diff --git a/website/docs/advanced/config-v2-sdk-compatibility.md b/website/docs/advanced/config-v2-sdk-compatibility.md index 6a21e598..651e1e20 100644 --- a/website/docs/advanced/config-v2-sdk-compatibility.md +++ b/website/docs/advanced/config-v2-sdk-compatibility.md @@ -7,13 +7,13 @@ description: This table shows which SDK versions support Config V2. This table shows which SDK versions support Config V2. | SDK | Version | Release Notes | -| ------------------------------- | ------- | --------------------------------------------------------------- | +| ------------------------------- |---------| --------------------------------------------------------------- | | .NET | v9.0.0 | https://github.com/configcat/.net-sdk/releases | | C++ | - | https://github.com/configcat/cpp-sdk/releases | | Dart (Flutter) | - | https://github.com/configcat/dart-sdk/releases | | Elixir | - | https://github.com/configcat/elixir-sdk/releases | | Go | - | https://github.com/configcat/go-sdk/releases | -| Java | - | https://github.com/configcat/java-sdk/releases | +| Java | v9.0.0 | https://github.com/configcat/java-sdk/releases | | Java (Android) | - | https://github.com/configcat/android-sdk/releases | | JavaScript | v9.0.0 | https://github.com/configcat/js-sdk/releases | | JavaScript (Chromium Extension) | v2.0.0 | https://github.com/configcat/js-chromium-extension-sdk/releases | diff --git a/website/docs/advanced/proxy/endpoints.md b/website/docs/advanced/proxy/endpoints.md index bbda7ebf..32055563 100644 --- a/website/docs/advanced/proxy/endpoints.md +++ b/website/docs/advanced/proxy/endpoints.md @@ -86,24 +86,24 @@ var configCatClient = configcat.getClient( The following SDK versions are supported by the `>=v0.3.X` Proxy's CDN endpoint: -| SDK | Version | -| ------- | ----------------------------------------------------- | -| .NET | [`>=v9.0.0`](https://github.com/configcat/.net-sdk/releases/tag/v9.0.0) | -| JS | [`>=v9.0.0`](https://github.com/configcat/js-sdk/releases/tag/v9.0.0) | -| JS SSR | [`>=v8.0.0`](https://github.com/configcat/js-ssr-sdk/releases/tag/v8.0.0) | -| React | [`>=v4.0.0`](https://github.com/configcat/react-sdk/releases/tag/v4.0.0) | -| Node | [`>=v11.0.0`](https://github.com/configcat/node-sdk/releases/tag/v11.0.0) | -| Python | [`>=v9.0.0`](https://github.com/configcat/python-sdk/releases/tag/v9.0.0) | -| Go | [`>=v9.0.0`](https://github.com/configcat/go-sdk/releases/tag/v9.0.0) | -| C++ | TBA | -| Dart | TBA | -| Elixir | TBA | -| Java | TBA | -| Android | TBA | -| Kotlin | TBA | -| PHP | TBA | -| Ruby | TBA | -| Swift | TBA | +| SDK | Version | +| ------- |---------------------------------------------------------------------------| +| .NET | [`>=v9.0.0`](https://github.com/configcat/.net-sdk/releases/tag/v9.0.0) | +| JS | [`>=v9.0.0`](https://github.com/configcat/js-sdk/releases/tag/v9.0.0) | +| JS SSR | [`>=v8.0.0`](https://github.com/configcat/js-ssr-sdk/releases/tag/v8.0.0) | +| React | [`>=v4.0.0`](https://github.com/configcat/react-sdk/releases/tag/v4.0.0) | +| Node | [`>=v11.0.0`](https://github.com/configcat/node-sdk/releases/tag/v11.0.0) | +| Python | [`>=v9.0.0`](https://github.com/configcat/python-sdk/releases/tag/v9.0.0) | +| Go | [`>=v9.0.0`](https://github.com/configcat/go-sdk/releases/tag/v9.0.0) | +| C++ | TBA | +| Dart | TBA | +| Elixir | TBA | +| Java | [`>=v9.0.0`](https://github.com/configcat/java-sdk/releases/tag/v9.0.0) | +| Android | TBA | +| Kotlin | TBA | +| PHP | TBA | +| Ruby | TBA | +| Swift | TBA | diff --git a/website/docs/sdk-reference/java.md b/website/docs/sdk-reference/java.md index 53ac86ce..26ccd8af 100644 --- a/website/docs/sdk-reference/java.md +++ b/website/docs/sdk-reference/java.md @@ -26,7 +26,7 @@ export const JavaSchema = require('@site/src/schema-markup/sdk-reference/java.js ```groovy title="build.gradle" dependencies { - implementation 'com.configcat:configcat-java-client:8.+' + implementation 'com.configcat:configcat-java-client:9.+' } ``` @@ -37,7 +37,7 @@ dependencies { com.configcat configcat-java-client - [8.0.0,) + [9.0.0,) ``` @@ -201,16 +201,16 @@ client.getValueDetailsAsync( The details result contains the following information: -| Property | Type | Description | -| -------------------------------------- | --------------------------------------- | ----------------------------------------------------------------------------------------- | -| `getValue()` | `boolean` / `String` / `int` / `double` | The evaluated value of the feature flag or setting. | -| `getKey()` | `String` | The key of the evaluated feature flag or setting. | -| `isDefaultValue()` | `boolean` | True when the default value passed to getValueDetails() is returned due to an error. | -| `getError()` | `String` | In case of an error, this field contains the error message. | -| `getUser()` | `User` | The user object that was used for evaluation. | -| `getMatchedEvaluationPercentageRule()` | `PercentageRule` | If the evaluation was based on a percentage rule, this field contains that specific rule. | -| `getMatchedEvaluationRule()` | `RolloutRule` | If the evaluation was based on a targeting rule, this field contains that specific rule. | -| `getFetchTimeUnixMilliseconds()` | `long` | The last download time of the current config in unix milliseconds format. | +| Property | Type | Description | +|------------------------------------|-----------------------------------------|------------------------------------------------------------------------------------------------------------| +| `getValue()` | `boolean` / `String` / `int` / `double` | The evaluated value of the feature flag or setting. | +| `getKey()` | `String` | The key of the evaluated feature flag or setting. | +| `isDefaultValue()` | `boolean` | True when the default value passed to getValueDetails() is returned due to an error. | +| `getError()` | `String` | In case of an error, this field contains the error message. | +| `getUser()` | `User` | The user object that was used for evaluation. | +| `getMatchedPercentageOption()` | `PercentageOption` | The percentage option (if any) that was used to select the evaluated value. | +| `getMatchedTargetingRule()` | `TargetingRule` | The targeting rule (if any) that matched during the evaluation and was used to return the evaluated value. | +| `getFetchTimeUnixMilliseconds()` | `long` | The last download time of the current config in unix milliseconds format. | ## User Object @@ -232,7 +232,7 @@ User user = User.newBuilder().build("john@example.com"); | `custom()` | Optional dictionary for custom attributes of a user for advanced targeting rule definitions. e.g. User role, Subscription type. | ```java -java.util.Map customAttributes = new java.util.HashMap(); +java.util.Map customAttributes = new java.util.HashMap(); customAttributes.put("SubscriptionType", "Pro"); customAttributes.put("UserRole", "Admin"); @@ -243,6 +243,50 @@ User user = User.newBuilder() .build("#UNIQUE-USER-IDENTIFIER#"); ``` +The `Custom` dictionary also allows attribute values other than `String` values: + +```java +java.util.Map customAttributes = new java.util.HashMap(); + customAttributes.put("Rating", 4.5); + customAttributes.put("RegisteredAt", new Date("2023-11-22 12:34:56 +00:00")); + customAttributes.put("Roles", new String[]{"Role1", "Role2"}); + +User user = User.newBuilder() + .email("john@example.com") + .country("United Kingdom") + .custom(customAttributes) + .build("#UNIQUE-USER-IDENTIFIER#"); +``` +### User Object Attribute Types + +All comparators support `String` values as User Object attribute (in some cases they need to be provided in a specific format though, see below), but some of them also support other types of values. It depends on the comparator how the values will be handled. The following rules apply: + +**Text-based comparators** (EQUALS, IS ONE OF, etc.) +* accept `String` values, +* all other values are automatically converted to `String` (a warning will be logged but evaluation will continue as normal). + +**SemVer-based comparators** (IS ONE OF, <, >=, etc.) +* accept `String` values containing a properly formatted, valid semver value, +* all other values are considered invalid (a warning will be logged and the currently evaluated targeting rule will be skipped). + +**Number-based comparators** (=, <, >=, etc.) +* accept `Double` values and all other numeric values which can safely be converted to `Double`, +* accept `String` values containing a properly formatted, valid `Double` value, +* all other values are considered invalid (a warning will be logged and the currently evaluated targeting rule will be skipped). + +**Date time-based comparators** (BEFORE / AFTER) +* accept `Date` or `Instant` values, which are automatically converted to a second-based Unix timestamp, +* accept `Double` values representing a second-based Unix timestamp and all other numeric values which can safely be converted to `Double`, +* accept `String` values containing a properly formatted, valid `Double` value, +* all other values are considered invalid (a warning will be logged and the currently evaluated targeting rule will be skipped). + +**String array-based comparators** (ARRAY CONTAINS ANY OF / ARRAY NOT CONTAINS ANY OF) +* accept arrays and list of `String`, +* accept `String` values containing a valid JSON string which can be deserialized to an array of `String`, +* all other values are considered invalid (a warning will be logged and the currently evaluated targeting rule will be skipped). + + + ### Default User There's an option to set a default user object that will be used at feature flag and setting evaluation. It can be useful when your application has a single user only, or rarely switches users. @@ -463,8 +507,169 @@ It allows the usage of all features you can do on the ConfigCat Dashboard. You can download your current config JSON from ConfigCat's CDN and use it as a baseline. -The URL to your current config JSON is based on your [Data Governance](advanced/data-governance.md) settings: + + + +A convenient way to get the config JSON for a specific SDK Key is to install the [ConfigCat CLI](https://github.com/configcat/cli) tool +and execute the following command: + +```bash +configcat config-json get -f v6 -p {YOUR-SDK-KEY} > config.json +``` + +(Depending on your [Data Governance](advanced/data-governance.md) settings, you may need to add the `--eu` switch.) + +Alternatively, you can download the config JSON manually, based on your [Data Governance](advanced/data-governance.md) settings: + +- GLOBAL: `https://cdn-global.configcat.com/configuration-files/{YOUR-SDK-KEY}/config_v6.json` +- EU: `https://cdn-eu.configcat.com/configuration-files/{YOUR-SDK-KEY}/config_v6.json` +```json +{ + "p": { + // hash salt, required only when sensitive text comparator(s) are used + "s": "80xCU/SlDz1lCiWFaxIBjyJeJecWjq46T4eu6GtozkM=" + }, + "s": [ // array of segments + { + "n": "Beta Users", // segment name + "r": [ // array of user conditions (there is a logical AND relation between the elements) + { + "a": "Email", // comparison attribute + "c": 0, // comparator (see below) + "l": [ // comparison value (see below) + "john@example.com", "jane@example.com" + ] + } + ] + } + ], + "f": { // key-value map of feature flags & settings + "isFeatureEnabled": { // key of a particular flag / setting + "t": 0, // setting type, possible values: + // 0 -> on/off setting (feature flag) + // 1 -> text setting + // 2 -> whole number setting + // 3 -> decimal number setting + "r": [ // array of targeting rules (there is a logical OR relation between the elements) + { + "c": [ // array of conditions (there is a logical AND relation between the elements) + { + "u": { // user condition + "a": "Email", // comparison attribute + "c": 2, // comparator, possible values and required comparison value types: + // 0 -> IS ONE OF (cleartext) + string array comparison value ("l") + // 1 -> IS NOT ONE OF (cleartext) + string array comparison value ("l") + // 2 -> CONTAINS ANY OF (cleartext) + string array comparison value ("l") + // 3 -> NOT CONTAINS ANY OF (cleartext) + string array comparison value ("l") + // 4 -> IS ONE OF (semver) + semver string array comparison value ("l") + // 5 -> IS NOT ONE OF (semver) + semver string array comparison value ("l") + // 6 -> < (semver) + semver string comparison value ("s") + // 7 -> <= (semver + semver string comparison value ("s") + // 8 -> > (semver) + semver string comparison value ("s") + // 9 -> >= (semver + semver string comparison value ("s") + // 10 -> = (number) + number comparison value ("d") + // 11 -> <> (number + number comparison value ("d") + // 12 -> < (number) + number comparison value ("d") + // 13 -> <= (number + number comparison value ("d") + // 14 -> > (number) + number comparison value ("d") + // 15 -> >= (number) + number comparison value ("d") + // 16 -> IS ONE OF (hashed) + string array comparison value ("l") + // 17 -> IS NOT ONE OF (hashed) + string array comparison value ("l") + // 18 -> BEFORE (UTC datetime) + second-based Unix timestamp number comparison value ("d") + // 19 -> AFTER (UTC datetime) + second-based Unix timestamp number comparison value ("d") + // 20 -> EQUALS (hashed) + string comparison value ("s") + // 21 -> NOT EQUALS (hashed) + string comparison value ("s") + // 22 -> STARTS WITH ANY OF (hashed) + string array comparison value ("l") + // 23 -> NOT STARTS WITH ANY OF (hashed) + string array comparison value ("l") + // 24 -> ENDS WITH ANY OF (hashed) + string array comparison value ("l") + // 25 -> NOT ENDS WITH ANY OF (hashed) + string array comparison value ("l") + // 26 -> ARRAY CONTAINS ANY OF (hashed) + string array comparison value ("l") + // 27 -> ARRAY NOT CONTAINS ANY OF (hashed) + string array comparison value ("l") + // 28 -> EQUALS (cleartext) + string comparison value ("s") + // 29 -> NOT EQUALS (cleartext) + string comparison value ("s") + // 30 -> STARTS WITH ANY OF (cleartext) + string array comparison value ("l") + // 31 -> NOT STARTS WITH ANY OF (cleartext) + string array comparison value ("l") + // 32 -> ENDS WITH ANY OF (cleartext) + string array comparison value ("l") + // 33 -> NOT ENDS WITH ANY OF (cleartext + string array comparison value ("l") + // 34 -> ARRAY CONTAINS ANY OF (cleartext) + string array comparison value ("l") + // 35 -> ARRAY NOT CONTAINS ANY OF (cleartext) + string array comparison value ("l") + "l": [ // comparison value - depending on the comparator, another type of value may need + // to be specified (see above): + // "s": string + // "d": number + "@example.com" + ] + } + }, + { + "p": { // prerequisite flag condition + "f": "mainIntFlag", // key of prerequisite flag + "c": 0, // comparator, possible values: 0 -> EQUALS, 1 -> NOT EQUALS + "v": { // comparison value (value's type must match the prerequisite flag's type) + "i": 42 + } + } + }, + { + "s": { // segment condition + "s": 0, // segment index, a valid index into the top-level segment array ("s") + "c": 1 // comparator, possible values: 0 -> IS IN SEGMENT, 1 -> IS NOT IN SEGMENT + } + } + ], + "s": { // alternatively, an array of percentage options ("p", see below) can also be specified + "v": { // the value served when the rule is selected during evaluation + "b": true + }, + "i": "bcfb84a7" + } + } + ], + "p": [ // array of percentage options + { + "p": 10, // % value + "v": { // the value served when the percentage option is selected during evaluation + "b": true + }, + "i": "bcfb84a7" + }, + { + "p": 90, + "v": { + "b": false + }, + "i": "bddac6ae" + } + ], + "v": { // fallback value, served when none of the targeting rules match, + // no percentage options are defined or evaluation of these is not possible + "b": false // depending on the setting type, another type of value may need to be specified: + // text setting -> "s": string + // whole number setting -> "i": number + // decimal number setting -> "d": number + }, + "i": "430bded3" // variation id (for analytical purposes) + } + } +} +``` + +For a more comprehensive specification of the config JSON v6 format, you may refer to [this JSON schema document](https://github.com/configcat/config-json/blob/main/V6/config.schema.json). + + + + +A convenient way to get the config JSON for a specific SDK Key is to install the [ConfigCat CLI](https://github.com/configcat/cli) tool +and execute the following command: + +```bash +configcat config-json get -f v5 -p {YOUR-SDK-KEY} > config.json +``` + +(Depending on your [Data Governance](advanced/data-governance.md) settings, you may need to add the `--eu` switch.) + +Alternatively, you can download the config JSON manually, based on your [Data Governance](advanced/data-governance.md) settings: - GLOBAL: `https://cdn-global.configcat.com/configuration-files/{YOUR-SDK-KEY}/config_v5.json` - EU: `https://cdn-eu.configcat.com/configuration-files/{YOUR-SDK-KEY}/config_v5.json` @@ -529,6 +734,8 @@ The URL to your current config JSON is based on your [Data Governance](advanced/ } } ``` + + ### Map @@ -709,10 +916,11 @@ Available log levels: Info level logging helps to inspect how a feature flag was evaluated: ```bash -INFO com.configcat.ConfigCatClient - [5000] Evaluating getValue(isPOCFeatureEnabled). -User object: User{Identifier=435170f4-8a8b-4b67-a723-505ac7cdea92, Email=john@example.com} -Evaluating rule: [Email:john@example.com] [CONTAINS] [@something.com] => no match -Evaluating rule: [Email:john@example.com] [CONTAINS] [@example.com] => match, returning "true" +INFO com.configcat.ConfigCatClient - [5000] Evaluating 'isPOCFeatureEnabled' for User '{"Identifier":"","Email":"configcat@example.com","Country":"US","SubscriptionType":"Pro","Role":"Admin","version":"1.0.0"}' + Evaluating targeting rules and applying the first match if any: + - IF User.Email CONTAINS ANY OF ['@something.com'] THEN 'False' => no match + - IF User.Email CONTAINS ANY OF ['@example.com'] THEN 'True' => MATCH, applying rule + Returning 'True'. ``` ### Logging Implementation