clerk-sdk-ruby 4.0.0.beta2 → 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 +25 -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 -179
  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 -157
  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 -19
  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,35 +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/users"
20
- require_relative "resources/jwks"
21
- require_relative "errors"
22
- require_relative "jwks_cache"
23
5
 
24
6
  module Clerk
25
- class SDK
7
+ class SDK < ClerkHttpClient::SDK
8
+ # TODO: Move to constants?
26
9
  DEFAULT_HEADERS = {
27
- "User-Agent" => "Clerk/#{Clerk::VERSION}; Faraday/#{Faraday::VERSION}; Ruby/#{RUBY_VERSION}",
28
- "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
29
12
  }
30
13
 
31
14
  # How often (in seconds) should JWKs be refreshed
32
- JWKS_CACHE_LIFETIME = 3600 # 1 hour
15
+ JWKS_CACHE_LIFETIME = 3600 # 1 hour / TODO: Move to constants?
33
16
 
34
17
  @@jwks_cache = JWKSCache.new(JWKS_CACHE_LIFETIME)
35
18
 
@@ -37,151 +20,57 @@ module Clerk
37
20
  @@jwks_cache
38
21
  end
39
22
 
40
- def initialize(api_key: nil, base_url: nil, logger: nil, ssl_verify: true,
41
- connection: nil)
42
- if connection
43
- # Inject a Faraday::Connection for testing or full control over Faraday
44
- @conn = connection
45
- return
46
- else
47
- base_url = base_url || Clerk.configuration.base_url
48
- base_uri = if !base_url.end_with?("/")
49
- URI("#{base_url}/")
50
- else
51
- URI(base_url)
52
- end
53
-
54
- api_key ||= Clerk.configuration.api_key
55
-
56
- if Faraday::VERSION.to_i >= 2 && api_key.nil?
57
- api_key = -> { raise ArgumentError, "Clerk secret key is not set" }
58
- end
59
-
60
- logger = logger || Clerk.configuration.logger
61
- @conn = Faraday.new(
62
- url: base_uri, headers: DEFAULT_HEADERS, ssl: {verify: ssl_verify}
63
- ) do |f|
64
- f.request :url_encoded
65
- f.request :authorization, "Bearer", api_key
66
- if logger
67
- f.response :logger, logger do |l|
68
- l.filter(/(Authorization: "Bearer) (\w+)/, '\1 [SECRET]')
69
- end
70
- end
71
- end
72
- end
73
- end
74
-
75
- def request(method, path, query: [], body: nil, timeout: nil)
76
- response = case method
77
- when :get
78
- @conn.get(path, query) do |req|
79
- req.options.timeout = timeout if timeout
80
- end
81
- when :post
82
- @conn.post(path, body) do |req|
83
- req.body = body.to_json
84
- req.headers[:content_type] = "application/json"
85
- req.options.timeout = timeout if timeout
86
- end
87
- when :patch
88
- @conn.patch(path, body) do |req|
89
- req.body = body.to_json
90
- req.headers[:content_type] = "application/json"
91
- req.options.timeout = timeout if timeout
92
- end
93
- when :delete
94
- @conn.delete(path) do |req|
95
- req.options.timeout = timeout if timeout
96
- end
97
- end
98
-
99
- body = if response[CONTENT_TYPE_HEADER] == "application/json"
100
- JSON.parse(response.body)
101
- else
102
- response.body
103
- end
104
-
105
- if response.success?
106
- body
107
- else
108
- klass = case body.dig("errors", 0, "code")
109
- when "cookie_invalid", "client_not_found", "resource_not_found"
110
- Errors::Authentication
111
- else
112
- Errors::Fatal
113
- end
114
- raise klass.new(body, status: response.status)
115
- end
116
- end
117
-
118
- def allowlist_identifiers
119
- Resources::AllowlistIdentifiers.new(self)
120
- end
121
-
122
- def allowlist
123
- Resources::Allowlist.new(self)
124
- end
125
-
126
- def clients
127
- Resources::Clients.new(self)
128
- end
129
-
130
- def email_addresses
131
- Resources::EmailAddresses.new(self)
132
- end
133
-
134
- def emails
135
- Resources::Emails.new(self)
136
- end
137
-
138
- def organizations
139
- Resources::Organizations.new(self)
140
- end
141
-
142
- def phone_numbers
143
- Resources::PhoneNumbers.new(self)
144
- end
145
-
146
- def sessions
147
- Resources::Sessions.new(self)
148
- end
149
-
150
- def users
151
- Resources::Users.new(self)
152
- end
153
-
154
- def jwks
155
- Resources::JWKS.new(self)
156
- end
157
-
158
- # Returns the decoded JWT payload without verifying if the signature is
159
- # valid.
23
+ # Returns the decoded JWT payload without verifying if the signature is valid.
160
24
  #
161
- # WARNING: This will not verify whether the signature is valid. You
162
- # should not use this for untrusted messages! You most likely want to use
163
- # 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`.
164
27
  def decode_token(token)
165
28
  JWT.decode(token, nil, false).first
166
29
  end
167
30
 
168
- # Decode the JWT and verify it's valid (verify claims, signature etc.) using
169
- # the provided algorithms.
31
+ # Decode the JWT and verify it's valid (verify claims, signature etc.) using the provided algorithms.
170
32
  #
171
- # JWKS are cached for JWKS_CACHE_LIFETIME seconds, in order to avoid
172
- # unecessary roundtrips. In order to invalidate the cache, pass
173
- # `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`.
174
35
  #
175
- # A timeout for the request to the JWKs endpoint can be set with the
176
- # `timeout` argument.
177
- 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)
178
38
  jwk_loader = ->(options) do
179
39
  # JWT.decode requires that the 'keys' key in the Hash is a symbol (as
180
40
  # opposed to a string which our SDK returns by default)
181
- { 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)}
182
42
  end
183
43
 
184
44
  JWT.decode(token, nil, true, algorithms: algorithms, exp_leeway: timeout, jwks: jwk_loader).first
185
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
186
75
  end
187
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.beta2"
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.fetch("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