hubssolib 3.5.0 → 3.6.1

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: 8fc6eb926595d43f0fe51803fb2dd817930adf4385a742461c13197c87ebd8a5
4
- data.tar.gz: 3f532eca169d7497d595aeb15a31d47650ee13fd4973859041d40f66144217cc
3
+ metadata.gz: eaea423d2eec6433b30522f6a6d46ab3c1b2868bb860d985cd7472116a17bc95
4
+ data.tar.gz: 19ae77e97c1e480b6b6cba78eee22523d9964af677ac9a9531582aa68883350b
5
5
  SHA512:
6
- metadata.gz: 1c645f57251251d560c8940d25aaaade1e98b83fe7cf8ab0dff194c3738bfb0c0e8699a1aed7c4889b656e7c0da81f005722773c1af52205eb233f7f9b7ef3dc
7
- data.tar.gz: 6927b5bd57c5d5c7eddb65cb36c20fc89f24d6cf8ac1ad341da2feedf4ae6e12f452351416d089eb046214a4895d2e7becf7c098bb4c1cd0c011bf8eaafd36c2
6
+ metadata.gz: db1d1f887967b480cb7ef066b1d038ca4540c5aa18fdb307a42fa825f20ca5c9cc7390bccdf9d5b7de247d9ccdfe0b7219f40cc7c7e7cc7b283d2c0a9a458be5
7
+ data.tar.gz: ccc14fd7c1015d61bc40cde504976d4eafc38368fa2bcb076b3f1e89fe9974b5c2179f55295789c28f25256335f02479161d8966cc774ce8b924bbb41d646119
data/CHANGELOG.md CHANGED
@@ -1,3 +1,21 @@
1
+ ## 3.6.1, 27-Mar-2025
2
+
3
+ Some fixes:
4
+
5
+ * Session reload could fail if a Hub flash happened to be persisted, because YAML won't load Symbol by default and it wasn't in the allow-list.
6
+ - Remove the flash message from the dump since some flash from "whenever" may well be confusing if reloaded and shown at some later time.
7
+ - Add Symbol so that if restarting under this gem version, a dump from an older version which _does_ contain flash data will still load OK.
8
+
9
+ * Secure services failure redirection _still_ wasn't quite in the right place - the earliest thing that runs is of course the 'before action' hook. Moved it there and added a Sentry warning too.
10
+
11
+ ## 3.6.0, 26-Mar-2025
12
+
13
+ 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.
14
+
15
+ * HubSsoLib::Core#enumerate_hub_sessions is deprecated. Use HubSsoLib::Core#enumerate_hub_session_keys instead.
16
+ * For any client code that might be "hitting the metal" and calling the DRb server directly, note that HubSsoLib::SessionFactory#get_hub_session_proxy no longer pays attention to IP address parameter and this should be removed (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.
17
+ * HubSsoLib::Core exception handling for the Hub app's "tasks" notification is more extensive. A few prior gem versions unwittingly restricted it to only one specific method call. Now it's done in the current user retrieval, which is an endpoint used by a majority of Core module method calls.
18
+
1
19
  ## 3.5.0, 25-Mar-2025
2
20
 
3
21
  Builds on the cleaner session interface with some changes and improvements:
@@ -29,7 +47,6 @@ In addition:
29
47
 
30
48
  * 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.
31
49
 
32
-
33
50
  ## 3.3.0, 16-Feb-2025
34
51
 
35
52
  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.6.1)
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)
@@ -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.5)
55
+ stringio (3.1.6)
56
56
 
57
57
  PLATFORMS
58
58
  ruby
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.5.0'
7
+ s.version = '3.6.1'
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
@@ -74,19 +75,33 @@ module HubSsoLib
74
75
  #
75
76
  HUB_IDLE_TIME_LIMIT = ENV['HUB_IDLE_TIME_LIMIT']&.to_i || 4 * 60 * 60
76
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
+
77
88
  # Shared cookie name.
89
+ #
78
90
  HUB_COOKIE_NAME = :hubapp_shared_id
