Updated auth api for customer profile
This commit is contained in:
@@ -1,142 +1,182 @@
|
||||
## File: your_app/services/payment_service.py
|
||||
|
||||
import logging
|
||||
from authorizenet import apicontractsv1
|
||||
from authorizenet.apicontrollers import createTransactionController
|
||||
from authorizenet.apicontrollers import (
|
||||
createTransactionController,
|
||||
createCustomerProfileController,
|
||||
createCustomerPaymentProfileController
|
||||
)
|
||||
from .. import schemas
|
||||
from config import load_config
|
||||
from config import load_config # Assuming you have this
|
||||
from decimal import Decimal
|
||||
|
||||
# Set up logging
|
||||
logger = logging.getLogger(__name__)
|
||||
|
||||
# Load Authorize.net credentials from config
|
||||
# Load Authorize.net credentials
|
||||
ApplicationConfig = load_config()
|
||||
API_LOGIN_ID = ApplicationConfig.API_LOGIN_ID
|
||||
TRANSACTION_KEY = ApplicationConfig.TRANSACTION_KEY
|
||||
# For sandbox, endpoint is https://apitest.authorize.net/xml/v1/request.api
|
||||
|
||||
def safe_string_convert(value):
|
||||
"""
|
||||
Safely convert any value to string, handling lxml objects properly.
|
||||
"""
|
||||
if value is None:
|
||||
# --- NEW CIM CORE FUNCTIONS ---
|
||||
|
||||
def create_customer_profile(customer: schemas.Customer, card_info: schemas.CardCreate):
|
||||
logger.info(f"Creating Auth.Net profile for internal customer ID: {customer.id}")
|
||||
|
||||
merchantAuth = apicontractsv1.merchantAuthenticationType()
|
||||
merchantAuth.name = API_LOGIN_ID
|
||||
merchantAuth.transactionKey = TRANSACTION_KEY
|
||||
|
||||
creditCard = apicontractsv1.creditCardType(
|
||||
cardNumber=card_info.card_number,
|
||||
expirationDate=card_info.expiration_date,
|
||||
cardCode=card_info.cvv
|
||||
)
|
||||
|
||||
billTo = apicontractsv1.customerAddressType(
|
||||
firstName=customer.customer_first_name,
|
||||
lastName=customer.customer_last_name,
|
||||
address=customer.customer_address,
|
||||
city=customer.customer_town,
|
||||
zip=customer.customer_zip,
|
||||
country="USA"
|
||||
)
|
||||
|
||||
paymentProfile = apicontractsv1.customerPaymentProfileType()
|
||||
paymentProfile.billTo = billTo
|
||||
paymentProfile.payment = apicontractsv1.paymentType(creditCard=creditCard)
|
||||
|
||||
customerProfile = apicontractsv1.customerProfileType(
|
||||
merchantCustomerId=str(customer.id),
|
||||
email=customer.customer_email,
|
||||
paymentProfiles=[paymentProfile]
|
||||
)
|
||||
|
||||
request = apicontractsv1.createCustomerProfileRequest(
|
||||
merchantAuthentication=merchantAuth,
|
||||
profile=customerProfile,
|
||||
validationMode="liveMode"
|
||||
)
|
||||
|
||||
controller = createCustomerProfileController(request)
|
||||
controller.execute()
|
||||
response = controller.getresponse()
|
||||
|
||||
if response is not None and response.messages.resultCode == "Ok":
|
||||
profile_id = response.customerProfileId
|
||||
payment_id = response.customerPaymentProfileIdList[0] if response.customerPaymentProfileIdList else None
|
||||
return str(profile_id), str(payment_id) if payment_id else None
|
||||
else:
|
||||
error_msg = response.messages.message[0].text.text if response and response.messages and response.messages.message else "Unknown Error"
|
||||
logger.error(f"Failed to create customer profile: {error_msg}")
|
||||
return None, None
|
||||
|
||||
def add_payment_profile_to_customer(customer_profile_id: str, customer: schemas.Customer, card_info: schemas.CardCreate):
|
||||
logger.info(f"Adding new payment profile to Auth.Net customer profile ID: {customer_profile_id}")
|
||||
|
||||
merchantAuth = apicontractsv1.merchantAuthenticationType(name=API_LOGIN_ID, transactionKey=TRANSACTION_KEY)
|
||||
|
||||
creditCard = apicontractsv1.creditCardType(
|
||||
cardNumber=card_info.card_number,
|
||||
expirationDate=card_info.expiration_date,
|
||||
cardCode=card_info.cvv
|
||||
)
|
||||
|
||||
paymentProfile = apicontractsv1.customerPaymentProfileType(
|
||||
billTo=apicontractsv1.customerAddressType(firstName=customer.customer_first_name, lastName=customer.customer_last_name),
|
||||
payment=apicontractsv1.paymentType(creditCard=creditCard)
|
||||
)
|
||||
|
||||
request = apicontractsv1.createCustomerPaymentProfileRequest(
|
||||
merchantAuthentication=merchantAuth,
|
||||
customerProfileId=customer_profile_id,
|
||||
paymentProfile=paymentProfile,
|
||||
validationMode="liveMode"
|
||||
)
|
||||
|
||||
controller = createCustomerPaymentProfileController(request)
|
||||
controller.execute()
|
||||
response = controller.getresponse()
|
||||
|
||||
if response is not None and response.messages.resultCode == "Ok":
|
||||
return str(response.customerPaymentProfileId)
|
||||
else:
|
||||
error_msg = response.messages.message[0].text.text if response and response.messages and response.messages.message else "Unknown Error"
|
||||
logger.error(f"Failed to add payment profile: {error_msg}")
|
||||
return None
|
||||
# Check if it's an lxml object with text attribute
|
||||
if hasattr(value, 'text'):
|
||||
return value.text
|
||||
# Otherwise use standard string conversion
|
||||
return str(value)
|
||||
|
||||
def charge_credit_card(transaction: schemas.TransactionCreate):
|
||||
logger.info(f"Processing charge for amount: {transaction.charge_amount}")
|
||||
# --- NEW CHARGE FUNCTION ---
|
||||
|
||||
merchantAuth = apicontractsv1.merchantAuthenticationType()
|
||||
merchantAuth.name = API_LOGIN_ID
|
||||
merchantAuth.transactionKey = TRANSACTION_KEY
|
||||
def charge_customer_profile(customer_profile_id: str, payment_profile_id: str, transaction_req: schemas.TransactionCreateByCardID):
|
||||
logger.info(f"Charging profile {customer_profile_id} / payment {payment_profile_id} for ${transaction_req.charge_amount}")
|
||||
|
||||
creditCard = apicontractsv1.creditCardType()
|
||||
creditCard.cardNumber = transaction.card_number
|
||||
creditCard.expirationDate = transaction.expiration_date
|
||||
creditCard.cardCode = transaction.cvv
|
||||
merchantAuth = apicontractsv1.merchantAuthenticationType(name=API_LOGIN_ID, transactionKey=TRANSACTION_KEY)
|
||||
|
||||
payment = apicontractsv1.paymentType()
|
||||
payment.creditCard = creditCard
|
||||
|
||||
transactionRequest = apicontractsv1.transactionRequestType()
|
||||
transactionRequest.transactionType = "authCaptureTransaction"
|
||||
transactionRequest.amount = transaction.charge_amount # ✅ Fixed: Use charge_amount
|
||||
transactionRequest.payment = payment
|
||||
|
||||
createtransactionrequest = apicontractsv1.createTransactionRequest()
|
||||
createtransactionrequest.merchantAuthentication = merchantAuth
|
||||
createtransactionrequest.refId = "ref_id" # Optional reference ID
|
||||
createtransactionrequest.transactionRequest = transactionRequest
|
||||
profile_to_charge = apicontractsv1.profileTransAuthCaptureType(
|
||||
customerProfileId=customer_profile_id,
|
||||
paymentProfileId=payment_profile_id
|
||||
)
|
||||
|
||||
transactionRequest = apicontractsv1.transactionRequestType(
|
||||
transactionType="authCaptureTransaction",
|
||||
amount=f"{transaction_req.charge_amount:.2f}",
|
||||
profile=profile_to_charge
|
||||
)
|
||||
|
||||
# --- THIS IS THE KEY FOR LOWER RATES (LEVEL 2/3 DATA) ---
|
||||
if transaction_req.tax_amount and transaction_req.tax_amount > 0:
|
||||
transactionRequest.tax = apicontractsv1.extendedAmountType(amount=f"{transaction_req.tax_amount:.2f}", name="Sales Tax")
|
||||
|
||||
createtransactionrequest = apicontractsv1.createTransactionRequest(
|
||||
merchantAuthentication=merchantAuth,
|
||||
transactionRequest=transactionRequest
|
||||
)
|
||||
|
||||
controller = createTransactionController(createtransactionrequest)
|
||||
controller.execute()
|
||||
return controller.getresponse()
|
||||
|
||||
response = controller.getresponse()
|
||||
# --- Your existing authorize/capture functions can remain ---
|
||||
# (They are not included here for brevity but should be kept in your file if you still need them)
|
||||
|
||||
# Log response status if payment failed
|
||||
if response is not None and response.messages is not None:
|
||||
logger.info(f"Charge response: {response.messages.resultCode}")
|
||||
|
||||
if hasattr(response.messages, 'message') and len(response.messages.message) > 0:
|
||||
for msg in response.messages.message:
|
||||
logger.info(f"Message: {msg.text.text if msg.text else 'No message text'}")
|
||||
else:
|
||||
logger.error("No response from Authorize.net")
|
||||
|
||||
return response
|
||||
def authorize_customer_profile(
|
||||
customer_profile_id: str,
|
||||
payment_profile_id: str,
|
||||
transaction_req: schemas.TransactionAuthorizeByCardID
|
||||
):
|
||||
"""
|
||||
Creates an AUTH_ONLY transaction against a customer profile.
|
||||
This holds the funds but does not capture them.
|
||||
"""
|
||||
logger.info(f"Authorizing profile {customer_profile_id} / payment {payment_profile_id} for ${transaction_req.preauthorize_amount}")
|
||||
|
||||
def authorize_credit_card(transaction: schemas.TransactionAuthorize):
|
||||
logger.info(f"Processing preauthorization for amount: {transaction.preauthorize_amount}")
|
||||
merchantAuth = apicontractsv1.merchantAuthenticationType(name=API_LOGIN_ID, transactionKey=TRANSACTION_KEY)
|
||||
|
||||
merchantAuth = apicontractsv1.merchantAuthenticationType()
|
||||
merchantAuth.name = API_LOGIN_ID
|
||||
merchantAuth.transactionKey = TRANSACTION_KEY
|
||||
# Note the type here: profileTransAuthOnlyType
|
||||
profile_to_authorize = apicontractsv1.profileTransAuthOnlyType(
|
||||
customerProfileId=customer_profile_id,
|
||||
paymentProfileId=payment_profile_id
|
||||
)
|
||||
|
||||
creditCard = apicontractsv1.creditCardType()
|
||||
creditCard.cardNumber = transaction.card_number
|
||||
creditCard.expirationDate = transaction.expiration_date
|
||||
creditCard.cardCode = transaction.cvv
|
||||
|
||||
payment = apicontractsv1.paymentType()
|
||||
payment.creditCard = creditCard
|
||||
|
||||
transactionRequest = apicontractsv1.transactionRequestType()
|
||||
transactionRequest.transactionType = "authOnlyTransaction"
|
||||
transactionRequest.amount = transaction.preauthorize_amount # ✅ Fixed: Use preauthorize_amount
|
||||
transactionRequest.payment = payment
|
||||
|
||||
createtransactionrequest = apicontractsv1.createTransactionRequest()
|
||||
createtransactionrequest.merchantAuthentication = merchantAuth
|
||||
createtransactionrequest.refId = "ref_id"
|
||||
createtransactionrequest.transactionRequest = transactionRequest
|
||||
transactionRequest = apicontractsv1.transactionRequestType(
|
||||
# The key difference: transactionType is "authOnlyTransaction"
|
||||
transactionType="authOnlyTransaction",
|
||||
amount=f"{transaction_req.preauthorize_amount:.2f}",
|
||||
profile=profile_to_authorize
|
||||
)
|
||||
|
||||
# --- LEVEL 2/3 DATA IS STILL CRITICAL HERE ---
|
||||
# The initial authorization is what the card issuers use to determine your rates.
|
||||
if transaction_req.tax_amount and transaction_req.tax_amount > 0:
|
||||
transactionRequest.tax = apicontractsv1.extendedAmountType(amount=f"{transaction_req.tax_amount:.2f}", name="Sales Tax")
|
||||
|
||||
createtransactionrequest = apicontractsv1.createTransactionRequest(
|
||||
merchantAuthentication=merchantAuth,
|
||||
transactionRequest=transactionRequest
|
||||
)
|
||||
|
||||
controller = createTransactionController(createtransactionrequest)
|
||||
controller.execute()
|
||||
|
||||
response = controller.getresponse()
|
||||
|
||||
# Log response status
|
||||
if response is not None and response.messages is not None:
|
||||
logger.info(f"Preauthorization response: {response.messages.resultCode}")
|
||||
|
||||
if hasattr(response.messages, 'message') and len(response.messages.message) > 0:
|
||||
for msg in response.messages.message:
|
||||
logger.info(f"Message: {msg.text.text if msg.text else 'No message text'}")
|
||||
else:
|
||||
logger.error("No response from Authorize.net for preauthorization")
|
||||
|
||||
return response
|
||||
|
||||
def capture_authorized_transaction(transaction: schemas.TransactionCapture):
|
||||
merchantAuth = apicontractsv1.merchantAuthenticationType()
|
||||
merchantAuth.name = API_LOGIN_ID
|
||||
merchantAuth.transactionKey = TRANSACTION_KEY
|
||||
|
||||
transactionRequest = apicontractsv1.transactionRequestType()
|
||||
transactionRequest.transactionType = "priorAuthCaptureTransaction"
|
||||
transactionRequest.amount = transaction.charge_amount # ✅ Fixed: Use charge_amount
|
||||
transactionRequest.refTransId = transaction.auth_net_transaction_id
|
||||
|
||||
createtransactionrequest = apicontractsv1.createTransactionRequest()
|
||||
createtransactionrequest.merchantAuthentication = merchantAuth
|
||||
createtransactionrequest.refId = "ref_id"
|
||||
createtransactionrequest.transactionRequest = transactionRequest
|
||||
|
||||
controller = createTransactionController(createtransactionrequest)
|
||||
controller.execute()
|
||||
|
||||
response = controller.getresponse()
|
||||
|
||||
# Log response status
|
||||
if response is not None and response.messages is not None:
|
||||
logger.info(f"Capture response: {response.messages.resultCode}")
|
||||
|
||||
if hasattr(response.messages, 'message') and len(response.messages.message) > 0:
|
||||
for msg in response.messages.message:
|
||||
logger.info(f"Message: {msg.text.text if msg.text else 'No message text'}")
|
||||
else:
|
||||
logger.error("No response from Authorize.net for capture")
|
||||
|
||||
return response
|
||||
return controller.getresponse()
|
||||
Reference in New Issue
Block a user