serverless-spy 2.3.9 → 2.3.11

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 (226) hide show
  1. package/.jsii +2 -2
  2. package/dist/releasetag.txt +1 -1
  3. package/lib/_virtual/rolldown_runtime.js +25 -0
  4. package/lib/_virtual/rolldown_runtime.mjs +11 -0
  5. package/lib/cli/cli.d.mts +1 -0
  6. package/lib/cli/cli.d.ts +1 -2
  7. package/lib/cli/cli.js +151 -208
  8. package/lib/cli/cli.js.map +1 -0
  9. package/lib/cli/cli.mjs +144 -205
  10. package/lib/cli/cli.mjs.map +1 -0
  11. package/lib/cli/sampleData.d.mts +896 -0
  12. package/lib/cli/sampleData.d.ts +860 -856
  13. package/lib/cli/sampleData.js +496 -480
  14. package/lib/cli/sampleData.js.map +1 -0
  15. package/lib/cli/sampleData.mjs +495 -477
  16. package/lib/cli/sampleData.mjs.map +1 -0
  17. package/lib/common/SpyEventSender.d.mts +26 -0
  18. package/lib/common/SpyEventSender.d.ts +23 -18
  19. package/lib/common/SpyEventSender.js +180 -230
  20. package/lib/common/SpyEventSender.js.map +1 -0
  21. package/lib/common/SpyEventSender.mjs +180 -227
  22. package/lib/common/SpyEventSender.mjs.map +1 -0
  23. package/lib/common/getWebSocketUrl.d.mts +7 -0
  24. package/lib/common/getWebSocketUrl.d.ts +7 -2
  25. package/lib/common/getWebSocketUrl.js +51 -62
  26. package/lib/common/getWebSocketUrl.js.map +1 -0
  27. package/lib/common/getWebSocketUrl.mjs +49 -60
  28. package/lib/common/getWebSocketUrl.mjs.map +1 -0
  29. package/lib/common/spyEvents/DynamoDBSpyEvent.d.mts +15 -0
  30. package/lib/common/spyEvents/DynamoDBSpyEvent.d.ts +14 -9
  31. package/lib/common/spyEvents/DynamoDBSpyEvent.js +0 -3
  32. package/lib/common/spyEvents/DynamoDBSpyEvent.mjs +1 -2
  33. package/lib/common/spyEvents/EventBridgeBaseSpyEvent.d.mts +14 -0
  34. package/lib/common/spyEvents/EventBridgeBaseSpyEvent.d.ts +13 -8
  35. package/lib/common/spyEvents/EventBridgeBaseSpyEvent.js +0 -3
  36. package/lib/common/spyEvents/EventBridgeBaseSpyEvent.mjs +1 -2
  37. package/lib/common/spyEvents/EventBridgeRuleSpyEvent.d.mts +9 -0
  38. package/lib/common/spyEvents/EventBridgeRuleSpyEvent.d.ts +8 -3
  39. package/lib/common/spyEvents/EventBridgeRuleSpyEvent.js +0 -3
  40. package/lib/common/spyEvents/EventBridgeRuleSpyEvent.mjs +1 -2
  41. package/lib/common/spyEvents/EventBridgeSpyEvent.d.mts +9 -0
  42. package/lib/common/spyEvents/EventBridgeSpyEvent.d.ts +8 -3
  43. package/lib/common/spyEvents/EventBridgeSpyEvent.js +0 -3
  44. package/lib/common/spyEvents/EventBridgeSpyEvent.mjs +1 -2
  45. package/lib/common/spyEvents/FunctionBaseSpyEvent.d.mts +11 -0
  46. package/lib/common/spyEvents/FunctionBaseSpyEvent.d.ts +10 -5
  47. package/lib/common/spyEvents/FunctionBaseSpyEvent.js +0 -3
  48. package/lib/common/spyEvents/FunctionBaseSpyEvent.mjs +1 -2
  49. package/lib/common/spyEvents/FunctionConsole.d.mts +10 -0
  50. package/lib/common/spyEvents/FunctionConsole.d.ts +9 -5
  51. package/lib/common/spyEvents/FunctionConsole.js +0 -3
  52. package/lib/common/spyEvents/FunctionConsole.mjs +1 -2
  53. package/lib/common/spyEvents/FunctionConsoleSpyEvent.d.mts +11 -0
  54. package/lib/common/spyEvents/FunctionConsoleSpyEvent.d.ts +10 -5
  55. package/lib/common/spyEvents/FunctionConsoleSpyEvent.js +0 -3
  56. package/lib/common/spyEvents/FunctionConsoleSpyEvent.mjs +1 -2
  57. package/lib/common/spyEvents/FunctionContext.d.mts +12 -0
  58. package/lib/common/spyEvents/FunctionContext.d.ts +11 -6
  59. package/lib/common/spyEvents/FunctionContext.js +0 -3
  60. package/lib/common/spyEvents/FunctionContext.mjs +1 -2
  61. package/lib/common/spyEvents/FunctionErrorSpyEvent.d.mts +13 -0
  62. package/lib/common/spyEvents/FunctionErrorSpyEvent.d.ts +12 -7
  63. package/lib/common/spyEvents/FunctionErrorSpyEvent.js +0 -3
  64. package/lib/common/spyEvents/FunctionErrorSpyEvent.mjs +1 -2
  65. package/lib/common/spyEvents/FunctionRequestSpyEvent.d.mts +12 -0
  66. package/lib/common/spyEvents/FunctionRequestSpyEvent.d.ts +11 -6
  67. package/lib/common/spyEvents/FunctionRequestSpyEvent.js +0 -3
  68. package/lib/common/spyEvents/FunctionRequestSpyEvent.mjs +1 -2
  69. package/lib/common/spyEvents/FunctionResponseSpyEvent.d.mts +10 -0
  70. package/lib/common/spyEvents/FunctionResponseSpyEvent.d.ts +9 -4
  71. package/lib/common/spyEvents/FunctionResponseSpyEvent.js +0 -3
  72. package/lib/common/spyEvents/FunctionResponseSpyEvent.mjs +1 -2
  73. package/lib/common/spyEvents/S3SpyEvent.d.mts +13 -0
  74. package/lib/common/spyEvents/S3SpyEvent.d.ts +12 -7
  75. package/lib/common/spyEvents/S3SpyEvent.js +0 -3
  76. package/lib/common/spyEvents/S3SpyEvent.mjs +1 -2
  77. package/lib/common/spyEvents/SnsSpyEventBase.d.mts +15 -0
  78. package/lib/common/spyEvents/SnsSpyEventBase.d.ts +14 -9
  79. package/lib/common/spyEvents/SnsSpyEventBase.js +0 -3
  80. package/lib/common/spyEvents/SnsSpyEventBase.mjs +1 -2
  81. package/lib/common/spyEvents/SnsSubscriptionSpyEvent.d.mts +9 -0
  82. package/lib/common/spyEvents/SnsSubscriptionSpyEvent.d.ts +8 -3
  83. package/lib/common/spyEvents/SnsSubscriptionSpyEvent.js +0 -3
  84. package/lib/common/spyEvents/SnsSubscriptionSpyEvent.mjs +1 -2
  85. package/lib/common/spyEvents/SnsTopicSpyEvent.d.mts +9 -0
  86. package/lib/common/spyEvents/SnsTopicSpyEvent.d.ts +8 -3
  87. package/lib/common/spyEvents/SnsTopicSpyEvent.js +0 -3
  88. package/lib/common/spyEvents/SnsTopicSpyEvent.mjs +1 -2
  89. package/lib/common/spyEvents/SpyEvent.d.mts +7 -0
  90. package/lib/common/spyEvents/SpyEvent.d.ts +6 -2
  91. package/lib/common/spyEvents/SpyEvent.js +0 -3
  92. package/lib/common/spyEvents/SpyEvent.mjs +1 -2
  93. package/lib/common/spyEvents/SpyMessage.d.mts +11 -0
  94. package/lib/common/spyEvents/SpyMessage.d.ts +10 -5
  95. package/lib/common/spyEvents/SpyMessage.js +0 -3
  96. package/lib/common/spyEvents/SpyMessage.mjs +1 -2
  97. package/lib/common/spyEvents/SqsSpyEvent.d.mts +12 -0
  98. package/lib/common/spyEvents/SqsSpyEvent.d.ts +11 -6
  99. package/lib/common/spyEvents/SqsSpyEvent.js +0 -3
  100. package/lib/common/spyEvents/SqsSpyEvent.mjs +1 -2
  101. package/lib/extension/dist/layer/nodejs/node_modules/interceptor.js +2 -2
  102. package/lib/extension/dist/layer/nodejs/node_modules/interceptor.js.map +2 -2
  103. package/lib/functions/onConnect.d.mts +1 -0
  104. package/lib/functions/onConnect.d.ts +1 -1
  105. package/lib/functions/onConnect.js +28 -26
  106. package/lib/functions/onConnect.js.map +1 -0
  107. package/lib/functions/onConnect.mjs +35 -26
  108. package/lib/functions/onConnect.mjs.map +1 -0
  109. package/lib/functions/onDisconnect.d.mts +10 -0
  110. package/lib/functions/onDisconnect.d.ts +9 -4
  111. package/lib/functions/onDisconnect.js +28 -27
  112. package/lib/functions/onDisconnect.js.map +1 -0
  113. package/lib/functions/onDisconnect.mjs +29 -25
  114. package/lib/functions/onDisconnect.mjs.map +1 -0
  115. package/lib/functions/sendMessage.d.mts +8 -0
  116. package/lib/functions/sendMessage.d.ts +7 -3
  117. package/lib/functions/sendMessage.js +26 -21
  118. package/lib/functions/sendMessage.js.map +1 -0
  119. package/lib/functions/sendMessage.mjs +28 -19
  120. package/lib/functions/sendMessage.mjs.map +1 -0
  121. package/lib/functions/sqsSubscriptionAndDropAllMessages.d.mts +5 -0
  122. package/lib/functions/sqsSubscriptionAndDropAllMessages.d.ts +5 -1
  123. package/lib/functions/sqsSubscriptionAndDropAllMessages.js +6 -5
  124. package/lib/functions/sqsSubscriptionAndDropAllMessages.js.map +1 -0
  125. package/lib/functions/sqsSubscriptionAndDropAllMessages.mjs +7 -3
  126. package/lib/functions/sqsSubscriptionAndDropAllMessages.mjs.map +1 -0
  127. package/lib/index.d.mts +4 -0
  128. package/lib/index.d.ts +4 -2
  129. package/lib/index.js +6 -19
  130. package/lib/index.mjs +5 -3
  131. package/lib/listener/PrettifyForDisplay.d.mts +5 -0
  132. package/lib/listener/PrettifyForDisplay.d.ts +5 -3
  133. package/lib/listener/PrettifyForDisplay.js +0 -3
  134. package/lib/listener/PrettifyForDisplay.mjs +1 -2
  135. package/lib/listener/RecursivePartial.d.mts +7 -0
  136. package/lib/listener/RecursivePartial.d.ts +7 -4
  137. package/lib/listener/RecursivePartial.js +0 -3
  138. package/lib/listener/RecursivePartial.mjs +1 -2
  139. package/lib/listener/ServerlessSpyListener.d.mts +23 -0
  140. package/lib/listener/ServerlessSpyListener.d.ts +22 -42
  141. package/lib/listener/ServerlessSpyListener.js +0 -3
  142. package/lib/listener/ServerlessSpyListener.mjs +1 -2
  143. package/lib/listener/ServerlessSpyListenerParams.d.mts +14 -0
  144. package/lib/listener/ServerlessSpyListenerParams.d.ts +13 -8
  145. package/lib/listener/ServerlessSpyListenerParams.js +0 -3
  146. package/lib/listener/ServerlessSpyListenerParams.mjs +1 -2
  147. package/lib/listener/SpyHandlers.ts.d.mts +158 -0
  148. package/lib/listener/SpyHandlers.ts.d.ts +144 -144
  149. package/lib/listener/SpyHandlers.ts.js +0 -3
  150. package/lib/listener/SpyHandlers.ts.mjs +1 -2
  151. package/lib/listener/WaitForParams.d.mts +10 -0
  152. package/lib/listener/WaitForParams.d.ts +9 -4
  153. package/lib/listener/WaitForParams.js +0 -3
  154. package/lib/listener/WaitForParams.mjs +1 -2
  155. package/lib/listener/WsListener.d.mts +27 -0
  156. package/lib/listener/WsListener.d.ts +26 -21
  157. package/lib/listener/WsListener.js +173 -231
  158. package/lib/listener/WsListener.js.map +1 -0
  159. package/lib/listener/WsListener.mjs +174 -228
  160. package/lib/listener/WsListener.mjs.map +1 -0
  161. package/lib/listener/createServerlessSpyListener.d.mts +8 -0
  162. package/lib/listener/createServerlessSpyListener.d.ts +8 -2
  163. package/lib/listener/createServerlessSpyListener.js +25 -25
  164. package/lib/listener/createServerlessSpyListener.js.map +1 -0
  165. package/lib/listener/createServerlessSpyListener.mjs +26 -23
  166. package/lib/listener/createServerlessSpyListener.mjs.map +1 -0
  167. package/lib/listener/index.d.mts +3 -0
  168. package/lib/listener/index.d.ts +3 -2
  169. package/lib/listener/index.js +3 -19
  170. package/lib/listener/index.mjs +3 -3
  171. package/lib/listener/iot-connection.d.mts +13 -0
  172. package/lib/listener/iot-connection.d.ts +12 -7
  173. package/lib/listener/iot-connection.js +48 -46
  174. package/lib/listener/iot-connection.js.map +1 -0
  175. package/lib/listener/iot-connection.mjs +46 -44
  176. package/lib/listener/iot-connection.mjs.map +1 -0
  177. package/lib/listener/matchers.d.mts +1 -0
  178. package/lib/listener/matchers.d.ts +1 -0
  179. package/lib/listener/matchers.js +0 -55
  180. package/lib/listener/matchers.mjs +1 -55
  181. package/lib/listener/setup.d.mts +1 -0
  182. package/lib/listener/setup.d.ts +1 -0
  183. package/lib/listener/setup.js +0 -21
  184. package/lib/listener/setup.mjs +1 -21
  185. package/lib/listener/topic.d.mts +6 -0
  186. package/lib/listener/topic.d.ts +6 -2
  187. package/lib/listener/topic.js +9 -7
  188. package/lib/listener/topic.js.map +1 -0
  189. package/lib/listener/topic.mjs +8 -4
  190. package/lib/listener/topic.mjs.map +1 -0
  191. package/lib/node_modules/tsdown/esm-shims.mjs +11 -0
  192. package/lib/node_modules/tsdown/esm-shims.mjs.map +1 -0
  193. package/lib/node_modules/uuid/dist/esm-node/native.js +10 -0
  194. package/lib/node_modules/uuid/dist/esm-node/native.js.map +1 -0
  195. package/lib/node_modules/uuid/dist/esm-node/native.mjs +8 -0
  196. package/lib/node_modules/uuid/dist/esm-node/native.mjs.map +1 -0
  197. package/lib/node_modules/uuid/dist/esm-node/rng.js +18 -0
  198. package/lib/node_modules/uuid/dist/esm-node/rng.js.map +1 -0
  199. package/lib/node_modules/uuid/dist/esm-node/rng.mjs +16 -0
  200. package/lib/node_modules/uuid/dist/esm-node/rng.mjs.map +1 -0
  201. package/lib/node_modules/uuid/dist/esm-node/stringify.js +15 -0
  202. package/lib/node_modules/uuid/dist/esm-node/stringify.js.map +1 -0
  203. package/lib/node_modules/uuid/dist/esm-node/stringify.mjs +14 -0
  204. package/lib/node_modules/uuid/dist/esm-node/stringify.mjs.map +1 -0
  205. package/lib/node_modules/uuid/dist/esm-node/v4.js +23 -0
  206. package/lib/node_modules/uuid/dist/esm-node/v4.js.map +1 -0
  207. package/lib/node_modules/uuid/dist/esm-node/v4.mjs +23 -0
  208. package/lib/node_modules/uuid/dist/esm-node/v4.mjs.map +1 -0
  209. package/lib/src/ServerlessSpy.d.mts +77 -0
  210. package/lib/src/ServerlessSpy.js +1 -1
  211. package/lib/src/ServerlessSpy.js.map +1 -0
  212. package/lib/src/ServerlessSpy.mjs +441 -618
  213. package/lib/src/ServerlessSpy.mjs.map +1 -0
  214. package/lib/src/common/envVariableNames.d.mts +35 -0
  215. package/lib/src/common/envVariableNames.js.map +1 -0
  216. package/lib/src/common/envVariableNames.mjs +43 -35
  217. package/lib/src/common/envVariableNames.mjs.map +1 -0
  218. package/lib/src/index.d.mts +2 -0
  219. package/lib/src/index.mjs +3 -2
  220. package/node_modules/debug/package.json +2 -3
  221. package/node_modules/debug/src/browser.js +1 -1
  222. package/node_modules/debug/src/common.js +1 -1
  223. package/package.json +3 -1
  224. package/tsdown.config.ts +16 -0
  225. package/node_modules/micromist/yarn.lock +0 -75
  226. package/node_modules/tmp/lib/tmp.js.orig +0 -471
@@ -0,0 +1 @@
1
+ {"version":3,"file":"SpyEventSender.js","names":["envVariableNames","getConnection","postDataPromises: Promise<any>[]","serviceKey: string","message: string","fluentEvent: Omit<SpyMessage, 'timestamp'>","body: string","v4","getTopic"],"sources":["../../common/SpyEventSender.ts"],"sourcesContent":["import { unmarshall } from '@aws-sdk/util-dynamodb';\nimport iot from 'aws-iot-device-sdk';\nimport {\n DynamoDBStreamEvent,\n S3Event,\n SNSEvent,\n EventBridgeEvent,\n SQSEvent,\n} from 'aws-lambda';\nimport { v4 } from 'uuid';\nimport { DynamoDBSpyEvent } from './spyEvents/DynamoDBSpyEvent';\nimport { EventBridgeRuleSpyEvent } from './spyEvents/EventBridgeRuleSpyEvent';\nimport { EventBridgeSpyEvent } from './spyEvents/EventBridgeSpyEvent';\nimport { S3SpyEvent } from './spyEvents/S3SpyEvent';\nimport { SnsSubscriptionSpyEvent } from './spyEvents/SnsSubscriptionSpyEvent';\nimport { SnsTopicSpyEvent } from './spyEvents/SnsTopicSpyEvent';\nimport { SpyMessage } from './spyEvents/SpyMessage';\nimport { SqsSpyEvent } from './spyEvents/SqsSpyEvent';\nimport { fragment, getConnection } from '../listener/iot-connection';\nimport { getTopic } from '../listener/topic';\nimport { envVariableNames } from '../src/common/envVariableNames';\n\nexport class SpyEventSender {\n debugMode = process.env[envVariableNames.SSPY_DEBUG] === 'true';\n connection: iot.device | undefined;\n scope: string;\n iotEndpoint: string;\n\n constructor(params: {\n log?: (message: string, ...optionalParams: any[]) => void;\n logError?: (message: string, ...optionalParams: any[]) => void;\n scope: string;\n iotEndpoint: string;\n }) {\n if (params.log) {\n this.log = params.log;\n }\n\n if (params.logError) {\n this.logError = params.logError;\n }\n\n this.scope = params.scope;\n this.iotEndpoint = params.iotEndpoint;\n }\n\n public async close() {\n this.connection?.end();\n }\n\n public async connect() {\n this.connection = await getConnection(this.debugMode, this.iotEndpoint);\n }\n\n public async publishSpyEvent(event: any) {\n this.log('Event', JSON.stringify(event));\n\n const mapping = JSON.parse(\n process.env[envVariableNames.SSPY_INFRA_MAPPING]!\n );\n this.log('ARN to names mapping', JSON.stringify(mapping));\n\n const postDataPromises: Promise<any>[] = [];\n\n if (event?.Records && event.Records[0]?.Sns) {\n //console.log('*** SNS ***');\n const eventSns = event as SNSEvent;\n for (const record of eventSns.Records) {\n const subscriptionArn = record.EventSubscriptionArn;\n\n let serviceKey: string;\n if (mapping[subscriptionArn]) {\n // subscription event that could contain filter based on existing subscription\n serviceKey = mapping[subscriptionArn];\n } else {\n // catch all subscription\n const topicArn = record.Sns.TopicArn;\n serviceKey = mapping[topicArn];\n }\n\n let message: string;\n\n try {\n message = JSON.parse(record.Sns.Message);\n } catch {\n message = record.Sns.Message;\n }\n\n const spyEventType = this.getSpyEventType(serviceKey) as\n | 'FunctionSnsTopic'\n | 'FunctionSnsSubscription';\n\n const data: SnsTopicSpyEvent | SnsSubscriptionSpyEvent = {\n spyEventType,\n message,\n subject: record.Sns.Subject,\n timestamp: record.Sns.Timestamp,\n topicArn: record.Sns.TopicArn,\n messageId: record.Sns.MessageId,\n messageAttributes: record.Sns.MessageAttributes,\n };\n\n const fluentEvent: Omit<SpyMessage, 'timestamp'> = {\n data,\n serviceKey,\n };\n postDataPromises.push(this.postData(fluentEvent));\n }\n } else if (event?.Records && event.Records[0]?.eventSource === 'aws:sqs') {\n //console.log('*** SQS ***');\n const eventSqs = event as SQSEvent;\n for (const record of eventSqs.Records) {\n const subscriptionArn = record.eventSourceARN;\n\n const serviceKey = mapping[subscriptionArn];\n let body: string;\n\n try {\n body = JSON.parse(record.body);\n } catch {\n body = record.body;\n }\n\n const data: SqsSpyEvent = {\n spyEventType: 'Sqs',\n body,\n messageAttributes: record.messageAttributes,\n };\n\n const fluentEvent: Omit<SpyMessage, 'timestamp'> = {\n data,\n serviceKey,\n };\n postDataPromises.push(this.postData(fluentEvent));\n }\n } else if (event?.Records && event.Records[0]?.s3) {\n //console.log('*** S3 ***');\n const eventS3 = event as S3Event;\n for (const record of eventS3.Records) {\n const bucketArn = record.s3.bucket.arn;\n\n const data: S3SpyEvent = {\n spyEventType: 'S3',\n eventName: record.eventName,\n eventTime: record.eventTime,\n bucket: record.s3.bucket.name,\n key: record.s3.object.key,\n };\n\n const fluentEvent: Omit<SpyMessage, 'timestamp'> = {\n data,\n serviceKey: mapping[bucketArn],\n };\n postDataPromises.push(this.postData(fluentEvent));\n }\n } else if (event.Records && event.Records[0]?.dynamodb) {\n //console.log('*** DYNAMODB ***');\n const eventDynamoDB = event as DynamoDBStreamEvent;\n for (const record of eventDynamoDB.Records) {\n let arn = record.eventSourceARN!;\n arn = arn.substring(0, arn.indexOf('/stream/'));\n\n const data: DynamoDBSpyEvent = {\n spyEventType: 'DynamoDB',\n eventName: record.eventName,\n newImage: record.dynamodb?.NewImage\n ? unmarshall(record.dynamodb?.NewImage as any)\n : undefined,\n keys: unmarshall(record.dynamodb?.Keys as any),\n oldImage: record.dynamodb?.OldImage\n ? unmarshall(record.dynamodb?.OldImage as any)\n : undefined,\n };\n\n const fluentEvent: Omit<SpyMessage, 'timestamp'> = {\n data,\n serviceKey: mapping[arn],\n };\n postDataPromises.push(this.postData(fluentEvent));\n }\n } else if (\n event.detail &&\n event['detail-type'] &&\n event.version &&\n event.source\n ) {\n //console.log('*** EventBridge ***');\n const eventEb = event as EventBridgeEvent<any, any>;\n\n const serviceKey = mapping.eventBridge; // the is new lambda for each subscription\n\n const spyEventType = this.getSpyEventType(serviceKey) as\n | 'EventBridge'\n | 'EventBridgeRule';\n\n const message = eventEb.detail;\n\n const data: EventBridgeSpyEvent | EventBridgeRuleSpyEvent = {\n spyEventType,\n detail: message,\n detailType: eventEb['detail-type'],\n eventBridgeId: eventEb['id'],\n source: eventEb.source,\n time: eventEb.time,\n account: eventEb.account,\n };\n\n const fluentEvent: Omit<SpyMessage, 'timestamp'> = {\n data,\n serviceKey,\n };\n postDataPromises.push(this.postData(fluentEvent));\n } else {\n //console.log('*** OTHER ***');\n const fluentEvent: Omit<SpyMessage, 'timestamp'> = event;\n postDataPromises.push(this.postData(fluentEvent));\n }\n\n await Promise.all(postDataPromises);\n }\n\n private encode(input: any): fragment[] {\n const payload = JSON.stringify(input);\n const parts = payload.match(/.{1,50000}/g);\n if (!parts) return [];\n this.log(`Encoded iot message, ${parts.length}`);\n const id = v4();\n return parts.map((part, index) => ({\n id,\n index,\n count: parts.length,\n data: part,\n }));\n }\n\n private async postData(\n spyMessage: Omit<SpyMessage, 'timestamp'> & { timestamp?: string }\n ) {\n if (this.connection === undefined) {\n throw new Error(\n 'No IoT connection created yet, did you forget to call connect()?'\n );\n }\n\n const withTimeStamp = {\n ...spyMessage,\n timestamp: spyMessage.timestamp || new Date().toISOString(),\n };\n\n this.log('Post spy message', JSON.stringify(withTimeStamp));\n\n const connection = this.connection;\n const topic = getTopic(this.scope);\n\n try {\n for (const fragment of this.encode(withTimeStamp)) {\n await new Promise<void>((resolve) => {\n connection.publish(\n topic,\n JSON.stringify(fragment),\n {\n qos: 1,\n },\n () => {\n this.log('Publishing finished');\n resolve();\n }\n );\n });\n this.log(\n `Published fragment ${fragment.index} out of ${fragment.count} to topic ${topic}`\n );\n }\n } catch (e) {\n this.logError(`Failed to send payload to iot: ${e}`);\n }\n\n this.log('Send spy message finish');\n }\n\n private getSpyEventType(serviceKey: string) {\n if (!serviceKey) {\n throw new Error('Missing serviceKey');\n }\n\n return serviceKey.substring(0, serviceKey.indexOf('#'));\n }\n\n private log(message: string, ...optionalParams: any[]) {\n if (this.debugMode) {\n console.debug('SSPY EXTENSION', message, ...optionalParams);\n }\n }\n\n private logError(message: string, ...optionalParams: any[]) {\n if (this.debugMode) {\n console.error('SSPY EXTENSION', message, ...optionalParams);\n }\n }\n}\n"],"mappings":"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"}
@@ -1,227 +1,180 @@
1
- import { unmarshall } from '@aws-sdk/util-dynamodb';
2
- import { v4 } from 'uuid';
3
- import { getConnection } from '../listener/iot-connection';
4
- import { getTopic } from '../listener/topic';
5
- import { envVariableNames } from '../src/common/envVariableNames';
6
- export class SpyEventSender {
7
- constructor(params) {
8
- this.debugMode = process.env[envVariableNames.SSPY_DEBUG] === 'true';
9
- if (params.log) {
10
- this.log = params.log;
11
- }
12
- if (params.logError) {
13
- this.logError = params.logError;
14
- }
15
- this.scope = params.scope;
16
- this.iotEndpoint = params.iotEndpoint;
17
- }
18
- async close() {
19
- this.connection?.end();
20
- }
21
- async connect() {
22
- this.connection = await getConnection(this.debugMode, this.iotEndpoint);
23
- }
24
- async publishSpyEvent(event) {
25
- this.log('Event', JSON.stringify(event));
26
- const mapping = JSON.parse(process.env[envVariableNames.SSPY_INFRA_MAPPING]);
27
- this.log('ARN to names mapping', JSON.stringify(mapping));
28
- const postDataPromises = [];
29
- if (event?.Records && event.Records[0]?.Sns) {
30
- //console.log('*** SNS ***');
31
- const eventSns = event;
32
- for (const record of eventSns.Records) {
33
- const subscriptionArn = record.EventSubscriptionArn;
34
- let serviceKey;
35
- if (mapping[subscriptionArn]) {
36
- // subscription event that could contain filter based on existing subscription
37
- serviceKey = mapping[subscriptionArn];
38
- }
39
- else {
40
- // catch all subscription
41
- const topicArn = record.Sns.TopicArn;
42
- serviceKey = mapping[topicArn];
43
- }
44
- let message;
45
- try {
46
- message = JSON.parse(record.Sns.Message);
47
- }
48
- catch {
49
- message = record.Sns.Message;
50
- }
51
- const spyEventType = this.getSpyEventType(serviceKey);
52
- const data = {
53
- spyEventType,
54
- message,
55
- subject: record.Sns.Subject,
56
- timestamp: record.Sns.Timestamp,
57
- topicArn: record.Sns.TopicArn,
58
- messageId: record.Sns.MessageId,
59
- messageAttributes: record.Sns.MessageAttributes,
60
- };
61
- const fluentEvent = {
62
- data,
63
- serviceKey,
64
- };
65
- postDataPromises.push(this.postData(fluentEvent));
66
- }
67
- }
68
- else if (event?.Records && event.Records[0]?.eventSource === 'aws:sqs') {
69
- //console.log('*** SQS ***');
70
- const eventSqs = event;
71
- for (const record of eventSqs.Records) {
72
- const subscriptionArn = record.eventSourceARN;
73
- const serviceKey = mapping[subscriptionArn];
74
- let body;
75
- try {
76
- body = JSON.parse(record.body);
77
- }
78
- catch {
79
- body = record.body;
80
- }
81
- const data = {
82
- spyEventType: 'Sqs',
83
- body,
84
- messageAttributes: record.messageAttributes,
85
- };
86
- const fluentEvent = {
87
- data,
88
- serviceKey,
89
- };
90
- postDataPromises.push(this.postData(fluentEvent));
91
- }
92
- }
93
- else if (event?.Records && event.Records[0]?.s3) {
94
- //console.log('*** S3 ***');
95
- const eventS3 = event;
96
- for (const record of eventS3.Records) {
97
- const bucketArn = record.s3.bucket.arn;
98
- const data = {
99
- spyEventType: 'S3',
100
- eventName: record.eventName,
101
- eventTime: record.eventTime,
102
- bucket: record.s3.bucket.name,
103
- key: record.s3.object.key,
104
- };
105
- const fluentEvent = {
106
- data,
107
- serviceKey: mapping[bucketArn],
108
- };
109
- postDataPromises.push(this.postData(fluentEvent));
110
- }
111
- }
112
- else if (event.Records && event.Records[0]?.dynamodb) {
113
- //console.log('*** DYNAMODB ***');
114
- const eventDynamoDB = event;
115
- for (const record of eventDynamoDB.Records) {
116
- let arn = record.eventSourceARN;
117
- arn = arn.substring(0, arn.indexOf('/stream/'));
118
- const data = {
119
- spyEventType: 'DynamoDB',
120
- eventName: record.eventName,
121
- newImage: record.dynamodb?.NewImage
122
- ? unmarshall(record.dynamodb?.NewImage)
123
- : undefined,
124
- keys: unmarshall(record.dynamodb?.Keys),
125
- oldImage: record.dynamodb?.OldImage
126
- ? unmarshall(record.dynamodb?.OldImage)
127
- : undefined,
128
- };
129
- const fluentEvent = {
130
- data,
131
- serviceKey: mapping[arn],
132
- };
133
- postDataPromises.push(this.postData(fluentEvent));
134
- }
135
- }
136
- else if (event.detail &&
137
- event['detail-type'] &&
138
- event.version &&
139
- event.source) {
140
- //console.log('*** EventBridge ***');
141
- const eventEb = event;
142
- const serviceKey = mapping.eventBridge; // the is new lambda for each subscription
143
- const spyEventType = this.getSpyEventType(serviceKey);
144
- const message = eventEb.detail;
145
- const data = {
146
- spyEventType,
147
- detail: message,
148
- detailType: eventEb['detail-type'],
149
- eventBridgeId: eventEb['id'],
150
- source: eventEb.source,
151
- time: eventEb.time,
152
- account: eventEb.account,
153
- };
154
- const fluentEvent = {
155
- data,
156
- serviceKey,
157
- };
158
- postDataPromises.push(this.postData(fluentEvent));
159
- }
160
- else {
161
- //console.log('*** OTHER ***');
162
- const fluentEvent = event;
163
- postDataPromises.push(this.postData(fluentEvent));
164
- }
165
- await Promise.all(postDataPromises);
166
- }
167
- encode(input) {
168
- const payload = JSON.stringify(input);
169
- const parts = payload.match(/.{1,50000}/g);
170
- if (!parts)
171
- return [];
172
- this.log(`Encoded iot message, ${parts.length}`);
173
- const id = v4();
174
- return parts.map((part, index) => ({
175
- id,
176
- index,
177
- count: parts.length,
178
- data: part,
179
- }));
180
- }
181
- async postData(spyMessage) {
182
- if (this.connection === undefined) {
183
- throw new Error('No IoT connection created yet, did you forget to call connect()?');
184
- }
185
- const withTimeStamp = {
186
- ...spyMessage,
187
- timestamp: spyMessage.timestamp || new Date().toISOString(),
188
- };
189
- this.log('Post spy message', JSON.stringify(withTimeStamp));
190
- const connection = this.connection;
191
- const topic = getTopic(this.scope);
192
- try {
193
- for (const fragment of this.encode(withTimeStamp)) {
194
- await new Promise((resolve) => {
195
- connection.publish(topic, JSON.stringify(fragment), {
196
- qos: 1,
197
- }, () => {
198
- this.log('Publishing finished');
199
- resolve();
200
- });
201
- });
202
- this.log(`Published fragment ${fragment.index} out of ${fragment.count} to topic ${topic}`);
203
- }
204
- }
205
- catch (e) {
206
- this.logError(`Failed to send payload to iot: ${e}`);
207
- }
208
- this.log('Send spy message finish');
209
- }
210
- getSpyEventType(serviceKey) {
211
- if (!serviceKey) {
212
- throw new Error('Missing serviceKey');
213
- }
214
- return serviceKey.substring(0, serviceKey.indexOf('#'));
215
- }
216
- log(message, ...optionalParams) {
217
- if (this.debugMode) {
218
- console.debug('SSPY EXTENSION', message, ...optionalParams);
219
- }
220
- }
221
- logError(message, ...optionalParams) {
222
- if (this.debugMode) {
223
- console.error('SSPY EXTENSION', message, ...optionalParams);
224
- }
225
- }
226
- }
227
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiU3B5RXZlbnRTZW5kZXIuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9jb21tb24vU3B5RXZlbnRTZW5kZXIudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFVBQVUsRUFBRSxNQUFNLHdCQUF3QixDQUFDO0FBU3BELE9BQU8sRUFBRSxFQUFFLEVBQUUsTUFBTSxNQUFNLENBQUM7QUFTMUIsT0FBTyxFQUFZLGFBQWEsRUFBRSxNQUFNLDRCQUE0QixDQUFDO0FBQ3JFLE9BQU8sRUFBRSxRQUFRLEVBQUUsTUFBTSxtQkFBbUIsQ0FBQztBQUM3QyxPQUFPLEVBQUUsZ0JBQWdCLEVBQUUsTUFBTSxnQ0FBZ0MsQ0FBQztBQUVsRSxNQUFNLE9BQU8sY0FBYztJQU16QixZQUFZLE1BS1g7UUFWRCxjQUFTLEdBQUcsT0FBTyxDQUFDLEdBQUcsQ0FBQyxnQkFBZ0IsQ0FBQyxVQUFVLENBQUMsS0FBSyxNQUFNLENBQUM7UUFXOUQsSUFBSSxNQUFNLENBQUMsR0FBRyxFQUFFLENBQUM7WUFDZixJQUFJLENBQUMsR0FBRyxHQUFHLE1BQU0sQ0FBQyxHQUFHLENBQUM7UUFDeEIsQ0FBQztRQUVELElBQUksTUFBTSxDQUFDLFFBQVEsRUFBRSxDQUFDO1lBQ3BCLElBQUksQ0FBQyxRQUFRLEdBQUcsTUFBTSxDQUFDLFFBQVEsQ0FBQztRQUNsQyxDQUFDO1FBRUQsSUFBSSxDQUFDLEtBQUssR0FBRyxNQUFNLENBQUMsS0FBSyxDQUFDO1FBQzFCLElBQUksQ0FBQyxXQUFXLEdBQUcsTUFBTSxDQUFDLFdBQVcsQ0FBQztJQUN4QyxDQUFDO0lBRU0sS0FBSyxDQUFDLEtBQUs7UUFDaEIsSUFBSSxDQUFDLFVBQVUsRUFBRSxHQUFHLEVBQUUsQ0FBQztJQUN6QixDQUFDO0lBRU0sS0FBSyxDQUFDLE9BQU87UUFDbEIsSUFBSSxDQUFDLFVBQVUsR0FBRyxNQUFNLGFBQWEsQ0FBQyxJQUFJLENBQUMsU0FBUyxFQUFFLElBQUksQ0FBQyxXQUFX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Z0JBQWdCLEVBQUUsT0FBTyxFQUFFLEdBQUcsY0FBYyxDQUFDLENBQUM7UUFDOUQsQ0FBQztJQUNILENBQUM7SUFFTyxRQUFRLENBQUMsT0FBZSxFQUFFLEdBQUcsY0FBcUI7UUFDeEQsSUFBSSxJQUFJLENBQUMsU0FBUyxFQUFFLENBQUM7WUFDbkIsT0FBTyxDQUFDLEtBQUssQ0FBQyxnQkFBZ0IsRUFBRSxPQUFPLEVBQUUsR0FBRyxjQUFjLENBQUMsQ0FBQztRQUM5RCxDQUFDO0lBQ0gsQ0FBQztDQUNGIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgdW5tYXJzaGFsbCB9IGZyb20gJ0Bhd3Mtc2RrL3V0aWwtZHluYW1vZGInO1xuaW1wb3J0IGlvdCBmcm9tICdhd3MtaW90LWRldmljZS1zZGsnO1xuaW1wb3J0IHtcbiAgRHluYW1vREJTdHJlYW1FdmVudCxcbiAgUzNFdmVudCxcbiAgU05TRXZlbnQsXG4gIEV2ZW50QnJpZGdlRXZlbnQsXG4gIFNRU0V2ZW50LFxufSBmcm9tICdhd3MtbGFtYmRhJztcbmltcG9ydCB7IHY0IH0gZnJvbSAndXVpZCc7XG5pbXBvcnQgeyBEeW5hbW9EQlNweUV2ZW50IH0gZnJvbSAnLi9zcHlFdmVudHMvRHluYW1vREJTcHlFdmVudCc7XG5pbXBvcnQgeyBFdmVudEJyaWRnZVJ1bGVTcHlFdmVudCB9IGZyb20gJy4vc3B5RXZlbnRzL0V2ZW50QnJpZGdlUnVsZVNweUV2ZW50JztcbmltcG9ydCB7IEV2ZW50QnJpZGdlU3B5RXZlbnQgfSBmcm9tICcuL3NweUV2ZW50cy9FdmVudEJyaWRnZVNweUV2ZW50JztcbmltcG9ydCB7IFMzU3B5RXZlbnQgfSBmcm9tICcuL3NweUV2ZW50cy9TM1NweUV2ZW50JztcbmltcG9ydCB7IFNuc1N1YnNjcmlwdGlvblNweUV2ZW50IH0gZnJvbSAnLi9zcHlFdmVudHMvU25zU3Vic2NyaXB0aW9uU3B5RXZlbnQnO1xuaW1wb3J0IHsgU25zVG9waWNTcHlFdmVudCB9IGZyb20gJy4vc3B5RXZlbnRzL1Nuc1RvcGljU3B5RXZlbnQnO1xuaW1wb3J0IHsgU3B5TWVzc2FnZSB9IGZyb20gJy4vc3B5RXZlbnRzL1NweU1lc3NhZ2UnO1xuaW1wb3J0IHsgU3FzU3B5RXZlbnQgfSBmcm9tICcuL3NweUV2ZW50cy9TcXNTcHlFdmVudCc7XG5pbXBvcnQgeyBmcmFnbWVudCwgZ2V0Q29ubmVjdGlvbiB9IGZyb20gJy4uL2xpc3RlbmVyL2lvdC1jb25uZWN0aW9uJztcbmltcG9ydCB7IGdldFRvcGljIH0gZnJvbSAnLi4vbGlzdGVuZXIvdG9waWMnO1xuaW1wb3J0IHsgZW52VmFyaWFibGVOYW1lcyB9IGZyb20gJy4uL3NyYy9jb21tb24vZW52VmFyaWFibGVOYW1lcyc7XG5cbmV4cG9ydCBjbGFzcyBTcHlFdmVudFNlbmRlciB7XG4gIGRlYnVnTW9kZSA9IHByb2Nlc3MuZW52W2VudlZhcmlhYmxlTmFtZXMuU1NQWV9ERUJVR10gPT09ICd0cnVlJztcbiAgY29ubmVjdGlvbjogaW90LmRldmljZSB8IHVuZGVmaW5lZDtcbiAgc2NvcGU6IHN0cmluZztcbiAgaW90RW5kcG9pbnQ6IHN0cmluZztcblxuICBjb25zdHJ1Y3RvcihwYXJhbXM6IHtcbiAgICBsb2c/OiAobWVzc2FnZTogc3RyaW5nLCAuLi5vcHRpb25hbFBhcmFtczogYW55W10pID0+IHZvaWQ7XG4gICAgbG9nRXJyb3I/OiAobWVzc2FnZTogc3RyaW5nLCAuLi5vcHRpb25hbFBhcmFtczogYW55W10pID0+IHZvaWQ7XG4gICAgc2NvcGU6IHN0cmluZztcbiAgICBpb3RFbmRwb2ludDogc3RyaW5nO1xuICB9KSB7XG4gICAgaWYgKHBhcmFtcy5sb2cpIHtcbiAgICAgIHRoaXMubG9nID0gcGFyYW1zLmxvZztcbiAgICB9XG5cbiAgICBpZiAocGFyYW1zLmxvZ0Vycm9yKSB7XG4gICAgICB0aGlzLmxvZ0Vycm9yID0gcGFyYW1zLmxvZ0Vycm9yO1xuICAgIH1cblxuICAgIHRoaXMuc2NvcGUgPSBwYXJhbXMuc2NvcGU7XG4gICAgdGhpcy5pb3RFbmRwb2ludCA9IHBhcmFtcy5pb3RFbmRwb2ludDtcbiAgfVxuXG4gIHB1YmxpYyBhc3luYyBjbG9zZSgpIHtcbiAgICB0aGlzLmNvbm5lY3Rpb24/LmVuZCgpO1xuICB9XG5cbiAgcHVibGljIGFzeW5jIGNvbm5lY3QoKSB7XG4gICAgdGhpcy5jb25uZWN0aW9uID0gYXdhaXQgZ2V0Q29ubmVjdGlvbih0aGlzLmRlYnVnTW9kZSwgdGhpcy5pb3RFbmRwb2ludCk7XG4gIH1cblxuICBwdWJsaWMgYXN5bmMgcHVibGlzaFNweUV2ZW50KGV2ZW50OiBhbnkpIHtcbiAgICB0aGlzLmxvZygnRXZlbnQnLCBKU09OLnN0cmluZ2lmeShldmVudCkpO1xuXG4gICAgY29uc3QgbWFwcGluZyA9IEpTT04ucGFyc2UoXG4gICAgICBwcm9jZXNzLmVudltlbnZWYXJpYWJsZU5hbWVzLlNTUFlfSU5GUkFfTUFQUElOR10hXG4gICAgKTtcbiAgICB0aGlzLmxvZygnQVJOIHRvIG5hbWVzIG1hcHBpbmcnLCBKU09OLnN0cmluZ2lmeShtYXBwaW5nKSk7XG5cbiAgICBjb25zdCBwb3N0RGF0YVByb21pc2VzOiBQcm9taXNlPGFueT5bXSA9IFtdO1xuXG4gICAgaWYgKGV2ZW50Py5SZWNvcmRzICYmIGV2ZW50LlJlY29yZHNbMF0/LlNucykge1xuICAgICAgLy9jb25zb2xlLmxvZygnKioqIFNOUyAqKionKTtcbiAgICAgIGNvbnN0IGV2ZW50U25zID0gZXZlbnQgYXMgU05TRXZlbnQ7XG4gICAgICBmb3IgKGNvbnN0IHJlY29yZCBvZiBldmVudFNucy5SZWNvcmRzKSB7XG4gICAgICAgIGNvbnN0IHN1YnNjcmlwdGlvbkFybiA9IHJlY29yZC5FdmVudFN1YnNjcmlwdGlvbkFybjtcblxuICAgICAgICBsZXQgc2VydmljZUtleTogc3RyaW5nO1xuICAgICAgICBpZiAobWFwcGluZ1tzdWJzY3JpcHRpb25Bcm5dKSB7XG4gICAgICAgICAgLy8gc3Vic2NyaXB0aW9uIGV2ZW50IHRoYXQgY291bGQgY29udGFpbiBmaWx0ZXIgYmFzZWQgb24gZXhpc3Rpbmcgc3Vic2NyaXB0aW9uXG4gICAgICAgICAgc2VydmljZUtleSA9IG1hcHBpbmdbc3Vic2NyaXB0aW9uQXJuXTtcbiAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAvLyBjYXRjaCBhbGwgc3Vic2NyaXB0aW9uXG4gICAgICAgICAgY29uc3QgdG9waWNBcm4gPSByZWNvcmQuU25zLlRvcGljQXJuO1xuICAgICAgICAgIHNlcnZpY2VLZXkgPSBtYXBwaW5nW3RvcGljQXJuXTtcbiAgICAgICAgfVxuXG4gICAgICAgIGxldCBtZXNzYWdlOiBzdHJpbmc7XG5cbiAgICAgICAgdHJ5IHtcbiAgICAgICAgICBtZXNzYWdlID0gSlNPTi5wYXJzZShyZWNvcmQuU25zLk1lc3NhZ2UpO1xuICAgICAgICB9IGNhdGNoIHtcbiAgICAgICAgICBtZXNzYWdlID0gcmVjb3JkLlNucy5NZXNzYWdlO1xuICAgICAgICB9XG5cbiAgICAgICAgY29uc3Qgc3B5RXZlbnRUeXBlID0gdGhpcy5nZXRTcHlFdmVudFR5cGUoc2VydmljZUtleSkgYXNcbiAgICAgICAgICB8ICdGdW5jdGlvblNuc1RvcGljJ1xuICAgICAgICAgIHwgJ0Z1bmN0aW9uU25zU3Vic2NyaXB0aW9uJztcblxuICAgICAgICBjb25zdCBkYXRhOiBTbnNUb3BpY1NweUV2ZW50IHwgU25zU3Vic2NyaXB0aW9uU3B5RXZlbnQgPSB7XG4gICAgICAgICAgc3B5RXZlbnRUeXBlLFxuICAgICAgICAgIG1lc3NhZ2UsXG4gICAgICAgICAgc3ViamVjdDogcmVjb3JkLlNucy5TdWJqZWN0LFxuICAgICAgICAgIHRpbWVzdGFtcDogcmVjb3JkLlNucy5UaW1lc3RhbXAsXG4gICAgICAgICAgdG9waWNBcm46IHJlY29yZC5TbnMuVG9waWNBcm4sXG4gICAgICAgICAgbWVzc2FnZUlkOiByZWNvcmQuU25zLk1lc3NhZ2VJZCxcbiAgICAgICAgICBtZXNzYWdlQXR0cmlidXRlczogcmVjb3JkLlNucy5NZXNzYWdlQXR0cmlidXRlcyxcbiAgICAgICAgfTtcblxuICAgICAgICBjb25zdCBmbHVlbnRFdmVudDogT21pdDxTcHlNZXNzYWdlLCAndGltZXN0YW1wJz4gPSB7XG4gICAgICAgICAgZGF0YSxcbiAgICAgICAgICBzZXJ2aWNlS2V5LFxuICAgICAgICB9O1xuICAgICAgICBwb3N0RGF0YVByb21pc2VzLnB1c2godGhpcy5wb3N0RGF0YShmbHVlbnRFdmVudCkpO1xuICAgICAgfVxuICAgIH0gZWxzZSBpZiAoZXZlbnQ/LlJlY29yZHMgJiYgZXZlbnQuUmVjb3Jkc1swXT8uZXZlbnRTb3VyY2UgPT09ICdhd3M6c3FzJykge1xuICAgICAgLy9jb25zb2xlLmxvZygnKioqIFNRUyAqKionKTtcbiAgICAgIGNvbnN0IGV2ZW50U3FzID0gZXZlbnQgYXMgU1FTRXZlbnQ7XG4gICAgICBmb3IgKGNvbnN0IHJlY29yZCBvZiBldmVudFNxcy5SZWNvcmRzKSB7XG4gICAgICAgIGNvbnN0IHN1YnNjcmlwdGlvbkFybiA9IHJlY29yZC5ldmVudFNvdXJjZUFSTjtcblxuICAgICAgICBjb25zdCBzZXJ2aWNlS2V5ID0gbWFwcGluZ1tzdWJzY3JpcHRpb25Bcm5dO1xuICAgICAgICBsZXQgYm9keTogc3RyaW5nO1xuXG4gICAgICAgIHRyeSB7XG4gICAgICAgICAgYm9keSA9IEpTT04ucGFyc2UocmVjb3JkLmJvZHkpO1xuICAgICAgICB9IGNhdGNoIHtcbiAgICAgICAgICBib2R5ID0gcmVjb3JkLmJvZHk7XG4gICAgICAgIH1cblxuICAgICAgICBjb25zdCBkYXRhOiBTcXNTcHlFdmVudCA9IHtcbiAgICAgICAgICBzcHlFdmVudFR5cGU6ICdTcXMnLFxuICAgICAgICAgIGJvZHksXG4gICAgICAgICAgbWVzc2FnZUF0dHJpYnV0ZXM6IHJlY29yZC5tZXNzYWdlQXR0cmlidXRlcyxcbiAgICAgICAgfTtcblxuICAgICAgICBjb25zdCBmbHVlbnRFdmVudDogT21pdDxTcHlNZXNzYWdlLCAndGltZXN0YW1wJz4gPSB7XG4gICAgICAgICAgZGF0YSxcbiAgICAgICAgICBzZXJ2aWNlS2V5LFxuICAgICAgICB9O1xuICAgICAgICBwb3N0RGF0YVByb21pc2VzLnB1c2godGhpcy5wb3N0RGF0YShmbHVlbnRFdmVudCkpO1xuICAgICAgfVxuICAgIH0gZWxzZSBpZiAoZXZlbnQ/LlJlY29yZHMgJiYgZXZlbnQuUmVjb3Jkc1swXT8uczMpIHtcbiAgICAgIC8vY29uc29sZS5sb2coJyoqKiBTMyAqKionKTtcbiAgICAgIGNvbnN0IGV2ZW50UzMgPSBldmVudCBhcyBTM0V2ZW50O1xuICAgICAgZm9yIChjb25zdCByZWNvcmQgb2YgZXZlbnRTMy5SZWNvcmRzKSB7XG4gICAgICAgIGNvbnN0IGJ1Y2tldEFybiA9IHJlY29yZC5zMy5idWNrZXQuYXJuO1xuXG4gICAgICAgIGNvbnN0IGRhdGE6IFMzU3B5RXZlbnQgPSB7XG4gICAgICAgICAgc3B5RXZlbnRUeXBlOiAnUzMnLFxuICAgICAgICAgIGV2ZW50TmFtZTogcmVjb3JkLmV2ZW50TmFtZSxcbiAgICAgICAgICBldmVudFRpbWU6IHJlY29yZC5ldmVudFRpbWUsXG4gICAgICAgICAgYnVja2V0OiByZWNvcmQuczMuYnVja2V0Lm5hbWUsXG4gICAgICAgICAga2V5OiByZWNvcmQuczMub2JqZWN0LmtleSxcbiAgICAgICAgfTtcblxuICAgICAgICBjb25zdCBmbHVlbnRFdmVudDogT21pdDxTcHlNZXNzYWdlLCAndGltZXN0YW1wJz4gPSB7XG4gICAgICAgICAgZGF0YSxcbiAgICAgICAgICBzZXJ2aWNlS2V5OiBtYXBwaW5nW2J1Y2tldEFybl0sXG4gICAgICAgIH07XG4gICAgICAgIHBvc3REYXRhUHJvbWlzZXMucHVzaCh0aGlzLnBvc3REYXRhKGZsdWVudEV2ZW50KSk7XG4gICAgICB9XG4gICAgfSBlbHNlIGlmIChldmVudC5SZWNvcmRzICYmIGV2ZW50LlJlY29yZHNbMF0/LmR5bmFtb2RiKSB7XG4gICAgICAvL2NvbnNvbGUubG9nKCcqKiogRFlOQU1PREIgKioqJyk7XG4gICAgICBjb25zdCBldmVudER5bmFtb0RCID0gZXZlbnQgYXMgRHluYW1vREJTdHJlYW1FdmVudDtcbiAgICAgIGZvciAoY29uc3QgcmVjb3JkIG9mIGV2ZW50RHluYW1vREIuUmVjb3Jkcykge1xuICAgICAgICBsZXQgYXJuID0gcmVjb3JkLmV2ZW50U291cmNlQVJOITtcbiAgICAgICAgYXJuID0gYXJuLnN1YnN0cmluZygwLCBhcm4uaW5kZXhPZignL3N0cmVhbS8nKSk7XG5cbiAgICAgICAgY29uc3QgZGF0YTogRHluYW1vREJTcHlFdmVudCA9IHtcbiAgICAgICAgICBzcHlFdmVudFR5cGU6ICdEeW5hbW9EQicsXG4gICAgICAgICAgZXZlbnROYW1lOiByZWNvcmQuZXZlbnROYW1lLFxuICAgICAgICAgIG5ld0ltYWdlOiByZWNvcmQuZHluYW1vZGI/Lk5ld0ltYWdlXG4gICAgICAgICAgICA/IHVubWFyc2hhbGwocmVjb3JkLmR5bmFtb2RiPy5OZXdJbWFnZSBhcyBhbnkpXG4gICAgICAgICAgICA6IHVuZGVmaW5lZCxcbiAgICAgICAgICBrZXlzOiB1bm1hcnNoYWxsKHJlY29yZC5keW5hbW9kYj8uS2V5cyBhcyBhbnkpLFxuICAgICAgICAgIG9sZEltYWdlOiByZWNvcmQuZHluYW1vZGI/Lk9sZEltYWdlXG4gICAgICAgICAgICA/IHVubWFyc2hhbGwocmVjb3JkLmR5bmFtb2RiPy5PbGRJbWFnZSBhcyBhbnkpXG4gICAgICAgICAgICA6IHVuZGVmaW5lZCxcbiAgICAgICAgfTtcblxuICAgICAgICBjb25zdCBmbHVlbnRFdmVudDogT21pdDxTcHlNZXNzYWdlLCAndGltZXN0YW1wJz4gPSB7XG4gICAgICAgICAgZGF0YSxcbiAgICAgICAgICBzZXJ2aWNlS2V5OiBtYXBwaW5nW2Fybl0sXG4gICAgICAgIH07XG4gICAgICAgIHBvc3REYXRhUHJvbWlzZXMucHVzaCh0aGlzLnBvc3REYXRhKGZsdWVudEV2ZW50KSk7XG4gICAgICB9XG4gICAgfSBlbHNlIGlmIChcbiAgICAgIGV2ZW50LmRldGFpbCAmJlxuICAgICAgZXZlbnRbJ2RldGFpbC10eXBlJ10gJiZcbiAgICAgIGV2ZW50LnZlcnNpb24gJiZcbiAgICAgIGV2ZW50LnNvdXJjZVxuICAgICkge1xuICAgICAgLy9jb25zb2xlLmxvZygnKioqIEV2ZW50QnJpZGdlICoqKicpO1xuICAgICAgY29uc3QgZXZlbnRFYiA9IGV2ZW50IGFzIEV2ZW50QnJpZGdlRXZlbnQ8YW55LCBhbnk+O1xuXG4gICAgICBjb25zdCBzZXJ2aWNlS2V5ID0gbWFwcGluZy5ldmVudEJyaWRnZTsgLy8gdGhlIGlzIG5ldyBsYW1iZGEgZm9yIGVhY2ggc3Vic2NyaXB0aW9uXG5cbiAgICAgIGNvbnN0IHNweUV2ZW50VHlwZSA9IHRoaXMuZ2V0U3B5RXZlbnRUeXBlKHNlcnZpY2VLZXkpIGFzXG4gICAgICAgIHwgJ0V2ZW50QnJpZGdlJ1xuICAgICAgICB8ICdFdmVudEJyaWRnZVJ1bGUnO1xuXG4gICAgICBjb25zdCBtZXNzYWdlID0gZXZlbnRFYi5kZXRhaWw7XG5cbiAgICAgIGNvbnN0IGRhdGE6IEV2ZW50QnJpZGdlU3B5RXZlbnQgfCBFdmVudEJyaWRnZVJ1bGVTcHlFdmVudCA9IHtcbiAgICAgICAgc3B5RXZlbnRUeXBlLFxuICAgICAgICBkZXRhaWw6IG1lc3NhZ2UsXG4gICAgICAgIGRldGFpbFR5cGU6IGV2ZW50RWJbJ2RldGFpbC10eXBlJ10sXG4gICAgICAgIGV2ZW50QnJpZGdlSWQ6IGV2ZW50RWJbJ2lkJ10sXG4gICAgICAgIHNvdXJjZTogZXZlbnRFYi5zb3VyY2UsXG4gICAgICAgIHRpbWU6IGV2ZW50RWIudGltZSxcbiAgICAgICAgYWNjb3VudDogZXZlbnRFYi5hY2NvdW50LFxuICAgICAgfTtcblxuICAgICAgY29uc3QgZmx1ZW50RXZlbnQ6IE9taXQ8U3B5TWVzc2FnZSwgJ3RpbWVzdGFtcCc+ID0ge1xuICAgICAgICBkYXRhLFxuICAgICAgICBzZXJ2aWNlS2V5LFxuICAgICAgfTtcbiAgICAgIHBvc3REYXRhUHJvbWlzZXMucHVzaCh0aGlzLnBvc3REYXRhKGZsdWVudEV2ZW50KSk7XG4gICAgfSBlbHNlIHtcbiAgICAgIC8vY29uc29sZS5sb2coJyoqKiBPVEhFUiAqKionKTtcbiAgICAgIGNvbnN0IGZsdWVudEV2ZW50OiBPbWl0PFNweU1lc3NhZ2UsICd0aW1lc3RhbXAnPiA9IGV2ZW50O1xuICAgICAgcG9zdERhdGFQcm9taXNlcy5wdXNoKHRoaXMucG9zdERhdGEoZmx1ZW50RXZlbnQpKTtcbiAgICB9XG5cbiAgICBhd2FpdCBQcm9taXNlLmFsbChwb3N0RGF0YVByb21pc2VzKTtcbiAgfVxuXG4gIHByaXZhdGUgZW5jb2RlKGlucHV0OiBhbnkpOiBmcmFnbWVudFtdIHtcbiAgICBjb25zdCBwYXlsb2FkID0gSlNPTi5zdHJpbmdpZnkoaW5wdXQpO1xuICAgIGNvbnN0IHBhcnRzID0gcGF5bG9hZC5tYXRjaCgvLnsxLDUwMDAwfS9nKTtcbiAgICBpZiAoIXBhcnRzKSByZXR1cm4gW107XG4gICAgdGhpcy5sb2coYEVuY29kZWQgaW90IG1lc3NhZ2UsICR7cGFydHMubGVuZ3RofWApO1xuICAgIGNvbnN0IGlkID0gdjQoKTtcbiAgICByZXR1cm4gcGFydHMubWFwKChwYXJ0LCBpbmRleCkgPT4gKHtcbiAgICAgIGlkLFxuICAgICAgaW5kZXgsXG4gICAgICBjb3VudDogcGFydHMubGVuZ3RoLFxuICAgICAgZGF0YTogcGFydCxcbiAgICB9KSk7XG4gIH1cblxuICBwcml2YXRlIGFzeW5jIHBvc3REYXRhKFxuICAgIHNweU1lc3NhZ2U6IE9taXQ8U3B5TWVzc2FnZSwgJ3RpbWVzdGFtcCc+ICYgeyB0aW1lc3RhbXA/OiBzdHJpbmcgfVxuICApIHtcbiAgICBpZiAodGhpcy5jb25uZWN0aW9uID09PSB1bmRlZmluZWQpIHtcbiAgICAgIHRocm93IG5ldyBFcnJvcihcbiAgICAgICAgJ05vIElvVCBjb25uZWN0aW9uIGNyZWF0ZWQgeWV0LCBkaWQgeW91IGZvcmdldCB0byBjYWxsIGNvbm5lY3QoKT8nXG4gICAgICApO1xuICAgIH1cblxuICAgIGNvbnN0IHdpdGhUaW1lU3RhbXAgPSB7XG4gICAgICAuLi5zcHlNZXNzYWdlLFxuICAgICAgdGltZXN0YW1wOiBzcHlNZXNzYWdlLnRpbWVzdGFtcCB8fCBuZXcgRGF0ZSgpLnRvSVNPU3RyaW5nKCksXG4gICAgfTtcblxuICAgIHRoaXMubG9nKCdQb3N0IHNweSBtZXNzYWdlJywgSlNPTi5zdHJpbmdpZnkod2l0aFRpbWVTdGFtcCkpO1xuXG4gICAgY29uc3QgY29ubmVjdGlvbiA9IHRoaXMuY29ubmVjdGlvbjtcbiAgICBjb25zdCB0b3BpYyA9IGdldFRvcGljKHRoaXMuc2NvcGUpO1xuXG4gICAgdHJ5IHtcbiAgICAgIGZvciAoY29uc3QgZnJhZ21lbnQgb2YgdGhpcy5lbmNvZGUod2l0aFRpbWVTdGFtcCkpIHtcbiAgICAgICAgYXdhaXQgbmV3IFByb21pc2U8dm9pZD4oKHJlc29sdmUpID0+IHtcbiAgICAgICAgICBjb25uZWN0aW9uLnB1Ymxpc2goXG4gICAgICAgICAgICB0b3BpYyxcbiAgICAgICAgICAgIEpTT04uc3RyaW5naWZ5KGZyYWdtZW50KSxcbiAgICAgICAgICAgIHtcbiAgICAgICAgICAgICAgcW9zOiAxLFxuICAgICAgICAgICAgfSxcbiAgICAgICAgICAgICgpID0+IHtcbiAgICAgICAgICAgICAgdGhpcy5sb2coJ1B1Ymxpc2hpbmcgZmluaXNoZWQnKTtcbiAgICAgICAgICAgICAgcmVzb2x2ZSgpO1xuICAgICAgICAgICAgfVxuICAgICAgICAgICk7XG4gICAgICAgIH0pO1xuICAgICAgICB0aGlzLmxvZyhcbiAgICAgICAgICBgUHVibGlzaGVkIGZyYWdtZW50ICR7ZnJhZ21lbnQuaW5kZXh9IG91dCBvZiAke2ZyYWdtZW50LmNvdW50fSB0byB0b3BpYyAke3RvcGljfWBcbiAgICAgICAgKTtcbiAgICAgIH1cbiAgICB9IGNhdGNoIChlKSB7XG4gICAgICB0aGlzLmxvZ0Vycm9yKGBGYWlsZWQgdG8gc2VuZCBwYXlsb2FkIHRvIGlvdDogJHtlfWApO1xuICAgIH1cblxuICAgIHRoaXMubG9nKCdTZW5kIHNweSBtZXNzYWdlIGZpbmlzaCcpO1xuICB9XG5cbiAgcHJpdmF0ZSBnZXRTcHlFdmVudFR5cGUoc2VydmljZUtleTogc3RyaW5nKSB7XG4gICAgaWYgKCFzZXJ2aWNlS2V5KSB7XG4gICAgICB0aHJvdyBuZXcgRXJyb3IoJ01pc3Npbmcgc2VydmljZUtleScpO1xuICAgIH1cblxuICAgIHJldHVybiBzZXJ2aWNlS2V5LnN1YnN0cmluZygwLCBzZXJ2aWNlS2V5LmluZGV4T2YoJyMnKSk7XG4gIH1cblxuICBwcml2YXRlIGxvZyhtZXNzYWdlOiBzdHJpbmcsIC4uLm9wdGlvbmFsUGFyYW1zOiBhbnlbXSkge1xuICAgIGlmICh0aGlzLmRlYnVnTW9kZSkge1xuICAgICAgY29uc29sZS5kZWJ1ZygnU1NQWSBFWFRFTlNJT04nLCBtZXNzYWdlLCAuLi5vcHRpb25hbFBhcmFtcyk7XG4gICAgfVxuICB9XG5cbiAgcHJpdmF0ZSBsb2dFcnJvcihtZXNzYWdlOiBzdHJpbmcsIC4uLm9wdGlvbmFsUGFyYW1zOiBhbnlbXSkge1xuICAgIGlmICh0aGlzLmRlYnVnTW9kZSkge1xuICAgICAgY29uc29sZS5lcnJvcignU1NQWSBFWFRFTlNJT04nLCBtZXNzYWdlLCAuLi5vcHRpb25hbFBhcmFtcyk7XG4gICAgfVxuICB9XG59XG4iXX0=
1
+ import { envVariableNames, init_envVariableNames } from "../src/common/envVariableNames.mjs";
2
+ import { getConnection } from "../listener/iot-connection.mjs";
3
+ import { getTopic } from "../listener/topic.mjs";
4
+ import v4_default from "../node_modules/uuid/dist/esm-node/v4.mjs";
5
+ import { unmarshall } from "@aws-sdk/util-dynamodb";
6
+
7
+ //#region common/SpyEventSender.ts
8
+ init_envVariableNames();
9
+ var SpyEventSender = class {
10
+ constructor(params) {
11
+ this.debugMode = process.env[envVariableNames.SSPY_DEBUG] === "true";
12
+ if (params.log) this.log = params.log;
13
+ if (params.logError) this.logError = params.logError;
14
+ this.scope = params.scope;
15
+ this.iotEndpoint = params.iotEndpoint;
16
+ }
17
+ async close() {
18
+ this.connection?.end();
19
+ }
20
+ async connect() {
21
+ this.connection = await getConnection(this.debugMode, this.iotEndpoint);
22
+ }
23
+ async publishSpyEvent(event) {
24
+ this.log("Event", JSON.stringify(event));
25
+ const mapping = JSON.parse(process.env[envVariableNames.SSPY_INFRA_MAPPING]);
26
+ this.log("ARN to names mapping", JSON.stringify(mapping));
27
+ const postDataPromises = [];
28
+ if (event?.Records && event.Records[0]?.Sns) {
29
+ const eventSns = event;
30
+ for (const record of eventSns.Records) {
31
+ const subscriptionArn = record.EventSubscriptionArn;
32
+ let serviceKey;
33
+ if (mapping[subscriptionArn]) serviceKey = mapping[subscriptionArn];
34
+ else serviceKey = mapping[record.Sns.TopicArn];
35
+ let message;
36
+ try {
37
+ message = JSON.parse(record.Sns.Message);
38
+ } catch {
39
+ message = record.Sns.Message;
40
+ }
41
+ const fluentEvent = {
42
+ data: {
43
+ spyEventType: this.getSpyEventType(serviceKey),
44
+ message,
45
+ subject: record.Sns.Subject,
46
+ timestamp: record.Sns.Timestamp,
47
+ topicArn: record.Sns.TopicArn,
48
+ messageId: record.Sns.MessageId,
49
+ messageAttributes: record.Sns.MessageAttributes
50
+ },
51
+ serviceKey
52
+ };
53
+ postDataPromises.push(this.postData(fluentEvent));
54
+ }
55
+ } else if (event?.Records && event.Records[0]?.eventSource === "aws:sqs") {
56
+ const eventSqs = event;
57
+ for (const record of eventSqs.Records) {
58
+ const serviceKey = mapping[record.eventSourceARN];
59
+ let body;
60
+ try {
61
+ body = JSON.parse(record.body);
62
+ } catch {
63
+ body = record.body;
64
+ }
65
+ const fluentEvent = {
66
+ data: {
67
+ spyEventType: "Sqs",
68
+ body,
69
+ messageAttributes: record.messageAttributes
70
+ },
71
+ serviceKey
72
+ };
73
+ postDataPromises.push(this.postData(fluentEvent));
74
+ }
75
+ } else if (event?.Records && event.Records[0]?.s3) {
76
+ const eventS3 = event;
77
+ for (const record of eventS3.Records) {
78
+ const bucketArn = record.s3.bucket.arn;
79
+ const fluentEvent = {
80
+ data: {
81
+ spyEventType: "S3",
82
+ eventName: record.eventName,
83
+ eventTime: record.eventTime,
84
+ bucket: record.s3.bucket.name,
85
+ key: record.s3.object.key
86
+ },
87
+ serviceKey: mapping[bucketArn]
88
+ };
89
+ postDataPromises.push(this.postData(fluentEvent));
90
+ }
91
+ } else if (event.Records && event.Records[0]?.dynamodb) {
92
+ const eventDynamoDB = event;
93
+ for (const record of eventDynamoDB.Records) {
94
+ let arn = record.eventSourceARN;
95
+ arn = arn.substring(0, arn.indexOf("/stream/"));
96
+ const fluentEvent = {
97
+ data: {
98
+ spyEventType: "DynamoDB",
99
+ eventName: record.eventName,
100
+ newImage: record.dynamodb?.NewImage ? unmarshall(record.dynamodb?.NewImage) : void 0,
101
+ keys: unmarshall(record.dynamodb?.Keys),
102
+ oldImage: record.dynamodb?.OldImage ? unmarshall(record.dynamodb?.OldImage) : void 0
103
+ },
104
+ serviceKey: mapping[arn]
105
+ };
106
+ postDataPromises.push(this.postData(fluentEvent));
107
+ }
108
+ } else if (event.detail && event["detail-type"] && event.version && event.source) {
109
+ const eventEb = event;
110
+ const serviceKey = mapping.eventBridge;
111
+ const fluentEvent = {
112
+ data: {
113
+ spyEventType: this.getSpyEventType(serviceKey),
114
+ detail: eventEb.detail,
115
+ detailType: eventEb["detail-type"],
116
+ eventBridgeId: eventEb["id"],
117
+ source: eventEb.source,
118
+ time: eventEb.time,
119
+ account: eventEb.account
120
+ },
121
+ serviceKey
122
+ };
123
+ postDataPromises.push(this.postData(fluentEvent));
124
+ } else {
125
+ const fluentEvent = event;
126
+ postDataPromises.push(this.postData(fluentEvent));
127
+ }
128
+ await Promise.all(postDataPromises);
129
+ }
130
+ encode(input) {
131
+ const parts = JSON.stringify(input).match(/.{1,50000}/g);
132
+ if (!parts) return [];
133
+ this.log(`Encoded iot message, ${parts.length}`);
134
+ const id = v4_default();
135
+ return parts.map((part, index) => ({
136
+ id,
137
+ index,
138
+ count: parts.length,
139
+ data: part
140
+ }));
141
+ }
142
+ async postData(spyMessage) {
143
+ if (this.connection === void 0) throw new Error("No IoT connection created yet, did you forget to call connect()?");
144
+ const withTimeStamp = {
145
+ ...spyMessage,
146
+ timestamp: spyMessage.timestamp || (/* @__PURE__ */ new Date()).toISOString()
147
+ };
148
+ this.log("Post spy message", JSON.stringify(withTimeStamp));
149
+ const connection = this.connection;
150
+ const topic = getTopic(this.scope);
151
+ try {
152
+ for (const fragment of this.encode(withTimeStamp)) {
153
+ await new Promise((resolve) => {
154
+ connection.publish(topic, JSON.stringify(fragment), { qos: 1 }, () => {
155
+ this.log("Publishing finished");
156
+ resolve();
157
+ });
158
+ });
159
+ this.log(`Published fragment ${fragment.index} out of ${fragment.count} to topic ${topic}`);
160
+ }
161
+ } catch (e) {
162
+ this.logError(`Failed to send payload to iot: ${e}`);
163
+ }
164
+ this.log("Send spy message finish");
165
+ }
166
+ getSpyEventType(serviceKey) {
167
+ if (!serviceKey) throw new Error("Missing serviceKey");
168
+ return serviceKey.substring(0, serviceKey.indexOf("#"));
169
+ }
170
+ log(message, ...optionalParams) {
171
+ if (this.debugMode) console.debug("SSPY EXTENSION", message, ...optionalParams);
172
+ }
173
+ logError(message, ...optionalParams) {
174
+ if (this.debugMode) console.error("SSPY EXTENSION", message, ...optionalParams);
175
+ }
176
+ };
177
+
178
+ //#endregion
179
+ export { SpyEventSender };
180
+ //# sourceMappingURL=SpyEventSender.mjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"SpyEventSender.mjs","names":["postDataPromises: Promise<any>[]","serviceKey: string","message: string","fluentEvent: Omit<SpyMessage, 'timestamp'>","body: string","v4"],"sources":["../../common/SpyEventSender.ts"],"sourcesContent":["import { unmarshall } from '@aws-sdk/util-dynamodb';\nimport iot from 'aws-iot-device-sdk';\nimport {\n DynamoDBStreamEvent,\n S3Event,\n SNSEvent,\n EventBridgeEvent,\n SQSEvent,\n} from 'aws-lambda';\nimport { v4 } from 'uuid';\nimport { DynamoDBSpyEvent } from './spyEvents/DynamoDBSpyEvent';\nimport { EventBridgeRuleSpyEvent } from './spyEvents/EventBridgeRuleSpyEvent';\nimport { EventBridgeSpyEvent } from './spyEvents/EventBridgeSpyEvent';\nimport { S3SpyEvent } from './spyEvents/S3SpyEvent';\nimport { SnsSubscriptionSpyEvent } from './spyEvents/SnsSubscriptionSpyEvent';\nimport { SnsTopicSpyEvent } from './spyEvents/SnsTopicSpyEvent';\nimport { SpyMessage } from './spyEvents/SpyMessage';\nimport { SqsSpyEvent } from './spyEvents/SqsSpyEvent';\nimport { fragment, getConnection } from '../listener/iot-connection';\nimport { getTopic } from '../listener/topic';\nimport { envVariableNames } from '../src/common/envVariableNames';\n\nexport class SpyEventSender {\n debugMode = process.env[envVariableNames.SSPY_DEBUG] === 'true';\n connection: iot.device | undefined;\n scope: string;\n iotEndpoint: string;\n\n constructor(params: {\n log?: (message: string, ...optionalParams: any[]) => void;\n logError?: (message: string, ...optionalParams: any[]) => void;\n scope: string;\n iotEndpoint: string;\n }) {\n if (params.log) {\n this.log = params.log;\n }\n\n if (params.logError) {\n this.logError = params.logError;\n }\n\n this.scope = params.scope;\n this.iotEndpoint = params.iotEndpoint;\n }\n\n public async close() {\n this.connection?.end();\n }\n\n public async connect() {\n this.connection = await getConnection(this.debugMode, this.iotEndpoint);\n }\n\n public async publishSpyEvent(event: any) {\n this.log('Event', JSON.stringify(event));\n\n const mapping = JSON.parse(\n process.env[envVariableNames.SSPY_INFRA_MAPPING]!\n );\n this.log('ARN to names mapping', JSON.stringify(mapping));\n\n const postDataPromises: Promise<any>[] = [];\n\n if (event?.Records && event.Records[0]?.Sns) {\n //console.log('*** SNS ***');\n const eventSns = event as SNSEvent;\n for (const record of eventSns.Records) {\n const subscriptionArn = record.EventSubscriptionArn;\n\n let serviceKey: string;\n if (mapping[subscriptionArn]) {\n // subscription event that could contain filter based on existing subscription\n serviceKey = mapping[subscriptionArn];\n } else {\n // catch all subscription\n const topicArn = record.Sns.TopicArn;\n serviceKey = mapping[topicArn];\n }\n\n let message: string;\n\n try {\n message = JSON.parse(record.Sns.Message);\n } catch {\n message = record.Sns.Message;\n }\n\n const spyEventType = this.getSpyEventType(serviceKey) as\n | 'FunctionSnsTopic'\n | 'FunctionSnsSubscription';\n\n const data: SnsTopicSpyEvent | SnsSubscriptionSpyEvent = {\n spyEventType,\n message,\n subject: record.Sns.Subject,\n timestamp: record.Sns.Timestamp,\n topicArn: record.Sns.TopicArn,\n messageId: record.Sns.MessageId,\n messageAttributes: record.Sns.MessageAttributes,\n };\n\n const fluentEvent: Omit<SpyMessage, 'timestamp'> = {\n data,\n serviceKey,\n };\n postDataPromises.push(this.postData(fluentEvent));\n }\n } else if (event?.Records && event.Records[0]?.eventSource === 'aws:sqs') {\n //console.log('*** SQS ***');\n const eventSqs = event as SQSEvent;\n for (const record of eventSqs.Records) {\n const subscriptionArn = record.eventSourceARN;\n\n const serviceKey = mapping[subscriptionArn];\n let body: string;\n\n try {\n body = JSON.parse(record.body);\n } catch {\n body = record.body;\n }\n\n const data: SqsSpyEvent = {\n spyEventType: 'Sqs',\n body,\n messageAttributes: record.messageAttributes,\n };\n\n const fluentEvent: Omit<SpyMessage, 'timestamp'> = {\n data,\n serviceKey,\n };\n postDataPromises.push(this.postData(fluentEvent));\n }\n } else if (event?.Records && event.Records[0]?.s3) {\n //console.log('*** S3 ***');\n const eventS3 = event as S3Event;\n for (const record of eventS3.Records) {\n const bucketArn = record.s3.bucket.arn;\n\n const data: S3SpyEvent = {\n spyEventType: 'S3',\n eventName: record.eventName,\n eventTime: record.eventTime,\n bucket: record.s3.bucket.name,\n key: record.s3.object.key,\n };\n\n const fluentEvent: Omit<SpyMessage, 'timestamp'> = {\n data,\n serviceKey: mapping[bucketArn],\n };\n postDataPromises.push(this.postData(fluentEvent));\n }\n } else if (event.Records && event.Records[0]?.dynamodb) {\n //console.log('*** DYNAMODB ***');\n const eventDynamoDB = event as DynamoDBStreamEvent;\n for (const record of eventDynamoDB.Records) {\n let arn = record.eventSourceARN!;\n arn = arn.substring(0, arn.indexOf('/stream/'));\n\n const data: DynamoDBSpyEvent = {\n spyEventType: 'DynamoDB',\n eventName: record.eventName,\n newImage: record.dynamodb?.NewImage\n ? unmarshall(record.dynamodb?.NewImage as any)\n : undefined,\n keys: unmarshall(record.dynamodb?.Keys as any),\n oldImage: record.dynamodb?.OldImage\n ? unmarshall(record.dynamodb?.OldImage as any)\n : undefined,\n };\n\n const fluentEvent: Omit<SpyMessage, 'timestamp'> = {\n data,\n serviceKey: mapping[arn],\n };\n postDataPromises.push(this.postData(fluentEvent));\n }\n } else if (\n event.detail &&\n event['detail-type'] &&\n event.version &&\n event.source\n ) {\n //console.log('*** EventBridge ***');\n const eventEb = event as EventBridgeEvent<any, any>;\n\n const serviceKey = mapping.eventBridge; // the is new lambda for each subscription\n\n const spyEventType = this.getSpyEventType(serviceKey) as\n | 'EventBridge'\n | 'EventBridgeRule';\n\n const message = eventEb.detail;\n\n const data: EventBridgeSpyEvent | EventBridgeRuleSpyEvent = {\n spyEventType,\n detail: message,\n detailType: eventEb['detail-type'],\n eventBridgeId: eventEb['id'],\n source: eventEb.source,\n time: eventEb.time,\n account: eventEb.account,\n };\n\n const fluentEvent: Omit<SpyMessage, 'timestamp'> = {\n data,\n serviceKey,\n };\n postDataPromises.push(this.postData(fluentEvent));\n } else {\n //console.log('*** OTHER ***');\n const fluentEvent: Omit<SpyMessage, 'timestamp'> = event;\n postDataPromises.push(this.postData(fluentEvent));\n }\n\n await Promise.all(postDataPromises);\n }\n\n private encode(input: any): fragment[] {\n const payload = JSON.stringify(input);\n const parts = payload.match(/.{1,50000}/g);\n if (!parts) return [];\n this.log(`Encoded iot message, ${parts.length}`);\n const id = v4();\n return parts.map((part, index) => ({\n id,\n index,\n count: parts.length,\n data: part,\n }));\n }\n\n private async postData(\n spyMessage: Omit<SpyMessage, 'timestamp'> & { timestamp?: string }\n ) {\n if (this.connection === undefined) {\n throw new Error(\n 'No IoT connection created yet, did you forget to call connect()?'\n );\n }\n\n const withTimeStamp = {\n ...spyMessage,\n timestamp: spyMessage.timestamp || new Date().toISOString(),\n };\n\n this.log('Post spy message', JSON.stringify(withTimeStamp));\n\n const connection = this.connection;\n const topic = getTopic(this.scope);\n\n try {\n for (const fragment of this.encode(withTimeStamp)) {\n await new Promise<void>((resolve) => {\n connection.publish(\n topic,\n JSON.stringify(fragment),\n {\n qos: 1,\n },\n () => {\n this.log('Publishing finished');\n resolve();\n }\n );\n });\n this.log(\n `Published fragment ${fragment.index} out of ${fragment.count} to topic ${topic}`\n );\n }\n } catch (e) {\n this.logError(`Failed to send payload to iot: ${e}`);\n }\n\n this.log('Send spy message finish');\n }\n\n private getSpyEventType(serviceKey: string) {\n if (!serviceKey) {\n throw new Error('Missing serviceKey');\n }\n\n return serviceKey.substring(0, serviceKey.indexOf('#'));\n }\n\n private log(message: string, ...optionalParams: any[]) {\n if (this.debugMode) {\n console.debug('SSPY EXTENSION', message, ...optionalParams);\n }\n }\n\n private logError(message: string, ...optionalParams: any[]) {\n if (this.debugMode) {\n console.error('SSPY EXTENSION', message, ...optionalParams);\n }\n }\n}\n"],"mappings":"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"}
@@ -0,0 +1,7 @@
1
+ import { Credentials } from "@aws-sdk/types";
2
+
3
+ //#region common/getWebSocketUrl.d.ts
4
+ declare function getSignedWebSocketUrl(url: string, credentials?: Credentials): Promise<string>;
5
+ //#endregion
6
+ export { getSignedWebSocketUrl };
7
+ //# sourceMappingURL=getWebSocketUrl.d.mts.map
@@ -1,2 +1,7 @@
1
- import { Credentials } from '@aws-sdk/types';
2
- export declare function getSignedWebSocketUrl(url: string, credentials?: Credentials): Promise<string>;
1
+ import { Credentials } from "@aws-sdk/types";
2
+
3
+ //#region common/getWebSocketUrl.d.ts
4
+ declare function getSignedWebSocketUrl(url: string, credentials?: Credentials): Promise<string>;
5
+ //#endregion
6
+ export { getSignedWebSocketUrl };
7
+ //# sourceMappingURL=getWebSocketUrl.d.ts.map