rapid_stack 1.0.4 → 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: a0f811ef1816391b9f0efbb459a472d19735071288c70b214e2d84e65bb4b866
4
- data.tar.gz: bd85ce600a38b610fd60754bce4d88ddefa537e47d4644425b8f44cb251450b0
3
+ metadata.gz: 0763d9f0a35056c31bb84c5a0689be33fcb18325ba78675255d14d26114543b8
4
+ data.tar.gz: 3302f2d5138ac5497ee77a293fd52a5f693c7894bfbd0bff1a0afa9f12ee6489
5
5
  SHA512:
6
- metadata.gz: 23ca2a82b234bb502fd825cb21ea77eb6f46e7cdcc362b5de34d67e4b0cbf93f017cced1e429a41a70fa347c489d8a45314d76b6aec19c1184a206e46864676e
7
- data.tar.gz: 647787a1ac4904a82e4acbfe2fd5c76bbbf3004b9f35818c4ac64ac1e2be48ea78e63f1de127a861d1fba95d0b901fad915e15b094b6eeed7804aebdf531e7d3
6
+ metadata.gz: 83cdb6c08e7887b26449de7bee95045d6a2ebe36fdf256d7f611d8afc1c483100b6c67ef1664525c7923bcb32eba7b301d696b98174f69622025436abe1df784
7
+ data.tar.gz: 39d36e1ee38c35cac2dee6811a42ea10e031523ca3839adaed47ebaef508cf020e989dcc066c37ce8630d8ab6b2f4090fee58273b70a814de6b781e6482cb22a
data/CHANGELOG.md CHANGED
@@ -1,3 +1,17 @@
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
+
11
+ ## [1.0.5] - 2025-04-03
12
+ ### Fixed
13
+ - remove npm as gem dependency
14
+
1
15
  ## [1.0.4] - 2025-04-03
2
16
  ### Added
3
17
  - Ability to generate company with frontend:company
@@ -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.4"
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.4
4
+ version: 1.0.6
5
5
  platform: ruby
6
6
  authors:
7
7
  - Kingsley Ijomah
@@ -23,20 +23,6 @@ dependencies:
23
23
  - - "~>"
24
24
  - !ruby/object:Gem::Version
25
25
  version: '2.7'
26
- - !ruby/object:Gem::Dependency
27
- name: npm
28
- requirement: !ruby/object:Gem::Requirement
29
- requirements:
30
- - - "~>"
31
- - !ruby/object:Gem::Version
32
- version: 0.1.0
33
- type: :runtime
34
- prerelease: false
35
- version_requirements: !ruby/object:Gem::Requirement
36
- requirements:
37
- - - "~>"
38
- - !ruby/object:Gem::Version
39
- version: 0.1.0
40
26
  - !ruby/object:Gem::Dependency
41
27
  name: rails
42
28
  requirement: !ruby/object:Gem::Requirement
@@ -15846,9 +15832,9 @@ licenses:
15846
15832
  - MIT
15847
15833
  metadata:
15848
15834
  homepage_uri: https://github.com/kingsley-ijomah/rapid_stack
15849
- source_code_uri: https://github.com/kingsley-ijomah/rapid_stack/tree/v1.0.4
15850
- changelog_uri: https://github.com/kingsley-ijomah/rapid_stack/blob/v1.0.4/CHANGELOG.md
15851
- documentation_uri: https://github.com/kingsley-ijomah/rapid_stack/blob/v1.0.4/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
15852
15838
  post_install_message: |
15853
15839
  =======================================
15854
15840
  Thank you for installing Rapid Stack!
@@ -15858,10 +15844,10 @@ post_install_message: |
15858
15844
  1. Node.js version 18.0.0 or higher
15859
15845
  2. npm version 9.0.0 or higher
15860
15846
 
15861
- To complete the installation and set up the generators:
15862
-
15863
- 1. Run the following command:
15847
+ ###########################################
15848
+ IMPORTANT: Run this command to complete setup:
15864
15849
  rapid_stack_setup
15850
+ ###########################################
15865
15851
 
15866
15852
  For more information, visit: https://github.com/kingsley-ijomah/rapid_stack
15867
15853
  rdoc_options: []