23 Managing ASP.NET Session State

You can manage ASP.NET session state in a Coherence cluster by using a Coherence session provider. The Coherence session store uses the Coherence .NET extend client to store ASP.NET sessions in Coherence caches.

This chapter includes the following sections:

Overview of ASP.NET Session State

Coherence for .NET allows ASP.NET session state to be stored and managed in a Coherence cluster, which has some benefits as compared to the out-of-the-box options offered by Microsoft.
  • Session state is stored in a highly available Coherence cluster, making sessions resilient to Web server failures.

  • Sessions are stored in memory which allows for much faster access than when they are serialized to disk using SQL Server session provider.

  • Unlike relational databases, Coherence cluster is easy to scale out to support additional load.

  • In some cases, session data can be accessed at in-process speed by leveraging Coherence near caching features.

  • Session state can be shared across multiple ASP.NET applications.

Setting Up Coherence ASP.NET Session Management

To manage ASP .NET sessions, you must enable a Coherence session provider and configure ASP session caches.

This section includes the following topics:

Configure Coherence Clusters for ASP.NET Session Management

For cluster-side support, add coherence-aspnet-session.jar in the classpath.

This will configure two distributed caches that are used for session storage and session attribute overflow, and start an Extend Proxy that an ASP.NET application can connect to. You can configure the address and port of the proxy service by using the coherence.aspnet.extend.address and coherence.aspnet.extend.port system properties.

Configure ASP.NET Applications

The Coherence session provider requires an extend client's cache configuration file to include remote cache schemes for the session storage and session overflow caches. As with any remote cache, the cache name on the server and the client must be same name. The out of the box configuration is ready to be used with the default server-side configuration.

After adding Coherence.dll to the solution, you have to enable the session middleware. To do so, Program.cs must contain:
  • A call to configure Coherence ASP.NET session support

  • A call to AddSession to add services required for application session state

  • A call to UseSession to automatically enable session state for the application

You can configure Coherence ASP.NET session support either by specifying configuration options in the CoherenceSession section of the appsettings.json file or by directly specifying configuration options in the UseCoherenceSession call.

appsettings.json example

{
  "AllowedHosts": "*",
  "CoherenceSession": {
    "ApplicationId": "demo-app",
    "Model": "split",
    "MinOverflowAttributeSize": 10240
  }
}

Program.cs example

var builder = WebApplication.CreateBuilder(args);

builder.Services.AddControllers();

// configure Coherence session support based on Coherence configuration specified in appsettings.json:
builder.Services.UseCoherenceSession(builder.Configuration.GetSection(CoherenceSessionOptions.CONFIG));

// or configure Coherence session support directly in the code:
builder.Services.UseCoherenceSession(options =>
    {
        options.ApplicationId = "demo-app";
        options.Model = CoherenceSessionOptions.SessionModel.Split;
        options.MinOverflowAttributeSize = 10240;
    }
);
// uncomment when using .NET 8:
/*
builder.Services.AddDataProtection()
        .PersistKeysToFileSystem(new DirectoryInfo(@"/tmp"))
        .SetApplicationName("DemoApp");
*/
builder.Services.AddSession(options =>
    {
        options.IdleTimeout = TimeSpan.FromSeconds(900);
	options.Cookie.Name = ".DemoApp.Session";
        options.Cookie.HttpOnly = true;
        options.Cookie.IsEssential = true;
    }
);

var app = builder.Build();

// Adds the SessionMiddleware automatically enable session state for the application
app.UseSession();

app.MapControllers();
app.Run();

