@nlabs/reaktor 0.4.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.
- 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,571 @@
|
|
|
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 __objRest = (source, exclude) => {
|
|
26
|
+
var target = {};
|
|
27
|
+
for (var prop in source)
|
|
28
|
+
if (__hasOwnProp.call(source, prop) && exclude.indexOf(prop) < 0)
|
|
29
|
+
target[prop] = source[prop];
|
|
30
|
+
if (source != null && __getOwnPropSymbols)
|
|
31
|
+
for (var prop of __getOwnPropSymbols(source)) {
|
|
32
|
+
if (exclude.indexOf(prop) < 0 && __propIsEnum.call(source, prop))
|
|
33
|
+
target[prop] = source[prop];
|
|
34
|
+
}
|
|
35
|
+
return target;
|
|
36
|
+
};
|
|
37
|
+
var __export = (target, all) => {
|
|
38
|
+
__markAsModule(target);
|
|
39
|
+
for (var name in all)
|
|
40
|
+
__defProp(target, name, { get: all[name], enumerable: true });
|
|
41
|
+
};
|
|
42
|
+
var __reExport = (target, module2, desc) => {
|
|
43
|
+
if (module2 && typeof module2 === "object" || typeof module2 === "function") {
|
|
44
|
+
for (let key of __getOwnPropNames(module2))
|
|
45
|
+
if (!__hasOwnProp.call(target, key) && key !== "default")
|
|
46
|
+
__defProp(target, key, { get: () => module2[key], enumerable: !(desc = __getOwnPropDesc(module2, key)) || desc.enumerable });
|
|
47
|
+
}
|
|
48
|
+
return target;
|
|
49
|
+
};
|
|
50
|
+
var __toModule = (module2) => {
|
|
51
|
+
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);
|
|
52
|
+
};
|
|
53
|
+
__export(exports, {
|
|
54
|
+
UserAccess: () => UserAccess,
|
|
55
|
+
addUser: () => addUser,
|
|
56
|
+
confirmCode: () => confirmCode,
|
|
57
|
+
createToken: () => createToken,
|
|
58
|
+
deactivateUser: () => deactivateUser,
|
|
59
|
+
deleteUser: () => deleteUser,
|
|
60
|
+
getActiveUserCount: () => getActiveUserCount,
|
|
61
|
+
getDisplayName: () => getDisplayName,
|
|
62
|
+
getSessionUser: () => getSessionUser,
|
|
63
|
+
getUser: () => getUser,
|
|
64
|
+
getUserOptional: () => getUserOptional,
|
|
65
|
+
getUsers: () => getUsers,
|
|
66
|
+
getUsersByLatest: () => getUsersByLatest,
|
|
67
|
+
getUsersByReactions: () => getUsersByReactions,
|
|
68
|
+
getUsersByTags: () => getUsersByTags,
|
|
69
|
+
parseUserOptions: () => parseUserOptions,
|
|
70
|
+
refreshSession: () => refreshSession,
|
|
71
|
+
signIn: () => signIn,
|
|
72
|
+
signOut: () => signOut,
|
|
73
|
+
updateUser: () => updateUser
|
|
74
|
+
});
|
|
75
|
+
var import_utils = __toModule(require("@nlabs/utils"));
|
|
76
|
+
var import_arangojs = __toModule(require("arangojs"));
|
|
77
|
+
var import_isEmpty = __toModule(require("lodash/isEmpty"));
|
|
78
|
+
var import_luxon = __toModule(require("luxon"));
|
|
79
|
+
var import_stripe = __toModule(require("stripe"));
|
|
80
|
+
var import_config = __toModule(require("../config"));
|
|
81
|
+
var import_utils2 = __toModule(require("../utils"));
|
|
82
|
+
var import_session = __toModule(require("../utils/session"));
|
|
83
|
+
const eventCategory = "users";
|
|
84
|
+
const apiVersion = "2020-03-02";
|
|
85
|
+
var UserAccess;
|
|
86
|
+
(function(UserAccess2) {
|
|
87
|
+
UserAccess2[UserAccess2["DEACTIVATED"] = 0] = "DEACTIVATED";
|
|
88
|
+
UserAccess2[UserAccess2["ACTIVE"] = 1] = "ACTIVE";
|
|
89
|
+
UserAccess2[UserAccess2["PREMIUM"] = 2] = "PREMIUM";
|
|
90
|
+
UserAccess2[UserAccess2["CONTENT_ADMIN"] = 3] = "CONTENT_ADMIN";
|
|
91
|
+
UserAccess2[UserAccess2["ADMIN"] = 4] = "ADMIN";
|
|
92
|
+
})(UserAccess || (UserAccess = {}));
|
|
93
|
+
const createToken = (userId, username, userAccess, expires = 15) => {
|
|
94
|
+
const now = import_luxon.DateTime.local();
|
|
95
|
+
const sessionExpires = now.plus({ minutes: expires });
|
|
96
|
+
const iat = Math.floor(now.toSeconds());
|
|
97
|
+
const exp = Math.floor(sessionExpires.toSeconds());
|
|
98
|
+
const token = (0, import_session.setSession)({
|
|
99
|
+
exp,
|
|
100
|
+
iat,
|
|
101
|
+
username,
|
|
102
|
+
userAccess,
|
|
103
|
+
userId
|
|
104
|
+
});
|
|
105
|
+
return {
|
|
106
|
+
expires: sessionExpires.toMillis(),
|
|
107
|
+
issued: now.toMillis(),
|
|
108
|
+
token
|
|
109
|
+
};
|
|
110
|
+
};
|
|
111
|
+
const getUserOptional = (fields = []) => fields.reduce((selects, field) => {
|
|
112
|
+
if (field.includes("Count")) {
|
|
113
|
+
return (0, import_utils2.selectReactionCountByType)("users", "u", field, selects);
|
|
114
|
+
}
|
|
115
|
+
return selects;
|
|
116
|
+
}, { objects: [], queries: [] });
|
|
117
|
+
const parseUserOptions = (options = {}) => {
|
|
118
|
+
const {
|
|
119
|
+
from = 0,
|
|
120
|
+
to = 30
|
|
121
|
+
} = options;
|
|
122
|
+
return __spreadProps(__spreadValues({}, options), {
|
|
123
|
+
limit: (0, import_utils2.getLimit)(from, to)
|
|
124
|
+
});
|
|
125
|
+
};
|
|
126
|
+
const addUser = async (context, args) => {
|
|
127
|
+
const action = "addUser";
|
|
128
|
+
const { database } = context;
|
|
129
|
+
const { user } = args;
|
|
130
|
+
const { email, password, phone, username } = user;
|
|
131
|
+
const salt = (0, import_utils.createHash)(`${username}${password}`, null);
|
|
132
|
+
const encryptedPassword = (0, import_utils.createPassword)(password, salt);
|
|
133
|
+
const formatUsername = (0, import_utils.parseUsername)(username);
|
|
134
|
+
const formatEmail = (0, import_utils.parseEmail)(email);
|
|
135
|
+
const formatPhone = (0, import_utils.parsePhone)(phone);
|
|
136
|
+
if ((0, import_isEmpty.default)(formatUsername) || (0, import_isEmpty.default)(password) || !(0, import_isEmpty.default)(formatPhone) && !(0, import_isEmpty.default)(formatEmail)) {
|
|
137
|
+
return (0, import_utils2.logException)({
|
|
138
|
+
action,
|
|
139
|
+
args: { username },
|
|
140
|
+
category: eventCategory,
|
|
141
|
+
value: import_utils2.ErrorTypes.INVALID_ARGUMENTS
|
|
142
|
+
}, context);
|
|
143
|
+
}
|
|
144
|
+
const filters = [`u.username == "${formatUsername}"`];
|
|
145
|
+
if (!(0, import_isEmpty.default)(formatEmail)) {
|
|
146
|
+
filters.push(`u.email == "${formatEmail}"`);
|
|
147
|
+
}
|
|
148
|
+
if (!(0, import_isEmpty.default)(formatPhone)) {
|
|
149
|
+
filters.push(`u.phone == ${formatPhone}`);
|
|
150
|
+
}
|
|
151
|
+
const checkQuery = `FOR u IN users
|
|
152
|
+
FILTER ${filters.join(" || ")}
|
|
153
|
+
RETURN u`;
|
|
154
|
+
const existingUsers = await database.query(checkQuery).then((cursor) => cursor.all()).catch((error) => (0, import_utils2.logError)({
|
|
155
|
+
action,
|
|
156
|
+
args: { username },
|
|
157
|
+
category: eventCategory,
|
|
158
|
+
label: import_utils2.ErrorTypes.DATABASE_ERROR
|
|
159
|
+
}, error, context).then(() => null));
|
|
160
|
+
if (existingUsers.length) {
|
|
161
|
+
return (0, import_utils2.logException)({
|
|
162
|
+
action,
|
|
163
|
+
args: { username },
|
|
164
|
+
category: eventCategory,
|
|
165
|
+
value: import_utils2.ErrorTypes.EXISTING_USERNAME
|
|
166
|
+
}, context);
|
|
167
|
+
}
|
|
168
|
+
const verifiedEmailCode = Math.floor(1e5 + Math.random() * 9e5);
|
|
169
|
+
const verifiedPhoneCode = Math.floor(1e5 + Math.random() * 9e5);
|
|
170
|
+
const insert = {
|
|
171
|
+
_key: (0, import_utils.createHash)(username, null),
|
|
172
|
+
added: Date.now(),
|
|
173
|
+
email: formatEmail,
|
|
174
|
+
modified: Date.now(),
|
|
175
|
+
password: encryptedPassword,
|
|
176
|
+
phone: formatPhone,
|
|
177
|
+
salt,
|
|
178
|
+
username: formatUsername,
|
|
179
|
+
userAccess: 1,
|
|
180
|
+
verifiedEmail: false,
|
|
181
|
+
verifiedEmailCode,
|
|
182
|
+
verifiedPhone: false,
|
|
183
|
+
verifiedPhoneCode
|
|
184
|
+
};
|
|
185
|
+
const insertQuery = import_arangojs.aql`INSERT ${insert} IN users RETURN NEW`;
|
|
186
|
+
return await database.query(insertQuery).then((cursor) => cursor.next() || {}).catch((error) => (0, import_utils2.logError)({
|
|
187
|
+
action,
|
|
188
|
+
args: { username },
|
|
189
|
+
category: eventCategory,
|
|
190
|
+
label: import_utils2.ErrorTypes.DATABASE_ERROR
|
|
191
|
+
}, error, context).then(() => null));
|
|
192
|
+
};
|
|
193
|
+
const updateUser = async (context, user) => {
|
|
194
|
+
const action = "updateUser";
|
|
195
|
+
const { database } = context;
|
|
196
|
+
const _a = user, { _key, _id, id, tags = [], userId } = _a, updated = __objRest(_a, ["_key", "_id", "id", "tags", "userId"]);
|
|
197
|
+
let userDocId;
|
|
198
|
+
if (_id || id) {
|
|
199
|
+
userDocId = _id || id;
|
|
200
|
+
} else if (_key || userId) {
|
|
201
|
+
userDocId = `users/${_key || userId}`;
|
|
202
|
+
}
|
|
203
|
+
const userQuery = import_arangojs.aql`LET u = DOCUMENT(${userDocId})
|
|
204
|
+
UPDATE u WITH ${updated} IN users
|
|
205
|
+
RETURN NEW`;
|
|
206
|
+
const updatedUser = await database.query(userQuery).then((cursor) => cursor.next()).catch((error) => {
|
|
207
|
+
console.log(error);
|
|
208
|
+
throw error;
|
|
209
|
+
});
|
|
210
|
+
const tagCollection = database.collection("isTagged");
|
|
211
|
+
await Promise.all(tags.map(({ id: tagDocId, name }) => {
|
|
212
|
+
const edge = {
|
|
213
|
+
_from: tagDocId,
|
|
214
|
+
_key: (0, import_utils.createHash)(`isTagged-${tagDocId}-${userDocId}`),
|
|
215
|
+
_to: userDocId,
|
|
216
|
+
added: Date.now(),
|
|
217
|
+
name
|
|
218
|
+
};
|
|
219
|
+
const tagQuery = import_arangojs.aql`FOR it IN isTagged
|
|
220
|
+
FILTER it._from == ${tagDocId} && it._to == ${userDocId} && it.name == ${name}
|
|
221
|
+
LIMIT 1
|
|
222
|
+
RETURN it`;
|
|
223
|
+
return database.query(tagQuery).then((cursor) => cursor.next()).then((tagEdge) => {
|
|
224
|
+
if (!!tagEdge) {
|
|
225
|
+
return tagEdge;
|
|
226
|
+
}
|
|
227
|
+
return tagCollection.save(edge, { returnNew: true }).then(() => edge);
|
|
228
|
+
}).catch((error) => (0, import_utils2.logError)({
|
|
229
|
+
action,
|
|
230
|
+
category: eventCategory,
|
|
231
|
+
label: "db_error"
|
|
232
|
+
}, error, context).then(() => null));
|
|
233
|
+
}));
|
|
234
|
+
return updatedUser;
|
|
235
|
+
};
|
|
236
|
+
const confirmCode = async (context, args) => {
|
|
237
|
+
const action = "confirmEmail";
|
|
238
|
+
const { database, session: { userId: sessionId } } = context;
|
|
239
|
+
const { code, type } = args;
|
|
240
|
+
const userDocId = `users/${sessionId}`;
|
|
241
|
+
const aqlQuery = import_arangojs.aql`LET u = DOCUMENT(${userDocId}) RETURN u`;
|
|
242
|
+
try {
|
|
243
|
+
return database.query(aqlQuery).then((cursor) => cursor.next() || {}).then(({ verifiedEmailCode, verifiedPhoneCode }) => {
|
|
244
|
+
switch (type) {
|
|
245
|
+
case "email":
|
|
246
|
+
return code === verifiedEmailCode;
|
|
247
|
+
case "phone":
|
|
248
|
+
return code === verifiedPhoneCode;
|
|
249
|
+
default:
|
|
250
|
+
return false;
|
|
251
|
+
}
|
|
252
|
+
}).catch((error) => (0, import_utils2.logError)({
|
|
253
|
+
action,
|
|
254
|
+
args: { code, type, userId: sessionId },
|
|
255
|
+
category: eventCategory,
|
|
256
|
+
label: import_utils2.ErrorTypes.DATABASE_ERROR
|
|
257
|
+
}, error, context));
|
|
258
|
+
} catch (error) {
|
|
259
|
+
return false;
|
|
260
|
+
}
|
|
261
|
+
};
|
|
262
|
+
const deleteUser = (context, args) => {
|
|
263
|
+
const action = "deleteUser";
|
|
264
|
+
const { database, session: { userId: sessionId, userAccess: sessionAccess } } = context;
|
|
265
|
+
const { userId } = args;
|
|
266
|
+
const isAdmin = sessionAccess > 2;
|
|
267
|
+
if (!isAdmin && sessionId !== userId) {
|
|
268
|
+
(0, import_utils2.logException)({
|
|
269
|
+
action,
|
|
270
|
+
args,
|
|
271
|
+
category: eventCategory,
|
|
272
|
+
label: "unauthorized",
|
|
273
|
+
value: "invalid_session"
|
|
274
|
+
}, context);
|
|
275
|
+
return null;
|
|
276
|
+
}
|
|
277
|
+
const aqlQuery = import_arangojs.aql`FOR u IN users
|
|
278
|
+
FILTER u._key == ${userId}
|
|
279
|
+
LIMIT 1
|
|
280
|
+
REMOVE u IN users
|
|
281
|
+
RETURN OLD`;
|
|
282
|
+
const stripeClient = new import_stripe.default(import_config.Config.get("stripe.token"), { apiVersion, typescript: true });
|
|
283
|
+
return database.query(aqlQuery).then((cursor) => cursor.next()).then((user = {}) => stripeClient.customers.del(user.stripeCustomerId).then(() => stripeClient.accounts.del(user.stripeAccountId)).then(() => user)).catch((error) => {
|
|
284
|
+
throw error;
|
|
285
|
+
});
|
|
286
|
+
};
|
|
287
|
+
const deactivateUser = (context, userId) => {
|
|
288
|
+
const action = "delete";
|
|
289
|
+
const { database, session: { userId: sessionId, userAccess: sessionAccess } } = context;
|
|
290
|
+
const isAdmin = sessionAccess > 2;
|
|
291
|
+
if (!isAdmin && sessionId !== userId) {
|
|
292
|
+
(0, import_utils2.logException)({
|
|
293
|
+
action,
|
|
294
|
+
category: eventCategory,
|
|
295
|
+
label: "unauthorized",
|
|
296
|
+
value: "invalid_session"
|
|
297
|
+
}, context);
|
|
298
|
+
return null;
|
|
299
|
+
}
|
|
300
|
+
const updated = {
|
|
301
|
+
userAccess: 0
|
|
302
|
+
};
|
|
303
|
+
const aqlQuery = import_arangojs.aql`UPDATE ${userId} WITH ${updated} IN users LIMIT 1 RETURN NEW`;
|
|
304
|
+
return database.query(aqlQuery).then((cursor) => cursor.next()).catch((error) => {
|
|
305
|
+
throw error;
|
|
306
|
+
});
|
|
307
|
+
};
|
|
308
|
+
const getDisplayName = (user = {}) => {
|
|
309
|
+
const { first, last, name = "", username = "" } = user;
|
|
310
|
+
const fullname = [first, last].join(" ").trim();
|
|
311
|
+
if (!(0, import_isEmpty.default)(name)) {
|
|
312
|
+
return name;
|
|
313
|
+
} else if (fullname !== "") {
|
|
314
|
+
return fullname;
|
|
315
|
+
} else if (!(0, import_isEmpty.default)(username)) {
|
|
316
|
+
return username;
|
|
317
|
+
}
|
|
318
|
+
return "Unknown";
|
|
319
|
+
};
|
|
320
|
+
const getSessionUser = (context) => {
|
|
321
|
+
const action = "getSessionUser";
|
|
322
|
+
const { database, fields, session: { userId: sessionId, username } } = context;
|
|
323
|
+
const { objects: selectObjects, queries: selectQueries } = getUserOptional(fields);
|
|
324
|
+
const aqlQuery = `LET u = DOCUMENT("users/${sessionId}")
|
|
325
|
+
${selectQueries.join("\n")}
|
|
326
|
+
RETURN MERGE(u, {${selectObjects.join(", ")}})`;
|
|
327
|
+
return database.query(aqlQuery).then((cursor) => cursor.next()).then((user = {}) => user).catch((error) => (0, import_utils2.logError)({
|
|
328
|
+
action,
|
|
329
|
+
args: { username, userId: sessionId },
|
|
330
|
+
category: eventCategory,
|
|
331
|
+
label: import_utils2.ErrorTypes.DATABASE_ERROR
|
|
332
|
+
}, error, context));
|
|
333
|
+
};
|
|
334
|
+
const getUser = (context, args) => {
|
|
335
|
+
const action = "getUser";
|
|
336
|
+
const { userId } = args;
|
|
337
|
+
const { database, fields } = context;
|
|
338
|
+
const formatUserId = (0, import_utils.parseId)(userId);
|
|
339
|
+
const { objects: selectObjects, queries: selectQueries } = getUserOptional(fields);
|
|
340
|
+
const aqlQuery = `LET u = DOCUMENT("users/${formatUserId}")
|
|
341
|
+
${selectQueries.join("\n")}
|
|
342
|
+
FILTER u.userAccess > 0
|
|
343
|
+
RETURN MERGE(u, {${selectObjects.join(", ")}})`;
|
|
344
|
+
return database.query(aqlQuery).then((cursor) => cursor.next()).then((user = {}) => user).catch((error) => (0, import_utils2.logError)({
|
|
345
|
+
action,
|
|
346
|
+
args,
|
|
347
|
+
category: eventCategory,
|
|
348
|
+
label: import_utils2.ErrorTypes.DATABASE_ERROR
|
|
349
|
+
}, error, context).then(() => {
|
|
350
|
+
}));
|
|
351
|
+
};
|
|
352
|
+
const getUsers = (context, options) => {
|
|
353
|
+
const action = "getUserList";
|
|
354
|
+
const { database, fields } = context;
|
|
355
|
+
const { limit, username } = parseUserOptions(options);
|
|
356
|
+
const { objects: selectObjects, queries: selectQueries } = getUserOptional(fields);
|
|
357
|
+
const filterBy = ["u.userAccess > 0"];
|
|
358
|
+
if (!(0, import_isEmpty.default)(username)) {
|
|
359
|
+
filterBy.push(`CONTAINS(u.username, "${(0, import_utils.parseUsername)(username)}")`);
|
|
360
|
+
}
|
|
361
|
+
const aqlQuery = `FOR u IN users
|
|
362
|
+
FILTER ${filterBy.join(" && ")}
|
|
363
|
+
${selectQueries.join("\n")}
|
|
364
|
+
${limit.aql}
|
|
365
|
+
SORT u.username
|
|
366
|
+
RETURN MERGE(u, {${selectObjects.join(", ")}})`;
|
|
367
|
+
return database.query(aqlQuery).then((cursor) => cursor.all()).catch((error) => (0, import_utils2.logError)({
|
|
368
|
+
action,
|
|
369
|
+
category: eventCategory,
|
|
370
|
+
label: import_utils2.ErrorTypes.DATABASE_ERROR
|
|
371
|
+
}, error, context).then(() => []));
|
|
372
|
+
};
|
|
373
|
+
const getUsersByReactions = (context, { reactions = [], username }, options) => {
|
|
374
|
+
const action = "getUsersByReactions";
|
|
375
|
+
const { database, fields, session: { userId: sessionId } } = context;
|
|
376
|
+
const formatReactions = reactions.map((reactionName) => (0, import_utils.parseChar)(reactionName, 32).toLowerCase());
|
|
377
|
+
const { limit } = parseUserOptions(options);
|
|
378
|
+
const { objects: selectObjects, queries: selectQueries } = getUserOptional(fields);
|
|
379
|
+
const formatSessionId = `users/${sessionId}`;
|
|
380
|
+
const formatUsername = (0, import_utils.parseUsername)(username);
|
|
381
|
+
const filterBy = [`POSITION(${JSON.stringify(formatReactions)}, LOWER(r.name))`];
|
|
382
|
+
if (!(0, import_isEmpty.default)(username)) {
|
|
383
|
+
filterBy.push(`CONTAINS(u.username, "${formatUsername}")`);
|
|
384
|
+
}
|
|
385
|
+
const aqlQuery = `FOR u, r IN OUTBOUND "${formatSessionId}" hasReactions
|
|
386
|
+
OPTIONS {vertexCollections: "users"}
|
|
387
|
+
${selectQueries.join("\n")}
|
|
388
|
+
${filterBy.length ? `FILTER ${filterBy.join(" && ")}` : ""}
|
|
389
|
+
${limit.aql}
|
|
390
|
+
RETURN MERGE(u, {${selectObjects.join(", ")}})`;
|
|
391
|
+
console.log({ aqlQuery });
|
|
392
|
+
return database.query(aqlQuery).then((cursor) => cursor.all()).catch((error) => (0, import_utils2.logError)({
|
|
393
|
+
action,
|
|
394
|
+
category: eventCategory,
|
|
395
|
+
label: import_utils2.ErrorTypes.DATABASE_ERROR
|
|
396
|
+
}, error, context).then(() => []));
|
|
397
|
+
};
|
|
398
|
+
const getUsersByTags = (context, { tags, username }, options) => {
|
|
399
|
+
const action = "getUsersByTags";
|
|
400
|
+
const { database, fields, session: { userId: sessionId } } = context;
|
|
401
|
+
const formatTags = tags.reduce((list, tagName) => {
|
|
402
|
+
if (!(0, import_isEmpty.default)(tagName)) {
|
|
403
|
+
list.push((0, import_utils.parseChar)(tagName, 32).toLowerCase());
|
|
404
|
+
}
|
|
405
|
+
return list;
|
|
406
|
+
}, []);
|
|
407
|
+
const { limit } = parseUserOptions(options);
|
|
408
|
+
const { objects: selectObjects, queries: selectQueries } = getUserOptional(fields);
|
|
409
|
+
const formatUsername = (0, import_utils.parseUsername)(username);
|
|
410
|
+
const filterBy = [`u._key != "${sessionId}"`];
|
|
411
|
+
if (!(0, import_isEmpty.default)(username)) {
|
|
412
|
+
filterBy.push(`CONTAINS(u.username, "${formatUsername}")`);
|
|
413
|
+
}
|
|
414
|
+
const aqlQuery = `FOR t IN tags
|
|
415
|
+
FILTER POSITION(${JSON.stringify(formatTags)}, LOWER(t.name))
|
|
416
|
+
FOR u, it IN OUTBOUND t isTagged
|
|
417
|
+
OPTIONS {bfs: true, uniqueVertices: "global", vertexCollections: "users"}
|
|
418
|
+
${selectQueries.join("\n")}
|
|
419
|
+
${filterBy.length ? `FILTER ${filterBy.join(" && ")}` : ""}
|
|
420
|
+
${limit.aql}
|
|
421
|
+
RETURN DISTINCT MERGE(u, {${selectObjects.join(", ")}})`;
|
|
422
|
+
console.log("getUsersByTags", aqlQuery);
|
|
423
|
+
return database.query(aqlQuery).then((cursor) => cursor.all()).catch((error) => (0, import_utils2.logError)({
|
|
424
|
+
action,
|
|
425
|
+
category: eventCategory,
|
|
426
|
+
label: import_utils2.ErrorTypes.DATABASE_ERROR
|
|
427
|
+
}, error, context).then(() => []));
|
|
428
|
+
};
|
|
429
|
+
const getUsersByLatest = (context, { username }, options) => {
|
|
430
|
+
const action = "getUsersByLatest";
|
|
431
|
+
const { database, fields, session: { userId } } = context;
|
|
432
|
+
const { limit } = parseUserOptions(options);
|
|
433
|
+
const filter = ["u._id != session._id"];
|
|
434
|
+
const { objects: selectObjects, queries: selectQueries } = getUserOptional(fields);
|
|
435
|
+
if (!(0, import_isEmpty.default)(username)) {
|
|
436
|
+
filter.push(`CONTAINS(u.username, "${(0, import_utils.parseUsername)(username)}")`);
|
|
437
|
+
}
|
|
438
|
+
const aqlQuery = `FOR u IN users
|
|
439
|
+
LET session = DOCUMENT("users/${userId}")
|
|
440
|
+
FILTER ${filter.join(" && ")}
|
|
441
|
+
${selectQueries.join("\n")}
|
|
442
|
+
LET distance = DISTANCE(u.latitude || 0, u.longitude || 0, session.latitude || 0, session.longitude || 0)
|
|
443
|
+
${limit.aql}
|
|
444
|
+
SORT distance ASC, u.added DESC
|
|
445
|
+
RETURN MERGE(u, {${selectObjects.join(", ")}})`;
|
|
446
|
+
return database.query(aqlQuery).then((cursor) => cursor.all()).catch((error) => (0, import_utils2.logError)({
|
|
447
|
+
action,
|
|
448
|
+
category: eventCategory,
|
|
449
|
+
label: import_utils2.ErrorTypes.DATABASE_ERROR
|
|
450
|
+
}, error, context).then(() => []));
|
|
451
|
+
};
|
|
452
|
+
const refreshSession = async (context, { expires, token }) => {
|
|
453
|
+
const { error } = (0, import_session.getSession)(token);
|
|
454
|
+
if (error) {
|
|
455
|
+
return { errors: [error] };
|
|
456
|
+
}
|
|
457
|
+
const { exp, userId, username, userAccess } = (0, import_session.getSession)(token);
|
|
458
|
+
const now = Math.ceil(import_luxon.DateTime.local().toSeconds());
|
|
459
|
+
if (exp > now) {
|
|
460
|
+
return createToken(userId, username, userAccess, expires);
|
|
461
|
+
}
|
|
462
|
+
return { errors: ["session_expired"] };
|
|
463
|
+
};
|
|
464
|
+
const signIn = async (context, args) => {
|
|
465
|
+
const action = "signIn";
|
|
466
|
+
const { expires, password, username } = args;
|
|
467
|
+
const formatUsername = (0, import_utils.parseUsername)(username);
|
|
468
|
+
const formatPassword = (0, import_utils.parsePassword)(password);
|
|
469
|
+
const formatExpires = (0, import_utils.parseNum)(expires) || 15;
|
|
470
|
+
const { database } = context;
|
|
471
|
+
if ((0, import_isEmpty.default)(formatUsername) || (0, import_isEmpty.default)(formatPassword)) {
|
|
472
|
+
return (0, import_utils2.logException)({
|
|
473
|
+
action,
|
|
474
|
+
args: { username },
|
|
475
|
+
category: eventCategory,
|
|
476
|
+
value: import_utils2.ErrorTypes.INVALID_ARGUMENTS
|
|
477
|
+
}, context);
|
|
478
|
+
}
|
|
479
|
+
const checkQuery = import_arangojs.aql`FOR u IN users
|
|
480
|
+
FILTER u.username == ${formatUsername}
|
|
481
|
+
LIMIT 1
|
|
482
|
+
RETURN u`;
|
|
483
|
+
const checkUser = await database.query(checkQuery).then((cursor) => cursor.next()).catch((error) => (0, import_utils2.logError)({
|
|
484
|
+
action,
|
|
485
|
+
args: { username: formatUsername },
|
|
486
|
+
category: eventCategory,
|
|
487
|
+
label: import_utils2.ErrorTypes.DATABASE_ERROR
|
|
488
|
+
}, error, context).then(() => null));
|
|
489
|
+
if (!checkUser) {
|
|
490
|
+
return (0, import_utils2.logException)({
|
|
491
|
+
action,
|
|
492
|
+
args: { username },
|
|
493
|
+
category: eventCategory,
|
|
494
|
+
value: import_utils2.ErrorTypes.INVALID_AUTHENTICATION
|
|
495
|
+
}, context);
|
|
496
|
+
}
|
|
497
|
+
const { _key: userId, salt, userAccess } = checkUser;
|
|
498
|
+
const authPassword = (0, import_utils.createPassword)(formatPassword, salt);
|
|
499
|
+
if (checkUser.password !== authPassword) {
|
|
500
|
+
return (0, import_utils2.logException)({
|
|
501
|
+
action,
|
|
502
|
+
args: { username },
|
|
503
|
+
category: eventCategory,
|
|
504
|
+
value: import_utils2.ErrorTypes.INVALID_AUTHENTICATION
|
|
505
|
+
}, context);
|
|
506
|
+
}
|
|
507
|
+
return createToken(userId, username, userAccess, formatExpires);
|
|
508
|
+
};
|
|
509
|
+
const signOut = async (context, args) => {
|
|
510
|
+
const action = "signOut";
|
|
511
|
+
const { database, session: { userId: sessionId, username } } = context;
|
|
512
|
+
const userDocId = `users/${sessionId}`;
|
|
513
|
+
const update = {
|
|
514
|
+
lastOnline: Date.now(),
|
|
515
|
+
sessionId: null
|
|
516
|
+
};
|
|
517
|
+
const sessionQuery = import_arangojs.aql`LET u = DOCUMENT(${userDocId})
|
|
518
|
+
UPDATE u WITH ${update} IN users
|
|
519
|
+
LIMIT 1
|
|
520
|
+
RETURN NEW`;
|
|
521
|
+
try {
|
|
522
|
+
await database.query(sessionQuery).then((cursor) => cursor.next()).catch((error) => (0, import_utils2.logError)({
|
|
523
|
+
action,
|
|
524
|
+
args: { username, userId: sessionId },
|
|
525
|
+
category: eventCategory,
|
|
526
|
+
label: import_utils2.ErrorTypes.DATABASE_ERROR
|
|
527
|
+
}, error, context));
|
|
528
|
+
} catch (error) {
|
|
529
|
+
return false;
|
|
530
|
+
}
|
|
531
|
+
return true;
|
|
532
|
+
};
|
|
533
|
+
const getActiveUserCount = (context) => {
|
|
534
|
+
const action = "getActiveUserCount";
|
|
535
|
+
const { database } = context;
|
|
536
|
+
const countQuery = import_arangojs.aql`LET docs = (
|
|
537
|
+
FOR u IN users
|
|
538
|
+
FILTER u.active == true
|
|
539
|
+
RETURN u
|
|
540
|
+
)
|
|
541
|
+
RETURN LENGTH(docs)`;
|
|
542
|
+
return database.query(countQuery).then((cursor) => cursor.next()).catch((error) => (0, import_utils2.logError)({
|
|
543
|
+
action,
|
|
544
|
+
category: eventCategory,
|
|
545
|
+
label: import_utils2.ErrorTypes.DATABASE_ERROR
|
|
546
|
+
}, error, context).then(() => 0));
|
|
547
|
+
};
|
|
548
|
+
// Annotate the CommonJS export names for ESM import in node:
|
|
549
|
+
0 && (module.exports = {
|
|
550
|
+
UserAccess,
|
|
551
|
+
addUser,
|
|
552
|
+
confirmCode,
|
|
553
|
+
createToken,
|
|
554
|
+
deactivateUser,
|
|
555
|
+
deleteUser,
|
|
556
|
+
getActiveUserCount,
|
|
557
|
+
getDisplayName,
|
|
558
|
+
getSessionUser,
|
|
559
|
+
getUser,
|
|
560
|
+
getUserOptional,
|
|
561
|
+
getUsers,
|
|
562
|
+
getUsersByLatest,
|
|
563
|
+
getUsersByReactions,
|
|
564
|
+
getUsersByTags,
|
|
565
|
+
parseUserOptions,
|
|
566
|
+
refreshSession,
|
|
567
|
+
signIn,
|
|
568
|
+
signOut,
|
|
569
|
+
updateUser
|
|
570
|
+
});
|
|
571
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../src/actions/users.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 {\n  createHash,\n  createPassword,\n  parseChar,\n  parseEmail,\n  parseId,\n  parseNum,\n  parsePassword,\n  parsePhone,\n  parseUsername\n} from '@nlabs/utils';\nimport {aql} from 'arangojs';\nimport {AqlQuery} from 'arangojs/aql';\nimport {EdgeCollection} from 'arangojs/collection';\nimport {ArrayCursor} from 'arangojs/cursor';\nimport isEmpty from 'lodash/isEmpty';\nimport {DateTime} from 'luxon';\nimport Stripe from 'stripe';\n\nimport {Config} from '../config';\nimport {ApiContext} from '../types/auth';\nimport {User} from '../types/users';\nimport {ErrorTypes, getLimit, logError, logException, selectReactionCountByType} from '../utils';\nimport {getSession, SessionError, SessionToken, setSession} from '../utils/session';\n\nconst eventCategory: string = 'users';\nconst apiVersion: any = '2020-03-02';\n\nexport interface UserOptions {\n  readonly from?: number;\n  readonly to?: number;\n  readonly username?: string;\n}\n\nexport enum UserAccess {\n  DEACTIVATED = 0,\n  ACTIVE = 1,\n  PREMIUM = 2,\n  CONTENT_ADMIN = 3,\n  ADMIN = 4\n}\n\nexport const createToken = (\n  userId: string,\n  username: string,\n  userAccess: number,\n  expires: number = 15\n): SessionToken => {\n  const now: DateTime = DateTime.local();\n  const sessionExpires: DateTime = now.plus({minutes: expires});\n  const iat: number = Math.floor(now.toSeconds());\n  const exp: number = Math.floor(sessionExpires.toSeconds());\n\n  const token = setSession({\n    exp,\n    iat,\n    username,\n    userAccess,\n    userId\n  });\n\n  return {\n    expires: sessionExpires.toMillis(),\n    issued: now.toMillis(),\n    token\n  };\n};\n\nexport const getUserOptional = (fields: string[] = []) =>\n  fields.reduce((selects: any, field: string) => {\n    if(field.includes('Count')) {\n      return selectReactionCountByType('users', 'u', field, selects);\n    }\n\n    return selects;\n  }, {objects: [], queries: []});\n\nexport const parseUserOptions = (options: UserOptions = {}) => {\n  const {\n    from = 0,\n    to = 30\n  } = options;\n\n  return {\n    ...options,\n    limit: getLimit(from, to)\n  };\n};\n\nexport const addUser = async (context: ApiContext, args: any): Promise<User> => {\n  const action: string = 'addUser';\n  const {database} = context;\n  const {user} = args;\n  const {email, password, phone, username} = user;\n  const salt: string = createHash(`${username}${password}`, null);\n  const encryptedPassword = createPassword(password, salt);\n  const formatUsername: string = parseUsername(username);\n  const formatEmail: string = parseEmail(email);\n  const formatPhone: string = parsePhone(phone);\n\n  if(isEmpty(formatUsername) || isEmpty(password) || (!isEmpty(formatPhone) && !isEmpty(formatEmail))) {\n    return logException({\n      action,\n      args: {username},\n      category: eventCategory,\n      value: ErrorTypes.INVALID_ARGUMENTS\n    }, context);\n  }\n\n  const filters: string[] = [`u.username == \"${formatUsername}\"`];\n\n  if(!isEmpty(formatEmail)) {\n    filters.push(`u.email == \"${formatEmail}\"`);\n  }\n\n  if(!isEmpty(formatPhone)) {\n    filters.push(`u.phone == ${formatPhone}`);\n  }\n\n  const checkQuery: string = `FOR u IN users\n    FILTER ${filters.join(' || ')}\n    RETURN u`;\n  const existingUsers = await database.query(checkQuery)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .catch((error) => logError({\n      action,\n      args: {username},\n      category: eventCategory,\n      label: ErrorTypes.DATABASE_ERROR\n    }, error, context)\n      .then(() => null)\n    );\n\n  if(existingUsers.length) {\n    return logException({\n      action,\n      args: {username},\n      category: eventCategory,\n      value: ErrorTypes.EXISTING_USERNAME\n    }, context);\n  }\n\n  const verifiedEmailCode: number = Math.floor(100000 + (Math.random() * 900000));\n  const verifiedPhoneCode: number = Math.floor(100000 + (Math.random() * 900000));\n\n  // Add new user properties\n  const insert: User = {\n    _key: createHash(username, null),\n    added: Date.now(),\n    email: formatEmail,\n    modified: Date.now(),\n    password: encryptedPassword,\n    phone: formatPhone,\n    salt,\n    username: formatUsername,\n    userAccess: 1,\n    verifiedEmail: false,\n    verifiedEmailCode,\n    verifiedPhone: false,\n    verifiedPhoneCode\n  };\n\n  // Add new user in ArangoDB\n  const insertQuery: AqlQuery = aql`INSERT ${insert} IN users RETURN NEW`;\n  return await database.query(insertQuery)\n    .then((cursor: ArrayCursor) => cursor.next() || {})\n    .catch((error) => logError({\n      action,\n      args: {username},\n      category: eventCategory,\n      label: ErrorTypes.DATABASE_ERROR\n    }, error, context)\n      .then(() => null)\n    );\n};\n\nexport const updateUser = async (context: ApiContext, user: User): Promise<any> => {\n  const action: string = 'updateUser';\n  const {database} = context;\n  const {_key, _id, id, tags = [], userId, ...updated} = user;\n  let userDocId: string;\n\n  if(_id || id) {\n    userDocId = _id || id;\n  } else if(_key || userId) {\n    userDocId = `users/${_key || userId}`;\n  }\n\n  const userQuery: AqlQuery = aql`LET u = DOCUMENT(${userDocId})\n    UPDATE u WITH ${updated} IN users\n    RETURN NEW`;\n\n  const updatedUser = await database.query(userQuery)\n    .then((cursor) => cursor.next())\n    .catch((error: Error) => {\n      console.log(error);\n      throw error;\n    });\n\n  const tagCollection: EdgeCollection = database.collection('isTagged');\n  await Promise.all(tags.map(({id: tagDocId, name}) => {\n    const edge = {\n      _from: tagDocId,\n      _key: createHash(`isTagged-${tagDocId}-${userDocId}`),\n      _to: userDocId,\n      added: Date.now(),\n      name\n    };\n    const tagQuery: AqlQuery = aql`FOR it IN isTagged\n      FILTER it._from == ${tagDocId} && it._to == ${userDocId} && it.name == ${name}\n      LIMIT 1\n      RETURN it`;\n\n    return database.query(tagQuery)\n      .then((cursor: ArrayCursor) => cursor.next())\n      .then((tagEdge) => {\n        if(!!tagEdge) {\n          return tagEdge;\n        }\n\n        return tagCollection.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\n  return updatedUser;\n};\n\nexport const confirmCode = async (context: ApiContext, args): Promise<boolean> => {\n  const action: string = 'confirmEmail';\n  const {database, session: {userId: sessionId}} = context;\n  const {code, type} = args;\n  const userDocId: string = `users/${sessionId}`;\n\n  const aqlQuery: AqlQuery = aql`LET u = DOCUMENT(${userDocId}) RETURN u`;\n\n  try {\n    return database.query(aqlQuery)\n      .then((cursor) => cursor.next() || {})\n      .then(({verifiedEmailCode, verifiedPhoneCode}: User) => {\n        switch(type) {\n          case 'email':\n            return code === verifiedEmailCode;\n          case 'phone':\n            return code === verifiedPhoneCode;\n          default:\n            return false;\n        }\n      })\n      .catch((error) => logError({\n        action,\n        args: {code, type, userId: sessionId},\n        category: eventCategory,\n        label: ErrorTypes.DATABASE_ERROR\n      }, error, context)\n      );\n  } catch(error) {\n    return false;\n  }\n};\n\nexport const deleteUser = (context: ApiContext, args: any): Promise<any> => {\n  const action: string = 'deleteUser';\n  const {database, session: {userId: sessionId, userAccess: sessionAccess}} = context;\n  const {userId} = args;\n  const isAdmin: boolean = sessionAccess > 2;\n\n  if(!isAdmin && (sessionId !== userId)) {\n    logException({\n      action,\n      args,\n      category: eventCategory,\n      label: 'unauthorized',\n      value: 'invalid_session'\n    }, context);\n    return null;\n  }\n\n  const aqlQuery: AqlQuery = aql`FOR u IN users\n    FILTER u._key == ${userId}\n    LIMIT 1\n    REMOVE u IN users\n    RETURN OLD`;\n\n  // Stripe\n  const stripeClient = new Stripe(Config.get('stripe.token'), {apiVersion, typescript: true});\n\n  return database.query(aqlQuery)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((user: User = {}) => stripeClient.customers.del(user.stripeCustomerId)\n      .then(() => stripeClient.accounts.del(user.stripeAccountId))\n      .then(() => user))\n    .catch((error: Error) => {\n      throw error;\n    });\n};\n\nexport const deactivateUser = (context: ApiContext, userId: string): Promise<User> => {\n  const action: string = 'delete';\n  const {database, session: {userId: sessionId, userAccess: sessionAccess}} = context;\n  const isAdmin: boolean = sessionAccess > 2;\n\n  if(!isAdmin && (sessionId !== userId)) {\n    logException({\n      action,\n      category: eventCategory,\n      label: 'unauthorized',\n      value: 'invalid_session'\n    }, context);\n    return null;\n  }\n\n  const updated: User = {\n    userAccess: 0\n  };\n  const aqlQuery: AqlQuery = aql`UPDATE ${userId} WITH ${updated} IN users LIMIT 1 RETURN NEW`;\n\n  return database.query(aqlQuery)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .catch((error: Error) => {\n      throw error;\n    });\n};\n\nexport const getDisplayName = (user: User = {}): string => {\n  const {first, last, name = '', username = ''} = user;\n  const fullname: string = ([first, last]).join(' ').trim();\n\n  if(!isEmpty(name)) {\n    return name;\n  } else if(fullname !== '') {\n    return fullname;\n  } else if(!isEmpty(username)) {\n    return username;\n  }\n\n  return 'Unknown';\n};\n\nexport const getSessionUser = (context: ApiContext): Promise<User> => {\n  const action: string = 'getSessionUser';\n  const {database, fields, session: {userId: sessionId, username}} = context;\n  const {objects: selectObjects, queries: selectQueries} = getUserOptional(fields);\n\n  const aqlQuery: string = `LET u = DOCUMENT(\"users/${sessionId}\")\n  ${selectQueries.join('\\n')}\n  RETURN MERGE(u, {${selectObjects.join(', ')}})`;\n\n  return database.query(aqlQuery)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((user = {}) => user)\n    .catch((error: Error) => logError({\n      action,\n      args: {username, userId: sessionId},\n      category: eventCategory,\n      label: ErrorTypes.DATABASE_ERROR\n    }, error, context));\n};\n\nexport const getUser = (context: ApiContext, args: any): Promise<User> => {\n  const action: string = 'getUser';\n  const {userId} = args;\n  const {database, fields} = context;\n  const formatUserId: string = parseId(userId);\n  const {objects: selectObjects, queries: selectQueries} = getUserOptional(fields);\n\n  // Get data from database\n  const aqlQuery: string = `LET u = DOCUMENT(\"users/${formatUserId}\")\n    ${selectQueries.join('\\n')}\n    FILTER u.userAccess > 0\n    RETURN MERGE(u, {${selectObjects.join(', ')}})`;\n\n  return database.query(aqlQuery)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((user = {}) => user)\n    .catch((error: Error) => logError({\n      action,\n      args,\n      category: eventCategory,\n      label: ErrorTypes.DATABASE_ERROR\n    }, error, context)\n      .then(() => { }));\n};\n\nexport const getUsers = (context: ApiContext, options?: UserOptions): Promise<User[]> => {\n  const action: string = 'getUserList';\n  const {database, fields} = context;\n  const {limit, username} = parseUserOptions(options);\n  const {objects: selectObjects, queries: selectQueries} = getUserOptional(fields);\n  const filterBy: string[] = ['u.userAccess > 0'];\n\n  if(!isEmpty(username)) {\n    filterBy.push(`CONTAINS(u.username, \"${parseUsername(username)}\")`);\n  }\n\n  // Get data from database\n  const aqlQuery: string = `FOR u IN users\n    FILTER ${filterBy.join(' && ')}\n    ${selectQueries.join('\\n')}\n    ${limit.aql}\n    SORT u.username\n    RETURN MERGE(u, {${selectObjects.join(', ')}})`;\n\n  return database.query(aqlQuery)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: ErrorTypes.DATABASE_ERROR\n    }, error, context).then(() => []));\n};\n\nexport const getUsersByReactions = (\n  context: ApiContext,\n  {reactions = [], username}: any,\n  options?: UserOptions\n): Promise<User[]> => {\n  const action: string = 'getUsersByReactions';\n  const {database, fields, session: {userId: sessionId}} = context;\n  const formatReactions: string[] =  reactions.map((reactionName: string) => parseChar(reactionName, 32).toLowerCase());\n  const {limit} = parseUserOptions(options);\n  const {objects: selectObjects, queries: selectQueries} = getUserOptional(fields);\n\n  const formatSessionId: string = `users/${sessionId}`;\n  const formatUsername: string = parseUsername(username);\n  const filterBy: string[] = [`POSITION(${JSON.stringify(formatReactions)}, LOWER(r.name))`];\n\n  if(!isEmpty(username)) {\n    filterBy.push(`CONTAINS(u.username, \"${formatUsername}\")`);\n  }\n\n  // Get data from database\n  const aqlQuery: string = `FOR u, r IN OUTBOUND \"${formatSessionId}\" hasReactions\n    OPTIONS {vertexCollections: \"users\"}\n    ${selectQueries.join('\\n')}\n    ${filterBy.length ? `FILTER ${filterBy.join(' && ')}` : ''}\n    ${limit.aql}\n    RETURN MERGE(u, {${selectObjects.join(', ')}})`;\n\n  console.log({aqlQuery});\n  return database.query(aqlQuery)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: ErrorTypes.DATABASE_ERROR\n    }, error, context).then(() => []));\n};\n\nexport const getUsersByTags = (\n  context: ApiContext,\n  {tags, username}: any,\n  options?: UserOptions\n): Promise<User[]> => {\n  const action: string = 'getUsersByTags';\n  const {database, fields, session: {userId: sessionId}} = context;\n  const formatTags: string[] =  tags.reduce((list: string[], tagName: string) => {\n    if(!isEmpty(tagName)) {\n      list.push(parseChar(tagName, 32).toLowerCase());\n    }\n\n    return list;\n  }, []);\n  const {limit} = parseUserOptions(options);\n  const {objects: selectObjects, queries: selectQueries} = getUserOptional(fields);\n\n  const formatUsername: string = parseUsername(username);\n  const filterBy: string[] = [`u._key != \"${sessionId}\"`];\n\n  if(!isEmpty(username)) {\n    filterBy.push(`CONTAINS(u.username, \"${formatUsername}\")`);\n  }\n\n  // Get data from database\n  const aqlQuery: string = `FOR t IN tags\n    FILTER POSITION(${JSON.stringify(formatTags)}, LOWER(t.name))\n    FOR u, it IN OUTBOUND t isTagged\n    OPTIONS {bfs: true, uniqueVertices: \"global\", vertexCollections: \"users\"}\n    ${selectQueries.join('\\n')}\n    ${filterBy.length ? `FILTER ${filterBy.join(' && ')}` : ''}\n    ${limit.aql}\n    RETURN DISTINCT MERGE(u, {${selectObjects.join(', ')}})`;\n\n  console.log('getUsersByTags', aqlQuery);\n  return database.query(aqlQuery)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: ErrorTypes.DATABASE_ERROR\n    }, error, context).then(() => []));\n};\n\nexport const getUsersByLatest = (context: ApiContext, {username}, options?: UserOptions): Promise<User[]> => {\n  const action: string = 'getUsersByLatest';\n  const {database, fields, session: {userId}} = context;\n  const {limit} = parseUserOptions(options);\n  const filter = ['u._id != session._id'];\n  const {objects: selectObjects, queries: selectQueries} = getUserOptional(fields);\n\n  if(!isEmpty(username)) {\n    filter.push(`CONTAINS(u.username, \"${parseUsername(username)}\")`);\n  }\n\n  // Get data from database\n  const aqlQuery: string = `FOR u IN users\n    LET session = DOCUMENT(\"users/${userId}\")\n    FILTER ${filter.join(' && ')}\n    ${selectQueries.join('\\n')}\n    LET distance = DISTANCE(u.latitude || 0, u.longitude || 0, session.latitude || 0, session.longitude || 0)\n    ${limit.aql}\n    SORT distance ASC, u.added DESC\n    RETURN MERGE(u, {${selectObjects.join(', ')}})`;\n\n  return database.query(aqlQuery)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: ErrorTypes.DATABASE_ERROR\n    }, error, context).then(() => []));\n};\n\nexport const refreshSession = async (context: ApiContext, {expires, token}): Promise<SessionToken | SessionError> => {\n  const {error} = getSession(token);\n\n  if(error) {\n    return {errors: [error]};\n  }\n\n  const {exp, userId, username, userAccess} = getSession(token);\n  const now: number = Math.ceil(DateTime.local().toSeconds());\n\n  if(exp > now) {\n    return createToken(userId, username, userAccess, expires);\n  }\n\n  return {errors: ['session_expired']};\n};\n\nexport const signIn = async (context: ApiContext, args): Promise<SessionToken> => {\n  const action: string = 'signIn';\n  const {expires, password, username} = args;\n  const formatUsername: string = parseUsername(username);\n  const formatPassword: string = parsePassword(password);\n  const formatExpires: number = parseNum(expires) || 15;\n  const {database} = context;\n\n  if(isEmpty(formatUsername) || isEmpty(formatPassword)) {\n    return logException({\n      action,\n      args: {username},\n      category: eventCategory,\n      value: ErrorTypes.INVALID_ARGUMENTS\n    }, context);\n  }\n\n  const checkQuery: AqlQuery = aql`FOR u IN users\n    FILTER u.username == ${formatUsername}\n    LIMIT 1\n    RETURN u`;\n  const checkUser: User = await database.query(checkQuery)\n    .then((cursor) => cursor.next())\n    .catch((error) => logError({\n      action,\n      args: {username: formatUsername},\n      category: eventCategory,\n      label: ErrorTypes.DATABASE_ERROR\n    }, error, context)\n      .then(() => null)\n    );\n\n  if(!checkUser) {\n    return logException({\n      action,\n      args: {username},\n      category: eventCategory,\n      value: ErrorTypes.INVALID_AUTHENTICATION\n    }, context);\n  }\n\n  const {_key: userId, salt, userAccess} = checkUser;\n  const authPassword: string = createPassword(formatPassword, salt);\n\n  if(checkUser.password !== authPassword) {\n    return logException({\n      action,\n      args: {username},\n      category: eventCategory,\n      value: ErrorTypes.INVALID_AUTHENTICATION\n    }, context);\n  }\n\n  return createToken(userId, username, userAccess, formatExpires);\n};\n\nexport const signOut = async (context: ApiContext, args): Promise<boolean> => {\n  const action: string = 'signOut';\n  const {database, session: {userId: sessionId, username}} = context;\n  const userDocId: string = `users/${sessionId}`;\n\n  const update = {\n    lastOnline: Date.now(),\n    sessionId: null\n  };\n  const sessionQuery: AqlQuery = aql`LET u = DOCUMENT(${userDocId})\n    UPDATE u WITH ${update} IN users\n    LIMIT 1\n    RETURN NEW`;\n\n  try {\n    await database.query(sessionQuery)\n      .then((cursor) => cursor.next())\n      .catch((error) => logError({\n        action,\n        args: {username, userId: sessionId},\n        category: eventCategory,\n        label: ErrorTypes.DATABASE_ERROR\n      }, error, context)\n      );\n  } catch(error) {\n    return false;\n  }\n\n  return true;\n};\n\nexport const getActiveUserCount = (context: ApiContext) => {\n  const action: string = 'getActiveUserCount';\n  const {database} = context;\n  const countQuery: AqlQuery = aql`LET docs = (\n    FOR u IN users\n    FILTER u.active == true\n    RETURN u\n  )\n  RETURN LENGTH(docs)`;\n\n  return database.query(countQuery)\n    .then((cursor) => cursor.next())\n    .catch((error) => logError({\n      action,\n      category: eventCategory,\n      label: ErrorTypes.DATABASE_ERROR\n    }, error, context)\n      .then(() => 0)\n    );\n};\n"],
  "mappings": "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",
  "names": []
}

|
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
import { ApiContext } from '../types/auth';
|
|
2
2
|
export declare const addConnection: (connectionId: string, token: string) => Promise<any>;
|
|
3
|
-
export declare const
|
|
3
|
+
export declare const getConnectionByUser: (userId: string) => Promise<any>;
|
|
4
4
|
export declare const getConnectionById: (connectionId: string) => Promise<any>;
|
|
5
5
|
export declare const getConnectionByConvo: (context: ApiContext, convoId: string) => Promise<any>;
|
|
6
6
|
export declare const removeConnection: (connectionId: string) => Promise<any>;
|