-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmp4tomkv.py
executable file
·105 lines (88 loc) · 2.79 KB
/
mp4tomkv.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
#!/usr/bin/python3
import argparse
import logging
import logging.handlers
import os
import os.path
import shlex
import sys
import subprocess
from cetools import * # noqa: F403
from tagmp4 import * # noqa: F403
prog = "mp4tomkv"
version = "0.1"
author = "Carl Edman ([email protected])"
desc = """Convert all mp4 files within directory to mkv, perserving meta-data."""
parser = None
args = None
log = logging.getLogger()
def main():
for root, _, files in os.walk(args.directory):
for file in sorted(files):
f = os.path.join(root, file)
if not os.path.isfile(f):
log.warning(f"{f} is not a regular file: skipping")
continue
base, ext = os.path.splitext(f)
if ext not in set([".mp4"]):
continue
log.info(f'Processing "{f}"')
meta = get_meta_mutagen(f)
if meta["type"] not in ["tvshow", "movie"]:
log.warning(f'Type of "{f}"={meta["type"]} not recognized: skipping')
continue
xml = set_meta_mkvxml(meta)
log.debug(f"XML: {xml}")
xmlfile = f"{base}.xml"
a = ["mkvmerge", "--output", f"{base}.mkv", "--global-tags", xmlfile, "=", f]
log.info(shlex.join(a))
if not args.dryrun:
try:
with open(xmlfile, mode="wt", encoding="utf-8") as tf:
tf.write(xml)
ret = subprocess.run(a, check=False, capture_output=True, encoding="utf-8")
finally:
os.remove(xmlfile)
if ret.returncode != 0:
log.warning(
f"Converting {f} to mkv failed, skipping: {repr(ret.stderr or ret.stdout)}"
)
continue
log.info(f'Deleting "{f}"')
if not args.dryrun:
pass
# os.remove(f)
if __name__ == "__main__":
parser = argparse.ArgumentParser(
fromfile_prefix_chars="@", prog=prog, epilog="Written by: " + author
)
parser.set_defaults(loglevel=logging.WARN)
parser.add_argument("--version", action="version", version="%(prog)s " + version)
parser.add_argument(
"-v", "--verbose", dest="loglevel", action="store_const", const=logging.INFO
)
parser.add_argument(
"-d", "--debug", dest="loglevel", action="store_const", const=logging.DEBUG
)
parser.add_argument(
"--dryrun",
action="store_true",
help="do not perform operations, but only print them.",
)
parser.add_argument(
"directory",
nargs="?",
default=".",
help="Directory to convert in (default: current)",
)
args = parser.parse_args()
if args.dryrun and args.loglevel > logging.INFO:
args.loglevel = logging.INFO
log = logging.getLogger()
log.setLevel(0)
slogger = logging.StreamHandler()
slogger.setLevel(args.loglevel)
slogger.setFormatter(logging.Formatter("[%(levelname)s] %(asctime)s: %(message)s"))
log.addHandler(slogger)
sys.stdout.reconfigure(encoding="utf-8")
main()