79
91
 
80
92
  # Principally for #hubssolib_account_link.
93
+ #
81
94
  HUB_LOGIN_INDICATOR_COOKIE = :hubapp_shared_id_alive
82
95
  HUB_LOGIN_INDICATOR_COOKIE_VALUE = 'Y'
83
96
 
84
97
  # Bypass SSL, for testing purposes? Rails 'production' mode will
85
98
  # insist on SSL otherwise. Development & test environments do not,
86
99
  # so do not need this variable setting.
100
+ #
87
101
  HUB_BYPASS_SSL = ( ENV['HUB_BYPASS_SSL'] == "true" )
88
102
 
89
103
  # Thread safety.
104
+ #
90
105
  HUB_MUTEX = Mutex.new
91
106
 
92
107
  #######################################################################
@@ -420,7 +435,6 @@ module HubSsoLib
420
435
  attr_accessor :session_flash
421
436
  attr_accessor :session_user
422
437
  attr_accessor :session_rotated_key
423
- attr_accessor :session_ip
424
438
 
425
439
  def initialize
426
440
  @session_last_used = Time.now.utc
@@ -428,7 +442,6 @@ module HubSsoLib
428
442
  @session_flash = {}
429
443
  @session_user = HubSsoLib::User.new
430
444
  @session_rotated_key = nil
431
- @session_ip = nil
432
445
 
433
446
  rescue => e
434
447
  Sentry.capture_exception(e) if defined?(Sentry) && Sentry.respond_to?(:capture_exception)
@@ -464,13 +477,14 @@ module HubSsoLib
464
477
  permitted_classes: [
465
478
  ::HubSsoLib::Session,
466
479
  ::HubSsoLib::User,
467
- Time
480
+ Time,
481
+ Symbol
468
482
  ]
469
483
  )
470
484
 
471
485
  @hub_sessions = restored_sessions || {}
472
486
  self.destroy_ancient_sessions()
473
- puts "Session factory: Reloaded #{@hub_sessions.keys.size} from archive" unless @hub_be_quiet
487
+ puts "Session factory: Reloaded #{@hub_sessions.size} from archive" unless @hub_be_quiet
474
488
 
475
489
  rescue => e
476
490
  puts "Session factory: Ignored archive due to error #{e.message.inspect}" unless @hub_be_quiet
@@ -499,15 +513,20 @@ module HubSsoLib
499
513
  #
500
514
  # The returned object is proxied via DRb - it is shared between processes.
501
515
  #
502
- # +key+:: Session key; lazy-initialises a new session under this key
503
- # if none is found, then immediately rotates it by default,
504
- # but may return no session for unrecognised keys depending
505
- # on the +create+ parameter, described below.
506
- #
507
- # +remote_ip+:: Request's remote IP address. If there is an existing
508
- # session which matches this, it's returned. If there is an
509
- # existing session but the IP mismatches, it's treated as
510
- # invalid and discarded.
516
+ # +key+:: Session key; lazy-initialises a new session under this key if
517
+ # none is found, then immediately rotates it by default, but may
518
+ # return no session for unrecognised keys depending on the +create+
519
+ # parameter, described below.
520
+ #
521
+ # The "_ignored" parameter is for backwards compatibility for older clients
522
+ # calling into a newer gem. This used to take an IP address of the request
523
+ # and would discard a session if the current IP address had changed, but
524
+ # since DHCP is a Thing then - even though in practice most IP addresses
525
+ # from ISPs are very stable - this wasn't really a valid security measure.
526
+ # It required us to process and store IP data which is now often considered
527
+ # PII and we'd rather not (especially given their arising storage in
528
+ # HUB_SESSION_ARCHIVE on shutdown). This is, of course, quite ironic given
529
+ # the reason for removal is IP address unreliability when used as PII!
511
530
  #
512
531
  # In addition, the following optional named parameters can be given:
513
532
  #
@@ -515,7 +534,7 @@ module HubSsoLib
515
534
  # new session under that key. If +false+, attempts to read with
