หมายเหตุ
การเข้าถึงหน้านี้ต้องได้รับการอนุญาต คุณสามารถลอง ลงชื่อเข้าใช้หรือเปลี่ยนไดเรกทอรีได้
การเข้าถึงหน้านี้ต้องได้รับการอนุญาต คุณสามารถลองเปลี่ยนไดเรกทอรีได้
Orleans grains use an observable lifecycle (see Orleans Lifecycle) for ordered activation and deactivation. This allows grain logic, system components, and application logic to start and stop in an ordered manner during grain activation and collection.
Stages
The predefined grain lifecycle stages are as follows:
public static class GrainLifecycleStage
{
public const int First = int.MinValue;
public const int SetupState = 1_000;
public const int Activate = 2_000;
public const int Last = int.MaxValue;
}
First: First stage in a grain's lifecycle.SetupState: Set up grain state before activation. For stateful grains, this is the stage where Orleans loads IStorage<TState>.State from storage if IStorage.RecordExists istrue.Activate: Stage where Orleans calls Grain.OnActivateAsync and Grain.OnDeactivateAsync.Last: Last stage in a grain's lifecycle.
While Orleans uses the grain lifecycle during grain activation, grains aren't always deactivated during some error cases (such as silo crashes). Therefore, applications shouldn't rely on the grain lifecycle always executing during grain deactivations.
Memory-based activation shedding
Memory-based activation shedding automatically deactivates grain activations when the silo is under memory pressure. This helps prevent out-of-memory conditions by intelligently removing grains from memory based on their activity patterns.
How it works
When enabled, Orleans monitors memory usage and begins deactivating grains when usage exceeds the configured threshold:
- Orleans polls memory usage at a configurable interval (default: 5 seconds)
- When memory usage exceeds
MemoryUsageLimitPercentage, Orleans begins deactivating grains - Orleans prioritizes deactivating older and less-recently-used grains first
- Deactivation continues until memory usage falls below
MemoryUsageTargetPercentage
Configuration
Configure memory-based activation shedding using GrainCollectionOptions:
builder.Configure<GrainCollectionOptions>(options =>
{
// Enable memory-based activation shedding
options.EnableActivationSheddingOnMemoryPressure = true;
// Memory usage percentage (0-100) at which grain collection triggers
options.MemoryUsageLimitPercentage = 80; // default: 80
// Target memory usage percentage to reach after collection
options.MemoryUsageTargetPercentage = 75; // default: 75
// How often to poll memory usage
options.MemoryUsagePollingPeriod = TimeSpan.FromSeconds(5); // default: 5s
});
GrainCollectionOptions properties
| Option | Type | Default | Description |
|---|---|---|---|
EnableActivationSheddingOnMemoryPressure |
bool |
false |
Enable automatic deactivation under memory pressure. |
MemoryUsageLimitPercentage |
int |
80 | Memory usage percentage at which shedding begins. Valid range: 0-100. |
MemoryUsageTargetPercentage |
int |
75 | Target memory usage percentage after shedding. Valid range: 0-100. |
MemoryUsagePollingPeriod |
TimeSpan | 5 seconds | How often to check memory usage. |
CollectionAge |
TimeSpan | 15 minutes | Minimum time a grain must be idle before eligible for collection. |
CollectionQuantum |
TimeSpan | 1 minute | How often idle grain collection runs. |
Best practices
- Set thresholds conservatively: Leave headroom between
MemoryUsageTargetPercentageandMemoryUsageLimitPercentageto avoid oscillation - Monitor collection metrics: Track grain deactivations to understand the impact on your workload
- Consider grain importance: Critical grains can extend their lifetime using timers with
KeepAlive = trueor by receiving periodic calls - Test under load: Verify behavior under production-like memory conditions
Grain lifecycle participation
Application logic can participate in a grain's lifecycle in two ways:
- The grain can participate in its own lifecycle.
- Components can access the lifecycle via the grain activation context (see IGrainContext.ObservableLifecycle).
- The grain can participate in its own lifecycle.
- Components can access the lifecycle via the grain activation context (see IGrainActivationContext.ObservableLifecycle).
A grain always participates in its lifecycle, so application logic can be introduced by overriding the participate method.
Example participation
public override void Participate(IGrainLifecycle lifecycle)
{
base.Participate(lifecycle);
lifecycle.Subscribe(
this.GetType().FullName,
GrainLifecycleStage.SetupState,
OnSetupState);
}
In the preceding example, Grain<TGrainState> overrides the Grain.Participate method to tell the lifecycle to call its OnSetupState method during the GrainLifecycleStage.SetupState stage of the lifecycle.
Components created during a grain's construction can also participate in the lifecycle without adding any special grain logic. Since Orleans creates the grain's context (IGrainContext), including its lifecycle (IGrainContext.ObservableLifecycle), before creating the grain, any component injected into the grain by the container can participate in the grain's lifecycle.
Components created during a grain's construction can also participate in the lifecycle without adding any special grain logic. Since Orleans creates the grain's activation context (IGrainActivationContext), including its lifecycle (IGrainActivationContext.ObservableLifecycle), before creating the grain, any component injected into the grain by the container can participate in the grain's lifecycle.
Example: Component participation
The following component participates in the grain's lifecycle when created using its factory function Create(...). This logic could exist in the component's constructor, but that risks adding the component to the lifecycle before it's fully constructed, which might not be safe.
public class MyComponent : ILifecycleParticipant<IGrainLifecycle>
{
public static MyComponent Create(IGrainContext context)
{
var component = new MyComponent();
component.Participate(context.ObservableLifecycle);
return component;
}
public void Participate(IGrainLifecycle lifecycle)
{
lifecycle.Subscribe<MyComponent>(GrainLifecycleStage.Activate, OnActivate);
}
private Task OnActivate(CancellationToken ct)
{
// Do stuff
}
}
public class MyComponent : ILifecycleParticipant<IGrainLifecycle>
{
public static MyComponent Create(IGrainActivationContext context)
{
var component = new MyComponent();
component.Participate(context.ObservableLifecycle);
return component;
}
public void Participate(IGrainLifecycle lifecycle)
{
lifecycle.Subscribe<MyComponent>(GrainLifecycleStage.Activate, OnActivate);
}
private Task OnActivate(CancellationToken ct)
{
// Do stuff
}
}
By registering the example component in the service container using its Create(...) factory function, any grain constructed with the component as a dependency has the component participate in its lifecycle without requiring any special logic in the grain itself.
Register component in container
services.AddTransient<MyComponent>(sp =>
MyComponent.Create(sp.GetRequiredService<IGrainContext>());
services.AddTransient<MyComponent>(sp =>
MyComponent.Create(sp.GetRequiredService<IGrainActivationContext>());
Grain with component as a dependency
public class MyGrain : Grain, IMyGrain
{
private readonly MyComponent _component;
public MyGrain(MyComponent component)
{
_component = component;
}
}
Grain migration
Grain migration allows grain activations to move from one silo to another while preserving their in-memory state. This enables load balancing and cluster optimization without losing grain state. The migration process involves:
- Dehydration: Serializing the grain's state on the source silo before deactivation
- Transfer: Sending the serialized state to the target silo
- Rehydration: Restoring the grain's state on the new activation before
OnActivateAsyncis called
When migration occurs
Migration can occur in several scenarios:
- Application-initiated: A grain can request migration by calling
this.MigrateOnIdle() - Activation repartitioner: Automatically collocates frequently communicating grains to optimize call locality (experimental)
- Activation rebalancer: Distributes activations to balance load across silos (experimental)
Implementing migration support
Grains can participate in migration by implementing IGrainMigrationParticipant:
public interface IGrainMigrationParticipant
{
void OnDehydrate(IDehydrationContext dehydrationContext);
void OnRehydrate(IRehydrationContext rehydrationContext);
}
Option 1: Implement IGrainMigrationParticipant directly
For grains with custom in-memory state that should be preserved during migration:
public class MyMigratableGrain : Grain, IMyGrain, IGrainMigrationParticipant
{
private int _cachedValue;
private string _sessionData;
public void OnDehydrate(IDehydrationContext dehydrationContext)
{
// Save state before migration
dehydrationContext.TryAddValue("cached", _cachedValue);
dehydrationContext.TryAddValue("session", _sessionData);
}
public void OnRehydrate(IRehydrationContext rehydrationContext)
{
// Restore state after migration
rehydrationContext.TryGetValue("cached", out _cachedValue);
rehydrationContext.TryGetValue("session", out _sessionData);
}
}
Option 2: Use Grain<TState> or IPersistentState<T>
Grains using Grain<TGrainState> or IPersistentState<TState> automatically support migration. Their state is serialized and restored without additional code:
// Automatic migration support via Grain<TState>
public class MyStatefulGrain : Grain<MyGrainState>, IMyGrain
{
public Task UpdateValue(int value)
{
State.Value = value;
return WriteStateAsync();
}
}
// Automatic migration support via IPersistentState<T>
public class MyOtherGrain : Grain, IMyGrain
{
private readonly IPersistentState<MyGrainState> _state;
public MyOtherGrain(
[PersistentState("state", "storage")] IPersistentState<MyGrainState> state)
{
_state = state;
}
}
Triggering migration
A grain can request migration to a new silo:
public class MyGrain : Grain, IMyGrain
{
public Task RequestMigration()
{
// Request migration when the grain becomes idle
this.MigrateOnIdle();
return Task.CompletedTask;
}
}
You can also specify a preferred target silo using placement hints:
public Task MigrateToSilo(SiloAddress targetSilo)
{
RequestContext.Set(IPlacementDirector.PlacementHintKey, targetSilo);
this.MigrateOnIdle();
return Task.CompletedTask;
}
Preventing automatic migration
Use the ImmovableAttribute to prevent automatic migration by the repartitioner or rebalancer:
// Prevent all automatic migration
[Immovable]
public class MyImmovableGrain : Grain, IMyGrain { }
// Prevent only repartitioner migration
[Immovable(ImmovableKind.Repartitioner)]
public class MyPartiallyImmovableGrain : Grain, IMyGrain { }
The ImmovableKind enum provides these options:
| Value | Description |
|---|---|
Repartitioner |
Won't be migrated by the activation repartitioner |
Rebalancer |
Won't be migrated by the activation rebalancer |
Any |
Won't be migrated by any automatic process (default) |
Note
The [Immovable] attribute only prevents automatic migration. User-initiated migration via MigrateOnIdle() still works.
Non-migratable grain types
The following grain types cannot be migrated:
- Client grains
- System targets
- Grain services
- Stateless worker grains