mirror of
https://github.com/simstudioai/sim.git
synced 2026-01-23 22:08:09 -05:00
151 lines
3.9 KiB
TypeScript
151 lines
3.9 KiB
TypeScript
import { ToolConfig, HttpMethod } from '../types';
|
|
|
|
interface RequestParams {
|
|
url: string;
|
|
method?: HttpMethod;
|
|
headers?: Record<string, string>;
|
|
body?: any;
|
|
queryParams?: Record<string, string>;
|
|
pathParams?: Record<string, string>;
|
|
formData?: Record<string, string | Blob>;
|
|
timeout?: number;
|
|
validateStatus?: (status: number) => boolean;
|
|
}
|
|
|
|
interface RequestResponse {
|
|
data: any;
|
|
status: number;
|
|
headers: Record<string, string>;
|
|
}
|
|
|
|
export const requestTool: ToolConfig<RequestParams, RequestResponse> = {
|
|
id: 'http.request',
|
|
name: 'HTTP Request',
|
|
description: 'Make HTTP requests to any endpoint with support for CRUD operations',
|
|
version: '1.0.0',
|
|
|
|
params: {
|
|
url: {
|
|
type: 'string',
|
|
required: true,
|
|
description: 'The URL to send the request to'
|
|
},
|
|
method: {
|
|
type: 'string',
|
|
default: 'GET',
|
|
description: 'HTTP method (GET, POST, PUT, PATCH, DELETE)'
|
|
},
|
|
headers: {
|
|
type: 'object',
|
|
description: 'HTTP headers to include'
|
|
},
|
|
body: {
|
|
type: 'object',
|
|
description: 'Request body (for POST, PUT, PATCH)'
|
|
},
|
|
queryParams: {
|
|
type: 'object',
|
|
description: 'URL query parameters to append'
|
|
},
|
|
pathParams: {
|
|
type: 'object',
|
|
description: 'URL path parameters to replace (e.g., :id in /users/:id)'
|
|
},
|
|
formData: {
|
|
type: 'object',
|
|
description: 'Form data to send (will set appropriate Content-Type)'
|
|
},
|
|
timeout: {
|
|
type: 'number',
|
|
default: 10000,
|
|
description: 'Request timeout in milliseconds'
|
|
},
|
|
validateStatus: {
|
|
type: 'object',
|
|
description: 'Custom status validation function'
|
|
}
|
|
},
|
|
|
|
request: {
|
|
url: (params: RequestParams) => {
|
|
let url = params.url;
|
|
|
|
// Replace path parameters
|
|
if (params.pathParams) {
|
|
Object.entries(params.pathParams).forEach(([key, value]) => {
|
|
url = url.replace(`:${key}`, encodeURIComponent(value));
|
|
});
|
|
}
|
|
|
|
// Append query parameters
|
|
if (params.queryParams) {
|
|
const queryString = Object.entries(params.queryParams)
|
|
.map(([key, value]) => `${encodeURIComponent(key)}=${encodeURIComponent(value)}`)
|
|
.join('&');
|
|
url += (url.includes('?') ? '&' : '?') + queryString;
|
|
}
|
|
|
|
return url;
|
|
},
|
|
method: 'POST' as HttpMethod,
|
|
headers: (params: RequestParams) => {
|
|
const headers: Record<string, string> = {
|
|
...params.headers
|
|
};
|
|
|
|
// Set appropriate Content-Type
|
|
if (params.formData) {
|
|
// Don't set Content-Type for FormData, browser will set it with boundary
|
|
return headers;
|
|
} else if (params.body) {
|
|
headers['Content-Type'] = 'application/json';
|
|
}
|
|
|
|
return headers;
|
|
},
|
|
body: (params: RequestParams) => {
|
|
if (params.formData) {
|
|
const formData = new FormData();
|
|
Object.entries(params.formData).forEach(([key, value]) => {
|
|
formData.append(key, value);
|
|
});
|
|
return formData;
|
|
}
|
|
|
|
if (params.body) {
|
|
return params.body;
|
|
}
|
|
|
|
return undefined;
|
|
}
|
|
},
|
|
|
|
transformResponse: (response) => {
|
|
// Try to parse response based on content-type
|
|
const contentType = response.headers?.['content-type'] || '';
|
|
let data = response.data;
|
|
|
|
if (contentType.includes('application/json')) {
|
|
try {
|
|
data = typeof response.data === 'string' ? JSON.parse(response.data) : response.data;
|
|
} catch (e) {
|
|
// Keep original data if parsing fails
|
|
}
|
|
}
|
|
|
|
return {
|
|
data,
|
|
status: response.status,
|
|
headers: response.headers
|
|
};
|
|
},
|
|
|
|
transformError: (error) => {
|
|
const message = error.message || error.error?.message;
|
|
const code = error.status || error.error?.status;
|
|
const details = error.response?.data
|
|
? `\nDetails: ${JSON.stringify(error.response.data)}`
|
|
: '';
|
|
return `${message} (${code})${details}`;
|
|
}
|
|
};
|