hubssolib 3.4.0 → 3.6.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: c1a83a70b87a6c3cabd8bbb821dfc06ee467d382c9593cb7c1733152e136f5a1
4
- data.tar.gz: 87f1dda92220f25326c8bbce597ecd97e74c3dd7e22fd0fbf89bfed6269b6b23
3
+ metadata.gz: faacd8f740f867ee072f0f8e0f15324109941026b836cb3fcc8618f61ee68e02
4
+ data.tar.gz: 6ef08dccd8bbe03a974c238353a27d7f249a40911a40d65dfcd481dd6fe7b48e
5
5
  SHA512:
6
- metadata.gz: cf2ae3afb91a4fcc3391cf4b156e3dd9e4d0be4cecafc10204cf6dc8a7d6b9d07d51dc08cd7fd22ee12a478025d03730227fa2602a194d4b0e6785d72af21295
7
- data.tar.gz: 6ac8d8a0a1920d7f93604204ef0c254a094bd4a8559110bba2f630c1fe406f87966b0ba2289d017a9a48e90cd0d6e2eab0e1ae3872032c7bad389942965fe2b2
6
+ metadata.gz: 67b9f58743d8f9ed2983d3b9b970fd00d50ba335dcd761f8c9626866c9c91a1e65c378bb0139835c1e8eb48557e681ea510f6391373d06e1a6806aaf8b8fc2e2
7
+ data.tar.gz: d4e6efe525fb55b25f40c888ab4d05a7041c90c1cddcba7bab74002d299b2a1f96e35f58ea211cb666da68fda3d129400a7bf874bc8b7ddf09e9fdd6a8d1fbd2
data/CHANGELOG.md CHANGED
@@ -1,3 +1,22 @@
1
+ ## 3.6.0, 26-Mar-2025
2
+
3
+ Cleans up and offers new enumeration features. Ordering by last-recently-active first allows clients to be deterministic about enumerated sessions. Features created to support improvements in the Hub app v3.6.0.
4
+
5
+ Note that the session generator in the factory - HubSsoLib::SessionFactory#get_hub_session_proxy - no longer pays attention to IP address parameter, which should now be omitted (it is now an ignored parameter that defaults to +nil+). See implementation comments for rationale, but basically, IP addresses can legitimately change for users due to DHCP (even if that's rare) and given v3.5.0's on-shutdown session store, it didn't seem wise to keep IP addresses around inside there for any length of time. It was cleanest to just drop them. PII in persisted data is once again limited to "real name" and e-mail address.
6
+ s
7
+ ## 3.5.0, 25-Mar-2025
8
+
9
+ Builds on the cleaner session interface with some changes and improvements:
10
+
11
+ * Removed the 'clean up other sessions under that user ID' code, since that stopped you being able to log in on more than one browser - notably, both e.g. a desktop/laptop computer's browser and a mobile device's browser. This does carry a risk of stale session cookies for a user building up over time, but the sweeper Rake task will attend to those in due course.
12
+ * On the other hand the Hub core explicitly say whether or not it is trying to read or create a session, so the internal lazy-create no longer triggers for outdated session cookies, resulting in unnecessary "empty" sessions where all the user properties are "nil". This prevents accumulation of sessions for some cases.
13
+
14
+ Minor version number arises due to a new feature:
15
+
16
+ * If the DRb server receives signals `TERM` (e.g. default `kill`) or `INT` (e.g. `Ctrl`+`C`) it now gracefully shuts down, dumping user sessions to a YAML file in a location of `~/.hub_ses_arc` by default, or the filename in environment variable `HUB_SESSION_ARCHIVE`, if defined.
17
+ * On startup, this file is loaded. If exceptions are encountered they are ignored and session data is ignored, leading to a clean start. Likewise simply deleting the file leads to a clean start - otherwise, it becomes possible to cycle the DRb server and retain session data! **A self-sweep of ancient sessions is conducted at startup**, which takes the same action as the `hub_sessions:sweep` Rake task.
18
+ * Once the server is running the file is deleted, so in case of hard crash or forced exit via other signals (e.g. `kill -9`) - so a new session dump is _not_ made - then session data is lost, by design. The assumption in such circumstances is that session data is unreliable and may even have been the cause of a crash.
19
+
1
20
  ## 3.4.0, 24-Mar-2025
2
21
 
3
22
  The session interface has now been cleaned up. While the public API is unchanged, you:
@@ -16,7 +35,6 @@ In addition:
16
35
 
17
36
  * 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
37
 
19
-
20
38
  ## 3.3.0, 16-Feb-2025
