123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250 |
- 'use strict';
- const Buffer = require('buffer').Buffer;
- const Long = require('../long');
- const Double = require('../double');
- const Timestamp = require('../timestamp');
- const ObjectId = require('../objectid');
- const BSONSymbol = require('../symbol');
- const BSONRegExp = require('../regexp');
- const Code = require('../code');
- const Decimal128 = require('../decimal128');
- const MinKey = require('../min_key');
- const MaxKey = require('../max_key');
- const DBRef = require('../db_ref');
- const Binary = require('../binary');
- const normalizedFunctionString = require('./utils').normalizedFunctionString;
- const constants = require('../constants');
- // To ensure that 0.4 of node works correctly
- function isDate(d) {
- return typeof d === 'object' && Object.prototype.toString.call(d) === '[object Date]';
- }
- function calculateObjectSize(object, serializeFunctions, ignoreUndefined) {
- let totalLength = 4 + 1;
- if (Array.isArray(object)) {
- for (let i = 0; i < object.length; i++) {
- totalLength += calculateElement(
- i.toString(),
- object[i],
- serializeFunctions,
- true,
- ignoreUndefined
- );
- }
- } else {
- // If we have toBSON defined, override the current object
- if (object.toBSON) {
- object = object.toBSON();
- }
- // Calculate size
- for (let key in object) {
- totalLength += calculateElement(key, object[key], serializeFunctions, false, ignoreUndefined);
- }
- }
- return totalLength;
- }
- /**
- * @ignore
- * @api private
- */
- function calculateElement(name, value, serializeFunctions, isArray, ignoreUndefined) {
- // If we have toBSON defined, override the current object
- if (value && value.toBSON) {
- value = value.toBSON();
- }
- switch (typeof value) {
- case 'string':
- return 1 + Buffer.byteLength(name, 'utf8') + 1 + 4 + Buffer.byteLength(value, 'utf8') + 1;
- case 'number':
- if (
- Math.floor(value) === value &&
- value >= constants.JS_INT_MIN &&
- value <= constants.JS_INT_MAX
- ) {
- if (value >= constants.BSON_INT32_MIN && value <= constants.BSON_INT32_MAX) {
- // 32 bit
- return (name != null ? Buffer.byteLength(name, 'utf8') + 1 : 0) + (4 + 1);
- } else {
- return (name != null ? Buffer.byteLength(name, 'utf8') + 1 : 0) + (8 + 1);
- }
- } else {
- // 64 bit
- return (name != null ? Buffer.byteLength(name, 'utf8') + 1 : 0) + (8 + 1);
- }
- case 'undefined':
- if (isArray || !ignoreUndefined)
- return (name != null ? Buffer.byteLength(name, 'utf8') + 1 : 0) + 1;
- return 0;
- case 'boolean':
- return (name != null ? Buffer.byteLength(name, 'utf8') + 1 : 0) + (1 + 1);
- case 'object':
- if (
- value == null ||
- value instanceof MinKey ||
- value instanceof MaxKey ||
- value['_bsontype'] === 'MinKey' ||
- value['_bsontype'] === 'MaxKey'
- ) {
- return (name != null ? Buffer.byteLength(name, 'utf8') + 1 : 0) + 1;
- } else if (value instanceof ObjectId || value['_bsontype'] === 'ObjectId') {
- return (name != null ? Buffer.byteLength(name, 'utf8') + 1 : 0) + (12 + 1);
- } else if (value instanceof Date || isDate(value)) {
- return (name != null ? Buffer.byteLength(name, 'utf8') + 1 : 0) + (8 + 1);
- } else if (typeof Buffer !== 'undefined' && Buffer.isBuffer(value)) {
- return (
- (name != null ? Buffer.byteLength(name, 'utf8') + 1 : 0) + (1 + 4 + 1) + value.length
- );
- } else if (
- value instanceof Long ||
- value instanceof Double ||
- value instanceof Timestamp ||
- value['_bsontype'] === 'Long' ||
- value['_bsontype'] === 'Double' ||
- value['_bsontype'] === 'Timestamp'
- ) {
- return (name != null ? Buffer.byteLength(name, 'utf8') + 1 : 0) + (8 + 1);
- } else if (value instanceof Decimal128 || value['_bsontype'] === 'Decimal128') {
- return (name != null ? Buffer.byteLength(name, 'utf8') + 1 : 0) + (16 + 1);
- } else if (value instanceof Code || value['_bsontype'] === 'Code') {
- // Calculate size depending on the availability of a scope
- if (value.scope != null && Object.keys(value.scope).length > 0) {
- return (
- (name != null ? Buffer.byteLength(name, 'utf8') + 1 : 0) +
- 1 +
- 4 +
- 4 +
- Buffer.byteLength(value.code.toString(), 'utf8') +
- 1 +
- calculateObjectSize(value.scope, serializeFunctions, ignoreUndefined)
- );
- } else {
- return (
- (name != null ? Buffer.byteLength(name, 'utf8') + 1 : 0) +
- 1 +
- 4 +
- Buffer.byteLength(value.code.toString(), 'utf8') +
- 1
- );
- }
- } else if (value instanceof Binary || value['_bsontype'] === 'Binary') {
- // Check what kind of subtype we have
- if (value.sub_type === Binary.SUBTYPE_BYTE_ARRAY) {
- return (
- (name != null ? Buffer.byteLength(name, 'utf8') + 1 : 0) +
- (value.position + 1 + 4 + 1 + 4)
- );
- } else {
- return (
- (name != null ? Buffer.byteLength(name, 'utf8') + 1 : 0) + (value.position + 1 + 4 + 1)
- );
- }
- } else if (value instanceof BSONSymbol || value['_bsontype'] === 'Symbol') {
- return (
- (name != null ? Buffer.byteLength(name, 'utf8') + 1 : 0) +
- Buffer.byteLength(value.value, 'utf8') +
- 4 +
- 1 +
- 1
- );
- } else if (value instanceof DBRef || value['_bsontype'] === 'DBRef') {
- // Set up correct object for serialization
- const ordered_values = Object.assign(
- {
- $ref: value.collection,
- $id: value.oid
- },
- value.fields
- );
- // Add db reference if it exists
- if (value.db != null) {
- ordered_values['$db'] = value.db;
- }
- return (
- (name != null ? Buffer.byteLength(name, 'utf8') + 1 : 0) +
- 1 +
- calculateObjectSize(ordered_values, serializeFunctions, ignoreUndefined)
- );
- } else if (
- value instanceof RegExp ||
- Object.prototype.toString.call(value) === '[object RegExp]'
- ) {
- return (
- (name != null ? Buffer.byteLength(name, 'utf8') + 1 : 0) +
- 1 +
- Buffer.byteLength(value.source, 'utf8') +
- 1 +
- (value.global ? 1 : 0) +
- (value.ignoreCase ? 1 : 0) +
- (value.multiline ? 1 : 0) +
- 1
- );
- } else if (value instanceof BSONRegExp || value['_bsontype'] === 'BSONRegExp') {
- return (
- (name != null ? Buffer.byteLength(name, 'utf8') + 1 : 0) +
- 1 +
- Buffer.byteLength(value.pattern, 'utf8') +
- 1 +
- Buffer.byteLength(value.options, 'utf8') +
- 1
- );
- } else {
- return (
- (name != null ? Buffer.byteLength(name, 'utf8') + 1 : 0) +
- calculateObjectSize(value, serializeFunctions, ignoreUndefined) +
- 1
- );
- }
- case 'function':
- // WTF for 0.4.X where typeof /someregexp/ === 'function'
- if (
- value instanceof RegExp ||
- Object.prototype.toString.call(value) === '[object RegExp]' ||
- String.call(value) === '[object RegExp]'
- ) {
- return (
- (name != null ? Buffer.byteLength(name, 'utf8') + 1 : 0) +
- 1 +
- Buffer.byteLength(value.source, 'utf8') +
- 1 +
- (value.global ? 1 : 0) +
- (value.ignoreCase ? 1 : 0) +
- (value.multiline ? 1 : 0) +
- 1
- );
- } else {
- if (serializeFunctions && value.scope != null && Object.keys(value.scope).length > 0) {
- return (
- (name != null ? Buffer.byteLength(name, 'utf8') + 1 : 0) +
- 1 +
- 4 +
- 4 +
- Buffer.byteLength(normalizedFunctionString(value), 'utf8') +
- 1 +
- calculateObjectSize(value.scope, serializeFunctions, ignoreUndefined)
- );
- } else if (serializeFunctions) {
- return (
- (name != null ? Buffer.byteLength(name, 'utf8') + 1 : 0) +
- 1 +
- 4 +
- Buffer.byteLength(normalizedFunctionString(value), 'utf8') +
- 1
- );
- }
- }
- }
- return 0;
- }
- module.exports = calculateObjectSize;
|