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 CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: 8605fe6f2fc1d82961b7dc3ae84cdf955a6d39b2a0be6240c41925373542a3e1
4
- data.tar.gz: cc70eb26684e84648c8351fff19f37869c4e1bb364984a83ad3be229851632f7
3
+ metadata.gz: 0763d9f0a35056c31bb84c5a0689be33fcb18325ba78675255d14d26114543b8
4
+ data.tar.gz: 3302f2d5138ac5497ee77a293fd52a5f693c7894bfbd0bff1a0afa9f12ee6489
5
5
  SHA512:
6
- metadata.gz: a98a9e5ee829d35e855816ebe85eb23ffdf833308f4ae63ce47e4d362993818b6dee9cb59719edec4266ee3ad251f3fd3d00a5102f17cf2c3ee0958183d8a6df
7
- data.tar.gz: ef12dc2f92e9924d9b8610309c44cbfd2d428aad6054ce2b81ba4dc2e10506e21ba298591372ebfa18068c38ba0f48c32b8666b76f6e04ca165bb40379c3107e
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
@@ -33,6 +33,7 @@ class AuthGenerator extends Generator {
33
33
  'reset_password_token',
34
34
  'reset_password_sent_at',
35
35
  'remember_created_at',
36
+ 'reset_password_token_expires_at',
36
37
  'role',
37
38
  'created_at',
38
39
  'updated_at',
@@ -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
- 'name' => get_secret('app_name', Rails.application.credentials.app&.name),
12
- 'domain' => get_secret('app_domain', Rails.application.credentials.app&.domain),
13
- 'support_email' => get_secret('app_support_email', Rails.application.credentials.app&.support_email)
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
- 'from_address' => get_secret('mailer_from_address', Rails.application.credentials.mailer&.from_address),
20
- 'from_name' => get_secret('mailer_from_name', Rails.application.credentials.mailer&.from_name),
21
- 'support_address' => get_secret('app_support_email', Rails.application.credentials.app&.support_email)
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['JWT_SECRET_KEY']
26
+ ENV["JWT_SECRET_KEY"]
27
27
  end
28
28
 
29
29
  def self.secret_key_base
30
- ENV['SECRET_KEY_BASE']
30
+ ENV["SECRET_KEY_BASE"]
31
31
  end
32
32
 
33
33
  def self.rails_master_key
34
- ENV['RAILS_MASTER_KEY']
34
+ ENV["RAILS_MASTER_KEY"]
35
35
  end
36
36
 
37
37
  def self.postmark_api_token
38
- ENV['POSTMARK_API_TOKEN']
38
+ ENV["POSTMARK_API_TOKEN"]
39
39
  end
40
- end
40
+ end
@@ -94,6 +94,8 @@ class FrontendAuthGenerator extends Generator {
94
94
  'reset_password_token',
95
95
  'reset_password_sent_at',
96
96
  'remember_created_at',
97
+ 'reset_password_token_expires_at',
98
+ 'role',
97
99
  'created_at',
98
100
  'updated_at',
99
101
  '_id',
@@ -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 => arg.name !== 'encrypted_password').forEach(function(arg, index) { %>$<%= snakeToCamel(arg.name) %>: <%= arg.type %><%= arg.required ? '!' : '' %><%= index < args.length - 1 ? ',' : '' %>
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 => arg.name !== 'encrypted_password').forEach(function(arg, index) { %><%= snakeToCamel(arg.name) %>: $<%= snakeToCamel(arg.name) %><%= index < args.length - 1 ? ',' : '' %>
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: () => this.router.navigate(['/auth/verify-otp'], {
82
- queryParams: { email: formData.email }
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 role fields, then separate remaining fields
19
- args.filter(arg => arg.name !== 'encrypted_password' && arg.name !== 'role').forEach(function(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')) {
@@ -61,6 +61,10 @@
61
61
  --border-color: transparent;
62
62
  --highlight-color-focused: transparent;
63
63
 
64
+ ion-checkbox {
65
+ max-width: 30px;
66
+ }
67
+
64
68
  ion-label {
65
69
  margin-left: 8px;
66
70
  font-size: 14px;
@@ -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 => arg.name !== 'encrypted_password' && arg.name !== 'role').forEach(function(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 => arg.name !== 'encrypted_password' && arg.name !== 'role').forEach(function(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
@@ -1,5 +1,5 @@
1
1
  # frozen_string_literal: true
2
2
 
3
3
  module RapidStack
4
- VERSION = "1.0.5"
4
+ VERSION = "1.0.6"
5
5
  end
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.5
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.5
15836
- changelog_uri: https://github.com/kingsley-ijomah/rapid_stack/blob/v1.0.5/CHANGELOG.md
15837
- documentation_uri: https://github.com/kingsley-ijomah/rapid_stack/blob/v1.0.5/README.md
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
- To complete the installation and set up the generators:
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: []