hubssolib 3.3.0 → 3.4.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: b434d369e25dc81a74c87385df70cade7dfd40c47abfae2c05732b471b6e1f5e
4
- data.tar.gz: 41d2303f125f52a5f957f851f1a8d5ae7067ab8233fcfa5359640f88e3bceb06
3
+ metadata.gz: c1a83a70b87a6c3cabd8bbb821dfc06ee467d382c9593cb7c1733152e136f5a1
4
+ data.tar.gz: 87f1dda92220f25326c8bbce597ecd97e74c3dd7e22fd0fbf89bfed6269b6b23
5
5
  SHA512:
6
- metadata.gz: 7dc3e04c0b1e75eecf504c61d19834151179666b7e601d974cbf74509f1ab1197bbf74cf347c63b4e1aef61d744bceb96621b7fccb8017654f6d4266eea21246
7
- data.tar.gz: ba3e8e1985dd91185c6049e25212dcade4dfe064faf31c4e1c1d013ac8851984f3b48ee47df5de16d097863ea36c9165c93790e05703f8fa640908e7260c3b4b
6
+ metadata.gz: cf2ae3afb91a4fcc3391cf4b156e3dd9e4d0be4cecafc10204cf6dc8a7d6b9d07d51dc08cd7fd22ee12a478025d03730227fa2602a194d4b0e6785d72af21295
7
+ data.tar.gz: 6ac8d8a0a1920d7f93604204ef0c254a094bd4a8559110bba2f630c1fe406f87966b0ba2289d017a9a48e90cd0d6e2eab0e1ae3872032c7bad389942965fe2b2
data/CHANGELOG.md CHANGED
@@ -1,3 +1,22 @@
1
+ ## 3.4.0, 24-Mar-2025
2
+
3
+ The session interface has now been cleaned up. While the public API is unchanged, you:
4
+
5
+ * ...are encouraged to change uses of `hubssolib_current_user = foo` over to `hubssolib_log_in()`
6
+ * ...are likewise encouraged to change `hubssolib_current_user = nil` over to `hubssolib_log_out()`
7
+
8
+ There are many fixes in the overhauled session management:
9
+
10
+ * Session data is only stored when a user logs in, else no session cookie is built
11
+ * If a user logs in, possible stale older sessions they might have are swept out
12
+ * When the user logs out their session data is entirely deleted
13
+ * The Hub app includes a Rake task that makes use of a new internal session factory feature to clean up sessions with a very old idle timeout (more than 3 times the usual in-app 'you were logged out' notification mechanism's delay via `HUB_IDLE_TIME_LIMIT`, or 2 days, whichever is larger) and delete such sessions - yes, this means that if a user did then come in even later on a now-deleted session ID, they'd simply be logged out without the warning message about why, but it was a serious oversight prior to allow the sessions to just accumulate.
14
+
15
+ In addition:
16
+
17
+ * Iteration over the sessions object for active user enumeration could cause failures. The process could be time consuming and, during it, the client side is iterating over a remote object that the DRb server maintains for any new, inbound sessions which may be started by other web server request threads into the Hub app. This would lead to `can't add a new key into hash during iteration` exceptions. This is fixed via more aggressive mutex use.
18
+
19
+
1
20
  ## 3.3.0, 16-Feb-2025
2
21
 
3
22
  Sentry support, for use by the DRb server. If you use Sentry, define your account's `SENTRY_DSN` in the environment where the DRb server runs and exceptions will be reported.
data/Gemfile.lock CHANGED
@@ -1,7 +1,7 @@
1
1
  PATH
2
2
  remote: .
3
3
  specs:
4
- hubssolib (3.1.0)
4
+ hubssolib (3.4.0)
5
5
  base64 (~> 0.2)
6
6
  drb (~> 2.2)
7
7
 
@@ -29,7 +29,7 @@ GEM
29
29
  psych (5.2.3)
30
30
  date
31
31
  stringio
32
- rdoc (6.12.0)
32
+ rdoc (6.13.0)
33
33
  psych (>= 4.0.0)
34
34
  reline (0.6.0)
35
35
  io-console (~> 0.5)
@@ -52,7 +52,7 @@ GEM
52
52
  simplecov_json_formatter (~> 0.1)
53
53
  simplecov-html (0.13.1)
54
54
  simplecov_json_formatter (0.1.4)
55
- stringio (3.1.3)
55
+ stringio (3.1.5)
56
56
 
57
57
  PLATFORMS
58
58
  ruby
data/README.md CHANGED
@@ -202,6 +202,14 @@ The four arguments are guaranteed to be present and non-empty, with leading or t
202
202
 
203
203
 
204
204
 
205
+ ## Session maintenance
206
+
207
+ User-idle session expiry is routinely handled in the "beforehand" callback so that we can see the session is valid but expired, expire it and add a via-flash warning about what happened so the user is fully informed. The Hub gem tries to be nice about `POST`, `PATCH` and `PUT` operations too and allows those to complete before the expected subsequent redirection `GET` causing expiry, to try and avoid users losing form submission data entirely.
208
+
209
+ Users who log in but then go idle will not cause any natural session expiry by virtue of there being no new page fetch activity provoking the idle timeout check. For users who've just gone away full stop, this means a session record is left hanging around in DRb server memory (or if future iterations support other storage methods, they'd persist in whatever storage that is). For this reason, the Hub _application_ provides a Rake task which sweeps sessions based on things idle for three times the `HUB_IDLE_TIME_LIMIT`, or two days, whichever is longer (so short idle timers still give users a couple of days to potentially come back to their old session and get the "nice" expiry message). Run `bundle exec rake hub_sessions:sweep` whenever you want (in practice, once a day is enough). **Be sure to set up any custom Hub environment variables for your setup** so that the Rake task knows where to find the DRb socket for the running server, what your expiry time is if so customised, and so-on.
210
+
211
+
212
+
205
213
  ## Hub library API
