trusona 0.16.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.
Files changed (84) hide show
  1. checksums.yaml +7 -0
  2. checksums.yaml.gz.sig +0 -0
  3. data.tar.gz.sig +0 -0
  4. data/.env.example +4 -0
  5. data/.gitignore +20 -0
  6. data/.rspec +3 -0
  7. data/.ruby-version +1 -0
  8. data/.travis.yml +24 -0
  9. data/Gemfile +8 -0
  10. data/Guardfile +54 -0
  11. data/LICENSE +201 -0
  12. data/README.md +234 -0
  13. data/Rakefile +8 -0
  14. data/bin/console +21 -0
  15. data/bin/setup +8 -0
  16. data/certs/trusona.pem +26 -0
  17. data/checksum/trusona-0.16.0.gem.sha512 +1 -0
  18. data/integrations/device_user_binding_integration_spec.rb +6 -0
  19. data/integrations/identity_documents_spec.rb +28 -0
  20. data/integrations/spec_helper.rb +23 -0
  21. data/integrations/tru_code_spec.rb +80 -0
  22. data/integrations/trusonafication_spec.rb +61 -0
  23. data/integrations/user_accounts_spec.rb +39 -0
  24. data/integrations/user_identifiers_spec.rb +32 -0
  25. data/lib/trusona.rb +157 -0
  26. data/lib/trusona/api/client.rb +60 -0
  27. data/lib/trusona/api/hashed_message.rb +71 -0
  28. data/lib/trusona/api/signed_request.rb +96 -0
  29. data/lib/trusona/api/verified_response.rb +85 -0
  30. data/lib/trusona/device.rb +31 -0
  31. data/lib/trusona/device_user_binding.rb +56 -0
  32. data/lib/trusona/errors.rb +51 -0
  33. data/lib/trusona/helpers/key_normalizer.rb +15 -0
  34. data/lib/trusona/helpers/time_normalizer.rb +17 -0
  35. data/lib/trusona/identity_document.rb +64 -0
  36. data/lib/trusona/mappers/base_mapper.rb +69 -0
  37. data/lib/trusona/mappers/device_mapper.rb +13 -0
  38. data/lib/trusona/mappers/device_user_binding_mapper.rb +13 -0
  39. data/lib/trusona/mappers/identity_document_mapper.rb +17 -0
  40. data/lib/trusona/mappers/nil_mapper.rb +11 -0
  41. data/lib/trusona/mappers/tru_code_mapper.rb +13 -0
  42. data/lib/trusona/mappers/trusonafication_mapper.rb +19 -0
  43. data/lib/trusona/mappers/user_account_mapper.rb +13 -0
  44. data/lib/trusona/mappers/user_identifier_mapper.rb +13 -0
  45. data/lib/trusona/resources/base_resource.rb +29 -0
  46. data/lib/trusona/resources/device.rb +22 -0
  47. data/lib/trusona/resources/device_user_binding.rb +30 -0
  48. data/lib/trusona/resources/device_user_binding_activation.rb +27 -0
  49. data/lib/trusona/resources/identity_document.rb +36 -0
  50. data/lib/trusona/resources/tru_code.rb +42 -0
  51. data/lib/trusona/resources/trusonafication.rb +137 -0
  52. data/lib/trusona/resources/user_account.rb +84 -0
  53. data/lib/trusona/resources/user_identifier.rb +49 -0
  54. data/lib/trusona/resources/validators.rb +22 -0
  55. data/lib/trusona/services/account_lookups_service.rb +17 -0
  56. data/lib/trusona/services/base_service.rb +117 -0
  57. data/lib/trusona/services/device_user_bindings_service.rb +22 -0
  58. data/lib/trusona/services/devices_service.rb +15 -0
  59. data/lib/trusona/services/identity_documents_service.rb +31 -0
  60. data/lib/trusona/services/tru_codes_service.rb +26 -0
  61. data/lib/trusona/services/trusonafication_service.rb +16 -0
  62. data/lib/trusona/services/user_accounts_service.rb +17 -0
  63. data/lib/trusona/services/user_identifiers_service.rb +16 -0
  64. data/lib/trusona/tru_code.rb +38 -0
  65. data/lib/trusona/tru_code_config.rb +45 -0
  66. data/lib/trusona/trusonafication.rb +133 -0
  67. data/lib/trusona/user_account.rb +17 -0
  68. data/lib/trusona/user_identifier.rb +15 -0
  69. data/lib/trusona/version.rb +5 -0
  70. data/lib/trusona/workers/device_finder.rb +18 -0
  71. data/lib/trusona/workers/device_user_binding_activator.rb +25 -0
  72. data/lib/trusona/workers/device_user_binding_creator.rb +26 -0
  73. data/lib/trusona/workers/identity_document_finder.rb +25 -0
  74. data/lib/trusona/workers/tru_code_creator.rb +17 -0
  75. data/lib/trusona/workers/tru_code_finder.rb +19 -0
  76. data/lib/trusona/workers/trusonafication_creator.rb +46 -0
  77. data/lib/trusona/workers/trusonafication_finder.rb +27 -0
  78. data/lib/trusona/workers/user_account_finder.rb +39 -0
  79. data/lib/trusona/workers/user_identifier_creator.rb +17 -0
  80. data/lib/trusona/workers/user_identifier_finder.rb +29 -0
  81. data/release-gem +17 -0
  82. data/trusona.gemspec +43 -0
  83. metadata +333 -0
  84. metadata.gz.sig +1 -0
