Table Maintenance¶
Metadata File Cleanup¶
Lakekeeper honors the Iceberg table properties write.metadata.delete-after-commit.enabled and write.metadata.previous-versions-max. Starting with Lakekeeper v0.10.0, delete-after-commit is enabled by default (it was disabled in earlier versions). On each table commit, when delete-after-commit is enabled, Lakekeeper keeps the current table metadata file plus up to write.metadata.previous-versions-max previous metadata files (default: 100) and deletes the oldest tracked metadata file from the metadata log once that limit is exceeded. This cleanup applies only to metadata files tracked in the metadata log; it does not remove orphaned metadata files.
For example: if write.metadata.previous-versions-max=20, Lakekeeper retains 21 files in total (the current plus 20 previous); committing a 22nd version deletes the oldest tracked metadata file.
Link to Expire Snapshots
Expire Snapshots ¶
Lakekeeper automatically expires old table snapshots based on configurable age and retention policies. This helps manage storage costs and performance by removing outdated snapshot metadata and associated data files.
Expire snapshots can be configured per warehouse and optionally overridden at the table level using Iceberg table properties.
Configuration¶
Configuration can be set via the Management UI or REST API endpoints:
- GET
/management/v1/warehouse/{warehouse_id}/task-queue/expire_snapshots/config - POST
/management/v1/warehouse/{warehouse_id}/task-queue/expire_snapshots/config
| Parameter | Type | Default | Description |
|---|---|---|---|
enable-expire-snapshots |
boolean | false |
Enable automatic snapshot expiration for all tables in the warehouse. Can be overridden per table with lakekeeper.history.expire.enabled |
max-snapshot-age-ms |
integer | 432000000 (5 days) |
Maximum age of snapshots in milliseconds before expiration. Override per table with history.expire.max-snapshot-age-ms |
min-snapshots-to-keep |
integer | 1 |
Minimum snapshots to retain on each table branch. Override per table with history.expire.min-snapshots-to-keep |
min-snapshots-to-expire |
integer | 20 |
Minimum snapshots required before expiration job is scheduled (prevents expensive jobs for few snapshots). Override per table with lakekeeper.history.expire.min-snapshots-to-expire |
max-ref-age-ms |
integer | 9223372036854775807 (no limit) |
Maximum age for snapshot references (except main branch). Main branch references never expire |
Table-Level Overrides¶
Individual tables can override warehouse settings using these Iceberg table properties:
lakekeeper.history.expire.enabled- Enable/disable for specific tablehistory.expire.max-snapshot-age-ms- Custom max age for table snapshotshistory.expire.min-snapshots-to-keep- Custom minimum retention for tablelakekeeper.history.expire.min-snapshots-to-expire- Custom threshold for table
Note
Tables with gc.enabled=false are excluded from automatic expiration regardless of other settings.
Production Deployment¶
For production workloads, we recommend running expire snapshots workers in dedicated pods to avoid impacting REST API performance. This can be achieved by:
- API pods: Set
LAKEKEEPER__TASK_EXPIRE_SNAPSHOTS_WORKERS=0to disable workers - Worker pods: Use default worker configuration (2 workers) to handle expire snapshots tasks or set
LAKEKEEPER__TASK_EXPIRE_SNAPSHOTS_WORKERSto desired number of workers
Task Scheduling¶
Expire snapshots tasks are intelligently scheduled immediately after table commits when needed, eliminating the overhead of cron-based polling. This ensures timely cleanup while maintaining optimal performance.
Remove Orphan Files ¶
Lakekeeper can detect and remove orphan files — files in a table's storage location that are no longer referenced by any snapshot, manifest, statistics file, or metadata log entry. Orphans typically come from failed writes (optimistic-concurrency conflicts) or incomplete maintenance jobs.
The queue is opt-in per warehouse and disabled by default. Once enabled, tables are scheduled adaptively: the next run is timed to the observed rate at which orphans accumulate, with a 1-day floor and a configurable ceiling. Idle tables get a periodic safety check at the ceiling; busy tables get reclaimed more often. No cron-based polling is involved.
How It Works¶
- Scan referenced files: All files referenced by the current table metadata are collected (manifest lists, manifests, data files, statistics files, partition statistics files, and metadata log entries).
- List storage: The table's storage location is listed recursively.
- Identify orphans: Files present in storage but not in the referenced set are orphan candidates.
- Age filter: Only files older than the configured threshold are deleted. Recently created files are preserved to avoid deleting data from in-progress writes.
- Delete: Orphans are deleted in micro-batches (using cloud-native batch-delete APIs where available).
- Reschedule: The worker self-schedules the next run based on the observed orphan-bytes-per-second rate.
Configuration¶
Configuration can be set via the Management UI or REST API endpoints:
- GET
/management/v1/warehouse/{warehouse_id}/task-queue/remove_orphan_files/config - POST
/management/v1/warehouse/{warehouse_id}/task-queue/remove_orphan_files/config
| Parameter | Type | Default | Description |
|---|---|---|---|
enable-remove-orphan-files |
boolean | false |
Master switch for the warehouse. When false, no tables are scheduled unless they set lakekeeper.remove-orphan-files.enabled=true. |
default-older-than-ms |
integer | 604800000 (7 days) |
Minimum file age before deletion. 24-hour safety floor enforced at task pickup (see below). Override per table with lakekeeper.remove-orphan-files.older-than-ms. |
target-reclaim-bytes |
integer | 1073741824 (1 GiB) |
Adaptive scheduler target: the next run is timed to reclaim roughly this many bytes based on the last run's observed rate. Clamped to [1 day, maximum-interval-seconds]. |
maximum-interval-seconds |
integer | 7776000 (90 days) |
Upper bound on the adaptive next-run interval. Idle tables get a safety check at this cadence. Must be ≥ 86400 (1 day). |
max-run-time-seconds |
integer | 3600 (1 hour) |
Wall-clock timeout for a single task attempt. Must be ≥ 60. |
dry-run |
boolean | false |
Identify orphans but do not delete. Useful for previewing what a run would remove. |
disable-min-older-than-check |
boolean | false |
Bypass the 24h floor on default-older-than-ms. Set only for dev/test setups that genuinely need sub-day retention. |
Table-Level Overrides¶
Individual tables can override warehouse settings using these Iceberg table properties:
lakekeeper.remove-orphan-files.enabled—trueincludes a table even when the warehouse master switch is off;falseexcludes it.lakekeeper.remove-orphan-files.older-than-ms— per-table deletion-age threshold. Not subject to the 24h safety floor; use only for known-stale tables.lakekeeper.remove-orphan-files.dry-run— per-table dry-run. Safer mode wins: if the warehouse is dry-run, the table cannot downgrade to live.
Safety Mechanisms¶
gc.enabledcheck: Tables withgc.enabled=falseare excluded. Attempting to remove orphan files on such a table returns an error immediately.- 24h safety floor on
default-older-than-ms: At task pickup, the worker refuses to run with a queue-config retention shorter than 24 hours, returningRemoveOrphanFilesRetentionTooShort. A typo likedefault-older-than-ms: 6000(six seconds) would otherwise delete in-flight writer uploads. Mirrors Spark/Iceberg'sretentionDurationCheck.enabled=falseescape — setdisable-min-older-than-check=trueto override. - Unknown age preservation: Files where the storage backend does not report a last-modified timestamp are skipped (counted as
skipped_unknown_age_countin the result).
Recommended Usage¶
Run remove orphan files after expire snapshots. Expiring snapshots first ensures that files from removed snapshots are no longer referenced, so they can be correctly identified as orphans.
Production Deployment¶
For production workloads, we recommend running remove orphan files workers in dedicated pods, similar to expire snapshots:
- API pods: Set
LAKEKEEPER__TASK_REMOVE_ORPHAN_FILES_WORKERS=0to disable workers. - Worker pods: Use the default (2 workers) or set
LAKEKEEPER__TASK_REMOVE_ORPHAN_FILES_WORKERSto the desired number.
Warning
Every run performs a full recursive listing of the table's storage location, which can be expensive for tables with many files. The adaptive scheduler stretches the cadence on tables that produce few orphans, but each run still pays the listing cost.