mirror of
https://github.com/danny-avila/LibreChat.git
synced 2025-09-22 06:00:56 +02:00
refactor express to use routes
This commit is contained in:
parent
cd71bb0727
commit
ed44daf8b3
11 changed files with 157 additions and 128 deletions
2
.vscode/settings.json
vendored
2
.vscode/settings.json
vendored
|
@ -9,5 +9,5 @@
|
||||||
"files.trimTrailingWhitespace": true,
|
"files.trimTrailingWhitespace": true,
|
||||||
"javascript.suggest.enabled": false,
|
"javascript.suggest.enabled": false,
|
||||||
"javascript.updateImportsOnFileMove.enabled": "never",
|
"javascript.updateImportsOnFileMove.enabled": "never",
|
||||||
"javascript.validate.enable": false
|
"javascript.validate.enable": true
|
||||||
}
|
}
|
||||||
|
|
|
@ -6,7 +6,7 @@
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"start": "webpack-dev-server .",
|
"start": "webpack-dev-server .",
|
||||||
"build": "Webpack . --watch",
|
"build": "Webpack . --watch",
|
||||||
"server": "node server/index.js",
|
"server": "npx nodemon server/index.js",
|
||||||
"test": "test"
|
"test": "test"
|
||||||
},
|
},
|
||||||
"repository": {
|
"repository": {
|
||||||
|
|
132
server/index.js
132
server/index.js
|
@ -1,141 +1,27 @@
|
||||||
const express = require('express');
|
const express = require('express');
|
||||||
const dbConnect = require('../models/dbConnect');
|
const dbConnect = require('../models/dbConnect');
|
||||||
const { ask, titleConversation } = require('../app/chatgpt');
|
|
||||||
const { saveMessage, getMessages, deleteMessages } = require('../models/Message');
|
|
||||||
const { saveConvo, getConvos, deleteConvos, updateConvo } = require('../models/Conversation');
|
|
||||||
const { savePrompt, getPrompts, deletePrompts } = require('../models/Prompt');
|
|
||||||
const crypto = require('crypto');
|
|
||||||
const path = require('path');
|
const path = require('path');
|
||||||
const cors = require('cors');
|
const cors = require('cors');
|
||||||
|
const routes = require('./routes');
|
||||||
const app = express();
|
const app = express();
|
||||||
const port = 3050;
|
const port = 3050;
|
||||||
|
const projectPath = path.join(__dirname, '..');
|
||||||
|
dbConnect().then(() => console.log('Connected to MongoDB'));
|
||||||
|
|
||||||
app.use(cors());
|
app.use(cors());
|
||||||
app.use(express.json());
|
app.use(express.json());
|
||||||
|
|
||||||
const projectPath = path.join(__dirname, '..');
|
|
||||||
app.use(express.static(path.join(projectPath, 'public')));
|
app.use(express.static(path.join(projectPath, 'public')));
|
||||||
|
|
||||||
dbConnect().then((connection) => console.log('Connected to MongoDB'));
|
|
||||||
|
|
||||||
app.get('/', function (req, res) {
|
app.get('/', function (req, res) {
|
||||||
console.log(path.join(projectPath, 'public', 'index.html'));
|
console.log(path.join(projectPath, 'public', 'index.html'));
|
||||||
res.sendFile(path.join(projectPath, 'public', 'index.html'));
|
res.sendFile(path.join(projectPath, 'public', 'index.html'));
|
||||||
});
|
});
|
||||||
|
|
||||||
app.get('/convos', async (req, res) => {
|
app.use('/ask', routes.ask);
|
||||||
res.status(200).send(await getConvos());
|
app.use('/messages', routes.messages);
|
||||||
});
|
app.use('/convos', routes.convos);
|
||||||
|
app.use('/prompts', routes.prompts);
|
||||||
app.get('/prompts', async (req, res) => {
|
|
||||||
let filter = {};
|
|
||||||
// const { search } = req.body.arg;
|
|
||||||
// if (!!search) {
|
|
||||||
// filter = { conversationId };
|
|
||||||
// }
|
|
||||||
res.status(200).send(await getPrompts(filter));
|
|
||||||
});
|
|
||||||
|
|
||||||
app.get('/messages/:conversationId', async (req, res) => {
|
|
||||||
const { conversationId } = req.params;
|
|
||||||
res.status(200).send(await getMessages({ conversationId }));
|
|
||||||
});
|
|
||||||
|
|
||||||
app.post('/clear_convos', async (req, res) => {
|
|
||||||
let filter = {};
|
|
||||||
const { conversationId } = req.body.arg;
|
|
||||||
if (!!conversationId) {
|
|
||||||
filter = { conversationId };
|
|
||||||
}
|
|
||||||
|
|
||||||
try {
|
|
||||||
const dbResponse = await deleteConvos(filter);
|
|
||||||
res.status(201).send(dbResponse);
|
|
||||||
} catch (error) {
|
|
||||||
console.error(error);
|
|
||||||
res.status(500).send(error);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
app.post('/update_convo', async (req, res) => {
|
|
||||||
const update = req.body.arg;
|
|
||||||
|
|
||||||
try {
|
|
||||||
const dbResponse = await updateConvo(update);
|
|
||||||
res.status(201).send(dbResponse);
|
|
||||||
} catch (error) {
|
|
||||||
console.error(error);
|
|
||||||
res.status(500).send(error);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
app.post('/ask', async (req, res) => {
|
|
||||||
const { text, parentMessageId, conversationId } = req.body;
|
|
||||||
if (!text.trim().includes(' ') && text.length < 5) {
|
|
||||||
res.status(500).write('Prompt empty or too short');
|
|
||||||
res.end();
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const userMessageId = crypto.randomUUID();
|
|
||||||
let userMessage = { id: userMessageId, sender: 'User', text };
|
|
||||||
|
|
||||||
console.log('initial ask log', userMessage);
|
|
||||||
|
|
||||||
res.writeHead(200, {
|
|
||||||
Connection: 'keep-alive',
|
|
||||||
'Content-Type': 'text/event-stream',
|
|
||||||
'Cache-Control': 'no-cache, no-transform',
|
|
||||||
'Access-Control-Allow-Origin': '*',
|
|
||||||
'X-Accel-Buffering': 'no'
|
|
||||||
});
|
|
||||||
|
|
||||||
try {
|
|
||||||
let i = 0;
|
|
||||||
const progressCallback = async (partial) => {
|
|
||||||
if (i === 0) {
|
|
||||||
userMessage.parentMessageId = parentMessageId ? parentMessageId : partial.id;
|
|
||||||
userMessage.conversationId = conversationId ? conversationId : partial.conversationId;
|
|
||||||
await saveMessage(userMessage);
|
|
||||||
res.write(
|
|
||||||
`event: message\ndata: ${JSON.stringify({ ...partial, initial: true })}\n\n`
|
|
||||||
);
|
|
||||||
i++;
|
|
||||||
}
|
|
||||||
const data = JSON.stringify({ ...partial, message: true });
|
|
||||||
res.write(`event: message\ndata: ${data}\n\n`);
|
|
||||||
};
|
|
||||||
|
|
||||||
let gptResponse = await ask(text, progressCallback, { parentMessageId, conversationId });
|
|
||||||
if (!!parentMessageId) {
|
|
||||||
gptResponse = { ...gptResponse, parentMessageId };
|
|
||||||
} else {
|
|
||||||
gptResponse.title = await titleConversation(text, gptResponse.text);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (
|
|
||||||
(gptResponse.text.includes('2023') && !gptResponse.text.trim().includes(' ')) ||
|
|
||||||
gptResponse.text.toLowerCase().includes('no response') ||
|
|
||||||
gptResponse.text.toLowerCase().includes('no answer')
|
|
||||||
) {
|
|
||||||
res.status(500).write('event: error\ndata: Prompt empty or too short');
|
|
||||||
res.end();
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
gptResponse.sender = 'GPT';
|
|
||||||
await saveMessage(gptResponse);
|
|
||||||
await saveConvo(gptResponse);
|
|
||||||
|
|
||||||
res.write(`event: message\ndata: ${JSON.stringify(gptResponse)}\n\n`);
|
|
||||||
res.end();
|
|
||||||
} catch (error) {
|
|
||||||
console.log(error);
|
|
||||||
await deleteMessages({ id: userMessageId });
|
|
||||||
res.status(500).write('event: error\ndata: ' + error.message);
|
|
||||||
res.end();
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
app.listen(port, () => {
|
app.listen(port, () => {
|
||||||
console.log(`Server listening at http://localhost:${port}`);
|
console.log(`Server listening at http://localhost:${port}`);
|
||||||
});
|
});
|
76
server/routes/ask.js
Normal file
76
server/routes/ask.js
Normal file
|
@ -0,0 +1,76 @@
|
||||||
|
const express = require('express');
|
||||||
|
const crypto = require('crypto');
|
||||||
|
const router = express.Router();
|
||||||
|
const { ask, titleConversation } = require('../../app/chatgpt');
|
||||||
|
const { saveMessage, deleteMessages } = require('../../models/Message');
|
||||||
|
const { saveConvo } = require('../../models/Conversation');
|
||||||
|
|
||||||
|
router.post('/', async (req, res) => {
|
||||||
|
const { text, parentMessageId, conversationId } = req.body;
|
||||||
|
if (!text.trim().includes(' ') && text.length < 5) {
|
||||||
|
res.status(500).write('Prompt empty or too short');
|
||||||
|
res.end();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const userMessageId = crypto.randomUUID();
|
||||||
|
let userMessage = { id: userMessageId, sender: 'User', text };
|
||||||
|
|
||||||
|
console.log('initial ask log', userMessage);
|
||||||
|
|
||||||
|
res.writeHead(200, {
|
||||||
|
Connection: 'keep-alive',
|
||||||
|
'Content-Type': 'text/event-stream',
|
||||||
|
'Cache-Control': 'no-cache, no-transform',
|
||||||
|
'Access-Control-Allow-Origin': '*',
|
||||||
|
'X-Accel-Buffering': 'no'
|
||||||
|
});
|
||||||
|
|
||||||
|
try {
|
||||||
|
let i = 0;
|
||||||
|
const progressCallback = async (partial) => {
|
||||||
|
if (i === 0) {
|
||||||
|
userMessage.parentMessageId = parentMessageId ? parentMessageId : partial.id;
|
||||||
|
userMessage.conversationId = conversationId ? conversationId : partial.conversationId;
|
||||||
|
await saveMessage(userMessage);
|
||||||
|
res.write(
|
||||||
|
`event: message\ndata: ${JSON.stringify({ ...partial, initial: true })}\n\n`
|
||||||
|
);
|
||||||
|
i++;
|
||||||
|
}
|
||||||
|
const data = JSON.stringify({ ...partial, message: true });
|
||||||
|
res.write(`event: message\ndata: ${data}\n\n`);
|
||||||
|
};
|
||||||
|
|
||||||
|
let gptResponse = await ask(text, progressCallback, { parentMessageId, conversationId });
|
||||||
|
if (!!parentMessageId) {
|
||||||
|
gptResponse = { ...gptResponse, parentMessageId };
|
||||||
|
} else {
|
||||||
|
gptResponse.title = await titleConversation(text, gptResponse.text);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (
|
||||||
|
(gptResponse.text.includes('2023') && !gptResponse.text.trim().includes(' ')) ||
|
||||||
|
gptResponse.text.toLowerCase().includes('no response') ||
|
||||||
|
gptResponse.text.toLowerCase().includes('no answer')
|
||||||
|
) {
|
||||||
|
res.status(500).write('event: error\ndata: Prompt empty or too short');
|
||||||
|
res.end();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
gptResponse.sender = 'GPT';
|
||||||
|
await saveMessage(gptResponse);
|
||||||
|
await saveConvo(gptResponse);
|
||||||
|
|
||||||
|
res.write(`event: message\ndata: ${JSON.stringify(gptResponse)}\n\n`);
|
||||||
|
res.end();
|
||||||
|
} catch (error) {
|
||||||
|
console.log(error);
|
||||||
|
await deleteMessages({ id: userMessageId });
|
||||||
|
res.status(500).write('event: error\ndata: ' + error.message);
|
||||||
|
res.end();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
module.exports = router;
|
37
server/routes/convos.js
Normal file
37
server/routes/convos.js
Normal file
|
@ -0,0 +1,37 @@
|
||||||
|
const express = require('express');
|
||||||
|
const router = express.Router();
|
||||||
|
const { getConvos, deleteConvos, updateConvo } = require('../../models/Conversation');
|
||||||
|
|
||||||
|
router.get('/', async (req, res) => {
|
||||||
|
res.status(200).send(await getConvos());
|
||||||
|
});
|
||||||
|
|
||||||
|
router.post('/clear', async (req, res) => {
|
||||||
|
let filter = {};
|
||||||
|
const { conversationId } = req.body.arg;
|
||||||
|
if (!!conversationId) {
|
||||||
|
filter = { conversationId };
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
const dbResponse = await deleteConvos(filter);
|
||||||
|
res.status(201).send(dbResponse);
|
||||||
|
} catch (error) {
|
||||||
|
console.error(error);
|
||||||
|
res.status(500).send(error);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
router.post('/update', async (req, res) => {
|
||||||
|
const update = req.body.arg;
|
||||||
|
|
||||||
|
try {
|
||||||
|
const dbResponse = await updateConvo(update);
|
||||||
|
res.status(201).send(dbResponse);
|
||||||
|
} catch (error) {
|
||||||
|
console.error(error);
|
||||||
|
res.status(500).send(error);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
module.exports = router;
|
6
server/routes/index.js
Normal file
6
server/routes/index.js
Normal file
|
@ -0,0 +1,6 @@
|
||||||
|
const ask = require('./ask');
|
||||||
|
const messages = require('./messages');
|
||||||
|
const convos = require('./convos');
|
||||||
|
const prompts = require('./prompts');
|
||||||
|
|
||||||
|
module.exports = { ask, messages, convos, prompts };
|
10
server/routes/messages.js
Normal file
10
server/routes/messages.js
Normal file
|
@ -0,0 +1,10 @@
|
||||||
|
const express = require('express');
|
||||||
|
const router = express.Router();
|
||||||
|
const { getMessages } = require('../../models/Message');
|
||||||
|
|
||||||
|
router.get('/:conversationId', async (req, res) => {
|
||||||
|
const { conversationId } = req.params;
|
||||||
|
res.status(200).send(await getMessages({ conversationId }));
|
||||||
|
});
|
||||||
|
|
||||||
|
module.exports = router;
|
14
server/routes/prompts.js
Normal file
14
server/routes/prompts.js
Normal file
|
@ -0,0 +1,14 @@
|
||||||
|
const express = require('express');
|
||||||
|
const router = express.Router();
|
||||||
|
const { savePrompt, getPrompts, deletePrompts } = require('../../models/Prompt');
|
||||||
|
|
||||||
|
router.get('/', async (req, res) => {
|
||||||
|
let filter = {};
|
||||||
|
// const { search } = req.body.arg;
|
||||||
|
// if (!!search) {
|
||||||
|
// filter = { conversationId };
|
||||||
|
// }
|
||||||
|
res.status(200).send(await getPrompts(filter));
|
||||||
|
});
|
||||||
|
|
||||||
|
module.exports = router;
|
|
@ -14,7 +14,7 @@ export default function Conversation({ id, parentMessageId, conversationId, titl
|
||||||
const inputRef = useRef(null);
|
const inputRef = useRef(null);
|
||||||
const dispatch = useDispatch();
|
const dispatch = useDispatch();
|
||||||
const { trigger, isMutating } = manualSWR(`http://localhost:3050/messages/${id}`, 'get');
|
const { trigger, isMutating } = manualSWR(`http://localhost:3050/messages/${id}`, 'get');
|
||||||
const rename = manualSWR(`http://localhost:3050/update_convo`, 'post');
|
const rename = manualSWR(`http://localhost:3050/convos/update`, 'post');
|
||||||
|
|
||||||
const clickHandler = async () => {
|
const clickHandler = async () => {
|
||||||
if (conversationId === id) {
|
if (conversationId === id) {
|
||||||
|
|
|
@ -9,7 +9,7 @@ import { setMessages } from '~/store/messageSlice';
|
||||||
export default function DeleteButton({ conversationId, renaming, cancelHandler }) {
|
export default function DeleteButton({ conversationId, renaming, cancelHandler }) {
|
||||||
const dispatch = useDispatch();
|
const dispatch = useDispatch();
|
||||||
const { trigger, isMutating } = manualSWR(
|
const { trigger, isMutating } = manualSWR(
|
||||||
'http://localhost:3050/clear_convos',
|
'http://localhost:3050/convos/clear',
|
||||||
'post',
|
'post',
|
||||||
() => {
|
() => {
|
||||||
dispatch(setMessages([]));
|
dispatch(setMessages([]));
|
||||||
|
|
|
@ -9,7 +9,7 @@ export default function ClearConvos() {
|
||||||
const dispatch = useDispatch();
|
const dispatch = useDispatch();
|
||||||
|
|
||||||
const { trigger, isMutating } = manualSWR(
|
const { trigger, isMutating } = manualSWR(
|
||||||
'http://localhost:3050/clear_convos',
|
'http://localhost:3050/convos/clear',
|
||||||
'post',
|
'post',
|
||||||
() => {
|
() => {
|
||||||
dispatch(setMessages([]));
|
dispatch(setMessages([]));
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue