From a0011bd2e90f822f682edf268e57efe6c22c4135 Mon Sep 17 00:00:00 2001 From: Maikel Linke Date: Mon, 12 Jan 2026 16:11:08 +1100 Subject: [PATCH 1/9] Remove orphaned translations of payment methods --- config/locales/en.yml | 2 -- 1 file changed, 2 deletions(-) diff --git a/config/locales/en.yml b/config/locales/en.yml index 3cec061b65..81a1c2bf5b 100644 --- a/config/locales/en.yml +++ b/config/locales/en.yml @@ -4664,9 +4664,7 @@ en: providers: provider: "Provider" check: "Cash/EFT/etc. (payments for which automatic validation is not required)" - pin: "Pin Payments" paypalexpress: "PayPal Express" - stripeconnect: "Stripe" stripesca: "Stripe SCA" payments: source_forms: From b3a1d1269a92874321cb830aadffc434f02342f2 Mon Sep 17 00:00:00 2001 From: Maikel Linke Date: Mon, 12 Jan 2026 17:12:11 +1100 Subject: [PATCH 2/9] Add Taler as payment method It doesn't take payments yet but can be selected during checkout. --- .../spree/admin/payment_methods_controller.rb | 1 + app/models/spree/payment_method/taler.rb | 39 +++++++++++++++++++ .../permitted_attributes/payment_method.rb | 2 +- config/locales/en.yml | 5 +++ .../admin/payment_methods_controller_spec.rb | 1 + 5 files changed, 47 insertions(+), 1 deletion(-) create mode 100644 app/models/spree/payment_method/taler.rb diff --git a/app/controllers/spree/admin/payment_methods_controller.rb b/app/controllers/spree/admin/payment_methods_controller.rb index 6584dd87ce..8073fb544a 100644 --- a/app/controllers/spree/admin/payment_methods_controller.rb +++ b/app/controllers/spree/admin/payment_methods_controller.rb @@ -14,6 +14,7 @@ module Spree Spree::Gateway::PayPalExpress Spree::Gateway::StripeSCA Spree::PaymentMethod::Check + Spree::PaymentMethod::Taler }.freeze def create diff --git a/app/models/spree/payment_method/taler.rb b/app/models/spree/payment_method/taler.rb new file mode 100644 index 0000000000..2a4e496d7c --- /dev/null +++ b/app/models/spree/payment_method/taler.rb @@ -0,0 +1,39 @@ +# frozen_string_literal: true + +module Spree + class PaymentMethod + # GNU Taler is a distributed, open source payment system. + # You need a hosted Taler backend server to process payments. + # + # For testing, you can use the official demo backend: + # + # - Merchant UX: https://backend.demo.taler.net + # - Username: sandbox + # - Password: sandbox + # + # Configure this payment method for testing with: + # + # - backend_url: https://backend.demo.taler.net/instances/sandbox + # - api_key: sandbox + class Taler < PaymentMethod + preference :backend_url, :string + preference :api_key, :password + + # Name of the view to display during checkout + def method_type + "check" # empty view + end + + def external_gateway? + true + end + + # The backend provides this URL. It can look like this: + # https://backend.demo.taler.net/instances/blog/orders/2026..?token=S8Y..&session_id=b0b.. + def external_payment_url(options) + # order = options.fetch(:order) + # Taler.create_order(backend_url, api_key, order.total, "OFN Order", "https://ofn.example.net") + end + end + end +end diff --git a/app/services/permitted_attributes/payment_method.rb b/app/services/permitted_attributes/payment_method.rb index aac274a9f2..2ee863e3b7 100644 --- a/app/services/permitted_attributes/payment_method.rb +++ b/app/services/permitted_attributes/payment_method.rb @@ -11,7 +11,7 @@ module PermittedAttributes [:name, :description, :type, :active, :environment, :display_on, :tag_list, :preferred_enterprise_id, :preferred_server, :preferred_login, :preferred_password, - :calculator_type, :preferred_api_key, + :calculator_type, :preferred_api_key, :preferred_backend_url, :preferred_signature, :preferred_solution, :preferred_landing_page, :preferred_logourl, :preferred_test_mode, :calculator_type, { distributor_ids: [] }, { calculator_attributes: PermittedAttributes::Calculator.attributes }] diff --git a/config/locales/en.yml b/config/locales/en.yml index 81a1c2bf5b..254b8b811b 100644 --- a/config/locales/en.yml +++ b/config/locales/en.yml @@ -4344,6 +4344,10 @@ en: thumbnail: "Thumbnail" back_to_images_list: "Back To Images List" + # Attributes of the Taler payment gateway + backend_url: "Backend URL" + api_key: "API key" + # TODO: remove `email` key once we get to Spree 2.0 email: Email # TODO: remove 'account_updated' key once we get to Spree 2.0 @@ -4666,6 +4670,7 @@ en: check: "Cash/EFT/etc. (payments for which automatic validation is not required)" paypalexpress: "PayPal Express" stripesca: "Stripe SCA" + taler: "Taler" payments: source_forms: stripe: diff --git a/spec/controllers/spree/admin/payment_methods_controller_spec.rb b/spec/controllers/spree/admin/payment_methods_controller_spec.rb index 311b83b7fa..8937bff352 100644 --- a/spec/controllers/spree/admin/payment_methods_controller_spec.rb +++ b/spec/controllers/spree/admin/payment_methods_controller_spec.rb @@ -19,6 +19,7 @@ RSpec.describe Spree::Admin::PaymentMethodsController do expect(providers).to eq %w[ Spree::Gateway::PayPalExpress Spree::PaymentMethod::Check + Spree::PaymentMethod::Taler ] end From 5971cdc6e2acd0dc48f1689ce4e76d592d97e519 Mon Sep 17 00:00:00 2001 From: Maikel Linke Date: Thu, 15 Jan 2026 15:20:46 +1100 Subject: [PATCH 3/9] Add new taler gem --- Gemfile | 1 + Gemfile.lock | 2 ++ 2 files changed, 3 insertions(+) diff --git a/Gemfile b/Gemfile index 44c1d371f1..7f663812c2 100644 --- a/Gemfile +++ b/Gemfile @@ -58,6 +58,7 @@ gem 'stringex', '~> 2.8.5', require: false gem 'paypal-sdk-merchant', '1.117.2' gem 'stripe', '~> 15' +gem "taler" gem 'devise' gem 'devise-encryptable' diff --git a/Gemfile.lock b/Gemfile.lock index 500251293e..316a74c595 100644 --- a/Gemfile.lock +++ b/Gemfile.lock @@ -884,6 +884,7 @@ GEM faraday (~> 2.0) faraday-follow_redirects sysexits (1.2.0) + taler (0.1.0) temple (0.10.4) terminal-table (4.0.0) unicode-display_width (>= 1.1.1, < 4) @@ -1100,6 +1101,7 @@ DEPENDENCIES stimulus_reflex_testing! stringex (~> 2.8.5) stripe (~> 15) + taler turbo-rails turbo_power undercover From 61e068839287897c7407d7536fe4c71d4d2e0b32 Mon Sep 17 00:00:00 2001 From: Maikel Linke Date: Thu, 15 Jan 2026 15:56:23 +1100 Subject: [PATCH 4/9] Demonstrate retrieving the payment Taler URL --- app/models/spree/payment_method/taler.rb | 14 ++- .../retrieves_a_URL_to_pay_at.yml | 106 ++++++++++++++++++ .../models/spree/payment_method/taler_spec.rb | 19 ++++ 3 files changed, 137 insertions(+), 2 deletions(-) create mode 100644 spec/fixtures/vcr_cassettes/Spree_PaymentMethod_Taler/external_payment_url/retrieves_a_URL_to_pay_at.yml create mode 100644 spec/models/spree/payment_method/taler_spec.rb diff --git a/app/models/spree/payment_method/taler.rb b/app/models/spree/payment_method/taler.rb index 2a4e496d7c..fed3239d7f 100644 --- a/app/models/spree/payment_method/taler.rb +++ b/app/models/spree/payment_method/taler.rb @@ -31,8 +31,18 @@ module Spree # The backend provides this URL. It can look like this: # https://backend.demo.taler.net/instances/blog/orders/2026..?token=S8Y..&session_id=b0b.. def external_payment_url(options) - # order = options.fetch(:order) - # Taler.create_order(backend_url, api_key, order.total, "OFN Order", "https://ofn.example.net") + order = options.fetch(:order) + total_amount = order&.total || 5 + taler_amount = "KUDOS:#{total_amount}" + new_order = client.create_order(taler_amount, "OFN Order", "https://ofn.example.net") + order = client.fetch_order(new_order["order_id"]) + order["order_status_url"] + end + + private + + def client + @client ||= ::Taler::Client.new(preferred_backend_url, preferred_api_key) end end end diff --git a/spec/fixtures/vcr_cassettes/Spree_PaymentMethod_Taler/external_payment_url/retrieves_a_URL_to_pay_at.yml b/spec/fixtures/vcr_cassettes/Spree_PaymentMethod_Taler/external_payment_url/retrieves_a_URL_to_pay_at.yml new file mode 100644 index 0000000000..89a829b59d --- /dev/null +++ b/spec/fixtures/vcr_cassettes/Spree_PaymentMethod_Taler/external_payment_url/retrieves_a_URL_to_pay_at.yml @@ -0,0 +1,106 @@ +--- +http_interactions: +- request: + method: post + uri: https://backend.demo.taler.net/instances/sandbox/private/orders + body: + encoding: UTF-8 + string: '{"order":{"amount":"KUDOS:5","summary":"OFN Order","fulfillment_url":"https://ofn.example.net"},"create_token":false}' + headers: + Authorization: + - "" + Accept: + - application/json + User-Agent: + - Taler Ruby + Content-Type: + - application/json + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + response: + status: + code: 200 + message: OK + headers: + Server: + - nginx/1.26.3 + Date: + - Thu, 15 Jan 2026 04:18:27 GMT + Content-Type: + - application/json + Content-Length: + - '42' + Connection: + - keep-alive + Access-Control-Allow-Origin: + - "*" + Access-Control-Expose-Headers: + - "*" + Cache-Control: + - no-store + Via: + - 1.1 Caddy + Strict-Transport-Security: + - max-age=63072000; includeSubDomains; preload + body: + encoding: UTF-8 + string: |- + { + "order_id": "2026.015-02T676V4VARMT" + } + recorded_at: Thu, 15 Jan 2026 04:18:28 GMT +- request: + method: get + uri: https://backend.demo.taler.net/instances/sandbox/private/orders/2026.015-02T676V4VARMT + body: + encoding: US-ASCII + string: '' + headers: + Authorization: + - "" + Accept: + - application/json + User-Agent: + - Taler Ruby + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + response: + status: + code: 200 + message: OK + headers: + Server: + - nginx/1.26.3 + Date: + - Thu, 15 Jan 2026 04:18:29 GMT + Content-Type: + - application/json + Content-Length: + - '336' + Connection: + - keep-alive + Access-Control-Allow-Origin: + - "*" + Access-Control-Expose-Headers: + - "*" + Cache-Control: + - no-store + Via: + - 1.1 Caddy + Strict-Transport-Security: + - max-age=63072000; includeSubDomains; preload + body: + encoding: ASCII-8BIT + string: |- + { + "taler_pay_uri": "taler://pay/backend.demo.taler.net/instances/sandbox/2026.015-02T676V4VARMT/", + "order_status_url": "https://backend.demo.taler.net/instances/sandbox/orders/2026.015-02T676V4VARMT", + "order_status": "unpaid", + "total_amount": "KUDOS:5", + "summary": "OFN Order", + "creation_time": { + "t_s": 1768450707 + } + } + recorded_at: Thu, 15 Jan 2026 04:18:29 GMT +recorded_with: VCR 6.3.1 diff --git a/spec/models/spree/payment_method/taler_spec.rb b/spec/models/spree/payment_method/taler_spec.rb new file mode 100644 index 0000000000..c30d7064f4 --- /dev/null +++ b/spec/models/spree/payment_method/taler_spec.rb @@ -0,0 +1,19 @@ +# frozen_string_literal: true + +require 'spec_helper' + +RSpec.describe Spree::PaymentMethod::Taler do + subject(:taler) { + Spree::PaymentMethod::Taler.new( + preferred_backend_url: "https://backend.demo.taler.net/instances/sandbox", + preferred_api_key: "sandbox", + ) + } + + describe "external_payment_url", vcr: true do + it "retrieves a URL to pay at" do + url = subject.external_payment_url(order: nil) + expect(url).to match %r{\Ahttps://backend.demo.taler.net/instances/sandb} + end + end +end From c11b93a4dcd22ffbc1c61223cd75023823e30f6f Mon Sep 17 00:00:00 2001 From: Maikel Linke Date: Mon, 19 Jan 2026 14:28:00 +1100 Subject: [PATCH 5/9] Demo Taler flow without validating payment yet --- .../payment_gateways/taler_controller.rb | 16 +++++++ app/models/spree/payment_method/taler.rb | 43 ++++++++++++++++--- config/locales/en.yml | 2 + config/routes.rb | 2 + .../models/spree/payment_method/taler_spec.rb | 9 +++- spec/requests/payment_gateways/taler_spec.rb | 27 ++++++++++++ 6 files changed, 92 insertions(+), 7 deletions(-) create mode 100644 app/controllers/payment_gateways/taler_controller.rb create mode 100644 spec/requests/payment_gateways/taler_spec.rb diff --git a/app/controllers/payment_gateways/taler_controller.rb b/app/controllers/payment_gateways/taler_controller.rb new file mode 100644 index 0000000000..aff2799242 --- /dev/null +++ b/app/controllers/payment_gateways/taler_controller.rb @@ -0,0 +1,16 @@ +# frozen_string_literal: true + +module PaymentGateways + class TalerController < BaseController + include OrderCompletion + + # The Taler merchant backend has taken the payment. + # Now we just need to confirm that and update our local database + # before finalising the order. + def confirm + payment = Spree::Payment.find(params[:payment_id]) + @order = payment.order + process_payment_completion! + end + end +end diff --git a/app/models/spree/payment_method/taler.rb b/app/models/spree/payment_method/taler.rb index fed3239d7f..41d0881a43 100644 --- a/app/models/spree/payment_method/taler.rb +++ b/app/models/spree/payment_method/taler.rb @@ -1,5 +1,7 @@ # frozen_string_literal: true +require "taler" + module Spree class PaymentMethod # GNU Taler is a distributed, open source payment system. @@ -32,15 +34,46 @@ module Spree # https://backend.demo.taler.net/instances/blog/orders/2026..?token=S8Y..&session_id=b0b.. def external_payment_url(options) order = options.fetch(:order) - total_amount = order&.total || 5 - taler_amount = "KUDOS:#{total_amount}" - new_order = client.create_order(taler_amount, "OFN Order", "https://ofn.example.net") - order = client.fetch_order(new_order["order_id"]) - order["order_status_url"] + payment = load_payment(order) + + payment.source ||= self + payment.response_code ||= create_taler_order(payment) + payment.redirect_auth_url ||= fetch_order_url(payment) + payment.save! if payment.changed? + + payment.redirect_auth_url + end + + def purchase(_money, _creditcard, _gateway_options) + # TODO: implement + ActiveMerchant::Billing::Response.new(true, "test") end private + def load_payment(order) + order.payments.checkout.where(payment_method: self).last + end + + def create_taler_order(payment) + # We are ignoring currency for now so that we can test with the + # current demo backend only working with the KUDOS currency. + taler_amount = "KUDOS:#{payment.amount}" + urls = Rails.application.routes.url_helpers + new_order = client.create_order( + taler_amount, + I18n.t("payment_method_taler.order_summary"), + urls.payment_gateways_confirm_taler_url(payment_id: payment.id), + ) + + new_order["order_id"] + end + + def fetch_order_url(payment) + order = client.fetch_order(payment.response_code) + order["order_status_url"] + end + def client @client ||= ::Taler::Client.new(preferred_backend_url, preferred_api_key) end diff --git a/config/locales/en.yml b/config/locales/en.yml index 254b8b811b..1d081df210 100644 --- a/config/locales/en.yml +++ b/config/locales/en.yml @@ -3502,6 +3502,8 @@ en: payment_processing_failed: "Payment could not be processed, please check the details you entered" payment_method_not_supported: "That payment method is unsupported. Please choose another one." payment_updated: "Payment Updated" + payment_method_taler: + order_summary: "Open Food Network order" cannot_perform_operation: "Could not update the payment" action_required: "Action required" tag_rules: "Tag Rules" diff --git a/config/routes.rb b/config/routes.rb index 0946d21ab7..abc399163b 100644 --- a/config/routes.rb +++ b/config/routes.rb @@ -82,6 +82,8 @@ Openfoodnetwork::Application.routes.draw do get "/stripe/confirm", to: "stripe#confirm", as: :confirm_stripe get "/stripe/authorize/:order_number", to: "stripe#authorize", as: :authorize_stripe + + get "/taler/:payment_id", to: "taler#confirm", as: :confirm_taler end get '/checkout', to: 'checkout#edit' diff --git a/spec/models/spree/payment_method/taler_spec.rb b/spec/models/spree/payment_method/taler_spec.rb index c30d7064f4..25c619b5fc 100644 --- a/spec/models/spree/payment_method/taler_spec.rb +++ b/spec/models/spree/payment_method/taler_spec.rb @@ -11,9 +11,14 @@ RSpec.describe Spree::PaymentMethod::Taler do } describe "external_payment_url", vcr: true do - it "retrieves a URL to pay at" do - url = subject.external_payment_url(order: nil) + it "retrieves a URL to pay at and stores it on the payment record" do + order = create(:order_ready_for_confirmation, payment_method: taler) + url = subject.external_payment_url(order:) expect(url).to match %r{\Ahttps://backend.demo.taler.net/instances/sandb} + + payment = order.payments.last.reload + expect(payment.response_code).to match "2026.022-0284X4GE8WKMJ" + expect(payment.redirect_auth_url).to eq url end end end diff --git a/spec/requests/payment_gateways/taler_spec.rb b/spec/requests/payment_gateways/taler_spec.rb new file mode 100644 index 0000000000..ed9969e583 --- /dev/null +++ b/spec/requests/payment_gateways/taler_spec.rb @@ -0,0 +1,27 @@ +# frozen_string_literal: true + +require 'spec_helper' + +RSpec.describe "/payment_gateways/taler/:id" do + it "completes the order" do + shop = create(:distributor_enterprise) + taler = Spree::PaymentMethod::Taler.create!( + name: "Taler", + environment: "test", + distributors: [shop], + ) + order = create(:order_ready_for_confirmation, payment_method: taler) + payment = Spree::Payment.last + payment.update!( + payment_method: taler, + response_code: "taler-order-id:12345", + redirect_auth_url: "https://merchant.backend.where-we-paid.com", + ) + + get payment_gateways_confirm_taler_path(payment_id: payment.id) + expect(response).to redirect_to(order_path(order, order_token: order.token)) + + payment.reload + expect(payment.state).to eq "completed" + end +end From b9c7925008bca27bd82a13474c860b6a6ae43954 Mon Sep 17 00:00:00 2001 From: Maikel Linke Date: Thu, 22 Jan 2026 16:01:57 +1100 Subject: [PATCH 6/9] Complete Taler payment success scenario --- app/models/spree/payment/processing.rb | 1 + app/models/spree/payment_method/taler.rb | 17 +- ...at_and_stores_it_on_the_payment_record.yml | 106 +++++++++ .../taler/_id/completes_the_order.yml | 209 ++++++++++++++++++ spec/requests/payment_gateways/taler_spec.rb | 10 +- spec/system/consumer/checkout/payment_spec.rb | 29 +++ 6 files changed, 367 insertions(+), 5 deletions(-) create mode 100644 spec/fixtures/vcr_cassettes/Spree_PaymentMethod_Taler/external_payment_url/retrieves_a_URL_to_pay_at_and_stores_it_on_the_payment_record.yml create mode 100644 spec/fixtures/vcr_cassettes/payment_gateways/taler/_id/completes_the_order.yml diff --git a/app/models/spree/payment/processing.rb b/app/models/spree/payment/processing.rb index 57b4f4335a..599a562097 100644 --- a/app/models/spree/payment/processing.rb +++ b/app/models/spree/payment/processing.rb @@ -183,6 +183,7 @@ module Spree options.merge!({ billing_address: order.bill_address.try(:active_merchant_hash), shipping_address: order.ship_address.try(:active_merchant_hash) }) + options.merge!(payment: self) options end diff --git a/app/models/spree/payment_method/taler.rb b/app/models/spree/payment_method/taler.rb index 41d0881a43..273b11ee9d 100644 --- a/app/models/spree/payment_method/taler.rb +++ b/app/models/spree/payment_method/taler.rb @@ -44,9 +44,20 @@ module Spree payment.redirect_auth_url end - def purchase(_money, _creditcard, _gateway_options) - # TODO: implement - ActiveMerchant::Billing::Response.new(true, "test") + # Main method called by Spree::Payment::Processing during checkout + # when the user is redirected back to the app. + # + # The payment has already been made and we need to verify the success. + def purchase(_money, _source, gateway_options) + payment = gateway_options[:payment] + + return unless payment.response_code + + taler_order = client.fetch_order(payment.response_code) + status = taler_order["order_status"] + success = (status == "paid") + + ActiveMerchant::Billing::Response.new(success, status) end private diff --git a/spec/fixtures/vcr_cassettes/Spree_PaymentMethod_Taler/external_payment_url/retrieves_a_URL_to_pay_at_and_stores_it_on_the_payment_record.yml b/spec/fixtures/vcr_cassettes/Spree_PaymentMethod_Taler/external_payment_url/retrieves_a_URL_to_pay_at_and_stores_it_on_the_payment_record.yml new file mode 100644 index 0000000000..5c658e01f3 --- /dev/null +++ b/spec/fixtures/vcr_cassettes/Spree_PaymentMethod_Taler/external_payment_url/retrieves_a_URL_to_pay_at_and_stores_it_on_the_payment_record.yml @@ -0,0 +1,106 @@ +--- +http_interactions: +- request: + method: post + uri: https://backend.demo.taler.net/instances/sandbox/private/orders + body: + encoding: UTF-8 + string: '{"order":{"amount":"KUDOS:10.0","summary":"Open Food Network order","fulfillment_url":"http://test.host/payment_gateways/taler/61"},"create_token":false}' + headers: + Authorization: + - "" + Accept: + - application/json + User-Agent: + - Taler Ruby + Content-Type: + - application/json + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + response: + status: + code: 200 + message: OK + headers: + Server: + - nginx/1.26.3 + Date: + - Thu, 22 Jan 2026 04:43:32 GMT + Content-Type: + - application/json + Content-Length: + - '42' + Connection: + - keep-alive + Access-Control-Allow-Origin: + - "*" + Access-Control-Expose-Headers: + - "*" + Cache-Control: + - no-store + Via: + - 1.1 Caddy + Strict-Transport-Security: + - max-age=63072000; includeSubDomains; preload + body: + encoding: UTF-8 + string: |- + { + "order_id": "2026.022-0284X4GE8WKMJ" + } + recorded_at: Thu, 22 Jan 2026 04:43:33 GMT +- request: + method: get + uri: https://backend.demo.taler.net/instances/sandbox/private/orders/2026.022-0284X4GE8WKMJ + body: + encoding: US-ASCII + string: '' + headers: + Authorization: + - "" + Accept: + - application/json + User-Agent: + - Taler Ruby + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + response: + status: + code: 200 + message: OK + headers: + Server: + - nginx/1.26.3 + Date: + - Thu, 22 Jan 2026 04:43:34 GMT + Content-Type: + - application/json + Content-Length: + - '351' + Connection: + - keep-alive + Access-Control-Allow-Origin: + - "*" + Access-Control-Expose-Headers: + - "*" + Cache-Control: + - no-store + Via: + - 1.1 Caddy + Strict-Transport-Security: + - max-age=63072000; includeSubDomains; preload + body: + encoding: ASCII-8BIT + string: |- + { + "taler_pay_uri": "taler://pay/backend.demo.taler.net/instances/sandbox/2026.022-0284X4GE8WKMJ/", + "order_status_url": "https://backend.demo.taler.net/instances/sandbox/orders/2026.022-0284X4GE8WKMJ", + "order_status": "unpaid", + "total_amount": "KUDOS:10", + "summary": "Open Food Network order", + "creation_time": { + "t_s": 1769057012 + } + } + recorded_at: Thu, 22 Jan 2026 04:43:34 GMT +recorded_with: VCR 6.3.1 diff --git a/spec/fixtures/vcr_cassettes/payment_gateways/taler/_id/completes_the_order.yml b/spec/fixtures/vcr_cassettes/payment_gateways/taler/_id/completes_the_order.yml new file mode 100644 index 0000000000..d84ef9f88a --- /dev/null +++ b/spec/fixtures/vcr_cassettes/payment_gateways/taler/_id/completes_the_order.yml @@ -0,0 +1,209 @@ +--- +http_interactions: +- request: + method: get + uri: https://backend.demo.taler.net/instances/sandbox/private/orders/taler-order-id:12345 + body: + encoding: US-ASCII + string: '' + headers: + Authorization: + - "" + Accept: + - application/json + User-Agent: + - Taler Ruby + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + response: + status: + code: 404 + message: Not Found + headers: + Server: + - nginx/1.26.3 + Date: + - Sat, 24 Jan 2026 00:51:32 GMT + Content-Type: + - application/json + Content-Length: + - '109' + Connection: + - keep-alive + Access-Control-Allow-Origin: + - "*" + Access-Control-Expose-Headers: + - "*" + Cache-Control: + - no-store + Via: + - 1.1 Caddy + body: + encoding: ASCII-8BIT + string: |- + { + "code": 2005, + "hint": "The proposal is not known to the backend.", + "detail": "taler-order-id:12345" + } + recorded_at: Sat, 24 Jan 2026 00:51:31 GMT +- request: + method: get + uri: https://backend.demo.taler.net/instances/sandbox/private/orders/2026.020-03R3ETNZZ0DVA + body: + encoding: US-ASCII + string: '' + headers: + Authorization: + - "" + Accept: + - application/json + User-Agent: + - Taler Ruby + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + response: + status: + code: 200 + message: OK + headers: + Server: + - nginx/1.26.3 + Date: + - Sat, 24 Jan 2026 00:55:33 GMT + Content-Type: + - application/json + Content-Length: + - '19676' + Connection: + - keep-alive + Access-Control-Allow-Origin: + - "*" + Access-Control-Expose-Headers: + - "*" + Cache-Control: + - no-store + Via: + - 1.1 Caddy + Strict-Transport-Security: + - max-age=63072000; includeSubDomains; preload + body: + encoding: ASCII-8BIT + string: |- + { + "wire_reports": [], + "exchange_code": 0, + "exchange_http_status": 0, + "exchange_ec": 0, + "exchange_hc": 0, + "deposit_total": "KUDOS:5.5", + "contract_terms": { + "nonce": "H8BTDASTRX0WMBQTEWN6JSVS3KK7MZ6WAZGTKDM28FQ0CE9W59EG", + "amount": "KUDOS:5.5", + "h_wire": "1HZYJW67Y7D9GDERGJ8R2B0MTYXRW425M0N11AWHY4VC3DAYCR14CG30X7B7NGM1G93YYC7V4FAS314NBF40GRF1EC8GB0FD5SSJET0", + "max_fee": "KUDOS:0", + "summary": "Open Food Network order", + "version": 0, + "merchant": { + "logo": "", + "name": "sandbox merchant", + "address": {}, + "jurisdiction": {} + }, + "order_id": "2026.020-03R3ETNZZ0DVA", + "products": [], + "exchanges": [ + { + "url": "https://exchange.demo.taler.net/", + "priority": 1024, + "master_pub": "F80MFRG8HVH6R9CQ47KRFQSJP3T6DBJ4K1D9B703RJY3Z39TBMJ0", + "max_contribution": "KUDOS:5.5" + } + ], + "timestamp": { + "t_s": 1768885088 + }, + "minimum_age": 0, + "wire_method": "x-taler-bank", + "merchant_pub": "74DV5N6T2ANY1DJFMD41BN7KJYXK70S18CM22TQABEYMA2GSDRD0", + "pay_deadline": { + "t_s": 1768885388 + }, + "fulfillment_url": "http://localhost:3000/payment_gateways/taler/17", + "refund_deadline": { + "t_s": 0 + }, + "merchant_base_url": "https://backend.demo.taler.net/instances/sandbox/", + "wire_transfer_deadline": { + "t_s": 1768885808 + } + }, + "order_status": "paid", + "last_payment": { + "t_s": 1768885096 + }, + "refunded": false, + "wired": true, + "refund_pending": false, + "refund_amount": "KUDOS:0", + "wire_details": [ + { + "wtid": "4E2DRXCHTBPVY2F72CZMK31X3ZTZTJW2RG4NF5KQA5T3C2AE2KVG", + "exchange_url": "https://exchange.demo.taler.net/", + "amount": "KUDOS:0.38", + "execution_time": { + "t_s": 1768885096 + }, + "confirmed": false + }, + { + "wtid": "4E2DRXCHTBPVY2F72CZMK31X3ZTZTJW2RG4NF5KQA5T3C2AE2KVG", + "exchange_url": "https://exchange.demo.taler.net/", + "amount": "KUDOS:1.28", + "execution_time": { + "t_s": 1768885096 + }, + "confirmed": false + }, + { + "wtid": "4E2DRXCHTBPVY2F72CZMK31X3ZTZTJW2RG4NF5KQA5T3C2AE2KVG", + "exchange_url": "https://exchange.demo.taler.net/", + "amount": "KUDOS:0.64", + "execution_time": { + "t_s": 1768885096 + }, + "confirmed": false + }, + { + "wtid": "4E2DRXCHTBPVY2F72CZMK31X3ZTZTJW2RG4NF5KQA5T3C2AE2KVG", + "exchange_url": "https://exchange.demo.taler.net/", + "amount": "KUDOS:1.28", + "execution_time": { + "t_s": 1768885096 + }, + "confirmed": false + }, + { + "wtid": "4E2DRXCHTBPVY2F72CZMK31X3ZTZTJW2RG4NF5KQA5T3C2AE2KVG", + "exchange_url": "https://exchange.demo.taler.net/", + "amount": "KUDOS:0.64", + "execution_time": { + "t_s": 1768885096 + }, + "confirmed": false + }, + { + "wtid": "4E2DRXCHTBPVY2F72CZMK31X3ZTZTJW2RG4NF5KQA5T3C2AE2KVG", + "exchange_url": "https://exchange.demo.taler.net/", + "amount": "KUDOS:1.28", + "execution_time": { + "t_s": 1768885096 + }, + "confirmed": false + } + ], + "refund_details": [], + "order_status_url": "https://backend.demo.taler.net/instances/sandbox/orders/2026.020-03R3ETNZZ0DVA" + } + recorded_at: Sat, 24 Jan 2026 00:55:32 GMT +recorded_with: VCR 6.3.1 diff --git a/spec/requests/payment_gateways/taler_spec.rb b/spec/requests/payment_gateways/taler_spec.rb index ed9969e583..0ddc88b927 100644 --- a/spec/requests/payment_gateways/taler_spec.rb +++ b/spec/requests/payment_gateways/taler_spec.rb @@ -3,18 +3,24 @@ require 'spec_helper' RSpec.describe "/payment_gateways/taler/:id" do - it "completes the order" do + it "completes the order", :vcr do shop = create(:distributor_enterprise) taler = Spree::PaymentMethod::Taler.create!( name: "Taler", environment: "test", distributors: [shop], + preferred_backend_url: "https://backend.demo.taler.net/instances/sandbox", + preferred_api_key: "sandbox", ) order = create(:order_ready_for_confirmation, payment_method: taler) payment = Spree::Payment.last payment.update!( + source: taler, payment_method: taler, - response_code: "taler-order-id:12345", + # This is a Taler order id of a paid order on the test backend. + # It may be gone when you try to re-record this test. + # To create a new order, you need user interaction with a wallet. + response_code: "2026.020-03R3ETNZZ0DVA", redirect_auth_url: "https://merchant.backend.where-we-paid.com", ) diff --git a/spec/system/consumer/checkout/payment_spec.rb b/spec/system/consumer/checkout/payment_spec.rb index 81c26225bd..eac1eb1de0 100644 --- a/spec/system/consumer/checkout/payment_spec.rb +++ b/spec/system/consumer/checkout/payment_spec.rb @@ -340,6 +340,35 @@ RSpec.describe "As a consumer, I want to checkout my order" do it_behaves_like "different payment methods", "Stripe SCA" end + + context "Taler" do + let!(:taler) do + Spree::PaymentMethod::Taler.create!( + name: "Taler", + environment: "test", + distributors: [distributor] + ) + end + + before do + # Shortcut the user interaction and go straight to our + # confirmation action. + taler_order_id = { "order_id" => "taler-order:123" } + expect_any_instance_of(Taler::Client) + .to receive(:create_order).and_return(taler_order_id) + + # And fake the payment status to avoid user interaction. + allow_any_instance_of(Taler::Client) + .to receive(:fetch_order) do + payment = Spree::Payment.last + url = payment_gateways_confirm_taler_path(payment_id: payment.id) + + { "order_status_url" => url, "order_status" => "paid" } + end + end + + it_behaves_like "different payment methods", "Taler" + end end end From d748972fca6c918a936fc3a2cbc1d97d17643179 Mon Sep 17 00:00:00 2001 From: Maikel Linke Date: Mon, 26 Jan 2026 10:30:11 +1100 Subject: [PATCH 7/9] Resolve flaky spec with defined order --- app/models/invoice/data_presenter.rb | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/app/models/invoice/data_presenter.rb b/app/models/invoice/data_presenter.rb index 405afdaf1f..3b851bf1d9 100644 --- a/app/models/invoice/data_presenter.rb +++ b/app/models/invoice/data_presenter.rb @@ -95,8 +95,8 @@ class Invoice def display_line_item_tax_rate(item) all_tax_adjustments.select { |a| a.adjustable.type == 'Spree::LineItem' && a.adjustable.id == item.id - }.map(&:originator).map { |tr| - number_to_percentage(tr.amount * 100, precision: 1) + }.map(&:originator).map(&:amount).sort.map { |amount| + number_to_percentage(amount * 100, precision: 1) }.join(", ") end From f3428494fc8e2b4dc1d1445f9fbf0c6514fb42dd Mon Sep 17 00:00:00 2001 From: Maikel Linke Date: Wed, 28 Jan 2026 14:01:01 +1100 Subject: [PATCH 8/9] Better name method spec --- .../retrieves_a_URL_to_pay_at.yml | 0 ...eves_a_URL_to_pay_at_and_stores_it_on_the_payment_record.yml | 0 spec/models/spree/payment_method/taler_spec.rb | 2 +- 3 files changed, 1 insertion(+), 1 deletion(-) rename spec/fixtures/vcr_cassettes/Spree_PaymentMethod_Taler/{external_payment_url => _external_payment_url}/retrieves_a_URL_to_pay_at.yml (100%) rename spec/fixtures/vcr_cassettes/Spree_PaymentMethod_Taler/{external_payment_url => _external_payment_url}/retrieves_a_URL_to_pay_at_and_stores_it_on_the_payment_record.yml (100%) diff --git a/spec/fixtures/vcr_cassettes/Spree_PaymentMethod_Taler/external_payment_url/retrieves_a_URL_to_pay_at.yml b/spec/fixtures/vcr_cassettes/Spree_PaymentMethod_Taler/_external_payment_url/retrieves_a_URL_to_pay_at.yml similarity index 100% rename from spec/fixtures/vcr_cassettes/Spree_PaymentMethod_Taler/external_payment_url/retrieves_a_URL_to_pay_at.yml rename to spec/fixtures/vcr_cassettes/Spree_PaymentMethod_Taler/_external_payment_url/retrieves_a_URL_to_pay_at.yml diff --git a/spec/fixtures/vcr_cassettes/Spree_PaymentMethod_Taler/external_payment_url/retrieves_a_URL_to_pay_at_and_stores_it_on_the_payment_record.yml b/spec/fixtures/vcr_cassettes/Spree_PaymentMethod_Taler/_external_payment_url/retrieves_a_URL_to_pay_at_and_stores_it_on_the_payment_record.yml similarity index 100% rename from spec/fixtures/vcr_cassettes/Spree_PaymentMethod_Taler/external_payment_url/retrieves_a_URL_to_pay_at_and_stores_it_on_the_payment_record.yml rename to spec/fixtures/vcr_cassettes/Spree_PaymentMethod_Taler/_external_payment_url/retrieves_a_URL_to_pay_at_and_stores_it_on_the_payment_record.yml diff --git a/spec/models/spree/payment_method/taler_spec.rb b/spec/models/spree/payment_method/taler_spec.rb index 25c619b5fc..1d9bf77ec8 100644 --- a/spec/models/spree/payment_method/taler_spec.rb +++ b/spec/models/spree/payment_method/taler_spec.rb @@ -10,7 +10,7 @@ RSpec.describe Spree::PaymentMethod::Taler do ) } - describe "external_payment_url", vcr: true do + describe "#external_payment_url", vcr: true do it "retrieves a URL to pay at and stores it on the payment record" do order = create(:order_ready_for_confirmation, payment_method: taler) url = subject.external_payment_url(order:) From c115ab7a0d713bf44e8424de9e1b84c0e4ac2f73 Mon Sep 17 00:00:00 2001 From: Maikel Linke Date: Wed, 28 Jan 2026 15:03:56 +1100 Subject: [PATCH 9/9] Translate Taler payment status to error message --- app/models/spree/payment_method/taler.rb | 3 +- config/locales/en.yml | 3 ++ .../models/spree/payment_method/taler_spec.rb | 32 +++++++++++++++++-- 3 files changed, 35 insertions(+), 3 deletions(-) diff --git a/app/models/spree/payment_method/taler.rb b/app/models/spree/payment_method/taler.rb index 273b11ee9d..d50625b309 100644 --- a/app/models/spree/payment_method/taler.rb +++ b/app/models/spree/payment_method/taler.rb @@ -56,8 +56,9 @@ module Spree taler_order = client.fetch_order(payment.response_code) status = taler_order["order_status"] success = (status == "paid") + message = I18n.t(status, default: status, scope: "taler.order_status") - ActiveMerchant::Billing::Response.new(success, status) + ActiveMerchant::Billing::Response.new(success, message) end private diff --git a/config/locales/en.yml b/config/locales/en.yml index 1d081df210..e34ed5c0d2 100644 --- a/config/locales/en.yml +++ b/config/locales/en.yml @@ -288,6 +288,9 @@ en: success_code: disconnected: "Stripe account disconnected." + taler: + order_status: + claimed: "The payment request expired. Please try again." activemodel: errors: messages: diff --git a/spec/models/spree/payment_method/taler_spec.rb b/spec/models/spree/payment_method/taler_spec.rb index 1d9bf77ec8..fb314043b1 100644 --- a/spec/models/spree/payment_method/taler_spec.rb +++ b/spec/models/spree/payment_method/taler_spec.rb @@ -5,20 +5,48 @@ require 'spec_helper' RSpec.describe Spree::PaymentMethod::Taler do subject(:taler) { Spree::PaymentMethod::Taler.new( - preferred_backend_url: "https://backend.demo.taler.net/instances/sandbox", + preferred_backend_url: backend_url, preferred_api_key: "sandbox", ) } + let(:backend_url) { "https://backend.demo.taler.net/instances/sandbox" } describe "#external_payment_url", vcr: true do it "retrieves a URL to pay at and stores it on the payment record" do order = create(:order_ready_for_confirmation, payment_method: taler) url = subject.external_payment_url(order:) - expect(url).to match %r{\Ahttps://backend.demo.taler.net/instances/sandb} + expect(url).to start_with backend_url payment = order.payments.last.reload expect(payment.response_code).to match "2026.022-0284X4GE8WKMJ" expect(payment.redirect_auth_url).to eq url end end + + describe "#purchase" do + let(:money) { 100 } + let(:source) { taler } + let(:payment) { build(:payment, response_code: "taler-order-7") } + let(:order_url) { "#{backend_url}/private/orders/taler-order-7" } + + it "returns an ActiveMerchant response" do + order_status = "paid" + stub_request(:get, order_url).to_return(body: { order_status: }.to_json) + + response = taler.purchase(nil, nil, payment:) + + expect(response.success?).to eq true + expect(response.message).to eq "paid" + end + + it "translates error messages" do + order_status = "claimed" + stub_request(:get, order_url).to_return(body: { order_status: }.to_json) + + response = taler.purchase(nil, nil, payment:) + + expect(response.success?).to eq false + expect(response.message).to eq "The payment request expired. Please try again." + end + end end