meteor/meteor

View on GitHub
packages/minimongo/minimongo_tests_client.js

Summary

Maintainability
F
1 mo
Test Coverage
import {hasOwn} from './common';

// Hack to make LocalCollection generate ObjectIDs by default.
LocalCollection._useOID = true;

// assert that f is a strcmp-style comparison function that puts
// 'values' in the provided order

const assert_ordering = (test, f, values) => {
  for (let i = 0; i < values.length; i++) {
    let x = f(values[i], values[i]);
    if (x !== 0) {
      // XXX super janky
      test.fail({type: 'minimongo-ordering',
        message: "value doesn't order as equal to itself",
        value: JSON.stringify(values[i]),
        should_be_zero_but_got: JSON.stringify(x)});
    }
    if (i + 1 < values.length) {
      const less = values[i];
      const more = values[i + 1];
      x = f(less, more);
      if (!(x < 0)) {
        // XXX super janky
        test.fail({type: 'minimongo-ordering',
          message: 'ordering test failed',
          first: JSON.stringify(less),
          second: JSON.stringify(more),
          should_be_negative_but_got: JSON.stringify(x)});
      }
      x = f(more, less);
      if (!(x > 0)) {
        // XXX super janky
        test.fail({type: 'minimongo-ordering',
          message: 'ordering test failed',
          first: JSON.stringify(less),
          second: JSON.stringify(more),
          should_be_positive_but_got: JSON.stringify(x)});
      }
    }
  }
};

const log_callbacks = operations => ({
  addedAt(obj, idx, before) {
    delete obj._id;
    operations.push(EJSON.clone(['added', obj, idx, before]));
  },

  changedAt(obj, old_obj, at) {
    delete obj._id;
    delete old_obj._id;
    operations.push(EJSON.clone(['changed', obj, at, old_obj]));
  },

  movedTo(obj, old_at, new_at, before) {
    delete obj._id;
    operations.push(EJSON.clone(['moved', obj, old_at, new_at, before]));
  },

  removedAt(old_obj, at) {
    const id = old_obj._id;
    delete old_obj._id;
    operations.push(EJSON.clone(['removed', id, at, old_obj]));
  },
});

// XXX test shared structure in all MM entrypoints
Tinytest.add('minimongo - basics', test => {
  const c = new LocalCollection();
  let fluffyKitten_id;
  let count;

  fluffyKitten_id = c.insert({type: 'kitten', name: 'fluffy'});
  c.insert({type: 'kitten', name: 'snookums'});
  c.insert({type: 'cryptographer', name: 'alice'});
  c.insert({type: 'cryptographer', name: 'bob'});
  c.insert({type: 'cryptographer', name: 'cara'});
  test.equal(c.find().count(), 5);
  test.equal(c.find({type: 'kitten'}).count(), 2);
  test.equal(c.find({type: 'cryptographer'}).count(), 3);
  test.length(c.find({type: 'kitten'}).fetch(), 2);
  test.length(c.find({type: 'cryptographer'}).fetch(), 3);
  test.equal(fluffyKitten_id, c.findOne({type: 'kitten', name: 'fluffy'})._id);

  c.remove({name: 'cara'});
  test.equal(c.find().count(), 4);
  test.equal(c.find({type: 'kitten'}).count(), 2);
  test.equal(c.find({type: 'cryptographer'}).count(), 2);
  test.length(c.find({type: 'kitten'}).fetch(), 2);
  test.length(c.find({type: 'cryptographer'}).fetch(), 2);

  count = c.update({name: 'snookums'}, {$set: {type: 'cryptographer'}});
  test.equal(count, 1);
  test.equal(c.find().count(), 4);
  test.equal(c.find({type: 'kitten'}).count(), 1);
  test.equal(c.find({type: 'cryptographer'}).count(), 3);
  test.length(c.find({type: 'kitten'}).fetch(), 1);
  test.length(c.find({type: 'cryptographer'}).fetch(), 3);

  c.remove(null);
  c.remove(false);
  c.remove(undefined);
  test.equal(c.find().count(), 4);

  c.remove({_id: null});
  c.remove({_id: false});
  c.remove({_id: undefined});
  count = c.remove();
  test.equal(count, 0);
  test.equal(c.find().count(), 4);

  count = c.remove({});
  test.equal(count, 4);
  test.equal(c.find().count(), 0);

  c.insert({_id: 1, name: 'strawberry', tags: ['fruit', 'red', 'squishy']});
  c.insert({_id: 2, name: 'apple', tags: ['fruit', 'red', 'hard']});
  c.insert({_id: 3, name: 'rose', tags: ['flower', 'red', 'squishy']});

  test.equal(c.find({tags: 'flower'}).count(), 1);
  test.equal(c.find({tags: 'fruit'}).count(), 2);
  test.equal(c.find({tags: 'red'}).count(), 3);
  test.length(c.find({tags: 'flower'}).fetch(), 1);
  test.length(c.find({tags: 'fruit'}).fetch(), 2);
  test.length(c.find({tags: 'red'}).fetch(), 3);

  test.equal(c.findOne(1).name, 'strawberry');
  test.equal(c.findOne(2).name, 'apple');
  test.equal(c.findOne(3).name, 'rose');
  test.equal(c.findOne(4), undefined);
  test.equal(c.findOne('abc'), undefined);
  test.equal(c.findOne(undefined), undefined);

  test.equal(c.find(1).count(), 1);
  test.equal(c.find(4).count(), 0);
  test.equal(c.find('abc').count(), 0);
  test.equal(c.find(undefined).count(), 0);
  test.equal(c.find().count(), 3);
  test.equal(c.find(1, {skip: 1}).count(false), 0);
  test.equal(c.find(1, {skip: 1}).count(), 0);
  test.equal(c.find({_id: 1}, {skip: 1}).count(false), 0);
  test.equal(c.find({_id: 1}, {skip: 1}).count(), 0);
  test.equal(c.find({_id: undefined}).count(), 0);
  test.equal(c.find({_id: false}).count(), 0);
  test.equal(c.find({_id: null}).count(), 0);
  test.equal(c.find({_id: ''}).count(), 0);
  test.equal(c.find({_id: 0}).count(), 0);
  test.equal(c.find({}, {skip: 1}).count(false), 2);
  test.equal(c.find({}, {skip: 1}).count(), 2);
  test.equal(c.find({}, {skip: 2}).count(), 1);
  test.equal(c.find({}, {limit: 2}).count(false), 2);
  test.equal(c.find({}, {limit: 2}).count(), 2);
  test.equal(c.find({}, {limit: 1}).count(), 1);
  test.equal(c.find({}, {skip: 1, limit: 1}).count(false), 1);
  test.equal(c.find({}, {skip: 1, limit: 1}).count(), 1);
  test.equal(c.find({tags: 'fruit'}, {skip: 1}).count(false), 1);
  test.equal(c.find({tags: 'fruit'}, {skip: 1}).count(), 1);
  test.equal(c.find({tags: 'fruit'}, {limit: 1}).count(false), 1);
  test.equal(c.find({tags: 'fruit'}, {limit: 1}).count(), 1);
  test.equal(c.find({tags: 'fruit'}, {skip: 1, limit: 1}).count(false), 1);
  test.equal(c.find({tags: 'fruit'}, {skip: 1, limit: 1}).count(), 1);
  test.equal(c.find(1, {sort: ['_id', 'desc'], skip: 1}).count(false), 0);
  test.equal(c.find(1, {sort: ['_id', 'desc'], skip: 1}).count(), 0);
  test.equal(c.find({_id: 1}, {sort: ['_id', 'desc'], skip: 1}).count(false), 0);
  test.equal(c.find({_id: 1}, {sort: ['_id', 'desc'], skip: 1}).count(), 0);
  test.equal(c.find({}, {sort: ['_id', 'desc'], skip: 1}).count(false), 2);
  test.equal(c.find({}, {sort: ['_id', 'desc'], skip: 1}).count(), 2);
  test.equal(c.find({}, {sort: ['_id', 'desc'], skip: 2}).count(), 1);
  test.equal(c.find({}, {sort: ['_id', 'desc'], limit: 2}).count(false), 2);
  test.equal(c.find({}, {sort: ['_id', 'desc'], limit: 2}).count(), 2);
  test.equal(c.find({}, {sort: ['_id', 'desc'], limit: 1}).count(), 1);
  test.equal(c.find({}, {sort: ['_id', 'desc'], skip: 1, limit: 1}).count(false), 1);
  test.equal(c.find({}, {sort: ['_id', 'desc'], skip: 1, limit: 1}).count(), 1);
  test.equal(c.find({tags: 'fruit'}, {sort: ['_id', 'desc'], skip: 1}).count(false), 1);
  test.equal(c.find({tags: 'fruit'}, {sort: ['_id', 'desc'], skip: 1}).count(), 1);
  test.equal(c.find({tags: 'fruit'}, {sort: ['_id', 'desc'], limit: 1}).count(false), 1);
  test.equal(c.find({tags: 'fruit'}, {sort: ['_id', 'desc'], limit: 1}).count(), 1);
  test.equal(c.find({tags: 'fruit'}, {sort: ['_id', 'desc'], skip: 1, limit: 1}).count(false), 1);
  test.equal(c.find({tags: 'fruit'}, {sort: ['_id', 'desc'], skip: 1, limit: 1}).count(), 1);

  // Regression test for #455.
  c.insert({foo: {bar: 'baz'}});
  test.equal(c.find({foo: {bam: 'baz'}}).count(), 0);
  test.equal(c.find({foo: {bar: 'baz'}}).count(), 1);

  // Regression test for #5301
  c.remove({});
  c.insert({a: 'a', b: 'b'});
  const noop = () => null;
  test.equal(c.find({a: noop}).count(), 1);
  test.equal(c.find({a: 'a', b: noop}).count(), 1);
  test.equal(c.find({c: noop}).count(), 1);
  test.equal(c.find({a: noop, c: 'c'}).count(), 0);

  // Regression test for #4260
  // Only insert enumerable, own properties from the object
  c.remove({});
  function Thing() {
    this.a = 1;
    this.b = 2;
    Object.defineProperty(this, 'b', { enumerable: false });
  }
  Thing.prototype.c = 3;
  Thing.prototype.d = () => null;
  const before = new Thing();
  c.insert(before);
  const after = c.findOne();
  test.equal(after.a, 1);
  test.equal(after.b, undefined);
  test.equal(after.c, undefined);
  test.equal(after.d, undefined);
});

Tinytest.add('minimongo - error - no options', test => {
  try {
    throw MinimongoError('Not fun to have errors');
  } catch (e) {
    test.equal(e.message, 'Not fun to have errors');
  }
});

Tinytest.add('minimongo - error - with field', test => {
  try {
    throw MinimongoError('Cats are no fun', { field: 'mice' });
  } catch (e) {
    test.equal(e.message, "Cats are no fun for field 'mice'");
  }
});

Tinytest.add('minimongo - cursors', test => {
  const c = new LocalCollection();
  let res;

  for (let i = 0; i < 20; i++) {c.insert({i});}

  const q = c.find();
  test.equal(q.count(), 20);

  // fetch
  res = q.fetch();
  test.length(res, 20);
  for (let i = 0; i < 20; i++) {
    test.equal(res[i].i, i);
  }
  // call it again, it still works
  test.length(q.fetch(), 20);

  // forEach
  let count = 0;
  const context = {};
  q.forEach(function(obj, i, cursor) {
    test.equal(obj.i, count++);
    test.equal(obj.i, i);
    test.isTrue(context === this);
    test.isTrue(cursor === q);
  }, context);
  test.equal(count, 20);
  // call it again, it still works
  test.length(q.fetch(), 20);

  // iterator
  count = 0;
  for (let obj of q) {
    test.equal(obj.i, count++);
  };
  test.equal(count, 20);
  // call it again, it still works
  test.length(q.fetch(), 20);
  // test spread operator
  test.equal([...q], q.fetch());

  // map
  res = q.map(function(obj, i, cursor) {
    test.equal(obj.i, i);
    test.isTrue(context === this);
    test.isTrue(cursor === q);
    return obj.i * 2;
  }, context);
  test.length(res, 20);
  for (let i = 0; i < 20; i++) {test.equal(res[i], i * 2);}
  // call it again, it still works
  test.length(q.fetch(), 20);

  // findOne (and no rewind first)
  test.equal(c.findOne({i: 0}).i, 0);
  test.equal(c.findOne({i: 1}).i, 1);
  const id = c.findOne({i: 2})._id;
  test.equal(c.findOne(id).i, 2);
});

Tinytest.add('minimongo - transform', test => {
  const c = new LocalCollection;
  c.insert({});
  // transform functions must return objects
  const invalidTransform = doc => doc._id;
  test.throws(() => {
    c.findOne({}, {transform: invalidTransform});
  });

  // transformed documents get _id field transplanted if not present
  const transformWithoutId = doc => {
    const docWithoutId = Object.assign({}, doc);
    delete docWithoutId._id;
    return docWithoutId;
  };
  test.equal(c.findOne({}, {transform: transformWithoutId})._id,
    c.findOne()._id);
});

Tinytest.add('minimongo - misc', test => {
  // deepcopy
  let a = {a: [1, 2, 3], b: 'x', c: true, d: {x: 12, y: [12]},
    f: null, g: new Date()};
  let b = EJSON.clone(a);
  test.equal(a, b);
  test.isTrue(LocalCollection._f._equal(a, b));
  a.a.push(4);
  test.length(b.a, 3);
  a.c = false;
  test.isTrue(b.c);
  b.d.z = 15;
  a.d.z = 14;
  test.equal(b.d.z, 15);
  a.d.y.push(88);
  test.length(b.d.y, 1);
  test.equal(a.g, b.g);
  b.g.setDate(b.g.getDate() + 1);
  test.notEqual(a.g, b.g);

  a = {x() {}};
  b = EJSON.clone(a);
  a.x.a = 14;
  test.equal(b.x.a, 14); // just to document current behavior
});

Tinytest.add('minimongo - lookup', test => {
  const lookupA = MinimongoTest.makeLookupFunction('a');
  test.equal(lookupA({}), [{value: undefined}]);
  test.equal(lookupA({a: 1}), [{value: 1}]);
  test.equal(lookupA({a: [1]}), [{value: [1]}]);

  const lookupAX = MinimongoTest.makeLookupFunction('a.x');
  test.equal(lookupAX({a: {x: 1}}), [{value: 1}]);
  test.equal(lookupAX({a: {x: [1]}}), [{value: [1]}]);
  test.equal(lookupAX({a: 5}), [{value: undefined}]);
  test.equal(lookupAX({a: [{x: 1}, {x: [2]}, {y: 3}]}),
    [{value: 1, arrayIndices: [0]},
      {value: [2], arrayIndices: [1]},
      {value: undefined, arrayIndices: [2]}]);

  const lookupA0X = MinimongoTest.makeLookupFunction('a.0.x');
  test.equal(lookupA0X({a: [{x: 1}]}), [
    // From interpreting '0' as "0th array element".
    {value: 1, arrayIndices: [0, 'x']},
    // From interpreting '0' as "after branching in the array, look in the
    // object {x:1} for a field named 0".
    {value: undefined, arrayIndices: [0]}]);
  test.equal(lookupA0X({a: [{x: [1]}]}), [
    {value: [1], arrayIndices: [0, 'x']},
    {value: undefined, arrayIndices: [0]}]);
  test.equal(lookupA0X({a: 5}), [{value: undefined}]);
  test.equal(lookupA0X({a: [{x: 1}, {x: [2]}, {y: 3}]}), [
    // From interpreting '0' as "0th array element".
    {value: 1, arrayIndices: [0, 'x']},
    // From interpreting '0' as "after branching in the array, look in the
    // object {x:1} for a field named 0".
    {value: undefined, arrayIndices: [0]},
    {value: undefined, arrayIndices: [1]},
    {value: undefined, arrayIndices: [2]},
  ]);

  test.equal(
    MinimongoTest.makeLookupFunction('w.x.0.z')({
      w: [{x: [{z: 5}]}]}), [
      // From interpreting '0' as "0th array element".
      {value: 5, arrayIndices: [0, 0, 'x']},
      // From interpreting '0' as "after branching in the array, look in the
      // object {z:5} for a field named "0".
      {value: undefined, arrayIndices: [0, 0]},
    ]);
});

