@nlabs/reaktor 0.1.2 → 0.1.3
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/.DS_Store +0 -0
- package/lib/config.d.ts +21 -0
- package/lib/config.js +130 -0
- package/lib/data/conversations.d.ts +6 -0
- package/lib/data/conversations.js +197 -0
- package/lib/data/dynamodb.d.ts +8 -0
- package/lib/data/dynamodb.js +139 -0
- package/lib/data/email.d.ts +7 -0
- package/lib/data/email.js +163 -0
- package/lib/data/files.d.ts +16 -0
- package/lib/data/files.js +406 -0
- package/lib/data/groups.d.ts +13 -0
- package/lib/data/groups.js +354 -0
- package/lib/data/images.d.ts +12 -0
- package/lib/data/images.js +667 -0
- package/{src/data/index.ts → lib/data/index.d.ts} +1 -5
- package/lib/data/index.js +24 -0
- package/lib/data/ios.d.ts +6 -0
- package/lib/data/ios.js +302 -0
- package/lib/data/locations.d.ts +3 -0
- package/lib/data/locations.js +132 -0
- package/lib/data/messages.d.ts +9 -0
- package/lib/data/messages.js +248 -0
- package/lib/data/notifications.d.ts +5 -0
- package/lib/data/notifications.js +42 -0
- package/lib/data/payments.d.ts +11 -0
- package/lib/data/payments.js +748 -0
- package/lib/data/posts.d.ts +22 -0
- package/lib/data/posts.js +578 -0
- package/lib/data/reactions.d.ts +6 -0
- package/lib/data/reactions.js +218 -0
- package/lib/data/s3.d.ts +6 -0
- package/lib/data/s3.js +103 -0
- package/lib/data/search.d.ts +3 -0
- package/lib/data/search.js +98 -0
- package/lib/data/sms.d.ts +3 -0
- package/lib/data/sms.js +59 -0
- package/lib/data/subscription.d.ts +7 -0
- package/lib/data/subscription.js +284 -0
- package/lib/data/tags.d.ts +14 -0
- package/lib/data/tags.js +304 -0
- package/lib/data/users.d.ts +12 -0
- package/lib/data/users.js +310 -0
- package/lib/index.d.ts +3 -0
- package/lib/index.js +8 -0
- package/lib/types/apps.d.ts +43 -0
- package/lib/types/apps.js +2 -0
- package/lib/types/arangodb.d.ts +17 -0
- package/lib/types/arangodb.js +2 -0
- package/lib/types/auth.d.ts +10 -0
- package/lib/types/auth.js +2 -0
- package/lib/types/conversations.d.ts +6 -0
- package/lib/types/conversations.js +2 -0
- package/lib/types/email.d.ts +12 -0
- package/lib/types/email.js +2 -0
- package/lib/types/files.d.ts +26 -0
- package/lib/types/files.js +2 -0
- package/lib/types/google.d.ts +27 -0
- package/lib/types/google.js +2 -0
- package/lib/types/groups.d.ts +21 -0
- package/lib/types/groups.js +2 -0
- package/lib/types/images.d.ts +24 -0
- package/lib/types/images.js +2 -0
- package/{src/types/index.ts → lib/types/index.d.ts} +0 -4
- package/lib/types/index.js +22 -0
- package/lib/types/locations.d.ts +20 -0
- package/lib/types/locations.js +2 -0
- package/lib/types/messages.d.ts +12 -0
- package/lib/types/messages.js +2 -0
- package/lib/types/notifications.d.ts +19 -0
- package/lib/types/notifications.js +2 -0
- package/lib/types/payments.d.ts +114 -0
- package/lib/types/payments.js +2 -0
- package/lib/types/posts.d.ts +28 -0
- package/lib/types/posts.js +2 -0
- package/lib/types/reactions.d.ts +4 -0
- package/lib/types/reactions.js +2 -0
- package/lib/types/tags.d.ts +9 -0
- package/lib/types/tags.js +2 -0
- package/lib/types/users.d.ts +78 -0
- package/lib/types/users.js +2 -0
- package/lib/utils/analytics.d.ts +3 -0
- package/lib/utils/analytics.js +47 -0
- package/lib/utils/arangodb.d.ts +9 -0
- package/lib/utils/arangodb.js +98 -0
- package/lib/utils/auth.d.ts +7 -0
- package/lib/utils/auth.js +80 -0
- package/lib/utils/graphql.d.ts +1 -0
- package/lib/utils/graphql.js +7 -0
- package/{src/utils/index.ts → lib/utils/index.d.ts} +0 -4
- package/lib/utils/index.js +11 -0
- package/lib/utils/objects.d.ts +3 -0
- package/lib/utils/objects.js +34 -0
- package/lib/utils/redis.d.ts +1 -0
- package/lib/utils/redis.js +15 -0
- package/package.json +5 -5
- package/.vscode/extensions.json +0 -15
- package/.vscode/settings.json +0 -82
- package/lex.config.js +0 -4
- package/src/config.ts +0 -127
- package/src/data/conversations.ts +0 -181
- package/src/data/dynamodb.ts +0 -157
- package/src/data/email.ts +0 -163
- package/src/data/files.ts +0 -352
- package/src/data/groups.ts +0 -308
- package/src/data/images.ts +0 -606
- package/src/data/ios.ts +0 -249
- package/src/data/locations.ts +0 -114
- package/src/data/messages.ts +0 -237
- package/src/data/notifications.ts +0 -48
- package/src/data/payments.ts +0 -675
- package/src/data/posts.ts +0 -580
- package/src/data/reactions.ts +0 -186
- package/src/data/s3.ts +0 -117
- package/src/data/search.ts +0 -74
- package/src/data/sms.ts +0 -60
- package/src/data/subscription.ts +0 -228
- package/src/data/tags.ts +0 -230
- package/src/data/users.ts +0 -254
- package/src/index.ts +0 -7
- package/src/types/apps.ts +0 -56
- package/src/types/arangodb.ts +0 -23
- package/src/types/auth.ts +0 -20
- package/src/types/conversations.ts +0 -11
- package/src/types/email.ts +0 -17
- package/src/types/files.ts +0 -31
- package/src/types/google.ts +0 -37
- package/src/types/groups.ts +0 -27
- package/src/types/images.ts +0 -32
- package/src/types/locations.ts +0 -24
- package/src/types/messages.ts +0 -16
- package/src/types/notifications.ts +0 -26
- package/src/types/payments.ts +0 -129
- package/src/types/posts.ts +0 -34
- package/src/types/reactions.ts +0 -8
- package/src/types/tags.ts +0 -13
- package/src/types/users.ts +0 -89
- package/src/utils/analytics.ts +0 -41
- package/src/utils/arangodb.ts +0 -100
- package/src/utils/auth.ts +0 -61
- package/src/utils/graphql.ts +0 -7
- package/src/utils/objects.ts +0 -34
- package/src/utils/redis.ts +0 -17
- package/tsconfig.json +0 -45
|
@@ -0,0 +1,163 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Copyright (c) 2019-Present, Nitrogen Labs, Inc.
|
|
3
|
+
* Copyrights licensed under the MIT License. See the accompanying LICENSE file for terms.
|
|
4
|
+
*/
|
|
5
|
+
import aws, { SES } from 'aws-sdk';
|
|
6
|
+
import fs from 'fs';
|
|
7
|
+
import isEmpty from 'lodash/isEmpty';
|
|
8
|
+
import { Config } from '../config';
|
|
9
|
+
import { s3Get } from './s3'; // const eventCategory: string = 'email';
|
|
10
|
+
|
|
11
|
+
export var appTemplate = function appTemplate(app) {
|
|
12
|
+
var message = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : '';
|
|
13
|
+
var vars = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : {};
|
|
14
|
+
var updatedMessage = '';
|
|
15
|
+
|
|
16
|
+
if (!isEmpty(app)) {
|
|
17
|
+
updatedMessage = message.replace(/\[appName\]/g, app.name).replace(/\[appUrl\]/g, app.url).replace(/\[appEmail\]/g, app.email);
|
|
18
|
+
}
|
|
19
|
+
|
|
20
|
+
Object.keys(vars).forEach(function (key) {
|
|
21
|
+
if (vars[key]) {
|
|
22
|
+
updatedMessage = message.replace(new RegExp("\\[".concat(key, "\\]"), 'g'), vars[key]);
|
|
23
|
+
}
|
|
24
|
+
});
|
|
25
|
+
return updatedMessage;
|
|
26
|
+
};
|
|
27
|
+
export var sendEmail = function sendEmail(params) {
|
|
28
|
+
var _params$app = params.app,
|
|
29
|
+
app = _params$app === void 0 ? {} : _params$app,
|
|
30
|
+
html = params.html,
|
|
31
|
+
subject = params.subject,
|
|
32
|
+
_params$subTitle = params.subTitle,
|
|
33
|
+
subTitle = _params$subTitle === void 0 ? '' : _params$subTitle,
|
|
34
|
+
text = params.text,
|
|
35
|
+
_params$title = params.title,
|
|
36
|
+
title = _params$title === void 0 ? '' : _params$title,
|
|
37
|
+
_params$user = params.user,
|
|
38
|
+
user = _params$user === void 0 ? {} : _params$user;
|
|
39
|
+
var to = [user.email];
|
|
40
|
+
var promise;
|
|
41
|
+
var formatSubject = appTemplate(app, subject);
|
|
42
|
+
var formatSubTitle = appTemplate(app, subTitle);
|
|
43
|
+
var formatTitle = appTemplate(app, title);
|
|
44
|
+
var formatText = appTemplate(app, text);
|
|
45
|
+
var formatHtml = appTemplate(app, html);
|
|
46
|
+
var hasCustomEmail = app._key,
|
|
47
|
+
urlFacebook = app.urlFacebook,
|
|
48
|
+
urlTwitter = app.urlTwitter;
|
|
49
|
+
|
|
50
|
+
var parseTemplate = function parseTemplate(body) {
|
|
51
|
+
var templateBody = appTemplate(app, body || ''); // Links
|
|
52
|
+
|
|
53
|
+
var links = '';
|
|
54
|
+
var spacer = '<td width="5"><img width="1" height="1" border="0" title="" alt="" ' + 'src="https://box.reaktor.io/images/email/spacer.gif"/></td>';
|
|
55
|
+
|
|
56
|
+
if (urlFacebook) {
|
|
57
|
+
links = "<td width=\"30\"><a href=\"".concat(urlFacebook, "\"><img width=\"30\" height=\"30\" border=\"0\" ") + 'title="Facebook" alt="Facebook" src="https://box.reaktor.io/images/email/icon-facebook.png"/></a></td>';
|
|
58
|
+
}
|
|
59
|
+
|
|
60
|
+
if (urlTwitter) {
|
|
61
|
+
if (urlFacebook) {
|
|
62
|
+
links += spacer;
|
|
63
|
+
}
|
|
64
|
+
|
|
65
|
+
links += "<td width=\"30\"><a href=\"".concat(urlTwitter, "\"><img width=\"30\" height=\"30\" border=\"0\" ") + 'title="Twitter" alt="Twitter" src="https://box.reaktor.io/images/email/icon-twitter.png"/></a></td>';
|
|
66
|
+
}
|
|
67
|
+
|
|
68
|
+
if (!links.length) {
|
|
69
|
+
links += spacer;
|
|
70
|
+
}
|
|
71
|
+
|
|
72
|
+
return templateBody.replace(/\[title\]/g, formatTitle).replace(/\[subTitle\]/g, formatSubTitle).replace(/\[subject\]/g, formatSubject).replace(/\[message\]/g, formatHtml).replace(/\[links\]/g, links);
|
|
73
|
+
};
|
|
74
|
+
|
|
75
|
+
if (hasCustomEmail) {
|
|
76
|
+
var s3Params = {
|
|
77
|
+
Bucket: null,
|
|
78
|
+
Key: "apps/templates/email.html"
|
|
79
|
+
};
|
|
80
|
+
promise = s3Get(s3Params).then(function (results) {
|
|
81
|
+
return parseTemplate(results.Body.toString());
|
|
82
|
+
});
|
|
83
|
+
} else {
|
|
84
|
+
promise = new Promise(function (resolve, reject) {
|
|
85
|
+
fs.readFile('./templates/email/layout.html', 'utf8', function (err, data) {
|
|
86
|
+
if (err) {
|
|
87
|
+
return reject(err);
|
|
88
|
+
}
|
|
89
|
+
|
|
90
|
+
return resolve(parseTemplate(data));
|
|
91
|
+
});
|
|
92
|
+
});
|
|
93
|
+
}
|
|
94
|
+
|
|
95
|
+
return promise.then(function (parsedHtml) {
|
|
96
|
+
var emailParams = {
|
|
97
|
+
Destination: {
|
|
98
|
+
ToAddresses: to
|
|
99
|
+
},
|
|
100
|
+
Message: {
|
|
101
|
+
Body: {
|
|
102
|
+
Html: {
|
|
103
|
+
Data: parsedHtml
|
|
104
|
+
},
|
|
105
|
+
Text: {
|
|
106
|
+
Data: formatText
|
|
107
|
+
}
|
|
108
|
+
},
|
|
109
|
+
Subject: {
|
|
110
|
+
Data: formatSubject
|
|
111
|
+
}
|
|
112
|
+
},
|
|
113
|
+
Source: "\"".concat(app.supportName, "\" <noreply@").concat(Config.get('app.url'), ">")
|
|
114
|
+
};
|
|
115
|
+
return sesSend(emailParams);
|
|
116
|
+
});
|
|
117
|
+
};
|
|
118
|
+
export var sendTemplate = function sendTemplate(templateName, emailParams) {
|
|
119
|
+
var app = emailParams.app,
|
|
120
|
+
user = emailParams.user,
|
|
121
|
+
subject = emailParams.subject,
|
|
122
|
+
subTitle = emailParams.subTitle,
|
|
123
|
+
title = emailParams.title,
|
|
124
|
+
vars = emailParams.vars;
|
|
125
|
+
|
|
126
|
+
try {
|
|
127
|
+
var html = fs.readFileSync("./templates/email/".concat(templateName, ".html"), 'utf8');
|
|
128
|
+
var templateHtml = appTemplate(app, html, vars);
|
|
129
|
+
var text = fs.readFileSync("./templates/sms/".concat(templateName, ".txt"), 'utf8');
|
|
130
|
+
var templateText = appTemplate(app, text, vars);
|
|
131
|
+
var params = {
|
|
132
|
+
app: app,
|
|
133
|
+
html: templateHtml,
|
|
134
|
+
subTitle: subTitle,
|
|
135
|
+
subject: subject,
|
|
136
|
+
text: templateText,
|
|
137
|
+
title: title,
|
|
138
|
+
user: user
|
|
139
|
+
};
|
|
140
|
+
return sendEmail(params).then(function () {
|
|
141
|
+
return true;
|
|
142
|
+
}).catch(function () {
|
|
143
|
+
return false;
|
|
144
|
+
});
|
|
145
|
+
} catch (error) {
|
|
146
|
+
return Promise.reject(error);
|
|
147
|
+
}
|
|
148
|
+
}; // AWS::SES
|
|
149
|
+
|
|
150
|
+
export var sesSend = function sesSend(params) {
|
|
151
|
+
return new Promise(function (resolve, reject) {
|
|
152
|
+
aws.config.update(Config.get('aws'));
|
|
153
|
+
var ses = new SES();
|
|
154
|
+
ses.sendEmail(params, function (error, response) {
|
|
155
|
+
if (error) {
|
|
156
|
+
return reject(error);
|
|
157
|
+
}
|
|
158
|
+
|
|
159
|
+
return resolve(response);
|
|
160
|
+
});
|
|
161
|
+
});
|
|
162
|
+
};
|
|
163
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9kYXRhL2VtYWlsLnRzIl0sIm5hbWVzIjpbImF3cyIsIlNFUyIsImZzIiwiaXNFbXB0eSIsIkNvbmZpZyIsInMzR2V0IiwiYXBwVGVtcGxhdGUiLCJhcHAiLCJtZXNzYWdlIiwidmFycyIsInVwZGF0ZWRNZXNzYWdlIiwicmVwbGFjZSIsIm5hbWUiLCJ1cmwiLCJlbWFpbCIsIk9iamVjdCIsImtleXMiLCJmb3JFYWNoIiwia2V5IiwiUmVnRXhwIiwic2VuZEVtYWlsIiwicGFyYW1zIiwiaHRtbCIsInN1YmplY3QiLCJzdWJUaXRsZSIsInRleHQiLCJ0aXRsZSIsInVzZXIiLCJ0byIsInByb21pc2UiLCJmb3JtYXRTdWJqZWN0IiwiZm9ybWF0U3ViVGl0bGUiLCJmb3JtYXRUaXRsZSIsImZvcm1hdFRleHQiLCJmb3JtYXRIdG1sIiwiaGFzQ3VzdG9tRW1haWwiLCJfa2V5IiwidXJsRmFjZWJvb2siLCJ1cmxUd2l0dGVyIiwicGFyc2VUZW1wbGF0ZSIsImJvZHkiLCJ0ZW1wbGF0ZUJvZHkiLCJsaW5rcyIsInNwYWNlciIsImxlbmd0aCIsInMzUGFyYW1zIiwiQnVja2V0IiwiS2V5IiwidGhlbiIsInJlc3VsdHMiLCJCb2R5IiwidG9TdHJpbmciLCJQcm9taXNlIiwicmVzb2x2ZSIsInJlamVjdCIsInJlYWRGaWxlIiwiZXJyIiwiZGF0YSIsInBhcnNlZEh0bWwiLCJlbWFpbFBhcmFtcyIsIkRlc3RpbmF0aW9uIiwiVG9BZGRyZXNzZXMiLCJNZXNzYWdlIiwiSHRtbCIsIkRhdGEiLCJUZXh0IiwiU3ViamVjdCIsIlNvdXJjZSIsInN1cHBvcnROYW1lIiwiZ2V0Iiwic2VzU2VuZCIsInNlbmRUZW1wbGF0ZSIsInRlbXBsYXRlTmFtZSIsInJlYWRGaWxlU3luYyIsInRlbXBsYXRlSHRtbCIsInRlbXBsYXRlVGV4dCIsImNhdGNoIiwiZXJyb3IiLCJjb25maWciLCJ1cGRhdGUiLCJzZXMiLCJyZXNwb25zZSJdLCJtYXBwaW5ncyI6IkFBQUE7Ozs7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QUFHRCxHQW5CRCxDQW1CRSxPQUFNQyxLQUFOLEVBQWE7QUFDYixXQUFPekIsT0FBTyxDQUFDRSxNQUFSLENBQWV1QixLQUFmLENBQVA7QUFDRDtBQUNGLENBekJNLEMsQ0EyQlA7O0FBQ0EsT0FBTyxJQUFNUCxPQUFPLEdBQUcsU0FBVkEsT0FBVSxDQUFDakQsTUFBRCxFQUEwRDtBQUMvRSxTQUFPLElBQUkrQixPQUFKLENBQVksVUFBQ0MsT0FBRCxFQUFVQyxNQUFWLEVBQXFCO0FBQ3RDdEQsSUFBQUEsR0FBRyxDQUFDOEUsTUFBSixDQUFXQyxNQUFYLENBQWtCM0UsTUFBTSxDQUFDaUUsR0FBUCxDQUFXLEtBQVgsQ0FBbEI7QUFDQSxRQUFNVyxHQUFRLEdBQUcsSUFBSS9FLEdBQUosRUFBakI7QUFFQStFLElBQUFBLEdBQUcsQ0FBQzVELFNBQUosQ0FBY0MsTUFBZCxFQUFzQixVQUFDd0QsS0FBRCxFQUFlSSxRQUFmLEVBQStDO0FBQ25FLFVBQUdKLEtBQUgsRUFBVTtBQUNSLGVBQU92QixNQUFNLENBQUN1QixLQUFELENBQWI7QUFDRDs7QUFFRCxhQUFPeEIsT0FBTyxDQUFDNEIsUUFBRCxDQUFkO0FBQ0QsS0FORDtBQU9ELEdBWE0sQ0FBUDtBQVlELENBYk0iLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIENvcHlyaWdodCAoYykgMjAxOS1QcmVzZW50LCBOaXRyb2dlbiBMYWJzLCBJbmMuXG4gKiBDb3B5cmlnaHRzIGxpY2Vuc2VkIHVuZGVyIHRoZSBNSVQgTGljZW5zZS4gU2VlIHRoZSBhY2NvbXBhbnlpbmcgTElDRU5TRSBmaWxlIGZvciB0ZXJtcy5cbiAqL1xuaW1wb3J0IGF3cywge1NFU30gZnJvbSAnYXdzLXNkayc7XG5pbXBvcnQge0dldE9iamVjdFJlcXVlc3R9IGZyb20gJ2F3cy1zZGsvY2xpZW50cy9zMyc7XG5pbXBvcnQge1NlbmRFbWFpbFJlcXVlc3QsIFNlbmRFbWFpbFJlc3BvbnNlfSBmcm9tICdhd3Mtc2RrL2NsaWVudHMvc2VzJztcbmltcG9ydCBmcyBmcm9tICdmcyc7XG5pbXBvcnQgaXNFbXB0eSBmcm9tICdsb2Rhc2gvaXNFbXB0eSc7XG5cbmltcG9ydCB7Q29uZmlnfSBmcm9tICcuLi9jb25maWcnO1xuaW1wb3J0IHtBcHBUeXBlfSBmcm9tICcuLi90eXBlcy9hcHBzJztcbmltcG9ydCB7RW1haWxUeXBlfSBmcm9tICcuLi90eXBlcy9lbWFpbCc7XG5pbXBvcnQge3MzR2V0fSBmcm9tICcuL3MzJztcblxuLy8gY29uc3QgZXZlbnRDYXRlZ29yeTogc3RyaW5nID0gJ2VtYWlsJztcblxuZXhwb3J0IGNvbnN0IGFwcFRlbXBsYXRlID0gKGFwcDogQXBwVHlwZSwgbWVzc2FnZTogc3RyaW5nID0gJycsIHZhcnMgPSB7fSk6IHN0cmluZyA9PiB7XG4gIGxldCB1cGRhdGVkTWVzc2FnZTogc3RyaW5nID0gJyc7XG5cbiAgaWYoIWlzRW1wdHkoYXBwKSkge1xuICAgIHVwZGF0ZWRNZXNzYWdlID0gbWVzc2FnZS5yZXBsYWNlKC9cXFthcHBOYW1lXFxdL2csIGFwcC5uYW1lKVxuICAgICAgLnJlcGxhY2UoL1xcW2FwcFVybFxcXS9nLCBhcHAudXJsKVxuICAgICAgLnJlcGxhY2UoL1xcW2FwcEVtYWlsXFxdL2csIGFwcC5lbWFpbCk7XG4gIH1cblxuICBPYmplY3Qua2V5cyh2YXJzKS5mb3JFYWNoKChrZXk6IHN0cmluZykgPT4ge1xuICAgIGlmKHZhcnNba2V5XSkge1xuICAgICAgdXBkYXRlZE1lc3NhZ2UgPSBtZXNzYWdlLnJlcGxhY2UobmV3IFJlZ0V4cChgXFxcXFske2tleX1cXFxcXWAsICdnJyksIHZhcnNba2V5XSk7XG4gICAgfVxuICB9KTtcblxuICByZXR1cm4gdXBkYXRlZE1lc3NhZ2U7XG59O1xuXG5leHBvcnQgY29uc3Qgc2VuZEVtYWlsID0gKHBhcmFtczogRW1haWxUeXBlKTogUHJvbWlzZTxib29sZWFuPiA9PiB7XG4gIGNvbnN0IHtcbiAgICBhcHAgPSB7fSxcbiAgICBodG1sLFxuICAgIHN1YmplY3QsXG4gICAgc3ViVGl0bGUgPSAnJyxcbiAgICB0ZXh0LFxuICAgIHRpdGxlID0gJycsXG4gICAgdXNlciA9IHt9XG4gIH0gPSBwYXJhbXM7XG5cbiAgY29uc3QgdG8gPSBbdXNlci5lbWFpbF07XG4gIGxldCBwcm9taXNlO1xuXG4gIGNvbnN0IGZvcm1hdFN1YmplY3Q6IHN0cmluZyA9IGFwcFRlbXBsYXRlKGFwcCwgc3ViamVjdCk7XG4gIGNvbnN0IGZvcm1hdFN1YlRpdGxlOiBzdHJpbmcgPSBhcHBUZW1wbGF0ZShhcHAsIHN1YlRpdGxlKTtcbiAgY29uc3QgZm9ybWF0VGl0bGU6IHN0cmluZyA9IGFwcFRlbXBsYXRlKGFwcCwgdGl0bGUpO1xuICBjb25zdCBmb3JtYXRUZXh0OiBzdHJpbmcgPSBhcHBUZW1wbGF0ZShhcHAsIHRleHQpO1xuICBjb25zdCBmb3JtYXRIdG1sOiBzdHJpbmcgPSBhcHBUZW1wbGF0ZShhcHAsIGh0bWwpO1xuICBjb25zdCB7X2tleTogaGFzQ3VzdG9tRW1haWwsIHVybEZhY2Vib29rLCB1cmxUd2l0dGVyfSA9IGFwcDtcblxuICBjb25zdCBwYXJzZVRlbXBsYXRlID0gKGJvZHk6IHN0cmluZykgPT4ge1xuICAgIGNvbnN0IHRlbXBsYXRlQm9keTogc3RyaW5nID0gYXBwVGVtcGxhdGUoYXBwLCBib2R5IHx8ICcnKTtcblxuICAgIC8vIExpbmtzXG4gICAgbGV0IGxpbmtzOiBzdHJpbmcgPSAnJztcbiAgICBjb25zdCBzcGFjZXIgPSAnPHRkIHdpZHRoPVwiNVwiPjxpbWcgd2lkdGg9XCIxXCIgaGVpZ2h0PVwiMVwiIGJvcmRlcj1cIjBcIiB0aXRsZT1cIlwiIGFsdD1cIlwiICcgK1xuICAgICAgJ3NyYz1cImh0dHBzOi8vYm94LnJlYWt0b3IuaW8vaW1hZ2VzL2VtYWlsL3NwYWNlci5naWZcIi8+PC90ZD4nO1xuXG4gICAgaWYodXJsRmFjZWJvb2spIHtcbiAgICAgIGxpbmtzID0gYDx0ZCB3aWR0aD1cIjMwXCI+PGEgaHJlZj1cIiR7dXJsRmFjZWJvb2t9XCI+PGltZyB3aWR0aD1cIjMwXCIgaGVpZ2h0PVwiMzBcIiBib3JkZXI9XCIwXCIgYCArXG4gICAgICAgICd0aXRsZT1cIkZhY2Vib29rXCIgYWx0PVwiRmFjZWJvb2tcIiBzcmM9XCJodHRwczovL2JveC5yZWFrdG9yLmlvL2ltYWdlcy9lbWFpbC9pY29uLWZhY2Vib29rLnBuZ1wiLz48L2E+PC90ZD4nO1xuICAgIH1cblxuICAgIGlmKHVybFR3aXR0ZXIpIHtcbiAgICAgIGlmKHVybEZhY2Vib29rKSB7XG4gICAgICAgIGxpbmtzICs9IHNwYWNlcjtcbiAgICAgIH1cblxuICAgICAgbGlua3MgKz0gYDx0ZCB3aWR0aD1cIjMwXCI+PGEgaHJlZj1cIiR7dXJsVHdpdHRlcn1cIj48aW1nIHdpZHRoPVwiMzBcIiBoZWlnaHQ9XCIzMFwiIGJvcmRlcj1cIjBcIiBgICtcbiAgICAgICAgJ3RpdGxlPVwiVHdpdHRlclwiIGFsdD1cIlR3aXR0ZXJcIiBzcmM9XCJodHRwczovL2JveC5yZWFrdG9yLmlvL2ltYWdlcy9lbWFpbC9pY29uLXR3aXR0ZXIucG5nXCIvPjwvYT48L3RkPic7XG4gICAgfVxuXG4gICAgaWYoIWxpbmtzLmxlbmd0aCkge1xuICAgICAgbGlua3MgKz0gc3BhY2VyO1xuICAgIH1cblxuICAgIHJldHVybiB0ZW1wbGF0ZUJvZHkucmVwbGFjZSgvXFxbdGl0bGVcXF0vZywgZm9ybWF0VGl0bGUpXG4gICAgICAucmVwbGFjZSgvXFxbc3ViVGl0bGVcXF0vZywgZm9ybWF0U3ViVGl0bGUpXG4gICAgICAucmVwbGFjZSgvXFxbc3ViamVjdFxcXS9nLCBmb3JtYXRTdWJqZWN0KVxuICAgICAgLnJlcGxhY2UoL1xcW21lc3NhZ2VcXF0vZywgZm9ybWF0SHRtbClcbiAgICAgIC5yZXBsYWNlKC9cXFtsaW5rc1xcXS9nLCBsaW5rcyk7XG4gIH07XG5cbiAgaWYoaGFzQ3VzdG9tRW1haWwpIHtcbiAgICBjb25zdCBzM1BhcmFtczogR2V0T2JqZWN0UmVxdWVzdCA9IHtCdWNrZXQ6IG51bGwsIEtleTogYGFwcHMvdGVtcGxhdGVzL2VtYWlsLmh0bWxgfTtcbiAgICBwcm9taXNlID0gczNHZXQoczNQYXJhbXMpLnRoZW4oKHJlc3VsdHMpID0+IHBhcnNlVGVtcGxhdGUocmVzdWx0cy5Cb2R5LnRvU3RyaW5nKCkpKTtcbiAgfSBlbHNlIHtcbiAgICBwcm9taXNlID0gbmV3IFByb21pc2UoKHJlc29sdmUsIHJlamVjdCkgPT4ge1xuICAgICAgZnMucmVhZEZpbGUoJy4vdGVtcGxhdGVzL2VtYWlsL2xheW91dC5odG1sJywgJ3V0ZjgnLCAoZXJyLCBkYXRhKSA9PiB7XG4gICAgICAgIGlmKGVycikge1xuICAgICAgICAgIHJldHVybiByZWplY3QoZXJyKTtcbiAgICAgICAgfVxuXG4gICAgICAgIHJldHVybiByZXNvbHZlKHBhcnNlVGVtcGxhdGUoZGF0YSkpO1xuICAgICAgfSk7XG4gICAgfSk7XG4gIH1cblxuICByZXR1cm4gcHJvbWlzZS50aGVuKChwYXJzZWRIdG1sOiBzdHJpbmcpID0+IHtcbiAgICBjb25zdCBlbWFpbFBhcmFtczogU2VuZEVtYWlsUmVxdWVzdCA9IHtcbiAgICAgIERlc3RpbmF0aW9uOiB7VG9BZGRyZXNzZXM6IHRvfSxcbiAgICAgIE1lc3NhZ2U6IHtcbiAgICAgICAgQm9keToge1xuICAgICAgICAgIEh0bWw6IHtEYXRhOiBwYXJzZWRIdG1sfSxcbiAgICAgICAgICBUZXh0OiB7RGF0YTogZm9ybWF0VGV4dH1cbiAgICAgICAgfSxcbiAgICAgICAgU3ViamVjdDoge0RhdGE6IGZvcm1hdFN1YmplY3R9XG4gICAgICB9LFxuICAgICAgU291cmNlOiBgXCIke2FwcC5zdXBwb3J0TmFtZX1cIiA8bm9yZXBseUAke0NvbmZpZy5nZXQoJ2FwcC51cmwnKX0+YFxuICAgIH07XG5cbiAgICByZXR1cm4gc2VzU2VuZChlbWFpbFBhcmFtcyk7XG4gIH0pO1xufTtcblxuZXhwb3J0IGNvbnN0IHNlbmRUZW1wbGF0ZSA9ICh0ZW1wbGF0ZU5hbWU6IHN0cmluZywgZW1haWxQYXJhbXM6IEVtYWlsVHlwZSk6IFByb21pc2U8Ym9vbGVhbj4gPT4ge1xuICBjb25zdCB7YXBwLCB1c2VyLCBzdWJqZWN0LCBzdWJUaXRsZSwgdGl0bGUsIHZhcnN9OiBFbWFpbFR5cGUgPSBlbWFpbFBhcmFtcztcblxuICB0cnkge1xuICAgIGNvbnN0IGh0bWw6IHN0cmluZyA9IGZzLnJlYWRGaWxlU3luYyhgLi90ZW1wbGF0ZXMvZW1haWwvJHt0ZW1wbGF0ZU5hbWV9Lmh0bWxgLCAndXRmOCcpO1xuICAgIGNvbnN0IHRlbXBsYXRlSHRtbDogc3RyaW5nID0gYXBwVGVtcGxhdGUoYXBwLCBodG1sLCB2YXJzKTtcbiAgICBjb25zdCB0ZXh0OiBzdHJpbmcgPSBmcy5yZWFkRmlsZVN5bmMoYC4vdGVtcGxhdGVzL3Ntcy8ke3RlbXBsYXRlTmFtZX0udHh0YCwgJ3V0ZjgnKTtcbiAgICBjb25zdCB0ZW1wbGF0ZVRleHQ6IHN0cmluZyA9IGFwcFRlbXBsYXRlKGFwcCwgdGV4dCwgdmFycyk7XG5cbiAgICBjb25zdCBwYXJhbXMgPSB7XG4gICAgICBhcHAsXG4gICAgICBodG1sOiB0ZW1wbGF0ZUh0bWwsXG4gICAgICBzdWJUaXRsZSxcbiAgICAgIHN1YmplY3QsXG4gICAgICB0ZXh0OiB0ZW1wbGF0ZVRleHQsXG4gICAgICB0aXRsZSxcbiAgICAgIHVzZXJcbiAgICB9O1xuXG4gICAgcmV0dXJuIHNlbmRFbWFpbChwYXJhbXMpXG4gICAgICAudGhlbigoKSA9PiB0cnVlKVxuICAgICAgLmNhdGNoKCgpID0+IGZhbHNlKTtcbiAgfSBjYXRjaChlcnJvcikge1xuICAgIHJldHVybiBQcm9taXNlLnJlamVjdChlcnJvcik7XG4gIH1cbn07XG5cbi8vIEFXUzo6U0VTXG5leHBvcnQgY29uc3Qgc2VzU2VuZCA9IChwYXJhbXM6IFNlbmRFbWFpbFJlcXVlc3QpOiBQcm9taXNlPFNlbmRFbWFpbFJlc3BvbnNlPiA9PiB7XG4gIHJldHVybiBuZXcgUHJvbWlzZSgocmVzb2x2ZSwgcmVqZWN0KSA9PiB7XG4gICAgYXdzLmNvbmZpZy51cGRhdGUoQ29uZmlnLmdldCgnYXdzJykpO1xuICAgIGNvbnN0IHNlczogU0VTID0gbmV3IFNFUygpO1xuXG4gICAgc2VzLnNlbmRFbWFpbChwYXJhbXMsIChlcnJvcjogRXJyb3IsIHJlc3BvbnNlOiBTZW5kRW1haWxSZXNwb25zZSkgPT4ge1xuICAgICAgaWYoZXJyb3IpIHtcbiAgICAgICAgcmV0dXJuIHJlamVjdChlcnJvcik7XG4gICAgICB9XG5cbiAgICAgIHJldHVybiByZXNvbHZlKHJlc3BvbnNlKTtcbiAgICB9KTtcbiAgfSk7XG59O1xuIl19
|
|
@@ -0,0 +1,16 @@
|
|
|
1
|
+
/// <reference types="node" />
|
|
2
|
+
import { Database } from 'arangojs';
|
|
3
|
+
import { ApiContext } from 'types/auth';
|
|
4
|
+
import { FileType } from '../types/files';
|
|
5
|
+
export declare const addFile: (context: ApiContext, item?: FileType) => Promise<FileType>;
|
|
6
|
+
export declare const getGiphyTrends: (context: ApiContext, limit?: number) => Promise<any[]>;
|
|
7
|
+
export declare const getGiphySearch: (context: ApiContext, query: string, limit?: number) => Promise<any[]>;
|
|
8
|
+
export declare const getYouTubeTrends: (context: ApiContext, limit?: number) => Promise<any[]>;
|
|
9
|
+
export declare const getYouTubeSearch: (context: ApiContext, query: string, limit?: number) => Promise<any[]>;
|
|
10
|
+
export declare const getPathUserFiles: (userId: string, filename: string) => string;
|
|
11
|
+
export declare const getUrlUserFiles: (userId: string, filename: string, dir?: string, type?: string) => string;
|
|
12
|
+
export declare const linkFiles: (db: Database, files: FileType[], postId: string) => Promise<any>;
|
|
13
|
+
export declare const updateFiles: (db: Database, postId: string, files: FileType[]) => Promise<any>;
|
|
14
|
+
export declare const createFile: (db: Database, file: FileType) => Promise<FileType>;
|
|
15
|
+
export declare const encodeBase64: (buffer: Buffer) => string;
|
|
16
|
+
export declare const decodeBase64: (dataString: string) => object;
|
|
@@ -0,0 +1,406 @@
|
|
|
1
|
+
function _templateObject3() {
|
|
2
|
+
var data = _taggedTemplateLiteral(["UPSERT {_key: ", "}\n INSERT ", "\n UPDATE {}\n IN files RETURN NEW"]);
|
|
3
|
+
|
|
4
|
+
_templateObject3 = function _templateObject3() {
|
|
5
|
+
return data;
|
|
6
|
+
};
|
|
7
|
+
|
|
8
|
+
return data;
|
|
9
|
+
}
|
|
10
|
+
|
|
11
|
+
function _templateObject2() {
|
|
12
|
+
var data = _taggedTemplateLiteral(["REMOVE {_key:", "} IN isPosted"]);
|
|
13
|
+
|
|
14
|
+
_templateObject2 = function _templateObject2() {
|
|
15
|
+
return data;
|
|
16
|
+
};
|
|
17
|
+
|
|
18
|
+
return data;
|
|
19
|
+
}
|
|
20
|
+
|
|
21
|
+
function _objectSpread(target) { for (var i = 1; i < arguments.length; i++) { var source = arguments[i] != null ? arguments[i] : {}; var ownKeys = Object.keys(source); if (typeof Object.getOwnPropertySymbols === 'function') { ownKeys = ownKeys.concat(Object.getOwnPropertySymbols(source).filter(function (sym) { return Object.getOwnPropertyDescriptor(source, sym).enumerable; })); } ownKeys.forEach(function (key) { _defineProperty(target, key, source[key]); }); } return target; }
|
|
22
|
+
|
|
23
|
+
function _defineProperty(obj, key, value) { if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; }
|
|
24
|
+
|
|
25
|
+
function _templateObject() {
|
|
26
|
+
var data = _taggedTemplateLiteral(["INSERT ", " IN files RETURN NEW"]);
|
|
27
|
+
|
|
28
|
+
_templateObject = function _templateObject() {
|
|
29
|
+
return data;
|
|
30
|
+
};
|
|
31
|
+
|
|
32
|
+
return data;
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
function _taggedTemplateLiteral(strings, raw) { if (!raw) { raw = strings.slice(0); } return Object.freeze(Object.defineProperties(strings, { raw: { value: Object.freeze(raw) } })); }
|
|
36
|
+
|
|
37
|
+
/**
|
|
38
|
+
* Copyright (c) 2019-Present, Nitrogen Labs, Inc.
|
|
39
|
+
* Copyrights licensed under the MIT License. See the accompanying LICENSE file for terms.
|
|
40
|
+
*/
|
|
41
|
+
import { get as httpGet } from '@nlabs/rip-hunter';
|
|
42
|
+
import { createHash, parseId, parseString } from '@nlabs/utils';
|
|
43
|
+
import { aql } from 'arangojs';
|
|
44
|
+
import { google } from 'googleapis';
|
|
45
|
+
import { UserError } from 'graphql-errors';
|
|
46
|
+
import * as request from 'request-promise';
|
|
47
|
+
import { Config } from '../config';
|
|
48
|
+
import { useDb } from '../utils';
|
|
49
|
+
import { resizeSaveImage } from './images';
|
|
50
|
+
import { createPostEdge } from './posts';
|
|
51
|
+
var youtube = google.youtube({
|
|
52
|
+
auth: Config.get('google.key'),
|
|
53
|
+
version: 'v3'
|
|
54
|
+
});
|
|
55
|
+
request.defaults({
|
|
56
|
+
encoding: null
|
|
57
|
+
}); // const eventCategory: string = 'files';
|
|
58
|
+
// Upload file
|
|
59
|
+
|
|
60
|
+
export var addFile = function addFile(context) {
|
|
61
|
+
var item = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : {};
|
|
62
|
+
var database = context.database,
|
|
63
|
+
sessionId = context.userId,
|
|
64
|
+
userType = context.userType;
|
|
65
|
+
var _item$description = item.description,
|
|
66
|
+
description = _item$description === void 0 ? '' : _item$description,
|
|
67
|
+
_item$fileType = item.fileType,
|
|
68
|
+
fileType = _item$fileType === void 0 ? '' : _item$fileType,
|
|
69
|
+
id = item.id,
|
|
70
|
+
_item$name = item.name,
|
|
71
|
+
name = _item$name === void 0 ? '' : _item$name,
|
|
72
|
+
_item$url = item.url,
|
|
73
|
+
url = _item$url === void 0 ? '' : _item$url; // Id
|
|
74
|
+
|
|
75
|
+
var fileId = id ? parseId(id) : createHash("file-".concat(sessionId)); // Name
|
|
76
|
+
|
|
77
|
+
var isUrl = url !== ''; // If no name, get it from url path
|
|
78
|
+
|
|
79
|
+
var formatName = parseString(name, 160);
|
|
80
|
+
var formatType = parseString(fileType, 16);
|
|
81
|
+
|
|
82
|
+
if (formatName === '' && isUrl) {
|
|
83
|
+
formatName = url.substring(url.lastIndexOf('/') + 1);
|
|
84
|
+
}
|
|
85
|
+
|
|
86
|
+
if (formatType === '') {
|
|
87
|
+
var nameArr = formatName.split('.');
|
|
88
|
+
var ext = nameArr[nameArr.length - 1];
|
|
89
|
+
|
|
90
|
+
switch (ext) {
|
|
91
|
+
case 'jpeg':
|
|
92
|
+
case 'jpg':
|
|
93
|
+
formatType = 'image/jpeg';
|
|
94
|
+
break;
|
|
95
|
+
|
|
96
|
+
case 'png':
|
|
97
|
+
formatType = 'image/png';
|
|
98
|
+
break;
|
|
99
|
+
|
|
100
|
+
case 'zip':
|
|
101
|
+
formatType = 'application/zip';
|
|
102
|
+
break;
|
|
103
|
+
|
|
104
|
+
default:
|
|
105
|
+
break;
|
|
106
|
+
}
|
|
107
|
+
}
|
|
108
|
+
|
|
109
|
+
var isImage;
|
|
110
|
+
|
|
111
|
+
switch (formatType) {
|
|
112
|
+
case 'image/jpeg':
|
|
113
|
+
case 'image/png':
|
|
114
|
+
isImage = true;
|
|
115
|
+
break;
|
|
116
|
+
|
|
117
|
+
default:
|
|
118
|
+
isImage = false;
|
|
119
|
+
break;
|
|
120
|
+
} // Description
|
|
121
|
+
|
|
122
|
+
|
|
123
|
+
var formatDesc = parseString(description, 500); // Only allow file uploads to premium users
|
|
124
|
+
|
|
125
|
+
if (!isImage && userType !== 2) {
|
|
126
|
+
throw new UserError('account_restriction');
|
|
127
|
+
}
|
|
128
|
+
|
|
129
|
+
var saveToDb = function saveToDb(insert) {
|
|
130
|
+
var aqlQry = aql(_templateObject(), insert);
|
|
131
|
+
return useDb(database).query(aqlQry).then(function (cursor) {
|
|
132
|
+
return cursor.next();
|
|
133
|
+
}).then(function () {
|
|
134
|
+
var file = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {};
|
|
135
|
+
return file;
|
|
136
|
+
}).catch(function (error) {
|
|
137
|
+
throw error;
|
|
138
|
+
});
|
|
139
|
+
};
|
|
140
|
+
|
|
141
|
+
var uploadFile = function uploadFile(buf, uploadType) {
|
|
142
|
+
var now = Date.now(); // If image, resize and create a thumbnail
|
|
143
|
+
|
|
144
|
+
if (isImage) {
|
|
145
|
+
return resizeSaveImage(sessionId, fileId, buf, uploadType).then(function (resizedImage) {
|
|
146
|
+
var insert = _objectSpread({}, resizedImage, {
|
|
147
|
+
_key: fileId,
|
|
148
|
+
added: now,
|
|
149
|
+
description: formatDesc,
|
|
150
|
+
fileType: formatType,
|
|
151
|
+
modified: now,
|
|
152
|
+
name: formatName,
|
|
153
|
+
userId: sessionId
|
|
154
|
+
});
|
|
155
|
+
|
|
156
|
+
return saveToDb(insert);
|
|
157
|
+
}).catch(function (error) {
|
|
158
|
+
throw error;
|
|
159
|
+
});
|
|
160
|
+
}
|
|
161
|
+
|
|
162
|
+
var insert = {
|
|
163
|
+
_key: fileId,
|
|
164
|
+
added: now,
|
|
165
|
+
description: formatDesc,
|
|
166
|
+
fileType: formatType,
|
|
167
|
+
modified: now,
|
|
168
|
+
name: formatName,
|
|
169
|
+
userId: sessionId
|
|
170
|
+
};
|
|
171
|
+
return saveToDb(insert);
|
|
172
|
+
}; // If file is a url path, download the file and save
|
|
173
|
+
|
|
174
|
+
|
|
175
|
+
if (isUrl) {
|
|
176
|
+
return request.get({
|
|
177
|
+
encoding: null,
|
|
178
|
+
uri: url
|
|
179
|
+
}).then(function (body) {
|
|
180
|
+
return uploadFile(new Buffer(body, 'binary'), formatType);
|
|
181
|
+
}).catch(function () {
|
|
182
|
+
throw new UserError('file_request');
|
|
183
|
+
});
|
|
184
|
+
} else if (item.base64 !== '') {
|
|
185
|
+
var buffer = new Buffer(item.base64);
|
|
186
|
+
return uploadFile(buffer, formatType);
|
|
187
|
+
}
|
|
188
|
+
|
|
189
|
+
throw new Error('file_required');
|
|
190
|
+
}; // Giphy
|
|
191
|
+
|
|
192
|
+
export var getGiphyTrends = function getGiphyTrends(context) {
|
|
193
|
+
var limit = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : 30;
|
|
194
|
+
var gifUrl = "http://api.giphy.com/v1/gifs/trending?api_key=".concat(Config.get('giphy.key'), "&limit=").concat(limit);
|
|
195
|
+
return httpGet(gifUrl).then(function (res) {
|
|
196
|
+
return res.json();
|
|
197
|
+
}).then(function (json) {
|
|
198
|
+
return json.data.map(function () {
|
|
199
|
+
var gifImage = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {
|
|
200
|
+
id: null,
|
|
201
|
+
images: null
|
|
202
|
+
};
|
|
203
|
+
var id = gifImage.id,
|
|
204
|
+
_gifImage$images = gifImage.images;
|
|
205
|
+
_gifImage$images = _gifImage$images === void 0 ? {} : _gifImage$images;
|
|
206
|
+
var _gifImage$images$orig = _gifImage$images.original;
|
|
207
|
+
_gifImage$images$orig = _gifImage$images$orig === void 0 ? {} : _gifImage$images$orig;
|
|
208
|
+
var _gifImage$images$orig2 = _gifImage$images$orig.url,
|
|
209
|
+
url = _gifImage$images$orig2 === void 0 ? '' : _gifImage$images$orig2,
|
|
210
|
+
_gifImage$images$fixe = _gifImage$images.fixed_height_small;
|
|
211
|
+
_gifImage$images$fixe = _gifImage$images$fixe === void 0 ? {} : _gifImage$images$fixe;
|
|
212
|
+
var _gifImage$images$fixe2 = _gifImage$images$fixe.url,
|
|
213
|
+
thumb = _gifImage$images$fixe2 === void 0 ? '' : _gifImage$images$fixe2;
|
|
214
|
+
return {
|
|
215
|
+
id: id,
|
|
216
|
+
thumb: thumb,
|
|
217
|
+
type: 'giphy',
|
|
218
|
+
url: url
|
|
219
|
+
};
|
|
220
|
+
});
|
|
221
|
+
});
|
|
222
|
+
};
|
|
223
|
+
export var getGiphySearch = function getGiphySearch(context, query) {
|
|
224
|
+
var limit = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : 30;
|
|
225
|
+
var formatQuery = encodeURI(query);
|
|
226
|
+
var gifUrl = "http://api.giphy.com/v1/gifs/search?q=".concat(formatQuery, "&api_key=").concat(Config.get('giphy.key'), "&limit=").concat(limit);
|
|
227
|
+
return fetch(gifUrl).then(function (res) {
|
|
228
|
+
return res.json();
|
|
229
|
+
}).then(function (json) {
|
|
230
|
+
return json.data.map(function () {
|
|
231
|
+
var gifImage = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {
|
|
232
|
+
id: null,
|
|
233
|
+
images: null
|
|
234
|
+
};
|
|
235
|
+
var id = gifImage.id,
|
|
236
|
+
_gifImage$images2 = gifImage.images;
|
|
237
|
+
_gifImage$images2 = _gifImage$images2 === void 0 ? {} : _gifImage$images2;
|
|
238
|
+
var _gifImage$images2$ori = _gifImage$images2.original;
|
|
239
|
+
_gifImage$images2$ori = _gifImage$images2$ori === void 0 ? {} : _gifImage$images2$ori;
|
|
240
|
+
var _gifImage$images2$ori2 = _gifImage$images2$ori.url,
|
|
241
|
+
url = _gifImage$images2$ori2 === void 0 ? '' : _gifImage$images2$ori2,
|
|
242
|
+
_gifImage$images2$fix = _gifImage$images2.fixed_height_small;
|
|
243
|
+
_gifImage$images2$fix = _gifImage$images2$fix === void 0 ? {} : _gifImage$images2$fix;
|
|
244
|
+
var _gifImage$images2$fix2 = _gifImage$images2$fix.url,
|
|
245
|
+
thumb = _gifImage$images2$fix2 === void 0 ? '' : _gifImage$images2$fix2;
|
|
246
|
+
return {
|
|
247
|
+
id: id,
|
|
248
|
+
thumb: thumb,
|
|
249
|
+
type: 'giphy',
|
|
250
|
+
url: url
|
|
251
|
+
};
|
|
252
|
+
});
|
|
253
|
+
});
|
|
254
|
+
};
|
|
255
|
+
export var getYouTubeTrends = function getYouTubeTrends(context) {
|
|
256
|
+
var limit = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : 30;
|
|
257
|
+
return new Promise(function (resolve, reject) {
|
|
258
|
+
youtube.videos.list({
|
|
259
|
+
chart: 'mostPopular',
|
|
260
|
+
maxResults: limit,
|
|
261
|
+
part: 'snippet',
|
|
262
|
+
regionCode: 'US'
|
|
263
|
+
}, function (error, data) {
|
|
264
|
+
if (error) {
|
|
265
|
+
console.error(error);
|
|
266
|
+
reject(new Error(error[0].message));
|
|
267
|
+
} else if (data) {
|
|
268
|
+
var list = data.items.map(function (item) {
|
|
269
|
+
return {
|
|
270
|
+
id: item.id,
|
|
271
|
+
thumb: item.snippet.thumbnails.high.url,
|
|
272
|
+
type: 'youtube',
|
|
273
|
+
url: "http://www.youtube.com/embed/".concat(item.id)
|
|
274
|
+
};
|
|
275
|
+
});
|
|
276
|
+
resolve(list);
|
|
277
|
+
}
|
|
278
|
+
});
|
|
279
|
+
});
|
|
280
|
+
};
|
|
281
|
+
export var getYouTubeSearch = function getYouTubeSearch(context, query) {
|
|
282
|
+
var limit = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : 30;
|
|
283
|
+
return new Promise(function (resolve, reject) {
|
|
284
|
+
youtube.search.list({
|
|
285
|
+
maxResults: limit,
|
|
286
|
+
part: 'snippet',
|
|
287
|
+
q: query,
|
|
288
|
+
regionCode: 'US'
|
|
289
|
+
}, function (error, data) {
|
|
290
|
+
if (error) {
|
|
291
|
+
console.error(error);
|
|
292
|
+
reject(new Error(error[0].message));
|
|
293
|
+
} else if (data) {
|
|
294
|
+
var items = data.items;
|
|
295
|
+
var list = items.map(function (item) {
|
|
296
|
+
return {
|
|
297
|
+
id: item.id,
|
|
298
|
+
thumb: item.snippet.thumbnails.high.url,
|
|
299
|
+
type: 'youtube',
|
|
300
|
+
url: "http://www.youtube.com/embed/".concat(item.id)
|
|
301
|
+
};
|
|
302
|
+
});
|
|
303
|
+
resolve(list);
|
|
304
|
+
}
|
|
305
|
+
});
|
|
306
|
+
});
|
|
307
|
+
}; // Files
|
|
308
|
+
|
|
309
|
+
export var getPathUserFiles = function getPathUserFiles(userId, filename) {
|
|
310
|
+
return "users/".concat(userId, "/files/").concat(filename);
|
|
311
|
+
};
|
|
312
|
+
export var getUrlUserFiles = function getUrlUserFiles(userId, filename) {
|
|
313
|
+
var dir = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : 'files';
|
|
314
|
+
var type = arguments.length > 3 && arguments[3] !== undefined ? arguments[3] : 'profile';
|
|
315
|
+
|
|
316
|
+
if (filename) {
|
|
317
|
+
return "https://box.".concat(Config.get('app.url'), "/users/").concat(userId, "/").concat(dir, "/").concat(filename);
|
|
318
|
+
}
|
|
319
|
+
|
|
320
|
+
if (type === 'profile') {
|
|
321
|
+
return "https://box.".concat(Config.get('app.url'), "/defaults/user_bk.jpg");
|
|
322
|
+
}
|
|
323
|
+
|
|
324
|
+
return "https://box.".concat(Config.get('app.url'), "/defaults/user_wh.jpg");
|
|
325
|
+
};
|
|
326
|
+
export var linkFiles = function linkFiles(db, files, postId) {
|
|
327
|
+
return Promise.all(files.map(function (file) {
|
|
328
|
+
return createFile(db, file).then(function (file) {
|
|
329
|
+
return createPostEdge(db, file, postId);
|
|
330
|
+
});
|
|
331
|
+
}));
|
|
332
|
+
};
|
|
333
|
+
export var updateFiles = function updateFiles(db, postId, files) {
|
|
334
|
+
var edgeCollection = db.edgeCollection('isPosted');
|
|
335
|
+
return edgeCollection.inEdges(postId).then(function (edges) {
|
|
336
|
+
if (edges.length) {
|
|
337
|
+
// Remove linked edges
|
|
338
|
+
return Promise.all(edges.map(function (edge) {
|
|
339
|
+
var edgeKey = edge._key;
|
|
340
|
+
var aqlQry = aql(_templateObject2(), edgeKey);
|
|
341
|
+
return db.query(aqlQry).catch(function (error) {
|
|
342
|
+
throw error;
|
|
343
|
+
});
|
|
344
|
+
})).then(function () {
|
|
345
|
+
if (files.length) {
|
|
346
|
+
// Link files
|
|
347
|
+
return linkFiles(db, files, postId).then(function () {
|
|
348
|
+
return files;
|
|
349
|
+
});
|
|
350
|
+
}
|
|
351
|
+
|
|
352
|
+
return files;
|
|
353
|
+
});
|
|
354
|
+
} else if (files.length) {
|
|
355
|
+
// Link files
|
|
356
|
+
return linkFiles(db, files, postId).then(function () {
|
|
357
|
+
return files;
|
|
358
|
+
});
|
|
359
|
+
}
|
|
360
|
+
|
|
361
|
+
return files;
|
|
362
|
+
}).catch(function (error) {
|
|
363
|
+
throw error;
|
|
364
|
+
});
|
|
365
|
+
};
|
|
366
|
+
export var createFile = function createFile(db, file) {
|
|
367
|
+
var insert = {
|
|
368
|
+
_key: file.id,
|
|
369
|
+
added: Date.now()
|
|
370
|
+
};
|
|
371
|
+
var aqlQry = aql(_templateObject3(), file.id, insert);
|
|
372
|
+
return db.query(aqlQry).then(function (cursor) {
|
|
373
|
+
return cursor.next();
|
|
374
|
+
}).then(function () {
|
|
375
|
+
var updatedFile = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {};
|
|
376
|
+
return updatedFile;
|
|
377
|
+
}).catch(function (error) {
|
|
378
|
+
throw error;
|
|
379
|
+
});
|
|
380
|
+
};
|
|
381
|
+
export var encodeBase64 = function encodeBase64(buffer) {
|
|
382
|
+
return new Buffer(buffer).toString('base64');
|
|
383
|
+
};
|
|
384
|
+
export var decodeBase64 = function decodeBase64(dataString) {
|
|
385
|
+
// const getData = (str: string) => str.match(/^data:([A-Za-z-+\/]+);base64,(.+)$/) || [];
|
|
386
|
+
var getData = function getData(str) {
|
|
387
|
+
return str.match(/^data:([A-Za-z-+/]+);base64,(.+)$/) || [];
|
|
388
|
+
};
|
|
389
|
+
|
|
390
|
+
var matches = getData(dataString);
|
|
391
|
+
|
|
392
|
+
if (matches.length !== 3) {
|
|
393
|
+
// If invalid make sure we don't need to decode
|
|
394
|
+
matches = getData(decodeURIComponent(dataString)); // Check it again.
|
|
395
|
+
|
|
396
|
+
if (matches.length !== 3) {
|
|
397
|
+
throw Error('Invalid input string');
|
|
398
|
+
}
|
|
399
|
+
}
|
|
400
|
+
|
|
401
|
+
return {
|
|
402
|
+
data: new Buffer(matches[2], 'base64'),
|
|
403
|
+
type: matches[1]
|
|
404
|
+
};
|
|
405
|
+
};
|
|
406
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9kYXRhL2ZpbGVzLnRzIl0sIm5hbWVzIjpbImdldCIsImh0dHBHZXQiLCJjcmVhdGVIYXNoIiwicGFyc2VJZCIsInBhcnNlU3RyaW5nIiwiYXFsIiwiZ29vZ2xlIiwiVXNlckVycm9yIiwicmVxdWVzdCIsIkNvbmZpZyIsInVzZURiIiwicmVzaXplU2F2ZUltYWdlIiwiY3JlYXRlUG9zdEVkZ2UiLCJ5b3V0dWJlIiwiYXV0aCIsInZlcnNpb24iLCJkZWZhdWx0cyIsImVuY29kaW5nIiwiYWRkRmlsZSIsImNvbnRleHQiLCJpdGVtIiwiZGF0YWJhc2UiLCJzZXNzaW9uSWQiLCJ1c2VySWQiLCJ1c2VyVHlwZSIsImRlc2NyaXB0aW9uIiwiZmlsZVR5cGUiLCJpZCIsIm5hbWUiLCJ1cmwiLCJmaWxlSWQiLCJpc1VybCIsImZvcm1hdE5hbWUiLCJmb3JtYXRUeXBlIiwic3Vic3RyaW5nIiwibGFzdEluZGV4T2YiLCJuYW1lQXJyIiwic3BsaXQiLCJleHQiLCJsZW5ndGgiLCJpc0ltYWdlIiwiZm9ybWF0RGVzYyIsInNhdmVUb0RiIiwiaW5zZXJ0IiwiYXFsUXJ5IiwicXVlcnkiLCJ0aGVuIiwiY3Vyc29yIiwibmV4dCIsImZpbGUiLCJjYXRjaCIsImVycm9yIiwidXBsb2FkRmlsZSIsImJ1ZiIsInVwbG9hZFR5cGUiLCJub3ciLCJEYXRlIiwicmVzaXplZEltYWdlIiwiX2tleSIsImFkZGVkIiwibW9kaWZpZWQiLCJ1cmkiLCJib2R5IiwiQnVmZmVyIiwiYmFzZTY0IiwiYnVmZmVyIiwiRXJyb3IiLCJnZXRHaXBoeVRyZW5kcyIsImxpbWl0IiwiZ2lmVXJsIiwicmVzIiwianNvbiIsImRhdGEiLCJtYXAiLCJnaWZJbWFnZSIsImltYWdlcyIsIm9yaWdpbmFsIiwiZml4ZWRfaGVpZ2h0X3NtYWxsIiwidGh1bWIiLCJ0eXBlIiwiZ2V0R2lwaHlTZWFyY2giLCJmb3JtYXRRdWVyeSIsImVuY29kZVVSSSIsImZldGNoIiwiZ2V0WW91VHViZVRyZW5kcyIsIlByb21pc2UiLCJyZXNvbHZlIiwicmVqZWN0IiwidmlkZW9zIiwibGlzdCIsImNoYXJ0IiwibWF4UmVzdWx0cyIsInBhcnQiLCJyZWdpb25Db2RlIiwiY29uc29sZSIsIm1lc3NhZ2UiLCJpdGVtcyIsInNuaXBwZXQiLCJ0aHVtYm5haWxzIiwiaGlnaCIsImdldFlvdVR1YmVTZWFyY2giLCJzZWFyY2giLCJxIiwiZ2V0UGF0aFVzZXJGaWxlcyIsImZpbGVuYW1lIiwiZ2V0VXJsVXNlckZpbGVzIiwiZGlyIiwibGlua0ZpbGVzIiwiZGIiLCJmaWxlcyIsInBvc3RJZCIsImFsbCIsImNyZWF0ZUZpbGUiLCJ1cGRhdGVGaWxlcyIsImVkZ2VDb2xsZWN0aW9uIiwiaW5FZGdlcyIsImVkZ2VzIiwiZWRnZSIsImVkZ2VLZXkiLCJ1cGRhdGVkRmlsZSIsImVuY29kZUJhc2U2NCIsInRvU3RyaW5nIiwiZGVjb2RlQmFzZTY0IiwiZGF0YVN0cmluZyIsImdldERhdGEiLCJzdHIiLCJtYXRjaCIsIm1hdGNoZXMiLCJkZWNvZGVVUklDb21wb25lbnQiXSwibWFwcGluZ3MiOiI7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztBQUFBOzs7O0FBSUEsU0FBUUEsR0FBRyxJQUFJQyxPQUFmLFFBQTZCLG1CQUE3QjtBQUNBLFNBQVFDLFVBQVIsRUFBb0JDLE9BQXBCLEVBQTZCQyxXQUE3QixRQUErQyxjQUEvQztBQUNBLFNBQVFDLEdBQVIsUUFBNEIsVUFBNUI7QUFHQSxTQUFRQyxNQUFSLFFBQXFCLFlBQXJCO0FBQ0EsU0FBUUMsU0FBUixRQUF3QixnQkFBeEI7QUFDQSxPQUFPLEtBQUtDLE9BQVosTUFBeUIsaUJBQXpCO0FBR0EsU0FBUUMsTUFBUixRQUFxQixXQUFyQjtBQUVBLFNBQVFDLEtBQVIsUUFBb0IsVUFBcEI7QUFDQSxTQUFRQyxlQUFSLFFBQThCLFVBQTlCO0FBQ0EsU0FBUUMsY0FBUixRQUE2QixTQUE3QjtBQUVBLElBQU1DLE9BQU8sR0FBR1AsTUFBTSxDQUFDTyxPQUFQLENBQWU7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QWdDO0FBQzFEO0FBQ0EsTUFBTUMsT0FBTyxHQUFHLFNBQVZBLE9BQVUsQ0FBQ0MsR0FBRDtBQUFBLFdBQWlCQSxHQUFHLENBQUNDLEtBQUosQ0FBVSxtQ0FBVixLQUFrRCxFQUFuRTtBQUFBLEdBQWhCOztBQUNBLE1BQUlDLE9BQU8sR0FBR0gsT0FBTyxDQUFDRCxVQUFELENBQXJCOztBQUVBLE1BQUdJLE9BQU8sQ0FBQ3hGLE1BQVIsS0FBbUIsQ0FBdEIsRUFBeUI7QUFDdkI7QUFDQXdGLElBQUFBLE9BQU8sR0FBR0gsT0FBTyxDQUFDSSxrQkFBa0IsQ0FBQ0wsVUFBRCxDQUFuQixDQUFqQixDQUZ1QixDQUl2Qjs7QUFDQSxRQUFHSSxPQUFPLENBQUN4RixNQUFSLEtBQW1CLENBQXRCLEVBQXlCO0FBQ3ZCLFlBQU0yQixLQUFLLENBQUMsc0JBQUQsQ0FBWDtBQUNEO0FBQ0Y7O0FBRUQsU0FBTztBQUNMTSxJQUFBQSxJQUFJLEVBQUUsSUFBSVQsTUFBSixDQUFXZ0UsT0FBTyxDQUFDLENBQUQsQ0FBbEIsRUFBdUIsUUFBdkIsQ0FERDtBQUVMaEQsSUFBQUEsSUFBSSxFQUFFZ0QsT0FBTyxDQUFDLENBQUQ7QUFGUixHQUFQO0FBSUQsQ0FuQk0iLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIENvcHlyaWdodCAoYykgMjAxOS1QcmVzZW50LCBOaXRyb2dlbiBMYWJzLCBJbmMuXG4gKiBDb3B5cmlnaHRzIGxpY2Vuc2VkIHVuZGVyIHRoZSBNSVQgTGljZW5zZS4gU2VlIHRoZSBhY2NvbXBhbnlpbmcgTElDRU5TRSBmaWxlIGZvciB0ZXJtcy5cbiAqL1xuaW1wb3J0IHtnZXQgYXMgaHR0cEdldH0gZnJvbSAnQG5sYWJzL3JpcC1odW50ZXInO1xuaW1wb3J0IHtjcmVhdGVIYXNoLCBwYXJzZUlkLCBwYXJzZVN0cmluZ30gZnJvbSAnQG5sYWJzL3V0aWxzJztcbmltcG9ydCB7YXFsLCBEYXRhYmFzZX0gZnJvbSAnYXJhbmdvanMnO1xuaW1wb3J0IHtBcWxRdWVyeX0gZnJvbSAnYXJhbmdvanMvbGliL2Nqcy9hcWwtcXVlcnknO1xuaW1wb3J0IHtBcnJheUN1cnNvcn0gZnJvbSAnYXJhbmdvanMvbGliL2Nqcy9jdXJzb3InO1xuaW1wb3J0IHtnb29nbGV9IGZyb20gJ2dvb2dsZWFwaXMnO1xuaW1wb3J0IHtVc2VyRXJyb3J9IGZyb20gJ2dyYXBocWwtZXJyb3JzJztcbmltcG9ydCAqIGFzIHJlcXVlc3QgZnJvbSAncmVxdWVzdC1wcm9taXNlJztcbmltcG9ydCB7QXBpQ29udGV4dH0gZnJvbSAndHlwZXMvYXV0aCc7XG5cbmltcG9ydCB7Q29uZmlnfSBmcm9tICcuLi9jb25maWcnO1xuaW1wb3J0IHtGaWxlVHlwZX0gZnJvbSAnLi4vdHlwZXMvZmlsZXMnO1xuaW1wb3J0IHt1c2VEYn0gZnJvbSAnLi4vdXRpbHMnO1xuaW1wb3J0IHtyZXNpemVTYXZlSW1hZ2V9IGZyb20gJy4vaW1hZ2VzJztcbmltcG9ydCB7Y3JlYXRlUG9zdEVkZ2V9IGZyb20gJy4vcG9zdHMnO1xuXG5jb25zdCB5b3V0dWJlID0gZ29vZ2xlLnlvdXR1YmUoe2F1dGg6IENvbmZpZy5nZXQoJ2dvb2dsZS5rZXknKSwgdmVyc2lvbjogJ3YzJ30pO1xucmVxdWVzdC5kZWZhdWx0cyh7ZW5jb2Rpbmc6IG51bGx9KTtcblxuLy8gY29uc3QgZXZlbnRDYXRlZ29yeTogc3RyaW5nID0gJ2ZpbGVzJztcblxuLy8gVXBsb2FkIGZpbGVcbmV4cG9ydCBjb25zdCBhZGRGaWxlID0gKGNvbnRleHQ6IEFwaUNvbnRleHQsIGl0ZW06IEZpbGVUeXBlID0ge30pOiBQcm9taXNlPEZpbGVUeXBlPiA9PiB7XG4gIGNvbnN0IHtkYXRhYmFzZSwgdXNlcklkOiBzZXNzaW9uSWQsIHVzZXJUeXBlfSA9IGNvbnRleHQ7XG4gIGNvbnN0IHtcbiAgICBkZXNjcmlwdGlvbiA9ICcnLFxuICAgIGZpbGVUeXBlID0gJycsXG4gICAgaWQsXG4gICAgbmFtZSA9ICcnLFxuICAgIHVybCA9ICcnXG4gIH0gPSBpdGVtO1xuXG4gIC8vIElkXG4gIGNvbnN0IGZpbGVJZDogc3RyaW5nID0gaWQgPyBwYXJzZUlkKGlkKSA6IGNyZWF0ZUhhc2goYGZpbGUtJHtzZXNzaW9uSWR9YCk7XG5cbiAgLy8gTmFtZVxuICBjb25zdCBpc1VybDogYm9vbGVhbiA9IHVybCAhPT0gJyc7XG5cbiAgLy8gSWYgbm8gbmFtZSwgZ2V0IGl0IGZyb20gdXJsIHBhdGhcbiAgbGV0IGZvcm1hdE5hbWU6IHN0cmluZyA9IHBhcnNlU3RyaW5nKG5hbWUsIDE2MCk7XG4gIGxldCBmb3JtYXRUeXBlOiBzdHJpbmcgPSBwYXJzZVN0cmluZyhmaWxlVHlwZSwgMTYpO1xuXG4gIGlmKGZvcm1hdE5hbWUgPT09ICcnICYmIGlzVXJsKSB7XG4gICAgZm9ybWF0TmFtZSA9IHVybC5zdWJzdHJpbmcodXJsLmxhc3RJbmRleE9mKCcvJykgKyAxKTtcbiAgfVxuXG4gIGlmKGZvcm1hdFR5cGUgPT09ICcnKSB7XG4gICAgY29uc3QgbmFtZUFycjogc3RyaW5nW10gPSBmb3JtYXROYW1lLnNwbGl0KCcuJyk7XG4gICAgY29uc3QgZXh0OiBzdHJpbmcgPSBuYW1lQXJyW25hbWVBcnIubGVuZ3RoIC0gMV07XG5cbiAgICBzd2l0Y2goZXh0KSB7XG4gICAgICBjYXNlICdqcGVnJzpcbiAgICAgIGNhc2UgJ2pwZyc6XG4gICAgICAgIGZvcm1hdFR5cGUgPSAnaW1hZ2UvanBlZyc7XG4gICAgICAgIGJyZWFrO1xuICAgICAgY2FzZSAncG5nJzpcbiAgICAgICAgZm9ybWF0VHlwZSA9ICdpbWFnZS9wbmcnO1xuICAgICAgICBicmVhaztcbiAgICAgIGNhc2UgJ3ppcCc6XG4gICAgICAgIGZvcm1hdFR5cGUgPSAnYXBwbGljYXRpb24vemlwJztcbiAgICAgICAgYnJlYWs7XG4gICAgICBkZWZhdWx0OlxuICAgICAgICBicmVhaztcbiAgICB9XG4gIH1cblxuICBsZXQgaXNJbWFnZTogYm9vbGVhbjtcblxuICBzd2l0Y2goZm9ybWF0VHlwZSkge1xuICAgIGNhc2UgJ2ltYWdlL2pwZWcnOlxuICAgIGNhc2UgJ2ltYWdlL3BuZyc6XG4gICAgICBpc0ltYWdlID0gdHJ1ZTtcbiAgICAgIGJyZWFrO1xuICAgIGRlZmF1bHQ6XG4gICAgICBpc0ltYWdlID0gZmFsc2U7XG4gICAgICBicmVhaztcbiAgfVxuXG4gIC8vIERlc2NyaXB0aW9uXG4gIGNvbnN0IGZvcm1hdERlc2M6IHN0cmluZyA9IHBhcnNlU3RyaW5nKGRlc2NyaXB0aW9uLCA1MDApO1xuXG4gIC8vIE9ubHkgYWxsb3cgZmlsZSB1cGxvYWRzIHRvIHByZW1pdW0gdXNlcnNcbiAgaWYoIWlzSW1hZ2UgJiYgdXNlclR5cGUgIT09IDIpIHtcbiAgICB0aHJvdyBuZXcgVXNlckVycm9yKCdhY2NvdW50X3Jlc3RyaWN0aW9uJyk7XG4gIH1cblxuICBjb25zdCBzYXZlVG9EYiA9IChpbnNlcnQ6IEZpbGVUeXBlKSA9PiB7XG4gICAgY29uc3QgYXFsUXJ5OiBBcWxRdWVyeSA9IGFxbGBJTlNFUlQgJHtpbnNlcnR9IElOIGZpbGVzIFJFVFVSTiBORVdgO1xuXG4gICAgcmV0dXJuIHVzZURiKGRhdGFiYXNlKS5xdWVyeShhcWxRcnkpXG4gICAgICAudGhlbigoY3Vyc29yOiBBcnJheUN1cnNvcikgPT4gY3Vyc29yLm5leHQoKSlcbiAgICAgIC50aGVuKChmaWxlID0ge30pID0+IGZpbGUpXG4gICAgICAuY2F0Y2goKGVycm9yOiBFcnJvcikgPT4ge1xuICAgICAgICB0aHJvdyBlcnJvcjtcbiAgICAgIH0pO1xuICB9O1xuXG4gIGNvbnN0IHVwbG9hZEZpbGUgPSAoYnVmOiBCdWZmZXIsIHVwbG9hZFR5cGU6IHN0cmluZykgPT4ge1xuICAgIGNvbnN0IG5vdzogbnVtYmVyID0gRGF0ZS5ub3coKTtcblxuICAgIC8vIElmIGltYWdlLCByZXNpemUgYW5kIGNyZWF0ZSBhIHRodW1ibmFpbFxuICAgIGlmKGlzSW1hZ2UpIHtcbiAgICAgIHJldHVybiByZXNpemVTYXZlSW1hZ2Uoc2Vzc2lvbklkLCBmaWxlSWQsIGJ1ZiwgdXBsb2FkVHlwZSlcbiAgICAgICAgLnRoZW4oKHJlc2l6ZWRJbWFnZTogRmlsZVR5cGUpID0+IHtcbiAgICAgICAgICBjb25zdCBpbnNlcnQ6IEZpbGVUeXBlID0ge1xuICAgICAgICAgICAgLi4ucmVzaXplZEltYWdlLFxuICAgICAgICAgICAgX2tleTogZmlsZUlkLFxuICAgICAgICAgICAgYWRkZWQ6IG5vdyxcbiAgICAgICAgICAgIGRlc2NyaXB0aW9uOiBmb3JtYXREZXNjLFxuICAgICAgICAgICAgZmlsZVR5cGU6IGZvcm1hdFR5cGUsXG4gICAgICAgICAgICBtb2RpZmllZDogbm93LFxuICAgICAgICAgICAgbmFtZTogZm9ybWF0TmFtZSxcbiAgICAgICAgICAgIHVzZXJJZDogc2Vzc2lvbklkXG4gICAgICAgICAgfTtcblxuICAgICAgICAgIHJldHVybiBzYXZlVG9EYihpbnNlcnQpO1xuICAgICAgICB9KVxuICAgICAgICAuY2F0Y2goKGVycm9yOiBFcnJvcikgPT4ge1xuICAgICAgICAgIHRocm93IGVycm9yO1xuICAgICAgICB9KTtcbiAgICB9XG4gICAgY29uc3QgaW5zZXJ0OiBGaWxlVHlwZSA9IHtcbiAgICAgIF9rZXk6IGZpbGVJZCxcbiAgICAgIGFkZGVkOiBub3csXG4gICAgICBkZXNjcmlwdGlvbjogZm9ybWF0RGVzYyxcbiAgICAgIGZpbGVUeXBlOiBmb3JtYXRUeXBlLFxuICAgICAgbW9kaWZpZWQ6IG5vdyxcbiAgICAgIG5hbWU6IGZvcm1hdE5hbWUsXG4gICAgICB1c2VySWQ6IHNlc3Npb25JZFxuICAgIH07XG5cbiAgICByZXR1cm4gc2F2ZVRvRGIoaW5zZXJ0KTtcbiAgfTtcblxuICAvLyBJZiBmaWxlIGlzIGEgdXJsIHBhdGgsIGRvd25sb2FkIHRoZSBmaWxlIGFuZCBzYXZlXG4gIGlmKGlzVXJsKSB7XG4gICAgcmV0dXJuIHJlcXVlc3QuZ2V0KHtlbmNvZGluZzogbnVsbCwgdXJpOiB1cmx9KVxuICAgICAgLnRoZW4oKGJvZHkpID0+IHVwbG9hZEZpbGUobmV3IEJ1ZmZlcihib2R5LCAnYmluYXJ5JyksIGZvcm1hdFR5cGUpKVxuICAgICAgLmNhdGNoKCgpID0+IHtcbiAgICAgICAgdGhyb3cgbmV3IFVzZXJFcnJvcignZmlsZV9yZXF1ZXN0Jyk7XG4gICAgICB9KTtcbiAgfSBlbHNlIGlmKGl0ZW0uYmFzZTY0ICE9PSAnJykge1xuICAgIGNvbnN0IGJ1ZmZlcjogQnVmZmVyID0gbmV3IEJ1ZmZlcihpdGVtLmJhc2U2NCk7XG4gICAgcmV0dXJuIHVwbG9hZEZpbGUoYnVmZmVyLCBmb3JtYXRUeXBlKTtcbiAgfVxuICB0aHJvdyBuZXcgRXJyb3IoJ2ZpbGVfcmVxdWlyZWQnKTtcbn07XG5cbi8vIEdpcGh5XG5leHBvcnQgY29uc3QgZ2V0R2lwaHlUcmVuZHMgPSAoY29udGV4dDogQXBpQ29udGV4dCwgbGltaXQ6IG51bWJlciA9IDMwKTogUHJvbWlzZTxhbnlbXT4gPT4ge1xuICBjb25zdCBnaWZVcmw6IHN0cmluZyA9IGBodHRwOi8vYXBpLmdpcGh5LmNvbS92MS9naWZzL3RyZW5kaW5nP2FwaV9rZXk9JHtDb25maWcuZ2V0KCdnaXBoeS5rZXknKX0mbGltaXQ9JHtsaW1pdH1gO1xuXG4gIHJldHVybiBodHRwR2V0KGdpZlVybClcbiAgICAudGhlbigocmVzOiBSZXNwb25zZSkgPT4gcmVzLmpzb24oKSlcbiAgICAudGhlbigoanNvbikgPT4ganNvbi5kYXRhLm1hcCgoZ2lmSW1hZ2UgPSB7aWQ6IG51bGwsIGltYWdlczogbnVsbH0pID0+IHtcbiAgICAgIGNvbnN0IHtcbiAgICAgICAgaWQsXG4gICAgICAgIGltYWdlczoge1xuICAgICAgICAgIG9yaWdpbmFsOiB7dXJsID0gJyd9ID0ge30sXG4gICAgICAgICAgZml4ZWRfaGVpZ2h0X3NtYWxsOiB7dXJsOiB0aHVtYiA9ICcnfSA9IHt9XG4gICAgICAgIH0gPSB7fVxuICAgICAgfSA9IGdpZkltYWdlO1xuXG4gICAgICByZXR1cm4ge1xuICAgICAgICBpZCxcbiAgICAgICAgdGh1bWIsXG4gICAgICAgIHR5cGU6ICdnaXBoeScsXG4gICAgICAgIHVybFxuICAgICAgfTtcbiAgICB9KSk7XG59O1xuXG5leHBvcnQgY29uc3QgZ2V0R2lwaHlTZWFyY2ggPSAoY29udGV4dDogQXBpQ29udGV4dCwgcXVlcnk6IHN0cmluZywgbGltaXQ6IG51bWJlciA9IDMwKTogUHJvbWlzZTxhbnlbXT4gPT4ge1xuICBjb25zdCBmb3JtYXRRdWVyeTogc3RyaW5nID0gZW5jb2RlVVJJKHF1ZXJ5KTtcbiAgY29uc3QgZ2lmVXJsOiBzdHJpbmcgPSBgaHR0cDovL2FwaS5naXBoeS5jb20vdjEvZ2lmcy9zZWFyY2g/cT0ke2Zvcm1hdFF1ZXJ5fSZhcGlfa2V5PSR7Q29uZmlnLmdldCgnZ2lwaHkua2V5Jyl9JmxpbWl0PSR7bGltaXR9YDtcblxuICByZXR1cm4gZmV0Y2goZ2lmVXJsKVxuICAgIC50aGVuKChyZXM6IFJlc3BvbnNlKSA9PiByZXMuanNvbigpKVxuICAgIC50aGVuKChqc29uKSA9PiBqc29uLmRhdGEubWFwKChnaWZJbWFnZSA9IHtpZDogbnVsbCwgaW1hZ2VzOiBudWxsfSkgPT4ge1xuICAgICAgY29uc3Qge1xuICAgICAgICBpZCxcbiAgICAgICAgaW1hZ2VzOiB7XG4gICAgICAgICAgb3JpZ2luYWw6IHt1cmwgPSAnJ30gPSB7fSxcbiAgICAgICAgICBmaXhlZF9oZWlnaHRfc21hbGw6IHt1cmw6IHRodW1iID0gJyd9ID0ge31cbiAgICAgICAgfSA9IHt9XG4gICAgICB9ID0gZ2lmSW1hZ2U7XG5cbiAgICAgIHJldHVybiB7XG4gICAgICAgIGlkLFxuICAgICAgICB0aHVtYixcbiAgICAgICAgdHlwZTogJ2dpcGh5JyxcbiAgICAgICAgdXJsXG4gICAgICB9O1xuICAgIH0pKTtcbn07XG5cbmV4cG9ydCBjb25zdCBnZXRZb3VUdWJlVHJlbmRzID0gKGNvbnRleHQ6IEFwaUNvbnRleHQsIGxpbWl0OiBudW1iZXIgPSAzMCk6IFByb21pc2U8YW55W10+ID0+IHtcbiAgcmV0dXJuIG5ldyBQcm9taXNlKChyZXNvbHZlLCByZWplY3QpID0+IHtcbiAgICB5b3V0dWJlLnZpZGVvcy5saXN0KHtcbiAgICAgIGNoYXJ0OiAnbW9zdFBvcHVsYXInLFxuICAgICAgbWF4UmVzdWx0czogbGltaXQsXG4gICAgICBwYXJ0OiAnc25pcHBldCcsXG4gICAgICByZWdpb25Db2RlOiAnVVMnXG4gICAgfSwgKGVycm9yOiBFcnJvciwgZGF0YTogYW55KSA9PiB7XG4gICAgICBpZihlcnJvcikge1xuICAgICAgICBjb25zb2xlLmVycm9yKGVycm9yKTtcbiAgICAgICAgcmVqZWN0KG5ldyBFcnJvcihlcnJvclswXS5tZXNzYWdlKSk7XG4gICAgICB9IGVsc2UgaWYoZGF0YSkge1xuICAgICAgICBjb25zdCBsaXN0ID0gZGF0YS5pdGVtcy5tYXAoKGl0ZW0pID0+ICh7XG4gICAgICAgICAgaWQ6IGl0ZW0uaWQsXG4gICAgICAgICAgdGh1bWI6IGl0ZW0uc25pcHBldC50aHVtYm5haWxzLmhpZ2gudXJsLFxuICAgICAgICAgIHR5cGU6ICd5b3V0dWJlJyxcbiAgICAgICAgICB1cmw6IGBodHRwOi8vd3d3LnlvdXR1YmUuY29tL2VtYmVkLyR7aXRlbS5pZH1gXG4gICAgICAgIH0pKTtcblxuICAgICAgICByZXNvbHZlKGxpc3QpO1xuICAgICAgfVxuICAgIH0pO1xuICB9KTtcbn07XG5cbmV4cG9ydCBjb25zdCBnZXRZb3VUdWJlU2VhcmNoID0gKGNvbnRleHQ6IEFwaUNvbnRleHQsIHF1ZXJ5OiBzdHJpbmcsIGxpbWl0OiBudW1iZXIgPSAzMCk6IFByb21pc2U8YW55W10+ID0+IHtcbiAgcmV0dXJuIG5ldyBQcm9taXNlKChyZXNvbHZlLCByZWplY3QpID0+IHtcbiAgICB5b3V0dWJlLnNlYXJjaC5saXN0KHtcbiAgICAgIG1heFJlc3VsdHM6IGxpbWl0LFxuICAgICAgcGFydDogJ3NuaXBwZXQnLFxuICAgICAgcTogcXVlcnksXG4gICAgICByZWdpb25Db2RlOiAnVVMnXG4gICAgfSwgKGVycm9yOiBFcnJvciwgZGF0YTogYW55KSA9PiB7XG4gICAgICBpZihlcnJvcikge1xuICAgICAgICBjb25zb2xlLmVycm9yKGVycm9yKTtcbiAgICAgICAgcmVqZWN0KG5ldyBFcnJvcihlcnJvclswXS5tZXNzYWdlKSk7XG4gICAgICB9IGVsc2UgaWYoZGF0YSkge1xuICAgICAgICBjb25zdCB7aXRlbXN9ID0gZGF0YTtcbiAgICAgICAgY29uc3QgbGlzdCA9IGl0ZW1zLm1hcCgoaXRlbSkgPT4gKHtcbiAgICAgICAgICBpZDogaXRlbS5pZCxcbiAgICAgICAgICB0aHVtYjogaXRlbS5zbmlwcGV0LnRodW1ibmFpbHMuaGlnaC51cmwsXG4gICAgICAgICAgdHlwZTogJ3lvdXR1YmUnLFxuICAgICAgICAgIHVybDogYGh0dHA6Ly93d3cueW91dHViZS5jb20vZW1iZWQvJHtpdGVtLmlkfWBcbiAgICAgICAgfSkpO1xuXG4gICAgICAgIHJlc29sdmUobGlzdCk7XG4gICAgICB9XG4gICAgfSk7XG4gIH0pO1xufTtcblxuLy8gRmlsZXNcbmV4cG9ydCBjb25zdCBnZXRQYXRoVXNlckZpbGVzID0gKHVzZXJJZDogc3RyaW5nLCBmaWxlbmFtZTogc3RyaW5nKTogc3RyaW5nID0+IHtcbiAgcmV0dXJuIGB1c2Vycy8ke3VzZXJJZH0vZmlsZXMvJHtmaWxlbmFtZX1gO1xufTtcblxuZXhwb3J0IGNvbnN0IGdldFVybFVzZXJGaWxlcyA9ICh1c2VySWQ6IHN0cmluZywgZmlsZW5hbWU6IHN0cmluZywgZGlyOiBzdHJpbmcgPSAnZmlsZXMnLCB0eXBlOiBzdHJpbmcgPSAncHJvZmlsZScpOiBzdHJpbmcgPT4ge1xuICBpZihmaWxlbmFtZSkge1xuICAgIHJldHVybiBgaHR0cHM6Ly9ib3guJHtDb25maWcuZ2V0KCdhcHAudXJsJyl9L3VzZXJzLyR7dXNlcklkfS8ke2Rpcn0vJHtmaWxlbmFtZX1gO1xuICB9XG5cbiAgaWYodHlwZSA9PT0gJ3Byb2ZpbGUnKSB7XG4gICAgcmV0dXJuIGBodHRwczovL2JveC4ke0NvbmZpZy5nZXQoJ2FwcC51cmwnKX0vZGVmYXVsdHMvdXNlcl9iay5qcGdgO1xuICB9XG5cbiAgcmV0dXJuIGBodHRwczovL2JveC4ke0NvbmZpZy5nZXQoJ2FwcC51cmwnKX0vZGVmYXVsdHMvdXNlcl93aC5qcGdgO1xufTtcblxuZXhwb3J0IGNvbnN0IGxpbmtGaWxlcyA9IChkYjogRGF0YWJhc2UsIGZpbGVzOiBGaWxlVHlwZVtdLCBwb3N0SWQ6IHN0cmluZyk6IFByb21pc2U8YW55PiA9PiB7XG4gIHJldHVybiBQcm9taXNlLmFsbChcbiAgICBmaWxlcy5tYXAoKGZpbGU6IEZpbGVUeXBlKSA9PiBjcmVhdGVGaWxlKGRiLCBmaWxlKVxuICAgICAgLnRoZW4oKGZpbGU6IEZpbGVUeXBlKSA9PiBjcmVhdGVQb3N0RWRnZShkYiwgZmlsZSwgcG9zdElkKSkpXG4gICk7XG59O1xuXG5leHBvcnQgY29uc3QgdXBkYXRlRmlsZXMgPSAoZGI6IERhdGFiYXNlLCBwb3N0SWQ6IHN0cmluZywgZmlsZXM6IEZpbGVUeXBlW10pOiBQcm9taXNlPGFueT4gPT4ge1xuICBjb25zdCBlZGdlQ29sbGVjdGlvbiA9IGRiLmVkZ2VDb2xsZWN0aW9uKCdpc1Bvc3RlZCcpO1xuXG4gIHJldHVybiBlZGdlQ29sbGVjdGlvbi5pbkVkZ2VzKHBvc3RJZClcbiAgICAudGhlbigoZWRnZXMpID0+IHtcbiAgICAgIGlmKGVkZ2VzLmxlbmd0aCkge1xuICAgICAgICAvLyBSZW1vdmUgbGlua2VkIGVkZ2VzXG4gICAgICAgIHJldHVybiBQcm9taXNlLmFsbChcbiAgICAgICAgICBlZGdlcy5tYXAoKGVkZ2UpID0+IHtcbiAgICAgICAgICAgIGNvbnN0IHtfa2V5OiBlZGdlS2V5fSA9IGVkZ2U7XG4gICAgICAgICAgICBjb25zdCBhcWxRcnk6IEFxbFF1ZXJ5ID0gYXFsYFJFTU9WRSB7X2tleToke2VkZ2VLZXl9fSBJTiBpc1Bvc3RlZGA7XG5cbiAgICAgICAgICAgIHJldHVybiBkYi5xdWVyeShhcWxRcnkpLmNhdGNoKChlcnJvcjogRXJyb3IpID0+IHtcbiAgICAgICAgICAgICAgdGhyb3cgZXJyb3I7XG4gICAgICAgICAgICB9KTtcbiAgICAgICAgICB9KSlcbiAgICAgICAgICAudGhlbigoKSA9PiB7XG4gICAgICAgICAgICBpZihmaWxlcy5sZW5ndGgpIHtcbiAgICAgICAgICAgICAgLy8gTGluayBmaWxlc1xuICAgICAgICAgICAgICByZXR1cm4gbGlua0ZpbGVzKGRiLCBmaWxlcywgcG9zdElkKS50aGVuKCgpID0+IGZpbGVzKTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgICAgIHJldHVybiBmaWxlcztcbiAgICAgICAgICB9KTtcbiAgICAgIH0gZWxzZSBpZihmaWxlcy5sZW5ndGgpIHtcbiAgICAgICAgLy8gTGluayBmaWxlc1xuICAgICAgICByZXR1cm4gbGlua0ZpbGVzKGRiLCBmaWxlcywgcG9zdElkKS50aGVuKCgpID0+IGZpbGVzKTtcbiAgICAgIH1cbiAgICAgIHJldHVybiBmaWxlcztcbiAgICB9KVxuICAgIC5jYXRjaCgoZXJyb3I6IEVycm9yKSA9PiB7XG4gICAgICB0aHJvdyBlcnJvcjtcbiAgICB9KTtcbn07XG5cbmV4cG9ydCBjb25zdCBjcmVhdGVGaWxlID0gKGRiOiBEYXRhYmFzZSwgZmlsZTogRmlsZVR5cGUpOiBQcm9taXNlPEZpbGVUeXBlPiA9PiB7XG4gIGNvbnN0IGluc2VydDogYW55ID0ge1xuICAgIF9rZXk6IGZpbGUuaWQsXG4gICAgYWRkZWQ6IERhdGUubm93KClcbiAgfTtcblxuICBjb25zdCBhcWxRcnk6IEFxbFF1ZXJ5ID0gYXFsYFVQU0VSVCB7X2tleTogJHtmaWxlLmlkfX1cbiAgICBJTlNFUlQgJHtpbnNlcnR9XG4gICAgVVBEQVRFIHt9XG4gICAgSU4gZmlsZXMgUkVUVVJOIE5FV2A7XG5cbiAgcmV0dXJuIGRiLnF1ZXJ5KGFxbFFyeSlcbiAgICAudGhlbigoY3Vyc29yOiBBcnJheUN1cnNvcikgPT4gY3Vyc29yLm5leHQoKSlcbiAgICAudGhlbigodXBkYXRlZEZpbGU6IEZpbGVUeXBlID0ge30pID0+IHVwZGF0ZWRGaWxlKVxuICAgIC5jYXRjaCgoZXJyb3I6IEVycm9yKSA9PiB7XG4gICAgICB0aHJvdyBlcnJvcjtcbiAgICB9KTtcbn07XG5cbmV4cG9ydCBjb25zdCBlbmNvZGVCYXNlNjQgPSAoYnVmZmVyOiBCdWZmZXIpOiBzdHJpbmcgPT4ge1xuICByZXR1cm4gbmV3IEJ1ZmZlcihidWZmZXIpLnRvU3RyaW5nKCdiYXNlNjQnKTtcbn1cblxuZXhwb3J0IGNvbnN0IGRlY29kZUJhc2U2NCA9IChkYXRhU3RyaW5nOiBzdHJpbmcpOiBvYmplY3QgPT4ge1xuICAvLyBjb25zdCBnZXREYXRhID0gKHN0cjogc3RyaW5nKSA9PiBzdHIubWF0Y2goL15kYXRhOihbQS1aYS16LStcXC9dKyk7YmFzZTY0LCguKykkLykgfHwgW107XG4gIGNvbnN0IGdldERhdGEgPSAoc3RyOiBzdHJpbmcpID0+IHN0ci5tYXRjaCgvXmRhdGE6KFtBLVphLXotKy9dKyk7YmFzZTY0LCguKykkLykgfHwgW107XG4gIGxldCBtYXRjaGVzID0gZ2V0RGF0YShkYXRhU3RyaW5nKTtcblxuICBpZihtYXRjaGVzLmxlbmd0aCAhPT0gMykge1xuICAgIC8vIElmIGludmFsaWQgbWFrZSBzdXJlIHdlIGRvbid0IG5lZWQgdG8gZGVjb2RlXG4gICAgbWF0Y2hlcyA9IGdldERhdGEoZGVjb2RlVVJJQ29tcG9uZW50KGRhdGFTdHJpbmcpKTtcblxuICAgIC8vIENoZWNrIGl0IGFnYWluLlxuICAgIGlmKG1hdGNoZXMubGVuZ3RoICE9PSAzKSB7XG4gICAgICB0aHJvdyBFcnJvcignSW52YWxpZCBpbnB1dCBzdHJpbmcnKTtcbiAgICB9XG4gIH1cblxuICByZXR1cm4ge1xuICAgIGRhdGE6IG5ldyBCdWZmZXIobWF0Y2hlc1syXSwgJ2Jhc2U2NCcpLFxuICAgIHR5cGU6IG1hdGNoZXNbMV1cbiAgfTtcbn07XG4iXX0=
|