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 +4 -4
- data/CHANGELOG.md +18 -1
- data/Gemfile.lock +3 -3
- data/hubssolib.gemspec +1 -1
- data/lib/hub_sso_lib.rb +210 -100
- metadata +2 -2
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: eaea423d2eec6433b30522f6a6d46ab3c1b2868bb860d985cd7472116a17bc95
|
4
|
+
data.tar.gz: 19ae77e97c1e480b6b6cba78eee22523d9964af677ac9a9531582aa68883350b
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
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
|
+
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.
|
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.
|
55
|
+
stringio (3.1.6)
|
56
56
|
|
57
57
|
PLATFORMS
|
58
58
|
ruby
|
data/hubssolib.gemspec
CHANGED
data/lib/hub_sso_lib.rb
CHANGED
@@ -1,6 +1,6 @@
|
|
1
1
|
#######################################################################
|
2
2
|
# Module: HubSsoLib #
|
3
|
-
# (C) Hipposoft 2006-
|
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.
|
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+::
|
503
|
-
#
|
504
|
-
#
|
505
|
-
#
|
506
|
-
#
|
507
|
-
#
|
508
|
-
#
|
509
|
-
#
|
510
|
-
#
|
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,
|
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
|
-
|
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
|
-
|
560
|
-
|
561
|
-
|
562
|
-
|
563
|
-
|
564
|
-
|
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
|
-
#
|
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
|
-
#
|
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
|
-
|
576
|
-
|
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 #{
|
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.
|
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 >
|
708
|
+
last_used.nil? || Time.now.utc - last_used > HUB_ARCHIVE_TIME_LIMIT
|
647
709
|
end
|
648
710
|
|
649
|
-
count_after = @hub_sessions.
|
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,
|
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,
|
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.
|
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-
|
10
|
+
date: 2025-03-26 00:00:00.000000000 Z
|
11
11
|
dependencies:
|
12
12
|
- !ruby/object:Gem::Dependency
|
13
13
|
name: drb
|