@nlabs/reaktor 0.4.0 → 0.4.2
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.
- package/lib/actions/conversations.d.ts +14 -0
- package/lib/actions/conversations.js +333 -0
- package/lib/actions/dynamodb.js +155 -0
- package/lib/actions/email.js +177 -0
- package/lib/actions/files.js +319 -0
- package/lib/{data → actions}/groups.d.ts +4 -3
- package/lib/actions/groups.js +282 -0
- package/lib/actions/images.d.ts +22 -0
- package/lib/actions/images.js +682 -0
- package/lib/actions/index.js +40 -0
- package/lib/{data → actions}/ios.d.ts +2 -1
- package/lib/actions/ios.js +179 -0
- package/lib/actions/locations.js +112 -0
- package/lib/actions/messages.d.ts +13 -0
- package/lib/actions/messages.js +216 -0
- package/lib/{data → actions}/notifications.d.ts +2 -2
- package/lib/actions/notifications.js +63 -0
- package/lib/{data → actions}/payments.d.ts +2 -2
- package/lib/actions/payments.js +491 -0
- package/lib/actions/posts.d.ts +19 -0
- package/lib/actions/posts.js +538 -0
- package/lib/actions/reactions.d.ts +30 -0
- package/lib/actions/reactions.js +340 -0
- package/lib/{data → actions}/s3.d.ts +1 -1
- package/lib/actions/s3.js +122 -0
- package/lib/{data → actions}/search.d.ts +2 -2
- package/lib/actions/search.js +99 -0
- package/lib/actions/sms.js +76 -0
- package/lib/actions/statistics.d.ts +2 -0
- package/lib/actions/statistics.js +63 -0
- package/lib/actions/subscription.js +209 -0
- package/lib/actions/tags.d.ts +26 -0
- package/lib/actions/tags.js +340 -0
- package/lib/actions/users.d.ts +44 -0
- package/lib/actions/users.js +571 -0
- package/lib/{data → actions}/websockets.d.ts +1 -1
- package/lib/actions/websockets.js +156 -0
- package/lib/config.d.ts +2 -3
- package/lib/config.js +116 -149
- package/lib/index.d.ts +1 -1
- package/lib/index.js +23 -45
- package/lib/templates/email/layout.d.ts +2 -0
- package/lib/templates/email/layout.js +292 -0
- package/lib/templates/email/passwordForgot.d.ts +2 -0
- package/lib/templates/email/passwordForgot.js +28 -0
- package/lib/templates/email/passwordRecovery.d.ts +2 -0
- package/lib/templates/email/passwordRecovery.js +25 -0
- package/lib/templates/email/verifyEmail.d.ts +2 -0
- package/lib/templates/email/verifyEmail.js +28 -0
- package/lib/templates/email/welcome.d.ts +2 -0
- package/lib/templates/email/welcome.js +28 -0
- package/lib/templates/sms/passwordForgot.d.ts +2 -0
- package/lib/templates/sms/passwordForgot.js +14 -0
- package/lib/templates/sms/passwordRecovery.d.ts +2 -0
- package/lib/templates/sms/passwordRecovery.js +14 -0
- package/lib/templates/sms/verifyEmail.d.ts +2 -0
- package/lib/templates/sms/verifyEmail.js +14 -0
- package/lib/templates/sms/verifyPhone.d.ts +2 -0
- package/lib/templates/sms/verifyPhone.js +14 -0
- package/lib/templates/sms/welcome.d.ts +2 -0
- package/lib/templates/sms/welcome.js +14 -0
- package/lib/types/apps.d.ts +2 -2
- package/lib/types/apps.js +4 -2
- package/lib/types/arangodb.js +4 -2
- package/lib/types/auth.d.ts +4 -8
- package/lib/types/auth.js +4 -2
- package/lib/types/conversations.d.ts +3 -3
- package/lib/types/conversations.js +4 -2
- package/lib/types/email.d.ts +2 -2
- package/lib/types/email.js +4 -2
- package/lib/types/files.js +4 -2
- package/lib/types/google.js +4 -2
- package/lib/types/groups.d.ts +2 -1
- package/lib/types/groups.js +4 -2
- package/lib/types/images.d.ts +8 -5
- package/lib/types/images.js +4 -2
- package/lib/types/index.d.ts +1 -1
- package/lib/types/index.js +37 -227
- package/lib/types/locations.js +4 -2
- package/lib/types/messages.d.ts +12 -2
- package/lib/types/messages.js +4 -2
- package/lib/types/notifications.d.ts +2 -2
- package/lib/types/notifications.js +4 -2
- package/lib/types/payments.js +4 -2
- package/lib/types/posts.d.ts +18 -1
- package/lib/types/posts.js +4 -2
- package/lib/types/statistics.d.ts +3 -0
- package/lib/types/statistics.js +4 -0
- package/lib/types/tags.d.ts +6 -0
- package/lib/types/tags.js +4 -2
- package/lib/types/users.d.ts +15 -11
- package/lib/types/users.js +4 -2
- package/lib/utils/analytics.d.ts +7 -0
- package/lib/utils/analytics.js +101 -77
- package/lib/utils/arangodb.d.ts +1 -1
- package/lib/utils/arangodb.js +93 -114
- package/lib/utils/auth.js +58 -55
- package/lib/utils/graphql.js +38 -19
- package/lib/utils/index.d.ts +1 -1
- package/lib/utils/index.js +26 -84
- package/lib/utils/objects.js +44 -53
- package/lib/utils/session.d.ts +18 -0
- package/lib/utils/session.js +42 -0
- package/package.json +32 -30
- package/lib/data/conversations.d.ts +0 -8
- package/lib/data/conversations.js +0 -311
- package/lib/data/dynamodb.js +0 -206
- package/lib/data/email.js +0 -222
- package/lib/data/files.js +0 -525
- package/lib/data/groups.js +0 -435
- package/lib/data/images.d.ts +0 -22
- package/lib/data/images.js +0 -1051
- package/lib/data/index.js +0 -266
- package/lib/data/ios.js +0 -355
- package/lib/data/locations.js +0 -172
- package/lib/data/messages.d.ts +0 -9
- package/lib/data/messages.js +0 -299
- package/lib/data/notifications.js +0 -59
- package/lib/data/payments.js +0 -771
- package/lib/data/posts.d.ts +0 -23
- package/lib/data/posts.js +0 -766
- package/lib/data/reactions.d.ts +0 -14
- package/lib/data/reactions.js +0 -529
- package/lib/data/s3.js +0 -155
- package/lib/data/search.js +0 -155
- package/lib/data/sms.js +0 -83
- package/lib/data/subscription.js +0 -337
- package/lib/data/tags.d.ts +0 -14
- package/lib/data/tags.js +0 -397
- package/lib/data/users.d.ts +0 -20
- package/lib/data/users.js +0 -470
- package/lib/data/websockets.js +0 -250
- package/lib/types/reactions.d.ts +0 -17
- package/lib/types/reactions.js +0 -2
- package/lib/utils/redis.d.ts +0 -1
- package/lib/utils/redis.js +0 -36
- package/templates/email/layout.html +0 -279
- package/templates/email/passwordForgot.html +0 -15
- package/templates/email/passwordRecovery.html +0 -12
- package/templates/email/verifyEmail.html +0 -15
- package/templates/sms/passwordForgot.txt +0 -1
- package/templates/sms/passwordRecovery.txt +0 -1
- package/templates/sms/verifyEmail.txt +0 -1
- package/templates/sms/verifyPhone.txt +0 -1
- /package/lib/{data → actions}/dynamodb.d.ts +0 -0
- /package/lib/{data → actions}/email.d.ts +0 -0
- /package/lib/{data → actions}/files.d.ts +0 -0
- /package/lib/{data → actions}/index.d.ts +0 -0
- /package/lib/{data → actions}/locations.d.ts +0 -0
- /package/lib/{data → actions}/sms.d.ts +0 -0
- /package/lib/{data → actions}/subscription.d.ts +0 -0
|
@@ -0,0 +1,14 @@
|
|
|
1
|
+
import { ApiContext } from 'types/auth';
|
|
2
|
+
import { ConversationType } from '../types';
|
|
3
|
+
export declare const getConversations: (context: ApiContext, { from, to }: {
|
|
4
|
+
from: any;
|
|
5
|
+
to: any;
|
|
6
|
+
}) => Promise<ConversationType[]>;
|
|
7
|
+
export declare const getDirectConversation: (context: ApiContext, fromId: string, toId: string) => Promise<ConversationType>;
|
|
8
|
+
export declare const getConnectionUsers: (context: ApiContext, conversationId: string, includeAll?: boolean, isDirect?: boolean) => Promise<ConversationType[]>;
|
|
9
|
+
export declare const getConversationUsers: (context: ApiContext, conversationId: string, includeAll?: boolean, isDirect?: boolean) => Promise<ConversationType[]>;
|
|
10
|
+
export declare const getDirectConversationOld: (context: ApiContext, userId: string) => Promise<ConversationType>;
|
|
11
|
+
export declare const getConversation: (context: ApiContext, convoId: string) => Promise<ConversationType>;
|
|
12
|
+
export declare const updateConversation: (context: ApiContext, conversation: any) => Promise<ConversationType>;
|
|
13
|
+
export declare const addUserToConversation: (context: ApiContext, convoId: string, userId: string) => Promise<any>;
|
|
14
|
+
export declare const deleteUserFromConversation: (context: ApiContext, convoId: string, userId: string) => Promise<any>;
|
|
@@ -0,0 +1,333 @@
|
|
|
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
|
+
addUserToConversation: () => addUserToConversation,
|
|
43
|
+
deleteUserFromConversation: () => deleteUserFromConversation,
|
|
44
|
+
getConnectionUsers: () => getConnectionUsers,
|
|
45
|
+
getConversation: () => getConversation,
|
|
46
|
+
getConversationUsers: () => getConversationUsers,
|
|
47
|
+
getConversations: () => getConversations,
|
|
48
|
+
getDirectConversation: () => getDirectConversation,
|
|
49
|
+
getDirectConversationOld: () => getDirectConversationOld,
|
|
50
|
+
updateConversation: () => updateConversation
|
|
51
|
+
});
|
|
52
|
+
var import_utils = __toModule(require("@nlabs/utils"));
|
|
53
|
+
var import_arangojs = __toModule(require("arangojs"));
|
|
54
|
+
var import_cloneDeep = __toModule(require("lodash/cloneDeep"));
|
|
55
|
+
var import_isEmpty = __toModule(require("lodash/isEmpty"));
|
|
56
|
+
var import_utils2 = __toModule(require("../utils"));
|
|
57
|
+
var import_analytics = __toModule(require("../utils/analytics"));
|
|
58
|
+
var import_images = __toModule(require("./images"));
|
|
59
|
+
var import_users = __toModule(require("./users"));
|
|
60
|
+
const eventCategory = "conversations";
|
|
61
|
+
const getConversations = (context, { from, to }) => {
|
|
62
|
+
const action = "getConversations";
|
|
63
|
+
const { database, session: { userId: sessionId } } = context;
|
|
64
|
+
const limit = (0, import_utils2.getLimit)(from, to);
|
|
65
|
+
const sessionDocId = `users/${sessionId}`;
|
|
66
|
+
const aqlQry = `FOR c, e IN 1..1 OUTBOUND "${sessionDocId}" hasConversations
|
|
67
|
+
${limit.aql}
|
|
68
|
+
LET users = (
|
|
69
|
+
FOR u, uc IN 1..1 INBOUND c._id hasConversations
|
|
70
|
+
FILTER u._id != "${sessionDocId}"
|
|
71
|
+
RETURN u
|
|
72
|
+
)
|
|
73
|
+
RETURN MERGE(c, {users: users})`;
|
|
74
|
+
return database.query(aqlQry).then((cursor) => cursor.all()).catch((error) => (0, import_analytics.logError)({
|
|
75
|
+
action,
|
|
76
|
+
category: eventCategory,
|
|
77
|
+
label: "db_error"
|
|
78
|
+
}, error, context).then(() => null));
|
|
79
|
+
};
|
|
80
|
+
const getDirectConversation = async (context, fromId, toId) => {
|
|
81
|
+
const action = "getDirectConversation";
|
|
82
|
+
const { database } = context;
|
|
83
|
+
try {
|
|
84
|
+
const formatFromId = (0, import_utils.parseId)(fromId);
|
|
85
|
+
const formatToId = (0, import_utils.parseId)(toId);
|
|
86
|
+
const userIds = [formatFromId, formatToId];
|
|
87
|
+
const sortedUserIds = userIds.sort((a, b) => {
|
|
88
|
+
if (a < b) {
|
|
89
|
+
return -1;
|
|
90
|
+
}
|
|
91
|
+
if (a > b) {
|
|
92
|
+
return 1;
|
|
93
|
+
}
|
|
94
|
+
return 0;
|
|
95
|
+
});
|
|
96
|
+
const conversationId = (0, import_utils.createHash)(`conversations-${sortedUserIds.join("-")}`, null);
|
|
97
|
+
const user = await (0, import_users.getUser)(context, { userId: formatToId });
|
|
98
|
+
const aqlQry = import_arangojs.aql`LET c = DOCUMENT(${`conversations/${conversationId}`}) RETURN c`;
|
|
99
|
+
const conversation = await database.query(aqlQry).then((cursor) => cursor.next()).catch((error) => (0, import_analytics.logError)({
|
|
100
|
+
action,
|
|
101
|
+
category: eventCategory,
|
|
102
|
+
label: "db_error"
|
|
103
|
+
}, error, context).then(() => null));
|
|
104
|
+
if (conversation) {
|
|
105
|
+
return __spreadProps(__spreadValues({}, conversation), { users: [user] });
|
|
106
|
+
}
|
|
107
|
+
const now = Date.now();
|
|
108
|
+
const insert = {
|
|
109
|
+
_key: conversationId,
|
|
110
|
+
added: now,
|
|
111
|
+
isDirect: true,
|
|
112
|
+
modified: now
|
|
113
|
+
};
|
|
114
|
+
const insertQuery = import_arangojs.aql`INSERT ${insert} IN conversations RETURN NEW`;
|
|
115
|
+
const updatedConversation = await database.query(insertQuery).then((cursor) => cursor.next());
|
|
116
|
+
await Promise.all(userIds.map((userId) => {
|
|
117
|
+
const edgeId = (0, import_utils.createHash)(`hasConversations-${userId}-${conversationId}`);
|
|
118
|
+
const edge = {
|
|
119
|
+
_from: `users/${userId}`,
|
|
120
|
+
_key: edgeId,
|
|
121
|
+
_to: `conversations/${conversationId}`,
|
|
122
|
+
added: Date.now()
|
|
123
|
+
};
|
|
124
|
+
const insertEdgeQuery = import_arangojs.aql`INSERT ${edge} IN hasConversations RETURN NEW`;
|
|
125
|
+
return database.query(insertEdgeQuery);
|
|
126
|
+
}));
|
|
127
|
+
return __spreadProps(__spreadValues({}, updatedConversation), { users: [user] });
|
|
128
|
+
} catch (updateError) {
|
|
129
|
+
return (0, import_analytics.logError)({
|
|
130
|
+
action,
|
|
131
|
+
category: eventCategory,
|
|
132
|
+
label: "db_error"
|
|
133
|
+
}, updateError, context).then(() => null);
|
|
134
|
+
}
|
|
135
|
+
};
|
|
136
|
+
const getConnectionUsers = (context, conversationId, includeAll = false, isDirect) => {
|
|
137
|
+
const action = "getConversationUsers";
|
|
138
|
+
const { database, session: { userId: sessionUserId } } = context;
|
|
139
|
+
const filters = ["c.userId == u._key"];
|
|
140
|
+
if (!includeAll) {
|
|
141
|
+
filters.push(`u._key != ${sessionUserId}`);
|
|
142
|
+
}
|
|
143
|
+
if (isDirect) {
|
|
144
|
+
filters.push(`p.vertices[0].isDirect == ${isDirect}`);
|
|
145
|
+
}
|
|
146
|
+
const aqlQry = import_arangojs.aql`FOR u, e,p IN 1..1 INBOUND ${`conversations/${conversationId}`} hasConversations
|
|
147
|
+
OPTIONS {vertexCollections: "users"}
|
|
148
|
+
FOR c IN connections
|
|
149
|
+
FILTER ${filters.join(" && ")}
|
|
150
|
+
RETURN c`;
|
|
151
|
+
console.log({ aqlQry });
|
|
152
|
+
return database.query(aqlQry).then((cursor) => cursor.all()).catch((error) => (0, import_analytics.logError)({
|
|
153
|
+
action,
|
|
154
|
+
category: eventCategory,
|
|
155
|
+
label: "db_error"
|
|
156
|
+
}, error, context).then(() => null));
|
|
157
|
+
};
|
|
158
|
+
const getConversationUsers = (context, conversationId, includeAll = false, isDirect) => {
|
|
159
|
+
const action = "getConversationUsers";
|
|
160
|
+
const { database, session: { userId: sessionUserId } } = context;
|
|
161
|
+
const filters = ["c.userId == u._key"];
|
|
162
|
+
if (!includeAll) {
|
|
163
|
+
filters.push(`u._key != ${sessionUserId}`);
|
|
164
|
+
}
|
|
165
|
+
if (isDirect) {
|
|
166
|
+
filters.push(`p.vertices[0].isDirect == ${isDirect}`);
|
|
167
|
+
}
|
|
168
|
+
const aqlQry = import_arangojs.aql`FOR u, e,p IN 1..1 INBOUND ${`conversations/${conversationId}`} hasConversations
|
|
169
|
+
OPTIONS {vertexCollections: "users"}
|
|
170
|
+
FILTER ${filters.join(" && ")}
|
|
171
|
+
RETURN u`;
|
|
172
|
+
console.log({ aqlQry });
|
|
173
|
+
return database.query(aqlQry).then((cursor) => cursor.all()).catch((error) => (0, import_analytics.logError)({
|
|
174
|
+
action,
|
|
175
|
+
category: eventCategory,
|
|
176
|
+
label: "db_error"
|
|
177
|
+
}, error, context).then(() => null));
|
|
178
|
+
};
|
|
179
|
+
const getDirectConversationOld = (context, userId) => {
|
|
180
|
+
const action = "getDirectConversation";
|
|
181
|
+
const { database, session: { userId: sessionId } } = context;
|
|
182
|
+
const sessionDocId = `users/${sessionId}`;
|
|
183
|
+
const formatUserId = (0, import_utils.parseArangoId)(userId);
|
|
184
|
+
const aqlQry = import_arangojs.aql`LET from = (
|
|
185
|
+
FOR c, e IN 1..1 OUTBOUND ${sessionDocId} hasConversations
|
|
186
|
+
FILTER c.direct == true
|
|
187
|
+
RETURN c
|
|
188
|
+
)
|
|
189
|
+
LET to = (
|
|
190
|
+
FOR c, e IN 1..1 OUTBOUND ${formatUserId} hasConversations
|
|
191
|
+
FILTER c.direct == true
|
|
192
|
+
RETURN c
|
|
193
|
+
)
|
|
194
|
+
LET conversation = FIRST(INTERSECTION(from, to))
|
|
195
|
+
LET users = !IS_NULL(conversation) && (
|
|
196
|
+
FOR g IN[${sessionDocId}, ${formatUserId}]
|
|
197
|
+
FOR u IN users
|
|
198
|
+
FILTER g == u._id
|
|
199
|
+
RETURN u
|
|
200
|
+
)
|
|
201
|
+
RETURN IS_NULL(conversation) ? null : MERGE(conversation, {users: users})`;
|
|
202
|
+
return database.query(aqlQry).then((cursor) => cursor.next()).catch((error) => (0, import_analytics.logError)({
|
|
203
|
+
action,
|
|
204
|
+
category: eventCategory,
|
|
205
|
+
label: "db_error"
|
|
206
|
+
}, error, context).then(() => null));
|
|
207
|
+
};
|
|
208
|
+
const getConversation = (context, convoId) => {
|
|
209
|
+
const action = "getItem";
|
|
210
|
+
const { database, session: { userId: sessionId } } = context;
|
|
211
|
+
const sessionDocId = `users/${sessionId}`;
|
|
212
|
+
const convoDocId = `conversations/${(0, import_utils.parseId)(convoId)}`;
|
|
213
|
+
const aqlQry = import_arangojs.aql`FOR c, e IN 1..1 OUTBOUND ${sessionDocId} hasConversations
|
|
214
|
+
FILTER c._id == ${convoDocId}
|
|
215
|
+
LET u = (
|
|
216
|
+
FOR inUser, inEdge IN 1..1 INBOUND ${convoDocId} hasConversations
|
|
217
|
+
RETURN inUser
|
|
218
|
+
)
|
|
219
|
+
LIMIT 1
|
|
220
|
+
RETURN MERGE(c, {users: u})`;
|
|
221
|
+
return database.query(aqlQry).then((cursor) => cursor.next()).then((conversation = {}) => {
|
|
222
|
+
if (!(0, import_isEmpty.default)(conversation)) {
|
|
223
|
+
conversation.users = conversation.users.map((user) => {
|
|
224
|
+
const { imageId, userId } = user;
|
|
225
|
+
const thumbUrlData = {
|
|
226
|
+
imageId,
|
|
227
|
+
isThumb: true,
|
|
228
|
+
userId
|
|
229
|
+
};
|
|
230
|
+
return __spreadProps(__spreadValues({}, user), {
|
|
231
|
+
name: (0, import_users.getDisplayName)(user),
|
|
232
|
+
thumb: (0, import_images.getUserImageUrl)(thumbUrlData)
|
|
233
|
+
});
|
|
234
|
+
});
|
|
235
|
+
return conversation;
|
|
236
|
+
}
|
|
237
|
+
return {};
|
|
238
|
+
}).catch((error) => (0, import_analytics.logError)({
|
|
239
|
+
action,
|
|
240
|
+
category: eventCategory,
|
|
241
|
+
label: "db_error"
|
|
242
|
+
}, error, context).then(() => null));
|
|
243
|
+
};
|
|
244
|
+
const updateConversation = (context, conversation) => {
|
|
245
|
+
const action = "updateConversation";
|
|
246
|
+
const { database, session: { userId: sessionId } } = context;
|
|
247
|
+
const {
|
|
248
|
+
conversationId = (0, import_utils.createHash)(`conversation-${sessionId} `),
|
|
249
|
+
direct = false,
|
|
250
|
+
name
|
|
251
|
+
} = conversation;
|
|
252
|
+
const formatConversationId = (0, import_utils.parseId)(conversationId);
|
|
253
|
+
const now = Date.now();
|
|
254
|
+
const update = {
|
|
255
|
+
direct,
|
|
256
|
+
modified: now,
|
|
257
|
+
name
|
|
258
|
+
};
|
|
259
|
+
const insert = __spreadProps(__spreadValues({}, (0, import_cloneDeep.default)(update)), {
|
|
260
|
+
_key: formatConversationId,
|
|
261
|
+
added: now
|
|
262
|
+
});
|
|
263
|
+
const aqlQry = import_arangojs.aql`UPSERT {_key: ${formatConversationId} }
|
|
264
|
+
INSERT ${insert}
|
|
265
|
+
UPDATE ${update}
|
|
266
|
+
IN conversations RETURN NEW`;
|
|
267
|
+
return database.query(aqlQry).then((cursor) => cursor.next()).then((conversation2 = {}) => conversation2).catch((error) => (0, import_analytics.logError)({
|
|
268
|
+
action,
|
|
269
|
+
category: eventCategory,
|
|
270
|
+
label: "db_error"
|
|
271
|
+
}, error, context).then(() => null));
|
|
272
|
+
};
|
|
273
|
+
const addUserToConversation = (context, convoId, userId) => {
|
|
274
|
+
const action = "addUserToConversation";
|
|
275
|
+
const { database } = context;
|
|
276
|
+
const formatConvoId = (0, import_utils.parseId)(convoId);
|
|
277
|
+
const formatUserId = (0, import_utils.parseId)(userId);
|
|
278
|
+
const convoDocId = `conversations/${formatConvoId} `;
|
|
279
|
+
const userDocId = `users/${formatUserId} `;
|
|
280
|
+
const aqlQry = import_arangojs.aql`FOR c IN hasConversations
|
|
281
|
+
FILTER c._from == ${userDocId} && c._to == ${convoDocId}
|
|
282
|
+
LIMIT 1
|
|
283
|
+
RETURN c`;
|
|
284
|
+
return database.query(aqlQry).then((cursor) => cursor.next()).then((conversationEdge) => {
|
|
285
|
+
if (!!conversationEdge) {
|
|
286
|
+
return conversationEdge;
|
|
287
|
+
}
|
|
288
|
+
const edgeCollection = database.collection("hasConversations");
|
|
289
|
+
const edgeId = (0, import_utils.createHash)(`convo - ${formatConvoId} -${formatUserId} `);
|
|
290
|
+
const edge = {
|
|
291
|
+
_from: userDocId,
|
|
292
|
+
_key: edgeId,
|
|
293
|
+
_to: convoDocId,
|
|
294
|
+
added: Date.now()
|
|
295
|
+
};
|
|
296
|
+
return edgeCollection.save(edge, { returnNew: true }).then(() => edge);
|
|
297
|
+
}).catch((error) => (0, import_analytics.logError)({
|
|
298
|
+
action,
|
|
299
|
+
category: eventCategory,
|
|
300
|
+
label: "db_error"
|
|
301
|
+
}, error, context).then(() => null));
|
|
302
|
+
};
|
|
303
|
+
const deleteUserFromConversation = (context, convoId, userId) => {
|
|
304
|
+
const action = "deleteUserFromConversation";
|
|
305
|
+
const { database } = context;
|
|
306
|
+
const formatConvoId = (0, import_utils.parseId)(convoId);
|
|
307
|
+
const formatUserId = (0, import_utils.parseId)(userId);
|
|
308
|
+
const convoDocId = `conversations / ${formatConvoId} `;
|
|
309
|
+
const userDocId = `users/${formatUserId} `;
|
|
310
|
+
const aqlQry = import_arangojs.aql`FOR c IN hasConversations
|
|
311
|
+
FILTER c._from == ${userDocId} && c._to == ${convoDocId}
|
|
312
|
+
LIMIT 1
|
|
313
|
+
REMOVE c IN hasConversations
|
|
314
|
+
RETURN OLD`;
|
|
315
|
+
return database.query(aqlQry).then((cursor) => cursor.next()).catch((error) => (0, import_analytics.logError)({
|
|
316
|
+
action,
|
|
317
|
+
category: eventCategory,
|
|
318
|
+
label: "db_error"
|
|
319
|
+
}, error, context).then(() => null));
|
|
320
|
+
};
|
|
321
|
+
// Annotate the CommonJS export names for ESM import in node:
|
|
322
|
+
0 && (module.exports = {
|
|
323
|
+
addUserToConversation,
|
|
324
|
+
deleteUserFromConversation,
|
|
325
|
+
getConnectionUsers,
|
|
326
|
+
getConversation,
|
|
327
|
+
getConversationUsers,
|
|
328
|
+
getConversations,
|
|
329
|
+
getDirectConversation,
|
|
330
|
+
getDirectConversationOld,
|
|
331
|
+
updateConversation
|
|
332
|
+
});
|
|
333
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../src/actions/conversations.ts"],
  "sourcesContent": ["/**\n * Copyright (c) 2019-Present, Nitrogen Labs, Inc.\n * Copyrights licensed under the MIT License. See the accompanying LICENSE file for terms.\n */\nimport {createHash, parseArangoId, parseId} from '@nlabs/utils';\nimport {aql} from 'arangojs';\nimport {AqlQuery} from 'arangojs/aql';\nimport {EdgeCollection} from 'arangojs/collection';\nimport {ArrayCursor} from 'arangojs/cursor';\nimport cloneDeep from 'lodash/cloneDeep';\nimport isEmpty from 'lodash/isEmpty';\nimport {ApiContext} from 'types/auth';\n\nimport {ArangoDBLimit, ConversationType, ImageUrlData, User} from '../types';\nimport {getLimit} from '../utils';\nimport {logError} from '../utils/analytics';\nimport {getUserImageUrl} from './images';\nimport {getDisplayName, getUser} from './users';\n\nconst eventCategory: string = 'conversations';\n\nexport const getConversations = (context: ApiContext, {from, to}): Promise<ConversationType[]> => {\n  const action: string = 'getConversations';\n  const {database, session: {userId: sessionId}} = context;\n  const limit: ArangoDBLimit = getLimit(from, to);\n  const sessionDocId: string = `users/${sessionId}`;\n  const aqlQry: string = `FOR c, e IN 1..1 OUTBOUND \"${sessionDocId}\" hasConversations\n    ${limit.aql}\n    LET users = (\n      FOR u, uc IN 1..1 INBOUND c._id hasConversations\n      FILTER u._id != \"${sessionDocId}\"\n      RETURN u\n    )\n    RETURN MERGE(c, {users: users})`;\n\n  return database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, context).then(() => null));\n};\n\nexport const getDirectConversation = async (\n  context: ApiContext,\n  fromId: string,\n  toId: string\n): Promise<ConversationType> => {\n  const action: string = 'getDirectConversation';\n  const {database} = context;\n\n  try {\n    const formatFromId: string = parseId(fromId);\n    const formatToId: string = parseId(toId);\n    const userIds: string[] = [formatFromId, formatToId];\n    const sortedUserIds: string[] = userIds.sort((a, b) => {\n      if(a < b) {\n        return -1;\n      }\n      if(a > b) {\n        return 1;\n      }\n      return 0;\n    });\n    const conversationId: string = createHash(`conversations-${sortedUserIds.join('-')}`, null);\n    const user: User = await getUser(context, {userId: formatToId});\n\n    const aqlQry: AqlQuery = aql`LET c = DOCUMENT(${`conversations/${conversationId}`}) RETURN c`;\n    const conversation = await database.query(aqlQry)\n      .then((cursor: ArrayCursor) => cursor.next())\n      .catch((error: Error) => logError({\n        action,\n        category: eventCategory,\n        label: 'db_error'\n      }, error, context).then(() => null));\n\n    if(conversation) {\n      return {...conversation, users: [user]};\n    }\n\n    const now: number = Date.now();\n    const insert = {\n      _key: conversationId,\n      added: now,\n      isDirect: true,\n      modified: now\n    };\n    const insertQuery: AqlQuery = aql`INSERT ${insert} IN conversations RETURN NEW`;\n    const updatedConversation = await database.query(insertQuery)\n      .then((cursor: ArrayCursor) => cursor.next());\n\n    await Promise.all(userIds.map((userId) => {\n      const edgeId: string = createHash(`hasConversations-${userId}-${conversationId}`);\n      const edge: any = {\n        _from: `users/${userId}`,\n        _key: edgeId,\n        _to: `conversations/${conversationId}`,\n        added: Date.now()\n      };\n      const insertEdgeQuery: AqlQuery = aql`INSERT ${edge} IN hasConversations RETURN NEW`;\n      return database.query(insertEdgeQuery);\n    }));\n\n    return {...updatedConversation, users: [user]};\n  } catch(updateError) {\n    return logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, updateError, context).then(() => null);\n  }\n};\n\nexport const getConnectionUsers = (\n  context: ApiContext,\n  conversationId: string,\n  includeAll: boolean = false,\n  isDirect?: boolean\n): Promise<ConversationType[]> => {\n  const action: string = 'getConversationUsers';\n  const {database, session: {userId: sessionUserId}} = context;\n  const filters: string[] = ['c.userId == u._key'];\n\n  if(!includeAll) {\n    filters.push(`u._key != ${sessionUserId}`);\n  }\n  if(isDirect) {\n    filters.push(`p.vertices[0].isDirect == ${isDirect}`);\n  }\n\n  const aqlQry: AqlQuery = aql`FOR u, e,p IN 1..1 INBOUND ${`conversations/${conversationId}`} hasConversations\n    OPTIONS {vertexCollections: \"users\"}\n    FOR c IN connections\n    FILTER ${filters.join(' && ')}\n    RETURN c`;\n\n  console.log({aqlQry});\n  return database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, context).then(() => null));\n};\n\nexport const getConversationUsers = (\n  context: ApiContext,\n  conversationId: string,\n  includeAll: boolean = false,\n  isDirect?: boolean\n): Promise<ConversationType[]> => {\n  const action: string = 'getConversationUsers';\n  const {database, session: {userId: sessionUserId}} = context;\n  const filters: string[] = ['c.userId == u._key'];\n\n  if(!includeAll) {\n    filters.push(`u._key != ${sessionUserId}`);\n  }\n  if(isDirect) {\n    filters.push(`p.vertices[0].isDirect == ${isDirect}`);\n  }\n\n  const aqlQry: AqlQuery = aql`FOR u, e,p IN 1..1 INBOUND ${`conversations/${conversationId}`} hasConversations\n    OPTIONS {vertexCollections: \"users\"}\n    FILTER ${filters.join(' && ')}\n    RETURN u`;\n\n  console.log({aqlQry});\n  return database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, context).then(() => null));\n};\n\nexport const getDirectConversationOld = (context: ApiContext, userId: string): Promise<ConversationType> => {\n  const action: string = 'getDirectConversation';\n  const {database, session: {userId: sessionId}} = context;\n  const sessionDocId: string = `users/${sessionId}`;\n  const formatUserId: string = parseArangoId(userId);\n\n  const aqlQry: AqlQuery = aql`LET from = (\n      FOR c, e IN 1..1 OUTBOUND ${sessionDocId} hasConversations\n      FILTER c.direct == true\n      RETURN c\n    )\n    LET to = (\n      FOR c, e IN 1..1 OUTBOUND ${formatUserId} hasConversations\n      FILTER c.direct == true\n      RETURN c\n    )\n    LET conversation = FIRST(INTERSECTION(from, to))\n    LET users = !IS_NULL(conversation) && (\n      FOR g IN[${sessionDocId}, ${formatUserId}]\n      FOR u IN users\n      FILTER g == u._id\n      RETURN u\n    )\n    RETURN IS_NULL(conversation) ? null : MERGE(conversation, {users: users})`;\n\n  return database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, context).then(() => null));\n};\n\nexport const getConversation = (context: ApiContext, convoId: string): Promise<ConversationType> => {\n  const action: string = 'getItem';\n  const {database, session: {userId: sessionId}} = context;\n  const sessionDocId: string = `users/${sessionId}`;\n  const convoDocId: string = `conversations/${parseId(convoId)}`;\n\n  const aqlQry: AqlQuery = aql`FOR c, e IN 1..1 OUTBOUND ${sessionDocId} hasConversations\n    FILTER c._id == ${convoDocId}\n    LET u = (\n      FOR inUser, inEdge IN 1..1 INBOUND ${convoDocId} hasConversations\n      RETURN inUser\n    )\n    LIMIT 1\n    RETURN MERGE(c, {users: u})`;\n\n  return database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((conversation: ConversationType = {}) => {\n      if(!isEmpty(conversation)) {\n        conversation.users = conversation.users.map((user: any) => {\n          const {imageId, userId} = user;\n          const thumbUrlData: ImageUrlData = {\n            imageId,\n            isThumb: true,\n            userId\n          };\n          return {\n            ...user,\n            name: getDisplayName(user),\n            thumb: getUserImageUrl(thumbUrlData)\n          };\n        });\n\n        return conversation;\n      }\n      return {};\n    })\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, context).then(() => null));\n};\n\nexport const updateConversation = (context: ApiContext, conversation): Promise<ConversationType> => {\n  const action: string = 'updateConversation';\n  const {database, session: {userId: sessionId}} = context;\n  const {\n    conversationId = createHash(`conversation-${sessionId} `),\n    direct = false,\n    name\n  } = conversation;\n  const formatConversationId: string = parseId(conversationId);\n  const now: number = Date.now();\n  const update: any = {\n    // Direct message\n    direct,\n    modified: now,\n    name\n  };\n\n  const insert: any = {\n    ...cloneDeep(update),\n    _key: formatConversationId,\n    added: now\n  };\n\n  const aqlQry: AqlQuery = aql`UPSERT {_key: ${formatConversationId} }\n    INSERT ${insert}\n    UPDATE ${update}\n    IN conversations RETURN NEW`;\n\n  return database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((conversation: ConversationType = {}) => conversation)\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, context).then(() => null));\n};\n\nexport const addUserToConversation = (context: ApiContext, convoId: string, userId: string) => {\n  const action: string = 'addUserToConversation';\n  const {database} = context;\n  const formatConvoId: string = parseId(convoId);\n  const formatUserId: string = parseId(userId);\n  const convoDocId: string = `conversations/${formatConvoId} `;\n  const userDocId: string = `users/${formatUserId} `;\n\n  const aqlQry: AqlQuery = aql`FOR c IN hasConversations\n    FILTER c._from == ${userDocId} && c._to == ${convoDocId}\n    LIMIT 1\n    RETURN c`;\n\n  return database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((conversationEdge) => {\n      if(!!conversationEdge) {\n        return conversationEdge;\n      }\n\n      const edgeCollection: EdgeCollection = database.collection('hasConversations');\n      const edgeId: string = createHash(`convo - ${formatConvoId} -${formatUserId} `);\n\n      const edge: ConversationType = {\n        _from: userDocId,\n        _key: edgeId,\n        _to: convoDocId,\n        added: Date.now()\n      };\n\n      return edgeCollection.save(edge, {returnNew: true}).then(() => edge);\n    })\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, context).then(() => null));\n};\n\nexport const deleteUserFromConversation = (context: ApiContext, convoId: string, userId: string) => {\n  const action: string = 'deleteUserFromConversation';\n  const {database} = context;\n  const formatConvoId: string = parseId(convoId);\n  const formatUserId: string = parseId(userId);\n  const convoDocId: string = `conversations / ${formatConvoId} `;\n  const userDocId: string = `users/${formatUserId} `;\n\n  const aqlQry: AqlQuery = aql`FOR c IN hasConversations\n    FILTER c._from == ${userDocId} && c._to == ${convoDocId}\n    LIMIT 1\n    REMOVE c IN hasConversations\n    RETURN OLD`;\n\n  return database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, context).then(() => null));\n};\n"],
  "mappings": "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",
  "names": []
}

