diff --git a/app/main.py b/app/main.py index c7bbfed..ae17291 100644 --- a/app/main.py +++ b/app/main.py @@ -12,6 +12,7 @@ import json import pathlib from ytdl import DownloadQueueNotifier, DownloadQueue +from yt_dlp.version import __version__ as yt_dlp_version log = logging.getLogger('main') @@ -231,6 +232,10 @@ def robots(request): ) return response +@routes.get(config.URL_PREFIX + 'version') +def version(request): + return web.json_response({"version": yt_dlp_version}) + if config.URL_PREFIX != '/': @routes.get('/') def index_redirect_root(request): diff --git a/ui/src/app/app.component.html b/ui/src/app/app.component.html index dc5e3bb..781f60b 100644 --- a/ui/src/app/app.component.html +++ b/ui/src/app/app.component.html @@ -218,5 +218,7 @@ - + diff --git a/ui/src/app/app.component.ts b/ui/src/app/app.component.ts index dd7319b..c8f9e6c 100644 --- a/ui/src/app/app.component.ts +++ b/ui/src/app/app.component.ts @@ -1,4 +1,5 @@ import { Component, ViewChild, ElementRef, AfterViewInit } from '@angular/core'; +import { HttpClient } from '@angular/common/http'; import { faTrashAlt, faCheckCircle, faTimesCircle, IconDefinition } from '@fortawesome/free-regular-svg-icons'; import { faRedoAlt, faSun, faMoon, faCircleHalfStroke, faCheck, faExternalLinkAlt, faDownload } from '@fortawesome/free-solid-svg-icons'; import { CookieService } from 'ngx-cookie-service'; @@ -30,6 +31,7 @@ export class AppComponent implements AfterViewInit { themes: Theme[] = Themes; activeTheme: Theme; customDirs$: Observable; + versionInfo: string | null = null; @ViewChild('queueMasterCheckbox') queueMasterCheckbox: MasterCheckboxComponent; @ViewChild('queueDelSelected') queueDelSelected: ElementRef; @@ -52,7 +54,7 @@ export class AppComponent implements AfterViewInit { faDownload = faDownload; faExternalLinkAlt = faExternalLinkAlt; - constructor(public downloads: DownloadsService, private cookieService: CookieService) { + constructor(public downloads: DownloadsService, private cookieService: CookieService, private http: HttpClient) { this.format = cookieService.get('metube_format') || 'any'; // Needs to be set or qualities won't automatically be set this.setQualities() @@ -91,6 +93,7 @@ export class AppComponent implements AfterViewInit { this.doneClearFailed.nativeElement.disabled = failed === 0; this.doneRetryFailed.nativeElement.disabled = failed === 0; }); + this.fetchVersionInfo(); } // workaround to allow fetching of Map values in the order they were inserted @@ -293,4 +296,18 @@ export class AppComponent implements AfterViewInit { event.preventDefault(); } } + + fetchVersionInfo(): void { + const baseUrl = `${window.location.origin}${window.location.pathname.replace(/\/[^\/]*$/, '/')}`; + const versionUrl = `${baseUrl}version`; + this.http.get<{ version: string}>(versionUrl) + .subscribe({ + next: (data) => { + this.versionInfo = `yt-dlp version: ${data.version}`; + }, + error: () => { + this.versionInfo = ''; + } + }); + } }