mirror of
https://github.com/arc53/DocsGPT.git
synced 2025-11-29 08:33:20 +00:00
synced with upstream
This commit is contained in:
2
frontend/package-lock.json
generated
2
frontend/package-lock.json
generated
@@ -1490,7 +1490,7 @@
|
|||||||
"version": "18.0.10",
|
"version": "18.0.10",
|
||||||
"resolved": "https://registry.npmjs.org/@types/react-dom/-/react-dom-18.0.10.tgz",
|
"resolved": "https://registry.npmjs.org/@types/react-dom/-/react-dom-18.0.10.tgz",
|
||||||
"integrity": "sha512-E42GW/JA4Qv15wQdqJq8DL4JhNpB3prJgjgapN3qJT9K2zO5IIAQh4VXvCEDupoqAwnz0cY4RlXeC/ajX5SFHg==",
|
"integrity": "sha512-E42GW/JA4Qv15wQdqJq8DL4JhNpB3prJgjgapN3qJT9K2zO5IIAQh4VXvCEDupoqAwnz0cY4RlXeC/ajX5SFHg==",
|
||||||
"dev": true,
|
"devOptional": true,
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@types/react": "*"
|
"@types/react": "*"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,46 +1,48 @@
|
|||||||
import { useEffect, useRef, useState } from 'react';
|
import { useEffect, useRef, useState } from 'react';
|
||||||
|
import { useTranslation } from 'react-i18next';
|
||||||
import { useDispatch, useSelector } from 'react-redux';
|
import { useDispatch, useSelector } from 'react-redux';
|
||||||
import { NavLink, useNavigate } from 'react-router-dom';
|
import { NavLink, useNavigate } from 'react-router-dom';
|
||||||
|
|
||||||
|
import conversationService from './api/services/conversationService';
|
||||||
|
import userService from './api/services/userService';
|
||||||
|
import Add from './assets/add.svg';
|
||||||
import DocsGPT3 from './assets/cute_docsgpt3.svg';
|
import DocsGPT3 from './assets/cute_docsgpt3.svg';
|
||||||
import Discord from './assets/discord.svg';
|
import Discord from './assets/discord.svg';
|
||||||
import Expand from './assets/expand.svg';
|
import Expand from './assets/expand.svg';
|
||||||
import Github from './assets/github.svg';
|
import Github from './assets/github.svg';
|
||||||
import Hamburger from './assets/hamburger.svg';
|
|
||||||
import HamburgerDark from './assets/hamburger-dark.svg';
|
import HamburgerDark from './assets/hamburger-dark.svg';
|
||||||
|
import Hamburger from './assets/hamburger.svg';
|
||||||
import Info from './assets/info.svg';
|
import Info from './assets/info.svg';
|
||||||
import SettingGear from './assets/settingGear.svg';
|
import SettingGear from './assets/settingGear.svg';
|
||||||
import Twitter from './assets/TwitterX.svg';
|
import Twitter from './assets/TwitterX.svg';
|
||||||
import Add from './assets/add.svg';
|
|
||||||
import UploadIcon from './assets/upload.svg';
|
import UploadIcon from './assets/upload.svg';
|
||||||
import { ActiveState } from './models/misc';
|
import SourceDropdown from './components/SourceDropdown';
|
||||||
import APIKeyModal from './preferences/APIKeyModal';
|
|
||||||
import DeleteConvModal from './modals/DeleteConvModal';
|
|
||||||
|
|
||||||
import {
|
|
||||||
selectApiKeyStatus,
|
|
||||||
selectSelectedDocs,
|
|
||||||
selectSelectedDocsStatus,
|
|
||||||
selectSourceDocs,
|
|
||||||
setSelectedDocs,
|
|
||||||
selectConversations,
|
|
||||||
setConversations,
|
|
||||||
selectConversationId,
|
|
||||||
selectModalStateDeleteConv,
|
|
||||||
setModalStateDeleteConv,
|
|
||||||
setSourceDocs,
|
|
||||||
} from './preferences/preferenceSlice';
|
|
||||||
import {
|
import {
|
||||||
setConversation,
|
setConversation,
|
||||||
updateConversationId,
|
updateConversationId,
|
||||||
} from './conversation/conversationSlice';
|
} from './conversation/conversationSlice';
|
||||||
import { useMediaQuery, useOutsideAlerter } from './hooks';
|
|
||||||
import Upload from './upload/Upload';
|
|
||||||
import { Doc, getConversations, getDocs } from './preferences/preferenceApi';
|
|
||||||
import SelectDocsModal from './preferences/SelectDocsModal';
|
|
||||||
import ConversationTile from './conversation/ConversationTile';
|
import ConversationTile from './conversation/ConversationTile';
|
||||||
import { useDarkTheme } from './hooks';
|
import { useDarkTheme, useMediaQuery, useOutsideAlerter } from './hooks';
|
||||||
import SourceDropdown from './components/SourceDropdown';
|
import DeleteConvModal from './modals/DeleteConvModal';
|
||||||
import { useTranslation } from 'react-i18next';
|
import { ActiveState } from './models/misc';
|
||||||
|
import APIKeyModal from './preferences/APIKeyModal';
|
||||||
|
import { Doc, getConversations, getDocs } from './preferences/preferenceApi';
|
||||||
|
import {
|
||||||
|
selectApiKeyStatus,
|
||||||
|
selectConversationId,
|
||||||
|
selectConversations,
|
||||||
|
selectModalStateDeleteConv,
|
||||||
|
selectSelectedDocs,
|
||||||
|
selectSelectedDocsStatus,
|
||||||
|
selectSourceDocs,
|
||||||
|
setConversations,
|
||||||
|
setModalStateDeleteConv,
|
||||||
|
setSelectedDocs,
|
||||||
|
setSourceDocs,
|
||||||
|
} from './preferences/preferenceSlice';
|
||||||
|
import SelectDocsModal from './preferences/SelectDocsModal';
|
||||||
|
import Upload from './upload/Upload';
|
||||||
|
|
||||||
interface NavigationProps {
|
interface NavigationProps {
|
||||||
navOpen: boolean;
|
navOpen: boolean;
|
||||||
setNavOpen: React.Dispatch<React.SetStateAction<boolean>>;
|
setNavOpen: React.Dispatch<React.SetStateAction<boolean>>;
|
||||||
@@ -85,7 +87,6 @@ export default function Navigation({ navOpen, setNavOpen }: NavigationProps) {
|
|||||||
useState<ActiveState>('INACTIVE');
|
useState<ActiveState>('INACTIVE');
|
||||||
|
|
||||||
const navRef = useRef(null);
|
const navRef = useRef(null);
|
||||||
const apiHost = import.meta.env.VITE_API_HOST || 'https://docsapi.arc53.com';
|
|
||||||
|
|
||||||
const navigate = useNavigate();
|
const navigate = useNavigate();
|
||||||
|
|
||||||
@@ -106,9 +107,8 @@ export default function Navigation({ navOpen, setNavOpen }: NavigationProps) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const handleDeleteAllConversations = () => {
|
const handleDeleteAllConversations = () => {
|
||||||
fetch(`${apiHost}/api/delete_all_conversations`, {
|
conversationService
|
||||||
method: 'POST',
|
.deleteAll({})
|
||||||
})
|
|
||||||
.then(() => {
|
.then(() => {
|
||||||
fetchConversations();
|
fetchConversations();
|
||||||
})
|
})
|
||||||
@@ -116,9 +116,8 @@ export default function Navigation({ navOpen, setNavOpen }: NavigationProps) {
|
|||||||
};
|
};
|
||||||
|
|
||||||
const handleDeleteConversation = (id: string) => {
|
const handleDeleteConversation = (id: string) => {
|
||||||
fetch(`${apiHost}/api/delete_conversation?id=${id}`, {
|
conversationService
|
||||||
method: 'POST',
|
.delete(id, {})
|
||||||
})
|
|
||||||
.then(() => {
|
.then(() => {
|
||||||
fetchConversations();
|
fetchConversations();
|
||||||
})
|
})
|
||||||
@@ -128,17 +127,9 @@ export default function Navigation({ navOpen, setNavOpen }: NavigationProps) {
|
|||||||
const handleDeleteClick = (doc: Doc) => {
|
const handleDeleteClick = (doc: Doc) => {
|
||||||
const docPath = `indexes/local/${doc.name}`;
|
const docPath = `indexes/local/${doc.name}`;
|
||||||
|
|
||||||
fetch(`${apiHost}/api/delete_old?path=${docPath}`, {
|
userService
|
||||||
method: 'GET',
|
.deletePath(docPath)
|
||||||
})
|
|
||||||
.then(() => {
|
.then(() => {
|
||||||
// remove the image element from the DOM
|
|
||||||
// const imageElement = document.querySelector(
|
|
||||||
// `#img-${index}`,
|
|
||||||
// ) as HTMLElement;
|
|
||||||
// const parentElement = imageElement.parentNode as HTMLElement;
|
|
||||||
// parentElement.parentNode?.removeChild(parentElement);
|
|
||||||
|
|
||||||
return getDocs();
|
return getDocs();
|
||||||
})
|
})
|
||||||
.then((updatedDocs) => {
|
.then((updatedDocs) => {
|
||||||
@@ -153,10 +144,8 @@ export default function Navigation({ navOpen, setNavOpen }: NavigationProps) {
|
|||||||
};
|
};
|
||||||
|
|
||||||
const handleConversationClick = (index: string) => {
|
const handleConversationClick = (index: string) => {
|
||||||
// fetch the conversation from the server and setConversation in the store
|
conversationService
|
||||||
fetch(`${apiHost}/api/get_single_conversation?id=${index}`, {
|
.getConversation(index)
|
||||||
method: 'GET',
|
|
||||||
})
|
|
||||||
.then((response) => response.json())
|
.then((response) => response.json())
|
||||||
.then((data) => {
|
.then((data) => {
|
||||||
navigate('/');
|
navigate('/');
|
||||||
@@ -173,13 +162,8 @@ export default function Navigation({ navOpen, setNavOpen }: NavigationProps) {
|
|||||||
name: string;
|
name: string;
|
||||||
id: string;
|
id: string;
|
||||||
}) {
|
}) {
|
||||||
await fetch(`${apiHost}/api/update_conversation_name`, {
|
await conversationService
|
||||||
method: 'POST',
|
.update(updatedConversation)
|
||||||
headers: {
|
|
||||||
'Content-Type': 'application/json',
|
|
||||||
},
|
|
||||||
body: JSON.stringify(updatedConversation),
|
|
||||||
})
|
|
||||||
.then((response) => response.json())
|
.then((response) => response.json())
|
||||||
.then((data) => {
|
.then((data) => {
|
||||||
if (data) {
|
if (data) {
|
||||||
|
|||||||
69
frontend/src/api/client.ts
Normal file
69
frontend/src/api/client.ts
Normal file
@@ -0,0 +1,69 @@
|
|||||||
|
const baseURL = import.meta.env.VITE_API_HOST || 'https://docsapi.arc53.com';
|
||||||
|
|
||||||
|
const defaultHeaders = {
|
||||||
|
'Content-Type': 'application/json',
|
||||||
|
};
|
||||||
|
|
||||||
|
const apiClient = {
|
||||||
|
get: (url: string, headers = {}, signal?: AbortSignal): Promise<any> =>
|
||||||
|
fetch(`${baseURL}${url}`, {
|
||||||
|
method: 'GET',
|
||||||
|
headers: {
|
||||||
|
...defaultHeaders,
|
||||||
|
...headers,
|
||||||
|
},
|
||||||
|
signal,
|
||||||
|
}).then((response) => {
|
||||||
|
return response;
|
||||||
|
}),
|
||||||
|
|
||||||
|
post: (
|
||||||
|
url: string,
|
||||||
|
data: any,
|
||||||
|
headers = {},
|
||||||
|
signal?: AbortSignal,
|
||||||
|
): Promise<any> =>
|
||||||
|
fetch(`${baseURL}${url}`, {
|
||||||
|
method: 'POST',
|
||||||
|
headers: {
|
||||||
|
...defaultHeaders,
|
||||||
|
...headers,
|
||||||
|
},
|
||||||
|
body: JSON.stringify(data),
|
||||||
|
signal,
|
||||||
|
}).then((response) => {
|
||||||
|
return response;
|
||||||
|
}),
|
||||||
|
|
||||||
|
put: (
|
||||||
|
url: string,
|
||||||
|
data: any,
|
||||||
|
headers = {},
|
||||||
|
signal?: AbortSignal,
|
||||||
|
): Promise<any> =>
|
||||||
|
fetch(`${baseURL}${url}`, {
|
||||||
|
method: 'PUT',
|
||||||
|
headers: {
|
||||||
|
...defaultHeaders,
|
||||||
|
...headers,
|
||||||
|
},
|
||||||
|
body: JSON.stringify(data),
|
||||||
|
signal,
|
||||||
|
}).then((response) => {
|
||||||
|
return response;
|
||||||
|
}),
|
||||||
|
|
||||||
|
delete: (url: string, headers = {}, signal?: AbortSignal): Promise<any> =>
|
||||||
|
fetch(`${baseURL}${url}`, {
|
||||||
|
method: 'DELETE',
|
||||||
|
headers: {
|
||||||
|
...defaultHeaders,
|
||||||
|
...headers,
|
||||||
|
},
|
||||||
|
signal,
|
||||||
|
}).then((response) => {
|
||||||
|
return response;
|
||||||
|
}),
|
||||||
|
};
|
||||||
|
|
||||||
|
export default apiClient;
|
||||||
33
frontend/src/api/endpoints.ts
Normal file
33
frontend/src/api/endpoints.ts
Normal file
@@ -0,0 +1,33 @@
|
|||||||
|
const endpoints = {
|
||||||
|
USER: {
|
||||||
|
DOCS: '/api/combine',
|
||||||
|
DOCS_CHECK: '/api/docs_check',
|
||||||
|
API_KEYS: '/api/get_api_keys',
|
||||||
|
CREATE_API_KEY: '/api/create_api_key',
|
||||||
|
DELETE_API_KEY: '/api/delete_api_key',
|
||||||
|
PROMPTS: '/api/get_prompts',
|
||||||
|
CREATE_PROMPT: '/api/create_prompt',
|
||||||
|
DELETE_PROMPT: '/api/delete_prompt',
|
||||||
|
UPDATE_PROMPT: '/api/update_prompt',
|
||||||
|
SINGLE_PROMPT: (id: string) => `/api/get_single_prompt?id=${id}`,
|
||||||
|
DELETE_PATH: (docPath: string) => `/api/delete_old?path=${docPath}`,
|
||||||
|
TASK_STATUS: (task_id: string) => `/api/task_status?task_id=${task_id}`,
|
||||||
|
},
|
||||||
|
CONVERSATION: {
|
||||||
|
ANSWER: '/api/answer',
|
||||||
|
ANSWER_STREAMING: '/stream',
|
||||||
|
SEARCH: '/api/search',
|
||||||
|
FEEDBACK: '/api/feedback',
|
||||||
|
CONVERSATION: (id: string) => `/api/get_single_conversation?id=${id}`,
|
||||||
|
CONVERSATIONS: '/api/get_conversations',
|
||||||
|
SHARE_CONVERSATION: (isPromptable: boolean) =>
|
||||||
|
`/api/share?isPromptable=${isPromptable}`,
|
||||||
|
SHARED_CONVERSATION: (identifier: string) =>
|
||||||
|
`/api/shared_conversation/${identifier}`,
|
||||||
|
DELETE: (id: string) => `/api/delete_conversation?id=${id}`,
|
||||||
|
DELETE_ALL: '/api/delete_all_conversations',
|
||||||
|
UPDATE: '/api/update_conversation_name',
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
export default endpoints;
|
||||||
32
frontend/src/api/services/conversationService.ts
Normal file
32
frontend/src/api/services/conversationService.ts
Normal file
@@ -0,0 +1,32 @@
|
|||||||
|
import apiClient from '../client';
|
||||||
|
import endpoints from '../endpoints';
|
||||||
|
|
||||||
|
const conversationService = {
|
||||||
|
answer: (data: any, signal: AbortSignal): Promise<any> =>
|
||||||
|
apiClient.post(endpoints.CONVERSATION.ANSWER, data, {}, signal),
|
||||||
|
answerStream: (data: any, signal: AbortSignal): Promise<any> =>
|
||||||
|
apiClient.post(endpoints.CONVERSATION.ANSWER_STREAMING, data, {}, signal),
|
||||||
|
search: (data: any): Promise<any> =>
|
||||||
|
apiClient.post(endpoints.CONVERSATION.SEARCH, data),
|
||||||
|
feedback: (data: any): Promise<any> =>
|
||||||
|
apiClient.post(endpoints.CONVERSATION.FEEDBACK, data),
|
||||||
|
getConversation: (id: string): Promise<any> =>
|
||||||
|
apiClient.get(endpoints.CONVERSATION.CONVERSATION(id)),
|
||||||
|
getConversations: (): Promise<any> =>
|
||||||
|
apiClient.get(endpoints.CONVERSATION.CONVERSATIONS),
|
||||||
|
shareConversation: (isPromptable: boolean, data: any): Promise<any> =>
|
||||||
|
apiClient.post(
|
||||||
|
endpoints.CONVERSATION.SHARE_CONVERSATION(isPromptable),
|
||||||
|
data,
|
||||||
|
),
|
||||||
|
getSharedConversation: (identifier: string): Promise<any> =>
|
||||||
|
apiClient.get(endpoints.CONVERSATION.SHARED_CONVERSATION(identifier)),
|
||||||
|
delete: (id: string, data: any): Promise<any> =>
|
||||||
|
apiClient.post(endpoints.CONVERSATION.DELETE(id), data),
|
||||||
|
deleteAll: (data: any): Promise<any> =>
|
||||||
|
apiClient.post(endpoints.CONVERSATION.DELETE_ALL, data),
|
||||||
|
update: (data: any): Promise<any> =>
|
||||||
|
apiClient.post(endpoints.CONVERSATION.UPDATE, data),
|
||||||
|
};
|
||||||
|
|
||||||
|
export default conversationService;
|
||||||
28
frontend/src/api/services/userService.ts
Normal file
28
frontend/src/api/services/userService.ts
Normal file
@@ -0,0 +1,28 @@
|
|||||||
|
import apiClient from '../client';
|
||||||
|
import endpoints from '../endpoints';
|
||||||
|
|
||||||
|
const userService = {
|
||||||
|
getDocs: (): Promise<any> => apiClient.get(endpoints.USER.DOCS),
|
||||||
|
checkDocs: (data: any): Promise<any> =>
|
||||||
|
apiClient.post(endpoints.USER.DOCS_CHECK, data),
|
||||||
|
getAPIKeys: (): Promise<any> => apiClient.get(endpoints.USER.API_KEYS),
|
||||||
|
createAPIKey: (data: any): Promise<any> =>
|
||||||
|
apiClient.post(endpoints.USER.CREATE_API_KEY, data),
|
||||||
|
deleteAPIKey: (data: any): Promise<any> =>
|
||||||
|
apiClient.post(endpoints.USER.DELETE_API_KEY, data),
|
||||||
|
getPrompts: (): Promise<any> => apiClient.get(endpoints.USER.PROMPTS),
|
||||||
|
createPrompt: (data: any): Promise<any> =>
|
||||||
|
apiClient.post(endpoints.USER.CREATE_PROMPT, data),
|
||||||
|
deletePrompt: (data: any): Promise<any> =>
|
||||||
|
apiClient.post(endpoints.USER.DELETE_PROMPT, data),
|
||||||
|
updatePrompt: (data: any): Promise<any> =>
|
||||||
|
apiClient.post(endpoints.USER.UPDATE_PROMPT, data),
|
||||||
|
getSinglePrompt: (id: string): Promise<any> =>
|
||||||
|
apiClient.get(endpoints.USER.SINGLE_PROMPT(id)),
|
||||||
|
deletePath: (docPath: string): Promise<any> =>
|
||||||
|
apiClient.get(endpoints.USER.DELETE_PATH(docPath)),
|
||||||
|
getTaskStatus: (task_id: string): Promise<any> =>
|
||||||
|
apiClient.get(endpoints.USER.TASK_STATUS(task_id)),
|
||||||
|
};
|
||||||
|
|
||||||
|
export default userService;
|
||||||
@@ -1,9 +1,22 @@
|
|||||||
import { Fragment, useEffect, useRef, useState } from 'react';
|
import { Fragment, useEffect, useRef, useState } from 'react';
|
||||||
|
import { useTranslation } from 'react-i18next';
|
||||||
import { useDispatch, useSelector } from 'react-redux';
|
import { useDispatch, useSelector } from 'react-redux';
|
||||||
import { useDarkTheme } from '../hooks';
|
|
||||||
|
import ArrowDown from '../assets/arrow-down.svg';
|
||||||
|
import Send from '../assets/send.svg';
|
||||||
|
import SendDark from '../assets/send_dark.svg';
|
||||||
|
import ShareIcon from '../assets/share.svg';
|
||||||
|
import SpinnerDark from '../assets/spinner-dark.svg';
|
||||||
|
import Spinner from '../assets/spinner.svg';
|
||||||
|
import RetryIcon from '../components/RetryIcon';
|
||||||
import Hero from '../Hero';
|
import Hero from '../Hero';
|
||||||
|
import { useDarkTheme } from '../hooks';
|
||||||
|
import { ShareConversationModal } from '../modals/ShareConversationModal';
|
||||||
|
import { selectConversationId } from '../preferences/preferenceSlice';
|
||||||
import { AppDispatch } from '../store';
|
import { AppDispatch } from '../store';
|
||||||
import ConversationBubble from './ConversationBubble';
|
import ConversationBubble from './ConversationBubble';
|
||||||
|
import { handleSendFeedback } from './conversationHandlers';
|
||||||
|
import { FEEDBACK, Query } from './conversationModels';
|
||||||
import {
|
import {
|
||||||
addQuery,
|
addQuery,
|
||||||
fetchAnswer,
|
fetchAnswer,
|
||||||
@@ -11,18 +24,6 @@ import {
|
|||||||
selectStatus,
|
selectStatus,
|
||||||
updateQuery,
|
updateQuery,
|
||||||
} from './conversationSlice';
|
} from './conversationSlice';
|
||||||
import { selectConversationId } from '../preferences/preferenceSlice';
|
|
||||||
import Send from './../assets/send.svg';
|
|
||||||
import SendDark from './../assets/send_dark.svg';
|
|
||||||
import Spinner from './../assets/spinner.svg';
|
|
||||||
import SpinnerDark from './../assets/spinner-dark.svg';
|
|
||||||
import { FEEDBACK, Query } from './conversationModels';
|
|
||||||
import { sendFeedback } from './conversationApi';
|
|
||||||
import { useTranslation } from 'react-i18next';
|
|
||||||
import ArrowDown from './../assets/arrow-down.svg';
|
|
||||||
import RetryIcon from '../components/RetryIcon';
|
|
||||||
import ShareIcon from '../assets/share.svg';
|
|
||||||
import { ShareConversationModal } from '../modals/ShareConversationModal';
|
|
||||||
|
|
||||||
export default function Conversation() {
|
export default function Conversation() {
|
||||||
const queries = useSelector(selectQueries);
|
const queries = useSelector(selectQueries);
|
||||||
@@ -112,7 +113,7 @@ export default function Conversation() {
|
|||||||
const handleFeedback = (query: Query, feedback: FEEDBACK, index: number) => {
|
const handleFeedback = (query: Query, feedback: FEEDBACK, index: number) => {
|
||||||
const prevFeedback = query.feedback;
|
const prevFeedback = query.feedback;
|
||||||
dispatch(updateQuery({ index, query: { feedback } }));
|
dispatch(updateQuery({ index, query: { feedback } }));
|
||||||
sendFeedback(query.prompt, query.response!, feedback).catch(() =>
|
handleSendFeedback(query.prompt, query.response!, feedback).catch(() =>
|
||||||
dispatch(updateQuery({ index, query: { feedback: prevFeedback } })),
|
dispatch(updateQuery({ index, query: { feedback: prevFeedback } })),
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -1,8 +1,9 @@
|
|||||||
import { useEffect, useRef } from 'react';
|
|
||||||
import { useParams } from 'react-router-dom';
|
|
||||||
import { useNavigate } from 'react-router-dom';
|
|
||||||
import { Query } from './conversationModels';
|
import { Query } from './conversationModels';
|
||||||
|
import { Fragment, useEffect, useRef } from 'react';
|
||||||
import { useTranslation } from 'react-i18next';
|
import { useTranslation } from 'react-i18next';
|
||||||
|
import { useNavigate, useParams } from 'react-router-dom';
|
||||||
|
|
||||||
|
import conversationService from '../api/services/conversationService';
|
||||||
import ConversationBubble from './ConversationBubble';
|
import ConversationBubble from './ConversationBubble';
|
||||||
import Send from '../assets/send.svg';
|
import Send from '../assets/send.svg';
|
||||||
import Spinner from '../assets/spinner.svg';
|
import Spinner from '../assets/spinner.svg';
|
||||||
@@ -15,7 +16,6 @@ import {
|
|||||||
selectQueries,
|
selectQueries,
|
||||||
} from './sharedConversationSlice';
|
} from './sharedConversationSlice';
|
||||||
import { useSelector } from 'react-redux';
|
import { useSelector } from 'react-redux';
|
||||||
import { Fragment } from 'react';
|
|
||||||
const apiHost = import.meta.env.VITE_API_HOST || 'https://docsapi.arc53.com';
|
const apiHost = import.meta.env.VITE_API_HOST || 'https://docsapi.arc53.com';
|
||||||
const SharedConversation = () => {
|
const SharedConversation = () => {
|
||||||
const params = useParams();
|
const params = useParams();
|
||||||
@@ -64,7 +64,8 @@ const SharedConversation = () => {
|
|||||||
}
|
}
|
||||||
const fetchQueris = () => {
|
const fetchQueris = () => {
|
||||||
identifier &&
|
identifier &&
|
||||||
fetch(`${apiHost}/api/shared_conversation/${identifier}`)
|
conversationService
|
||||||
|
.getSharedConversation(identifier || '')
|
||||||
.then((res) => {
|
.then((res) => {
|
||||||
if (res.status === 404 || res.status === 400)
|
if (res.status === 404 || res.status === 400)
|
||||||
navigate('/pagenotfound');
|
navigate('/pagenotfound');
|
||||||
@@ -200,5 +201,3 @@ const SharedConversation = () => {
|
|||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
export default SharedConversation;
|
|
||||||
|
|||||||
@@ -1,11 +1,9 @@
|
|||||||
import { Answer, FEEDBACK } from './conversationModels';
|
import conversationService from '../api/services/conversationService';
|
||||||
import { Doc } from '../preferences/preferenceApi';
|
import { Doc } from '../preferences/preferenceApi';
|
||||||
|
import { Answer, FEEDBACK } from './conversationModels';
|
||||||
const apiHost = import.meta.env.VITE_API_HOST || 'https://docsapi.arc53.com';
|
|
||||||
|
|
||||||
function getDocPath(selectedDocs: Doc | null): string {
|
function getDocPath(selectedDocs: Doc | null): string {
|
||||||
let docPath = 'default';
|
let docPath = 'default';
|
||||||
|
|
||||||
if (selectedDocs) {
|
if (selectedDocs) {
|
||||||
let namePath = selectedDocs.name;
|
let namePath = selectedDocs.name;
|
||||||
if (selectedDocs.language === namePath) {
|
if (selectedDocs.language === namePath) {
|
||||||
@@ -27,10 +25,10 @@ function getDocPath(selectedDocs: Doc | null): string {
|
|||||||
docPath = selectedDocs.docLink;
|
docPath = selectedDocs.docLink;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return docPath;
|
return docPath;
|
||||||
}
|
}
|
||||||
export function fetchAnswerApi(
|
|
||||||
|
export function handleFetchAnswer(
|
||||||
question: string,
|
question: string,
|
||||||
signal: AbortSignal,
|
signal: AbortSignal,
|
||||||
selectedDocs: Doc | null,
|
selectedDocs: Doc | null,
|
||||||
@@ -57,27 +55,22 @@ export function fetchAnswerApi(
|
|||||||
}
|
}
|
||||||
> {
|
> {
|
||||||
const docPath = getDocPath(selectedDocs);
|
const docPath = getDocPath(selectedDocs);
|
||||||
//in history array remove all keys except prompt and response
|
|
||||||
history = history.map((item) => {
|
history = history.map((item) => {
|
||||||
return { prompt: item.prompt, response: item.response };
|
return { prompt: item.prompt, response: item.response };
|
||||||
});
|
});
|
||||||
|
return conversationService
|
||||||
return fetch(apiHost + '/api/answer', {
|
.answer(
|
||||||
method: 'POST',
|
{
|
||||||
headers: {
|
question: question,
|
||||||
'Content-Type': 'application/json',
|
history: history,
|
||||||
},
|
active_docs: docPath,
|
||||||
body: JSON.stringify({
|
conversation_id: conversationId,
|
||||||
question: question,
|
prompt_id: promptId,
|
||||||
history: history,
|
chunks: chunks,
|
||||||
active_docs: docPath,
|
token_limit: token_limit,
|
||||||
conversation_id: conversationId,
|
},
|
||||||
prompt_id: promptId,
|
signal,
|
||||||
chunks: chunks,
|
)
|
||||||
token_limit: token_limit,
|
|
||||||
}),
|
|
||||||
signal,
|
|
||||||
})
|
|
||||||
.then((response) => {
|
.then((response) => {
|
||||||
if (response.ok) {
|
if (response.ok) {
|
||||||
return response.json();
|
return response.json();
|
||||||
@@ -97,7 +90,7 @@ export function fetchAnswerApi(
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
export function fetchAnswerSteaming(
|
export function handleFetchAnswerSteaming(
|
||||||
question: string,
|
question: string,
|
||||||
signal: AbortSignal,
|
signal: AbortSignal,
|
||||||
selectedDocs: Doc | null,
|
selectedDocs: Doc | null,
|
||||||
@@ -109,29 +102,23 @@ export function fetchAnswerSteaming(
|
|||||||
onEvent: (event: MessageEvent) => void,
|
onEvent: (event: MessageEvent) => void,
|
||||||
): Promise<Answer> {
|
): Promise<Answer> {
|
||||||
const docPath = getDocPath(selectedDocs);
|
const docPath = getDocPath(selectedDocs);
|
||||||
|
|
||||||
history = history.map((item) => {
|
history = history.map((item) => {
|
||||||
return { prompt: item.prompt, response: item.response };
|
return { prompt: item.prompt, response: item.response };
|
||||||
});
|
});
|
||||||
|
|
||||||
return new Promise<Answer>((resolve, reject) => {
|
return new Promise<Answer>((resolve, reject) => {
|
||||||
const body = {
|
conversationService
|
||||||
question: question,
|
.answerStream(
|
||||||
active_docs: docPath,
|
{
|
||||||
history: JSON.stringify(history),
|
question: question,
|
||||||
conversation_id: conversationId,
|
active_docs: docPath,
|
||||||
prompt_id: promptId,
|
history: JSON.stringify(history),
|
||||||
chunks: chunks,
|
conversation_id: conversationId,
|
||||||
token_limit: token_limit,
|
prompt_id: promptId,
|
||||||
};
|
chunks: chunks,
|
||||||
fetch(apiHost + '/stream', {
|
token_limit: token_limit,
|
||||||
method: 'POST',
|
},
|
||||||
headers: {
|
signal,
|
||||||
'Content-Type': 'application/json',
|
)
|
||||||
},
|
|
||||||
body: JSON.stringify(body),
|
|
||||||
signal,
|
|
||||||
})
|
|
||||||
.then((response) => {
|
.then((response) => {
|
||||||
if (!response.body) throw Error('No response body');
|
if (!response.body) throw Error('No response body');
|
||||||
|
|
||||||
@@ -179,7 +166,8 @@ export function fetchAnswerSteaming(
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
export function searchEndpoint(
|
|
||||||
|
export function handleSearch(
|
||||||
question: string,
|
question: string,
|
||||||
selectedDocs: Doc | null,
|
selectedDocs: Doc | null,
|
||||||
conversation_id: string | null,
|
conversation_id: string | null,
|
||||||
@@ -188,50 +176,40 @@ export function searchEndpoint(
|
|||||||
token_limit: number,
|
token_limit: number,
|
||||||
) {
|
) {
|
||||||
const docPath = getDocPath(selectedDocs);
|
const docPath = getDocPath(selectedDocs);
|
||||||
|
return conversationService
|
||||||
const body = {
|
.search({
|
||||||
question: question,
|
question: question,
|
||||||
active_docs: docPath,
|
active_docs: docPath,
|
||||||
conversation_id,
|
conversation_id,
|
||||||
history,
|
history,
|
||||||
chunks: chunks,
|
chunks: chunks,
|
||||||
token_limit: token_limit,
|
token_limit: token_limit,
|
||||||
};
|
})
|
||||||
return fetch(`${apiHost}/api/search`, {
|
|
||||||
method: 'POST',
|
|
||||||
headers: {
|
|
||||||
'Content-Type': 'application/json',
|
|
||||||
},
|
|
||||||
body: JSON.stringify(body),
|
|
||||||
})
|
|
||||||
.then((response) => response.json())
|
.then((response) => response.json())
|
||||||
.then((data) => {
|
.then((data) => {
|
||||||
return data;
|
return data;
|
||||||
})
|
})
|
||||||
.catch((err) => console.log(err));
|
.catch((err) => console.log(err));
|
||||||
}
|
}
|
||||||
export function sendFeedback(
|
|
||||||
|
export function handleSendFeedback(
|
||||||
prompt: string,
|
prompt: string,
|
||||||
response: string,
|
response: string,
|
||||||
feedback: FEEDBACK,
|
feedback: FEEDBACK,
|
||||||
) {
|
) {
|
||||||
return fetch(`${apiHost}/api/feedback`, {
|
return conversationService
|
||||||
method: 'POST',
|
.feedback({
|
||||||
headers: {
|
|
||||||
'Content-Type': 'application/json',
|
|
||||||
},
|
|
||||||
body: JSON.stringify({
|
|
||||||
question: prompt,
|
question: prompt,
|
||||||
answer: response,
|
answer: response,
|
||||||
feedback: feedback,
|
feedback: feedback,
|
||||||
}),
|
})
|
||||||
}).then((response) => {
|
.then((response) => {
|
||||||
if (response.ok) {
|
if (response.ok) {
|
||||||
return Promise.resolve();
|
return Promise.resolve();
|
||||||
} else {
|
} else {
|
||||||
return Promise.reject();
|
return Promise.reject();
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
export function fetchSharedAnswerSteaming( //for shared conversations
|
export function fetchSharedAnswerSteaming( //for shared conversations
|
||||||
@@ -1,10 +1,14 @@
|
|||||||
import { createAsyncThunk, createSlice, PayloadAction } from '@reduxjs/toolkit';
|
import { createAsyncThunk, createSlice, PayloadAction } from '@reduxjs/toolkit';
|
||||||
import store from '../store';
|
|
||||||
import { fetchAnswerApi, fetchAnswerSteaming } from './conversationApi';
|
|
||||||
import { searchEndpoint } from './conversationApi';
|
|
||||||
import { Answer, ConversationState, Query, Status } from './conversationModels';
|
|
||||||
import { getConversations } from '../preferences/preferenceApi';
|
import { getConversations } from '../preferences/preferenceApi';
|
||||||
import { setConversations } from '../preferences/preferenceSlice';
|
import { setConversations } from '../preferences/preferenceSlice';
|
||||||
|
import store from '../store';
|
||||||
|
import {
|
||||||
|
handleFetchAnswer,
|
||||||
|
handleFetchAnswerSteaming,
|
||||||
|
handleSearch,
|
||||||
|
} from './conversationHandlers';
|
||||||
|
import { Answer, ConversationState, Query, Status } from './conversationModels';
|
||||||
|
|
||||||
const initialState: ConversationState = {
|
const initialState: ConversationState = {
|
||||||
queries: [],
|
queries: [],
|
||||||
@@ -20,7 +24,7 @@ export const fetchAnswer = createAsyncThunk<Answer, { question: string }>(
|
|||||||
const state = getState() as RootState;
|
const state = getState() as RootState;
|
||||||
if (state.preference) {
|
if (state.preference) {
|
||||||
if (API_STREAMING) {
|
if (API_STREAMING) {
|
||||||
await fetchAnswerSteaming(
|
await handleFetchAnswerSteaming(
|
||||||
question,
|
question,
|
||||||
signal,
|
signal,
|
||||||
state.preference.selectedDocs!,
|
state.preference.selectedDocs!,
|
||||||
@@ -45,7 +49,7 @@ export const fetchAnswer = createAsyncThunk<Answer, { question: string }>(
|
|||||||
console.error('Failed to fetch conversations: ', error);
|
console.error('Failed to fetch conversations: ', error);
|
||||||
});
|
});
|
||||||
|
|
||||||
searchEndpoint(
|
handleSearch(
|
||||||
//search for sources post streaming
|
//search for sources post streaming
|
||||||
question,
|
question,
|
||||||
state.preference.selectedDocs!,
|
state.preference.selectedDocs!,
|
||||||
@@ -89,7 +93,7 @@ export const fetchAnswer = createAsyncThunk<Answer, { question: string }>(
|
|||||||
},
|
},
|
||||||
);
|
);
|
||||||
} else {
|
} else {
|
||||||
const answer = await fetchAnswerApi(
|
const answer = await handleFetchAnswer(
|
||||||
question,
|
question,
|
||||||
signal,
|
signal,
|
||||||
state.preference.selectedDocs!,
|
state.preference.selectedDocs!,
|
||||||
|
|||||||
162
frontend/src/modals/CreateAPIKeyModal.tsx
Normal file
162
frontend/src/modals/CreateAPIKeyModal.tsx
Normal file
@@ -0,0 +1,162 @@
|
|||||||
|
import React from 'react';
|
||||||
|
import { useTranslation } from 'react-i18next';
|
||||||
|
import { useSelector } from 'react-redux';
|
||||||
|
|
||||||
|
import userService from '../api/services/userService';
|
||||||
|
import Exit from '../assets/exit.svg';
|
||||||
|
import Dropdown from '../components/Dropdown';
|
||||||
|
import Input from '../components/Input';
|
||||||
|
import { CreateAPIKeyModalProps, Doc } from '../models/misc';
|
||||||
|
import { selectSourceDocs } from '../preferences/preferenceSlice';
|
||||||
|
|
||||||
|
const embeddingsName =
|
||||||
|
import.meta.env.VITE_EMBEDDINGS_NAME ||
|
||||||
|
'huggingface_sentence-transformers/all-mpnet-base-v2';
|
||||||
|
|
||||||
|
export default function CreateAPIKeyModal({
|
||||||
|
close,
|
||||||
|
createAPIKey,
|
||||||
|
}: CreateAPIKeyModalProps) {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
const docs = useSelector(selectSourceDocs);
|
||||||
|
|
||||||
|
const [APIKeyName, setAPIKeyName] = React.useState<string>('');
|
||||||
|
const [sourcePath, setSourcePath] = React.useState<{
|
||||||
|
label: string;
|
||||||
|
value: string;
|
||||||
|
} | null>(null);
|
||||||
|
const [prompt, setPrompt] = React.useState<{
|
||||||
|
name: string;
|
||||||
|
id: string;
|
||||||
|
type: string;
|
||||||
|
} | null>(null);
|
||||||
|
const [activePrompts, setActivePrompts] = React.useState<
|
||||||
|
{ name: string; id: string; type: string }[]
|
||||||
|
>([]);
|
||||||
|
const [chunk, setChunk] = React.useState<string>('2');
|
||||||
|
const chunkOptions = ['0', '2', '4', '6', '8', '10'];
|
||||||
|
|
||||||
|
const extractDocPaths = () =>
|
||||||
|
docs
|
||||||
|
? docs
|
||||||
|
.filter((doc) => doc.model === embeddingsName)
|
||||||
|
.map((doc: Doc) => {
|
||||||
|
let namePath = doc.name;
|
||||||
|
if (doc.language === namePath) {
|
||||||
|
namePath = '.project';
|
||||||
|
}
|
||||||
|
let docPath = 'default';
|
||||||
|
if (doc.location === 'local') {
|
||||||
|
docPath = 'local' + '/' + doc.name + '/';
|
||||||
|
} else if (doc.location === 'remote') {
|
||||||
|
docPath =
|
||||||
|
doc.language +
|
||||||
|
'/' +
|
||||||
|
namePath +
|
||||||
|
'/' +
|
||||||
|
doc.version +
|
||||||
|
'/' +
|
||||||
|
doc.model +
|
||||||
|
'/';
|
||||||
|
}
|
||||||
|
return {
|
||||||
|
label: doc.name,
|
||||||
|
value: docPath,
|
||||||
|
};
|
||||||
|
})
|
||||||
|
: [];
|
||||||
|
|
||||||
|
React.useEffect(() => {
|
||||||
|
const handleFetchPrompts = async () => {
|
||||||
|
try {
|
||||||
|
const response = await userService.getPrompts();
|
||||||
|
if (!response.ok) {
|
||||||
|
throw new Error('Failed to fetch prompts');
|
||||||
|
}
|
||||||
|
const promptsData = await response.json();
|
||||||
|
setActivePrompts(promptsData);
|
||||||
|
} catch (error) {
|
||||||
|
console.error(error);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
handleFetchPrompts();
|
||||||
|
}, []);
|
||||||
|
return (
|
||||||
|
<div className="fixed top-0 left-0 z-30 flex h-screen w-screen items-center justify-center bg-gray-alpha bg-opacity-50">
|
||||||
|
<div className="relative w-11/12 rounded-2xl bg-white p-10 dark:bg-outer-space sm:w-[512px]">
|
||||||
|
<button className="absolute top-3 right-4 m-2 w-3" onClick={close}>
|
||||||
|
<img className="filter dark:invert" src={Exit} />
|
||||||
|
</button>
|
||||||
|
<div className="mb-6">
|
||||||
|
<span className="text-xl text-jet dark:text-bright-gray">
|
||||||
|
{t('modals.createAPIKey.label')}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
<div className="relative mt-5 mb-4">
|
||||||
|
<span className="absolute left-2 -top-2 bg-white px-2 text-xs text-gray-4000 dark:bg-outer-space dark:text-silver">
|
||||||
|
{t('modals.createAPIKey.apiKeyName')}
|
||||||
|
</span>
|
||||||
|
<Input
|
||||||
|
type="text"
|
||||||
|
className="rounded-md"
|
||||||
|
value={APIKeyName}
|
||||||
|
onChange={(e) => setAPIKeyName(e.target.value)}
|
||||||
|
></Input>
|
||||||
|
</div>
|
||||||
|
<div className="my-4">
|
||||||
|
<Dropdown
|
||||||
|
placeholder={t('modals.createAPIKey.sourceDoc')}
|
||||||
|
selectedValue={sourcePath}
|
||||||
|
onSelect={(selection: { label: string; value: string }) =>
|
||||||
|
setSourcePath(selection)
|
||||||
|
}
|
||||||
|
options={extractDocPaths()}
|
||||||
|
size="w-full"
|
||||||
|
rounded="xl"
|
||||||
|
border="border"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div className="my-4">
|
||||||
|
<Dropdown
|
||||||
|
options={activePrompts}
|
||||||
|
selectedValue={prompt ? prompt.name : null}
|
||||||
|
placeholder={t('modals.createAPIKey.prompt')}
|
||||||
|
onSelect={(value: { name: string; id: string; type: string }) =>
|
||||||
|
setPrompt(value)
|
||||||
|
}
|
||||||
|
size="w-full"
|
||||||
|
border="border"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div className="my-4">
|
||||||
|
<p className="mb-2 ml-2 font-semibold text-jet dark:text-bright-gray">
|
||||||
|
{t('modals.createAPIKey.chunks')}
|
||||||
|
</p>
|
||||||
|
<Dropdown
|
||||||
|
options={chunkOptions}
|
||||||
|
selectedValue={chunk}
|
||||||
|
onSelect={(value: string) => setChunk(value)}
|
||||||
|
size="w-full"
|
||||||
|
border="border"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<button
|
||||||
|
disabled={!sourcePath || APIKeyName.length === 0 || !prompt}
|
||||||
|
onClick={() =>
|
||||||
|
sourcePath &&
|
||||||
|
prompt &&
|
||||||
|
createAPIKey({
|
||||||
|
name: APIKeyName,
|
||||||
|
source: sourcePath.value,
|
||||||
|
prompt_id: prompt.id,
|
||||||
|
chunks: chunk,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
className="float-right mt-4 rounded-full bg-purple-30 px-5 py-2 text-sm text-white hover:bg-[#6F3FD1] disabled:opacity-50"
|
||||||
|
>
|
||||||
|
{t('modals.createAPIKey.create')}
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
52
frontend/src/modals/SaveAPIKeyModal.tsx
Normal file
52
frontend/src/modals/SaveAPIKeyModal.tsx
Normal file
@@ -0,0 +1,52 @@
|
|||||||
|
import React from 'react';
|
||||||
|
import { useTranslation } from 'react-i18next';
|
||||||
|
|
||||||
|
import Exit from '../assets/exit.svg';
|
||||||
|
import { SaveAPIKeyModalProps } from '../models/misc';
|
||||||
|
|
||||||
|
export default function SaveAPIKeyModal({
|
||||||
|
apiKey,
|
||||||
|
close,
|
||||||
|
}: SaveAPIKeyModalProps) {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
const [isCopied, setIsCopied] = React.useState(false);
|
||||||
|
|
||||||
|
const handleCopyKey = () => {
|
||||||
|
navigator.clipboard.writeText(apiKey);
|
||||||
|
setIsCopied(true);
|
||||||
|
};
|
||||||
|
return (
|
||||||
|
<div className="fixed top-0 left-0 z-30 flex h-screen w-screen items-center justify-center bg-gray-alpha bg-opacity-50">
|
||||||
|
<div className="relative w-11/12 rounded-3xl bg-white px-6 py-8 dark:bg-outer-space dark:text-bright-gray sm:w-[512px]">
|
||||||
|
<button className="absolute top-3 right-4 m-2 w-3" onClick={close}>
|
||||||
|
<img className="filter dark:invert" src={Exit} />
|
||||||
|
</button>
|
||||||
|
<h1 className="my-0 text-xl font-medium">
|
||||||
|
{' '}
|
||||||
|
{t('modals.saveKey.note')}
|
||||||
|
</h1>
|
||||||
|
<h3 className="text-sm font-normal text-outer-space">
|
||||||
|
{t('modals.saveKey.disclaimer')}
|
||||||
|
</h3>
|
||||||
|
<div className="flex justify-between py-2">
|
||||||
|
<div>
|
||||||
|
<h2 className="text-base font-semibold">API Key</h2>
|
||||||
|
<span className="text-sm font-normal leading-7 ">{apiKey}</span>
|
||||||
|
</div>
|
||||||
|
<button
|
||||||
|
className="my-1 h-10 w-20 rounded-full border border-solid border-purple-30 p-2 text-sm text-purple-30 hover:bg-purple-30 hover:text-white"
|
||||||
|
onClick={handleCopyKey}
|
||||||
|
>
|
||||||
|
{isCopied ? t('modals.saveKey.copied') : t('modals.saveKey.copy')}
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
<button
|
||||||
|
onClick={close}
|
||||||
|
className="rounded-full bg-philippine-yellow px-4 py-3 font-medium text-black hover:bg-[#E6B91A]"
|
||||||
|
>
|
||||||
|
{t('modals.saveKey.confirm')}
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -18,6 +18,8 @@ const embeddingsName =
|
|||||||
|
|
||||||
type StatusType = 'loading' | 'idle' | 'fetched' | 'failed';
|
type StatusType = 'loading' | 'idle' | 'fetched' | 'failed';
|
||||||
|
|
||||||
|
import conversationService from '../api/services/conversationService';
|
||||||
|
|
||||||
export const ShareConversationModal = ({
|
export const ShareConversationModal = ({
|
||||||
close,
|
close,
|
||||||
conversationId,
|
conversationId,
|
||||||
@@ -100,13 +102,8 @@ export const ShareConversationModal = ({
|
|||||||
payload.prompt_id = selectedPrompt.id;
|
payload.prompt_id = selectedPrompt.id;
|
||||||
sourcePath && (payload.source = sourcePath.value);
|
sourcePath && (payload.source = sourcePath.value);
|
||||||
}
|
}
|
||||||
fetch(`${apiHost}/api/share?isPromptable=${isPromptable}`, {
|
conversationService
|
||||||
method: 'POST',
|
.shareConversation(isPromptable, payload)
|
||||||
headers: {
|
|
||||||
'Content-Type': 'application/json',
|
|
||||||
},
|
|
||||||
body: JSON.stringify(payload),
|
|
||||||
})
|
|
||||||
.then((res) => {
|
.then((res) => {
|
||||||
console.log(res.status);
|
console.log(res.status);
|
||||||
return res.json();
|
return res.json();
|
||||||
|
|||||||
@@ -21,7 +21,6 @@ export type PromptProps = {
|
|||||||
selectedPrompt: { name: string; id: string; type: string };
|
selectedPrompt: { name: string; id: string; type: string };
|
||||||
onSelectPrompt: (name: string, id: string, type: string) => void;
|
onSelectPrompt: (name: string, id: string, type: string) => void;
|
||||||
setPrompts: (prompts: { name: string; id: string; type: string }[]) => void;
|
setPrompts: (prompts: { name: string; id: string; type: string }[]) => void;
|
||||||
apiHost: string;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
export type DocumentsProps = {
|
export type DocumentsProps = {
|
||||||
|
|||||||
@@ -1,3 +1,6 @@
|
|||||||
|
import conversationService from '../api/services/conversationService';
|
||||||
|
import userService from '../api/services/userService';
|
||||||
|
|
||||||
// not all properties in Doc are going to be present. Make some optional
|
// not all properties in Doc are going to be present. Make some optional
|
||||||
export type Doc = {
|
export type Doc = {
|
||||||
location: string;
|
location: string;
|
||||||
@@ -14,10 +17,7 @@ export type Doc = {
|
|||||||
//Fetches all JSON objects from the source. We only use the objects with the "model" property in SelectDocsModal.tsx. Hopefully can clean up the source file later.
|
//Fetches all JSON objects from the source. We only use the objects with the "model" property in SelectDocsModal.tsx. Hopefully can clean up the source file later.
|
||||||
export async function getDocs(): Promise<Doc[] | null> {
|
export async function getDocs(): Promise<Doc[] | null> {
|
||||||
try {
|
try {
|
||||||
const apiHost =
|
const response = await userService.getDocs();
|
||||||
import.meta.env.VITE_API_HOST || 'https://docsapi.arc53.com';
|
|
||||||
|
|
||||||
const response = await fetch(apiHost + '/api/combine');
|
|
||||||
const data = await response.json();
|
const data = await response.json();
|
||||||
|
|
||||||
const docs: Doc[] = [];
|
const docs: Doc[] = [];
|
||||||
@@ -37,10 +37,7 @@ export async function getConversations(): Promise<
|
|||||||
{ name: string; id: string }[] | null
|
{ name: string; id: string }[] | null
|
||||||
> {
|
> {
|
||||||
try {
|
try {
|
||||||
const apiHost =
|
const response = await conversationService.getConversations();
|
||||||
import.meta.env.VITE_API_HOST || 'https://docsapi.arc53.com';
|
|
||||||
|
|
||||||
const response = await fetch(apiHost + '/api/get_conversations');
|
|
||||||
const data = await response.json();
|
const data = await response.json();
|
||||||
|
|
||||||
const conversations: { name: string; id: string }[] = [];
|
const conversations: { name: string; id: string }[] = [];
|
||||||
@@ -93,14 +90,9 @@ export function setLocalRecentDocs(doc: Doc): void {
|
|||||||
docPath =
|
docPath =
|
||||||
doc.language + '/' + namePath + '/' + doc.version + '/' + doc.model + '/';
|
doc.language + '/' + namePath + '/' + doc.version + '/' + doc.model + '/';
|
||||||
}
|
}
|
||||||
const apiHost = import.meta.env.VITE_API_HOST || 'https://docsapi.arc53.com';
|
userService
|
||||||
fetch(apiHost + '/api/docs_check', {
|
.checkDocs({
|
||||||
method: 'POST',
|
|
||||||
headers: {
|
|
||||||
'Content-Type': 'application/json',
|
|
||||||
},
|
|
||||||
body: JSON.stringify({
|
|
||||||
docs: docPath,
|
docs: docPath,
|
||||||
}),
|
})
|
||||||
}).then((response) => response.json());
|
.then((response) => response.json());
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,22 +1,12 @@
|
|||||||
import React from 'react';
|
import React from 'react';
|
||||||
import { useSelector } from 'react-redux';
|
|
||||||
import Dropdown from '../components/Dropdown';
|
|
||||||
import {
|
|
||||||
Doc,
|
|
||||||
CreateAPIKeyModalProps,
|
|
||||||
SaveAPIKeyModalProps,
|
|
||||||
} from '../models/misc';
|
|
||||||
import { selectSourceDocs } from '../preferences/preferenceSlice';
|
|
||||||
import Exit from '../assets/exit.svg';
|
|
||||||
import Trash from '../assets/trash.svg';
|
|
||||||
import { useTranslation } from 'react-i18next';
|
import { useTranslation } from 'react-i18next';
|
||||||
import Input from '../components/Input';
|
|
||||||
const apiHost = import.meta.env.VITE_API_HOST || 'https://docsapi.arc53.com';
|
|
||||||
const embeddingsName =
|
|
||||||
import.meta.env.VITE_EMBEDDINGS_NAME ||
|
|
||||||
'huggingface_sentence-transformers/all-mpnet-base-v2';
|
|
||||||
|
|
||||||
const APIKeys: React.FC = () => {
|
import userService from '../api/services/userService';
|
||||||
|
import Trash from '../assets/trash.svg';
|
||||||
|
import CreateAPIKeyModal from '../modals/CreateAPIKeyModal';
|
||||||
|
import SaveAPIKeyModal from '../modals/SaveAPIKeyModal';
|
||||||
|
|
||||||
|
export default function APIKeys() {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const [isCreateModalOpen, setCreateModal] = React.useState(false);
|
const [isCreateModalOpen, setCreateModal] = React.useState(false);
|
||||||
const [isSaveKeyModalOpen, setSaveKeyModal] = React.useState(false);
|
const [isSaveKeyModalOpen, setSaveKeyModal] = React.useState(false);
|
||||||
@@ -24,14 +14,23 @@ const APIKeys: React.FC = () => {
|
|||||||
const [apiKeys, setApiKeys] = React.useState<
|
const [apiKeys, setApiKeys] = React.useState<
|
||||||
{ name: string; key: string; source: string; id: string }[]
|
{ name: string; key: string; source: string; id: string }[]
|
||||||
>([]);
|
>([]);
|
||||||
|
|
||||||
|
const handleFetchKeys = async () => {
|
||||||
|
try {
|
||||||
|
const response = await userService.getAPIKeys();
|
||||||
|
if (!response.ok) {
|
||||||
|
throw new Error('Failed to fetch API Keys');
|
||||||
|
}
|
||||||
|
const apiKeys = await response.json();
|
||||||
|
setApiKeys(apiKeys);
|
||||||
|
} catch (error) {
|
||||||
|
console.log(error);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
const handleDeleteKey = (id: string) => {
|
const handleDeleteKey = (id: string) => {
|
||||||
fetch(`${apiHost}/api/delete_api_key`, {
|
userService
|
||||||
method: 'POST',
|
.deleteAPIKey({ id })
|
||||||
headers: {
|
|
||||||
'Content-Type': 'application/json',
|
|
||||||
},
|
|
||||||
body: JSON.stringify({ id }),
|
|
||||||
})
|
|
||||||
.then((response) => {
|
.then((response) => {
|
||||||
if (!response.ok) {
|
if (!response.ok) {
|
||||||
throw new Error('Failed to delete API Key');
|
throw new Error('Failed to delete API Key');
|
||||||
@@ -46,34 +45,15 @@ const APIKeys: React.FC = () => {
|
|||||||
console.error(error);
|
console.error(error);
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
React.useEffect(() => {
|
|
||||||
fetchAPIKeys();
|
const handleCreateKey = (payload: {
|
||||||
}, []);
|
|
||||||
const fetchAPIKeys = async () => {
|
|
||||||
try {
|
|
||||||
const response = await fetch(`${apiHost}/api/get_api_keys`);
|
|
||||||
if (!response.ok) {
|
|
||||||
throw new Error('Failed to fetch API Keys');
|
|
||||||
}
|
|
||||||
const apiKeys = await response.json();
|
|
||||||
setApiKeys(apiKeys);
|
|
||||||
} catch (error) {
|
|
||||||
console.log(error);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
const createAPIKey = (payload: {
|
|
||||||
name: string;
|
name: string;
|
||||||
source: string;
|
source: string;
|
||||||
prompt_id: string;
|
prompt_id: string;
|
||||||
chunks: string;
|
chunks: string;
|
||||||
}) => {
|
}) => {
|
||||||
fetch(`${apiHost}/api/create_api_key`, {
|
userService
|
||||||
method: 'POST',
|
.createAPIKey(payload)
|
||||||
headers: {
|
|
||||||
'Content-Type': 'application/json',
|
|
||||||
},
|
|
||||||
body: JSON.stringify(payload),
|
|
||||||
})
|
|
||||||
.then((response) => {
|
.then((response) => {
|
||||||
if (!response.ok) {
|
if (!response.ok) {
|
||||||
throw new Error('Failed to create API Key');
|
throw new Error('Failed to create API Key');
|
||||||
@@ -85,12 +65,16 @@ const APIKeys: React.FC = () => {
|
|||||||
setCreateModal(false);
|
setCreateModal(false);
|
||||||
setNewKey(data.key);
|
setNewKey(data.key);
|
||||||
setSaveKeyModal(true);
|
setSaveKeyModal(true);
|
||||||
fetchAPIKeys();
|
handleFetchKeys();
|
||||||
})
|
})
|
||||||
.catch((error) => {
|
.catch((error) => {
|
||||||
console.error(error);
|
console.error(error);
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
|
React.useEffect(() => {
|
||||||
|
handleFetchKeys();
|
||||||
|
}, []);
|
||||||
return (
|
return (
|
||||||
<div className="mt-8">
|
<div className="mt-8">
|
||||||
<div className="flex w-full flex-col lg:w-max">
|
<div className="flex w-full flex-col lg:w-max">
|
||||||
@@ -104,8 +88,8 @@ const APIKeys: React.FC = () => {
|
|||||||
</div>
|
</div>
|
||||||
{isCreateModalOpen && (
|
{isCreateModalOpen && (
|
||||||
<CreateAPIKeyModal
|
<CreateAPIKeyModal
|
||||||
|
createAPIKey={handleCreateKey}
|
||||||
close={() => setCreateModal(false)}
|
close={() => setCreateModal(false)}
|
||||||
createAPIKey={createAPIKey}
|
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
{isSaveKeyModalOpen && (
|
{isSaveKeyModalOpen && (
|
||||||
@@ -155,192 +139,4 @@ const APIKeys: React.FC = () => {
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
};
|
}
|
||||||
|
|
||||||
const CreateAPIKeyModal: React.FC<CreateAPIKeyModalProps> = ({
|
|
||||||
close,
|
|
||||||
createAPIKey,
|
|
||||||
}) => {
|
|
||||||
const [APIKeyName, setAPIKeyName] = React.useState<string>('');
|
|
||||||
const [sourcePath, setSourcePath] = React.useState<{
|
|
||||||
label: string;
|
|
||||||
value: string;
|
|
||||||
} | null>(null);
|
|
||||||
|
|
||||||
const chunkOptions = ['0', '2', '4', '6', '8', '10'];
|
|
||||||
const [chunk, setChunk] = React.useState<string>('2');
|
|
||||||
const [activePrompts, setActivePrompts] = React.useState<
|
|
||||||
{ name: string; id: string; type: string }[]
|
|
||||||
>([]);
|
|
||||||
const [prompt, setPrompt] = React.useState<{
|
|
||||||
name: string;
|
|
||||||
id: string;
|
|
||||||
type: string;
|
|
||||||
} | null>(null);
|
|
||||||
const docs = useSelector(selectSourceDocs);
|
|
||||||
React.useEffect(() => {
|
|
||||||
const fetchPrompts = async () => {
|
|
||||||
try {
|
|
||||||
const response = await fetch(`${apiHost}/api/get_prompts`);
|
|
||||||
if (!response.ok) {
|
|
||||||
throw new Error('Failed to fetch prompts');
|
|
||||||
}
|
|
||||||
const promptsData = await response.json();
|
|
||||||
setActivePrompts(promptsData);
|
|
||||||
} catch (error) {
|
|
||||||
console.error(error);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
fetchPrompts();
|
|
||||||
}, []);
|
|
||||||
const extractDocPaths = () =>
|
|
||||||
docs
|
|
||||||
? docs
|
|
||||||
.filter((doc) => doc.model === embeddingsName)
|
|
||||||
.map((doc: Doc) => {
|
|
||||||
let namePath = doc.name;
|
|
||||||
if (doc.language === namePath) {
|
|
||||||
namePath = '.project';
|
|
||||||
}
|
|
||||||
let docPath = 'default';
|
|
||||||
if (doc.location === 'local') {
|
|
||||||
docPath = 'local' + '/' + doc.name + '/';
|
|
||||||
} else if (doc.location === 'remote') {
|
|
||||||
docPath =
|
|
||||||
doc.language +
|
|
||||||
'/' +
|
|
||||||
namePath +
|
|
||||||
'/' +
|
|
||||||
doc.version +
|
|
||||||
'/' +
|
|
||||||
doc.model +
|
|
||||||
'/';
|
|
||||||
}
|
|
||||||
return {
|
|
||||||
label: doc.name,
|
|
||||||
value: docPath,
|
|
||||||
};
|
|
||||||
})
|
|
||||||
: [];
|
|
||||||
const { t } = useTranslation();
|
|
||||||
return (
|
|
||||||
<div className="fixed top-0 left-0 z-30 flex h-screen w-screen items-center justify-center bg-gray-alpha bg-opacity-50">
|
|
||||||
<div className="relative w-11/12 rounded-2xl bg-white p-10 dark:bg-outer-space sm:w-[512px]">
|
|
||||||
<button className="absolute top-3 right-4 m-2 w-3" onClick={close}>
|
|
||||||
<img className="filter dark:invert" src={Exit} />
|
|
||||||
</button>
|
|
||||||
<div className="mb-6">
|
|
||||||
<span className="text-xl text-jet dark:text-bright-gray">
|
|
||||||
{t('modals.createAPIKey.label')}
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
<div className="relative mt-5 mb-4">
|
|
||||||
<span className="absolute left-2 -top-2 bg-white px-2 text-xs text-gray-4000 dark:bg-outer-space dark:text-silver">
|
|
||||||
{t('modals.createAPIKey.apiKeyName')}
|
|
||||||
</span>
|
|
||||||
<Input
|
|
||||||
type="text"
|
|
||||||
className="rounded-md"
|
|
||||||
value={APIKeyName}
|
|
||||||
onChange={(e) => setAPIKeyName(e.target.value)}
|
|
||||||
></Input>
|
|
||||||
</div>
|
|
||||||
<div className="my-4">
|
|
||||||
<Dropdown
|
|
||||||
placeholder={t('modals.createAPIKey.sourceDoc')}
|
|
||||||
selectedValue={sourcePath}
|
|
||||||
onSelect={(selection: { label: string; value: string }) =>
|
|
||||||
setSourcePath(selection)
|
|
||||||
}
|
|
||||||
options={extractDocPaths()}
|
|
||||||
size="w-full"
|
|
||||||
rounded="xl"
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
<div className="my-4">
|
|
||||||
<Dropdown
|
|
||||||
options={activePrompts}
|
|
||||||
selectedValue={prompt ? prompt.name : null}
|
|
||||||
placeholder={t('modals.createAPIKey.prompt')}
|
|
||||||
onSelect={(value: { name: string; id: string; type: string }) =>
|
|
||||||
setPrompt(value)
|
|
||||||
}
|
|
||||||
size="w-full"
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
<div className="my-4">
|
|
||||||
<p className="mb-2 ml-2 font-bold text-jet dark:text-bright-gray">
|
|
||||||
{t('modals.createAPIKey.chunks')}
|
|
||||||
</p>
|
|
||||||
<Dropdown
|
|
||||||
options={chunkOptions}
|
|
||||||
selectedValue={chunk}
|
|
||||||
onSelect={(value: string) => setChunk(value)}
|
|
||||||
size="w-full"
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
<button
|
|
||||||
disabled={!sourcePath || APIKeyName.length === 0 || !prompt}
|
|
||||||
onClick={() =>
|
|
||||||
sourcePath &&
|
|
||||||
prompt &&
|
|
||||||
createAPIKey({
|
|
||||||
name: APIKeyName,
|
|
||||||
source: sourcePath.value,
|
|
||||||
prompt_id: prompt.id,
|
|
||||||
chunks: chunk,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
className="float-right mt-4 rounded-full bg-purple-30 px-5 py-2 text-sm text-white hover:bg-[#6F3FD1] disabled:opacity-50"
|
|
||||||
>
|
|
||||||
{t('modals.createAPIKey.create')}
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|
||||||
const SaveAPIKeyModal: React.FC<SaveAPIKeyModalProps> = ({ apiKey, close }) => {
|
|
||||||
const [isCopied, setIsCopied] = React.useState(false);
|
|
||||||
const { t } = useTranslation();
|
|
||||||
const handleCopyKey = () => {
|
|
||||||
navigator.clipboard.writeText(apiKey);
|
|
||||||
setIsCopied(true);
|
|
||||||
};
|
|
||||||
return (
|
|
||||||
<div className="fixed top-0 left-0 z-30 flex h-screen w-screen items-center justify-center bg-gray-alpha bg-opacity-50">
|
|
||||||
<div className="relative w-11/12 rounded-3xl bg-white px-6 py-8 dark:bg-outer-space dark:text-bright-gray sm:w-[512px]">
|
|
||||||
<button className="absolute top-3 right-4 m-2 w-3" onClick={close}>
|
|
||||||
<img className="filter dark:invert" src={Exit} />
|
|
||||||
</button>
|
|
||||||
<h1 className="my-0 text-xl font-medium">
|
|
||||||
{' '}
|
|
||||||
{t('modals.saveKey.note')}
|
|
||||||
</h1>
|
|
||||||
<h3 className="text-sm font-normal text-outer-space">
|
|
||||||
{t('modals.saveKey.disclaimer')}
|
|
||||||
</h3>
|
|
||||||
<div className="flex justify-between py-2">
|
|
||||||
<div>
|
|
||||||
<h2 className="text-base font-semibold">API Key</h2>
|
|
||||||
<span className="text-sm font-normal leading-7 ">{apiKey}</span>
|
|
||||||
</div>
|
|
||||||
<button
|
|
||||||
className="my-1 h-10 w-20 rounded-full border border-solid border-purple-30 p-2 text-sm text-purple-30 hover:bg-purple-30 hover:text-white"
|
|
||||||
onClick={handleCopyKey}
|
|
||||||
>
|
|
||||||
{isCopied ? t('modals.saveKey.copied') : t('modals.saveKey.copy')}
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
<button
|
|
||||||
onClick={close}
|
|
||||||
className="rounded-full bg-philippine-yellow px-4 py-3 font-medium text-black hover:bg-[#E6B91A]"
|
|
||||||
>
|
|
||||||
{t('modals.saveKey.confirm')}
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|
||||||
export default APIKeys;
|
|
||||||
|
|||||||
@@ -1,22 +1,22 @@
|
|||||||
import React from 'react';
|
import React from 'react';
|
||||||
import { useSelector, useDispatch } from 'react-redux';
|
|
||||||
import Prompts from './Prompts';
|
|
||||||
import { useDarkTheme } from '../hooks';
|
|
||||||
import { useTranslation } from 'react-i18next';
|
import { useTranslation } from 'react-i18next';
|
||||||
|
import { useDispatch, useSelector } from 'react-redux';
|
||||||
|
|
||||||
|
import userService from '../api/services/userService';
|
||||||
import Dropdown from '../components/Dropdown';
|
import Dropdown from '../components/Dropdown';
|
||||||
|
import { useDarkTheme } from '../hooks';
|
||||||
import {
|
import {
|
||||||
selectPrompt,
|
|
||||||
setPrompt,
|
|
||||||
setChunks,
|
|
||||||
selectChunks,
|
selectChunks,
|
||||||
setTokenLimit,
|
selectPrompt,
|
||||||
selectTokenLimit,
|
selectTokenLimit,
|
||||||
|
setChunks,
|
||||||
setModalStateDeleteConv,
|
setModalStateDeleteConv,
|
||||||
|
setPrompt,
|
||||||
|
setTokenLimit,
|
||||||
} from '../preferences/preferenceSlice';
|
} from '../preferences/preferenceSlice';
|
||||||
|
import Prompts from './Prompts';
|
||||||
|
|
||||||
const apiHost = import.meta.env.VITE_API_HOST || 'https://docsapi.arc53.com';
|
export default function General() {
|
||||||
|
|
||||||
const General: React.FC = () => {
|
|
||||||
const {
|
const {
|
||||||
t,
|
t,
|
||||||
i18n: { changeLanguage, language },
|
i18n: { changeLanguage, language },
|
||||||
@@ -69,9 +69,9 @@ const General: React.FC = () => {
|
|||||||
const selectedPrompt = useSelector(selectPrompt);
|
const selectedPrompt = useSelector(selectPrompt);
|
||||||
|
|
||||||
React.useEffect(() => {
|
React.useEffect(() => {
|
||||||
const fetchPrompts = async () => {
|
const handleFetchPrompts = async () => {
|
||||||
try {
|
try {
|
||||||
const response = await fetch(`${apiHost}/api/get_prompts`);
|
const response = await userService.getPrompts();
|
||||||
if (!response.ok) {
|
if (!response.ok) {
|
||||||
throw new Error('Failed to fetch prompts');
|
throw new Error('Failed to fetch prompts');
|
||||||
}
|
}
|
||||||
@@ -81,14 +81,13 @@ const General: React.FC = () => {
|
|||||||
console.error(error);
|
console.error(error);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
fetchPrompts();
|
handleFetchPrompts();
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
React.useEffect(() => {
|
React.useEffect(() => {
|
||||||
localStorage.setItem('docsgpt-locale', selectedLanguage?.value as string);
|
localStorage.setItem('docsgpt-locale', selectedLanguage?.value as string);
|
||||||
changeLanguage(selectedLanguage?.value);
|
changeLanguage(selectedLanguage?.value);
|
||||||
}, [selectedLanguage, changeLanguage]);
|
}, [selectedLanguage, changeLanguage]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="mt-[59px]">
|
<div className="mt-[59px]">
|
||||||
<div className="mb-5">
|
<div className="mb-5">
|
||||||
@@ -171,7 +170,6 @@ const General: React.FC = () => {
|
|||||||
dispatch(setPrompt({ name: name, id: id, type: type }))
|
dispatch(setPrompt({ name: name, id: id, type: type }))
|
||||||
}
|
}
|
||||||
setPrompts={setPrompts}
|
setPrompts={setPrompts}
|
||||||
apiHost={apiHost}
|
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
<div className="w-56">
|
<div className="w-56">
|
||||||
@@ -189,6 +187,4 @@ const General: React.FC = () => {
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
};
|
}
|
||||||
|
|
||||||
export default General;
|
|
||||||
|
|||||||
@@ -1,15 +1,17 @@
|
|||||||
import React from 'react';
|
import React from 'react';
|
||||||
import { PromptProps, ActiveState } from '../models/misc';
|
|
||||||
import Dropdown from '../components/Dropdown';
|
|
||||||
import PromptsModal from '../preferences/PromptsModal';
|
|
||||||
import { useTranslation } from 'react-i18next';
|
import { useTranslation } from 'react-i18next';
|
||||||
const apiHost = import.meta.env.VITE_API_HOST || 'https://docsapi.arc53.com';
|
|
||||||
const Prompts: React.FC<PromptProps> = ({
|
import userService from '../api/services/userService';
|
||||||
|
import Dropdown from '../components/Dropdown';
|
||||||
|
import { ActiveState, PromptProps } from '../models/misc';
|
||||||
|
import PromptsModal from '../preferences/PromptsModal';
|
||||||
|
|
||||||
|
export default function Prompts({
|
||||||
prompts,
|
prompts,
|
||||||
selectedPrompt,
|
selectedPrompt,
|
||||||
onSelectPrompt,
|
onSelectPrompt,
|
||||||
setPrompts,
|
setPrompts,
|
||||||
}) => {
|
}: PromptProps) {
|
||||||
const handleSelectPrompt = ({
|
const handleSelectPrompt = ({
|
||||||
name,
|
name,
|
||||||
id,
|
id,
|
||||||
@@ -37,17 +39,12 @@ const Prompts: React.FC<PromptProps> = ({
|
|||||||
t,
|
t,
|
||||||
i18n: { changeLanguage, language },
|
i18n: { changeLanguage, language },
|
||||||
} = useTranslation();
|
} = useTranslation();
|
||||||
|
|
||||||
const handleAddPrompt = async () => {
|
const handleAddPrompt = async () => {
|
||||||
try {
|
try {
|
||||||
const response = await fetch(`${apiHost}/api/create_prompt`, {
|
const response = await userService.createPrompt({
|
||||||
method: 'POST',
|
name: newPromptName,
|
||||||
headers: {
|
content: newPromptContent,
|
||||||
'Content-Type': 'application/json',
|
|
||||||
},
|
|
||||||
body: JSON.stringify({
|
|
||||||
name: newPromptName,
|
|
||||||
content: newPromptContent,
|
|
||||||
}),
|
|
||||||
});
|
});
|
||||||
if (!response.ok) {
|
if (!response.ok) {
|
||||||
throw new Error('Failed to add prompt');
|
throw new Error('Failed to add prompt');
|
||||||
@@ -69,18 +66,12 @@ const Prompts: React.FC<PromptProps> = ({
|
|||||||
|
|
||||||
const handleDeletePrompt = (id: string) => {
|
const handleDeletePrompt = (id: string) => {
|
||||||
setPrompts(prompts.filter((prompt) => prompt.id !== id));
|
setPrompts(prompts.filter((prompt) => prompt.id !== id));
|
||||||
fetch(`${apiHost}/api/delete_prompt`, {
|
userService
|
||||||
method: 'POST',
|
.deletePrompt({ id })
|
||||||
headers: {
|
|
||||||
'Content-Type': 'application/json',
|
|
||||||
},
|
|
||||||
body: JSON.stringify({ id: id }),
|
|
||||||
})
|
|
||||||
.then((response) => {
|
.then((response) => {
|
||||||
if (!response.ok) {
|
if (!response.ok) {
|
||||||
throw new Error('Failed to delete prompt');
|
throw new Error('Failed to delete prompt');
|
||||||
}
|
}
|
||||||
// get 1st prompt and set it as selected
|
|
||||||
if (prompts.length > 0) {
|
if (prompts.length > 0) {
|
||||||
onSelectPrompt(prompts[0].name, prompts[0].id, prompts[0].type);
|
onSelectPrompt(prompts[0].name, prompts[0].id, prompts[0].type);
|
||||||
}
|
}
|
||||||
@@ -90,18 +81,9 @@ const Prompts: React.FC<PromptProps> = ({
|
|||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
const fetchPromptContent = async (id: string) => {
|
const handleFetchPromptContent = async (id: string) => {
|
||||||
console.log('fetching prompt content');
|
|
||||||
try {
|
try {
|
||||||
const response = await fetch(
|
const response = await userService.getSinglePrompt(id);
|
||||||
`${apiHost}/api/get_single_prompt?id=${id}`,
|
|
||||||
{
|
|
||||||
method: 'GET',
|
|
||||||
headers: {
|
|
||||||
'Content-Type': 'application/json',
|
|
||||||
},
|
|
||||||
},
|
|
||||||
);
|
|
||||||
if (!response.ok) {
|
if (!response.ok) {
|
||||||
throw new Error('Failed to fetch prompt content');
|
throw new Error('Failed to fetch prompt content');
|
||||||
}
|
}
|
||||||
@@ -113,17 +95,12 @@ const Prompts: React.FC<PromptProps> = ({
|
|||||||
};
|
};
|
||||||
|
|
||||||
const handleSaveChanges = (id: string, type: string) => {
|
const handleSaveChanges = (id: string, type: string) => {
|
||||||
fetch(`${apiHost}/api/update_prompt`, {
|
userService
|
||||||
method: 'POST',
|
.updatePrompt({
|
||||||
headers: {
|
|
||||||
'Content-Type': 'application/json',
|
|
||||||
},
|
|
||||||
body: JSON.stringify({
|
|
||||||
id: id,
|
id: id,
|
||||||
name: editPromptName,
|
name: editPromptName,
|
||||||
content: editPromptContent,
|
content: editPromptContent,
|
||||||
}),
|
})
|
||||||
})
|
|
||||||
.then((response) => {
|
.then((response) => {
|
||||||
if (!response.ok) {
|
if (!response.ok) {
|
||||||
throw new Error('Failed to update prompt');
|
throw new Error('Failed to update prompt');
|
||||||
@@ -154,7 +131,6 @@ const Prompts: React.FC<PromptProps> = ({
|
|||||||
console.error(error);
|
console.error(error);
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<div>
|
<div>
|
||||||
@@ -183,7 +159,7 @@ const Prompts: React.FC<PromptProps> = ({
|
|||||||
}) => {
|
}) => {
|
||||||
setModalType('EDIT');
|
setModalType('EDIT');
|
||||||
setEditPromptName(name);
|
setEditPromptName(name);
|
||||||
fetchPromptContent(id);
|
handleFetchPromptContent(id);
|
||||||
setCurrentPromptEdit({ id: id, name: name, type: type });
|
setCurrentPromptEdit({ id: id, name: name, type: type });
|
||||||
setModalState('ACTIVE');
|
setModalState('ACTIVE');
|
||||||
}}
|
}}
|
||||||
@@ -219,6 +195,4 @@ const Prompts: React.FC<PromptProps> = ({
|
|||||||
/>
|
/>
|
||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
};
|
}
|
||||||
|
|
||||||
export default Prompts;
|
|
||||||
|
|||||||
@@ -1,22 +1,22 @@
|
|||||||
import React from 'react';
|
import React from 'react';
|
||||||
import { useSelector, useDispatch } from 'react-redux';
|
import { useTranslation } from 'react-i18next';
|
||||||
import General from './General';
|
import { useDispatch, useSelector } from 'react-redux';
|
||||||
import Documents from './Documents';
|
|
||||||
import APIKeys from './APIKeys';
|
import userService from '../api/services/userService';
|
||||||
import Widgets from './Widgets';
|
import ArrowLeft from '../assets/arrow-left.svg';
|
||||||
|
import ArrowRight from '../assets/arrow-right.svg';
|
||||||
|
import i18n from '../locale/i18n';
|
||||||
|
import { Doc } from '../preferences/preferenceApi';
|
||||||
import {
|
import {
|
||||||
selectSourceDocs,
|
selectSourceDocs,
|
||||||
setSourceDocs,
|
setSourceDocs,
|
||||||
} from '../preferences/preferenceSlice';
|
} from '../preferences/preferenceSlice';
|
||||||
import { Doc } from '../preferences/preferenceApi';
|
import APIKeys from './APIKeys';
|
||||||
import ArrowLeft from '../assets/arrow-left.svg';
|
import Documents from './Documents';
|
||||||
import ArrowRight from '../assets/arrow-right.svg';
|
import General from './General';
|
||||||
import { useTranslation } from 'react-i18next';
|
import Widgets from './Widgets';
|
||||||
import i18n from '../locale/i18n';
|
|
||||||
|
|
||||||
const apiHost = import.meta.env.VITE_API_HOST || 'https://docsapi.arc53.com';
|
export default function Settings() {
|
||||||
|
|
||||||
const Settings: React.FC = () => {
|
|
||||||
const dispatch = useDispatch();
|
const dispatch = useDispatch();
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const tabs = [
|
const tabs = [
|
||||||
@@ -33,11 +33,11 @@ const Settings: React.FC = () => {
|
|||||||
const updateWidgetScreenshot = (screenshot: File | null) => {
|
const updateWidgetScreenshot = (screenshot: File | null) => {
|
||||||
setWidgetScreenshot(screenshot);
|
setWidgetScreenshot(screenshot);
|
||||||
};
|
};
|
||||||
|
|
||||||
const handleDeleteClick = (index: number, doc: Doc) => {
|
const handleDeleteClick = (index: number, doc: Doc) => {
|
||||||
const docPath = 'indexes/' + 'local' + '/' + doc.name;
|
const docPath = 'indexes/' + 'local' + '/' + doc.name;
|
||||||
fetch(`${apiHost}/api/delete_old?path=${docPath}`, {
|
userService
|
||||||
method: 'GET',
|
.deletePath(docPath)
|
||||||
})
|
|
||||||
.then((response) => {
|
.then((response) => {
|
||||||
if (response.ok && documents) {
|
if (response.ok && documents) {
|
||||||
const updatedDocuments = [
|
const updatedDocuments = [
|
||||||
@@ -50,7 +50,6 @@ const Settings: React.FC = () => {
|
|||||||
.catch((error) => console.error(error));
|
.catch((error) => console.error(error));
|
||||||
};
|
};
|
||||||
|
|
||||||
// persist active tab as the translated version of 'general' per language change
|
|
||||||
React.useEffect(() => {
|
React.useEffect(() => {
|
||||||
setActiveTab(t('settings.general.label'));
|
setActiveTab(t('settings.general.label'));
|
||||||
}, [i18n.language]);
|
}, [i18n.language]);
|
||||||
@@ -134,6 +133,4 @@ const Settings: React.FC = () => {
|
|||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
};
|
}
|
||||||
|
|
||||||
export default Settings;
|
|
||||||
|
|||||||
@@ -1,13 +1,14 @@
|
|||||||
import React, { useRef } from 'react';
|
import React, { useCallback, useEffect, useRef, useState } from 'react';
|
||||||
import { useCallback, useEffect, useState } from 'react';
|
|
||||||
import { useDropzone } from 'react-dropzone';
|
import { useDropzone } from 'react-dropzone';
|
||||||
|
import { useTranslation } from 'react-i18next';
|
||||||
import { useDispatch } from 'react-redux';
|
import { useDispatch } from 'react-redux';
|
||||||
|
|
||||||
|
import userService from '../api/services/userService';
|
||||||
|
import Dropdown from '../components/Dropdown';
|
||||||
|
import Input from '../components/Input';
|
||||||
import { ActiveState } from '../models/misc';
|
import { ActiveState } from '../models/misc';
|
||||||
import { getDocs } from '../preferences/preferenceApi';
|
import { getDocs } from '../preferences/preferenceApi';
|
||||||
import { setSelectedDocs, setSourceDocs } from '../preferences/preferenceSlice';
|
import { setSelectedDocs, setSourceDocs } from '../preferences/preferenceSlice';
|
||||||
import Dropdown from '../components/Dropdown';
|
|
||||||
import { useTranslation } from 'react-i18next';
|
|
||||||
import Input from '../components/Input';
|
|
||||||
|
|
||||||
function Upload({
|
function Upload({
|
||||||
modalState,
|
modalState,
|
||||||
@@ -95,20 +96,6 @@ function Upload({
|
|||||||
|
|
||||||
{/* progress bar */}
|
{/* progress bar */}
|
||||||
<ProgressBar progressPercent={progress?.percentage as number} />
|
<ProgressBar progressPercent={progress?.percentage as number} />
|
||||||
|
|
||||||
<button
|
|
||||||
onClick={() => {
|
|
||||||
setDocName('');
|
|
||||||
setfiles([]);
|
|
||||||
setProgress(undefined);
|
|
||||||
setModalState('INACTIVE');
|
|
||||||
}}
|
|
||||||
className={`rounded-3xl bg-purple-30 px-4 py-2 text-sm font-medium text-white ${
|
|
||||||
isCancellable ? '' : 'hidden'
|
|
||||||
}`}
|
|
||||||
>
|
|
||||||
Finish
|
|
||||||
</button>
|
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
@@ -125,8 +112,8 @@ function Upload({
|
|||||||
|
|
||||||
if ((progress?.percentage ?? 0) < 100) {
|
if ((progress?.percentage ?? 0) < 100) {
|
||||||
timeoutID = setTimeout(() => {
|
timeoutID = setTimeout(() => {
|
||||||
const apiHost = import.meta.env.VITE_API_HOST;
|
userService
|
||||||
fetch(`${apiHost}/api/task_status?task_id=${progress?.taskId}`)
|
.getTaskStatus(progress?.taskId as string)
|
||||||
.then((data) => data.json())
|
.then((data) => data.json())
|
||||||
.then((data) => {
|
.then((data) => {
|
||||||
if (data.status == 'SUCCESS') {
|
if (data.status == 'SUCCESS') {
|
||||||
@@ -164,6 +151,10 @@ function Upload({
|
|||||||
failed: false,
|
failed: false,
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
|
setDocName('');
|
||||||
|
setfiles([]);
|
||||||
|
setProgress(undefined);
|
||||||
|
setModalState('INACTIVE');
|
||||||
}
|
}
|
||||||
} else if (data.status == 'PROGRESS') {
|
} else if (data.status == 'PROGRESS') {
|
||||||
setProgress(
|
setProgress(
|
||||||
|
|||||||
Reference in New Issue
Block a user