kaze 0.5.0 → 0.6.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 (123) hide show
  1. checksums.yaml +4 -4
  2. data/lib/kaze/commands/install_command.rb +8 -0
  3. data/lib/kaze/commands/installs_hotwire_stack.rb +5 -2
  4. data/lib/kaze/commands/installs_inertia_stacks.rb +13 -3
  5. data/lib/kaze/version.rb +1 -1
  6. data/stubs/default/app/forms/auth/login_form.rb +2 -8
  7. data/stubs/default/app/forms/update_profile_information_form.rb +1 -1
  8. data/stubs/default/app/models/auth.rb +57 -0
  9. data/stubs/default/app/models/current.rb +1 -1
  10. data/stubs/default/app/validators/current_password_validator.rb +1 -1
  11. data/stubs/default/app/views/layouts/mailer.html.erb +367 -372
  12. data/stubs/default/app/views/layouts/mailer.text.erb +1 -4
  13. data/stubs/default/app/views/user_mailer/reset_password.html.erb +21 -26
  14. data/stubs/default/test/factories/users.rb +7 -0
  15. data/stubs/default/test/integration/auth/authentication_test.rb +43 -0
  16. data/stubs/default/test/integration/auth/password_reset_test.rb +41 -0
  17. data/stubs/default/test/integration/auth/registration_test.rb +21 -0
  18. data/stubs/default/test/integration/password_update_test.rb +28 -0
  19. data/stubs/default/test/integration/profile_test.rb +51 -0
  20. data/stubs/default/test/test_helper.rb +38 -0
  21. data/stubs/hotwire/app/components/dropdown_component.html.erb +17 -18
  22. data/stubs/hotwire/app/components/modal_component.html.erb +55 -59
  23. data/stubs/hotwire/app/controllers/application_controller.rb +1 -0
  24. data/stubs/hotwire/app/controllers/auth/authenticated_session_controller.rb +10 -7
  25. data/stubs/hotwire/app/controllers/auth/new_password_controller.rb +3 -1
  26. data/stubs/hotwire/app/controllers/auth/password_reset_link_controller.rb +3 -1
  27. data/stubs/hotwire/app/controllers/auth/registered_user_controller.rb +4 -2
  28. data/stubs/hotwire/app/controllers/concerns/authenticate.rb +5 -20
  29. data/stubs/hotwire/app/controllers/concerns/redirect_if_authenticated.rb +19 -0
  30. data/stubs/hotwire/app/controllers/concerns/set_current_auth.rb +9 -0
  31. data/stubs/hotwire/app/controllers/password_controller.rb +1 -1
  32. data/stubs/hotwire/app/controllers/profile_controller.rb +6 -4
  33. data/stubs/hotwire/app/controllers/welcome_controller.rb +1 -1
  34. data/stubs/hotwire/app/javascript/application.js +3 -3
  35. data/stubs/hotwire/app/views/auth/forgot_password.html.erb +12 -17
  36. data/stubs/hotwire/app/views/auth/login.html.erb +0 -9
  37. data/stubs/hotwire/app/views/auth/register.html.erb +0 -13
  38. data/stubs/hotwire/app/views/auth/reset_password.html.erb +0 -7
  39. data/stubs/hotwire/app/views/dashboard/index.html.erb +9 -10
  40. data/stubs/hotwire/app/views/layouts/_navigation.html.erb +77 -87
  41. data/stubs/hotwire/app/views/layouts/application.html.erb +0 -9
  42. data/stubs/hotwire/app/views/layouts/guest.html.erb +0 -6
  43. data/stubs/hotwire/app/views/profile/edit.html.erb +19 -22
  44. data/stubs/hotwire/app/views/profile/partials/_delete_user_form.html.erb +32 -42
  45. data/stubs/hotwire/app/views/profile/partials/_update_password_form.html.erb +42 -55
  46. data/stubs/hotwire/app/views/profile/partials/_update_profile_information_form.html.erb +36 -46
  47. data/stubs/hotwire/app/views/welcome/index.html.erb +34 -46
  48. data/stubs/hotwire/config/tailwind.config.js +2 -2
  49. data/stubs/inertia-common/app/controllers/application_controller.rb +1 -0
  50. data/stubs/inertia-common/app/controllers/auth/authenticated_session_controller.rb +10 -7
  51. data/stubs/inertia-common/app/controllers/auth/new_password_controller.rb +3 -1
  52. data/stubs/inertia-common/app/controllers/auth/password_reset_link_controller.rb +3 -1
  53. data/stubs/inertia-common/app/controllers/auth/registered_user_controller.rb +4 -2
  54. data/stubs/inertia-common/app/controllers/concerns/authenticate.rb +5 -20
  55. data/stubs/inertia-common/app/controllers/concerns/handle_inertia_requests.rb +1 -1
  56. data/stubs/inertia-common/app/controllers/concerns/redirect_if_authenticated.rb +19 -0
  57. data/stubs/inertia-common/app/controllers/concerns/set_current_auth.rb +9 -0
  58. data/stubs/inertia-common/app/controllers/password_controller.rb +1 -1
  59. data/stubs/inertia-common/app/controllers/profile_controller.rb +5 -3
  60. data/stubs/inertia-common/app/controllers/welcome_controller.rb +1 -1
  61. data/stubs/inertia-common/test/integration/password_update_test.rb +28 -0
  62. data/stubs/inertia-common/test/integration/profile_test.rb +51 -0
  63. data/stubs/inertia-react-ts/app/javascript/Components/ApplicationLogo.tsx +13 -9
  64. data/stubs/inertia-react-ts/app/javascript/Components/Checkbox.tsx +15 -12
  65. data/stubs/inertia-react-ts/app/javascript/Components/DangerButton.tsx +20 -15
  66. data/stubs/inertia-react-ts/app/javascript/Components/Dropdown.tsx +119 -87
  67. data/stubs/inertia-react-ts/app/javascript/Components/InputError.tsx +14 -7
  68. data/stubs/inertia-react-ts/app/javascript/Components/InputLabel.tsx +18 -7
  69. data/stubs/inertia-react-ts/app/javascript/Components/Modal.tsx +60 -60
  70. data/stubs/inertia-react-ts/app/javascript/Components/NavLink.tsx +21 -16
  71. data/stubs/inertia-react-ts/app/javascript/Components/PrimaryButton.tsx +20 -15
  72. data/stubs/inertia-react-ts/app/javascript/Components/ResponsiveNavLink.tsx +19 -14
  73. data/stubs/inertia-react-ts/app/javascript/Components/SecondaryButton.tsx +22 -16
  74. data/stubs/inertia-react-ts/app/javascript/Components/TextInput.tsx +35 -24
  75. data/stubs/inertia-react-ts/app/javascript/Layouts/AuthenticatedLayout.tsx +157 -117
  76. data/stubs/inertia-react-ts/app/javascript/Layouts/GuestLayout.tsx +15 -15
  77. data/stubs/inertia-react-ts/app/javascript/Pages/Auth/ForgotPassword.tsx +52 -49
  78. data/stubs/inertia-react-ts/app/javascript/Pages/Auth/Login.tsx +90 -82
  79. data/stubs/inertia-react-ts/app/javascript/Pages/Auth/Register.tsx +118 -115
  80. data/stubs/inertia-react-ts/app/javascript/Pages/Auth/ResetPassword.tsx +63 -60
  81. data/stubs/inertia-react-ts/app/javascript/Pages/Dashboard.tsx +23 -17
  82. data/stubs/inertia-react-ts/app/javascript/Pages/Profile/Edit.tsx +31 -27
  83. data/stubs/inertia-react-ts/app/javascript/Pages/Profile/Partials/DeleteUserForm.tsx +109 -99
  84. data/stubs/inertia-react-ts/app/javascript/Pages/Profile/Partials/UpdatePasswordForm.tsx +121 -113
  85. data/stubs/inertia-react-ts/app/javascript/Pages/Profile/Partials/UpdateProfileInformationForm.tsx +76 -69
  86. data/stubs/inertia-react-ts/app/javascript/Pages/Welcome.tsx +87 -63
  87. data/stubs/inertia-react-ts/app/javascript/entrypoints/application.tsx +32 -25
  88. data/stubs/inertia-react-ts/app/views/layouts/application.html.erb +0 -4
  89. data/stubs/inertia-react-ts/config/tailwind.config.js +2 -2
  90. data/stubs/inertia-react-ts/vite.config.ts +2 -5
  91. data/stubs/inertia-vue-ts/app/javascript/Components/ApplicationLogo.vue +10 -6
  92. data/stubs/inertia-vue-ts/app/javascript/Components/Checkbox.vue +18 -18
  93. data/stubs/inertia-vue-ts/app/javascript/Components/DangerButton.vue +5 -5
  94. data/stubs/inertia-vue-ts/app/javascript/Components/Dropdown.vue +60 -57
  95. data/stubs/inertia-vue-ts/app/javascript/Components/DropdownLink.vue +9 -9
  96. data/stubs/inertia-vue-ts/app/javascript/Components/InputError.vue +7 -7
  97. data/stubs/inertia-vue-ts/app/javascript/Components/InputLabel.vue +6 -6
  98. data/stubs/inertia-vue-ts/app/javascript/Components/Modal.vue +84 -74
  99. data/stubs/inertia-vue-ts/app/javascript/Components/NavLink.vue +12 -12
  100. data/stubs/inertia-vue-ts/app/javascript/Components/PrimaryButton.vue +5 -5
  101. data/stubs/inertia-vue-ts/app/javascript/Components/ResponsiveNavLink.vue +12 -12
  102. data/stubs/inertia-vue-ts/app/javascript/Components/SecondaryButton.vue +13 -13
  103. data/stubs/inertia-vue-ts/app/javascript/Components/TextInput.vue +13 -13
  104. data/stubs/inertia-vue-ts/app/javascript/Layouts/AuthenticatedLayout.vue +168 -136
  105. data/stubs/inertia-vue-ts/app/javascript/Layouts/GuestLayout.vue +15 -13
  106. data/stubs/inertia-vue-ts/app/javascript/Pages/Auth/ForgotPassword.vue +56 -49
  107. data/stubs/inertia-vue-ts/app/javascript/Pages/Auth/Login.vue +78 -72
  108. data/stubs/inertia-vue-ts/app/javascript/Pages/Auth/Register.vue +101 -97
  109. data/stubs/inertia-vue-ts/app/javascript/Pages/Auth/ResetPassword.vue +71 -68
  110. data/stubs/inertia-vue-ts/app/javascript/Pages/Dashboard.vue +22 -14
  111. data/stubs/inertia-vue-ts/app/javascript/Pages/Profile/Edit.vue +34 -30
  112. data/stubs/inertia-vue-ts/app/javascript/Pages/Profile/Partials/DeleteUserForm.vue +87 -83
  113. data/stubs/inertia-vue-ts/app/javascript/Pages/Profile/Partials/UpdatePasswordForm.vue +105 -98
  114. data/stubs/inertia-vue-ts/app/javascript/Pages/Profile/Partials/UpdateProfileInformationForm.vue +69 -59
  115. data/stubs/inertia-vue-ts/app/javascript/Pages/Welcome.vue +74 -47
  116. data/stubs/inertia-vue-ts/app/views/layouts/application.html.erb +0 -4
  117. data/stubs/inertia-vue-ts/config/tailwind.config.js +2 -2
  118. data/stubs/inertia-vue-ts/vite.config.ts +2 -5
  119. metadata +18 -6
  120. data/stubs/hotwire/bin/vite +0 -27
  121. data/stubs/inertia-common/Procfile.dev +0 -3
  122. /data/stubs/{hotwire → default}/Procfile.dev +0 -0
  123. /data/stubs/hotwire/app/javascript/{alpinejs.js → alpinejs.stub} +0 -0
