n8n-nodes-cala 0.2.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/LICENSE ADDED
@@ -0,0 +1,21 @@
1
+ MIT License
2
+
3
+ Copyright (c) 2026 Cala AI
4
+
5
+ Permission is hereby granted, free of charge, to any person obtaining a copy
6
+ of this software and associated documentation files (the "Software"), to deal
7
+ in the Software without restriction, including without limitation the rights
8
+ to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9
+ copies of the Software, and to permit persons to whom the Software is
10
+ furnished to do so, subject to the following conditions:
11
+
12
+ The above copyright notice and this permission notice shall be included in all
13
+ copies or substantial portions of the Software.
14
+
15
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16
+ IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17
+ FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18
+ AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19
+ LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20
+ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21
+ SOFTWARE.
package/README.md ADDED
@@ -0,0 +1,122 @@
1
+ # n8n-nodes-cala
2
+
3
+ This is an n8n community node for [Cala AI](https://cala.ai) - a platform that transforms internet information into structured, trustworthy context for AI agents.
4
+
5
+ ## Installation
6
+
7
+ ### n8n Cloud (Verified Nodes)
8
+
9
+ 1. Go to **Settings** → **Community Nodes**
10
+ 2. Click **Install**
11
+ 3. Enter `n8n-nodes-cala`
12
+ 4. Click **Install**
13
+
14
+ ### Self-hosted n8n
15
+
16
+ ```bash
17
+ npm install n8n-nodes-cala
18
+ ```
19
+
20
+ ## Credentials
21
+
22
+ You need a Cala API key to use this node:
23
+
24
+ 1. Sign up at [console.cala.ai](https://console.cala.ai)
25
+ 2. Create an API key
26
+ 3. In n8n, create new credentials of type **Cala API**
27
+ 4. Enter your API key
28
+
29
+ ## Operations
30
+
31
+ ### Knowledge
32
+
33
+ | Operation | Description |
34
+ |-----------|-------------|
35
+ | **Search** | Search verified knowledge using natural language queries |
36
+
37
+ ### Example
38
+
39
+ **Input:** "How many students were enrolled at MIT in 2024?"
40
+
41
+ **Output:**
42
+ ```json
43
+ {
44
+ "content": "In 2024, MIT had approximately 11,800 students enrolled, including 4,600 undergraduate and 7,200 graduate students.",
45
+ "explainability": [
46
+ {
47
+ "content": "MIT enrollment data shows total student population across undergraduate and graduate programs",
48
+ "references": ["a1b2c3d4-5678-90ab-cdef-123456789abc"]
49
+ }
50
+ ],
51
+ "context": [
52
+ {
53
+ "id": "a1b2c3d4-5678-90ab-cdef-123456789abc",
54
+ "content": "In 2024, the Massachusetts Institute of Technology enrolled approximately 11,800 students: 4,600 undergraduates and 7,200 graduate students.",
55
+ "origins": [
56
+ {
57
+ "source": { "name": "MIT", "url": "https://mit.edu" },
58
+ "document": { "name": "MIT Facts", "url": "https://mit.edu/about" }
59
+ }
60
+ ]
61
+ }
62
+ ],
63
+ "entities": [
64
+ { "id": 1, "name": "MIT", "entity_type": "ORGANIZATION" },
65
+ { "id": 2, "name": "Massachusetts Institute of Technology", "entity_type": "ORGANIZATION" }
66
+ ]
67
+ }
68
+ ```
69
+
70
+ ## Resources
71
+
72
+ - [Cala Documentation](https://docs.cala.ai)
73
+ - [Cala Console](https://console.cala.ai)
74
+ - [n8n Community Nodes Documentation](https://docs.n8n.io/integrations/community-nodes/)
75
+
76
+ ## Development
77
+
78
+ ### Requirements
79
+
80
+ - Node.js >= 22
81
+ - pnpm >= 10
82
+
83
+ ### Project Structure
84
+
85
+ ```text
86
+ cala-n8n/
87
+ ├── credentials/
88
+ │ └── CalaApi.credentials.ts # API credentials definition
89
+ ├── nodes/
90
+ │ └── Cala/
91
+ │ ├── Cala.node.ts # Main node logic
92
+ │ └── cala.svg # Node icon
93
+ ├── dist/ # Compiled output
94
+ ├── Makefile # Development commands
95
+ ├── package.json
96
+ ├── tsconfig.json
97
+ └── gulpfile.js
98
+ ```
99
+
100
+ ### Quick Start
101
+
102
+ ```bash
103
+ make start # Build + start n8n at http://localhost:5678
104
+ make stop # Stop n8n
105
+ ```
106
+
107
+ ### Commands
108
+
109
+ | Command | Description |
110
+ | ------- | ----------- |
111
+ | `make install` | Install dependencies |
112
+ | `make build` | Build the project |
113
+ | `make dev` | Development mode (watch) |
114
+ | `make start` | Start n8n locally |
115
+ | `make stop` | Stop n8n |
116
+ | `make publish` | Publish to npm |
117
+ | `make verify` | Run n8n linter |
118
+ | `make clean` | Remove build artifacts |
119
+
120
+ ## License
121
+
122
+ MIT
@@ -0,0 +1,7 @@
1
+ import { ICredentialType, INodeProperties } from 'n8n-workflow';
2
+ export declare class CalaApi implements ICredentialType {
3
+ name: string;
4
+ displayName: string;
5
+ documentationUrl: string;
6
+ properties: INodeProperties[];
7
+ }
@@ -0,0 +1,32 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.CalaApi = void 0;
4
+ class CalaApi {
5
+ constructor() {
6
+ this.name = 'calaApi';
7
+ this.displayName = 'Cala API';
8
+ this.documentationUrl = 'https://docs.cala.ai';
9
+ this.properties = [
10
+ {
11
+ displayName: 'API Key',
12
+ name: 'apiKey',
13
+ type: 'string',
14
+ typeOptions: {
15
+ password: true,
16
+ },
17
+ default: '',
18
+ required: true,
19
+ description: 'Your Cala API key',
20
+ },
21
+ {
22
+ displayName: 'Base URL',
23
+ name: 'baseUrl',
24
+ type: 'string',
25
+ default: 'https://api.cala.ai',
26
+ required: true,
27
+ description: 'The base URL of the Cala API',
28
+ },
29
+ ];
30
+ }
31
+ }
32
+ exports.CalaApi = CalaApi;
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,32 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ const CalaApi_credentials_1 = require("./CalaApi.credentials");
4
+ describe('CalaApi Credentials', () => {
5
+ let credentials;
6
+ beforeEach(() => {
7
+ credentials = new CalaApi_credentials_1.CalaApi();
8
+ });
9
+ it('should have correct name', () => {
10
+ expect(credentials.name).toBe('calaApi');
11
+ });
12
+ it('should have correct display name', () => {
13
+ expect(credentials.displayName).toBe('Cala API');
14
+ });
15
+ it('should have documentation URL', () => {
16
+ expect(credentials.documentationUrl).toBe('https://docs.cala.ai');
17
+ });
18
+ it('should have API key property', () => {
19
+ var _a;
20
+ const apiKeyProp = credentials.properties.find(p => p.name === 'apiKey');
21
+ expect(apiKeyProp).toBeDefined();
22
+ expect(apiKeyProp === null || apiKeyProp === void 0 ? void 0 : apiKeyProp.type).toBe('string');
23
+ expect(apiKeyProp === null || apiKeyProp === void 0 ? void 0 : apiKeyProp.required).toBe(true);
24
+ expect((_a = apiKeyProp === null || apiKeyProp === void 0 ? void 0 : apiKeyProp.typeOptions) === null || _a === void 0 ? void 0 : _a.password).toBe(true);
25
+ });
26
+ it('should have base URL property with default value', () => {
27
+ const baseUrlProp = credentials.properties.find(p => p.name === 'baseUrl');
28
+ expect(baseUrlProp).toBeDefined();
29
+ expect(baseUrlProp === null || baseUrlProp === void 0 ? void 0 : baseUrlProp.type).toBe('string');
30
+ expect(baseUrlProp === null || baseUrlProp === void 0 ? void 0 : baseUrlProp.default).toBe('https://api.cala.ai');
31
+ });
32
+ });
@@ -0,0 +1,5 @@
1
+ import { IExecuteFunctions, INodeExecutionData, INodeType, INodeTypeDescription } from 'n8n-workflow';
2
+ export declare class Cala implements INodeType {
3
+ description: INodeTypeDescription;
4
+ execute(this: IExecuteFunctions): Promise<INodeExecutionData[][]>;
5
+ }
@@ -0,0 +1,112 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.Cala = void 0;
4
+ class Cala {
5
+ constructor() {
6
+ this.description = {
7
+ displayName: 'Cala',
8
+ name: 'cala',
9
+ icon: 'file:cala.svg',
10
+ group: ['transform'],
11
+ version: 1,
12
+ subtitle: '={{$parameter["operation"] + ": " + $parameter["resource"]}}',
13
+ description: 'Search verified knowledge with Cala AI',
14
+ defaults: {
15
+ name: 'Cala',
16
+ },
17
+ inputs: ['main'],
18
+ outputs: ['main'],
19
+ credentials: [
20
+ {
21
+ name: 'calaApi',
22
+ required: true,
23
+ },
24
+ ],
25
+ properties: [
26
+ {
27
+ displayName: 'Resource',
28
+ name: 'resource',
29
+ type: 'options',
30
+ noDataExpression: true,
31
+ options: [
32
+ {
33
+ name: 'Knowledge',
34
+ value: 'knowledge',
35
+ },
36
+ ],
37
+ default: 'knowledge',
38
+ },
39
+ {
40
+ displayName: 'Operation',
41
+ name: 'operation',
42
+ type: 'options',
43
+ noDataExpression: true,
44
+ displayOptions: {
45
+ show: {
46
+ resource: ['knowledge'],
47
+ },
48
+ },
49
+ options: [
50
+ {
51
+ name: 'Search',
52
+ value: 'search',
53
+ description: 'Search verified knowledge',
54
+ action: 'Search verified knowledge',
55
+ },
56
+ ],
57
+ default: 'search',
58
+ },
59
+ {
60
+ displayName: 'Query',
61
+ name: 'query',
62
+ type: 'string',
63
+ required: true,
64
+ displayOptions: {
65
+ show: {
66
+ resource: ['knowledge'],
67
+ operation: ['search'],
68
+ },
69
+ },
70
+ default: '',
71
+ placeholder: 'What is the refund policy?',
72
+ description: 'The search query to find verified knowledge',
73
+ },
74
+ ],
75
+ };
76
+ }
77
+ async execute() {
78
+ const items = this.getInputData();
79
+ const returnData = [];
80
+ const credentials = await this.getCredentials('calaApi');
81
+ const baseUrl = credentials.baseUrl.replace(/\/$/, '');
82
+ const apiKey = credentials.apiKey;
83
+ for (let i = 0; i < items.length; i++) {
84
+ const resource = this.getNodeParameter('resource', i);
85
+ const operation = this.getNodeParameter('operation', i);
86
+ if (resource === 'knowledge' && operation === 'search') {
87
+ const query = this.getNodeParameter('query', i);
88
+ const headers = {
89
+ 'Content-Type': 'application/json',
90
+ };
91
+ if (apiKey) {
92
+ headers['X-API-KEY'] = apiKey;
93
+ }
94
+ const response = await this.helpers.httpRequest({
95
+ method: 'POST',
96
+ url: `${baseUrl}/v1/knowledge/search`,
97
+ headers,
98
+ body: {
99
+ input: query,
100
+ },
101
+ json: true,
102
+ });
103
+ returnData.push({
104
+ json: response,
105
+ pairedItem: { item: i },
106
+ });
107
+ }
108
+ }
109
+ return [returnData];
110
+ }
111
+ }
112
+ exports.Cala = Cala;
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,128 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ const Cala_node_1 = require("./Cala.node");
4
+ describe('Cala Node', () => {
5
+ let node;
6
+ beforeEach(() => {
7
+ node = new Cala_node_1.Cala();
8
+ });
9
+ describe('execute', () => {
10
+ const createExecutionContext = ({ baseUrl = 'https://api.cala.ai/', apiKey = 'test-key', resource = 'knowledge', operation = 'search', query = 'What is Cala?', response = { content: 'ok' }, } = {}) => {
11
+ const httpRequest = jest.fn(async () => response);
12
+ return {
13
+ getInputData: jest.fn(() => [{ json: { input: 'one' } }]),
14
+ getNodeParameter: jest.fn((name, _index) => {
15
+ if (name === 'resource') {
16
+ return resource;
17
+ }
18
+ if (name === 'operation') {
19
+ return operation;
20
+ }
21
+ if (name === 'query') {
22
+ return query;
23
+ }
24
+ throw new Error(`Unexpected parameter name: ${name}`);
25
+ }),
26
+ getCredentials: jest.fn(async () => ({ baseUrl, apiKey })),
27
+ helpers: {
28
+ httpRequest,
29
+ },
30
+ };
31
+ };
32
+ it('should call Cala API with normalized base URL', async () => {
33
+ const context = createExecutionContext();
34
+ const result = await node.execute.call(context);
35
+ expect(context.helpers.httpRequest).toHaveBeenCalledWith({
36
+ method: 'POST',
37
+ url: 'https://api.cala.ai/v1/knowledge/search',
38
+ headers: {
39
+ 'Content-Type': 'application/json',
40
+ 'X-API-KEY': 'test-key',
41
+ },
42
+ body: {
43
+ input: 'What is Cala?',
44
+ },
45
+ json: true,
46
+ });
47
+ expect(result).toEqual([[{ json: { content: 'ok' }, pairedItem: { item: 0 } }]]);
48
+ });
49
+ it('should omit API key header when missing', async () => {
50
+ const context = createExecutionContext({ apiKey: '' });
51
+ await node.execute.call(context);
52
+ expect(context.helpers.httpRequest).toHaveBeenCalledWith({
53
+ method: 'POST',
54
+ url: 'https://api.cala.ai/v1/knowledge/search',
55
+ headers: {
56
+ 'Content-Type': 'application/json',
57
+ },
58
+ body: {
59
+ input: 'What is Cala?',
60
+ },
61
+ json: true,
62
+ });
63
+ });
64
+ it('should process multiple items', async () => {
65
+ const queries = ['Query 1', 'Query 2', 'Query 3'];
66
+ const httpRequest = jest.fn()
67
+ .mockResolvedValueOnce({ answer: 'Answer 1' })
68
+ .mockResolvedValueOnce({ answer: 'Answer 2' })
69
+ .mockResolvedValueOnce({ answer: 'Answer 3' });
70
+ const context = {
71
+ getInputData: jest.fn(() => queries.map(q => ({ json: { query: q } }))),
72
+ getNodeParameter: jest.fn((name, index) => {
73
+ if (name === 'resource')
74
+ return 'knowledge';
75
+ if (name === 'operation')
76
+ return 'search';
77
+ if (name === 'query')
78
+ return queries[index];
79
+ throw new Error(`Unexpected parameter: ${name}`);
80
+ }),
81
+ getCredentials: jest.fn(async () => ({ baseUrl: 'https://api.cala.ai', apiKey: 'test-key' })),
82
+ helpers: { httpRequest },
83
+ };
84
+ const result = await node.execute.call(context);
85
+ expect(httpRequest).toHaveBeenCalledTimes(3);
86
+ expect(result[0]).toHaveLength(3);
87
+ expect(result[0][0].json).toEqual({ answer: 'Answer 1' });
88
+ expect(result[0][1].json).toEqual({ answer: 'Answer 2' });
89
+ expect(result[0][2].json).toEqual({ answer: 'Answer 3' });
90
+ });
91
+ it('should propagate HTTP errors', async () => {
92
+ const httpRequest = jest.fn().mockRejectedValue(new Error('API Error: 500 Internal Server Error'));
93
+ const context = {
94
+ getInputData: jest.fn(() => [{ json: {} }]),
95
+ getNodeParameter: jest.fn((name) => {
96
+ if (name === 'resource')
97
+ return 'knowledge';
98
+ if (name === 'operation')
99
+ return 'search';
100
+ if (name === 'query')
101
+ return 'test query';
102
+ throw new Error(`Unexpected parameter: ${name}`);
103
+ }),
104
+ getCredentials: jest.fn(async () => ({ baseUrl: 'https://api.cala.ai', apiKey: 'test-key' })),
105
+ helpers: { httpRequest },
106
+ };
107
+ await expect(node.execute.call(context)).rejects.toThrow('API Error: 500 Internal Server Error');
108
+ });
109
+ it('should return empty array for unsupported resource/operation', async () => {
110
+ const httpRequest = jest.fn();
111
+ const context = {
112
+ getInputData: jest.fn(() => [{ json: {} }]),
113
+ getNodeParameter: jest.fn((name) => {
114
+ if (name === 'resource')
115
+ return 'unsupported';
116
+ if (name === 'operation')
117
+ return 'unknown';
118
+ throw new Error(`Unexpected parameter: ${name}`);
119
+ }),
120
+ getCredentials: jest.fn(async () => ({ baseUrl: 'https://api.cala.ai', apiKey: 'test-key' })),
121
+ helpers: { httpRequest },
122
+ };
123
+ const result = await node.execute.call(context);
124
+ expect(httpRequest).not.toHaveBeenCalled();
125
+ expect(result).toEqual([[]]);
126
+ });
127
+ });
128
+ });
@@ -0,0 +1,8 @@
1
+ <?xml version="1.0" encoding="UTF-8" standalone="no"?>
2
+ <!DOCTYPE svg PUBLIC "-//W3C//DTD SVG 1.1//EN" "http://www.w3.org/Graphics/SVG/1.1/DTD/svg11.dtd">
3
+ <svg width="100%" height="100%" viewBox="0 0 440 158" version="1.1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" xml:space="preserve" xmlns:serif="http://www.serif.com/" style="fill-rule:evenodd;clip-rule:evenodd;stroke-linejoin:round;stroke-miterlimit:2;">
4
+ <use xlink:href="#_Image1" x="0" y="0.811" width="440px" height="157px"/>
5
+ <defs>
6
+ <image id="_Image1" width="440px" height="157px" xlink:href=""/>
7
+ </defs>
8
+ </svg>
package/package.json ADDED
@@ -0,0 +1,68 @@
1
+ {
2
+ "name": "n8n-nodes-cala",
3
+ "version": "0.2.0",
4
+ "description": "n8n nodes for Cala AI knowledge search",
5
+ "keywords": [
6
+ "n8n-community-node-package",
7
+ "n8n",
8
+ "cala",
9
+ "knowledge",
10
+ "search"
11
+ ],
12
+ "license": "MIT",
13
+ "author": {
14
+ "name": "Cala AI",
15
+ "email": "support@cala.ai",
16
+ "url": "https://cala.ai"
17
+ },
18
+ "repository": {
19
+ "type": "git",
20
+ "url": "https://gitlab.com/cala-ai/cala-n8n.git"
21
+ },
22
+ "homepage": "https://gitlab.com/cala-ai/cala-n8n",
23
+ "bugs": {
24
+ "url": "https://gitlab.com/cala-ai/cala-n8n/-/issues"
25
+ },
26
+ "files": [
27
+ "dist",
28
+ "LICENSE",
29
+ "README.md"
30
+ ],
31
+ "n8n": {
32
+ "n8nNodesApiVersion": 1,
33
+ "credentials": [
34
+ "dist/credentials/CalaApi.credentials.js"
35
+ ],
36
+ "nodes": [
37
+ "dist/nodes/Cala/Cala.node.js"
38
+ ]
39
+ },
40
+ "devDependencies": {
41
+ "@eslint/js": "^9.39.2",
42
+ "@types/jest": "^30.0.0",
43
+ "@types/node": "^20.10.0",
44
+ "eslint": "^9.39.2",
45
+ "gulp": "^5.0.0",
46
+ "jest": "^30.2.0",
47
+ "n8n-workflow": "^1.0.0",
48
+ "ts-jest": "^29.4.6",
49
+ "typescript": "^5.3.0",
50
+ "typescript-eslint": "^8.53.1"
51
+ },
52
+ "peerDependencies": {
53
+ "n8n-workflow": "*"
54
+ },
55
+ "engines": {
56
+ "node": ">=22",
57
+ "pnpm": ">=10"
58
+ },
59
+ "volta": {
60
+ "node": "22.13.1"
61
+ },
62
+ "scripts": {
63
+ "build": "tsc && gulp build:icons",
64
+ "dev": "tsc --watch",
65
+ "test": "jest",
66
+ "lint": "eslint nodes credentials"
67
+ }
68
+ }