Document updated on Feb 9, 2023
The policies engine lets you write custom rules validated on runtime during requests, responses, and token validation. The policies allow you to implement all sorts of validations and user access control, from parameter compliance, to RBAC (Role-Based Access Control) and ABAC (Attribute-Based Access Control) strategies.
A policy is a test rule based on the CEL language and advanced macros that evaluates the gateway’s input or output and determines if it can continue executing or if it must break the flow with an error. For example, a simple policy could look like this:
has(JWT.department) && JWT.department in ["marketing", "sales"]
The policy above checks if the JWT token contains a claim named department
and its value is one of the listed ones. If it doesn’t there is a configurable error. The JWT validator would have previously validated that the token is valid, signed, and so on.
Policies extend the rest of existing features (security related or not) like token validation, CORS, HTTPS headers, authentication, parameter validation, etc., with your set of rules.
The rules are business logic that you will apply to your API. These rules have access to query strings, URLs, parameters, cookies, tokens, time functions, geolocation, cryptography, and a long etcetera, and you can combine all this data.
For instance, real-world examples could be:
/blackfriday
will become available and unavailable automatically based on dates and without any intervention.Because CEL evaluates an expression from the Abstract Syntax Tree in nanoseconds to microseconds, the ideal use case for CEL is applications with performance-critical paths. For example, executing a security policy with each HTTP request to a service is a perfect use case for CEL because the security policy does not change constantly, and CEL will have a negligible impact on the response time.
The rules are pre-compiled to the AST during startup time, so when the rule is evaluated, there is only the execution part. The interpretation is never needed as it was performed when the service started.
CEL is maintained by Google and used in hundreds of mission-critical projects like Google Cloud, Openshift, Digital Ocean, or Kubernetes. Even though there are other popular policy engines, we chose CEL because of its simplicity, extensibility, and, most importantly, security and performance.
JavaScript and Lua are rich languages that require sandboxing to execute safely. Unfortunately, sandboxing is costly and heavily factors into the “what will I let users evaluate?” question when the answer is anything more than O(n) complexity.
CEL evaluates linearly concerning the expression size and the input evaluated when macros are disabled. The host environment provides the only functions beyond the built-ins you can invoke.
Then, why not WASM? WASM is an excellent choice for specific applications. It is far superior to embedded JavaScript and Lua, but it does not support garbage collection, and non-primitive object types require semi-expensive calls across modules.
In most cases, CEL will be faster and more portable for its intended use case.
You can set policies between the end-user and KrakenD via endpoint policies, or you can set policies between KrakenD and the services you want to consume via backend policies. Policies can work independently or together in two directions, both in the request and response.
You decide in which direction you want to apply the policies by setting them inside the proper context:
jwt
: During the token validation (endpoint
only) in auth/validator
. If you don’t have any JWT validation configuration, policies remain disabled. This context evaluates in the first place.req
: During the requestresp
: Before delivering the final responseThe following example shows the three different places to apply policies with a few checks at various stages during endpoint and backend processing. The code is @commented
to understand each evaluation block:
{
"version": 3,
"$schema": "https://www.krakend.io/schema/v2.2/krakend.json",
"debug_endpoint": true,
"extra_config": {
"router": {
"return_error_msg": true
}
},
"endpoints": [
{
"endpoint": "/legal/document/{docId}",
"input_query_strings": [
"q"
],
"extra_config": {
"security/policies": {
"auto_join_policies": true,
"debug": false,
"req": {
"@comment": "Validates that the URL parameter docId begins with 'EU' and there is a ?q= parameter",
"policies": [
"req_params.DocId.startsWith('EU')",
"hasQuerystring('q')"
],
"error": {
"@comment": "If any rule fails, we will return this JSON body and content-type. Plus an HTTP status code 400",
"body": "{\"message\":\"wrong request, try again\"}",
"status": 400,
"content_type": "application/json"
}
},
"resp": {
"@comment": "Check that when we get the response, contains all involved backend calls",
"policies": [
"resp_completed"
],
"error": {
"body": "The service didn't return all the requested information",
"status": 500,
"content_type": "text/plain"
}
},
"jwt": {
"@comment": "The token was already validated, but user is from legal?",
"policies": [
"has(JWT.user_id)",
"'legal' == JWT.department"
]
}
},
"auth/validator": {
"@comment": "Add here your auth/validator settings when using 'jwt' policies"
}
},
"backend": [
{
"url_pattern": "/cms/contracts/docid/{docId}",
"extra_config": {
"security/policies": {
"auto_join_policies": true,
"debug": false,
"req": {
"@comment": "DocId is like EU123456Z. We could have done this earlier too.",
"policies": [
"req_params.DocId.isAlphanumeric()"
],
"error": {
"body": "The document requested does not exist",
"status": 400,
"content_type": "text/plain"
}
},
"resp": {
"policies": [
"!isEmpty(resp_data.message)"
],
"error": {
"body": "We received an empty response from the service",
"status": 500
}
}
}
}
}
]
}
]
}
Notice the presence of return_error_msg
when we want to return a custom body to the end-user.
These are the options available in the component:
req
, or
resp
, or
jwt
| When true, all policies of the same type concatenate with an AND operation to evaluate a single expression. Performs faster, but its harder the debug. Defaults to false | ||||||||||
| When true, all the inputs and evaluation results are printed in the console. Defaults to false | ||||||||||
| Advanced macros can be disabled in those policies not needing them for a faster evaluation. Defaults to false | ||||||||||
| All the policies applied in the JWT context (token validation). You must configure auth/validator for the policies to run, otherwise they will be skipped. Any policy failing will generate a 401 Unauthorized error. Works in endpoint context only, this option is not available under backend .
| ||||||||||
| All the policies applied in the request context.
| ||||||||||
| All the policies applied in the response context.
|
The documentation is only a piece of the help you can get! Whether you are looking for Open Source or Enterprise support, see more support channels that can help you.