# Copyright (c) 2012 RightScale Inc # # Permission is hereby granted, free of charge, to any person obtaining # a copy of this software and associated documentation files (the # "Software"), to deal in the Software without restriction, including # without limitation the rights to use, copy, modify, merge, publish, # distribute, sublicense, and/or sell copies of the Software, and to # permit persons to whom the Software is furnished to do so, subject to # the following conditions: # # The above copyright notice and this permission notice shall be # included in all copies or substantial portions of the Software. # # THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, # EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF # MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND # NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE # LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION # OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION # WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. require File.expand_path(File.join(File.dirname(__FILE__), "..", "global_session")) # Make sure the namespace exists, to satisfy Rails auto-loading module GlobalSession module Rack # Global session middleware. Note: this class relies on # Rack::Cookies being used higher up in the chain. class Middleware NUMERIC_HOST = /^[0-9.]+$/.freeze LOCAL_SESSION_KEY = "rack.session".freeze # @return [GlobalSession::Configuration] attr_accessor :configuration # @return [GlobalSession::Directory] attr_accessor :directory # @return [GlobalSession::Keystore] attr_accessor :keystore # Make a new global session middleware. # # The optional block here controls an alternate ticket retrieval # method. If no ticket is stored in the cookie jar, this # function is called. If it returns a non-nil value, that value # is the ticket. # # @param [Configuration] configuration # @param optional [String,Directory] directory the disk-directory in which keys live (DEPRECATED), or an actual instance of Directory # # @yield if a block is provided, yields to the block to fetch session data from request state # @yieldparam [Hash] env Rack request environment is passed as a yield parameter def initialize(app, configuration, directory=nil, &block) @app = app # Initialize shared configuration # @deprecated require Configuration object in v4 if configuration.instance_of?(String) @configuration = Configuration.new(configuration, ENV['RACK_ENV'] || 'development') else @configuration = configuration end klass = nil begin # v0.9.0 - v3.0.4: class name is the value of the 'directory' key klass_name = @configuration['directory'] case klass_name when Hash # v3.0.5 and beyond: class name is in 'class' subkey klass_name = klass_name['class'] when NilClass # the eternal default, if the class name is not provided klass_name = 'GlobalSession::Directory' end if klass_name.is_a?(String) # for apps klass = klass_name.to_const else # for specs that need to directly inject a class/object klass = klass_name end rescue Exception => e raise GlobalSession::ConfigurationError, "Invalid/unknown directory class name: #{klass_name.inspect}" end # Initialize the directory object if directory.is_a?(Directory) # In v4-style initialization, the directory is always passed in @directory = directory elsif klass.is_a?(Class) # @deprecated v3-style initialization where the config file names the directory class @directory = klass.new(@configuration, directory) else raise GlobalSession::ConfigurationError, "Cannot determine directory class/instance; method parameter is a #{directory.class.name} and configuration parameter is #{klass.class.name}" end @cookie_retrieval = block @cookie_name = @configuration['cookie']['name'] end # Rack request chain. Sets up the global session ticket from # the environment and passes it up the chain. # # @return [Array] valid Rack response tuple e.g. [200, 'hello world'] # @param [Hash] env Rack request environment def call(env) env['rack.cookies'] = {} unless env['rack.cookies'] begin err = nil read_authorization_header(env) || read_cookie(env) || create_session(env) rescue Exception => read_err err = read_err # Catch "double whammy" errors begin env['global_session'] = @directory.create_session rescue Exception => create_err err = create_err end handle_error('reading session cookie', env, err) end tuple = nil begin tuple = @app.call(env) rescue Exception => read_err handle_error('processing request', env, read_err) return tuple else renew_cookie(env) update_cookie(env) return tuple end end # Read a global session from the HTTP Authorization header, if present. If an authorization # header was found, also disable global session cookie update and renewal by setting the # corresponding keys of the Rack environment. # # @return [Boolean] true if the environment was populated, false otherwise # @param [Hash] env Rack request environment def read_authorization_header(env) if env.has_key? 'X-HTTP_AUTHORIZATION' # RFC2617 style (preferred by OAuth 2.0 spec) header_data = env['X-HTTP_AUTHORIZATION'].to_s.split elsif env.has_key? 'HTTP_AUTHORIZATION' # Fallback style (generally when no load balancer is present, e.g. dev/test) header_data = env['HTTP_AUTHORIZATION'].to_s.split else header_data = nil end if header_data && header_data.size == 2 && header_data.first.downcase == 'bearer' env['global_session.req.renew'] = false env['global_session.req.update'] = false env['global_session'] = @directory.load_session(header_data.last) true else false end end # Read a global session from HTTP cookies, if present. # # @return [Boolean] true if the environment was populated, false otherwise # @param [Hash] env Rack request environment def read_cookie(env) if @cookie_retrieval && (cookie = @cookie_retrieval.call(env)) env['global_session'] = @directory.load_session(cookie) true elsif env['rack.cookies'].has_key?(@cookie_name) env['global_session'] = @directory.load_session(env['rack.cookies'][@cookie_name]) true else false end end # Ensure that the Rack environment contains a global session object; create a session # if necessary. # # @return [true] always returns true # @param [Hash] env Rack request environment def create_session(env) env['global_session'] ||= @directory.create_session true end # Renew the session ticket. # # @return [true] always returns true # @param [Hash] env Rack request environment def renew_cookie(env) return true unless @directory.local_authority_name return true if env['global_session.req.renew'] == false if (renew = @configuration['renew']) && env['global_session'] && env['global_session'].expired_at < Time.at(Time.now.utc + 60 * renew.to_i) env['global_session'].renew! end true end # Update the cookie jar with the revised ticket. # # @return [true] always returns true # @param [Hash] env Rack request environment def update_cookie(env) return true unless @directory.local_authority_name return true if env['global_session.req.update'] == false session = env['global_session'] if session unless session.valid? old_session = session session = @directory.create_session perform_invalidation_callbacks(env, old_session, session) env['global_session'] = session end value = session.to_s expires = @configuration['ephemeral'] ? nil : session.expired_at unless env['rack.cookies'][@cookie_name] == value secure = (env['HTTP_X_FORWARDED_PROTO'] == 'https') || (env['rack.url_scheme'] == 'https') env['rack.cookies'][@cookie_name] = { :value => value, :domain => cookie_domain(env), :expires => expires, :httponly => true, :secure => secure, } end else # write an empty cookie wipe_cookie(env) end true rescue Exception => e wipe_cookie(env) raise e end # Delete the global session cookie from the cookie jar. # # @return [true] always returns true # @param [Hash] env Rack request environment def wipe_cookie(env) return true unless @directory.local_authority_name return true if env['global_session.req.update'] == false env['rack.cookies'][@cookie_name] = {:value => nil, :domain => cookie_domain(env), :expires => Time.at(0)} true end # Handle exceptions that occur during app invocation. This will either save the error # in the Rack environment or raise it, depending on the type of error. The error may # also be logged. # # @return [true] always returns true # @param [String] activity name of activity during which the error happened # @param [Hash] env Rack request environment # @param [Exception] e error that happened def handle_error(activity, env, e) if env['rack.logger'] msg = "#{e.class} while #{activity}: #{e}" msg += " #{e.backtrace}" unless e.is_a?(ExpiredSession) env['rack.logger'].error(msg) end if e.is_a?(ClientError) || e.is_a?(SecurityError) env['global_session.error'] = e wipe_cookie(env) elsif e.is_a? ConfigurationError env['global_session.error'] = e else # Don't intercept errors unless they're GlobalSession-related raise e end true end # Perform callbacks to directory and/or local session # informing them that this session has been invalidated. # # @return [true] always returns true # @param [Hash] env Rack request environment # @param [GlobalSession::Session] old_session now-invalidated session # @param [GlobalSession::Session] new_session new session that will be sent to the client def perform_invalidation_callbacks(env, old_session, new_session) if (local_session = env[LOCAL_SESSION_KEY]) && local_session.respond_to?(:rename!) local_session.rename!(old_session, new_session) end true end # Determine the domain name for which we should set the cookie. Uses the domain specified # in the configuration if one is found; otherwise, uses the SERVER_NAME from the request # but strips off the first component if the domain name contains more than two components. # # @param [Hash] env Rack request environment def cookie_domain(env) name = env['HTTP_X_FORWARDED_HOST'] || env['SERVER_NAME'] if @configuration['cookie'].has_key?('domain') # Use the explicitly provided domain name domain = @configuration['cookie']['domain'] elsif name =~ NUMERIC_HOST # Don't set a domain if the browser requested an IP-based host domain = nil else # Guess an appropriate domain for the cookie. Strip one level of # subdomain; leave SLDs unmolested; omit domain entirely for # one-component domains (e.g. localhost). parts = name.split('.') case parts.length when 0..1 domain = nil when 2 domain = parts.join('.') else domain = parts[1..-1].join('.') end end domain end end end end module Rack GlobalSession = ::GlobalSession::Rack::Middleware unless defined?(::Rack::GlobalSession) end