21
39
 
22
40
  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.4.0)
4
+ hubssolib (3.5.0)
5
5
  base64 (~> 0.2)
6
6
  drb (~> 2.2)
7
7
 
@@ -13,7 +13,7 @@ GEM
13
13
  debug (1.10.0)
14
14
  irb (~> 1.10)
15
15
  reline (>= 0.3.8)
16
- diff-lcs (1.6.0)
16
+ diff-lcs (1.6.1)
17
17
  docile (1.4.1)
18
18
  doggo (1.4.0)
19
19
  rspec-core (~> 3.13)
data/README.md CHANGED
@@ -208,6 +208,8 @@ User-idle session expiry is routinely handled in the "beforehand" callback so th
208
208
 
209
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
210
 
211
+ * **Note:** The DRb server persists sessions upon graceful shutdown (`INT` or `TERM` signals) into file `~/.hub_ses_arc` by default, or the filename in environment variable `HUB_SESSION_ARCHIVE` if defined. When the server restarts, it loads this data **and runs the same very old session expiry** algorithm. This means that shutting down and restarting the DRb server is another way to expire very old sessions, but that results in downtime, even if only brief; so for general maintenance with uptime maintained, the Rake task should be used.
212
+
211
213
 
212
214
 
213
215
  ## Hub library API
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.4.0'
7
+ s.version = '3.6.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
@@ -1,6 +1,6 @@
1
1
  #######################################################################
2
2
  # Module: HubSsoLib #
3
- # (C) Hipposoft 2006-2019 #
3
+ # (C) Hipposoft 2006-2025 #
4
4
  # #
5
5
  # Purpose: Cross-application same domain single sign-on support. #
6
6
  # #
@@ -13,6 +13,7 @@
13
13
  # 09-Mar-2011 (ADH): Updated for Hub on Rails 2.3.11 along #
14
14
  # with several important bug fixes. #
15
15
  # 01-May-2019 (ADH): Updated for Ruby 2.5.x. #
16
+ # 25-Mar-2025 (ADH): Major 2025 overhaul for ROOL site. #
16
17
  #######################################################################
17
18
 
18
19
  module HubSsoLib
@@ -20,8 +21,10 @@ module HubSsoLib
20
21
  require 'drb'
21
22
  require 'securerandom'
22
23
  require 'json'
24
+ require 'yaml'
23
25
 
24
- # DRb connection
26
+ # DRb connection.
27
+ #
25
28
  HUB_CONNECTION_URI = ENV['HUB_CONNECTION_URI'] || 'drbunix:' + File.join( ENV['HOME'] || '/', '/.hub_drb')
26
29
 
27
30
  unless HUB_CONNECTION_URI.downcase.start_with?('drbunix:')
@@ -34,7 +37,8 @@ module HubSsoLib
34
37
  raise 'Exiting'
35
38
  end
36
39
 
37
- # External application command registry for on-user-change events
40
+ # External application command registry for on-user-change events.
41
+ #
38
42
  HUB_COMMAND_REGISTRY = ENV['HUB_COMMAND_REGISTRY'] || File.join( ENV['HOME'] || '/', '/.hub_cmd_reg')
39
43
 
40
44
  unless Dir.exist?(File.dirname(HUB_COMMAND_REGISTRY))
@@ -47,6 +51,20 @@ module HubSsoLib
47
51
  raise 'Exiting'
48
52
  end
49
53
 
54
+ # DRb session on-shutdown dumped cache/archive.
55
+ #
56
+ HUB_SESSION_ARCHIVE = ENV['HUB_SESSION_ARCHIVE'] || File.join( ENV['HOME'] || '/', '/.hub_ses_arc')
57
+
58
+ unless Dir.exist?(File.dirname(HUB_SESSION_ARCHIVE))
59
+ puts
60
+ puts '*' * 80
61
+ puts "Invalid path specified by HUB_SESSION_ARCHIVE (#{ HUB_SESSION_ARCHIVE.inspect })"
62
+ puts '*' * 80
63
+ puts
64
+
65
+ raise 'Exiting'
66
+ end
67
+
50
68
  # Location of Hub application root.
51
69
  #
52
70
  HUB_PATH_PREFIX = ENV['HUB_PATH_PREFIX'] || ''
@@ -57,19 +75,33 @@ module HubSsoLib
57
75
  #
58
76
  HUB_IDLE_TIME_LIMIT = ENV['HUB_IDLE_TIME_LIMIT']&.to_i || 4 * 60 * 60
59
77
 
