DingTalkController.php 6.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177
  1. <?php
  2. namespace App\Http\Controllers\Api;
  3. use App\Service\DingCallbackCrypto;
  4. use App\Service\DingTalkCrypto;
  5. use App\Service\DrbService;
  6. use Illuminate\Http\Request;
  7. use Illuminate\Support\Facades\Http;
  8. use Illuminate\Support\Facades\Log;
  9. class DingTalkController extends BaseController
  10. {
  11. public function getAccessToken(Request $request)
  12. {
  13. $service = new DrbService();
  14. list($status,$data) = $service->getAccessToken();
  15. if($status){
  16. return $this->json_return(200,'',$data);
  17. }else{
  18. return $this->json_return(201,$data);
  19. }
  20. }
  21. public function getUserByCode(Request $request)
  22. {
  23. $service = new DrbService();
  24. list($status,$data) = $service->getUserByCode($request->all());
  25. if($status){
  26. return $this->json_return(200,'',$data);
  27. }else{
  28. return $this->json_return(201,$data);
  29. }
  30. }
  31. public function createProcessInstance(Request $request)
  32. {
  33. $service = new DrbService();
  34. $userData = $request->userData;
  35. list($status,$data) = $service->createProcessInstance($request->all(), $userData);
  36. if($status){
  37. return $this->json_return(200,'',$data);
  38. }else{
  39. return $this->json_return(201,$data);
  40. }
  41. }
  42. public function getTemplateFields(Request $request)
  43. {
  44. $service = new DrbService();
  45. list($status,$data) = $service->getTemplateFields($request->all());
  46. if($status){
  47. return $this->json_return(200,'',$data);
  48. }else{
  49. return $this->json_return(201,$data);
  50. }
  51. }
  52. /**
  53. * 钉钉事件回调
  54. */
  55. public function dinCallback(Request $request)
  56. {
  57. $token = config('dingtalk.token'); // 钉钉事件订阅 token
  58. $aesKey = config('dingtalk.aes_key'); // 钉钉事件订阅 aes_key
  59. $corpid = config('dingtalk.app_key'); // 钉钉事件订阅 corpid
  60. $crypt = new DingCallbackCrypto($token, $aesKey, $corpid);
  61. try {
  62. $encrypt = $request->input('encrypt');
  63. $msgSignature = $request->get('msg_signature');
  64. $timeStamp = $request->get('timestamp');
  65. $nonce = $request->get('nonce');
  66. // 1️⃣ 解密
  67. $event = $crypt->getDecryptMsg($msgSignature, $timeStamp, $nonce, $encrypt);
  68. $event = json_decode($event, true);
  69. Log::info('钉钉回调解密后的数据', $event);
  70. // 2️⃣ 处理事件
  71. if (isset($event['EventType'])) {
  72. switch ($event['EventType']) {
  73. case 'bpms_instance_change':
  74. $processInstanceId = $event['processInstanceId'] ?? null;
  75. $result = $event['result'] ?? null;
  76. Log::info('审批实例变更', compact('processInstanceId','result'));
  77. break;
  78. case 'bpms_task_change':
  79. $processInstanceId = $event['processInstanceId'] ?? null;
  80. $taskId = $event['taskId'] ?? null;
  81. $result = $event['result'] ?? null;
  82. Log::info('审批任务变更', compact('processInstanceId','taskId','result'));
  83. break;
  84. }
  85. }
  86. // 3️⃣ 返回加密 success
  87. $res = $crypt->getEncryptedMap('success');
  88. return response($res, 200)->header('Content-Type', 'application/json');
  89. } catch (\Exception $e) {
  90. Log::error('钉钉回调解密异常', ['msg' => $e->getMessage()]);
  91. return response()->json(['errcode'=>1, 'errmsg'=>'解密失败']);
  92. }
  93. return;
  94. $crypt = new DingTalkCrypto($token, $aesKey);
  95. // 1. 获取钉钉传来的参数
  96. $msgSignature = $request->get('msg_signature'); // 注意
  97. $signature = $request->get('signature');
  98. $timestamp = $request->get('timestamp');
  99. $nonce = $request->get('nonce');
  100. $encrypt = $request->input('encrypt');
  101. // 2. 解密
  102. $event = $crypt->decryptMsg($msgSignature, $timestamp, $nonce, $encrypt);
  103. if ($event === false) {
  104. Log::error('钉钉解密失败', compact('msgSignature','timestamp','nonce','encrypt'));
  105. return response()->json([
  106. 'errcode' => 40001,
  107. 'errmsg' => 'decrypt fail'
  108. ]);
  109. }
  110. Log::info('钉钉解密后的事件数据', $event);
  111. // 3. 验证 challenge
  112. if (isset($event['EventType']) && $event['EventType'] === 'check_url') {
  113. $encryptMsg = $crypt->encryptMsg($event['Random'], $timestamp, $nonce);
  114. return response()->json([
  115. "msg_signature" => $encryptMsg['msg_signature'],
  116. "timeStamp" => $timestamp,
  117. "nonce" => $nonce,
  118. "encrypt" => $encryptMsg['encrypt']
  119. ]);
  120. }
  121. // 4. 处理事件
  122. if (isset($event['EventType'])) {
  123. switch ($event['EventType']) {
  124. case 'bpms_instance_change': // 审批实例变更
  125. $processInstanceId = $event['processInstanceId'] ?? null;
  126. $result = $event['result'] ?? null; // agree / oppose / terminate
  127. Log::info("审批实例变更事件", compact('processInstanceId', 'result'));
  128. // TODO: 在这里执行你的业务逻辑,例如更新数据库状态
  129. break;
  130. case 'bpms_task_change': // 审批任务变更
  131. $processInstanceId = $event['processInstanceId'] ?? null;
  132. $taskId = $event['taskId'] ?? null;
  133. $result = $event['result'] ?? null; // agree / oppose
  134. Log::info("审批任务变更事件", compact('processInstanceId', 'taskId', 'result'));
  135. // TODO: 在这里执行你的业务逻辑,例如记录审批人操作
  136. break;
  137. default:
  138. Log::warning("未处理的事件类型", ['EventType' => $event['EventType']]);
  139. }
  140. }
  141. // 5. 返回 success(加密)
  142. $resEncrypt = $crypt->encryptMsg('success', $timestamp, $nonce);
  143. return response()->json([
  144. "msg_signature" => $resEncrypt['msg_signature'],
  145. "timeStamp" => $timestamp,
  146. "nonce" => $nonce,
  147. "encrypt" => $resEncrypt['encrypt']
  148. ]);
  149. }
  150. }