workarea-klarna 1.0.0.beta1
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +7 -0
- data/.editorconfig +20 -0
- data/.eslintrc.json +36 -0
- data/.github/workflows/ci.yml +61 -0
- data/.gitignore +24 -0
- data/.rubocop.yml +2 -0
- data/.stylelintrc.json +8 -0
- data/.yardopts +1 -0
- data/CHANGELOG.md +81 -0
- data/Gemfile +17 -0
- data/LICENSE +52 -0
- data/README.md +79 -0
- data/Rakefile +56 -0
- data/app/assets/javascripts/workarea/storefront/klarna/modules/klarna_widget.js +99 -0
- data/app/controllers/workarea/storefront/checkout/addresses_controller.decorator +8 -0
- data/app/controllers/workarea/storefront/checkout/place_order_controller.decorator +9 -0
- data/app/controllers/workarea/storefront/checkouts_controller.decorator +13 -0
- data/app/models/workarea/checkout/steps/payment.decorator +22 -0
- data/app/models/workarea/order.decorator +7 -0
- data/app/models/workarea/payment.decorator +34 -0
- data/app/models/workarea/payment/authorize/klarna.rb +21 -0
- data/app/models/workarea/payment/capture/klarna.rb +21 -0
- data/app/models/workarea/payment/klarna_session.rb +18 -0
- data/app/models/workarea/payment/purchase/klarna.rb +21 -0
- data/app/models/workarea/payment/refund/klarna.rb +18 -0
- data/app/models/workarea/payment/tender/klarna.rb +53 -0
- data/app/services/workarea/setup_klarna_session.rb +24 -0
- data/app/view_models/workarea/storefront/checkout/payment_view_model.decorator +41 -0
- data/app/views/workarea/admin/orders/tenders/_klarna.html.haml +4 -0
- data/app/views/workarea/storefront/checkouts/_klarna_payments.html.haml +22 -0
- data/app/views/workarea/storefront/klarna/_sdk.html.haml +2 -0
- data/app/views/workarea/storefront/order_mailer/tenders/_klarna.html.haml +2 -0
- data/app/views/workarea/storefront/orders/tenders/_klarna.html.haml +6 -0
- data/bin/rails +20 -0
- data/config/initializers/appends.rb +14 -0
- data/config/initializers/configuration.rb +32 -0
- data/config/initializers/fields.rb +43 -0
- data/config/locales/en.yml +30 -0
- data/config/routes.rb +2 -0
- data/lib/workarea/klarna.rb +30 -0
- data/lib/workarea/klarna/bogus_gateway.rb +19 -0
- data/lib/workarea/klarna/engine.rb +10 -0
- data/lib/workarea/klarna/gateway.rb +92 -0
- data/lib/workarea/klarna/gateway/cancel_request.rb +16 -0
- data/lib/workarea/klarna/gateway/capture_request.rb +24 -0
- data/lib/workarea/klarna/gateway/create_session_request.rb +22 -0
- data/lib/workarea/klarna/gateway/order.rb +249 -0
- data/lib/workarea/klarna/gateway/place_order_request.rb +34 -0
- data/lib/workarea/klarna/gateway/refund_request.rb +24 -0
- data/lib/workarea/klarna/gateway/release_request.rb +16 -0
- data/lib/workarea/klarna/gateway/request.rb +91 -0
- data/lib/workarea/klarna/gateway/response.rb +38 -0
- data/lib/workarea/klarna/gateway/update_session_request.rb +30 -0
- data/lib/workarea/klarna/version.rb +5 -0
- data/package.json +9 -0
- data/test/dummy/Rakefile +6 -0
- data/test/dummy/app/assets/config/manifest.js +4 -0
- data/test/dummy/app/assets/javascripts/application.js +13 -0
- data/test/dummy/app/assets/stylesheets/application.css +15 -0
- data/test/dummy/app/controllers/application_controller.rb +3 -0
- data/test/dummy/app/helpers/application_helper.rb +2 -0
- data/test/dummy/app/jobs/application_job.rb +2 -0
- data/test/dummy/app/mailers/application_mailer.rb +4 -0
- data/test/dummy/app/views/layouts/application.html.erb +14 -0
- data/test/dummy/app/views/layouts/mailer.html.erb +13 -0
- data/test/dummy/app/views/layouts/mailer.text.erb +1 -0
- data/test/dummy/bin/bundle +3 -0
- data/test/dummy/bin/rails +4 -0
- data/test/dummy/bin/rake +4 -0
- data/test/dummy/bin/setup +34 -0
- data/test/dummy/bin/update +29 -0
- data/test/dummy/config.ru +5 -0
- data/test/dummy/config/application.rb +24 -0
- data/test/dummy/config/boot.rb +5 -0
- data/test/dummy/config/cable.yml +9 -0
- data/test/dummy/config/environment.rb +5 -0
- data/test/dummy/config/environments/development.rb +54 -0
- data/test/dummy/config/environments/production.rb +86 -0
- data/test/dummy/config/environments/test.rb +43 -0
- data/test/dummy/config/initializers/application_controller_renderer.rb +6 -0
- data/test/dummy/config/initializers/assets.rb +11 -0
- data/test/dummy/config/initializers/backtrace_silencers.rb +7 -0
- data/test/dummy/config/initializers/cookies_serializer.rb +5 -0
- data/test/dummy/config/initializers/filter_parameter_logging.rb +4 -0
- data/test/dummy/config/initializers/inflections.rb +16 -0
- data/test/dummy/config/initializers/mime_types.rb +4 -0
- data/test/dummy/config/initializers/new_framework_defaults.rb +21 -0
- data/test/dummy/config/initializers/workarea.rb +5 -0
- data/test/dummy/config/initializers/wrap_parameters.rb +14 -0
- data/test/dummy/config/locales/en.yml +23 -0
- data/test/dummy/config/puma.rb +47 -0
- data/test/dummy/config/routes.rb +5 -0
- data/test/dummy/config/secrets.yml +22 -0
- data/test/dummy/config/spring.rb +6 -0
- data/test/dummy/db/seeds.rb +2 -0
- data/test/dummy/log/.keep +0 -0
- data/test/factories/workarea/klarna.rb +146 -0
- data/test/integration/workarea/storefront/klarna_session_integration_test.rb +40 -0
- data/test/lib/workarea/klarna/gateway/capture_request_test.rb +26 -0
- data/test/lib/workarea/klarna/gateway/create_session_request_test.rb +26 -0
- data/test/lib/workarea/klarna/gateway/order_test.rb +135 -0
- data/test/lib/workarea/klarna/gateway/place_order_request_test.rb +34 -0
- data/test/lib/workarea/klarna/gateway/refund_request_test.rb +26 -0
- data/test/lib/workarea/klarna/gateway/request_test.rb +87 -0
- data/test/lib/workarea/klarna/gateway/response_test.rb +75 -0
- data/test/lib/workarea/klarna/gateway/update_session_request_test.rb +27 -0
- data/test/lib/workarea/klarna/gateway_test.rb +58 -0
- data/test/models/workarea/checkout/steps/klarna_payment_test.rb +79 -0
- data/test/models/workarea/klarna_payment_test.rb +45 -0
- data/test/models/workarea/payment/authorize/klarna_test.rb +34 -0
- data/test/models/workarea/payment/capture/klarna_test.rb +51 -0
- data/test/models/workarea/payment/tender/klarna_test.rb +81 -0
- data/test/services/workarea/setup_klarna_session_test.rb +42 -0
- data/test/teaspoon_env.rb +6 -0
- data/test/test_helper.rb +10 -0
- data/test/vcr_cassettes/klarna_authorize.yml +62 -0
- data/test/vcr_cassettes/klarna_capture.yml +60 -0
- data/test/vcr_cassettes/klarna_create_session.yml +61 -0
- data/test/vcr_cassettes/klarna_refund.yml +60 -0
- data/test/view_models/workarea/storefront/checkout/klarna_payment_view_model_test.rb +62 -0
- data/workarea-klarna.gemspec +20 -0
- metadata +177 -0
@@ -0,0 +1,99 @@
|
|
1
|
+
/**
|
2
|
+
* @namespace WORKAREA.klarnaWidget
|
3
|
+
*/
|
4
|
+
WORKAREA.registerModule('klarnaWidget', (function () {
|
5
|
+
'use strict';
|
6
|
+
|
7
|
+
var authorizePayment = function(session, event) {
|
8
|
+
var $submitButton = $(event.target),
|
9
|
+
$form = $submitButton.closest('form'),
|
10
|
+
$selectedPayment = $form.find('input[name=payment]:checked'),
|
11
|
+
paymentCategory = $selectedPayment.data('paymentCategory');
|
12
|
+
|
13
|
+
if (_.isEmpty(paymentCategory)) { return; }
|
14
|
+
|
15
|
+
event.preventDefault();
|
16
|
+
|
17
|
+
Klarna.Payments.authorize(
|
18
|
+
{
|
19
|
+
payment_method_category: paymentCategory
|
20
|
+
},
|
21
|
+
session.order,
|
22
|
+
function(res) {
|
23
|
+
if (res.approved && res.show_form) {
|
24
|
+
$form.find('#klarna_authorization_token').val(res.authorization_token);
|
25
|
+
$submitButton
|
26
|
+
.removeAttr('disabled')
|
27
|
+
.trigger('click');
|
28
|
+
|
29
|
+
} else if (res.show_form) {
|
30
|
+
setupListener($form, session);
|
31
|
+
|
32
|
+
$submitButton
|
33
|
+
.removeAttr('disabled')
|
34
|
+
.text(I18n.t('workarea.storefront.checkouts.place_order'));
|
35
|
+
|
36
|
+
} else {
|
37
|
+
setupListener($form, session);
|
38
|
+
|
39
|
+
$selectedPayment
|
40
|
+
.closest('.checkout-payment__primary-method--klarna')
|
41
|
+
.remove();
|
42
|
+
|
43
|
+
$form.find('input[name=payment]').first().trigger('click');
|
44
|
+
|
45
|
+
WORKAREA.messages.insertMessage(
|
46
|
+
I18n.t('workarea.storefront.widget.invalid_payment'),
|
47
|
+
'error'
|
48
|
+
);
|
49
|
+
}
|
50
|
+
}
|
51
|
+
);
|
52
|
+
},
|
53
|
+
|
54
|
+
setupWidget = function(session, payment) {
|
55
|
+
var $payment = $(payment);
|
56
|
+
|
57
|
+
Klarna.Payments.load(
|
58
|
+
{
|
59
|
+
container: '#' + $payment.prop('id'),
|
60
|
+
payment_method_category: $payment.data('klarnaPayment')
|
61
|
+
},
|
62
|
+
_.omit(session.order, 'shipping_address', 'billing_address'),
|
63
|
+
function(res) {
|
64
|
+
if ( ! res.show_form) {
|
65
|
+
$payment
|
66
|
+
.closest('.checkout-payment__primary-method')
|
67
|
+
.remove();
|
68
|
+
}
|
69
|
+
}
|
70
|
+
);
|
71
|
+
},
|
72
|
+
|
73
|
+
setupListener = function($form, session) {
|
74
|
+
$form.one('click', '[type=submit]', _.partial(authorizePayment, session));
|
75
|
+
},
|
76
|
+
|
77
|
+
/**
|
78
|
+
* @method
|
79
|
+
* @name init
|
80
|
+
* @memberof WORKAREA.klarnaWidget
|
81
|
+
*/
|
82
|
+
init = function () {
|
83
|
+
var $container = $('[data-klarna-session]'),
|
84
|
+
session = $container.data('klarnaSession'),
|
85
|
+
$payments = $('[data-klarna-payment]');
|
86
|
+
|
87
|
+
if (session === undefined || _.isEmpty($payments)) {
|
88
|
+
return;
|
89
|
+
}
|
90
|
+
|
91
|
+
Klarna.Payments.init({ client_token: session.client_token });
|
92
|
+
|
93
|
+
_.each($payments, _.partial(setupWidget, session));
|
94
|
+
|
95
|
+
setupListener($container.closest('form'), session);
|
96
|
+
};
|
97
|
+
|
98
|
+
window.klarnaAsyncCallback = init;
|
99
|
+
}()));
|
@@ -0,0 +1,22 @@
|
|
1
|
+
module Workarea
|
2
|
+
decorate Checkout::Steps::Payment, with: :klarna do
|
3
|
+
private
|
4
|
+
# decorating this method to inject this logic at the right part of
|
5
|
+
# the payment step without redefining all of #update.
|
6
|
+
def set_credit_card(params)
|
7
|
+
payment_type = params[:payment]
|
8
|
+
auth_token = params.fetch(:klarna, {})[:authorization_token]
|
9
|
+
|
10
|
+
|
11
|
+
unless payment_type.to_s.starts_with?('klarna') && auth_token.present?
|
12
|
+
payment.clear_klarna
|
13
|
+
return super
|
14
|
+
end
|
15
|
+
|
16
|
+
payment.set_klarna(
|
17
|
+
authorization_token: auth_token,
|
18
|
+
payment_method_category: payment_type.sub('klarna_', '')
|
19
|
+
)
|
20
|
+
end
|
21
|
+
end
|
22
|
+
end
|
@@ -0,0 +1,34 @@
|
|
1
|
+
module Workarea
|
2
|
+
decorate Payment, with: :klarna do
|
3
|
+
decorated do
|
4
|
+
embeds_one :klarna,
|
5
|
+
class_name: "Workarea::Payment::Tender::Klarna",
|
6
|
+
inverse_of: :payment
|
7
|
+
|
8
|
+
delegate :redirect_url, to: :klarna, allow_nil: true
|
9
|
+
end
|
10
|
+
|
11
|
+
def set_klarna(attrs)
|
12
|
+
build_klarna unless klarna
|
13
|
+
klarna.attributes = attrs.slice(
|
14
|
+
:authorization_token,
|
15
|
+
:authorization_token_expires_at,
|
16
|
+
:payment_method_category
|
17
|
+
)
|
18
|
+
|
19
|
+
save
|
20
|
+
end
|
21
|
+
|
22
|
+
def clear_klarna
|
23
|
+
self.klarna = nil
|
24
|
+
end
|
25
|
+
|
26
|
+
def eligible_for_klarna?
|
27
|
+
address.present? &&
|
28
|
+
address.country&.continent.present? &&
|
29
|
+
address.country.continent.in?(
|
30
|
+
Workarea.config.klarna_continent_keys.keys
|
31
|
+
)
|
32
|
+
end
|
33
|
+
end
|
34
|
+
end
|
@@ -0,0 +1,21 @@
|
|
1
|
+
module Workarea
|
2
|
+
class Payment
|
3
|
+
module Authorize
|
4
|
+
class Klarna
|
5
|
+
include OperationImplementation
|
6
|
+
|
7
|
+
def complete!
|
8
|
+
transaction.response =
|
9
|
+
Workarea::Klarna.gateway.authorize(tender, transaction.amount)
|
10
|
+
end
|
11
|
+
|
12
|
+
def cancel!
|
13
|
+
return unless transaction.success?
|
14
|
+
|
15
|
+
transaction.cancellation = Workarea::Klarna.gateway.cancel(tender)
|
16
|
+
tender.clear_authorization!
|
17
|
+
end
|
18
|
+
end
|
19
|
+
end
|
20
|
+
end
|
21
|
+
end
|
@@ -0,0 +1,21 @@
|
|
1
|
+
module Workarea
|
2
|
+
class Payment
|
3
|
+
class Capture
|
4
|
+
class Klarna
|
5
|
+
include OperationImplementation
|
6
|
+
|
7
|
+
def complete!
|
8
|
+
transaction.response =
|
9
|
+
Workarea::Klarna.gateway.capture(tender, transaction.amount)
|
10
|
+
end
|
11
|
+
|
12
|
+
def cancel!
|
13
|
+
return unless transaction.success?
|
14
|
+
|
15
|
+
transaction.cancellation =
|
16
|
+
Workarea::Klarna.gateway.refund(tender, transaction.amount)
|
17
|
+
end
|
18
|
+
end
|
19
|
+
end
|
20
|
+
end
|
21
|
+
end
|
@@ -0,0 +1,18 @@
|
|
1
|
+
module Workarea
|
2
|
+
class Payment
|
3
|
+
class KlarnaSession
|
4
|
+
include ApplicationDocument
|
5
|
+
|
6
|
+
# The _id field will be the order ID
|
7
|
+
field :_id, type: String, default: -> { BSON::ObjectId.new.to_s }
|
8
|
+
field :session_id, type: String
|
9
|
+
field :client_token, type: String
|
10
|
+
field :payment_method_categories, type: Array, default: []
|
11
|
+
|
12
|
+
index(
|
13
|
+
{ created_at: 1 },
|
14
|
+
{ expire_after_seconds: Workarea.config.klarn_session_expiration.to_i }
|
15
|
+
)
|
16
|
+
end
|
17
|
+
end
|
18
|
+
end
|
@@ -0,0 +1,21 @@
|
|
1
|
+
module Workarea
|
2
|
+
class Payment
|
3
|
+
module Purchase
|
4
|
+
class Klarna
|
5
|
+
include OperationImplementation
|
6
|
+
|
7
|
+
def complete!
|
8
|
+
transaction.response =
|
9
|
+
Workarea::Klarna.gateway.purchase(tender, transaction.amount)
|
10
|
+
end
|
11
|
+
|
12
|
+
def cancel!
|
13
|
+
return unless transaction.success?
|
14
|
+
|
15
|
+
transaction.cancellation =
|
16
|
+
Workarea::Klarna.gateway.refund(tender, transaction.amount)
|
17
|
+
end
|
18
|
+
end
|
19
|
+
end
|
20
|
+
end
|
21
|
+
end
|
@@ -0,0 +1,18 @@
|
|
1
|
+
module Workarea
|
2
|
+
class Payment
|
3
|
+
class Refund
|
4
|
+
class Klarna
|
5
|
+
include OperationImplementation
|
6
|
+
|
7
|
+
def complete!
|
8
|
+
transaction.response =
|
9
|
+
Workarea::Klarna.gateway.refund(tender, transaction.amount)
|
10
|
+
end
|
11
|
+
|
12
|
+
def cancel!
|
13
|
+
# noop
|
14
|
+
end
|
15
|
+
end
|
16
|
+
end
|
17
|
+
end
|
18
|
+
end
|
@@ -0,0 +1,53 @@
|
|
1
|
+
module Workarea
|
2
|
+
class Payment
|
3
|
+
class Tender
|
4
|
+
class Klarna < Tender
|
5
|
+
field :authorization_token, type: String
|
6
|
+
field :authorization_token_expires_at, type: Time
|
7
|
+
field :payment_method_category, type: String
|
8
|
+
|
9
|
+
embedded_in :payment, class_name: 'Workarea::Payment'
|
10
|
+
|
11
|
+
def slug
|
12
|
+
:klarna
|
13
|
+
end
|
14
|
+
|
15
|
+
def authorization_token=(val)
|
16
|
+
if val.present? && authorization_token != val
|
17
|
+
self.authorization_token_expires_at = 1.hour.from_now
|
18
|
+
end
|
19
|
+
|
20
|
+
super
|
21
|
+
end
|
22
|
+
|
23
|
+
def authorization_token_expired?
|
24
|
+
authorization_token.present? &&
|
25
|
+
authorization_token_expires_at < Time.current
|
26
|
+
end
|
27
|
+
|
28
|
+
def clear_authorization!
|
29
|
+
update!(
|
30
|
+
authorization_token: nil,
|
31
|
+
authorization_token_expires_at: nil
|
32
|
+
)
|
33
|
+
end
|
34
|
+
|
35
|
+
def placed_order_data
|
36
|
+
txn = transactions.successful.not_canceled.authorizes.first ||
|
37
|
+
transactions.successful.not_canceled.captures_or_purchased.first
|
38
|
+
|
39
|
+
return {} unless txn.present?
|
40
|
+
txn.response.params
|
41
|
+
end
|
42
|
+
|
43
|
+
def order_id
|
44
|
+
placed_order_data['order_id']
|
45
|
+
end
|
46
|
+
|
47
|
+
def redirect_url
|
48
|
+
placed_order_data['redirect_url']
|
49
|
+
end
|
50
|
+
end
|
51
|
+
end
|
52
|
+
end
|
53
|
+
end
|
@@ -0,0 +1,24 @@
|
|
1
|
+
module Workarea
|
2
|
+
class SetupKlarnaSession
|
3
|
+
delegate :order, :payment, to: :@checkout
|
4
|
+
|
5
|
+
def initialize(checkout)
|
6
|
+
@checkout = checkout
|
7
|
+
end
|
8
|
+
|
9
|
+
def perform
|
10
|
+
return unless payment.eligible_for_klarna?
|
11
|
+
|
12
|
+
session = Payment::KlarnaSession.find_or_initialize_by(id: order.id)
|
13
|
+
|
14
|
+
if session.persisted?
|
15
|
+
Klarna.gateway.update_session(order, session.session_id)
|
16
|
+
else
|
17
|
+
response = Klarna.gateway.create_session(order)
|
18
|
+
return unless response.success?
|
19
|
+
|
20
|
+
session.update!(response.body)
|
21
|
+
end
|
22
|
+
end
|
23
|
+
end
|
24
|
+
end
|
@@ -0,0 +1,41 @@
|
|
1
|
+
module Workarea
|
2
|
+
decorate Storefront::Checkout::PaymentViewModel, with: :klarna do
|
3
|
+
def klarna_session
|
4
|
+
@klarna_session ||=
|
5
|
+
Payment::KlarnaSession.find_or_initialize_by(id: order.id)
|
6
|
+
end
|
7
|
+
|
8
|
+
def offer_klarna_payments?
|
9
|
+
payment.eligible_for_klarna? &&
|
10
|
+
klarna_session.client_token.present?
|
11
|
+
end
|
12
|
+
|
13
|
+
def klarna_data
|
14
|
+
@klarna_data ||= klarna_session
|
15
|
+
.attributes
|
16
|
+
.slice('client_token', 'payment_method_categories')
|
17
|
+
.merge(
|
18
|
+
order: Klarna::Gateway::Order
|
19
|
+
.new(order, payment: payment, shippings: shippings)
|
20
|
+
.to_h
|
21
|
+
.except(:merchant_urls)
|
22
|
+
)
|
23
|
+
end
|
24
|
+
|
25
|
+
def klarna_selected?
|
26
|
+
payment.klarna&.authorization_token.present?
|
27
|
+
end
|
28
|
+
|
29
|
+
def klarna_payment_category_selected?(category)
|
30
|
+
klarna_selected? && payment.klarna.payment_method_category == category
|
31
|
+
end
|
32
|
+
|
33
|
+
def klarna_expired?
|
34
|
+
klarna_selected? && payment.klarna.authorization_token_expired?
|
35
|
+
end
|
36
|
+
|
37
|
+
def using_new_card?
|
38
|
+
super && !klarna_selected?
|
39
|
+
end
|
40
|
+
end
|
41
|
+
end
|
@@ -0,0 +1,22 @@
|
|
1
|
+
- if step.offer_klarna_payments?
|
2
|
+
.hidden{ data: { klarna_session: step.klarna_data.to_json } }
|
3
|
+
= hidden_field_tag 'klarna[authorization_token]', step.payment.klarna&.authorization_token, id: 'klarna_authorization_token'
|
4
|
+
|
5
|
+
- step.klarna_session.payment_method_categories.each do |category|
|
6
|
+
- identifier = category['identifier']
|
7
|
+
.checkout-payment__primary-method.checkout-payment__primary-method--klarna.checkout-payment__primary-method--new{ class: "checkout-payment__primary-method--klarna_#{identifier} #{step.klarna_payment_category_selected?(identifier) ? 'checkout-payment__primary-method--selected' : nil}" }
|
8
|
+
.button-property
|
9
|
+
.value{ role: 'radiogroup', aria: { labelledby: 'aria_payment_radiogroup' } }
|
10
|
+
= radio_button_tag 'payment', "klarna_#{identifier}", step.klarna_payment_category_selected?(identifier), data: { payment_category: identifier, analytics: checkout_payment_selected_analytics_data("klarna_#{identifier}").to_json }
|
11
|
+
|
12
|
+
= label_tag 'payment[klarna]', nil, class: 'button-property__name' do
|
13
|
+
= image_tag(category['asset_urls']['standard'], class: "payment-icon payment-icon--#{identifier.dasherize}", title: category['name'])
|
14
|
+
%span.button-property__text= category['name']
|
15
|
+
|
16
|
+
- if step.klarna_payment_category_selected?(identifier) && !step.klarna_expired?
|
17
|
+
.checkout-payment__primary-method-edit= t('workarea.workarea.storefront.checkouts.klarna_payment_received')
|
18
|
+
- else
|
19
|
+
- if step.klarna_expired?
|
20
|
+
.checkout-payment__primary-method-edit= t('workarea.workarea.storefront.checkouts.klarna_payment_expired')
|
21
|
+
|
22
|
+
.checkout-payment__primary-method-edit{ data: { klarna_payment: identifier }, id: "klarna-payment--#{identifier}" }
|