Tinytest.add('minimongo - selector_compiler', test => {
  const matches = (shouldMatch, selector, doc) => {
    const doesMatch = new Minimongo.Matcher(selector).documentMatches(doc).result;
    if (doesMatch != shouldMatch) {
      // XXX super janky
      test.fail({message: `minimongo match failure: document ${shouldMatch ? "should match, but doesn't" :
        "shouldn't match, but does"}`,
      selector: JSON.stringify(selector),
      document: JSON.stringify(doc),
      });
    }
  };

  const match = matches.bind(null, true);
  const nomatch = matches.bind(null, false);

  // XXX blog post about what I learned while writing these tests (weird
  // mongo edge cases)

  // empty selectors
  match({}, {});
  match({}, {a: 12});

  // scalars
  match(1, {_id: 1, a: 'foo'});
  nomatch(1, {_id: 2, a: 'foo'});
  match('a', {_id: 'a', a: 'foo'});
  nomatch('a', {_id: 'b', a: 'foo'});

  // safety
  nomatch(undefined, {});
  nomatch(undefined, {_id: 'foo'});
  nomatch(false, {_id: 'foo'});
  nomatch(null, {_id: 'foo'});
  nomatch({_id: undefined}, {_id: 'foo'});
  nomatch({_id: false}, {_id: 'foo'});
  nomatch({_id: null}, {_id: 'foo'});
  nomatch({_id: ''}, {_id: ''});
  nomatch({_id: 0}, {_id: 0});

  // matching one or more keys
  nomatch({a: 12}, {});
  match({a: 12}, {a: 12});
  match({a: 12}, {a: 12, b: 13});
  match({a: 12, b: 13}, {a: 12, b: 13});
  match({a: 12, b: 13}, {a: 12, b: 13, c: 14});
  nomatch({a: 12, b: 13, c: 14}, {a: 12, b: 13});
  nomatch({a: 12, b: 13}, {b: 13, c: 14});

  match({a: 12}, {a: [12]});
  match({a: 12}, {a: [11, 12, 13]});
  nomatch({a: 12}, {a: [11, 13]});
  match({a: 12, b: 13}, {a: [11, 12, 13], b: [13, 14, 15]});
  nomatch({a: 12, b: 13}, {a: [11, 12, 13], b: [14, 15]});

  // dates
  const date1 = new Date;
  const date2 = new Date(date1.getTime() + 1000);
  const date3 = new Date('');
  match({a: date1}, {a: date1});
  nomatch({a: date1}, {a: date2});
  match({a: date3}, {a: date3});
  nomatch({a: date1}, {a: date3});
  nomatch({a: date3}, {a: date1});
  match({a: {$gt: date3}}, {a: date1});
  match({a: {$gte: date3}}, {a: date1});
  nomatch({a: {$lt: date3}}, {a: date1});
  nomatch({a: {$lte: date3}}, {a: date1});


  // arrays
  match({a: [1, 2]}, {a: [1, 2]});
  match({a: [1, 2]}, {a: [[1, 2]]});
  match({a: [1, 2]}, {a: [[3, 4], [1, 2]]});
  nomatch({a: [1, 2]}, {a: [3, 4]});
  nomatch({a: [1, 2]}, {a: [[[1, 2]]]});

  // literal documents
  match({a: {b: 12}}, {a: {b: 12}});
  nomatch({a: {b: 12, c: 13}}, {a: {b: 12}});
  nomatch({a: {b: 12}}, {a: {b: 12, c: 13}});
  match({a: {b: 12, c: 13}}, {a: {b: 12, c: 13}});
  nomatch({a: {b: 12, c: 13}}, {a: {c: 13, b: 12}}); // tested on mongodb
  nomatch({a: {}}, {a: {b: 12}});
  nomatch({a: {b: 12}}, {a: {}});
  match(
    {a: {b: 12, c: [13, true, false, 2.2, 'a', null, {d: 14}]}},
    {a: {b: 12, c: [13, true, false, 2.2, 'a', null, {d: 14}]}});
  match({a: {b: 12}}, {a: {b: 12}, k: 99});

  match({a: {b: 12}}, {a: [{b: 12}]});
  nomatch({a: {b: 12}}, {a: [[{b: 12}]]});
  match({a: {b: 12}}, {a: [{b: 11}, {b: 12}, {b: 13}]});
  nomatch({a: {b: 12}}, {a: [{b: 11}, {b: 12, c: 20}, {b: 13}]});
  nomatch({a: {b: 12, c: 20}}, {a: [{b: 11}, {b: 12}, {c: 20}]});
  match({a: {b: 12, c: 20}}, {a: [{b: 11}, {b: 12, c: 20}, {b: 13}]});

  // null
  match({a: null}, {a: null});
  match({a: null}, {b: 12});
  nomatch({a: null}, {a: 12});
  match({a: null}, {a: [1, 2, null, 3]}); // tested on mongodb
  nomatch({a: null}, {a: [1, 2, {}, 3]}); // tested on mongodb

  // order comparisons: $lt, $gt, $lte, $gte
  match({a: {$lt: 10}}, {a: 9});
  nomatch({a: {$lt: 10}}, {a: 10});
  nomatch({a: {$lt: 10}}, {a: 11});

  match({a: {$gt: 10}}, {a: 11});
  nomatch({a: {$gt: 10}}, {a: 10});
  nomatch({a: {$gt: 10}}, {a: 9});

  match({a: {$lte: 10}}, {a: 9});
  match({a: {$lte: 10}}, {a: 10});
  nomatch({a: {$lte: 10}}, {a: 11});

  match({a: {$gte: 10}}, {a: 11});
  match({a: {$gte: 10}}, {a: 10});
  nomatch({a: {$gte: 10}}, {a: 9});

  match({a: {$lt: 10}}, {a: [11, 9, 12]});
  nomatch({a: {$lt: 10}}, {a: [11, 12]});

  // (there's a full suite of ordering test elsewhere)
  nomatch({a: {$lt: 'null'}}, {a: null});
  match({a: {$lt: {x: [2, 3, 4]}}}, {a: {x: [1, 3, 4]}});
  match({a: {$gt: {x: [2, 3, 4]}}}, {a: {x: [3, 3, 4]}});
  nomatch({a: {$gt: {x: [2, 3, 4]}}}, {a: {x: [1, 3, 4]}});
  nomatch({a: {$gt: {x: [2, 3, 4]}}}, {a: {x: [2, 3, 4]}});
  nomatch({a: {$lt: {x: [2, 3, 4]}}}, {a: {x: [2, 3, 4]}});
  match({a: {$gte: {x: [2, 3, 4]}}}, {a: {x: [2, 3, 4]}});
  match({a: {$lte: {x: [2, 3, 4]}}}, {a: {x: [2, 3, 4]}});

  nomatch({a: {$gt: [2, 3]}}, {a: [1, 2]}); // tested against mongodb

  // composition of two qualifiers
  nomatch({a: {$lt: 11, $gt: 9}}, {a: 8});
  nomatch({a: {$lt: 11, $gt: 9}}, {a: 9});
  match({a: {$lt: 11, $gt: 9}}, {a: 10});
  nomatch({a: {$lt: 11, $gt: 9}}, {a: 11});
  nomatch({a: {$lt: 11, $gt: 9}}, {a: 12});

  match({a: {$lt: 11, $gt: 9}}, {a: [8, 9, 10, 11, 12]});
  match({a: {$lt: 11, $gt: 9}}, {a: [8, 9, 11, 12]}); // tested against mongodb

  // $all
  match({a: {$all: [1, 2]}}, {a: [1, 2]});
  nomatch({a: {$all: [1, 2, 3]}}, {a: [1, 2]});
  match({a: {$all: [1, 2]}}, {a: [3, 2, 1]});
  match({a: {$all: [1, 'x']}}, {a: [3, 'x', 1]});
  nomatch({a: {$all: ['2']}}, {a: 2});
  nomatch({a: {$all: [2]}}, {a: '2'});
  match({a: {$all: [[1, 2], [1, 3]]}}, {a: [[1, 3], [1, 2], [1, 4]]});
  nomatch({a: {$all: [[1, 2], [1, 3]]}}, {a: [[1, 4], [1, 2], [1, 4]]});
  match({a: {$all: [2, 2]}}, {a: [2]}); // tested against mongodb
  nomatch({a: {$all: [2, 3]}}, {a: [2, 2]});

  nomatch({a: {$all: [1, 2]}}, {a: [[1, 2]]}); // tested against mongodb
  nomatch({a: {$all: [1, 2]}}, {}); // tested against mongodb, field doesn't exist
  nomatch({a: {$all: [1, 2]}}, {a: {foo: 'bar'}}); // tested against mongodb, field is not an object
  nomatch({a: {$all: []}}, {a: []});
  nomatch({a: {$all: []}}, {a: [5]});
  match({a: {$all: [/i/, /e/i]}}, {a: ['foo', 'bEr', 'biz']});
  nomatch({a: {$all: [/i/, /e/i]}}, {a: ['foo', 'bar', 'biz']});
  match({a: {$all: [{b: 3}]}}, {a: [{b: 3}]});
  // Members of $all other than regexps are *equality matches*, not document
  // matches.
  nomatch({a: {$all: [{b: 3}]}}, {a: [{b: 3, k: 4}]});
  test.throws(() => {
    match({a: {$all: [{$gt: 4}]}}, {});
  });

  // $exists
  match({a: {$exists: true}}, {a: 12});
  nomatch({a: {$exists: true}}, {b: 12});
  nomatch({a: {$exists: false}}, {a: 12});
  match({a: {$exists: false}}, {b: 12});

  match({a: {$exists: true}}, {a: []});
  nomatch({a: {$exists: true}}, {b: []});
  nomatch({a: {$exists: false}}, {a: []});
  match({a: {$exists: false}}, {b: []});

  match({a: {$exists: true}}, {a: [1]});
  nomatch({a: {$exists: true}}, {b: [1]});
  nomatch({a: {$exists: false}}, {a: [1]});
  match({a: {$exists: false}}, {b: [1]});

  match({a: {$exists: 1}}, {a: 5});
  match({a: {$exists: 0}}, {b: 5});

  nomatch({'a.x': {$exists: false}}, {a: [{}, {x: 5}]});
  match({'a.x': {$exists: true}}, {a: [{}, {x: 5}]});
  match({'a.x': {$exists: true}}, {a: [{}, {x: 5}]});
  match({'a.x': {$exists: true}}, {a: {x: []}});
  match({'a.x': {$exists: true}}, {a: {x: null}});

  // $mod
  match({a: {$mod: [10, 1]}}, {a: 11});
  nomatch({a: {$mod: [10, 1]}}, {a: 12});
  match({a: {$mod: [10, 1]}}, {a: [10, 11, 12]});
  nomatch({a: {$mod: [10, 1]}}, {a: [10, 12]});
  [
    5,
    [10],
    [10, 1, 2],
    'foo',
    {bar: 1},
    [],
  ].forEach(badMod => {
    test.throws(() => {
      match({a: {$mod: badMod}}, {a: 11});
    });
  });

  // $eq
  nomatch({a: {$eq: 1}}, {a: 2});
  match({a: {$eq: 2}}, {a: 2});
  nomatch({a: {$eq: [1]}}, {a: [2]});

  match({a: {$eq: [1, 2]}}, {a: [1, 2]});
  match({a: {$eq: 1}}, {a: [1, 2]});
  match({a: {$eq: 2}}, {a: [1, 2]});
  nomatch({a: {$eq: 3}}, {a: [1, 2]});
  match({'a.b': {$eq: 1}}, {a: [{b: 1}, {b: 2}]});
  match({'a.b': {$eq: 2}}, {a: [{b: 1}, {b: 2}]});
  nomatch({'a.b': {$eq: 3}}, {a: [{b: 1}, {b: 2}]});

  match({a: {$eq: {x: 1}}}, {a: {x: 1}});
  nomatch({a: {$eq: {x: 1}}}, {a: {x: 2}});
  nomatch({a: {$eq: {x: 1}}}, {a: {x: 1, y: 2}});

  // $ne
  match({a: {$ne: 1}}, {a: 2});
  nomatch({a: {$ne: 2}}, {a: 2});
  match({a: {$ne: [1]}}, {a: [2]});

  nomatch({a: {$ne: [1, 2]}}, {a: [1, 2]}); // all tested against mongodb
  nomatch({a: {$ne: 1}}, {a: [1, 2]});
  nomatch({a: {$ne: 2}}, {a: [1, 2]});
  match({a: {$ne: 3}}, {a: [1, 2]});
  nomatch({'a.b': {$ne: 1}}, {a: [{b: 1}, {b: 2}]});
  nomatch({'a.b': {$ne: 2}}, {a: [{b: 1}, {b: 2}]});
  match({'a.b': {$ne: 3}}, {a: [{b: 1}, {b: 2}]});

  nomatch({a: {$ne: {x: 1}}}, {a: {x: 1}});
  match({a: {$ne: {x: 1}}}, {a: {x: 2}});
  match({a: {$ne: {x: 1}}}, {a: {x: 1, y: 2}});

  // This query means: All 'a.b' must be non-5, and some 'a.b' must be >6.
  match({'a.b': {$ne: 5, $gt: 6}}, {a: [{b: 2}, {b: 10}]});
  nomatch({'a.b': {$ne: 5, $gt: 6}}, {a: [{b: 2}, {b: 4}]});
  nomatch({'a.b': {$ne: 5, $gt: 6}}, {a: [{b: 2}, {b: 5}]});
  nomatch({'a.b': {$ne: 5, $gt: 6}}, {a: [{b: 10}, {b: 5}]});
  // Should work the same if the branch is at the bottom.
  match({a: {$ne: 5, $gt: 6}}, {a: [2, 10]});
  nomatch({a: {$ne: 5, $gt: 6}}, {a: [2, 4]});
  nomatch({a: {$ne: 5, $gt: 6}}, {a: [2, 5]});
  nomatch({a: {$ne: 5, $gt: 6}}, {a: [10, 5]});

  // $in
  match({a: {$in: [1, 2, 3]}}, {a: 2});
  nomatch({a: {$in: [1, 2, 3]}}, {a: 4});
  match({a: {$in: [[1], [2], [3]]}}, {a: [2]});
  nomatch({a: {$in: [[1], [2], [3]]}}, {a: [4]});
  match({a: {$in: [{b: 1}, {b: 2}, {b: 3}]}}, {a: {b: 2}});
  nomatch({a: {$in: [{b: 1}, {b: 2}, {b: 3}]}}, {a: {b: 4}});

  match({a: {$in: [1, 2, 3]}}, {a: [2]}); // tested against mongodb
  match({a: {$in: [{x: 1}, {x: 2}, {x: 3}]}}, {a: [{x: 2}]});
  match({a: {$in: [1, 2, 3]}}, {a: [4, 2]});
  nomatch({a: {$in: [1, 2, 3]}}, {a: [4]});

  match({a: {$in: ['x', /foo/i]}}, {a: 'x'});
  match({a: {$in: ['x', /foo/i]}}, {a: 'fOo'});
  match({a: {$in: ['x', /foo/i]}}, {a: ['f', 'fOo']});
  nomatch({a: {$in: ['x', /foo/i]}}, {a: ['f', 'fOx']});

  match({a: {$in: [1, null]}}, {});
  match({'a.b': {$in: [1, null]}}, {});
  match({'a.b': {$in: [1, null]}}, {a: {}});
  match({'a.b': {$in: [1, null]}}, {a: {b: null}});
  nomatch({'a.b': {$in: [1, null]}}, {a: {b: 5}});
  nomatch({'a.b': {$in: [1]}}, {a: {b: null}});
  nomatch({'a.b': {$in: [1]}}, {a: {}});
  nomatch({'a.b': {$in: [1, null]}}, {a: [{b: 5}]});
  match({'a.b': {$in: [1, null]}}, {a: [{b: 5}, {}]});
  nomatch({'a.b': {$in: [1, null]}}, {a: [{b: 5}, []]});
  nomatch({'a.b': {$in: [1, null]}}, {a: [{b: 5}, 5]});

  // $nin
  nomatch({a: {$nin: [1, 2, 3]}}, {a: 2});
  match({a: {$nin: [1, 2, 3]}}, {a: 4});
  nomatch({a: {$nin: [[1], [2], [3]]}}, {a: [2]});
  match({a: {$nin: [[1], [2], [3]]}}, {a: [4]});
  nomatch({a: {$nin: [{b: 1}, {b: 2}, {b: 3}]}}, {a: {b: 2}});
  match({a: {$nin: [{b: 1}, {b: 2}, {b: 3}]}}, {a: {b: 4}});

  nomatch({a: {$nin: [1, 2, 3]}}, {a: [2]}); // tested against mongodb
  nomatch({a: {$nin: [{x: 1}, {x: 2}, {x: 3}]}}, {a: [{x: 2}]});
  nomatch({a: {$nin: [1, 2, 3]}}, {a: [4, 2]});
  nomatch({'a.b': {$nin: [1, 2, 3]}}, {a: [{b: 4}, {b: 2}]});
  match({a: {$nin: [1, 2, 3]}}, {a: [4]});
  match({'a.b': {$nin: [1, 2, 3]}}, {a: [{b: 4}]});

  nomatch({a: {$nin: ['x', /foo/i]}}, {a: 'x'});
  nomatch({a: {$nin: ['x', /foo/i]}}, {a: 'fOo'});
  nomatch({a: {$nin: ['x', /foo/i]}}, {a: ['f', 'fOo']});
  match({a: {$nin: ['x', /foo/i]}}, {a: ['f', 'fOx']});

  nomatch({a: {$nin: [1, null]}}, {});
  nomatch({'a.b': {$nin: [1, null]}}, {});
  nomatch({'a.b': {$nin: [1, null]}}, {a: {}});
  nomatch({'a.b': {$nin: [1, null]}}, {a: {b: null}});
  match({'a.b': {$nin: [1, null]}}, {a: {b: 5}});
  match({'a.b': {$nin: [1]}}, {a: {b: null}});
  match({'a.b': {$nin: [1]}}, {a: {}});
  match({'a.b': {$nin: [1, null]}}, {a: [{b: 5}]});
  nomatch({'a.b': {$nin: [1, null]}}, {a: [{b: 5}, {}]});
  match({'a.b': {$nin: [1, null]}}, {a: [{b: 5}, []]});
  match({'a.b': {$nin: [1, null]}}, {a: [{b: 5}, 5]});

  // $size
  match({a: {$size: 0}}, {a: []});
  match({a: {$size: 1}}, {a: [2]});
  match({a: {$size: 2}}, {a: [2, 2]});
  nomatch({a: {$size: 0}}, {a: [2]});
  nomatch({a: {$size: 1}}, {a: []});
  nomatch({a: {$size: 1}}, {a: [2, 2]});
  nomatch({a: {$size: 0}}, {a: '2'});
  nomatch({a: {$size: 1}}, {a: '2'});
  nomatch({a: {$size: 2}}, {a: '2'});

  nomatch({a: {$size: 2}}, {a: [[2, 2]]}); // tested against mongodb


  // $bitsAllClear - number
  match({a: {$bitsAllClear: [0, 1, 2, 3]}}, {a: 0});
  match({a: {$bitsAllClear: [0, 1, 2, 3]}}, {a: 0b10000});
  nomatch({a: {$bitsAllClear: [0, 1, 2, 3]}}, {a: 0b1});
  nomatch({a: {$bitsAllClear: [0, 1, 2, 3]}}, {a: 0b10});
  nomatch({a: {$bitsAllClear: [0, 1, 2, 3]}}, {a: 0b100});
  nomatch({a: {$bitsAllClear: [0, 1, 2, 3]}}, {a: 0b1000});

  // $bitsAllClear - buffer
  match({a: {$bitsAllClear: new Uint8Array([3])}}, {a: new Uint8Array([4])});
  match({a: {$bitsAllClear: new Uint8Array([0, 1])}}, {a: new Uint8Array([255])});  // 256 should not be set for 255.
  match({a: {$bitsAllClear: new Uint8Array([3])}}, {a: 4 });

  match({a: {$bitsAllClear: new Uint8Array([3])}}, {a: 0 });

  // $bitsAllSet - number
  match({a: {$bitsAllSet: [0, 1, 2, 3]}}, {a: 0b1111});
  nomatch({a: {$bitsAllSet: [0, 1, 2, 3]}}, {a: 0b111});
  nomatch({a: {$bitsAllSet: [0, 1, 2, 3]}}, {a: 256});
  nomatch({a: {$bitsAllSet: [0, 1, 2, 3]}}, {a: 50000});
  match({a: {$bitsAllSet: [0, 1, 2]}}, {a: 15});
  match({a: {$bitsAllSet: [0, 12]}}, {a: 0b1000000000001});
  nomatch({a: {$bitsAllSet: [0, 12]}}, {a: 0b1000000000000});
  nomatch({a: {$bitsAllSet: [0, 12]}}, {a: 0b1});

  // $bitsAllSet - buffer
  match({a: {$bitsAllSet: new Uint8Array([3])}}, {a: new Uint8Array([3])});
  match({a: {$bitsAllSet: new Uint8Array([7])}}, {a: new Uint8Array([15])});
  match({a: {$bitsAllSet: new Uint8Array([3])}}, {a: 3 });

  // $bitsAnySet - number
  match({a: {$bitsAnySet: [0, 1, 2, 3]}}, {a: 0b1});
  match({a: {$bitsAnySet: [0, 1, 2, 3]}}, {a: 0b10});
  match({a: {$bitsAnySet: [0, 1, 2, 3]}}, {a: 0b100});
  match({a: {$bitsAnySet: [0, 1, 2, 3]}}, {a: 0b1000});
  match({a: {$bitsAnySet: [4]}}, {a: 0b10000});
  nomatch({a: {$bitsAnySet: [0, 1, 2, 3]}}, {a: 0b10000});
  nomatch({a: {$bitsAnySet: [0, 1, 2, 3]}}, {a: 0});

  // $bitsAnySet - buffer
  match({a: {$bitsAnySet: new Uint8Array([3])}}, {a: new Uint8Array([7])});
  match({a: {$bitsAnySet: new Uint8Array([15])}}, {a: new Uint8Array([7])});
  match({a: {$bitsAnySet: new Uint8Array([3])}}, {a: 1 });

  // $bitsAnyClear - number
  match({a: {$bitsAnyClear: [0, 1, 2, 3]}}, {a: 0});
  match({a: {$bitsAnyClear: [0, 1, 2, 3]}}, {a: 0b1});
  match({a: {$bitsAnyClear: [0, 1, 2, 3]}}, {a: 0b10});
  match({a: {$bitsAnyClear: [0, 1, 2, 3]}}, {a: 0b100});
  match({a: {$bitsAnyClear: [0, 1, 2, 3]}}, {a: 0b1000});
  match({a: {$bitsAnyClear: [0, 1, 2, 3]}}, {a: 0b10000});
  nomatch({a: {$bitsAnyClear: [0, 1, 2, 3]}}, {a: 0b1111});
  match({a: {$bitsAnyClear: [0, 1, 2, 3]}}, {a: 0b111});
  nomatch({a: {$bitsAnyClear: [0, 1, 2]}}, {a: 0b111});
  match({a: {$bitsAnyClear: [0, 1, 2, 3]}}, {a: 0b11});
  nomatch({a: {$bitsAnyClear: [0, 1]}}, {a: 0b11});
  match({a: {$bitsAnyClear: [0, 1, 2, 3]}}, {a: 0b1});
  nomatch({a: {$bitsAnyClear: [0]}}, {a: 0b1});
  nomatch({a: {$bitsAnyClear: [4]}}, {a: 0b10000});

  // $bitsAnyClear - buffer
  match({a: {$bitsAnyClear: new Uint8Array([8])}}, {a: new Uint8Array([7])});
  match({a: {$bitsAnyClear: new Uint8Array([1])}}, {a: new Uint8Array([0])});
  match({a: {$bitsAnyClear: new Uint8Array([1])}}, {a: 4 });

  // taken from: https://github.com/mongodb/mongo/blob/master/jstests/core/bittest.js
  const c = new LocalCollection;
  function matchCount(query, count) {
    const matches = c.find(query).count();
    if (matches !== count) {
      test.fail({message: `minimongo match count failure: matched ${matches} times, but should match ${count} times`,
        query: JSON.stringify(query),
        count: JSON.stringify(count),
      });
    }
  }

  // Tests on numbers.

  c.insert({a: 0});
  c.insert({a: 1});
  c.insert({a: 54});
  c.insert({a: 88});
  c.insert({a: 255});

  // Tests with bitmask.
  matchCount({a: {$bitsAllSet: 0}}, 5);
  matchCount({a: {$bitsAllSet: 1}}, 2);
  matchCount({a: {$bitsAllSet: 16}}, 3);
  matchCount({a: {$bitsAllSet: 54}}, 2);
  matchCount({a: {$bitsAllSet: 55}}, 1);
  matchCount({a: {$bitsAllSet: 88}}, 2);
  matchCount({a: {$bitsAllSet: 255}}, 1);
  matchCount({a: {$bitsAllClear: 0}}, 5);
  matchCount({a: {$bitsAllClear: 1}}, 3);
  matchCount({a: {$bitsAllClear: 16}}, 2);
  matchCount({a: {$bitsAllClear: 129}}, 3);
  matchCount({a: {$bitsAllClear: 255}}, 1);
  matchCount({a: {$bitsAnySet: 0}}, 0);
  matchCount({a: {$bitsAnySet: 9}}, 3);
  matchCount({a: {$bitsAnySet: 255}}, 4);
  matchCount({a: {$bitsAnyClear: 0}}, 0);
  matchCount({a: {$bitsAnyClear: 18}}, 3);
  matchCount({a: {$bitsAnyClear: 24}}, 3);
  matchCount({a: {$bitsAnyClear: 255}}, 4);

  // Tests with array of bit positions.
  matchCount({a: {$bitsAllSet: []}}, 5);
  matchCount({a: {$bitsAllSet: [0]}}, 2);
  matchCount({a: {$bitsAllSet: [4]}}, 3);
  matchCount({a: {$bitsAllSet: [1, 2, 4, 5]}}, 2);
  matchCount({a: {$bitsAllSet: [0, 1, 2, 4, 5]}}, 1);
  matchCount({a: {$bitsAllSet: [3, 4, 6]}}, 2);
  matchCount({a: {$bitsAllSet: [0, 1, 2, 3, 4, 5, 6, 7]}}, 1);
  matchCount({a: {$bitsAllClear: []}}, 5);
  matchCount({a: {$bitsAllClear: [0]}}, 3);
  matchCount({a: {$bitsAllClear: [4]}}, 2);
  matchCount({a: {$bitsAllClear: [1, 7]}}, 3);
  matchCount({a: {$bitsAllClear: [0, 1, 2, 3, 4, 5, 6, 7]}}, 1);
  matchCount({a: {$bitsAnySet: []}}, 0);
  matchCount({a: {$bitsAnySet: [1, 3]}}, 3);
  matchCount({a: {$bitsAnySet: [0, 1, 2, 3, 4, 5, 6, 7]}}, 4);
  matchCount({a: {$bitsAnyClear: []}}, 0);
  matchCount({a: {$bitsAnyClear: [1, 4]}}, 3);
  matchCount({a: {$bitsAnyClear: [3, 4]}}, 3);
  matchCount({a: {$bitsAnyClear: [0, 1, 2, 3, 4, 5, 6, 7]}}, 4);

  // Tests with multiple predicates.
  matchCount({a: {$bitsAllSet: 54, $bitsAllClear: 201}}, 1);

  // Tests on negative numbers

  c.remove({});
  c.insert({a: -0});
  c.insert({a: -1});
  c.insert({a: -54});

  // Tests with bitmask.
  matchCount({a: {$bitsAllSet: 0}}, 3);
  matchCount({a: {$bitsAllSet: 2}}, 2);
  matchCount({a: {$bitsAllSet: 127}}, 1);
  matchCount({a: {$bitsAllSet: 74}}, 2);
  matchCount({a: {$bitsAllClear: 0}}, 3);
  matchCount({a: {$bitsAllClear: 53}}, 2);
  matchCount({a: {$bitsAllClear: 127}}, 1);
  matchCount({a: {$bitsAnySet: 0}}, 0);
  matchCount({a: {$bitsAnySet: 2}}, 2);
  matchCount({a: {$bitsAnySet: 127}}, 2);
  matchCount({a: {$bitsAnyClear: 0}}, 0);
  matchCount({a: {$bitsAnyClear: 53}}, 2);
  matchCount({a: {$bitsAnyClear: 127}}, 2);

  // Tests with array of bit positions.
  const allPositions = [];
  for (let i = 0; i < 64; i++) {
    allPositions.push(i);
  }

  matchCount({a: {$bitsAllSet: []}}, 3);
  matchCount({a: {$bitsAllSet: [1]}}, 2);
  matchCount({a: {$bitsAllSet: allPositions}}, 1);
  matchCount({a: {$bitsAllSet: [1, 7, 6, 3, 100]}}, 2);
  matchCount({a: {$bitsAllClear: []}}, 3);
  matchCount({a: {$bitsAllClear: [5, 4, 2, 0]}}, 2);
  matchCount({a: {$bitsAllClear: allPositions}}, 1);
  matchCount({a: {$bitsAnySet: []}}, 0);
  matchCount({a: {$bitsAnySet: [1]}}, 2);
  matchCount({a: {$bitsAnySet: allPositions}}, 2);
  matchCount({a: {$bitsAnyClear: []}}, 0);
  matchCount({a: {$bitsAnyClear: [0, 2, 4, 5, 100]}}, 2);
  matchCount({a: {$bitsAnyClear: allPositions}}, 2);

  // Tests with multiple predicates.
  matchCount({a: {$bitsAllSet: 74, $bitsAllClear: 53}}, 1);

  // Tests on BinData.

  c.remove({});
  c.insert({a: EJSON.parse('{"$binary": "AAAAAAAAAAAAAAAAAAAAAAAAAAAA"}')});
  c.insert({a: EJSON.parse('{"$binary": "AANgAAAAAAAAAAAAAAAAAAAAAAAA"}')});
  c.insert({a: EJSON.parse('{"$binary": "JANgqwetkqwklEWRbWERKKJREtbq"}')});
  c.insert({a: EJSON.parse('{"$binary": "////////////////////////////"}')});

  // Tests with binary string bitmask.
  matchCount({a: {$bitsAllSet: EJSON.parse('{"$binary": "AAAAAAAAAAAAAAAAAAAAAAAAAAAA"}')}}, 4);
  matchCount({a: {$bitsAllSet: EJSON.parse('{"$binary": "AANgAAAAAAAAAAAAAAAAAAAAAAAA"}')}}, 3);
  matchCount({a: {$bitsAllSet: EJSON.parse('{"$binary": "JANgqwetkqwklEWRbWERKKJREtbq"}')}}, 2);
  matchCount({a: {$bitsAllSet: EJSON.parse('{"$binary": "////////////////////////////"}')}}, 1);
  matchCount({a: {$bitsAllClear: EJSON.parse('{"$binary": "AAAAAAAAAAAAAAAAAAAAAAAAAAAA"}')}}, 4);
  matchCount({a: {$bitsAllClear: EJSON.parse('{"$binary": "AAyfAAAAAAAAAAAAAAAAAAAAAAAA"}')}}, 3);
  matchCount({a: {$bitsAllClear: EJSON.parse('{"$binary": "JAyfqwetkqwklEWRbWERKKJREtbq"}')}}, 2);
  matchCount({a: {$bitsAllClear: EJSON.parse('{"$binary": "////////////////////////////"}')}}, 1);
  matchCount({a: {$bitsAnySet: EJSON.parse('{"$binary": "AAAAAAAAAAAAAAAAAAAAAAAAAAAA"}')}}, 0);
  matchCount({a: {$bitsAnySet: EJSON.parse('{"$binary": "AAyfAAAAAAAAAAAAAAAAAAAAAAAA"}')}}, 1);
  matchCount({a: {$bitsAnySet: EJSON.parse('{"$binary": "JAyfqwetkqwklEWRbWERKKJREtbq"}')}}, 2);
  matchCount({a: {$bitsAnySet: EJSON.parse('{"$binary": "////////////////////////////"}')}}, 3);
  matchCount({a: {$bitsAnyClear: EJSON.parse('{"$binary": "AAAAAAAAAAAAAAAAAAAAAAAAAAAA"}')}}, 0);
  matchCount({a: {$bitsAnyClear: EJSON.parse('{"$binary": "AANgAAAAAAAAAAAAAAAAAAAAAAAA"}')}}, 1);
  matchCount({a: {$bitsAnyClear: EJSON.parse('{"$binary": "JANgqwetkqwklEWRbWERKKJREtbq"}')}}, 2);
  matchCount({a: {$bitsAnyClear: EJSON.parse('{"$binary": "////////////////////////////"}')}}, 3);

  // Tests with multiple predicates.
  matchCount({
    a: {
      $bitsAllSet: EJSON.parse('{"$binary": "AANgAAAAAAAAAAAAAAAAAAAAAAAA"}'),
      $bitsAllClear: EJSON.parse('{"$binary": "//yf////////////////////////"}'),
    },
  }, 1);

  c.remove({});

  nomatch({a: {$bitsAllSet: 1}}, {a: false});
  nomatch({a: {$bitsAllSet: 1}}, {a: NaN});
  nomatch({a: {$bitsAllSet: 1}}, {a: Infinity});
  nomatch({a: {$bitsAllSet: 1}}, {a: null});
  nomatch({a: {$bitsAllSet: 1}}, {a: 'asdf'});
  nomatch({a: {$bitsAllSet: 1}}, {a: ['a', 'b']});
  nomatch({a: {$bitsAllSet: 1}}, {a: {foo: 'bar'}});
  nomatch({a: {$bitsAllSet: 1}}, {a: 1.2});
  nomatch({a: {$bitsAllSet: 1}}, {a: '1'});

  [
    false,
    NaN,
    Infinity,
    null,
    'asdf',
    ['a', 'b'],
    {foo: 'bar'},
    1.2,
    '1',
    [0, -1],
  ].forEach(badValue => {
    test.throws(() => {
      match({a: {$bitsAllSet: badValue}}, {a: 42});
    });
  });

  // $type
  match({a: {$type: 1}}, {a: 1.1});
  match({a: {$type: 'double'}}, {a: 1.1});
  match({a: {$type: 1}}, {a: 1});
  nomatch({a: {$type: 1}}, {a: '1'});
  match({a: {$type: 2}}, {a: '1'});
  match({a: {$type: 'string'}}, {a: '1'});
  nomatch({a: {$type: 2}}, {a: 1});
  match({a: {$type: 3}}, {a: {}});
  match({a: {$type: 'object'}}, {a: {}});
  match({a: {$type: 3}}, {a: {b: 2}});
  nomatch({a: {$type: 3}}, {a: []});
  nomatch({a: {$type: 3}}, {a: [1]});
  nomatch({a: {$type: 3}}, {a: null});
  match({a: {$type: 5}}, {a: EJSON.newBinary(0)});
  match({a: {$type: 'binData'}}, {a: EJSON.newBinary(0)});
  match({a: {$type: 5}}, {a: EJSON.newBinary(4)});
  nomatch({a: {$type: 5}}, {a: []});
  nomatch({a: {$type: 5}}, {a: [42]});
  match({a: {$type: 7}}, {a: new MongoID.ObjectID()});
  match({a: {$type: 'objectId'}}, {a: new MongoID.ObjectID()});
  nomatch({a: {$type: 7}}, {a: '1234567890abcd1234567890'});
  match({a: {$type: 8}}, {a: true});
  match({a: {$type: 'bool'}}, {a: true});
  match({a: {$type: 8}}, {a: false});
  nomatch({a: {$type: 8}}, {a: 'true'});
  nomatch({a: {$type: 8}}, {a: 0});
  nomatch({a: {$type: 8}}, {a: null});
  nomatch({a: {$type: 8}}, {a: ''});
  nomatch({a: {$type: 8}}, {});
  match({a: {$type: 9}}, {a: new Date});
  match({a: {$type: 'date'}}, {a: new Date});
  nomatch({a: {$type: 9}}, {a: +new Date});
  match({a: {$type: 10}}, {a: null});
  match({a: {$type: 'null'}}, {a: null});
  nomatch({a: {$type: 10}}, {a: false});
  nomatch({a: {$type: 10}}, {a: ''});
  nomatch({a: {$type: 10}}, {a: 0});
  nomatch({a: {$type: 10}}, {});
  match({a: {$type: 11}}, {a: /x/});
  match({a: {$type: 'regex'}}, {a: /x/});
  nomatch({a: {$type: 11}}, {a: 'x'});
  nomatch({a: {$type: 11}}, {});

  // The normal rule for {$type:4} (4 means array) is that it NOT good enough to
  // just have an array that's the leaf that matches the path.  (An array inside
  // that array is good, though.)
  nomatch({a: {$type: 4}}, {a: []});
  nomatch({a: {$type: 4}}, {a: [1]}); // tested against mongodb
  match({a: {$type: 1}}, {a: [1]});
  nomatch({a: {$type: 2}}, {a: [1]});
  match({a: {$type: 1}}, {a: ['1', 1]});
  match({a: {$type: 2}}, {a: ['1', 1]});
  nomatch({a: {$type: 3}}, {a: ['1', 1]});
  nomatch({a: {$type: 4}}, {a: ['1', 1]});
  nomatch({a: {$type: 1}}, {a: ['1', []]});
  match({a: {$type: 2}}, {a: ['1', []]});
  match({a: {$type: 4}}, {a: ['1', []]}); // tested against mongodb
  // An exception to the normal rule is that an array found via numeric index is
  // examined itself, and its elements are not.
  match({'a.0': {$type: 4}}, {a: [[0]]});
  match({'a.0': {$type: 'array'}}, {a: [[0]]});
  nomatch({'a.0': {$type: 1}}, {a: [[0]]});

  // invalid types should throw errors
  test.throws(() => {
    match({a: {$type: 'foo'}}, {a: 1});
  });
  test.throws(() => {
    match({a: {$type: -2}}, {a: 1});
  });
  test.throws(() => {
    match({a: {$type: 0}}, {a: 1});
  });
  test.throws(() => {
    match({a: {$type: 20}}, {a: 1});
  });

  // regular expressions
  match({a: /a/}, {a: 'cat'});
  nomatch({a: /a/}, {a: 'cut'});
  nomatch({a: /a/}, {a: 'CAT'});
  match({a: /a/i}, {a: 'CAT'});
  match({a: /a/}, {a: ['foo', 'bar']});  // search within array...
  nomatch({a: /,/}, {a: ['foo', 'bar']});  // but not by stringifying
  match({a: {$regex: 'a'}}, {a: ['foo', 'bar']});
  nomatch({a: {$regex: ','}}, {a: ['foo', 'bar']});
  match({a: {$regex: /a/}}, {a: 'cat'});
  nomatch({a: {$regex: /a/}}, {a: 'cut'});
  nomatch({a: {$regex: /a/}}, {a: 'CAT'});
  match({a: {$regex: /a/i}}, {a: 'CAT'});
  match({a: {$regex: /a/, $options: 'i'}}, {a: 'CAT'}); // tested
  match({a: {$regex: /a/i, $options: 'i'}}, {a: 'CAT'}); // tested
  nomatch({a: {$regex: /a/i, $options: ''}}, {a: 'CAT'}); // tested
  match({a: {$regex: 'a'}}, {a: 'cat'});
  nomatch({a: {$regex: 'a'}}, {a: 'cut'});
  nomatch({a: {$regex: 'a'}}, {a: 'CAT'});
  match({a: {$regex: 'a', $options: 'i'}}, {a: 'CAT'});
  match({a: {$regex: '', $options: 'i'}}, {a: 'foo'});
  nomatch({a: {$regex: '', $options: 'i'}}, {});
  nomatch({a: {$regex: '', $options: 'i'}}, {a: 5});
  nomatch({a: /undefined/}, {});
  nomatch({a: {$regex: 'undefined'}}, {});
  nomatch({a: /xxx/}, {});
  nomatch({a: {$regex: 'xxx'}}, {});

  // GitHub issue #2817:
  // Regexps with a global flag ('g') keep a state when tested against the same
  // string. Selector shouldn't return different result for similar documents
  // because of this state.
  const reusedRegexp = /sh/ig;
  match({a: reusedRegexp}, {a: 'Shorts'});
  match({a: reusedRegexp}, {a: 'Shorts'});
  match({a: reusedRegexp}, {a: 'Shorts'});

  match({a: {$regex: reusedRegexp}}, {a: 'Shorts'});
  match({a: {$regex: reusedRegexp}}, {a: 'Shorts'});
  match({a: {$regex: reusedRegexp}}, {a: 'Shorts'});

  test.throws(() => {
    match({a: {$options: 'i'}}, {a: 12});
  });

  match({a: /a/}, {a: ['dog', 'cat']});
  nomatch({a: /a/}, {a: ['dog', 'puppy']});

  // we don't support regexps in minimongo very well (eg, there's no EJSON
  // encoding so it won't go over the wire), but run these tests anyway
  match({a: /a/}, {a: /a/});
  match({a: /a/}, {a: ['x', /a/]});
  nomatch({a: /a/}, {a: /a/i});
  nomatch({a: /a/m}, {a: /a/});
  nomatch({a: /a/}, {a: /b/});
  nomatch({a: /5/}, {a: 5});
  nomatch({a: /t/}, {a: true});
  match({a: /m/i}, {a: ['x', 'xM']});

  test.throws(() => {
    match({a: {$regex: /a/, $options: 'x'}}, {a: 'cat'});
  });
  test.throws(() => {
    match({a: {$regex: /a/, $options: 's'}}, {a: 'cat'});
  });

  // $not
  match({x: {$not: {$gt: 7}}}, {x: 6});
  nomatch({x: {$not: {$gt: 7}}}, {x: 8});
  match({x: {$not: {$lt: 10, $gt: 7}}}, {x: 11});
  nomatch({x: {$not: {$lt: 10, $gt: 7}}}, {x: 9});
  match({x: {$not: {$lt: 10, $gt: 7}}}, {x: 6});

  match({x: {$not: {$gt: 7}}}, {x: [2, 3, 4]});
  match({'x.y': {$not: {$gt: 7}}}, {x: [{y: 2}, {y: 3}, {y: 4}]});
  nomatch({x: {$not: {$gt: 7}}}, {x: [2, 3, 4, 10]});
  nomatch({'x.y': {$not: {$gt: 7}}}, {x: [{y: 2}, {y: 3}, {y: 4}, {y: 10}]});

  match({x: {$not: /a/}}, {x: 'dog'});
  nomatch({x: {$not: /a/}}, {x: 'cat'});
  match({x: {$not: /a/}}, {x: ['dog', 'puppy']});
  nomatch({x: {$not: /a/}}, {x: ['kitten', 'cat']});

  // dotted keypaths: bare values
  match({'a.b': 1}, {a: {b: 1}});
  nomatch({'a.b': 1}, {a: {b: 2}});
  match({'a.b': [1, 2, 3]}, {a: {b: [1, 2, 3]}});
  nomatch({'a.b': [1, 2, 3]}, {a: {b: [4]}});
  match({'a.b': /a/}, {a: {b: 'cat'}});
  nomatch({'a.b': /a/}, {a: {b: 'dog'}});
  match({'a.b.c': null}, {});
  match({'a.b.c': null}, {a: 1});
  match({'a.b': null}, {a: 1});
  match({'a.b.c': null}, {a: {b: 4}});

  // dotted keypaths, nulls, numeric indices, arrays
  nomatch({'a.b': null}, {a: [1]});
  match({'a.b': []}, {a: {b: []}});
  const big = {a: [{b: 1}, 2, {}, {b: [3, 4]}]};
  match({'a.b': 1}, big);
  match({'a.b': [3, 4]}, big);
  match({'a.b': 3}, big);
  match({'a.b': 4}, big);
  match({'a.b': null}, big);  // matches on slot 2
  match({'a.1': 8}, {a: [7, 8, 9]});
  nomatch({'a.1': 7}, {a: [7, 8, 9]});
  nomatch({'a.1': null}, {a: [7, 8, 9]});
  match({'a.1': [8, 9]}, {a: [7, [8, 9]]});
  nomatch({'a.1': 6}, {a: [[6, 7], [8, 9]]});
  nomatch({'a.1': 7}, {a: [[6, 7], [8, 9]]});
  nomatch({'a.1': 8}, {a: [[6, 7], [8, 9]]});
  nomatch({'a.1': 9}, {a: [[6, 7], [8, 9]]});
  match({'a.1': 2}, {a: [0, {1: 2}, 3]});
  match({'a.1': {1: 2}}, {a: [0, {1: 2}, 3]});
  match({'x.1.y': 8}, {x: [7, {y: 8}, 9]});
  // comes from trying '1' as key in the plain object
  match({'x.1.y': null}, {x: [7, {y: 8}, 9]});
  match({'a.1.b': 9}, {a: [7, {b: 9}, {1: {b: 'foo'}}]});
  match({'a.1.b': 'foo'}, {a: [7, {b: 9}, {1: {b: 'foo'}}]});
  match({'a.1.b': null}, {a: [7, {b: 9}, {1: {b: 'foo'}}]});
  match({'a.1.b': 2}, {a: [1, [{b: 2}], 3]});
  nomatch({'a.1.b': null}, {a: [1, [{b: 2}], 3]});
  // this is new behavior in mongo 2.5
  nomatch({'a.0.b': null}, {a: [5]});
  match({'a.1': 4}, {a: [{1: 4}, 5]});
  match({'a.1': 5}, {a: [{1: 4}, 5]});
  nomatch({'a.1': null}, {a: [{1: 4}, 5]});
  match({'a.1.foo': 4}, {a: [{1: {foo: 4}}, {foo: 5}]});
  match({'a.1.foo': 5}, {a: [{1: {foo: 4}}, {foo: 5}]});
  match({'a.1.foo': null}, {a: [{1: {foo: 4}}, {foo: 5}]});

  // trying to access a dotted field that is undefined at some point
  // down the chain
  nomatch({'a.b': 1}, {x: 2});
  nomatch({'a.b.c': 1}, {a: {x: 2}});
  nomatch({'a.b.c': 1}, {a: {b: {x: 2}}});
  nomatch({'a.b.c': 1}, {a: {b: 1}});
  nomatch({'a.b.c': 1}, {a: {b: 0}});

  // dotted keypaths: literal objects
  match({'a.b': {c: 1}}, {a: {b: {c: 1}}});
  nomatch({'a.b': {c: 1}}, {a: {b: {c: 2}}});
  nomatch({'a.b': {c: 1}}, {a: {b: 2}});
  match({'a.b': {c: 1, d: 2}}, {a: {b: {c: 1, d: 2}}});
  nomatch({'a.b': {c: 1, d: 2}}, {a: {b: {c: 1, d: 1}}});
  nomatch({'a.b': {c: 1, d: 2}}, {a: {b: {d: 2}}});

  // dotted keypaths: $ operators
  match({'a.b': {$in: [1, 2, 3]}}, {a: {b: [2]}}); // tested against mongodb
  match({'a.b': {$in: [{x: 1}, {x: 2}, {x: 3}]}}, {a: {b: [{x: 2}]}});
  match({'a.b': {$in: [1, 2, 3]}}, {a: {b: [4, 2]}});
  nomatch({'a.b': {$in: [1, 2, 3]}}, {a: {b: [4]}});

  // $or
  test.throws(() => {
    match({$or: []}, {});
  });
  test.throws(() => {
    match({$or: [5]}, {});
  });
  test.throws(() => {
    match({$or: []}, {a: 1});
  });
  match({$or: [{a: 1}]}, {a: 1});
  nomatch({$or: [{b: 2}]}, {a: 1});
  match({$or: [{a: 1}, {b: 2}]}, {a: 1});
  nomatch({$or: [{c: 3}, {d: 4}]}, {a: 1});
  match({$or: [{a: 1}, {b: 2}]}, {a: [1, 2, 3]});
  nomatch({$or: [{a: 1}, {b: 2}]}, {c: [1, 2, 3]});
  nomatch({$or: [{a: 1}, {b: 2}]}, {a: [2, 3, 4]});
  match({$or: [{a: 1}, {a: 2}]}, {a: 1});
  match({$or: [{a: 1}, {a: 2}], b: 2}, {a: 1, b: 2});
  nomatch({$or: [{a: 2}, {a: 3}], b: 2}, {a: 1, b: 2});
  nomatch({$or: [{a: 1}, {a: 2}], b: 3}, {a: 1, b: 2});

  // Combining $or with equality
  match({x: 1, $or: [{a: 1}, {b: 1}]}, {x: 1, b: 1});
  match({$or: [{a: 1}, {b: 1}], x: 1}, {x: 1, b: 1});
  nomatch({x: 1, $or: [{a: 1}, {b: 1}]}, {b: 1});
  nomatch({x: 1, $or: [{a: 1}, {b: 1}]}, {x: 1});

  // $or and $lt, $lte, $gt, $gte
  match({$or: [{a: {$lte: 1}}, {a: 2}]}, {a: 1});
  nomatch({$or: [{a: {$lt: 1}}, {a: 2}]}, {a: 1});
  match({$or: [{a: {$gte: 1}}, {a: 2}]}, {a: 1});
  nomatch({$or: [{a: {$gt: 1}}, {a: 2}]}, {a: 1});
  match({$or: [{b: {$gt: 1}}, {b: {$lt: 3}}]}, {b: 2});
  nomatch({$or: [{b: {$lt: 1}}, {b: {$gt: 3}}]}, {b: 2});

  // $or and $in
  match({$or: [{a: {$in: [1, 2, 3]}}]}, {a: 1});
  nomatch({$or: [{a: {$in: [4, 5, 6]}}]}, {a: 1});
  match({$or: [{a: {$in: [1, 2, 3]}}, {b: 2}]}, {a: 1});
  match({$or: [{a: {$in: [1, 2, 3]}}, {b: 2}]}, {b: 2});
  nomatch({$or: [{a: {$in: [1, 2, 3]}}, {b: 2}]}, {c: 3});
  match({$or: [{a: {$in: [1, 2, 3]}}, {b: {$in: [1, 2, 3]}}]}, {b: 2});
  nomatch({$or: [{a: {$in: [1, 2, 3]}}, {b: {$in: [4, 5, 6]}}]}, {b: 2});

  // $or and $nin
  nomatch({$or: [{a: {$nin: [1, 2, 3]}}]}, {a: 1});
  match({$or: [{a: {$nin: [4, 5, 6]}}]}, {a: 1});
  nomatch({$or: [{a: {$nin: [1, 2, 3]}}, {b: 2}]}, {a: 1});
  match({$or: [{a: {$nin: [1, 2, 3]}}, {b: 2}]}, {b: 2});
  match({$or: [{a: {$nin: [1, 2, 3]}}, {b: 2}]}, {c: 3});
  match({$or: [{a: {$nin: [1, 2, 3]}}, {b: {$nin: [1, 2, 3]}}]}, {b: 2});
  nomatch({$or: [{a: {$nin: [1, 2, 3]}}, {b: {$nin: [1, 2, 3]}}]}, {a: 1, b: 2});
  match({$or: [{a: {$nin: [1, 2, 3]}}, {b: {$nin: [4, 5, 6]}}]}, {b: 2});

  // $or and dot-notation
  match({$or: [{'a.b': 1}, {'a.b': 2}]}, {a: {b: 1}});
  match({$or: [{'a.b': 1}, {'a.c': 1}]}, {a: {b: 1}});
  nomatch({$or: [{'a.b': 2}, {'a.c': 1}]}, {a: {b: 1}});

  // $or and nested objects
  match({$or: [{a: {b: 1, c: 2}}, {a: {b: 2, c: 1}}]}, {a: {b: 1, c: 2}});
  nomatch({$or: [{a: {b: 1, c: 3}}, {a: {b: 2, c: 1}}]}, {a: {b: 1, c: 2}});

  // $or and regexes
  match({$or: [{a: /a/}]}, {a: 'cat'});
  nomatch({$or: [{a: /o/}]}, {a: 'cat'});
  match({$or: [{a: /a/}, {a: /o/}]}, {a: 'cat'});
  nomatch({$or: [{a: /i/}, {a: /o/}]}, {a: 'cat'});
  match({$or: [{a: /i/}, {b: /o/}]}, {a: 'cat', b: 'dog'});

  // $or and $ne
  match({$or: [{a: {$ne: 1}}]}, {});
  nomatch({$or: [{a: {$ne: 1}}]}, {a: 1});
  match({$or: [{a: {$ne: 1}}]}, {a: 2});
  match({$or: [{a: {$ne: 1}}]}, {b: 1});
  match({$or: [{a: {$ne: 1}}, {a: {$ne: 2}}]}, {a: 1});
  match({$or: [{a: {$ne: 1}}, {b: {$ne: 1}}]}, {a: 1});
  nomatch({$or: [{a: {$ne: 1}}, {b: {$ne: 2}}]}, {a: 1, b: 2});

  // $or and $not
  match({$or: [{a: {$not: {$mod: [10, 1]}}}]}, {});
  nomatch({$or: [{a: {$not: {$mod: [10, 1]}}}]}, {a: 1});
  match({$or: [{a: {$not: {$mod: [10, 1]}}}]}, {a: 2});
  match({$or: [{a: {$not: {$mod: [10, 1]}}}, {a: {$not: {$mod: [10, 2]}}}]}, {a: 1});
  nomatch({$or: [{a: {$not: {$mod: [10, 1]}}}, {a: {$mod: [10, 2]}}]}, {a: 1});
  match({$or: [{a: {$not: {$mod: [10, 1]}}}, {a: {$mod: [10, 2]}}]}, {a: 2});
  match({$or: [{a: {$not: {$mod: [10, 1]}}}, {a: {$mod: [10, 2]}}]}, {a: 3});
  // this is possibly an open-ended task, so we stop here ...

  // $nor
  test.throws(() => {
    match({$nor: []}, {});
  });
  test.throws(() => {
    match({$nor: [5]}, {});
  });
  test.throws(() => {
    match({$nor: []}, {a: 1});
  });
  nomatch({$nor: [{a: 1}]}, {a: 1});
  match({$nor: [{b: 2}]}, {a: 1});
  nomatch({$nor: [{a: 1}, {b: 2}]}, {a: 1});
  match({$nor: [{c: 3}, {d: 4}]}, {a: 1});
  nomatch({$nor: [{a: 1}, {b: 2}]}, {a: [1, 2, 3]});
  match({$nor: [{a: 1}, {b: 2}]}, {c: [1, 2, 3]});
  match({$nor: [{a: 1}, {b: 2}]}, {a: [2, 3, 4]});
  nomatch({$nor: [{a: 1}, {a: 2}]}, {a: 1});

  // $nor and $lt, $lte, $gt, $gte
  nomatch({$nor: [{a: {$lte: 1}}, {a: 2}]}, {a: 1});
  match({$nor: [{a: {$lt: 1}}, {a: 2}]}, {a: 1});
  nomatch({$nor: [{a: {$gte: 1}}, {a: 2}]}, {a: 1});
  match({$nor: [{a: {$gt: 1}}, {a: 2}]}, {a: 1});
  nomatch({$nor: [{b: {$gt: 1}}, {b: {$lt: 3}}]}, {b: 2});
  match({$nor: [{b: {$lt: 1}}, {b: {$gt: 3}}]}, {b: 2});

  // $nor and $in
  nomatch({$nor: [{a: {$in: [1, 2, 3]}}]}, {a: 1});
  match({$nor: [{a: {$in: [4, 5, 6]}}]}, {a: 1});
  nomatch({$nor: [{a: {$in: [1, 2, 3]}}, {b: 2}]}, {a: 1});
  nomatch({$nor: [{a: {$in: [1, 2, 3]}}, {b: 2}]}, {b: 2});
  match({$nor: [{a: {$in: [1, 2, 3]}}, {b: 2}]}, {c: 3});
  nomatch({$nor: [{a: {$in: [1, 2, 3]}}, {b: {$in: [1, 2, 3]}}]}, {b: 2});
  match({$nor: [{a: {$in: [1, 2, 3]}}, {b: {$in: [4, 5, 6]}}]}, {b: 2});

  // $nor and $nin
  match({$nor: [{a: {$nin: [1, 2, 3]}}]}, {a: 1});
  nomatch({$nor: [{a: {$nin: [4, 5, 6]}}]}, {a: 1});
  match({$nor: [{a: {$nin: [1, 2, 3]}}, {b: 2}]}, {a: 1});
  nomatch({$nor: [{a: {$nin: [1, 2, 3]}}, {b: 2}]}, {b: 2});
  nomatch({$nor: [{a: {$nin: [1, 2, 3]}}, {b: 2}]}, {c: 3});
  nomatch({$nor: [{a: {$nin: [1, 2, 3]}}, {b: {$nin: [1, 2, 3]}}]}, {b: 2});
  match({$nor: [{a: {$nin: [1, 2, 3]}}, {b: {$nin: [1, 2, 3]}}]}, {a: 1, b: 2});
  nomatch({$nor: [{a: {$nin: [1, 2, 3]}}, {b: {$nin: [4, 5, 6]}}]}, {b: 2});

  // $nor and dot-notation
  nomatch({$nor: [{'a.b': 1}, {'a.b': 2}]}, {a: {b: 1}});
  nomatch({$nor: [{'a.b': 1}, {'a.c': 1}]}, {a: {b: 1}});
  match({$nor: [{'a.b': 2}, {'a.c': 1}]}, {a: {b: 1}});

  // $nor and nested objects
  nomatch({$nor: [{a: {b: 1, c: 2}}, {a: {b: 2, c: 1}}]}, {a: {b: 1, c: 2}});
  match({$nor: [{a: {b: 1, c: 3}}, {a: {b: 2, c: 1}}]}, {a: {b: 1, c: 2}});

  // $nor and regexes
  nomatch({$nor: [{a: /a/}]}, {a: 'cat'});
  match({$nor: [{a: /o/}]}, {a: 'cat'});
  nomatch({$nor: [{a: /a/}, {a: /o/}]}, {a: 'cat'});
  match({$nor: [{a: /i/}, {a: /o/}]}, {a: 'cat'});
  nomatch({$nor: [{a: /i/}, {b: /o/}]}, {a: 'cat', b: 'dog'});

  // $nor and $ne
  nomatch({$nor: [{a: {$ne: 1}}]}, {});
  match({$nor: [{a: {$ne: 1}}]}, {a: 1});
  nomatch({$nor: [{a: {$ne: 1}}]}, {a: 2});
  nomatch({$nor: [{a: {$ne: 1}}]}, {b: 1});
  nomatch({$nor: [{a: {$ne: 1}}, {a: {$ne: 2}}]}, {a: 1});
  nomatch({$nor: [{a: {$ne: 1}}, {b: {$ne: 1}}]}, {a: 1});
  match({$nor: [{a: {$ne: 1}}, {b: {$ne: 2}}]}, {a: 1, b: 2});

  // $nor and $not
  nomatch({$nor: [{a: {$not: {$mod: [10, 1]}}}]}, {});
  match({$nor: [{a: {$not: {$mod: [10, 1]}}}]}, {a: 1});
  nomatch({$nor: [{a: {$not: {$mod: [10, 1]}}}]}, {a: 2});
  nomatch({$nor: [{a: {$not: {$mod: [10, 1]}}}, {a: {$not: {$mod: [10, 2]}}}]}, {a: 1});
  match({$nor: [{a: {$not: {$mod: [10, 1]}}}, {a: {$mod: [10, 2]}}]}, {a: 1});
  nomatch({$nor: [{a: {$not: {$mod: [10, 1]}}}, {a: {$mod: [10, 2]}}]}, {a: 2});
  nomatch({$nor: [{a: {$not: {$mod: [10, 1]}}}, {a: {$mod: [10, 2]}}]}, {a: 3});

  // $and

  test.throws(() => {
    match({$and: []}, {});
  });
  test.throws(() => {
    match({$and: [5]}, {});
  });
  test.throws(() => {
    match({$and: []}, {a: 1});
  });
  match({$and: [{a: 1}]}, {a: 1});
  nomatch({$and: [{a: 1}, {a: 2}]}, {a: 1});
  nomatch({$and: [{a: 1}, {b: 1}]}, {a: 1});
  match({$and: [{a: 1}, {b: 2}]}, {a: 1, b: 2});
  nomatch({$and: [{a: 1}, {b: 1}]}, {a: 1, b: 2});
  match({$and: [{a: 1}, {b: 2}], c: 3}, {a: 1, b: 2, c: 3});
  nomatch({$and: [{a: 1}, {b: 2}], c: 4}, {a: 1, b: 2, c: 3});

  // $and and regexes
  match({$and: [{a: /a/}]}, {a: 'cat'});
  match({$and: [{a: /a/i}]}, {a: 'CAT'});
  nomatch({$and: [{a: /o/}]}, {a: 'cat'});
  nomatch({$and: [{a: /a/}, {a: /o/}]}, {a: 'cat'});
  match({$and: [{a: /a/}, {b: /o/}]}, {a: 'cat', b: 'dog'});
  nomatch({$and: [{a: /a/}, {b: /a/}]}, {a: 'cat', b: 'dog'});

  // $and, dot-notation, and nested objects
  match({$and: [{'a.b': 1}]}, {a: {b: 1}});
  match({$and: [{a: {b: 1}}]}, {a: {b: 1}});
  nomatch({$and: [{'a.b': 2}]}, {a: {b: 1}});
  nomatch({$and: [{'a.c': 1}]}, {a: {b: 1}});
  nomatch({$and: [{'a.b': 1}, {'a.b': 2}]}, {a: {b: 1}});
  nomatch({$and: [{'a.b': 1}, {a: {b: 2}}]}, {a: {b: 1}});
  match({$and: [{'a.b': 1}, {'c.d': 2}]}, {a: {b: 1}, c: {d: 2}});
  nomatch({$and: [{'a.b': 1}, {'c.d': 1}]}, {a: {b: 1}, c: {d: 2}});
  match({$and: [{'a.b': 1}, {c: {d: 2}}]}, {a: {b: 1}, c: {d: 2}});
  nomatch({$and: [{'a.b': 1}, {c: {d: 1}}]}, {a: {b: 1}, c: {d: 2}});
  nomatch({$and: [{'a.b': 2}, {c: {d: 2}}]}, {a: {b: 1}, c: {d: 2}});
  match({$and: [{a: {b: 1}}, {c: {d: 2}}]}, {a: {b: 1}, c: {d: 2}});
  nomatch({$and: [{a: {b: 2}}, {c: {d: 2}}]}, {a: {b: 1}, c: {d: 2}});

  // $and and $in
  nomatch({$and: [{a: {$in: []}}]}, {});
  match({$and: [{a: {$in: [1, 2, 3]}}]}, {a: 1});
  nomatch({$and: [{a: {$in: [4, 5, 6]}}]}, {a: 1});
  nomatch({$and: [{a: {$in: [1, 2, 3]}}, {a: {$in: [4, 5, 6]}}]}, {a: 1});
  nomatch({$and: [{a: {$in: [1, 2, 3]}}, {b: {$in: [1, 2, 3]}}]}, {a: 1, b: 4});
  match({$and: [{a: {$in: [1, 2, 3]}}, {b: {$in: [4, 5, 6]}}]}, {a: 1, b: 4});


  // $and and $nin
  match({$and: [{a: {$nin: []}}]}, {});
  nomatch({$and: [{a: {$nin: [1, 2, 3]}}]}, {a: 1});
  match({$and: [{a: {$nin: [4, 5, 6]}}]}, {a: 1});
  nomatch({$and: [{a: {$nin: [1, 2, 3]}}, {a: {$nin: [4, 5, 6]}}]}, {a: 1});
  nomatch({$and: [{a: {$nin: [1, 2, 3]}}, {b: {$nin: [1, 2, 3]}}]}, {a: 1, b: 4});
  nomatch({$and: [{a: {$nin: [1, 2, 3]}}, {b: {$nin: [4, 5, 6]}}]}, {a: 1, b: 4});

  // $and and $lt, $lte, $gt, $gte
  match({$and: [{a: {$lt: 2}}]}, {a: 1});
  nomatch({$and: [{a: {$lt: 1}}]}, {a: 1});
  match({$and: [{a: {$lte: 1}}]}, {a: 1});
  match({$and: [{a: {$gt: 0}}]}, {a: 1});
  nomatch({$and: [{a: {$gt: 1}}]}, {a: 1});
  match({$and: [{a: {$gte: 1}}]}, {a: 1});
  match({$and: [{a: {$gt: 0}}, {a: {$lt: 2}}]}, {a: 1});
  nomatch({$and: [{a: {$gt: 1}}, {a: {$lt: 2}}]}, {a: 1});
  nomatch({$and: [{a: {$gt: 0}}, {a: {$lt: 1}}]}, {a: 1});
  match({$and: [{a: {$gte: 1}}, {a: {$lte: 1}}]}, {a: 1});
  nomatch({$and: [{a: {$gte: 2}}, {a: {$lte: 0}}]}, {a: 1});

  // $and and $ne
  match({$and: [{a: {$ne: 1}}]}, {});
  nomatch({$and: [{a: {$ne: 1}}]}, {a: 1});
  match({$and: [{a: {$ne: 1}}]}, {a: 2});
  nomatch({$and: [{a: {$ne: 1}}, {a: {$ne: 2}}]}, {a: 2});
  match({$and: [{a: {$ne: 1}}, {a: {$ne: 3}}]}, {a: 2});

  // $and and $not
  match({$and: [{a: {$not: {$gt: 2}}}]}, {a: 1});
  nomatch({$and: [{a: {$not: {$lt: 2}}}]}, {a: 1});
  match({$and: [{a: {$not: {$lt: 0}}}, {a: {$not: {$gt: 2}}}]}, {a: 1});
  nomatch({$and: [{a: {$not: {$lt: 2}}}, {a: {$not: {$gt: 0}}}]}, {a: 1});

  // $where
  match({$where: 'this.a === 1'}, {a: 1});
  match({$where: 'obj.a === 1'}, {a: 1});
  nomatch({$where: 'this.a !== 1'}, {a: 1});
  nomatch({$where: 'obj.a !== 1'}, {a: 1});
  nomatch({$where: 'this.a === 1', a: 2}, {a: 1});
  match({$where: 'this.a === 1', b: 2}, {a: 1, b: 2});
  match({$where: 'this.a === 1 && this.b === 2'}, {a: 1, b: 2});
  match({$where: 'this.a instanceof Array'}, {a: []});
  nomatch({$where: 'this.a instanceof Array'}, {a: 1});

  // reaching into array
  match({'dogs.0.name': 'Fido'}, {dogs: [{name: 'Fido'}, {name: 'Rex'}]});
  match({'dogs.1.name': 'Rex'}, {dogs: [{name: 'Fido'}, {name: 'Rex'}]});
  nomatch({'dogs.1.name': 'Fido'}, {dogs: [{name: 'Fido'}, {name: 'Rex'}]});
  match({'room.1b': 'bla'}, {room: {'1b': 'bla'}});

  match({'dogs.name': 'Fido'}, {dogs: [{name: 'Fido'}, {name: 'Rex'}]});
  match({'dogs.name': 'Rex'}, {dogs: [{name: 'Fido'}, {name: 'Rex'}]});
  match({'animals.dogs.name': 'Fido'},
    {animals: [{dogs: [{name: 'Rover'}]},
      {},
      {dogs: [{name: 'Fido'}, {name: 'Rex'}]}]});
  match({'animals.dogs.name': 'Fido'},
    {animals: [{dogs: {name: 'Rex'}},
      {dogs: {name: 'Fido'}}]});
  match({'animals.dogs.name': 'Fido'},
    {animals: [{dogs: [{name: 'Rover'}]},
      {},
      {dogs: [{name: ['Fido']}, {name: 'Rex'}]}]});
  nomatch({'dogs.name': 'Fido'}, {dogs: []});

  // $elemMatch
  match({dogs: {$elemMatch: {name: /e/}}},
    {dogs: [{name: 'Fido'}, {name: 'Rex'}]});
  nomatch({dogs: {$elemMatch: {name: /a/}}},
    {dogs: [{name: 'Fido'}, {name: 'Rex'}]});
  match({dogs: {$elemMatch: {age: {$gt: 4}}}},
    {dogs: [{name: 'Fido', age: 5}, {name: 'Rex', age: 3}]});
  match({dogs: {$elemMatch: {name: 'Fido', age: {$gt: 4}}}},
    {dogs: [{name: 'Fido', age: 5}, {name: 'Rex', age: 3}]});
  nomatch({dogs: {$elemMatch: {name: 'Fido', age: {$gt: 5}}}},
    {dogs: [{name: 'Fido', age: 5}, {name: 'Rex', age: 3}]});
  match({dogs: {$elemMatch: {name: /i/, age: {$gt: 4}}}},
    {dogs: [{name: 'Fido', age: 5}, {name: 'Rex', age: 3}]});
  nomatch({dogs: {$elemMatch: {name: /e/, age: 5}}},
    {dogs: [{name: 'Fido', age: 5}, {name: 'Rex', age: 3}]});

  // Tests for https://github.com/meteor/meteor/issues/9111.
  match(
    { dogs: { $elemMatch: { name: 'Rex' } } },
    { dogs: [{ name: 'Rex', age: 3 }] });
  nomatch(
    { dogs: { $not: { $elemMatch: { name: 'Rex' } } } },
    { dogs: [{ name: 'Rex', age: 3 }] });
  match({
    $or: [
      { dogs: { $elemMatch: { name: 'Rex' } } },
      { dogs: { $elemMatch: { name: 'Rex', age: 5 } } }
    ]
  }, {
    dogs: [{ name: 'Rex', age: 3 }]
  });
  nomatch({
    $or: [
      { dogs: { $not: { $elemMatch: { name: 'Rex' } } } },
      { dogs: { $elemMatch: { name: 'Rex', age: 5 } } }
    ]
  }, {
    dogs: [{ name: 'Rex', age: 3 }]
  });

  match({x: {$elemMatch: {y: 9}}}, {x: [{y: 9}]});
  nomatch({x: {$elemMatch: {y: 9}}}, {x: [[{y: 9}]]});
  match({x: {$elemMatch: {$gt: 5, $lt: 9}}}, {x: [8]});
  nomatch({x: {$elemMatch: {$gt: 5, $lt: 9}}}, {x: [[8]]});
  match({'a.x': {$elemMatch: {y: 9}}},
    {a: [{x: []}, {x: [{y: 9}]}]});
  nomatch({a: {$elemMatch: {x: 5}}}, {a: {x: 5}});
  match({a: {$elemMatch: {0: {$gt: 5, $lt: 9}}}}, {a: [[6]]});
  match({a: {$elemMatch: {'0.b': {$gt: 5, $lt: 9}}}}, {a: [[{b: 6}]]});
  match({a: {$elemMatch: {x: 1, $or: [{a: 1}, {b: 1}]}}},
    {a: [{x: 1, b: 1}]});
  match({a: {$elemMatch: {$or: [{a: 1}, {b: 1}], x: 1}}},
    {a: [{x: 1, b: 1}]});
  match({a: {$elemMatch: {$or: [{a: 1}, {b: 1}]}}},
    {a: [{x: 1, b: 1}]});
  match({a: {$elemMatch: {$or: [{a: 1}, {b: 1}]}}},
    {a: [{x: 1, b: 1}]});
  match({a: {$elemMatch: {$and: [{b: 1}, {x: 1}]}}},
    {a: [{x: 1, b: 1}]});
  nomatch({a: {$elemMatch: {x: 1, $or: [{a: 1}, {b: 1}]}}},
    {a: [{b: 1}]});
  nomatch({a: {$elemMatch: {x: 1, $or: [{a: 1}, {b: 1}]}}},
    {a: [{x: 1}]});
  nomatch({a: {$elemMatch: {x: 1, $or: [{a: 1}, {b: 1}]}}},
    {a: [{x: 1}, {b: 1}]});

  test.throws(() => {
    match({a: {$elemMatch: {$gte: 1, $or: [{a: 1}, {b: 1}]}}},
      {a: [{x: 1, b: 1}]});
  });

  test.throws(() => {
    match({x: {$elemMatch: {$and: [{$gt: 5, $lt: 9}]}}}, {x: [8]});
  });

  // $comment
  match({a: 5, $comment: 'asdf'}, {a: 5});
  nomatch({a: 6, $comment: 'asdf'}, {a: 5});

  // XXX still needs tests:
  // - non-scalar arguments to $gt, $lt, etc
});

