Successfully split up data provider
All checks were successful
continuous-integration/drone/push Build is passing

This commit is contained in:
Brandon Watson 2021-12-28 18:38:08 -05:00
parent 506c170746
commit c22a8a0325
9 changed files with 38 additions and 468 deletions

View File

@ -1,5 +1,4 @@
import { PlatformAccessory, RemoteController, Service } from "homebridge"; import { PlatformAccessory, RemoteController, Service } from "homebridge";
import HarmonyDataProvider from "../dataProviders/harmonyDataProvider";
import { IActivity } from "../models/config"; import { IActivity } from "../models/config";
import { Platform } from "../platform"; import { Platform } from "../platform";
import { ActivityService } from "../services/activityService"; import { ActivityService } from "../services/activityService";
@ -151,7 +150,7 @@ export class ControlUnit {
*/ */
private onGetAccessoryActive = async () => { private onGetAccessoryActive = async () => {
//@ts-ignore //@ts-ignore
return this._dataProvider.getIsActive(this._accessory.displayName) return this._activityService.getIsActive(this._accessory.displayName)
? this._platform.Characteristic.Active.ACTIVE ? this._platform.Characteristic.Active.ACTIVE
: this._platform.Characteristic.Active.INACTIVE; : this._platform.Characteristic.Active.INACTIVE;
}; };

View File

@ -1,5 +1,5 @@
import { PlatformAccessory, Service } from "homebridge"; import { PlatformAccessory, Service } from "homebridge";
import HarmonyDataProvider from "../dataProviders/harmonyDataProvider"; import { HarmonyDataProvider } from "../dataProviders/harmonyDataProvider";
import { IDeviceButton } from "../models/config"; import { IDeviceButton } from "../models/config";
import { HarmonyDevice } from "../models/harmonyDevice"; import { HarmonyDevice } from "../models/harmonyDevice";
import { Platform } from "../platform"; import { Platform } from "../platform";

View File

@ -5,7 +5,7 @@ import {
PlatformAccessory, PlatformAccessory,
Service, Service,
} from "homebridge"; } from "homebridge";
import HarmonyDataProvider from "../dataProviders/harmonyDataProvider"; import { HarmonyDataProvider } from "../dataProviders/harmonyDataProvider";
import { ISequence } from "../models/config/sequence"; import { ISequence } from "../models/config/sequence";
import { HarmonyDevice } from "../models/harmonyDevice"; import { HarmonyDevice } from "../models/harmonyDevice";
import { Platform } from "../platform"; import { Platform } from "../platform";

View File

@ -1,46 +1,20 @@
import { IActivity } from "../models/config/activity"; import { Logging } from "homebridge";
import { IDeviceSetupItem } from "../models/config/deviceSetupItem"; import { inject, injectable } from "tsyringe";
import { IInput, IMatrix, IOutput } from "../models/config/matrix"; import { IConfig } from "../models/config";
import { RemoteKey } from "../accessories/controlUnit";
import { EventEmitter } from "events";
import { IHub } from "../models/config/hub";
import { IDeviceConfig } from "../models/config/deviceConfig"; import { IDeviceConfig } from "../models/config/deviceConfig";
import { IHub } from "../models/config/hub";
import { HarmonyDevice } from "../models/harmonyDevice"; import { HarmonyDevice } from "../models/harmonyDevice";
import { HarmonyHub } from "../models/harmonyHub"; import { HarmonyHub } from "../models/harmonyHub";
import { IConfig } from "../models/config";
import { inject, injectable } from "tsyringe";
import { Logger, Logging } from "homebridge";
const Harmony = require("harmony-websocket");
interface IActivityState {
currentActivity: IActivity;
}
interface IHarmonyDataProviderProps {
hubs: Array<IHub>;
deviceConfigs: Array<IDeviceConfig>;
log: any;
matrix: IMatrix;
}
@injectable() @injectable()
class HarmonyDataProvider extends EventEmitter { export class HarmonyDataProvider {
private _hubsByDevice: { [deviceName: string]: string } = {};
private _hubs: { [hubName: string]: HarmonyHub } = {}; private _hubs: { [hubName: string]: HarmonyHub } = {};
// private _devicesByHub: { [hubName: string]: { [deviceName: string]: HarmonyDevice } } = {}; private _hubsByDevice: { [deviceName: string]: string } = {};
private _states: {
[controlUnitName: string]: IActivityState | undefined;
} = {};
private _matrix: IMatrix;
constructor( constructor(
@inject("IConfig") private _config: IConfig, @inject("IConfig") private _config: IConfig,
@inject("log") private _log: Logging @inject("log") private _log: Logging
) { ) {
super();
this._matrix = _config.Matrix;
_config.Devices.forEach((deviceConfig: IDeviceConfig) => { _config.Devices.forEach((deviceConfig: IDeviceConfig) => {
this._hubsByDevice[deviceConfig.Name] = deviceConfig.Hub; this._hubsByDevice[deviceConfig.Name] = deviceConfig.Hub;
}); });
@ -50,87 +24,9 @@ class HarmonyDataProvider extends EventEmitter {
this.emitInfo(); this.emitInfo();
} }
// public get devicesByHub(): { [hubName: string]: { [deviceName: string]: HarmonyDevice } } { public async turnOnDevices(devices: Array<HarmonyDevice>): Promise<void> {
// return this._devicesByHub;
// }
public get hubs(): { [hubName: string]: HarmonyHub } {
return this._hubs;
}
/**
* Power on all devices in an activity.
*/
public powerOn = async (controlUnitName: string, activity: IActivity) => {
//Only power on if not alread on
let currentActivity = this._states[controlUnitName]
? this._states[controlUnitName]!.currentActivity
: undefined;
if (!currentActivity) {
await this.startActivity(controlUnitName, activity);
}
};
/**
* Power off all devices in an activity that aren't being used.
*/
public powerOff = async (controlUnitName: string) => {
if (!this._states[controlUnitName]) {
return;
}
//Build potential list of devices to turn off
let devicesToTurnOff: Array<HarmonyDevice> = this._states[
controlUnitName
]!.currentActivity.DeviceSetupList.map(
(value: IDeviceSetupItem): HarmonyDevice => {
return this.getDeviceFromName(value.DeviceName);
}
);
//Resolve device conflicts with other controlUnits
devicesToTurnOff = this.sanitizeDeviceList(
devicesToTurnOff,
controlUnitName
);
//Turn off devices
devicesToTurnOff.forEach(async (device: HarmonyDevice) => {
if (device) {
await device.powerOff();
}
});
this._states[controlUnitName] = undefined;
};
/**
* Start an activity
*/
public startActivity = async (
controlUnitName: string,
activity: IActivity
) => {
this._log.info(
`Starting activity ${activity.DisplayName} for controlUnit: ${controlUnitName}`
);
let lastActivity: IActivity | undefined = undefined;
if (this._states[controlUnitName]) {
lastActivity = this._states[controlUnitName]!.currentActivity;
}
//Build potential list of devices to to turn on
let devicesToTurnOn: Array<HarmonyDevice> = activity.DeviceSetupList.map(
(value: IDeviceSetupItem): HarmonyDevice => {
return this.getDeviceFromName(value.DeviceName);
}
);
//Resolve device conflicts with other controlUnits
devicesToTurnOn = this.sanitizeDeviceList(devicesToTurnOn, controlUnitName);
//Turn on devices
await Promise.all( await Promise.all(
devicesToTurnOn.map(async (device: HarmonyDevice) => { devices.map(async (device: HarmonyDevice) => {
if (device && device.name) { if (device && device.name) {
if (!device.on) { if (!device.on) {
this._log.info(`Turning on device ${device.name}`); this._log.info(`Turning on device ${device.name}`);
@ -139,84 +35,12 @@ class HarmonyDataProvider extends EventEmitter {
} }
}) })
); );
//Assign correct input
await Promise.all(
activity.DeviceSetupList.map(async (value: IDeviceSetupItem) => {
let device: HarmonyDevice = this.getDeviceFromName(value.DeviceName);
if (device && device.supportsCommand(`Input${value.Input}`)) {
await device.sendCommand(`Input${value.Input}`);
}
})
);
if (activity.UseMatrix) {
//get input and output
let input: IInput = this._matrix.Inputs.filter(
(e) => e.InputDevice === activity.ControlDevice
)[0];
let output: IOutput = this._matrix.Outputs.filter(
(e) => e.OutputDevice === activity.OutputDevice
)[0];
let inputCommandName: string = `In ${input.InputNumber}`;
let outputCommandName: string = `Out ${output.OutputLetter}`;
let matrixDevice: HarmonyDevice = this.getDeviceFromName(
this._matrix.DeviceName
);
//Route hdmi
if (
matrixDevice.supportsCommand(inputCommandName) &&
matrixDevice.supportsCommand(outputCommandName)
) {
await matrixDevice.sendCommand(outputCommandName);
await matrixDevice.sendCommand(inputCommandName);
await matrixDevice.sendCommand(outputCommandName);
await matrixDevice.sendCommand(inputCommandName);
}
} }
//Build potential list of devices to turn off public async turnOffDevices(devices: Array<HarmonyDevice>) {
if (lastActivity) {
let devicesToTurnOff: Array<HarmonyDevice> =
lastActivity.DeviceSetupList.map(
(value: IDeviceSetupItem): HarmonyDevice => {
return this.getDeviceFromName(value.DeviceName);
}
);
//remove devices that will be used for next activity from list
//delete array[index] is stupid because it just nulls out the index. But now i have to deal with nulls
devicesToTurnOff.forEach((device: HarmonyDevice, index: number) => {
if (
device &&
device.name &&
activity.DeviceSetupList.some((e) => {
return e && e.DeviceName === device.name;
})
) {
delete devicesToTurnOff[index];
}
});
//Resolve device conflicts with other controlUnits
devicesToTurnOff = this.sanitizeDeviceList(
devicesToTurnOff,
controlUnitName
);
this._log.info(
`Sanatized devices to turn off: ${JSON.stringify(
devicesToTurnOff.map((e) => (e ? e.name : ""))
)}`
);
await Promise.all( await Promise.all(
//Turn off devices //Turn off devices
devicesToTurnOff.map(async (device: HarmonyDevice) => { devices.map(async (device: HarmonyDevice) => {
if (device) { if (device) {
if (device.on) { if (device.on) {
this._log.info(`Turning off device ${device.name}`); this._log.info(`Turning off device ${device.name}`);
@ -227,102 +51,6 @@ class HarmonyDataProvider extends EventEmitter {
); );
} }
//Assign current activity
this._states[controlUnitName] = { currentActivity: activity };
};
/**
* Turn the volume up for the current running activity.
*/
public volumeUp = async (controlUnitName: string) => {
let volumeUpCommand: string = "Volume Up";
if (this._states[controlUnitName]) {
let volumeDevice: HarmonyDevice = this.getDeviceFromName(
this._states[controlUnitName]!.currentActivity.VolumeDevice
);
await volumeDevice.sendCommand(volumeUpCommand);
}
};
/**
* Volume down for current running activity.
*/
public volumeDown = async (controlUnitName: string) => {
let volumeDownCommand: string = "Volume Down";
if (this._states[controlUnitName]) {
let volumeDevice: HarmonyDevice = this.getDeviceFromName(
this._states[controlUnitName]!.currentActivity.VolumeDevice
);
await volumeDevice.sendCommand(volumeDownCommand);
}
};
/**
* Send key press for current activity.
*
* @param controlUnitName The name of the control unit to act on.
* @param key The key to send.
*/
public sendKeyPress = async (controlUnitName: string, key: any) => {
if (this._states[controlUnitName]) {
let commandName: string = "";
let device: HarmonyDevice = this.getDeviceFromName(
this._states[controlUnitName]!.currentActivity.ControlDevice
);
switch (key) {
case RemoteKey.ARROW_UP: {
commandName = "Direction Up";
break;
}
case RemoteKey.ARROW_DOWN: {
commandName = "Direction Down";
break;
}
case RemoteKey.ARROW_LEFT: {
commandName = "Direction Left";
break;
}
case RemoteKey.ARROW_RIGHT: {
commandName = "Direction Right";
break;
}
case RemoteKey.SELECT: {
commandName = "Select";
break;
}
case RemoteKey.PLAY_PAUSE: {
commandName = "Pause";
break;
}
case RemoteKey.INFORMATION: {
commandName = "Menu";
break;
}
case RemoteKey.BACK: {
commandName = "Back";
break;
}
case RemoteKey.EXIT: {
commandName = "Back";
break;
}
}
await device.sendCommand(commandName);
}
};
/**
* Return if a control unit is active
* @param controlUnitName
*/
public getIsActive(controlUnitName: string): IActivity | undefined {
return this._states[controlUnitName]
? this._states[controlUnitName]!.currentActivity
: undefined;
}
/** /**
* Get the IDevice by name. * Get the IDevice by name.
* @param deviceName The device to retrieve. * @param deviceName The device to retrieve.
@ -339,20 +67,6 @@ class HarmonyDataProvider extends EventEmitter {
return device!; return device!;
} }
// /**
// * Gets device button commands
// * @param deviceCommandName The device command name
// * @param deviceName The device name
// */
// public getCommand(deviceCommandName: string, deviceName: string): ICommand | undefined {
// const device: HarmonyDevice = this.getDeviceFromName(deviceName);
// if (device && device.supportsCommand(deviceCommandName)) {
// return device.getCommand(deviceCommandName);
// } else {
// return undefined;
// }
// }
private connect = async (hubs: Array<IHub>) => { private connect = async (hubs: Array<IHub>) => {
let readyCount = 0; let readyCount = 0;
await Promise.all( await Promise.all(
@ -362,7 +76,7 @@ class HarmonyDataProvider extends EventEmitter {
newHarmonyHub.on("Ready", () => { newHarmonyHub.on("Ready", () => {
readyCount++; readyCount++;
if (readyCount === Object.keys(this._hubs).length) { if (readyCount === Object.keys(this._hubs).length) {
this.emit("Ready"); // this.emit("Ready");
} }
}); });
await newHarmonyHub.initialize(); await newHarmonyHub.initialize();
@ -370,45 +84,11 @@ class HarmonyDataProvider extends EventEmitter {
); );
}; };
/**
* Helper function to make sure no control unit depends on device list.
* @param devicesToTurnOn The list of devices to modify.
* @param controlUnitName The name of the control unit in question.
*/
private sanitizeDeviceList(
devicesToTurnOn: Array<HarmonyDevice>,
controlUnitName: string
): Array<HarmonyDevice> {
for (let controlUnitKey in this._states) {
//Skip self
if (controlUnitKey === controlUnitName) {
continue;
}
let currentOtherState: IActivityState = this._states[controlUnitKey]!;
if (currentOtherState) {
currentOtherState.currentActivity.DeviceSetupList.forEach(
(value: IDeviceSetupItem) => {
//there are devices to remove
if (devicesToTurnOn.some((e) => e && e.name === value.DeviceName)) {
let deviceToRemove: HarmonyDevice = devicesToTurnOn.filter(
(i) => i.name === value.DeviceName
)[0];
delete devicesToTurnOn[devicesToTurnOn.indexOf(deviceToRemove)];
}
}
);
}
}
return devicesToTurnOn;
}
private emitInfo(): void { private emitInfo(): void {
//Emit devices if requested //Emit devices if requested
this._log.info("All hubs connected"); this._log.info("All hubs connected");
if (this._config.EmitDevicesOnStartup) { if (this._config.EmitDevicesOnStartup) {
const hubs = this.hubs; const hubs = this._hubs;
Object.values(hubs).forEach((hub: HarmonyHub) => { Object.values(hubs).forEach((hub: HarmonyHub) => {
const deviceDictionary = hub.devices; const deviceDictionary = hub.devices;
this._log.info(`${hub.hubName}`); this._log.info(`${hub.hubName}`);
@ -422,5 +102,3 @@ class HarmonyDataProvider extends EventEmitter {
} }
} }
} }
export default HarmonyDataProvider;

View File

@ -1,105 +0,0 @@
import { Logging } from "homebridge";
import { inject } from "tsyringe";
import { IConfig, IMatrix } from "../models/config";
import { IDeviceConfig } from "../models/config/deviceConfig";
import { IHub } from "../models/config/hub";
import { HarmonyDevice } from "../models/harmonyDevice";
import { HarmonyHub } from "../models/harmonyHub";
export class HarmonyDataProvider2 {
private _matrix: IMatrix;
private _hubs: { [hubName: string]: HarmonyHub } = {};
private _hubsByDevice: { [deviceName: string]: string } = {};
constructor(
@inject("IConfig") private _config: IConfig,
@inject("log") private _log: Logging
) {
this._matrix = _config.Matrix;
_config.Devices.forEach((deviceConfig: IDeviceConfig) => {
this._hubsByDevice[deviceConfig.Name] = deviceConfig.Hub;
});
// this._deviceConfigs = props.deviceConfigs;
this.connect(_config.Hubs);
this.emitInfo();
}
public async turnOnDevices(devices: Array<HarmonyDevice>): Promise<void> {
await Promise.all(
devices.map(async (device: HarmonyDevice) => {
if (device && device.name) {
if (!device.on) {
this._log.info(`Turning on device ${device.name}`);
await device.powerOn();
}
}
})
);
}
public async turnOffDevices(devices: Array<HarmonyDevice>) {
await Promise.all(
//Turn off devices
devices.map(async (device: HarmonyDevice) => {
if (device) {
if (device.on) {
this._log.info(`Turning off device ${device.name}`);
await device.powerOff();
}
}
})
);
}
/**
* Get the IDevice by name.
* @param deviceName The device to retrieve.
*/
public getDeviceFromName(deviceName: string): HarmonyDevice {
let device: HarmonyDevice | undefined;
try {
device =
this._hubs[this._hubsByDevice[deviceName]].getDeviceByName(deviceName);
} catch (err) {
this._log.info(`Error retrieving device from hub: ${err}`);
}
return device!;
}
private connect = async (hubs: Array<IHub>) => {
let readyCount = 0;
await Promise.all(
hubs.map(async (hub: IHub): Promise<void> => {
const newHarmonyHub = new HarmonyHub(hub.Name, hub.Ip, this._log.info);
this._hubs[hub.Name] = newHarmonyHub;
newHarmonyHub.on("Ready", () => {
readyCount++;
if (readyCount === Object.keys(this._hubs).length) {
// this.emit("Ready");
}
});
await newHarmonyHub.initialize();
})
);
};
private emitInfo(): void {
//Emit devices if requested
this._log.info("All hubs connected");
if (this._config.EmitDevicesOnStartup) {
const hubs = this._hubs;
Object.values(hubs).forEach((hub: HarmonyHub) => {
const deviceDictionary = hub.devices;
this._log.info(`${hub.hubName}`);
Object.values(deviceDictionary).forEach((device: HarmonyDevice) => {
this._log.info(` ${device.name} : ${device.id}`);
Object.keys(device.commands).forEach((command: string) => {
this._log.info(` ${command}`);
});
});
});
}
}
}

View File

@ -10,12 +10,11 @@ import {
} from "homebridge"; } from "homebridge";
import { ControlUnit, DeviceButton } from "./accessories"; import { ControlUnit, DeviceButton } from "./accessories";
import { Sequence } from "./accessories/sequence"; import { Sequence } from "./accessories/sequence";
import HarmonyDataProvider from "./dataProviders/harmonyDataProvider";
import { IConfig, IControlUnit, IDeviceButton } from "./models/config"; import { IConfig, IControlUnit, IDeviceButton } from "./models/config";
import { ISequence } from "./models/config/sequence"; import { ISequence } from "./models/config/sequence";
import { PLATFORM_NAME, PLUGIN_NAME } from "./settings"; import { PLATFORM_NAME, PLUGIN_NAME } from "./settings";
import { container } from "tsyringe"; import { container } from "tsyringe";
import { HarmonyDataProvider2 } from "./dataProviders/harmonyDataProvider2"; import { HarmonyDataProvider } from "./dataProviders/harmonyDataProvider";
import { StateDataProvider } from "./dataProviders/stateDataProvider"; import { StateDataProvider } from "./dataProviders/stateDataProvider";
import { CommandService } from "./services/commandService"; import { CommandService } from "./services/commandService";
import { ActivityService } from "./services/activityService"; import { ActivityService } from "./services/activityService";
@ -221,9 +220,9 @@ export class Platform implements DynamicPlatformPlugin {
container.register<Platform>(Platform, { useValue: this }); container.register<Platform>(Platform, { useValue: this });
container.register<IConfig>("IConfig", { useValue: this.config }); container.register<IConfig>("IConfig", { useValue: this.config });
container.register<Logger>("log", { useValue: this.log }); container.register<Logger>("log", { useValue: this.log });
container.registerSingleton<HarmonyDataProvider>(HarmonyDataProvider);
container.registerSingleton<ActivityService>(ActivityService);
container.registerSingleton<StateDataProvider>(StateDataProvider); container.registerSingleton<StateDataProvider>(StateDataProvider);
container.registerSingleton<CommandService>(CommandService); container.registerSingleton<CommandService>(CommandService);
container.registerSingleton<HarmonyDataProvider>(HarmonyDataProvider);
container.registerSingleton<HarmonyDataProvider2>(HarmonyDataProvider2);
} }
} }

View File

@ -1,6 +1,6 @@
import { Logging } from "homebridge"; import { Logging } from "homebridge";
import { inject, injectable } from "tsyringe"; import { inject, injectable } from "tsyringe";
import { HarmonyDataProvider2 } from "../dataProviders/harmonyDataProvider2"; import { HarmonyDataProvider } from "../dataProviders/harmonyDataProvider";
import { StateDataProvider } from "../dataProviders/stateDataProvider"; import { StateDataProvider } from "../dataProviders/stateDataProvider";
import { IActivityState } from "../models/activityState"; import { IActivityState } from "../models/activityState";
import { import {
@ -15,8 +15,8 @@ import { HarmonyDevice } from "../models/harmonyDevice";
@injectable() @injectable()
export class ActivityService { export class ActivityService {
constructor( constructor(
@inject(HarmonyDataProvider2) @inject(HarmonyDataProvider)
private _harmonyDataProvider: HarmonyDataProvider2, private _harmonyDataProvider: HarmonyDataProvider,
@inject(StateDataProvider) private _stateDataProvider: StateDataProvider, @inject(StateDataProvider) private _stateDataProvider: StateDataProvider,
@inject("IConfig") private _config: IConfig, @inject("IConfig") private _config: IConfig,
@inject("log") private _log: Logging @inject("log") private _log: Logging

View File

@ -1,15 +1,15 @@
import { inject } from "tsyringe"; import { inject, injectable } from "tsyringe";
import { RemoteKey } from "../accessories/controlUnit"; import { RemoteKey } from "../accessories/controlUnit";
import { HarmonyDataProvider2 } from "../dataProviders/harmonyDataProvider2"; import { HarmonyDataProvider } from "../dataProviders/harmonyDataProvider";
import { StateDataProvider } from "../dataProviders/stateDataProvider"; import { StateDataProvider } from "../dataProviders/stateDataProvider";
import { IConfig } from "../models/config";
import { HarmonyDevice } from "../models/harmonyDevice"; import { HarmonyDevice } from "../models/harmonyDevice";
@injectable()
export class CommandService { export class CommandService {
constructor( constructor(
@inject(StateDataProvider) private _stateDataProvider: StateDataProvider, @inject(StateDataProvider) private _stateDataProvider: StateDataProvider,
@inject(HarmonyDataProvider2) @inject(HarmonyDataProvider)
private _harmonyDataProvider: HarmonyDataProvider2 private _harmonyDataProvider: HarmonyDataProvider
) {} ) {}
/** /**

View File

@ -1,15 +1,14 @@
import { Logging } from "homebridge"; import { inject, injectable } from "tsyringe";
import { inject } from "tsyringe"; import { HarmonyDataProvider } from "../dataProviders/harmonyDataProvider";
import { HarmonyDataProvider2 } from "../dataProviders/harmonyDataProvider2";
import { StateDataProvider } from "../dataProviders/stateDataProvider"; import { StateDataProvider } from "../dataProviders/stateDataProvider";
import { IConfig } from "../models/config";
import { HarmonyDevice } from "../models/harmonyDevice"; import { HarmonyDevice } from "../models/harmonyDevice";
@injectable()
export class VolumeService { export class VolumeService {
constructor( constructor(
@inject(StateDataProvider) private _stateDataProvider: StateDataProvider, @inject(StateDataProvider) private _stateDataProvider: StateDataProvider,
@inject(HarmonyDataProvider2) @inject(HarmonyDataProvider)
private _harmonyDataProvider: HarmonyDataProvider2 private _harmonyDataProvider: HarmonyDataProvider
) {} ) {}
/** /**