require 'data_mapper' require 'base64' module Merb module SessionMixin def setup_session MERB_LOGGER.info("Setting up session") before = cookies[_session_id_key] request.session, cookies[_session_id_key] = Merb::DataMapperSession.persist(cookies[_session_id_key]) @_fingerprint = Marshal.dump(request.session.data).hash @_new_cookie = cookies[_session_id_key] != before end def finalize_session MERB_LOGGER.info("Finalize session") request.session.save if @_fingerprint != Marshal.dump(request.session.data).hash set_cookie(_session_id_key, request.session.session_id, _session_expiry) if (@_new_cookie || request.session.needs_new_cookie) end end table_name = (Merb::Plugins.config[:merb_data_mapper][:session_table_name] || "sessions") class DataMapperSession < DataMapper::Base set_table_name "sessions" property :session_id, :text property :data, :text attr_accessor :needs_new_cookie class << self # Generates a new session ID and creates a row for the new session in the database. def generate create(:session_id => Merb::SessionMixin::rand_uuid, :data =>{}) end # Gets the existing session based on the session_id available in cookies. # If none is found, generates a new session. def persist(session_id) if session_id session = self[:session_id => session_id] end unless session session = generate end [session, session.session_id] end def marshal(data) Base64.encode64(Marshal.dump(data)) if data end def unmarshal(data) Marshal.load(Base64.decode64(data)) if data end end # Regenerate the Session ID def regenerate self.session_id = Merb::SessionMixin::rand_uuid self.needs_new_cookie = true self.save end # Recreates the cookie with the default expiration time # Useful during log in for pushing back the expiration date def refresh_expiration self.needs_new_cookie = true end # Lazy-delete of session data def delete data = {} end def [](key) data[key] end def []=(key, val) data[key] = val end def data @unmarshalled_data || @unmarshalled_data = self.class.unmarshal(@data) end private before_save :serialize_data def serialize_data @data = self.class.marshal(self.data) end end unless DataMapper.database.table_exists?(table_name) puts "Warning: The database did not contain a '#{table_name}' table for sessions." DataMapperSession.auto_migrate! puts "Created sessions table." end end