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.
Files changed (51) hide show
  1. checksums.yaml +5 -5
  2. checksums.yaml.gz.sig +0 -0
  3. data.tar.gz.sig +0 -0
  4. data/.gitignore +3 -0
  5. data/.rubocop.yml +13 -0
  6. data/.ruby-version +1 -1
  7. data/.travis.yml +26 -16
  8. data/DEVELOP.md +11 -0
  9. data/README.md +4 -3
  10. data/bin/console +0 -1
  11. data/integrations/buster.rb +15 -0
  12. data/integrations/spec_helper.rb +3 -0
  13. data/integrations/trusonafication_spec.rb +18 -3
  14. data/lib/trusona.rb +3 -1
  15. data/lib/trusona/api/signed_request.rb +4 -1
  16. data/lib/trusona/api/verified_response.rb +6 -1
  17. data/lib/trusona/identity_document.rb +0 -6
  18. data/lib/trusona/mappers/base_mapper.rb +5 -0
  19. data/lib/trusona/mappers/paired_tru_code_mapper.rb +1 -0
  20. data/lib/trusona/resources/base_resource.rb +1 -1
  21. data/lib/trusona/resources/device_user_binding.rb +1 -0
  22. data/lib/trusona/resources/device_user_binding_activation.rb +2 -1
  23. data/lib/trusona/resources/identity_document.rb +4 -4
  24. data/lib/trusona/resources/paired_tru_code.rb +1 -0
  25. data/lib/trusona/resources/tru_code.rb +2 -1
  26. data/lib/trusona/resources/trusonafication.rb +29 -20
  27. data/lib/trusona/resources/user_account.rb +6 -2
  28. data/lib/trusona/resources/user_identifier.rb +3 -1
  29. data/lib/trusona/resources/validators.rb +1 -0
  30. data/lib/trusona/services/base_service.rb +9 -2
  31. data/lib/trusona/services/device_user_bindings_service.rb +3 -3
  32. data/lib/trusona/services/identity_documents_service.rb +3 -3
  33. data/lib/trusona/services/paired_tru_code_service.rb +1 -0
  34. data/lib/trusona/trusonafication.rb +32 -3
  35. data/lib/trusona/version.rb +1 -1
  36. data/lib/trusona/workers/device_finder.rb +1 -0
  37. data/lib/trusona/workers/identity_document_finder.rb +1 -0
  38. data/lib/trusona/workers/paired_tru_code_finder.rb +1 -0
  39. data/lib/trusona/workers/tru_code_finder.rb +1 -0
  40. data/lib/trusona/workers/trusonafication_canceler.rb +27 -0
  41. data/lib/trusona/workers/trusonafication_creator.rb +2 -1
  42. data/lib/trusona/workers/trusonafication_finder.rb +1 -1
  43. data/lib/trusona/workers/user_account_finder.rb +1 -3
  44. data/lib/trusona/workers/user_deactivator.rb +1 -3
  45. data/lib/trusona/workers/user_identifier_finder.rb +1 -0
  46. data/trusona.cnf +14 -0
  47. data/trusona.gemspec +9 -7
  48. metadata +85 -57
  49. metadata.gz.sig +0 -0
  50. data/certs/trusona.pem +0 -34
  51. data/trusona.key.pem.enc +0 -0
@@ -2,6 +2,7 @@
2
2
 
3
3
  module Trusona
4
4
  module Resources
5
+ ## A Paired Trucode
5
6
  class PairedTruCode < BaseResource
6
7
  attr_reader :identifier
7
8
 
@@ -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(device_identifier: device_identifier,
48
- user_identifier: user_identifier,
49
- trucode_id: trucode_id,
50
- trusona_id: trusona_id,
51
- email: email,
52
- resource: resource,
53
- action: action,
54
- desired_level: level,
55
- id: id,
56
- status: @status,
57
- prompt: prompt,
58
- user_presence: user_presence,
59
- custom_fields: custom_fields,
60
- expires_at: expires_at&.iso8601)
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 = params_with_symbol_keys
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
@@ -7,6 +7,7 @@ module Trusona
7
7
  module Validators
8
8
  def present?(item)
9
9
  return false if item.nil? || item.to_s.empty?
10
+
10
11
  true
11
12
  end
12
13
 
@@ -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
- client: Trusona::Api::HTTPClient.new(Trusona.config.api_host),
10
- mapper: Trusona::Mappers::DeviceUserBindingMapper.new
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
- client: Trusona::Api::HTTPClient.new(Trusona.config.api_host),
10
- mapper: Trusona::Mappers::IdentityDocumentMapper.new
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'
@@ -2,6 +2,7 @@
2
2
 
3
3
  module Trusona
4
4
  module Services
5
+ ## Paired trucode service
5
6
  class PairedTruCodeService < BaseService
6
7
  def initialize(client: Trusona::Api::HTTPClient.new)
7
8
  super(client: client, mapper: Trusona::Mappers::PairedTruCodeMapper.new)
@@ -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?+
@@ -1,5 +1,5 @@
1
1
  # frozen_string_literal: true
2
2
 
3
3
  module Trusona
4
- VERSION = '2.2.0'
4
+ VERSION = '2.5.2'
5
5
  end
@@ -11,6 +11,7 @@ module Trusona
11
11
 
12
12
  def find(id = nil)
13
13
  raise(ArgumentError, 'A device identifier is required.') unless id
14
+
14
15
  @service.get(Trusona::Resources::Device.new(id: id))
15
16
  end
16
17
  end
@@ -18,6 +18,7 @@ module Trusona
18
18
 
19
19
  def find(id = nil)
20
20
  raise(ArgumentError, 'An Identity Document id is required.') unless id
21
+
21
22
  @service.get(Trusona::Resources::IdentityDocument.new(id: id))
22
23
  end
23
24
  end
@@ -2,6 +2,7 @@
2
2
 
3
3
  module Trusona
4
4
  module Workers
5
+ # A Paired Trucode Finder
5
6
  class PairedTruCodeFinder
6
7
  def initialize(service: Trusona::Services::PairedTruCodeService.new)
7
8
  @service = service
@@ -11,6 +11,7 @@ module Trusona
11
11
 
12
12
  def find(id)
13
13
  raise ArgumentError, 'Missing TruCode Id' unless id
14
+
14
15
  resource = Trusona::Resources::BaseResource.new(id: id)
15
16
  @service.get(resource)
16
17
  end
@@ -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
@@ -24,4 +24,4 @@ module Trusona
24
24
  end
25
25
  end
26
26
  end
27
- end
27
+ end