Tinytest.add('minimongo - projection_compiler', test => {
  const testProjection = (projection, tests) => {
    const projection_f = LocalCollection._compileProjection(projection);
    const equalNonStrict = (a, b, desc) => {
      test.isTrue(EJSON.equals(a, b), desc);
    };

    tests.forEach(testCase => {
      equalNonStrict(projection_f(testCase[0]), testCase[1], testCase[2]);
    });
  };

  const testCompileProjectionThrows = (projection, expectedError) => {
    test.throws(() => {
      LocalCollection._compileProjection(projection);
    }, expectedError);
  };

  testProjection({ foo: 1, bar: 1 }, [
    [{ foo: 42, bar: 'something', baz: 'else' },
      { foo: 42, bar: 'something' },
      'simplest - whitelist'],

    [{ foo: { nested: 17 }, baz: {} },
      { foo: { nested: 17 } },
      'nested whitelisted field'],

    [{ _id: 'uid', bazbaz: 42 },
      { _id: 'uid' },
      'simplest whitelist - preserve _id'],
  ]);

  testProjection({ foo: 0, bar: 0 }, [
    [{ foo: 42, bar: 'something', baz: 'else' },
      { baz: 'else' },
      'simplest - blacklist'],

    [{ foo: { nested: 17 }, baz: { foo: 'something' } },
      { baz: { foo: 'something' } },
      'nested blacklisted field'],

    [{ _id: 'uid', bazbaz: 42 },
      { _id: 'uid', bazbaz: 42 },
      'simplest blacklist - preserve _id'],
  ]);

  testProjection({ _id: 0, foo: 1 }, [
    [{ foo: 42, bar: 33, _id: 'uid' },
      { foo: 42 },
      'whitelist - _id blacklisted'],
  ]);

  testProjection({ _id: 0, foo: 0 }, [
    [{ foo: 42, bar: 33, _id: 'uid' },
      { bar: 33 },
      'blacklist - _id blacklisted'],
  ]);

  testProjection({ 'foo.bar.baz': 1 }, [
    [{ foo: { meh: 'fur', bar: { baz: 42 }, tr: 1 }, bar: 33, baz: 'trolololo' },
      { foo: { bar: { baz: 42 } } },
      'whitelist nested'],

    // Behavior of this test is looked up in actual mongo
    [{ foo: { meh: 'fur', bar: 'nope', tr: 1 }, bar: 33, baz: 'trolololo' },
      { foo: {} },
      'whitelist nested - path not found in doc, different type'],

    // Behavior of this test is looked up in actual mongo
    [{ foo: { meh: 'fur', bar: [], tr: 1 }, bar: 33, baz: 'trolololo' },
      { foo: { bar: [] } },
      'whitelist nested - path not found in doc'],
  ]);

  testProjection({ 'hope.humanity': 0, 'hope.people': 0 }, [
    [{ hope: { humanity: 'lost', people: 'broken', candies: 'long live!' } },
      { hope: { candies: 'long live!' } },
      'blacklist nested'],

    [{ hope: 'new' },
      { hope: 'new' },
      'blacklist nested - path not found in doc'],
  ]);

  testProjection({ _id: 1 }, [
    [{ _id: 42, x: 1, y: { z: '2' } },
      { _id: 42 },
      '_id whitelisted'],
    [{ _id: 33 },
      { _id: 33 },
      '_id whitelisted, _id only'],
    [{ x: 1 },
      {},
      '_id whitelisted, no _id'],
  ]);

  testProjection({ _id: 0 }, [
    [{ _id: 42, x: 1, y: { z: '2' } },
      { x: 1, y: { z: '2' } },
      '_id blacklisted'],
    [{ _id: 33 },
      {},
      '_id blacklisted, _id only'],
    [{ x: 1 },
      { x: 1 },
      '_id blacklisted, no _id'],
  ]);

  testProjection({}, [
    [{ a: 1, b: 2, c: '3' },
      { a: 1, b: 2, c: '3' },
      'empty projection'],
  ]);

  testCompileProjectionThrows(
    { inc: 1, excl: 0 },
    'You cannot currently mix including and excluding fields');
  testCompileProjectionThrows(
    { _id: 1, a: 0 },
    'You cannot currently mix including and excluding fields');

  testCompileProjectionThrows(
    { a: 1, 'a.b': 1 },
    'using both of them may trigger unexpected behavior');
  testCompileProjectionThrows(
    { 'a.b.c': 1, 'a.b': 1, a: 1 },
    'using both of them may trigger unexpected behavior');

  testCompileProjectionThrows('some string', 'fields option must be an object');
});

