Skip to content

Commit

Permalink
[add] Webcrypto example misc/aes_gcm
Browse files Browse the repository at this point in the history
  • Loading branch information
xeioex committed Jul 21, 2022
1 parent 7e65def commit 38cce8a
Show file tree
Hide file tree
Showing 3 changed files with 157 additions and 0 deletions.
91 changes: 91 additions & 0 deletions README.rst
Original file line number Diff line number Diff line change
Expand Up @@ -1545,6 +1545,97 @@ example.js:
curl http://localhost/read
200 <empty reply>
Webcrypto (AES-GSM) [misc/aes_gsm]
----------------------------------

nginx.conf:

.. code-block:: nginx
http {
js_path "/etc/nginx/njs/";
js_import main from misc/aes_gsm.js;
server {
listen 80;
location /encrypt {
js_content main.encrypt;
}
location /decrypt {
js_content main.decrypt;
}
}
}
example.js:

.. code-block:: js
async function encryptUAM(key_in, iv, text) {
const alg = { name: 'AES-GCM', iv: iv ? Buffer.from(iv, 'hex')
: crypto.getRandomValues(new Uint8Array(12)) };
const sha256 = await crypto.subtle.digest('SHA-256', new TextEncoder().encode(key_in));
const key = await crypto.subtle.importKey('raw', sha256, alg, false, ['encrypt']);
const cipher = await crypto.subtle.encrypt(alg, key, new TextEncoder().encode(text));
return JSON.stringify({
cipher: btoa(String.fromCharCode.apply(null, new Uint8Array(cipher))),
iv: btoa(String.fromCharCode.apply(null, new Uint8Array(alg.iv))),
});
}
async function decryptUAM(key_in, value) {
value = JSON.parse(value);
ngx.log(ngx.ERR, njs.dump(value))
const alg = { name: 'AES-GCM', iv: Buffer.from(value.iv, 'base64') };
const sha256 = await crypto.subtle.digest('SHA-256', new TextEncoder().encode(key_in));
const key = await crypto.subtle.importKey('raw', sha256, alg, false, ['decrypt']);
const decrypt = await crypto.subtle.decrypt(alg, key, Buffer.from(value.cipher, 'base64'));
ngx.log(ngx.ERR, njs.dump(new Uint8Array(decrypt)))
return new TextDecoder().decode(decrypt);
}
async function encrypt(r) {
try {
let encrypted = await encryptUAM(r.args.key, r.args.iv, r.requestText);
r.return(200, encrypted);
} catch (e) {
r.return(500, `encryption failed with ${e.message}`);
}
}
async function decrypt(r) {
try {
let decrypted = await decryptUAM(r.args.key, r.requestText);
r.return(200, decrypted);
} catch (e) {
r.return(500, `decryption failed with ${e.message}`);
}
}
export default {encrypt, decrypt};
.. code-block:: shell
curl 'http://localhost/encrypt?key=mySecret&iv=000000000000000000000001' -d TEXT-TO-BE-ENCODED
{"cipher":"kLKXeb/h1inwXYlP7M504xCD+/1sF4yesCSUc7/OJiyPyw==","iv":"AAAAAAAAAAAAAAAB"}
curl 'http://localhost/decrypt?key=mySecret' -d '{"cipher":"kLKXeb/h1inwXYlP7M504xCD+/1sF4yesCSUc7/OJiyPyw==","iv":"AAAAAAAAAAAAAAAA"}'
decryption failed with EVP_DecryptFinal_ex() failed
curl 'http://localhost/decrypt?key=mySecre' -d '{"cipher":"kLKXeb/h1inwXYlP7M504xCD+/1sF4yesCSUc7/OJiyPyw==","iv":"AAAAAAAAAAAAAAAB"}'
decryption failed with EVP_DecryptFinal_ex() failed
curl 'http://localhost/decrypt?key=mySecret' -d '{"cipher":"kLKXeb/h1inwXYlP7M504xCD+/1sF4yesCSUc7/OJiyPyw==","iv":"AAAAAAAAAAAAAAAB"}'
TEXT-TO-BE-ENCODED
Command line interface
======================

Expand Down
21 changes: 21 additions & 0 deletions conf/misc/aes_gcm.conf
Original file line number Diff line number Diff line change
@@ -0,0 +1,21 @@
load_module modules/ngx_http_js_module.so;

events { }

http {
js_path "/etc/nginx/njs/";

js_import main from misc/aes_gcm.js;

server {
listen 80;

location /encrypt {
js_content main.encrypt;
}

location /decrypt {
js_content main.decrypt;
}
}
}
45 changes: 45 additions & 0 deletions njs/misc/aes_gcm.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,45 @@
async function encryptUAM(key_in, iv, text) {
const alg = { name: 'AES-GCM', iv: iv ? Buffer.from(iv, 'hex')
: crypto.getRandomValues(new Uint8Array(12)) };

const sha256 = await crypto.subtle.digest('SHA-256', new TextEncoder().encode(key_in));
const key = await crypto.subtle.importKey('raw', sha256, alg, false, ['encrypt']);

const cipher = await crypto.subtle.encrypt(alg, key, new TextEncoder().encode(text));

return JSON.stringify({
cipher: btoa(String.fromCharCode.apply(null, new Uint8Array(cipher))),
iv: btoa(String.fromCharCode.apply(null, new Uint8Array(alg.iv))),
});
}

async function decryptUAM(key_in, value) {
value = JSON.parse(value);

const alg = { name: 'AES-GCM', iv: Buffer.from(value.iv, 'base64') };
const sha256 = await crypto.subtle.digest('SHA-256', new TextEncoder().encode(key_in));
const key = await crypto.subtle.importKey('raw', sha256, alg, false, ['decrypt']);

const decrypt = await crypto.subtle.decrypt(alg, key, Buffer.from(value.cipher, 'base64'));
return new TextDecoder().decode(decrypt);
}

async function encrypt(r) {
try {
let encrypted = await encryptUAM(r.args.key, r.args.iv, r.requestText);
r.return(200, encrypted);
} catch (e) {
r.return(500, `encryption failed with ${e.message}`);
}
}

async function decrypt(r) {
try {
let decrypted = await decryptUAM(r.args.key, r.requestText);
r.return(200, decrypted);
} catch (e) {
r.return(500, `decryption failed with ${e.message}`);
}
}

export default {encrypt, decrypt};

0 comments on commit 38cce8a

Please sign in to comment.