Add native local device integrations

This commit is contained in:
2026-05-05 18:26:11 +00:00
parent accfa82f36
commit 282283d344
69 changed files with 9713 additions and 182 deletions
@@ -0,0 +1,85 @@
import { expect, tap } from '@git.zone/tstest/tapbundle';
import { AndroidIpWebcamMapper, type IAndroidIpWebcamSnapshot } from '../../ts/integrations/android_ip_webcam/index.js';
const snapshot: IAndroidIpWebcamSnapshot = {
deviceInfo: {
id: 'kitchen-phone',
name: 'Kitchen Phone',
manufacturer: 'Android IP Webcam',
host: '192.168.1.20',
port: 8080,
protocol: 'http',
online: true,
},
camera: {
id: 'camera',
name: 'Kitchen Phone Camera',
mjpegUrl: 'http://192.168.1.20:8080/video',
imageUrl: 'http://192.168.1.20:8080/shot.jpg',
rtspUrl: 'rtsp://192.168.1.20:8080/h264_aac.sdp',
supportedFeatures: ['stream'],
available: true,
},
sensors: [
{ key: 'audio_connections', name: 'Audio connections', value: 1, stateClass: 'total', entityCategory: 'diagnostic', available: true },
{ key: 'battery_level', name: 'Battery level', value: 87, unit: '%', deviceClass: 'battery', stateClass: 'measurement', entityCategory: 'diagnostic', available: true },
],
binarySensors: [{ key: 'motion_active', name: 'Motion active', isOn: true, deviceClass: 'motion', available: true }],
switches: [
{ key: 'torch', name: 'Torch', isOn: false, command: 'torch', entityCategory: 'config', available: true },
{ key: 'motion_detect', name: 'Motion detection', isOn: true, command: 'setting', entityCategory: 'config', available: true },
],
statusData: { audio_connections: 1, curvals: { torch: 'off', motion_detect: 'on' } },
sensorData: { battery_level: { unit: '%', data: [[ [87, 123] ]] }, motion_active: { data: [[ [1, 123] ]] } },
currentSettings: { torch: false, motion_detect: true },
enabledSensors: ['battery_level', 'motion_active'],
enabledSettings: ['torch', 'motion_detect'],
availableSettings: {},
connected: true,
updatedAt: '2026-01-01T00:00:00.000Z',
};
tap.test('maps Android IP Webcam camera, sensors, binary sensor, and switches', async () => {
const devices = AndroidIpWebcamMapper.toDevices(snapshot);
const entities = AndroidIpWebcamMapper.toEntities(snapshot);
expect(devices.length).toEqual(1);
expect(devices[0].features.some((featureArg) => featureArg.capability === 'camera')).toBeTrue();
expect(entities.find((entityArg) => entityArg.id === 'camera.kitchen_phone_camera')?.attributes?.mjpegUrl).toEqual('http://192.168.1.20:8080/video');
expect(entities.find((entityArg) => entityArg.id === 'sensor.battery_level')?.state).toEqual(87);
expect(entities.find((entityArg) => entityArg.id === 'binary_sensor.motion_active')?.state).toEqual('on');
expect(entities.find((entityArg) => entityArg.id === 'switch.torch')?.state).toEqual('off');
});
tap.test('models camera and setting services as explicit commands', async () => {
const streamCommand = AndroidIpWebcamMapper.commandForService(snapshot, {
domain: 'camera',
service: 'stream_source',
target: { entityId: 'camera.kitchen_phone_camera' },
});
const snapshotCommand = AndroidIpWebcamMapper.commandForService(snapshot, {
domain: 'camera',
service: 'snapshot',
target: { entityId: 'camera.kitchen_phone_camera' },
});
const torchCommand = AndroidIpWebcamMapper.commandForService(snapshot, {
domain: 'switch',
service: 'turn_on',
target: { entityId: 'switch.torch' },
});
const zoomCommand = AndroidIpWebcamMapper.commandForService(snapshot, {
domain: 'android_ip_webcam',
service: 'set_zoom',
target: {},
data: { zoom: '42' },
});
expect(streamCommand?.type).toEqual('stream_source');
expect(snapshotCommand?.type).toEqual('snapshot_image');
expect(torchCommand?.type).toEqual('torch');
expect(torchCommand?.activate).toBeTrue();
expect(zoomCommand?.type).toEqual('set_zoom');
expect(zoomCommand?.zoom).toEqual(42);
});
export default tap.start();