keeweb/app/scripts/mixins/protected-value-ex.js

141 lines
3.7 KiB
JavaScript

const kdbxweb = require('kdbxweb');
const ExpectedFieldRefChars = '{REF:0@I:00000000000000000000000000000000}'.split('');
const ExpectedFieldRefByteLength = ExpectedFieldRefChars.length;
kdbxweb.ProtectedValue.prototype.isProtected = true;
kdbxweb.ProtectedValue.prototype.forEachChar = function(fn) {
const value = this._value;
const salt = this._salt;
let b, b1, b2, b3;
for (let i = 0, len = value.length; i < len; i++) {
b = value[i] ^ salt[i];
if (b < 128) {
if (fn(b) === false) {
return;
}
continue;
}
i++;
b1 = value[i] ^ salt[i];
if (i === len) {
break;
}
if (b >= 192 && b < 224) {
if (fn(((b & 0x1f) << 6) | (b1 & 0x3f)) === false) {
return;
}
continue;
}
i++;
b2 = value[i] ^ salt[i];
if (i === len) {
break;
}
if (b >= 224 && b < 240) {
if (fn(((b & 0xf) << 12) | ((b1 & 0x3f) << 6) | (b2 & 0x3f)) === false) {
return;
}
}
i++;
b3 = value[i] ^ salt[i];
if (i === len) {
break;
}
if (b >= 240 && b < 248) {
let c = ((b & 7) << 18) | ((b1 & 0x3f) << 12) | ((b2 & 0x3f) << 6) | (b3 & 0x3f);
if (c <= 0xffff) {
if (fn(c) === false) {
return;
}
} else {
c ^= 0x10000;
if (fn(0xd800 | (c >> 10)) === false) {
return;
}
if (fn(0xdc00 | (c & 0x3ff)) === false) {
return;
}
}
}
// skip error
}
};
Object.defineProperty(kdbxweb.ProtectedValue.prototype, 'textLength', {
get: function() {
let textLength = 0;
this.forEachChar(() => {
textLength++;
});
return textLength;
}
});
kdbxweb.ProtectedValue.prototype.includesLower = function(findLower) {
let matches = false;
const foundSeqs = [];
const len = findLower.length;
this.forEachChar(ch => {
ch = String.fromCharCode(ch).toLowerCase();
if (matches) {
return;
}
for (let i = 0; i < foundSeqs.length; i++) {
const seqIx = ++foundSeqs[i];
if (findLower[seqIx] !== ch) {
foundSeqs.splice(i, 1);
i--;
continue;
}
if (seqIx === len - 1) {
matches = true;
return;
}
}
if (findLower[0] === ch) {
foundSeqs.push(0);
}
});
return matches;
};
kdbxweb.ProtectedValue.prototype.equals = function(other) {
if (!other) {
return false;
}
if (!other.isProtected) {
return this.textLength === other.length && this.includes(other);
}
if (other === this) {
return true;
}
const len = this.byteLength;
if (len !== other.byteLength) {
return false;
}
for (let i = 0; i < len; i++) {
if ((this._value[i] ^ this._salt[i]) !== (other._value[i] ^ other._salt[i])) {
return false;
}
}
return true;
};
kdbxweb.ProtectedValue.prototype.isFieldReference = function() {
if (this.byteLength !== ExpectedFieldRefByteLength) {
return false;
}
let ix = 0;
this.forEachChar(ch => {
const expected = ExpectedFieldRefChars[ix++];
if (expected !== '0' && ch !== expected) {
return false;
}
});
return true;
};
module.exports = kdbxweb.ProtectedValue;