516
535
  # an unrecognised key yield +nil+.
517
536
  #
518
- def get_hub_session_proxy(key, remote_ip, create: true)
537
+ def get_hub_session_proxy(key, _ignored = nil, create: true)
519
538
  hub_session = HUB_MUTEX.synchronize { @hub_sessions[key] }
520
539
  return nil if create == false && hub_session.nil? # NOTE EARLY EXIT
521
540
 
@@ -526,18 +545,7 @@ module HubSsoLib
526
545
  puts "Session factory: #{ message } session for key #{ key } and rotating to #{ new_key }"
527
546
  end
528
547
 
529
- unless hub_session.nil? || hub_session.session_ip == remote_ip
530
- unless @hub_be_quiet
531
- puts "Session factory: WARNING: IP address changed from #{ hub_session.session_ip } to #{ remote_ip } -> discarding session"
532
- end
533
-
534
- hub_session = nil
535
- end
536
-
537
- if hub_session.nil?
538
- hub_session = HubSsoLib::Session.new
539
- hub_session.session_ip = remote_ip
540
- end
548
+ hub_session = HubSsoLib::Session.new if hub_session.nil?
541
549
 
542
550
  HUB_MUTEX.synchronize do
543
551
  @hub_sessions.delete(key)
@@ -552,34 +560,90 @@ module HubSsoLib
552
560
  raise
553
561
  end
554
562
 
563
+ # THIS INTERFACE IS DEPRECATED and will be removed in Hub 4. Change to
564
+ # #enumerate_hub_session_keys instead.
565
+ #
555
566
  # Enumerate all currently known sessions. The format is a Hash, with the
556
567
  # session key UUIDs as keys and the related HubSsoLib::Session instances as
557
- # values.
568
+ # values. If you attempt it iterate over this data YOU MUST USE A COPY of
569
+ # the keys to do so, since Hub users may log in or out at any time and Ruby
570
+ # will raise an exception if the session data changes during enumeration -
571
+ # end users will see errors.
558
572
  #
559
- # WARNING: This returns all Hub sessions maintained by the server instance
560
- # but it's returning a reference to the live object with everything being
561
- # managed over DRb. If a caller iterates over this object, then it can fall
562
- # foul of other request threads making changes to the underlying data and
563
- # Ruby raising exceptions about e.g. hashes being modified during
564
- # enumeration.
573
+ def enumerate_hub_sessions()
574
+ @hub_sessions
575
+
576
+ rescue => e
577
+ Sentry.capture_exception(e) if defined?(Sentry) && Sentry.respond_to?(:capture_exception)
578
+ raise
579
+ end
580
+
581
+ # Returns all currently known session keys. This is a COPY of the internal
582
+ # keyset, since otherwise it would be necessary to try and enumerate keys
583
+ # on a Hash which is subject to change at any moment if a user logs in or
584
+ # out. Since Ruby raises an exception should an under-iteration Hash be
585
+ # changed, we can't do that, which is why the keys are returned as a copied
586
+ # array instead.
565
587
  #
566
- # To prevent this, if intending to iterate over the collection, really the
567
- # only safe way is to duplicate it first on "your side" (the client side)
568
- # of the DRb connection. This of course consumes RAM, so you might choose
569
- # to evaluate e.g. the number of keys in the returned session data and only
570
- # permit enumeration if they fall below some previously-measured "allowed"
571
- # value wherein RAM consumption is acceptable.
588
+ # Keys are ordered by least-recently-active first to most-recent last.
572
589
  #
573
- # See HubSsoLib::Core#hubssolib_enumerate_users for an example.
590
+ # Call #retrieve_session_by_key(...) to get session details for that key.
591
+ # Bear in mind that +nil+ returns are possible, since the session data may
592
+ # be changing rapidly and a user might've logged out or had their session
593
+ # expired in the time between you retrieving the list of current keys here,
594
+ # then requesting details of the session for that key later.
574
595
  #
575
- def enumerate_hub_sessions()
576
- @hub_sessions
596
+ # To avoid unbounded RAM requirements arising, the maximum number of keys
597
+ # returned herein is limited to HUB_SESSION_ENUMERATION_KEY_MAX.
598
+ #
599
+ # Returns a Hash with Symbol keys that have values as follows:
600
+ #
601
+ # +count+:: The number of known sessions - just a key count.
602
+ # +keys+:: If +count+ exceeds HUB_SESSION_ENUMERATION_KEY_MAX, this is
603
+ # +nil+, else an array of zero or more session keys.
604
+ #
605
+ def enumerate_hub_session_ids()
606
+ count = @hub_sessions.size
607
+ keys = if count > HUB_SESSION_ENUMERATION_KEY_MAX
608
+ nil
609
+ else
610
+ @hub_sessions.keys # (Hash#keys returns a new array)
611
+ end
612
+
613
+ return { count: count, keys: keys }
577
614
 
578
615
  rescue => e
579
616
  Sentry.capture_exception(e) if defined?(Sentry) && Sentry.respond_to?(:capture_exception)
580
617
  raise
581
618
  end
582
619
 
620
+ # Retrieve session data (as a HubSsoLib::Session instance) based on the
621
+ # given session key. No key rotation occurs. Returns +nil+ if no entry is
622
+ # found for that key.
623
+ #
624
+ def retrieve_session_by_key(key)
625
+ @hub_sessions[key]
626
+ end
627
+
628
+ # WARNING: Comparatively slow.
629
+ #
630
+ # This is usually only called in administrative interfaces to look at the
631
+ # known sessions for a specific user of interest. An Hash of session key
632
+ # values yielding HubSsoLib::Session instances as values is returned.
633
+ #
634
+ # The array is ordered by least-recently-active first to most-recent last.
635
+ #
636
+ # IN THE CURRENT IMPLEMENTATION THIS JUST SEQUENTIALLY SCANS ALL ACTIVE
637
+ # SESSIONS IN THE HASH and must therefore lock on mutex for the duration.
638
+ #
639
+ def retrieve_sessions_by_user_id(user_id)
640
+ HUB_MUTEX.synchronize do
641
+ @hub_sessions.select do | key, session |
642
+ session&.session_user&.user_id == user_id
643
+ end
644
+ end
645
+ end
646
+
583
647
  # Given a session key (which, if a session has been looked up and the key
584
648
  # thus rotated, ought to be that new, rotated key), destroy the associated
585
649
  # session data. Does nothing if the key is not found.
@@ -594,7 +658,7 @@ module HubSsoLib
594
658
  raise
595
659
  end
596
660
 
597
- # WARNING: Comparatively slow
661
+ # WARNING: Comparatively slow.
598
662
  #
599
663
  # This is called in rare cases such as user deletion or being asked for a
600
664
  # session under an old key, indicating loss of key rotation sequence.
@@ -615,7 +679,7 @@ module HubSsoLib
615
679
  raise
616
680
  end
617
681
 
618
- # WARNING: Slow
682
+ # WARNING: Slow.
619
683
  #
620
684
  # This is a housekeeping task which checks sessions against Hub expiry and,
621
685
  # if the session keys look to be substantially older than the value set in
@@ -630,23 +694,21 @@ module HubSsoLib
630
694
  # session data while the method runs will block until method finishes.
631
695
  #
632
696
  def destroy_ancient_sessions
633
- time_limit = HUB_IDLE_TIME_LIMIT * 3 # (TODO: This is fairly arbitrary...)
634
- time_limit = 172_800 if time_limit < 172_800 # (2 days)
635
- destroyed = 0
636
-
637
697
  unless @hub_be_quiet
638
- puts "Session factory: Sweeping sessions inactive for more than #{ time_limit } seconds..."
698
+ puts "Session factory: Sweeping sessions inactive for more than #{ HUB_ARCHIVE_TIME_LIMIT } seconds..."
639
699
  end
640
700
 
701
+ destroyed = 0
702
+
641
703
  HUB_MUTEX.synchronize do
