Skip to main content

State persistence

Learn how to maintain an Actor's state to prevent data loss during unexpected restarts. Includes code examples for handling server migrations.


Long-running Actor jobs may need to migrate between servers. Without state persistence, your job's progress is lost during migration, causing it to restart from the beginning on the new server. This can be costly and time-consuming.

To prevent data loss, long-running Actors should:

  • Periodically save (persist) their state.
  • Listen for migration events
  • Check for persisted state when starting, allowing them to resume from where they left off.

For short-running Actors, the risk of restarts and the cost of repeated runs are low, so you can typically ignore state persistence.

Understanding migrations

A migration occurs when a process running on one server must stop and move to another. During this process:

  • All in-progress processes on the current server are stopped
  • Unless you've saved your state, the Actor run will restart on the new server with an empty internal state
  • You only have a few seconds to save your work when a migration event occurs

Causes of migration

Migrations can happen for several reasons:

  • Server workload optimization
  • Server crashes (rare)
  • New feature releases and bug fixes

Frequency of migrations

Migrations don't follow a specific schedule. They can occur at any time due to the events mentioned above.

Why state is lost during migration

By default, an Actor keeps its state in the server's memory. During a server switch, the run loses access to the previous server's memory. Even if data were saved on the server's disk, access to that would also be lost. Note that the Actor run's default dataset, key-value store and request queue are preserved across migrations, by state we mean the contents of runtime variables in the Actor's code.

Implementing state persistence

The Apify SDKs handle state persistence automatically.

This is done using the Actor.on() method and the migrating event.

  • The migrating event is triggered just before a migration occurs, allowing you to save your state.
  • To retrieve previously saved state, you can use the Actor.getValue/Actor.get_value methods.

Code examples

To manually persist state, use the Actor.on method in the Apify SDK:

import { Actor } from 'apify';

await Actor.init();
// ...
Actor.on('migrating', () => {
Actor.setValue('my-crawling-state', {
foo: 'bar',
});
});
// ...
await Actor.exit();

To check for state saved in a previous run:

import { Actor } from 'apify';

await Actor.init();
// ...
const previousCrawlingState = await Actor.getValue('my-crawling-state') || {};
// ...
await Actor.exit();

For improved Actor performance consider caching repeated page data.