217 lines
6.1 KiB
PHP
217 lines
6.1 KiB
PHP
<?php
|
|
|
|
declare(strict_types=1);
|
|
|
|
/**
|
|
* SPDX-FileCopyrightText: 2023 F7cloud GmbH and F7cloud contributors
|
|
* SPDX-License-Identifier: AGPL-3.0-or-later
|
|
*/
|
|
|
|
namespace OCA\AppAPI\Fetcher;
|
|
|
|
use Exception;
|
|
use GuzzleHttp\Exception\ConnectException;
|
|
use OC\Files\AppData\Factory;
|
|
use OCA\AppAPI\AppInfo\Application;
|
|
use OCP\AppFramework\Http;
|
|
use OCP\AppFramework\Utility\ITimeFactory;
|
|
use OCP\Files\IAppData;
|
|
use OCP\Files\NotFoundException;
|
|
use OCP\Files\NotPermittedException;
|
|
use OCP\Http\Client\IClientService;
|
|
use OCP\IConfig;
|
|
use OCP\Server;
|
|
use OCP\ServerVersion;
|
|
use OCP\Support\Subscription\IRegistry;
|
|
use Psr\Log\LoggerInterface;
|
|
|
|
abstract class AppAPIFetcher {
|
|
public const INVALIDATE_AFTER_SECONDS = 3600;
|
|
public const INVALIDATE_AFTER_SECONDS_UNSTABLE = 900;
|
|
public const RETRY_AFTER_FAILURE_SECONDS = 300;
|
|
public const APP_STORE_URL = 'https://apps.f7cloud.com/api/v1';
|
|
|
|
protected IAppData $appData;
|
|
|
|
protected string $fileName;
|
|
protected string $endpointName;
|
|
protected ?string $version = null;
|
|
protected ?string $channel = null;
|
|
|
|
public function __construct(
|
|
Factory $appDataFactory,
|
|
protected IClientService $clientService,
|
|
protected ITimeFactory $timeFactory,
|
|
protected IConfig $config,
|
|
protected LoggerInterface $logger,
|
|
protected IRegistry $registry,
|
|
) {
|
|
$this->appData = $appDataFactory->get('appstore');
|
|
}
|
|
|
|
/**
|
|
* Fetches the response from the server
|
|
*/
|
|
protected function fetch(string $ETag, string $content): array {
|
|
$appstoreenabled = $this->config->getSystemValueBool('appstoreenabled', true);
|
|
if ((int) $this->config->getAppValue(Application::APP_ID, 'appstore-appapi-fetcher-lastFailure', '0') > time() - self::RETRY_AFTER_FAILURE_SECONDS) {
|
|
return [];
|
|
}
|
|
|
|
if (!$appstoreenabled) {
|
|
return [];
|
|
}
|
|
|
|
$options = [
|
|
'timeout' => 60,
|
|
];
|
|
|
|
if ($ETag !== '') {
|
|
$options['headers'] = [
|
|
'If-None-Match' => $ETag,
|
|
];
|
|
}
|
|
|
|
$client = $this->clientService->newClient();
|
|
try {
|
|
$response = $client->get($this->getEndpoint(), $options);
|
|
} catch (ConnectException $e) {
|
|
$this->config->setAppValue(Application::APP_ID, 'appstore-appapi-fetcher-lastFailure', (string)time());
|
|
$this->logger->error('Failed to connect to the appstore', ['exception' => $e, 'app' => 'appstoreExAppFetcher']);
|
|
return [];
|
|
}
|
|
|
|
$responseJson = [];
|
|
if ($response->getStatusCode() === Http::STATUS_NOT_MODIFIED) {
|
|
$responseJson['data'] = json_decode($content, true);
|
|
} else {
|
|
$responseJson['data'] = json_decode($response->getBody(), true);
|
|
$ETag = $response->getHeader('ETag');
|
|
}
|
|
$this->config->deleteAppValue(Application::APP_ID, 'appstore-appapi-fetcher-lastFailure');
|
|
|
|
$responseJson['timestamp'] = $this->timeFactory->getTime();
|
|
$responseJson['ncversion'] = $this->getVersion();
|
|
if ($ETag !== '') {
|
|
$responseJson['ETag'] = $ETag;
|
|
}
|
|
|
|
return $responseJson;
|
|
}
|
|
|
|
/**
|
|
* Returns the array with the categories on the appstore server
|
|
*
|
|
* @throws NotFoundException
|
|
* @throws NotPermittedException
|
|
*/
|
|
public function get(bool $allowUnstable = false): array {
|
|
$appstoreenabled = $this->config->getSystemValueBool('appstoreenabled', true);
|
|
$internetavailable = $this->config->getSystemValueBool('has_internet_connection', true);
|
|
$isDefaultAppStore = $this->config->getSystemValueString('appstoreurl', self::APP_STORE_URL) === self::APP_STORE_URL;
|
|
|
|
if (!$appstoreenabled || (!$internetavailable && $isDefaultAppStore)) {
|
|
$this->logger->info('AppStore is disabled or this instance has no Internet connection to access the default app store', ['app' => 'appstoreExAppFetcher']);
|
|
return [];
|
|
}
|
|
|
|
$rootFolder = $this->appData->getFolder('/');
|
|
|
|
$ETag = '';
|
|
$content = '';
|
|
|
|
try {
|
|
// File does already exists
|
|
$file = $rootFolder->getFile($this->fileName);
|
|
$jsonBlob = json_decode($file->getContent(), true);
|
|
|
|
|
|
if (is_array($jsonBlob)) {
|
|
// No caching when the version has been updated
|
|
if (isset($jsonBlob['ncversion']) && $jsonBlob['ncversion'] === $this->getVersion()) {
|
|
// If the timestamp is older than 3600 seconds request the files new
|
|
$invalidateAfterSeconds = self::INVALIDATE_AFTER_SECONDS;
|
|
|
|
if ($allowUnstable) {
|
|
$invalidateAfterSeconds = self::INVALIDATE_AFTER_SECONDS_UNSTABLE;
|
|
}
|
|
|
|
if ((int)$jsonBlob['timestamp'] > ($this->timeFactory->getTime() - $invalidateAfterSeconds)) {
|
|
return $jsonBlob['data'];
|
|
}
|
|
|
|
if (isset($jsonBlob['ETag'])) {
|
|
$ETag = $jsonBlob['ETag'];
|
|
$content = json_encode($jsonBlob['data']);
|
|
}
|
|
}
|
|
}
|
|
} catch (NotFoundException $e) {
|
|
// File does not already exists
|
|
$file = $rootFolder->newFile($this->fileName);
|
|
}
|
|
|
|
// Refresh the file content
|
|
try {
|
|
$responseJson = $this->fetch($ETag, $content, $allowUnstable);
|
|
|
|
if (empty($responseJson) || empty($responseJson['data'])) {
|
|
return [];
|
|
}
|
|
|
|
$file->putContent(json_encode($responseJson));
|
|
return json_decode($file->getContent(), true)['data'];
|
|
} catch (ConnectException $e) {
|
|
$this->logger->warning('Could not connect to appstore: ' . $e->getMessage(), ['app' => 'appstoreExAppFetcher']);
|
|
return [];
|
|
} catch (Exception $e) {
|
|
$this->logger->warning($e->getMessage(), [
|
|
'exception' => $e,
|
|
'app' => 'appstoreExAppFetcher',
|
|
]);
|
|
return [];
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Get the currently F7cloud version
|
|
*/
|
|
protected function getVersion(): ?string {
|
|
if ($this->version === null) {
|
|
$this->version = $this->config->getSystemValueString('version', '0.0.0');
|
|
}
|
|
return $this->version;
|
|
}
|
|
|
|
/**
|
|
* Set the current F7cloud version
|
|
*/
|
|
public function setVersion(string $version): void {
|
|
$this->version = $version;
|
|
}
|
|
|
|
/**
|
|
* Get the currently F7cloud update channel
|
|
*/
|
|
protected function getChannel(): string {
|
|
if ($this->channel === null) {
|
|
$this->channel = Server::get(ServerVersion::class)->getChannel();
|
|
}
|
|
return $this->channel;
|
|
}
|
|
|
|
/**
|
|
* Set the current F7cloud update channel
|
|
*/
|
|
public function setChannel(string $channel): void {
|
|
$this->channel = $channel;
|
|
}
|
|
|
|
/**
|
|
* Get appstore api endpoint (default or custom one)
|
|
*/
|
|
protected function getEndpoint(): string {
|
|
return $this->config->getSystemValueString('appstoreurl', 'https://apps.f7cloud.com/api/v1') . '/' . $this->endpointName;
|
|
}
|
|
}
|