@@ -0,0 +1,15 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Trusona
4
+ module Helpers
5
+ ##
6
+ # Noramlizes keys by turning all key values into symbols
7
+ module KeyNormalizer
8
+ def normalize_keys(hash)
9
+ hash.each_with_object({}) do |(k, v), memo|
10
+ memo[k.to_sym] = v
11
+ end
12
+ end
13
+ end
14
+ end
15
+ end
@@ -0,0 +1,17 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Trusona
4
+ module Helpers
5
+ ##
6
+ # Normalizes Strings, Date, DateTime, and Time into a UTC Time object
7
+ module TimeNormalizer
8
+ def normalize_time(time)
9
+ return nil if time.nil?
10
+ return time.to_time.gmtime if time.is_a?(DateTime)
11
+ return time.gmtime if time.is_a?(Time)
12
+ return Time.parse(time).gmtime if time.is_a?(String)
13
+ return nil if time.is_a?(Date)
14
+ end
15
+ end
16
+ end
17
+ end
@@ -0,0 +1,64 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Trusona
4
+ ##
5
+ # Helpful for finding Identity Documents given a user identifier or
6
+ # Identity Document identifier
7
+ class IdentityDocument
8
+ ##
9
+ # Finds all of the identity documents associated with the user identifier
10
+ #
11
+ # @param [String] user_identifier The user identifier associated with the
12
+ # requested identity documents.
13
+ # @return [Array<Trusona::Resources::IdentityDocument>] A collection of
14
+ # identity
15
+ # documents.
16
+ # @raise ArgumentError if the user_identifier is nil
17
+ #
18
+ # @raise [Trusona::InvalidResourceError] if the resource is not +valid?+
19
+ # @see Trusona::Resources::BaseResource#valid?
20
+ # @raise [Trusona::BadRequestError] if the request is improperly formatted
21
+ # @raise [Trusona::UnauthorizedRequestError] if the request is unauthorized.
22
+ # Typically the result of invalid or revoked Trusona SDK keys.
23
+ # @raise [Trusona::ApiError] if the Trusona API is experiencing problems.
24
+ #
25
+ # @example
26
+ #
27
+ # Trusona::IdentityDocument.find_all('user-1234')
28
+ #
29
+
30
+ def self.all(user_identifier: nil)
31
+ # rubocop:disable Metrics/LineLength
32
+ raise ArgumentError, 'A user identifier is required.' unless user_identifier
33
+ # rubocop:enable Metrics/LineLength
34
+
35
+ Trusona::Workers::IdentityDocumentFinder.new.find_all(user_identifier)
36
+ end
37
+
38
+ ##
39
+ # Finds the specified identity document
40
+ #
41
+ # @param [String] id The id of the Identity Document
42
+ # @return [Trusona::Resources::IdentityDocument] the Identity Document
43
+ # @raise ArgumentError if the user_identifier is nil
44
+ #
45
+ # @raise [Trusona::InvalidResourceError] if the resource is not +valid?+
46
+ # @see Trusona::Resources::BaseResource#valid?
47
+ # @raise [Trusona::BadRequestError] if the request is improperly formatted
48
+ # @raise [Trusona::UnauthorizedRequestError] if the request is unauthorized.
49
+ # Typically the result of invalid or revoked Trusona SDK keys.
50
+ # @raise [Trusona::ApiError] if the Trusona API is experiencing problems.
51
+ #
52
+ # @example
53
+ #
54
+ # Trusona::IdentityDocument.find('4FDF044D-89FB-4043-947D-A029CF785B5F')
55
+ #
56
+ def self.find(id: nil)
57
+ # rubocop:disable Metrics/LineLength
58
+ raise ArgumentError, 'An Identity Document identifier is required.' unless id
59
+ # rubocop:enable Metrics/LineLength
60
+
61
+ Trusona::Workers::IdentityDocumentFinder.new.find(id)
62
+ end
63
+ end
64
+ end
@@ -0,0 +1,69 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Trusona
4
+ module Mappers
5
+ #
6
+ ## A starting point for specific resource mappers
7
+ class BaseMapper
8
+ include Trusona::Helpers::KeyNormalizer
9
+ attr_accessor :custom_mappings, :resource
10
+
11
+ def map(response, existing = {})
12
+ raise ArgumentError if response_invalid?(response)
13
+ raise ArgumentError if existing_invalid?(existing)
14
+
15
+ raise if existing.nil?
16
+
17
+ parsed_response = response.to_h
18
+
19
+ if parsed_response.is_a?(Array)
20
+ parsed_response.map { |r| map_item(r, existing) }.compact
21
+ else
22
+ map_item(parsed_response, existing)
23
+ end
24
+ end
25
+
26
+ private
27
+
28
+ def map_item(item, existing)
29
+ return nil if item.nil? || item.empty?
30
+ item = normalize_keys(item)
31
+ item = merge_existing_state(item, existing.to_h)
32
+ item = map_custom_fields(item)
33
+
34
+ (resource || Trusona::Resources::BaseResource).send(:new, item)
35
+ end
36
+
37
+ def merge_existing_state(response, existing)
38
+ existing.to_h.compact.merge(response)
39
+ end
40
+
41
+ def map_custom_fields(response)
42
+ return response if custom_mappings.nil? || custom_mappings.empty?
43
+ custom_mappings.each do |original_key, new_key|
44
+ value = response.delete(original_key)
45
+ response[new_key] = value
46
+ end
47
+
48
+ response
49
+ end
50
+
51
+ def response_invalid?(response)
52
+ return true if response.nil?
53
+ return true unless response.respond_to?(:to_h)
54
+ false
55
+ end
56
+
57
+ def resource_invalid?(resource)
58
+ return true if resource.nil?
59
+ return true unless resource.respond_to?(:new)
60
+ false
61
+ end
62
+
63
+ def existing_invalid?(existing)
64
+ return true unless existing.respond_to?(:to_h)
65
+ false
66
+ end
67
+ end
68
+ end
69
+ end
@@ -0,0 +1,13 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Trusona
4
+ module Mappers
5
+ #
6
+ ## Maps Device Responses
7
+ class DeviceMapper < BaseMapper
8
+ def resource
9
+ Trusona::Resources::Device
10
+ end
11
+ end
12
+ end
13
+ end
@@ -0,0 +1,13 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Trusona
4
+ module Mappers
5
+ #
6
+ ## Maps Device User Binding Responses
7
+ class DeviceUserBindingMapper < BaseMapper
8
+ def resource
9
+ Trusona::Resources::DeviceUserBinding
10
+ end
11
+ end
12
+ end
13
+ end
@@ -0,0 +1,17 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Trusona
4
+ module Mappers
5
+ #
6
+ ## Maps Identity Document Responses
7
+ class IdentityDocumentMapper < BaseMapper
8
+ def resource
9
+ Trusona::Resources::IdentityDocument
10
+ end
11
+
12
+ def custom_mappings
13
+ { hash: :document_hash }
14
+ end
15
+ end
16
+ end
17
+ end
@@ -0,0 +1,11 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Trusona
4
+ module Mappers
5
+ #
6
+ ## A default mapper that does nothing
7
+ class NilMapper
8
+ def map(_json, _resource); end
9
+ end
10
+ end
11
+ end
@@ -0,0 +1,13 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Trusona
4
+ module Mappers
5
+ #
6
+ ## A Mapper for TruCodes
7
+ class TruCodeMapper < BaseMapper
8
+ def resource
9
+ Trusona::Resources::TruCode
10
+ end
11
+ end
12
+ end
13
+ end
@@ -0,0 +1,19 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Trusona
4
+ module Mappers
5
+ ##
6
+ # A mapper for Trusonafication
7
+ class TrusonaficationMapper < BaseMapper
8
+ def resource
9
+ Trusona::Resources::Trusonafication
10
+ end
11
+
12
+ def custom_mappings
13
+ {
14
+ desired_level: :level
15
+ }
16
+ end
17
+ end
18
+ end
19
+ end
@@ -0,0 +1,13 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Trusona
4
+ module Mappers
5
+ #
6
+ ## A Mapper for User Accounts
7
+ class UserAccountMapper < BaseMapper
8
+ def resource
9
+ Trusona::Resources::UserAccount
10
+ end
11
+ end
12
+ end
13
+ end
@@ -0,0 +1,13 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Trusona
4
+ module Mappers
5
+ #
6
+ ## A Mapper for UserIdentifiers
7
+ class UserIdentifierMapper < BaseMapper
8
+ def resource
9
+ Trusona::Resources::UserIdentifier
10
+ end
11
+ end
12
+ end
13
+ end
@@ -0,0 +1,29 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Trusona
4
+ module Resources
5
+ #
6
+ ## A base resource
7
+ class BaseResource
8
+ include Trusona::Resources::Validators
9
+ attr_reader :id
10
+
11
+ def initialize(params = {})
12
+ @params = params
13
+ @id = params[:id] || params['id']
14
+ end
15
+
16
+ def to_h
17
+ @params
18
+ end
19
+
20
+ def to_json
21
+ JSON(to_h)
22
+ end
23
+
24
+ def valid?
25
+ true
26
+ end
27
+ end
28
+ end
29
+ end
@@ -0,0 +1,22 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Trusona
4
+ module Resources
5
+ #
6
+ ## A record representing a device in the Trusona API
7
+ class Device < BaseResource
8
+ include Trusona::Helpers::KeyNormalizer
9
+ include Trusona::Helpers::TimeNormalizer
10
+
11
+ attr_reader :active, :activated_at
12
+
13
+ def initialize(params = {})
14
+ super
15
+ @params = normalize_keys(params)
16
+
17
+ @active = @params[:active]
18
+ @activated_at = normalize_time(@params[:activated_at])
19
+ end
20
+ end
21
+ end
22
+ end
@@ -0,0 +1,30 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Trusona
4
+ module Resources
5
+ #
6
+ ## A record representing a device and user binding in the Trusona API
7
+ class DeviceUserBinding < BaseResource
8
+ include Trusona::Resources::Validators
9
+ include Trusona::Helpers::KeyNormalizer
10
+ attr_reader :user_identifier, :device_identifier, :active, :id
11
+
12
+ def initialize(params = {})
13
+ normalized_params = normalize_keys(params)
14
+ @user_identifier = normalized_params[:user_identifier]
15
+ @device_identifier = normalized_params[:device_identifier]
16
+ @active = normalized_params[:active]
17
+ @id = normalized_params[:id]
18
+
19
+ @params = normalized_params
20
+ raise Trusona::InvalidResourceError unless validate
21
+ end
22
+
23
+ def validate
24
+ return false unless @user_identifier
25
+ return false unless @device_identifier
26
+ true
27
+ end
28
+ end
29
+ end
30
+ end
@@ -0,0 +1,27 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Trusona
4
+ module Resources
5
+ #
6
+ ## A Device User Binding Activation
7
+ class DeviceUserBindingActivation < BaseResource
8
+ include Trusona::Resources::Validators
9
+ include Trusona::Helpers::KeyNormalizer
10
+ attr_reader :id, :active
11
+
12
+ def initialize(params = {})
13
+ normalized_params = normalize_keys(params)
14
+ @id = normalized_params[:id]
15
+ @active = normalized_params[:active]
16
+
17
+ @params = normalized_params
18
+ raise Trusona::InvalidResourceError unless validate
19
+ end
20
+
21
+ def validate
22
+ return false unless @id
23
+ true
24
+ end
25
+ end
26
+ end
27
+ end
@@ -0,0 +1,36 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Trusona
4
+ module Resources
5
+ ##
6
+ # A record representing an identity document in the Trusona API
7
+ class IdentityDocument < BaseResource
8
+ include Trusona::Resources::Validators
9
+ include Trusona::Helpers::KeyNormalizer
10
+
11
+ attr_accessor :document_hash, :id, :type, :verification_status,
12
+ :user_identifier
13
+
14
+ def initialize(params)
15
+ normalized = normalize_keys(params)
16
+
17
+ @params = normalized
18
+ @document_hash = normalized[:document_hash]
19
+ @id = normalized[:id]
20
+ @type = normalized[:type]
21
+ @verification_status = normalized[:verification_status]
22
+ @user_identifier = normalized[:user_identifier]
23
+ end
24
+
25
+ def to_json
26
+ JSON(
27
+ hash: @document_hash,
28
+ id: @id,
29
+ type: @type,
30
+ verification_status: @verification_status,
31
+ user_identifier: @user_identifier
32
+ )
33
+ end
34
+ end
35
+ end
36
+ end