encryptme/scripts/aes.js

206 lines
4.5 KiB
JavaScript
Raw Normal View History

2023-01-03 15:34:57 -05:00
let encForm = new Form({label: "Encryption"});
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
});
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"});
let decMsg = decForm.createTextArea({
2022-12-31 22:06:13 -05:00
label: "Encrypted message",
dataType: "json-b64",
});
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
});
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;
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 {
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);
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) {
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
});