govuk_personalisation 0.2.0 → 0.6.0

Sign up to get free protection for your applications and to get access to all the features.
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: 4f19f176c2aa1874082e94b2798bc865b4b612ec92682ba41d8b6ed386239543
4
- data.tar.gz: 50c849e85a7d2115316c7db368e16f2c15649f82265b1a2c87a787aae49018e1
3
+ metadata.gz: '08b691342c4c22592162f9eb11e8c8f74b8fea75af5669568c71d15337166626'
4
+ data.tar.gz: 0e266ec1d1f91a7537ccef2c1a4398e90b4c0435546772d41877cfd34e9df092
5
5
  SHA512:
6
- metadata.gz: e9ddd7d63d119782c1a8663ba076f233881dbfd04fadd9007d53afaaa68555145a10c30e1a0e2e04f65867d79c13ba662a9febb0e2ba4ba657bf056056efa6f2
7
- data.tar.gz: '02950b7e918f99b08f2cbeeec51da89e8aeaaa7be115250c4e918aebf0a06e6fb7768c59cf208f20dc17c11c7edf346fcc2edbe1c4ff9a3c2caabad9673d4941'
6
+ metadata.gz: 5f265633173eee3b60e436c781ddb4047d849471cc1760d88e8c3e497a79d8a0fd188a4c3908feb33fecf3b8c2b171dbbc1ea7f08ed837fae2d58cfc5658ae35
7
+ data.tar.gz: d90dc8a74720981e16d7d379939c08072806ead930d012bb40c1a57ac2e90ae3879a10d7fb0bfbe3d874d39c70312952d72c0c9eca452fd757906c253ecb402e
data/CHANGELOG.md CHANGED
@@ -1,7 +1,26 @@
1
- ## [0.2.0]
1
+ # 0.6.0
2
+
3
+ - Add `GovukPersonalisation::Flash` and helper methods to the concern ([#9](https://github.com/alphagov/govuk_personalisation/pull/9))
4
+ - Ensure every method has RDoc ([#10](https://github.com/alphagov/govuk_personalisation/pull/10))
5
+ - Remove unused `GovukPersonalisation::Error` class ([#10](https://github.com/alphagov/govuk_personalisation/pull/10))
6
+ - BREAKING: Rename `GovukPersonalisation::AccountConcern` to `GovukPersonalisation::ControllerConcern` ([#11](https://github.com/alphagov/govuk_personalisation/pull/11))
7
+
8
+ # 0.5.0
9
+
10
+ - Rename header name constants ([#7](https://github.com/alphagov/govuk_personalisation/pull/7))
11
+
12
+ # 0.4.0
13
+
14
+ - Add ability to set GOVUK-Account-Session ([#6](https://github.com/alphagov/govuk_personalisation/pull/6))
15
+
16
+ # 0.3.0
17
+
18
+ - Always set `Vary: GOVUK-Account-Session` response header ([#4](https://github.com/alphagov/govuk_personalisation/pull/4))
19
+
20
+ # 0.2.0
2
21
 
3
22
  - Add AccountConcern to extract common account-related functionality ([#3](https://github.com/alphagov/govuk_personalisation/pull/3))
4
23
 
5
- ## [0.1.0] - 2021-05-12
24
+ # 0.1.0
6
25
 
7
26
  - Initial release
data/README.md CHANGED
@@ -1,6 +1,6 @@
1
- # GovukPersonalisation
1
+ # GOV.UK Personalisation
2
2
 
3
- A gem to hold shared code which other GOV.UK apps will use to implement
3
+ A gem to hold shared code which other GOV.UK apps use to implement
4
4
  accounts-related functionality.
5
5
 
6
6
  ## Installation
@@ -17,13 +17,64 @@ And then execute:
17
17
  $ bundle install
18
18
  ```
19
19
 
20
- ## Technical documentation
20
+ ## Usage
21
21
 
22
- <!-- TODO -->...
22
+ ### Rails concern
23
23
 
24
- ### Testing
24
+ Include the concern into a controller:
25
25
 
26
- <!-- TODO -->...
26
+ ```ruby
27
+ include GovukPersonalisation::AccountConcern
28
+ ```
29
+
30
+ And it will add `before_action` methods to:
31
+
32
+ - fetch the account session identifier from the request headers, making it available as `account_session_header`
33
+ - set a `Vary` response header, to ensure responses for different users are cached differently by the CDN
34
+
35
+ The following functions are available:
36
+
37
+ - `logged_in?` - check if there is an account session header
38
+ - `set_account_session_header` - replace the current session value, and set the response header the CDN uses to update the user's cookie
39
+ - `logout!` - clear the session value, and set the response header the CDN uses to remove the user's cookie
40
+
41
+ When run in development mode (`RAILS_ENV=development`), a cookie on
42
+ `dev.gov.uk` is used instead of custom headers.
43
+
44
+ ### Test helpers
45
+
46
+ There are test helpers for both request and feature specs to log the
47
+ user in. Include the relevant helper:
48
+
49
+ ```ruby
50
+ include GovukPersonalisation::TestHelpers::Requests
51
+ ```
52
+
53
+ *or*
54
+
55
+ ```ruby
56
+ include GovukPersonalisation::TestHelpers::Features
57
+ ```
58
+
59
+ And then log the user in:
60
+
61
+ ```ruby
62
+ before do
63
+ mock_logged_in_session
64
+ end
65
+ ```
66
+
67
+ If you need a specific session identifier, for example to match
68
+ against it in WebMock stubs or with the [gds-api-adapters][] test
69
+ helpers, you can pass it as an argument:
70
+
71
+ ```ruby
72
+ before do
73
+ mock_logged_in_session("your-session-identifier-goes-here")
74
+ end
75
+ ```
76
+
77
+ [gds-api-adapters]: https://github.com/alphagov/gds-api-adapters
27
78
 
28
79
  ## License
29
80
 
@@ -1,9 +1,9 @@
1
1
  # frozen_string_literal: true
2
2
 
3
3
  require "govuk_personalisation/version"
4
- require "govuk_personalisation/account_concern"
4
+ require "govuk_personalisation/controller_concern"
5
+ require "govuk_personalisation/flash"
6
+ require "govuk_personalisation/test_helpers/features"
7
+ require "govuk_personalisation/test_helpers/requests"
5
8
 
6
- module GovukPersonalisation
7
- class Error < StandardError; end
8
- # Your code goes here...
9
- end
9
+ module GovukPersonalisation; end
@@ -0,0 +1,129 @@
1
+ # frozen_string_literal: true
2
+
3
+ require "active_support/concern"
4
+
5
+ module GovukPersonalisation
6
+ module ControllerConcern
7
+ extend ActiveSupport::Concern
8
+
9
+ ACCOUNT_SESSION_INTERNAL_HEADER_NAME = "HTTP_GOVUK_ACCOUNT_SESSION"
10
+ ACCOUNT_SESSION_HEADER_NAME = "GOVUK-Account-Session"
11
+ ACCOUNT_END_SESSION_HEADER_NAME = "GOVUK-Account-End-Session"
12
+ ACCOUNT_SESSION_DEV_COOKIE_NAME = "govuk_account_session"
13
+
14
+ included do
15
+ before_action :fetch_account_session_header
16
+ before_action :set_account_vary_header
17
+ attr_accessor :account_session_header
18
+ attr_reader :account_flash
19
+ end
20
+
21
+ # Read the `GOVUK-Account-Session` request header and set the
22
+ # `@account_session_header` and `@account_flash` variables. Also
23
+ # sets a response header with an empty flash if there is a flash
24
+ # in the request.
25
+ #
26
+ # This is called as a `before_action`
27
+ #
28
+ # This should not be called after either of the
29
+ # `@govuk_account_session` or flash to return to the user have
30
+ # been changed, as those changes will be overwritten.
31
+ def fetch_account_session_header
32
+ session_with_flash =
33
+ if request.headers[ACCOUNT_SESSION_INTERNAL_HEADER_NAME]
34
+ request.headers[ACCOUNT_SESSION_INTERNAL_HEADER_NAME].presence
35
+ elsif Rails.env.development?
36
+ cookies[ACCOUNT_SESSION_DEV_COOKIE_NAME]
37
+ end
38
+
39
+ @account_session_header, flash = GovukPersonalisation::Flash.decode_session(session_with_flash)
40
+ @account_flash = (flash || []).index_with { |_| true }
41
+ @new_account_flash = {}
42
+
43
+ set_account_session_header unless @account_flash.empty?
44
+ end
45
+
46
+ # Set the `Vary: GOVUK-Account-Session` response header.
47
+ #
48
+ # This is called as a `before_action`, to ensure that pages
49
+ # rendered using one user's session are not served to another by
50
+ # our CDN. You should only skip this action if you are certain
51
+ # that the response does not include any personalisation, or if
52
+ # you prevent caching in some other way (for example, with
53
+ # `Cache-Control: no-store`).
54
+ def set_account_vary_header
55
+ response.headers["Vary"] = [response.headers["Vary"], ACCOUNT_SESSION_HEADER_NAME].compact.join(", ")
56
+ end
57
+
58
+ # Check if the user has a session.
59
+ #
60
+ # This does not call account-api to verify that the session is
61
+ # valid, but an invalid session would not allow a user to access
62
+ # any personal data anyway.
63
+ #
64
+ # @return [true, false] whether the user has a session
65
+ def logged_in?
66
+ account_session_header.present?
67
+ end
68
+
69
+ # Set a new session header.
70
+ #
71
+ # This should be called after any API call to account-api which
72
+ # returns a new session value. This is called automatically after
73
+ # updating the flash with `account_flash_add` or
74
+ # `account_flash_keep`
75
+ #
76
+ # Calling this after calling `logout!` will not prevent the user
77
+ # from being logged out.
78
+ #
79
+ # @param govuk_account_session [String, nil] the new session identifier
80
+ def set_account_session_header(govuk_account_session = nil)
81
+ @account_session_header = govuk_account_session if govuk_account_session
82
+
83
+ session_with_flash = GovukPersonalisation::Flash.encode_session(@account_session_header, @new_account_flash.keys)
84
+
85
+ response.headers[ACCOUNT_SESSION_HEADER_NAME] = session_with_flash
86
+ if Rails.env.development?
87
+ cookies[ACCOUNT_SESSION_DEV_COOKIE_NAME] = {
88
+ value: session_with_flash,
89
+ domain: "dev.gov.uk",
90
+ }
91
+ end
92
+ end
93
+
94
+ # Clear the `@account_session_header` and set the logout response
95
+ # header.
96
+ def logout!
97
+ response.headers[ACCOUNT_END_SESSION_HEADER_NAME] = "1"
98
+ @account_session_header = nil
99
+ if Rails.env.development?
100
+ cookies[ACCOUNT_SESSION_DEV_COOKIE_NAME] = {
101
+ value: "",
102
+ domain: "dev.gov.uk",
103
+ expires: 1.second.ago,
104
+ }
105
+ end
106
+ end
107
+
108
+ # Add a message to the flash to return to the user. This does not
109
+ # change `account_flash`
110
+ #
111
+ # @param message [String] the message to add
112
+ #
113
+ # @return [true, false] whether the message is valid (and so has been added)
114
+ def account_flash_add(message)
115
+ return false unless GovukPersonalisation::Flash.valid_message? message
116
+
117
+ @new_account_flash[message] = true
118
+ set_account_session_header
119
+ true
120
+ end
121
+
122
+ # Copy all messages from the `account_flash` into the flash to
123
+ # return to the user.
124
+ def account_flash_keep
125
+ @new_account_flash = @account_flash.merge(@new_account_flash)
126
+ set_account_session_header
127
+ end
128
+ end
129
+ end
@@ -0,0 +1,50 @@
1
+ # frozen_string-literal: true
2
+
3
+ module GovukPersonalisation::Flash
4
+ SESSION_SEPARATOR = "$$"
5
+ MESSAGE_SEPARATOR = ","
6
+ MESSAGE_REGEX = /\A[a-zA-Z0-9._\-]+\z/.freeze
7
+
8
+ # Splits the session header into a session value (suitable for using
9
+ # in account-api calls) and flash messages.
10
+ #
11
+ # @param encoded_session [String] the value of the `GOVUK-Account-Session` header
12
+ #
13
+ # @return [Array(String, Array<String>), nil] the session value and the flash messages
14
+ def self.decode_session(encoded_session)
15
+ session_bits = encoded_session&.split(SESSION_SEPARATOR)
16
+ return if session_bits.blank?
17
+
18
+ if session_bits.length == 1
19
+ [session_bits[0], []]
20
+ else
21
+ [session_bits[0], session_bits[1].split(MESSAGE_SEPARATOR)]
22
+ end
23
+ end
24
+
25
+ # Encodes the session value and a list of flash messages into a
26
+ # session header which can be returned to the user.
27
+ #
28
+ # @param session [String] the session value
29
+ # @param flash [Array<String>] the flash messages, which must all be `valid_message?`
30
+ #
31
+ # @return [String] the encoded session header value
32
+ def self.encode_session(session, flash)
33
+ if flash.blank?
34
+ session
35
+ else
36
+ "#{session}#{SESSION_SEPARATOR}#{flash.join(MESSAGE_SEPARATOR)}"
37
+ end
38
+ end
39
+
40
+ # Check if a string is valid as a flash message.
41
+ #
42
+ # @param message [String, nil] the flash message
43
+ #
44
+ # @return [true, false] whether the message is valid or not.
45
+ def self.valid_message?(message)
46
+ return false if message.nil?
47
+
48
+ message.match? MESSAGE_REGEX
49
+ end
50
+ end
@@ -0,0 +1,14 @@
1
+ module GovukPersonalisation
2
+ module TestHelpers
3
+ module Features
4
+ # Set the `GOVUK-Account-Session` request header in the page
5
+ # driver.
6
+ #
7
+ # @param session_id [String] the session identifier
8
+ # @param flash [Array<String>, nil] the flash messages
9
+ def mock_logged_in_session(session_id = "placeholder", flash = nil)
10
+ page.driver.header("GOVUK-Account-Session", GovukPersonalisation::Flash.encode_session(session_id, flash))
11
+ end
12
+ end
13
+ end
14
+ end
@@ -0,0 +1,13 @@
1
+ module GovukPersonalisation
2
+ module TestHelpers
3
+ module Requests
4
+ # Set the `GOVUK-Account-Session` request header.
5
+ #
6
+ # @param session_id [String] the session identifier
7
+ # @param flash [Array<String>, nil] the flash messages
8
+ def mock_logged_in_session(session_id = "placeholder", flash = nil)
9
+ request.headers["GOVUK-Account-Session"] = GovukPersonalisation::Flash.encode_session(session_id, flash)
10
+ end
11
+ end
12
+ end
13
+ end
@@ -1,5 +1,5 @@
1
1
  # frozen_string_literal: true
2
2
 
3
3
  module GovukPersonalisation
4
- VERSION = "0.2.0"
4
+ VERSION = "0.6.0"
5
5
  end
metadata CHANGED
@@ -1,14 +1,14 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: govuk_personalisation
3
3
  version: !ruby/object:Gem::Version
4
- version: 0.2.0
4
+ version: 0.6.0
5
5
  platform: ruby
6
6
  authors:
7
7
  - GOV.UK Dev
8
8
  autorequire:
9
9
  bindir: exe
10
10
  cert_chain: []
11
- date: 2021-06-01 00:00:00.000000000 Z
11
+ date: 2021-08-06 00:00:00.000000000 Z
12
12
  dependencies:
13
13
  - !ruby/object:Gem::Dependency
14
14
  name: rails
@@ -116,7 +116,10 @@ files:
116
116
  - bin/setup
117
117
  - govuk_personalisation.gemspec
118
118
  - lib/govuk_personalisation.rb
119
- - lib/govuk_personalisation/account_concern.rb
119
+ - lib/govuk_personalisation/controller_concern.rb
120
+ - lib/govuk_personalisation/flash.rb
121
+ - lib/govuk_personalisation/test_helpers/features.rb
122
+ - lib/govuk_personalisation/test_helpers/requests.rb
120
123
  - lib/govuk_personalisation/version.rb
121
124
  homepage: https://github.com/alphagov/govuk_personalisation
122
125
  licenses:
@@ -137,7 +140,7 @@ required_rubygems_version: !ruby/object:Gem::Requirement
137
140
  - !ruby/object:Gem::Version
138
141
  version: '0'
139
142
  requirements: []
140
- rubygems_version: 3.1.6
143
+ rubygems_version: 3.0.3
141
144
  signing_key:
142
145
  specification_version: 4
143
146
  summary: A gem to hold shared code which other GOV.UK apps will use to implement accounts-related
@@ -1,55 +0,0 @@
1
- # frozen_string_literal: true
2
-
3
- require "active_support/concern"
4
-
5
- module GovukPersonalisation
6
- module AccountConcern
7
- extend ActiveSupport::Concern
8
-
9
- ACCOUNT_SESSION_REQUEST_HEADER_NAME = "HTTP_GOVUK_ACCOUNT_SESSION"
10
- ACCOUNT_SESSION_RESPONSE_HEADER_NAME = "GOVUK-Account-Session"
11
- ACCOUNT_END_SESSION_RESPONSE_HEADER_NAME = "GOVUK-Account-End-Session"
12
- ACCOUNT_SESSION_DEV_COOKIE_NAME = "govuk_account_session"
13
-
14
- included do
15
- before_action :fetch_account_session_header
16
- attr_accessor :account_session_header
17
- end
18
-
19
- def logged_in?
20
- account_session_header.present?
21
- end
22
-
23
- def fetch_account_session_header
24
- @account_session_header =
25
- if request.headers[ACCOUNT_SESSION_REQUEST_HEADER_NAME]
26
- request.headers[ACCOUNT_SESSION_REQUEST_HEADER_NAME].presence
27
- elsif Rails.env.development?
28
- cookies[ACCOUNT_SESSION_DEV_COOKIE_NAME]
29
- end
30
- end
31
-
32
- def set_account_session_header(govuk_account_session = nil)
33
- @account_session_header = govuk_account_session if govuk_account_session
34
- response.headers[ACCOUNT_SESSION_RESPONSE_HEADER_NAME] = @account_session_header
35
- if Rails.env.development?
36
- cookies[ACCOUNT_SESSION_DEV_COOKIE_NAME] = {
37
- value: @account_session_header,
38
- domain: "dev.gov.uk",
39
- }
40
- end
41
- end
42
-
43
- def logout!
44
- response.headers[ACCOUNT_END_SESSION_RESPONSE_HEADER_NAME] = "1"
45
- @account_session_header = nil
46
- if Rails.env.development?
47
- cookies[ACCOUNT_SESSION_DEV_COOKIE_NAME] = {
48
- value: "",
49
- domain: "dev.gov.uk",
50
- expires: 1.second.ago,
51
- }
52
- end
53
- end
54
- end
55
- end