fix(FDroidRepoBridge): unlink when json file is absent from archive (#4056)

This commit is contained in:
Dag 2024-04-04 17:43:07 +02:00 committed by GitHub
parent 82606a479a
commit 3cba984d22
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194

View File

@ -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);
} }
} }