Persistent Storage
Miren provides two options for persistent storage: Local Storage (simple, node-local) and Miren Disks (managed persistent volumes). Both are configured as disks in your app.toml.
Both storage options are node-local — your data lives on the server where your app runs. See each section below for backup options.
Local Storage
Local storage gives your app a persistent directory on the server's filesystem. Data survives container restarts and redeployments.
Configuration
Add a disk with provider = "local" to your service in .miren/app.toml:
[services.web]
command = "node server.js"
[[services.web.disks]]
name = "data"
provider = "local"
mount_path = "/miren/data/local"
You can mount to any path — for example, directly to a database's data directory:
[services.db]
image = "postgres:16"
[[services.db.disks]]
name = "pgdata"
provider = "local"
mount_path = "/var/lib/postgresql/data"
How It Works
- Persistent: Data survives container restarts and redeployments
- Shared: All containers within your app share the same storage
- Host-local: Data lives on the server's filesystem
- Node-pinned: Apps with local storage are scheduled to the coordinator node
When to Use Local Storage
- SQLite databases
- File uploads and user content
- Application cache
- Session storage
- Any data that needs to persist across restarts
Limitations
- Host-local: Data is tied to the server. If you move your app to a different server, you'll need to migrate the data manually.
- No managed backups: Back up your data by copying the host directory, or use your own backup tooling.
- Shared access: All containers in your app can read/write simultaneously—your application needs to handle concurrent access (SQLite handles this well when configured with
PRAGMA journal_mode=WAL). - Node affinity: Apps with any disk (local or miren) are pinned to the coordinator and won't be scheduled to distributed runners.
Migrating from Automatic Local Storage
Previously, Miren automatically mounted /miren/data/local for every app. This is now opt-in via the disk config above.
If any of your environment variables reference /miren/data/local, Miren will automatically add the local storage volume for you — so most apps will keep working without changes. You'll see a log message when this happens, and we recommend adding the explicit disk config when convenient.
Miren Disks
Miren Disks live on your server. Back up important data with miren disk backup and restore it with miren disk restore. Cloud backup is on the roadmap.
Miren Disks provide managed persistent storage for your applications. Disks are provisioned with a specific size and filesystem, support exclusive leasing for data consistency, and persist across app restarts and redeployments.
Why Use Disks?
- Managed lifecycle: Miren handles disk creation, formatting, and attachment automatically
- Configurable size and filesystem: Specify exactly what you need
- Thin provisioning: Storage is allocated as needed, not all at once
- Persist across redeployments: Disks survive app deletion — reattach by name
How Disks Work
When you configure a disk for your application:
- Miren creates the disk with the size and filesystem you specify
- Your app instance acquires a lease on the disk (exclusive access)
- The disk is mounted at the path you specified in your container
When your app stops or restarts:
- The lease is released
- Data remains on the disk
- Your next instance can acquire the lease and continue where it left off
How Much Storage Does Miren Provide?
During the Developer Preview, we're providing unmetered storage. The intention is to implement a free tier and usage-based pricing on the storage. We'll be sure to communicate often and clearly how we intend to proceed.
The feature is designed to keep our costs low, and our intention is to pass that low cost on to our users.
Configuring Disks
Add a disk to your application by including a disks section in your service configuration in .miren/app.toml:
[services.web]
image = "myapp:latest"
[[services.web.disks]]
name = "my-app-data"
mount_path = "/data"
size_gb = 10
filesystem = "ext4"
Configuration Options
| Option | Required | Description |
|---|---|---|
name | Yes | Unique name for the disk (alphanumeric, hyphens allowed) |
mount_path | Yes | Where to mount the disk in your container |
size_gb | Yes* | Size in gigabytes (required for auto-creation) |
filesystem | No | Filesystem type: ext4 (default), xfs, or btrfs |
read_only | No | Mount as read-only (default: false) |
*size_gb is required when the disk doesn't already exist. If the disk exists, this field is ignored.
Example: PostgreSQL with Persistent Storage
[services.db]
image = "postgres:16"
[[services.db.env]]
key = "POSTGRES_PASSWORD"
value = "secret"
[[services.db.env]]
key = "PGDATA"
value = "/var/lib/postgresql/data/pgdata"
[[services.db.disks]]
name = "myapp-postgres"
mount_path = "/var/lib/postgresql/data"
size_gb = 20
filesystem = "ext4"
Example: File Upload Storage
[services.web]
image = "myapp:latest"
[[services.web.disks]]
name = "myapp-uploads"
mount_path = "/app/uploads"
size_gb = 50
Disk Lifecycle
Creation
Disks are automatically created when your app first deploys with a volume configuration that includes size_gb. The disk is provisioned with the specified size and filesystem.
Reuse
If you deploy an app with a name that matches an existing disk, Miren will attach that disk instead of creating a new one. This allows you to:
- Share data between app versions
- Preserve data across complete redeployments
- Reference disks created by other apps
Deletion
Disks are not automatically deleted when you delete an app. This is intentional - your data is precious. To delete a disk:
miren debug disk delete -i <disk-id>
Inspecting Disks
List all disks:
miren debug disk list
Check a specific disk's status:
miren debug disk status -i <disk-id>
View active disk leases:
miren debug disk lease-list
See CLI Reference - Disk Commands for complete command documentation.
Important Considerations
One Instance per Disk
Disks use exclusive leasing - only one app instance can mount a disk at a time. This ensures data consistency but means:
- Multiple replicas of your app cannot share the same disk
- If you need shared storage, use separate disks per instance or external storage
Disk Sizing
- Disks use thin provisioning, so storage is only allocated as needed
- Choose a size that accommodates growth
Filesystem Choice
- ext4: Best general-purpose choice, widely compatible
- xfs: Better for large files and high-throughput workloads
NOTE: Your server must have the mkfs tools to format the disk types.
Roadmap: Cloud Backup & Sync
We're building toward cloud-connected storage for Miren Disks. Here's what's planned:
- Remote backup & restore (next up): Trigger backups of your disks to Miren Cloud and restore them on any cluster. This extends the existing local backup/restore functionality to work remotely.
- Automatic cloud sync: Background replication of disk data to Miren Cloud, enabling seamless portability across clusters.
We'll update this page and the changelog as these capabilities land.
Next Steps
- app.toml Reference — Disks — Complete field reference for disk configuration (including
lease_timeout) - Services — Define services that use persistent storage
- Getting Started — Deploy your first app
- CLI Reference - Disk Commands — Complete disk CLI reference
- Miren Cloud — Set up cloud features