78
+ # Archive time for very old sessions that are kicked out foricbly.
79
+ #
80
+ HUB_ARCHIVE_TIME_LIMIT = [ HUB_IDLE_TIME_LIMIT * 3, 172_800 ].max() # (2 days or more)
81
+
82
+ # If you use live session enumeration, this is the hard-coded limit for the
83
+ # number of session keys that can be returned. If there are more active
84
+ # sessions that this, bulk session enumeration is not permitted.
85
+ #
86
+ HUB_SESSION_ENUMERATION_KEY_MAX = ENV['HUB_SESSION_ENUMERATION_KEY_MAX']&.to_i || 2000
87
+
60
88
  # Shared cookie name.
89
+ #
61
90
  HUB_COOKIE_NAME = :hubapp_shared_id
62
91
 
63
92
  # Principally for #hubssolib_account_link.
93
+ #
64
94
  HUB_LOGIN_INDICATOR_COOKIE = :hubapp_shared_id_alive
65
95
  HUB_LOGIN_INDICATOR_COOKIE_VALUE = 'Y'
66
96
 
67
97
  # Bypass SSL, for testing purposes? Rails 'production' mode will
68
98
  # insist on SSL otherwise. Development & test environments do not,
69
99
  # so do not need this variable setting.
100
+ #
70
101
  HUB_BYPASS_SSL = ( ENV['HUB_BYPASS_SSL'] == "true" )
71
102
 
72
103
  # Thread safety.
104
+ #
73
105
  HUB_MUTEX = Mutex.new
74
106
 
75
107
  #######################################################################
@@ -403,7 +435,6 @@ module HubSsoLib
403
435
  attr_accessor :session_flash
404
436
  attr_accessor :session_user
405
437
  attr_accessor :session_rotated_key
406
- attr_accessor :session_ip
407
438
 
408
439
  def initialize
409
440
  @session_last_used = Time.now.utc
@@ -411,7 +442,6 @@ module HubSsoLib
411
442
  @session_flash = {}
412
443
  @session_user = HubSsoLib::User.new
413
444
  @session_rotated_key = nil
414
- @session_ip = nil
415
445
 
416
446
  rescue => e
417
447
  Sentry.capture_exception(e) if defined?(Sentry) && Sentry.respond_to?(:capture_exception)
@@ -438,7 +468,33 @@ module HubSsoLib
438
468
  @hub_be_quiet = ! ENV['HUB_QUIET_SERVER'].nil?
439
469
  @hub_sessions = {}
440
470
 
441
- puts "Session factory: Awaken" unless @hub_be_quiet
471
+ puts "Session factory: Awakening..." unless @hub_be_quiet
472
+
473
+ if File.exist?(HUB_SESSION_ARCHIVE)
474
+ begin
475
+ restored_sessions = ::YAML.load_file(
476
+ HUB_SESSION_ARCHIVE,
477
+ permitted_classes: [
478
+ ::HubSsoLib::Session,
479
+ ::HubSsoLib::User,
480
+ Time
481
+ ]
482
+ )
483
+
484
+ @hub_sessions = restored_sessions || {}
485
+ self.destroy_ancient_sessions()
486
+ puts "Session factory: Reloaded #{@hub_sessions.size} from archive" unless @hub_be_quiet
487
+
488
+ rescue => e
489
+ puts "Session factory: Ignored archive due to error #{e.message.inspect}" unless @hub_be_quiet
490
+
491
+ ensure
492
+ File.unlink(HUB_SESSION_ARCHIVE)
493
+
494
+ end
495
+ end
496
+
497
+ puts "Session factory: ...Awakened" unless @hub_be_quiet
442
498
 
443
499
  rescue => e
444
500
  Sentry.capture_exception(e) if defined?(Sentry) && Sentry.respond_to?(:capture_exception)
@@ -456,35 +512,39 @@ module HubSsoLib
456
512
  #
457
513
  # The returned object is proxied via DRb - it is shared between processes.
458
514
  #
459
- # +key+:: Session key; lazy-initialises a new session under this key
460
- # if none is found, then immediately rotates it.
515
+ # +key+:: Session key; lazy-initialises a new session under this key if
516
+ # none is found, then immediately rotates it by default, but may
517
+ # return no session for unrecognised keys depending on the +create+
518
+ # parameter, described below.
519
+ #
520
+ # The "_ignored" parameter is for backwards compatibility for older clients
521
+ # calling into a newer gem. This used to take an IP address of the request
522
+ # and would discard a session if the current IP address had changed, but
523
+ # since DHCP is a Thing then - even though in practice most IP addresses
524
+ # from ISPs are very stable - this wasn't really a valid security measure.
525
+ # It required us to process and store IP data which is now often considered
526
+ # PII and we'd rather not (especially given their arising storage in
527
+ # HUB_SESSION_ARCHIVE on shutdown). This is, of course, quite ironic given
528
+ # the reason for removal is IP address unreliability when used as PII!
461
529
  #
