add requests tools
This commit is contained in:
parent
d3de0d33d7
commit
aac9160a9f
|
|
@ -1,6 +1,6 @@
|
||||||
import { INode, INodeData, INodeParams } from '../../../src/Interface'
|
import { INode, INodeData, INodeParams } from '../../../src/Interface'
|
||||||
import { getBaseClasses } from '../../../src/utils'
|
import { getBaseClasses } from '../../../src/utils'
|
||||||
import { RequestsGetTool } from 'langchain/tools'
|
import { desc, RequestParameters, RequestsGetTool } from './core'
|
||||||
|
|
||||||
class RequestsGet_Tools implements INode {
|
class RequestsGet_Tools implements INode {
|
||||||
label: string
|
label: string
|
||||||
|
|
@ -22,9 +22,22 @@ class RequestsGet_Tools implements INode {
|
||||||
this.baseClasses = [this.type, ...getBaseClasses(RequestsGetTool)]
|
this.baseClasses = [this.type, ...getBaseClasses(RequestsGetTool)]
|
||||||
this.inputs = [
|
this.inputs = [
|
||||||
{
|
{
|
||||||
label: 'Max Output Length',
|
label: 'URL',
|
||||||
name: 'maxOutputLength',
|
name: 'url',
|
||||||
type: 'number',
|
type: 'string',
|
||||||
|
description:
|
||||||
|
'Agent will make call to this exact URL. If not specified, agent will try to figure out itself from AIPlugin if provided',
|
||||||
|
additionalParams: true,
|
||||||
|
optional: true
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: 'Description',
|
||||||
|
name: 'description',
|
||||||
|
type: 'string',
|
||||||
|
rows: 4,
|
||||||
|
default: desc,
|
||||||
|
description: 'Acts like a prompt to tell agent when it should use this tool',
|
||||||
|
additionalParams: true,
|
||||||
optional: true
|
optional: true
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
|
@ -39,18 +52,18 @@ class RequestsGet_Tools implements INode {
|
||||||
|
|
||||||
async init(nodeData: INodeData): Promise<any> {
|
async init(nodeData: INodeData): Promise<any> {
|
||||||
const headers = nodeData.inputs?.headers as string
|
const headers = nodeData.inputs?.headers as string
|
||||||
const maxOutputLength = nodeData.inputs?.maxOutputLength as string
|
const url = nodeData.inputs?.url as string
|
||||||
|
const description = nodeData.inputs?.description as string
|
||||||
const obj: any = {}
|
|
||||||
if (maxOutputLength) {
|
|
||||||
obj.maxOutputLength = parseInt(maxOutputLength, 10)
|
|
||||||
}
|
|
||||||
|
|
||||||
|
const obj: RequestParameters = {}
|
||||||
|
if (url) obj.url = url
|
||||||
|
if (description) obj.description = description
|
||||||
if (headers) {
|
if (headers) {
|
||||||
const parsedHeaders = typeof headers === 'object' ? headers : JSON.parse(headers)
|
const parsedHeaders = typeof headers === 'object' ? headers : JSON.parse(headers)
|
||||||
return Object.keys(obj).length ? new RequestsGetTool(parsedHeaders, obj) : new RequestsGetTool(parsedHeaders)
|
obj.headers = parsedHeaders
|
||||||
}
|
}
|
||||||
return Object.keys(obj).length ? new RequestsGetTool(undefined, obj) : new RequestsGetTool()
|
|
||||||
|
return new RequestsGetTool(obj)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -0,0 +1,42 @@
|
||||||
|
import { Tool } from 'langchain/tools'
|
||||||
|
|
||||||
|
export const desc = `A portal to the internet. Use this when you need to get specific content from a website.
|
||||||
|
Input should be a url (i.e. https://www.google.com). The output will be the text response of the GET request.`
|
||||||
|
|
||||||
|
export interface Headers {
|
||||||
|
[key: string]: string
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface RequestParameters {
|
||||||
|
headers?: Headers
|
||||||
|
url?: string
|
||||||
|
description?: string
|
||||||
|
maxOutputLength?: number
|
||||||
|
}
|
||||||
|
|
||||||
|
export class RequestsGetTool extends Tool {
|
||||||
|
name = 'requests_get'
|
||||||
|
url = ''
|
||||||
|
description = desc
|
||||||
|
maxOutputLength = 2000
|
||||||
|
headers = {}
|
||||||
|
|
||||||
|
constructor(args?: RequestParameters) {
|
||||||
|
super()
|
||||||
|
this.url = args?.url ?? this.url
|
||||||
|
this.headers = args?.headers ?? this.headers
|
||||||
|
this.description = args?.description ?? this.description
|
||||||
|
this.maxOutputLength = args?.maxOutputLength ?? this.maxOutputLength
|
||||||
|
}
|
||||||
|
|
||||||
|
/** @ignore */
|
||||||
|
async _call(input: string) {
|
||||||
|
const inputUrl = this.url ?? input
|
||||||
|
if (process.env.DEBUG === 'true') console.info(`Making GET API call to ${inputUrl}`)
|
||||||
|
const res = await fetch(inputUrl, {
|
||||||
|
headers: this.headers
|
||||||
|
})
|
||||||
|
const text = await res.text()
|
||||||
|
return text.slice(0, this.maxOutputLength)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
@ -1,6 +1,6 @@
|
||||||
import { INode, INodeData, INodeParams } from '../../../src/Interface'
|
import { INode, INodeData, INodeParams } from '../../../src/Interface'
|
||||||
import { getBaseClasses } from '../../../src/utils'
|
import { getBaseClasses } from '../../../src/utils'
|
||||||
import { RequestsPostTool } from 'langchain/tools'
|
import { RequestParameters, desc, RequestsPostTool } from './core'
|
||||||
|
|
||||||
class RequestsPost_Tools implements INode {
|
class RequestsPost_Tools implements INode {
|
||||||
label: string
|
label: string
|
||||||
|
|
@ -22,9 +22,31 @@ class RequestsPost_Tools implements INode {
|
||||||
this.baseClasses = [this.type, ...getBaseClasses(RequestsPostTool)]
|
this.baseClasses = [this.type, ...getBaseClasses(RequestsPostTool)]
|
||||||
this.inputs = [
|
this.inputs = [
|
||||||
{
|
{
|
||||||
label: 'Max Output Length',
|
label: 'URL',
|
||||||
name: 'maxOutputLength',
|
name: 'url',
|
||||||
type: 'number',
|
type: 'string',
|
||||||
|
description:
|
||||||
|
'Agent will make call to this exact URL. If not specified, agent will try to figure out itself from AIPlugin if provided',
|
||||||
|
additionalParams: true,
|
||||||
|
optional: true
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: 'Body',
|
||||||
|
name: 'body',
|
||||||
|
type: 'json',
|
||||||
|
description:
|
||||||
|
'JSON body for the POST request. If not specified, agent will try to figure out itself from AIPlugin if provided',
|
||||||
|
additionalParams: true,
|
||||||
|
optional: true
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: 'Description',
|
||||||
|
name: 'description',
|
||||||
|
type: 'string',
|
||||||
|
rows: 4,
|
||||||
|
default: desc,
|
||||||
|
description: 'Acts like a prompt to tell agent when it should use this tool',
|
||||||
|
additionalParams: true,
|
||||||
optional: true
|
optional: true
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
|
@ -39,18 +61,23 @@ class RequestsPost_Tools implements INode {
|
||||||
|
|
||||||
async init(nodeData: INodeData): Promise<any> {
|
async init(nodeData: INodeData): Promise<any> {
|
||||||
const headers = nodeData.inputs?.headers as string
|
const headers = nodeData.inputs?.headers as string
|
||||||
const maxOutputLength = nodeData.inputs?.maxOutputLength as string
|
const url = nodeData.inputs?.url as string
|
||||||
|
const description = nodeData.inputs?.description as string
|
||||||
const obj: any = {}
|
const body = nodeData.inputs?.body as string
|
||||||
if (maxOutputLength) {
|
|
||||||
obj.maxOutputLength = parseInt(maxOutputLength, 10)
|
|
||||||
}
|
|
||||||
|
|
||||||
|
const obj: RequestParameters = {}
|
||||||
|
if (url) obj.url = url
|
||||||
|
if (description) obj.description = description
|
||||||
if (headers) {
|
if (headers) {
|
||||||
const parsedHeaders = typeof headers === 'object' ? headers : JSON.parse(headers)
|
const parsedHeaders = typeof headers === 'object' ? headers : JSON.parse(headers)
|
||||||
return Object.keys(obj).length ? new RequestsPostTool(parsedHeaders, obj) : new RequestsPostTool(parsedHeaders)
|
obj.headers = parsedHeaders
|
||||||
}
|
}
|
||||||
return Object.keys(obj).length ? new RequestsPostTool(undefined, obj) : new RequestsPostTool()
|
if (body) {
|
||||||
|
const parsedBody = typeof body === 'object' ? body : JSON.parse(body)
|
||||||
|
obj.body = parsedBody
|
||||||
|
}
|
||||||
|
|
||||||
|
return new RequestsPostTool(obj)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -0,0 +1,68 @@
|
||||||
|
import { Tool } from 'langchain/tools'
|
||||||
|
|
||||||
|
export const desc = `Use this when you want to POST to a website.
|
||||||
|
Input should be a json string with two keys: "url" and "data".
|
||||||
|
The value of "url" should be a string, and the value of "data" should be a dictionary of
|
||||||
|
key-value pairs you want to POST to the url as a JSON body.
|
||||||
|
Be careful to always use double quotes for strings in the json string
|
||||||
|
The output will be the text response of the POST request.`
|
||||||
|
|
||||||
|
export interface Headers {
|
||||||
|
[key: string]: string
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface Body {
|
||||||
|
[key: string]: any
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface RequestParameters {
|
||||||
|
headers?: Headers
|
||||||
|
body?: Body
|
||||||
|
url?: string
|
||||||
|
description?: string
|
||||||
|
maxOutputLength?: number
|
||||||
|
}
|
||||||
|
|
||||||
|
export class RequestsPostTool extends Tool {
|
||||||
|
name = 'requests_post'
|
||||||
|
url = ''
|
||||||
|
description = desc
|
||||||
|
maxOutputLength = Infinity
|
||||||
|
headers = {}
|
||||||
|
body = {}
|
||||||
|
|
||||||
|
constructor(args?: RequestParameters) {
|
||||||
|
super()
|
||||||
|
this.url = args?.url ?? this.url
|
||||||
|
this.headers = args?.headers ?? this.headers
|
||||||
|
this.body = args?.body ?? this.body
|
||||||
|
this.description = args?.description ?? this.description
|
||||||
|
this.maxOutputLength = args?.maxOutputLength ?? this.maxOutputLength
|
||||||
|
}
|
||||||
|
|
||||||
|
/** @ignore */
|
||||||
|
async _call(input: string) {
|
||||||
|
try {
|
||||||
|
let inputUrl = ''
|
||||||
|
let inputBody = {}
|
||||||
|
if (Object.keys(this.body).length || this.url) {
|
||||||
|
if (this.url) inputUrl = this.url
|
||||||
|
if (Object.keys(this.body).length) inputBody = this.body
|
||||||
|
} else {
|
||||||
|
const { url, data } = JSON.parse(input)
|
||||||
|
inputUrl = url
|
||||||
|
inputBody = data
|
||||||
|
}
|
||||||
|
if (process.env.DEBUG === 'true') console.info(`Making POST API call to ${inputUrl} with body ${JSON.stringify(inputBody)}`)
|
||||||
|
const res = await fetch(inputUrl, {
|
||||||
|
method: 'POST',
|
||||||
|
headers: this.headers,
|
||||||
|
body: JSON.stringify(inputBody)
|
||||||
|
})
|
||||||
|
const text = await res.text()
|
||||||
|
return text.slice(0, this.maxOutputLength)
|
||||||
|
} catch (error) {
|
||||||
|
return `${error}`
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
@ -4,18 +4,27 @@ import parser from 'html-react-parser'
|
||||||
import PropTypes from 'prop-types'
|
import PropTypes from 'prop-types'
|
||||||
import { useSelector } from 'react-redux'
|
import { useSelector } from 'react-redux'
|
||||||
|
|
||||||
export const TooltipWithParser = ({ title }) => {
|
export const TooltipWithParser = ({ title, style }) => {
|
||||||
const customization = useSelector((state) => state.customization)
|
const customization = useSelector((state) => state.customization)
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Tooltip title={parser(title)} placement='right'>
|
<Tooltip title={parser(title)} placement='right'>
|
||||||
<IconButton sx={{ height: 25, width: 25 }}>
|
<IconButton sx={{ height: 15, width: 15 }}>
|
||||||
<Info style={{ background: 'transparent', color: customization.isDarkMode ? 'white' : 'inherit', height: 18, width: 18 }} />
|
<Info
|
||||||
|
style={{
|
||||||
|
...style,
|
||||||
|
background: 'transparent',
|
||||||
|
color: customization.isDarkMode ? 'white' : 'inherit',
|
||||||
|
height: 15,
|
||||||
|
width: 15
|
||||||
|
}}
|
||||||
|
/>
|
||||||
</IconButton>
|
</IconButton>
|
||||||
</Tooltip>
|
</Tooltip>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
TooltipWithParser.propTypes = {
|
TooltipWithParser.propTypes = {
|
||||||
title: PropTypes.node
|
title: PropTypes.node,
|
||||||
|
style: PropTypes.any
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -134,7 +134,16 @@ const CanvasNode = ({ data }) => {
|
||||||
<NodeInputHandler key={index} inputParam={inputParam} data={data} />
|
<NodeInputHandler key={index} inputParam={inputParam} data={data} />
|
||||||
))}
|
))}
|
||||||
{data.inputParams.find((param) => param.additionalParams) && (
|
{data.inputParams.find((param) => param.additionalParams) && (
|
||||||
<div style={{ textAlign: 'center' }}>
|
<div
|
||||||
|
style={{
|
||||||
|
textAlign: 'center',
|
||||||
|
marginTop:
|
||||||
|
data.inputParams.filter((param) => param.additionalParams).length ===
|
||||||
|
data.inputParams.length + data.inputAnchors.length
|
||||||
|
? 20
|
||||||
|
: 0
|
||||||
|
}}
|
||||||
|
>
|
||||||
<Button sx={{ borderRadius: 25, width: '90%', mb: 2 }} variant='outlined' onClick={onDialogClicked}>
|
<Button sx={{ borderRadius: 25, width: '90%', mb: 2 }} variant='outlined' onClick={onDialogClicked}>
|
||||||
Additional Parameters
|
Additional Parameters
|
||||||
</Button>
|
</Button>
|
||||||
|
|
|
||||||
|
|
@ -17,6 +17,7 @@ import { SwitchInput } from 'ui-component/switch/Switch'
|
||||||
import { flowContext } from 'store/context/ReactFlowContext'
|
import { flowContext } from 'store/context/ReactFlowContext'
|
||||||
import { isValidConnection, getAvailableNodesForVariable } from 'utils/genericHelper'
|
import { isValidConnection, getAvailableNodesForVariable } from 'utils/genericHelper'
|
||||||
import { JsonEditorInput } from 'ui-component/json/JsonEditor'
|
import { JsonEditorInput } from 'ui-component/json/JsonEditor'
|
||||||
|
import { TooltipWithParser } from 'ui-component/tooltip/TooltipWithParser'
|
||||||
|
|
||||||
const CustomWidthTooltip = styled(({ className, ...props }) => <Tooltip {...props} classes={{ popper: className }} />)({
|
const CustomWidthTooltip = styled(({ className, ...props }) => <Tooltip {...props} classes={{ popper: className }} />)({
|
||||||
[`& .${tooltipClasses.tooltip}`]: {
|
[`& .${tooltipClasses.tooltip}`]: {
|
||||||
|
|
@ -123,6 +124,7 @@ const NodeInputHandler = ({ inputAnchor, inputParam, data, disabled = false, isA
|
||||||
<Typography>
|
<Typography>
|
||||||
{inputParam.label}
|
{inputParam.label}
|
||||||
{!inputParam.optional && <span style={{ color: 'red' }}> *</span>}
|
{!inputParam.optional && <span style={{ color: 'red' }}> *</span>}
|
||||||
|
{inputParam.description && <TooltipWithParser style={{ marginLeft: 10 }} title={inputParam.description} />}
|
||||||
</Typography>
|
</Typography>
|
||||||
<div style={{ flexGrow: 1 }}></div>
|
<div style={{ flexGrow: 1 }}></div>
|
||||||
{inputParam.type === 'string' && inputParam.rows && (
|
{inputParam.type === 'string' && inputParam.rows && (
|
||||||
|
|
|
||||||
Loading…
Reference in New Issue