642
- count_before = @hub_sessions.keys.size
704
+ count_before = @hub_sessions.size
643
705
 
644
706
  @hub_sessions.reject! do | key, session |
645
707
  last_used = session&.session_last_used
646
- last_used.nil? || Time.now.utc - last_used > time_limit
708
+ last_used.nil? || Time.now.utc - last_used > HUB_ARCHIVE_TIME_LIMIT
647
709
  end
648
710
 
649
- count_after = @hub_sessions.keys.size
711
+ count_after = @hub_sessions.size
650
712
  destroyed = count_before - count_after
651
713
  end
652
714
 
@@ -680,6 +742,8 @@ module HubSsoLib
680
742
  @hub_sessions.each do | key, session |
681
743
  next if session&.session_user&.user_id.nil? # NOTE EARLY LOOP RESTART
682
744
 
745
+ session.session_flash = nil
746
+
683
747
  dump = ::YAML.dump({key => session})
684
748
  dump.sub!(/^---\n/, '') # (avoid multiple document markers)
685
749
 
@@ -971,6 +1035,80 @@ module HubSsoLib
971
1035
  user ? "#{user.user_real_name} (#{user.user_id})" : 'Anonymous'
972
1036
  end
973
1037
 
1038
+ # WARNING: Slow.
1039
+ #
1040
+ # Return an Array of HubSsoLib::User objects for all logged-in users, in an
1041
+ # array. If a user is logged into more than one browser and thus has more
1042
+ # than one session active, they will accordingly appear more than once in
1043
+ # the returned data. This can also happen if a user loses key rotation and
1044
+ # leaves an old, but not yet expired session behind when they log in anew.
1045
+ #
1046
+ # In accordance with HubSsoLib::SessionFactory#enumerate_hub_session_ids
1047
+ # documentation, a maximum of HUB_SESSION_ENUMERATION_KEY_MAX users can be
1048
+ # returned here. If this is exceeded, an *empty* array is returned. If you
1049
+ # are processing this information for display in a UI which itself requires
1050
+ # a logged in user of some sort (which is very likely) and therefore know
1051
+ # that at least *one* session does exist, you can treat an empty array as
1052
+ # confirmation of "lots of sessions". If you can't be sure of at least one
1053
+ # logged in user, then there is obvious arising ambiguity and this method
1054
+ # does not solve it for you - it's just "zero, or very many users".
1055
+ #
1056
+ # Users are ordered by least-recently-active first, most-recent last.
1057
+ #
1058
+ # For information about performance limitations, see
1059
+ # HubSsoLib::SessionFactory#enumerate_hub_session_ids.
1060
+ #
1061
+ def hubssolib_enumerate_users
1062
+ hub_session_info = hubssolib_factory().enumerate_hub_session_ids()
1063
+ hub_users = []
1064
+
1065
+ unless hub_session_info[:keys].nil? # (keyset too large, enumeration prohibited)
1066
+ hub_session_info[:keys].each do | key |
1067
+ session = hubssolib_factory().retrieve_session_by_key(key)
1068
+ hub_users << session.session_user unless session&.session_user&.user_id.nil?
1069
+ end
1070
+ end
1071
+
1072
+ return hub_users
1073
+ end
1074
+
1075
+ # WARNING: Comparatively slow.
1076
+ #
1077
+ # For a given HubSsoLib::User's user ID, return any known sessions held by
1078
+ # the DRb server, as an Hash of session keys with HubSsoLib::Session
1079
+ # instances as values.
1080
+ #
1081
+ # Returns an empty Hash if the given ID is +nil+.
1082
+ #
1083
+ # Note that Hub sessions can disappear at any moment, so the session keys
1084
+ # you find in the Hash might refer to extinct sessions by the time you get
1085
+ # to do something with them. You can still access the data, but if you were
1086
+ # to try and ask the DRb server for that key, it'd return +nil+.
1087
+ #
1088
+ # Sessions are ordered by least-recently-active first, most-recent last.
1089
+ #
1090
+ # For information about performance limitations, see
1091
+ # HubSsoLib::SessionFactory#retrieve_sessions_by_user_id.
1092
+ #
1093
+ def hubssolib_retrieve_user_sessions(hub_user_id)
1094
+ if hub_user_id.nil?
1095
+ {}
1096
+ else
1097
+ hubssolib_factory().retrieve_sessions_by_user_id(hub_user_id)
1098
+ end
1099
+ end
1100
+
1101
+ # WARNING: Comparatively slow.
1102
+ #
1103
+ # Remove all sessions under a given ID.
1104
+ #
1105
+ # For information about performance limitations, see
1106
+ # HubSsoLib::SessionFactory#destroy_sessions_by_user_id.
1107
+ #
1108
+ def hubssolib_destroy_user_sessions(hub_user_id)
1109
+ hubssolib_factory().destroy_sessions_by_user_id(hub_user_id) unless hub_user_id.nil?
1110
+ end
1111
+
974
1112
  # If an application needs to know about changes of a user e-mail address
975
1113
  # or display name (e.g. because of sync to a local relational store of
976
1114
  # users related to other application-managed resources, with therefore a
@@ -1138,6 +1276,19 @@ module HubSsoLib
1138
1276
  return true # true -> let action processing continue
1139
1277
 
1140
1278
  end
1279
+
1280
+ rescue Exception => e
1281
+ Sentry.capture_exception(e) if defined?(Sentry) && Sentry.respond_to?(:capture_exception)
1282
+
1283
+ # At this point there tends to be no Session data, so we're going to have
1284
+ # to encode the exception data into the URI... It must be escaped twice,
1285
+ # as many servers treat "%2F" in a URI as a "/". Apache can then fail to
1286
+ # serve the page, raising a 404 error unless "AllowEncodedSlashes on" is
1287
+ # specified in its configuration.
1288
+ #
1289
+ suffix = '/' + CGI::escape(CGI::escape(hubssolib_set_exception_data(e)))
1290
+ new_path = HUB_PATH_PREFIX + '/tasks/service'
1291
+ redirect_to(new_path + suffix) unless request.path.include?(new_path)
1141
1292
  end
1142
1293
 
1143
1294
  # Mandatory controller "after_action" callback method to tidy up after Hub
@@ -1158,18 +1309,17 @@ module HubSsoLib
1158
1309
  # We can return to this location by calling #redirect_back_or_default.
1159
1310
  #
1160
1311
  def hubssolib_store_location(uri_str = request.url)
1161
-
1162
1312
  if (uri_str && !uri_str.empty?)
1163
1313
  uri_str = hubssolib_promote_uri_to_ssl(uri_str, request.host) unless request.ssl?
1164
1314
  hubssolib_set_return_to(uri_str)
1165
1315
  else
1166
1316
  hubssolib_set_return_to(nil)
1167
1317
  end
1168
-
1169
1318
  end
1170
1319
 
1171
1320
  # Redirect to the URI stored by the most recent store_location call or
1172
1321
  # to the passed default.
1322
+ #
1173
1323
  def hubssolib_redirect_back_or_default(default)
1174
1324
  url = hubssolib_get_return_to()
1175
1325
  hubssolib_set_return_to(nil)
@@ -1181,7 +1331,7 @@ module HubSsoLib
1181
1331
  # sets the host you provide (or leaves it alone if you omit the
1182
1332
  # parameter), then forces the scheme to 'https'. Returns the result
1183
1333
  # as a flat string.
1184
-
1334
+ #
1185
1335
  def hubssolib_promote_uri_to_ssl(uri_str, host = nil)
1186
1336
  uri = URI.parse(uri_str)
1187
1337
  uri.host = host if host
@@ -1337,7 +1487,7 @@ module HubSsoLib
1337
1487
  self.hubssolib_destroy_session! unless old_session.nil?
1338
1488
 
1339
1489
  start_key = SecureRandom.uuid