462
- # +remote_ip+:: Request's remote IP address. If there is an existing
463
- # session which matches this, it's returned. If there is an
464
- # existing session but the IP mismatches, it's treated as
465
- # invalid and discarded.
530
+ # In addition, the following optional named parameters can be given:
466
531
  #
467
- def get_hub_session_proxy(key, remote_ip)
532
+ # +create+:: Default +true+ - an unknown key causes creation of an empty,
533
+ # new session under that key. If +false+, attempts to read with
534
+ # an unrecognised key yield +nil+.
535
+ #
536
+ def get_hub_session_proxy(key, _ignored = nil, create: true)
468
537
  hub_session = HUB_MUTEX.synchronize { @hub_sessions[key] }
469
- message = hub_session.nil? ? 'Created' : 'Retrieving'
470
- new_key = SecureRandom.uuid
538
+ return nil if create == false && hub_session.nil? # NOTE EARLY EXIT
539
+
540
+ message = hub_session.nil? ? 'Created' : 'Retrieving'
541
+ new_key = SecureRandom.uuid
471
542
 
472
543
  unless @hub_be_quiet
473
544
  puts "Session factory: #{ message } session for key #{ key } and rotating to #{ new_key }"
474
545
  end
475
546
 
476
- unless hub_session.nil? || hub_session.session_ip == remote_ip
477
- unless @hub_be_quiet
478
- puts "Session factory: WARNING: IP address changed from #{ hub_session.session_ip } to #{ remote_ip } -> discarding session"
479
- end
480
-
481
- hub_session = nil
482
- end
483
-
484
- if hub_session.nil?
485
- hub_session = HubSsoLib::Session.new
486
- hub_session.session_ip = remote_ip
487
- end
547
+ hub_session = HubSsoLib::Session.new if hub_session.nil?
488
548
 
489
549
  HUB_MUTEX.synchronize do
490
550
  @hub_sessions.delete(key)
@@ -499,34 +559,90 @@ module HubSsoLib
499
559
  raise
500
560
  end
501
561
 
562
+ # THIS INTERFACE IS DEPRECATED and will be removed in Hub 4. Change to
563
+ # #enumerate_hub_session_keys instead.
564
+ #
502
565
  # Enumerate all currently known sessions. The format is a Hash, with the
503
566
  # session key UUIDs as keys and the related HubSsoLib::Session instances as
504
- # values.
567
+ # values. If you attempt it iterate over this data YOU MUST USE A COPY of
568
+ # the keys to do so, since Hub users may log in or out at any time and Ruby
569
+ # will raise an exception if the session data changes during enumeration -
570
+ # end users will see errors.
571
+ #
572
+ def enumerate_hub_sessions()
573
+ @hub_sessions
574
+
575
+ rescue => e
576
+ Sentry.capture_exception(e) if defined?(Sentry) && Sentry.respond_to?(:capture_exception)
577
+ raise
578
+ end
579
+
580
+ # Returns all currently known session keys. This is a COPY of the internal
581
+ # keyset, since otherwise it would be necessary to try and enumerate keys
582
+ # on a Hash which is subject to change at any moment if a user logs in or
583
+ # out. Since Ruby raises an exception should an under-iteration Hash be
584
+ # changed, we can't do that, which is why the keys are returned as a copied
585
+ # array instead.
505
586
  #
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.
587
+ # Keys are ordered by least-recently-active first to most-recent last.
512
588
  #
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.
589
+ # Call #retrieve_session_by_key(...) to get session details for that key.
590
+ # Bear in mind that +nil+ returns are possible, since the session data may
591
+ # be changing rapidly and a user might've logged out or had their session
592
+ # expired in the time between you retrieving the list of current keys here,
593
+ # then requesting details of the session for that key later.
519
594
  #
520
- # See HubSsoLib::Core#hubssolib_enumerate_users for an example.
595
+ # To avoid unbounded RAM requirements arising, the maximum number of keys
596
+ # returned herein is limited to HUB_SESSION_ENUMERATION_KEY_MAX.
521
597
  #
