camstreamerlib 4.0.0-beta.3 → 4.0.0-beta.31

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (194) hide show
  1. package/README.md +8 -2
  2. package/cjs/CamOverlayAPI.d.ts +48 -26
  3. package/cjs/CamOverlayAPI.js +171 -88
  4. package/cjs/CamOverlayDrawingAPI.d.ts +2 -47
  5. package/cjs/CamOverlayDrawingAPI.js +6 -3
  6. package/cjs/CamOverlayPainter/Frame.d.ts +8 -37
  7. package/cjs/CamOverlayPainter/Frame.js +33 -0
  8. package/cjs/CamOverlayPainter/Painter.d.ts +16 -10
  9. package/cjs/CamOverlayPainter/Painter.js +6 -5
  10. package/cjs/CamOverlayPainter/ResourceManager.d.ts +3 -2
  11. package/cjs/CamOverlayPainter/ResourceManager.js +8 -11
  12. package/cjs/CamScripterAPI.d.ts +34 -16
  13. package/cjs/CamScripterAPI.js +50 -41
  14. package/cjs/CamScripterAPICameraEventsGenerator.d.ts +1 -1
  15. package/cjs/CamScripterAPICameraEventsGenerator.js +6 -3
  16. package/cjs/CamStreamerAPI.d.ts +17 -14
  17. package/cjs/CamStreamerAPI.js +32 -32
  18. package/cjs/CamSwitcherAPI.d.ts +43 -37
  19. package/cjs/CamSwitcherAPI.js +123 -114
  20. package/cjs/CamSwitcherEvents.d.ts +1 -1
  21. package/cjs/PlaneTrackerAPI.d.ts +43 -0
  22. package/cjs/PlaneTrackerAPI.js +211 -0
  23. package/cjs/VapixAPI.d.ts +56 -42
  24. package/cjs/VapixAPI.js +305 -216
  25. package/cjs/VapixEvents.d.ts +1 -1
  26. package/cjs/VapixEvents.js +3 -3
  27. package/cjs/errors/errors.d.ts +3 -0
  28. package/cjs/errors/errors.js +8 -1
  29. package/cjs/events/AxisCameraStationEvents.d.ts +7 -4
  30. package/cjs/events/AxisCameraStationEvents.js +23 -18
  31. package/cjs/events/GenetecAgent.d.ts +6 -3
  32. package/cjs/events/GenetecAgent.js +30 -19
  33. package/cjs/index.d.ts +14 -1
  34. package/cjs/index.js +23 -2
  35. package/cjs/internal/Digest.js +6 -6
  36. package/cjs/internal/ProxyClient.d.ts +8 -9
  37. package/cjs/internal/ProxyClient.js +25 -29
  38. package/cjs/internal/types.d.ts +42 -0
  39. package/cjs/internal/types.js +2 -0
  40. package/cjs/internal/utils.d.ts +4 -1
  41. package/cjs/internal/utils.js +22 -3
  42. package/cjs/internal/versionCompare.d.ts +2 -2
  43. package/cjs/node/DefaultClient.d.ts +5 -6
  44. package/cjs/node/DefaultClient.js +12 -14
  45. package/cjs/node/HttpRequestSender.d.ts +1 -0
  46. package/cjs/node/HttpRequestSender.js +13 -3
  47. package/cjs/node/HttpServer.js +1 -1
  48. package/cjs/node/WsClient.d.ts +2 -1
  49. package/cjs/node/index.d.ts +2 -0
  50. package/cjs/node/index.js +18 -1
  51. package/cjs/types/CamOverlayAPI/CamOverlayAPI.d.ts +3071 -0
  52. package/cjs/types/CamOverlayAPI/CamOverlayAPI.js +127 -0
  53. package/cjs/types/CamOverlayAPI/accuweatherSchema.d.ts +114 -0
  54. package/cjs/types/CamOverlayAPI/accuweatherSchema.js +50 -0
  55. package/cjs/types/CamOverlayAPI/customGraphicsSchema.d.ts +783 -0
  56. package/cjs/types/CamOverlayAPI/customGraphicsSchema.js +75 -0
  57. package/cjs/types/CamOverlayAPI/imagesSchema.d.ts +122 -0
  58. package/cjs/types/CamOverlayAPI/imagesSchema.js +12 -0
  59. package/cjs/types/CamOverlayAPI/index.d.ts +9 -0
  60. package/cjs/types/CamOverlayAPI/index.js +25 -0
  61. package/cjs/types/CamOverlayAPI/infotickerSchema.d.ts +130 -0
  62. package/cjs/types/CamOverlayAPI/infotickerSchema.js +29 -0
  63. package/cjs/types/CamOverlayAPI/pipSchema.d.ts +166 -0
  64. package/cjs/types/CamOverlayAPI/pipSchema.js +42 -0
  65. package/cjs/types/CamOverlayAPI/ptzCompassSchema.d.ts +126 -0
  66. package/cjs/types/CamOverlayAPI/ptzCompassSchema.js +28 -0
  67. package/cjs/types/CamOverlayAPI/ptzSchema.d.ts +146 -0
  68. package/cjs/types/CamOverlayAPI/ptzSchema.js +15 -0
  69. package/cjs/types/CamOverlayAPI/screenSharingSchema.d.ts +79 -0
  70. package/cjs/types/CamOverlayAPI/screenSharingSchema.js +11 -0
  71. package/cjs/types/CamOverlayAPI/webCameraSharingSchema.d.ts +79 -0
  72. package/cjs/types/CamOverlayAPI/webCameraSharingSchema.js +11 -0
  73. package/cjs/types/CamOverlayDrawingAPI.d.ts +58 -0
  74. package/cjs/types/CamOverlayDrawingAPI.js +2 -0
  75. package/cjs/types/CamOverlayPainter.d.ts +74 -0
  76. package/cjs/types/CamOverlayPainter.js +2 -0
  77. package/cjs/types/CamScripterAPI.d.ts +82 -17
  78. package/cjs/types/CamScripterAPI.js +22 -7
  79. package/cjs/types/CamStreamerAPI.d.ts +16 -5
  80. package/cjs/types/CamStreamerAPI.js +5 -1
  81. package/cjs/types/CamSwitcherAPI.d.ts +4 -6
  82. package/cjs/types/CamSwitcherEvents.d.ts +77 -0
  83. package/cjs/types/CamSwitcherEvents.js +8 -0
  84. package/cjs/types/PlaneTrackerAPI.d.ts +8 -0
  85. package/cjs/types/PlaneTrackerAPI.js +2 -0
  86. package/cjs/types/VapixAPI.d.ts +635 -520
  87. package/cjs/types/VapixAPI.js +62 -24
  88. package/cjs/types/common.d.ts +14 -5
  89. package/cjs/web/DefaultClient.d.ts +5 -5
  90. package/cjs/web/DefaultClient.js +22 -10
  91. package/cjs/web/WsClient.js +2 -2
  92. package/esm/CamOverlayAPI.d.ts +48 -26
  93. package/esm/CamOverlayAPI.js +167 -84
  94. package/esm/CamOverlayDrawingAPI.d.ts +2 -47
  95. package/esm/CamOverlayDrawingAPI.js +6 -3
  96. package/esm/CamOverlayPainter/Frame.d.ts +8 -37
  97. package/esm/CamOverlayPainter/Frame.js +33 -0
  98. package/esm/CamOverlayPainter/Painter.d.ts +16 -10
  99. package/esm/CamOverlayPainter/Painter.js +5 -3
  100. package/esm/CamOverlayPainter/ResourceManager.d.ts +3 -2
  101. package/esm/CamOverlayPainter/ResourceManager.js +7 -11
  102. package/esm/CamScripterAPI.d.ts +34 -16
  103. package/esm/CamScripterAPI.js +46 -37
  104. package/esm/CamScripterAPICameraEventsGenerator.d.ts +1 -1
  105. package/esm/CamScripterAPICameraEventsGenerator.js +6 -3
  106. package/esm/CamStreamerAPI.d.ts +17 -14
  107. package/esm/CamStreamerAPI.js +32 -32
  108. package/esm/CamSwitcherAPI.d.ts +43 -37
  109. package/esm/CamSwitcherAPI.js +116 -107
  110. package/esm/CamSwitcherEvents.d.ts +1 -1
  111. package/esm/PlaneTrackerAPI.d.ts +43 -0
  112. package/esm/PlaneTrackerAPI.js +207 -0
  113. package/esm/VapixAPI.d.ts +56 -42
  114. package/esm/VapixAPI.js +297 -208
  115. package/esm/VapixEvents.d.ts +1 -1
  116. package/esm/VapixEvents.js +3 -3
  117. package/esm/errors/errors.d.ts +3 -0
  118. package/esm/errors/errors.js +6 -0
  119. package/esm/events/AxisCameraStationEvents.d.ts +7 -4
  120. package/esm/events/AxisCameraStationEvents.js +18 -13
  121. package/esm/events/GenetecAgent.d.ts +6 -3
  122. package/esm/events/GenetecAgent.js +20 -9
  123. package/esm/index.d.ts +14 -1
  124. package/esm/index.js +14 -1
  125. package/esm/internal/Digest.js +6 -6
  126. package/esm/internal/ProxyClient.d.ts +8 -9
  127. package/esm/internal/ProxyClient.js +25 -29
  128. package/esm/internal/types.d.ts +42 -0
  129. package/esm/internal/types.js +1 -0
  130. package/esm/internal/utils.d.ts +4 -1
  131. package/esm/internal/utils.js +17 -1
  132. package/esm/internal/versionCompare.d.ts +2 -2
  133. package/esm/node/DefaultClient.d.ts +5 -6
  134. package/esm/node/DefaultClient.js +12 -14
  135. package/esm/node/HttpRequestSender.d.ts +1 -0
  136. package/esm/node/HttpRequestSender.js +13 -3
  137. package/esm/node/HttpServer.js +1 -1
  138. package/esm/node/WsClient.d.ts +2 -1
  139. package/esm/node/index.d.ts +2 -0
  140. package/esm/node/index.js +2 -0
  141. package/esm/types/CamOverlayAPI/CamOverlayAPI.d.ts +3071 -0
  142. package/esm/types/CamOverlayAPI/CamOverlayAPI.js +124 -0
  143. package/esm/types/CamOverlayAPI/accuweatherSchema.d.ts +114 -0
  144. package/esm/types/CamOverlayAPI/accuweatherSchema.js +46 -0
  145. package/esm/types/CamOverlayAPI/customGraphicsSchema.d.ts +783 -0
  146. package/esm/types/CamOverlayAPI/customGraphicsSchema.js +71 -0
  147. package/esm/types/CamOverlayAPI/imagesSchema.d.ts +122 -0
  148. package/esm/types/CamOverlayAPI/imagesSchema.js +8 -0
  149. package/esm/types/CamOverlayAPI/index.d.ts +9 -0
  150. package/esm/types/CamOverlayAPI/index.js +9 -0
  151. package/esm/types/CamOverlayAPI/infotickerSchema.d.ts +130 -0
  152. package/esm/types/CamOverlayAPI/infotickerSchema.js +25 -0
  153. package/esm/types/CamOverlayAPI/pipSchema.d.ts +166 -0
  154. package/esm/types/CamOverlayAPI/pipSchema.js +38 -0
  155. package/esm/types/CamOverlayAPI/ptzCompassSchema.d.ts +126 -0
  156. package/esm/types/CamOverlayAPI/ptzCompassSchema.js +24 -0
  157. package/esm/types/CamOverlayAPI/ptzSchema.d.ts +146 -0
  158. package/esm/types/CamOverlayAPI/ptzSchema.js +11 -0
  159. package/esm/types/CamOverlayAPI/screenSharingSchema.d.ts +79 -0
  160. package/esm/types/CamOverlayAPI/screenSharingSchema.js +7 -0
  161. package/esm/types/CamOverlayAPI/webCameraSharingSchema.d.ts +79 -0
  162. package/esm/types/CamOverlayAPI/webCameraSharingSchema.js +7 -0
  163. package/esm/types/CamOverlayDrawingAPI.d.ts +58 -0
  164. package/esm/types/CamOverlayDrawingAPI.js +1 -0
  165. package/esm/types/CamOverlayPainter.d.ts +74 -0
  166. package/esm/types/CamOverlayPainter.js +1 -0
  167. package/esm/types/CamScripterAPI.d.ts +82 -17
  168. package/esm/types/CamScripterAPI.js +21 -6
  169. package/esm/types/CamStreamerAPI.d.ts +16 -5
  170. package/esm/types/CamStreamerAPI.js +4 -0
  171. package/esm/types/CamSwitcherAPI.d.ts +4 -6
  172. package/esm/types/CamSwitcherEvents.d.ts +77 -0
  173. package/esm/types/CamSwitcherEvents.js +8 -0
  174. package/esm/types/PlaneTrackerAPI.d.ts +8 -0
  175. package/esm/types/PlaneTrackerAPI.js +1 -0
  176. package/esm/types/VapixAPI.d.ts +635 -520
  177. package/esm/types/VapixAPI.js +61 -23
  178. package/esm/types/common.d.ts +14 -5
  179. package/esm/web/DefaultClient.d.ts +5 -5
  180. package/esm/web/DefaultClient.js +22 -10
  181. package/esm/web/WsClient.js +2 -2
  182. package/package.json +9 -8
  183. package/cjs/internal/common.d.ts +0 -39
  184. package/cjs/internal/common.js +0 -27
  185. package/cjs/node/WsEventClient.d.ts +0 -13
  186. package/cjs/node/WsEventClient.js +0 -22
  187. package/cjs/types/CamOverlayAPI.d.ts +0 -188
  188. package/cjs/types/CamOverlayAPI.js +0 -47
  189. package/esm/internal/common.d.ts +0 -39
  190. package/esm/internal/common.js +0 -20
  191. package/esm/node/WsEventClient.d.ts +0 -13
  192. package/esm/node/WsEventClient.js +0 -18
  193. package/esm/types/CamOverlayAPI.d.ts +0 -188
  194. package/esm/types/CamOverlayAPI.js +0 -44
