agsamantha/node_modules/@langchain/community/dist/tools/connery.js

217 lines
8.3 KiB
JavaScript
Raw Normal View History

2024-10-02 20:15:21 +00:00
import { AsyncCaller, } from "@langchain/core/utils/async_caller";
import { getEnvironmentVariable } from "@langchain/core/utils/env";
import { StructuredTool } from "@langchain/core/tools";
import { z } from "zod";
/**
* A LangChain Tool object wrapping a Connery action.
* ConneryAction is a structured tool that can be used only in the agents supporting structured tools.
* @extends StructuredTool
*/
export class ConneryAction extends StructuredTool {
/**
* Creates a ConneryAction instance based on the provided Connery Action.
* @param _action The Connery Action.
* @param _service The ConneryService instance.
* @returns A ConneryAction instance.
*/
constructor(_action, _service) {
super();
Object.defineProperty(this, "_action", {
enumerable: true,
configurable: true,
writable: true,
value: _action
});
Object.defineProperty(this, "_service", {
enumerable: true,
configurable: true,
writable: true,
value: _service
});
Object.defineProperty(this, "name", {
enumerable: true,
configurable: true,
writable: true,
value: void 0
});
Object.defineProperty(this, "description", {
enumerable: true,
configurable: true,
writable: true,
value: void 0
});
Object.defineProperty(this, "schema", {
enumerable: true,
configurable: true,
writable: true,
value: void 0
});
this.name = this._action.id;
this.description =
this._action.title +
(this._action.description ? `: ${this._action.description}` : "");
this.schema = this.createInputSchema();
}
/**
* Runs the Connery Action with the provided input.
* @param arg The input object expected by the action.
* @returns A promise that resolves to a JSON string containing the output of the action.
*/
_call(arg) {
return this._service.runAction(this._action.id, arg);
}
/**
* Creates a Zod schema for the input object expected by the Connery action.
* @returns A Zod schema for the input object expected by the Connery action.
*/
createInputSchema() {
const dynamicInputFields = {};
this._action.inputParameters.forEach((param) => {
const isRequired = param.validation?.required ?? false;
let fieldSchema = z.string();
fieldSchema = isRequired ? fieldSchema : fieldSchema.optional();
const fieldDescription = param.title + (param.description ? `: ${param.description}` : "");
fieldSchema = fieldSchema.describe(fieldDescription);
dynamicInputFields[param.key] = fieldSchema;
});
return z.object(dynamicInputFields);
}
}
/**
* A service for working with Connery Actions.
*/
export class ConneryService {
/**
* Creates a ConneryService instance.
* @param params A ConneryServiceParams object.
* If not provided, the values are retrieved from the CONNERY_RUNNER_URL
* and CONNERY_RUNNER_API_KEY environment variables.
* @returns A ConneryService instance.
*/
constructor(params) {
Object.defineProperty(this, "runnerUrl", {
enumerable: true,
configurable: true,
writable: true,
value: void 0
});
Object.defineProperty(this, "apiKey", {
enumerable: true,
configurable: true,
writable: true,
value: void 0
});
Object.defineProperty(this, "asyncCaller", {
enumerable: true,
configurable: true,
writable: true,
value: void 0
});
const runnerUrl = params?.runnerUrl ?? getEnvironmentVariable("CONNERY_RUNNER_URL");
const apiKey = params?.apiKey ?? getEnvironmentVariable("CONNERY_RUNNER_API_KEY");
if (!runnerUrl || !apiKey) {
throw new Error("CONNERY_RUNNER_URL and CONNERY_RUNNER_API_KEY environment variables must be set.");
}
this.runnerUrl = runnerUrl;
this.apiKey = apiKey;
this.asyncCaller = new AsyncCaller(params ?? {});
}
/**
* Returns the list of Connery Actions wrapped as a LangChain StructuredTool objects.
* @returns A promise that resolves to an array of ConneryAction objects.
*/
async listActions() {
const actions = await this._listActions();
return actions.map((action) => new ConneryAction(action, this));
}
/**
* Returns the specified Connery action wrapped as a LangChain StructuredTool object.
* @param actionId The ID of the action to return.
* @returns A promise that resolves to a ConneryAction object.
*/
async getAction(actionId) {
const action = await this._getAction(actionId);
return new ConneryAction(action, this);
}
/**
* Runs the specified Connery action with the provided input.
* @param actionId The ID of the action to run.
* @param input The input object expected by the action.
* @returns A promise that resolves to a JSON string containing the output of the action.
*/
async runAction(actionId, input = {}) {
const result = await this._runAction(actionId, input);
return JSON.stringify(result);
}
/**
* Returns the list of actions available in the Connery runner.
* @returns A promise that resolves to an array of Action objects.
*/
async _listActions() {
const response = await this.asyncCaller.call(fetch, `${this.runnerUrl}/v1/actions`, {
method: "GET",
headers: this._getHeaders(),
});
await this._handleError(response, "Failed to list actions");
const apiResponse = await response.json();
return apiResponse.data;
}
/**
* Returns the specified action available in the Connery runner.
* @param actionId The ID of the action to return.
* @returns A promise that resolves to an Action object.
* @throws An error if the action with the specified ID is not found.
*/
async _getAction(actionId) {
const actions = await this._listActions();
const action = actions.find((a) => a.id === actionId);
if (!action) {
throw new Error(`The action with ID "${actionId}" was not found in the list of available actions in the Connery runner.`);
}
return action;
}
/**
* Runs the specified Connery action with the provided input.
* @param actionId The ID of the action to run.
* @param input The input object expected by the action.
* @returns A promise that resolves to a RunActionResult object.
*/
async _runAction(actionId, input = {}) {
const response = await this.asyncCaller.call(fetch, `${this.runnerUrl}/v1/actions/${actionId}/run`, {
method: "POST",
headers: this._getHeaders(),
body: JSON.stringify({
input,
}),
});
await this._handleError(response, "Failed to run action");
const apiResponse = await response.json();
return apiResponse.data.output;
}
/**
* Returns a standard set of HTTP headers to be used in API calls to the Connery runner.
* @returns An object containing the standard set of HTTP headers.
*/
_getHeaders() {
return {
"Content-Type": "application/json",
"x-api-key": this.apiKey,
};
}
/**
* Shared error handler for API calls to the Connery runner.
* If the response is not ok, an error is thrown containing the error message returned by the Connery runner.
* Otherwise, the promise resolves to void.
* @param response The response object returned by the Connery runner.
* @param errorMessage The error message to be used in the error thrown if the response is not ok.
* @returns A promise that resolves to void.
* @throws An error containing the error message returned by the Connery runner.
*/
async _handleError(response, errorMessage) {
if (response.ok)
return;
const apiErrorResponse = await response.json();
throw new Error(`${errorMessage}. Status code: ${response.status}. Error message: ${apiErrorResponse.error.message}`);
}
}