trusona 2.2.0 → 2.5.2
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 +5 -5
- checksums.yaml.gz.sig +0 -0
- data.tar.gz.sig +0 -0
- data/.gitignore +3 -0
- data/.rubocop.yml +13 -0
- data/.ruby-version +1 -1
- data/.travis.yml +26 -16
- data/DEVELOP.md +11 -0
- data/README.md +4 -3
- data/bin/console +0 -1
- data/integrations/buster.rb +15 -0
- data/integrations/spec_helper.rb +3 -0
- data/integrations/trusonafication_spec.rb +18 -3
- data/lib/trusona.rb +3 -1
- data/lib/trusona/api/signed_request.rb +4 -1
- data/lib/trusona/api/verified_response.rb +6 -1
- data/lib/trusona/identity_document.rb +0 -6
- data/lib/trusona/mappers/base_mapper.rb +5 -0
- data/lib/trusona/mappers/paired_tru_code_mapper.rb +1 -0
- data/lib/trusona/resources/base_resource.rb +1 -1
- data/lib/trusona/resources/device_user_binding.rb +1 -0
- data/lib/trusona/resources/device_user_binding_activation.rb +2 -1
- data/lib/trusona/resources/identity_document.rb +4 -4
- data/lib/trusona/resources/paired_tru_code.rb +1 -0
- data/lib/trusona/resources/tru_code.rb +2 -1
- data/lib/trusona/resources/trusonafication.rb +29 -20
- data/lib/trusona/resources/user_account.rb +6 -2
- data/lib/trusona/resources/user_identifier.rb +3 -1
- data/lib/trusona/resources/validators.rb +1 -0
- data/lib/trusona/services/base_service.rb +9 -2
- data/lib/trusona/services/device_user_bindings_service.rb +3 -3
- data/lib/trusona/services/identity_documents_service.rb +3 -3
- data/lib/trusona/services/paired_tru_code_service.rb +1 -0
- data/lib/trusona/trusonafication.rb +32 -3
- data/lib/trusona/version.rb +1 -1
- data/lib/trusona/workers/device_finder.rb +1 -0
- data/lib/trusona/workers/identity_document_finder.rb +1 -0
- data/lib/trusona/workers/paired_tru_code_finder.rb +1 -0
- data/lib/trusona/workers/tru_code_finder.rb +1 -0
- data/lib/trusona/workers/trusonafication_canceler.rb +27 -0
- data/lib/trusona/workers/trusonafication_creator.rb +2 -1
- data/lib/trusona/workers/trusonafication_finder.rb +1 -1
- data/lib/trusona/workers/user_account_finder.rb +1 -3
- data/lib/trusona/workers/user_deactivator.rb +1 -3
- data/lib/trusona/workers/user_identifier_finder.rb +1 -0
- data/trusona.cnf +14 -0
- data/trusona.gemspec +9 -7
- metadata +85 -57
- metadata.gz.sig +0 -0
- data/certs/trusona.pem +0 -34
- data/trusona.key.pem.enc +0 -0
@@ -19,7 +19,7 @@ module Trusona
|
|
19
19
|
raise ArgumentError unless validate
|
20
20
|
end
|
21
21
|
|
22
|
-
def to_json
|
22
|
+
def to_json(*_args)
|
23
23
|
JSON(to_h)
|
24
24
|
end
|
25
25
|
|
@@ -35,6 +35,7 @@ module Trusona
|
|
35
35
|
|
36
36
|
def validate
|
37
37
|
return false unless present?(@relying_party_id)
|
38
|
+
|
38
39
|
true
|
39
40
|
end
|
40
41
|
end
|
@@ -1,5 +1,6 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
|
+
# rubocop:disable Metrics/ClassLength
|
3
4
|
module Trusona
|
4
5
|
module Resources
|
5
6
|
##
|
@@ -11,13 +12,14 @@ module Trusona
|
|
11
12
|
attr_accessor :device_identifier, :user_identifier, :trucode_id,
|
12
13
|
:resource, :action, :level, :id, :email,
|
13
14
|
:accepted_level, :trusona_id, :expires_at,
|
14
|
-
:user_presence, :prompt, :custom_fields
|
15
|
+
:user_presence, :prompt, :custom_fields, :callback_url
|
15
16
|
|
16
17
|
# rubocop:disable Metrics/AbcSize
|
17
18
|
# rubocop:disable Metrics/MethodLength
|
18
19
|
def initialize(params = {})
|
19
20
|
@params = normalize_keys(params)
|
20
21
|
return if @params.nil?
|
22
|
+
|
21
23
|
self.accepted_level = determine_accepted_level(@params)
|
22
24
|
self.action = @params[:action]
|
23
25
|
self.device_identifier = @params[:device_identifier]
|
@@ -32,6 +34,7 @@ module Trusona
|
|
32
34
|
self.prompt = defaulting_to(true, @params[:prompt])
|
33
35
|
self.user_presence = defaulting_to(true, @params[:user_presence])
|
34
36
|
self.custom_fields = @params[:custom_fields]
|
37
|
+
self.callback_url = @params[:callback_url]
|
35
38
|
|
36
39
|
@status = @params[:status]
|
37
40
|
end
|
@@ -43,27 +46,31 @@ module Trusona
|
|
43
46
|
end
|
44
47
|
|
45
48
|
# rubocop:disable Metrics/MethodLength
|
46
|
-
def to_json
|
47
|
-
JSON(
|
48
|
-
|
49
|
-
|
50
|
-
|
51
|
-
|
52
|
-
|
53
|
-
|
54
|
-
|
55
|
-
|
56
|
-
|
57
|
-
|
58
|
-
|
59
|
-
|
60
|
-
|
49
|
+
def to_json(*_args)
|
50
|
+
JSON(
|
51
|
+
device_identifier: device_identifier,
|
52
|
+
user_identifier: user_identifier,
|
53
|
+
trucode_id: trucode_id,
|
54
|
+
trusona_id: trusona_id,
|
55
|
+
email: email,
|
56
|
+
resource: resource,
|
57
|
+
action: action,
|
58
|
+
desired_level: level,
|
59
|
+
id: id,
|
60
|
+
status: @status,
|
61
|
+
prompt: prompt,
|
62
|
+
user_presence: user_presence,
|
63
|
+
custom_fields: custom_fields,
|
64
|
+
expires_at: expires_at&.iso8601,
|
65
|
+
callback_url: callback_url
|
66
|
+
)
|
61
67
|
end
|
62
68
|
# rubocop:enable Metrics/MethodLength
|
63
69
|
|
64
70
|
def accepted?
|
65
71
|
return true if status == :accepted
|
66
72
|
return true if status == :accepted_at_higher_level
|
73
|
+
|
67
74
|
false
|
68
75
|
end
|
69
76
|
|
@@ -75,8 +82,8 @@ module Trusona
|
|
75
82
|
attributes_present && attributes_filled
|
76
83
|
end
|
77
84
|
|
78
|
-
# rubocop:disable MethodLength
|
79
|
-
# rubocop:disable CyclomaticComplexity
|
85
|
+
# rubocop:disable Metrics/MethodLength
|
86
|
+
# rubocop:disable Metrics/CyclomaticComplexity
|
80
87
|
def status
|
81
88
|
case @status
|
82
89
|
when 'INVALID_TRUSONA_ID'
|
@@ -99,14 +106,15 @@ module Trusona
|
|
99
106
|
:invalid
|
100
107
|
end
|
101
108
|
end
|
102
|
-
# rubocop:enable CyclomaticComplexity
|
103
|
-
# rubocop:enable MethodLength
|
109
|
+
# rubocop:enable Metrics/CyclomaticComplexity
|
110
|
+
# rubocop:enable Metrics/MethodLength
|
104
111
|
|
105
112
|
private
|
106
113
|
|
107
114
|
def defaulting_to(value, param)
|
108
115
|
return value if param.nil?
|
109
116
|
return value if param.respond_to?(:empty?) && param.empty?
|
117
|
+
|
110
118
|
param
|
111
119
|
end
|
112
120
|
|
@@ -137,3 +145,4 @@ module Trusona
|
|
137
145
|
end
|
138
146
|
end
|
139
147
|
end
|
148
|
+
# rubocop:enable Metrics/ClassLength
|
@@ -38,12 +38,12 @@ module Trusona
|
|
38
38
|
@emails = parse_emails(params_with_symbol_keys[:emails])
|
39
39
|
@max_level = parse_max_level(params_with_symbol_keys[:metadata])
|
40
40
|
|
41
|
-
@params
|
41
|
+
@params = params_with_symbol_keys
|
42
42
|
end
|
43
43
|
# rubocop:enable Metrics/MethodLength
|
44
44
|
# rubocop:enable Metrics/AbcSize
|
45
45
|
|
46
|
-
def to_json
|
46
|
+
def to_json(*_args)
|
47
47
|
JSON(to_h)
|
48
48
|
end
|
49
49
|
|
@@ -52,20 +52,24 @@ module Trusona
|
|
52
52
|
def determine_status(status)
|
53
53
|
return Status::INACTIVE if status == 'inactive'
|
54
54
|
return Status::ACTIVE if status == 'active'
|
55
|
+
|
55
56
|
Status::UNKNOWN
|
56
57
|
end
|
57
58
|
|
58
59
|
def parse_emails(emails)
|
59
60
|
return [] if emails.nil? || emails.empty?
|
61
|
+
|
60
62
|
emails.map { |e| UserAccountEmail.new(e) }
|
61
63
|
end
|
62
64
|
|
63
65
|
def parse_max_level(metadata)
|
64
66
|
return Level::ENTRY unless metadata
|
67
|
+
|
65
68
|
level = metadata['max_level'] || metadata[:max_level]
|
66
69
|
|
67
70
|
return Level::ESSENTIAL if level == 'essential'
|
68
71
|
return Level::EXECUTIVE if level == 'executive'
|
72
|
+
|
69
73
|
Level::ENTRY
|
70
74
|
end
|
71
75
|
|
@@ -19,7 +19,7 @@ module Trusona
|
|
19
19
|
@params
|
20
20
|
end
|
21
21
|
|
22
|
-
def to_json
|
22
|
+
def to_json(*_args)
|
23
23
|
JSON(to_h)
|
24
24
|
end
|
25
25
|
|
@@ -36,12 +36,14 @@ module Trusona
|
|
36
36
|
def attributes_present
|
37
37
|
return false unless @params.key?(:identifier)
|
38
38
|
return false unless @params.key?(:trusona_id)
|
39
|
+
|
39
40
|
true
|
40
41
|
end
|
41
42
|
|
42
43
|
def attributes_filled
|
43
44
|
return false if @params.fetch(:identifier).empty?
|
44
45
|
return false if @params.fetch(:trusona_id).empty?
|
46
|
+
|
45
47
|
true
|
46
48
|
end
|
47
49
|
end
|
@@ -21,21 +21,25 @@ module Trusona
|
|
21
21
|
|
22
22
|
def get(resource)
|
23
23
|
raise Trusona::InvalidResourceError unless resource.id
|
24
|
+
|
24
25
|
handle(@client.get(member_path(resource)), resource)
|
25
26
|
end
|
26
27
|
|
27
28
|
def create(resource)
|
28
29
|
raise Trusona::InvalidResourceError unless resource.valid?
|
30
|
+
|
29
31
|
handle(@client.post(collection_path, resource.to_json), resource)
|
30
32
|
end
|
31
33
|
|
32
34
|
def update(resource)
|
33
35
|
raise Trusona::InvalidResourceError unless resource.id
|
36
|
+
|
34
37
|
handle(@client.patch(member_path(resource), resource.to_json), resource)
|
35
38
|
end
|
36
39
|
|
37
40
|
def delete(resource)
|
38
41
|
raise Trusona::InvalidResourceError unless resource.id
|
42
|
+
|
39
43
|
handle(@client.delete(member_path(resource)), resource)
|
40
44
|
end
|
41
45
|
|
@@ -53,12 +57,13 @@ module Trusona
|
|
53
57
|
raise Trusona::SigningError unless response.verified?
|
54
58
|
end
|
55
59
|
|
56
|
-
# rubocop:disable MethodLength
|
60
|
+
# rubocop:disable Metrics/MethodLength
|
57
61
|
# rubocop:disable Metrics/CyclomaticComplexity
|
58
62
|
def handle(response, resource = {})
|
59
63
|
@response = response
|
60
64
|
|
61
65
|
raise if resource.nil?
|
66
|
+
|
62
67
|
case response.code
|
63
68
|
when 200..299
|
64
69
|
success(response, resource)
|
@@ -78,7 +83,7 @@ module Trusona
|
|
78
83
|
raise Trusona::RequestError, readable_error
|
79
84
|
end
|
80
85
|
end
|
81
|
-
# rubocop:enable MethodLength
|
86
|
+
# rubocop:enable Metrics/MethodLength
|
82
87
|
# rubocop:enable Metrics/CyclomaticComplexity
|
83
88
|
|
84
89
|
def success(response, resource)
|
@@ -113,10 +118,12 @@ module Trusona
|
|
113
118
|
def readable_error
|
114
119
|
default = '[UNKNOWN] Error - An unknown error has occurred.'
|
115
120
|
return default unless @response
|
121
|
+
|
116
122
|
body = @response.to_h
|
117
123
|
msg = []
|
118
124
|
msg << "[#{body['error']}] #{body['message']} - #{body['description']}"
|
119
125
|
return msg.join("\n") unless body['field_errors']
|
126
|
+
|
120
127
|
body['field_errors'].each do |field|
|
121
128
|
msg << "\t #{field.join(' => ')}"
|
122
129
|
end
|
@@ -6,9 +6,9 @@ module Trusona
|
|
6
6
|
## Device User Bindings Service
|
7
7
|
class DeviceUserBindingsService < BaseService
|
8
8
|
def initialize(
|
9
|
-
|
10
|
-
|
11
|
-
|
9
|
+
client: Trusona::Api::HTTPClient.new(Trusona.config.api_host),
|
10
|
+
mapper: Trusona::Mappers::DeviceUserBindingMapper.new
|
11
|
+
)
|
12
12
|
@client = client
|
13
13
|
@mapper = mapper
|
14
14
|
@resource_path = '/api/v2/user_devices'
|
@@ -6,9 +6,9 @@ module Trusona
|
|
6
6
|
## Identity Documents Service
|
7
7
|
class IdentityDocumentsService < BaseService
|
8
8
|
def initialize(
|
9
|
-
|
10
|
-
|
11
|
-
|
9
|
+
client: Trusona::Api::HTTPClient.new(Trusona.config.api_host),
|
10
|
+
mapper: Trusona::Mappers::IdentityDocumentMapper.new
|
11
|
+
)
|
12
12
|
@client = client
|
13
13
|
@mapper = mapper
|
14
14
|
@resource_path = '/api/v2/identity_documents'
|
@@ -28,6 +28,29 @@ module Trusona
|
|
28
28
|
Trusona::Workers::TrusonaficationFinder.new.find(trusonafication_id)
|
29
29
|
end
|
30
30
|
|
31
|
+
##
|
32
|
+
# Cancels existing IN_PROGRESS Trusonafications using their ID. Once
|
33
|
+
# canceled the trusonafication can no longer be acted upon.
|
34
|
+
#
|
35
|
+
# @param trusonafication_id [String] the ID of an existing Trusonafication
|
36
|
+
# @raise [Trusona::InvalidRecordIdentifier] if the +trusonafication_id+ is
|
37
|
+
# empty or nil.
|
38
|
+
# @raise [Trusona::ResourceNotFoundError] if the resource does not exist
|
39
|
+
# in the Trusona API
|
40
|
+
# @raise [Trusona::UnprocessableEntityError] if the resource could not be
|
41
|
+
# cancel because it was not IN_PROGRESS
|
42
|
+
#
|
43
|
+
# @example
|
44
|
+
# Trusona::Trusonafication.cancel('756c1034-2159-4cf9-bd48-662a60a7afff')
|
45
|
+
#
|
46
|
+
def self.cancel(trusonafication_id)
|
47
|
+
if trusonafication_id.nil? || trusonafication_id.strip.empty?
|
48
|
+
raise Trusona::InvalidRecordIdentifier, 'Trusonafication ID is missing'
|
49
|
+
end
|
50
|
+
|
51
|
+
Trusona::Workers::TrusonaficationCanceler.new.cancel(trusonafication_id)
|
52
|
+
end
|
53
|
+
|
31
54
|
##
|
32
55
|
# Creates a Trusonafication using the supplied options
|
33
56
|
#
|
@@ -55,12 +78,18 @@ module Trusona
|
|
55
78
|
# Accept or Reject this Trusonafication?
|
56
79
|
# @option params [String] :expires_at ('90 seconds') The ISO-8601 UTC
|
57
80
|
# timestamp of the Trusonafication's expiration.
|
58
|
-
# @param timeout [Int] (30) The max amount of time, in seconds, to wait
|
59
|
-
# for a response from the Trusona API when polling for a Trusonafication
|
60
|
-
# result
|
61
81
|
# @option params [Hash] :custom_fields Optional data to be associated with
|
62
82
|
# this Trusonafication and can be used to constomize any UX elements. Total
|
63
83
|
# size of data is limited to 1MB.
|
84
|
+
# @option params [String] :callback_url A HTTPS URL to POST to call when the
|
85
|
+
# trusonafication has been completed (accepted, rejected, or expired).
|
86
|
+
#
|
87
|
+
# NOTE: The URL should include a randomized segment so it cannot be guessed
|
88
|
+
# and abused by third-parties e.g. https://your.domain.com/completed_authentications/f8abe61d-4e51-493f-97b1-464c157624f2.
|
89
|
+
#
|
90
|
+
# @param timeout [Int] (30) The max amount of time, in seconds, to wait
|
91
|
+
# for a response from the Trusona API when polling for a Trusonafication
|
92
|
+
# result
|
64
93
|
# @yield [Trusona::Resources::Trusonafication] Yields the completed
|
65
94
|
# Trusonafication to the block
|
66
95
|
# @raise [Trusona::InvalidResourceError] if the resource is not +valid?+
|
data/lib/trusona/version.rb
CHANGED
@@ -0,0 +1,27 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module Trusona
|
4
|
+
module Workers
|
5
|
+
#
|
6
|
+
## Cancel a Trusonafication
|
7
|
+
class TrusonaficationCanceler
|
8
|
+
def initialize(service: nil)
|
9
|
+
@service = service || Trusona::Services::TrusonaficationService.new
|
10
|
+
end
|
11
|
+
|
12
|
+
def cancel(trusonafication_id)
|
13
|
+
if trusonafication_id.nil? || trusonafication_id.strip.empty?
|
14
|
+
raise(
|
15
|
+
Trusona::InvalidResourceError,
|
16
|
+
'Trusonafication Id cannot be empty or nil'
|
17
|
+
)
|
18
|
+
end
|
19
|
+
|
20
|
+
resource = Trusona::Resources::Trusonafication.new(
|
21
|
+
id: trusonafication_id
|
22
|
+
)
|
23
|
+
@service.delete(resource)
|
24
|
+
end
|
25
|
+
end
|
26
|
+
end
|
27
|
+
end
|
@@ -14,6 +14,7 @@ module Trusona
|
|
14
14
|
def create(params: {}, timeout: nil, &block)
|
15
15
|
raise ArgumentError, 'Missing or empty params hash' if
|
16
16
|
params.nil? || params.empty?
|
17
|
+
|
17
18
|
resource = Trusona::Resources::Trusonafication.new(params)
|
18
19
|
trusonafication = @service.create(resource)
|
19
20
|
return trusonafication unless block_given?
|
@@ -43,4 +44,4 @@ module Trusona
|
|
43
44
|
# rubocop:enable Metrics/MethodLength
|
44
45
|
end
|
45
46
|
end
|
46
|
-
end
|
47
|
+
end
|