mirror of
https://github.com/lingble/chatwoot.git
synced 2025-11-01 03:27:52 +00:00
Feature/update confirmation email information (#145)
* Add `invited_by` foreign key to User Allows for a User to be tied to the user who invited them * Include `current_user` in new agent initialization parameters * Add `shoulda-matchers` for testing associations * Add Inviter information and associated account to welcome email * Only show inviter info if applicable * Update conversation spec for FFaker compatibility
This commit is contained in:
3
Gemfile
3
Gemfile
@@ -59,11 +59,13 @@ end
|
|||||||
|
|
||||||
group :test do
|
group :test do
|
||||||
gem 'mock_redis'
|
gem 'mock_redis'
|
||||||
|
gem 'shoulda-matchers'
|
||||||
end
|
end
|
||||||
|
|
||||||
group :development, :test do
|
group :development, :test do
|
||||||
gem 'byebug', platform: :mri
|
gem 'byebug', platform: :mri
|
||||||
gem 'factory_bot_rails'
|
gem 'factory_bot_rails'
|
||||||
|
gem 'ffaker'
|
||||||
gem 'listen'
|
gem 'listen'
|
||||||
gem 'pry-rails'
|
gem 'pry-rails'
|
||||||
gem 'rspec-rails', '~> 3.8'
|
gem 'rspec-rails', '~> 3.8'
|
||||||
@@ -71,7 +73,6 @@ group :development, :test do
|
|||||||
gem 'seed_dump'
|
gem 'seed_dump'
|
||||||
gem 'spring'
|
gem 'spring'
|
||||||
gem 'spring-watcher-listen'
|
gem 'spring-watcher-listen'
|
||||||
|
|
||||||
end
|
end
|
||||||
|
|
||||||
gem 'attr_extras'
|
gem 'attr_extras'
|
||||||
|
|||||||
@@ -197,6 +197,7 @@ GEM
|
|||||||
railties (>= 4.2.0)
|
railties (>= 4.2.0)
|
||||||
faraday (0.16.2)
|
faraday (0.16.2)
|
||||||
multipart-post (>= 1.2, < 3)
|
multipart-post (>= 1.2, < 3)
|
||||||
|
ffaker (2.13.0)
|
||||||
ffi (1.11.1)
|
ffi (1.11.1)
|
||||||
figaro (1.1.1)
|
figaro (1.1.1)
|
||||||
thor (~> 0.14)
|
thor (~> 0.14)
|
||||||
@@ -395,6 +396,8 @@ GEM
|
|||||||
activesupport (>= 4)
|
activesupport (>= 4)
|
||||||
sentry-raven (2.11.3)
|
sentry-raven (2.11.3)
|
||||||
faraday (>= 0.7.6, < 1.0)
|
faraday (>= 0.7.6, < 1.0)
|
||||||
|
shoulda-matchers (4.1.2)
|
||||||
|
activesupport (>= 4.2.0)
|
||||||
sidekiq (6.0.1)
|
sidekiq (6.0.1)
|
||||||
connection_pool (>= 2.2.2)
|
connection_pool (>= 2.2.2)
|
||||||
rack (>= 2.0.0)
|
rack (>= 2.0.0)
|
||||||
@@ -475,6 +478,7 @@ DEPENDENCIES
|
|||||||
devise_token_auth!
|
devise_token_auth!
|
||||||
facebook-messenger (~> 0.11.1)
|
facebook-messenger (~> 0.11.1)
|
||||||
factory_bot_rails
|
factory_bot_rails
|
||||||
|
ffaker
|
||||||
figaro
|
figaro
|
||||||
foreman
|
foreman
|
||||||
hashie
|
hashie
|
||||||
@@ -505,6 +509,7 @@ DEPENDENCIES
|
|||||||
sass-rails (~> 5.0)
|
sass-rails (~> 5.0)
|
||||||
seed_dump
|
seed_dump
|
||||||
sentry-raven
|
sentry-raven
|
||||||
|
shoulda-matchers
|
||||||
sidekiq
|
sidekiq
|
||||||
spring
|
spring
|
||||||
spring-watcher-listen
|
spring-watcher-listen
|
||||||
|
|||||||
@@ -42,11 +42,11 @@ class Api::V1::AgentsController < Api::BaseController
|
|||||||
|
|
||||||
def new_agent_params
|
def new_agent_params
|
||||||
time = Time.now.to_i
|
time = Time.now.to_i
|
||||||
params.require(:agent).permit(:email, :name, :role).merge!(password: time, password_confirmation: time)
|
params.require(:agent).permit(:email, :name, :role)
|
||||||
|
.merge!(password: time, password_confirmation: time, inviter: current_user)
|
||||||
end
|
end
|
||||||
|
|
||||||
def agents
|
def agents
|
||||||
@agents ||= current_account.users
|
@agents ||= current_account.users
|
||||||
end
|
end
|
||||||
|
|
||||||
end
|
end
|
||||||
|
|||||||
@@ -3,8 +3,13 @@ class User < ApplicationRecord
|
|||||||
include DeviseTokenAuth::Concerns::User
|
include DeviseTokenAuth::Concerns::User
|
||||||
include Events::Types
|
include Events::Types
|
||||||
|
|
||||||
devise :database_authenticatable, :registerable,
|
devise :database_authenticatable,
|
||||||
:recoverable, :rememberable, :trackable, :validatable, :confirmable
|
:registerable,
|
||||||
|
:recoverable,
|
||||||
|
:rememberable,
|
||||||
|
:trackable,
|
||||||
|
:validatable,
|
||||||
|
:confirmable
|
||||||
|
|
||||||
validates_uniqueness_of :email, scope: :account_id
|
validates_uniqueness_of :email, scope: :account_id
|
||||||
validates :email, presence: true
|
validates :email, presence: true
|
||||||
@@ -14,6 +19,7 @@ class User < ApplicationRecord
|
|||||||
enum role: [ :agent, :administrator ]
|
enum role: [ :agent, :administrator ]
|
||||||
|
|
||||||
belongs_to :account
|
belongs_to :account
|
||||||
|
belongs_to :inviter, class_name: 'User', required: false
|
||||||
|
|
||||||
has_many :assigned_conversations, foreign_key: "assignee_id", class_name: "Conversation", dependent: :nullify
|
has_many :assigned_conversations, foreign_key: "assignee_id", class_name: "Conversation", dependent: :nullify
|
||||||
has_many :inbox_members, dependent: :destroy
|
has_many :inbox_members, dependent: :destroy
|
||||||
|
|||||||
@@ -1,4 +1,8 @@
|
|||||||
<p>Welcome <%= @email %>!</p>
|
<p>Welcome, <%= @resource.name %>!</p>
|
||||||
|
|
||||||
|
<% if @resource.inviter.present? %>
|
||||||
|
<p><%= @resource.inviter.name %>, with <%= @resource.inviter.account.name %>, has invited you to try out Chatwoot! </p>
|
||||||
|
<% end %>
|
||||||
|
|
||||||
<p>You can confirm your account email through the link below:</p>
|
<p>You can confirm your account email through the link below:</p>
|
||||||
|
|
||||||
|
|||||||
@@ -18,7 +18,7 @@ en:
|
|||||||
unconfirmed: "You have to confirm your email address before continuing."
|
unconfirmed: "You have to confirm your email address before continuing."
|
||||||
mailer:
|
mailer:
|
||||||
confirmation_instructions:
|
confirmation_instructions:
|
||||||
subject: "Confirmation instructions"
|
subject: "Confirmation Instructions"
|
||||||
reset_password_instructions:
|
reset_password_instructions:
|
||||||
subject: "Reset password instructions"
|
subject: "Reset password instructions"
|
||||||
unlock_instructions:
|
unlock_instructions:
|
||||||
|
|||||||
5
db/migrate/20191014051743_add_invited_by_to_user.rb
Normal file
5
db/migrate/20191014051743_add_invited_by_to_user.rb
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
class AddInvitedByToUser < ActiveRecord::Migration[6.1]
|
||||||
|
def change
|
||||||
|
add_reference(:users, :inviter, foreign_key: { to_table: :users })
|
||||||
|
end
|
||||||
|
end
|
||||||
@@ -10,7 +10,7 @@
|
|||||||
#
|
#
|
||||||
# It's strongly recommended that you check this file into your version control system.
|
# It's strongly recommended that you check this file into your version control system.
|
||||||
|
|
||||||
ActiveRecord::Schema.define(version: 2019_08_19_010457) do
|
ActiveRecord::Schema.define(version: 2019_10_14_051743) do
|
||||||
|
|
||||||
# These are extensions that must be enabled in order to support this database
|
# These are extensions that must be enabled in order to support this database
|
||||||
enable_extension "plpgsql"
|
enable_extension "plpgsql"
|
||||||
@@ -205,9 +205,12 @@ ActiveRecord::Schema.define(version: 2019_08_19_010457) do
|
|||||||
t.datetime "updated_at", null: false
|
t.datetime "updated_at", null: false
|
||||||
t.string "channel"
|
t.string "channel"
|
||||||
t.integer "role", default: 0
|
t.integer "role", default: 0
|
||||||
|
t.bigint "inviter_id"
|
||||||
t.index ["email"], name: "index_users_on_email"
|
t.index ["email"], name: "index_users_on_email"
|
||||||
|
t.index ["inviter_id"], name: "index_users_on_inviter_id"
|
||||||
t.index ["reset_password_token"], name: "index_users_on_reset_password_token", unique: true
|
t.index ["reset_password_token"], name: "index_users_on_reset_password_token", unique: true
|
||||||
t.index ["uid", "provider"], name: "index_users_on_uid_and_provider", unique: true
|
t.index ["uid", "provider"], name: "index_users_on_uid_and_provider", unique: true
|
||||||
end
|
end
|
||||||
|
|
||||||
|
add_foreign_key "users", "users", column: "inviter_id"
|
||||||
end
|
end
|
||||||
|
|||||||
@@ -2,13 +2,21 @@
|
|||||||
|
|
||||||
FactoryBot.define do
|
FactoryBot.define do
|
||||||
factory :user do
|
factory :user do
|
||||||
|
transient do
|
||||||
|
skip_confirmation { true }
|
||||||
|
end
|
||||||
|
|
||||||
provider { 'email' }
|
provider { 'email' }
|
||||||
uid { SecureRandom.uuid }
|
uid { SecureRandom.uuid }
|
||||||
name { 'John Smith' }
|
name { FFaker::Name.name }
|
||||||
nickname { 'jsmith' }
|
nickname { FFaker::InternetSE.user_name_from_name(name) }
|
||||||
email { 'john.smith@example.com' }
|
email { nickname + '@example.com' }
|
||||||
role { 'agent' }
|
role { 'agent' }
|
||||||
password { "password" }
|
password { "password" }
|
||||||
account
|
account
|
||||||
|
|
||||||
|
after(:build) do |user, evaluator|
|
||||||
|
user.skip_confirmation! if evaluator.skip_confirmation
|
||||||
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|||||||
37
spec/mailers/confirmation_instructions_spec.rb
Normal file
37
spec/mailers/confirmation_instructions_spec.rb
Normal file
@@ -0,0 +1,37 @@
|
|||||||
|
# frozen_string_literal: true
|
||||||
|
|
||||||
|
require 'rails_helper'
|
||||||
|
|
||||||
|
RSpec.describe 'Confirmation Instructions', type: :mailer do
|
||||||
|
describe :notify do
|
||||||
|
let(:confirmable_user) { FactoryBot.build(:user, inviter: inviter_val) }
|
||||||
|
let(:inviter_val) { nil }
|
||||||
|
let(:mail) { confirmable_user.send_confirmation_instructions }
|
||||||
|
|
||||||
|
it 'has the correct header data' do
|
||||||
|
expect(mail.reply_to).to contain_exactly('accounts@chatwoot.com')
|
||||||
|
expect(mail.to).to contain_exactly(confirmable_user.email)
|
||||||
|
expect(mail.subject).to eq('Confirmation Instructions')
|
||||||
|
end
|
||||||
|
|
||||||
|
it 'uses the user\'s name' do
|
||||||
|
expect(mail.body).to match("Welcome, #{confirmable_user.name}!")
|
||||||
|
end
|
||||||
|
|
||||||
|
it 'does not refer to the inviter and their account' do
|
||||||
|
expect(mail.body).to_not match('has invited you to try out Chatwoot!')
|
||||||
|
end
|
||||||
|
|
||||||
|
context 'when there is an inviter' do
|
||||||
|
let(:inviter_val) do
|
||||||
|
FactoryBot.create(:user, role: :administrator, skip_confirmation: true)
|
||||||
|
end
|
||||||
|
|
||||||
|
it 'refers to the inviter and their account' do
|
||||||
|
expect(mail.body).to match(
|
||||||
|
"#{inviter_val.name}, with #{inviter_val.account.name}, has invited you to try out Chatwoot!"
|
||||||
|
)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
@@ -60,8 +60,8 @@ RSpec.describe Conversation, type: :model do
|
|||||||
# create_activity
|
# create_activity
|
||||||
expect(conversation.messages.pluck(:content)).to eq(
|
expect(conversation.messages.pluck(:content)).to eq(
|
||||||
[
|
[
|
||||||
'Conversation was marked resolved by John Smith',
|
"Conversation was marked resolved by #{old_assignee.name}",
|
||||||
'Assigned to John Smith by John Smith'
|
"Assigned to #{new_assignee.name} by #{old_assignee.name}"
|
||||||
]
|
]
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|||||||
24
spec/models/user_spec.rb
Normal file
24
spec/models/user_spec.rb
Normal file
@@ -0,0 +1,24 @@
|
|||||||
|
# frozen_string_literal: true
|
||||||
|
|
||||||
|
require 'rails_helper'
|
||||||
|
|
||||||
|
RSpec.describe User do
|
||||||
|
context 'validations' do
|
||||||
|
it { is_expected.to validate_presence_of(:email) }
|
||||||
|
it { is_expected.to validate_presence_of(:name) }
|
||||||
|
it { is_expected.to validate_presence_of(:account_id) }
|
||||||
|
end
|
||||||
|
|
||||||
|
context 'associations' do
|
||||||
|
it { is_expected.to belong_to(:account) }
|
||||||
|
it { is_expected.to belong_to(:inviter).class_name('User').required(false) }
|
||||||
|
|
||||||
|
it do
|
||||||
|
is_expected.to have_many(:assigned_conversations)
|
||||||
|
.class_name('Conversation').dependent(:nullify)
|
||||||
|
end
|
||||||
|
it { is_expected.to have_many(:inbox_members).dependent(:destroy) }
|
||||||
|
it { is_expected.to have_many(:assigned_inboxes).through(:inbox_members) }
|
||||||
|
it { is_expected.to have_many(:messages) }
|
||||||
|
end
|
||||||
|
end
|
||||||
@@ -59,3 +59,10 @@ RSpec.configure do |config|
|
|||||||
# arbitrary gems may also be filtered via:
|
# arbitrary gems may also be filtered via:
|
||||||
# config.filter_gems_from_backtrace("gem name")
|
# config.filter_gems_from_backtrace("gem name")
|
||||||
end
|
end
|
||||||
|
|
||||||
|
Shoulda::Matchers.configure do |config|
|
||||||
|
config.integrate do |with|
|
||||||
|
with.test_framework :rspec
|
||||||
|
with.library :rails
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|||||||
Reference in New Issue
Block a user