@@ -1,58 +1,45 @@
1
1
  <turbo-frame id="update_password_form">
2
- <section>
3
- <header>
4
- <h2 class="text-lg font-medium text-gray-900 dark:text-gray-100">
5
- Update Password
6
- </h2>
7
-
8
- <p class="mt-1 text-sm text-gray-600 dark:text-gray-400">
9
- Ensure your account is using a long, random password to stay secure.
10
- </p>
11
- </header>
12
-
13
- <%= form_with model: @update_password_form, url: password_update_path, method: "put", class: "mt-6 space-y-6" do |update_password_form| %>
14
- <!-- Current Password -->
15
- <div>
16
- <%= render(InputLabelComponent.new({ for: "update_password_current_password", value: "Current Password" })) %>
17
-
18
- <%= render(TextInputComponent.new({ id: "update_password_current_password", class: "block mt-1 w-full", type: "password", name: "current_password", autocomplete: "current-password" })) %>
19
-
20
- <%= render(InputErrorComponent.new({ class: "mt-2", message: @update_password_form.error_messages[:current_password] })) %>
21
- </div>
22
-
23
- <!-- New Password -->
24
- <div>
25
- <%= render(InputLabelComponent.new({ for: "update_password_password", value: "New Password" })) %>
26
-
27
- <%= render(TextInputComponent.new({ id: "update_password_password", class: "block mt-1 w-full", type: "password", name: "password", autocomplete: "new-password" })) %>
28
-
29
- <%= render(InputErrorComponent.new({ class: "mt-2", message: @update_password_form.error_messages[:password] })) %>
30
- </div>
31
-
32
- <!-- Confirm Password -->
33
- <div class="mt-4">
34
- <%= render(InputLabelComponent.new({ for: "update_password_password_confirmation", value: "Confirm Password" })) %>
35
-
36
- <%= render(TextInputComponent.new({ id: "update_password_password_confirmation", class: "block mt-1 w-full", type: "password", name: "password_confirmation", autocomplete: "new-password" })) %>
37
-
38
- <%= render(InputErrorComponent.new({ class: "mt-2", message: @update_password_form.error_messages[:password_confirmation] })) %>
39
- </div>
40
-
41
- <div class="flex items-center gap-4">
42
- <%= render(PrimaryButtonComponent.new) do %>
43
- Save
44
- <% end %>
45
-
46
- <% if flash[:status] == "password-updated" %>
47
- <p
48
- x-data="{ show: true }"
49
- x-show="show"
50
- x-transition
51
- x-init="setTimeout(() => show = false, 2000)"
52
- class="text-sm text-gray-600 dark:text-gray-400"
53
- >Saved.</p>
54
- <% end %>
55
- </div>
2
+ <section>
3
+ <header>
4
+ <h2 class="text-lg font-medium text-gray-900 dark:text-gray-100">
5
+ Update Password
6
+ </h2>
7
+ <p class="mt-1 text-sm text-gray-600 dark:text-gray-400">
8
+ Ensure your account is using a long, random password to stay secure.
9
+ </p>
10
+ </header>
11
+ <%= form_with model: @update_password_form, url: password_update_path, method: "put", class: "mt-6 space-y-6" do |update_password_form| %>
12
+ <!-- Current Password -->
13
+ <div>
14
+ <%= render(InputLabelComponent.new({ for: "update_password_current_password", value: "Current Password" })) %>
15
+ <%= render(TextInputComponent.new({ id: "update_password_current_password", class: "block mt-1 w-full", type: "password", name: "current_password", autocomplete: "current-password" })) %>
16
+ <%= render(InputErrorComponent.new({ class: "mt-2", message: @update_password_form.error_messages[:current_password] })) %>
17
+ </div>
18
+ <!-- New Password -->
19
+ <div>
20
+ <%= render(InputLabelComponent.new({ for: "update_password_password", value: "New Password" })) %>
21
+ <%= render(TextInputComponent.new({ id: "update_password_password", class: "block mt-1 w-full", type: "password", name: "password", autocomplete: "new-password" })) %>
22
+ <%= render(InputErrorComponent.new({ class: "mt-2", message: @update_password_form.error_messages[:password] })) %>
23
+ </div>
24
+ <!-- Confirm Password -->
25
+ <div class="mt-4">
26
+ <%= render(InputLabelComponent.new({ for: "update_password_password_confirmation", value: "Confirm Password" })) %>
27
+ <%= render(TextInputComponent.new({ id: "update_password_password_confirmation", class: "block mt-1 w-full", type: "password", name: "password_confirmation", autocomplete: "new-password" })) %>
28
+ <%= render(InputErrorComponent.new({ class: "mt-2", message: @update_password_form.error_messages[:password_confirmation] })) %>
29
+ </div>
30
+ <div class="flex items-center gap-4">
31
+ <%= render(PrimaryButtonComponent.new) do %>
32
+ Save
56
33
  <% end %>
