| name | stash-encryption |
| description | Implement field-level encryption with @cipherstash/stack. Covers schema definition, encrypt/decrypt operations, searchable encryption (equality, free-text, range, JSON), bulk operations, model operations, identity-aware encryption with LockContext, multi-tenant keysets, and the full TypeScript type system. Use when adding encryption to a project, defining encrypted schemas, or working with the CipherStash Encryption API. |
CipherStash Stack - Encryption
Comprehensive guide for implementing field-level encryption with @cipherstash/stack. Every value is encrypted with its own unique key via ZeroKMS (backed by AWS KMS). Encryption happens client-side before data leaves the application.
When to Use This Skill
- Adding field-level encryption to a TypeScript/Node.js project
- Defining encrypted table schemas
- Encrypting and decrypting individual values or entire models
- Implementing searchable encryption (equality, free-text, range, JSON queries)
- Bulk encrypting/decrypting large datasets
- Implementing identity-aware encryption with JWT-based lock contexts
- Setting up multi-tenant encryption with keysets
- Migrating from
@cipherstash/protect to @cipherstash/stack
Installation
npm install @cipherstash/stack
[!IMPORTANT]
Exclude @cipherstash/stack from bundling — required for any project with a bundler (Next.js, webpack, esbuild, vite SSR, etc.). The package wraps a native FFI module (@cipherstash/protect-ffi) that cannot be bundled. Importing the encryption client from server code without this exclusion will fail at runtime with errors about missing native modules. Configure as soon as you install the package; do not skip this step.
Concrete configuration for the most common bundlers:
Next.js (next.config.{js,ts,mjs}):
const nextConfig = {
serverExternalPackages: ['@cipherstash/stack', '@cipherstash/protect-ffi'],
}
export default nextConfig
(Older Next.js — pre-15 — uses experimental.serverComponentsExternalPackages with the same value.)
webpack (next/nuxt/remix/etc. that compose webpack directly):
config.externals.push('@cipherstash/stack', '@cipherstash/protect-ffi')
esbuild:
{ external: ['@cipherstash/stack', '@cipherstash/protect-ffi'] }
Vite SSR:
ssr: { external: ['@cipherstash/stack', '@cipherstash/protect-ffi'] }
If you skip this step, you'll see runtime errors like Cannot find module '@cipherstash/protect-ffi-darwin-arm64' or dlopen failed once the bundler tries to inline the native binding.
Configuration
Environment Variables
Set these in .env or your hosting platform:
CS_WORKSPACE_CRN=crn:ap-southeast-2.aws:your-workspace-id
CS_CLIENT_ID=your-client-id
CS_CLIENT_KEY=your-client-key
CS_CLIENT_ACCESS_KEY=your-access-key
Sign up at cipherstash.com/signup to generate credentials.
Programmatic Config
const client = await Encryption({
schemas: [users],
config: {
workspaceCrn: "crn:ap-southeast-2.aws:your-workspace-id",
clientId: "your-client-id",
clientKey: "your-client-key",
accessKey: "your-access-key",
keyset: { name: "my-keyset" },
},
})
If config is omitted, the client reads CS_* environment variables automatically.
Logging
Logging is enabled by default at the error level. Configure the log level with the STASH_STACK_LOG environment variable:
STASH_STACK_LOG=error
| Value | What is logged |
|---|
error | Errors only (default) |
info | Info and errors |
debug | Debug, info, and errors |
When STASH_STACK_LOG is not set, the SDK defaults to error (errors only).
The SDK never logs plaintext data.
Subpath Exports
| Import Path | Provides |
|---|
@cipherstash/stack | Encryption function, Secrets class, encryptedTable, encryptedColumn, encryptedField (convenience re-exports) |
@cipherstash/stack/schema | encryptedTable, encryptedColumn, encryptedField, schema types |
@cipherstash/stack/identity | LockContext class and identity types |
@cipherstash/stack/secrets | Secrets class and secrets types |
@cipherstash/stack/drizzle | encryptedType, extractEncryptionSchema, createEncryptionOperators for Drizzle ORM |
@cipherstash/stack/supabase | encryptedSupabase wrapper for Supabase |
@cipherstash/stack/dynamodb | encryptedDynamoDB helper for DynamoDB |
@cipherstash/stack/encryption | EncryptionClient class, Encryption function |
@cipherstash/stack/errors | EncryptionErrorTypes, StackError, error subtypes, getErrorMessage |
@cipherstash/stack/client | Client-safe exports: schema builders, schema types, EncryptionClient type (no native FFI) |
@cipherstash/stack/types | All TypeScript types |
Schema Definition
Define which tables and columns to encrypt using encryptedTable and encryptedColumn:
import { encryptedTable, encryptedColumn } from "@cipherstash/stack/schema"
const users = encryptedTable("users", {
email: encryptedColumn("email")
.equality()
.freeTextSearch()
.orderAndRange(),
age: encryptedColumn("age")
.dataType("number")
.equality()
.orderAndRange(),
address: encryptedColumn("address"),
})
const documents = encryptedTable("documents", {
metadata: encryptedColumn("metadata")
.searchableJson(),
})
Index Types
| Method | Purpose | Query Type |
|---|
.equality(tokenFilters?) | Exact match lookups. Accepts an optional array of token filters (e.g., [{ kind: 'downcase' }]) for case-insensitive matching. | 'equality' |
.freeTextSearch(opts?) | Full-text / fuzzy search | 'freeTextSearch' |
.orderAndRange() | Sorting, comparison, range queries | 'orderAndRange' |
.searchableJson() | Encrypted JSONB path and containment queries (auto-sets dataType to 'json') | 'searchableJson' |
.dataType(cast) | Set plaintext data type | N/A |
Supported data types: 'string' (default), 'text', 'number', 'boolean', 'date', 'bigint', 'json'
Methods are chainable - call as many as you need on a single column.
Free-Text Search Options
encryptedColumn("bio").freeTextSearch({
tokenizer: { kind: "ngram", token_length: 3 },
token_filters: [{ kind: "downcase" }],
k: 6,
m: 2048,
include_original: true,
})
Type Inference
import type { InferPlaintext, InferEncrypted } from "@cipherstash/stack/schema"
type UserPlaintext = InferPlaintext<typeof users>
type UserEncrypted = InferEncrypted<typeof users>
Client Initialization
import { Encryption } from "@cipherstash/stack"
const client = await Encryption({ schemas: [users, documents] })
The Encryption() function returns Promise<EncryptionClient> and throws on error (e.g., bad credentials, missing config, invalid keyset UUID). At least one schema is required.
try {
const client = await Encryption({ schemas: [users] })
} catch (error) {
console.error("Init failed:", error.message)
}
Encrypt and Decrypt Single Values
const encrypted = await client.encrypt("hello@example.com", {
column: users.email,
table: users,
})
if (encrypted.failure) {
console.error(encrypted.failure.message)
} else {
console.log(encrypted.data)
}
const decrypted = await client.decrypt(encrypted.data)
if (!decrypted.failure) {
console.log(decrypted.data)
}
All plaintext values must be non-null. Null handling is managed at the model level by encryptModel and decryptModel.
Model Operations
Encrypt or decrypt an entire object. Only fields matching your schema are encrypted; other fields pass through unchanged.
The return type is schema-aware: fields matching the table schema are typed as Encrypted, while other fields retain their original types. For best results, let TypeScript infer the type parameters from the arguments rather than providing an explicit <User>.
type User = { id: string; email: string; createdAt: Date }
const user = {
id: "user_123",
email: "alice@example.com",
createdAt: new Date(),
}
const encResult = await client.encryptModel(user, users)
if (!encResult.failure) {
}
const decResult = await client.decryptModel(encResult.data)
if (!decResult.failure) {
console.log(decResult.data.email)
}
The Decrypted<T> type maps encrypted fields back to their plaintext types.
Passing an explicit type parameter (e.g., client.encryptModel<User>(...)) still works for backward compatibility — the return type degrades to User in that case.
Bulk Operations
All bulk methods make a single call to ZeroKMS regardless of record count, while still using a unique key per value.
Bulk Encrypt / Decrypt (Raw Values)
const plaintexts = [
{ id: "u1", plaintext: "alice@example.com" },
{ id: "u2", plaintext: "bob@example.com" },
{ id: "u3", plaintext: "charlie@example.com" },
]
const encrypted = await client.bulkEncrypt(plaintexts, {
column: users.email,
table: users,
})
const decrypted = await client.bulkDecrypt(encrypted.data)
for (const item of decrypted.data) {
if ("data" in item) {
console.log(`${item.id}: ${item.data}`)
} else {
console.error(`${item.id} failed: ${item.error}`)
}
}
Bulk Encrypt / Decrypt Models
const userModels = [
{ id: "1", email: "alice@example.com" },
{ id: "2", email: "bob@example.com" },
]
const encrypted = await client.bulkEncryptModels(userModels, users)
const decrypted = await client.bulkDecryptModels(encrypted.data)
Searchable Encryption
Encrypt query terms so you can search encrypted data in PostgreSQL.
Single Query Encryption
const eqQuery = await client.encryptQuery("alice@example.com", {
column: users.email,
table: users,
queryType: "equality",
})
const matchQuery = await client.encryptQuery("alice", {
column: users.email,
table: users,
queryType: "freeTextSearch",
})
const rangeQuery = await client.encryptQuery(25, {
column: users.age,
table: users,
queryType: "orderAndRange",
})
const pathQuery = await client.encryptQuery("$.user.email", {
column: documents.metadata,
table: documents,
queryType: "steVecSelector",
})
const containsQuery = await client.encryptQuery({ role: "admin" }, {
column: documents.metadata,
table: documents,
queryType: "steVecTerm",
})
If queryType is omitted, it's auto-inferred from the column's configured indexes (priority: unique > match > ore > ste_vec).
Query Result Formatting (returnType)
By default encryptQuery returns an Encrypted object (the raw EQL JSON payload). Use returnType to change the output format:
returnType | Output | Use case |
|---|
'eql' (default) | Encrypted object | Parameterized queries, ORMs accepting JSON |
'composite-literal' | string | Supabase .eq(), string-based APIs |
'escaped-composite-literal' | string | Embedding inside another string or JSON value |
const term = await client.encryptQuery("alice@example.com", {
column: users.email,
table: users,
queryType: "equality",
returnType: "composite-literal",
})
Each term in a batch can have its own returnType.
Searchable JSON
For columns using .searchableJson(), the query type is auto-inferred from the plaintext:
const pathQuery = await client.encryptQuery("$.user.email", {
column: documents.metadata,
table: documents,
})
const containsQuery = await client.encryptQuery({ role: "admin" }, {
column: documents.metadata,
table: documents,
})
Batch Query Encryption
Encrypt multiple query terms in one ZeroKMS call:
const terms = [
{ value: "alice@example.com", column: users.email, table: users, queryType: "equality" as const },
{ value: "bob", column: users.email, table: users, queryType: "freeTextSearch" as const },
]
const results = await client.encryptQuery(terms)
All values in the array must be non-null.
Identity-Aware Encryption (Lock Contexts)
Lock encryption to a specific user by requiring a valid JWT for decryption.
import { LockContext } from "@cipherstash/stack/identity"
const lc = new LockContext()
const identifyResult = await lc.identify(userJwt)
if (identifyResult.failure) {
throw new Error(identifyResult.failure.message)
}
const lockContext = identifyResult.data
const encrypted = await client
.encrypt("sensitive data", { column: users.email, table: users })
.withLockContext(lockContext)
const decrypted = await client
.decrypt(encrypted.data)
.withLockContext(lockContext)
Lock contexts work with ALL operations: encrypt, decrypt, encryptModel, decryptModel, bulkEncrypt, bulkDecrypt, bulkEncryptModels, bulkDecryptModels, encryptQuery.
CTS Token Service
The lock context exchanges the JWT for a CTS (CipherStash Token Service) token. Set the endpoint:
CS_CTS_ENDPOINT=https://ap-southeast-2.aws.auth.viturhosted.net
Multi-Tenant Encryption (Keysets)
Isolate encryption keys per tenant:
const client = await Encryption({
schemas: [users],
config: { keyset: { name: "Company A" } },
})
const client = await Encryption({
schemas: [users],
config: { keyset: { id: "123e4567-e89b-12d3-a456-426614174000" } },
})
Each keyset provides full cryptographic isolation between tenants.
Operation Chaining
All operations return thenable objects that support chaining:
const result = await client
.encrypt(plaintext, { column: users.email, table: users })
.withLockContext(lockContext)
.audit({ metadata: { action: "create" } })
Error Handling
All async methods return a Result object - a discriminated union with either data (success) or failure (error), never both.
const result = await client.encrypt("hello", { column: users.email, table: users })
if (result.failure) {
console.error(result.failure.type, result.failure.message)
} else {
console.log(result.data)
}
Error Types
| Type | When |
|---|
ClientInitError | Client initialization fails (bad credentials, missing config) |
EncryptionError | An encrypt operation fails (has optional code field) |
DecryptionError | A decrypt operation fails |
LockContextError | Lock context creation or usage fails |
CtsTokenError | Identity token exchange fails |
StackError is a discriminated union of all the error types above, enabling exhaustive switch handling. EncryptionErrorTypes provides runtime constants for each error type string. Use getErrorMessage(error: unknown): string to safely extract a message from any thrown value.
import { EncryptionErrorTypes, type StackError, getErrorMessage } from "@cipherstash/stack/errors"
function handleError(error: StackError) {
switch (error.type) {
case EncryptionErrorTypes.ClientInitError:
console.error("Init failed:", error.message)
break
case EncryptionErrorTypes.EncryptionError:
console.error("Encrypt failed:", error.message, error.code)
break
case EncryptionErrorTypes.DecryptionError:
console.error("Decrypt failed:", error.message)
break
case EncryptionErrorTypes.LockContextError:
console.error("Lock context failed:", error.message)
break
case EncryptionErrorTypes.CtsTokenError:
console.error("CTS token failed:", error.message)
break
default:
const _exhaustive: never = error
}
}
try {
await client.encrypt("data", { column: users.email, table: users })
} catch (e) {
console.error(getErrorMessage(e))
}
Validation Rules
- NaN and Infinity are rejected for numeric values
freeTextSearch index only supports string values
- At least one
encryptedTable schema must be provided
- Keyset UUIDs must be valid format
Ordering Encrypted Data
ORDER BY on encrypted columns requires operator family support in the database.
On databases without operator families (e.g. Supabase, or when EQL is installed with --exclude-operator-family), sorting on encrypted columns is not currently supported — regardless of the client or ORM used. This applies to Drizzle, the Supabase JS SDK, raw SQL, and any other database client.
Workaround: Sort application-side after decrypting the results.
Operator family support for Supabase is being developed in collaboration with the Supabase and CipherStash teams and will be available in a future release.
PostgreSQL Storage
Encrypted data is stored as EQL (Encrypt Query Language) JSON payloads. Install the EQL extension in PostgreSQL:
CREATE EXTENSION IF NOT EXISTS eql_v2;
CREATE TABLE users (
id BIGINT GENERATED ALWAYS AS IDENTITY PRIMARY KEY,
email eql_v2_encrypted
);
Or store as JSONB if not using the EQL extension directly:
CREATE TABLE users (
id SERIAL PRIMARY KEY,
email jsonb NOT NULL
);
Rolling Encryption Out to Production
Adding a fresh encrypted column to a table you don't yet write to is the easy case — declare it in the schema, run the migration, start writing. The harder case is taking an existing plaintext column with live data and turning it into an encrypted one without dropping a write or returning the wrong value mid-cutover.
CipherStash splits that into two named steps with a hard production-deploy gate between them:
ENCRYPTION ROLLOUT → ⛔ deploy gate → ENCRYPTION CUTOVER
───────────────────── ──────────────────────
schema-add backfill historical rows
dual-write code switch reads to encrypted
then drop the plaintext column when reads are decrypting.
The gate is the rule that backfill is only safe once the dual-write code is running in the production environment that owns the database — not on the developer's laptop, not in CI. Any row inserted during the backfill window must be written to both columns by the application; otherwise it lands in plaintext only and creates silent migration drift.
Runner note. stash init adds stash to the project as a dev dependency, so stash <command> runs through whichever package manager the project uses (Bun, pnpm, Yarn, or npm) — examples below show this bare form. Before init has run, prefix with your package manager's one-shot runner: bunx, pnpm dlx, yarn dlx, or npx. The CLI's behaviour is identical across all of them; only the prefix changes. The stash-cli skill has the full mapping.
Where am I?
Always start with stash status (stash status / pnpm dlx stash status / etc., per the runner note above). It is disk-only, idempotent, and tells you which encryption rollouts are in flight, what's been deployed, and what the next move is per column. Re-run it after every transition. Never act blind.
Step 1 — Encryption rollout
Everything that lands in the repo and ships in one PR:
| Action | What changes |
|---|
| Schema-add | Migration adds <col>_encrypted (nullable jsonb) alongside the existing plaintext column. Plaintext column unchanged; application still writes only plaintext. |
| Dual-write code | Application now writes both <col> and <col>_encrypted on every persistence path that mutates the row, in the same transaction, on every code branch. Reads still come from the plaintext column. |
If you use CipherStash Proxy: After the schema-add, run stash db push to register the new column in eql_v2_configuration. With no active config yet it writes directly to active; with an existing active config it writes pending (cutover will promote it). Required for Proxy-based queries.
The dual-write definition matters. "Writes both columns" is not enough. The rule is: every persistence path that mutates this row writes both columns, in the same transaction, on every code branch. A single missed branch — a CSV import, an admin action, a background job, a third-party webhook handler — means rows inserted in production after deploy land in plaintext only, and backfill won't catch them. Grep for every site that writes the plaintext column before declaring rollout complete.
⛔ Deploy gate
Stop. The rollout PR ships to production. The deployed environment must be running this code before any cutover-step work is safe.
When the deploy is live, run stash status. Look for the active quest's "Next move" hint to confirm dual-writes are recorded. Then run stash plan again — the CLI detects that dual-writes are live and writes a separate cutover plan.
stash impl will refuse to run a cutover-step plan if cs_migrations has no dual_writing event for the targeted columns. That refusal is intentional; it's the safety net for cases where someone runs cutover work locally before the code is actually live.
Step 2 — Encryption cutover
Once dual-writes are recorded as live in cs_migrations:
| Action | What changes |
|---|
stash encrypt backfill | Walks the table in keyset-pagination order, encrypts each chunk, writes a single transactional UPDATE per chunk plus a cs_migrations checkpoint. SIGINT-safe; idempotent re-runs converge. |
| Schema rename | Update the schema file: drop the _encrypted suffix; switch the original column declaration onto the encrypted type. |
stash encrypt cutover | One transaction: renames <col> → <col>_plaintext, <col>_encrypted → <col>, and promotes pending → active. Application reads of <col> now return decrypted ciphertext transparently. |
| Wire reads through the encryption client | Read paths must decrypt before returning the value to callers (decryptModel(row, table) for Drizzle; encryptedSupabase wrapper for Supabase; decrypt/bulkDecryptModels otherwise). Without this step, reads return raw eql_v2_encrypted payloads to end users. |
| Remove dual-write code | The plaintext column is now <col>_plaintext and is no longer authoritative. Delete the dual-write logic. |
stash encrypt drop | Emits a migration that removes <col>_plaintext. Apply with the project's normal migration tooling. |
If you use CipherStash Proxy: After the schema rename, run stash db push to register the renamed shape as pending. This is required for Proxy-based queries; SDK users skip this step.
State storage
Three sources of truth, kept separate on purpose:
.cipherstash/migrations.json (repo) — intent. Which columns the developer wants to encrypt and at which phase, code-reviewable.
eql_v2_configuration (DB, EQL-managed) — EQL intent. Which columns are encrypted and with which indexes; drives the CipherStash Proxy.
cipherstash.cs_migrations (DB, CipherStash-managed) — runtime state. Append-only event log: phase transitions, backfill cursors, error rows. Latest row per (table, column) is the current state.
stash encrypt status shows all three side-by-side and flags drift (e.g. EQL says registered, the physical <col>_encrypted column is missing). stash status (the quest log) rolls them up into the per-column "what's the next move" view used during a rollout.
Note on internal phase names. The runtime event log uses schema-added → dual-writing → backfilling → backfilled → cut-over → dropped as machine-readable phase names. They appear in cs_migrations rows and stash encrypt status output. Treat them as internal mechanism detail — the user-facing story is "encryption rollout, then cutover, with a deploy gate in between."
CLI sequence for a single column
Known limitation: stash encrypt cutover currently requires a pending EQL configuration registered via stash db push. SDK-only users may hit a "No pending EQL configuration" error. Workaround: Run stash db push once before stash encrypt cutover, even if you don't use CipherStash Proxy. Decoupling cutover from EQL config for SDK users is tracked in issue #447 follow-up work.
stash status
stash status
stash plan
stash encrypt backfill --table users --column email
stash encrypt backfill --table users --column email --force
stash db push
stash encrypt cutover --table users --column email
stash encrypt drop --table users --column email
If you use CipherStash Proxy
Register and promote encryption config at each phase:
stash status
stash db push
stash status
stash plan
stash encrypt backfill --table users --column email
stash encrypt backfill --table users --column email --force
stash db push
stash encrypt cutover --table users --column email
stash encrypt drop --table users --column email
Library use
Long-running backfills can also embed the engine directly without the CLI:
import { runBackfill } from '@cipherstash/migrate'
import { Encryption } from '@cipherstash/stack'
const encryptionClient = await Encryption({ schemas: [usersTable] })
await runBackfill({
db,
encryptionClient,
tableSchema: usersTable,
tableName: 'users',
schemaColumnKey: 'email',
plaintextColumn: 'email',
encryptedColumn: 'email_encrypted',
pkColumn: 'id',
chunkSize: 1000,
signal: abortCtrl.signal,
})
Useful when the backfill needs to run in a worker, on a schedule, or alongside an existing job runner.
Invariants the rollout preserves
- Reads never return the wrong value. Until cutover, reads come from the plaintext column. After cutover, the same
SELECT email returns the decrypted ciphertext via Proxy or the encryption client. There is no in-between.
- Writes never drop. Dual-writing keeps both columns in sync until the cutover moment. After cutover, writes go to the encrypted column.
- The deploy gate is a one-way door for production. Backfill against rows the dual-write code never saw produces silent drift. The CLI refuses to run cutover-step plans without a
dual_writing event recorded; do not paper over that refusal.
- Re-runs are safe. Backfill is idempotent (
<col> IS NOT NULL AND <col>_encrypted IS NULL guards every chunk). cs_migrations is append-only.
- Rollback is possible up to cutover. Until the rename happens, the plaintext column is authoritative; aborting just leaves the encrypted twin partially populated. After cutover, rollback is a manual restore — treat cutover as the one-way door for data.
Migration from @cipherstash/protect
@cipherstash/protect | @cipherstash/stack | Import Path |
|---|
protect(config) | Encryption(config) | @cipherstash/stack |
csTable(name, cols) | encryptedTable(name, cols) | @cipherstash/stack/schema |
csColumn(name) | encryptedColumn(name) | @cipherstash/stack/schema |
LockContext from /identify | LockContext from /identity | @cipherstash/stack/identity |
All method signatures on the encryption client remain the same. The Result pattern is unchanged.
Complete API Reference
EncryptionClient Methods
| Method | Signature | Returns |
|---|
encrypt | (plaintext, { column, table }) | EncryptOperation |
decrypt | (encryptedData) | DecryptOperation |
encryptQuery | (plaintext, { column, table, queryType?, returnType? }) | EncryptQueryOperation |
encryptQuery | (terms: readonly ScalarQueryTerm[]) | BatchEncryptQueryOperation |
encryptModel | (model, table) | EncryptModelOperation<EncryptedFromSchema<T, S>> |
decryptModel | (encryptedModel) | DecryptModelOperation<T> — resolves to Decrypted<T> |
bulkEncrypt | (plaintexts, { column, table }) | BulkEncryptOperation |
bulkDecrypt | (encryptedPayloads) | BulkDecryptOperation |
bulkEncryptModels | (models, table) | BulkEncryptModelsOperation<EncryptedFromSchema<T, S>> |
bulkDecryptModels | (encryptedModels) | BulkDecryptModelsOperation<T> — resolves to Decrypted<T>[] |
All operations are thenable (awaitable) and support .withLockContext() and .audit() chaining.
Schema Builders
encryptedTable(tableName: string, columns: Record<string, EncryptedColumn | EncryptedField | nested>)
encryptedColumn(columnName: string)
encryptedField(valueName: string)