mirror of
https://github.com/danny-avila/LibreChat.git
synced 2025-12-16 16:30:15 +01:00
* WIP: initial logging changes add several transports in ~/config/winston omit messages in logs, truncate long strings add short blurb in dotenv for debug logging GoogleClient: using logger OpenAIClient: using logger, handleOpenAIErrors Adding typedef for payload message bumped winston and using winston-daily-rotate-file moved config for server paths to ~/config dir Added `DEBUG_LOGGING=true` to .env.example * WIP: Refactor logging statements in code * WIP: Refactor logging statements and import configurations * WIP: Refactor logging statements and import configurations * refactor: broadcast Redis initialization message with `info` not `debug` * refactor: complete Refactor logging statements and import configurations * chore: delete unused tools * fix: circular dependencies due to accessing logger * refactor(handleText): handle booleans and write tests * refactor: redact sensitive values, better formatting * chore: improve log formatting, avoid passing strings to 2nd arg * fix(ci): fix jest tests due to logger changes * refactor(getAvailablePluginsController): cache plugins as they are static and avoids async addOpenAPISpecs call every time * chore: update docs * chore: update docs * chore: create separate meiliSync logger, clean up logs to avoid being unnecessarily verbose * chore: spread objects where they are commonly logged to allow string truncation * chore: improve error log formatting
165 lines
5.8 KiB
JavaScript
165 lines
5.8 KiB
JavaScript
const { StructuredTool } = require('langchain/tools');
|
|
const axios = require('axios');
|
|
const { z } = require('zod');
|
|
|
|
const headers = {
|
|
'Content-Type': 'application/json',
|
|
};
|
|
|
|
function getServerURL() {
|
|
const url = process.env.CODESHERPA_SERVER_URL || '';
|
|
if (!url) {
|
|
throw new Error('Missing CODESHERPA_SERVER_URL environment variable.');
|
|
}
|
|
return url;
|
|
}
|
|
|
|
class RunCode extends StructuredTool {
|
|
constructor() {
|
|
super();
|
|
this.name = 'RunCode';
|
|
this.description =
|
|
'Use this plugin to run code with the following parameters\ncode: your code\nlanguage: either Python, Rust, or C++.';
|
|
this.headers = headers;
|
|
this.schema = z.object({
|
|
code: z.string().describe('The code to be executed in the REPL-like environment.'),
|
|
language: z.string().describe('The programming language of the code to be executed.'),
|
|
});
|
|
}
|
|
|
|
async _call({ code, language = 'python' }) {
|
|
// logger.debug('<--------------- Running Code --------------->', { code, language });
|
|
const response = await axios({
|
|
url: `${this.url}/repl`,
|
|
method: 'post',
|
|
headers: this.headers,
|
|
data: { code, language },
|
|
});
|
|
// logger.debug('<--------------- Sucessfully ran Code --------------->', response.data);
|
|
return response.data.result;
|
|
}
|
|
}
|
|
|
|
class RunCommand extends StructuredTool {
|
|
constructor() {
|
|
super();
|
|
this.name = 'RunCommand';
|
|
this.description =
|
|
'Runs the provided terminal command and returns the output or error message.';
|
|
this.headers = headers;
|
|
this.schema = z.object({
|
|
command: z.string().describe('The terminal command to be executed.'),
|
|
});
|
|
}
|
|
|
|
async _call({ command }) {
|
|
const response = await axios({
|
|
url: `${this.url}/command`,
|
|
method: 'post',
|
|
headers: this.headers,
|
|
data: {
|
|
command,
|
|
},
|
|
});
|
|
return response.data.result;
|
|
}
|
|
}
|
|
|
|
class CodeSherpa extends StructuredTool {
|
|
constructor(fields) {
|
|
super();
|
|
this.name = 'CodeSherpa';
|
|
this.url = fields.CODESHERPA_SERVER_URL || getServerURL();
|
|
// this.description = `A plugin for interactive code execution, and shell command execution.
|
|
|
|
// Run code: provide "code" and "language"
|
|
// - Execute Python code interactively for general programming, tasks, data analysis, visualizations, and more.
|
|
// - Pre-installed packages: matplotlib, seaborn, pandas, numpy, scipy, openpyxl. If you need to install additional packages, use the \`pip install\` command.
|
|
// - When a user asks for visualization, save the plot to \`static/images/\` directory, and embed it in the response using \`http://localhost:3333/static/images/\` URL.
|
|
// - Always save all media files created to \`static/images/\` directory, and embed them in responses using \`http://localhost:3333/static/images/\` URL.
|
|
|
|
// Run command: provide "command" only
|
|
// - Run terminal commands and interact with the filesystem, run scripts, and more.
|
|
// - Install python packages using \`pip install\` command.
|
|
// - Always embed media files created or uploaded using \`http://localhost:3333/static/images/\` URL in responses.
|
|
// - Access user-uploaded files in \`static/uploads/\` directory using \`http://localhost:3333/static/uploads/\` URL.`;
|
|
this.description = `This plugin allows interactive code and shell command execution.
|
|
|
|
To run code, supply "code" and "language". Python has pre-installed packages: matplotlib, seaborn, pandas, numpy, scipy, openpyxl. Additional ones can be installed via pip.
|
|
|
|
To run commands, provide "command" only. This allows interaction with the filesystem, script execution, and package installation using pip. Created or uploaded media files are embedded in responses using a specific URL.`;
|
|
this.schema = z.object({
|
|
code: z
|
|
.string()
|
|
.optional()
|
|
.describe(
|
|
`The code to be executed in the REPL-like environment. You must save all media files created to \`${this.url}/static/images/\` and embed them in responses with markdown`,
|
|
),
|
|
language: z
|
|
.string()
|
|
.optional()
|
|
.describe(
|
|
'The programming language of the code to be executed, you must also include code.',
|
|
),
|
|
command: z
|
|
.string()
|
|
.optional()
|
|
.describe(
|
|
'The terminal command to be executed. Only provide this if you want to run a command instead of code.',
|
|
),
|
|
});
|
|
|
|
this.RunCode = new RunCode({ url: this.url });
|
|
this.RunCommand = new RunCommand({ url: this.url });
|
|
this.runCode = this.RunCode._call.bind(this);
|
|
this.runCommand = this.RunCommand._call.bind(this);
|
|
}
|
|
|
|
async _call({ code, language, command }) {
|
|
if (code?.length > 0) {
|
|
return await this.runCode({ code, language });
|
|
} else if (command) {
|
|
return await this.runCommand({ command });
|
|
} else {
|
|
return 'Invalid parameters provided.';
|
|
}
|
|
}
|
|
}
|
|
|
|
/* TODO: support file upload */
|
|
// class UploadFile extends StructuredTool {
|
|
// constructor(fields) {
|
|
// super();
|
|
// this.name = 'UploadFile';
|
|
// this.url = fields.CODESHERPA_SERVER_URL || getServerURL();
|
|
// this.description = 'Endpoint to upload a file.';
|
|
// this.headers = headers;
|
|
// this.schema = z.object({
|
|
// file: z.string().describe('The file to be uploaded.'),
|
|
// });
|
|
// }
|
|
|
|
// async _call(data) {
|
|
// const formData = new FormData();
|
|
// formData.append('file', fs.createReadStream(data.file));
|
|
|
|
// const response = await axios({
|
|
// url: `${this.url}/upload`,
|
|
// method: 'post',
|
|
// headers: {
|
|
// ...this.headers,
|
|
// 'Content-Type': `multipart/form-data; boundary=${formData._boundary}`,
|
|
// },
|
|
// data: formData,
|
|
// });
|
|
// return response.data;
|
|
// }
|
|
// }
|
|
|
|
// module.exports = [
|
|
// RunCode,
|
|
// RunCommand,
|
|
// // UploadFile
|
|
// ];
|
|
|
|
module.exports = CodeSherpa;
|