Getting Started with Cedarling#
Cedarling is a lightweight, embeddable Policy Decision Point (PDP) that enables fast, fine-grained, and decentralized access control across modern applications. Built on the Rust-based Cedar engine, Cedarling is designed for both client-side and server-side use, supporting environments like browsers, mobile apps, cloud-native services, and API gateways.
Cedarling supports both Token-Based Access Control (TBAC) using JWTs and unsigned authorization requests. In both cases, it enforces policies locally for low-latency and consistent Zero Trust security.
You can integrate Cedarling into your application using the following language libraries:
Alternatively, you can use the Cedarling Sidecar for a drop-in deployment.
From here, you can either jump directly to the language-specific examples above or continue reading for a high-level overview of how Cedarling works.
Cedarling Interfaces#
The main way you will interact with Cedarling are through the following interfaces
Initialization#
The initialization or init interface is how you will initialize Cedarling. Initialization involves loading:
Bootstrap Configuration
- A set of properties that will tells how Cedarling behaves within your application.
- Learn more in the bootstrap properties guide.
Policy Store
- A JSON file containing the schema, policies, trusted issuers, and token metadata schema used for making authorization decisions.
- Learn more in the policy store guide.
The bootstrap configuration and policy store directly influence how Cedarling performs authorization.
import initWasm, { init } from "/pkg/cedarling_wasm.js";
let cedarling = await init({});
cedarling = Cedarling(BootstrapConfig.load_from_file(
  "./bootstrap_props.json"
))
let cedarling = Cedarling::new(&BootstrapConfig::from_env().unwrap());
// Assigning a multi-line JSON string representing the Bootstrap configuration for Cedarling
val bootstrapConfig: String = """
    {
        "CEDARLING_APPLICATION_NAME": "My App",
    ...
    }
""";
val cedarling: Cedarling? = Cedarling.loadFromJson(bootstrapConfig);
// Assigning a multi-line JSON string representing the Bootstrap configuration for Cedarling
let bootstrapConfig: String = """
    {
        "CEDARLING_APPLICATION_NAME": "My App",
    ...
    }
"""
let cedarling = try Cedarling.loadFromJson(config: bootstrapConfig)
config := map[string]any{
    "CEDARLING_APPLICATION_NAME": "My App",
}
instance, err := cedarling_go.NewCedarling(config)
if err != nil {
    panic!("Error during init: %s", err)
}
import uniffi.cedarling_uniffi.*;
...
 /*
* In a production environment, the bootstrap configuration should not be hardcoded.
* Instead, it should be loaded dynamically from external sources such as environment variables,
* configuration files, or a centralized configuration service.
*/
String bootstrapJsonStr = """
    {
        "CEDARLING_APPLICATION_NAME":   "MyApp",
        ...
    }
""";
try {
    CedarlingAdapter cedarlingAdapter = new CedarlingAdapter();
    cedarlingAdapter.loadFromJson(bootstrapJsonStr);
} catch (CedarlingException e) {
    System.out.println("Unable to initialize Cedarling" + e.getMessage());
} catch (Exception e) {
    System.out.println("Unable to initialize Cedarling" + e.getMessage());
}
Authorization#
The authorization, or authz, interface is used to evaluate access control decisions by answering the question:
Is this Action, on this Resource, in this Context, allowed for these Principals?
When using Cedarling, Action and Resource are typically defined in the policy store, while Principal and Context are supplied at runtime via the authz interface.
Cedarling currently provides two modes of authorization:
Standard (Token-based) Interface
- Extracts the Principal from a JWT.
- Accepts the Context as a structured map.
const tokens = {
  access_token: "<access_token>",
  id_token: "<id_token>",
  userinfo_token: "<userinfo_token>",
};
const action = 'Jans::Action::"Read"';
const resource = {
  cedar_entity_mapping: {
    entity_type: "Jans::Application",
    id: "app_id_001"
  },
  name: "Some Application",
  url: {
    host: "example.com",
    path: "/admin-dashboard",
    protocol: "https",
  },
};
const context = {
  current_time: Date.now(),
  device_health: ["Healthy"],
  fraud_indicators: ["Allowed"],
  geolocation: ["America"],
  network: "127.0.0.1",
  network_type: "Local",
  operating_system: "Linux",
};
const result = await cedarling.authorize({
  tokens,
  action,
  resource,
  context,
});
import time
tokens = {
    "access_token": "<access_token>",
    "id_token": "<id_token>",
    "userinfo_token": "<userinfo_token>"
}
action = 'Jans::Action::"Read"'
resource = {
    "cedar_entity_mapping": {
        "entity_type": "Jans::Application",
        "id": "app_id_001"
    },
    "name": "Some Application",
    "url": {
        "host": "example.com",
        "path": "/admin-dashboard",
        "protocol": "https"
    }
}
context = {
    "current_time": int(time.time() * 1000),
    "device_health": ["Healthy"],
    "fraud_indicators": ["Allowed"],
    "geolocation": ["America"],
    "network": "127.0.0.1",
    "network_type": "Local",
    "operating_system": "Linux",
}
result = cedarling.authorize(Request(
    tokens,
    action,
    resource,
    context,
))
use std::collections::HashMap;
use chrono::Utc;
use serde_json::json;
let tokens = HashMap::from([
    ("access_token".to_string(), "<access_token>".to_string()),
    ("id_token".to_string(), "<id_token>".to_string()),
    ("userinfo_token".to_string(), "<userinfo_token>".to_string()),
]);
let action = "Jans::Action::\"Read\"".to_string();
let resource = json!({
    "cedar_entity_mapping": {
        "entity_type": "Jans::Application",
        "id": "app_id_001"
    },
    "name": "Some Application",
    "url": {
        "host": "example.com",
        "path": "/admin-dashboard",
        "protocol": "https"
    }
});
let context = json!({
    "current_time": Utc::now().timestamp_millis(),
    "device_health": ["Healthy"],
    "fraud_indicators": ["Allowed"],
    "geolocation": ["America"],
    "network": "127.0.0.1",
    "network_type": "Local",
    "operating_system": "Linux",
});
let result = cedarling.authorize(Request {
    tokens,
    action,
    resource,
    context,
});
val tokens = mapOf(
    "access_token" to "<access_token>",
    "id_token" to "<id_token>",
    "userinfo_token" to "<userinfo_token>"
)
val action = """Jans::Action::"Read""""
val resource = mapOf(
    "cedar_entity_mapping" to mapOf(
        "entity_type" to "Jans::Application",
        "id" to "app_id_001"
    ),
    "name" to "Some Application",
    "url" to mapOf(
        "host" to "example.com",
        "path" to "/admin-dashboard",
        "protocol" to "https"
    )
)
val context = mapOf(
    "current_time" to System.currentTimeMillis(),
    "device_health" to listOf("Healthy"),
    "fraud_indicators" to listOf("Allowed"),
    "geolocation" to listOf("America"),
    "network" to "127.0.0.1",
    "network_type" to "Local",
    "operating_system" to "Linux"
)
val result = cedarling?.authorize(
    tokens,
    action,
    resource["cedar_entity_mapping"]?.get("entity_type") as String,
    resource["cedar_entity_mapping"]?.get("id") as String,
    anyToJson(resource),
    context
)
let tokens: [String: String] = [
    "access_token": "<access_token>",
    "id_token": "<id_token>",
    "userinfo_token": "<userinfo_token>"
]
let action = #"Jans::Action::"Read""#
let resource: [String: Any] = [
    "cedar_entity_mapping": [
        "entity_type": "Jans::Application",
        "id": "app_id_001"
    ],
    "name": "Some Application",
    "url": [
        "host": "example.com",
        "path": "/admin-dashboard",
        "protocol": "https"
    ]
]
let context: [String: Any] = [
    "current_time": Int(Date().timeIntervalSince1970 * 1000),
    "device_health": ["Healthy"],
    "fraud_indicators": ["Allowed"],
    "geolocation": ["America"],
    "network": "127.0.0.1",
    "network_type": "Local",
    "operating_system": "Linux"
]
let payloadJsonString = try JSONSerialization.data(
    withJSONObject: resource,
    options: []
).toUtf8String()
let result: AuthorizeResult = try cedarling.authorize(
    tokens: tokens,
    action: action,
    resourceType: resource["cedar_entity_mapping"]?.get("entity_type") as! String,
    resourceId: resource["cedar_entity_mapping"]?.get("id") as! String,
    payload: payloadJsonString,
    context: context
)
tokens := map[string]string {
    "access_token": "<access_token>",
    "id_token": "<id_token>",
    "userinfo_token": "<userinfo_token>"
}
action := "Jans::Action::\"Read\""
resource := cedarling_go.EntityData{
    CedarMapping: cedarling_go.CedarMapping{
        EntityType: "Jans::Application",
        ID: "app_id_001",
    },
    Payload: map[string]any{
        "name": "Some Application",
        "url": map[string]string{
            "host": "example.com",
            "path": "/admin-dashboard",
            "protocol": "https",
        },
    },
}
context := map[string]any{
    "current_time": time.Now().Unix(),
    "device_health": ["Healthy"],
    "fraud_indicators": ["Allowed"],
    "geolocation": ["America"],
    "network": "127.0.0.1",
    "network_type": "Local",
    "operating_system": "Linux",
}
request := cedarling_go.Request{
    Tokens: tokens,
    Action: action,
    Resource: resource,
    Context: context,
}
result, err := cedarling_instance.Authorize(request)
if err != nil {
    panic!(err)
}
String resource = """
    {
      "cedar_entity_mapping": {
        "entity_type": "Jans::Application",
        "id": "app_id_001"
      },
      "name": "Some Application",
      "url": {
        "host": "example.com",
        "path": "/admin-dashboard",
        "protocol": "https"
      }
    }
""";
String action = "Jans::Action::\"Update\"";
String context = """
    {
      "device_health": ["Healthy"],
      "fraud_indicators": ["Allowed"],
      "geolocation": ["America"],
      "network": "127.0.0.1",
      "network_type": "Local",
      "operating_system": "Linux",
      "user_agent": "Linux"
    }
""";
Map<String, String> tokens = Map.of(
    "access_token", "<access_token>",
    "id_token", "<id_token>",
    "userinfo_token", "<userinfo_token>"
);
// Perform authorization
AuthorizeResult result = adapter.authorize(tokens, action, new JSONObject(resource), new JSONObject(context));
if(result.getDecision()) {
    System.out.println("Access granted");
} else {
    System.out.println("Access denied");
}
Unsigned Authorization
- Accepts the Principal directly without requiring a JWT.
- This makes authorization decisions by passing a set of Principals directly.
- Similar to the standard interface, the Context is passed in as-is in a map-like structure.
Logging#
Cedarling supports logging of both decision and system events, useful for auditing and troubleshooting. Logging is optional and can be configured (or disabled) via the bootstrap properties.
What's next?#
You're now ready to dive deeper into Cedarling. From here, you could either:
- See how you can use RBAC with Cedarling.
- Explore how to use Cedarling's Unsigned interface.
- Use the Cedarling Sidecar for a quick, zero-code deployment.
- Learn more about why Cedarling exists and the problems it solves.
