217 lines
8.3 KiB
JavaScript
217 lines
8.3 KiB
JavaScript
|
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}`);
|
||
|
}
|
||
|
}
|