mirror of
https://github.com/RSS-Bridge/rss-bridge.git
synced 2025-04-05 17:19:37 +00:00
fix(FDroidRepoBridge): unlink when json file is absent from archive (#4056)
This commit is contained in:
parent
82606a479a
commit
3cba984d22
@ -49,7 +49,7 @@ class FDroidRepoBridge extends BridgeAbstract
|
|||||||
throw new \Exception('FDroidRepoBridge requires the php-zip extension');
|
throw new \Exception('FDroidRepoBridge requires the php-zip extension');
|
||||||
}
|
}
|
||||||
|
|
||||||
$this->repo = $this->getRepo();
|
$this->repo = $this->fetchData();
|
||||||
switch ($this->queriedContext) {
|
switch ($this->queriedContext) {
|
||||||
case 'Latest Updates':
|
case 'Latest Updates':
|
||||||
$this->getAllUpdates();
|
$this->getAllUpdates();
|
||||||
@ -58,63 +58,40 @@ class FDroidRepoBridge extends BridgeAbstract
|
|||||||
$this->getPackage($this->getInput('package'));
|
$this->getPackage($this->getInput('package'));
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
returnServerError('Unimplemented Context (collectData)');
|
throw new \Exception('Unimplemented Context (collectData)');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public function getURI()
|
/**
|
||||||
{
|
* This method fetches data from arbitrary url and writes to os temp file.
|
||||||
if (empty($this->queriedContext)) {
|
* I don't think there's any security problem here but might be DOS problems.
|
||||||
return parent::getURI();
|
*/
|
||||||
}
|
private function fetchData()
|
||||||
|
|
||||||
$url = rtrim($this->GetInput('url'), '/');
|
|
||||||
return strstr($url, '?', true) ?: $url;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function getName()
|
|
||||||
{
|
|
||||||
if (empty($this->queriedContext)) {
|
|
||||||
return parent::getName();
|
|
||||||
}
|
|
||||||
|
|
||||||
$name = $this->repo['repo']['name'];
|
|
||||||
switch ($this->queriedContext) {
|
|
||||||
case 'Latest Updates':
|
|
||||||
return $name;
|
|
||||||
case 'Follow Package':
|
|
||||||
return $this->getInput('package') . ' - ' . $name;
|
|
||||||
default:
|
|
||||||
returnServerError('Unimplemented Context (getName)');
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private function getRepo()
|
|
||||||
{
|
{
|
||||||
$url = $this->getURI();
|
$url = $this->getURI();
|
||||||
|
|
||||||
// Get repo information (only available as JAR)
|
$zipFile = getContents($url . '/index-v1.jar');
|
||||||
$jar = getContents($url . '/index-v1.jar');
|
// On linux this creates a temp file in /tmp/
|
||||||
$jar_loc = tempnam(sys_get_temp_dir(), '');
|
$temporaryFile = tempnam(sys_get_temp_dir(), 'rssbridge_');
|
||||||
file_put_contents($jar_loc, $jar);
|
file_put_contents($temporaryFile, $zipFile);
|
||||||
|
|
||||||
// JAR files are specially formatted ZIP files
|
$archive = new \ZipArchive();
|
||||||
$jar = new \ZipArchive();
|
if ($archive->open($temporaryFile) !== true) {
|
||||||
if ($jar->open($jar_loc) !== true) {
|
unlink($temporaryFile);
|
||||||
unlink($jar_loc);
|
|
||||||
throw new \Exception('Failed to extract archive');
|
throw new \Exception('Failed to extract archive');
|
||||||
}
|
}
|
||||||
|
|
||||||
// Get file pointer to the relevant JSON inside
|
$fp = $archive->getStream('index-v1.json');
|
||||||
$fp = $jar->getStream('index-v1.json');
|
|
||||||
if (!$fp) {
|
if (!$fp) {
|
||||||
returnServerError('Failed to get file pointer');
|
unlink($temporaryFile);
|
||||||
|
throw new \Exception('Failed to get file pointer');
|
||||||
}
|
}
|
||||||
|
|
||||||
$data = json_decode(stream_get_contents($fp), true);
|
$json = stream_get_contents($fp);
|
||||||
fclose($fp);
|
fclose($fp);
|
||||||
$jar->close();
|
$data = Json::decode($json);
|
||||||
unlink($jar_loc);
|
$archive->close();
|
||||||
|
unlink($temporaryFile);
|
||||||
return $data;
|
return $data;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -158,9 +135,9 @@ class FDroidRepoBridge extends BridgeAbstract
|
|||||||
$summary = $lang['summary'] ?? $app['summary'] ?? '';
|
$summary = $lang['summary'] ?? $app['summary'] ?? '';
|
||||||
$description = markdownToHtml(trim($lang['description'] ?? $app['description'] ?? 'None'));
|
$description = markdownToHtml(trim($lang['description'] ?? $app['description'] ?? 'None'));
|
||||||
$whatsNew = markdownToHtml(trim($lang['whatsNew'] ?? 'None'));
|
$whatsNew = markdownToHtml(trim($lang['whatsNew'] ?? 'None'));
|
||||||
$website = $this->link($lang['webSite'] ?? $app['webSite'] ?? $app['authorWebSite'] ?? null);
|
$website = $this->createAnchor($lang['webSite'] ?? $app['webSite'] ?? $app['authorWebSite'] ?? null);
|
||||||
$source = $this->link($app['sourceCode'] ?? null);
|
$source = $this->createAnchor($app['sourceCode'] ?? null);
|
||||||
$issueTracker = $this->link($app['issueTracker'] ?? null);
|
$issueTracker = $this->createAnchor($app['issueTracker'] ?? null);
|
||||||
$license = $app['license'] ?? 'None';
|
$license = $app['license'] ?? 'None';
|
||||||
$item['content'] = <<<EOD
|
$item['content'] = <<<EOD
|
||||||
{$icon}
|
{$icon}
|
||||||
@ -182,7 +159,7 @@ EOD;
|
|||||||
private function getPackage($package)
|
private function getPackage($package)
|
||||||
{
|
{
|
||||||
if (!isset($this->repo['packages'][$package])) {
|
if (!isset($this->repo['packages'][$package])) {
|
||||||
returnClientError('Invalid Package Name');
|
throw new \Exception('Invalid Package Name');
|
||||||
}
|
}
|
||||||
$package = $this->repo['packages'][$package];
|
$package = $this->repo['packages'][$package];
|
||||||
|
|
||||||
@ -192,7 +169,7 @@ EOD;
|
|||||||
$item['uri'] = $this->getURI() . '/' . $version['apkName'];
|
$item['uri'] = $this->getURI() . '/' . $version['apkName'];
|
||||||
$item['title'] = $version['versionName'];
|
$item['title'] = $version['versionName'];
|
||||||
$item['timestamp'] = date(DateTime::ISO8601, (int) ($version['added'] / 1000));
|
$item['timestamp'] = date(DateTime::ISO8601, (int) ($version['added'] / 1000));
|
||||||
$item['uid'] = $version['versionCode'];
|
$item['uid'] = (string) $version['versionCode'];
|
||||||
$size = round($version['size'] / 1048576, 1); // Bytes -> MB
|
$size = round($version['size'] / 1048576, 1); // Bytes -> MB
|
||||||
$sdk_link = 'https://developer.android.com/studio/releases/platforms';
|
$sdk_link = 'https://developer.android.com/studio/releases/platforms';
|
||||||
$item['content'] = <<<EOD
|
$item['content'] = <<<EOD
|
||||||
@ -208,11 +185,42 @@ EOD;
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private function link($url)
|
public function getURI()
|
||||||
|
{
|
||||||
|
if (empty($this->queriedContext)) {
|
||||||
|
return parent::getURI();
|
||||||
|
}
|
||||||
|
|
||||||
|
$url = rtrim($this->getInput('url'), '/');
|
||||||
|
if (strstr($url, '?', true)) {
|
||||||
|
return strstr($url, '?', true);
|
||||||
|
} else {
|
||||||
|
return $url;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getName()
|
||||||
|
{
|
||||||
|
if (empty($this->queriedContext)) {
|
||||||
|
return parent::getName();
|
||||||
|
}
|
||||||
|
|
||||||
|
$name = $this->repo['repo']['name'];
|
||||||
|
switch ($this->queriedContext) {
|
||||||
|
case 'Latest Updates':
|
||||||
|
return $name;
|
||||||
|
case 'Follow Package':
|
||||||
|
return $this->getInput('package') . ' - ' . $name;
|
||||||
|
default:
|
||||||
|
throw new \Exception('Unimplemented Context (getName)');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private function createAnchor($url)
|
||||||
{
|
{
|
||||||
if (empty($url)) {
|
if (empty($url)) {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
return '<a href="' . $url . '">' . $url . '</a>';
|
return sprintf('<a href="%s">%s</a>', $url, $url);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user