Write a ScalarDL Application with Generic Contracts
This document explains how to write ScalarDL applications with generic contracts. You will learn how to interact with ScalarDL in your applications, handle errors, and validate your data when using generic contracts in ScalarDL.
Use the ScalarDL Client SDK for generic contracts
You have two options to interact with ScalarDL when using generic contracts:
- Using commands, as shown in Use Generic Contracts and Functions
- Using the Java Client SDK
Using commands is convenient because you don't need to write applications. However, they invoke a process for each execution, which is slow, so they are mainly for quickly testing generic contracts. Instead, using the Client SDK is usually recommended when you write ScalarDL-based applications because it is more efficient.
The Client SDK is available on Maven Central. You can install it in your application by using a build tool such as Gradle. For example in Gradle, you can add the following dependency to build.gradle
, replacing VERSION
with the version of ScalarDL that you want to use.
dependencies {
implementation group: 'com.scalar-labs', name: 'scalardl-java-client-sdk', version: '<VERSION>'
}
The Client SDK APIs for generic contracts are provided by a service class called GenericContractClientService. The following is a code snippet that shows how to use GenericContractClientService
to execute a contract.
// ClientServiceFactory should always be reused.
ClientServiceFactory factory = new ClientServiceFactory();
// ClientServiceFactory creates a new GenericContractClientService object in every create method call
// but reuses the internal objects and connections as much as possible for better performance and resource usage.
GenericContractClientService service = factory.createForGenericContracts(new ClientConfig(new File(properties));
try {
// create an application-specific argument that matches the generic contract specification
JsonNode jsonArgument = ...;
ContractExecutionResult result = service.executeContract(contractId, jsonArgument);
result.getContractResult().ifPresent(System.out::println);
} catch (ClientException e) {
System.err.println(e.getStatusCode());
System.err.println(e.getMessage());
}
factory.close();
You should always use ClientServiceFactory
to create GenericContractClientService
objects. ClientServiceFactory
caches objects that are required to create GenericContractClientService
and reuses them on the basis of the given configurations, so ClientServiceFactory
object should always be reused.
GenericContractClientService
is a thread-safe client that interacts with ScalarDL components, like Ledger and Auditor, to register certificates, register contracts, execute contracts, and validate data. When you execute a generic contract, you need to specify a JsonNode
argument. For details about the specification of the input argument, see Generic Contracts and Functions Reference Guide.
Do not register and execute your custom contracts through GenericContractClientService
. Using the generic contracts and your original contracts together is not supported because the proper asset management cannot be guaranteed.
For more information about ClientServiceFactory
and GenericContractClientService
, see the scalardl-java-client-sdk
Javadoc.
Handle errors
If an error occurs in your application, the Client SDK will return an exception with a status code and an error message with an error code. You should check the status code and the error code to identify the cause of the error. For details about the status code and the error codes, see Status codes and Error codes.
Implement error handling
The SDK throws ClientException when an error occurs. You can handle errors by catching the exception as follows:
GenericContractClientService clientService = ...;
try {
// interact with ScalarDL through a ClientService object
} catch (ClientException) {
// e.getStatusCode() returns the status of the error
}
Validate your data
In ScalarDL, you occasionally need to validate your data to make sure all the data is in a valid state. Since you can learn the basics of how ScalarDL validates your data in Write a ScalarDL Application in Java, this section mainly describes differences between the validateLedger
method in the regular ClientService
and the validateLedger
method in GenericContractClientService
.
When validating assets created by generic contracts, you need to specify the type of the asset and a list of keys to identify the asset. Currently, generic contracts create two types of assets: an object (AssetType.OBJECT
) and a collection (AssetType.COLLECTION
). For keys, you can specify the object ID or collection ID.
An example code for validating an object is as follows:
GenericContractClientService service = ...
try {
LedgerValidationResult result = service.validateLedger(AssetType.OBJECT, ImmutableList.of("an_object_ID"));
// You can also specify age range.
// LedgerValidationResult result = service.validateLedger(AssetType.OBJECT, ImmutableList.of("an_object_ID"), startAge, endAge);
} catch (ClientException e) {
}
Generic contracts internally assign a dedicated asset ID to an asset record that represents an object or a collection. The asset ID consists of a prefix for the asset type and keys; for example, a prefix o_
and an object ID for AssetType.OBJECT
. Therefore, you will see such raw asset IDs in AssetProof
in LedgerValidationResult
.
Use other languages
To interact with ScalarDL in languages other than Java, you can use ScalarDL Gateway.
Documentation for ScalarDL Gateway is currently being created and will be ready in the near future.