clerk-sdk-ruby 4.0.0.beta3 → 4.0.0.beta4

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 (204) hide show
  1. checksums.yaml +4 -4
  2. data/.env.example +3 -0
  3. data/.github/workflows/main.yml +22 -14
  4. data/.gitignore +7 -1
  5. data/.rspec +3 -0
  6. data/.ruby-version +1 -0
  7. data/CHANGELOG.md +22 -0
  8. data/Gemfile +26 -3
  9. data/Gemfile.lock +269 -13
  10. data/Guardfile +14 -0
  11. data/README.md +71 -11
  12. data/Rakefile +50 -6
  13. data/apps/rack/app.rb +67 -0
  14. data/apps/rack/config.ru +17 -0
  15. data/apps/rack/middleware/disable_paths.rb +13 -0
  16. data/apps/rails-api/.dockerignore +41 -0
  17. data/apps/rails-api/.gitattributes +9 -0
  18. data/apps/rails-api/.gitignore +32 -0
  19. data/apps/rails-api/.kamal/hooks/docker-setup.sample +3 -0
  20. data/apps/rails-api/.kamal/hooks/post-deploy.sample +14 -0
  21. data/apps/rails-api/.kamal/hooks/post-proxy-reboot.sample +3 -0
  22. data/apps/rails-api/.kamal/hooks/pre-build.sample +51 -0
  23. data/apps/rails-api/.kamal/hooks/pre-connect.sample +47 -0
  24. data/apps/rails-api/.kamal/hooks/pre-deploy.sample +109 -0
  25. data/apps/rails-api/.kamal/hooks/pre-proxy-reboot.sample +3 -0
  26. data/apps/rails-api/.kamal/secrets +17 -0
  27. data/apps/rails-api/.rubocop.yml +8 -0
  28. data/apps/rails-api/.ruby-version +1 -0
  29. data/apps/rails-api/Dockerfile +69 -0
  30. data/apps/rails-api/Gemfile +54 -0
  31. data/apps/rails-api/Gemfile.lock +374 -0
  32. data/apps/rails-api/README.md +24 -0
  33. data/apps/rails-api/Rakefile +6 -0
  34. data/apps/rails-api/app/controllers/application_controller.rb +3 -0
  35. data/apps/rails-api/app/controllers/home_controller.rb +5 -0
  36. data/apps/rails-api/app/jobs/application_job.rb +7 -0
  37. data/apps/rails-api/app/mailers/application_mailer.rb +4 -0
  38. data/apps/rails-api/app/models/application_record.rb +3 -0
  39. data/apps/rails-api/app/views/layouts/mailer.html.erb +13 -0
  40. data/apps/rails-api/app/views/layouts/mailer.text.erb +1 -0
  41. data/apps/rails-api/bin/brakeman +7 -0
  42. data/apps/rails-api/bin/bundle +109 -0
  43. data/apps/rails-api/bin/dev +2 -0
  44. data/apps/rails-api/bin/docker-entrypoint +14 -0
  45. data/apps/rails-api/bin/jobs +6 -0
  46. data/apps/rails-api/bin/kamal +27 -0
  47. data/apps/rails-api/bin/rails +4 -0
  48. data/apps/rails-api/bin/rake +4 -0
  49. data/apps/rails-api/bin/rubocop +8 -0
  50. data/apps/rails-api/bin/setup +34 -0
  51. data/apps/rails-api/bin/thrust +5 -0
  52. data/apps/rails-api/config/application.rb +36 -0
  53. data/apps/rails-api/config/boot.rb +4 -0
  54. data/apps/rails-api/config/cable.yml +17 -0
  55. data/apps/rails-api/config/cache.yml +16 -0
  56. data/apps/rails-api/config/credentials.yml.enc +1 -0
  57. data/apps/rails-api/config/database.yml +41 -0
  58. data/apps/rails-api/config/deploy.yml +116 -0
  59. data/apps/rails-api/config/environment.rb +5 -0
  60. data/apps/rails-api/config/environments/development.rb +70 -0
  61. data/apps/rails-api/config/environments/production.rb +88 -0
  62. data/apps/rails-api/config/environments/test.rb +53 -0
  63. data/apps/rails-api/config/initializers/cors.rb +16 -0
  64. data/apps/rails-api/config/initializers/filter_parameter_logging.rb +8 -0
  65. data/apps/rails-api/config/initializers/inflections.rb +16 -0
  66. data/apps/rails-api/config/locales/en.yml +31 -0
  67. data/apps/rails-api/config/puma.rb +41 -0
  68. data/apps/rails-api/config/queue.yml +18 -0
  69. data/apps/rails-api/config/recurring.yml +10 -0
  70. data/apps/rails-api/config/routes.rb +10 -0
  71. data/apps/rails-api/config/storage.yml +34 -0
  72. data/apps/rails-api/config.ru +6 -0
  73. data/apps/rails-api/db/cable_schema.rb +11 -0
  74. data/apps/rails-api/db/cache_schema.rb +14 -0
  75. data/apps/rails-api/db/queue_schema.rb +129 -0
  76. data/apps/rails-api/db/seeds.rb +9 -0
  77. data/apps/rails-api/public/robots.txt +1 -0
  78. data/apps/rails-api/test/controllers/home_controller_test.rb +7 -0
  79. data/apps/rails-api/test/test_helper.rb +15 -0
  80. data/apps/rails-full/.dockerignore +47 -0
  81. data/apps/rails-full/.gitattributes +9 -0
  82. data/apps/rails-full/.gitignore +34 -0
  83. data/apps/rails-full/.kamal/hooks/docker-setup.sample +3 -0
  84. data/apps/rails-full/.kamal/hooks/post-deploy.sample +14 -0
  85. data/apps/rails-full/.kamal/hooks/post-proxy-reboot.sample +3 -0
  86. data/apps/rails-full/.kamal/hooks/pre-build.sample +51 -0
  87. data/apps/rails-full/.kamal/hooks/pre-connect.sample +47 -0
  88. data/apps/rails-full/.kamal/hooks/pre-deploy.sample +109 -0
  89. data/apps/rails-full/.kamal/hooks/pre-proxy-reboot.sample +3 -0
  90. data/apps/rails-full/.kamal/secrets +17 -0
  91. data/apps/rails-full/.rubocop.yml +8 -0
  92. data/apps/rails-full/.ruby-version +1 -0
  93. data/apps/rails-full/Dockerfile +72 -0
  94. data/apps/rails-full/Gemfile +70 -0
  95. data/apps/rails-full/Gemfile.lock +429 -0
  96. data/apps/rails-full/README.md +24 -0
  97. data/apps/rails-full/Rakefile +6 -0
  98. data/apps/rails-full/app/assets/stylesheets/application.css +10 -0
  99. data/apps/rails-full/app/controllers/application_controller.rb +6 -0
  100. data/apps/rails-full/app/controllers/home_controller.rb +11 -0
  101. data/apps/rails-full/app/helpers/application_helper.rb +2 -0
  102. data/apps/rails-full/app/helpers/home_helper.rb +2 -0
  103. data/apps/rails-full/app/javascript/application.js +3 -0
  104. data/apps/rails-full/app/javascript/controllers/application.js +9 -0
  105. data/apps/rails-full/app/javascript/controllers/hello_controller.js +7 -0
  106. data/apps/rails-full/app/javascript/controllers/index.js +4 -0
  107. data/apps/rails-full/app/jobs/application_job.rb +7 -0
  108. data/apps/rails-full/app/mailers/application_mailer.rb +4 -0
  109. data/apps/rails-full/app/models/application_record.rb +3 -0
  110. data/apps/rails-full/app/views/home/index.html.erb +7 -0
  111. data/apps/rails-full/app/views/layouts/application.html.erb +60 -0
  112. data/apps/rails-full/app/views/layouts/mailer.html.erb +13 -0
  113. data/apps/rails-full/app/views/layouts/mailer.text.erb +1 -0
  114. data/apps/rails-full/app/views/pwa/manifest.json.erb +22 -0
  115. data/apps/rails-full/app/views/pwa/service-worker.js +26 -0
  116. data/apps/rails-full/bin/brakeman +7 -0
  117. data/apps/rails-full/bin/bundle +109 -0
  118. data/apps/rails-full/bin/dev +2 -0
  119. data/apps/rails-full/bin/docker-entrypoint +14 -0
  120. data/apps/rails-full/bin/importmap +4 -0
  121. data/apps/rails-full/bin/jobs +6 -0
  122. data/apps/rails-full/bin/kamal +27 -0
  123. data/apps/rails-full/bin/rails +4 -0
  124. data/apps/rails-full/bin/rake +4 -0
  125. data/apps/rails-full/bin/rubocop +8 -0
  126. data/apps/rails-full/bin/setup +34 -0
  127. data/apps/rails-full/bin/thrust +5 -0
  128. data/apps/rails-full/config/application.rb +31 -0
  129. data/apps/rails-full/config/boot.rb +4 -0
  130. data/apps/rails-full/config/cable.yml +17 -0
  131. data/apps/rails-full/config/cache.yml +16 -0
  132. data/apps/rails-full/config/credentials.yml.enc +1 -0
  133. data/apps/rails-full/config/database.yml +41 -0
  134. data/apps/rails-full/config/deploy.yml +116 -0
  135. data/apps/rails-full/config/environment.rb +5 -0
  136. data/apps/rails-full/config/environments/development.rb +72 -0
  137. data/apps/rails-full/config/environments/production.rb +91 -0
  138. data/apps/rails-full/config/environments/test.rb +53 -0
  139. data/apps/rails-full/config/importmap.rb +7 -0
  140. data/apps/rails-full/config/initializers/assets.rb +7 -0
  141. data/apps/rails-full/config/initializers/clerk.rb +4 -0
  142. data/apps/rails-full/config/initializers/content_security_policy.rb +25 -0
  143. data/apps/rails-full/config/initializers/filter_parameter_logging.rb +8 -0
  144. data/apps/rails-full/config/initializers/inflections.rb +16 -0
  145. data/apps/rails-full/config/locales/en.yml +31 -0
  146. data/apps/rails-full/config/puma.rb +41 -0
  147. data/apps/rails-full/config/queue.yml +18 -0
  148. data/apps/rails-full/config/recurring.yml +10 -0
  149. data/apps/rails-full/config/routes.rb +15 -0
  150. data/apps/rails-full/config/storage.yml +34 -0
  151. data/apps/rails-full/config.ru +6 -0
  152. data/apps/rails-full/db/cable_schema.rb +11 -0
  153. data/apps/rails-full/db/cache_schema.rb +14 -0
  154. data/apps/rails-full/db/queue_schema.rb +129 -0
  155. data/apps/rails-full/db/seeds.rb +9 -0
  156. data/apps/rails-full/public/400.html +114 -0
  157. data/apps/rails-full/public/404.html +114 -0
  158. data/apps/rails-full/public/406-unsupported-browser.html +114 -0
  159. data/apps/rails-full/public/422.html +114 -0
  160. data/apps/rails-full/public/500.html +114 -0
  161. data/apps/rails-full/public/icon.png +0 -0
  162. data/apps/rails-full/public/icon.svg +3 -0
  163. data/apps/rails-full/public/robots.txt +1 -0
  164. data/apps/rails-full/test/application_system_test_case.rb +5 -0
  165. data/apps/rails-full/test/controllers/home_controller_test.rb +7 -0
  166. data/apps/rails-full/test/test_helper.rb +15 -0
  167. data/apps/sinatra/app.rb +29 -0
  168. data/apps/sinatra/config.ru +2 -0
  169. data/apps/sinatra/views/index.erb +44 -0
  170. data/clerk-sdk-ruby.gemspec +2 -1
  171. data/lib/clerk/authenticatable.rb +14 -79
  172. data/lib/clerk/authenticate_context.rb +164 -181
  173. data/lib/clerk/authenticate_request.rb +238 -230
  174. data/lib/clerk/configuration.rb +78 -0
  175. data/lib/clerk/constants.rb +68 -46
  176. data/lib/clerk/error.rb +17 -0
  177. data/lib/clerk/jwks_cache.rb +27 -22
  178. data/lib/clerk/proxy.rb +135 -0
  179. data/lib/clerk/rack.rb +2 -0
  180. data/lib/clerk/rack_middleware.rb +88 -73
  181. data/lib/clerk/rails.rb +3 -0
  182. data/lib/clerk/railtie.rb +7 -6
  183. data/lib/clerk/sdk.rb +46 -156
  184. data/lib/clerk/sinatra.rb +52 -0
  185. data/lib/clerk/utils.rb +52 -6
  186. data/lib/clerk/version.rb +1 -1
  187. data/lib/clerk.rb +15 -51
  188. metadata +187 -25
  189. data/CODEOWNERS +0 -1
  190. data/lib/clerk/errors.rb +0 -22
  191. data/lib/clerk/rack_middleware_v2.rb +0 -167
  192. data/lib/clerk/resources/allowlist.rb +0 -16
  193. data/lib/clerk/resources/allowlist_identifiers.rb +0 -16
  194. data/lib/clerk/resources/clients.rb +0 -23
  195. data/lib/clerk/resources/email_addresses.rb +0 -17
  196. data/lib/clerk/resources/emails.rb +0 -16
  197. data/lib/clerk/resources/jwks.rb +0 -18
  198. data/lib/clerk/resources/organizations.rb +0 -73
  199. data/lib/clerk/resources/phone_numbers.rb +0 -17
  200. data/lib/clerk/resources/plural_resource.rb +0 -38
  201. data/lib/clerk/resources/sessions.rb +0 -26
  202. data/lib/clerk/resources/singular_resource.rb +0 -14
  203. data/lib/clerk/resources/users.rb +0 -37
  204. data/lib/clerk/resources.rb +0 -10
