feat: exporting contacts takes the filters into account (#9347)
- This PR allows contacts to be exported using the current filter in CRM view Co-authored-by: Sojan Jose <sojan@pepalo.com>
This commit is contained in:
@@ -2,6 +2,16 @@ require 'rails_helper'
|
||||
|
||||
RSpec.describe 'Contacts API', type: :request do
|
||||
let(:account) { create(:account) }
|
||||
let(:email_filter) do
|
||||
{
|
||||
attribute_key: 'email',
|
||||
filter_operator: 'contains',
|
||||
values: 'looped',
|
||||
query_operator: 'and',
|
||||
attribute_model: 'standard',
|
||||
custom_attribute_type: ''
|
||||
}
|
||||
end
|
||||
|
||||
describe 'GET /api/v1/accounts/{account.id}/contacts' do
|
||||
context 'when it is an unauthenticated user' do
|
||||
@@ -175,7 +185,7 @@ RSpec.describe 'Contacts API', type: :request do
|
||||
describe 'POST /api/v1/accounts/{account.id}/contacts/export' do
|
||||
context 'when it is an unauthenticated user' do
|
||||
it 'returns unauthorized' do
|
||||
get "/api/v1/accounts/#{account.id}/contacts/export"
|
||||
post "/api/v1/accounts/#{account.id}/contacts/export"
|
||||
|
||||
expect(response).to have_http_status(:unauthorized)
|
||||
end
|
||||
@@ -185,9 +195,9 @@ RSpec.describe 'Contacts API', type: :request do
|
||||
let(:agent) { create(:user, account: account, role: :agent) }
|
||||
|
||||
it 'returns unauthorized' do
|
||||
get "/api/v1/accounts/#{account.id}/contacts/export",
|
||||
headers: agent.create_new_auth_token,
|
||||
as: :json
|
||||
post "/api/v1/accounts/#{account.id}/contacts/export",
|
||||
headers: agent.create_new_auth_token,
|
||||
as: :json
|
||||
|
||||
expect(response).to have_http_status(:unauthorized)
|
||||
end
|
||||
@@ -197,21 +207,35 @@ RSpec.describe 'Contacts API', type: :request do
|
||||
let(:admin) { create(:user, account: account, role: :administrator) }
|
||||
|
||||
it 'enqueues a contact export job' do
|
||||
expect(Account::ContactsExportJob).to receive(:perform_later).with(account.id, nil, admin.email).once
|
||||
expect(Account::ContactsExportJob).to receive(:perform_later).with(account.id, admin.id, nil, { :payload => nil, :label => nil }).once
|
||||
|
||||
get "/api/v1/accounts/#{account.id}/contacts/export",
|
||||
headers: admin.create_new_auth_token,
|
||||
params: { column_names: nil }
|
||||
post "/api/v1/accounts/#{account.id}/contacts/export",
|
||||
headers: admin.create_new_auth_token
|
||||
|
||||
expect(response).to have_http_status(:success)
|
||||
end
|
||||
|
||||
it 'enqueues a contact export job with sent_columns' do
|
||||
expect(Account::ContactsExportJob).to receive(:perform_later).with(account.id, %w[phone_number email], admin.email).once
|
||||
expect(Account::ContactsExportJob).to receive(:perform_later).with(account.id, admin.id, %w[phone_number email],
|
||||
{ :payload => nil, :label => nil }).once
|
||||
|
||||
get "/api/v1/accounts/#{account.id}/contacts/export",
|
||||
headers: admin.create_new_auth_token,
|
||||
params: { column_names: %w[phone_number email] }
|
||||
post "/api/v1/accounts/#{account.id}/contacts/export",
|
||||
headers: admin.create_new_auth_token,
|
||||
params: { column_names: %w[phone_number email] }
|
||||
|
||||
expect(response).to have_http_status(:success)
|
||||
end
|
||||
|
||||
it 'enqueues a contact export job with payload' do
|
||||
expect(Account::ContactsExportJob).to receive(:perform_later).with(account.id, admin.id, nil,
|
||||
{
|
||||
:payload => [ActionController::Parameters.new(email_filter).permit!],
|
||||
:label => nil
|
||||
}).once
|
||||
|
||||
post "/api/v1/accounts/#{account.id}/contacts/export",
|
||||
headers: admin.create_new_auth_token,
|
||||
params: { payload: [email_filter] }
|
||||
|
||||
expect(response).to have_http_status(:success)
|
||||
end
|
||||
|
||||
@@ -3,7 +3,43 @@ require 'rails_helper'
|
||||
RSpec.describe Account::ContactsExportJob do
|
||||
subject(:job) { described_class.perform_later }
|
||||
|
||||
let!(:account) { create(:account) }
|
||||
let(:account) { create(:account) }
|
||||
let(:user) { create(:user, account: account, email: 'account-user-test@test.com') }
|
||||
let(:label) { create(:label, title: 'spec-billing', maccount: account) }
|
||||
|
||||
let(:email_filter) do
|
||||
{
|
||||
:attribute_key => 'email',
|
||||
:filter_operator => 'contains',
|
||||
:values => 'looped',
|
||||
:query_operator => 'and',
|
||||
:attribute_model => 'standard',
|
||||
:custom_attribute_type => ''
|
||||
}
|
||||
end
|
||||
|
||||
let(:city_filter) do
|
||||
{
|
||||
:attribute_key => 'country_code',
|
||||
:filter_operator => 'equal_to',
|
||||
:values => ['India'],
|
||||
:query_operator => 'and',
|
||||
:attribute_model => 'standard',
|
||||
:custom_attribute_type => ''
|
||||
}
|
||||
end
|
||||
|
||||
let(:single_filter) do
|
||||
{
|
||||
:payload => [email_filter.merge(:query_operator => nil)]
|
||||
}
|
||||
end
|
||||
|
||||
let(:multiple_filters) do
|
||||
{
|
||||
:payload => [city_filter, email_filter.merge(:query_operator => nil)]
|
||||
}
|
||||
end
|
||||
|
||||
it 'enqueues the job' do
|
||||
expect { job }.to have_enqueued_job(described_class)
|
||||
@@ -13,8 +49,11 @@ RSpec.describe Account::ContactsExportJob do
|
||||
context 'when export_contacts' do
|
||||
before do
|
||||
create(:contact, account: account, phone_number: '+910808080818', email: 'test1@text.example')
|
||||
8.times do
|
||||
create(:contact, account: account)
|
||||
4.times do |i|
|
||||
create(:contact, account: account, email: "looped-#{i + 3}@text.example.com")
|
||||
end
|
||||
4.times do |i|
|
||||
create(:contact, account: account, additional_attributes: { :country_code => 'India' }, email: "looped-#{i + 10}@text.example.com")
|
||||
end
|
||||
create(:contact, account: account, phone_number: '+910808080808', email: 'test2@text.example')
|
||||
end
|
||||
@@ -24,17 +63,17 @@ RSpec.describe Account::ContactsExportJob do
|
||||
allow(AdministratorNotifications::ChannelNotificationsMailer).to receive(:with).with(account: account).and_return(mailer)
|
||||
allow(mailer).to receive(:contact_export_complete)
|
||||
|
||||
described_class.perform_now(account.id, [], 'test@test.com')
|
||||
described_class.perform_now(account.id, user.id, [], {})
|
||||
|
||||
file_url = Rails.application.routes.url_helpers.rails_blob_url(account.contacts_export)
|
||||
|
||||
expect(account.contacts_export).to be_present
|
||||
expect(file_url).to be_present
|
||||
expect(mailer).to have_received(:contact_export_complete).with(file_url, 'test@test.com')
|
||||
expect(mailer).to have_received(:contact_export_complete).with(file_url, user.email)
|
||||
end
|
||||
|
||||
it 'generates valid data export file' do
|
||||
described_class.perform_now(account.id, [], 'test@test.com')
|
||||
described_class.perform_now(account.id, user.id, %w[id name email phone_number column_not_present], {})
|
||||
|
||||
csv_data = CSV.parse(account.contacts_export.download, headers: true)
|
||||
emails = csv_data.pluck('email')
|
||||
@@ -45,5 +84,46 @@ RSpec.describe Account::ContactsExportJob do
|
||||
expect(emails).to include('test1@text.example', 'test2@text.example')
|
||||
expect(phone_numbers).to include('+910808080818', '+910808080808')
|
||||
end
|
||||
|
||||
it 'returns all resolved contacts as results when filter is not prvoided' do
|
||||
create(:contact, account: account, email: nil, phone_number: nil)
|
||||
described_class.perform_now(account.id, user.id, %w[id name email column_not_present], {})
|
||||
csv_data = CSV.parse(account.contacts_export.download, headers: true)
|
||||
expect(csv_data.length).to eq(account.contacts.resolved_contacts.count)
|
||||
end
|
||||
|
||||
it 'returns resolved contacts filtered if labels are provided' do
|
||||
# Adding label to a resolved contact
|
||||
Contact.last.add_labels(['spec-billing'])
|
||||
contact = create(:contact, account: account, email: nil, phone_number: nil)
|
||||
contact.add_labels(['spec-billing'])
|
||||
described_class.perform_now(account.id, user.id, [], { :payload => nil, :label => 'spec-billing' })
|
||||
csv_data = CSV.parse(account.contacts_export.download, headers: true)
|
||||
# since there is only 1 resolved contact with 'spec-billing' label
|
||||
expect(csv_data.length).to eq(1)
|
||||
end
|
||||
|
||||
# TODO: This returns unresolved contacts as well since filter service returns the same
|
||||
# Change this when we make changes to filter service and ensure only resolved contacts are returned
|
||||
it 'returns filtered data which inclues unresolved contacts when filter is provided' do
|
||||
create(:contact, account: account, email: nil, phone_number: nil, additional_attributes: { :country_code => 'India' })
|
||||
described_class.perform_now(account.id, user.id, [], { :payload => [city_filter.merge(:query_operator => nil)] }.with_indifferent_access)
|
||||
csv_data = CSV.parse(account.contacts_export.download, headers: true)
|
||||
expect(csv_data.length).to eq(5)
|
||||
end
|
||||
|
||||
it 'returns filtered data when multiple filters are provided' do
|
||||
described_class.perform_now(account.id, user.id, [], multiple_filters.with_indifferent_access)
|
||||
csv_data = CSV.parse(account.contacts_export.download, headers: true)
|
||||
# since there are only 4 contacts with 'looped' in email and 'India' as country_code
|
||||
expect(csv_data.length).to eq(4)
|
||||
end
|
||||
|
||||
it 'returns filtered data when a single filter is provided' do
|
||||
described_class.perform_now(account.id, user.id, [], single_filter.with_indifferent_access)
|
||||
csv_data = CSV.parse(account.contacts_export.download, headers: true)
|
||||
# since there are only 8 contacts with 'looped' in email
|
||||
expect(csv_data.length).to eq(8)
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
@@ -16,7 +16,6 @@ describe Contacts::FilterService do
|
||||
create(:inbox_member, user: second_user, inbox: inbox)
|
||||
create(:conversation, account: account, inbox: inbox, assignee: first_user, contact: en_contact)
|
||||
create(:conversation, account: account, inbox: inbox, contact: el_contact)
|
||||
Current.account = account
|
||||
|
||||
create(:custom_attribute_definition,
|
||||
attribute_key: 'contact_additional_information',
|
||||
@@ -59,7 +58,7 @@ describe Contacts::FilterService do
|
||||
}.with_indifferent_access
|
||||
]
|
||||
|
||||
result = filter_service.new(params, first_user).perform
|
||||
result = filter_service.new(account, first_user, params).perform
|
||||
expect(result[:count]).to be 1
|
||||
expect(result[:contacts].length).to be 1
|
||||
expect(result[:contacts].first.name).to eq(en_contact.name)
|
||||
@@ -71,7 +70,7 @@ describe Contacts::FilterService do
|
||||
blocked_contact = create(:contact, account: account, blocked: true)
|
||||
params = { payload: [{ attribute_key: 'blocked', filter_operator: 'equal_to', values: ['true'],
|
||||
query_operator: nil }.with_indifferent_access] }
|
||||
result = filter_service.new(params, first_user).perform
|
||||
result = filter_service.new(account, first_user, params).perform
|
||||
expect(result[:count]).to be 1
|
||||
expect(result[:contacts].first.id).to eq(blocked_contact.id)
|
||||
end
|
||||
@@ -79,7 +78,7 @@ describe Contacts::FilterService do
|
||||
it 'filter contacts by not_blocked' do
|
||||
params = { payload: [{ attribute_key: 'blocked', filter_operator: 'equal_to', values: [false],
|
||||
query_operator: nil }.with_indifferent_access] }
|
||||
result = filter_service.new(params, first_user).perform
|
||||
result = filter_service.new(account, first_user, params).perform
|
||||
# existing contacts are not blocked
|
||||
expect(result[:count]).to be 3
|
||||
end
|
||||
@@ -96,7 +95,7 @@ describe Contacts::FilterService do
|
||||
}.with_indifferent_access
|
||||
]
|
||||
|
||||
result = filter_service.new(params, first_user).perform
|
||||
result = filter_service.new(account, first_user, params).perform
|
||||
expect(result[:contacts].length).to be 2
|
||||
expect(result[:contacts].first.label_list).to include('support')
|
||||
expect(result[:contacts].last.label_list).to include('support')
|
||||
@@ -112,7 +111,7 @@ describe Contacts::FilterService do
|
||||
}.with_indifferent_access
|
||||
]
|
||||
|
||||
result = filter_service.new(params, first_user).perform
|
||||
result = filter_service.new(account, first_user, params).perform
|
||||
expect(result[:contacts].length).to be 1
|
||||
expect(result[:contacts].first.id).to eq el_contact.id
|
||||
end
|
||||
@@ -127,7 +126,7 @@ describe Contacts::FilterService do
|
||||
}.with_indifferent_access
|
||||
]
|
||||
|
||||
result = filter_service.new(params, first_user).perform
|
||||
result = filter_service.new(account, first_user, params).perform
|
||||
expect(result[:contacts].length).to be 2
|
||||
expect(result[:contacts].first.label_list).to include('support')
|
||||
expect(result[:contacts].last.label_list).to include('support')
|
||||
@@ -143,7 +142,7 @@ describe Contacts::FilterService do
|
||||
}.with_indifferent_access
|
||||
]
|
||||
|
||||
result = filter_service.new(params, first_user).perform
|
||||
result = filter_service.new(account, first_user, params).perform
|
||||
expect(result[:contacts].length).to be 1
|
||||
expect(result[:contacts].first.id).to eq el_contact.id
|
||||
end
|
||||
@@ -180,7 +179,7 @@ describe Contacts::FilterService do
|
||||
'test custom data'
|
||||
).count
|
||||
|
||||
result = filter_service.new(params, first_user).perform
|
||||
result = filter_service.new(account, first_user, params).perform
|
||||
expect(result[:contacts].length).to be expected_count
|
||||
expect(result[:contacts].first.id).to eq(el_contact.id)
|
||||
end
|
||||
@@ -197,7 +196,7 @@ describe Contacts::FilterService do
|
||||
|
||||
expected_count = Contact.where('last_activity_at < ?', (Time.zone.today - 2.days)).count
|
||||
|
||||
result = filter_service.new(params, first_user).perform
|
||||
result = filter_service.new(account, first_user, params).perform
|
||||
expect(result[:contacts].length).to be expected_count
|
||||
expect(result[:contacts].pluck(:id)).to include(el_contact.id)
|
||||
expect(result[:contacts].pluck(:id)).to include(cs_contact.id)
|
||||
@@ -219,7 +218,7 @@ describe Contacts::FilterService do
|
||||
|
||||
it 'filter contacts by additional_attributes' do
|
||||
params[:payload] = payload
|
||||
result = filter_service.new(params, first_user).perform
|
||||
result = filter_service.new(account, first_user, params).perform
|
||||
expect(result[:count]).to be 1
|
||||
expect(result[:contacts].first.id).to eq(en_contact.id)
|
||||
end
|
||||
@@ -247,7 +246,7 @@ describe Contacts::FilterService do
|
||||
query_operator: nil
|
||||
}.with_indifferent_access
|
||||
]
|
||||
result = filter_service.new(params, first_user).perform
|
||||
result = filter_service.new(account, first_user, params).perform
|
||||
expect(result[:contacts].length).to be 1
|
||||
expect(result[:contacts].first.id).to eq(cs_contact.id)
|
||||
end
|
||||
@@ -273,7 +272,7 @@ describe Contacts::FilterService do
|
||||
query_operator: nil
|
||||
}.with_indifferent_access
|
||||
]
|
||||
result = filter_service.new(params, first_user).perform
|
||||
result = filter_service.new(account, first_user, params).perform
|
||||
expect(result[:contacts].length).to be 1
|
||||
expect(result[:contacts].first.id).to eq(el_contact.id)
|
||||
end
|
||||
@@ -294,7 +293,7 @@ describe Contacts::FilterService do
|
||||
query_operator: nil
|
||||
}.with_indifferent_access
|
||||
]
|
||||
result = filter_service.new(params, first_user).perform
|
||||
result = filter_service.new(account, first_user, params).perform
|
||||
expected_count = Contact.where("created_at < ? AND custom_attributes->>'customer_type' = ?", Date.tomorrow, 'platinum').count
|
||||
|
||||
expect(result[:contacts].length).to be expected_count
|
||||
|
||||
Reference in New Issue
Block a user