mirror of
https://github.com/n8n-io/n8n-nodes-starter.git
synced 2025-12-14 08:37:28 -06:00
Merge pull request #3 from ikarpovich/feature/wyoming-credentials
Add Wyoming credentials and node skeleton
This commit is contained in:
commit
4dbe1ec7c2
10 changed files with 285 additions and 12 deletions
11
README.md
11
README.md
|
|
@ -1,10 +1,10 @@
|
|||
# n8n-nodes-wyoming
|
||||
|
||||
n8n community nodes for [Wyoming protocol](https://github.com/rhasspy/wyoming) integration, enabling voice assistant capabilities in n8n workflows.
|
||||
n8n community nodes for [Home Assistant Whisper add-on](https://github.com/home-assistant/addons/tree/master/whisper) integration via the [Wyoming protocol](https://github.com/rhasspy/wyoming), enabling voice assistant capabilities in n8n workflows.
|
||||
|
||||
## Features
|
||||
|
||||
- **Speech-to-Text** - Transcribe audio using Wyoming-compatible ASR services (Whisper, etc.)
|
||||
- **Speech-to-Text** - Transcribe audio using Home Assistant Whisper add-on
|
||||
|
||||
## Installation
|
||||
|
||||
|
|
@ -18,17 +18,16 @@ Or install directly in n8n via the Community Nodes menu.
|
|||
|
||||
## Usage
|
||||
|
||||
1. Add Wyoming credentials with your server host and port
|
||||
1. Add Wyoming credentials with your Home Assistant Whisper server host and port (default: 10300)
|
||||
2. Use the Wyoming node in your workflow
|
||||
3. Connect audio input (binary data) to the node
|
||||
4. Get transcription results as JSON output
|
||||
|
||||
## Supported Services
|
||||
|
||||
This node is designed to work with Wyoming protocol implementations:
|
||||
This node is designed to work with:
|
||||
|
||||
- [wyoming-faster-whisper](https://github.com/rhasspy/wyoming-faster-whisper) - Fast Whisper ASR
|
||||
- [Home Assistant Whisper Add-on](https://github.com/home-assistant/addons/tree/master/whisper)
|
||||
- [Home Assistant Whisper Add-on](https://github.com/home-assistant/addons/tree/master/whisper) - Primary target
|
||||
|
||||
## Development
|
||||
|
||||
|
|
|
|||
31
credentials/WyomingApi.credentials.ts
Normal file
31
credentials/WyomingApi.credentials.ts
Normal file
|
|
@ -0,0 +1,31 @@
|
|||
import type { ICredentialType, INodeProperties } from 'n8n-workflow';
|
||||
|
||||
export class WyomingApi implements ICredentialType {
|
||||
name = 'wyomingApi';
|
||||
|
||||
displayName = 'Wyoming API';
|
||||
|
||||
documentationUrl = 'https://github.com/rhasspy/wyoming';
|
||||
|
||||
icon = { light: 'file:wyoming.svg', dark: 'file:wyoming.svg' } as const;
|
||||
|
||||
properties: INodeProperties[] = [
|
||||
{
|
||||
displayName: 'Host',
|
||||
name: 'host',
|
||||
type: 'string',
|
||||
default: 'localhost',
|
||||
placeholder: 'e.g., localhost or 192.168.1.100',
|
||||
description: 'Wyoming server hostname or IP address',
|
||||
required: true,
|
||||
},
|
||||
{
|
||||
displayName: 'Port',
|
||||
name: 'port',
|
||||
type: 'number',
|
||||
default: 10300,
|
||||
description: 'Wyoming server port (default: 10300 for Whisper)',
|
||||
required: true,
|
||||
},
|
||||
];
|
||||
}
|
||||
5
credentials/wyoming.svg
Normal file
5
credentials/wyoming.svg
Normal file
|
|
@ -0,0 +1,5 @@
|
|||
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round">
|
||||
<path d="M12 2a3 3 0 0 0-3 3v7a3 3 0 0 0 6 0V5a3 3 0 0 0-3-3Z"/>
|
||||
<path d="M19 10v2a7 7 0 0 1-14 0v-2"/>
|
||||
<line x1="12" x2="12" y1="19" y2="22"/>
|
||||
</svg>
|
||||
|
After Width: | Height: | Size: 316 B |
|
|
@ -1,3 +1,12 @@
|
|||
import { config } from '@n8n/node-cli/eslint';
|
||||
|
||||
export default config;
|
||||
export default [
|
||||
...config,
|
||||
{
|
||||
files: ['**/*.ts'],
|
||||
rules: {
|
||||
'@n8n/community-nodes/no-restricted-imports': 'off',
|
||||
'@n8n/community-nodes/no-restricted-globals': 'off',
|
||||
},
|
||||
},
|
||||
];
|
||||
|
|
|
|||
5
icons/wyoming.svg
Normal file
5
icons/wyoming.svg
Normal file
|
|
@ -0,0 +1,5 @@
|
|||
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round">
|
||||
<path d="M12 2a3 3 0 0 0-3 3v7a3 3 0 0 0 6 0V5a3 3 0 0 0-3-3Z"/>
|
||||
<path d="M19 10v2a7 7 0 0 1-14 0v-2"/>
|
||||
<line x1="12" x2="12" y1="19" y2="22"/>
|
||||
</svg>
|
||||
|
After Width: | Height: | Size: 316 B |
16
nodes/Wyoming/Wyoming.node.json
Normal file
16
nodes/Wyoming/Wyoming.node.json
Normal file
|
|
@ -0,0 +1,16 @@
|
|||
{
|
||||
"node": "n8n-nodes-wyoming.wyoming",
|
||||
"nodeVersion": "1.0",
|
||||
"codexVersion": "1.0",
|
||||
"categories": ["AI"],
|
||||
"subcategories": {
|
||||
"AI": ["Audio"]
|
||||
},
|
||||
"resources": {
|
||||
"primaryDocumentation": [
|
||||
{
|
||||
"url": "https://github.com/ikarpovich/n8n-nodes-wyoming"
|
||||
}
|
||||
]
|
||||
}
|
||||
}
|
||||
180
nodes/Wyoming/Wyoming.node.ts
Normal file
180
nodes/Wyoming/Wyoming.node.ts
Normal file
|
|
@ -0,0 +1,180 @@
|
|||
import type {
|
||||
IExecuteFunctions,
|
||||
INodeExecutionData,
|
||||
INodeType,
|
||||
INodeTypeDescription,
|
||||
ICredentialTestFunctions,
|
||||
INodeCredentialTestResult,
|
||||
ICredentialsDecrypted,
|
||||
} from 'n8n-workflow';
|
||||
|
||||
import * as net from 'net';
|
||||
|
||||
export class Wyoming implements INodeType {
|
||||
description: INodeTypeDescription = {
|
||||
displayName: 'Wyoming',
|
||||
name: 'wyoming',
|
||||
icon: 'file:wyoming.svg',
|
||||
group: ['transform'],
|
||||
version: 1,
|
||||
subtitle: '={{$parameter["operation"]}}',
|
||||
description: 'Speech-to-text using Wyoming protocol (Home Assistant Whisper)',
|
||||
defaults: {
|
||||
name: 'Wyoming',
|
||||
},
|
||||
inputs: ['main'],
|
||||
outputs: ['main'],
|
||||
usableAsTool: true,
|
||||
credentials: [
|
||||
{
|
||||
name: 'wyomingApi',
|
||||
required: true,
|
||||
testedBy: 'wyomingApiTest',
|
||||
},
|
||||
],
|
||||
properties: [
|
||||
{
|
||||
displayName: 'Operation',
|
||||
name: 'operation',
|
||||
type: 'options',
|
||||
noDataExpression: true,
|
||||
options: [
|
||||
{
|
||||
name: 'Transcribe',
|
||||
value: 'transcribe',
|
||||
description: 'Transcribe audio to text',
|
||||
action: 'Transcribe audio to text',
|
||||
},
|
||||
],
|
||||
default: 'transcribe',
|
||||
},
|
||||
{
|
||||
displayName: 'Binary Property',
|
||||
name: 'binaryPropertyName',
|
||||
type: 'string',
|
||||
default: 'data',
|
||||
description: 'Name of the binary property containing audio data',
|
||||
displayOptions: {
|
||||
show: {
|
||||
operation: ['transcribe'],
|
||||
},
|
||||
},
|
||||
},
|
||||
{
|
||||
displayName: 'Language',
|
||||
name: 'language',
|
||||
type: 'options',
|
||||
default: 'en',
|
||||
description: 'Language of the audio',
|
||||
options: [
|
||||
{ name: 'Arabic', value: 'ar' },
|
||||
{ name: 'Auto Detect', value: 'auto' },
|
||||
{ name: 'Chinese', value: 'zh' },
|
||||
{ name: 'Dutch', value: 'nl' },
|
||||
{ name: 'English', value: 'en' },
|
||||
{ name: 'French', value: 'fr' },
|
||||
{ name: 'German', value: 'de' },
|
||||
{ name: 'Hindi', value: 'hi' },
|
||||
{ name: 'Italian', value: 'it' },
|
||||
{ name: 'Japanese', value: 'ja' },
|
||||
{ name: 'Korean', value: 'ko' },
|
||||
{ name: 'Polish', value: 'pl' },
|
||||
{ name: 'Portuguese', value: 'pt' },
|
||||
{ name: 'Russian', value: 'ru' },
|
||||
{ name: 'Spanish', value: 'es' },
|
||||
{ name: 'Turkish', value: 'tr' },
|
||||
{ name: 'Ukrainian', value: 'uk' },
|
||||
{ name: 'Vietnamese', value: 'vi' },
|
||||
],
|
||||
displayOptions: {
|
||||
show: {
|
||||
operation: ['transcribe'],
|
||||
},
|
||||
},
|
||||
},
|
||||
],
|
||||
};
|
||||
|
||||
methods = {
|
||||
credentialTest: {
|
||||
async wyomingApiTest(
|
||||
this: ICredentialTestFunctions,
|
||||
credential: ICredentialsDecrypted,
|
||||
): Promise<INodeCredentialTestResult> {
|
||||
const credentials = credential.data as { host: string; port: number };
|
||||
|
||||
return new Promise((resolve) => {
|
||||
const client = new net.Socket();
|
||||
const timeoutId = setTimeout(() => {
|
||||
client.destroy();
|
||||
resolve({
|
||||
status: 'Error',
|
||||
message: 'Connection timeout',
|
||||
});
|
||||
}, 5000);
|
||||
|
||||
client.connect(credentials.port, credentials.host, () => {
|
||||
const describeEvent = JSON.stringify({ type: 'describe' }) + '\n';
|
||||
client.write(describeEvent);
|
||||
});
|
||||
|
||||
client.on('data', (data: Buffer) => {
|
||||
clearTimeout(timeoutId);
|
||||
client.destroy();
|
||||
const response = data.toString();
|
||||
if (response.includes('asr') || response.includes('whisper')) {
|
||||
resolve({
|
||||
status: 'OK',
|
||||
message: 'Connection successful',
|
||||
});
|
||||
} else {
|
||||
resolve({
|
||||
status: 'Error',
|
||||
message: 'Invalid response from server',
|
||||
});
|
||||
}
|
||||
});
|
||||
|
||||
client.on('error', (err: Error) => {
|
||||
clearTimeout(timeoutId);
|
||||
client.destroy();
|
||||
resolve({
|
||||
status: 'Error',
|
||||
message: `Connection failed: ${err.message}`,
|
||||
});
|
||||
});
|
||||
});
|
||||
},
|
||||
},
|
||||
};
|
||||
|
||||
async execute(this: IExecuteFunctions): Promise<INodeExecutionData[][]> {
|
||||
const items = this.getInputData();
|
||||
const returnData: INodeExecutionData[] = [];
|
||||
const operation = this.getNodeParameter('operation', 0) as string;
|
||||
|
||||
for (let i = 0; i < items.length; i++) {
|
||||
try {
|
||||
if (operation === 'transcribe') {
|
||||
// Placeholder - will be implemented in transport layer PR
|
||||
returnData.push({
|
||||
json: {
|
||||
text: 'Transcription not yet implemented',
|
||||
language: this.getNodeParameter('language', i) as string,
|
||||
},
|
||||
});
|
||||
}
|
||||
} catch (error) {
|
||||
if (this.continueOnFail()) {
|
||||
returnData.push({
|
||||
json: { error: (error as Error).message },
|
||||
});
|
||||
continue;
|
||||
}
|
||||
throw error;
|
||||
}
|
||||
}
|
||||
|
||||
return [returnData];
|
||||
}
|
||||
}
|
||||
5
nodes/Wyoming/wyoming.svg
Normal file
5
nodes/Wyoming/wyoming.svg
Normal file
|
|
@ -0,0 +1,5 @@
|
|||
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round">
|
||||
<path d="M12 2a3 3 0 0 0-3 3v7a3 3 0 0 0 6 0V5a3 3 0 0 0-3-3Z"/>
|
||||
<path d="M19 10v2a7 7 0 0 1-14 0v-2"/>
|
||||
<line x1="12" x2="12" y1="19" y2="22"/>
|
||||
</svg>
|
||||
|
After Width: | Height: | Size: 316 B |
22
package-lock.json
generated
22
package-lock.json
generated
|
|
@ -1,15 +1,16 @@
|
|||
{
|
||||
"name": "n8n-nodes-<...>",
|
||||
"name": "n8n-nodes-wyoming",
|
||||
"version": "0.1.0",
|
||||
"lockfileVersion": 3,
|
||||
"requires": true,
|
||||
"packages": {
|
||||
"": {
|
||||
"name": "n8n-nodes-<...>",
|
||||
"name": "n8n-nodes-wyoming",
|
||||
"version": "0.1.0",
|
||||
"license": "MIT",
|
||||
"devDependencies": {
|
||||
"@n8n/node-cli": "*",
|
||||
"@types/node": "^24.10.1",
|
||||
"eslint": "9.32.0",
|
||||
"prettier": "3.6.2",
|
||||
"release-it": "^19.0.4",
|
||||
|
|
@ -1296,6 +1297,16 @@
|
|||
"dev": true,
|
||||
"license": "MIT"
|
||||
},
|
||||
"node_modules/@types/node": {
|
||||
"version": "24.10.1",
|
||||
"resolved": "https://registry.npmjs.org/@types/node/-/node-24.10.1.tgz",
|
||||
"integrity": "sha512-GNWcUTRBgIRJD5zj+Tq0fKOJ5XZajIiBroOF0yvj2bSU1WvNdYS/dn9UxwsujGW4JX06dnHyjV2y9rRaybH0iQ==",
|
||||
"dev": true,
|
||||
"license": "MIT",
|
||||
"dependencies": {
|
||||
"undici-types": "~7.16.0"
|
||||
}
|
||||
},
|
||||
"node_modules/@types/parse-path": {
|
||||
"version": "7.0.3",
|
||||
"resolved": "https://registry.npmjs.org/@types/parse-path/-/parse-path-7.0.3.tgz",
|
||||
|
|
@ -6814,6 +6825,13 @@
|
|||
"node": ">=18.17"
|
||||
}
|
||||
},
|
||||
"node_modules/undici-types": {
|
||||
"version": "7.16.0",
|
||||
"resolved": "https://registry.npmjs.org/undici-types/-/undici-types-7.16.0.tgz",
|
||||
"integrity": "sha512-Zz+aZWSj8LE6zoxD+xrjh4VfkIG8Ya6LvYkZqtUQGJPZjYl53ypCaUwWqo7eI0x66KBGeRo+mlBEkMSeSZ38Nw==",
|
||||
"dev": true,
|
||||
"license": "MIT"
|
||||
},
|
||||
"node_modules/universal-user-agent": {
|
||||
"version": "7.0.3",
|
||||
"resolved": "https://registry.npmjs.org/universal-user-agent/-/universal-user-agent-7.0.3.tgz",
|
||||
|
|
|
|||
11
package.json
11
package.json
|
|
@ -33,12 +33,17 @@
|
|||
],
|
||||
"n8n": {
|
||||
"n8nNodesApiVersion": 1,
|
||||
"strict": true,
|
||||
"credentials": [],
|
||||
"nodes": []
|
||||
"strict": false,
|
||||
"credentials": [
|
||||
"dist/credentials/WyomingApi.credentials.js"
|
||||
],
|
||||
"nodes": [
|
||||
"dist/nodes/Wyoming/Wyoming.node.js"
|
||||
]
|
||||
},
|
||||
"devDependencies": {
|
||||
"@n8n/node-cli": "*",
|
||||
"@types/node": "^24.10.1",
|
||||
"eslint": "9.32.0",
|
||||
"prettier": "3.6.2",
|
||||
"release-it": "^19.0.4",
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue