From 5c9becfcffc3a0ec37c002c060edf597859cb9a2 Mon Sep 17 00:00:00 2001 From: boyska Date: Tue, 17 Sep 2019 12:31:06 +0200 Subject: [PATCH 1/9] new bridge: AutoPodcasterBridge if you have a feed of a radio show which does not seem to be a valid podcast, this bridge will transformt it into a valid podcast --- bridges/AutoPodcasterBridge.php | 93 +++++++++++++++++++++++++++++++++ 1 file changed, 93 insertions(+) create mode 100644 bridges/AutoPodcasterBridge.php diff --git a/bridges/AutoPodcasterBridge.php b/bridges/AutoPodcasterBridge.php new file mode 100644 index 00000000..870cadae --- /dev/null +++ b/bridges/AutoPodcasterBridge.php @@ -0,0 +1,93 @@ + array( + 'url' => array( + 'name' => 'URL', + 'required' => true + ))); + + private function archiveIsAudioFormat($formatString) { + return strpos($formatString, 'MP3') !== false || + strpos($formatString, 'Ogg') === 0; + } + + protected function parseItem($newItem){ + $item = parent::parseItem($newItem); + + $dom = getSimpleHTMLDOMCached($item['uri']); + $audios = []; + + /* 1st extraction method: by "audio" tag */ + foreach($dom->find('audio') as $audioEl) { + $sources = []; + if($audioEl->src !== false) { + $sources[] = $audioEl->src; + } + foreach($audioEl->find('source') as $sourceEl) { + $sources[] = $sourceEl->src; + } + if($sources) { + $audios[$sources[0]] = ['sources' => [$sources]]; + } + } + + /* 2nd extraction method: by "iframe" tag */ + foreach($dom->find('iframe') as $iframeEl) { + if(strpos($iframeEl->src, "https://archive.org/embed/") === 0) { + $listURL = preg_replace("/\/embed\//", "/details/", $iframeEl->src, 1) . "?output=json"; + $baseURL = preg_replace("/\/embed\//", "/download/", $iframeEl->src, 1); + $list = json_decode(file_get_contents($listURL)); + $audios = []; + foreach($list->files as $name =>$data) { + if($data->source === 'original' && + $this->archiveIsAudioFormat($data->format)) { + $audios[$baseURL . $name] = ['sources' => [$baseURL . $name]]; + } + } + foreach($list->files as $name =>$data) { + if($data->source === 'derivative' && + $this->archiveIsAudioFormat($data->format) && + isset($audios[$baseURL . "/" . $data->original])) { + $audios[$baseURL . "/" . $data->original]['sources'][] = $baseURL . $name; + } + } + } + } + + + + if(count($audios) === 0) { + return null; + } + $item['enclosures'] = array_values($audios); + $item['enclosures'] = []; + foreach(array_values($audios) as $audio) { + $item['enclosures'][] = $audio['sources'][0]; + } + return $item; + } + public function collectData(){ + if($this->getInput('url') && substr($this->getInput('url'), 0, strlen('http')) !== 'http') { + // just in case someone find a way to access local files by playing with the url + returnClientError('The url parameter must either refer to http or https protocol.'); + } + $this->collectExpandableDatas($this->getURI()); + } + public function getName(){ + if(!is_null($this->getInput('url'))) { + return self::NAME . ' : ' . $this->getInput('url'); + } + + return parent::getName(); + } + public function getURI(){ + return $this->getInput('url'); + } + +} + From fb5424ac9ddac992e6802e4a637e866afe732cdf Mon Sep 17 00:00:00 2001 From: boyska Date: Tue, 17 Sep 2019 12:36:59 +0200 Subject: [PATCH 2/9] [AutoPodcasterBridge] FIX nested lists --- bridges/AutoPodcasterBridge.php | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/bridges/AutoPodcasterBridge.php b/bridges/AutoPodcasterBridge.php index 870cadae..c0fabbf6 100644 --- a/bridges/AutoPodcasterBridge.php +++ b/bridges/AutoPodcasterBridge.php @@ -32,7 +32,7 @@ class AutoPodcasterBridge extends FeedExpander { $sources[] = $sourceEl->src; } if($sources) { - $audios[$sources[0]] = ['sources' => [$sources]]; + $audios[$sources[0]] = ['sources' => $sources]; } } From 69b79c7e1bbd01af4f6935c852805595e78bdd39 Mon Sep 17 00:00:00 2001 From: boyska Date: Tue, 17 Sep 2019 12:40:06 +0200 Subject: [PATCH 3/9] [AutoPodcasterBridge] fix metadata --- bridges/AutoPodcasterBridge.php | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/bridges/AutoPodcasterBridge.php b/bridges/AutoPodcasterBridge.php index c0fabbf6..ac6e357d 100644 --- a/bridges/AutoPodcasterBridge.php +++ b/bridges/AutoPodcasterBridge.php @@ -3,8 +3,8 @@ class AutoPodcasterBridge extends FeedExpander { const MAINTAINER='boyska'; const NAME='Auto Podcaster'; const URI = ''; - const CACHE_TIMEOUT = 0; - const DESCRIPTION='Crea un podcast multimediale a partire da un feed normale'; + const CACHE_TIMEOUT = 300; // 5 minuti + const DESCRIPTION='Make a "multimedia" podcast out of a normal feed'; const PARAMETERS = array('url' => array( 'url' => array( 'name' => 'URL', From 5d5f1fe965b43e92e7ffab25f7657f0aba7a9e88 Mon Sep 17 00:00:00 2001 From: boyska Date: Thu, 26 Sep 2019 11:00:46 +0200 Subject: [PATCH 4/9] [AutoPodcaster] FIX feeds w/o url, but w/ content --- bridges/AutoPodcasterBridge.php | 34 ++++++++++++++++++++++++++------- 1 file changed, 27 insertions(+), 7 deletions(-) diff --git a/bridges/AutoPodcasterBridge.php b/bridges/AutoPodcasterBridge.php index ac6e357d..bdd0729f 100644 --- a/bridges/AutoPodcasterBridge.php +++ b/bridges/AutoPodcasterBridge.php @@ -16,13 +16,8 @@ class AutoPodcasterBridge extends FeedExpander { strpos($formatString, 'Ogg') === 0; } - protected function parseItem($newItem){ - $item = parent::parseItem($newItem); - - $dom = getSimpleHTMLDOMCached($item['uri']); + private function extractAudio($dom) { $audios = []; - - /* 1st extraction method: by "audio" tag */ foreach($dom->find('audio') as $audioEl) { $sources = []; if($audioEl->src !== false) { @@ -35,8 +30,11 @@ class AutoPodcasterBridge extends FeedExpander { $audios[$sources[0]] = ['sources' => $sources]; } } + return $audios; + } + private function extractIframeArchive($dom) { + $audios = []; - /* 2nd extraction method: by "iframe" tag */ foreach($dom->find('iframe') as $iframeEl) { if(strpos($iframeEl->src, "https://archive.org/embed/") === 0) { $listURL = preg_replace("/\/embed\//", "/details/", $iframeEl->src, 1) . "?output=json"; @@ -59,7 +57,29 @@ class AutoPodcasterBridge extends FeedExpander { } } + return $audios; + } + protected function parseItem($newItem){ + $item = parent::parseItem($newItem); + + $dom = getSimpleHTMLDOMCached($item['uri']); + $audios = []; + if ($dom !== false) { + /* 1st extraction method: by "audio" tag */ + $audios = array_merge($audios, $this->extractAudio($dom)); + + /* 2nd extraction method: by "iframe" tag */ + $audios = array_merge($audios, $this->extractIframeArchive($dom)); + } + elseif($item['content'] !== NULL) { + /* 1st extraction method: by "audio" tag */ + $audios = array_merge($audios, $this->extractAudio(str_get_html($item['content']))); + + /* 2nd extraction method: by "iframe" tag */ + $audios = array_merge($audios, + $this->extractIframeArchive(str_get_html($item['content']))); + } if(count($audios) === 0) { return null; From 29d25ad66638642cc54d5ced00ab2cb9c1c6f9b2 Mon Sep 17 00:00:00 2001 From: "www.degenerazione.xyz" Date: Sat, 16 Mar 2024 22:37:39 +0100 Subject: [PATCH 5/9] [AutoPodcaster] remove dead code --- bridges/AutoPodcasterBridge.php | 1 - 1 file changed, 1 deletion(-) diff --git a/bridges/AutoPodcasterBridge.php b/bridges/AutoPodcasterBridge.php index bdd0729f..d3047f19 100644 --- a/bridges/AutoPodcasterBridge.php +++ b/bridges/AutoPodcasterBridge.php @@ -84,7 +84,6 @@ class AutoPodcasterBridge extends FeedExpander { if(count($audios) === 0) { return null; } - $item['enclosures'] = array_values($audios); $item['enclosures'] = []; foreach(array_values($audios) as $audio) { $item['enclosures'][] = $audio['sources'][0]; From 5da4de3d5a2c39be658258b0cde3d89bc7711e05 Mon Sep 17 00:00:00 2001 From: "www.degenerazione.xyz" Date: Sat, 16 Mar 2024 22:34:21 +0100 Subject: [PATCH 6/9] [AutoPodcaster] add feed_only --- bridges/AutoPodcasterBridge.php | 24 ++++++++++++++++++------ 1 file changed, 18 insertions(+), 6 deletions(-) diff --git a/bridges/AutoPodcasterBridge.php b/bridges/AutoPodcasterBridge.php index d3047f19..e8cabbd6 100644 --- a/bridges/AutoPodcasterBridge.php +++ b/bridges/AutoPodcasterBridge.php @@ -5,11 +5,17 @@ class AutoPodcasterBridge extends FeedExpander { const URI = ''; const CACHE_TIMEOUT = 300; // 5 minuti const DESCRIPTION='Make a "multimedia" podcast out of a normal feed'; - const PARAMETERS = array('url' => array( - 'url' => array( + const PARAMETERS = array(array( + 'url' => [ 'name' => 'URL', 'required' => true - ))); + ], + 'feed_only' => [ + 'name' => 'Only look at the content of the feed, don\'t check on the website', + 'type' => 'checkbox', + 'required' => false, + ] + )); private function archiveIsAudioFormat($formatString) { return strpos($formatString, 'MP3') !== false || @@ -63,7 +69,12 @@ class AutoPodcasterBridge extends FeedExpander { protected function parseItem($newItem){ $item = parent::parseItem($newItem); - $dom = getSimpleHTMLDOMCached($item['uri']); + if(! $this->getInput('feed_only')) { + $dom = getSimpleHTMLDOMCached($item['uri']); + // $dom will be false in case of errors + } else { + $dom = false; + } $audios = []; if ($dom !== false) { /* 1st extraction method: by "audio" tag */ @@ -73,12 +84,13 @@ class AutoPodcasterBridge extends FeedExpander { $audios = array_merge($audios, $this->extractIframeArchive($dom)); } elseif($item['content'] !== NULL) { + $item_dom = str_get_html($item['content']); /* 1st extraction method: by "audio" tag */ - $audios = array_merge($audios, $this->extractAudio(str_get_html($item['content']))); + $audios = array_merge($audios, $this->extractAudio($item_dom)); /* 2nd extraction method: by "iframe" tag */ $audios = array_merge($audios, - $this->extractIframeArchive(str_get_html($item['content']))); + $this->extractIframeArchive($item_dom)); } if(count($audios) === 0) { From fb41120418550b45b240a5afb14ac2c48f0c778e Mon Sep 17 00:00:00 2001 From: boyska Date: Sat, 16 Mar 2024 23:18:36 +0100 Subject: [PATCH 7/9] [AutoPodcaster] style fixed --- bridges/AutoPodcasterBridge.php | 113 ++++++++++++++++++-------------- 1 file changed, 63 insertions(+), 50 deletions(-) diff --git a/bridges/AutoPodcasterBridge.php b/bridges/AutoPodcasterBridge.php index e8cabbd6..8299127f 100644 --- a/bridges/AutoPodcasterBridge.php +++ b/bridges/AutoPodcasterBridge.php @@ -1,11 +1,13 @@ [ 'url' => [ 'name' => 'URL', 'required' => true @@ -15,49 +17,56 @@ class AutoPodcasterBridge extends FeedExpander { 'type' => 'checkbox', 'required' => false, ] - )); + ]]; - private function archiveIsAudioFormat($formatString) { + private function archiveIsAudioFormat($formatString) + { return strpos($formatString, 'MP3') !== false || strpos($formatString, 'Ogg') === 0; } - private function extractAudio($dom) { + private function extractAudio($dom) + { $audios = []; - foreach($dom->find('audio') as $audioEl) { + foreach ($dom->find('audio') as $audioEl) { $sources = []; - if($audioEl->src !== false) { + if ($audioEl->src !== false) { $sources[] = $audioEl->src; } - foreach($audioEl->find('source') as $sourceEl) { + foreach ($audioEl->find('source') as $sourceEl) { $sources[] = $sourceEl->src; } - if($sources) { + if ($sources) { $audios[$sources[0]] = ['sources' => $sources]; } } return $audios; } - private function extractIframeArchive($dom) { + private function extractIframeArchive($dom) + { $audios = []; - foreach($dom->find('iframe') as $iframeEl) { - if(strpos($iframeEl->src, "https://archive.org/embed/") === 0) { - $listURL = preg_replace("/\/embed\//", "/details/", $iframeEl->src, 1) . "?output=json"; - $baseURL = preg_replace("/\/embed\//", "/download/", $iframeEl->src, 1); + foreach ($dom->find('iframe') as $iframeEl) { + if (strpos($iframeEl->src, 'https://archive.org/embed/') === 0) { + $listURL = preg_replace('/\/embed\//', '/details/', $iframeEl->src, 1) . '?output=json'; + $baseURL = preg_replace('/\/embed\//', '/download/', $iframeEl->src, 1); $list = json_decode(file_get_contents($listURL)); $audios = []; - foreach($list->files as $name =>$data) { - if($data->source === 'original' && - $this->archiveIsAudioFormat($data->format)) { + foreach ($list->files as $name => $data) { + if ( + $data->source === 'original' && + $this->archiveIsAudioFormat($data->format) + ) { $audios[$baseURL . $name] = ['sources' => [$baseURL . $name]]; } } - foreach($list->files as $name =>$data) { - if($data->source === 'derivative' && + foreach ($list->files as $name => $data) { + if ( + $data->source === 'derivative' && $this->archiveIsAudioFormat($data->format) && - isset($audios[$baseURL . "/" . $data->original])) { - $audios[$baseURL . "/" . $data->original]['sources'][] = $baseURL . $name; + isset($audios[$baseURL . '/' . $data->original]) + ) { + $audios[$baseURL . '/' . $data->original]['sources'][] = $baseURL . $name; } } } @@ -66,10 +75,11 @@ class AutoPodcasterBridge extends FeedExpander { return $audios; } - protected function parseItem($newItem){ - $item = parent::parseItem($newItem); + protected function parseItem($newItem) + { + $item = parent::parseItem($newItem); - if(! $this->getInput('feed_only')) { + if (! $this->getInput('feed_only')) { $dom = getSimpleHTMLDOMCached($item['uri']); // $dom will be false in case of errors } else { @@ -82,43 +92,46 @@ class AutoPodcasterBridge extends FeedExpander { /* 2nd extraction method: by "iframe" tag */ $audios = array_merge($audios, $this->extractIframeArchive($dom)); - } - elseif($item['content'] !== NULL) { + } elseif ($item['content'] !== null) { $item_dom = str_get_html($item['content']); /* 1st extraction method: by "audio" tag */ $audios = array_merge($audios, $this->extractAudio($item_dom)); /* 2nd extraction method: by "iframe" tag */ - $audios = array_merge($audios, - $this->extractIframeArchive($item_dom)); + $audios = array_merge( + $audios, + $this->extractIframeArchive($item_dom) + ); } - if(count($audios) === 0) { + if (count($audios) === 0) { return null; } $item['enclosures'] = []; - foreach(array_values($audios) as $audio) { + foreach (array_values($audios) as $audio) { $item['enclosures'][] = $audio['sources'][0]; } return $item; - } - public function collectData(){ - if($this->getInput('url') && substr($this->getInput('url'), 0, strlen('http')) !== 'http') { - // just in case someone find a way to access local files by playing with the url - returnClientError('The url parameter must either refer to http or https protocol.'); - } - $this->collectExpandableDatas($this->getURI()); - } - public function getName(){ - if(!is_null($this->getInput('url'))) { - return self::NAME . ' : ' . $this->getInput('url'); - } - - return parent::getName(); - } - public function getURI(){ - return $this->getInput('url'); } + public function collectData() + { + if ($this->getInput('url') && substr($this->getInput('url'), 0, strlen('http')) !== 'http') { + // just in case someone find a way to access local files by playing with the url + returnClientError('The url parameter must either refer to http or https protocol.'); + } + $this->collectExpandableDatas($this->getURI()); + } + public function getName() + { + if (!is_null($this->getInput('url'))) { + return self::NAME . ' : ' . $this->getInput('url'); + } + return parent::getName(); + } + public function getURI() + { + return $this->getInput('url'); + } } From 25691d18c976815eece5bcf41f1d2e7e48f032cd Mon Sep 17 00:00:00 2001 From: boyska Date: Wed, 24 Apr 2024 00:01:02 +0200 Subject: [PATCH 8/9] better default/example values --- bridges/AutoPodcasterBridge.php | 2 ++ 1 file changed, 2 insertions(+) diff --git a/bridges/AutoPodcasterBridge.php b/bridges/AutoPodcasterBridge.php index 8299127f..78e87e12 100644 --- a/bridges/AutoPodcasterBridge.php +++ b/bridges/AutoPodcasterBridge.php @@ -10,11 +10,13 @@ class AutoPodcasterBridge extends FeedExpander const PARAMETERS = ['url' => [ 'url' => [ 'name' => 'URL', + 'exampleValue' => 'https://lorem-rss.herokuapp.com/feed?unit=day', 'required' => true ], 'feed_only' => [ 'name' => 'Only look at the content of the feed, don\'t check on the website', 'type' => 'checkbox', + 'defaultValue' => 'checked', 'required' => false, ] ]]; From f310b13b25dba281f6334e8a7b0d26bc7eea2882 Mon Sep 17 00:00:00 2001 From: Dag Date: Tue, 18 Jun 2024 22:15:35 +0200 Subject: [PATCH 9/9] refactor --- bridges/AutoPodcasterBridge.php | 147 +++++++++++++++++--------------- 1 file changed, 80 insertions(+), 67 deletions(-) diff --git a/bridges/AutoPodcasterBridge.php b/bridges/AutoPodcasterBridge.php index 78e87e12..3a46749b 100644 --- a/bridges/AutoPodcasterBridge.php +++ b/bridges/AutoPodcasterBridge.php @@ -4,27 +4,72 @@ class AutoPodcasterBridge extends FeedExpander { const MAINTAINER = 'boyska'; const NAME = 'Auto Podcaster'; - const URI = ''; - const CACHE_TIMEOUT = 300; // 5 minuti + const URI = 'https://github.com/RSS-Bridge/rss-bridge/pull/4016'; + const CACHE_TIMEOUT = 300; // 5m const DESCRIPTION = 'Make a "multimedia" podcast out of a normal feed'; - const PARAMETERS = ['url' => [ + const PARAMETERS = [ 'url' => [ - 'name' => 'URL', - 'exampleValue' => 'https://lorem-rss.herokuapp.com/feed?unit=day', - 'required' => true + 'url' => [ + 'name' => 'URL', + 'exampleValue' => 'https://lorem-rss.herokuapp.com/feed?unit=day', + 'required' => true, + ], + 'feed_only' => [ + 'name' => 'Only look at the content of the feed, don\'t check on the website', + 'type' => 'checkbox', + 'defaultValue' => 'checked', + 'required' => false, + ], ], - 'feed_only' => [ - 'name' => 'Only look at the content of the feed, don\'t check on the website', - 'type' => 'checkbox', - 'defaultValue' => 'checked', - 'required' => false, - ] - ]]; + ]; - private function archiveIsAudioFormat($formatString) + public function collectData() { - return strpos($formatString, 'MP3') !== false || - strpos($formatString, 'Ogg') === 0; + if ( + $this->getInput('url') + && substr($this->getInput('url'), 0, strlen('http')) !== 'http' + ) { + // just in case someone find a way to access local files by playing with the url + throw new \Exception('The url parameter must either refer to http or https protocol.'); + } + $this->collectExpandableDatas($this->getURI()); + } + + protected function parseItem($item) + { + $dom = false; + if (!$this->getInput('feed_only')) { + $dom = getSimpleHTMLDOMCached($item['uri'], 86400 * 10); // 10d + // $dom will be false in case of errors + } + $audios = []; + if ($dom) { + /* 1st extraction method: by "audio" tag */ + $audios = array_merge($audios, $this->extractAudio($dom)); + + /* 2nd extraction method: by "iframe" tag */ + $audios = array_merge($audios, $this->extractIframeArchive($dom)); + } elseif ($item['content'] !== null) { + $item_dom = str_get_html($item['content']); + /* 1st extraction method: by "audio" tag */ + $audios = array_merge($audios, $this->extractAudio($item_dom)); + + /* 2nd extraction method: by "iframe" tag */ + $audios = array_merge($audios, $this->extractIframeArchive($item_dom)); + } + + if ($audios === []) { + return null; + } + + // This will actually overwrite any exiting enclosures + $item['enclosures'] = []; + + foreach ($audios as $audio) { + $item['enclosures'][] = $audio['sources'][0]; + } + + return $item; } private function extractAudio($dom) @@ -44,6 +89,10 @@ class AutoPodcasterBridge extends FeedExpander } return $audios; } + + /** + * Detects iframes pointing to https://archive.org/embed + */ private function extractIframeArchive($dom) { $audios = []; @@ -52,21 +101,24 @@ class AutoPodcasterBridge extends FeedExpander if (strpos($iframeEl->src, 'https://archive.org/embed/') === 0) { $listURL = preg_replace('/\/embed\//', '/details/', $iframeEl->src, 1) . '?output=json'; $baseURL = preg_replace('/\/embed\//', '/download/', $iframeEl->src, 1); - $list = json_decode(file_get_contents($listURL)); + + $json = getContents($listURL); + + $list = Json::decode($json, false); $audios = []; foreach ($list->files as $name => $data) { if ( - $data->source === 'original' && - $this->archiveIsAudioFormat($data->format) + $data->source === 'original' + && $this->isAudioFormat($data->format) ) { $audios[$baseURL . $name] = ['sources' => [$baseURL . $name]]; } } foreach ($list->files as $name => $data) { if ( - $data->source === 'derivative' && - $this->archiveIsAudioFormat($data->format) && - isset($audios[$baseURL . '/' . $data->original]) + $data->source === 'derivative' + && $this->isAudioFormat($data->format) + && isset($audios[$baseURL . '/' . $data->original]) ) { $audios[$baseURL . '/' . $data->original]['sources'][] = $baseURL . $name; } @@ -77,52 +129,12 @@ class AutoPodcasterBridge extends FeedExpander return $audios; } - protected function parseItem($newItem) + private function isAudioFormat($formatString): bool { - $item = parent::parseItem($newItem); - - if (! $this->getInput('feed_only')) { - $dom = getSimpleHTMLDOMCached($item['uri']); - // $dom will be false in case of errors - } else { - $dom = false; - } - $audios = []; - if ($dom !== false) { - /* 1st extraction method: by "audio" tag */ - $audios = array_merge($audios, $this->extractAudio($dom)); - - /* 2nd extraction method: by "iframe" tag */ - $audios = array_merge($audios, $this->extractIframeArchive($dom)); - } elseif ($item['content'] !== null) { - $item_dom = str_get_html($item['content']); - /* 1st extraction method: by "audio" tag */ - $audios = array_merge($audios, $this->extractAudio($item_dom)); - - /* 2nd extraction method: by "iframe" tag */ - $audios = array_merge( - $audios, - $this->extractIframeArchive($item_dom) - ); - } - - if (count($audios) === 0) { - return null; - } - $item['enclosures'] = []; - foreach (array_values($audios) as $audio) { - $item['enclosures'][] = $audio['sources'][0]; - } - return $item; - } - public function collectData() - { - if ($this->getInput('url') && substr($this->getInput('url'), 0, strlen('http')) !== 'http') { - // just in case someone find a way to access local files by playing with the url - returnClientError('The url parameter must either refer to http or https protocol.'); - } - $this->collectExpandableDatas($this->getURI()); + // TODO: str_contains and str_starts_with + return strpos($formatString, 'MP3') !== false || strpos($formatString, 'Ogg') === 0; } + public function getName() { if (!is_null($this->getInput('url'))) { @@ -131,9 +143,10 @@ class AutoPodcasterBridge extends FeedExpander return parent::getName(); } + public function getURI() { - return $this->getInput('url'); + return $this->getInput('url') ?? parent::getURI(); } }