synced with upstream

This commit is contained in:
ManishMadan2882
2024-07-26 15:36:27 +05:30
20 changed files with 611 additions and 527 deletions

View File

@@ -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": "*"
} }

View File

@@ -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) {

View 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;

View 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;

View 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;

View 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;

View File

@@ -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 } })),
); );
}; };

View File

@@ -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;

View File

@@ -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

View File

@@ -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!,

View 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>
);
}

View 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>
);
}

View File

@@ -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();

View File

@@ -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 = {

View File

@@ -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());
} }

View File

@@ -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;

View File

@@ -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;

View File

@@ -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;

View File

@@ -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;

View File

@@ -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(