QrcodeExportJobs.php 6.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172
  1. <?php
  2. namespace App\Jobs\OpenWork\Contactway;
  3. use Illuminate\Bus\Queueable;
  4. use Illuminate\Contracts\Queue\ShouldQueue;
  5. use Illuminate\Foundation\Bus\Dispatchable;
  6. use Illuminate\Queue\InteractsWithQueue;
  7. use Illuminate\Queue\SerializesModels;
  8. use App\Facades\Servers\Logs\Log;
  9. use App\Servers\DB\DbService;
  10. use App\Models\OpenWork\job\Records;
  11. use App\Models\OpenWork\Contactway\Qrcode as Model;
  12. use App\Servers\Aliyun\Oss;
  13. use App\Models\OpenWork\DownloadTask as DownloadTaskModel;
  14. /**
  15. * SCRM 渠道活码异步导出
  16. * @author 唐远望
  17. * @version 1.0
  18. * @date 2025-09-28
  19. */
  20. class QrcodeExportJobs implements ShouldQueue
  21. {
  22. use Dispatchable, InteractsWithQueue, Queueable, SerializesModels;
  23. protected $message_data;
  24. protected $Records;
  25. /**
  26. * Create a new job instance.
  27. *
  28. * @return void
  29. */
  30. public function __construct(array $message_data)
  31. {
  32. $this->message_data = $message_data;
  33. }
  34. public function handle()
  35. {
  36. try {
  37. // 商户ID
  38. $companyId = $this->message_data['company_id'];
  39. $file_id = $this->message_data['file_id'];
  40. $corpid = $this->message_data['corpid'];
  41. // 切换商户ID
  42. $result = (new DbService())->getConnectionNameByCompanyId($companyId);
  43. // 数据库链接失败
  44. if ( isset($result['error']) ) {
  45. Log::info('job_error', '渠道活码队列执行失败-切换数据库失败', ['data' => $this->message_data, 'error' => $result['error']]);
  46. return;
  47. }
  48. // 创建任务记录
  49. $this->Records = Records::create([
  50. 'job_id' => $companyId . '_CompanySyncTagGroupJobs',
  51. 'name' => static::class,
  52. 'payload' => json_encode($this->message_data),
  53. 'status' => 'processing',
  54. 'started_at' => now()
  55. ]);
  56. // 查询下载任务
  57. $downloadTask = DownloadTaskModel::where(['file_id' => $file_id, 'corpid' => $corpid])->first();
  58. // 下载任务不存在
  59. if (!$downloadTask) {
  60. // 删除任务记录
  61. $this->Records->delete();
  62. Log::info('job_error', '渠道活码队列执行失败-下载任务不存在', ['data' => $this->message_data]);
  63. return;
  64. }
  65. // 下载任务状态为2,则返回
  66. $oss_url = $this->download();
  67. // 上传失败
  68. if (!$oss_url) {
  69. $downloadTask->status = 2;
  70. $downloadTask->update_time = time();
  71. $downloadTask->save();
  72. }
  73. //删除任务记录
  74. if ($this->Records) {
  75. $this->Records->delete();
  76. }
  77. } catch (\Exception $e) {
  78. // 失败处理...
  79. if ($this->Records) {
  80. $this->Records->delete();
  81. }
  82. Log::info('job_error', '渠道活码队列执行失败', ['data' => $this->message_data, 'error' => $e->getMessage()]);
  83. }
  84. }
  85. /**
  86. * 执行下载打包任务
  87. * @author 唐远望
  88. * @version 1.0
  89. * @date 2025-09-28
  90. */
  91. public function download()
  92. {
  93. $Model = new Model();
  94. $ossClient = new Oss();
  95. $chunkSize = 100;
  96. // 文件ID
  97. $file_id = $this->message_data['file_id'];
  98. $companyId = $this->message_data['company_id'];
  99. $corpid = $this->message_data['corpid'];
  100. $checkedList = isset($this->message_data['request_data']['ids']) ? explode(',', $this->message_data['request_data']['ids']) : '';
  101. // 生成文件名
  102. $filename = '渠道活码_' . $file_id . '.zip';
  103. $fullPath = public_path('uploads/c/'.$companyId.'/zip');
  104. // 确保目录存在
  105. if ( !is_dir($fullPath) ) mkdir($fullPath, 0755, true);
  106. // 生成文件路径
  107. $fullPath = rtrim($fullPath,'/').'/'. $filename;
  108. // 查询数据
  109. $query = $Model->query()->where([['corpid', '=', $corpid]]);
  110. if ($checkedList) $query = $query->whereIn('id', $checkedList);
  111. $query = $query->select(['id', 'remark', 'qr_code']);
  112. // 压缩实例
  113. $zip = new \ZipArchive;
  114. // 创建压缩文件
  115. $zip->open($fullPath, \ZipArchive::CREATE | \ZipArchive::OVERWRITE);
  116. //分块处理数据
  117. $query->chunk($chunkSize, function ($chunk) use ($zip,$companyId,$Model) {
  118. // 循环数据
  119. foreach ($chunk as $item) {
  120. // 转换数据格式
  121. $item = $item->toArray();
  122. // 生成二维码图片
  123. $filePath = $Model->writeQrcodeToFile($item['qr_code'],$companyId);
  124. // 添加文件到压缩包
  125. if( $filePath ) $zip->addFile($filePath,$item['remark'] . '_' . $item['id'] . '.png');
  126. }
  127. // 每处理完一个块,释放内存
  128. unset($chunk);
  129. gc_collect_cycles();
  130. });
  131. // 关闭压缩文件
  132. $zip->close();
  133. // 上传文件到OSS
  134. $ossPath = 'uploads/c/'.$companyId.'/scrm/contactway/'.$filename;
  135. // 上传文件
  136. $oss_url = $ossClient->putObject($ossPath, file_get_contents($fullPath));
  137. // OSS上传失败
  138. if ( !$oss_url ) Log::info('job_error', '渠道活码队列执行失败-OSS上传失败', ['data' => $this->message_data]);
  139. // 删除临时文件
  140. unlink($fullPath);
  141. //查询下载任务
  142. $downloadTask = DownloadTaskModel::where(['file_id' => $file_id, 'corpid' => $corpid])->first();
  143. if ($downloadTask) {
  144. $downloadTask->url = $oss_url;
  145. $downloadTask->file_dir_name= $filename;
  146. $downloadTask->update_time = time();
  147. $downloadTask->status = 1;
  148. $downloadTask->save();
  149. }
  150. return $oss_url;
  151. }
  152. public function failed(\Throwable $exception)
  153. {
  154. Log::info('job_error', '渠道活码队列执行完全失败', ['data' => $this->message_data, 'error' => $exception]);
  155. if ($this->Records) {
  156. $this->Records->delete();
  157. }
  158. }
  159. }