206
214
 
207
215
  The Hub component interfaces that should be used by application authors when integrating with the Hub single sign-on mechanism are described below. If you want a complete list of all public interfaces, consult the file `hub_sso_lib.rb` inside the Hub gem. All functions and classes therein are fully commented to describe the purpose of each class, along with the purpose, input parameters and return values of class methods and instance methods.
data/hubssolib.gemspec CHANGED
@@ -4,7 +4,7 @@ spec = Gem::Specification.new do |s|
4
4
  s.platform = Gem::Platform::RUBY
5
5
  s.name = 'hubssolib'
6
6
 
7
- s.version = '3.3.0'
7
+ s.version = '3.4.0'
8
8
  s.author = 'Andrew Hodgkinson and others'
9
9
  s.email = 'ahodgkin@rowing.org.uk'
10
10
  s.homepage = 'http://pond.org.uk/'
data/lib/hub_sso_lib.rb CHANGED
@@ -51,9 +51,9 @@ module HubSsoLib
51
51
  #
52
52
  HUB_PATH_PREFIX = ENV['HUB_PATH_PREFIX'] || ''
53
53
 
54
- # Time limit, *in seconds*, for the account inactivity timeout.
55
- # If a user performs no Hub actions during this time they will
56
- # be automatically logged out upon their next action.
54
+ # Time limit, *in seconds*, for the account inactivity timeout. If a user
55
+ # performs no Hub actions during this time they will be automatically logged
56
+ # out upon their next action, with a Flash message set to explain why.
57
57
  #
58
58
  HUB_IDLE_TIME_LIMIT = ENV['HUB_IDLE_TIME_LIMIT']&.to_i || 4 * 60 * 60
59
59
 
@@ -402,16 +402,16 @@ module HubSsoLib
402
402
  attr_accessor :session_return_to
403
403
  attr_accessor :session_flash
404
404
  attr_accessor :session_user
405
- attr_accessor :session_key_rotation
405
+ attr_accessor :session_rotated_key
406
406
  attr_accessor :session_ip
407
407
 
408
408
  def initialize
409
- @session_last_used = Time.now.utc
410
- @session_return_to = nil
411
- @session_flash = {}
412
- @session_user = HubSsoLib::User.new
413
- @session_key_rotation = nil
414
- @session_ip = nil
409
+ @session_last_used = Time.now.utc
410
+ @session_return_to = nil
411
+ @session_flash = {}
412
+ @session_user = HubSsoLib::User.new
413
+ @session_rotated_key = nil
414
+ @session_ip = nil
415
415
 
416
416
  rescue => e
417
417
  Sentry.capture_exception(e) if defined?(Sentry) && Sentry.respond_to?(:capture_exception)
@@ -423,8 +423,10 @@ module HubSsoLib
423
423
  # Class: SessionFactory #
424
424
  # (C) Hipposoft 2006 #
425
425
  # #
426
- # Purpose: Build Session objects for DRb server clients. Maintains a #
427
- # hash of Session objects. #
426
+ # Purpose: Manage Session objects for DRb server clients. This class #
427
+ # implements the API exposed by the HubSsoLib::Server DRb #
428
+ # endpoint, so this is the remote object that clients will #
429
+ # be calling into. #
428
430
  # #
429
431
  # Author: A.D.Hodgkinson #
430
432
  # #
@@ -448,7 +450,7 @@ module HubSsoLib
448
450
  # recorded in existing session data.
449
451
  #
450
452
  # Whether new or pre-existing, the returned session will have changed key
451
- # as a result of being read; check the #session_key_rotation property to
453
+ # as a result of being read; check the #session_rotated_key property to
452
454
  # find out the new key. If you fail to do this, you'll lose access to the
453
455
  # session data as you won't know which key it lies under.
454
456
  #
@@ -463,17 +465,17 @@ module HubSsoLib
463
465
  # invalid and discarded.
464
466
  #
465
467
  def get_hub_session_proxy(key, remote_ip)
466
- hub_session = @hub_sessions[key]
468
+ hub_session = HUB_MUTEX.synchronize { @hub_sessions[key] }
467
469
  message = hub_session.nil? ? 'Created' : 'Retrieving'
