@nlabs/reaktor 0.9.0 → 0.10.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/.env +1 -0
- package/.env.example +1 -0
- package/DATABASE_I18N_GUIDE.md +434 -0
- package/README.md +9 -0
- package/TEST_UTILITIES_GUIDE.md +360 -0
- package/coverage/index.html +61 -46
- package/dist/actions/apps.js +242 -0
- package/dist/actions/connections.js +90 -0
- package/dist/actions/conversations.js +350 -0
- package/dist/actions/dynamodb.js +150 -0
- package/dist/actions/email.js +152 -0
- package/dist/actions/files.js +283 -0
- package/dist/actions/groups.js +292 -0
- package/dist/actions/images.js +735 -0
- package/dist/actions/index.js +66 -0
- package/dist/actions/ios.js +164 -0
- package/dist/actions/locations.js +122 -0
- package/dist/actions/messages.js +208 -0
- package/dist/actions/notifications.js +59 -0
- package/dist/actions/payments.js +497 -0
- package/dist/actions/personas.js +110 -0
- package/dist/actions/posts.js +595 -0
- package/dist/actions/reactions.js +322 -0
- package/dist/actions/s3.js +133 -0
- package/dist/actions/search.js +90 -0
- package/dist/actions/sms.js +108 -0
- package/dist/actions/statistics.js +62 -0
- package/dist/actions/subscription.js +220 -0
- package/dist/actions/tags.js +292 -0
- package/dist/actions/users.js +784 -0
- package/dist/actions/websockets.js +174 -0
- package/dist/adapters/arangoAdapter.js +46 -0
- package/dist/adapters/fileAdapter.js +76 -0
- package/dist/adapters/imageAdapter.js +40 -0
- package/dist/adapters/messageAdapter.js +49 -0
- package/dist/adapters/postAdapter.js +70 -0
- package/dist/adapters/reaktorAdapter.js +44 -0
- package/dist/adapters/tagAdapter.js +50 -0
- package/dist/adapters/userAdapter.js +115 -0
- package/dist/config.js +125 -0
- package/dist/index.js +66 -0
- package/dist/lambdas/actions/websockets.js +132 -0
- package/dist/lambdas/authorizer.js +67 -0
- package/dist/lambdas/connection.js +91 -0
- package/dist/lambdas/utils/message.js +42 -0
- package/dist/lambdas/utils/websocket.js +105 -0
- package/dist/mocks/conversation.js +35 -0
- package/dist/mocks/file.js +38 -0
- package/dist/mocks/group.js +47 -0
- package/dist/mocks/image.js +44 -0
- package/dist/mocks/nlabs.png +0 -0
- package/dist/mocks/post.js +55 -0
- package/dist/mocks/tag.js +37 -0
- package/dist/mocks/user.js +88 -0
- package/dist/mutations/index.js +26 -0
- package/dist/mutations/locations.js +44 -0
- package/dist/mutations/messages.js +86 -0
- package/dist/mutations/personas.js +100 -0
- package/dist/mutations/posts.js +53 -0
- package/dist/mutations/reactions.js +51 -0
- package/dist/mutations/statistics.js +39 -0
- package/dist/mutations/subscriptions.js +56 -0
- package/dist/mutations/tags.js +120 -0
- package/dist/mutations/users.js +116 -0
- package/dist/objectTypes/app.js +173 -0
- package/dist/objectTypes/bankAccount.js +76 -0
- package/dist/objectTypes/connection.js +48 -0
- package/dist/objectTypes/conversation.js +77 -0
- package/dist/objectTypes/creditCard.js +86 -0
- package/dist/objectTypes/document.js +46 -0
- package/dist/objectTypes/error.js +46 -0
- package/dist/objectTypes/external.js +74 -0
- package/dist/objectTypes/file.js +100 -0
- package/dist/objectTypes/filter.js +43 -0
- package/dist/objectTypes/group.js +123 -0
- package/dist/objectTypes/iapSubscription.js +40 -0
- package/dist/objectTypes/image.js +129 -0
- package/dist/objectTypes/index.js +68 -0
- package/dist/objectTypes/location.js +109 -0
- package/dist/objectTypes/message.js +96 -0
- package/dist/objectTypes/passcode.js +42 -0
- package/dist/objectTypes/persona.js +87 -0
- package/dist/objectTypes/plan.js +95 -0
- package/dist/objectTypes/post.js +125 -0
- package/dist/objectTypes/reaction.js +61 -0
- package/dist/objectTypes/relation.js +49 -0
- package/dist/objectTypes/search.js +72 -0
- package/dist/objectTypes/statistics.js +39 -0
- package/dist/objectTypes/subscription.js +117 -0
- package/dist/objectTypes/tag.js +65 -0
- package/dist/objectTypes/user.js +144 -0
- package/dist/queries/index.js +33 -0
- package/dist/queries/locations.js +45 -0
- package/dist/queries/messages.js +52 -0
- package/dist/queries/posts.js +154 -0
- package/dist/queries/reactions.js +56 -0
- package/dist/queries/statistics.js +39 -0
- package/dist/queries/subscriptions.js +44 -0
- package/dist/queries/tags.js +75 -0
- package/dist/queries/users.js +64 -0
- package/dist/templates/email/layout.js +302 -0
- package/dist/templates/email/passwordForgot.js +38 -0
- package/dist/templates/email/passwordRecovery.js +35 -0
- package/dist/templates/email/verifyEmail.js +38 -0
- package/dist/templates/email/welcome.js +38 -0
- package/dist/templates/sms/passwordForgot.js +24 -0
- package/dist/templates/sms/passwordRecovery.js +24 -0
- package/dist/templates/sms/verifyEmail.js +24 -0
- package/dist/templates/sms/verifyPhone.js +24 -0
- package/dist/templates/sms/welcome.js +24 -0
- package/dist/types/apps.js +32 -0
- package/{lib → dist}/types/arangodb.js +1 -1
- package/{lib → dist}/types/auth.js +1 -1
- package/{lib → dist}/types/connections.js +1 -1
- package/dist/types/conversations.js +16 -0
- package/{lib → dist}/types/email.js +1 -1
- package/dist/types/files.js +16 -0
- package/dist/types/google.js +16 -0
- package/{lib → dist}/types/groups.js +1 -1
- package/dist/types/images.js +16 -0
- package/dist/types/index.js +60 -0
- package/{lib → dist}/types/locations.js +1 -1
- package/{lib → dist}/types/messages.js +1 -1
- package/{lib → dist}/types/notifications.js +1 -1
- package/dist/types/payments.js +16 -0
- package/dist/types/personas.js +16 -0
- package/dist/types/posts.js +16 -0
- package/{lib → dist}/types/tags.js +1 -1
- package/dist/types/users.js +16 -0
- package/dist/types/websockets.js +16 -0
- package/dist/utils/adapterUtils.js +45 -0
- package/dist/utils/analyticsUtils.js +72 -0
- package/dist/utils/arangodbUtils.js +165 -0
- package/dist/utils/auth.js +57 -0
- package/dist/utils/index.js +30 -0
- package/{lib → dist}/utils/session.js +10 -7
- package/index.js +1 -1
- package/jest.config.js +17 -0
- package/jest.setup.js +36 -0
- package/lex.config.cjs +13 -0
- package/lib/actions/apps.js +17 -249
- package/lib/actions/connections.js +7 -0
- package/lib/actions/content.js +17 -0
- package/lib/actions/conversations.js +19 -325
- package/lib/actions/dynamodb.js +2 -150
- package/lib/actions/email.js +2 -152
- package/lib/actions/files.js +5 -287
- package/lib/actions/groups.js +23 -263
- package/lib/actions/images.js +31 -646
- package/lib/actions/index.js +2 -62
- package/lib/actions/ios.js +9 -162
- package/lib/actions/locations.js +7 -110
- package/lib/actions/messages.js +21 -193
- package/lib/actions/notifications.js +2 -59
- package/lib/actions/payments.js +11 -461
- package/lib/actions/posts.js +77 -515
- package/lib/actions/profiles.js +8 -0
- package/lib/actions/reactions.js +25 -312
- package/lib/actions/s3.js +2 -133
- package/lib/actions/search.js +5 -90
- package/lib/actions/sms.js +2 -106
- package/lib/actions/statistics.js +6 -60
- package/lib/actions/subscriptions.js +12 -0
- package/lib/actions/tags.js +19 -262
- package/lib/actions/users.js +67 -537
- package/lib/actions/websockets.js +16 -158
- package/lib/adapters/arangoAdapter.js +2 -46
- package/lib/adapters/contentAdapter.js +2 -0
- package/lib/adapters/fileAdapter.js +2 -76
- package/lib/adapters/imageAdapter.js +2 -0
- package/lib/adapters/messageAdapter.js +2 -0
- package/lib/adapters/postAdapter.js +2 -70
- package/lib/adapters/reaktorAdapter.js +2 -44
- package/lib/adapters/tagAdapter.js +2 -50
- package/lib/adapters/userAdapter.js +2 -110
- package/lib/config.js +2 -125
- package/lib/handlers/graphqlHandler.js +2 -0
- package/lib/index.js +2 -30
- package/lib/lambdas/actions/websockets.js +14 -112
- package/lib/lambdas/authorizer.js +2 -67
- package/lib/lambdas/connection.js +2 -92
- package/lib/lambdas/utils/message.js +2 -42
- package/lib/lambdas/utils/websocket.js +2 -105
- package/lib/mocks/conversation.js +2 -35
- package/lib/mocks/file.js +2 -38
- package/lib/mocks/group.js +2 -47
- package/lib/mocks/image.js +2 -43
- package/lib/mocks/post.js +2 -55
- package/lib/mocks/tag.js +2 -37
- package/lib/mocks/user.js +2 -88
- package/lib/mutations/index.js +2 -0
- package/lib/mutations/locations.integration.js +2 -0
- package/lib/mutations/locations.js +2 -0
- package/lib/mutations/messages.integration.js +2 -0
- package/lib/mutations/messages.js +2 -0
- package/lib/mutations/posts.integration.js +2 -0
- package/lib/mutations/posts.js +2 -0
- package/lib/mutations/profiles.integration.js +2 -0
- package/lib/mutations/profiles.js +2 -0
- package/lib/mutations/reactions.integration.js +2 -0
- package/lib/mutations/reactions.js +2 -0
- package/lib/mutations/statistics.integration.js +2 -0
- package/lib/mutations/statistics.js +2 -0
- package/lib/mutations/subscriptions.integration.js +2 -0
- package/lib/mutations/subscriptions.js +2 -0
- package/lib/mutations/tags.integration.js +2 -0
- package/lib/mutations/tags.js +2 -0
- package/lib/mutations/users.integration.js +2 -0
- package/lib/mutations/users.js +2 -0
- package/lib/objectTypes/app.js +2 -0
- package/lib/objectTypes/bankAccount.js +2 -0
- package/lib/objectTypes/connection.js +2 -0
- package/lib/objectTypes/conversation.js +2 -0
- package/lib/objectTypes/creditCard.js +2 -0
- package/lib/objectTypes/document.js +2 -0
- package/lib/objectTypes/error.js +2 -0
- package/lib/objectTypes/external.js +2 -0
- package/lib/objectTypes/file.js +2 -0
- package/lib/objectTypes/filter.js +2 -0
- package/lib/objectTypes/group.js +2 -0
- package/lib/objectTypes/iapSubscription.js +2 -0
- package/lib/objectTypes/image.js +2 -0
- package/lib/objectTypes/index.js +2 -0
- package/lib/objectTypes/location.js +2 -0
- package/lib/objectTypes/message.js +2 -0
- package/lib/objectTypes/passcode.js +2 -0
- package/lib/objectTypes/plan.js +2 -0
- package/lib/objectTypes/post.js +2 -0
- package/lib/objectTypes/profile.js +2 -0
- package/lib/objectTypes/reaction.js +2 -0
- package/lib/objectTypes/relation.js +2 -0
- package/lib/objectTypes/search.js +2 -0
- package/lib/objectTypes/statistics.js +2 -0
- package/lib/objectTypes/subscription.js +2 -0
- package/lib/objectTypes/tag.js +2 -0
- package/lib/objectTypes/user.js +2 -0
- package/lib/queries/index.js +2 -0
- package/lib/queries/locations.integration.js +2 -0
- package/lib/queries/locations.js +2 -0
- package/lib/queries/messages.integration.js +2 -0
- package/lib/queries/messages.js +2 -0
- package/lib/queries/posts.integration.js +2 -0
- package/lib/queries/posts.js +2 -0
- package/lib/queries/reactions.integration.js +2 -0
- package/lib/queries/reactions.js +2 -0
- package/lib/queries/statistics.js +2 -0
- package/lib/queries/subscriptions.js +2 -0
- package/lib/queries/tags.integration.js +2 -0
- package/lib/queries/tags.js +2 -0
- package/lib/queries/users.integration.js +2 -0
- package/lib/queries/users.js +2 -0
- package/lib/templates/email/layout.js +3 -25
- package/lib/templates/email/passwordForgot.js +3 -25
- package/lib/templates/email/passwordRecovery.js +3 -25
- package/lib/templates/email/verifyEmail.js +3 -25
- package/lib/templates/email/welcome.js +3 -25
- package/lib/templates/sms/passwordForgot.js +2 -24
- package/lib/templates/sms/passwordRecovery.js +2 -24
- package/lib/templates/sms/verifyEmail.js +2 -24
- package/lib/templates/sms/verifyPhone.js +2 -24
- package/lib/templates/sms/welcome.js +2 -24
- package/lib/types/apps.types.js +2 -0
- package/lib/types/arangodb.types.js +1 -0
- package/lib/types/auth.types.js +1 -0
- package/lib/types/connections.types.js +1 -0
- package/lib/types/content.types.js +1 -0
- package/lib/types/conversations.types.js +1 -0
- package/lib/types/email.types.js +1 -0
- package/lib/types/error.types.js +2 -0
- package/lib/types/files.types.js +1 -0
- package/lib/types/google.types.js +1 -0
- package/lib/types/groups.types.js +1 -0
- package/lib/types/images.types.js +1 -0
- package/lib/types/index.js +2 -56
- package/lib/types/locations.types.js +1 -0
- package/lib/types/messages.types.js +1 -0
- package/lib/types/notifications.types.js +1 -0
- package/lib/types/payments.types.js +1 -0
- package/lib/types/posts.types.js +1 -0
- package/lib/types/profiles.types.js +1 -0
- package/lib/types/statistics.types.js +1 -0
- package/lib/types/tags.types.js +1 -0
- package/lib/types/users.types.js +1 -0
- package/lib/types/websockets.types.js +1 -0
- package/lib/utils/adapterUtils.js +2 -45
- package/lib/utils/analyticsUtils.js +2 -72
- package/lib/utils/arangodbUtils.js +5 -142
- package/lib/utils/authUtils.js +2 -0
- package/lib/utils/contextUtils.js +2 -0
- package/lib/utils/dbI18n.example.js +6 -0
- package/lib/utils/dbI18n.js +2 -0
- package/lib/utils/googleTranslate.js +2 -0
- package/lib/utils/graphqlUtils.js +2 -0
- package/lib/utils/index.js +2 -30
- package/lib/utils/languageDetection.js +2 -0
- package/lib/utils/localeUtils.example.js +2 -0
- package/lib/utils/localeUtils.js +2 -0
- package/lib/utils/middlewareUtils.js +2 -0
- package/lib/utils/sessionUtils.js +2 -0
- package/lib/utils/stripeUtils.js +2 -0
- package/lib/utils/templateUtils.js +2 -0
- package/lib/utils/testUtils.js +2 -0
- package/lib/utils/translationQueue.example.js +2 -0
- package/lib/utils/translationQueue.js +2 -0
- package/package.json +41 -37
- package/.eslintrc +0 -10
- package/.prettierrc.js +0 -4
- package/lib/actions/apps.d.ts +0 -25
- package/lib/actions/conversations.d.ts +0 -14
- package/lib/actions/dynamodb.d.ts +0 -8
- package/lib/actions/email.d.ts +0 -5
- package/lib/actions/files.d.ts +0 -19
- package/lib/actions/groups.d.ts +0 -14
- package/lib/actions/images.d.ts +0 -26
- package/lib/actions/index.d.ts +0 -21
- package/lib/actions/ios.d.ts +0 -7
- package/lib/actions/locations.d.ts +0 -5
- package/lib/actions/messages.d.ts +0 -13
- package/lib/actions/notifications.d.ts +0 -5
- package/lib/actions/payments.d.ts +0 -10
- package/lib/actions/posts.d.ts +0 -19
- package/lib/actions/reactions.d.ts +0 -30
- package/lib/actions/s3.d.ts +0 -7
- package/lib/actions/search.d.ts +0 -3
- package/lib/actions/sms.d.ts +0 -3
- package/lib/actions/statistics.d.ts +0 -3
- package/lib/actions/subscription.d.ts +0 -7
- package/lib/actions/subscription.js +0 -218
- package/lib/actions/tags.d.ts +0 -29
- package/lib/actions/users.d.ts +0 -47
- package/lib/actions/websockets.d.ts +0 -19
- package/lib/adapters/arangoAdapter.d.ts +0 -2
- package/lib/adapters/fileAdapter.d.ts +0 -3
- package/lib/adapters/postAdapter.d.ts +0 -2
- package/lib/adapters/reaktorAdapter.d.ts +0 -6
- package/lib/adapters/tagAdapter.d.ts +0 -2
- package/lib/adapters/userAdapter.d.ts +0 -2
- package/lib/config.d.ts +0 -20
- package/lib/index.d.ts +0 -5
- package/lib/lambdas/actions/websockets.d.ts +0 -6
- package/lib/lambdas/authorizer.d.ts +0 -20
- package/lib/lambdas/connection.d.ts +0 -12
- package/lib/lambdas/utils/message.d.ts +0 -1
- package/lib/lambdas/utils/websocket.d.ts +0 -7
- package/lib/mocks/conversation.d.ts +0 -8
- package/lib/mocks/file.d.ts +0 -11
- package/lib/mocks/group.d.ts +0 -17
- package/lib/mocks/image.d.ts +0 -3
- package/lib/mocks/post.d.ts +0 -38
- package/lib/mocks/tag.d.ts +0 -2
- package/lib/mocks/user.d.ts +0 -4
- package/lib/templates/email/layout.d.ts +0 -2
- package/lib/templates/email/passwordForgot.d.ts +0 -2
- package/lib/templates/email/passwordRecovery.d.ts +0 -2
- package/lib/templates/email/verifyEmail.d.ts +0 -2
- package/lib/templates/email/welcome.d.ts +0 -2
- package/lib/templates/sms/passwordForgot.d.ts +0 -2
- package/lib/templates/sms/passwordRecovery.d.ts +0 -2
- package/lib/templates/sms/verifyEmail.d.ts +0 -2
- package/lib/templates/sms/verifyPhone.d.ts +0 -2
- package/lib/templates/sms/welcome.d.ts +0 -2
- package/lib/types/apps.d.ts +0 -46
- package/lib/types/apps.js +0 -32
- package/lib/types/arangodb.d.ts +0 -30
- package/lib/types/auth.d.ts +0 -7
- package/lib/types/connections.d.ts +0 -7
- package/lib/types/conversations.d.ts +0 -29
- package/lib/types/conversations.js +0 -16
- package/lib/types/email.d.ts +0 -13
- package/lib/types/error.d.ts +0 -20
- package/lib/types/files.d.ts +0 -26
- package/lib/types/files.js +0 -16
- package/lib/types/google.d.ts +0 -29
- package/lib/types/google.js +0 -16
- package/lib/types/groups.d.ts +0 -21
- package/lib/types/images.d.ts +0 -51
- package/lib/types/images.js +0 -16
- package/lib/types/index.d.ts +0 -18
- package/lib/types/locations.d.ts +0 -20
- package/lib/types/messages.d.ts +0 -16
- package/lib/types/notifications.d.ts +0 -17
- package/lib/types/payments.d.ts +0 -112
- package/lib/types/payments.js +0 -16
- package/lib/types/posts.d.ts +0 -31
- package/lib/types/posts.js +0 -16
- package/lib/types/statistics.d.ts +0 -3
- package/lib/types/tags.d.ts +0 -10
- package/lib/types/users.d.ts +0 -76
- package/lib/types/users.js +0 -16
- package/lib/types/websocket.d.ts +0 -13
- package/lib/types/websocket.js +0 -16
- package/lib/utils/adapterUtils.d.ts +0 -1
- package/lib/utils/analyticsUtils.d.ts +0 -21
- package/lib/utils/arangodbUtils.d.ts +0 -65
- package/lib/utils/auth.d.ts +0 -20
- package/lib/utils/auth.js +0 -50
- package/lib/utils/index.d.ts +0 -5
- package/lib/utils/session.d.ts +0 -16
- /package/{lib → dist}/types/error.js +0 -0
- /package/{lib → dist}/types/statistics.js +0 -0
package/lib/actions/posts.js
CHANGED
|
@@ -1,346 +1,112 @@
|
|
|
1
|
-
|
|
2
|
-
var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
|
|
3
|
-
var __getOwnPropNames = Object.getOwnPropertyNames;
|
|
4
|
-
var __hasOwnProp = Object.prototype.hasOwnProperty;
|
|
5
|
-
var __export = (target, all) => {
|
|
6
|
-
for (var name in all)
|
|
7
|
-
__defProp(target, name, { get: all[name], enumerable: true });
|
|
8
|
-
};
|
|
9
|
-
var __copyProps = (to, from, except, desc) => {
|
|
10
|
-
if (from && typeof from === "object" || typeof from === "function") {
|
|
11
|
-
for (let key of __getOwnPropNames(from))
|
|
12
|
-
if (!__hasOwnProp.call(to, key) && key !== except)
|
|
13
|
-
__defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
|
|
14
|
-
}
|
|
15
|
-
return to;
|
|
16
|
-
};
|
|
17
|
-
var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
|
|
18
|
-
var posts_exports = {};
|
|
19
|
-
__export(posts_exports, {
|
|
20
|
-
addPost: () => addPost,
|
|
21
|
-
createPostEdge: () => createPostEdge,
|
|
22
|
-
deletePost: () => deletePost,
|
|
23
|
-
getPost: () => getPost,
|
|
24
|
-
getPostComments: () => getPostComments,
|
|
25
|
-
getPostOptional: () => getPostOptional,
|
|
26
|
-
getPostsByArea: () => getPostsByArea,
|
|
27
|
-
getPostsByLatest: () => getPostsByLatest,
|
|
28
|
-
getPostsByReactions: () => getPostsByReactions,
|
|
29
|
-
getPostsByTags: () => getPostsByTags,
|
|
30
|
-
getPostsByUser: () => getPostsByUser,
|
|
31
|
-
parsePostOptions: () => parsePostOptions,
|
|
32
|
-
updatePost: () => updatePost
|
|
33
|
-
});
|
|
34
|
-
module.exports = __toCommonJS(posts_exports);
|
|
35
|
-
var import_arangojs = require("arangojs");
|
|
36
|
-
var import_utils = require("@nlabs/utils");
|
|
37
|
-
var import_postAdapter = require("../adapters/postAdapter");
|
|
38
|
-
var import_error = require("../types/error");
|
|
39
|
-
var import_analyticsUtils = require("../utils/analyticsUtils");
|
|
40
|
-
var import_arangodbUtils = require("../utils/arangodbUtils");
|
|
41
|
-
var import_tags = require("./tags");
|
|
42
|
-
const MAX_CONTENT_LENGTH = 1e5;
|
|
43
|
-
const eventCategory = "posts";
|
|
44
|
-
const parsePostOptions = (options = {}) => {
|
|
45
|
-
const {
|
|
46
|
-
from = 0,
|
|
47
|
-
latitude = 0,
|
|
48
|
-
longitude = 0,
|
|
49
|
-
to = 30,
|
|
50
|
-
type = "post"
|
|
51
|
-
} = options;
|
|
52
|
-
return {
|
|
53
|
-
latitude: (0, import_utils.parseNum)(latitude, 32),
|
|
54
|
-
limit: (0, import_arangodbUtils.getLimit)(from, to),
|
|
55
|
-
longitude: (0, import_utils.parseNum)(longitude, 32),
|
|
56
|
-
type: (0, import_utils.parseChar)(type, 32)
|
|
57
|
-
};
|
|
58
|
-
};
|
|
59
|
-
const getPostOptional = (fields, sessionId) => (fields || []).reduce((selects, field) => {
|
|
60
|
-
switch (field) {
|
|
61
|
-
case "hasRsvp": {
|
|
62
|
-
selects.queries.push(`LET hasRsvp = TO_BOOL(FIRST(
|
|
1
|
+
import{createHash as B,parseArangoId as S,parseChar as _,parseId as D,parseNum as L,parseString as q,parseVarChar as k}from"@nlabs/utils";import{aql as A}from"arangojs";import{extractTags as w,getTagsByName as U,updateTagsInItem as j}from"./tags";import{parsePost as H}from"../adapters/postAdapter";import{ErrorTypes as N}from"../types/error.types";import{logError as f,logException as Q}from"../utils/analyticsUtils";import{getLimit as V,useDb as y}from"../utils/arangodbUtils";const W=1e5,O="posts",$=(e={})=>{const{from:r=0,latitude:t=0,longitude:o=0,to:i=30,type:n="post"}=e;return{latitude:L(t,32),limit:V(r,i),longitude:L(o,32),type:_(n,32)}},C=(e,r)=>(e||[]).reduce((t,o)=>{switch(o){case"hasRsvp":return t.queries.push(`LET hasRsvp = TO_BOOL(FIRST(
|
|
63
2
|
FOR post, r IN INBOUND p._id hasReaction
|
|
64
|
-
FILTER r.name == "rsvp" && r.type == "posts" && r._from == "users/${
|
|
3
|
+
FILTER r.name == "rsvp" && r.type == "posts" && r._from == "users/${r}"
|
|
65
4
|
COLLECT WITH COUNT INTO count
|
|
66
5
|
RETURN count
|
|
67
|
-
))`);
|
|
68
|
-
selects.objects.push("hasRsvp:hasRsvp");
|
|
69
|
-
return selects;
|
|
70
|
-
}
|
|
71
|
-
case "isSaved": {
|
|
72
|
-
selects.queries.push(`LET isSaved = TO_BOOL(FIRST(
|
|
6
|
+
))`),t.objects.push("hasRsvp:hasRsvp"),t;case"isSaved":return t.queries.push(`LET isSaved = TO_BOOL(FIRST(
|
|
73
7
|
FOR post, r IN INBOUND p._id hasReaction
|
|
74
|
-
FILTER r.name == "pin" && r.type == "posts" && r._from == "users/${
|
|
8
|
+
FILTER r.name == "pin" && r.type == "posts" && r._from == "users/${r}"
|
|
75
9
|
COLLECT WITH COUNT INTO count
|
|
76
10
|
RETURN count
|
|
77
|
-
))`);
|
|
78
|
-
selects.objects.push("isSaved:isSaved");
|
|
79
|
-
return selects;
|
|
80
|
-
}
|
|
81
|
-
case "reactions": {
|
|
82
|
-
selects.queries.push(`LET reactions = (
|
|
11
|
+
))`),t.objects.push("isSaved:isSaved"),t;case"reactions":return t.queries.push(`LET reactions = (
|
|
83
12
|
FOR post, r IN INBOUND p._id hasReaction
|
|
84
13
|
COLLECT reactionName = r.value INTO reactionItems
|
|
85
14
|
RETURN {name: reactionName, count: LENGTH(reactionItems[*].r.value)}
|
|
86
|
-
)`);
|
|
87
|
-
selects.objects.push("reactions:reactions");
|
|
88
|
-
return selects;
|
|
89
|
-
}
|
|
90
|
-
case "rsvpCount": {
|
|
91
|
-
selects.queries.push(`LET rsvpCount = FIRST(
|
|
15
|
+
)`),t.objects.push("reactions:reactions"),t;case"rsvpCount":return t.queries.push(`LET rsvpCount = FIRST(
|
|
92
16
|
FOR post, r IN INBOUND p._id hasReaction
|
|
93
17
|
FILTER r.name == "rsvp" && r.type == "posts"
|
|
94
18
|
COLLECT WITH COUNT INTO count
|
|
95
19
|
RETURN count
|
|
96
|
-
)`);
|
|
97
|
-
selects.objects.push("rsvpCount:rsvpCount");
|
|
98
|
-
return selects;
|
|
99
|
-
}
|
|
100
|
-
case "viewCount": {
|
|
101
|
-
selects.queries.push(`LET viewCount = FIRST(
|
|
20
|
+
)`),t.objects.push("rsvpCount:rsvpCount"),t;case"viewCount":return t.queries.push(`LET viewCount = FIRST(
|
|
102
21
|
FOR post, r IN INBOUND p._id hasReaction
|
|
103
22
|
FILTER r.name == "view" && r.type == "posts"
|
|
104
23
|
COLLECT WITH COUNT INTO count
|
|
105
24
|
RETURN count
|
|
106
|
-
)`);
|
|
107
|
-
|
|
108
|
-
return selects;
|
|
109
|
-
}
|
|
110
|
-
default: {
|
|
111
|
-
return selects;
|
|
112
|
-
}
|
|
113
|
-
}
|
|
114
|
-
}, { objects: [], queries: [] });
|
|
115
|
-
const getPost = async (context, postId, options) => {
|
|
116
|
-
const action = "getPost";
|
|
117
|
-
const { database, fields, session: { userId: sessionId } } = context;
|
|
118
|
-
const formatItemId = (0, import_utils.parseId)(postId);
|
|
119
|
-
const { type } = parsePostOptions(options);
|
|
120
|
-
const db = database;
|
|
121
|
-
const { objects: selectObjects, queries: selectQueries } = getPostOptional(fields, sessionId);
|
|
122
|
-
const aqlQry = import_arangojs.aql`FOR p IN posts
|
|
123
|
-
FILTER p._key == ${formatItemId} && p.type == ${type}
|
|
25
|
+
)`),t.objects.push("viewCount:viewCount"),t;default:return t}},{objects:[],queries:[]}),et=async(e,r,t)=>{const o="getPost",{databaseName:i,fields:n,session:{userId:p}}=e,R=D(r),{type:c}=$(t),s=y(i),{objects:a,queries:u}=C(n,p),d=A`FOR p IN posts
|
|
26
|
+
FILTER p._key == ${R} && p.type == ${c}
|
|
124
27
|
LIMIT 1
|
|
125
|
-
RETURN p`;
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
_id: postDocId,
|
|
129
|
-
userId,
|
|
130
|
-
groupId,
|
|
131
|
-
privacy = "default"
|
|
132
|
-
} = post;
|
|
133
|
-
let privacyAqlQry;
|
|
134
|
-
if (userId === sessionId) {
|
|
135
|
-
return post;
|
|
136
|
-
}
|
|
137
|
-
if (groupId && privacy === "group") {
|
|
138
|
-
privacyAqlQry = `LET p = DOCUMENT("${postDocId}")
|
|
139
|
-
${selectQueries.join("\n")}
|
|
28
|
+
RETURN p`;return s.query(d).then(I=>I.next()).then(I=>{const{_id:E,userId:T,groupId:h,privacy:m="default"}=I;let g;return T===p?I:(h&&m==="group"?g=`LET p = DOCUMENT("${E}")
|
|
29
|
+
${u.join(`
|
|
30
|
+
`)}
|
|
140
31
|
FOR group IN groups
|
|
141
32
|
FILTER group._key == p.groupId
|
|
142
|
-
FOR u, e IN OUTBOUND group._id
|
|
143
|
-
FILTER u._key == "${
|
|
33
|
+
FOR u, e IN OUTBOUND group._id hasConnection
|
|
34
|
+
FILTER u._key == "${p}"
|
|
144
35
|
LIMIT 1
|
|
145
|
-
RETURN MERGE(p, {${
|
|
146
|
-
|
|
147
|
-
|
|
148
|
-
${selectQueries.join("\n")}
|
|
36
|
+
RETURN MERGE(p, {${a.join(", ")}})`:m==="public"&&(g=`LET p = DOCUMENT("${E}")
|
|
37
|
+
${u.join(`
|
|
38
|
+
`)}
|
|
149
39
|
LIMIT 1
|
|
150
|
-
RETURN MERGE(p, {${
|
|
151
|
-
}
|
|
152
|
-
|
|
153
|
-
return db.query(privacyAqlQry).then((cursor) => cursor.next()).catch((error) => (0, import_analyticsUtils.logError)({
|
|
154
|
-
action,
|
|
155
|
-
category: eventCategory,
|
|
156
|
-
label: import_error.ErrorTypes.DATABASE_ERROR
|
|
157
|
-
}, error, context));
|
|
158
|
-
}
|
|
159
|
-
return {};
|
|
160
|
-
}).catch((error) => (0, import_analyticsUtils.logError)({
|
|
161
|
-
action,
|
|
162
|
-
category: eventCategory,
|
|
163
|
-
label: import_error.ErrorTypes.DATABASE_ERROR
|
|
164
|
-
}, error, context));
|
|
165
|
-
};
|
|
166
|
-
const getPostsByArea = (context, latitude, longitude, options) => {
|
|
167
|
-
const action = "getPostsByArea";
|
|
168
|
-
const { database, fields, session: { userId: sessionId } } = context;
|
|
169
|
-
const { limit, type } = parsePostOptions(options);
|
|
170
|
-
const formatLatitude = (0, import_utils.parseNum)(latitude);
|
|
171
|
-
const formatLongitude = (0, import_utils.parseNum)(longitude);
|
|
172
|
-
const { objects: selectObjects, queries: selectQueries } = getPostOptional(fields, sessionId);
|
|
173
|
-
selectQueries.push(`LET distance = DISTANCE(
|
|
174
|
-
${formatLatitude},
|
|
175
|
-
${formatLongitude},
|
|
40
|
+
RETURN MERGE(p, {${a.join(", ")}})`),g?s.query(g).then(l=>l.next()).catch(l=>f({action:o,category:O,label:N.DATABASE_ERROR},l,e)):{})}).catch(I=>f({action:o,category:O,label:N.DATABASE_ERROR},I,e))},st=(e,r,t,o)=>{const i="getPostsByArea",{databaseName:n,fields:p,session:{userId:R}}=e,{limit:c,type:s}=$(o),a=L(r),u=L(t),{objects:d,queries:I}=C(p,R);I.push(`LET distance = DISTANCE(
|
|
41
|
+
${a},
|
|
42
|
+
${u},
|
|
176
43
|
NOT_NULL(p.latitude, 0),
|
|
177
44
|
NOT_NULL(p.longitude, 0))
|
|
178
|
-
`);
|
|
179
|
-
|
|
180
|
-
|
|
181
|
-
${
|
|
182
|
-
|
|
183
|
-
${limit.aql}
|
|
45
|
+
`),d.push("distance:distance");const E=`FOR p IN posts
|
|
46
|
+
${I.join(`
|
|
47
|
+
`)}
|
|
48
|
+
FILTER p.type == "${s}" && p.privacy == "public" && p.parentId == null
|
|
49
|
+
${c.aql}
|
|
184
50
|
SORT distance, p.added
|
|
185
|
-
RETURN DISTINCT MERGE(p, {${
|
|
186
|
-
|
|
187
|
-
|
|
188
|
-
|
|
189
|
-
|
|
190
|
-
}, error, context));
|
|
191
|
-
};
|
|
192
|
-
const getPostsByLatest = (context, options) => {
|
|
193
|
-
const { database, fields, session: { userId: sessionId } } = context;
|
|
194
|
-
const { limit, type } = parsePostOptions(options);
|
|
195
|
-
const { objects: selectObjects, queries: selectQueries } = getPostOptional(fields, sessionId);
|
|
196
|
-
const aqlQry = `FOR p IN posts
|
|
197
|
-
FILTER p.type == "${type}" && p.privacy == "public" && p.parent == null
|
|
198
|
-
${selectQueries.join("\n")}
|
|
199
|
-
${limit.aql}
|
|
51
|
+
RETURN DISTINCT MERGE(p, {${d.join(", ")}})`;return y(n).query(E).then(T=>T.all()).catch(T=>(f({action:i,category:O,label:N.DATABASE_ERROR},T,e),[]))},ot=(e,r)=>{const{databaseName:t,fields:o,session:{userId:i}}=e,{limit:n,type:p}=$(r),{objects:R,queries:c}=C(o,i),s=`FOR p IN posts
|
|
52
|
+
FILTER p.type == "${p}" && p.privacy == "public" && p.parent == null
|
|
53
|
+
${c.join(`
|
|
54
|
+
`)}
|
|
55
|
+
${n.aql}
|
|
200
56
|
SORT p.added DESC
|
|
201
|
-
RETURN DISTINCT MERGE(p, {${
|
|
202
|
-
|
|
203
|
-
|
|
204
|
-
});
|
|
205
|
-
};
|
|
206
|
-
const getPostsByReactions = (context, reactions = [], options) => {
|
|
207
|
-
const action = "getPostsByReactions";
|
|
208
|
-
const { database, fields, session: { userId: sessionId } } = context;
|
|
209
|
-
const { latitude, limit, longitude, type } = parsePostOptions(options);
|
|
210
|
-
const { objects: selectObjects, queries: selectQueries } = getPostOptional(fields, sessionId);
|
|
211
|
-
const formatSessionId = `users/${sessionId}`;
|
|
212
|
-
const formatReactions = JSON.stringify(reactions.map((reaction) => (0, import_utils.parseChar)(reaction, 32).toLowerCase()));
|
|
213
|
-
const sortBy = [];
|
|
214
|
-
const filters = [`p.type == "${type}"`, 'p.privacy == "public"'];
|
|
215
|
-
const formatLatitude = (0, import_utils.parseNum)(latitude);
|
|
216
|
-
const formatLongitude = (0, import_utils.parseNum)(longitude);
|
|
217
|
-
if (formatLatitude && formatLongitude) {
|
|
218
|
-
selectQueries.push(`LET distance = DISTANCE(
|
|
219
|
-
${formatLatitude},
|
|
220
|
-
${formatLongitude},
|
|
57
|
+
RETURN DISTINCT MERGE(p, {${R.join(", ")}})`;return y(t).query(s).then(a=>a.all()).catch(a=>{throw a})},rt=(e,r=[],t)=>{const o="getPostsByReactions",{databaseName:i,fields:n,session:{userId:p}}=e,{latitude:R,limit:c,longitude:s,type:a}=$(t),{objects:u,queries:d}=C(n,p),I=`users/${p}`,E=JSON.stringify(r.map(P=>_(P,32).toLowerCase())),T=[],h=[`p.type == "${a}"`,'p.privacy == "public"'],m=L(R),g=L(s);m&&g&&(d.push(`LET distance = DISTANCE(
|
|
58
|
+
${m},
|
|
59
|
+
${g},
|
|
221
60
|
NOT_NULL(p.latitude, 0),
|
|
222
61
|
NOT_NULL(p.longitude, 0))
|
|
223
|
-
`)
|
|
224
|
-
selectObjects.push("distance:distance");
|
|
225
|
-
sortBy.push("distance");
|
|
226
|
-
}
|
|
227
|
-
if (reactions.length) {
|
|
228
|
-
sortBy.push("matchedTags DESC");
|
|
229
|
-
selectQueries.push(`LET matchedReactions = LENGTH(
|
|
62
|
+
`),u.push("distance:distance"),T.push("distance")),r.length&&(T.push("matchedTags DESC"),d.push(`LET matchedReactions = LENGTH(
|
|
230
63
|
FOR mr IN reactions
|
|
231
64
|
FILTER mr.matched == true
|
|
232
65
|
RETURN mr
|
|
233
|
-
)`);
|
|
234
|
-
selectObjects.push("matchedReactions:matchedReactions");
|
|
235
|
-
filters.push("matchedReactions > 0");
|
|
236
|
-
}
|
|
237
|
-
sortBy.push("p.added DESC");
|
|
238
|
-
selectObjects.push("reactions:reactions");
|
|
239
|
-
const aqlQry = `FOR p, r IN OUTBOUND "${formatSessionId}" hasReaction
|
|
66
|
+
)`),u.push("matchedReactions:matchedReactions"),h.push("matchedReactions > 0")),T.push("p.added DESC"),u.push("reactions:reactions");const l=`FOR p, r IN OUTBOUND "${I}" hasReaction
|
|
240
67
|
LET reactions = (
|
|
241
68
|
FOR reaction, hr IN 1..1 INBOUND p isTagged
|
|
242
|
-
LET matched = LENGTH(${
|
|
69
|
+
LET matched = LENGTH(${E}) > 0 && POSITION(${E}, reaction.name)
|
|
243
70
|
SORT reaction.name
|
|
244
71
|
RETURN MERGE(reaction, {matched:matched})
|
|
245
72
|
)
|
|
246
|
-
${
|
|
247
|
-
|
|
248
|
-
${
|
|
249
|
-
|
|
250
|
-
|
|
251
|
-
|
|
252
|
-
|
|
253
|
-
label: import_error.ErrorTypes.DATABASE_ERROR
|
|
254
|
-
}, error, context));
|
|
255
|
-
};
|
|
256
|
-
const getPostsByTags = (context, tags = [], options) => {
|
|
257
|
-
const action = "getPostsByTags";
|
|
258
|
-
const { database, fields, session: { userId: sessionId } } = context;
|
|
259
|
-
const { latitude, limit, longitude, type } = parsePostOptions(options);
|
|
260
|
-
const { objects: selectObjects, queries: selectQueries } = getPostOptional(fields, sessionId);
|
|
261
|
-
const formatTagNames = JSON.stringify(tags.map((tag) => (0, import_utils.parseChar)(tag, 32).toLowerCase()));
|
|
262
|
-
const sortBy = [];
|
|
263
|
-
const filters = [`p.type == "${type}"`, 'p.privacy == "public"'];
|
|
264
|
-
const formatLatitude = (0, import_utils.parseNum)(latitude);
|
|
265
|
-
const formatLongitude = (0, import_utils.parseNum)(longitude);
|
|
266
|
-
if (formatLatitude && formatLongitude) {
|
|
267
|
-
selectQueries.push(`LET distance = DISTANCE(
|
|
268
|
-
${formatLatitude},
|
|
269
|
-
${formatLongitude},
|
|
73
|
+
${d.join(`
|
|
74
|
+
`)}
|
|
75
|
+
FILTER ${h.join(" && ")}
|
|
76
|
+
${c.aql}
|
|
77
|
+
RETURN DISTINCT MERGE(p, {${u.join(", ")}})`;return y(i).query(l).then(P=>P.all()).catch(P=>(f({action:o,category:O,label:N.DATABASE_ERROR},P,e),[]))},nt=(e,r=[],t)=>{const o="getPostsByTags",{databaseName:i,fields:n,session:{userId:p}}=e,{latitude:R,limit:c,longitude:s,type:a}=$(t),{objects:u,queries:d}=C(n,p),I=JSON.stringify(r.map(l=>_(l,32).toLowerCase())),E=[],T=[`p.type == "${a}"`,'p.privacy == "public"'],h=L(R),m=L(s);h&&m&&(d.push(`LET distance = DISTANCE(
|
|
78
|
+
${h},
|
|
79
|
+
${m},
|
|
270
80
|
NOT_NULL(p.latitude, 0),
|
|
271
81
|
NOT_NULL(p.longitude, 0))
|
|
272
|
-
`)
|
|
273
|
-
selectObjects.push("distance:distance");
|
|
274
|
-
sortBy.push("distance");
|
|
275
|
-
}
|
|
276
|
-
if (tags.length) {
|
|
277
|
-
sortBy.push("matchedTags DESC");
|
|
278
|
-
selectQueries.push(`LET matchedTags = LENGTH(
|
|
82
|
+
`),u.push("distance:distance"),E.push("distance")),r.length&&(E.push("matchedTags DESC"),d.push(`LET matchedTags = LENGTH(
|
|
279
83
|
FOR t IN tags
|
|
280
84
|
FILTER t.matched == true
|
|
281
85
|
RETURN t
|
|
282
|
-
)`);
|
|
283
|
-
selectObjects.push("matchedTags:matchedTags");
|
|
284
|
-
filters.push("matchedTags > 0");
|
|
285
|
-
}
|
|
286
|
-
sortBy.push("p.added DESC");
|
|
287
|
-
selectObjects.push("tags:tags");
|
|
288
|
-
const aqlQry = `FOR p IN posts
|
|
86
|
+
)`),u.push("matchedTags:matchedTags"),T.push("matchedTags > 0")),E.push("p.added DESC"),u.push("tags:tags");const g=`FOR p IN posts
|
|
289
87
|
LET tags = (
|
|
290
88
|
FOR tag, it IN 1..1 INBOUND p isTagged
|
|
291
|
-
LET matched = LENGTH(${
|
|
89
|
+
LET matched = LENGTH(${I}) > 0 && POSITION(${I}, tag.name)
|
|
292
90
|
SORT tag.name
|
|
293
91
|
RETURN MERGE(tag, {matched:matched})
|
|
294
92
|
)
|
|
295
|
-
${
|
|
296
|
-
|
|
297
|
-
${
|
|
298
|
-
|
|
299
|
-
|
|
300
|
-
|
|
301
|
-
|
|
302
|
-
|
|
303
|
-
|
|
304
|
-
|
|
305
|
-
};
|
|
306
|
-
const getPostsByUser = (context, userId, options) => {
|
|
307
|
-
const action = "getPostsByUser";
|
|
308
|
-
const { database, fields, session: { userId: sessionId } } = context;
|
|
309
|
-
const { limit, type } = parsePostOptions(options);
|
|
310
|
-
const formatUserId = (0, import_utils.parseId)(userId);
|
|
311
|
-
const { objects: selectObjects, queries: selectQueries } = getPostOptional(fields, sessionId);
|
|
312
|
-
const aqlQry = `FOR p IN posts
|
|
313
|
-
FILTER p.userId == "${formatUserId}" && p.type == "${type}" && p.privacy == "public" && p.parent == null
|
|
314
|
-
${selectQueries.join("\n")}
|
|
315
|
-
${limit.aql}
|
|
93
|
+
${d.join(`
|
|
94
|
+
`)}
|
|
95
|
+
FILTER ${T.join(" && ")}
|
|
96
|
+
${c.aql}
|
|
97
|
+
SORT ${E.join(", ")}
|
|
98
|
+
RETURN DISTINCT MERGE(p, {${u.join(", ")}})`;return y(i).query(g).then(l=>l.all()).catch(l=>(f({action:o,category:O,label:N.DATABASE_ERROR},l,e),[]))},at=(e,r,t)=>{const o="getPostsByUser",{databaseName:i,fields:n,session:{userId:p}}=e,{limit:R,type:c}=$(t),s=D(r),{objects:a,queries:u}=C(n,p),d=`FOR p IN posts
|
|
99
|
+
FILTER p.userId == "${s}" && p.type == "${c}" && p.privacy == "public" && p.parent == null
|
|
100
|
+
${u.join(`
|
|
101
|
+
`)}
|
|
102
|
+
${R.aql}
|
|
316
103
|
SORT p.added
|
|
317
|
-
RETURN DISTINCT MERGE(p, {${
|
|
318
|
-
|
|
319
|
-
action,
|
|
320
|
-
category: eventCategory,
|
|
321
|
-
label: import_error.ErrorTypes.DATABASE_ERROR
|
|
322
|
-
}, error, context));
|
|
323
|
-
};
|
|
324
|
-
const getPostComments = (context, postId, options) => {
|
|
325
|
-
const action = "getPostComments";
|
|
326
|
-
const { database, session: { userId: sessionId } } = context;
|
|
327
|
-
const { limit, type } = parsePostOptions(options);
|
|
328
|
-
const formatItemId = (0, import_utils.parseId)(postId);
|
|
329
|
-
const aqlQry = import_arangojs.aql`FOR p IN posts
|
|
330
|
-
FILTER p.type == ${type} && p._key == ${formatItemId}
|
|
104
|
+
RETURN DISTINCT MERGE(p, {${a.join(", ")}})`;return y(i).query(d).then(I=>I.all()).catch(I=>(f({action:o,category:O,label:N.DATABASE_ERROR},I,e),[]))},it=(e,r,t)=>{const o="getPostComments",{databaseName:i,session:{userId:n}}=e,{limit:p,type:R}=$(t),c=D(r),s=A`FOR p IN posts
|
|
105
|
+
FILTER p.type == ${R} && p._key == ${c}
|
|
331
106
|
LIMIT 1
|
|
332
|
-
RETURN p`;
|
|
333
|
-
return database.query(aqlQry).then((cursor) => cursor.next()).then((post) => {
|
|
334
|
-
const {
|
|
335
|
-
_key,
|
|
336
|
-
groupId,
|
|
337
|
-
privacy = "default"
|
|
338
|
-
} = post;
|
|
339
|
-
let privacyAqlQry;
|
|
340
|
-
if (groupId && privacy === "group") {
|
|
341
|
-
privacyAqlQry = `FOR p IN posts
|
|
107
|
+
RETURN p`;return y(i).query(s).then(a=>a.next()).then(a=>{const{_key:u,groupId:d,privacy:I="default"}=a;let E;return d&&I==="group"?E=`FOR p IN posts
|
|
342
108
|
FOR user IN users
|
|
343
|
-
FILTER p.parent == "${
|
|
109
|
+
FILTER p.parent == "${u}" && user._key == p.userId
|
|
344
110
|
LET reactions = (
|
|
345
111
|
FOR post, r IN INBOUND p._id reactions
|
|
346
112
|
COLLECT reactionName = r.value INTO reactionItems
|
|
@@ -348,234 +114,30 @@ const getPostComments = (context, postId, options) => {
|
|
|
348
114
|
)
|
|
349
115
|
FOR group IN groups
|
|
350
116
|
FILTER group._key == p.groupId
|
|
351
|
-
FOR u, e IN OUTBOUND group._id
|
|
352
|
-
FILTER u._key == "${
|
|
117
|
+
FOR u, e IN OUTBOUND group._id hasConnection
|
|
118
|
+
FILTER u._key == "${n}"
|
|
353
119
|
SORT p.added
|
|
354
|
-
${
|
|
355
|
-
RETURN MERGE(p, {user: user, reactions: reactions})
|
|
356
|
-
} else if (privacy === "public") {
|
|
357
|
-
privacyAqlQry = `FOR p IN posts
|
|
120
|
+
${p.aql}
|
|
121
|
+
RETURN MERGE(p, {user: user, reactions: reactions})`:I==="public"&&(E=`FOR p IN posts
|
|
358
122
|
FOR user IN users
|
|
359
|
-
FILTER p.parent == "${
|
|
123
|
+
FILTER p.parent == "${u}" && user._key == p.userId
|
|
360
124
|
LET reactions = (
|
|
361
125
|
FOR post, r IN INBOUND p._id reactions
|
|
362
126
|
COLLECT reactionName = r.value INTO reactionItems
|
|
363
127
|
RETURN {name: reactionName, count: LENGTH(reactionItems[*].r.value)}
|
|
364
128
|
)
|
|
365
129
|
SORT p.added
|
|
366
|
-
${
|
|
367
|
-
RETURN MERGE(p, {user: user, reactions: reactions})`;
|
|
368
|
-
}
|
|
369
|
-
|
|
370
|
-
|
|
371
|
-
|
|
372
|
-
|
|
373
|
-
|
|
374
|
-
|
|
375
|
-
|
|
376
|
-
return [];
|
|
377
|
-
}).catch((error) => (0, import_analyticsUtils.logError)({
|
|
378
|
-
action,
|
|
379
|
-
category: eventCategory,
|
|
380
|
-
label: import_error.ErrorTypes.DATABASE_ERROR
|
|
381
|
-
}, error, context));
|
|
382
|
-
};
|
|
383
|
-
const addPost = async (context, {
|
|
384
|
-
content = "",
|
|
385
|
-
endDate,
|
|
386
|
-
groupId = "",
|
|
387
|
-
location,
|
|
388
|
-
latitude,
|
|
389
|
-
longitude,
|
|
390
|
-
name = "",
|
|
391
|
-
parentId = null,
|
|
392
|
-
privacy = "public",
|
|
393
|
-
tags = [],
|
|
394
|
-
startDate,
|
|
395
|
-
type = "default"
|
|
396
|
-
}) => {
|
|
397
|
-
const action = "addPost";
|
|
398
|
-
const { database, session: { userId: sessionId } } = context;
|
|
399
|
-
const now = Date.now();
|
|
400
|
-
const postId = (0, import_utils.createHash)(`post-${sessionId}`);
|
|
401
|
-
const insert = {
|
|
402
|
-
_id: `posts/${postId}`,
|
|
403
|
-
_key: postId,
|
|
404
|
-
added: now,
|
|
405
|
-
content: (0, import_utils.parseString)(content, MAX_CONTENT_LENGTH),
|
|
406
|
-
endDate: endDate ? (0, import_utils.parseNum)(endDate, 13) : void 0,
|
|
407
|
-
groupId: groupId ? (0, import_utils.parseId)(groupId) : void 0,
|
|
408
|
-
latitude: latitude !== void 0 ? (0, import_utils.parseNum)(latitude) : void 0,
|
|
409
|
-
location: location ? (0, import_utils.parseString)(location, 160) : void 0,
|
|
410
|
-
longitude: longitude !== void 0 ? (0, import_utils.parseNum)(longitude) : void 0,
|
|
411
|
-
modified: now,
|
|
412
|
-
name: (0, import_utils.parseString)(name, 160),
|
|
413
|
-
parentId: parentId ? (0, import_utils.parseId)(parentId) : void 0,
|
|
414
|
-
privacy: (0, import_utils.parseVarChar)(privacy, 16),
|
|
415
|
-
startDate: startDate ? (0, import_utils.parseNum)(startDate, 13) : void 0,
|
|
416
|
-
type: (0, import_utils.parseChar)(type, 32),
|
|
417
|
-
userId: sessionId
|
|
418
|
-
};
|
|
419
|
-
const aqlQry = import_arangojs.aql`INSERT ${insert} IN posts RETURN NEW`;
|
|
420
|
-
try {
|
|
421
|
-
const savedPost = await database.query(aqlQry).then((cursor) => cursor.next()).catch((error) => (0, import_analyticsUtils.logError)({
|
|
422
|
-
action,
|
|
423
|
-
category: eventCategory,
|
|
424
|
-
label: import_error.ErrorTypes.DATABASE_ERROR
|
|
425
|
-
}, error, context));
|
|
426
|
-
const { _id: postDocId } = savedPost;
|
|
427
|
-
const contentTagNames = await (0, import_tags.extractTags)(insert.content);
|
|
428
|
-
if (tags.length || contentTagNames.length) {
|
|
429
|
-
const userTags = (await (0, import_tags.getTagsByName)(context, tags.map(({ name: name2 }) => name2))).map((tag) => ({ ...tag, tagBy: sessionId }));
|
|
430
|
-
const contentTags = (await (0, import_tags.getTagsByName)(context, contentTagNames)).map((tag) => ({ ...tag, tagBy: "extract" }));
|
|
431
|
-
const updatedTags = await (0, import_tags.updateTagsInItem)(
|
|
432
|
-
context,
|
|
433
|
-
{
|
|
434
|
-
itemDocId: postDocId,
|
|
435
|
-
tags: [...contentTags, ...userTags]
|
|
436
|
-
}
|
|
437
|
-
);
|
|
438
|
-
return {
|
|
439
|
-
...savedPost,
|
|
440
|
-
tags: updatedTags
|
|
441
|
-
};
|
|
442
|
-
}
|
|
443
|
-
return savedPost;
|
|
444
|
-
} catch (error) {
|
|
445
|
-
throw error;
|
|
446
|
-
}
|
|
447
|
-
};
|
|
448
|
-
const updatePost = async (context, post) => {
|
|
449
|
-
const action = "updatePost";
|
|
450
|
-
const { database, session: { userId: sessionId } } = context;
|
|
451
|
-
const now = Date.now();
|
|
452
|
-
const parsedPost = (0, import_postAdapter.parsePost)(post);
|
|
453
|
-
const {
|
|
454
|
-
postId,
|
|
455
|
-
tags = []
|
|
456
|
-
} = parsedPost;
|
|
457
|
-
const update = {
|
|
458
|
-
...parsedPost,
|
|
459
|
-
modified: now
|
|
460
|
-
};
|
|
461
|
-
if (!postId) {
|
|
462
|
-
return (0, import_analyticsUtils.logException)({
|
|
463
|
-
action,
|
|
464
|
-
category: eventCategory,
|
|
465
|
-
value: import_error.ErrorTypes.INVALID_ID
|
|
466
|
-
}, {});
|
|
467
|
-
}
|
|
468
|
-
const insert = {
|
|
469
|
-
...update,
|
|
470
|
-
_key: postId,
|
|
471
|
-
added: now,
|
|
472
|
-
userId: sessionId
|
|
473
|
-
};
|
|
474
|
-
const aqlQry = import_arangojs.aql`UPSERT {_key: ${postId}, userId: ${sessionId}}
|
|
475
|
-
INSERT ${insert}
|
|
476
|
-
UPDATE ${update}
|
|
477
|
-
IN posts RETURN NEW`;
|
|
478
|
-
try {
|
|
479
|
-
const updatedPost = await database.query(aqlQry).then((cursor) => cursor.next()).catch((error) => (0, import_analyticsUtils.logError)({
|
|
480
|
-
action,
|
|
481
|
-
category: eventCategory,
|
|
482
|
-
value: import_error.ErrorTypes.DATABASE_ERROR
|
|
483
|
-
}, error, {}));
|
|
484
|
-
const { _id: updatedPostId } = updatedPost;
|
|
485
|
-
const contentTagNames = await (0, import_tags.extractTags)(insert.content);
|
|
486
|
-
if (tags?.length || contentTagNames?.length) {
|
|
487
|
-
const userTags = tags?.length ? (await (0, import_tags.getTagsByName)(context, tags.map(({ name }) => name))).map((tag) => ({ ...tag, tagBy: sessionId })) : [];
|
|
488
|
-
const contentTags = contentTagNames?.length ? (await (0, import_tags.getTagsByName)(context, contentTagNames)).map((tag) => ({ ...tag, tagBy: "extract" })) : [];
|
|
489
|
-
const updatedTags = await (0, import_tags.updateTagsInItem)(
|
|
490
|
-
context,
|
|
491
|
-
{
|
|
492
|
-
itemDocId: updatedPostId,
|
|
493
|
-
tags: [...contentTags, ...userTags]
|
|
494
|
-
}
|
|
495
|
-
);
|
|
496
|
-
return {
|
|
497
|
-
...updatedPost,
|
|
498
|
-
tags: updatedTags
|
|
499
|
-
};
|
|
500
|
-
}
|
|
501
|
-
return updatedPost;
|
|
502
|
-
} catch (error) {
|
|
503
|
-
throw error;
|
|
504
|
-
}
|
|
505
|
-
};
|
|
506
|
-
const deletePost = async (context, postDocId) => {
|
|
507
|
-
const action = "deletePost";
|
|
508
|
-
const { database, session: { userId: sessionId } } = context;
|
|
509
|
-
const formatPostId = (0, import_utils.parseArangoId)(postDocId);
|
|
510
|
-
if (!formatPostId) {
|
|
511
|
-
return (0, import_analyticsUtils.logException)({
|
|
512
|
-
action,
|
|
513
|
-
category: eventCategory,
|
|
514
|
-
value: import_error.ErrorTypes.INVALID_ID
|
|
515
|
-
}, {});
|
|
516
|
-
}
|
|
517
|
-
const edgeAqlQry = import_arangojs.aql`FOR t IN isTagged
|
|
518
|
-
FILTER t._to == ${formatPostId}
|
|
519
|
-
REMOVE t IN isTagged`;
|
|
520
|
-
await database.query(edgeAqlQry).catch((error) => {
|
|
521
|
-
throw error;
|
|
522
|
-
});
|
|
523
|
-
const fileAqlQry = import_arangojs.aql`FOR f IN hasFile
|
|
524
|
-
FILTER f._to == ${formatPostId}
|
|
525
|
-
REMOVE f IN hasFile`;
|
|
526
|
-
await database.query(fileAqlQry).catch((error) => {
|
|
527
|
-
throw error;
|
|
528
|
-
});
|
|
529
|
-
const aqlQry = import_arangojs.aql`FOR p IN posts
|
|
530
|
-
FILTER p._id == ${formatPostId} && p.userId == ${sessionId}
|
|
130
|
+
${p.aql}
|
|
131
|
+
RETURN MERGE(p, {user: user, reactions: reactions})`),E?y(i).query(E).then(T=>T.all()).catch(T=>(f({action:o,category:O,label:N.DATABASE_ERROR},T,e),[])):[]}).catch(a=>(f({action:o,category:O,label:N.DATABASE_ERROR},a,e),[]))},pt=async(e,{content:r="",endDate:t,groupId:o="",location:i,latitude:n,longitude:p,name:R="",parentId:c=null,privacy:s="public",tags:a=[],startDate:u,type:d="default"})=>{const I="addPost",{databaseName:E,session:{userId:T}}=e,h=Date.now(),m=B(`post-${T}`),g={_id:`posts/${m}`,_key:m,added:h,content:q(r,W),endDate:t?L(t,13):void 0,groupId:o?D(o):void 0,latitude:n!==void 0?L(n):void 0,location:i?q(i,160):void 0,longitude:p!==void 0?L(p):void 0,modified:h,name:q(R,160),parentId:c?D(c):void 0,privacy:k(s,16),startDate:u?L(u,13):void 0,type:_(d,32),userId:T},l=A`INSERT ${g} IN posts RETURN NEW`;try{const P=await y(E).query(l).then(b=>b.next()).catch(b=>f({action:I,category:O,label:N.DATABASE_ERROR},b,e)),{_id:M}=P,v=await w(g.content);if(a.length||v.length){const b=(await U(e,a.map(({name:F})=>F))).map(F=>({...F,tagBy:T})),x=(await U(e,v)).map(F=>({...F,tagBy:"extract"})),G=await j(e,{itemDocId:M,tags:[...x,...b]});return{...P,tags:G}}return P}catch(P){throw P}},ct=async(e,r)=>{const t="updatePost",{databaseName:o,session:{userId:i}}=e,n=Date.now(),p=H(r),{postId:R,tags:c=[]}=p,s={...p,modified:n};if(!R)return Q({action:t,category:O,value:N.INVALID_ID},{});const a={...s,_key:R,added:n,userId:i},u=A`UPSERT {_key: ${R}, userId: ${i}}
|
|
132
|
+
INSERT ${a}
|
|
133
|
+
UPDATE ${s}
|
|
134
|
+
IN posts RETURN NEW`;try{const d=await y(o).query(u).then(T=>T.next()).catch(T=>f({action:t,category:O,value:N.DATABASE_ERROR},T,{})),{_id:I}=d,E=await w(a.content);if(c?.length||E?.length){const T=c?.length?(await U(e,c.map(({name:g})=>g))).map(g=>({...g,tagBy:i})):[],h=E?.length?(await U(e,E)).map(g=>({...g,tagBy:"extract"})):[],m=await j(e,{itemDocId:I,tags:[...h,...T]});return{...d,tags:m}}return d}catch(d){throw d}},ut=async(e,r)=>{const t="deletePost",{databaseName:o,session:{userId:i}}=e,n=S(r);if(!n)return Q({action:t,category:O,value:N.INVALID_ID},{});const p=A`FOR t IN isTagged
|
|
135
|
+
FILTER t._to == ${n}
|
|
136
|
+
REMOVE t IN isTagged`;await y(o).query(p).catch(s=>{throw s});const R=A`FOR f IN hasFile
|
|
137
|
+
FILTER f._to == ${n}
|
|
138
|
+
REMOVE f IN hasFile`;await y(o).query(R).catch(s=>{throw s});const c=A`FOR p IN posts
|
|
139
|
+
FILTER p._id == ${n} && p.userId == ${i}
|
|
531
140
|
LIMIT 1
|
|
532
141
|
REMOVE p IN posts
|
|
533
|
-
RETURN OLD`;
|
|
534
|
-
return database.query(aqlQry).then((cursor) => cursor.next()).catch((error) => {
|
|
535
|
-
throw error;
|
|
536
|
-
});
|
|
537
|
-
};
|
|
538
|
-
const createPostEdge = (context, postDocId, itemDocId, edgeType = "isPosted", props = {}) => {
|
|
539
|
-
const action = "createPostEdge";
|
|
540
|
-
const { database } = context;
|
|
541
|
-
const edgeCollection = database.collection(edgeType);
|
|
542
|
-
const formatPostId = (0, import_utils.parseArangoId)(postDocId);
|
|
543
|
-
const formatDocId = (0, import_utils.parseArangoId)(itemDocId);
|
|
544
|
-
if (!formatDocId || !formatPostId) {
|
|
545
|
-
return (0, import_analyticsUtils.logException)({
|
|
546
|
-
action,
|
|
547
|
-
category: eventCategory,
|
|
548
|
-
value: import_error.ErrorTypes.INVALID_ID
|
|
549
|
-
}, {});
|
|
550
|
-
}
|
|
551
|
-
const edgeId = (0, import_utils.createHash)(`postEdge-${formatPostId}-${formatDocId}`);
|
|
552
|
-
const edge = {
|
|
553
|
-
_from: formatPostId,
|
|
554
|
-
_key: edgeId,
|
|
555
|
-
_to: formatDocId,
|
|
556
|
-
added: Date.now(),
|
|
557
|
-
...props
|
|
558
|
-
};
|
|
559
|
-
return edgeCollection.save(edge, { returnNew: true }).catch((error) => (0, import_analyticsUtils.logError)({
|
|
560
|
-
action,
|
|
561
|
-
category: eventCategory,
|
|
562
|
-
value: import_error.ErrorTypes.DATABASE_ERROR
|
|
563
|
-
}, error, {}));
|
|
564
|
-
};
|
|
565
|
-
// Annotate the CommonJS export names for ESM import in node:
|
|
566
|
-
0 && (module.exports = {
|
|
567
|
-
addPost,
|
|
568
|
-
createPostEdge,
|
|
569
|
-
deletePost,
|
|
570
|
-
getPost,
|
|
571
|
-
getPostComments,
|
|
572
|
-
getPostOptional,
|
|
573
|
-
getPostsByArea,
|
|
574
|
-
getPostsByLatest,
|
|
575
|
-
getPostsByReactions,
|
|
576
|
-
getPostsByTags,
|
|
577
|
-
getPostsByUser,
|
|
578
|
-
parsePostOptions,
|
|
579
|
-
updatePost
|
|
580
|
-
});
|
|
581
|
-
//# 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 {aql} from 'arangojs';\nimport {AqlQuery} from 'arangojs/aql';\nimport {EdgeCollection} from 'arangojs/collection';\nimport {ArrayCursor} from 'arangojs/cursor';\nimport {\n  createHash, parseArangoId, parseChar, parseId, parseNum, parseString, parseVarChar\n} from '@nlabs/utils';\n\nimport {parsePost} from '../adapters/postAdapter';\nimport {ErrorTypes} from '../types/error';\nimport {logError, logException} from '../utils/analyticsUtils';\nimport {getLimit} from '../utils/arangodbUtils';\nimport {extractTags, getTagsByName, updateTagsInItem} from './tags';\nimport type {ApiContext, ArangoDbLimit, FileType, PostInputType, PostOptions, PostType, TagType} from '../types';\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) as ArangoDbLimit,\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 hasReaction\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 hasReaction\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 hasReaction\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 hasReaction\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 hasReaction\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 = async (\n  context: ApiContext,\n  postId: string,\n  options?: PostOptions\n): Promise<PostType> => {\n  const action: string = 'getPost';\n  const {database, fields, session: {userId: sessionId}} = context;\n  const formatItemId: string = parseId(postId);\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        userId,\n        groupId,\n        privacy = 'default'\n      }: PostType = post;\n\n      let privacyAqlQry: string;\n\n      if(userId === sessionId) {\n        return post;\n      }\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: ErrorTypes.DATABASE_ERROR\n          }, error, context));\n      }\n\n      return {};\n    })\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: ErrorTypes.DATABASE_ERROR\n    }, error, context));\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//   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//       throw error;\n//     });\n// };\n\nexport const getPostsByArea = (\n  context: ApiContext,\n  latitude: number,\n  longitude: number,\n  options?: PostOptions\n): Promise<PostType[]> => {\n  const action: string = 'getPostsByArea';\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) => logError({\n      action,\n      category: eventCategory,\n      label: ErrorTypes.DATABASE_ERROR\n    }, error, context));\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\nexport const getPostsByLatest = (context: ApiContext, options?: PostOptions): Promise<PostType[]> => {\n  // const action: string = 'getListByLatest';\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  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}\" hasReaction\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: ErrorTypes.DATABASE_ERROR\n    }, error, context));\n};\n\nexport const getPostsByTags = (\n  context: ApiContext,\n  tags: string[] = [],\n  options?: PostOptions\n): Promise<PostType[]> => {\n  const action: string = 'getPostsByTags';\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  return database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: ErrorTypes.DATABASE_ERROR\n    }, error, context));\n};\n\nexport const getPostsByUser = (context: ApiContext, userId: string, options?: PostOptions): Promise<PostType[]> => {\n  const action: string = 'getPostsByUser';\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) => logError({\n      action,\n      category: eventCategory,\n      label: ErrorTypes.DATABASE_ERROR\n    }, error, context));\n};\n\nexport const getPostComments = (context: ApiContext, postId: string, options?: PostOptions): Promise<PostType[]> => {\n  const action: string = 'getPostComments';\n  const {database, session: {userId: sessionId}} = context;\n  const {limit, type} = parsePostOptions(options);\n  const formatItemId: string = parseId(postId);\n\n  // Get the parent post to get restrictions\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 database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((post: PostType) => {\n      const {\n        _key,\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 = `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 database.query(privacyAqlQry)\n          .then((cursor: ArrayCursor) => cursor.all())\n          .catch((error: Error) => logError({\n            action,\n            category: eventCategory,\n            label: ErrorTypes.DATABASE_ERROR\n          }, error, context));\n      }\n\n      return [];\n    })\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: ErrorTypes.DATABASE_ERROR\n    }, error, context));\n};\n\nexport const addPost = async (\n  context: ApiContext,\n  {\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  }: PostInputType\n): Promise<PostType> => {\n  const action: string = 'addPost';\n  const {database, session: {userId: sessionId}} = context;\n  const now = Date.now();\n  const postId = createHash(`post-${sessionId}`);\n  const insert: PostType = {\n    _id: `posts/${postId}`,\n    _key: postId,\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: parseVarChar(privacy, 16),\n    startDate: startDate ? parseNum(startDate, 13) : undefined,\n    type: parseChar(type, 32),\n    userId: sessionId\n  };\n  const aqlQry: AqlQuery = aql`INSERT ${insert} IN posts RETURN NEW`;\n\n  try {\n    const savedPost: PostType = await database.query(aqlQry)\n      .then((cursor: ArrayCursor) => cursor.next())\n      .catch((error: Error) => logError({\n        action,\n        category: eventCategory,\n        label: ErrorTypes.DATABASE_ERROR\n      }, error, context));\n    const {_id: postDocId} = savedPost;\n    const contentTagNames = await extractTags(insert.content);\n\n    if(tags.length || contentTagNames.length) {\n      const userTags = (await getTagsByName(context, tags.map(({name}) => name)))\n        .map((tag) => ({...tag, tagBy: sessionId}));\n      const contentTags = (await getTagsByName(context, contentTagNames))\n        .map((tag) => ({...tag, tagBy: 'extract'}));\n      const updatedTags: TagType[] = await updateTagsInItem(\n        context,\n        {\n          itemDocId: postDocId,\n          tags: [...contentTags, ...userTags]\n        }\n      );\n\n      return {\n        ...savedPost,\n        tags: updatedTags\n      };\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 = 'updatePost';\n  const {database, session: {userId: sessionId}} = context;\n  const now: number = Date.now();\n  const parsedPost = parsePost(post);\n  const {\n    postId,\n    tags = []\n  } = parsedPost;\n\n  const update: PostType = {\n    ...parsedPost,\n    modified: now\n  };\n\n  if(!postId) {\n    return logException({\n      action,\n      category: eventCategory,\n      value: ErrorTypes.INVALID_ID\n    }, {});\n  }\n\n  const insert: any = {\n    ...update,\n    _key: postId,\n    added: now,\n    userId: sessionId\n  };\n  const aqlQry: AqlQuery = aql`UPSERT {_key: ${postId}, userId: ${sessionId}}\n    INSERT ${insert}\n    UPDATE ${update}\n    IN posts RETURN NEW`;\n\n  try {\n    const updatedPost: PostType = await database\n      .query(aqlQry)\n      .then((cursor: ArrayCursor) => cursor.next())\n      .catch((error: Error) => logError({\n        action,\n        category: eventCategory,\n        value: ErrorTypes.DATABASE_ERROR\n      }, error, {}));\n    const {_id: updatedPostId} = updatedPost;\n    const contentTagNames = await extractTags(insert.content);\n\n    if(tags?.length || contentTagNames?.length) {\n      const userTags = tags?.length ? (await getTagsByName(context, tags.map(({name}) => name)))\n        .map((tag) => ({...tag, tagBy: sessionId})) : [];\n      const contentTags = contentTagNames?.length ? (await getTagsByName(context, contentTagNames))\n        .map((tag) => ({...tag, tagBy: 'extract'})) : [];\n      const updatedTags: TagType[] = await updateTagsInItem(\n        context,\n        {\n          itemDocId: updatedPostId,\n          tags: [...contentTags, ...userTags]\n        }\n      );\n\n      return {\n        ...updatedPost,\n        tags: updatedTags\n      };\n    }\n\n    return updatedPost;\n  } catch(error) {\n    throw error;\n  }\n};\n\nexport const deletePost = async (context: ApiContext, postDocId: string): Promise<PostType> => {\n  const action: string = 'deletePost';\n  const {database, session: {userId: sessionId}} = context;\n  const formatPostId: string = parseArangoId(postDocId);\n\n  if(!formatPostId) {\n    return logException({\n      action,\n      category: eventCategory,\n      value: ErrorTypes.INVALID_ID\n    }, {});\n  }\n\n  const edgeAqlQry: AqlQuery = aql`FOR t IN isTagged\n  FILTER t._to == ${formatPostId}\n  REMOVE t IN isTagged`;\n\n  await database.query(edgeAqlQry)\n    .catch((error: Error) => {\n      throw error;\n    });\n\n  const fileAqlQry: AqlQuery = aql`FOR f IN hasFile\n    FILTER f._to == ${formatPostId}\n    REMOVE f IN hasFile`;\n\n  await database.query(fileAqlQry)\n    .catch((error: Error) => {\n      throw error;\n    });\n\n  const aqlQry = aql`FOR p IN posts\n      FILTER p._id == ${formatPostId} && p.userId == ${sessionId}\n      LIMIT 1\n      REMOVE p IN posts\n      RETURN OLD`;\n\n  return database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .catch((error: Error) => {\n      throw error;\n    });\n};\n\nexport const createPostEdge = (\n  context: ApiContext,\n  postDocId: string,\n  itemDocId: string,\n  edgeType: string = 'isPosted',\n  props: any = {}\n): Promise<FileType> => {\n  const action = 'createPostEdge';\n  const {database} = context;\n  const edgeCollection: EdgeCollection = database.collection(edgeType);\n  const formatPostId: string = parseArangoId(postDocId);\n  const formatDocId: string = parseArangoId(itemDocId);\n\n  if(!formatDocId || !formatPostId) {\n    return logException({\n      action,\n      category: eventCategory,\n      value: ErrorTypes.INVALID_ID\n    }, {});\n  }\n\n  const edgeId: string = createHash(`postEdge-${formatPostId}-${formatDocId}`);\n  const edge: any = {\n    _from: formatPostId,\n    _key: edgeId,\n    _to: formatDocId,\n    added: Date.now(),\n    ...props\n  };\n\n  return edgeCollection.save(edge, {returnNew: true})\n    .catch((error: Error) =>\n      logError({\n        action,\n        category: eventCategory,\n        value: ErrorTypes.DATABASE_ERROR\n      }, error, {}));\n};\n"],
  "mappings": "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",
  "names": ["name"]
}

|
|
142
|
+
RETURN OLD`;return y(o).query(c).then(s=>s.next()).catch(s=>{throw s})},dt=(e,r,t,o="isPosted",i={})=>{const n="createPostEdge",{databaseName:p}=e,R=y(p).collection(o),c=S(r),s=S(t);if(!s||!c)return Promise.reject(new Error(N.INVALID_ID));const a=B(`postEdge-${c}-${s}`),u={_from:c,_key:a,_to:s,added:Date.now(),...i};return R.save(u,{returnNew:!0}).catch(d=>f({action:n,category:O,value:N.DATABASE_ERROR},d,{}))};export{pt as addPost,dt as createPostEdge,ut as deletePost,et as getPost,it as getPostComments,C as getPostOptional,st as getPostsByArea,ot as getPostsByLatest,rt as getPostsByReactions,nt as getPostsByTags,at as getPostsByUser,$ as parsePostOptions,ct as updatePost};
|
|
143
|
+
//# 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 {\n  createHash, parseArangoId, parseChar, parseId, parseNum, parseString, parseVarChar\n} from '@nlabs/utils';\nimport {aql} from 'arangojs';\nimport {AqlQuery} from 'arangojs/aql';\n\nimport {extractTags, getTagsByName, updateTagsInItem} from './tags';\nimport {parsePost} from '../adapters/postAdapter';\nimport {ErrorTypes} from '../types/error.types';\nimport {logError, logException} from '../utils/analyticsUtils';\nimport {getLimit, useDb} from '../utils/arangodbUtils';\n\nimport type {ApiContext, ArangoDbLimit, FileType, PostInputType, PostOptions, PostType, TagType} from '../types';\nimport type {EdgeCollection} from 'arangojs/collections';\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) as ArangoDbLimit,\n    longitude: parseNum(longitude, 32),\n    type: parseChar(type, 32)\n  };\n};\n\nexport const getPostOptional = (fields: string[], sessionId: string) =>\n  (fields || []).reduce((selects, 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 hasReaction\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 hasReaction\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 hasReaction\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 hasReaction\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 hasReaction\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 = async (\n  context: ApiContext,\n  postId: string,\n  options?: PostOptions\n): Promise<PostType> => {\n  const action: string = 'getPost';\n  const {databaseName, fields, session: {userId: sessionId}} = context;\n  const formatItemId: string = parseId(postId);\n  const {type} = parsePostOptions(options);\n  const database = useDb(databaseName);\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 database.query(aqlQry)\n    .then((cursor) => cursor.next())\n    .then((post: PostType) => {\n      const {\n        _id: postDocId,\n        userId,\n        groupId,\n        privacy = 'default'\n      }: PostType = post;\n\n      let privacyAqlQry: string;\n\n      if(userId === sessionId) {\n        return post;\n      }\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 hasConnection\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 database.query(privacyAqlQry)\n          .then((cursor) => cursor.next())\n          .catch((error: Error) => logError({\n            action,\n            category: eventCategory,\n            label: ErrorTypes.DATABASE_ERROR\n          }, error, context));\n      }\n\n      return {};\n    })\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: ErrorTypes.DATABASE_ERROR\n    }, error, context));\n};\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//   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) => cursor.all())\n//     .catch((error: Error) => {\n//       throw error;\n//     });\n// };\n\nexport const getPostsByArea = (\n  context: ApiContext,\n  latitude: number,\n  longitude: number,\n  options?: PostOptions\n): Promise<PostType[]> => {\n  const action: string = 'getPostsByArea';\n  const {databaseName, 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 useDb(databaseName).query(aqlQry)\n    .then((cursor) => cursor.all() as unknown as PostType[])\n    .catch((error: Error) => {\n      logError({\n        action,\n        category: eventCategory,\n        label: ErrorTypes.DATABASE_ERROR\n      }, error, context);\n\n      return [] as PostType[];\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) => 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) => 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 getPostsByLatest = (context: ApiContext, options?: PostOptions): Promise<PostType[]> => {\n  // const action: string = 'getListByLatest';\n  const {databaseName, 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  return useDb(databaseName).query(aqlQry)\n    .then((cursor) => 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 {databaseName, 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}\" hasReaction\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 useDb(databaseName).query(aqlQry)\n    .then((cursor) => cursor.all() as unknown as PostType[])\n    .catch((error: Error) => {\n      logError({\n        action,\n        category: eventCategory,\n        label: ErrorTypes.DATABASE_ERROR\n      }, error, context);\n\n      return [] as PostType[];\n    });\n};\n\nexport const getPostsByTags = (\n  context: ApiContext,\n  tags: string[] = [],\n  options?: PostOptions\n): Promise<PostType[]> => {\n  const action: string = 'getPostsByTags';\n  const {databaseName, 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  return useDb(databaseName).query(aqlQry)\n    .then((cursor) => cursor.all() as unknown as PostType[])\n    .catch((error: Error) => {\n      logError({\n        action,\n        category: eventCategory,\n        label: ErrorTypes.DATABASE_ERROR\n      }, error, context);\n\n      return [] as PostType[];\n    });\n};\n\nexport const getPostsByUser = (context: ApiContext, userId: string, options?: PostOptions): Promise<PostType[]> => {\n  const action: string = 'getPostsByUser';\n  const {databaseName, 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 useDb(databaseName).query(aqlQry)\n    .then((cursor) => cursor.all() as unknown as PostType[])\n    .catch((error: Error) => {\n      logError({\n        action,\n        category: eventCategory,\n        label: ErrorTypes.DATABASE_ERROR\n      }, error, context);\n\n      return [] as PostType[];\n    });\n};\n\nexport const getPostComments = (context: ApiContext, postId: string, options?: PostOptions): Promise<PostType[]> => {\n  const action: string = 'getPostComments';\n  const {databaseName, session: {userId: sessionId}} = context;\n  const {limit, type} = parsePostOptions(options);\n  const formatItemId: string = parseId(postId);\n\n  // Get the parent post to get restrictions\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 useDb(databaseName).query(aqlQry)\n    .then((cursor) => cursor.next())\n    .then((post: PostType) => {\n      const {\n        _key,\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 = `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 hasConnection\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 useDb(databaseName).query(privacyAqlQry)\n          .then((cursor) => cursor.all() as unknown as PostType[])\n          .catch((error: Error) => {\n            logError({\n              action,\n              category: eventCategory,\n              label: ErrorTypes.DATABASE_ERROR\n            }, error, context);\n\n            return [] as PostType[];\n          });\n      }\n\n      return [] as PostType[];\n    })\n    .catch((error: Error) => {\n      logError({\n        action,\n        category: eventCategory,\n        label: ErrorTypes.DATABASE_ERROR\n      }, error, context);\n\n      return [] as PostType[];\n    });\n};\n\nexport const addPost = async (\n  context: ApiContext,\n  {\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  }: PostInputType\n): Promise<PostType> => {\n  const action: string = 'addPost';\n  const {databaseName, session: {userId: sessionId}} = context;\n  const now = Date.now();\n  const postId = createHash(`post-${sessionId}`);\n  const insert: PostType = {\n    _id: `posts/${postId}`,\n    _key: postId,\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: parseVarChar(privacy, 16),\n    startDate: startDate ? parseNum(startDate, 13) : undefined,\n    type: parseChar(type, 32),\n    userId: sessionId\n  };\n  const aqlQry: AqlQuery = aql`INSERT ${insert} IN posts RETURN NEW`;\n\n  try {\n    const savedPost: PostType = await useDb(databaseName).query(aqlQry)\n      .then((cursor) => cursor.next())\n      .catch((error: Error) => logError({\n        action,\n        category: eventCategory,\n        label: ErrorTypes.DATABASE_ERROR\n      }, error, context));\n    const {_id: postDocId} = savedPost;\n    const contentTagNames = await extractTags(insert.content);\n\n    if(tags.length || contentTagNames.length) {\n      const userTags = (await getTagsByName(context, tags.map(({name}) => name)))\n        .map((tag) => ({...tag, tagBy: sessionId}));\n      const contentTags = (await getTagsByName(context, contentTagNames))\n        .map((tag) => ({...tag, tagBy: 'extract'}));\n      const updatedTags: TagType[] = await updateTagsInItem(\n        context,\n        {\n          itemDocId: postDocId,\n          tags: [...contentTags, ...userTags]\n        }\n      );\n\n      return {\n        ...savedPost,\n        tags: updatedTags\n      };\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 = 'updatePost';\n  const {databaseName, session: {userId: sessionId}} = context;\n  const now: number = Date.now();\n  const parsedPost = parsePost(post);\n  const {\n    postId,\n    tags = []\n  } = parsedPost;\n\n  const update: PostType = {\n    ...parsedPost,\n    modified: now\n  };\n\n  if(!postId) {\n    return logException({\n      action,\n      category: eventCategory,\n      value: ErrorTypes.INVALID_ID\n    }, {});\n  }\n\n  const insert = {\n    ...update,\n    _key: postId,\n    added: now,\n    userId: sessionId\n  };\n  const aqlQry: AqlQuery = aql`UPSERT {_key: ${postId}, userId: ${sessionId}}\n    INSERT ${insert}\n    UPDATE ${update}\n    IN posts RETURN NEW`;\n\n  try {\n    const updatedPost: PostType = await useDb(databaseName)\n      .query(aqlQry)\n      .then((cursor) => cursor.next())\n      .catch((error: Error) => logError({\n        action,\n        category: eventCategory,\n        value: ErrorTypes.DATABASE_ERROR\n      }, error, {}));\n    const {_id: updatedPostId} = updatedPost;\n    const contentTagNames = await extractTags(insert.content);\n\n    if(tags?.length || contentTagNames?.length) {\n      const userTags = tags?.length ? (await getTagsByName(context, tags.map(({name}) => name)))\n        .map((tag) => ({...tag, tagBy: sessionId})) : [];\n      const contentTags = contentTagNames?.length ? (await getTagsByName(context, contentTagNames))\n        .map((tag) => ({...tag, tagBy: 'extract'})) : [];\n      const updatedTags: TagType[] = await updateTagsInItem(\n        context,\n        {\n          itemDocId: updatedPostId,\n          tags: [...contentTags, ...userTags]\n        }\n      );\n\n      return {\n        ...updatedPost,\n        tags: updatedTags\n      };\n    }\n\n    return updatedPost;\n  } catch(error) {\n    throw error;\n  }\n};\n\nexport const deletePost = async (context: ApiContext, postDocId: string): Promise<PostType> => {\n  const action: string = 'deletePost';\n  const {databaseName, session: {userId: sessionId}} = context;\n  const formatPostId: string = parseArangoId(postDocId);\n\n  if(!formatPostId) {\n    return logException({\n      action,\n      category: eventCategory,\n      value: ErrorTypes.INVALID_ID\n    }, {});\n  }\n\n  const edgeAqlQry: AqlQuery = aql`FOR t IN isTagged\n  FILTER t._to == ${formatPostId}\n  REMOVE t IN isTagged`;\n\n  await useDb(databaseName).query(edgeAqlQry)\n    .catch((error: Error) => {\n      throw error;\n    });\n\n  const fileAqlQry: AqlQuery = aql`FOR f IN hasFile\n    FILTER f._to == ${formatPostId}\n    REMOVE f IN hasFile`;\n\n  await useDb(databaseName).query(fileAqlQry)\n    .catch((error: Error) => {\n      throw error;\n    });\n\n  const aqlQry = aql`FOR p IN posts\n      FILTER p._id == ${formatPostId} && p.userId == ${sessionId}\n      LIMIT 1\n      REMOVE p IN posts\n      RETURN OLD`;\n\n  return useDb(databaseName).query(aqlQry)\n    .then((cursor) => cursor.next())\n    .catch((error: Error) => {\n      throw error;\n    });\n};\n\nexport const createPostEdge = (\n  context: ApiContext,\n  postDocId: string,\n  itemDocId: string,\n  edgeType: string = 'isPosted',\n  props = {}\n): Promise<FileType> => {\n  const action = 'createPostEdge';\n  const {databaseName} = context;\n  const edgeCollection: EdgeCollection = useDb(databaseName).collection(edgeType);\n  const formatPostId: string = parseArangoId(postDocId);\n  const formatDocId: string = parseArangoId(itemDocId);\n\n  if(!formatDocId || !formatPostId) {\n    return Promise.reject(new Error(ErrorTypes.INVALID_ID));\n  }\n\n  const edgeId: string = createHash(`postEdge-${formatPostId}-${formatDocId}`);\n  const edge = {\n    _from: formatPostId,\n    _key: edgeId,\n    _to: formatDocId,\n    added: Date.now(),\n    ...props\n  };\n\n  return edgeCollection.save(edge, {returnNew: true})\n    .catch((error: Error) =>\n      logError({\n        action,\n        category: eventCategory,\n        value: ErrorTypes.DATABASE_ERROR\n      }, error, {}));\n};\n"],
  "mappings": "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",
  "names": ["createHash", "parseArangoId", "parseChar", "parseId", "parseNum", "parseString", "parseVarChar", "aql", "extractTags", "getTagsByName", "updateTagsInItem", "parsePost", "ErrorTypes", "logError", "logException", "getLimit", "useDb", "MAX_CONTENT_LENGTH", "eventCategory", "parsePostOptions", "options", "from", "latitude", "longitude", "to", "type", "getPostOptional", "fields", "sessionId", "selects", "field", "getPost", "context", "postId", "action", "databaseName", "formatItemId", "database", "selectObjects", "selectQueries", "aqlQry", "cursor", "post", "postDocId", "userId", "groupId", "privacy", "privacyAqlQry", "error", "getPostsByArea", "limit", "formatLatitude", "formatLongitude", "getPostsByLatest", "getPostsByReactions", "reactions", "formatSessionId", "formatReactions", "reaction", "sortBy", "filters", "getPostsByTags", "tags", "formatTagNames", "tag", "getPostsByUser", "formatUserId", "getPostComments", "_key", "addPost", "content", "endDate", "location", "name", "parentId", "startDate", "now", "insert", "savedPost", "contentTagNames", "userTags", "contentTags", "updatedTags", "updatePost", "parsedPost", "update", "updatedPost", "updatedPostId", "deletePost", "formatPostId", "edgeAqlQry", "fileAqlQry", "createPostEdge", "itemDocId", "edgeType", "props", "edgeCollection", "formatDocId", "edgeId", "edge"]
}

|