name: encrypted-saved-objects description: Encrypted Saved Objects (ESO) in Kibana — registration, AAD attribute choices, partial update safety, model version migrations with createModelVersion, canEncrypt checks, and Serverless constraints. Use when creating, modifying, or working with ESO types.
Encrypted Saved Objects (ESO)
Sensitive Data Protection: Encrypted Saved Objects protect credentials, API keys, PII, and other secrets stored in Kibana. Incorrect ESO changes can make objects permanently undecryptable.
Overview
An Encrypted Saved Object (ESO) is a Saved Object type registered with the ESO Service to specify:
attributesToEncrypt: Attributes containing sensitive data (encrypted at rest)attributesToIncludeInAAD: Attributes used as Additional Authenticated Data (bound to the encrypted data, must match exactly during decryption)
The ESO Service encrypts using the xpack.encryptedSavedObjects.encryptionKey Kibana config setting. In development, a static key is auto-configured.
Definitive reference: dev_docs/key_concepts/encrypted_saved_objects.mdx
When to Use ESOs
Only register a Saved Object type as encrypted if it stores genuinely sensitive data:
- Credentials: passwords, API keys, access keys, tokens
- PII: social security numbers, credit card numbers, bank routing numbers
- Other secrets: private endpoints, signing keys, certificates
Most Saved Object types do not need encryption. When in doubt, consult #kibana-security.
Registration
Step 1: Register the Saved Object type with Core
savedObjects.registerType({
name: 'my_encrypted_type',
hidden: true,
namespaceType: 'multiple-isolated',
mappings: { /* ... */ },
modelVersions: myModelVersions,
});
Step 2: Register with the ESO Service
encryptedSavedObjects.registerType({
type: 'my_encrypted_type', // must match the Core registration name
attributesToEncrypt: new Set(['secrets']),
attributesToIncludeInAAD: new Set(['connectorType', 'createdAt']),
});
Key rules:
- The
typestring must exactly match the name used in Core'ssavedObjects.registerType attributesToEncryptmust not be empty- Use
{ key: 'fieldName', dangerouslyExposeValue: true }only when decrypted values must be exposed through standard SO client APIs (e.g.get,find); this requires thorough justification and documentation
Choosing attributesToEncrypt
Encrypt any attribute containing sensitive data. By default, encrypted attributes are stripped from responses when accessed via standard Saved Object Client APIs (get, find, etc.). To access decrypted values, use the dedicated ESO Client APIs (getDecryptedAsInternalUser, createPointInTimeFinderDecryptedAsInternalUser).
Choosing attributesToIncludeInAAD
AAD attributes are not encrypted but are cryptographically bound to the encrypted data. If any AAD attribute changes, all encrypted attributes must be re-encrypted.
INCLUDE in AAD — attributes that:
- Are associated with or describe the encrypted data (e.g., connector type, token type, URL)
- Never change after creation (e.g.,
createdAt,createdBy, type identifiers)
EXCLUDE from AAD — attributes that:
- Are present in
attributesToEncrypt - Can be changed by end users independently of encrypted data (e.g., display name, UI settings)
- May be optional, absent, or calculated (e.g., statistics,
updatedAt) - May be removed or refactored in the future (deprecated or experimental fields)
- Contain large data that would slow encryption/decryption in bulk operations
Be conservative: only include attributes the team is 100% confident should be included. Adding an existing populated attribute to AAD later is not supported in Serverless.
Nested attributes: When an attribute is included in AAD, all of its subfields are inherently included. For more granular control, use dotted keys like rule.apiKeyOwner instead of the entire rule object.
Partial Update Safety
Critical: Partial updates (savedObjectsClient.update or savedObjectsRepository.update) on ESOs must never modify encrypted attributes or AAD-included attributes. Doing so corrupts the object, making it permanently undecryptable.
Required pattern: Create a type-safe partial update helper that strips encrypted and AAD attributes:
export const MyTypeAttributesToEncrypt = ['secrets'];
export const MyTypeAttributesIncludedInAAD = ['connectorType', 'createdAt'];
export type MyTypeAttributesNotPartiallyUpdatable = 'secrets' | 'connectorType' | 'createdAt';
export type PartiallyUpdateableMyTypeAttributes = Partial<
Omit<MyTypeSO, MyTypeAttributesNotPartiallyUpdatable>
>;
export async function partiallyUpdateMyType(
savedObjectsClient: Pick<SavedObjectsClient, 'update'>,
id: string,
attributes: PartiallyUpdateableMyTypeAttributes,
options: SavedObjectsUpdateOptions = {}
): Promise<void> {
const safeAttributes = omit(attributes, [
...MyTypeAttributesToEncrypt,
...MyTypeAttributesIncludedInAAD,
]);
await savedObjectsClient.update('my_encrypted_type', id, safeAttributes, options);
}
Any code that calls savedObjectsClient.update or savedObjectsRepository.update on an ESO type must verify that encrypted and AAD attributes are excluded from the update payload.
Accessing Decrypted Data
ESO Client APIs (server-side only)
Use the dedicated ESO Client for accessing decrypted attributes. These run as the internal Kibana user and should not expose secrets to end users unless absolutely necessary:
// Single object
const decrypted = await encryptedSavedObjectsClient.getDecryptedAsInternalUser<MyType>(
'my_encrypted_type',
objectId,
{ namespace }
);
// Bulk find with decryption
const finder = await encryptedSavedObjectsClient
.createPointInTimeFinderDecryptedAsInternalUser<MyType>({
type: 'my_encrypted_type',
perPage: 100,
});
Decrypted values should not be returned directly in HTTP responses without explicit justification. Calling getDecryptedAsInternalUser with a type that is not registered as encrypted throws at runtime.
Graceful Degradation with canEncrypt
The ESO encryption key is optional. Plugins must check canEncrypt and handle the case where encryption is unavailable:
// Setup phase: store the canEncrypt flag
const canEncrypt = plugins.encryptedSavedObjects.canEncrypt;
// Runtime: degrade gracefully or reject operations
if (!canEncrypt) {
// Option 1: Reject the operation with a clear error
throw new Error('Encryption key is not configured. Cannot create encrypted objects.');
// Option 2: Degrade gracefully (e.g., skip encryption-dependent features)
logger.warn('Encryption key not set. Feature X is unavailable.');
}
Any plugin that uses ESO features (registers types, calls getDecryptedAsInternalUser, etc.) without checking canEncrypt or handling the absence of an encryption key has a bug.
Model Version Migrations
When an ESO type's encrypted attributes or AAD-included attributes change, use createModelVersion to wrap the model version definition with automatic decryption/re-encryption:
import type { EncryptedSavedObjectTypeRegistration } from '@kbn/encrypted-saved-objects-plugin/server';
// Previous version's registration (for decryption)
const inputType: EncryptedSavedObjectTypeRegistration = {
type: 'my_encrypted_type',
attributesToEncrypt: new Set(['secrets']),
attributesToIncludeInAAD: new Set(['connectorType']),
};
// New version's registration (for re-encryption)
const outputType: EncryptedSavedObjectTypeRegistration = {
type: 'my_encrypted_type',
attributesToEncrypt: new Set(['secrets']),
attributesToIncludeInAAD: new Set(['connectorType', 'createdAt']),
};
// In the Saved Object type registration:
modelVersions: {
2: plugins.encryptedSavedObjects.createModelVersion({
modelVersion: {
changes: [
{
type: 'data_backfill',
backfillFn: (doc) => ({
attributes: { createdAt: doc.attributes.createdAt ?? new Date().toISOString() },
}),
},
],
schemas: {
forwardCompatibility: mySchemaV2.extends({}, { unknowns: 'ignore' }),
create: mySchemaV2,
},
},
inputType,
outputType,
shouldTransformIfDecryptionFails: true, // optional: proceed even if decryption fails
}),
},
Key rules:
createModelVersionrequires at least one change in thechangesarrayinputTypemust match the ESO registration from the previous model versionoutputTypemust match the ESO registration for the new model version- All transform functions (
unsafe_transform,data_backfill,data_removal) are merged into a single decrypt-transform-encrypt pass createModelVersionis only needed when encrypted or AAD attributes change; purely unencrypted, non-AAD changes can use standard model versions
Reference implementation: examples/eso_model_version_example/server/plugin.ts
Serverless Considerations (Zero Downtime Upgrades)
In Serverless, both the current and previous Kibana versions may run simultaneously. The previous version must be able to decrypt ESOs migrated by the new version without knowledge of the new model version.
Critical constraints
- Cannot add an existing populated attribute to AAD. The previous version will never successfully decrypt because it does not include the attribute in its AAD construction.
- Cannot remove an attribute from AAD. The previous version will always include it in AAD construction, causing decryption to fail.
- Cannot change an attribute from unencrypted to encrypted. The previous version will not attempt decryption.
- Cannot change an attribute from encrypted to unencrypted. The previous version will always attempt decryption.
Multi-stage release patterns
Some changes require 2 Serverless releases:
Adding a new AAD attribute:
- Release 1: Add the attribute to
attributesToIncludeInAADin the registration. Do NOT populate or use the attribute yet. - Release 2: Implement a model version with
createModelVersionto backfill and start using the attribute.
Removing an attribute (when previous version depends on it):
- Release 1: Update all business logic to handle the type without the attribute.
- Release 2: Implement a model version to remove the attribute.
forwardCompatibility schema
Set unknowns: 'ignore' in the forwardCompatibility schema when the previous version should drop unknown fields. This is helpful if the additional fields are not compatible or problematic in the previous version.
During model version transformation, decryption occurs BEFORE the forwardCompatibility schema is applied. This supports hierarchical AAD — when subfields of an AAD attribute are added or removed, the previous version can still successfully construct AAD, ensuring objects can be decrypted before being adapted for the previous version.
Quick Change Reference
| Change | Encrypted? | In AAD? | Needs createModelVersion? |
Serverless stages |
|---|---|---|---|---|
| Add new attribute | No | No | No | 1 (with forwardCompatibility if needed) |
| Add new attribute | No | Yes | Yes | 2 |
| Add new attribute | Yes | N/A | Yes | 1 (with forwardCompatibility if needed) |
| Remove attribute | No | No | No | 1-2 depending on business logic |
| Remove attribute | No | Yes | Yes | 1-2 depending on business logic |
| Remove attribute | Yes | N/A | No | 1-2 depending on business logic |
| Modify attribute (add/remove subfield) | No | No | No | 1-2 depending on business logic |
| Modify attribute (add/remove subfield) | No | Yes | Yes | 1-2 depending on business logic |
| Modify attribute (add/remove subfield) | Yes | N/A | Yes | 1-2 depending on business logic |
| Add existing attribute to AAD | No | No->Yes | Not supported | N/A |
| Remove attribute from AAD | No | Yes->No | Not supported | N/A |
| Change unencrypted to encrypted | No->Yes | Any | Not supported | N/A |
| Change encrypted to unencrypted | Yes->No | N/A | Not supported | N/A |
Checklist
When working with ESO-related code, verify:
Registration correctness
-
typematches the Core Saved Object registration name -
attributesToEncryptcontains only genuinely sensitive attributes -
attributesToIncludeInAADfollows the inclusion/exclusion guidelines above -
dangerouslyExposeValueis only used with documented justification
-
Partial update safety
- No
savedObjectsClient.updatecalls modify encrypted or AAD attributes - A type-safe partial update helper exists that strips unsafe attributes
- The
NotPartiallyUpdatabletype is kept in sync with the ESO registration
- No
Model version migrations
-
createModelVersionis used when encrypted or AAD attributes change -
inputTypematches the previous version's ESO registration -
outputTypematches the new version's ESO registration -
forwardCompatibilityschema is set withunknowns: 'ignore'when appropriate
-
Serverless compatibility
- Changes do not add existing populated attributes to AAD
- Changes do not remove attributes from AAD
- Multi-stage releases are used when required (see table above)
- Business logic handles objects with or without new/removed attributes
Encryption availability
-
canEncryptis checked before using ESO-dependent features - Graceful degradation or clear error when encryption is unavailable
-
Secret exposure
- Decrypted values from
getDecryptedAsInternalUserare consumed internally, not exposed in API responses - Any exposure of decrypted values is explicitly justified and documented
- Decrypted values from
References
- Encrypted Saved Objects dev docs
- Secure Saved Objects (Elastic docs)
- Model Versions tutorial
- ESO Model Version example plugin
- ESO plugin source:
x-pack/platform/plugins/shared/encrypted_saved_objects/