123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648 |
- "use strict";
- var _sliceInstanceProperty2 = require("@babel/runtime-corejs3/core-js-stable/instance/slice");
- var _Array$from = require("@babel/runtime-corejs3/core-js-stable/array/from");
- var _Symbol = require("@babel/runtime-corejs3/core-js-stable/symbol");
- var _getIteratorMethod = require("@babel/runtime-corejs3/core-js/get-iterator-method");
- var _Array$isArray2 = require("@babel/runtime-corejs3/core-js-stable/array/is-array");
- var _interopRequireDefault = require("@babel/runtime-corejs3/helpers/interopRequireDefault");
- var _typeof2 = _interopRequireDefault(require("@babel/runtime-corejs3/helpers/typeof"));
- var _slicedToArray2 = _interopRequireDefault(require("@babel/runtime-corejs3/helpers/slicedToArray"));
- var _isArray = _interopRequireDefault(require("@babel/runtime-corejs3/core-js-stable/array/is-array"));
- var _indexOf = _interopRequireDefault(require("@babel/runtime-corejs3/core-js-stable/instance/index-of"));
- var _filter = _interopRequireDefault(require("@babel/runtime-corejs3/core-js-stable/instance/filter"));
- var _slice = _interopRequireDefault(require("@babel/runtime-corejs3/core-js-stable/instance/slice"));
- var _isInteger = _interopRequireDefault(require("@babel/runtime-corejs3/core-js-stable/number/is-integer"));
- var _concat = _interopRequireDefault(require("@babel/runtime-corejs3/core-js-stable/instance/concat"));
- var _map = _interopRequireDefault(require("@babel/runtime-corejs3/core-js-stable/instance/map"));
- var _forEach = _interopRequireDefault(require("@babel/runtime-corejs3/core-js-stable/instance/for-each"));
- var _keys = _interopRequireDefault(require("@babel/runtime-corejs3/core-js-stable/object/keys"));
- function _createForOfIteratorHelper(o, allowArrayLike) {
- var it = typeof _Symbol !== "undefined" && _getIteratorMethod(o) || o["@@iterator"];
- if (!it) {
- if (_Array$isArray2(o) || (it = _unsupportedIterableToArray(o)) || allowArrayLike && o && typeof o.length === "number") {
- if (it) o = it;
- var i = 0;
- var F = function () {};
- return {
- s: F,
- n: function () {
- if (i >= o.length) return {
- done: true
- };
- return {
- done: false,
- value: o[i++]
- };
- },
- e: function (_e) {
- throw _e;
- },
- f: F
- };
- }
- throw new TypeError("Invalid attempt to iterate non-iterable instance.\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method.");
- }
- var normalCompletion = true,
- didErr = false,
- err;
- return {
- s: function () {
- it = it.call(o);
- },
- n: function () {
- var step = it.next();
- normalCompletion = step.done;
- return step;
- },
- e: function (_e2) {
- didErr = true;
- err = _e2;
- },
- f: function () {
- try {
- if (!normalCompletion && it.return != null) it.return();
- } finally {
- if (didErr) throw err;
- }
- }
- };
- }
- function _unsupportedIterableToArray(o, minLen) {
- var _context6;
- if (!o) return;
- if (typeof o === "string") return _arrayLikeToArray(o, minLen);
- var n = _sliceInstanceProperty2(_context6 = Object.prototype.toString.call(o)).call(_context6, 8, -1);
- if (n === "Object" && o.constructor) n = o.constructor.name;
- if (n === "Map" || n === "Set") return _Array$from(o);
- if (n === "Arguments" || /^(?:Ui|I)nt(?:8|16|32)(?:Clamped)?Array$/.test(n)) return _arrayLikeToArray(o, minLen);
- }
- function _arrayLikeToArray(arr, len) {
- if (len == null || len > arr.length) len = arr.length;
- for (var i = 0, arr2 = new Array(len); i < len; i++) arr2[i] = arr[i];
- return arr2;
- }
- var equalObjects = require('./equals').default;
- var decode = require('./decode').default;
- var ParseError = require('./ParseError').default;
- var ParsePolygon = require('./ParsePolygon').default;
- var ParseGeoPoint = require('./ParseGeoPoint').default;
- /**
- * contains -- Determines if an object is contained in a list with special handling for Parse pointers.
- *
- * @param haystack
- * @param needle
- * @private
- * @returns {boolean}
- */
- function contains(haystack, needle) {
- if (needle && needle.__type && (needle.__type === 'Pointer' || needle.__type === 'Object')) {
- for (var i in haystack) {
- var ptr = haystack[i];
- if (typeof ptr === 'string' && ptr === needle.objectId) {
- return true;
- }
- if (ptr.className === needle.className && ptr.objectId === needle.objectId) {
- return true;
- }
- }
- return false;
- }
- if ((0, _isArray.default)(needle)) {
- var _iterator = _createForOfIteratorHelper(needle),
- _step;
- try {
- for (_iterator.s(); !(_step = _iterator.n()).done;) {
- var need = _step.value;
- if (contains(haystack, need)) {
- return true;
- }
- }
- } catch (err) {
- _iterator.e(err);
- } finally {
- _iterator.f();
- }
- }
- return (0, _indexOf.default)(haystack).call(haystack, needle) > -1;
- }
- function transformObject(object) {
- if (object._toFullJSON) {
- return object._toFullJSON();
- }
- return object;
- }
- /**
- * matchesQuery -- Determines if an object would be returned by a Parse Query
- * It's a lightweight, where-clause only implementation of a full query engine.
- * Since we find queries that match objects, rather than objects that match
- * queries, we can avoid building a full-blown query tool.
- *
- * @param className
- * @param object
- * @param objects
- * @param query
- * @private
- * @returns {boolean}
- */
- function matchesQuery(className, object, objects, query) {
- if (object.className !== className) {
- return false;
- }
- var obj = object;
- var q = query;
- if (object.toJSON) {
- obj = object.toJSON();
- }
- if (query.toJSON) {
- q = query.toJSON().where;
- }
- obj.className = className;
- for (var field in q) {
- if (!matchesKeyConstraints(className, obj, objects, field, q[field])) {
- return false;
- }
- }
- return true;
- }
- function equalObjectsGeneric(obj, compareTo, eqlFn) {
- if ((0, _isArray.default)(obj)) {
- for (var i = 0; i < obj.length; i++) {
- if (eqlFn(obj[i], compareTo)) {
- return true;
- }
- }
- return false;
- }
- return eqlFn(obj, compareTo);
- }
- /**
- * @typedef RelativeTimeToDateResult
- * @property {string} status The conversion status, `error` if conversion failed or
- * `success` if conversion succeeded.
- * @property {string} info The error message if conversion failed, or the relative
- * time indication (`past`, `present`, `future`) if conversion succeeded.
- * @property {Date|undefined} result The converted date, or `undefined` if conversion
- * failed.
- */
- /**
- * Converts human readable relative date string, for example, 'in 10 days' to a date
- * relative to now.
- *
- * @param {string} text The text to convert.
- * @param {Date} [now=new Date()] The date from which add or subtract. Default is now.
- * @returns {RelativeTimeToDateResult}
- */
- function relativeTimeToDate(text) {
- var now = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : new Date();
- text = text.toLowerCase();
- var parts = text.split(' ');
- // Filter out whitespace
- parts = (0, _filter.default)(parts).call(parts, function (part) {
- return part !== '';
- });
- var future = parts[0] === 'in';
- var past = parts[parts.length - 1] === 'ago';
- if (!future && !past && text !== 'now') {
- return {
- status: 'error',
- info: "Time should either start with 'in' or end with 'ago'"
- };
- }
- if (future && past) {
- return {
- status: 'error',
- info: "Time cannot have both 'in' and 'ago'"
- };
- }
- // strip the 'ago' or 'in'
- if (future) {
- parts = (0, _slice.default)(parts).call(parts, 1);
- } else {
- // past
- parts = (0, _slice.default)(parts).call(parts, 0, parts.length - 1);
- }
- if (parts.length % 2 !== 0 && text !== 'now') {
- return {
- status: 'error',
- info: 'Invalid time string. Dangling unit or number.'
- };
- }
- var pairs = [];
- while (parts.length) {
- pairs.push([parts.shift(), parts.shift()]);
- }
- var seconds = 0;
- for (var _i = 0, _pairs = pairs; _i < _pairs.length; _i++) {
- var _pairs$_i = (0, _slicedToArray2.default)(_pairs[_i], 2),
- num = _pairs$_i[0],
- interval = _pairs$_i[1];
- var val = Number(num);
- if (!(0, _isInteger.default)(val)) {
- return {
- status: 'error',
- info: "'".concat(num, "' is not an integer.")
- };
- }
- switch (interval) {
- case 'yr':
- case 'yrs':
- case 'year':
- case 'years':
- seconds += val * 31536000; // 365 * 24 * 60 * 60
- break;
- case 'wk':
- case 'wks':
- case 'week':
- case 'weeks':
- seconds += val * 604800; // 7 * 24 * 60 * 60
- break;
- case 'd':
- case 'day':
- case 'days':
- seconds += val * 86400; // 24 * 60 * 60
- break;
- case 'hr':
- case 'hrs':
- case 'hour':
- case 'hours':
- seconds += val * 3600; // 60 * 60
- break;
- case 'min':
- case 'mins':
- case 'minute':
- case 'minutes':
- seconds += val * 60;
- break;
- case 'sec':
- case 'secs':
- case 'second':
- case 'seconds':
- seconds += val;
- break;
- default:
- return {
- status: 'error',
- info: "Invalid interval: '".concat(interval, "'")
- };
- }
- }
- var milliseconds = seconds * 1000;
- if (future) {
- return {
- status: 'success',
- info: 'future',
- result: new Date(now.valueOf() + milliseconds)
- };
- } else if (past) {
- return {
- status: 'success',
- info: 'past',
- result: new Date(now.valueOf() - milliseconds)
- };
- } else {
- return {
- status: 'success',
- info: 'present',
- result: new Date(now.valueOf())
- };
- }
- }
- /**
- * Determines whether an object matches a single key's constraints
- *
- * @param className
- * @param object
- * @param objects
- * @param key
- * @param constraints
- * @private
- * @returns {boolean}
- */
- function matchesKeyConstraints(className, object, objects, key, constraints) {
- if (constraints === null) {
- return false;
- }
- if ((0, _indexOf.default)(key).call(key, '.') >= 0) {
- // Key references a subobject
- var keyComponents = key.split('.');
- var subObjectKey = keyComponents[0];
- var keyRemainder = (0, _slice.default)(keyComponents).call(keyComponents, 1).join('.');
- return matchesKeyConstraints(className, object[subObjectKey] || {}, objects, keyRemainder, constraints);
- }
- var i;
- if (key === '$or') {
- for (i = 0; i < constraints.length; i++) {
- if (matchesQuery(className, object, objects, constraints[i])) {
- return true;
- }
- }
- return false;
- }
- if (key === '$and') {
- for (i = 0; i < constraints.length; i++) {
- if (!matchesQuery(className, object, objects, constraints[i])) {
- return false;
- }
- }
- return true;
- }
- if (key === '$nor') {
- for (i = 0; i < constraints.length; i++) {
- if (matchesQuery(className, object, objects, constraints[i])) {
- return false;
- }
- }
- return true;
- }
- if (key === '$relatedTo') {
- // Bail! We can't handle relational queries locally
- return false;
- }
- if (!/^[A-Za-z][0-9A-Za-z_]*$/.test(key)) {
- throw new ParseError(ParseError.INVALID_KEY_NAME, "Invalid Key: ".concat(key));
- }
- // Equality (or Array contains) cases
- if ((0, _typeof2.default)(constraints) !== 'object') {
- if ((0, _isArray.default)(object[key])) {
- var _context;
- return (0, _indexOf.default)(_context = object[key]).call(_context, constraints) > -1;
- }
- return object[key] === constraints;
- }
- var compareTo;
- if (constraints.__type) {
- if (constraints.__type === 'Pointer') {
- return equalObjectsGeneric(object[key], constraints, function (obj, ptr) {
- return typeof obj !== 'undefined' && ptr.className === obj.className && ptr.objectId === obj.objectId;
- });
- }
- return equalObjectsGeneric(decode(object[key]), decode(constraints), equalObjects);
- }
- // More complex cases
- for (var condition in constraints) {
- compareTo = constraints[condition];
- if (compareTo.__type) {
- compareTo = decode(compareTo);
- }
- // is it a $relativeTime? convert to date
- if (compareTo['$relativeTime']) {
- var parserResult = relativeTimeToDate(compareTo['$relativeTime']);
- if (parserResult.status !== 'success') {
- var _context2;
- throw new ParseError(ParseError.INVALID_JSON, (0, _concat.default)(_context2 = "bad $relativeTime (".concat(key, ") value. ")).call(_context2, parserResult.info));
- }
- compareTo = parserResult.result;
- }
- // Compare Date Object or Date String
- if (toString.call(compareTo) === '[object Date]' || typeof compareTo === 'string' && new Date(compareTo) !== 'Invalid Date' && !isNaN(new Date(compareTo))) {
- object[key] = new Date(object[key].iso ? object[key].iso : object[key]);
- }
- switch (condition) {
- case '$lt':
- if (object[key] >= compareTo) {
- return false;
- }
- break;
- case '$lte':
- if (object[key] > compareTo) {
- return false;
- }
- break;
- case '$gt':
- if (object[key] <= compareTo) {
- return false;
- }
- break;
- case '$gte':
- if (object[key] < compareTo) {
- return false;
- }
- break;
- case '$ne':
- if (equalObjects(object[key], compareTo)) {
- return false;
- }
- break;
- case '$in':
- if (!contains(compareTo, object[key])) {
- return false;
- }
- break;
- case '$nin':
- if (contains(compareTo, object[key])) {
- return false;
- }
- break;
- case '$all':
- for (i = 0; i < compareTo.length; i++) {
- var _context3;
- if ((0, _indexOf.default)(_context3 = object[key]).call(_context3, compareTo[i]) < 0) {
- return false;
- }
- }
- break;
- case '$exists':
- {
- var propertyExists = typeof object[key] !== 'undefined';
- var existenceIsRequired = constraints['$exists'];
- if (typeof constraints['$exists'] !== 'boolean') {
- // The SDK will never submit a non-boolean for $exists, but if someone
- // tries to submit a non-boolean for $exits outside the SDKs, just ignore it.
- break;
- }
- if (!propertyExists && existenceIsRequired || propertyExists && !existenceIsRequired) {
- return false;
- }
- break;
- }
- case '$regex':
- {
- if ((0, _typeof2.default)(compareTo) === 'object') {
- return compareTo.test(object[key]);
- }
- // JS doesn't support perl-style escaping
- var expString = '';
- var escapeEnd = -2;
- var escapeStart = (0, _indexOf.default)(compareTo).call(compareTo, '\\Q');
- while (escapeStart > -1) {
- // Add the unescaped portion
- expString += compareTo.substring(escapeEnd + 2, escapeStart);
- escapeEnd = (0, _indexOf.default)(compareTo).call(compareTo, '\\E', escapeStart);
- if (escapeEnd > -1) {
- expString += compareTo.substring(escapeStart + 2, escapeEnd).replace(/\\\\\\\\E/g, '\\E').replace(/\W/g, '\\$&');
- }
- escapeStart = (0, _indexOf.default)(compareTo).call(compareTo, '\\Q', escapeEnd);
- }
- expString += compareTo.substring(Math.max(escapeStart, escapeEnd + 2));
- var modifiers = constraints.$options || '';
- modifiers = modifiers.replace('x', '').replace('s', '');
- // Parse Server / Mongo support x and s modifiers but JS RegExp doesn't
- var exp = new RegExp(expString, modifiers);
- if (!exp.test(object[key])) {
- return false;
- }
- break;
- }
- case '$nearSphere':
- {
- if (!compareTo || !object[key]) {
- return false;
- }
- var distance = compareTo.radiansTo(object[key]);
- var max = constraints.$maxDistance || Infinity;
- return distance <= max;
- }
- case '$within':
- {
- if (!compareTo || !object[key]) {
- return false;
- }
- var southWest = compareTo.$box[0];
- var northEast = compareTo.$box[1];
- if (southWest.latitude > northEast.latitude || southWest.longitude > northEast.longitude) {
- // Invalid box, crosses the date line
- return false;
- }
- return object[key].latitude > southWest.latitude && object[key].latitude < northEast.latitude && object[key].longitude > southWest.longitude && object[key].longitude < northEast.longitude;
- }
- case '$options':
- // Not a query type, but a way to add options to $regex. Ignore and
- // avoid the default
- break;
- case '$maxDistance':
- // Not a query type, but a way to add a cap to $nearSphere. Ignore and
- // avoid the default
- break;
- case '$select':
- {
- var subQueryObjects = (0, _filter.default)(objects).call(objects, function (obj, index, arr) {
- return matchesQuery(compareTo.query.className, obj, arr, compareTo.query.where);
- });
- for (var _i2 = 0; _i2 < subQueryObjects.length; _i2 += 1) {
- var subObject = transformObject(subQueryObjects[_i2]);
- return equalObjects(object[key], subObject[compareTo.key]);
- }
- return false;
- }
- case '$dontSelect':
- {
- var _subQueryObjects = (0, _filter.default)(objects).call(objects, function (obj, index, arr) {
- return matchesQuery(compareTo.query.className, obj, arr, compareTo.query.where);
- });
- for (var _i3 = 0; _i3 < _subQueryObjects.length; _i3 += 1) {
- var _subObject = transformObject(_subQueryObjects[_i3]);
- return !equalObjects(object[key], _subObject[compareTo.key]);
- }
- return false;
- }
- case '$inQuery':
- {
- var _subQueryObjects2 = (0, _filter.default)(objects).call(objects, function (obj, index, arr) {
- return matchesQuery(compareTo.className, obj, arr, compareTo.where);
- });
- for (var _i4 = 0; _i4 < _subQueryObjects2.length; _i4 += 1) {
- var _subObject2 = transformObject(_subQueryObjects2[_i4]);
- if (object[key].className === _subObject2.className && object[key].objectId === _subObject2.objectId) {
- return true;
- }
- }
- return false;
- }
- case '$notInQuery':
- {
- var _subQueryObjects3 = (0, _filter.default)(objects).call(objects, function (obj, index, arr) {
- return matchesQuery(compareTo.className, obj, arr, compareTo.where);
- });
- for (var _i5 = 0; _i5 < _subQueryObjects3.length; _i5 += 1) {
- var _subObject3 = transformObject(_subQueryObjects3[_i5]);
- if (object[key].className === _subObject3.className && object[key].objectId === _subObject3.objectId) {
- return false;
- }
- }
- return true;
- }
- case '$containedBy':
- {
- var _iterator2 = _createForOfIteratorHelper(object[key]),
- _step2;
- try {
- for (_iterator2.s(); !(_step2 = _iterator2.n()).done;) {
- var value = _step2.value;
- if (!contains(compareTo, value)) {
- return false;
- }
- }
- } catch (err) {
- _iterator2.e(err);
- } finally {
- _iterator2.f();
- }
- return true;
- }
- case '$geoWithin':
- {
- if (compareTo.$polygon) {
- var _context4;
- var points = (0, _map.default)(_context4 = compareTo.$polygon).call(_context4, function (geoPoint) {
- return [geoPoint.latitude, geoPoint.longitude];
- });
- var polygon = new ParsePolygon(points);
- return polygon.containsPoint(object[key]);
- }
- if (compareTo.$centerSphere) {
- var _compareTo$$centerSph = (0, _slicedToArray2.default)(compareTo.$centerSphere, 2),
- WGS84Point = _compareTo$$centerSph[0],
- maxDistance = _compareTo$$centerSph[1];
- var centerPoint = new ParseGeoPoint({
- latitude: WGS84Point[1],
- longitude: WGS84Point[0]
- });
- var point = new ParseGeoPoint(object[key]);
- var _distance = point.radiansTo(centerPoint);
- return _distance <= maxDistance;
- }
- break;
- }
- case '$geoIntersects':
- {
- var _polygon = new ParsePolygon(object[key].coordinates);
- var _point = new ParseGeoPoint(compareTo.$point);
- return _polygon.containsPoint(_point);
- }
- default:
- return false;
- }
- }
- return true;
- }
- function validateQuery(query /*: any*/) {
- var _context5;
- var q = query;
- if (query.toJSON) {
- q = query.toJSON().where;
- }
- var specialQuerykeys = ['$and', '$or', '$nor', '_rperm', '_wperm', '_perishable_token', '_email_verify_token', '_email_verify_token_expires_at', '_account_lockout_expires_at', '_failed_login_count'];
- (0, _forEach.default)(_context5 = (0, _keys.default)(q)).call(_context5, function (key) {
- if (q && q[key] && q[key].$regex) {
- if (typeof q[key].$options === 'string') {
- if (!q[key].$options.match(/^[imxs]+$/)) {
- throw new ParseError(ParseError.INVALID_QUERY, "Bad $options value for query: ".concat(q[key].$options));
- }
- }
- }
- if ((0, _indexOf.default)(specialQuerykeys).call(specialQuerykeys, key) < 0 && !key.match(/^[a-zA-Z][a-zA-Z0-9_\.]*$/)) {
- throw new ParseError(ParseError.INVALID_KEY_NAME, "Invalid key name: ".concat(key));
- }
- });
- }
- var OfflineQuery = {
- matchesQuery: matchesQuery,
- validateQuery: validateQuery
- };
- module.exports = OfflineQuery;
|