clavis 0.7.1

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 (74) hide show
  1. checksums.yaml +7 -0
  2. data/.actrc +4 -0
  3. data/.cursor/rules/ruby-gem.mdc +49 -0
  4. data/.gemignore +6 -0
  5. data/.rspec +3 -0
  6. data/.rubocop.yml +88 -0
  7. data/.vscode/settings.json +22 -0
  8. data/CHANGELOG.md +127 -0
  9. data/CODE_OF_CONDUCT.md +3 -0
  10. data/LICENSE.txt +21 -0
  11. data/README.md +838 -0
  12. data/Rakefile +341 -0
  13. data/UPGRADE.md +57 -0
  14. data/app/assets/stylesheets/clavis.css +133 -0
  15. data/app/controllers/clavis/auth_controller.rb +133 -0
  16. data/config/database.yml +16 -0
  17. data/config/routes.rb +49 -0
  18. data/docs/SECURITY.md +340 -0
  19. data/docs/TESTING.md +78 -0
  20. data/docs/integration.md +272 -0
  21. data/error_handling.md +355 -0
  22. data/file_structure.md +221 -0
  23. data/gemfiles/rails_80.gemfile +17 -0
  24. data/gemfiles/rails_80.gemfile.lock +286 -0
  25. data/implementation_plan.md +523 -0
  26. data/lib/clavis/configuration.rb +196 -0
  27. data/lib/clavis/controllers/concerns/authentication.rb +232 -0
  28. data/lib/clavis/controllers/concerns/session_management.rb +117 -0
  29. data/lib/clavis/engine.rb +191 -0
  30. data/lib/clavis/errors.rb +205 -0
  31. data/lib/clavis/logging.rb +116 -0
  32. data/lib/clavis/models/concerns/oauth_authenticatable.rb +169 -0
  33. data/lib/clavis/oauth_identity.rb +174 -0
  34. data/lib/clavis/providers/apple.rb +135 -0
  35. data/lib/clavis/providers/base.rb +432 -0
  36. data/lib/clavis/providers/custom_provider_example.rb +57 -0
  37. data/lib/clavis/providers/facebook.rb +84 -0
  38. data/lib/clavis/providers/generic.rb +63 -0
  39. data/lib/clavis/providers/github.rb +87 -0
  40. data/lib/clavis/providers/google.rb +98 -0
  41. data/lib/clavis/providers/microsoft.rb +57 -0
  42. data/lib/clavis/security/csrf_protection.rb +79 -0
  43. data/lib/clavis/security/https_enforcer.rb +90 -0
  44. data/lib/clavis/security/input_validator.rb +192 -0
  45. data/lib/clavis/security/parameter_filter.rb +64 -0
  46. data/lib/clavis/security/rate_limiter.rb +109 -0
  47. data/lib/clavis/security/redirect_uri_validator.rb +124 -0
  48. data/lib/clavis/security/session_manager.rb +220 -0
  49. data/lib/clavis/security/token_storage.rb +114 -0
  50. data/lib/clavis/user_info_normalizer.rb +74 -0
  51. data/lib/clavis/utils/nonce_store.rb +14 -0
  52. data/lib/clavis/utils/secure_token.rb +17 -0
  53. data/lib/clavis/utils/state_store.rb +18 -0
  54. data/lib/clavis/version.rb +6 -0
  55. data/lib/clavis/view_helpers.rb +260 -0
  56. data/lib/clavis.rb +132 -0
  57. data/lib/generators/clavis/controller/controller_generator.rb +48 -0
  58. data/lib/generators/clavis/controller/templates/controller.rb.tt +137 -0
  59. data/lib/generators/clavis/controller/templates/views/login.html.erb.tt +145 -0
  60. data/lib/generators/clavis/install_generator.rb +182 -0
  61. data/lib/generators/clavis/templates/add_oauth_to_users.rb +28 -0
  62. data/lib/generators/clavis/templates/clavis.css +133 -0
  63. data/lib/generators/clavis/templates/initializer.rb +47 -0
  64. data/lib/generators/clavis/templates/initializer.rb.tt +76 -0
  65. data/lib/generators/clavis/templates/migration.rb +18 -0
  66. data/lib/generators/clavis/templates/migration.rb.tt +16 -0
  67. data/lib/generators/clavis/user_method/user_method_generator.rb +219 -0
  68. data/lib/tasks/provider_verification.rake +77 -0
  69. data/llms.md +487 -0
  70. data/log/development.log +20 -0
  71. data/log/test.log +0 -0
  72. data/sig/clavis.rbs +4 -0
  73. data/testing_plan.md +710 -0
  74. metadata +258 -0