Configuration options:

  • Model

    • Monolithic - Stores all session state as a single entity, serializing and deserializing all attributes as a single operation.

    • Split - Separates the larger session attributes into independent physical entities, one cache entry per attribute, and stores them in separate caches.

  • MinOverflowAttributeSize - Defines the minimum size of the attributes that will be stored separately (applicable only for the Split session model).

  • CoherenceConfig - Path to the Coherence configuration file (by default, set to assembly://Coherence/Tangosol.Config/coherence-config.xml).

  • CacheConfig - Path to the Coherence cache configuration file (by default, set to assembly://Coherence.SessionStore/Tangosol.Config/coherence-aspnet-cache-config.xml).

  • ApplicationId - Application ID that is used as a part of a session key. If not set, then the IHostEnvironment.ApplicationName value will be used instead.

If needed, you can provide the custom configuration, which has to be in sync with the server-side configuration, by overriding the default CoherenceConfig and CacheConfig files.

Selecting a Session Model

You can configure a Coherence session provider to store session state using different models depending on an application’s requirements.

This section includes the following topics:

Overview of Session Models

A session model describes how the Coherence session provider physically represents and stores session state in the cluster. The provider includes two different session model implementations out of the box:

  • Monolithic Model – Stores all session state as a single entity, serializing and deserializing all attributes as a single operation.

  • Split Model – Stores smaller session attributes in a single entity, just like the monolithic model, but separates the larger session attributes into independent physical entities, one entry per attribute, stored in a separate cache.

The monolithic model is the default.

Specify the Session Model

You configure the session model by setting the Model configuration option either in Program.cs or in the CoherenceSession section of appsettings.json file. When using the split model, you can configure the minimum size of the attributes that will be stored separately by setting the MinOverflowAttributeSize option to the minimum size (by default, it is set to 1024).

Sharing ASP.NET Session State Across Applications

In some cases, it is beneficial to share sessions across ASP.NET applications. By default, a session key is determined by combining the application identifier (as returned by the IHostEnvironment.ApplicationName property) with the session identifier. This effectively prevents session sharing.

In order to share sessions, applications must have the same application ID. You can configure Coherence to use a specific application identifier by setting the applicationId option of the CoherenceSessionOptions instance or by specifying CoherenceSession.ApplicationId in the appsettings.json.

Advanced Configuration

When defining custom session storage and session overflow caches, the cache names must be aspnet-session-storage and aspnet-session-overflow, respectively. In addition, the storage cache must be configured to use the ConfigurablePofContext class as the serializer.

Default cluster cache configuration coherence-aspnet-config.xml file

<cache-config xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
              xmlns="http://xmlns.oracle.com/coherence/coherence-cache-config"
              xsi:schemaLocation="http://xmlns.oracle.com/coherence/coherence-cache-config coherence-cache-config.xsd"
              xml-override="{coherence.aspnet.cacheconfig.override}">

  <defaults>
    <scope-name system-property="coherence.aspnet.scope">AspNet</scope-name>
    <serializer>pof</serializer>
  </defaults>

  <caching-scheme-mapping>
    <cache-mapping>
      <cache-name>aspnet-session-storage</cache-name>
      <scheme-name>aspnet-session-storage</scheme-name>
      <interceptors>
        <interceptor>
          <name>session-cleanup-interceptor</name>
          <instance>
            <class-name>com.oracle.coherence.aspnet.session.internal.SessionCleanupInterceptor</class-name>
          </instance>
        </interceptor>
      </interceptors>
    </cache-mapping>

    <cache-mapping>
      <cache-name>aspnet-session-overflow</cache-name>
      <scheme-name>aspnet-session-overflow</scheme-name>
      <interceptors>
        <interceptor>
          <name>index-interceptor</name>
          <instance>
            <class-name>com.oracle.coherence.aspnet.session.internal.IndexInterceptor</class-name>
          </instance>
        </interceptor>
      </interceptors>
    </cache-mapping>
  </caching-scheme-mapping>

  <caching-schemes>
    <distributed-scheme>
      <scheme-name>aspnet-session-storage</scheme-name>
      <service-name>Sessions</service-name>
      <backing-map-scheme>
        <local-scheme>
          <unit-calculator>BINARY</unit-calculator>
        </local-scheme>
      </backing-map-scheme>
      <autostart>true</autostart>
    </distributed-scheme>

    <distributed-scheme>
      <scheme-name>aspnet-session-overflow</scheme-name>
      <service-name>Sessions</service-name>
      <backing-map-scheme>
        <local-scheme>
          <unit-calculator>BINARY</unit-calculator>
        </local-scheme>
      </backing-map-scheme>
    </distributed-scheme>

    <proxy-scheme>
      <scheme-name>aspnet-proxy</scheme-name>
      <service-name>SessionProxy</service-name>
      <acceptor-config>
        <tcp-acceptor>
          <local-address>
            <address system-property="coherence.aspnet.extend.address"/>
            <port system-property="coherence.aspnet.extend.port"/>
          </local-address>
        </tcp-acceptor>
      </acceptor-config>
      <autostart>true</autostart>
    </proxy-scheme>
  </caching-schemes>
</cache-config>

Default .NET client cache configuration coherence-aspnet-cache-config.xml file

<cache-config xmlns="http://schemas.tangosol.com/cache">
  <caching-scheme-mapping>
    <cache-mapping>
      <cache-name>aspnet-session-storage</cache-name>
      <scheme-name>aspnet-session-remote</scheme-name>
    </cache-mapping>
    <cache-mapping>
      <cache-name>aspnet-session-overflow</cache-name>
      <scheme-name>aspnet-session-remote</scheme-name>
    </cache-mapping>
  </caching-scheme-mapping>

  <caching-schemes>
    <remote-cache-scheme>
      <scheme-name>aspnet-session-remote</scheme-name>
      <service-name>AspNet:SessionProxy</service-name>

      <initiator-config>
        <tcp-initiator>
          <name-service-addresses>
            <socket-address>
              <address system-property="coherence.ns.address">127.0.0.1</address>
              <port system-property="coherence.ns.port">7574</port>
            </socket-address>
          </name-service-addresses>
        </tcp-initiator>
        <serializer>
          <class-name>Tangosol.IO.Pof.ConfigurablePofContext, Coherence</class-name>
          <init-params>
            <init-param>
              <param-type>string</param-type>
              <param-value>assembly://Coherence.SessionStore/Tangosol.Config/coherence-aspnet-pof-config.xml</param-value>
            </init-param>
          </init-params>
        </serializer>
      </initiator-config>
    </remote-cache-scheme>
  </caching-schemes>
</cache-config>

Registering the Event Interceptors

Session attributes are stored in two caches when utilizing the split session model. Small HTTP session attributes are managed within one cache and large attributes are split out into another cache. This allows support for very large HTTP session objects without incurring overhead for frequently accessed small attributes. Therefore; the event listener (com.oracle.coherence.aspnet.session.internal.SessionCleanupInterceptor) is recommended to keep both caches synchronized. This ensures that if a session is terminated explicitly by the user and removed by eviction or expiry, that both the removal of the small and large segments of the session are coherently removed from the two caches.

To speed up access to the cached data, it is recommended that you create an index on the overflow cache. The com.oracle.coherence.aspnet.session.internal.IndexInterceptor interceptor will create an index aftert the overflow cache is created.

The default configuration is preconfigured with both interceptors.