Tinytest.add('minimongo - fetch with fields', test => {
  const c = new LocalCollection();
  Array.from({length: 30}, (x, i) => {
    c.insert({
      something: Random.id(),
      anything: {
        foo: 'bar',
        cool: 'hot',
      },
      nothing: i,
      i,
    });
  });

  // Test just a regular fetch with some projection
  let fetchResults = c.find({}, { fields: {
    something: 1,
    'anything.foo': 1,
  } }).fetch();

  test.isTrue(fetchResults.every(x => x &&
         x.something &&
         x.anything &&
         x.anything.foo &&
         x.anything.foo === 'bar' &&
         !hasOwn.call(x, 'nothing') &&
         !hasOwn.call(x.anything, 'cool')));

  // Test with a selector, even field used in the selector is excluded in the
  // projection
  fetchResults = c.find({
    nothing: { $gte: 5 },
  }, {
    fields: { nothing: 0 },
  }).fetch();

  test.isTrue(fetchResults.every(x => x &&
         x.something &&
         x.anything &&
         x.anything.foo === 'bar' &&
         x.anything.cool === 'hot' &&
         !hasOwn.call(x, 'nothing') &&
         x.i &&
         x.i >= 5));

  test.isTrue(fetchResults.length === 25);

  // Test that we can sort, based on field excluded from the projection, use
  // skip and limit as well!
  // following find will get indexes [10..20) sorted by nothing
  fetchResults = c.find({}, {
    sort: {
      nothing: 1,
    },
    limit: 10,
    skip: 10,
    fields: {
      i: 1,
      something: 1,
    },
  }).fetch();

  test.isTrue(fetchResults.every(x => x &&
         x.something &&
         x.i >= 10 && x.i < 20));

  fetchResults.forEach((x, i, arr) => {
    if (!i) return;
    test.isTrue(x.i === arr[i - 1].i + 1);
  });

  // Temporary unsupported operators
  // queries are taken from MongoDB docs examples
  test.throws(() => {
    c.find({}, { fields: { 'grades.$': 1 } });
  });
  test.throws(() => {
    c.find({}, { fields: { grades: { $elemMatch: { mean: 70 } } } });
  });
  test.throws(() => {
    c.find({}, { fields: { grades: { $slice: [20, 10] } } });
  });
});

Tinytest.add('minimongo - fetch with projection, subarrays', test => {
  // Apparently projection of type 'foo.bar.x' for
  // { foo: [ { bar: { x: 42 } }, { bar: { x: 3 } } ] }
  // should return exactly this object. More precisely, arrays are considered as
  // sets and are queried separately and then merged back to result set
  const c = new LocalCollection();

  // Insert a test object with two set fields
  c.insert({
    setA: [{
      fieldA: 42,
      fieldB: 33,
    }, {
      fieldA: 'the good',
      fieldB: 'the bad',
      fieldC: 'the ugly',
    }],
    setB: [null, {
      anotherA: { },
      anotherB: 'meh',
    }, null, {
      anotherA: 1234,
      anotherB: 431,
    }, null],
  });

  const equalNonStrict = (a, b, desc) => {
    test.isTrue(EJSON.equals(a, b), desc);
  };

  const testForProjection = (projection, expected) => {
    const fetched = c.find({}, { fields: projection }).fetch()[0];
    equalNonStrict(fetched, expected, `failed sub-set projection: ${JSON.stringify(projection)}`);
  };

  testForProjection({ 'setA.fieldA': 1, 'setB.anotherB': 1, _id: 0 },
    {
      setA: [{ fieldA: 42 }, { fieldA: 'the good' }],
      setB: [null, { anotherB: 'meh' }, null, { anotherB: 431 }, null],
    });

  testForProjection({ 'setA.fieldA': 0, 'setB.anotherA': 0, _id: 0 },
    {
      setA: [{fieldB: 33}, {fieldB: 'the bad', fieldC: 'the ugly'}],
      setB: [null, { anotherB: 'meh' }, null, { anotherB: 431 }, null],
    });

  c.remove({});
  c.insert({a: [[{b: 1, c: 2}, {b: 2, c: 4}], {b: 3, c: 5}, [{b: 4, c: 9}]]});

  testForProjection({ 'a.b': 1, _id: 0 },
    {a: [ [ { b: 1 }, { b: 2 } ], { b: 3 }, [ { b: 4 } ] ] });
  testForProjection({ 'a.b': 0, _id: 0 },
    {a: [ [ { c: 2 }, { c: 4 } ], { c: 5 }, [ { c: 9 } ] ] });
});

Tinytest.add('minimongo - fetch with projection, deep copy', test => {
  // Compiled fields projection defines the contract: returned document doesn't
  // retain anything from the passed argument.
  const doc = {
    a: { x: 42 },
    b: {
      y: { z: 33 },
    },
    c: 'asdf',
  };

  let fields = {
    a: 1,
    'b.y': 1,
  };

  let projectionFn = LocalCollection._compileProjection(fields);
  let filteredDoc = projectionFn(doc);
  doc.a.x++;
  doc.b.y.z--;
  test.equal(filteredDoc.a.x, 42, 'projection returning deep copy - including');
  test.equal(filteredDoc.b.y.z, 33, 'projection returning deep copy - including');

  fields = { c: 0 };
  projectionFn = LocalCollection._compileProjection(fields);
  filteredDoc = projectionFn(doc);

  doc.a.x = 5;
  test.equal(filteredDoc.a.x, 43, 'projection returning deep copy - excluding');
});

