November 28, 2024 12:23
Video
Note: Without proper reencoding the filesize will stay the same and some devices still don’t open the results - even if they support e.g. mp4
- this does not mean they support the used codec. To fix this, please re-encode the files properly - this will take much more time, but provides better filesizes and results!
As ffmpeg
is not always the most stable program, I recommend to use this script for conversion:
import os, sys, subprocess
# Config
inFolder = '.'
inExtensions = ['mkv', 'avi', 'mp4', 'flv', 'mov']
outFolder = './out'
outExtension = 'mkv' # If you change this, you must also change the ffmpeg command (e.g. if subtitles can't be preserved)
retries = 10
justCopy = False # Copy files without re-encoding
sendMail = True # At the end send a notify email, only useful if you receive the email somehow
# Script
class Task:
def __init__(self, inFile, outFile):
self.inF = inFile
self.out = outFile
self.retries = retries
def run(self) -> bool:
if os.path.isfile(self.out):
return True # Oh, we already finished that file
if self.retries > 0:
if justCopy:
cmnd = ['ffmpeg', '-fflags', '+genpts', '-i', self.inF, '-codec', 'copy', self.out]
else:
cmnd = ['ffmpeg', '-i', self.inF, '-fflags', '+genpts', '-vcodec', 'libx265', '-crf', '28', '-map', '0', '-scodec', 'copy', '-acodec', 'copy', self.out]
print('Running (' + str(self.retries) + '): ' + ' '.join(cmnd))
res = subprocess.run(cmnd).returncode
print('Done with code ' + str(res) + '...')
else:
return False
if res == 0:
return True
else:
self.retries -= 1
return False
# Queue all files
queue = []
for path, dirs, files in os.walk(inFolder):
if path.startswith(outFolder):
continue
for file in files:
name, ext = os.path.splitext(file)
ext = ext.lstrip('.')
if ext.lower() in inExtensions:
inFile = os.path.join(path, file)
outFile = os.path.join(outFolder, name + '.' + outExtension)
queue.append(Task(inFile, outFile))
# Run
os.makedirs(outFolder, exist_ok=True)
results = []
while len(queue):
task = queue.pop(0)
if task.run():
results.append((True, task))
else:
try:
os.remove(task.out)
except:
pass
if task.retries > 0:
queue.append(task)
print('Retrying...')
else:
results.append((False, task))
print(f'Progress: {len(queue)} remaining, {len(results)} processed')
# Print results
print()
ok = 0
for r in results:
if r[0]:
ok += 1
print(f'{"OK" if r[0] else "!!"}: {r[1].inF} -> {r[1].out}')
# Send email about results
if sendMail:
os.system(f"echo '{ok} passed of {len(results)}' | mail -s '{sys.argv[0]} finished' $USER")
Audio
Normalize
This normalizes to 0db
- NOTE that the parent folder is used since we don’t change the audio format extension and therefore we would loop for a infinite time…
#!/bin/bash
set -xe
mkdir -p ../normalized/
for i in *.mp3;
do name=`echo ${i%.*}`;
echo $name;
ffmpeg -i "$i" -af "volume=0dB" "../normalized/${name}.mp3";
sleep 1
done
FLAC to MP3
#!/bin/bash
set -xe
mkdir -p ./converted/
for i in *.flac;
do name=`echo ${i%.*}`;
ffmpeg -i "$i" -acodec libmp3lame "./converted/${name}.mp3";
sleep 1
done