TimeAlbumTask.py 3.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293
  1. import logging
  2. import os
  3. import subprocess
  4. from celery import shared_task
  5. from Ansjer.config import LOGGER
  6. import django
  7. # 设置 Django 的环境变量
  8. os.environ.setdefault("DJANGO_SETTINGS_MODULE", 'Ansjer.cn_config.test_settings') # 确保替换为实际的 settings 模块路径
  9. django.setup()
  10. def start_xvfb():
  11. try:
  12. # 检查 Xvfb 是否已经在运行
  13. output = subprocess.check_output("pgrep -f Xvfb", shell=True)
  14. if output:
  15. LOGGER.info('虚拟显示服务已在运行')
  16. os.environ['DISPLAY'] = ':99'
  17. else:
  18. # 启动虚拟显示服务
  19. process = subprocess.Popen(['Xvfb', ':99', '-screen', '0', '1024x768x16'])
  20. LOGGER.info('虚拟显示服务启动成功,进程ID: %s', process.pid)
  21. os.environ['DISPLAY'] = ':99'
  22. except subprocess.CalledProcessError:
  23. # 如果没有运行,启动 Xvfb
  24. process = subprocess.Popen(['Xvfb', ':99', '-screen', '0', '1024x768x16'])
  25. LOGGER.info('虚拟显示服务启动成功,进程ID: %s', process.pid)
  26. os.environ['DISPLAY'] = ':99'
  27. except Exception as e:
  28. LOGGER.error('启动虚拟显示服务失败: %s', e)
  29. @shared_task
  30. def generate_video(image_files, output_path):
  31. LOGGER.info('start开始视频生成任务')
  32. # 启动虚拟显示服务
  33. start_xvfb()
  34. try:
  35. video_files = []
  36. music_path = "/web/test/ASJServer/static/temp/music.mp3"
  37. # 定义特效列表
  38. transitions = [
  39. "BowTieHorizontal.glsl",
  40. "burn.glsl",
  41. "cube.glsl",
  42. "pinwheel.glsl",
  43. "windowslice.glsl",
  44. "Radial.glsl",
  45. "rotateTransition.glsl",
  46. "wind.glsl",
  47. "squeeze.glsl"
  48. ]
  49. # 生成相邻图片的视频
  50. for i in range(len(image_files) - 1):
  51. output_video = os.path.join(output_path, f'{i + 1}_{i + 2}.mp4')
  52. video_files.append(output_video)
  53. cmd = [
  54. 'ffmpeg', '-loop', '1', '-i', image_files[i],
  55. '-loop', '1', '-i', image_files[i + 1],
  56. '-filter_complex',
  57. f'gltransition=duration=1:offset=0.7:source=/web/test/ASJServer/static/gl-transitions/{transitions[i]}',
  58. '-t', '2', output_video
  59. ]
  60. subprocess.run(cmd, check=True)
  61. # 生成拼接列表文件
  62. with open(os.path.join(output_path, 'filelist.txt'), 'w') as f:
  63. for video in video_files:
  64. f.write(f"file '{video}'\n")
  65. # 拼接视频
  66. final_output = os.path.join(output_path, 'final_output.mp4')
  67. subprocess.run(
  68. ['ffmpeg', '-f', 'concat', '-safe', '0', '-i', '/web/test/ASJServer/static/temp/filelist.txt', '-c', 'copy',
  69. final_output],
  70. check=True)
  71. # 添加背景音乐
  72. final_output_with_music = os.path.join(output_path, 'final_output_with_music.mp4')
  73. subprocess.run(['ffmpeg', '-i', final_output, '-i', music_path, '-c:v', 'copy', '-c:a', 'aac', '-b:a', '192k',
  74. '-shortest', final_output_with_music], check=True)
  75. # 清理临时文件
  76. for video in video_files:
  77. os.remove(video)
  78. os.remove('/web/test/ASJServer/static/temp/filelist.txt')
  79. except Exception as e:
  80. LOGGER.info(f'视频合成失败: {e}')