Configuration
Cratis Arc can be configured both through appsettings.json and programmatically to customize its behavior. The main configuration is handled through the ArcOptions class.
Default Configuration Section
By default, the Arc looks for configuration under the Cratis:Arc section in your appsettings.json file.
Configuration Options
Complete Configuration Example
Here's a complete example showing all available configuration options in appsettings.json:
{
"Cratis": {
"Arc": {
"CorrelationId": {
"HttpHeader": "X-Correlation-ID"
},
"Tenancy": {
"HttpHeader": "x-cratis-tenant-id"
},
"GeneratedApis": {
"RoutePrefix": "api",
"SegmentsToSkipForRoute": 0,
"IncludeCommandNameInRoute": true,
"IncludeQueryNameInRoute": true
}
}
}
}
Configuration Properties
CorrelationId
Controls how correlation IDs are handled in HTTP requests.
- HttpHeader (string, default:
"X-Correlation-ID"): The HTTP header name to use for correlation ID tracking.
Tenancy
Controls multi-tenancy support through HTTP headers.
- HttpHeader (string, default:
"x-cratis-tenant-id"): The HTTP header name to use for tenant identification.
GeneratedApis
Controls how automatically generated API endpoints are configured for commands and queries.
- RoutePrefix (string, default:
"api"): The base route prefix for all generated API endpoints. - SegmentsToSkipForRoute (int, default:
0): Number of namespace segments to skip when constructing routes from type namespaces. - IncludeCommandNameInRoute (bool, default:
true): Whether to include the command type name as the last segment of the route for command endpoints. - IncludeQueryNameInRoute (bool, default:
true): Whether to include the query type name as the last segment of the route for query endpoints.
IdentityDetailsProvider
- IdentityDetailsProvider (Type, default:
null): Specifies a custom identity details provider type. If not specified, the system will use type discovery to find one automatically.
Setup Methods
Using Configuration File
The most common approach is to use the configuration file with the default section:
var builder = Host.CreateDefaultBuilder(args);
// Use default configuration section (Cratis:Arc)
builder.UseCratisArc();
var app = builder.Build();
app.UseCratisArc();
Using Custom Configuration Section
You can specify a custom configuration section path:
var builder = Host.CreateDefaultBuilder(args);
// Use custom configuration section
builder.UseCratisArc("MyApp:CratisConfig");
var app = builder.Build();
app.UseCratisArc();
Programmatic Configuration
You can configure the options entirely through code:
var builder = Host.CreateDefaultBuilder(args);
builder.UseCratisArc(options =>
{
// Configure correlation ID
options.CorrelationId.HttpHeader = "X-My-Correlation-ID";
// Configure tenancy
options.Tenancy.HttpHeader = "X-Tenant-ID";
// Configure generated APIs
options.GeneratedApis.RoutePrefix = "myapi";
options.GeneratedApis.SegmentsToSkipForRoute = 2;
options.GeneratedApis.IncludeCommandNameInRoute = false;
options.GeneratedApis.IncludeQueryNameInRoute = false;
// Set custom identity details provider
options.IdentityDetailsProvider = typeof(MyCustomIdentityDetailsProvider);
});
var app = builder.Build();
app.UseCratisArc();
Hybrid Configuration
You can combine configuration file settings with programmatic overrides:
var builder = Host.CreateDefaultBuilder(args);
// First bind from configuration, then override specific values
builder.UseCratisArc("Cratis:Arc")
.ConfigureServices(services =>
{
services.Configure<ArcOptions>(options =>
{
// Override specific settings programmatically
options.GeneratedApis.RoutePrefix = "v1/api";
});
});
var app = builder.Build();
app.UseCratisArc();
Environment-Specific Configuration
You can use different configurations for different environments using the standard ASP.NET Core configuration pattern:
appsettings.json (base configuration):
{
"Cratis": {
"Arc": {
"GeneratedApis": {
"RoutePrefix": "api"
}
}
}
}
appsettings.Development.json (development overrides):
{
"Cratis": {
"Arc": {
"CorrelationId": {
"HttpHeader": "X-Dev-Correlation-ID"
}
}
}
}
appsettings.Production.json (production overrides):
{
"Cratis": {
"Arc": {
"GeneratedApis": {
"RoutePrefix": "v1"
}
}
}
}
Route Generation Examples
The GeneratedApis configuration affects how routes are generated for your commands and queries. Here are some examples:
Given a command class MyApp.Sales.Commands.CreateOrderCommand:
Default Configuration
{
"GeneratedApis": {
"RoutePrefix": "api",
"SegmentsToSkipForRoute": 0,
"IncludeCommandNameInRoute": true,
"IncludeQueryNameInRoute": true
}
}
Generated route: /api/MyApp/Sales/Commands/CreateOrderCommand
Skip Namespace Segments
{
"GeneratedApis": {
"RoutePrefix": "api",
"SegmentsToSkipForRoute": 2,
"IncludeCommandNameInRoute": true,
"IncludeQueryNameInRoute": true
}
}
Generated route: /api/Sales/Commands/CreateOrderCommand
Exclude Type Names
{
"GeneratedApis": {
"RoutePrefix": "api",
"SegmentsToSkipForRoute": 3,
"IncludeCommandNameInRoute": false,
"IncludeQueryNameInRoute": false
}
}
Generated route: /api/Commands (for commands) or /api/Queries (for queries)
Note: When IncludeCommandNameInRoute or IncludeQueryNameInRoute is set to false, the system automatically detects route conflicts. If multiple commands or queries exist in the same namespace (after skipping segments), the type name will be automatically included in the route to prevent conflicts. This ensures that:
- Single command/query in a namespace: Route remains clean without the type name
- Multiple commands/queries in the same namespace: Type names are automatically added to prevent route collisions
- Both runtime endpoint mapping and proxy generation apply this logic consistently
For example, with the configuration above:
- If you have only
MyApp.Sales.Commands.CreateOrder, the route will be/api/commands - If you have both
MyApp.Sales.Commands.CreateOrderandMyApp.Sales.Commands.UpdateOrder, the routes will be/api/commands/create-orderand/api/commands/update-orderrespectively (type names added automatically to avoid conflict)
JSON Serialization
Arc provides a centralized JsonSerializerOptions configuration through ArcOptions. This ensures consistent JSON serialization across your entire application, including controller actions, manual serialization, and generated API endpoints.
Default Configuration
Arc configures JsonSerializerOptions with the following defaults:
- Property Naming: Camel case with acronym-friendly handling (e.g.,
XMLParserbecomesxmlParser) - Null Handling: Null values are ignored when writing JSON
- Enums: Serialized as integers (not strings)
- Concepts: Full support for Cratis Concepts (strongly-typed primitives)
- Date/Time: Support for
DateOnlyandTimeOnlytypes - Types: Support for
System.TypeandSystem.Uriserialization - Derived Types: Polymorphic serialization support when derived types are discovered
Accessing JsonSerializerOptions
The configured JsonSerializerOptions is available through dependency injection:
public class MyService
{
public MyService(JsonSerializerOptions jsonOptions)
{
// Use the Arc-configured options
var json = JsonSerializer.Serialize(myObject, jsonOptions);
}
}
Or through ArcOptions:
public class MyService
{
public MyService(IOptions<ArcOptions> arcOptions)
{
var jsonOptions = arcOptions.Value.JsonSerializerOptions;
}
}
Customizing JSON Serialization
You can add custom converters or modify the configuration through the options pattern:
builder.UseCratisArc(options =>
{
// Add a custom converter
options.JsonSerializerOptions.Converters.Add(new MyCustomConverter());
});
Any customizations made to ArcOptions.JsonSerializerOptions will automatically be applied to ASP.NET Core controller actions as well, ensuring consistency throughout your application.
Best Practices
Use Configuration Files: For most scenarios, use
appsettings.jsonconfiguration as it allows easy environment-specific overrides without code changes.Environment-Specific Settings: Leverage
appsettings.{Environment}.jsonfiles for environment-specific configurations.Programmatic Configuration: Use programmatic configuration when you need to:
- Set configuration based on runtime conditions
- Use custom identity providers
- Override specific settings that can't be easily expressed in JSON
Route Planning: Consider your API route structure carefully when configuring
GeneratedApisoptions, especially in public-facing APIs where route stability is important.Header Standardization: Use standard HTTP header names for correlation IDs and tenant IDs that align with your organization's conventions and any API gateways or load balancers in use.
JSON Consistency: Always use the injected
JsonSerializerOptionswhen manually serializing/deserializing JSON to maintain consistency with controller actions and generated APIs.