diff --git a/api-reference-v2/openapi_spec.json b/api-reference-v2/openapi_spec.json index 3f1a0ae01de8..c5238b055849 100644 --- a/api-reference-v2/openapi_spec.json +++ b/api-reference-v2/openapi_spec.json @@ -18131,6 +18131,10 @@ "type": "object", "description": "Product authentication ids", "nullable": true + }, + "is_tokenize_before_payment_enabled": { + "type": "boolean", + "description": "Indicates if network tokenization before first payment is enabled or not" } }, "additionalProperties": false @@ -18164,7 +18168,8 @@ "is_tax_connector_enabled", "is_network_tokenization_enabled", "should_collect_cvv_during_payment", - "is_click_to_pay_enabled" + "is_click_to_pay_enabled", + "is_tokenize_before_payment_enabled" ], "properties": { "merchant_id": { @@ -18355,6 +18360,10 @@ "type": "object", "description": "Product authentication ids", "nullable": true + }, + "is_tokenize_before_payment_enabled": { + "type": "boolean", + "description": "Indicates if network tokenization before first payment is enabled or not" } } }, diff --git a/api-reference/openapi_spec.json b/api-reference/openapi_spec.json index 0d750ea7bf53..9f3ce8d82204 100644 --- a/api-reference/openapi_spec.json +++ b/api-reference/openapi_spec.json @@ -22432,6 +22432,10 @@ "type": "object", "description": "Product authentication ids", "nullable": true + }, + "is_tokenize_before_payment_enabled": { + "type": "boolean", + "description": "Indicates if network tokenization before first payment is enabled or not" } }, "additionalProperties": false @@ -22465,7 +22469,8 @@ "is_tax_connector_enabled", "is_network_tokenization_enabled", "is_auto_retries_enabled", - "is_click_to_pay_enabled" + "is_click_to_pay_enabled", + "is_tokenize_before_payment_enabled" ], "properties": { "merchant_id": { @@ -22673,6 +22678,10 @@ "type": "object", "description": "Product authentication ids", "nullable": true + }, + "is_tokenize_before_payment_enabled": { + "type": "boolean", + "description": "Indicates if network tokenization before first payment is enabled or not" } } }, diff --git a/crates/api_models/src/admin.rs b/crates/api_models/src/admin.rs index 76db62bbeaff..9849f727c90f 100644 --- a/crates/api_models/src/admin.rs +++ b/crates/api_models/src/admin.rs @@ -1974,6 +1974,10 @@ pub struct ProfileCreate { /// Product authentication ids #[schema(value_type = Option, example = r#"{ "key1": "value-1", "key2": "value-2" }"#)] pub authentication_product_ids: Option>, + + /// Indicates if network tokenization before first payment is enabled or not + #[serde(default)] + pub is_tokenize_before_payment_enabled: bool, } #[nutype::nutype( @@ -2090,6 +2094,10 @@ pub struct ProfileCreate { /// Product authentication ids #[schema(value_type = Option, example = r#"{ "key1": "value-1", "key2": "value-2" }"#)] pub authentication_product_ids: Option>, + + /// Indicates if network tokenization before first payment is enabled or not + #[serde(default)] + pub is_tokenize_before_payment_enabled: bool, } #[cfg(feature = "v1")] @@ -2226,6 +2234,9 @@ pub struct ProfileResponse { /// Product authentication ids #[schema(value_type = Option, example = r#"{ "key1": "value-1", "key2": "value-2" }"#)] pub authentication_product_ids: Option, + + /// Indicates if network tokenization before first payment is enabled or not + pub is_tokenize_before_payment_enabled: bool, } #[cfg(feature = "v2")] @@ -2349,6 +2360,9 @@ pub struct ProfileResponse { /// Product authentication ids #[schema(value_type = Option, example = r#"{ "key1": "value-1", "key2": "value-2" }"#)] pub authentication_product_ids: Option, + + /// Indicates if network tokenization before first payment is enabled or not + pub is_tokenize_before_payment_enabled: bool, } #[cfg(feature = "v1")] @@ -2479,6 +2493,10 @@ pub struct ProfileUpdate { /// Product authentication ids #[schema(value_type = Option, example = r#"{ "key1": "value-1", "key2": "value-2" }"#)] pub authentication_product_ids: Option>, + + /// Indicates if network tokenization before first payment is enabled or not + #[schema(default = false, example = false)] + pub is_tokenize_before_payment_enabled: Option, } #[cfg(feature = "v2")] @@ -2590,6 +2608,10 @@ pub struct ProfileUpdate { /// Product authentication ids #[schema(value_type = Option, example = r#"{ "key1": "value-1", "key2": "value-2" }"#)] pub authentication_product_ids: Option>, + + /// Indicates if network tokenization before first payment is enabled or not + #[schema(default = false, example = false)] + pub is_tokenize_before_payment_enabled: Option, } #[derive(Clone, Debug, serde::Deserialize, serde::Serialize, ToSchema)] diff --git a/crates/diesel_models/src/business_profile.rs b/crates/diesel_models/src/business_profile.rs index dff3f174fc50..f6cbc0d34883 100644 --- a/crates/diesel_models/src/business_profile.rs +++ b/crates/diesel_models/src/business_profile.rs @@ -59,6 +59,7 @@ pub struct Profile { pub max_auto_retries_enabled: Option, pub is_click_to_pay_enabled: bool, pub authentication_product_ids: Option, + pub is_tokenize_before_payment_enabled: bool, } #[cfg(feature = "v1")] @@ -104,6 +105,7 @@ pub struct ProfileNew { pub max_auto_retries_enabled: Option, pub is_click_to_pay_enabled: bool, pub authentication_product_ids: Option, + pub is_tokenize_before_payment_enabled: Option, } #[cfg(feature = "v1")] @@ -146,6 +148,7 @@ pub struct ProfileUpdateInternal { pub max_auto_retries_enabled: Option, pub is_click_to_pay_enabled: Option, pub authentication_product_ids: Option, + pub is_tokenize_before_payment_enabled: Option, } #[cfg(feature = "v1")] @@ -187,6 +190,7 @@ impl ProfileUpdateInternal { max_auto_retries_enabled, is_click_to_pay_enabled, authentication_product_ids, + is_tokenize_before_payment_enabled, } = self; Profile { profile_id: source.profile_id, @@ -250,6 +254,8 @@ impl ProfileUpdateInternal { .unwrap_or(source.is_click_to_pay_enabled), authentication_product_ids: authentication_product_ids .or(source.authentication_product_ids), + is_tokenize_before_payment_enabled: is_tokenize_before_payment_enabled + .unwrap_or(source.is_tokenize_before_payment_enabled), } } } diff --git a/crates/diesel_models/src/schema.rs b/crates/diesel_models/src/schema.rs index 61a8a7e19b9e..0608748b3029 100644 --- a/crates/diesel_models/src/schema.rs +++ b/crates/diesel_models/src/schema.rs @@ -216,6 +216,7 @@ diesel::table! { max_auto_retries_enabled -> Nullable, is_click_to_pay_enabled -> Bool, authentication_product_ids -> Nullable, + is_tokenize_before_payment_enabled -> Bool, } } diff --git a/crates/hyperswitch_domain_models/src/business_profile.rs b/crates/hyperswitch_domain_models/src/business_profile.rs index 1df474f18d5f..66cbf8dbee3b 100644 --- a/crates/hyperswitch_domain_models/src/business_profile.rs +++ b/crates/hyperswitch_domain_models/src/business_profile.rs @@ -60,6 +60,7 @@ pub struct Profile { pub max_auto_retries_enabled: Option, pub is_click_to_pay_enabled: bool, pub authentication_product_ids: Option, + pub is_tokenize_before_payment_enabled: bool, } #[cfg(feature = "v1")] @@ -102,6 +103,7 @@ pub struct ProfileSetter { pub max_auto_retries_enabled: Option, pub is_click_to_pay_enabled: bool, pub authentication_product_ids: Option, + pub is_tokenize_before_payment_enabled: bool, } #[cfg(feature = "v1")] @@ -151,6 +153,7 @@ impl From for Profile { max_auto_retries_enabled: value.max_auto_retries_enabled, is_click_to_pay_enabled: value.is_click_to_pay_enabled, authentication_product_ids: value.authentication_product_ids, + is_tokenize_before_payment_enabled: value.is_tokenize_before_payment_enabled, } } } @@ -202,6 +205,7 @@ pub struct ProfileGeneralUpdate { pub max_auto_retries_enabled: Option, pub is_click_to_pay_enabled: Option, pub authentication_product_ids: Option, + pub is_tokenize_before_payment_enabled: Option, } #[cfg(feature = "v1")] @@ -266,6 +270,7 @@ impl From for ProfileUpdateInternal { max_auto_retries_enabled, is_click_to_pay_enabled, authentication_product_ids, + is_tokenize_before_payment_enabled, } = *update; Self { @@ -305,6 +310,7 @@ impl From for ProfileUpdateInternal { max_auto_retries_enabled, is_click_to_pay_enabled, authentication_product_ids, + is_tokenize_before_payment_enabled, } } ProfileUpdate::RoutingAlgorithmUpdate { @@ -346,6 +352,7 @@ impl From for ProfileUpdateInternal { max_auto_retries_enabled: None, is_click_to_pay_enabled: None, authentication_product_ids: None, + is_tokenize_before_payment_enabled: None, }, ProfileUpdate::DynamicRoutingAlgorithmUpdate { dynamic_routing_algorithm, @@ -385,6 +392,7 @@ impl From for ProfileUpdateInternal { max_auto_retries_enabled: None, is_click_to_pay_enabled: None, authentication_product_ids: None, + is_tokenize_before_payment_enabled: None, }, ProfileUpdate::ExtendedCardInfoUpdate { is_extended_card_info_enabled, @@ -424,6 +432,7 @@ impl From for ProfileUpdateInternal { max_auto_retries_enabled: None, is_click_to_pay_enabled: None, authentication_product_ids: None, + is_tokenize_before_payment_enabled: None, }, ProfileUpdate::ConnectorAgnosticMitUpdate { is_connector_agnostic_mit_enabled, @@ -463,6 +472,7 @@ impl From for ProfileUpdateInternal { max_auto_retries_enabled: None, is_click_to_pay_enabled: None, authentication_product_ids: None, + is_tokenize_before_payment_enabled: None, }, ProfileUpdate::NetworkTokenizationUpdate { is_network_tokenization_enabled, @@ -502,6 +512,7 @@ impl From for ProfileUpdateInternal { max_auto_retries_enabled: None, is_click_to_pay_enabled: None, authentication_product_ids: None, + is_tokenize_before_payment_enabled: None, }, } } @@ -560,6 +571,7 @@ impl super::behaviour::Conversion for Profile { max_auto_retries_enabled: self.max_auto_retries_enabled, is_click_to_pay_enabled: self.is_click_to_pay_enabled, authentication_product_ids: self.authentication_product_ids, + is_tokenize_before_payment_enabled: self.is_tokenize_before_payment_enabled, }) } @@ -630,6 +642,7 @@ impl super::behaviour::Conversion for Profile { max_auto_retries_enabled: item.max_auto_retries_enabled, is_click_to_pay_enabled: item.is_click_to_pay_enabled, authentication_product_ids: item.authentication_product_ids, + is_tokenize_before_payment_enabled: item.is_network_tokenization_enabled, }) } .await @@ -684,6 +697,7 @@ impl super::behaviour::Conversion for Profile { max_auto_retries_enabled: self.max_auto_retries_enabled, is_click_to_pay_enabled: self.is_click_to_pay_enabled, authentication_product_ids: self.authentication_product_ids, + is_tokenize_before_payment_enabled: Some(self.is_tokenize_before_payment_enabled), }) } } @@ -730,6 +744,7 @@ pub struct Profile { pub is_network_tokenization_enabled: bool, pub is_click_to_pay_enabled: bool, pub authentication_product_ids: Option, + pub is_tokenize_before_payment_enabled: bool, } #[cfg(feature = "v2")] @@ -772,6 +787,7 @@ pub struct ProfileSetter { pub is_network_tokenization_enabled: bool, pub is_click_to_pay_enabled: bool, pub authentication_product_ids: Option, + pub is_tokenize_before_payment_enabled: bool, } #[cfg(feature = "v2")] @@ -821,6 +837,7 @@ impl From for Profile { is_network_tokenization_enabled: value.is_network_tokenization_enabled, is_click_to_pay_enabled: value.is_click_to_pay_enabled, authentication_product_ids: value.authentication_product_ids, + is_tokenize_before_payment_enabled: value.is_tokenize_before_payment_enabled, } } } @@ -973,6 +990,7 @@ impl From for ProfileUpdateInternal { max_auto_retries_enabled: None, is_click_to_pay_enabled: None, authentication_product_ids, + is_tokenize_before_payment_enabled: None, } } ProfileUpdate::RoutingAlgorithmUpdate { @@ -1016,6 +1034,7 @@ impl From for ProfileUpdateInternal { max_auto_retries_enabled: None, is_click_to_pay_enabled: None, authentication_product_ids: None, + is_tokenize_before_payment_enabled: None, }, ProfileUpdate::ExtendedCardInfoUpdate { is_extended_card_info_enabled, @@ -1057,6 +1076,7 @@ impl From for ProfileUpdateInternal { max_auto_retries_enabled: None, is_click_to_pay_enabled: None, authentication_product_ids: None, + is_tokenize_before_payment_enabled: None, }, ProfileUpdate::ConnectorAgnosticMitUpdate { is_connector_agnostic_mit_enabled, @@ -1098,6 +1118,7 @@ impl From for ProfileUpdateInternal { max_auto_retries_enabled: None, is_click_to_pay_enabled: None, authentication_product_ids: None, + is_tokenize_before_payment_enabled: None, }, ProfileUpdate::DefaultRoutingFallbackUpdate { default_fallback_routing, @@ -1139,6 +1160,7 @@ impl From for ProfileUpdateInternal { max_auto_retries_enabled: None, is_click_to_pay_enabled: None, authentication_product_ids: None, + is_tokenize_before_payment_enabled: None, }, ProfileUpdate::NetworkTokenizationUpdate { is_network_tokenization_enabled, @@ -1180,6 +1202,7 @@ impl From for ProfileUpdateInternal { max_auto_retries_enabled: None, is_click_to_pay_enabled: None, authentication_product_ids: None, + is_tokenize_before_payment_enabled: None, }, ProfileUpdate::CollectCvvDuringPaymentUpdate { should_collect_cvv_during_payment, @@ -1221,6 +1244,7 @@ impl From for ProfileUpdateInternal { max_auto_retries_enabled: None, is_click_to_pay_enabled: None, authentication_product_ids: None, + is_tokenize_before_payment_enabled: None, }, } } @@ -1282,6 +1306,7 @@ impl super::behaviour::Conversion for Profile { max_auto_retries_enabled: None, is_click_to_pay_enabled: self.is_click_to_pay_enabled, authentication_product_ids: self.authentication_product_ids, + is_tokenize_before_payment_enabled: self.is_tokenize_before_payment_enabled, }) } @@ -1352,6 +1377,7 @@ impl super::behaviour::Conversion for Profile { is_network_tokenization_enabled: item.is_network_tokenization_enabled, is_click_to_pay_enabled: item.is_click_to_pay_enabled, authentication_product_ids: item.authentication_product_ids, + is_tokenize_before_payment_enabled: item.is_tokenize_before_payment_enabled, }) } .await @@ -1409,6 +1435,7 @@ impl super::behaviour::Conversion for Profile { max_auto_retries_enabled: None, is_click_to_pay_enabled: self.is_click_to_pay_enabled, authentication_product_ids: self.authentication_product_ids, + is_tokenize_before_payment_enabled: self.is_tokenize_before_payment_enabled, }) } } diff --git a/crates/router/src/core/admin.rs b/crates/router/src/core/admin.rs index 8e6d74c83753..e7f5e947bb84 100644 --- a/crates/router/src/core/admin.rs +++ b/crates/router/src/core/admin.rs @@ -3701,6 +3701,7 @@ impl ProfileCreateBridge for api::ProfileCreate { max_auto_retries_enabled: self.max_auto_retries_enabled.map(i16::from), is_click_to_pay_enabled: self.is_click_to_pay_enabled, authentication_product_ids, + is_tokenize_before_payment_enabled: self.is_tokenize_before_payment_enabled, })) } @@ -4075,6 +4076,7 @@ impl ProfileUpdateBridge for api::ProfileUpdate { max_auto_retries_enabled: self.max_auto_retries_enabled.map(i16::from), is_click_to_pay_enabled: self.is_click_to_pay_enabled, authentication_product_ids, + is_tokenize_before_payment_enabled: self.is_tokenize_before_payment_enabled, }, ))) } diff --git a/crates/router/src/core/payment_methods/network_tokenization.rs b/crates/router/src/core/payment_methods/network_tokenization.rs index bb730caad2b5..deac9efbcc44 100644 --- a/crates/router/src/core/payment_methods/network_tokenization.rs +++ b/crates/router/src/core/payment_methods/network_tokenization.rs @@ -79,21 +79,22 @@ pub struct GetCardToken { } #[derive(Debug, Deserialize)] pub struct AuthenticationDetails { - cryptogram: Secret, - token: CardNumber, //network token + pub cryptogram: Secret, + pub token: CardNumber, //network token } #[derive(Debug, Serialize, Deserialize)] pub struct TokenDetails { - exp_month: Secret, - exp_year: Secret, + pub exp_month: Secret, + pub exp_year: Secret, } #[derive(Debug, Deserialize)] pub struct TokenResponse { - authentication_details: AuthenticationDetails, - network: api_enums::CardNetwork, - token_details: TokenDetails, + pub authentication_details: AuthenticationDetails, + pub network: api_enums::CardNetwork, + pub token_details: TokenDetails, + pub eci: Option, } #[derive(Debug, Serialize, Deserialize)] diff --git a/crates/router/src/core/payments.rs b/crates/router/src/core/payments.rs index e68bd09ba462..870520e8f35b 100644 --- a/crates/router/src/core/payments.rs +++ b/crates/router/src/core/payments.rs @@ -435,6 +435,15 @@ where } _ => (), }; + let customer_acceptance = payment_data + .get_payment_attempt() + .customer_acceptance + .clone(); + let customer_id = payment_data.get_payment_intent().customer_id.clone(); + let payment_method_data = payment_data.get_payment_method_data(); + let is_pre_tokenization_enabled = business_profile.is_network_tokenization_enabled + && business_profile.is_tokenize_before_payment_enabled + && customer_acceptance.is_some(); payment_data = match connector_details { ConnectorCallType::PreDetermined(connector) => { #[cfg(all(feature = "dynamic_routing", feature = "v1"))] @@ -455,6 +464,64 @@ where } else { None }; + let filtered_nt_supported_connectors = + get_filtered_nt_supported_connectors(&state, [connector.clone()].to_vec()); + + let is_nt_supported_connector_available = + filtered_nt_supported_connectors.first().is_some(); + + if is_pre_tokenization_enabled && is_nt_supported_connector_available { + let pre_tokenization_response = tokenization::pre_payment_tokenization( + state, + customer_id, + payment_method_data, + ) + .await?; + let pm_data = payment_data.get_payment_method_data(); + match pre_tokenization_response { + (Some(token_response), Some(_token_ref)) => { + let token_data = domain::NetworkTokenData { + token_number: token_response.authentication_details.token, + token_exp_month: token_response.token_details.exp_month, + token_exp_year: token_response.token_details.exp_year, + token_cryptogram: Some( + token_response.authentication_details.cryptogram, + ), + card_issuer: None, + card_network: Some(token_response.network), + card_type: None, + card_issuing_country: None, + bank_code: None, + nick_name: None, + eci: token_response.eci, + }; + match pm_data { + Some(domain::PaymentMethodData::Card(card_data)) => { + let card_and_network_token_data = CardAndNetworkTokenData { + card_data: card_data.clone(), + network_token_data: token_data.clone(), + }; + payment_data.set_vault_operation( + PaymentMethodDataAction::SaveCardAndNetworkTokenData( + card_and_network_token_data.clone(), + ), + ) + } + _ => (), + } + payment_data.set_payment_method_data(Some( + domain::PaymentMethodData::NetworkToken(token_data), + )); + } + _ => match pm_data { + Some(domain::PaymentMethodData::Card(card_data)) => payment_data + .set_vault_operation(PaymentMethodDataAction::SaveCardData( + card_data.clone(), + )), + _ => (), + }, + } + } let (router_data, mca) = call_connector_service( state, req_state.clone(), @@ -540,10 +607,68 @@ where .map_err(|e| logger::error!(routable_connector_error=?e)) .unwrap_or_default(); + let filtered_nt_supported_connectors = + get_filtered_nt_supported_connectors(&state, connectors.clone()); + let is_nt_supported_connector_available = + filtered_nt_supported_connectors.first().is_some(); + let mut connectors = connectors.into_iter(); let connector_data = get_connector_data(&mut connectors)?; + if is_pre_tokenization_enabled && is_nt_supported_connector_available { + let pre_tokenization_response = tokenization::pre_payment_tokenization( + state, + customer_id, + payment_method_data, + ) + .await?; + let pm_data = payment_data.get_payment_method_data(); + match pre_tokenization_response { + (Some(token_response), Some(_token_ref)) => { + let token_data = domain::NetworkTokenData { + token_number: token_response.authentication_details.token, + token_exp_month: token_response.token_details.exp_month, + token_exp_year: token_response.token_details.exp_year, + token_cryptogram: Some( + token_response.authentication_details.cryptogram, + ), + card_issuer: None, + card_network: Some(token_response.network), + card_type: None, + card_issuing_country: None, + bank_code: None, + nick_name: None, + eci: token_response.eci, + }; + match pm_data { + Some(domain::PaymentMethodData::Card(card_data)) => { + let card_and_network_token_data = CardAndNetworkTokenData { + card_data: card_data.clone(), + network_token_data: token_data.clone(), + }; + payment_data.set_vault_operation( + PaymentMethodDataAction::SaveCardAndNetworkTokenData( + card_and_network_token_data.clone(), + ), + ) + } + _ => (), + } + payment_data.set_payment_method_data(Some( + domain::PaymentMethodData::NetworkToken(token_data), + )); + } + _ => match pm_data { + Some(domain::PaymentMethodData::Card(card_data)) => payment_data + .set_vault_operation(PaymentMethodDataAction::SaveCardData( + card_data.clone(), + )), + _ => (), + }, + } + } + let schedule_time = if should_add_task_to_process_tracker { payment_sync::get_sync_process_schedule_time( &*state.store, @@ -4399,6 +4524,19 @@ where pub tax_data: Option, pub session_id: Option, pub service_details: Option, + pub vault_operation: Option, +} + +#[derive(Clone, serde::Serialize, Debug)] +pub enum PaymentMethodDataAction { + SaveCardData(hyperswitch_domain_models::payment_method_data::Card), + SaveCardAndNetworkTokenData(CardAndNetworkTokenData), +} + +#[derive(Default, Clone, serde::Serialize, Debug)] +pub struct CardAndNetworkTokenData { + pub card_data: hyperswitch_domain_models::payment_method_data::Card, + pub network_token_data: hyperswitch_domain_models::payment_method_data::NetworkTokenData, } #[derive(Clone, serde::Serialize, Debug)] @@ -5592,25 +5730,8 @@ where .get_required_value("payment_method_info")? .clone(); - //fetch connectors that support ntid flow - let ntid_supported_connectors = &state - .conf - .network_transaction_id_supported_connectors - .connector_list; - //filered connectors list with ntid_supported_connectors - let filtered_ntid_supported_connectors = - filter_ntid_supported_connectors(connectors.clone(), ntid_supported_connectors); - - //fetch connectors that support network tokenization flow - let network_tokenization_supported_connectors = &state - .conf - .network_tokenization_supported_connectors - .connector_list; - //filered connectors list with ntid_supported_connectors and network_tokenization_supported_connectors - let filtered_nt_supported_connectors = filter_network_tokenization_supported_connectors( - filtered_ntid_supported_connectors, - network_tokenization_supported_connectors, - ); + let filtered_nt_supported_connectors = + get_filtered_nt_supported_connectors(&state, connectors.clone()); let action_type = decide_action_type( state, @@ -5895,6 +6016,31 @@ pub fn filter_network_tokenization_supported_connectors( .collect() } +pub fn get_filtered_nt_supported_connectors( + state: &SessionState, + connectors: Vec, +) -> Vec { + //fetch connectors that support ntid flow + let ntid_supported_connectors = &state + .conf + .network_transaction_id_supported_connectors + .connector_list; + //filered connectors list with ntid_supported_connectors + let filtered_ntid_supported_connectors = + filter_ntid_supported_connectors(connectors.clone(), ntid_supported_connectors); + + //fetch connectors that support network tokenization flow + let network_tokenization_supported_connectors = &state + .conf + .network_tokenization_supported_connectors + .connector_list; + //filered connectors list with ntid_supported_connectors and network_tokenization_supported_connectors + filter_network_tokenization_supported_connectors( + filtered_ntid_supported_connectors, + network_tokenization_supported_connectors, + ) +} + #[cfg(feature = "v1")] pub async fn decide_action_type( state: &SessionState, @@ -6913,6 +7059,7 @@ pub trait OperationSessionSetters { straight_through_algorithm: serde_json::Value, ); fn set_connector_in_payment_attempt(&mut self, connector: Option); + fn set_vault_operation(&mut self, vault_operation: PaymentMethodDataAction); } #[cfg(feature = "v1")] @@ -7161,6 +7308,10 @@ impl OperationSessionSetters for PaymentData { fn set_connector_in_payment_attempt(&mut self, connector: Option) { self.payment_attempt.connector = connector; } + + fn set_vault_operation(&mut self, vault_operation: PaymentMethodDataAction) { + self.vault_operation = Some(vault_operation); + } } #[cfg(feature = "v2")] @@ -7374,6 +7525,10 @@ impl OperationSessionSetters for PaymentIntentData { fn set_connector_in_payment_attempt(&mut self, _connector: Option) { todo!() } + + fn set_vault_operation(&mut self, vault_operation: PaymentMethodDataAction) { + todo!() + } } #[cfg(feature = "v2")] @@ -7587,6 +7742,10 @@ impl OperationSessionSetters for PaymentConfirmData { fn set_connector_in_payment_attempt(&mut self, connector: Option) { self.payment_attempt.connector = connector; } + + fn set_vault_operation(&mut self, vault_operation: PaymentMethodDataAction) { + self.vault_operation = Some(vault_operation); + } } #[cfg(feature = "v2")] @@ -7800,6 +7959,10 @@ impl OperationSessionSetters for PaymentStatusData { fn set_connector_in_payment_attempt(&mut self, connector: Option) { todo!() } + + fn set_vault_operation(&mut self, vault_operation: PaymentMethodDataAction) { + todo!() + } } #[cfg(feature = "v2")] diff --git a/crates/router/src/core/payments/operations/payment_approve.rs b/crates/router/src/core/payments/operations/payment_approve.rs index a5993eb2f012..82da4cead21b 100644 --- a/crates/router/src/core/payments/operations/payment_approve.rs +++ b/crates/router/src/core/payments/operations/payment_approve.rs @@ -195,6 +195,7 @@ impl GetTracker, api::PaymentsCaptureR tax_data: None, session_id: None, service_details: None, + vault_operation: None, }; let get_trackers_response = operations::GetTrackerResponse { diff --git a/crates/router/src/core/payments/operations/payment_cancel.rs b/crates/router/src/core/payments/operations/payment_cancel.rs index 427c10aab629..2fa132ae9430 100644 --- a/crates/router/src/core/payments/operations/payment_cancel.rs +++ b/crates/router/src/core/payments/operations/payment_cancel.rs @@ -206,6 +206,7 @@ impl GetTracker, api::PaymentsCancelRe tax_data: None, session_id: None, service_details: None, + vault_operation: None, }; let get_trackers_response = operations::GetTrackerResponse { diff --git a/crates/router/src/core/payments/operations/payment_capture.rs b/crates/router/src/core/payments/operations/payment_capture.rs index f8d304bcb79c..767325f225ae 100644 --- a/crates/router/src/core/payments/operations/payment_capture.rs +++ b/crates/router/src/core/payments/operations/payment_capture.rs @@ -255,6 +255,7 @@ impl GetTracker, api::Paymen tax_data: None, session_id: None, service_details: None, + vault_operation: None, }; let get_trackers_response = operations::GetTrackerResponse { diff --git a/crates/router/src/core/payments/operations/payment_complete_authorize.rs b/crates/router/src/core/payments/operations/payment_complete_authorize.rs index 914ddf251efe..df8aead52b7e 100644 --- a/crates/router/src/core/payments/operations/payment_complete_authorize.rs +++ b/crates/router/src/core/payments/operations/payment_complete_authorize.rs @@ -350,6 +350,7 @@ impl GetTracker, api::PaymentsRequest> tax_data: None, session_id: None, service_details: None, + vault_operation: None, }; let customer_details = Some(CustomerDetails { diff --git a/crates/router/src/core/payments/operations/payment_confirm.rs b/crates/router/src/core/payments/operations/payment_confirm.rs index 96406932445a..8f8ebc803b12 100644 --- a/crates/router/src/core/payments/operations/payment_confirm.rs +++ b/crates/router/src/core/payments/operations/payment_confirm.rs @@ -822,6 +822,7 @@ impl GetTracker, api::PaymentsRequest> tax_data: None, session_id: None, service_details: request.ctp_service_details.clone(), + vault_operation: None, }; let get_trackers_response = operations::GetTrackerResponse { diff --git a/crates/router/src/core/payments/operations/payment_create.rs b/crates/router/src/core/payments/operations/payment_create.rs index 36b338f2d528..6ac0f03af6ee 100644 --- a/crates/router/src/core/payments/operations/payment_create.rs +++ b/crates/router/src/core/payments/operations/payment_create.rs @@ -614,6 +614,7 @@ impl GetTracker, api::PaymentsRequest> tax_data: None, session_id: None, service_details: None, + vault_operation: None, }; let get_trackers_response = operations::GetTrackerResponse { diff --git a/crates/router/src/core/payments/operations/payment_post_session_tokens.rs b/crates/router/src/core/payments/operations/payment_post_session_tokens.rs index 7337f0808847..d7be71213ebc 100644 --- a/crates/router/src/core/payments/operations/payment_post_session_tokens.rs +++ b/crates/router/src/core/payments/operations/payment_post_session_tokens.rs @@ -166,6 +166,7 @@ impl GetTracker, api::PaymentsPostSess tax_data: None, session_id: None, service_details: None, + vault_operation: None, }; let get_trackers_response = operations::GetTrackerResponse { operation: Box::new(self), diff --git a/crates/router/src/core/payments/operations/payment_reject.rs b/crates/router/src/core/payments/operations/payment_reject.rs index a6a10be8e9ab..a53ce46b6037 100644 --- a/crates/router/src/core/payments/operations/payment_reject.rs +++ b/crates/router/src/core/payments/operations/payment_reject.rs @@ -193,6 +193,7 @@ impl GetTracker, PaymentsCancelRequest tax_data: None, session_id: None, service_details: None, + vault_operation: None, }; let get_trackers_response = operations::GetTrackerResponse { diff --git a/crates/router/src/core/payments/operations/payment_response.rs b/crates/router/src/core/payments/operations/payment_response.rs index 988429a0eecb..09e47096efee 100644 --- a/crates/router/src/core/payments/operations/payment_response.rs +++ b/crates/router/src/core/payments/operations/payment_response.rs @@ -163,6 +163,8 @@ impl PostUpdateTracker, types::PaymentsAuthor .and_then(|billing_details| billing_details.address.as_ref()) .and_then(|address| address.get_optional_full_name()); let mut should_avoid_saving = false; + let vault_operation = payment_data.vault_operation.clone(); + let payment_method_info = payment_data.payment_method_info.clone(); if let Some(payment_method_info) = &payment_data.payment_method_info { if payment_data.payment_intent.off_session.is_none() && resp.response.is_ok() { @@ -200,6 +202,8 @@ impl PostUpdateTracker, types::PaymentsAuthor payment_method_billing_address, business_profile, connector_mandate_reference_id.clone(), + vault_operation.clone(), + payment_method_info.clone(), )); let is_connector_mandate = resp.request.customer_acceptance.is_some() @@ -315,6 +319,8 @@ impl PostUpdateTracker, types::PaymentsAuthor payment_method_billing_address.as_ref(), &business_profile, connector_mandate_reference_id, + vault_operation.clone(), + payment_method_info.clone(), )) .await; @@ -1081,7 +1087,8 @@ impl PostUpdateTracker, types::SetupMandateRequestDa .connector_mandate_detail .as_ref() .map(|detail| ConnectorMandateReferenceId::foreign_from(detail.clone())); - + let vault_operation = payment_data.vault_operation.clone(); + let payment_method_info = payment_data.payment_method_info.clone(); let merchant_connector_id = payment_data.payment_attempt.merchant_connector_id.clone(); let tokenization::SavePaymentMethodDataResponse { payment_method_id, @@ -1099,6 +1106,8 @@ impl PostUpdateTracker, types::SetupMandateRequestDa payment_method_billing_address, business_profile, connector_mandate_reference_id, + vault_operation, + payment_method_info, )) .await?; diff --git a/crates/router/src/core/payments/operations/payment_session.rs b/crates/router/src/core/payments/operations/payment_session.rs index f93327b275cf..76937c10b5ee 100644 --- a/crates/router/src/core/payments/operations/payment_session.rs +++ b/crates/router/src/core/payments/operations/payment_session.rs @@ -213,6 +213,7 @@ impl GetTracker, api::PaymentsSessionR tax_data: None, session_id: None, service_details: None, + vault_operation: None, }; let get_trackers_response = operations::GetTrackerResponse { diff --git a/crates/router/src/core/payments/operations/payment_start.rs b/crates/router/src/core/payments/operations/payment_start.rs index a895855cefc8..93b584eff707 100644 --- a/crates/router/src/core/payments/operations/payment_start.rs +++ b/crates/router/src/core/payments/operations/payment_start.rs @@ -200,6 +200,7 @@ impl GetTracker, api::PaymentsStartReq tax_data: None, session_id: None, service_details: None, + vault_operation: None, }; let get_trackers_response = operations::GetTrackerResponse { diff --git a/crates/router/src/core/payments/operations/payment_status.rs b/crates/router/src/core/payments/operations/payment_status.rs index 9aa905345c8c..66860f8caf76 100644 --- a/crates/router/src/core/payments/operations/payment_status.rs +++ b/crates/router/src/core/payments/operations/payment_status.rs @@ -522,6 +522,7 @@ async fn get_tracker_for_sync< tax_data: None, session_id: None, service_details: None, + vault_operation: None, }; let get_trackers_response = operations::GetTrackerResponse { diff --git a/crates/router/src/core/payments/operations/payment_update.rs b/crates/router/src/core/payments/operations/payment_update.rs index c75794dd17f9..fb8482877f39 100644 --- a/crates/router/src/core/payments/operations/payment_update.rs +++ b/crates/router/src/core/payments/operations/payment_update.rs @@ -492,6 +492,7 @@ impl GetTracker, api::PaymentsRequest> tax_data: None, session_id: None, service_details: None, + vault_operation: None, }; let get_trackers_response = operations::GetTrackerResponse { diff --git a/crates/router/src/core/payments/operations/payments_incremental_authorization.rs b/crates/router/src/core/payments/operations/payments_incremental_authorization.rs index 035c4e8e2ecf..36fab81e7511 100644 --- a/crates/router/src/core/payments/operations/payments_incremental_authorization.rs +++ b/crates/router/src/core/payments/operations/payments_incremental_authorization.rs @@ -172,6 +172,7 @@ impl tax_data: None, session_id: None, service_details: None, + vault_operation: None, }; let get_trackers_response = operations::GetTrackerResponse { diff --git a/crates/router/src/core/payments/operations/tax_calculation.rs b/crates/router/src/core/payments/operations/tax_calculation.rs index 859422920e16..3f1293c89b20 100644 --- a/crates/router/src/core/payments/operations/tax_calculation.rs +++ b/crates/router/src/core/payments/operations/tax_calculation.rs @@ -180,6 +180,7 @@ impl tax_data: Some(tax_data), session_id: request.session_id.clone(), service_details: None, + vault_operation: None, }; let get_trackers_response = operations::GetTrackerResponse { operation: Box::new(self), diff --git a/crates/router/src/core/payments/tokenization.rs b/crates/router/src/core/payments/tokenization.rs index 94f221dd722f..d606fed2f675 100644 --- a/crates/router/src/core/payments/tokenization.rs +++ b/crates/router/src/core/payments/tokenization.rs @@ -12,7 +12,9 @@ use common_enums::{ConnectorMandateStatus, PaymentMethod}; use common_utils::{ crypto::Encryptable, ext_traits::{AsyncExt, Encode, ValueExt}, - id_type, pii, + id_type, + metrics::utils::record_operation_time, + pii, }; use error_stack::{report, ResultExt}; use masking::{ExposeInterface, Secret}; @@ -83,6 +85,8 @@ pub async fn save_payment_method( payment_method_billing_address: Option<&hyperswitch_domain_models::address::Address>, business_profile: &domain::Profile, mut original_connector_mandate_reference_id: Option, + vault_operation: Option, + payment_method_info: Option, ) -> RouterResult where FData: mandate::MandateBehaviour + Clone, @@ -202,41 +206,188 @@ where .await?; ((res, dc, None), None) } else { - pm_status = Some(common_enums::PaymentMethodStatus::from( + let payment_method_status = common_enums::PaymentMethodStatus::from( save_payment_method_data.attempt_status, - )); - let (res, dc) = Box::pin(save_in_locker( - state, - merchant_account, - payment_method_create_request.to_owned(), - )) - .await?; + ); + pm_status = Some(payment_method_status); - if is_network_tokenization_enabled { - let pm_data = &save_payment_method_data.request.get_payment_method_data(); - match pm_data { - domain::PaymentMethodData::Card(card) => { - let ( - network_token_resp, - _network_token_duplication_check, //the duplication check is discarded, since each card has only one token, handling card duplication check will be suffice - network_token_requestor_ref_id, - ) = Box::pin(save_network_token_in_locker( - state, - merchant_account, - card, - payment_method_create_request.clone(), - )) - .await?; + if let Some(payment_method_data_action) = vault_operation { + let network_token_requestor_reference_id = + payment_method_info.and_then(|pm_info| { + pm_info.network_token_requestor_reference_id.clone() + }); + match payment_method_data_action { + payments::PaymentMethodDataAction::SaveCardData(card) => { + if payment_method_status + == common_enums::PaymentMethodStatus::Inactive + { + if let (Some(nt_ref_id), Some(tokenization_service)) = ( + network_token_requestor_reference_id.clone(), + &state.conf.network_tokenization_service, + ) { + let _ = record_operation_time( + async { + network_tokenization::delete_network_token_from_tokenization_service( + state, + nt_ref_id.clone(), + &customer_id, + tokenization_service.get_inner(), + ) + .await + }, + &metrics::DELETE_NETWORK_TOKEN_TIME, + &[], + ) + .await; + } + } + let card_data = api::CardDetail { + card_number: card.card_number.clone(), + card_exp_month: card.card_exp_month.clone(), + card_exp_year: card.card_exp_year.clone(), + card_holder_name: None, + nick_name: None, + card_issuing_country: None, + card_network: card.card_network.clone(), + card_issuer: None, + card_type: None, + }; + let (res, dc) = + Box::pin(payment_methods::cards::add_card_to_locker( + state, + payment_method_create_request.to_owned(), + &card_data, + &customer_id, + merchant_account, + None, + )) + .await + .change_context(errors::ApiErrorResponse::InternalServerError) + .attach_printable("Add Card In Locker Failed")?; + + ((res, dc, None), None) + } + payments::PaymentMethodDataAction::SaveCardAndNetworkTokenData( + save_card_and_network_token_data, + ) => { + let card_data = api::CardDetail { + card_number: save_card_and_network_token_data + .card_data + .card_number + .clone(), + card_exp_month: save_card_and_network_token_data + .card_data + .card_exp_month + .clone(), + card_exp_year: save_card_and_network_token_data + .card_data + .card_exp_year + .clone(), + card_holder_name: None, + nick_name: None, + card_issuing_country: None, + card_network: save_card_and_network_token_data + .card_data + .card_network + .clone(), + card_issuer: None, + card_type: None, + }; + let (res, dc) = + Box::pin(payment_methods::cards::add_card_to_locker( + state, + payment_method_create_request.to_owned(), + &card_data, + &customer_id, + merchant_account, + None, + )) + .await + .change_context(errors::ApiErrorResponse::InternalServerError) + .attach_printable("Add Card In Locker Failed")?; + + let network_token_data = api::CardDetail { + card_number: save_card_and_network_token_data + .network_token_data + .token_number + .clone(), + card_exp_month: save_card_and_network_token_data + .network_token_data + .token_exp_month + .clone(), + card_exp_year: save_card_and_network_token_data + .network_token_data + .token_exp_year + .clone(), + card_holder_name: None, + nick_name: None, + card_issuing_country: None, + card_network: save_card_and_network_token_data + .network_token_data + .card_network + .clone(), + card_issuer: None, + card_type: None, + }; + + let card_reference = network_token_requestor_reference_id + .as_ref() + .map(|x| x.as_str()); + + let (network_token_resp, _dc) = + Box::pin(payment_methods::cards::add_card_to_locker( + state, + payment_method_create_request.to_owned(), + &network_token_data, + &customer_id, + merchant_account, + card_reference, + )) + .await + .change_context(errors::ApiErrorResponse::InternalServerError) + .attach_printable("Add Network Token In Locker Failed")?; ( - (res, dc, network_token_requestor_ref_id), - network_token_resp, + (res, dc, network_token_requestor_reference_id), + Some(network_token_resp), ) } - _ => ((res, dc, None), None), //network_token_resp is None in case of other payment methods } } else { - ((res, dc, None), None) + let (res, dc) = Box::pin(save_in_locker( + state, + merchant_account, + payment_method_create_request.to_owned(), + )) + .await?; + + if is_network_tokenization_enabled { + let pm_data = + &save_payment_method_data.request.get_payment_method_data(); + match pm_data { + domain::PaymentMethodData::Card(card) => { + let ( + network_token_resp, + _network_token_duplication_check, //the duplication check is discarded, since each card has only one token, handling card duplication check will be suffice + network_token_requestor_ref_id, + ) = Box::pin(save_network_token_in_locker( + state, + merchant_account, + card, + payment_method_create_request.clone(), + )) + .await?; + + ( + (res, dc, network_token_requestor_ref_id), + network_token_resp, + ) + } + _ => ((res, dc, None), None), //network_token_resp is None in case of other payment methods + } + } else { + ((res, dc, None), None) + } } }; let network_token_locker_id = match network_token_resp { @@ -773,6 +924,82 @@ where todo!() } +pub async fn pre_payment_tokenization( + state: &SessionState, + customer_id: Option, + payment_method_data: Option<&domain::PaymentMethodData>, +) -> RouterResult<(Option, Option)> { + let customer_id = customer_id.to_owned().get_required_value("customer_id")?; + match payment_method_data { + Some(domain::PaymentMethodData::Card(card)) => { + let network_tokenization_supported_card_networks = &state + .conf + .network_tokenization_supported_card_networks + .card_networks; + + if card + .card_network + .as_ref() + .filter(|cn| network_tokenization_supported_card_networks.contains(cn)) + .is_some() + { + match network_tokenization::make_card_network_tokenization_request( + state, + card, + &customer_id, + ) + .await + { + Ok((_token_response, network_token_requestor_ref_id)) => { + let network_tokenization_service = &state.conf.network_tokenization_service; + match ( + network_token_requestor_ref_id.clone(), + network_tokenization_service, + ) { + (Some(token_ref), Some(network_tokenization_service)) => { + let network_token = record_operation_time( + async { + network_tokenization::get_network_token( + state, + customer_id, + token_ref, + network_tokenization_service.get_inner(), + ) + .await + .inspect_err( + |e| logger::error!(error=?e, "Error while fetching token from tokenization service") + ) + .change_context(errors::ApiErrorResponse::InternalServerError) + .attach_printable("Fetch network token failed") + }, + &metrics::FETCH_NETWORK_TOKEN_TIME, + &[], + ) + .await; + match network_token { + Ok(token_response) => Ok(( + Some(token_response), + network_token_requestor_ref_id.clone(), + )), + _ => Ok((None, None)), + } + } + _ => Ok((None, None)), + } + } + Err(err) => { + logger::error!("Failed to tokenize card: {:?}", err); + Ok((None, None)) //None will be returned in case of error when calling network tokenization service + } + } + } else { + Ok((None, None)) //None will be returned in case of unsupported card network. + } + } + _ => Ok((None, None)), //network_token_resp is None in case of other payment methods + } +} + #[cfg(all( any(feature = "v1", feature = "v2"), not(feature = "payment_methods_v2") diff --git a/crates/router/src/types/api/admin.rs b/crates/router/src/types/api/admin.rs index ff72af484016..6d9eb86a555b 100644 --- a/crates/router/src/types/api/admin.rs +++ b/crates/router/src/types/api/admin.rs @@ -178,6 +178,7 @@ impl ForeignTryFrom for ProfileResponse { max_auto_retries_enabled: item.max_auto_retries_enabled, is_click_to_pay_enabled: item.is_click_to_pay_enabled, authentication_product_ids: item.authentication_product_ids, + is_tokenize_before_payment_enabled: item.is_tokenize_before_payment_enabled, }) } } @@ -384,5 +385,6 @@ pub async fn create_profile_from_merchant_account( max_auto_retries_enabled: request.max_auto_retries_enabled.map(i16::from), is_click_to_pay_enabled: request.is_click_to_pay_enabled, authentication_product_ids, + is_tokenize_before_payment_enabled: request.is_network_tokenization_enabled, })) } diff --git a/migrations/2024-12-02-113838_add_is_tokenize_before_payment_enabled_in_business_profile/down.sql b/migrations/2024-12-02-113838_add_is_tokenize_before_payment_enabled_in_business_profile/down.sql new file mode 100644 index 000000000000..56559f452ee4 --- /dev/null +++ b/migrations/2024-12-02-113838_add_is_tokenize_before_payment_enabled_in_business_profile/down.sql @@ -0,0 +1,2 @@ +-- This file should undo anything in `up.sql` +ALTER TABLE business_profile DROP COLUMN IF EXISTS is_tokenize_before_payment_enabled; \ No newline at end of file diff --git a/migrations/2024-12-02-113838_add_is_tokenize_before_payment_enabled_in_business_profile/up.sql b/migrations/2024-12-02-113838_add_is_tokenize_before_payment_enabled_in_business_profile/up.sql new file mode 100644 index 000000000000..d481781c3a8a --- /dev/null +++ b/migrations/2024-12-02-113838_add_is_tokenize_before_payment_enabled_in_business_profile/up.sql @@ -0,0 +1,2 @@ +-- Your SQL goes here +ALTER TABLE business_profile ADD COLUMN IF NOT EXISTS is_tokenize_before_payment_enabled BOOLEAN NOT NULL DEFAULT FALSE; \ No newline at end of file