var Meta = module.exports; var deduplicate = require("./common-util").deduplicateString; /* Metadata fields and the commands that can modify them we assume that these commands can only be performed by owners or in some cases pending owners. Thus the owners field is guaranteed to exist. * channel <STRING> * validateKey <STRING> * owners <ARRAY> * ADD_OWNERS * RM_OWNERS * RESET_OWNERS * pending_owners <ARRAY> * ADD_PENDING_OWNERS * RM_PENDING_OWNERS * expire <NUMBER> * UPDATE_EXPIRATION (NOT_IMPLEMENTED) * restricted <BOOLEAN> * RESTRICT_ACCESS * allowed <ARRAY> * ADD_ALLOWED * RM_ALLOWED * RESET_ALLOWED * ADD_OWNERS * RESET_OWNERS * mailbox <STRING|MAP> * ADD_MAILBOX * RM_MAILBOX */ var commands = {}; var isValidPublicKey = function (owner) { return typeof(owner) === 'string' && owner.length === 44; }; // isValidPublicKey is a better indication of what the above function does // I'm preserving this function name in case we ever want to expand its // criteria at a later time... var isValidOwner = isValidPublicKey; // ["RESTRICT_ACCESS", [true], 1561623438989] // ["RESTRICT_ACCESS", [false], 1561623438989] commands.RESTRICT_ACCESS = function (meta, args) { if (!Array.isArray(args) || typeof(args[0]) !== 'boolean') { throw new Error('INVALID_STATE'); } var bool = args[0]; // reject the proposed command if there is no change in state if (meta.restricted === bool) { return false; } // apply the new state meta.restricted = args[0]; // if you're disabling access restrictions then you can assume // then there is nothing more to do. Leave the existing list as-is if (!bool) { return true; } // you're all set if an allow list already exists if (Array.isArray(meta.allowed)) { return true; } // otherwise define it meta.allowed = []; return true; }; // ["ADD_ALLOWED", ["7eEqelGso3EBr5jHlei6av4r9w2B9XZiGGwA1EgZ-5I=", ...], 1561623438989] commands.ADD_ALLOWED = function (meta, args) { if (!Array.isArray(args)) { throw new Error("INVALID_ARGS"); } var allowed = meta.allowed || []; var changed = false; args.forEach(function (arg) { // don't add invalid public keys if (!isValidPublicKey(arg)) { return; } // don't add owners to the allow list if (meta.owners.indexOf(arg) >= 0) { return; } // don't duplicate entries in the allow list if (allowed.indexOf(arg) >= 0) { return; } allowed.push(arg); changed = true; }); if (changed) { meta.allowed = meta.allowed || allowed; } return changed; }; // ["RM_ALLOWED", ["7eEqelGso3EBr5jHlei6av4r9w2B9XZiGGwA1EgZ-5I=", ...], 1561623438989] commands.RM_ALLOWED = function (meta, args) { if (!Array.isArray(args)) { throw new Error("INVALID_ARGS"); } // there may not be anything to remove if (!meta.allowed) { return false; } var changed = false; args.forEach(function (arg) { var index = meta.allowed.indexOf(arg); if (index < 0) { return; } meta.allowed.splice(index, 1); changed = true; }); return changed; }; var arrayHasChanged = function (A, B) { var changed; A.some(function (a) { if (B.indexOf(a) < 0) { return (changed = true); } }); if (changed) { return true; } B.some(function (b) { if (A.indexOf(b) < 0) { return (changed = true); } }); return changed; }; var filterInPlace = function (A, f) { for (var i = A.length - 1; i >= 0; i--) { if (f(A[i], i, A)) { A.splice(i, 1); } } }; // ["RESET_ALLOWED", ["7eEqelGso3EBr5jHlei6av4r9w2B9XZiGGwA1EgZ-5I=", ...], 1561623438989] commands.RESET_ALLOWED = function (meta, args) { if (!Array.isArray(args)) { throw new Error("INVALID_ARGS"); } var updated = args.filter(function (arg) { // don't allow invalid public keys if (!isValidPublicKey(arg)) { return false; } // don't ever add owners to the allow list if (meta.owners.indexOf(arg)) { return false; } return true; }); // this is strictly an optimization... // a change in length is a clear indicator of a functional change if (meta.allowed && meta.allowed.length !== updated.length) { meta.allowed = updated; return true; } // otherwise we must check that the arrays contain distinct elements // if there is no functional change, then return false if (!arrayHasChanged(meta.allowed, updated)) { return false; } // otherwise overwrite the in-memory data and indicate that there was a change meta.allowed = updated; return true; }; // ["ADD_OWNERS", ["7eEqelGso3EBr5jHlei6av4r9w2B9XZiGGwA1EgZ-5I="], 1561623438989] commands.ADD_OWNERS = function (meta, args) { // bail out if args isn't an array if (!Array.isArray(args)) { throw new Error('METADATA_INVALID_OWNERS'); } // you shouldn't be able to get here if there are no owners // because only an owner should be able to change the owners if (!Array.isArray(meta.owners)) { throw new Error("METADATA_NONSENSE_OWNERS"); } var changed = false; args.forEach(function (owner) { if (!isValidOwner(owner)) { return; } if (meta.owners.indexOf(owner) >= 0) { return; } meta.owners.push(owner); changed = true; }); if (changed && Array.isArray(meta.allowed)) { // make sure owners are not included in the allow list filterInPlace(meta.allowed, function (member) { return meta.owners.indexOf(member) !== -1; }); } return changed; }; // ["RM_OWNERS", ["CrufexqXcY-z+eKJlEbNELVy5Sb7E-EAAEFI8GnEtZ0="], 1561623439989] commands.RM_OWNERS = function (meta, args) { // what are you doing if you don't have owners to remove? if (!Array.isArray(args)) { throw new Error('METADATA_INVALID_OWNERS'); } // if there aren't any owners to start, this is also pointless if (!Array.isArray(meta.owners)) { throw new Error("METADATA_NONSENSE_OWNERS"); } var changed = false; // remove owners one by one // we assume there are no duplicates args.forEach(function (owner) { var index = meta.owners.indexOf(owner); if (index < 0) { return; } if (meta.mailbox) { if (typeof(meta.mailbox) === "string") { delete meta.mailbox; } else { delete meta.mailbox[owner]; } } meta.owners.splice(index, 1); changed = true; }); if (meta.owners.length === 0 && meta.restricted) { meta.restricted = false; } return changed; }; // ["ADD_PENDING_OWNERS", ["7eEqelGso3EBr5jHlei6av4r9w2B9XZiGGwA1EgZ-5I="], 1561623438989] commands.ADD_PENDING_OWNERS = function (meta, args) { // bail out if args isn't an array if (!Array.isArray(args)) { throw new Error('METADATA_INVALID_PENDING_OWNERS'); } // you shouldn't be able to get here if there are no owners // because only an owner should be able to change the owners if (meta.pending_owners && !Array.isArray(meta.pending_owners)) { throw new Error("METADATA_NONSENSE_PENDING_OWNERS"); } var changed = false; // Add pending_owners array if it doesn't exist if (!meta.pending_owners) { meta.pending_owners = deduplicate(args); return true; } // or fill it args.forEach(function (owner) { if (!isValidOwner(owner)) { return; } if (meta.pending_owners.indexOf(owner) >= 0) { return; } meta.pending_owners.push(owner); changed = true; }); return changed; }; // ["RM_PENDING_OWNERS", ["CrufexqXcY-z+eKJlEbNELVy5Sb7E-EAAEFI8GnEtZ0="], 1561623439989] commands.RM_PENDING_OWNERS = function (meta, args) { // what are you doing if you don't have owners to remove? if (!Array.isArray(args)) { throw new Error('METADATA_INVALID_PENDING_OWNERS'); } // if there aren't any owners to start, this is also pointless if (!Array.isArray(meta.pending_owners)) { throw new Error("METADATA_NONSENSE_PENDING_OWNERS"); } var changed = false; // remove owners one by one // we assume there are no duplicates args.forEach(function (owner) { var index = meta.pending_owners.indexOf(owner); if (index < 0) { return; } meta.pending_owners.splice(index, 1); changed = true; }); return changed; }; // ["RESET_OWNERS", ["7eEqelGso3EBr5jHlei6av4r9w2B9XZiGGwA1EgZ-5I="], 1561623439989] commands.RESET_OWNERS = function (meta, args) { // expect a new array, even if it's empty if (!Array.isArray(args)) { throw new Error('METADATA_INVALID_OWNERS'); } // assume there are owners to start if (!Array.isArray(meta.owners)) { throw new Error("METADATA_NONSENSE_OWNERS"); } // overwrite the existing owners with the new one meta.owners = deduplicate(args.filter(isValidOwner)); if (Array.isArray(meta.allowed)) { // make sure owners are not included in the allow list filterInPlace(meta.allowed, function (member) { return meta.owners.indexOf(member) !== -1; }); } if (meta.owners.length === 0 && meta.restricted) { meta.restricted = false; } return true; }; // ["ADD_MAILBOX", {"7eEqelGso3EBr5jHlei6av4r9w2B9XZiGGwA1EgZ-5I=": mailbox, ...}, 1561623439989] commands.ADD_MAILBOX = function (meta, args) { // expect a new array, even if it's empty if (!args || typeof(args) !== "object") { throw new Error('METADATA_INVALID_MAILBOX'); } // assume there are owners to start if (!Array.isArray(meta.owners)) { throw new Error("METADATA_NONSENSE_OWNERS"); } var changed = false; // For each mailbox we try to add, check if the associated edPublic is an owner // If they are, add or replace the mailbox Object.keys(args).forEach(function (edPublic) { if (meta.owners.indexOf(edPublic) === -1) { return; } if (typeof(meta.mailbox) === "string") { var str = meta.mailbox; meta.mailbox = {}; meta.mailbox[meta.owners[0]] = str; } // Make sure mailbox is defined if (!meta.mailbox) { meta.mailbox = {}; } meta.mailbox[edPublic] = args[edPublic]; changed = true; }); return changed; }; commands.RM_MAILBOX = function (meta, args) { if (!Array.isArray(args)) { throw new Error("INVALID_ARGS"); } if (!meta.mailbox || typeof(meta.mailbox) === 'undefined') { return false; } if (typeof(meta.mailbox) === 'string' && args.length === 0) { delete meta.mailbox; return true; } var changed = false; args.forEach(function (arg) { if (meta.mailbox[arg] === 'undefined') { return; } delete meta.mailbox[arg]; changed = true; }); return changed; }; commands.UPDATE_EXPIRATION = function () { throw new Error("E_NOT_IMPLEMENTED"); }; var handleCommand = Meta.handleCommand = function (meta, line) { var command = line[0]; var args = line[1]; //var time = line[2]; if (typeof(commands[command]) !== 'function') { throw new Error("METADATA_UNSUPPORTED_COMMAND"); } return commands[command](meta, args); }; Meta.commands = Object.keys(commands); Meta.createLineHandler = function (ref, errorHandler) { ref.meta = {}; ref.index = 0; ref.logged = {}; return function (err, line) { if (err) { // it's not abnormal that metadata exists without a corresponding log // so ENOENT is fine if (ref.index === 0 && err.code === 'ENOENT') { return; } // any other errors are abnormal return void errorHandler('METADATA_HANDLER_LINE_ERR', { error: err, index: ref.index, line: JSON.stringify(line), }); } // the case above is special, everything else should increment the index var index = ref.index++; if (typeof(line) === 'undefined') { return; } if (Array.isArray(line)) { try { handleCommand(ref.meta, line); } catch (err2) { var code = err2.message; if (ref.logged[code]) { return; } ref.logged[code] = true; errorHandler("METADATA_COMMAND_ERR", { error: err2.stack, line: line, }); } return; } // the first line of a channel is processed before the dedicated metadata log. // it can contain a map, in which case it should be used as the initial state. // it's possible that a trim-history command was interrupted, in which case // this first message might exist in parallel with the more recent metadata log // which will contain the computed state of the previous metadata log // which has since been archived. // Thus, accept both the first and second lines you process as valid initial state // preferring the second if it exists if (index < 2 && line && typeof(line) === 'object') { // special case! ref.meta = line; return; } errorHandler("METADATA_HANDLER_WEIRDLINE", { line: line, index: index, }); }; };