/* * Copyright (C) 2020-2021 Yomichan Authors * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see <https://www.gnu.org/licenses/>. */ const assert = require('assert'); const {testMain} = require('../dev/util'); const {VM} = require('../dev/vm'); const vm = new VM({}); vm.execute('js/general/object-property-accessor.js'); const ObjectPropertyAccessor = vm.get('ObjectPropertyAccessor'); function createTestObject() { return { 0: null, value1: { value2: {}, value3: [], value4: null }, value5: [ {}, [], null ] }; } function testGet1() { const data = [ [[], (object) => object], [['0'], (object) => object['0']], [['value1'], (object) => object.value1], [['value1', 'value2'], (object) => object.value1.value2], [['value1', 'value3'], (object) => object.value1.value3], [['value1', 'value4'], (object) => object.value1.value4], [['value5'], (object) => object.value5], [['value5', 0], (object) => object.value5[0]], [['value5', 1], (object) => object.value5[1]], [['value5', 2], (object) => object.value5[2]] ]; for (const [pathArray, getExpected] of data) { const object = createTestObject(); const accessor = new ObjectPropertyAccessor(object); const expected = getExpected(object); assert.strictEqual(accessor.get(pathArray), expected); } } function testGet2() { const object = createTestObject(); const accessor = new ObjectPropertyAccessor(object); const data = [ [[0], 'Invalid path: [0]'], [['0', 'invalid'], 'Invalid path: ["0"].invalid'], [['invalid'], 'Invalid path: invalid'], [['value1', 'invalid'], 'Invalid path: value1.invalid'], [['value1', 'value2', 'invalid'], 'Invalid path: value1.value2.invalid'], [['value1', 'value2', 0], 'Invalid path: value1.value2[0]'], [['value1', 'value3', 'invalid'], 'Invalid path: value1.value3.invalid'], [['value1', 'value3', 0], 'Invalid path: value1.value3[0]'], [['value1', 'value4', 'invalid'], 'Invalid path: value1.value4.invalid'], [['value1', 'value4', 0], 'Invalid path: value1.value4[0]'], [['value5', 'length'], 'Invalid path: value5.length'], [['value5', 0, 'invalid'], 'Invalid path: value5[0].invalid'], [['value5', 0, 0], 'Invalid path: value5[0][0]'], [['value5', 1, 'invalid'], 'Invalid path: value5[1].invalid'], [['value5', 1, 0], 'Invalid path: value5[1][0]'], [['value5', 2, 'invalid'], 'Invalid path: value5[2].invalid'], [['value5', 2, 0], 'Invalid path: value5[2][0]'], [['value5', 2, 0, 'invalid'], 'Invalid path: value5[2][0]'], [['value5', 2.5], 'Invalid index'] ]; for (const [pathArray, message] of data) { assert.throws(() => accessor.get(pathArray), {message}); } } function testSet1() { const testValue = {}; const data = [ ['0'], ['value1', 'value2'], ['value1', 'value3'], ['value1', 'value4'], ['value1'], ['value5', 0], ['value5', 1], ['value5', 2], ['value5'] ]; for (const pathArray of data) { const object = createTestObject(); const accessor = new ObjectPropertyAccessor(object); accessor.set(pathArray, testValue); assert.strictEqual(accessor.get(pathArray), testValue); } } function testSet2() { const object = createTestObject(); const accessor = new ObjectPropertyAccessor(object); const testValue = {}; const data = [ [[], 'Invalid path'], [[0], 'Invalid path: [0]'], [['0', 'invalid'], 'Invalid path: ["0"].invalid'], [['value1', 'value2', 0], 'Invalid path: value1.value2[0]'], [['value1', 'value3', 'invalid'], 'Invalid path: value1.value3.invalid'], [['value1', 'value4', 'invalid'], 'Invalid path: value1.value4.invalid'], [['value1', 'value4', 0], 'Invalid path: value1.value4[0]'], [['value5', 1, 'invalid'], 'Invalid path: value5[1].invalid'], [['value5', 2, 'invalid'], 'Invalid path: value5[2].invalid'], [['value5', 2, 0], 'Invalid path: value5[2][0]'], [['value5', 2, 0, 'invalid'], 'Invalid path: value5[2][0]'], [['value5', 2.5], 'Invalid index'] ]; for (const [pathArray, message] of data) { assert.throws(() => accessor.set(pathArray, testValue), {message}); } } function testDelete1() { const hasOwn = (object, property) => Object.prototype.hasOwnProperty.call(object, property); const data = [ [['0'], (object) => !hasOwn(object, '0')], [['value1', 'value2'], (object) => !hasOwn(object.value1, 'value2')], [['value1', 'value3'], (object) => !hasOwn(object.value1, 'value3')], [['value1', 'value4'], (object) => !hasOwn(object.value1, 'value4')], [['value1'], (object) => !hasOwn(object, 'value1')], [['value5'], (object) => !hasOwn(object, 'value5')] ]; for (const [pathArray, validate] of data) { const object = createTestObject(); const accessor = new ObjectPropertyAccessor(object); accessor.delete(pathArray); assert.ok(validate(object)); } } function testDelete2() { const data = [ [[], 'Invalid path'], [[0], 'Invalid path: [0]'], [['0', 'invalid'], 'Invalid path: ["0"].invalid'], [['value1', 'value2', 0], 'Invalid path: value1.value2[0]'], [['value1', 'value3', 'invalid'], 'Invalid path: value1.value3.invalid'], [['value1', 'value4', 'invalid'], 'Invalid path: value1.value4.invalid'], [['value1', 'value4', 0], 'Invalid path: value1.value4[0]'], [['value5', 1, 'invalid'], 'Invalid path: value5[1].invalid'], [['value5', 2, 'invalid'], 'Invalid path: value5[2].invalid'], [['value5', 2, 0], 'Invalid path: value5[2][0]'], [['value5', 2, 0, 'invalid'], 'Invalid path: value5[2][0]'], [['value5', 2.5], 'Invalid index'], [['value5', 0], 'Invalid type'], [['value5', 1], 'Invalid type'], [['value5', 2], 'Invalid type'] ]; for (const [pathArray, message] of data) { const object = createTestObject(); const accessor = new ObjectPropertyAccessor(object); assert.throws(() => accessor.delete(pathArray), {message}); } } function testSwap1() { const data = [ [['0'], true], [['value1', 'value2'], true], [['value1', 'value3'], true], [['value1', 'value4'], true], [['value1'], false], [['value5', 0], true], [['value5', 1], true], [['value5', 2], true], [['value5'], false] ]; for (const [pathArray1, compareValues1] of data) { for (const [pathArray2, compareValues2] of data) { const object = createTestObject(); const accessor = new ObjectPropertyAccessor(object); const value1a = accessor.get(pathArray1); const value2a = accessor.get(pathArray2); accessor.swap(pathArray1, pathArray2); if (!compareValues1 || !compareValues2) { continue; } const value1b = accessor.get(pathArray1); const value2b = accessor.get(pathArray2); assert.deepStrictEqual(value1a, value2b); assert.deepStrictEqual(value2a, value1b); } } } function testSwap2() { const data = [ [[], [], false, 'Invalid path 1'], [['0'], [], false, 'Invalid path 2'], [[], ['0'], false, 'Invalid path 1'], [[0], ['0'], false, 'Invalid path 1: [0]'], [['0'], [0], false, 'Invalid path 2: [0]'] ]; for (const [pathArray1, pathArray2, checkRevert, message] of data) { const object = createTestObject(); const accessor = new ObjectPropertyAccessor(object); let value1a; let value2a; if (checkRevert) { try { value1a = accessor.get(pathArray1); value2a = accessor.get(pathArray2); } catch (e) { // NOP } } assert.throws(() => accessor.swap(pathArray1, pathArray2), {message}); if (!checkRevert) { continue; } const value1b = accessor.get(pathArray1); const value2b = accessor.get(pathArray2); assert.deepStrictEqual(value1a, value1b); assert.deepStrictEqual(value2a, value2b); } } function testGetPathString1() { const data = [ [[], ''], [[0], '[0]'], [['escape\\'], '["escape\\\\"]'], [['\'quote\''], '["\'quote\'"]'], [['"quote"'], '["\\"quote\\""]'], [['part1', 'part2'], 'part1.part2'], [['part1', 'part2', 3], 'part1.part2[3]'], [['part1', 'part2', '3'], 'part1.part2["3"]'], [['part1', 'part2', '3part'], 'part1.part2["3part"]'], [['part1', 'part2', '3part', 'part4'], 'part1.part2["3part"].part4'], [['part1', 'part2', '3part', '4part'], 'part1.part2["3part"]["4part"]'] ]; for (const [pathArray, expected] of data) { assert.strictEqual(ObjectPropertyAccessor.getPathString(pathArray), expected); } } function testGetPathString2() { const data = [ [[1.5], 'Invalid index'], [[null], 'Invalid type: object'] ]; for (const [pathArray, message] of data) { assert.throws(() => ObjectPropertyAccessor.getPathString(pathArray), {message}); } } function testGetPathArray1() { const data = [ ['', []], ['[0]', [0]], ['["escape\\\\"]', ['escape\\']], ['["\'quote\'"]', ['\'quote\'']], ['["\\"quote\\""]', ['"quote"']], ['part1.part2', ['part1', 'part2']], ['part1.part2[3]', ['part1', 'part2', 3]], ['part1.part2["3"]', ['part1', 'part2', '3']], ['part1.part2[\'3\']', ['part1', 'part2', '3']], ['part1.part2["3part"]', ['part1', 'part2', '3part']], ['part1.part2[\'3part\']', ['part1', 'part2', '3part']], ['part1.part2["3part"].part4', ['part1', 'part2', '3part', 'part4']], ['part1.part2[\'3part\'].part4', ['part1', 'part2', '3part', 'part4']], ['part1.part2["3part"]["4part"]', ['part1', 'part2', '3part', '4part']], ['part1.part2[\'3part\'][\'4part\']', ['part1', 'part2', '3part', '4part']] ]; for (const [pathString, expected] of data) { vm.assert.deepStrictEqual(ObjectPropertyAccessor.getPathArray(pathString), expected); } } function testGetPathArray2() { const data = [ ['?', 'Unexpected character: ?'], ['.', 'Unexpected character: .'], ['0', 'Unexpected character: 0'], ['part1.[0]', 'Unexpected character: ['], ['part1?', 'Unexpected character: ?'], ['[part1]', 'Unexpected character: p'], ['[0a]', 'Unexpected character: a'], ['["part1"x]', 'Unexpected character: x'], ['[\'part1\'x]', 'Unexpected character: x'], ['["part1"]x', 'Unexpected character: x'], ['[\'part1\']x', 'Unexpected character: x'], ['part1..part2', 'Unexpected character: .'], ['[', 'Path not terminated correctly'], ['part1.', 'Path not terminated correctly'], ['part1[', 'Path not terminated correctly'], ['part1["', 'Path not terminated correctly'], ['part1[\'', 'Path not terminated correctly'], ['part1[""', 'Path not terminated correctly'], ['part1[\'\'', 'Path not terminated correctly'], ['part1[0', 'Path not terminated correctly'], ['part1[0].', 'Path not terminated correctly'] ]; for (const [pathString, message] of data) { assert.throws(() => ObjectPropertyAccessor.getPathArray(pathString), {message}); } } function testHasProperty() { const data = [ [{}, 'invalid', false], [{}, 0, false], [{valid: 0}, 'valid', true], [{null: 0}, null, false], [[], 'invalid', false], [[], 0, false], [[0], 0, true], [[0], null, false], ['string', 0, false], ['string', 'length', false], ['string', null, false] ]; for (const [object, property, expected] of data) { assert.strictEqual(ObjectPropertyAccessor.hasProperty(object, property), expected); } } function testIsValidPropertyType() { const data = [ [{}, 'invalid', true], [{}, 0, false], [{valid: 0}, 'valid', true], [{null: 0}, null, false], [[], 'invalid', false], [[], 0, true], [[0], 0, true], [[0], null, false], ['string', 0, false], ['string', 'length', false], ['string', null, false] ]; for (const [object, property, expected] of data) { assert.strictEqual(ObjectPropertyAccessor.isValidPropertyType(object, property), expected); } } function main() { testGet1(); testGet2(); testSet1(); testSet2(); testDelete1(); testDelete2(); testSwap1(); testSwap2(); testGetPathString1(); testGetPathString2(); testGetPathArray1(); testGetPathArray2(); testHasProperty(); testIsValidPropertyType(); } if (require.main === module) { testMain(main); }