Skip to content

How to store sensitive data in your database with demande access from user

Posted on:March 12, 2023 at 03:22 PM

Table of contents

Open Table of contents

Introduction

During the development of my latest project, I stumbled upon a problem that seemed to me like it needed a very specific solution. The TLDR of the problem was that I needed to store some data that was specific to a user, this data needed to be encrypted in a way that makes it inaccessible to me as a developer or by anyone on the team. On the other hand, the data needed to be decrypted by the backend to be used to communicate with other services but the decryption needed to happen by demand of the user with a decryption key that only the user knows. The other important criterion was to allow the user to change his key with minimum friction. This implementation needed to be generalised to be used across the app and not be specific to a certain data type. does that ring a bell?

General idea

Encryption procedure

  1. Generate encryption keys
  2. Encrypt the user data
  3. Recover the user’s secret decryption key as an input
  4. Encrypt the previously generated encryption keys using the user’s secret
  5. Store the encrypted encryption keys in the database

Decription procedure

  1. Recover the user’s secret decryption key as input
  2. Recover the encrypted encryption key from the database
  3. Decrypt the encrypted encryption key using the user’s secret key
  4. Decrypt the sensitive information using the decrypted encryption key
  5. Use the decrypted data as needed.

Changing the user secret decryption key

  1. Recover the OLD and the NEW decryption secret user key as input
  2. Recover the encrypted encryption key from the database
  3. Decrypt the encrypted encryption key using the OLD user secret
  4. Encrypt the decrypted encryption key using the NEW user secret
  5. Store the newly encrypted encryption keys in the database for future use

Diving deeper

Storing the user data

storing-the-sensitive-data

In this scenario, the backend service will receive the user’s sensitive data as well as the encryption secret key as input from the user. The backend will use two different encryption algorithms. The first one is based on a key derivation algorithm such as PBKDF2 to encrypt the encryption keys using keys that were derived from the user secret. The second algorithm is an AES-based encryption that uses a decryption key and initiation vector pair to encrypt the user data. This pair will be later encrypted using the derived keys to be stored in the database in a encrypted format.

Using the user data

using-the-user-data.

In this scenario, the backend service with receive a request from a user to a certain service that requires on-demand access to the user’s sensitive data. The user needs to provide his secret as part of the request. The service will retrieve the encrypted decryption keys from the database and proceed to decrypt them using the user’s secret key. The final part is to retrieve the encrypted data from the database and decrypt them using the freshly decrypted encryption keys.

Changing the user secret key

changing-the-user-secret.

The other scenario that needs to be handled is changing the user’s secret key. This procedure needs to have minimum friction. The user needs to provide the OLD secret key and a NEW secret key to be set. Initially, the backend retrieves the encrypted encryption keys from the database and proceeds to decrypt them using the OLD secret. The next step is to encrypt the freshly decrypted data and finally store the results in the database. This method has minimum friction with all the other user’s data. The previously encrypted user data will not be affected nor decrypted while changing the secret key.

Code example

The code example below should be treated as pseudo-code. The purpose of this code is to represent the steps that need to be followed to achieve this kind of solution and not a copy-past functioning solution.

Storing the user data code example

// get the user data
const { userSecretData, userSecret } = req.body;

// Hash the activation key to store in database
const salt = await bcrypt.genSalt();
const hashedActivationKey = await bcrypt.hash(userSecret, salt);

// Generate securityKey and init vector for node
// informations encryption
const [Securitykey, initVector] = generateEncryptionKeyAndIv();

// Encrypt the userSecretData using the activation key
const encryptedSecretData = encryptUsingEncryptionKey(
  userSecret,
  Securitykey,
  initVector
);

// store the secretData in the database
const node = await storeEncryptedSecretData(encryptedSecretData);

const encryptedEncryptionKey = encryptUsingPassword(
  encryptedSecretData,
  userSecret
);

const encryptedIV = encryptUsingPassword(initVector, userSecret);

await storeEncryptedDecriptionKeys(encryptedEncryptionKey, encryptedIV);

Using the secret user data code example

// get the encrypted secret data
const encryptedSecretData = getEncryptedSecretDataFromDatabase();

// get the encrypted encryption keys
const encryptedEncryptionKeys = await getNodeEncryptionKeys(node.id);

// get the user secret from the request body
const userSecret = req.userSecret;

const { encryption_key, init_vector } = encryptedEncryptionKeys;

// decrypt the encryption keys using the user secret
const decryptedEncryptionKey = decryptUsingPassword(encryption_key, userSecret);

const decryptedIV = decryptUsingPassword(init_vector, userSecret);

// decrypt the user data using the decrypted key pair
const decryptedUserData = decryptUsingEncryptionKey(
  encryptedSecretData,
  decryptedEncryptionKey,
  decryptedIV
);

// forward the decrypted data to the next service
next(decryptedUserData);

Conclusion

While coming up with the initial plan for these procedures required some planning, the actual implementation of this service was quite straightforward. The solution performed all the needed functionalities in-house without needing external services or dependencies. In the end, we can present the pros and cons of this kind of implementation as following

PROS

Cons