@nlabs/reaktor 0.3.0 → 0.4.1

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 (128) hide show
  1. package/lib/actions/conversations.d.ts +14 -0
  2. package/lib/actions/conversations.js +333 -0
  3. package/lib/actions/dynamodb.js +155 -0
  4. package/lib/actions/email.js +177 -0
  5. package/lib/actions/files.js +319 -0
  6. package/lib/{data → actions}/groups.d.ts +4 -3
  7. package/lib/actions/groups.js +282 -0
  8. package/lib/actions/images.d.ts +22 -0
  9. package/lib/actions/images.js +682 -0
  10. package/lib/actions/index.js +40 -0
  11. package/lib/{data → actions}/ios.d.ts +2 -1
  12. package/lib/actions/ios.js +179 -0
  13. package/lib/actions/locations.js +112 -0
  14. package/lib/actions/messages.d.ts +13 -0
  15. package/lib/actions/messages.js +216 -0
  16. package/lib/actions/notifications.js +63 -0
  17. package/lib/{data → actions}/payments.d.ts +2 -2
  18. package/lib/actions/payments.js +491 -0
  19. package/lib/actions/posts.d.ts +19 -0
  20. package/lib/actions/posts.js +538 -0
  21. package/lib/actions/reactions.d.ts +30 -0
  22. package/lib/actions/reactions.js +340 -0
  23. package/lib/{data → actions}/s3.d.ts +1 -1
  24. package/lib/actions/s3.js +122 -0
  25. package/lib/{data → actions}/search.d.ts +2 -2
  26. package/lib/actions/search.js +99 -0
  27. package/lib/actions/sms.js +76 -0
  28. package/lib/actions/statistics.d.ts +2 -0
  29. package/lib/actions/statistics.js +63 -0
  30. package/lib/actions/subscription.js +209 -0
  31. package/lib/actions/tags.d.ts +26 -0
  32. package/lib/actions/tags.js +340 -0
  33. package/lib/actions/users.d.ts +44 -0
  34. package/lib/actions/users.js +571 -0
  35. package/lib/{data → actions}/websockets.d.ts +1 -1
  36. package/lib/actions/websockets.js +156 -0
  37. package/lib/config.d.ts +2 -3
  38. package/lib/config.js +120 -0
  39. package/lib/index.d.ts +1 -1
  40. package/lib/index.js +23 -0
  41. package/lib/templates/email/layout.d.ts +2 -0
  42. package/lib/templates/email/layout.js +292 -0
  43. package/lib/templates/email/passwordForgot.d.ts +2 -0
  44. package/lib/templates/email/passwordForgot.js +28 -0
  45. package/lib/templates/email/passwordRecovery.d.ts +2 -0
  46. package/lib/templates/email/passwordRecovery.js +25 -0
  47. package/lib/templates/email/verifyEmail.d.ts +2 -0
  48. package/lib/templates/email/verifyEmail.js +28 -0
  49. package/lib/templates/email/welcome.d.ts +2 -0
  50. package/lib/templates/email/welcome.js +28 -0
  51. package/lib/templates/sms/passwordForgot.d.ts +2 -0
  52. package/lib/templates/sms/passwordForgot.js +14 -0
  53. package/lib/templates/sms/passwordRecovery.d.ts +2 -0
  54. package/lib/templates/sms/passwordRecovery.js +14 -0
  55. package/lib/templates/sms/verifyEmail.d.ts +2 -0
  56. package/lib/templates/sms/verifyEmail.js +14 -0
  57. package/lib/templates/sms/verifyPhone.d.ts +2 -0
  58. package/lib/templates/sms/verifyPhone.js +14 -0
  59. package/lib/templates/sms/welcome.d.ts +2 -0
  60. package/lib/templates/sms/welcome.js +14 -0
  61. package/lib/types/apps.d.ts +2 -2
  62. package/lib/types/apps.js +4 -0
  63. package/lib/types/arangodb.js +4 -0
  64. package/lib/types/auth.d.ts +4 -8
  65. package/lib/types/auth.js +4 -0
  66. package/lib/types/conversations.d.ts +5 -3
  67. package/lib/types/conversations.js +4 -0
  68. package/lib/types/email.d.ts +2 -2
  69. package/lib/types/email.js +4 -0
  70. package/lib/types/files.js +4 -0
  71. package/lib/types/google.js +4 -0
  72. package/lib/types/groups.d.ts +2 -1
  73. package/lib/types/groups.js +4 -0
  74. package/lib/types/images.d.ts +8 -5
  75. package/lib/types/images.js +4 -0
  76. package/lib/types/index.d.ts +1 -1
  77. package/lib/types/index.js +37 -0
  78. package/lib/types/locations.js +4 -0
  79. package/lib/types/messages.d.ts +12 -2
  80. package/lib/types/messages.js +4 -0
  81. package/lib/types/notifications.d.ts +2 -2
  82. package/lib/types/notifications.js +4 -0
  83. package/lib/types/payments.js +4 -0
  84. package/lib/types/posts.d.ts +18 -1
  85. package/lib/types/posts.js +4 -0
  86. package/lib/types/statistics.d.ts +3 -0
  87. package/lib/types/statistics.js +4 -0
  88. package/lib/types/tags.d.ts +6 -0
  89. package/lib/types/tags.js +4 -0
  90. package/lib/types/users.d.ts +15 -10
  91. package/lib/types/users.js +4 -0
  92. package/lib/utils/analytics.d.ts +7 -0
  93. package/lib/utils/analytics.js +107 -0
  94. package/lib/utils/arangodb.d.ts +1 -1
  95. package/lib/utils/arangodb.js +122 -0
  96. package/lib/utils/auth.js +78 -0
  97. package/lib/utils/graphql.js +40 -0
  98. package/lib/utils/index.d.ts +1 -1
  99. package/lib/utils/index.js +26 -0
  100. package/lib/utils/objects.js +53 -0
  101. package/lib/utils/session.d.ts +18 -0
  102. package/lib/utils/session.js +42 -0
  103. package/package.json +35 -33
  104. package/lib/data/conversations.d.ts +0 -8
  105. package/lib/data/images.d.ts +0 -21
  106. package/lib/data/messages.d.ts +0 -9
  107. package/lib/data/posts.d.ts +0 -23
  108. package/lib/data/reactions.d.ts +0 -14
  109. package/lib/data/tags.d.ts +0 -14
  110. package/lib/data/users.d.ts +0 -17
  111. package/lib/types/reactions.d.ts +0 -15
  112. package/lib/utils/redis.d.ts +0 -1
  113. package/templates/email/layout.html +0 -279
  114. package/templates/email/passwordForgot.html +0 -15
  115. package/templates/email/passwordRecovery.html +0 -12
  116. package/templates/email/verifyEmail.html +0 -15
  117. package/templates/sms/passwordForgot.txt +0 -1
  118. package/templates/sms/passwordRecovery.txt +0 -1
  119. package/templates/sms/verifyEmail.txt +0 -1
  120. package/templates/sms/verifyPhone.txt +0 -1
  121. /package/lib/{data → actions}/dynamodb.d.ts +0 -0
  122. /package/lib/{data → actions}/email.d.ts +0 -0
  123. /package/lib/{data → actions}/files.d.ts +0 -0
  124. /package/lib/{data → actions}/index.d.ts +0 -0
  125. /package/lib/{data → actions}/locations.d.ts +0 -0
  126. /package/lib/{data → actions}/notifications.d.ts +0 -0
  127. /package/lib/{data → actions}/sms.d.ts +0 -0
  128. /package/lib/{data → actions}/subscription.d.ts +0 -0
@@ -0,0 +1,40 @@
1
+ var __create = Object.create;
2
+ var __defProp = Object.defineProperty;
3
+ var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
4
+ var __getOwnPropNames = Object.getOwnPropertyNames;
5
+ var __getProtoOf = Object.getPrototypeOf;
6
+ var __hasOwnProp = Object.prototype.hasOwnProperty;
7
+ var __markAsModule = (target) => __defProp(target, "__esModule", { value: true });
8
+ var __reExport = (target, module2, desc) => {
9
+ if (module2 && typeof module2 === "object" || typeof module2 === "function") {
10
+ for (let key of __getOwnPropNames(module2))
11
+ if (!__hasOwnProp.call(target, key) && key !== "default")
12
+ __defProp(target, key, { get: () => module2[key], enumerable: !(desc = __getOwnPropDesc(module2, key)) || desc.enumerable });
13
+ }
14
+ return target;
15
+ };
16
+ var __toModule = (module2) => {
17
+ return __reExport(__markAsModule(__defProp(module2 != null ? __create(__getProtoOf(module2)) : {}, "default", module2 && module2.__esModule && "default" in module2 ? { get: () => module2.default, enumerable: true } : { value: module2, enumerable: true })), module2);
18
+ };
19
+ __markAsModule(exports);
20
+ __reExport(exports, __toModule(require("./conversations")));
21
+ __reExport(exports, __toModule(require("./dynamodb")));
22
+ __reExport(exports, __toModule(require("./email")));
23
+ __reExport(exports, __toModule(require("./files")));
24
+ __reExport(exports, __toModule(require("./groups")));
25
+ __reExport(exports, __toModule(require("./images")));
26
+ __reExport(exports, __toModule(require("./ios")));
27
+ __reExport(exports, __toModule(require("./locations")));
28
+ __reExport(exports, __toModule(require("./messages")));
29
+ __reExport(exports, __toModule(require("./notifications")));
30
+ __reExport(exports, __toModule(require("./payments")));
31
+ __reExport(exports, __toModule(require("./posts")));
32
+ __reExport(exports, __toModule(require("./reactions")));
33
+ __reExport(exports, __toModule(require("./s3")));
34
+ __reExport(exports, __toModule(require("./search")));
35
+ __reExport(exports, __toModule(require("./sms")));
36
+ __reExport(exports, __toModule(require("./subscription")));
37
+ __reExport(exports, __toModule(require("./tags")));
38
+ __reExport(exports, __toModule(require("./users")));
39
+ __reExport(exports, __toModule(require("./websockets")));
40
+ //# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFsiLi4vLi4vc3JjL2FjdGlvbnMvaW5kZXgudHMiXSwKICAic291cmNlc0NvbnRlbnQiOiBbIi8qKlxuICogQ29weXJpZ2h0IChjKSAyMDE5LVByZXNlbnQsIE5pdHJvZ2VuIExhYnMsIEluYy5cbiAqIENvcHlyaWdodHMgbGljZW5zZWQgdW5kZXIgdGhlIE1JVCBMaWNlbnNlLiBTZWUgdGhlIGFjY29tcGFueWluZyBMSUNFTlNFIGZpbGUgZm9yIHRlcm1zLlxuICovXG5leHBvcnQgKiBmcm9tICcuL2NvbnZlcnNhdGlvbnMnO1xuZXhwb3J0ICogZnJvbSAnLi9keW5hbW9kYic7XG5leHBvcnQgKiBmcm9tICcuL2VtYWlsJztcbmV4cG9ydCAqIGZyb20gJy4vZmlsZXMnO1xuZXhwb3J0ICogZnJvbSAnLi9ncm91cHMnO1xuZXhwb3J0ICogZnJvbSAnLi9pbWFnZXMnO1xuZXhwb3J0ICogZnJvbSAnLi9pb3MnO1xuZXhwb3J0ICogZnJvbSAnLi9sb2NhdGlvbnMnO1xuZXhwb3J0ICogZnJvbSAnLi9tZXNzYWdlcyc7XG5leHBvcnQgKiBmcm9tICcuL25vdGlmaWNhdGlvbnMnO1xuZXhwb3J0ICogZnJvbSAnLi9wYXltZW50cyc7XG5leHBvcnQgKiBmcm9tICcuL3Bvc3RzJztcbmV4cG9ydCAqIGZyb20gJy4vcmVhY3Rpb25zJztcbmV4cG9ydCAqIGZyb20gJy4vczMnO1xuZXhwb3J0ICogZnJvbSAnLi9zZWFyY2gnO1xuZXhwb3J0ICogZnJvbSAnLi9zbXMnO1xuZXhwb3J0ICogZnJvbSAnLi9zdWJzY3JpcHRpb24nO1xuZXhwb3J0ICogZnJvbSAnLi90YWdzJztcbmV4cG9ydCAqIGZyb20gJy4vdXNlcnMnO1xuZXhwb3J0ICogZnJvbSAnLi93ZWJzb2NrZXRzJztcbiJdLAogICJtYXBwaW5ncyI6ICI7Ozs7Ozs7Ozs7Ozs7Ozs7OztBQUFBO0FBSUEsb0JBQWM7QUFDZCxvQkFBYztBQUNkLG9CQUFjO0FBQ2Qsb0JBQWM7QUFDZCxvQkFBYztBQUNkLG9CQUFjO0FBQ2Qsb0JBQWM7QUFDZCxvQkFBYztBQUNkLG9CQUFjO0FBQ2Qsb0JBQWM7QUFDZCxvQkFBYztBQUNkLG9CQUFjO0FBQ2Qsb0JBQWM7QUFDZCxvQkFBYztBQUNkLG9CQUFjO0FBQ2Qsb0JBQWM7QUFDZCxvQkFBYztBQUNkLG9CQUFjO0FBQ2Qsb0JBQWM7QUFDZCxvQkFBYzsiLAogICJuYW1lcyI6IFtdCn0K
@@ -1,6 +1,7 @@
1
+ import { Database } from 'arangojs';
1
2
  import { ApiContext, PaymentIOSInAppError, PaymentIOSSubscriptionUpdate, PaymentSubscription } from '../types';
2
3
  export declare const getIosInAppError: (statusCode: string) => PaymentIOSInAppError;
