-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathcutter.py
executable file
·360 lines (272 loc) · 11.4 KB
/
cutter.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
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
#!/usr/bin/python3
import glob
import os
import sys
from tinytag import TinyTag
import argparse
import signal
import platform
import tempfile
import shutil
import remsi
import json
from pathlib import Path
# costanti di default
NOISE = -40
DURATION = 0.80
WORKING = ""
ORIGINAL_FILES_DESTINATION_FOLDER = "originals"
EDITED_FILES_DESTINATION_FOLDER = "edited"
EDITED_FILE_MARK = "[EDIT]"
TMP_FILE_MARK = "[TMP]"
# necessario per l'export di windows
try:
if (os.environ["EXPORT"] == "win"):
base_path = sys._MEIPASS
FFMPEG_CMD = base_path + "\\ffmpeg.exe"
FFPROBE_CMD = base_path + "\\ffprobe.exe -v quiet -print_format json -show_format -show_streams"
elif (os.environ["EXPORT"] == "linux"):
base_path = sys._MEIPASS
FFMPEG_CMD = base_path + "/ffmpeg"
FFPROBE_CMD = base_path + "/ffprobe -v quiet -print_format json -show_format -show_streams"
except:
FFMPEG_CMD = "ffmpeg"
FFPROBE_CMD = "ffprobe -v quiet -print_format json -show_format -show_streams"
# ==================================
# calculate difference between two video duration
def durationDiff(original, edited):
try:
ffprobe_json1 = json.loads(os.popen(f'{FFPROBE_CMD} "{original}"').read())
ffprobe_json2 = json.loads(os.popen(f'{FFPROBE_CMD} "{edited}"').read())
if "streams" in ffprobe_json1 and "duration" in ffprobe_json1["streams"][0]:
video1 = int(float(ffprobe_json1["streams"][0]["duration"]))
video2 = int(float(ffprobe_json2["streams"][0]["duration"]))
elif "format" in ffprobe_json1 and "duration" in ffprobe_json1["format"]:
video1 = int(float(ffprobe_json1["format"]["duration"]))
video2 = int(float(ffprobe_json2["format"]["duration"]))
else:
return "??:??"
minutes = str(int((abs(video1 - video2))/60))
seconds = f"{int((abs(video1 - video2))%60):02d}"
return minutes + ":" + seconds
except Exception as e:
print(f"ffprobe error: {e}")
print(original)
print(edited)
return "??:??"
# argomenti
parser = argparse.ArgumentParser()
parser.add_argument("path", help="the path of the video or the folder (for many videos) to cut")
parser.add_argument('-d', type=float, required=False, default=DURATION, help='duration of silence in seconds')
parser.add_argument('-n', type=int, choices=range(-80, -19), metavar="[-80,-20]", required=False, default=NOISE, help='noise level in dB (from -80 to -20)')
parser.add_argument('-fr', type=int, required=False, help='output video frame rate')
parser.add_argument('-x', type=float, default=-1, required=False, help='Speed of the video')
parser.add_argument('-vfr', default=False, required=False, action="store_true", help='variable frame rate on output (if -fr is given, it indicates the max number of duplicates frames per second)')
parser.add_argument('--keep-cfr', default=False, required=False, action="store_true", help='keeps Constant Frame Rate version of the file for future editing.')
parser.add_argument('--preview', default=False, required=False, action="store_true", help='makes a preview of 180 seconds')
args = parser.parse_args()
# print text at center of screen
def print_centered(text):
print("\033[1;37;40m")
print(text.center(os.get_terminal_size().columns))
print("\033[0;37;40m")
# print full line of "="
def print_line():
print("─"*os.get_terminal_size().columns)
# stampa le informazioni sui comandi eseguiti
def fancy_print(arg1, arg2=""):
print_line()
print_centered(arg1)
# print("\x1B[3m"+arg2+"\x1B[0m")
print_line()
print("\n")
# GESTIONE CHIUSURA IMPROVVISA
def signal_handler(sig, frame):
print()
print_line()
print_centered("⛔️ \033[91mForced close\033[0m detected ⛔️")
print_centered("In progress: " + WORKING)
if (WORKING != ""):
filename, file_extension = os.path.splitext(WORKING)
# elimino i file non completati
try:
os.remove(f'{filename}[JUNK]{file_extension}')
except:
pass
try:
os.remove(f'{filename}[CUT]{file_extension}')
except:
pass
try:
os.remove(f'{filename}[TMP]{file_extension}')
except:
pass
print_line()
print("\n\n")
sys.exit(0)
# handler di CTRL+C
signal.signal(signal.SIGINT, signal_handler)
# controlla se un video è a framerate variabile
def check_variable_framerate(media_file):
try:
ffprobe_json = json.loads(os.popen(f'{FFPROBE_CMD} "{media_file}"').read())
denom = int(ffprobe_json["streams"][0]["avg_frame_rate"].split("/")[1])
# se il denominatore è 1, allora il framerate è statico
if(denom == 1):
return False
# altrimenti è variabile
return True
# se si verifica qualche problema continuo ipotizzando
# che il file sia cfr
except Exception as e:
print(f"CFR not checked correctly! {e}")
return True
def check_mp4(filename):
if (filename[-4:] == ".mp4"):
return True
return False
# converte un video a framerate variabile in un video a framerate statico
def convert_to_cfr(_input_file, _output_file):
# se è già stato correttamente processato
if ( os.path.exists(_output_file) and TinyTag.get(_output_file).comment == "cfr version"):
return
filename, file_extension = os.path.splitext(_input_file)
name = os.path.basename(filename)
tmp_path= tempfile.gettempdir()
_tmp_file = f"{tmp_path}/{name}{TMP_FILE_MARK}{file_extension}"
#_tmp_output = f"{tmp_path}/{name}{file_extension}" # creo il nome del file di output temporaneo
# decode delle opzioni
_fr = f"-r {args.fr}" if (args.fr) else "" # framerate
_t = f"-t 180" if (args.preview) else ""
_ignore_chapters = "-ignore_chapters 1" if (check_mp4(filename)) else ""
# effettua la conversione
command = f"{FFMPEG_CMD} -y {_ignore_chapters} -i \"{_input_file}\" {_t} -hide_banner -loglevel info -vsync cfr -metadata comment=\"cfr version\" {_fr} -preset ultrafast \"{_tmp_file}\""
fancy_print(f"🔧 \033[35mConverting\033[0m file in CFR ({_input_file})", command)
os.system(command)
shutil.move(_tmp_file, _output_file)
def speed(_input_file):
filename, file_extension = os.path.splitext(_input_file) # recupero il filename ed estensione
name = os.path.basename(filename)
tmp_path= tempfile.gettempdir()
_tmp_output = f"{tmp_path}/{name}{file_extension}" # creo il nome del file di output temporaneo
# decode
_t = f"-t 180" if (args.preview) else "" # modalità preview
_ignore_chapters = "-ignore_chapters 1" if (check_mp4(filename)) else ""
pts = 1/args.x
command = f'{FFMPEG_CMD} -y {_ignore_chapters} -i "{_input_file}" {_t} -hide_banner -filter:v "setpts=PTS*{pts}" -filter:a "atempo={args.x}" -preset ultrafast "{_tmp_output}"'
fancy_print(f"🚀 \033[33mSpeeding up\033[0m video ({_input_file})", command)
os.system(command)
# os.replace(_tmp_output, _input_file)
shutil.move(_tmp_output, _input_file)
# taglia il file
def cut(__file__):
filename, file_extension = os.path.splitext(__file__) # recupero il filename ed estensione
output = f"{filename}{TMP_FILE_MARK}{file_extension}" # creo il nome del file di output
input_file = __file__
if(check_variable_framerate(__file__)):
input_file = f"{filename}[CFR]{file_extension}"
convert_to_cfr(__file__, input_file)
# decoding delle impostazioni per ffmpeg
_fr = f"-r {args.fr}" if (args.fr) else "" # framerate
_vfr = f"-vsync vfr" if (args.vfr) else "" # framerate
_t = f"-t 180" if (args.preview) else "" # modalità preview
# eseguo remsi per la rilevazione dei silenzi
fancy_print(f"💡 \033[93mCreating\033[0m the cutting command ({__file__})")
afilter, vfilter = remsi.elaborate(input_file, FFMPEG_CMD, args.n, args.d)
name = os.path.basename(filename)
tmp_path = tempfile.gettempdir()
_tmp_output = f"{tmp_path}/{name}{TMP_FILE_MARK}{file_extension}"
_ignore_chapters = "-ignore_chapters 1" if (check_mp4(filename)) else ""
# eseguo il comando di taglio
command = f'{FFMPEG_CMD} -y {_ignore_chapters} -i "{input_file}" {_t} -hide_banner -filter_script:v "{vfilter}" -filter_script:a "{afilter}" {_vfr} {_fr} -metadata comment="edited" "{_tmp_output}"'
fancy_print(f"✂️ \033[94mCutting\033[0m file ({__file__})", command)
os.system(command)
shutil.move(_tmp_output, output)
# elimino il CFR nel caso fosse una preview e se il file esiste
if( ( (not args.keep_cfr and os.path.exists(f"{filename}[CFR]{file_extension}") ) or args.preview) and os.path.exists(f"{filename}[CFR]{file_extension}")):
try:
os.remove(f"{filename}[CFR]{file_extension}")
except:
print(f"\n\033[33m[WARN]\033[0m The file {filename}[CFR]{file_extension} hasn't been deleted because was in progress by another process.")
# elimino i file temporanei relativi al taglio
if os.path.exists(f"{afilter}"):
os.remove(f"{afilter}")
if os.path.exists(f"{vfilter}"):
os.remove(f"{vfilter}")
# restituisci il nome del file di output
return output
# funzione che riordina in modo alfanumerico
def sort(lst):
lst = [str(i) for i in lst]
lst.sort()
lst = [int(i) if i.isdigit() else i for i in lst ]
return lst
# =================== MAIN ===================
if __name__ == "__main__":
path = args.path # recupero il nome della cartella
location = os.path.abspath(path) # recupero la posizione della cartella
# se il path è una cartella
if os.path.isdir(location):
folder = location
# se il path è un file
elif os.path.isfile(location):
folder = os.path.dirname(location)
# path errato
else:
print("Error, file not valid.")
sys.exit(0)
# creo la cartella fatti se non presente
if not os.path.exists(folder+f"/{ORIGINAL_FILES_DESTINATION_FOLDER}"):
os.makedirs(folder+f"/{ORIGINAL_FILES_DESTINATION_FOLDER}")
# creo la cartella cut se non presente
if not os.path.exists(folder+f"/{EDITED_FILES_DESTINATION_FOLDER}"):
os.makedirs(folder+f"/{EDITED_FILES_DESTINATION_FOLDER}")
y = []
# scansiono tutti i file della cartella
if os.path.isdir(location):
for path in os.scandir(location):
filename, file_extension = os.path.splitext(path)
if not path.is_file():
continue
if file_extension not in [".mp4", ",mkv"]:
continue
if TinyTag.get(Path(path).as_posix()).comment == "cfr version":
continue
if TinyTag.get(Path(path).as_posix()).comment == "edited":
continue
y.append(f"{filename}{file_extension}")
elif os.path.isfile(location):
y.append(location)
z = sort(y)
# per ogni video esaminato
for i in z:
WORKING = i # setto il file che sto elaborando
filename = cut(i) # eseguo il taglio
if(args.x != -1):
speed(filename)
# rimuovo i file inutili
try:
if(not args.preview):
edited_file = filename.replace(TMP_FILE_MARK, EDITED_FILE_MARK)
else:
edited_file = filename.replace(TMP_FILE_MARK, "[PREVIEW]")
# rinomino il file in definitivo
shutil.move(filename, edited_file)
# calcolo i secondi risparmiati
elapsed = durationDiff(i, edited_file)
print("\n")
fancy_print(f"✅ {i} \033[92mCompleted!\033[0m You have saved {elapsed} minutes")
if (not args.preview):
# sposto in cut il file elaborato
pos = os.path.abspath(folder + f"/{EDITED_FILES_DESTINATION_FOLDER}/" + os.path.basename(edited_file))
shutil.move(edited_file, pos)
# sposto il file originale
pos = os.path.abspath(folder + f"/{ORIGINAL_FILES_DESTINATION_FOLDER}/" + os.path.basename(i))
if (not os.path.exists(pos)):
shutil.move(i, pos)
else:
print("⚠️ The file " + i+ " hasn't been moved because was already in the originals' folder.")
except Exception as e:
print(f"error: {e}")
print("⚠️ The file " + i+ " hasn't been moved because was already in the originals' folder.")
# [TODO] se all'apertura la durata del CFR è differente da quella del video originale lo elimina e lo rigenera