kaze 0.5.0 → 0.7.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (123) hide show
  1. checksums.yaml +4 -4
  2. data/lib/kaze/commands/install_command.rb +18 -4
  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 +12 -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 +115 -88
  67. data/stubs/inertia-react-ts/app/javascript/Components/InputError.tsx +14 -7
  68. data/stubs/inertia-react-ts/app/javascript/Components/InputLabel.tsx +15 -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 +32 -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