Using ffmpeg to obtain video durations in python

16,911

Solution 1

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

ffprobe -v error -show_entries format=duration -of default=noprint_wrappers=1:nokey=1 <input_video>

You can use the following method instead to get the duration:

def get_length(input_video):
    result = subprocess.run(['ffprobe', '-v', 'error', '-show_entries', 'format=duration', '-of', 'default=noprint_wrappers=1:nokey=1', input_video], stdout=subprocess.PIPE, stderr=subprocess.STDOUT)
    return float(result.stdout)

Solution 2

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.

Solution 3

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

Solution 4

Updated solution using ffprobe based on @llogan guidance with the pointed link:

import subprocess

def get_duration(input_video):
    cmd = ["ffprobe", "-i", input_video, "-show_entries", "format=duration",
           "-v", "quiet", "-sexagesimal", "-of", "csv=p=0"]
    return subprocess.check_output(cmd).decode("utf-8").strip()

Fragile Solution due to stderr output:

the stderr output from ffmpeg is not intended for machine parsing and is considered fragile.

I get help from the following documentation (https://codingwithcody.com/2014/05/14/get-video-duration-with-ffmpeg-and-python/) and https://stackoverflow.com/a/6239379/2402577

Actually, sed is unnecessary: ffmpeg -i file.mp4 2>&1 | grep -o -P "(?<=Duration: ).*?(?=,)"


You can use the following method to get the duration in HH:MM:SS format:

import subprocess

def get_duration(input_video):
    # cmd: ffmpeg -i file.mkv 2>&1 | grep -o -P "(?<=Duration: ).*?(?=,)"
    p1 = subprocess.Popen(['ffmpeg',  '-i', input_video], stdout=subprocess.PIPE, stderr=subprocess.STDOUT)
    p2 = subprocess.Popen(["grep",  "-o", "-P", "(?<=Duration: ).*?(?=,)"], stdin=p1.stdout, stdout=subprocess.PIPE)
    p1.stdout.close()
    return p2.communicate()[0].decode("utf-8").strip()

Example output for both: 01:37:11.83

Share:
16,911
OSK
Author by

OSK

Updated on June 19, 2022

Comments

  • OSK
    OSK about 2 years

    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!