mirror of
https://github.com/google-gemini/gemini-cli.git
synced 2026-03-21 03:21:11 -07:00
Cleanup extension update logic (#10514)
Co-authored-by: gemini-code-assist[bot] <176961590+gemini-code-assist[bot]@users.noreply.github.com>
This commit is contained in:
@@ -76,7 +76,7 @@ describe('useExtensionUpdates', () => {
|
||||
const cwd = '/test/cwd';
|
||||
|
||||
vi.mocked(checkForAllExtensionUpdates).mockImplementation(
|
||||
async (extensions, dispatch) => {
|
||||
async (_extensions, dispatch, _cwd) => {
|
||||
dispatch({
|
||||
type: 'SET_STATE',
|
||||
payload: {
|
||||
@@ -122,7 +122,7 @@ describe('useExtensionUpdates', () => {
|
||||
const addItem = vi.fn();
|
||||
|
||||
vi.mocked(checkForAllExtensionUpdates).mockImplementation(
|
||||
async (extensions, dispatch) => {
|
||||
async (_extensions, dispatch, _cwd) => {
|
||||
dispatch({
|
||||
type: 'SET_STATE',
|
||||
payload: {
|
||||
@@ -195,7 +195,7 @@ describe('useExtensionUpdates', () => {
|
||||
const addItem = vi.fn();
|
||||
|
||||
vi.mocked(checkForAllExtensionUpdates).mockImplementation(
|
||||
async (extensions, dispatch) => {
|
||||
async (_extensions, dispatch, _cwd) => {
|
||||
dispatch({
|
||||
type: 'SET_STATE',
|
||||
payload: {
|
||||
@@ -280,7 +280,7 @@ describe('useExtensionUpdates', () => {
|
||||
const cwd = '/test/cwd';
|
||||
|
||||
vi.mocked(checkForAllExtensionUpdates).mockImplementation(
|
||||
async (extensions, dispatch) => {
|
||||
async (_extensions, dispatch, _cwd) => {
|
||||
dispatch({ type: 'BATCH_CHECK_START' });
|
||||
dispatch({
|
||||
type: 'SET_STATE',
|
||||
|
||||
@@ -18,7 +18,10 @@ import {
|
||||
checkForAllExtensionUpdates,
|
||||
updateExtension,
|
||||
} from '../../config/extensions/update.js';
|
||||
import { requestConsentInteractive } from '../../config/extension.js';
|
||||
import {
|
||||
requestConsentInteractive,
|
||||
type ExtensionUpdateInfo,
|
||||
} from '../../config/extension.js';
|
||||
import { checkExhaustive } from '../../utils/checks.js';
|
||||
|
||||
type ConfirmationRequestWrapper = {
|
||||
@@ -41,7 +44,6 @@ function confirmationRequestsReducer(
|
||||
return state.filter((r) => r !== action.request);
|
||||
default:
|
||||
checkExhaustive(action);
|
||||
return state;
|
||||
}
|
||||
}
|
||||
|
||||
@@ -80,40 +82,77 @@ export const useExtensionUpdates = (
|
||||
);
|
||||
|
||||
useEffect(() => {
|
||||
(async () => {
|
||||
await checkForAllExtensionUpdates(
|
||||
extensions,
|
||||
dispatchExtensionStateUpdate,
|
||||
const extensionsToCheck = extensions.filter((extension) => {
|
||||
const currentStatus = extensionsUpdateState.extensionStatuses.get(
|
||||
extension.name,
|
||||
);
|
||||
})();
|
||||
}, [extensions, extensions.length, dispatchExtensionStateUpdate]);
|
||||
if (!currentStatus) return true;
|
||||
const currentState = currentStatus.status;
|
||||
return !currentState || currentState === ExtensionUpdateState.UNKNOWN;
|
||||
});
|
||||
if (extensionsToCheck.length === 0) return;
|
||||
checkForAllExtensionUpdates(
|
||||
extensionsToCheck,
|
||||
dispatchExtensionStateUpdate,
|
||||
cwd,
|
||||
);
|
||||
}, [
|
||||
extensions,
|
||||
extensionsUpdateState.extensionStatuses,
|
||||
cwd,
|
||||
dispatchExtensionStateUpdate,
|
||||
]);
|
||||
|
||||
useEffect(() => {
|
||||
if (extensionsUpdateState.batchChecksInProgress > 0) {
|
||||
return;
|
||||
}
|
||||
const scheduledUpdate = extensionsUpdateState.scheduledUpdate;
|
||||
if (scheduledUpdate) {
|
||||
dispatchExtensionStateUpdate({
|
||||
type: 'CLEAR_SCHEDULED_UPDATE',
|
||||
});
|
||||
}
|
||||
|
||||
function shouldDoUpdate(extension: GeminiCLIExtension): boolean {
|
||||
if (scheduledUpdate) {
|
||||
if (scheduledUpdate.all) {
|
||||
return true;
|
||||
}
|
||||
return scheduledUpdate.names?.includes(extension.name) === true;
|
||||
} else {
|
||||
return extension.installMetadata?.autoUpdate === true;
|
||||
}
|
||||
}
|
||||
|
||||
let extensionsWithUpdatesCount = 0;
|
||||
// We only notify if we have unprocessed extensions in the UPDATE_AVAILABLE
|
||||
// state.
|
||||
let shouldNotifyOfUpdates = false;
|
||||
const updatePromises: Array<Promise<ExtensionUpdateInfo | undefined>> = [];
|
||||
for (const extension of extensions) {
|
||||
const currentState = extensionsUpdateState.extensionStatuses.get(
|
||||
extension.name,
|
||||
);
|
||||
if (
|
||||
!currentState ||
|
||||
currentState.processed ||
|
||||
currentState.status !== ExtensionUpdateState.UPDATE_AVAILABLE
|
||||
) {
|
||||
continue;
|
||||
}
|
||||
|
||||
// Mark as processed immediately to avoid re-triggering.
|
||||
dispatchExtensionStateUpdate({
|
||||
type: 'SET_PROCESSED',
|
||||
payload: { name: extension.name, processed: true },
|
||||
});
|
||||
|
||||
if (extension.installMetadata?.autoUpdate) {
|
||||
updateExtension(
|
||||
const shouldUpdate = shouldDoUpdate(extension);
|
||||
if (!shouldUpdate) {
|
||||
extensionsWithUpdatesCount++;
|
||||
if (!currentState.notified) {
|
||||
// Mark as processed immediately to avoid re-triggering.
|
||||
dispatchExtensionStateUpdate({
|
||||
type: 'SET_NOTIFIED',
|
||||
payload: { name: extension.name, notified: true },
|
||||
});
|
||||
shouldNotifyOfUpdates = true;
|
||||
}
|
||||
} else {
|
||||
const updatePromise = updateExtension(
|
||||
extension,
|
||||
cwd,
|
||||
(description) =>
|
||||
@@ -123,7 +162,9 @@ export const useExtensionUpdates = (
|
||||
),
|
||||
currentState.status,
|
||||
dispatchExtensionStateUpdate,
|
||||
)
|
||||
);
|
||||
updatePromises.push(updatePromise);
|
||||
updatePromise
|
||||
.then((result) => {
|
||||
if (!result) return;
|
||||
addItem(
|
||||
@@ -143,11 +184,9 @@ export const useExtensionUpdates = (
|
||||
Date.now(),
|
||||
);
|
||||
});
|
||||
} else {
|
||||
extensionsWithUpdatesCount++;
|
||||
}
|
||||
}
|
||||
if (extensionsWithUpdatesCount > 0) {
|
||||
if (shouldNotifyOfUpdates) {
|
||||
const s = extensionsWithUpdatesCount > 1 ? 's' : '';
|
||||
addItem(
|
||||
{
|
||||
@@ -157,6 +196,18 @@ export const useExtensionUpdates = (
|
||||
Date.now(),
|
||||
);
|
||||
}
|
||||
if (scheduledUpdate) {
|
||||
Promise.all(updatePromises).then((results) => {
|
||||
const nonNullResults = results.filter((result) => result != null);
|
||||
scheduledUpdate.onCompleteCallbacks.forEach((callback) => {
|
||||
try {
|
||||
callback(nonNullResults);
|
||||
} catch (e) {
|
||||
console.error(getErrorMessage(e));
|
||||
}
|
||||
});
|
||||
});
|
||||
}
|
||||
}, [
|
||||
extensions,
|
||||
extensionsUpdateState,
|
||||
|
||||
Reference in New Issue
Block a user