Files
HRM-System/app/Http/Controllers/MidtransPaymentController.php
2026-04-13 08:16:56 +08:00

188 lines
6.6 KiB
PHP

<?php
namespace App\Http\Controllers;
use App\Models\Plan;
use Illuminate\Http\Request;
class MidtransPaymentController extends Controller
{
public function processPayment(Request $request)
{
$validated = validatePaymentRequest($request, [
'transaction_status' => 'required|string',
'order_id' => 'required|string',
]);
try {
$plan = Plan::findOrFail($validated['plan_id']);
$settings = getPaymentGatewaySettings();
if (!isset($settings['payment_settings']['midtrans_secret_key'])) {
return back()->withErrors(['error' => __('Midtrans not configured')]);
}
if (in_array($validated['transaction_status'], ['capture', 'settlement'])) {
processPaymentSuccess([
'user_id' => auth()->id(),
'plan_id' => $plan->id,
'billing_cycle' => $validated['billing_cycle'],
'payment_method' => 'midtrans',
'coupon_code' => $validated['coupon_code'] ?? null,
'payment_id' => $validated['order_id'],
]);
return back()->with('success', __('Payment successful and plan activated'));
}
return back()->withErrors(['error' => __('Payment failed or cancelled')]);
} catch (\Exception $e) {
return handlePaymentError($e, 'midtrans');
}
}
public function createPayment(Request $request)
{
$validated = validatePaymentRequest($request);
try {
$plan = Plan::findOrFail($validated['plan_id']);
$pricing = calculatePlanPricing($plan, $validated['coupon_code'] ?? null);
$settings = getPaymentGatewaySettings();
if (!isset($settings['payment_settings']['midtrans_secret_key'])) {
return response()->json(['error' => __('Midtrans not configured')], 400);
}
$user = auth()->user();
$orderId = 'plan_' . $plan->id . '_' . $user->id . '_' . time();
// Convert to IDR (whole numbers only, no cents)
$amount = intval($pricing['final_price']);
$paymentData = [
'transaction_details' => [
'order_id' => $orderId,
'gross_amount' => $amount
],
'credit_card' => [
'secure' => true
],
'customer_details' => [
'first_name' => $user->name ?? 'Customer',
'email' => $user->email,
],
'item_details' => [
[
'id' => $plan->id,
'price' => $amount,
'quantity' => 1,
'name' => $plan->name
]
]
];
$snapToken = $this->createSnapToken($paymentData, $settings['payment_settings']);
if ($snapToken) {
$baseUrl = $settings['payment_settings']['midtrans_mode'] === 'live'
? 'https://app.midtrans.com'
: 'https://app.sandbox.midtrans.com';
return response()->json([
'success' => true,
'snap_token' => $snapToken,
'payment_url' => $baseUrl . '/snap/v1/transactions/' . $snapToken,
'order_id' => $orderId
]);
}
throw new \Exception(__('Failed to create Midtrans snap token'));
} catch (\Exception $e) {
return response()->json(['error' => __('Payment creation failed')], 500);
}
}
public function callback(Request $request)
{
try {
$orderId = $request->input('order_id');
$transactionStatus = $request->input('transaction_status');
if ($orderId && in_array($transactionStatus, ['capture', 'settlement'])) {
$parts = explode('_', $orderId);
if (count($parts) >= 3) {
$planId = $parts[1];
$userId = $parts[2];
$plan = Plan::find($planId);
$user = \App\Models\User::find($userId);
if ($plan && $user) {
processPaymentSuccess([
'user_id' => $user->id,
'plan_id' => $plan->id,
'billing_cycle' => 'monthly',
'payment_method' => 'midtrans',
'payment_id' => $request->input('transaction_id'),
]);
}
}
}
return response()->json(['status' => 'success']);
} catch (\Exception $e) {
return response()->json(['error' => __('Callback processing failed')], 500);
}
}
private function createSnapToken($paymentData, $settings)
{
try {
$baseUrl = $settings['midtrans_mode'] === 'live'
? 'https://app.midtrans.com'
: 'https://app.sandbox.midtrans.com';
$ch = curl_init();
curl_setopt($ch, CURLOPT_URL, $baseUrl . '/snap/v1/transactions');
curl_setopt($ch, CURLOPT_POST, true);
curl_setopt($ch, CURLOPT_POSTFIELDS, json_encode($paymentData));
curl_setopt($ch, CURLOPT_HTTPHEADER, [
'Authorization: Basic ' . base64_encode($settings['midtrans_secret_key'] . ':'),
'Content-Type: application/json',
'Accept: application/json'
]);
curl_setopt($ch, CURLOPT_RETURNTRANSFER, true);
curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, false);
curl_setopt($ch, CURLOPT_TIMEOUT, 30);
$response = curl_exec($ch);
$httpCode = curl_getinfo($ch, CURLINFO_HTTP_CODE);
$curlError = curl_error($ch);
curl_close($ch);
if ($curlError) {
throw new \Exception(__('cURL Error: ') . $curlError);
}
if ($httpCode !== 201) {
throw new \Exception(__('HTTP Error: ') . $httpCode . ' - ' . $response);
}
$result = json_decode($response, true);
if (!isset($result['token'])) {
throw new \Exception(__('No token in response: ') . $response);
}
return $result['token'];
} catch (\Exception $e) {
return false;
}
}
}