solidus_braintree 1.4.0 → 2.0.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (204) hide show
  1. checksums.yaml +4 -4
  2. data/.circleci/config.yml +12 -0
  3. data/.gem_release.yml +5 -0
  4. data/.github_changelog_generator +2 -0
  5. data/.gitignore +20 -11
  6. data/.rspec +1 -1
  7. data/.rubocop.yml +79 -0
  8. data/CHANGELOG.md +178 -18
  9. data/Gemfile +31 -23
  10. data/LICENSE +26 -0
  11. data/README.md +387 -26
  12. data/Rakefile +4 -19
  13. data/app/assets/config/solidus_braintree_manifest.js +1 -0
  14. data/app/assets/images/solidus_braintree/venmo/venmo_active_blue_button_280x48.svg +19 -0
  15. data/app/assets/images/solidus_braintree/venmo/venmo_active_blue_button_320x48.svg +19 -0
  16. data/app/assets/images/solidus_braintree/venmo/venmo_active_blue_button_375x48.svg +19 -0
  17. data/app/assets/images/solidus_braintree/venmo/venmo_active_white_button_280x48.svg +19 -0
  18. data/app/assets/images/solidus_braintree/venmo/venmo_active_white_button_320x48.svg +19 -0
  19. data/app/assets/images/solidus_braintree/venmo/venmo_active_white_button_375x48.svg +19 -0
  20. data/app/assets/images/solidus_braintree/venmo/venmo_blue_acceptance_mark.svg +15 -0
  21. data/app/assets/images/solidus_braintree/venmo/venmo_blue_button_280x48.svg +19 -0
  22. data/app/assets/images/solidus_braintree/venmo/venmo_blue_button_320x48.svg +19 -0
  23. data/app/assets/images/solidus_braintree/venmo/venmo_blue_button_375x48.svg +19 -0
  24. data/app/assets/images/solidus_braintree/venmo/venmo_blue_logo.svg +18 -0
  25. data/app/assets/images/solidus_braintree/venmo/venmo_white_acceptance_mark.svg +20 -0
  26. data/app/assets/images/solidus_braintree/venmo/venmo_white_button_280x48.svg +19 -0
  27. data/app/assets/images/solidus_braintree/venmo/venmo_white_button_320x48.svg +19 -0
  28. data/app/assets/images/solidus_braintree/venmo/venmo_white_button_375x48.svg +19 -0
  29. data/app/assets/images/solidus_braintree/venmo/venmo_white_logo.svg +18 -0
  30. data/app/assets/javascripts/solidus_braintree/apple_pay_button.js +179 -0
  31. data/app/assets/javascripts/solidus_braintree/checkout.js +108 -0
  32. data/app/assets/javascripts/solidus_braintree/client.js +239 -0
  33. data/app/assets/javascripts/solidus_braintree/constants.js +89 -0
  34. data/app/assets/javascripts/solidus_braintree/frontend.js +14 -0
  35. data/app/assets/javascripts/solidus_braintree/hosted_form.js +46 -0
  36. data/app/assets/javascripts/solidus_braintree/paypal_button.js +178 -0
  37. data/app/assets/javascripts/solidus_braintree/paypal_messaging.js +22 -0
  38. data/app/assets/javascripts/solidus_braintree/promise.js +20 -0
  39. data/app/assets/javascripts/solidus_braintree/venmo_button.js +86 -0
  40. data/app/assets/javascripts/spree/backend/solidus_braintree.js +96 -0
  41. data/app/assets/javascripts/spree/frontend/paypal_button.js +34 -0
  42. data/app/assets/javascripts/spree/frontend/solidus_braintree.js +1 -0
  43. data/app/assets/stylesheets/spree/backend/solidus_braintree.scss +28 -0
  44. data/app/assets/stylesheets/spree/frontend/solidus_braintree.scss +51 -0
  45. data/app/decorators/controllers/solidus_braintree/admin_payments_controller_decorator.rb +11 -0
  46. data/app/decorators/controllers/solidus_braintree/checkout_controller_decorator.rb +11 -0
  47. data/app/decorators/controllers/solidus_braintree/client_tokens_controller.rb +41 -0
  48. data/app/decorators/controllers/solidus_braintree/orders_controller_decorator.rb +11 -0
  49. data/app/decorators/models/solidus_braintree/spree/store_decorator.rb +20 -0
  50. data/app/decorators/models/solidus_braintree/spree/user_decorator.rb +13 -0
  51. data/app/helpers/solidus_braintree/braintree_admin_helper.rb +23 -0
  52. data/app/helpers/solidus_braintree/braintree_checkout_helper.rb +60 -0
  53. data/app/models/application_record.rb +5 -0
  54. data/app/models/solidus_braintree/address.rb +64 -0
  55. data/app/models/solidus_braintree/avs_result.rb +69 -0
  56. data/app/models/solidus_braintree/configuration.rb +39 -0
  57. data/app/models/solidus_braintree/customer.rb +8 -0
  58. data/app/models/solidus_braintree/gateway.rb +433 -0
  59. data/app/models/solidus_braintree/response.rb +80 -0
  60. data/app/models/solidus_braintree/source.rb +135 -0
  61. data/app/models/solidus_braintree/transaction.rb +31 -0
  62. data/app/models/solidus_braintree/transaction_address.rb +88 -0
  63. data/app/models/solidus_braintree/transaction_import.rb +98 -0
  64. data/app/overrides/spree/payments/payment/add_paypal_funding_source_to_payment.rb +9 -0
  65. data/app/views/spree/api/payments/source_views/_braintree.json.jbuilder +1 -1
  66. data/app/views/spree/checkout/existing_payment/_braintree.html.erb +10 -0
  67. data/app/views/spree/shared/_apple_pay_button.html.erb +27 -0
  68. data/app/views/spree/shared/_braintree_errors.html.erb +16 -0
  69. data/app/views/spree/shared/_braintree_head_scripts.html.erb +26 -0
  70. data/app/views/spree/shared/_braintree_hosted_fields.html.erb +43 -0
  71. data/app/views/spree/shared/_paypal_cart_button.html.erb +38 -0
  72. data/app/views/spree/shared/_paypal_messaging.html.erb +13 -0
  73. data/app/views/spree/shared/_venmo_button.html.erb +33 -0
  74. data/bin/console +4 -1
  75. data/bin/rails +5 -5
  76. data/bin/rails-engine +13 -0
  77. data/bin/rails-sandbox +16 -0
  78. data/bin/rake +7 -0
  79. data/bin/sandbox +103 -0
  80. data/bin/setup +5 -4
  81. data/config/locales/en.yml +94 -2
  82. data/config/locales/it.yml +56 -0
  83. data/config/routes.rb +12 -3
  84. data/db/migrate/20160830061749_create_solidus_paypal_braintree_sources.rb +16 -0
  85. data/db/migrate/20160906201711_create_solidus_paypal_braintree_customers.rb +13 -0
  86. data/db/migrate/20161114231422_create_solidus_paypal_braintree_configurations.rb +11 -0
  87. data/db/migrate/20161125172005_add_braintree_configuration_to_stores.rb +7 -0
  88. data/db/migrate/20170203191030_add_credit_card_to_braintree_configuration.rb +6 -0
  89. data/db/migrate/20170505193712_add_null_constraint_to_sources.rb +38 -0
  90. data/db/migrate/20170508085402_add_not_null_constraint_to_sources_payment_type.rb +14 -0
  91. data/db/migrate/20190705115327_add_paypal_button_preferences_to_braintree_configurations.rb +5 -0
  92. data/db/migrate/20190911141712_add_3d_secure_to_braintree_configuration.rb +5 -0
  93. data/db/migrate/20211222170950_add_paypal_funding_source_to_solidus_paypal_braintree_sources.rb +5 -0
  94. data/db/migrate/20220104150301_add_venmo_to_braintree_configuration.rb +5 -0
  95. data/db/migrate/20230109080950_rename_solidus_paypal_braintree_source_type.rb +31 -0
  96. data/lib/controllers/backend/solidus_braintree/configurations_controller.rb +48 -0
  97. data/lib/controllers/frontend/solidus_braintree/checkouts_controller.rb +31 -0
  98. data/lib/controllers/frontend/solidus_braintree/transactions_controller.rb +67 -0
  99. data/lib/generators/solidus_braintree/install/install_generator.rb +54 -18
  100. data/lib/generators/solidus_braintree/install/templates/initializer.rb +6 -0
  101. data/lib/solidus_braintree/country_mapper.rb +37 -0
  102. data/lib/solidus_braintree/engine.rb +55 -10
  103. data/lib/solidus_braintree/extension_configuration.rb +23 -0
  104. data/lib/solidus_braintree/request_protection.rb +21 -0
  105. data/lib/solidus_braintree/testing_support/factories.rb +53 -0
  106. data/lib/solidus_braintree/version.rb +3 -1
  107. data/lib/solidus_braintree.rb +14 -2
  108. data/lib/solidus_paypal_braintree.rb +6 -0
  109. data/lib/views/backend/solidus_braintree/configurations/list.html.erb +63 -0
  110. data/lib/views/backend/spree/admin/payments/source_forms/_braintree.html.erb +16 -0
  111. data/lib/views/backend/spree/admin/payments/source_views/_braintree.html.erb +39 -0
  112. data/lib/views/backend/spree/admin/shared/preference_fields/_preference_select.html.erb +13 -0
  113. data/lib/views/backend_v1.2/spree/admin/payments/source_forms/_braintree.html.erb +16 -0
  114. data/lib/views/backend_v2.4/spree/admin/shared/preference_fields/_hash.html.erb +12 -0
  115. data/lib/views/frontend/solidus_braintree/payments/_payment.html.erb +12 -0
  116. data/lib/views/frontend/spree/checkout/payment/_braintree.html.erb +23 -0
  117. data/lib/views/frontend/spree/shared/_paypal_checkout_button.html.erb +32 -0
  118. data/solidus_braintree.gemspec +39 -38
  119. data/spec/controllers/solidus_braintree/checkouts_controller_spec.rb +99 -0
  120. data/spec/controllers/solidus_braintree/client_tokens_controller_spec.rb +55 -0
  121. data/spec/controllers/solidus_braintree/configurations_controller_spec.rb +73 -0
  122. data/spec/controllers/solidus_braintree/transactions_controller_spec.rb +183 -0
  123. data/spec/features/backend/configuration_spec.rb +23 -0
  124. data/spec/features/backend/new_payment_spec.rb +137 -0
  125. data/spec/features/frontend/braintree_credit_card_checkout_spec.rb +191 -0
  126. data/spec/features/frontend/paypal_checkout_spec.rb +166 -0
  127. data/spec/features/frontend/venmo_checkout_spec.rb +194 -0
  128. data/spec/fixtures/cassettes/admin/invalid_credit_card.yml +63 -0
  129. data/spec/fixtures/cassettes/admin/resubmit_credit_card.yml +352 -0
  130. data/spec/fixtures/cassettes/admin/valid_credit_card.yml +412 -0
  131. data/spec/fixtures/cassettes/braintree/create_profile.yml +71 -0
  132. data/spec/fixtures/cassettes/braintree/generate_token.yml +63 -0
  133. data/spec/fixtures/cassettes/braintree/token.yml +63 -0
  134. data/spec/fixtures/cassettes/checkout/invalid_credit_card.yml +63 -0
  135. data/spec/fixtures/cassettes/checkout/resubmit_credit_card.yml +216 -0
  136. data/spec/fixtures/cassettes/checkout/update.yml +71 -0
  137. data/spec/fixtures/cassettes/checkout/valid_credit_card.yml +171 -0
  138. data/spec/fixtures/cassettes/checkout/valid_venmo_transaction.yml +599 -0
  139. data/spec/fixtures/cassettes/gateway/authorize/credit_card/address.yml +86 -0
  140. data/spec/fixtures/cassettes/gateway/authorize/merchant_account/EUR.yml +154 -0
  141. data/spec/fixtures/cassettes/gateway/authorize/paypal/EUR.yml +90 -0
  142. data/spec/fixtures/cassettes/gateway/authorize/paypal/address.yml +90 -0
  143. data/spec/fixtures/cassettes/gateway/authorize.yml +86 -0
  144. data/spec/fixtures/cassettes/gateway/authorized_transaction.yml +73 -0
  145. data/spec/fixtures/cassettes/gateway/cancel/missing.yml +63 -0
  146. data/spec/fixtures/cassettes/gateway/cancel/refunds.yml +272 -0
  147. data/spec/fixtures/cassettes/gateway/cancel/void.yml +201 -0
  148. data/spec/fixtures/cassettes/gateway/capture.yml +141 -0
  149. data/spec/fixtures/cassettes/gateway/complete.yml +157 -0
  150. data/spec/fixtures/cassettes/gateway/credit.yml +208 -0
  151. data/spec/fixtures/cassettes/gateway/customer.yml +79 -0
  152. data/spec/fixtures/cassettes/gateway/purchase.yml +87 -0
  153. data/spec/fixtures/cassettes/gateway/settled_transaction.yml +140 -0
  154. data/spec/fixtures/cassettes/gateway/void.yml +137 -0
  155. data/spec/fixtures/cassettes/source/bin.yml +295 -0
  156. data/spec/fixtures/cassettes/source/card_type.yml +267 -0
  157. data/spec/fixtures/cassettes/source/last4.yml +267 -0
  158. data/spec/fixtures/cassettes/transaction/import/valid/capture.yml +224 -0
  159. data/spec/fixtures/cassettes/transaction/import/valid.yml +71 -0
  160. data/spec/fixtures/views/spree/orders/edit.html.erb +50 -0
  161. data/spec/helpers/solidus_braintree/braintree_admin_helper_spec.rb +17 -0
  162. data/spec/helpers/solidus_braintree/braintree_checkout_helper_spec.rb +70 -0
  163. data/spec/models/solidus_braintree/address_spec.rb +71 -0
  164. data/spec/models/solidus_braintree/avs_result_spec.rb +317 -0
  165. data/spec/models/solidus_braintree/gateway_spec.rb +742 -0
  166. data/spec/models/solidus_braintree/response_spec.rb +280 -0
  167. data/spec/models/solidus_braintree/source_spec.rb +539 -0
  168. data/spec/models/solidus_braintree/transaction_address_spec.rb +235 -0
  169. data/spec/models/solidus_braintree/transaction_import_spec.rb +302 -0
  170. data/spec/models/solidus_braintree/transaction_spec.rb +86 -0
  171. data/spec/models/spree/store_spec.rb +14 -0
  172. data/spec/requests/spree/api/orders_controller_spec.rb +36 -0
  173. data/spec/spec_helper.rb +32 -0
  174. data/spec/support/capybara.rb +7 -0
  175. data/spec/support/gateway_helpers.rb +29 -0
  176. data/spec/support/order_ready_for_payment.rb +37 -0
  177. data/spec/support/vcr.rb +42 -0
  178. data/spec/support/views.rb +1 -0
  179. metadata +276 -224
  180. data/LICENSE.txt +0 -21
  181. data/app/controllers/spree/api/braintree_client_token_controller.rb +0 -13
  182. data/app/decorators/lib/solidus_braintree/spree/permitted_attributes_decorator.rb +0 -9
  183. data/app/decorators/models/solidus_braintree/spree/credit_card_decorator.rb +0 -11
  184. data/app/decorators/models/solidus_braintree/spree/payment_decorator.rb +0 -10
  185. data/app/helpers/braintree_view_helpers.rb +0 -20
  186. data/app/models/concerns/solidus_braintree/add_name_validation_concern.rb +0 -8
  187. data/app/models/concerns/solidus_braintree/inject_device_data_concern.rb +0 -18
  188. data/app/models/concerns/solidus_braintree/payment_braintree_nonce_concern.rb +0 -8
  189. data/app/models/concerns/solidus_braintree/permitted_attributes_concern.rb +0 -11
  190. data/app/models/concerns/solidus_braintree/skip_require_card_numbers_concern.rb +0 -14
  191. data/app/models/concerns/solidus_braintree/use_data_field_concern.rb +0 -23
  192. data/app/models/solidus/gateway/braintree_gateway.rb +0 -306
  193. data/app/overrides/spree/checkout/_confirm/braintree_security.html.erb.deface +0 -9
  194. data/app/views/spree/admin/payments/source_forms/_braintree.html.erb +0 -38
  195. data/app/views/spree/admin/payments/source_views/_braintree.html.erb +0 -30
  196. data/app/views/spree/checkout/payment/_braintree.html.erb +0 -55
  197. data/app/views/spree/checkout/payment/_braintree_initialization.html.erb +0 -12
  198. data/config/initializers/braintree.rb +0 -3
  199. data/db/migrate/20150910170527_add_data_to_credit_card.rb +0 -5
  200. data/db/migrate/20160426221931_add_braintree_device_data_to_order.rb +0 -5
  201. data/lib/assets/javascripts/spree/backend/braintree/solidus_braintree.js +0 -59
  202. data/lib/assets/javascripts/spree/frontend/braintree/solidus_braintree.js +0 -144
  203. data/lib/assets/javascripts/vendor/braintree.js +0 -8
  204. data/lib/assets/stylesheets/spree/frontend/solidus_braintree.scss +0 -26