57
- </section>
34
+ <% if flash[:status] == "password-updated" %>
35
+ <p
36
+ x-data="{ show: true }"
37
+ x-show="show"
38
+ x-transition
39
+ x-init="setTimeout(() => show = false, 2000)"
40
+ class="text-sm text-gray-600 dark:text-gray-400">Saved.</p>
41
+ <% end %>
42
+ </div>
43
+ <% end %>
44
+ </section>
58
45
  </turbo-frame>
@@ -1,49 +1,39 @@
1
1
  <turbo-frame id="update_profile_information_form">
2
- <section>
3
- <header>
4
- <h2 class="text-lg font-medium text-gray-900 dark:text-gray-100">
5
- Profile Information
6
- </h2>
7
-
8
- <p class="mt-1 text-sm text-gray-600 dark:text-gray-400">
9
- Update your account's profile information and email address.
10
- </p>
11
- </header>
12
-
13
- <%= form_with model: @update_profile_information_form, url: profile_update_path, method: "patch", class: "mt-6 space-y-6" do %>
14
- <!-- Name -->
15
- <div>
16
- <%= render(InputLabelComponent.new({ for: "name", value: "Name" })) %>
17
-
18
- <%= render(TextInputComponent.new({ id: "name", class: "block mt-1 w-full", type: "text", name: "name", value: @update_profile_information_form.name, required: true, autofocus: true, autocomplete: "name" })) %>
19
-
20
- <%= render(InputErrorComponent.new({ class: "mt-2", message: @update_profile_information_form.error_messages[:name] })) %>
21
- </div>
22
-
23
- <!-- Email Address -->
24
- <div class="mt-4">
25
- <%= render(InputLabelComponent.new({ for: "email", value: "Email" })) %>
26
-
27
- <%= render(TextInputComponent.new({ id: "email", class: "block mt-1 w-full", type: "email", name: "email", value: @update_profile_information_form.email, required: true, autocomplete: "username" })) %>
28
-
29
- <%= render(InputErrorComponent.new({ class: "mt-2", message: @update_profile_information_form.error_messages[:email] })) %>
30
- </div>
31
-
32
- <div class="flex items-center gap-4">
33
- <%= render(PrimaryButtonComponent.new) do %>
34
- Save
35
- <% end %>
36
-
37
- <% if flash[:status] == "profile-updated" %>
38
- <p
39
- x-data="{ show: true }"
40
- x-show="show"
41
- x-transition
42
- x-init="setTimeout(() => show = false, 2000)"
43
- class="text-sm text-gray-600 dark:text-gray-400"
44
- >Saved.</p>
45
- <% end %>
46
- </div>
2
+ <section>
3
+ <header>
4
+ <h2 class="text-lg font-medium text-gray-900 dark:text-gray-100">
5
+ Profile Information
6
+ </h2>
7
+ <p class="mt-1 text-sm text-gray-600 dark:text-gray-400">
8
+ Update your account's profile information and email address.
9
+ </p>
10
+ </header>
11
+ <%= form_with model: @update_profile_information_form, url: profile_update_path, method: "patch", class: "mt-6 space-y-6" do %>
12
+ <!-- Name -->
13
+ <div>
14
+ <%= render(InputLabelComponent.new({ for: "name", value: "Name" })) %>
15
+ <%= render(TextInputComponent.new({ id: "name", class: "block mt-1 w-full", type: "text", name: "name", value: @update_profile_information_form.name, required: true, autofocus: true, autocomplete: "name" })) %>
16
+ <%= render(InputErrorComponent.new({ class: "mt-2", message: @update_profile_information_form.error_messages[:name] })) %>
17
+ </div>
18
+ <!-- Email Address -->
19
+ <div class="mt-4">
20
+ <%= render(InputLabelComponent.new({ for: "email", value: "Email" })) %>
21
+ <%= render(TextInputComponent.new({ id: "email", class: "block mt-1 w-full", type: "email", name: "email", value: @update_profile_information_form.email, required: true, autocomplete: "username" })) %>
22
+ <%= render(InputErrorComponent.new({ class: "mt-2", message: @update_profile_information_form.error_messages[:email] })) %>
23
+ </div>
24
+ <div class="flex items-center gap-4">
25
+ <%= render(PrimaryButtonComponent.new) do %>
26
+ Save
47
27
  <% end %>
48
- </section>
28
+ <% if flash[:status] == "profile-updated" %>
29
+ <p
30
+ x-data="{ show: true }"
31
+ x-show="show"
32
+ x-transition
33
+ x-init="setTimeout(() => show = false, 2000)"
34
+ class="text-sm text-gray-600 dark:text-gray-400">Saved.</p>
35
+ <% end %>
36
+ </div>
37
+ <% end %>
38
+ </section>
49
39
  </turbo-frame>
@@ -4,65 +4,53 @@
4
4
  <meta charset="utf-8">
5
5
  <meta name="viewport" content="width=device-width,initial-scale=1">
6
6
  <meta name="turbo-visit-control" content="reload">
7
-
8
7
  <title><%= content_for(:title) || "Rails" %></title>
9
-
10
8
  <%= csrf_meta_tags %>
11
9
  <%= csp_meta_tag %>
12
-
13
10
  <%= stylesheet_link_tag "tailwind", "inter-font", "data-turbo-track": "reload" %>
14
-
15
11
  <%= stylesheet_link_tag "application", "data-turbo-track": "reload" %>
16
12
  <%= javascript_importmap_tags %>
17
-
18
13
  <%= hotwire_livereload_tags if Rails.env.development? %>
19
14
  <%= turbo_refreshes_with method: :morph, scroll: :preserve %>
