Using ffmpeg to obtain video durations in python

2019-01-24 17:57发布

I've installed ffprobe using the pip ffprobe command on my PC, and installed ffmpeg from here.

However, I'm still having trouble running the code listed here.

I try to use the following code unsuccessfully.

SyntaxError: Non-ASCII character '\xe2' in file GetVideoDurations.py on line 12, but no encoding declared; see http://python.org/dev/peps/pep-0263/ for details

Does anyone know what's wrong? Am I not referencing the directories correctly? Do I need to make sure the .py and video files are in a specific location?

import subprocess

def getLength(filename):
  result = subprocess.Popen(["ffprobe", "filename"],
    stdout = subprocess.PIPE, stderr = subprocess.STDOUT)
  return [x for x in result.stdout.readlines() if "Duration" in x]

fileToWorkWith = ‪'C:\Users\PC\Desktop\Video.mkv'

getLength(fileToWorkWith)

Apologies if the question is somewhat basic. All I need is to be able to iterate over a group of video files and get their start time and end time.

Thank you!

6条回答
一纸荒年 Trace。
2楼-- · 2019-01-24 18:36

I'd suggest using FFprobe (comes with FFmpeg).

The answer Chamath gave was pretty close, but ultimately failed for me.

Just as a note, I'm using Python 3.5 and 3.6 and this is what worked for me.

import subprocess 

def get_duration(file):
    """Get the duration of a video using ffprobe."""
    cmd = 'ffprobe -i {} -show_entries format=duration -v quiet -of csv="p=0"'.format(file)
    output = subprocess.check_output(
        cmd,
        shell=True, # Let this run in the shell
        stderr=subprocess.STDOUT
    )
    # return round(float(output))  # ugly, but rounds your seconds up or down
    return float(output)

If you want to throw this function into a class and use it in Django (1.8 - 1.11), just change one line and put this function into your class, like so:

def get_duration(file):

to:

def get_duration(self, file):

Note: Using a relative path worked for me locally, but the production server required an absolute path. You can use os.path.abspath(os.path.dirname(file)) to get the path to your video or audio file.

查看更多
仙女界的扛把子
3楼-- · 2019-01-24 18:36

Python Code

<code>
cmnd = ['/root/bin/ffmpeg',  '-i', videopath]
process = subprocess.Popen(cmnd, stdout=subprocess.PIPE, stderr=subprocess.STDOUT)
stdout, stderr = process.communicate()

#This matches regex to get the time in H:M:S format
matches = re.search(r"Duration:\s{1}(?P<hours>\d+?):(?P<minutes>\d+?):(?P<seconds>\d+\.\d+?),", stdout, re.DOTALL).groupdict()
t_hour = matches['hours']
t_min  = matches['minutes']
t_sec  = matches['seconds']

t_hour_sec = int(t_hour) * 3600
t_min_sec = int(t_min) * 60
t_s_sec   = int(round(float(t_sec)))

total_sec = t_hour_sec + t_min_sec + t_s_sec

#This matches1 is to get the frame rate of a video
matches1 = re.search(r'(\d+) fps', stdout)
frame_rate = matches1.group(0) // This will give 20fps
frame_rate = matches1.group(1) //It will give 20

</code>
查看更多
Rolldiameter
4楼-- · 2019-01-24 18:41

I think Chamath's second comment answers the question: you have a strange character somewhere in your script, either because you are using a ` instead of a ' or you have a word with non-english accents, something like this.

As a remark, for what you are doing you can also try MoviePy which parses the ffmpeg output like you do (but maybe in the future I'll use Chamath's ffprobe method it looks cleaner):

import moviepy.editor as mp
duration =  mp.VideoFileClip("my_video.mp4").duration
查看更多
对你真心纯属浪费
5楼-- · 2019-01-24 18:43

we can also use ffmpeg to get the duration of any video or audio files.

To install ffmpeg follow this link

import subprocess
import re

process = subprocess.Popen(['ffmpeg',  '-i', path_of_video_file], stdout=subprocess.PIPE, stderr=subprocess.STDOUT)
stdout, stderr = process.communicate()
matches = re.search(r"Duration:\s{1}(?P<hours>\d+?):(?P<minutes>\d+?):(?P<seconds>\d+\.\d+?),", stdout, re.DOTALL).groupdict()

print matches['hours']
print matches['minutes']
print matches['seconds']
查看更多
可以哭但决不认输i
6楼-- · 2019-01-24 18:44

There is no need to iterate though the output of FFprobe. There is one simple command which result only the duration of the input file.

ffprobe -i input_audio -show_entries format=duration -v quiet -of csv="p=0"

you can use the following method instead to get the duration.

def getLength(input_video):
    result = subprocess.Popen('ffprobe -i input_video -show_entries format=duration -v quiet -of csv="p=0"', stdout=subprocess.PIPE,stderr=subprocess.STDOUT)
    output = result.communicate()
    return output[0]

Also you need to provide the absolute path to each of your file.

Hope this helps!

查看更多
再贱就再见
7楼-- · 2019-01-24 18:44

Have you tried adding the encoding? That error is typical of that, as Chamath said. Add the utf-8 encoding to your script header:

#!/usr/bin/env python
# -*- coding: utf-8 -*- 
查看更多
登录 后发表回答