mirror of
https://github.com/alexta69/metube.git
synced 2025-04-04 20:57:45 +03:00
Merge pull request #602 from rdiaz738/master
Added different download options and url import/exports
This commit is contained in:
commit
2872e8e6ab
7 changed files with 368 additions and 66 deletions
|
@ -61,6 +61,13 @@ Certain values can be set via environment variables, using the `-e` parameter on
|
||||||
* __YTDL_OPTIONS__: Additional options to pass to youtube-dl, in JSON format. [See available options here](https://github.com/yt-dlp/yt-dlp/blob/master/yt_dlp/YoutubeDL.py#L183). They roughly correspond to command-line options, though some do not have exact equivalents here, for example `--recode-video` has to be specified via `postprocessors`. Also note that dashes are replaced with underscores.
|
* __YTDL_OPTIONS__: Additional options to pass to youtube-dl, in JSON format. [See available options here](https://github.com/yt-dlp/yt-dlp/blob/master/yt_dlp/YoutubeDL.py#L183). They roughly correspond to command-line options, though some do not have exact equivalents here, for example `--recode-video` has to be specified via `postprocessors`. Also note that dashes are replaced with underscores.
|
||||||
* __YTDL_OPTIONS_FILE__: A path to a JSON file that will be loaded and used for populating `YTDL_OPTIONS` above. Please note that if both `YTDL_OPTIONS_FILE` and `YTDL_OPTIONS` are specified, the options in `YTDL_OPTIONS` take precedence.
|
* __YTDL_OPTIONS_FILE__: A path to a JSON file that will be loaded and used for populating `YTDL_OPTIONS` above. Please note that if both `YTDL_OPTIONS_FILE` and `YTDL_OPTIONS` are specified, the options in `YTDL_OPTIONS` take precedence.
|
||||||
* __ROBOTS_TXT__: A path to a `robots.txt` file mounted in the container
|
* __ROBOTS_TXT__: A path to a `robots.txt` file mounted in the container
|
||||||
|
* __DOWNLOAD_MODE__ :This flag controls how downloads are scheduled and executed. Options are `sequential`, `concurrent`, and `limited`. Defaults to `limited`:
|
||||||
|
* `sequential`: Downloads are processed one at a time. A new download won’t start until the previous one has finished. This mode is useful for conserving system resources or ensuring downloads occur in a strict order.
|
||||||
|
* `concurrent`: Downloads are started immediately as they are added, with no built-in limit on how many run simultaneously. This mode may overwhelm your system if too many downloads start at once.
|
||||||
|
* `limited`: Downloads are started concurrently but are capped by a concurrency limit. In this mode, a semaphore is used so that at most a fixed number of downloads run at any given time.
|
||||||
|
* **MAX\_CONCURRENT\_DOWNLOADS** This flag is used only when **DOWNLOAD\_MODE** is set to **limited**.
|
||||||
|
It specifies the maximum number of simultaneous downloads allowed. For example, if set to `5`, then at most five downloads will run concurrently, and any additional downloads will wait until one of the active downloads completes. Defaults to `3`.
|
||||||
|
|
||||||
|
|
||||||
The following example value for `YTDL_OPTIONS` embeds English subtitles and chapter markers (for videos that have them), and also changes the permissions on the downloaded video and sets the file modification timestamp to the date of when it was downloaded:
|
The following example value for `YTDL_OPTIONS` embeds English subtitles and chapter markers (for videos that have them), and also changes the permissions on the downloaded video and sets the file modification timestamp to the date of when it was downloaded:
|
||||||
|
|
||||||
|
|
28
app/main.py
28
app/main.py
|
@ -43,17 +43,19 @@ class Config:
|
||||||
'CERTFILE': '',
|
'CERTFILE': '',
|
||||||
'KEYFILE': '',
|
'KEYFILE': '',
|
||||||
'BASE_DIR': '',
|
'BASE_DIR': '',
|
||||||
'DEFAULT_THEME': 'auto'
|
'DEFAULT_THEME': 'auto',
|
||||||
|
'DOWNLOAD_MODE': 'limited',
|
||||||
|
'MAX_CONCURRENT_DOWNLOADS': 3,
|
||||||
}
|
}
|
||||||
|
|
||||||
_BOOLEAN = ('DOWNLOAD_DIRS_INDEXABLE', 'CUSTOM_DIRS', 'CREATE_CUSTOM_DIRS', 'DELETE_FILE_ON_TRASHCAN', 'DEFAULT_OPTION_PLAYLIST_STRICT_MODE', 'HTTPS')
|
_BOOLEAN = ('DOWNLOAD_DIRS_INDEXABLE', 'CUSTOM_DIRS', 'CREATE_CUSTOM_DIRS', 'DELETE_FILE_ON_TRASHCAN', 'DEFAULT_OPTION_PLAYLIST_STRICT_MODE', 'HTTPS')
|
||||||
|
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
for k, v in self._DEFAULTS.items():
|
for k, v in self._DEFAULTS.items():
|
||||||
setattr(self, k, os.environ[k] if k in os.environ else v)
|
setattr(self, k, os.environ.get(k, v))
|
||||||
|
|
||||||
for k, v in self.__dict__.items():
|
for k, v in self.__dict__.items():
|
||||||
if v.startswith('%%'):
|
if isinstance(v, str) and v.startswith('%%'):
|
||||||
setattr(self, k, getattr(self, v[2:]))
|
setattr(self, k, getattr(self, v[2:]))
|
||||||
if k in self._BOOLEAN:
|
if k in self._BOOLEAN:
|
||||||
if v not in ('true', 'false', 'True', 'False', 'on', 'off', '1', '0'):
|
if v not in ('true', 'false', 'True', 'False', 'on', 'off', '1', '0'):
|
||||||
|
@ -102,18 +104,23 @@ routes = web.RouteTableDef()
|
||||||
|
|
||||||
class Notifier(DownloadQueueNotifier):
|
class Notifier(DownloadQueueNotifier):
|
||||||
async def added(self, dl):
|
async def added(self, dl):
|
||||||
|
log.info(f"Notifier: Download added - {dl.title}")
|
||||||
await sio.emit('added', serializer.encode(dl))
|
await sio.emit('added', serializer.encode(dl))
|
||||||
|
|
||||||
async def updated(self, dl):
|
async def updated(self, dl):
|
||||||
|
log.info(f"Notifier: Download updated - {dl.title}")
|
||||||
await sio.emit('updated', serializer.encode(dl))
|
await sio.emit('updated', serializer.encode(dl))
|
||||||
|
|
||||||
async def completed(self, dl):
|
async def completed(self, dl):
|
||||||
|
log.info(f"Notifier: Download completed - {dl.title}")
|
||||||
await sio.emit('completed', serializer.encode(dl))
|
await sio.emit('completed', serializer.encode(dl))
|
||||||
|
|
||||||
async def canceled(self, id):
|
async def canceled(self, id):
|
||||||
|
log.info(f"Notifier: Download canceled - {id}")
|
||||||
await sio.emit('canceled', serializer.encode(id))
|
await sio.emit('canceled', serializer.encode(id))
|
||||||
|
|
||||||
async def cleared(self, id):
|
async def cleared(self, id):
|
||||||
|
log.info(f"Notifier: Download cleared - {id}")
|
||||||
await sio.emit('cleared', serializer.encode(id))
|
await sio.emit('cleared', serializer.encode(id))
|
||||||
|
|
||||||
dqueue = DownloadQueue(config, Notifier())
|
dqueue = DownloadQueue(config, Notifier())
|
||||||
|
@ -121,10 +128,13 @@ app.on_startup.append(lambda app: dqueue.initialize())
|
||||||
|
|
||||||
@routes.post(config.URL_PREFIX + 'add')
|
@routes.post(config.URL_PREFIX + 'add')
|
||||||
async def add(request):
|
async def add(request):
|
||||||
|
log.info("Received request to add download")
|
||||||
post = await request.json()
|
post = await request.json()
|
||||||
|
log.info(f"Request data: {post}")
|
||||||
url = post.get('url')
|
url = post.get('url')
|
||||||
quality = post.get('quality')
|
quality = post.get('quality')
|
||||||
if not url or not quality:
|
if not url or not quality:
|
||||||
|
log.error("Bad request: missing 'url' or 'quality'")
|
||||||
raise web.HTTPBadRequest()
|
raise web.HTTPBadRequest()
|
||||||
format = post.get('format')
|
format = post.get('format')
|
||||||
folder = post.get('folder')
|
folder = post.get('folder')
|
||||||
|
@ -153,14 +163,17 @@ async def delete(request):
|
||||||
ids = post.get('ids')
|
ids = post.get('ids')
|
||||||
where = post.get('where')
|
where = post.get('where')
|
||||||
if not ids or where not in ['queue', 'done']:
|
if not ids or where not in ['queue', 'done']:
|
||||||
|
log.error("Bad request: missing 'ids' or incorrect 'where' value")
|
||||||
raise web.HTTPBadRequest()
|
raise web.HTTPBadRequest()
|
||||||
status = await (dqueue.cancel(ids) if where == 'queue' else dqueue.clear(ids))
|
status = await (dqueue.cancel(ids) if where == 'queue' else dqueue.clear(ids))
|
||||||
|
log.info(f"Download delete request processed for ids: {ids}, where: {where}")
|
||||||
return web.Response(text=serializer.encode(status))
|
return web.Response(text=serializer.encode(status))
|
||||||
|
|
||||||
@routes.post(config.URL_PREFIX + 'start')
|
@routes.post(config.URL_PREFIX + 'start')
|
||||||
async def start(request):
|
async def start(request):
|
||||||
post = await request.json()
|
post = await request.json()
|
||||||
ids = post.get('ids')
|
ids = post.get('ids')
|
||||||
|
log.info(f"Received request to start pending downloads for ids: {ids}")
|
||||||
status = await dqueue.start_pending(ids)
|
status = await dqueue.start_pending(ids)
|
||||||
return web.Response(text=serializer.encode(status))
|
return web.Response(text=serializer.encode(status))
|
||||||
|
|
||||||
|
@ -168,17 +181,19 @@ async def start(request):
|
||||||
async def history(request):
|
async def history(request):
|
||||||
history = { 'done': [], 'queue': [], 'pending': []}
|
history = { 'done': [], 'queue': [], 'pending': []}
|
||||||
|
|
||||||
for _ ,v in dqueue.queue.saved_items():
|
for _, v in dqueue.queue.saved_items():
|
||||||
history['queue'].append(v)
|
history['queue'].append(v)
|
||||||
for _ ,v in dqueue.done.saved_items():
|
for _, v in dqueue.done.saved_items():
|
||||||
history['done'].append(v)
|
history['done'].append(v)
|
||||||
for _ ,v in dqueue.pending.saved_items():
|
for _, v in dqueue.pending.saved_items():
|
||||||
history['pending'].append(v)
|
history['pending'].append(v)
|
||||||
|
|
||||||
|
log.info("Sending download history")
|
||||||
return web.Response(text=serializer.encode(history))
|
return web.Response(text=serializer.encode(history))
|
||||||
|
|
||||||
@sio.event
|
@sio.event
|
||||||
async def connect(sid, environ):
|
async def connect(sid, environ):
|
||||||
|
log.info(f"Client connected: {sid}")
|
||||||
await sio.emit('all', serializer.encode(dqueue.get()), to=sid)
|
await sio.emit('all', serializer.encode(dqueue.get()), to=sid)
|
||||||
await sio.emit('configuration', serializer.encode(config), to=sid)
|
await sio.emit('configuration', serializer.encode(config), to=sid)
|
||||||
if config.CUSTOM_DIRS:
|
if config.CUSTOM_DIRS:
|
||||||
|
@ -262,7 +277,6 @@ async def add_cors(request):
|
||||||
|
|
||||||
app.router.add_route('OPTIONS', config.URL_PREFIX + 'add', add_cors)
|
app.router.add_route('OPTIONS', config.URL_PREFIX + 'add', add_cors)
|
||||||
|
|
||||||
|
|
||||||
async def on_prepare(request, response):
|
async def on_prepare(request, response):
|
||||||
if 'Origin' in request.headers:
|
if 'Origin' in request.headers:
|
||||||
response.headers['Access-Control-Allow-Origin'] = request.headers['Origin']
|
response.headers['Access-Control-Allow-Origin'] = request.headers['Origin']
|
||||||
|
|
150
app/ytdl.py
150
app/ytdl.py
|
@ -63,8 +63,8 @@ class Download:
|
||||||
self.loop = None
|
self.loop = None
|
||||||
self.notifier = None
|
self.notifier = None
|
||||||
|
|
||||||
|
|
||||||
def _download(self):
|
def _download(self):
|
||||||
|
log.info(f"Starting download for: {self.info.title} ({self.info.url})")
|
||||||
try:
|
try:
|
||||||
def put_status(st):
|
def put_status(st):
|
||||||
self.status_queue.put({k: v for k, v in st.items() if k in (
|
self.status_queue.put({k: v for k, v in st.items() if k in (
|
||||||
|
@ -78,6 +78,7 @@ class Download:
|
||||||
'speed',
|
'speed',
|
||||||
'eta',
|
'eta',
|
||||||
)})
|
)})
|
||||||
|
|
||||||
def put_status_postprocessor(d):
|
def put_status_postprocessor(d):
|
||||||
if d['postprocessor'] == 'MoveFiles' and d['status'] == 'finished':
|
if d['postprocessor'] == 'MoveFiles' and d['status'] == 'finished':
|
||||||
if '__finaldir' in d['info_dict']:
|
if '__finaldir' in d['info_dict']:
|
||||||
|
@ -85,10 +86,10 @@ class Download:
|
||||||
else:
|
else:
|
||||||
filename = d['info_dict']['filepath']
|
filename = d['info_dict']['filepath']
|
||||||
self.status_queue.put({'status': 'finished', 'filename': filename})
|
self.status_queue.put({'status': 'finished', 'filename': filename})
|
||||||
|
|
||||||
ret = yt_dlp.YoutubeDL(params={
|
ret = yt_dlp.YoutubeDL(params={
|
||||||
'quiet': True,
|
'quiet': True,
|
||||||
'no_color': True,
|
'no_color': True,
|
||||||
#'skip_download': True,
|
|
||||||
'paths': {"home": self.download_dir, "temp": self.temp_dir},
|
'paths': {"home": self.download_dir, "temp": self.temp_dir},
|
||||||
'outtmpl': { "default": self.output_template, "chapter": self.output_template_chapter },
|
'outtmpl': { "default": self.output_template, "chapter": self.output_template_chapter },
|
||||||
'format': self.format,
|
'format': self.format,
|
||||||
|
@ -99,10 +100,13 @@ class Download:
|
||||||
**self.ytdl_opts,
|
**self.ytdl_opts,
|
||||||
}).download([self.info.url])
|
}).download([self.info.url])
|
||||||
self.status_queue.put({'status': 'finished' if ret == 0 else 'error'})
|
self.status_queue.put({'status': 'finished' if ret == 0 else 'error'})
|
||||||
|
log.info(f"Finished download for: {self.info.title}")
|
||||||
except yt_dlp.utils.YoutubeDLError as exc:
|
except yt_dlp.utils.YoutubeDLError as exc:
|
||||||
|
log.error(f"Download error for {self.info.title}: {str(exc)}")
|
||||||
self.status_queue.put({'status': 'error', 'msg': str(exc)})
|
self.status_queue.put({'status': 'error', 'msg': str(exc)})
|
||||||
|
|
||||||
async def start(self, notifier):
|
async def start(self, notifier):
|
||||||
|
log.info(f"Preparing download for: {self.info.title}")
|
||||||
if Download.manager is None:
|
if Download.manager is None:
|
||||||
Download.manager = multiprocessing.Manager()
|
Download.manager = multiprocessing.Manager()
|
||||||
self.status_queue = Download.manager.Queue()
|
self.status_queue = Download.manager.Queue()
|
||||||
|
@ -116,14 +120,22 @@ class Download:
|
||||||
return await self.loop.run_in_executor(None, self.proc.join)
|
return await self.loop.run_in_executor(None, self.proc.join)
|
||||||
|
|
||||||
def cancel(self):
|
def cancel(self):
|
||||||
|
log.info(f"Cancelling download: {self.info.title}")
|
||||||
if self.running():
|
if self.running():
|
||||||
self.proc.kill()
|
try:
|
||||||
|
self.proc.kill()
|
||||||
|
except Exception as e:
|
||||||
|
log.error(f"Error killing process for {self.info.title}: {e}")
|
||||||
self.canceled = True
|
self.canceled = True
|
||||||
|
if self.status_queue is not None:
|
||||||
|
self.status_queue.put(None)
|
||||||
|
|
||||||
def close(self):
|
def close(self):
|
||||||
|
log.info(f"Closing download process for: {self.info.title}")
|
||||||
if self.started():
|
if self.started():
|
||||||
self.proc.close()
|
self.proc.close()
|
||||||
self.status_queue.put(None)
|
if self.status_queue is not None:
|
||||||
|
self.status_queue.put(None)
|
||||||
|
|
||||||
def running(self):
|
def running(self):
|
||||||
try:
|
try:
|
||||||
|
@ -138,15 +150,17 @@ class Download:
|
||||||
while True:
|
while True:
|
||||||
status = await self.loop.run_in_executor(None, self.status_queue.get)
|
status = await self.loop.run_in_executor(None, self.status_queue.get)
|
||||||
if status is None:
|
if status is None:
|
||||||
|
log.info(f"Status update finished for: {self.info.title}")
|
||||||
|
return
|
||||||
|
if self.canceled:
|
||||||
|
log.info(f"Download {self.info.title} is canceled; stopping status updates.")
|
||||||
return
|
return
|
||||||
self.tmpfilename = status.get('tmpfilename')
|
self.tmpfilename = status.get('tmpfilename')
|
||||||
if 'filename' in status:
|
if 'filename' in status:
|
||||||
fileName = status.get('filename')
|
fileName = status.get('filename')
|
||||||
self.info.filename = os.path.relpath(fileName, self.download_dir)
|
self.info.filename = os.path.relpath(fileName, self.download_dir)
|
||||||
self.info.size = os.path.getsize(fileName) if os.path.exists(fileName) else None
|
self.info.size = os.path.getsize(fileName) if os.path.exists(fileName) else None
|
||||||
|
if self.info.format == 'thumbnail':
|
||||||
# Set correct file extension for thumbnails
|
|
||||||
if(self.info.format == 'thumbnail'):
|
|
||||||
self.info.filename = re.sub(r'\.webm$', '.jpg', self.info.filename)
|
self.info.filename = re.sub(r'\.webm$', '.jpg', self.info.filename)
|
||||||
self.info.status = status['status']
|
self.info.status = status['status']
|
||||||
self.info.msg = status.get('msg')
|
self.info.msg = status.get('msg')
|
||||||
|
@ -156,6 +170,7 @@ class Download:
|
||||||
self.info.percent = status['downloaded_bytes'] / total * 100
|
self.info.percent = status['downloaded_bytes'] / total * 100
|
||||||
self.info.speed = status.get('speed')
|
self.info.speed = status.get('speed')
|
||||||
self.info.eta = status.get('eta')
|
self.info.eta = status.get('eta')
|
||||||
|
log.info(f"Updating status for {self.info.title}: {status}")
|
||||||
await self.notifier.updated(self.info)
|
await self.notifier.updated(self.info)
|
||||||
|
|
||||||
class PersistentQueue:
|
class PersistentQueue:
|
||||||
|
@ -192,9 +207,10 @@ class PersistentQueue:
|
||||||
shelf[key] = value.info
|
shelf[key] = value.info
|
||||||
|
|
||||||
def delete(self, key):
|
def delete(self, key):
|
||||||
del self.dict[key]
|
if key in self.dict:
|
||||||
with shelve.open(self.path, 'w') as shelf:
|
del self.dict[key]
|
||||||
shelf.pop(key)
|
with shelve.open(self.path, 'w') as shelf:
|
||||||
|
shelf.pop(key, None)
|
||||||
|
|
||||||
def next(self):
|
def next(self):
|
||||||
k, v = next(iter(self.dict.items()))
|
k, v = next(iter(self.dict.items()))
|
||||||
|
@ -210,6 +226,14 @@ class DownloadQueue:
|
||||||
self.queue = PersistentQueue(self.config.STATE_DIR + '/queue')
|
self.queue = PersistentQueue(self.config.STATE_DIR + '/queue')
|
||||||
self.done = PersistentQueue(self.config.STATE_DIR + '/completed')
|
self.done = PersistentQueue(self.config.STATE_DIR + '/completed')
|
||||||
self.pending = PersistentQueue(self.config.STATE_DIR + '/pending')
|
self.pending = PersistentQueue(self.config.STATE_DIR + '/pending')
|
||||||
|
self.active_downloads = set()
|
||||||
|
self.semaphore = None
|
||||||
|
# For sequential mode, use an asyncio lock to ensure one-at-a-time execution.
|
||||||
|
if self.config.DOWNLOAD_MODE == 'sequential':
|
||||||
|
self.seq_lock = asyncio.Lock()
|
||||||
|
elif self.config.DOWNLOAD_MODE == 'limited':
|
||||||
|
self.semaphore = asyncio.Semaphore(self.config.MAX_CONCURRENT_DOWNLOADS)
|
||||||
|
|
||||||
self.done.load()
|
self.done.load()
|
||||||
|
|
||||||
async def __import_queue(self):
|
async def __import_queue(self):
|
||||||
|
@ -217,10 +241,56 @@ class DownloadQueue:
|
||||||
await self.add(v.url, v.quality, v.format, v.folder, v.custom_name_prefix, v.playlist_strict_mode, v.playlist_item_limit)
|
await self.add(v.url, v.quality, v.format, v.folder, v.custom_name_prefix, v.playlist_strict_mode, v.playlist_item_limit)
|
||||||
|
|
||||||
async def initialize(self):
|
async def initialize(self):
|
||||||
self.event = asyncio.Event()
|
log.info("Initializing DownloadQueue")
|
||||||
asyncio.create_task(self.__download())
|
|
||||||
asyncio.create_task(self.__import_queue())
|
asyncio.create_task(self.__import_queue())
|
||||||
|
|
||||||
|
async def __start_download(self, download):
|
||||||
|
if download.canceled:
|
||||||
|
log.info(f"Download {download.info.title} was canceled, skipping start.")
|
||||||
|
return
|
||||||
|
if self.config.DOWNLOAD_MODE == 'sequential':
|
||||||
|
async with self.seq_lock:
|
||||||
|
log.info("Starting sequential download.")
|
||||||
|
await download.start(self.notifier)
|
||||||
|
self._post_download_cleanup(download)
|
||||||
|
elif self.config.DOWNLOAD_MODE == 'limited' and self.semaphore is not None:
|
||||||
|
await self.__limited_concurrent_download(download)
|
||||||
|
else:
|
||||||
|
await self.__concurrent_download(download)
|
||||||
|
|
||||||
|
async def __concurrent_download(self, download):
|
||||||
|
log.info("Starting concurrent download without limits.")
|
||||||
|
asyncio.create_task(self._run_download(download))
|
||||||
|
|
||||||
|
async def __limited_concurrent_download(self, download):
|
||||||
|
log.info("Starting limited concurrent download.")
|
||||||
|
async with self.semaphore:
|
||||||
|
await self._run_download(download)
|
||||||
|
|
||||||
|
async def _run_download(self, download):
|
||||||
|
if download.canceled:
|
||||||
|
log.info(f"Download {download.info.title} is canceled; skipping start.")
|
||||||
|
return
|
||||||
|
await download.start(self.notifier)
|
||||||
|
self._post_download_cleanup(download)
|
||||||
|
|
||||||
|
def _post_download_cleanup(self, download):
|
||||||
|
if download.info.status != 'finished':
|
||||||
|
if download.tmpfilename and os.path.isfile(download.tmpfilename):
|
||||||
|
try:
|
||||||
|
os.remove(download.tmpfilename)
|
||||||
|
except:
|
||||||
|
pass
|
||||||
|
download.info.status = 'error'
|
||||||
|
download.close()
|
||||||
|
if self.queue.exists(download.info.url):
|
||||||
|
self.queue.delete(download.info.url)
|
||||||
|
if download.canceled:
|
||||||
|
asyncio.create_task(self.notifier.canceled(download.info.url))
|
||||||
|
else:
|
||||||
|
self.done.put(download)
|
||||||
|
asyncio.create_task(self.notifier.completed(download.info))
|
||||||
|
|
||||||
def __extract_info(self, url, playlist_strict_mode):
|
def __extract_info(self, url, playlist_strict_mode):
|
||||||
return yt_dlp.YoutubeDL(params={
|
return yt_dlp.YoutubeDL(params={
|
||||||
'quiet': True,
|
'quiet': True,
|
||||||
|
@ -234,12 +304,6 @@ class DownloadQueue:
|
||||||
}).extract_info(url, download=False)
|
}).extract_info(url, download=False)
|
||||||
|
|
||||||
def __calc_download_path(self, quality, format, folder):
|
def __calc_download_path(self, quality, format, folder):
|
||||||
"""Calculates download path from quality, format and folder attributes.
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
Tuple dldirectory, error_message both of which might be None (but not at the same time)
|
|
||||||
"""
|
|
||||||
# Keep consistent with frontend
|
|
||||||
base_directory = self.config.DOWNLOAD_DIR if (quality != 'audio' and format not in AUDIO_FORMATS) else self.config.AUDIO_DOWNLOAD_DIR
|
base_directory = self.config.DOWNLOAD_DIR if (quality != 'audio' and format not in AUDIO_FORMATS) else self.config.AUDIO_DOWNLOAD_DIR
|
||||||
if folder:
|
if folder:
|
||||||
if not self.config.CUSTOM_DIRS:
|
if not self.config.CUSTOM_DIRS:
|
||||||
|
@ -283,7 +347,7 @@ class DownloadQueue:
|
||||||
log.info(f'Playlist item limit is set. Processing only first {playlist_item_limit} entries')
|
log.info(f'Playlist item limit is set. Processing only first {playlist_item_limit} entries')
|
||||||
entries = entries[:playlist_item_limit]
|
entries = entries[:playlist_item_limit]
|
||||||
for index, etr in enumerate(entries, start=1):
|
for index, etr in enumerate(entries, start=1):
|
||||||
etr["_type"] = "video" # Prevents video to be treated as url and lose below properties during processing
|
etr["_type"] = "video"
|
||||||
etr["playlist"] = entry["id"]
|
etr["playlist"] = entry["id"]
|
||||||
etr["playlist_index"] = '{{0:0{0:d}d}}'.format(playlist_index_digits).format(index)
|
etr["playlist_index"] = '{{0:0{0:d}d}}'.format(playlist_index_digits).format(index)
|
||||||
for property in ("id", "title", "uploader", "uploader_id"):
|
for property in ("id", "title", "uploader", "uploader_id"):
|
||||||
|
@ -293,10 +357,11 @@ class DownloadQueue:
|
||||||
if any(res['status'] == 'error' for res in results):
|
if any(res['status'] == 'error' for res in results):
|
||||||
return {'status': 'error', 'msg': ', '.join(res['msg'] for res in results if res['status'] == 'error' and 'msg' in res)}
|
return {'status': 'error', 'msg': ', '.join(res['msg'] for res in results if res['status'] == 'error' and 'msg' in res)}
|
||||||
return {'status': 'ok'}
|
return {'status': 'ok'}
|
||||||
elif etype == 'video' or etype.startswith('url') and 'id' in entry and 'title' in entry:
|
elif etype == 'video' or (etype.startswith('url') and 'id' in entry and 'title' in entry):
|
||||||
log.debug('Processing as a video')
|
log.debug('Processing as a video')
|
||||||
if not self.queue.exists(entry['id']):
|
key = entry.get('webpage_url') or entry['url']
|
||||||
dl = DownloadInfo(entry['id'], entry.get('title') or entry['id'], entry.get('webpage_url') or entry['url'], quality, format, folder, custom_name_prefix, error)
|
if not self.queue.exists(key):
|
||||||
|
dl = DownloadInfo(entry['id'], entry.get('title') or entry['id'], key, quality, format, folder, custom_name_prefix, error)
|
||||||
dldirectory, error_message = self.__calc_download_path(quality, format, folder)
|
dldirectory, error_message = self.__calc_download_path(quality, format, folder)
|
||||||
if error_message is not None:
|
if error_message is not None:
|
||||||
return error_message
|
return error_message
|
||||||
|
@ -305,20 +370,17 @@ class DownloadQueue:
|
||||||
if 'playlist' in entry and entry['playlist'] is not None:
|
if 'playlist' in entry and entry['playlist'] is not None:
|
||||||
if len(self.config.OUTPUT_TEMPLATE_PLAYLIST):
|
if len(self.config.OUTPUT_TEMPLATE_PLAYLIST):
|
||||||
output = self.config.OUTPUT_TEMPLATE_PLAYLIST
|
output = self.config.OUTPUT_TEMPLATE_PLAYLIST
|
||||||
|
|
||||||
for property, value in entry.items():
|
for property, value in entry.items():
|
||||||
if property.startswith("playlist"):
|
if property.startswith("playlist"):
|
||||||
output = output.replace(f"%({property})s", str(value))
|
output = output.replace(f"%({property})s", str(value))
|
||||||
|
|
||||||
ytdl_options = dict(self.config.YTDL_OPTIONS)
|
ytdl_options = dict(self.config.YTDL_OPTIONS)
|
||||||
|
|
||||||
if playlist_item_limit > 0:
|
if playlist_item_limit > 0:
|
||||||
log.info(f'playlist limit is set. Processing only first {playlist_item_limit} entries')
|
log.info(f'playlist limit is set. Processing only first {playlist_item_limit} entries')
|
||||||
ytdl_options['playlistend'] = playlist_item_limit
|
ytdl_options['playlistend'] = playlist_item_limit
|
||||||
|
|
||||||
if auto_start is True:
|
if auto_start is True:
|
||||||
self.queue.put(Download(dldirectory, self.config.TEMP_DIR, output, output_chapter, quality, format, ytdl_options, dl))
|
download = Download(dldirectory, self.config.TEMP_DIR, output, output_chapter, quality, format, ytdl_options, dl)
|
||||||
self.event.set()
|
self.queue.put(download)
|
||||||
|
asyncio.create_task(self.__start_download(download))
|
||||||
else:
|
else:
|
||||||
self.pending.put(Download(dldirectory, self.config.TEMP_DIR, output, output_chapter, quality, format, ytdl_options, dl))
|
self.pending.put(Download(dldirectory, self.config.TEMP_DIR, output, output_chapter, quality, format, ytdl_options, dl))
|
||||||
await self.notifier.added(dl)
|
await self.notifier.added(dl)
|
||||||
|
@ -347,7 +409,7 @@ class DownloadQueue:
|
||||||
dl = self.pending.get(id)
|
dl = self.pending.get(id)
|
||||||
self.queue.put(dl)
|
self.queue.put(dl)
|
||||||
self.pending.delete(id)
|
self.pending.delete(id)
|
||||||
self.event.set()
|
asyncio.create_task(self.__start_download(dl))
|
||||||
return {'status': 'ok'}
|
return {'status': 'ok'}
|
||||||
|
|
||||||
async def cancel(self, ids):
|
async def cancel(self, ids):
|
||||||
|
@ -383,30 +445,6 @@ class DownloadQueue:
|
||||||
return {'status': 'ok'}
|
return {'status': 'ok'}
|
||||||
|
|
||||||
def get(self):
|
def get(self):
|
||||||
return(list((k, v.info) for k, v in self.queue.items()) + list((k, v.info) for k, v in self.pending.items()),
|
return (list((k, v.info) for k, v in self.queue.items()) +
|
||||||
list((k, v.info) for k, v in self.done.items()))
|
list((k, v.info) for k, v in self.pending.items()),
|
||||||
|
list((k, v.info) for k, v in self.done.items()))
|
||||||
async def __download(self):
|
|
||||||
while True:
|
|
||||||
while self.queue.empty():
|
|
||||||
log.info('waiting for item to download')
|
|
||||||
await self.event.wait()
|
|
||||||
self.event.clear()
|
|
||||||
id, entry = self.queue.next()
|
|
||||||
log.info(f'downloading {entry.info.title}')
|
|
||||||
await entry.start(self.notifier)
|
|
||||||
if entry.info.status != 'finished':
|
|
||||||
if entry.tmpfilename and os.path.isfile(entry.tmpfilename):
|
|
||||||
try:
|
|
||||||
os.remove(entry.tmpfilename)
|
|
||||||
except:
|
|
||||||
pass
|
|
||||||
entry.info.status = 'error'
|
|
||||||
entry.close()
|
|
||||||
if self.queue.exists(id):
|
|
||||||
self.queue.delete(id)
|
|
||||||
if entry.canceled:
|
|
||||||
await self.notifier.canceled(id)
|
|
||||||
else:
|
|
||||||
self.done.put(entry)
|
|
||||||
await self.notifier.completed(entry.info)
|
|
||||||
|
|
|
@ -75,7 +75,7 @@
|
||||||
</select>
|
</select>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="col-md-6 col-lg-3 add-url-component">
|
<div class="col-md-6 col-lg-3 add-url-component d-flex align-items-center">
|
||||||
<div [attr.class]="showAdvanced() ? 'btn-group add-url-group' : 'add-url-group'" ngbDropdown #advancedDropdown="ngbDropdown" display="dynamic" placement="bottom-end">
|
<div [attr.class]="showAdvanced() ? 'btn-group add-url-group' : 'add-url-group'" ngbDropdown #advancedDropdown="ngbDropdown" display="dynamic" placement="bottom-end">
|
||||||
<button class="btn btn-primary add-url" type="submit" (click)="addDownload()" [disabled]="addInProgress || downloads.loading">
|
<button class="btn btn-primary add-url" type="submit" (click)="addDownload()" [disabled]="addInProgress || downloads.loading">
|
||||||
<span class="spinner-border spinner-border-sm" role="status" id="add-spinner" *ngIf="addInProgress"></span>
|
<span class="spinner-border spinner-border-sm" role="status" id="add-spinner" *ngIf="addInProgress"></span>
|
||||||
|
@ -115,11 +115,60 @@
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
<div class="btn-group ms-2" ngbDropdown>
|
||||||
|
<button type="button" class="btn btn-secondary" (click)="openBatchImportModal()">
|
||||||
|
Import
|
||||||
|
</button>
|
||||||
|
<button type="button" class="btn btn-secondary dropdown-toggle dropdown-toggle-split" ngbDropdownToggle>
|
||||||
|
<span class="visually-hidden">Toggle Dropdown</span>
|
||||||
|
</button>
|
||||||
|
<ul ngbDropdownMenu class="dropdown-menu dropdown-menu-end">
|
||||||
|
<li>
|
||||||
|
<button class="dropdown-item" (click)="exportBatchUrls('all')">
|
||||||
|
Export All URLs
|
||||||
|
</button>
|
||||||
|
</li>
|
||||||
|
<li>
|
||||||
|
<button class="dropdown-item" (click)="copyBatchUrls('all')">
|
||||||
|
Copy All URLs
|
||||||
|
</button>
|
||||||
|
</li>
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</form>
|
</form>
|
||||||
|
|
||||||
|
<!-- Batch Import Modal -->
|
||||||
|
<div class="modal fade" tabindex="-1" role="dialog" [ngClass]="{'show': batchImportModalOpen}" [ngStyle]="{'display': batchImportModalOpen ? 'block' : 'none'}">
|
||||||
|
<div class="modal-dialog" role="document">
|
||||||
|
<div class="modal-content">
|
||||||
|
<div class="modal-header">
|
||||||
|
<h5 class="modal-title">Batch Import URLs</h5>
|
||||||
|
<button type="button" class="btn-close" aria-label="Close" (click)="closeBatchImportModal()"></button>
|
||||||
|
</div>
|
||||||
|
<div class="modal-body">
|
||||||
|
<textarea [(ngModel)]="batchImportText" class="form-control" rows="6"
|
||||||
|
placeholder="Paste one video URL per line"></textarea>
|
||||||
|
<div class="mt-2">
|
||||||
|
<small *ngIf="batchImportStatus">{{ batchImportStatus }}</small>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="modal-footer">
|
||||||
|
<button type="button" class="btn btn-danger me-auto" *ngIf="importInProgress" (click)="cancelBatchImport()">
|
||||||
|
Cancel Import
|
||||||
|
</button>
|
||||||
|
<button type="button" class="btn btn-secondary" (click)="closeBatchImportModal()">Close</button>
|
||||||
|
<button type="button" class="btn btn-primary" (click)="startBatchImport()" [disabled]="importInProgress">
|
||||||
|
Import URLs
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
|
||||||
<div *ngIf="downloads.loading" class="alert alert-info" role="alert">
|
<div *ngIf="downloads.loading" class="alert alert-info" role="alert">
|
||||||
Connecting to server...
|
Connecting to server...
|
||||||
</div>
|
</div>
|
||||||
|
|
|
@ -65,3 +65,41 @@ td
|
||||||
.download-progressbar
|
.download-progressbar
|
||||||
width: 12rem
|
width: 12rem
|
||||||
margin-left: auto
|
margin-left: auto
|
||||||
|
|
||||||
|
.batch-panel
|
||||||
|
margin-top: 15px
|
||||||
|
border: 1px solid #ccc
|
||||||
|
border-radius: 8px
|
||||||
|
padding: 15px
|
||||||
|
background-color: #fff
|
||||||
|
box-shadow: 0 2px 8px rgba(0, 0, 0, 0.1)
|
||||||
|
|
||||||
|
.batch-panel-header
|
||||||
|
border-bottom: 1px solid #eee
|
||||||
|
padding-bottom: 8px
|
||||||
|
margin-bottom: 15px
|
||||||
|
h4
|
||||||
|
font-size: 1.5rem
|
||||||
|
margin: 0
|
||||||
|
|
||||||
|
.batch-panel-body
|
||||||
|
textarea.form-control
|
||||||
|
resize: vertical
|
||||||
|
|
||||||
|
.batch-status
|
||||||
|
font-size: 0.9rem
|
||||||
|
color: #555
|
||||||
|
|
||||||
|
.d-flex.my-3
|
||||||
|
margin-top: 1rem
|
||||||
|
margin-bottom: 1rem
|
||||||
|
|
||||||
|
.modal.fade.show
|
||||||
|
background-color: rgba(0, 0, 0, 0.5)
|
||||||
|
|
||||||
|
.modal-header
|
||||||
|
border-bottom: 1px solid #eee
|
||||||
|
.modal-body
|
||||||
|
textarea.form-control
|
||||||
|
resize: vertical
|
||||||
|
|
||||||
|
|
|
@ -31,6 +31,12 @@ export class AppComponent implements AfterViewInit {
|
||||||
themes: Theme[] = Themes;
|
themes: Theme[] = Themes;
|
||||||
activeTheme: Theme;
|
activeTheme: Theme;
|
||||||
customDirs$: Observable<string[]>;
|
customDirs$: Observable<string[]>;
|
||||||
|
showBatchPanel: boolean = false;
|
||||||
|
batchImportModalOpen = false;
|
||||||
|
batchImportText = '';
|
||||||
|
batchImportStatus = '';
|
||||||
|
importInProgress = false;
|
||||||
|
cancelImportFlag = false;
|
||||||
versionInfo: string | null = null;
|
versionInfo: string | null = null;
|
||||||
|
|
||||||
@ViewChild('queueMasterCheckbox') queueMasterCheckbox: MasterCheckboxComponent;
|
@ViewChild('queueMasterCheckbox') queueMasterCheckbox: MasterCheckboxComponent;
|
||||||
|
@ -297,6 +303,134 @@ export class AppComponent implements AfterViewInit {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Toggle inline batch panel (if you want to use an inline panel for export; not used for import modal)
|
||||||
|
toggleBatchPanel(): void {
|
||||||
|
this.showBatchPanel = !this.showBatchPanel;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Open the Batch Import modal
|
||||||
|
openBatchImportModal(): void {
|
||||||
|
this.batchImportModalOpen = true;
|
||||||
|
this.batchImportText = '';
|
||||||
|
this.batchImportStatus = '';
|
||||||
|
this.importInProgress = false;
|
||||||
|
this.cancelImportFlag = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Close the Batch Import modal
|
||||||
|
closeBatchImportModal(): void {
|
||||||
|
this.batchImportModalOpen = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Start importing URLs from the batch modal textarea
|
||||||
|
startBatchImport(): void {
|
||||||
|
const urls = this.batchImportText
|
||||||
|
.split(/\r?\n/)
|
||||||
|
.map(url => url.trim())
|
||||||
|
.filter(url => url.length > 0);
|
||||||
|
if (urls.length === 0) {
|
||||||
|
alert('No valid URLs found.');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
this.importInProgress = true;
|
||||||
|
this.cancelImportFlag = false;
|
||||||
|
this.batchImportStatus = `Starting to import ${urls.length} URLs...`;
|
||||||
|
let index = 0;
|
||||||
|
const delayBetween = 1000;
|
||||||
|
const processNext = () => {
|
||||||
|
if (this.cancelImportFlag) {
|
||||||
|
this.batchImportStatus = `Import cancelled after ${index} of ${urls.length} URLs.`;
|
||||||
|
this.importInProgress = false;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (index >= urls.length) {
|
||||||
|
this.batchImportStatus = `Finished importing ${urls.length} URLs.`;
|
||||||
|
this.importInProgress = false;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const url = urls[index];
|
||||||
|
this.batchImportStatus = `Importing URL ${index + 1} of ${urls.length}: ${url}`;
|
||||||
|
this.downloads.addDownloadByUrl(url)
|
||||||
|
.then(() => {
|
||||||
|
index++;
|
||||||
|
setTimeout(processNext, delayBetween);
|
||||||
|
})
|
||||||
|
.catch(err => {
|
||||||
|
console.error(`Error importing URL ${url}:`, err);
|
||||||
|
index++;
|
||||||
|
setTimeout(processNext, delayBetween);
|
||||||
|
});
|
||||||
|
};
|
||||||
|
processNext();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Cancel the batch import process
|
||||||
|
cancelBatchImport(): void {
|
||||||
|
if (this.importInProgress) {
|
||||||
|
this.cancelImportFlag = true;
|
||||||
|
this.batchImportStatus += ' Cancelling...';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Export URLs based on filter: 'pending', 'completed', 'failed', or 'all'
|
||||||
|
exportBatchUrls(filter: 'pending' | 'completed' | 'failed' | 'all'): void {
|
||||||
|
let urls: string[];
|
||||||
|
if (filter === 'pending') {
|
||||||
|
urls = Array.from(this.downloads.queue.values()).map(dl => dl.url);
|
||||||
|
} else if (filter === 'completed') {
|
||||||
|
// Only finished downloads in the "done" Map
|
||||||
|
urls = Array.from(this.downloads.done.values()).filter(dl => dl.status === 'finished').map(dl => dl.url);
|
||||||
|
} else if (filter === 'failed') {
|
||||||
|
// Only error downloads from the "done" Map
|
||||||
|
urls = Array.from(this.downloads.done.values()).filter(dl => dl.status === 'error').map(dl => dl.url);
|
||||||
|
} else {
|
||||||
|
// All: pending + both finished and error in done
|
||||||
|
urls = [
|
||||||
|
...Array.from(this.downloads.queue.values()).map(dl => dl.url),
|
||||||
|
...Array.from(this.downloads.done.values()).map(dl => dl.url)
|
||||||
|
];
|
||||||
|
}
|
||||||
|
if (!urls.length) {
|
||||||
|
alert('No URLs found for the selected filter.');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const content = urls.join('\n');
|
||||||
|
const blob = new Blob([content], { type: 'text/plain' });
|
||||||
|
const downloadUrl = window.URL.createObjectURL(blob);
|
||||||
|
const a = document.createElement('a');
|
||||||
|
a.href = downloadUrl;
|
||||||
|
a.download = 'metube_urls.txt';
|
||||||
|
document.body.appendChild(a);
|
||||||
|
a.click();
|
||||||
|
document.body.removeChild(a);
|
||||||
|
window.URL.revokeObjectURL(downloadUrl);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Copy URLs to clipboard based on filter: 'pending', 'completed', 'failed', or 'all'
|
||||||
|
copyBatchUrls(filter: 'pending' | 'completed' | 'failed' | 'all'): void {
|
||||||
|
let urls: string[];
|
||||||
|
if (filter === 'pending') {
|
||||||
|
urls = Array.from(this.downloads.queue.values()).map(dl => dl.url);
|
||||||
|
} else if (filter === 'completed') {
|
||||||
|
urls = Array.from(this.downloads.done.values()).filter(dl => dl.status === 'finished').map(dl => dl.url);
|
||||||
|
} else if (filter === 'failed') {
|
||||||
|
urls = Array.from(this.downloads.done.values()).filter(dl => dl.status === 'error').map(dl => dl.url);
|
||||||
|
} else {
|
||||||
|
urls = [
|
||||||
|
...Array.from(this.downloads.queue.values()).map(dl => dl.url),
|
||||||
|
...Array.from(this.downloads.done.values()).map(dl => dl.url)
|
||||||
|
];
|
||||||
|
}
|
||||||
|
if (!urls.length) {
|
||||||
|
alert('No URLs found for the selected filter.');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const content = urls.join('\n');
|
||||||
|
navigator.clipboard.writeText(content)
|
||||||
|
.then(() => alert('URLs copied to clipboard.'))
|
||||||
|
.catch(() => alert('Failed to copy URLs.'));
|
||||||
|
}
|
||||||
|
|
||||||
fetchVersionInfo(): void {
|
fetchVersionInfo(): void {
|
||||||
const baseUrl = `${window.location.origin}${window.location.pathname.replace(/\/[^\/]*$/, '/')}`;
|
const baseUrl = `${window.location.origin}${window.location.pathname.replace(/\/[^\/]*$/, '/')}`;
|
||||||
const versionUrl = `${baseUrl}version`;
|
const versionUrl = `${baseUrl}version`;
|
||||||
|
|
|
@ -129,4 +129,26 @@ export class DownloadsService {
|
||||||
this[where].forEach((dl: Download) => { if (filter(dl)) ids.push(dl.url) });
|
this[where].forEach((dl: Download) => { if (filter(dl)) ids.push(dl.url) });
|
||||||
return this.delById(where, ids);
|
return this.delById(where, ids);
|
||||||
}
|
}
|
||||||
|
public addDownloadByUrl(url: string): Promise<any> {
|
||||||
|
const defaultQuality = 'best';
|
||||||
|
const defaultFormat = 'mp4';
|
||||||
|
const defaultFolder = '';
|
||||||
|
const defaultCustomNamePrefix = '';
|
||||||
|
const defaultPlaylistStrictMode = false;
|
||||||
|
const defaultPlaylistItemLimit = 0;
|
||||||
|
const defaultAutoStart = true;
|
||||||
|
|
||||||
|
return new Promise((resolve, reject) => {
|
||||||
|
this.add(url, defaultQuality, defaultFormat, defaultFolder, defaultCustomNamePrefix, defaultPlaylistStrictMode, defaultPlaylistItemLimit, defaultAutoStart)
|
||||||
|
.subscribe(
|
||||||
|
response => resolve(response),
|
||||||
|
error => reject(error)
|
||||||
|
);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
public exportQueueUrls(): string[] {
|
||||||
|
return Array.from(this.queue.values()).map(download => download.url);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue