mirror of
https://github.com/QwenLM/qwen-code.git
synced 2025-12-20 08:47:44 +00:00
136 lines
3.7 KiB
TypeScript
136 lines
3.7 KiB
TypeScript
/**
|
|
* @license
|
|
* Copyright 2025 Google LLC
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
*/
|
|
|
|
import { useState, useCallback, useEffect } from 'react';
|
|
import type { LoadedSettings, SettingScope } from '../../config/settings.js';
|
|
import { AuthType, type Config } from '@qwen-code/qwen-code-core';
|
|
import {
|
|
clearCachedCredentialFile,
|
|
getErrorMessage,
|
|
} from '@qwen-code/qwen-code-core';
|
|
import { runExitCleanup } from '../../utils/cleanup.js';
|
|
import { AuthState } from '../types.js';
|
|
import { validateAuthMethod } from '../../config/auth.js';
|
|
|
|
export function validateAuthMethodWithSettings(
|
|
authType: AuthType,
|
|
settings: LoadedSettings,
|
|
): string | null {
|
|
const enforcedType = settings.merged.security?.auth?.enforcedType;
|
|
if (enforcedType && enforcedType !== authType) {
|
|
return `Authentication is enforced to be ${enforcedType}, but you are currently using ${authType}.`;
|
|
}
|
|
if (settings.merged.security?.auth?.useExternal) {
|
|
return null;
|
|
}
|
|
return validateAuthMethod(authType);
|
|
}
|
|
|
|
export const useAuthCommand = (settings: LoadedSettings, config: Config) => {
|
|
const unAuthenticated =
|
|
settings.merged.security?.auth?.selectedType === undefined;
|
|
|
|
const [authState, setAuthState] = useState<AuthState>(
|
|
unAuthenticated ? AuthState.Updating : AuthState.Unauthenticated,
|
|
);
|
|
|
|
const [authError, setAuthError] = useState<string | null>(null);
|
|
|
|
const [isAuthenticating, setIsAuthenticating] = useState(false);
|
|
const [isAuthDialogOpen, setIsAuthDialogOpen] = useState(unAuthenticated);
|
|
|
|
const onAuthError = useCallback(
|
|
(error: string | null) => {
|
|
setAuthError(error);
|
|
if (error) {
|
|
setAuthState(AuthState.Updating);
|
|
}
|
|
},
|
|
[setAuthError, setAuthState],
|
|
);
|
|
|
|
// Authentication flow
|
|
useEffect(() => {
|
|
const authFlow = async () => {
|
|
const authType = settings.merged.security?.auth?.selectedType;
|
|
if (isAuthDialogOpen || !authType) {
|
|
return;
|
|
}
|
|
|
|
const validationError = validateAuthMethodWithSettings(
|
|
authType,
|
|
settings,
|
|
);
|
|
if (validationError) {
|
|
onAuthError(validationError);
|
|
return;
|
|
}
|
|
|
|
try {
|
|
setIsAuthenticating(true);
|
|
await config.refreshAuth(authType);
|
|
console.log(`Authenticated via "${authType}".`);
|
|
setAuthError(null);
|
|
setAuthState(AuthState.Authenticated);
|
|
} catch (e) {
|
|
onAuthError(`Failed to login. Message: ${getErrorMessage(e)}`);
|
|
} finally {
|
|
setIsAuthenticating(false);
|
|
}
|
|
};
|
|
|
|
void authFlow();
|
|
}, [isAuthDialogOpen, settings, config, onAuthError]);
|
|
|
|
// Handle auth selection from dialog
|
|
const handleAuthSelect = useCallback(
|
|
async (authType: AuthType | undefined, scope: SettingScope) => {
|
|
if (authType) {
|
|
await clearCachedCredentialFile();
|
|
|
|
settings.setValue(scope, 'security.auth.selectedType', authType);
|
|
|
|
if (
|
|
authType === AuthType.LOGIN_WITH_GOOGLE &&
|
|
config.isBrowserLaunchSuppressed()
|
|
) {
|
|
await runExitCleanup();
|
|
console.log(`
|
|
----------------------------------------------------------------
|
|
Logging in with Google... Please restart Gemini CLI to continue.
|
|
----------------------------------------------------------------
|
|
`);
|
|
process.exit(0);
|
|
}
|
|
}
|
|
|
|
setIsAuthDialogOpen(false);
|
|
setAuthError(null);
|
|
},
|
|
[settings, config],
|
|
);
|
|
|
|
const openAuthDialog = useCallback(() => {
|
|
setIsAuthDialogOpen(true);
|
|
}, []);
|
|
|
|
const cancelAuthentication = useCallback(() => {
|
|
setIsAuthenticating(false);
|
|
}, []);
|
|
|
|
return {
|
|
authState,
|
|
setAuthState,
|
|
authError,
|
|
onAuthError,
|
|
isAuthDialogOpen,
|
|
isAuthenticating,
|
|
handleAuthSelect,
|
|
openAuthDialog,
|
|
cancelAuthentication,
|
|
};
|
|
};
|