hubssolib 3.5.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 +4 -4
- data/CHANGELOG.md +6 -1
- data/Gemfile.lock +2 -2
- data/hubssolib.gemspec +1 -1
- data/lib/hub_sso_lib.rb +205 -99
- 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: faacd8f740f867ee072f0f8e0f15324109941026b836cb3fcc8618f61ee68e02
|
4
|
+
data.tar.gz: 6ef08dccd8bbe03a974c238353a27d7f249a40911a40d65dfcd481dd6fe7b48e
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: 67b9f58743d8f9ed2983d3b9b970fd00d50ba335dcd761f8c9626866c9c91a1e65c378bb0139835c1e8eb48557e681ea510f6391373d06e1a6806aaf8b8fc2e2
|
7
|
+
data.tar.gz: d4e6efe525fb55b25f40c888ab4d05a7041c90c1cddcba7bab74002d299b2a1f96e35f58ea211cb666da68fda3d129400a7bf874bc8b7ddf09e9fdd6a8d1fbd2
|
data/CHANGELOG.md
CHANGED
@@ -1,3 +1,9 @@
|
|
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
|
1
7
|
## 3.5.0, 25-Mar-2025
|
2
8
|
|
3
9
|
Builds on the cleaner session interface with some changes and improvements:
|
@@ -29,7 +35,6 @@ In addition:
|
|
29
35
|
|
30
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.
|
31
37
|
|
32
|
-
|
33
38
|
## 3.3.0, 16-Feb-2025
|
34
39
|
|
35
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
|
+
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.
|
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/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)
|
@@ -470,7 +483,7 @@ module HubSsoLib
|
|
470
483
|
|
471
484
|
@hub_sessions = restored_sessions || {}
|
472
485
|
self.destroy_ancient_sessions()
|
473
|
-
puts "Session factory: Reloaded #{@hub_sessions.
|
486
|
+
puts "Session factory: Reloaded #{@hub_sessions.size} from archive" unless @hub_be_quiet
|
474
487
|
|
475
488
|
rescue => e
|
476
489
|
puts "Session factory: Ignored archive due to error #{e.message.inspect}" unless @hub_be_quiet
|
@@ -499,15 +512,20 @@ module HubSsoLib
|
|
499
512
|
#
|
500
513
|
# The returned object is proxied via DRb - it is shared between processes.
|
501
514
|
#
|
502
|
-
# +key+::
|
503
|
-
#
|
504
|
-
#
|
505
|
-
#
|
506
|
-
#
|
507
|
-
#
|
508
|
-
#
|
509
|
-
#
|
510
|
-
#
|
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!
|
511
529
|
#
|
512
530
|
# In addition, the following optional named parameters can be given:
|
513
531
|
#
|
@@ -515,7 +533,7 @@ module HubSsoLib
|
|
515
533
|
# new session under that key. If +false+, attempts to read with
|
516
534
|
# an unrecognised key yield +nil+.
|
517
535
|
#
|
518
|
-
def get_hub_session_proxy(key,
|
536
|
+
def get_hub_session_proxy(key, _ignored = nil, create: true)
|
519
537
|
hub_session = HUB_MUTEX.synchronize { @hub_sessions[key] }
|
520
538
|
return nil if create == false && hub_session.nil? # NOTE EARLY EXIT
|
521
539
|
|
@@ -526,18 +544,7 @@ module HubSsoLib
|
|
526
544
|
puts "Session factory: #{ message } session for key #{ key } and rotating to #{ new_key }"
|
527
545
|
end
|
528
546
|
|
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
|
547
|
+
hub_session = HubSsoLib::Session.new if hub_session.nil?
|
541
548
|
|
542
549
|
HUB_MUTEX.synchronize do
|
543
550
|
@hub_sessions.delete(key)
|
@@ -552,34 +559,90 @@ module HubSsoLib
|
|
552
559
|
raise
|
553
560
|
end
|
554
561
|
|
562
|
+
# THIS INTERFACE IS DEPRECATED and will be removed in Hub 4. Change to
|
563
|
+
# #enumerate_hub_session_keys instead.
|
564
|
+
#
|
555
565
|
# Enumerate all currently known sessions. The format is a Hash, with the
|
556
566
|
# session key UUIDs as keys and the related HubSsoLib::Session instances as
|
557
|
-
# 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.
|
558
571
|
#
|
559
|
-
|
560
|
-
|
561
|
-
|
562
|
-
|
563
|
-
|
564
|
-
|
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.
|
565
586
|
#
|
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.
|
587
|
+
# Keys are ordered by least-recently-active first to most-recent last.
|
572
588
|
#
|
573
|
-
#
|
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.
|
574
594
|
#
|
575
|
-
|
576
|
-
|
595
|
+
# To avoid unbounded RAM requirements arising, the maximum number of keys
|
596
|
+
# returned herein is limited to HUB_SESSION_ENUMERATION_KEY_MAX.
|
597
|
+
#
|
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 }
|
577
613
|
|
578
614
|
rescue => e
|
579
615
|
Sentry.capture_exception(e) if defined?(Sentry) && Sentry.respond_to?(:capture_exception)
|
580
616
|
raise
|
581
617
|
end
|
582
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
|
+
|
583
646
|
# Given a session key (which, if a session has been looked up and the key
|
584
647
|
# thus rotated, ought to be that new, rotated key), destroy the associated
|
585
648
|
# session data. Does nothing if the key is not found.
|
@@ -594,7 +657,7 @@ module HubSsoLib
|
|
594
657
|
raise
|
595
658
|
end
|
596
659
|
|
597
|
-
# WARNING: Comparatively slow
|
660
|
+
# WARNING: Comparatively slow.
|
598
661
|
#
|
599
662
|
# This is called in rare cases such as user deletion or being asked for a
|
600
663
|
# session under an old key, indicating loss of key rotation sequence.
|
@@ -615,7 +678,7 @@ module HubSsoLib
|
|
615
678
|
raise
|
616
679
|
end
|
617
680
|
|
618
|
-
# WARNING: Slow
|
681
|
+
# WARNING: Slow.
|
619
682
|
#
|
620
683
|
# This is a housekeeping task which checks sessions against Hub expiry and,
|
621
684
|
# if the session keys look to be substantially older than the value set in
|
@@ -630,23 +693,21 @@ module HubSsoLib
|
|
630
693
|
# session data while the method runs will block until method finishes.
|
631
694
|
#
|
632
695
|
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
696
|
unless @hub_be_quiet
|
638
|
-
puts "Session factory: Sweeping sessions inactive for more than #{
|
697
|
+
puts "Session factory: Sweeping sessions inactive for more than #{ HUB_ARCHIVE_TIME_LIMIT } seconds..."
|
639
698
|
end
|
640
699
|
|
700
|
+
destroyed = 0
|
701
|
+
|
641
702
|
HUB_MUTEX.synchronize do
|
642
|
-
count_before = @hub_sessions.
|
703
|
+
count_before = @hub_sessions.size
|
643
704
|
|
644
705
|
@hub_sessions.reject! do | key, session |
|
645
706
|
last_used = session&.session_last_used
|
646
|
-
last_used.nil? || Time.now.utc - last_used >
|
707
|
+
last_used.nil? || Time.now.utc - last_used > HUB_ARCHIVE_TIME_LIMIT
|
647
708
|
end
|
648
709
|
|
649
|
-
count_after = @hub_sessions.
|
710
|
+
count_after = @hub_sessions.size
|
650
711
|
destroyed = count_before - count_after
|
651
712
|
end
|
652
713
|
|
@@ -802,6 +863,18 @@ module HubSsoLib
|
|
802
863
|
user = hub_session&.session_user
|
803
864
|
|
804
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)
|
805
878
|
end
|
806
879
|
|
807
880
|
# Sets the currently signed in user. Note that although this works and is
|
@@ -971,6 +1044,80 @@ module HubSsoLib
|
|
971
1044
|
user ? "#{user.user_real_name} (#{user.user_id})" : 'Anonymous'
|
972
1045
|
end
|
973
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
|
+
|
974
1121
|
# If an application needs to know about changes of a user e-mail address
|
975
1122
|
# or display name (e.g. because of sync to a local relational store of
|
976
1123
|
# users related to other application-managed resources, with therefore a
|
@@ -1158,18 +1305,17 @@ module HubSsoLib
|
|
1158
1305
|
# We can return to this location by calling #redirect_back_or_default.
|
1159
1306
|
#
|
1160
1307
|
def hubssolib_store_location(uri_str = request.url)
|
1161
|
-
|
1162
1308
|
if (uri_str && !uri_str.empty?)
|
1163
1309
|
uri_str = hubssolib_promote_uri_to_ssl(uri_str, request.host) unless request.ssl?
|
1164
1310
|
hubssolib_set_return_to(uri_str)
|
1165
1311
|
else
|
1166
1312
|
hubssolib_set_return_to(nil)
|
1167
1313
|
end
|
1168
|
-
|
1169
1314
|
end
|
1170
1315
|
|
1171
1316
|
# Redirect to the URI stored by the most recent store_location call or
|
1172
1317
|
# to the passed default.
|
1318
|
+
#
|
1173
1319
|
def hubssolib_redirect_back_or_default(default)
|
1174
1320
|
url = hubssolib_get_return_to()
|
1175
1321
|
hubssolib_set_return_to(nil)
|
@@ -1181,7 +1327,7 @@ module HubSsoLib
|
|
1181
1327
|
# sets the host you provide (or leaves it alone if you omit the
|
1182
1328
|
# parameter), then forces the scheme to 'https'. Returns the result
|
1183
1329
|
# as a flat string.
|
1184
|
-
|
1330
|
+
#
|
1185
1331
|
def hubssolib_promote_uri_to_ssl(uri_str, host = nil)
|
1186
1332
|
uri = URI.parse(uri_str)
|
1187
1333
|
uri.host = host if host
|
@@ -1337,7 +1483,7 @@ module HubSsoLib
|
|
1337
1483
|
self.hubssolib_destroy_session! unless old_session.nil?
|
1338
1484
|
|
1339
1485
|
start_key = SecureRandom.uuid
|
1340
|
-
@hubssolib_session = hubssolib_factory().get_hub_session_proxy(start_key,
|
1486
|
+
@hubssolib_session = hubssolib_factory().get_hub_session_proxy(start_key, create: true)
|
1341
1487
|
|
1342
1488
|
# The session is now stored under the rotated key, so put that into the
|
1343
1489
|
# Hub session cookie so that on the next request, we can retrieve the
|
@@ -1374,7 +1520,7 @@ module HubSsoLib
|
|
1374
1520
|
if key.nil? || key.size != 36
|
1375
1521
|
@hubssolib_session = nil
|
1376
1522
|
else
|
1377
|
-
hub_session = hubssolib_factory().get_hub_session_proxy(key,
|
1523
|
+
hub_session = hubssolib_factory().get_hub_session_proxy(key, create: false)
|
1378
1524
|
|
1379
1525
|
if hub_session.nil? # Invalid key in cookie
|
1380
1526
|
@hubssolib_session = nil
|
@@ -1434,13 +1580,14 @@ module HubSsoLib
|
|
1434
1580
|
# halted (since the overall return value is therefore 'false').
|
1435
1581
|
#
|
1436
1582
|
def hubssolib_must_login
|
1583
|
+
|
1437
1584
|
# If HTTP, redirect to the same place, but HTTPS. Then we can store the
|
1438
1585
|
# flash and return-to in the session data. We'll have the same set of
|
1439
1586
|
# before-filter operations running and they'll find out we're either
|
1440
1587
|
# authorised after all, or come back to this very function, which will
|
1441
1588
|
# now be happily running from an HTTPS connection and will go on to set
|
1442
1589
|
# the flash and redirect to the login page.
|
1443
|
-
|
1590
|
+
#
|
1444
1591
|
if hubssolib_ensure_https
|
1445
1592
|
hubssolib_set_flash(:alert, 'You must log in before you can continue.')
|
1446
1593
|
redirect_to HUB_PATH_PREFIX + '/account/login'
|
@@ -1502,47 +1649,6 @@ module HubSsoLib
|
|
1502
1649
|
)
|
1503
1650
|
end
|
1504
1651
|
|
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
1652
|
# Encode exception data into a string suitable for using in a URL
|
1547
1653
|
# if CGI escaped first. Pass the exception object; stores only the
|
1548
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
|
+
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-
|
10
|
+
date: 2025-03-26 00:00:00.000000000 Z
|
11
11
|
dependencies:
|
12
12
|
- !ruby/object:Gem::Dependency
|
13
13
|
name: drb
|