UNPKG

@google-cloud/firestore

Version:
326 lines 12.1 kB
"use strict"; /*! * Copyright 2017 Google Inc. All Rights Reserved. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ Object.defineProperty(exports, "__esModule", { value: true }); exports.primitiveComparator = primitiveComparator; exports.compareArrays = compareArrays; exports.compareUtf8Strings = compareUtf8Strings; exports.compare = compare; const convert_1 = require("./convert"); const path_1 = require("./path"); /*! * The type order as defined by the backend. */ var TypeOrder; (function (TypeOrder) { TypeOrder[TypeOrder["NULL"] = 0] = "NULL"; TypeOrder[TypeOrder["BOOLEAN"] = 1] = "BOOLEAN"; TypeOrder[TypeOrder["NUMBER"] = 2] = "NUMBER"; TypeOrder[TypeOrder["TIMESTAMP"] = 3] = "TIMESTAMP"; TypeOrder[TypeOrder["STRING"] = 4] = "STRING"; TypeOrder[TypeOrder["BLOB"] = 5] = "BLOB"; TypeOrder[TypeOrder["REF"] = 6] = "REF"; TypeOrder[TypeOrder["GEO_POINT"] = 7] = "GEO_POINT"; TypeOrder[TypeOrder["ARRAY"] = 8] = "ARRAY"; TypeOrder[TypeOrder["VECTOR"] = 9] = "VECTOR"; TypeOrder[TypeOrder["OBJECT"] = 10] = "OBJECT"; })(TypeOrder || (TypeOrder = {})); /*! * @private * @internal */ function typeOrder(val) { const valueType = (0, convert_1.detectValueType)(val); switch (valueType) { case 'nullValue': return TypeOrder.NULL; case 'integerValue': return TypeOrder.NUMBER; case 'doubleValue': return TypeOrder.NUMBER; case 'stringValue': return TypeOrder.STRING; case 'booleanValue': return TypeOrder.BOOLEAN; case 'arrayValue': return TypeOrder.ARRAY; case 'timestampValue': return TypeOrder.TIMESTAMP; case 'geoPointValue': return TypeOrder.GEO_POINT; case 'bytesValue': return TypeOrder.BLOB; case 'referenceValue': return TypeOrder.REF; case 'mapValue': return TypeOrder.OBJECT; case 'vectorValue': return TypeOrder.VECTOR; default: throw new Error('Unexpected value type: ' + valueType); } } /*! * @private * @internal */ function primitiveComparator(left, right) { if (left < right) { return -1; } if (left > right) { return 1; } return 0; } /*! * Utility function to compare doubles (using Firestore semantics for NaN). * @private * @internal */ function compareNumbers(left, right) { if (left < right) { return -1; } if (left > right) { return 1; } if (left === right) { return 0; } // one or both are NaN. if (isNaN(left)) { return isNaN(right) ? 0 : -1; } return 1; } /*! * @private * @internal */ function compareNumberProtos(left, right) { let leftValue, rightValue; if (left.integerValue !== undefined) { leftValue = Number(left.integerValue); } else { leftValue = Number(left.doubleValue); } if (right.integerValue !== undefined) { rightValue = Number(right.integerValue); } else { rightValue = Number(right.doubleValue); } return compareNumbers(leftValue, rightValue); } /*! * @private * @internal */ function compareTimestamps(left, right) { const seconds = primitiveComparator(left.seconds || 0, right.seconds || 0); if (seconds !== 0) { return seconds; } return primitiveComparator(left.nanos || 0, right.nanos || 0); } /*! * @private * @internal */ function compareBlobs(left, right) { if (!(left instanceof Buffer) || !(right instanceof Buffer)) { throw new Error('Blobs can only be compared if they are Buffers.'); } return Buffer.compare(left, right); } /*! * @private * @internal */ function compareReferenceProtos(left, right) { const leftPath = path_1.QualifiedResourcePath.fromSlashSeparatedString(left.referenceValue); const rightPath = path_1.QualifiedResourcePath.fromSlashSeparatedString(right.referenceValue); return leftPath.compareTo(rightPath); } /*! * @private * @internal */ function compareGeoPoints(left, right) { return (primitiveComparator(left.latitude || 0, right.latitude || 0) || primitiveComparator(left.longitude || 0, right.longitude || 0)); } /*! * @private * @internal */ function compareArrays(left, right) { for (let i = 0; i < left.length && i < right.length; i++) { const valueComparison = compare(left[i], right[i]); if (valueComparison !== 0) { return valueComparison; } } // If all the values matched so far, just check the length. return primitiveComparator(left.length, right.length); } /*! * @private * @internal */ function compareObjects(left, right) { // This requires iterating over the keys in the object in order and doing a // deep comparison. const leftKeys = Object.keys(left); const rightKeys = Object.keys(right); leftKeys.sort(); rightKeys.sort(); for (let i = 0; i < leftKeys.length && i < rightKeys.length; i++) { const keyComparison = compareUtf8Strings(leftKeys[i], rightKeys[i]); if (keyComparison !== 0) { return keyComparison; } const key = leftKeys[i]; const valueComparison = compare(left[key], right[key]); if (valueComparison !== 0) { return valueComparison; } } // If all the keys matched so far, just check the length. return primitiveComparator(leftKeys.length, rightKeys.length); } /*! * @private * @internal */ function compareVectors(left, right) { var _a, _b, _c, _d, _e, _f; // The vector is a map, but only vector value is compared. const leftArray = (_c = (_b = (_a = left === null || left === void 0 ? void 0 : left['value']) === null || _a === void 0 ? void 0 : _a.arrayValue) === null || _b === void 0 ? void 0 : _b.values) !== null && _c !== void 0 ? _c : []; const rightArray = (_f = (_e = (_d = right === null || right === void 0 ? void 0 : right['value']) === null || _d === void 0 ? void 0 : _d.arrayValue) === null || _e === void 0 ? void 0 : _e.values) !== null && _f !== void 0 ? _f : []; const lengthCompare = primitiveComparator(leftArray.length, rightArray.length); if (lengthCompare !== 0) { return lengthCompare; } return compareArrays(leftArray, rightArray); } /*! * Compare strings in UTF-8 encoded byte order * @private * @internal */ function compareUtf8Strings(left, right) { // Find the first differing character (a.k.a. "UTF-16 code unit") in the two strings and, // if found, use that character to determine the relative ordering of the two strings as a // whole. Comparing UTF-16 strings in UTF-8 byte order can be done simply and efficiently by // comparing the UTF-16 code units (chars). This serendipitously works because of the way UTF-8 // and UTF-16 happen to represent Unicode code points. // // After finding the first pair of differing characters, there are two cases: // // Case 1: Both characters are non-surrogates (code points less than or equal to 0xFFFF) or // both are surrogates from a surrogate pair (that collectively represent code points greater // than 0xFFFF). In this case their numeric order as UTF-16 code units is the same as the // lexicographical order of their corresponding UTF-8 byte sequences. A direct comparison is // sufficient. // // Case 2: One character is a surrogate and the other is not. In this case the surrogate- // containing string is always ordered after the non-surrogate. This is because surrogates are // used to represent code points greater than 0xFFFF which have 4-byte UTF-8 representations // and are lexicographically greater than the 1, 2, or 3-byte representations of code points // less than or equal to 0xFFFF. // // An example of why Case 2 is required is comparing the following two Unicode code points: // // |-----------------------|------------|---------------------|-----------------| // | Name | Code Point | UTF-8 Encoding | UTF-16 Encoding | // |-----------------------|------------|---------------------|-----------------| // | Replacement Character | U+FFFD | 0xEF 0xBF 0xBD | 0xFFFD | // | Grinning Face | U+1F600 | 0xF0 0x9F 0x98 0x80 | 0xD83D 0xDE00 | // |-----------------------|------------|---------------------|-----------------| // // A lexicographical comparison of the UTF-8 encodings of these code points would order // "Replacement Character" _before_ "Grinning Face" because 0xEF is less than 0xF0. However, a // direct comparison of the UTF-16 code units, as would be done in case 1, would erroneously // produce the _opposite_ ordering, because 0xFFFD is _greater than_ 0xD83D. As it turns out, // this relative ordering holds for all comparisons of UTF-16 code points requiring a surrogate // pair with those that do not. const length = Math.min(left.length, right.length); for (let i = 0; i < length; i++) { const leftChar = left.charAt(i); const rightChar = right.charAt(i); if (leftChar !== rightChar) { return isSurrogate(leftChar) === isSurrogate(rightChar) ? primitiveComparator(leftChar, rightChar) : isSurrogate(leftChar) ? 1 : -1; } } // Use the lengths of the strings to determine the overall comparison result since either the // strings were equal or one is a prefix of the other. return primitiveComparator(left.length, right.length); } const MIN_SURROGATE = 0xd800; const MAX_SURROGATE = 0xdfff; function isSurrogate(s) { const c = s.charCodeAt(0); return c >= MIN_SURROGATE && c <= MAX_SURROGATE; } /*! * @private * @internal */ function compare(left, right) { // First compare the types. const leftType = typeOrder(left); const rightType = typeOrder(right); const typeComparison = primitiveComparator(leftType, rightType); if (typeComparison !== 0) { return typeComparison; } // So they are the same type. switch (leftType) { case TypeOrder.NULL: // Nulls are all equal. return 0; case TypeOrder.BOOLEAN: return primitiveComparator(left.booleanValue, right.booleanValue); case TypeOrder.STRING: return compareUtf8Strings(left.stringValue, right.stringValue); case TypeOrder.NUMBER: return compareNumberProtos(left, right); case TypeOrder.TIMESTAMP: return compareTimestamps(left.timestampValue, right.timestampValue); case TypeOrder.BLOB: return compareBlobs(left.bytesValue, right.bytesValue); case TypeOrder.REF: return compareReferenceProtos(left, right); case TypeOrder.GEO_POINT: return compareGeoPoints(left.geoPointValue, right.geoPointValue); case TypeOrder.ARRAY: return compareArrays(left.arrayValue.values || [], right.arrayValue.values || []); case TypeOrder.OBJECT: return compareObjects(left.mapValue.fields || {}, right.mapValue.fields || {}); case TypeOrder.VECTOR: return compareVectors(left.mapValue.fields || {}, right.mapValue.fields || {}); default: throw new Error(`Encountered unknown type order: ${leftType}`); } } //# sourceMappingURL=order.js.map