2018-12-12 08:57:30 +00:00
require 'rails_helper'
RSpec . describe 'External Credentials' , type : :request do
let ( :admin_user ) { create ( :admin_user ) }
context 'without authentication' do
describe '#index' do
it 'returns 401 unauthorized' do
get '/api/v1/external_credentials' , as : :json
2019-04-15 01:41:17 +00:00
expect ( response ) . to have_http_status ( :unauthorized )
2018-12-12 08:57:30 +00:00
expect ( json_response ) . to include ( 'error' = > 'authentication failed' )
end
end
describe '#app_verify' do
it 'returns 401 unauthorized' do
post '/api/v1/external_credentials/facebook/app_verify' , as : :json
2019-04-15 01:41:17 +00:00
expect ( response ) . to have_http_status ( :unauthorized )
2018-12-12 08:57:30 +00:00
expect ( json_response ) . to include ( 'error' = > 'authentication failed' )
end
end
describe '#link_account' do
it 'returns 401 unauthorized' do
get '/api/v1/external_credentials/facebook/link_account' , as : :json
2019-04-15 01:41:17 +00:00
expect ( response ) . to have_http_status ( :unauthorized )
2018-12-12 08:57:30 +00:00
expect ( json_response ) . to include ( 'error' = > 'authentication failed' )
end
end
describe '#callback' do
it 'returns 401 unauthorized' do
get '/api/v1/external_credentials/facebook/callback' , as : :json
2019-04-15 01:41:17 +00:00
expect ( response ) . to have_http_status ( :unauthorized )
2018-12-12 08:57:30 +00:00
expect ( json_response ) . to include ( 'error' = > 'authentication failed' )
end
end
end
context 'authenticated as admin' do
2020-03-13 05:04:33 +00:00
before { authenticated_as ( admin_user , via : :browser ) }
2018-12-12 08:57:30 +00:00
describe '#index' do
it 'responds with an array of ExternalCredential records' do
get '/api/v1/external_credentials' , as : :json
2019-04-15 01:41:17 +00:00
expect ( response ) . to have_http_status ( :ok )
2018-12-12 08:57:30 +00:00
expect ( json_response ) . to eq ( [ ] )
end
context 'with expand=true URL parameters' do
it 'responds with an array of ExternalCredential records and their association data' do
get '/api/v1/external_credentials?expand=true' , as : :json
2019-04-15 01:41:17 +00:00
expect ( response ) . to have_http_status ( :ok )
2018-12-12 08:57:30 +00:00
expect ( json_response ) . to eq ( [ ] )
end
end
end
context 'for Facebook' do
let ( :invalid_credentials ) do
{ application_id : 123 , application_secret : 123 }
end
describe '#app_verify' do
describe 'failure cases' do
context 'when permission for Facebook channel is deactivated' do
before { Permission . find_by ( name : 'admin.channel_facebook' ) . update ( active : false ) }
it 'returns 401 unauthorized with internal (Zammad) error' do
post '/api/v1/external_credentials/facebook/app_verify' , as : :json
2019-04-15 01:41:17 +00:00
expect ( response ) . to have_http_status ( :unauthorized )
2018-12-12 08:57:30 +00:00
expect ( json_response ) . to include ( 'error' = > 'Not authorized (user)!' )
end
end
context 'with no credentials' do
it 'returns 200 with internal (Zammad) error' do
post '/api/v1/external_credentials/facebook/app_verify' , as : :json
2019-04-15 01:41:17 +00:00
expect ( response ) . to have_http_status ( :ok )
2018-12-12 08:57:30 +00:00
expect ( json_response ) . to include ( 'error' = > 'No application_id param!' )
end
end
context 'with invalid credentials, via request params' do
Refactoring: Automatic RSpec VCR cassette name helper
This commit was prepared to support upcoming additions to the test suite
(specifically, better coverage for existing Twitter functionality).
These upcoming changes will depend heavily on VCR.[0]
(VCR is a Ruby gem that makes it easier to write and run tests
that call out to external services over HTTP
by "recording" HTTP transactions to a YAML file
and "replaying" them later.)
VCR is widely-used (4600 GitHub stars), but its API is a little clumsy--
You have to manually specify the name of a "cassette" file every time:
it 'does something' do
VCR.use_cassette('path/to/cassette') do
...
end
end
This commit adds an RSpec metadata config option
as a shorthand for the syntax above:
it 'does something', :use_vcr do
...
end
This config option automatically generates a cassette filename
based on the description of the example it's applied to.
=== Analysis of alternative approaches
Ideally, these auto-generated cassette filenames should be unique:
if filenames collide, multiple examples will share the same cassette.
A first attempt generated names based on `example.full_description`,
but that led to errors:
Errno::ENAMETOOLONG:
File name too long @ rb_sysopen - /opt/zammad/test/data/vcr_cassettes/models/ticket/article/ticket_article_callbacks_observers_async_transactions_-_auto-setting_of_outgoing_twitter_article_attributes_via_bg_jobs_when_the_original_channel_specified_in_ticket_preferences_was_deleted_but_a_new_one_with_the_same_screen_name_exists_sets_appropriate_status_attributes_on_the_new_channel.yml
Another idea was to use MD5 digests of the above,
but in fact both of these approaches share another problem:
even minor changes to the description could break tests
(unless the committer remembers to rename the cassette file to match):
an altered description means VCR will record a new cassette file
instead of replaying from the original.
(Normally, this would only slow down the test instead of breaking it,
but sometimes we modify tests and cassettes after recording them
to hide sensitive data like API keys or login credentials.)
The approach taken by this commit was to use partial descriptions,
combining the parent `describe`/`context` label with the `it` label.
This does not guarantee uniqueness--
even in the present refactoring, it produced a filename collision--
but it's a good middle ground.
[0]: https://relishapp.com/vcr/vcr/docs
2019-11-12 08:17:21 +00:00
it 'returns 200 with remote (Facebook auth) error' , :use_vcr do
post '/api/v1/external_credentials/facebook/app_verify' , params : invalid_credentials , as : :json
2018-12-12 08:57:30 +00:00
2019-04-15 01:41:17 +00:00
expect ( response ) . to have_http_status ( :ok )
2018-12-12 08:57:30 +00:00
expect ( json_response ) . to include ( 'error' = > 'type: OAuthException, code: 101, message: Error validating application. Cannot get application info due to a system error. [HTTP 400]' )
end
end
context 'with invalid credentials, via ExternalCredential record' do
before { create ( :facebook_credential , credentials : invalid_credentials ) }
Refactoring: Automatic RSpec VCR cassette name helper
This commit was prepared to support upcoming additions to the test suite
(specifically, better coverage for existing Twitter functionality).
These upcoming changes will depend heavily on VCR.[0]
(VCR is a Ruby gem that makes it easier to write and run tests
that call out to external services over HTTP
by "recording" HTTP transactions to a YAML file
and "replaying" them later.)
VCR is widely-used (4600 GitHub stars), but its API is a little clumsy--
You have to manually specify the name of a "cassette" file every time:
it 'does something' do
VCR.use_cassette('path/to/cassette') do
...
end
end
This commit adds an RSpec metadata config option
as a shorthand for the syntax above:
it 'does something', :use_vcr do
...
end
This config option automatically generates a cassette filename
based on the description of the example it's applied to.
=== Analysis of alternative approaches
Ideally, these auto-generated cassette filenames should be unique:
if filenames collide, multiple examples will share the same cassette.
A first attempt generated names based on `example.full_description`,
but that led to errors:
Errno::ENAMETOOLONG:
File name too long @ rb_sysopen - /opt/zammad/test/data/vcr_cassettes/models/ticket/article/ticket_article_callbacks_observers_async_transactions_-_auto-setting_of_outgoing_twitter_article_attributes_via_bg_jobs_when_the_original_channel_specified_in_ticket_preferences_was_deleted_but_a_new_one_with_the_same_screen_name_exists_sets_appropriate_status_attributes_on_the_new_channel.yml
Another idea was to use MD5 digests of the above,
but in fact both of these approaches share another problem:
even minor changes to the description could break tests
(unless the committer remembers to rename the cassette file to match):
an altered description means VCR will record a new cassette file
instead of replaying from the original.
(Normally, this would only slow down the test instead of breaking it,
but sometimes we modify tests and cassettes after recording them
to hide sensitive data like API keys or login credentials.)
The approach taken by this commit was to use partial descriptions,
combining the parent `describe`/`context` label with the `it` label.
This does not guarantee uniqueness--
even in the present refactoring, it produced a filename collision--
but it's a good middle ground.
[0]: https://relishapp.com/vcr/vcr/docs
2019-11-12 08:17:21 +00:00
it 'returns 200 with remote (Facebook auth) error' , :use_vcr do
post '/api/v1/external_credentials/facebook/app_verify' , as : :json
2018-12-12 08:57:30 +00:00
2019-04-15 01:41:17 +00:00
expect ( response ) . to have_http_status ( :ok )
2018-12-12 08:57:30 +00:00
expect ( json_response ) . to include ( 'error' = > 'type: OAuthException, code: 101, message: Error validating application. Cannot get application info due to a system error. [HTTP 400]' )
end
end
end
end
describe '#link_account' do
describe 'failure cases' do
context 'with no credentials' do
it 'returns 422 unprocessable entity with internal (Zammad) error' do
get '/api/v1/external_credentials/facebook/link_account' , as : :json
2019-04-15 01:41:17 +00:00
expect ( response ) . to have_http_status ( :unprocessable_entity )
2018-12-12 08:57:30 +00:00
expect ( json_response ) . to include ( 'error' = > 'No facebook app configured!' )
end
end
context 'with invalid credentials, via request params' do
it 'returns 422 unprocessable entity with internal (Zammad) error' do
get '/api/v1/external_credentials/facebook/link_account' , params : invalid_credentials , as : :json
2019-04-15 01:41:17 +00:00
expect ( response ) . to have_http_status ( :unprocessable_entity )
2018-12-12 08:57:30 +00:00
expect ( json_response ) . to include ( 'error' = > 'No facebook app configured!' )
end
end
context 'with invalid credentials, via ExternalCredential record' do
before { create ( :facebook_credential , credentials : invalid_credentials ) }
Refactoring: Automatic RSpec VCR cassette name helper
This commit was prepared to support upcoming additions to the test suite
(specifically, better coverage for existing Twitter functionality).
These upcoming changes will depend heavily on VCR.[0]
(VCR is a Ruby gem that makes it easier to write and run tests
that call out to external services over HTTP
by "recording" HTTP transactions to a YAML file
and "replaying" them later.)
VCR is widely-used (4600 GitHub stars), but its API is a little clumsy--
You have to manually specify the name of a "cassette" file every time:
it 'does something' do
VCR.use_cassette('path/to/cassette') do
...
end
end
This commit adds an RSpec metadata config option
as a shorthand for the syntax above:
it 'does something', :use_vcr do
...
end
This config option automatically generates a cassette filename
based on the description of the example it's applied to.
=== Analysis of alternative approaches
Ideally, these auto-generated cassette filenames should be unique:
if filenames collide, multiple examples will share the same cassette.
A first attempt generated names based on `example.full_description`,
but that led to errors:
Errno::ENAMETOOLONG:
File name too long @ rb_sysopen - /opt/zammad/test/data/vcr_cassettes/models/ticket/article/ticket_article_callbacks_observers_async_transactions_-_auto-setting_of_outgoing_twitter_article_attributes_via_bg_jobs_when_the_original_channel_specified_in_ticket_preferences_was_deleted_but_a_new_one_with_the_same_screen_name_exists_sets_appropriate_status_attributes_on_the_new_channel.yml
Another idea was to use MD5 digests of the above,
but in fact both of these approaches share another problem:
even minor changes to the description could break tests
(unless the committer remembers to rename the cassette file to match):
an altered description means VCR will record a new cassette file
instead of replaying from the original.
(Normally, this would only slow down the test instead of breaking it,
but sometimes we modify tests and cassettes after recording them
to hide sensitive data like API keys or login credentials.)
The approach taken by this commit was to use partial descriptions,
combining the parent `describe`/`context` label with the `it` label.
This does not guarantee uniqueness--
even in the present refactoring, it produced a filename collision--
but it's a good middle ground.
[0]: https://relishapp.com/vcr/vcr/docs
2019-11-12 08:17:21 +00:00
it 'returns 500 with remote (Facebook auth) error' , :use_vcr do
get '/api/v1/external_credentials/facebook/link_account' , as : :json
2018-12-12 08:57:30 +00:00
2019-04-15 01:41:17 +00:00
expect ( response ) . to have_http_status ( :internal_server_error )
2018-12-12 08:57:30 +00:00
expect ( json_response ) . to include ( 'error' = > 'type: OAuthException, code: 101, message: Error validating application. Cannot get application info due to a system error. [HTTP 400]' )
end
end
end
end
describe '#callback' do
describe 'failure cases' do
context 'with no credentials' do
it 'returns 422 unprocessable entity with internal (Zammad) error' do
get '/api/v1/external_credentials/facebook/callback' , as : :json
2019-04-15 01:41:17 +00:00
expect ( response ) . to have_http_status ( :unprocessable_entity )
2018-12-12 08:57:30 +00:00
expect ( json_response ) . to include ( 'error' = > 'No facebook app configured!' )
end
end
context 'with invalid credentials, via request params' do
it 'returns 422 unprocessable entity with internal (Zammad) error' do
get '/api/v1/external_credentials/facebook/callback' , params : invalid_credentials , as : :json
2019-04-15 01:41:17 +00:00
expect ( response ) . to have_http_status ( :unprocessable_entity )
2018-12-12 08:57:30 +00:00
expect ( json_response ) . to include ( 'error' = > 'No facebook app configured!' )
end
end
context 'with invalid credentials, via ExternalCredential record' do
before { create ( :facebook_credential , credentials : invalid_credentials ) }
Refactoring: Automatic RSpec VCR cassette name helper
This commit was prepared to support upcoming additions to the test suite
(specifically, better coverage for existing Twitter functionality).
These upcoming changes will depend heavily on VCR.[0]
(VCR is a Ruby gem that makes it easier to write and run tests
that call out to external services over HTTP
by "recording" HTTP transactions to a YAML file
and "replaying" them later.)
VCR is widely-used (4600 GitHub stars), but its API is a little clumsy--
You have to manually specify the name of a "cassette" file every time:
it 'does something' do
VCR.use_cassette('path/to/cassette') do
...
end
end
This commit adds an RSpec metadata config option
as a shorthand for the syntax above:
it 'does something', :use_vcr do
...
end
This config option automatically generates a cassette filename
based on the description of the example it's applied to.
=== Analysis of alternative approaches
Ideally, these auto-generated cassette filenames should be unique:
if filenames collide, multiple examples will share the same cassette.
A first attempt generated names based on `example.full_description`,
but that led to errors:
Errno::ENAMETOOLONG:
File name too long @ rb_sysopen - /opt/zammad/test/data/vcr_cassettes/models/ticket/article/ticket_article_callbacks_observers_async_transactions_-_auto-setting_of_outgoing_twitter_article_attributes_via_bg_jobs_when_the_original_channel_specified_in_ticket_preferences_was_deleted_but_a_new_one_with_the_same_screen_name_exists_sets_appropriate_status_attributes_on_the_new_channel.yml
Another idea was to use MD5 digests of the above,
but in fact both of these approaches share another problem:
even minor changes to the description could break tests
(unless the committer remembers to rename the cassette file to match):
an altered description means VCR will record a new cassette file
instead of replaying from the original.
(Normally, this would only slow down the test instead of breaking it,
but sometimes we modify tests and cassettes after recording them
to hide sensitive data like API keys or login credentials.)
The approach taken by this commit was to use partial descriptions,
combining the parent `describe`/`context` label with the `it` label.
This does not guarantee uniqueness--
even in the present refactoring, it produced a filename collision--
but it's a good middle ground.
[0]: https://relishapp.com/vcr/vcr/docs
2019-11-12 08:17:21 +00:00
it 'returns 500 with remote (Facebook auth) error' , :use_vcr do
get '/api/v1/external_credentials/facebook/callback' , as : :json
2018-12-12 08:57:30 +00:00
2019-04-15 01:41:17 +00:00
expect ( response ) . to have_http_status ( :internal_server_error )
2018-12-12 08:57:30 +00:00
expect ( json_response ) . to include ( 'error' = > 'type: OAuthException, code: 101, message: Error validating application. Cannot get application info due to a system error. [HTTP 400]' )
end
end
end
end
end
2020-03-13 05:04:33 +00:00
context 'for Twitter' , :use_vcr do
shared_context 'for callback URL configuration' do
# NOTE: When recording a new VCR cassette for these tests,
# the URL below must match the callback URL
# registered with developer.twitter.com.
before do
Setting . set ( 'http_type' , 'https' )
Setting . set ( 'fqdn' , 'zammad.example.com' )
end
2018-12-12 08:57:30 +00:00
end
2020-03-13 05:04:33 +00:00
shared_examples 'for failure cases' do
it 'responds with the appropriate status and error message' do
send ( * endpoint , as : :json , params : try ( :params ) || { } )
2018-12-12 08:57:30 +00:00
2020-03-13 05:04:33 +00:00
expect ( response ) . to have_http_status ( status )
expect ( json_response ) . to include ( 'error' = > error_message )
end
end
let ( :valid_credentials ) { attributes_for ( :twitter_credential ) [ :credentials ] }
let ( :invalid_credentials ) { attributes_for ( :twitter_credential , :invalid ) [ :credentials ] }
describe 'POST /api/v1/external_credentials/twitter/app_verify' do
let ( :endpoint ) { [ :post , '/api/v1/external_credentials/twitter/app_verify' ] }
context 'when permission for Twitter channel is deactivated' do
before { Permission . find_by ( name : 'admin.channel_twitter' ) . update ( active : false ) }
include_examples 'for failure cases' do
let ( :status ) { :unauthorized }
let ( :error_message ) { 'Not authorized (user)!' }
end
end
context 'with no credentials' do
include_examples 'for failure cases' do
let ( :status ) { :ok }
let ( :error_message ) { 'No consumer_key param!' }
2018-12-12 08:57:30 +00:00
end
2020-03-13 05:04:33 +00:00
end
2018-12-12 08:57:30 +00:00
2020-03-13 05:04:33 +00:00
context 'with invalid credential params' do
let ( :params ) { invalid_credentials }
2018-12-12 08:57:30 +00:00
2020-03-13 05:04:33 +00:00
include_examples 'for failure cases' do
let ( :status ) { :ok }
2020-03-20 10:29:50 +00:00
let ( :error_message ) { << ~ ERR . chomp }
401 Authorization Required ( Invalid credentials may be to blame . )
ERR
2018-12-12 08:57:30 +00:00
end
2020-03-13 05:04:33 +00:00
end
2018-12-12 08:57:30 +00:00
2020-03-13 05:04:33 +00:00
context 'with valid credential params but misconfigured callback URL' do
let ( :params ) { valid_credentials }
include_examples 'for failure cases' do
let ( :status ) { :ok }
let ( :error_message ) { << ~ ERR . chomp }
2020-03-20 10:29:50 +00:00
403 Forbidden ( Your app ' s callback URL configuration on developer . twitter . com may be to blame . )
2020-03-13 05:04:33 +00:00
ERR
end
end
context 'with valid credential params and callback URL but no dev env registered' do
let ( :params ) { valid_credentials }
include_context 'for callback URL configuration'
include_examples 'for failure cases' do
let ( :status ) { :ok }
let ( :error_message ) { << ~ ERR . chomp }
Unable to get list of webooks . Maybe you do not have an Twitter developer approval right now or you use the wrong 'Dev environment label' : Forbidden .
ERR
end
end
context 'with valid credential params and callback URL but wrong dev env label' do
let ( :params ) { valid_credentials . merge ( env : 'foo' ) }
include_context 'for callback URL configuration'
include_examples 'for failure cases' do
let ( :status ) { :ok }
let ( :error_message ) { << ~ ERR . chomp }
Unable to get list of webooks . Maybe you do not have an Twitter developer approval right now or you use the wrong 'Dev environment label' : Unable to get list of webooks . You use the wrong 'Dev environment label' , only { :environments = > [ { :environment_name = > \ " zammad \" , :webhooks=>[]}]} available.
ERR
end
end
context 'with valid credential params, callback URL, and dev env label' do
let ( :env_name ) { valid_credentials [ :env ] }
include_context 'for callback URL configuration'
shared_examples 'for successful webhook connection' do
let ( :webhook_id ) { '1241980494134145024' }
it 'responds 200 OK with the new webhook ID' do
send ( * endpoint , as : :json , params : valid_credentials )
2018-12-12 08:57:30 +00:00
2019-04-15 01:41:17 +00:00
expect ( response ) . to have_http_status ( :ok )
2020-03-13 05:04:33 +00:00
expect ( json_response ) . to match ( 'attributes' = > hash_including ( 'webhook_id' = > webhook_id ) )
2018-12-12 08:57:30 +00:00
end
end
2020-03-13 05:04:33 +00:00
context 'with no existing webhooks' do
let ( :webhook_url ) { " #{ Setting . get ( 'http_type' ) } :// #{ Setting . get ( 'fqdn' ) } #{ Rails . configuration . api_path } /channels_twitter_webhook " }
2018-12-12 08:57:30 +00:00
2020-03-13 05:04:33 +00:00
include_examples 'for successful webhook connection'
2018-12-12 08:57:30 +00:00
2020-03-13 05:04:33 +00:00
it 'registers a new webhook' do
send ( * endpoint , as : :json , params : valid_credentials )
expect ( WebMock )
. to have_requested ( :post , " https://api.twitter.com/1.1/account_activity/all/ #{ env_name } /webhooks.json " )
. with ( body : " url= #{ CGI . escape ( webhook_url ) } " )
2018-12-12 08:57:30 +00:00
end
end
2020-03-13 05:04:33 +00:00
context 'with an existing webhook registered to another app' do
include_examples 'for successful webhook connection'
2018-12-12 08:57:30 +00:00
2020-03-13 05:04:33 +00:00
it 'deletes all existing webhooks first' do
send ( * endpoint , as : :json , params : valid_credentials )
expect ( WebMock )
. to have_requested ( :delete , " https://api.twitter.com/1.1/account_activity/all/ #{ env_name } /webhooks/1241981813595049984.json " )
2018-12-12 08:57:30 +00:00
end
end
2020-03-13 05:04:33 +00:00
context 'with an existing, invalid webhook registered to Zammad' do
include_examples 'for successful webhook connection'
2018-12-12 08:57:30 +00:00
2020-03-13 05:04:33 +00:00
it 'revalidates by manually triggering a challenge-response check' do
send ( * endpoint , as : :json , params : valid_credentials )
expect ( WebMock )
. to have_requested ( :put , " https://api.twitter.com/1.1/account_activity/all/ #{ env_name } /webhooks/1241980494134145024.json " )
2018-12-12 08:57:30 +00:00
end
end
2020-03-13 05:04:33 +00:00
context 'with an existing, valid webhook registered to Zammad' do
include_examples 'for successful webhook connection'
2018-12-12 08:57:30 +00:00
2020-03-13 05:04:33 +00:00
it 'uses the existing webhook' do
send ( * endpoint , as : :json , params : valid_credentials )
2018-12-12 08:57:30 +00:00
2020-03-13 05:04:33 +00:00
expect ( WebMock )
. not_to have_requested ( :post , " https://api.twitter.com/1.1/account_activity/all/ #{ env_name } /webhooks.json " )
2018-12-12 08:57:30 +00:00
end
end
end
end
2020-03-13 05:04:33 +00:00
describe 'GET /api/v1/external_credentials/twitter/link_account' do
let ( :endpoint ) { [ :get , '/api/v1/external_credentials/twitter/link_account' ] }
2018-12-12 08:57:30 +00:00
2020-03-13 05:04:33 +00:00
context 'with no Twitter app' do
include_examples 'for failure cases' do
let ( :status ) { :unprocessable_entity }
let ( :error_message ) { 'No twitter app configured!' }
2018-12-12 08:57:30 +00:00
end
2020-03-13 05:04:33 +00:00
end
2018-12-12 08:57:30 +00:00
2020-03-13 05:04:33 +00:00
context 'with invalid Twitter app (configured with invalid credentials)' do
let! ( :twitter_credential ) { create ( :twitter_credential , :invalid ) }
2018-12-12 08:57:30 +00:00
2020-03-13 05:04:33 +00:00
include_examples 'for failure cases' do
let ( :status ) { :internal_server_error }
2020-03-20 10:29:50 +00:00
let ( :error_message ) { << ~ ERR . chomp }
401 Authorization Required ( Invalid credentials may be to blame . )
ERR
2018-12-12 08:57:30 +00:00
end
2020-03-13 05:04:33 +00:00
end
2018-12-12 08:57:30 +00:00
2020-03-13 05:04:33 +00:00
context 'with a valid Twitter app but misconfigured callback URL' do
let! ( :twitter_credential ) { create ( :twitter_credential ) }
2018-12-12 08:57:30 +00:00
2020-03-13 05:04:33 +00:00
include_examples 'for failure cases' do
let ( :status ) { :internal_server_error }
let ( :error_message ) { << ~ ERR . chomp }
2020-03-20 10:29:50 +00:00
403 Forbidden ( Your app ' s callback URL configuration on developer . twitter . com may be to blame . )
2020-03-13 05:04:33 +00:00
ERR
end
end
2018-12-12 08:57:30 +00:00
2020-03-13 05:04:33 +00:00
context 'with a valid Twitter app and callback URL' do
let! ( :twitter_credential ) { create ( :twitter_credential ) }
include_context 'for callback URL configuration'
it 'requests OAuth request token from Twitter API' do
send ( * endpoint , as : :json )
expect ( WebMock )
. to have_requested ( :post , 'https://api.twitter.com/oauth/request_token' )
. with ( headers : { 'Authorization' = > / oauth_consumer_key=" #{ twitter_credential . credentials [ :consumer_key ] } " / } )
end
it 'redirects to Twitter authorization URL' do
send ( * endpoint , as : :json )
expect ( response ) . to redirect_to ( %r{ ^https://api.twitter.com/oauth/authorize \ ?oauth_token= \ w+$ } )
end
it 'saves request token to session hash' do
send ( * endpoint , as : :json )
expect ( session [ :request_token ] ) . to be_a ( OAuth :: RequestToken )
end
end
end
describe 'GET /api/v1/external_credentials/twitter/callback' do
let ( :endpoint ) { [ :get , '/api/v1/external_credentials/twitter/callback' ] }
context 'with no Twitter app' do
include_examples 'for failure cases' do
let ( :status ) { :unprocessable_entity }
let ( :error_message ) { 'No twitter app configured!' }
end
end
context 'with valid Twitter app but no request token' do
let! ( :twitter_credential ) { create ( :twitter_credential ) }
include_examples 'for failure cases' do
let ( :status ) { :unprocessable_entity }
let ( :error_message ) { 'No request_token for session found!' }
end
end
context 'with valid Twitter app and request token but non-matching OAuth token (via params)' do
include_context 'for callback URL configuration'
let! ( :twitter_credential ) { create ( :twitter_credential ) }
before { get '/api/v1/external_credentials/twitter/link_account' , as : :json }
include_examples 'for failure cases' do
let ( :status ) { :unprocessable_entity }
let ( :error_message ) { 'Invalid oauth_token given!' }
end
end
# NOTE: Want to delete/regenerate the VCR cassettes for these examples?
# It's gonna be messy--each one is actually two cassettes merged into one.
#
# Why? The OAuth flow can't be fully reproduced in a request spec:
#
# 1. User clicks "Add Twitter account" in Zammad.
# Zammad asks Twitter for request token, saves it to session,
# and redirects user to Twitter.
# 2. User clicks "Authorize app" on Twitter.
# Twitter generates temporary OAuth credentials
# and redirects user back to this endpoint (with creds in URL query string).
# 3. Zammad asks Twitter for an access token
# (using request token from Step 1 + OAuth creds from Step 2).
#
# In these tests (Step 2), the user hits this endpoint
# with parameters that ONLY the Twitter OAuth server can generate.
# In the VCR cassette for Step 3,
# Zammad sends these parameters back to Twitter for validation.
# Without valid credentials in Step 2, Step 3 will always fail.
#
# Instead, we have to record the VCR cassette in a live development instance
# and stitch the cassette together with a cassette for Step 1.
#
# tl;dr A feature spec might have made more sense here.
context 'with valid Twitter app, request token, and matching OAuth token (via params)' do
include_context 'for callback URL configuration'
let! ( :twitter_credential ) { create ( :twitter_credential ) }
# For illustrative purposes only.
# These parameters cannot be used to record a new VCR cassette (see note above).
let ( :params ) { { oauth_token : oauth_token , oauth_verifier : oauth_verifier } }
let ( :oauth_token ) { 'DyhnyQAAAAAA9CNXAAABcSxAexs' }
let ( :oauth_verifier ) { '15DOeRkjP4JkOSVqULkTKA1SCuIPP105' }
before { get '/api/v1/external_credentials/twitter/link_account' , as : :json }
context 'if Twitter account has already been added' do
let! ( :channel ) { create ( :twitter_channel , custom_options : channel_options ) }
let ( :channel_options ) do
{
user : {
id : '1205290247124217856' ,
screen_name : 'pennbrooke1' ,
}
}
end
it 'uses the existing channel' do
expect { send ( * endpoint , as : :json , params : params ) }
. not_to change ( Channel , :count )
end
it 'updates channel properties' do
expect { send ( * endpoint , as : :json , params : params ) }
. to change { channel . reload . options [ :user ] [ :name ] }
. and change { channel . reload . options [ :auth ] [ :external_credential_id ] }
. and change { channel . reload . options [ :auth ] [ :oauth_token ] }
. and change { channel . reload . options [ :auth ] [ :oauth_token_secret ] }
end
it 'subscribes to webhooks' do
send ( * endpoint , as : :json , params : params )
expect ( WebMock )
. to have_requested ( :post , " https://api.twitter.com/1.1/account_activity/all/ #{ twitter_credential . credentials [ :env ] } /subscriptions.json " )
expect ( channel . reload . options [ 'subscribed_to_webhook_id' ] )
. to eq ( twitter_credential . credentials [ :webhook_id ] )
2018-12-12 08:57:30 +00:00
end
end
2020-03-13 05:04:33 +00:00
it 'creates a new channel' do
expect { send ( * endpoint , as : :json , params : params ) }
. to change ( Channel , :count ) . by ( 1 )
expect ( Channel . last . options )
. to include ( 'adapter' = > 'twitter' )
. and include ( 'user' = > hash_including ( 'id' , 'screen_name' , 'name' ) )
. and include ( 'auth' = > hash_including ( 'external_credential_id' , 'oauth_token' , 'oauth_token_secret' ) )
end
it 'redirects to the newly created channel' do
send ( * endpoint , as : :json , params : params )
expect ( response ) . to redirect_to ( %r{ / # channels/twitter/ #{ Channel . last . id } $ } )
end
it 'clears the :request_token session variable' do
send ( * endpoint , as : :json , params : params )
expect ( session [ :request_token ] ) . to be ( nil )
end
it 'subscribes to webhooks' do
send ( * endpoint , as : :json , params : params )
expect ( WebMock )
. to have_requested ( :post , " https://api.twitter.com/1.1/account_activity/all/ #{ twitter_credential . credentials [ :env ] } /subscriptions.json " )
expect ( Channel . last . options [ 'subscribed_to_webhook_id' ] )
. to eq ( twitter_credential . credentials [ :webhook_id ] )
end
2018-12-12 08:57:30 +00:00
end
end
end
end
end