Summary: Extracted plugin marketplace API to a separate file and updated it to load full plugin manifests. Reviewed By: passy Differential Revision: D25181759 fbshipit-source-id: a63f9ce16249ccc170df148cef5c209fdc6d4d6d
133 lines
3.8 KiB
TypeScript
133 lines
3.8 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 PluginDetails from './PluginDetails';
|
|
import {getInstalledPlugins} from './getInstalledPlugins';
|
|
import semver from 'semver';
|
|
import {getNpmHostedPlugins, NpmPackageDescriptor} from './getNpmHostedPlugins';
|
|
import NpmApi from 'npm-api';
|
|
import {getPluginDetails} from './getPluginDetails';
|
|
import {getPluginInstallationDir} from './pluginPaths';
|
|
import pmap from 'p-map';
|
|
import {notNull} from './typeUtils';
|
|
const npmApi = new NpmApi();
|
|
|
|
export type UpdateResult =
|
|
| {kind: 'not-installed'; version: string}
|
|
| {kind: 'pending'}
|
|
| {kind: 'up-to-date'}
|
|
| {kind: 'error'; error: Error}
|
|
| {kind: 'update-available'; version: string};
|
|
|
|
export type UpdatablePlugin = {
|
|
updateStatus: UpdateResult;
|
|
};
|
|
|
|
export type UpdatablePluginDetails = PluginDetails & UpdatablePlugin;
|
|
|
|
export async function getUpdatablePlugins(
|
|
query?: string,
|
|
): Promise<UpdatablePluginDetails[]> {
|
|
const installedPlugins = await getInstalledPlugins();
|
|
const npmHostedPlugins = new Map<string, NpmPackageDescriptor>(
|
|
(await getNpmHostedPlugins()).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 getPluginDetails(
|
|
getPluginInstallationDir(npmPackageDescriptor.name),
|
|
pkg,
|
|
);
|
|
return {
|
|
...npmPluginDetails,
|
|
updateStatus: {
|
|
kind: 'update-available',
|
|
version: npmPluginDetails.version,
|
|
},
|
|
};
|
|
}
|
|
}
|
|
const updateStatus: UpdateResult =
|
|
installedPlugin.installationStatus === 'installed'
|
|
? {kind: 'up-to-date'}
|
|
: {kind: 'pending'};
|
|
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 = await getPluginDetails(
|
|
getPluginInstallationDir(notInstalledPlugin.name),
|
|
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}`,
|
|
);
|
|
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),
|
|
);
|
|
}
|