chore: Refactor RefreshOauthTokenService to improve readability (#8820)
- Added a trait called microsoft_email for the Channel::Email factory. - Rewrote the logic to make it simple to understand - Rewrote the specs for readability
This commit is contained in:
@@ -8,5 +8,22 @@ FactoryBot.define do
|
||||
after(:create) do |channel_email|
|
||||
create(:inbox, channel: channel_email, account: channel_email.account)
|
||||
end
|
||||
|
||||
trait :microsoft_email do
|
||||
imap_enabled { true }
|
||||
imap_address { 'outlook.office365.com' }
|
||||
imap_port { 993 }
|
||||
imap_login { 'email@example.com' }
|
||||
imap_password { '' }
|
||||
imap_enable_ssl { true }
|
||||
provider_config do
|
||||
{
|
||||
expires_on: Time.zone.now + 3600,
|
||||
access_token: SecureRandom.hex,
|
||||
refresh_token: SecureRandom.hex
|
||||
}
|
||||
end
|
||||
provider { 'microsoft' }
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
@@ -8,11 +8,8 @@ RSpec.describe Inboxes::FetchImapEmailsJob do
|
||||
create(:channel_email, imap_enabled: true, imap_address: 'imap.gmail.com', imap_port: 993, imap_login: 'imap@gmail.com',
|
||||
imap_password: 'password', account: account)
|
||||
end
|
||||
let(:microsoft_imap_email_channel) do
|
||||
create(:channel_email, provider: 'microsoft', imap_enabled: true, imap_address: 'outlook.office365.com',
|
||||
imap_port: 993, imap_login: 'imap@outlook.com', imap_password: 'password', account: account,
|
||||
provider_config: { access_token: 'access_token' })
|
||||
end
|
||||
|
||||
let(:microsoft_imap_email_channel) { create(:channel_email, :microsoft_email) }
|
||||
let(:ms_email_inbox) { create(:inbox, channel: microsoft_imap_email_channel, account: account) }
|
||||
let!(:conversation) { create(:conversation, inbox: imap_email_channel.inbox, account: account) }
|
||||
let(:inbound_mail) { create_inbound_email_from_mail(from: 'testemail@gmail.com', to: 'imap@outlook.com', subject: 'Hello!') }
|
||||
|
||||
@@ -1,51 +1,87 @@
|
||||
require 'rails_helper'
|
||||
|
||||
RSpec.describe Microsoft::RefreshOauthTokenService do
|
||||
let(:access_token) { SecureRandom.hex }
|
||||
let(:refresh_token) { SecureRandom.hex }
|
||||
let(:expires_on) { Time.zone.now + 3600 }
|
||||
|
||||
let!(:microsoft_email_channel) do
|
||||
create(:channel_email, provider_config: { access_token: access_token, refresh_token: refresh_token, expires_on: expires_on })
|
||||
let!(:microsoft_channel) { create(:channel_email, :microsoft_email) }
|
||||
let!(:microsoft_channel_with_expired_token) do
|
||||
create(
|
||||
:channel_email, :microsoft_email, provider_config: {
|
||||
expires_on: Time.zone.now - 3600,
|
||||
access_token: SecureRandom.hex,
|
||||
refresh_token: SecureRandom.hex
|
||||
}
|
||||
)
|
||||
end
|
||||
let(:new_tokens) { { access_token: access_token, refresh_token: refresh_token, expires_at: expires_on.to_i, token_type: 'bearer' } }
|
||||
|
||||
describe '#access_token' do
|
||||
context 'when token is not expired' do
|
||||
it 'returns the existing access token' do
|
||||
expect(described_class.new(channel: microsoft_email_channel).access_token).to eq(access_token)
|
||||
expect(microsoft_email_channel.reload.provider_config['refresh_token']).to eq(refresh_token)
|
||||
end
|
||||
let(:new_tokens) do
|
||||
{
|
||||
access_token: SecureRandom.hex,
|
||||
refresh_token: SecureRandom.hex,
|
||||
expires_at: (Time.zone.now + 3600).to_i,
|
||||
token_type: 'bearer'
|
||||
}
|
||||
end
|
||||
|
||||
context 'when token is not expired' do
|
||||
it 'returns the existing access token' do
|
||||
service = described_class.new(channel: microsoft_channel)
|
||||
|
||||
expect(service.access_token).to eq(microsoft_channel.provider_config['access_token'])
|
||||
expect(microsoft_channel.reload.provider_config['refresh_token']).to eq(microsoft_channel.provider_config['refresh_token'])
|
||||
end
|
||||
end
|
||||
|
||||
describe 'on expired token or invalid expiry' do
|
||||
before do
|
||||
stub_request(:post, 'https://login.microsoftonline.com/common/oauth2/v2.0/token').with(
|
||||
body: { 'grant_type' => 'refresh_token', 'refresh_token' => microsoft_channel_with_expired_token.provider_config['refresh_token'] }
|
||||
).to_return(status: 200, body: new_tokens.to_json, headers: { 'Content-Type' => 'application/json' })
|
||||
end
|
||||
|
||||
context 'when token is expired' do
|
||||
let(:expires_on) { 1.minute.from_now }
|
||||
|
||||
before do
|
||||
stub_request(:post, 'https://login.microsoftonline.com/common/oauth2/v2.0/token').with(
|
||||
body: { 'grant_type' => 'refresh_token', 'refresh_token' => refresh_token }
|
||||
).to_return(status: 200, body: new_tokens.to_json, headers: { 'Content-Type' => 'application/json' })
|
||||
end
|
||||
|
||||
context 'when token is invalid' do
|
||||
it 'fetches new access token and refresh tokens' do
|
||||
microsoft_email_channel.provider_config['expires_on'] = Time.zone.now - 3600
|
||||
microsoft_email_channel.save!
|
||||
with_modified_env AZURE_APP_ID: SecureRandom.uuid, AZURE_APP_SECRET: SecureRandom.hex do
|
||||
provider_config = microsoft_channel_with_expired_token.provider_config
|
||||
service = described_class.new(channel: microsoft_channel_with_expired_token)
|
||||
expect(service.access_token).not_to eq(provider_config['access_token'])
|
||||
|
||||
expect(described_class.new(channel: microsoft_email_channel).access_token).not_to eq(access_token)
|
||||
expect(microsoft_email_channel.reload.provider_config['access_token']).to eq(new_tokens[:access_token])
|
||||
expect(microsoft_email_channel.reload.provider_config['refresh_token']).to eq(new_tokens[:refresh_token])
|
||||
expect(microsoft_email_channel.reload.provider_config['expires_on']).to eq(Time.at(new_tokens[:expires_at]).utc.to_s)
|
||||
new_provider_config = microsoft_channel_with_expired_token.reload.provider_config
|
||||
expect(new_provider_config['access_token']).to eq(new_tokens[:access_token])
|
||||
expect(new_provider_config['refresh_token']).to eq(new_tokens[:refresh_token])
|
||||
expect(new_provider_config['expires_on']).to eq(Time.at(new_tokens[:expires_at]).utc.to_s)
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
context 'when refresh token is not present in provider config and access token is expired' do
|
||||
it 'throws an error' do
|
||||
microsoft_email_channel.update(provider_config: {
|
||||
access_token: access_token,
|
||||
expires_on: expires_on - 3600
|
||||
})
|
||||
context 'when expiry time is missing' do
|
||||
it 'fetches new access token and refresh tokens' do
|
||||
with_modified_env AZURE_APP_ID: SecureRandom.uuid, AZURE_APP_SECRET: SecureRandom.hex do
|
||||
microsoft_channel_with_expired_token.provider_config['expires_on'] = nil
|
||||
microsoft_channel_with_expired_token.save!
|
||||
provider_config = microsoft_channel_with_expired_token.provider_config
|
||||
service = described_class.new(channel: microsoft_channel_with_expired_token)
|
||||
expect(service.access_token).not_to eq(provider_config['access_token'])
|
||||
|
||||
new_provider_config = microsoft_channel_with_expired_token.reload.provider_config
|
||||
expect(new_provider_config['access_token']).to eq(new_tokens[:access_token])
|
||||
expect(new_provider_config['refresh_token']).to eq(new_tokens[:refresh_token])
|
||||
expect(new_provider_config['expires_on']).to eq(Time.at(new_tokens[:expires_at]).utc.to_s)
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
context 'when refresh token is not present in provider config and access token is expired' do
|
||||
it 'throws an error' do
|
||||
with_modified_env AZURE_APP_ID: SecureRandom.uuid, AZURE_APP_SECRET: SecureRandom.hex do
|
||||
microsoft_channel.update(
|
||||
provider_config: {
|
||||
access_token: SecureRandom.hex,
|
||||
expires_on: Time.zone.now - 3600
|
||||
}
|
||||
)
|
||||
|
||||
expect do
|
||||
described_class.new(channel: microsoft_email_channel).access_token
|
||||
described_class.new(channel: microsoft_channel).access_token
|
||||
end.to raise_error(RuntimeError, 'A refresh_token is not available')
|
||||
end
|
||||
end
|
||||
|
||||
Reference in New Issue
Block a user