20
15
  </head>
21
-
22
16
  <body class="font-sans antialiased">
23
17
  <div
24
- class="flex flex-col items-center justify-center bg-[#F0E7E9] bg-center bg-cover text-[#261B23] not-italic font-normal leading-tight min-h-screen text-center"
25
- style="background-image: url();font-family: Sans-Serif;font-size: calc(0.9em + 0.5vw);"
26
- >
27
- <header class="absolute top-0 right-0 grid grid-cols-2 items-center gap-2 py-10 lg:grid-cols-3">
28
- <div class="-mx-3 flex flex-1 justify-end">
29
- <% if Current.user %>
30
- <a
31
- href="<%= dashboard_url %>"
32
- class="rounded-md px-3 py-2 text-black ring-1 ring-transparent transition hover:text-black/70 focus:outline-none focus-visible:ring-[#FF2D20] dark:text-white dark:hover:text-white/80 dark:focus-visible:ring-white"
33
- v-if="$page.props.auth.user"
34
- >
35
- Dashboard
36
- </a>
37
- <% else %>
38
- <a
39
- href="<%= login_url %>"
40
- class="rounded-md px-3 py-2 text-black ring-1 ring-transparent transition hover:text-black/70 focus:outline-none focus-visible:ring-[#FF2D20] dark:text-white dark:hover:text-white/80 dark:focus-visible:ring-white"
41
- >
42
- Log in
43
- </a>
44
- <a
45
- href="<%= register_url %>"
46
- class="rounded-md px-3 py-2 text-black ring-1 ring-transparent transition hover:text-black/70 focus:outline-none focus-visible:ring-[#FF2D20] dark:text-white dark:hover:text-white/80 dark:focus-visible:ring-white"
47
- >
48
- Register
49
- </a>
50
- <% end %>
51
- </div>
52
- </header>
53
- <nav style="font-size: 0;height: 20vw;line-height: 0;max-height: 280px;max-width: 280px;min-height: 86px;min-width: 86px;width: 20vw;">
18
+ class="flex flex-col items-center justify-center bg-[#F0E7E9] bg-center bg-cover text-[#261B23] not-italic font-normal leading-tight min-h-screen text-center"
19
+ style="background-image: url();font-family: Sans-Serif;font-size: calc(0.9em + 0.5vw);">
20
+ <header class="absolute top-0 right-0 grid grid-cols-2 items-center gap-2 py-10 lg:grid-cols-3">
21
+ <div class="-mx-3 flex flex-1 justify-end">
22
+ <% if Current.auth.check? %>
54
23
  <a
