@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,19 @@
|
|
|
1
|
+
import { Database } from 'arangojs';
|
|
2
|
+
import { ApiContext, FileType, PostInputType, PostOptions, PostType } from '../types';
|
|
3
|
+
export declare const parsePostOptions: (options?: PostOptions) => {
|
|
4
|
+
latitude: number;
|
|
5
|
+
limit: import("../types").ArangoDBLimit;
|
|
6
|
+
longitude: number;
|
|
7
|
+
type: string;
|
|
8
|
+
};
|
|
9
|
+
export declare const getPostOptional: (fields: string[], sessionId: string) => any;
|
|
10
|
+
export declare const getPost: (context: ApiContext, itemId: string, options: PostOptions) => Promise<Partial<PostType>>;
|
|
11
|
+
export declare const getPostsByReactions: (context: ApiContext, reactions?: string[], options?: PostOptions) => Promise<PostType[]>;
|
|
12
|
+
export declare const getPostsByTags: (context: ApiContext, tags?: string[], options?: PostOptions) => Promise<PostType[]>;
|
|
13
|
+
export declare const getPostsByUser: (context: ApiContext, userId: string, options?: PostOptions) => Promise<PostType[]>;
|
|
14
|
+
export declare const getPostComments: (context: ApiContext, itemId: string, options?: PostOptions) => Promise<PostType[]>;
|
|
15
|
+
export declare const addPost: (context: ApiContext, post: PostInputType) => Promise<PostType>;
|
|
16
|
+
export declare const updatePost: (context: ApiContext, post: PostInputType) => Promise<PostType>;
|
|
17
|
+
export declare const deletePost: (context: ApiContext, postId: string) => Promise<PostType>;
|
|
18
|
+
export declare const cleanPosts: (database: Database) => Promise<number>;
|
|
19
|
+
export declare const createPostEdge: (db: Database, file: FileType, postId: string) => Promise<FileType>;
|
|
@@ -0,0 +1,538 @@
|
|
|
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
|
+
addPost: () => addPost,
|
|
43
|
+
cleanPosts: () => cleanPosts,
|
|
44
|
+
createPostEdge: () => createPostEdge,
|
|
45
|
+
deletePost: () => deletePost,
|
|
46
|
+
getPost: () => getPost,
|
|
47
|
+
getPostComments: () => getPostComments,
|
|
48
|
+
getPostOptional: () => getPostOptional,
|
|
49
|
+
getPostsByReactions: () => getPostsByReactions,
|
|
50
|
+
getPostsByTags: () => getPostsByTags,
|
|
51
|
+
getPostsByUser: () => getPostsByUser,
|
|
52
|
+
parsePostOptions: () => parsePostOptions,
|
|
53
|
+
updatePost: () => updatePost
|
|
54
|
+
});
|
|
55
|
+
var import_utils = __toModule(require("@nlabs/utils"));
|
|
56
|
+
var import_arangojs = __toModule(require("arangojs"));
|
|
57
|
+
var import_utils2 = __toModule(require("../utils"));
|
|
58
|
+
var import_files = __toModule(require("./files"));
|
|
59
|
+
var import_tags = __toModule(require("./tags"));
|
|
60
|
+
const MAX_CONTENT_LENGTH = 1e5;
|
|
61
|
+
const eventCategory = "posts";
|
|
62
|
+
const parsePostOptions = (options = {}) => {
|
|
63
|
+
const {
|
|
64
|
+
from = 0,
|
|
65
|
+
latitude = 0,
|
|
66
|
+
longitude = 0,
|
|
67
|
+
to = 30,
|
|
68
|
+
type = "post"
|
|
69
|
+
} = options;
|
|
70
|
+
return {
|
|
71
|
+
latitude: (0, import_utils.parseNum)(latitude, 32),
|
|
72
|
+
limit: (0, import_utils2.getLimit)(from, to),
|
|
73
|
+
longitude: (0, import_utils.parseNum)(longitude, 32),
|
|
74
|
+
type: (0, import_utils.parseChar)(type, 32)
|
|
75
|
+
};
|
|
76
|
+
};
|
|
77
|
+
const getPostOptional = (fields, sessionId) => fields.reduce((selects, field) => {
|
|
78
|
+
switch (field) {
|
|
79
|
+
case "hasRsvp": {
|
|
80
|
+
selects.queries.push(`LET hasRsvp = TO_BOOL(FIRST(
|
|
81
|
+
FOR post, r IN INBOUND p._id hasReactions
|
|
82
|
+
FILTER r.name == "rsvp" && r.type == "posts" && r._from == "users/${sessionId}"
|
|
83
|
+
COLLECT WITH COUNT INTO count
|
|
84
|
+
RETURN count
|
|
85
|
+
))`);
|
|
86
|
+
selects.objects.push("hasRsvp:hasRsvp");
|
|
87
|
+
return selects;
|
|
88
|
+
}
|
|
89
|
+
case "isSaved": {
|
|
90
|
+
selects.queries.push(`LET isSaved = TO_BOOL(FIRST(
|
|
91
|
+
FOR post, r IN INBOUND p._id hasReactions
|
|
92
|
+
FILTER r.name == "pin" && r.type == "posts" && r._from == "users/${sessionId}"
|
|
93
|
+
COLLECT WITH COUNT INTO count
|
|
94
|
+
RETURN count
|
|
95
|
+
))`);
|
|
96
|
+
selects.objects.push("isSaved:isSaved");
|
|
97
|
+
return selects;
|
|
98
|
+
}
|
|
99
|
+
case "reactions": {
|
|
100
|
+
selects.queries.push(`LET reactions = (
|
|
101
|
+
FOR post, r IN INBOUND p._id hasReactions
|
|
102
|
+
COLLECT reactionName = r.value INTO reactionItems
|
|
103
|
+
RETURN {name: reactionName, count: LENGTH(reactionItems[*].r.value)}
|
|
104
|
+
)`);
|
|
105
|
+
selects.objects.push("reactions:reactions");
|
|
106
|
+
return selects;
|
|
107
|
+
}
|
|
108
|
+
case "rsvpCount": {
|
|
109
|
+
selects.queries.push(`LET rsvpCount = FIRST(
|
|
110
|
+
FOR post, r IN INBOUND p._id hasReactions
|
|
111
|
+
FILTER r.name == "rsvp" && r.type == "posts"
|
|
112
|
+
COLLECT WITH COUNT INTO count
|
|
113
|
+
RETURN count
|
|
114
|
+
)`);
|
|
115
|
+
selects.objects.push("rsvpCount:rsvpCount");
|
|
116
|
+
return selects;
|
|
117
|
+
}
|
|
118
|
+
case "viewCount": {
|
|
119
|
+
selects.queries.push(`LET viewCount = FIRST(
|
|
120
|
+
FOR post, r IN INBOUND p._id hasReactions
|
|
121
|
+
FILTER r.name == "view" && r.type == "posts"
|
|
122
|
+
COLLECT WITH COUNT INTO count
|
|
123
|
+
RETURN count
|
|
124
|
+
)`);
|
|
125
|
+
selects.objects.push("viewCount:viewCount");
|
|
126
|
+
return selects;
|
|
127
|
+
}
|
|
128
|
+
default: {
|
|
129
|
+
return selects;
|
|
130
|
+
}
|
|
131
|
+
}
|
|
132
|
+
}, { objects: [], queries: [] });
|
|
133
|
+
const getPost = (context, itemId, options) => {
|
|
134
|
+
const action = "getPost";
|
|
135
|
+
const { database, fields, session: { userId: sessionId } } = context;
|
|
136
|
+
const formatItemId = (0, import_utils.parseId)(itemId);
|
|
137
|
+
const { type } = parsePostOptions(options);
|
|
138
|
+
const db = database;
|
|
139
|
+
const { objects: selectObjects, queries: selectQueries } = getPostOptional(fields, sessionId);
|
|
140
|
+
const aqlQry = import_arangojs.aql`FOR p IN posts
|
|
141
|
+
FILTER p._key == ${formatItemId} && p.type == ${type}
|
|
142
|
+
LIMIT 1
|
|
143
|
+
RETURN p`;
|
|
144
|
+
return db.query(aqlQry).then((cursor) => cursor.next()).then((post = {}) => {
|
|
145
|
+
const {
|
|
146
|
+
_id: postDocId,
|
|
147
|
+
groupId,
|
|
148
|
+
privacy = "default"
|
|
149
|
+
} = post;
|
|
150
|
+
let privacyAqlQry;
|
|
151
|
+
if (groupId && privacy === "group") {
|
|
152
|
+
privacyAqlQry = `LET p = DOCUMENT("${postDocId}")
|
|
153
|
+
${selectQueries.join("\n")}
|
|
154
|
+
FOR group IN groups
|
|
155
|
+
FILTER group._key == p.groupId
|
|
156
|
+
FOR u, e IN OUTBOUND group._id isGrouped
|
|
157
|
+
FILTER u._key == "${sessionId}"
|
|
158
|
+
LIMIT 1
|
|
159
|
+
RETURN MERGE(p, {${selectObjects.join(", ")}})`;
|
|
160
|
+
} else if (privacy === "public") {
|
|
161
|
+
privacyAqlQry = `LET p = DOCUMENT("${postDocId}")
|
|
162
|
+
${selectQueries.join("\n")}
|
|
163
|
+
LIMIT 1
|
|
164
|
+
RETURN MERGE(p, {${selectObjects.join(", ")}})`;
|
|
165
|
+
}
|
|
166
|
+
if (privacyAqlQry) {
|
|
167
|
+
return db.query(privacyAqlQry).then((cursor) => cursor.next() || {}).catch((error) => (0, import_utils2.logError)({
|
|
168
|
+
action,
|
|
169
|
+
category: eventCategory,
|
|
170
|
+
label: "db_error"
|
|
171
|
+
}, error, context).then(() => ({})));
|
|
172
|
+
}
|
|
173
|
+
return {};
|
|
174
|
+
}).catch((error) => (0, import_utils2.logError)({
|
|
175
|
+
action,
|
|
176
|
+
category: eventCategory,
|
|
177
|
+
label: "db_error"
|
|
178
|
+
}, error, context).then(() => ({})));
|
|
179
|
+
};
|
|
180
|
+
const getPostsByReactions = (context, reactions = [], options) => {
|
|
181
|
+
const action = "getPostsByReactions";
|
|
182
|
+
const { database, fields, session: { userId: sessionId } } = context;
|
|
183
|
+
const { latitude, limit, longitude, type } = parsePostOptions(options);
|
|
184
|
+
const { objects: selectObjects, queries: selectQueries } = getPostOptional(fields, sessionId);
|
|
185
|
+
const formatSessionId = `users/${sessionId}`;
|
|
186
|
+
const formatReactions = JSON.stringify(reactions.map((reaction) => (0, import_utils.parseChar)(reaction, 32).toLowerCase()));
|
|
187
|
+
const sortBy = [];
|
|
188
|
+
const filters = [`p.type == "${type}"`, 'p.privacy == "public"'];
|
|
189
|
+
const formatLatitude = (0, import_utils.parseNum)(latitude);
|
|
190
|
+
const formatLongitude = (0, import_utils.parseNum)(longitude);
|
|
191
|
+
if (formatLatitude && formatLongitude) {
|
|
192
|
+
selectQueries.push(`LET distance = DISTANCE(
|
|
193
|
+
${formatLatitude},
|
|
194
|
+
${formatLongitude},
|
|
195
|
+
NOT_NULL(p.latitude, 0),
|
|
196
|
+
NOT_NULL(p.longitude, 0))
|
|
197
|
+
`);
|
|
198
|
+
selectObjects.push("distance:distance");
|
|
199
|
+
sortBy.push("distance");
|
|
200
|
+
}
|
|
201
|
+
if (reactions.length) {
|
|
202
|
+
sortBy.push("matchedTags DESC");
|
|
203
|
+
selectQueries.push(`LET matchedReactions = LENGTH(
|
|
204
|
+
FOR mr IN reactions
|
|
205
|
+
FILTER mr.matched == true
|
|
206
|
+
RETURN mr
|
|
207
|
+
)`);
|
|
208
|
+
selectObjects.push("matchedReactions:matchedReactions");
|
|
209
|
+
filters.push("matchedReactions > 0");
|
|
210
|
+
}
|
|
211
|
+
sortBy.push("p.added DESC");
|
|
212
|
+
selectObjects.push("reactions:reactions");
|
|
213
|
+
const aqlQry = `FOR p, r IN OUTBOUND "${formatSessionId}" hasReactions
|
|
214
|
+
LET reactions = (
|
|
215
|
+
FOR reaction, hr IN 1..1 INBOUND p isTagged
|
|
216
|
+
LET matched = LENGTH(${formatReactions}) > 0 && POSITION(${formatReactions}, reaction.name)
|
|
217
|
+
SORT reaction.name
|
|
218
|
+
RETURN MERGE(reaction, {matched:matched})
|
|
219
|
+
)
|
|
220
|
+
${selectQueries.join("\n")}
|
|
221
|
+
FILTER ${filters.join(" && ")}
|
|
222
|
+
${limit.aql}
|
|
223
|
+
RETURN DISTINCT MERGE(p, {${selectObjects.join(", ")}})`;
|
|
224
|
+
return database.query(aqlQry).then((cursor) => cursor.all()).catch((error) => (0, import_utils2.logError)({
|
|
225
|
+
action,
|
|
226
|
+
category: eventCategory,
|
|
227
|
+
label: "db_error"
|
|
228
|
+
}, error, context).then(() => []));
|
|
229
|
+
};
|
|
230
|
+
const getPostsByTags = (context, tags = [], options) => {
|
|
231
|
+
const { database, fields, session: { userId: sessionId } } = context;
|
|
232
|
+
const { latitude, limit, longitude, type } = parsePostOptions(options);
|
|
233
|
+
const { objects: selectObjects, queries: selectQueries } = getPostOptional(fields, sessionId);
|
|
234
|
+
const formatTagNames = JSON.stringify(tags.map((tag) => (0, import_utils.parseChar)(tag, 32).toLowerCase()));
|
|
235
|
+
const sortBy = [];
|
|
236
|
+
const filters = [`p.type == "${type}"`, 'p.privacy == "public"'];
|
|
237
|
+
const formatLatitude = (0, import_utils.parseNum)(latitude);
|
|
238
|
+
const formatLongitude = (0, import_utils.parseNum)(longitude);
|
|
239
|
+
if (formatLatitude && formatLongitude) {
|
|
240
|
+
selectQueries.push(`LET distance = DISTANCE(
|
|
241
|
+
${formatLatitude},
|
|
242
|
+
${formatLongitude},
|
|
243
|
+
NOT_NULL(p.latitude, 0),
|
|
244
|
+
NOT_NULL(p.longitude, 0))
|
|
245
|
+
`);
|
|
246
|
+
selectObjects.push("distance:distance");
|
|
247
|
+
sortBy.push("distance");
|
|
248
|
+
}
|
|
249
|
+
if (tags.length) {
|
|
250
|
+
sortBy.push("matchedTags DESC");
|
|
251
|
+
selectQueries.push(`LET matchedTags = LENGTH(
|
|
252
|
+
FOR t IN tags
|
|
253
|
+
FILTER t.matched == true
|
|
254
|
+
RETURN t
|
|
255
|
+
)`);
|
|
256
|
+
selectObjects.push("matchedTags:matchedTags");
|
|
257
|
+
filters.push("matchedTags > 0");
|
|
258
|
+
}
|
|
259
|
+
sortBy.push("p.added DESC");
|
|
260
|
+
selectObjects.push("tags:tags");
|
|
261
|
+
const aqlQry = `FOR p IN posts
|
|
262
|
+
LET tags = (
|
|
263
|
+
FOR tag, it IN 1..1 INBOUND p isTagged
|
|
264
|
+
LET matched = LENGTH(${formatTagNames}) > 0 && POSITION(${formatTagNames}, tag.name)
|
|
265
|
+
SORT tag.name
|
|
266
|
+
RETURN MERGE(tag, {matched:matched})
|
|
267
|
+
)
|
|
268
|
+
${selectQueries.join("\n")}
|
|
269
|
+
FILTER ${filters.join(" && ")}
|
|
270
|
+
${limit.aql}
|
|
271
|
+
SORT ${sortBy.join(", ")}
|
|
272
|
+
RETURN DISTINCT MERGE(p, {${selectObjects.join(", ")}})`;
|
|
273
|
+
console.log({ aqlQry, options });
|
|
274
|
+
return database.query(aqlQry).then((cursor) => cursor.all()).catch(() => []);
|
|
275
|
+
};
|
|
276
|
+
const getPostsByUser = (context, userId, options) => {
|
|
277
|
+
const { database, fields, session: { userId: sessionId } } = context;
|
|
278
|
+
const { limit, type } = parsePostOptions(options);
|
|
279
|
+
const formatUserId = (0, import_utils.parseId)(userId);
|
|
280
|
+
const { objects: selectObjects, queries: selectQueries } = getPostOptional(fields, sessionId);
|
|
281
|
+
const aqlQry = `FOR p IN posts
|
|
282
|
+
FILTER p.userId == "${formatUserId}" && p.type == "${type}" && p.privacy == "public" && p.parent == null
|
|
283
|
+
${selectQueries.join("\n")}
|
|
284
|
+
${limit.aql}
|
|
285
|
+
SORT p.added
|
|
286
|
+
RETURN DISTINCT MERGE(p, {${selectObjects.join(", ")}})`;
|
|
287
|
+
return database.query(aqlQry).then((cursor) => cursor.all()).catch((error) => {
|
|
288
|
+
throw error;
|
|
289
|
+
});
|
|
290
|
+
};
|
|
291
|
+
const getPostComments = (context, itemId, options) => {
|
|
292
|
+
const { database, session: { userId: sessionId } } = context;
|
|
293
|
+
const { limit, type } = parsePostOptions(options);
|
|
294
|
+
const formatItemId = (0, import_utils.parseId)(itemId);
|
|
295
|
+
const db = database;
|
|
296
|
+
const aqlQry = import_arangojs.aql`FOR p IN posts
|
|
297
|
+
FILTER p.type == ${type} && p._key == ${formatItemId}
|
|
298
|
+
LIMIT 1
|
|
299
|
+
RETURN p`;
|
|
300
|
+
return db.query(aqlQry).then((cursor) => cursor.next()).then((post = {}) => {
|
|
301
|
+
const {
|
|
302
|
+
_key,
|
|
303
|
+
groupId,
|
|
304
|
+
privacy = "public"
|
|
305
|
+
} = post;
|
|
306
|
+
let privacyAqlQry;
|
|
307
|
+
if (groupId && privacy === "group") {
|
|
308
|
+
privacyAqlQry = `FOR p IN posts
|
|
309
|
+
FOR user IN users
|
|
310
|
+
FILTER p.parent == "${_key}" && user._key == p.userId
|
|
311
|
+
LET reactions = (
|
|
312
|
+
FOR post, r IN INBOUND p._id reactions
|
|
313
|
+
COLLECT reactionName = r.value INTO reactionItems
|
|
314
|
+
RETURN {name: reactionName, count: LENGTH(reactionItems[*].r.value)}
|
|
315
|
+
)
|
|
316
|
+
FOR group IN groups
|
|
317
|
+
FILTER group._key == p.groupId
|
|
318
|
+
FOR u, e IN OUTBOUND group._id isGrouped
|
|
319
|
+
FILTER u._key == "${sessionId}"
|
|
320
|
+
SORT p.added
|
|
321
|
+
${limit.aql}
|
|
322
|
+
RETURN MERGE(p, {user: user, reactions: reactions})`;
|
|
323
|
+
} else if (privacy === "public") {
|
|
324
|
+
privacyAqlQry = `FOR p IN posts
|
|
325
|
+
FOR user IN users
|
|
326
|
+
FILTER p.parent == "${_key}" && user._key == p.userId
|
|
327
|
+
LET reactions = (
|
|
328
|
+
FOR post, r IN INBOUND p._id reactions
|
|
329
|
+
COLLECT reactionName = r.value INTO reactionItems
|
|
330
|
+
RETURN {name: reactionName, count: LENGTH(reactionItems[*].r.value)}
|
|
331
|
+
)
|
|
332
|
+
SORT p.added
|
|
333
|
+
${limit.aql}
|
|
334
|
+
RETURN MERGE(p, {user: user, reactions: reactions})`;
|
|
335
|
+
}
|
|
336
|
+
if (privacyAqlQry) {
|
|
337
|
+
return db.query(privacyAqlQry).then((cursor) => cursor.all()).catch((error) => {
|
|
338
|
+
throw error;
|
|
339
|
+
});
|
|
340
|
+
}
|
|
341
|
+
return [];
|
|
342
|
+
}).catch((error) => {
|
|
343
|
+
throw error;
|
|
344
|
+
});
|
|
345
|
+
};
|
|
346
|
+
const addPost = async (context, post) => {
|
|
347
|
+
const { database, session: { userId: sessionId } } = context;
|
|
348
|
+
const {
|
|
349
|
+
content = "",
|
|
350
|
+
endDate,
|
|
351
|
+
groupId = "",
|
|
352
|
+
location,
|
|
353
|
+
latitude,
|
|
354
|
+
longitude,
|
|
355
|
+
name = "",
|
|
356
|
+
parentId = null,
|
|
357
|
+
privacy = "public",
|
|
358
|
+
tags = [],
|
|
359
|
+
startDate,
|
|
360
|
+
type = "default"
|
|
361
|
+
} = post;
|
|
362
|
+
const now = Date.now();
|
|
363
|
+
const insert = {
|
|
364
|
+
_key: (0, import_utils.createHash)(`post-${sessionId}`),
|
|
365
|
+
added: now,
|
|
366
|
+
content: (0, import_utils.parseString)(content, MAX_CONTENT_LENGTH),
|
|
367
|
+
endDate: endDate ? (0, import_utils.parseNum)(endDate, 13) : void 0,
|
|
368
|
+
groupId: groupId ? (0, import_utils.parseId)(groupId) : void 0,
|
|
369
|
+
latitude: latitude !== void 0 ? (0, import_utils.parseNum)(latitude) : void 0,
|
|
370
|
+
location: location ? (0, import_utils.parseString)(location, 160) : void 0,
|
|
371
|
+
longitude: longitude !== void 0 ? (0, import_utils.parseNum)(longitude) : void 0,
|
|
372
|
+
modified: now,
|
|
373
|
+
name: (0, import_utils.parseString)(name, 160),
|
|
374
|
+
parentId: parentId ? (0, import_utils.parseId)(parentId) : void 0,
|
|
375
|
+
privacy: privacy ? (0, import_utils.parseVarChar)(privacy, 16) : void 0,
|
|
376
|
+
startDate: startDate ? (0, import_utils.parseNum)(startDate, 13) : void 0,
|
|
377
|
+
type: (0, import_utils.parseChar)(type, 32),
|
|
378
|
+
userId: sessionId
|
|
379
|
+
};
|
|
380
|
+
const db = database;
|
|
381
|
+
const aqlQry = import_arangojs.aql`INSERT ${insert} IN posts RETURN NEW`;
|
|
382
|
+
try {
|
|
383
|
+
const savedPost = await db.query(aqlQry).then((cursor) => cursor.next() || {});
|
|
384
|
+
const { _id: postDocId } = savedPost;
|
|
385
|
+
console.log({ tags });
|
|
386
|
+
if (tags && tags.length) {
|
|
387
|
+
const tagNames = tags.map(({ name: name2 }) => (0, import_utils.parseChar)(name2, 32));
|
|
388
|
+
console.log({ tagNames, postDocId });
|
|
389
|
+
await (0, import_tags.linkTags)(db, tagNames, postDocId);
|
|
390
|
+
} else {
|
|
391
|
+
const tagList = await (0, import_tags.extractTags)(db, postDocId, insert.content);
|
|
392
|
+
savedPost.tags = tagList;
|
|
393
|
+
}
|
|
394
|
+
return savedPost;
|
|
395
|
+
} catch (error) {
|
|
396
|
+
throw error;
|
|
397
|
+
}
|
|
398
|
+
};
|
|
399
|
+
const updatePost = async (context, post) => {
|
|
400
|
+
const { database, session: { userId: sessionId } } = context;
|
|
401
|
+
const now = Date.now();
|
|
402
|
+
const {
|
|
403
|
+
content,
|
|
404
|
+
endDate,
|
|
405
|
+
groupId,
|
|
406
|
+
name,
|
|
407
|
+
parentId,
|
|
408
|
+
postId,
|
|
409
|
+
privacy,
|
|
410
|
+
startDate,
|
|
411
|
+
tags = [],
|
|
412
|
+
type
|
|
413
|
+
} = post;
|
|
414
|
+
const update = {
|
|
415
|
+
content: content ? (0, import_utils.parseString)(content, MAX_CONTENT_LENGTH) : void 0,
|
|
416
|
+
endDate: endDate ? (0, import_utils.parseNum)(endDate, 13) : void 0,
|
|
417
|
+
modified: now,
|
|
418
|
+
name: name ? (0, import_utils.parseString)(name, 160) : void 0,
|
|
419
|
+
parentId: parentId ? (0, import_utils.parseString)(parentId, 160) : void 0,
|
|
420
|
+
privacy: privacy ? (0, import_utils.parseVarChar)(privacy, 16) : void 0,
|
|
421
|
+
startDate: startDate ? (0, import_utils.parseNum)(startDate, 13) : void 0,
|
|
422
|
+
type: type !== void 0 ? (0, import_utils.parseChar)(type, 16) : void 0
|
|
423
|
+
};
|
|
424
|
+
let formatId = (0, import_utils.parseId)(postId);
|
|
425
|
+
formatId = formatId === "" ? (0, import_utils.createHash)(`post-${sessionId}`) : formatId;
|
|
426
|
+
const formatGroupId = (0, import_utils.parseId)(groupId);
|
|
427
|
+
const insert = __spreadProps(__spreadValues({}, update), {
|
|
428
|
+
_key: formatId,
|
|
429
|
+
added: now,
|
|
430
|
+
groupId: formatGroupId,
|
|
431
|
+
privacy,
|
|
432
|
+
userId: sessionId
|
|
433
|
+
});
|
|
434
|
+
const db = database;
|
|
435
|
+
const aqlQry = import_arangojs.aql`UPSERT {_key: ${formatId}, userId: ${sessionId}}
|
|
436
|
+
INSERT ${insert}
|
|
437
|
+
UPDATE ${update}
|
|
438
|
+
IN posts RETURN NEW`;
|
|
439
|
+
try {
|
|
440
|
+
const updatedPost = await db.query(aqlQry).then((cursor) => cursor.next() || {});
|
|
441
|
+
const { _id: updatedPostId } = updatedPost;
|
|
442
|
+
if (tags && tags.length) {
|
|
443
|
+
const tagNames = tags.map(({ name: name2 }) => (0, import_utils.parseChar)(name2, 32));
|
|
444
|
+
const tagQuery = import_arangojs.aql`FOR t IN tags
|
|
445
|
+
FILTER POSITION(${JSON.stringify(tagNames)}, t.name)
|
|
446
|
+
RETURN t`;
|
|
447
|
+
const existingTags = await db.query(tagQuery).then((cursor) => cursor.all());
|
|
448
|
+
const tagIds = existingTags.map(({ _id: tagId }) => tagId);
|
|
449
|
+
await (0, import_tags.linkTags)(db, tagIds, updatedPostId);
|
|
450
|
+
} else {
|
|
451
|
+
const tagList = await (0, import_tags.extractTags)(db, updatedPostId, update.content || "") || [];
|
|
452
|
+
updatedPost.tags = tagList;
|
|
453
|
+
}
|
|
454
|
+
const files = updatedPost.files || [];
|
|
455
|
+
if (files.length) {
|
|
456
|
+
const fileList = await (0, import_files.updateFiles)(db, formatId, files) || [];
|
|
457
|
+
updatedPost.files = fileList;
|
|
458
|
+
} else {
|
|
459
|
+
updatedPost.files = [];
|
|
460
|
+
}
|
|
461
|
+
return updatedPost;
|
|
462
|
+
} catch (error) {
|
|
463
|
+
throw error;
|
|
464
|
+
}
|
|
465
|
+
};
|
|
466
|
+
const deletePost = (context, postId) => {
|
|
467
|
+
const { database, session: { userId: sessionId } } = context;
|
|
468
|
+
const formatItemId = (0, import_utils.parseId)(postId);
|
|
469
|
+
const db = database;
|
|
470
|
+
const aqlQry = import_arangojs.aql`FOR p IN posts
|
|
471
|
+
FILTER p._key == ${formatItemId} && p.userId == ${sessionId}
|
|
472
|
+
LIMIT 1
|
|
473
|
+
REMOVE p IN posts
|
|
474
|
+
RETURN OLD`;
|
|
475
|
+
return db.query(aqlQry).then((cursor) => cursor.next()).then((post = {}) => {
|
|
476
|
+
if (post) {
|
|
477
|
+
const edgeAqlQry = import_arangojs.aql`FOR t IN isTagged
|
|
478
|
+
FILTER t._to == ${formatItemId}
|
|
479
|
+
REMOVE t IN isTagged`;
|
|
480
|
+
return db.query(edgeAqlQry).then(() => {
|
|
481
|
+
const fileAqlQry = import_arangojs.aql`FOR f IN hasFile
|
|
482
|
+
FILTER f._to == ${formatItemId}
|
|
483
|
+
REMOVE f IN hasFile`;
|
|
484
|
+
return db.query(fileAqlQry).then(() => post).catch((error) => {
|
|
485
|
+
throw error;
|
|
486
|
+
});
|
|
487
|
+
}).catch((error) => {
|
|
488
|
+
throw error;
|
|
489
|
+
});
|
|
490
|
+
}
|
|
491
|
+
return {};
|
|
492
|
+
}).catch((error) => {
|
|
493
|
+
throw error;
|
|
494
|
+
});
|
|
495
|
+
};
|
|
496
|
+
const cleanPosts = (database) => {
|
|
497
|
+
const aqlQry = import_arangojs.aql`FOR p IN posts
|
|
498
|
+
FILTER p.added < DATE_TIMESTAMP(DATE_SUBTRACT(DATE_NOW(), 60, 'day')) && p.type == "default"
|
|
499
|
+
REMOVE p IN posts
|
|
500
|
+
RETURN OLD`;
|
|
501
|
+
return database.query(aqlQry).then((cursor) => cursor.all()).then((results = []) => results.length).catch((error) => {
|
|
502
|
+
throw error;
|
|
503
|
+
});
|
|
504
|
+
};
|
|
505
|
+
const createPostEdge = (db, file, postId) => {
|
|
506
|
+
const edgeCollection = db.collection("isPosted");
|
|
507
|
+
const { fileId } = file;
|
|
508
|
+
const formatFileId = (0, import_utils.parseId)(fileId);
|
|
509
|
+
const edgeId = (0, import_utils.createHash)(`file-${postId}-${formatFileId}`);
|
|
510
|
+
const formatPostId = (0, import_utils.parseId)(postId);
|
|
511
|
+
const fileType = (0, import_utils.parseChar)(file.fileType, 16);
|
|
512
|
+
const edge = {
|
|
513
|
+
_from: `posts/${formatPostId}`,
|
|
514
|
+
_key: edgeId,
|
|
515
|
+
_to: `files/${formatFileId}`,
|
|
516
|
+
added: Date.now(),
|
|
517
|
+
type: fileType
|
|
518
|
+
};
|
|
519
|
+
return edgeCollection.save(edge, { returnNew: true }).then(() => file).catch((error) => {
|
|
520
|
+
throw error;
|
|
521
|
+
});
|
|
522
|
+
};
|
|
523
|
+
// Annotate the CommonJS export names for ESM import in node:
|
|
524
|
+
0 && (module.exports = {
|
|
525
|
+
addPost,
|
|
526
|
+
cleanPosts,
|
|
527
|
+
createPostEdge,
|
|
528
|
+
deletePost,
|
|
529
|
+
getPost,
|
|
530
|
+
getPostComments,
|
|
531
|
+
getPostOptional,
|
|
532
|
+
getPostsByReactions,
|
|
533
|
+
getPostsByTags,
|
|
534
|
+
getPostsByUser,
|
|
535
|
+
parsePostOptions,
|
|
536
|
+
updatePost
|
|
537
|
+
});
|
|
538
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../src/actions/posts.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, parseChar, parseId, parseNum, parseString, parseVarChar} from '@nlabs/utils';\nimport {aql, Database} from 'arangojs';\nimport {AqlQuery} from 'arangojs/aql';\nimport {EdgeCollection} from 'arangojs/collection';\nimport {ArrayCursor} from 'arangojs/cursor';\n\nimport {ApiContext, FileType, PostInputType, PostOptions, PostType, TagType} from '../types';\nimport {getLimit, logError} from '../utils';\nimport {updateFiles} from './files';\nimport {extractTags, linkTags} from './tags';\n\nconst MAX_CONTENT_LENGTH: number = 100000;\nconst eventCategory: string = 'posts';\n\nexport const parsePostOptions = (options: PostOptions = {}) => {\n  const {\n    from = 0,\n    latitude = 0,\n    longitude = 0,\n    to = 30,\n    type = 'post'\n  } = options;\n\n  return {\n    latitude: parseNum(latitude, 32),\n    limit: getLimit(from, to),\n    longitude: parseNum(longitude, 32),\n    type: parseChar(type, 32)\n  };\n};\n\nexport const getPostOptional = (fields: string[], sessionId: string) =>\n  fields.reduce((selects: any, field: string) => {\n    switch(field) {\n      case 'hasRsvp': {\n        selects.queries.push(`LET hasRsvp = TO_BOOL(FIRST(\n          FOR post, r IN INBOUND p._id hasReactions\n          FILTER r.name == \"rsvp\" && r.type == \"posts\" && r._from == \"users/${sessionId}\"\n          COLLECT WITH COUNT INTO count\n          RETURN count\n        ))`);\n        selects.objects.push('hasRsvp:hasRsvp');\n        return selects;\n      }\n      case 'isSaved': {\n        selects.queries.push(`LET isSaved = TO_BOOL(FIRST(\n          FOR post, r IN INBOUND p._id hasReactions\n          FILTER r.name == \"pin\" && r.type == \"posts\" && r._from == \"users/${sessionId}\"\n          COLLECT WITH COUNT INTO count\n          RETURN count\n        ))`);\n        selects.objects.push('isSaved:isSaved');\n        return selects;\n      }\n      case 'reactions': {\n        selects.queries.push(`LET reactions = (\n          FOR post, r IN INBOUND p._id hasReactions\n          COLLECT reactionName = r.value INTO reactionItems\n          RETURN {name: reactionName, count: LENGTH(reactionItems[*].r.value)}\n        )`);\n        selects.objects.push('reactions:reactions');\n        return selects;\n      }\n      case 'rsvpCount': {\n        selects.queries.push(`LET rsvpCount = FIRST(\n          FOR post, r IN INBOUND p._id hasReactions\n          FILTER r.name == \"rsvp\" && r.type == \"posts\"\n          COLLECT WITH COUNT INTO count\n          RETURN count\n        )`);\n        selects.objects.push('rsvpCount:rsvpCount');\n        return selects;\n      }\n      case 'viewCount': {\n        selects.queries.push(`LET viewCount = FIRST(\n          FOR post, r IN INBOUND p._id hasReactions\n          FILTER r.name == \"view\" && r.type == \"posts\"\n          COLLECT WITH COUNT INTO count\n          RETURN count\n        )`);\n        selects.objects.push('viewCount:viewCount');\n        return selects;\n      }\n      default: {\n        return selects;\n      }\n    }\n  }, {objects: [], queries: []});\n\nexport const getPost = (context: ApiContext, itemId: string, options: PostOptions): Promise<Partial<PostType>> => {\n  const action: string = 'getPost';\n  const {database, fields, session: {userId: sessionId}} = context;\n  const formatItemId: string = parseId(itemId);\n  const {type} = parsePostOptions(options);\n  const db = database;\n  const {objects: selectObjects, queries: selectQueries} = getPostOptional(fields, sessionId);\n  const aqlQry: AqlQuery = aql`FOR p IN posts\n    FILTER p._key == ${formatItemId} && p.type == ${type}\n    LIMIT 1\n    RETURN p`;\n\n  return db.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((post: PostType = {}) => {\n      const {\n        _id: postDocId,\n        groupId,\n        privacy = 'default'\n      }: PostType = post;\n\n      // Query based on privacy level\n      let privacyAqlQry: string;\n\n      if(groupId && privacy === 'group') {\n        privacyAqlQry = `LET p = DOCUMENT(\"${postDocId}\")\n          ${selectQueries.join('\\n')}\n          FOR group IN groups\n          FILTER group._key == p.groupId\n          FOR u, e IN OUTBOUND group._id isGrouped\n          FILTER u._key == \"${sessionId}\"\n          LIMIT 1\n          RETURN MERGE(p, {${selectObjects.join(', ')}})`;\n      } else if(privacy === 'public') {\n        privacyAqlQry = `LET p = DOCUMENT(\"${postDocId}\")\n          ${selectQueries.join('\\n')}\n          LIMIT 1\n          RETURN MERGE(p, {${selectObjects.join(', ')}})`;\n      }\n\n      if(privacyAqlQry) {\n        return db.query(privacyAqlQry)\n          .then((cursor: ArrayCursor) => cursor.next() || {})\n          .catch((error: Error) => logError({\n            action,\n            category: eventCategory,\n            label: 'db_error'\n          }, error, context).then(() => ({})));\n      }\n\n      return {};\n    })\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, context).then(() => ({})));\n};\n\n// export const getPostList = (context: ApiContext, options?: PostOptions): Promise<PostType[]> => {\n//   // const action: string = 'getListByApp';\n//   const {database, fields, session: {userId: sessionId}} = context;\n//   console.log('getPostList::context', context);\n//   const {limit, type} = parsePostOptions(options);\n//   const {objects: selectObjects, queries: selectQueries} = getPostOptional(fields, sessionId);\n//   const aqlQry: string = `FOR p IN posts\n//     FILTER p.type == \"${type}\" && p.privacy == \"public\" && p.parent == null\n//     ${selectQueries.join('\\n')}\n//     ${limit.aql}\n//     SORT p.added\n//     RETURN DISTINCT MERGE(p, {${selectObjects.join(', ')}})`;\n\n//   return database.query(aqlQry)\n//     .then((cursor: ArrayCursor) => cursor.all())\n//     .catch((error: Error) => {\n//       console.log('getPostList::error', error);\n//       throw error;\n//     });\n// };\n\n// export const getPostsByArea = (\n//   context: ApiContext,\n//   latitude: number,\n//   longitude: number,\n//   options?: PostOptions\n// ): Promise<PostType[]> => {\n//   // const action: string = 'getListByUser';\n//   const {database, fields, session: {userId: sessionId}} = context;\n//   const {limit, type} = parsePostOptions(options);\n//   const formatLatitude: number = parseNum(latitude);\n//   const formatLongitude: number = parseNum(longitude);\n//   const {objects: selectObjects, queries: selectQueries} = getPostOptional(fields, sessionId);\n//   selectQueries.push(`LET distance = DISTANCE(\n//     ${formatLatitude},\n//     ${formatLongitude},\n//     NOT_NULL(p.latitude, 0),\n//     NOT_NULL(p.longitude, 0))\n//   `);\n//   selectObjects.push('distance:distance');\n\n//   const aqlQry: string = `FOR p IN posts\n//     ${selectQueries.join('\\n')}\n//     FILTER p.type == \"${type}\" && p.privacy == \"public\" && p.parentId == null\n//     ${limit.aql}\n//     SORT distance, p.added\n//     RETURN DISTINCT MERGE(p, {${selectObjects.join(', ')}})`;\n\n//   return database.query(aqlQry)\n//     .then((cursor: ArrayCursor) => cursor.all())\n//     .catch((error: Error) => {\n//       throw error;\n//     });\n// };\n\n// export const getPostsByGroup = (\n//   context: ApiContext,\n//   groupId: string,\n//   options?: PostOptions\n// ): Promise<PostType[]> => {\n//   // const action: string = 'getListByGroup';\n//   const {database, fields, session: {userId: sessionId}} = context;\n//   const {objects: selectObjects, queries: selectQueries} = getPostOptional(fields, sessionId);\n\n//   // Group id\n//   const formatGroupId: string = parseId(groupId);\n//   const db = database;\n//   const aqlQry: string = `FOR u, g IN INBOUND ${formatGroupId} hasGroup\n//       FILTER u._key == ${sessionId}\n//       RETURN g`;\n\n//   return db.query(aqlQry)\n//     .then((cursor: ArrayCursor) => cursor.all())\n//     .then((groups: GroupType[] = []) => {\n//       if(groups.length) {\n//         const {limit, type} = parsePostOptions(options);\n//         const postAqlQry: string = `FOR p IN posts\n//           FILTER p.type == \"${type}\" && p.groupId == \"${formatGroupId}\" && p.parent == null\n//           ${selectQueries.join('\\n')}\n//           ${limit.aql}\n//           SORT p.added\n//           RETURN DISTINCT MERGE(p, {${selectObjects.join(', ')}})`;\n\n//         return db.query(postAqlQry)\n//           .then((cursor: ArrayCursor) => cursor.all())\n//           .catch((error: Error) => {\n//             throw error;\n//           });\n//       }\n\n//       return [];\n//     })\n//     .catch((error: Error) => {\n//       throw error;\n//     });\n// };\n\n// export const getPostsByLatest = (context: ApiContext, options?: PostOptions): Promise<PostType[]> => {\n//   // const action: string = 'getListByLatest';\n//   console.log('getPostsByLatest::options', options);\n//   const {database, fields, session: {userId: sessionId}} = context;\n//   const {limit, type} = parsePostOptions(options);\n//   const {objects: selectObjects, queries: selectQueries} = getPostOptional(fields, sessionId);\n//   const aqlQry: string = `FOR p IN posts\n//     FILTER p.type == \"${type}\" && p.privacy == \"public\" && p.parent == null\n//     ${selectQueries.join('\\n')}\n//     ${limit.aql}\n//     SORT p.added DESC\n//     RETURN DISTINCT MERGE(p, {${selectObjects.join(', ')}})`;\n\n//   console.log('getPostsByLatest::aqlQry', aqlQry);\n//   return database.query(aqlQry)\n//     .then((cursor: ArrayCursor) => cursor.all())\n//     .catch((error: Error) => {\n//       throw error;\n//     });\n// };\n\nexport const getPostsByReactions = (\n  context: ApiContext,\n  reactions: string[] = [],\n  options?: PostOptions\n): Promise<PostType[]> => {\n  const action: string = 'getPostsByReactions';\n  const {database, fields, session: {userId: sessionId}} = context;\n  const {latitude, limit, longitude, type} = parsePostOptions(options);\n  const {objects: selectObjects, queries: selectQueries} = getPostOptional(fields, sessionId);\n  const formatSessionId: string = `users/${sessionId}`;\n  const formatReactions: string = JSON.stringify(reactions.map((reaction) => parseChar(reaction, 32).toLowerCase()));\n  const sortBy: string[] = [];\n  const filters: string[] = [`p.type == \"${type}\"`, 'p.privacy == \"public\"'];\n  const formatLatitude: number = parseNum(latitude);\n  const formatLongitude: number = parseNum(longitude);\n\n  if(formatLatitude && formatLongitude) {\n    selectQueries.push(`LET distance = DISTANCE(\n      ${formatLatitude},\n      ${formatLongitude},\n      NOT_NULL(p.latitude, 0),\n      NOT_NULL(p.longitude, 0))\n    `);\n    selectObjects.push('distance:distance');\n    sortBy.push('distance');\n  }\n\n  if(reactions.length) {\n    sortBy.push('matchedTags DESC');\n    selectQueries.push(`LET matchedReactions = LENGTH(\n      FOR mr IN reactions\n      FILTER mr.matched == true\n      RETURN mr\n    )`);\n    selectObjects.push('matchedReactions:matchedReactions');\n    filters.push('matchedReactions > 0');\n  }\n\n  sortBy.push('p.added DESC');\n  selectObjects.push('reactions:reactions');\n\n  // Get data from database\n  const aqlQry: string = `FOR p, r IN OUTBOUND \"${formatSessionId}\" hasReactions\n    LET reactions = (\n      FOR reaction, hr IN 1..1 INBOUND p isTagged\n      LET matched = LENGTH(${formatReactions}) > 0 && POSITION(${formatReactions}, reaction.name)\n      SORT reaction.name\n      RETURN MERGE(reaction, {matched:matched})\n    )\n    ${selectQueries.join('\\n')}\n    FILTER ${filters.join(' && ')}\n    ${limit.aql}\n    RETURN DISTINCT MERGE(p, {${selectObjects.join(', ')}})`;\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(() => []));\n};\n\nexport const getPostsByTags = (\n  context: ApiContext,\n  tags: string[] = [],\n  options?: PostOptions\n): Promise<PostType[]> => {\n  // const action: string = 'getListByTags';\n  const {database, fields, session: {userId: sessionId}} = context;\n  const {latitude, limit, longitude, type} = parsePostOptions(options);\n  const {objects: selectObjects, queries: selectQueries} = getPostOptional(fields, sessionId);\n  const formatTagNames: string = JSON.stringify(tags.map((tag) => parseChar(tag, 32).toLowerCase()));\n  const sortBy: string[] = [];\n  const filters: string[] = [`p.type == \"${type}\"`, 'p.privacy == \"public\"'];\n  const formatLatitude: number = parseNum(latitude);\n  const formatLongitude: number = parseNum(longitude);\n\n  if(formatLatitude && formatLongitude) {\n    selectQueries.push(`LET distance = DISTANCE(\n      ${formatLatitude},\n      ${formatLongitude},\n      NOT_NULL(p.latitude, 0),\n      NOT_NULL(p.longitude, 0))\n    `);\n    selectObjects.push('distance:distance');\n    sortBy.push('distance');\n  }\n\n  if(tags.length) {\n    sortBy.push('matchedTags DESC');\n    selectQueries.push(`LET matchedTags = LENGTH(\n      FOR t IN tags\n      FILTER t.matched == true\n      RETURN t\n    )`);\n    selectObjects.push('matchedTags:matchedTags');\n    filters.push('matchedTags > 0');\n  }\n\n  sortBy.push('p.added DESC');\n  selectObjects.push('tags:tags');\n\n  const aqlQry: string = `FOR p IN posts\n    LET tags = (\n      FOR tag, it IN 1..1 INBOUND p isTagged\n      LET matched = LENGTH(${formatTagNames}) > 0 && POSITION(${formatTagNames}, tag.name)\n      SORT tag.name\n      RETURN MERGE(tag, {matched:matched})\n    )\n    ${selectQueries.join('\\n')}\n    FILTER ${filters.join(' && ')}\n    ${limit.aql}\n    SORT ${sortBy.join(', ')}\n    RETURN DISTINCT MERGE(p, {${selectObjects.join(', ')}})`;\n\n  console.log({aqlQry, options});\n  return database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .catch(() => []);\n};\n\nexport const getPostsByUser = (context: ApiContext, userId: string, options?: PostOptions): Promise<PostType[]> => {\n  // const action: string = 'getListByUser';\n  const {database, fields, session: {userId: sessionId}} = context;\n  const {limit, type} = parsePostOptions(options);\n  const formatUserId: string = parseId(userId);\n  const {objects: selectObjects, queries: selectQueries} = getPostOptional(fields, sessionId);\n  const aqlQry: string = `FOR p IN posts\n    FILTER p.userId == \"${formatUserId}\" && p.type == \"${type}\" && p.privacy == \"public\" && p.parent == null\n    ${selectQueries.join('\\n')}\n    ${limit.aql}\n    SORT p.added\n    RETURN DISTINCT MERGE(p, {${selectObjects.join(', ')}})`;\n\n  return database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .catch((error: Error) => {\n      throw error;\n    });\n};\n\nexport const getPostComments = (context: ApiContext, itemId: string, options?: PostOptions): Promise<PostType[]> => {\n  // const action: string = 'getComments';\n  const {database, session: {userId: sessionId}} = context;\n  const {limit, type} = parsePostOptions(options);\n  const formatItemId: string = parseId(itemId);\n\n  // Get the parent post to get restrictions\n  const db = database;\n  const aqlQry: AqlQuery = aql`FOR p IN posts\n    FILTER p.type == ${type} && p._key == ${formatItemId}\n    LIMIT 1\n    RETURN p`;\n\n  return db.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((post: PostType = {}) => {\n      const {\n        _key,\n        groupId,\n        privacy = 'public'\n      }: PostType = post;\n\n      // Query based on privacy level\n      let privacyAqlQry: string;\n\n      if(groupId && privacy === 'group') {\n        privacyAqlQry = `FOR p IN posts\n          FOR user IN users\n          FILTER p.parent == \"${_key}\" && user._key == p.userId\n          LET reactions = (\n            FOR post, r IN INBOUND p._id reactions\n            COLLECT reactionName = r.value INTO reactionItems\n            RETURN {name: reactionName, count: LENGTH(reactionItems[*].r.value)}\n          )\n          FOR group IN groups\n          FILTER group._key == p.groupId\n          FOR u, e IN OUTBOUND group._id isGrouped\n          FILTER u._key == \"${sessionId}\"\n          SORT p.added\n          ${limit.aql}\n          RETURN MERGE(p, {user: user, reactions: reactions})`;\n      } else if(privacy === 'public') {\n        privacyAqlQry = `FOR p IN posts\n          FOR user IN users\n          FILTER p.parent == \"${_key}\" && user._key == p.userId\n          LET reactions = (\n            FOR post, r IN INBOUND p._id reactions\n            COLLECT reactionName = r.value INTO reactionItems\n            RETURN {name: reactionName, count: LENGTH(reactionItems[*].r.value)}\n          )\n          SORT p.added\n          ${limit.aql}\n          RETURN MERGE(p, {user: user, reactions: reactions})`;\n      }\n\n      if(privacyAqlQry) {\n        return db.query(privacyAqlQry)\n          .then((cursor: ArrayCursor) => cursor.all())\n          .catch((error: Error) => {\n            throw error;\n          });\n      }\n\n      return [];\n    })\n    .catch((error: Error) => {\n      throw error;\n    });\n};\n\nexport const addPost = async (context: ApiContext, post: PostInputType): Promise<PostType> => {\n  // const action: string = 'add';\n  const {database, session: {userId: sessionId}} = context;\n\n  const {\n    content = '',\n    endDate,\n    groupId = '',\n    location,\n    latitude,\n    longitude,\n    name = '',\n    parentId = null,\n    privacy = 'public',\n    tags = [],\n    startDate,\n    type = 'default'\n  } = post;\n\n  const now: number = Date.now();\n\n  const insert: PostType = {\n    _key: createHash(`post-${sessionId}`),\n    added: now,\n    content: parseString(content, MAX_CONTENT_LENGTH),\n    endDate: endDate ? parseNum(endDate, 13) : undefined,\n    groupId: groupId ? parseId(groupId) : undefined,\n    latitude: latitude !== undefined ? parseNum(latitude) : undefined,\n    location: location ? parseString(location, 160) : undefined,\n    longitude: longitude !== undefined ? parseNum(longitude) : undefined,\n    modified: now,\n    name: parseString(name, 160),\n    parentId: parentId ? parseId(parentId) : undefined,\n    privacy: privacy ? parseVarChar(privacy, 16) : undefined,\n    startDate: startDate ? parseNum(startDate, 13) : undefined,\n    type: parseChar(type, 32),\n    userId: sessionId\n  };\n\n  const db: Database = database;\n  const aqlQry: AqlQuery = aql`INSERT ${insert} IN posts RETURN NEW`;\n\n  try {\n    const savedPost: PostType = await db.query(aqlQry)\n      .then((cursor: ArrayCursor) => cursor.next() || {});\n\n    const {_id: postDocId} = savedPost;\n\n    console.log({tags});\n    if(tags && tags.length) {\n      const tagNames: string[] = tags.map(({name}) => parseChar(name, 32));\n      console.log({tagNames, postDocId});\n      await linkTags(db, tagNames, postDocId);\n    } else {\n      // Update linked tags within posts\n      const tagList: TagType[] = await extractTags(db, postDocId, insert.content);\n      savedPost.tags = tagList;\n    }\n\n    return savedPost;\n  } catch(error) {\n    throw error;\n  }\n};\n\nexport const updatePost = async (context: ApiContext, post: PostInputType): Promise<PostType> => {\n  // const action: string = 'update';\n  const {database, session: {userId: sessionId}} = context;\n  const now: number = Date.now();\n  const {\n    content,\n    endDate,\n    groupId,\n    name,\n    parentId,\n    postId,\n    privacy,\n    startDate,\n    tags = [],\n    type\n  } = post;\n\n  const update: PostType = {\n    content: content ? parseString(content, MAX_CONTENT_LENGTH) : undefined,\n    endDate: endDate ? parseNum(endDate, 13) : undefined,\n    modified: now,\n    name: name ? parseString(name, 160) : undefined,\n    parentId: parentId ? parseString(parentId, 160) : undefined,\n    privacy: privacy ? parseVarChar(privacy, 16) : undefined,\n    startDate: startDate ? parseNum(startDate, 13) : undefined,\n    type: type !== undefined ? parseChar(type, 16) : undefined\n  };\n\n  let formatId: string = parseId(postId);\n  formatId = formatId === '' ? createHash(`post-${sessionId}`) : formatId;\n  const formatGroupId: string = parseId(groupId);\n  const insert: any = {\n    ...update,\n    _key: formatId,\n    added: now,\n    groupId: formatGroupId,\n    privacy,\n    userId: sessionId\n  };\n  const db: Database = database;\n  const aqlQry: AqlQuery = aql`UPSERT {_key: ${formatId}, userId: ${sessionId}}\n    INSERT ${insert}\n    UPDATE ${update}\n    IN posts RETURN NEW`;\n\n  try {\n    const updatedPost: PostType = await db.query(aqlQry)\n      .then((cursor: ArrayCursor) => cursor.next() || {});\n\n    const {_id: updatedPostId} = updatedPost;\n\n    if(tags && tags.length) {\n      const tagNames: string[] = tags.map(({name}) => parseChar(name, 32));\n      const tagQuery: AqlQuery = aql`FOR t IN tags\n        FILTER POSITION(${JSON.stringify(tagNames)}, t.name)\n        RETURN t`;\n\n      const existingTags = await db.query(tagQuery).then((cursor: ArrayCursor) => cursor.all());\n      const tagIds: string[] = existingTags.map(({_id: tagId}) => tagId);\n      await linkTags(db, tagIds, updatedPostId);\n    } else {\n      // Update linked tags\n      const tagList = await extractTags(db, updatedPostId, update.content || '') || [];\n      updatedPost.tags = tagList;\n    }\n\n    // Update linked files\n    const files: FileType[] = updatedPost.files || [];\n\n    if(files.length) {\n      const fileList = await updateFiles(db, formatId, files) || [];\n      updatedPost.files = fileList;\n    } else {\n      updatedPost.files = [];\n    }\n\n    return updatedPost;\n  } catch(error) {\n    throw error;\n  }\n};\n\nexport const deletePost = (context: ApiContext, postId: string): Promise<PostType> => {\n  // const action: string = 'delete';\n  const {database, session: {userId: sessionId}} = context;\n  const formatItemId: string = parseId(postId);\n  const db: Database = database;\n  const aqlQry = aql`FOR p IN posts\n      FILTER p._key == ${formatItemId} && p.userId == ${sessionId}\n      LIMIT 1\n      REMOVE p IN posts\n      RETURN OLD`;\n\n  return db.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((post: PostType = {}) => {\n      if(post) {\n        // Remove tag links\n        const edgeAqlQry: AqlQuery = aql`FOR t IN isTagged\n            FILTER t._to == ${formatItemId}\n            REMOVE t IN isTagged`;\n\n        return db.query(edgeAqlQry)\n          .then(() => {\n            // Remove attached files\n            const fileAqlQry: AqlQuery = aql`FOR f IN hasFile\n                FILTER f._to == ${formatItemId}\n                REMOVE f IN hasFile`;\n\n            return db.query(fileAqlQry)\n              .then(() => post)\n              .catch((error: Error) => {\n                throw error;\n              });\n          })\n          .catch((error: Error) => {\n            throw error;\n          });\n      }\n      return {};\n    })\n    .catch((error: Error) => {\n      throw error;\n    });\n};\n\nexport const cleanPosts = (database: Database): Promise<number> => {\n  // Remove all messages that are over 60 days and not saved\n  const aqlQry: AqlQuery = aql`FOR p IN posts\n      FILTER p.added < DATE_TIMESTAMP(DATE_SUBTRACT(DATE_NOW(), 60, 'day')) && p.type == \"default\"\n      REMOVE p IN posts\n      RETURN OLD`;\n\n  return database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .then((results: PostType[] = []) => results.length)\n    .catch((error: Error) => {\n      throw error;\n    });\n};\n\nexport const createPostEdge = (db: Database, file: FileType, postId: string): Promise<FileType> => {\n  const edgeCollection: EdgeCollection = db.collection('isPosted');\n  const {fileId} = file;\n  const formatFileId: string = parseId(fileId);\n  const edgeId: string = createHash(`file-${postId}-${formatFileId}`);\n  const formatPostId: string = parseId(postId);\n  const fileType: string = parseChar(file.fileType, 16);\n\n  const edge: any = {\n    _from: `posts/${formatPostId}`,\n    _key: edgeId,\n    _to: `files/${formatFileId}`,\n    added: Date.now(),\n    type: fileType\n  };\n\n  return edgeCollection.save(edge, {returnNew: true})\n    .then(() => file)\n    .catch((error: Error) => {\n      throw error;\n    });\n};\n"],
  "mappings": "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",
  "names": []
}

|