GeneratePreviewVideo.php 6.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212
  1. <?php
  2. namespace App\Console\Commands;
  3. use App\Libraries\CommonHelper;
  4. use Illuminate\Console\Command;
  5. use Illuminate\Http\File;
  6. use Illuminate\Support\Carbon;
  7. use Illuminate\Support\Facades\DB;
  8. use Illuminate\Support\Facades\Log;
  9. use Dcat\Admin\Traits\HasUploadedFile;
  10. use FFMpeg\FFMpeg;
  11. use FFMpeg\Coordinate\TimeCode;
  12. /*
  13. * 生成预览10秒视频
  14. *
  15. * @author <EMAIL>
  16. * 运行命令:php artisan command:preview-video
  17. */
  18. class GeneratePreviewVideo extends Command
  19. {
  20. use HasUploadedFile;
  21. /**
  22. * The name and signature of the console command.
  23. *
  24. * @var string
  25. */
  26. protected $signature = 'command:preview-video';
  27. /**
  28. * The console command description.
  29. *
  30. * @var string
  31. */
  32. protected $description = '生成预览视频';
  33. public $timeSecond = 5;
  34. /**
  35. * Execute the console command.
  36. *
  37. * @return int
  38. */
  39. public function handle()
  40. {
  41. // $this->importVideo();
  42. // exit;
  43. $disk = $this->disk('oss');
  44. //如果目录不存在则创建
  45. $localPreviewPath = storage_path('tmp/previews');
  46. if (!file_exists($localPreviewPath)) {
  47. mkdir($localPreviewPath, 0777, true);
  48. }
  49. $reslut = DB::table('site_preview_video')
  50. ->where('status', '==', 0) // 过滤非空 video 字段
  51. ->orderBy('id') // 按主键排序确保顺序
  52. ->limit(1) // 限制处理 1 条
  53. ->get();
  54. foreach ($reslut as $item) {
  55. try {
  56. $video_oss_url = $this->ossUrl($item->video_url);
  57. $fileName = basename($video_oss_url);
  58. $fileInfo = pathinfo($fileName);
  59. //下载$video_url到本地
  60. $localFilePath = $localPreviewPath. '/'. $fileName;
  61. $this->saveLocal($video_oss_url, $localFilePath);
  62. $previewName = $fileInfo['filename'] . '_preview.' . $fileInfo['extension'];
  63. $previewPath = storage_path('tmp/previews/'.$previewName);
  64. //截取视频前5秒
  65. $this->generatePreview($localFilePath, $previewPath);
  66. //上传到OSS
  67. $uploadPath = 'videos/uploads/previews/'. date('Ym');
  68. $file = new File($previewPath);
  69. $disk->putFileAs($uploadPath, $file, $previewName);
  70. DB::table('site_preview_video')
  71. ->where('id', $item->id)
  72. ->update([
  73. 'preview_url' => $uploadPath . '/' . $previewName,
  74. 'status' => 1
  75. ]);
  76. //删除本地文件
  77. unlink($localFilePath);
  78. unlink($previewPath);
  79. } catch (\Exception $e) {
  80. DB::table('site_preview_video')
  81. ->where('id', $item->id)
  82. ->update([
  83. 'remark' => $e->getMessage(),
  84. 'status' => -1
  85. ]);
  86. }
  87. }
  88. //$this->importVideo();
  89. return dd('success');
  90. }
  91. private function saveLocal($url, $localPath) {
  92. $remote = fopen($url, 'rb');
  93. $local = fopen($localPath, 'wb');
  94. while (!feof($remote)) {
  95. fwrite($local, fread($remote, 8192)); // 8KB分块
  96. }
  97. fclose($remote);
  98. fclose($local);
  99. return $localPath;
  100. }
  101. private function ossUrl($url)
  102. {
  103. if (strpos($url, 'http:') === 0 || strpos($url, 'https:') === 0) {
  104. return $url;
  105. }
  106. return "https://mietublcom.oss-cn-hongkong.aliyuncs.com".'/'.$url;
  107. }
  108. public function generatePreview($inputUrl, $outputPath) {
  109. $ffmpeg = FFMpeg::create([
  110. 'ffmpeg.binaries' => '/usr/bin/ffmpeg',
  111. 'ffprobe.binaries' => '/usr/bin/ffprobe',
  112. 'timeout' => 300
  113. ]);
  114. $video = $ffmpeg->open($inputUrl);
  115. $video->filters()->clip(TimeCode::fromSeconds(0), TimeCode::fromSeconds($this->timeSecond));
  116. // 输出为H.264编码的MP4
  117. $format = new \FFMpeg\Format\Video\X264();
  118. $format->setAdditionalParameters([
  119. '-an' // 禁用音频
  120. ]);
  121. $video->save($format, $outputPath);
  122. return $outputPath;
  123. }
  124. /*ssssssss
  125. * 把旧数据导入新的预览表
  126. */
  127. function importVideo()
  128. {
  129. try {
  130. // 分批处理数据,避免内存溢出
  131. DB::table('site_album')
  132. ->where('video', '<>', '[]') // 过滤非空 video 字段
  133. ->orderBy('id') // 按主键排序确保顺序
  134. ->chunkById(10000, function ($albums) { // 每次处理 100 条
  135. $insertData = [];
  136. foreach ($albums as $album) {
  137. // 跳过无效 JSON 数据
  138. $videos = json_decode($album->video, true);
  139. if (json_last_error() !== JSON_ERROR_NONE || !is_array($videos)) {
  140. Log::warning("Invalid JSON in album ID: {$album->id}");
  141. continue;
  142. }
  143. foreach ($videos as $video) {
  144. // 校验必要字段存在性
  145. if (empty($video['video_src']) || empty($video['cover'])) {
  146. Log::warning("Missing video_src or cover in album ID: {$album->id}");
  147. continue;
  148. }
  149. // 处理 URL 长度限制
  150. $videoUrl = substr($video['video_src'], 0, 255);
  151. // 构建插入数据
  152. $insertData[] = [
  153. 'video_url' => $videoUrl,
  154. 'preview_url' => '',
  155. 'status' => 0,
  156. 'album_id' => $album->id,
  157. 'created_at' => Carbon::now(),
  158. 'updated_at' => Carbon::now(),
  159. ];
  160. }
  161. }
  162. // 插入剩余数据
  163. if (!empty($insertData)) {
  164. DB::table('site_preview_video')->insert($insertData);
  165. }
  166. });
  167. return true;
  168. } catch (\Exception $e) {
  169. Log::error("Video import failed: " . $e->getMessage());
  170. return false;
  171. }
  172. }
  173. }