55
- class="bg-[#D30001] hover:bg-[#261B23] flex rounded-full" href="https://rubyonrails.org" target="_blank"
56
- style="transition: background 0.25s cubic-bezier(0.33, 1, 0.68, 1);filter: drop-shadow(0 20px 13px rgb(0 0 0 / 0.03)) drop-shadow(0 8px 5px rgb(0 0 0 / 0.08));"
57
- >
58
- <img class="h-auto max-w-full w-full cursor-pointer" alt="" src="" />
24
+ href="<%= dashboard_url %>"
25
+ class="rounded-md px-3 py-2 text-black ring-1 ring-transparent transition hover:text-black/70 focus:outline-none focus-visible:ring-[#FF2D20] dark:text-white dark:hover:text-white/80 dark:focus-visible:ring-white"
26
+ v-if="$page.props.auth.user">
27
+ Dashboard
59
28
  </a>
60
- </nav>
61
-
62
- <ul class="bottom-0 left-0 list-none mx-8 mt-0 mb-8 right-0 absolute">
63
- <li><strong>Rails version:</strong> <%= Rails.version %></li>
64
- <li><strong>Ruby version:</strong> <%= RUBY_DESCRIPTION %></li>
65
- </ul>
29
+ <% else %>
30
+ <a
31
+ href="<%= login_url %>"
32
+ class="rounded-md px-3 py-2 text-black ring-1 ring-transparent transition hover:text-black/70 focus:outline-none focus-visible:ring-[#FF2D20] dark:text-white dark:hover:text-white/80 dark:focus-visible:ring-white">
33
+ Log in
34
+ </a>
35
+ <a
36
+ href="<%= register_url %>"
37
+ class="rounded-md px-3 py-2 text-black ring-1 ring-transparent transition hover:text-black/70 focus:outline-none focus-visible:ring-[#FF2D20] dark:text-white dark:hover:text-white/80 dark:focus-visible:ring-white">
38
+ Register
39
+ </a>
40
+ <% end %>
41
+ </div>
42
+ </header>
43
+ <nav style="font-size: 0;height: 20vw;line-height: 0;max-height: 280px;max-width: 280px;min-height: 86px;min-width: 86px;width: 20vw;">
44
+ <a
45
+ class="bg-[#D30001] hover:bg-[#261B23] flex rounded-full" href="https://rubyonrails.org" target="_blank"
46
+ style="transition: background 0.25s cubic-bezier(0.33, 1, 0.68, 1);filter: drop-shadow(0 20px 13px rgb(0 0 0 / 0.03)) drop-shadow(0 8px 5px rgb(0 0 0 / 0.08));">
47
+ <img class="h-auto max-w-full w-full cursor-pointer" alt="" src="">
48
+ </a>
49
+ </nav>
50
+ <ul class="bottom-0 left-0 list-none mx-8 mt-0 mb-8 right-0 absolute">
51
+ <li><strong>Rails version:</strong> <%= Rails.version %></li>
52
+ <li><strong>Ruby version:</strong> <%= RUBY_DESCRIPTION %></li>
53
+ </ul>
66
54
  </div>
67
55
  </body>
68
56
  </html>
