You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					1880 lines
				
				62 KiB
			
		
		
			
		
	
	
					1880 lines
				
				62 KiB
			| 
											3 years ago
										 | 'use strict'; | ||
|  | 
 | ||
|  | /** | ||
|  |  * Test dependencies. | ||
|  |  */ | ||
|  | 
 | ||
|  | const mpath = require('../'); | ||
|  | const assert = require('assert'); | ||
|  | 
 | ||
|  | /** | ||
|  |  * logging helper | ||
|  |  */ | ||
|  | 
 | ||
|  | function log(o) { | ||
|  |   console.log(); | ||
|  |   console.log(require('util').inspect(o, false, 1000)); | ||
|  | } | ||
|  | 
 | ||
|  | /** | ||
|  |  * special path for override tests | ||
|  |  */ | ||
|  | 
 | ||
|  | const special = '_doc'; | ||
|  | 
 | ||
|  | /** | ||
|  |  * Tests | ||
|  |  */ | ||
|  | 
 | ||
|  | describe('mpath', function() { | ||
|  | 
 | ||
|  |   /** | ||
|  |    * test doc creator | ||
|  |    */ | ||
|  | 
 | ||
|  |   function doc() { | ||
|  |     const o = { first: { second: { third: [3, { name: 'aaron' }, 9] } } }; | ||
|  |     o.comments = [ | ||
|  |       { name: 'one' }, | ||
|  |       { name: 'two', _doc: { name: '2' } }, | ||
|  |       { name: 'three', | ||
|  |         comments: [{}, { comments: [{ val: 'twoo' }] }], | ||
|  |         _doc: { name: '3', comments: [{}, { _doc: { comments: [{ val: 2 }] } }] } } | ||
|  |     ]; | ||
|  |     o.name = 'jiro'; | ||
|  |     o.array = [ | ||
|  |       { o: { array: [{ x: { b: [4, 6, 8] } }, { y: 10 }] } }, | ||
|  |       { o: { array: [{ x: { b: [1, 2, 3] } }, { x: { z: 10 } }, { x: { b: 'hi' } }] } }, | ||
|  |       { o: { array: [{ x: { b: null } }, { x: { b: [null, 1] } }] } }, | ||
|  |       { o: { array: [{ x: null }] } }, | ||
|  |       { o: { array: [{ y: 3 }] } }, | ||
|  |       { o: { array: [3, 0, null] } }, | ||
|  |       { o: { name: 'ha' } } | ||
|  |     ]; | ||
|  |     o.arr = [ | ||
|  |       { arr: [{ a: { b: 47 } }, { a: { c: 48 } }, { d: 'yep' }] }, | ||
|  |       { yep: true } | ||
|  |     ]; | ||
|  |     return o; | ||
|  |   } | ||
|  | 
 | ||
|  |   describe('get', function() { | ||
|  |     const o = doc(); | ||
|  | 
 | ||
|  |     it('`path` must be a string or array', function(done) { | ||
|  |       assert.throws(function() { | ||
|  |         mpath.get({}, o); | ||
|  |       }, /Must be either string or array/); | ||
|  |       assert.throws(function() { | ||
|  |         mpath.get(4, o); | ||
|  |       }, /Must be either string or array/); | ||
|  |       assert.throws(function() { | ||
|  |         mpath.get(function() {}, o); | ||
|  |       }, /Must be either string or array/); | ||
|  |       assert.throws(function() { | ||
|  |         mpath.get(/asdf/, o); | ||
|  |       }, /Must be either string or array/); | ||
|  |       assert.throws(function() { | ||
|  |         mpath.get(Math, o); | ||
|  |       }, /Must be either string or array/); | ||
|  |       assert.throws(function() { | ||
|  |         mpath.get(Buffer, o); | ||
|  |       }, /Must be either string or array/); | ||
|  |       assert.doesNotThrow(function() { | ||
|  |         mpath.get('string', o); | ||
|  |       }); | ||
|  |       assert.doesNotThrow(function() { | ||
|  |         mpath.get([], o); | ||
|  |       }); | ||
|  |       done(); | ||
|  |     }); | ||
|  | 
 | ||
|  |     describe('without `special`', function() { | ||
|  |       it('works', function(done) { | ||
|  |         assert.equal('jiro', mpath.get('name', o)); | ||
|  | 
 | ||
|  |         assert.deepEqual( | ||
|  |           { second: { third: [3, { name: 'aaron' }, 9] } } | ||
|  |           , mpath.get('first', o) | ||
|  |         ); | ||
|  | 
 | ||
|  |         assert.deepEqual( | ||
|  |           { third: [3, { name: 'aaron' }, 9] } | ||
|  |           , mpath.get('first.second', o) | ||
|  |         ); | ||
|  | 
 | ||
|  |         assert.deepEqual( | ||
|  |           [3, { name: 'aaron' }, 9] | ||
|  |           , mpath.get('first.second.third', o) | ||
|  |         ); | ||
|  | 
 | ||
|  |         assert.deepEqual( | ||
|  |           3 | ||
|  |           , mpath.get('first.second.third.0', o) | ||
|  |         ); | ||
|  | 
 | ||
|  |         assert.deepEqual( | ||
|  |           9 | ||
|  |           , mpath.get('first.second.third.2', o) | ||
|  |         ); | ||
|  | 
 | ||
|  |         assert.deepEqual( | ||
|  |           { name: 'aaron' } | ||
|  |           , mpath.get('first.second.third.1', o) | ||
|  |         ); | ||
|  | 
 | ||
|  |         assert.deepEqual( | ||
|  |           'aaron' | ||
|  |           , mpath.get('first.second.third.1.name', o) | ||
|  |         ); | ||
|  | 
 | ||
|  |         assert.deepEqual([ | ||
|  |           { name: 'one' }, | ||
|  |           { name: 'two', _doc: { name: '2' } }, | ||
|  |           { name: 'three', | ||
|  |             comments: [{}, { comments: [{ val: 'twoo' }] }], | ||
|  |             _doc: { name: '3', comments: [{}, { _doc: { comments: [{ val: 2 }] } }] } }], | ||
|  |         mpath.get('comments', o)); | ||
|  | 
 | ||
|  |         assert.deepEqual({ name: 'one' }, mpath.get('comments.0', o)); | ||
|  |         assert.deepEqual('one', mpath.get('comments.0.name', o)); | ||
|  |         assert.deepEqual('two', mpath.get('comments.1.name', o)); | ||
|  |         assert.deepEqual('three', mpath.get('comments.2.name', o)); | ||
|  | 
 | ||
|  |         assert.deepEqual([{}, { comments: [{ val: 'twoo' }] }] | ||
|  |           , mpath.get('comments.2.comments', o)); | ||
|  | 
 | ||
|  |         assert.deepEqual({ comments: [{ val: 'twoo' }] } | ||
|  |           , mpath.get('comments.2.comments.1', o)); | ||
|  | 
 | ||
|  |         assert.deepEqual('twoo', mpath.get('comments.2.comments.1.comments.0.val', o)); | ||
|  | 
 | ||
|  |         done(); | ||
|  |       }); | ||
|  | 
 | ||
|  |       it('handles array.property dot-notation', function(done) { | ||
|  |         assert.deepEqual( | ||
|  |           ['one', 'two', 'three'] | ||
|  |           , mpath.get('comments.name', o) | ||
|  |         ); | ||
|  |         done(); | ||
|  |       }); | ||
|  | 
 | ||
|  |       it('handles array.array notation', function(done) { | ||
|  |         assert.deepEqual( | ||
|  |           [undefined, undefined, [{}, { comments: [{ val: 'twoo' }] }]] | ||
|  |           , mpath.get('comments.comments', o) | ||
|  |         ); | ||
|  |         done(); | ||
|  |       }); | ||
|  | 
 | ||
|  |       it('handles prop.prop.prop.arrayProperty notation', function(done) { | ||
|  |         assert.deepEqual( | ||
|  |           [undefined, 'aaron', undefined] | ||
|  |           , mpath.get('first.second.third.name', o) | ||
|  |         ); | ||
|  |         assert.deepEqual( | ||
|  |           [1, 'aaron', 1] | ||
|  |           , mpath.get('first.second.third.name', o, function(v) { | ||
|  |             return undefined === v ? 1 : v; | ||
|  |           }) | ||
|  |         ); | ||
|  |         done(); | ||
|  |       }); | ||
|  | 
 | ||
|  |       it('handles array.prop.array', function(done) { | ||
|  |         assert.deepEqual( | ||
|  |           [[{ x: { b: [4, 6, 8] } }, { y: 10 }], | ||
|  |             [{ x: { b: [1, 2, 3] } }, { x: { z: 10 } }, { x: { b: 'hi' } }], | ||
|  |             [{ x: { b: null } }, { x: { b: [null, 1] } }], | ||
|  |             [{ x: null }], | ||
|  |             [{ y: 3 }], | ||
|  |             [3, 0, null], | ||
|  |             undefined | ||
|  |           ] | ||
|  |           , mpath.get('array.o.array', o) | ||
|  |         ); | ||
|  |         done(); | ||
|  |       }); | ||
|  | 
 | ||
|  |       it('handles array.prop.array.index', function(done) { | ||
|  |         assert.deepEqual( | ||
|  |           [{ x: { b: [4, 6, 8] } }, | ||
|  |             { x: { b: [1, 2, 3] } }, | ||
|  |             { x: { b: null } }, | ||
|  |             { x: null }, | ||
|  |             { y: 3 }, | ||
|  |             3, | ||
|  |             undefined | ||
|  |           ] | ||
|  |           , mpath.get('array.o.array.0', o) | ||
|  |         ); | ||
|  |         done(); | ||
|  |       }); | ||
|  | 
 | ||
|  |       it('handles array.prop.array.index.prop', function(done) { | ||
|  |         assert.deepEqual( | ||
|  |           [{ b: [4, 6, 8] }, | ||
|  |             { b: [1, 2, 3] }, | ||
|  |             { b: null }, | ||
|  |             null, | ||
|  |             undefined, | ||
|  |             undefined, | ||
|  |             undefined | ||
|  |           ] | ||
|  |           , mpath.get('array.o.array.0.x', o) | ||
|  |         ); | ||
|  |         done(); | ||
|  |       }); | ||
|  | 
 | ||
|  |       it('handles array.prop.array.prop', function(done) { | ||
|  |         assert.deepEqual( | ||
|  |           [[undefined, 10], | ||
|  |             [undefined, undefined, undefined], | ||
|  |             [undefined, undefined], | ||
|  |             [undefined], | ||
|  |             [3], | ||
|  |             [undefined, undefined, undefined], | ||
|  |             undefined | ||
|  |           ] | ||
|  |           , mpath.get('array.o.array.y', o) | ||
|  |         ); | ||
|  |         assert.deepEqual( | ||
|  |           [[{ b: [4, 6, 8] }, undefined], | ||
|  |             [{ b: [1, 2, 3] }, { z: 10 }, { b: 'hi' }], | ||
|  |             [{ b: null }, { b: [null, 1] }], | ||
|  |             [null], | ||
|  |             [undefined], | ||
|  |             [undefined, undefined, undefined], | ||
|  |             undefined | ||
|  |           ] | ||
|  |           , mpath.get('array.o.array.x', o) | ||
|  |         ); | ||
|  |         done(); | ||
|  |       }); | ||
|  | 
 | ||
|  |       it('handles array.prop.array.prop.prop', function(done) { | ||
|  |         assert.deepEqual( | ||
|  |           [[[4, 6, 8], undefined], | ||
|  |             [[1, 2, 3], undefined, 'hi'], | ||
|  |             [null, [null, 1]], | ||
|  |             [null], | ||
|  |             [undefined], | ||
|  |             [undefined, undefined, undefined], | ||
|  |             undefined | ||
|  |           ] | ||
|  |           , mpath.get('array.o.array.x.b', o) | ||
|  |         ); | ||
|  |         done(); | ||
|  |       }); | ||
|  | 
 | ||
|  |       it('handles array.prop.array.prop.prop.index', function(done) { | ||
|  |         assert.deepEqual( | ||
|  |           [[6, undefined], | ||
|  |             [2, undefined, 'i'], // undocumented feature (string indexing)
 | ||
|  |             [null, 1], | ||
|  |             [null], | ||
|  |             [undefined], | ||
|  |             [undefined, undefined, undefined], | ||
|  |             undefined | ||
|  |           ] | ||
|  |           , mpath.get('array.o.array.x.b.1', o) | ||
|  |         ); | ||
|  |         assert.deepEqual( | ||
|  |           [[6, 0], | ||
|  |             [2, 0, 'i'], // undocumented feature (string indexing)
 | ||
|  |             [null, 1], | ||
|  |             [null], | ||
|  |             [0], | ||
|  |             [0, 0, 0], | ||
|  |             0 | ||
|  |           ] | ||
|  |           , mpath.get('array.o.array.x.b.1', o, function(v) { | ||
|  |             return undefined === v ? 0 : v; | ||
|  |           }) | ||
|  |         ); | ||
|  |         done(); | ||
|  |       }); | ||
|  | 
 | ||
|  |       it('handles array.index.prop.prop', function(done) { | ||
|  |         assert.deepEqual( | ||
|  |           [{ x: { b: [1, 2, 3] } }, { x: { z: 10 } }, { x: { b: 'hi' } }] | ||
|  |           , mpath.get('array.1.o.array', o) | ||
|  |         ); | ||
|  |         assert.deepEqual( | ||
|  |           ['hi', 'hi', 'hi'] | ||
|  |           , mpath.get('array.1.o.array', o, function(v) { | ||
|  |             if (Array.isArray(v)) { | ||
|  |               return v.map(function(val) { | ||
|  |                 return 'hi'; | ||
|  |               }); | ||
|  |             } | ||
|  |             return v; | ||
|  |           }) | ||
|  |         ); | ||
|  |         done(); | ||
|  |       }); | ||
|  | 
 | ||
|  |       it('handles array.array.index', function(done) { | ||
|  |         assert.deepEqual( | ||
|  |           [{ a: { c: 48 } }, undefined] | ||
|  |           , mpath.get('arr.arr.1', o) | ||
|  |         ); | ||
|  |         assert.deepEqual( | ||
|  |           ['woot', undefined] | ||
|  |           , mpath.get('arr.arr.1', o, function(v) { | ||
|  |             if (v && v.a && v.a.c) return 'woot'; | ||
|  |             return v; | ||
|  |           }) | ||
|  |         ); | ||
|  |         done(); | ||
|  |       }); | ||
|  | 
 | ||
|  |       it('handles array.array.index.prop', function(done) { | ||
|  |         assert.deepEqual( | ||
|  |           [{ c: 48 }, 'woot'] | ||
|  |           , mpath.get('arr.arr.1.a', o, function(v) { | ||
|  |             if (undefined === v) return 'woot'; | ||
|  |             return v; | ||
|  |           }) | ||
|  |         ); | ||
|  |         assert.deepEqual( | ||
|  |           [{ c: 48 }, undefined] | ||
|  |           , mpath.get('arr.arr.1.a', o) | ||
|  |         ); | ||
|  |         mpath.set('arr.arr.1.a', [{ c: 49 }, undefined], o); | ||
|  |         assert.deepEqual( | ||
|  |           [{ c: 49 }, undefined] | ||
|  |           , mpath.get('arr.arr.1.a', o) | ||
|  |         ); | ||
|  |         mpath.set('arr.arr.1.a', [{ c: 48 }, undefined], o); | ||
|  |         done(); | ||
|  |       }); | ||
|  | 
 | ||
|  |       it('handles array.array.index.prop.prop', function(done) { | ||
|  |         assert.deepEqual( | ||
|  |           [48, undefined] | ||
|  |           , mpath.get('arr.arr.1.a.c', o) | ||
|  |         ); | ||
|  |         assert.deepEqual( | ||
|  |           [48, 'woot'] | ||
|  |           , mpath.get('arr.arr.1.a.c', o, function(v) { | ||
|  |             if (undefined === v) return 'woot'; | ||
|  |             return v; | ||
|  |           }) | ||
|  |         ); | ||
|  |         done(); | ||
|  |       }); | ||
|  | 
 | ||
|  |     }); | ||
|  | 
 | ||
|  |     describe('with `special`', function() { | ||
|  |       describe('that is a string', function() { | ||
|  |         it('works', function(done) { | ||
|  |           assert.equal('jiro', mpath.get('name', o, special)); | ||
|  | 
 | ||
|  |           assert.deepEqual( | ||
|  |             { second: { third: [3, { name: 'aaron' }, 9] } } | ||
|  |             , mpath.get('first', o, special) | ||
|  |           ); | ||
|  | 
 | ||
|  |           assert.deepEqual( | ||
|  |             { third: [3, { name: 'aaron' }, 9] } | ||
|  |             , mpath.get('first.second', o, special) | ||
|  |           ); | ||
|  | 
 | ||
|  |           assert.deepEqual( | ||
|  |             [3, { name: 'aaron' }, 9] | ||
|  |             , mpath.get('first.second.third', o, special) | ||
|  |           ); | ||
|  | 
 | ||
|  |           assert.deepEqual( | ||
|  |             3 | ||
|  |             , mpath.get('first.second.third.0', o, special) | ||
|  |           ); | ||
|  | 
 | ||
|  |           assert.deepEqual( | ||
|  |             4 | ||
|  |             , mpath.get('first.second.third.0', o, special, function(v) { | ||
|  |               return 3 === v ? 4 : v; | ||
|  |             }) | ||
|  |           ); | ||
|  | 
 | ||
|  |           assert.deepEqual( | ||
|  |             9 | ||
|  |             , mpath.get('first.second.third.2', o, special) | ||
|  |           ); | ||
|  | 
 | ||
|  |           assert.deepEqual( | ||
|  |             { name: 'aaron' } | ||
|  |             , mpath.get('first.second.third.1', o, special) | ||
|  |           ); | ||
|  | 
 | ||
|  |           assert.deepEqual( | ||
|  |             'aaron' | ||
|  |             , mpath.get('first.second.third.1.name', o, special) | ||
|  |           ); | ||
|  | 
 | ||
|  |           assert.deepEqual([ | ||
|  |             { name: 'one' }, | ||
|  |             { name: 'two', _doc: { name: '2' } }, | ||
|  |             { name: 'three', | ||
|  |               comments: [{}, { comments: [{ val: 'twoo' }] }], | ||
|  |               _doc: { name: '3', comments: [{}, { _doc: { comments: [{ val: 2 }] } }] } }], | ||
|  |           mpath.get('comments', o, special)); | ||
|  | 
 | ||
|  |           assert.deepEqual({ name: 'one' }, mpath.get('comments.0', o, special)); | ||
|  |           assert.deepEqual('one', mpath.get('comments.0.name', o, special)); | ||
|  |           assert.deepEqual('2', mpath.get('comments.1.name', o, special)); | ||
|  |           assert.deepEqual('3', mpath.get('comments.2.name', o, special)); | ||
|  |           assert.deepEqual('nice', mpath.get('comments.2.name', o, special, function(v) { | ||
|  |             return '3' === v ? 'nice' : v; | ||
|  |           })); | ||
|  | 
 | ||
|  |           assert.deepEqual([{}, { _doc: { comments: [{ val: 2 }] } }] | ||
|  |             , mpath.get('comments.2.comments', o, special)); | ||
|  | 
 | ||
|  |           assert.deepEqual({ _doc: { comments: [{ val: 2 }] } } | ||
|  |             , mpath.get('comments.2.comments.1', o, special)); | ||
|  | 
 | ||
|  |           assert.deepEqual(2, mpath.get('comments.2.comments.1.comments.0.val', o, special)); | ||
|  |           done(); | ||
|  |         }); | ||
|  | 
 | ||
|  |         it('handles array.property dot-notation', function(done) { | ||
|  |           assert.deepEqual( | ||
|  |             ['one', '2', '3'] | ||
|  |             , mpath.get('comments.name', o, special) | ||
|  |           ); | ||
|  |           assert.deepEqual( | ||
|  |             ['one', 2, '3'] | ||
|  |             , mpath.get('comments.name', o, special, function(v) { | ||
|  |               return '2' === v ? 2 : v; | ||
|  |             }) | ||
|  |           ); | ||
|  |           done(); | ||
|  |         }); | ||
|  | 
 | ||
|  |         it('handles array.array notation', function(done) { | ||
|  |           assert.deepEqual( | ||
|  |             [undefined, undefined, [{}, { _doc: { comments: [{ val: 2 }] } }]] | ||
|  |             , mpath.get('comments.comments', o, special) | ||
|  |           ); | ||
|  |           done(); | ||
|  |         }); | ||
|  | 
 | ||
|  |         it('handles array.array.index.array', function(done) { | ||
|  |           assert.deepEqual( | ||
|  |             [undefined, undefined, [{ val: 2 }]] | ||
|  |             , mpath.get('comments.comments.1.comments', o, special) | ||
|  |           ); | ||
|  |           done(); | ||
|  |         }); | ||
|  | 
 | ||
|  |         it('handles array.array.index.array.prop', function(done) { | ||
|  |           assert.deepEqual( | ||
|  |             [undefined, undefined, [2]] | ||
|  |             , mpath.get('comments.comments.1.comments.val', o, special) | ||
|  |           ); | ||
|  |           assert.deepEqual( | ||
|  |             ['nil', 'nil', [2]] | ||
|  |             , mpath.get('comments.comments.1.comments.val', o, special, function(v) { | ||
|  |               return undefined === v ? 'nil' : v; | ||
|  |             }) | ||
|  |           ); | ||
|  |           done(); | ||
|  |         }); | ||
|  |       }); | ||
|  | 
 | ||
|  |       describe('that is a function', function() { | ||
|  |         const special = function(obj, key) { | ||
|  |           return obj[key]; | ||
|  |         }; | ||
|  | 
 | ||
|  |         it('works', function(done) { | ||
|  |           assert.equal('jiro', mpath.get('name', o, special)); | ||
|  | 
 | ||
|  |           assert.deepEqual( | ||
|  |             { second: { third: [3, { name: 'aaron' }, 9] } } | ||
|  |             , mpath.get('first', o, special) | ||
|  |           ); | ||
|  | 
 | ||
|  |           assert.deepEqual( | ||
|  |             { third: [3, { name: 'aaron' }, 9] } | ||
|  |             , mpath.get('first.second', o, special) | ||
|  |           ); | ||
|  | 
 | ||
|  |           assert.deepEqual( | ||
|  |             [3, { name: 'aaron' }, 9] | ||
|  |             , mpath.get('first.second.third', o, special) | ||
|  |           ); | ||
|  | 
 | ||
|  |           assert.deepEqual( | ||
|  |             3 | ||
|  |             , mpath.get('first.second.third.0', o, special) | ||
|  |           ); | ||
|  | 
 | ||
|  |           assert.deepEqual( | ||
|  |             4 | ||
|  |             , mpath.get('first.second.third.0', o, special, function(v) { | ||
|  |               return 3 === v ? 4 : v; | ||
|  |             }) | ||
|  |           ); | ||
|  | 
 | ||
|  |           assert.deepEqual( | ||
|  |             9 | ||
|  |             , mpath.get('first.second.third.2', o, special) | ||
|  |           ); | ||
|  | 
 | ||
|  |           assert.deepEqual( | ||
|  |             { name: 'aaron' } | ||
|  |             , mpath.get('first.second.third.1', o, special) | ||
|  |           ); | ||
|  | 
 | ||
|  |           assert.deepEqual( | ||
|  |             'aaron' | ||
|  |             , mpath.get('first.second.third.1.name', o, special) | ||
|  |           ); | ||
|  | 
 | ||
|  |           assert.deepEqual([ | ||
|  |             { name: 'one' }, | ||
|  |             { name: 'two', _doc: { name: '2' } }, | ||
|  |             { name: 'three', | ||
|  |               comments: [{}, { comments: [{ val: 'twoo' }] }], | ||
|  |               _doc: { name: '3', comments: [{}, { _doc: { comments: [{ val: 2 }] } }] } }], | ||
|  |           mpath.get('comments', o, special)); | ||
|  | 
 | ||
|  |           assert.deepEqual({ name: 'one' }, mpath.get('comments.0', o, special)); | ||
|  |           assert.deepEqual('one', mpath.get('comments.0.name', o, special)); | ||
|  |           assert.deepEqual('two', mpath.get('comments.1.name', o, special)); | ||
|  |           assert.deepEqual('three', mpath.get('comments.2.name', o, special)); | ||
|  |           assert.deepEqual('nice', mpath.get('comments.2.name', o, special, function(v) { | ||
|  |             return 'three' === v ? 'nice' : v; | ||
|  |           })); | ||
|  | 
 | ||
|  |           assert.deepEqual([{}, { comments: [{ val: 'twoo' }] }] | ||
|  |             , mpath.get('comments.2.comments', o, special)); | ||
|  | 
 | ||
|  |           assert.deepEqual({ comments: [{ val: 'twoo' }] } | ||
|  |             , mpath.get('comments.2.comments.1', o, special)); | ||
|  | 
 | ||
|  |           assert.deepEqual('twoo', mpath.get('comments.2.comments.1.comments.0.val', o, special)); | ||
|  | 
 | ||
|  |           let overide = false; | ||
|  |           assert.deepEqual('twoo', mpath.get('comments.8.comments.1.comments.0.val', o, function(obj, path) { | ||
|  |             if (Array.isArray(obj) && 8 == path) { | ||
|  |               overide = true; | ||
|  |               return obj[2]; | ||
|  |             } | ||
|  |             return obj[path]; | ||
|  |           })); | ||
|  |           assert.ok(overide); | ||
|  | 
 | ||
|  |           done(); | ||
|  |         }); | ||
|  | 
 | ||
|  |         it('in combination with map', function(done) { | ||
|  |           const special = function(obj, key) { | ||
|  |             if (Array.isArray(obj)) return obj[key]; | ||
|  |             return obj.mpath; | ||
|  |           }; | ||
|  |           const map = function(val) { | ||
|  |             return 'convert' == val | ||
|  |               ? 'mpath' | ||
|  |               : val; | ||
|  |           }; | ||
|  |           const o = { mpath: [{ mpath: 'converse' }, { mpath: 'convert' }] }; | ||
|  | 
 | ||
|  |           assert.equal('mpath', mpath.get('something.1.kewl', o, special, map)); | ||
|  |           done(); | ||
|  |         }); | ||
|  |       }); | ||
|  |     }); | ||
|  |   }); | ||
|  | 
 | ||
|  |   describe('set', function() { | ||
|  |     it('prevents writing to __proto__', function() { | ||
|  |       const obj = {}; | ||
|  |       mpath.set('__proto__.x', 'foobar', obj); | ||
|  |       assert.ok(!({}.x)); | ||
|  | 
 | ||
|  |       mpath.set('constructor.prototype.x', 'foobar', obj); | ||
|  |       assert.ok(!({}.x)); | ||
|  |     }); | ||
|  | 
 | ||
|  |     describe('without `special`', function() { | ||
|  |       const o = doc(); | ||
|  | 
 | ||
|  |       it('works', function(done) { | ||
|  |         mpath.set('name', 'a new val', o, function(v) { | ||
|  |           return 'a new val' === v ? 'changed' : v; | ||
|  |         }); | ||
|  |         assert.deepEqual('changed', o.name); | ||
|  | 
 | ||
|  |         mpath.set('name', 'changed', o); | ||
|  |         assert.deepEqual('changed', o.name); | ||
|  | 
 | ||
|  |         mpath.set('first.second.third', [1, { name: 'x' }, 9], o); | ||
|  |         assert.deepEqual([1, { name: 'x' }, 9], o.first.second.third); | ||
|  | 
 | ||
|  |         mpath.set('first.second.third.1.name', 'y', o); | ||
|  |         assert.deepEqual([1, { name: 'y' }, 9], o.first.second.third); | ||
|  | 
 | ||
|  |         mpath.set('comments.1.name', 'ttwwoo', o); | ||
|  |         assert.deepEqual({ name: 'ttwwoo', _doc: { name: '2' } }, o.comments[1]); | ||
|  | 
 | ||
|  |         mpath.set('comments.2.comments.1.comments.0.expand', 'added', o); | ||
|  |         assert.deepEqual( | ||
|  |           { val: 'twoo', expand: 'added' } | ||
|  |           , o.comments[2].comments[1].comments[0]); | ||
|  | 
 | ||
|  |         mpath.set('comments.2.comments.1.comments.2', 'added', o); | ||
|  |         assert.equal(3, o.comments[2].comments[1].comments.length); | ||
|  |         assert.deepEqual( | ||
|  |           { val: 'twoo', expand: 'added' } | ||
|  |           , o.comments[2].comments[1].comments[0]); | ||
|  |         assert.deepEqual( | ||
|  |           undefined | ||
|  |           , o.comments[2].comments[1].comments[1]); | ||
|  |         assert.deepEqual( | ||
|  |           'added' | ||
|  |           , o.comments[2].comments[1].comments[2]); | ||
|  | 
 | ||
|  |         done(); | ||
|  |       }); | ||
|  | 
 | ||
|  |       describe('array.path', function() { | ||
|  |         describe('with single non-array value', function() { | ||
|  |           it('works', function(done) { | ||
|  |             mpath.set('arr.yep', false, o, function(v) { | ||
|  |               return false === v ? true : v; | ||
|  |             }); | ||
|  |             assert.deepEqual([ | ||
|  |               { yep: true, arr: [{ a: { b: 47 } }, { a: { c: 48 } }, { d: 'yep' }] }, | ||
|  |               { yep: true } | ||
|  |             ], o.arr); | ||
|  | 
 | ||
|  |             mpath.set('arr.yep', false, o); | ||
|  | 
 | ||
|  |             assert.deepEqual([ | ||
|  |               { yep: false, arr: [{ a: { b: 47 } }, { a: { c: 48 } }, { d: 'yep' }] }, | ||
|  |               { yep: false } | ||
|  |             ], o.arr); | ||
|  | 
 | ||
|  |             done(); | ||
|  |           }); | ||
|  |         }); | ||
|  |         describe('with array of values', function() { | ||
|  |           it('that are equal in length', function(done) { | ||
|  |             mpath.set('arr.yep', ['one', 2], o, function(v) { | ||
|  |               return 'one' === v ? 1 : v; | ||
|  |             }); | ||
|  |             assert.deepEqual([ | ||
|  |               { yep: 1, arr: [{ a: { b: 47 } }, { a: { c: 48 } }, { d: 'yep' }] }, | ||
|  |               { yep: 2 } | ||
|  |             ], o.arr); | ||
|  |             mpath.set('arr.yep', ['one', 2], o); | ||
|  | 
 | ||
|  |             assert.deepEqual([ | ||
|  |               { yep: 'one', arr: [{ a: { b: 47 } }, { a: { c: 48 } }, { d: 'yep' }] }, | ||
|  |               { yep: 2 } | ||
|  |             ], o.arr); | ||
|  | 
 | ||
|  |             done(); | ||
|  |           }); | ||
|  | 
 | ||
|  |           it('that is less than length', function(done) { | ||
|  |             mpath.set('arr.yep', [47], o, function(v) { | ||
|  |               return 47 === v ? 4 : v; | ||
|  |             }); | ||
|  |             assert.deepEqual([ | ||
|  |               { yep: 4, arr: [{ a: { b: 47 } }, { a: { c: 48 } }, { d: 'yep' }] }, | ||
|  |               { yep: 2 } | ||
|  |             ], o.arr); | ||
|  | 
 | ||
|  |             mpath.set('arr.yep', [47], o); | ||
|  |             assert.deepEqual([ | ||
|  |               { yep: 47, arr: [{ a: { b: 47 } }, { a: { c: 48 } }, { d: 'yep' }] }, | ||
|  |               { yep: 2 } | ||
|  |             ], o.arr); | ||
|  | 
 | ||
|  |             done(); | ||
|  |           }); | ||
|  | 
 | ||
|  |           it('that is greater than length', function(done) { | ||
|  |             mpath.set('arr.yep', [5, 6, 7], o, function(v) { | ||
|  |               return 5 === v ? 'five' : v; | ||
|  |             }); | ||
|  |             assert.deepEqual([ | ||
|  |               { yep: 'five', arr: [{ a: { b: 47 } }, { a: { c: 48 } }, { d: 'yep' }] }, | ||
|  |               { yep: 6 } | ||
|  |             ], o.arr); | ||
|  | 
 | ||
|  |             mpath.set('arr.yep', [5, 6, 7], o); | ||
|  |             assert.deepEqual([ | ||
|  |               { yep: 5, arr: [{ a: { b: 47 } }, { a: { c: 48 } }, { d: 'yep' }] }, | ||
|  |               { yep: 6 } | ||
|  |             ], o.arr); | ||
|  | 
 | ||
|  |             done(); | ||
|  |           }); | ||
|  |         }); | ||
|  |       }); | ||
|  | 
 | ||
|  |       describe('array.$.path', function() { | ||
|  |         describe('with single non-array value', function() { | ||
|  |           it('copies the value to each item in array', function(done) { | ||
|  |             mpath.set('arr.$.yep', { xtra: 'double good' }, o, function(v) { | ||
|  |               return v && v.xtra ? 'hi' : v; | ||
|  |             }); | ||
|  |             assert.deepEqual([ | ||
|  |               { yep: 'hi', arr: [{ a: { b: 47 } }, { a: { c: 48 } }, { d: 'yep' }] }, | ||
|  |               { yep: 'hi' } | ||
|  |             ], o.arr); | ||
|  | 
 | ||
|  |             mpath.set('arr.$.yep', { xtra: 'double good' }, o); | ||
|  |             assert.deepEqual([ | ||
|  |               { yep: { xtra: 'double good' }, arr: [{ a: { b: 47 } }, { a: { c: 48 } }, { d: 'yep' }] }, | ||
|  |               { yep: { xtra: 'double good' } } | ||
|  |             ], o.arr); | ||
|  | 
 | ||
|  |             done(); | ||
|  |           }); | ||
|  |         }); | ||
|  |         describe('with array of values', function() { | ||
|  |           it('copies the value to each item in array', function(done) { | ||
|  |             mpath.set('arr.$.yep', [15], o, function(v) { | ||
|  |               return v.length === 1 ? [] : v; | ||
|  |             }); | ||
|  |             assert.deepEqual([ | ||
|  |               { yep: [], arr: [{ a: { b: 47 } }, { a: { c: 48 } }, { d: 'yep' }] }, | ||
|  |               { yep: [] } | ||
|  |             ], o.arr); | ||
|  | 
 | ||
|  |             mpath.set('arr.$.yep', [15], o); | ||
|  |             assert.deepEqual([ | ||
|  |               { yep: [15], arr: [{ a: { b: 47 } }, { a: { c: 48 } }, { d: 'yep' }] }, | ||
|  |               { yep: [15] } | ||
|  |             ], o.arr); | ||
|  | 
 | ||
|  |             done(); | ||
|  |           }); | ||
|  |         }); | ||
|  |       }); | ||
|  | 
 | ||
|  |       describe('array.index.path', function() { | ||
|  |         it('works', function(done) { | ||
|  |           mpath.set('arr.1.yep', 0, o, function(v) { | ||
|  |             return 0 === v ? 'zero' : v; | ||
|  |           }); | ||
|  |           assert.deepEqual([ | ||
|  |             { yep: [15], arr: [{ a: { b: 47 } }, { a: { c: 48 } }, { d: 'yep' }] }, | ||
|  |             { yep: 'zero' } | ||
|  |           ], o.arr); | ||
|  | 
 | ||
|  |           mpath.set('arr.1.yep', 0, o); | ||
|  |           assert.deepEqual([ | ||
|  |             { yep: [15], arr: [{ a: { b: 47 } }, { a: { c: 48 } }, { d: 'yep' }] }, | ||
|  |             { yep: 0 } | ||
|  |           ], o.arr); | ||
|  | 
 | ||
|  |           done(); | ||
|  |         }); | ||
|  |       }); | ||
|  | 
 | ||
|  |       describe('array.index.array.path', function() { | ||
|  |         it('with single value', function(done) { | ||
|  |           mpath.set('arr.0.arr.e', 35, o, function(v) { | ||
|  |             return 35 === v ? 3 : v; | ||
|  |           }); | ||
|  |           assert.deepEqual([ | ||
|  |             { yep: [15], arr: [{ a: { b: 47 }, e: 3 }, { a: { c: 48 }, e: 3 }, { d: 'yep', e: 3 }] }, | ||
|  |             { yep: 0 } | ||
|  |           ], o.arr); | ||
|  | 
 | ||
|  |           mpath.set('arr.0.arr.e', 35, o); | ||
|  |           assert.deepEqual([ | ||
|  |             { yep: [15], arr: [{ a: { b: 47 }, e: 35 }, { a: { c: 48 }, e: 35 }, { d: 'yep', e: 35 }] }, | ||
|  |             { yep: 0 } | ||
|  |           ], o.arr); | ||
|  | 
 | ||
|  |           done(); | ||
|  |         }); | ||
|  |         it('with array', function(done) { | ||
|  |           mpath.set('arr.0.arr.e', ['a', 'b'], o, function(v) { | ||
|  |             return 'a' === v ? 'x' : v; | ||
|  |           }); | ||
|  |           assert.deepEqual([ | ||
|  |             { yep: [15], arr: [{ a: { b: 47 }, e: 'x' }, { a: { c: 48 }, e: 'b' }, { d: 'yep', e: 35 }] }, | ||
|  |             { yep: 0 } | ||
|  |           ], o.arr); | ||
|  | 
 | ||
|  |           mpath.set('arr.0.arr.e', ['a', 'b'], o); | ||
|  |           assert.deepEqual([ | ||
|  |             { yep: [15], arr: [{ a: { b: 47 }, e: 'a' }, { a: { c: 48 }, e: 'b' }, { d: 'yep', e: 35 }] }, | ||
|  |             { yep: 0 } | ||
|  |           ], o.arr); | ||
|  | 
 | ||
|  |           done(); | ||
|  |         }); | ||
|  |       }); | ||
|  | 
 | ||
|  |       describe('array.index.array.path.path', function() { | ||
|  |         it('with single value', function(done) { | ||
|  |           mpath.set('arr.0.arr.a.b', 36, o, function(v) { | ||
|  |             return 36 === v ? 3 : v; | ||
|  |           }); | ||
|  |           assert.deepEqual([ | ||
|  |             { yep: [15], arr: [{ a: { b: 3 }, e: 'a' }, { a: { c: 48, b: 3 }, e: 'b' }, { d: 'yep', e: 35 }] }, | ||
|  |             { yep: 0 } | ||
|  |           ], o.arr); | ||
|  | 
 | ||
|  |           mpath.set('arr.0.arr.a.b', 36, o); | ||
|  |           assert.deepEqual([ | ||
|  |             { yep: [15], arr: [{ a: { b: 36 }, e: 'a' }, { a: { c: 48, b: 36 }, e: 'b' }, { d: 'yep', e: 35 }] }, | ||
|  |             { yep: 0 } | ||
|  |           ], o.arr); | ||
|  | 
 | ||
|  |           done(); | ||
|  |         }); | ||
|  |         it('with array', function(done) { | ||
|  |           mpath.set('arr.0.arr.a.b', [1, 2, 3, 4], o, function(v) { | ||
|  |             return 2 === v ? 'two' : v; | ||
|  |           }); | ||
|  |           assert.deepEqual([ | ||
|  |             { yep: [15], arr: [{ a: { b: 1 }, e: 'a' }, { a: { c: 48, b: 'two' }, e: 'b' }, { d: 'yep', e: 35 }] }, | ||
|  |             { yep: 0 } | ||
|  |           ], o.arr); | ||
|  | 
 | ||
|  |           mpath.set('arr.0.arr.a.b', [1, 2, 3, 4], o); | ||
|  |           assert.deepEqual([ | ||
|  |             { yep: [15], arr: [{ a: { b: 1 }, e: 'a' }, { a: { c: 48, b: 2 }, e: 'b' }, { d: 'yep', e: 35 }] }, | ||
|  |             { yep: 0 } | ||
|  |           ], o.arr); | ||
|  | 
 | ||
|  |           done(); | ||
|  |         }); | ||
|  |       }); | ||
|  | 
 | ||
|  |       describe('array.index.array.$.path.path', function() { | ||
|  |         it('with single value', function(done) { | ||
|  |           mpath.set('arr.0.arr.$.a.b', '$', o, function(v) { | ||
|  |             return '$' === v ? 'dolla billz' : v; | ||
|  |           }); | ||
|  |           assert.deepEqual([ | ||
|  |             { yep: [15], arr: [{ a: { b: 'dolla billz' }, e: 'a' }, { a: { c: 48, b: 'dolla billz' }, e: 'b' }, { d: 'yep', e: 35 }] }, | ||
|  |             { yep: 0 } | ||
|  |           ], o.arr); | ||
|  | 
 | ||
|  |           mpath.set('arr.0.arr.$.a.b', '$', o); | ||
|  |           assert.deepEqual([ | ||
|  |             { yep: [15], arr: [{ a: { b: '$' }, e: 'a' }, { a: { c: 48, b: '$' }, e: 'b' }, { d: 'yep', e: 35 }] }, | ||
|  |             { yep: 0 } | ||
|  |           ], o.arr); | ||
|  | 
 | ||
|  |           done(); | ||
|  |         }); | ||
|  |         it('with array', function(done) { | ||
|  |           mpath.set('arr.0.arr.$.a.b', [1], o, function(v) { | ||
|  |             return Array.isArray(v) ? {} : v; | ||
|  |           }); | ||
|  |           assert.deepEqual([ | ||
|  |             { yep: [15], arr: [{ a: { b: {} }, e: 'a' }, { a: { c: 48, b: {} }, e: 'b' }, { d: 'yep', e: 35 }] }, | ||
|  |             { yep: 0 } | ||
|  |           ], o.arr); | ||
|  | 
 | ||
|  |           mpath.set('arr.0.arr.$.a.b', [1], o); | ||
|  |           assert.deepEqual([ | ||
|  |             { yep: [15], arr: [{ a: { b: [1] }, e: 'a' }, { a: { c: 48, b: [1] }, e: 'b' }, { d: 'yep', e: 35 }] }, | ||
|  |             { yep: 0 } | ||
|  |           ], o.arr); | ||
|  | 
 | ||
|  |           done(); | ||
|  |         }); | ||
|  |       }); | ||
|  | 
 | ||
|  |       describe('array.array.index.path', function() { | ||
|  |         it('with single value', function(done) { | ||
|  |           mpath.set('arr.arr.0.a', 'single', o, function(v) { | ||
|  |             return 'single' === v ? 'double' : v; | ||
|  |           }); | ||
|  |           assert.deepEqual([ | ||
|  |             { yep: [15], arr: [{ a: 'double', e: 'a' }, { a: { c: 48, b: [1] }, e: 'b' }, { d: 'yep', e: 35 }] }, | ||
|  |             { yep: 0 } | ||
|  |           ], o.arr); | ||
|  | 
 | ||
|  |           mpath.set('arr.arr.0.a', 'single', o); | ||
|  |           assert.deepEqual([ | ||
|  |             { yep: [15], arr: [{ a: 'single', e: 'a' }, { a: { c: 48, b: [1] }, e: 'b' }, { d: 'yep', e: 35 }] }, | ||
|  |             { yep: 0 } | ||
|  |           ], o.arr); | ||
|  | 
 | ||
|  |           done(); | ||
|  |         }); | ||
|  |         it('with array', function(done) { | ||
|  |           mpath.set('arr.arr.0.a', [4, 8, 15, 16, 23, 42], o, function(v) { | ||
|  |             return 4 === v ? 3 : v; | ||
|  |           }); | ||
|  |           assert.deepEqual([ | ||
|  |             { yep: [15], arr: [{ a: 3, e: 'a' }, { a: { c: 48, b: [1] }, e: 'b' }, { d: 'yep', e: 35 }] }, | ||
|  |             { yep: false } | ||
|  |           ], o.arr); | ||
|  | 
 | ||
|  |           mpath.set('arr.arr.0.a', [4, 8, 15, 16, 23, 42], o); | ||
|  |           assert.deepEqual([ | ||
|  |             { yep: [15], arr: [{ a: 4, e: 'a' }, { a: { c: 48, b: [1] }, e: 'b' }, { d: 'yep', e: 35 }] }, | ||
|  |             { yep: false } | ||
|  |           ], o.arr); | ||
|  | 
 | ||
|  |           done(); | ||
|  |         }); | ||
|  |       }); | ||
|  | 
 | ||
|  |       describe('array.array.$.index.path', function() { | ||
|  |         it('with single value', function(done) { | ||
|  |           mpath.set('arr.arr.$.0.a', 'singles', o, function(v) { | ||
|  |             return 0; | ||
|  |           }); | ||
|  |           assert.deepEqual([ | ||
|  |             { yep: [15], arr: [{ a: 0, e: 'a' }, { a: { c: 48, b: [1] }, e: 'b' }, { d: 'yep', e: 35 }] }, | ||
|  |             { yep: 0 } | ||
|  |           ], o.arr); | ||
|  | 
 | ||
|  |           mpath.set('arr.arr.$.0.a', 'singles', o); | ||
|  |           assert.deepEqual([ | ||
|  |             { yep: [15], arr: [{ a: 'singles', e: 'a' }, { a: { c: 48, b: [1] }, e: 'b' }, { d: 'yep', e: 35 }] }, | ||
|  |             { yep: 0 } | ||
|  |           ], o.arr); | ||
|  | 
 | ||
|  |           mpath.set('$.arr.arr.0.a', 'single', o); | ||
|  |           assert.deepEqual([ | ||
|  |             { yep: [15], arr: [{ a: 'single', e: 'a' }, { a: { c: 48, b: [1] }, e: 'b' }, { d: 'yep', e: 35 }] }, | ||
|  |             { yep: 0 } | ||
|  |           ], o.arr); | ||
|  | 
 | ||
|  |           done(); | ||
|  |         }); | ||
|  |         it('with array', function(done) { | ||
|  |           mpath.set('arr.arr.$.0.a', [4, 8, 15, 16, 23, 42], o, function(v) { | ||
|  |             return 'nope'; | ||
|  |           }); | ||
|  |           assert.deepEqual([ | ||
|  |             { yep: [15], arr: [{ a: 'nope', e: 'a' }, { a: { c: 48, b: [1] }, e: 'b' }, { d: 'yep', e: 35 }] }, | ||
|  |             { yep: 0 } | ||
|  |           ], o.arr); | ||
|  | 
 | ||
|  |           mpath.set('arr.arr.$.0.a', [4, 8, 15, 16, 23, 42], o); | ||
|  |           assert.deepEqual([ | ||
|  |             { yep: [15], arr: [{ a: [4, 8, 15, 16, 23, 42], e: 'a' }, { a: { c: 48, b: [1] }, e: 'b' }, { d: 'yep', e: 35 }] }, | ||
|  |             { yep: 0 } | ||
|  |           ], o.arr); | ||
|  | 
 | ||
|  |           mpath.set('arr.$.arr.0.a', [4, 8, 15, 16, 23, 42, 108], o); | ||
|  |           assert.deepEqual([ | ||
|  |             { yep: [15], arr: [{ a: [4, 8, 15, 16, 23, 42, 108], e: 'a' }, { a: { c: 48, b: [1] }, e: 'b' }, { d: 'yep', e: 35 }] }, | ||
|  |             { yep: 0 } | ||
|  |           ], o.arr); | ||
|  | 
 | ||
|  |           done(); | ||
|  |         }); | ||
|  |       }); | ||
|  | 
 | ||
|  |       describe('array.array.path.index', function() { | ||
|  |         it('with single value', function(done) { | ||
|  |           mpath.set('arr.arr.a.7', 47, o, function(v) { | ||
|  |             return 1; | ||
|  |           }); | ||
|  |           assert.deepEqual([ | ||
|  |             { yep: [15], arr: [{ a: [4, 8, 15, 16, 23, 42, 108, 1], e: 'a' }, { a: { c: 48, b: [1], 7: 1 }, e: 'b' }, { d: 'yep', e: 35 }] }, | ||
|  |             { yep: 0 } | ||
|  |           ], o.arr); | ||
|  | 
 | ||
|  |           mpath.set('arr.arr.a.7', 47, o); | ||
|  |           assert.deepEqual([ | ||
|  |             { yep: [15], arr: [{ a: [4, 8, 15, 16, 23, 42, 108, 47], e: 'a' }, { a: { c: 48, b: [1], 7: 47 }, e: 'b' }, { d: 'yep', e: 35 }] }, | ||
|  |             { yep: 0 } | ||
|  |           ], o.arr); | ||
|  | 
 | ||
|  |           done(); | ||
|  |         }); | ||
|  |         it('with array', function(done) { | ||
|  |           o.arr[1].arr = [{ a: [] }, { a: [] }, { a: null }]; | ||
|  |           mpath.set('arr.arr.a.7', [[null, 46], [undefined, 'woot']], o); | ||
|  | 
 | ||
|  |           const a1 = []; | ||
|  |           const a2 = []; | ||
|  |           a1[7] = undefined; | ||
|  |           a2[7] = 'woot'; | ||
|  | 
 | ||
|  |           assert.deepEqual([ | ||
|  |             { yep: [15], arr: [{ a: [4, 8, 15, 16, 23, 42, 108, null], e: 'a' }, { a: { c: 48, b: [1], 7: 46 }, e: 'b' }, { d: 'yep', e: 35 }] }, | ||
|  |             { yep: 0, arr: [{ a: a1 }, { a: a2 }, { a: null }] } | ||
|  |           ], o.arr); | ||
|  | 
 | ||
|  |           done(); | ||
|  |         }); | ||
|  |       }); | ||
|  | 
 | ||
|  |       describe('handles array.array.path', function() { | ||
|  |         it('with single', function(done) { | ||
|  |           o.arr[1].arr = [{}, {}]; | ||
|  |           assert.deepEqual([{}, {}], o.arr[1].arr); | ||
|  |           o.arr.push({ arr: 'something else' }); | ||
|  |           o.arr.push({ arr: ['something else'] }); | ||
|  |           o.arr.push({ arr: [[]] }); | ||
|  |           o.arr.push({ arr: [5] }); | ||
|  | 
 | ||
|  |           const weird = []; | ||
|  |           weird.e = 'xmas'; | ||
|  | 
 | ||
|  |           // test
 | ||
|  |           mpath.set('arr.arr.e', 47, o, function(v) { | ||
|  |             return 'xmas'; | ||
|  |           }); | ||
|  |           assert.deepEqual([ | ||
|  |             { yep: [15], arr: [ | ||
|  |               { a: [4, 8, 15, 16, 23, 42, 108, null], e: 'xmas' }, | ||
|  |               { a: { c: 48, b: [1], 7: 46 }, e: 'xmas' }, | ||
|  |               { d: 'yep', e: 'xmas' } | ||
|  |             ] | ||
|  |             }, | ||
|  |             { yep: 0, arr: [{ e: 'xmas' }, { e: 'xmas' }] }, | ||
|  |             { arr: 'something else' }, | ||
|  |             { arr: ['something else'] }, | ||
|  |             { arr: [weird] }, | ||
|  |             { arr: [5] } | ||
|  |           ] | ||
|  |           , o.arr); | ||
|  | 
 | ||
|  |           weird.e = 47; | ||
|  | 
 | ||
|  |           mpath.set('arr.arr.e', 47, o); | ||
|  |           assert.deepEqual([ | ||
|  |             { yep: [15], arr: [ | ||
|  |               { a: [4, 8, 15, 16, 23, 42, 108, null], e: 47 }, | ||
|  |               { a: { c: 48, b: [1], 7: 46 }, e: 47 }, | ||
|  |               { d: 'yep', e: 47 } | ||
|  |             ] | ||
|  |             }, | ||
|  |             { yep: 0, arr: [{ e: 47 }, { e: 47 }] }, | ||
|  |             { arr: 'something else' }, | ||
|  |             { arr: ['something else'] }, | ||
|  |             { arr: [weird] }, | ||
|  |             { arr: [5] } | ||
|  |           ] | ||
|  |           , o.arr); | ||
|  | 
 | ||
|  |           done(); | ||
|  |         }); | ||
|  |         it('with arrays', function(done) { | ||
|  |           mpath.set('arr.arr.e', [[1, 2, 3], [4, 5], null, [], [6], [7, 8, 9]], o, function(v) { | ||
|  |             return 10; | ||
|  |           }); | ||
|  | 
 | ||
|  |           const weird = []; | ||
|  |           weird.e = 10; | ||
|  | 
 | ||
|  |           assert.deepEqual([ | ||
|  |             { yep: [15], arr: [ | ||
|  |               { a: [4, 8, 15, 16, 23, 42, 108, null], e: 10 }, | ||
|  |               { a: { c: 48, b: [1], 7: 46 }, e: 10 }, | ||
|  |               { d: 'yep', e: 10 } | ||
|  |             ] | ||
|  |             }, | ||
|  |             { yep: 0, arr: [{ e: 10 }, { e: 10 }] }, | ||
|  |             { arr: 'something else' }, | ||
|  |             { arr: ['something else'] }, | ||
|  |             { arr: [weird] }, | ||
|  |             { arr: [5] } | ||
|  |           ] | ||
|  |           , o.arr); | ||
|  | 
 | ||
|  |           mpath.set('arr.arr.e', [[1, 2, 3], [4, 5], null, [], [6], [7, 8, 9]], o); | ||
|  | 
 | ||
|  |           weird.e = 6; | ||
|  | 
 | ||
|  |           assert.deepEqual([ | ||
|  |             { yep: [15], arr: [ | ||
|  |               { a: [4, 8, 15, 16, 23, 42, 108, null], e: 1 }, | ||
|  |               { a: { c: 48, b: [1], 7: 46 }, e: 2 }, | ||
|  |               { d: 'yep', e: 3 } | ||
|  |             ] | ||
|  |             }, | ||
|  |             { yep: 0, arr: [{ e: 4 }, { e: 5 }] }, | ||
|  |             { arr: 'something else' }, | ||
|  |             { arr: ['something else'] }, | ||
|  |             { arr: [weird] }, | ||
|  |             { arr: [5] } | ||
|  |           ] | ||
|  |           , o.arr); | ||
|  | 
 | ||
|  |           done(); | ||
|  |         }); | ||
|  |       }); | ||
|  |     }); | ||
|  | 
 | ||
|  |     describe('with `special`', function() { | ||
|  |       const o = doc(); | ||
|  | 
 | ||
|  |       it('works', function(done) { | ||
|  |         mpath.set('name', 'chan', o, special, function(v) { | ||
|  |           return 'hi'; | ||
|  |         }); | ||
|  |         assert.deepEqual('hi', o.name); | ||
|  | 
 | ||
|  |         mpath.set('name', 'changer', o, special); | ||
|  |         assert.deepEqual('changer', o.name); | ||
|  | 
 | ||
|  |         mpath.set('first.second.third', [1, { name: 'y' }, 9], o, special); | ||
|  |         assert.deepEqual([1, { name: 'y' }, 9], o.first.second.third); | ||
|  | 
 | ||
|  |         mpath.set('first.second.third.1.name', 'z', o, special); | ||
|  |         assert.deepEqual([1, { name: 'z' }, 9], o.first.second.third); | ||
|  | 
 | ||
|  |         mpath.set('comments.1.name', 'ttwwoo', o, special); | ||
|  |         assert.deepEqual({ name: 'two', _doc: { name: 'ttwwoo' } }, o.comments[1]); | ||
|  | 
 | ||
|  |         mpath.set('comments.2.comments.1.comments.0.expander', 'adder', o, special, function(v) { | ||
|  |           return 'super'; | ||
|  |         }); | ||
|  |         assert.deepEqual( | ||
|  |           { val: 2, expander: 'super' } | ||
|  |           , o.comments[2]._doc.comments[1]._doc.comments[0]); | ||
|  | 
 | ||
|  |         mpath.set('comments.2.comments.1.comments.0.expander', 'adder', o, special); | ||
|  |         assert.deepEqual( | ||
|  |           { val: 2, expander: 'adder' } | ||
|  |           , o.comments[2]._doc.comments[1]._doc.comments[0]); | ||
|  | 
 | ||
|  |         mpath.set('comments.2.comments.1.comments.2', 'set', o, special); | ||
|  |         assert.equal(3, o.comments[2]._doc.comments[1]._doc.comments.length); | ||
|  |         assert.deepEqual( | ||
|  |           { val: 2, expander: 'adder' } | ||
|  |           , o.comments[2]._doc.comments[1]._doc.comments[0]); | ||
|  |         assert.deepEqual( | ||
|  |           undefined | ||
|  |           , o.comments[2]._doc.comments[1]._doc.comments[1]); | ||
|  |         assert.deepEqual( | ||
|  |           'set' | ||
|  |           , o.comments[2]._doc.comments[1]._doc.comments[2]); | ||
|  |         done(); | ||
|  |       }); | ||
|  | 
 | ||
|  |       describe('array.path', function() { | ||
|  |         describe('with single non-array value', function() { | ||
|  |           it('works', function(done) { | ||
|  |             o.arr[1]._doc = { special: true }; | ||
|  | 
 | ||
|  |             mpath.set('arr.yep', false, o, special, function(v) { | ||
|  |               return 'yes'; | ||
|  |             }); | ||
|  |             assert.deepEqual([ | ||
|  |               { yep: 'yes', arr: [{ a: { b: 47 } }, { a: { c: 48 } }, { d: 'yep' }] }, | ||
|  |               { yep: true, _doc: { special: true, yep: 'yes' } } | ||
|  |             ], o.arr); | ||
|  | 
 | ||
|  |             mpath.set('arr.yep', false, o, special); | ||
|  |             assert.deepEqual([ | ||
|  |               { yep: false, arr: [{ a: { b: 47 } }, { a: { c: 48 } }, { d: 'yep' }] }, | ||
|  |               { yep: true, _doc: { special: true, yep: false } } | ||
|  |             ], o.arr); | ||
|  | 
 | ||
|  |             done(); | ||
|  |           }); | ||
|  |         }); | ||
|  |         describe('with array of values', function() { | ||
|  |           it('that are equal in length', function(done) { | ||
|  |             mpath.set('arr.yep', ['one', 2], o, special, function(v) { | ||
|  |               return 2 === v ? 20 : v; | ||
|  |             }); | ||
|  |             assert.deepEqual([ | ||
|  |               { yep: 'one', arr: [{ a: { b: 47 } }, { a: { c: 48 } }, { d: 'yep' }] }, | ||
|  |               { yep: true, _doc: { special: true, yep: 20 } } | ||
|  |             ], o.arr); | ||
|  | 
 | ||
|  |             mpath.set('arr.yep', ['one', 2], o, special); | ||
|  |             assert.deepEqual([ | ||
|  |               { yep: 'one', arr: [{ a: { b: 47 } }, { a: { c: 48 } }, { d: 'yep' }] }, | ||
|  |               { yep: true, _doc: { special: true, yep: 2 } } | ||
|  |             ], o.arr); | ||
|  | 
 | ||
|  |             done(); | ||
|  |           }); | ||
|  | 
 | ||
|  |           it('that is less than length', function(done) { | ||
|  |             mpath.set('arr.yep', [47], o, special, function(v) { | ||
|  |               return 80; | ||
|  |             }); | ||
|  |             assert.deepEqual([ | ||
|  |               { yep: 80, arr: [{ a: { b: 47 } }, { a: { c: 48 } }, { d: 'yep' }] }, | ||
|  |               { yep: true, _doc: { special: true, yep: 2 } } | ||
|  |             ], o.arr); | ||
|  | 
 | ||
|  |             mpath.set('arr.yep', [47], o, special); | ||
|  |             assert.deepEqual([ | ||
|  |               { yep: 47, arr: [{ a: { b: 47 } }, { a: { c: 48 } }, { d: 'yep' }] }, | ||
|  |               { yep: true, _doc: { special: true, yep: 2 } } | ||
|  |             ], o.arr); | ||
|  | 
 | ||
|  |             // add _doc to first element
 | ||
|  |             o.arr[0]._doc = { yep: 46, arr: [{ a: { b: 47 } }, { a: { c: 48 } }, { d: 'yep' }] }; | ||
|  | 
 | ||
|  |             mpath.set('arr.yep', [20], o, special); | ||
|  |             assert.deepEqual([ | ||
|  |               { yep: 47, arr: [{ a: { b: 47 } }, { a: { c: 48 } }, { d: 'yep' }], _doc: { yep: 20, arr: [{ a: { b: 47 } }, { a: { c: 48 } }, { d: 'yep' }] } }, | ||
|  |               { yep: true, _doc: { special: true, yep: 2 } } | ||
|  |             ], o.arr); | ||
|  | 
 | ||
|  |             done(); | ||
|  |           }); | ||
|  | 
 | ||
|  |           it('that is greater than length', function(done) { | ||
|  |             mpath.set('arr.yep', [5, 6, 7], o, special, function() { | ||
|  |               return 'x'; | ||
|  |             }); | ||
|  |             assert.deepEqual([ | ||
|  |               { yep: 47, arr: [{ a: { b: 47 } }, { a: { c: 48 } }, { d: 'yep' }], _doc: { yep: 'x', arr: [{ a: { b: 47 } }, { a: { c: 48 } }, { d: 'yep' }] } }, | ||
|  |               { yep: true, _doc: { special: true, yep: 'x' } } | ||
|  |             ], o.arr); | ||
|  | 
 | ||
|  |             mpath.set('arr.yep', [5, 6, 7], o, special); | ||
|  |             assert.deepEqual([ | ||
|  |               { yep: 47, arr: [{ a: { b: 47 } }, { a: { c: 48 } }, { d: 'yep' }], _doc: { yep: 5, arr: [{ a: { b: 47 } }, { a: { c: 48 } }, { d: 'yep' }] } }, | ||
|  |               { yep: true, _doc: { special: true, yep: 6 } } | ||
|  |             ], o.arr); | ||
|  | 
 | ||
|  |             done(); | ||
|  |           }); | ||
|  |         }); | ||
|  |       }); | ||
|  | 
 | ||
|  |       describe('array.$.path', function() { | ||
|  |         describe('with single non-array value', function() { | ||
|  |           it('copies the value to each item in array', function(done) { | ||
|  |             mpath.set('arr.$.yep', { xtra: 'double good' }, o, special, function(v) { | ||
|  |               return 9; | ||
|  |             }); | ||
|  |             assert.deepEqual([ | ||
|  |               { yep: 47, arr: [{ a: { b: 47 } }, { a: { c: 48 } }, { d: 'yep' }], | ||
|  |                 _doc: { yep: 9, arr: [{ a: { b: 47 } }, { a: { c: 48 } }, { d: 'yep' }] } }, | ||
|  |               { yep: true, _doc: { special: true, yep: 9 } } | ||
|  |             ], o.arr); | ||
|  | 
 | ||
|  |             mpath.set('arr.$.yep', { xtra: 'double good' }, o, special); | ||
|  |             assert.deepEqual([ | ||
|  |               { yep: 47, arr: [{ a: { b: 47 } }, { a: { c: 48 } }, { d: 'yep' }], | ||
|  |                 _doc: { yep: { xtra: 'double good' }, arr: [{ a: { b: 47 } }, { a: { c: 48 } }, { d: 'yep' }] } }, | ||
|  |               { yep: true, _doc: { special: true, yep: { xtra: 'double good' } } } | ||
|  |             ], o.arr); | ||
|  | 
 | ||
|  |             done(); | ||
|  |           }); | ||
|  |         }); | ||
|  |         describe('with array of values', function() { | ||
|  |           it('copies the value to each item in array', function(done) { | ||
|  |             mpath.set('arr.$.yep', [15], o, special, function(v) { | ||
|  |               return 'array'; | ||
|  |             }); | ||
|  |             assert.deepEqual([ | ||
|  |               { yep: 47, arr: [{ a: { b: 47 } }, { a: { c: 48 } }, { d: 'yep' }], | ||
|  |                 _doc: { yep: 'array', arr: [{ a: { b: 47 } }, { a: { c: 48 } }, { d: 'yep' }] } }, | ||
|  |               { yep: true, _doc: { special: true, yep: 'array' } } | ||
|  |             ], o.arr); | ||
|  | 
 | ||
|  |             mpath.set('arr.$.yep', [15], o, special); | ||
|  |             assert.deepEqual([ | ||
|  |               { yep: 47, arr: [{ a: { b: 47 } }, { a: { c: 48 } }, { d: 'yep' }], | ||
|  |                 _doc: { yep: [15], arr: [{ a: { b: 47 } }, { a: { c: 48 } }, { d: 'yep' }] } }, | ||
|  |               { yep: true, _doc: { special: true, yep: [15] } } | ||
|  |             ], o.arr); | ||
|  | 
 | ||
|  |             done(); | ||
|  |           }); | ||
|  |         }); | ||
|  |       }); | ||
|  | 
 | ||
|  |       describe('array.index.path', function() { | ||
|  |         it('works', function(done) { | ||
|  |           mpath.set('arr.1.yep', 0, o, special, function(v) { | ||
|  |             return 1; | ||
|  |           }); | ||
|  |           assert.deepEqual([ | ||
|  |             { yep: 47, arr: [{ a: { b: 47 } }, { a: { c: 48 } }, { d: 'yep' }], | ||
|  |               _doc: { yep: [15], arr: [{ a: { b: 47 } }, { a: { c: 48 } }, { d: 'yep' }] } }, | ||
|  |             { yep: true, _doc: { special: true, yep: 1 } } | ||
|  |           ], o.arr); | ||
|  | 
 | ||
|  |           mpath.set('arr.1.yep', 0, o, special); | ||
|  |           assert.deepEqual([ | ||
|  |             { yep: 47, arr: [{ a: { b: 47 } }, { a: { c: 48 } }, { d: 'yep' }], | ||
|  |               _doc: { yep: [15], arr: [{ a: { b: 47 } }, { a: { c: 48 } }, { d: 'yep' }] } }, | ||
|  |             { yep: true, _doc: { special: true, yep: 0 } } | ||
|  |           ], o.arr); | ||
|  | 
 | ||
|  |           done(); | ||
|  |         }); | ||
|  |       }); | ||
|  | 
 | ||
|  |       describe('array.index.array.path', function() { | ||
|  |         it('with single value', function(done) { | ||
|  |           mpath.set('arr.0.arr.e', 35, o, special, function(v) { | ||
|  |             return 30; | ||
|  |           }); | ||
|  |           assert.deepEqual([ | ||
|  |             { yep: 47, arr: [{ a: { b: 47 } }, { a: { c: 48 } }, { d: 'yep' }], | ||
|  |               _doc: { yep: [15], arr: [{ a: { b: 47 }, e: 30 }, { a: { c: 48 }, e: 30 }, { d: 'yep', e: 30 }] } }, | ||
|  |             { yep: true, _doc: { special: true, yep: 0 } } | ||
|  |           ], o.arr); | ||
|  | 
 | ||
|  |           mpath.set('arr.0.arr.e', 35, o, special); | ||
|  |           assert.deepEqual([ | ||
|  |             { yep: 47, arr: [{ a: { b: 47 } }, { a: { c: 48 } }, { d: 'yep' }], | ||
|  |               _doc: { yep: [15], arr: [{ a: { b: 47 }, e: 35 }, { a: { c: 48 }, e: 35 }, { d: 'yep', e: 35 }] } }, | ||
|  |             { yep: true, _doc: { special: true, yep: 0 } } | ||
|  |           ], o.arr); | ||
|  | 
 | ||
|  |           done(); | ||
|  |         }); | ||
|  |         it('with array', function(done) { | ||
|  |           mpath.set('arr.0.arr.e', ['a', 'b'], o, special, function(v) { | ||
|  |             return 'a' === v ? 'A' : v; | ||
|  |           }); | ||
|  |           assert.deepEqual([ | ||
|  |             { yep: 47, arr: [{ a: { b: 47 } }, { a: { c: 48 } }, { d: 'yep' }], | ||
|  |               _doc: { yep: [15], arr: [{ a: { b: 47 }, e: 'A' }, { a: { c: 48 }, e: 'b' }, { d: 'yep', e: 35 }] } }, | ||
|  |             { yep: true, _doc: { special: true, yep: 0 } } | ||
|  |           ], o.arr); | ||
|  | 
 | ||
|  |           mpath.set('arr.0.arr.e', ['a', 'b'], o, special); | ||
|  |           assert.deepEqual([ | ||
|  |             { yep: 47, arr: [{ a: { b: 47 } }, { a: { c: 48 } }, { d: 'yep' }], | ||
|  |               _doc: { yep: [15], arr: [{ a: { b: 47 }, e: 'a' }, { a: { c: 48 }, e: 'b' }, { d: 'yep', e: 35 }] } }, | ||
|  |             { yep: true, _doc: { special: true, yep: 0 } } | ||
|  |           ], o.arr); | ||
|  | 
 | ||
|  |           done(); | ||
|  |         }); | ||
|  |       }); | ||
|  | 
 | ||
|  |       describe('array.index.array.path.path', function() { | ||
|  |         it('with single value', function(done) { | ||
|  |           mpath.set('arr.0.arr.a.b', 36, o, special, function(v) { | ||
|  |             return 20; | ||
|  |           }); | ||
|  |           assert.deepEqual([ | ||
|  |             { yep: 47, arr: [{ a: { b: 47 } }, { a: { c: 48 } }, { d: 'yep' }], | ||
|  |               _doc: { yep: [15], arr: [{ a: { b: 20 }, e: 'a' }, { a: { c: 48, b: 20 }, e: 'b' }, { d: 'yep', e: 35 }] } }, | ||
|  |             { yep: true, _doc: { special: true, yep: 0 } } | ||
|  |           ], o.arr); | ||
|  | 
 | ||
|  |           mpath.set('arr.0.arr.a.b', 36, o, special); | ||
|  |           assert.deepEqual([ | ||
|  |             { yep: 47, arr: [{ a: { b: 47 } }, { a: { c: 48 } }, { d: 'yep' }], | ||
|  |               _doc: { yep: [15], arr: [{ a: { b: 36 }, e: 'a' }, { a: { c: 48, b: 36 }, e: 'b' }, { d: 'yep', e: 35 }] } }, | ||
|  |             { yep: true, _doc: { special: true, yep: 0 } } | ||
|  |           ], o.arr); | ||
|  | 
 | ||
|  |           done(); | ||
|  |         }); | ||
|  |         it('with array', function(done) { | ||
|  |           mpath.set('arr.0.arr.a.b', [1, 2, 3, 4], o, special, function(v) { | ||
|  |             return v * 2; | ||
|  |           }); | ||
|  |           assert.deepEqual([ | ||
|  |             { yep: 47, arr: [{ a: { b: 47 } }, { a: { c: 48 } }, { d: 'yep' }], | ||
|  |               _doc: { yep: [15], arr: [{ a: { b: 2 }, e: 'a' }, { a: { c: 48, b: 4 }, e: 'b' }, { d: 'yep', e: 35 }] } }, | ||
|  |             { yep: true, _doc: { special: true, yep: 0 } } | ||
|  |           ], o.arr); | ||
|  | 
 | ||
|  |           mpath.set('arr.0.arr.a.b', [1, 2, 3, 4], o, special); | ||
|  |           assert.deepEqual([ | ||
|  |             { yep: 47, arr: [{ a: { b: 47 } }, { a: { c: 48 } }, { d: 'yep' }], | ||
|  |               _doc: { yep: [15], arr: [{ a: { b: 1 }, e: 'a' }, { a: { c: 48, b: 2 }, e: 'b' }, { d: 'yep', e: 35 }] } }, | ||
|  |             { yep: true, _doc: { special: true, yep: 0 } } | ||
|  |           ], o.arr); | ||
|  | 
 | ||
|  |           done(); | ||
|  |         }); | ||
|  |       }); | ||
|  | 
 | ||
|  |       describe('array.index.array.$.path.path', function() { | ||
|  |         it('with single value', function(done) { | ||
|  |           mpath.set('arr.0.arr.$.a.b', '$', o, special, function(v) { | ||
|  |             return 'dollaz'; | ||
|  |           }); | ||
|  |           assert.deepEqual([ | ||
|  |             { yep: 47, arr: [{ a: { b: 47 } }, { a: { c: 48 } }, { d: 'yep' }], | ||
|  |               _doc: { yep: [15], arr: [{ a: { b: 'dollaz' }, e: 'a' }, { a: { c: 48, b: 'dollaz' }, e: 'b' }, { d: 'yep', e: 35 }] } }, | ||
|  |             { yep: true, _doc: { special: true, yep: 0 } } | ||
|  |           ], o.arr); | ||
|  | 
 | ||
|  |           mpath.set('arr.0.arr.$.a.b', '$', o, special); | ||
|  |           assert.deepEqual([ | ||
|  |             { yep: 47, arr: [{ a: { b: 47 } }, { a: { c: 48 } }, { d: 'yep' }], | ||
|  |               _doc: { yep: [15], arr: [{ a: { b: '$' }, e: 'a' }, { a: { c: 48, b: '$' }, e: 'b' }, { d: 'yep', e: 35 }] } }, | ||
|  |             { yep: true, _doc: { special: true, yep: 0 } } | ||
|  |           ], o.arr); | ||
|  | 
 | ||
|  |           done(); | ||
|  |         }); | ||
|  |         it('with array', function(done) { | ||
|  |           mpath.set('arr.0.arr.$.a.b', [1], o, special, function(v) { | ||
|  |             return {}; | ||
|  |           }); | ||
|  |           assert.deepEqual([ | ||
|  |             { yep: 47, arr: [{ a: { b: 47 } }, { a: { c: 48 } }, { d: 'yep' }], | ||
|  |               _doc: { yep: [15], arr: [{ a: { b: {} }, e: 'a' }, { a: { c: 48, b: {} }, e: 'b' }, { d: 'yep', e: 35 }] } }, | ||
|  |             { yep: true, _doc: { special: true, yep: 0 } } | ||
|  |           ], o.arr); | ||
|  | 
 | ||
|  |           mpath.set('arr.0.arr.$.a.b', [1], o, special); | ||
|  |           assert.deepEqual([ | ||
|  |             { yep: 47, arr: [{ a: { b: 47 } }, { a: { c: 48 } }, { d: 'yep' }], | ||
|  |               _doc: { yep: [15], arr: [{ a: { b: [1] }, e: 'a' }, { a: { c: 48, b: [1] }, e: 'b' }, { d: 'yep', e: 35 }] } }, | ||
|  |             { yep: true, _doc: { special: true, yep: 0 } } | ||
|  |           ], o.arr); | ||
|  | 
 | ||
|  |           done(); | ||
|  |         }); | ||
|  |       }); | ||
|  | 
 | ||
|  |       describe('array.array.index.path', function() { | ||
|  |         it('with single value', function(done) { | ||
|  |           mpath.set('arr.arr.0.a', 'single', o, special, function(v) { | ||
|  |             return 88; | ||
|  |           }); | ||
|  |           assert.deepEqual([ | ||
|  |             { yep: 47, arr: [{ a: { b: 47 } }, { a: { c: 48 } }, { d: 'yep' }], | ||
|  |               _doc: { yep: [15], arr: [{ a: 88, e: 'a' }, { a: { c: 48, b: [1] }, e: 'b' }, { d: 'yep', e: 35 }] } }, | ||
|  |             { yep: true, _doc: { special: true, yep: 0 } } | ||
|  |           ], o.arr); | ||
|  | 
 | ||
|  |           mpath.set('arr.arr.0.a', 'single', o, special); | ||
|  |           assert.deepEqual([ | ||
|  |             { yep: 47, arr: [{ a: { b: 47 } }, { a: { c: 48 } }, { d: 'yep' }], | ||
|  |               _doc: { yep: [15], arr: [{ a: 'single', e: 'a' }, { a: { c: 48, b: [1] }, e: 'b' }, { d: 'yep', e: 35 }] } }, | ||
|  |             { yep: true, _doc: { special: true, yep: 0 } } | ||
|  |           ], o.arr); | ||
|  | 
 | ||
|  |           done(); | ||
|  |         }); | ||
|  |         it('with array', function(done) { | ||
|  |           mpath.set('arr.arr.0.a', [4, 8, 15, 16, 23, 42], o, special, function(v) { | ||
|  |             return v * 2; | ||
|  |           }); | ||
|  |           assert.deepEqual([ | ||
|  |             { yep: 47, arr: [{ a: { b: 47 } }, { a: { c: 48 } }, { d: 'yep' }], | ||
|  |               _doc: { yep: [15], arr: [{ a: 8, e: 'a' }, { a: { c: 48, b: [1] }, e: 'b' }, { d: 'yep', e: 35 }] } }, | ||
|  |             { yep: true, _doc: { special: true, yep: 0 } } | ||
|  |           ], o.arr); | ||
|  | 
 | ||
|  |           mpath.set('arr.arr.0.a', [4, 8, 15, 16, 23, 42], o, special); | ||
|  |           assert.deepEqual([ | ||
|  |             { yep: 47, arr: [{ a: { b: 47 } }, { a: { c: 48 } }, { d: 'yep' }], | ||
|  |               _doc: { yep: [15], arr: [{ a: 4, e: 'a' }, { a: { c: 48, b: [1] }, e: 'b' }, { d: 'yep', e: 35 }] } }, | ||
|  |             { yep: true, _doc: { special: true, yep: 0 } } | ||
|  |           ], o.arr); | ||
|  | 
 | ||
|  |           done(); | ||
|  |         }); | ||
|  |       }); | ||
|  | 
 | ||
|  |       describe('array.array.$.index.path', function() { | ||
|  |         it('with single value', function(done) { | ||
|  |           mpath.set('arr.arr.$.0.a', 'singles', o, special, function(v) { | ||
|  |             return v.toUpperCase(); | ||
|  |           }); | ||
|  |           assert.deepEqual([ | ||
|  |             { yep: 47, arr: [{ a: { b: 47 } }, { a: { c: 48 } }, { d: 'yep' }], | ||
|  |               _doc: { yep: [15], arr: [{ a: 'SINGLES', e: 'a' }, { a: { c: 48, b: [1] }, e: 'b' }, { d: 'yep', e: 35 }] } }, | ||
|  |             { yep: true, _doc: { special: true, yep: 0 } } | ||
|  |           ], o.arr); | ||
|  | 
 | ||
|  |           mpath.set('arr.arr.$.0.a', 'singles', o, special); | ||
|  |           assert.deepEqual([ | ||
|  |             { yep: 47, arr: [{ a: { b: 47 } }, { a: { c: 48 } }, { d: 'yep' }], | ||
|  |               _doc: { yep: [15], arr: [{ a: 'singles', e: 'a' }, { a: { c: 48, b: [1] }, e: 'b' }, { d: 'yep', e: 35 }] } }, | ||
|  |             { yep: true, _doc: { special: true, yep: 0 } } | ||
|  |           ], o.arr); | ||
|  | 
 | ||
|  |           mpath.set('$.arr.arr.0.a', 'single', o, special); | ||
|  |           assert.deepEqual([ | ||
|  |             { yep: 47, arr: [{ a: { b: 47 } }, { a: { c: 48 } }, { d: 'yep' }], | ||
|  |               _doc: { yep: [15], arr: [{ a: 'single', e: 'a' }, { a: { c: 48, b: [1] }, e: 'b' }, { d: 'yep', e: 35 }] } }, | ||
|  |             { yep: true, _doc: { special: true, yep: 0 } } | ||
|  |           ], o.arr); | ||
|  | 
 | ||
|  |           done(); | ||
|  |         }); | ||
|  |         it('with array', function(done) { | ||
|  |           mpath.set('arr.arr.$.0.a', [4, 8, 15, 16, 23, 42], o, special, function(v) { | ||
|  |             return Array; | ||
|  |           }); | ||
|  |           assert.deepEqual([ | ||
|  |             { yep: 47, arr: [{ a: { b: 47 } }, { a: { c: 48 } }, { d: 'yep' }], | ||
|  |               _doc: { yep: [15], arr: [{ a: Array, e: 'a' }, { a: { c: 48, b: [1] }, e: 'b' }, { d: 'yep', e: 35 }] } }, | ||
|  |             { yep: true, _doc: { special: true, yep: 0 } } | ||
|  |           ], o.arr); | ||
|  | 
 | ||
|  |           mpath.set('arr.arr.$.0.a', [4, 8, 15, 16, 23, 42], o, special); | ||
|  |           assert.deepEqual([ | ||
|  |             { yep: 47, arr: [{ a: { b: 47 } }, { a: { c: 48 } }, { d: 'yep' }], | ||
|  |               _doc: { yep: [15], arr: [{ a: [4, 8, 15, 16, 23, 42], e: 'a' }, { a: { c: 48, b: [1] }, e: 'b' }, { d: 'yep', e: 35 }] } }, | ||
|  |             { yep: true, _doc: { special: true, yep: 0 } } | ||
|  |           ], o.arr); | ||
|  | 
 | ||
|  |           mpath.set('arr.$.arr.0.a', [4, 8, 15, 16, 23, 42, 108], o, special); | ||
|  |           assert.deepEqual([ | ||
|  |             { yep: 47, arr: [{ a: { b: 47 } }, { a: { c: 48 } }, { d: 'yep' }], | ||
|  |               _doc: { yep: [15], arr: [{ a: [4, 8, 15, 16, 23, 42, 108], e: 'a' }, { a: { c: 48, b: [1] }, e: 'b' }, { d: 'yep', e: 35 }] } }, | ||
|  |             { yep: true, _doc: { special: true, yep: 0 } } | ||
|  |           ], o.arr); | ||
|  | 
 | ||
|  |           done(); | ||
|  |         }); | ||
|  |       }); | ||
|  | 
 | ||
|  |       describe('array.array.path.index', function() { | ||
|  |         it('with single value', function(done) { | ||
|  |           mpath.set('arr.arr.a.7', 47, o, special, function(v) { | ||
|  |             return Object; | ||
|  |           }); | ||
|  |           assert.deepEqual([ | ||
|  |             { yep: 47, arr: [{ a: { b: 47 } }, { a: { c: 48 } }, { d: 'yep' }], | ||
|  |               _doc: { yep: [15], arr: [{ a: [4, 8, 15, 16, 23, 42, 108, Object], e: 'a' }, { a: { c: 48, b: [1], 7: Object }, e: 'b' }, { d: 'yep', e: 35 }] } }, | ||
|  |             { yep: true, _doc: { special: true, yep: 0 } } | ||
|  |           ], o.arr); | ||
|  | 
 | ||
|  |           mpath.set('arr.arr.a.7', 47, o, special); | ||
|  |           assert.deepEqual([ | ||
|  |             { yep: 47, arr: [{ a: { b: 47 } }, { a: { c: 48 } }, { d: 'yep' }], | ||
|  |               _doc: { yep: [15], arr: [{ a: [4, 8, 15, 16, 23, 42, 108, 47], e: 'a' }, { a: { c: 48, b: [1], 7: 47 }, e: 'b' }, { d: 'yep', e: 35 }] } }, | ||
|  |             { yep: true, _doc: { special: true, yep: 0 } } | ||
|  |           ], o.arr); | ||
|  | 
 | ||
|  |           done(); | ||
|  |         }); | ||
|  |         it('with array', function(done) { | ||
|  |           o.arr[1]._doc.arr = [{ a: [] }, { a: [] }, { a: null }]; | ||
|  |           mpath.set('arr.arr.a.7', [[null, 46], [undefined, 'woot']], o, special, function(v) { | ||
|  |             return undefined === v ? 'nope' : v; | ||
|  |           }); | ||
|  | 
 | ||
|  |           const a1 = []; | ||
|  |           const a2 = []; | ||
|  |           a1[7] = 'nope'; | ||
|  |           a2[7] = 'woot'; | ||
|  | 
 | ||
|  |           assert.deepEqual([ | ||
|  |             { yep: 47, arr: [{ a: { b: 47 } }, { a: { c: 48 } }, { d: 'yep' }], | ||
|  |               _doc: { yep: [15], arr: [{ a: [4, 8, 15, 16, 23, 42, 108, null], e: 'a' }, { a: { c: 48, b: [1], 7: 46 }, e: 'b' }, { d: 'yep', e: 35 }] } }, | ||
|  |             { yep: true, _doc: { arr: [{ a: a1 }, { a: a2 }, { a: null }], special: true, yep: 0 } } | ||
|  |           ], o.arr); | ||
|  | 
 | ||
|  |           mpath.set('arr.arr.a.7', [[null, 46], [undefined, 'woot']], o, special); | ||
|  | 
 | ||
|  |           a1[7] = undefined; | ||
|  | 
 | ||
|  |           assert.deepEqual([ | ||
|  |             { yep: 47, arr: [{ a: { b: 47 } }, { a: { c: 48 } }, { d: 'yep' }], | ||
|  |               _doc: { yep: [15], arr: [{ a: [4, 8, 15, 16, 23, 42, 108, null], e: 'a' }, { a: { c: 48, b: [1], 7: 46 }, e: 'b' }, { d: 'yep', e: 35 }] } }, | ||
|  |             { yep: true, _doc: { arr: [{ a: a1 }, { a: a2 }, { a: null }], special: true, yep: 0 } } | ||
|  |           ], o.arr); | ||
|  | 
 | ||
|  |           done(); | ||
|  |         }); | ||
|  |       }); | ||
|  | 
 | ||
|  |       describe('handles array.array.path', function() { | ||
|  |         it('with single', function(done) { | ||
|  |           o.arr[1]._doc.arr = [{}, {}]; | ||
|  |           assert.deepEqual([{}, {}], o.arr[1]._doc.arr); | ||
|  |           o.arr.push({ _doc: { arr: 'something else' } }); | ||
|  |           o.arr.push({ _doc: { arr: ['something else'] } }); | ||
|  |           o.arr.push({ _doc: { arr: [[]] } }); | ||
|  |           o.arr.push({ _doc: { arr: [5] } }); | ||
|  | 
 | ||
|  |           // test
 | ||
|  |           mpath.set('arr.arr.e', 47, o, special); | ||
|  | 
 | ||
|  |           const weird = []; | ||
|  |           weird.e = 47; | ||
|  | 
 | ||
|  |           assert.deepEqual([ | ||
|  |             { yep: 47, arr: [{ a: { b: 47 } }, { a: { c: 48 } }, { d: 'yep' }], | ||
|  |               _doc: { | ||
|  |                 yep: [15], | ||
|  |                 arr: [ | ||
|  |                   { a: [4, 8, 15, 16, 23, 42, 108, null], e: 47 }, | ||
|  |                   { a: { c: 48, b: [1], 7: 46 }, e: 47 }, | ||
|  |                   { d: 'yep', e: 47 } | ||
|  |                 ] | ||
|  |               } | ||
|  |             }, | ||
|  |             { yep: true, | ||
|  |               _doc: { | ||
|  |                 arr: [ | ||
|  |                   { e: 47 }, | ||
|  |                   { e: 47 } | ||
|  |                 ], | ||
|  |                 special: true, | ||
|  |                 yep: 0 | ||
|  |               } | ||
|  |             }, | ||
|  |             { _doc: { arr: 'something else' } }, | ||
|  |             { _doc: { arr: ['something else'] } }, | ||
|  |             { _doc: { arr: [weird] } }, | ||
|  |             { _doc: { arr: [5] } } | ||
|  |           ] | ||
|  |           , o.arr); | ||
|  | 
 | ||
|  |           done(); | ||
|  |         }); | ||
|  |         it('with arrays', function(done) { | ||
|  |           mpath.set('arr.arr.e', [[1, 2, 3], [4, 5], null, [], [6], [7, 8, 9]], o, special); | ||
|  | 
 | ||
|  |           const weird = []; | ||
|  |           weird.e = 6; | ||
|  | 
 | ||
|  |           assert.deepEqual([ | ||
|  |             { yep: 47, arr: [{ a: { b: 47 } }, { a: { c: 48 } }, { d: 'yep' }], | ||
|  |               _doc: { | ||
|  |                 yep: [15], | ||
|  |                 arr: [ | ||
|  |                   { a: [4, 8, 15, 16, 23, 42, 108, null], e: 1 }, | ||
|  |                   { a: { c: 48, b: [1], 7: 46 }, e: 2 }, | ||
|  |                   { d: 'yep', e: 3 } | ||
|  |                 ] | ||
|  |               } | ||
|  |             }, | ||
|  |             { yep: true, | ||
|  |               _doc: { | ||
|  |                 arr: [ | ||
|  |                   { e: 4 }, | ||
|  |                   { e: 5 } | ||
|  |                 ], | ||
|  |                 special: true, | ||
|  |                 yep: 0 | ||
|  |               } | ||
|  |             }, | ||
|  |             { _doc: { arr: 'something else' } }, | ||
|  |             { _doc: { arr: ['something else'] } }, | ||
|  |             { _doc: { arr: [weird] } }, | ||
|  |             { _doc: { arr: [5] } } | ||
|  |           ] | ||
|  |           , o.arr); | ||
|  | 
 | ||
|  |           done(); | ||
|  |         }); | ||
|  |       }); | ||
|  | 
 | ||
|  |       describe('that is a function', function() { | ||
|  |         describe('without map', function() { | ||
|  |           it('works on array value', function(done) { | ||
|  |             const o = { hello: { world: [{ how: 'are' }, { you: '?' }] } }; | ||
|  |             const special = function(obj, key, val) { | ||
|  |               if (val) { | ||
|  |                 obj[key] = val; | ||
|  |               } else { | ||
|  |                 return 'thing' == key | ||
|  |                   ? obj.world | ||
|  |                   : obj[key]; | ||
|  |               } | ||
|  |             }; | ||
|  |             mpath.set('hello.thing.how', 'arrrr', o, special); | ||
|  |             assert.deepEqual(o, { hello: { world: [{ how: 'arrrr' }, { you: '?', how: 'arrrr' }] } }); | ||
|  |             done(); | ||
|  |           }); | ||
|  |           it('works on non-array value', function(done) { | ||
|  |             const o = { hello: { world: { how: 'are you' } } }; | ||
|  |             const special = function(obj, key, val) { | ||
|  |               if (val) { | ||
|  |                 obj[key] = val; | ||
|  |               } else { | ||
|  |                 return 'thing' == key | ||
|  |                   ? obj.world | ||
|  |                   : obj[key]; | ||
|  |               } | ||
|  |             }; | ||
|  |             mpath.set('hello.thing.how', 'RU', o, special); | ||
|  |             assert.deepEqual(o, { hello: { world: { how: 'RU' } } }); | ||
|  |             done(); | ||
|  |           }); | ||
|  |         }); | ||
|  |         it('works with map', function(done) { | ||
|  |           const o = { hello: { world: [{ how: 'are' }, { you: '?' }] } }; | ||
|  |           const special = function(obj, key, val) { | ||
|  |             if (val) { | ||
|  |               obj[key] = val; | ||
|  |             } else { | ||
|  |               return 'thing' == key | ||
|  |                 ? obj.world | ||
|  |                 : obj[key]; | ||
|  |             } | ||
|  |           }; | ||
|  |           const map = function(val) { | ||
|  |             return 'convert' == val | ||
|  |               ? 'ºº' | ||
|  |               : val; | ||
|  |           }; | ||
|  |           mpath.set('hello.thing.how', 'convert', o, special, map); | ||
|  |           assert.deepEqual(o, { hello: { world: [{ how: 'ºº' }, { you: '?', how: 'ºº' }] } }); | ||
|  |           done(); | ||
|  |         }); | ||
|  |       }); | ||
|  | 
 | ||
|  |     }); | ||
|  | 
 | ||
|  |     describe('get/set integration', function() { | ||
|  |       const o = doc(); | ||
|  | 
 | ||
|  |       it('works', function(done) { | ||
|  |         const vals = mpath.get('array.o.array.x.b', o); | ||
|  | 
 | ||
|  |         vals[0][0][2] = 10; | ||
|  |         vals[1][0][1] = 0; | ||
|  |         vals[1][1] = 'Rambaldi'; | ||
|  |         vals[1][2] = [12, 14]; | ||
|  |         vals[2] = [{ changed: true }, [null, ['changed', 'to', 'array']]]; | ||
|  | 
 | ||
|  |         mpath.set('array.o.array.x.b', vals, o); | ||
|  | 
 | ||
|  |         const t = [ | ||
|  |           { o: { array: [{ x: { b: [4, 6, 10] } }, { y: 10 }] } }, | ||
|  |           { o: { array: [{ x: { b: [1, 0, 3] } }, { x: { b: 'Rambaldi', z: 10 } }, { x: { b: [12, 14] } }] } }, | ||
|  |           { o: { array: [{ x: { b: { changed: true } } }, { x: { b: [null, ['changed', 'to', 'array']] } }] } }, | ||
|  |           { o: { array: [{ x: null }] } }, | ||
|  |           { o: { array: [{ y: 3 }] } }, | ||
|  |           { o: { array: [3, 0, null] } }, | ||
|  |           { o: { name: 'ha' } } | ||
|  |         ]; | ||
|  |         assert.deepEqual(t, o.array); | ||
|  |         done(); | ||
|  |       }); | ||
|  | 
 | ||
|  |       it('array.prop', function(done) { | ||
|  |         mpath.set('comments.name', ['this', 'was', 'changed'], o); | ||
|  | 
 | ||
|  |         assert.deepEqual([ | ||
|  |           { name: 'this' }, | ||
|  |           { name: 'was', _doc: { name: '2' } }, | ||
|  |           { name: 'changed', | ||
|  |             comments: [{}, { comments: [{ val: 'twoo' }] }], | ||
|  |             _doc: { name: '3', comments: [{}, { _doc: { comments: [{ val: 2 }] } }] } } | ||
|  |         ], o.comments); | ||
|  | 
 | ||
|  |         mpath.set('comments.name', ['also', 'changed', 'this'], o, special); | ||
|  | 
 | ||
|  |         assert.deepEqual([ | ||
|  |           { name: 'also' }, | ||
|  |           { name: 'was', _doc: { name: 'changed' } }, | ||
|  |           { name: 'changed', | ||
|  |             comments: [{}, { comments: [{ val: 'twoo' }] }], | ||
|  |             _doc: { name: 'this', comments: [{}, { _doc: { comments: [{ val: 2 }] } }] } } | ||
|  |         ], o.comments); | ||
|  | 
 | ||
|  |         done(); | ||
|  |       }); | ||
|  | 
 | ||
|  |       it('nested array', function(done) { | ||
|  |         const obj = { arr: [[{ test: 41 }]] }; | ||
|  |         mpath.set('arr.test', [[42]], obj); | ||
|  |         assert.deepEqual(obj.arr, [[{ test: 42 }]]); | ||
|  |         done(); | ||
|  |       }); | ||
|  |     }); | ||
|  | 
 | ||
|  |     describe('multiple $ use', function() { | ||
|  |       const o = doc(); | ||
|  |       it('is ok', function(done) { | ||
|  |         assert.doesNotThrow(function() { | ||
|  |           mpath.set('arr.$.arr.$.a', 35, o); | ||
|  |         }); | ||
|  |         done(); | ||
|  |       }); | ||
|  |     }); | ||
|  | 
 | ||
|  |     it('has', function(done) { | ||
|  |       assert.ok(mpath.has('a', { a: 1 })); | ||
|  |       assert.ok(mpath.has('a', { a: undefined })); | ||
|  |       assert.ok(!mpath.has('a', {})); | ||
|  |       assert.ok(!mpath.has('a', null)); | ||
|  | 
 | ||
|  |       assert.ok(mpath.has('a.b', { a: { b: 1 } })); | ||
|  |       assert.ok(mpath.has('a.b', { a: { b: undefined } })); | ||
|  |       assert.ok(!mpath.has('a.b', { a: 1 })); | ||
|  |       assert.ok(!mpath.has('a.b', { a: null })); | ||
|  | 
 | ||
|  |       done(); | ||
|  |     }); | ||
|  | 
 | ||
|  |     it('underneath a map', function(done) { | ||
|  |       if (!global.Map) { | ||
|  |         done(); | ||
|  |         return; | ||
|  |       } | ||
|  |       assert.equal(mpath.get('a.b', { a: new Map([['b', 1]]) }), 1); | ||
|  | 
 | ||
|  |       const m = new Map([['b', 1]]); | ||
|  |       const obj = { a: m }; | ||
|  |       mpath.set('a.c', 2, obj); | ||
|  |       assert.equal(m.get('c'), 2); | ||
|  | 
 | ||
|  |       done(); | ||
|  |     }); | ||
|  | 
 | ||
|  |     it('unset', function(done) { | ||
|  |       let o = { a: 1 }; | ||
|  |       mpath.unset('a', o); | ||
|  |       assert.deepEqual(o, {}); | ||
|  | 
 | ||
|  |       o = { a: { b: 1 } }; | ||
|  |       mpath.unset('a.b', o); | ||
|  |       assert.deepEqual(o, { a: {} }); | ||
|  | 
 | ||
|  |       o = { a: null }; | ||
|  |       mpath.unset('a.b', o); | ||
|  |       assert.deepEqual(o, { a: null }); | ||
|  | 
 | ||
|  |       done(); | ||
|  |     }); | ||
|  | 
 | ||
|  |     it('unset with __proto__', function(done) { | ||
|  |       // Should refuse to set __proto__
 | ||
|  |       function Clazz() {} | ||
|  |       Clazz.prototype.foobar = true; | ||
|  | 
 | ||
|  |       mpath.unset('__proto__.foobar', new Clazz()); | ||
|  |       assert.ok(Clazz.prototype.foobar); | ||
|  | 
 | ||
|  |       mpath.unset('constructor.prototype.foobar', new Clazz()); | ||
|  |       assert.ok(Clazz.prototype.foobar); | ||
|  | 
 | ||
|  |       done(); | ||
|  |     }); | ||
|  | 
 | ||
|  |     it('get() underneath subclassed array', function(done) { | ||
|  |       class MyArray extends Array {} | ||
|  | 
 | ||
|  |       const obj = { | ||
|  |         arr: new MyArray() | ||
|  |       }; | ||
|  |       obj.arr.push({ test: 2 }); | ||
|  | 
 | ||
|  |       const arr = mpath.get('arr.test', obj); | ||
|  |       assert.equal(arr.constructor.name, 'Array'); | ||
|  |       assert.ok(!(arr instanceof MyArray)); | ||
|  | 
 | ||
|  |       done(); | ||
|  |     }); | ||
|  | 
 | ||
|  |     it('ignores setting a nested path that doesnt exist', function(done) { | ||
|  |       const o = doc(); | ||
|  |       assert.doesNotThrow(function() { | ||
|  |         mpath.set('thing.that.is.new', 10, o); | ||
|  |       }); | ||
|  |       done(); | ||
|  |     }); | ||
|  |   }); | ||
|  | }); |