问题描述
我正在使用 react 和 Flask/python 开发视频到音频转换器。 我收到了带有此错误的 500:
raise IOError(("MoviePy error: the file %s Could not be found!\n"
OSError: MoviePy error: the file guitar.mp4 Could not be found!
Please check that you entered the correct path.
编辑:如评论中所述,moviepy VideoFileClip 正在寻找路径。根据建议,我现在尝试将传入的视频文件写入应用程序后端的临时目录。更新后的堆栈跟踪显示了文件路径打印,但是当呈现给 VideoFileClip 时它仍然不满意。
const onSubmit = async (e) => {
e.preventDefault()
const data = new FormData()
console.log('hopefully the mp4',videoData)
data.append('mp3',videoData)
console.log('hopefully a form object with mp4',data)
const response = await fetch('/api/convert',{
method: "POST",body: data
})
if (response.ok) {
const converted = await response.json()
setMp3(converted)
console.log(mp3)
} else {
window.alert("something went wrong :(");
}
}
Here is a link to an image depicting the console output of my file upload 从 init.py
中app = Flask(__name__)
app.config.from_object(Config)
app.register_blueprint(convert,url_prefix='/api/convert')
CORS(app)
来自 converter.py
import os
from flask import Blueprint,jsonify,request
import imageio
from moviepy.editor import *
convert = Blueprint('convert',__name__)
@convert.route('',methods=['POST'])
def convert_mp4():
if request.files['mp3'].filename:
os.getcwd()
filename = request.files['mp3'].filename
print('hey its a file again',filename)
safe_filename = secure_filename(filename)
video_file = os.path.join("/temp/",safe_filename)
print('hey its the file path',video_file)
video_clip = VideoFileClip(video_file)
print('hey its the VideoFileClip',video_clip)
audio_clip = video_clip.audio
audio_clip.write_audiofile(os.path.join("/temp/",f"{safe_filename}-converted.mp3"))
video_clip.close()
audio_clip.close()
return jsonify(send_from_directory(os.path.join("/temp/",f"{safe_filename}-converted.mp3")))
else:
return {'error': 'something went wrong :('}
在下面的堆栈跟踪中,您可以看到打印视频名称的文件,关于为什么这可能不起作用的唯一其他想法是因为它在发布请求中丢失了,但是事实上它在我的{ {1}} 支票让我很困惑。
if file:
提前感谢您查看/未来的建议。 Stack Overflow 上的第一篇官方文章 :)
解决方法
我认为问题在于您如何使用 file = request.files['mp3'].filename
。
分配给 file
的值不是指向上传文件的指针。它只是文件的名称,一个字符串。只是 request.files['mp3']
是记录 here 的 werkzeug.datastructures.FileStorage
类的一个实例。
您传递该字符串的库将其解释为它们应该打开的文件的路径。
由于您没有将文件保存在任何地方,因此他们的图书馆找不到任何东西。
我不熟悉您使用的库,但他们可能有办法将内存中的文件数据直接发送给他们,而无需保存文件然后让他们再次打开它。
如果没有,那么您可能希望将文件保存到某个临时位置,然后打开库并读取文件。
,看起来python找不到guitar.mp4
:(
我似乎需要在处理之前将文件内容保存在磁盘上。查看 MoviePy
的 docs 您需要将文件名或绝对路径传递到 VideoFileClip
构造函数中,此对象将在磁盘上打开文件并在实例化后处理。
在请求中保存文件应该很简单。下面的代码应该能够处理这个
file.save(os.path.join("/path/to/some/dir",filename))
现在您可以为文件提供 VideoFileClip
一个正确的 URI。
video_clip = VideoFileClip(os.path.join("/path/to/some/dir",filename))
这就是我要为 convert_mp4
编写的内容,尽管它没有经过测试。
@convert.route('',methods=["POST"])
def convert_mp4():
if request.files.get("mp3"):
# clean the filename
safe_filename = secure_filename(request.files["mp3"].filename)
# save file to some directory on disk
request.files["mp3"].save(os.path.join("/path/to/some/dir",safe_filename))
video_clip = VideoFileClip(os.path.join("/path/to/some/dir",safe_filename))
audio_clip = video_clip.audio # convert to audio
# you may need to change the name or save to a different directory
audio_clip.write_audiofile(os.path.join("/path/to/some/dir",f"{safe_filename}.mp3"))
# close resources,maybe use a context manager for better readability
video_clip.close()
audio_clip.close()
# responds with data from a specific file
send_from_directory("/path/to/some/dir",f"{safe_filename}.mp3")
else:
return jsonify(error="File 'mp3' does not exist!")
每当您通过 Flask 将数据保存到磁盘时,您都应该使用 secure_filename
,它从 werkzeug 项目内置到烧瓶中。此函数将清理输入名称,以便攻击者无法创建恶意文件名。
我建议甚至更进一步,也许创建 2 个端点。一个用于提交数据以进行处理,第二个用于检索数据。这使您的请求保持快速,并允许 Flask 同时处理其他请求(但是您需要一些后台进程来处理转换)。
2021 年 4 月 30 日更新
我知道我们在 Discord 上解决了这个问题,但我想记录解决方案。
您的 MP4 数据未使用 save
方法保存到磁盘(请参阅 this)。你可以检查上面实现这个的代码。
一旦完成,我们现在知道这些数据在哪里,并且可以使用已知的文件路径实例化 VideoFileClip
对象,这将允许进行转换,然后您需要保存转换后的 MP3 文件在文件系统中的某个位置。
将 MP3 保存到磁盘后,您可以使用 Flask send_from_directory
函数在响应中发回数据。此响应不能包含 JSON 内容,因为内容类型已根据 MP3 文件内容设置为 audio/mpeg
。