3
- export declare const iosSubscriptionUpdate: (database: string, subscriptionUpdate: PaymentIOSSubscriptionUpdate) => Promise<void>;
4
+ export declare const iosSubscriptionUpdate: (database: Database, subscriptionUpdate: PaymentIOSSubscriptionUpdate) => Promise<void>;
4
5
  export declare const getIosSubscription: (context: ApiContext) => Promise<PaymentSubscription>;
5
6
  export declare const addIosInApp: (context: ApiContext, subscription: PaymentSubscription) => Promise<PaymentSubscription>;
6
7
  export declare const getIosInApp: (context: ApiContext) => Promise<PaymentSubscription>;
@@ -0,0 +1,179 @@
1
+ var __create = Object.create;
2
+ var __defProp = Object.defineProperty;
3
+ var __defProps = Object.defineProperties;
4
+ var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
5
+ var __getOwnPropDescs = Object.getOwnPropertyDescriptors;
6
+ var __getOwnPropNames = Object.getOwnPropertyNames;
7
+ var __getOwnPropSymbols = Object.getOwnPropertySymbols;
8
+ var __getProtoOf = Object.getPrototypeOf;
9
+ var __hasOwnProp = Object.prototype.hasOwnProperty;
10
+ var __propIsEnum = Object.prototype.propertyIsEnumerable;
11
+ var __defNormalProp = (obj, key, value) => key in obj ? __defProp(obj, key, { enumerable: true, configurable: true, writable: true, value }) : obj[key] = value;
12
+ var __spreadValues = (a, b) => {
13
+ for (var prop in b || (b = {}))
14
+ if (__hasOwnProp.call(b, prop))
15
+ __defNormalProp(a, prop, b[prop]);
16
+ if (__getOwnPropSymbols)
17
+ for (var prop of __getOwnPropSymbols(b)) {
18
+ if (__propIsEnum.call(b, prop))
19
+ __defNormalProp(a, prop, b[prop]);
20
+ }
21
+ return a;
22
+ };
23
+ var __spreadProps = (a, b) => __defProps(a, __getOwnPropDescs(b));
24
+ var __markAsModule = (target) => __defProp(target, "__esModule", { value: true });
25
+ var __export = (target, all) => {
26
+ __markAsModule(target);
27
+ for (var name in all)
28
+ __defProp(target, name, { get: all[name], enumerable: true });
29
+ };
30
+ var __reExport = (target, module2, desc) => {
31
+ if (module2 && typeof module2 === "object" || typeof module2 === "function") {
32
+ for (let key of __getOwnPropNames(module2))
33
+ if (!__hasOwnProp.call(target, key) && key !== "default")
34
+ __defProp(target, key, { get: () => module2[key], enumerable: !(desc = __getOwnPropDesc(module2, key)) || desc.enumerable });
35
+ }
36
+ return target;
37
+ };
38
+ var __toModule = (module2) => {
39
+ return __reExport(__markAsModule(__defProp(module2 != null ? __create(__getProtoOf(module2)) : {}, "default", module2 && module2.__esModule && "default" in module2 ? { get: () => module2.default, enumerable: true } : { value: module2, enumerable: true })), module2);
40
+ };
41
+ __export(exports, {
42
+ addIosInApp: () => addIosInApp,
43
+ getIosInApp: () => getIosInApp,
44
+ getIosInAppError: () => getIosInAppError,
45
+ getIosSubscription: () => getIosSubscription,
46
+ iosSubscriptionUpdate: () => iosSubscriptionUpdate
47
+ });
48
+ var import_utils = __toModule(require("@nlabs/utils"));
49
+ var import_arangojs = __toModule(require("arangojs"));
50
+ var import_utils2 = __toModule(require("../utils"));
51
+ const eventCategory = "ios";
52
+ const getIosInAppError = (statusCode) => {
53
+ const codes = {
54
+ [0]: { error: false, message: "Active", valid: true },
55
+ [21e3]: { error: true, id: "payment_receipt_unreadable", message: "App store could not read", valid: false },
56
+ [21002]: { error: true, id: "payment_data_malformed", message: "Data was malformed", valid: false },
57
+ [21003]: { error: true, id: "payment_receipt_unauthorized", message: "Receipt not authenticated", valid: false },
58
+ [21004]: { error: true, id: "payment_invalid_secret", message: "Shared secret does not match", valid: false },
59
+ [21005]: { error: true, id: "payment_server_unavailable", message: "Receipt server unavailable", valid: false },
60
+ [21006]: {
61
+ error: false,
62
+ id: "payment_subscription_expired",
63
+ message: "Receipt valid but sub expired",
64
+ valid: true
65
+ },
66
+ [21007]: {
67
+ error: true,
68
+ id: "payment_invalid_server",
69
+ message: "Sandbox receipt sent to Production environment",
70
+ redirect: true,
71
+ valid: false
72
+ },
73
+ [21008]: {
74
+ error: true,
75
+ id: "payment_invalid_server",
76
+ message: "Production receipt sent to Sandbox environment",
77
+ valid: false
78
+ }
79
+ };
80
+ return codes[statusCode] || {};
81
+ };
82
+ const iosSubscriptionUpdate = (database, subscriptionUpdate) => {
83
+ const action = "iosSubscriptionUpdate";
84
+ const {
85
+ auto_renew_adam_id: planId,
86
+ cancellation_date: cancelDate,
87
+ notification_type: status,
88
+ latest_receipt: latestReceipt,
89
+ latest_receipt_info: latestInfo,
90
+ latest_expired_receipt: expiredReceipt,
91
+ latest_expired_receipt_info: expiredInfo,
92
+ original_transaction_id: transactionId
93
+ } = subscriptionUpdate;
94
+ const formatStatus = (status || "").toLowerCase();
95
+ const now = Date.now();
96
+ const receiptInfo = latestInfo || expiredInfo;
97
+ const id = (0, import_utils.createHash)(`payment-${transactionId}`);
98
+ const update = {
99
+ cancelDate,
100
+ isValid: receiptInfo[0].expires_date_ms > now,
101
+ modified: now,
102
+ planId,
103
+ receipt: latestReceipt || expiredReceipt,
104
+ status: formatStatus
105
+ };
106
+ const insert = __spreadProps(__spreadValues({}, update), {
107
+ _key: id,
108
+ added: now,
109
+ type: "ios_payment"
110
+ });
111
+ const aqlQry = import_arangojs.aql`UPSERT {transactionId: ${transactionId}}
112
+ INSERT ${insert}
113
+ UPDATE ${update} IN subscriptions
114
+ LIMIT 1`;
115
+ return database.query(aqlQry).then(() => null).catch((dbError) => (0, import_utils2.logError)({
116
+ action,
117
+ category: eventCategory,
118
+ label: "db_error"
119
+ }, dbError, {}).then(() => null));
120
+ };
121
+ const getIosSubscription = (context) => {
122
+ const action = "getIosSubscription";
123
+ const { database, session: { userId: sessionId } } = context;
124
+ const aqlQry = import_arangojs.aql`FOR s IN subscriptions
125
+ FILTER s.userId == ${sessionId} && s.type == "ios_payment"
126
+ LIMIT 1
127
+ RETURN s`;
128
+ return database.query(aqlQry).then((cursor) => cursor.next()).then((subscription = {}) => subscription).catch((error) => (0, import_utils2.logError)({
129
+ action,
130
+ category: eventCategory,
131
+ label: "db_error"
132
+ }, error, context).then(() => null));
133
+ };
134
+ const addIosInApp = (context, subscription) => {
135
+ const action = "addIosInApp";
136
+ const { database, session: { userId: sessionId } } = context;
137
+ const now = Date.now();
138
+ const { added = now, planId, receipt, transactionId, trialEnd } = subscription;
139
+ const subscriptionId = (0, import_utils.createHash)(`subscription-${transactionId}`);
140
+ const insert = {
141
+ _key: subscriptionId,
142
+ added,
143
+ modified: now,
144
+ planId,
145
+ receipt,
146
+ transactionId,
147
+ trialEnd,
148
+ type: "ios_subscription",
149
+ userId: sessionId
150
+ };
151
+ const aqlQry = import_arangojs.aql`INSERT ${insert} IN subscriptions RETURN NEW`;
152
+ return database.query(aqlQry).then((cursor) => cursor.next()).then((updatedSubscription = {}) => updatedSubscription).catch((error) => (0, import_utils2.logError)({
153
+ action,
154
+ category: eventCategory,
155
+ label: "db_error"
156
+ }, error, context).then(() => null));
157
+ };
158
+ const getIosInApp = (context) => {
159
+ const action = "getIosInApp";
160
+ const { database, session: { userId: sessionId } } = context;
161
+ const aqlQry = import_arangojs.aql`FOR s IN subscription
162
+ FILTER s.userId == ${sessionId} && s.type == "ios"
163
+ LIMIT 1
164
+ RETURN s`;
165
+ return database.query(aqlQry).then((cursor) => cursor.next()).then((subscription = {}) => subscription).catch((error) => (0, import_utils2.logError)({
166
+ action,
167
+ category: eventCategory,
168
+ label: "db_error"
169
+ }, error, context).then(() => null));
170
+ };
171
+ // Annotate the CommonJS export names for ESM import in node:
172
+ 0 && (module.exports = {
173
+ addIosInApp,
174
+ getIosInApp,
175
+ getIosInAppError,
176
+ getIosSubscription,
177
+ iosSubscriptionUpdate
178
+ });
179
+ //# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFsiLi4vLi4vc3JjL2FjdGlvbnMvaW9zLnRzIl0sCiAgInNvdXJjZXNDb250ZW50IjogWyJpbXBvcnQge2NyZWF0ZUhhc2h9IGZyb20gJ0BubGFicy91dGlscyc7XG5pbXBvcnQge2FxbCwgRGF0YWJhc2V9IGZyb20gJ2FyYW5nb2pzJztcbmltcG9ydCB7QXFsUXVlcnl9IGZyb20gJ2FyYW5nb2pzL2FxbCc7XG5pbXBvcnQge0FycmF5Q3Vyc29yfSBmcm9tICdhcmFuZ29qcy9jdXJzb3InO1xuXG5pbXBvcnQge1xuICBBcGlDb250ZXh0LFxuICBQYXltZW50SU9TSW5BcHBFcnJvcixcbiAgUGF5bWVudElPU1N1YnNjcmlwdGlvbkluZm8sXG4gIFBheW1lbnRJT1NTdWJzY3JpcHRpb25VcGRhdGUsXG4gIFBheW1lbnRTdWJzY3JpcHRpb25cbn0gZnJvbSAnLi4vdHlwZXMnO1xuaW1wb3J0IHtsb2dFcnJvcn0gZnJvbSAnLi4vdXRpbHMnO1xuXG5jb25zdCBldmVudENhdGVnb3J5OiBzdHJpbmcgPSAnaW9zJztcblxuZXhwb3J0IGNvbnN0IGdldElvc0luQXBwRXJyb3IgPSAoc3RhdHVzQ29kZTogc3RyaW5nKTogUGF5bWVudElPU0luQXBwRXJyb3IgPT4ge1xuICBjb25zdCBjb2RlcyA9IHtcbiAgICBbMF06IHtlcnJvcjogZmFsc2UsIG1lc3NhZ2U6ICdBY3RpdmUnLCB2YWxpZDogdHJ1ZX0sXG4gICAgWzIxMDAwXToge2Vycm9yOiB0cnVlLCBpZDogJ3BheW1lbnRfcmVjZWlwdF91bnJlYWRhYmxlJywgbWVzc2FnZTogJ0FwcCBzdG9yZSBjb3VsZCBub3QgcmVhZCcsIHZhbGlkOiBmYWxzZX0sXG4gICAgWzIxMDAyXToge2Vycm9yOiB0cnVlLCBpZDogJ3BheW1lbnRfZGF0YV9tYWxmb3JtZWQnLCBtZXNzYWdlOiAnRGF0YSB3YXMgbWFsZm9ybWVkJywgdmFsaWQ6IGZhbHNlfSxcbiAgICBbMjEwMDNdOiB7ZXJyb3I6IHRydWUsIGlkOiAncGF5bWVudF9yZWNlaXB0X3VuYXV0aG9yaXplZCcsIG1lc3NhZ2U6ICdSZWNlaXB0IG5vdCBhdXRoZW50aWNhdGVkJywgdmFsaWQ6IGZhbHNlfSxcbiAgICBbMjEwMDRdOiB7ZXJyb3I6IHRydWUsIGlkOiAncGF5bWVudF9pbnZhbGlkX3NlY3JldCcsIG1lc3NhZ2U6ICdTaGFyZWQgc2VjcmV0IGRvZXMgbm90IG1hdGNoJywgdmFsaWQ6IGZhbHNlfSxcbiAgICBbMjEwMDVdOiB7ZXJyb3I6IHRydWUsIGlkOiAncGF5bWVudF9zZXJ2ZXJfdW5hdmFpbGFibGUnLCBtZXNzYWdlOiAnUmVjZWlwdCBzZXJ2ZXIgdW5hdmFpbGFibGUnLCB2YWxpZDogZmFsc2V9LFxuICAgIFsyMTAwNl06IHtcbiAgICAgIGVycm9yOiBmYWxzZSxcbiAgICAgIGlkOiAncGF5bWVudF9zdWJzY3JpcHRpb25fZXhwaXJlZCcsXG4gICAgICBtZXNzYWdlOiAnUmVjZWlwdCB2YWxpZCBidXQgc3ViIGV4cGlyZWQnLFxuICAgICAgdmFsaWQ6IHRydWVcbiAgICB9LFxuICAgIC8qKlxuICAgICAqIHNwZWNpYWwgY2FzZSBmb3IgYXBwIHJldmlldyBoYW5kbGluZyAtIGZvcndhcmQgYW55IHJlcXVlc3QgdGhhdCBpcyBpbnRlbmRlZCBmb3IgdGhlIFNhbmRib3ggYnV0IHdhcyBzZW50IHRvXG4gICAgICogUHJvZHVjdGlvbiwgdGhpcyBpcyB3aGF0IHRoZSBhcHAgcmV2aWV3IHRlYW0gZG9lc1xuICAgICAqL1xuICAgIFsyMTAwN106IHtcbiAgICAgIGVycm9yOiB0cnVlLFxuICAgICAgaWQ6ICdwYXltZW50X2ludmFsaWRfc2VydmVyJyxcbiAgICAgIG1lc3NhZ2U6ICdTYW5kYm94IHJlY2VpcHQgc2VudCB0byBQcm9kdWN0aW9uIGVudmlyb25tZW50JyxcbiAgICAgIHJlZGlyZWN0OiB0cnVlLFxuICAgICAgdmFsaWQ6IGZhbHNlXG4gICAgfSxcbiAgICBbMjEwMDhdOiB7XG4gICAgICBlcnJvcjogdHJ1ZSxcbiAgICAgIGlkOiAncGF5bWVudF9pbnZhbGlkX3NlcnZlcicsXG4gICAgICBtZXNzYWdlOiAnUHJvZHVjdGlvbiByZWNlaXB0IHNlbnQgdG8gU2FuZGJveCBlbnZpcm9ubWVudCcsXG4gICAgICB2YWxpZDogZmFsc2VcbiAgICB9XG4gIH07XG5cbiAgcmV0dXJuIGNvZGVzW3N0YXR1c0NvZGVdIHx8IHt9O1xufTtcblxuZXhwb3J0IGNvbnN0IGlvc1N1YnNjcmlwdGlvblVwZGF0ZSA9IChcbiAgZGF0YWJhc2U6IERhdGFiYXNlLFxuICBzdWJzY3JpcHRpb25VcGRhdGU6IFBheW1lbnRJT1NTdWJzY3JpcHRpb25VcGRhdGVcbik6IFByb21pc2U8dm9pZD4gPT4ge1xuICBjb25zdCBhY3Rpb246IHN0cmluZyA9ICdpb3NTdWJzY3JpcHRpb25VcGRhdGUnO1xuICBjb25zdCB7XG4gICAgYXV0b19yZW5ld19hZGFtX2lkOiBwbGFuSWQsXG4gICAgY2FuY2VsbGF0aW9uX2RhdGU6IGNhbmNlbERhdGUsXG4gICAgbm90aWZpY2F0aW9uX3R5cGU6IHN0YXR1cyxcbiAgICBsYXRlc3RfcmVjZWlwdDogbGF0ZXN0UmVjZWlwdCxcbiAgICBsYXRlc3RfcmVjZWlwdF9pbmZvOiBsYXRlc3RJbmZvLFxuICAgIGxhdGVzdF9leHBpcmVkX3JlY2VpcHQ6IGV4cGlyZWRSZWNlaXB0LFxuICAgIGxhdGVzdF9leHBpcmVkX3JlY2VpcHRfaW5mbzogZXhwaXJlZEluZm8sXG4gICAgb3JpZ2luYWxfdHJhbnNhY3Rpb25faWQ6IHRyYW5zYWN0aW9uSWRcbiAgfSA9IHN1YnNjcmlwdGlvblVwZGF0ZTtcbiAgY29uc3QgZm9ybWF0U3RhdHVzOiBzdHJpbmcgPSAoc3RhdHVzIHx8ICcnKS50b0xvd2VyQ2FzZSgpO1xuICBjb25zdCBub3c6IG51bWJlciA9IERhdGUubm93KCk7XG4gIGNvbnN0IHJlY2VpcHRJbmZvOiBQYXltZW50SU9TU3Vic2NyaXB0aW9uSW5mb1tdID0gbGF0ZXN0SW5mbyB8fCBleHBpcmVkSW5mbztcbiAgY29uc3QgaWQ6IHN0cmluZyA9IGNyZWF0ZUhhc2goYHBheW1lbnQtJHt0cmFuc2FjdGlvbklkfWApO1xuXG4gIGNvbnN0IHVwZGF0ZTogUGF5bWVudFN1YnNjcmlwdGlvbiA9IHtcbiAgICBjYW5jZWxEYXRlLFxuICAgIGlzVmFsaWQ6IHJlY2VpcHRJbmZvWzBdLmV4cGlyZXNfZGF0ZV9tcyA+IG5vdyxcbiAgICBtb2RpZmllZDogbm93LFxuICAgIHBsYW5JZCxcbiAgICByZWNlaXB0OiBsYXRlc3RSZWNlaXB0IHx8IGV4cGlyZWRSZWNlaXB0LFxuICAgIHN0YXR1czogZm9ybWF0U3RhdHVzXG4gIH07XG4gIGNvbnN0IGluc2VydDogUGF5bWVudFN1YnNjcmlwdGlvbiA9IHtcbiAgICAuLi51cGRhdGUsXG4gICAgX2tleTogaWQsXG4gICAgYWRkZWQ6IG5vdyxcbiAgICB0eXBlOiAnaW9zX3BheW1lbnQnXG4gIH07XG4gIGNvbnN0IGFxbFFyeTogQXFsUXVlcnkgPSBhcWxgVVBTRVJUIHt0cmFuc2FjdGlvbklkOiAke3RyYW5zYWN0aW9uSWR9fVxuICAgICBJTlNFUlQgJHtpbnNlcnR9XG4gICAgIFVQREFURSAke3VwZGF0ZX0gSU4gc3Vic2NyaXB0aW9uc1xuICAgICBMSU1JVCAxYDtcblxuICByZXR1cm4gZGF0YWJhc2UucXVlcnkoYXFsUXJ5KVxuICAgIC50aGVuKCgpID0+IG51bGwpXG4gICAgLmNhdGNoKChkYkVycm9yOiBFcnJvcikgPT4gbG9nRXJyb3Ioe1xuICAgICAgYWN0aW9uLFxuICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICBsYWJlbDogJ2RiX2Vycm9yJ1xuICAgIH0sIGRiRXJyb3IsIHt9KS50aGVuKCgpID0+IG51bGwpKTtcbn07XG5cbi8vIGV4cG9ydCBjb25zdCBhZGRJb3NJbkFwcCA9IChcbi8vICAgY29udGV4dDogQXBpQ29udGV4dCxcbi8vICAgYXBwLFxuLy8gICBzdWJzY3JpcHRpb246IFBheW1lbnRTdWJzY3JpcHRpb25cbi8vICk6IFByb21pc2U8UGF5bWVudFN1YnNjcmlwdGlvbj4gPT4ge1xuLy8gICBjb25zdCBhY3Rpb246IHN0cmluZyA9ICdhZGRJb3NJbkFwcCc7XG4vLyAgIGNvbnN0IHtpc1Byb2R1Y3Rpb24sIHJlY2VpcHR9ID0gc3Vic2NyaXB0aW9uO1xuLy8gICBjb25zdCB7ZGF0YWJhc2UsIHVzZXJJZDogc2Vzc2lvbklkfSA9IGNvbnRleHQ7XG5cbi8vICAgY29uc3QgcHJvZHVjdGlvbkhvc3Q6IHN0cmluZyA9ICdidXkuaXR1bmVzLmFwcGxlLmNvbSc7XG4vLyAgIGNvbnN0IHNhbmRib3hIb3N0OiBzdHJpbmcgPSAnc2FuZGJveC5pdHVuZXMuYXBwbGUuY29tJztcbi8vICAgY29uc3QgZW5kcG9pbnQ6IHN0cmluZyA9IGlzUHJvZHVjdGlvbiA/IHByb2R1Y3Rpb25Ib3N0IDogc2FuZGJveEhvc3Q7XG4vLyAgIGNvbnN0IHZlcmlmeVVybDogc3RyaW5nID0gYGh0dHBzOi8vJHtlbmRwb2ludH0vdmVyaWZ5UmVjZWlwdGA7XG4vLyAgIGNvbnN0IHBheWxvYWQ6IG9iamVjdCA9IHtcbi8vICAgICBwYXNzd29yZDogYXBwLmlhcCxcbi8vICAgICAncmVjZWlwdC1kYXRhJzogZGVjb2RlVVJJQ29tcG9uZW50KHJlY2VpcHQpXG4vLyAgIH07XG4vLyAgIGNvbnN0IG9wdGlvbnM6IG9iamVjdCA9IHtcbi8vICAgICBoZWFkZXJzOiB7XG4vLyAgICAgICAnQ29udGVudC1UeXBlJzogJ2FwcGxpY2F0aW9uL3gtd3d3LWZvcm0tdXJsZW5jb2RlZCdcbi8vICAgICB9XG4vLyAgIH07XG5cbi8vICAgcmV0dXJuIHBvc3QodmVyaWZ5VXJsLCBwYXlsb2FkLCBvcHRpb25zKVxuLy8gICAgIC50aGVuKChyZXN1bHRzKSA9PiB7XG4vLyAgICAgICBjb25zdCBqc29uID0gSlNPTi5wYXJzZShyZXN1bHRzKTtcbi8vICAgICAgIGNvbnN0IHtcbi8vICAgICAgICAgbGF0ZXN0X3JlY2VpcHQ6IGxhdGVzdFJlY2VpcHQsXG4vLyAgICAgICAgIGxhdGVzdF9yZWNlaXB0X2luZm86IGxhdGVzdEluZm8sXG4vLyAgICAgICAgIHN0YXR1c1xuLy8gICAgICAgfSA9IGpzb247XG5cbi8vICAgICAgIGNvbnN0IHtcbi8vICAgICAgICAgaXNfdHJpYWxfcGVyaW9kOiB0cmlhbFBlcmlvZCxcbi8vICAgICAgICAgb3JpZ2luYWxfcHVyY2hhc2VfZGF0ZV9tczogYWRkZWQsXG4vLyAgICAgICAgIGV4cGlyZXNfZGF0ZTogZXhwaXJlcyxcbi8vICAgICAgICAgcHJvZHVjdF9pZDogcGxhbklkLFxuLy8gICAgICAgICB0cmFuc2FjdGlvbl9pZDogdHJhbnNhY3Rpb25JZFxuLy8gICAgICAgfSA9IGxhdGVzdEluZm87XG5cbi8vICAgICAgIGlmKHN0YXR1cyA9PT0gMCB8fCBzdGF0dXMgPT09IDIxMDA2KSB7XG4vLyAgICAgICAgIGNvbnN0IGlkOiBzdHJpbmcgPSBjcmVhdGVIYXNoKGBwYXltZW50LSR7dHJhbnNhY3Rpb25JZH1gKTtcbi8vICAgICAgICAgY29uc3Qgbm93OiBudW1iZXIgPSBEYXRlLm5vdygpO1xuLy8gICAgICAgICBjb25zdCB1cGRhdGU6IFBheW1lbnRTdWJzY3JpcHRpb24gPSB7XG4vLyAgICAgICAgICAgZXhwaXJlcyxcbi8vICAgICAgICAgICBpc1RyaWFsOiB0cmlhbFBlcmlvZCA9PT0gJ3RydWUnLFxuLy8gICAgICAgICAgIGlzVmFsaWQ6IGV4cGlyZXMgPiBub3csXG4vLyAgICAgICAgICAgbW9kaWZpZWQ6IG5vdyxcbi8vICAgICAgICAgICBwbGFuSWQsXG4vLyAgICAgICAgICAgcmVjZWlwdDogbGF0ZXN0UmVjZWlwdCxcbi8vICAgICAgICAgICB0cmFuc2FjdGlvbklkLFxuLy8gICAgICAgICAgIHVzZXJJZDogc2Vzc2lvbklkXG4vLyAgICAgICAgIH07XG4vLyAgICAgICAgIGNvbnN0IGluc2VydDogUGF5bWVudFN1YnNjcmlwdGlvbiA9IHtcbi8vICAgICAgICAgICAuLi51cGRhdGUsXG4vLyAgICAgICAgICAgX2tleTogaWQsXG4vLyAgICAgICAgICAgYWRkZWQsXG4vLyAgICAgICAgICAgdHlwZTogJ2lvc19wYXltZW50J1xuLy8gICAgICAgICB9O1xuXG4vLyAgICAgICAgIGNvbnN0IGFxbFFyeTogQXFsUXVlcnkgPSBhcWxgVVBTRVJUIHt0cmFuc2FjdGlvbklkOiAke3RyYW5zYWN0aW9uSWR9fVxuLy8gICAgICAgICAgICAgICBJTlNFUlQgJHtpbnNlcnR9XG4vLyAgICAgICAgICAgICAgIFVQREFURSAke3VwZGF0ZX0gSU4gc3Vic2NyaXB0aW9uc1xuLy8gICAgICAgICAgICAgICBSRVRVUk4gTkVXYDtcblxuLy8gICAgICAgICByZXR1cm4gZGF0YWJhc2UucXVlcnkoYXFsUXJ5KVxuLy8gICAgICAgICAgIC50aGVuKChjdXJzb3I6IEFycmF5Q3Vyc29yKSA9PiBjdXJzb3IubmV4dCgpKVxuLy8gICAgICAgICAgIC50aGVuKCh1cGRhdGVkU3Vic2NyaXB0aW9uOiBQYXltZW50U3Vic2NyaXB0aW9uID0ge30pID0+IHVwZGF0ZWRTdWJzY3JpcHRpb24pXG4vLyAgICAgICAgICAgLmNhdGNoKChkYkVycm9yOiBFcnJvcikgPT4gbG9nRXJyb3Ioe1xuLy8gICAgICAgICAgICAgYWN0aW9uLFxuLy8gICAgICAgICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4vLyAgICAgICAgICAgICBsYWJlbDogJ2RiX2Vycm9yJ1xuLy8gICAgICAgICAgIH0sIGRiRXJyb3IsIGNvbnRleHQpLnRoZW4oKCkgPT4gbnVsbCkpO1xuLy8gICAgICAgfVxuXG4vLyAgICAgICAvLyBFcnJvcnNcbi8vICAgICAgIGNvbnN0IGVycm9yOiBQYXltZW50SU9TSW5BcHBFcnJvciA9IGdldElvc0luQXBwRXJyb3IocmVzdWx0cy5zdGF0dXMpO1xuXG4vLyAgICAgICByZXR1cm4gbG9nRXhjZXB0aW9uKHtcbi8vICAgICAgICAgYWN0aW9uLFxuLy8gICAgICAgICBjYXRlZ29yeTogZXZlbnRDYXRlZ29yeSxcbi8vICAgICAgICAgbGFiZWw6ICdwYXltZW50X2Vycm9yJyxcbi8vICAgICAgICAgbWVzc2FnZTogZXJyb3IubWVzc2FnZSxcbi8vICAgICAgICAgdmFsdWU6IGVycm9yLmlkXG4vLyAgICAgICB9LCBjb250ZXh0KS50aGVuKCgpID0+IG51bGwpO1xuLy8gICAgIH0pO1xuLy8gfTtcblxuZXhwb3J0IGNvbnN0IGdldElvc1N1YnNjcmlwdGlvbiA9IChjb250ZXh0OiBBcGlDb250ZXh0KTogUHJvbWlzZTxQYXltZW50U3Vic2NyaXB0aW9uPiA9PiB7XG4gIGNvbnN0IGFjdGlvbjogc3RyaW5nID0gJ2dldElvc1N1YnNjcmlwdGlvbic7XG4gIGNvbnN0IHtkYXRhYmFzZSwgc2Vzc2lvbjoge3VzZXJJZDogc2Vzc2lvbklkfX0gPSBjb250ZXh0O1xuICBjb25zdCBhcWxRcnk6IEFxbFF1ZXJ5ID0gYXFsYEZPUiBzIElOIHN1YnNjcmlwdGlvbnNcbiAgICAgIEZJTFRFUiBzLnVzZXJJZCA9PSAke3Nlc3Npb25JZH0gJiYgcy50eXBlID09IFwiaW9zX3BheW1lbnRcIlxuICAgICAgTElNSVQgMVxuICAgICAgUkVUVVJOIHNgO1xuXG4gIHJldHVybiBkYXRhYmFzZS5xdWVyeShhcWxRcnkpXG4gICAgLnRoZW4oKGN1cnNvcjogQXJyYXlDdXJzb3IpID0+IGN1cnNvci5uZXh0KCkpXG4gICAgLnRoZW4oKHN1YnNjcmlwdGlvbjogUGF5bWVudFN1YnNjcmlwdGlvbiA9IHt9KSA9PiBzdWJzY3JpcHRpb24pXG4gICAgLmNhdGNoKChlcnJvcjogRXJyb3IpID0+IGxvZ0Vycm9yKHtcbiAgICAgIGFjdGlvbixcbiAgICAgIGNhdGVnb3J5OiBldmVudENhdGVnb3J5LFxuICAgICAgbGFiZWw6ICdkYl9lcnJvcidcbiAgICB9LCBlcnJvciwgY29udGV4dCkudGhlbigoKSA9PiBudWxsKSk7XG59O1xuXG5leHBvcnQgY29uc3QgYWRkSW9zSW5BcHAgPSAoY29udGV4dDogQXBpQ29udGV4dCwgc3Vic2NyaXB0aW9uOiBQYXltZW50U3Vic2NyaXB0aW9uKTogUHJvbWlzZTxQYXltZW50U3Vic2NyaXB0aW9uPiA9PiB7XG4gIGNvbnN0IGFjdGlvbjogc3RyaW5nID0gJ2FkZElvc0luQXBwJztcbiAgY29uc3Qge2RhdGFiYXNlLCBzZXNzaW9uOiB7dXNlcklkOiBzZXNzaW9uSWR9fSA9IGNvbnRleHQ7XG4gIGNvbnN0IG5vdzogbnVtYmVyID0gRGF0ZS5ub3coKTtcbiAgY29uc3Qge2FkZGVkID0gbm93LCBwbGFuSWQsIHJlY2VpcHQsIHRyYW5zYWN0aW9uSWQsIHRyaWFsRW5kfSA9IHN1YnNjcmlwdGlvbjtcbiAgY29uc3Qgc3Vic2NyaXB0aW9uSWQ6IHN0cmluZyA9IGNyZWF0ZUhhc2goYHN1YnNjcmlwdGlvbi0ke3RyYW5zYWN0aW9uSWR9YCk7XG4gIGNvbnN0IGluc2VydDogUGF5bWVudFN1YnNjcmlwdGlvbiA9IHtcbiAgICBfa2V5OiBzdWJzY3JpcHRpb25JZCxcbiAgICBhZGRlZCxcbiAgICBtb2RpZmllZDogbm93LFxuICAgIHBsYW5JZCxcbiAgICByZWNlaXB0LFxuICAgIHRyYW5zYWN0aW9uSWQsXG4gICAgdHJpYWxFbmQsXG4gICAgdHlwZTogJ2lvc19zdWJzY3JpcHRpb24nLFxuICAgIHVzZXJJZDogc2Vzc2lvbklkXG4gIH07XG4gIGNvbnN0IGFxbFFyeTogQXFsUXVlcnkgPSBhcWxgSU5TRVJUICR7aW5zZXJ0fSBJTiBzdWJzY3JpcHRpb25zIFJFVFVSTiBORVdgO1xuXG4gIHJldHVybiBkYXRhYmFzZS5xdWVyeShhcWxRcnkpXG4gICAgLnRoZW4oKGN1cnNvcjogQXJyYXlDdXJzb3IpID0+IGN1cnNvci5uZXh0KCkpXG4gICAgLnRoZW4oKHVwZGF0ZWRTdWJzY3JpcHRpb246IFBheW1lbnRTdWJzY3JpcHRpb24gPSB7fSkgPT4gdXBkYXRlZFN1YnNjcmlwdGlvbilcbiAgICAuY2F0Y2goKGVycm9yOiBFcnJvcikgPT4gbG9nRXJyb3Ioe1xuICAgICAgYWN0aW9uLFxuICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICBsYWJlbDogJ2RiX2Vycm9yJ1xuICAgIH0sIGVycm9yLCBjb250ZXh0KS50aGVuKCgpID0+IG51bGwpKTtcbn07XG5cbmV4cG9ydCBjb25zdCBnZXRJb3NJbkFwcCA9IChjb250ZXh0OiBBcGlDb250ZXh0KTogUHJvbWlzZTxQYXltZW50U3Vic2NyaXB0aW9uPiA9PiB7XG4gIGNvbnN0IGFjdGlvbjogc3RyaW5nID0gJ2dldElvc0luQXBwJztcbiAgY29uc3Qge2RhdGFiYXNlLCBzZXNzaW9uOiB7dXNlcklkOiBzZXNzaW9uSWR9fSA9IGNvbnRleHQ7XG4gIGNvbnN0IGFxbFFyeTogQXFsUXVlcnkgPSBhcWxgRk9SIHMgSU4gc3Vic2NyaXB0aW9uXG4gICAgICBGSUxURVIgcy51c2VySWQgPT0gJHtzZXNzaW9uSWR9ICYmIHMudHlwZSA9PSBcImlvc1wiXG4gICAgICBMSU1JVCAxXG4gICAgICBSRVRVUk4gc2A7XG5cbiAgcmV0dXJuIGRhdGFiYXNlLnF1ZXJ5KGFxbFFyeSlcbiAgICAudGhlbigoY3Vyc29yOiBBcnJheUN1cnNvcikgPT4gY3Vyc29yLm5leHQoKSlcbiAgICAudGhlbigoc3Vic2NyaXB0aW9uOiBQYXltZW50U3Vic2NyaXB0aW9uID0ge30pID0+IHN1YnNjcmlwdGlvbilcbiAgICAuY2F0Y2goKGVycm9yOiBFcnJvcikgPT4gbG9nRXJyb3Ioe1xuICAgICAgYWN0aW9uLFxuICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICBsYWJlbDogJ2RiX2Vycm9yJ1xuICAgIH0sIGVycm9yLCBjb250ZXh0KS50aGVuKCgpID0+IG51bGwpKTtcbn07XG4iXSwKICAibWFwcGluZ3MiOiAiOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQSxtQkFBeUI7QUFDekIsc0JBQTRCO0FBVzVCLG9CQUF1QjtBQUV2QixNQUFNLGdCQUF3QjtBQUV2QixNQUFNLG1CQUFtQixDQUFDLGVBQTZDO0FBQzVFLFFBQU0sUUFBUTtBQUFBLEtBQ1gsSUFBSSxF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
@@ -0,0 +1,112 @@
1
+ var __create = Object.create;
2
+ var __defProp = Object.defineProperty;
3
+ var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
4
+ var __getOwnPropNames = Object.getOwnPropertyNames;
5
+ var __getProtoOf = Object.getPrototypeOf;
6
+ var __hasOwnProp = Object.prototype.hasOwnProperty;
7
+ var __markAsModule = (target) => __defProp(target, "__esModule", { value: true });
8
+ var __export = (target, all) => {
9
+ __markAsModule(target);
10
+ for (var name in all)
11
+ __defProp(target, name, { get: all[name], enumerable: true });
12
+ };
13
+ var __reExport = (target, module2, desc) => {
14
+ if (module2 && typeof module2 === "object" || typeof module2 === "function") {
15
+ for (let key of __getOwnPropNames(module2))
16
+ if (!__hasOwnProp.call(target, key) && key !== "default")
17
+ __defProp(target, key, { get: () => module2[key], enumerable: !(desc = __getOwnPropDesc(module2, key)) || desc.enumerable });
18
+ }
19
+ return target;
20
+ };
21
+ var __toModule = (module2) => {
22
+ return __reExport(__markAsModule(__defProp(module2 != null ? __create(__getProtoOf(module2)) : {}, "default", module2 && module2.__esModule && "default" in module2 ? { get: () => module2.default, enumerable: true } : { value: module2, enumerable: true })), module2);
23
+ };
24
+ __export(exports, {
25
+ addLocation: () => addLocation
26
+ });
27
+ var import_rip_hunter = __toModule(require("@nlabs/rip-hunter"));
28
+ var import_utils = __toModule(require("@nlabs/utils"));
29
+ var import_arangojs = __toModule(require("arangojs"));
30
+ var import_get = __toModule(require("lodash/get"));
31
+ var import_config = __toModule(require("../config"));
32
+ var import_utils2 = __toModule(require("../utils"));
33
+ const eventCategory = "locations";
34
+ const addLocation = (context, location) => {
35
+ const action = "add";
36
+ const { database, session: { userId: sessionId } } = context;
37
+ const { address, itemId, itemType } = location;
38
+ const geocodeUrl = import_config.Config.get("google.geocode.url");
39
+ const params = {
40
+ address,
41
+ key: import_config.Config.get("google.geocode.key")
42
+ };
43
+ const url = `${geocodeUrl}?${(0, import_rip_hunter.queryString)(params)}`;
44
+ return (0, import_rip_hunter.get)(url).then((json) => {
45
+ const { results = [], status = "" } = json;
46
+ if (status === "OK" && results.length) {
47
+ const geodata = results[0];
48
+ const addressComponents = (0, import_get.default)(geodata, "address_components");
49
+ const getFieldVal = (name) => {
50
+ const { short_name: shortName = "" } = addressComponents.find((addressField) => {
51
+ const types = addressField.types || [];
52
+ return types.findIndex((fieldType) => fieldType === name) >= 0;
53
+ }) || {};
54
+ return shortName;
55
+ };
56
+ const streetNumber = getFieldVal("street_number");
57
+ const route = getFieldVal("route");
58
+ const now = Date.now();
59
+ const formatItemType = (0, import_utils.parseChar)(itemType).toLowerCase();
60
+ const formatItemId = (0, import_utils.parseId)(itemId);
61
+ const googleId = (0, import_get.default)(geodata, "place_id");
62
+ const update = {};
63
+ const insert = {
64
+ _key: (0, import_utils.createHash)(`post-${sessionId}`),
65
+ added: now,
66
+ city: getFieldVal("locality"),
67
+ country: getFieldVal("country"),
68
+ formatted: (0, import_get.default)(geodata, "formatted_address"),
69
+ googleId,
70
+ latitude: (0, import_get.default)(geodata, "geometry.location.lat"),
71
+ longitude: (0, import_get.default)(geodata, "geometry.location.lng"),
72
+ state: getFieldVal("administrative_area_level_1"),
73
+ street: [streetNumber, route].filter((obj) => obj).join(" "),
74
+ zip: getFieldVal("postal_code")
75
+ };
76
+ const aqlQry = import_arangojs.aql`UPSERT {googleId: ${googleId}}
77
+ INSERT ${insert}
78
+ UPDATE ${update}
79
+ IN locations RETURN NEW`;
80
+ return database.query(aqlQry).then((cursor) => cursor.next()).then((updatedLocation = {}) => {
81
+ let itemDocId;
82
+ const { _key: updatedLocationKey } = updatedLocation;
83
+ switch (formatItemType) {
84
+ case "posts":
85
+ itemDocId = `posts/${formatItemId}`;
86
+ break;
87
+ default:
88
+ itemDocId = "";
89
+ }
90
+ if (itemDocId) {
91
+ const edgeCollection = database.collection("hasLocation");
92
+ const locationId = updatedLocationKey;
93
+ const locationDocId = `locations/${locationId}`;
94
+ const edgeId = (0, import_utils.createHash)(`hasLocation-${locationId}-${sessionId}`);
95
+ const edge = { _from: itemDocId, _key: edgeId, _to: locationDocId };
96
+ return edgeCollection.save(edge, { returnNew: true }).then(() => updatedLocation);
97
+ }
98
+ return updatedLocation;
99
+ }).catch((error) => (0, import_utils2.logError)({
100
+ action,
101
+ category: eventCategory,
102
+ label: "db_error"
103
+ }, error, context).then(() => null));
104
+ }
105
+ return {};
106
+ });
107
+ };
108
+ // Annotate the CommonJS export names for ESM import in node:
109
+ 0 && (module.exports = {
110
+ addLocation
111
+ });
112
+ //# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFsiLi4vLi4vc3JjL2FjdGlvbnMvbG9jYXRpb25zLnRzIl0sCiAgInNvdXJjZXNDb250ZW50IjogWyIvKipcbiAqIENvcHlyaWdodCAoYykgMjAxOS1QcmVzZW50LCBOaXRyb2dlbiBMYWJzLCBJbmMuXG4gKiBDb3B5cmlnaHRzIGxpY2Vuc2VkIHVuZGVyIHRoZSBNSVQgTGljZW5zZS4gU2VlIHRoZSBhY2NvbXBhbnlpbmcgTElDRU5TRSBmaWxlIGZvciB0ZXJtcy5cbiAqL1xuaW1wb3J0IHtnZXQgYXMgaHR0cEdldCwgcXVlcnlTdHJpbmd9IGZyb20gJ0BubGFicy9yaXAtaHVudGVyJztcbmltcG9ydCB7Y3JlYXRlSGFzaCwgcGFyc2VDaGFyLCBwYXJzZUlkfSBmcm9tICdAbmxhYnMvdXRpbHMnO1xuaW1wb3J0IHthcWx9IGZyb20gJ2FyYW5nb2pzJztcbmltcG9ydCB7QXFsUXVlcnl9IGZyb20gJ2FyYW5nb2pzL2FxbCc7XG5pbXBvcnQge0VkZ2VDb2xsZWN0aW9ufSBmcm9tICdhcmFuZ29qcy9jb2xsZWN0aW9uJztcbmltcG9ydCB7QXJyYXlDdXJzb3J9IGZyb20gJ2FyYW5nb2pzL2N1cnNvcic7XG5pbXBvcnQgZ2V0IGZyb20gJ2xvZGFzaC9nZXQnO1xuXG5pbXBvcnQge0NvbmZpZ30gZnJvbSAnLi4vY29uZmlnJztcbmltcG9ydCB7QXBpQ29udGV4dH0gZnJvbSAnLi4vdHlwZXMvYXV0aCc7XG5pbXBvcnQge0dvb2dsZU1hcHNBZGRyZXNzVHlwZX0gZnJvbSAnLi4vdHlwZXMvZ29vZ2xlJztcbmltcG9ydCB7TG9jYXRpb25UeXBlfSBmcm9tICcuLi90eXBlcy9sb2NhdGlvbnMnO1xuaW1wb3J0IHtsb2dFcnJvcn0gZnJvbSAnLi4vdXRpbHMnO1xuXG5jb25zdCBldmVudENhdGVnb3J5OiBzdHJpbmcgPSAnbG9jYXRpb25zJztcblxuZXhwb3J0IGNvbnN0IGFkZExvY2F0aW9uID0gKGNvbnRleHQ6IEFwaUNvbnRleHQsIGxvY2F0aW9uOiBMb2NhdGlvblR5cGUpOiBQcm9taXNlPExvY2F0aW9uVHlwZT4gPT4ge1xuICBjb25zdCBhY3Rpb246IHN0cmluZyA9ICdhZGQnO1xuICBjb25zdCB7ZGF0YWJhc2UsIHNlc3Npb246IHt1c2VySWQ6IHNlc3Npb25JZH19ID0gY29udGV4dDtcbiAgY29uc3Qge2FkZHJlc3MsIGl0ZW1JZCwgaXRlbVR5cGV9ID0gbG9jYXRpb247XG4gIGNvbnN0IGdlb2NvZGVVcmw6IHN0cmluZyA9IENvbmZpZy5nZXQoJ2dvb2dsZS5nZW9jb2RlLnVybCcpO1xuICBjb25zdCBwYXJhbXM6IG9iamVjdCA9IHtcbiAgICBhZGRyZXNzLFxuICAgIGtleTogQ29uZmlnLmdldCgnZ29vZ2xlLmdlb2NvZGUua2V5JylcbiAgfTtcbiAgY29uc3QgdXJsOiBzdHJpbmcgPSBgJHtnZW9jb2RlVXJsfT8ke3F1ZXJ5U3RyaW5nKHBhcmFtcyl9YDtcblxuICByZXR1cm4gaHR0cEdldCh1cmwpXG4gICAgLnRoZW4oKGpzb24pID0+IHtcbiAgICAgIGNvbnN0IHtyZXN1bHRzID0gW10sIHN0YXR1cyA9ICcnfSA9IGpzb247XG5cbiAgICAgIGlmKHN0YXR1cyA9PT0gJ09LJyAmJiByZXN1bHRzLmxlbmd0aCkge1xuICAgICAgICBjb25zdCBnZW9kYXRhOiBvYmplY3QgPSByZXN1bHRzWzBdO1xuICAgICAgICBjb25zdCBhZGRyZXNzQ29tcG9uZW50czogR29vZ2xlTWFwc0FkZHJlc3NUeXBlW10gPSBnZXQoZ2VvZGF0YSwgJ2FkZHJlc3NfY29tcG9uZW50cycpO1xuICAgICAgICBjb25zdCBnZXRGaWVsZFZhbCA9IChuYW1lOiBzdHJpbmcpOiBzdHJpbmcgPT4ge1xuICAgICAgICAgIGNvbnN0IHtzaG9ydF9uYW1lOiBzaG9ydE5hbWUgPSAnJ30gPSBhZGRyZXNzQ29tcG9uZW50cy5maW5kKChhZGRyZXNzRmllbGQ6IEdvb2dsZU1hcHNBZGRyZXNzVHlwZSkgPT4ge1xuICAgICAgICAgICAgY29uc3QgdHlwZXMgPSBhZGRyZXNzRmllbGQudHlwZXMgfHwgW107XG4gICAgICAgICAgICByZXR1cm4gdHlwZXMuZmluZEluZGV4KChmaWVsZFR5cGU6IHN0cmluZykgPT4gZmllbGRUeXBlID09PSBuYW1lKSA+PSAwO1xuICAgICAgICAgIH0pIHx8IHt9O1xuXG4gICAgICAgICAgcmV0dXJuIHNob3J0TmFtZTtcbiAgICAgICAgfTtcblxuICAgICAgICAvLyBTdHJlZXRcbiAgICAgICAgY29uc3Qgc3RyZWV0TnVtYmVyOiBzdHJpbmcgPSBnZXRGaWVsZFZhbCgnc3RyZWV0X251bWJlcicpO1xuICAgICAgICBjb25zdCByb3V0ZTogc3RyaW5nID0gZ2V0RmllbGRWYWwoJ3JvdXRlJyk7XG5cbiAgICAgICAgLy8gTG9jYXRpb25cbiAgICAgICAgY29uc3Qgbm93OiBudW1iZXIgPSBEYXRlLm5vdygpO1xuXG4gICAgICAgIC8vIEl0ZW1cbiAgICAgICAgY29uc3QgZm9ybWF0SXRlbVR5cGU6IHN0cmluZyA9IHBhcnNlQ2hhcihpdGVtVHlwZSkudG9Mb3dlckNhc2UoKTtcbiAgICAgICAgY29uc3QgZm9ybWF0SXRlbUlkOiBzdHJpbmcgPSBwYXJzZUlkKGl0ZW1JZCk7XG4gICAgICAgIGNvbnN0IGdvb2dsZUlkOiBzdHJpbmcgPSBnZXQoZ2VvZGF0YSwgJ3BsYWNlX2lkJyk7XG4gICAgICAgIGNvbnN0IHVwZGF0ZTogTG9jYXRpb25UeXBlID0ge307XG4gICAgICAgIGNvbnN0IGluc2VydDogTG9jYXRpb25UeXBlID0ge1xuICAgICAgICAgIF9rZXk6IGNyZWF0ZUhhc2goYHBvc3QtJHtzZXNzaW9uSWR9YCksXG4gICAgICAgICAgYWRkZWQ6IG5vdyxcbiAgICAgICAgICBjaXR5OiBnZXRGaWVsZFZhbCgnbG9jYWxpdHknKSxcbiAgICAgICAgICBjb3VudHJ5OiBnZXRGaWVsZFZhbCgnY291bnRyeScpLFxuICAgICAgICAgIGZvcm1hdHRlZDogZ2V0KGdlb2RhdGEsICdmb3JtYXR0ZWRfYWRkcmVzcycpLFxuICAgICAgICAgIGdvb2dsZUlkLFxuICAgICAgICAgIGxhdGl0dWRlOiBnZXQoZ2VvZGF0YSwgJ2dlb21ldHJ5LmxvY2F0aW9uLmxhdCcpLFxuICAgICAgICAgIGxvbmdpdHVkZTogZ2V0KGdlb2RhdGEsICdnZW9tZXRyeS5sb2NhdGlvbi5sbmcnKSxcbiAgICAgICAgICBzdGF0ZTogZ2V0RmllbGRWYWwoJ2FkbWluaXN0cmF0aXZlX2FyZWFfbGV2ZWxfMScpLFxuICAgICAgICAgIHN0cmVldDogW3N0cmVldE51bWJlciwgcm91dGVdLmZpbHRlcigob2JqKSA9PiBvYmopLmpvaW4oJyAnKSxcbiAgICAgICAgICB6aXA6IGdldEZpZWxkVmFsKCdwb3N0YWxfY29kZScpXG4gICAgICAgIH07XG4gICAgICAgIGNvbnN0IGFxbFFyeTogQXFsUXVlcnkgPSBhcWxgVVBTRVJUIHtnb29nbGVJZDogJHtnb29nbGVJZH19XG4gICAgICAgICAgICBJTlNFUlQgJHtpbnNlcnR9XG4gICAgICAgICAgICBVUERBVEUgJHt1cGRhdGV9XG4gICAgICAgICAgICBJTiBsb2NhdGlvbnMgUkVUVVJOIE5FV2A7XG5cbiAgICAgICAgcmV0dXJuIGRhdGFiYXNlLnF1ZXJ5KGFxbFFyeSlcbiAgICAgICAgICAudGhlbigoY3Vyc29yOiBBcnJheUN1cnNvcikgPT4gY3Vyc29yLm5leHQoKSlcbiAgICAgICAgICAudGhlbigodXBkYXRlZExvY2F0aW9uOiBMb2NhdGlvblR5cGUgPSB7fSkgPT4ge1xuICAgICAgICAgICAgbGV0IGl0ZW1Eb2NJZDogc3RyaW5nO1xuICAgICAgICAgICAgY29uc3Qge19rZXk6IHVwZGF0ZWRMb2NhdGlvbktleX0gPSB1cGRhdGVkTG9jYXRpb247XG5cbiAgICAgICAgICAgIHN3aXRjaChmb3JtYXRJdGVtVHlwZSkge1xuICAgICAgICAgICAgICBjYXNlICdwb3N0cyc6XG4gICAgICAgICAgICAgICAgaXRlbURvY0lkID0gYHBvc3RzLyR7Zm9ybWF0SXRlbUlkfWA7XG4gICAgICAgICAgICAgICAgYnJlYWs7XG4gICAgICAgICAgICAgIGRlZmF1bHQ6XG4gICAgICAgICAgICAgICAgaXRlbURvY0lkID0gJyc7XG4gICAgICAgICAgICB9XG5cbiAgICAgICAgICAgIC8vIElmIHRoZXJlIGlzIGEgdmFsaWQgaXRlbSBkb2MgaWQsIGNyZWF0ZSBhbiBlZGdlXG4gICAgICAgICAgICBpZihpdGVtRG9jSWQpIHtcbiAgICAgICAgICAgICAgY29uc3QgZWRnZUNvbGxlY3Rpb246IEVkZ2VDb2xsZWN0aW9uID0gZGF0YWJhc2UuY29sbGVjdGlvbignaGFzTG9jYXRpb24nKTtcbiAgICAgICAgICAgICAgY29uc3QgbG9jYXRpb25JZDogc3RyaW5nID0gdXBkYXRlZExvY2F0aW9uS2V5O1xuICAgICAgICAgICAgICBjb25zdCBsb2NhdGlvbkRvY0lkOiBzdHJpbmcgPSBgbG9jYXRpb25zLyR7bG9jYXRpb25JZH1gO1xuICAgICAgICAgICAgICBjb25zdCBlZGdlSWQgPSBjcmVhdGVIYXNoKGBoYXNMb2NhdGlvbi0ke2xvY2F0aW9uSWR9LSR7c2Vzc2lvbklkfWApO1xuICAgICAgICAgICAgICBjb25zdCBlZGdlOiBhbnkgPSB7X2Zyb206IGl0ZW1Eb2NJZCwgX2tleTogZWRnZUlkLCBfdG86IGxvY2F0aW9uRG9jSWR9O1xuXG4gICAgICAgICAgICAgIHJldHVybiBlZGdlQ29sbGVjdGlvbi5zYXZlKGVkZ2UsIHtyZXR1cm5OZXc6IHRydWV9KS50aGVuKCgpID0+IHVwZGF0ZWRMb2NhdGlvbik7XG4gICAgICAgICAgICB9XG5cbiAgICAgICAgICAgIC8vIE90aGVyd2lzZSBqdXN0IHJldHVybiB0aGUgbmV3IGxvY2F0aW9uXG4gICAgICAgICAgICByZXR1cm4gdXBkYXRlZExvY2F0aW9uO1xuICAgICAgICAgIH0pXG4gICAgICAgICAgLmNhdGNoKChlcnJvcjogRXJyb3IpID0+IGxvZ0Vycm9yKHtcbiAgICAgICAgICAgIGFjdGlvbixcbiAgICAgICAgICAgIGNhdGVnb3J5OiBldmVudENhdGVnb3J5LFxuICAgICAgICAgICAgbGFiZWw6ICdkYl9lcnJvcidcbiAgICAgICAgICB9LCBlcnJvciwgY29udGV4dCkudGhlbigoKSA9PiBudWxsKSk7XG4gICAgICB9XG5cbiAgICAgIHJldHVybiB7fTtcbiAgICB9KTtcbn07XG4iXSwKICAibWFwcGluZ3MiOiAiOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztBQUFBO0FBQUE7QUFBQTtBQUlBLHdCQUEwQztBQUMxQyxtQkFBNkM7QUFDN0Msc0JBQWtCO0FBSWxCLGlCQUFnQjtBQUVoQixvQkFBcUI7QUFJckIsb0JBQXVCO0FBRXZCLE1BQU0sZ0JBQXdCO0FBRXZCLE1BQU0sY0FBYyxDQUFDLFNBQXFCLGFBQWtEO0FBQ2pHLFFBQU0sU0FBaUI7QUFDdkIsUUFBTSxFQUFDLFVBQVUsU0FBUyxFQUFDLFFBQVEsZ0JBQWM7QUFDakQsUUFBTSxFQUFDLFNBQVMsUUFBUSxhQUFZO0FBQ3BDLFFBQU0sYUFBcUIscUJBQU8sSUFBSTtBQUN0QyxRQUFNLFNBQWlCO0FBQUEsSUFDckI7QUFBQSxJQUNBLEtBQUsscUJBQU8sSUFBSTtBQUFBO0FBRWxCLFFBQU0sTUFBYyxHQUFHLGNBQWMsbUNBQVk7QUFFakQsU0FBTywyQkFBUSxLQUNaLEtBQUssQ0FBQyxTQUFTO0FBQ2QsVUFBTSxFQUFDLFVBQVUsSUFBSSxTQUFTLE9BQU07QUFFcEMsUUFBRyxXQUFXLFFBQVEsUUFBUSxRQUFRO0FBQ3BDLFlBQU0sVUFBa0IsUUFBUTtBQUNoQyxZQUFNLG9CQUE2Qyx3QkFBSSxTQUFTO0FBQ2hFLFlBQU0sY0FBYyxDQUFDLFNBQXlCO0FBQzVDLGNBQU0sRUFBQyxZQUFZLFlBQVksT0FBTSxrQkFBa0IsS0FBSyxDQUFDLGlCQUF3QztBQUNuRyxnQkFBTSxRQUFRLGFBQWEsU0FBUztBQUNwQyxpQkFBTyxNQUFNLFVBQVUsQ0FBQyxjQUFzQixjQUFjLFNBQVM7QUFBQSxjQUNqRTtBQUVOLGVBQU87QUFBQTtBQUlULFlBQU0sZUFBdUIsWUFBWTtBQUN6QyxZQUFNLFFBQWdCLFlBQVk7QUFHbEMsWUFBTSxNQUFjLEtBQUs7QUFHekIsWUFBTSxpQkFBeUIsNEJBQVUsVUFBVTtBQUNuRCxZQUFNLGVBQXVCLDBCQUFRO0FBQ3JDLFlBQU0sV0FBbUIsd0JBQUksU0FBUztBQUN0QyxZQUFNLFNBQXVCO0FBQzdCLFlBQU0sU0FBdUI7QUFBQSxRQUMzQixNQUFNLDZCQUFXLFFBQVE7QUFBQSxRQUN6QixPQUFPO0FBQUEsUUFDUCxNQUFNLFlBQVk7QUFBQSxRQUNsQixTQUFTLFlBQVk7QUFBQSxRQUNyQixXQUFXLHdCQUFJLFNBQVM7QUFBQSxRQUN4QjtBQUFBLFFBQ0EsVUFBVSx3QkFBSSxTQUFTO0FBQUEsUUFDdkIsV0FBVyx3QkFBSSxTQUFTO0FBQUEsUUFDeEIsT0FBTyxZQUFZO0FBQUEsUUFDbkIsUUFBUSxDQUFDLGNBQWMsT0FBTyxPQUFPLENBQUMsUUFBUSxLQUFLLEtBQUs7QUFBQSxRQUN4RCxLQUFLLFlBQVk7QUFBQTtBQUVuQixZQUFNLFNBQW1CLHdDQUF3QjtBQUFBLHFCQUNwQztBQUFBLHFCQUNBO0FBQUE7QUFHYixhQUFPLFNBQVMsTUFBTSxRQUNuQixLQUFLLENBQUMsV0FBd0IsT0FBTyxRQUNyQyxLQUFLLENBQUMsa0JBQWdDLE9BQU87QUFDNUMsWUFBSTtBQUNKLGNBQU0sRUFBQyxNQUFNLHVCQUFzQjtBQUVuQyxnQkFBTztBQUFBLGVBQ0E7QUFDSCx3QkFBWSxTQUFTO0FBQ3JCO0FBQUE7QUFFQSx3QkFBWTtBQUFBO0FBSWhCLFlBQUcsV0FBVztBQUNaLGdCQUFNLGlCQUFpQyxTQUFTLFdBQVc7QUFDM0QsZ0JBQU0sYUFBcUI7QUFDM0IsZ0JBQU0sZ0JBQXdCLGFBQWE7QUFDM0MsZ0JBQU0sU0FBUyw2QkFBVyxlQUFlLGNBQWM7QUFDdkQsZ0JBQU0sT0FBWSxFQUFDLE9BQU8sV0FBVyxNQUFNLFFBQVEsS0FBSztBQUV4RCxpQkFBTyxlQUFlLEtBQUssTUFBTSxFQUFDLFdBQVcsUUFBTyxLQUFLLE1BQU07QUFBQTtBQUlqRSxlQUFPO0FBQUEsU0FFUixNQUFNLENBQUMsVUFBaUIsNEJBQVM7QUFBQSxRQUNoQztBQUFBLFFBQ0EsVUFBVTtBQUFBLFFBQ1YsT0FBTztBQUFBLFNBQ04sT0FBTyxTQUFTLEtBQUssTUFBTTtBQUFBO0FBR2xDLFdBQU87QUFBQTtBQUFBOyIsCiAgIm5hbWVzIjogW10KfQo=
@@ -0,0 +1,13 @@
1
+ import { Database } from 'arangojs';
2
+ import { ApiContext } from '../types/auth';
3
+ import { MessageInputType, MessageType } from '../types/messages';
4
+ export declare const getMessages: (context: ApiContext, conversationId: string, { from, to }: {
5
+ from: any;
6
+ to: any;
7
+ }) => Promise<MessageType[]>;
8
+ export declare const updateMessage: (context: ApiContext, message: MessageInputType) => Promise<MessageType>;
9
+ export declare const saveMessage: (context: ApiContext, messageId: string, conversationId: string) => Promise<MessageType>;
10
+ export declare const unsaveMessage: (context: ApiContext, messageId: string, conversationId: string) => Promise<MessageType>;
11
+ export declare const deleteMessage: (context: ApiContext, messageId: string, conversationId: string) => Promise<MessageType>;
12
+ export declare const getMessage: (data: MessageType) => MessageType;
13
+ export declare const cleanMessages: (database: Database) => Promise<number>;
@@ -0,0 +1,216 @@
1
+ var __create = Object.create;
2
+ var __defProp = Object.defineProperty;
3
+ var __defProps = Object.defineProperties;
4
+ var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
5
+ var __getOwnPropDescs = Object.getOwnPropertyDescriptors;
6
+ var __getOwnPropNames = Object.getOwnPropertyNames;
7
+ var __getOwnPropSymbols = Object.getOwnPropertySymbols;
8
+ var __getProtoOf = Object.getPrototypeOf;
9
+ var __hasOwnProp = Object.prototype.hasOwnProperty;
10
+ var __propIsEnum = Object.prototype.propertyIsEnumerable;
11
+ var __defNormalProp = (obj, key, value) => key in obj ? __defProp(obj, key, { enumerable: true, configurable: true, writable: true, value }) : obj[key] = value;
12
+ var __spreadValues = (a, b) => {
13
+ for (var prop in b || (b = {}))
14
+ if (__hasOwnProp.call(b, prop))
15
+ __defNormalProp(a, prop, b[prop]);
16
+ if (__getOwnPropSymbols)
17
+ for (var prop of __getOwnPropSymbols(b)) {
18
+ if (__propIsEnum.call(b, prop))
19
+ __defNormalProp(a, prop, b[prop]);
20
+ }
21
+ return a;
22
+ };
23
+ var __spreadProps = (a, b) => __defProps(a, __getOwnPropDescs(b));
24
+ var __markAsModule = (target) => __defProp(target, "__esModule", { value: true });
25
+ var __export = (target, all) => {
26
+ __markAsModule(target);
27
+ for (var name in all)
28
+ __defProp(target, name, { get: all[name], enumerable: true });
29
+ };
30
+ var __reExport = (target, module2, desc) => {
31
+ if (module2 && typeof module2 === "object" || typeof module2 === "function") {
32
+ for (let key of __getOwnPropNames(module2))
33
+ if (!__hasOwnProp.call(target, key) && key !== "default")
34
+ __defProp(target, key, { get: () => module2[key], enumerable: !(desc = __getOwnPropDesc(module2, key)) || desc.enumerable });
35
+ }
36
+ return target;
37
+ };
38
+ var __toModule = (module2) => {
39
+ return __reExport(__markAsModule(__defProp(module2 != null ? __create(__getProtoOf(module2)) : {}, "default", module2 && module2.__esModule && "default" in module2 ? { get: () => module2.default, enumerable: true } : { value: module2, enumerable: true })), module2);
40
+ };
41
+ __export(exports, {
42
+ cleanMessages: () => cleanMessages,
43
+ deleteMessage: () => deleteMessage,
44
+ getMessage: () => getMessage,
45
+ getMessages: () => getMessages,
46
+ saveMessage: () => saveMessage,
47
+ unsaveMessage: () => unsaveMessage,
48
+ updateMessage: () => updateMessage
49
+ });
50
+ var import_utils = __toModule(require("@nlabs/utils"));
51
+ var import_arangojs = __toModule(require("arangojs"));
52
+ var import_cloneDeep = __toModule(require("lodash/cloneDeep"));
53
+ var import_isEmpty = __toModule(require("lodash/isEmpty"));
54
+ var import_utils2 = __toModule(require("../utils"));
55
+ var import_conversations2 = __toModule(require("./conversations"));
56
+ const getMessages = (context, conversationId, { from, to }) => {
57
+ const { database, session: { userId: sessionId } } = context;
58
+ const formatConversationId = (0, import_utils.parseId)(conversationId);
59
+ const sessionDocId = `users/${sessionId}`;
60
+ const conversationDocId = `conversations/${formatConversationId}`;
61
+ const conversationQry = import_arangojs.aql`FOR e IN hasConversations
62
+ FILTER e._from == ${sessionDocId} && e._to == ${conversationDocId}
63
+ LIMIT 1
64
+ RETURN e`;
65
+ return database.query(conversationQry).then((cursor) => cursor.all()).then((conversations) => {
66
+ if (!conversations.length) {
67
+ return [];
68
+ }
69
+ const limit = (0, import_utils2.getLimit)(from, to);
70
+ const aqlQry = `FOR m IN messages
71
+ FILTER m.conversationId == "${formatConversationId}"
72
+ SORT m.added DESC
73
+ ${limit.aql}
74
+ SORT m.added ASC
75
+ LET user = FIRST(
76
+ FOR u IN users
77
+ FILTER u._key == m.userId
78
+ LIMIT 1
79
+ RETURN u
80
+ )
81
+ RETURN MERGE(m, {user: user})`;
82
+ return database.query(aqlQry).then((cursor) => cursor.all()).catch((error) => {
83
+ throw error;
84
+ });
85
+ });
86
+ };
87
+ const updateMessage = (context, message) => {
88
+ const { database, session: { userId: sessionId } } = context;
89
+ const {
90
+ conversationId,
91
+ files,
92
+ images,
93
+ messageId = (0, import_utils.createHash)(`message-${sessionId}`)
94
+ } = message;
95
+ const formatConversationId = (0, import_utils.parseId)(conversationId);
96
+ if (formatConversationId === "") {
97
+ throw new Error("required_conversationId");
98
+ }
99
+ const now = Date.now();
100
+ const update = {
101
+ content: (0, import_utils.parseString)(message.content, 640),
102
+ files: files || [],
103
+ images: images || [],
104
+ modified: now
105
+ };
106
+ const formatMsgId = (0, import_utils.parseId)(messageId);
107
+ const insert = __spreadProps(__spreadValues({}, (0, import_cloneDeep.default)(update)), {
108
+ _key: formatMsgId,
109
+ added: now,
110
+ conversationId: formatConversationId,
111
+ userId: sessionId
112
+ });
113
+ const aqlQry = import_arangojs.aql`UPSERT {_key: ${formatMsgId}, conversationId: ${formatConversationId}, userId: ${sessionId}}
114
+ INSERT ${insert}
115
+ UPDATE ${update} IN messages
116
+ RETURN NEW`;
117
+ return database.query(aqlQry).then((cursor) => cursor.next()).catch((error) => {
118
+ throw error;
119
+ });
120
+ };
121
+ const saveMessage = (context, messageId, conversationId) => {
122
+ const { database, session: { userId: sessionId } } = context;
123
+ const formatId = (0, import_utils.parseId)(messageId);
124
+ return (0, import_conversations2.getConversation)(context, conversationId).then((conversation) => {
125
+ const { _key: conversationKey } = conversation;
126
+ const aqlQry = import_arangojs.aql`FOR m IN messages
127
+ FILTER m._key == ${formatId} && m.to == ${conversationKey}
128
+ UPDATE m WITH {saved: APPEND(m.saved, ${sessionId}, true)} IN messages
129
+ RETURN NEW`;
130
+ return database.query(aqlQry).then((cursor) => cursor.next()).then((message = {}) => {
131
+ if (!(0, import_isEmpty.default)(message)) {
132
+ }
133
+ return message;
134
+ }).catch((error) => {
135
+ throw error;
136
+ });
137
+ });
138
+ };
139
+ const unsaveMessage = (context, messageId, conversationId) => {
140
+ const { database, session: { userId: sessionId } } = context;
141
+ const formatId = (0, import_utils.parseId)(messageId);
142
+ return (0, import_conversations2.getConversation)(context, conversationId).then((conversation) => {
143
+ const { _key: conversationKey } = conversation;
144
+ const aqlQry = import_arangojs.aql`FOR m IN messages
145
+ FILTER m._key == ${formatId} && m.to == ${conversationKey}
146
+ UPDATE m WITH {saved: REMOVE_VALUE(m.saved, ${sessionId})} IN messages
147
+ RETURN NEW`;
148
+ return database.query(aqlQry).then((cursor) => cursor.next()).then((message = {}) => {
149
+ if (!(0, import_isEmpty.default)(message)) {
150
+ }
151
+ return message;
152
+ }).catch((error) => {
153
+ throw error;
154
+ });
155
+ });
156
+ };
157
+ const deleteMessage = (context, messageId, conversationId) => {
158
+ const { database, session: { userId: sessionId } } = context;
159
+ const formatId = (0, import_utils.parseId)(messageId);
160
+ return (0, import_conversations2.getConversation)(context, conversationId).then((conversation) => {
161
+ const { _key: conversationKey } = conversation;
162
+ const aqlQry = import_arangojs.aql`FOR m IN messages
163
+ FILTER m._key == ${formatId} && m.to == ${conversationKey} && m.from == ${sessionId}
164
+ REMOVE m IN messages
165
+ RETURN OLD`;
166
+ return database.query(aqlQry).then((cursor) => cursor.next()).then((message = {}) => {
167
+ if (!(0, import_isEmpty.default)(message)) {
168
+ }
169
+ return message;
170
+ }).catch((error) => {
171
+ throw error;
172
+ });
173
+ });
174
+ };
175
+ const getMessage = (data) => {
176
+ const {
177
+ added,
178
+ content,
179
+ conversationId,
180
+ files,
181
+ _key: messageId,
182
+ read,
183
+ saved,
184
+ userId
185
+ } = data;
186
+ return {
187
+ added,
188
+ content,
189
+ conversationId,
190
+ files,
191
+ messageId,
192
+ read,
193
+ saved,
194
+ userId
195
+ };
196
+ };
197
+ const cleanMessages = (database) => {
198
+ const aqlQry = import_arangojs.aql`FOR m IN messages
199
+ FILTER m.added < DATE_TIMESTAMP(DATE_SUBTRACT(DATE_NOW(), 60, 'day')) && LENGTH(m.saved) > 0
200
+ REMOVE m IN messages
201
+ RETURN OLD`;
202
+ return database.query(aqlQry).then((cursor) => cursor.all()).then((list) => list.length).catch((error) => {
203
+ throw error;
204
+ });
205
+ };
206
+ // Annotate the CommonJS export names for ESM import in node:
207
+ 0 && (module.exports = {
208
+ cleanMessages,
209
+ deleteMessage,
210
+ getMessage,
211
+ getMessages,
212
+ saveMessage,
213
+ unsaveMessage,
214
+ updateMessage
215
+ });
216
+ //# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFsiLi4vLi4vc3JjL2FjdGlvbnMvbWVzc2FnZXMudHMiXSwKICAic291cmNlc0NvbnRlbnQiOiBbIi8qKlxuICogQ29weXJpZ2h0IChjKSAyMDE5LVByZXNlbnQsIE5pdHJvZ2VuIExhYnMsIEluYy5cbiAqIENvcHlyaWdodHMgbGljZW5zZWQgdW5kZXIgdGhlIE1JVCBMaWNlbnNlLiBTZWUgdGhlIGFjY29tcGFueWluZyBMSUNFTlNFIGZpbGUgZm9yIHRlcm1zLlxuICovXG5pbXBvcnQge2NyZWF0ZUhhc2gsIHBhcnNlSWQsIHBhcnNlU3RyaW5nfSBmcm9tICdAbmxhYnMvdXRpbHMnO1xuaW1wb3J0IHthcWwsIERhdGFiYXNlfSBmcm9tICdhcmFuZ29qcyc7XG5pbXBvcnQge0FxbFF1ZXJ5fSBmcm9tICdhcmFuZ29qcy9hcWwnO1xuaW1wb3J0IHtBcnJheUN1cnNvcn0gZnJvbSAnYXJhbmdvanMvY3Vyc29yJztcbmltcG9ydCBjbG9uZURlZXAgZnJvbSAnbG9kYXNoL2Nsb25lRGVlcCc7XG5pbXBvcnQgaXNFbXB0eSBmcm9tICdsb2Rhc2gvaXNFbXB0eSc7XG5cbmltcG9ydCB7QXJhbmdvREJMaW1pdH0gZnJvbSAnLi4vdHlwZXMvYXJhbmdvZGInO1xuaW1wb3J0IHtBcGlDb250ZXh0fSBmcm9tICcuLi90eXBlcy9hdXRoJztcbmltcG9ydCB7Q29udmVyc2F0aW9uVHlwZX0gZnJvbSAnLi4vdHlwZXMvY29udmVyc2F0aW9ucyc7XG5pbXBvcnQge01lc3NhZ2VJbnB1dFR5cGUsIE1lc3NhZ2VUeXBlfSBmcm9tICcuLi90eXBlcy9tZXNzYWdlcyc7XG5pbXBvcnQge2dldExpbWl0fSBmcm9tICcuLi91dGlscyc7XG5pbXBvcnQge2dldENvbnZlcnNhdGlvbn0gZnJvbSAnLi9jb252ZXJzYXRpb25zJztcblxuLy8gY29uc3QgZXZlbnRDYXRlZ29yeTogc3RyaW5nID0gJ21lc3NhZ2VzJztcblxuZXhwb3J0IGNvbnN0IGdldE1lc3NhZ2VzID0gKGNvbnRleHQ6IEFwaUNvbnRleHQsIGNvbnZlcnNhdGlvbklkOiBzdHJpbmcsIHtmcm9tLCB0b30pOiBQcm9taXNlPE1lc3NhZ2VUeXBlW10+ID0+IHtcbiAgY29uc3Qge2RhdGFiYXNlLCBzZXNzaW9uOiB7dXNlcklkOiBzZXNzaW9uSWR9fSA9IGNvbnRleHQ7XG4gIGNvbnN0IGZvcm1hdENvbnZlcnNhdGlvbklkOiBzdHJpbmcgPSBwYXJzZUlkKGNvbnZlcnNhdGlvbklkKTtcbiAgY29uc3Qgc2Vzc2lvbkRvY0lkOiBzdHJpbmcgPSBgdXNlcnMvJHtzZXNzaW9uSWR9YDtcbiAgY29uc3QgY29udmVyc2F0aW9uRG9jSWQ6IHN0cmluZyA9IGBjb252ZXJzYXRpb25zLyR7Zm9ybWF0Q29udmVyc2F0aW9uSWR9YDtcblxuICBjb25zdCBjb252ZXJzYXRpb25RcnkgPSBhcWxgRk9SIGUgSU4gaGFzQ29udmVyc2F0aW9uc1xuICAgIEZJTFRFUiBlLl9mcm9tID09ICR7c2Vzc2lvbkRvY0lkfSAmJiBlLl90byA9PSAke2NvbnZlcnNhdGlvbkRvY0lkfVxuICAgIExJTUlUIDFcbiAgICBSRVRVUk4gZWA7XG5cbiAgcmV0dXJuIGRhdGFiYXNlLnF1ZXJ5KGNvbnZlcnNhdGlvblFyeSlcbiAgICAudGhlbigoY3Vyc29yOiBBcnJheUN1cnNvcikgPT4gY3Vyc29yLmFsbCgpKVxuICAgIC50aGVuKChjb252ZXJzYXRpb25zKSA9PiB7XG4gICAgICBpZighY29udmVyc2F0aW9ucy5sZW5ndGgpIHtcbiAgICAgICAgcmV0dXJuIFtdO1xuICAgICAgfVxuXG4gICAgICBjb25zdCBsaW1pdDogQXJhbmdvREJMaW1pdCA9IGdldExpbWl0KGZyb20sIHRvKTtcbiAgICAgIGNvbnN0IGFxbFFyeTogc3RyaW5nID0gYEZPUiBtIElOIG1lc3NhZ2VzXG4gICAgICAgICAgRklMVEVSIG0uY29udmVyc2F0aW9uSWQgPT0gXCIke2Zvcm1hdENvbnZlcnNhdGlvbklkfVwiXG4gICAgICAgICAgU09SVCBtLmFkZGVkIERFU0NcbiAgICAgICAgICAke2xpbWl0LmFxbH1cbiAgICAgICAgICBTT1JUIG0uYWRkZWQgQVNDXG4gICAgICAgICAgTEVUIHVzZXIgPSBGSVJTVChcbiAgICAgICAgICAgIEZPUiB1IElOIHVzZXJzXG4gICAgICAgICAgICBGSUxURVIgdS5fa2V5ID09IG0udXNlcklkXG4gICAgICAgICAgICBMSU1JVCAxXG4gICAgICAgICAgICBSRVRVUk4gdVxuICAgICAgICAgIClcbiAgICAgICAgICBSRVRVUk4gTUVSR0UobSwge3VzZXI6IHVzZXJ9KWA7XG5cbiAgICAgIHJldHVybiBkYXRhYmFzZS5xdWVyeShhcWxRcnkpXG4gICAgICAgIC50aGVuKChjdXJzb3I6IEFycmF5Q3Vyc29yKSA9PiBjdXJzb3IuYWxsKCkpXG4gICAgICAgIC5jYXRjaCgoZXJyb3I6IEVycm9yKSA9PiB7XG4gICAgICAgICAgdGhyb3cgZXJyb3I7XG4gICAgICAgIH0pO1xuICAgIH0pO1xufTtcblxuZXhwb3J0IGNvbnN0IHVwZGF0ZU1lc3NhZ2UgPSAoY29udGV4dDogQXBpQ29udGV4dCwgbWVzc2FnZTogTWVzc2FnZUlucHV0VHlwZSk6IFByb21pc2U8TWVzc2FnZVR5cGU+ID0+IHtcbiAgY29uc3Qge2RhdGFiYXNlLCBzZXNzaW9uOiB7dXNlcklkOiBzZXNzaW9uSWR9fSA9IGNvbnRleHQ7XG4gIGNvbnN0IHtcbiAgICBjb252ZXJzYXRpb25JZCxcbiAgICBmaWxlcyxcbiAgICBpbWFnZXMsXG4gICAgbWVzc2FnZUlkID0gY3JlYXRlSGFzaChgbWVzc2FnZS0ke3Nlc3Npb25JZH1gKVxuICB9ID0gbWVzc2FnZTtcbiAgY29uc3QgZm9ybWF0Q29udmVyc2F0aW9uSWQ6IHN0cmluZyA9IHBhcnNlSWQoY29udmVyc2F0aW9uSWQpO1xuXG4gIGlmKGZvcm1hdENvbnZlcnNhdGlvbklkID09PSAnJykge1xuICAgIHRocm93IG5ldyBFcnJvcigncmVxdWlyZWRfY29udmVyc2F0aW9uSWQnKTtcbiAgfVxuXG4gIGNvbnN0IG5vdzogbnVtYmVyID0gRGF0ZS5ub3coKTtcbiAgY29uc3QgdXBkYXRlOiBNZXNzYWdlVHlwZSA9IHtcbiAgICBjb250ZW50OiBwYXJzZVN0cmluZyhtZXNzYWdlLmNvbnRlbnQsIDY0MCksXG4gICAgZmlsZXM6IGZpbGVzIHx8IFtdLFxuICAgIGltYWdlczogaW1hZ2VzIHx8IFtdLFxuICAgIG1vZGlmaWVkOiBub3dcbiAgfTtcbiAgY29uc3QgZm9ybWF0TXNnSWQ6IHN0cmluZyA9IHBhcnNlSWQobWVzc2FnZUlkKTtcbiAgY29uc3QgaW5zZXJ0OiBNZXNzYWdlVHlwZSA9IHtcbiAgICAuLi5jbG9uZURlZXAodXBkYXRlKSxcbiAgICBfa2V5OiBmb3JtYXRNc2dJZCxcbiAgICBhZGRlZDogbm93LFxuICAgIGNvbnZlcnNhdGlvbklkOiBmb3JtYXRDb252ZXJzYXRpb25JZCxcbiAgICB1c2VySWQ6IHNlc3Npb25JZFxuICB9O1xuICBjb25zdCBhcWxRcnk6IEFxbFF1ZXJ5ID0gYXFsYFVQU0VSVCB7X2tleTogJHtmb3JtYXRNc2dJZH0sIGNvbnZlcnNhdGlvbklkOiAke2Zvcm1hdENvbnZlcnNhdGlvbklkfSwgdXNlcklkOiAke3Nlc3Npb25JZH19XG4gICAgSU5TRVJUICR7aW5zZXJ0fVxuICAgIFVQREFURSAke3VwZGF0ZX0gSU4gbWVzc2FnZXNcbiAgICBSRVRVUk4gTkVXYDtcblxuICByZXR1cm4gZGF0YWJhc2UucXVlcnkoYXFsUXJ5KVxuICAgIC50aGVuKChjdXJzb3I6IEFycmF5Q3Vyc29yKSA9PiBjdXJzb3IubmV4dCgpKVxuICAgIC5jYXRjaCgoZXJyb3I6IEVycm9yKSA9PiB7XG4gICAgICB0aHJvdyBlcnJvcjtcbiAgICB9KTtcbn07XG5cbmV4cG9ydCBjb25zdCBzYXZlTWVzc2FnZSA9IChjb250ZXh0OiBBcGlDb250ZXh0LCBtZXNzYWdlSWQ6IHN0cmluZywgY29udmVyc2F0aW9uSWQ6IHN0cmluZyk6IFByb21pc2U8TWVzc2FnZVR5cGU+ID0+IHtcbiAgY29uc3Qge2RhdGFiYXNlLCBzZXNzaW9uOiB7dXNlcklkOiBzZXNzaW9uSWR9fSA9IGNvbnRleHQ7XG4gIGNvbnN0IGZvcm1hdElkOiBzdHJpbmcgPSBwYXJzZUlkKG1lc3NhZ2VJZCk7XG5cbiAgcmV0dXJuIGdldENvbnZlcnNhdGlvbihjb250ZXh0LCBjb252ZXJzYXRpb25JZClcbiAgICAudGhlbigoY29udmVyc2F0aW9uOiBDb252ZXJzYXRpb25UeXBlKSA9PiB7XG4gICAgICBjb25zdCB7X2tleTogY29udmVyc2F0aW9uS2V5fSA9IGNvbnZlcnNhdGlvbjtcbiAgICAgIGNvbnN0IGFxbFFyeSA9IGFxbGBGT1IgbSBJTiBtZXNzYWdlc1xuICAgICAgICBGSUxURVIgbS5fa2V5ID09ICR7Zm9ybWF0SWR9ICYmIG0udG8gPT0gJHtjb252ZXJzYXRpb25LZXl9XG4gICAgICAgIFVQREFURSBtIFdJVEgge3NhdmVkOiBBUFBFTkQobS5zYXZlZCwgJHtzZXNzaW9uSWR9LCB0cnVlKX0gSU4gbWVzc2FnZXNcbiAgICAgICAgUkVUVVJOIE5FV2A7XG5cbiAgICAgIHJldHVybiBkYXRhYmFzZS5xdWVyeShhcWxRcnkpXG4gICAgICAgIC50aGVuKChjdXJzb3I6IEFycmF5Q3Vyc29yKSA9PiBjdXJzb3IubmV4dCgpKVxuICAgICAgICAudGhlbigobWVzc2FnZTogTWVzc2FnZVR5cGUgPSB7fSkgPT4ge1xuICAgICAgICAgIGlmKCFpc0VtcHR5KG1lc3NhZ2UpKSB7XG4gICAgICAgICAgICAvLyBCcm9hZGNhc3QgdG8gbWVzc2FnZSBib3hcbiAgICAgICAgICAgIC8vIGNvbnN0IG1lc3NhZ2U6IE1lc3NhZ2VUeXBlID0gTWVzc2FnZXMuZ2V0TWVzc2FnZU9iamVjdChtZXNzYWdlKTtcbiAgICAgICAgICAgIC8vIGlvLnRvKG1lc3NhZ2UudG9JZCkuZW1pdCgnbWVzc2FnZV9zYXZlJywgbWVzc2FnZSk7XG4gICAgICAgICAgfVxuXG4gICAgICAgICAgcmV0dXJuIG1lc3NhZ2U7XG4gICAgICAgIH0pXG4gICAgICAgIC5jYXRjaCgoZXJyb3I6IEVycm9yKSA9PiB7XG4gICAgICAgICAgdGhyb3cgZXJyb3I7XG4gICAgICAgIH0pO1xuICAgIH0pO1xufTtcblxuZXhwb3J0IGNvbnN0IHVuc2F2ZU1lc3NhZ2UgPSAoY29udGV4dDogQXBpQ29udGV4dCwgbWVzc2FnZUlkOiBzdHJpbmcsIGNvbnZlcnNhdGlvbklkOiBzdHJpbmcpOiBQcm9taXNlPE1lc3NhZ2VUeXBlPiA9PiB7XG4gIGNvbnN0IHtkYXRhYmFzZSwgc2Vzc2lvbjoge3VzZXJJZDogc2Vzc2lvbklkfX0gPSBjb250ZXh0O1xuICBjb25zdCBmb3JtYXRJZDogc3RyaW5nID0gcGFyc2VJZChtZXNzYWdlSWQpO1xuXG4gIHJldHVybiBnZXRDb252ZXJzYXRpb24oY29udGV4dCwgY29udmVyc2F0aW9uSWQpXG4gICAgLnRoZW4oKGNvbnZlcnNhdGlvbjogQ29udmVyc2F0aW9uVHlwZSkgPT4ge1xuICAgICAgY29uc3Qge19rZXk6IGNvbnZlcnNhdGlvbktleX0gPSBjb252ZXJzYXRpb247XG4gICAgICBjb25zdCBhcWxRcnk6IEFxbFF1ZXJ5ID0gYXFsYEZPUiBtIElOIG1lc3NhZ2VzXG4gICAgICAgICAgRklMVEVSIG0uX2tleSA9PSAke2Zvcm1hdElkfSAmJiBtLnRvID09ICR7Y29udmVyc2F0aW9uS2V5fVxuICAgICAgICAgIFVQREFURSBtIFdJVEgge3NhdmVkOiBSRU1PVkVfVkFMVUUobS5zYXZlZCwgJHtzZXNzaW9uSWR9KX0gSU4gbWVzc2FnZXNcbiAgICAgICAgICBSRVRVUk4gTkVXYDtcblxuICAgICAgcmV0dXJuIGRhdGFiYXNlLnF1ZXJ5KGFxbFFyeSlcbiAgICAgICAgLnRoZW4oKGN1cnNvcjogQXJyYXlDdXJzb3IpID0+IGN1cnNvci5uZXh0KCkpXG4gICAgICAgIC50aGVuKChtZXNzYWdlOiBNZXNzYWdlVHlwZSA9IHt9KSA9PiB7XG4gICAgICAgICAgaWYoIWlzRW1wdHkobWVzc2FnZSkpIHtcbiAgICAgICAgICAgIC8vIEJyb2FkY2FzdCB0byBtZXNzYWdlIGJveFxuICAgICAgICAgICAgLy8gY29uc3QgbWVzc2FnZTogTWVzc2FnZVR5cGUgPSBNZXNzYWdlcy5nZXRNZXNzYWdlT2JqZWN0KG1lc3NhZ2UpO1xuICAgICAgICAgICAgLy8gaW8udG8obWVzc2FnZS5pZCkuZW1pdCgnbWVzc2FnZV91bnNhdmUnLCBtZXNzYWdlKTtcbiAgICAgICAgICB9XG5cbiAgICAgICAgICByZXR1cm4gbWVzc2FnZTtcbiAgICAgICAgfSlcbiAgICAgICAgLmNhdGNoKChlcnJvcjogRXJyb3IpID0+IHtcbiAgICAgICAgICB0aHJvdyBlcnJvcjtcbiAgICAgICAgfSk7XG4gICAgfSk7XG59O1xuXG5leHBvcnQgY29uc3QgZGVsZXRlTWVzc2FnZSA9IChjb250ZXh0OiBBcGlDb250ZXh0LCBtZXNzYWdlSWQ6IHN0cmluZywgY29udmVyc2F0aW9uSWQ6IHN0cmluZyk6IFByb21pc2U8TWVzc2FnZVR5cGU+ID0+IHtcbiAgY29uc3Qge2RhdGFiYXNlLCBzZXNzaW9uOiB7dXNlcklkOiBzZXNzaW9uSWR9fSA9IGNvbnRleHQ7XG4gIGNvbnN0IGZvcm1hdElkOiBzdHJpbmcgPSBwYXJzZUlkKG1lc3NhZ2VJZCk7XG5cbiAgcmV0dXJuIGdldENvbnZlcnNhdGlvbihjb250ZXh0LCBjb252ZXJzYXRpb25JZClcbiAgICAudGhlbigoY29udmVyc2F0aW9uOiBDb252ZXJzYXRpb25UeXBlKSA9PiB7XG4gICAgICBjb25zdCB7X2tleTogY29udmVyc2F0aW9uS2V5fSA9IGNvbnZlcnNhdGlvbjtcbiAgICAgIGNvbnN0IGFxbFFyeSA9IGFxbGBGT1IgbSBJTiBtZXNzYWdlc1xuICAgICAgICBGSUxURVIgbS5fa2V5ID09ICR7Zm9ybWF0SWR9ICYmIG0udG8gPT0gJHtjb252ZXJzYXRpb25LZXl9ICYmIG0uZnJvbSA9PSAke3Nlc3Npb25JZH1cbiAgICAgICAgUkVNT1ZFIG0gSU4gbWVzc2FnZXNcbiAgICAgICAgUkVUVVJOIE9MRGA7XG5cbiAgICAgIHJldHVybiBkYXRhYmFzZS5xdWVyeShhcWxRcnkpXG4gICAgICAgIC50aGVuKChjdXJzb3I6IEFycmF5Q3Vyc29yKSA9PiBjdXJzb3IubmV4dCgpKVxuICAgICAgICAudGhlbigobWVzc2FnZTogTWVzc2FnZVR5cGUgPSB7fSkgPT4ge1xuICAgICAgICAgIGlmKCFpc0VtcHR5KG1lc3NhZ2UpKSB7XG4gICAgICAgICAgICAvLyBCcm9hZGNhc3QgdG8gbWVzc2FnZSBib3hcbiAgICAgICAgICAgIC8vIGNvbnN0IG1lc3NhZ2U6IE1lc3NhZ2VUeXBlID0gTWVzc2FnZXMuZ2V0TWVzc2FnZU9iamVjdChtZXNzYWdlKTtcbiAgICAgICAgICAgIC8vIGlvLnRvKG1lc3NhZ2UudG9JZCkuZW1pdCgnbWVzc2FnZV9kZWxldGUnLCBtZXNzYWdlKTtcbiAgICAgICAgICB9XG5cbiAgICAgICAgICByZXR1cm4gbWVzc2FnZTtcbiAgICAgICAgfSlcbiAgICAgICAgLmNhdGNoKChlcnJvcjogRXJyb3IpID0+IHtcbiAgICAgICAgICB0aHJvdyBlcnJvcjtcbiAgICAgICAgfSk7XG4gICAgfSk7XG59O1xuXG5leHBvcnQgY29uc3QgZ2V0TWVzc2FnZSA9IChkYXRhOiBNZXNzYWdlVHlwZSk6IE1lc3NhZ2VUeXBlID0+IHtcbiAgY29uc3Qge1xuICAgIGFkZGVkLFxuICAgIGNvbnRlbnQsXG4gICAgY29udmVyc2F0aW9uSWQsXG4gICAgZmlsZXMsXG4gICAgX2tleTogbWVzc2FnZUlkLFxuICAgIHJlYWQsXG4gICAgc2F2ZWQsXG4gICAgdXNlcklkXG4gIH0gPSBkYXRhO1xuXG4gIHJldHVybiB7XG4gICAgYWRkZWQsXG4gICAgY29udGVudCxcbiAgICBjb252ZXJzYXRpb25JZCxcbiAgICBmaWxlcyxcbiAgICBtZXNzYWdlSWQsXG4gICAgcmVhZCxcbiAgICBzYXZlZCxcbiAgICB1c2VySWRcbiAgfTtcbn07XG5cbmV4cG9ydCBjb25zdCBjbGVhbk1lc3NhZ2VzID0gKGRhdGFiYXNlOiBEYXRhYmFzZSk6IFByb21pc2U8bnVtYmVyPiA9PiB7XG4gIC8vIFJlbW92ZSBhbGwgbWVzc2FnZXMgdGhhdCBhcmUgb3ZlciA2MCBkYXlzIGFuZCBub3Qgc2F2ZWRcbiAgY29uc3QgYXFsUXJ5ID0gYXFsYEZPUiBtIElOIG1lc3NhZ2VzXG4gICAgICBGSUxURVIgbS5hZGRlZCA8IERBVEVfVElNRVNUQU1QKERBVEVfU1VCVFJBQ1QoREFURV9OT1coKSwgNjAsICdkYXknKSkgJiYgTEVOR1RIKG0uc2F2ZWQpID4gMFxuICAgICAgUkVNT1ZFIG0gSU4gbWVzc2FnZXNcbiAgICAgIFJFVFVSTiBPTERgO1xuXG4gIHJldHVybiBkYXRhYmFzZS5xdWVyeShhcWxRcnkpXG4gICAgLnRoZW4oKGN1cnNvcjogQXJyYXlDdXJzb3IpID0+IGN1cnNvci5hbGwoKSlcbiAgICAudGhlbigobGlzdDogTWVzc2FnZVR5cGVbXSkgPT4gbGlzdC5sZW5ndGgpXG4gICAgLmNhdGNoKChlcnJvcjogRXJyb3IpID0+IHtcbiAgICAgIHRocm93IGVycm9yO1xuICAgIH0pO1xufTtcbiJdLAogICJtYXBwaW5ncyI6ICI7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFJQSxtQkFBK0M7QUFDL0Msc0JBQTRCO0FBRzVCLHVCQUFzQjtBQUN0QixxQkFBb0I7QUFNcEIsb0JBQXVCO0FBQ3ZCLDRCQUE4QjtBQUl2QixNQUFNLGNBQWMsQ0FBQyxTQUFxQixnQkFBd0IsRUFBQyxNQUFNLFNBQWdDO0FBQzlHLFFBQU0sRUFBQyxVQUFVLFNBQVMsRUFBQyxRQUFRLGdCQUFjO0FBQ2pELFFBQU0sdUJBQStCLDBCQUFRO0FBQzdDLFFBQU0sZUFBdUIsU0FBUztBQUN0QyxRQUFNLG9CQUE0QixpQkFBaUI7QUFFbkQsUUFBTSxrQkFBa0I7QUFBQSx3QkFDRiw0QkFBNEI7QUFBQTtBQUFBO0FBSWxELFNBQU8sU0FBUyxNQUFNLGlCQUNuQixLQUFLLENBQUMsV0FBd0IsT0FBTyxPQUNyQyxLQUFLLENBQUMsa0JBQWtCO0FBQ3ZCLFFBQUcsQ0FB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