@@ -1,5 +1,5 @@
1
- import defaultTheme from 'tailwindcss/defaultTheme';
2
- import forms from '@tailwindcss/forms';
1
+ import defaultTheme from 'tailwindcss/defaultTheme'
2
+ import forms from '@tailwindcss/forms'
3
3
 
4
4
  /** @type {import('tailwindcss').Config} */
5
5
  export default {
@@ -1,4 +1,5 @@
1
1
  class ApplicationController < ActionController::Base
2
+ include SetCurrentAuth
2
3
  include Authenticate
3
4
  include HandleInertiaRequests
4
5
  include VerifyCsrfToken
@@ -1,5 +1,8 @@
1
1
  class Auth::AuthenticatedSessionController < ApplicationController
2
- skip_authentication only: %i[new create]
2
+ include RedirectIfAuthenticated
3
+
4
+ skip_authenticate only: %i[new create]
5
+ skip_redirect_if_authenticated only: %i[destroy]
3
6
 
4
7
  def new
5
8
  render inertia: 'Auth/Login', props: {
@@ -11,18 +14,18 @@ class Auth::AuthenticatedSessionController < ApplicationController
11
14
  def create
12
15
  form = Auth::LoginForm.new params.permit(:email, :password)
13
16
 
14
- user = form.authenticate
15
-
16
- return redirect_back_or_to login_path, inertia: { errors: form.error_messages } if user.nil?
17
+ form.authenticate
17
18
 
18
- login user
19
+ return redirect_back_or_to login_path, inertia: { errors: form.error_messages } if Current.auth.user.nil?
19
20
 
20
21
  redirect_to dashboard_path
21
22
  end
22
23
 
23
24
  def destroy
24
- logout
25
+ Current.auth.logout
26
+
27
+ reset_session
25
28
 
26
- redirect_to login_path
29
+ redirect_to '/'
27
30
  end
28
31
  end
@@ -1,5 +1,7 @@
1
1
  class Auth::NewPasswordController < ApplicationController
2
- skip_authentication
2
+ include RedirectIfAuthenticated
3
+
4
+ skip_authenticate
3
5
 
4
6
  def new
5
7
  render inertia: 'Auth/ResetPassword', props: {
@@ -1,5 +1,7 @@
1
1
  class Auth::PasswordResetLinkController < ApplicationController
2
- skip_authentication
2
+ include RedirectIfAuthenticated
3
+
4
+ skip_authenticate
3
5
 
4
6
  def new
5
7
  render inertia: 'Auth/ForgotPassword', props: {
@@ -1,5 +1,7 @@
1
1
  class Auth::RegisteredUserController < ApplicationController
2
- skip_authentication
2
+ include RedirectIfAuthenticated
3
+
4
+ skip_authenticate
3
5
 
4
6
  def new
5
7
  render inertia: 'Auth/Register'
@@ -12,7 +14,7 @@ class Auth::RegisteredUserController < ApplicationController
12
14
 
13
15
  user = User.create(name: form.name, email: form.email, password: form.password)
14
16
 
15
- login user
17
+ Current.auth.login user
16
18
 
17
19
  redirect_to dashboard_path
18
20
  end
@@ -2,33 +2,18 @@ module Authenticate
2
2
  extend ActiveSupport::Concern
3
3
 
4
4
  included do
5
- before_action :authenticate_user!
5
+ before_action :authenticate!
6
6
  end
7
7
 
8
8
  class_methods do
9
- def skip_authentication(**options)
10
- skip_before_action :authenticate_user!, **options
9
+ def skip_authenticate(**options)
10
+ skip_before_action :authenticate!, **options
11
11
  end
12
12
  end
13
13
 
14
14
  private
15
15
 
16
- def authenticate_user!
17
- if user = User.find_by(id: session[:user_id])
18
- Current.user = user
19
- else
20
- redirect_to login_path
21
- end
22
- end
23
-
24
- def login(user)
25
- Current.user = user
26
- reset_session
27
- session[:user_id] = user.id
28
- end
29
-
30
- def logout
31
- Current.user = nil
32
- reset_session
16
+ def authenticate!
17
+ redirect_to login_path unless Current.auth.check?
33
18
  end
34
19
  end
@@ -3,7 +3,7 @@ module HandleInertiaRequests
3
3
 
4
4
  included do
5
5
  inertia_share do
6
- { auth: { user: Current.user } }
6
+ { auth: { user: Current.auth.get_user } }
7
7
  end
8
8
  end
9
9
  end
@@ -0,0 +1,19 @@
1
+ module RedirectIfAuthenticated
2
+ extend ActiveSupport::Concern
3
+
4
+ included do
5
+ before_action :redirect_if_authenticated!
6
+ end
7
+
8
+ class_methods do
9
+ def skip_redirect_if_authenticated(**options)
10
+ skip_before_action :redirect_if_authenticated!, **options
11
+ end
12
+ end
13
+
14
+ private
15
+
16
+ def redirect_if_authenticated!
17
+ redirect_to dashboard_path if Current.auth.check?
18
+ end
19
+ end
@@ -0,0 +1,9 @@
1
+ module SetCurrentAuth
2
+ extend ActiveSupport::Concern
3
+
4
+ included do
5
+ before_action do
6
+ Current.auth = Auth.new('web', session)
7
+ end
8
+ end
9
+ end
@@ -4,7 +4,7 @@ class PasswordController < ApplicationController
4
4
 
5
5
  return redirect_to profile_edit_path, inertia: { errors: form.error_messages } if form.invalid?
6
6
 
7
- Current.user.update(password: form.password)
7
+ Current.auth.user.update(password: form.password)
8
8
 
9
9
  redirect_back_or_to profile_edit_path
10
10
  end
@@ -10,7 +10,7 @@ class ProfileController < ApplicationController
10
10
 
11
11
  return redirect_to profile_edit_path, inertia: { errors: form.error_messages } if form.invalid?
12
12
 
13
- Current.user.update(name: form.name, email: form.email)
13
+ Current.auth.user.update(name: form.name, email: form.email)
14
14
 
15
15
  redirect_to profile_edit_path
16
16
  end
@@ -20,12 +20,14 @@ class ProfileController < ApplicationController
20
20
 
21
21
  return redirect_back_or_to profile_edit_path, inertia: { errors: form.error_messages } if form.invalid?
22
22
 
23
- user = Current.user
23
+ user = Current.auth.user
24
24
 
25
- logout
25
+ Current.auth.logout
26
26
 
27
27
  user.delete
28
28
 
29
+ reset_session
30
+
29
31
  redirect_to '/'
30
32
  end
31
33
  end
@@ -1,5 +1,5 @@
1
1
  class WelcomeController < ApplicationController
2
- skip_authentication
2
+ skip_authenticate
3
3
 
4
4
  def index
5
5
  render inertia: 'Welcome', props: {
@@ -0,0 +1,28 @@
1
+ require 'test_helper'
2
+
3
+ class PasswordUpdateTest < ActionDispatch::IntegrationTest
4
+ test 'password can be updated' do
5
+ user = FactoryBot.create :user
6
+
7
+ acting_as(user).put password_update_path, params: {
8
+ current_password: 'password',
9
+ password: 'new-password',
10
+ password_confirmation: 'new-password'
11
+ }
12
+
13
+ assert_redirected_to profile_edit_path
14
+ assert BCrypt::Password.new(user.reload.password_digest).is_password?('new-password')
15
+ end
16
+
17
+ test 'correct password must be provided to update password' do
18
+ user = FactoryBot.create :user
19
+
20
+ acting_as(user).put password_update_path, params: {
21
+ current_password: 'wrong-password',
22
+ password: 'new-password',
23
+ password_confirmation: 'new-password'
24
+ }
25
+
26
+ assert_redirected_to profile_edit_path
27
+ end
28
+ end
@@ -0,0 +1,51 @@
1
+ require 'test_helper'
2
+
3
+ class ProfileTest < ActionDispatch::IntegrationTest
4
+ test 'profile page is displayed' do
5
+ user = FactoryBot.create :user
6
+
7
+ acting_as(user).get profile_edit_path
8
+
9
+ assert_response :success
10
+ end
11
+
12
+ test 'profile information can be updated' do
13
+ user = FactoryBot.create :user
14
+
15
+ acting_as(user).patch profile_edit_path, params: {
16
+ name: 'Test User',
17
+ email: 'test@example.com'
18
+ }
19
+
20
+ assert_redirected_to profile_edit_path
21
+
22
+ user.reload
23
+
24
+ assert_equal 'Test User', user.name
25
+ assert_equal 'test@example.com', user.email
26
+ end
27
+
28
+ test 'user can delete their account' do
29
+ user = FactoryBot.create :user
30
+
31
+ acting_as(user).delete profile_destroy_path, params: {
32
+ password: 'password'
33
+ }
34
+
35
+ assert_redirected_to '/'
36
+
37
+ assert_guest
38
+ assert_raise(ActiveRecord::RecordNotFound) { user.reload }
39
+ end
40
+
41
+ test 'correct password must be provided to delete account' do
42
+ user = FactoryBot.create :user
43
+
44
+ acting_as(user).delete profile_destroy_path, params: {
45
+ password: 'wrong-password'
46
+ }
47
+
48
+ assert_redirected_to profile_edit_path
49
+ assert_not_nil user.reload
50
+ end
51
+ end