refactor express to use routes

This commit is contained in:
Daniel Avila 2023-02-12 16:38:33 -05:00
parent cd71bb0727
commit ed44daf8b3
11 changed files with 157 additions and 128 deletions

View file

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

View file

@ -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": {

View file

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

View file

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

View file

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

View file

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