data/lib/clerk/sdk.rb CHANGED
@@ -1,34 +1,18 @@
1
- # frozen_string_literal: true
2
-
3
- require "faraday"
4
- require "logger"
5
- require "net/http"
6
- require "json"
1
+ require "clerk-http-client"
2
+ require "clerk/jwks_cache"
3
+ require "clerk/version"
7
4
  require "jwt"
8
- require "concurrent-ruby"
9
-
10
- require_relative "resources/allowlist_identifiers"
11
- require_relative "resources/allowlist"
12
- require_relative "resources/clients"
13
- require_relative "resources/email_addresses"
14
- require_relative "resources/emails"
15
- require_relative "resources/organizations"
16
- require_relative "resources/phone_numbers"
17
- require_relative "resources/sessions"
18
- require_relative "resources/users"
19
- require_relative "resources/jwks"
20
- require_relative "errors"
21
- require_relative "jwks_cache"
22
5
 
23
6
  module Clerk
24
- class SDK
7
+ class SDK < ClerkHttpClient::SDK
8
+ # TODO: Move to constants?
25
9
  DEFAULT_HEADERS = {
26
- "User-Agent" => "Clerk/#{Clerk::VERSION}; Faraday/#{Faraday::VERSION}; Ruby/#{RUBY_VERSION}",
27
- "X-Clerk-SDK" => "ruby/#{Clerk::VERSION}"
10
+ "User-Agent": "Clerk/#{Clerk::VERSION}; Faraday/#{Faraday::VERSION}; Ruby/#{RUBY_VERSION}",
11
+ "X-Clerk-SDK": "ruby/#{Clerk::VERSION}" # TODO: Add framework identifier
28
12
  }
29
13
 
30
14
  # How often (in seconds) should JWKs be refreshed
31
- JWKS_CACHE_LIFETIME = 3600 # 1 hour
15
+ JWKS_CACHE_LIFETIME = 3600 # 1 hour / TODO: Move to constants?
32
16
 
33
17
  @@jwks_cache = JWKSCache.new(JWKS_CACHE_LIFETIME)
34
18
 
@@ -36,151 +20,57 @@ module Clerk
36
20
  @@jwks_cache
37
21
  end
38
22
 
39
- def initialize(api_key: nil, base_url: nil, logger: nil, ssl_verify: true,
40
- connection: nil)
41
- if connection
42
- # Inject a Faraday::Connection for testing or full control over Faraday
43
- @conn = connection
44
- return
45
- else
46
- base_url = base_url || Clerk.configuration.base_url
47
- base_uri = if !base_url.end_with?("/")
48
- URI("#{base_url}/")
49
- else
50
- URI(base_url)
51
- end
52
-
53
- api_key ||= Clerk.configuration.api_key
54
-
55
- if Faraday::VERSION.to_i >= 2 && api_key.nil?
56
- api_key = -> { raise ArgumentError, "Clerk secret key is not set" }
57
- end
58
-
59
- logger = logger || Clerk.configuration.logger
60
- @conn = Faraday.new(
61
- url: base_uri, headers: DEFAULT_HEADERS, ssl: {verify: ssl_verify}
62
- ) do |f|
63
- f.request :url_encoded
64
- f.request :authorization, "Bearer", api_key
65
- if logger
66
- f.response :logger, logger do |l|
67
- l.filter(/(Authorization: "Bearer) (\w+)/, '\1 [SECRET]')
68
- end
69
- end
70
- end
71
- end
72
- end
73
-
74
- def request(method, path, query: [], body: nil, timeout: nil)
75
- response = case method
76
- when :get
77
- @conn.get(path, query) do |req|
78
- req.options.timeout = timeout if timeout
79
- end
80
- when :post
81
- @conn.post(path, body) do |req|
82
- req.body = body.to_json
83
- req.headers[:content_type] = "application/json"
84
- req.options.timeout = timeout if timeout
85
- end
86
- when :patch
87
- @conn.patch(path, body) do |req|
88
- req.body = body.to_json
89
- req.headers[:content_type] = "application/json"
90
- req.options.timeout = timeout if timeout
91
- end
92
- when :delete
93
- @conn.delete(path) do |req|
94
- req.options.timeout = timeout if timeout
95
- end
96
- end
97
-
98
- body = if response[CONTENT_TYPE_HEADER] == "application/json"
99
- JSON.parse(response.body)
100
- else
101
- response.body
102
- end
103
-
104
- if response.success?
105
- body
106
- else
107
- klass = case body.dig("errors", 0, "code")
108
- when "cookie_invalid", "client_not_found", "resource_not_found"
109
- Errors::Authentication
110
- else
111
- Errors::Fatal
112
- end
113
- raise klass.new(body, status: response.status)
114
- end
115
- end
116
-
117
- def allowlist_identifiers
118
- Resources::AllowlistIdentifiers.new(self)
119
- end
120
-
121
- def allowlist
122
- Resources::Allowlist.new(self)
123
- end
124
-
125
- def clients
126
- Resources::Clients.new(self)
127
- end
128
-
129
- def email_addresses
130
- Resources::EmailAddresses.new(self)
131
- end
132
-
133
- def emails
134
- Resources::Emails.new(self)
135
- end
136
-
137
- def organizations
138
- Resources::Organizations.new(self)
139
- end
140
-
141
- def phone_numbers
142
- Resources::PhoneNumbers.new(self)
143
- end
144
-
145
- def sessions
146
- Resources::Sessions.new(self)
147
- end
148
-
149
- def users
150
- Resources::Users.new(self)
151
- end
152
-
153
- def jwks
154
- Resources::JWKS.new(self)
155
- end
156
-
157
- # Returns the decoded JWT payload without verifying if the signature is
158
- # valid.
23
+ # Returns the decoded JWT payload without verifying if the signature is valid.
159
24
  #
160
- # WARNING: This will not verify whether the signature is valid. You
161
- # should not use this for untrusted messages! You most likely want to use
162
- # verify_token.
25
+ # WARNING: This will not verify whether the signature is valid. You should not
26
+ # use this for untrusted messages! You most likely want to use `verify_token`.
163
27
  def decode_token(token)
164
28
  JWT.decode(token, nil, false).first
165
29
  end
166
30
 
167
- # Decode the JWT and verify it's valid (verify claims, signature etc.) using
168
- # the provided algorithms.
31
+ # Decode the JWT and verify it's valid (verify claims, signature etc.) using the provided algorithms.
169
32
  #
170
- # JWKS are cached for JWKS_CACHE_LIFETIME seconds, in order to avoid
171
- # unecessary roundtrips. In order to invalidate the cache, pass
172
- # `force_refresh_jwks: true`.
33
+ # JWKS are cached for JWKS_CACHE_LIFETIME seconds, in order to avoid unecessary roundtrips.
34
+ # In order to invalidate the cache, pass `force_refresh_jwks: true`.
173
35
  #
174
- # A timeout for the request to the JWKs endpoint can be set with the
175
- # `timeout` argument.
176
- def verify_token(token, force_refresh_jwks: false, algorithms: ['RS256'], timeout: 5)
36
+ # A timeout for the request to the JWKs endpoint can be set with the `timeout` argument.
37
+ def verify_token(token, force_refresh_jwks: false, algorithms: ["RS256"], timeout: 5)
177
38
  jwk_loader = ->(options) do
178
39
  # JWT.decode requires that the 'keys' key in the Hash is a symbol (as
179
40
  # opposed to a string which our SDK returns by default)
180
- { keys: SDK.jwks_cache.fetch(self, kid_not_found: (options[:invalidate] || options[:kid_not_found]), force_refresh: force_refresh_jwks) }
41
+ {keys: SDK.jwks_cache.fetch(self, kid_not_found: options[:invalidate] || options[:kid_not_found], force_refresh: force_refresh_jwks)}
181
42
  end
182
43
 
183
44
  JWT.decode(token, nil, true, algorithms: algorithms, exp_leeway: timeout, jwks: jwk_loader).first
184
45
  end
46
+
47
+ private
48
+
49
+ # TODO: Temporary solution until generators are improved
50
+ HTTP_CLIENT_CUSTOM_MAPPING = {
51
+ allowlist: "AllowListBlockList",
52
+ blocklist: "AllowListBlockList",
53
+ email_sms_templates: "EmailSMSTemplates",
54
+ oauth_applications: "OAuthApplications",
55
+ jwt_templates: "JWTTemplates",
56
+ redirect_urls: "RedirectURLs",
57
+ saml_connections: "SAMLConnections"
58
+ }
59
+
60
+ def generate_const_name(method_name)
61
+ "#{HTTP_CLIENT_CUSTOM_MAPPING[method_name] || Utils.camel_case(method_name)}Api"
62
+ end
63
+
64
+ def respond_to_missing?(method_name, include_private = false)
65
+ ClerkHttpClient.const_get(generate_const_name(method_name)).respond_to?(:new)
66
+ rescue NameError
67
+ false
68
+ end
69
+
70
+ def method_missing(method_name, *arguments)
71
+ ClerkHttpClient.const_get(generate_const_name(method_name)).new
72
+ rescue NameError
73
+ raise NoMethodError, "undefined method `#{method_name}` for #{self.class.name}"
74
+ end
185
75
  end
186
76
  end
@@ -0,0 +1,52 @@
1
+ require "sinatra/base"
2
+ require "clerk/rack"
3
+
4
+ module Sinatra
5
+ module Clerk
6
+ module Helpers
7
+ def clerk
8
+ env["clerk"]
9
+ end
10
+
11
+ def require_reverification!(preset = ::Clerk::StepUp::Preset::STRICT, &block)
12
+ clerk.user_require_reverification!(preset) do
13
+ return yield(preset) if block_given?
14
+ render_reverification!(preset)
15
+ end
16
+ end
17
+
18
+ def render_reverification!(preset = nil)
19
+ halt 403, ::Clerk::StepUp::Reverification.error_payload(preset).to_json
20
+ end
21
+
22
+ def clerk_sdk
23
+ @@sdk ||= ::Clerk::SDK.new
24
+ end
25
+ end
26
+
27
+ def self.registered(app)
28
+ app.helpers Clerk::Helpers
29
+ app.use ::Clerk::Rack::Middleware
30
+
31
+ app.set(:auth) do |active|
32
+ condition do
33
+ redirect clerk.sign_in_url if active && !clerk.session
34
+ end
35
+ end
36
+
37
+ app.set(:reverify) do |preset|
38
+ condition do
39
+ if preset === true
40
+ preset = ::Clerk::StepUp::Preset::STRICT
41
+ end
42
+
43
+ if preset
44
+ require_reverification!(preset)
45
+ end
46
+ end
47
+ end
48
+ end
49
+ end
50
+
51
+ register Clerk
52
+ end
data/lib/clerk/utils.rb CHANGED
@@ -1,11 +1,57 @@
1
+ # frozen_string_literal: true
2
+
3
+ require "base64"
4
+
1
5
  module Clerk
2
6
  module Utils
3
- module_function
4
- def camelize(term)
5
- string = term.to_s
6
- string = string.sub(/^[a-z\d]*/) { match.capitalize }
7
- string.gsub!(/(?:_|(\/))([a-z\d]*)/) { "#{$1}#{$2.capitalize}" }
8
- string
7
+ class << self
8
+ def camel_case(str)
9
+ str = str.to_s
10
+ return str if str !~ /_/ && str =~ /[A-Z]+.*/
11
+ str.split("_").map { |s| s.capitalize }.join
12
+ end
13
+
14
+ def decode_publishable_key(publishable_key)
15
+ Base64.decode64(publishable_key.split("_")[2].to_s)
16
+ end
17
+
18
+ def filter_routes(routes)
19
+ filtered_routes = {}
20
+ filtered_wildcard_routes = []
21
+
22
+ routes.each do |route|
23
+ route = route.strip
24
+
25
+ if route.end_with?("/*")
26
+ filtered_wildcard_routes << route[0..-2]
27
+ else
28
+ filtered_routes[route] = true
29
+ end
30
+ end
31
+
32
+ filtered_wildcard_routes.uniq!
33
+
34
+ [filtered_routes, filtered_wildcard_routes]
35
+ end
36
+
37
+ def retrieve_from_query_string(url, key)
38
+ ::Rack::Utils.parse_query(url.query)[key]
39
+ end
40
+
41
+ def valid_publishable_key?(publishable_key)
42
+ raise ArgumentError, "publishable_key must be a string" unless publishable_key.is_a?(String)
43
+
44
+ key = publishable_key.to_s
45
+ valid_publishable_key_prefix?(key) && valid_publishable_key_postfix?(key)
46
+ end
47
+
48
+ def valid_publishable_key_postfix?(publishable_key)
49
+ decode_publishable_key(publishable_key).end_with?("$")
50
+ end
51
+
52
+ def valid_publishable_key_prefix?(publishable_key)
53
+ publishable_key.start_with?("pk_live_", "pk_test_")
54
+ end
9
55
  end
10
56
  end
11
57
  end
data/lib/clerk/version.rb CHANGED
@@ -1,5 +1,5 @@
1
1
  # frozen_string_literal: true
2
2
 
3
3
  module Clerk
4
- VERSION = "4.0.0.beta3"
4
+ VERSION = "4.0.0.beta4"
5
5
  end
data/lib/clerk.rb CHANGED
@@ -1,63 +1,27 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- require_relative "clerk/version"
4
- require_relative "clerk/constants"
5
- require_relative "clerk/sdk"
3
+ require "clerk/configuration"
4
+ require "clerk/constants"
5
+ require "clerk/error"
6
+ require "clerk/sdk"
7
+ require "clerk/version"
8
+
9
+ if defined?(::Rails)
10
+ require "clerk/rails"
11
+ end
6
12
 
7
13
  module Clerk
8
14
  class << self
9
15
  def configure
10
- yield(configuration)
16
+ if block_given?
17
+ yield(configuration)
18
+ else
19
+ configuration
20
+ end
11
21
  end
12
22
 
13
23
  def configuration
14
- @configuration ||= Config.new
15
- end
16
- end
17
-
18
- class Config
19
- PRODUCTION_BASE_URL = "https://api.clerk.dev/v1/".freeze
20
- attr_accessor :api_key, :base_url, :publishable_key, :logger, :middleware_cache_store
21
-
22
- # An array of route paths on which the middleware will not execute.
23
- #
24
- # Only request paths that match _exactly_ one of the routes will be skipped.
25
- # As a special case, if a route ends with '/*', then all request paths that
26
- # match the route's prefix will be skipped.
27
- #
28
- # For example, given the following configuration:
29
- #
30
- # excluded_routes = ["/foo", "/bar/*"]
31
- #
32
- # the following requests will be excluded:
33
- #
34
- # - /foo
35
- # - /bar/baz
36
- # - /bar/abc/xyz
37
- #
38
- # while the following requests will NOT be excluded:
39
- #
40
- # - /foo/bar
41
- # - /bar
42
- #
43
- attr_accessor :excluded_routes
44
-
45
- def initialize
46
- @base_url = ENV.fetch("CLERK_API_BASE", PRODUCTION_BASE_URL)
47
- @api_key = ENV["CLERK_API_KEY"]
48
-
49
- secret_key = ENV["CLERK_SECRET_KEY"]
50
- if secret_key && !secret_key.empty?
51
- @api_key = secret_key
52
- end
53
-
54
- @publishable_key = ENV["CLERK_PUBLISHABLE_KEY"]
55
-
56
- @excluded_routes = []
24
+ @configuration ||= Clerk::Configuration.default
57
25
  end
58
26
  end
59
27
  end
60
-
61
- if defined?(::Rails)
62
- require_relative "clerk/railtie"
63
- end