Summary: *Stack summary*: this stack refactors plugin management actions to perform them in a dispatcher rather than in the root reducer (store.tsx) as all of these actions has side effects. To do that, we store requested plugin management actions (install/update/uninstall, star/unstar) in a queue which is then handled by pluginManager dispatcher. This dispatcher then dispatches all required state updates. *Diff summary*: refactored "uninstall plugin" operation to perform it in pluginManager dispatcher Reviewed By: mweststrate Differential Revision: D26166198 fbshipit-source-id: d74a1d690102d9036c6d3d8612d2428f5ecef4e6
160 lines
4.7 KiB
TypeScript
160 lines
4.7 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 type {Store} from '../reducers/index';
|
|
import type {Logger} from '../fb-interfaces/Logger';
|
|
import {clearPluginState} from '../reducers/pluginStates';
|
|
import {
|
|
LoadPluginActionPayload,
|
|
pluginCommandsProcessed,
|
|
UninstallPluginActionPayload,
|
|
} from '../reducers/pluginManager';
|
|
import {
|
|
getInstalledPlugins,
|
|
cleanupOldInstalledPluginVersions,
|
|
removePlugins,
|
|
ActivatablePluginDetails,
|
|
} from 'flipper-plugin-lib';
|
|
import {sideEffect} from '../utils/sideEffect';
|
|
import {requirePlugin} from './plugins';
|
|
import {registerPluginUpdate} from '../reducers/connections';
|
|
import {showErrorNotification} from '../utils/notifications';
|
|
import type Client from '../Client';
|
|
import {unloadModule} from '../utils/electronModuleCache';
|
|
import {pluginUninstalled, registerInstalledPlugins} from '../reducers/plugins';
|
|
import {defaultEnabledBackgroundPlugins} from '../utils/pluginUtils';
|
|
|
|
const maxInstalledPluginVersionsToKeep = 2;
|
|
|
|
function refreshInstalledPlugins(store: Store) {
|
|
removePlugins(store.getState().plugins.uninstalledPlugins.values())
|
|
.then(() =>
|
|
cleanupOldInstalledPluginVersions(maxInstalledPluginVersionsToKeep),
|
|
)
|
|
.then(() => getInstalledPlugins())
|
|
.then((plugins) => store.dispatch(registerInstalledPlugins(plugins)));
|
|
}
|
|
|
|
export default (
|
|
store: Store,
|
|
_logger: Logger,
|
|
{runSideEffectsSynchronously}: {runSideEffectsSynchronously: boolean} = {
|
|
runSideEffectsSynchronously: false,
|
|
},
|
|
) => {
|
|
// This needn't happen immediately and is (light) I/O work.
|
|
if (window.requestIdleCallback) {
|
|
window.requestIdleCallback(() => {
|
|
refreshInstalledPlugins(store);
|
|
});
|
|
}
|
|
|
|
const unsubscribeHandlePluginCommands = sideEffect(
|
|
store,
|
|
{
|
|
name: 'handlePluginCommands',
|
|
throttleMs: 0,
|
|
fireImmediately: true,
|
|
runSynchronously: runSideEffectsSynchronously, // Used to simplify writing tests, if "true" passed, the all side effects will be called synchronously and immediately after changes
|
|
noTimeBudgetWarns: true, // These side effects are critical, so we're doing them with zero throttling and want to avoid unnecessary warns
|
|
},
|
|
(state) => state.pluginManager.pluginCommandsQueue,
|
|
(queue, store) => {
|
|
for (const command of queue) {
|
|
switch (command.type) {
|
|
case 'LOAD_PLUGIN':
|
|
loadPlugin(store, command.payload);
|
|
break;
|
|
case 'UNINSTALL_PLUGIN':
|
|
uninstallPlugin(store, command.payload);
|
|
break;
|
|
default:
|
|
console.error('Unexpected plugin command', command);
|
|
break;
|
|
}
|
|
}
|
|
store.dispatch(pluginCommandsProcessed(queue.length));
|
|
},
|
|
);
|
|
return async () => {
|
|
unsubscribeHandlePluginCommands();
|
|
};
|
|
};
|
|
|
|
function loadPlugin(store: Store, payload: LoadPluginActionPayload) {
|
|
try {
|
|
const plugin = requirePlugin(payload.plugin);
|
|
const enablePlugin = payload.enable;
|
|
store.dispatch(
|
|
registerPluginUpdate({
|
|
plugin,
|
|
enablePlugin,
|
|
}),
|
|
);
|
|
} catch (err) {
|
|
console.error(
|
|
`Failed to activate plugin ${payload.plugin.title} v${payload.plugin.version}`,
|
|
err,
|
|
);
|
|
if (payload.notifyIfFailed) {
|
|
showErrorNotification(
|
|
`Failed to activate plugin "${payload.plugin.title}" v${payload.plugin.version}`,
|
|
);
|
|
}
|
|
}
|
|
}
|
|
|
|
function uninstallPlugin(store: Store, {plugin}: UninstallPluginActionPayload) {
|
|
try {
|
|
const state = store.getState();
|
|
const clients = state.connections.clients;
|
|
clients.forEach((client) => {
|
|
stopPlugin(client, plugin.id);
|
|
});
|
|
store.dispatch(clearPluginState({pluginId: plugin.id}));
|
|
if (!plugin.details.isBundled) {
|
|
unloadPluginModule(plugin.details);
|
|
}
|
|
store.dispatch(pluginUninstalled(plugin.details));
|
|
} catch (err) {
|
|
console.error(
|
|
`Failed to uninstall plugin ${plugin.title} v${plugin.version}`,
|
|
err,
|
|
);
|
|
showErrorNotification(
|
|
`Failed to uninstall plugin "${plugin.title}" v${plugin.version}`,
|
|
);
|
|
}
|
|
}
|
|
|
|
function stopPlugin(
|
|
client: Client,
|
|
pluginId: string,
|
|
forceInitBackgroundPlugin: boolean = false,
|
|
): boolean {
|
|
if (
|
|
(forceInitBackgroundPlugin ||
|
|
!defaultEnabledBackgroundPlugins.includes(pluginId)) &&
|
|
client?.isBackgroundPlugin(pluginId)
|
|
) {
|
|
client.deinitPlugin(pluginId);
|
|
}
|
|
// stop sandy plugins
|
|
client.stopPluginIfNeeded(pluginId);
|
|
return true;
|
|
}
|
|
|
|
function unloadPluginModule(plugin: ActivatablePluginDetails) {
|
|
if (plugin.isBundled) {
|
|
// We cannot unload bundled plugin.
|
|
return;
|
|
}
|
|
unloadModule(plugin.entry);
|
|
}
|