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 +4 -4
- data/CHANGELOG.md +19 -1
- data/Gemfile.lock +2 -2
- data/README.md +2 -0
- data/hubssolib.gemspec +1 -1
- data/lib/hub_sso_lib.rb +330 -116
- 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,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
|
+
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/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
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
|
@@ -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:
|
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+::
|
460
|
-
#
|
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
|
-
#
|
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
|
-
|
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
|
-
|
470
|
-
|
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
|
-
|
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
|
-
#
|
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
|
-
#
|
514
|
-
#
|
515
|
-
#
|
516
|
-
#
|
517
|
-
#
|
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
|
-
#
|
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
|
-
|
523
|
-
|
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 #{
|
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.
|
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 >
|
707
|
+
last_used.nil? || Time.now.utc - last_used > HUB_ARCHIVE_TIME_LIMIT
|
594
708
|
end
|
595
709
|
|
596
|
-
count_after = @hub_sessions.
|
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
|
-
|
778
|
+
::HubSsoLib::Server::Runner.run
|
779
|
+
end
|
630
780
|
|
631
|
-
|
632
|
-
|
633
|
-
DRb.thread.join
|
781
|
+
class Runner
|
782
|
+
QUEUE = ::Queue.new
|
634
783
|
|
635
|
-
|
636
|
-
|
637
|
-
|
638
|
-
|
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,
|
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
|
-
|
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,
|
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
|
+
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
|