Personalization
If you want to customize the output of the Ntropy API you can do so by defining a ruleset through the
personalization API in /v2/rules/
.
A ruleset is a set of rules that change some of the fields returned by the enrichment API, such as
entity attributes or adding and removing labels. If not specified otherwise, rules are applied to all transactions.
In order to apply them to a subset of transactions, you can filter by the values that transaction fields of
this set would have. To apply to consumer or business transactions you could use account_holder_type
. To have more
fine-grained control and apply it to specific account holders or subsets thereof
please see this example.
Rules
There are two types of rules: actions and conditionals. Actions are used to change the fields of transactions. Conditionals, execute actions depending on a boolean expression. To replace the logo of a website, we could use:
- cURL
- Python SDK
$ curl \
-H "X-API-KEY: <YOUR-API-KEY>" \
-H "Content-Type: application/json" \
-X POST \
--data '[
{
"if": {
"==": [
{"get": "website"},
"ntropy.com"
]
},
"then": [
{"set": "logo", "to": "https://logos.ntropy.com/ntropy.com"}
],
}
]' \
https://api.ntropy.com/v2/rules/
rules = [
{
"if": {
"==": [
{"get": "website"},
"ntropy.com"
]
},
"then": [
{"set": "logo", "to": "https://logos.ntropy.com/ntropy.com"}
],
}
]
sdk.upload_rules(rules)
Please refer to our examples section for more examples on how to override labels in the enrichment result or how to segment rules by account holders or by subsets of their accounts.
Uploading and modifying rulesets
Rulesets are type checked when modified. They can be uploaded via POST request to
/v2/rules/
and retrieved by a GET request to the same endpoint. Only one ruleset is allowed per API key.
To add rules to a ruleset, you can do it via a POST request to
/v2/rules/append
.
Deleting and replacing individual rules can be done via
DELETE /v2/rules/{index}
and PATCH /v2/rules/{index}
respectively.
There is a size limit of 200,000 bytes for API key rulesets which is enough for approximately 8000 basic rules.
Rule Specification
Actions
There are 3 types of actions a rule can perform. They take expressions as arguments and can set general properties, modify labels and modify mccs.
Set property
{"set": "property", "to": expression}
Modify labels
You can add labels, remove labels, or replace all the labels. These expect strings.
{"set_labels": [string...]}
{"add_label": string}
{"remove_label": string}
Modify MCCs
Similarly, you can add, removes, or replace all the MCCs. These expect numbers.
{"set_mcc": [number...]}
{"add_mcc": number}
{"remove_mcc": number}
Conditionals
Execute a set of actions if condition
is true. A condition is an expression that evaluates to a boolean.
{"if": condition, "then": [rules...]}
{"if": condition, "then": [rules...], "else": [rules...]}
Note that conditionals support the specification of rules in the then
clause. This means that nested conditionals
are supported. Building on the previous example:
- cURL
- Python SDK
$ curl \
-H "X-API-KEY: <YOUR-API-KEY>" \
-H "Content-Type: application/json" \
-X POST \
--data '[
{
"if": {
"==": [
{"get": "website"},
"ntropy.com"
]
},
"then": [
{
"if": {
"==": [
{"get": "website"},
"api.ntropy.com"
]
},
"then": [
{"set": "logo", "to": "https://logos.ntropy.com/api.ntropy.com"}
],
"else": [
{"set": "logo", "to": "https://logos.ntropy.com/ntropy.com"}
]
}
],
}
]' \
https://api.ntropy.com/v2/rules/
rules = [
{
"if": {
"==": [
{"get": "website"},
"ntropy.com"
]
},
"then": [
{
"if": {
"==": [
{"get": "website"},
"api.ntropy.com"
]
},
"then": [
{"set": "logo", "to": "https://logos.ntropy.com/api.ntropy.com"}
],
"else": [
{"set": "logo", "to": "https://logos.ntropy.com/ntropy.com"}
]
}
],
}
]
sdk.upload_rules(rules)
Expressions
Expressions can have one of three types: number, boolean, or string. null
is not an allowed value.
All numbers are interpreted as floats.
Expressions can be literals, transformations or operators.
Transformations
There are currently 7 transformations supported:
transformation | args | description |
---|---|---|
get | read-property | return the value of the transaction property. it needs to be a readable property. |
is_substring | [expression, expression] | takes two strings and returns a boolean, indicating whether the 2nd argument is a substring of the 1st. |
starts_with | [expression, expression] | takes two strings and returns a boolean, indicating whether the 2nd argument is a prefix of the 1st. |
ends_with | [expression, expression] | takes two strings and returns a boolean, indicating whether the 2nd argument is a suffix of the 1st. |
to_lower | expression | takes a string converts it to lowercase. |
to_upper | expression | takes a string converts it to uppercase. |
has_label | expression | takes a string and returns a boolean |
Operators
Operators evaluate to a boolean literal and take an arbitrary number of arguments and are evaluated as
((arg1 <op> arg2) <op> arg3) <op> ...
:
{"!": expression} // logical not, accepts a boolean
{"&&": [expressions...]} // logical and, accepts a list of booleans
{"||": [expressions...]} // logical or, accepts a list of booleans
{"==": [expressions...]} // equals, accepts a list of expressions of the same type
{"+": [expressions...]} // accepts a list of numbers
{"*": [expressions...]} // accepts a list of numbers
{"/": [expressions...]} // accepts a list of numbers
{"-": [expressions...]} // accepts a list of of numbers
{"//": [expressions...]} // integer division, accepts a list of numbers
{"<": [expression, expression]} // accepts two numbers
{"<=": [expression, expression]} // accepts two numbers
{">": [expression, expression]} // accepts two numbers
{">=": [expression, expression]} // accepts two numbers
Properties
Each property corresponds to an enrichment transaction field of the same name
. write
and read
indicate whether
the property is writable and/or readable.
name | type | write | read |
---|---|---|---|
logo | string | ✔ | ✔ |
website | string | ✔ | ✔ |
merchant | string | ✔ | ✔ |
merchant_id | string | ✔ | ✔ |
location | string | ✔ | ✔ |
person | string | ✔ | ✔ |
transaction_type | string | ✔ | ✔ |
description | string | ✔ | |
amount | number | ✔ | |
entry_type | string ("incoming" or "outgoing") | ✔ | |
account_holder_type | string | ✔ | |
account_holder_id | string | ✔ | |
account_holder_name | string | ✔ |
Examples
Adding income label to interest payments
- cURL
- Python SDK
$ curl \
-H "X-API-KEY: <YOUR-API-KEY>" \
-H "Content-Type: application/json" \
-X POST \
--data '[
{
"if": {
"||": [
{"has_label": "interest"},
{"is_substring": [{"to_lower": {"get": "description"}}, "interest"]},
]
},
"then": [{"add_label": "income"}],
}
]' \
https://api.ntropy.com/v2/rules/
rules = [
{
"if": {
"||": [
{"has_label": "interest"},
{"is_substring": [{"to_lower": {"get": "description"}}, "interest"]},
]
},
"then": [{"add_label": "income"}],
}
]
sdk.upload_rules(rules)
Account holder rule segmentation
For further control over the subset of transactions rules apply to, if you only want to apply a set of
rules to an account holder ntropy
, you can use the property account_holder_id
to do so:
- cURL
- Python SDK
$ curl \
-H "X-API-KEY: <YOUR-API-KEY>" \
-H "Content-Type: application/json" \
-X POST \
--data '[
{
"if": {
"==": [
{"get": "account_holder_id"},
"ntropy"
]
},
"then": ...
}
]' \
https://api.ntropy.com/v2/rules/
rules = [
{
"if": {
"==": [
{"get": "account_holder_id"},
"ntropy"
]
},
"then": ...
}
]
sdk.upload_rules(rules)
Taking this a step further, there are some use-cases where instead of applying rules to specific account holders,
you may want to apply it to all future account holders that follow a specific pattern. This could simplify the
management of rules and account holder management on your side. To this end, you can use a combination of starts_with
and account_holder_id
. For example, if for every account under the account holder ntropy
we want to apply a set of
rules, we could adopt the naming convention of ntropy_<specific_id>
and set rules of this form:
- cURL
- Python SDK
$ curl \
-H "X-API-KEY: <YOUR-API-KEY>" \
-H "Content-Type: application/json" \
-X POST \
--data '[
{
"if": {
"starts_with": [
{"get": "account_holder_id"},
"ntropy_"
]
},
"then": ...
}
]' \
https://api.ntropy.com/v2/rules/
rules = [
{
"if": {
"starts_with": [
{"get": "account_holder_id"},
"ntropy_"
]
},
"then": ...
}
]
sdk.upload_rules(rules)