"use strict"; /* * Hashing and comparison utility functions for the RSpade framework. * These functions handle object hashing and deep comparison. */ // ============================================================================ // HASHING AND COMPARISON // ============================================================================ /** * Generates a unique hash for any value (handles objects, arrays, circular references) * @param {*} the_var - Value to hash * @param {boolean} [calc_sha1=true] - If true, returns SHA1 hash; if false, returns JSON * @param {Array} [ignored_keys=null] - Keys to ignore when hashing objects * @returns {string} SHA1 hash or JSON string of the value */ function hash(the_var) { let calc_sha1 = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : true; let ignored_keys = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : null; if (typeof the_var == undef) { the_var = '__undefined__'; } if (ignored_keys === null) { ignored_keys = ['$']; } // Converts value to json, discarding circular references let json_stringify_nocirc = function (value) { const cache = []; return JSON.stringify(value, function (key, v) { if (typeof v === 'object' && typeof the_var._cache_key == 'function') { return the_var._hash_key(); } else if (typeof v === 'object' && v !== null) { if (cache.indexOf(v) !== -1) { // Duplicate reference found, discard key return; } cache.push(v); } return v; }); }; // Turn every property and all its children into a single depth array of values that we can then // sort and hash as a whole let flat_var = {}; let _flatten = function (the_var, prefix) { let depth = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : 0; // If a class object is provided, circular references can make the call stack recursive. // For the purposes of how the hash function is called, this should be sufficient. if (depth > 10) { return; } // Does not account for dates i think... if (is_object(the_var) && typeof the_var._cache_key == 'function') { // Use _cache_key to hash components flat_var[prefix] = the_var._hash_key(); } else if (is_object(the_var) && typeof Abstract !== 'undefined' && the_var instanceof Abstract) { // Stringify all class objects flat_var[prefix] = json_stringify_nocirc(the_var); } else if (is_object(the_var)) { // Iterate other objects flat_var[prefix] = {}; for (let k in the_var) { if (the_var.hasOwnProperty(k) && ignored_keys.indexOf(k) == -1) { _flatten(the_var[k], prefix + '..' + k, depth + 1); } } } else if (is_array(the_var)) { // Iterate arrays flat_var[prefix] = []; let i = 0; foreach(the_var, v => { _flatten(v, prefix + '..' + i, depth + 1); i++; }); } else if (is_function(the_var)) { // nothing } else if (!is_numeric(the_var)) { flat_var[prefix] = String(the_var); } else { flat_var[prefix] = the_var; } }; _flatten(the_var, '_'); let sorter = []; foreach(flat_var, function (v, k) { sorter.push([k, v]); }); sorter.sort(function (a, b) { return a[0] > b[0]; }); let json = JSON.stringify(sorter); if (calc_sha1) { let hashed = sha1.sha1(json); return hashed; } else { return json; } } /** * Deep comparison of two values (ignores property order and functions) * @param {*} a - First value to compare * @param {*} b - Second value to compare * @returns {boolean} True if values are deeply equal */ function deep_equal(a, b) { return hash(a, false) == hash(b, false); } //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["hash","the_var","calc_sha1","arguments","length","undefined","ignored_keys","undef","json_stringify_nocirc","value","cache","JSON","stringify","key","v","_cache_key","_hash_key","indexOf","push","flat_var","_flatten","prefix","depth","is_object","Abstract","k","hasOwnProperty","is_array","i","foreach","is_function","is_numeric","String","sorter","sort","a","b","json","hashed","sha1","deep_equal"],"sources":["app/RSpade/Core/Js/hash.js"],"sourcesContent":["/*\n * Hashing and comparison utility functions for the RSpade framework.\n * These functions handle object hashing and deep comparison.\n */\n\n// ============================================================================\n// HASHING AND COMPARISON\n// ============================================================================\n\n/**\n * Generates a unique hash for any value (handles objects, arrays, circular references)\n * @param {*} the_var - Value to hash\n * @param {boolean} [calc_sha1=true] - If true, returns SHA1 hash; if false, returns JSON\n * @param {Array<string>} [ignored_keys=null] - Keys to ignore when hashing objects\n * @returns {string} SHA1 hash or JSON string of the value\n */\nfunction hash(the_var, calc_sha1 = true, ignored_keys = null) {\n    if (typeof the_var == undef) {\n        the_var = '__undefined__';\n    }\n\n    if (ignored_keys === null) {\n        ignored_keys = ['$'];\n    }\n\n    // Converts value to json, discarding circular references\n    let json_stringify_nocirc = function (value) {\n        const cache = [];\n        return JSON.stringify(value, function (key, v) {\n            if (typeof v === 'object' && typeof the_var._cache_key == 'function') {\n                return the_var._hash_key();\n            } else if (typeof v === 'object' && v !== null) {\n                if (cache.indexOf(v) !== -1) {\n                    // Duplicate reference found, discard key\n                    return;\n                }\n                cache.push(v);\n            }\n            return v;\n        });\n    };\n\n    // Turn every property and all its children into a single depth array of values that we can then\n    // sort and hash as a whole\n    let flat_var = {};\n    let _flatten = function (the_var, prefix, depth = 0) {\n        // If a class object is provided, circular references can make the call stack recursive.\n        // For the purposes of how the hash function is called, this should be sufficient.\n        if (depth > 10) {\n            return;\n        }\n\n        // Does not account for dates i think...\n\n        if (is_object(the_var) && typeof the_var._cache_key == 'function') {\n            // Use _cache_key to hash components\n            flat_var[prefix] = the_var._hash_key();\n        } else if (is_object(the_var) && typeof Abstract !== 'undefined' && the_var instanceof Abstract) {\n            // Stringify all class objects\n            flat_var[prefix] = json_stringify_nocirc(the_var);\n        } else if (is_object(the_var)) {\n            // Iterate other objects\n            flat_var[prefix] = {};\n            for (let k in the_var) {\n                if (the_var.hasOwnProperty(k) && ignored_keys.indexOf(k) == -1) {\n                    _flatten(the_var[k], prefix + '..' + k, depth + 1);\n                }\n            }\n        } else if (is_array(the_var)) {\n            // Iterate arrays\n            flat_var[prefix] = [];\n            let i = 0;\n            foreach(the_var, (v) => {\n                _flatten(v, prefix + '..' + i, depth + 1);\n                i++;\n            });\n        } else if (is_function(the_var)) {\n            // nothing\n        } else if (!is_numeric(the_var)) {\n            flat_var[prefix] = String(the_var);\n        } else {\n            flat_var[prefix] = the_var;\n        }\n    };\n\n    _flatten(the_var, '_');\n\n    let sorter = [];\n\n    foreach(flat_var, function (v, k) {\n        sorter.push([k, v]);\n    });\n\n    sorter.sort(function (a, b) {\n        return a[0] > b[0];\n    });\n\n    let json = JSON.stringify(sorter);\n\n    if (calc_sha1) {\n        let hashed = sha1.sha1(json);\n        return hashed;\n    } else {\n        return json;\n    }\n}\n\n/**\n * Deep comparison of two values (ignores property order and functions)\n * @param {*} a - First value to compare\n * @param {*} b - Second value to compare\n * @returns {boolean} True if values are deeply equal\n */\nfunction deep_equal(a, b) {\n    return hash(a, false) == hash(b, false);\n}"],"mappings":"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","ignoreList":[]}