468
470
  new_key = SecureRandom.uuid
469
471
 
470
472
  unless @hub_be_quiet
471
- puts "#{ message } session for key #{ key } and rotating to #{ new_key }"
473
+ puts "Session factory: #{ message } session for key #{ key } and rotating to #{ new_key }"
472
474
  end
473
475
 
474
476
  unless hub_session.nil? || hub_session.session_ip == remote_ip
475
477
  unless @hub_be_quiet
476
- puts "WARNING: IP address changed from #{ hub_session.session_ip } to #{ remote_ip } -> discarding session"
478
+ puts "Session factory: WARNING: IP address changed from #{ hub_session.session_ip } to #{ remote_ip } -> discarding session"
477
479
  end
478
480
 
479
481
  hub_session = nil
@@ -484,10 +486,12 @@ module HubSsoLib
484
486
  hub_session.session_ip = remote_ip
485
487
  end
486
488
 
487
- @hub_sessions.delete(key)
488
- @hub_sessions[new_key] = hub_session
489
+ HUB_MUTEX.synchronize do
490
+ @hub_sessions.delete(key)
491
+ @hub_sessions[new_key] = hub_session
492
+ end
489
493
 
490
- hub_session.session_key_rotation = new_key
494
+ hub_session.session_rotated_key = new_key
491
495
  return hub_session
492
496
 
493
497
  rescue => e
@@ -495,6 +499,26 @@ module HubSsoLib
495
499
  raise
496
500
  end
497
501
 
502
+ # Enumerate all currently known sessions. The format is a Hash, with the
503
+ # session key UUIDs as keys and the related HubSsoLib::Session instances as
504
+ # values.
505
+ #
506
+ # WARNING: This returns all Hub sessions maintained by the server instance
507
+ # but it's returning a reference to the live object with everything being
508
+ # managed over DRb. If a caller iterates over this object, then it can fall
509
+ # foul of other request threads making changes to the underlying data and
510
+ # Ruby raising exceptions about e.g. hashes being modified during
511
+ # enumeration.
512
+ #
513
+ # To prevent this, if intending to iterate over the collection, really the
514
+ # only safe way is to duplicate it first on "your side" (the client side)
515
+ # of the DRb connection. This of course consumes RAM, so you might choose
516
+ # to evaluate e.g. the number of keys in the returned session data and only
517
+ # permit enumeration if they fall below some previously-measured "allowed"
518
+ # value wherein RAM consumption is acceptable.
519
+ #
520
+ # See HubSsoLib::Core#hubssolib_enumerate_users for an example.
521
+ #
498
522
  def enumerate_hub_sessions()
499
523
  @hub_sessions
500
524
 
@@ -502,6 +526,86 @@ module HubSsoLib
502
526
  Sentry.capture_exception(e) if defined?(Sentry) && Sentry.respond_to?(:capture_exception)
503
527
  raise
504
528
  end
529
+
530
+ # Given a session key (which, if a session has been looked up and the key
531
+ # thus rotated, ought to be that new, rotated key), destroy the associated
532
+ # session data. Does nothing if the key is not found.
533
+ #
534
+ def destroy_session_by_key(key)
535
+ HUB_MUTEX.synchronize do
536
+ @hub_sessions.delete(key)
537
+ end
538
+
539
+ rescue => e
540
+ Sentry.capture_exception(e) if defined?(Sentry) && Sentry.respond_to?(:capture_exception)
541
+ raise
542
+ end
543
+
544
+ # WARNING: Comparatively slow
545
+ #
546
+ # This is called in rare cases such as user deletion or being asked for a
547
+ # session under an old key, indicating loss of key rotation sequence.
548
+ # Removes all sessions found for a given user ID.
549
+ #
550
+ # IN THE CURRENT IMPLEMENTATION THIS JUST SEQUENTIALLY SCANS ALL ACTIVE
551
+ # SESSIONS IN THE HASH and must therefore lock on mutex for the duration.
552
+ #
553
+ def destroy_sessions_by_user_id(user_id)
554
+ HUB_MUTEX.synchronize do
555
+ @hub_sessions.reject! do | key, session |
556
+ session&.session_user&.user_id == user_id
557
+ end
558
+ end
559
+
560
+ rescue => e
561
+ Sentry.capture_exception(e) if defined?(Sentry) && Sentry.respond_to?(:capture_exception)
562
+ raise
563
+ end
564
+
565
+ # WARNING: Slow
566
+ #
567
+ # This is a housekeeping task which checks sessions against Hub expiry and,
568
+ # if the session keys look to be substantially older than the value set in
569
+ # HUB_IDLE_TIME_LIMIT, the session simply deleted. If a user does return
570
+ # later, they'll see themselves in logged out state without the Flash
571
+ # warning them of an expired session, but we can't allow session keys to
572
+ # just hang around forever so *some* kind of sweep is needed.
573
+ #
574
+ # This method clearly needs to iterate over all sessions under a mutex and
575
+ # makes relatively complex checks for each, so it's fairly slow compared
576
+ # to most methods. Call it infrequently; any and all other attempts to read
577
+ # session data while the method runs will block until method finishes.
578
+ #
579
+ def destroy_ancient_sessions
580
+ time_limit = HUB_IDLE_TIME_LIMIT * 3 # (TODO: This is fairly arbitrary...)
581
+ time_limit = 172_800 if time_limit < 172_800 # (2 days)
582
+ destroyed = 0
583
+
584
+ unless @hub_be_quiet
585
+ puts "Session factory: Sweeping sessions inactive for more than #{ time_limit } seconds..."
586
+ end
587
+
588
+ HUB_MUTEX.synchronize do
589
+ count_before = @hub_sessions.keys.size
590
+
591
+ @hub_sessions.reject! do | key, session |
592
+ last_used = session&.session_last_used
593
+ last_used.nil? || Time.now.utc - last_used > time_limit
594
+ end
595
+
596
+ count_after = @hub_sessions.keys.size
597
+ destroyed = count_before - count_after
598
+ end
599
+
600
+ unless @hub_be_quiet
601
+ puts "Session factory: ...Destroyed #{destroyed} session(s)"
602
+ end
603
+
604
+ rescue => e
605
+ Sentry.capture_exception(e) if defined?(Sentry) && Sentry.respond_to?(:capture_exception)
606
+ raise
607
+ end
608
+
505
609
  end
