Add loadOptions for Home Assistant

This commit is contained in:
pemontto 2021-10-17 17:48:24 +01:00
parent 0243fc68d9
commit 095ce93cf7
No known key found for this signature in database
GPG key ID: EDCB93C3DA1B5DA9
5 changed files with 96 additions and 7 deletions

View file

@ -33,7 +33,10 @@ export const cameraProxyFields = [
{ {
displayName: 'Camera Entity ID', displayName: 'Camera Entity ID',
name: 'cameraEntityId', name: 'cameraEntityId',
type: 'string', type: 'options',
typeOptions: {
loadOptionsMethod: 'getCameraEntities',
},
default: '', default: '',
required: true, required: true,
displayOptions: { displayOptions: {

View file

@ -4,15 +4,17 @@ import {
import { import {
IExecuteFunctions, IExecuteFunctions,
ILoadOptionsFunctions,
} from 'n8n-core'; } from 'n8n-core';
import { import {
IDataObject, IDataObject,
INodePropertyOptions,
NodeApiError, NodeApiError,
NodeOperationError, NodeOperationError,
} from 'n8n-workflow'; } from 'n8n-workflow';
export async function homeAssistantApiRequest(this: IExecuteFunctions, method: string, resource: string, body: IDataObject = {}, qs: IDataObject = {}, uri?: string, option: IDataObject = {}) { export async function homeAssistantApiRequest(this: IExecuteFunctions | ILoadOptionsFunctions, method: string, resource: string, body: IDataObject = {}, qs: IDataObject = {}, uri?: string, option: IDataObject = {}) {
const credentials = await this.getCredentials('homeAssistantApi'); const credentials = await this.getCredentials('homeAssistantApi');
if (credentials === undefined) { if (credentials === undefined) {
@ -35,8 +37,51 @@ export async function homeAssistantApiRequest(this: IExecuteFunctions, method: s
delete options.body; delete options.body;
} }
try { try {
return await this.helpers.request(options); if (this.helpers.request) {
return await this.helpers.request(options);
}
} catch (error) { } catch (error) {
throw new NodeApiError(this.getNode(), error); throw new NodeApiError(this.getNode(), error);
} }
} }
export async function getHomeAssistantEntities(this: IExecuteFunctions | ILoadOptionsFunctions, domain = '') {
const returnData: INodePropertyOptions[] = [];
const entities = await homeAssistantApiRequest.call(this, 'GET', '/states');
for (const entity of entities) {
const entityId = entity.entity_id as string;
if (domain === '' || domain && entityId.startsWith(domain)) {
const entityName = entity.attributes.friendly_name as string || entityId;
returnData.push({
name: entityName,
value: entityId,
});
}
}
return returnData;
}
export async function getHomeAssistantServices(this: IExecuteFunctions | ILoadOptionsFunctions, domain = '') {
const returnData: INodePropertyOptions[] = [];
const services = await homeAssistantApiRequest.call(this, 'GET', '/services');
if (domain === '') {
// If no domain specified return domains
const domains = services.map(({ domain }: IDataObject) => domain as string).sort();
returnData.push(...domains.map((service: string) => ({ name: service, value: service })));
return returnData;
} else {
// If we have a domain, return all relevant services
const domainServices = services.filter((service: IDataObject) => service.domain === domain);
for (const domainService of domainServices) {
for (const [serviceID, value] of Object.entries(domainService.services)) {
const serviceProperties = value as IDataObject;
const serviceName = serviceProperties.description || serviceID;
returnData.push({
name: serviceName as string,
value: serviceID,
});
}
}
}
return returnData;
}

View file

@ -4,7 +4,9 @@ import {
import { import {
IDataObject, IDataObject,
ILoadOptionsFunctions,
INodeExecutionData, INodeExecutionData,
INodePropertyOptions,
INodeType, INodeType,
INodeTypeDescription, INodeTypeDescription,
} from 'n8n-workflow'; } from 'n8n-workflow';
@ -49,6 +51,8 @@ import {
} from './CameraProxyDescription'; } from './CameraProxyDescription';
import { import {
getHomeAssistantEntities,
getHomeAssistantServices,
homeAssistantApiRequest, homeAssistantApiRequest,
} from './GenericFunctions'; } from './GenericFunctions';
@ -133,6 +137,28 @@ export class HomeAssistant implements INodeType {
], ],
}; };
methods = {
loadOptions: {
async getAllEntities(this: ILoadOptionsFunctions): Promise<INodePropertyOptions[]> {
return await getHomeAssistantEntities.call(this);
},
async getCameraEntities(this: ILoadOptionsFunctions): Promise<INodePropertyOptions[]> {
return await getHomeAssistantEntities.call(this, 'camera');
},
async getDomains(this: ILoadOptionsFunctions): Promise<INodePropertyOptions[]> {
return await getHomeAssistantServices.call(this);
},
async getDomainServices(this: ILoadOptionsFunctions): Promise<INodePropertyOptions[]> {
const currentDomain = this.getCurrentNodeParameter('domain') as string;
if (currentDomain) {
return await getHomeAssistantServices.call(this, currentDomain);
} else {
return [];
}
},
},
};
async execute(this: IExecuteFunctions): Promise<INodeExecutionData[][]> { async execute(this: IExecuteFunctions): Promise<INodeExecutionData[][]> {
const items = this.getInputData(); const items = this.getInputData();
const returnData: IDataObject[] = []; const returnData: IDataObject[] = [];

View file

@ -83,7 +83,10 @@ export const serviceFields = [
{ {
displayName: 'Domain', displayName: 'Domain',
name: 'domain', name: 'domain',
type: 'string', type: 'options',
typeOptions: {
loadOptionsMethod: 'getDomains',
},
default: '', default: '',
required: true, required: true,
displayOptions: { displayOptions: {
@ -100,7 +103,13 @@ export const serviceFields = [
{ {
displayName: 'Service', displayName: 'Service',
name: 'service', name: 'service',
type: 'string', type: 'options',
typeOptions: {
loadOptionsDependsOn: [
'domain',
],
loadOptionsMethod: 'getDomainServices',
},
default: '', default: '',
required: true, required: true,
displayOptions: { displayOptions: {

View file

@ -43,7 +43,10 @@ export const stateFields = [
{ {
displayName: 'Entity ID', displayName: 'Entity ID',
name: 'entityId', name: 'entityId',
type: 'string', type: 'options',
typeOptions: {
loadOptionsMethod: 'getAllEntities',
},
displayOptions: { displayOptions: {
show: { show: {
operation: [ operation: [
@ -110,7 +113,10 @@ export const stateFields = [
{ {
displayName: 'Entity ID', displayName: 'Entity ID',
name: 'entityId', name: 'entityId',
type: 'string', type: 'options',
typeOptions: {
loadOptionsMethod: 'getAllEntities',
},
displayOptions: { displayOptions: {
show: { show: {
operation: [ operation: [