2023-01-03 15:34:57 -05:00
|
|
|
let encForm = new Form({label: "Encryption"});
|
2023-01-02 14:05:53 -05:00
|
|
|
|
|
|
|
let encMsg = encForm.createTextArea({label: "Message"});
|
2023-01-03 20:44:47 -05:00
|
|
|
let encPass = encForm.createPasswordInput({
|
|
|
|
label: "Password",
|
|
|
|
enabledFunc: function() {return !encManualKey.value}
|
2023-01-03 15:34:57 -05:00
|
|
|
});
|
2023-01-03 20:44:47 -05:00
|
|
|
let encSalt = encForm.createMediumTextBox({
|
2023-01-03 15:34:57 -05:00
|
|
|
label: "PBKDF2 salt",
|
|
|
|
dataType: "b64",
|
|
|
|
advanced: true,
|
2023-01-03 20:44:47 -05:00
|
|
|
enabled: false,
|
|
|
|
enabledFunc: function() {return encManualSalt.value && !encManualKey.value}
|
|
|
|
});
|
|
|
|
let encManualSalt = encForm.createCheckBox({
|
|
|
|
label: "Use fixed salt instead of random",
|
|
|
|
advanced: true
|
|
|
|
});
|
|
|
|
let encKey = encForm.createMediumTextBox({
|
|
|
|
label: "Key",
|
|
|
|
dataType: "b64",
|
|
|
|
advanced: true,
|
|
|
|
enabled: false,
|
|
|
|
enabledFunc: function() {return encManualKey.value}
|
2023-01-03 15:34:57 -05:00
|
|
|
});
|
2023-01-03 20:44:47 -05:00
|
|
|
let encManualKey = encForm.createCheckBox({
|
|
|
|
label: "Use fixed key instead of password",
|
|
|
|
advanced: true
|
|
|
|
});
|
|
|
|
let encIV = encForm.createMediumTextBox({
|
2023-01-03 15:34:57 -05:00
|
|
|
label: "IV",
|
|
|
|
dataType: "b64",
|
|
|
|
advanced: true,
|
2023-01-03 20:44:47 -05:00
|
|
|
enabledFunc: function() {return encManualIV.value}
|
|
|
|
});
|
|
|
|
let encManualIV = encForm.createCheckBox({
|
|
|
|
label: "Use fixed IV instead of random",
|
|
|
|
advanced: true
|
2023-01-03 15:34:57 -05:00
|
|
|
});
|
2023-01-02 14:05:53 -05:00
|
|
|
let encButton = encForm.createButton({label: "Encrypt"});
|
|
|
|
let encOut = encForm.createOutput({
|
2022-12-31 22:06:13 -05:00
|
|
|
label: "Output",
|
|
|
|
dataType: "json-b64",
|
|
|
|
});
|
2023-01-03 15:34:57 -05:00
|
|
|
let encOutRaw = encForm.createOutput({
|
|
|
|
label: "Raw ciphertext",
|
|
|
|
dataType: "b64",
|
|
|
|
advanced: true
|
|
|
|
});
|
2022-12-31 22:06:13 -05:00
|
|
|
|
2023-01-03 15:34:57 -05:00
|
|
|
let decForm = new Form({label: "Decryption"});
|
2023-01-02 14:05:53 -05:00
|
|
|
|
|
|
|
let decMsg = decForm.createTextArea({
|
2022-12-31 22:06:13 -05:00
|
|
|
label: "Encrypted message",
|
|
|
|
dataType: "json-b64",
|
|
|
|
});
|
2023-01-03 21:50:27 -05:00
|
|
|
let decPass = decForm.createPasswordInput({
|
|
|
|
label: "Password",
|
|
|
|
enabledFunc: function() {return !decManualKey.value}
|
|
|
|
});
|
|
|
|
let decKey = decForm.createMediumTextBox({
|
|
|
|
label: "Key",
|
|
|
|
dataType: "b64",
|
|
|
|
advanced: true,
|
|
|
|
enabled: false,
|
|
|
|
enabledFunc: function() {return decManualKey.value}
|
|
|
|
});
|
|
|
|
let decManualKey = decForm.createCheckBox({
|
|
|
|
label: "Use fixed key instead of password",
|
|
|
|
advanced: true
|
|
|
|
});
|
2023-01-02 14:05:53 -05:00
|
|
|
let decButton = decForm.createButton({label: "Decrypt"});
|
|
|
|
let decOut = decForm.createOutput({label: "Output"});
|
|
|
|
|
2022-12-31 22:06:13 -05:00
|
|
|
function getKeyMaterial(password) {
|
|
|
|
let enc = new TextEncoder();
|
|
|
|
return window.crypto.subtle.importKey(
|
|
|
|
"raw",
|
|
|
|
enc.encode(password),
|
|
|
|
"PBKDF2",
|
|
|
|
false,
|
|
|
|
["deriveKey"]
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
function getKey(keyMaterial, salt) {
|
|
|
|
return window.crypto.subtle.deriveKey(
|
|
|
|
{
|
|
|
|
"name": "PBKDF2",
|
|
|
|
"hash": "SHA-256",
|
|
|
|
"salt": salt,
|
|
|
|
"iterations": 300000
|
|
|
|
},
|
|
|
|
keyMaterial,
|
|
|
|
{
|
|
|
|
"name": "AES-GCM",
|
|
|
|
"length": 256
|
|
|
|
},
|
|
|
|
true,
|
|
|
|
["encrypt", "decrypt"]
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2023-01-03 20:44:47 -05:00
|
|
|
encButton.handle.addEventListener("click", async function() {
|
2022-12-31 22:06:13 -05:00
|
|
|
let keyMaterial = await getKeyMaterial(encPass.value);
|
2023-01-03 20:44:47 -05:00
|
|
|
let key;
|
2023-01-03 21:50:27 -05:00
|
|
|
let salt = encSalt.value;
|
|
|
|
|
2023-01-03 20:44:47 -05:00
|
|
|
if (encManualKey.value) {
|
|
|
|
key = await window.crypto.subtle.importKey(
|
|
|
|
"raw",
|
|
|
|
encKey.value,
|
|
|
|
{"name": "AES-GCM"},
|
|
|
|
true,
|
|
|
|
["encrypt", "decrypt"]
|
|
|
|
);
|
|
|
|
} else {
|
2023-01-03 21:09:00 -05:00
|
|
|
if (encSalt.enabledFunc()) {
|
|
|
|
salt = encSalt.value;
|
|
|
|
} else {
|
|
|
|
salt = window.crypto.getRandomValues(new Uint8Array(16));
|
|
|
|
encSalt.value = salt;
|
|
|
|
}
|
|
|
|
|
2023-01-03 20:44:47 -05:00
|
|
|
key = await getKey(keyMaterial, salt);
|
|
|
|
encKey.value = await window.crypto.subtle.exportKey("raw", key);
|
|
|
|
}
|
2022-12-31 22:06:13 -05:00
|
|
|
|
2023-01-03 20:44:47 -05:00
|
|
|
let iv;
|
|
|
|
if (encManualIV.value) {
|
|
|
|
iv = encIV.value;
|
|
|
|
} else {
|
|
|
|
iv = window.crypto.getRandomValues(new Uint8Array(16));
|
|
|
|
encIV.value = iv;
|
|
|
|
}
|
2022-12-31 22:06:13 -05:00
|
|
|
|
|
|
|
let enc = new TextEncoder();
|
|
|
|
let msgEncoded = enc.encode(encMsg.value);
|
|
|
|
|
|
|
|
let ciphertext = await window.crypto.subtle.encrypt(
|
|
|
|
{
|
|
|
|
"name": "AES-GCM",
|
|
|
|
"iv": iv
|
|
|
|
},
|
|
|
|
key,
|
|
|
|
msgEncoded
|
|
|
|
);
|
|
|
|
|
2023-01-03 15:34:57 -05:00
|
|
|
encOutRaw.value = ciphertext;
|
|
|
|
|
2022-12-31 22:06:13 -05:00
|
|
|
encOut.value = {
|
|
|
|
"ciphertext": bufToB64(ciphertext),
|
|
|
|
"salt": bufToB64(salt),
|
|
|
|
"iv": bufToB64(iv)
|
|
|
|
}
|
2023-01-03 20:44:47 -05:00
|
|
|
});
|
2022-12-31 22:06:13 -05:00
|
|
|
|
2023-01-03 20:44:47 -05:00
|
|
|
decButton.handle.addEventListener("click", async function() {
|
2022-12-31 22:06:13 -05:00
|
|
|
let msgEncoded = decMsg.value;
|
|
|
|
|
2023-01-02 15:01:19 -05:00
|
|
|
let ciphertext, iv, salt;
|
|
|
|
try {
|
|
|
|
ciphertext = new b64ToBuf(msgEncoded.ciphertext);
|
|
|
|
iv = new Uint8Array(b64ToBuf(msgEncoded.iv));
|
|
|
|
salt = new Uint8Array(b64ToBuf(msgEncoded.salt));
|
|
|
|
} catch (e) {
|
|
|
|
decMsg.alertBox("alert-error", "Invalid base64 value.");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ciphertext === undefined || iv === undefined || salt === undefined) {
|
|
|
|
return;
|
|
|
|
}
|
2022-12-31 22:06:13 -05:00
|
|
|
|
|
|
|
let keyMaterial = await getKeyMaterial(decPass.value);
|
2023-01-03 21:50:27 -05:00
|
|
|
let key;
|
|
|
|
if (decManualKey.value) {
|
|
|
|
key = await window.crypto.subtle.importKey(
|
|
|
|
"raw",
|
|
|
|
decKey.value,
|
|
|
|
{"name": "AES-GCM"},
|
|
|
|
true,
|
|
|
|
["encrypt", "decrypt"]
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
key = await getKey(keyMaterial, salt);
|
|
|
|
}
|
2022-12-31 22:06:13 -05:00
|
|
|
|
|
|
|
let plaintext;
|
|
|
|
|
|
|
|
try {
|
|
|
|
plaintext = await window.crypto.subtle.decrypt(
|
|
|
|
{
|
|
|
|
"name": "AES-GCM",
|
|
|
|
"iv": iv
|
|
|
|
},
|
|
|
|
key,
|
|
|
|
ciphertext
|
|
|
|
);
|
|
|
|
} catch (e) {
|
2022-12-31 22:12:42 -05:00
|
|
|
decPass.alertBox("alert-error", "Decryption error: incorrect password?");
|
2022-12-31 22:06:13 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
let dec = new TextDecoder();
|
|
|
|
decOut.value = `${dec.decode(plaintext)}`;
|
2023-01-03 20:44:47 -05:00
|
|
|
});
|