522
- def enumerate_hub_sessions()
523
- @hub_sessions
598
+ # Returns a Hash with Symbol keys that have values as follows:
599
+ #
600
+ # +count+:: The number of known sessions - just a key count.
601
+ # +keys+:: If +count+ exceeds HUB_SESSION_ENUMERATION_KEY_MAX, this is
602
+ # +nil+, else an array of zero or more session keys.
603
+ #
604
+ def enumerate_hub_session_ids()
605
+ count = @hub_sessions.size
606
+ keys = if count > HUB_SESSION_ENUMERATION_KEY_MAX
607
+ nil
608
+ else
609
+ @hub_sessions.keys # (Hash#keys returns a new array)
610
+ end
611
+
612
+ return { count: count, keys: keys }
524
613
 
525
614
  rescue => e
526
615
  Sentry.capture_exception(e) if defined?(Sentry) && Sentry.respond_to?(:capture_exception)
527
616
  raise
528
617
  end
529
618
 
619
+ # Retrieve session data (as a HubSsoLib::Session instance) based on the
620
+ # given session key. No key rotation occurs. Returns +nil+ if no entry is
621
+ # found for that key.
622
+ #
623
+ def retrieve_session_by_key(key)
624
+ @hub_sessions[key]
625
+ end
626
+
627
+ # WARNING: Comparatively slow.
628
+ #
629
+ # This is usually only called in administrative interfaces to look at the
630
+ # known sessions for a specific user of interest. An Hash of session key
631
+ # values yielding HubSsoLib::Session instances as values is returned.
632
+ #
633
+ # The array is ordered by least-recently-active first to most-recent last.
634
+ #
635
+ # IN THE CURRENT IMPLEMENTATION THIS JUST SEQUENTIALLY SCANS ALL ACTIVE
636
+ # SESSIONS IN THE HASH and must therefore lock on mutex for the duration.
637
+ #
638
+ def retrieve_sessions_by_user_id(user_id)
639
+ HUB_MUTEX.synchronize do
640
+ @hub_sessions.select do | key, session |
641
+ session&.session_user&.user_id == user_id
642
+ end
643
+ end
644
+ end
645
+
530
646
  # Given a session key (which, if a session has been looked up and the key
531
647
  # thus rotated, ought to be that new, rotated key), destroy the associated
532
648
  # session data. Does nothing if the key is not found.
@@ -541,7 +657,7 @@ module HubSsoLib
541
657
  raise
542
658
  end
543
659
 
544
- # WARNING: Comparatively slow
660
+ # WARNING: Comparatively slow.
545
661
  #
546
662
  # This is called in rare cases such as user deletion or being asked for a
547
663
  # session under an old key, indicating loss of key rotation sequence.
@@ -562,7 +678,7 @@ module HubSsoLib
562
678
  raise
563
679
  end
564
680
 
565
- # WARNING: Slow
681
+ # WARNING: Slow.
566
682
  #
567
683
  # This is a housekeeping task which checks sessions against Hub expiry and,
568
684
  # if the session keys look to be substantially older than the value set in
@@ -577,23 +693,21 @@ module HubSsoLib
577
693
  # session data while the method runs will block until method finishes.
578
694
  #
579
695
  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
696
  unless @hub_be_quiet
585
- puts "Session factory: Sweeping sessions inactive for more than #{ time_limit } seconds..."
697
+ puts "Session factory: Sweeping sessions inactive for more than #{ HUB_ARCHIVE_TIME_LIMIT } seconds..."
586
698
  end
587
699
 
700
+ destroyed = 0
701
+
588
702
  HUB_MUTEX.synchronize do
589
- count_before = @hub_sessions.keys.size
703
+ count_before = @hub_sessions.size
590
704
 
591
705
  @hub_sessions.reject! do | key, session |
592
706
  last_used = session&.session_last_used
593
- last_used.nil? || Time.now.utc - last_used > time_limit
707
+ last_used.nil? || Time.now.utc - last_used > HUB_ARCHIVE_TIME_LIMIT
594
708
  end
595
709
 
596
- count_after = @hub_sessions.keys.size
710
+ count_after = @hub_sessions.size
597
711
  destroyed = count_before - count_after
598
712
  end
599
713
 
@@ -606,6 +720,41 @@ module HubSsoLib
606
720
  raise
607
721
  end
608
722
 
723
+ # Lock the session store and dump all sessions with a non-nil session user
724
+ # ID to a YAML file at HUB_SESSION_ARCHIVE. This is expected to only be
725
+ # called by the graceful shutdown code in HubSsoLib::Server.
726
+ #
727
+ def dump_sessions!
728
+ written_record_count = 0
729
+
730
+ # Why not just do ::YAML.dump(@hub_sessions)? Well, it'd be faster, but
731
+ # it builds the YAML data all in RAM which would cause a huge RAM spike
732
+ # of unknown size (depends on live session count) and that's Bad.
733
+ #
734
+ # If any no-user sessions have crept in for any reason, this also gives
735
+ # us a chance to skip them.
736
+ #
737
+ HUB_MUTEX.synchronize do
738
+ File.open(HUB_SESSION_ARCHIVE, 'w') do | f |
739
+ f.write("---\n") # (document marker)
740
+
741
+ @hub_sessions.each do | key, session |
742
+ next if session&.session_user&.user_id.nil? # NOTE EARLY LOOP RESTART
743
+
744
+ dump = ::YAML.dump({key => session})
745
+ dump.sub!(/^---\n/, '') # (avoid multiple document markers)
746
+
747
+ f.write(dump)
748
+ written_record_count += 1
749
+ end
750
+ end
751
+ end
752
+
753
+ # Simple if slightly inefficient way to deal with zero actual useful
754
+ # session records being present - an unusual real-world edge case.
755
+ #
756
+ File.unlink(HUB_SESSION_ARCHIVE) if written_record_count == 0
757
+ end
609
758
  end
610
759
 
611
760
  #######################################################################
@@ -626,16 +775,41 @@ module HubSsoLib
626
775
 
627
776
  module Server
628
777
  def hubssolib_launch_server
629
- puts "Server: Starting at #{ HUB_CONNECTION_URI }" unless ENV['HUB_QUIET_SERVER'].nil?
778
+ ::HubSsoLib::Server::Runner.run
779
+ end
630
780
 
631
- @@hub_session_factory = HubSsoLib::SessionFactory.new
632
- DRb.start_service(HUB_CONNECTION_URI, @@hub_session_factory, { :safe_level => 1 })
633
- DRb.thread.join
781
+ class Runner
782
+ QUEUE = ::Queue.new
634
783
 
635
- rescue => e
636
- Sentry.capture_exception(e) if defined?(Sentry) && Sentry.respond_to?(:capture_exception)
637
- raise
638
- end
784
+ def self.run
785
+ puts "Server: Starting at #{ HUB_CONNECTION_URI }" if ENV['HUB_QUIET_SERVER'].nil?
786
+
787
+ @@hub_session_factory = HubSsoLib::SessionFactory.new
788
+
789
+ Signal.trap('INT' ) { QUEUE << :INT }
790
+ Signal.trap('TERM') { QUEUE << :TERM }
791
+
792
+ DRb.start_service(HUB_CONNECTION_URI, @@hub_session_factory, { :safe_level => 1 })
793
+
794
+ QUEUE.pop
795
+
796
+ self.shutdown()
797
+ exit
798
+
799
+ rescue => e
800
+ Sentry.capture_exception(e) if defined?(Sentry) && Sentry.respond_to?(:capture_exception)
801
+ raise
802
+ end
803
+
804
+ def self.shutdown
805
+ puts "Server: Graceful shutdown..."
806
+
807
+ @@hub_session_factory.dump_sessions!
808
+ DRb.stop_service
809
+
810
+ puts "Server: ...completed."
811
+ end
812
+ end # Runner class
639
813
  end # Server module
640
814
 
641
815
  #######################################################################
@@ -689,6 +863,18 @@ module HubSsoLib
689
863
  user = hub_session&.session_user
690
864
 
691
865
  return (user&.user_id.nil? ? nil : user)
866
+
867
+ rescue Exception => e
868
+
869
+ # At this point there tends to be no Session data, so we're going to have
870
+ # to encode the exception data into the URI... It must be escaped twice,
871
+ # as many servers treat "%2F" in a URI as a "/". Apache can then fail to
872
+ # serve the page, raising a 404 error unless "AllowEncodedSlashes on" is
873
+ # specified in its configuration.
874
+ #
875
+ suffix = '/' + CGI::escape(CGI::escape(hubssolib_set_exception_data(e)))
876
+ new_path = HUB_PATH_PREFIX + '/tasks/service'
877
+ redirect_to(new_path + suffix) unless request.path.include?(new_path)
692
878
  end
693
879
 
694
880
  # Sets the currently signed in user. Note that although this works and is
@@ -701,13 +887,6 @@ module HubSsoLib
701
887
  if user.nil?
702
888
  self.hubssolib_destroy_session!
703
889
  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
890
  hub_session = self.hubssolib_create_session()
712
891
  hub_session.session_user = user
