From 7c1d893c91d22d7402504513d79fc94e0315dd10 Mon Sep 17 00:00:00 2001 From: Pa1NarK <69745008+pixincreate@users.noreply.github.com> Date: Wed, 21 May 2025 12:45:46 +0530 Subject: [PATCH] feat(connector): [nordea] template code (#8056) Co-authored-by: hyperswitch-bot[bot] <148525504+hyperswitch-bot[bot]@users.noreply.github.com> --- config/config.example.toml | 1 + config/deployments/integration_test.toml | 1 + config/deployments/production.toml | 1 + config/deployments/sandbox.toml | 1 + config/development.toml | 2 + config/docker_compose.toml | 2 + crates/common_enums/src/connector_enums.rs | 5 + crates/connector_configs/src/connector.rs | 1 + .../hyperswitch_connectors/src/connectors.rs | 9 +- .../src/connectors/nordea.rs | 568 ++++++++++++++++++ .../src/connectors/nordea/transformers.rs | 228 +++++++ .../src/default_implementations.rs | 44 ++ .../src/default_implementations_v2.rs | 25 + .../hyperswitch_domain_models/src/configs.rs | 1 + crates/router/src/connector.rs | 22 +- crates/router/src/core/admin.rs | 4 + crates/router/src/types/api.rs | 1 + crates/router/src/types/transformers.rs | 1 + crates/router/tests/connectors/main.rs | 1 + crates/router/tests/connectors/nordea.rs | 420 +++++++++++++ .../router/tests/connectors/sample_auth.toml | 4 + crates/test_utils/src/connector_auth.rs | 1 + loadtest/config/development.toml | 2 + scripts/add_connector.sh | 2 +- 24 files changed, 1331 insertions(+), 16 deletions(-) create mode 100644 crates/hyperswitch_connectors/src/connectors/nordea.rs create mode 100644 crates/hyperswitch_connectors/src/connectors/nordea/transformers.rs create mode 100644 crates/router/tests/connectors/nordea.rs diff --git a/config/config.example.toml b/config/config.example.toml index 5bc5523829..52ee69bec1 100644 --- a/config/config.example.toml +++ b/config/config.example.toml @@ -245,6 +245,7 @@ nexixpay.base_url = "https://xpaysandbox.nexigroup.com/api/phoenix-0.0/psp/api/v nmi.base_url = "https://secure.nmi.com/" nomupay.base_url = "https://payout-api.sandbox.nomupay.com" noon.base_url = "https://api-test.noonpayments.com/" +nordea.base_url = "https://api.nordeaopenbanking.com" novalnet.base_url = "https://payport.novalnet.de/v2" noon.key_mode = "Test" nuvei.base_url = "https://ppp-test.nuvei.com/" diff --git a/config/deployments/integration_test.toml b/config/deployments/integration_test.toml index a05db876f2..734fa08c88 100644 --- a/config/deployments/integration_test.toml +++ b/config/deployments/integration_test.toml @@ -88,6 +88,7 @@ nexixpay.base_url = "https://xpaysandbox.nexigroup.com/api/phoenix-0.0/psp/api/v nmi.base_url = "https://secure.nmi.com/" nomupay.base_url = "https://payout-api.sandbox.nomupay.com" noon.base_url = "https://api-test.noonpayments.com/" +nordea.base_url = "https://api.nordeaopenbanking.com" noon.key_mode = "Test" novalnet.base_url = "https://payport.novalnet.de/v2" nuvei.base_url = "https://ppp-test.nuvei.com/" diff --git a/config/deployments/production.toml b/config/deployments/production.toml index 828b144f1b..16e114a754 100644 --- a/config/deployments/production.toml +++ b/config/deployments/production.toml @@ -92,6 +92,7 @@ nexixpay.base_url = "https://xpay.nexigroup.com/api/phoenix-0.0/psp/api/v1" nmi.base_url = "https://secure.nmi.com/" nomupay.base_url = "https://payout-api.nomupay.com" noon.base_url = "https://api.noonpayments.com/" +nordea.base_url = "https://openapi.portal.nordea.com" noon.key_mode = "Live" novalnet.base_url = "https://payport.novalnet.de/v2" nuvei.base_url = "https://secure.safecharge.com/" diff --git a/config/deployments/sandbox.toml b/config/deployments/sandbox.toml index b12149d0dc..ec6572fd71 100644 --- a/config/deployments/sandbox.toml +++ b/config/deployments/sandbox.toml @@ -92,6 +92,7 @@ nexixpay.base_url = "https://xpaysandbox.nexigroup.com/api/phoenix-0.0/psp/api/v nmi.base_url = "https://secure.nmi.com/" nomupay.base_url = "https://payout-api.sandbox.nomupay.com" noon.base_url = "https://api-test.noonpayments.com/" +nordea.base_url = "https://api.nordeaopenbanking.com" noon.key_mode = "Test" novalnet.base_url = "https://payport.novalnet.de/v2" nuvei.base_url = "https://ppp-test.nuvei.com/" diff --git a/config/development.toml b/config/development.toml index 035671dec1..7deb3b4699 100644 --- a/config/development.toml +++ b/config/development.toml @@ -167,6 +167,7 @@ cards = [ "nmi", "nomupay", "noon", + "nordea", "novalnet", "nuvei", "opayo", @@ -295,6 +296,7 @@ nexixpay.base_url = "https://xpaysandbox.nexigroup.com/api/phoenix-0.0/psp/api/v nmi.base_url = "https://secure.nmi.com/" nomupay.base_url = "https://payout-api.sandbox.nomupay.com" noon.base_url = "https://api-test.noonpayments.com/" +nordea.base_url = "https://api.nordeaopenbanking.com" novalnet.base_url = "https://payport.novalnet.de/v2" noon.key_mode = "Test" nuvei.base_url = "https://ppp-test.nuvei.com/" diff --git a/config/docker_compose.toml b/config/docker_compose.toml index 08c32cb894..855bed5ebd 100644 --- a/config/docker_compose.toml +++ b/config/docker_compose.toml @@ -176,6 +176,7 @@ nexixpay.base_url = "https://xpaysandbox.nexigroup.com/api/phoenix-0.0/psp/api/v nmi.base_url = "https://secure.nmi.com/" nomupay.base_url = "https://payout-api.sandbox.nomupay.com" noon.base_url = "https://api-test.noonpayments.com/" +nordea.base_url = "https://api.nordeaopenbanking.com" novalnet.base_url = "https://payport.novalnet.de/v2" noon.key_mode = "Test" nuvei.base_url = "https://ppp-test.nuvei.com/" @@ -288,6 +289,7 @@ cards = [ "nexixpay", "nmi", "noon", + "nordea", "novalnet", "nuvei", "opayo", diff --git a/crates/common_enums/src/connector_enums.rs b/crates/common_enums/src/connector_enums.rs index 76979d26d6..0b3d03e6a9 100644 --- a/crates/common_enums/src/connector_enums.rs +++ b/crates/common_enums/src/connector_enums.rs @@ -111,6 +111,7 @@ pub enum RoutableConnectors { Nmi, Nomupay, Noon, + // Nordea, Novalnet, Nuvei, // Opayo, added as template code for future usage @@ -267,6 +268,7 @@ pub enum Connector { Nmi, Nomupay, Noon, + // Nordea, Novalnet, Nuvei, // Opayo, added as template code for future usage @@ -432,6 +434,7 @@ impl Connector { | Self::Nexinets | Self::Nexixpay | Self::Nomupay + // | Self::Nordea | Self::Novalnet | Self::Nuvei | Self::Opennode @@ -586,6 +589,7 @@ impl From for Connector { RoutableConnectors::Nmi => Self::Nmi, RoutableConnectors::Nomupay => Self::Nomupay, RoutableConnectors::Noon => Self::Noon, + // RoutableConnectors::Nordea => Self::Nordea, RoutableConnectors::Novalnet => Self::Novalnet, RoutableConnectors::Nuvei => Self::Nuvei, RoutableConnectors::Opennode => Self::Opennode, @@ -699,6 +703,7 @@ impl TryFrom for RoutableConnectors { Connector::Nmi => Ok(Self::Nmi), Connector::Nomupay => Ok(Self::Nomupay), Connector::Noon => Ok(Self::Noon), + // Connector::Nordea => Ok(Self::Nordea), Connector::Novalnet => Ok(Self::Novalnet), Connector::Nuvei => Ok(Self::Nuvei), Connector::Opennode => Ok(Self::Opennode), diff --git a/crates/connector_configs/src/connector.rs b/crates/connector_configs/src/connector.rs index bfa63d5939..45a2648911 100644 --- a/crates/connector_configs/src/connector.rs +++ b/crates/connector_configs/src/connector.rs @@ -224,6 +224,7 @@ pub struct ConnectorConfig { pub nmi: Option, pub nomupay_payout: Option, pub noon: Option, + pub nordea: Option, pub novalnet: Option, pub nuvei: Option, pub paybox: Option, diff --git a/crates/hyperswitch_connectors/src/connectors.rs b/crates/hyperswitch_connectors/src/connectors.rs index b2f6bb1e70..a635800152 100644 --- a/crates/hyperswitch_connectors/src/connectors.rs +++ b/crates/hyperswitch_connectors/src/connectors.rs @@ -56,6 +56,7 @@ pub mod nexixpay; pub mod nmi; pub mod nomupay; pub mod noon; +pub mod nordea; pub mod novalnet; pub mod nuvei; pub mod opayo; @@ -114,10 +115,10 @@ pub use self::{ itaubank::Itaubank, jpmorgan::Jpmorgan, juspaythreedsserver::Juspaythreedsserver, klarna::Klarna, mifinity::Mifinity, mollie::Mollie, moneris::Moneris, multisafepay::Multisafepay, netcetera::Netcetera, nexinets::Nexinets, nexixpay::Nexixpay, - nmi::Nmi, nomupay::Nomupay, noon::Noon, novalnet::Novalnet, nuvei::Nuvei, opayo::Opayo, - opennode::Opennode, paybox::Paybox, payeezy::Payeezy, payme::Payme, payone::Payone, - paypal::Paypal, paystack::Paystack, payu::Payu, placetopay::Placetopay, plaid::Plaid, - powertranz::Powertranz, prophetpay::Prophetpay, rapyd::Rapyd, razorpay::Razorpay, + nmi::Nmi, nomupay::Nomupay, noon::Noon, nordea::Nordea, novalnet::Novalnet, nuvei::Nuvei, + opayo::Opayo, opennode::Opennode, paybox::Paybox, payeezy::Payeezy, payme::Payme, + payone::Payone, paypal::Paypal, paystack::Paystack, payu::Payu, placetopay::Placetopay, + plaid::Plaid, powertranz::Powertranz, prophetpay::Prophetpay, rapyd::Rapyd, razorpay::Razorpay, recurly::Recurly, redsys::Redsys, riskified::Riskified, shift4::Shift4, signifyd::Signifyd, square::Square, stax::Stax, stripe::Stripe, stripebilling::Stripebilling, taxjar::Taxjar, threedsecureio::Threedsecureio, thunes::Thunes, trustpay::Trustpay, tsys::Tsys, diff --git a/crates/hyperswitch_connectors/src/connectors/nordea.rs b/crates/hyperswitch_connectors/src/connectors/nordea.rs new file mode 100644 index 0000000000..9230d856f8 --- /dev/null +++ b/crates/hyperswitch_connectors/src/connectors/nordea.rs @@ -0,0 +1,568 @@ +pub mod transformers; + +use common_utils::{ + errors::CustomResult, + ext_traits::BytesExt, + request::{Method, Request, RequestBuilder, RequestContent}, + types::{AmountConvertor, StringMinorUnit, StringMinorUnitForConnector}, +}; +use error_stack::{report, ResultExt}; +use hyperswitch_domain_models::{ + router_data::{AccessToken, ConnectorAuthType, ErrorResponse, RouterData}, + router_flow_types::{ + access_token_auth::AccessTokenAuth, + payments::{Authorize, Capture, PSync, PaymentMethodToken, Session, SetupMandate, Void}, + refunds::{Execute, RSync}, + }, + router_request_types::{ + AccessTokenRequestData, PaymentMethodTokenizationData, PaymentsAuthorizeData, + PaymentsCancelData, PaymentsCaptureData, PaymentsSessionData, PaymentsSyncData, + RefundsData, SetupMandateRequestData, + }, + router_response_types::{PaymentsResponseData, RefundsResponseData}, + types::{ + PaymentsAuthorizeRouterData, PaymentsCaptureRouterData, PaymentsSyncRouterData, + RefundSyncRouterData, RefundsRouterData, + }, +}; +use hyperswitch_interfaces::{ + api::{ + self, ConnectorCommon, ConnectorCommonExt, ConnectorIntegration, ConnectorSpecifications, + ConnectorValidation, + }, + configs::Connectors, + errors, + events::connector_api_logs::ConnectorEvent, + types::{self, Response}, + webhooks, +}; +use masking::{ExposeInterface, Mask}; +use transformers as nordea; + +use crate::{constants::headers, types::ResponseRouterData, utils}; + +#[derive(Clone)] +pub struct Nordea { + amount_converter: &'static (dyn AmountConvertor + Sync), +} + +impl Nordea { + pub fn new() -> &'static Self { + &Self { + amount_converter: &StringMinorUnitForConnector, + } + } +} + +impl api::Payment for Nordea {} +impl api::PaymentSession for Nordea {} +impl api::ConnectorAccessToken for Nordea {} +impl api::MandateSetup for Nordea {} +impl api::PaymentAuthorize for Nordea {} +impl api::PaymentSync for Nordea {} +impl api::PaymentCapture for Nordea {} +impl api::PaymentVoid for Nordea {} +impl api::Refund for Nordea {} +impl api::RefundExecute for Nordea {} +impl api::RefundSync for Nordea {} +impl api::PaymentToken for Nordea {} + +impl ConnectorIntegration + for Nordea +{ + // Not Implemented (R) +} + +impl ConnectorCommonExt for Nordea +where + Self: ConnectorIntegration, +{ + fn build_headers( + &self, + req: &RouterData, + _connectors: &Connectors, + ) -> CustomResult)>, errors::ConnectorError> { + let mut header = vec![( + headers::CONTENT_TYPE.to_string(), + self.get_content_type().to_string().into(), + )]; + let mut api_key = self.get_auth_header(&req.connector_auth_type)?; + header.append(&mut api_key); + Ok(header) + } +} + +impl ConnectorCommon for Nordea { + fn id(&self) -> &'static str { + "nordea" + } + + fn get_currency_unit(&self) -> api::CurrencyUnit { + api::CurrencyUnit::Base + } + + fn common_get_content_type(&self) -> &'static str { + "application/json" + } + + fn base_url<'a>(&self, connectors: &'a Connectors) -> &'a str { + connectors.nordea.base_url.as_ref() + } + + fn get_auth_header( + &self, + auth_type: &ConnectorAuthType, + ) -> CustomResult)>, errors::ConnectorError> { + let auth = nordea::NordeaAuthType::try_from(auth_type) + .change_context(errors::ConnectorError::FailedToObtainAuthType)?; + Ok(vec![( + headers::AUTHORIZATION.to_string(), + auth.api_key.expose().into_masked(), + )]) + } + + fn build_error_response( + &self, + res: Response, + event_builder: Option<&mut ConnectorEvent>, + ) -> CustomResult { + let response: nordea::NordeaErrorResponse = res + .response + .parse_struct("NordeaErrorResponse") + .change_context(errors::ConnectorError::ResponseDeserializationFailed)?; + + event_builder.map(|i| i.set_response_body(&response)); + router_env::logger::info!(connector_response=?response); + + Ok(ErrorResponse { + status_code: res.status_code, + code: response.code, + message: response.message, + reason: response.reason, + attempt_status: None, + connector_transaction_id: None, + network_decline_code: None, + network_advice_code: None, + network_error_message: None, + }) + } +} + +impl ConnectorValidation for Nordea { + //TODO: implement functions when support enabled +} + +impl ConnectorIntegration for Nordea { + //TODO: implement sessions flow +} + +impl ConnectorIntegration for Nordea {} + +impl ConnectorIntegration for Nordea {} + +impl ConnectorIntegration for Nordea { + fn get_headers( + &self, + req: &PaymentsAuthorizeRouterData, + connectors: &Connectors, + ) -> CustomResult)>, errors::ConnectorError> { + self.build_headers(req, connectors) + } + + fn get_content_type(&self) -> &'static str { + self.common_get_content_type() + } + + fn get_url( + &self, + _req: &PaymentsAuthorizeRouterData, + _connectors: &Connectors, + ) -> CustomResult { + Err(errors::ConnectorError::NotImplemented("get_url method".to_string()).into()) + } + + fn get_request_body( + &self, + req: &PaymentsAuthorizeRouterData, + _connectors: &Connectors, + ) -> CustomResult { + let amount = utils::convert_amount( + self.amount_converter, + req.request.minor_amount, + req.request.currency, + )?; + + let connector_router_data = nordea::NordeaRouterData::from((amount, req)); + let connector_req = nordea::NordeaPaymentsRequest::try_from(&connector_router_data)?; + Ok(RequestContent::Json(Box::new(connector_req))) + } + + fn build_request( + &self, + req: &PaymentsAuthorizeRouterData, + connectors: &Connectors, + ) -> CustomResult, errors::ConnectorError> { + Ok(Some( + RequestBuilder::new() + .method(Method::Post) + .url(&types::PaymentsAuthorizeType::get_url( + self, req, connectors, + )?) + .attach_default_headers() + .headers(types::PaymentsAuthorizeType::get_headers( + self, req, connectors, + )?) + .set_body(types::PaymentsAuthorizeType::get_request_body( + self, req, connectors, + )?) + .build(), + )) + } + + fn handle_response( + &self, + data: &PaymentsAuthorizeRouterData, + event_builder: Option<&mut ConnectorEvent>, + res: Response, + ) -> CustomResult { + let response: nordea::NordeaPaymentsResponse = res + .response + .parse_struct("Nordea PaymentsAuthorizeResponse") + .change_context(errors::ConnectorError::ResponseDeserializationFailed)?; + event_builder.map(|i| i.set_response_body(&response)); + router_env::logger::info!(connector_response=?response); + RouterData::try_from(ResponseRouterData { + response, + data: data.clone(), + http_code: res.status_code, + }) + } + + fn get_error_response( + &self, + res: Response, + event_builder: Option<&mut ConnectorEvent>, + ) -> CustomResult { + self.build_error_response(res, event_builder) + } +} + +impl ConnectorIntegration for Nordea { + fn get_headers( + &self, + req: &PaymentsSyncRouterData, + connectors: &Connectors, + ) -> CustomResult)>, errors::ConnectorError> { + self.build_headers(req, connectors) + } + + fn get_content_type(&self) -> &'static str { + self.common_get_content_type() + } + + fn get_url( + &self, + _req: &PaymentsSyncRouterData, + _connectors: &Connectors, + ) -> CustomResult { + Err(errors::ConnectorError::NotImplemented("get_url method".to_string()).into()) + } + + fn build_request( + &self, + req: &PaymentsSyncRouterData, + connectors: &Connectors, + ) -> CustomResult, errors::ConnectorError> { + Ok(Some( + RequestBuilder::new() + .method(Method::Get) + .url(&types::PaymentsSyncType::get_url(self, req, connectors)?) + .attach_default_headers() + .headers(types::PaymentsSyncType::get_headers(self, req, connectors)?) + .build(), + )) + } + + fn handle_response( + &self, + data: &PaymentsSyncRouterData, + event_builder: Option<&mut ConnectorEvent>, + res: Response, + ) -> CustomResult { + let response: nordea::NordeaPaymentsResponse = res + .response + .parse_struct("nordea PaymentsSyncResponse") + .change_context(errors::ConnectorError::ResponseDeserializationFailed)?; + event_builder.map(|i| i.set_response_body(&response)); + router_env::logger::info!(connector_response=?response); + RouterData::try_from(ResponseRouterData { + response, + data: data.clone(), + http_code: res.status_code, + }) + } + + fn get_error_response( + &self, + res: Response, + event_builder: Option<&mut ConnectorEvent>, + ) -> CustomResult { + self.build_error_response(res, event_builder) + } +} + +impl ConnectorIntegration for Nordea { + fn get_headers( + &self, + req: &PaymentsCaptureRouterData, + connectors: &Connectors, + ) -> CustomResult)>, errors::ConnectorError> { + self.build_headers(req, connectors) + } + + fn get_content_type(&self) -> &'static str { + self.common_get_content_type() + } + + fn get_url( + &self, + _req: &PaymentsCaptureRouterData, + _connectors: &Connectors, + ) -> CustomResult { + Err(errors::ConnectorError::NotImplemented("get_url method".to_string()).into()) + } + + fn get_request_body( + &self, + _req: &PaymentsCaptureRouterData, + _connectors: &Connectors, + ) -> CustomResult { + Err(errors::ConnectorError::NotImplemented("get_request_body method".to_string()).into()) + } + + fn build_request( + &self, + req: &PaymentsCaptureRouterData, + connectors: &Connectors, + ) -> CustomResult, errors::ConnectorError> { + Ok(Some( + RequestBuilder::new() + .method(Method::Post) + .url(&types::PaymentsCaptureType::get_url(self, req, connectors)?) + .attach_default_headers() + .headers(types::PaymentsCaptureType::get_headers( + self, req, connectors, + )?) + .set_body(types::PaymentsCaptureType::get_request_body( + self, req, connectors, + )?) + .build(), + )) + } + + fn handle_response( + &self, + data: &PaymentsCaptureRouterData, + event_builder: Option<&mut ConnectorEvent>, + res: Response, + ) -> CustomResult { + let response: nordea::NordeaPaymentsResponse = res + .response + .parse_struct("Nordea PaymentsCaptureResponse") + .change_context(errors::ConnectorError::ResponseDeserializationFailed)?; + event_builder.map(|i| i.set_response_body(&response)); + router_env::logger::info!(connector_response=?response); + RouterData::try_from(ResponseRouterData { + response, + data: data.clone(), + http_code: res.status_code, + }) + } + + fn get_error_response( + &self, + res: Response, + event_builder: Option<&mut ConnectorEvent>, + ) -> CustomResult { + self.build_error_response(res, event_builder) + } +} + +impl ConnectorIntegration for Nordea {} + +impl ConnectorIntegration for Nordea { + fn get_headers( + &self, + req: &RefundsRouterData, + connectors: &Connectors, + ) -> CustomResult)>, errors::ConnectorError> { + self.build_headers(req, connectors) + } + + fn get_content_type(&self) -> &'static str { + self.common_get_content_type() + } + + fn get_url( + &self, + _req: &RefundsRouterData, + _connectors: &Connectors, + ) -> CustomResult { + Err(errors::ConnectorError::NotImplemented("get_url method".to_string()).into()) + } + + fn get_request_body( + &self, + req: &RefundsRouterData, + _connectors: &Connectors, + ) -> CustomResult { + let refund_amount = utils::convert_amount( + self.amount_converter, + req.request.minor_refund_amount, + req.request.currency, + )?; + + let connector_router_data = nordea::NordeaRouterData::from((refund_amount, req)); + let connector_req = nordea::NordeaRefundRequest::try_from(&connector_router_data)?; + Ok(RequestContent::Json(Box::new(connector_req))) + } + + fn build_request( + &self, + req: &RefundsRouterData, + connectors: &Connectors, + ) -> CustomResult, errors::ConnectorError> { + let request = RequestBuilder::new() + .method(Method::Post) + .url(&types::RefundExecuteType::get_url(self, req, connectors)?) + .attach_default_headers() + .headers(types::RefundExecuteType::get_headers( + self, req, connectors, + )?) + .set_body(types::RefundExecuteType::get_request_body( + self, req, connectors, + )?) + .build(); + Ok(Some(request)) + } + + fn handle_response( + &self, + data: &RefundsRouterData, + event_builder: Option<&mut ConnectorEvent>, + res: Response, + ) -> CustomResult, errors::ConnectorError> { + let response: nordea::RefundResponse = + res.response + .parse_struct("nordea RefundResponse") + .change_context(errors::ConnectorError::ResponseDeserializationFailed)?; + event_builder.map(|i| i.set_response_body(&response)); + router_env::logger::info!(connector_response=?response); + RouterData::try_from(ResponseRouterData { + response, + data: data.clone(), + http_code: res.status_code, + }) + } + + fn get_error_response( + &self, + res: Response, + event_builder: Option<&mut ConnectorEvent>, + ) -> CustomResult { + self.build_error_response(res, event_builder) + } +} + +impl ConnectorIntegration for Nordea { + fn get_headers( + &self, + req: &RefundSyncRouterData, + connectors: &Connectors, + ) -> CustomResult)>, errors::ConnectorError> { + self.build_headers(req, connectors) + } + + fn get_content_type(&self) -> &'static str { + self.common_get_content_type() + } + + fn get_url( + &self, + _req: &RefundSyncRouterData, + _connectors: &Connectors, + ) -> CustomResult { + Err(errors::ConnectorError::NotImplemented("get_url method".to_string()).into()) + } + + fn build_request( + &self, + req: &RefundSyncRouterData, + connectors: &Connectors, + ) -> CustomResult, errors::ConnectorError> { + Ok(Some( + RequestBuilder::new() + .method(Method::Get) + .url(&types::RefundSyncType::get_url(self, req, connectors)?) + .attach_default_headers() + .headers(types::RefundSyncType::get_headers(self, req, connectors)?) + .set_body(types::RefundSyncType::get_request_body( + self, req, connectors, + )?) + .build(), + )) + } + + fn handle_response( + &self, + data: &RefundSyncRouterData, + event_builder: Option<&mut ConnectorEvent>, + res: Response, + ) -> CustomResult { + let response: nordea::RefundResponse = res + .response + .parse_struct("nordea RefundSyncResponse") + .change_context(errors::ConnectorError::ResponseDeserializationFailed)?; + event_builder.map(|i| i.set_response_body(&response)); + router_env::logger::info!(connector_response=?response); + RouterData::try_from(ResponseRouterData { + response, + data: data.clone(), + http_code: res.status_code, + }) + } + + fn get_error_response( + &self, + res: Response, + event_builder: Option<&mut ConnectorEvent>, + ) -> CustomResult { + self.build_error_response(res, event_builder) + } +} + +#[async_trait::async_trait] +impl webhooks::IncomingWebhook for Nordea { + fn get_webhook_object_reference_id( + &self, + _request: &webhooks::IncomingWebhookRequestDetails<'_>, + ) -> CustomResult { + Err(report!(errors::ConnectorError::WebhooksNotImplemented)) + } + + fn get_webhook_event_type( + &self, + _request: &webhooks::IncomingWebhookRequestDetails<'_>, + ) -> CustomResult { + Err(report!(errors::ConnectorError::WebhooksNotImplemented)) + } + + fn get_webhook_resource_object( + &self, + _request: &webhooks::IncomingWebhookRequestDetails<'_>, + ) -> CustomResult, errors::ConnectorError> { + Err(report!(errors::ConnectorError::WebhooksNotImplemented)) + } +} + +impl ConnectorSpecifications for Nordea {} diff --git a/crates/hyperswitch_connectors/src/connectors/nordea/transformers.rs b/crates/hyperswitch_connectors/src/connectors/nordea/transformers.rs new file mode 100644 index 0000000000..bea06d53e3 --- /dev/null +++ b/crates/hyperswitch_connectors/src/connectors/nordea/transformers.rs @@ -0,0 +1,228 @@ +use common_enums::enums; +use common_utils::types::StringMinorUnit; +use hyperswitch_domain_models::{ + payment_method_data::PaymentMethodData, + router_data::{ConnectorAuthType, RouterData}, + router_flow_types::refunds::{Execute, RSync}, + router_request_types::ResponseId, + router_response_types::{PaymentsResponseData, RefundsResponseData}, + types::{PaymentsAuthorizeRouterData, RefundsRouterData}, +}; +use hyperswitch_interfaces::errors; +use masking::Secret; +use serde::{Deserialize, Serialize}; + +use crate::{ + types::{RefundsResponseRouterData, ResponseRouterData}, + utils::PaymentsAuthorizeRequestData, +}; + +//TODO: Fill the struct with respective fields +pub struct NordeaRouterData { + pub amount: StringMinorUnit, // The type of amount that a connector accepts, for example, String, i64, f64, etc. + pub router_data: T, +} + +impl From<(StringMinorUnit, T)> for NordeaRouterData { + fn from((amount, item): (StringMinorUnit, T)) -> Self { + //Todo : use utils to convert the amount to the type of amount that a connector accepts + Self { + amount, + router_data: item, + } + } +} + +//TODO: Fill the struct with respective fields +#[derive(Default, Debug, Serialize, PartialEq)] +pub struct NordeaPaymentsRequest { + amount: StringMinorUnit, + card: NordeaCard, +} + +#[derive(Default, Debug, Serialize, Eq, PartialEq)] +pub struct NordeaCard { + number: cards::CardNumber, + expiry_month: Secret, + expiry_year: Secret, + cvc: Secret, + complete: bool, +} + +impl TryFrom<&NordeaRouterData<&PaymentsAuthorizeRouterData>> for NordeaPaymentsRequest { + type Error = error_stack::Report; + fn try_from( + item: &NordeaRouterData<&PaymentsAuthorizeRouterData>, + ) -> Result { + match item.router_data.request.payment_method_data.clone() { + PaymentMethodData::Card(req_card) => { + let card = NordeaCard { + number: req_card.card_number, + expiry_month: req_card.card_exp_month, + expiry_year: req_card.card_exp_year, + cvc: req_card.card_cvc, + complete: item.router_data.request.is_auto_capture()?, + }; + Ok(Self { + amount: item.amount.clone(), + card, + }) + } + _ => Err(errors::ConnectorError::NotImplemented("Payment method".to_string()).into()), + } + } +} + +//TODO: Fill the struct with respective fields +// Auth Struct +pub struct NordeaAuthType { + pub(super) api_key: Secret, +} + +impl TryFrom<&ConnectorAuthType> for NordeaAuthType { + type Error = error_stack::Report; + fn try_from(auth_type: &ConnectorAuthType) -> Result { + match auth_type { + ConnectorAuthType::HeaderKey { api_key } => Ok(Self { + api_key: api_key.to_owned(), + }), + _ => Err(errors::ConnectorError::FailedToObtainAuthType.into()), + } + } +} +// PaymentsResponse +//TODO: Append the remaining status flags +#[derive(Debug, Clone, Default, Serialize, Deserialize, PartialEq)] +#[serde(rename_all = "lowercase")] +pub enum NordeaPaymentStatus { + Succeeded, + Failed, + #[default] + Processing, +} + +impl From for common_enums::AttemptStatus { + fn from(item: NordeaPaymentStatus) -> Self { + match item { + NordeaPaymentStatus::Succeeded => Self::Charged, + NordeaPaymentStatus::Failed => Self::Failure, + NordeaPaymentStatus::Processing => Self::Authorizing, + } + } +} + +//TODO: Fill the struct with respective fields +#[derive(Default, Debug, Clone, Serialize, Deserialize, PartialEq)] +pub struct NordeaPaymentsResponse { + status: NordeaPaymentStatus, + id: String, +} + +impl TryFrom> + for RouterData +{ + type Error = error_stack::Report; + fn try_from( + item: ResponseRouterData, + ) -> Result { + Ok(Self { + status: common_enums::AttemptStatus::from(item.response.status), + response: Ok(PaymentsResponseData::TransactionResponse { + resource_id: ResponseId::ConnectorTransactionId(item.response.id), + redirection_data: Box::new(None), + mandate_reference: Box::new(None), + connector_metadata: None, + network_txn_id: None, + connector_response_reference_id: None, + incremental_authorization_allowed: None, + charges: None, + }), + ..item.data + }) + } +} + +//TODO: Fill the struct with respective fields +// REFUND : +// Type definition for RefundRequest +#[derive(Default, Debug, Serialize)] +pub struct NordeaRefundRequest { + pub amount: StringMinorUnit, +} + +impl TryFrom<&NordeaRouterData<&RefundsRouterData>> for NordeaRefundRequest { + type Error = error_stack::Report; + fn try_from(item: &NordeaRouterData<&RefundsRouterData>) -> Result { + Ok(Self { + amount: item.amount.to_owned(), + }) + } +} + +// Type definition for Refund Response + +#[allow(dead_code)] +#[derive(Debug, Serialize, Default, Deserialize, Clone)] +pub enum RefundStatus { + Succeeded, + Failed, + #[default] + Processing, +} + +impl From for enums::RefundStatus { + fn from(item: RefundStatus) -> Self { + match item { + RefundStatus::Succeeded => Self::Success, + RefundStatus::Failed => Self::Failure, + RefundStatus::Processing => Self::Pending, + //TODO: Review mapping + } + } +} + +//TODO: Fill the struct with respective fields +#[derive(Default, Debug, Clone, Serialize, Deserialize)] +pub struct RefundResponse { + id: String, + status: RefundStatus, +} + +impl TryFrom> for RefundsRouterData { + type Error = error_stack::Report; + fn try_from( + item: RefundsResponseRouterData, + ) -> Result { + Ok(Self { + response: Ok(RefundsResponseData { + connector_refund_id: item.response.id.to_string(), + refund_status: enums::RefundStatus::from(item.response.status), + }), + ..item.data + }) + } +} + +impl TryFrom> for RefundsRouterData { + type Error = error_stack::Report; + fn try_from( + item: RefundsResponseRouterData, + ) -> Result { + Ok(Self { + response: Ok(RefundsResponseData { + connector_refund_id: item.response.id.to_string(), + refund_status: enums::RefundStatus::from(item.response.status), + }), + ..item.data + }) + } +} + +//TODO: Fill the struct with respective fields +#[derive(Default, Debug, Serialize, Deserialize, PartialEq)] +pub struct NordeaErrorResponse { + pub status_code: u16, + pub code: String, + pub message: String, + pub reason: Option, +} diff --git a/crates/hyperswitch_connectors/src/default_implementations.rs b/crates/hyperswitch_connectors/src/default_implementations.rs index ed5f0135a6..f4689525cc 100644 --- a/crates/hyperswitch_connectors/src/default_implementations.rs +++ b/crates/hyperswitch_connectors/src/default_implementations.rs @@ -168,6 +168,7 @@ default_imp_for_authorize_session_token!( connectors::Nmi, connectors::Nomupay, connectors::Noon, + connectors::Nordea, connectors::Novalnet, connectors::Nexinets, connectors::Nexixpay, @@ -293,6 +294,7 @@ default_imp_for_calculate_tax!( connectors::Nmi, connectors::Nomupay, connectors::Noon, + connectors::Nordea, connectors::Novalnet, connectors::Nuvei, connectors::Payeezy, @@ -410,6 +412,7 @@ default_imp_for_session_update!( connectors::Nmi, connectors::Nomupay, connectors::Noon, + connectors::Nordea, connectors::Novalnet, connectors::Nexinets, connectors::Nexixpay, @@ -527,6 +530,7 @@ default_imp_for_post_session_tokens!( connectors::Nmi, connectors::Nomupay, connectors::Noon, + connectors::Nordea, connectors::Novalnet, connectors::Nexinets, connectors::Nexixpay, @@ -641,6 +645,7 @@ default_imp_for_update_metadata!( connectors::Netcetera, connectors::Nomupay, connectors::Noon, + connectors::Nordea, connectors::Novalnet, connectors::Nexinets, connectors::Nexixpay, @@ -747,6 +752,7 @@ default_imp_for_complete_authorize!( connectors::Netcetera, connectors::Nomupay, connectors::Noon, + connectors::Nordea, connectors::Novalnet, connectors::Nexinets, connectors::Opayo, @@ -850,6 +856,7 @@ default_imp_for_incremental_authorization!( connectors::Nmi, connectors::Nomupay, connectors::Noon, + connectors::Nordea, connectors::Novalnet, connectors::Nexinets, connectors::Nexixpay, @@ -970,6 +977,7 @@ default_imp_for_create_customer!( connectors::Nmi, connectors::Nomupay, connectors::Noon, + connectors::Nordea, connectors::Novalnet, connectors::Nexinets, connectors::Nexixpay, @@ -1079,6 +1087,7 @@ default_imp_for_connector_redirect_response!( connectors::Netcetera, connectors::Nexinets, connectors::Nexixpay, + connectors::Nordea, connectors::Opayo, connectors::Opennode, connectors::Payeezy, @@ -1177,6 +1186,7 @@ default_imp_for_pre_processing_steps!( connectors::Netcetera, connectors::Nomupay, connectors::Noon, + connectors::Nordea, connectors::Novalnet, connectors::Nexinets, connectors::Opayo, @@ -1288,6 +1298,7 @@ default_imp_for_post_processing_steps!( connectors::Nmi, connectors::Nomupay, connectors::Noon, + connectors::Nordea, connectors::Novalnet, connectors::Nexinets, connectors::Nexixpay, @@ -1405,6 +1416,7 @@ default_imp_for_approve!( connectors::Netcetera, connectors::Nomupay, connectors::Noon, + connectors::Nordea, connectors::Novalnet, connectors::Nexinets, connectors::Nexixpay, @@ -1525,6 +1537,7 @@ default_imp_for_reject!( connectors::Nmi, connectors::Nomupay, connectors::Noon, + connectors::Nordea, connectors::Novalnet, connectors::Nexinets, connectors::Nexixpay, @@ -1644,6 +1657,7 @@ default_imp_for_webhook_source_verification!( connectors::Nmi, connectors::Nomupay, connectors::Noon, + connectors::Nordea, connectors::Novalnet, connectors::Nexinets, connectors::Nexixpay, @@ -1761,6 +1775,7 @@ default_imp_for_accept_dispute!( connectors::Nmi, connectors::Nomupay, connectors::Noon, + connectors::Nordea, connectors::Novalnet, connectors::Nexinets, connectors::Nexixpay, @@ -1878,6 +1893,7 @@ default_imp_for_submit_evidence!( connectors::Nmi, connectors::Nomupay, connectors::Noon, + connectors::Nordea, connectors::Novalnet, connectors::Nexinets, connectors::Nexixpay, @@ -1994,6 +2010,7 @@ default_imp_for_defend_dispute!( connectors::Nmi, connectors::Nomupay, connectors::Noon, + connectors::Nordea, connectors::Novalnet, connectors::Nexinets, connectors::Nexixpay, @@ -2120,6 +2137,7 @@ default_imp_for_file_upload!( connectors::Nmi, connectors::Nomupay, connectors::Noon, + connectors::Nordea, connectors::Novalnet, connectors::Nexinets, connectors::Nexixpay, @@ -2229,6 +2247,7 @@ default_imp_for_payouts!( connectors::Multisafepay, connectors::Nexinets, connectors::Nexixpay, + connectors::Nordea, connectors::Opayo, connectors::Opennode, connectors::Paybox, @@ -2342,6 +2361,7 @@ default_imp_for_payouts_create!( connectors::Novalnet, connectors::Nexinets, connectors::Nexixpay, + connectors::Nordea, connectors::Opayo, connectors::Opennode, connectors::Nuvei, @@ -2456,6 +2476,7 @@ default_imp_for_payouts_retrieve!( connectors::Netcetera, connectors::Nmi, connectors::Noon, + connectors::Nordea, connectors::Novalnet, connectors::Nexinets, connectors::Nexixpay, @@ -2573,6 +2594,7 @@ default_imp_for_payouts_eligibility!( connectors::Netcetera, connectors::Nmi, connectors::Noon, + connectors::Nordea, connectors::Novalnet, connectors::Nexinets, connectors::Nexixpay, @@ -2687,6 +2709,7 @@ default_imp_for_payouts_fulfill!( connectors::Klarna, connectors::Netcetera, connectors::Noon, + connectors::Nordea, connectors::Novalnet, connectors::Nexinets, connectors::Nexixpay, @@ -2802,6 +2825,7 @@ default_imp_for_payouts_cancel!( connectors::Netcetera, connectors::Nmi, connectors::Noon, + connectors::Nordea, connectors::Novalnet, connectors::Nexinets, connectors::Nexixpay, @@ -2919,6 +2943,7 @@ default_imp_for_payouts_quote!( connectors::Netcetera, connectors::Nmi, connectors::Noon, + connectors::Nordea, connectors::Novalnet, connectors::Nexinets, connectors::Nexixpay, @@ -3037,6 +3062,7 @@ default_imp_for_payouts_recipient!( connectors::Netcetera, connectors::Nmi, connectors::Noon, + connectors::Nordea, connectors::Novalnet, connectors::Nexinets, connectors::Nexixpay, @@ -3155,6 +3181,7 @@ default_imp_for_payouts_recipient_account!( connectors::Netcetera, connectors::Nmi, connectors::Noon, + connectors::Nordea, connectors::Novalnet, connectors::Nexinets, connectors::Nexixpay, @@ -3275,6 +3302,7 @@ default_imp_for_frm_sale!( connectors::Nomupay, connectors::Nmi, connectors::Noon, + connectors::Nordea, connectors::Novalnet, connectors::Nexinets, connectors::Nexixpay, @@ -3394,6 +3422,7 @@ default_imp_for_frm_checkout!( connectors::Nmi, connectors::Nomupay, connectors::Noon, + connectors::Nordea, connectors::Novalnet, connectors::Nexinets, connectors::Nexixpay, @@ -3513,6 +3542,7 @@ default_imp_for_frm_transaction!( connectors::Nmi, connectors::Nomupay, connectors::Noon, + connectors::Nordea, connectors::Novalnet, connectors::Nexinets, connectors::Nexixpay, @@ -3632,6 +3662,7 @@ default_imp_for_frm_fulfillment!( connectors::Nmi, connectors::Nomupay, connectors::Noon, + connectors::Nordea, connectors::Novalnet, connectors::Nexinets, connectors::Nexixpay, @@ -3751,6 +3782,7 @@ default_imp_for_frm_record_return!( connectors::Nmi, connectors::Nomupay, connectors::Noon, + connectors::Nordea, connectors::Novalnet, connectors::Nexinets, connectors::Nexixpay, @@ -3864,6 +3896,7 @@ default_imp_for_revoking_mandates!( connectors::Netcetera, connectors::Nmi, connectors::Nomupay, + connectors::Nordea, connectors::Novalnet, connectors::Nexinets, connectors::Nexixpay, @@ -3981,6 +4014,7 @@ default_imp_for_uas_pre_authentication!( connectors::Nmi, connectors::Nomupay, connectors::Noon, + connectors::Nordea, connectors::Novalnet, connectors::Nexinets, connectors::Nexixpay, @@ -4097,6 +4131,7 @@ default_imp_for_uas_post_authentication!( connectors::Nmi, connectors::Nomupay, connectors::Noon, + connectors::Nordea, connectors::Novalnet, connectors::Nexinets, connectors::Nexixpay, @@ -4214,6 +4249,7 @@ default_imp_for_uas_authentication_confirmation!( connectors::Nmi, connectors::Nomupay, connectors::Noon, + connectors::Nordea, connectors::Novalnet, connectors::Nexinets, connectors::Nexixpay, @@ -4322,6 +4358,7 @@ default_imp_for_connector_request_id!( connectors::Netcetera, connectors::Nmi, connectors::Nomupay, + connectors::Nordea, connectors::Novalnet, connectors::Noon, connectors::Nexixpay, @@ -4434,6 +4471,7 @@ default_imp_for_fraud_check!( connectors::Netcetera, connectors::Nmi, connectors::Nomupay, + connectors::Nordea, connectors::Novalnet, connectors::Noon, connectors::Nexinets, @@ -4574,6 +4612,7 @@ default_imp_for_connector_authentication!( connectors::Nmi, connectors::Nomupay, connectors::Noon, + connectors::Nordea, connectors::Novalnet, connectors::Nexinets, connectors::Nexixpay, @@ -4688,6 +4727,7 @@ default_imp_for_uas_authentication!( connectors::Netcetera, connectors::Nmi, connectors::Nomupay, + connectors::Nordea, connectors::Novalnet, connectors::Nexinets, connectors::Nexixpay, @@ -4799,6 +4839,7 @@ default_imp_for_revenue_recovery! { connectors::Nmi, connectors::Nomupay, connectors::Noon, + connectors::Nordea, connectors::Novalnet, connectors::Nexinets, connectors::Nexixpay, @@ -4919,6 +4960,7 @@ default_imp_for_billing_connector_payment_sync!( connectors::Netcetera, connectors::Nmi, connectors::Noon, + connectors::Nordea, connectors::Novalnet, connectors::Nexinets, connectors::Nexixpay, @@ -5037,6 +5079,7 @@ default_imp_for_revenue_recovery_record_back!( connectors::Nmi, connectors::Nomupay, connectors::Noon, + connectors::Nordea, connectors::Novalnet, connectors::Nexinets, connectors::Nexixpay, @@ -5154,6 +5197,7 @@ default_imp_for_billing_connector_invoice_sync!( connectors::Nomupay, connectors::Nmi, connectors::Noon, + connectors::Nordea, connectors::Novalnet, connectors::Netcetera, connectors::Nexinets, diff --git a/crates/hyperswitch_connectors/src/default_implementations_v2.rs b/crates/hyperswitch_connectors/src/default_implementations_v2.rs index 9148bdb113..3188126f4c 100644 --- a/crates/hyperswitch_connectors/src/default_implementations_v2.rs +++ b/crates/hyperswitch_connectors/src/default_implementations_v2.rs @@ -288,6 +288,7 @@ default_imp_for_new_connector_integration_payment!( connectors::Klarna, connectors::Nomupay, connectors::Noon, + connectors::Nordea, connectors::Novalnet, connectors::Netcetera, connectors::Nexinets, @@ -407,6 +408,7 @@ default_imp_for_new_connector_integration_refund!( connectors::Klarna, connectors::Nomupay, connectors::Noon, + connectors::Nordea, connectors::Novalnet, connectors::Netcetera, connectors::Nexinets, @@ -521,6 +523,7 @@ default_imp_for_new_connector_integration_connector_access_token!( connectors::Klarna, connectors::Nomupay, connectors::Noon, + connectors::Nordea, connectors::Novalnet, connectors::Netcetera, connectors::Nexinets, @@ -640,6 +643,7 @@ default_imp_for_new_connector_integration_accept_dispute!( connectors::Klarna, connectors::Nomupay, connectors::Noon, + connectors::Nordea, connectors::Novalnet, connectors::Netcetera, connectors::Nexinets, @@ -758,6 +762,7 @@ default_imp_for_new_connector_integration_submit_evidence!( connectors::Klarna, connectors::Nomupay, connectors::Noon, + connectors::Nordea, connectors::Novalnet, connectors::Netcetera, connectors::Nexinets, @@ -877,6 +882,7 @@ default_imp_for_new_connector_integration_defend_dispute!( connectors::Klarna, connectors::Nomupay, connectors::Noon, + connectors::Nordea, connectors::Novalnet, connectors::Netcetera, connectors::Nexinets, @@ -1006,6 +1012,7 @@ default_imp_for_new_connector_integration_file_upload!( connectors::Klarna, connectors::Nomupay, connectors::Noon, + connectors::Nordea, connectors::Novalnet, connectors::Netcetera, connectors::Nexinets, @@ -1127,6 +1134,7 @@ default_imp_for_new_connector_integration_payouts_create!( connectors::Klarna, connectors::Nomupay, connectors::Noon, + connectors::Nordea, connectors::Novalnet, connectors::Netcetera, connectors::Nexinets, @@ -1248,6 +1256,7 @@ default_imp_for_new_connector_integration_payouts_eligibility!( connectors::Klarna, connectors::Nomupay, connectors::Noon, + connectors::Nordea, connectors::Novalnet, connectors::Netcetera, connectors::Nexinets, @@ -1369,6 +1378,7 @@ default_imp_for_new_connector_integration_payouts_fulfill!( connectors::Klarna, connectors::Nomupay, connectors::Noon, + connectors::Nordea, connectors::Novalnet, connectors::Netcetera, connectors::Nexinets, @@ -1490,6 +1500,7 @@ default_imp_for_new_connector_integration_payouts_cancel!( connectors::Klarna, connectors::Nomupay, connectors::Noon, + connectors::Nordea, connectors::Novalnet, connectors::Netcetera, connectors::Nexinets, @@ -1611,6 +1622,7 @@ default_imp_for_new_connector_integration_payouts_quote!( connectors::Klarna, connectors::Nomupay, connectors::Noon, + connectors::Nordea, connectors::Novalnet, connectors::Netcetera, connectors::Nexinets, @@ -1732,6 +1744,7 @@ default_imp_for_new_connector_integration_payouts_recipient!( connectors::Klarna, connectors::Nomupay, connectors::Noon, + connectors::Nordea, connectors::Novalnet, connectors::Netcetera, connectors::Nexinets, @@ -1853,6 +1866,7 @@ default_imp_for_new_connector_integration_payouts_sync!( connectors::Klarna, connectors::Nomupay, connectors::Noon, + connectors::Nordea, connectors::Novalnet, connectors::Netcetera, connectors::Nexinets, @@ -1974,6 +1988,7 @@ default_imp_for_new_connector_integration_payouts_recipient_account!( connectors::Klarna, connectors::Nomupay, connectors::Noon, + connectors::Nordea, connectors::Novalnet, connectors::Netcetera, connectors::Nexinets, @@ -2093,6 +2108,7 @@ default_imp_for_new_connector_integration_webhook_source_verification!( connectors::Klarna, connectors::Nomupay, connectors::Noon, + connectors::Nordea, connectors::Novalnet, connectors::Netcetera, connectors::Nexinets, @@ -2214,6 +2230,7 @@ default_imp_for_new_connector_integration_frm_sale!( connectors::Klarna, connectors::Nomupay, connectors::Noon, + connectors::Nordea, connectors::Novalnet, connectors::Netcetera, connectors::Nexinets, @@ -2335,6 +2352,7 @@ default_imp_for_new_connector_integration_frm_checkout!( connectors::Klarna, connectors::Nomupay, connectors::Noon, + connectors::Nordea, connectors::Novalnet, connectors::Netcetera, connectors::Nexinets, @@ -2456,6 +2474,7 @@ default_imp_for_new_connector_integration_frm_transaction!( connectors::Klarna, connectors::Nomupay, connectors::Noon, + connectors::Nordea, connectors::Novalnet, connectors::Netcetera, connectors::Nexinets, @@ -2577,6 +2596,7 @@ default_imp_for_new_connector_integration_frm_fulfillment!( connectors::Klarna, connectors::Nomupay, connectors::Noon, + connectors::Nordea, connectors::Novalnet, connectors::Netcetera, connectors::Nexinets, @@ -2698,6 +2718,7 @@ default_imp_for_new_connector_integration_frm_record_return!( connectors::Klarna, connectors::Nomupay, connectors::Noon, + connectors::Nordea, connectors::Novalnet, connectors::Netcetera, connectors::Nexinets, @@ -2816,6 +2837,7 @@ default_imp_for_new_connector_integration_revoking_mandates!( connectors::Klarna, connectors::Nomupay, connectors::Noon, + connectors::Nordea, connectors::Novalnet, connectors::Netcetera, connectors::Nexinets, @@ -2912,6 +2934,7 @@ default_imp_for_new_connector_integration_frm!( connectors::Jpmorgan, connectors::Juspaythreedsserver, connectors::Nomupay, + connectors::Nordea, connectors::Novalnet, connectors::Netcetera, connectors::Nexinets, @@ -3030,6 +3053,7 @@ default_imp_for_new_connector_integration_connector_authentication!( connectors::Jpmorgan, connectors::Juspaythreedsserver, connectors::Nomupay, + connectors::Nordea, connectors::Novalnet, connectors::Netcetera, connectors::Nexinets, @@ -3137,6 +3161,7 @@ default_imp_for_new_connector_integration_revenue_recovery!( connectors::Jpmorgan, connectors::Juspaythreedsserver, connectors::Nomupay, + connectors::Nordea, connectors::Novalnet, connectors::Netcetera, connectors::Nexinets, diff --git a/crates/hyperswitch_domain_models/src/configs.rs b/crates/hyperswitch_domain_models/src/configs.rs index 7ce8b2cad0..bc06fbeb49 100644 --- a/crates/hyperswitch_domain_models/src/configs.rs +++ b/crates/hyperswitch_domain_models/src/configs.rs @@ -73,6 +73,7 @@ pub struct Connectors { pub nmi: ConnectorParams, pub nomupay: ConnectorParams, pub noon: ConnectorParamsWithModeType, + pub nordea: ConnectorParams, pub novalnet: ConnectorParams, pub nuvei: ConnectorParams, pub opayo: ConnectorParams, diff --git a/crates/router/src/connector.rs b/crates/router/src/connector.rs index 20f09c0ab9..6106aa27bf 100644 --- a/crates/router/src/connector.rs +++ b/crates/router/src/connector.rs @@ -23,17 +23,17 @@ pub use hyperswitch_connectors::connectors::{ juspaythreedsserver::Juspaythreedsserver, klarna, klarna::Klarna, mifinity, mifinity::Mifinity, mollie, mollie::Mollie, moneris, moneris::Moneris, multisafepay, multisafepay::Multisafepay, netcetera, netcetera::Netcetera, nexinets, nexinets::Nexinets, nexixpay, nexixpay::Nexixpay, - nmi, nmi::Nmi, nomupay, nomupay::Nomupay, noon, noon::Noon, novalnet, novalnet::Novalnet, - nuvei, nuvei::Nuvei, opayo, opayo::Opayo, opennode, opennode::Opennode, paybox, paybox::Paybox, - payeezy, payeezy::Payeezy, payme, payme::Payme, payone, payone::Payone, paypal, paypal::Paypal, - paystack, paystack::Paystack, payu, payu::Payu, placetopay, placetopay::Placetopay, plaid, - plaid::Plaid, powertranz, powertranz::Powertranz, prophetpay, prophetpay::Prophetpay, rapyd, - rapyd::Rapyd, razorpay, razorpay::Razorpay, recurly, recurly::Recurly, redsys, redsys::Redsys, - riskified, riskified::Riskified, shift4, shift4::Shift4, signifyd, signifyd::Signifyd, square, - square::Square, stax, stax::Stax, stripe, stripe::Stripe, stripebilling, - stripebilling::Stripebilling, taxjar, taxjar::Taxjar, threedsecureio, - threedsecureio::Threedsecureio, thunes, thunes::Thunes, trustpay, trustpay::Trustpay, tsys, - tsys::Tsys, unified_authentication_service, + nmi, nmi::Nmi, nomupay, nomupay::Nomupay, noon, noon::Noon, nordea, nordea::Nordea, novalnet, + novalnet::Novalnet, nuvei, nuvei::Nuvei, opayo, opayo::Opayo, opennode, opennode::Opennode, + paybox, paybox::Paybox, payeezy, payeezy::Payeezy, payme, payme::Payme, payone, payone::Payone, + paypal, paypal::Paypal, paystack, paystack::Paystack, payu, payu::Payu, placetopay, + placetopay::Placetopay, plaid, plaid::Plaid, powertranz, powertranz::Powertranz, prophetpay, + prophetpay::Prophetpay, rapyd, rapyd::Rapyd, razorpay, razorpay::Razorpay, recurly, + recurly::Recurly, redsys, redsys::Redsys, riskified, riskified::Riskified, shift4, + shift4::Shift4, signifyd, signifyd::Signifyd, square, square::Square, stax, stax::Stax, stripe, + stripe::Stripe, stripebilling, stripebilling::Stripebilling, taxjar, taxjar::Taxjar, + threedsecureio, threedsecureio::Threedsecureio, thunes, thunes::Thunes, trustpay, + trustpay::Trustpay, tsys, tsys::Tsys, unified_authentication_service, unified_authentication_service::UnifiedAuthenticationService, vgs, vgs::Vgs, volt, volt::Volt, wellsfargo, wellsfargo::Wellsfargo, wellsfargopayout, wellsfargopayout::Wellsfargopayout, wise, wise::Wise, worldline, worldline::Worldline, worldpay, worldpay::Worldpay, worldpayxml, diff --git a/crates/router/src/core/admin.rs b/crates/router/src/core/admin.rs index 4a877092d8..5db9e708a4 100644 --- a/crates/router/src/core/admin.rs +++ b/crates/router/src/core/admin.rs @@ -1554,6 +1554,10 @@ impl ConnectorAuthTypeAndMetadataValidation<'_> { noon::transformers::NoonAuthType::try_from(self.auth_type)?; Ok(()) } + // api_enums::Connector::Nordea => { + // nordea::transformers::NordeaAuthType::try_from(self.auth_type)?; + // Ok(()) + // } api_enums::Connector::Novalnet => { novalnet::transformers::NovalnetAuthType::try_from(self.auth_type)?; Ok(()) diff --git a/crates/router/src/types/api.rs b/crates/router/src/types/api.rs index 9296503c36..6838ec5b04 100644 --- a/crates/router/src/types/api.rs +++ b/crates/router/src/types/api.rs @@ -484,6 +484,7 @@ impl ConnectorData { Ok(ConnectorEnum::Old(Box::new(connector::Nomupay::new()))) } enums::Connector::Noon => Ok(ConnectorEnum::Old(Box::new(connector::Noon::new()))), + // enums::Connector::Nordea => Ok(ConnectorEnum::Old(Box::new(connector::Nordea::new()))), enums::Connector::Novalnet => { Ok(ConnectorEnum::Old(Box::new(connector::Novalnet::new()))) } diff --git a/crates/router/src/types/transformers.rs b/crates/router/src/types/transformers.rs index e40456b7d9..a882ee1927 100644 --- a/crates/router/src/types/transformers.rs +++ b/crates/router/src/types/transformers.rs @@ -288,6 +288,7 @@ impl ForeignTryFrom for common_enums::RoutableConnectors { api_enums::Connector::Nmi => Self::Nmi, api_enums::Connector::Nomupay => Self::Nomupay, api_enums::Connector::Noon => Self::Noon, + // api_enums::Connector::Nordea => Self::Nordea, api_enums::Connector::Novalnet => Self::Novalnet, api_enums::Connector::Nuvei => Self::Nuvei, api_enums::Connector::Opennode => Self::Opennode, diff --git a/crates/router/tests/connectors/main.rs b/crates/router/tests/connectors/main.rs index 2726363635..a51c85c1dc 100644 --- a/crates/router/tests/connectors/main.rs +++ b/crates/router/tests/connectors/main.rs @@ -62,6 +62,7 @@ mod nexixpay; mod nmi; mod nomupay; mod noon; +mod nordea; mod novalnet; mod nuvei; #[cfg(feature = "dummy_connector")] diff --git a/crates/router/tests/connectors/nordea.rs b/crates/router/tests/connectors/nordea.rs new file mode 100644 index 0000000000..af5d7b4d4b --- /dev/null +++ b/crates/router/tests/connectors/nordea.rs @@ -0,0 +1,420 @@ +use masking::Secret; +use router::types::{self, api, domain, storage::enums}; +use test_utils::connector_auth; + +use crate::utils::{self, ConnectorActions}; + +#[derive(Clone, Copy)] +struct NordeaTest; +impl ConnectorActions for NordeaTest {} +impl utils::Connector for NordeaTest { + fn get_data(&self) -> api::ConnectorData { + use router::connector::Nordea; + utils::construct_connector_data_old( + Box::new(Nordea::new()), + types::Connector::DummyConnector1, + api::GetToken::Connector, + None, + ) + } + + fn get_auth_token(&self) -> types::ConnectorAuthType { + utils::to_connector_auth_type( + connector_auth::ConnectorAuthentication::new() + .nordea + .expect("Missing connector authentication configuration") + .into(), + ) + } + + fn get_name(&self) -> String { + "nordea".to_string() + } +} + +static CONNECTOR: NordeaTest = NordeaTest {}; + +fn get_default_payment_info() -> Option { + None +} + +fn payment_method_details() -> Option { + None +} + +// Cards Positive Tests +// Creates a payment using the manual capture flow (Non 3DS). +#[actix_web::test] +async fn should_only_authorize_payment() { + let response = CONNECTOR + .authorize_payment(payment_method_details(), get_default_payment_info()) + .await + .expect("Authorize payment response"); + assert_eq!(response.status, enums::AttemptStatus::Authorized); +} + +// Captures a payment using the manual capture flow (Non 3DS). +#[actix_web::test] +async fn should_capture_authorized_payment() { + let response = CONNECTOR + .authorize_and_capture_payment(payment_method_details(), None, get_default_payment_info()) + .await + .expect("Capture payment response"); + assert_eq!(response.status, enums::AttemptStatus::Charged); +} + +// Partially captures a payment using the manual capture flow (Non 3DS). +#[actix_web::test] +async fn should_partially_capture_authorized_payment() { + let response = CONNECTOR + .authorize_and_capture_payment( + payment_method_details(), + Some(types::PaymentsCaptureData { + amount_to_capture: 50, + ..utils::PaymentCaptureType::default().0 + }), + get_default_payment_info(), + ) + .await + .expect("Capture payment response"); + assert_eq!(response.status, enums::AttemptStatus::Charged); +} + +// Synchronizes a payment using the manual capture flow (Non 3DS). +#[actix_web::test] +async fn should_sync_authorized_payment() { + let authorize_response = CONNECTOR + .authorize_payment(payment_method_details(), get_default_payment_info()) + .await + .expect("Authorize payment response"); + let txn_id = utils::get_connector_transaction_id(authorize_response.response); + let response = CONNECTOR + .psync_retry_till_status_matches( + enums::AttemptStatus::Authorized, + Some(types::PaymentsSyncData { + connector_transaction_id: types::ResponseId::ConnectorTransactionId( + txn_id.unwrap(), + ), + ..Default::default() + }), + get_default_payment_info(), + ) + .await + .expect("PSync response"); + assert_eq!(response.status, enums::AttemptStatus::Authorized,); +} + +// Voids a payment using the manual capture flow (Non 3DS). +#[actix_web::test] +async fn should_void_authorized_payment() { + let response = CONNECTOR + .authorize_and_void_payment( + payment_method_details(), + Some(types::PaymentsCancelData { + connector_transaction_id: String::from(""), + cancellation_reason: Some("requested_by_customer".to_string()), + ..Default::default() + }), + get_default_payment_info(), + ) + .await + .expect("Void payment response"); + assert_eq!(response.status, enums::AttemptStatus::Voided); +} + +// Refunds a payment using the manual capture flow (Non 3DS). +#[actix_web::test] +async fn should_refund_manually_captured_payment() { + let response = CONNECTOR + .capture_payment_and_refund( + payment_method_details(), + None, + None, + get_default_payment_info(), + ) + .await + .unwrap(); + assert_eq!( + response.response.unwrap().refund_status, + enums::RefundStatus::Success, + ); +} + +// Partially refunds a payment using the manual capture flow (Non 3DS). +#[actix_web::test] +async fn should_partially_refund_manually_captured_payment() { + let response = CONNECTOR + .capture_payment_and_refund( + payment_method_details(), + None, + Some(types::RefundsData { + refund_amount: 50, + ..utils::PaymentRefundType::default().0 + }), + get_default_payment_info(), + ) + .await + .unwrap(); + assert_eq!( + response.response.unwrap().refund_status, + enums::RefundStatus::Success, + ); +} + +// Synchronizes a refund using the manual capture flow (Non 3DS). +#[actix_web::test] +async fn should_sync_manually_captured_refund() { + let refund_response = CONNECTOR + .capture_payment_and_refund( + payment_method_details(), + None, + None, + get_default_payment_info(), + ) + .await + .unwrap(); + let response = CONNECTOR + .rsync_retry_till_status_matches( + enums::RefundStatus::Success, + refund_response.response.unwrap().connector_refund_id, + None, + get_default_payment_info(), + ) + .await + .unwrap(); + assert_eq!( + response.response.unwrap().refund_status, + enums::RefundStatus::Success, + ); +} + +// Creates a payment using the automatic capture flow (Non 3DS). +#[actix_web::test] +async fn should_make_payment() { + let authorize_response = CONNECTOR + .make_payment(payment_method_details(), get_default_payment_info()) + .await + .unwrap(); + assert_eq!(authorize_response.status, enums::AttemptStatus::Charged); +} + +// Synchronizes a payment using the automatic capture flow (Non 3DS). +#[actix_web::test] +async fn should_sync_auto_captured_payment() { + let authorize_response = CONNECTOR + .make_payment(payment_method_details(), get_default_payment_info()) + .await + .unwrap(); + assert_eq!(authorize_response.status, enums::AttemptStatus::Charged); + let txn_id = utils::get_connector_transaction_id(authorize_response.response); + assert_ne!(txn_id, None, "Empty connector transaction id"); + let response = CONNECTOR + .psync_retry_till_status_matches( + enums::AttemptStatus::Charged, + Some(types::PaymentsSyncData { + connector_transaction_id: types::ResponseId::ConnectorTransactionId( + txn_id.unwrap(), + ), + capture_method: Some(enums::CaptureMethod::Automatic), + ..Default::default() + }), + get_default_payment_info(), + ) + .await + .unwrap(); + assert_eq!(response.status, enums::AttemptStatus::Charged,); +} + +// Refunds a payment using the automatic capture flow (Non 3DS). +#[actix_web::test] +async fn should_refund_auto_captured_payment() { + let response = CONNECTOR + .make_payment_and_refund(payment_method_details(), None, get_default_payment_info()) + .await + .unwrap(); + assert_eq!( + response.response.unwrap().refund_status, + enums::RefundStatus::Success, + ); +} + +// Partially refunds a payment using the automatic capture flow (Non 3DS). +#[actix_web::test] +async fn should_partially_refund_succeeded_payment() { + let refund_response = CONNECTOR + .make_payment_and_refund( + payment_method_details(), + Some(types::RefundsData { + refund_amount: 50, + ..utils::PaymentRefundType::default().0 + }), + get_default_payment_info(), + ) + .await + .unwrap(); + assert_eq!( + refund_response.response.unwrap().refund_status, + enums::RefundStatus::Success, + ); +} + +// Creates multiple refunds against a payment using the automatic capture flow (Non 3DS). +#[actix_web::test] +async fn should_refund_succeeded_payment_multiple_times() { + CONNECTOR + .make_payment_and_multiple_refund( + payment_method_details(), + Some(types::RefundsData { + refund_amount: 50, + ..utils::PaymentRefundType::default().0 + }), + get_default_payment_info(), + ) + .await; +} + +// Synchronizes a refund using the automatic capture flow (Non 3DS). +#[actix_web::test] +async fn should_sync_refund() { + let refund_response = CONNECTOR + .make_payment_and_refund(payment_method_details(), None, get_default_payment_info()) + .await + .unwrap(); + let response = CONNECTOR + .rsync_retry_till_status_matches( + enums::RefundStatus::Success, + refund_response.response.unwrap().connector_refund_id, + None, + get_default_payment_info(), + ) + .await + .unwrap(); + assert_eq!( + response.response.unwrap().refund_status, + enums::RefundStatus::Success, + ); +} + +// Cards Negative scenarios +// Creates a payment with incorrect CVC. +#[actix_web::test] +async fn should_fail_payment_for_incorrect_cvc() { + let response = CONNECTOR + .make_payment( + Some(types::PaymentsAuthorizeData { + payment_method_data: domain::PaymentMethodData::Card(domain::Card { + card_cvc: Secret::new("12345".to_string()), + ..utils::CCardType::default().0 + }), + ..utils::PaymentAuthorizeType::default().0 + }), + get_default_payment_info(), + ) + .await + .unwrap(); + assert_eq!( + response.response.unwrap_err().message, + "Your card's security code is invalid.".to_string(), + ); +} + +// Creates a payment with incorrect expiry month. +#[actix_web::test] +async fn should_fail_payment_for_invalid_exp_month() { + let response = CONNECTOR + .make_payment( + Some(types::PaymentsAuthorizeData { + payment_method_data: domain::PaymentMethodData::Card(domain::Card { + card_exp_month: Secret::new("20".to_string()), + ..utils::CCardType::default().0 + }), + ..utils::PaymentAuthorizeType::default().0 + }), + get_default_payment_info(), + ) + .await + .unwrap(); + assert_eq!( + response.response.unwrap_err().message, + "Your card's expiration month is invalid.".to_string(), + ); +} + +// Creates a payment with incorrect expiry year. +#[actix_web::test] +async fn should_fail_payment_for_incorrect_expiry_year() { + let response = CONNECTOR + .make_payment( + Some(types::PaymentsAuthorizeData { + payment_method_data: domain::PaymentMethodData::Card(domain::Card { + card_exp_year: Secret::new("2000".to_string()), + ..utils::CCardType::default().0 + }), + ..utils::PaymentAuthorizeType::default().0 + }), + get_default_payment_info(), + ) + .await + .unwrap(); + assert_eq!( + response.response.unwrap_err().message, + "Your card's expiration year is invalid.".to_string(), + ); +} + +// Voids a payment using automatic capture flow (Non 3DS). +#[actix_web::test] +async fn should_fail_void_payment_for_auto_capture() { + let authorize_response = CONNECTOR + .make_payment(payment_method_details(), get_default_payment_info()) + .await + .unwrap(); + assert_eq!(authorize_response.status, enums::AttemptStatus::Charged); + let txn_id = utils::get_connector_transaction_id(authorize_response.response); + assert_ne!(txn_id, None, "Empty connector transaction id"); + let void_response = CONNECTOR + .void_payment(txn_id.unwrap(), None, get_default_payment_info()) + .await + .unwrap(); + assert_eq!( + void_response.response.unwrap_err().message, + "You cannot cancel this PaymentIntent because it has a status of succeeded." + ); +} + +// Captures a payment using invalid connector payment id. +#[actix_web::test] +async fn should_fail_capture_for_invalid_payment() { + let capture_response = CONNECTOR + .capture_payment("123456789".to_string(), None, get_default_payment_info()) + .await + .unwrap(); + assert_eq!( + capture_response.response.unwrap_err().message, + String::from("No such payment_intent: '123456789'") + ); +} + +// Refunds a payment with refund amount higher than payment amount. +#[actix_web::test] +async fn should_fail_for_refund_amount_higher_than_payment_amount() { + let response = CONNECTOR + .make_payment_and_refund( + payment_method_details(), + Some(types::RefundsData { + refund_amount: 150, + ..utils::PaymentRefundType::default().0 + }), + get_default_payment_info(), + ) + .await + .unwrap(); + assert_eq!( + response.response.unwrap_err().message, + "Refund amount (₹1.50) is greater than charge amount (₹1.00)", + ); +} + +// Connector dependent test cases goes here + +// [#478]: add unit tests for non 3DS, wallets & webhooks in connector tests diff --git a/crates/router/tests/connectors/sample_auth.toml b/crates/router/tests/connectors/sample_auth.toml index 7d208c5464..00dbfb6698 100644 --- a/crates/router/tests/connectors/sample_auth.toml +++ b/crates/router/tests/connectors/sample_auth.toml @@ -285,6 +285,10 @@ api_secret = "Consumer Secret" [taxjar] api_key = "API Key" +[nordea] +api_key = "Client Secret" +key1 = "Client ID" + [novalnet] api_key="API Key" diff --git a/crates/test_utils/src/connector_auth.rs b/crates/test_utils/src/connector_auth.rs index b9873776e0..dd2cfe6cf7 100644 --- a/crates/test_utils/src/connector_auth.rs +++ b/crates/test_utils/src/connector_auth.rs @@ -69,6 +69,7 @@ pub struct ConnectorAuthentication { pub nexixpay: Option, pub nomupay: Option, pub noon: Option, + pub nordea: Option, pub novalnet: Option, pub nmi: Option, pub nuvei: Option, diff --git a/loadtest/config/development.toml b/loadtest/config/development.toml index 8ccb694a38..d3edab0510 100644 --- a/loadtest/config/development.toml +++ b/loadtest/config/development.toml @@ -143,6 +143,7 @@ nexixpay.base_url = "https://xpaysandbox.nexigroup.com/api/phoenix-0.0/psp/api/v nmi.base_url = "https://secure.nmi.com/" nomupay.base_url = "https://payout-api.sandbox.nomupay.com" noon.base_url = "https://api-test.noonpayments.com/" +nordea.base_url = "https://api.nordeaopenbanking.com" noon.key_mode = "Test" novalnet.base_url = "https://payport.novalnet.de/v2" nuvei.base_url = "https://ppp-test.nuvei.com/" @@ -254,6 +255,7 @@ cards = [ "nexixpay", "nmi", "noon", + "nordea", "novalnet", "nuvei", "opayo", diff --git a/scripts/add_connector.sh b/scripts/add_connector.sh index bf27a84958..9b0fab8505 100755 --- a/scripts/add_connector.sh +++ b/scripts/add_connector.sh @@ -6,7 +6,7 @@ function find_prev_connector() { git checkout $self cp $self $self.tmp # Add new connector to existing list and sort it - connectors=(aci adyen adyenplatform airwallex amazonpay applepay archipel authorizedotnet bambora bamboraapac bankofamerica barclaycard billwerk bitpay bluesnap boku braintree cashtocode chargebee checkout coinbase cryptopay ctp_visa cybersource datatrans deutschebank digitalvirgo dlocal dummyconnector ebanx elavon facilitapay fiserv fiservemea fiuu forte getnet globalpay globepay gocardless gpayments helcim hipay iatapay inespay itaubank jpmorgan juspaythreedsserver klarna mifinity mollie moneris multisafepay netcetera nexinets nexixpay nomupay noon novalnet nuvei opayo opennode paybox payeezy payme payone paypal paystack payu placetopay plaid powertranz prophetpay rapyd razorpay recurly redsys shift4 square stax stripe stripebilling taxjar threedsecureio thunes trustpay tsys unified_authentication_service vgs volt wellsfargo wellsfargopayout wise worldline worldpay worldpayxml xendit zsl "$1") + connectors=(aci adyen adyenplatform airwallex amazonpay applepay archipel authorizedotnet bambora bamboraapac bankofamerica barclaycard billwerk bitpay bluesnap boku braintree cashtocode chargebee checkout coinbase cryptopay ctp_visa cybersource datatrans deutschebank digitalvirgo dlocal dummyconnector ebanx elavon facilitapay fiserv fiservemea fiuu forte getnet globalpay globepay gocardless gpayments helcim hipay iatapay inespay itaubank jpmorgan juspaythreedsserver klarna mifinity mollie moneris multisafepay netcetera nexinets nexixpay nomupay noon nordea novalnet nuvei opayo opennode paybox payeezy payme payone paypal paystack payu placetopay plaid powertranz prophetpay rapyd razorpay recurly redsys shift4 square stax stripe stripebilling taxjar threedsecureio thunes trustpay tsys unified_authentication_service vgs volt wellsfargo wellsfargopayout wise worldline worldpay worldpayxml xendit zsl "$1") IFS=$'\n' sorted=($(sort <<<"${connectors[*]}")); unset IFS res="$(echo ${sorted[@]})" sed -i'' -e "s/^ connectors=.*/ connectors=($res \"\$1\")/" $self.tmp