package/cjs/VapixAPI.js CHANGED
@@ -2,57 +2,67 @@
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
3
  exports.VapixAPI = void 0;
4
4
  const prettifyXml = require("prettify-xml");
5
- const xml2js_1 = require("xml2js");
6
- const common_1 = require("./internal/common");
5
+ const utils_1 = require("./internal/utils");
7
6
  const VapixAPI_1 = require("./types/VapixAPI");
8
7
  const errors_1 = require("./errors/errors");
9
8
  const ProxyClient_1 = require("./internal/ProxyClient");
10
- const utils_1 = require("./internal/utils");
11
9
  const zod_1 = require("zod");
10
+ const fast_xml_parser_1 = require("fast-xml-parser");
12
11
  class VapixAPI {
13
12
  client;
14
- constructor(client, getProxyUrl) {
15
- this.client = new ProxyClient_1.ProxyClient(client, getProxyUrl);
13
+ constructor(client) {
14
+ this.client = client;
15
+ }
16
+ getClient(proxyParams) {
17
+ return proxyParams ? new ProxyClient_1.ProxyClient(this.client, proxyParams) : this.client;
16
18
  }
17
- async getUrlEncoded(proxy = null, path, parameters, headers = {}) {
19
+ async postUrlEncoded(path, parameters, headers, options) {
18
20
  const data = (0, utils_1.paramToUrl)(parameters);
19
21
  const head = { ...headers, 'Content-Type': 'application/x-www-form-urlencoded' };
20
- const res = await this.client.post(proxy, path, data, {}, head);
22
+ const agent = this.getClient(options?.proxyParams);
23
+ const res = await agent.post({ path, data, headers: head, timeout: options?.timeout });
21
24
  if (!res.ok) {
22
- throw new Error(await (0, common_1.responseStringify)(res));
25
+ throw new Error(await (0, utils_1.responseStringify)(res));
23
26
  }
24
27
  return res;
25
28
  }
26
- async postJson(proxy = null, path, jsonData, headers = {}) {
29
+ async postJson(path, jsonData, headers, options) {
27
30
  const data = JSON.stringify(jsonData);
28
31
  const head = { ...headers, 'Content-Type': 'application/json' };
29
- const res = await this.client.post(proxy, path, data, {}, head);
32
+ const agent = this.getClient(options?.proxyParams);
33
+ const res = await agent.post({ path, data, headers: head, timeout: options?.timeout });
30
34
  if (!res.ok) {
31
- throw new Error(await (0, common_1.responseStringify)(res));
35
+ throw new Error(await (0, utils_1.responseStringify)(res));
32
36
  }
33
37
  return res;
34
38
  }
35
- async getCameraImage(params, proxy = null) {
36
- return await this.client.get(proxy, '/axis-cgi/jpg/image.cgi', params);
39
+ async getCameraImage(parameters, options) {
40
+ const agent = this.getClient(options?.proxyParams);
41
+ return await agent.get({ path: '/axis-cgi/jpg/image.cgi', parameters, timeout: options?.timeout });
37
42
  }
38
- async getEventDeclarations(proxy = null) {
43
+ async getEventDeclarations(options) {
39
44
  const data = '<s:Envelope xmlns:s="http://www.w3.org/2003/05/soap-envelope">' +
40
45
  '<s:Body xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"' +
41
46
  'xmlns:xsd="http://www.w3.org/2001/XMLSchema">' +
42
47
  '<GetEventInstances xmlns="http://www.axis.com/vapix/ws/event1"/>' +
43
48
  '</s:Body>' +
44
49
  '</s:Envelope>';
45
- const res = await this.client.post(proxy, '/vapix/services', data, { 'Content-Type': 'application/soap+xml' });
50
+ const agent = this.getClient(options?.proxyParams);
51
+ const res = await agent.post({
52
+ path: '/vapix/services',
53
+ data,
54
+ headers: { 'Content-Type': 'application/soap+xml' },
55
+ });
46
56
  if (!res.ok) {
47
- throw new Error(await (0, common_1.responseStringify)(res));
57
+ throw new Error(await (0, utils_1.responseStringify)(res));
48
58
  }
49
59
  const declarations = await res.text();
50
60
  return prettifyXml(declarations);
51
61
  }
52
- async getSupportedAudioSampleRate(proxy = null) {
53
- const url = '/axis-cgi/audio/streamingcapabilities.cgi';
54
- const formData = { apiVersion: '1.0', method: 'list' };
55
- const res = await this.postJson(proxy, url, formData);
62
+ async getSupportedAudioSampleRate(options) {
63
+ const path = '/axis-cgi/audio/streamingcapabilities.cgi';
64
+ const jsonData = { apiVersion: '1.0', method: 'list' };
65
+ const res = await this.postJson(path, jsonData, undefined, options);
56
66
  const encoders = VapixAPI_1.audioSampleRatesResponseSchema.parse(await res.json()).data.encoders;
57
67
  const data = encoders.aac ?? encoders.AAC ?? [];
58
68
  return data.map((item) => {
@@ -62,7 +72,7 @@ class VapixAPI {
62
72
  };
63
73
  });
64
74
  }
65
- async performAutofocus(proxy = null) {
75
+ async performAutofocus(options) {
66
76
  try {
67
77
  const data = {
68
78
  apiVersion: '1',
@@ -75,24 +85,26 @@ class VapixAPI {
75
85
  ],
76
86
  },
77
87
  };
78
- await this.postJson(proxy, '/axis-cgi/opticscontrol.cgi', data);
88
+ await this.postJson('/axis-cgi/opticscontrol.cgi', data, undefined, options);
79
89
  }
80
90
  catch (err) {
81
- await this.postJson(proxy, '/axis-cgi/opticssetup.cgi', {
91
+ await this.postUrlEncoded('/axis-cgi/opticssetup.cgi', {
82
92
  autofocus: 'perform',
83
93
  source: '1',
84
- });
94
+ }, undefined, options);
85
95
  }
86
96
  }
87
- async checkSDCard(proxy = null) {
88
- const res = await this.getUrlEncoded(proxy, '/axis-cgi/disks/list.cgi', {
97
+ async checkSDCard(options) {
98
+ const res = await this.postUrlEncoded('/axis-cgi/disks/list.cgi', {
89
99
  diskid: 'SD_DISK',
100
+ }, undefined, options);
101
+ const xmlText = await res.text();
102
+ const parser = new fast_xml_parser_1.XMLParser({
103
+ ignoreAttributes: false,
104
+ attributeNamePrefix: '',
105
+ allowBooleanAttributes: true,
90
106
  });
91
- const result = await (0, xml2js_1.parseStringPromise)(await res.text(), {
92
- ignoreAttrs: false,
93
- mergeAttrs: true,
94
- explicitArray: false,
95
- });
107
+ const result = parser.parse(xmlText);
96
108
  const data = result.root.disks.disk;
97
109
  return {
98
110
  totalSize: parseInt(data.totalsize),
@@ -100,53 +112,56 @@ class VapixAPI {
100
112
  status: VapixAPI_1.sdCardWatchedStatuses.includes(data.status) ? data.status : 'disconnected',
101
113
  };
102
114
  }
103
- mountSDCard(proxy = null) {
104
- return this._doSDCardMountAction('MOUNT', proxy);
115
+ mountSDCard(options) {
116
+ return this._doSDCardMountAction('MOUNT', options);
105
117
  }
106
- unmountSDCard(proxy = null) {
107
- return this._doSDCardMountAction('UNMOUNT', proxy);
118
+ unmountSDCard(options) {
119
+ return this._doSDCardMountAction('UNMOUNT', options);
108
120
  }
109
- async _doSDCardMountAction(action, proxy = null) {
110
- const res = await this.getUrlEncoded(proxy, '/axis-cgi/disks/mount.cgi', {
121
+ async _doSDCardMountAction(action, options) {
122
+ const res = await this.postUrlEncoded('/axis-cgi/disks/mount.cgi', {
111
123
  action: action,
112
124
  diskid: 'SD_DISK',
125
+ }, undefined, options);
126
+ const textXml = await res.text();
127
+ const parser = new fast_xml_parser_1.XMLParser({
128
+ ignoreAttributes: false,
129
+ attributeNamePrefix: '',
130
+ allowBooleanAttributes: true,
113
131
  });
114
- const result = await (0, xml2js_1.parseStringPromise)(await res.text(), {
115
- ignoreAttrs: false,
116
- mergeAttrs: true,
117
- explicitArray: false,
118
- });
132
+ const result = parser.parse(textXml);
119
133
  const job = result.root.job;
120
134
  if (job.result !== 'OK') {
121
- throw new errors_1.SDCardActionError(action, await (0, common_1.responseStringify)(res));
135
+ throw new errors_1.SDCardActionError(action, await (0, utils_1.responseStringify)(res));
122
136
  }
123
137
  return Number(job.jobid);
124
138
  }
125
- async fetchSDCardJobProgress(jobId, proxy = null) {
126
- const res = await this.getUrlEncoded(proxy, '/disks/job.cgi', {
139
+ async fetchSDCardJobProgress(jobId, options) {
140
+ const res = await this.postUrlEncoded('/disks/job.cgi', {
127
141
  jobid: String(jobId),
128
142
  diskid: 'SD_DISK',
143
+ }, undefined, options);
144
+ const textXml = await res.text();
145
+ const parser = new fast_xml_parser_1.XMLParser({
146
+ ignoreAttributes: false,
147
+ attributeNamePrefix: '',
148
+ allowBooleanAttributes: true,
129
149
  });
130
- const result = await (0, xml2js_1.parseStringPromise)(await res.text(), {
131
- ignoreAttrs: false,
132
- mergeAttrs: true,
133
- explicitArray: false,
134
- });
135
- const job = result.root.job;
150
+ const job = parser.parse(textXml).root.job;
136
151
  if (job.result !== 'OK') {
137
152
  throw new errors_1.SDCardJobError();
138
153
  }
139
154
  return Number(job.progress);
140
155
  }
141
- downloadCameraReport(proxy = null) {
142
- return this.getUrlEncoded(proxy, '/axis-cgi/serverreport.cgi', { mode: 'text' });
156
+ downloadCameraReport(options) {
157
+ return this.postUrlEncoded('/axis-cgi/serverreport.cgi', { mode: 'text' }, undefined, options);
143
158
  }
144
- getSystemLog(proxy = null) {
145
- return this.getUrlEncoded(proxy, '/axis-cgi/admin/systemlog.cgi');
159
+ getSystemLog(options) {
160
+ return this.postUrlEncoded('/axis-cgi/admin/systemlog.cgi', undefined, undefined, options);
146
161
  }
147
- async getMaxFps(channel, proxy = null) {
162
+ async getMaxFps(channel, options) {
148
163
  const data = { apiVersion: '1.0', method: 'getCaptureModes' };
149
- const res = await this.postJson(proxy, '/axis-cgi/capturemode.cgi', data);
164
+ const res = await this.postJson('/axis-cgi/capturemode.cgi', data, undefined, options);
150
165
  const response = VapixAPI_1.maxFpsResponseSchema.parse(await res.json());
151
166
  const channels = response.data;
152
167
  if (channels === undefined) {
@@ -161,82 +176,97 @@ class VapixAPI {
161
176
  if (captureMode === undefined) {
162
177
  throw new errors_1.MaxFPSError('CAPTURE_MODE_NOT_FOUND');
163
178
  }
164
- if ((0, common_1.isNullish)(captureMode.maxFPS)) {
179
+ if ((0, utils_1.isNullish)(captureMode.maxFPS)) {
165
180
  throw new errors_1.MaxFPSError('FPS_NOT_SPECIFIED');
166
181
  }
167
182
  return captureMode.maxFPS;
168
183
  }
169
- async getTimezone(proxy = null) {
170
- const data = { apiVersion: '1.0', method: 'getDateTimeInfo' };
171
- const res = await this.postJson(proxy, '/axis-cgi/time.cgi', data);
172
- return (await res.json())?.timeZone ?? 'Europe/Prague';
184
+ async getTimezone(options) {
185
+ try {
186
+ const agent = this.getClient(options?.proxyParams);
187
+ const resV2 = await agent.get({ path: '/config/rest/time/v2/timeZone', timeout: options?.timeout });
188
+ if (!resV2.ok) {
189
+ throw new Error(await (0, utils_1.responseStringify)(resV2));
190
+ }
191
+ const json = await resV2.json();
192
+ const data = VapixAPI_1.timeZoneSchema.parse(json);
193
+ if (data.status === 'error') {
194
+ throw new Error(data.error.message);
195
+ }
196
+ return data.data.activeTimeZone;
197
+ }
198
+ catch (error) {
199
+ console.warn('Failed to fetch time zone data from time API v2:', error instanceof Error ? error.message : JSON.stringify(error));
200
+ console.warn('Falling back to deprecated time API v1');
201
+ }
202
+ const data = await this.getDateTimeInfo(options);
203
+ if (data.data.timeZone === undefined) {
204
+ throw new Error('Time zone not setup on the device');
205
+ }
206
+ return data.data.timeZone;
173
207
  }
174
- async getDateTimeInfo(proxy = null) {
208
+ async getDateTimeInfo(options) {
175
209
  const data = { apiVersion: '1.0', method: 'getDateTimeInfo' };
176
- const res = await this.postJson(proxy, '/axis-cgi/time.cgi', data);
210
+ const res = await this.postJson('/axis-cgi/time.cgi', data, undefined, options);
177
211
  return VapixAPI_1.dateTimeinfoSchema.parse(await res.json());
178
212
  }
179
- async getDevicesSettings(proxy = null) {
213
+ async getDevicesSettings(options) {
180
214
  const data = { apiVersion: '1.0', method: 'getDevicesSettings' };
181
- const res = await this.postJson(proxy, '/axis-cgi/audiodevicecontrol.cgi', data);
215
+ const res = await this.postJson('/axis-cgi/audiodevicecontrol.cgi', data, undefined, options);
182
216
  const result = VapixAPI_1.audioDeviceRequestSchema.parse(await res.json());
183
- return result.devices.map((device) => ({
217
+ return result.data.devices.map((device) => ({
184
218
  ...device,
185
219
  inputs: (device.inputs || []).sort((a, b) => a.id.localeCompare(b.id)),
186
220
  outputs: (device.outputs || []).sort((a, b) => a.id.localeCompare(b.id)),
187
221
  }));
188
222
  }
189
- async fetchRemoteDeviceInfo(payload, proxy = null) {
190
- const res = await this.postJson(proxy, '/axis-cgi/basicdeviceinfo.cgi', payload);
191
- const result = await (0, xml2js_1.parseStringPromise)(await res.text(), {
192
- ignoreAttrs: false,
193
- mergeAttrs: true,
194
- explicitArray: false,
195
- });
196
- if ((0, common_1.isNullish)(result.body.data)) {
223
+ async fetchRemoteDeviceInfo(payload, options) {
224
+ const res = await this.postJson('/axis-cgi/basicdeviceinfo.cgi', payload, undefined, options);
225
+ const json = await res.json();
226
+ if ((0, utils_1.isNullish)(json.data)) {
197
227
  throw new errors_1.NoDeviceInfoError();
198
228
  }
199
- return result.data;
229
+ return json.data;
200
230
  }
201
- async getHeaders(proxy = null) {
231
+ async getHeaders(options) {
202
232
  const data = { apiVersion: '1.0', method: 'list' };
203
- const res = await this.postJson(proxy, '/axis-cgi/customhttpheader.cgi', data);
233
+ const res = await this.postJson('/axis-cgi/customhttpheader.cgi', data, undefined, options);
204
234
  return zod_1.z.object({ data: zod_1.z.record(zod_1.z.string()) }).parse(await res.json()).data;
205
235
  }
206
- async setHeaders(headers, proxy = null) {
236
+ async setHeaders(headers, options) {
207
237
  const data = { apiVersion: '1.0', method: 'set', params: headers };
208
- return this.postJson(proxy, '/axis-cgi/customhttpheader.cgi', data);
238
+ return this.postJson('/axis-cgi/customhttpheader.cgi', data, undefined, options);
209
239
  }
210
- async getParameter(paramNames, proxy = null) {
211
- const response = await this.getUrlEncoded(proxy, '/axis-cgi/param.cgi', {
240
+ async getParameter(paramNames, options) {
241
+ const response = await this.postUrlEncoded('/axis-cgi/param.cgi', {
212
242
  action: 'list',
213
243
  group: (0, utils_1.arrayToUrl)(paramNames),
214
- });
215
- return parseParameters(await response.text());
244
+ }, undefined, options);
245
+ return VapixAPI.parseParameters(await response.text());
216
246
  }
217
- async setParameter(params, proxy = null) {
218
- const res = await this.getUrlEncoded(proxy, '/axis-cgi/param.cgi', {
247
+ async setParameter(params, options) {
248
+ const res = await this.postUrlEncoded('/axis-cgi/param.cgi', {
219
249
  ...params,
220
250
  action: 'update',
221
- });
251
+ }, undefined, options);
222
252
  const responseText = await res.text();
223
253
  if (responseText.startsWith('# Error')) {
224
254
  throw new Error(responseText);
225
255
  }
226
256
  return true;
227
257
  }
228
- async getGuardTourList(proxy = null) {
258
+ async getGuardTourList(options) {
229
259
  const gTourList = new Array();
230
- const response = await this.getParameter('GuardTour', proxy);
260
+ const response = await this.getParameter('GuardTour', options);
231
261
  for (let i = 0; i < 20; i++) {
232
262
  const gTourBaseName = 'root.GuardTour.G' + i;
233
263
  if (gTourBaseName + '.CamNbr' in response) {
234
264
  const gTour = {
235
265
  id: gTourBaseName,
236
266
  camNbr: response[gTourBaseName + '.CamNbr'],
237
- name: response[gTourBaseName + '.Name'],
267
+ name: response[gTourBaseName + '.Name'] ?? 'Guard Tour ' + (i + 1),
238
268
  randomEnabled: response[gTourBaseName + '.RandomEnabled'],
239
- running: response[gTourBaseName + '.Running'],
269
+ running: response[gTourBaseName + '.Running'] ?? 'no',
240
270
  timeBetweenSequences: response[gTourBaseName + '.TimeBetweenSequences'],
241
271
  tour: [],
242
272
  };
@@ -261,16 +291,16 @@ class VapixAPI {
261
291
  }
262
292
  return gTourList;
263
293
  }
264
- setGuardTourEnabled(guardTourID, enable, proxy = null) {
265
- const options = {};
266
- options[guardTourID + '.Running'] = enable ? 'yes' : 'no';
267
- return this.setParameter(options, proxy);
294
+ setGuardTourEnabled(guardTourID, enable, options) {
295
+ const params = {};
296
+ params[guardTourID + '.Running'] = enable ? 'yes' : 'no';
297
+ return this.setParameter(params, options);
268
298
  }
269
- async getPTZPresetList(channel, proxy = null) {
270
- const res = await this.getUrlEncoded(proxy, '/axis-cgi/com/ptz.cgi', {
299
+ async getPTZPresetList(channel, options) {
300
+ const res = await this.postUrlEncoded('/axis-cgi/com/ptz.cgi', {
271
301
  query: 'presetposcam',
272
302
  camera: channel.toString(),
273
- });
303
+ }, undefined, options);
274
304
  const text = await res.text();
275
305
  const lines = text.split(/[\r\n]/);
276
306
  const positions = [];
@@ -285,174 +315,233 @@ class VapixAPI {
285
315
  }
286
316
  return positions;
287
317
  }
288
- async listPTZ(camera, proxy = null) {
318
+ async listPTZ(camera, options) {
289
319
  const url = `/axis-cgi/com/ptz.cgi`;
290
- const response = await this.getUrlEncoded(proxy, url, {
320
+ const response = await this.postUrlEncoded(url, {
291
321
  camera,
292
322
  query: 'presetposcamdata',
293
323
  format: 'json',
294
- });
295
- return parseCameraPtzResponse(await response.text())[camera] ?? [];
324
+ }, undefined, options);
325
+ const text = await response.text();
326
+ if (text === '') {
327
+ throw new errors_1.PtzNotSupportedError();
328
+ }
329
+ return VapixAPI.parseCameraPtzResponse(text)[camera] ?? [];
296
330
  }
297
- async listPtzVideoSourceOverview(proxy = null) {
298
- const response = await this.getUrlEncoded(proxy, '/axis-cgi/com/ptz.cgi', {
331
+ async listPtzVideoSourceOverview(options) {
332
+ const response = await this.postUrlEncoded('/axis-cgi/com/ptz.cgi', {
299
333
  query: 'presetposall',
300
334
  format: 'json',
301
- });
302
- const data = parseCameraPtzResponse(await response.text());
335
+ }, undefined, options);
336
+ const text = await response.text();
337
+ if (text === '') {
338
+ throw new errors_1.PtzNotSupportedError();
339
+ }
340
+ const data = VapixAPI.parseCameraPtzResponse(text);
303
341
  const res = {};
304
- Object.keys(data).forEach((camera) => {
305
- res[Number(camera) - 1] = data[Number(camera)].map(({ data: itemData, ...d }) => d);
342
+ Object.keys(data)
343
+ .map(Number)
344
+ .forEach((camera) => {
345
+ if (data[camera] !== undefined) {
346
+ res[camera - 1] = data[camera]?.map(({ data: itemData, ...d }) => d);
347
+ }
306
348
  });
307
349
  return res;
308
350
  }
309
- goToPreset(channel, presetName, proxy = null) {
310
- return this.getUrlEncoded(proxy, '/axis-cgi/com/ptz.cgi', {
351
+ goToPreset(channel, presetName, options) {
352
+ return this.postUrlEncoded('/axis-cgi/com/ptz.cgi', {
311
353
  camera: channel.toString(),
312
354
  gotoserverpresetname: presetName,
313
- });
355
+ }, undefined, options);
314
356
  }
315
- async getPtzPosition(camera, proxy = null) {
316
- const res = await this.getUrlEncoded(proxy, '/axis-cgi/com/ptz.cgi', {
357
+ async getPtzPosition(camera, options) {
358
+ const res = await this.postUrlEncoded('/axis-cgi/com/ptz.cgi', {
317
359
  query: 'position',
318
360
  camera: camera.toString(),
319
- });
320
- const params = parseParameters(await res.text());
361
+ }, undefined, options);
362
+ const params = VapixAPI.parseParameters(await res.text());
321
363
  return {
322
364
  pan: Number(params.pan),
323
365
  tilt: Number(params.tilt),
324
366
  zoom: Number(params.zoom),
325
367
  };
326
368
  }
327
- async getInputState(port, proxy = null) {
328
- const response = await (await this.getUrlEncoded(proxy, '/axis-cgi/io/port.cgi', { checkactive: port.toString() })).text();
329
- return response.split('=')[1].indexOf('active') === 0;
330
- }
331
- async setOutputState(port, active, proxy = null) {
332
- return this.getUrlEncoded(proxy, '/axis-cgi/io/port.cgi', { action: active ? `${port}:/` : `${port}:\\` });
333
- }
334
- async getApplicationList(proxy = null) {
335
- const res = await this.getUrlEncoded(proxy, '/axis-cgi/applications/list.cgi');
369
+ async getPorts(options) {
370
+ const res = await this.postJson('/axis-cgi/io/portmanagement.cgi', {
371
+ apiVersion: '1.0',
372
+ context: '',
373
+ method: 'getPorts',
374
+ }, undefined, options);
375
+ const portResponseParsed = await VapixAPI_1.getPortsResponseSchema.parse(res.json());
376
+ return portResponseParsed.data.items;
377
+ }
378
+ async setPorts(ports, options) {
379
+ await this.postJson('/axis-cgi/io/portmanagement.cgi', {
380
+ apiVersion: '1.0',
381
+ context: '',
382
+ method: 'setPorts',
383
+ params: { ports },
384
+ }, undefined, options);
385
+ }
386
+ async setPortStateSequence(port, sequence, options) {
387
+ await this.postJson('/axis-cgi/io/portmanagement.cgi', {
388
+ apiVersion: '1.0',
389
+ context: '',
390
+ method: 'setStateSequence',
391
+ params: { port, sequence },
392
+ }, undefined, options);
393
+ }
394
+ async getApplicationList(options) {
395
+ const agent = this.getClient(options?.proxyParams);
396
+ const res = await agent.get({ path: '/axis-cgi/applications/list.cgi', timeout: options?.timeout });
336
397
  const xml = await res.text();
337
- const result = (await (0, xml2js_1.parseStringPromise)(xml));
338
- const apps = [];
339
- for (let i = 0; i < result.reply.application.length; i++) {
340
- apps.push({
341
- ...result.reply.application[i].$,
342
- appId: VapixAPI_1.APP_IDS.find((id) => id.toLowerCase() === result.reply.application[i].$.Name.toLowerCase()) ?? null,
343
- });
398
+ const parser = new fast_xml_parser_1.XMLParser({
399
+ ignoreAttributes: false,
400
+ attributeNamePrefix: '',
401
+ allowBooleanAttributes: true,
402
+ });
403
+ const result = parser.parse(xml);
404
+ let apps = result.reply.application ?? [];
405
+ if (!Array.isArray(apps)) {
406
+ apps = [apps];
344
407
  }
345
- return apps;
408
+ return apps.map((app) => {
409
+ return {
410
+ ...app,
411
+ appId: VapixAPI_1.APP_IDS.find((id) => id.toLowerCase() === app.Name.toLowerCase()) ?? null,
412
+ };
413
+ });
346
414
  }
347
- async startApplication(applicationID, proxy = null) {
348
- const res = await this.getUrlEncoded(proxy, '/axis-cgi/applications/control.cgi', {
349
- package: applicationID.toLowerCase(),
350
- action: 'start',
415
+ async startApplication(applicationID, options) {
416
+ const agent = this.getClient(options?.proxyParams);
417
+ const res = await agent.get({
418
+ path: '/axis-cgi/applications/control.cgi',
419
+ parameters: {
420
+ package: applicationID.toLowerCase(),
421
+ action: 'start',
422
+ },
423
+ timeout: options?.timeout,
351
424
  });
352
425
  const text = (await res.text()).trim().toLowerCase();
353
426
  if (text !== 'ok' && !(text.startsWith('error:') && text.substring(7) === '6')) {
354
- throw new errors_1.ApplicationAPIError('START', await (0, common_1.responseStringify)(res));
427
+ throw new errors_1.ApplicationAPIError('START', await (0, utils_1.responseStringify)(res));
355
428
  }
356
429
  }
357
- async restartApplication(applicationID, proxy = null) {
358
- const res = await this.getUrlEncoded(proxy, '/axis-cgi/applications/control.cgi', {
359
- package: applicationID.toLowerCase(),
360
- action: 'restart',
430
+ async restartApplication(applicationID, options) {
431
+ const agent = this.getClient(options?.proxyParams);
432
+ const res = await agent.get({
433
+ path: '/axis-cgi/applications/control.cgi',
434
+ parameters: {
435
+ package: applicationID.toLowerCase(),
436
+ action: 'restart',
437
+ },
438
+ timeout: options?.timeout,
361
439
  });
362
440
  const text = (await res.text()).trim().toLowerCase();
363
441
  if (text !== 'ok') {
364
- throw new errors_1.ApplicationAPIError('RESTART', await (0, common_1.responseStringify)(res));
442
+ throw new errors_1.ApplicationAPIError('RESTART', await (0, utils_1.responseStringify)(res));
365
443
  }
366
444
  }
367
- async stopApplication(applicationID, proxy = null) {
368
- const res = await this.getUrlEncoded(proxy, '/axis-cgi/applications/control.cgi', {
369
- package: applicationID.toLowerCase(),
370
- action: 'stop',
445
+ async stopApplication(applicationID, options) {
446
+ const agent = this.getClient(options?.proxyParams);
447
+ const res = await agent.get({
448
+ path: '/axis-cgi/applications/control.cgi',
449
+ parameters: {
450
+ package: applicationID.toLowerCase(),
451
+ action: 'stop',
452
+ },
453
+ timeout: options?.timeout,
371
454
  });
372
455
  const text = (await res.text()).trim().toLowerCase();
373
456
  if (text !== 'ok' && !(text.startsWith('error:') && text.substring(7) === '6')) {
374
- throw new errors_1.ApplicationAPIError('STOP', await (0, common_1.responseStringify)(res));
457
+ throw new errors_1.ApplicationAPIError('STOP', await (0, utils_1.responseStringify)(res));
375
458
  }
376
459
  }
377
- async installApplication(data, fileName) {
460
+ async installApplication(data, fileName, options) {
378
461
  const formData = new FormData();
379
462
  formData.append('packfil', data, fileName);
380
- const res = await this.client.post(null, '/axis-cgi/applications/upload.cgi', formData, {}, {
381
- contentType: 'application/octet-stream',
463
+ const agent = this.getClient(options?.proxyParams);
464
+ const res = await agent.post({
465
+ path: '/axis-cgi/applications/upload.cgi',
466
+ data: formData,
467
+ headers: {
468
+ contentType: 'application/octet-stream',
469
+ },
470
+ timeout: options?.timeout ?? 120000,
382
471
  });
383
472
  if (!res.ok) {
384
- throw new Error(await (0, common_1.responseStringify)(res));
473
+ throw new Error(await (0, utils_1.responseStringify)(res));
385
474
  }
386
475
  const text = await res.text();
387
476
  if (text.length > 5) {
388
477
  throw new Error('installing error: ' + text);
389
478
  }
390
479
  }
391
- }
392
- exports.VapixAPI = VapixAPI;
393
- const parseParameters = (response) => {
394
- const params = {};
395
- const lines = response.split(/[\r\n]/);
396
- for (let i = 0; i < lines.length; i++) {
397
- if (lines[i].length === 0 || lines[i].substring(0, 7) === '# Error') {
398
- continue;
399
- }
400
- const delimiterPos = lines[i].indexOf('=');
401
- if (delimiterPos !== -1) {
402
- const paramName = lines[i].substring(0, delimiterPos);
403
- const paramValue = lines[i].substring(delimiterPos + 1);
404
- params[paramName] = paramValue;
405
- }
406
- }
407
- return params;
408
- };
409
- const parseCameraPtzResponse = (response) => {
410
- const json = JSON.parse(response);
411
- const parsed = {};
412
- Object.keys(json).forEach((key) => {
413
- if (!key.startsWith('Camera ')) {
414
- return;
415
- }
416
- const camera = Number(key.replace('Camera ', ''));
417
- if (json[key].presets !== undefined) {
418
- parsed[camera] = parsePtz(json[key].presets);
419
- }
420
- });
421
- return parsed;
422
- };
423
- const parsePtz = (parsed) => {
424
- const res = [];
425
- parsed.forEach((value) => {
426
- const delimiterPos = value.indexOf('=');
427
- if (delimiterPos === -1) {
428
- return;
429
- }
430
- if (!value.startsWith('presetposno')) {
431
- return;
432
- }
433
- const id = Number(value.substring(11, delimiterPos));
434
- if (Number.isNaN(id)) {
435
- return;
480
+ static parseParameters = (response) => {
481
+ const params = {};
482
+ const lines = response.split(/[\r\n]/);
483
+ for (const line of lines) {
484
+ if (line.length === 0 || line.substring(0, 7) === '# Error') {
485
+ continue;
486
+ }
487
+ const delimiterPos = line.indexOf('=');
488
+ if (delimiterPos !== -1) {
489
+ const paramName = line.substring(0, delimiterPos).replace('root.', '');
490
+ const paramValue = line.substring(delimiterPos + 1);
491
+ params[paramName] = paramValue;
492
+ }
436
493
  }
437
- const data = value.substring(delimiterPos + 1).split(':');
438
- const getValue = (valueName) => {
439
- for (const d of data) {
440
- const p = d.split('=');
441
- if (p[0] === valueName) {
442
- return Number(p[1]);
443
- }
494
+ return params;
495
+ };
496
+ static parseCameraPtzResponse = (response) => {
497
+ const json = JSON.parse(response);
498
+ const parsed = {};
499
+ Object.keys(json).forEach((key) => {
500
+ if (!key.startsWith('Camera ')) {
501
+ return;
444
502
  }
445
- return 0;
446
- };
447
- res.push({
448
- id,
449
- name: data[0],
450
- data: {
451
- pan: getValue('pan'),
452
- tilt: getValue('tilt'),
453
- zoom: getValue('zoom'),
454
- },
503
+ const camera = Number(key.replace('Camera ', ''));
504
+ if (json[key].presets !== undefined) {
505
+ parsed[camera] = VapixAPI.parsePtz(json[key].presets);
506
+ }
507
+ });
508
+ return parsed;
509
+ };
510
+ static parsePtz = (parsed) => {
511
+ const res = [];
512
+ parsed.forEach((value) => {
513
+ const delimiterPos = value.indexOf('=');
514
+ if (delimiterPos === -1) {
515
+ return;
516
+ }
517
+ if (!value.startsWith('presetposno')) {
518
+ return;
519
+ }
520
+ const id = Number(value.substring(11, delimiterPos));
521
+ if (Number.isNaN(id)) {
522
+ return;
523
+ }
524
+ const data = value.substring(delimiterPos + 1).split(':');
525
+ const getValue = (valueName) => {
526
+ for (const d of data) {
527
+ const p = d.split('=');
528
+ if (p[0] === valueName) {
529
+ return Number(p[1]);
530
+ }
531
+ }
532
+ return 0;
533
+ };
534
+ res.push({
535
+ id,
536
+ name: data[0] ?? 'Preset ' + id,
537
+ data: {
538
+ pan: getValue('pan'),
539
+ tilt: getValue('tilt'),
540
+ zoom: getValue('zoom'),
541
+ },
542
+ });
455
543
  });
456
- });
457
- return res;
458
- };
544
+ return res;
545
+ };
546
+ }
547
+ exports.VapixAPI = VapixAPI;