March 11, 2025
AES
cryptography
article cover

Encryption is a fundamental technique used to protect data from unauthorized access

In this blog post, we will learn how to use CryptoJS, a powerful JavaScript library, to encrypt and decrypt data using the AES-256 encryption algorithm.

šŸ” Cryptography: The Science of Secure Communication

Cryptography is the practice of securing information so that only authorized parties can access it. It protects data from hackers, ensures privacy, and is used in banking, messaging apps, online payments, and digital signatures.

Types of Cryptography
  • Symmetric Encryption (Same key for encryption & decryption)
  • Asymmetric Encryption (Public key for encryption, private key for decryption)
  • Hashing (One-way conversion, no decryption)
  • What is AES Encryption?

    AES (Advanced Encryption Standard) is a secure encryption method used to protect data. It works by converting readable data into scrambled text using a secret key. The same key is used to decrypt the data back into its original form.

    šŸ”‘ Key Points:
  • Fast & Secure ā€“ Used worldwide for encryption.
  • Key Sizes ā€“ 128-bit, 192-bit, or 256-bit.
  • Common Uses ā€“ Secure messaging, banking, passwords, and file encryption.
  • A visual depiction of what is being written about

    Generating a Secure Secret Key for AES Encryption

    The first and most crucial step in AES encryption is generating a strong secret key. The following function, generateSecretKey, creates a random 256-bit key to ensure robust security.

    javascript
    const crypto = require("crypto");
    const genrateSecretKey = () => {
    const keyLength = 32; //// 32 bytes = 256 bits (AES-256)
    return crypto.randomBytes(keyLength).toString("hex");
    };
    const secretKey = genrateSecretKey();
    console.log("Generated Secret Key:",secretKey);

    āœ… 256-bit strength ensures high security. āœ… Randomized output prevents predictability. āœ… Hex encoding makes storage and sharing easy.

    Sample Output (Varies Each Time)
    powershell
    Generated Secret Key: e9f3a8b62d4c89f7b10c3e9d58723a1f7d2b6a9c5e1f4d8c7b2e0f6a9c3d2b1e

    Each time you run the script, you'll get a different 64-character hex string (because 32 bytes = 64 hex characters).

    šŸ” Deep Explanation of AES-256-CBC Encryption in Node.js

    Your code securely encrypts an array of objects using AES-256-CBC encryption in Node.js. Let's break it down step by step.


    javascript
    const crypto = require("crypto"); // Import Node.js crypto module
    // šŸ›  Generate a secure 256-bit secret key (32 bytes = 256 bits)
    // The key is represented in hexadecimal (binary-to-text conversion for storage)
    const generateSecretKey = () => crypto.randomBytes(32).toString("hex");
    const secretKey = generateSecretKey(); // Generate a new random secret key
    console.log("šŸ”‘ Generated Secret Key:", secretKey);
    // šŸ“Œ Sample user data to encrypt
    const data = [
    { _id: "jhon", username: "DevX@123", password: "1200012" },
    { _id: "jhon2", username: "DevX@1232", password: "12000122" },
    ];
    /**
    * šŸ” AES-256-CBC Encryption Function
    *
    * @param {Object} data - The data to encrypt (JSON object)
    * @param {string} secretKey - The secret key (256-bit hex string)
    * @returns {string} - The encrypted data as a hex-encoded string (IV + CipherText)
    */
    const encryptData = (data, secretKey) => {
    const algorithm = "aes-256-cbc"; // AES algorithm with 256-bit key in CBC mode
    const iv = crypto.randomBytes(16); // 16-byte IV (random for each encryption)
    // šŸ—ļø Create AES cipher with secret key & IV
    const cipher = crypto.createCipheriv(algorithm, Buffer.from(secretKey, "hex"), iv);
    // šŸ“ Encrypt JSON Data (Convert UTF-8 text to hex format)
    let encrypted = cipher.update(JSON.stringify(data), "utf-8", "hex");
    encrypted += cipher.final("hex"); // Finalize encryption process
    // šŸ·ļø Return the IV (for decryption) + Encrypted Data (Hex encoded)
    return iv.toString("hex") + ":" + encrypted;
    };
    // šŸ”’ Encrypt the user data
    const encryptedData = encryptData(data, secretKey);
    console.log("šŸ›”ļø Encrypted Data:", encryptedData);

    javascript
    šŸ”‘ Generated Secret Key: 692d033ed7e582cb76704d6b02e5eb28411a00f7449bfda03a56103b1c0eb90f
    šŸ›”ļø Encrypted Data: 8704dc1e2da674be89589cc419b11855:603c3f867e34cd0afc94af9d7732527cc0d571179cec5a5 3c03758e1c770b25f664bf5ea57edd9ca337101baaac418fe2b29852e11b64f8b8faf4f94037238a360ba729b7
    fa1176a08eb8f9ea6eb201fe4f7149b61c054dafdf5742f967480005ffee35a85d36034a5b9f03fc0ad743597eb0042ef68cc6bc899717ed29da370

    šŸ›  Components of the Encryption Process

    The encryption function uses several key concepts:

  • AES-256-CBC Encryption
  • Hexadecimal Encoding (hex)
  • UTF-8 Encoding (utf-8)
  • Initialization Vector (IV)
  • Cipher Block Chaining (CBC) Mode
  • Secret Key (256-bit)

  • 1ļøāƒ£ AES-256-CBC Encryption

    AES (Advanced Encryption Standard) is a symmetric encryption algorithm that encrypts data securely.

  • AES-256: Uses a 256-bit key (32 bytes) for encryption, making it very strong.
  • CBC Mode (Cipher Block Chaining) is a way of encrypting data where each block of plaintext is linked to the previous one. This makes the encryption stronger because changes in one block affect all the following blocks. However, it requires an Initialization Vector (IV) to start the process securely.
  • šŸ“Œ Why AES-256-CBC?

    āœ… Highly secure (used by governments & banks).

    āœ… Requires a secret key + IV, making brute-force attacks difficult.

    āœ… Ensures randomness with IV, preventing pattern attacks.


    2ļøāƒ£ Understanding Encodings (utf-8 & hex)

    Your code uses UTF-8 and Hexadecimal (hex) encoding. Let's understand both.

    šŸ”¹ UTF-8 (Unicode Transformation Format - 8 bit)
  • Used for text encoding (e.g., English, Chinese, emojis, etc.).
  • Represents characters efficiently (1 to 4 bytes per character).
  • Used when converting JSON or text into a binary format.
  • šŸ”¹ Example:

    javascript
    const text = "Hello";
    const buffer = Buffer.from(text, "utf-8");
    console.log(buffer);

    šŸ“Œ Output:

    javascript
    <Buffer 48 65 6c 6c 6f>

    Each character (H, e, l, l, o) is represented as a byte in memory.


    šŸ”¹ Hexadecimal (hex) Encoding
  • Used for binary-to-text conversion.
  • Represents each byte as two hexadecimal characters.
  • Output is double the size of input (e.g., 32 bytes ā†’ 64 hex characters).
  • šŸ”¹ Example:

    javascript
    const hex = buffer.toString("hex");
    console.log(hex);

    šŸ“Œ Output:

    javascript
    48656c6c6f

    Each byte is converted into a 2-character hex code.

    šŸ“Œ Why use hex?

    āœ… Makes binary data human-readable.

    āœ… Prevents encoding issues (e.g., special characters breaking encryption).


    3ļøāƒ£ Initialization Vector (IV)

    šŸ”¹ What is IV?

    An Initialization Vector (IV) is a random 16-byte value used in encryption. It makes sure that even if you encrypt the same data multiple times, the output (ciphertext) is different each time. This adds extra security and prevents patterns from forming in the encrypted data.

    šŸ”¹ How does IV work?

  • First, AES XORs IV with the first block of data before encryption.
  • Each subsequent block is XORed with the previous encrypted block.
  • This makes encryption unique, even if the same data is encrypted multiple times.
  • šŸ“Œ Example:

    javascript
    const iv = crypto.randomBytes(16); // 16-byte IV
    console.log(iv.toString("hex"));

    Each run will generate a different IV


    šŸ”“ AES-256-CBC Decryption

    Now that you've encrypted your data, let's implement decryption to restore the original JSON.

    javascript
    const decryptData = (encryptedData, secretKey) => {
    const algorithm = "aes-256-cbc";
    // šŸ·ļø Split IV and encrypted text (IV is required for decryption)
    const [ivHex, encryptedText] = encryptedData.split(":");
    // šŸ—ļø Create AES decipher with the same secret key & IV
    const decipher = crypto.createDecipheriv(
    algorithm,
    Buffer.from(secretKey, "hex"),
    Buffer.from(ivHex, "hex")
    );
    // šŸ”„ Decrypt data (Convert hex back to UTF-8 text)
    let decrypted = decipher.update(encryptedText, "hex", "utf-8");
    decrypted += decipher.final("utf-8"); // Finalize decryption
    return JSON.parse(decrypted); // Convert string back to JSON
    };
    // šŸ”“ Decrypt the encrypted data
    const decryptedData = decryptData(encryptedData, secretKey);
    console.log("āœ… Decrypted Data:", decryptedData);

    šŸ“Œ Output

    powershell
    āœ… Decrypted Data: [
    { _id: 'jhon', username: 'DevX@123', password: '1200012' },
    { _id: 'jhon2', username: 'DevX@1232', password: '12000122' }
    ]

    šŸ” AES-256-CBC Encryption & Decryption (Final Code)

    javascript
    const crypto = require("crypto"); // Import Node.js crypto module
    // šŸ”‘ Generate a secure 256-bit secret key (32 bytes)
    const generateSecretKey = () => crypto.randomBytes(32).toString("hex");
    const secretKey = generateSecretKey(); // Generate a new random secret key
    console.log("šŸ”‘ Secret Key:", secretKey);
    // šŸ“Œ Sample user data to encrypt
    const data = [
    { _id: "john", username: "DevX@123", password: "1200012" },
    { _id: "john2", username: "DevX@1232", password: "12000122" },
    ];
    /**
    * šŸ” AES-256-CBC Encryption Function
    * @param {Object} data - Data to encrypt (JSON object)
    * @param {string} secretKey - 256-bit hex secret key
    * @returns {string} - IV + Encrypted data (hex encoded)
    */
    const encryptData = (data, secretKey) => {
    const algorithm = "aes-256-cbc"; // AES algorithm
    const iv = crypto.randomBytes(16); // 16-byte IV for randomness
    // šŸ—ļø Create AES cipher
    const cipher = crypto.createCipheriv(algorithm, Buffer.from(secretKey, "hex"), iv);
    // šŸ”„ Encrypt JSON data (Convert UTF-8 to hex)
    let encrypted = cipher.update(JSON.stringify(data), "utf-8", "hex");
    encrypted += cipher.final("hex");
    // šŸ·ļø Return IV + Encrypted Data (Both in hex format)
    return iv.toString("hex") + ":" + encrypted;
    };
    /**
    * šŸ”“ AES-256-CBC Decryption Function
    * @param {string} encryptedData - Encrypted data string (IV + CipherText)
    * @param {string} secretKey - 256-bit hex secret key
    * @returns {Object} - Decrypted original data (JSON object)
    */
    const decryptData = (encryptedData, secretKey) => {
    const algorithm = "aes-256-cbc";
    // šŸ·ļø Extract IV and encrypted text (IV is required for decryption)
    const [ivHex, encryptedText] = encryptedData.split(":");
    // šŸ—ļø Create AES decipher
    const decipher = crypto.createDecipheriv(
    algorithm,
    Buffer.from(secretKey, "hex"),
    Buffer.from(ivHex, "hex")
    );
    // šŸ”„ Decrypt data (Convert hex back to UTF-8)
    let decrypted = decipher.update(encryptedText, "hex", "utf-8");
    decrypted += decipher.final("utf-8");
    return JSON.parse(decrypted); // Convert string back to JSON
    };
    // šŸ”’ Encrypt the user data
    const encryptedData = encryptData(data, secretKey);
    console.log("šŸ›”ļø Encrypted Data:", encryptedData);
    // šŸ”“ Decrypt the encrypted data
    const decryptedData = decryptData(encryptedData, secretKey);
    console.log("āœ… Decrypted Data:", decryptedData);

    šŸ“Œ Output

    javascript
    šŸ”‘ Secret Key: 692d033ed7e582cb76704d6b02e5eb28411a00f7449bfda03a56103b1c0eb90f
    šŸ›”ļø Encrypted Data: 8704dc1e2da674be89589cc419b11855:603c3f867e34cd0afc94af9d7732527c...
    āœ… Decrypted Data: [
    { _id: "john", username: "DevX@123", password: "1200012" },
    { _id: "john2", username: "DevX@1232", password: "12000122" }
    ]

    šŸ› ļø Explanation
  • šŸ”‘ Secret Key Generation
  • plain text
    1b2d4a5f9c44a6b3e3f7a15d78c1e9b2e9b8c4d7a8a2f7e9d4b2a8c3d1f0e9c
  • šŸ” Encryption Process
  • šŸ”“ Decryption Process
  • šŸš€ Best Practices

    āœ… Use a unique IV for every encryption (never reuse IVs).

    āœ… Store the secret key securely (Environment Variables, Vaults, etc.).

    āœ… IV is NOT secret (It must be included with ciphertext).

    āœ… CBC mode prevents identical plaintext from having identical ciphertext (more secure than ECB)

    šŸ™ Thank You!

    Thank you for taking the time to read this blog! šŸŽ‰ I hope you found it useful in understanding AES encryption and decryption in Node.js.

    šŸš€ Stay secure, keep coding, and happy encrypting! šŸ”

    ;