Tinytest.add('minimongo - observe ordered with projection', test => {
  // These tests are copy-paste from "minimongo -observe ordered",
  // slightly modified to test projection
  const operations = [];
  const cbs = log_callbacks(operations);
  let handle;

  const c = new LocalCollection();
  handle = c.find({}, {sort: {a: 1}, fields: { a: 1 }}).observe(cbs);
  test.isTrue(handle.collection === c);

  c.insert({_id: 'foo', a: 1, b: 2});
  test.equal(operations.shift(), ['added', {a: 1}, 0, null]);
  c.update({a: 1}, {$set: {a: 2, b: 1}});
  test.equal(operations.shift(), ['changed', {a: 2}, 0, {a: 1}]);
  c.insert({_id: 'bar', a: 10, c: 33});
  test.equal(operations.shift(), ['added', {a: 10}, 1, null]);
  c.update({}, {$inc: {a: 1}}, {multi: true});
  c.update({}, {$inc: {c: 1}}, {multi: true});
  test.equal(operations.shift(), ['changed', {a: 3}, 0, {a: 2}]);
  test.equal(operations.shift(), ['changed', {a: 11}, 1, {a: 10}]);
  c.update({a: 11}, {a: 1, b: 44});
  test.equal(operations.shift(), ['changed', {a: 1}, 1, {a: 11}]);
  test.equal(operations.shift(), ['moved', {a: 1}, 1, 0, 'foo']);
  c.remove({a: 2});
  test.equal(operations.shift(), undefined);
  c.remove({a: 3});
  test.equal(operations.shift(), ['removed', 'foo', 1, {a: 3}]);

  // test stop
  handle.stop();
  const idA2 = Random.id();
  c.insert({_id: idA2, a: 2});
  test.equal(operations.shift(), undefined);

  const cursor = c.find({}, {fields: {a: 1, _id: 0}});
  test.throws(() => {
    cursor.observeChanges({added() {}});
  });
  test.throws(() => {
    cursor.observe({added() {}});
  });

  // test initial inserts (and backwards sort)
  handle = c.find({}, {sort: {a: -1}, fields: { a: 1 } }).observe(cbs);
  test.equal(operations.shift(), ['added', {a: 2}, 0, null]);
  test.equal(operations.shift(), ['added', {a: 1}, 1, null]);
  handle.stop();

  // test _suppress_initial
  handle = c.find({}, {sort: {a: -1}, fields: { a: 1 }}).observe(Object.assign(cbs, {_suppress_initial: true}));
  test.equal(operations.shift(), undefined);
  c.insert({a: 100, b: { foo: 'bar' }});
  test.equal(operations.shift(), ['added', {a: 100}, 0, idA2]);
  handle.stop();

  // test skip and limit.
  c.remove({});
  handle = c.find({}, {sort: {a: 1}, skip: 1, limit: 2, fields: { blacklisted: 0 }}).observe(cbs);
  test.equal(operations.shift(), undefined);
  c.insert({a: 1, blacklisted: 1324});
  test.equal(operations.shift(), undefined);
  c.insert({_id: 'foo', a: 2, blacklisted: ['something']});
  test.equal(operations.shift(), ['added', {a: 2}, 0, null]);
  c.insert({a: 3, blacklisted: { 2: 3 }});
  test.equal(operations.shift(), ['added', {a: 3}, 1, null]);
  c.insert({a: 4, blacklisted: 6});
  test.equal(operations.shift(), undefined);
  c.update({a: 1}, {a: 0, blacklisted: 4444});
  test.equal(operations.shift(), undefined);
  c.update({a: 0}, {a: 5, blacklisted: 11111});
  test.equal(operations.shift(), ['removed', 'foo', 0, {a: 2}]);
  test.equal(operations.shift(), ['added', {a: 4}, 1, null]);
  c.update({a: 3}, {a: 3.5, blacklisted: 333.4444});
  test.equal(operations.shift(), ['changed', {a: 3.5}, 0, {a: 3}]);
  handle.stop();

  // test _no_indices

  c.remove({});
  handle = c.find({}, {sort: {a: 1}, fields: { a: 1 }}).observe(Object.assign(cbs, {_no_indices: true}));
  c.insert({_id: 'foo', a: 1, zoo: 'crazy'});
  test.equal(operations.shift(), ['added', {a: 1}, -1, null]);
  c.update({a: 1}, {$set: {a: 2, foobar: 'player'}});
  test.equal(operations.shift(), ['changed', {a: 2}, -1, {a: 1}]);
  c.insert({a: 10, b: 123.45});
  test.equal(operations.shift(), ['added', {a: 10}, -1, null]);
  c.update({}, {$inc: {a: 1, b: 2}}, {multi: true});
  test.equal(operations.shift(), ['changed', {a: 3}, -1, {a: 2}]);
  test.equal(operations.shift(), ['changed', {a: 11}, -1, {a: 10}]);
  c.update({a: 11, b: 125.45}, {a: 1, b: 444});
  test.equal(operations.shift(), ['changed', {a: 1}, -1, {a: 11}]);
  test.equal(operations.shift(), ['moved', {a: 1}, -1, -1, 'foo']);
  c.remove({a: 2});
  test.equal(operations.shift(), undefined);
  c.remove({a: 3});
  test.equal(operations.shift(), ['removed', 'foo', -1, {a: 3}]);
  handle.stop();
});


Tinytest.add('minimongo - ordering', test => {
  const shortBinary = EJSON.newBinary(1);
  shortBinary[0] = 128;
  const longBinary1 = EJSON.newBinary(2);
  longBinary1[1] = 42;
  const longBinary2 = EJSON.newBinary(2);
  longBinary2[1] = 50;

  const date1 = new Date;
  const date2 = new Date(date1.getTime() + 1000);

  // value ordering
  assert_ordering(test, LocalCollection._f._cmp, [
    null,
    1, 2.2, 3,
    '03', '1', '11', '2', 'a', 'aaa',
    {}, {a: 2}, {a: 3}, {a: 3, b: 4}, {b: 4}, {b: 4, a: 3},
    {b: {}}, {b: [1, 2, 3]}, {b: [1, 2, 4]},
    [], [1, 2], [1, 2, 3], [1, 2, 4], [1, 2, '4'], [1, 2, [4]],
    shortBinary, longBinary1, longBinary2,
    new MongoID.ObjectID('1234567890abcd1234567890'),
    new MongoID.ObjectID('abcd1234567890abcd123456'),
    false, true,
    date1, date2,
  ]);

  // document ordering under a sort specification
  const verify = (sorts, docs) => {
    (Array.isArray(sorts) ? sorts : [sorts]).forEach(sort => {
      const sorter = new Minimongo.Sorter(sort);
      assert_ordering(test, sorter.getComparator(), docs);
    });
  };

  // note: [] doesn't sort with "arrays", it sorts as "undefined". the position
  // of arrays in _typeorder only matters for things like $lt. (This behavior
  // verified with MongoDB 2.2.1.) We don't define the relative order of {a: []}
  // and {c: 1} is undefined (MongoDB does seem to care but it's not clear how
  // or why).
  verify([{a: 1}, ['a'], [['a', 'asc']]],
    [{a: []}, {a: 1}, {a: {}}, {a: true}]);
  verify([{a: 1}, ['a'], [['a', 'asc']]],
    [{c: 1}, {a: 1}, {a: {}}, {a: true}]);
  verify([{a: -1}, [['a', 'desc']]],
    [{a: true}, {a: {}}, {a: 1}, {c: 1}]);
  verify([{a: -1}, [['a', 'desc']]],
    [{a: true}, {a: {}}, {a: 1}, {a: []}]);

  verify([{a: 1, b: -1}, ['a', ['b', 'desc']],
    [['a', 'asc'], ['b', 'desc']]],
  [{c: 1}, {a: 1, b: 3}, {a: 1, b: 2}, {a: 2, b: 0}]);

  verify([{a: 1, b: 1}, ['a', 'b'],
    [['a', 'asc'], ['b', 'asc']]],
  [{c: 1}, {a: 1, b: 2}, {a: 1, b: 3}, {a: 2, b: 0}]);

  test.throws(() => {
    new Minimongo.Sorter('a');
  });

  test.throws(() => {
    new Minimongo.Sorter(123);
  });

  // We don't support $natural:1 (since we don't actually have Mongo's on-disk
  // ordering available!)
  test.throws(() => {
    new Minimongo.Sorter({$natural: 1});
  });

  // No sort spec implies everything equal.
  test.equal(new Minimongo.Sorter({}).getComparator()({a: 1}, {a: 2}), 0);

  // All sorts of array edge cases!
  // Increasing sort sorts by the smallest element it finds; 1 < 2.
  verify({a: 1}, [
    {a: [1, 10, 20]},
    {a: [5, 2, 99]},
  ]);
  // Decreasing sorts by largest it finds; 99 > 20.
  verify({a: -1}, [
    {a: [5, 2, 99]},
    {a: [1, 10, 20]},
  ]);
  // Can also sort by specific array indices.
  verify({'a.1': 1}, [
    {a: [5, 2, 99]},
    {a: [1, 10, 20]},
  ]);
  // We do NOT expand sub-arrays, so the minimum in the second doc is 5, not
  // -20. (Numbers always sort before arrays.)
  verify({a: 1}, [
    {a: [1, [10, 15], 20]},
    {a: [5, [-5, -20], 18]},
  ]);
  // The maximum in each of these is the array, since arrays are "greater" than
  // numbers. And [10, 15] is greater than [-5, -20].
  verify({a: -1}, [
    {a: [1, [10, 15], 20]},
    {a: [5, [-5, -20], 18]},
  ]);
  // 'a.0' here ONLY means "first element of a", not "first element of something
  // found in a", so it CANNOT find the 10 or -5.
  verify({'a.0': 1}, [
    {a: [1, [10, 15], 20]},
    {a: [5, [-5, -20], 18]},
  ]);
  verify({'a.0': -1}, [
    {a: [5, [-5, -20], 18]},
    {a: [1, [10, 15], 20]},
  ]);
  // Similarly, this is just comparing [-5,-20] to [10, 15].
  verify({'a.1': 1}, [
    {a: [5, [-5, -20], 18]},
    {a: [1, [10, 15], 20]},
  ]);
  verify({'a.1': -1}, [
    {a: [1, [10, 15], 20]},
    {a: [5, [-5, -20], 18]},
  ]);
  // Here we are just comparing [10,15] directly to [19,3] (and NOT also
  // iterating over the numbers; this is implemented by setting dontIterate in
  // makeLookupFunction).  So [10,15]<[19,3] even though 3 is the smallest
  // number you can find there.
  verify({'a.1': 1}, [
    {a: [1, [10, 15], 20]},
    {a: [5, [19, 3], 18]},
  ]);
  verify({'a.1': -1}, [
    {a: [5, [19, 3], 18]},
    {a: [1, [10, 15], 20]},
  ]);
  // Minimal elements are 1 and 5.
  verify({a: 1}, [
    {a: [1, [10, 15], 20]},
    {a: [5, [19, 3], 18]},
  ]);
  // Maximal elements are [19,3] and [10,15] (because arrays sort higher than
  // numbers), even though there's a 20 floating around.
  verify({a: -1}, [
    {a: [5, [19, 3], 18]},
    {a: [1, [10, 15], 20]},
  ]);
  // Maximal elements are [10,15] and [3,19].  [10,15] is bigger even though 19
  // is the biggest number in them, because array comparison is lexicographic.
  verify({a: -1}, [
    {a: [1, [10, 15], 20]},
    {a: [5, [3, 19], 18]},
  ]);

  // (0,4) < (0,5), so they go in this order.  It's not correct to consider
  // (0,3) as a sort key for the second document because they come from
  // different a-branches.
  verify({'a.x': 1, 'a.y': 1}, [
    {a: [{x: 0, y: 4}]},
    {a: [{x: 0, y: 5}, {x: 1, y: 3}]},
  ]);

  verify({'a.0.s': 1}, [
    {a: [ {s: 1} ]},
    {a: [ {s: 2} ]},
  ]);
});

Tinytest.add('minimongo - sort', test => {
  const c = new LocalCollection();
  for (let i = 0; i < 50; i++) {
    for (let j = 0; j < 2; j++) {c.insert({a: i, b: j, _id: `${i}_${j}`});}
  }

  test.equal(c.find(null, {sort: {b: -1, a: 1}, limit: 5}).fetch(), []);
  test.equal(c.find(undefined, {sort: {b: -1, a: 1}, limit: 5}).fetch(), []);
  test.equal(c.find(false, {sort: {b: -1, a: 1}, limit: 5}).fetch(), []);

  test.equal(
    c.find({a: {$gt: 10}}, {sort: {b: -1, a: 1}, limit: 5}).fetch(), [
      {a: 11, b: 1, _id: '11_1'},
      {a: 12, b: 1, _id: '12_1'},
      {a: 13, b: 1, _id: '13_1'},
      {a: 14, b: 1, _id: '14_1'},
      {a: 15, b: 1, _id: '15_1'}]);

  test.equal(
    c.find({a: {$gt: 10}}, {sort: {b: -1, a: 1}, skip: 3, limit: 5}).fetch(), [
      {a: 14, b: 1, _id: '14_1'},
      {a: 15, b: 1, _id: '15_1'},
      {a: 16, b: 1, _id: '16_1'},
      {a: 17, b: 1, _id: '17_1'},
      {a: 18, b: 1, _id: '18_1'}]);

  test.equal(
    c.find({a: {$gte: 20}}, {sort: {a: 1, b: -1}, skip: 50, limit: 5}).fetch(), [
      {a: 45, b: 1, _id: '45_1'},
      {a: 45, b: 0, _id: '45_0'},
      {a: 46, b: 1, _id: '46_1'},
      {a: 46, b: 0, _id: '46_0'},
      {a: 47, b: 1, _id: '47_1'}]);
});

Tinytest.add('minimongo - subkey sort', test => {
  const c = new LocalCollection();

  // normal case
  c.insert({a: {b: 2}});
  c.insert({a: {b: 1}});
  c.insert({a: {b: 3}});
  test.equal(
    c.find({}, {sort: {'a.b': -1}}).fetch().map(doc => doc.a),
    [{b: 3}, {b: 2}, {b: 1}]);

  // isn't an object
  c.insert({a: 1});
  test.equal(
    c.find({}, {sort: {'a.b': 1}}).fetch().map(doc => doc.a),
    [1, {b: 1}, {b: 2}, {b: 3}]);

  // complex object
  c.insert({a: {b: {c: 1}}});
  test.equal(
    c.find({}, {sort: {'a.b': -1}}).fetch().map(doc => doc.a),
    [{b: {c: 1}}, {b: 3}, {b: 2}, {b: 1}, 1]);

  // no such top level prop
  c.insert({c: 1});
  test.equal(
    c.find({}, {sort: {'a.b': -1}}).fetch().map(doc => doc.a),
    [{b: {c: 1}}, {b: 3}, {b: 2}, {b: 1}, 1, undefined]);

  // no such mid level prop. just test that it doesn't throw.
  test.equal(c.find({}, {sort: {'a.nope.c': -1}}).count(), 6);
});

Tinytest.add('minimongo - array sort', test => {
  const c = new LocalCollection();

  // "up" and "down" are the indices that the docs should have when sorted
  // ascending and descending by "a.x" respectively. They are not reverses of
  // each other: when sorting ascending, you use the minimum value you can find
  // in the document, and when sorting descending, you use the maximum value you
  // can find. So [1, 4] shows up in the 1 slot when sorting ascending and the 4
  // slot when sorting descending.
  //
  // Similarly, "selected" is the index that the doc should have in the query
  // that sorts ascending on "a.x" and selects {'a.x': {$gt: 1}}. In this case,
  // the 1 in [1, 4] may not be used as a sort key.
  c.insert({up: 1, down: 1, selected: 0, a: {x: [1, 4]}});
  c.insert({up: 2, down: 2, selected: 1, a: [{x: [2]}, {x: 3}]});
  c.insert({up: 0, down: 4,              a: {x: 0}});
  c.insert({up: 3, down: 3, selected: 2, a: {x: 2.5}});
  c.insert({up: 4, down: 0, selected: 3, a: {x: 5}});

  // Test that the the documents in "cursor" contain values with the name
  // "field" running from 0 to the max value of that name in the collection.
  const testCursorMatchesField = (cursor, field) => {
    const fieldValues = [];
    c.find().forEach(doc => {
      if (hasOwn.call(doc, field)) {fieldValues.push(doc[field]);}
    });
    test.equal(cursor.fetch().map(doc => doc[field]),
      Array.from({length: Math.max(...fieldValues) + 1}, (x, i) => i));
  };

  testCursorMatchesField(c.find({}, {sort: {'a.x': 1}}), 'up');
  testCursorMatchesField(c.find({}, {sort: {'a.x': -1}}), 'down');
  testCursorMatchesField(c.find({'a.x': {$gt: 1}}, {sort: {'a.x': 1}}),
    'selected');
});

Tinytest.add('minimongo - nested array sort', test => {
  const c = new LocalCollection();

  // the short fields represent the order it should be when sorting for those keys
  // e.g. the cdx_cdy field represents the order when you sort: { 'c.d.x': 1, 'c.d.y': 1 }
  c.insert({ ab0x: 0, ab0x_g: 0, g_ab0x: 0, cdx: 0, cdx_cdy: 0, cdy_cdx: 0, n: 0 });
  c.insert({ ab0x: 1, ab0x_g: 2, g_ab0x: 3, cdx: 1, cdx_cdy: 2, cdy_cdx: 4, n: 1 , g: 2, c: { d: [{ y: 2}, { y: 3}] } });
  c.insert({ ab0x: 2, ab0x_g: 1, g_ab0x: 1, cdx: 2, cdx_cdy: 3, cdy_cdx: 5, n: 2 , c: { d: [{ y: 2}] }, g: 0 });
  c.insert({ ab0x: 3, ab0x_g: 3, g_ab0x: 2, cdx: 6, cdx_cdy: 6, cdy_cdx: 8, n: 3 , a: { b: [{ x: 0 }] }, c: { d: [{ x: 1, y: 2}] }, g: 1 });
  c.insert({ ab0x: 4, ab0x_g: 4, g_ab0x: 4, cdx: 3, cdx_cdy: 1, cdy_cdx: 1, n: 4 , a: { b: [{ x: [1, 4] }] }, c: { d: [] }, g: 2 });
  c.insert({ ab0x: 5, ab0x_g: 5, g_ab0x: 5, cdx: 7, cdx_cdy: 7, cdy_cdx: 3, n: 5 , a: { b: [{ x: [2] }, { x: 3 }]}, c: { d: [{x: 2, y: 2}, {x: 3, y: 1}] }, g: 3 });
  c.insert({ ab0x: 6, ab0x_g: 6, g_ab0x: 6, cdx: 8, cdx_cdy: 8, cdy_cdx: 2, n: 6 , a: { b: [{ x: 2.5 }] }, c: { d: [{x: 2, y: 2}, {x: 3}] }, g: 4 });
  c.insert({ ab0x: 7, ab0x_g: 7, g_ab0x: 7, cdx: 4, cdx_cdy: 4, cdy_cdx: 6, n: 7 , a: { b: [{ x: 5 }] }, c: { d: [{ y: 2}, { y: 3}] }, g: 5 });
  c.insert({ ab0x: 8, ab0x_g: 8, g_ab0x: 8, cdx: 5, cdx_cdy: 5, cdy_cdx: 7, n: 8 , a: { b: [{ x: 6 }, { x: 7 }] }, c: { d: [{ y: 2}, { x: 1.5, y: 2}] }, g: 6 });

  // Test that the the documents in "cursor" contain values with the name
  // "field" running from 0 to the max value of that name in the collection.
  const testCursorMatchesField = (cursor, field) => {
    const fieldValues = [];
    c.find().forEach(doc => {
      if (hasOwn.call(doc, field)) {
        fieldValues.push(doc[field]);
      }
    });
    test.equal(cursor.fetch().map(doc => doc[field]),
      Array.from({ length: Math.max(...fieldValues) + 1 }, (x, i) => i));
  };

  testCursorMatchesField(c.find({}, { sort: { 'a.b.0.x': 1 } }), 'ab0x');
  testCursorMatchesField(c.find({}, { sort: { 'a.b.0.x': 1, 'g': 1 } }), 'ab0x_g');
  testCursorMatchesField(c.find({}, { sort: { 'g': 1, 'a.b.0.x': 1 } }), 'g_ab0x');
  testCursorMatchesField(c.find({}, { sort: { 'c.d.x': 1 } }), 'cdx');
  testCursorMatchesField(c.find({}, { sort: { 'c.d.x': 1, 'c.d.y': 1 } }), 'cdx_cdy');
  testCursorMatchesField(c.find({}, { sort: { 'c.d.y': 1, 'c.d.x': 1 } }), 'cdy_cdx');

});

Tinytest.add('minimongo - sort keys', test => {
  const keyListToObject = keyList => {
    const obj = {};
    keyList.forEach(key => {
      obj[EJSON.stringify(key)] = true;
    });
    return obj;
  };

  const testKeys = (sortSpec, doc, expectedKeyList) => {
    const expectedKeys = keyListToObject(expectedKeyList);
    const sorter = new Minimongo.Sorter(sortSpec);

    const actualKeyList = [];
    sorter._generateKeysFromDoc(doc, key => {
      actualKeyList.push(key);
    });
    const actualKeys = keyListToObject(actualKeyList);
    test.equal(actualKeys, expectedKeys);
  };

  const testParallelError = (sortSpec, doc) => {
    const sorter = new Minimongo.Sorter(sortSpec);
    test.throws(() => {
      sorter._generateKeysFromDoc(doc, () => {});
    }, /parallel arrays/);
  };

  // Just non-array fields.
  testKeys({'a.x': 1, 'a.y': 1},
    {a: {x: 0, y: 5}},
    [[0, 5]]);

  // Ensure that we don't get [0,3] and [1,5].
  testKeys({'a.x': 1, 'a.y': 1},
    {a: [{x: 0, y: 5}, {x: 1, y: 3}]},
    [[0, 5], [1, 3]]);

  // Ensure we can combine "array fields" with "non-array fields".
  testKeys({'a.x': 1, 'a.y': 1, b: -1},
    {a: [{x: 0, y: 5}, {x: 1, y: 3}], b: 42},
    [[0, 5, 42], [1, 3, 42]]);
  testKeys({b: -1, 'a.x': 1, 'a.y': 1},
    {a: [{x: 0, y: 5}, {x: 1, y: 3}], b: 42},
    [[42, 0, 5], [42, 1, 3]]);
  testKeys({'a.x': 1, b: -1, 'a.y': 1},
    {a: [{x: 0, y: 5}, {x: 1, y: 3}], b: 42},
    [[0, 42, 5], [1, 42, 3]]);
  testKeys({a: 1, b: 1},
    {a: [1, 2, 3], b: 42},
    [[1, 42], [2, 42], [3, 42]]);

  testKeys({'a.0.x': 1},
    {a: [{x: 0}]},
    [[0]]);

  testKeys({'a.0.x': 1},
    {a: []},
    [[undefined]]);

  // Don't support multiple arrays at the same level.
  testParallelError({a: 1, b: 1},
    {a: [1, 2, 3], b: [42]});

  // We are MORE STRICT than Mongo here; Mongo supports this!
  // XXX support this too  #NestedArraySort
  testParallelError({'a.x': 1, 'a.y': 1},
    {a: [{x: 1, y: [2, 3]},
      {x: 2, y: [4, 5]}]});
});

Tinytest.add('minimongo - sort function', test => {
  const c = new LocalCollection();

  c.insert({a: 1});
  c.insert({a: 10});
  c.insert({a: 5});
  c.insert({a: 7});
  c.insert({a: 2});
  c.insert({a: 4});
  c.insert({a: 3});

  const sortFunction = (doc1, doc2) => doc2.a - doc1.a;

  test.equal(c.find({}, {sort: sortFunction}).fetch(), c.find({}).fetch().sort(sortFunction));
  test.notEqual(c.find({}).fetch(), c.find({}).fetch().sort(sortFunction));
  test.equal(c.find({}, {sort: {a: -1}}).fetch(), c.find({}).fetch().sort(sortFunction));
});

Tinytest.add('minimongo - binary search', test => {
  const forwardCmp = (a, b) => a - b;

  const backwardCmp = (a, b) => -1 * forwardCmp(a, b);

  const checkSearch = (cmp, array, value, expected, message) => {
    const actual = LocalCollection._binarySearch(cmp, array, value);
    if (expected != actual) {
      test.fail({type: 'minimongo-binary-search',
        message: `${message} : Expected index ${expected} but had ${actual}`,
      });
    }
  };

  const checkSearchForward = (array, value, expected, message) => {
    checkSearch(forwardCmp, array, value, expected, message);
  };
  const checkSearchBackward = (array, value, expected, message) => {
    checkSearch(backwardCmp, array, value, expected, message);
  };

  checkSearchForward([1, 2, 5, 7], 4, 2, 'Inner insert');
  checkSearchForward([1, 2, 3, 4], 3, 3, 'Inner insert, equal value');
  checkSearchForward([1, 2, 5], 4, 2, 'Inner insert, odd length');
  checkSearchForward([1, 3, 5, 6], 9, 4, 'End insert');
  checkSearchForward([1, 3, 5, 6], 0, 0, 'Beginning insert');
  checkSearchForward([1], 0, 0, 'Single array, less than.');
  checkSearchForward([1], 1, 1, 'Single array, equal.');
  checkSearchForward([1], 2, 1, 'Single array, greater than.');
  checkSearchForward([], 1, 0, 'Empty array');
  checkSearchForward([1, 1, 1, 2, 2, 2, 2], 1, 3, 'Highly degenerate array, lower');
  checkSearchForward([1, 1, 1, 2, 2, 2, 2], 2, 7, 'Highly degenerate array, upper');
  checkSearchForward([2, 2, 2, 2, 2, 2, 2], 1, 0, 'Highly degenerate array, lower');
  checkSearchForward([2, 2, 2, 2, 2, 2, 2], 2, 7, 'Highly degenerate array, equal');
  checkSearchForward([2, 2, 2, 2, 2, 2, 2], 3, 7, 'Highly degenerate array, upper');

  checkSearchBackward([7, 5, 2, 1], 4, 2, 'Backward: Inner insert');
  checkSearchBackward([4, 3, 2, 1], 3, 2, 'Backward: Inner insert, equal value');
  checkSearchBackward([5, 2, 1], 4, 1, 'Backward: Inner insert, odd length');
  checkSearchBackward([6, 5, 3, 1], 9, 0, 'Backward: Beginning insert');
  checkSearchBackward([6, 5, 3, 1], 0, 4, 'Backward: End insert');
  checkSearchBackward([1], 0, 1, 'Backward: Single array, less than.');
  checkSearchBackward([1], 1, 1, 'Backward: Single array, equal.');
  checkSearchBackward([1], 2, 0, 'Backward: Single array, greater than.');
  checkSearchBackward([], 1, 0, 'Backward: Empty array');
  checkSearchBackward([2, 2, 2, 2, 1, 1, 1], 1, 7, 'Backward: Degenerate array, lower');
  checkSearchBackward([2, 2, 2, 2, 1, 1, 1], 2, 4, 'Backward: Degenerate array, upper');
  checkSearchBackward([2, 2, 2, 2, 2, 2, 2], 1, 7, 'Backward: Highly degenerate array, upper');
  checkSearchBackward([2, 2, 2, 2, 2, 2, 2], 2, 7, 'Backward: Highly degenerate array, upper');
  checkSearchBackward([2, 2, 2, 2, 2, 2, 2], 3, 0, 'Backward: Highly degenerate array, upper');
});

