kaze 0.10.0 → 0.12.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.
- checksums.yaml +4 -4
- data/LICENSE.md +21 -0
- data/README.md +23 -11
- data/lib/kaze/commands/install_command.rb +13 -15
- data/lib/kaze/commands/installs_inertia_stacks.rb +2 -2
- data/lib/kaze/version.rb +1 -1
- data/stubs/default/app/models/concerns/must_verify_email.rb +0 -2
- data/stubs/default/db/migrate/20240101000000_create_users.rb +5 -5
- data/stubs/hotwire/app/components/modal_component.rb +1 -1
- data/stubs/hotwire/app/controllers/profile_controller.rb +1 -1
- data/stubs/hotwire/app/views/profile/partials/_confirm_user_deletion_modal.html.erb +25 -0
- data/stubs/hotwire/app/views/profile/partials/_delete_user_form.html.erb +1 -25
- data/stubs/hotwire/app/views/welcome/index.html.erb +0 -1
- metadata +11 -8
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: c65dc7f1acffb3ae6d361fbd4f1172fb46b857611dc0cc96b9229219869a3576
|
4
|
+
data.tar.gz: 5fa72ea32080be5c77a451bbc9cf24d6e8ff1c83af914c0b97004086cfd76e14
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: 51c93d3f462be29745003941d34a1d8d42da5ff6ab8127c59ae4fc9c3ab001cf23094cc5f298262b63d661507bb5e79d93657fce24ea1fd1c82d8de7d7e5ac7f
|
7
|
+
data.tar.gz: 3abcbcda1957bf7415d10fc5c142183ca0227925843b6a1aeebd9c5a158fc5eeed0b855963c60cd785ce4201ac13dcbc169d27cdfe6d3dd12d9b41798d0e7931
|
data/LICENSE.md
ADDED
@@ -0,0 +1,21 @@
|
|
1
|
+
The MIT License (MIT)
|
2
|
+
|
3
|
+
Copyright (c) Cuong Giang
|
4
|
+
|
5
|
+
Permission is hereby granted, free of charge, to any person obtaining a copy
|
6
|
+
of this software and associated documentation files (the "Software"), to deal
|
7
|
+
in the Software without restriction, including without limitation the rights
|
8
|
+
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
9
|
+
copies of the Software, and to permit persons to whom the Software is
|
10
|
+
furnished to do so, subject to the following conditions:
|
11
|
+
|
12
|
+
The above copyright notice and this permission notice shall be included in
|
13
|
+
all copies or substantial portions of the Software.
|
14
|
+
|
15
|
+
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
16
|
+
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
17
|
+
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
18
|
+
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
19
|
+
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
20
|
+
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
21
|
+
THE SOFTWARE.
|
data/README.md
CHANGED
@@ -2,29 +2,41 @@
|
|
2
2
|
|
3
3
|
# Kaze
|
4
4
|
|
5
|
-
Heavily inspired by [Laravel Breeze](https://github.com/laravel/breeze), this gem offers authentication and application starter kits to give you a head start building your new Rails application. These kits automatically scaffold your application with the routes, controllers, and views you need to register and authenticate your application's users.
|
5
|
+
Heavily inspired by [Laravel Breeze](https://github.com/laravel/breeze), this gem offers authentication and application starter kits to give you a head start building your new [Rails](https://rubyonrails.org) application. These kits automatically scaffold your application with the routes, controllers, and views you need to register and authenticate your application's users.
|
6
6
|
|
7
|
-
[Kaze](https://github.com/gtkvn/kaze) is a minimal, simple implementation of all
|
7
|
+
[Kaze](https://github.com/gtkvn/kaze) is a opinionated minimal, simple implementation of all authentication features that a modern web application should have, including login, registration, password reset, email verification. In addition, Kaze includes a simple "profile" page where the user may update their name, email address, and password.
|
8
8
|
|
9
9
|
Kaze provides scaffolding options based on [Hotwire](https://hotwired.dev) or [Inertia](https://inertiajs.com), with the choice of using Vue or React for the Inertia-based scaffolding.
|
10
10
|
|
11
11
|
## Installation
|
12
12
|
|
13
|
-
|
13
|
+
Before creating your first project powered by Kaze, make sure that your local machine has Ruby installed. Ruby can be installed in minutes via [mise](https://mise.jdx.dev).
|
14
14
|
|
15
|
-
|
15
|
+
```
|
16
|
+
mise use -g ruby@3.3
|
17
|
+
```
|
18
|
+
|
19
|
+
After you have installed Ruby, you may install `rails` and `kaze` gems globally:
|
16
20
|
|
17
21
|
```
|
22
|
+
gem install rails
|
18
23
|
gem install kaze
|
19
24
|
```
|
20
25
|
|
21
|
-
|
26
|
+
Then, you may create a new Rails application:
|
27
|
+
|
28
|
+
```
|
29
|
+
rails new kaze-example-app
|
30
|
+
cd kaze-example-app
|
31
|
+
```
|
32
|
+
|
33
|
+
Once the project has been created, you may scaffold your application using one of the Kaze "stacks" discussed in the documentation below.
|
22
34
|
|
23
|
-
## Kaze
|
35
|
+
## Kaze and Hotwire
|
24
36
|
|
25
37
|
The default Kaze "stack" is the Hotwire stack. Hotwire is a powerful way to building dynamic, reactive, front-end UIs primarily using Ruby and ERB templates without using much JavaScript by sending HTML instead of JSON over the wire.
|
26
38
|
|
27
|
-
The Hotwire stack may be installed by invoking the `install` command with no other additional arguments inside your app directory.
|
39
|
+
The Hotwire stack may be installed by invoking the `kaze install` command with no other additional arguments inside your app directory. This command publishes the authentication, views, routes, controllers, and other resources to your application.
|
28
40
|
|
29
41
|
```
|
30
42
|
kaze install
|
@@ -37,13 +49,13 @@ bin/setup
|
|
37
49
|
bin/dev
|
38
50
|
```
|
39
51
|
|
40
|
-
Next, you may navigate to your application's `/login` or `/register` URLs in your web browser.
|
52
|
+
Next, you may navigate to your application's `/login` or `/register` URLs in your web browser. All of Kaze's routes are defined within the `config/routes.rb` file.
|
41
53
|
|
42
|
-
## Kaze
|
54
|
+
## Kaze and React / Vue
|
43
55
|
|
44
56
|
Kaze offers React and Vue scaffolding via an Inertia frontend implementation. Inertia allows you to build modern, single-page React and Vue applications using classic server-side routing and controllers.
|
45
57
|
|
46
|
-
Inertia lets you enjoy the frontend power of React and Vue combined with the incredible backend productivity of Rails and lightning-fast Vite compilation. To use an Inertia stack, specify `react` or `vue` as your desired stack when executing the `install` command
|
58
|
+
Inertia lets you enjoy the frontend power of React and Vue combined with the incredible backend productivity of Rails and lightning-fast Vite compilation. To use an Inertia stack, specify `react` or `vue` as your desired stack when executing the `kaze install` command:
|
47
59
|
|
48
60
|
```
|
49
61
|
kaze install react
|
@@ -60,4 +72,4 @@ bin/setup
|
|
60
72
|
bin/dev
|
61
73
|
```
|
62
74
|
|
63
|
-
Next, you may navigate to your application's `/login` or `/register` URLs in your web browser.
|
75
|
+
Next, you may navigate to your application's `/login` or `/register` URLs in your web browser. All of Kaze's routes are defined within the `config/routes.rb` file.
|
@@ -11,17 +11,11 @@ class Kaze::Commands::InstallCommand < Thor
|
|
11
11
|
def install(stack = 'hotwire')
|
12
12
|
return say 'Kaze must be run in a new Rails application.', :red unless File.exist?("#{Dir.pwd}/bin/rails")
|
13
13
|
|
14
|
-
if stack == 'hotwire'
|
15
|
-
return install_hotwire_stack
|
16
|
-
end
|
14
|
+
return install_hotwire_stack if stack == 'hotwire'
|
17
15
|
|
18
|
-
if stack == 'react'
|
19
|
-
return install_inertia_react_stack
|
20
|
-
end
|
16
|
+
return install_inertia_react_stack if stack == 'react'
|
21
17
|
|
22
|
-
if stack == 'vue'
|
23
|
-
return install_inertia_vue_stack
|
24
|
-
end
|
18
|
+
return install_inertia_vue_stack if stack == 'vue'
|
25
19
|
|
26
20
|
say 'Invalid stack. Supported stacks are [hotwire], [react], [vue].', :red
|
27
21
|
end
|
@@ -53,14 +47,14 @@ class Kaze::Commands::InstallCommand < Thor
|
|
53
47
|
end
|
54
48
|
|
55
49
|
def install_migrations
|
56
|
-
ensure_directory_exists("#{Dir.pwd}/db/migrate")
|
57
|
-
FileUtils.copy_entry("#{File.dirname(__FILE__)}/../../../stubs/default/db/migrate", "#{Dir.pwd}/db/migrate")
|
58
50
|
stdin, _ = Open3.capture3("#{Dir.pwd}/bin/rails version")
|
59
51
|
versions = stdin.gsub!('Rails ', '').split('.')
|
60
52
|
railsVersion = [ versions[0], versions[1] ].join('.')
|
53
|
+
|
54
|
+
ensure_directory_exists("#{Dir.pwd}/db/migrate")
|
55
|
+
FileUtils.copy_entry("#{File.dirname(__FILE__)}/../../../stubs/default/db/migrate", "#{Dir.pwd}/db/migrate")
|
61
56
|
Dir.children("#{Dir.pwd}/db/migrate").each do |file|
|
62
|
-
|
63
|
-
File.write(path, File.read(path).gsub!(/ActiveRecord::Migration$/, "ActiveRecord::Migration[#{railsVersion}]"))
|
57
|
+
replace_in_file(/ActiveRecord::Migration$/, "ActiveRecord::Migration[#{railsVersion}]", "#{Dir.pwd}/db/migrate/#{file}")
|
64
58
|
end
|
65
59
|
end
|
66
60
|
|
@@ -76,15 +70,19 @@ class Kaze::Commands::InstallCommand < Thor
|
|
76
70
|
FileUtils.mkdir_p(path) unless File.directory?(path)
|
77
71
|
end
|
78
72
|
|
73
|
+
def replace_in_file(search, replace, path)
|
74
|
+
File.write(path, File.read(path).gsub!(search, replace))
|
75
|
+
end
|
76
|
+
|
79
77
|
def run_command(command)
|
80
78
|
Open3.popen3(command) do |stdin, stdout, stderr, wait_thr|
|
81
79
|
stdout_thread = Thread.new do
|
82
|
-
while
|
80
|
+
while line = stdout.gets do
|
83
81
|
say line
|
84
82
|
end
|
85
83
|
end
|
86
84
|
stderr_thread = Thread.new do
|
87
|
-
while
|
85
|
+
while line = stderr.gets do
|
88
86
|
say line, :red
|
89
87
|
end
|
90
88
|
end
|
@@ -60,7 +60,7 @@ module Kaze::Commands::InstallsInertiaStacks
|
|
60
60
|
FileUtils.copy_file("#{File.dirname(__FILE__)}/../../../stubs/default/bin/dev", "#{Dir.pwd}/bin/dev")
|
61
61
|
FileUtils.copy_file("#{File.dirname(__FILE__)}/../../../stubs/inertia-common/bin/vite", "#{Dir.pwd}/bin/vite")
|
62
62
|
FileUtils.copy_file("#{File.dirname(__FILE__)}/../../../stubs/default/Procfile.dev", "#{Dir.pwd}/Procfile.dev")
|
63
|
-
File.write("#{Dir.pwd}/Procfile.dev", "#{File.read("#{Dir.pwd}/Procfile.dev")}\nvite:
|
63
|
+
File.write("#{Dir.pwd}/Procfile.dev", "#{File.read("#{Dir.pwd}/Procfile.dev")}\nvite: bin/vite dev\n")
|
64
64
|
run_command("#{Dir.pwd}/bin/rails generate js_routes:middleware")
|
65
65
|
run_command("#{Dir.pwd}/bin/rails tailwindcss:build")
|
66
66
|
|
@@ -140,7 +140,7 @@ module Kaze::Commands::InstallsInertiaStacks
|
|
140
140
|
FileUtils.copy_file("#{File.dirname(__FILE__)}/../../../stubs/default/bin/dev", "#{Dir.pwd}/bin/dev")
|
141
141
|
FileUtils.copy_file("#{File.dirname(__FILE__)}/../../../stubs/inertia-common/bin/vite", "#{Dir.pwd}/bin/vite")
|
142
142
|
FileUtils.copy_file("#{File.dirname(__FILE__)}/../../../stubs/default/Procfile.dev", "#{Dir.pwd}/Procfile.dev")
|
143
|
-
File.write("#{Dir.pwd}/Procfile.dev", "#{File.read("#{Dir.pwd}/Procfile.dev")}\nvite:
|
143
|
+
File.write("#{Dir.pwd}/Procfile.dev", "#{File.read("#{Dir.pwd}/Procfile.dev")}\nvite: bin/vite dev\n")
|
144
144
|
run_command("#{Dir.pwd}/bin/rails generate js_routes:middleware")
|
145
145
|
run_command("#{Dir.pwd}/bin/rails tailwindcss:build")
|
146
146
|
|
data/lib/kaze/version.rb
CHANGED
@@ -1,11 +1,11 @@
|
|
1
1
|
class CreateUsers < ActiveRecord::Migration
|
2
2
|
def self.up
|
3
3
|
create_table :users do |table|
|
4
|
-
table.string :name
|
5
|
-
table.string :email
|
6
|
-
table.timestamp :email_verified_at
|
7
|
-
table.string :password_digest
|
8
|
-
table.string :remember_token
|
4
|
+
table.string :name, null: false
|
5
|
+
table.string :email, null: false
|
6
|
+
table.timestamp :email_verified_at
|
7
|
+
table.string :password_digest, null: false
|
8
|
+
table.string :remember_token
|
9
9
|
table.timestamps
|
10
10
|
end
|
11
11
|
end
|
@@ -8,7 +8,7 @@ class ModalComponent < ViewComponent::Base
|
|
8
8
|
lg: 'sm:max-w-lg',
|
9
9
|
xl: 'sm:max-w-xl',
|
10
10
|
'2xl': 'sm:max-w-2xl'
|
11
|
-
}[attributes[:max_width] || '2xl']
|
11
|
+
}[(attributes[:max_width] || '2xl').to_sym]
|
12
12
|
@attributes = attributes.without(:name, :show, :max_width)
|
13
13
|
end
|
14
14
|
end
|
@@ -22,7 +22,7 @@ class ProfileController < ApplicationController
|
|
22
22
|
def destroy
|
23
23
|
@delete_user_form = DeleteUserForm.new(params.permit(:password))
|
24
24
|
|
25
|
-
return render partial: 'profile/partials/
|
25
|
+
return render turbo_stream: turbo_stream.replace('confirm_user_deletion_modal', partial: 'profile/partials/confirm_user_deletion_modal'), status: :unprocessable_entity if @delete_user_form.invalid?
|
26
26
|
|
27
27
|
user = Current.auth.user
|
28
28
|
|
@@ -0,0 +1,25 @@
|
|
1
|
+
<div id="confirm_user_deletion_modal">
|
2
|
+
<%= render(ModalComponent.new({ name: "confirm-user-deletion", show: @delete_user_form.error_messages.has_key?(:password), focusable: true })) do %>
|
3
|
+
<%= form_with model: @delete_user_form, url: profile_destroy_path, method: "delete", class: "p-6" do %>
|
4
|
+
<h2 class="text-lg font-medium text-gray-900 dark:text-gray-100">
|
5
|
+
Are you sure you want to delete your account?
|
6
|
+
</h2>
|
7
|
+
<p class="mt-1 text-sm text-gray-600 dark:text-gray-400">
|
8
|
+
Once your account is deleted, all of its resources and data will be permanently deleted. Please enter your password to confirm you would like to permanently delete your account.
|
9
|
+
</p>
|
10
|
+
<div class="mt-6">
|
11
|
+
<%= render(InputLabelComponent.new({ for: "password", value: "Password", class: "sr-only" })) %>
|
12
|
+
<%= render(TextInputComponent.new({ id: "password", class: "mt-1 block w-3/4", type: "password", name: "password", placeholder: "Password" })) %>
|
13
|
+
<%= render(InputErrorComponent.new({ class: "mt-2", message: @delete_user_form.error_messages[:password] })) %>
|
14
|
+
</div>
|
15
|
+
<div class="mt-6 flex justify-end">
|
16
|
+
<%= render(SecondaryButtonComponent.new({ "x-on:click": "$dispatch('close')" })) do %>
|
17
|
+
Cancel
|
18
|
+
<% end %>
|
19
|
+
<%= render(DangerButtonComponent.new({ class: "ms-3" })) do %>
|
20
|
+
Delete Account
|
21
|
+
<% end %>
|
22
|
+
</div>
|
23
|
+
<% end %>
|
24
|
+
<% end %>
|
25
|
+
</div>
|
@@ -10,29 +10,5 @@
|
|
10
10
|
<%= render(DangerButtonComponent.new({"x-data": "", "x-on:click.prevent": "$dispatch('open-modal', 'confirm-user-deletion')"})) do %>
|
11
11
|
Delete Account
|
12
12
|
<% end %>
|
13
|
-
<%= render
|
14
|
-
<turbo-frame id="delete_user_form">
|
15
|
-
<%= form_with model: @delete_user_form, url: profile_destroy_path, method: "delete", class: "p-6" do %>
|
16
|
-
<h2 class="text-lg font-medium text-gray-900 dark:text-gray-100">
|
17
|
-
Are you sure you want to delete your account?
|
18
|
-
</h2>
|
19
|
-
<p class="mt-1 text-sm text-gray-600 dark:text-gray-400">
|
20
|
-
Once your account is deleted, all of its resources and data will be permanently deleted. Please enter your password to confirm you would like to permanently delete your account.
|
21
|
-
</p>
|
22
|
-
<div class="mt-6">
|
23
|
-
<%= render(InputLabelComponent.new({ for: "password", value: "Password", class: "sr-only" })) %>
|
24
|
-
<%= render(TextInputComponent.new({ id: "password", class: "mt-1 block w-3/4", type: "password", name: "password", placeholder: "Password" })) %>
|
25
|
-
<%= render(InputErrorComponent.new({ class: "mt-2", message: @delete_user_form.error_messages[:password] })) %>
|
26
|
-
</div>
|
27
|
-
<div class="mt-6 flex justify-end">
|
28
|
-
<%= render(SecondaryButtonComponent.new({ "x-on:click": "$dispatch('close')" })) do %>
|
29
|
-
Cancel
|
30
|
-
<% end %>
|
31
|
-
<%= render(DangerButtonComponent.new({ class: "ms-3" })) do %>
|
32
|
-
Delete Account
|
33
|
-
<% end %>
|
34
|
-
</div>
|
35
|
-
<% end %>
|
36
|
-
</turbo-frame>
|
37
|
-
<% end %>
|
13
|
+
<%= render 'profile/partials/confirm_user_deletion_modal' %>
|
38
14
|
</section>
|
metadata
CHANGED
@@ -1,14 +1,14 @@
|
|
1
1
|
--- !ruby/object:Gem::Specification
|
2
2
|
name: kaze
|
3
3
|
version: !ruby/object:Gem::Version
|
4
|
-
version: 0.
|
4
|
+
version: 0.12.0
|
5
5
|
platform: ruby
|
6
6
|
authors:
|
7
7
|
- Cuong Giang
|
8
8
|
autorequire:
|
9
9
|
bindir: bin
|
10
10
|
cert_chain: []
|
11
|
-
date: 2024-06-
|
11
|
+
date: 2024-06-28 00:00:00.000000000 Z
|
12
12
|
dependencies:
|
13
13
|
- !ruby/object:Gem::Dependency
|
14
14
|
name: thor
|
@@ -16,28 +16,28 @@ dependencies:
|
|
16
16
|
requirements:
|
17
17
|
- - "~>"
|
18
18
|
- !ruby/object:Gem::Version
|
19
|
-
version: '1.
|
19
|
+
version: '1.3'
|
20
20
|
type: :runtime
|
21
21
|
prerelease: false
|
22
22
|
version_requirements: !ruby/object:Gem::Requirement
|
23
23
|
requirements:
|
24
24
|
- - "~>"
|
25
25
|
- !ruby/object:Gem::Version
|
26
|
-
version: '1.
|
26
|
+
version: '1.3'
|
27
27
|
- !ruby/object:Gem::Dependency
|
28
28
|
name: zeitwerk
|
29
29
|
requirement: !ruby/object:Gem::Requirement
|
30
30
|
requirements:
|
31
31
|
- - "~>"
|
32
32
|
- !ruby/object:Gem::Version
|
33
|
-
version: '2.
|
33
|
+
version: '2.6'
|
34
34
|
type: :runtime
|
35
35
|
prerelease: false
|
36
36
|
version_requirements: !ruby/object:Gem::Requirement
|
37
37
|
requirements:
|
38
38
|
- - "~>"
|
39
39
|
- !ruby/object:Gem::Version
|
40
|
-
version: '2.
|
40
|
+
version: '2.6'
|
41
41
|
- !ruby/object:Gem::Dependency
|
42
42
|
name: railties
|
43
43
|
requirement: !ruby/object:Gem::Requirement
|
@@ -60,6 +60,7 @@ executables:
|
|
60
60
|
extensions: []
|
61
61
|
extra_rdoc_files: []
|
62
62
|
files:
|
63
|
+
- LICENSE.md
|
63
64
|
- README.md
|
64
65
|
- bin/kaze
|
65
66
|
- lib/kaze.rb
|
@@ -150,6 +151,7 @@ files:
|
|
150
151
|
- stubs/hotwire/app/views/layouts/application.html.erb
|
151
152
|
- stubs/hotwire/app/views/layouts/guest.html.erb
|
152
153
|
- stubs/hotwire/app/views/profile/edit.html.erb
|
154
|
+
- stubs/hotwire/app/views/profile/partials/_confirm_user_deletion_modal.html.erb
|
153
155
|
- stubs/hotwire/app/views/profile/partials/_delete_user_form.html.erb
|
154
156
|
- stubs/hotwire/app/views/profile/partials/_update_password_form.html.erb
|
155
157
|
- stubs/hotwire/app/views/profile/partials/_update_profile_information_form.html.erb
|
@@ -260,7 +262,7 @@ required_ruby_version: !ruby/object:Gem::Requirement
|
|
260
262
|
requirements:
|
261
263
|
- - ">="
|
262
264
|
- !ruby/object:Gem::Version
|
263
|
-
version:
|
265
|
+
version: 3.1.0
|
264
266
|
required_rubygems_version: !ruby/object:Gem::Requirement
|
265
267
|
requirements:
|
266
268
|
- - ">="
|
@@ -270,5 +272,6 @@ requirements: []
|
|
270
272
|
rubygems_version: 3.5.9
|
271
273
|
signing_key:
|
272
274
|
specification_version: 4
|
273
|
-
summary:
|
275
|
+
summary: Opinionated minimal Rails authentication scaffolding with Hotwire, React,
|
276
|
+
or Vue + Tailwind.
|
274
277
|
test_files: []
|