1340
- @hubssolib_session = hubssolib_factory().get_hub_session_proxy(start_key, request.remote_ip, create: true)
1490
+ @hubssolib_session = hubssolib_factory().get_hub_session_proxy(start_key, create: true)
1341
1491
 
1342
1492
  # The session is now stored under the rotated key, so put that into the
1343
1493
  # Hub session cookie so that on the next request, we can retrieve the
@@ -1374,7 +1524,7 @@ module HubSsoLib
1374
1524
  if key.nil? || key.size != 36
1375
1525
  @hubssolib_session = nil
1376
1526
  else
1377
- hub_session = hubssolib_factory().get_hub_session_proxy(key, request.remote_ip, create: false)
1527
+ hub_session = hubssolib_factory().get_hub_session_proxy(key, create: false)
1378
1528
 
1379
1529
  if hub_session.nil? # Invalid key in cookie
1380
1530
  @hubssolib_session = nil
@@ -1434,13 +1584,14 @@ module HubSsoLib
1434
1584
  # halted (since the overall return value is therefore 'false').
1435
1585
  #
1436
1586
  def hubssolib_must_login
1587
+
1437
1588
  # If HTTP, redirect to the same place, but HTTPS. Then we can store the
1438
1589
  # flash and return-to in the session data. We'll have the same set of
1439
1590
  # before-filter operations running and they'll find out we're either
1440
1591
  # authorised after all, or come back to this very function, which will
1441
1592
  # now be happily running from an HTTPS connection and will go on to set
1442
1593
  # the flash and redirect to the login page.
1443
-
1594
+ #
1444
1595
  if hubssolib_ensure_https
1445
1596
  hubssolib_set_flash(:alert, 'You must log in before you can continue.')
1446
1597
  redirect_to HUB_PATH_PREFIX + '/account/login'
@@ -1502,47 +1653,6 @@ module HubSsoLib
1502
1653
  )
1503
1654
  end
1504
1655
 
1505
- # Return an array of Hub User objects representing users based on a list of
1506
- # known sessions returned by the DRb server. Note that if an application
1507
- # exposes this method to a view, it is up to the application to ensure
1508
- # sufficient access permission protection for that view according to the
1509
- # webmaster's choice of site security level. Generally, normal users should
1510
- # not be allowed access!
1511
- #
1512
- # Due to the session pool being held in the DRb server and subject to
1513
- # alteration at any time by other requests (assuming the server supports
1514
- # more than just one request at a time, that is!) then the session data
1515
- # must be copied locally before iterating. Otherwise, exceptions arise from
1516
- # attempts to alter an under-iteration Hash. This in turn raises a worry
1517
- # about RAM usage. For that reason, a (somewhat arbitrary) limit of
1518
- # 2000 active users is applied. More than that and the method returns an
1519
- # empty array.
1520
- #
1521
- def hubssolib_enumerate_users
1522
- hub_session_data = hubssolib_factory().enumerate_hub_sessions()
1523
- return [] if hub_session_data.keys.size > 2000 # NOTE EARLY EXIT
1524
-
1525
- sessions = hub_session_data.values.dup
1526
- users = sessions.inject( [] ) do | memo, session |
1527
- user = session.session_user
1528
- memo << user unless user&.user_id.nil?
1529
- memo
1530
- end
1531
-
1532
- return users
1533
-
1534
- rescue Exception => e
1535
-
1536
- # At this point there tends to be no Session data, so we're
1537
- # going to have to encode the exception data into the URI...
1538
- # See earlier for double-escaping rationale.
1539
-
1540
- suffix = '/' + CGI::escape(CGI::escape(hubssolib_set_exception_data(e)))
1541
- new_path = HUB_PATH_PREFIX + '/tasks/service'
1542
- redirect_to new_path + suffix unless request.path.include?(new_path)
1543
- return nil
1544
- end
1545
-
1546
1656
  # Encode exception data into a string suitable for using in a URL
1547
1657
  # if CGI escaped first. Pass the exception object; stores only the
1548
1658
  # 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.5.0
4
+ version: 3.6.1
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-25 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