diff --git a/.travis.yml b/.travis.yml index 29d4f67c..aa52ca55 100644 --- a/.travis.yml +++ b/.travis.yml @@ -1,54 +1,63 @@ -sudo: required -dist: trusty +--- +os: linux +dist: bionic + +addons: + apt: + sources: + - google-chrome + packages: + - google-chrome-stable + +services: + - mysql + - postgresql language: ruby -script: - - bundle exec rake test_app - - bundle exec rake spec +rvm: + - 2.5 + - 2.6 + - 2.7 env: - DB=mysql - DB=postgres -addons: - chrome: stable - postgresql: 9.4 - -rvm: - - 2.3.8 - - 2.4.4 - - 2.5.1 - gemfile: - - gemfiles/spree_3_5.gemfile - gemfiles/spree_3_7.gemfile - gemfiles/spree_4_0.gemfile + - gemfiles/spree_4_1.gemfile - gemfiles/spree_master.gemfile -before_install: - - mysql -u root -e "GRANT ALL ON *.* TO 'travis'@'%';" - -matrix: +jobs: exclude: - - rvm: 2.5.1 - gemfile: gemfiles/spree_3_5.gemfile - - rvm: 2.4.4 + - rvm: 2.6 + gemfile: gemfiles/spree_3_7.gemfile + - rvm: 2.7 + gemfile: gemfiles/spree_3_7.gemfile + - rvm: 2.5 gemfile: gemfiles/spree_4_0.gemfile - - rvm: 2.3.8 + - rvm: 2.7 gemfile: gemfiles/spree_4_0.gemfile - - rvm: 2.4.4 - gemfile: gemfiles/spree_master.gemfile - - rvm: 2.3.8 + - rvm: 2.5 + gemfile: gemfiles/spree_4_1.gemfile + - rvm: 2.7 + gemfile: gemfiles/spree_4_1.gemfile + - rvm: 2.5 gemfile: gemfiles/spree_master.gemfile allow_failures: - - gemfile: gemfiles/spree_master.gemfile + - gemfile: gemfiles/spree_master.gemfile before_install: - mysql -u root -e "GRANT ALL ON *.* TO 'travis'@'%';" - - wget -N https://chromedriver.storage.googleapis.com/2.35/chromedriver_linux64.zip -P ~/ - - unzip ~/chromedriver_linux64.zip -d ~/ - - rm ~/chromedriver_linux64.zip - - sudo mv -f ~/chromedriver /usr/local/share/ - - sudo chmod +x /usr/local/share/chromedriver - - sudo ln -s /usr/local/share/chromedriver /usr/local/bin/chromedriver + +before_script: + - CHROME_MAIN_VERSION=`google-chrome-stable --version | sed -E 's/(^Google Chrome |\.[0-9]+ )//g'` + - CHROMEDRIVER_VERSION=`curl -s "https://chromedriver.storage.googleapis.com/LATEST_RELEASE_$CHROME_MAIN_VERSION"` + - curl "https://chromedriver.storage.googleapis.com/${CHROMEDRIVER_VERSION}/chromedriver_linux64.zip" -O + - unzip chromedriver_linux64.zip -d ~/bin + +script: + - bundle exec rake test_app + - bundle exec rake spec diff --git a/Appraisals b/Appraisals index 15dc1ed5..fafce4bb 100644 --- a/Appraisals +++ b/Appraisals @@ -1,15 +1,15 @@ -appraise 'spree-3-5' do - gem 'spree', '~> 3.5.0' - gem 'rails-controller-testing' -end - appraise 'spree-3-7' do gem 'spree', '~> 3.7.0' gem 'rails-controller-testing' end appraise 'spree-4-0' do - gem 'spree', '~> 4.0.0.rc2' + gem 'spree', '~> 4.0.0' + gem 'rails-controller-testing' +end + +appraise 'spree-4-1' do + gem 'spree', '~> 4.1.0' gem 'rails-controller-testing' end diff --git a/app/assets/javascripts/spree/frontend/spree_paypal_express.js b/app/assets/javascripts/spree/frontend/spree_paypal_express.js index 09c80d8d..551c9cc7 100644 --- a/app/assets/javascripts/spree/frontend/spree_paypal_express.js +++ b/app/assets/javascripts/spree/frontend/spree_paypal_express.js @@ -23,7 +23,7 @@ SpreePaypalExpress = { } } -$(document).ready(function() { +document.addEventListener("DOMContentLoaded", function() { SpreePaypalExpress.updateSaveAndContinueVisibility(); paymentMethods = $('div[data-hook="checkout_payment_step"] input[type="radio"]').click(function (e) { SpreePaypalExpress.updateSaveAndContinueVisibility(); diff --git a/app/models/spree/gateway/pay_pal_express.rb b/app/models/spree/gateway/pay_pal_express.rb index fa47cb0c..01b55d22 100644 --- a/app/models/spree/gateway/pay_pal_express.rb +++ b/app/models/spree/gateway/pay_pal_express.rb @@ -19,11 +19,13 @@ def provider_class def provider ::PayPal::SDK.configure( - :mode => preferred_server.present? ? preferred_server : "sandbox", - :username => preferred_login, - :password => preferred_password, - :signature => preferred_signature) - provider_class.new + mode: preferred_server.present? ? preferred_server : "sandbox", + username: preferred_login, + password: preferred_password, + signature: preferred_signature, + ssl_options: { ca_file: nil } + ) + provider_class.new end def auto_capture? diff --git a/app/views/spree/checkout/payment/_paypal.html.erb b/app/views/spree/checkout/payment/_paypal.html.erb index f01452e3..d4db433e 100644 --- a/app/views/spree/checkout/payment/_paypal.html.erb +++ b/app/views/spree/checkout/payment/_paypal.html.erb @@ -2,5 +2,7 @@ <%= link_to(image_tag("https://www.paypal.com/en_US/i/btn/btn_xpressCheckout.gif"), paypal_express_url(:payment_method_id => payment_method.id), :method => :post, :id => "paypal_button") %> \ No newline at end of file + document.addEventListener("DOMContentLoaded", function() { + SpreePaypalExpress.paymentMethodID = "<%= payment_method.id %>" + }) + diff --git a/config/locales/en.yml b/config/locales/en.yml index e0722847..e399e972 100644 --- a/config/locales/en.yml +++ b/config/locales/en.yml @@ -3,7 +3,7 @@ en: spree: paypal: already_refunded: "This payment has been refunded and no further action can be taken on it." - no_payment_via_admin_backend: "You cannot charge PayPal accounts through the admin backend at this time." + no_payment_via_admin_backend: "You cannot charge PayPal accounts through the admin back-end at this time." transaction: "PayPal Transaction" payer_id: "Payer ID" transaction_id: "Transaction ID" @@ -18,4 +18,4 @@ en: flash: cancel: "Don't want to use PayPal? No problems." connection_failed: "Could not connect to PayPal." - generic_error: "PayPal failed. %{reasons}" \ No newline at end of file + generic_error: "PayPal failed. %{reasons}" diff --git a/gemfiles/spree_3_5.gemfile b/gemfiles/spree_3_5.gemfile deleted file mode 100644 index 823705d5..00000000 --- a/gemfiles/spree_3_5.gemfile +++ /dev/null @@ -1,8 +0,0 @@ -# This file was generated by Appraisal - -source "https://rubygems.org" - -gem "spree", "~> 3.5.0" -gem "rails-controller-testing" - -gemspec path: "../" diff --git a/gemfiles/spree_3_7.gemfile b/gemfiles/spree_3_7.gemfile index ae4b22d4..1af800cf 100644 --- a/gemfiles/spree_3_7.gemfile +++ b/gemfiles/spree_3_7.gemfile @@ -1,9 +1,9 @@ # This file was generated by Appraisal -source "https://rubygems.org" +source 'https://rubygems.org' -gem "spree", "~> 3.7.0" -gem "rails-controller-testing" -gem "sass-rails" +gem 'spree', '~> 3.7.0' +gem 'rails-controller-testing' +gem 'sass-rails' -gemspec path: "../" +gemspec path: '../' diff --git a/gemfiles/spree_4_0.gemfile b/gemfiles/spree_4_0.gemfile index cda9cc3f..fb229df6 100644 --- a/gemfiles/spree_4_0.gemfile +++ b/gemfiles/spree_4_0.gemfile @@ -1,8 +1,8 @@ # This file was generated by Appraisal -source "https://rubygems.org" +source 'https://rubygems.org' -gem "spree", "~> 4.0.0.rc2" -gem "rails-controller-testing" +gem 'spree', '~> 4.0.0' +gem 'rails-controller-testing' -gemspec path: "../" +gemspec path: '../' diff --git a/gemfiles/spree_4_1.gemfile b/gemfiles/spree_4_1.gemfile new file mode 100644 index 00000000..429ff44c --- /dev/null +++ b/gemfiles/spree_4_1.gemfile @@ -0,0 +1,8 @@ +# This file was generated by Appraisal + +source 'https://rubygems.org' + +gem 'spree', '~> 4.1.0' +gem 'rails-controller-testing' + +gemspec path: '../' diff --git a/gemfiles/spree_master.gemfile b/gemfiles/spree_master.gemfile index 14964f7c..d30062b0 100644 --- a/gemfiles/spree_master.gemfile +++ b/gemfiles/spree_master.gemfile @@ -1,8 +1,8 @@ # This file was generated by Appraisal -source "https://rubygems.org" +source 'https://rubygems.org' -gem "spree", github: "spree/spree", branch: "master" -gem "rails-controller-testing" +gem 'spree', github: 'spree/spree', branch: 'master' +gem 'rails-controller-testing' -gemspec path: "../" +gemspec path: '../' diff --git a/spec/features/paypal_spec.rb b/spec/features/paypal_spec.rb index 754a0622..2144468d 100644 --- a/spec/features/paypal_spec.rb +++ b/spec/features/paypal_spec.rb @@ -1,7 +1,13 @@ -describe 'PayPal', js: true do - let!(:product) { create(:product, name: 'iPad') } - let!(:country) { create(:country, name: 'United States') } - let!(:state) { create(:state, country: country)} +require 'spec_helper' + +describe 'PayPal', type: :feature, js: true do + let!(:country) { create(:country, name: 'United States', states_required: true) } + let!(:state) { create(:state, country: country) } + let!(:shipping_method) { create(:shipping_method) } + let!(:stock_location) { create(:stock_location) } + let!(:product) { create(:product, name: 'RoR Mug') } + let!(:zone) { create(:zone) } + let!(:store) { create(:store) } before do @gateway = Spree::Gateway::PayPalExpress.create!({ @@ -11,23 +17,11 @@ name: 'PayPal', active: true }) - create(:shipping_method) - end - - def fill_in_billing - fill_in :order_bill_address_attributes_firstname, with: 'Test' - fill_in :order_bill_address_attributes_lastname, with: 'User' - fill_in :order_bill_address_attributes_address1, with: '1 User Lane' - fill_in :order_bill_address_attributes_city, with: 'Adamsville' - select 'United States', from: :order_bill_address_attributes_country_id - find('#order_bill_address_attributes_state_id').find(:xpath, 'option[2]').select_option - fill_in :order_bill_address_attributes_zipcode, with: '35005' - fill_in :order_bill_address_attributes_phone, with: '555-123-4567' end def switch_to_paypal_login unless page.has_selector?('#login #email') - if page.has_css?('.changeLanguage') + if page.has_css?('.changeLanguage') wait_for { !page.has_css?('div#preloaderSpinner') } find('.changeLanguage').click find_all('a', text: 'English')[0].click @@ -64,16 +58,19 @@ def within_transaction_cart(container_class, expected_texts, unexpected_texts) find('#closeCart').click end - def add_to_cart(product) - visit spree.root_path - click_link product.name - click_button 'Add To Cart' - sleep(1) - visit spree.cart_path + def fill_in_billing + fill_in :order_bill_address_attributes_firstname, with: 'Test' + fill_in :order_bill_address_attributes_lastname, with: 'User' + fill_in :order_bill_address_attributes_address1, with: '1 User Lane' + fill_in :order_bill_address_attributes_city, with: 'Adamsville' + select 'United States', from: :order_bill_address_attributes_country_id + find('#order_bill_address_attributes_state_id').find(:xpath, 'option[2]').select_option + fill_in :order_bill_address_attributes_zipcode, with: '35005' + fill_in :order_bill_address_attributes_phone, with: '555-123-4567' end def fill_in_guest - fill_in :order_email, with: 'test@example.com' + fill_in :order_email, with: 'ryanbigg@me.com' end def click_pay_now_button @@ -100,11 +97,27 @@ def expect_successfully_processed_order expect(page).to have_content('Your order has been processed successfully') end - it 'pays for an order successfully' do - add_to_cart(product) - click_button 'Checkout' - fill_in_guest - fill_in_billing + def proceed_through_address_stage + if Spree.version.to_f < 4.1 + visit spree.root_path + click_link product.name + click_button 'Add To Cart' + sleep(1) + visit spree.cart_path + click_button 'Checkout' + fill_in_guest + fill_in_billing + else + add_to_cart(product) + visit spree.checkout_path + fill_in_guest + fill_in_address + end + end + + it 'pays for an order successfully', js: true do + proceed_through_address_stage + click_button 'Save and Continue' click_button 'Save and Continue' @@ -123,10 +136,8 @@ def expect_successfully_processed_order end it 'passes user details to PayPal' do - add_to_cart(product) - click_button 'Checkout' - fill_in_guest - fill_in_billing + proceed_through_address_stage + click_button 'Save and Continue' click_button 'Save and Continue' @@ -135,20 +146,27 @@ def expect_successfully_processed_order login_to_paypal stay_logged_in_for_faster_checkout click_pay_now_button - wait_for { page.has_text?('555-123-4567') } - within('#order_summary') do - expect(page).to have_selector '[data-hook=order-bill-address] .fn', text: 'Test User' - expect(page).to have_selector '[data-hook=order-bill-address] .adr', text: '1 User Lane' - expect(page).to have_selector '[data-hook=order-bill-address] .adr .local .locality', text: 'Adamsville' - expect(page).to have_selector '[data-hook=order-bill-address] .adr .local .postal-code', text: '35005' - expect(page).to have_selector '[data-hook=order-bill-address] .tel', text: '555-123-4567' + if Spree.version.to_f < 4.1 + wait_for { page.has_text?('555-123-4567') } + within('#order_summary') do + expect(page).to have_selector '[data-hook=order-bill-address] .fn', text: 'Test User' + expect(page).to have_selector '[data-hook=order-bill-address] .adr', text: '1 User Lane' + expect(page).to have_selector '[data-hook=order-bill-address] .adr .local .locality', text: 'Adamsville' + expect(page).to have_selector '[data-hook=order-bill-address] .adr .local .postal-code', text: '35005' + expect(page).to have_selector '[data-hook=order-bill-address] .tel', text: '555-123-4567' + end + else + wait_for { page.has_text?('Order Summary') } + expect(page).to have_content('Order placed successfully') end end end - it 'includes adjustments in PayPal summary' do - add_to_cart(product) - # TODO: Is there a better way to find this current order? + xit 'includes adjustments in PayPal summary' do + visit spree.root_path + click_link product.name + click_button 'Add To Cart' + order = Spree::Order.last Spree::Adjustment.create!(label: '$5 off', adjustable: order, order: order, amount: -5) Spree::Adjustment.create!(label: '$10 on', adjustable: order, order: order, amount: 10) @@ -178,6 +196,7 @@ def expect_successfully_processed_order end end + context 'line item adjustments' do let(:promotion) { Spree::Promotion.create(name: '10% off') } before do @@ -187,19 +206,8 @@ def expect_successfully_processed_order end it 'includes line item adjustments in PayPal summary' do - add_to_cart(product) - # TODO: Is there a better way to find this current order? - order = Spree::Order.last - expect(order.line_item_adjustments.count).to eq 1 - - visit '/cart' - within('#cart_adjustments') do - expect(page).to have_content('10% off') - end + proceed_through_address_stage - click_button 'Checkout' - fill_in_guest - fill_in_billing click_button 'Save and Continue' click_button 'Save and Continue' @@ -209,9 +217,15 @@ def expect_successfully_processed_order login_to_paypal stay_logged_in_for_faster_checkout click_pay_now_button - wait_for { page.has_css?('strong', text: '10% off') } - within('#price-adjustments') do - expect(page).to have_content('10% off') + + if Spree.version.to_f < 4.1 + wait_for { page.has_css?('strong', text: '10% off') } + within('#price-adjustments') do + expect(page).to have_content('10% off') + end + else + wait_for { page.has_text?('Order Summary') } + expect(page).to have_content('PROMOTION') end end end @@ -228,14 +242,14 @@ def expect_successfully_processed_order order_id = script_content.strip.split('\"')[1] order = Spree::Order.find_by(number: order_id) order.line_items.last.update_attribute(:price, 0) - click_button 'Checkout' - fill_in_guest - fill_in_billing + + proceed_through_address_stage + click_button 'Save and Continue' click_button 'Save and Continue' click_paypal_button - + within_transaction_cart('.cartContainer', ['iPad'], ['iPod']) wait_for { page.has_css?('.transactionDetails') } switch_to_paypal_login @@ -243,11 +257,9 @@ def expect_successfully_processed_order stay_logged_in_for_faster_checkout within_transaction_cart('.transctionCartDetails', ['iPad'], ['iPod']) click_pay_now_button - wait_for { page.has_text?('iPad') } - within('#line-items') do - expect(page).to have_content('iPad') - expect(page).to have_content('iPod') - end + wait_for { page.has_text?('Order Summary') } + expect(page).to have_content('iPad') + expect(page).to have_content('iPod') end end @@ -289,10 +301,8 @@ def expect_successfully_processed_order end specify do - add_to_cart(product) - click_button 'Checkout' - fill_in 'Customer E-Mail', with: 'test@example.com' - fill_in_billing + proceed_through_address_stage + click_button 'Save and Continue' click_button 'Save and Continue' @@ -314,7 +324,7 @@ def expect_successfully_processed_order end end - context 'as an admin' do +context 'as an admin' do context 'refunding payments' do before do stub_authorization! diff --git a/spec/spec_helper.rb b/spec/spec_helper.rb index ef1d9144..39930669 100644 --- a/spec/spec_helper.rb +++ b/spec/spec_helper.rb @@ -1,102 +1,11 @@ -# Run Coverage report -require 'simplecov' -SimpleCov.start do - add_filter 'spec/dummy' - add_group 'Controllers', 'app/controllers' - add_group 'Helpers', 'app/helpers' - add_group 'Mailers', 'app/mailers' - add_group 'Models', 'app/models' - add_group 'Views', 'app/views' - add_group 'Libraries', 'lib' -end - -# CodeClimate coverage -require "simplecov" -SimpleCov.start - # Configure Rails Environment ENV['RAILS_ENV'] = 'test' require File.expand_path('../dummy/config/environment.rb', __FILE__) -require 'rspec/rails' +require 'spree_dev_tools/rspec/spec_helper' require 'rspec/active_model/mocks' -require 'database_cleaner' -require 'ffaker' # Requires supporting ruby files with custom matchers and macros, etc, # in spec/support/ and its subdirectories. -Dir[File.join(File.dirname(__FILE__), 'support/**/*.rb')].each { |f| require f } - -# Requires factories and other useful helpers defined in spree_core. -require 'spree/testing_support/authorization_helpers' -require 'spree/testing_support/capybara_ext' -require 'spree/testing_support/controller_requests' -require 'spree/testing_support/factories' -require 'spree/testing_support/url_helpers' -require 'spree/testing_support/order_walkthrough' - -require 'pry' -require 'webdrivers' - -RSpec.configure do |config| - config.include FactoryBot::Syntax::Methods - - # Infer an example group's spec type from the file location. - config.infer_spec_type_from_file_location! - - # == URL Helpers - # - # Allows access to Spree's routes in specs: - # - # visit spree.admin_path - # current_path.should eql(spree.products_path) - config.include Spree::TestingSupport::UrlHelpers - config.include Spree::TestingSupport::AuthorizationHelpers::Controller - - # == Requests support - # - # Adds convenient methods to request Spree's controllers - # spree_get :index - config.include Spree::TestingSupport::ControllerRequests, type: :controller - - # == Mock Framework - # - # If you prefer to use mocha, flexmock or RR, uncomment the appropriate line: - # - # config.mock_with :mocha - # config.mock_with :flexmock - # config.mock_with :rr - config.mock_with :rspec do |mocks| - mocks.syntax = [:expect, :should] - end - config.color = true - - # Remove this line if you're not using ActiveRecord or ActiveRecord fixtures - config.fixture_path = "#{::Rails.root}/spec/fixtures" - - # Capybara javascript drivers require transactional fixtures set to false, and we use DatabaseCleaner - # to cleanup after each test instead. Without transactional fixtures set to false the records created - # to setup a test will be unavailable to the browser, which runs under a separate server instance. - config.use_transactional_fixtures = false - - # Ensure Suite is set to use transactions for speed. - config.before :suite do - DatabaseCleaner.strategy = :transaction - DatabaseCleaner.clean_with :truncation - end - - # Before each spec check if it is a Javascript test and switch between using database transactions or not where necessary. - config.before :each do - DatabaseCleaner.strategy = RSpec.current_example.metadata[:js] ? :truncation : :transaction - DatabaseCleaner.start - end - - # After each spec clean the database. - config.after :each do - DatabaseCleaner.clean - end - - config.fail_fast = ENV['FAIL_FAST'] || false - config.order = 'random' -end +Dir[File.join(File.dirname(__FILE__), 'support/**/*.rb')].sort.each { |f| require f } diff --git a/spec/support/capybara.rb b/spec/support/capybara.rb deleted file mode 100644 index 663c10c0..00000000 --- a/spec/support/capybara.rb +++ /dev/null @@ -1,17 +0,0 @@ -require 'capybara' -require 'capybara/rspec' -require 'capybara/rails' -require 'selenium-webdriver' -require 'webdrivers' - -RSpec.configure do |config| - Capybara.register_driver :chrome do |app| - Selenium::WebDriver.logger.level = :error - - Capybara::Selenium::Driver.new app, - browser: :chrome, - options: Selenium::WebDriver::Chrome::Options.new(args: %w[disable-popup-blocking headless disable-gpu window-size=1920,1080]) - end - - Capybara.javascript_driver = :chrome -end diff --git a/spec/support/capybara_ext.rb b/spec/support/capybara_ext.rb deleted file mode 100644 index c682fb5d..00000000 --- a/spec/support/capybara_ext.rb +++ /dev/null @@ -1,13 +0,0 @@ -module CapybaraExt - def wait_for(options = {}) - default_options = { error: nil, seconds: 5 }.merge(options) - - Selenium::WebDriver::Wait.new(timeout: default_options[:seconds]).until { yield } - rescue Selenium::WebDriver::Error::TimeOutError - default_options[:error].nil? ? false : raise(default_options[:error]) - end - - RSpec.configure do |c| - c.include CapybaraExt - end -end \ No newline at end of file diff --git a/spree_paypal_express.gemspec b/spree_paypal_express.gemspec index 71d36675..cb7417eb 100644 --- a/spree_paypal_express.gemspec +++ b/spree_paypal_express.gemspec @@ -1,5 +1,4 @@ -# coding: utf-8 -lib = File.expand_path('../lib/', __FILE__) +lib = File.expand_path('lib', __dir__) $:.unshift lib unless $:.include?(lib) require 'spree_paypal_express/version' @@ -15,7 +14,7 @@ Gem::Specification.new do |s| s.author = 'Spree Commerce' s.email = 'gems@spreecommerce.com' s.homepage = 'http://www.spreecommerce.com' - s.license = %q{BSD-3} + s.license = 'BSD-3' s.files = `git ls-files`.split("\n") s.test_files = `git ls-files -- spec/*`.split("\n") @@ -27,23 +26,6 @@ Gem::Specification.new do |s| s.add_dependency 'paypal-sdk-merchant' s.add_dependency 'spree_extension' - s.add_development_dependency 'capybara' - s.add_development_dependency 'capybara-screenshot' - s.add_development_dependency 'selenium-webdriver' - s.add_development_dependency 'webdrivers', '~> 4.1' - s.add_development_dependency 'coffee-rails' - s.add_development_dependency 'database_cleaner' - s.add_development_dependency 'factory_bot', '~> 4.7' - s.add_development_dependency 'ffaker' - s.add_development_dependency 'rspec-rails', '~> 4.0.0.beta2' s.add_development_dependency 'rspec-activemodel-mocks' - s.add_development_dependency 'sass-rails' - s.add_development_dependency 'simplecov' - s.add_development_dependency 'pry' - s.add_development_dependency 'require_all' - s.add_development_dependency 'pg' - s.add_development_dependency 'puma' - s.add_development_dependency 'mysql2' - s.add_development_dependency 'sqlite3' - s.add_development_dependency 'appraisal' + s.add_development_dependency 'spree_dev_tools' end