506
610
 
507
611
  #######################################################################
@@ -550,14 +654,65 @@ module HubSsoLib
550
654
 
551
655
  module Core
552
656
 
553
- # Returns true or false if the user is logged in.
657
+ # Log in the user. This is just syntax sugar for setting the current user
658
+ # via #hubssolib_current_user, really. You can freely use either approach
659
+ # according to your preferred aesthetics, but this method is preferred.
554
660
  #
555
- # Preloads @hubssolib_current_user with user data if logged in.
661
+ # +user+:: A valid HubSsoLib::User instance. If this has a +nil+ value for
662
+ # +user_id+, or if +user+ is itself +nil+, you'll cause the same
663
+ # effect as if explicitly logging *out*.
664
+ #
665
+ def hubssolib_log_in(user)
666
+ self.hubssolib_current_user = user # (which deals with all related session and cookie consequences)
667
+ end
668
+
669
+ # Log out the user. Very few applications should ever need to call this,
670
+ # though Hub certainly does and it gets used internally too.
671
+ #
672
+ def hubssolib_log_out
673
+ self.hubssolib_current_user = nil # (which deals with all related session and cookie consequences)
674
+ end
675
+
676
+ # Returns true or false if a user is logged in or not, respectively.
556
677
  #
557
678
  def hubssolib_logged_in?
558
679
  !!self.hubssolib_current_user
559
680
  end
560
681
 
682
+ # Accesses the current user, via the DRb server if necessary. Returns a
683
+ # HubSsoLib::User object, or +nil+ if none is available (this indicates
684
+ # that nobody is logged in, but #hubssolib_logged_in? should be used to
685
+ # check that, to allow for possible future more advanced logic within).
686
+ #
687
+ def hubssolib_current_user
688
+ hub_session = self.hubssolib_get_session()
689
+ user = hub_session&.session_user
690
+
691
+ return (user&.user_id.nil? ? nil : user)
692
+ end
693
+
694
+ # Sets the currently signed in user. Note that although this works and is
695
+ # maintained, it is recommended that #hubssolib_log_in gets called instead.
696
+ #
697
+ # +user+:: A valid HubSsoLib::User. This will replace any existing logged
698
+ # in user. If there is no session yet, one will be created.
699
+ #
700
+ def hubssolib_current_user=(user)
701
+ if user.nil?
702
+ self.hubssolib_destroy_session!
703
+ else
704
+
705
+ # Stale sessions may exist for this user. The frequency of logins is
706
+ # very low compared to frequency of page requests, so we perform this
707
+ # expensive method here as a least-worst option!
708
+ #
709
+ hubssolib_factory().destroy_sessions_by_user_id(user.user_id)
710
+
711
+ hub_session = self.hubssolib_create_session()
712
+ hub_session.session_user = user
713
+ end
714
+ end
715
+
561
716
  # Returns markup for a link that leads to Hub's conditional login endpoint,
562
717
  # inline-styled as a red "Log in" or green "Account" button. This can be
563
718
  # used in page templates to avoid needing any additional images or other
@@ -570,7 +725,6 @@ module HubSsoLib
570
725
  #
571
726
  def hubssolib_account_link
572
727
  logged_in = self.hubssolib_logged_in?()
573
-
574
728
  ui_href = "#{HUB_PATH_PREFIX}/account/login_conditional"
575
729
  noscript_img_src = "#{HUB_PATH_PREFIX}/account/login_indication.png"
576
730
  noscript_img_tag = helpers.image_tag(noscript_img_src, size: '90x22', border: '0', alt: 'Log in or out')
