Merge pull request #8579 from nextcloud/feature/2692/merge-can-i-update
Display a list of unavailable apps for updates (aka. merge "Can I update?")
This commit is contained in:
commit
c88112bb41
|
@ -34,6 +34,7 @@
|
||||||
!/apps/updatenotification
|
!/apps/updatenotification
|
||||||
/apps/updatenotification/js/merged.js
|
/apps/updatenotification/js/merged.js
|
||||||
/apps/updatenotification/js/merged.js.map
|
/apps/updatenotification/js/merged.js.map
|
||||||
|
/apps/updatenotification/js/*.hot-update.*
|
||||||
/apps/updatenotification/node_modules
|
/apps/updatenotification/node_modules
|
||||||
!/apps/theming
|
!/apps/theming
|
||||||
!/apps/twofactor_backupcodes
|
!/apps/twofactor_backupcodes
|
||||||
|
|
|
@ -21,10 +21,12 @@
|
||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
|
|
||||||
use OCA\UpdateNotification\AppInfo\Application;
|
return [
|
||||||
|
'routes' => [
|
||||||
$application = new Application();
|
['name' => 'Admin#createCredentials', 'url' => '/credentials', 'verb' => 'GET'],
|
||||||
$application->registerRoutes($this, ['routes' => [
|
['name' => 'Admin#setChannel', 'url' => '/channel', 'verb' => 'POST'],
|
||||||
['name' => 'Admin#createCredentials', 'url' => '/credentials', 'verb' => 'GET'],
|
],
|
||||||
['name' => 'Admin#setChannel', 'url' => '/channel', 'verb' => 'POST'],
|
'ocs' => [
|
||||||
]]);
|
['name' => 'API#getAppList', 'url' => '/api/{apiVersion}/applist/{newVersion}', 'verb' => 'GET', 'requirements' => ['apiVersion' => 'v1']],
|
||||||
|
],
|
||||||
|
];
|
||||||
|
|
|
@ -7,6 +7,7 @@ $baseDir = $vendorDir;
|
||||||
|
|
||||||
return array(
|
return array(
|
||||||
'OCA\\UpdateNotification\\AppInfo\\Application' => $baseDir . '/../lib/AppInfo/Application.php',
|
'OCA\\UpdateNotification\\AppInfo\\Application' => $baseDir . '/../lib/AppInfo/Application.php',
|
||||||
|
'OCA\\UpdateNotification\\Controller\\APIController' => $baseDir . '/../lib/Controller/APIController.php',
|
||||||
'OCA\\UpdateNotification\\Controller\\AdminController' => $baseDir . '/../lib/Controller/AdminController.php',
|
'OCA\\UpdateNotification\\Controller\\AdminController' => $baseDir . '/../lib/Controller/AdminController.php',
|
||||||
'OCA\\UpdateNotification\\Notification\\BackgroundJob' => $baseDir . '/../lib/Notification/BackgroundJob.php',
|
'OCA\\UpdateNotification\\Notification\\BackgroundJob' => $baseDir . '/../lib/Notification/BackgroundJob.php',
|
||||||
'OCA\\UpdateNotification\\Notification\\Notifier' => $baseDir . '/../lib/Notification/Notifier.php',
|
'OCA\\UpdateNotification\\Notification\\Notifier' => $baseDir . '/../lib/Notification/Notifier.php',
|
||||||
|
|
|
@ -22,6 +22,7 @@ class ComposerStaticInitUpdateNotification
|
||||||
|
|
||||||
public static $classMap = array (
|
public static $classMap = array (
|
||||||
'OCA\\UpdateNotification\\AppInfo\\Application' => __DIR__ . '/..' . '/../lib/AppInfo/Application.php',
|
'OCA\\UpdateNotification\\AppInfo\\Application' => __DIR__ . '/..' . '/../lib/AppInfo/Application.php',
|
||||||
|
'OCA\\UpdateNotification\\Controller\\APIController' => __DIR__ . '/..' . '/../lib/Controller/APIController.php',
|
||||||
'OCA\\UpdateNotification\\Controller\\AdminController' => __DIR__ . '/..' . '/../lib/Controller/AdminController.php',
|
'OCA\\UpdateNotification\\Controller\\AdminController' => __DIR__ . '/..' . '/../lib/Controller/AdminController.php',
|
||||||
'OCA\\UpdateNotification\\Notification\\BackgroundJob' => __DIR__ . '/..' . '/../lib/Notification/BackgroundJob.php',
|
'OCA\\UpdateNotification\\Notification\\BackgroundJob' => __DIR__ . '/..' . '/../lib/Notification/BackgroundJob.php',
|
||||||
'OCA\\UpdateNotification\\Notification\\Notifier' => __DIR__ . '/..' . '/../lib/Notification/Notifier.php',
|
'OCA\\UpdateNotification\\Notification\\Notifier' => __DIR__ . '/..' . '/../lib/Notification/Notifier.php',
|
||||||
|
|
|
@ -1,4 +1,33 @@
|
||||||
#updatenotification p,
|
#updatenotification {
|
||||||
#oca_updatenotification_section p {
|
margin-top: -25px;
|
||||||
margin: 25px 0;
|
}
|
||||||
|
|
||||||
|
#updatenotification div.update,
|
||||||
|
#updatenotification ul,
|
||||||
|
#updatenotification p:not(.inlineblock) {
|
||||||
|
margin-bottom: 25px;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
#updatenotification h2.inlineblock {
|
||||||
|
margin-top: 25px;
|
||||||
|
}
|
||||||
|
|
||||||
|
#updatenotification h3,
|
||||||
|
#updatenotification h3 .icon {
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
|
|
||||||
|
#updatenotification h3:first-of-type {
|
||||||
|
margin-top: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
#updatenotification .icon {
|
||||||
|
display: inline-block;
|
||||||
|
margin-bottom: -3px;
|
||||||
|
}
|
||||||
|
|
||||||
|
#updatenotification .icon-triangle-s,
|
||||||
|
#updatenotification .icon-triangle-n {
|
||||||
|
opacity: 0.5;
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,9 +1,36 @@
|
||||||
<template>
|
<template>
|
||||||
<div id="updatenotification" class="followupsection">
|
<div id="updatenotification" class="followupsection">
|
||||||
<p>
|
<div class="update">
|
||||||
<template v-if="isNewVersionAvailable">
|
<template v-if="isNewVersionAvailable">
|
||||||
<strong>{{newVersionAvailableString}}</strong>
|
<p>
|
||||||
<input v-if="updaterEnabled" type="button" @click="clickUpdaterButton" id="oca_updatenotification_button" :value="t('updatenotification', 'Open updater')">
|
<span v-html="newVersionAvailableString"></span><br>
|
||||||
|
<span v-if="!isListFetched" class="icon icon-loading-small"></span>
|
||||||
|
<span v-html="statusText"></span>
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<template v-if="missingAppUpdates.length">
|
||||||
|
<h3 @click="toggleHideMissingUpdates">
|
||||||
|
{{ t('updatenotification', 'Apps missing updates') }}
|
||||||
|
<span v-if="!hideMissingUpdates" class="icon icon-triangle-n"></span>
|
||||||
|
<span v-if="hideMissingUpdates" class="icon icon-triangle-s"></span>
|
||||||
|
</h3>
|
||||||
|
<ul class="applist" v-if="!hideMissingUpdates">
|
||||||
|
<li v-for="app in missingAppUpdates"><a :href="'https://apps.nextcloud.com/apps/' + app.appId" :title="t('settings', 'View in store')">{{app.appName}} ↗</a></li>
|
||||||
|
</ul>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<template v-if="availableAppUpdates.length">
|
||||||
|
<h3 @click="toggleHideAvailableUpdates">
|
||||||
|
{{ t('updatenotification', 'Apps with available updates') }}
|
||||||
|
<span v-if="!hideAvailableUpdates" class="icon icon-triangle-n"></span>
|
||||||
|
<span v-if="hideAvailableUpdates" class="icon icon-triangle-s"></span>
|
||||||
|
</h3>
|
||||||
|
<ul class="applist">
|
||||||
|
<li v-for="app in availableAppUpdates" v-if="!hideAvailableUpdates"><a :href="'https://apps.nextcloud.com/apps/' + app.appId" :title="t('settings', 'View in store')">{{app.appName}} ↗</a></li>
|
||||||
|
</ul>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<a v-if="updaterEnabled" href="#" class="button" @click="clickUpdaterButton">{{ t('updatenotification', 'Open updater') }}</a>
|
||||||
<a v-if="downloadLink" :href="downloadLink" class="button" :class="{ hidden: !updaterEnabled }">{{ t('updatenotification', 'Download now') }}</a>
|
<a v-if="downloadLink" :href="downloadLink" class="button" :class="{ hidden: !updaterEnabled }">{{ t('updatenotification', 'Download now') }}</a>
|
||||||
</template>
|
</template>
|
||||||
<template v-else-if="!isUpdateChecked">{{ t('updatenotification', 'The update check is not yet finished. Please refresh the page.') }}</template>
|
<template v-else-if="!isUpdateChecked">{{ t('updatenotification', 'The update check is not yet finished. Please refresh the page.') }}</template>
|
||||||
|
@ -16,7 +43,7 @@
|
||||||
<br />
|
<br />
|
||||||
<em>{{ t('updatenotification', 'A non-default update server is in use to be checked for updates:') }} <code>{{updateServerURL}}</code></em>
|
<em>{{ t('updatenotification', 'A non-default update server is in use to be checked for updates:') }} <code>{{updateServerURL}}</code></em>
|
||||||
</template>
|
</template>
|
||||||
</p>
|
</div>
|
||||||
|
|
||||||
<p>
|
<p>
|
||||||
<label for="release-channel">{{ t('updatenotification', 'Update channel:') }}</label>
|
<label for="release-channel">{{ t('updatenotification', 'Update channel:') }}</label>
|
||||||
|
@ -58,7 +85,15 @@
|
||||||
notifyGroups: '',
|
notifyGroups: '',
|
||||||
availableGroups: [],
|
availableGroups: [],
|
||||||
isDefaultUpdateServerURL: true,
|
isDefaultUpdateServerURL: true,
|
||||||
enableChangeWatcher: false
|
enableChangeWatcher: false,
|
||||||
|
|
||||||
|
availableAppUpdates: [],
|
||||||
|
missingAppUpdates: [],
|
||||||
|
appStoreFailed: false,
|
||||||
|
appStoreDisabled: false,
|
||||||
|
isListFetched: false,
|
||||||
|
hideMissingUpdates: false,
|
||||||
|
hideAvailableUpdates: true
|
||||||
};
|
};
|
||||||
},
|
},
|
||||||
|
|
||||||
|
@ -78,19 +113,67 @@
|
||||||
});
|
});
|
||||||
|
|
||||||
OCP.AppConfig.setValue('updatenotification', 'notify_groups', JSON.stringify(selectedGroups));
|
OCP.AppConfig.setValue('updatenotification', 'notify_groups', JSON.stringify(selectedGroups));
|
||||||
|
},
|
||||||
|
isNewVersionAvailable: function() {
|
||||||
|
if (!this.isNewVersionAvailable) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$.ajax({
|
||||||
|
url: OC.linkToOCS('apps/updatenotification/api/v1/applist', 2) + this.newVersionString,
|
||||||
|
type: 'GET',
|
||||||
|
beforeSend: function (request) {
|
||||||
|
request.setRequestHeader('Accept', 'application/json');
|
||||||
|
},
|
||||||
|
success: function(response) {
|
||||||
|
this.availableAppUpdates = response.ocs.data.available;
|
||||||
|
this.missingAppUpdates = response.ocs.data.missing;
|
||||||
|
this.isListFetched = true;
|
||||||
|
this.appStoreFailed = false;
|
||||||
|
}.bind(this),
|
||||||
|
error: function(xhr) {
|
||||||
|
this.availableAppUpdates = [];
|
||||||
|
this.missingAppUpdates = [];
|
||||||
|
this.appStoreDisabled = xhr.responseJSON.ocs.data.appstore_disabled;
|
||||||
|
this.isListFetched = true;
|
||||||
|
this.appStoreFailed = true;
|
||||||
|
}.bind(this)
|
||||||
|
});
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
computed: {
|
computed: {
|
||||||
newVersionAvailableString: function() {
|
newVersionAvailableString: function() {
|
||||||
return t('updatenotification', 'A new version is available: {newVersionString}', {
|
return t('updatenotification', 'A new version is available: <strong>{newVersionString}</strong>', {
|
||||||
newVersionString: this.newVersionString
|
newVersionString: this.newVersionString
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
|
|
||||||
lastCheckedOnString: function() {
|
lastCheckedOnString: function() {
|
||||||
return t('updatenotification', 'Checked on {lastCheckedDate}', {
|
return t('updatenotification', 'Checked on {lastCheckedDate}', {
|
||||||
lastCheckedDate: this.lastCheckedDate
|
lastCheckedDate: this.lastCheckedDate
|
||||||
});
|
});
|
||||||
|
},
|
||||||
|
|
||||||
|
statusText: function() {
|
||||||
|
if (!this.isListFetched) {
|
||||||
|
return t('updatenotification', 'Checking apps for compatible updates');
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this.appstoreDisabled) {
|
||||||
|
return t('updatenotification', 'Please make sure your config.php does not set <samp>appstoreenabled</samp> to false.');
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this.appstoreFailed) {
|
||||||
|
return t('updatenotification', 'Could not connect to the appstore or the appstore returned no updates at all. Search manually for updates or make sure your server has access to the internet and can connect to the appstore.');
|
||||||
|
}
|
||||||
|
|
||||||
|
return this.missingAppUpdates.length === 0 ? t('updatenotification', '<strong>All</strong> apps have an update for this version available', this) : n('updatenotification',
|
||||||
|
'<strong>%n</strong> app has no update for this version available',
|
||||||
|
'<strong>%n</strong> apps have no update for this version available',
|
||||||
|
this.missingAppUpdates.length, {
|
||||||
|
version: this.newVersionString
|
||||||
|
});
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
|
@ -144,6 +227,12 @@
|
||||||
OC.msg.finishedAction('#channel_save_msg', data);
|
OC.msg.finishedAction('#channel_save_msg', data);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
},
|
||||||
|
toggleHideMissingUpdates: function() {
|
||||||
|
this.hideMissingUpdates = !this.hideMissingUpdates;
|
||||||
|
},
|
||||||
|
toggleHideAvailableUpdates: function() {
|
||||||
|
this.hideAvailableUpdates = !this.hideAvailableUpdates;
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,134 @@
|
||||||
|
<?php
|
||||||
|
declare(strict_types=1);
|
||||||
|
/**
|
||||||
|
* @copyright Copyright (c) 2017 Joas Schilling <coding@schilljs.com>
|
||||||
|
*
|
||||||
|
* @license GNU AGPL version 3 or any later version
|
||||||
|
*
|
||||||
|
* This program is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU Affero General Public License as
|
||||||
|
* published by the Free Software Foundation, either version 3 of the
|
||||||
|
* License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU Affero General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Affero General Public License
|
||||||
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
namespace OCA\UpdateNotification\Controller;
|
||||||
|
|
||||||
|
use OC\App\AppStore\Fetcher\AppFetcher;
|
||||||
|
use OCP\App\AppPathNotFoundException;
|
||||||
|
use OCP\App\IAppManager;
|
||||||
|
use OCP\AppFramework\Http;
|
||||||
|
use OCP\AppFramework\Http\DataResponse;
|
||||||
|
use OCP\AppFramework\OCSController;
|
||||||
|
use OCP\IConfig;
|
||||||
|
use OCP\IRequest;
|
||||||
|
|
||||||
|
class APIController extends OCSController {
|
||||||
|
|
||||||
|
/** @var IConfig */
|
||||||
|
protected $config;
|
||||||
|
|
||||||
|
/** @var IAppManager */
|
||||||
|
protected $appManager;
|
||||||
|
|
||||||
|
/** @var AppFetcher */
|
||||||
|
protected $appFetcher;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param string $appName
|
||||||
|
* @param IRequest $request
|
||||||
|
* @param IConfig $config
|
||||||
|
* @param IAppManager $appManager
|
||||||
|
* @param AppFetcher $appFetcher
|
||||||
|
*/
|
||||||
|
public function __construct($appName,
|
||||||
|
IRequest $request,
|
||||||
|
IConfig $config,
|
||||||
|
IAppManager $appManager,
|
||||||
|
AppFetcher $appFetcher) {
|
||||||
|
parent::__construct($appName, $request);
|
||||||
|
|
||||||
|
$this->config = $config;
|
||||||
|
$this->appManager = $appManager;
|
||||||
|
$this->appFetcher = $appFetcher;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param string $newVersion
|
||||||
|
* @return DataResponse
|
||||||
|
*/
|
||||||
|
public function getAppList(string $newVersion): DataResponse {
|
||||||
|
if (!$this->config->getSystemValue('appstoreenabled', true)) {
|
||||||
|
return new DataResponse([
|
||||||
|
'appstore_disabled' => true,
|
||||||
|
], Http::STATUS_NOT_FOUND);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Get list of installed custom apps
|
||||||
|
$installedApps = $this->appManager->getInstalledApps();
|
||||||
|
$installedApps = array_filter($installedApps, function($app) {
|
||||||
|
try {
|
||||||
|
$this->appManager->getAppPath($app);
|
||||||
|
} catch (AppPathNotFoundException $e) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return !$this->appManager->isShipped($app);
|
||||||
|
});
|
||||||
|
|
||||||
|
if (empty($installedApps)) {
|
||||||
|
return new DataResponse([
|
||||||
|
'missing' => [],
|
||||||
|
'available' => [],
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->appFetcher->setVersion($newVersion, 'future-apps.json');
|
||||||
|
|
||||||
|
// Apps available on the app store for that version
|
||||||
|
$availableApps = array_map(function(array $app) {
|
||||||
|
return $app['id'];
|
||||||
|
}, $this->appFetcher->get());
|
||||||
|
|
||||||
|
if (empty($availableApps)) {
|
||||||
|
return new DataResponse([
|
||||||
|
'appstore_disabled' => false,
|
||||||
|
'already_on_latest' => false,
|
||||||
|
], Http::STATUS_NOT_FOUND);
|
||||||
|
}
|
||||||
|
|
||||||
|
$missing = array_diff($installedApps, $availableApps);
|
||||||
|
$missing = array_map([$this, 'getAppDetails'], $missing);
|
||||||
|
sort($missing);
|
||||||
|
|
||||||
|
$available = array_intersect($installedApps, $availableApps);
|
||||||
|
$available = array_map([$this, 'getAppDetails'], $available);
|
||||||
|
sort($available);
|
||||||
|
|
||||||
|
return new DataResponse([
|
||||||
|
'missing' => $missing,
|
||||||
|
'available' => $available,
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get translated app name
|
||||||
|
*
|
||||||
|
* @param string $appId
|
||||||
|
* @return string[]
|
||||||
|
*/
|
||||||
|
protected function getAppDetails($appId): array {
|
||||||
|
$app = $this->appManager->getAppInfo($appId);
|
||||||
|
return [
|
||||||
|
'appId' => $appId,
|
||||||
|
'appName' => $app['name'] ?? $appId,
|
||||||
|
];
|
||||||
|
}
|
||||||
|
}
|
|
@ -133,9 +133,11 @@ class AppFetcher extends Fetcher {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @param string $version
|
* @param string $version
|
||||||
|
* @param string $filename
|
||||||
*/
|
*/
|
||||||
public function setVersion($version) {
|
public function setVersion(string $version, string $fileName = 'apps.json') {
|
||||||
parent::setVersion($version);
|
parent::setVersion($version);
|
||||||
|
$this->fileName = $fileName;
|
||||||
$this->setEndpoint();
|
$this->setEndpoint();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -194,7 +194,7 @@ abstract class Fetcher {
|
||||||
* Set the current Nextcloud version
|
* Set the current Nextcloud version
|
||||||
* @param string $version
|
* @param string $version
|
||||||
*/
|
*/
|
||||||
public function setVersion($version) {
|
public function setVersion(string $version) {
|
||||||
$this->version = $version;
|
$this->version = $version;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue