new arguments
This commit is contained in:
+9
-3
@@ -1,4 +1,6 @@
|
|||||||
usage: video-randomizer.py [-h] [-o OUTPUT] [-d DURATION] [-s SAMPLE] [-p HEIGHT] [-w WIDTH] [-f FRAMERATE] [-i IGNORE] [--dry] [-q] [-qf] [--crf CRF] [-r SEED] [--ffmpeg FFMPEG] file [file ...]
|
usage: video-randomizer.py [-h] [-o OUTPUT] [-d DURATION] [-s SAMPLE] [-p HEIGHT] [-w WIDTH] [-f FRAMERATE] [-i IGNORE] [--dry] [-q] [-qf] [--crf CRF] [-r SEED]
|
||||||
|
[--ffmpeg FFMPEG] [-nc] [-na] [-ab AUDIO_BITRATE]
|
||||||
|
file [file ...]
|
||||||
|
|
||||||
randomize videos by taking small random samples and merging them together
|
randomize videos by taking small random samples and merging them together
|
||||||
|
|
||||||
@@ -14,11 +16,11 @@ optional arguments:
|
|||||||
-s SAMPLE, --sample SAMPLE
|
-s SAMPLE, --sample SAMPLE
|
||||||
floating samples duration in seconds (default: 1s)
|
floating samples duration in seconds (default: 1s)
|
||||||
-p HEIGHT, --height HEIGHT
|
-p HEIGHT, --height HEIGHT
|
||||||
output video height (default: 1080p)
|
output video height (default: 1080p if multiple videos)
|
||||||
-w WIDTH, --width WIDTH
|
-w WIDTH, --width WIDTH
|
||||||
output video height (default: auto for 16:9)
|
output video height (default: auto for 16:9)
|
||||||
-f FRAMERATE, --framerate FRAMERATE
|
-f FRAMERATE, --framerate FRAMERATE
|
||||||
output video framerate (default: 30fps)
|
output video framerate (default: 30fps if multiple videos)
|
||||||
-i IGNORE, --ignore IGNORE
|
-i IGNORE, --ignore IGNORE
|
||||||
video input content start/end ignore in % (default: 10)
|
video input content start/end ignore in % (default: 10)
|
||||||
--dry dry mode, do not output video
|
--dry dry mode, do not output video
|
||||||
@@ -27,3 +29,7 @@ optional arguments:
|
|||||||
--crf CRF libx264 Constant Rate Factor (default: 23)
|
--crf CRF libx264 Constant Rate Factor (default: 23)
|
||||||
-r SEED, --seed SEED random seed
|
-r SEED, --seed SEED random seed
|
||||||
--ffmpeg FFMPEG ffmpeg binary path (default is found on PATH)
|
--ffmpeg FFMPEG ffmpeg binary path (default is found on PATH)
|
||||||
|
-nc, --no-convert don't convert videos (default for one video, might fail on multiple)
|
||||||
|
-na, --no-audio only keep video track
|
||||||
|
-ab AUDIO_BITRATE, --audio-bitrate AUDIO_BITRATE
|
||||||
|
audio bitrate in Kbps (default: 128)
|
||||||
+67
-14
@@ -46,7 +46,7 @@ def parse_args() -> argparse.Namespace:
|
|||||||
"--height",
|
"--height",
|
||||||
type=int,
|
type=int,
|
||||||
default=None,
|
default=None,
|
||||||
help="output video height (default: 1080p)",
|
help="output video height (default: 1080p if multiple videos)",
|
||||||
)
|
)
|
||||||
parser.add_argument(
|
parser.add_argument(
|
||||||
"-w",
|
"-w",
|
||||||
@@ -58,9 +58,9 @@ def parse_args() -> argparse.Namespace:
|
|||||||
parser.add_argument(
|
parser.add_argument(
|
||||||
"-f",
|
"-f",
|
||||||
"--framerate",
|
"--framerate",
|
||||||
type=int,
|
type=float,
|
||||||
default=30,
|
default=None,
|
||||||
help="output video framerate (default: 30fps)",
|
help="output video framerate (default: 30fps if multiple videos)",
|
||||||
)
|
)
|
||||||
parser.add_argument(
|
parser.add_argument(
|
||||||
"-i",
|
"-i",
|
||||||
@@ -108,6 +108,27 @@ def parse_args() -> argparse.Namespace:
|
|||||||
default=None,
|
default=None,
|
||||||
help="ffmpeg binary path (default is found on PATH)",
|
help="ffmpeg binary path (default is found on PATH)",
|
||||||
)
|
)
|
||||||
|
parser.add_argument(
|
||||||
|
"-nc",
|
||||||
|
"--no-convert",
|
||||||
|
action="store_true",
|
||||||
|
default=False,
|
||||||
|
help="don't convert videos (default for one video, might fail on multiple)",
|
||||||
|
)
|
||||||
|
parser.add_argument(
|
||||||
|
"-na",
|
||||||
|
"--no-audio",
|
||||||
|
action="store_true",
|
||||||
|
default=False,
|
||||||
|
help="only keep video track",
|
||||||
|
)
|
||||||
|
parser.add_argument(
|
||||||
|
"-ab",
|
||||||
|
"--audio-bitrate",
|
||||||
|
type=int,
|
||||||
|
default=128,
|
||||||
|
help="audio bitrate in Kbps (default: 128)",
|
||||||
|
)
|
||||||
parser.add_argument("file", type=str, nargs="+", help="input files")
|
parser.add_argument("file", type=str, nargs="+", help="input files")
|
||||||
return parser.parse_args()
|
return parser.parse_args()
|
||||||
|
|
||||||
@@ -124,6 +145,12 @@ def get_video_frame_count(path: str) -> int:
|
|||||||
return cv2.VideoCapture(path).get(cv2.CAP_PROP_FRAME_COUNT)
|
return cv2.VideoCapture(path).get(cv2.CAP_PROP_FRAME_COUNT)
|
||||||
|
|
||||||
|
|
||||||
|
def get_framerate(path: str, args: argparse.Namespace) -> float:
|
||||||
|
if not args.no_convert:
|
||||||
|
return args.framerate
|
||||||
|
return cv2.VideoCapture(path).get(cv2.CAP_PROP_FPS)
|
||||||
|
|
||||||
|
|
||||||
def get_timestamp(frame_number: int, framerate: float) -> str:
|
def get_timestamp(frame_number: int, framerate: float) -> str:
|
||||||
t = frame_number / framerate
|
t = frame_number / framerate
|
||||||
return f"{t//60:.0f}:{t%60:.3f}"
|
return f"{t//60:.0f}:{t%60:.3f}"
|
||||||
@@ -164,9 +191,23 @@ def get_scale(args: argparse.Namespace) -> str:
|
|||||||
return f"{args.width}:{args.height}"
|
return f"{args.width}:{args.height}"
|
||||||
|
|
||||||
|
|
||||||
|
def no_convert(args: argparse.Namespace) -> bool:
|
||||||
|
if args.no_convert:
|
||||||
|
return True
|
||||||
|
if args.no_audio or args.width is not None or args.height is not None or args.framerate is not None:
|
||||||
|
return False
|
||||||
|
return len(args.file) == 1
|
||||||
|
|
||||||
|
|
||||||
# === MAIN ===
|
# === MAIN ===
|
||||||
|
|
||||||
|
|
||||||
|
def fix_arguments(args: argparse.Namespace) -> argparse.Namespace:
|
||||||
|
args.no_convert = no_convert(args)
|
||||||
|
args.framerate = args.framerate if args.framerate is not None else 30
|
||||||
|
return args
|
||||||
|
|
||||||
|
|
||||||
def get_output_file(args: argparse.Namespace) -> str:
|
def get_output_file(args: argparse.Namespace) -> str:
|
||||||
if args.output is not None:
|
if args.output is not None:
|
||||||
return args.output
|
return args.output
|
||||||
@@ -175,9 +216,9 @@ def get_output_file(args: argparse.Namespace) -> str:
|
|||||||
|
|
||||||
def get_build_dir(args: argparse.Namespace) -> str:
|
def get_build_dir(args: argparse.Namespace) -> str:
|
||||||
path = os.path.join(
|
path = os.path.join(
|
||||||
os.getcwd(), f"build_{get_scale(args).replace(':','x')}_{args.framerate}fps"
|
os.getcwd(), f"build_{get_scale(args).replace(':','x')}_{args.framerate}fps{'_na' if args.no_audio else ''}"
|
||||||
)
|
)
|
||||||
if not os.path.exists(path):
|
if not os.path.exists(path) and not args.no_convert:
|
||||||
os.mkdir(path)
|
os.mkdir(path)
|
||||||
return path
|
return path
|
||||||
|
|
||||||
@@ -197,9 +238,12 @@ def convert_video(in_path: str, out_path: str, args: argparse.Namespace) -> bool
|
|||||||
str(args.crf),
|
str(args.crf),
|
||||||
"-video_track_timescale",
|
"-video_track_timescale",
|
||||||
"90000",
|
"90000",
|
||||||
"-an",
|
|
||||||
out_path,
|
|
||||||
]
|
]
|
||||||
|
if args.no_audio:
|
||||||
|
parameters += ["-an"]
|
||||||
|
else:
|
||||||
|
parameters += ["-c:a", "aac", "-b:a", f"{args.audio_bitrate}k"]
|
||||||
|
parameters += [out_path]
|
||||||
return ffmpeg(parameters, args)
|
return ffmpeg(parameters, args)
|
||||||
|
|
||||||
|
|
||||||
@@ -208,6 +252,9 @@ def convert_all_videos(build_dir: str, args: argparse.Namespace) -> typing.List[
|
|||||||
to_convert = []
|
to_convert = []
|
||||||
for path in args.file:
|
for path in args.file:
|
||||||
if os.path.exists(path):
|
if os.path.exists(path):
|
||||||
|
if args.no_convert:
|
||||||
|
converted += [os.path.realpath(path)]
|
||||||
|
else:
|
||||||
output_path = os.path.join(build_dir, get_file_hash(path) + ".mp4")
|
output_path = os.path.join(build_dir, get_file_hash(path) + ".mp4")
|
||||||
if os.path.exists(output_path):
|
if os.path.exists(output_path):
|
||||||
converted += [output_path]
|
converted += [output_path]
|
||||||
@@ -241,17 +288,18 @@ def generate_concat_file(videos: typing.List[str], args: argparse.Namespace) ->
|
|||||||
while t < args.duration:
|
while t < args.duration:
|
||||||
file = random.choice(videos)
|
file = random.choice(videos)
|
||||||
framecount = get_video_frame_count(file)
|
framecount = get_video_frame_count(file)
|
||||||
|
framerate = get_framerate(file, args)
|
||||||
if framecount > 0:
|
if framecount > 0:
|
||||||
tmp.write(f"file '{file}'\n".encode())
|
tmp.write(f"file '{file}'\n".encode())
|
||||||
inpoint = round(
|
inpoint = round(
|
||||||
random.random() * framecount * (1 - args.ignore / 100.0 * 2)
|
random.random() * framecount * (1 - args.ignore / 100.0 * 2)
|
||||||
)
|
)
|
||||||
tmp.write(
|
tmp.write(
|
||||||
f"inpoint {get_timestamp(inpoint, args.framerate)}\n".encode()
|
f"inpoint {get_timestamp(inpoint, framerate)}\n".encode()
|
||||||
)
|
)
|
||||||
outpoint = inpoint + round(args.sample * args.framerate)
|
outpoint = inpoint + round(args.sample * framerate)
|
||||||
tmp.write(
|
tmp.write(
|
||||||
f"outpoint {get_timestamp(outpoint, args.framerate)}\n".encode()
|
f"outpoint {get_timestamp(outpoint, framerate)}\n".encode()
|
||||||
)
|
)
|
||||||
t += args.sample
|
t += args.sample
|
||||||
if not args.quiet:
|
if not args.quiet:
|
||||||
@@ -273,10 +321,13 @@ def make_output_video(
|
|||||||
"-c:v",
|
"-c:v",
|
||||||
"libx264",
|
"libx264",
|
||||||
"-async",
|
"-async",
|
||||||
"1",
|
"1"
|
||||||
"-an",
|
|
||||||
output_file,
|
|
||||||
]
|
]
|
||||||
|
if args.no_audio:
|
||||||
|
parameters += ["-an"]
|
||||||
|
else:
|
||||||
|
parameters += ["-c:a", "aac", "-b:a", f"{args.audio_bitrate}k"]
|
||||||
|
parameters += [output_file]
|
||||||
if not ffmpeg(parameters, args):
|
if not ffmpeg(parameters, args):
|
||||||
sys.exit(1)
|
sys.exit(1)
|
||||||
|
|
||||||
@@ -284,6 +335,8 @@ def make_output_video(
|
|||||||
if __name__ == "__main__":
|
if __name__ == "__main__":
|
||||||
args = parse_args()
|
args = parse_args()
|
||||||
|
|
||||||
|
args = fix_arguments(args)
|
||||||
|
|
||||||
output_file = get_output_file(args)
|
output_file = get_output_file(args)
|
||||||
|
|
||||||
build_dir = get_build_dir(args)
|
build_dir = get_build_dir(args)
|
||||||
|
|||||||
Reference in New Issue
Block a user