Tinytest.add('minimongo - modify', test => {
  const modifyWithQuery = (doc, query, mod, expected) => {
    const coll = new LocalCollection;
    coll.insert(doc);
    // The query is relevant for 'a.$.b'.
    coll.update(query, mod);
    const actual = coll.findOne();

    if (!expected._id) {
      delete actual._id;  // added by insert
    }

    if (typeof expected === 'function') {
      expected(actual, EJSON.stringify({input: doc, mod}));
    } else {
      test.equal(actual, expected, EJSON.stringify({input: doc, mod}));
    }
  };
  const modify = (doc, mod, expected) => {
    modifyWithQuery(doc, {}, mod, expected);
  };
  const exceptionWithQuery = (doc, query, mod) => {
    const coll = new LocalCollection;
    coll.insert(doc);
    test.throws(() => {
      coll.update(query, mod);
    });
  };
  const exception = (doc, mod) => {
    exceptionWithQuery(doc, {}, mod);
  };

  const upsert = (query, mod, expected) => {
    const coll = new LocalCollection;

    const result = coll.upsert(query, mod);

    const actual = coll.findOne();

    if (expected._id) {
      test.equal(result.insertedId, expected._id);
    } else {
      delete actual._id;
    }

    test.equal(actual, expected);
  };

  const upsertUpdate = (initialDoc, query, mod, expected) => {
    const collection = new LocalCollection;

    collection.insert(initialDoc);

    const result = collection.upsert(query, mod);
    const actual = collection.findOne();

    if (!expected._id) {
      delete actual._id;
    }

    test.equal(actual, expected);
  };

  const upsertException = (query, mod) => {
    const coll = new LocalCollection;
    test.throws(() => {
      coll.upsert(query, mod);
    });
  };

  // document replacement
  modify({}, {}, {});
  modify({a: 12}, {}, {}); // tested against mongodb
  modify({a: 12}, {a: 13}, {a: 13});
  modify({a: 12, b: 99}, {a: 13}, {a: 13});
  exception({a: 12}, {a: 13, $set: {b: 13}});
  exception({a: 12}, {$set: {b: 13}, a: 13});

  exception({a: 12}, {$a: 13}); // invalid operator
  exception({a: 12}, {b: {$a: 13}});
  exception({a: 12}, {b: {'a.b': 13}});
  exception({a: 12}, {b: {'\0a': 13}});

  // keys
  modify({}, {$set: {a: 12}}, {a: 12});
  modify({}, {$set: {'a.b': 12}}, {a: {b: 12}});
  modify({}, {$set: {'a.b.c': 12}}, {a: {b: {c: 12}}});
  modify({a: {d: 99}}, {$set: {'a.b.c': 12}}, {a: {d: 99, b: {c: 12}}});
  modify({}, {$set: {'a.b.3.c': 12}}, {a: {b: {3: {c: 12}}}});
  modify({a: {b: []}}, {$set: {'a.b.3.c': 12}}, {
    a: {b: [null, null, null, {c: 12}]}});
  exception({a: [null, null, null]}, {$set: {'a.1.b': 12}});
  exception({a: [null, 1, null]}, {$set: {'a.1.b': 12}});
  exception({a: [null, 'x', null]}, {$set: {'a.1.b': 12}});
  exception({a: [null, [], null]}, {$set: {'a.1.b': 12}});
  modify({a: [null, null, null]}, {$set: {'a.3.b': 12}}, {
    a: [null, null, null, {b: 12}]});
  exception({a: []}, {$set: {'a.b': 12}});
  exception({a: 12}, {$set: {'a.b': 99}}); // tested on mongo
  exception({a: 'x'}, {$set: {'a.b': 99}});
  exception({a: true}, {$set: {'a.b': 99}});
  exception({a: null}, {$set: {'a.b': 99}});
  modify({a: {}}, {$set: {'a.3': 12}}, {a: {3: 12}});
  modify({a: []}, {$set: {'a.3': 12}}, {a: [null, null, null, 12]});
  exception({}, {$set: {'': 12}}); // tested on mongo
  exception({}, {$set: {'.': 12}}); // tested on mongo
  exception({}, {$set: {'a.': 12}}); // tested on mongo
  exception({}, {$set: {'. ': 12}}); // tested on mongo
  exception({}, {$inc: {'... ': 12}}); // tested on mongo
  exception({}, {$set: {'a..b': 12}}); // tested on mongo
  modify({a: [1, 2, 3]}, {$set: {'a.01': 99}}, {a: [1, 99, 3]});
  modify({a: [1, {a: 98}, 3]}, {$set: {'a.01.b': 99}}, {a: [1, {a: 98, b: 99}, 3]});
  modify({}, {$set: {'2.a.b': 12}}, {2: {a: {b: 12}}}); // tested
  exception({x: []}, {$set: {'x.2..a': 99}});
  modify({x: [null, null]}, {$set: {'x.2.a': 1}}, {x: [null, null, {a: 1}]});
  exception({x: [null, null]}, {$set: {'x.1.a': 1}});

  // a.$.b
  modifyWithQuery({a: [{x: 2}, {x: 4}]}, {'a.x': 4}, {$set: {'a.$.z': 9}},
    {a: [{x: 2}, {x: 4, z: 9}]});
  exception({a: [{x: 2}, {x: 4}]}, {$set: {'a.$.z': 9}});
  exceptionWithQuery({a: [{x: 2}, {x: 4}], b: 5}, {b: 5}, {$set: {'a.$.z': 9}});
  // can't have two $
  exceptionWithQuery({a: [{x: [2]}]}, {'a.x': 2}, {$set: {'a.$.x.$': 9}});
  modifyWithQuery({a: [5, 6, 7]}, {a: 6}, {$set: {'a.$': 9}}, {a: [5, 9, 7]});
  modifyWithQuery({a: [{b: [{c: 9}, {c: 10}]}, {b: {c: 11}}]}, {'a.b.c': 10},
    {$unset: {'a.$.b': 1}}, {a: [{}, {b: {c: 11}}]});
  modifyWithQuery({a: [{b: [{c: 9}, {c: 10}]}, {b: {c: 11}}]}, {'a.b.c': 11},
    {$unset: {'a.$.b': 1}},
    {a: [{b: [{c: 9}, {c: 10}]}, {}]});
  modifyWithQuery({a: [1]}, {'a.0': 1}, {$set: {'a.$': 5}}, {a: [5]});
  modifyWithQuery({a: [9]}, {a: {$mod: [2, 1]}}, {$set: {'a.$': 5}}, {a: [5]});
  // Negatives don't set '$'.
  exceptionWithQuery({a: [1]}, {$not: {a: 2}}, {$set: {'a.$': 5}});
  exceptionWithQuery({a: [1]}, {'a.0': {$ne: 2}}, {$set: {'a.$': 5}});
  // One $or clause works.
  modifyWithQuery({a: [{x: 2}, {x: 4}]},
    {$or: [{'a.x': 4}]}, {$set: {'a.$.z': 9}},
    {a: [{x: 2}, {x: 4, z: 9}]});
  // More $or clauses throw.
  exceptionWithQuery({a: [{x: 2}, {x: 4}]},
    {$or: [{'a.x': 4}, {'a.x': 4}]},
    {$set: {'a.$.z': 9}});
  // $and uses the last one.
  modifyWithQuery({a: [{x: 1}, {x: 3}]},
    {$and: [{'a.x': 1}, {'a.x': 3}]},
    {$set: {'a.$.x': 5}},
    {a: [{x: 1}, {x: 5}]});
  modifyWithQuery({a: [{x: 1}, {x: 3}]},
    {$and: [{'a.x': 3}, {'a.x': 1}]},
    {$set: {'a.$.x': 5}},
    {a: [{x: 5}, {x: 3}]});
  // Same goes for the implicit AND of a document selector.
  modifyWithQuery({a: [{x: 1}, {y: 3}]},
    {'a.x': 1, 'a.y': 3},
    {$set: {'a.$.z': 5}},
    {a: [{x: 1}, {y: 3, z: 5}]});
  modifyWithQuery({a: [{x: 1}, {y: 1}, {x: 1, y: 1}]},
    {a: {$elemMatch: {x: 1, y: 1}}},
    {$set: {'a.$.x': 2}},
    {a: [{x: 1}, {y: 1}, {x: 2, y: 1}]});
  modifyWithQuery({a: [{b: [{x: 1}, {y: 1}, {x: 1, y: 1}]}]},
    {'a.b': {$elemMatch: {x: 1, y: 1}}},
    {$set: {'a.$.b': 3}},
    {a: [{b: 3}]});
  // with $near, make sure it does not find the closest one (#3599)
  modifyWithQuery({a: []},
    {'a.b': {$near: [5, 5]}},
    {$set: {'a.$.b': 'k'}},
    {a: []});
  modifyWithQuery({a: [{b: [ [3, 3], [4, 4] ]}]},
    {'a.b': {$near: [5, 5]}},
    {$set: {'a.$.b': 'k'}},
    {a: [{b: 'k'}]});
  modifyWithQuery({a: [{b: [1, 1]},
    {b: [ [3, 3], [4, 4] ]},
    {b: [9, 9]}]},
  {'a.b': {$near: [5, 5]}},
  {$set: {'a.$.b': 'k'}},
  {a: [{b: 'k'}, {b: [[3, 3], [4, 4]]}, {b: [9, 9]}]});
  modifyWithQuery({a: [{b: [1, 1]},
    {b: [ [3, 3], [4, 4] ]},
    {b: [9, 9]}]},
  {'a.b': {$near: [9, 9], $maxDistance: 1}},
  {$set: {'a.$.b': 'k'}},
  {a: [{b: 'k'}, {b: [[3, 3], [4, 4]]}, {b: [9, 9]}]});
  modifyWithQuery({a: [{b: [1, 1]},
    {b: [ [3, 3], [4, 4] ]},
    {b: [9, 9]}]},
  {'a.b': {$near: [9, 9]}},
  {$set: {'a.$.b': 'k'}},
  {a: [{b: 'k'}, {b: [[3, 3], [4, 4]]}, {b: [9, 9]}]});
  modifyWithQuery({a: [{b: [9, 9]},
    {b: [ [3, 3], [4, 4] ]},
    {b: [9, 9]}]},
  {'a.b': {$near: [9, 9]}},
  {$set: {'a.$.b': 'k'}},
  {a: [{b: 'k'}, {b: [[3, 3], [4, 4]]}, {b: [9, 9]}]});
  modifyWithQuery({a: [{b: [4, 3]},
    {c: [1, 1]}]},
  {'a.c': {$near: [1, 1]}},
  {$set: {'a.$.c': 'k'}},
  {a: [{c: 'k', b: [4, 3]}, {c: [1, 1]}]});
  modifyWithQuery({a: [{c: [9, 9]},
    {b: [ [3, 3], [4, 4] ]},
    {b: [1, 1]}]},
  {'a.b': {$near: [1, 1]}},
  {$set: {'a.$.b': 'k'}},
  {a: [{c: [9, 9], b: 'k'}, {b: [ [3, 3], [4, 4]]}, {b: [1, 1]}]});
  modifyWithQuery({a: [{c: [9, 9], b: [4, 3]},
    {b: [ [3, 3], [4, 4] ]},
    {b: [1, 1]}]},
  {'a.b': {$near: [1, 1]}},
  {$set: {'a.$.b': 'k'}},
  {a: [{c: [9, 9], b: 'k'}, {b: [ [3, 3], [4, 4]]}, {b: [1, 1]}]});

  // $inc
  modify({a: 1, b: 2}, {$inc: {a: 10}}, {a: 11, b: 2});
  modify({a: 1, b: 2}, {$inc: {c: 10}}, {a: 1, b: 2, c: 10});
  exception({a: 1}, {$inc: {a: '10'}});
  exception({a: 1}, {$inc: {a: true}});
  exception({a: 1}, {$inc: {a: [10]}});
  exception({a: '1'}, {$inc: {a: 10}});
  exception({a: [1]}, {$inc: {a: 10}});
  exception({a: {}}, {$inc: {a: 10}});
  exception({a: false}, {$inc: {a: 10}});
  exception({a: null}, {$inc: {a: 10}});
  modify({a: [1, 2]}, {$inc: {'a.1': 10}}, {a: [1, 12]});
  modify({a: [1, 2]}, {$inc: {'a.2': 10}}, {a: [1, 2, 10]});
  modify({a: [1, 2]}, {$inc: {'a.3': 10}}, {a: [1, 2, null, 10]});
  modify({a: {b: 2}}, {$inc: {'a.b': 10}}, {a: {b: 12}});
  modify({a: {b: 2}}, {$inc: {'a.c': 10}}, {a: {b: 2, c: 10}});
  exception({}, {$inc: {_id: 1}});

  // $currentDate
  modify({}, {$currentDate: {a: true}}, (result, msg) => { test.instanceOf(result.a, Date, msg); });
  modify({}, {$currentDate: {a: {$type: 'date'}}}, (result, msg) => { test.instanceOf(result.a, Date, msg); });
  exception({}, {$currentDate: {a: false}});
  exception({}, {$currentDate: {a: {}}});
  exception({}, {$currentDate: {a: {$type: 'timestamp'}}});

  // $min
  modify({a: 1, b: 2}, {$min: {b: 1}}, {a: 1, b: 1});
  modify({a: 1, b: 2}, {$min: {b: 3}}, {a: 1, b: 2});
  modify({a: 1, b: 2}, {$min: {c: 10}}, {a: 1, b: 2, c: 10});
  exception({a: 1}, {$min: {a: '10'}});
  exception({a: 1}, {$min: {a: true}});
  exception({a: 1}, {$min: {a: [10]}});
  exception({a: '1'}, {$min: {a: 10}});
  exception({a: [1]}, {$min: {a: 10}});
  exception({a: {}}, {$min: {a: 10}});
  exception({a: false}, {$min: {a: 10}});
  exception({a: null}, {$min: {a: 10}});
  modify({a: [1, 2]}, {$min: {'a.1': 1}}, {a: [1, 1]});
  modify({a: [1, 2]}, {$min: {'a.1': 3}}, {a: [1, 2]});
  modify({a: [1, 2]}, {$min: {'a.2': 10}}, {a: [1, 2, 10]});
  modify({a: [1, 2]}, {$min: {'a.3': 10}}, {a: [1, 2, null, 10]});
  modify({a: {b: 2}}, {$min: {'a.b': 1}}, {a: {b: 1}});
  modify({a: {b: 2}}, {$min: {'a.c': 10}}, {a: {b: 2, c: 10}});
  exception({}, {$min: {_id: 1}});

  //$mul
  modify({a: 1, b: 1}, {$mul: {b: 2}}, {a: 1, b: 2});
  modify({a: 1, b: 1}, {$mul: {c: 2}}, {a: 1, b: 1, c: 0});
  modify({a: 1, b: 2}, {$mul: {b: 2}}, {a: 1, b: 4});
  modify({a: 1, b: 2}, {$mul: {b: 10}}, {a: 1, b: 20});
  exception({a: 1}, {$mul: {a: '10'}});
  exception({a: 1}, {$mul: {a: true}});
  exception({a: 1}, {$mul: {a: [10]}});
  exception({a: '1'}, {$mul: {a: 10}});
  exception({a: [1]}, {$mul: {a: 10}});
  exception({a: {}}, {$mul: {a: 10}});
  exception({a: false}, {$mul: {a: 10}});
  exception({a: null}, {$mul: {a: 10}});
  exception({}, {$mul: {_id: 1}});
  modify({a: [1, 2]}, {$mul: {'a.0': 2}}, {a: [2, 2]});
  modify({a: [1, 2]}, {$mul: {'a.1': 3}}, {a: [1, 6]});
  modify({a: [1, 2]}, {$mul: {'a.1': 10}}, {a: [1, 20]});
  modify({a: [1, 2]}, {$mul: {'a.2': 10}}, {a: [1, 2, 0]});
  modify({a: {b: 2}}, {$mul: {'a.b': 1}}, {a: {b: 2}});
  modify({a: {b: 2}}, {$mul: {'a.c': 10}}, {a: {b: 2, c: 0}});

  // $max
  modify({a: 1, b: 2}, {$max: {b: 1}}, {a: 1, b: 2});
  modify({a: 1, b: 2}, {$max: {b: 3}}, {a: 1, b: 3});
  modify({a: 1, b: 2}, {$max: {c: 10}}, {a: 1, b: 2, c: 10});
  exception({a: 1}, {$max: {a: '10'}});
  exception({a: 1}, {$max: {a: true}});
  exception({a: 1}, {$max: {a: [10]}});
  exception({a: '1'}, {$max: {a: 10}});
  exception({a: [1]}, {$max: {a: 10}});
  exception({a: {}}, {$max: {a: 10}});
  exception({a: false}, {$max: {a: 10}});
  exception({a: null}, {$max: {a: 10}});
  modify({a: [1, 2]}, {$max: {'a.1': 3}}, {a: [1, 3]});
  modify({a: [1, 2]}, {$max: {'a.1': 1}}, {a: [1, 2]});
  modify({a: [1, 2]}, {$max: {'a.2': 10}}, {a: [1, 2, 10]});
  modify({a: [1, 2]}, {$max: {'a.3': 10}}, {a: [1, 2, null, 10]});
  modify({a: {b: 2}}, {$max: {'a.b': 3}}, {a: {b: 3}});
  modify({a: {b: 2}}, {$max: {'a.c': 10}}, {a: {b: 2, c: 10}});
  exception({}, {$max: {_id: 1}});

  // $set
  modify({a: 1, b: 2}, {$set: {a: 10}}, {a: 10, b: 2});
  modify({a: 1, b: 2}, {$set: {c: 10}}, {a: 1, b: 2, c: 10});
  modify({a: 1, b: 2}, {$set: {a: {c: 10}}}, {a: {c: 10}, b: 2});
  modify({a: [1, 2], b: 2}, {$set: {a: [3, 4]}}, {a: [3, 4], b: 2});
  modify({a: [1, 2, 3], b: 2}, {$set: {'a.1': [3, 4]}},
    {a: [1, [3, 4], 3], b: 2});
  modify({a: [1], b: 2}, {$set: {'a.1': 9}}, {a: [1, 9], b: 2});
  modify({a: [1], b: 2}, {$set: {'a.2': 9}}, {a: [1, null, 9], b: 2});
  modify({a: {b: 1}}, {$set: {'a.c': 9}}, {a: {b: 1, c: 9}});
  modify({}, {$set: {'x._id': 4}}, {x: {_id: 4}});

  // Changing _id is disallowed
  exception({}, {$set: {_id: 4}});
  exception({_id: 1}, {$set: {_id: 4}});
  modify({_id: 4}, {$set: {_id: 4}}, {_id: 4});  // not-changing _id is not bad
  // restricted field names
  exception({a: {}}, {$set: {a: {$a: 1}}});
  exception({ a: {} }, { $set: { a: { c:
              [{ b: { $a: 1 } }] } } });
  exception({a: {}}, {$set: {a: {'\0a': 1}}});
  exception({a: {}}, {$set: {a: {'a.b': 1}}});

  // $unset
  modify({}, {$unset: {a: 1}}, {});
  modify({a: 1}, {$unset: {a: 1}}, {});
  modify({a: 1, b: 2}, {$unset: {a: 1}}, {b: 2});
  modify({a: 1, b: 2}, {$unset: {a: 0}}, {b: 2});
  modify({a: 1, b: 2}, {$unset: {a: false}}, {b: 2});
  modify({a: 1, b: 2}, {$unset: {a: null}}, {b: 2});
  modify({a: 1, b: 2}, {$unset: {a: [1]}}, {b: 2});
  modify({a: 1, b: 2}, {$unset: {a: {}}}, {b: 2});
  modify({a: {b: 2, c: 3}}, {$unset: {'a.b': 1}}, {a: {c: 3}});
  modify({a: [1, 2, 3]}, {$unset: {'a.1': 1}}, {a: [1, null, 3]}); // tested
  modify({a: [1, 2, 3]}, {$unset: {'a.2': 1}}, {a: [1, 2, null]}); // tested
  modify({a: [1, 2, 3]}, {$unset: {'a.x': 1}}, {a: [1, 2, 3]}); // tested
  modify({a: {b: 1}}, {$unset: {'a.b.c.d': 1}}, {a: {b: 1}});
  modify({a: {b: 1}}, {$unset: {'a.x.c.d': 1}}, {a: {b: 1}});
  modify({a: {b: {c: 1}}}, {$unset: {'a.b.c': 1}}, {a: {b: {}}});
  exception({}, {$unset: {_id: 1}});

  // $push
  modify({}, {$push: {a: 1}}, {a: [1]});
  modify({a: []}, {$push: {a: 1}}, {a: [1]});
  modify({a: [1]}, {$push: {a: 2}}, {a: [1, 2]});
  exception({a: true}, {$push: {a: 1}});
  modify({a: [1]}, {$push: {a: [2]}}, {a: [1, [2]]});
  modify({a: []}, {$push: {'a.1': 99}}, {a: [null, [99]]}); // tested
  modify({a: {}}, {$push: {'a.x': 99}}, {a: {x: [99]}});
  modify({}, {$push: {a: {$each: [1, 2, 3]}}},
    {a: [1, 2, 3]});
  modify({a: []}, {$push: {a: {$each: [1, 2, 3]}}},
    {a: [1, 2, 3]});
  modify({a: [true]}, {$push: {a: {$each: [1, 2, 3]}}},
    {a: [true, 1, 2, 3]});
  modify({a: [true]}, {$push: {a: {$each: [1, 2, 3], $slice: -2}}},
    {a: [2, 3]});
  modify({a: [false, true]}, {$push: {a: {$each: [1], $slice: -2}}},
    {a: [true, 1]});
  modify(
    {a: [{x: 3}, {x: 1}]},
    {$push: {a: {
      $each: [{x: 4}, {x: 2}],
      $slice: -2,
      $sort: {x: 1},
    }}},
    {a: [{x: 3}, {x: 4}]});
  modify({}, {$push: {a: {$each: [1, 2, 3], $slice: 0}}}, {a: []});
  modify({a: [1, 2]}, {$push: {a: {$each: [1, 2, 3], $slice: 0}}}, {a: []});
  // $push with $position modifier
  // No negative number for $position
  exception({a: []}, {$push: {a: {$each: [0], $position: -1}}});
  modify({a: [1, 2]}, {$push: {a: {$each: [0], $position: 0}}},
    {a: [0, 1, 2]});
  modify({a: [1, 2]}, {$push: {a: {$each: [-1, 0], $position: 0}}},
    {a: [-1, 0, 1, 2]});
  modify({a: [1, 3]}, {$push: {a: {$each: [2], $position: 1}}}, {a: [1, 2, 3]});
  modify({a: [1, 4]}, {$push: {a: {$each: [2, 3], $position: 1}}},
    {a: [1, 2, 3, 4]});
  modify({a: [1, 2]}, {$push: {a: {$each: [3], $position: 3}}}, {a: [1, 2, 3]});
  modify({a: [1, 2]}, {$push: {a: {$each: [3], $position: 99}}},
    {a: [1, 2, 3]});
  modify({a: [1, 2]}, {$push: {a: {$each: [3], $position: 99, $slice: -2}}},
    {a: [2, 3]});
  modify(
    {a: [{x: 1}, {x: 2}]},
    {$push: {a: {$each: [{x: 3}], $position: 0, $sort: {x: 1}, $slice: -3}}},
    {a: [{x: 1}, {x: 2}, {x: 3}]}
  );
  modify(
    {a: [{x: 1}, {x: 2}]},
    {$push: {a: {$each: [{x: 3}], $position: 0, $sort: {x: 1}, $slice: 0}}},
    {a: []}
  );
  // restricted field names
  exception({}, {$push: {$a: 1}});
  exception({}, {$push: {'\0a': 1}});
  exception({}, {$push: {a: {$a: 1}}});
  exception({}, {$push: {a: {$each: [{$a: 1}]}}});
  exception({}, {$push: {a: {$each: [{'a.b': 1}]}}});
  exception({}, {$push: {a: {$each: [{'\0a': 1}]}}});
  modify({}, {$push: {a: {$each: [{'': 1}]}}}, {a: [ { '': 1 } ]});
  modify({}, {$push: {a: {$each: [{' ': 1}]}}}, {a: [ { ' ': 1 } ]});
  exception({}, {$push: {a: {$each: [{'.': 1}]}}});

  // #issue 5167
  // $push $slice with positive numbers
  modify({}, {$push: {a: {$each: [], $slice: 5}}}, {a: []});
  modify({a: [1, 2, 3]}, {$push: {a: {$each: [], $slice: 1}}}, {a: [1]});
  modify({a: [1, 2, 3]}, {$push: {a: {$each: [4, 5], $slice: 1}}}, {a: [1]});
  modify({a: [1, 2, 3]}, {$push: {a: {$each: [4, 5], $slice: 2}}}, {a: [1, 2]});
  modify({a: [1, 2, 3]}, {$push: {a: {$each: [4, 5], $slice: 4}}}, {a: [1, 2, 3, 4]});
  modify({a: [1, 2, 3]}, {$push: {a: {$each: [4, 5], $slice: 5}}}, {a: [1, 2, 3, 4, 5]});
  modify({a: [1, 2, 3]}, {$push: {a: {$each: [4, 5], $slice: 10}}}, {a: [1, 2, 3, 4, 5]});


  // $pushAll
  modify({}, {$pushAll: {a: [1]}}, {a: [1]});
  modify({a: []}, {$pushAll: {a: [1]}}, {a: [1]});
  modify({a: [1]}, {$pushAll: {a: [2]}}, {a: [1, 2]});
  modify({}, {$pushAll: {a: [1, 2]}}, {a: [1, 2]});
  modify({a: []}, {$pushAll: {a: [1, 2]}}, {a: [1, 2]});
  modify({a: [1]}, {$pushAll: {a: [2, 3]}}, {a: [1, 2, 3]});
  modify({}, {$pushAll: {a: []}}, {a: []});
  modify({a: []}, {$pushAll: {a: []}}, {a: []});
  modify({a: [1]}, {$pushAll: {a: []}}, {a: [1]});
  exception({a: true}, {$pushAll: {a: [1]}});
  exception({a: []}, {$pushAll: {a: 1}});
  modify({a: []}, {$pushAll: {'a.1': [99]}}, {a: [null, [99]]});
  modify({a: []}, {$pushAll: {'a.1': []}}, {a: [null, []]});
  modify({a: {}}, {$pushAll: {'a.x': [99]}}, {a: {x: [99]}});
  modify({a: {}}, {$pushAll: {'a.x': []}}, {a: {x: []}});
  exception({a: [1]}, {$pushAll: {a: [{$a: 1}]}});
  exception({a: [1]}, {$pushAll: {a: [{'\0a': 1}]}});
  exception({a: [1]}, {$pushAll: {a: [{'a.b': 1}]}});

  // $addToSet
  modify({}, {$addToSet: {a: 1}}, {a: [1]});
  modify({a: []}, {$addToSet: {a: 1}}, {a: [1]});
  modify({a: [1]}, {$addToSet: {a: 2}}, {a: [1, 2]});
  modify({a: [1, 2]}, {$addToSet: {a: 1}}, {a: [1, 2]});
  modify({a: [1, 2]}, {$addToSet: {a: 2}}, {a: [1, 2]});
  modify({a: [1, 2]}, {$addToSet: {a: 3}}, {a: [1, 2, 3]});
  exception({a: true}, {$addToSet: {a: 1}});
  modify({a: [1]}, {$addToSet: {a: [2]}}, {a: [1, [2]]});
  modify({}, {$addToSet: {a: {x: 1}}}, {a: [{x: 1}]});
  modify({a: [{x: 1}]}, {$addToSet: {a: {x: 1}}}, {a: [{x: 1}]});
  modify({a: [{x: 1}]}, {$addToSet: {a: {x: 2}}}, {a: [{x: 1}, {x: 2}]});
  modify({a: [{x: 1, y: 2}]}, {$addToSet: {a: {x: 1, y: 2}}},
    {a: [{x: 1, y: 2}]});
  modify({a: [{x: 1, y: 2}]}, {$addToSet: {a: {y: 2, x: 1}}},
    {a: [{x: 1, y: 2}, {y: 2, x: 1}]});
  modify({a: [1, 2]}, {$addToSet: {a: {$each: [3, 1, 4]}}}, {a: [1, 2, 3, 4]});
  modify({}, {$addToSet: {a: {$each: []}}}, {a: []});
  modify({}, {$addToSet: {a: {$each: [1]}}}, {a: [1]});
  modify({a: []}, {$addToSet: {'a.1': 99}}, {a: [null, [99]]});
  modify({a: {}}, {$addToSet: {'a.x': 99}}, {a: {x: [99]}});

  // invalid field names
  exception({}, {$addToSet: {a: {$b: 1}}});
  exception({}, {$addToSet: {a: {'a.b': 1}}});
  exception({}, {$addToSet: {a: {'a.': 1}}});
  exception({}, {$addToSet: {a: {'\u0000a': 1}}});
  exception({a: [1, 2]}, {$addToSet: {a: {$each: [3, 1, {$a: 1}]}}});
  exception({a: [1, 2]}, {$addToSet: {a: {$each: [3, 1, {'\0a': 1}]}}});
  exception({a: [1, 2]}, {$addToSet: {a: {$each: [3, 1, [{$a: 1}]]}}});
  exception({a: [1, 2]}, {$addToSet: {a: {$each: [3, 1, [{b: {c: [{a: 1}, {'d.s': 2}]}}]]}}});
  exception({a: [1, 2]}, {$addToSet: {a: {b: [3, 1, [{b: {c: [{a: 1}, {'d.s': 2}]}}]]}}});
  // $each is first element and thus an operator
  modify({a: [1, 2]}, {$addToSet: {a: {$each: [3, 1, 4], b: 12}}}, {a: [ 1, 2, 3, 4 ]});
  // this should fail because $each is now a field name (not first in object) and thus invalid field name with $
  exception({a: [1, 2]}, {$addToSet: {a: {b: 12, $each: [3, 1, 4]}}});

  // $pop
  modify({}, {$pop: {a: 1}}, {}); // tested
  modify({}, {$pop: {a: -1}}, {}); // tested
  modify({a: []}, {$pop: {a: 1}}, {a: []});
  modify({a: []}, {$pop: {a: -1}}, {a: []});
  modify({a: [1, 2, 3]}, {$pop: {a: 1}}, {a: [1, 2]});
  modify({a: [1, 2, 3]}, {$pop: {a: 10}}, {a: [1, 2]});
  modify({a: [1, 2, 3]}, {$pop: {a: 0.001}}, {a: [1, 2]});
  modify({a: [1, 2, 3]}, {$pop: {a: 0}}, {a: [1, 2]});
  modify({a: [1, 2, 3]}, {$pop: {a: 'stuff'}}, {a: [1, 2]});
  modify({a: [1, 2, 3]}, {$pop: {a: -1}}, {a: [2, 3]});
  modify({a: [1, 2, 3]}, {$pop: {a: -10}}, {a: [2, 3]});
  modify({a: [1, 2, 3]}, {$pop: {a: -0.001}}, {a: [2, 3]});
  exception({a: true}, {$pop: {a: 1}});
  exception({a: true}, {$pop: {a: -1}});
  modify({a: []}, {$pop: {'a.1': 1}}, {a: []}); // tested
  modify({a: [1, [2, 3], 4]}, {$pop: {'a.1': 1}}, {a: [1, [2], 4]});
  modify({a: {}}, {$pop: {'a.x': 1}}, {a: {}}); // tested
  modify({a: {x: [2, 3]}}, {$pop: {'a.x': 1}}, {a: {x: [2]}});

  // $pull
  modify({}, {$pull: {a: 1}}, {});
  modify({}, {$pull: {'a.x': 1}}, {});
  modify({a: {}}, {$pull: {'a.x': 1}}, {a: {}});
  exception({a: true}, {$pull: {a: 1}});
  modify({a: [2, 1, 2]}, {$pull: {a: 1}}, {a: [2, 2]});
  modify({a: [2, 1, 2]}, {$pull: {a: 2}}, {a: [1]});
  modify({a: [2, 1, 2]}, {$pull: {a: 3}}, {a: [2, 1, 2]});
  modify({a: [1, null, 2, null]}, {$pull: {a: null}}, {a: [1, 2]});
  modify({a: []}, {$pull: {a: 3}}, {a: []});
  modify({a: [[2], [2, 1], [3]]}, {$pull: {a: [2, 1]}},
    {a: [[2], [3]]}); // tested
  modify({a: [{b: 1, c: 2}, {b: 2, c: 2}]}, {$pull: {a: {b: 1}}},
    {a: [{b: 2, c: 2}]});
  modify({a: [{b: 1, c: 2}, {b: 2, c: 2}]}, {$pull: {a: {c: 2}}},
    {a: []});
  // XXX implement this functionality!
  // probably same refactoring as $elemMatch?
  // modify({a: [1, 2, 3, 4]}, {$pull: {$gt: 2}}, {a: [1,2]}); fails!

  // $pullAll
  modify({}, {$pullAll: {a: [1]}}, {});
  modify({a: [1, 2, 3]}, {$pullAll: {a: []}}, {a: [1, 2, 3]});
  modify({a: [1, 2, 3]}, {$pullAll: {a: [2]}}, {a: [1, 3]});
  modify({a: [1, 2, 3]}, {$pullAll: {a: [2, 1]}}, {a: [3]});
  modify({a: [1, 2, 3]}, {$pullAll: {a: [1, 2]}}, {a: [3]});
  modify({}, {$pullAll: {'a.b.c': [2]}}, {});
  exception({a: true}, {$pullAll: {a: [1]}});
  exception({a: [1, 2, 3]}, {$pullAll: {a: 1}});
  modify({x: [{a: 1}, {a: 1, b: 2}]}, {$pullAll: {x: [{a: 1}]}},
    {x: [{a: 1, b: 2}]});

  // $rename
  modify({}, {$rename: {a: 'b'}}, {});
  modify({a: [12]}, {$rename: {a: 'b'}}, {b: [12]});
  modify({a: {b: 12}}, {$rename: {a: 'c'}}, {c: {b: 12}});
  modify({a: {b: 12}}, {$rename: {'a.b': 'a.c'}}, {a: {c: 12}});
  modify({a: {b: 12}}, {$rename: {'a.b': 'x'}}, {a: {}, x: 12}); // tested
  modify({a: {b: 12}}, {$rename: {'a.b': 'q.r'}}, {a: {}, q: {r: 12}});
  modify({a: {b: 12}}, {$rename: {'a.b': 'q.2.r'}}, {a: {}, q: {2: {r: 12}}});
  modify({a: {b: 12}, q: {}}, {$rename: {'a.b': 'q.2.r'}},
    {a: {}, q: {2: {r: 12}}});
  exception({a: {b: 12}, q: []}, {$rename: {'a.b': 'q.2'}}); // tested
  exception({a: {b: 12}, q: []}, {$rename: {'a.b': 'q.2.r'}}); // tested
  // These strange MongoDB behaviors throw.
  // modify({a: {b: 12}, q: []}, {$rename: {'q.1': 'x'}},
  //        {a: {b: 12}, x: []}); // tested
  // modify({a: {b: 12}, q: []}, {$rename: {'q.1.j': 'x'}},
  //        {a: {b: 12}, x: []}); // tested
  exception({}, {$rename: {a: 'a'}});
  exception({}, {$rename: {'a.b': 'a.b'}});
  modify({a: 12, b: 13}, {$rename: {a: 'b'}}, {b: 12});
  exception({a: [12]}, {$rename: {a: '$b'}});
  exception({a: [12]}, {$rename: {a: '\0a'}});

  // $setOnInsert
  modify({a: 0}, {$setOnInsert: {a: 12}}, {a: 0});
  upsert({a: 12}, {$setOnInsert: {b: 12}}, {a: 12, b: 12});
  upsert({a: 12}, {$setOnInsert: {_id: 'test'}}, {_id: 'test', a: 12});
  upsert({'a.b': 10}, {$setOnInsert: {a: {b: 10, c: 12}}}, {a: {b: 10, c: 12}});
  upsert({'a.b': 10}, {$setOnInsert: {c: 12}}, {a: {b: 10}, c: 12});
  upsert({_id: 'test'}, {$setOnInsert: {c: 12}}, {_id: 'test', c: 12});
  upsert('test', {$setOnInsert: {c: 12}}, {_id: 'test', c: 12});
  upsertException({a: 0}, {$setOnInsert: {$a: 12}});
  upsertException({a: 0}, {$setOnInsert: {'\0a': 12}});
  upsert({a: 0}, {$setOnInsert: {b: {a: 1}}}, {a: 0, b: {a: 1}});
  upsertException({a: 0}, {$setOnInsert: {b: {$a: 1}}});
  upsertException({a: 0}, {$setOnInsert: {b: {'a.b': 1}}});
  upsertException({a: 0}, {$setOnInsert: {b: {'\0a': 1}}});

  // Test for https://github.com/meteor/meteor/issues/8775.
  upsert(
    { a: { $exists: true }},
    { $setOnInsert: { a: 123 }},
    { a: 123 }
  );

  // Tests for https://github.com/meteor/meteor/issues/8794.
  const testObjectId = new MongoID.ObjectID();
  upsert(
    { _id: testObjectId },
    { $setOnInsert: { a: 123 } },
    { _id: testObjectId, a: 123 },
  );
  upsert(
    { someOtherId: testObjectId },
    { $setOnInsert: { a: 123 } },
    { someOtherId: testObjectId, a: 123 },
  );
  upsert(
    { a: { $eq: testObjectId } },
    { $setOnInsert: { a: 123 } },
    { a: 123 },
  );
  const testDate = new Date('2017-01-01');
  upsert(
    { someDate: testDate },
    { $setOnInsert: { a: 123 } },
    { someDate: testDate, a: 123 },
  );
  upsert(
    {
      a: Object.create(null, {
        $exists: {
          writable: true,
          configurable: true,
          value: true,
        },
      }),
    },
    { $setOnInsert: { a: 123 } },
    { a: 123 },
  );
  upsert(
    { foo: { $exists: true, $type: 2 }},
    { $setOnInsert: { bar: 'baz' } },
    { bar: 'baz' }
  );
  upsert(
    { foo: {} },
    { $setOnInsert: { bar: 'baz' } },
    { foo: {}, bar: 'baz' }
  );

  // Tests for https://github.com/meteor/meteor/issues/8806
  upsert({"a": {"b": undefined, "c": null}}, {"$set": {"c": "foo"}}, {"a": {"b": undefined, "c": null}, "c": "foo"})
  upsert({"a": {"$eq": "bar" }}, {"$set": {"c": "foo"}}, {"a": "bar", "c": "foo"})
  // $all with 1 statement is similar to $eq
  upsert({"a": {"$all": ["bar"] }}, {"$set": {"c": "foo"}}, {"a": "bar", "c": "foo"})
  upsert({"a": {"$eq": "bar" }, "b": "baz"}, {"$set": {"c": "foo"}}, {"a": "bar", "b": "baz", "c": "foo"})
   upsert({"a": {"$exists": true}}, {"$set": {"c": "foo"}}, {"c": "foo"})
  upsert({"a": {"$exists": true, "$eq": "foo"}}, {"$set": {"c": "foo"}}, {"a": "foo", "c": "foo"})
  upsert({"a": {"$gt": 3, "$eq": 2}}, {"$set": {"c": "foo"}}, {"a": 2, "c": "foo"})
   // $and
  upsert({"$and": [{"a": {"$eq": "bar"}}]}, {"$set": {"c": "foo"}}, {"a": "bar", "c": "foo"})
  upsert({"$and": [{"a": {"$all": ["bar"]}}]}, {"$set": {"c": "foo"}}, {"a": "bar", "c": "foo"})
  upsert({"$and": [{"a": {"$all": ["bar"]}}]}, {"$set": {"c": "foo"}}, {"a": "bar", "c": "foo"})
   // $or with one statement is handled similar to $and
  upsert({"$or": [{"a": "bar"}]}, {"$set": {"c": "foo"}}, {"a": "bar", "c": "foo"})
   // $or with multiple statements is ignored
  upsert({"$or": [{"a": "bar"}, {"b": "baz"}]}, {"$set": {"c": "foo"}}, {"c": "foo"})
   // Negative logical operators are ignored
  upsert({"$nor": [{"a": "bar"}]}, {"$set": {"c": "foo"}}, {"c": "foo"})
   // Filter out empty objects after filtering out operators
  upsert({"a": {"$exists": true}}, {"$set": {"c": "foo"}}, {"c": "foo"})
   // But leave actual empty objects
  upsert({"a": {}}, {"$set": {"c": "foo"}}, {"a": {}, "c": "foo"})
    // Also filter out shorthand regexp notation
  upsert({"a": /a/}, {"$set": {"c": "foo"}}, {"c": "foo"})
   // Test nested fields
  upsert({"$and": [{"a.a": "foo"}, {"$or": [{"a.b": "baz"}]}]}, {"$set": {"c": "foo"}}, {"a": {"a": "foo", "b": "baz"}, "c": "foo"})
   // Test for https://github.com/meteor/meteor/issues/5294
  upsert({"a": {"$ne": 444}}, {"$push": {"a": 123}}, {"a": [123]})
   // Mod takes precedence over query
  upsert({"a": "foo"}, {"a": "bar"}, {"a": "bar"})
  upsert({"a": "foo"}, {"$set":{"a": "bar"}}, {"a": "bar"})
   // Replacement can take _id from query
  upsert({"_id": "foo", "foo": "bar"}, {"bar": "foo"}, {"_id": "foo", "bar": "foo"})
   // Replacement update keeps _id
  upsertUpdate({"_id": "foo", "bar": "baz"}, {"_id":"foo"}, {"bar": "crow"}, {"_id": "foo", "bar": "crow"});
  // Test for https://github.com/meteor/meteor/issues/9167
  upsert({key: 123, keyName: '321'}, {$set: {name: 'Todo'}}, {key: 123, keyName: '321', name: 'Todo'});
  upsertException({key: 123, "key.name": '321'}, {$set:{}});

  // Nested fields don't work with literal objects
  upsertException({"a": {}, "a.b": "foo"}, {});
   // You can't have an ambiguous ID
  upsertException({"_id":"foo"}, {"_id":"bar"});
  upsertException({"_id":"foo"}, {"$set":{"_id":"bar"}});
   // You can't set the same field twice
  upsertException({"$and": [{"a": "foo"}, {"a": "foo"}]}, {}); //not even with same value
  upsertException({"a": {"$all": ["foo", "bar"]}}, {});
  upsertException({"$and": [{"a": {"$eq": "foo"}}, {"$or": [{"a": {"$all": ["bar"]}}]}]}, {});
   // You can't have nested dotted fields
  upsertException({"a": {"foo.bar": "baz"}}, {});
   // You can't have dollar-prefixed fields above the first level (logical operators not counted)
  upsertException({"a": {"a": {"$eq": "foo"}}}, {});
  upsertException({"a": {"a": {"$exists": true}}}, {});
   // You can't mix operators with other fields
  upsertException({"a": {"$eq": "bar", "b": "foo"}}, {})
  upsertException({"a": {"b": "foo", "$eq": "bar"}}, {})

  const mongoIdForUpsert = new MongoID.ObjectID('44915733af80844fa1cef07a');
  upsert({_id: mongoIdForUpsert}, {$setOnInsert: {a: 123}}, {a: 123})

  // Test for https://github.com/meteor/meteor/issues/7758
  upsert({n_id: mongoIdForUpsert, c_n: "bar"},
    {$set: { t_t_o: "foo"}},
    {n_id: mongoIdForUpsert, t_t_o: "foo", c_n: "bar"});

  exception({}, {$set: {_id: 'bad'}});

  // $bit
  // unimplemented

  // XXX test case sensitivity of modops
  // XXX for each (most) modop, test that it performs a deep copy
});

