platform.ts 7.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204
  1. import { API, DynamicPlatformPlugin, Logger, PlatformAccessory, PlatformConfig, Service, Characteristic } from 'homebridge'
  2. import { PLATFORM_NAME, PLUGIN_NAME } from './settings'
  3. import { accessory } from './accessory'
  4. import { device } from './device'
  5. import { klikAanKlikUitAccessory } from './klikAanKlikUitAccessory'
  6. import { powerSocketAccessory } from './powerSocketAccessory'
  7. import { powerSwitchAccessory } from './powerSwitchAccessory'
  8. import { environmentSensorAccessory } from './environmentSensorAccessory'
  9. import { doorWindowSensorAccessory } from './doorWindowSensorAccessory'
  10. import { motionSensorAccessory } from './motionSensorAccessory'
  11. import { cameraAccessory } from './cameraAccessory'
  12. import { httpServer } from './httpServer'
  13. export class domoticaPlatform implements DynamicPlatformPlugin {
  14. public readonly Service: typeof Service = this.api.hap.Service
  15. public readonly Characteristic: typeof Characteristic = this.api.hap.Characteristic
  16. public readonly accessories: accessory[] = []
  17. public readonly cachedAccessories: PlatformAccessory[] = []
  18. public readonly presentAccessories: PlatformAccessory[] = []
  19. private readonly discoveries: String[] = []
  20. private readonly completedDiscoveries: String[] = []
  21. private httpServer: httpServer = new httpServer(this, this.config)
  22. constructor(
  23. public readonly log: Logger,
  24. public readonly config: PlatformConfig,
  25. public readonly api: API
  26. ) {
  27. this.api.on('didFinishLaunching', () => {
  28. this.discoverDevices()
  29. this.httpServer.start()
  30. })
  31. }
  32. configureAccessory(accessory: PlatformAccessory) {
  33. this.cachedAccessories.push(accessory)
  34. }
  35. discoverDevices() {
  36. this.discoveryStarted('klikAanKlikUitAccessory')
  37. klikAanKlikUitAccessory.discoverDevices(this)
  38. .then((devices) => {
  39. this.processDiscoveredDevices(devices)
  40. this.discoveryCompleted('klikAanKlikUitAccessory')
  41. })
  42. .catch((error) => {
  43. this.log.debug('klikAanKlikUitAccessory.discoverDevices Error ->' + error)
  44. })
  45. this.discoveryStarted('powerSocketAccessory')
  46. powerSocketAccessory.discoverDevices(this)
  47. .then((devices) => {
  48. this.processDiscoveredDevices(devices)
  49. this.discoveryCompleted('powerSocketAccessory')
  50. })
  51. .catch((error) => {
  52. this.log.debug('powerSocketAccessory.discoverDevices Error ->' + error)
  53. })
  54. this.discoveryStarted('powerSwitchAccessory')
  55. powerSwitchAccessory.discoverDevices(this)
  56. .then((devices) => {
  57. this.processDiscoveredDevices(devices)
  58. this.discoveryCompleted('powerSwitchAccessory')
  59. })
  60. .catch((error) => {
  61. this.log.debug('powerSwitchAccessory.discoverDevices Error ->' + error)
  62. })
  63. this.discoveryStarted('environmentSensorAccessory')
  64. environmentSensorAccessory.discoverDevices(this)
  65. .then((devices) => {
  66. this.processDiscoveredDevices(devices)
  67. this.discoveryCompleted('environmentSensorAccessory')
  68. })
  69. .catch((error) => {
  70. this.log.debug('environmentSensorAccessory.discoverDevices Error ->' + error)
  71. })
  72. this.discoveryStarted('doorWindowSensorAccessory')
  73. doorWindowSensorAccessory.discoverDevices(this)
  74. .then((devices) => {
  75. this.processDiscoveredDevices(devices)
  76. this.discoveryCompleted('doorWindowSensorAccessory')
  77. })
  78. .catch((error) => {
  79. this.log.debug('doorWindowSensorAccessory.discoverDevices Error ->' + error)
  80. })
  81. this.discoveryStarted('motionSensorAccessory')
  82. motionSensorAccessory.discoverDevices(this)
  83. .then((devices) => {
  84. this.processDiscoveredDevices(devices)
  85. this.discoveryCompleted('motionSensorAccessory')
  86. })
  87. .catch((error) => {
  88. this.log.debug('motionSensorAccessory.discoverDevices Error ->' + error)
  89. })
  90. this.discoveryStarted('cameraAccessory')
  91. cameraAccessory.discoverDevices(this)
  92. .then((devices) => {
  93. this.processDiscoveredDevices(devices)
  94. this.discoveryCompleted('cameraAccessory')
  95. })
  96. .catch((error) => {
  97. this.log.debug('cameraAccessory.discoverDevices Error ->' + error)
  98. })
  99. }
  100. processDiscoveredDevices(domoticaDevices: device[]) {
  101. for (const device of domoticaDevices) {
  102. const uuid = this.api.hap.uuid.generate(device.uniqueId)
  103. const existingAccessory = this.cachedAccessories.find(accessory => accessory.UUID === uuid)
  104. if (existingAccessory) {
  105. this.log.info('| ' + existingAccessory.context.device.type + ': ' + existingAccessory.displayName)
  106. this.presentAccessories.push(existingAccessory)
  107. if (existingAccessory.displayName != device.deviceName) {
  108. this.log.info('! ' + existingAccessory.displayName + ' => ' + device.deviceName)
  109. existingAccessory.context.device = device
  110. this.api.updatePlatformAccessories([existingAccessory])
  111. }
  112. if (existingAccessory.context.device.type == 'KlikAanKlikUit') {
  113. this.accessories.push(new klikAanKlikUitAccessory(this, existingAccessory))
  114. } else if (existingAccessory.context.device.type == 'PowerSocket') {
  115. this.accessories.push(new powerSocketAccessory(this, existingAccessory))
  116. } else if (existingAccessory.context.device.type == 'PowerSwitch') {
  117. this.accessories.push(new powerSwitchAccessory(this, existingAccessory))
  118. } else if (existingAccessory.context.device.type == 'EnvironmentSensor') {
  119. this.accessories.push(new environmentSensorAccessory(this, existingAccessory))
  120. } else if (existingAccessory.context.device.type == 'DoorWindowSensor') {
  121. this.accessories.push(new doorWindowSensorAccessory(this, existingAccessory))
  122. } else if (existingAccessory.context.device.type == 'MotionSensor') {
  123. this.accessories.push(new motionSensorAccessory(this, existingAccessory))
  124. } else if (existingAccessory.context.device.type == 'Camera') {
  125. this.accessories.push(new cameraAccessory(this, existingAccessory))
  126. }
  127. } else {
  128. this.log.info('+ ' + device.type + ': ' + device.deviceName)
  129. const accessory = new this.api.platformAccessory(device.deviceName, uuid)
  130. accessory.context.device = device
  131. if (accessory.context.device.type == 'KlikAanKlikUit') {
  132. this.accessories.push(new klikAanKlikUitAccessory(this, accessory))
  133. } else if (accessory.context.device.type == 'PowerSocket') {
  134. this.accessories.push(new powerSocketAccessory(this, accessory))
  135. } else if (accessory.context.device.type == 'PowerSwitch') {
  136. this.accessories.push(new powerSwitchAccessory(this, accessory))
  137. } else if (accessory.context.device.type == 'EnvironmentSensor') {
  138. this.accessories.push(new environmentSensorAccessory(this, accessory))
  139. } else if (accessory.context.device.type == 'DoorWindowSensor') {
  140. this.accessories.push(new doorWindowSensorAccessory(this, accessory))
  141. } else if (accessory.context.device.type == 'MotionSensor') {
  142. this.accessories.push(new motionSensorAccessory(this, accessory))
  143. } else if (accessory.context.device.type == 'Camera') {
  144. this.accessories.push(new cameraAccessory(this, accessory))
  145. }
  146. this.api.registerPlatformAccessories(PLUGIN_NAME, PLATFORM_NAME, [accessory])
  147. }
  148. }
  149. }
  150. discoveryStarted(deviceType) {
  151. this.discoveries.push(deviceType)
  152. }
  153. discoveryCompleted(deviceType) {
  154. this.completedDiscoveries.push(deviceType)
  155. if (this.completedDiscoveries.length == this.discoveries.length) {
  156. this.cleanupRemovedDevices()
  157. }
  158. //setInterval(this.updateAccessories.bind(this), 60 * 1000); // Poll every minute
  159. }
  160. updateAccessories() {
  161. for (const accessory of this.accessories) {
  162. accessory.update()
  163. }
  164. }
  165. cleanupRemovedDevices() {
  166. for (const existingAccessory of this.cachedAccessories) {
  167. const presentAccessory = this.presentAccessories.find(accessory => accessory.UUID === existingAccessory.UUID)
  168. if (!presentAccessory) {
  169. this.log.info('- ' + existingAccessory.context.device.type + ': ' + existingAccessory.displayName)
  170. this.api.unregisterPlatformAccessories(PLUGIN_NAME, PLATFORM_NAME, [existingAccessory])
  171. }
  172. }
  173. }
  174. }