|
|
@@ -0,0 +1,155 @@
|
|
|
1
|
+
var __create = Object.create;
|
|
2
|
+
var __defProp = Object.defineProperty;
|
|
3
|
+
var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
|
|
4
|
+
var __getOwnPropNames = Object.getOwnPropertyNames;
|
|
5
|
+
var __getOwnPropSymbols = Object.getOwnPropertySymbols;
|
|
6
|
+
var __getProtoOf = Object.getPrototypeOf;
|
|
7
|
+
var __hasOwnProp = Object.prototype.hasOwnProperty;
|
|
8
|
+
var __propIsEnum = Object.prototype.propertyIsEnumerable;
|
|
9
|
+
var __defNormalProp = (obj, key, value) => key in obj ? __defProp(obj, key, { enumerable: true, configurable: true, writable: true, value }) : obj[key] = value;
|
|
10
|
+
var __spreadValues = (a, b) => {
|
|
11
|
+
for (var prop in b || (b = {}))
|
|
12
|
+
if (__hasOwnProp.call(b, prop))
|
|
13
|
+
__defNormalProp(a, prop, b[prop]);
|
|
14
|
+
if (__getOwnPropSymbols)
|
|
15
|
+
for (var prop of __getOwnPropSymbols(b)) {
|
|
16
|
+
if (__propIsEnum.call(b, prop))
|
|
17
|
+
__defNormalProp(a, prop, b[prop]);
|
|
18
|
+
}
|
|
19
|
+
return a;
|
|
20
|
+
};
|
|
21
|
+
var __markAsModule = (target) => __defProp(target, "__esModule", { value: true });
|
|
22
|
+
var __export = (target, all) => {
|
|
23
|
+
__markAsModule(target);
|
|
24
|
+
for (var name in all)
|
|
25
|
+
__defProp(target, name, { get: all[name], enumerable: true });
|
|
26
|
+
};
|
|
27
|
+
var __reExport = (target, module2, desc) => {
|
|
28
|
+
if (module2 && typeof module2 === "object" || typeof module2 === "function") {
|
|
29
|
+
for (let key of __getOwnPropNames(module2))
|
|
30
|
+
if (!__hasOwnProp.call(target, key) && key !== "default")
|
|
31
|
+
__defProp(target, key, { get: () => module2[key], enumerable: !(desc = __getOwnPropDesc(module2, key)) || desc.enumerable });
|
|
32
|
+
}
|
|
33
|
+
return target;
|
|
34
|
+
};
|
|
35
|
+
var __toModule = (module2) => {
|
|
36
|
+
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);
|
|
37
|
+
};
|
|
38
|
+
__export(exports, {
|
|
39
|
+
dynamoDel: () => dynamoDel,
|
|
40
|
+
dynamoGet: () => dynamoGet,
|
|
41
|
+
dynamoGetList: () => dynamoGetList,
|
|
42
|
+
dynamoPut: () => dynamoPut,
|
|
43
|
+
dynamoPutList: () => dynamoPutList,
|
|
44
|
+
dynamoUpdate: () => dynamoUpdate,
|
|
45
|
+
parseDynamo: () => parseDynamo
|
|
46
|
+
});
|
|
47
|
+
var import_aws_sdk = __toModule(require("aws-sdk"));
|
|
48
|
+
var import_config = __toModule(require("../config"));
|
|
49
|
+
const dynamoGet = (params) => new Promise((resolve, reject) => {
|
|
50
|
+
import_aws_sdk.default.config.update(import_config.Config.get("aws"));
|
|
51
|
+
const dynamodb = new import_aws_sdk.DynamoDB();
|
|
52
|
+
params.ConsistentRead = false;
|
|
53
|
+
dynamodb.getItem(params, (error, results) => {
|
|
54
|
+
if (error) {
|
|
55
|
+
return reject(error);
|
|
56
|
+
}
|
|
57
|
+
return resolve(results);
|
|
58
|
+
});
|
|
59
|
+
});
|
|
60
|
+
const parseDynamo = (results) => {
|
|
61
|
+
const { Items: items } = results;
|
|
62
|
+
const data = [];
|
|
63
|
+
for (let idx = 0, len = items.length; idx < len; idx++) {
|
|
64
|
+
const tmp = items[idx];
|
|
65
|
+
const obj = {};
|
|
66
|
+
let key;
|
|
67
|
+
for (key in tmp) {
|
|
68
|
+
if (tmp.hasOwnProperty(key)) {
|
|
69
|
+
if (tmp[key].S) {
|
|
70
|
+
obj[key] = tmp[key].S.toString();
|
|
71
|
+
} else if (tmp[key].N) {
|
|
72
|
+
obj[key] = parseFloat(tmp[key].N);
|
|
73
|
+
} else {
|
|
74
|
+
obj[key] = tmp[key].B;
|
|
75
|
+
}
|
|
76
|
+
}
|
|
77
|
+
}
|
|
78
|
+
data.push(obj);
|
|
79
|
+
}
|
|
80
|
+
return data;
|
|
81
|
+
};
|
|
82
|
+
const dynamoGetList = (params, index) => new Promise((resolve, reject) => {
|
|
83
|
+
import_aws_sdk.default.config.update(import_config.Config.get("aws"));
|
|
84
|
+
const dynamodb = new import_aws_sdk.DynamoDB();
|
|
85
|
+
const { ConsistentRead: consistentRead } = params;
|
|
86
|
+
const updatedParams = __spreadValues({}, params);
|
|
87
|
+
if (index) {
|
|
88
|
+
updatedParams.ExclusiveStartKey = index;
|
|
89
|
+
}
|
|
90
|
+
if (!consistentRead) {
|
|
91
|
+
updatedParams.ConsistentRead = false;
|
|
92
|
+
}
|
|
93
|
+
dynamodb.query(updatedParams, (error, output) => {
|
|
94
|
+
if (error) {
|
|
95
|
+
return reject(error);
|
|
96
|
+
}
|
|
97
|
+
const list = parseDynamo(output);
|
|
98
|
+
const { LastEvaluatedKey: lastKey } = output;
|
|
99
|
+
if (lastKey !== void 0) {
|
|
100
|
+
return dynamoGetList(params, lastKey);
|
|
101
|
+
}
|
|
102
|
+
return resolve(list);
|
|
103
|
+
});
|
|
104
|
+
});
|
|
105
|
+
const dynamoPut = (params) => new Promise((resolve, reject) => {
|
|
106
|
+
import_aws_sdk.default.config.update(import_config.Config.get("aws"));
|
|
107
|
+
const dynamodb = new import_aws_sdk.DynamoDB();
|
|
108
|
+
dynamodb.putItem(params, (error, output) => {
|
|
109
|
+
if (error) {
|
|
110
|
+
return reject(error);
|
|
111
|
+
}
|
|
112
|
+
return resolve(output);
|
|
113
|
+
});
|
|
114
|
+
});
|
|
115
|
+
const dynamoUpdate = (params) => new Promise((resolve, reject) => {
|
|
116
|
+
import_aws_sdk.default.config.update(import_config.Config.get("aws"));
|
|
117
|
+
const dynamodb = new import_aws_sdk.DynamoDB();
|
|
118
|
+
dynamodb.updateItem(params, (error, output) => {
|
|
119
|
+
if (error) {
|
|
120
|
+
return reject(error);
|
|
121
|
+
}
|
|
122
|
+
return resolve(output);
|
|
123
|
+
});
|
|
124
|
+
});
|
|
125
|
+
const dynamoPutList = (params) => new Promise((resolve, reject) => {
|
|
126
|
+
import_aws_sdk.default.config.update(import_config.Config.get("aws"));
|
|
127
|
+
const dynamodb = new import_aws_sdk.DynamoDB();
|
|
128
|
+
dynamodb.batchWriteItem(params, (error, output) => {
|
|
129
|
+
if (error) {
|
|
130
|
+
return reject(error);
|
|
131
|
+
}
|
|
132
|
+
return resolve(output);
|
|
133
|
+
});
|
|
134
|
+
});
|
|
135
|
+
const dynamoDel = (params) => new Promise((resolve, reject) => {
|
|
136
|
+
import_aws_sdk.default.config.update(import_config.Config.get("aws"));
|
|
137
|
+
const dynamodb = new import_aws_sdk.DynamoDB();
|
|
138
|
+
dynamodb.deleteItem(params, (error, output) => {
|
|
139
|
+
if (error) {
|
|
140
|
+
return reject(error);
|
|
141
|
+
}
|
|
142
|
+
return resolve(output);
|
|
143
|
+
});
|
|
144
|
+
});
|
|
145
|
+
// Annotate the CommonJS export names for ESM import in node:
|
|
146
|
+
0 && (module.exports = {
|
|
147
|
+
dynamoDel,
|
|
148
|
+
dynamoGet,
|
|
149
|
+
dynamoGetList,
|
|
150
|
+
dynamoPut,
|
|
151
|
+
dynamoPutList,
|
|
152
|
+
dynamoUpdate,
|
|
153
|
+
parseDynamo
|
|
154
|
+
});
|
|
155
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../src/actions/dynamodb.ts"],
  "sourcesContent": ["/**\n * Copyright (c) 2019-Present, Nitrogen Labs, Inc.\n * Copyrights licensed under the MIT License. See the accompanying LICENSE file for terms.\n */\nimport aws, {DynamoDB} from 'aws-sdk';\nimport {\n  BatchWriteItemInput,\n  BatchWriteItemOutput,\n  DeleteItemInput,\n  DeleteItemOutput,\n  GetItemInput,\n  GetItemOutput,\n  PutItemInput,\n  PutItemOutput,\n  QueryInput,\n  QueryOutput,\n  UpdateItemInput,\n  UpdateItemOutput\n} from 'aws-sdk/clients/dynamodb';\n\nimport {Config} from '../config';\n\n// const eventCategory: string = 'dynamodb';\n\nexport const dynamoGet = (params: GetItemInput): Promise<GetItemOutput> => new Promise((resolve, reject) => {\n  aws.config.update(Config.get('aws'));\n  const dynamodb: DynamoDB = new DynamoDB();\n  params.ConsistentRead = false;\n\n  dynamodb.getItem(params, (error: Error, results: GetItemOutput) => {\n    if(error) {\n      return reject(error);\n    }\n\n    return resolve(results);\n  });\n});\n\nexport const parseDynamo = (results: QueryOutput): any[] => {\n  const {Items: items} = results;\n  const data: any[] = [];\n\n  for(let idx: number = 0, len = items.length; idx < len; idx++) {\n    const tmp = items[idx];\n    const obj = {};\n    let key;\n\n    for(key in tmp) {\n      if(tmp.hasOwnProperty(key)) {\n        if(tmp[key].S) {\n          obj[key] = tmp[key].S.toString();\n        } else if(tmp[key].N) {\n          obj[key] = parseFloat(tmp[key].N);\n        } else {\n          obj[key] = tmp[key].B;\n        }\n      }\n    }\n\n    data.push(obj);\n  }\n\n  return data;\n};\n\nexport const dynamoGetList = (params: QueryInput, index): Promise<GetItemOutput[]> => new Promise((resolve, reject) => {\n  aws.config.update(Config.get('aws'));\n  const dynamodb: DynamoDB = new DynamoDB();\n  const {ConsistentRead: consistentRead} = params;\n  const updatedParams = {...params};\n\n  if(index) {\n    updatedParams.ExclusiveStartKey = index;\n  }\n\n  if(!consistentRead) {\n    updatedParams.ConsistentRead = false;\n  }\n\n  dynamodb.query(updatedParams, (error: Error, output: QueryOutput) => {\n    if(error) {\n      return reject(error);\n    }\n    // Save list of items\n    const list: any[] = parseDynamo(output);\n    const {LastEvaluatedKey: lastKey} = output;\n\n    if(lastKey !== undefined) {\n      // Save last index before looping\n      return dynamoGetList(params, lastKey);\n    }\n\n    // Query complete, return final list\n    return resolve(list);\n  });\n});\n\nexport const dynamoPut = (params: PutItemInput): Promise<PutItemOutput> => new Promise((resolve, reject) => {\n  aws.config.update(Config.get('aws'));\n  const dynamodb: DynamoDB = new DynamoDB();\n\n  dynamodb.putItem(params, (error, output: PutItemOutput) => {\n    if(error) {\n      return reject(error);\n    }\n    return resolve(output);\n  });\n});\n\nexport const dynamoUpdate = (params: UpdateItemInput): Promise<UpdateItemOutput> => new Promise((resolve, reject) => {\n  aws.config.update(Config.get('aws'));\n  const dynamodb: DynamoDB = new DynamoDB();\n\n  dynamodb.updateItem(params, (error: Error, output: UpdateItemOutput) => {\n    if(error) {\n      return reject(error);\n    }\n    return resolve(output);\n  });\n});\n\nexport const dynamoPutList = (params: BatchWriteItemInput): Promise<BatchWriteItemOutput> =>\n  new Promise((resolve, reject) => {\n    aws.config.update(Config.get('aws'));\n    const dynamodb: DynamoDB = new DynamoDB();\n\n    dynamodb.batchWriteItem(params, (error: Error, output: BatchWriteItemOutput) => {\n      if(error) {\n        return reject(error);\n      }\n      return resolve(output);\n    });\n  });\n\nexport const dynamoDel = (params: DeleteItemInput): Promise<DeleteItemOutput> => new Promise((resolve, reject) => {\n  aws.config.update(Config.get('aws'));\n  const dynamodb: DynamoDB = new DynamoDB();\n\n  dynamodb.deleteItem(params, (error, output: DeleteItemOutput) => {\n    if(error) {\n      return reject(error);\n    }\n    return resolve(output);\n  });\n});\n"],
  "mappings": "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",
  "names": []
}

|