// XXX test update() (selecting docs, multi, upsert..)

Tinytest.add('minimongo - observe ordered', test => {
  const operations = [];
  const cbs = log_callbacks(operations);
  let handle;

  const c = new LocalCollection();
  handle = c.find({}, {sort: {a: 1}}).observe(cbs);
  test.isTrue(handle.collection === c);

  c.insert({_id: 'foo', a: 1});
  test.equal(operations.shift(), ['added', {a: 1}, 0, null]);
  c.update({a: 1}, {$set: {a: 2}});
  test.equal(operations.shift(), ['changed', {a: 2}, 0, {a: 1}]);
  c.insert({a: 10});
  test.equal(operations.shift(), ['added', {a: 10}, 1, null]);
  c.update({}, {$inc: {a: 1}}, {multi: true});
  test.equal(operations.shift(), ['changed', {a: 3}, 0, {a: 2}]);
  test.equal(operations.shift(), ['changed', {a: 11}, 1, {a: 10}]);
  c.update({a: 11}, {a: 1});
  test.equal(operations.shift(), ['changed', {a: 1}, 1, {a: 11}]);
  test.equal(operations.shift(), ['moved', {a: 1}, 1, 0, 'foo']);
  c.remove({a: 2});
  test.equal(operations.shift(), undefined);
  c.remove({a: 3});
  test.equal(operations.shift(), ['removed', 'foo', 1, {a: 3}]);

  // test stop
  handle.stop();
  const idA2 = Random.id();
  c.insert({_id: idA2, a: 2});
  test.equal(operations.shift(), undefined);

  // test initial inserts (and backwards sort)
  handle = c.find({}, {sort: {a: -1}}).observe(cbs);
  test.equal(operations.shift(), ['added', {a: 2}, 0, null]);
  test.equal(operations.shift(), ['added', {a: 1}, 1, null]);
  handle.stop();

  // test _suppress_initial
  handle = c.find({}, {sort: {a: -1}}).observe(Object.assign({
    _suppress_initial: true}, cbs));
  test.equal(operations.shift(), undefined);
  c.insert({a: 100});
  test.equal(operations.shift(), ['added', {a: 100}, 0, idA2]);
  handle.stop();

  // test skip and limit.
  c.remove({});
  handle = c.find({}, {sort: {a: 1}, skip: 1, limit: 2}).observe(cbs);
  test.equal(operations.shift(), undefined);
  c.insert({a: 1});
  test.equal(operations.shift(), undefined);
  c.insert({_id: 'foo', a: 2});
  test.equal(operations.shift(), ['added', {a: 2}, 0, null]);
  c.insert({a: 3});
  test.equal(operations.shift(), ['added', {a: 3}, 1, null]);
  c.insert({a: 4});
  test.equal(operations.shift(), undefined);
  c.update({a: 1}, {a: 0});
  test.equal(operations.shift(), undefined);
  c.update({a: 0}, {a: 5});
  test.equal(operations.shift(), ['removed', 'foo', 0, {a: 2}]);
  test.equal(operations.shift(), ['added', {a: 4}, 1, null]);
  c.update({a: 3}, {a: 3.5});
  test.equal(operations.shift(), ['changed', {a: 3.5}, 0, {a: 3}]);
  handle.stop();

  // test observe limit with pre-existing docs
  c.remove({});
  c.insert({a: 1});
  c.insert({_id: 'two', a: 2});
  c.insert({a: 3});
  handle = c.find({}, {sort: {a: 1}, limit: 2}).observe(cbs);
  test.equal(operations.shift(), ['added', {a: 1}, 0, null]);
  test.equal(operations.shift(), ['added', {a: 2}, 1, null]);
  test.equal(operations.shift(), undefined);
  c.remove({a: 2});
  test.equal(operations.shift(), ['removed', 'two', 1, {a: 2}]);
  test.equal(operations.shift(), ['added', {a: 3}, 1, null]);
  test.equal(operations.shift(), undefined);
  handle.stop();

  // test _no_indices

  c.remove({});
  handle = c.find({}, {sort: {a: 1}}).observe(Object.assign(cbs, {_no_indices: true}));
  c.insert({_id: 'foo', a: 1});
  test.equal(operations.shift(), ['added', {a: 1}, -1, null]);
  c.update({a: 1}, {$set: {a: 2}});
  test.equal(operations.shift(), ['changed', {a: 2}, -1, {a: 1}]);
  c.insert({a: 10});
  test.equal(operations.shift(), ['added', {a: 10}, -1, null]);
  c.update({}, {$inc: {a: 1}}, {multi: true});
  test.equal(operations.shift(), ['changed', {a: 3}, -1, {a: 2}]);
  test.equal(operations.shift(), ['changed', {a: 11}, -1, {a: 10}]);
  c.update({a: 11}, {a: 1});
  test.equal(operations.shift(), ['changed', {a: 1}, -1, {a: 11}]);
  test.equal(operations.shift(), ['moved', {a: 1}, -1, -1, 'foo']);
  c.remove({a: 2});
  test.equal(operations.shift(), undefined);
  c.remove({a: 3});
  test.equal(operations.shift(), ['removed', 'foo', -1, {a: 3}]);
  handle.stop();
});