@@ -0,0 +1,55 @@
1
+ require 'spec_helper'
2
+
3
+ describe SolidusBraintree::ClientTokensController do
4
+ routes { SolidusBraintree::Engine.routes }
5
+
6
+ cassette_options = { cassette_name: "braintree/token" }
7
+ describe "POST create", vcr: cassette_options do
8
+ let!(:gateway) { create_gateway }
9
+ let(:user) { create(:user) }
10
+ let(:json) { JSON.parse(response.body) }
11
+
12
+ before { user.generate_spree_api_key! }
13
+
14
+ context 'without a payment method id' do
15
+ subject(:response) do
16
+ post :create, params: { token: user.spree_api_key }
17
+ end
18
+
19
+ it "returns a client token", aggregate_failures: true do
20
+ expect(response).to have_http_status(:success)
21
+ expect(response.content_type).to include 'application/json'
22
+ expect(json["client_token"]).to be_present
23
+ expect(json["client_token"]).to be_a String
24
+ expect(json["payment_method_id"]).to eq gateway.id
25
+ end
26
+
27
+ context "when there's two gateway's for different stores" do
28
+ let!(:store1) { create(:store, code: 'store_1') }
29
+ let!(:store2) { create(:store, code: 'store_2') }
30
+ let!(:gateway_for_store1) { create_gateway.tap{ |gw| store1.payment_methods << gw } }
31
+ let!(:gateway_for_store2) { create_gateway.tap{ |gw| store2.payment_methods << gw } }
32
+
33
+ it "returns the correct gateway for store1" do
34
+ allow_any_instance_of(described_class).to receive(:current_store).and_return store1
35
+ expect(json["payment_method_id"]).to eq gateway_for_store1.id
36
+ end
37
+
38
+ it "returns the correct gateway for store2" do
39
+ allow_any_instance_of(described_class).to receive(:current_store).and_return store2
40
+ expect(json["payment_method_id"]).to eq gateway_for_store2.id
41
+ end
42
+ end
43
+ end
44
+
45
+ context 'with a payment method id' do
46
+ subject(:response) do
47
+ post :create, params: { token: user.spree_api_key, payment_method_id: gateway.id }
48
+ end
49
+
50
+ it 'uses the selected gateway' do
51
+ expect(json["payment_method_id"]).to eq gateway.id
52
+ end
53
+ end
54
+ end
55
+ end
@@ -0,0 +1,73 @@
1
+ require 'spec_helper'
2
+
3
+ describe SolidusBraintree::ConfigurationsController, type: :controller do
4
+ routes { SolidusBraintree::Engine.routes }
5
+
6
+ let!(:store_1) { create :store }
7
+ let!(:store_2) { create :store }
8
+
9
+ stub_authorization!
10
+
11
+ describe "GET #list" do
12
+ subject { get :list }
13
+
14
+ it "assigns all store's configurations as @configurations" do
15
+ subject
16
+ expect(assigns(:configurations)).
17
+ to eq [store_1.braintree_configuration, store_2.braintree_configuration]
18
+ end
19
+
20
+ it "renders the correct view" do
21
+ expect(subject).to render_template :list
22
+ end
23
+ end
24
+
25
+ describe "POST #update" do
26
+ subject { post :update, params: configurations_params }
27
+
28
+ let(:paypal_button_color) { 'blue' }
29
+ let(:configurations_params) do
30
+ {
31
+ configurations: {
32
+ configuration_fields: {
33
+ store_1.braintree_configuration.id.to_s => { paypal: true, apple_pay: true },
34
+ store_2.braintree_configuration.id.to_s => {
35
+ paypal: true,
36
+ apple_pay: false,
37
+ preferred_paypal_button_color: paypal_button_color
38
+ }
39
+ }
40
+ }
41
+ }
42
+ end
43
+
44
+ context "with valid parameters" do
45
+ it "updates the configuration" do
46
+ expect { subject }.to change { store_1.braintree_configuration.reload.paypal }.
47
+ from(false).to(true)
48
+ end
49
+
50
+ it "displays a success message to the user" do
51
+ subject
52
+ expect(flash[:success]).to eq "Successfully updated Braintree configurations."
53
+ end
54
+
55
+ it "displays all configurations" do
56
+ expect(subject).to redirect_to action: :list
57
+ end
58
+ end
59
+
60
+ context "with invalid parameters" do
61
+ let(:paypal_button_color) { 'invalid-color' }
62
+
63
+ it "displays an error message to the user" do
64
+ subject
65
+ expect(flash[:error]).to eq "An error occurred while updating Braintree configurations."
66
+ end
67
+
68
+ it "returns the user to the edit page" do
69
+ expect(subject).to redirect_to action: :list
70
+ end
71
+ end
72
+ end
73
+ end
@@ -0,0 +1,183 @@
1
+ require 'spec_helper'
2
+
3
+ RSpec.describe SolidusBraintree::TransactionsController, type: :controller do
4
+ routes { SolidusBraintree::Engine.routes }
5
+
6
+ let!(:country) { create :country }
7
+ let(:line_item) { create :line_item, price: 50 }
8
+ let(:order) do
9
+ Spree::Order.create!(
10
+ line_items: [line_item],
11
+ email: 'test@example.com',
12
+ bill_address: create(:address, country: country),
13
+ ship_address: create(:address, country: country),
14
+ user: create(:user)
15
+ )
16
+ end
17
+
18
+ let(:payment_method) { create_gateway }
19
+
20
+ before do
21
+ allow(controller).to receive(:spree_current_user) { order.user }
22
+ allow(controller).to receive(:current_order) { order }
23
+ create :shipping_method, cost: 5
24
+ create :state, abbr: "WA", country: country
25
+ end
26
+
27
+ cassette_options = {
28
+ cassette_name: "transactions_controller/create",
29
+ match_requests_on: [:braintree_uri]
30
+ }
31
+ describe "POST create", vcr: cassette_options do
32
+ subject(:post_create) { post :create, params: params }
33
+
34
+ let!(:country) { create :country, iso: 'US' }
35
+
36
+ let(:params) do
37
+ {
38
+ transaction: {
39
+ nonce: "fake-valid-nonce",
40
+ payment_type: SolidusBraintree::Source::PAYPAL,
41
+ phone: "1112223333",
42
+ email: "batman@example.com",
43
+ address_attributes: {
44
+ name: "Wade Wilson",
45
+ address_line_1: "123 Fake Street",
46
+ city: "Seattle",
47
+ zip: "98101",
48
+ state_code: "WA",
49
+ country_code: "US"
50
+ }
51
+ },
52
+ payment_method_id: payment_method.id
53
+ }
54
+ end
55
+
56
+ before do
57
+ create :state, abbr: "WA", country: country
58
+ end
59
+
60
+ context "when import has invalid address" do
61
+ before { params[:transaction][:address_attributes][:city] = nil }
62
+
63
+ it "raises a validation error" do
64
+ expect { post_create }.to raise_error(
65
+ SolidusBraintree::TransactionsController::InvalidImportError,
66
+ "Import invalid: " \
67
+ "Address is invalid, " \
68
+ "Address City can't be blank"
69
+ )
70
+ end
71
+ end
72
+
73
+ context "when the transaction is valid", vcr: {
74
+ cassette_name: 'transaction/import/valid',
75
+ match_requests_on: [:braintree_uri]
76
+ } do
77
+ it "imports the payment" do
78
+ expect { post_create }.to change { order.payments.count }.by(1)
79
+ expect(order.payments.first.amount).to eq 55
80
+ end
81
+
82
+ context "when no end state is provided" do
83
+ it "advances the order to confirm" do
84
+ post_create
85
+ expect(order).to be_confirm
86
+ end
87
+ end
88
+
89
+ context "when end state provided is delivery" do
90
+ let(:params) { super().merge(state: 'delivery') }
91
+
92
+ it "advances the order to delivery" do
93
+ post_create
94
+ expect(order).to be_delivery
95
+ end
96
+ end
97
+
98
+ context "with provided address" do
99
+ it "creates a new address" do
100
+ # Creating the order also creates 3 addresses, we want to make sure
101
+ # the transaction import only creates 1 new one
102
+ order
103
+ expect { post_create }.to change(Spree::Address, :count).by(1)
104
+ expect(Spree::Address.last.address1).to eq "123 Fake Street"
105
+ end
106
+ end
107
+
108
+ context "without country ISO" do
109
+ before do
110
+ params[:transaction][:address_attributes][:country_code] = ""
111
+ params[:transaction][:address_attributes][:country_name] = "United States"
112
+ end
113
+
114
+ it "creates a new address, looking up the ISO by country name" do
115
+ order
116
+ expect { post_create }.to change(Spree::Address, :count).by(1)
117
+ expect(Spree::Address.last.country.iso).to eq "US"
118
+ end
119
+ end
120
+
121
+ context "without transaction address" do
122
+ before { params[:transaction].delete(:address_attributes) }
123
+
124
+ it "does not create a new address" do
125
+ order
126
+ expect { post_create }.not_to(change(Spree::Address, :count))
127
+ end
128
+ end
129
+
130
+ context "when format is HTML" do
131
+ context "when import! leaves the order in confirm" do
132
+ it "redirects the user to the confirm page" do
133
+ expect(post_create).to redirect_to spree.checkout_state_path("confirm")
134
+ end
135
+ end
136
+
137
+ context "when import! completes the order" do
138
+ before { allow(order).to receive(:complete?).and_return(true) }
139
+
140
+ it "displays the order to the user" do
141
+ expect(post_create).to redirect_to spree.order_path(order)
142
+ end
143
+ end
144
+ end
145
+
146
+ context "when format is JSON" do
147
+ before { params[:format] = :json }
148
+
149
+ it "has a successful response" do
150
+ post_create
151
+ expect(response).to be_successful
152
+ end
153
+ end
154
+ end
155
+
156
+ context "when the transaction is invalid" do
157
+ before { params[:transaction][:email] = nil }
158
+
159
+ context "when format is HTML" do
160
+ it "raises an error including the validation messages" do
161
+ expect { post_create }.to raise_error(
162
+ SolidusBraintree::TransactionsController::InvalidImportError,
163
+ "Import invalid: Email can't be blank"
164
+ )
165
+ end
166
+ end
167
+
168
+ context "when format is JSON" do
169
+ before { params[:format] = :json }
170
+
171
+ it "has a failed status" do
172
+ post_create
173
+ expect(response).to have_http_status :unprocessable_entity
174
+ end
175
+
176
+ it "returns the errors as JSON" do
177
+ post_create
178
+ expect(JSON.parse(response.body)["errors"]["email"]).to eq ["can't be blank"]
179
+ end
180
+ end
181
+ end
182
+ end
183
+ end
@@ -0,0 +1,23 @@
1
+ require 'spec_helper'
2
+
3
+ RSpec.describe "viewing the configuration interface" do
4
+ stub_authorization!
5
+
6
+ # Regression to ensure this page still renders on old versions of solidus
7
+ it "doesn't raise any errors due to unavailable route helpers" do
8
+ visit "/solidus_braintree/configurations/list"
9
+ expect(page).to have_content("Braintree Configurations")
10
+ end
11
+
12
+ # Regression to ensure this page renders on Solidus 2.4
13
+ it "doesn't raise any errors due to unavailable preference field partial" do
14
+ Rails.application.config.spree.payment_methods << SolidusBraintree::Gateway
15
+ Spree::PaymentMethod.create!(
16
+ type: 'SolidusBraintree::Gateway',
17
+ name: 'Braintree Payments'
18
+ )
19
+ visit '/admin/payment_methods'
20
+ page.find('a[title="Edit"]').click
21
+ expect(page).to have_field 'Name', with: 'Braintree Payments'
22
+ end
23
+ end
@@ -0,0 +1,137 @@
1
+ require 'spec_helper'
2
+ require 'spree/testing_support/order_walkthrough'
3
+
4
+ shared_context "with backend checkout setup" do
5
+ let(:braintree) { new_gateway(active: true) }
6
+ let!(:gateway) { create :payment_method }
7
+ let(:order) { create(:completed_order_with_totals, number: 'R9999999') }
8
+ let(:pending_case_insensitive) { /pending/i }
9
+ let(:expiration) { "02/#{Date.current.year.next}" }
10
+
11
+ before do
12
+ braintree.save!
13
+ create(:store, payment_methods: [gateway, braintree]).tap do |store|
14
+ store.braintree_configuration.update!(credit_card: true)
15
+ end
16
+
17
+ allow_any_instance_of(SolidusBraintree::Source).to receive(:nonce).and_return("fake-valid-nonce")
18
+
19
+ # Order and payment numbers must be identical between runs to re-use the VCR
20
+ # cassette
21
+ allow_any_instance_of(Spree::Payment).to receive(:number).and_return("123ABC")
22
+ end
23
+
24
+ around do |example|
25
+ Capybara.using_wait_time(20) { example.run }
26
+ end
27
+ end
28
+
29
+ describe 'creating a new payment', type: :feature, js: true do
30
+ stub_authorization!
31
+
32
+ context "with valid credit card data", vcr: {
33
+ cassette_name: 'admin/valid_credit_card',
34
+ match_requests_on: [:braintree_uri]
35
+ } do
36
+ include_context "with backend checkout setup"
37
+
38
+ it "checks out successfully" do
39
+ visit "/admin/orders/#{order.number}/payments/new"
40
+ choose('Braintree')
41
+ expect(page).to have_selector("#payment_method_#{braintree.id}", visible: :visible)
42
+ expect(page).to have_selector("iframe#braintree-hosted-field-number")
43
+
44
+ within_frame("braintree-hosted-field-number") do
45
+ fill_in("credit-card-number", with: "4111111111111111")
46
+ end
47
+ within_frame("braintree-hosted-field-expirationDate") do
48
+ fill_in("expiration", with: expiration)
49
+ end
50
+ within_frame("braintree-hosted-field-cvv") do
51
+ fill_in("cvv", with: "123")
52
+ end
53
+
54
+ click_button("Update")
55
+
56
+ within('table#payments') do
57
+ expect(page).to have_content('Braintree')
58
+ expect(page).to have_content(pending_case_insensitive)
59
+ end
60
+
61
+ click_icon(:capture)
62
+
63
+ expect(page).not_to have_content('Cannot perform requested operation')
64
+ expect(page).to have_content('Payment Updated')
65
+ end
66
+ end
67
+
68
+ context "with invalid credit card data" do
69
+ include_context "with backend checkout setup"
70
+
71
+ # Attempt to submit an invalid form once
72
+ before do
73
+ visit "/admin/orders/#{order.number}/payments/new"
74
+ choose('Braintree')
75
+
76
+ within_frame("braintree-hosted-field-number") do
77
+ fill_in("credit-card-number", with: "1111111111111111")
78
+ end
79
+ within_frame("braintree-hosted-field-expirationDate") do
80
+ fill_in("expiration", with: expiration)
81
+ end
82
+ within_frame("braintree-hosted-field-cvv") do
83
+ fill_in("cvv", with: "123")
84
+ end
85
+
86
+ click_button "Update"
87
+ end
88
+
89
+ it "displays a meaningful error message" do
90
+ expect(page).to have_text(
91
+ "BraintreeError: Some payment input fields are invalid. Cannot tokenize invalid card fields."
92
+ )
93
+ end
94
+
95
+ # Same error should be produced when submitting an empty form again
96
+ context "when user tries to resubmit another invalid form", vcr: {
97
+ cassette_name: "admin/invalid_credit_card",
98
+ match_requests_on: [:braintree_uri]
99
+ } do
100
+ it "displays a meaningful error message" do
101
+ click_button "Update"
102
+ expect(page).to have_text(
103
+ "BraintreeError: Some payment input fields are invalid. Cannot tokenize invalid card fields."
104
+ )
105
+ end
106
+ end
107
+
108
+ # User should be able to checkout after submit fails once
109
+ context "when user enters valid data", vcr: {
110
+ cassette_name: "admin/resubmit_credit_card",
111
+ match_requests_on: [:braintree_uri]
112
+ } do
113
+ it "creates the payment successfully" do
114
+ within_frame("braintree-hosted-field-number") do
115
+ fill_in("credit-card-number", with: "4111111111111111")
116
+ end
117
+ within_frame("braintree-hosted-field-expirationDate") do
118
+ fill_in("expiration", with: expiration)
119
+ end
120
+ within_frame("braintree-hosted-field-cvv") do
121
+ fill_in("cvv", with: "123")
122
+ end
123
+ click_button("Update")
124
+
125
+ within('table#payments') do
126
+ expect(page).to have_content('Braintree')
127
+ expect(page).to have_content(pending_case_insensitive)
128
+ end
129
+
130
+ click_icon(:capture)
131
+
132
+ expect(page).not_to have_content('Cannot perform requested operation')
133
+ expect(page).to have_content('Payment Updated')
134
+ end
135
+ end
136
+ end
137
+ end
@@ -0,0 +1,191 @@
1
+ require 'spec_helper'
2
+ require 'spree/testing_support/order_walkthrough'
3
+
4
+ shared_context "with frontend checkout setup" do
5
+ let(:braintree) { new_gateway(active: true) }
6
+ let!(:gateway) { create :payment_method }
7
+ let(:three_d_secure_enabled) { false }
8
+ let(:venmo_enabled) { false }
9
+ let(:card_number) { "4111111111111111" }
10
+ let(:card_expiration) { "01/#{Time.now.utc.year + 3}" }
11
+
12
+ before do
13
+ braintree.save!
14
+
15
+ create(:store, payment_methods: [gateway, braintree]).tap do |store|
16
+ store.braintree_configuration.update!(
17
+ credit_card: true,
18
+ three_d_secure: three_d_secure_enabled,
19
+ venmo: venmo_enabled
20
+ )
21
+
22
+ braintree.update(
23
+ preferred_credit_card_fields_style: { input: { 'font-size': '30px' } },
24
+ preferred_placeholder_text: { number: "Enter Your Card Number" }
25
+ )
26
+ end
27
+
28
+ order = if Spree.solidus_gem_version >= Gem::Version.new('2.6.0')
29
+ Spree::TestingSupport::OrderWalkthrough.up_to(:delivery)
30
+ else
31
+ OrderWalkthrough.up_to(:delivery)
32
+ end
33
+
34
+ user = create(:user)
35
+ order.user = user
36
+ order.number = "R9999999"
37
+ order.recalculate
38
+
39
+ allow_any_instance_of(Spree::CheckoutController).to receive_messages(current_order: order)
40
+ allow_any_instance_of(Spree::CheckoutController).to receive_messages(try_spree_current_user: user)
41
+ allow_any_instance_of(Spree::CheckoutController).to receive_messages(spree_current_user: user)
42
+ allow_any_instance_of(Spree::Payment).to receive(:number).and_return("123ABC")
43
+ allow_any_instance_of(SolidusBraintree::Source).to receive(:nonce).and_return("fake-valid-nonce")
44
+
45
+ visit spree.checkout_state_path(:delivery)
46
+ click_button "Save and Continue"
47
+ choose("Braintree")
48
+ end
49
+
50
+ around do |example|
51
+ Capybara.using_wait_time(20) { example.run }
52
+ end
53
+ end
54
+
55
+ describe 'entering credit card details', type: :feature, js: true do
56
+ context 'when page loads' do
57
+ include_context "with frontend checkout setup"
58
+
59
+ it "selectors display correctly" do
60
+ expect(page).to have_selector("#payment_method_#{braintree.id}", visible: :visible)
61
+ expect(page).to have_selector("iframe#braintree-hosted-field-number")
62
+ expect(page).to have_selector("iframe[type='number']")
63
+ end
64
+
65
+ it "credit card field style variable is set" do
66
+ within_frame("braintree-hosted-field-number") do
67
+ expect(find("#credit-card-number").style("font-size")).to eq({ "font-size" => "30px" })
68
+ end
69
+ end
70
+
71
+ it "sets the placeholder text correctly" do
72
+ within_frame("braintree-hosted-field-number") do
73
+ expect(find("#credit-card-number")['placeholder']).to eq("Enter Your Card Number")
74
+ end
75
+ end
76
+ end
77
+
78
+ context "with valid credit card data", vcr: {
79
+ cassette_name: 'checkout/valid_credit_card',
80
+ match_requests_on: [:braintree_uri]
81
+ } do
82
+ include_context "with frontend checkout setup"
83
+ # To ensure Venmo inputs do not conflict with checkout
84
+ let(:venmo_enabled) { true }
85
+
86
+ before do
87
+ within_frame("braintree-hosted-field-number") do
88
+ fill_in("credit-card-number", with: card_number)
89
+ end
90
+ within_frame("braintree-hosted-field-expirationDate") do
91
+ fill_in("expiration", with: card_expiration)
92
+ end
93
+ within_frame("braintree-hosted-field-cvv") do
94
+ fill_in("cvv", with: "123")
95
+ end
96
+
97
+ click_button("Save and Continue")
98
+ end
99
+
100
+ it "checks out successfully" do
101
+ within("#order_details") do
102
+ expect(page).to have_content("CONFIRM")
103
+ end
104
+ click_button("Place Order")
105
+ expect(page).to have_content("Your order has been processed successfully")
106
+ end
107
+
108
+ context 'with 3D secure enabled' do
109
+ let(:three_d_secure_enabled) { true }
110
+
111
+ it 'checks out successfully' do
112
+ authenticate_3ds
113
+
114
+ within("#order_details") do
115
+ expect(page).to have_content("CONFIRM")
116
+ end
117
+
118
+ click_button("Place Order")
119
+ expect(page).to have_content("Your order has been processed successfully")
120
+ end
121
+
122
+ context 'with 3ds authentication error' do
123
+ let(:card_number) { "4000000000001125" }
124
+
125
+ it 'shows a 3ds authentication error' do
126
+ authenticate_3ds
127
+ expect(page).to have_content(
128
+ "3D Secure authentication failed. Please try again using a different payment method."
129
+ )
130
+ end
131
+ end
132
+ end
133
+ end
134
+
135
+ context "with invalid credit card data" do
136
+ include_context "with frontend checkout setup"
137
+
138
+ # Attempt to submit an empty form once
139
+ before do
140
+ click_button "Save and Continue"
141
+ end
142
+
143
+ it "displays an alert with a meaningful error message" do
144
+ expect(page).to have_text I18n.t("solidus_braintree.errors.empty_fields")
145
+ expect(page).to have_selector("input[type='submit']:enabled")
146
+ end
147
+
148
+ # Same error should be produced when submitting an empty form again
149
+ context "when user tries to resubmit an empty form", vcr: { cassette_name: "checkout/invalid_credit_card" } do
150
+ it "displays an alert with a meaningful error message" do
151
+ expect(page).to have_selector("input[type='submit']:enabled")
152
+
153
+ click_button "Save and Continue"
154
+ expect(page).to have_text I18n.t("solidus_braintree.errors.empty_fields")
155
+ end
156
+ end
157
+
158
+ # User should be able to checkout after submit fails once
159
+ context "when user enters valid data", vcr: {
160
+ cassette_name: "checkout/resubmit_credit_card",
161
+ match_requests_on: [:braintree_uri]
162
+ } do
163
+ it "allows them to resubmit and complete the purchase" do
164
+ within_frame("braintree-hosted-field-number") do
165
+ fill_in("credit-card-number", with: "4111111111111111")
166
+ end
167
+ within_frame("braintree-hosted-field-expirationDate") do
168
+ fill_in("expiration", with: card_expiration)
169
+ end
170
+ within_frame("braintree-hosted-field-cvv") do
171
+ fill_in("cvv", with: "123")
172
+ end
173
+ click_button("Save and Continue")
174
+ within("#order_details") do
175
+ expect(page).to have_content("CONFIRM")
176
+ end
177
+ click_button("Place Order")
178
+ expect(page).to have_content("Your order has been processed successfully")
179
+ end
180
+ end
181
+ end
182
+
183
+ def authenticate_3ds
184
+ within_frame("Cardinal-CCA-IFrame") do
185
+ fill_in("challengeDataEntry", with: "1234")
186
+ continue_button = find_button("SUBMIT")
187
+ continue_button.scroll_to(continue_button)
188
+ continue_button.click
189
+ end
190
+ end
191
+ end