713
892
  end
@@ -865,6 +1044,80 @@ module HubSsoLib
865
1044
  user ? "#{user.user_real_name} (#{user.user_id})" : 'Anonymous'
866
1045
  end
867
1046
 
1047
+ # WARNING: Slow.
1048
+ #
1049
+ # Return an Array of HubSsoLib::User objects for all logged-in users, in an
1050
+ # array. If a user is logged into more than one browser and thus has more
1051
+ # than one session active, they will accordingly appear more than once in
1052
+ # the returned data. This can also happen if a user loses key rotation and
1053
+ # leaves an old, but not yet expired session behind when they log in anew.
1054
+ #
1055
+ # In accordance with HubSsoLib::SessionFactory#enumerate_hub_session_ids
1056
+ # documentation, a maximum of HUB_SESSION_ENUMERATION_KEY_MAX users can be
1057
+ # returned here. If this is exceeded, an *empty* array is returned. If you
1058
+ # are processing this information for display in a UI which itself requires
1059
+ # a logged in user of some sort (which is very likely) and therefore know
1060
+ # that at least *one* session does exist, you can treat an empty array as
1061
+ # confirmation of "lots of sessions". If you can't be sure of at least one
1062
+ # logged in user, then there is obvious arising ambiguity and this method
1063
+ # does not solve it for you - it's just "zero, or very many users".
1064
+ #
1065
+ # Users are ordered by least-recently-active first, most-recent last.
1066
+ #
1067
+ # For information about performance limitations, see
1068
+ # HubSsoLib::SessionFactory#enumerate_hub_session_ids.
1069
+ #
1070
+ def hubssolib_enumerate_users
1071
+ hub_session_info = hubssolib_factory().enumerate_hub_session_ids()
1072
+ hub_users = []
1073
+
1074
+ unless hub_session_info[:keys].nil? # (keyset too large, enumeration prohibited)
1075
+ hub_session_info[:keys].each do | key |
1076
+ session = hubssolib_factory().retrieve_session_by_key(key)
1077
+ hub_users << session.session_user unless session&.session_user&.user_id.nil?
1078
+ end
1079
+ end
1080
+
1081
+ return hub_users
1082
+ end
1083
+
1084
+ # WARNING: Comparatively slow.
1085
+ #
1086
+ # For a given HubSsoLib::User's user ID, return any known sessions held by
1087
+ # the DRb server, as an Hash of session keys with HubSsoLib::Session
1088
+ # instances as values.
1089
+ #
1090
+ # Returns an empty Hash if the given ID is +nil+.
1091
+ #
1092
+ # Note that Hub sessions can disappear at any moment, so the session keys
1093
+ # you find in the Hash might refer to extinct sessions by the time you get
1094
+ # to do something with them. You can still access the data, but if you were
1095
+ # to try and ask the DRb server for that key, it'd return +nil+.
1096
+ #
1097
+ # Sessions are ordered by least-recently-active first, most-recent last.
1098
+ #
1099
+ # For information about performance limitations, see
1100
+ # HubSsoLib::SessionFactory#retrieve_sessions_by_user_id.
1101
+ #
1102
+ def hubssolib_retrieve_user_sessions(hub_user_id)
1103
+ if hub_user_id.nil?
1104
+ {}
1105
+ else
1106
+ hubssolib_factory().retrieve_sessions_by_user_id(hub_user_id)
1107
+ end
1108
+ end
1109
+
1110
+ # WARNING: Comparatively slow.
1111
+ #
1112
+ # Remove all sessions under a given ID.
1113
+ #
1114
+ # For information about performance limitations, see
1115
+ # HubSsoLib::SessionFactory#destroy_sessions_by_user_id.
1116
+ #
1117
+ def hubssolib_destroy_user_sessions(hub_user_id)
1118
+ hubssolib_factory().destroy_sessions_by_user_id(hub_user_id) unless hub_user_id.nil?
1119
+ end
1120
+
868
1121
  # If an application needs to know about changes of a user e-mail address
869
1122
  # or display name (e.g. because of sync to a local relational store of
870
1123
  # users related to other application-managed resources, with therefore a
@@ -1052,18 +1305,17 @@ module HubSsoLib
1052
1305
  # We can return to this location by calling #redirect_back_or_default.
1053
1306
  #
1054
1307
  def hubssolib_store_location(uri_str = request.url)
1055
-
1056
1308
  if (uri_str && !uri_str.empty?)
1057
1309
  uri_str = hubssolib_promote_uri_to_ssl(uri_str, request.host) unless request.ssl?
