Table of Contents

Installation

Note: You will need a Software Vendor account at elm.evoleap.com to use the elm client API. Contact evoleap to open an account.

The elm client API for .NET is distributed via NuGet.

Install-Package evoleap.Licensing -Pre

Licensing the server

First, embed your application's product id and version somewhere in the server code. You can get the product id from the elm web portal.

static class Constants
{
    internal static readonly Guid ProductId = new Guid("XXXXXXXXXXXX-XXXX-XXXX-XXXX-XXXXXXXX");
    internal static readonly string ProductVersion = "1.0.0.0";
}

Then download the public key file for the product from the elm web portal and load it into your application:

// From configuration:
var keyFile = PublicKeyFile.Create(WebConfigurationManager.AppSettings["PublicKey"]);

// From a resource string:
var keyFile = PublicKeyFile.Create(Resources.PublicKey);

// From an external file:
var keyFile = PublicKeyFile.ReadFrom(filePath);

Next, load the license key. How that happens is up to you, but typically a server would load it from configuration or show an initial setup screen asking for the license key.

var licenseKey = WebConfigurationManager.AppSettings["LicenseKey"];

Once you have the license key, create an instance of the ServerControlManager class, passing in the product id and version, license key, public key file, any configuration options, and any saved state.

var controlManager = new ServerControlManager(
  Constants.ProductId,
  Constants.ProductVersion, 
  keyFile,
  controlStrategy: <strategyObject>,
  savedState: <stateObject>);

Next, check the ServerControlManager.ServerState.Registered property to determine whether registration is required.

bool shouldRegister;
if (controlManager.ServerState.Registered) {
  shouldRegister = false;
}
else {
  shouldRegister = true;
}
Note: if your application supports an unregistered grace period, these registration steps are not required until the grace period has expired.

Next, call the RegisterAsync method to register the current instance of the application with the licensing system. If the current instance is already registered, this call does not contact the server.

if (shouldRegister) {
  // Use current hardware info as instance identity:
  var instanceIdentity = await HardwareBasedInstanceIdentity.GetCurrentAsync();

  // Use custom identity
  var instanceIdentity = new InstanceIdentity();
  instanceIdentity.Data.Add("key", "value");
  ...
  
  var result = await controlManager.RegisterAsync(licenseKey, instanceIdentity);
  if (!result.Success) {
    // Not registered, deny access or try again.
  }

Once registration succeeds, call the ValidateSessionAsync method to verify that the user is authorized to use the application and start a new session.

var validity = await controlManager.ValidateSessionAsync(instanceIdentity);
if (validity.IsValid) {
  // Allow access
}
else {
  // Deny access
}

Storing license information

The client API provides full support for remembering registration and validation information across sessions of an application. Storing licensing information is highly recommended, as it reduces the number of calls to the licensing server on application startup and enables the client to detect if the user is tampering with the system clock or trying to spoof hardware. However, the data must be stored securely, as third-party modifications of the stored data could allow users access to the application without authorization.

The IServerState interfaces define all of the data that must be stored.

Important: The application should update the stored version of the data after every call to ServerControlManager.ValidateSessionAsync method.

// Update stored data after every ControlManager call:

await controlManager.RegisterInstanceAsync(...);
StoreLicensingData(controlManager);

// Later

await controlManager.ValidateSessionAsync(...);
StoreLicensingData(controlManager);

// etc.

void StoreLicensingData(ServerControlManager controlManager)
{
    var data = controlManager.ServerState;
    SaveDataSomewhere(data);
}

The ServerControlManager can be initialized with the saved state by passing in a IServerState object to its constructor:

var controlManager = new ServerControlManager(productId, licenseKey, keyFile, savedState: <state>);