hubssolib 3.3.0 → 3.4.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 -0
- data/Gemfile.lock +3 -3
- data/README.md +8 -0
- data/hubssolib.gemspec +1 -1
- data/lib/hub_sso_lib.rb +357 -162
- 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: c1a83a70b87a6c3cabd8bbb821dfc06ee467d382c9593cb7c1733152e136f5a1
|
4
|
+
data.tar.gz: 87f1dda92220f25326c8bbce597ecd97e74c3dd7e22fd0fbf89bfed6269b6b23
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: cf2ae3afb91a4fcc3391cf4b156e3dd9e4d0be4cecafc10204cf6dc8a7d6b9d07d51dc08cd7fd22ee12a478025d03730227fa2602a194d4b0e6785d72af21295
|
7
|
+
data.tar.gz: 6ac8d8a0a1920d7f93604204ef0c254a094bd4a8559110bba2f630c1fe406f87966b0ba2289d017a9a48e90cd0d6e2eab0e1ae3872032c7bad389942965fe2b2
|
data/CHANGELOG.md
CHANGED
@@ -1,3 +1,22 @@
|
|
1
|
+
## 3.4.0, 24-Mar-2025
|
2
|
+
|
3
|
+
The session interface has now been cleaned up. While the public API is unchanged, you:
|
4
|
+
|
5
|
+
* ...are encouraged to change uses of `hubssolib_current_user = foo` over to `hubssolib_log_in()`
|
6
|
+
* ...are likewise encouraged to change `hubssolib_current_user = nil` over to `hubssolib_log_out()`
|
7
|
+
|
8
|
+
There are many fixes in the overhauled session management:
|
9
|
+
|
10
|
+
* Session data is only stored when a user logs in, else no session cookie is built
|
11
|
+
* If a user logs in, possible stale older sessions they might have are swept out
|
12
|
+
* When the user logs out their session data is entirely deleted
|
13
|
+
* The Hub app includes a Rake task that makes use of a new internal session factory feature to clean up sessions with a very old idle timeout (more than 3 times the usual in-app 'you were logged out' notification mechanism's delay via `HUB_IDLE_TIME_LIMIT`, or 2 days, whichever is larger) and delete such sessions - yes, this means that if a user did then come in even later on a now-deleted session ID, they'd simply be logged out without the warning message about why, but it was a serious oversight prior to allow the sessions to just accumulate.
|
14
|
+
|
15
|
+
In addition:
|
16
|
+
|
17
|
+
* 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
|
+
|
19
|
+
|
1
20
|
## 3.3.0, 16-Feb-2025
|
2
21
|
|
3
22
|
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.4.0)
|
5
5
|
base64 (~> 0.2)
|
6
6
|
drb (~> 2.2)
|
7
7
|
|
@@ -29,7 +29,7 @@ GEM
|
|
29
29
|
psych (5.2.3)
|
30
30
|
date
|
31
31
|
stringio
|
32
|
-
rdoc (6.
|
32
|
+
rdoc (6.13.0)
|
33
33
|
psych (>= 4.0.0)
|
34
34
|
reline (0.6.0)
|
35
35
|
io-console (~> 0.5)
|
@@ -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.5)
|
56
56
|
|
57
57
|
PLATFORMS
|
58
58
|
ruby
|
data/README.md
CHANGED
@@ -202,6 +202,14 @@ The four arguments are guaranteed to be present and non-empty, with leading or t
|
|
202
202
|
|
203
203
|
|
204
204
|
|
205
|
+
## Session maintenance
|
206
|
+
|
207
|
+
User-idle session expiry is routinely handled in the "beforehand" callback so that we can see the session is valid but expired, expire it and add a via-flash warning about what happened so the user is fully informed. The Hub gem tries to be nice about `POST`, `PATCH` and `PUT` operations too and allows those to complete before the expected subsequent redirection `GET` causing expiry, to try and avoid users losing form submission data entirely.
|
208
|
+
|
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
|
+
|
211
|
+
|
212
|
+
|
205
213
|
## Hub library API
|
206
214
|
|
207
215
|
The Hub component interfaces that should be used by application authors when integrating with the Hub single sign-on mechanism are described below. If you want a complete list of all public interfaces, consult the file `hub_sso_lib.rb` inside the Hub gem. All functions and classes therein are fully commented to describe the purpose of each class, along with the purpose, input parameters and return values of class methods and instance methods.
|
data/hubssolib.gemspec
CHANGED
data/lib/hub_sso_lib.rb
CHANGED
@@ -51,9 +51,9 @@ module HubSsoLib
|
|
51
51
|
#
|
52
52
|
HUB_PATH_PREFIX = ENV['HUB_PATH_PREFIX'] || ''
|
53
53
|
|
54
|
-
# Time limit, *in seconds*, for the account inactivity timeout.
|
55
|
-
#
|
56
|
-
#
|
54
|
+
# Time limit, *in seconds*, for the account inactivity timeout. If a user
|
55
|
+
# performs no Hub actions during this time they will be automatically logged
|
56
|
+
# out upon their next action, with a Flash message set to explain why.
|
57
57
|
#
|
58
58
|
HUB_IDLE_TIME_LIMIT = ENV['HUB_IDLE_TIME_LIMIT']&.to_i || 4 * 60 * 60
|
59
59
|
|
@@ -402,16 +402,16 @@ module HubSsoLib
|
|
402
402
|
attr_accessor :session_return_to
|
403
403
|
attr_accessor :session_flash
|
404
404
|
attr_accessor :session_user
|
405
|
-
attr_accessor :
|
405
|
+
attr_accessor :session_rotated_key
|
406
406
|
attr_accessor :session_ip
|
407
407
|
|
408
408
|
def initialize
|
409
|
-
@session_last_used
|
410
|
-
@session_return_to
|
411
|
-
@session_flash
|
412
|
-
@session_user
|
413
|
-
@
|
414
|
-
@session_ip
|
409
|
+
@session_last_used = Time.now.utc
|
410
|
+
@session_return_to = nil
|
411
|
+
@session_flash = {}
|
412
|
+
@session_user = HubSsoLib::User.new
|
413
|
+
@session_rotated_key = nil
|
414
|
+
@session_ip = nil
|
415
415
|
|
416
416
|
rescue => e
|
417
417
|
Sentry.capture_exception(e) if defined?(Sentry) && Sentry.respond_to?(:capture_exception)
|
@@ -423,8 +423,10 @@ module HubSsoLib
|
|
423
423
|
# Class: SessionFactory #
|
424
424
|
# (C) Hipposoft 2006 #
|
425
425
|
# #
|
426
|
-
# Purpose:
|
427
|
-
#
|
426
|
+
# Purpose: Manage Session objects for DRb server clients. This class #
|
427
|
+
# implements the API exposed by the HubSsoLib::Server DRb #
|
428
|
+
# endpoint, so this is the remote object that clients will #
|
429
|
+
# be calling into. #
|
428
430
|
# #
|
429
431
|
# Author: A.D.Hodgkinson #
|
430
432
|
# #
|
@@ -448,7 +450,7 @@ module HubSsoLib
|
|
448
450
|
# recorded in existing session data.
|
449
451
|
#
|
450
452
|
# Whether new or pre-existing, the returned session will have changed key
|
451
|
-
# as a result of being read; check the #
|
453
|
+
# as a result of being read; check the #session_rotated_key property to
|
452
454
|
# find out the new key. If you fail to do this, you'll lose access to the
|
453
455
|
# session data as you won't know which key it lies under.
|
454
456
|
#
|
@@ -463,17 +465,17 @@ module HubSsoLib
|
|
463
465
|
# invalid and discarded.
|
464
466
|
#
|
465
467
|
def get_hub_session_proxy(key, remote_ip)
|
466
|
-
hub_session = @hub_sessions[key]
|
468
|
+
hub_session = HUB_MUTEX.synchronize { @hub_sessions[key] }
|
467
469
|
message = hub_session.nil? ? 'Created' : 'Retrieving'
|
468
470
|
new_key = SecureRandom.uuid
|
469
471
|
|
470
472
|
unless @hub_be_quiet
|
471
|
-
puts "#{ message } session for key #{ key } and rotating to #{ new_key }"
|
473
|
+
puts "Session factory: #{ message } session for key #{ key } and rotating to #{ new_key }"
|
472
474
|
end
|
473
475
|
|
474
476
|
unless hub_session.nil? || hub_session.session_ip == remote_ip
|
475
477
|
unless @hub_be_quiet
|
476
|
-
puts "WARNING: IP address changed from #{ hub_session.session_ip } to #{ remote_ip } -> discarding session"
|
478
|
+
puts "Session factory: WARNING: IP address changed from #{ hub_session.session_ip } to #{ remote_ip } -> discarding session"
|
477
479
|
end
|
478
480
|
|
479
481
|
hub_session = nil
|
@@ -484,10 +486,12 @@ module HubSsoLib
|
|
484
486
|
hub_session.session_ip = remote_ip
|
485
487
|
end
|
486
488
|
|
487
|
-
|
488
|
-
|
489
|
+
HUB_MUTEX.synchronize do
|
490
|
+
@hub_sessions.delete(key)
|
491
|
+
@hub_sessions[new_key] = hub_session
|
492
|
+
end
|
489
493
|
|
490
|
-
hub_session.
|
494
|
+
hub_session.session_rotated_key = new_key
|
491
495
|
return hub_session
|
492
496
|
|
493
497
|
rescue => e
|
@@ -495,6 +499,26 @@ module HubSsoLib
|
|
495
499
|
raise
|
496
500
|
end
|
497
501
|
|
502
|
+
# Enumerate all currently known sessions. The format is a Hash, with the
|
503
|
+
# session key UUIDs as keys and the related HubSsoLib::Session instances as
|
504
|
+
# values.
|
505
|
+
#
|
506
|
+
# WARNING: This returns all Hub sessions maintained by the server instance
|
507
|
+
# but it's returning a reference to the live object with everything being
|
508
|
+
# managed over DRb. If a caller iterates over this object, then it can fall
|
509
|
+
# foul of other request threads making changes to the underlying data and
|
510
|
+
# Ruby raising exceptions about e.g. hashes being modified during
|
511
|
+
# enumeration.
|
512
|
+
#
|
513
|
+
# To prevent this, if intending to iterate over the collection, really the
|
514
|
+
# only safe way is to duplicate it first on "your side" (the client side)
|
515
|
+
# of the DRb connection. This of course consumes RAM, so you might choose
|
516
|
+
# to evaluate e.g. the number of keys in the returned session data and only
|
517
|
+
# permit enumeration if they fall below some previously-measured "allowed"
|
518
|
+
# value wherein RAM consumption is acceptable.
|
519
|
+
#
|
520
|
+
# See HubSsoLib::Core#hubssolib_enumerate_users for an example.
|
521
|
+
#
|
498
522
|
def enumerate_hub_sessions()
|
499
523
|
@hub_sessions
|
500
524
|
|
@@ -502,6 +526,86 @@ module HubSsoLib
|
|
502
526
|
Sentry.capture_exception(e) if defined?(Sentry) && Sentry.respond_to?(:capture_exception)
|
503
527
|
raise
|
504
528
|
end
|
529
|
+
|
530
|
+
# Given a session key (which, if a session has been looked up and the key
|
531
|
+
# thus rotated, ought to be that new, rotated key), destroy the associated
|
532
|
+
# session data. Does nothing if the key is not found.
|
533
|
+
#
|
534
|
+
def destroy_session_by_key(key)
|
535
|
+
HUB_MUTEX.synchronize do
|
536
|
+
@hub_sessions.delete(key)
|
537
|
+
end
|
538
|
+
|
539
|
+
rescue => e
|
540
|
+
Sentry.capture_exception(e) if defined?(Sentry) && Sentry.respond_to?(:capture_exception)
|
541
|
+
raise
|
542
|
+
end
|
543
|
+
|
544
|
+
# WARNING: Comparatively slow
|
545
|
+
#
|
546
|
+
# This is called in rare cases such as user deletion or being asked for a
|
547
|
+
# session under an old key, indicating loss of key rotation sequence.
|
548
|
+
# Removes all sessions found for a given user ID.
|
549
|
+
#
|
550
|
+
# IN THE CURRENT IMPLEMENTATION THIS JUST SEQUENTIALLY SCANS ALL ACTIVE
|
551
|
+
# SESSIONS IN THE HASH and must therefore lock on mutex for the duration.
|
552
|
+
#
|
553
|
+
def destroy_sessions_by_user_id(user_id)
|
554
|
+
HUB_MUTEX.synchronize do
|
555
|
+
@hub_sessions.reject! do | key, session |
|
556
|
+
session&.session_user&.user_id == user_id
|
557
|
+
end
|
558
|
+
end
|
559
|
+
|
560
|
+
rescue => e
|
561
|
+
Sentry.capture_exception(e) if defined?(Sentry) && Sentry.respond_to?(:capture_exception)
|
562
|
+
raise
|
563
|
+
end
|
564
|
+
|
565
|
+
# WARNING: Slow
|
566
|
+
#
|
567
|
+
# This is a housekeeping task which checks sessions against Hub expiry and,
|
568
|
+
# if the session keys look to be substantially older than the value set in
|
569
|
+
# HUB_IDLE_TIME_LIMIT, the session simply deleted. If a user does return
|
570
|
+
# later, they'll see themselves in logged out state without the Flash
|
571
|
+
# warning them of an expired session, but we can't allow session keys to
|
572
|
+
# just hang around forever so *some* kind of sweep is needed.
|
573
|
+
#
|
574
|
+
# This method clearly needs to iterate over all sessions under a mutex and
|
575
|
+
# makes relatively complex checks for each, so it's fairly slow compared
|
576
|
+
# to most methods. Call it infrequently; any and all other attempts to read
|
577
|
+
# session data while the method runs will block until method finishes.
|
578
|
+
#
|
579
|
+
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
|
+
unless @hub_be_quiet
|
585
|
+
puts "Session factory: Sweeping sessions inactive for more than #{ time_limit } seconds..."
|
586
|
+
end
|
587
|
+
|
588
|
+
HUB_MUTEX.synchronize do
|
589
|
+
count_before = @hub_sessions.keys.size
|
590
|
+
|
591
|
+
@hub_sessions.reject! do | key, session |
|
592
|
+
last_used = session&.session_last_used
|
593
|
+
last_used.nil? || Time.now.utc - last_used > time_limit
|
594
|
+
end
|
595
|
+
|
596
|
+
count_after = @hub_sessions.keys.size
|
597
|
+
destroyed = count_before - count_after
|
598
|
+
end
|
599
|
+
|
600
|
+
unless @hub_be_quiet
|
601
|
+
puts "Session factory: ...Destroyed #{destroyed} session(s)"
|
602
|
+
end
|
603
|
+
|
604
|
+
rescue => e
|
605
|
+
Sentry.capture_exception(e) if defined?(Sentry) && Sentry.respond_to?(:capture_exception)
|
606
|
+
raise
|
607
|
+
end
|
608
|
+
|
505
609
|
end
|
506
610
|
|
507
611
|
#######################################################################
|
@@ -550,14 +654,65 @@ module HubSsoLib
|
|
550
654
|
|
551
655
|
module Core
|
552
656
|
|
553
|
-
#
|
657
|
+
# Log in the user. This is just syntax sugar for setting the current user
|
658
|
+
# via #hubssolib_current_user, really. You can freely use either approach
|
659
|
+
# according to your preferred aesthetics, but this method is preferred.
|
554
660
|
#
|
555
|
-
#
|
661
|
+
# +user+:: A valid HubSsoLib::User instance. If this has a +nil+ value for
|
662
|
+
# +user_id+, or if +user+ is itself +nil+, you'll cause the same
|
663
|
+
# effect as if explicitly logging *out*.
|
664
|
+
#
|
665
|
+
def hubssolib_log_in(user)
|
666
|
+
self.hubssolib_current_user = user # (which deals with all related session and cookie consequences)
|
667
|
+
end
|
668
|
+
|
669
|
+
# Log out the user. Very few applications should ever need to call this,
|
670
|
+
# though Hub certainly does and it gets used internally too.
|
671
|
+
#
|
672
|
+
def hubssolib_log_out
|
673
|
+
self.hubssolib_current_user = nil # (which deals with all related session and cookie consequences)
|
674
|
+
end
|
675
|
+
|
676
|
+
# Returns true or false if a user is logged in or not, respectively.
|
556
677
|
#
|
557
678
|
def hubssolib_logged_in?
|
558
679
|
!!self.hubssolib_current_user
|
559
680
|
end
|
560
681
|
|
682
|
+
# Accesses the current user, via the DRb server if necessary. Returns a
|
683
|
+
# HubSsoLib::User object, or +nil+ if none is available (this indicates
|
684
|
+
# that nobody is logged in, but #hubssolib_logged_in? should be used to
|
685
|
+
# check that, to allow for possible future more advanced logic within).
|
686
|
+
#
|
687
|
+
def hubssolib_current_user
|
688
|
+
hub_session = self.hubssolib_get_session()
|
689
|
+
user = hub_session&.session_user
|
690
|
+
|
691
|
+
return (user&.user_id.nil? ? nil : user)
|
692
|
+
end
|
693
|
+
|
694
|
+
# Sets the currently signed in user. Note that although this works and is
|
695
|
+
# maintained, it is recommended that #hubssolib_log_in gets called instead.
|
696
|
+
#
|
697
|
+
# +user+:: A valid HubSsoLib::User. This will replace any existing logged
|
698
|
+
# in user. If there is no session yet, one will be created.
|
699
|
+
#
|
700
|
+
def hubssolib_current_user=(user)
|
701
|
+
if user.nil?
|
702
|
+
self.hubssolib_destroy_session!
|
703
|
+
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
|
+
hub_session = self.hubssolib_create_session()
|
712
|
+
hub_session.session_user = user
|
713
|
+
end
|
714
|
+
end
|
715
|
+
|
561
716
|
# Returns markup for a link that leads to Hub's conditional login endpoint,
|
562
717
|
# inline-styled as a red "Log in" or green "Account" button. This can be
|
563
718
|
# used in page templates to avoid needing any additional images or other
|
@@ -570,7 +725,6 @@ module HubSsoLib
|
|
570
725
|
#
|
571
726
|
def hubssolib_account_link
|
572
727
|
logged_in = self.hubssolib_logged_in?()
|
573
|
-
|
574
728
|
ui_href = "#{HUB_PATH_PREFIX}/account/login_conditional"
|
575
729
|
noscript_img_src = "#{HUB_PATH_PREFIX}/account/login_indication.png"
|
576
730
|
noscript_img_tag = helpers.image_tag(noscript_img_src, size: '90x22', border: '0', alt: 'Log in or out')
|
@@ -667,51 +821,12 @@ module HubSsoLib
|
|
667
821
|
return (puser && !puser.empty? && puser != pnormal)
|
668
822
|
end
|
669
823
|
|
670
|
-
# Log out the user. Very few applications should ever need to call this,
|
671
|
-
# though Hub certainly does and it gets used internally too.
|
672
|
-
#
|
673
|
-
def hubssolib_log_out
|
674
|
-
# Causes the "hubssolib_current_[foo]=" methods to run, which
|
675
|
-
# deal with everything else.
|
676
|
-
self.hubssolib_current_user = nil
|
677
|
-
@hubssolib_current_session_proxy = nil
|
678
|
-
end
|
679
|
-
|
680
|
-
# Accesses the current session from the cookie. Creates a new session
|
681
|
-
# object if need be, but can return +nil+ if e.g. attempting to access
|
682
|
-
# session cookie data without SSL.
|
683
|
-
#
|
684
|
-
def hubssolib_current_session
|
685
|
-
@hubssolib_current_session_proxy ||= hubssolib_get_session_proxy()
|
686
|
-
end
|
687
|
-
|
688
|
-
# Accesses the current user, via the DRb server if necessary.
|
689
|
-
#
|
690
|
-
def hubssolib_current_user
|
691
|
-
hub_session = self.hubssolib_current_session
|
692
|
-
user = hub_session.nil? ? nil : hub_session.session_user
|
693
|
-
|
694
|
-
if (user && user.user_id)
|
695
|
-
return user
|
696
|
-
else
|
697
|
-
return nil
|
698
|
-
end
|
699
|
-
end
|
700
|
-
|
701
|
-
# Store the given user data in the cookie
|
702
|
-
#
|
703
|
-
def hubssolib_current_user=(user)
|
704
|
-
hub_session = self.hubssolib_current_session
|
705
|
-
hub_session.session_user = user unless hub_session.nil?
|
706
|
-
end
|
707
|
-
|
708
824
|
# Public read-only accessor methods for common user activities:
|
709
825
|
# return the current user's roles as a Roles object, or nil if
|
710
826
|
# there's no user.
|
711
827
|
#
|
712
828
|
def hubssolib_get_user_roles
|
713
|
-
|
714
|
-
user ? user.user_roles.to_authenticated_roles : nil
|
829
|
+
self.hubssolib_current_user&.user_roles&.to_authenticated_roles
|
715
830
|
end
|
716
831
|
|
717
832
|
# Public read-only accessor methods for common user activities:
|
@@ -719,8 +834,7 @@ module HubSsoLib
|
|
719
834
|
# no user. See also hubssolib_unique_name.
|
720
835
|
#
|
721
836
|
def hubssolib_get_user_name
|
722
|
-
|
723
|
-
user ? user.user_real_name : nil
|
837
|
+
self.hubssolib_current_user&.user_real_name
|
724
838
|
end
|
725
839
|
|
726
840
|
# Public read-only accessor methods for common user activities:
|
@@ -728,8 +842,7 @@ module HubSsoLib
|
|
728
842
|
# nil if there's no user. See also hubssolib_unique_name.
|
729
843
|
#
|
730
844
|
def hubssolib_get_user_id
|
731
|
-
|
732
|
-
user ? user.user_id : nil
|
845
|
+
self.hubssolib_current_user&.user_id
|
733
846
|
end
|
734
847
|
|
735
848
|
# Public read-only accessor methods for common user activities:
|
@@ -737,8 +850,7 @@ module HubSsoLib
|
|
737
850
|
# no user.
|
738
851
|
#
|
739
852
|
def hubssolib_get_user_address
|
740
|
-
|
741
|
-
user ? user.user_email : nil
|
853
|
+
self.hubssolib_current_user&.user_email
|
742
854
|
end
|
743
855
|
|
744
856
|
# Return a human-readable unique ID for a user. We don't want to
|
@@ -863,15 +975,13 @@ module HubSsoLib
|
|
863
975
|
cookies.delete(HUB_LOGIN_INDICATOR_COOKIE, domain: :all, path: '/')
|
864
976
|
|
865
977
|
if login_is_required
|
866
|
-
hubssolib_store_location
|
867
|
-
return hubssolib_must_login
|
978
|
+
hubssolib_store_location()
|
979
|
+
return hubssolib_must_login()
|
868
980
|
else
|
869
981
|
return true
|
870
982
|
end
|
871
983
|
end
|
872
984
|
|
873
|
-
# Definitely logged in.
|
874
|
-
#
|
875
985
|
cookies[HUB_LOGIN_INDICATOR_COOKIE] = {
|
876
986
|
value: HUB_LOGIN_INDICATOR_COOKIE_VALUE,
|
877
987
|
path: '/',
|
@@ -892,8 +1002,8 @@ module HubSsoLib
|
|
892
1002
|
# if OK, else indicate that access is denied.
|
893
1003
|
|
894
1004
|
if (hubssolib_session_expired?)
|
895
|
-
hubssolib_store_location
|
896
|
-
hubssolib_log_out
|
1005
|
+
hubssolib_store_location()
|
1006
|
+
hubssolib_log_out()
|
897
1007
|
hubssolib_set_flash(:attention, 'Sorry, your session timed out; you need to log in again to continue.')
|
898
1008
|
|
899
1009
|
# We mean this: redirect_to :controller => 'account', :action => 'login'
|
@@ -902,7 +1012,7 @@ module HubSsoLib
|
|
902
1012
|
redirect_to HUB_PATH_PREFIX + '/account/login'
|
903
1013
|
else
|
904
1014
|
hubssolib_set_last_used(Time.now.utc)
|
905
|
-
return hubssolib_authorized? ? true : hubssolib_access_denied
|
1015
|
+
return hubssolib_authorized? ? true : hubssolib_access_denied()
|
906
1016
|
end
|
907
1017
|
|
908
1018
|
else
|
@@ -955,7 +1065,7 @@ module HubSsoLib
|
|
955
1065
|
# Redirect to the URI stored by the most recent store_location call or
|
956
1066
|
# to the passed default.
|
957
1067
|
def hubssolib_redirect_back_or_default(default)
|
958
|
-
url = hubssolib_get_return_to
|
1068
|
+
url = hubssolib_get_return_to()
|
959
1069
|
hubssolib_set_return_to(nil)
|
960
1070
|
|
961
1071
|
redirect_to(url || default)
|
@@ -987,26 +1097,29 @@ module HubSsoLib
|
|
987
1097
|
end
|
988
1098
|
end
|
989
1099
|
|
990
|
-
#
|
991
|
-
#
|
992
|
-
#
|
993
|
-
#
|
1100
|
+
# Flash data can be carried across the Hub session, stored in the DRb
|
1101
|
+
# server as a result, and is thus cleared automatically if a session gets
|
1102
|
+
# dropped. However, we also want this to work without being logged in, so
|
1103
|
+
# in that case it uses the normal flash as a backup when *writing*.
|
994
1104
|
#
|
995
|
-
def hubssolib_get_flash
|
996
|
-
|
997
|
-
|
1105
|
+
def hubssolib_get_flash
|
1106
|
+
session = self.hubssolib_get_session()
|
1107
|
+
session&.session_flash || {}
|
998
1108
|
end
|
999
1109
|
|
1000
1110
|
def hubssolib_set_flash(symbol, message)
|
1001
|
-
|
1002
|
-
f
|
1003
|
-
f
|
1004
|
-
|
1111
|
+
session = self.hubssolib_get_session()
|
1112
|
+
f = hubssolib_get_flash() unless session.nil?
|
1113
|
+
f = self.flash if f.nil? && self.respond_to?(:flash)
|
1114
|
+
|
1115
|
+
f[symbol] = message
|
1116
|
+
|
1117
|
+
session.session_flash = f unless session.nil?
|
1005
1118
|
end
|
1006
1119
|
|
1007
1120
|
def hubssolib_clear_flash
|
1008
|
-
|
1009
|
-
|
1121
|
+
session = self.hubssolib_get_session()
|
1122
|
+
session.session_flash = {} unless session.nil?
|
1010
1123
|
end
|
1011
1124
|
|
1012
1125
|
# Return flash data for known keys, then all remaining keys, from both
|
@@ -1063,21 +1176,22 @@ module HubSsoLib
|
|
1063
1176
|
hubssolib_get_exception_data(CGI::unescape(id_data))
|
1064
1177
|
end
|
1065
1178
|
|
1066
|
-
# Inclusion hook to make various methods available as ActionView
|
1067
|
-
# helper methods.
|
1179
|
+
# Inclusion hook to make various methods available as ActionView helpers.
|
1068
1180
|
#
|
1069
1181
|
def self.included(base)
|
1070
|
-
base.
|
1071
|
-
|
1072
|
-
|
1073
|
-
|
1074
|
-
|
1075
|
-
|
1076
|
-
|
1077
|
-
|
1078
|
-
|
1079
|
-
|
1080
|
-
|
1182
|
+
if base.respond_to?(:helper_method)
|
1183
|
+
base.send(
|
1184
|
+
:helper_method,
|
1185
|
+
|
1186
|
+
:hubssolib_current_user,
|
1187
|
+
:hubssolib_unique_name,
|
1188
|
+
:hubssolib_logged_in?,
|
1189
|
+
:hubssolib_authorized?,
|
1190
|
+
:hubssolib_privileged?,
|
1191
|
+
:hubssolib_account_link,
|
1192
|
+
:hubssolib_flash_data
|
1193
|
+
)
|
1194
|
+
end
|
1081
1195
|
end
|
1082
1196
|
|
1083
1197
|
private
|
@@ -1104,6 +1218,108 @@ module HubSsoLib
|
|
1104
1218
|
HUB_BYPASS_SSL || ! Rails.env.production?
|
1105
1219
|
end
|
1106
1220
|
|
1221
|
+
# Create a new session. This MUST ONLY BE CALLED at a "log in" phase, since
|
1222
|
+
# it discards any existing session and creates a new, valid one, whether or
|
1223
|
+
# not a valid session key was being presented for the old session. It is,
|
1224
|
+
# in essence, a "clean slate" start.
|
1225
|
+
#
|
1226
|
+
# On exit, @hubssolib_session is updated; see #hubssolib_get_session for
|
1227
|
+
# the significance of that.
|
1228
|
+
#
|
1229
|
+
def hubssolib_create_session
|
1230
|
+
old_session = self.hubssolib_get_session()
|
1231
|
+
self.hubssolib_destroy_session! unless old_session.nil?
|
1232
|
+
|
1233
|
+
start_key = SecureRandom.uuid
|
1234
|
+
@hubssolib_session = hubssolib_factory().get_hub_session_proxy(start_key, request.remote_ip)
|
1235
|
+
|
1236
|
+
# The session is now stored under the rotated key, so put that into the
|
1237
|
+
# Hub session cookie so that on the next request, we can retrieve the
|
1238
|
+
# session data (and at that time, once again rotate the key).
|
1239
|
+
#
|
1240
|
+
next_key = @hubssolib_session.session_rotated_key
|
1241
|
+
self.hubssolib_store_key(next_key)
|
1242
|
+
|
1243
|
+
return @hubssolib_session
|
1244
|
+
end
|
1245
|
+
|
1246
|
+
# Gets session data based on the current +request+ details.
|
1247
|
+
#
|
1248
|
+
# * If there is no session cookie, returns +nil+
|
1249
|
+
# * If there is a session cookie but the session key is invalid, returns
|
1250
|
+
# +nil+.
|
1251
|
+
# * If there is a session cookie with valid key, session data is retrieved
|
1252
|
+
# and returned; the session key is rotated and the session cookie updated
|
1253
|
+
# with new key (that is to say that this request's response, under normal
|
1254
|
+
# conditions, will include an appropriate Set-Cookie header).
|
1255
|
+
#
|
1256
|
+
# The ivar @hubssolib_session is consulted first to avoid repeating work
|
1257
|
+
# during request processing, where it's likely that more than one call
|
1258
|
+
# will occur. This ivar is an known internal implementation detail; it is
|
1259
|
+
# also set by #hubssolib_create_session, and is cleared by calls to
|
1260
|
+
# #hubssolib_destroy_session!.
|
1261
|
+
#
|
1262
|
+
def hubssolib_get_session
|
1263
|
+
if @hubssolib_session.nil?
|
1264
|
+
key = cookies[HUB_COOKIE_NAME]
|
1265
|
+
|
1266
|
+
if key.nil? || key == ''
|
1267
|
+
@hubssolib_session = nil
|
1268
|
+
else
|
1269
|
+
hub_session = hubssolib_factory().get_hub_session_proxy(key, request.remote_ip)
|
1270
|
+
|
1271
|
+
if hub_session.nil? # Invalid key in cookie
|
1272
|
+
@hubssolib_session = nil
|
1273
|
+
else
|
1274
|
+
@hubssolib_session = hub_session
|
1275
|
+
|
1276
|
+
next_key = @hubssolib_session.session_rotated_key
|
1277
|
+
self.hubssolib_store_key(next_key)
|
1278
|
+
end
|
1279
|
+
end
|
1280
|
+
end
|
1281
|
+
|
1282
|
+
return @hubssolib_session
|
1283
|
+
end
|
1284
|
+
|
1285
|
+
# Destroys the current session. If there isn't one, then it has few side
|
1286
|
+
# effects other than making sure all session-related cookies are deleted.
|
1287
|
+
#
|
1288
|
+
def hubssolib_destroy_session!
|
1289
|
+
session = self.hubssolib_get_session()
|
1290
|
+
|
1291
|
+
# Remember, if creating or retrieving a session for a request, the key is
|
1292
|
+
# rotated and stored in the session under #session_rotated_key. To delete
|
1293
|
+
# that session within that same request - since rotation happens even if
|
1294
|
+
# the first place the session gets read is right here, in this method -
|
1295
|
+
# we must use the rotated key.
|
1296
|
+
#
|
1297
|
+
unless session.nil?
|
1298
|
+
hubssolib_factory().destroy_session_by_key(session.session_rotated_key)
|
1299
|
+
end
|
1300
|
+
|
1301
|
+
@hubssolib_session = nil
|
1302
|
+
|
1303
|
+
cookies.delete(HUB_COOKIE_NAME, domain: :all, path: '/')
|
1304
|
+
cookies.delete(HUB_LOGIN_INDICATOR_COOKIE, domain: :all, path: '/')
|
1305
|
+
end
|
1306
|
+
|
1307
|
+
# Store the Hub's session key in the Hub cookie.
|
1308
|
+
#
|
1309
|
+
# +key+:: Session key used to retrieve the session again on the *next*
|
1310
|
+
# request. The session has already been stored under that new key
|
1311
|
+
# internally. Do not call with +nil+ or blank keys.
|
1312
|
+
#
|
1313
|
+
def hubssolib_store_key(key)
|
1314
|
+
cookies[HUB_COOKIE_NAME] = {
|
1315
|
+
value: key,
|
1316
|
+
path: '/',
|
1317
|
+
domain: :all,
|
1318
|
+
secure: ! hub_bypass_ssl?,
|
1319
|
+
httponly: true
|
1320
|
+
}
|
1321
|
+
end
|
1322
|
+
|
1107
1323
|
# Indicate that the user must log in to complete their request.
|
1108
1324
|
# Returns false to enable a before_filter to return through this
|
1109
1325
|
# method while ensuring that the previous action processing is
|
@@ -1164,64 +1380,45 @@ module HubSsoLib
|
|
1164
1380
|
# have not logged out anyway, and the Hub isn't intended for Fort Knox.
|
1165
1381
|
# At the time of writing the trade-off of usability vs security is
|
1166
1382
|
# considered acceptable, though who knows, the view may change in future.
|
1383
|
+
#
|
1384
|
+
# Same applies for PATCH and PUT.
|
1385
|
+
#
|
1386
|
+
last_used = self.hubssolib_get_last_used
|
1167
1387
|
|
1168
|
-
|
1169
|
-
|
1170
|
-
|
1171
|
-
|
1172
|
-
def hubssolib_get_session_proxy
|
1173
|
-
# If we're not using SSL, forget it
|
1174
|
-
return nil unless request.ssl? || hub_bypass_ssl?
|
1175
|
-
|
1176
|
-
key = cookies[HUB_COOKIE_NAME] || SecureRandom.uuid
|
1177
|
-
hub_session = hubssolib_factory().get_hub_session_proxy(key, request.remote_ip)
|
1178
|
-
key = hub_session.session_key_rotation unless hub_session.nil?
|
1179
|
-
|
1180
|
-
cookies[HUB_COOKIE_NAME] = {
|
1181
|
-
value: key,
|
1182
|
-
path: '/',
|
1183
|
-
domain: :all,
|
1184
|
-
secure: ! hub_bypass_ssl?,
|
1185
|
-
httponly: true
|
1186
|
-
}
|
1187
|
-
|
1188
|
-
return hub_session
|
1189
|
-
|
1190
|
-
rescue Exception => e
|
1191
|
-
|
1192
|
-
# At this point there tends to be no Session data, so we're
|
1193
|
-
# going to have to encode the exception data into the URI...
|
1194
|
-
# It must be escaped twice, as many servers treat "%2F" in a
|
1195
|
-
# URI as a "/" and Apache may flat refuse to serve the page,
|
1196
|
-
# raising a 404 error unless "AllowEncodedSlashes on" is
|
1197
|
-
# specified in its configuration.
|
1198
|
-
|
1199
|
-
suffix = '/' + CGI::escape(CGI::escape(hubssolib_set_exception_data(e)))
|
1200
|
-
new_path = HUB_PATH_PREFIX + '/tasks/service'
|
1201
|
-
redirect_to(new_path + suffix) unless request.path.include?(new_path)
|
1202
|
-
|
1203
|
-
return nil
|
1204
|
-
end
|
1388
|
+
(
|
1389
|
+
request.method != :post &&
|
1390
|
+
request.method != :patch &&
|
1391
|
+
request.method != :put &&
|
1205
1392
|
|
1206
|
-
|
1207
|
-
|
1393
|
+
last_used && Time.now.utc - last_used > HUB_IDLE_TIME_LIMIT
|
1394
|
+
)
|
1208
1395
|
end
|
1209
1396
|
|
1210
|
-
# Return an array of Hub User objects representing users based
|
1211
|
-
#
|
1212
|
-
#
|
1213
|
-
#
|
1214
|
-
#
|
1215
|
-
#
|
1216
|
-
#
|
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.
|
1217
1412
|
#
|
1218
1413
|
def hubssolib_enumerate_users
|
1219
|
-
|
1220
|
-
|
1221
|
-
|
1222
|
-
sessions
|
1223
|
-
|
1224
|
-
|
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
|
1225
1422
|
end
|
1226
1423
|
|
1227
1424
|
return users
|
@@ -1263,23 +1460,21 @@ module HubSsoLib
|
|
1263
1460
|
# the session data is available, else return default values.
|
1264
1461
|
|
1265
1462
|
def hubssolib_get_last_used
|
1266
|
-
|
1267
|
-
session ? session.session_last_used : Time.now.utc
|
1463
|
+
self.hubssolib_get_session()&.session_last_used || Time.now.utc
|
1268
1464
|
end
|
1269
1465
|
|
1270
1466
|
def hubssolib_set_last_used(time)
|
1271
|
-
|
1272
|
-
|
1467
|
+
session = self.hubssolib_get_session()
|
1468
|
+
session.session_last_used = time unless session.nil?
|
1273
1469
|
end
|
1274
1470
|
|
1275
1471
|
def hubssolib_get_return_to
|
1276
|
-
|
1277
|
-
session ? session.session_return_to : nil
|
1472
|
+
self.hubssolib_get_session()&.session_return_to
|
1278
1473
|
end
|
1279
1474
|
|
1280
1475
|
def hubssolib_set_return_to(uri)
|
1281
|
-
|
1282
|
-
|
1476
|
+
session = self.hubssolib_get_session()
|
1477
|
+
session.session_return_to = uri unless session.nil?
|
1283
1478
|
end
|
1284
1479
|
|
1285
1480
|
end # Core module
|
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.4.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-
|
10
|
+
date: 2025-03-24 00:00:00.000000000 Z
|
11
11
|
dependencies:
|
12
12
|
- !ruby/object:Gem::Dependency
|
13
13
|
name: drb
|