All Downloads are FREE. Search and download functionalities are using the official Maven repository.

cgi.session.java_servlet_store.rb Maven / Gradle / Ivy

Go to download

A servlet bridge for (Ruby-based) Rack applications that allow them to run in Java Application servers using JRuby.

There is a newer version: 1.2.2
Show newest version
#--
# Copyright (c) 2010-2012 Engine Yard, Inc.
# Copyright (c) 2007-2009 Sun Microsystems, Inc.
# This source code is available under the MIT license.
# See the file LICENSE.txt for details.
#++

require 'openssl'

class CGI #:nodoc:all
  class Session
    class JavaServletStore
      RAILS_SESSION_KEY = "__current_rails_session"

      def initialize(session, option=nil)
        @java_request = option["java_servlet_request"] if option
        unless @java_request
          raise 'JavaServletStore requires that HttpServletRequest is made available to the session'
        end
        @digest = option['digest'] || 'SHA1'
        @session_data = {}
      end

      # Restore session state from the Java session
      def restore
        @session_data = {}
        java_session = @java_request.getSession(false)
        if java_session
          java_session.getAttributeNames.each do |k|
            if k == RAILS_SESSION_KEY
              marshalled_bytes = java_session.getAttribute(RAILS_SESSION_KEY)
              if marshalled_bytes
                data = Marshal.load(String.from_java_bytes(marshalled_bytes))
                @session_data.update data if Hash === data
              end
            else
              @session_data[k] = java_session.getAttribute(k)
            end
          end
        end
        @session_data
      end

      # Save session state to the Java session
      def update
        java_session = @java_request.getSession(true)
        hash = @session_data.dup
        hash.delete_if do |k,v|
          if String === k
            case v
            when String, Numeric, true, false, nil
              java_session.setAttribute k, v
              true
            else
              if v.respond_to?(:java_object)
                java_session.setAttribute k, v
                true
              else
                false
              end
            end
          end
        end
        unless hash.empty?
          marshalled_string = Marshal.dump(hash)
          marshalled_bytes = marshalled_string.to_java_bytes
          java_session.setAttribute(RAILS_SESSION_KEY, marshalled_bytes)
        end
      end

      # Update and close the Java session entry
      def close
        update
      end

      # Delete the Java session entry
      def delete
        java_session = @java_request.getSession(false)
        java_session.invalidate if java_session
      end

      def generate_digest(data)
        java_session = @java_request.getSession(true)
        @secret ||= java_session.getAttribute("__rails_secret")
        unless @secret
          @secret = java_session.getId + OpenSSL::Random.random_bytes(32) + java_session.getLastAccessedTime.to_s
          java_session.setAttribute("__rails_secret", @secret)
        end
        OpenSSL::HMAC.hexdigest(OpenSSL::Digest::Digest.new(@digest), @secret, data)
      end

      # The session state
      def data
        @session_data
      end

      def []=(k, v)
        @session_data[k] = v
      end

      def [](k)
        @session_data[k]
      end

      def each(&b)
        @session_data.each(&b)
      end

      private
      # Attempts to redirect any messages to the data object.
      def method_missing(name, *args, &block)
        @session_data.send(name, *args, &block)
      end
    end
  end
end




© 2015 - 2025 Weber Informatics LLC | Privacy Policy