container->get('view'); assert($view instanceof Smarty); $this->view = $view; // Plugins $this->view->registerPlugin('modifier', 'filter_var', 'filter_var'); } /** * Display index page. * * @param Request $request PSR-7 request * @param Response $response PSR-7 response * * @return Response HTTP response */ public function index(Request $request, Response $response): Response { $this->view->render( $response, 'index.tpl', [ 'class' => 'index', 'description' => $this->localeManager->t( 'Easily download videos from YouTube, Dailymotion, Vimeo and other websites.' ), 'supportedLocales' => $this->localeManager->getSupportedLocales(), ] ); return $response; } /** * Switch locale. * * @param Request $request PSR-7 request * @param Response $response PSR-7 response * @param string[] $data Query parameters * * @return Response */ public function locale(Request $request, Response $response, array $data): Response { $this->localeManager->setLocale(new Locale($data['locale'])); return $response->withRedirect($this->router->pathFor('index')); } /** * Display a list of extractors. * * @param Request $request PSR-7 request * @param Response $response PSR-7 response * * @return Response HTTP response * @throws AlltubeLibraryException */ public function extractors(Request $request, Response $response): Response { $this->view->render( $response, 'extractors.tpl', [ 'extractors' => $this->downloader->getExtractors(), 'class' => 'extractors', 'title' => $this->localeManager->t('Supported websites'), 'description' => $this->localeManager->t('List of all supported websites from which AllTube Download ' . 'can extract video or audio files'), ] ); return $response; } /** * Display a password prompt. * * @param Request $request PSR-7 request * @param Response $response PSR-7 response * * @return Response HTTP response */ public function password(Request $request, Response $response): Response { $this->view->render( $response, 'password.tpl', [ 'class' => 'password', 'title' => $this->localeManager->t('Password prompt'), 'description' => $this->localeManager->t( 'You need a password in order to download this video with AllTube Download' ), ] ); return $response->withStatus(StatusCode::HTTP_FORBIDDEN); } /** * Return the video description page. * * @param Request $request PSR-7 request * @param Response $response PSR-7 response * * @return Response HTTP response * @throws AlltubeLibraryException */ private function getInfoResponse(Request $request, Response $response): Response { try { $this->video->getJson(); } catch (PasswordException $e) { return $this->password($request, $response); } catch (WrongPasswordException $e) { return $this->displayError($request, $response, $this->localeManager->t('Wrong password')); } if (isset($this->video->entries)) { $template = 'playlist.tpl'; } else { $template = 'info.tpl'; } $title = $this->localeManager->t('Video download'); $description = $this->localeManager->t( 'Download video from @extractor', ['@extractor' => $this->video->extractor_key] ); if (isset($this->video->title)) { $title = $this->video->title; $description = $this->localeManager->t( 'Download @title from @extractor', [ '@title' => $this->video->title, '@extractor' => $this->video->extractor_key ] ); } $formats = []; $genericFormatsLabel = $this->localeManager->t('Generic formats'); $detailedFormatsLabel = $this->localeManager->t('Detailed formats'); foreach ($this->config->genericFormats as $id => $genericFormat) { $formats[$genericFormatsLabel][$id] = $this->localeManager->t($genericFormat); } $json = $this->video->getJson(); if (isset($json->formats)) { /** @var stdClass $format */ foreach ($json->formats as $format) { if ($this->config->stream || in_array($format->protocol, ['http', 'https'])) { $formatParts = [ // File extension $format->ext, ]; if (isset($format->width) || isset($format->height)) { // Video dimensions $formatParts[] = implode('x', array_filter([$format->width, $format->height])); } if (isset($format->filesize)) { // File size $formatParts[] = round($format->filesize / 1000000, 2) . ' MB'; } if (isset($format->format_note)) { // Format name $formatParts[] = $format->format_note; } if (isset($format->format_id)) { // Format ID $formatParts[] = '(' . $format->format_id . ')'; } $formats[$detailedFormatsLabel][$format->format_id] = implode(' ', $formatParts); } } } /* Fetch the thumbnail, if it exists, and add a data URI to the video object */ if (isset($this->video->thumbnail) && $this->video->thumbnail !== '') { /* Fetch the thumbnail */ $thumbnailData = file_get_contents($this->video->thumbnail); if ($thumbnailData !== false) { $thumbnailData = base64_encode($thumbnailData); /* Guess the mime type */ $thumbnailMime = 'image/jpeg'; if (strpos($this->video->thumbnail, '.png') !== false) { $thumbnailMime = 'image/png'; } elseif (strpos($this->video->thumbnail, '.gif') !== false) { $thumbnailMime = 'image/gif'; } elseif (strpos($this->video->thumbnail, '.webp') !== false) { $thumbnailMime = 'image/webp'; } $this->video->thumbnail = 'data:' . $thumbnailMime . ';base64,' . $thumbnailData; } } $this->view->render( $response, $template, [ 'video' => $this->video, 'class' => 'info', 'title' => $title, 'description' => $description, 'defaultFormat' => $this->defaultFormat, 'formats' => $formats ] ); return $response; } /** * Dislay information about the video. * * @param Request $request PSR-7 request * @param Response $response PSR-7 response * * @return Response HTTP response * @throws AlltubeLibraryException * @throws InvalidURLException */ public function info(Request $request, Response $response): Response { $url = $this->getVideoPageUrl($request); $this->video = $this->downloader->getVideo($url, $this->getFormat($request), $this->getPassword($request)); if ($this->config->convert && $request->getQueryParam('audio')) { // We skip the info page and get directly to the download. return $response->withRedirect( $this->router->pathFor('download', [], $request->getQueryParams()) ); } else { return $this->getInfoResponse($request, $response); } } /** * Display an user-friendly error. * * @param Request $request PSR-7 request * @param Response $response PSR-7 response * @param string $message Error message * * @return Response HTTP response */ protected function displayError(Request $request, Response $response, string $message): Response { $this->view->render( $response, 'error.tpl', [ 'error' => $message, 'class' => 'video', 'title' => $this->localeManager->t('Error'), ] ); return $response->withStatus(StatusCode::HTTP_INTERNAL_SERVER_ERROR); } /** * @param Request $request * @param Response $response * @return Response */ public function notFound(Request $request, Response $response): Response { return $this->displayError($request, $response, $this->localeManager->t('Page not found')) ->withStatus(StatusCode::HTTP_NOT_FOUND); } /** * @param Request $request * @param Response $response * @return Response */ public function notAllowed(Request $request, Response $response): Response { return $this->displayError($request, $response, $this->localeManager->t('Method not allowed')) ->withStatus(StatusCode::HTTP_METHOD_NOT_ALLOWED); } /** * Display an error page. * * @param Request $request PSR-7 request * @param Response $response PSR-7 response * @param Throwable $error Error to display * * @return Response HTTP response */ public function error(Request $request, Response $response, Throwable $error): Response { $this->logger->error($error); // We apply the CSP manually because middlewares are not called on error pages. $cspMiddleware = new CspMiddleware($this->container); /** @var Response $response */ $response = $cspMiddleware->applyHeader($response); if ($this->config->debug) { $projectDir = $this->container->get('root_path'); assert(is_string($projectDir) || is_null($projectDir)); $renderer = new HtmlErrorRenderer(true, null, null, $projectDir); $exception = $renderer->render($error); $response->getBody()->write($exception->getAsString()); foreach ($exception->getHeaders() as $header => $value) { $response = $response->withHeader($header, $value); } return $response->withStatus($exception->getStatusCode()); } else { if ($error instanceof Exception) { $message = $error->getMessage(); } else { $message = ''; } return $this->displayError($request, $response, $message); } } /** * Route that mimics YouTube video URLs ("/watch?v=foo") * * @param Request $request * @param Response $response * @return Response */ public function watch(Request $request, Response $response): Response { // We build a full YouTube URL from the video ID. $youtubeUri = Uri::createFromString('https://www.youtube.com/watch') ->withQuery(http_build_query(['v' => $request->getQueryParam('v')])); // Then pass it to the info route. return $response->withRedirect( Uri::createFromString($this->router->pathFor('info')) ->withQuery(http_build_query(['url' => strval($youtubeUri)])) ); } }