diff --git a/packages/nodes-base/credentials/AirtopApi.credentials.ts b/packages/nodes-base/credentials/AirtopApi.credentials.ts new file mode 100644 index 0000000000..75afe9764a --- /dev/null +++ b/packages/nodes-base/credentials/AirtopApi.credentials.ts @@ -0,0 +1,51 @@ +import type { + IAuthenticateGeneric, + ICredentialType, + ICredentialTestRequest, + INodeProperties, +} from 'n8n-workflow'; + +export class AirtopApi implements ICredentialType { + name = 'airtopApi'; + + displayName = 'Airtop API'; + + documentationUrl = 'airtop'; + + properties: INodeProperties[] = [ + { + displayName: 'API Key', + name: 'apiKey', + type: 'string', + default: '', + description: + 'The Airtop API key. You can create one at Airtop for free.', + required: true, + typeOptions: { + password: true, + }, + noDataExpression: true, + }, + ]; + + authenticate: IAuthenticateGeneric = { + type: 'generic', + properties: { + headers: { + Authorization: '=Bearer {{$credentials.apiKey}}', + 'api-key': '={{$credentials.apiKey}}', + }, + }, + }; + + test: ICredentialTestRequest = { + request: { + method: 'GET', + baseURL: 'https://api.airtop.ai/api/v1', + url: '/sessions', + qs: { + limit: 10, + }, + }, + }; +} diff --git a/packages/nodes-base/nodes/Airtop/Airtop.node.json b/packages/nodes-base/nodes/Airtop/Airtop.node.json new file mode 100644 index 0000000000..8050ecdf40 --- /dev/null +++ b/packages/nodes-base/nodes/Airtop/Airtop.node.json @@ -0,0 +1,18 @@ +{ + "node": "n8n-nodes-base.airtop", + "nodeVersion": "1.0", + "codexVersion": "1.0", + "categories": ["Productivity", "Development"], + "resources": { + "credentialDocumentation": [ + { + "url": "https://docs.n8n.io/integrations/builtin/credentials/airtop/" + } + ], + "primaryDocumentation": [ + { + "url": "https://docs.n8n.io/integrations/builtin/app-nodes/n8n-nodes-base.airtop/" + } + ] + } +} diff --git a/packages/nodes-base/nodes/Airtop/Airtop.node.ts b/packages/nodes-base/nodes/Airtop/Airtop.node.ts new file mode 100644 index 0000000000..526a83dbcf --- /dev/null +++ b/packages/nodes-base/nodes/Airtop/Airtop.node.ts @@ -0,0 +1,67 @@ +import { NodeConnectionTypes } from 'n8n-workflow'; +import type { IExecuteFunctions, INodeType, INodeTypeDescription } from 'n8n-workflow'; + +import * as extraction from './actions/extraction/Extraction.resource'; +import * as interaction from './actions/interaction/Interaction.resource'; +import { router } from './actions/router'; +import * as session from './actions/session/Session.resource'; +import * as window from './actions/window/Window.resource'; +export class Airtop implements INodeType { + description: INodeTypeDescription = { + displayName: 'Airtop', + name: 'airtop', + icon: 'file:airtop.svg', + group: ['transform'], + defaultVersion: 1, + version: [1], + subtitle: '={{ $parameter["operation"] + ": " + $parameter["resource"] }}', + description: 'Scrape and control any site with Airtop', + usableAsTool: true, + defaults: { + name: 'Airtop', + }, + inputs: [NodeConnectionTypes.Main], + outputs: [NodeConnectionTypes.Main], + credentials: [ + { + name: 'airtopApi', + required: true, + }, + ], + properties: [ + { + displayName: 'Resource', + name: 'resource', + type: 'options', + noDataExpression: true, + options: [ + { + name: 'Session', + value: 'session', + }, + { + name: 'Window', + value: 'window', + }, + { + name: 'Extraction', + value: 'extraction', + }, + { + name: 'Interaction', + value: 'interaction', + }, + ], + default: 'session', + }, + ...session.description, + ...window.description, + ...extraction.description, + ...interaction.description, + ], + }; + + async execute(this: IExecuteFunctions) { + return await router.call(this); + } +} diff --git a/packages/nodes-base/nodes/Airtop/GenericFunctions.ts b/packages/nodes-base/nodes/Airtop/GenericFunctions.ts new file mode 100644 index 0000000000..b92e4915b6 --- /dev/null +++ b/packages/nodes-base/nodes/Airtop/GenericFunctions.ts @@ -0,0 +1,318 @@ +import { NodeApiError, type IExecuteFunctions, type INode } from 'n8n-workflow'; +import { NodeOperationError } from 'n8n-workflow'; + +import { SESSION_MODE } from './actions/common/fields'; +import { + ERROR_MESSAGES, + DEFAULT_TIMEOUT_MINUTES, + MIN_TIMEOUT_MINUTES, + MAX_TIMEOUT_MINUTES, + INTEGRATION_URL, +} from './constants'; +import { apiRequest } from './transport'; +import type { IAirtopResponse } from './transport/types'; + +/** + * Validate a required string field + * @param this - The execution context + * @param index - The index of the node + * @param field - The field to validate + * @param fieldName - The name of the field + */ +export function validateRequiredStringField( + this: IExecuteFunctions, + index: number, + field: string, + fieldName: string, +) { + let value = this.getNodeParameter(field, index) as string; + value = (value || '').trim(); + const errorMessage = ERROR_MESSAGES.REQUIRED_PARAMETER.replace('{{field}}', fieldName); + + if (!value) { + throw new NodeOperationError(this.getNode(), errorMessage, { + itemIndex: index, + }); + } + + return value; +} + +/** + * Validate the session ID parameter + * @param this - The execution context + * @param index - The index of the node + * @returns The validated session ID + */ +export function validateSessionId(this: IExecuteFunctions, index: number) { + let sessionId = this.getNodeParameter('sessionId', index, '') as string; + sessionId = (sessionId || '').trim(); + + if (!sessionId) { + throw new NodeOperationError(this.getNode(), ERROR_MESSAGES.SESSION_ID_REQUIRED, { + itemIndex: index, + }); + } + + return sessionId; +} + +/** + * Validate the session ID and window ID parameters + * @param this - The execution context + * @param index - The index of the node + * @returns The validated session ID and window ID parameters + */ +export function validateSessionAndWindowId(this: IExecuteFunctions, index: number) { + let sessionId = this.getNodeParameter('sessionId', index, '') as string; + let windowId = this.getNodeParameter('windowId', index, '') as string; + sessionId = (sessionId || '').trim(); + windowId = (windowId || '').trim(); + + if (!sessionId) { + throw new NodeOperationError(this.getNode(), ERROR_MESSAGES.SESSION_ID_REQUIRED, { + itemIndex: index, + }); + } + + if (!windowId) { + throw new NodeOperationError(this.getNode(), ERROR_MESSAGES.WINDOW_ID_REQUIRED, { + itemIndex: index, + }); + } + + return { + sessionId, + windowId, + }; +} + +/** + * Validate the profile name parameter + * @param this - The execution context + * @param index - The index of the node + * @returns The validated profile name + */ +export function validateProfileName(this: IExecuteFunctions, index: number) { + let profileName = this.getNodeParameter('profileName', index) as string; + profileName = (profileName || '').trim(); + + if (!profileName) { + return profileName; + } + + if (!/^[a-zA-Z0-9-]+$/.test(profileName)) { + throw new NodeOperationError(this.getNode(), ERROR_MESSAGES.PROFILE_NAME_INVALID, { + itemIndex: index, + }); + } + + return profileName; +} + +/** + * Validate the timeout minutes parameter + * @param this - The execution context + * @param index - The index of the node + * @returns The validated timeout minutes + */ +export function validateTimeoutMinutes(this: IExecuteFunctions, index: number) { + const timeoutMinutes = this.getNodeParameter( + 'timeoutMinutes', + index, + DEFAULT_TIMEOUT_MINUTES, + ) as number; + + if (timeoutMinutes < MIN_TIMEOUT_MINUTES || timeoutMinutes > MAX_TIMEOUT_MINUTES) { + throw new NodeOperationError(this.getNode(), ERROR_MESSAGES.TIMEOUT_MINUTES_INVALID, { + itemIndex: index, + }); + } + + return timeoutMinutes; +} + +/** + * Validate the URL parameter + * @param this - The execution context + * @param index - The index of the node + * @returns The validated URL + */ +export function validateUrl(this: IExecuteFunctions, index: number) { + let url = this.getNodeParameter('url', index) as string; + url = (url || '').trim(); + + if (!url) { + return ''; + } + + if (!url.startsWith('http')) { + throw new NodeOperationError(this.getNode(), ERROR_MESSAGES.URL_INVALID, { + itemIndex: index, + }); + } + + return url; +} + +/** + * Validate the Proxy URL parameter + * @param this - The execution context + * @param index - The index of the node + * @param proxy - The value of the Proxy parameter + * @returns The validated proxy URL + */ +export function validateProxyUrl(this: IExecuteFunctions, index: number, proxy: string) { + let proxyUrl = this.getNodeParameter('proxyUrl', index, '') as string; + proxyUrl = (proxyUrl || '').trim(); + + // only validate proxyUrl if proxy is custom + if (proxy !== 'custom') { + return ''; + } + + if (!proxyUrl) { + throw new NodeOperationError(this.getNode(), ERROR_MESSAGES.PROXY_URL_REQUIRED, { + itemIndex: index, + }); + } + + if (!proxyUrl.startsWith('http')) { + throw new NodeOperationError(this.getNode(), ERROR_MESSAGES.PROXY_URL_INVALID, { + itemIndex: index, + }); + } + + return proxyUrl; +} + +/** + * Validate the screen resolution parameter + * @param this - The execution context + * @param index - The index of the node + * @returns The validated screen resolution + */ +export function validateScreenResolution(this: IExecuteFunctions, index: number) { + let screenResolution = this.getNodeParameter('screenResolution', index, '') as string; + screenResolution = (screenResolution || '').trim().toLowerCase(); + const regex = /^\d{3,4}x\d{3,4}$/; // Expected format: 1280x720 + + if (!screenResolution) { + return ''; + } + + if (!regex.test(screenResolution)) { + throw new NodeOperationError(this.getNode(), ERROR_MESSAGES.SCREEN_RESOLUTION_INVALID, { + itemIndex: index, + }); + } + + return screenResolution; +} + +/** + * Validate the save profile on termination parameter + * @param this - The execution context + * @param index - The index of the node + * @param profileName - The profile name + * @returns The validated save profile on termination + */ +export function validateSaveProfileOnTermination( + this: IExecuteFunctions, + index: number, + profileName: string, +) { + const saveProfileOnTermination = this.getNodeParameter( + 'saveProfileOnTermination', + index, + false, + ) as boolean; + + if (saveProfileOnTermination && !profileName) { + throw new NodeOperationError(this.getNode(), ERROR_MESSAGES.PROFILE_NAME_REQUIRED, { + itemIndex: index, + }); + } + + return saveProfileOnTermination; +} + +/** + * Check if there is an error in the API response and throw NodeApiError + * @param node - The node instance + * @param response - The response from the API + */ +export function validateAirtopApiResponse(node: INode, response: IAirtopResponse) { + if (response?.errors?.length) { + const errorMessage = response.errors.map((error) => error.message).join('\n'); + throw new NodeApiError(node, { + message: errorMessage, + }); + } +} + +/** + * Convert a screenshot from the API response to a binary buffer + * @param screenshot - The screenshot from the API response + * @returns The processed screenshot + */ +export function convertScreenshotToBinary(screenshot: { dataUrl: string }): Buffer { + const base64Data = screenshot.dataUrl.replace('data:image/jpeg;base64,', ''); + const buffer = Buffer.from(base64Data, 'base64'); + return buffer; +} + +/** + * Check if a new session should be created + * @param this - The execution context + * @param index - The index of the node + * @returns True if a new session should be created, false otherwise + */ +export function shouldCreateNewSession(this: IExecuteFunctions, index: number) { + const sessionMode = this.getNodeParameter('sessionMode', index) as string; + return Boolean(sessionMode && sessionMode === SESSION_MODE.NEW); +} + +/** + * Create a new session and window + * @param this - The execution context + * @param index - The index of the node + * @returns The session ID and window ID + */ +export async function createSessionAndWindow( + this: IExecuteFunctions, + index: number, +): Promise<{ sessionId: string; windowId: string }> { + const node = this.getNode(); + const noCodeEndpoint = `${INTEGRATION_URL}/create-session`; + const profileName = validateProfileName.call(this, index); + const url = validateRequiredStringField.call(this, index, 'url', 'URL'); + + const { sessionId } = await apiRequest.call(this, 'POST', noCodeEndpoint, { + configuration: { + profileName, + }, + }); + + if (!sessionId) { + throw new NodeApiError(node, { + message: 'Failed to create session', + code: 500, + }); + } + this.logger.info(`[${node.name}] Session successfully created.`); + + const windowResponse = await apiRequest.call(this, 'POST', `/sessions/${sessionId}/windows`, { + url, + }); + const windowId = windowResponse?.data?.windowId as string; + + if (!windowId) { + throw new NodeApiError(node, { + message: 'Failed to create window', + code: 500, + }); + } + this.logger.info(`[${node.name}] Window successfully created.`); + return { sessionId, windowId }; +} diff --git a/packages/nodes-base/nodes/Airtop/actions/common/fields.ts b/packages/nodes-base/nodes/Airtop/actions/common/fields.ts new file mode 100644 index 0000000000..71ee973733 --- /dev/null +++ b/packages/nodes-base/nodes/Airtop/actions/common/fields.ts @@ -0,0 +1,156 @@ +import type { INodeProperties } from 'n8n-workflow'; + +export const SESSION_MODE = { + NEW: 'new', + EXISTING: 'existing', +} as const; + +/** + * Session related fields + */ + +export const sessionIdField: INodeProperties = { + displayName: 'Session ID', + name: 'sessionId', + type: 'string', + required: true, + default: '={{ $json["sessionId"] }}', + description: + 'The ID of the Session to use', +}; + +export const windowIdField: INodeProperties = { + displayName: 'Window ID', + name: 'windowId', + type: 'string', + required: true, + default: '={{ $json["windowId"] }}', + description: + 'The ID of the Window to use', +}; + +export const profileNameField: INodeProperties = { + displayName: 'Profile Name', + name: 'profileName', + type: 'string', + default: '', + description: 'The name of the Airtop profile to load or create', + hint: 'Learn more about Airtop profiles', + placeholder: 'e.g. my-x-profile', +}; + +export const urlField: INodeProperties = { + displayName: 'URL', + name: 'url', + type: 'string', + default: '', + placeholder: 'e.g. https://google.com', + description: 'URL to load in the window', +}; + +/** + * Extraction related fields + */ + +export const outputSchemaField: INodeProperties = { + displayName: 'JSON Output Schema', + name: 'outputSchema', + description: 'JSON schema defining the structure of the output', + hint: 'If you want to force your output to be JSON, provide a valid JSON schema describing the output. You can generate one automatically in the Airtop API Playground.', + type: 'json', + default: '', +}; + +/** + * Interaction related fields + */ + +export const elementDescriptionField: INodeProperties = { + displayName: 'Element Description', + name: 'elementDescription', + type: 'string', + default: '', + description: 'A specific description of the element to execute the interaction on', + placeholder: 'e.g. the search box at the top of the page', +}; + +export function getSessionModeFields(resource: string, operations: string[]): INodeProperties[] { + return [ + { + displayName: 'Session Mode', + name: 'sessionMode', + type: 'options', + default: 'existing', + description: 'Choose between creating a new session or using an existing one', + options: [ + { + name: 'Automatically Create Session', + description: 'Automatically create a new session and window for this operation', + value: SESSION_MODE.NEW, + }, + { + name: 'Use Existing Session', + description: 'Use an existing session and window for this operation', + value: SESSION_MODE.EXISTING, + }, + ], + displayOptions: { + show: { + resource: [resource], + operation: operations, + }, + }, + }, + { + ...sessionIdField, + displayOptions: { + show: { + resource: [resource], + sessionMode: [SESSION_MODE.EXISTING], + }, + }, + }, + { + ...windowIdField, + displayOptions: { + show: { + resource: [resource], + sessionMode: [SESSION_MODE.EXISTING], + }, + }, + }, + { + ...urlField, + required: true, + displayOptions: { + show: { + resource: [resource], + sessionMode: [SESSION_MODE.NEW], + }, + }, + }, + { + ...profileNameField, + displayOptions: { + show: { + resource: [resource], + sessionMode: [SESSION_MODE.NEW], + }, + }, + }, + { + displayName: 'Auto-Terminate Session', + name: 'autoTerminateSession', + type: 'boolean', + default: true, + description: + 'Whether to terminate the session after the operation is complete. When disabled, you must manually terminate the session. By default, idle sessions timeout after 10 minutes', + displayOptions: { + show: { + resource: [resource], + sessionMode: [SESSION_MODE.NEW], + }, + }, + }, + ]; +} diff --git a/packages/nodes-base/nodes/Airtop/actions/common/session.utils.ts b/packages/nodes-base/nodes/Airtop/actions/common/session.utils.ts new file mode 100644 index 0000000000..b3d33287f7 --- /dev/null +++ b/packages/nodes-base/nodes/Airtop/actions/common/session.utils.ts @@ -0,0 +1,45 @@ +import type { INodeExecutionData, IExecuteFunctions, IDataObject } from 'n8n-workflow'; + +import { + validateSessionAndWindowId, + createSessionAndWindow, + shouldCreateNewSession, + validateAirtopApiResponse, +} from '../../GenericFunctions'; +import { apiRequest } from '../../transport'; + +/** + * Execute the node operation. Creates and terminates a new session if needed. + * @param this - The execution context + * @param index - The index of the node + * @param request - The request to execute + * @returns The response from the request + */ +export async function executeRequestWithSessionManagement( + this: IExecuteFunctions, + index: number, + request: { + method: 'POST' | 'DELETE'; + path: string; + body: IDataObject; + }, +): Promise { + const { sessionId, windowId } = shouldCreateNewSession.call(this, index) + ? await createSessionAndWindow.call(this, index) + : validateSessionAndWindowId.call(this, index); + + const shouldTerminateSession = this.getNodeParameter('autoTerminateSession', index, false); + + const endpoint = request.path.replace('{sessionId}', sessionId).replace('{windowId}', windowId); + const response = await apiRequest.call(this, request.method, endpoint, request.body); + + validateAirtopApiResponse(this.getNode(), response); + + if (shouldTerminateSession) { + await apiRequest.call(this, 'DELETE', `/sessions/${sessionId}`); + this.logger.info(`[${this.getNode().name}] Session terminated.`); + return this.helpers.returnJsonArray({ ...response }); + } + + return this.helpers.returnJsonArray({ sessionId, windowId, ...response }); +} diff --git a/packages/nodes-base/nodes/Airtop/actions/extraction/Extraction.resource.ts b/packages/nodes-base/nodes/Airtop/actions/extraction/Extraction.resource.ts new file mode 100644 index 0000000000..f7513b2deb --- /dev/null +++ b/packages/nodes-base/nodes/Airtop/actions/extraction/Extraction.resource.ts @@ -0,0 +1,46 @@ +import type { INodeProperties } from 'n8n-workflow'; + +import * as getPaginated from './getPaginated.operation'; +import * as query from './query.operation'; +import * as scrape from './scrape.operation'; +import { getSessionModeFields } from '../common/fields'; + +export { getPaginated, query, scrape }; + +export const description: INodeProperties[] = [ + { + displayName: 'Operation', + name: 'operation', + type: 'options', + noDataExpression: true, + displayOptions: { + show: { + resource: ['extraction'], + }, + }, + options: [ + { + name: 'Query Page', + value: 'query', + description: 'Query a page to extract data or ask a question given the data on the page', + action: 'Query page', + }, + { + name: 'Query Page with Pagination', + value: 'getPaginated', + description: 'Extract content from paginated or dynamically loaded pages', + action: 'Query page with pagination', + }, + { + name: 'Smart Scrape', + value: 'scrape', + description: 'Scrape a page and return the data as markdown', + action: 'Smart scrape page', + }, + ], + default: 'getPaginated', + }, + ...getSessionModeFields('extraction', ['getPaginated', 'query', 'scrape']), + ...getPaginated.description, + ...query.description, +]; diff --git a/packages/nodes-base/nodes/Airtop/actions/extraction/getPaginated.operation.ts b/packages/nodes-base/nodes/Airtop/actions/extraction/getPaginated.operation.ts new file mode 100644 index 0000000000..16edfcccea --- /dev/null +++ b/packages/nodes-base/nodes/Airtop/actions/extraction/getPaginated.operation.ts @@ -0,0 +1,114 @@ +import { + type IExecuteFunctions, + type INodeExecutionData, + type INodeProperties, +} from 'n8n-workflow'; + +import { outputSchemaField } from '../common/fields'; +import { executeRequestWithSessionManagement } from '../common/session.utils'; + +export const description: INodeProperties[] = [ + { + displayName: 'Prompt', + name: 'prompt', + type: 'string', + typeOptions: { + rows: 4, + }, + required: true, + default: '', + displayOptions: { + show: { + resource: ['extraction'], + operation: ['getPaginated'], + }, + }, + description: 'The prompt to extract data from the pages', + placeholder: 'e.g. Extract all the product names and prices', + }, + { + displayName: 'Additional Fields', + name: 'additionalFields', + type: 'collection', + placeholder: 'Add Field', + default: {}, + displayOptions: { + show: { + resource: ['extraction'], + operation: ['getPaginated'], + }, + }, + options: [ + { + ...outputSchemaField, + }, + { + displayName: 'Interaction Mode', + name: 'interactionMode', + type: 'options', + default: 'auto', + description: 'The strategy for interacting with the page', + options: [ + { + name: 'Auto', + description: 'Automatically choose the most cost-effective mode', + value: 'auto', + }, + { + name: 'Accurate', + description: 'Prioritize accuracy over cost', + value: 'accurate', + }, + { + name: 'Cost Efficient', + description: 'Minimize costs while ensuring effectiveness', + value: 'cost-efficient', + }, + ], + }, + { + displayName: 'Pagination Mode', + name: 'paginationMode', + type: 'options', + default: 'auto', + description: 'The pagination approach to use', + options: [ + { + name: 'Auto', + description: 'Look for pagination links first, then try infinite scrolling', + value: 'auto', + }, + { + name: 'Paginated', + description: 'Only use pagination links', + value: 'paginated', + }, + { + name: 'Infinite Scroll', + description: 'Scroll the page to load more content', + value: 'infinite-scroll', + }, + ], + }, + ], + }, +]; + +export async function execute( + this: IExecuteFunctions, + index: number, +): Promise { + const prompt = this.getNodeParameter('prompt', index, '') as string; + const additionalFields = this.getNodeParameter('additionalFields', index); + + return await executeRequestWithSessionManagement.call(this, index, { + method: 'POST', + path: '/sessions/{sessionId}/windows/{windowId}/paginated-extraction', + body: { + prompt, + configuration: { + ...additionalFields, + }, + }, + }); +} diff --git a/packages/nodes-base/nodes/Airtop/actions/extraction/query.operation.ts b/packages/nodes-base/nodes/Airtop/actions/extraction/query.operation.ts new file mode 100644 index 0000000000..9f966aa5b3 --- /dev/null +++ b/packages/nodes-base/nodes/Airtop/actions/extraction/query.operation.ts @@ -0,0 +1,66 @@ +import { + type IExecuteFunctions, + type INodeExecutionData, + type INodeProperties, +} from 'n8n-workflow'; + +import { outputSchemaField } from '../common/fields'; +import { executeRequestWithSessionManagement } from '../common/session.utils'; + +export const description: INodeProperties[] = [ + { + displayName: 'Prompt', + name: 'prompt', + type: 'string', + typeOptions: { + rows: 4, + }, + required: true, + default: '', + placeholder: 'e.g. Is there a login form in this page?', + displayOptions: { + show: { + resource: ['extraction'], + operation: ['query'], + }, + }, + description: 'The prompt to query the page content', + }, + { + displayName: 'Additional Fields', + name: 'additionalFields', + type: 'collection', + placeholder: 'Add Field', + default: {}, + displayOptions: { + show: { + resource: ['extraction'], + operation: ['query'], + }, + }, + options: [ + { + ...outputSchemaField, + }, + ], + }, +]; + +export async function execute( + this: IExecuteFunctions, + index: number, +): Promise { + const prompt = this.getNodeParameter('prompt', index, '') as string; + const additionalFields = this.getNodeParameter('additionalFields', index); + + return await executeRequestWithSessionManagement.call(this, index, { + method: 'POST', + path: '/sessions/{sessionId}/windows/{windowId}/page-query', + body: { + prompt, + configuration: { + ...additionalFields, + }, + }, + }); +} diff --git a/packages/nodes-base/nodes/Airtop/actions/extraction/scrape.operation.ts b/packages/nodes-base/nodes/Airtop/actions/extraction/scrape.operation.ts new file mode 100644 index 0000000000..574bea5292 --- /dev/null +++ b/packages/nodes-base/nodes/Airtop/actions/extraction/scrape.operation.ts @@ -0,0 +1,14 @@ +import type { IExecuteFunctions, INodeExecutionData } from 'n8n-workflow'; + +import { executeRequestWithSessionManagement } from '../common/session.utils'; + +export async function execute( + this: IExecuteFunctions, + index: number, +): Promise { + return await executeRequestWithSessionManagement.call(this, index, { + method: 'POST', + path: '/sessions/{sessionId}/windows/{windowId}/scrape-content', + body: {}, + }); +} diff --git a/packages/nodes-base/nodes/Airtop/actions/interaction/Interaction.resource.ts b/packages/nodes-base/nodes/Airtop/actions/interaction/Interaction.resource.ts new file mode 100644 index 0000000000..2a0ecda3da --- /dev/null +++ b/packages/nodes-base/nodes/Airtop/actions/interaction/Interaction.resource.ts @@ -0,0 +1,131 @@ +import type { INodeProperties } from 'n8n-workflow'; + +import * as click from './click.operation'; +import * as hover from './hover.operation'; +import * as type from './type.operation'; +import { sessionIdField, windowIdField } from '../common/fields'; +export { click, hover, type }; + +export const description: INodeProperties[] = [ + { + displayName: 'Operation', + name: 'operation', + type: 'options', + noDataExpression: true, + displayOptions: { + show: { + resource: ['interaction'], + }, + }, + options: [ + { + name: 'Click an Element', + value: 'click', + description: 'Execute a click on an element given a description', + action: 'Click an element', + }, + { + name: 'Hover on an Element', + value: 'hover', + description: 'Execute a hover action on an element given a description', + action: 'Hover on an element', + }, + { + name: 'Type', + value: 'type', + description: 'Execute a Type action on an element given a description', + action: 'Type text', + }, + ], + default: 'click', + }, + { + ...sessionIdField, + displayOptions: { + show: { + resource: ['interaction'], + }, + }, + }, + { + ...windowIdField, + displayOptions: { + show: { + resource: ['interaction'], + }, + }, + }, + ...click.description, + ...hover.description, + ...type.description, + { + displayName: 'Additional Fields', + name: 'additionalFields', + type: 'collection', + placeholder: 'Add Field', + default: {}, + displayOptions: { + show: { + resource: ['interaction'], + }, + }, + options: [ + { + displayName: 'Visual Scope', + name: 'visualScope', + type: 'options', + default: 'auto', + description: 'Defines the strategy for visual analysis of the current window', + options: [ + { + name: 'Auto', + description: 'Provides the simplest out-of-the-box experience for most web pages', + value: 'auto', + }, + { + name: 'Viewport', + description: 'For analysis of the current browser view only', + value: 'viewport', + }, + { + name: 'Page', + description: 'For analysis of the entire page', + value: 'page', + }, + { + name: 'Scan', + description: + "For a full page analysis on sites that have compatibility issues with 'Page' mode", + value: 'scan', + }, + ], + }, + { + displayName: 'Wait Until Event After Navigation', + name: 'waitForNavigation', + type: 'options', + default: 'load', + description: + "The condition to wait for the navigation to complete after an interaction (click, type or hover). Defaults to 'Fully Loaded'.", + options: [ + { + name: 'Fully Loaded (Slower)', + value: 'load', + }, + { + name: 'DOM Only Loaded (Faster)', + value: 'domcontentloaded', + }, + { + name: 'All Network Activity Has Stopped', + value: 'networkidle0', + }, + { + name: 'Most Network Activity Has Stopped', + value: 'networkidle2', + }, + ], + }, + ], + }, +]; diff --git a/packages/nodes-base/nodes/Airtop/actions/interaction/click.operation.ts b/packages/nodes-base/nodes/Airtop/actions/interaction/click.operation.ts new file mode 100644 index 0000000000..6aa4e7283f --- /dev/null +++ b/packages/nodes-base/nodes/Airtop/actions/interaction/click.operation.ts @@ -0,0 +1,56 @@ +import { + type IExecuteFunctions, + type INodeExecutionData, + type INodeProperties, +} from 'n8n-workflow'; + +import { constructInteractionRequest } from './helpers'; +import { + validateRequiredStringField, + validateSessionAndWindowId, + validateAirtopApiResponse, +} from '../../GenericFunctions'; +import { apiRequest } from '../../transport'; +import { elementDescriptionField } from '../common/fields'; + +export const description: INodeProperties[] = [ + { + ...elementDescriptionField, + placeholder: 'e.g. the green "save" button at the top of the page', + required: true, + displayOptions: { + show: { + resource: ['interaction'], + operation: ['click'], + }, + }, + }, +]; + +export async function execute( + this: IExecuteFunctions, + index: number, +): Promise { + const { sessionId, windowId } = validateSessionAndWindowId.call(this, index); + const elementDescription = validateRequiredStringField.call( + this, + index, + 'elementDescription', + 'Element Description', + ); + + const request = constructInteractionRequest.call(this, index, { + elementDescription, + }); + + const response = await apiRequest.call( + this, + 'POST', + `/sessions/${sessionId}/windows/${windowId}/click`, + request, + ); + + validateAirtopApiResponse(this.getNode(), response); + + return this.helpers.returnJsonArray({ sessionId, windowId, ...response }); +} diff --git a/packages/nodes-base/nodes/Airtop/actions/interaction/helpers.ts b/packages/nodes-base/nodes/Airtop/actions/interaction/helpers.ts new file mode 100644 index 0000000000..bdcbd1dec1 --- /dev/null +++ b/packages/nodes-base/nodes/Airtop/actions/interaction/helpers.ts @@ -0,0 +1,31 @@ +import type { IExecuteFunctions } from 'n8n-workflow'; + +import type { IAirtopInteractionRequest } from '../../transport/types'; + +export function constructInteractionRequest( + this: IExecuteFunctions, + index: number, + parameters: Partial = {}, +): IAirtopInteractionRequest { + const additionalFields = this.getNodeParameter('additionalFields', index); + const request: IAirtopInteractionRequest = { + configuration: {}, + }; + + if (additionalFields.visualScope) { + request.configuration.visualAnalysis = { + scope: additionalFields.visualScope as string, + }; + } + + if (additionalFields.waitForNavigation) { + request.waitForNavigation = true; + request.configuration.waitForNavigationConfig = { + waitUntil: additionalFields.waitForNavigation as string, + }; + } + + Object.assign(request, parameters); + + return request; +} diff --git a/packages/nodes-base/nodes/Airtop/actions/interaction/hover.operation.ts b/packages/nodes-base/nodes/Airtop/actions/interaction/hover.operation.ts new file mode 100644 index 0000000000..db46dd475f --- /dev/null +++ b/packages/nodes-base/nodes/Airtop/actions/interaction/hover.operation.ts @@ -0,0 +1,56 @@ +import { + type IExecuteFunctions, + type INodeExecutionData, + type INodeProperties, +} from 'n8n-workflow'; + +import { constructInteractionRequest } from './helpers'; +import { + validateRequiredStringField, + validateSessionAndWindowId, + validateAirtopApiResponse, +} from '../../GenericFunctions'; +import { apiRequest } from '../../transport'; +import { elementDescriptionField } from '../common/fields'; + +export const description: INodeProperties[] = [ + { + ...elementDescriptionField, + required: true, + placeholder: 'e.g. the rounded user profile image at the top right of the page', + displayOptions: { + show: { + resource: ['interaction'], + operation: ['hover'], + }, + }, + }, +]; + +export async function execute( + this: IExecuteFunctions, + index: number, +): Promise { + const { sessionId, windowId } = validateSessionAndWindowId.call(this, index); + const elementDescription = validateRequiredStringField.call( + this, + index, + 'elementDescription', + 'Element Description', + ); + + const request = constructInteractionRequest.call(this, index, { + elementDescription, + }); + + const response = await apiRequest.call( + this, + 'POST', + `/sessions/${sessionId}/windows/${windowId}/hover`, + request, + ); + + validateAirtopApiResponse(this.getNode(), response); + + return this.helpers.returnJsonArray({ sessionId, windowId, ...response }); +} diff --git a/packages/nodes-base/nodes/Airtop/actions/interaction/type.operation.ts b/packages/nodes-base/nodes/Airtop/actions/interaction/type.operation.ts new file mode 100644 index 0000000000..b43ca2b9c6 --- /dev/null +++ b/packages/nodes-base/nodes/Airtop/actions/interaction/type.operation.ts @@ -0,0 +1,81 @@ +import { + type IExecuteFunctions, + type INodeExecutionData, + type INodeProperties, +} from 'n8n-workflow'; + +import { constructInteractionRequest } from './helpers'; +import { + validateRequiredStringField, + validateSessionAndWindowId, + validateAirtopApiResponse, +} from '../../GenericFunctions'; +import { apiRequest } from '../../transport'; +import { elementDescriptionField } from '../common/fields'; + +export const description: INodeProperties[] = [ + { + displayName: 'Text', + name: 'text', + type: 'string', + required: true, + default: '', + displayOptions: { + show: { + resource: ['interaction'], + operation: ['type'], + }, + }, + description: 'The text to type into the browser window', + placeholder: 'e.g. email@example.com', + }, + { + displayName: 'Press Enter Key', + name: 'pressEnterKey', + type: 'boolean', + default: false, + description: 'Whether to press the Enter key after typing the text', + displayOptions: { + show: { + resource: ['interaction'], + operation: ['type'], + }, + }, + }, + { + ...elementDescriptionField, + displayOptions: { + show: { + resource: ['interaction'], + operation: ['type'], + }, + }, + }, +]; + +export async function execute( + this: IExecuteFunctions, + index: number, +): Promise { + const { sessionId, windowId } = validateSessionAndWindowId.call(this, index); + const text = validateRequiredStringField.call(this, index, 'text', 'Text'); + const pressEnterKey = this.getNodeParameter('pressEnterKey', index) as boolean; + const elementDescription = this.getNodeParameter('elementDescription', index) as string; + + const request = constructInteractionRequest.call(this, index, { + text, + pressEnterKey, + elementDescription, + }); + + const response = await apiRequest.call( + this, + 'POST', + `/sessions/${sessionId}/windows/${windowId}/type`, + request, + ); + + validateAirtopApiResponse(this.getNode(), response); + + return this.helpers.returnJsonArray({ sessionId, windowId, ...response }); +} diff --git a/packages/nodes-base/nodes/Airtop/actions/node.type.ts b/packages/nodes-base/nodes/Airtop/actions/node.type.ts new file mode 100644 index 0000000000..6175d22a56 --- /dev/null +++ b/packages/nodes-base/nodes/Airtop/actions/node.type.ts @@ -0,0 +1,10 @@ +import type { AllEntities } from 'n8n-workflow'; + +type NodeMap = { + session: 'create' | 'save' | 'terminate'; + window: 'create' | 'close' | 'takeScreenshot' | 'load'; + extraction: 'getPaginated' | 'query' | 'scrape'; + interaction: 'click' | 'hover' | 'type'; +}; + +export type AirtopType = AllEntities; diff --git a/packages/nodes-base/nodes/Airtop/actions/router.ts b/packages/nodes-base/nodes/Airtop/actions/router.ts new file mode 100644 index 0000000000..826001010f --- /dev/null +++ b/packages/nodes-base/nodes/Airtop/actions/router.ts @@ -0,0 +1,63 @@ +import type { IDataObject, IExecuteFunctions, INodeExecutionData } from 'n8n-workflow'; +import { NodeOperationError } from 'n8n-workflow'; + +import * as extraction from './extraction/Extraction.resource'; +import * as interaction from './interaction/Interaction.resource'; +import type { AirtopType } from './node.type'; +import * as session from './session/Session.resource'; +import * as window from './window/Window.resource'; + +export async function router(this: IExecuteFunctions): Promise { + const operationResult: INodeExecutionData[] = []; + let responseData: IDataObject | IDataObject[] = []; + + const items = this.getInputData(); + const resource = this.getNodeParameter('resource', 0); + const operation = this.getNodeParameter('operation', 0); + + const airtopNodeData = { + resource, + operation, + } as AirtopType; + + for (let i = 0; i < items.length; i++) { + try { + switch (airtopNodeData.resource) { + case 'session': + responseData = await session[airtopNodeData.operation].execute.call(this, i); + break; + case 'window': + responseData = await window[airtopNodeData.operation].execute.call(this, i); + break; + case 'interaction': + responseData = await interaction[airtopNodeData.operation].execute.call(this, i); + break; + case 'extraction': + responseData = await extraction[airtopNodeData.operation].execute.call(this, i); + break; + default: + throw new NodeOperationError( + this.getNode(), + `The resource "${resource}" is not supported!`, + ); + } + + const executionData = this.helpers.constructExecutionMetaData( + this.helpers.returnJsonArray(responseData), + { itemData: { item: i } }, + ); + operationResult.push(...executionData); + } catch (error) { + if (this.continueOnFail()) { + operationResult.push({ + json: this.getInputData(i)[0].json, + error: error as NodeOperationError, + }); + } else { + throw error; + } + } + } + + return [operationResult]; +} diff --git a/packages/nodes-base/nodes/Airtop/actions/session/Session.resource.ts b/packages/nodes-base/nodes/Airtop/actions/session/Session.resource.ts new file mode 100644 index 0000000000..367cdf8d46 --- /dev/null +++ b/packages/nodes-base/nodes/Airtop/actions/session/Session.resource.ts @@ -0,0 +1,46 @@ +import type { INodeProperties } from 'n8n-workflow'; + +import * as create from './create.operation'; +import * as save from './save.operation'; +import * as terminate from './terminate.operation'; + +export { create, save, terminate }; + +export const description: INodeProperties[] = [ + { + displayName: 'Operation', + name: 'operation', + type: 'options', + noDataExpression: true, + displayOptions: { + show: { + resource: ['session'], + }, + }, + options: [ + { + name: 'Create Session', + value: 'create', + description: 'Create an Airtop browser session', + action: 'Create a session', + }, + { + name: 'Save Profile on Termination', + value: 'save', + description: + 'Save in a profile changes made in your browsing session such as cookies and local storage', + action: 'Save a profile on session termination', + }, + { + name: 'Terminate Session', + value: 'terminate', + description: 'Terminate a session', + action: 'Terminate a session', + }, + ], + default: 'create', + }, + ...create.description, + ...save.description, + ...terminate.description, +]; diff --git a/packages/nodes-base/nodes/Airtop/actions/session/create.operation.ts b/packages/nodes-base/nodes/Airtop/actions/session/create.operation.ts new file mode 100644 index 0000000000..40e3161bbf --- /dev/null +++ b/packages/nodes-base/nodes/Airtop/actions/session/create.operation.ts @@ -0,0 +1,136 @@ +import { + type IDataObject, + type IExecuteFunctions, + type INodeExecutionData, + type INodeProperties, +} from 'n8n-workflow'; + +import { INTEGRATION_URL } from '../../constants'; +import { + validateAirtopApiResponse, + validateProfileName, + validateProxyUrl, + validateSaveProfileOnTermination, + validateTimeoutMinutes, +} from '../../GenericFunctions'; +import { apiRequest } from '../../transport'; +import { profileNameField } from '../common/fields'; + +export const description: INodeProperties[] = [ + { + ...profileNameField, + displayOptions: { + show: { + resource: ['session'], + operation: ['create'], + }, + }, + }, + { + displayName: 'Save Profile', + name: 'saveProfileOnTermination', + type: 'boolean', + default: false, + description: + 'Whether to automatically save the Airtop profile for this session upon termination', + displayOptions: { + show: { + resource: ['session'], + operation: ['create'], + }, + }, + }, + { + displayName: 'Idle Timeout', + name: 'timeoutMinutes', + type: 'number', + default: 10, + validateType: 'number', + description: 'Minutes to wait before the session is terminated due to inactivity', + displayOptions: { + show: { + resource: ['session'], + operation: ['create'], + }, + }, + }, + { + displayName: 'Proxy', + name: 'proxy', + type: 'options', + default: 'none', + description: 'Choose how to configure the proxy for this session', + options: [ + { + name: 'None', + value: 'none', + description: 'No proxy will be used', + }, + { + name: 'Integrated', + value: 'integrated', + description: 'Use Airtop-provided proxy', + }, + { + name: 'Custom', + value: 'custom', + description: 'Configure a custom proxy', + }, + ], + displayOptions: { + show: { + resource: ['session'], + operation: ['create'], + }, + }, + }, + { + displayName: 'Proxy URL', + name: 'proxyUrl', + type: 'string', + default: '', + description: 'The URL of the proxy to use', + displayOptions: { + show: { + proxy: ['custom'], + }, + }, + }, +]; + +export async function execute( + this: IExecuteFunctions, + index: number, +): Promise { + const url = `${INTEGRATION_URL}/create-session`; + + const profileName = validateProfileName.call(this, index); + const timeoutMinutes = validateTimeoutMinutes.call(this, index); + const saveProfileOnTermination = validateSaveProfileOnTermination.call(this, index, profileName); + const proxyParam = this.getNodeParameter('proxy', index, 'none') as string; + const proxyUrl = validateProxyUrl.call(this, index, proxyParam); + + const body: IDataObject = { + configuration: { + profileName, + timeoutMinutes, + proxy: proxyParam === 'custom' ? proxyUrl : Boolean(proxyParam === 'integrated'), + }, + }; + + const response = await apiRequest.call(this, 'POST', url, body); + const sessionId = response.sessionId; + + // validate response + validateAirtopApiResponse(this.getNode(), response); + + if (saveProfileOnTermination) { + await apiRequest.call( + this, + 'PUT', + `/sessions/${sessionId}/save-profile-on-termination/${profileName}`, + ); + } + + return this.helpers.returnJsonArray({ sessionId } as IDataObject); +} diff --git a/packages/nodes-base/nodes/Airtop/actions/session/save.operation.ts b/packages/nodes-base/nodes/Airtop/actions/session/save.operation.ts new file mode 100644 index 0000000000..f81904b1e3 --- /dev/null +++ b/packages/nodes-base/nodes/Airtop/actions/session/save.operation.ts @@ -0,0 +1,73 @@ +import { + type IDataObject, + type IExecuteFunctions, + type INodeExecutionData, + type INodeProperties, +} from 'n8n-workflow'; + +import { + validateAirtopApiResponse, + validateProfileName, + validateRequiredStringField, + validateSessionId, +} from '../../GenericFunctions'; +import { apiRequest } from '../../transport'; +import { sessionIdField, profileNameField } from '../common/fields'; + +export const description: INodeProperties[] = [ + { + displayName: + "Note: This operation is not needed if you enabled 'Save Profile' in the 'Create Session' operation", + name: 'notice', + type: 'notice', + displayOptions: { + show: { + resource: ['session'], + operation: ['save'], + }, + }, + default: 'This operation will save the profile on session termination', + }, + { + ...sessionIdField, + displayOptions: { + show: { + resource: ['session'], + operation: ['save'], + }, + }, + }, + { + ...profileNameField, + required: true, + description: + 'The name of the Profile to save', + displayOptions: { + show: { + resource: ['session'], + operation: ['save'], + }, + }, + hint: 'Name of the profile you want to save. Must consist only of alphanumeric characters and hyphens "-"', + }, +]; + +export async function execute( + this: IExecuteFunctions, + index: number, +): Promise { + const sessionId = validateSessionId.call(this, index); + let profileName = validateRequiredStringField.call(this, index, 'profileName', 'Profile Name'); + profileName = validateProfileName.call(this, index); + + const response = await apiRequest.call( + this, + 'PUT', + `/sessions/${sessionId}/save-profile-on-termination/${profileName}`, + ); + + // validate response + validateAirtopApiResponse(this.getNode(), response); + + return this.helpers.returnJsonArray({ sessionId, profileName, ...response } as IDataObject); +} diff --git a/packages/nodes-base/nodes/Airtop/actions/session/terminate.operation.ts b/packages/nodes-base/nodes/Airtop/actions/session/terminate.operation.ts new file mode 100644 index 0000000000..66bfc52c0f --- /dev/null +++ b/packages/nodes-base/nodes/Airtop/actions/session/terminate.operation.ts @@ -0,0 +1,35 @@ +import { + type IDataObject, + type IExecuteFunctions, + type INodeExecutionData, + type INodeProperties, +} from 'n8n-workflow'; + +import { validateAirtopApiResponse, validateSessionId } from '../../GenericFunctions'; +import { apiRequest } from '../../transport'; +import { sessionIdField } from '../common/fields'; + +export const description: INodeProperties[] = [ + { + ...sessionIdField, + displayOptions: { + show: { + resource: ['session'], + operation: ['terminate'], + }, + }, + }, +]; + +export async function execute( + this: IExecuteFunctions, + index: number, +): Promise { + const sessionId = validateSessionId.call(this, index); + const response = await apiRequest.call(this, 'DELETE', `/sessions/${sessionId}`); + + // validate response + validateAirtopApiResponse(this.getNode(), response); + + return this.helpers.returnJsonArray({ success: true } as IDataObject); +} diff --git a/packages/nodes-base/nodes/Airtop/actions/window/Window.resource.ts b/packages/nodes-base/nodes/Airtop/actions/window/Window.resource.ts new file mode 100644 index 0000000000..524b85071c --- /dev/null +++ b/packages/nodes-base/nodes/Airtop/actions/window/Window.resource.ts @@ -0,0 +1,72 @@ +import type { INodeProperties } from 'n8n-workflow'; + +import * as close from './close.operation'; +import * as create from './create.operation'; +import * as load from './load.operation'; +import * as takeScreenshot from './takeScreenshot.operation'; +import { sessionIdField, windowIdField } from '../common/fields'; + +export { create, close, takeScreenshot, load }; + +export const description: INodeProperties[] = [ + { + displayName: 'Operation', + name: 'operation', + type: 'options', + noDataExpression: true, + typeOptions: { + sortable: false, + }, + displayOptions: { + show: { + resource: ['window'], + }, + }, + options: [ + { + name: 'Create a New Browser Window', + value: 'create', + description: 'Create a new browser window inside a session. Can load a URL when created.', + action: 'Create a window', + }, + { + name: 'Load URL', + value: 'load', + description: 'Load a URL in an existing window', + action: 'Load a page', + }, + { + name: 'Take Screenshot', + value: 'takeScreenshot', + description: 'Take a screenshot of the current window', + action: 'Take screenshot', + }, + { + name: 'Close Window', + value: 'close', + description: 'Close a window inside a session', + action: 'Close a window', + }, + ], + default: 'create', + }, + { + ...sessionIdField, + displayOptions: { + show: { + resource: ['window'], + }, + }, + }, + { + ...windowIdField, + displayOptions: { + show: { + resource: ['window'], + operation: ['close', 'takeScreenshot', 'load'], + }, + }, + }, + ...create.description, + ...load.description, +]; diff --git a/packages/nodes-base/nodes/Airtop/actions/window/close.operation.ts b/packages/nodes-base/nodes/Airtop/actions/window/close.operation.ts new file mode 100644 index 0000000000..1b08f06ac3 --- /dev/null +++ b/packages/nodes-base/nodes/Airtop/actions/window/close.operation.ts @@ -0,0 +1,22 @@ +import type { IExecuteFunctions, INodeExecutionData } from 'n8n-workflow'; + +import { validateAirtopApiResponse, validateSessionAndWindowId } from '../../GenericFunctions'; +import { apiRequest } from '../../transport'; + +export async function execute( + this: IExecuteFunctions, + index: number, +): Promise { + const { sessionId, windowId } = validateSessionAndWindowId.call(this, index); + + const response = await apiRequest.call( + this, + 'DELETE', + `/sessions/${sessionId}/windows/${windowId}`, + ); + + // validate response + validateAirtopApiResponse(this.getNode(), response); + + return this.helpers.returnJsonArray({ sessionId, windowId, ...response }); +} diff --git a/packages/nodes-base/nodes/Airtop/actions/window/create.operation.ts b/packages/nodes-base/nodes/Airtop/actions/window/create.operation.ts new file mode 100644 index 0000000000..1a13413a33 --- /dev/null +++ b/packages/nodes-base/nodes/Airtop/actions/window/create.operation.ts @@ -0,0 +1,186 @@ +import type { + IExecuteFunctions, + INodeExecutionData, + IDataObject, + INodeProperties, +} from 'n8n-workflow'; +import { NodeApiError } from 'n8n-workflow'; + +import { + validateAirtopApiResponse, + validateSessionId, + validateUrl, + validateScreenResolution, +} from '../../GenericFunctions'; +import { apiRequest } from '../../transport'; +import type { IAirtopResponse } from '../../transport/types'; +import { urlField } from '../common/fields'; + +export const description: INodeProperties[] = [ + { + ...urlField, + description: 'Initial URL to load in the window. Defaults to https://www.google.com.', + displayOptions: { + show: { + resource: ['window'], + operation: ['create'], + }, + }, + }, + // Live View Options + { + displayName: 'Get Live View', + name: 'getLiveView', + type: 'boolean', + default: false, + description: + 'Whether to get the URL of the window\'s Live View', + displayOptions: { + show: { + resource: ['window'], + operation: ['create'], + }, + }, + }, + { + displayName: 'Include Navigation Bar', + name: 'includeNavigationBar', + type: 'boolean', + default: false, + description: + 'Whether to include the navigation bar in the Live View. When enabled, the navigation bar will be visible allowing you to navigate between pages.', + displayOptions: { + show: { + resource: ['window'], + operation: ['create'], + getLiveView: [true], + }, + }, + }, + { + displayName: 'Screen Resolution', + name: 'screenResolution', + type: 'string', + default: '', + description: + 'The screen resolution of the Live View. Setting a resolution will force the window to open at that specific size.', + placeholder: 'e.g. 1280x720', + displayOptions: { + show: { + resource: ['window'], + operation: ['create'], + getLiveView: [true], + }, + }, + }, + { + displayName: 'Disable Resize', + name: 'disableResize', + type: 'boolean', + default: false, + description: 'Whether to disable the window from being resized in the Live View', + displayOptions: { + show: { + resource: ['window'], + operation: ['create'], + getLiveView: [true], + }, + }, + }, + { + displayName: 'Additional Fields', + name: 'additionalFields', + type: 'collection', + placeholder: 'Add Field', + default: {}, + displayOptions: { + show: { + resource: ['window'], + operation: ['create'], + }, + }, + options: [ + { + displayName: 'Wait Until', + name: 'waitUntil', + type: 'options', + description: 'Wait until the specified loading event occurs', + default: 'load', + options: [ + { + name: 'Load', + value: 'load', + description: "Wait until the page dom and it's assets have loaded", + }, + { + name: 'DOM Content Loaded', + value: 'domContentLoaded', + description: 'Wait until the page DOM has loaded', + }, + { + name: 'Complete', + value: 'complete', + description: 'Wait until all iframes in the page have loaded', + }, + { + name: 'No Wait', + value: 'noWait', + description: 'Do not wait for any loading event and it will return immediately', + }, + ], + }, + ], + }, +]; + +export async function execute( + this: IExecuteFunctions, + index: number, +): Promise { + const sessionId = validateSessionId.call(this, index); + const url = validateUrl.call(this, index); + const additionalFields = this.getNodeParameter('additionalFields', index); + // Live View Options + const getLiveView = this.getNodeParameter('getLiveView', index, false); + const includeNavigationBar = this.getNodeParameter('includeNavigationBar', index, false); + const screenResolution = validateScreenResolution.call(this, index); + const disableResize = this.getNodeParameter('disableResize', index, false); + + let response: IAirtopResponse; + + const body: IDataObject = { + url, + ...additionalFields, + }; + + response = await apiRequest.call(this, 'POST', `/sessions/${sessionId}/windows`, body); + + if (!response?.data?.windowId) { + throw new NodeApiError(this.getNode(), { + message: 'Failed to create window', + code: 500, + }); + } + + const windowId = String(response.data.windowId); + + if (getLiveView) { + // Get Window info + response = await apiRequest.call( + this, + 'GET', + `/sessions/${sessionId}/windows/${windowId}`, + undefined, + { + ...(includeNavigationBar && { includeNavigationBar: true }), + ...(screenResolution && { screenResolution }), + ...(disableResize && { disableResize: true }), + }, + ); + } + + // validate response + validateAirtopApiResponse(this.getNode(), response); + + return this.helpers.returnJsonArray({ sessionId, windowId, ...response }); +} diff --git a/packages/nodes-base/nodes/Airtop/actions/window/load.operation.ts b/packages/nodes-base/nodes/Airtop/actions/window/load.operation.ts new file mode 100644 index 0000000000..548cb0357b --- /dev/null +++ b/packages/nodes-base/nodes/Airtop/actions/window/load.operation.ts @@ -0,0 +1,95 @@ +import { + type IExecuteFunctions, + type INodeExecutionData, + type INodeProperties, +} from 'n8n-workflow'; + +import { + validateRequiredStringField, + validateSessionAndWindowId, + validateUrl, + validateAirtopApiResponse, +} from '../../GenericFunctions'; +import { apiRequest } from '../../transport'; +import { urlField } from '../common/fields'; + +export const description: INodeProperties[] = [ + { + ...urlField, + required: true, + displayOptions: { + show: { + resource: ['window'], + operation: ['load'], + }, + }, + }, + { + displayName: 'Additional Fields', + name: 'additionalFields', + type: 'collection', + placeholder: 'Add Field', + default: {}, + displayOptions: { + show: { + resource: ['window'], + operation: ['load'], + }, + }, + options: [ + { + displayName: 'Wait Until', + name: 'waitUntil', + type: 'options', + default: 'load', + description: "Wait until the specified loading event occurs. Defaults to 'Fully Loaded'.", + options: [ + { + name: 'Complete', + value: 'complete', + description: "Wait until the page and all it's iframes have loaded it's dom and assets", + }, + { + name: 'DOM Only Loaded', + value: 'domContentLoaded', + description: 'Wait until the dom has loaded', + }, + { + name: 'Fully Loaded', + value: 'load', + description: "Wait until the page dom and it's assets have loaded", + }, + { + name: 'No Wait', + value: 'noWait', + description: 'Do not wait for any loading event and will return immediately', + }, + ], + }, + ], + }, +]; + +export async function execute( + this: IExecuteFunctions, + index: number, +): Promise { + const { sessionId, windowId } = validateSessionAndWindowId.call(this, index); + let url = validateRequiredStringField.call(this, index, 'url', 'URL'); + url = validateUrl.call(this, index); + const additionalFields = this.getNodeParameter('additionalFields', index); + + const response = await apiRequest.call( + this, + 'POST', + `/sessions/${sessionId}/windows/${windowId}`, + { + url, + waitUntil: additionalFields.waitUntil, + }, + ); + + validateAirtopApiResponse(this.getNode(), response); + + return this.helpers.returnJsonArray({ sessionId, windowId, ...response }); +} diff --git a/packages/nodes-base/nodes/Airtop/actions/window/takeScreenshot.operation.ts b/packages/nodes-base/nodes/Airtop/actions/window/takeScreenshot.operation.ts new file mode 100644 index 0000000000..5ad0294571 --- /dev/null +++ b/packages/nodes-base/nodes/Airtop/actions/window/takeScreenshot.operation.ts @@ -0,0 +1,41 @@ +import type { IExecuteFunctions, INodeExecutionData, IBinaryData } from 'n8n-workflow'; + +import { + validateSessionAndWindowId, + validateAirtopApiResponse, + convertScreenshotToBinary, +} from '../../GenericFunctions'; +import { apiRequest } from '../../transport'; + +export async function execute( + this: IExecuteFunctions, + index: number, +): Promise { + const { sessionId, windowId } = validateSessionAndWindowId.call(this, index); + let data: IBinaryData | undefined; // for storing the binary data + const response = await apiRequest.call( + this, + 'POST', + `/sessions/${sessionId}/windows/${windowId}/screenshot`, + ); + + // validate response + validateAirtopApiResponse(this.getNode(), response); + + // process screenshot on success + if (response.meta?.screenshots?.length) { + const buffer = convertScreenshotToBinary(response.meta.screenshots[0]); + data = await this.helpers.prepareBinaryData(buffer, 'screenshot.jpg', 'image/jpeg'); + } + + return [ + { + json: { + sessionId, + windowId, + ...response, + }, + ...(data ? { binary: { data } } : {}), + }, + ]; +} diff --git a/packages/nodes-base/nodes/Airtop/airtop.svg b/packages/nodes-base/nodes/Airtop/airtop.svg new file mode 100644 index 0000000000..91f9c93b63 --- /dev/null +++ b/packages/nodes-base/nodes/Airtop/airtop.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/packages/nodes-base/nodes/Airtop/constants.ts b/packages/nodes-base/nodes/Airtop/constants.ts new file mode 100644 index 0000000000..6380c5869c --- /dev/null +++ b/packages/nodes-base/nodes/Airtop/constants.ts @@ -0,0 +1,21 @@ +export const BASE_URL = 'https://api.airtop.ai/api/v1'; +export const INTEGRATION_URL = 'https://portal-api.airtop.ai/integrations/v1/no-code'; + +export const DEFAULT_TIMEOUT_MINUTES = 10; +export const MIN_TIMEOUT_MINUTES = 1; +export const MAX_TIMEOUT_MINUTES = 10080; + +export const ERROR_MESSAGES = { + SESSION_ID_REQUIRED: "Please fill the 'Session ID' parameter", + WINDOW_ID_REQUIRED: "Please fill the 'Window ID' parameter", + URL_REQUIRED: "Please fill the 'URL' parameter", + PROFILE_NAME_INVALID: "'Profile Name' should only contain letters, numbers and dashes", + TIMEOUT_MINUTES_INVALID: `Timeout must be between ${MIN_TIMEOUT_MINUTES} and ${MAX_TIMEOUT_MINUTES} minutes`, + URL_INVALID: "'URL' must start with 'http' or 'https'", + PROFILE_NAME_REQUIRED: "'Profile Name' is required when 'Save Profile' is enabled", + REQUIRED_PARAMETER: "Please fill the '{{field}}' parameter", + PROXY_URL_REQUIRED: "Please fill the 'Proxy URL' parameter", + PROXY_URL_INVALID: "'Proxy URL' must start with 'http' or 'https'", + SCREEN_RESOLUTION_INVALID: + "'Screen Resolution' must be in the format 'width x height' (e.g. '1280x720')", +}; diff --git a/packages/nodes-base/nodes/Airtop/test/node/extraction/getPaginated.test.ts b/packages/nodes-base/nodes/Airtop/test/node/extraction/getPaginated.test.ts new file mode 100644 index 0000000000..2667000fc7 --- /dev/null +++ b/packages/nodes-base/nodes/Airtop/test/node/extraction/getPaginated.test.ts @@ -0,0 +1,280 @@ +import type { IExecuteFunctions } from 'n8n-workflow'; +import nock from 'nock'; + +import * as getPaginated from '../../../actions/extraction/getPaginated.operation'; +import { ERROR_MESSAGES } from '../../../constants'; +import * as GenericFunctions from '../../../GenericFunctions'; +import * as transport from '../../../transport'; +import { createMockExecuteFunction } from '../helpers'; + +const baseNodeParameters = { + resource: 'extraction', + operation: 'getPaginated', + sessionId: 'test-session-123', + windowId: 'win-123', + sessionMode: 'existing', + additionalFields: {}, +}; + +const mockResponse = { + data: { + modelResponse: + '{"items": [{"title": "Item 1", "price": "$10.99"}, {"title": "Item 2", "price": "$20.99"}]}', + }, +}; + +const mockJsonSchema = + '{"type":"object","properties":{"title":{"type":"string"},"price":{"type":"string"}}}'; + +jest.mock('../../../transport', () => { + const originalModule = jest.requireActual('../../../transport'); + return { + ...originalModule, + apiRequest: jest.fn(async (method: string, endpoint: string) => { + // For paginated extraction requests + if (endpoint.includes('/paginated-extraction')) { + return mockResponse; + } + + // For session deletion + if (method === 'DELETE' && endpoint.includes('/sessions/')) { + return { status: 'success' }; + } + + return { success: true }; + }), + }; +}); + +jest.mock('../../../GenericFunctions', () => { + const originalModule = jest.requireActual('../../../GenericFunctions'); + return { + ...originalModule, + createSessionAndWindow: jest.fn().mockImplementation(async () => { + return { + sessionId: 'new-session-123', + windowId: 'new-window-123', + }; + }), + shouldCreateNewSession: jest.fn().mockImplementation(function ( + this: IExecuteFunctions, + index: number, + ) { + const sessionMode = this.getNodeParameter('sessionMode', index) as string; + return sessionMode === 'new'; + }), + validateAirtopApiResponse: jest.fn(), + }; +}); + +describe('Test Airtop, getPaginated operation', () => { + beforeAll(() => { + nock.disableNetConnect(); + }); + + afterAll(() => { + nock.restore(); + jest.unmock('../../../transport'); + jest.unmock('../../../GenericFunctions'); + }); + + afterEach(() => { + jest.clearAllMocks(); + }); + + it('should extract data with minimal parameters', async () => { + const nodeParameters = { + ...baseNodeParameters, + prompt: 'Extract all product titles and prices', + }; + + const result = await getPaginated.execute.call(createMockExecuteFunction(nodeParameters), 0); + + expect(GenericFunctions.shouldCreateNewSession).toHaveBeenCalledTimes(1); + expect(GenericFunctions.createSessionAndWindow).not.toHaveBeenCalled(); + expect(transport.apiRequest).toHaveBeenCalledTimes(1); + expect(transport.apiRequest).toHaveBeenCalledWith( + 'POST', + '/sessions/test-session-123/windows/win-123/paginated-extraction', + { + prompt: 'Extract all product titles and prices', + configuration: {}, + }, + ); + + expect(result).toEqual([ + { + json: { + sessionId: 'test-session-123', + windowId: 'win-123', + data: mockResponse.data, + }, + }, + ]); + }); + + it('should extract data with output schema', async () => { + const nodeParameters = { + ...baseNodeParameters, + prompt: 'Extract all product titles and prices', + additionalFields: { + outputSchema: mockJsonSchema, + }, + }; + + const result = await getPaginated.execute.call(createMockExecuteFunction(nodeParameters), 0); + + expect(GenericFunctions.shouldCreateNewSession).toHaveBeenCalledTimes(1); + expect(GenericFunctions.createSessionAndWindow).not.toHaveBeenCalled(); + expect(transport.apiRequest).toHaveBeenCalledTimes(1); + expect(transport.apiRequest).toHaveBeenCalledWith( + 'POST', + '/sessions/test-session-123/windows/win-123/paginated-extraction', + { + prompt: 'Extract all product titles and prices', + configuration: { + outputSchema: mockJsonSchema, + }, + }, + ); + + expect(result).toEqual([ + { + json: { + sessionId: 'test-session-123', + windowId: 'win-123', + data: mockResponse.data, + }, + }, + ]); + }); + + ['auto', 'accurate', 'cost-efficient'].forEach((interactionMode) => { + it(`interactionMode > Should extract data with '${interactionMode}' mode`, async () => { + const nodeParameters = { + ...baseNodeParameters, + prompt: 'Extract all product titles and prices', + additionalFields: { + interactionMode, + }, + }; + + const result = await getPaginated.execute.call(createMockExecuteFunction(nodeParameters), 0); + + expect(transport.apiRequest).toHaveBeenCalledTimes(1); + expect(transport.apiRequest).toHaveBeenCalledWith( + 'POST', + '/sessions/test-session-123/windows/win-123/paginated-extraction', + { + prompt: 'Extract all product titles and prices', + configuration: { + interactionMode, + }, + }, + ); + + expect(result).toEqual([ + { + json: { + sessionId: 'test-session-123', + windowId: 'win-123', + data: mockResponse.data, + }, + }, + ]); + }); + }); + + ['auto', 'paginated', 'infinite-scroll'].forEach((paginationMode) => { + it(`paginationMode > Should extract data with '${paginationMode}' mode`, async () => { + const nodeParameters = { + ...baseNodeParameters, + prompt: 'Extract all product titles and prices', + additionalFields: { + paginationMode, + }, + }; + + const result = await getPaginated.execute.call(createMockExecuteFunction(nodeParameters), 0); + + expect(transport.apiRequest).toHaveBeenCalledTimes(1); + expect(transport.apiRequest).toHaveBeenCalledWith( + 'POST', + '/sessions/test-session-123/windows/win-123/paginated-extraction', + { + prompt: 'Extract all product titles and prices', + configuration: { + paginationMode, + }, + }, + ); + + expect(result).toEqual([ + { + json: { + sessionId: 'test-session-123', + windowId: 'win-123', + data: mockResponse.data, + }, + }, + ]); + }); + }); + + it('should extract data using a new session', async () => { + const nodeParameters = { + ...baseNodeParameters, + sessionMode: 'new', + autoTerminateSession: true, + url: 'https://example.com', + prompt: 'Extract all product titles and prices', + }; + + const result = await getPaginated.execute.call(createMockExecuteFunction(nodeParameters), 0); + + expect(GenericFunctions.shouldCreateNewSession).toHaveBeenCalledTimes(1); + expect(GenericFunctions.createSessionAndWindow).toHaveBeenCalledTimes(1); + expect(transport.apiRequest).toHaveBeenCalledTimes(2); // One for extraction, one for session deletion + expect(transport.apiRequest).toHaveBeenNthCalledWith( + 1, + 'POST', + '/sessions/new-session-123/windows/new-window-123/paginated-extraction', + { + prompt: 'Extract all product titles and prices', + configuration: {}, + }, + ); + + expect(result).toEqual([ + { + json: { + data: mockResponse.data, + }, + }, + ]); + }); + + it("should throw error when 'sessionId' is empty and session mode is 'existing'", async () => { + const nodeParameters = { + ...baseNodeParameters, + sessionId: '', + prompt: 'Extract data', + }; + + await expect( + getPaginated.execute.call(createMockExecuteFunction(nodeParameters), 0), + ).rejects.toThrow(ERROR_MESSAGES.SESSION_ID_REQUIRED); + }); + + it("should throw error when 'windowId' is empty and session mode is 'existing'", async () => { + const nodeParameters = { + ...baseNodeParameters, + windowId: '', + prompt: 'Extract data', + }; + + await expect( + getPaginated.execute.call(createMockExecuteFunction(nodeParameters), 0), + ).rejects.toThrow(ERROR_MESSAGES.WINDOW_ID_REQUIRED); + }); +}); diff --git a/packages/nodes-base/nodes/Airtop/test/node/extraction/query.test.ts b/packages/nodes-base/nodes/Airtop/test/node/extraction/query.test.ts new file mode 100644 index 0000000000..f3a0bd3f3b --- /dev/null +++ b/packages/nodes-base/nodes/Airtop/test/node/extraction/query.test.ts @@ -0,0 +1,196 @@ +import nock from 'nock'; + +import * as query from '../../../actions/extraction/query.operation'; +import { ERROR_MESSAGES } from '../../../constants'; +import * as GenericFunctions from '../../../GenericFunctions'; +import * as transport from '../../../transport'; +import { createMockExecuteFunction } from '../helpers'; + +const baseNodeParameters = { + resource: 'extraction', + operation: 'query', + sessionId: 'test-session-123', + windowId: 'win-123', + sessionMode: 'existing', +}; + +const mockResponse = { + data: { + modelResponse: { + answer: 'The page contains 5 products with prices ranging from $10.99 to $50.99', + }, + }, +}; + +const mockJsonSchema = + '{"type":"object","properties":{"productCount":{"type":"number"},"priceRange":{"type":"object"}}}'; + +jest.mock('../../../transport', () => { + const originalModule = jest.requireActual('../../../transport'); + return { + ...originalModule, + apiRequest: jest.fn(async function (method: string, endpoint: string) { + if (method === 'DELETE' && endpoint.includes('/sessions/')) { + return { status: 'success' }; + } + return mockResponse; + }), + }; +}); + +jest.mock('../../../GenericFunctions', () => { + const originalModule = jest.requireActual('../../../GenericFunctions'); + return { + ...originalModule, + createSessionAndWindow: jest.fn().mockImplementation(async () => { + return { + sessionId: 'new-session-456', + windowId: 'new-win-456', + }; + }), + shouldCreateNewSession: jest.fn().mockImplementation(function (this: any) { + const sessionMode = this.getNodeParameter('sessionMode', 0); + return sessionMode === 'new'; + }), + }; +}); + +describe('Test Airtop, query page operation', () => { + beforeAll(() => { + nock.disableNetConnect(); + }); + + afterAll(() => { + nock.restore(); + jest.unmock('../../../transport'); + jest.unmock('../../../GenericFunctions'); + }); + + afterEach(() => { + jest.clearAllMocks(); + }); + + it('should query the page with minimal parameters using existing session', async () => { + const nodeParameters = { + ...baseNodeParameters, + prompt: 'How many products are on the page and what is their price range?', + }; + + const result = await query.execute.call(createMockExecuteFunction(nodeParameters), 0); + + expect(GenericFunctions.shouldCreateNewSession).toHaveBeenCalledTimes(1); + expect(GenericFunctions.createSessionAndWindow).not.toHaveBeenCalled(); + expect(transport.apiRequest).toHaveBeenCalledTimes(1); + expect(transport.apiRequest).toHaveBeenCalledWith( + 'POST', + '/sessions/test-session-123/windows/win-123/page-query', + { + prompt: 'How many products are on the page and what is their price range?', + configuration: {}, + }, + ); + + expect(result).toEqual([ + { + json: { + sessionId: 'test-session-123', + windowId: 'win-123', + data: mockResponse.data, + }, + }, + ]); + }); + + it('should query the page with output schema using existing session', async () => { + const nodeParameters = { + ...baseNodeParameters, + prompt: 'How many products are on the page and what is their price range?', + additionalFields: { + outputSchema: mockJsonSchema, + }, + }; + + const result = await query.execute.call(createMockExecuteFunction(nodeParameters), 0); + + expect(GenericFunctions.shouldCreateNewSession).toHaveBeenCalledTimes(1); + expect(GenericFunctions.createSessionAndWindow).not.toHaveBeenCalled(); + expect(transport.apiRequest).toHaveBeenCalledTimes(1); + expect(transport.apiRequest).toHaveBeenCalledWith( + 'POST', + '/sessions/test-session-123/windows/win-123/page-query', + { + prompt: 'How many products are on the page and what is their price range?', + configuration: { + outputSchema: mockJsonSchema, + }, + }, + ); + + expect(result).toEqual([ + { + json: { + sessionId: 'test-session-123', + windowId: 'win-123', + data: mockResponse.data, + }, + }, + ]); + }); + + it('should query the page using a new session', async () => { + const nodeParameters = { + ...baseNodeParameters, + sessionMode: 'new', + url: 'https://example.com', + prompt: 'How many products are on the page and what is their price range?', + autoTerminateSession: true, + }; + + const result = await query.execute.call(createMockExecuteFunction(nodeParameters), 0); + + expect(GenericFunctions.shouldCreateNewSession).toHaveBeenCalledTimes(1); + expect(GenericFunctions.createSessionAndWindow).toHaveBeenCalledTimes(1); + expect(transport.apiRequest).toHaveBeenCalledTimes(2); // One for query, one for session deletion + expect(transport.apiRequest).toHaveBeenNthCalledWith( + 1, + 'POST', + '/sessions/new-session-456/windows/new-win-456/page-query', + { + prompt: 'How many products are on the page and what is their price range?', + configuration: {}, + }, + ); + + expect(result).toEqual([ + { + json: { + data: mockResponse.data, + }, + }, + ]); + }); + + it("should throw error when 'sessionId' is empty in 'existing' session mode", async () => { + const nodeParameters = { + ...baseNodeParameters, + sessionId: '', + prompt: 'Query data', + }; + + await expect(query.execute.call(createMockExecuteFunction(nodeParameters), 0)).rejects.toThrow( + ERROR_MESSAGES.SESSION_ID_REQUIRED, + ); + }); + + it("should throw error when 'windowId' is empty in 'existing' session mode", async () => { + const nodeParameters = { + ...baseNodeParameters, + windowId: '', + prompt: 'Query data', + }; + + await expect(query.execute.call(createMockExecuteFunction(nodeParameters), 0)).rejects.toThrow( + ERROR_MESSAGES.WINDOW_ID_REQUIRED, + ); + }); +}); diff --git a/packages/nodes-base/nodes/Airtop/test/node/extraction/scrape.test.ts b/packages/nodes-base/nodes/Airtop/test/node/extraction/scrape.test.ts new file mode 100644 index 0000000000..1b053539de --- /dev/null +++ b/packages/nodes-base/nodes/Airtop/test/node/extraction/scrape.test.ts @@ -0,0 +1,172 @@ +import nock from 'nock'; + +import * as scrape from '../../../actions/extraction/scrape.operation'; +import { ERROR_MESSAGES } from '../../../constants'; +import * as GenericFunctions from '../../../GenericFunctions'; +import * as transport from '../../../transport'; +import { createMockExecuteFunction } from '../helpers'; + +const baseNodeParameters = { + resource: 'extraction', + operation: 'scrape', + sessionId: 'test-session-123', + windowId: 'win-123', + sessionMode: 'existing', +}; + +const mockResponse = { + data: { + content: 'Scraped content', + }, +}; + +jest.mock('../../../transport', () => { + const originalModule = jest.requireActual('../../../transport'); + return { + ...originalModule, + apiRequest: jest.fn(async function (method: string, endpoint: string) { + if (method === 'DELETE' && endpoint.includes('/sessions/')) { + return { status: 'success' }; + } + return mockResponse; + }), + }; +}); + +jest.mock('../../../GenericFunctions', () => { + const originalModule = jest.requireActual('../../../GenericFunctions'); + return { + ...originalModule, + createSessionAndWindow: jest.fn().mockImplementation(async () => { + return { + sessionId: 'new-session-456', + windowId: 'new-win-456', + }; + }), + shouldCreateNewSession: jest.fn().mockImplementation(function (this: any) { + const sessionMode = this.getNodeParameter('sessionMode', 0); + return sessionMode === 'new'; + }), + }; +}); + +describe('Test Airtop, scrape operation', () => { + beforeAll(() => { + nock.disableNetConnect(); + }); + + afterAll(() => { + nock.restore(); + jest.unmock('../../../transport'); + jest.unmock('../../../GenericFunctions'); + }); + + afterEach(() => { + jest.clearAllMocks(); + }); + + it('should scrape content with minimal parameters using existing session', async () => { + const result = await scrape.execute.call(createMockExecuteFunction(baseNodeParameters), 0); + + expect(GenericFunctions.shouldCreateNewSession).toHaveBeenCalledTimes(1); + expect(GenericFunctions.createSessionAndWindow).not.toHaveBeenCalled(); + expect(transport.apiRequest).toHaveBeenCalledTimes(1); + expect(transport.apiRequest).toHaveBeenCalledWith( + 'POST', + '/sessions/test-session-123/windows/win-123/scrape-content', + {}, + ); + + expect(result).toEqual([ + { + json: { + sessionId: 'test-session-123', + windowId: 'win-123', + data: mockResponse.data, + }, + }, + ]); + }); + + it('should scrape content with additional parameters using existing session', async () => { + const nodeParameters = { + ...baseNodeParameters, + additionalFields: { + waitForSelector: '.product-list', + waitForTimeout: 5000, + }, + }; + + const result = await scrape.execute.call(createMockExecuteFunction(nodeParameters), 0); + + expect(GenericFunctions.shouldCreateNewSession).toHaveBeenCalledTimes(1); + expect(GenericFunctions.createSessionAndWindow).not.toHaveBeenCalled(); + expect(transport.apiRequest).toHaveBeenCalledTimes(1); + expect(transport.apiRequest).toHaveBeenCalledWith( + 'POST', + '/sessions/test-session-123/windows/win-123/scrape-content', + {}, + ); + + expect(result).toEqual([ + { + json: { + sessionId: 'test-session-123', + windowId: 'win-123', + data: mockResponse.data, + }, + }, + ]); + }); + + it('should scrape content using a new session', async () => { + const nodeParameters = { + ...baseNodeParameters, + sessionMode: 'new', + url: 'https://example.com', + autoTerminateSession: true, + }; + + const result = await scrape.execute.call(createMockExecuteFunction(nodeParameters), 0); + + expect(GenericFunctions.shouldCreateNewSession).toHaveBeenCalledTimes(1); + expect(GenericFunctions.createSessionAndWindow).toHaveBeenCalledTimes(1); + expect(transport.apiRequest).toHaveBeenCalledTimes(2); // One for scrape, one for session deletion + expect(transport.apiRequest).toHaveBeenNthCalledWith( + 1, + 'POST', + '/sessions/new-session-456/windows/new-win-456/scrape-content', + {}, + ); + + expect(result).toEqual([ + { + json: { + data: mockResponse.data, + }, + }, + ]); + }); + + it("should throw error when sessionId is empty in 'existing' session mode", async () => { + const nodeParameters = { + ...baseNodeParameters, + sessionId: '', + }; + + await expect(scrape.execute.call(createMockExecuteFunction(nodeParameters), 0)).rejects.toThrow( + ERROR_MESSAGES.SESSION_ID_REQUIRED, + ); + }); + + it("should throw error when windowId is empty in 'existing' session mode", async () => { + const nodeParameters = { + ...baseNodeParameters, + windowId: '', + }; + + await expect(scrape.execute.call(createMockExecuteFunction(nodeParameters), 0)).rejects.toThrow( + ERROR_MESSAGES.WINDOW_ID_REQUIRED, + ); + }); +}); diff --git a/packages/nodes-base/nodes/Airtop/test/node/helpers.ts b/packages/nodes-base/nodes/Airtop/test/node/helpers.ts new file mode 100644 index 0000000000..ba231bfe50 --- /dev/null +++ b/packages/nodes-base/nodes/Airtop/test/node/helpers.ts @@ -0,0 +1,57 @@ +import { get } from 'lodash'; +import { constructExecutionMetaData } from 'n8n-core'; +import type { + IDataObject, + IExecuteFunctions, + IGetNodeParameterOptions, + INode, + INodeExecutionData, +} from 'n8n-workflow'; + +export const node: INode = { + id: '1', + name: 'Airtop node', + typeVersion: 1, + type: 'n8n-nodes-base.airtop', + position: [10, 10], + parameters: {}, +}; + +export const createMockExecuteFunction = (nodeParameters: IDataObject) => { + const fakeExecuteFunction = { + getInputData(): INodeExecutionData[] { + return [{ json: {} }]; + }, + getNodeParameter( + parameterName: string, + _itemIndex: number, + fallbackValue?: IDataObject | undefined, + options?: IGetNodeParameterOptions | undefined, + ) { + const parameter = options?.extractValue ? `${parameterName}.value` : parameterName; + return get(nodeParameters, parameter, fallbackValue); + }, + getNode() { + return node; + }, + helpers: { + constructExecutionMetaData, + returnJsonArray: (data: IDataObject | IDataObject[]) => { + return [{ json: data }] as INodeExecutionData[]; + }, + prepareBinaryData: async (data: Buffer) => { + return { + mimeType: 'image/jpeg', + fileType: 'jpg', + fileName: 'screenshot.jpg', + data: data.toString('base64'), + }; + }, + }, + continueOnFail: () => false, + logger: { + info: () => {}, + }, + } as unknown as IExecuteFunctions; + return fakeExecuteFunction; +}; diff --git a/packages/nodes-base/nodes/Airtop/test/node/interaction/click.test.ts b/packages/nodes-base/nodes/Airtop/test/node/interaction/click.test.ts new file mode 100644 index 0000000000..5cada782be --- /dev/null +++ b/packages/nodes-base/nodes/Airtop/test/node/interaction/click.test.ts @@ -0,0 +1,137 @@ +import nock from 'nock'; + +import * as click from '../../../actions/interaction/click.operation'; +import { ERROR_MESSAGES } from '../../../constants'; +import * as transport from '../../../transport'; +import { createMockExecuteFunction } from '../helpers'; + +const baseNodeParameters = { + resource: 'interaction', + operation: 'click', + sessionId: 'test-session-123', + windowId: 'win-123', + elementDescription: 'the login button', + additionalFields: {}, +}; + +const mockResponse = { + success: true, + message: 'Click executed successfully', +}; + +jest.mock('../../../transport', () => { + const originalModule = jest.requireActual('../../../transport'); + return { + ...originalModule, + apiRequest: jest.fn(async function () { + return { + status: 'success', + data: mockResponse, + }; + }), + }; +}); + +describe('Test Airtop, click operation', () => { + beforeAll(() => { + nock.disableNetConnect(); + }); + + afterAll(() => { + nock.restore(); + jest.unmock('../../../transport'); + }); + + afterEach(() => { + jest.clearAllMocks(); + }); + + it('should execute click with minimal parameters', async () => { + const result = await click.execute.call(createMockExecuteFunction(baseNodeParameters), 0); + + expect(transport.apiRequest).toHaveBeenCalledTimes(1); + expect(transport.apiRequest).toHaveBeenCalledWith( + 'POST', + '/sessions/test-session-123/windows/win-123/click', + { + elementDescription: 'the login button', + configuration: {}, + }, + ); + + expect(result).toEqual([ + { + json: { + sessionId: baseNodeParameters.sessionId, + windowId: baseNodeParameters.windowId, + status: 'success', + data: mockResponse, + }, + }, + ]); + }); + + it("should throw error when 'elementDescription' parameter is empty", async () => { + const nodeParameters = { + ...baseNodeParameters, + elementDescription: '', + }; + const errorMessage = ERROR_MESSAGES.REQUIRED_PARAMETER.replace( + '{{field}}', + 'Element Description', + ); + + await expect(click.execute.call(createMockExecuteFunction(nodeParameters), 0)).rejects.toThrow( + errorMessage, + ); + }); + + it("should include 'visualScope' parameter when specified", async () => { + const nodeParameters = { + ...baseNodeParameters, + additionalFields: { + visualScope: 'viewport', + }, + }; + + await click.execute.call(createMockExecuteFunction(nodeParameters), 0); + + expect(transport.apiRequest).toHaveBeenCalledWith( + 'POST', + '/sessions/test-session-123/windows/win-123/click', + { + configuration: { + visualAnalysis: { + scope: 'viewport', + }, + }, + elementDescription: 'the login button', + }, + ); + }); + + it("should include 'waitForNavigation' parameter when specified", async () => { + const nodeParameters = { + ...baseNodeParameters, + additionalFields: { + waitForNavigation: 'load', + }, + }; + + await click.execute.call(createMockExecuteFunction(nodeParameters), 0); + + expect(transport.apiRequest).toHaveBeenCalledWith( + 'POST', + '/sessions/test-session-123/windows/win-123/click', + { + configuration: { + waitForNavigationConfig: { + waitUntil: 'load', + }, + }, + waitForNavigation: true, + elementDescription: 'the login button', + }, + ); + }); +}); diff --git a/packages/nodes-base/nodes/Airtop/test/node/interaction/helpers.test.ts b/packages/nodes-base/nodes/Airtop/test/node/interaction/helpers.test.ts new file mode 100644 index 0000000000..d4c3353f5a --- /dev/null +++ b/packages/nodes-base/nodes/Airtop/test/node/interaction/helpers.test.ts @@ -0,0 +1,77 @@ +import { constructInteractionRequest } from '../../../actions/interaction/helpers'; +import { createMockExecuteFunction } from '../helpers'; + +describe('Test Airtop interaction helpers', () => { + describe('constructInteractionRequest', () => { + it('should construct basic request with default values', () => { + const mockExecute = createMockExecuteFunction({ + additionalFields: {}, + }); + + const request = constructInteractionRequest.call(mockExecute, 0); + + expect(request).toEqual({ + configuration: {}, + }); + }); + + it("should include 'visualScope' parameter when specified", () => { + const mockExecute = createMockExecuteFunction({ + additionalFields: { + visualScope: 'viewport', + }, + }); + + const request = constructInteractionRequest.call(mockExecute, 0); + + expect(request).toEqual({ + configuration: { + visualAnalysis: { + scope: 'viewport', + }, + }, + }); + }); + + it("should include 'waitForNavigation' parameter when specified", () => { + const mockExecute = createMockExecuteFunction({ + additionalFields: { + waitForNavigation: 'load', + }, + }); + + const request = constructInteractionRequest.call(mockExecute, 0); + + expect(request).toEqual({ + configuration: { + waitForNavigationConfig: { + waitUntil: 'load', + }, + }, + waitForNavigation: true, + }); + }); + + it('should merge additional parameters', () => { + const mockExecute = createMockExecuteFunction({ + additionalFields: { + waitForNavigation: 'load', + }, + }); + + const request = constructInteractionRequest.call(mockExecute, 0, { + elementDescription: 'test element', + }); + + expect(request).toEqual({ + configuration: { + waitForNavigationConfig: { + waitUntil: 'load', + }, + }, + waitForNavigation: true, + elementDescription: 'test element', + }); + }); + }); +}); diff --git a/packages/nodes-base/nodes/Airtop/test/node/interaction/hover.test.ts b/packages/nodes-base/nodes/Airtop/test/node/interaction/hover.test.ts new file mode 100644 index 0000000000..0f91917259 --- /dev/null +++ b/packages/nodes-base/nodes/Airtop/test/node/interaction/hover.test.ts @@ -0,0 +1,137 @@ +import nock from 'nock'; + +import * as hover from '../../../actions/interaction/hover.operation'; +import { ERROR_MESSAGES } from '../../../constants'; +import * as transport from '../../../transport'; +import { createMockExecuteFunction } from '../helpers'; + +const baseNodeParameters = { + resource: 'interaction', + operation: 'hover', + sessionId: 'test-session-123', + windowId: 'win-123', + elementDescription: 'the user profile image', + additionalFields: {}, +}; + +const mockResponse = { + success: true, + message: 'Hover interaction executed successfully', +}; + +jest.mock('../../../transport', () => { + const originalModule = jest.requireActual('../../../transport'); + return { + ...originalModule, + apiRequest: jest.fn(async function () { + return { + status: 'success', + data: mockResponse, + }; + }), + }; +}); + +describe('Test Airtop, hover operation', () => { + beforeAll(() => { + nock.disableNetConnect(); + }); + + afterAll(() => { + nock.restore(); + jest.unmock('../../../transport'); + }); + + afterEach(() => { + jest.clearAllMocks(); + }); + + it('should execute hover with minimal parameters', async () => { + const result = await hover.execute.call(createMockExecuteFunction(baseNodeParameters), 0); + + expect(transport.apiRequest).toHaveBeenCalledTimes(1); + expect(transport.apiRequest).toHaveBeenCalledWith( + 'POST', + '/sessions/test-session-123/windows/win-123/hover', + { + configuration: {}, + elementDescription: 'the user profile image', + }, + ); + + expect(result).toEqual([ + { + json: { + sessionId: 'test-session-123', + windowId: 'win-123', + status: 'success', + data: mockResponse, + }, + }, + ]); + }); + + it("should throw error when 'elementDescription' parameter is empty", async () => { + const nodeParameters = { + ...baseNodeParameters, + elementDescription: '', + }; + const errorMessage = ERROR_MESSAGES.REQUIRED_PARAMETER.replace( + '{{field}}', + 'Element Description', + ); + + await expect(hover.execute.call(createMockExecuteFunction(nodeParameters), 0)).rejects.toThrow( + errorMessage, + ); + }); + + it("should include 'visualScope' parameter when specified", async () => { + const nodeParameters = { + ...baseNodeParameters, + additionalFields: { + visualScope: 'viewport', + }, + }; + + await hover.execute.call(createMockExecuteFunction(nodeParameters), 0); + + expect(transport.apiRequest).toHaveBeenCalledWith( + 'POST', + '/sessions/test-session-123/windows/win-123/hover', + { + configuration: { + visualAnalysis: { + scope: 'viewport', + }, + }, + elementDescription: 'the user profile image', + }, + ); + }); + + it("should include 'waitForNavigation' parameter when specified", async () => { + const nodeParameters = { + ...baseNodeParameters, + additionalFields: { + waitForNavigation: 'load', + }, + }; + + await hover.execute.call(createMockExecuteFunction(nodeParameters), 0); + + expect(transport.apiRequest).toHaveBeenCalledWith( + 'POST', + '/sessions/test-session-123/windows/win-123/hover', + { + configuration: { + waitForNavigationConfig: { + waitUntil: 'load', + }, + }, + waitForNavigation: true, + elementDescription: 'the user profile image', + }, + ); + }); +}); diff --git a/packages/nodes-base/nodes/Airtop/test/node/interaction/type.test.ts b/packages/nodes-base/nodes/Airtop/test/node/interaction/type.test.ts new file mode 100644 index 0000000000..b5471dfaec --- /dev/null +++ b/packages/nodes-base/nodes/Airtop/test/node/interaction/type.test.ts @@ -0,0 +1,181 @@ +import nock from 'nock'; + +import * as type from '../../../actions/interaction/type.operation'; +import { ERROR_MESSAGES } from '../../../constants'; +import * as transport from '../../../transport'; +import { createMockExecuteFunction } from '../helpers'; + +const baseNodeParameters = { + resource: 'interaction', + operation: 'type', + sessionId: 'test-session-123', + windowId: 'win-123', + text: 'Hello World', + pressEnterKey: false, + elementDescription: '', + additionalFields: {}, +}; + +const mockResponse = { + success: true, + message: 'Text typed successfully', +}; + +jest.mock('../../../transport', () => { + const originalModule = jest.requireActual('../../../transport'); + return { + ...originalModule, + apiRequest: jest.fn(async function () { + return { + status: 'success', + data: mockResponse, + }; + }), + }; +}); + +describe('Test Airtop, type operation', () => { + beforeAll(() => { + nock.disableNetConnect(); + }); + + afterAll(() => { + nock.restore(); + jest.unmock('../../../transport'); + }); + + afterEach(() => { + jest.clearAllMocks(); + }); + + it('should execute type with minimal parameters', async () => { + const result = await type.execute.call(createMockExecuteFunction(baseNodeParameters), 0); + + expect(transport.apiRequest).toHaveBeenCalledTimes(1); + expect(transport.apiRequest).toHaveBeenCalledWith( + 'POST', + '/sessions/test-session-123/windows/win-123/type', + { + configuration: {}, + text: 'Hello World', + pressEnterKey: false, + elementDescription: '', + }, + ); + + expect(result).toEqual([ + { + json: { + sessionId: baseNodeParameters.sessionId, + windowId: baseNodeParameters.windowId, + status: 'success', + data: mockResponse, + }, + }, + ]); + }); + + it("should throw error when 'text' parameter is empty", async () => { + const nodeParameters = { + ...baseNodeParameters, + text: '', + }; + + await expect(type.execute.call(createMockExecuteFunction(nodeParameters), 0)).rejects.toThrow( + ERROR_MESSAGES.REQUIRED_PARAMETER.replace('{{field}}', 'Text'), + ); + }); + + it("should include 'elementDescription' parameter when specified", async () => { + const nodeParameters = { + ...baseNodeParameters, + elementDescription: 'the search box', + }; + + await type.execute.call(createMockExecuteFunction(nodeParameters), 0); + + expect(transport.apiRequest).toHaveBeenCalledWith( + 'POST', + '/sessions/test-session-123/windows/win-123/type', + { + configuration: {}, + text: 'Hello World', + pressEnterKey: false, + elementDescription: 'the search box', + }, + ); + }); + + it("should include 'pressEnterKey' parameter when specified", async () => { + const nodeParameters = { + ...baseNodeParameters, + pressEnterKey: true, + }; + + await type.execute.call(createMockExecuteFunction(nodeParameters), 0); + + expect(transport.apiRequest).toHaveBeenCalledWith( + 'POST', + '/sessions/test-session-123/windows/win-123/type', + { + configuration: {}, + text: 'Hello World', + pressEnterKey: true, + elementDescription: '', + }, + ); + }); + + it("should include 'waitForNavigation' parameter when specified", async () => { + const nodeParameters = { + ...baseNodeParameters, + additionalFields: { + waitForNavigation: 'load', + }, + }; + + await type.execute.call(createMockExecuteFunction(nodeParameters), 0); + + expect(transport.apiRequest).toHaveBeenCalledWith( + 'POST', + '/sessions/test-session-123/windows/win-123/type', + { + configuration: { + waitForNavigationConfig: { + waitUntil: 'load', + }, + }, + waitForNavigation: true, + text: 'Hello World', + pressEnterKey: false, + elementDescription: '', + }, + ); + }); + + it("should include 'visualScope' parameter when specified", async () => { + const nodeParameters = { + ...baseNodeParameters, + additionalFields: { + visualScope: 'viewport', + }, + }; + + await type.execute.call(createMockExecuteFunction(nodeParameters), 0); + + expect(transport.apiRequest).toHaveBeenCalledWith( + 'POST', + '/sessions/test-session-123/windows/win-123/type', + { + configuration: { + visualAnalysis: { + scope: 'viewport', + }, + }, + text: 'Hello World', + pressEnterKey: false, + elementDescription: '', + }, + ); + }); +}); diff --git a/packages/nodes-base/nodes/Airtop/test/node/session/create.test.ts b/packages/nodes-base/nodes/Airtop/test/node/session/create.test.ts new file mode 100644 index 0000000000..04a7d34127 --- /dev/null +++ b/packages/nodes-base/nodes/Airtop/test/node/session/create.test.ts @@ -0,0 +1,202 @@ +import * as create from '../../../actions/session/create.operation'; +import { ERROR_MESSAGES } from '../../../constants'; +import * as transport from '../../../transport'; +import { createMockExecuteFunction } from '../helpers'; + +jest.mock('../../../transport', () => { + const originalModule = jest.requireActual('../../../transport'); + return { + ...originalModule, + apiRequest: jest.fn(async function () { + return { + sessionId: 'test-session-123', + status: 'success', + }; + }), + }; +}); + +describe('Test Airtop, session create operation', () => { + afterAll(() => { + jest.unmock('../../../transport'); + }); + + afterEach(() => { + jest.clearAllMocks(); + }); + + it('should create a session with minimal parameters', async () => { + const nodeParameters = { + resource: 'session', + operation: 'create', + profileName: 'test-profile', + timeoutMinutes: 10, + saveProfileOnTermination: false, + proxy: 'none', + }; + + const result = await create.execute.call(createMockExecuteFunction(nodeParameters), 0); + + expect(transport.apiRequest).toHaveBeenCalledTimes(1); + expect(transport.apiRequest).toHaveBeenCalledWith( + 'POST', + 'https://portal-api.airtop.ai/integrations/v1/no-code/create-session', + { + configuration: { + profileName: 'test-profile', + timeoutMinutes: 10, + proxy: false, + }, + }, + ); + + expect(result).toEqual([ + { + json: { + sessionId: 'test-session-123', + }, + }, + ]); + }); + + it('should create a session with save profile enabled', async () => { + const nodeParameters = { + resource: 'session', + operation: 'create', + profileName: 'test-profile', + timeoutMinutes: 15, + saveProfileOnTermination: true, + proxy: 'none', + }; + + const result = await create.execute.call(createMockExecuteFunction(nodeParameters), 0); + + expect(transport.apiRequest).toHaveBeenCalledTimes(2); + expect(transport.apiRequest).toHaveBeenNthCalledWith( + 1, + 'POST', + 'https://portal-api.airtop.ai/integrations/v1/no-code/create-session', + { + configuration: { + profileName: 'test-profile', + timeoutMinutes: 15, + proxy: false, + }, + }, + ); + expect(transport.apiRequest).toHaveBeenNthCalledWith( + 2, + 'PUT', + '/sessions/test-session-123/save-profile-on-termination/test-profile', + ); + + expect(result).toEqual([ + { + json: { + sessionId: 'test-session-123', + }, + }, + ]); + }); + + it('should create a session with integrated proxy', async () => { + const nodeParameters = { + resource: 'session', + operation: 'create', + profileName: 'test-profile', + timeoutMinutes: 10, + saveProfileOnTermination: false, + proxy: 'integrated', + }; + + const result = await create.execute.call(createMockExecuteFunction(nodeParameters), 0); + + expect(transport.apiRequest).toHaveBeenCalledTimes(1); + expect(transport.apiRequest).toHaveBeenCalledWith( + 'POST', + 'https://portal-api.airtop.ai/integrations/v1/no-code/create-session', + { + configuration: { + profileName: 'test-profile', + timeoutMinutes: 10, + proxy: true, + }, + }, + ); + + expect(result).toEqual([ + { + json: { + sessionId: 'test-session-123', + }, + }, + ]); + }); + + it('should create a session with custom proxy', async () => { + const nodeParameters = { + resource: 'session', + operation: 'create', + profileName: 'test-profile', + timeoutMinutes: 10, + saveProfileOnTermination: false, + proxy: 'custom', + proxyUrl: 'http://proxy.example.com:8080', + }; + + const result = await create.execute.call(createMockExecuteFunction(nodeParameters), 0); + + expect(transport.apiRequest).toHaveBeenCalledTimes(1); + expect(transport.apiRequest).toHaveBeenCalledWith( + 'POST', + 'https://portal-api.airtop.ai/integrations/v1/no-code/create-session', + { + configuration: { + profileName: 'test-profile', + timeoutMinutes: 10, + proxy: 'http://proxy.example.com:8080', + }, + }, + ); + + expect(result).toEqual([ + { + json: { + sessionId: 'test-session-123', + }, + }, + ]); + }); + + it('should throw error when custom proxy URL is invalid', async () => { + const nodeParameters = { + resource: 'session', + operation: 'create', + profileName: 'test-profile', + timeoutMinutes: 10, + saveProfileOnTermination: false, + proxy: 'custom', + proxyUrl: 'invalid-url', + }; + + await expect(create.execute.call(createMockExecuteFunction(nodeParameters), 0)).rejects.toThrow( + ERROR_MESSAGES.PROXY_URL_INVALID, + ); + }); + + it('should throw error when custom proxy URL is empty', async () => { + const nodeParameters = { + resource: 'session', + operation: 'create', + profileName: 'test-profile', + timeoutMinutes: 10, + saveProfileOnTermination: false, + proxy: 'custom', + proxyUrl: '', + }; + + await expect(create.execute.call(createMockExecuteFunction(nodeParameters), 0)).rejects.toThrow( + ERROR_MESSAGES.PROXY_URL_REQUIRED, + ); + }); +}); diff --git a/packages/nodes-base/nodes/Airtop/test/node/session/save.test.ts b/packages/nodes-base/nodes/Airtop/test/node/session/save.test.ts new file mode 100644 index 0000000000..115354e06a --- /dev/null +++ b/packages/nodes-base/nodes/Airtop/test/node/session/save.test.ts @@ -0,0 +1,103 @@ +import * as save from '../../../actions/session/save.operation'; +import { ERROR_MESSAGES } from '../../../constants'; +import * as transport from '../../../transport'; +import { createMockExecuteFunction } from '../helpers'; + +jest.mock('../../../transport', () => { + const originalModule = jest.requireActual('../../../transport'); + return { + ...originalModule, + apiRequest: jest.fn(async function () { + return { + status: 'success', + message: 'Profile will be saved on session termination', + }; + }), + }; +}); + +const baseParameters = { + resource: 'session', + operation: 'save', + sessionId: 'test-session-123', + profileName: 'test-profile', +}; + +describe('Test Airtop, session save operation', () => { + afterAll(() => { + jest.unmock('../../../transport'); + }); + + afterEach(() => { + jest.clearAllMocks(); + }); + + it('should save a profile on session termination successfully', async () => { + const nodeParameters = { + ...baseParameters, + }; + + const result = await save.execute.call(createMockExecuteFunction(nodeParameters), 0); + + expect(transport.apiRequest).toHaveBeenCalledTimes(1); + expect(transport.apiRequest).toHaveBeenCalledWith( + 'PUT', + '/sessions/test-session-123/save-profile-on-termination/test-profile', + ); + + expect(result).toEqual([ + { + json: { + sessionId: 'test-session-123', + profileName: 'test-profile', + status: 'success', + message: 'Profile will be saved on session termination', + }, + }, + ]); + }); + + it('should throw error when sessionId is empty', async () => { + const nodeParameters = { + ...baseParameters, + sessionId: '', + }; + + await expect(save.execute.call(createMockExecuteFunction(nodeParameters), 0)).rejects.toThrow( + ERROR_MESSAGES.SESSION_ID_REQUIRED, + ); + }); + + it('should throw error when sessionId is whitespace', async () => { + const nodeParameters = { + ...baseParameters, + sessionId: ' ', + }; + + await expect(save.execute.call(createMockExecuteFunction(nodeParameters), 0)).rejects.toThrow( + ERROR_MESSAGES.SESSION_ID_REQUIRED, + ); + }); + + it('should throw error when profileName is empty', async () => { + const nodeParameters = { + ...baseParameters, + profileName: '', + }; + + await expect(save.execute.call(createMockExecuteFunction(nodeParameters), 0)).rejects.toThrow( + "Please fill the 'Profile Name' parameter", + ); + }); + + it('should throw error when profileName is whitespace', async () => { + const nodeParameters = { + ...baseParameters, + profileName: ' ', + }; + + await expect(save.execute.call(createMockExecuteFunction(nodeParameters), 0)).rejects.toThrow( + "Please fill the 'Profile Name' parameter", + ); + }); +}); diff --git a/packages/nodes-base/nodes/Airtop/test/node/session/terminate.test.ts b/packages/nodes-base/nodes/Airtop/test/node/session/terminate.test.ts new file mode 100644 index 0000000000..ede8cd0757 --- /dev/null +++ b/packages/nodes-base/nodes/Airtop/test/node/session/terminate.test.ts @@ -0,0 +1,71 @@ +import * as terminate from '../../../actions/session/terminate.operation'; +import { ERROR_MESSAGES } from '../../../constants'; +import * as transport from '../../../transport'; +import { createMockExecuteFunction } from '../helpers'; + +jest.mock('../../../transport', () => { + const originalModule = jest.requireActual('../../../transport'); + return { + ...originalModule, + apiRequest: jest.fn(async function () { + return { + status: 'success', + }; + }), + }; +}); + +describe('Test Airtop, session terminate operation', () => { + afterAll(() => { + jest.unmock('../../../transport'); + }); + + afterEach(() => { + jest.clearAllMocks(); + }); + + it('should terminate a session successfully', async () => { + const nodeParameters = { + resource: 'session', + operation: 'terminate', + sessionId: 'test-session-123', + }; + + const result = await terminate.execute.call(createMockExecuteFunction(nodeParameters), 0); + + expect(transport.apiRequest).toHaveBeenCalledTimes(1); + expect(transport.apiRequest).toHaveBeenCalledWith('DELETE', '/sessions/test-session-123'); + + expect(result).toEqual([ + { + json: { + success: true, + }, + }, + ]); + }); + + it('should throw error when sessionId is empty', async () => { + const nodeParameters = { + resource: 'session', + operation: 'terminate', + sessionId: '', + }; + + await expect( + terminate.execute.call(createMockExecuteFunction(nodeParameters), 0), + ).rejects.toThrow(ERROR_MESSAGES.SESSION_ID_REQUIRED); + }); + + it('should throw error when sessionId is whitespace', async () => { + const nodeParameters = { + resource: 'session', + operation: 'terminate', + sessionId: ' ', + }; + + await expect( + terminate.execute.call(createMockExecuteFunction(nodeParameters), 0), + ).rejects.toThrow(ERROR_MESSAGES.SESSION_ID_REQUIRED); + }); +}); diff --git a/packages/nodes-base/nodes/Airtop/test/node/window/close.test.ts b/packages/nodes-base/nodes/Airtop/test/node/window/close.test.ts new file mode 100644 index 0000000000..5d023d2c06 --- /dev/null +++ b/packages/nodes-base/nodes/Airtop/test/node/window/close.test.ts @@ -0,0 +1,92 @@ +import nock from 'nock'; + +import * as close from '../../../actions/window/close.operation'; +import { ERROR_MESSAGES } from '../../../constants'; +import * as transport from '../../../transport'; +import { createMockExecuteFunction } from '../helpers'; + +jest.mock('../../../transport', () => { + const originalModule = jest.requireActual('../../../transport'); + return { + ...originalModule, + apiRequest: jest.fn(async function () { + return { + status: 'success', + data: { + closed: true, + }, + }; + }), + }; +}); + +describe('Test Airtop, window close operation', () => { + beforeAll(() => { + nock.disableNetConnect(); + }); + + afterAll(() => { + nock.restore(); + jest.unmock('../../../transport'); + }); + + afterEach(() => { + jest.clearAllMocks(); + }); + + it('should close a window successfully', async () => { + const nodeParameters = { + resource: 'window', + operation: 'close', + sessionId: 'test-session-123', + windowId: 'win-123', + }; + + const result = await close.execute.call(createMockExecuteFunction(nodeParameters), 0); + + expect(transport.apiRequest).toHaveBeenCalledTimes(1); + expect(transport.apiRequest).toHaveBeenCalledWith( + 'DELETE', + '/sessions/test-session-123/windows/win-123', + ); + + expect(result).toEqual([ + { + json: { + sessionId: 'test-session-123', + windowId: 'win-123', + status: 'success', + data: { + closed: true, + }, + }, + }, + ]); + }); + + it('should throw error when sessionId is empty', async () => { + const nodeParameters = { + resource: 'window', + operation: 'close', + sessionId: '', + windowId: 'win-123', + }; + + await expect(close.execute.call(createMockExecuteFunction(nodeParameters), 0)).rejects.toThrow( + ERROR_MESSAGES.SESSION_ID_REQUIRED, + ); + }); + + it('should throw error when windowId is empty', async () => { + const nodeParameters = { + resource: 'window', + operation: 'close', + sessionId: 'test-session-123', + windowId: '', + }; + + await expect(close.execute.call(createMockExecuteFunction(nodeParameters), 0)).rejects.toThrow( + ERROR_MESSAGES.WINDOW_ID_REQUIRED, + ); + }); +}); diff --git a/packages/nodes-base/nodes/Airtop/test/node/window/create.test.ts b/packages/nodes-base/nodes/Airtop/test/node/window/create.test.ts new file mode 100644 index 0000000000..8dedc11452 --- /dev/null +++ b/packages/nodes-base/nodes/Airtop/test/node/window/create.test.ts @@ -0,0 +1,308 @@ +import nock from 'nock'; + +import * as create from '../../../actions/window/create.operation'; +import { ERROR_MESSAGES } from '../../../constants'; +import * as transport from '../../../transport'; +import { createMockExecuteFunction } from '../helpers'; + +const baseNodeParameters = { + resource: 'window', + operation: 'create', + sessionId: 'test-session-123', + url: 'https://example.com', + getLiveView: false, + disableResize: false, + additionalFields: {}, +}; + +jest.mock('../../../transport', () => { + const originalModule = jest.requireActual('../../../transport'); + return { + ...originalModule, + apiRequest: jest.fn(async function (method: string) { + if (method === 'GET') { + return { + status: 'success', + data: { + liveViewUrl: 'https://live.airtop.ai/123-abcd', + }, + }; + } + return { + status: 'success', + data: { + windowId: 'win-123', + }, + }; + }), + }; +}); + +describe('Test Airtop, window create operation', () => { + beforeAll(() => { + nock.disableNetConnect(); + }); + + afterAll(() => { + nock.restore(); + jest.unmock('../../../transport'); + }); + + afterEach(() => { + jest.clearAllMocks(); + }); + + it('should create a window with minimal parameters', async () => { + const result = await create.execute.call(createMockExecuteFunction(baseNodeParameters), 0); + + expect(transport.apiRequest).toHaveBeenCalledTimes(1); + expect(transport.apiRequest).toHaveBeenCalledWith( + 'POST', + '/sessions/test-session-123/windows', + { + url: 'https://example.com', + }, + ); + + expect(result).toEqual([ + { + json: { + sessionId: baseNodeParameters.sessionId, + windowId: 'win-123', + status: 'success', + data: { + windowId: 'win-123', + }, + }, + }, + ]); + }); + + it('should create a window with live view', async () => { + const nodeParameters = { + ...baseNodeParameters, + getLiveView: true, + }; + + const result = await create.execute.call(createMockExecuteFunction(nodeParameters), 0); + + expect(transport.apiRequest).toHaveBeenCalledTimes(2); + expect(transport.apiRequest).toHaveBeenNthCalledWith( + 1, + 'POST', + '/sessions/test-session-123/windows', + { + url: 'https://example.com', + }, + ); + expect(transport.apiRequest).toHaveBeenNthCalledWith( + 2, + 'GET', + '/sessions/test-session-123/windows/win-123', + undefined, + {}, + ); + + expect(result).toEqual([ + { + json: { + sessionId: baseNodeParameters.sessionId, + windowId: 'win-123', + status: 'success', + data: { + liveViewUrl: 'https://live.airtop.ai/123-abcd', + }, + }, + }, + ]); + }); + + it('should create a window with live view and disabled resize', async () => { + const nodeParameters = { + ...baseNodeParameters, + getLiveView: true, + disableResize: true, + }; + + const result = await create.execute.call(createMockExecuteFunction(nodeParameters), 0); + + expect(transport.apiRequest).toHaveBeenCalledTimes(2); + expect(transport.apiRequest).toHaveBeenNthCalledWith( + 1, + 'POST', + '/sessions/test-session-123/windows', + { + url: 'https://example.com', + }, + ); + expect(transport.apiRequest).toHaveBeenNthCalledWith( + 2, + 'GET', + '/sessions/test-session-123/windows/win-123', + undefined, + { disableResize: true }, + ); + + expect(result).toEqual([ + { + json: { + sessionId: baseNodeParameters.sessionId, + windowId: 'win-123', + status: 'success', + data: { + liveViewUrl: 'https://live.airtop.ai/123-abcd', + }, + }, + }, + ]); + }); + + it('should create a window with live view and navigation bar', async () => { + const nodeParameters = { + ...baseNodeParameters, + getLiveView: true, + includeNavigationBar: true, + }; + + const result = await create.execute.call(createMockExecuteFunction(nodeParameters), 0); + + expect(transport.apiRequest).toHaveBeenCalledTimes(2); + expect(transport.apiRequest).toHaveBeenNthCalledWith( + 1, + 'POST', + '/sessions/test-session-123/windows', + { + url: 'https://example.com', + }, + ); + expect(transport.apiRequest).toHaveBeenNthCalledWith( + 2, + 'GET', + '/sessions/test-session-123/windows/win-123', + undefined, + { includeNavigationBar: true }, + ); + + expect(result).toEqual([ + { + json: { + sessionId: baseNodeParameters.sessionId, + windowId: 'win-123', + status: 'success', + data: { + liveViewUrl: 'https://live.airtop.ai/123-abcd', + }, + }, + }, + ]); + }); + + it('should create a window with live view and screen resolution', async () => { + const nodeParameters = { + ...baseNodeParameters, + getLiveView: true, + screenResolution: '1280x720', + }; + + const result = await create.execute.call(createMockExecuteFunction(nodeParameters), 0); + + expect(transport.apiRequest).toHaveBeenCalledTimes(2); + expect(transport.apiRequest).toHaveBeenNthCalledWith( + 1, + 'POST', + '/sessions/test-session-123/windows', + { + url: 'https://example.com', + }, + ); + expect(transport.apiRequest).toHaveBeenNthCalledWith( + 2, + 'GET', + '/sessions/test-session-123/windows/win-123', + undefined, + { screenResolution: '1280x720' }, + ); + + expect(result).toEqual([ + { + json: { + sessionId: baseNodeParameters.sessionId, + windowId: 'win-123', + status: 'success', + data: { + liveViewUrl: 'https://live.airtop.ai/123-abcd', + }, + }, + }, + ]); + }); + + it('should create a window with all live view options', async () => { + const nodeParameters = { + ...baseNodeParameters, + getLiveView: true, + includeNavigationBar: true, + screenResolution: '1920x1080', + disableResize: true, + }; + + const result = await create.execute.call(createMockExecuteFunction(nodeParameters), 0); + + expect(transport.apiRequest).toHaveBeenCalledTimes(2); + expect(transport.apiRequest).toHaveBeenNthCalledWith( + 1, + 'POST', + '/sessions/test-session-123/windows', + { + url: 'https://example.com', + }, + ); + expect(transport.apiRequest).toHaveBeenNthCalledWith( + 2, + 'GET', + '/sessions/test-session-123/windows/win-123', + undefined, + { + includeNavigationBar: true, + screenResolution: '1920x1080', + disableResize: true, + }, + ); + + expect(result).toEqual([ + { + json: { + sessionId: baseNodeParameters.sessionId, + windowId: 'win-123', + status: 'success', + data: { + liveViewUrl: 'https://live.airtop.ai/123-abcd', + }, + }, + }, + ]); + }); + + it("should throw error when 'sessionId' parameter is empty", async () => { + const nodeParameters = { + ...baseNodeParameters, + sessionId: '', + }; + + await expect(create.execute.call(createMockExecuteFunction(nodeParameters), 0)).rejects.toThrow( + ERROR_MESSAGES.SESSION_ID_REQUIRED, + ); + }); + + it('should throw error when screen resolution format is invalid', async () => { + const nodeParameters = { + ...baseNodeParameters, + getLiveView: true, + screenResolution: 'invalid-format', + }; + + await expect(create.execute.call(createMockExecuteFunction(nodeParameters), 0)).rejects.toThrow( + ERROR_MESSAGES.SCREEN_RESOLUTION_INVALID, + ); + }); +}); diff --git a/packages/nodes-base/nodes/Airtop/test/node/window/load.test.ts b/packages/nodes-base/nodes/Airtop/test/node/window/load.test.ts new file mode 100644 index 0000000000..289e393627 --- /dev/null +++ b/packages/nodes-base/nodes/Airtop/test/node/window/load.test.ts @@ -0,0 +1,115 @@ +import nock from 'nock'; + +import * as load from '../../../actions/window/load.operation'; +import { ERROR_MESSAGES } from '../../../constants'; +import * as transport from '../../../transport'; +import { createMockExecuteFunction } from '../helpers'; + +const baseNodeParameters = { + resource: 'window', + operation: 'load', + sessionId: 'test-session-123', + windowId: 'win-123', + url: 'https://example.com', + additionalFields: {}, +}; + +const mockResponse = { + success: true, + message: 'Page loaded successfully', +}; + +jest.mock('../../../transport', () => { + const originalModule = jest.requireActual('../../../transport'); + return { + ...originalModule, + apiRequest: jest.fn(async function () { + return { + status: 'success', + data: mockResponse, + }; + }), + }; +}); + +describe('Test Airtop, window load operation', () => { + beforeAll(() => { + nock.disableNetConnect(); + }); + + afterAll(() => { + nock.restore(); + jest.unmock('../../../transport'); + }); + + afterEach(() => { + jest.clearAllMocks(); + }); + + it('should load URL with minimal parameters', async () => { + const result = await load.execute.call(createMockExecuteFunction(baseNodeParameters), 0); + + expect(transport.apiRequest).toHaveBeenCalledTimes(1); + expect(transport.apiRequest).toHaveBeenCalledWith( + 'POST', + '/sessions/test-session-123/windows/win-123', + { + url: 'https://example.com', + }, + ); + + expect(result).toEqual([ + { + json: { + sessionId: baseNodeParameters.sessionId, + windowId: baseNodeParameters.windowId, + status: 'success', + data: mockResponse, + }, + }, + ]); + }); + + it('should throw error when URL is empty', async () => { + const nodeParameters = { + ...baseNodeParameters, + url: '', + }; + const errorMessage = ERROR_MESSAGES.REQUIRED_PARAMETER.replace('{{field}}', 'URL'); + + await expect(load.execute.call(createMockExecuteFunction(nodeParameters), 0)).rejects.toThrow( + errorMessage, + ); + }); + + it('should throw error when URL is invalid', async () => { + const nodeParameters = { + ...baseNodeParameters, + url: 'not-a-valid-url', + }; + + await expect(load.execute.call(createMockExecuteFunction(nodeParameters), 0)).rejects.toThrow( + ERROR_MESSAGES.URL_INVALID, + ); + }); + + it("should include 'waitUntil' parameter when specified", async () => { + const nodeParameters = { + ...baseNodeParameters, + additionalFields: { + waitUntil: 'domContentLoaded', + }, + }; + + await load.execute.call(createMockExecuteFunction(nodeParameters), 0); + + expect(transport.apiRequest).toHaveBeenCalledWith( + 'POST', + '/sessions/test-session-123/windows/win-123', + { + url: 'https://example.com', + waitUntil: 'domContentLoaded', + }, + ); + }); +}); diff --git a/packages/nodes-base/nodes/Airtop/test/node/window/takeScreenshot.test.ts b/packages/nodes-base/nodes/Airtop/test/node/window/takeScreenshot.test.ts new file mode 100644 index 0000000000..1d4f2337c0 --- /dev/null +++ b/packages/nodes-base/nodes/Airtop/test/node/window/takeScreenshot.test.ts @@ -0,0 +1,141 @@ +import * as takeScreenshot from '../../../actions/window/takeScreenshot.operation'; +import { ERROR_MESSAGES } from '../../../constants'; +import * as GenericFunctions from '../../../GenericFunctions'; +import * as transport from '../../../transport'; +import { createMockExecuteFunction } from '../helpers'; + +const baseNodeParameters = { + resource: 'window', + operation: 'takeScreenshot', + sessionId: 'test-session-123', + windowId: 'win-123', +}; + +const mockResponse = { + meta: { + screenshots: [{ dataUrl: 'base64-encoded-image-data' }], + }, +}; + +const mockBinaryBuffer = Buffer.from('mock-binary-data'); + +const expectedBinaryResult = { + binary: { + data: { + mimeType: 'image/jpeg', + fileType: 'jpg', + fileName: 'screenshot.jpg', + data: mockBinaryBuffer.toString('base64'), + }, + }, +}; + +jest.mock('../../../transport', () => { + const originalModule = jest.requireActual('../../../transport'); + return { + ...originalModule, + apiRequest: jest.fn(async function () { + return { + status: 'success', + ...mockResponse, + }; + }), + }; +}); + +jest.mock('../../../GenericFunctions', () => { + const originalModule = jest.requireActual('../../../GenericFunctions'); + return { + ...originalModule, + convertScreenshotToBinary: jest.fn(() => mockBinaryBuffer), + }; +}); + +describe('Test Airtop, take screenshot operation', () => { + afterAll(() => { + jest.unmock('../../../transport'); + jest.unmock('../../../GenericFunctions'); + }); + + afterEach(() => { + jest.clearAllMocks(); + }); + + it('should take screenshot successfully', async () => { + const result = await takeScreenshot.execute.call( + createMockExecuteFunction({ ...baseNodeParameters }), + 0, + ); + + expect(transport.apiRequest).toHaveBeenCalledTimes(1); + expect(transport.apiRequest).toHaveBeenCalledWith( + 'POST', + '/sessions/test-session-123/windows/win-123/screenshot', + ); + + expect(result).toEqual([ + { + json: { + sessionId: 'test-session-123', + windowId: 'win-123', + status: 'success', + ...mockResponse, + }, + ...expectedBinaryResult, + }, + ]); + }); + + it('should transform screenshot to binary data', async () => { + const result = await takeScreenshot.execute.call( + createMockExecuteFunction({ + ...baseNodeParameters, + }), + 0, + ); + + expect(transport.apiRequest).toHaveBeenCalledTimes(1); + expect(transport.apiRequest).toHaveBeenCalledWith( + 'POST', + '/sessions/test-session-123/windows/win-123/screenshot', + ); + + expect(GenericFunctions.convertScreenshotToBinary).toHaveBeenCalledWith( + mockResponse.meta.screenshots[0], + ); + + expect(result).toEqual([ + { + json: { + sessionId: 'test-session-123', + windowId: 'win-123', + status: 'success', + ...mockResponse, + }, + ...expectedBinaryResult, + }, + ]); + }); + + it('should throw error when sessionId is empty', async () => { + const nodeParameters = { + ...baseNodeParameters, + sessionId: '', + }; + + await expect( + takeScreenshot.execute.call(createMockExecuteFunction(nodeParameters), 0), + ).rejects.toThrow(ERROR_MESSAGES.SESSION_ID_REQUIRED); + }); + + it('should throw error when windowId is empty', async () => { + const nodeParameters = { + ...baseNodeParameters, + windowId: '', + }; + + await expect( + takeScreenshot.execute.call(createMockExecuteFunction(nodeParameters), 0), + ).rejects.toThrow(ERROR_MESSAGES.WINDOW_ID_REQUIRED); + }); +}); diff --git a/packages/nodes-base/nodes/Airtop/test/session-utils.test.ts b/packages/nodes-base/nodes/Airtop/test/session-utils.test.ts new file mode 100644 index 0000000000..a42c655721 --- /dev/null +++ b/packages/nodes-base/nodes/Airtop/test/session-utils.test.ts @@ -0,0 +1,201 @@ +import type { IExecuteFunctions } from 'n8n-workflow'; + +import { createMockExecuteFunction } from './node/helpers'; +import { SESSION_MODE } from '../actions/common/fields'; +import { executeRequestWithSessionManagement } from '../actions/common/session.utils'; +import * as transport from '../transport'; + +jest.mock('../transport', () => { + const originalModule = jest.requireActual('../transport'); + return { + ...originalModule, + apiRequest: jest.fn(async () => { + return { + success: true, + }; + }), + }; +}); + +jest.mock('../GenericFunctions', () => ({ + shouldCreateNewSession: jest.fn(function (this: IExecuteFunctions, index: number) { + const sessionMode = this.getNodeParameter('sessionMode', index); + return sessionMode === SESSION_MODE.NEW; + }), + createSessionAndWindow: jest.fn(async () => ({ + sessionId: 'new-session-123', + windowId: 'new-window-123', + })), + validateSessionAndWindowId: jest.fn(() => ({ + sessionId: 'existing-session-123', + windowId: 'existing-window-123', + })), + validateAirtopApiResponse: jest.fn(), +})); + +describe('executeRequestWithSessionManagement', () => { + afterAll(() => { + jest.unmock('../transport'); + }); + + afterEach(() => { + jest.clearAllMocks(); + }); + + describe("When 'sessionMode' is 'new'", () => { + it("should create a new session and window when 'sessionMode' is 'new'", async () => { + const nodeParameters = { + sessionMode: SESSION_MODE.NEW, + url: 'https://example.com', + autoTerminateSession: true, + }; + + const result = await executeRequestWithSessionManagement.call( + createMockExecuteFunction(nodeParameters), + 0, + { + method: 'POST', + path: '/sessions/{sessionId}/windows/{windowId}/action', + body: {}, + }, + ); + + expect(result).toEqual([ + { + json: { + success: true, + }, + }, + ]); + }); + + it("should not terminate session when 'autoTerminateSession' is false", async () => { + const nodeParameters = { + sessionMode: SESSION_MODE.NEW, + url: 'https://example.com', + autoTerminateSession: false, + }; + + const result = await executeRequestWithSessionManagement.call( + createMockExecuteFunction(nodeParameters), + 0, + { + method: 'POST', + path: '/sessions/{sessionId}/windows/{windowId}/action', + body: {}, + }, + ); + + expect(transport.apiRequest).not.toHaveBeenCalledWith( + 'DELETE', + '/sessions/existing-session-123', + ); + + expect(result).toEqual([ + { + json: { + sessionId: 'new-session-123', + windowId: 'new-window-123', + success: true, + }, + }, + ]); + }); + + it("should terminate session when 'autoTerminateSession' is true", async () => { + const nodeParameters = { + sessionMode: SESSION_MODE.NEW, + url: 'https://example.com', + autoTerminateSession: true, + }; + + await executeRequestWithSessionManagement.call(createMockExecuteFunction(nodeParameters), 0, { + method: 'POST', + path: '/sessions/{sessionId}/windows/{windowId}/action', + body: {}, + }); + + expect(transport.apiRequest).toHaveBeenNthCalledWith( + 2, + 'DELETE', + '/sessions/new-session-123', + ); + }); + + it("should call the operation passed in the 'request' parameter", async () => { + const nodeParameters = { + sessionMode: SESSION_MODE.NEW, + url: 'https://example.com', + autoTerminateSession: true, + }; + + await executeRequestWithSessionManagement.call(createMockExecuteFunction(nodeParameters), 0, { + method: 'POST', + path: '/sessions/{sessionId}/windows/{windowId}/action', + body: { + operation: 'test-operation', + }, + }); + + expect(transport.apiRequest).toHaveBeenNthCalledWith( + 1, + 'POST', + '/sessions/new-session-123/windows/new-window-123/action', + { + operation: 'test-operation', + }, + ); + }); + }); + + describe("When 'sessionMode' is 'existing'", () => { + it('should not create a new session and window', async () => { + const nodeParameters = { + sessionMode: SESSION_MODE.EXISTING, + url: 'https://example.com', + sessionId: 'existing-session-123', + windowId: 'existing-window-123', + }; + + await executeRequestWithSessionManagement.call(createMockExecuteFunction(nodeParameters), 0, { + method: 'POST', + path: '/sessions/{sessionId}/windows/{windowId}/action', + body: {}, + }); + + expect(transport.apiRequest).toHaveBeenCalledTimes(1); + + expect(transport.apiRequest).toHaveBeenCalledWith( + 'POST', + '/sessions/existing-session-123/windows/existing-window-123/action', + {}, + ); + }); + + it("should call the operation passed in the 'request' parameter", async () => { + const nodeParameters = { + sessionMode: SESSION_MODE.EXISTING, + url: 'https://example.com', + sessionId: 'existing-session-123', + windowId: 'existing-window-123', + }; + + await executeRequestWithSessionManagement.call(createMockExecuteFunction(nodeParameters), 0, { + method: 'POST', + path: '/sessions/{sessionId}/windows/{windowId}/action', + body: { + operation: 'test-operation', + }, + }); + + expect(transport.apiRequest).toHaveBeenNthCalledWith( + 1, + 'POST', + '/sessions/existing-session-123/windows/existing-window-123/action', + { + operation: 'test-operation', + }, + ); + }); + }); +}); diff --git a/packages/nodes-base/nodes/Airtop/test/utils.test.ts b/packages/nodes-base/nodes/Airtop/test/utils.test.ts new file mode 100644 index 0000000000..08f744a1c2 --- /dev/null +++ b/packages/nodes-base/nodes/Airtop/test/utils.test.ts @@ -0,0 +1,435 @@ +import { NodeApiError } from 'n8n-workflow'; + +import { createMockExecuteFunction } from './node/helpers'; +import { ERROR_MESSAGES } from '../constants'; +import { + createSessionAndWindow, + validateProfileName, + validateTimeoutMinutes, + validateSaveProfileOnTermination, + validateSessionAndWindowId, + validateAirtopApiResponse, + validateSessionId, + validateUrl, + validateRequiredStringField, + shouldCreateNewSession, + convertScreenshotToBinary, +} from '../GenericFunctions'; +import type * as transport from '../transport'; + +jest.mock('../transport', () => { + const originalModule = jest.requireActual('../transport'); + return { + ...originalModule, + apiRequest: jest.fn(async (method: string, endpoint: string) => { + // create session + if (endpoint.includes('/create-session')) { + return { sessionId: 'new-session-123' }; + } + + // create window + if (method === 'POST' && endpoint.endsWith('/windows')) { + return { data: { windowId: 'new-window-123' } }; + } + + return { + success: true, + }; + }), + }; +}); + +describe('Test convertScreenshotToBinary', () => { + it('should convert base64 screenshot data to buffer', () => { + const mockScreenshot = { + dataUrl: '', // "Hello World" in base64 + }; + + const result = convertScreenshotToBinary(mockScreenshot); + + expect(Buffer.isBuffer(result)).toBe(true); + expect(result.toString()).toBe('Hello World'); + }); + + it('should handle empty base64 data', () => { + const mockScreenshot = { + dataUrl: 'data:image/jpeg;base64,', + }; + + const result = convertScreenshotToBinary(mockScreenshot); + + expect(Buffer.isBuffer(result)).toBe(true); + expect(result.length).toBe(0); + }); +}); + +describe('Test Airtop utils', () => { + describe('validateRequiredStringField', () => { + it('should validate non-empty string field', () => { + const nodeParameters = { + testField: 'test-value', + }; + + const result = validateRequiredStringField.call( + createMockExecuteFunction(nodeParameters), + 0, + 'testField', + 'Test Field', + ); + expect(result).toBe('test-value'); + }); + + it('should trim whitespace from string field', () => { + const nodeParameters = { + testField: ' test-value ', + }; + + const result = validateRequiredStringField.call( + createMockExecuteFunction(nodeParameters), + 0, + 'testField', + 'Test Field', + ); + expect(result).toBe('test-value'); + }); + + it('should throw error for empty string field', () => { + const nodeParameters = { + testField: '', + }; + + expect(() => + validateRequiredStringField.call( + createMockExecuteFunction(nodeParameters), + 0, + 'testField', + 'Test Field', + ), + ).toThrow(ERROR_MESSAGES.REQUIRED_PARAMETER.replace('{{field}}', 'Test Field')); + }); + + it('should throw error for whitespace-only string field', () => { + const nodeParameters = { + testField: ' ', + }; + + expect(() => + validateRequiredStringField.call( + createMockExecuteFunction(nodeParameters), + 0, + 'testField', + 'Test Field', + ), + ).toThrow(ERROR_MESSAGES.REQUIRED_PARAMETER.replace('{{field}}', 'Test Field')); + }); + }); + + describe('validateProfileName', () => { + it('should validate valid profile names', () => { + const nodeParameters = { + profileName: 'test-profile-123', + }; + + const result = validateProfileName.call(createMockExecuteFunction(nodeParameters), 0); + expect(result).toBe('test-profile-123'); + }); + + it('should allow empty profile name', () => { + const nodeParameters = { + profileName: '', + }; + + const result = validateProfileName.call(createMockExecuteFunction(nodeParameters), 0); + expect(result).toBe(''); + }); + + it('should throw error for invalid profile name', () => { + const nodeParameters = { + profileName: 'test@profile#123', + }; + + expect(() => validateProfileName.call(createMockExecuteFunction(nodeParameters), 0)).toThrow( + ERROR_MESSAGES.PROFILE_NAME_INVALID, + ); + }); + }); + + describe('validateTimeoutMinutes', () => { + it('should validate valid timeout', () => { + const nodeParameters = { + timeoutMinutes: 10, + }; + + const result = validateTimeoutMinutes.call(createMockExecuteFunction(nodeParameters), 0); + expect(result).toBe(10); + }); + + it('should throw error for timeout below minimum', () => { + const nodeParameters = { + timeoutMinutes: 0, + }; + + expect(() => + validateTimeoutMinutes.call(createMockExecuteFunction(nodeParameters), 0), + ).toThrow(ERROR_MESSAGES.TIMEOUT_MINUTES_INVALID); + }); + + it('should throw error for timeout above maximum', () => { + const nodeParameters = { + timeoutMinutes: 10081, + }; + + expect(() => + validateTimeoutMinutes.call(createMockExecuteFunction(nodeParameters), 0), + ).toThrow(ERROR_MESSAGES.TIMEOUT_MINUTES_INVALID); + }); + }); + + describe('validateSaveProfileOnTermination', () => { + it('should validate when save profile is false', () => { + const nodeParameters = { + saveProfileOnTermination: false, + }; + + const result = validateSaveProfileOnTermination.call( + createMockExecuteFunction(nodeParameters), + 0, + '', + ); + expect(result).toBe(false); + }); + + it('should validate when save profile is true with profile name', () => { + const nodeParameters = { + saveProfileOnTermination: true, + }; + + const result = validateSaveProfileOnTermination.call( + createMockExecuteFunction(nodeParameters), + 0, + 'test-profile', + ); + expect(result).toBe(true); + }); + + it('should throw error when save profile is true without profile name', () => { + const nodeParameters = { + saveProfileOnTermination: true, + }; + + expect(() => + validateSaveProfileOnTermination.call(createMockExecuteFunction(nodeParameters), 0, ''), + ).toThrow(ERROR_MESSAGES.PROFILE_NAME_REQUIRED); + }); + }); + + describe('validateSessionId', () => { + it('should validate session ID', () => { + const nodeParameters = { + sessionId: 'test-session-123', + }; + + const result = validateSessionId.call(createMockExecuteFunction(nodeParameters), 0); + expect(result).toBe('test-session-123'); + }); + + it('should throw error for empty session ID', () => { + const nodeParameters = { + sessionId: '', + }; + + expect(() => validateSessionId.call(createMockExecuteFunction(nodeParameters), 0)).toThrow( + ERROR_MESSAGES.SESSION_ID_REQUIRED, + ); + }); + + it('should trim whitespace from session ID', () => { + const nodeParameters = { + sessionId: ' test-session-123 ', + }; + + const result = validateSessionId.call(createMockExecuteFunction(nodeParameters), 0); + expect(result).toBe('test-session-123'); + }); + }); + + describe('validateSessionAndWindowId', () => { + it('should validate session and window IDs', () => { + const nodeParameters = { + sessionId: 'test-session-123', + windowId: 'win-123', + }; + + const result = validateSessionAndWindowId.call(createMockExecuteFunction(nodeParameters), 0); + expect(result).toEqual({ + sessionId: 'test-session-123', + windowId: 'win-123', + }); + }); + + it('should throw error for empty session ID', () => { + const nodeParameters = { + sessionId: '', + windowId: 'win-123', + }; + + expect(() => + validateSessionAndWindowId.call(createMockExecuteFunction(nodeParameters), 0), + ).toThrow(ERROR_MESSAGES.SESSION_ID_REQUIRED); + }); + + it('should throw error for empty window ID', () => { + const nodeParameters = { + sessionId: 'test-session-123', + windowId: '', + }; + + expect(() => + validateSessionAndWindowId.call(createMockExecuteFunction(nodeParameters), 0), + ).toThrow(ERROR_MESSAGES.WINDOW_ID_REQUIRED); + }); + + it('should trim whitespace from IDs', () => { + const nodeParameters = { + sessionId: ' test-session-123 ', + windowId: ' win-123 ', + }; + + const result = validateSessionAndWindowId.call(createMockExecuteFunction(nodeParameters), 0); + expect(result).toEqual({ + sessionId: 'test-session-123', + windowId: 'win-123', + }); + }); + }); + + describe('validateUrl', () => { + it('should validate valid URL', () => { + const nodeParameters = { + url: 'https://example.com', + }; + + const result = validateUrl.call(createMockExecuteFunction(nodeParameters), 0); + expect(result).toBe('https://example.com'); + }); + + it('should throw error for invalid URL', () => { + const nodeParameters = { + url: 'invalid-url', + }; + + expect(() => validateUrl.call(createMockExecuteFunction(nodeParameters), 0)).toThrow( + ERROR_MESSAGES.URL_INVALID, + ); + }); + + it('should return empty string for empty URL', () => { + const nodeParameters = { + url: '', + }; + + const result = validateUrl.call(createMockExecuteFunction(nodeParameters), 0); + expect(result).toBe(''); + }); + + it('should throw error for URL without http or https', () => { + const nodeParameters = { + url: 'example.com', + }; + + expect(() => validateUrl.call(createMockExecuteFunction(nodeParameters), 0)).toThrow( + ERROR_MESSAGES.URL_INVALID, + ); + }); + }); + + describe('validateAirtopApiResponse', () => { + const mockNode = { + id: '1', + name: 'Airtop node', + type: 'n8n-nodes-base.airtop', + typeVersion: 1, + position: [0, 0] as [number, number], + parameters: {}, + }; + + it('should not throw error for valid response', () => { + const response = { + status: 'success', + data: {}, + meta: {}, + errors: [], + warnings: [], + }; + + expect(() => validateAirtopApiResponse(mockNode, response)).not.toThrow(); + }); + + it('should throw error for response with errors', () => { + const response = { + status: 'error', + data: {}, + meta: {}, + errors: [{ message: 'Error 1' }, { message: 'Error 2' }], + warnings: [], + }; + + const expectedError = new NodeApiError(mockNode, { message: 'Error 1\nError 2' }); + expect(() => validateAirtopApiResponse(mockNode, response)).toThrow(expectedError); + }); + }); + + describe('shouldCreateNewSession', () => { + it("should return true when 'sessionMode' is 'new'", () => { + const nodeParameters = { + sessionMode: 'new', + }; + + const result = shouldCreateNewSession.call(createMockExecuteFunction(nodeParameters), 0); + expect(result).toBe(true); + }); + + it("should return false when 'sessionMode' is 'existing'", () => { + const nodeParameters = { + sessionMode: 'existing', + }; + + const result = shouldCreateNewSession.call(createMockExecuteFunction(nodeParameters), 0); + expect(result).toBe(false); + }); + + it("should return false when 'sessionMode' is empty", () => { + const nodeParameters = { + sessionMode: '', + }; + + const result = shouldCreateNewSession.call(createMockExecuteFunction(nodeParameters), 0); + expect(result).toBe(false); + }); + + it("should return false when 'sessionMode' is not set", () => { + const nodeParameters = {}; + + const result = shouldCreateNewSession.call(createMockExecuteFunction(nodeParameters), 0); + expect(result).toBe(false); + }); + }); + + describe('createSessionAndWindow', () => { + it("should create a new session and window when sessionMode is 'new'", async () => { + const nodeParameters = { + sessionMode: 'new', + url: 'https://example.com', + }; + + const result = await createSessionAndWindow.call( + createMockExecuteFunction(nodeParameters), + 0, + ); + expect(result).toEqual({ + sessionId: 'new-session-123', + windowId: 'new-window-123', + }); + }); + }); +}); diff --git a/packages/nodes-base/nodes/Airtop/transport/index.ts b/packages/nodes-base/nodes/Airtop/transport/index.ts new file mode 100644 index 0000000000..a4b1cbad0e --- /dev/null +++ b/packages/nodes-base/nodes/Airtop/transport/index.ts @@ -0,0 +1,39 @@ +import type { + IDataObject, + IExecuteFunctions, + ILoadOptionsFunctions, + IHttpRequestMethods, + IHttpRequestOptions, +} from 'n8n-workflow'; + +import type { IAirtopResponse } from './types'; +import { BASE_URL } from '../constants'; + +export async function apiRequest( + this: IExecuteFunctions | ILoadOptionsFunctions, + method: IHttpRequestMethods, + endpoint: string, + body: IDataObject = {}, + query: IDataObject = {}, +) { + const options: IHttpRequestOptions = { + headers: { + 'Content-Type': 'application/json', + }, + method, + body, + qs: query, + url: endpoint.startsWith('http') ? endpoint : `${BASE_URL}${endpoint}`, + json: true, + }; + + if (Object.keys(body).length === 0) { + delete options.body; + } + + return (await this.helpers.httpRequestWithAuthentication.call( + this, + 'airtopApi', + options, + )) as IAirtopResponse; +} diff --git a/packages/nodes-base/nodes/Airtop/transport/types.ts b/packages/nodes-base/nodes/Airtop/transport/types.ts new file mode 100644 index 0000000000..18139fba22 --- /dev/null +++ b/packages/nodes-base/nodes/Airtop/transport/types.ts @@ -0,0 +1,29 @@ +import type { IDataObject } from 'n8n-workflow'; + +export interface IAirtopResponse extends IDataObject { + sessionId?: string; + data: IDataObject & { + modelResponse?: string; + }; + meta: IDataObject & { + status?: string; + screenshots?: Array<{ dataUrl: string }>; + }; + errors: IDataObject[]; + warnings: IDataObject[]; +} + +export interface IAirtopInteractionRequest extends IDataObject { + text?: string; + waitForNavigation?: boolean; + elementDescription?: string; + pressEnterKey?: boolean; + configuration: { + visualAnalysis?: { + scope: string; + }; + waitForNavigationConfig?: { + waitUntil: string; + }; + }; +} diff --git a/packages/nodes-base/package.json b/packages/nodes-base/package.json index da00638650..8700afb86a 100644 --- a/packages/nodes-base/package.json +++ b/packages/nodes-base/package.json @@ -31,6 +31,7 @@ "dist/credentials/AirtableApi.credentials.js", "dist/credentials/AirtableOAuth2Api.credentials.js", "dist/credentials/AirtableTokenApi.credentials.js", + "dist/credentials/AirtopApi.credentials.js", "dist/credentials/AlienVaultApi.credentials.js", "dist/credentials/Amqp.credentials.js", "dist/credentials/ApiTemplateIoApi.credentials.js", @@ -413,6 +414,7 @@ "dist/nodes/AgileCrm/AgileCrm.node.js", "dist/nodes/Airtable/Airtable.node.js", "dist/nodes/Airtable/AirtableTrigger.node.js", + "dist/nodes/Airtop/Airtop.node.js", "dist/nodes/AiTransform/AiTransform.node.js", "dist/nodes/Amqp/Amqp.node.js", "dist/nodes/Amqp/AmqpTrigger.node.js",