SolidNetsEasyClient 0.1.1

dotnet add package SolidNetsEasyClient --version 0.1.1                
NuGet\Install-Package SolidNetsEasyClient -Version 0.1.1                
This command is intended to be used within the Package Manager Console in Visual Studio, as it uses the NuGet module's version of Install-Package.
<PackageReference Include="SolidNetsEasyClient" Version="0.1.1" />                
For projects that support PackageReference, copy this XML node into the project file to reference the package.
paket add SolidNetsEasyClient --version 0.1.1                
#r "nuget: SolidNetsEasyClient, 0.1.1"                
#r directive can be used in F# Interactive and Polyglot Notebooks. Copy this into the interactive tool or source code of the script to reference the package.
// Install SolidNetsEasyClient as a Cake Addin
#addin nuget:?package=SolidNetsEasyClient&version=0.1.1

// Install SolidNetsEasyClient as a Cake Tool
#tool nuget:?package=SolidNetsEasyClient&version=0.1.1                

NuGet Version

What is it?

This is a client for making type safe requests to the Easy Payment API endpoint.

Status

Current development status is: under development, somewhat functional library. Since Nexi and Nets have had a merger, this library uses a mixture of naming references to Nets and Nexi.

Quickstart

Add the package to your project:

$ `dotnet add package SolidNetsEasyClient`

Register the service in the startup process:

// Register nets services
builder.Services
  .AddNetsEasy(options =>
  {
    // Configure all the options here...
    // MUST set these either through appsettings.json or via env vars:
    options.ApiKey = "my-api-key";
    options.TermsUrl = "https://my-site/terms-and-conditions";
    options.PrivacyPolicyUrl = "https://my-site/privacy-policy";

    // If embedded checkout:
    options.IntegrationType = Integration.EmbeddedCheckout;
    options.CheckoutUrl = "https://exact.url.to/checkout";
    options.CheckoutKey = "my-checkout-key";

    // If hosted on nexi:
    options.IntegrationType = Integration.HostedPaymentPage;
    options.ReturnUrl = "https://redirect-on-success.to/my-page";
    options.CancelUrl = "https://redirect-on-cancel.to/my-other-page";
  });

Use the payment builder to construct a payment object, or make a PaymentRequest yourself:

Order order = new Order { /* With order items */};
NetsPaymentBuilder builder; // Inject by dependency injection
PaymentRequest payment = builder.CreatePayment(order)
                        .WithPrivateCustomer(
                            customerId: "myInternalCustomerId",
                            firstName: "John",
                            lastName: "Doe",
                            email: "john@doe.com"
                        )
                        .MerchantHandlesCustomerData()
                        .AddCustomer()
                        .ChargeImmidiately()
                        .AddWebhook("https://my-site/callback/charge", EventName.ChargeCreated, "authorizationheader123")
                        .AddWebhook("https://my-site/callback", EventName.PaymentCreated, "authorizationheader123")
                        .Build();

Then start a checkout session using the http client NexiClient or the typed IPaymentClient.

NexiClient client; // Inject by dependency injection
PaymentResult? payment = await client.StartCheckoutPayment(payment);

If using the EmbeddedCheckout - see this documentation for more information: https://developer.nexigroup.com/nexi-checkout/en-EU/docs/web-integration/integrate-checkout-on-your-website-embedded/

If using the HostedPaymentPage - see this documentation for more information: https://developer.nexigroup.com/nexi-checkout/en-EU/docs/web-integration/integrate-checkout-on-your-website-hosted/

Embedded checkout

The overall steps for the embedded checkout is as follows:

  1. Customer creates an order
  2. Your backend receives the order
  3. Initiate a checkout by contacting Nexi/Nets (backend)
  4. Get paymentId from Nets (backend)
  5. Make the frontend use paymentId and your checkoutKey
  6. Customer inputs their payment details
  7. Nets receives the payment details
  8. Nets charges the customer
  9. Nets calls your endpoint via webhooks

The customer never leaves your checkout page, the Nets integration is displayed in an iframe by using the Nets Checkout JS SDK.
The benefit is the possibility for a more streamlined customer experience, by only having the customer in 1 place and being able to customize the look and feel of your checkout page.

The drawback is more work for the developer, and the result is only as good as the developer makes it.

Hosted checkout

  1. Customer creates an order
  2. Your backend receives the order
  3. Initiate checkout by contacting Nexi/Nets (backend)
  4. Redirect customer to Nets
  5. Customer inputs their payment details on Nets
  6. Nets redirects customer to your page
  7. Nets charges the customer
  8. Nets calls your endpoint via webhooks

The customer will leave your page and be redirected upon the start and end of their customer journey.

The benefit is a simpler developer flow and maybe easier to customize some parts. The base checkout is already developed by Nets/Nexi.
The drawback is the customer redirection, and limited customization.

MVC

TODO

Minimal Api

TODO

Webhooks

The webhook notifications, will be directed to the endpoint you specified in the AddWebhook builder method.
It is highly recommended to use these for processing customer payments. The callback webhook will be an http POST request with an authorization header as specified on the payment request.

For example, you can either use the generic interface IWebhook<WebhookData> to catch any webhook callbacks OR you can specify the expected callback type (e.g) ChargeCreated.

[HttpPost("/callback")]
public IResult Method(IWebhook<WebhookData> payload)
{
    if (payload is PaymentCreated paymentCreated)
    {
        // Handle payment creation..
    }
    else if (payload is ChargeFailed chargeFailed)
    {
        // etc.
    }
}
// Route: https://my-site/callback/charge
public IResult Method(ChargeCreation payload)
{
    // Handle charge creation..
}
Nets event name Description (from Nets Easy) DTO
payment.created A payment has been created. PaymentCreated
payment.reservation.created The amount of the payment has been reserved. ReservationCreatedV1
payment.reservation.created.v2 The amount of the payment has been reserved. ReservationCreatedV2
payment.reservation.failed A reservation attempt has failed. ReservationFailed
payment.checkout.completed The customer has completed the checkout. CheckoutCompleted
payment.charge.created.v2 The customer has successfully been charged, partially or fully. ChargeCreated
payment.charge.failed A charge attempt has failed. ChargeFailed
payment.refund.initiated.v2 A refund has been initiated. RefundInitiated
payment.refund.failed A refund attempt has failed. RefundFailed
payment.refund.completed A refund has successfully been completed. RefundCompleted
payment.cancel.created A reservation has been canceled. PaymentCancelled
payment.cancel.failed A cancellation has failed. PaymentCancellationFailed

Proxy

You must remember to configure the forwarded headers, if you want ASP.NET Core to work with proxy servers and load balancers.
Documentation reference: https://learn.microsoft.com/en-us/aspnet/core/host-and-deploy/proxy-load-balancer?view=aspnetcore-8.0

Summary

Configure the forwarded headers for X-Forwarded-For header field.

builder.Services.Configure<ForwardedHeadersOptions>(options =>
{
    options.ForwardedHeaders =
        ForwardedHeaders.XForwardedFor | ForwardedHeaders.XForwardedProto;
});

Then insert the middleware as the first item in the pipeline:

app.UseForwardedHeaders();

IP-filter

The SolidNetsEasyIPFilterAttribute can both whitelist and blacklist ips or ip-ranges (using the CIDR-format). Thus minimally ensuring that any webhook requests are coming from an accepted source.

For the minimal API you can use the endpoint filter WebhookFilter to ensure that the IP is on the whitelist.
Or just use the MapNetsWebhook extension method, which also ensures the response to be 200 OK and listens on POST requests.

Terminology for Nexi-Nets

According to my own understanding.

Term Definition
Merchant You, the seller.
Consumer The customer.
Checkout The time when the customer is inputting their payment details.
Payment A request from merchant to the consumer, to get paid.
Charge When money has been transfered thus finalizing the payment.
Order The items the consumer has bought.
Subscription A completely regular recurring payment, which cannot change in either Amount or in Time. E.g. Always 50 DKK on the last day of the month. Has an exact end date.
Unscheduled subscription A variable recurring payment, which can change in Amount and in Time. Consumer MUST agree to the terms that Nexi/Nets stores the consumer's payment details. Has no end date.

Features

Use the client to make and manage payments, from the backend. Remember that the you still need a frontend for a customer to input payment details.

Client

Using the NexiClient you must ensure that the http client is not used in a singleton.
See the guidelines here: https://learn.microsoft.com/en-us/dotnet/fundamentals/networking/http/httpclient-guidelines

The ASP.NET Core will use the HttpClient and dispose it after use, but the HttpMessageHandler will be reused depending on their lifetimes.

The pooling ensures that port and socket exhaustion is limited.
The lifetime restriction ensures that new DNS changes can be updated.

There is only 1 client with 4 typed interfaces.

  • IPaymentClient - handles the checkout, and payment.
  • IChargeClient - handles the charge and refund of a payment.
  • ISubscriptionClient - handles the subscriptions, bulk charging and bulk verification.
  • IUnscheduledSubscriptionClient - handles unscheduled subscriptions, bulk charging and bulk verification of cards.

Roadmap

  • Handle payments, subscriptions and webhooks in a type safe and easy to use way.
  • Create nuget package
  • Add unit tests
  • Add easy to use configuration for handling API keys and other client settings
  • Add example site
  • Consolidate the various Models into related models
  • Add more unit tests
  • Proof read README
  • Add options section that explains each option
  • Add Nets Terminology for vocabulary such as: Payment, checkout, charge...
Product Compatible and additional computed target framework versions.
.NET net8.0 is compatible.  net8.0-android was computed.  net8.0-browser was computed.  net8.0-ios was computed.  net8.0-maccatalyst was computed.  net8.0-macos was computed.  net8.0-tvos was computed.  net8.0-windows was computed.  net9.0 was computed.  net9.0-android was computed.  net9.0-browser was computed.  net9.0-ios was computed.  net9.0-maccatalyst was computed.  net9.0-macos was computed.  net9.0-tvos was computed.  net9.0-windows was computed. 
Compatible target framework(s)
Included target framework(s) (in package)
Learn more about Target Frameworks and .NET Standard.

NuGet packages

This package is not used by any NuGet packages.

GitHub repositories

This package is not used by any popular GitHub repositories.

Version Downloads Last updated
0.1.1 432 5/16/2024
0.1.0 99 5/16/2024
0.0.24 190 9/6/2023
0.0.23 155 9/6/2023
0.0.22 220 4/27/2023
0.0.21 312 2/22/2023
0.0.20 258 2/22/2023
0.0.19 288 2/16/2023
0.0.18 301 2/11/2023
0.0.16 283 2/11/2023
0.0.15 290 2/10/2023
0.0.14 306 1/31/2023
0.0.13 302 1/30/2023
0.0.12 343 1/22/2023
0.0.11 339 1/21/2023
0.0.9 366 12/10/2022
0.0.8 327 12/9/2022
0.0.7 366 11/28/2022
0.0.6 361 11/24/2022
0.0.5 388 11/11/2022
0.0.4 409 11/2/2022
0.0.3 424 10/26/2022