[true, false].forEach(ordered => {
  Tinytest.add(`minimongo - observe ordered: ${ordered}`, test => {
    const c = new LocalCollection();

    let ev = '';
    const makecb = tag => {
      const ret = {};
      ['added', 'changed', 'removed'].forEach(fn => {
        const fnName = ordered ? `${fn}At` : fn;
        ret[fnName] = doc => {
          ev = `${ev + fn.substr(0, 1) + tag + doc._id}_`;
        };
      });
      return ret;
    };
    const expect = x => {
      test.equal(ev, x);
      ev = '';
    };

    c.insert({_id: 1, name: 'strawberry', tags: ['fruit', 'red', 'squishy']});
    c.insert({_id: 2, name: 'apple', tags: ['fruit', 'red', 'hard']});
    c.insert({_id: 3, name: 'rose', tags: ['flower', 'red', 'squishy']});

    // This should work equally well for ordered and unordered observations
    // (because the callbacks don't look at indices and there's no 'moved'
    // callback).
    let handle = c.find({tags: 'flower'}).observe(makecb('a'));
    expect('aa3_');
    c.update({name: 'rose'}, {$set: {tags: ['bloom', 'red', 'squishy']}});
    expect('ra3_');
    c.update({name: 'rose'}, {$set: {tags: ['flower', 'red', 'squishy']}});
    expect('aa3_');
    c.update({name: 'rose'}, {$set: {food: false}});
    expect('ca3_');
    c.remove({});
    expect('ra3_');
    c.insert({_id: 4, name: 'daisy', tags: ['flower']});
    expect('aa4_');
    handle.stop();
    // After calling stop, no more callbacks are called.
    c.insert({_id: 5, name: 'iris', tags: ['flower']});
    expect('');

    // Test that observing a lookup by ID works.
    handle = c.find(4).observe(makecb('b'));
    expect('ab4_');
    c.update(4, {$set: {eek: 5}});
    expect('cb4_');
    handle.stop();

    // Test observe with reactive: false.
    handle = c.find({tags: 'flower'}, {reactive: false}).observe(makecb('c'));
    expect('ac4_ac5_');
    // This insert shouldn't trigger a callback because it's not reactive.
    c.insert({_id: 6, name: 'river', tags: ['flower']});
    expect('');
    handle.stop();
  });
});


Tinytest.add('minimongo - saveOriginals', test => {
  // set up some data
  const c = new LocalCollection();

  let count;
  c.insert({_id: 'foo', x: 'untouched'});
  c.insert({_id: 'bar', x: 'updateme'});
  c.insert({_id: 'baz', x: 'updateme'});
  c.insert({_id: 'quux', y: 'removeme'});
  c.insert({_id: 'whoa', y: 'removeme'});

  // Save originals and make some changes.
  c.saveOriginals();
  c.insert({_id: 'hooray', z: 'insertme'});
  c.remove({y: 'removeme'});
  count = c.update({x: 'updateme'}, {$set: {z: 5}}, {multi: true});
  c.update('bar', {$set: {k: 7}});  // update same doc twice

  // Verify returned count is correct
  test.equal(count, 2);

  // Verify the originals.
  let originals = c.retrieveOriginals();
  const affected = ['bar', 'baz', 'quux', 'whoa', 'hooray'];
  test.equal(originals.size(), affected.length);
  affected.forEach(id => {
    test.isTrue(originals.has(id));
  });
  test.equal(originals.get('bar'), {_id: 'bar', x: 'updateme'});
  test.equal(originals.get('baz'), {_id: 'baz', x: 'updateme'});
  test.equal(originals.get('quux'), {_id: 'quux', y: 'removeme'});
  test.equal(originals.get('whoa'), {_id: 'whoa', y: 'removeme'});
  test.equal(originals.get('hooray'), undefined);

  // Verify that changes actually occured.
  test.equal(c.find().count(), 4);
  test.equal(c.findOne('foo'), {_id: 'foo', x: 'untouched'});
  test.equal(c.findOne('bar'), {_id: 'bar', x: 'updateme', z: 5, k: 7});
  test.equal(c.findOne('baz'), {_id: 'baz', x: 'updateme', z: 5});
  test.equal(c.findOne('hooray'), {_id: 'hooray', z: 'insertme'});

  // The next call doesn't get the same originals again.
  c.saveOriginals();
  originals = c.retrieveOriginals();
  test.isTrue(originals);
  test.isTrue(originals.empty());

  // Insert and remove a document during the period.
  c.saveOriginals();
  c.insert({_id: 'temp', q: 8});
  c.remove('temp');
  originals = c.retrieveOriginals();
  test.equal(originals.size(), 1);
  test.isTrue(originals.has('temp'));
  test.equal(originals.get('temp'), undefined);
});

Tinytest.add('minimongo - saveOriginals errors', test => {
  const c = new LocalCollection();
  // Can't call retrieve before save.
  test.throws(() => { c.retrieveOriginals(); });
  c.saveOriginals();
  // Can't call save twice.
  test.throws(() => { c.saveOriginals(); });
});

Tinytest.add('minimongo - objectid transformation', test => {
  const testId = item => {
    test.equal(item, MongoID.idParse(MongoID.idStringify(item)));
  };
  const randomOid = new MongoID.ObjectID();
  testId(randomOid);
  testId('FOO');
  testId('ffffffffffff');
  testId('0987654321abcdef09876543');
  testId(new MongoID.ObjectID());
  testId('--a string');

  test.equal('ffffffffffff', MongoID.idParse(MongoID.idStringify('ffffffffffff')));
});


Tinytest.add('minimongo - objectid', test => {
  const randomOid = new MongoID.ObjectID();
  const anotherRandomOid = new MongoID.ObjectID();
  test.notEqual(randomOid, anotherRandomOid);
  test.throws(() => { new MongoID.ObjectID('qqqqqqqqqqqqqqqqqqqqqqqq');});
  test.throws(() => { new MongoID.ObjectID('ABCDEF'); });
  test.equal(randomOid, new MongoID.ObjectID(randomOid.valueOf()));
});

Tinytest.add('minimongo - pause', test => {
  const operations = [];
  const cbs = log_callbacks(operations);

  const c = new LocalCollection();
  const h = c.find({}).observe(cbs);

  // remove and add cancel out.
  c.insert({_id: 1, a: 1});
  test.equal(operations.shift(), ['added', {a: 1}, 0, null]);

  c.pauseObservers();

  c.remove({_id: 1});
  test.length(operations, 0);
  c.insert({_id: 1, a: 1});
  test.length(operations, 0);

  c.resumeObservers();
  test.length(operations, 0);


  // two modifications become one
  c.pauseObservers();

  c.update({_id: 1}, {a: 2});
  c.update({_id: 1}, {a: 3});

  c.resumeObservers();
  test.equal(operations.shift(), ['changed', {a: 3}, 0, {a: 1}]);
  test.length(operations, 0);

  // test special case for remove({})
  c.pauseObservers();
  test.equal(c.remove({}), 1);
  test.length(operations, 0);
  c.resumeObservers();
  test.equal(operations.shift(), ['removed', 1, 0, {a: 3}]);
  test.length(operations, 0);

  h.stop();
});

Tinytest.add('minimongo - ids matched by selector', test => {
  const check = (selector, ids) => {
    const idsFromSelector = LocalCollection._idsMatchedBySelector(selector);
    // XXX normalize order, in a way that also works for ObjectIDs?
    test.equal(idsFromSelector, ids);
  };
  check('foo', ['foo']);
  check({_id: 'foo'}, ['foo']);
  const oid1 = new MongoID.ObjectID();
  check(oid1, [oid1]);
  check({_id: oid1}, [oid1]);
  check({_id: 'foo', x: 42}, ['foo']);
  check({}, null);
  check({_id: {$in: ['foo', oid1]}}, ['foo', oid1]);
  check({_id: {$ne: 'foo'}}, null);
  // not actually valid, but works for now...
  check({$and: ['foo']}, ['foo']);
  check({$and: [{x: 42}, {_id: oid1}]}, [oid1]);
  check({$and: [{x: 42}, {_id: {$in: [oid1]}}]}, [oid1]);
});

Tinytest.add('minimongo - reactive stop', test => {
  const coll = new LocalCollection();
  coll.insert({_id: 'A'});
  coll.insert({_id: 'B'});
  coll.insert({_id: 'C'});

  const addBefore = (str, newChar, before) => {
    const idx = str.indexOf(before);
    if (idx === -1) {return str + newChar;}
    return str.slice(0, idx) + newChar + str.slice(idx);
  };

  let x, y;
  const sortOrder = ReactiveVar(1);

  const c = Tracker.autorun(() => {
    const q = coll.find({}, {sort: {_id: sortOrder.get()}});
    x = '';
    q.observe({ addedAt(doc, atIndex, before) {
      x = addBefore(x, doc._id, before);
    }});
    y = '';
    q.observeChanges({ addedBefore(id, fields, before) {
      y = addBefore(y, id, before);
    }});
  });

  test.equal(x, 'ABC');
  test.equal(y, 'ABC');

  sortOrder.set(-1);
  test.equal(x, 'ABC');
  test.equal(y, 'ABC');
  Tracker.flush();
  test.equal(x, 'CBA');
  test.equal(y, 'CBA');

  coll.insert({_id: 'D'});
  coll.insert({_id: 'E'});
  test.equal(x, 'EDCBA');
  test.equal(y, 'EDCBA');

  c.stop();
  // stopping kills the observes immediately
  coll.insert({_id: 'F'});
  test.equal(x, 'EDCBA');
  test.equal(y, 'EDCBA');
});

Tinytest.add('minimongo - immediate invalidate', test => {
  const coll = new LocalCollection();
  coll.insert({_id: 'A'});

  // This has two separate findOnes.  findOne() uses skip/limit, which means
  // that its response to an update() call involves a recompute. We used to have
  // a bug where we would first calculate all the calls that need to be
  // recomputed, then recompute them one by one, without checking to see if the
  // callbacks from recomputing one query stopped the second query, which
  // crashed.
  const c = Tracker.autorun(() => {
    coll.findOne('A');
    coll.findOne('A');
  });

  coll.update('A', {$set: {x: 42}});

  c.stop();
});


Tinytest.add('minimongo - count on cursor with limit', test => {
  const coll = new LocalCollection();
  let count, unlimitedCount;

  coll.insert({_id: 'A'});
  coll.insert({_id: 'B'});
  coll.insert({_id: 'C'});
  coll.insert({_id: 'D'});

  const c = Tracker.autorun(c => {
    const cursor = coll.find({_id: {$exists: true}}, {sort: {_id: 1}, limit: 3});
    count = cursor.count();
  });

  test.equal(count, 3);

  coll.remove('A'); // still 3 in the collection
  Tracker.flush();
  test.equal(count, 3);

  coll.remove('B'); // expect count now 2
  Tracker.flush();
  test.equal(count, 2);


  coll.insert({_id: 'A'}); // now 3 again
  Tracker.flush();
  test.equal(count, 3);

  coll.insert({_id: 'B'}); // now 4 entries, but count should be 3 still
  Tracker.flush();
  test.equal(count, 3);

  c.stop();
});

Tinytest.add('minimongo - reactive count with cached cursor', test => {
  const coll = new LocalCollection;
  const cursor = coll.find({});
  let firstAutorunCount, secondAutorunCount;
  Tracker.autorun(() => {
    firstAutorunCount = cursor.count();
  });
  Tracker.autorun(() => {
    secondAutorunCount = coll.find({}).count();
  });
  test.equal(firstAutorunCount, 0);
  test.equal(secondAutorunCount, 0);
  coll.insert({i: 1});
  coll.insert({i: 2});
  coll.insert({i: 3});
  Tracker.flush();
  test.equal(firstAutorunCount, 3);
  test.equal(secondAutorunCount, 3);
});

Tinytest.add('minimongo - $near operator tests', test => {
  let coll = new LocalCollection();
  coll.insert({ rest: { loc: [2, 3] } });
  coll.insert({ rest: { loc: [-3, 3] } });
  coll.insert({ rest: { loc: [5, 5] } });

  test.equal(coll.find({ 'rest.loc': { $near: [0, 0], $maxDistance: 30 } }).count(), 3);
  test.equal(coll.find({ 'rest.loc': { $near: [0, 0], $maxDistance: 4 } }).count(), 1);
  const points = coll.find({ 'rest.loc': { $near: [0, 0], $maxDistance: 6 } }).fetch();
  points.forEach((point, i, points) => {
    test.isTrue(!i || distance([0, 0], point.rest.loc) >= distance([0, 0], points[i - 1].rest.loc));
  });

  function distance(a, b) {
    const x = a[0] - b[0];
    const y = a[1] - b[1];
    return Math.sqrt(x * x + y * y);
  }

  // GeoJSON tests
  coll = new LocalCollection();
  const data = [{ category: 'BURGLARY', descript: 'BURGLARY OF STORE, FORCIBLE ENTRY', address: '100 Block of 10TH ST', location: { type: 'Point', coordinates: [  -122.415449723856,  37.7749518087273 ] } },
    { category: 'WEAPON LAWS', descript: 'POSS OF PROHIBITED WEAPON', address: '900 Block of MINNA ST', location: { type: 'Point', coordinates: [  -122.415386041221,  37.7747879744156 ] } },
    { category: 'LARCENY/THEFT', descript: 'GRAND THEFT OF PROPERTY', address: '900 Block of MINNA ST', location: { type: 'Point', coordinates: [  -122.41538270191,  37.774683628213 ] } },
    { category: 'LARCENY/THEFT', descript: 'PETTY THEFT FROM LOCKED AUTO', address: '900 Block of MINNA ST', location: { type: 'Point', coordinates: [  -122.415396041221,  37.7747879744156 ] } },
    { category: 'OTHER OFFENSES', descript: 'POSSESSION OF BURGLARY TOOLS', address: '900 Block of MINNA ST', location: { type: 'Point', coordinates: [  -122.415386041221,  37.7747879734156 ] } },
  ];

  data.forEach((x, i) => { coll.insert(Object.assign(x, { x: i })); });

  const close15 = coll.find({ location: { $near: {
    $geometry: { type: 'Point',
      coordinates: [-122.4154282, 37.7746115] },
    $maxDistance: 15 } } }).fetch();
  test.length(close15, 1);
  test.equal(close15[0].descript, 'GRAND THEFT OF PROPERTY');

  const close20 = coll.find({ location: { $near: {
    $geometry: { type: 'Point',
      coordinates: [-122.4154282, 37.7746115] },
    $maxDistance: 20 } } }).fetch();
  test.length(close20, 4);
  test.equal(close20[0].descript, 'GRAND THEFT OF PROPERTY');
  test.equal(close20[1].descript, 'PETTY THEFT FROM LOCKED AUTO');
  test.equal(close20[2].descript, 'POSSESSION OF BURGLARY TOOLS');
  test.equal(close20[3].descript, 'POSS OF PROHIBITED WEAPON');

  // Any combinations of $near with $or/$and/$nor/$not should throw an error
  test.throws(() => {
    coll.find({ location: {
      $not: {
        $near: {
          $geometry: {
            type: 'Point',
            coordinates: [-122.4154282, 37.7746115],
          }, $maxDistance: 20 } } } });
  });
  test.throws(() => {
    coll.find({
      $and: [ { location: { $near: { $geometry: { type: 'Point', coordinates: [-122.4154282, 37.7746115] }, $maxDistance: 20 }}},
        { x: 0 }],
    });
  });
  test.throws(() => {
    coll.find({
      $or: [ { location: { $near: { $geometry: { type: 'Point', coordinates: [-122.4154282, 37.7746115] }, $maxDistance: 20 }}},
        { x: 0 }],
    });
  });
  test.throws(() => {
    coll.find({
      $nor: [ { location: { $near: { $geometry: { type: 'Point', coordinates: [-122.4154282, 37.7746115] }, $maxDistance: 1 }}},
        { x: 0 }],
    });
  });
  test.throws(() => {
    coll.find({
      $and: [{
        $and: [{
          location: {
            $near: {
              $geometry: {
                type: 'Point',
                coordinates: [-122.4154282, 37.7746115],
              },
              $maxDistance: 1,
            },
          },
        }],
      }],
    });
  });

  // array tests
  coll = new LocalCollection();
  coll.insert({
    _id: 'x',
    k: 9,
    a: [
      {b: [
        [100, 100],
        [1,  1]]},
      {b: [150,  150]}]});
  coll.insert({
    _id: 'y',
    k: 9,
    a: {b: [5, 5]}});
  const testNear = (near, md, expected) => {
    test.equal(
      coll.find({'a.b': {$near: near, $maxDistance: md}}).fetch().map(doc => doc._id),
      expected);
  };
  testNear([149, 149], 4, ['x']);
  testNear([149, 149], 1000, ['x', 'y']);
  // It's important that we figure out that 'x' is closer than 'y' to [2,2] even
  // though the first within-1000 point in 'x' (ie, [100,100]) is farther than
  // 'y'.
  testNear([2, 2], 1000, ['x', 'y']);

  // issue #3599
  // Ensure that distance is not used as a tie-breaker for sort.
  test.equal(
    coll.find({'a.b': {$near: [1, 1]}}, {sort: {k: 1}}).fetch().map(doc => doc._id),
    ['x', 'y']);
  test.equal(
    coll.find({'a.b': {$near: [5, 5]}}, {sort: {k: 1}}).fetch().map(doc => doc._id),
    ['x', 'y']);

  const operations = [];
  const cbs = log_callbacks(operations);
  const handle = coll.find({'a.b': {$near: [7, 7]}}).observe(cbs);

  test.length(operations, 2);
  test.equal(operations.shift(), ['added', {k: 9, a: {b: [5, 5]}}, 0, null]);
  test.equal(operations.shift(),
    ['added', {k: 9, a: [{b: [[100, 100], [1, 1]]}, {b: [150, 150]}]},
      1, null]);
  // This needs to be inserted in the MIDDLE of the two existing ones.
  coll.insert({a: {b: [3, 3]}});
  test.length(operations, 1);
  test.equal(operations.shift(), ['added', {a: {b: [3, 3]}}, 1, 'x']);

  handle.stop();
});

// issue #2077
Tinytest.add('minimongo - $near and $geometry for legacy coordinates', test => {
  const coll = new LocalCollection();

  coll.insert({
    loc: {
      x: 1,
      y: 1,
    },
  });
  coll.insert({
    loc: [-1, -1],
  });
  coll.insert({
    loc: [40, -10],
  });
  coll.insert({
    loc: {
      x: -10,
      y: 40,
    },
  });

  test.equal(coll.find({ loc: { $near: [0, 0], $maxDistance: 4 } }).count(), 2);
  test.equal(coll.find({ loc: { $near: {$geometry: {type: 'Point', coordinates: [0, 0]}}} }).count(), 4);
  test.equal(coll.find({ loc: { $near: {$geometry: {type: 'Point', coordinates: [0, 0]}, $maxDistance: 200000}}}).count(), 2);
});

// Regression test for #4377. Previously, "replace" updates didn't clone the
// argument.
Tinytest.add('minimongo - update should clone', test => {
  const x = [];
  const coll = new LocalCollection;
  const id = coll.insert({});
  coll.update(id, {x});
  x.push(1);
  test.equal(coll.findOne(id), {_id: id, x: []});
});

// See #2275.
Tinytest.add('minimongo - fetch in observe', test => {
  const coll = new LocalCollection;
  let callbackInvoked = false;
  const observe = coll.find().observeChanges({
    added(id, fields) {
      callbackInvoked = true;
      test.equal(fields, {foo: 1});
      const doc = coll.findOne({foo: 1});
      test.isTrue(doc);
      test.equal(doc.foo, 1);
    },
  });
  test.isFalse(callbackInvoked);
  const computation = Tracker.autorun(computation => {
    if (computation.firstRun) {
      coll.insert({foo: 1});
    }
  });
  test.isTrue(callbackInvoked);
  observe.stop();
  computation.stop();
});

// See #2254
Tinytest.add('minimongo - fine-grained reactivity of observe with fields projection', test => {
  const X = new LocalCollection;
  const id = 'asdf';
  X.insert({_id: id, foo: {bar: 123}});

  let callbackInvoked = false;
  const obs = X.find(id, {fields: {'foo.bar': 1}}).observeChanges({
    changed(id, fields) {
      callbackInvoked = true;
    },
  });

  test.isFalse(callbackInvoked);
  X.update(id, {$set: {'foo.baz': 456}});
  test.isFalse(callbackInvoked);

  obs.stop();
});
Tinytest.add('minimongo - fine-grained reactivity of query with fields projection', test => {
  const X = new LocalCollection;
  const id = 'asdf';
  X.insert({_id: id, foo: {bar: 123}});

  let callbackInvoked = false;
  const computation = Tracker.autorun(() => {
    callbackInvoked = true;
    return X.findOne(id, { fields: { 'foo.bar': 1 } });
  });
  test.isTrue(callbackInvoked);
  callbackInvoked = false;
  X.update(id, {$set: {'foo.baz': 456}});
  test.isFalse(callbackInvoked);
  X.update(id, {$set: {'foo.bar': 124}});
  Tracker.flush();
  test.isTrue(callbackInvoked);

  computation.stop();
});

// Tests that the logic in `LocalCollection.prototype.update`
// correctly deals with count() on a cursor with skip or limit (since
// then the result set is an IdMap, not an array)
Tinytest.add('minimongo - reactive skip/limit count while updating', test => {
  const X = new LocalCollection;
  let count = -1;

  const c = Tracker.autorun(() => {
    count = X.find({}, {skip: 1, limit: 1}).count();
  });

  test.equal(count, 0);

  X.insert({});
  Tracker.flush({_throwFirstError: true});
  test.equal(count, 0);

  X.insert({});
  Tracker.flush({_throwFirstError: true});
  test.equal(count, 1);

  X.update({}, {$set: {foo: 1}});
  Tracker.flush({_throwFirstError: true});
  test.equal(count, 1);

  // Make sure a second update also works
  X.update({}, {$set: {foo: 2}});
  Tracker.flush({_throwFirstError: true});
  test.equal(count, 1);

  c.stop();
});

// Makes sure inserts cannot be performed using field names that have
// Mongo restricted characters in them ('.', '$', '\0'):
// https://docs.mongodb.com/manual/reference/limits/#Restrictions-on-Field-Names
Tinytest.add('minimongo - cannot insert using invalid field names', test => {
  const collection = new LocalCollection();

  // Quick test to make sure non-dot field inserts are working
  collection.insert({ a: 'b' });

  // Quick test to make sure field values with dots are allowed
  collection.insert({ a: 'b.c' });

  // Verify top level dot-field inserts are prohibited
  ['a.b', '.b', 'a.', 'a.b.c'].forEach((field) => {
    test.throws(() => {
      collection.insert({ [field]: 'c' });
    }, `Key ${field} must not contain '.'`);
  });

  // Verify nested dot-field inserts are prohibited
  test.throws(() => {
    collection.insert({ a: { b: { 'c.d': 'e' } } });
  }, "Key c.d must not contain '.'");

  // Verify field names starting with $ are prohibited
  test.throws(() => {
    collection.insert({ $a: 'b' });
  }, "Key $a must not start with '$'");

  // Verify nested field names starting with $ are prohibited
  test.throws(() => {
    collection.insert({ a: { b: { $c: 'd' } } });
  }, "Key $c must not start with '$'");

  // Verify top level fields with null characters are prohibited
  ['\0a', 'a\0', 'a\0b', '\u0000a', 'a\u0000', 'a\u0000b'].forEach((field) => {
    test.throws(() => {
      collection.insert({ [field]: 'c' });
    }, `Key ${field} must not contain null bytes`);
  });

  // Verify nested field names with null characters are prohibited
  test.throws(() => {
    collection.insert({ a: { b: { '\0c': 'd' } } });
  }, 'Key \0c must not contain null bytes');
});

// Makes sure $set's cannot be performed using null bytes
// https://docs.mongodb.com/manual/reference/limits/#Restrictions-on-Field-Names
Tinytest.add('minimongo - cannot $set with null bytes', test => {
  const collection = new LocalCollection();

  // Quick test to make sure non-null byte $set's are working
  const id = collection.insert({ a: 'b', c: 'd' });
  collection.update({ _id: id }, { $set: { e: 'f' } });

  // Verify $set's with null bytes throw an exception
  test.throws(() => {
    collection.update({ _id: id }, { $set: { '\0a': 'b' } });
  }, 'Key \0a must not contain null bytes');
});

// Makes sure $rename's cannot be performed using null bytes
// https://docs.mongodb.com/manual/reference/limits/#Restrictions-on-Field-Names
Tinytest.add('minimongo - cannot $rename with null bytes', test => {
  const collection = new LocalCollection();

  // Quick test to make sure non-null byte $rename's are working
  let id = collection.insert({ a: 'b', c: 'd' });
  collection.update({ _id: id }, { $rename: { a: 'a1', c: 'c1' } });

  // Verify $rename's with null bytes throw an exception
  collection.remove({});
  id = collection.insert({ a: 'b', c: 'd' });
  test.throws(() => {
    collection.update({ _id: id }, { $rename: { a: '\0a', c: 'c\0' } });
  }, "The 'to' field for $rename cannot contain an embedded null byte");
});