rapid_stack 1.0.5 → 1.0.6
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/CHANGELOG.md +10 -0
- data/lib/rapid_stack/generators/generator-rapid/generators/backend-auth/index.js +1 -0
- data/lib/rapid_stack/generators/generator-rapid/generators/backend-schema-runner/templates/user.rb.template +5 -0
- data/lib/rapid_stack/generators/generator-rapid/generators/build-backend/index.js +32 -0
- data/lib/rapid_stack/generators/generator-rapid/generators/build-backend/templates/lib/config_helper.rb.erb +12 -12
- data/lib/rapid_stack/generators/generator-rapid/generators/frontend-auth/index.js +2 -0
- data/lib/rapid_stack/generators/generator-rapid/generators/frontend-auth/templates/createUser.mutation.ts.ejs +14 -2
- data/lib/rapid_stack/generators/generator-rapid/generators/frontend-auth/templates/pages/forgot-password/forgot-password.page.ts.ejs +9 -3
- data/lib/rapid_stack/generators/generator-rapid/generators/frontend-auth/templates/pages/signup/signup.page.html.ejs +8 -2
- data/lib/rapid_stack/generators/generator-rapid/generators/frontend-auth/templates/pages/signup/signup.page.scss.ejs +4 -0
- data/lib/rapid_stack/generators/generator-rapid/generators/frontend-auth/templates/pages/signup/signup.page.ts.ejs +14 -2
- data/lib/rapid_stack/generators/generator-rapid/generators/frontend-list-action/index.js +3 -1
- data/lib/rapid_stack/generators/generator-rapid/generators/graphql/index.js +3 -1
- data/lib/rapid_stack/version.rb +1 -1
- metadata +7 -7
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: 0763d9f0a35056c31bb84c5a0689be33fcb18325ba78675255d14d26114543b8
|
4
|
+
data.tar.gz: 3302f2d5138ac5497ee77a293fd52a5f693c7894bfbd0bff1a0afa9f12ee6489
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: 83cdb6c08e7887b26449de7bee95045d6a2ebe36fdf256d7f611d8afc1c483100b6c67ef1664525c7923bcb32eba7b301d696b98174f69622025436abe1df784
|
7
|
+
data.tar.gz: 39d36e1ee38c35cac2dee6811a42ea10e031523ca3839adaed47ebaef508cf020e989dcc066c37ce8630d8ab6b2f4090fee58273b70a814de6b781e6482cb22a
|
data/CHANGELOG.md
CHANGED
@@ -1,3 +1,13 @@
|
|
1
|
+
## [1.0.6] - 2025-04-03
|
2
|
+
### Added
|
3
|
+
- In dev mode forgot password will auto launch letter opener
|
4
|
+
|
5
|
+
### Changed
|
6
|
+
- make installation clearer in gemspec
|
7
|
+
|
8
|
+
### Fixed
|
9
|
+
- Fixed issue with dev environment not sending mail with letter opener
|
10
|
+
|
1
11
|
## [1.0.5] - 2025-04-03
|
2
12
|
### Fixed
|
3
13
|
- remove npm as gem dependency
|
@@ -15,6 +15,11 @@ class User
|
|
15
15
|
<%- roleEnum %>
|
16
16
|
}.freeze
|
17
17
|
|
18
|
+
# Add these fields for password recovery
|
19
|
+
field :reset_password_token, type: String
|
20
|
+
field :reset_password_sent_at, type: Time
|
21
|
+
field :reset_password_token_expires_at, type: Time
|
22
|
+
|
18
23
|
# Fields
|
19
24
|
<%
|
20
25
|
// Find polymorphic fields
|
@@ -659,6 +659,38 @@ module.exports = class extends Generator {
|
|
659
659
|
this.fs.commit(() => {
|
660
660
|
this._debugLog('Email templates copied successfully');
|
661
661
|
});
|
662
|
+
|
663
|
+
// Update development.rb with mailer configurations
|
664
|
+
const devConfigPath = this.destinationPath('config/environments/development.rb');
|
665
|
+
if (fs.existsSync(devConfigPath)) {
|
666
|
+
let content = fs.readFileSync(devConfigPath, 'utf8');
|
667
|
+
|
668
|
+
// Define the mailer configurations we want to set
|
669
|
+
const mailerConfigs = [
|
670
|
+
{ key: 'config.action_mailer.delivery_method', value: ':letter_opener_web' },
|
671
|
+
{ key: 'config.action_mailer.perform_deliveries', value: 'true' },
|
672
|
+
{ key: 'config.action_mailer.raise_delivery_errors', value: 'true' }
|
673
|
+
];
|
674
|
+
|
675
|
+
// Process each configuration
|
676
|
+
mailerConfigs.forEach(config => {
|
677
|
+
const regex = new RegExp(`^\\s*${config.key}\\s*=\\s*[^\\n]+$`, 'm');
|
678
|
+
const newLine = ` ${config.key} = ${config.value}`;
|
679
|
+
|
680
|
+
if (content.match(regex)) {
|
681
|
+
// Replace existing configuration
|
682
|
+
content = content.replace(regex, newLine);
|
683
|
+
} else {
|
684
|
+
// Add new configuration after the first config.action_mailer line
|
685
|
+
const insertRegex = /^(\s*config\.action_mailer\.[^\n]+)$/m;
|
686
|
+
content = content.replace(insertRegex, `$1\n${newLine}`);
|
687
|
+
}
|
688
|
+
});
|
689
|
+
|
690
|
+
// Write the updated content back to the file
|
691
|
+
fs.writeFileSync(devConfigPath, content, 'utf8');
|
692
|
+
this.log(`✓ Updated mailer configurations in development.rb`);
|
693
|
+
}
|
662
694
|
} catch (error) {
|
663
695
|
this.log('Error setting up mailers:', error.message);
|
664
696
|
if (this.options.debug) {
|
@@ -2,39 +2,39 @@
|
|
2
2
|
module ConfigHelper
|
3
3
|
def self.get_secret(path, default = nil)
|
4
4
|
# Convert path to uppercase and replace special characters for ENV var format
|
5
|
-
env_key = path.upcase.gsub(/[^A-Z0-9_]/,
|
5
|
+
env_key = path.upcase.gsub(/[^A-Z0-9_]/, "_")
|
6
6
|
ENV[env_key] || default
|
7
7
|
end
|
8
8
|
|
9
9
|
def self.app_credentials
|
10
10
|
{
|
11
|
-
|
12
|
-
|
13
|
-
|
11
|
+
"name" => get_secret("app_name", "Rapid Stack"),
|
12
|
+
"domain" => get_secret("app_domain", "example.com"),
|
13
|
+
"support_email" => get_secret("app_support_email", "support@example.com")
|
14
14
|
}
|
15
15
|
end
|
16
16
|
|
17
17
|
def self.mailer_credentials
|
18
18
|
{
|
19
|
-
|
20
|
-
|
21
|
-
|
19
|
+
"from_address" => get_secret("mailer_from_address", "no-reply@example.com"),
|
20
|
+
"from_name" => get_secret("mailer_from_name", "no-reply"),
|
21
|
+
"support_address" => get_secret("app_support_email", "support@example.com")
|
22
22
|
}
|
23
23
|
end
|
24
24
|
|
25
25
|
def self.jwt_secret_key
|
26
|
-
ENV[
|
26
|
+
ENV["JWT_SECRET_KEY"]
|
27
27
|
end
|
28
28
|
|
29
29
|
def self.secret_key_base
|
30
|
-
ENV[
|
30
|
+
ENV["SECRET_KEY_BASE"]
|
31
31
|
end
|
32
32
|
|
33
33
|
def self.rails_master_key
|
34
|
-
ENV[
|
34
|
+
ENV["RAILS_MASTER_KEY"]
|
35
35
|
end
|
36
36
|
|
37
37
|
def self.postmark_api_token
|
38
|
-
ENV[
|
38
|
+
ENV["POSTMARK_API_TOKEN"]
|
39
39
|
end
|
40
|
-
end
|
40
|
+
end
|
@@ -2,12 +2,24 @@ import gql from 'graphql-tag';
|
|
2
2
|
|
3
3
|
export const CreateUserMutation = gql`
|
4
4
|
mutation CreateUser(
|
5
|
-
<% args.filter(arg =>
|
5
|
+
<% args.filter(arg =>
|
6
|
+
arg.name !== 'encrypted_password' &&
|
7
|
+
arg.name !== 'role' &&
|
8
|
+
arg.name !== 'reset_password_token' &&
|
9
|
+
arg.name !== 'reset_password_sent_at' &&
|
10
|
+
arg.name !== 'reset_password_token_expires_at'
|
11
|
+
).forEach(function(arg, index) { %>$<%= snakeToCamel(arg.name) %>: <%= arg.type %><%= arg.required ? '!' : '' %><%= index < args.length - 1 ? ',' : '' %>
|
6
12
|
<% }); %>
|
7
13
|
) {
|
8
14
|
createUser(
|
9
15
|
input: {
|
10
|
-
<% args.filter(arg =>
|
16
|
+
<% args.filter(arg =>
|
17
|
+
arg.name !== 'encrypted_password' &&
|
18
|
+
arg.name !== 'role' &&
|
19
|
+
arg.name !== 'reset_password_token' &&
|
20
|
+
arg.name !== 'reset_password_sent_at' &&
|
21
|
+
arg.name !== 'reset_password_token_expires_at'
|
22
|
+
).forEach(function(arg, index) { %><%= snakeToCamel(arg.name) %>: $<%= snakeToCamel(arg.name) %><%= index < args.length - 1 ? ',' : '' %>
|
11
23
|
<% }); %>
|
12
24
|
}
|
13
25
|
) {
|
@@ -5,6 +5,7 @@ import { RouterLink } from '@angular/router';
|
|
5
5
|
import { OtpRequestMutation } from 'src/app/graphql/mutations/auth/otpRequest.mutation';
|
6
6
|
import { Router } from '@angular/router';
|
7
7
|
import { BaseGraphQLPage } from 'src/app/shared/base/base-graphql.page';
|
8
|
+
import { environment } from 'src/environments/environment';
|
8
9
|
import {
|
9
10
|
IonHeader,
|
10
11
|
IonToolbar,
|
@@ -78,9 +79,14 @@ export class ForgotPasswordPage extends BaseGraphQLPage implements OnInit {
|
|
78
79
|
responsePath: 'otpRequest',
|
79
80
|
successMessage: 'OTP sent to your email!',
|
80
81
|
errorMessage: 'Failed to send OTP. Please check your email and try again.',
|
81
|
-
onSuccess: () =>
|
82
|
-
|
83
|
-
|
82
|
+
onSuccess: () => {
|
83
|
+
if (environment.production === false) {
|
84
|
+
window.open('http://localhost:3000/letter_opener/', '_blank');
|
85
|
+
}
|
86
|
+
this.router.navigate(['/auth/verify-otp'], {
|
87
|
+
queryParams: { email: formData.email }
|
88
|
+
});
|
89
|
+
},
|
84
90
|
onError: (error) => this.backendErrors = this.errorService.errors
|
85
91
|
});
|
86
92
|
}
|
@@ -15,8 +15,14 @@
|
|
15
15
|
const regularFields = [];
|
16
16
|
const checkboxFields = [];
|
17
17
|
|
18
|
-
// Filter out encrypted_password and
|
19
|
-
args.filter(arg =>
|
18
|
+
// Filter out encrypted_password, role, and password reset fields, then separate remaining fields
|
19
|
+
args.filter(arg =>
|
20
|
+
arg.name !== 'encrypted_password' &&
|
21
|
+
arg.name !== 'role' &&
|
22
|
+
arg.name !== 'reset_password_token' &&
|
23
|
+
arg.name !== 'reset_password_sent_at' &&
|
24
|
+
arg.name !== 'reset_password_token_expires_at'
|
25
|
+
).forEach(function(arg) {
|
20
26
|
const fieldName = snakeToCamel(arg.name);
|
21
27
|
// Identify checkbox fields by type or naming convention
|
22
28
|
if (arg.type === 'Boolean' || fieldName.includes('accept') || fieldName.includes('agree') || fieldName.includes('consent')) {
|
@@ -55,7 +55,13 @@ export class SignupPage extends BaseGraphQLPage implements OnInit {
|
|
55
55
|
) {
|
56
56
|
super();
|
57
57
|
this.signupForm = this.formBuilder.group({
|
58
|
-
<% args.filter(arg =>
|
58
|
+
<% args.filter(arg =>
|
59
|
+
arg.name !== 'encrypted_password' &&
|
60
|
+
arg.name !== 'role' &&
|
61
|
+
arg.name !== 'reset_password_token' &&
|
62
|
+
arg.name !== 'reset_password_sent_at' &&
|
63
|
+
arg.name !== 'reset_password_token_expires_at'
|
64
|
+
).forEach(function(arg) { %>
|
59
65
|
<%- snakeToCamel(arg.name) %>: [<%- arg.type === 'Boolean' ? false : '\'\'' %>, [<%- arg.required ? 'Validators.required' : '' %><%- arg.type === 'String' ? ', Validators.minLength(3)' : '' %><%- arg.type === 'Boolean' ? ', Validators.pattern(/^(true|false)$/)' : '' %><%- arg.name === 'accept_terms' ? ', Validators.requiredTrue' : '' %>]],
|
60
66
|
<% }); %>
|
61
67
|
}, {
|
@@ -106,7 +112,13 @@ export class SignupPage extends BaseGraphQLPage implements OnInit {
|
|
106
112
|
this.executeMutation({
|
107
113
|
mutation: CreateUserMutation,
|
108
114
|
variables: {
|
109
|
-
<% args.filter(arg =>
|
115
|
+
<% args.filter(arg =>
|
116
|
+
arg.name !== 'encrypted_password' &&
|
117
|
+
arg.name !== 'role' &&
|
118
|
+
arg.name !== 'reset_password_token' &&
|
119
|
+
arg.name !== 'reset_password_sent_at' &&
|
120
|
+
arg.name !== 'reset_password_token_expires_at'
|
121
|
+
).forEach(function(arg) { %>
|
110
122
|
<%= snakeToCamel(arg.name) %>: signupData.<%= snakeToCamel(arg.name) %>,
|
111
123
|
<% }); %>
|
112
124
|
},
|
@@ -399,7 +399,9 @@ module.exports = class extends Generator {
|
|
399
399
|
'encrypted_password',
|
400
400
|
'reset_password_token',
|
401
401
|
'reset_password_sent_at',
|
402
|
-
'remember_created_at'
|
402
|
+
'remember_created_at',
|
403
|
+
'reset_password_token_expires_at',
|
404
|
+
'role'
|
403
405
|
];
|
404
406
|
|
405
407
|
// Filter out relationship fields and sensitive fields for User model
|
@@ -249,7 +249,9 @@ module.exports = class extends Generator {
|
|
249
249
|
'encrypted_password',
|
250
250
|
'reset_password_token',
|
251
251
|
'reset_password_sent_at',
|
252
|
-
'remember_created_at'
|
252
|
+
'remember_created_at',
|
253
|
+
'reset_password_token_expires_at',
|
254
|
+
'role'
|
253
255
|
];
|
254
256
|
|
255
257
|
// Filter out sensitive fields if this is the User model
|
data/lib/rapid_stack/version.rb
CHANGED
metadata
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
--- !ruby/object:Gem::Specification
|
2
2
|
name: rapid_stack
|
3
3
|
version: !ruby/object:Gem::Version
|
4
|
-
version: 1.0.
|
4
|
+
version: 1.0.6
|
5
5
|
platform: ruby
|
6
6
|
authors:
|
7
7
|
- Kingsley Ijomah
|
@@ -15832,9 +15832,9 @@ licenses:
|
|
15832
15832
|
- MIT
|
15833
15833
|
metadata:
|
15834
15834
|
homepage_uri: https://github.com/kingsley-ijomah/rapid_stack
|
15835
|
-
source_code_uri: https://github.com/kingsley-ijomah/rapid_stack/tree/v1.0.
|
15836
|
-
changelog_uri: https://github.com/kingsley-ijomah/rapid_stack/blob/v1.0.
|
15837
|
-
documentation_uri: https://github.com/kingsley-ijomah/rapid_stack/blob/v1.0.
|
15835
|
+
source_code_uri: https://github.com/kingsley-ijomah/rapid_stack/tree/v1.0.6
|
15836
|
+
changelog_uri: https://github.com/kingsley-ijomah/rapid_stack/blob/v1.0.6/CHANGELOG.md
|
15837
|
+
documentation_uri: https://github.com/kingsley-ijomah/rapid_stack/blob/v1.0.6/README.md
|
15838
15838
|
post_install_message: |
|
15839
15839
|
=======================================
|
15840
15840
|
Thank you for installing Rapid Stack!
|
@@ -15844,10 +15844,10 @@ post_install_message: |
|
|
15844
15844
|
1. Node.js version 18.0.0 or higher
|
15845
15845
|
2. npm version 9.0.0 or higher
|
15846
15846
|
|
15847
|
-
|
15848
|
-
|
15849
|
-
1. Run the following command:
|
15847
|
+
###########################################
|
15848
|
+
IMPORTANT: Run this command to complete setup:
|
15850
15849
|
rapid_stack_setup
|
15850
|
+
###########################################
|
15851
15851
|
|
15852
15852
|
For more information, visit: https://github.com/kingsley-ijomah/rapid_stack
|
15853
15853
|
rdoc_options: []
|