1058
1310
  hubssolib_set_return_to(uri_str)
1059
1311
  else
1060
1312
  hubssolib_set_return_to(nil)
1061
1313
  end
1062
-
1063
1314
  end
1064
1315
 
1065
1316
  # Redirect to the URI stored by the most recent store_location call or
1066
1317
  # to the passed default.
1318
+ #
1067
1319
  def hubssolib_redirect_back_or_default(default)
1068
1320
  url = hubssolib_get_return_to()
1069
1321
  hubssolib_set_return_to(nil)
@@ -1075,7 +1327,7 @@ module HubSsoLib
1075
1327
  # sets the host you provide (or leaves it alone if you omit the
1076
1328
  # parameter), then forces the scheme to 'https'. Returns the result
1077
1329
  # as a flat string.
1078
-
1330
+ #
1079
1331
  def hubssolib_promote_uri_to_ssl(uri_str, host = nil)
1080
1332
  uri = URI.parse(uri_str)
1081
1333
  uri.host = host if host
@@ -1231,7 +1483,7 @@ module HubSsoLib
1231
1483
  self.hubssolib_destroy_session! unless old_session.nil?
1232
1484
 
1233
1485
  start_key = SecureRandom.uuid
1234
- @hubssolib_session = hubssolib_factory().get_hub_session_proxy(start_key, request.remote_ip)
1486
+ @hubssolib_session = hubssolib_factory().get_hub_session_proxy(start_key, create: true)
1235
1487
 
1236
1488
  # The session is now stored under the rotated key, so put that into the
1237
1489
  # Hub session cookie so that on the next request, we can retrieve the
@@ -1263,10 +1515,12 @@ module HubSsoLib
1263
1515
  if @hubssolib_session.nil?
1264
1516
  key = cookies[HUB_COOKIE_NAME]
1265
1517
 
1266
- if key.nil? || key == ''
1518
+ # See #hubssolib_create_session - valid keys are 36-char UUIDs
1519
+ #
1520
+ if key.nil? || key.size != 36
1267
1521
  @hubssolib_session = nil
1268
1522
  else
1269
- hub_session = hubssolib_factory().get_hub_session_proxy(key, request.remote_ip)
1523
+ hub_session = hubssolib_factory().get_hub_session_proxy(key, create: false)
1270
1524
 
1271
1525
  if hub_session.nil? # Invalid key in cookie
1272
1526
  @hubssolib_session = nil
@@ -1326,13 +1580,14 @@ module HubSsoLib
1326
1580
  # halted (since the overall return value is therefore 'false').
1327
1581
  #
1328
1582
  def hubssolib_must_login
1583
+
1329
1584
  # If HTTP, redirect to the same place, but HTTPS. Then we can store the
1330
1585
  # flash and return-to in the session data. We'll have the same set of
1331
1586
  # before-filter operations running and they'll find out we're either
1332
1587
  # authorised after all, or come back to this very function, which will
1333
1588
  # now be happily running from an HTTPS connection and will go on to set
1334
1589
  # the flash and redirect to the login page.
1335
-
1590
+ #
1336
1591
  if hubssolib_ensure_https
1337
1592
  hubssolib_set_flash(:alert, 'You must log in before you can continue.')
1338
1593
  redirect_to HUB_PATH_PREFIX + '/account/login'
@@ -1394,47 +1649,6 @@ module HubSsoLib
1394
1649
  )
1395
1650
  end
1396
1651
 
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.
1412
- #
1413
- def hubssolib_enumerate_users
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
1422
- end
1423
-
1424
- return users
1425
-
1426
- rescue Exception => e
1427
-
1428
- # At this point there tends to be no Session data, so we're
1429
- # going to have to encode the exception data into the URI...
1430
- # See earlier for double-escaping rationale.
1431
-
1432
- suffix = '/' + CGI::escape(CGI::escape(hubssolib_set_exception_data(e)))
1433
- new_path = HUB_PATH_PREFIX + '/tasks/service'
1434
- redirect_to new_path + suffix unless request.path.include?(new_path)
1435
- return nil
1436
- end
1437
-
1438
1652
  # Encode exception data into a string suitable for using in a URL
1439
1653
  # if CGI escaped first. Pass the exception object; stores only the
1440
1654
  # message.
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.4.0
4
+ version: 3.6.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-03-24 00:00:00.000000000 Z
10
+ date: 2025-03-26 00:00:00.000000000 Z
11
11
  dependencies:
12
12
  - !ruby/object:Gem::Dependency
13
13
  name: drb