From db0011a26b2b15e5d047353657af724f8dc9078a Mon Sep 17 00:00:00 2001 From: mamadoudicko Date: Wed, 30 Aug 2023 09:09:26 +0200 Subject: [PATCH] feat: add openai api key validation --- .../SettingsTab/hooks/useSettingsTab.ts | 45 +++++++----- .../SettingsTab/utils/validateOpenAIKey.ts | 71 +++++++++++++++++++ .../ApiKeyConfig/hooks/useApiKeyConfig.ts | 21 ++++++ frontend/public/locales/en/config.json | 6 +- frontend/public/locales/es/config.json | 4 +- frontend/public/locales/fr/config.json | 4 +- frontend/public/locales/pt-br/config.json | 4 +- frontend/public/locales/ru/config.json | 4 +- frontend/public/locales/zh-cn/config.json | 4 +- 9 files changed, 138 insertions(+), 25 deletions(-) create mode 100644 frontend/app/brains-management/[brainId]/components/BrainManagementTabs/components/SettingsTab/utils/validateOpenAIKey.ts diff --git a/frontend/app/brains-management/[brainId]/components/BrainManagementTabs/components/SettingsTab/hooks/useSettingsTab.ts b/frontend/app/brains-management/[brainId]/components/BrainManagementTabs/components/SettingsTab/hooks/useSettingsTab.ts index a64101785f4f..60d9b41095ee 100644 --- a/frontend/app/brains-management/[brainId]/components/BrainManagementTabs/components/SettingsTab/hooks/useSettingsTab.ts +++ b/frontend/app/brains-management/[brainId]/components/BrainManagementTabs/components/SettingsTab/hooks/useSettingsTab.ts @@ -14,6 +14,8 @@ import { Brain } from "@/lib/context/BrainProvider/types"; import { defineMaxTokens } from "@/lib/helpers/defineMaxTokens"; import { useToast } from "@/lib/hooks"; +import { validateOpenAIKey } from "../utils/validateOpenAIKey"; + type UseSettingsTabProps = { brainId: UUID; }; @@ -95,8 +97,7 @@ export const useSettingsTab = ({ brainId }: UseSettingsTabProps) => { if (brain.model !== undefined) { setValue("model", brain.model); } - },50); - + }, 50); }; useEffect(() => { void fetchBrain(); @@ -142,7 +143,7 @@ export const useSettingsTab = ({ brainId }: UseSettingsTabProps) => { await setAsDefaultBrain(brainId); publish({ variant: "success", - text: t("defaultBrainSet",{ns:"config"}), + text: t("defaultBrainSet", { ns: "config" }), }); void fetchAllBrains(); void fetchDefaultBrain(); @@ -180,12 +181,12 @@ export const useSettingsTab = ({ brainId }: UseSettingsTabProps) => { void fetchBrain(); publish({ variant: "success", - text: t("promptRemoved",{ns:"config"}), + text: t("promptRemoved", { ns: "config" }), }); } catch (err) { publish({ variant: "danger", - text: t("errorRemovingPrompt",{ns:"config"}), + text: t("errorRemovingPrompt", { ns: "config" }), }); } finally { setIsUpdating(false); @@ -203,31 +204,39 @@ export const useSettingsTab = ({ brainId }: UseSettingsTabProps) => { } }; - const handleSubmit = async (checkDirty:boolean) => { + const handleSubmit = async (checkDirty: boolean) => { const hasChanges = Object.keys(dirtyFields).length > 0; if (!hasChanges && checkDirty) { return; } const { name: isNameDirty } = dirtyFields; - const { name } = getValues(); + const { name, openAiKey: openai_api_key } = getValues(); if (isNameDirty !== undefined && isNameDirty && name.trim() === "") { publish({ variant: "danger", - text: t("nameRequired",{ns:"config"}), + text: t("nameRequired", { ns: "config" }), }); return; } + if ( + openai_api_key !== undefined && + !(await validateOpenAIKey( + openai_api_key, + { + badApiKeyError: t("incorrectApiKey", { ns: "config" }), + invalidApiKeyError: t("invalidApiKeyError", { ns: "config" }), + }, + publish + )) + ) { + return; + } + try { setIsUpdating(true); - - const { - maxTokens: max_tokens, - openAiKey: openai_api_key, - prompt, - ...otherConfigs - } = getValues(); + const { maxTokens: max_tokens, prompt, ...otherConfigs } = getValues(); if ( dirtyFields["prompt"] && @@ -235,7 +244,7 @@ export const useSettingsTab = ({ brainId }: UseSettingsTabProps) => { ) { publish({ variant: "warning", - text: t("promptFieldsRequired",{ns:"config"}), + text: t("promptFieldsRequired", { ns: "config" }), }); return; @@ -279,7 +288,7 @@ export const useSettingsTab = ({ brainId }: UseSettingsTabProps) => { publish({ variant: "success", - text: t("brainUpdated",{ns:"config"}), + text: t("brainUpdated", { ns: "config" }), }); void fetchAllBrains(); } catch (err) { @@ -318,7 +327,7 @@ export const useSettingsTab = ({ brainId }: UseSettingsTabProps) => { setValue("prompt.content", content, { shouldDirty: true, }); - }; + }; return { handleSubmit, diff --git a/frontend/app/brains-management/[brainId]/components/BrainManagementTabs/components/SettingsTab/utils/validateOpenAIKey.ts b/frontend/app/brains-management/[brainId]/components/BrainManagementTabs/components/SettingsTab/utils/validateOpenAIKey.ts new file mode 100644 index 000000000000..3f158d1c3472 --- /dev/null +++ b/frontend/app/brains-management/[brainId]/components/BrainManagementTabs/components/SettingsTab/utils/validateOpenAIKey.ts @@ -0,0 +1,71 @@ +import axios from "axios"; + +import { ToastData } from "@/lib/components/ui/Toast/domain/types"; +import { getAxiosErrorParams } from "@/lib/helpers/getAxiosErrorParams"; + +export const getOpenAIKeyValidationStatusCode = async ( + key: string +): Promise => { + const url = "https://api.openai.com/v1/chat/completions"; + const headers = { + Authorization: `Bearer ${key}`, + "Content-Type": "application/json", + }; + + const data = JSON.stringify({ + model: "gpt-3.5-turbo", + messages: [ + { + role: "user", + content: "Hello!", + }, + ], + }); + + try { + await axios.post(url, data, { headers }); + + return 200; + } catch (error) { + return getAxiosErrorParams(error)?.status ?? 400; + } +}; + +type ErrorMessages = { + badApiKeyError: string; + invalidApiKeyError: string; +}; + +export const validateOpenAIKey = async ( + openai_api_key: string | undefined, + errorMessages: ErrorMessages, + publish: (toast: ToastData) => void +): Promise => { + if (openai_api_key !== undefined) { + const keyValidationStatusCode = await getOpenAIKeyValidationStatusCode( + openai_api_key + ); + + if (keyValidationStatusCode !== 200) { + if (keyValidationStatusCode === 401) { + publish({ + variant: "danger", + text: errorMessages.badApiKeyError, + }); + } + + if (keyValidationStatusCode === 429) { + publish({ + variant: "danger", + text: errorMessages.invalidApiKeyError, + }); + } + + return false; + } + + return true; + } + + return false; +}; diff --git a/frontend/app/user/components/ApiKeyConfig/hooks/useApiKeyConfig.ts b/frontend/app/user/components/ApiKeyConfig/hooks/useApiKeyConfig.ts index 529f16314fb3..22985b8f3ff3 100644 --- a/frontend/app/user/components/ApiKeyConfig/hooks/useApiKeyConfig.ts +++ b/frontend/app/user/components/ApiKeyConfig/hooks/useApiKeyConfig.ts @@ -1,6 +1,8 @@ /* eslint-disable max-lines */ import { useEffect, useState } from "react"; +import { useTranslation } from "react-i18next"; +import { validateOpenAIKey } from "@/app/brains-management/[brainId]/components/BrainManagementTabs/components/SettingsTab/utils/validateOpenAIKey"; import { useAuthApi } from "@/lib/api/auth/useAuthApi"; import { useUserApi } from "@/lib/api/user/useUserApi"; import { UserIdentity } from "@/lib/api/user/user"; @@ -20,6 +22,7 @@ export const useApiKeyConfig = () => { const { createApiKey } = useAuthApi(); const { publish } = useToast(); const [userIdentity, setUserIdentity] = useState(); + const { t } = useTranslation(["config"]); const fetchUserIdentity = async () => { setUserIdentity(await getUserIdentity()); @@ -56,6 +59,24 @@ export const useApiKeyConfig = () => { const changeOpenAiApiKey = async () => { try { setChangeOpenAiApiKeyRequestPending(true); + + if ( + openAiApiKey !== undefined && + openAiApiKey !== null && + !(await validateOpenAIKey( + openAiApiKey, + { + badApiKeyError: t("incorrectApiKey", { ns: "config" }), + invalidApiKeyError: t("invalidApiKeyError", { ns: "config" }), + }, + publish + )) + ) { + setChangeOpenAiApiKeyRequestPending(false); + + return; + } + await updateUserIdentity({ openai_api_key: openAiApiKey, }); diff --git a/frontend/public/locales/en/config.json b/frontend/public/locales/en/config.json index e6deccfdaa4e..41284569482e 100644 --- a/frontend/public/locales/en/config.json +++ b/frontend/public/locales/en/config.json @@ -46,5 +46,7 @@ "roleRequired": "You don't have the necessary role to access this tab 🧠💡🥲.", "requireAccess": "Please require access from the owner.", "ohno": "Oh no!", - "noUser": "No user" -} \ No newline at end of file + "noUser": "No user", + "incorrectApiKey": "Incorrect API Key", + "invalidApiKeyError": "Invalid API Key" +} \ No newline at end of file diff --git a/frontend/public/locales/es/config.json b/frontend/public/locales/es/config.json index cb98fe1a0fe8..ccdbfd567a8d 100644 --- a/frontend/public/locales/es/config.json +++ b/frontend/public/locales/es/config.json @@ -46,5 +46,7 @@ "supabaseURLPlaceHolder": "URL de Supabase", "temperature": "Temperatura", "title": "Configuración", - "updatingBrainSettings": "Actualizando configuración del cerebro..." + "updatingBrainSettings": "Actualizando configuración del cerebro...", + "incorrectApiKey": "Clave de API incorrecta", + "invalidApiKeyError": "Clave de API inválida" } \ No newline at end of file diff --git a/frontend/public/locales/fr/config.json b/frontend/public/locales/fr/config.json index d302ffbb26a5..3b47671b4945 100644 --- a/frontend/public/locales/fr/config.json +++ b/frontend/public/locales/fr/config.json @@ -46,5 +46,7 @@ "supabaseURLPlaceHolder": "URL Supabase", "temperature": "Température", "title": "Configuration", - "updatingBrainSettings": "Mise à jour des paramètres du cerveau..." + "updatingBrainSettings": "Mise à jour des paramètres du cerveau...", + "incorrectApiKey": "Clé API incorrecte", + "invalidApiKeyError": "Clé API invalide" } \ No newline at end of file diff --git a/frontend/public/locales/pt-br/config.json b/frontend/public/locales/pt-br/config.json index eb828a886a7f..fba4a4be2e21 100644 --- a/frontend/public/locales/pt-br/config.json +++ b/frontend/public/locales/pt-br/config.json @@ -46,5 +46,7 @@ "roleRequired": "Você não possui a função necessária para acessar esta aba 🧠💡🥲.", "requireAccess": "Por favor, solicite acesso ao proprietário.", "ohno": "Oh, não!", - "noUser": "Nenhum usuário" + "noUser": "Nenhum usuário", + "incorrectApiKey": "Chave de API incorreta", + "invalidApiKeyError": "Chave de API inválida" } \ No newline at end of file diff --git a/frontend/public/locales/ru/config.json b/frontend/public/locales/ru/config.json index ad27c9d8e636..4ee246231c14 100644 --- a/frontend/public/locales/ru/config.json +++ b/frontend/public/locales/ru/config.json @@ -46,5 +46,7 @@ "roleRequired": "У вас нет необходимой роли для доступа к этой вкладке 🧠💡🥲.", "requireAccess": "Пожалуйста, запросите доступ у владельца.", "ohno": "О нет!", - "noUser": "Пользователь не найден" + "noUser": "Пользователь не найден", + "incorrectApiKey": "Неверный ключ API", + "invalidApiKeyError": "Недействительный ключ API" } diff --git a/frontend/public/locales/zh-cn/config.json b/frontend/public/locales/zh-cn/config.json index c92017583278..a0afea2aaae1 100644 --- a/frontend/public/locales/zh-cn/config.json +++ b/frontend/public/locales/zh-cn/config.json @@ -46,5 +46,7 @@ "roleRequired": "您没有访问此选项卡所需的权限 🧠💡🥲.", "requireAccess": "请向所有者申请访问权限.", "ohno": "哎呀!", - "noUser": "没有用户" + "noUser": "没有用户", + "incorrectApiKey": "无效的API密钥", + "invalidApiKeyError": "无效的API密钥" } \ No newline at end of file