fix:E-Sign 알림톡 certKey 버그 수정 및 발송 상태 추적 기능 추가
- sendATKakaotalkEx() 호출 시 존재하지 않는 certKey 파라미터 제거 (TypeError 버그) - sendAlimtalk/dispatchNotification 결과 반환 (void → array) - send/remind 응답에 notification_results 포함 - 감사 로그 metadata에 서명자별 알림 발송 결과 저장 - EsignPublicController 다음 서명자/완료 알림에도 동일 수정 적용 - detail.blade.php: 발송 방식 배지, 서명자 연락처, 알림 오류 배너, 활동 로그 발송 결과 표시 - send.blade.php: 발송 후 알림 실패 시 경고 메시지 표시 Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
This commit is contained in:
@@ -310,19 +310,25 @@ public function submitSignature(Request $request, string $token): JsonResponse
|
||||
// 모든 서명자에게 완료 알림 발송
|
||||
$sendMethod = $contract->send_method ?? 'alimtalk';
|
||||
foreach ($allSigners as $completedSigner) {
|
||||
$completionResults = [];
|
||||
try {
|
||||
// 이메일 발송
|
||||
if (in_array($sendMethod, ['email', 'both']) || !$completedSigner->phone) {
|
||||
if ($completedSigner->email) {
|
||||
Mail::to($completedSigner->email)->send(
|
||||
new EsignCompletedMail($contract, $completedSigner, $allSigners)
|
||||
);
|
||||
try {
|
||||
Mail::to($completedSigner->email)->send(
|
||||
new EsignCompletedMail($contract, $completedSigner, $allSigners)
|
||||
);
|
||||
$completionResults[] = ['success' => true, 'channel' => 'email', 'error' => null];
|
||||
} catch (\Throwable $e) {
|
||||
$completionResults[] = ['success' => false, 'channel' => 'email', 'error' => $e->getMessage()];
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// 알림톡 발송
|
||||
if (in_array($sendMethod, ['alimtalk', 'both']) && $completedSigner->phone) {
|
||||
$this->sendCompletionAlimtalk($contract, $completedSigner);
|
||||
$completionResults[] = $this->sendCompletionAlimtalk($contract, $completedSigner);
|
||||
}
|
||||
|
||||
EsignAuditLog::create([
|
||||
@@ -332,7 +338,14 @@ public function submitSignature(Request $request, string $token): JsonResponse
|
||||
'action' => 'completion_notification_sent',
|
||||
'ip_address' => $request->ip(),
|
||||
'user_agent' => $request->userAgent(),
|
||||
'metadata' => ['send_method' => $sendMethod],
|
||||
'metadata' => [
|
||||
'send_method' => $sendMethod,
|
||||
'notification_results' => [[
|
||||
'signer_id' => $completedSigner->id,
|
||||
'signer_name' => $completedSigner->name,
|
||||
'results' => $completionResults,
|
||||
]],
|
||||
],
|
||||
'created_at' => now(),
|
||||
]);
|
||||
} catch (\Throwable $e) {
|
||||
@@ -358,6 +371,8 @@ public function submitSignature(Request $request, string $token): JsonResponse
|
||||
$nextSendMethod = $contract->send_method ?? 'alimtalk';
|
||||
$nextSmsFallback = $contract->sms_fallback ?? true;
|
||||
|
||||
$notificationResults = [];
|
||||
|
||||
// 알림톡 발송
|
||||
if (in_array($nextSendMethod, ['alimtalk', 'both']) && $nextSigner->phone) {
|
||||
try {
|
||||
@@ -367,9 +382,8 @@ public function submitSignature(Request $request, string $token): JsonResponse
|
||||
$barobill->setServerMode($member->is_test_mode ? 'test' : 'production');
|
||||
$nextSignUrl = config('app.url') . '/esign/sign/' . $nextSigner->access_token;
|
||||
$nextExpires = $contract->expires_at?->format('Y-m-d H:i') ?? '없음';
|
||||
$barobill->sendATKakaotalkEx(
|
||||
$atResult = $barobill->sendATKakaotalkEx(
|
||||
corpNum: $member->corp_num,
|
||||
certKey: $member->cert_key,
|
||||
senderId: $member->kakaotalk_sender_id ?? '',
|
||||
templateName: '전자계약_서명요청',
|
||||
receiverName: $nextSigner->name,
|
||||
@@ -379,16 +393,30 @@ public function submitSignature(Request $request, string $token): JsonResponse
|
||||
buttons: [['Name' => '계약서 확인하기', 'ButtonType' => 'WL', 'Url1' => $nextSignUrl, 'Url2' => $nextSignUrl]],
|
||||
smsMessage: $nextSmsFallback ? "[SAM] {$nextSigner->name}님, 전자계약 서명 요청이 도착했습니다. {$nextSignUrl}" : '',
|
||||
);
|
||||
$notificationResults[] = [
|
||||
'success' => $atResult['success'] ?? false,
|
||||
'channel' => 'alimtalk',
|
||||
'error' => $atResult['error'] ?? null,
|
||||
];
|
||||
} else {
|
||||
$notificationResults[] = ['success' => false, 'channel' => 'alimtalk', 'error' => '바로빌 회원 미등록'];
|
||||
}
|
||||
} catch (\Throwable $e) {
|
||||
Log::warning('다음 서명자 알림톡 발송 실패', ['error' => $e->getMessage()]);
|
||||
$notificationResults[] = ['success' => false, 'channel' => 'alimtalk', 'error' => $e->getMessage()];
|
||||
}
|
||||
}
|
||||
|
||||
// 이메일 발송
|
||||
if (in_array($nextSendMethod, ['email', 'both']) || ($nextSendMethod === 'alimtalk' && !$nextSigner->phone)) {
|
||||
if ($nextSigner->email) {
|
||||
Mail::to($nextSigner->email)->send(new EsignRequestMail($contract, $nextSigner));
|
||||
try {
|
||||
Mail::to($nextSigner->email)->send(new EsignRequestMail($contract, $nextSigner));
|
||||
$notificationResults[] = ['success' => true, 'channel' => 'email', 'error' => null];
|
||||
} catch (\Throwable $e) {
|
||||
Log::warning('다음 서명자 이메일 발송 실패', ['error' => $e->getMessage()]);
|
||||
$notificationResults[] = ['success' => false, 'channel' => 'email', 'error' => $e->getMessage()];
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -399,7 +427,14 @@ public function submitSignature(Request $request, string $token): JsonResponse
|
||||
'action' => 'sign_request_sent',
|
||||
'ip_address' => $request->ip(),
|
||||
'user_agent' => $request->userAgent(),
|
||||
'metadata' => ['triggered_by' => 'auto_after_sign'],
|
||||
'metadata' => [
|
||||
'triggered_by' => 'auto_after_sign',
|
||||
'notification_results' => [[
|
||||
'signer_id' => $nextSigner->id,
|
||||
'signer_name' => $nextSigner->name,
|
||||
'results' => $notificationResults,
|
||||
]],
|
||||
],
|
||||
'created_at' => now(),
|
||||
]);
|
||||
}
|
||||
@@ -484,12 +519,12 @@ public function downloadDocument(string $token): StreamedResponse|JsonResponse
|
||||
|
||||
// ─── Private ───
|
||||
|
||||
private function sendCompletionAlimtalk(EsignContract $contract, EsignSigner $signer): void
|
||||
private function sendCompletionAlimtalk(EsignContract $contract, EsignSigner $signer): array
|
||||
{
|
||||
try {
|
||||
$member = BarobillMember::where('tenant_id', $contract->tenant_id)->first();
|
||||
if (!$member) {
|
||||
return;
|
||||
return ['success' => false, 'channel' => 'alimtalk', 'error' => '바로빌 회원 미등록'];
|
||||
}
|
||||
|
||||
$barobill = app(BarobillService::class);
|
||||
@@ -498,9 +533,8 @@ private function sendCompletionAlimtalk(EsignContract $contract, EsignSigner $si
|
||||
$signUrl = config('app.url') . '/esign/sign/' . $signer->access_token;
|
||||
$completedAt = $contract->completed_at?->format('Y-m-d H:i') ?? now()->format('Y-m-d H:i');
|
||||
|
||||
$barobill->sendATKakaotalkEx(
|
||||
$result = $barobill->sendATKakaotalkEx(
|
||||
corpNum: $member->corp_num,
|
||||
certKey: $member->cert_key,
|
||||
senderId: $member->kakaotalk_sender_id ?? '',
|
||||
templateName: '전자계약_완료',
|
||||
receiverName: $signer->name,
|
||||
@@ -519,12 +553,24 @@ private function sendCompletionAlimtalk(EsignContract $contract, EsignSigner $si
|
||||
? "[SAM] {$signer->name}님, 전자계약이 완료되었습니다. {$signUrl}"
|
||||
: '',
|
||||
);
|
||||
|
||||
if (!($result['success'] ?? false)) {
|
||||
Log::warning('E-Sign 완료 알림톡 발송 실패', [
|
||||
'contract_id' => $contract->id,
|
||||
'signer_id' => $signer->id,
|
||||
'error' => $result['error'] ?? 'Unknown error',
|
||||
]);
|
||||
return ['success' => false, 'channel' => 'alimtalk', 'error' => $result['error'] ?? 'API 호출 실패'];
|
||||
}
|
||||
|
||||
return ['success' => true, 'channel' => 'alimtalk', 'error' => null];
|
||||
} catch (\Throwable $e) {
|
||||
Log::warning('E-Sign 완료 알림톡 발송 실패', [
|
||||
'contract_id' => $contract->id,
|
||||
'signer_id' => $signer->id,
|
||||
'error' => $e->getMessage(),
|
||||
]);
|
||||
return ['success' => false, 'channel' => 'alimtalk', 'error' => $e->getMessage()];
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
Reference in New Issue
Block a user