data/Rakefile ADDED
@@ -0,0 +1,341 @@
1
+ # frozen_string_literal: true
2
+
3
+ require "bundler/gem_tasks"
4
+ require "rspec/core/rake_task"
5
+
6
+ RSpec::Core::RakeTask.new(:spec)
7
+
8
+ task default: :ci
9
+
10
+ # Helper method to safely run a Rails command and fix known issues
11
+ def safe_rails_command(rails_app_dir, command)
12
+ Dir.chdir(rails_app_dir) do
13
+ # Fix bootsnap issue if needed
14
+ fix_bootsnap_issue
15
+
16
+ # Try running the command
17
+ puts "Running: #{command}"
18
+ result = system(command)
19
+
20
+ # If it fails, check for asset configuration issues
21
+ unless result
22
+ puts "Command failed, checking for asset configuration issues..."
23
+
24
+ # Try to fix assets.rb if it exists
25
+ assets_initializer = "config/initializers/assets.rb"
26
+ if File.exist?(assets_initializer)
27
+ puts "Fixing assets configuration..."
28
+ # Comment out the entire file to prevent issues
29
+ content = File.read(assets_initializer)
30
+ fixed_content = "# Assets configuration disabled for testing\n# Original content:\n# #{content.gsub("\n",
31
+ "\n# ")}"
32
+ File.write(assets_initializer, fixed_content)
33
+
34
+ # Try the command again
35
+ puts "Retrying: #{command}"
36
+ result = system(command)
37
+ end
38
+ end
39
+
40
+ return result
41
+ end
42
+ end
43
+
44
+ # Helper to fix bootsnap issues and other dependencies in the Rails app
45
+ def fix_bootsnap_issue
46
+ boot_rb_path = "config/boot.rb"
47
+ return unless File.exist?(boot_rb_path)
48
+
49
+ content = File.read(boot_rb_path)
50
+ return unless content.include?("bootsnap/setup") && !system("bundle list | grep bootsnap")
51
+
52
+ puts "Fixing bootsnap issue..."
53
+
54
+ # Option 1: Add bootsnap to Gemfile
55
+ unless File.read("Gemfile").include?("bootsnap")
56
+ puts "Adding bootsnap to Gemfile..."
57
+ File.open("Gemfile", "a") do |f|
58
+ f.puts "\n# Reduces boot times through caching; required in config/boot.rb"
59
+ f.puts "gem \"bootsnap\", require: false"
60
+ end
61
+ system("bundle install")
62
+ end
63
+
64
+ # Option 2 (fallback): Comment out bootsnap line in boot.rb
65
+ return if system("bundle list | grep bootsnap")
66
+
67
+ puts "Commenting out bootsnap in boot.rb..."
68
+ modified_content = content.gsub(
69
+ 'require "bootsnap/setup"',
70
+ '# require "bootsnap/setup" # Commented out to avoid dependency issues'
71
+ )
72
+ File.write(boot_rb_path, modified_content)
73
+ end
74
+
75
+ # Helper to test loading the clavis gem
76
+ def test_clavis_loading(rails_app_dir)
77
+ Dir.chdir(rails_app_dir) do
78
+ puts "Testing if the clavis gem can be loaded..."
79
+ test_code = "require 'clavis'; puts 'Clavis loaded successfully! Version: ' + Clavis::VERSION"
80
+ system("bundle exec ruby -e \"#{test_code}\"")
81
+ end
82
+ end
83
+
84
+ # Define an environment task for Rails-dependent tasks
85
+ task :environment do
86
+ # This is a no-op task to satisfy dependencies
87
+ # Rails would normally provide this task
88
+ end
89
+
90
+ # Task to run Rails-dependent tests
91
+ # rubocop:disable Metrics/BlockLength
92
+ namespace :test do
93
+ desc "Run Rails controller tests"
94
+ task controllers: :environment do
95
+ ENV["RAILS_ENV"] = "test"
96
+ system("bundle exec rspec spec/clavis/controllers/*_spec.rb")
97
+ end
98
+
99
+ desc "Run Rails integration tests"
100
+ task integration: :environment do
101
+ ENV["RAILS_ENV"] = "test"
102
+ system("bundle exec rspec spec/integration/*_spec.rb")
103
+ end
104
+
105
+ desc "Run Rails generator tests"
106
+ task generators: :environment do
107
+ ENV["RAILS_ENV"] = "test"
108
+ system("bundle exec rspec spec/generators/**/*_generator_spec.rb")
109
+ end
110
+
111
+ desc "Run all Rails-dependent tests"
112
+ task rails: %i[controllers integration generators]
113
+
114
+ desc "Test the actual generator in the rails-app directory"
115
+ task real_generator: :bootstrap_rails_app do
116
+ puts "Testing basic functionality in rails-app..."
117
+ rails_app_dir = File.expand_path("rails-app", __dir__)
118
+
119
+ # The bootstrap task ensures the directory exists, so we can remove this check
120
+ # or keep it for extra safety
121
+ unless File.directory?(rails_app_dir)
122
+ puts "Error: rails-app directory not found"
123
+ exit 1
124
+ end
125
+
126
+ Dir.chdir(rails_app_dir) do
127
+ puts "Adding bcrypt"
128
+ unless system("bundle add bcrypt")
129
+ puts "Error: Failed to add bcrypt"
130
+ exit 1
131
+ end
132
+
133
+ puts "Adding clavis"
134
+ unless system("bundle add clavis")
135
+ puts "Error: Failed to add clavis"
136
+ exit 1
137
+ end
138
+
139
+ # Install dependencies
140
+ puts "Installing dependencies in rails-app..."
141
+ unless system("bundle install")
142
+ puts "Error: Failed to install dependencies in rails-app"
143
+ exit 1
144
+ end
145
+
146
+ # Test loading the gem
147
+ unless test_clavis_loading(rails_app_dir)
148
+ puts "Error: Failed to load the clavis gem in rails-app"
149
+ exit 1
150
+ end
151
+
152
+ # Run our clavis generator
153
+ puts "Running Clavis generator..."
154
+ unless safe_rails_command(rails_app_dir, "bin/rails generate clavis:install")
155
+ puts "Error: Failed to run Clavis generator"
156
+ exit 1
157
+ end
158
+
159
+ # Run migrations again after the generator
160
+ puts "Running migrations after generator..."
161
+ unless safe_rails_command(rails_app_dir, "bin/rails db:migrate RAILS_ENV=development") &&
162
+ safe_rails_command(rails_app_dir, "bin/rails db:migrate RAILS_ENV=test")
163
+ puts "Error: Failed to run migrations after generator"
164
+ exit 1
165
+ end
166
+
167
+ puts "Checking routes"
168
+ unless safe_rails_command(rails_app_dir, "bin/rails routes")
169
+ puts "Error: Failed to check routes"
170
+ exit 1
171
+ end
172
+
173
+ puts "Generator dependency test in rails-app passed successfully!"
174
+ end
175
+ end
176
+ end
177
+ # Task to run all tests
178
+ desc "Run all tests including Rails controller tests and integration tests"
179
+ task all_tests: [:spec, "test:rails", "test:real_generator"]
180
+
181
+ # Helper to set up rails app authentication
182
+ def setup_rails_authentication(rails_app_dir, gemfile_content)
183
+ Dir.chdir(rails_app_dir) do
184
+ # Add bcrypt first to ensure it's available for has_secure_password
185
+ unless gemfile_content.include?("gem \"bcrypt\"")
186
+ puts "Adding bcrypt to Gemfile..."
187
+ File.open("Gemfile", "a") do |f|
188
+ f.puts "\n# Use Active Model has_secure_password"
189
+ f.puts "gem \"bcrypt\", \"~> 3.1.7\""
190
+ end
191
+ puts "Installing bcrypt..."
192
+ system("bundle install")
193
+ end
194
+
195
+ # Use the Rails 8 built-in authentication generator
196
+ puts "Using Rails built-in authentication generator..."
197
+ system("bin/rails generate authentication")
198
+ end
199
+ end
200
+
201
+ # Task to bootstrap rails-app if it doesn't exist
202
+ desc "Create a minimal Rails application for testing if rails-app doesn't exist"
203
+ task :bootstrap_rails_app do
204
+ rails_app_dir = File.expand_path("rails-app", __dir__)
205
+
206
+ if File.directory?(rails_app_dir)
207
+ puts "Rails application already exists at #{rails_app_dir}"
208
+
209
+ # Even if it exists, make sure the database is migrated
210
+ puts "Ensuring database is migrated..."
211
+ safe_rails_command(rails_app_dir, "bin/rails db:migrate RAILS_ENV=development")
212
+ safe_rails_command(rails_app_dir, "bin/rails db:migrate RAILS_ENV=test")
213
+ else
214
+ puts "Creating Rails application at #{rails_app_dir}..."
215
+
216
+ # Check if Rails is installed
217
+ unless system("gem list rails -i")
218
+ puts "Installing Rails..."
219
+ system("gem install rails")
220
+ end
221
+
222
+ # Create a new Rails application with minimal sensible flags
223
+ system("rails new #{rails_app_dir} --skip-git")
224
+
225
+ # Add the clavis gem to the Gemfile
226
+ gemfile_path = File.join(rails_app_dir, "Gemfile")
227
+
228
+ # Read the current Gemfile
229
+ gemfile_content = File.read(gemfile_path)
230
+
231
+ # Add clavis with path to local directory
232
+ unless gemfile_content.include?("gem \"clavis\"")
233
+ puts "Adding clavis gem to Gemfile..."
234
+ gem_line = "gem \"clavis\", path: \"../.\""
235
+
236
+ # Append to Gemfile
237
+ File.open(gemfile_path, "a") do |f|
238
+ f.puts "\n# Add local Clavis gem for testing"
239
+ f.puts gem_line
240
+ end
241
+ end
242
+
243
+ Dir.chdir(rails_app_dir) do
244
+ puts "Installing dependencies..."
245
+ system("bundle install")
246
+
247
+ # Fix bootsnap issue
248
+ fix_bootsnap_issue
249
+
250
+ # Fix asset configuration if needed
251
+ assets_initializer = "config/initializers/assets.rb"
252
+ if File.exist?(assets_initializer)
253
+ puts "Checking assets configuration..."
254
+ assets_content = File.read(assets_initializer)
255
+ if assets_content.include?("Rails.application.config.assets") &&
256
+ !system("bin/rails runner 'Rails.application.config.respond_to?(:assets)'")
257
+ puts "Fixing assets configuration..."
258
+ fixed_content = assets_content.gsub(/Rails\.application\.config\.assets.*$/,
259
+ "# Assets configuration disabled for testing")
260
+ File.write(assets_initializer, fixed_content)
261
+ end
262
+ end
263
+ end
264
+
265
+ safe_rails_command(rails_app_dir, "bin/rails g model User name:string")
266
+
267
+ # Set up authentication
268
+ setup_rails_authentication(rails_app_dir, gemfile_content)
269
+
270
+ # Use safe_rails_command for migrations
271
+ safe_rails_command(rails_app_dir, "bin/rails db:migrate RAILS_ENV=development")
272
+ safe_rails_command(rails_app_dir, "bin/rails db:migrate RAILS_ENV=test")
273
+
274
+ puts "Rails application created successfully with authentication!"
275
+ end
276
+ end
277
+ # rubocop:enable Metrics/BlockLength
278
+ # Tasks for the dummy Rails app
279
+ namespace :dummy do
280
+ desc "Prepare the dummy Rails app for testing"
281
+ task prepare: :environment do
282
+ app_path = File.expand_path("spec/dummy", __dir__)
283
+
284
+ # Ensure the db directory exists
285
+ FileUtils.mkdir_p(File.join(app_path, "db"))
286
+
287
+ # Set up environment
288
+ ENV["RAILS_ENV"] = "test"
289
+
290
+ # Make sure we can load ActiveRecord
291
+ require "active_record"
292
+
293
+ # Configure ActiveRecord for in-memory SQLite
294
+ ActiveRecord::Base.establish_connection adapter: "sqlite3", database: ":memory:"
295
+
296
+ # Load the schema
297
+ ActiveRecord::Schema.verbose = false
298
+ load File.expand_path("spec/dummy/db/schema.rb", __dir__)
299
+
300
+ Rails.logger.debug "Dummy Rails app prepared successfully!"
301
+ end
302
+
303
+ desc "Run RSpec tests with the dummy Rails app"
304
+ task tests: :prepare do
305
+ ENV["RAILS_ENV"] = "test"
306
+ Rake::Task["spec"].invoke
307
+ end
308
+ end
309
+
310
+ # Task to cleanup the dummy Rails app
311
+ task clean: :environment do
312
+ app_path = File.expand_path("spec/dummy", __dir__)
313
+ db_path = File.join(app_path, "db", "test.sqlite3")
314
+ FileUtils.rm_f(db_path)
315
+ end
316
+
317
+ begin
318
+ require "rubocop/rake_task"
319
+ RuboCop::RakeTask.new
320
+ rescue LoadError
321
+ desc "Run RuboCop"
322
+ task rubocop: :environment do
323
+ abort "RuboCop is not available. Run 'bundle install' first."
324
+ end
325
+ end
326
+
327
+ begin
328
+ require "brakeman"
329
+ desc "Run Brakeman"
330
+ task brakeman: :environment do
331
+ Brakeman.run(app_path: ".")
332
+ end
333
+ rescue LoadError
334
+ desc "Run Brakeman"
335
+ task brakeman: :environment do
336
+ abort "Brakeman is not available. Run 'bundle install' first."
337
+ end
338
+ end
339
+
340
+ desc "Run all CI checks"
341
+ task ci: %i[rubocop all_tests brakeman]
data/UPGRADE.md ADDED
@@ -0,0 +1,57 @@
1
+ # Clavis Upgrade Guide
2
+
3
+ ## Upgrading to 0.2.0 from 0.1.x
4
+
5
+ Version 0.2.0 brings several important fixes and improvements to the Clavis gem, focusing on better documentation, improved compatibility with existing Rails applications, and bug fixes for various integration issues.
6
+
7
+ ### Key Changes
8
+
9
+ 1. **Module Alias Consistency**:
10
+ - The module `Clavis::Models::OauthAuthenticatable` is now properly aliased to `Clavis::Models::Concerns::OauthAuthenticatable`
11
+ - This means either reference can be used in your application code: `include Clavis::Models::OauthAuthenticatable` or `include Clavis::Models::Concerns::OauthAuthenticatable`
12
+
13
+ 2. **View Helper Integration**:
14
+ - Improved automatic inclusion of view helpers in Rails applications
15
+ - Fixed issues with helper availability in `ApplicationHelper`
16
+ - View helpers like `clavis_oauth_button` should now work out of the box
17
+
18
+ 3. **Installation Generator Improvements**:
19
+ - Updated the `clavis:install` generator to properly handle the creation of OAuth identity tables
20
+ - Fixed issues with duplicate migrations
21
+ - Added better support for integrating with existing User models
22
+
23
+ 4. **Documentation Updates**:
24
+ - Added comprehensive guide for integrating with existing authentication systems
25
+ - Corrected inconsistencies between code examples and actual implementations
26
+ - Updated installation and configuration guidance
27
+
28
+ ### Migration Steps
29
+
30
+ **For most users, no changes are required**. The improvements in 0.2.0 are backward compatible.
31
+
32
+ If you encountered any of the specific issues addressed in this release, upgrade to 0.2.0 and follow these steps:
33
+
34
+ 1. Update your Gemfile:
35
+ ```ruby
36
+ gem 'clavis', '~> 0.2.0'
37
+ ```
38
+
39
+ 2. Run bundle install:
40
+ ```bash
41
+ bundle install
42
+ ```
43
+
44
+ 3. If you've manually implemented workarounds for any of the fixed issues, you can now remove them.
45
+
46
+ 4. If you're integrating with an existing application, refer to the new integration guide at `/docs/integration.md` for detailed instructions.
47
+
48
+ ### New Documentation
49
+
50
+ - `docs/integration.md` - Guide for integrating Clavis with existing authentication systems
51
+ - Updated README with clearer instructions and example code
52
+ - Improved error handling documentation
53
+
54
+ ### Known Issues
55
+
56
+ - Rails 7.1+ and Ruby 3.4+ compatibility testing is ongoing
57
+ - Full test suite coverage requires a Rails environment for some specs
@@ -0,0 +1,133 @@
1
+ /* Clavis OAuth Button Styles */
2
+
3
+ .clavis-oauth-button {
4
+ display: inline-flex;
5
+ align-items: center;
6
+ justify-content: center;
7
+ padding: 10px 16px;
8
+ border-radius: 4px;
9
+ font-size: 14px;
10
+ font-weight: 500;
11
+ text-decoration: none;
12
+ margin: 5px 0;
13
+ border: 1px solid rgba(0, 0, 0, 0.1);
14
+ transition: all 0.2s ease;
15
+ cursor: pointer;
16
+ min-width: 240px;
17
+ height: 40px;
18
+ box-sizing: border-box;
19
+ }
20
+
21
+ .clavis-oauth-button__icon,
22
+ .clavis-oauth-button span .clavis-icon {
23
+ width: 18px;
24
+ height: 18px;
25
+ margin-right: 10px;
26
+ fill: currentColor;
27
+ }
28
+
29
+ .clavis-oauth-button span {
30
+ line-height: 1;
31
+ }
32
+
33
+ /* Google - Following Google branding guidelines */
34
+ .clavis-oauth-button--google {
35
+ background-color: white;
36
+ color: rgba(0, 0, 0, 0.54);
37
+ border: 1px solid #dadce0;
38
+ box-shadow: 0 1px 3px rgba(0, 0, 0, 0.08);
39
+ font-weight: 500;
40
+ }
41
+
42
+ .clavis-oauth-button--google:hover {
43
+ background-color: #f8f8f8;
44
+ box-shadow: 0 1px 3px rgba(0, 0, 0, 0.12);
45
+ }
46
+
47
+ .clavis-oauth-button--google:focus {
48
+ box-shadow: 0 0 0 3px rgba(66, 133, 244, 0.3);
49
+ outline: none;
50
+ }
51
+
52
+ .clavis-oauth-button--google .clavis-icon {
53
+ width: 18px;
54
+ height: 18px;
55
+ }
56
+
57
+ /* GitHub */
58
+ .clavis-oauth-button--github {
59
+ background-color: #24292e;
60
+ color: white;
61
+ }
62
+
63
+ .clavis-oauth-button--github:hover {
64
+ background-color: #2c3238;
65
+ border-color: #24292e;
66
+ }
67
+
68
+ /* Apple - Following Apple's Sign in with Apple guidelines */
69
+ .clavis-oauth-button--apple {
70
+ background-color: black;
71
+ color: white;
72
+ border-radius: 4px;
73
+ }
74
+
75
+ .clavis-oauth-button--apple:hover {
76
+ background-color: #333;
77
+ }
78
+
79
+ .clavis-oauth-button--apple .clavis-icon {
80
+ width: 16px;
81
+ height: 16px;
82
+ }
83
+
84
+ /* Facebook - Following Facebook branding guidelines */
85
+ .clavis-oauth-button--facebook {
86
+ background-color: #1877F2;
87
+ color: white;
88
+ border: none;
89
+ font-weight: bold;
90
+ }
91
+
92
+ .clavis-oauth-button--facebook:hover {
93
+ background-color: #166fe5;
94
+ border-color: #166fe5;
95
+ }
96
+
97
+ /* Microsoft - Following Microsoft branding guidelines */
98
+ .clavis-oauth-button--microsoft {
99
+ background-color: white;
100
+ color: #5e5e5e;
101
+ border: 1px solid #8c8c8c;
102
+ }
103
+
104
+ .clavis-oauth-button--microsoft:hover {
105
+ background-color: #f0f0f0;
106
+ }
107
+
108
+ .clavis-oauth-button--microsoft .clavis-icon {
109
+ width: 16px;
110
+ height: 16px;
111
+ }
112
+
113
+ /* Generic OAuth button */
114
+ .clavis-oauth-button--oauth {
115
+ background-color: #f8f9fa;
116
+ color: #202124;
117
+ border: 1px solid #dadce0;
118
+ }
119
+
120
+ .clavis-oauth-button--oauth:hover {
121
+ background-color: #f1f3f4;
122
+ }
123
+
124
+ /* Error message */
125
+ .clavis-error {
126
+ color: #721c24;
127
+ background-color: #f8d7da;
128
+ border: 1px solid #f5c6cb;
129
+ padding: 10px;
130
+ border-radius: 4px;
131
+ margin: 10px 0;
132
+ font-size: 14px;
133
+ }
@@ -0,0 +1,133 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Clavis
4
+ # AuthController directly inherits from ActionController::Base to avoid inheriting
5
+ # host application's authentication requirements or before_actions that could
6
+ # interfere with the OAuth flow
7
+ class AuthController < ::ActionController::Base
8
+ include Clavis::Controllers::Concerns::Authentication
9
+ include Clavis::Controllers::Concerns::SessionManagement
10
+
11
+ # Add basic controller setup
12
+ protect_from_forgery with: :exception
13
+
14
+ # Allow access to main_app routes
15
+ helper Rails.application.routes.url_helpers if defined?(Rails)
16
+
17
+ # Skip CSRF protection for OAuth callback endpoints since they come from external redirects
18
+ skip_before_action :verify_authenticity_token, only: [:callback]
19
+
20
+ def authorize
21
+ # Store the current URL for returning after authentication
22
+ store_location if request.get?
23
+
24
+ oauth_authorize
25
+ end
26
+
27
+ def callback
28
+ oauth_callback do |user, auth_hash|
29
+ # This is a default implementation that can be overridden
30
+ # by the application using the gem
31
+ if respond_to?(:clavis_authentication_success)
32
+ clavis_authentication_success(user, auth_hash)
33
+ else
34
+ # Use the SessionManagement method to sign in the user with secure cookies
35
+ sign_in_user(user)
36
+
37
+ # Get the redirect path
38
+ redirect_path = after_login_path
39
+
40
+ # Force redirect to root path if it's redirecting to auth path
41
+ if redirect_path.include?("/auth/")
42
+ redirect_path = defined?(main_app) && main_app.respond_to?(:root_path) ? main_app.root_path : "/"
43
+ end
44
+
45
+ redirect_to redirect_path, notice: "Successfully signed in with #{params[:provider].capitalize}"
46
+ end
47
+ end
48
+ rescue StandardError => e
49
+ Clavis::Logging.log_error(e)
50
+
51
+ if respond_to?(:clavis_authentication_failure)
52
+ clavis_authentication_failure(e)
53
+ else
54
+ # Default behavior: redirect to sign in page with error
55
+ flash[:alert] = case e
56
+ when Clavis::AuthorizationDenied
57
+ "Authentication was cancelled"
58
+ when Clavis::InvalidState, Clavis::MissingState
59
+ "Authentication session expired. Please try again."
60
+ else
61
+ "Authentication failed: #{e.message}"
62
+ end
63
+
64
+ # Safe redirect to after_login_path
65
+ redirect_to after_login_path
66
+ end
67
+ end
68
+
69
+ private
70
+
71
+ def find_or_create_user_from_oauth(auth_hash)
72
+ user_class = Clavis.configuration.user_class.constantize
73
+ finder_method = Clavis.configuration.user_finder_method
74
+
75
+ if user_class.respond_to?(finder_method)
76
+ user_class.public_send(finder_method, auth_hash)
77
+ else
78
+ # If no suitable method is available, just return the auth hash
79
+ auth_hash
80
+ end
81
+ end
82
+
83
+ # Method to handle authentication requests with proper routing
84
+ def request_authentication
85
+ session[:return_to_after_authenticating] = request.url
86
+
87
+ # Only redirect to paths we control or to root_path
88
+ # This avoids assumptions about the host application's routes
89
+ redirect_to main_app.root_path, alert: "Authentication required. Please sign in to continue."
90
+ end
91
+
92
+ def after_authentication_url
93
+ return session.delete(:return_to_after_authenticating) if session[:return_to_after_authenticating].present?
94
+
95
+ # Try to get main_app's root_path, fall back to "/"
96
+ begin
97
+ main_app.respond_to?(:root_path) ? main_app.root_path : "/"
98
+ rescue StandardError
99
+ "/"
100
+ end
101
+ end
102
+
103
+ def safe_redirect(path = nil)
104
+ # Try default paths in order of preference
105
+ fallback_paths = [
106
+ -> { main_app.root_path if main_app.respond_to?(:root_path) },
107
+ -> { main_app.respond_to?(:login_path) ? main_app.login_path : nil },
108
+ -> { "/" } # Final fallback is always root
109
+ ]
110
+
111
+ # Use provided path or find first working fallback
112
+ target_path = path || fallback_paths.lazy.map(&:call).find(&:present?)
113
+
114
+ # Perform the redirect with exception handling
115
+ begin
116
+ redirect_to target_path
117
+ rescue StandardError => e
118
+ # Log the error and redirect to root as ultimate fallback
119
+ Clavis::Logging.log_error("Redirect error: #{e.message}. Falling back to '/'")
120
+ redirect_to "/"
121
+ end
122
+ end
123
+
124
+ # Override default_path to ensure we don't redirect back to auth paths
125
+ def default_path
126
+ if defined?(main_app) && main_app.respond_to?(:root_path)
127
+ main_app.root_path
128
+ else
129
+ "/"
130
+ end
131
+ end
132
+ end
133
+ end
@@ -0,0 +1,16 @@
1
+ default: &default
2
+ adapter: sqlite3
3
+ pool: <%= ENV.fetch("RAILS_MAX_THREADS") { 5 } %>
4
+ timeout: 5000
5
+
6
+ development:
7
+ <<: *default
8
+ database: db/development.sqlite3
9
+
10
+ test:
11
+ <<: *default
12
+ database: ":memory:"
13
+
14
+ production:
15
+ <<: *default
16
+ database: db/production.sqlite3