Summary: Follow up of D32665064, this diff moves all plugin management logic from flipper-ui to flipper-server. Things like downloading, installing, querying new plugins. Loading plugins is handled separately in the next diff. Reviewed By: nikoant Differential Revision: D32666537 fbshipit-source-id: 9786b82987f00180bb26200e38735b334dc4d5c3
122 lines
3.5 KiB
TypeScript
122 lines
3.5 KiB
TypeScript
/**
|
|
* Copyright (c) Facebook, Inc. and its affiliates.
|
|
*
|
|
* This source code is licensed under the MIT license found in the
|
|
* LICENSE file in the root directory of this source tree.
|
|
*
|
|
* @format
|
|
*/
|
|
|
|
import {
|
|
UpdatablePluginDetails,
|
|
UpdateResult,
|
|
getPluginDetails,
|
|
} from 'flipper-common';
|
|
import {getInstalledPlugins} from './pluginInstaller';
|
|
import semver from 'semver';
|
|
import {getNpmHostedPlugins, NpmPackageDescriptor} from './getNpmHostedPlugins';
|
|
import NpmApi from 'npm-api';
|
|
import {getInstalledPluginDetails} from './getPluginDetails';
|
|
import {getPluginVersionInstallationDir} from './pluginPaths';
|
|
import pmap from 'p-map';
|
|
import {notNull} from './typeUtils';
|
|
const npmApi = new NpmApi();
|
|
|
|
export async function getUpdatablePlugins(
|
|
query?: string,
|
|
): Promise<UpdatablePluginDetails[]> {
|
|
const installedPlugins = await getInstalledPlugins();
|
|
const npmHostedPlugins = new Map<string, NpmPackageDescriptor>(
|
|
(await getNpmHostedPlugins({query})).map((p) => [p.name, p]),
|
|
);
|
|
const annotatedInstalledPlugins = await pmap(
|
|
installedPlugins,
|
|
async (installedPlugin): Promise<UpdatablePluginDetails> => {
|
|
try {
|
|
const npmPackageDescriptor = npmHostedPlugins.get(installedPlugin.name);
|
|
if (npmPackageDescriptor) {
|
|
npmHostedPlugins.delete(installedPlugin.name);
|
|
if (
|
|
semver.lt(installedPlugin.version, npmPackageDescriptor.version)
|
|
) {
|
|
const pkg = await npmApi.repo(npmPackageDescriptor.name).package();
|
|
const npmPluginDetails = await getInstalledPluginDetails(
|
|
getPluginVersionInstallationDir(
|
|
npmPackageDescriptor.name,
|
|
npmPackageDescriptor.version,
|
|
),
|
|
pkg,
|
|
);
|
|
return {
|
|
...npmPluginDetails,
|
|
updateStatus: {
|
|
kind: 'update-available',
|
|
version: npmPluginDetails.version,
|
|
},
|
|
};
|
|
}
|
|
}
|
|
const updateStatus: UpdateResult = {kind: 'up-to-date'};
|
|
return {
|
|
...installedPlugin,
|
|
updateStatus,
|
|
};
|
|
} catch (error) {
|
|
return {
|
|
...installedPlugin,
|
|
updateStatus: {
|
|
kind: 'error',
|
|
error,
|
|
},
|
|
};
|
|
}
|
|
},
|
|
{
|
|
concurrency: 4,
|
|
},
|
|
);
|
|
const annotatedNotInstalledPlugins = await pmap(
|
|
npmHostedPlugins.values(),
|
|
async (notInstalledPlugin) => {
|
|
try {
|
|
const pkg = await npmApi.repo(notInstalledPlugin.name).package();
|
|
const npmPluginDetails = getPluginDetails(pkg);
|
|
if (npmPluginDetails.specVersion === 1) {
|
|
return null;
|
|
}
|
|
return {
|
|
...npmPluginDetails,
|
|
updateStatus: {
|
|
kind: 'not-installed',
|
|
version: npmPluginDetails.version,
|
|
},
|
|
} as UpdatablePluginDetails;
|
|
} catch (error) {
|
|
console.log(
|
|
`Failed to load details from npm for plugin ${notInstalledPlugin.name}`,
|
|
error,
|
|
);
|
|
return null;
|
|
}
|
|
},
|
|
{
|
|
concurrency: 4,
|
|
},
|
|
);
|
|
return [
|
|
...annotatedInstalledPlugins.sort((p1, p2) =>
|
|
p1.name.localeCompare(p2.name),
|
|
),
|
|
...annotatedNotInstalledPlugins
|
|
.filter(notNull)
|
|
.sort((p1, p2) => p1.name.localeCompare(p2.name)),
|
|
].filter(
|
|
(p) =>
|
|
!query ||
|
|
p.name.includes(query) ||
|
|
p.id.includes(query) ||
|
|
p.description?.includes(query) ||
|
|
p.title?.includes(query),
|
|
);
|
|
}
|