mirror of
http://10.0.2.1:3031/sauer/claude-code.git
synced 2026-06-30 16:26:58 +10:00
838 lines
116 KiB
TypeScript
838 lines
116 KiB
TypeScript
|
|
import { c as _c } from "react/compiler-runtime";
|
||
|
|
import figures from 'figures';
|
||
|
|
import * as React from 'react';
|
||
|
|
import { useEffect, useRef, useState } from 'react';
|
||
|
|
import { type AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, logEvent } from 'src/services/analytics/index.js';
|
||
|
|
import { ConfigurableShortcutHint } from '../../components/ConfigurableShortcutHint.js';
|
||
|
|
import { Byline } from '../../components/design-system/Byline.js';
|
||
|
|
import { KeyboardShortcutHint } from '../../components/design-system/KeyboardShortcutHint.js';
|
||
|
|
// eslint-disable-next-line custom-rules/prefer-use-keybindings -- useInput needed for marketplace-specific u/r shortcuts and y/n confirmation not in keybinding schema
|
||
|
|
import { Box, Text, useInput } from '../../ink.js';
|
||
|
|
import { useKeybinding, useKeybindings } from '../../keybindings/useKeybinding.js';
|
||
|
|
import type { LoadedPlugin } from '../../types/plugin.js';
|
||
|
|
import { count } from '../../utils/array.js';
|
||
|
|
import { shouldSkipPluginAutoupdate } from '../../utils/config.js';
|
||
|
|
import { errorMessage } from '../../utils/errors.js';
|
||
|
|
import { clearAllCaches } from '../../utils/plugins/cacheUtils.js';
|
||
|
|
import { createPluginId, formatMarketplaceLoadingErrors, getMarketplaceSourceDisplay, loadMarketplacesWithGracefulDegradation } from '../../utils/plugins/marketplaceHelpers.js';
|
||
|
|
import { loadKnownMarketplacesConfig, refreshMarketplace, removeMarketplaceSource, setMarketplaceAutoUpdate } from '../../utils/plugins/marketplaceManager.js';
|
||
|
|
import { updatePluginsForMarketplaces } from '../../utils/plugins/pluginAutoupdate.js';
|
||
|
|
import { loadAllPlugins } from '../../utils/plugins/pluginLoader.js';
|
||
|
|
import { isMarketplaceAutoUpdate } from '../../utils/plugins/schemas.js';
|
||
|
|
import { getSettingsForSource, updateSettingsForSource } from '../../utils/settings/settings.js';
|
||
|
|
import { plural } from '../../utils/stringUtils.js';
|
||
|
|
import type { ViewState } from './types.js';
|
||
|
|
type Props = {
|
||
|
|
setViewState: (state: ViewState) => void;
|
||
|
|
error?: string | null;
|
||
|
|
setError?: (error: string | null) => void;
|
||
|
|
setResult: (result: string | null) => void;
|
||
|
|
exitState: {
|
||
|
|
pending: boolean;
|
||
|
|
keyName: 'Ctrl-C' | 'Ctrl-D' | null;
|
||
|
|
};
|
||
|
|
onManageComplete?: () => void | Promise<void>;
|
||
|
|
targetMarketplace?: string;
|
||
|
|
action?: 'update' | 'remove';
|
||
|
|
};
|
||
|
|
type MarketplaceState = {
|
||
|
|
name: string;
|
||
|
|
source: string;
|
||
|
|
lastUpdated?: string;
|
||
|
|
pluginCount?: number;
|
||
|
|
installedPlugins?: LoadedPlugin[];
|
||
|
|
pendingUpdate?: boolean;
|
||
|
|
pendingRemove?: boolean;
|
||
|
|
autoUpdate?: boolean;
|
||
|
|
};
|
||
|
|
type InternalViewState = 'list' | 'details' | 'confirm-remove';
|
||
|
|
export function ManageMarketplaces({
|
||
|
|
setViewState,
|
||
|
|
error,
|
||
|
|
setError,
|
||
|
|
setResult,
|
||
|
|
exitState,
|
||
|
|
onManageComplete,
|
||
|
|
targetMarketplace,
|
||
|
|
action
|
||
|
|
}: Props): React.ReactNode {
|
||
|
|
const [marketplaceStates, setMarketplaceStates] = useState<MarketplaceState[]>([]);
|
||
|
|
const [loading, setLoading] = useState(true);
|
||
|
|
const [selectedIndex, setSelectedIndex] = useState(0);
|
||
|
|
const [isProcessing, setIsProcessing] = useState(false);
|
||
|
|
const [processError, setProcessError] = useState<string | null>(null);
|
||
|
|
const [successMessage, setSuccessMessage] = useState<string | null>(null);
|
||
|
|
const [progressMessage, setProgressMessage] = useState<string | null>(null);
|
||
|
|
const [internalView, setInternalView] = useState<InternalViewState>('list');
|
||
|
|
const [selectedMarketplace, setSelectedMarketplace] = useState<MarketplaceState | null>(null);
|
||
|
|
const [detailsMenuIndex, setDetailsMenuIndex] = useState(0);
|
||
|
|
const hasAttemptedAutoAction = useRef(false);
|
||
|
|
|
||
|
|
// Load marketplaces and their installed plugins
|
||
|
|
useEffect(() => {
|
||
|
|
async function loadMarketplaces() {
|
||
|
|
try {
|
||
|
|
const config = await loadKnownMarketplacesConfig();
|
||
|
|
const {
|
||
|
|
enabled,
|
||
|
|
disabled
|
||
|
|
} = await loadAllPlugins();
|
||
|
|
const allPlugins = [...enabled, ...disabled];
|
||
|
|
|
||
|
|
// Load marketplaces with graceful degradation
|
||
|
|
const {
|
||
|
|
marketplaces,
|
||
|
|
failures
|
||
|
|
} = await loadMarketplacesWithGracefulDegradation(config);
|
||
|
|
const states: MarketplaceState[] = [];
|
||
|
|
for (const {
|
||
|
|
name,
|
||
|
|
config: entry,
|
||
|
|
data: marketplace
|
||
|
|
} of marketplaces) {
|
||
|
|
// Get all plugins installed from this marketplace
|
||
|
|
const installedFromMarketplace = allPlugins.filter(plugin => plugin.source.endsWith(`@${name}`));
|
||
|
|
states.push({
|
||
|
|
name,
|
||
|
|
source: getMarketplaceSourceDisplay(entry.source),
|
||
|
|
lastUpdated: entry.lastUpdated,
|
||
|
|
pluginCount: marketplace?.plugins.length,
|
||
|
|
installedPlugins: installedFromMarketplace,
|
||
|
|
pendingUpdate: false,
|
||
|
|
pendingRemove: false,
|
||
|
|
autoUpdate: isMarketplaceAutoUpdate(name, entry)
|
||
|
|
});
|
||
|
|
}
|
||
|
|
|
||
|
|
// Sort: claude-plugin-directory first, then alphabetically
|
||
|
|
states.sort((a, b) => {
|
||
|
|
if (a.name === 'claude-plugin-directory') return -1;
|
||
|
|
if (b.name === 'claude-plugin-directory') return 1;
|
||
|
|
return a.name.localeCompare(b.name);
|
||
|
|
});
|
||
|
|
setMarketplaceStates(states);
|
||
|
|
|
||
|
|
// Handle marketplace loading errors/warnings
|
||
|
|
const successCount = count(marketplaces, m => m.data !== null);
|
||
|
|
const errorResult = formatMarketplaceLoadingErrors(failures, successCount);
|
||
|
|
if (errorResult) {
|
||
|
|
if (errorResult.type === 'warning') {
|
||
|
|
setProcessError(errorResult.message);
|
||
|
|
} else {
|
||
|
|
throw new Error(errorResult.message);
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
// Auto-execute if target and action provided
|
||
|
|
if (targetMarketplace && !hasAttemptedAutoAction.current && !error) {
|
||
|
|
hasAttemptedAutoAction.current = true;
|
||
|
|
const targetIndex = states.findIndex(s => s.name === targetMarketplace);
|
||
|
|
if (targetIndex >= 0) {
|
||
|
|
const targetState = states[targetIndex];
|
||
|
|
if (action) {
|
||
|
|
// Mark the action as pending and execute
|
||
|
|
setSelectedIndex(targetIndex + 1); // +1 because "Add Marketplace" is at index 0
|
||
|
|
const newStates = [...states];
|
||
|
|
if (action === 'update') {
|
||
|
|
newStates[targetIndex]!.pendingUpdate = true;
|
||
|
|
} else if (action === 'remove') {
|
||
|
|
newStates[targetIndex]!.pendingRemove = true;
|
||
|
|
}
|
||
|
|
setMarketplaceStates(newStates);
|
||
|
|
// Apply the change immediately
|
||
|
|
setTimeout(applyChanges, 100, newStates);
|
||
|
|
} else if (targetState) {
|
||
|
|
// No action - just show the details view for this marketplace
|
||
|
|
setSelectedIndex(targetIndex + 1); // +1 because "Add Marketplace" is at index 0
|
||
|
|
setSelectedMarketplace(targetState);
|
||
|
|
setInternalView('details');
|
||
|
|
}
|
||
|
|
} else if (setError) {
|
||
|
|
setError(`Marketplace not found: ${targetMarketplace}`);
|
||
|
|
}
|
||
|
|
}
|
||
|
|
} catch (err) {
|
||
|
|
if (setError) {
|
||
|
|
setError(err instanceof Error ? err.message : 'Failed to load marketplaces');
|
||
|
|
}
|
||
|
|
setProcessError(err instanceof Error ? err.message : 'Failed to load marketplaces');
|
||
|
|
} finally {
|
||
|
|
setLoading(false);
|
||
|
|
}
|
||
|
|
}
|
||
|
|
void loadMarketplaces();
|
||
|
|
// eslint-disable-next-line react-hooks/exhaustive-deps
|
||
|
|
// biome-ignore lint/correctness/useExhaustiveDependencies: intentional
|
||
|
|
}, [targetMarketplace, action, error]);
|
||
|
|
|
||
|
|
// Check if there are any pending changes
|
||
|
|
const hasPendingChanges = () => {
|
||
|
|
return marketplaceStates.some(state => state.pendingUpdate || state.pendingRemove);
|
||
|
|
};
|
||
|
|
|
||
|
|
// Get count of pending operations
|
||
|
|
const getPendingCounts = () => {
|
||
|
|
const updateCount = count(marketplaceStates, s => s.pendingUpdate);
|
||
|
|
const removeCount = count(marketplaceStates, s => s.pendingRemove);
|
||
|
|
return {
|
||
|
|
updateCount,
|
||
|
|
removeCount
|
||
|
|
};
|
||
|
|
};
|
||
|
|
|
||
|
|
// Apply all pending changes
|
||
|
|
const applyChanges = async (states?: MarketplaceState[]) => {
|
||
|
|
const statesToProcess = states || marketplaceStates;
|
||
|
|
const wasInDetailsView = internalView === 'details';
|
||
|
|
setIsProcessing(true);
|
||
|
|
setProcessError(null);
|
||
|
|
setSuccessMessage(null);
|
||
|
|
setProgressMessage(null);
|
||
|
|
try {
|
||
|
|
const settings = getSettingsForSource('userSettings');
|
||
|
|
let updatedCount = 0;
|
||
|
|
let removedCount = 0;
|
||
|
|
const refreshedMarketplaces = new Set<string>();
|
||
|
|
for (const state of statesToProcess) {
|
||
|
|
// Handle remove
|
||
|
|
if (state.pendingRemove) {
|
||
|
|
// First uninstall all plugins from this marketplace
|
||
|
|
if (state.installedPlugins && state.installedPlugins.length > 0) {
|
||
|
|
const newEnabledPlugins = {
|
||
|
|
...settings?.enabledPlugins
|
||
|
|
};
|
||
|
|
for (const plugin of state.installedPlugins) {
|
||
|
|
const pluginId = createPluginId(plugin.name, state.name);
|
||
|
|
// Mark as disabled/uninstalled
|
||
|
|
newEnabledPlugins[pluginId] = false;
|
||
|
|
}
|
||
|
|
updateSettingsForSource('userSettings', {
|
||
|
|
enabledPlugins: newEnabledPlugins
|
||
|
|
});
|
||
|
|
}
|
||
|
|
|
||
|
|
// Then remove the marketplace
|
||
|
|
await removeMarketplaceSource(state.name);
|
||
|
|
removedCount++;
|
||
|
|
logEvent('tengu_marketplace_removed', {
|
||
|
|
marketplace_name: state.name as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
|
||
|
|
plugins_uninstalled: state.installedPlugins?.length || 0
|
||
|
|
});
|
||
|
|
continue;
|
||
|
|
}
|
||
|
|
|
||
|
|
// Handle update
|
||
|
|
if (state.pendingUpdate) {
|
||
|
|
// Refresh individual marketplace for efficiency with progress reporting
|
||
|
|
await refreshMarketplace(state.name, (message: string) => {
|
||
|
|
setProgressMessage(message);
|
||
|
|
});
|
||
|
|
updatedCount++;
|
||
|
|
refreshedMarketplaces.add(state.name.toLowerCase());
|
||
|
|
logEvent('tengu_marketplace_updated', {
|
||
|
|
marketplace_name: state.name as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS
|
||
|
|
});
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
// After marketplace clones are refreshed, bump installed plugins from
|
||
|
|
// those marketplaces to the new version. Without this, the loader's
|
||
|
|
// cache-on-miss (copyPluginToVersionedCache) creates the new version
|
||
|
|
// dir on the next loadAllPlugins() call, but installed_plugins.json
|
||
|
|
// stays on the old version — so cleanupOrphanedPluginVersionsInBackground
|
||
|
|
// stamps the NEW dir with .orphaned_at on the next startup. See #29512.
|
||
|
|
// updatePluginOp (called inside the helper) is what actually writes
|
||
|
|
// installed_plugins.json via updateInstallationPathOnDisk.
|
||
|
|
let updatedPluginCount = 0;
|
||
|
|
if (refreshedMarketplaces.size > 0) {
|
||
|
|
const updatedPluginIds = await updatePluginsForMarketplaces(refreshedMarketplaces);
|
||
|
|
updatedPluginCount = updatedPluginIds.length;
|
||
|
|
}
|
||
|
|
|
||
|
|
// Clear caches after changes
|
||
|
|
clearAllCaches();
|
||
|
|
|
||
|
|
// Call completion callback
|
||
|
|
if (onManageComplete) {
|
||
|
|
await onManageComplete();
|
||
|
|
}
|
||
|
|
|
||
|
|
// Reload marketplace data to show updated timestamps
|
||
|
|
const config = await loadKnownMarketplacesConfig();
|
||
|
|
const {
|
||
|
|
enabled,
|
||
|
|
disabled
|
||
|
|
} = await loadAllPlugins();
|
||
|
|
const allPlugins = [...enabled, ...disabled];
|
||
|
|
const {
|
||
|
|
marketplaces
|
||
|
|
} = await loadMarketplacesWithGracefulDegradation(config);
|
||
|
|
const newStates: MarketplaceState[] = [];
|
||
|
|
for (const {
|
||
|
|
name,
|
||
|
|
config: entry,
|
||
|
|
data: marketplace
|
||
|
|
} of marketplaces) {
|
||
|
|
const installedFromMarketplace = allPlugins.filter(plugin => plugin.source.endsWith(`@${name}`));
|
||
|
|
newStates.push({
|
||
|
|
name,
|
||
|
|
source: getMarketplaceSourceDisplay(entry.source),
|
||
|
|
lastUpdated: entry.lastUpdated,
|
||
|
|
pluginCount: marketplace?.plugins.length,
|
||
|
|
installedPlugins: installedFromMarketplace,
|
||
|
|
pendingUpdate: false,
|
||
|
|
pendingRemove: false,
|
||
|
|
autoUpdate: isMarketplaceAutoUpdate(name, entry)
|
||
|
|
});
|
||
|
|
}
|
||
|
|
|
||
|
|
// Sort: claude-plugin-directory first, then alphabetically
|
||
|
|
newStates.sort((a, b) => {
|
||
|
|
if (a.name === 'claude-plugin-directory') return -1;
|
||
|
|
if (b.name === 'claude-plugin-directory') return 1;
|
||
|
|
return a.name.localeCompare(b.name);
|
||
|
|
});
|
||
|
|
setMarketplaceStates(newStates);
|
||
|
|
|
||
|
|
// Update selected marketplace reference with fresh data
|
||
|
|
if (wasInDetailsView && selectedMarketplace) {
|
||
|
|
const updatedMarketplace = newStates.find(s => s.name === selectedMarketplace.name);
|
||
|
|
if (updatedMarketplace) {
|
||
|
|
setSelectedMarketplace(updatedMarketplace);
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
// Build success message
|
||
|
|
const actions: string[] = [];
|
||
|
|
if (updatedCount > 0) {
|
||
|
|
const pluginPart = updatedPluginCount > 0 ? ` (${updatedPluginCount} ${plural(updatedPluginCount, 'plugin')} bumped)` : '';
|
||
|
|
actions.push(`Updated ${updatedCount} ${plural(updatedCount, 'marketplace')}${pluginPart}`);
|
||
|
|
}
|
||
|
|
if (removedCount > 0) {
|
||
|
|
actions.push(`Removed ${removedCount} ${plural(removedCount, 'marketplace')}`);
|
||
|
|
}
|
||
|
|
if (actions.length > 0) {
|
||
|
|
const successMsg = `${figures.tick} ${actions.join(', ')}`;
|
||
|
|
// If we were in details view, stay there and show success
|
||
|
|
if (wasInDetailsView) {
|
||
|
|
setSuccessMessage(successMsg);
|
||
|
|
} else {
|
||
|
|
// Otherwise show result and exit to menu
|
||
|
|
setResult(successMsg);
|
||
|
|
setTimeout(setViewState, 2000, {
|
||
|
|
type: 'menu' as const
|
||
|
|
});
|
||
|
|
}
|
||
|
|
} else if (!wasInDetailsView) {
|
||
|
|
setViewState({
|
||
|
|
type: 'menu'
|
||
|
|
});
|
||
|
|
}
|
||
|
|
} catch (err) {
|
||
|
|
const errorMsg = errorMessage(err);
|
||
|
|
setProcessError(errorMsg);
|
||
|
|
if (setError) {
|
||
|
|
setError(errorMsg);
|
||
|
|
}
|
||
|
|
} finally {
|
||
|
|
setIsProcessing(false);
|
||
|
|
setProgressMessage(null);
|
||
|
|
}
|
||
|
|
};
|
||
|
|
|
||
|
|
// Handle confirming marketplace removal
|
||
|
|
const confirmRemove = async () => {
|
||
|
|
if (!selectedMarketplace) return;
|
||
|
|
|
||
|
|
// Mark for removal and apply
|
||
|
|
const newStates = marketplaceStates.map(state => state.name === selectedMarketplace.name ? {
|
||
|
|
...state,
|
||
|
|
pendingRemove: true
|
||
|
|
} : state);
|
||
|
|
setMarketplaceStates(newStates);
|
||
|
|
await applyChanges(newStates);
|
||
|
|
};
|
||
|
|
|
||
|
|
// Build menu options for details view
|
||
|
|
const buildDetailsMenuOptions = (marketplace: MarketplaceState | null): Array<{
|
||
|
|
label: string;
|
||
|
|
secondaryLabel?: string;
|
||
|
|
value: string;
|
||
|
|
}> => {
|
||
|
|
if (!marketplace) return [];
|
||
|
|
const options: Array<{
|
||
|
|
label: string;
|
||
|
|
secondaryLabel?: string;
|
||
|
|
value: string;
|
||
|
|
}> = [{
|
||
|
|
label: `Browse plugins (${marketplace.pluginCount ?? 0})`,
|
||
|
|
value: 'browse'
|
||
|
|
}, {
|
||
|
|
label: 'Update marketplace',
|
||
|
|
secondaryLabel: marketplace.lastUpdated ? `(last updated ${new Date(marketplace.lastUpdated).toLocaleDateString()})` : undefined,
|
||
|
|
value: 'update'
|
||
|
|
}];
|
||
|
|
|
||
|
|
// Only show auto-update toggle if auto-updater is not globally disabled
|
||
|
|
if (!shouldSkipPluginAutoupdate()) {
|
||
|
|
options.push({
|
||
|
|
label: marketplace.autoUpdate ? 'Disable auto-update' : 'Enable auto-update',
|
||
|
|
value: 'toggle-auto-update'
|
||
|
|
});
|
||
|
|
}
|
||
|
|
options.push({
|
||
|
|
label: 'Remove marketplace',
|
||
|
|
value: 'remove'
|
||
|
|
});
|
||
|
|
return options;
|
||
|
|
};
|
||
|
|
|
||
|
|
// Handle toggling auto-update for a marketplace
|
||
|
|
const handleToggleAutoUpdate = async (marketplace: MarketplaceState) => {
|
||
|
|
const newAutoUpdate = !marketplace.autoUpdate;
|
||
|
|
try {
|
||
|
|
await setMarketplaceAutoUpdate(marketplace.name, newAutoUpdate);
|
||
|
|
|
||
|
|
// Update local state
|
||
|
|
setMarketplaceStates(prev => prev.map(state => state.name === marketplace.name ? {
|
||
|
|
...state,
|
||
|
|
autoUpdate: newAutoUpdate
|
||
|
|
} : state));
|
||
|
|
|
||
|
|
// Update selected marketplace reference
|
||
|
|
setSelectedMarketplace(prev => prev ? {
|
||
|
|
...prev,
|
||
|
|
autoUpdate: newAutoUpdate
|
||
|
|
} : prev);
|
||
|
|
} catch (err) {
|
||
|
|
setProcessError(err instanceof Error ? err.message : 'Failed to update setting');
|
||
|
|
}
|
||
|
|
};
|
||
|
|
|
||
|
|
// Escape in details or confirm-remove view - go back to list
|
||
|
|
useKeybinding('confirm:no', () => {
|
||
|
|
setInternalView('list');
|
||
|
|
setDetailsMenuIndex(0);
|
||
|
|
}, {
|
||
|
|
context: 'Confirmation',
|
||
|
|
isActive: !isProcessing && (internalView === 'details' || internalView === 'confirm-remove')
|
||
|
|
});
|
||
|
|
|
||
|
|
// Escape in list view with pending changes - clear pending changes
|
||
|
|
useKeybinding('confirm:no', () => {
|
||
|
|
setMarketplaceStates(prev => prev.map(state => ({
|
||
|
|
...state,
|
||
|
|
pendingUpdate: false,
|
||
|
|
pendingRemove: false
|
||
|
|
})));
|
||
|
|
setSelectedIndex(0);
|
||
|
|
}, {
|
||
|
|
context: 'Confirmation',
|
||
|
|
isActive: !isProcessing && internalView === 'list' && hasPendingChanges()
|
||
|
|
});
|
||
|
|
|
||
|
|
// Escape in list view without pending changes - exit to parent menu
|
||
|
|
useKeybinding('confirm:no', () => {
|
||
|
|
setViewState({
|
||
|
|
type: 'menu'
|
||
|
|
});
|
||
|
|
}, {
|
||
|
|
context: 'Confirmation',
|
||
|
|
isActive: !isProcessing && internalView === 'list' && !hasPendingChanges()
|
||
|
|
});
|
||
|
|
|
||
|
|
// List view — navigation (up/down/enter via configurable keybindings)
|
||
|
|
useKeybindings({
|
||
|
|
'select:previous': () => setSelectedIndex(prev => Math.max(0, prev - 1)),
|
||
|
|
'select:next': () => {
|
||
|
|
const totalItems = marketplaceStates.length + 1;
|
||
|
|
setSelectedIndex(prev => Math.min(totalItems - 1, prev + 1));
|
||
|
|
},
|
||
|
|
'select:accept': () => {
|
||
|
|
const marketplaceIndex = selectedIndex - 1;
|
||
|
|
if (selectedIndex === 0) {
|
||
|
|
setViewState({
|
||
|
|
type: 'add-marketplace'
|
||
|
|
});
|
||
|
|
} else if (hasPendingChanges()) {
|
||
|
|
void applyChanges();
|
||
|
|
} else {
|
||
|
|
const marketplace = marketplaceStates[marketplaceIndex];
|
||
|
|
if (marketplace) {
|
||
|
|
setSelectedMarketplace(marketplace);
|
||
|
|
setInternalView('details');
|
||
|
|
setDetailsMenuIndex(0);
|
||
|
|
}
|
||
|
|
}
|
||
|
|
}
|
||
|
|
}, {
|
||
|
|
context: 'Select',
|
||
|
|
isActive: !isProcessing && internalView === 'list'
|
||
|
|
});
|
||
|
|
|
||
|
|
// List view — marketplace-specific actions (u/r shortcuts)
|
||
|
|
useInput(input => {
|
||
|
|
const marketplaceIndex = selectedIndex - 1;
|
||
|
|
if ((input === 'u' || input === 'U') && marketplaceIndex >= 0) {
|
||
|
|
setMarketplaceStates(prev => prev.map((state, idx) => idx === marketplaceIndex ? {
|
||
|
|
...state,
|
||
|
|
pendingUpdate: !state.pendingUpdate,
|
||
|
|
pendingRemove: state.pendingUpdate ? state.pendingRemove : false
|
||
|
|
} : state));
|
||
|
|
} else if ((input === 'r' || input === 'R') && marketplaceIndex >= 0) {
|
||
|
|
const marketplace = marketplaceStates[marketplaceIndex];
|
||
|
|
if (marketplace) {
|
||
|
|
setSelectedMarketplace(marketplace);
|
||
|
|
setInternalView('confirm-remove');
|
||
|
|
}
|
||
|
|
}
|
||
|
|
}, {
|
||
|
|
isActive: !isProcessing && internalView === 'list'
|
||
|
|
});
|
||
|
|
|
||
|
|
// Details view — navigation
|
||
|
|
useKeybindings({
|
||
|
|
'select:previous': () => setDetailsMenuIndex(prev => Math.max(0, prev - 1)),
|
||
|
|
'select:next': () => {
|
||
|
|
const menuOptions = buildDetailsMenuOptions(selectedMarketplace);
|
||
|
|
setDetailsMenuIndex(prev => Math.min(menuOptions.length - 1, prev + 1));
|
||
|
|
},
|
||
|
|
'select:accept': () => {
|
||
|
|
if (!selectedMarketplace) return;
|
||
|
|
const menuOptions = buildDetailsMenuOptions(selectedMarketplace);
|
||
|
|
const selectedOption = menuOptions[detailsMenuIndex];
|
||
|
|
if (selectedOption?.value === 'browse') {
|
||
|
|
setViewState({
|
||
|
|
type: 'browse-marketplace',
|
||
|
|
targetMarketplace: selectedMarketplace.name
|
||
|
|
});
|
||
|
|
} else if (selectedOption?.value === 'update') {
|
||
|
|
const newStates = marketplaceStates.map(state => state.name === selectedMarketplace.name ? {
|
||
|
|
...state,
|
||
|
|
pendingUpdate: true
|
||
|
|
} : state);
|
||
|
|
setMarketplaceStates(newStates);
|
||
|
|
void applyChanges(newStates);
|
||
|
|
} else if (selectedOption?.value === 'toggle-auto-update') {
|
||
|
|
void handleToggleAutoUpdate(selectedMarketplace);
|
||
|
|
} else if (selectedOption?.value === 'remove') {
|
||
|
|
setInternalView('confirm-remove');
|
||
|
|
}
|
||
|
|
}
|
||
|
|
}, {
|
||
|
|
context: 'Select',
|
||
|
|
isActive: !isProcessing && internalView === 'details'
|
||
|
|
});
|
||
|
|
|
||
|
|
// Confirm-remove view — y/n input
|
||
|
|
useInput(input => {
|
||
|
|
if (input === 'y' || input === 'Y') {
|
||
|
|
void confirmRemove();
|
||
|
|
} else if (input === 'n' || input === 'N') {
|
||
|
|
setInternalView('list');
|
||
|
|
setSelectedMarketplace(null);
|
||
|
|
}
|
||
|
|
}, {
|
||
|
|
isActive: !isProcessing && internalView === 'confirm-remove'
|
||
|
|
});
|
||
|
|
if (loading) {
|
||
|
|
return <Text>Loading marketplaces…</Text>;
|
||
|
|
}
|
||
|
|
if (marketplaceStates.length === 0) {
|
||
|
|
return <Box flexDirection="column">
|
||
|
|
<Box marginBottom={1}>
|
||
|
|
<Text bold>Manage marketplaces</Text>
|
||
|
|
</Box>
|
||
|
|
|
||
|
|
{/* Add Marketplace option */}
|
||
|
|
<Box flexDirection="row" gap={1}>
|
||
|
|
<Text color="suggestion">{figures.pointer} +</Text>
|
||
|
|
<Text bold color="suggestion">
|
||
|
|
Add Marketplace
|
||
|
|
</Text>
|
||
|
|
</Box>
|
||
|
|
|
||
|
|
<Box marginLeft={3}>
|
||
|
|
<Text dimColor italic>
|
||
|
|
{exitState.pending ? <>Press {exitState.keyName} again to go back</> : <Byline>
|
||
|
|
<ConfigurableShortcutHint action="select:accept" context="Select" fallback="Enter" description="select" />
|
||
|
|
<ConfigurableShortcutHint action="confirm:no" context="Confirmation" fallback="Esc" description="go back" />
|
||
|
|
</Byline>}
|
||
|
|
</Text>
|
||
|
|
</Box>
|
||
|
|
</Box>;
|
||
|
|
}
|
||
|
|
|
||
|
|
// Show confirmation dialog
|
||
|
|
if (internalView === 'confirm-remove' && selectedMarketplace) {
|
||
|
|
const pluginCount = selectedMarketplace.installedPlugins?.length || 0;
|
||
|
|
return <Box flexDirection="column">
|
||
|
|
<Text bold color="warning">
|
||
|
|
Remove marketplace <Text italic>{selectedMarketplace.name}</Text>?
|
||
|
|
</Text>
|
||
|
|
<Box flexDirection="column">
|
||
|
|
{pluginCount > 0 && <Box marginTop={1}>
|
||
|
|
<Text color="warning">
|
||
|
|
This will also uninstall {pluginCount}{' '}
|
||
|
|
{plural(pluginCount, 'plugin')} from this marketplace:
|
||
|
|
</Text>
|
||
|
|
</Box>}
|
||
|
|
{selectedMarketplace.installedPlugins && selectedMarketplace.installedPlugins.length > 0 && <Box flexDirection="column" marginTop={1} marginLeft={2}>
|
||
|
|
{selectedMarketplace.installedPlugins.map(plugin => <Text key={plugin.name} dimColor>
|
||
|
|
• {plugin.name}
|
||
|
|
</Text>)}
|
||
|
|
</Box>}
|
||
|
|
<Box marginTop={1}>
|
||
|
|
<Text>
|
||
|
|
Press <Text bold>y</Text> to confirm or <Text bold>n</Text> to
|
||
|
|
cancel
|
||
|
|
</Text>
|
||
|
|
</Box>
|
||
|
|
</Box>
|
||
|
|
</Box>;
|
||
|
|
}
|
||
|
|
|
||
|
|
// Show marketplace details
|
||
|
|
if (internalView === 'details' && selectedMarketplace) {
|
||
|
|
// Check if this marketplace is currently being processed
|
||
|
|
// Check pendingUpdate first so we show updating state immediately when user presses Enter
|
||
|
|
const isUpdating = selectedMarketplace.pendingUpdate || isProcessing;
|
||
|
|
const menuOptions = buildDetailsMenuOptions(selectedMarketplace);
|
||
|
|
return <Box flexDirection="column">
|
||
|
|
<Text bold>{selectedMarketplace.name}</Text>
|
||
|
|
<Text dimColor>{selectedMarketplace.source}</Text>
|
||
|
|
<Box marginTop={1}>
|
||
|
|
<Text>
|
||
|
|
{selectedMarketplace.pluginCount || 0} available{' '}
|
||
|
|
{plural(selectedMarketplace.pluginCount || 0, 'plugin')}
|
||
|
|
</Text>
|
||
|
|
</Box>
|
||
|
|
|
||
|
|
{/* Installed plugins section */}
|
||
|
|
{selectedMarketplace.installedPlugins && selectedMarketplace.installedPlugins.length > 0 && <Box flexDirection="column" marginTop={1}>
|
||
|
|
<Text bold>
|
||
|
|
Installed plugins ({selectedMarketplace.installedPlugins.length}
|
||
|
|
):
|
||
|
|
</Text>
|
||
|
|
<Box flexDirection="column" marginLeft={1}>
|
||
|
|
{selectedMarketplace.installedPlugins.map(plugin => <Box key={plugin.name} flexDirection="row" gap={1}>
|
||
|
|
<Text>{figures.bullet}</Text>
|
||
|
|
<Box flexDirection="column">
|
||
|
|
<Text>{plugin.name}</Text>
|
||
|
|
<Text dimColor>{plugin.manifest.description}</Text>
|
||
|
|
</Box>
|
||
|
|
</Box>)}
|
||
|
|
</Box>
|
||
|
|
</Box>}
|
||
|
|
|
||
|
|
{/* Processing indicator */}
|
||
|
|
{isUpdating && <Box marginTop={1} flexDirection="column">
|
||
|
|
<Text color="claude">Updating marketplace…</Text>
|
||
|
|
{progressMessage && <Text dimColor>{progressMessage}</Text>}
|
||
|
|
</Box>}
|
||
|
|
|
||
|
|
{/* Success message */}
|
||
|
|
{!isUpdating && successMessage && <Box marginTop={1}>
|
||
|
|
<Text color="claude">{successMessage}</Text>
|
||
|
|
</Box>}
|
||
|
|
|
||
|
|
{/* Error message */}
|
||
|
|
{!isUpdating && processError && <Box marginTop={1}>
|
||
|
|
<Text color="error">{processError}</Text>
|
||
|
|
</Box>}
|
||
|
|
|
||
|
|
{/* Menu options */}
|
||
|
|
{!isUpdating && <Box flexDirection="column" marginTop={1}>
|
||
|
|
{menuOptions.map((option, idx) => {
|
||
|
|
if (!option) return null;
|
||
|
|
const isSelected = idx === detailsMenuIndex;
|
||
|
|
return <Box key={option.value}>
|
||
|
|
<Text color={isSelected ? 'suggestion' : undefined}>
|
||
|
|
{isSelected ? figures.pointer : ' '} {option.label}
|
||
|
|
</Text>
|
||
|
|
{option.secondaryLabel && <Text dimColor> {option.secondaryLabel}</Text>}
|
||
|
|
</Box>;
|
||
|
|
})}
|
||
|
|
</Box>}
|
||
|
|
|
||
|
|
{/* Show explanatory text at the bottom when auto-update is enabled */}
|
||
|
|
{!isUpdating && !shouldSkipPluginAutoupdate() && selectedMarketplace.autoUpdate && <Box marginTop={1}>
|
||
|
|
<Text dimColor>
|
||
|
|
Auto-update enabled. Claude Code will automatically update this
|
||
|
|
marketplace and its installed plugins.
|
||
|
|
</Text>
|
||
|
|
</Box>}
|
||
|
|
|
||
|
|
<Box marginLeft={3}>
|
||
|
|
<Text dimColor italic>
|
||
|
|
{isUpdating ? <>Please wait…</> : <Byline>
|
||
|
|
<ConfigurableShortcutHint action="select:accept" context="Select" fallback="Enter" description="select" />
|
||
|
|
<ConfigurableShortcutHint action="confirm:no" context="Confirmation" fallback="Esc" description="go back" />
|
||
|
|
</Byline>}
|
||
|
|
</Text>
|
||
|
|
</Box>
|
||
|
|
</Box>;
|
||
|
|
}
|
||
|
|
|
||
|
|
// Show marketplace list
|
||
|
|
const {
|
||
|
|
updateCount,
|
||
|
|
removeCount
|
||
|
|
} = getPendingCounts();
|
||
|
|
return <Box flexDirection="column">
|
||
|
|
<Box marginBottom={1}>
|
||
|
|
<Text bold>Manage marketplaces</Text>
|
||
|
|
</Box>
|
||
|
|
|
||
|
|
{/* Add Marketplace option */}
|
||
|
|
<Box flexDirection="row" gap={1} marginBottom={1}>
|
||
|
|
<Text color={selectedIndex === 0 ? 'suggestion' : undefined}>
|
||
|
|
{selectedIndex === 0 ? figures.pointer : ' '} +
|
||
|
|
</Text>
|
||
|
|
<Text bold color={selectedIndex === 0 ? 'suggestion' : undefined}>
|
||
|
|
Add Marketplace
|
||
|
|
</Text>
|
||
|
|
</Box>
|
||
|
|
|
||
|
|
{/* Marketplace list */}
|
||
|
|
<Box flexDirection="column">
|
||
|
|
{marketplaceStates.map((state, idx) => {
|
||
|
|
const isSelected = idx + 1 === selectedIndex; // +1 because Add Marketplace is at index 0
|
||
|
|
|
||
|
|
// Build status indicators
|
||
|
|
const indicators: string[] = [];
|
||
|
|
if (state.pendingUpdate) indicators.push('UPDATE');
|
||
|
|
if (state.pendingRemove) indicators.push('REMOVE');
|
||
|
|
return <Box key={state.name} flexDirection="row" gap={1} marginBottom={1}>
|
||
|
|
<Text color={isSelected ? 'suggestion' : undefined}>
|
||
|
|
{isSelected ? figures.pointer : ' '}{' '}
|
||
|
|
{state.pendingRemove ? figures.cross : figures.bullet}
|
||
|
|
</Text>
|
||
|
|
<Box flexDirection="column" flexGrow={1}>
|
||
|
|
<Box flexDirection="row" gap={1}>
|
||
|
|
<Text bold strikethrough={state.pendingRemove} dimColor={state.pendingRemove}>
|
||
|
|
{state.name === 'claude-plugins-official' && <Text color="claude">✻ </Text>}
|
||
|
|
{state.name}
|
||
|
|
{state.name === 'claude-plugins-official' && <Text color="claude"> ✻</Text>}
|
||
|
|
</Text>
|
||
|
|
{indicators.length > 0 && <Text color="warning">[{indicators.join(', ')}]</Text>}
|
||
|
|
</Box>
|
||
|
|
<Text dimColor>{state.source}</Text>
|
||
|
|
<Text dimColor>
|
||
|
|
{state.pluginCount !== undefined && <>{state.pluginCount} available</>}
|
||
|
|
{state.installedPlugins && state.installedPlugins.length > 0 && <> • {state.installedPlugins.length} installed</>}
|
||
|
|
{state.lastUpdated && <>
|
||
|
|
{' '}
|
||
|
|
• Updated{' '}
|
||
|
|
{new Date(state.lastUpdated).toLocaleDateString()}
|
||
|
|
</>}
|
||
|
|
</Text>
|
||
|
|
</Box>
|
||
|
|
</Box>;
|
||
|
|
})}
|
||
|
|
</Box>
|
||
|
|
|
||
|
|
{/* Pending changes summary */}
|
||
|
|
{hasPendingChanges() && <Box marginTop={1} flexDirection="column">
|
||
|
|
<Text>
|
||
|
|
<Text bold>Pending changes:</Text>{' '}
|
||
|
|
<Text dimColor>Enter to apply</Text>
|
||
|
|
</Text>
|
||
|
|
{updateCount > 0 && <Text>
|
||
|
|
• Update {updateCount} {plural(updateCount, 'marketplace')}
|
||
|
|
</Text>}
|
||
|
|
{removeCount > 0 && <Text color="warning">
|
||
|
|
• Remove {removeCount} {plural(removeCount, 'marketplace')}
|
||
|
|
</Text>}
|
||
|
|
</Box>}
|
||
|
|
|
||
|
|
{/* Processing indicator */}
|
||
|
|
{isProcessing && <Box marginTop={1}>
|
||
|
|
<Text color="claude">Processing changes…</Text>
|
||
|
|
</Box>}
|
||
|
|
|
||
|
|
{/* Error display */}
|
||
|
|
{processError && <Box marginTop={1}>
|
||
|
|
<Text color="error">{processError}</Text>
|
||
|
|
</Box>}
|
||
|
|
|
||
|
|
<ManageMarketplacesKeyHints exitState={exitState} hasPendingActions={hasPendingChanges()} />
|
||
|
|
</Box>;
|
||
|
|
}
|
||
|
|
type ManageMarketplacesKeyHintsProps = {
|
||
|
|
exitState: Props['exitState'];
|
||
|
|
hasPendingActions: boolean;
|
||
|
|
};
|
||
|
|
function ManageMarketplacesKeyHints(t0) {
|
||
|
|
const $ = _c(18);
|
||
|
|
const {
|
||
|
|
exitState,
|
||
|
|
hasPendingActions
|
||
|
|
} = t0;
|
||
|
|
if (exitState.pending) {
|
||
|
|
let t1;
|
||
|
|
if ($[0] !== exitState.keyName) {
|
||
|
|
t1 = <Box marginTop={1}><Text dimColor={true} italic={true}>Press {exitState.keyName} again to go back</Text></Box>;
|
||
|
|
$[0] = exitState.keyName;
|
||
|
|
$[1] = t1;
|
||
|
|
} else {
|
||
|
|
t1 = $[1];
|
||
|
|
}
|
||
|
|
return t1;
|
||
|
|
}
|
||
|
|
let t1;
|
||
|
|
if ($[2] !== hasPendingActions) {
|
||
|
|
t1 = hasPendingActions && <ConfigurableShortcutHint action="select:accept" context="Select" fallback="Enter" description="apply changes" />;
|
||
|
|
$[2] = hasPendingActions;
|
||
|
|
$[3] = t1;
|
||
|
|
} else {
|
||
|
|
t1 = $[3];
|
||
|
|
}
|
||
|
|
let t2;
|
||
|
|
if ($[4] !== hasPendingActions) {
|
||
|
|
t2 = !hasPendingActions && <ConfigurableShortcutHint action="select:accept" context="Select" fallback="Enter" description="select" />;
|
||
|
|
$[4] = hasPendingActions;
|
||
|
|
$[5] = t2;
|
||
|
|
} else {
|
||
|
|
t2 = $[5];
|
||
|
|
}
|
||
|
|
let t3;
|
||
|
|
if ($[6] !== hasPendingActions) {
|
||
|
|
t3 = !hasPendingActions && <KeyboardShortcutHint shortcut="u" action="update" />;
|
||
|
|
$[6] = hasPendingActions;
|
||
|
|
$[7] = t3;
|
||
|
|
} else {
|
||
|
|
t3 = $[7];
|
||
|
|
}
|
||
|
|
let t4;
|
||
|
|
if ($[8] !== hasPendingActions) {
|
||
|
|
t4 = !hasPendingActions && <KeyboardShortcutHint shortcut="r" action="remove" />;
|
||
|
|
$[8] = hasPendingActions;
|
||
|
|
$[9] = t4;
|
||
|
|
} else {
|
||
|
|
t4 = $[9];
|
||
|
|
}
|
||
|
|
const t5 = hasPendingActions ? "cancel" : "go back";
|
||
|
|
let t6;
|
||
|
|
if ($[10] !== t5) {
|
||
|
|
t6 = <ConfigurableShortcutHint action="confirm:no" context="Confirmation" fallback="Esc" description={t5} />;
|
||
|
|
$[10] = t5;
|
||
|
|
$[11] = t6;
|
||
|
|
} else {
|
||
|
|
t6 = $[11];
|
||
|
|
}
|
||
|
|
let t7;
|
||
|
|
if ($[12] !== t1 || $[13] !== t2 || $[14] !== t3 || $[15] !== t4 || $[16] !== t6) {
|
||
|
|
t7 = <Box marginTop={1}><Text dimColor={true} italic={true}><Byline>{t1}{t2}{t3}{t4}{t6}</Byline></Text></Box>;
|
||
|
|
$[12] = t1;
|
||
|
|
$[13] = t2;
|
||
|
|
$[14] = t3;
|
||
|
|
$[15] = t4;
|
||
|
|
$[16] = t6;
|
||
|
|
$[17] = t7;
|
||
|
|
} else {
|
||
|
|
t7 = $[17];
|
||
|
|
}
|
||
|
|
return t7;
|
||
|
|
}
|
||
|
|
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJmaWd1cmVzIiwiUmVhY3QiLCJ1c2VFZmZlY3QiLCJ1c2VSZWYiLCJ1c2VTdGF0ZSIsIkFuYWx5dGljc01ldGFkYXRhX0lfVkVSSUZJRURfVEhJU19JU19OT1RfQ09ERV9PUl9GSUxFUEFUSFMiLCJsb2dFdmVudCIsIkNvbmZpZ3VyYWJsZVNob3J0Y3V0SGludCIsIkJ5bGluZSIsIktleWJvYXJkU2hvcnRjdXRIaW50IiwiQm94IiwiVGV4dCIsInVzZUlucHV0IiwidXNlS2V5YmluZGluZyIsInVzZUtleWJpbmRpbmdzIiwiTG9hZGVkUGx1Z2luIiwiY291bnQiLCJzaG91bGRTa2lwUGx1Z2luQXV0b3VwZGF0ZSIsImVycm9yTWVzc2FnZSIsImNsZWFyQWxsQ2FjaGVzIiwiY3JlYXRlUGx1Z2luSWQiLCJmb3JtYXRNYXJrZXRwbGFjZUxvYWRpbmdFcnJvcnMiLCJnZXRNYXJrZXRwbGFjZVNvdXJjZURpc3BsYXkiLCJsb2FkTWFya2V0cGxhY2VzV2l0aEdyYWNlZnVsRGVncmFkYXRpb24iLCJsb2FkS25vd25NYXJrZXRwbGFjZXNDb25maWciLCJyZWZyZXNoTWFya2V0cGxhY2UiLCJyZW1vdmVNYXJrZXRwbGFjZVNvdXJjZSIsInNldE1hcmtldHBsYWNlQXV0b1VwZGF0ZSIsInVwZGF0ZVBsdWdpbnNGb3JNYXJrZXRwbGFjZXMiLCJsb2FkQWxsUGx1Z2lucyIsImlzTWFya2V0cGxhY2VBdXRvVXBkYXRlIiwiZ2V0U2V0dGluZ3NGb3JTb3VyY2UiLCJ1cGRhdGVTZXR0aW5nc0ZvclNvdXJjZSIsInBsdXJhbCIsIlZpZXdTdGF0ZSIsIlByb3BzIiwic2V0Vmlld1N0YXRlIiwic3RhdGUiLCJlcnJvciIsInNldEVycm9yIiwic2V0UmVzdWx0IiwicmVzdWx0IiwiZXhpdFN0YXRlIiwicGVuZGluZyIsImtleU5hbWUiLCJvbk1hbmFnZUNvbXBsZXRlIiwiUHJvbWlzZSIsInRhcmdldE1hcmtldHBsYWNlIiwiYWN0aW9uIiwiTWFya2V0cGxhY2VTdGF0ZSIsIm5hbWUiLCJzb3VyY2UiLCJsYXN0VXBkYXRlZCIsInBsdWdpbkNvdW50IiwiaW5zdGFsbGVkUGx1Z2lucyIsInBlbmRpbmdVcGRhdGUiLCJwZW5kaW5nUmVtb3ZlIiwiYXV0b1VwZGF0ZSIsIkludGVybmFsVmlld1N0YXRlIiwiTWFuYWdlTWFya2V0cGxhY2VzIiwiUmVhY3ROb2RlIiwibWFya2V0cGxhY2VTdGF0ZXMiLCJzZXRNYXJrZXRwbGFjZVN0YXRlcyIsImxvYWRpbmciLCJzZXRMb2FkaW5nIiwic2VsZWN0ZWRJbmRleCIsInNldFNlbGVjdGVkSW5kZXgiLCJpc1Byb2Nlc3NpbmciLCJzZXRJc1Byb2Nlc3NpbmciLCJwcm9jZXNzRXJyb3IiLCJzZXRQcm9jZXNzRXJyb3IiLCJzdWNjZXNzTWVzc2FnZSIsInNldFN1Y2Nlc3NNZXNzYWdlIiwicHJvZ3Jlc3NNZXNzYWdlIiwic2V0UHJvZ3Jlc3NNZXNzYWdlIiwiaW50ZXJuYWxWaWV3Iiwic2V0SW50ZXJuYWxWaWV3Iiwic2VsZWN0ZWRNYXJrZXRwbGFjZSIsInNldFNlbGVjdGVkTWFya2V0cGxhY2UiLCJkZXRhaWxzTWVudUluZGV4Iiwic2V0RGV0YWlsc01lbnVJbmRleCIsImhhc0F0dGVtcHRlZEF1dG9BY3Rpb24iLCJsb2FkTWFya2V0cGxhY2VzIiwiY29uZmlnIiwiZW5hYmxlZCIsImRpc2FibGVkIiwiYWxsUGx1Z2lucyIsIm1hcmtldHBsYWNlcyIsImZhaWx1cmVzIiwic3RhdGVzIiwiZW50cnkiLCJkYXRhIiwibWFya2V0cGxhY2UiLCJpbnN0YWxsZWRGcm9tTWFya2V0cGxhY2UiLCJmaWx0ZXIiLCJwbHVnaW4iLCJlbmRzV2l0aCIsInB1c2giLCJwbHVnaW5zIiwibGVuZ3RoIiwic29ydCIsImEiLCJiIiwibG9jYWxlQ29tcGFyZSIsInN1Y2Nlc3NDb3VudCIsIm0iLCJlcnJvclJlc3VsdCIsInR5cGUiLCJtZXNzYWdlIiwiRXJyb3IiLCJjdXJyZW50IiwidGFyZ2V0SW5kZXgiLCJmaW5kSW5kZXgiLCJzIiwidGFyZ2V0U3RhdGUiLCJuZXdTdGF0ZXMiLCJzZXRUaW1lb3V0IiwiYXBwbHlDaGFuZ2VzIiwiZXJyIiwiaGFzUGVuZGluZ0NoYW5nZXMiLCJzb21lIiwiZ2V0UGVuZGluZ0NvdW50cyIsInVwZGF0ZUNvdW50IiwicmVtb3ZlQ291bnQiLCJzdGF0ZXNUb1Byb2Nlc3MiLCJ3YXNJbkRldGFpbHNWaWV3Iiwic2V0dGluZ3MiLCJ1cGRhdGVkQ291bnQiLCJyZW1vdmVkQ291bnQiLCJyZWZyZXNoZWRNYXJrZXRwbGFjZXMiLCJTZXQiLCJuZXdFbmFibGVkUGx1Z2lucyIsImVuYWJsZWRQbHVnaW5zIiwicGx1Z2luSWQiLCJtYXJrZXRwbGFjZV9uYW1lIiwicGx1Z2luc191bmluc3RhbGxlZCIsImFkZCIsInRvTG93ZXJDYXNlIiwidXBkYXRlZFBsdWdpbkNvdW50Iiwic2l6ZSIsInVwZGF0ZWRQbHVnaW5JZHMiLCJ1cGRhdGVkTWFya2V0cGxhY2UiLCJmaW5kIiwiYWN0aW9ucyIsInBsdWdpblBhcnQiLCJzdWNjZXNzTXNnIiwidGljayIsImpvaW4iLCJjb25zdCIsImVycm9yTXNnIiwiY29uZmlybVJlbW92ZSIsIm1hcCIsImJ1aWxkRGV0YWlsc01lbnVPcHRpb25zIiwiQXJyYXkiLCJsYWJlbCIsInNlY29uZGFyeUxhYmVsIiwidmFsdWUiLCJvcHRpb25zIiwiRGF0ZSIsInRvTG9jYWxlRGF0ZVN0cmluZyIsInVuZGVmaW5lZCIsImhhbmRsZVRvZ2dsZUF1dG9VcGRhdGUiLCJuZXdBdXRvVXBkYXRlIiwicHJldiIsImNvbnRleHQiLCJpc0FjdGl2ZSIsInNlbGVjdDpwcmV2aW91cyIsIk1hdGgiLCJtYXgiLCJzZWxlY3Q6bmV4dCIsInRvdGFsSXRlbXMiLCJtaW4iLCJzZWxlY3Q6YWNjZXB0IiwibWFya2V0cGxhY2VJbmRleCIsImlucHV0IiwiaWR4IiwibWVudU9wdGlvbnMiLCJzZWxlY3RlZE9wdGlvbiIsInBvaW50ZXIiLCJpc1VwZGF0aW5nIiwiYnVsbGV0IiwibWFuaWZlc3QiLCJkZXNjcmlwdGlvbiIsIm9wdGlvbiIsImlzU2VsZWN0ZWQiLCJpbmRpY2F0b3JzIiwiY3Jvc3MiLCJNYW5hZ2VNYXJrZXRwbGFjZXNLZXlIaW50c1Byb3BzIiwiaGFzUGVuZGluZ0FjdGlvbnMiLCJNYW5hZ2VNYXJrZXRwbGFjZXNLZXlIaW50cyIsInQwIiwiJCIsIl9jIiwidDEiLCJ0MiIsInQzIiwidDQiLCJ0NSIsInQ2IiwidDciXSwic291cmNlcyI6WyJNYW5hZ2VNYXJrZXRwbGFjZXMudHN4Il0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBmaWd1cmVzIGZyb20gJ2ZpZ3VyZXMnXG5pbXBvcnQgKiBhcyBSZWFjdCBmcm9tICdyZWFjdCdcbmltcG9ydCB7IHVzZUVmZmVjdCwgdXNlUmVmLCB1c2VTdGF0ZSB9IGZyb20
|