@@ -667,51 +821,12 @@ module HubSsoLib
667
821
  return (puser && !puser.empty? && puser != pnormal)
668
822
  end
669
823
 
670
- # Log out the user. Very few applications should ever need to call this,
671
- # though Hub certainly does and it gets used internally too.
672
- #
673
- def hubssolib_log_out
674
- # Causes the "hubssolib_current_[foo]=" methods to run, which
675
- # deal with everything else.
676
- self.hubssolib_current_user = nil
677
- @hubssolib_current_session_proxy = nil
678
- end
679
-
680
- # Accesses the current session from the cookie. Creates a new session
681
- # object if need be, but can return +nil+ if e.g. attempting to access
682
- # session cookie data without SSL.
683
- #
684
- def hubssolib_current_session
685
- @hubssolib_current_session_proxy ||= hubssolib_get_session_proxy()
686
- end
687
-
688
- # Accesses the current user, via the DRb server if necessary.
689
- #
690
- def hubssolib_current_user
691
- hub_session = self.hubssolib_current_session
692
- user = hub_session.nil? ? nil : hub_session.session_user
693
-
694
- if (user && user.user_id)
695
- return user
696
- else
697
- return nil
698
- end
699
- end
700
-
701
- # Store the given user data in the cookie
702
- #
703
- def hubssolib_current_user=(user)
704
- hub_session = self.hubssolib_current_session
705
- hub_session.session_user = user unless hub_session.nil?
706
- end
707
-
708
824
  # Public read-only accessor methods for common user activities:
709
825
  # return the current user's roles as a Roles object, or nil if
710
826
  # there's no user.
711
827
  #
712
828
  def hubssolib_get_user_roles
713
- user = self.hubssolib_current_user
714
- user ? user.user_roles.to_authenticated_roles : nil
829
+ self.hubssolib_current_user&.user_roles&.to_authenticated_roles
715
830
  end
716
831
 
717
832
  # Public read-only accessor methods for common user activities:
@@ -719,8 +834,7 @@ module HubSsoLib
719
834
  # no user. See also hubssolib_unique_name.
720
835
  #
721
836
  def hubssolib_get_user_name
722
- user = self.hubssolib_current_user
723
- user ? user.user_real_name : nil
837
+ self.hubssolib_current_user&.user_real_name
724
838
  end
725
839
 
726
840
  # Public read-only accessor methods for common user activities:
@@ -728,8 +842,7 @@ module HubSsoLib
728
842
  # nil if there's no user. See also hubssolib_unique_name.
729
843
  #
730
844
  def hubssolib_get_user_id
731
- user = self.hubssolib_current_user
732
- user ? user.user_id : nil
845
+ self.hubssolib_current_user&.user_id
733
846
  end
734
847
 
735
848
  # Public read-only accessor methods for common user activities:
@@ -737,8 +850,7 @@ module HubSsoLib
737
850
  # no user.
738
851
  #
739
852
  def hubssolib_get_user_address
740
- user = self.hubssolib_current_user
741
- user ? user.user_email : nil
853
+ self.hubssolib_current_user&.user_email
742
854
  end
743
855
 
744
856
  # Return a human-readable unique ID for a user. We don't want to
@@ -863,15 +975,13 @@ module HubSsoLib
863
975
  cookies.delete(HUB_LOGIN_INDICATOR_COOKIE, domain: :all, path: '/')
864
976
 
865
977
  if login_is_required
866
- hubssolib_store_location
867
- return hubssolib_must_login
978
+ hubssolib_store_location()
979
+ return hubssolib_must_login()
868
980
  else
869
981
  return true
870
982
  end
871
983
  end
872
984
 
873
- # Definitely logged in.
874
- #
875
985
  cookies[HUB_LOGIN_INDICATOR_COOKIE] = {
876
986
  value: HUB_LOGIN_INDICATOR_COOKIE_VALUE,
877
987
  path: '/',
@@ -892,8 +1002,8 @@ module HubSsoLib
892
1002
  # if OK, else indicate that access is denied.
893
1003
 
894
1004
  if (hubssolib_session_expired?)
895
- hubssolib_store_location
896
- hubssolib_log_out
1005
+ hubssolib_store_location()
1006
+ hubssolib_log_out()
897
1007
  hubssolib_set_flash(:attention, 'Sorry, your session timed out; you need to log in again to continue.')
898
1008
 
899
1009
  # We mean this: redirect_to :controller => 'account', :action => 'login'
@@ -902,7 +1012,7 @@ module HubSsoLib
902
1012
  redirect_to HUB_PATH_PREFIX + '/account/login'
903
1013
  else
904
1014
  hubssolib_set_last_used(Time.now.utc)
905
- return hubssolib_authorized? ? true : hubssolib_access_denied
1015
+ return hubssolib_authorized? ? true : hubssolib_access_denied()
906
1016
  end
907
1017
 
908
1018
  else
@@ -955,7 +1065,7 @@ module HubSsoLib
955
1065
  # Redirect to the URI stored by the most recent store_location call or
956
1066
  # to the passed default.
957
1067
  def hubssolib_redirect_back_or_default(default)
958
- url = hubssolib_get_return_to
1068
+ url = hubssolib_get_return_to()
959
1069
  hubssolib_set_return_to(nil)
960
1070
 
961
1071
  redirect_to(url || default)
@@ -987,26 +1097,29 @@ module HubSsoLib
987
1097
  end
988
1098
  end
989
1099
 
990
- # Public methods to set some data that would normally go in @session,
991
- # but can't because it needs to be accessed across applications. It is
992
- # put in an insecure support cookie instead. There are some related
993
- # private methods for things like session expiry too.
1100
+ # Flash data can be carried across the Hub session, stored in the DRb
1101
+ # server as a result, and is thus cleared automatically if a session gets
1102
+ # dropped. However, we also want this to work without being logged in, so
1103
+ # in that case it uses the normal flash as a backup when *writing*.
994
1104
  #
995
- def hubssolib_get_flash()
996
- f = self.hubssolib_current_session ? self.hubssolib_current_session.session_flash : nil
997
- return f || {}
1105
+ def hubssolib_get_flash
1106
+ session = self.hubssolib_get_session()
1107
+ session&.session_flash || {}
998
1108
  end
999
1109
 
1000
1110
  def hubssolib_set_flash(symbol, message)
1001
- return unless self.hubssolib_current_session
1002
- f = hubssolib_get_flash
1003
- f[symbol.to_s] = message
1004
- self.hubssolib_current_session.session_flash = f
1111
+ session = self.hubssolib_get_session()
1112
+ f = hubssolib_get_flash() unless session.nil?
1113
+ f = self.flash if f.nil? && self.respond_to?(:flash)
1114
+
1115
+ f[symbol] = message
1116
+
1117
+ session.session_flash = f unless session.nil?
1005
1118
  end
1006
1119
 
1007
1120
  def hubssolib_clear_flash
1008
- return unless self.hubssolib_current_session
1009
- self.hubssolib_current_session.session_flash = {}
1121
+ session = self.hubssolib_get_session()
1122
+ session.session_flash = {} unless session.nil?
1010
1123
  end
1011
1124
 
1012
1125
  # Return flash data for known keys, then all remaining keys, from both
@@ -1063,21 +1176,22 @@ module HubSsoLib
1063
1176
  hubssolib_get_exception_data(CGI::unescape(id_data))
1064
1177
  end
1065
1178
 
1066
- # Inclusion hook to make various methods available as ActionView
1067
- # helper methods.
1179
+ # Inclusion hook to make various methods available as ActionView helpers.
1068
1180
  #
1069
1181
  def self.included(base)
1070
- base.send :helper_method,
1071
- :hubssolib_current_user,
1072
- :hubssolib_unique_name,
1073
- :hubssolib_logged_in?,
1074
- :hubssolib_account_link,
1075
- :hubssolib_authorized?,
1076
- :hubssolib_privileged?,
1077
- :hubssolib_flash_data
1078
- rescue
1079
- # We're not always included in controllers...
1080
- nil
1182
+ if base.respond_to?(:helper_method)
1183
+ base.send(
1184
+ :helper_method,
1185
+
1186
+ :hubssolib_current_user,
1187
+ :hubssolib_unique_name,
1188
+ :hubssolib_logged_in?,
1189
+ :hubssolib_authorized?,
1190
+ :hubssolib_privileged?,
1191
+ :hubssolib_account_link,
1192
+ :hubssolib_flash_data
1193
+ )
1194
+ end
1081
1195
  end
1082
1196
 
1083
1197
  private
@@ -1104,6 +1218,108 @@ module HubSsoLib
1104
1218
  HUB_BYPASS_SSL || ! Rails.env.production?
1105
1219
  end
1106
1220
 
1221
+ # Create a new session. This MUST ONLY BE CALLED at a "log in" phase, since
1222
+ # it discards any existing session and creates a new, valid one, whether or
1223
+ # not a valid session key was being presented for the old session. It is,
1224
+ # in essence, a "clean slate" start.
1225
+ #
1226
+ # On exit, @hubssolib_session is updated; see #hubssolib_get_session for
1227
+ # the significance of that.
1228
+ #
1229
+ def hubssolib_create_session
1230
+ old_session = self.hubssolib_get_session()
1231
+ self.hubssolib_destroy_session! unless old_session.nil?
1232
+
1233
+ start_key = SecureRandom.uuid
1234
+ @hubssolib_session = hubssolib_factory().get_hub_session_proxy(start_key, request.remote_ip)
1235
+
1236
+ # The session is now stored under the rotated key, so put that into the
1237
+ # Hub session cookie so that on the next request, we can retrieve the
1238
+ # session data (and at that time, once again rotate the key).
1239
+ #
1240
+ next_key = @hubssolib_session.session_rotated_key
1241
+ self.hubssolib_store_key(next_key)
1242
+
1243
+ return @hubssolib_session
1244
+ end
1245
+
1246
+ # Gets session data based on the current +request+ details.
1247
+ #
1248
+ # * If there is no session cookie, returns +nil+
1249
+ # * If there is a session cookie but the session key is invalid, returns
1250
+ # +nil+.
1251
+ # * If there is a session cookie with valid key, session data is retrieved
1252
+ # and returned; the session key is rotated and the session cookie updated
1253
+ # with new key (that is to say that this request's response, under normal
1254
+ # conditions, will include an appropriate Set-Cookie header).
1255
+ #
1256
+ # The ivar @hubssolib_session is consulted first to avoid repeating work
1257
+ # during request processing, where it's likely that more than one call
1258
+ # will occur. This ivar is an known internal implementation detail; it is
1259
+ # also set by #hubssolib_create_session, and is cleared by calls to
1260
+ # #hubssolib_destroy_session!.
1261
+ #
1262
+ def hubssolib_get_session
1263
+ if @hubssolib_session.nil?
1264
+ key = cookies[HUB_COOKIE_NAME]
1265
+
1266
+ if key.nil? || key == ''
1267
+ @hubssolib_session = nil
1268
+ else
1269
+ hub_session = hubssolib_factory().get_hub_session_proxy(key, request.remote_ip)
1270
+
1271
+ if hub_session.nil? # Invalid key in cookie
1272
+ @hubssolib_session = nil
1273
+ else
1274
+ @hubssolib_session = hub_session
1275
+
1276
+ next_key = @hubssolib_session.session_rotated_key
1277
+ self.hubssolib_store_key(next_key)
1278
+ end
1279
+ end
1280
+ end
1281
+
1282
+ return @hubssolib_session
1283
+ end
1284
+
1285
+ # Destroys the current session. If there isn't one, then it has few side
1286
+ # effects other than making sure all session-related cookies are deleted.
1287
+ #
1288
+ def hubssolib_destroy_session!
1289
+ session = self.hubssolib_get_session()
1290
+
1291
+ # Remember, if creating or retrieving a session for a request, the key is
1292
+ # rotated and stored in the session under #session_rotated_key. To delete
1293
+ # that session within that same request - since rotation happens even if
1294
+ # the first place the session gets read is right here, in this method -
1295
+ # we must use the rotated key.
1296
+ #
1297
+ unless session.nil?
1298
+ hubssolib_factory().destroy_session_by_key(session.session_rotated_key)
1299
+ end
1300
+
1301
+ @hubssolib_session = nil
1302
+
1303
+ cookies.delete(HUB_COOKIE_NAME, domain: :all, path: '/')
1304
+ cookies.delete(HUB_LOGIN_INDICATOR_COOKIE, domain: :all, path: '/')
1305
+ end
1306
+
1307
+ # Store the Hub's session key in the Hub cookie.
1308
+ #
1309
+ # +key+:: Session key used to retrieve the session again on the *next*
1310
+ # request. The session has already been stored under that new key
1311
+ # internally. Do not call with +nil+ or blank keys.
1312
+ #
1313
+ def hubssolib_store_key(key)
1314
+ cookies[HUB_COOKIE_NAME] = {
1315
+ value: key,
1316
+ path: '/',
1317
+ domain: :all,
1318
+ secure: ! hub_bypass_ssl?,
1319
+ httponly: true
1320
+ }
1321
+ end
1322
+
1107
1323
  # Indicate that the user must log in to complete their request.
1108
1324
  # Returns false to enable a before_filter to return through this
1109
1325
  # method while ensuring that the previous action processing is
@@ -1164,64 +1380,45 @@ module HubSsoLib
1164
1380
  # have not logged out anyway, and the Hub isn't intended for Fort Knox.
1165
1381
  # At the time of writing the trade-off of usability vs security is
1166
1382
  # considered acceptable, though who knows, the view may change in future.
1383
+ #
1384
+ # Same applies for PATCH and PUT.
1385
+ #
1386
+ last_used = self.hubssolib_get_last_used
1167
1387
 
1168
- last_used = hubssolib_get_last_used
1169
- (request.method != :post && last_used && Time.now.utc - last_used > HUB_IDLE_TIME_LIMIT)
1170
- end
1171
-
1172
- def hubssolib_get_session_proxy
1173
- # If we're not using SSL, forget it
1174
- return nil unless request.ssl? || hub_bypass_ssl?
1175
-
1176
- key = cookies[HUB_COOKIE_NAME] || SecureRandom.uuid
1177
- hub_session = hubssolib_factory().get_hub_session_proxy(key, request.remote_ip)
1178
- key = hub_session.session_key_rotation unless hub_session.nil?
1179
-
1180
- cookies[HUB_COOKIE_NAME] = {
1181
- value: key,
1182
- path: '/',
1183
- domain: :all,
1184
- secure: ! hub_bypass_ssl?,
1185
- httponly: true
1186
- }
1187
-
1188
- return hub_session
1189
-
1190
- rescue Exception => e
1191
-
1192
- # At this point there tends to be no Session data, so we're
1193
- # going to have to encode the exception data into the URI...
1194
- # It must be escaped twice, as many servers treat "%2F" in a
1195
- # URI as a "/" and Apache may flat refuse to serve the page,
1196
- # raising a 404 error unless "AllowEncodedSlashes on" is
1197
- # specified in its configuration.
1198
-
1199
- suffix = '/' + CGI::escape(CGI::escape(hubssolib_set_exception_data(e)))
1200
- new_path = HUB_PATH_PREFIX + '/tasks/service'
1201
- redirect_to(new_path + suffix) unless request.path.include?(new_path)
1202
-
1203
- return nil
1204
- end
1388
+ (
1389
+ request.method != :post &&
1390
+ request.method != :patch &&
1391
+ request.method != :put &&
1205
1392
 
1206
- def hubssolib_set_session_data(session)
1207
- # Nothing to do presently - DRb handles everything
1393
+ last_used && Time.now.utc - last_used > HUB_IDLE_TIME_LIMIT
1394
+ )
1208
1395
  end
1209
1396
 
1210
- # Return an array of Hub User objects representing users based
1211
- # on a list of known sessions returned by the DRb server. Note
1212
- # that if an application exposes this method to a view, it is
1213
- # up to the application to ensure sufficient access permission
1214
- # protection for that view according to the webmaster's choice
1215
- # of site security level. Generally, normal users should not
1216
- # be allowed access.
1397
+ # Return an array of Hub User objects representing users based on a list of
1398
+ # known sessions returned by the DRb server. Note that if an application
1399
+ # exposes this method to a view, it is up to the application to ensure
1400
+ # sufficient access permission protection for that view according to the
1401
+ # webmaster's choice of site security level. Generally, normal users should
1402
+ # not be allowed access!
1403
+ #
1404
+ # Due to the session pool being held in the DRb server and subject to
1405
+ # alteration at any time by other requests (assuming the server supports
1406
+ # more than just one request at a time, that is!) then the session data
1407
+ # must be copied locally before iterating. Otherwise, exceptions arise from
1408
+ # attempts to alter an under-iteration Hash. This in turn raises a worry
1409
+ # about RAM usage. For that reason, a (somewhat arbitrary) limit of
1410
+ # 2000 active users is applied. More than that and the method returns an
1411
+ # empty array.
1217
1412
  #
1218
1413
  def hubssolib_enumerate_users
1219
- sessions = hubssolib_factory().enumerate_hub_sessions()
1220
- users = []
1221
-
1222
- sessions.each do |key, value|
1223
- user = value.session_user
1224
- users.push(user) if (user && user.respond_to?(:user_id) && user.user_id)
1414
+ hub_session_data = hubssolib_factory().enumerate_hub_sessions()
1415
+ return [] if hub_session_data.keys.size > 2000 # NOTE EARLY EXIT
1416
+
1417
+ sessions = hub_session_data.values.dup
1418
+ users = sessions.inject( [] ) do | memo, session |
1419
+ user = session.session_user
1420
+ memo << user unless user&.user_id.nil?
1421
+ memo
1225
1422
  end
1226
1423
 
1227
1424
  return users
@@ -1263,23 +1460,21 @@ module HubSsoLib
1263
1460
  # the session data is available, else return default values.
1264
1461
 
1265
1462
  def hubssolib_get_last_used
1266
- session = self.hubssolib_current_session
1267
- session ? session.session_last_used : Time.now.utc
1463
+ self.hubssolib_get_session()&.session_last_used || Time.now.utc
1268
1464
  end
1269
1465
 
1270
1466
  def hubssolib_set_last_used(time)
1271
- return unless self.hubssolib_current_session
1272
- self.hubssolib_current_session.session_last_used = time
1467
+ session = self.hubssolib_get_session()
1468
+ session.session_last_used = time unless session.nil?
1273
1469
  end
1274
1470
 
1275
1471
  def hubssolib_get_return_to
1276
- session = self.hubssolib_current_session
1277
- session ? session.session_return_to : nil
1472
+ self.hubssolib_get_session()&.session_return_to
1278
1473
  end
1279
1474
 
1280
1475
  def hubssolib_set_return_to(uri)
1281
- return unless self.hubssolib_current_session
1282
- self.hubssolib_current_session.session_return_to = uri
1476
+ session = self.hubssolib_get_session()
1477
+ session.session_return_to = uri unless session.nil?
1283
1478
  end
1284
1479
 
1285
1480
  end # Core module
metadata CHANGED
@@ -1,13 +1,13 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: hubssolib
3
3
  version: !ruby/object:Gem::Version
4
- version: 3.3.0
4
+ version: 3.4.0
5
5
  platform: ruby
6
6
  authors:
7
7
  - Andrew Hodgkinson and others
8
8
  bindir: bin
9
9
  cert_chain: []
10
- date: 2025-02-16 00:00:00.000000000 Z
10
+ date: 2025-03-24 00:00:00.000000000 Z
11
11
  dependencies:
12
12
  - !ruby/object:Gem::Dependency
13
13
  name: drb