braintree = $braintree; $this->braintree->init(); } public function authorizeView(array $data) { $data['gateway'] = $this->braintree; $data['threeds_enable'] = $this->braintree->company_gateway->getConfigField('threeds') ? "true" : "false"; return render('gateways.braintree.credit_card.authorize', $data); } public function authorizeResponse($data): \Illuminate\Http\RedirectResponse { return back(); } /** * Credit card payment page. * * @param array $data * @return array */ private function threeDParameters(array $data) { return [ 'amount' => $data['amount_with_fee'], 'email' => $this->braintree->client->present()->email(), 'billingAddress' => [ 'givenName' => $this->braintree->client->present()->first_name() ?: $this->braintree->client->present()->name(), 'surname' => $this->braintree->client->present()->last_name() ?: '', 'phoneNumber' => $this->braintree->client->present()->phone(), 'streetAddress' => $this->braintree->client->address1 ?: '', 'extendedAddress' => $this->braintree->client->address2 ?: '', 'locality' => $this->braintree->client->city ?: '', 'postalCode' => $this->braintree->client->postal_code ?: '', 'countryCodeAlpha2' => $this->braintree->client->country ? $this->braintree->client->country->iso_3166_2 : 'US', ] ]; } public function paymentView(array $data) { $data['gateway'] = $this->braintree; $data['client_token'] = $this->braintree->gateway->clientToken()->generate(); $data['threeds'] = $this->threeDParameters($data); $data['threeds_enable'] = $this->braintree->company_gateway->getConfigField('threeds') ? "true" : "false"; if ($this->braintree->company_gateway->getConfigField('merchantAccountId')) { /** https://developer.paypal.com/braintree/docs/reference/request/client-token/generate#merchant_account_id */ $data['client_token'] = $this->braintree->gateway->clientToken()->generate([ //@phpstan-ignore-line 'merchantAccountId' => $this->braintree->company_gateway->getConfigField('merchantAccountId'), ]); } return render('gateways.braintree.credit_card.pay', $data); } /** * Process the credit card payments. * * @param PaymentResponseRequest $request * @return \Illuminate\Http\RedirectResponse|void * @throws PaymentFailed */ public function paymentResponse(PaymentResponseRequest $request) { $this->braintree->client->fresh(); $state = [ 'server_response' => json_decode($request->gateway_response), 'payment_hash' => $request->payment_hash, ]; $state = array_merge($state, $request->all()); $state['store_card'] = boolval($state['store_card']); $this->braintree->payment_hash->data = array_merge((array) $this->braintree->payment_hash->data, $state); $this->braintree->payment_hash->save(); $customer = $this->braintree->findOrCreateCustomer(); $token = $this->getPaymentToken($request->all(), $customer->id); $data = [ 'amount' => $this->braintree->payment_hash->data->amount_with_fee, //@phpstan-ignore-line 'paymentMethodToken' => $token, 'deviceData' => $state['client-data'], 'options' => [ 'submitForSettlement' => true, ], 'billing' => [ 'streetAddress' => $this->braintree->client->address1 ?: '', 'extendedAddress' => $this->braintree->client->address2 ?: '', 'locality' => $this->braintree->client->city ?: '', 'postalCode' => $this->braintree->client->postal_code ?: '', 'countryCodeAlpha2' => $this->braintree->client->country ? $this->braintree->client->country->iso_3166_2 : 'US', ] ]; if ($this->braintree->company_gateway->getConfigField('merchantAccountId')) { /** https://developer.paypal.com/braintree/docs/reference/request/transaction/sale/php#full-example */ $data['merchantAccountId'] = $this->braintree->company_gateway->getConfigField('merchantAccountId'); } try { $result = $this->braintree->gateway->transaction()->sale($data); } catch (\Exception $e) { if ($e instanceof \Braintree\Exception\Authorization) { $this->braintree->sendFailureMail(ctrans('texts.generic_gateway_error')); throw new PaymentFailed(ctrans('texts.generic_gateway_error'), $e->getCode()); } $this->braintree->sendFailureMail($e->getMessage()); throw new PaymentFailed($e->getMessage(), $e->getCode()); } if ($result->success) { $this->braintree->logSuccessfulGatewayResponse(['response' => $request->server_response, 'data' => $this->braintree->payment_hash], SystemLog::TYPE_BRAINTREE); if ($request->store_card && is_null($request->token)) { $payment_method = $this->braintree->gateway->paymentMethod()->find($token); $this->storePaymentMethod($payment_method, $customer->id); } return $this->processSuccessfulPayment($result); } $error = $result ?: 'Undefined gateway error'; return $this->processUnsuccessfulPayment($error); } private function getPaymentToken(array $data, $customerId): ?string { if (array_key_exists('token', $data) && ! is_null($data['token'])) { return $data['token']; } $gateway_response = \json_decode($data['gateway_response']); $data = [ 'customerId' => $customerId, 'paymentMethodNonce' => $gateway_response->nonce, 'options' => [ 'verifyCard' => true, ], 'billingAddress' => [ 'streetAddress' => $this->braintree->client->address1 ?: '', 'extendedAddress' => $this->braintree->client->address2 ?: '', 'locality' => $this->braintree->client->city ?: '', 'postalCode' => $this->braintree->client->postal_code ?: '', 'countryCodeAlpha2' => $this->braintree->client->country ? $this->braintree->client->country->iso_3166_2 : 'US', ] ]; if ($this->braintree->company_gateway->getConfigField('merchantAccountId')) { /** https://developer.paypal.com/braintree/docs/reference/request/transaction/sale/php#full-example */ $data['options']['verificationMerchantAccountId'] = $this->braintree->company_gateway->getConfigField('merchantAccountId'); } $response = $this->braintree->gateway->paymentMethod()->create($data); if ($response->success) { return $response->paymentMethod->token; } $this->braintree->sendFailureMail($response->message); throw new PaymentFailed($response->message); } private function processSuccessfulPayment($response) { $state = $this->braintree->payment_hash->data; $data = [ 'payment_type' => PaymentType::parseCardType(strtolower($response->transaction->creditCard['cardType'])), 'amount' => $this->braintree->payment_hash->data->amount_with_fee, 'transaction_reference' => $response->transaction->id, 'gateway_type_id' => GatewayType::CREDIT_CARD, ]; $payment = $this->braintree->createPayment($data, Payment::STATUS_COMPLETED); SystemLogger::dispatch( ['response' => $response, 'data' => $data], SystemLog::CATEGORY_GATEWAY_RESPONSE, SystemLog::EVENT_GATEWAY_SUCCESS, SystemLog::TYPE_BRAINTREE, $this->braintree->client, $this->braintree->client->company, ); return redirect()->route('client.payments.show', ['payment' => $this->braintree->encodePrimaryKey($payment->id)]); } /** * @throws PaymentFailed */ private function processUnsuccessfulPayment($response) { $this->braintree->sendFailureMail($response?->transaction?->additionalProcessorResponse); $message = [ 'server_response' => $response, 'data' => $this->braintree->payment_hash->data, ]; SystemLogger::dispatch( $message, SystemLog::CATEGORY_GATEWAY_RESPONSE, SystemLog::EVENT_GATEWAY_FAILURE, SystemLog::TYPE_BRAINTREE, $this->braintree->client, $this->braintree->client->company, ); throw new PaymentFailed($response?->transaction?->additionalProcessorResponse ?: 'Unhandled error, please contact merchant', $response?->transaction?->processorResponseCode ?: 500); } private function storePaymentMethod($method, $customer_reference) { try { $payment_meta = new \stdClass(); $payment_meta->exp_month = (string) $method->expirationMonth; $payment_meta->exp_year = (string) $method->expirationYear; $payment_meta->brand = (string) $method->cardType; $payment_meta->last4 = (string) $method->last4; $payment_meta->type = GatewayType::CREDIT_CARD; $data = [ 'payment_meta' => $payment_meta, 'token' => $method->token, 'payment_method_id' => $this->braintree->payment_hash->data->payment_method_id, ]; $this->braintree->storeGatewayToken($data, ['gateway_customer_reference' => $customer_reference]); } catch (\Exception $e) { return $this->braintree->processInternallyFailedPayment($this->braintree, $e); } } }