From 363951c080f173bf399fa6b9fdc879fa1f1aca65 Mon Sep 17 00:00:00 2001 From: ivarconr Date: Tue, 23 Jan 2024 17:21:32 +0000 Subject: [PATCH] Deploy website - based on e9d49d68fd356551ce73f6e70dddbc8f710b220c --- 404.html | 8 ++-- ...pired-ec15259f0a3c0f4fc41cbca389c19918.png | Bin 0 -> 23212 bytes ...huser-7dfff1dbf755b594a32bbe583f0a64be.png | Bin 0 -> 30571 bytes ...ssing-8bc0f18d979eff3dda9f45ed9af5b356.png | Bin 0 -> 21887 bytes ...rview-d5fde313e9b49397653607393dddb419.png | Bin 0 -> 215308 bytes ...3ee16.e8794ef8.js => 0833ee16.3310d1d2.js} | 2 +- ...5f8bd.cc1d2547.js => 1ff5f8bd.efba8cf1.js} | 2 +- ...af5ed.2fd805f8.js => 26faf5ed.bff6b0e9.js} | 2 +- ...7c5bd.cb71076b.js => 2d07c5bd.eb67c4c7.js} | 2 +- ...d7114.ef70ff89.js => 325d7114.ee7c8e38.js} | 2 +- ...677c6.84b6d2ea.js => 39e677c6.108329bf.js} | 2 +- ...6af39.9ab287cd.js => 3c56af39.4138b35e.js} | 2 +- ...3a177.fe43ad7c.js => 4db3a177.69b09904.js} | 2 +- assets/js/4fda00e5.4172f83f.js | 1 - assets/js/4fda00e5.98f4feb2.js | 1 + ...fa60f.8d879457.js => 54bfa60f.97e52020.js} | 2 +- ...fac95.8340428d.js => 5f3fac95.46a72b78.js} | 2 +- assets/js/65755815.ddb216c4.js | 1 + ...1326b.a64e6377.js => 6591326b.9da0ecfb.js} | 2 +- ...5c6ad.b5266fd3.js => 89b5c6ad.3e1a4eae.js} | 2 +- ...7ccfd.fb56cd79.js => 8cd7ccfd.b7f2c73b.js} | 2 +- ...26739.4b5ed859.js => 8f226739.d883ff57.js} | 2 +- assets/js/935f2afb.1fac2233.js | 1 - assets/js/935f2afb.eadfe24a.js | 1 + ...a64fb.e9b13c80.js => 93ba64fb.12ec40c1.js} | 2 +- ...00257.5fc071f8.js => 96600257.45142e3e.js} | 2 +- ...d4c50.7bc0a6b0.js => 9ead4c50.80844719.js} | 2 +- ...83c3a.cbf0d404.js => b9883c3a.841f9805.js} | 2 +- ...04f77.278561df.js => bce04f77.5a2a3a4f.js} | 2 +- ...7a5c1.92dc963b.js => d3a7a5c1.fd10dbdd.js} | 2 +- ...1e71a.fbfe1389.js => d751e71a.0a749ee7.js} | 2 +- ...16537.b2aee030.js => f4116537.a0bb269c.js} | 2 +- assets/js/main.777d27d6.js | 2 + ...CENSE.txt => main.777d27d6.js.LICENSE.txt} | 0 assets/js/main.d4effe3c.js | 2 - assets/js/runtime~main.06dad8fe.js | 1 - assets/js/runtime~main.c409ccd8.js | 1 + client-specification.html | 8 ++-- contributing.html | 8 ++-- contributing/ADRs.html | 8 ++-- .../ADRs/back-end/POST-PUT-api-payload.html | 8 ++-- .../ADRs/back-end/breaking-db-changes.html | 8 ++-- contributing/ADRs/back-end/naming.html | 8 ++-- .../ADRs/back-end/preferred-export.html | 8 ++-- .../ADRs/back-end/specificity-db-columns.html | 8 ++-- .../ADRs/front-end/component-naming.html | 8 ++-- .../ADRs/front-end/handling-tables.html | 8 ++-- .../ADRs/front-end/interface-naming.html | 8 ++-- .../preferred-component-props-usage.html | 8 ++-- .../preferred-data-fetching-method.html | 8 ++-- .../preferred-data-mutation-method.html | 8 ++-- .../ADRs/front-end/preferred-export.html | 8 ++-- .../front-end/preferred-folder-structure.html | 8 ++-- .../preferred-form-architecture.html | 8 ++-- .../front-end/preferred-function-type.html | 8 ++-- .../preferred-styles-import-placement.html | 8 ++-- .../front-end/preferred-styling-method.html | 8 ++-- .../ADRs/front-end/sdk-generator.html | 8 ++-- .../ADRs/overarching/domain-language.html | 8 ++-- .../separation-request-response-schemas.html | 8 ++-- contributing/backend/overview.html | 8 ++-- contributing/developer-guide.html | 8 ++-- contributing/frontend/overview.html | 8 ++-- .../deploy/license-keys/index.html | 11 +++++ feature-flag-tutorials.html | 8 ++-- .../flutter/a-b-testing.html | 8 ++-- .../nextjs/implementing-feature-flags.html | 8 ++-- feature-flag-tutorials/react.html | 8 ++-- feature-flag-tutorials/react/examples.html | 8 ++-- how-to.html | 8 ++-- how-to/api.html | 8 ++-- how-to/env.html | 8 ++-- ...w-to-add-feature-flag-naming-patterns.html | 8 ++-- how-to/how-to-add-sso-azure-saml.html | 8 ++-- how-to/how-to-add-sso-google.html | 8 ++-- how-to/how-to-add-sso-open-id-connect.html | 8 ++-- how-to/how-to-add-sso-saml-keycloak.html | 8 ++-- how-to/how-to-add-sso-saml.html | 8 ++-- how-to/how-to-add-strategy-constraints.html | 8 ++-- how-to/how-to-add-users-to-unleash.html | 8 ++-- how-to/how-to-capture-impression-data.html | 8 ++-- how-to/how-to-clone-environments.html | 8 ++-- ...reate-and-assign-custom-project-roles.html | 8 ++-- ...o-create-and-assign-custom-root-roles.html | 8 ++-- how-to/how-to-create-and-display-banners.html | 8 ++-- .../how-to-create-and-manage-user-groups.html | 8 ++-- how-to/how-to-create-api-tokens.html | 8 ++-- how-to/how-to-create-feature-toggles.html | 8 ++-- .../how-to-create-personal-access-tokens.html | 8 ++-- how-to/how-to-create-project-api-tokens.html | 8 ++-- how-to/how-to-create-service-accounts.html | 8 ++-- .../how-to-define-custom-context-fields.html | 8 ++-- how-to/how-to-download-login-history.html | 8 ++-- how-to/how-to-enable-openapi.html | 8 ++-- how-to/how-to-environment-import-export.html | 8 ++-- how-to/how-to-import-export.html | 8 ++-- .../how-to-manage-public-invite-tokens.html | 8 ++-- how-to/how-to-run-the-unleash-proxy.html | 8 ++-- how-to/how-to-schedule-feature-releases.html | 8 ++-- ...d-feature-updates-to-slack-deprecated.html | 8 ++-- how-to/how-to-set-up-group-sso-sync.html | 8 ++-- .../how-to-setup-sso-keycloak-group-sync.html | 8 ++-- .../how-to-synchronize-unleash-instances.html | 8 ++-- how-to/how-to-use-custom-strategies.html | 8 ++-- how-to/how-to-use-the-admin-api.html | 8 ++-- how-to/misc.html | 8 ++-- how-to/proxy.html | 8 ++-- how-to/sso.html | 8 ++-- how-to/users-and-permissions.html | 8 ++-- img/license-banner-expired.png | Bin 0 -> 23212 bytes img/license-banner-highuser.png | Bin 0 -> 30571 bytes img/license-banner-missing.png | Bin 0 -> 21887 bytes img/license-overview.png | Bin 0 -> 215308 bytes index.html | 8 ++-- quickstart.html | 8 ++-- reference.html | 8 ++-- reference/activation-strategies.html | 8 ++-- reference/api-tokens-and-client-keys.html | 8 ++-- reference/api/legacy/unleash.html | 8 ++-- .../api/legacy/unleash/admin/addons.html | 8 ++-- .../api/legacy/unleash/admin/archive.html | 8 ++-- .../api/legacy/unleash/admin/context.html | 8 ++-- .../api/legacy/unleash/admin/events.html | 8 ++-- .../legacy/unleash/admin/feature-types.html | 8 ++-- .../api/legacy/unleash/admin/features-v2.html | 8 ++-- .../api/legacy/unleash/admin/features.html | 8 ++-- .../api/legacy/unleash/admin/metrics.html | 8 ++-- .../api/legacy/unleash/admin/projects.html | 8 ++-- .../api/legacy/unleash/admin/segments.html | 8 ++-- reference/api/legacy/unleash/admin/state.html | 8 ++-- .../api/legacy/unleash/admin/strategies.html | 8 ++-- reference/api/legacy/unleash/admin/tags.html | 8 ++-- .../api/legacy/unleash/admin/user-admin.html | 8 ++-- reference/api/legacy/unleash/basic-auth.html | 8 ++-- .../api/legacy/unleash/client/features.html | 8 ++-- .../api/legacy/unleash/client/metrics.html | 8 ++-- .../api/legacy/unleash/client/register.html | 8 ++-- .../api/legacy/unleash/internal/health.html | 8 ++-- .../legacy/unleash/internal/prometheus.html | 8 ++-- reference/api/unleash.html | 8 ++-- .../api/unleash/add-access-to-project.html | 8 ++-- .../unleash/add-change-request-comment.html | 8 ++-- ...fault-strategy-to-project-environment.html | 8 ++-- .../unleash/add-environment-to-project.html | 8 ++-- .../api/unleash/add-favorite-feature.html | 8 ++-- .../api/unleash/add-favorite-project.html | 8 ++-- .../api/unleash/add-feature-dependency.html | 8 ++-- .../api/unleash/add-feature-strategy.html | 8 ++-- .../unleash/add-public-signup-token-user.html | 8 ++-- .../unleash/add-role-access-to-project.html | 8 ++-- reference/api/unleash/add-role-to-user.html | 8 ++-- .../api/unleash/add-tag-to-features.html | 8 ++-- reference/api/unleash/add-tag.html | 8 ++-- reference/api/unleash/addons.html | 8 ++-- reference/api/unleash/admin-ui.html | 8 ++-- reference/api/unleash/api-tokens.html | 8 ++-- reference/api/unleash/archive-feature.html | 8 ++-- reference/api/unleash/archive-features.html | 8 ++-- reference/api/unleash/archive.html | 8 ++-- reference/api/unleash/auth.html | 8 ++-- reference/api/unleash/banners.html | 8 ++-- reference/api/unleash/bulk-metrics.html | 8 ++-- .../bulk-toggle-features-environment-off.html | 8 ++-- .../bulk-toggle-features-environment-on.html | 8 ++-- reference/api/unleash/change-my-password.html | 8 ++-- reference/api/unleash/change-password.html | 8 ++-- reference/api/unleash/change-project.html | 8 ++-- reference/api/unleash/change-request.html | 8 ++-- reference/api/unleash/change-requests.html | 8 ++-- .../api/unleash/change-role-for-group.html | 8 ++-- .../api/unleash/change-role-for-user.html | 8 ++-- .../api/unleash/change-user-password.html | 8 ++-- .../api/unleash/check-dependencies-exist.html | 8 ++-- reference/api/unleash/check-license.html | 8 ++-- reference/api/unleash/client.html | 8 ++-- reference/api/unleash/clone-environment.html | 8 ++-- reference/api/unleash/clone-feature.html | 8 ++-- reference/api/unleash/context.html | 8 ++-- reference/api/unleash/create-addon.html | 8 ++-- reference/api/unleash/create-api-token.html | 8 ++-- reference/api/unleash/create-application.html | 8 ++-- reference/api/unleash/create-banner.html | 8 ++-- .../api/unleash/create-context-field.html | 8 ++-- reference/api/unleash/create-environment.html | 8 ++-- reference/api/unleash/create-feature.html | 8 ++-- reference/api/unleash/create-feedback.html | 8 ++-- reference/api/unleash/create-group.html | 8 ++-- reference/api/unleash/create-pat.html | 8 ++-- .../api/unleash/create-project-api-token.html | 8 ++-- reference/api/unleash/create-project.html | 8 ++-- .../unleash/create-public-signup-token.html | 8 ++-- reference/api/unleash/create-role.html | 8 ++-- reference/api/unleash/create-segment.html | 8 ++-- .../unleash/create-service-account-token.html | 8 ++-- .../api/unleash/create-service-account.html | 8 ++-- reference/api/unleash/create-strategy.html | 8 ++-- reference/api/unleash/create-tag-type.html | 8 ++-- reference/api/unleash/create-tag.html | 8 ++-- reference/api/unleash/create-user.html | 8 ++-- reference/api/unleash/delete-addon.html | 8 ++-- reference/api/unleash/delete-api-token.html | 8 ++-- reference/api/unleash/delete-application.html | 8 ++-- reference/api/unleash/delete-banner.html | 8 ++-- .../api/unleash/delete-change-request.html | 8 ++-- reference/api/unleash/delete-change.html | 8 ++-- .../api/unleash/delete-context-field.html | 8 ++-- .../unleash/delete-feature-dependencies.html | 8 ++-- .../unleash/delete-feature-dependency.html | 8 ++-- .../api/unleash/delete-feature-strategy.html | 8 ++-- reference/api/unleash/delete-feature.html | 8 ++-- reference/api/unleash/delete-features.html | 8 ++-- reference/api/unleash/delete-group.html | 8 ++-- reference/api/unleash/delete-pat.html | 8 ++-- .../api/unleash/delete-project-api-token.html | 8 ++-- reference/api/unleash/delete-project.html | 8 ++-- reference/api/unleash/delete-role.html | 8 ++-- .../unleash/delete-service-account-token.html | 8 ++-- .../api/unleash/delete-service-account.html | 8 ++-- reference/api/unleash/delete-tag-type.html | 8 ++-- reference/api/unleash/delete-tag.html | 8 ++-- reference/api/unleash/delete-user.html | 8 ++-- reference/api/unleash/dependencies.html | 8 ++-- reference/api/unleash/deprecate-strategy.html | 8 ++-- reference/api/unleash/disable-banner.html | 8 ++-- reference/api/unleash/edge.html | 8 ++-- reference/api/unleash/edit-change.html | 8 ++-- reference/api/unleash/enable-banner.html | 8 ++-- reference/api/unleash/environments.html | 8 ++-- reference/api/unleash/events.html | 8 ++-- reference/api/unleash/export-features.html | 8 ++-- reference/api/unleash/export.html | 8 ++-- reference/api/unleash/feature-types.html | 8 ++-- reference/api/unleash/features.html | 8 ++-- reference/api/unleash/frontend-api.html | 8 ++-- .../api/unleash/get-access-overview.html | 8 ++-- reference/api/unleash/get-addon.html | 8 ++-- reference/api/unleash/get-addons.html | 8 ++-- reference/api/unleash/get-admin-count.html | 8 ++-- .../api/unleash/get-advanced-playground.html | 8 ++-- reference/api/unleash/get-all-api-tokens.html | 8 ++-- .../api/unleash/get-all-client-features.html | 8 ++-- .../api/unleash/get-all-environments.html | 8 ++-- .../api/unleash/get-all-feature-types.html | 8 ++-- .../unleash/get-all-public-signup-tokens.html | 8 ++-- reference/api/unleash/get-all-strategies.html | 8 ++-- reference/api/unleash/get-all-toggles.html | 8 ++-- .../api/unleash/get-api-tokens-by-name.html | 8 ++-- reference/api/unleash/get-application.html | 8 ++-- reference/api/unleash/get-applications.html | 8 ++-- .../get-archived-features-by-project-id.html | 8 ++-- .../api/unleash/get-archived-features.html | 8 ++-- reference/api/unleash/get-banners.html | 8 ++-- .../unleash/get-base-users-and-groups.html | 8 ++-- reference/api/unleash/get-change-request.html | 8 ++-- .../get-change-requests-for-project.html | 8 ++-- reference/api/unleash/get-client-feature.html | 8 ++-- reference/api/unleash/get-context-field.html | 8 ++-- reference/api/unleash/get-context-fields.html | 8 ++-- .../get-deprecated-project-overview.html | 8 ++-- .../get-environment-feature-variants.html | 8 ++-- reference/api/unleash/get-environment.html | 8 ++-- .../api/unleash/get-events-for-toggle.html | 8 ++-- reference/api/unleash/get-events.html | 8 ++-- .../api/unleash/get-feature-environment.html | 8 ++-- .../api/unleash/get-feature-strategies.html | 8 ++-- .../api/unleash/get-feature-strategy.html | 8 ++-- .../unleash/get-feature-usage-summary.html | 8 ++-- .../api/unleash/get-feature-variants.html | 8 ++-- reference/api/unleash/get-feature.html | 8 ++-- reference/api/unleash/get-features.html | 8 ++-- reference/api/unleash/get-feedback.html | 8 ++-- .../api/unleash/get-frontend-features.html | 8 ++-- .../api/unleash/get-google-settings.html | 8 ++-- reference/api/unleash/get-group.html | 8 ++-- reference/api/unleash/get-groups.html | 8 ++-- reference/api/unleash/get-health.html | 8 ++-- .../unleash/get-instance-admin-stats-csv.html | 8 ++-- .../api/unleash/get-instance-admin-stats.html | 8 ++-- reference/api/unleash/get-login-history.html | 8 ++-- reference/api/unleash/get-maintenance.html | 8 ++-- reference/api/unleash/get-me.html | 8 ++-- reference/api/unleash/get-notifications.html | 8 ++-- reference/api/unleash/get-oidc-settings.html | 8 ++-- .../get-open-change-requests-for-user.html | 8 ++-- reference/api/unleash/get-pats.html | 8 ++-- ...t-pending-change-requests-for-feature.html | 8 ++-- .../get-pending-change-requests-for-user.html | 8 ++-- reference/api/unleash/get-permissions.html | 8 ++-- reference/api/unleash/get-playground.html | 8 ++-- reference/api/unleash/get-profile.html | 8 ++-- reference/api/unleash/get-project-access.html | 8 ++-- .../api/unleash/get-project-api-tokens.html | 8 ++-- .../get-project-change-request-config.html | 8 ++-- reference/api/unleash/get-project-dora.html | 8 ++-- .../api/unleash/get-project-environments.html | 8 ++-- .../unleash/get-project-health-report.html | 8 ++-- .../api/unleash/get-project-overview.html | 8 ++-- reference/api/unleash/get-project-users.html | 8 ++-- reference/api/unleash/get-projects.html | 8 ++-- .../api/unleash/get-public-signup-token.html | 8 ++-- .../api/unleash/get-raw-feature-metrics.html | 8 ++-- .../api/unleash/get-requests-per-second.html | 8 ++-- reference/api/unleash/get-role-by-id.html | 8 ++-- .../api/unleash/get-role-project-access.html | 8 ++-- reference/api/unleash/get-roles.html | 8 ++-- reference/api/unleash/get-saml-settings.html | 8 ++-- .../get-scheduled-change-requests.html | 8 ++-- reference/api/unleash/get-segment.html | 8 ++-- .../unleash/get-segments-by-strategy-id.html | 8 ++-- reference/api/unleash/get-segments.html | 8 ++-- .../unleash/get-service-account-tokens.html | 8 ++-- .../api/unleash/get-service-accounts.html | 8 ++-- .../api/unleash/get-simple-settings.html | 8 ++-- .../get-strategies-by-context-field.html | 8 ++-- .../unleash/get-strategies-by-segment-id.html | 8 ++-- reference/api/unleash/get-strategy.html | 8 ++-- reference/api/unleash/get-tag-type.html | 8 ++-- reference/api/unleash/get-tag-types.html | 8 ++-- reference/api/unleash/get-tag.html | 8 ++-- reference/api/unleash/get-tags-by-type.html | 8 ++-- reference/api/unleash/get-tags.html | 8 ++-- .../api/unleash/get-telemetry-settings.html | 8 ++-- reference/api/unleash/get-ui-config.html | 8 ++-- reference/api/unleash/get-user.html | 8 ++-- reference/api/unleash/get-users.html | 8 ++-- reference/api/unleash/get-valid-tokens.html | 8 ++-- reference/api/unleash/import-export.html | 8 ++-- reference/api/unleash/import-toggles.html | 8 ++-- reference/api/unleash/import.html | 8 ++-- reference/api/unleash/instance-admin.html | 8 ++-- .../api/unleash/list-parent-options.html | 8 ++-- reference/api/unleash/list-tags.html | 8 ++-- reference/api/unleash/login.html | 8 ++-- reference/api/unleash/maintenance.html | 8 ++-- .../unleash/mark-notifications-as-read.html | 8 ++-- reference/api/unleash/metrics.html | 8 ++-- reference/api/unleash/notifications.html | 8 ++-- reference/api/unleash/operational.html | 8 ++-- ...verwrite-environment-feature-variants.html | 8 ++-- ...rite-feature-variants-on-environments.html | 8 ++-- .../unleash/overwrite-feature-variants.html | 8 ++-- .../patch-environments-feature-variants.html | 8 ++-- .../api/unleash/patch-feature-strategy.html | 8 ++-- .../api/unleash/patch-feature-variants.html | 8 ++-- reference/api/unleash/patch-feature.html | 8 ++-- .../api/unleash/personal-access-tokens.html | 8 ++-- reference/api/unleash/playground.html | 8 ++-- reference/api/unleash/projects.html | 8 ++-- reference/api/unleash/provide-feedback.html | 8 ++-- .../api/unleash/public-signup-tokens.html | 8 ++-- .../api/unleash/reactivate-strategy.html | 8 ++-- reference/api/unleash/read-license.html | 8 ++-- .../unleash/register-client-application.html | 8 ++-- .../api/unleash/register-client-metrics.html | 8 ++-- .../api/unleash/register-frontend-client.html | 8 ++-- .../unleash/register-frontend-metrics.html | 8 ++-- .../remove-environment-from-project.html | 8 ++-- reference/api/unleash/remove-environment.html | 8 ++-- .../api/unleash/remove-favorite-feature.html | 8 ++-- .../api/unleash/remove-favorite-project.html | 8 ++-- .../api/unleash/remove-group-access.html | 8 ++-- .../api/unleash/remove-role-for-user.html | 8 ++-- .../api/unleash/remove-role-from-group.html | 8 ++-- reference/api/unleash/remove-segment.html | 8 ++-- reference/api/unleash/remove-strategy.html | 8 ++-- reference/api/unleash/remove-tag.html | 8 ++-- reference/api/unleash/remove-user-access.html | 8 ++-- .../api/unleash/reset-user-password.html | 8 ++-- reference/api/unleash/revive-feature.html | 8 ++-- reference/api/unleash/revive-features.html | 8 ++-- reference/api/unleash/search-events.html | 8 ++-- reference/api/unleash/search-features.html | 8 ++-- reference/api/unleash/search-users.html | 8 ++-- reference/api/unleash/search.html | 8 ++-- reference/api/unleash/segments.html | 8 ++-- .../unleash/send-reset-password-email.html | 8 ++-- reference/api/unleash/service-accounts.html | 8 ++-- .../api/unleash/set-google-settings.html | 8 ++-- reference/api/unleash/set-oidc-settings.html | 8 ++-- reference/api/unleash/set-project-access.html | 8 ++-- .../api/unleash/set-roles-for-group.html | 8 ++-- reference/api/unleash/set-roles-for-user.html | 8 ++-- reference/api/unleash/set-saml-settings.html | 8 ++-- .../api/unleash/set-simple-settings.html | 8 ++-- .../api/unleash/set-strategy-sort-order.html | 8 ++-- reference/api/unleash/set-ui-config.html | 8 ++-- reference/api/unleash/stale-features.html | 8 ++-- reference/api/unleash/strategies.html | 8 ++-- reference/api/unleash/tags.html | 8 ++-- reference/api/unleash/telemetry.html | 8 ++-- .../api/unleash/toggle-environment-off.html | 8 ++-- .../api/unleash/toggle-environment-on.html | 8 ++-- .../toggle-feature-environment-off.html | 8 ++-- .../toggle-feature-environment-on.html | 8 ++-- reference/api/unleash/toggle-maintenance.html | 8 ++-- reference/api/unleash/unleash-api.html | 8 ++-- reference/api/unleash/unstable.html | 8 ++-- reference/api/unleash/update-addon.html | 8 ++-- reference/api/unleash/update-api-token.html | 8 ++-- reference/api/unleash/update-banner.html | 8 ++-- .../unleash/update-change-request-state.html | 8 ++-- .../unleash/update-change-request-title.html | 8 ++-- .../api/unleash/update-context-field.html | 8 ++-- reference/api/unleash/update-environment.html | 8 ++-- .../update-feature-strategy-segments.html | 8 ++-- .../api/unleash/update-feature-strategy.html | 8 ++-- .../unleash/update-feature-type-lifetime.html | 8 ++-- reference/api/unleash/update-feature.html | 8 ++-- reference/api/unleash/update-feedback.html | 8 ++-- reference/api/unleash/update-group.html | 8 ++-- reference/api/unleash/update-license.html | 8 ++-- .../update-project-change-request-config.html | 8 ++-- .../update-project-enterprise-settings.html | 8 ++-- reference/api/unleash/update-project.html | 8 ++-- .../unleash/update-public-signup-token.html | 8 ++-- reference/api/unleash/update-role.html | 8 ++-- reference/api/unleash/update-segment.html | 8 ++-- .../api/unleash/update-service-account.html | 8 ++-- reference/api/unleash/update-sort-order.html | 8 ++-- .../api/unleash/update-splash-settings.html | 8 ++-- reference/api/unleash/update-strategy.html | 8 ++-- reference/api/unleash/update-tag-type.html | 8 ++-- reference/api/unleash/update-tags.html | 8 ++-- reference/api/unleash/update-user.html | 8 ++-- reference/api/unleash/users.html | 8 ++-- .../unleash/validate-archive-features.html | 8 ++-- .../api/unleash/validate-constraint.html | 8 ++-- .../unleash/validate-environment-name.html | 8 ++-- reference/api/unleash/validate-feature.html | 8 ++-- reference/api/unleash/validate-import.html | 8 ++-- reference/api/unleash/validate-password.html | 8 ++-- reference/api/unleash/validate-project.html | 8 ++-- .../unleash/validate-public-signup-token.html | 8 ++-- reference/api/unleash/validate-role.html | 8 ++-- reference/api/unleash/validate-segment.html | 8 ++-- reference/api/unleash/validate-tag-type.html | 8 ++-- reference/api/unleash/validate-token.html | 8 ++-- .../api/unleash/validate-user-password.html | 8 ++-- reference/api/unleash/validate.html | 8 ++-- reference/archived-toggles.html | 8 ++-- reference/banners.html | 8 ++-- reference/change-requests.html | 8 ++-- reference/custom-activation-strategies.html | 8 ++-- reference/dependent-features.html | 8 ++-- reference/environments.html | 8 ++-- reference/event-log.html | 8 ++-- reference/event-types.html | 8 ++-- reference/feature-flag-naming-patterns.html | 8 ++-- reference/feature-toggle-types.html | 8 ++-- reference/feature-toggle-variants.html | 8 ++-- reference/feature-toggles.html | 8 ++-- reference/front-end-api.html | 8 ++-- reference/impression-data.html | 8 ++-- reference/integrations.html | 8 ++-- reference/integrations/datadog.html | 8 ++-- .../jira-cloud-plugin-installation.html | 8 ++-- .../integrations/jira-cloud-plugin-usage.html | 8 ++-- .../jira-server-plugin-installation.html | 8 ++-- .../jira-server-plugin-usage.html | 8 ++-- reference/integrations/slack-app.html | 8 ++-- reference/integrations/slack.html | 8 ++-- reference/integrations/teams.html | 8 ++-- reference/integrations/webhook.html | 8 ++-- reference/login-history.html | 8 ++-- reference/maintenance-mode.html | 8 ++-- reference/network-view.html | 8 ++-- reference/notifications.html | 8 ++-- reference/playground.html | 8 ++-- reference/project-collaboration-mode.html | 8 ++-- reference/projects.html | 8 ++-- reference/public-signup.html | 8 ++-- reference/rbac.html | 8 ++-- reference/sdks.html | 8 ++-- reference/sdks/android-proxy.html | 10 ++--- reference/sdks/dotnet.html | 10 ++--- reference/sdks/flutter.html | 10 ++--- reference/sdks/go.html | 10 ++--- reference/sdks/ios-proxy.html | 10 ++--- reference/sdks/java.html | 10 ++--- reference/sdks/javascript-browser.html | 10 ++--- reference/sdks/next-js.html | 10 ++--- reference/sdks/node.html | 10 ++--- reference/sdks/php.html | 10 ++--- reference/sdks/python.html | 10 ++--- reference/sdks/react.html | 10 ++--- reference/sdks/ruby.html | 10 ++--- reference/sdks/rust.html | 10 ++--- reference/sdks/svelte.html | 10 ++--- reference/sdks/vue.html | 10 ++--- reference/segments.html | 8 ++-- reference/service-accounts.html | 8 ++-- reference/sso.html | 8 ++-- reference/stickiness.html | 8 ++-- reference/strategy-constraints.html | 8 ++-- reference/strategy-variants.html | 8 ++-- reference/tags.html | 8 ++-- reference/technical-debt.html | 8 ++-- reference/unleash-context.html | 8 ++-- reference/unleash-edge.html | 10 ++--- reference/unleash-proxy.html | 10 ++--- reference/whats-new-v4.html | 8 ++-- search.html | 8 ++-- sitemap.xml | 2 +- topics.html | 8 ++-- topics/a-b-testing.html | 8 ++-- .../business-case-feature-flag-migration.html | 8 ++-- ...feature-flag-migration-best-practices.html | 8 ++-- .../feature-flag-migration-scope.html | 8 ++-- ...how-to-execute-feature-flag-migration.html | 8 ++-- ...bording-users-to-feature-flag-service.html | 8 ++-- .../planning-feature-flag-migration.html | 8 ++-- .../availability-over-consistency.html | 8 ++-- .../democratize-feature-flag-access.html | 8 ++-- topics/feature-flags/enable-traceability.html | 8 ++-- .../evaluate-flags-close-to-user.html | 8 ++-- .../feature-flag-best-practices.html | 8 ++-- topics/feature-flags/limit-payloads.html | 8 ++-- topics/feature-flags/never-expose-pii.html | 8 ++-- topics/feature-flags/prioritize-ux.html | 8 ++-- topics/feature-flags/runtime-control.html | 8 ++-- topics/feature-flags/scale-horizontally.html | 8 ++-- .../short-lived-feature-flags.html | 8 ++-- topics/feature-flags/unique-names.html | 8 ++-- understanding-unleash.html | 8 ++-- understanding-unleash/data-collection.html | 8 ++-- .../managing-constraints.html | 8 ++-- understanding-unleash/proxy-hosting.html | 8 ++-- .../the-anatomy-of-unleash.html | 8 ++-- understanding-unleash/unleash-overview.html | 8 ++-- unleash-academy/advanced-for-devs.html | 8 ++-- unleash-academy/foundational.html | 8 ++-- unleash-academy/introduction.html | 8 ++-- .../managing-unleash-for-devops.html | 8 ++-- using-unleash.html | 10 ++--- using-unleash/deploy.html | 10 ++--- .../deploy/configuring-unleash-v3.html | 8 ++-- using-unleash/deploy/configuring-unleash.html | 10 ++--- using-unleash/deploy/database-backup.html | 10 ++--- using-unleash/deploy/database-setup.html | 10 ++--- using-unleash/deploy/email-service.html | 10 ++--- using-unleash/deploy/getting-started.html | 10 ++--- using-unleash/deploy/google-auth-hook.html | 10 ++--- using-unleash/deploy/google-auth-v3.html | 8 ++-- .../deploy/license-keys.htm/index.html | 11 +++++ using-unleash/deploy/license-keys.html | 38 ++++++++++++++++++ using-unleash/deploy/license-keys.html.html | 11 +++++ using-unleash/deploy/securing-unleash-v3.html | 8 ++-- using-unleash/deploy/securing-unleash.html | 10 ++--- using-unleash/deploy/upgrading-unleash.html | 10 ++--- using-unleash/troubleshooting.html | 10 ++--- using-unleash/troubleshooting/cors.html | 8 ++-- .../troubleshooting/email-service.html | 8 ++-- .../feature-not-available.html | 8 ++-- .../troubleshooting/flag-exposure.html | 8 ++-- .../troubleshooting/flag-not-returned.html | 8 ++-- 555 files changed, 2169 insertions(+), 2097 deletions(-) create mode 100644 assets/images/license-banner-expired-ec15259f0a3c0f4fc41cbca389c19918.png create mode 100644 assets/images/license-banner-highuser-7dfff1dbf755b594a32bbe583f0a64be.png create mode 100644 assets/images/license-banner-missing-8bc0f18d979eff3dda9f45ed9af5b356.png create mode 100644 assets/images/license-overview-d5fde313e9b49397653607393dddb419.png rename assets/js/{0833ee16.e8794ef8.js => 0833ee16.3310d1d2.js} (98%) rename assets/js/{1ff5f8bd.cc1d2547.js => 1ff5f8bd.efba8cf1.js} (99%) rename assets/js/{26faf5ed.2fd805f8.js => 26faf5ed.bff6b0e9.js} (99%) rename assets/js/{2d07c5bd.cb71076b.js => 2d07c5bd.eb67c4c7.js} (99%) rename assets/js/{325d7114.ef70ff89.js => 325d7114.ee7c8e38.js} (62%) rename assets/js/{39e677c6.84b6d2ea.js => 39e677c6.108329bf.js} (99%) rename assets/js/{3c56af39.9ab287cd.js => 3c56af39.4138b35e.js} (99%) rename assets/js/{4db3a177.fe43ad7c.js => 4db3a177.69b09904.js} (99%) delete mode 100644 assets/js/4fda00e5.4172f83f.js create mode 100644 assets/js/4fda00e5.98f4feb2.js rename assets/js/{54bfa60f.8d879457.js => 54bfa60f.97e52020.js} (98%) rename assets/js/{5f3fac95.8340428d.js => 5f3fac95.46a72b78.js} (99%) create mode 100644 assets/js/65755815.ddb216c4.js rename assets/js/{6591326b.a64e6377.js => 6591326b.9da0ecfb.js} (98%) rename assets/js/{89b5c6ad.b5266fd3.js => 89b5c6ad.3e1a4eae.js} (98%) rename assets/js/{8cd7ccfd.fb56cd79.js => 8cd7ccfd.b7f2c73b.js} (98%) rename assets/js/{8f226739.4b5ed859.js => 8f226739.d883ff57.js} (99%) delete mode 100644 assets/js/935f2afb.1fac2233.js create mode 100644 assets/js/935f2afb.eadfe24a.js rename assets/js/{93ba64fb.e9b13c80.js => 93ba64fb.12ec40c1.js} (99%) rename assets/js/{96600257.5fc071f8.js => 96600257.45142e3e.js} (99%) rename assets/js/{9ead4c50.7bc0a6b0.js => 9ead4c50.80844719.js} (99%) rename assets/js/{b9883c3a.cbf0d404.js => b9883c3a.841f9805.js} (99%) rename assets/js/{bce04f77.278561df.js => bce04f77.5a2a3a4f.js} (99%) rename assets/js/{d3a7a5c1.92dc963b.js => d3a7a5c1.fd10dbdd.js} (99%) rename assets/js/{d751e71a.fbfe1389.js => d751e71a.0a749ee7.js} (99%) rename assets/js/{f4116537.b2aee030.js => f4116537.a0bb269c.js} (99%) create mode 100644 assets/js/main.777d27d6.js rename assets/js/{main.d4effe3c.js.LICENSE.txt => main.777d27d6.js.LICENSE.txt} (100%) delete mode 100644 assets/js/main.d4effe3c.js delete mode 100644 assets/js/runtime~main.06dad8fe.js create mode 100644 assets/js/runtime~main.c409ccd8.js create mode 100644 docs/using-unleash/deploy/license-keys/index.html create mode 100644 img/license-banner-expired.png create mode 100644 img/license-banner-highuser.png create mode 100644 img/license-banner-missing.png create mode 100644 img/license-overview.png create mode 100644 using-unleash/deploy/license-keys.htm/index.html create mode 100644 using-unleash/deploy/license-keys.html create mode 100644 using-unleash/deploy/license-keys.html.html diff --git a/404.html b/404.html index 410dff99c5..4963c4ef6f 100644 --- a/404.html +++ b/404.html @@ -20,15 +20,15 @@ - - + +
Skip to main content

Page Not Found

We could not find what you were looking for.

Please contact the owner of the site that linked you to the original URL and let them know their link is broken.

- - + + \ No newline at end of file diff --git a/assets/images/license-banner-expired-ec15259f0a3c0f4fc41cbca389c19918.png b/assets/images/license-banner-expired-ec15259f0a3c0f4fc41cbca389c19918.png new file mode 100644 index 0000000000000000000000000000000000000000..1af308b06d6ccec862b1320555e363d1cf28ba28 GIT binary patch literal 23212 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 literal 0 HcmV?d00001 diff --git a/assets/images/license-banner-highuser-7dfff1dbf755b594a32bbe583f0a64be.png b/assets/images/license-banner-highuser-7dfff1dbf755b594a32bbe583f0a64be.png new file mode 100644 index 0000000000000000000000000000000000000000..940a40a33bc5ec823e616b8e8e0eb66f6bc62069 GIT binary patch literal 30571 zcmeFZWmH?;*Dne*I24K&mtw`GxKrGn0Kp2B7713|i+hVp(El#~6ciME6=iu{6cmgM zWwWbj!5+-3AXuqC;ViS~GDN!pk0&MZv| z@QjA)-O@@W2?i+tQ@eCP>+^J4d9GH%lT<^OkXm>{QNVWpxg3Cl$0URHkBt^h0mfxP#meoI zaK?fC0p*`t?-MqD?h z^en3Vs|TC_Vr+VYc7%!OQUAGNF>8n6=&{k!GX0Cr$AoO_*JIWHNXI`N2vvhI)C;`5 zKm0S-lsnae$<)mmZ#DnL`yu}qO^=QC-;I}uZF{Nf2ss=CPsk-HM^=ABHSjrx)#AJ$ zKlosN)+_LzNQ9%l|A+~aM;u)G&b964RbTPG$X=!y&uBI)^2om9z52V5?0W(DA3OZC zy$YPM7s>!AcM?9GDshT}Y!XDvB}A$QRz=p}c%j&+qxkjUV?X`;nd+q+7> znq!JI4Xmgms0*ctE1KC3h1hN4-)MX(jYg7j+2KA$cC-VaC+Tt<#K2N<35aw89upU{ z@rOUz;QO1H%z9yyIB8d2E+N2Yv;4@OJR!m@)Fcb&US1DbQW|NFOOy-b=cuky7WMiB zcD)sukJ$|D$J_Ju7xTuDf@q_B#cp7GHkMPX^58}}U*>IG1>`es3-Hsddv){TS*JVI zU6{8q=_Gu=+xMbK_hg;lZnEST#laUocRx@}%+G4jQBzzN`35N%Ej2d%gJvrGpl-n7 zTvK*#FYeWx&4FybY$pFisiU!RtF2lH%JW~RxJ5rX-*wP;+DSZ2g4ztJp6k?pa-91% zFO7Dk$f8|Pdmm9cuPsz%K_sdHRvb~eR)`@CKYoGHBX1^KgyUt2_6=Q?{QBeKdrK&O#kyM*w|t~oX&?@+a5k*^L@}Q3V_v(fT~;XZqa5nr=i^N=E@X=Y zjFwR@%!$IDxBLO?N*^aTU-y1Wu{a1eoT$9gNHT7z zycIlLTC5;DV?-rVeu>sM0nw}=!u}vbOZeT*HT|Mzy15UReS6`hZ`RAUz}qfi9MB$v z^cBy?Sjmn!*e<~)hki<_QS&zmersP)O5x2*#Mm}hk$p7Lb6lV+#o>U(<4Qw~`SPgK z%?e`#U--U8b!oDok&xpXfch2Jh2T67U(K7IGa4~f(VS18YOUH{%wVgir})>ti@vq% z$7^2#GZaJmpaC7`4@L=e>2iDB$u>hb9{lgR{4+J+#og``I4T;SLeUB`HD5L@hns_Kf;FQ8?D>D@z z^9-xv(&#a3PU|YD^##g`2s5}VEmj&Ub0OR<3wt>1cpJaV4;F0t`#*J%r08!w6PmKkK%WAUO9&WHD@UOee8VXRCfJWRCL~nKy8^8MwTyo7kT^V>hl{ zB`_B8({G11LrA~5?OO2L?wi(nPA_&Fdce|=!jFzL8M?v!OibEOu{$TeV_?b4cA1E* ziAZ5$D}C>&+5!x-RScv!W;B9x{A+EZ&m~t;uGN>0G$xLOK9!L@i=WQHWC?(=T$hxL zhAwcT?Ud86*D|MjqlLa6`;Z8w!wBa|=Sn8u8T)wP2V%}K{F(+@D6R2t@ejYPkEo)1 z-^e^46k$S4Vl*5RAep~{mn&dr;mopN6b$I@ha~yNk7~z`P_b&fIIOX}c=ypnWtDcJ z&R&qB#@`qfCh-s(=)bM*6O>4b$q7y@H8;DNcBc1gc=LQl{pol*hOPuNBu2cxA63y( zl`r=DAd=AEQ(wsweMd*zUuZ)4ImpaIPLYSLw06s`}ItNZ_D3s7qR;GyFNelZ4DUNV?u(<}t;?a4bR`qdpIC zN%hA9XX}kd93BCF6$g21uYnvTHWSURd6@#HzWSAn!a{Eu@9uxU^!#Px zs1Y!E!$?|rOtwG6k=F7PJxxQ&_O&Yfl(Y!=5ihejF!afndJzY#E-39HD!v(nUh~tQ zZc}Z|rcWRk()k+K0a#K&U6LmK-C-BTbs7rmi+S+{-YZf8QWPV0r^q_Sfks@v-D2;` z{?Z4{O&5HV@0yk^n2<0NO}@meHJ*!9i1CeYfz`h#ff`ec7zZJUsO|z z9a4vy*9{_as|8PEn{(V(UMSI|IWO3cF#{6CidBa#?5!RZ036uoj)emjpbAZ=y7Ffs z(8t|&CJ8GhygRV6uZ%<7K6Yzz*%I}VjjE*2MB2rI+8t+Wp4G1$JqnBppT>R*=~wWQ zJUW;Tn050A!b$^vF3AZD(&FQjUAFoOsUahRmX`|Y94cr39S z@daSsc5yT;vOk7N?L}OXb0w**f_0bJ!m-tLZm>`L0LLnV4&@&DJ{t*dE80=-b1o5P zemDviG4pi&Z|2+QXF3Uz$Ka`2_&M^DnpXuppU3?x15U>;;7cqD_i?Sf(Lkfh$0x7{ zwypiD-C+00B1ch4;gi66E!btj!@N|&{kxb;(~3J=#8wy-zWK%A-npH1LHH+TqRv&8 z#`F@yQ~D$4)G{~`r@h$$bbCPPiAbYfFV)VfwAzYD$Jn#p$asP8;+gJ=e4_i!!|{P+ zu;tKq{s$EshR`Htb<>WC!0k(XFZHm#RqUHrvb&55=%60?1e@_T3mW2h;lQ`um(2F? zaQ4;w6=Q}+K5JQTKc^Mt9a8MnR7bbVyA~xp)3)XDgWPvDhF1?RZhoZHxbuoS7y(33hrA$cfWY*k;~2N^H@ws znnwj zi@k9P+D%?S+OglE>~drJvj=CR3W!Q?EiU!&izQoMsgL~!JkSf#{oX9h3_CW7ASPc} zENMn4L1VhmlpkW@X3?`7Rd7@gpS-Br65`wPcXvb=?eA+_ z8C2<@tuEE@%5f^Q3*!UHJDQCIGjO{?Z0ZM=%EWZ`%+V>zH_4o4yqeqNx3Zo`K7&=I zCb-k~E+3{!C%c$TMG)jr+AI!@g-@m2KbCRL+_B*&)Y=W9Qx%C9`2sd`-6yqg4yzTJ zYCV+o1AEAnhGZixcyM3eaay+rsiyH{a<)u4y$K)kDDvz%oG2LB(*5k|cfD5U=a^RG z^!mWlBv5Ii+4;IKnjc=xQ+yyxJ|tJ!6VsP$eFE{+=c;_R4xWd|rP)iSmlr4D_r;_q zfdlTjk%K3h`rdI>Y{4xR==!d(-ypah+%F$z44mN3QmalUJFzM5_+I z&aa4m4Vy?!RS6+?MNpFc-aICp@x`O zm{!JO4<%kSphVAO9WfLU4=5yMS?7E18L{m(=G>8`!i6RG`{x#EhBgZeDZLBHBsP}+ z0D)BFv+m6scF71!cqDeM!xYi$T+Mq+A`D_Q$WCLO0`A%C>vvAiF0B~}iQO7Ba~%O9 zn;~M3KO5>W%GI(CT0DpCbL4vyh13I=Bz%%oBI!tJJPL6?t+JpMQlrW9b_5Tm-Cs~* z0TMD35Txo@pz0lflO_}QG>4vv`GiN#`@Jx?6WDKJr-e)|7SBCt4- zVuSb&qU!BVQ)j7c`40u=hhiFmZwt?fi~PJkx?n1^5o5WoE~z()~K4trQIBWg&70R>Z%!q@)`Tt zr`s{*^-hf)a=bvOQ_Ip;OFVxBbGfiN^=}LDxT|8D!UPAdcefH9V6zPjJO^3E+4eQB zj$WImhX}Z*qc9x3d%ejmdc9wv%(+-^_HAA)JsO5a^P1SbR<_Xga*T+#0Bs@3;c$m| zT7`Eerr{F&oqtC8i6sqsO-pJ*p0^YPZL+xe0c|U|#v4TD(ipqDqFw=%F;TQbJ0bb4^-3xc^CERBxJbeY zld5sOx+D2s#=BN2HA^D$jSi|#bZ%#>rifKr@)v1Vg)~_wlAKrR)*SCi4!H*S^n9O^bP^8$oqxCz-{yJ_>*po%XnT0Qn-9>7%;s?(o1D*cwi>(RjTb~{r zmjkWs_{Ryo-#)e`tr)7|=FTnPOSlV|U^>gjP>4tk-_W_y1IPL4-1w*LjFgUJSXntWj7M4Hwyc)g|VQ|?v(1#mgb=BOX<*i-5*DT86Q7HW~h)&SaHHA}; z@AYuWw4$ZR=4_DasBexz!_wiQ3mR4FlJr_>oL8er?O4AMmdTore!%@y)~K zTLStd^V}Nx`XSzDcdTC6oMwdGLp(+WCGYlppd)!Fp>dSgJkJeEVsJ<#i-9$|?C%43 z2#xyvAEb9(zHd!b91QY1bz|~5VtR^N9Ak2PnKlV*34M6oTTm`&@|9Hw_2p>e*Ww+ubo2b!cnUIzVAeIwP>u~VW)=7rMFYL>OzO4@3% z)n_A5D?3w+zgAAo1VE{1IaAApctLwYV)B$wF~xS@hkoskWj-x>aTyWrC_3dY6YnRQ z&wTjZW@Vnm6o3iC3UjM@55MEaC5l{sEZWN~*uaKazPC&-yX(Opo&Usi;CC*-jc>%A zV}C;d^UXRPXyFf>Is@wmvY-;N$phGUNeh$v%9CWu0iY}IvhB6>$7zC$7q^d@p#)^4 zN^pWZ+=~G`PEX6X03B56A4@KJ$t7;Vnna+Rf!SHpp5J`eEAz8dL?!?nR~lS~MT z|=QS;5c9xJSdj@TT+-L=Az zBZj^FP4gSZByYgy14mLguc-ggREHy5cZd5`&nk*2!uUpdPFI1%gnnZGE`|!I)_Qa`>H$i zSEhw1t~jVWzHygt8*R^HFf$svHhEdfT%^or&W=MhNp@bW;@h?695p+27m<*iC@Pmo z%gj<#IbZgWlCeVL4bakpXL12>?Spf<R@$83&R39_@5l+JJWmVLC*N|i9< zx8OA|eBU(CjJNa1;f`T*vweF;=_6~u3{_G8$xR>unTD^8z#+*T+T?t)F@ z?`izxW#9uLgN`vv(ivSYjg)?R)Xpz={C-(1U#tv&>93kj)+%-i($bz*)|rEB-9drl zzE6@A65jjqnQJ$>7HXowpW^)Vdb7jb??yJR7|38x7wdaSv}COvslONfP6r;`7=5Gl zLjbLxmm>*dDQ^wj926SC{?7S>vth5Fk<47QoM!cB$n9PKm`yH&OLe!vw*wxC;%$S} zf@JA9_m?{{u}fXR-l~z|)P0KShIwH-$(E?VUY%?q-vAw%-6xL7p3D{ojTz`UP)EI(WDRh9oc%5u(w6qqD7(gARCNSl(oI#f-ba=*p z-c)pEHA9KoTgeHiV^oM8R;mUO5)Yerh$+o`wkfBL(0wKo(u;F>aW+1_xf#2nedWxv z`3Apa!uD&HM91|nL+e3<5|E(i{cU=YPV8sf?qBJP<$Y2;zIgA65*|diHR?Jla9%~` z{r>Q7*Q3o}_|~(I=RdA&X0fhTMWmqHEWTT5?#7ZIKz`>&8uZZ;dhvHvJCE)Su5f@K=?ob{<2`>QS!Z0R>Qf^RGetKMi_l@kZ5li7o1UHNA6MB&@w2>IK*+{xH=HtT}3Rr0naF{hz>tKB9Iqy zMRgw8!1NZr3r)pfi9ZmHjm_9h_gJV4V^YtxV#Gytre27?w7GfYEmcs+HXj~*Gk3c* zsL^Pz#_az&dD{>q=L_;9ykUoLI@HycT{M+mM9mBPef=I2jfrJtm?BRBK?xJ_dyfP5 zjuo3*9T1w6Sp2QipJ_7|PzWEp3|;(X2NplR=4@kO$e0!Gs3DN+Y|9Xtk;3;<26qZZ zo&Zn-&a<~T;))+}KDv{eaz9EJ$~@`kY4fs}s_(v|lK+^?E<(S$jQ$Zlj$VR61{7|) zYvc#2oOe$dfo6?7eT_5yu$RhhT_I6(&(Od0DFWgfD&4CUcZD3cPxaSg{^bnkuiQ6_ zt6m-Q_qG>VPNi4)fW`Dm<<*jSyi47_Mx?azG zds`!zWoZ^S2%N&ddI12MuH9SrG4H<&pqqA`2*~XWY~=1OqGPFl!fVdJC8v4l5Eww> z^9oO1dCQo6&H+ZUc}t$EroD80DwncD@a21B;t?mv1|S18nE&BE?w){9-yh(1l!4kcYq6dLlY>lwNv8>b(3r*O6|igaDAomVVxE95u6;* zXur^8orsxmV`ek<&H2?jQG)YQt!Dq-Iki*-29Rr_DpP%6r%G`u8-h?aAVb&4!o$E7&~V(4+-n! z?1x2Kteo7hhWt4Fj3dgr7e;>@-qXhbOKw);igOJ1!)_Ht=rn*m)91(x5GX+0&ULQu z3V1BR3_s>jW+GfBF@`BWJ@PtWRgdby!D9SDy7!)dOc#=YoAu$F56k>3-ABnWOVA zvhoWUA()BLaAaA*$spC$jSmOW7r9lk2t*X%@kR(Pq0yM;O}VlP%K_UCdN=73e)+-G-}yvV2L`Yo5>8b$x6o(-wd_#eKB7e9w{ z#F|^I^A0l5eiB|ism=(CT0SA>nO-b_r8&RbWt6T06*%yZB`&$^wpE8;=$IP&NwS=A zrVa@@4EG4Jr$d^Xu9*=O8!J1*X_O2n=%na;nP7`gU1zj%Ph0Y9IPE1{$9-jYU3kJh zs6pm8=oPME5GT7^8fu;*`W&jXH&m6!yN3;VpD$}lcewXyt%ueBMAD#-b#Am~1>RTM z&E8SWj2hqgfBJ9b_S#p!r071%e3Q@~7bb=M^M{$hxGC8lDO7q3)|hT@QH$q(HrKhm z3x0MN80~H0dIrr)KKRV>P6F$O%jT??41C;wfBfy+QD7!@KKOWB6EPq4)CG=FQ{rDd z-#j+a0#1_5+Zdp84kKc|C{7si7^Mqi5hj`Rh@;nK2umeUE92kdpEg?ol&$Pz&DA@F zs^44))@?tTkYh4#x4X-Qe%;y4#oLf0@Q4J~$th$n?U_hPm9CHse8NQ|LjzrVh@I~> zmiDQo@Ht@g9gKwYh z<3>-tOl8Kx2ZX7nrr6Bz0j3K zc&x2qvGtdo3sp?CM%-{9DxxM(0T?B%cF*_ulV3;5V1jU<&uqY8Jt_DUayYTJm+)9l za#61Ph6s-NO7c_gba{V;rj8=m?5nKx)2iVIhv|7x_Jg*tGi<)Ae%zVL77zDpg9itq zwhFpw7gDRsdQXfVg$m&hN8cT^5BdF}SvS1JsFfjCHCS9XYRzLxm7G|uG#yHfGemGj z5r$b(hCZ3QgDcDp6P1$*>X~<5Mp0w-l4PVFP$MJX6G=pT--S91et1kXC3q;W*Pwso z5Picoq_J&`b8;t_;sDb?&d9)}#arF$IP}9rBQ?Rsq0ry4=>{$u^B)a&8 z$4Z40UNl5$q>l&<9g%OQ|7u@1dd(j&#S<{w{+3ue1-&i^Cg^%on%$J_5ue|)m0XZM zm_TFdu~bB(+bVw>%kcfHf3eF?kAVa(MR{Y1Ajc7>9MecM@hd4@Kq^7(GXRzu(C=+_ zjIQ`q5l1;ADJ-Uz`-I)F%o!$T(($@or7i%jxZ?Zr{5_#<&t;|Y)#+eIM-`g#<%)`_ zkZl{G%?75E*Wn^ENS(%;Cf|~?!blhdS=Q5h1S0D${59iDdp2r+R$|c4&z5;c+8Z!% zIofW!X~POo0LEUU@2@m7pGc1Ja9ts*d`Kihep5`muO@InI^Z7$0B(Cj+@-$8jD|mc zzCpA(TQ--iwqqOyw;rYuVj_JrQyk$}yMrEy8%bXtYGsgZaTC=EGw zH0|{XzEs;e^n1zr>i9DeT+NctriHMDiZyc_oYP6is6j`nvl!eExz@8p)rlGeRBvl( zKS^wE>P8^EPL6%2wb2(MH_}sM6KP(!4MF<#Ac7O((|V^ zGo_u|#$wA%PNgyUO2AYLG`=i_uJe>Twx5~s%M#s6eQw}_{LV6uM(swt1+Aw=ty4}# zH#(<#wbK-k%yF5TO3zZqJE9r*o=P`Ld-=o2pSAwWdmtv3p?o8mbC$&O+T+}jr`tx( zek&2oWo_wmT%2hqh8`)wd=!VG&N&iuKt6=oiq5GlHHf^FV^Q^e5Th&hTbDM*PMXrn zj-~ZJ0<3&2Z;ZEJor`-ea;aLc#QpHd@Le$d`x;$e&lF6TXE~K$K_OE1!{2t8kSQ&V zjinolI^)SL2<>G=DX4poEYHrx_~hg=uqRdp!YH9>@wlnk>e<HeHKlnvtc@;Vef3l>7JX`rjo{-#jqVRJZBc(Bq2i`=9e#0= z+btT}T1C-+JZj8?OFu;yc!Wq`$D`E=5;(N!>YGi!as!^f=B3za2>}9(aGzQsj-SdR zfe$sUk&vE0S!HI_iJH&*pao9!t7*PIFV5c?+04UFqH{024U}CXUNU-h;4Z~9sgz^$ zQ5u=@0nJeOz6`uMMXkHkd|fHT&h3<*I$VOo`Jn*gjtyl`ZR= zz_V0>;{A9$WTUX0scPT1JjxS+P>f9<+rYJaH} zcro&W9ETWvimnQa0nl9V4JQ&x`7U|OzPKN$*V-c3HbJ9RF7=^MWCRk6AsADK>iDIo z4#aiwUEo_s32HTx8J+-s1VpN<&i(apXTOGeEzk54_VTBQ?1DP`89uOkMGL?Btu_B1 zQ-GiU7paG}$ljlV{PCuYn@`AAt!cpVkj4X!8<=3=?5o;eY&j6x!dN3@Z_}J2ndK`u z<0S;ItSs5?OPL2g>}{8chA4L8_$Mn6s$_-p+YNjtb#*xs2{SHJOP!RV8`1BN9D@Bz zM(EeAYRAkTzMEPA4{!6i{T^4EjH+)pCG##f4h9a0`k9(h=X`N~@|o7sxY7o@DIeS> zvi5!SsuCR+9k;J~Dms;Cs(A>1`7E67ol7zrA1bk`A)DLe?D$pvo3|jYfC7+Ec}o{J zdkjdwI=6Ykn9S4}S8ih^TS&k`LSv$psubR;cL9>w4|6Dm5Ekd18(P&y;|Q;CI0KMT z<&ovdM7Mc5NXguu(?Yyo^4<_d6?sj6hQ~D8&kM)4K=4nM9r^cjCT&%;y0}>9Jx$p{ z9M!6?W&PBr=COIsN!9BgJ1zp34LBrPr*Vb_T50Z+Bgw0bDx%*(E;T9wf(NPKRo`Wf zuX%JKbkwNm4Nk$yHc*yD`fb=@Z$h^eVOrgsUfDYEJ-t-wgVRgX4p){hJIj@p z=9hK(zUI0e?N1hKFwDSVNJ~>4r{7Q2;lJN*GanNe;qVCF-dfdNh5u{{pM}%oq`u8e z?ePGS*;k>qKZzO!DUi)&sjdO!kjOltlMXG+2dIS+&SniP^hGQ+)RYWSh#i7nY+q$4 zXlTdbG=dyr!7{{<85eaS zLIRz1rStpvoyco;!1<4p(N@3)2_Uva?V>n(Tu`c_kzU`@ZDx6;ZQ~{Xm7p_SUmNWe zU);sj9xmRXgqo;<=IP4)l2G+sM`%OFllHD%ypqk;!C2)C&uyx@Hc-`%uh9HU?72hK>Bruy^ zKhX)m2siDZpc2Zm(_U<0m|CV;AL6s8h$8>bnm_uGZ*XdqO4y#J@SLv0p{dc_?`iUD zr^O=1%~Khnr^5`Il3yK$<_K-$zPWyO)H_%R^ufHa>sTRn35}G(RFj@Uy~F2`>#XFE zAC+f7rqqXBc}fFi$1?||sET~ey8}CZUnoda`yO!wI^nJ#%qZPWU>0Gvyfzb&wW6yC{n0>ZZ;<1 zVGqe=8!I!=95RZZUy#WRoAv5yoFQfX`-+bCl>W+u>EG2E7oX-mTjUkK2%ZjTg?b$1 zo=lX$6?D~ROtYkRHySUZA(wie4F@u6SrdI;3RM`hL8&~G6}b|lqqR%9IYavsea)9P z%&hID{q~IdRm)#5jqTMR=tDR6$ccOE9nBpoiYV(vbYb>OYN=&5FopLXW2Xwl@!>jO za*eXyN7j#HDby28G^n3;Z8IKS#!U!HV@zal!1QO9Fz)sm#DIK%u z(!kA|NOy1dD@<}RW3^JX;`uproN5Csw&?|J-slQjyl5s%3?L_mJ*GN;ipElNczdSR z)k{|4Z~5^NE;naB&QB&*RJ3WR$vu~e*Jgb1tu(AYsFHXm`n|?Y(N}vsaa8=v9cm#+I_JB*g@R1Gm*yP&vLMv_XzOD1m;5@>gj_{e z@0y_f$B-AlLlUCR8&~U4@w&Dzw@2{>TIaB8_DfWV!6_{U5^!?_L#KvebI30|jqhcAJ#hBS#xVVTXdk3SjbMrl1f^+jAGA$(e z9w+>054q@b6Ju#!B+l^@uDbiQ?`=JK?Ud{USGIL2!Ff(t<0R7&YFVo=qvN_H!qRN$ z4FQUkbs9VPT$uu{t1W*-r>FsTgCAfogTJzT4?SK7j(qoAq`ULL*EN-r&^niE*o&Q7^wV3JPcas<$?(g~Ua*G|a* zu&GOnXG8l)I8>1V7P3XWr~5(`>(zFGv5)9sK|)}_^L&K%lo$P5)oamx3qqDI^AKwIq2Ku{BzO5cFKrA4H3$d+@t0H zqtVJ#;!2p}?jk78n>g)ik5pjmtCjGOtkNRAWJ3Fr{1Vh5awOw6dL;WK=Q_9a7Fk)7 zTAn$&6_rejEP-iN2z++0hEDHckiE=n_h~ZL z!hsoD1i93`n>-b}Y@Go5R2qvK!H)@yJA`8VR@yEG**WeXXiPdj9+~l&`I@oj5VRN) zyuOn%lQ{Bv2Ai8j20<7Pg*x^Ze*30sal_CC;FiGY)yam!%Q7Ra{Gu zr~~zI!vHTHM>x&`SU}cYcQPr$$E_YA!(V*Ge4zT|1p}}40#SW7l=RA_uI^61A&yy0 zOEj}ajba&7J|+iW2t(L7=R@e=(P(2lB8axFY&NH<4H04QI?5fwWF1lbOMkA(hSSeg zZGfLx4z9l>K8{QdcITU{9=J`BCbI!Y#O4TLTAfX~+wpW4zpZvP#agzC=II=lte|1S zwM)mk1DgFNqwN%jw755TuYpriKYeE{xu1^SJmL1m3T^6LS2dau>smLbS(>8nHHAHMVkE^8EJNv%FXmku27dVVEN+u8$&fR63 zGt-u10XmYZsM9L}c^G%R4yoCTmZmkAf(gBsF{rzjwZC{V+_<<<`!;_2i(#&Z_%5<$ z(01fXYjAOQrPr=yhia%lOM_~L>j}~$owx1TTu6LiP$Rc#DqqTG>%n3sREJgO^2k!J zmREhkVTXKwPq>B8_J|u@Z*6mQ_(=S8;QGB^3s<%wLV*uKlvnL?7(y{X^Ie-fT*C%g z(7Uu(BBcts7`Q&(AZI4%XFyie(;=&nCw4#(7N!q70FVxrp%M=|@C$H_?Yr;pD=1DQ z&@7rEY%6&O{}rka5=6GXy3RU;OqS^fXKv%7R`^}W&VWxFr2G3p0$XZbqHJLS0+`1* zKDxOGxj;1I03tR^vZrmW`;uiQ0HBcPyS?g=aY8oZ=EAABv!AuU z*}+oAi|0|fE~=qbTazXV#r(B~3i;E+Uj{$DUC9uPqHGWw`)0eHmxlvGXHU|ZYHV?* zih_P_zGyODrgAd@Q3^knCdUg|YVo2JjzZTHQ$xaNlv7d6Hj7mgc9ru9iGmf9NHZ^S z@rXDU z-XeIQY;k(7Q&@2>wWU+%NeokZZ=kDT_7xj^{HJs=O6_}}#zr2;RnmVDUtv%3J4qBH z2b3-8m{4S~7>Uv3S%4f?78#{sjZv7cgC@dO79X-Dsjb;Q0U)y?WS#w4Pw)^zs1>q5 zgFu};buS#n5U(k7Q~b+`#sj26Uh^?Q2#A7XIV%)Itd>mbO4g-j<%OdEg9)M!?yPsB zvfY|^K5Ptp5lBn)A*SWOKy^vXGUPskMArU9vzS;iOtp;3PLBaB z9Q?8_wQ{6Bx;MyILiDDyr38zOJN1zGfS z(6W=p^(;ggV3`TvU~Xa2(`Gc`LjSlH_*ab61D3Ys{KJ^2sXAH+pkmhMS)6H0zyG(Y zv;9Eu-*JbR&S-s?N85E5$dnp4DwYaKGKle3{jWH-Ri*r>70tsxulkQIgdVB1@4vf9 z|Lw*98-o_igRE9<2R0`E4Go4A-am~#3W+VF=U>$N>qwDN4P;wY3r0)kAv^px2th1H zB;XIN5&!V{wB)r+>8wu|2u*IYryZ348mdTx%)Goe`@VWF}QxUruqLXd1Y$a zGgIFuxc(c)p4y2wghSE}K$sufio(07@z3O7TGc}G?;pU1|09T4 ze^d_z&+A|Dj~L0n@o>L?!#}h?{6pzk`bXv=JKX=_pZl8mzu_MWQa#W*U4;Jy(SoIs z>TwThMB{~7Ox}C?3+5qf7T{Q^^@;!rXT_H zWqgu;w4AJ;yep`EY!DaQR~Df<@p_n8Z6c{|C|l2K5E z2LJ2@fSmp1Ed6puYb#F_=ght8_DgFFJPqf$TZZ>jmK40y`p57FXZ7`Cs-AQArKP2; zpQ))uXWAD3em`-od-shfLsP1$8g$xlw%Pf}bX#@9ZtN2mPn(E-#Nz;Aqf={^e}20U z=>Sg~aOx;#W!mV$%+Q$qLK1iRuNWd-Uy#D{I|tk6^%XV~yPsK(FN$XKOF89P4Mymvjy42451Nh_#Ux5Z0o)w*klOWUExq^sxDWn z=__{Q(#qBe!Bw9DO=?wjtg)NsNi03~IVANt?ix7#hWGkkAlnBCfkC%Kb3g2S_*If( zOu;rD&!umJizb+2K$d8FKMNH$Yh(#jSK5`B%dM+uk;wrax4X^-n~S|gJrsB7`;oYR zvTK$d^udB@uw3YmikdJCx(_-!_vVGppar_+%V^HIHX(4{m0Azu^w#{kQvX4_-vXi* ze6>9*Rq<2*q;kHgUQw?GIClNJ0Mt~=|GP@j+*r8gQ;|_s@3z2gU zU7Z@!|1&Wi-!>S+Kq3~NA9N@f*ZqqH<>oSqE8CWoDW|KrgyKt zI|}@;=rc3ww$hdurto*nK7B|)3JPaBJ-g4N%e7eq-O{dP?-A1ne$+7NAzUls75^AuSS9U*fwFljO>EGsDl zXjh;15GXf8{GViKitUQ*&O=vc%y0@Pi&UO9oloFyUhi1o$8`VI<<$O#C1cYi-?}u9 zfk&6SOng$GD>_%WbjhP;dFp(ItW^jw+bnRTwpNx#Gk3)n{B|c{7C!kOBO*0CNl*gA zx?qHnkmNTWpLeh9n61^Cr9i|J{)4&bWPaKO;`?m z(IPlqI$DYrJH$nsgj`+4*w6(Ym$lxWQZ1K|4IP=ac%;EGK(t>GO*@yd0o$1WM>p#G zb5vk6@Cn8%Kz9aT2f(Cl^IPk!a-z`MI2DdxOX~sl&Ebq^N2B|8!bg_)76c6uwc3|p zB&4?VRZN4klEzM2DtBHR-bAr^^j=MYmv4+({c@>A@yy?YwwO%$5*Z1hkv#WD3;yivM#TsZ1S zkG;RcyvZHU3-U35z>;QYXe-iPD)K`cf^McU3rmee)KjW>vC2f!&G|;EElgQNw&Rb| zBjfWd7HBd}{}}pf1CZ)ge&AO7NXn~WRYlFb9kj`4R%)syesr3XUgAWqb2L3!0h?; zqNAPHu;a&6lX@n9t=|=Ru2rAhn#0o0@-@DYv6fk~xLqt1(n@qpqq?P8-h~A#29=<29P*JfmF9Fs)4q08T znyz~{_k!XECbPSIfcESKgA~$CaIXPn(T|$&zG8-vVigX4a^{#S`ubt7o^FPIJ|dug zIZy(LaHS&O#XS{U_Wxnuu~)JrH6@S<%{T>F?z!7ke+wniFL+6Cz@5r&kRs>+!%#pj zx4*liW#A3ibWriSws;=Q`)7jWuX)NYuK$ts z;i~9`18jukZ1+pvkeSzVvGW*dGP${T*SQ~-mUG*+p}AqG&;Pan>XcGJk$aDC{uQHT zMN@ghhLgh%v|;0 zhjlnSV_d?uKi7GtK!LE0(gC4N>trViKPl}`cx_A&Cki9_ucog+Q zohqcvedkhFhgBX&Poxncdph^!8IE46@k~Hm=b92V_rq1tLwcn_wvbbj>qh@zzHF!s zY{YmjVF#9wBko1Xhs!vV)OY3h{IGQSNcGD zayGu}(X5+w<5I)vV*6U*mPGewJW}JZO&W2}?{O3YHpg|LTZp5ManDcU$c=hX4E?_3 zcxIhkw;$) z=>>zgYc4D#^M9voyDCZ=1y+X56>)}cW~Y{&;bCvHMRptGOR)y#x#ofcL(b?V7!GIP z9i!^ay98?m=9Y`E9P{|H&&%0%(v{2)-4Sl!)KCd>E??pQ)!tWzwYha|77E3RHN}gS z0;M=D(&E}uycCzL4h=oU*JdVvK zM{^bNDh{n+O=-WsOntmNp7vYsXU$AXG4y03RS9~N91(q!3DE7PsPtSlYSOw(Uf ztG3`I8zDzHLWW}gRs$w5k!a|Dc#k9kDe!vOvmzKdpsRso$!F$j-L(sY%)4Hactpl7 z17A(|>qo!dpoEF*xWm5LUGqas^>ZVzdIERu=cqxCZRBQYrF|x(jJ_=9$j`RRf6hji zIV_Q@UN;R$NF2mKIL$Qn{#=LZcJQF(2wl5#)h^MD9#EoWDIFFNl_JgCCR8j(**0|} zjtl;SO8DM#KJQ~kssg-kj=H$2Y@5d{(&^G$u*Wce=Bi0jDh@1(MCkj9_RkDXE^ebDru`n8mvYVH$v<8LF zrcJ*hI)C;)>bEIxCLg4+pMU4J{p@;#xL)M};3!rE^+CUf#EAZOG}jTKp*1ine$O#c zK=#9L9U6J0FiQ&EE=Z4tzVSSuM21-9s8U%l4Z-PdbR_LZ)N4VTuVYE}TNAiR?+0TI{Hcnl=7CLGyT=-6h)d>nb zUCytBhLMR$dYAB$6y1!nu|Q8}_4qD)mPJoV{mOQlQBjsxpCm;B987{l#XcWj9ITD9 zFn3LhRD{HtA6H!;Ei6#HKrzIy2iI>aEqXq)BYyI|$qrTU{?6*POb$b+z}Gt3+Pl}5 zx1?;%!-YI{VT3dVQWjd<3x#z$B@ZDRo`V(stV#NJ)2UC!nEC7Q;F$rUK4zq*H> z(pPR6A6+<&3d6NMpK=?{H*wy#``Tl>Wt)t0x_c+iw?mq2bk&!>>gVa;;8eMyB2KG`(7|Y z@KYb(xtx@PuMhJ*Pj_^7f74btUzzUGqtK<{Ksf1)fuoc|kpDE|CwkMHamNK7YTjv7 zzTB`m&ntF+ycKw_eYctA*V0yI(mHt6&)fUVlu{GD%@}$=+v3w0fy2(uBT_LO;Mj(N zg9}T$+(-vuZDMjk=T%#Med>3LOY-a*YdxzCUNE&hTd~ntDUc5v1+Xe7F^&?(w=SXF z-a@b2@vsYuqCt&&0@jnz`1!NgId9g>Yq}*ZhMShN9h=Z!<^fLN52Mmp$!rc_0aQsm zs(TaI*BL9qaX{<^3?-(w&vD%zmxVx4_-oXJa6|}&!OJqex@5gO->AyS>PuReNwtle zTB|=&FpxI4!`dT-G7qQCnU@w(H}FzJCSs98iFsfBsEViFmxvVfVqZ0n+hA^LSE9gG zFS8?9#I0VS$ZK;OP5sTBdBXzBEw1;OS2U@W>;5!bYxG?=l*M^1@7OZhK*+N7_AI>| z8r;CG3;l3Vc|Qz$eEzOjEjJXLjgt~hdGD|=rK%5E@~K;NETOtSoCz(u-5np~;wP_2Id2u&|K59X zH6OQm;w!BCh?vf{%H>*CHB0f#&3D2rTQ-&74l8sQ{4(lX=OiXoiYZ0AugTA8;3GW3nCqWl;jD< z(JrG5m_Pj5FTrp;RM<-g^dv|}V!?O>B!+^lg2gc45x5T@Jtom^mfQ*m`9?_M+6GlG zRJIUh(0?d&02L!~pgU+h5PN%-nv0FBSfMV`MR$c`XMDFp8O%$N3q7) zVv5}ii5r+!x-{?tm-La;86kX$0n+EJ3J*Hz4(-LxdAQ1@*D`d{Xc%^y;D{Kck{pLI z8Pt*VlAw`z_q018AF_zVU2_N{qRm6c!b0(x`+|g%Zyd$EKp;OPmj=|pd*UfYFDXK^ zyP45g)gvLyvg)43XKwqZTB#swszh&e63<4(481^AT zhDA$X>Rq0s?xVjUMqDPNGX5qoPqsbl6!>xCm}?+4U_^48gd+J#2qSdwxP*H)0?(_f zQ(!=*k*th^z15rGTI_I|VlK=n%r|D)4FWh#2^1@@9Wji@v#aX8*$H=w5L!HAR6>jM zO^!B8BOi6R12#2%Ju+1O`jZRzr8r|nAc^3QGedFb+AQQrSC|lTeDkHsKnN7-!8wq0 zdg8c}{kV#-j2}t|WDfcmxQMmJ^j0l>l&%_;aX%PiO9?bsqgU=9azXOlR>G?c*Zh2j zbHH)Ns8uUT;d9GyNrl}W&Hh^N*VvMGT^P7jmKh$&q)$rQ{P8wv?3>yZp8>&_WL0Qn znn(otd4K)KuO!Y5qWFV{*FxTDkh6|5erM6^;OihgV4Ea`;mx$*g4(0t5_E^t_dPda za*IBK$N=^tMKfF^JFXfMUY=qDY3adbAo*AVbR=A?9=C{Pz5ZlPM;7 zsSWJuk|(8g*he@R=$>B# zxE<|b^xjt0an&PY3d^&DWJe}Ar=4v`9DJ7gG8f8KV<~Ew-Lm^i2*p5F;(au8fDmqp zprC}R=joId!cXOHRQs(&5+&+}q<%S&AlfC)CV!kpU>iT1owSqNy-#NO4rK}yHC#eY8pu)#o3jwB7XUKzkZXBNtaRlgj73o9^6YS_c(Y;sTe~OT;ci~=6t#{ zmFTwQ1=kkQ4ZtG@#)@4PwY?%DcP1$&9jn`K*|c}cxQd|o1!F58&tA$UtiX?^4umlM zrg%Pz(9(w_*F9VZ65p|N4Zef}AmE#?cP_uZ`ECG5F^z&t&efH+4ah9FG~RjgxG&<> z+qt$uS7v$C!3JhKNM&fntuANaW6caHM%Y?E3r$wrE{Ty!E-m_ zzcekFR~f!g;`i|_R48cic*PSj`a;*Y^FzLfA~m+sYC!_9*t!Wbj!dug^a(0t%;cQ) zRj(2bxF>xiQeJ?rmZZ*Bq?}=$llUmETGlX4|SLlgjo6Te8m7!9?M|ps+lo)k-_HFC;8E*4bfLEoI4mz!aioY@7 zMB(YrdS3|4J2c<7Ut37cD|0n^G-Mf5VW)UIOj$e}@MmX(y6;TZt1BAezXEKxP+xvi z%5;djISR9R!gaRWpxUoU#LDra8@J-gTQpcgY_u+Mgwgv#MJ~#5%vU->ED~mv2bgRW zhw4B}hK3%O--O^ENLX2-WYD92_VOkZ4n_u^cGRQ&>E(tovXr<{Jy44|2MGh}GUWdD zz^@09>9w1ru0je$1-hiMb1{wp{caI*{8@~LqGDS|)4sT>OFq{bZOxjA1xi>jt!)zB z-?oeuRTU;|91%u#;8T7r8so?NErC*1(Zi8+T>f4rk#O+sYM5ID^5c0l&29y8~YvG~*I)epL z+y$~;&V}8bs-ianduzA1qjuUJ62F1d?T2Y=l{#;a_k0{je=G{Tel}+>D8<`S=+$-^ zmaZh6Eb|LA=7PmVNo@Shb=JC&iqsVijw?BiQJx=bvbtYUtONnGWU9>eu_GcY1LL*K z_z=(hK5@L|NU@!tR{nd#TC_e#5;;HCdM|NImN*!}lFAD<&^G9;$COHSc<>13lPU~8 zsFF0)w0hX|ne}AhxNQ_$dN~-NJZjOu1~WTcmfVL+9Z}r^kengdCpzWW>e@F=VzrH1 zhUF?Rky|--dA{*q0=hLv!p}_t!w>c^G&BzupQb*-`C|4t2@vHH9_3_MxnCt;)g;$V zJAG_E3t>j>jj(~knohq8Z(Sx7Voq!fhu9pws+ei=P|yumAW;pJp=4kF*S&z*mGZu9t)8W8X1p4 z!KQ0cd(gF*MAl624qPHMZVYE@^$9z(AaHZnDAw%PWW~ixHbB!aL*Ro3WdXJ6JRu>E zhbY-QSaHAC{537$VA1IMDTl?ltWjXda0*k)B+k5SPli646^TBjyW-aWQ=fRVc$U{e zX%i^ClO_Oy&GmXS8c!1$b6TL z*;~00BiKg!ytL03U9F@d5@>MX{ImG&oUDvnP12OzQ2(<48mw>heiyuYYSXQDHitnU zYq7AQ57$^B@QR}my=ECWDeO~Fl%)3im1;sseXvL&T)p^q2*x=8D_3Fh;tu~5Ll&bS z->>u@+@Ie*`ZqC-M6r!5fPn}b&paF_>IFN4PJF+8kiv3&hg@5xsg{HDbp-pz0-L!{ zdnvwl(uR<@?^$*u)4v{Z#`HNmR|WACAN_>5E)_AVU>AI2*V_Kh`j+oE9%ck5%_kTVqHx8sidmabVW9)Yef&i{Ax3rLjWg} z*2bi+pAw}0%;Yrz)?}z4>F`vL#00>+D=`4ki#%vS5-NWtWA2IG$#{R@chA1Y z8l4iw-7*ag2$IFock$r#?m+b+O)JD2&>QR4^C^G{y^lm$EHhmSz%GyC+aVqgS$ubw z=|J06`2~?v#pccR72Y$BORqY=u%`L*G2S6$cI#}~f5zr1_*bCjK&|p6$z?+|0WLwy z*i-yuJw_KBG@t@d$d`AK+HC*e8z7I#jd!ob>szHse`+QW>$1$$i^zAkIf(**tAIZ#DazYlRt5iywUNlN%~Rmbx(tEV6f+l`T`SYEoNc z>Dz`c?a&mOZhUh@vfcW`GB*NTMywsm+#SDb8Ro0VIv)X2ZcuoDa%#aHBbcjW2s4Og zuhNX;yDi-Z(~wt<Vaa_QC?k@~tSWokA zFE_J%$_kr2SYA!gi2reeT6fZO$nCct_}1T^&7om&O9~nSv3~qs-fK*Y(T=e{NhEe* zL?vLGs+cY?Jc$9Bs%TNu_9vspgjHM}k2WKjXWlsIBr8hH{6%3@mL3jrr!|eU?a#k; zG9NPy`c(X>IC{xrleYdMAqvrSJn~wK8wU3{OS4=H*7%Hjxps4j?R9qpGeSjy%7sm_ z&p(|}YWJ3?<;tOk;Y(da9WHcd?X#r&NZYoWpksZ#=90O6Wvm~?((YHHOy_REY zAcQ*^!svbTr*{W`)qRcpsr`dntk)pnCkF2;PNY%=M3K7Z1*1_1{b#Se7Ss7{SQ1%P zVl?oUNOzB>1N@0=Rpgo}u*}vP=h*qw^ikbi8UMZvu#r??Ml3YwXiN4hi!}B&WyOox zRxQ%BGPIa>&;pRYiAdj5Nv}i8x6?vaB`&!XZz4czj_EB}wpn56B3Pi-HgLH@zq7Sc z9MlXMc4vKEz7WQFnJsmi{}y#wyu941JX zzmV34R}Bo)TSI6Z>WvJTLw}H7pjTQ1%y0e+48KE{-CpQ=w>sbBcw>hPT+}E$XQ}Pr zjV_qS8{I%_$_InoEzwaEp$EK{(fiw4vbUv}vieJjl%X5iS* zj9QK2?upSVH#Fda9tKXIzohSg2;=s+&4a^?uD30-da7uIElPEck>N}+B&8T;o${<1 zX7jt*;&OO~NV4Aq#(!1#Jv8>wr_xD&3vI|8vovzBk-^!S9%xXvD>KqWRRuMzc81oktFU)i;2CKZYJp0jQH#phz zJYe$dg}qA`^Jz%C5ZnCWG$g~V|KS^l+IaZqiVJ@sKJN>MG*O#cosund{tF(lC@b#i ze%0U$qo1t-vHCThfcQZBB5CjJifw11^4IoS)wTFA)$hDa3Z1#!Ixgos=_aZhrhC?8^7ruYbRfrbVdGAoKDhQRQD~pgD`llb+Ei zRstp=ovJGV93=R+EA zv|!nZW|}!zGrSXw%HEW&g!)dOLt8$T`-%oIB&({eo9aYeQ^2B%QllRGZwvOwdkD!siq3>3z+UXqg{3+_*o^{M}68jhpD;vuyQ$@zR-2bfdC{_%NGUVxTmmeE(o7{ksbB=_Yd zBjGp`TBO~u3)8basC>B;FBXv{t5&6$E=2Ohfi7_m2YcbcB!!F03(6Ju93gu>{9+Y~ zETFD{8<8RAN=Zf{;*VWcy@t~EakPO#{!;R$^u(gU>gR2}QUj}JlY(FR)B~(1iHE%OD+4*=ichE8RqRLhGcHgE__HCn73V9oM= ze($@!6c5`9;2)l`%;2nB@J+8A-}dqbCXS-(azp?Q9nb!;Eoaa`F!K{wKbTdAvF#_3 z1N(Vf5OFGpq1PTi6OW|+OoeYyn{HnyHUW8`lHlSCVb|S?Mtn1`YrDEZ^INXhng;Lb zJo0`$grUQ|YlN1UKj>%p(%zQnd;jTn+%#~*4}NpQ*mg=jTZ@)dwQsn^?mKX8Qw&_w zz-u87k2gnCXdtbUF>g0g^RQU*XhqyBNJ&A^qet~Ok)+n}g4!+X6I(M9&wQ!f0Jne}W_?uTX>B=6z1m>p{fgaR!amjm3{m zD4Z9vfAr#ra*k($hypr&^7e!=}dK+v#aaxE0_D;P-nlOJNw((wAdd^y2`K5(*O%KhLg$|;|{Ay z+4Fv^&B`V=bUNjCL=&VR(F#f`e-Y2VQCaKte)DMQ6nXw#IAD_Xb{|{q*i7Vaj9{M% z=3BB9TLa!-gp7{>uoXYpqqLAX!k?URS&+Py(Z+rn%jI2VU(ax59Jyk z9{8Yn)7aw@;ZC7!uiM4j0VpkRaP5E#AsIomO{ zTq-bAMJ=#-K)UILN;Y*LVvmg;%5Y33GDba^&oTB%US*CPjNX2|3bO)J88{!wS|r>I zy&iePl{0U8q2J(aZ9V(@SDIqhYfI&#Y16@AVS&;mj1)It49IaDD@Vq1@&I8PKVQCm zM#I%chl$UcNxaufPO;8gDwYC+6jw>IAcRb3D53dGl{sc52S?lLV;HGg@XreG-AdgW z8~Lw8)PYW#ar@YIKI6vek4ftft6_v#QE z>GsK>mP-~jlx^>x3l2qMg~)FRQ?=}T^+QB%tE>T#^(Fm*FSI+zW;kCVMQ-PK@nrjq z_6N^d+wI)5W`WYVk)NKc?*>XUt$V6p+6ZAI67{YV_bi3{(AJJOYt}hpNkuWWDpH4- zje+EMfT@(;zZt+81iNrko^x=FgP$zTDJvF-DU~vb1vA5+yilV~k(Tt`U-C18kY&Ye z0y408lr(KG~D4IE=9W>rX7@wCc3&-B{h58za~RYq8!ak931;i{W+ORRY}=2um< zF3fZ0p3MXDmswB~A?S9Kz)r4M`B+-aoikNzhSzX^QGg>k2kih&@VH-W=qAGQ?o%gV zNqT#;#+~pa6>}$zyHWhk{lJs#*Hc{HPWIpn1o~i^o7MFxvB^JVMhtz>oET1uoMLZXO?)+U9RdMf?t^`mzdn#5QFLX+N7veK65*02L^97CdsK6p za*`wDj~N7&29vn}H~BXHAs;3m7_$~bACfW(cIJdcVfj7w!h6wq@!h7J*ZwH9g|5XS zZJ0JN1UW!z&t}|?>ZR8~Qq8!l)%EMU^#Kr@wa{R1#*b)c5H|gJ!~^9-7FoCLbbGj3 z)$QX=Vpw)PX~JG&y&B8$e|Y9TC_c!+V42u^(Ug(bn|Gh5FsEqR3 z{TR_BKg|8h^Vs-1uRjzEzERY3E!K0Jb|LI9W^6ES=L?Q|M#~TKdo_{>eR4(_H?*3Y z5I-~fgGDG(ykFIS2i-GeF9`cY^uo#aq?~ zHfd;=3b4ZA7Ol71mvT8~vO6&Zbf`RlFjvAT^@l@r%GykDxlozjsD2@2Cfcm=fvfrC zQyOaXDVrjajO^}p9jMm-)^wJZszsd`@IFdXbml@2g`nx_CZ0a zA+ui;o`)DU<(PaW(M?S52qP)u1Mw>$0N0R#lHKlhT+qsez3LpXA6y ziA~~W>#I-bd4@P5$1shz9`jIG63%p{8Cw_$S&oPm-kuFEdhIpC?hint6PK2I-|#=q za3n99nEQA(YW}kOCKOcqr1az4YW9Q8?J}N8pmAlLP4ic^{mtSzcjYYXf^wo;MSacd zN=TCH@4Z->n8i7lOrqpiC=1JDDtKAhX8&uuMw6pII1T(u;;2w}ppuW)+}8KIl-D0U zh|FkCRMzv&Hiqv#=RJ|b5Bp1Y&lkbHptTX;lEE4D8wn3PX8!7kMBU`M?G;%M8;kFX zP)roX2PFG&hOl6*?cx)(0Ka|q(BmkVCP2akKd%E>EDni0dE3lx0VK7qH2~`Q`1suT z>_x~@P3-315?%}6wVQlW{4VlqEsN7|SD%=(Ov98(HQK+m+T`SfS;GGT1|}9R3B%z& zpwGhdevMnY)a<#LT))YOyIznL#mN%A&%6s9ngxD^b(h{PX#0|I~%_y=t#& z-;nC*anY$y70A5>i&5C|vA>QN^DZ&hG?mf>k=&@t`svUfqE>LVc%!j_Ut9jZBfffEkkk&Oa~DEe zTIMRl&)%z~tI`Rgv2&g2(eXErU`Tjs_tWsm?Q~IZww}J8DsE*XWO5mJx$I%3Nq6J~ zQ}!F z8|;}>OaF=u_9^SX#pX#LJ|JB%%X|Oip%3%Wj!>G~cI^0ZevZ7RvwUxw>~=2UNI2V4 zk{{56qNQ`8QOnI2#Y6V}lNZqEQ-Mw^;9SZ#B$=7}tLFvN2Lf%arl~|1bdo2I{A^?S(7pU+` zCrLR__S%bzuuvlKCTb{JZr;@Eyd+(KW| z=?crx0JEKP>IJZ2H}2J%^CFlM*Af&83Mc04&|V&vWUgC}eUA_Sx3T;yQQbEQZq`N{ zO-5qHsibYfoW*ua_6Xx)QUkTWVdJr(qa(_~=Oy3o0oE4keshyC9e-9=fZMKv21@tQ zp=X^2c3CuzYIT_EdNy9s9*5Efa2!vERlG;DCE?ego4aweB)=I&nf=D#e0PR|EhRZQ zajJ1V+V<(crpUj?<$q2Z3IDcVnAfgvy7yX_+;>8r>B7q9X*vzdg^_*c+&t(K^ks|B z?boAuC+;6Em0O>dm!!M`}>14$VE`wjo~jwse$ z&0URq&E+tq&uP597WG_uqYodqfuddE+~OB;$B(TWF=VOub$fQgIeS)Kd1(sW?o7pYsl0>fqjZ|890;WT>DM}8a<~aB5OSV^2>BYT zi@SZ>7_Z21$*|!fJJxCHP;}(x$v3kYzB1ux*>2?ZKy`J@e{^l~I$yj%?78)P=Z{UX zReVVLPtSw(@n89F^-nGvfIPty+yc3*1ls)Etcg;W9QQ9pnCo^NRlzn01yh$NfK6 z4Hpt`ltiDFt;G-u#m?S#=GxDt7NP%S9{ZSf`Cn&QhedMmnD@t09WSPe-L9ryZL4%8 zu_c;z1QVFdL(hAQ)Lhim)d5N%!R|Z3rAWBje&i<5owpWr4J-xTe}403cQmx(|14ww znHQh)C3JJQr)A&2gIj&MZ2MzH>m@S#zDCZ=&TH~FOlEklM$OAX`|*O{^Koj0PN1hs z(wDFK69F2|0o%^~jmrP@Y$Ob-z5$E~Vz%f1HA((^pe6VTUh-%h&m{`~<24L87}y2R z#zJ2F=dk~Gf`6_?pEb$fDq0Wt#}siSK_AF~VesJ}=kNa?$bW|B|D}lB(4R6qGyh6E R{;wl(W19K$_$`ypbiE@>hw3qk$14-WwuYz_hQr_0+1`|Se(0i6vA0sXdx{BtZD>OW`S z5obgHXCK1mPe+dc=O+jV0SHMEL1hr+Nd}xJx+u|E1ArZQHj;lz1NZ35$?RyN` zEK1d5N;XRW$xg&H+^W#!Z`f5H&o>s3oa06fh7Zi?ZTIO97h2fB@v-`|&(%+G$C zy3ju@b2In1vT{3gGQ>b2{Iv<7BcMkcEYPOHNcjG>u|U3$#f2vofcjfgpb#KpIt!(*iR>o_4%DJ5b4umL~?bwaI{=#mgOLl2}mGFa2;(C!?YPbFe*w24$!p$+~w zZf3`zP|j8?#6BUWjIt!bNDy|z3pjymZFoEH_l@xbf4C;m=Mf&f3)z$W--FDCf8&rw zAa{~1iJv;a7sAbWrL~nBKC)x&!>dOhNiqw^wf6qN)W&)DR9x zt-ueW+8@}x`jAZgM863vZ1x+^b$BUQ%Uu}s=LeSRkH7S=+e0*-(FD4PHwbHR{E;d; zF;vZcI*ka2nMDLbJf()%?XJ;av#P2%6scV?VBLf@SW2!ggA||-)p0P-C-*~ML_!&v zUZT!~l!tVb>-$EsYKt-z)$Vu59_jXcQmO*r6%KU*EtT^K8D2b|F*~&`P>Sk;Cf?U! zl|NVb$AK`_6rHSX|M})(aEoY5X0I^P=a%Q)tW--C1ztB1huVl!hj^zFP1f8S4mZBJ z(gyHyqmIZ?h>QN`B7))?_TA#jNuYh0co%EPg>H#aNle+Gv+00whQm&N{ubA%dO;Y^ zbh*Zl9Y*~Lq8Qb}i2PDrQ_8B-x|^A2+Ua$0hwv!o*H-nFr{0=Sb3_V1Gp{kS_mPI< zAQ45&Mk#d1O%w6aR=n-vxU+Eg=Z(v++{bn;>x)(Ss?T2)nj$~Fu?sg9M#!U-k7c2H z=1p55;ghN6fW-OZPZR8POJB&VKD*h0GiFcj(^FY@2ClCIa?c?+z|Sb(RU3~5?Q=5s zH7iniR&)CWhvKB-W7v^x=^KP-+X7J)2n$8Dmdi9t>jjdLSXnC66_}wY_H2g$qN>v~ zMc&YU#aLC<4trm*I_>jl#}}~_(9(H)R_Z0aSLzvaokiNJk8oIG*4S;oJ5sz-AvT?b zxHzBg5PW`Z{wRdh8~p<>4!}snqN^}?Cki)AshgWWO&RxU$*s(dN~a_1Jl>M6dJ>ZM z>CBQN9acxDszV!CG$yF=QOy#Kf39^@aIgIqRrG7ld?s|cf2v&eCY#m{B6 ztp)iKX;K}sk3AJrC+-omd)m#7qBLGXQmpw#B@!!B;gUha4cpbQ;*wI4yQRwx3XE;w z+7|OAM-ok!l1WL$4uhMJ!~<3JW-v3e>QGov=Wrt(&2rcl=O1OKWPm(P)zJ&2^@aJ! z3w3V9vKj%jLFT=GXLSK&)wv5?^BHlann^I7 z6m75^d7??&Ym4wV(N4;si~gCW^~=D2_to>JG}|1NPS9P^?Zi1fB`(ga@Lj`RVan@| za_c3So>#_0X}=|Gw(oQvE+fSamI1qTr3mi$F3HEsZS1+D(t(y7xkv{3_9TGF3#_IA z^lSr9I@bhE7FnBNA3|bd)o7`6zbooOCs}??v~yWZwxfNQYp}@e5cMIUdGb> zk>1H4>*aAL@^f=HSeKI1!$q_lEb@iY^2t5rdI~MnirAXujRD(lUJK*$WBvmv0Qt=; zJr#PUn$JGeKyK#AMaX>A3em0#qy%Z(C>m_b9nE>D>MFQN-de|e$(XMH8c3zp_`^~f zW26x(j3wEDt6qoO&1FN#4-Olj)LUuQar}!mA?KH)%WmmipW!k%e6(|vwDMz+=t9k2 z7j|&+@O$oPMWx8_k}ccW^4Q&BkBc^E%boGjv>Ws}4{Y{C7mHHs-Avxgf!c!0k#9(? zvIXIoaauTR@n}kABe4BD1^wa1kE&}Xf?R^hKAAQ%QUjEQ6#4m5mSm{oBR}H>ulCG0 zFqKHN6s8gwt#7S&h{6)k2vq#neW;Oo4$RFROd?aE=u>&pmiXfwlXzK?7``BUecqY=qc zWc%M+kIok9kN9R7-x+o1=Ye0?p6Xaelhw4>k)tSkHh2 zv3e0H6-}u&PQ+PZany;V}dLj&iI*)$_##qBzSq`i7D#JBIuT= z>qpS9){yA&=W3h6vydM*Qx7{nbe^PK9&VDg%E^D-qgJ`O&q|HW;eEL+am*^Q?1Nwb zY-w6@WFFkAi>E{b?P%5=Q?HLrn16(i9-mp5uQk0ARXky|G~vt-#lRCV_DwdF!*H2_;>)x zA-y$tjny)d?PB|I=B6Rk1M(4eE)Bst!F1V~$dirGi_$BUYJV4dU~g{DQgx)rqMUHv zOYhMw6yS&=QT|tpW<g*I%~*(@F1;*010!@#i9*HChuv8HuzV2_ba7*{gJ5xp?>UI{ZXrwm#3pz632|F;CYZylUBZB zLhJVy=LDIT9rL02dSz<03VMpnyYip+&qx{^ruD~bdryF zmJeklZS6}jsMY8mgk4U%)EDVrzq-j4Q58)#aUc48S1oD}TOyU-eIdOv^{&`TCGd`A zv2go#ywEP}0kqNNS9`fPueR{OF`wCB!g?BAd2wGPSW#=V)ylh?Ep^RjF>?tk*W%RC z#GOzoSfIFid60U2WLAgk4@-(+%czX!IuJj4g;S%?D%JQb$XX)g%n7z6(4n_VznO!Z ztQ>jWE~2+dIM!L}3;$Fq0Av>Rxcc^Rxi>C%n#eFE3N!MOwFRw z{Fd>EACtd6clo@U(^ol=Nx{#)aMM+Mqjt@~=&1g>-aDFh#B?!8M{p?0(366hrN?$4 z#Pgze$R4=9!|ineMA}+m7WU{gwEUhLb<#kj zK)5UR8vXPMVE(&*mNX z2nAk%pQpBqGNH|u_4`+it#X14h)zgLCjk_PP-Go~A+v>|y^vOLH%)~py$- z^;-UhY5jKPQH8|Tc)l}2?cDh+@6{b(>73pw2Jnx}6PU|;=4{e>1oUi$WCVjbKCJ#@yud zF~yFV^M_cv_614B26r?aV-hk4AtP;$BQ0xNlL0C7)fXWurLjAHN;VTt)tMICS71tZ z-{gXlxP&wr?yOoPQ8|!JG?6};uq^R6Rp;w{^3T*uKR#W>9mlT~U((`f3r3XB`^(HX z8*rQwdnem0kb6_&f@yLts3l7l6bHFqoj7jE>6pG^-If2oxY8>rJRanPk=Z)+o<%Et zCfk{tKpt7^Y$qRG)X9D)44(`SEC{hTpw!0Chst*t zq^3K)TuDIy%SGp~;#Ol|W#wXS2bZ(A9{$NGd&t(?dV|x#58iUbkLFN*g*t=pKY;fM zmywinKk?YClbOST2sZsNg#s>k(#u2;%K3viwiL|v5bo|X7wVpKCy);zNA{>%iRBop zH?FZ+^Y?U>V3aBVu;aSfY=Y)A! zbV|s?3vggcS=_+)@O1F!`&??g#d^J{OW{NLtW{!xOlORv*gz?AaCsj!4+Amb$FxR) z!1=^F@j1V%aAlp__zA=rCu`6V8uR*v%yQvHi`h#8r0fARmhHs;hm3TqOs=gAEpy{QyyVBEx!K zWcK9x!k!K>+crRxZ9d+!t;lakvbH`0`Sv1*#-U*(U}WpJv^9 z*jEit1eMts@Lb57H|>1xcc|6762EbCXv{spbL=A@I?{$5G5R0{##(3zS5wH`i&lfe z0J~3AWhmpcl^tC_h$13O5k##YvS5+6+P~|e^YP~T>AO_(H&?t4^=>b$ zPWWaF0Jstp&-@vofIef6p|*Dmw|fE3g`a!ta*F=}D{GMVR*!9Y zek<(z+%S?jwV7((>G&I_?Hi&)6%tFMotH@&{UM>);+#dbJ^DWLr=9sM$9r6mW@sb; z$+g&HTU4rxpn|W+etgjCeMqca$t%_RK@@$aVth=vN9j^I%3iXA_m^b)hxDxW!n2<= zecT{;o0VFblge}tBw=uX-Cgu$jL47qdJ&J&?x}G)7E6!J%SRk?8jR<{yrKw?uVbgY zLf^|+V@8M5Y6DU6H=diUM;xBRMT|$tEMqU42=HSKzj~<4j%3pLP@Rdxd(&}WPd+fH zaFbQESONKU-Lx`^0OJ^=55PIdv|$yAi|`EA>0Mg&I)~x;z4{6*wl9jrmSNM@)3S+U z1Z6?LblRAvmCtYmwK<4r9NT4np9+#q@BR?mInk)Hca^M|P$XE|5dh2BdY4AHcQ)x+ zu=H%#j9XNVQ*C7?9X2dVAh1FkO-2V8odF4K<2_bcvmV0j4ULIkUrwBNb~wd4gR+;N z%PAV>cDz|l(jlz_=eAa%l_t`fI@ML_RlFVtwYba}%X4TI7rj%4Xz^L#%4spS#k zLZK7)=JeOU1vaSyr}Y+RH+YeEQJ741UyQtYh-FjON`kH6MbBYAGnh7VR_NjsH(Z*O z9XN;u_DS*_k1I@tpinrR2Qa|^KmnZ4U{yfGx$)*n@Loe(ZGqx&cat(mWPRd#haSm?zuyG1$crf{tgAd_Sl!|8#%bsAZWP^EXk+TgAdc=I?eJ^XIK?x2^J(}vz$Sf zFvDL8gxtAJe@s%V=d(=;brJORt=`(n_!cZ!kCqwt96y(b0NM`-&O;$yDU8^HT*c@; zmk)X*^8QRTNtaauU1j6x5KoB_ChNlK7p0H_r**Fj*-4VD&t`e;4lDiU~69y>TgAjy&{iM4z$;cQ%q1%hsjT`J#om8@|Uv)WKBU zw6h5eKf0dBu{*{-s(otetw&TUYV^@Ne^L=b8G5I0dSHFD%4Y#K-U|Hrv#C`ND$V^M zJaVOV?b@04Jh|&#@|q}KCST^evb~vXqbu} zokYEXzS??z|6;cDesren%VKM#SwbQyA08#9oNGd;M;7eW0AZp{a47V#$of3W|JhgT#&ZFY6LK0i2noz3Y}VA z*9K@p&p(sfN1Kj=hk|#g))Q>}|6{TZPb|OYnHo1(=Q(PpV7b5Dw~BZHd-9&?8+hPM z(DXvl0!ZDVwYtM_h4~Zuk6Ahy%QCdvI#vE=#$O!jS8`g3No=g=?fN z40m`rRx2m@t+~b}Z(xDZ=h=a-z+gk$YjNT2a^-?34D?s*S8!OT^dSqEs++8LPQQrE z$k<|sLTIo>U6Pdf<;a}tAs1ncwKlhSQozB4@vsUi2vKNSyXla?=i#yIksmQL?i0;i z;??V25Wj4V!67~xq#u$}A(2q3FV$Z1?CFQ(fv6p~uV)YLnUq!?amqQVyD;`MAu$q0 zN#S=k(#$w6~jfGf6Rr$Q?yy( z$GufQy`bP;P1UZKeqwxwK|BG9||lfRW}?DoR#|E55IBU%5ni3O+Y%S;}d*m4|r{)iXDadU5B6m$VChk zhL79L!A3T(o`RW%oe>hXSY;Ym}4Gtev z{NmC@f?8un0%nGKS7a7+khc#llZM0!x)`dGwYkYXqoJ9=Aw~hKANxK+-!LY^*}vJV~UV;WDHrI0xM%wZ(_?HMwF^HxEo} zU-dHqN>p~Df;I#Z9c;tjT}cNDF?=k=vwLmmm0;DunHt)-h$7?vq5wR*#ndIY&i?Xd zC6v7<*aL1=eDLUMzg{jjL%Dhps(Is|n8AK05c9uEQsfjR>K9H(h7-_yTW&e`#R zk8U*KmujplFyKzo6A}RX7=-yJ%1{eyogLIPn6H^>B5rtR{O4r&vk|{t`cxF_PGctz z?zu@ROq0sha)}ECgn!Yv=nWy==mrCW84b6D=A>^JTv4W}3xP17Ng{azcWe>9-<$dv#F(?n z4v6PaUC`{au-rm)QOJU*h&zVN$prr8x{YhXVS!2AEGoMYJ6_AYpg_ zw5q_5uEmke46-;bJ!RGPHxLsI7OW=;@MnVYYD1#)OeYA}zrtDWldyqdW4cv74H-Kp z%E+()`Hi7ah*(I-sl8u?u!s71(NUmvyyDp_huyi*3jTBWR&`m%UUcj0Q7_q4vm7XV z@f=uszK4dP3ZUJq*3A~QY?=jkEUdp4-cAr|dGyL~-5?^HTX$*%AbPz9@r+)+{~J1Q z{15y3%aFciKM*KDvj|n`Oc;q08YdICY`6>C1>g=>+yFg+F^`nZalw}(ezTB04C9QALrSLIw6TwxE~_g{X1ALFZjTEM~#eX+lE!pSPM z?gr}XR*y&U-5ecx(1M?;^WV>+Wj0TEyf>@%5GFlqfUBD?t5T`Qse^w@L`m`y_T2lp zQq;NcWNun~3(#b`Ub)qMZm4fwzhbGddlSXn$%B{u9sd;3mM@z#%5*yGZcCRM-CDZ1 z$3PGIB%iH51Vn0|>xwtjn@rZ9wAhI5J0iZ|m9Xpish3n0Y;#TC#6wcsbf$)BYsw=- ziLhCPDHU3Yp7=F+m5pzq;LNAHEpR(5k_q3STUfaV$Xcm>Ny!Hjn(l*XFy$tl^`J38 z`fL_6!ip8n*lM(oDzbN#<+HV5p_!6dS=G*D@|Eg-b1wduY~EY-QJ(DJO8 z+Itm%T}B`$qBtCTIda#7Brh`3vFqHHO#&ev8g z+n#_MHmrYQL2an(kPk_pr{0z_%4newbmtr4RLrOgae}@-GeCc<9 z4hHAyTGTgQkL9WtAx_bck~7UKT;uOJe=?i1I`(vb3%>n>$wUOpCjxyI*EAb{qT}7t zxU}}?K4rNc=g)!ql($W3qX%Aw3kS98<6|wXMam&VA9qWGL1^a?(Nw8i|Oh6gQsr&&B*erVsS@~mp91FWdNEB#GvD`Ho0X%>k zA8&15^W(gMIQ{9T>_eTBV|3k zfzTHpP0#`ha1yegvQdqQe0zn*WgwB2E7LMu9`y{Ni1BCb(y%t2K-0ko-SkZ*WNS{R zv}G=*lQ?-WmIRjzh6G;z-ae`;j)YtuI=w&kFvzJdpoP*MW#*mNQ-jsvqv@>X{d|wM zY@BU-3pw9HM?hPG;q#IJzO~KG7JkI26vIF@36F?Q1c{aUZBoN-wid9;aqOta&061D zm6qQBiRlp95k#9+(I*MBe3Yp;N87`fAj(0N9bULr?GULeEu;& zvle2mW(97{yKyUbhTmpuDz5}C!)39k!>knBwyYINXMY;FR^-*N>Y73UeV^KbQ-rO~aS= z{I&S@T3REc<0^|*t9U;$K7%&*gPl;}eUl3#8*svDCf=`poBv&)FoHiueEsxV=tEvG zeZ3V&4^pV=4xwK)DKlT|$nF5E7-%KMJ(e{POCOleN$)-9=9$Po3qJ??pvegqG*ye`h zx?ox7?RJl8#P=JI$5t!BGP=J+{kHlhkfoE*&Jr6ls#KNg>k*LqI&&>86Ujcs z*iPGX+KO2sYg3&^Q^-TJtyz99MX>Kl$hR?%Ui=Zw^wB=yP<(IN5p8c~u<*kkO(2qt zm*{H`&99FjnbH#Q=6cIsI)+UVqLbC|5BVwMn!FaRu2c_@U9fL_mhZ~k#>SD6za^DQ z1JaZ{37-GOZ7qUJ*A~|1wOsdhSdH5~oT1@ng~SpS(Vqj1 zCU2jkd07+mYQ}^^G?97ts12_Q%N9wc(h4@B)*ow;K;qf^KK+dz056QtpYTgPM-hp4 zW=_VD?SWdkv}8`qsd7QRO%RhTVx^>S45?P~WD=j0}fw{9WZa<@|hA)4CVIZo{gkh&4(y~F`Y*4?H31(CRhM|A@>ck@T=>@tic4?;v z*2=$bt;Q2_5>!)r#T~6c;IcIU%jyRuvUyobAM{^$HS+trAs;G7AA&$& z;OEeFGO<=7CSuY}LZMXiRU31+N;9X+sHt7_uaYb*#1BQ_0fTRq6%|VX)w#4C0nhq(4Wf=>(w`dejb@`F zU#^mKy*Ay(hXNI#ufNPAX{Bev%TmR&=>_B(BMx9lEt#_*#0dO`-ySoJ$U-Vlfsx1t zp!)j4L!xFznxrZP0-Jy?^jpiLnYqjubRx7OT{8p(QYgO& zS?THBFVd*p@iA7EYDA)~S-k2M%$(Ar9h5n9!M|OHJv0MFAOST0ETO(I2I9W>u{DOH zsS?=|sMY{8VUyVPa5X7zctj|a7D*bCKC4T;)j=t%7{-RAoTqVhkRKo4p*Af;MIxc9G6u(=jM*m;XD+%Sd>J?gpsTt<~8s~p# zp%cAz370pmrF!G?zqgp;H)I64#rUr;eE&BGXlA&lx|`-+ae8-F0uca!~gD$jsn#`iBrF%E3(*Q^1o02_YqP` z$giVce?{eJ{C!aIe+DJhKGgW%SN+c&0slWT&&Ct3`|IqpDN){VXYuZF)&r;V$?rcT*AJ9@mJ-NkpH8_^@e5` ze1fqoe-~C2eQcCK(*VFA#qjgHH`zW>sgHJ52dNs#Hn&8u4iLJ+w$HWL3Ez}zC zEMK|*pen+p*J*S4zS$4WZDX@ouR$L1sv{QtA~_KGbQ5zb&zAWYab}yLNKm(HeW$TX zlG<^RQcSU;QeqO9N}MmlNvUfJG}?=gFF$6gwKnA(ngRtB%QZ3>*?aDD+ph5Msb6 zQkzJvOd}}*-S{y`kL7{iN7(a+2Y!aVqartOLlX+Yz5mhfuwaZSTCU(C`22(%KhF2O z$k%W9SJ6vIKt)=VC#0%0ip(}@R&3AddMdS$XxV`Z5{eZI26{J)L#+VVDoyCSB zNADl!D3^!gSv#Dro<$@(&lGUGoO_;*{hyLZ%K;0;3at^iH?eFFk?h|_;^atgWp8G+ zs*<#9+P^L>mCZFUJ&c!pUG|jwUtT)K7sg8P1C%UrxB(zK9cr>^%{$XqIUsG@O8gx6 zLQ+3ace=wwwo-nNe6gj#SsF7d5vKA}qg)yWhitvU6t!)_rEx7Z#@a@0rDB6kgRl8A zaSi$Q{hp6q_+^&xO11vV?C^`@1YbxA0Kv8oBs1h9#IqFOj0|ihw+=Ai>0jm zkuZ@h2)$BBCY^}h(ipUuy*rvF-R5$#o%MzoOK zRn2kLJBSkDrBF^t5-0k?s83-&OmE=(3Wj3@8)1edTG@pP31@He`ont^rV2N@aZnY! zl?P_fvsy&d=U7N<$~c4V=^(s+2A zs@8Y}bAh=oEzW-BY zgVicnLo5pS09)->TxefTvZh|Xa*1NCN7K_J$~$XIj)!ROQ;vg$vZ8d4v6c;^aIM8# z@%kjT3P;|xyVDgK>7*|SLva*)8*VyTN2HD(g*>EIFv^tH-K8(yhFTacDT^G%ZhPuRIhkM>3?)-=nA3kYX`SNBj;APLN zQAOT{R-WhUDTR`xptt8Zf1)7)eG-5KkI?erOz4KSS)XEvZl0Le7t>5PKV1kKnV;R$ z;hZqM^=O+(D5IM*Z1gzR9ue)goY&DPI7GWdZ@k8drn)vc&X%o4Zft&MScLK=7S=7- z3yCygl`i3X?XpbR0p~n~R;`N%z2j%e0E1l=janl~&PTH~Z1`9!0q>DRcV+SSu@6CN zB9}8|T5>k659H3mnk1Z@oc_T(_4OUM=6JVDtJ1pIu&bU_r`zp#a8$R!Pl6*H+| za@T?hyeRjwTst^F-KlBs>Ajz@AB@6tJN9!`WeNNtvh^!Z9JwS@RcF=xs)FZaFKrzr zitCWF<@e>t4BU%FMzcYG*!T4Ga$W)=@(E}rj$RRI$P1KQJ$Luka$tjIVwL(A(=4w> zZO2hohw+s=kz+o~rl+3pFY$8x&&ru@k5~;5o{V^`F30L6iVb1Euczc(6#vP%{xCL{ z;#ZLz|B4tp3MG0s)v9kqqzx2aU_lRRM3N4J$HXcKF}B;Pek<~RzkVTe*J>a;gC1kh z-aX7J{7qmgwbHX4orxk#hUiV)#vQYo{gN)>IHgXxxl z&Q<^NqU0=xo>9X|*VAdimvkQ1@*4u#RsZ@4Wa?o!uga6WehKBc)KqF{xR3GT1Rly? zTF;5!6gwG0@N|@YjaCAs(oK%9cP(~OfW zR3!WJK(wzhlTo7!#P1-Lrud(xo4o6>=$D#olQzyiMk{|(pf55R)eK9Mi@7VMjb{P6 zx&0>^5+F+;9($2{)*$M%Y>!yLzR0(`xv;I zRUJQ$9R!xuyTT!Mv7)ekx@RbZtEms7g<#y3> zCr*w(76Fao$C7O)-AbG59s4_coh+Y=mKWZB-kk8C@K#R+^! zC9=$j)SvX}McOyckdkV)28LsZg!A9x*Hq(3Y*r^nD0^Rlhg)vjJ#&|tQWfVHa@o}V zE5FA^?G^_I%T6ul$1^b>ifEq85IwF=wXS5W%VR)^ZfD2!@q*Hx@Y;Niq(D*=NJvZ%B3FT>H&7?X8!;dXa~s z4ZY)|`5)A6+D_O`kgQO9UUI6CKSglPgLV?sQj43xW0|~KK<@`*^XcN)=@PBMDcFdI zB%rsXomcJ}x6^*MM!MZ50vGZCGKl+aah6SEyAgl4MPRl(KUcBD3z^{ifakipQH2cG z+-P5s>U-be^)#$unn9-P^_&p#U>?D9zaHTB_}(0c&5AMr7HRS#%lo!O+v!5on_fUk z6%0IU*~4FGbuPPZ@2JuZ45S0V`Mf-sVqFq9IXc52iQV@q6VHA(rb-^uEOYlJmeFmO{3=@+g%MzCMF}q&Xg4eju0IlC@C45 zq6PA>PfQ$vwYG@%Q}9YrY&~$WYksAi&1zMSDA`u?*f?`sfe^xrM;gcH{T+dBGN~f& z{fpk}X_ZD)S-lMS7DU>-S3j8->-Hk`ZTCx-s~=kSve0Cp`(WV#rMqKkWxP%9H-`mj zfj2lrUxipol_w~G#G8j@Rf#yZV1+8LyZf!^qnoo#uUifCiQJ&EV1AbcWS(`+>2~*V(M9Bm?*Yj%* zqROlS{|;lZnwAH;ACReS0E`9Y`|9On_@6x%*Iv8@aiM1Uarr!(@wtUFv2uepXDhTu zLXgGHxIIfm!n4fo%6&|S;^cTro-Q6RGv{2=TnuGpB9K7|&*#s`mpzE|`{qT^upeT2 zKVH2Il97>#0^N2#(f#CkUB0@UQPVZQKvgo8^`!)Mc)E2r2f}-*>e6JJ2NCH{bnG}VJ=KXzi3|lF|P*8%Mhl+Y!0W`GbC6& zs*5u!(rXMXVLFfy+UwCoh9u)j3LJ_587jhvLPe6;xXm{z-sf~C^nIAZO?-fgoT^0o zr^PpT1i`Z)OkGSF=f-#3@St{O?QGxQ}4kY-*NKv{V{o<(J8WetA&kg5S zHxX<|^ox5|1H+=P;`bxg$==PcpW#}0K1H**;K#x5y*)TBbXX#$v|@KjxemR2uQ2#= zFVmwEqn!MebHN8Y)9HK|`$6{9gy#J1U91@uHQ){=HMOL;XqjSWyK0z;!Hyan>gDy`#CRIWPMmvLrJ>>}Q+cr_(mM zfk+$*=u@2aEMNrH@>%C=hxXE*rkl%;I!$H!MjXpa#d#O=Mv|^K3J4xUi64F~`UP0o`QqMkSxU3B z5sW{(w=N4m-jHSkmh5!Jakcp!J8!+*tkYz32cx3_@598zw7vMMbF{`YQq7=I!#F*b z!Tm1nPQXg|Gpj>9_+d8frN&@#;PDgCzqu)uUIJ={;3InkQaPrDi~nG2&RfF!4Zo>* zZKE~7_1tyMJIPcBnBMu8%Nh5FeOipSd*q5aaak@+D65|<y>J)zfu=@TR2 zNOf$4=X^dZd8se_+VP)&n*%079eP}d`Q4|7j7?b%V=<b2HF=_tzfd>C_rL#Q6sXq)c1(pqnU zVUHutgb=NDZID2SH{g&E_8T_gVG0BzT5c8sCH&UO9%qUy8ESSaq!m~fVo&V)tIGiDvz`J?6Qs-@5KG@;GG3*K- zz#!7M4JP4+FO0;Nqn`R!)%mFTcymON%m=?Qg!Ps<6^Z&6%BD)W2V5_k6(s2y!ogJ4 zMsmcZ9jueg*IUd`^BSW770)LfZEzj!Dt~19smter5PWiOAFFff2{QfX1@_abIxmv^ zppX4=4%TOzKEna}#(1uN^2`W+79wath;>8dTJ1us=|BzHs4%@F;<@mC*)e@(f&iWX z>wi&nrGO>i6lOzW3R{E1F^27j!>4Gnb9e`J zf%M!!Bo>R}EYfAgdh#qdG2L@bqw>JcVfcK_Xmz~K$#WrWXZ!LF83+W#9=&MMQL^I` znFwt+R-{~*>2`578SnDt!^(X2lhDc*wg;S?BifH)d=f69#)U*PUdjyZ9?s%%OI^bc zNgY5$dx&VttD`HF2}v9C<#E9pyRXRz&W3z&~QXGNL*|oPqx2{|2rD>G?Pn*(J zneF+mXRDKYt$dozU8nW0zpv!7txzHZ22+yMF{PepDAW?m%W1o(UAW)!fqrkCd-W4@ z1H^2J4--h;@a`NgM_SX-@+A(ZPVV4fy&d#dJ3e&*AL99*mZ^3}4H8O73+?Z;0a zHjEEHBx7Jxe!d-Z(GR;Fki~IU1%2wF`8?%c)v;^Q&bEh{<%3c@A2N_)UQd_W^egwk zqjzn}zhf?H0YE%ag{8%GIS+Xzxc8W*>-CU;p}*lAX1-UH?jCkq$&4ku6Mb#7HE`7G zyo$!~687mXQdEh3b!lrdy-C;c_P)?2**-&?bHN!`xk6Ee&3e_yH1dJ*5v485m8Uij z28Tb#!O;>XaRP$9w~sDzsU_x#tM3W^&@L*}e^ngZClMB7!#nKwUv>9jV#x9R#oiPX>w2WOe;g>}R4& z4EdtiAeKeqZ00PT!XaA`G48BT<~;`1rA%y_O~fOb!rwg})XNN}x466o7@Hq?mSpy) zOX4!W->>1W#cMdOzGVx{S_Ch@uf{LubUUa(9FUCfR`uY4#z#K@FI~{6O6@gT>}-qO z-##RdCBqZN-@bBR>CzE*OOIo`a5UFWM&};5T#k$9X0A~iWH#2L zBkk5XP46mDKFD%asgZzEK_Cmk}9BoyHaoDdaGfW=hEbl|}(Qc6*Lo~}pB7y0L%MtpUvp-2U`-)SX zKv#GI9A}^gyxKhinJ=A)#`VeXhYoKsKF}&xl8$e4N6#i%_&cThjROQ*WtjPBMx3v{!K7?0kNbJizZa_Ut-}-PU@S=nxWR9W zl{>a**?lTy{-y!yPNZOYRR4pXDrv)7*z^`5UEwDN{EImB1Uu7$tMfp z*JMVebIahgF_5N!Sum(Nhe_llu)s25ENA9zdJgqJ1 zdO&ZsYxz=sbKc7YT_U(`hEhxi>^QYwS&svGZ(tDt`sv2>>eT{8=Bv3ao*bLPI>(Pl zM>i($Z!vAXOa_;Vt*_tBcohEz=oLHZEh<(dfN4-(+pBavFj1l%51RC@?p<_9wt1T0 ziawxPSZQ^h0%tV3+jpTmhgc?0av$I5FSeoJ_;fAJj51GrnHwUw#a(@|YadwGaa|>Q ztNS&??QyPFdj~vraS2sfBgA4WNp38J${MU&Hyb{E!_!YJ*;FUDbY@>dCMP-ClPV zf-JUA^)wH$^OnRXUX63>+FHOsVr;Y@07&>97q~|;rb$Rt9AQDL$q z9b1&imSrqS_Q)Vcr5Q{1Wn>*oW)Q|Uga+ksejnd+&h@*l@5#U4>-t>J-_P@W?(2T; z=ly;?x8R_lb!|H>C$v0j>tQY9=(w)3Z8l46L@2mzRpd?%N(%Pm@~LaNtL|X()I!K^ z09o9>-bnx5-Hoc2juwUNQ1~G689B5WKNORwBVxx&!cJB*ZI1igDXe9&l!8lQI^2+nq-UB8 zvDY|`yo7ZeAHY3`6c2UW)tokLI zY~i%YvnM*Q=cNI{i(D_?B_+;unNivpe0}?6+iXoPZb1oBF`$S}roCu~s$&Pp_!$7B<5^Yl@evN1ev6rvYA?+>=9wBu|lUv|FN7$vw(|D+R^KAbbp zM5%Z0N*U6%_+-h4)(3R){`#=6QQX7jL#yVX4xnUq9Fw@PhlcFz*Z&i|ld-XwC}`3r zCorU2Nc`zPVdkE{;IKlGMPQ5`kj zrCs0cqP+;U&g7A=H5(JVeXVT?KJBf*VkXN?1h6x}vuNjH0x6q7+12aK#Su7VB`4@1 z!}c@ikf3uq8zPyBni-NwZ;`mIRtPSzcB7qFiMzMwg#ir;lVSIAE7C)EYwrxj&^NVf zzuw$x#AnY+*Yx8fUXw=OlIuxQfMX*)9DVFWzIv(L_PLRi_m=RwDd*5QeTGh92zz41 ziA%W-!4c@79Pd{S8eX3qFc-S-@MQ#%X$W)DI!AzJh{+ z1rAge(!rsWQzXgYQ0DiYd|1yZ=MU9`xNFy-r>;x!)jq278Z6FbQHE9M9@ba{Xw%Da z+mGaOPHL*E+5^f9tvgA1)##Q!^U#wN`Od|QrH9Apt&KMfgy#im8Obv6&!tmbzV@md zI<~>hhQ~h52|GyjQp}j|f(5TD6td+}3E>kcs~Ry4?{rZz{apm?y~>KR;^cdf3OnTH zzIb%Ga&M#kae##iLTtK5Y;mw8k8qni4d`-!pD3_M>CZ$oqWn&etZ9`a2_GIX7?=D1 ze6hz=qy%KUE5so2`4=wzll)yOlVeICPphuB^d=r&(oVed<&W{n$;mY|4UK`d4+2KB z0v-6sF>)m_-na^L>K}%}pkd*?04IM8!Xa4&GS#iB^cms>jtXZpk~zKX9~{y82`%+% zu4u(}EHmd`#4MX#wKrSSlKDBG}J++k;WQd-ET7M zZ`}4CYI>PL674yZ+Y5^MW~Ek#91oDhuxV*bpIBDc^^$^YN(ynAzV$=Dgy}^5JZv{G zqepILN0ZEbBMLwhBKxBj=sFuc#sz|I#T6TH5JE>v z%_vfLs+sr3baO2798KGIeO5bYlEawXBs577h&eqLS)*RLl0!cJB-%@PuW>t42i3jO z-`IJSkE>68uvI!;iY6j#nx!s_t@_&K^(O>w@`68v+;DG>C}JReMtl~{SoiN z&*gP!`*lxmsYPLx{*M_YHLV+;1LsHGo0mpLp6C2bBv-5G|8T>y;$Mk_l`nl}jRiOK z-ib`huN?4!jU(&L1fOacY>%g=NCRHi`sDeCN5aK$i1yL!Jc7s?F)()eyiSLca&t8* z@Pgzrg{WYo-C`*~re%k@7?l~D4T{-I+-usV4w-Ol6a;O^kkGUsYJqbmY?VM6w6;t+ zKSXdP^oFgkrBM=@T;qgdr8oUO0%bn1`K4QnoeXS=7h58B99n$jdE(7Gq+$i#Ds(%JH*|}sjOj^$=ni8ktthaJu}v@bodq0R5SmvXP=XmN<5Km)SXnXZ z3{SBWuG}T~8=&^5g!JF#lMgf1AfAq79wA3n4BR~PAA7Mu!u-&0Jg@;MC-&N-hDud~ zKnv+L#W6?#X|wGoVL}`|fjXNbCSK|mCl+F%-)i#4QBd@P9!csATcxOxmueChk7U<{ zlL~^wjJlcObVDf?NE|5r6$h6&?orJN+Ahq})Tq07+FJM-x%z&sM`p<&_^Wqe>sRie z28w_P!m$|(S6)Dj8FRZ&G~17eTdr|uEq{iGR{F0F*k!yJ>4($xIZb1l=hSjRi5|B% zIcv;&s80uW16rn`b7!TGGblSo1zu$9%9~Wf0GRqVOU>Q3I@W9&+W|C&22+ebxETz6 zBn0N%BS#?dqx7HBNKgBML{! zbFDkPCO@EsUd9>t?(8~)6}rA{#Zz=yQ#6OjEHOkp_F_QU!BA5;&-Q(!yByIAi&W~T z6dw^T9V|`Fy!M9w*6`DAJ#5GYpui(MyRm}(A{CP$qoh%Mkp8tGb zT2|I{g|LeDlb^;dk$s;#p@>d7MPi$tI z@4XkD8*o3LXiub7TePe+kW~0_;52o~Y>Xabkawo9?T8s*^JUqw1nVzsb=;$d#`p79 z8Sr5NX!tM6>AyzkCFbKp4hrg}-my0W0Q@JmD{HJJw2pZ|}7q|M0@%KQU2Gm%9y{6fZGepeob+4m^u1eH|;~Nu|gk=Qq zoLOjOF9zSg^Taa04f|e~8T-x!m#OKSXanD~pagkFdXH6WJKSsC2bq8Q0o<-x>Boz= zr9I~{-c2ez0~KTs>8=lNQa*JEFsj*PjWYy_3JcrKW$&kzJL_7E|5RsVCG|2vr3B>+10|IeXcMb7_nW486<+%qn1aWOXmy-yFB MpR+bDKkE|yf0__(=Kufz literal 0 HcmV?d00001 diff --git a/assets/images/license-overview-d5fde313e9b49397653607393dddb419.png b/assets/images/license-overview-d5fde313e9b49397653607393dddb419.png new file mode 100644 index 0000000000000000000000000000000000000000..def533cc2ede25e82625587627dfa52522d614dd GIT binary patch literal 215308 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

Client Specification 1.0

This document attempts to guide developers in implementing an Unleash Client SDK.

System Overview

Unleash is composed of three parts:

  • Unleash API - The service holding all feature toggles and their configurations. Configurations declare which activation strategies to use and which parameters they should get.
  • Unleash UI - The dashboard used to manage feature toggles, define new strategies, look at metrics, etc.
  • Unleash SDK - Used by clients to check if a feature is enabled or disabled. The SDK also collects metrics and sends them to the Unleash API. Activation Strategies are also implemented in the SDK. Unleash currently provides official SDKs for Java and Node.js

system_overview

To be super fast, the client SDK caches all feature toggles and their current configuration in memory. The activation strategies are also implemented in the SDK. This makes it really fast to check if a toggle is on or off because it is just a simple function operating on local state, without the need to poll data from the database.

The Basics

All client implementations should strive to have a consistent and straightforward user API. It should be a simple method, called isEnabled, to check if a feature toggle is enabled or not. The method should return a boolean value, true or false.

unleash.isEnabled('myAwesomeToggle');

The basic isEnabled method should also accept a default value. This should be used if the client does not know anything about a particular toggle. If the user does not specify a default value, false should be returned for unknown feature toggles.

Calling unleash with default value:

boolean value = unleash.isEnabled("unknownFeatureToggle", false);
//value==false because default value was used.

Implementation of isEnabled

A feature toggle is defined as:

{
"name": "Feature.B",
"description": "lorem ipsum",
"enabled": true,
"strategies": [
{
"name": "ActiveForUserWithId",
"parameters": {
"userIdList": "123,221,998"
}
},
{
"name": "GradualRolloutRandom",
"parameters": {
"percentage": "10"
}
}
],
"strategy": "ActiveForUserWithId",
"parameters": {
"userIdList": "123,221,998"
}
}

A simple demo of the isEnabled function in JavaScript style (most of the implementation will likely be more functional):

function isEnabled(name, unleashContext = {}, defaultValue = false) {
const toggle = toggleRepository.get(name);
let enabled = false;

if (!toggle) {
return defaultValue;
} else if (!toggle.isEnabled) {
return false;
} else {
for (let i = 0; i < toggle.strategies.length; i++) {
let strategyDef = toggle.strategies[i];
let strategyImpl = strategyImplRepository.get(strategyDef.name);
if (strategyImpl.isEnabled(toggle.parameters, unleashContext)) {
return true;
}
}
return false;
}
}

Activation Strategies

Activation strategies are defined and configured in the unleash-service. It is up to the client to provide the actual implementation of each activation strategy.

Unleash also ships with a few built-in strategies, and expects client SDK's to implement these. Read more about these activation strategies. For the built-in strategies to work as expected the client should also allow the user to define an unleash-context. The context should be possible to pass in as part of the isEnabled call.

Extension points

Client implementation should also provide a defined interface to make it easier for the user to implement their own activation strategies, and register those in the Unleash client.

Fetching feature toggles (polling)

The client implementation should fetch toggles in the background as regular polling. In a thread-based environment, such as Java, this needs to be done in a separate thread. The default poll interval should be 15 seconds, and it should also be configurable.

Client registration

On start-up, the clients should register with the Unleash server. The registration request must include the required fields specified in the API documentation.

Metrics

Clients are expected to send metrics back to Unleash API at regular intervals. The metrics are a list of used toggles and how many times they evaluated to yes or no in at the time of requesting the metrics. Read more about how to send metrics in the Metrics API documentation.

Backup Feature Toggles

The SDK also persists the latest known state to a local file on the instance where the client is running. It will store a local copy every time the client receives changes from the API. Having a local backup of the latest known state minimises the consequences of clients not being able to talk to the Unleash API on startup. This is necessary due to network unreliability.

- - + + \ No newline at end of file diff --git a/contributing.html b/contributing.html index 2d80a1dd53..2155afc5f5 100644 --- a/contributing.html +++ b/contributing.html @@ -20,15 +20,15 @@ - - + + - - + + \ No newline at end of file diff --git a/contributing/ADRs.html b/contributing/ADRs.html index 1369872934..5879db384f 100644 --- a/contributing/ADRs.html +++ b/contributing/ADRs.html @@ -20,8 +20,8 @@ - - + + @@ -29,7 +29,7 @@

ADR Overview

Introduction

Architectural decision records are a record of design decisions we have made in the past because we belived they would help our code quality over time. Any ADR can be challenged, but two conditions must be met to change an ADR:

  1. The proposed solution must provide a tangible benefit in terms of code quality.
  2. The benefits of the proposed solution must outweigh the effort of retroactively changing the entire codebase. One such example is the decision to re-write Unleash to TypeScript.

Overarching ADRs

These ADRs describe decisions that concern the entire codebase. They apply to back-end code, front-end code, and code that doesn't neatly fit into either of those categories.

Back-end ADRs

We are in the process of defining ADRs for the back end. At the time of writing we have created the following ADRS:

Front-end ADRs

We have created a set of ADRs to help guide the development of the front end:

- - + + \ No newline at end of file diff --git a/contributing/ADRs/back-end/POST-PUT-api-payload.html b/contributing/ADRs/back-end/POST-PUT-api-payload.html index 831b6f0fbb..966abbd642 100644 --- a/contributing/ADRs/back-end/POST-PUT-api-payload.html +++ b/contributing/ADRs/back-end/POST-PUT-api-payload.html @@ -20,15 +20,15 @@ - - + +

ADR: POST/PUT API payload

Background

Whenever we receive a payload in our backend for POST or PUT requests we need to take into account backwards compatibility. When we add a new field to an existing API payload, clients using the previous version of the payload will not know about that new field. This means that we need to make sure that the new field is optional. If we make the field required, clients using the previous version of the payload will override the value of the new field with an empty value or null.

Example: adding new setting field to project settings

Project settings on Unleash 5.3:

curl --location --request PUT 'http://localhost:4242/api/admin/projects/default' \
--header 'Authorization: INSERT_API_KEY' \
--header 'Content-Type: application/json' \
--data-raw '{
"id": "default",
"name": "Default",
"description": "Default project",
"defaultStickiness": "default",
"mode": "open"
}'

New version of project settings (Unleash 5.6):

curl --location --request PUT 'http://localhost:4242/api/admin/projects/default' \
--header 'Authorization: INSERT_API_KEY' \
--header 'Content-Type: application/json' \
--data-raw '{
"id": "default",
"name": "Default",
"description": "Default project",
"defaultStickiness": "default",
"featureLimit": 2
}'

Pay attention to the new field feature limit. If a customer updates Unleash to 5.6 but their integration still does not send that field, it may result in the unwanted behavior of setting that field to empty in the database, in case the server assumes that not sending the field means setting it to empty / null.

This bug can easily be an oversight but can be prevented by following some rules when designing the API payload.

Decision

When receiving a body from a request we need to take into account 3 possible cases:

  1. The field has a value
  2. The field is undefined or not part of the payload
  3. The field is null
  • If the field has a value, we need to update or set that value in the DB.
  • If the field is undefined or not part of the payload, we need to leave the value in the DB as it is.
  • If the field is null, we need to remove the value from the DB (set it as null on the DB).
- - + + \ No newline at end of file diff --git a/contributing/ADRs/back-end/breaking-db-changes.html b/contributing/ADRs/back-end/breaking-db-changes.html index 79235f474f..ebe3be84e9 100644 --- a/contributing/ADRs/back-end/breaking-db-changes.html +++ b/contributing/ADRs/back-end/breaking-db-changes.html @@ -20,8 +20,8 @@ - - + + @@ -30,7 +30,7 @@

ADR: Breaking DB changes

Background

During the evolution of a feature different clients may use different version of code e.g. behind a feature flag. If the code relies on breaking DB changes (column delete, table rename, deleting DB entries etc.) it may lead to errors.

The very same problem occurs when you apply a breaking migration just before the new version of the application starts e.g. during a zero-downtime deployment (whatever strategy you use). The code is still running against the old schema as the migration takes a few seconds to apply.

Decision

First please make sure to avoid breaking DB changes in the first place if possible.

If breaking change is inevitable please use the "expand/contract" pattern.

In the "expand phase":

  • maintain old and new DB schema in parallel
  • maintain code that works with old and new DB schema
  • keep it for 2 minor releases to give all clients a chance to upgrade the code
  • with a fallback of 2 version we can also downgrade in this range without running down migrations

In the "contract phase":

  • remove the old schema when you know that no client is using the old version

Action for a code reviewer:

  • when you spot a migration with ALTER table DROP COLUMN or ALTER table RENAME TO please raise a flag if the "expand phase" was missed
- - + + \ No newline at end of file diff --git a/contributing/ADRs/back-end/naming.html b/contributing/ADRs/back-end/naming.html index 26d2fb3a41..6bda721c5a 100644 --- a/contributing/ADRs/back-end/naming.html +++ b/contributing/ADRs/back-end/naming.html @@ -20,15 +20,15 @@ - - + +

ADR: Naming

Background

In the codebase, we have found a need to have a common way of naming things in order to ensure consistency. It's important that files are named after the contents of the file to ensure that it's easy to search for files. You should be able to find the file you need in the command line without the help of advanced IDEs. This can easily be solved by proper naming. It's also crucial that the naming is consistent across the project, if we are using different naming conventions in different places, it will be hard to navigate the codebase.

Decision

We have decided to use a naming convention where the files are named after the main class that it contains. Example:

feature-toggle-service.ts

class FeatureToggleService {
...
}

The reason for this decision is to remove mental clutter and free up capacity to easily navigate the codebase. Knowing that a file is named after the class that it contains allows you to quickly scan a file without watching for a context where the class is used in order to understand what it is.

- - + + \ No newline at end of file diff --git a/contributing/ADRs/back-end/preferred-export.html b/contributing/ADRs/back-end/preferred-export.html index 4015200ad4..bd7c18be4b 100644 --- a/contributing/ADRs/back-end/preferred-export.html +++ b/contributing/ADRs/back-end/preferred-export.html @@ -20,15 +20,15 @@ - - + +

ADR: Preferred export

Background

In the codebase, we have discovered that default exports create multiple problems. One is that you can rename the component when importing it, which can cause confusion. Another is that it is harder to find the component when you are looking for it, as you have to look for the file name instead of the component name (solved by ADR for naming, but still relevant).

Decision

We have decided to use named exports. This will allow us to eliminate the possiblity of exporting a component and renaming it in another file. It also allows us easy access to advanced refactors across the project, because renaming in one place will propagate to all the other places where that import is referenced. This resolves the issues described in the background without any significant downsides.

- - + + \ No newline at end of file diff --git a/contributing/ADRs/back-end/specificity-db-columns.html b/contributing/ADRs/back-end/specificity-db-columns.html index 706d72ddba..c72907a1c1 100644 --- a/contributing/ADRs/back-end/specificity-db-columns.html +++ b/contributing/ADRs/back-end/specificity-db-columns.html @@ -20,15 +20,15 @@ - - + +

ADR: Specificity in database column references

Background

We recently experienced an issue where a database migration that introduced a new column resulted in ambiguity errors in our application, which highlighted the need for clearer SQL query standards. Currently, our queries often reference columns without specifying their parent tables, leading to potential ambiguity in complex queries that join multiple tables. This issue becomes more pronounced during database schema changes and migrations, where ambiguity in column references can lead to hard-to-anticipate runtime errors.

Decision

To mitigate these risks, we will adopt a standard of explicitly specifying the full table name or alias for each column in our SQL queries. This standard is not just about improving readability, but is crucial for avoiding ambiguity in queries, especially when performing joins between tables. The decision to use the full table name or an alias will be left to the discretion of the developer, with a focus on maximizing clarity and maintainability.

Example: Preferred vs. discouraged syntax

To clarify the standards set out in the ADR, here's a quick comparison of the preferred syntax against the discouraged syntax, using hypothetical tables and columns:

1. Preferred syntax: Explicit table naming

const rows = await this.db
.select(
'u.id',
'u.name',
'u.email',
'o.description',
)
.from('users as u')
.join('orders as o', 'o.user_id', 'u.id')
.where('o.status', 'active')
.orderBy('o.created_at', 'desc')

Why preferred: Clearly indicates that id, name, and email are columns from the users table (aliased as u), and that description, status and created_at are from the orders table (aliased as o). This prevents ambiguity, especially useful in JOINs.

Note: Aliases (u for users, o for orders) are used here for brevity and readability, but they are optional. The key aspect is specifying the table for each column.

2. Discouraged syntax: Implicit table naming

const rows = await this.db
.select(
'id',
'name',
'email',
'description',
)
.from('users')
.join('orders', 'orders.user_id', 'users.id')
.where('status', 'active')
.orderBy('created_at', 'desc')

Why discouraged: Without specifying the table for each column, it becomes unclear which table each column belongs to. This ambiguity, especially in tables with identically named columns, can lead to runtime errors that are difficult to anticipate, particularly in queries involving joins.

Advantages

The primary benefits of this approach are:

  1. Clarity and reduced ambiguity: By explicitly specifying table names for each column, we eliminate ambiguity about which table a column belongs to. This is particularly beneficial in complex queries involving multiple tables and joins. This also reduces the risk of ambiguity errors during database migrations or schema changes.

  2. Ease of maintenance and adaptability to changes: During database migrations or schema changes, specifically referenced columns make it easier to identify and update relevant queries. This reduces the risk of overlooking changes that might affect query behavior.

  3. Improved readability in complex queries: In queries involving multiple tables and joins, explicitly specifying table names makes the query more readable and understandable, facilitating easier debugging and review, while ensuring consistency across the codebase.

Concerns

The adoption of this practice comes with minimal concerns:

  1. Slight increase in verbosity: The queries will be slightly longer due to the addition of table names in columns. However, the benefits in clarity and maintainability far outweigh this minor increase in verbosity.

  2. Initial adaptation curve: There might be a brief period of adjustment as developers adapt to this new standard. However, given its straightforward nature, this learning curve is expected to be minimal.

Conclusion

The adoption of explicit table name specification in our SQL queries is a strategic decision aimed at improving the clarity, maintainability, and reliability of our database interactions. This practice will ensure that our queries remain robust and clear, especially in the context of evolving database schemas and complex query scenarios. We will gradually implement this standard in our existing codebase by following the Girl Scout Rule. This change aligns with our commitment to writing clean, understandable, and maintainable code, thereby enhancing the overall quality of our software development processes.

- - + + \ No newline at end of file diff --git a/contributing/ADRs/front-end/component-naming.html b/contributing/ADRs/front-end/component-naming.html index 9293c5e6be..fd3fffce20 100644 --- a/contributing/ADRs/front-end/component-naming.html +++ b/contributing/ADRs/front-end/component-naming.html @@ -20,15 +20,15 @@ - - + +

ADR: Component naming and file naming

Background

In the codebase, we have found a need to have a common way of naming components so that components can be (a) easily searched for, (b) easily identified as react components and (c) be descriptive in what they do in the codebase.

Decision

We have decided to use a naming convention for components that uppercases the first letter of the component. This also extends to the filename of the component. The two should always be the same:

// Do:
// MyComponent.ts

const MyComponent = () => {};

// Don't:
// someRandomName.ts

const MyComponent = () => {};

The reason for this decision is to remove mental clutter and free up capacity to easily navigate the codebase. Knowing that a component name has the same name as the filename will remove any doubts about the file contents quickly and in the same way follow the React standard of uppercase component names.

Deviations

In some instances, for simplicity we might want to create internal components or child components for a larger component. If these child components are small enough in size and it makes sense to keep them in the same file as the parent (AND they are used in no other external components) it's fine to keep in the same file as the parent component.

- - + + \ No newline at end of file diff --git a/contributing/ADRs/front-end/handling-tables.html b/contributing/ADRs/front-end/handling-tables.html index 0af0393392..b4850585c4 100644 --- a/contributing/ADRs/front-end/handling-tables.html +++ b/contributing/ADRs/front-end/handling-tables.html @@ -20,8 +20,8 @@ - - + + @@ -34,7 +34,7 @@ between the definitions of the client side and server side powered tables.

Decision

We have decided to favor consistency over one-off simplicity. Using react-table comes at a cost but allows to change between client and server side data handling with lesser effort. It allows to revert decisions to client side and makes the migration to server side data handling easier.

- - + + \ No newline at end of file diff --git a/contributing/ADRs/front-end/interface-naming.html b/contributing/ADRs/front-end/interface-naming.html index 11f45185fc..a9a49b3a71 100644 --- a/contributing/ADRs/front-end/interface-naming.html +++ b/contributing/ADRs/front-end/interface-naming.html @@ -20,15 +20,15 @@ - - + +

ADR: Interface naming

Background

In the codebase, we have found a need to have a common way of naming interfaces in order to ensure consistency.

Decision

We have decided to use a naming convention of appending the letter I in front of interfaces to signify that we are in fact using an interface. For props, we use IComponentNameProps.

// Do:
interface IMyInterface {}
interface IMyComponentNameProps {}

// Don't:
interface MyInterface {}
interface MyComponentName {}

The reason for this decision is to remove mental clutter and free up capacity to easily navigate the codebase. Knowing that an interface is prefixed with I allows you to quickly scan a file without watching for a context where the interface is used in order to understand what it is.

- - + + \ No newline at end of file diff --git a/contributing/ADRs/front-end/preferred-component-props-usage.html b/contributing/ADRs/front-end/preferred-component-props-usage.html index 1034b563df..9e870509c0 100644 --- a/contributing/ADRs/front-end/preferred-component-props-usage.html +++ b/contributing/ADRs/front-end/preferred-component-props-usage.html @@ -20,15 +20,15 @@ - - + +

ADR: Preferred component props usage

Background

In the codebase, we have found a need to standardise how to use props, in order to easily be able to figure out what a component is doing and what properties it is given without having to look up the interface.

Decision

We have decided to use props destructuring inline in components in order to quickly display what properties a component is using.

// Do:
const MyComponent = ({ name, age, occupation }: IComponentProps) => {
return (
<div>
<p>{age}</p>
<p>{name}</p>
<p>{occupation}</p>
</>
)
};

// Don't:
function MyComponent(props) {
return (
<div>
<p>{props.age}</p>
<p>{props.name}</p>
<p>{props.occupation}</p>
</>
)
}

The reason for this decision is to remove mental clutter and free up capacity to easily navigate the codebase. In addition, when components grow, the ability to look at the signature and instantly know what dependencies this component uses gives you an advantage when scanning the codebase.

- - + + \ No newline at end of file diff --git a/contributing/ADRs/front-end/preferred-data-fetching-method.html b/contributing/ADRs/front-end/preferred-data-fetching-method.html index c712ba5735..49be2a32c0 100644 --- a/contributing/ADRs/front-end/preferred-data-fetching-method.html +++ b/contributing/ADRs/front-end/preferred-data-fetching-method.html @@ -20,15 +20,15 @@ - - + +

ADR: Preferred data fetching method

Background

We have found a need to standardise how we fetch data from APIs, in order to reduce complexity and simplify the data fetching process.

Decision

We have decided to remove redux from our application and fetch all of our data via a third party library called useSWR (SWR stands for stale-while-revalidate and is a common cache strategy).

// Do:
// useSegments.ts

import useSWR from 'swr';
import { useCallback } from 'react';
import { formatApiPath } from 'utils/formatPath';
import handleErrorResponses from '../httpErrorResponseHandler';
import { ISegment } from 'interfaces/segment';
import useUiConfig from 'hooks/api/getters/useUiConfig/useUiConfig';
import { IFlags } from 'interfaces/uiConfig';

export interface UseSegmentsOutput {
segments?: ISegment[];
refetchSegments: () => void;
loading: boolean;
error?: Error;
}

export const useSegments = (strategyId?: string): UseSegmentsOutput => {
const { uiConfig } = useUiConfig();

const { data, error, mutate } = useSWR(
[strategyId, uiConfig.flags],
fetchSegments
);

const refetchSegments = useCallback(() => {
mutate().catch(console.warn);
}, [mutate]);

return {
segments: data,
refetchSegments,
loading: !error && !data,
error,
};
};

export const fetchSegments = async (
strategyId?: string,
flags?: IFlags
): Promise<ISegment[]> => {
if (!flags?.SE) {
return [];
}

return fetch(formatSegmentsPath(strategyId))
.then(handleErrorResponses('Segments'))
.then(res => res.json())
.then(res => res.segments);
};

const formatSegmentsPath = (strategyId?: string): string => {
return strategyId
? formatApiPath(`api/admin/segments/strategies/${strategyId}`)
: formatApiPath('api/admin/segments');
};

// Don't:
const MyComponent = () => {
useEffect(() => {
const getData = () => {
fetch(API_URL)
.then(res => res.json())
.then(setData);
};
getData();
}, []);
};
- - + + \ No newline at end of file diff --git a/contributing/ADRs/front-end/preferred-data-mutation-method.html b/contributing/ADRs/front-end/preferred-data-mutation-method.html index 1dfbce59c9..09a25156c5 100644 --- a/contributing/ADRs/front-end/preferred-data-mutation-method.html +++ b/contributing/ADRs/front-end/preferred-data-mutation-method.html @@ -20,8 +20,8 @@ - - + + @@ -29,7 +29,7 @@

ADR: Preferred data mutation method

Background

Because our product is open-core, we have complexities and needs for our SaaS platform that are not compatible with the needs of our open-source product. We have found a need to standardise how we fetch data from APIs, in order to reduce complexity and simplify the data fetching process.

Decision

We have decided to standardise data-fetching and error handling by implementing a top level useAPI hook that will take care of formatting the request in the correct way adding the basePath if unleash is hosted on a subpath, wrap with error handlers and return the data in a consistent way.

Example:

import { ITagPayload } from 'interfaces/tags';
import useAPI from '../useApi/useApi';

export const useTagTypesApi = () => {
const { makeRequest, createRequest, errors, loading } = useAPI({
propagateErrors: true,
});

const createTag = async (payload: ITagPayload) => {
const path = `api/admin/tag-types`;
const req = createRequest(path, {
method: 'POST',
body: JSON.stringify(payload),
});

try {
const res = await makeRequest(req.caller, req.id);

return res;
} catch (e) {
throw e;
}
};

const validateTagName = async (name: string) => {
const path = `api/admin/tag-types/validate`;
const req = createRequest(path, {
method: 'POST',
body: JSON.stringify({ name }),
});
try {
const res = await makeRequest(req.caller, req.id);
return res;
} catch (e) {
throw e;
}
};
const updateTagType = async (tagName: string, payload: ITagPayload) => {
const path = `api/admin/tag-types/${tagName}`;
const req = createRequest(path, {
method: 'PUT',
body: JSON.stringify(payload),
});

try {
const res = await makeRequest(req.caller, req.id);
return res;
} catch (e) {
throw e;
}
};

const deleteTagType = async (tagName: string) => {
const path = `api/admin/tag-types/${tagName}`;
const req = createRequest(path, { method: 'DELETE' });

try {
const res = await makeRequest(req.caller, req.id);
return res;
} catch (e) {
throw e;
}
};

return {
createTag,
validateTagName,
updateTagType,
deleteTagType,
errors,
loading,
};
};
- - + + \ No newline at end of file diff --git a/contributing/ADRs/front-end/preferred-export.html b/contributing/ADRs/front-end/preferred-export.html index b9300f94f9..a88757e05f 100644 --- a/contributing/ADRs/front-end/preferred-export.html +++ b/contributing/ADRs/front-end/preferred-export.html @@ -20,15 +20,15 @@ - - + +

ADR: Preferred export

Background

We have seen a need to standardize how to export from files in the project, in order to achieve consistency and avoid situations where we can have a component default exported as one name and renamed as something else in a different file. For example:

// Problem example
// File A

const MyComponent = () => {

}

export default MyComponent;

// File B
import NewName from '../components/MyComponent/MyComponent.tsx';

The above can cause massive confusion and make it hard to navigate the codebase.

Decision

We have decided to standardise exports on named exports. This will allow us to eliminate the possiblity of exporting a component and renaming it in another file.

// Do:
export const MyComponent = () => {};

// Don't:
const MyComponent = () => {};

export default MyComponent;

The reason for this decision is to remove mental clutter and free up capacity to easily navigate the codebase. If you can always deduce that the component is named as it is defined, then finding that component becomes a lot easier. This will ensure that we remove unnecessary hurdles to understand and work within the codebase.

- - + + \ No newline at end of file diff --git a/contributing/ADRs/front-end/preferred-folder-structure.html b/contributing/ADRs/front-end/preferred-folder-structure.html index a777a291ab..ca113f7020 100644 --- a/contributing/ADRs/front-end/preferred-folder-structure.html +++ b/contributing/ADRs/front-end/preferred-folder-structure.html @@ -20,15 +20,15 @@ - - + +

ADR: Preferred folder structure

Background

Folder structure is important in how easy it is to navigate and reason about the codebase. It's important to have a clear structure that is easy to understand and follow, while grouping related files together in such a way that is easy to find and remove.

Decision

We have decided to create tree-like folder structure that mimics as closely as possible the relationship of the React components in the project. This has a number of benefits:

  • If you are looking for a component, you can easily find it by looking at the folder structure.
  • If you need to delete a component, you can be sure that all of the files connected to that component will be deleted if you delete the folder. This is supremely important, because it allows us to get rid of dead code easily and without having to worry about the consequences of deleting a file and worrying about whether it's used somewhere else.

Folder structure example:

ProfilePage
ProfilePage.tsx
ProfilePage.styles.ts
ProfileSettings
ProfileSettings.tsx
ProfileSettings.styles.ts
ProfilePicture
ProfilePicture.tsx
ProfilePicture.styles.ts

Now you can clearly see that if you need to delete the ProfilePage component, you can simply delete the ProfilePage folder and all of the files connected to that component will be deleted.

If you experience that you need to create a component that is used in multiple places, the component should be moved to the closest possible ancestor. If this is not possible, the component should be moved to the common folder.

- - + + \ No newline at end of file diff --git a/contributing/ADRs/front-end/preferred-form-architecture.html b/contributing/ADRs/front-end/preferred-form-architecture.html index c4a26cdc96..d76bc5b825 100644 --- a/contributing/ADRs/front-end/preferred-form-architecture.html +++ b/contributing/ADRs/front-end/preferred-form-architecture.html @@ -20,15 +20,15 @@ - - + +

ADR: Preferred form architecture

Background

Forms can be tricky. In software, we often want to write DRY components, repeating as little as possible. Yet we also want a clear separation of concerns. Forms represent a challenge in this way because you have to choose which principle is the most important. You can't both have it DRY and completely separated.

Decision

We have decided to architecture our forms in the following way:

  • Create a hook that contains all the logic for the form. This hook will return a form object that contains all the form state and functions to update the state.
  • Create a reusable form component that does not contain any logic
  • Create separate Create and Edit components that use the form component and the form hook to create the form and implements it's own logic for submitting the form.

In this way, we keep as much of the form as possible DRY, but we avoid passing state internally in the form so the form doesn't need to know whether it is in create or edit mode. This allows us to keep one thing in mind when working, and not have to worry about dual states of the component.

- - + + \ No newline at end of file diff --git a/contributing/ADRs/front-end/preferred-function-type.html b/contributing/ADRs/front-end/preferred-function-type.html index ce0e4ab037..c463093460 100644 --- a/contributing/ADRs/front-end/preferred-function-type.html +++ b/contributing/ADRs/front-end/preferred-function-type.html @@ -20,15 +20,15 @@ - - + +

ADR: Preferred function type

Background

In the codebase, we have found a need to standardise function types in order to keep the codebase recognizible across different sections, and to encourage / discourage certain patterns.

Decision

We have decided to use arrow functions across the board in the project. Both for helper functions and for react components.

// Do:
const myFunction = () => {};
const MyComponent = () => {};

// Don't:
function myFunction() {}
function MyComponent() {}

The reason for this decision is to remove mental clutter and free up capacity to easily navigate the codebase. In addition, using arrow functions allows you to avoid the complexity of losing the scope of this for nested functions, and keeps this stable without any huge drawbacks. Losing hoisting is an acceptable compromise.

- - + + \ No newline at end of file diff --git a/contributing/ADRs/front-end/preferred-styles-import-placement.html b/contributing/ADRs/front-end/preferred-styles-import-placement.html index 0809326f15..7aaffaf89b 100644 --- a/contributing/ADRs/front-end/preferred-styles-import-placement.html +++ b/contributing/ADRs/front-end/preferred-styles-import-placement.html @@ -20,15 +20,15 @@ - - + +

ADR: preferred styles import placement

Background

SUPERSEDED BY ADR: Preferred styling method

In the codebase, we have found a need to standardise where to locate the styles import. When using CSS modules, the styles import placement matters for the priority of the styles if you are passing through styles to other components. IE:

// import order matters, because the useStyles in MyComponent now
// is after the useStyles import it will not take precedence if it has
// a styling conflict.
import useStyles from './SecondComponent.styles.ts';
import MyComponent from '../MyComponent/MyComponent.tsx';

const SecondComponent = () => {
const styles = useStyles();

return <MyComponent className={styles.overrideStyles} />
}

Decision

We have decided to always place style imports as the last import in the file, so that any components that the file may use can safely be overriden with styles from the parent component.

// Do:
import MyComponent from '../MyComponent/MyComponent.tsx';

import useStyles from './SecondComponent.styles.ts';

const SecondComponent = () => {
const styles = useStyles();

return <MyComponent className={styles.overrideStyles} />;
};

// Don't:
import useStyles from './SecondComponent.styles.ts';
import MyComponent from '../MyComponent/MyComponent.tsx';

const SecondComponent = () => {
const styles = useStyles();

return <MyComponent className={styles.overrideStyles} />;
};

The reason for this decision is to remove the posibillity for hard to find bugs, that are not obvious to detect and that might be time consuming to find a solution to.

- - + + \ No newline at end of file diff --git a/contributing/ADRs/front-end/preferred-styling-method.html b/contributing/ADRs/front-end/preferred-styling-method.html index 66a937d60f..d190b6b24e 100644 --- a/contributing/ADRs/front-end/preferred-styling-method.html +++ b/contributing/ADRs/front-end/preferred-styling-method.html @@ -20,8 +20,8 @@ - - + + @@ -31,7 +31,7 @@ external interop package to maintain compatability with the latest version. The preferred path forward is to use styled components which is supported natively in @material/ui and sparingly use the sx prop available on all mui components.

Consequences: code sharing

With makeStyles it was common to reuse CSS fragments via library utilities. In the styled components approach we use themeable functions and object literals

import { Theme } from '@mui/material';

export const focusable = (theme: Theme) => ({
color: theme.palette.primary.main,
});

export const flexRow = {
display: 'flex',
alignItems: 'center',
};

Usage:

const StyledLink = styled(Link)(({ theme }) => ({
...focusable(theme),
}));

<IconButton sx={focusable}/>
- - + + \ No newline at end of file diff --git a/contributing/ADRs/front-end/sdk-generator.html b/contributing/ADRs/front-end/sdk-generator.html index d8310ac0b6..e200ac09b3 100644 --- a/contributing/ADRs/front-end/sdk-generator.html +++ b/contributing/ADRs/front-end/sdk-generator.html @@ -20,15 +20,15 @@ - - + +

ADR: OpenAPI SDK Generator

Background

In our current frontend setup we have a lot of code that can be generated out of the OpenAPI schema. Types have not been updated in a while, and even some new features rely on hard-coded types instead of auto-generated files in src/openapi. Fetchers and actions in the frontend involve custom-built code that is grouped in the src/hooks/api folder. There is a separation between getters and actions. Getters use the SWR library. API actions (POST/PUT/DELETE) are grouped by feature and are exposed to components reliant on useAPI hook.

Decisions

  • We will use the Orval package to generate the typescript types for our SDK.
  • We will consider using Orval to generate the HTTP getters and actions for our SDK in the future, but will first carefully test this approach in new features under development to weed out edge cases.
  • We will deprecate src/interfaces related to API calls and use src/openapi models instead.

Advantages

SDK generated out of it will be better than what we have right now. It will help reduce the risk of duplication and inconsistencies in our SDK, as it will be generated from a single source of truth.

  • We retain the flexibility of previous solution, because we can implement our own fetcher function, and substitute response and error type generics. See https://orval.dev/guides/custom-client
  • It supports anyOf and oneOf schema, which the previous generator did not support.
  • If we decide to use Orval to generate the HTTP getters and actions for our SDK, it will reduce the amount of boilerplate code required when working with the new APIs.

Concerns

  • We will need to ensure that we keep our OpenAPI specification up-to-date, as any changes in the specification will be reflected in the generated SDK. We need an enterprise version with all experimental endpoints enabled to get complete output.
  • Orval is well-maintained, but it appears to have just 1 core contributor. SDK is a very important thing and should be reliable.
  • We can revert to writing some API calls by hand if this approach is not flexible enough, but this can cause issues negating the benefits of code generation.

Alternative packages considered

  • @openapitools/openapi-generator - does not offer as many customization options as Orval. It struggles with anyOf and oneOf types. It fails
  • rapini: This package is less flexible and less actively maintained than Orval, and we therefore decided against it.
  • @openapi-codegen - does not generate SWR hooks
- - + + \ No newline at end of file diff --git a/contributing/ADRs/overarching/domain-language.html b/contributing/ADRs/overarching/domain-language.html index dd662daac7..6d858428d7 100644 --- a/contributing/ADRs/overarching/domain-language.html +++ b/contributing/ADRs/overarching/domain-language.html @@ -20,15 +20,15 @@ - - + +

ADR: Domain language

Background

In the codebase, we have seen the need to define a domain language that we use to refer to features, methods to keep it consistent across the codebase. This ADR will contain a growing list of domain language used to keep the consistency across the codebase.

Decision

We have decided to use the same domain language for the features we develop. Each feature will have it's own domain language to keep it consistent across the codebase.

Change requests domain language

  • Change request: An entity referring to the overarching data structure of a change request. A change request contains changes, and can be approved or rejected.
  • Change: A term referring to a single change within a change request
  • Changes: A term referring to a group of changes within a change request
  • Discard: A term used for deleting a single change of a change request, or discarding an entire change request.
  • Pending: A pending change request is one that has not yet been applied or discarded. In other words, it is in one of these three states:
    1. Draft
    2. In review
    3. Approved
  • Closed: A closed change request has either been applied or cancelled and can no longer be changed. Change requests that are either Applied or Cancelled are considered closed.
- - + + \ No newline at end of file diff --git a/contributing/ADRs/overarching/separation-request-response-schemas.html b/contributing/ADRs/overarching/separation-request-response-schemas.html index 55de7042df..7f16df6561 100644 --- a/contributing/ADRs/overarching/separation-request-response-schemas.html +++ b/contributing/ADRs/overarching/separation-request-response-schemas.html @@ -20,15 +20,15 @@ - - + +

ADR: Separation of request and response schemas

Background

During the updating of our OpenAPI documentation, we have encountered issues related to the scope and strictness of our schemas for requests and responses. Currently, we are reusing the same schema for both request and response, which has led to situations where the schemas are either too broad for a response or too strict for a request. This has caused difficulties in accurately defining the expected data structures for API interactions.

Decision

After careful consideration and discussion, it has been decided to separate the request and response schemas to address the challenges we have encountered. By creating distinct schemas for requests and responses, we aim to improve the flexibility and precision of our API documentation.

Advantages

Separating the schemas will allow us to establish more precise and constrained response types while enabling more forgiving request types. This approach will facilitate better alignment between the expected data structures and the actual data transmitted in API interactions.

The separation of request and response schemas will provide the following benefits:

  1. Enhanced clarity and correctness: With dedicated schemas for requests and responses, we can define the precise structure and constraints for each interaction. This will help prevent situations where the schemas are overly permissive or restrictive, reducing ambiguity and ensuring that the code handling the requests and responses is more reliable and easier to understand. By separating the schemas, we can define specific and precise structures for requests and responses, minimizing the use of undefined values and improving the overall correctness of the codebase and API implementation. The client knows exactly what data to send in the requests, and the server knows what to expect in the responses, ensuring that both parties are aligned in terms of data structures and expectations.

  2. Improved maintainability: By avoiding the reuse of schemas between requests and responses, we can modify and update them independently. This decoupling of schemas will simplify maintenance efforts and minimize the risk of unintended side effects caused by changes in one context affecting the other.

  3. Flexibility for future enhancements: Separating request and response schemas lays the foundation for introducing additional validation or transformation logic specific to each type of interaction. This modularity will enable us to incorporate future enhancements, such as custom validation rules or middleware, with ease.

Concerns

While this decision brings several benefits, we acknowledge the following concerns that may arise from the separation of request and response schemas:

  1. Increased schema maintenance: By having separate schemas for requests and responses, there will be a need to maintain and update two sets of schemas instead of a single shared schema. This could potentially increase the maintenance overhead and introduce the possibility of inconsistencies between the two schemas.

  2. Data duplication and redundancy: With the separation of schemas, there might be instances where certain data fields or structures are duplicated between the request and response schemas. This redundancy could lead to code duplication and increase the risk of inconsistencies if changes are not carefully synchronized between the two schemas.

Conclusion

By implementing the separation of request and response schemas, we aim to improve the robustness and maintainability of our API documentation. This decision will empower developers to build more reliable integrations by providing clearer guidelines for both request and response data structures.

Furthermore, this separation of schemas brings valuable benefits to our internal development process. It allows us to write more robust code and reduces the need for extensive manipulation of incoming requests to fit the correct shapes. By clearly defining the structure and constraints for each interaction, we minimize the likelihood of bugs and make the code significantly easier to reason about and work with.

While a big bang migration replacing all schemas at once is not feasible, we will follow the Boy Scout Rule and aim to complete the migration to separated schemas by the release of version 6.0. This means that as developers make changes or additions to the code, they will incorporate the separation of schemas, gradually updating the existing codebase over time. This approach ensures a smooth transition to the separated schemas, allowing us to continually improve the code handling requests and responses, reducing reliance on undefined values and promoting clarity, correctness, and maintainability throughout the development process.

Overall, this approach will facilitate better communication, reduce confusion, and enhance the overall developer experience when interacting with our APIs, while providing the aforementioned benefits of more robust code, correctness and improved maintainability.

- - + + \ No newline at end of file diff --git a/contributing/backend/overview.html b/contributing/backend/overview.html index 6b2b9d1aa9..fd9b14192b 100644 --- a/contributing/backend/overview.html +++ b/contributing/backend/overview.html @@ -20,15 +20,15 @@ - - + +

Back end

The backend is written in nodejs/typescript. It's written as a REST API following a CSR (controller, service, repository/store) pattern. The following ADRs are defined for the backend:

ADRs

We have created a set of ADRs to help guide the development of the backend:

Requirements

Before developing on this project you will need two things:

  • PostgreSQL 14.x or newer
  • Node.js 14.x or newer
yarn install
yarn dev

PostgreSQL

To run and develop unleash, you need to have PostgreSQL database (PostgreSQL v14.x or newer) locally.

Unleash currently also work with PostgreSQL v14+, but this might change in a future feature release, and we have stopped running automatic integration tests below PostgreSQL v12. The current recommendation is to use a role with Owner privileges since Unleash uses Postgres functions to simplify our database usage.

Create a local unleash databases in postgres

$ psql postgres <<SQL
CREATE USER unleash_user WITH PASSWORD 'password';
CREATE DATABASE unleash WITH OWNER unleash_user;
CREATE DATABASE unleash_test WITH OWNER unleash_user;
ALTER DATABASE unleash_test SET timezone TO 'UTC';
SQL

Then set env vars:

(Optional as unleash will assume these as default values).

export DATABASE_URL=postgres://unleash_user:password@localhost:5432/unleash
export TEST_DATABASE_URL=postgres://unleash_user:password@localhost:5432/unleash_test

PostgreSQL with docker

If you don't want to install PostgreSQL locally, you can spin up an Docker instance. We have created a script to ease this process: scripts/docker-postgres.sh

Start the application

In order to start the application you will need Node.js v14.x or newer installed locally.

// Install dependencies
yarn install

// Start Unleash in development
yarn dev

// Unleash UI
http://localhost:3000

// API:
http://localhost:3000/api/

// Execute tests in all packages:
yarn test

Database changes

We use database migrations to track database changes. Never change a migration that has been merged to main. If you need to change a migration, create a new migration that reverts the old one and then creates the new one.

Making a schema change

To run migrations, you will set the environment variable for DATABASE_URL

export DATABASE_URL=postgres://unleash_user:password@localhost:5432/unleash

Use db-migrate to create new migrations file.

> yarn run db-migrate create YOUR-MIGRATION-NAME

All migrations require one up and one down method. There are some migrations that will maintain the database integrity, but not the data integrity and may not be safe to run on a production database.

Example of a typical migration:

/* eslint camelcase: "off" */
'use strict';

exports.up = function(db, cb) {
db.createTable(
'examples',
{
id: { type: 'int', primaryKey: true, notNull: true },
created_at: { type: 'timestamp', defaultValue: 'now()' },
},
cb,
);
};

exports.down = function(db, cb) {
return db.dropTable('examples', cb);
};

Test your migrations:

> yarn run db-migrate up
> yarn run db-migrate down

Publishing / Releasing new packages

Please run yarn test checks before publishing.

Run npm run publish to start the publishing process.

npm run publish:dry

- - + + \ No newline at end of file diff --git a/contributing/developer-guide.html b/contributing/developer-guide.html index 66ac682ae2..9bdbc21534 100644 --- a/contributing/developer-guide.html +++ b/contributing/developer-guide.html @@ -20,15 +20,15 @@ - - + +

Developer guide

Introduction

This repository contains two main parts. The backend and the frontend of unleash. The backend is a Node.js application that is built using TypeScript. The frontend is a React application that is built using TypeScript. The backend specific code can be found in the src lib folder. The frontend specific code can be found in the frontend folder.

Development philosophy

The development philosophy at unleash is centered at delivering high quality software. We do this by following a set of principles that we believe will help us achieve this goal. We believe that these principles will also help us deliver software that is easy to maintain and extend, serving as our north star.

We believe that the following principles will help us achieve our goal of delivering high quality software:

  • We test our code always

Software is difficult. Being a software engineer is about acknowledging our limits, and taking every precaution necessary to avoid introducing bugs. We believe that testing is the best way to achieve this. We test our code always, and prefer automation over manual testing.

  • We strive to write code that is easy to understand and maintain

We believe code is a language. Written code is a way to communicate intent. It's about explaining to the reader what this code does, in the shortest amount of time possible. As such, writing clean code is supremely important to us. We believe that this contributes to keeping our codebase maintainable, and helps us maintain speed in the long run.

  • We think about solutions before comitting

We don't jump to implementation immediately. We think about the problem at hand, and try to examine the impact that this solution may have in a multitude of scenarios. As our product core is open source, we need to balance the solutions and avoid implementations that may be cumbersome for our community. The need to improve our paid offering must never come at the cost of our open source offering.

Required reading

The following resources should be read before contributing to the project:

- - + + \ No newline at end of file diff --git a/contributing/frontend/overview.html b/contributing/frontend/overview.html index 4f9f2379b0..4370bb8069 100644 --- a/contributing/frontend/overview.html +++ b/contributing/frontend/overview.html @@ -20,15 +20,15 @@ - - + + - - + + \ No newline at end of file diff --git a/docs/using-unleash/deploy/license-keys/index.html b/docs/using-unleash/deploy/license-keys/index.html new file mode 100644 index 0000000000..1ae7c80950 --- /dev/null +++ b/docs/using-unleash/deploy/license-keys/index.html @@ -0,0 +1,11 @@ + + + + + + + + + \ No newline at end of file diff --git a/feature-flag-tutorials.html b/feature-flag-tutorials.html index cd525bdcc8..9cd838500c 100644 --- a/feature-flag-tutorials.html +++ b/feature-flag-tutorials.html @@ -20,15 +20,15 @@ - - + +
- - + + \ No newline at end of file diff --git a/feature-flag-tutorials/flutter/a-b-testing.html b/feature-flag-tutorials/flutter/a-b-testing.html index c4f20bc0a5..de0acd5ac6 100644 --- a/feature-flag-tutorials/flutter/a-b-testing.html +++ b/feature-flag-tutorials/flutter/a-b-testing.html @@ -20,8 +20,8 @@ - - + + @@ -29,7 +29,7 @@

A/B Testing in Flutter using Unleash and Mixpanel

note

This article is a contribution by Ayush Bherwani as a part of the Community Content Program. You can also suggest a topic by opening an issue, or Write for Unleash as a part of the Community Content Program.

After successfully integrating the first feature flag in the Unsplash sample app, let’s talk about how you can use Unleash to perform experimentation, also known as A/B testing, in Flutter to ship features more confidently.

For this article, we’ll integrate feature flags for A/B testing to experiment with “like image” feature user experience. As an overview, the app is quite simple, with two screens displaying images and image details respectively. The behavior of the “image details” feature is controlled through an Unleash instance. You can check out the previous article, “How to set up feature flags in Flutter” for an overview of the code structure and implementation. For those who want to skip straight to the code, you can find it on GitHub.

Here’s a screenshot of the application: Unsplash App built on Flutter

Setup variants in Unleash

In your Unleash instance, create a new feature flag called likeOptionExperiment. Choose the toggle type called Experiment and enable the impression data. By default, the flag will be set to false.

Set Up Variant in Unleash

Now that you have created your feature toggle, let’s create two new variants “gridTile'' and “imageDetails” respectively. These variants will help you position your “like image” button.

Succesfully setting up variant in Unleash

Below is a screenshot of experimentation in action based on the likeOptionExperiment flag and corresponding variants.

Unsplash App built on Flutter

Setup Mixpanel

For analytics and metrics, we’ll use Mixpanel to track user behavior and usage patterns. We have chosen Mixpanel because it offers a user-friendly setup and in-depth user analytics and segmentation. Given that the project follows clean architecture and Test-Driven Development (TDD) principles, you’ll want to create an abstract layer to interact with the Mixpanel.

Whenever a user opens the app, we track like-variant if likeOptionExperiment is enabled to tag them with their assigned variant (gridTile or imageDetails). The stored variant in Mixpanel can be used later to analyze how each variant impacts user behavior to like an image.

Whenever a user interacts with the LikeButton, we track trackLikeEventForExperimentation, along with their assigned variants. By correlating the trackLikeEventForExperimentation with the like-variant, you can effectively measure the impact of a variant on user behavior and make data-driven decisions. To learn how to correlate and generate reports, see the Mixpanel docs.

abstract class MixpanelConfig {
/// ...

/// Helps you get the metrics of experimentation to analysis
/// the different position of the share image button.
void trackLikeEventForExperimentation({
required LikeButtonPosition likeButtonPosition,
required String photoId,
});

/// Help you get the variant based on which we can create funnel
/// for analytics.
void trackLikeVariant(LikeButtonPosition likeButtonPosition);
}

class MixpanelConfigImpl implements MixpanelConfig {
final TargetPlatformExtended targetPlatformExtended;

MixpanelConfigImpl(this.targetPlatformExtended);

Mixpanel get mixpanel {
return ServiceLocator.getIt<Mixpanel>();
}

/// ...

@override
void trackLikeEventForExperimentation({
required LikeButtonPosition likeButtonPosition,
required String photoId,
}) {
if (targetPlatformExtended.isMobile) {
mixpanel.track('like-experimentation', properties: {
"variant": describeEnum(likeButtonPosition),
"photoId": photoId,
});
}
}

@override
void trackLikeVariant(LikeButtonPosition likeButtonPosition) {
if (targetPlatformExtended.isMobile) {
mixpanel.track('like-variant', properties: {
"variant": describeEnum(likeButtonPosition),
});
}
}
}

Once you have your configuration in place, the next step is to create MixPanel and MixpanelConfig and add it to your service locator class. Make sure that you have a Mixpanel API key.

class ServiceLocator {
ServiceLocator._();

static GetIt get getIt => GetIt.instance;

static Future<void> initialize() async {
/// ...

final unleash = UnleashClient(
url: Uri.parse('http://127.0.0.1:4242/api/frontend'),
clientKey: dotenv.env["UNLEASH_API_KEY"] as String,
appName: 'unplash_demo',
);

await unleash.start();

getIt.registerLazySingleton(() => unleash);
getIt.registerLazySingleton<UnleashConfig>(
() => UnleashConfigImpl(getIt()),
);

getIt.registerLazySingleton<WebPlatformResolver>(
() => WebPlatformResolverImpl(),
);


final TargetPlatformExtended targetPlatformExtended =
TargetPlatformExtendedImpl(getIt());

getIt.registerLazySingleton<TargetPlatformExtended>(
() => targetPlatformExtended,
);

if (targetPlatformExtended.isMobile) {
final mixPanel = await Mixpanel.init(
dotenv.env["MIXPANEL_KEY"] as String,
trackAutomaticEvents: false,
);

getIt.registerLazySingleton(() => mixPanel);
}

getIt.registerLazySingleton<MixpanelConfig>(
() => MixpanelConfigImpl(getIt()),
);

/// ...
}
}

Integration in Flutter

Let’s dive into how you can use these variants in your Flutter application.

You’ll have to modify UnleashConfig which helps you test the Unleash functionalities in isolation from the rest of the app.

const String isImageDetailsEnabledToggleKey = "isImageDetailsEnabled";
const String likeOptionExperimentKey = "likeOptionExperiment";

/// Helps determine the position of like button
/// [gridTile] will be used to position like image button
/// on [ImageTile].
/// [imageDetails] will be used to position like image button
/// inside the [ImageDetails] page.
enum LikeButtonPosition { gridTile, imageDetails }

abstract class UnleashConfig {
/// ...
bool get isLikeOptionExperimentEnabled;
LikeButtonPosition get likeButtonPosition;
}

class UnleashConfigImpl extends UnleashConfig {
final UnleashClient unleash;

UnleashConfigImpl(this.unleash);

/// ...

@override
bool get isLikeOptionExperimentEnabled =>
unleash.isEnabled(likeOptionExperimentKey);

@override
LikeButtonPosition get likeButtonPosition {
final variant = unleash.getVariant(likeOptionExperimentKey);
return LikeButtonPosition.values.byName(variant.name);
}
}

After updating UnleashConfig you may want to create a _trackLikeExperimentationVariant method which you can call in initState of “HomePage” to get the variant details.

void _trackLikeExperimentationVariant() {
final unleashConfig = ServiceLocator.getIt<UnleashConfig>();
final mixpanelConfig = ServiceLocator.getIt<MixpanelConfig>();
if (unleashConfig.isLikeOptionExperimentEnabled) {
mixpanelConfig.trackLikeVariant(unleashConfig.likeButtonPosition);
}
}

You’ll create a new widget “LikeButton'' which can be utilized for both variants. Make sure to use MixpanelConfig to track user engagement for analytics purposes.

class LikeButton extends StatelessWidget {
final ValueNotifier<bool> isLikedNotifier;
/// Used to track the variant option for the mixpanel event.
final LikeButtonPosition likeButtonPosition;
final String photoId;

const LikeButton({
super.key,
required this.isLikedNotifier,
required this.likeButtonPosition,
required this.photoId,
});


/// Event fired to track the user engagement on liking an image
void _fireMixpanelEvent() {
final mixpanelConfig = ServiceLocator.getIt<MixpanelConfig>();
mixpanelConfig.trackLikeEventForExperimentation(
likeButtonPosition: likeButtonPosition,
photoId: photoId,
);
}

@override
Widget build(BuildContext context) {
return ValueListenableBuilder<bool>(
valueListenable: isLikedNotifier,
builder: (context, isLiked, child) {
return IconButton(
padding: EdgeInsets.zero,
visualDensity: VisualDensity.compact,
iconSize: 20,
isSelected: isLiked,
splashColor: Colors.red,
onPressed: () {
isLikedNotifier.value = !isLikedNotifier.value;
_fireMixpanelEvent();
},
selectedIcon: const Icon(
CupertinoIcons.heart_fill,
color: Colors.redAccent,
),
icon: const Icon(CupertinoIcons.heart),
);
},
);
}
}

Once you have created LikeButton, the next step is to use the LikeButtonPosition to add the button in the ImageTile widget, and ImageDetails page.

For ImageTile make sure the button is only visible if isLikeOptionExperiment is enabled and LikeButtonPosition is gridTile.

class ImageTile extends StatelessWidget {
final UnsplashImage image;
final UnleashConfig unleashConfig;

const ImageTile({
super.key,
required this.image,
required this.unleashConfig,
});

bool get isFooterEnabled {
return unleashConfig.isLikeOptionExperimentEnabled &&
unleashConfig.likeButtonPosition == LikeButtonPosition.gridTile;
}

Widget tileGap() => const SizedBox(height: 4);

@override
Widget build(BuildContext context) {
return CachedNetworkImage(
imageUrl: image.url,
cacheManager: ServiceLocator.getIt<DefaultCacheManager>(),
imageBuilder: (context, provider) {
return Column(
children: [
/// Some more widgets
if (isFooterEnabled) ...[
ImageTileFooter(image: image),
],
],
);
},
);
}
}

For ImageDetailsPage make sure the button is only visible if isLikeOptionExperiment is enabled and LikeButtonPosition is imageDetails.

@RoutePage()
class ImageDetailsPage extends StatefulWidget {
final String id;

const ImageDetailsPage({
super.key,
required this.id,
});

@override
State<ImageDetailsPage> createState() => _ImageDetailsPageState();
}

class _ImageDetailsPageState extends State<ImageDetailsPage> {
/// ...
late final MixpanelConfig mixPanelConfig;
late final ValueNotifier<bool> isLikedNotifier;
late final UnleashConfig unleashConfig;

@override
void initState() {
super.initState();
/// ...
mixPanelConfig = ServiceLocator.getIt<MixpanelConfig>();
isLikedNotifier = ValueNotifier<bool>(false);
unleashConfig = ServiceLocator.getIt<UnleashConfig>();
/// ...
_fetchImageDetails();
}

/// ...

void _fetchImageDetails() {
bloc.add(FetchImageDetailsEvent(widget.id));
}

bool get isLikeButtonVisible {
return unleashConfig.isLikeOptionExperimentEnabled &&
unleashConfig.likeButtonPosition == LikeButtonPosition.imageDetails;
}

@override
Widget build(BuildContext context) {
return Scaffold(
appBar: AppBar(
leading: IconButton(
onPressed: () {
Navigator.of(context).pop();
},
icon: const Icon(
CupertinoIcons.xmark,
),
),
actions: [
if (isLikeButtonVisible) ...[
LikeButton(
isLikedNotifier: isLikedNotifier,
photoId: widget.id,
likeButtonPosition: LikeButtonPosition.imageDetails,
),
]
],
),
body: /// Body widget;
}
}

Now that you have your pieces clubbed, you can toggle the likeOptionExperiment flag to enable the experimentation of the “like image” feature in the application.

Voila! Your experimentation is enabled for your feature. You’ve also ensured that users can access the “like image” feature depending on the state of the flag and variant.

Analytics

Once your experimentation is up and running, you can visit the Mixpanel dashboard to create a funnel and get insights on the user engagement and conversion rate for different variants.

Below is a funnel screenshot for “like image” experimentation:

Mixpanel Analytics for A/B Testing in Unleash for the Flutter Demo

Conclusion

A/B testing is a low-risk, high-returns approach that can help you make data-driven decisions for your feature releases to increase user engagement, minimize the risk, and increase conversion rates. As a developer, it helps you be confident in your releases by addressing the issues users face and reducing the bounce rates during experimentation with the help of data.

Some of the best practices for experimentation include:

  • You should be open to the results and avoid any hypotheses.
  • You should define the metrics for the success of the experimentation before you run the tests. Keep your success metrics simple and narrowed for better results.
  • Select a group of adequate size for the test to yield definitive results.
  • You should avoid running multiple tests simultaneously, as it may not give reasonable outcomes.

That’s it for today. I hope you found this helpful. Want to dive deep into the code used for this article? It’s all on GitHub.

- - + + \ No newline at end of file diff --git a/feature-flag-tutorials/nextjs/implementing-feature-flags.html b/feature-flag-tutorials/nextjs/implementing-feature-flags.html index af7297f90f..52ea35408e 100644 --- a/feature-flag-tutorials/nextjs/implementing-feature-flags.html +++ b/feature-flag-tutorials/nextjs/implementing-feature-flags.html @@ -20,8 +20,8 @@ - - + + @@ -30,7 +30,7 @@

How to Implement Feature Flags in Next.js using Unleash

note

This article is a contribution by Kafilat Adeleke as a part of the Community Content Program. You can also suggest a topic by opening an issue, or Write for Unleash as a part of the Community Content Program.

Imagine turning on and off features in your web application without redeploying your code. Sounds too good to be true? It's possible with feature flags and Next.js, a powerful framework for building fast and scalable web applications using React.

Feature flags are a powerful technique that allows you to toggle features on and off dynamically without redeploying your code. This can help you to deliver faster and safer web applications, as you can test new features in production, perform gradual rollouts, and revert changes quickly if something goes wrong.

Next.js provides features such as:

  • Server-side rendering
  • Static site generation
  • Code splitting
  • Dynamic routing
  • TypeScript support
  • CSS modules support
  • And many more

With Next.js, you can create hybrid applications that can use both static and dynamic pages. Static pages are pre-rendered at build time and served from a CDN, which makes them fast and secure. Dynamic pages are rendered on-demand by a Node.js server, which allows you to use data from any source and handle user interactions. You can also use incremental static regeneration to update static pages without rebuilding your entire application.

However, one of the challenges of using Next.js is that it requires you to rebuild your application every time you want to change something in your code. This can be time-consuming and risky, especially if you want to test new features in production or perform gradual rollouts. Therefore, you need a way to toggle features on and off dynamically without redeploying your code. This is where feature flags come in.

Unleash is an open-source, easy-to-use feature management platform that supports Next.js and other frameworks. With Unleash, you can create and manage feature flags from a user-friendly dashboard and use them in your code with a simple API. Unleash also provides advanced features such as segmentation, strategies, and integrations.

In this tutorial, you will learn how to use feature flags in a Next.js application that displays random activities to users using Unleash. We will use the @unleash/nextjs package, which provides easy integration of Unleash feature flags in a Next.js application. Note: If you only need a very simple setup for feature flags in your Next.js application, like toggling a feature on and off for all users, use the Vercel Edge Config. You can avoid making requests to the Unleash API from your application and instead use the edge config to inject the feature flag values into your pages. Next.js Feature Flag Architecture Diagram

Setup Unleash

Before you proceed, ensure you have an Unleash instance running. Run the following commands in the terminal:

wget getunleash.io/docker-compose.yml
docker-compose up -d

This will start Unleash in the background. Once Unleash is running, you can access it at http://localhost:4242.

Username: admin
Password: unleash4all

Create a New Feature

Create a new feature flag in your Unleash instance named "activity".

Create a new feature flag in Unleash

Integrating Unleash in Nextjs

To get started with Next.js and Unleash, you need to create a Next.js project and add the @unleash/nextjs package as a dependency.

You can run the following commands in your terminal to do this:

npx create-next-app activity-app
cd activity-app
npm install @unleash/nextjs

To make feature flags available to our Next.js application, we will wrap it with the FlagProvider component from the @unleash/nextjs package. This component will initialize the Unleash SDK and provide access to feature flags throughout our application. We will do this by adding it to our pages/_app.tsx file.

import type { AppProps } from "next/app"
import { FlagProvider } from "@unleash/nextjs/client"

export default function App({ Component, pageProps }: AppProps) {
return (
<FlagProvider>
<Component {...pageProps} />
</FlagProvider>
)
}

Next, we will use the Bored API to retrieve a random activity and then use the useFlag feature to determine whether the activity is displayed or not.

import { useEffect, useState } from "react"
import { useFlag } from "@unleash/nextjs/client"

const Activity = () => {
const [activityData, setActivityData] = useState({})

const showActivity = useFlag("activity")

useEffect(() => {
const fetchActivity = async () => {
try {
const response = await fetch("https://www.boredapi.com/api/activity/")
const data = await response.json()
setActivityData(data)
} catch (error) {
console.error("Error fetching activity:", error)
}
}

if (showActivity) {
fetchActivity()
}
}, [showActivity])

return (
<div className="bg-gray-100 min-h-screen flex items-center justify-center">
<div className="bg-white p-8 rounded shadow-lg">
<h1 className="text-3xl font-bold mb-4">
Here is an activity for you!
</h1>
{showActivity ? (
<>
<p className="mb-2">Activity: {activityData.activity}</p>
<p className="mb-2">Participants: {activityData.participants}</p>
<p>Price: ${activityData.price}</p>
</>
) : (
<p>Activity not available</p>
)}
</div>
</div>
)
}

export default Activity

Our feature flag can now be used to control whether or not activity is displayed. If we toggle on the gradual roll out:

Gradual Rollout

The activity is displayed:

Activity Successful

If we toggle it off:

Toggle Off

No activity is displayed:

Activity Successful

To configure access to Unleash beyond localhost development, follow these steps:

  • Self-host Unleash or run an instance on Unleash Cloud.

  • Get an API key from the Unleash dashboard.

  • Store the API key in your Vercel Project Environment Variable, which secures it and makes it accessible in your code.

Conclusion

Feature flags are a powerful tool for managing features in web applications. This tutorial showed us how to use feature flags with Next.js and Unleash. We have seen how to create and manage feature flags in the Unleash dashboard, and how to use them in our Next.js code with the @unleash/nextjs library. We have also seen how to test our feature flags by toggling them on and off in the Unleash dashboard.

Unleash is a powerful and easy-to-use feature management platform that can help you deliver faster and safer web applications with Next.js and other frameworks. If you are not already using feature flags in your Next.js applications, I encourage you to try Unleash and see how it can improve your development workflow.

- - + + \ No newline at end of file diff --git a/feature-flag-tutorials/react.html b/feature-flag-tutorials/react.html index 3e617e3b28..5161dce9c8 100644 --- a/feature-flag-tutorials/react.html +++ b/feature-flag-tutorials/react.html @@ -20,8 +20,8 @@ - - + + @@ -32,7 +32,7 @@ Since Yarn is required in order to run the app, make sure you have it installed globally. If you do not, run the command below.

npm install yarn@latest -g

In your app's directory, begin installing the dependencies and then run the app:

yarn
yarn dev

Note: We recommend using the default ports that exist in the app's configurations, which are explained in the README for it to point to. In order to ensure those function as expected, make sure no other apps are running on your machine that also port to localhost:3000 and localhost:3001.

In your browser at http://localhost:3000, you will be directed to the sign-in page of the Cypress Real World App. Utilize one of the pre-existing user accounts from the database to sign in.

Username: Allie2
Password: s3cret

For more detailed instructions on the setup process for this app, review the README.

It’s time to pull in your newly created feature flag in your app. Run the following command to install the Unleash React SDK in your repo:

yarn add @unleash/proxy-client-react unleash-proxy-client

Once Unleash has been installed, open up a code editor like VSCode to view your React repo.

In src/index.tsx, import the <FlagProvider>:

import { FlagProvider } from "@unleash/proxy-client-react";

Paste in a configuration object:

const config = {
url: "http://localhost:4242/api/frontend", // Your local instance Unleash API URL
clientKey: "<client_key>", // Your client-side API token
refreshInterval: 15, // How often (in seconds) the client should poll the proxy for updates
appName: "cypress-realworld-app", // The name of your application. It's only used for identifying your application
};

In the Router section of this file, wrap the FlagProvider around the existing <App /> component:

<FlagProvider config={config}>
<App />
</FlagProvider>

Next, replace the <client_key> string in the config object with the API token you generated. You can do this by copying the API token into your clipboard from the API Access view table and pasting it into the code.

This configuration object is used to populate the FlagProvider component that comes from Unleash and wraps around the application, using the credentials to target the specific feature flag you created for the project.

You can check our documentation on API tokens and client keys for more specifics and see additional use-cases in our Client-Side SDK with React documentation.

5. Use the feature flag to rollout a notifications badge

In a real world use case for your feature flag, you can gradually rollout new features to a percentage of users by configuring the flag's strategy.

In this case, we want to rollout a new notifications badge that will appear in the top navigation bar so users can see the latest updates from transactions between contacts. This will require us to modify the visibility of a React component that is rendered in our app.

In src/components/NavBar.tsx, import the useFlag feature:

import { useFlag } from "@unleash/proxy-client-react";

Within the NavBar component in the file, define and reference the flag you created.

const notificationsBadgeEnabled = useFlag("newNotificationsBadge");

This flag will be used to conditionally render the notification icon Badge that is pulled in from Material-UI. If the flag is enabled, the notification badge will display to users and will route them to the Notifications view.

Find the Badge component in the file and wrap it in a boolean operator:

{notificationsBadgeEnabled && (
<Badge
badgeContent={allNotifications?.length}
data-test="nav-top-notifications-count"
classes={{ badge: classes.customBadge }}
>
<NotificationsIcon />
</Badge>
)}

Note: Ensure you have the correct format in your file or the Prettier formatter will display an error.

6. Verify the toggle experience

In your Unleash instance, you can toggle your feature flag on or off to verify that the different UI experiences load accordingly.

Unleash turn on feature flag

Enabling the flag will result in being able to see the notifications icon in the top menu of the app.

Notification icon badge visible

If you disable the flag, this results in a view of a navigation menu without the notification badge for all users.

Notification icon badge not visible

You've successfully implemented a feature flag using best practices to control the release of a notifications feature in a real world app!

Conclusion

In this tutorial, we installed Unleash locally, created a new feature flag, installed Unleash into a React app, and toggled the visibility of a notifications feature within a real world open source project!

- - + + \ No newline at end of file diff --git a/feature-flag-tutorials/react/examples.html b/feature-flag-tutorials/react/examples.html index c99112dd02..4ec6d957f4 100644 --- a/feature-flag-tutorials/react/examples.html +++ b/feature-flag-tutorials/react/examples.html @@ -20,15 +20,15 @@ - - + +

React Feature Flag Examples

In our React tutorial, we implemented a simple on/off feature flag that could turned on and off. In the real world, many feature flag use cases have more nuance than this. This document will walk you through some common examples of using feature flags in React with some of those more advanced use cases in mind.

Applications evolve, and teams must manage all aspects of this evolution, including the flags used to control the application. We built multiple features into Unleash to address the complexities of releasing code and managing feature flags along the way:

  1. Gradual rollouts
  2. Canary deployments
  3. A/B testing
  4. Feature flag metrics & reporting
  5. Feature flag audit logs
  6. Change management & feature flag approvals
  7. Flag automation & workflow integration
  8. Common usage examples of React feature flags

Gradual Rollouts for React Apps

It’s common to use feature flags to roll out changes to a percentage of users. Flags allow you to monitor your application and infrastructure for undesired behavior (such as errors, memory bottlenecks, etc.) and to see if the changes improve the outcomes for your application (to increase sales, reduce support requests, etc.)

Doing a gradual rollout for a React-based application with Unleash is very straightforward. To see this in action, follow our How to Implement Feature Flags in React tutorial, which implements a notification feature using feature flags.

Once you have completed the tutorial, you can modify the basic setup to adjust the percentage of users who experience this feature with a gradual rollout. The percentage of users split between the notification feature being visible or not is cached so their user experience will remain consistent.

Navigate to the Gradual Rollout form in Unleash by using the "Edit strategy" button.

The &quot;edit strategy&quot; button uses a pencil icon and is located on every strategy.

Adjust the percentage of users to 50% or whichever value you choose, and refresh your app in the browser to see if your user has the new feature experience.

Gradual rollout form

You can achieve this same result using our API with the following code:

curl --location --request PUT 'http://localhost:4242/api/admin/projects/default/features/newNotificationsBadge/environments/development/strategies/{STRATEGY_ID}' \
--header 'Authorization: INSERT_API_KEY' \
--header 'Content-Type: application/json' \
--data-raw '{
"name": "flexibleRollout",
"title": "",
"constraints": [],
"parameters": {
"rollout": "50",
"stickiness": "default",
"groupId": "newNotificationsBadge"
},
"variants": [],
"segments": [],
"disabled": false
}'

Learn more about gradual rollouts in our docs.

Canary Deployments in React

What is a canary deployment?

Canary releases are a way to test and release code in different environments for a subset of your audience, which determines which features or versions of the platform people have access to.

Why use canary deployments?

Canary deployments are a safer and more gradual way to make changes in software development. They help find any abnormalities and align with the agile process for faster releases and quick reversions.

How to leverage feature flags for canary deployments in React?

Unleash has a few ways to help manage canary deployments for React apps at scale:

  • Using a gradual rollout (which we implemented in a previous section) would be a simple use case but would reduce the amount of control you have over who gets the new feature.

  • Using either constraints or segments (which are a collection of constraints) for a subset of your users to get the new feature vs. the old feature, for more control than a gradual rollout

  • Strategy variants are used to do the same canary deployment, but can be scaled to more advanced cases. For example, if you have 2+ new features and are testing to see if they are better than the old one, you can use variants to split your population of users and conduct an A/B test with them.

Let’s walk through how to utilize strategy constraints in our React app through the Unleash platform.

Configure strategy constraints for canary deployments

We will build a strategy constraint on top of our existing gradual rollout strategy. This will allow us to target a subset of users to rollout to.

In Unleash, start from the feature flag view and edit your Gradual Rollout strategy from your development environment.

Gradual Rollout configure strategy constraint image

This will take you to the gradual rollout form. Click on the ‘Add constraint’ button.

Add constraint button image

Let’s say we are experimenting with releasing the notifications feature for a limited time. We want to release it to all users, capture some usage data to compare it to the old experience, and then automatically turn it off.

We can configure the constraint in the form to match these requirements:

Create constraint image

  • The context field is set to currentTime
  • The operator is set to DATE_BEFORE
  • The date time is set (to any time) in the future

Once you’ve filled out the proper constraint fields, click ‘Done’ and save the strategy.

Graduall rollout with constraint image

Your release process is now configured with a datetime-based strategy constraint.

Alternatively, you can send an API command to apply the same requirements:

curl --location --request PUT 'http://localhost:4242/api/admin/projects/default/features/newNotificationsBadge/environments/development/strategies/806ebcbd-bb03-4713-8081-7dca3905e612' \
--header 'Authorization: INSERT_API_KEY' \
--header 'Content-Type: application/json' \
--data-raw '{
"name": "flexibleRollout",
"title": "",
"constraints": [
{
"value": "2024-02-27T17:00:00.000Z",
"values": [],
"inverted": false,
"operator": "DATE_BEFORE",
"contextName": "currentTime",
"caseInsensitive": false
}
],
"parameters": {
"rollout": "50",
"stickiness": "default",
"groupId": "newNotificationsBadge"
},
"variants": [],
"segments": [],
"disabled": false
}'

Read our documentation for more context on the robustness of strategy constraint configurations and use cases.

A/B Testing in React

A/B testing is a common way for teams to test out how users interact with two or more versions of a new feature that is released. At Unleash, we call these variants.

We can expose a particular version of the feature to select user bases when a flag is enabled. From there, a way to use the variants is to view the performance metrics and see which is more efficient.

We can create several variations of this feature to release to users and gather performance metrics to determine which one yields better results. While teams may have different goals for measuring performance, Unleash enables you to configure strategy for the feature variants within your application/service and the platform.

In the context of our React tutorial, we have a notifications badge feature that displays in the top navigation menu. To implement feature flag variants for an A/B Test in React, we will set up a variant in the feature flag and use an announcement icon from Material UI to render a different version.

In Unleash, navigate to the feature flag’s Variants tab and click ‘Add variant’.

Add variant image

In the Variants form, add two variants that will reference the different icons we will render in our app. Name them ‘notificationsIcon’ and ‘announcementsIcon’.

Note: We won’t use any particular payload from these variants other than their default returned objects. For this example, we can keep the variant at 50% weight for each variant, meaning there is a 50% chance that a user will see one icon versus the other. You can adjust the percentages based on your needs, such as making one icon a majority of users would see by increasing its weight percentage. Learn more about feature flag variant properties.

Variant form image

Once you click 'Save variants' at the bottom of the form, your view will display the list of variants in the development environment with their respective metadata.

Variant added image

Alternatively, we can create new variants via an API command below:

curl --location --request PATCH 'http://localhost:4242/api/admin/projects/default/features/newNotificationsBadge/environments/development/variants' \
--header 'Authorization: INSERT_API_KEY' \
--header 'Content-Type: application/json' \
--data-raw '[
{
"op": "replace",
"path": "/1/name",
"value": "announcementsIcon"
},
{
"op": "replace",
"path": "/0/name",
"value": "notificationsIcon"
}
]'

Now that we have configured our feature flag variants, we can reference them in the React code.

In NavBar.tsx, import the Announcements icon at the top of the file from Material UI, just like the Notifications icon.

Announcement as AnnouncementsIcon,

The full import line now looks like this:

import {
Menu as MenuIcon,
Notifications as NotificationsIcon,
Announcement as AnnouncementsIcon,
AttachMoney as AttachMoneyIcon,
} from "@material-ui/icons";

Unleash has a built-in React SDK hook called useVariant to retrieve variant data and perform different tasks against them.

Next, import the useVariant hook from the React SDK:

import { useFlag, useVariant } from "@unleash/proxy-client-react";

useVariant returns the name of the specific flag variant your user experiences, if it’s enabled, and if the overarching feature flag is enabled as well.

Within the NavBar component itself, we will:

  • Reference the flag that holds the variants
  • Reference the two variants to conditionally check that their enabled status for rendering

Next, we can modify what is returned in the NavBar component to account for the two variants the user might see on render.

Previously, we toggled the NotificationsIcon component based on whether or not the feature flag was enabled. We’ll take it one step further and conditionally display the icon components based on the feature flag’s variant status:

With our new variants implemented, 50% of users will see the announcements icon. The differences in the UI would look like this:

Compare icons in UI image

We have successfully configured our flag variants and implemented them into our React app for A/B testing in our development environment. Next, we can take a look at how Unleash can track the results of A/B testing and provide insights with data analytics.

Feature Flag Analytics and Reporting in React

Shipping code is one thing, but monitoring your applications is another aspect of managing code that developers have to account for. Some things to consider would be:

  • Security concerns
  • Performance metrics
  • Tracking user behavior

Unleash was built with all of these considerations in mind as part of our feature flag management approach. You can use feature flag events to send impression data to an analytics tool you choose to integrate. For example, a new feature you’ve released could be causing more autoscaling in your service resources than expected and you either can view that in your Analytics tool or get notified from a Slack integration. Our impression data gives developers a full view of the activity that could raise any alarms.

We make it easy to get our data, your application, and an analytics tool connected so you can collect, analyze, and report relevant data for your teams.

Let’s walk through how to enable impression data for the feature flag we created from the React tutorial and capture the data in our app for analytics usage.

Enable feature flag impression data

At the flag level in Unleash, navigate to the Settings view.

Edit Settings image

In the Settings view, click on the edit button. This will take us to the ‘Edit Feature toggle’ form.

Enable impression data image

Turn on the impression data and hit ‘Save’. Events will now be emitted every time the feature flag is triggered.

You can also use our API command to enable the impression data:

curl --location --request PATCH 'http://localhost:4242/api/admin/projects/default/features/newNotificationsBadge' \
--header 'Authorization: INSERT_API_KEY' \
--header 'Content-Type: application/json' \
--data-raw '[
{
"op": "replace",
"path": "/impressionData",
"value": true
}
]'

Capture impression data for flag analytics

Next, let’s configure our React app to capture the impression events that are emitted when our flag is triggered. To do this, we can import the useUnleashClient hook from the React SDK into our app. This hook allows us to interact directly with the Unleash client to listen for events and log them.

In NavBar.tsx, pull in useUnleashClient. Our updated import line should look like this:

import { useFlag, useVariant, useUnleashClient } from "@unleash/proxy-client-react";

Next, within the NavBar component itself, call useUnleashClient and wrap a useEffect hook around our function calls to grab impression data with this code snippet:

 const unleashClient = useUnleashClient();

useEffect(() => {
unleashClient.start();

unleashClient.on("ready", () => {
const enabledImpression = unleashClient.isEnabled("newNotificationsBadge");
console.log(enabledImpression);
});

unleashClient.on("impression", (impressionEvent: object) => {
console.log(impressionEvent);
// Capture the event here and pass it internal data lake or analytics provider
});
}, [unleashClient]);

This code snippet starts the Unleash client, checks that our flag is enabled, and then stores impression events for your use.

Note: We are passing in unleashClient into the dependency array in useEffect to prevent the app from unnecessarily mounting the component if the state of the data it holds has not changed.

Our flag impression data is now being logged!

In your browser, you can view the output for isEnabled:

{
"eventType": "isEnabled",
"eventId": "b4455218-4a88-43aa-8712-b99186f46548",
"context": {
"sessionId": 386689528,
"appName": "cypress-realworld-app",
"environment": "default"
},
"enabled": true,
"featureName": "newNotificationsBadge",
"impressionData": true
}

And the console.log for getVariant returns:

{
"eventType": "getVariant",
"eventId": "c41aa58b-d2c7-45cf-b668-7267f465e01a",
"context": {
"sessionId": 386689528,
"appName": "cypress-realworld-app",
"environment": "default"
},
"enabled": true,
"featureName": "newNotificationsBadge",
"impressionData": true,
"variant": "announcementsIcon"
}

You can find more information on isEnabled and getVariant in our impression data docs.

Now that the application is capturing impression events, you can configure the correct data fields and formatting to send to any analytics tool or data warehouse you use.

Application Metrics & Monitoring

Under the Metrics tab, you can see the general activity of the Cypress Real World App from our React tutorial in the development environment over different periods of time. If the app had a production environment enabled, we would also be able to view the amount of exposure and requests the app is receiving over time.

Metrics view image

Our metrics are great for understanding user traffic. You can get a better sense of:

  • What time(s) of the day or week are requests the highest?
  • Which feature flags are the most popular?

Another use case for reviewing metrics is verifying that the right users are being exposed to your feature based on how you’ve configured your strategies and/or variants.

Take a look at our Metrics API documentation to understand how it works from a code perspective.

Feature Flag Audit Logs in React

Because a Feature Flag service controls the way an application behaves in production, it can be highly important to have visibility into when changes have been made and by whom. This is especially true in highly regulated environments, such as health care, insurance, banking, and others. In these cases (or similar), you might find audit logging useful for:

  1. Organizational compliance
  2. Change control

Fortunately, this is straightforward in Unleash Enterprise.

Unleash provides the data to log any change that has happened over time, at the flag level from a global level. In conjunction with Unleash, tools like Splunk can help you combine logs and run advanced queries against them. Logs are useful for downstream data warehouses or data lakes.

In our React tutorial application, we can view Event logs to monitor the changes to flag strategies and statuses we have made throughout our examples, such as:

  • When the newNotificationsBadge flag was created
  • How the gradual rollout strategy was configured
  • When and how the variants were created and configured

Events log image

You can also retrieve event log data by using an API command below:

curl -L -X GET '<your-unleash-url>/api/admin/events/:featureName' \
-H 'Accept: application/json' \
-H 'Authorization: <API_KEY_VALUE>'

Read our documentation on Event logs and APIs to learn more.

Change Management & Feature Flag Approvals in React

Unleash makes it easy to toggle a feature. But the more users you have, the more risk with unexpected changes occurring. That’s why we implemented an approval workflow within Unleash Enterprise for making a change to a feature flag. This functions similar to GitHub's pull requests, and models a Git review workflow. You could have one or more approvals required to reduce risk of someone changing something they shouldn’t. It helps development teams to have access only to what they need. For example, you can use Unleash to track changes to your React feature flag’s configuration.

In Unleash Enterprise, we have a change request feature in your project settings to manage your feature flag approvals. When someone working in a project needs to update the status of a flag or strategy, they can follow our approval workflow to ensure that one or more team members review the change request.

Project settings

We have several statuses that indicate the stages a feature flag could be in as it progresses through the workflow. This facilitates collaboration on teams while reducing risk in environments like production. For larger scale change management, you can ensure the correct updates are made while having full visibility into who made the request for change and when.

Change requests

Learn more about how to configure your change requests and our project collaboration mode.

Flag Automation & Workflow Integration for React Apps

An advanced use case for leveraging feature flags at scale is flag automation in your development workflow. Many organizations use tools like Jira for managing projects and tracking releases across teams. Our Jira integration helps to manage feature flag lifecycles associated with your projects.

It’s common for teams to have a development phase, QA/testing and then a release to production. Let’s say the changes we’ve made in our React tutorial project need to go through a typical development workflow. The Unleash Jira plugin can connect to your Jira server or cloud to create feature flags automatically during the project creation phase. As your code progresses through development and Jira tickets are updated, the relevant flag can turn on in a development environment. The next stage could be canary deployments for testing code quality in subsequent environments to certain groups, like a QA team or beta users. The flag could be automatically turned on in QA and rollout and/or rollout to target audiences in production.

Here’s how this can be done via our API:

  1. Enable a flag.
curl -L -X POST '<your-unleash-url>/api/admin/projects/:projectId/features/:featureName/environments/:environment/on' \
-H 'Accept: application/json' \
-H 'Authorization: <API_KEY_VALUE>'

Review our API docs on flag enablement.

  1. Update a flag.
curl -L -X PUT '<your-unleash-url>/api/admin/projects/:projectId/features/:featureName' \
-H 'Content-Type: application/json' \
-H 'Accept: application/json' \
-H 'Authorization: <API_KEY_VALUE>' \
--data-raw '{
"description": "Controls disabling of the comments section in case of an incident",
"type": "kill-switch",
"stale": true,
"archived": true,
"impressionData": false
}'

And the required body field object would look like this:

{
"description": "Controls disabling of the comments section in case of an incident",
"type": "kill-switch",
"stale": true,
"archived": true,
"impressionData": false
}

Review our API docs on updating feature flags.

  1. Archive a flag.
curl -L -X DELETE '<your-unleash-url>/api/admin/projects/:projectId/features/:featureName' \
-H 'Authorization: <API_KEY_VALUE>'

Review API docs on archiving flags.

Common Usage Examples of React Feature Flags

To take full advantage of our React SDK, we’ve compiled a list of the most common functions to call in a React app.

FunctionDescriptionParametersOutput
useFlagdetermines whether or not the flag is enabledfeature flag name (string)true, false (boolean)
useVariantreturns the flag variant that the user falls intofeature flag name (string)flag and flag variant data (object)
useUnleashClientlistens to client events and performs actions against themnone
useUnleashContextretrieves information related to current flag request for you to updatenone
useFlagsretrieves a list of all flags within your projectnonean array of each flag object data
useFlagsStatusretrieves status information of al flags within your project; tells you whether they have been successfully fetched and whether there were any errorsnonean object of flag data

useFlag example

const newFeature = useFlag(“newFeatureFlag”);

// output
true

useVariant example

const variant = useVariant(“newFeatureFlag”);

// output
{ enabled: true, feature_enabled: true, name: “newVariant” }

useUnleashClient example

const unleashClient = useUnleashClient();

unleashClient.start();

unleashClient.on(“ready”, () => {
const impressionEnabled = unleashClient.isEnabled(“newFeatureFlag”);
// do something here
});

unleashClient.on(“impression”, (events) => {
console.log(events);
// do something with impression data here
});

useUnleashContext example

 const updateContext = useUnleashContext();

useEffect(() => {
// context is updated with userId
updateContext({ userId });
}, [userId]);

Read more on Unleash Context and the fields you can request and update.

useFlags example

const allFlags = useFlags();

// output
[
{
name: 'string',
enabled: true,
variant: {
name: 'string',
enabled: false,
},
impressionData: false,
},
{
name: 'string',
enabled: true,
variant: {
name: 'string',
enabled: false,
},
impressionData: false,
},
];

useFlagsStatus example

const flagsStatus = useFlagsStatus();

// output
{ flagsReady: true, flagsError: null }

Additional Examples

Many additional use cases exist for React, including deferring client start, usage with class components, React Native, and more, all of which can be found in our React SDK documentation.

- - + + \ No newline at end of file diff --git a/how-to.html b/how-to.html index 882ddd1f7b..bd0cc21f97 100644 --- a/how-to.html +++ b/how-to.html @@ -20,15 +20,15 @@ - - + + - - + + \ No newline at end of file diff --git a/how-to/api.html b/how-to/api.html index 29ffe05ce7..daa06f5f7a 100644 --- a/how-to/api.html +++ b/how-to/api.html @@ -20,15 +20,15 @@ - - + +
- - + + \ No newline at end of file diff --git a/how-to/env.html b/how-to/env.html index 04cce8dbff..c608496d06 100644 --- a/how-to/env.html +++ b/how-to/env.html @@ -20,15 +20,15 @@ - - + + - - + + \ No newline at end of file diff --git a/how-to/how-to-add-feature-flag-naming-patterns.html b/how-to/how-to-add-feature-flag-naming-patterns.html index 656f3af4cb..701168012f 100644 --- a/how-to/how-to-add-feature-flag-naming-patterns.html +++ b/how-to/how-to-add-feature-flag-naming-patterns.html @@ -20,15 +20,15 @@ - - + +

How to add feature flag naming patterns

Availability

Feature flag naming patterns is an in-development, enterprise-only feature.

This short guide will show you how to add feature flag naming patterns to a project.

Prerequisites

  • You must be using an Unleash Enterprise instance.
  • You must have permissions to edit project settings for the project you want to add feature flag naming patterns to.

Step 1: Navigate to project settings

Navigate to the project settings page for the project you want to add feature flag naming patterns to.

Step 2: Add a feature flag naming pattern

Use the "feature flag naming pattern" section of the project settings form to add a feature flag naming pattern, plus the optional example and/or description.

When you've entered you're data, save the changes.

The &quot;feature flag naming pattern&quot; part of the form. Input fields for pattern, example, and description

- - + + \ No newline at end of file diff --git a/how-to/how-to-add-sso-azure-saml.html b/how-to/how-to-add-sso-azure-saml.html index 03f8bd6930..2139cec606 100644 --- a/how-to/how-to-add-sso-azure-saml.html +++ b/how-to/how-to-add-sso-azure-saml.html @@ -20,8 +20,8 @@ - - + + @@ -32,7 +32,7 @@ Azure: The list of claims used by the SAML setup, including the optional claims for given name and surname

URLs and formats

Make sure to replace URLs with the public URL for your Unleash instance. This will require correct region prefix and the instance name.

The correct format is: https://[region].app.unleash-hosted.com/[instanceName]/auth/saml/callback

d) Sections 3 and 4: Azure AD setup details {#azure-details}

You will need some details from section 3 and 4 of the SAML setup form to configure the integration within Unleash. These details are:

  • Azure AD Identifier (from section 4)
  • Login URL (from section 4)
  • X.509 Certificate (in the Federation Metadata XML from section 3)
Section 3 contains a download link for the Federation Metadata XML file. Section 4 lists the Azure AD identifier and the login URL
Within the Federation Metadata XML file, find the `X509Certificate` tag. You'll need the content within that tag.

Step 3: Configure SAML 2.0 provider in Unleash

In order to configure SSO with SAML with your Unleash enterprise you should navigate to the Single-Sign-On configuration section and choose the "SAML 2.0" tab.

Unleash: sso-config screen

Use the values from the previous section to fill out the form:

  1. In the entity ID field, add the Azure AD identifier. It should look a little like this https://sts.windows.net/**[identifier]**.
  2. In the single sign-on URL field, add the login URL. It should look something like https://login.microsoftonline.com/**[identifier]**/saml2
  3. In the X.509 certificate field, add the content of the X509Certificate tag from the federation metadata XML.

Optionally, you may also choose to “Auto-create users”. This will make Unleash automatically create new users on the fly the first time they sign-in to Unleash with the given SSO provider (JIT). If you decide to automatically create users in Unleash you must also provide a list of valid email domains separated by commas. You must also decide which root Unleash role they will be assigned. Without this enabled you will need to manually add users to Unleash before SSO will work for their accounts and Unleash.

Unleash: SAML 2.0 filled out with entity ID, Single Sign-On URL, and X.509 certificate and auto-creating users for users with the &#39;@getunleash.ai&#39; and &#39;@getunleash.io&#39; emaiil domains.

Validate

If everything is set up correctly, you should now be able to sign in with the SAML 2.0 option. You can verify that this works by logging out of Unleash: the login screen should give you the option to sign in with SAML 2.0.

You can also test the integration in Azure by using the "test single sign on" step in the SAML setup wizard.

The SAML setup wizard contains a step that lets you test your SAML 2.0 integration. You can use this to verify that everything is configured correctly within Azure.

Group Syncing

Optionally, you can sync groups from Azure AD to Unleash to map them to groups in Unleash.

a) Add a group claim in Azure In section 2 (Attributes and claims) of the Azure SAML set-up, select the option to "Add a group claim".

Check the box to "Customize the name of the group claim" and update the "Name" to something simple, such as "groups".

Azure AD only supports sending a maximum of 150 groups in the SAML response. If you're using Azure AD and have users that are present in more than 150 groups, you'll need to add a filter in this section to the group claim to ensure that only the groups you want to sync are sent to Unleash.

Azure: section 2, attributes and claims, adding a group claim with the name &#39;group&#39;

b) Unleash SSO Setup In the Unleash Admin SSO section, enable the option to "Enable Group Syncing".

Add the same "Name" you used from the previous section (eg. "groups") as the "Group Field JSON Path".

Unleash: SAML 2.0 SSO setup, enabled group syncing with the Group Field JSON Path as &#39;groups&#39;

Note that Azure only supports sending up to 150 groups. If you have more groups than this, you can setup a filter in Azure to only send the relevant groups to Unleash.

Unleash: SAML 2.0 setup, filtering groups by name

- - + + \ No newline at end of file diff --git a/how-to/how-to-add-sso-google.html b/how-to/how-to-add-sso-google.html index f12bcfda78..10390972aa 100644 --- a/how-to/how-to-add-sso-google.html +++ b/how-to/how-to-add-sso-google.html @@ -20,15 +20,15 @@ - - + +

[Deprecated] How to add SSO with Google

Deprecation notice

Single Sign-on via the Google Authenticator provider has been removed in Unleash v5 (deprecated in v4). We recommend using OpenID Connect instead. If you're running a self hosted version of Unleash and you need to temporarily re-enable Google SSO, you can do so by setting the GOOGLE_AUTH_ENABLED environment variable to true. If you're running a hosted version of Unleash, you'll need to reach out to us and ask us to re-enable the flag. Note that this code will be removed in a future release and this is not safe to depend on.

Introduction

In this guide we will do a deep dive on the Single-Sign-On (SSO) using Google Authentication. Unleash supports other identity providers and protocols, have a look at all available Single-Sign-On options

Basic configuration

Step 1: Sign-in to Unleash

In order to configure SSO you will need to log in to the Unleash instance with a user that have "Admin" role. If you are self-hosting Unleash then a default user will be automatically created the first time you start Unleash:

  • username: admin
  • password: unleash4all

Step 2: Navigate to SSO configuration

In order to configure SSO with Google with your Unleash enterprise you should navigate to the Single-Sign-On configuration section and choose the "Google" tab.

sso-config

Step 3: Google Authentication

Navigate to https://console.developers.google.com/apis/credentials

  1. Click Create credentials
  2. Choose Oauth Client Id
  3. Choose Application Type: web application
  4. Add https://[unleash.hostname.com]/auth/google/callback as an authorized redirect URI.

You will then get a Client ID and a Client Secret that you will need in the next step.

Google OAuth: Secret

Step 4: Configure Unleash

Log in to Unleash and navigate to Admin menu -> Single-Sign-on -> Google.

First insert the Client Id and Client Secret from step 3.

You must also specify the hostname Unleash is running on. If Unleash is running on localhost you should specify the port as well (localhost:4242).

If you want to allow everyone in your organization to access Unleash, and have Unleash auto-create users you can enable this option. You should then also specify which email domains you want to allow logging in to Unleash.

Remember to click “Save” to store your settings.

Google OAuth: Secret

Step 5: Verify

Log out of Unleash and sign back in again. You should now be presented with the “SSO Authentication Option”. Click the button and follow the sign-in flow. If all goes well you should be successfully signed in to Unleash. If something is not working you can still sign-in with username and password.

Verify SSO

- - + + \ No newline at end of file diff --git a/how-to/how-to-add-sso-open-id-connect.html b/how-to/how-to-add-sso-open-id-connect.html index 5eb1ad63a0..c8cf829ba6 100644 --- a/how-to/how-to-add-sso-open-id-connect.html +++ b/how-to/how-to-add-sso-open-id-connect.html @@ -20,15 +20,15 @@ - - + +

How to add SSO with OpenID Connect

Availability

The Single-Sign-On capability is only available for customers on the Enterprise subscription. Check out the Unleash plans for details.

Introduction

In this guide we will do a deep dive on the Single-Sign-On (SSO) using the OpenID Connect protocol and connect it with Okta as IdP. Unleash supports other identity providers and protocols, have a look at all available Single-Sign-On options

Basic configuration

Step 1: Sign-in to Unleash

In order to configure SSO you will need to log in to the Unleash instance with a user that have "Admin" role. If you are self-hosting Unleash then a default user will be automatically created the first time you start Unleash:

  • username: admin
  • password: unleash4all

Step 2: Navigate to SSO configuration

Unleash enterprise supports multiple authentication providers, and we provide in depth guides for each of them. To find them navigate to "Admin" => "Single-Sign-On" section.

admin-authentication

Step 3: Okta with OpenID Connect

Open a new tab/window in your browser and sign in to your Okta account. We will need to create a new Application which will hold the settings we need for Unleash.

a) Create new Okta application

Navigate to “Admin/Applications” and click the “Add Apps” button.

Okta: Add Apps

Then click “Create Application” and choose a new “OIDC - OpenID Connect” application, and choose application type "Web Application" and click create.

Okta: Create Apps

b) Configure Application Integration

Give you application a name. And set the Sign-in redirect URI to:

https://[region].app.unleash-hosted.com/[instanceName]/auth/oidc/callback

(In a self-hosted scenario the URL must match your UNLEASH_URL configuration)

You can also configure the optional Sign-out redirect URIs: https://[region].app.unleash-hosted.com/[instanceName]/

Okta: Configure OpenID Connect

Save your new application and you will get the required details you need to configure the Unleash side of things:

Okta: Configure OpenID Connect

c) Configure OpenID Connect provider in Unleash

Navigate to Unleash and insert the details (Discover URL, Client Id and Client Secret) in to Unleash.

Pleas note that the Discover URL must be a valid URL and must include the https:// prefix. For example: https://dev-example-okta.com is a valid discovery URL.

You may also choose to “Auto-create users”. This will make Unleash automatically create new users on the fly the first time they sign-in to Unleash with the given SSO provider (JIT). If you decide to automatically create users in Unleash you must also provide a list of valid email domains. You must also decide which root Unleash role they will be assigned (Editor role will be the default).

Unleash: Configure OpenID Connect

Step 4: Verify

Log out of Unleash and sign back in again. You should now be presented with the "Sign in with OpenID Connect" option. Click the button and follow the sign-in flow. If all goes well you should be successfully signed in to Unleash.

(If something is not working you can still sign-in with username and password).

Verify SSO

Success!

- - + + \ No newline at end of file diff --git a/how-to/how-to-add-sso-saml-keycloak.html b/how-to/how-to-add-sso-saml-keycloak.html index 7169b34ab5..d5845b5c62 100644 --- a/how-to/how-to-add-sso-saml-keycloak.html +++ b/how-to/how-to-add-sso-saml-keycloak.html @@ -20,15 +20,15 @@ - - + +

How to add SSO with SAML 2.0 Keycloak

Availability

The Single-Sign-On capability is only available for customers on the Enterprise subscription. Check out the Unleash plans for details.

Introduction

In this guide we will do a deep dive on the Single-Sign-On (SSO) integration with SAML 2.0 and connect it with Keycloak as IdP. Unleash supports other identity providers and protocols, have a look at all available Single-Sign-On options

Basic configuration

Step 1: Sign-in to Unleash

In order to configure SSO you will need to log in to the Unleash instance with a user that have "Admin" role. If you are self-hosting Unleash then a default user will be automatically created the first time you start Unleash:

  • username: admin
  • password: unleash4all

Step 2: Navigate to SSO configuration

In order to configure SSO with SAML with your Unleash enterprise you should navigate to the Single-Sign-On configuration section and choose the "SAML 2.0" tab.

sso-config

Step 3: Keycloak with SAML 2.0

Open to the Keycloak dashboard and navigate to “Clients” and click “Add Client” button. Give it a unique clientId (e.g. unleash), use the “saml” protocol and specify the following SAML Endpoint:

https://<unleash.hostname.com>/auth/saml/callback

Keycloak: Add client

a) Change “Name ID format to “email” Unleash expects an email to be sent from the SSO provider so make sure Name ID format is set to email, see a). also you must give the IDP Initiated SSO URL Name, we have chosen to call it “unleash”, see 2). This gives us the Sign-on URL we will need in our Unleash configuration later.

Keycloak: step 2

b) Copy the Keycloak Entity ID an Signing key

Navigate to “Realm Settings” and open the “SAML 2.0 Identity Provider Metadata”. You will need copy the entityID (a) and the X509Certificate (B). These will be required when configuring SAML 2.0 in Unleash.

Keycloak: step 3

Step 4: Configure SAML 2.0 Authentication provider in Unleash

Go back to Unleash Admin Dashboard and navigate to Admin Menu -> Single-Sign-On -> SAML. Fill in the values captured in the step 3.

  • Entity ID (3b a)
  • Single Sing-On URL (3a b)
  • Certificate (3b b)

You may also choose to “auto create users”. This will make Unleash automatically create new users on the fly first time they sign-in to Unleash with the given SSO provider. You may also limit the auto-creation to certain email domains, shown in the example below.

Keycloak: step 4

Step 5: Validate

You have now successfully configured Unleash to use SAML 2.0 together with Keycloak as an IdP. Please note that you also must assign users to the application defined in Keycloak to actually be able to log-in to Unleash.

Try signing out of Unleash. If everything is configured correctly you should be presented with the option to sign in with SAML 2.0.

- - + + \ No newline at end of file diff --git a/how-to/how-to-add-sso-saml.html b/how-to/how-to-add-sso-saml.html index 8ff874d622..e0db218e4e 100644 --- a/how-to/how-to-add-sso-saml.html +++ b/how-to/how-to-add-sso-saml.html @@ -20,15 +20,15 @@ - - + +

How to add SSO with SAML 2.0 Okta

Availability

The Single-Sign-On capability is only available for customers on the Enterprise subscription. Check out the Unleash plans for details.

Introduction

In this guide we will do a deep dive on the Single-Sign-On (SSO) integration with SAML 2.0 and connect it with Okta as IdP. Unleash support other identity providers and protocols, have a look at all available Single-Sign-On options

Basic configuration

Step 1: Sign-in to Unleash

In order to configure SSO you will need to log in to the Unleash instance with a user that have "Admin" role. If you are self-hosting Unleash then a default user will be automatically created the first time you start Unleash:

  • username: admin
  • password: unleash4all

Step 2: Navigate to SSO configuration

In order to configure SSO with SAML with your Unleash enterprise you should navigate to the Single-Sign-On configuration section and choose the "SAML 2.0" tab.

sso-config

Step 3: Create an application in Okta

Open a new tab/window in your browser and sign in to your Okta account. We will need to create a new Application which will hold the settings we need for Unleash.

a) Navigate to “Admin -> Applications” and click the “Add Application” button.

Okta: Add Apps

b) Click “Create New App and choose a new “SAML 2.0” application and click create

Okta: Create Application

c) Configure SAML 2.0

Unleash expects an email to be sent from the SSO provider so make sure Name ID format is set to email. Also you must give the IdP Initiated SSO URL Name, we have chosen to call it “unleash-enterprise”. This gives us the Sign-on URL we will need in our Unleash configuration later.

In addition you may provide the following attributes:

  • firstName
  • lastName

(These will be used to enrich the user data in Unleash).

Okta: Configure SAML

Please make sure to replace URLs with the public URL for your Unleash instance. This will require correct region prefix and the instance name. The example above uses region="us" and instance-name="ushosted".

The correct format is: https://[region].app.unleash-hosted.com/[instanceName]/auth/saml/callback

d) Get the Okta Setup Instructions

Click the “View Setup Instructions” to get the necessary configuration required for Unleash.

Okta: Setup Instructions

Step 4: Configure SAML 2.0 provider in Unleash

Go back to Unleash Admin Dashboard and navigate to Admin Menu -> Single-Sign-On -> SAML. Fill in the values captured in the "Get the Okta Setup Instructions" step.

You may also choose to “Auto-create users”. This will make Unleash automatically create new users on the fly the first time they sign-in to Unleash with the given SSO provider (JIT). If you decide to automatically create users in Unleash you must also provide a list of valid email domains. You must also decide which root Unleash role they will be assigned (Editor role will be the default).

Unleash: SAML 2.0

Step 5: Validate

You have now successfully configured Unleash to use SAML 2.0 together with Okta as an IdP. Please note that you also must assign users to the application defined in Okta to actually be able to log-in to Unleash.

Try signing out of Unleash. If everything is configured correctly you should be presented with the option to sign in with SAML 2.0.

Single-Sign-Out

Available from Unleash Enterprise 4.1.0

You may also configure Unleash to to perform Single-Sign-Out. By enabling single-sign-out Unleash will redirect the user back to IdP as part of the sign-out process. You may optionally also sign the sign-out request (required by multiple IdP's such as Okta).

Step 1: Generate private key & public certificate

(This step is only required if you intend to sign the sign-out requests).

Before you can configure single-sign-out support with Okta you are required to generate a Private Key together with a public certificate for that key. We recommend to use SHA256 certificates.

To create a public certificate and private key pair, use the proceeding commands. They work in Linux® and Mac® terminals.

openssl genrsa -out private.pem 2048
openssl req -new -x509 -sha256 -key private.pem -out cert.pem -days 1095

Answer the promoted questions, and when you complete all the steps you should end up with two files:

  • private.pem - Private certificate, required by Unleash in order to sign requests.
  • cert.pem - Public certificate, required by the IdP in order to validate requests from Unleash.

Step 2: Configure sign-out url in Okta

Login in to Okta and navigate to your Applications. Select the "Unleash" application you created, click on "General" and then "Edit SAML Settings".

SAML 2.0 Okta edit



Next, navigate to "Configure SAML" and click "show Advanced Settings" and check the Enable Single Logout option.



SAML 2.0 Okta sing-out config



Please make sure to replace URLs with the public URL for your Unleash instance. This will require correct region prefix and the instance name. The example above uses region="us" and instance-name="ushosted".

The correct format is: https://[region].app.unleash-hosted.com/[instanceName]/auth/saml/logout/done

You need to fill out the following options:

  • Single Logout Url: https://[region].app.unleash-hosted.com/[instanceName]/auth/saml/logout/done
  • SP Issuer: https://[region].app.unleash-hosted.com/[instanceName]

Next upload the public Certificate you generated in the previous step (cert.pem) and save the Okta SAML settings. Upon completion of this step you should be provided with the ability to view setup instructions and now you should be provided with a "Identity Provider Single Logout URL"

SAML 2.0 Okta sing-out url

Step 3: Configure Single-Sign-Out in Unleash

Go back to Unleash Admin Dashboard and navigate to Admin Menu -> Single-Sign-On -> SAML. Fill in the values captured in the "Single Logout URL" from Okta.

In the "Service Provide X.509 Certificate" field you should insert the value of your private key (private-pem). This is required in order to make Unleash able to sign logout requests.

SAML 2.0 Okta sing-out config

After you save these settings users will now be redirected to your IdP (Okta) and back to Unleash again after successfully signing out.

- - + + \ No newline at end of file diff --git a/how-to/how-to-add-strategy-constraints.html b/how-to/how-to-add-strategy-constraints.html index 6b1a954a19..478ab66653 100644 --- a/how-to/how-to-add-strategy-constraints.html +++ b/how-to/how-to-add-strategy-constraints.html @@ -20,15 +20,15 @@ - - + +

How to add strategy constraints

Availability

Before Unleash 4.16, strategy constraints were only available to Unleash Pro and Enterprise users. From 4.16 onwards, they're available to everyone.

This guide shows you how to add strategy constraints to your feature toggles via the admin UI. For information on how to interact with strategy constraints from an Unleash client SDK, visit the specific SDKs documentation or see the relevant section in the strategy constraints documentation.

Prerequisites

You'll need to have an existing feature toggle with a defined strategy to add a constraint. The rest of this guide assumes you have a specific strategy that you're working with.

Step 1: Open the constraints menu

On the strategy you're working with, find and select the "edit strategy" button.

A feature toggle with one strategy. The &quot;edit strategy&quot; button is highlighted.

On the "edit strategy" screen, select the "add constraint" button to open the constraints menu.

A feature toggle strategy view showing a button labeled with add constraints.

Step 2: Add and configure the constraint

Refer to the constraint structure section of the strategy constraints reference for a thorough explanation of the constraint fields.

  1. From the "Context Field" dropdown, select the context field you would like to constrain the strategy on and choose the constraint operator you want.
  2. Define the values to use for this constraint. The operator you selected decides whether you can define one or multiple values and what format they can have.
  3. Save the constraint first.

A strategy constraint form with a constraint set to &quot;useid&quot;. The &quot;values&quot; input is a text input containing the values &quot;41&quot;, &quot;932&quot;, &quot;822&quot;.

Step 3: Save the strategy

A feature toggle strategy view showing a button at the end of the form labeled with save strategy.

How to update existing constraints

To update an existing constraint, find the constraint in the "edit strategy" screen and use the constraint's "edit" button.

A strategy form showing an existing constraint with existing values and 2 buttons, the &quot;edit&quot; button is highlighted.

- - + + \ No newline at end of file diff --git a/how-to/how-to-add-users-to-unleash.html b/how-to/how-to-add-users-to-unleash.html index cb44b29996..49e4427513 100644 --- a/how-to/how-to-add-users-to-unleash.html +++ b/how-to/how-to-add-users-to-unleash.html @@ -20,8 +20,8 @@ - - + + @@ -30,7 +30,7 @@

How to add new users to your Unleash instance

This feature was introduced in Unleash v4 for Unleash Open-Source.

You can add new users to Unleash in Admin > Users.

  1. From the top-line menu – click on the “Settings Wheel” then click on “Users”. A visual representation of the current step: the Unleash Admin UI with the steps highlighted.
  1. To add a new user to your Unleash instance, use the "new user" button: The Unleash users page with the &#39;add new user&#39; button being pointed to.

  2. Fill out the required fields in the "create user" form. Refer to the predefined roles overview for more information on roles.

    A form titled &quot;Add team member&quot;. It has the fields &quot;full name&quot;, &quot;email&quot;, and &quot;role&quot;. The role field is a radio button set with roles called &quot;admin&quot;, &quot;editor&quot;, and &quot;viewer&quot;.

    If you have configured an email server the user will receive the invite link in her inbox, otherwise you should share the magic invite link to Unleash presented in the confirmation dialogue.

- - + + \ No newline at end of file diff --git a/how-to/how-to-capture-impression-data.html b/how-to/how-to-capture-impression-data.html index 964f4d0701..20a7d9bcc0 100644 --- a/how-to/how-to-capture-impression-data.html +++ b/how-to/how-to-capture-impression-data.html @@ -20,8 +20,8 @@ - - + + @@ -32,7 +32,7 @@ The steps below will use extracts from said documentation.

After initializing the library, you'll probably also want to identify the current user, so that events can be correlated properly:

Identify the user.
posthog.identify(userId);

Capture and transform the event

Attach an event listener to Unleash that listens for "impression" events. Inside the listener, transform the event data to the format you want and send it to your analytics service.

Capture, transform, and send impression data.
// listen for impression events
unleash.on("impression", (event) => {

// capture and transform events
posthog.capture(event.eventType, {
...event.context,
distinct_id: event.context?.userId,
featureName: event.featureName,
enabled: event.enabled,
variant: event.variant,
});

});

Posthog expects an object with a distinct_id property that it uses to correlate data. Additionally, you can add whatever properties you want, such as the feature toggle name, its state and/or variant, or the whole Unleash context. The posthog.capture method sends the event data to your Posthog instance.

- - + + \ No newline at end of file diff --git a/how-to/how-to-clone-environments.html b/how-to/how-to-clone-environments.html index 48ff3a0a2c..1b4b37e719 100644 --- a/how-to/how-to-clone-environments.html +++ b/how-to/how-to-clone-environments.html @@ -20,15 +20,15 @@ - - + +

How to clone environments

availability

Environment cloning was made available in Unleash 4.19.

Environment cloning enables Unleash admins to duplicate existing environments, including all feature toggles strategies and their state.

Step 1: Navigate to the environments page

Navigate to the Environments page in the admin UI (available at the URL /environments). Use the navigation menu item "Configure" and select "Environments".

The admin UI navigation &quot;Configure&quot; submenu with the Environments item highlighted.

Step 2: Select an environment to clone

Select an environment to clone. On the right side, open the actions submenu and select "Clone".

The &quot;production&quot; environment actions submenu with the Clone option highlighted.

Step 3: Fill in the clone environment form

Give your new environment a name. The name must be unique and cannot be the same as the original environment. The name is pre-filled with a suggestion, but you can change it to whatever you like.

Select an environment type, which projects should have their environment configuration cloned, and whether to keep the existing user permissions for the new environment.

The clone environment form filled with some example data, and the Clone environment button highlighted at the bottom.

You can optionally generate an API token for the new environment right away. Select which projects the token should have access to, and the token will be generated when you submit the form.

The clone environment form with the API Token section highlighted and the Generate an API token now option selected

The token details with the &quot;Copy Token&quot; element highlighted.

You can always create API tokens for the new environment by following the Generating an API token guide.

- - + + \ No newline at end of file diff --git a/how-to/how-to-create-and-assign-custom-project-roles.html b/how-to/how-to-create-and-assign-custom-project-roles.html index 2778111441..96a15c447c 100644 --- a/how-to/how-to-create-and-assign-custom-project-roles.html +++ b/how-to/how-to-create-and-assign-custom-project-roles.html @@ -20,8 +20,8 @@ - - + + @@ -36,7 +36,7 @@ A project overview with the &#39;access&#39; tab highlighted.
  • This step depends on whether the user has already been added to the project or not:
    • If the user has already been added to the project, click on the edit icon coresponding with its line and from the overlay that will show up select the new role you want to assign it from the dropdown and save the changes. A list of users with access to the current project. To the right of each user is a dropdown input labeled role.
    • If the user hasn't been added to the project, add them using the button 'Assing user/group'. From the overlay that will show up select the user, assign it a role and save the changes. Now you should be able to see the new user in the table. Adding a user to a project. The add user form is filled out with data for an &quot;Alexis&quot;. The Role input is open and the custom &quot;Developer&quot; role is highlighted.
  • - - + + \ No newline at end of file diff --git a/how-to/how-to-create-and-assign-custom-root-roles.html b/how-to/how-to-create-and-assign-custom-root-roles.html index 0947a1827e..da09aff931 100644 --- a/how-to/how-to-create-and-assign-custom-root-roles.html +++ b/how-to/how-to-create-and-assign-custom-root-roles.html @@ -20,15 +20,15 @@ - - + +

    How to create and assign custom root roles

    availability

    Custom root roles were introduced in Unleash 5.4 and are only available in Unleash Enterprise.

    This guide takes you through how to create and assign custom root roles. Custom root roles allow you to fine-tune access rights and permissions to root resources in your Unleash instance.

    Creating custom root roles

    Step 1: Navigate to the custom root roles page

    Navigate to the roles page in the admin UI (available at the URL /admin/roles). Use the settings button in the navigation menu and select "roles".

    The admin UI admin menu with the Roles item highlighted.

    Step 2: Click the "new root role" button.

    Use the "new root role" button to open the "new root role" form.

    The &quot;root roles&quot; table with the &quot;new root role&quot; button highlighted.

    Step 3: Fill in the root role form

    Give the root role a name, a description, and the set of permissions you'd like it to have. For a full overview of all the options, consult the custom root roles reference documentation.

    The root role form filled with some example data, and the &quot;add role&quot; button highlighted at the bottom.

    Assigning custom root roles

    You can assign custom root roles just like you would assign any other predefined root role. Root roles can be assigned to users, service accounts, and groups.

    - - + + \ No newline at end of file diff --git a/how-to/how-to-create-and-display-banners.html b/how-to/how-to-create-and-display-banners.html index 2adcededdd..7e1c5f465e 100644 --- a/how-to/how-to-create-and-display-banners.html +++ b/how-to/how-to-create-and-display-banners.html @@ -20,15 +20,15 @@ - - + +

    How to create and display banners

    Availability

    Banners were introduced as a beta feature in Unleash 5.6 and are only available in Unleash Enterprise. We plan to make this feature generally available to all Enterprise users in Unleash 5.7.

    This guide takes you through how to create and display banners.

    Creating banners

    Step 1: Navigate to the banners page

    Navigate to the banners page in the admin UI (available at the URL /admin/banners). Use the settings button in the navigation menu and select "banners".

    The admin UI admin menu with the Banners item highlighted.

    Step 2: Use the "new banner" button

    Use the "new banner" button to open the "new banner" form.

    The &quot;banners&quot; table with the &quot;new banner&quot; button highlighted.

    Step 3: Fill in the banner form

    Choose whether the banner should be enabled right away. If enabled, the banner will be visible to all users in your Unleash instance. Select the banner type, icon, and write the message that you'd like to see displayed on the banner. The message and dialog fields support Markdown. Optionally, you can also configure a banner action for user interactivity. For a full overview of all the banner options, consult the banners reference documentation.

    You'll be able to preview the banner at the top as you fill in the form.

    Once you're satisfied, use the "add banner" button to create the banner.

    The banner form filled with some example data, and the &quot;add banner&quot; button highlighted at the bottom.

    Displaying banners

    You can choose whether a banner is currently displayed to all users of your Unleash instance by toggling the "enabled" switch on the banner table.

    Alternatively, you can edit the banner by using the "edit" button on the banner table and then toggle the "banner status" switch.

    The &quot;banners&quot; table with some example data, and the &quot;enable&quot; switch highlighted.

    - - + + \ No newline at end of file diff --git a/how-to/how-to-create-and-manage-user-groups.html b/how-to/how-to-create-and-manage-user-groups.html index 8f26681db9..471f119172 100644 --- a/how-to/how-to-create-and-manage-user-groups.html +++ b/how-to/how-to-create-and-manage-user-groups.html @@ -20,15 +20,15 @@ - - + +

    How to create and manage user groups

    availability

    User groups are available to Unleash Enterprise users since Unleash 4.14.

    This guide takes you through how to use user groups to manage permissions on your projects. User groups allow you to manage large groups of users more easily than assigning roles directly to those users. Refer to the section on user groups in the RBAC documentation for more information.

    Creating user groups

    1. Navigate to groups by using the admin menu (the gear icon) and selecting the groups option.

    The Unleash Admin UI with the steps highlighted to navigate to groups.

    1. Navigate to new group.

    The groups screen with the new group button highlighted.

    1. Give the group a name, an optional description, an optional root role, and select the users you'd like to be in the group.

    The new group screen with the users drop down open and highlighted.

    1. Review the details of the group and save them if you're happy.

    The new group screen with the users selected and the save button highlighted.

    Managing users within a group

    1. Navigate to groups by using the admin menu (the gear icon) and selecting the groups option.

    The Unleash Admin UI with the steps highlighted to navigate to groups.

    1. Select the card of the group you want to edit.

    The manage groups with a pointer to a group card.

    1. Remove users by using the remove user button (displayed as a bin).

    The manage group page with the remove user button highlighted.

    1. Confirm the remove.

    The manage group page with the confirm user removal dialog shown.

    1. Add users by selecting the add button.

    The groups page shown with the add user button highlighted.

    1. Find the user you'd like to add to the group and select them.

    The groups page shown with a user selected.

    1. Review the group users and save when you're happy.

    The edit groups page shown with the save button highlighted.

    Assigning groups to projects

    1. Navigate to projects

    The landing page with the projects navigation link highlighted.

    1. Select the project you want to manage.

    The projects page with a project highlighted.

    1. Navigate to the access tab and then use the assign user/group button.

    The project page with the access tab and assign button highlighted.

    1. Find your group in the drop down.

    The access sidepane for a project with a group selected.

    1. Select the role that the group should have in this project. You can review the list of permissions that the group users will gain by having this role before confirming.

    The access sidepane for a project with a role selected.

    - - + + \ No newline at end of file diff --git a/how-to/how-to-create-api-tokens.html b/how-to/how-to-create-api-tokens.html index d0dd5b8cfc..12132eca59 100644 --- a/how-to/how-to-create-api-tokens.html +++ b/how-to/how-to-create-api-tokens.html @@ -20,15 +20,15 @@ - - + +

    How to create API Tokens

    Permissions

    Creating API tokens requires you to have the CREATE_API_TOKEN permission. This means that, as of now, only instance admins can create API tokens.

    All users can see tokens with CLIENT level access, but only instance admins can see tokens with ADMIN level access.

    Unleash SDKs use API tokens to authenticate to the Unleash API. Unleash supports different types of API tokens, each with different levels of access and privileges. Refer to the API tokens and client keys article for complete overview of the different token types.

    Step 1: Navigate to the API token creation form

    Navigate to the API access page in the admin UI (available at the URL /admin/api). Use the navigation menu item "Configure" and select "API access".

    The admin UI navigation &quot;Configure&quot; submenu with the API access item highlighted.

    On the API access page, use the "New API token" button to navigate to the token creation form.

    The API access page with the &quot;New API token&quot; button highlighted.

    Step 2: Fill in the API token form

    API token creation form.

    Fill in the form with the desired values for the token you want to create. Refer to the API tokens and client keys article for a detailed explanation of what all the fields mean.

    Using API tokens

    When you have created the API token, it will be listed on the API access page. If you have the required permissions to see the token, you can copy it for easy use in your applications.

    API access token table with a &quot;copy token&quot; button highlighted.

    - - + + \ No newline at end of file diff --git a/how-to/how-to-create-feature-toggles.html b/how-to/how-to-create-feature-toggles.html index df40f4c800..dcd29b3489 100644 --- a/how-to/how-to-create-feature-toggles.html +++ b/how-to/how-to-create-feature-toggles.html @@ -20,15 +20,15 @@ - - + +

    How to create a feature flag

    Feature flags or feature toggles?

    This document uses feature flags and feature toggles interchangeably. Some people prefer flag; others prefer toggle. We use both - they are synonyms for us.

    Feature Flags (or Feature toggles in the UI) are the foundation of Unleash. They are at the core of everything we do and are a fundamental building block in any feature management system. This guide shows you how to create feature flags in Unleash and how to add any optional constraints, segments, variants, and more. Links to learn more about these concepts will be scattered throughout the text.

    You can perform every action both via the UI and the admin API. This guide includes screenshots to highlight the relevant UI controls and links to the relevant API methods for each step.

    This guide is split into three sections:

    1. Prerequisites: you need these before you can create a flag.
    2. Required steps: all the required steps to create a flag and activate it in production.
    3. Optional steps: optional steps you can take to further target and configure your feature flag and its audience.

    Prerequisites

    To perform all the steps in this guide, you will need:

    • A running Unleash instance
    • A project to hold the flag
    • A user with an editor or admin role OR a user with the following permissions inside the target project:
    roles

    Refer to the documentation on role-based access control for more information about the available roles and their permissions.

    Required steps

    This section takes you through the required steps to create and activate a feature toggle. It assumes that you have all the prerequisites from the previous section done.

    Step 1: Create a toggle

    API: create a toggle

    Use the Admin API endpoint for creating a feature toggle. The payload accepts all the same fields as the Admin UI form. The Admin UI also displays the corresponding cURL command when you use the form.

    In the project that you want to create the toggle in, use the "new feature toggle" button and fill the form out with your desired configuration. Refer to the feature toggle reference documentation for the full list of configuration options and explanations.

    Step 2: Add a strategy

    API: Add a strategy

    Use the API for adding a strategy to a feature toggle. You can find the configuration options for each strategy in the activation strategy reference documentation.

    Decide which environment you want to enable the toggle in. Select that environment and add an activation strategy. Different activation strategies will act differently as described in the activation strategy documentation. The configuration for each strategy differs accordingly. After selecting a strategy, complete the steps to configure it.

    Step 3: Enable the toggle

    API: Enable a toggle

    Use the API for enabling an environment for a toggle and specify the environment you'd like to enable.

    Use the environments toggles to switch on the environment that you chose above. Depending on the activation strategy you added in the previous step, the toggle should now evaluate to true or false depending on the Unleash context you provide it.

    Optional steps

    These optional steps allow you to further configure your feature toggles to add optional payloads, variants for A/B testing, more detailed user targeting and exceptions/overrides.

    Add constraints and segmentation

    Constraints and segmentation allow you to set filters on your strategies, so that they will only be evaluated for users and applications that match the specified preconditions. Refer to the strategy constraints and segments reference documentation for more information.

    To add constraints and segmentation, use the "edit strategy" button for the desired strategy.

    Constraints

    info

    Constraints aren't fixed and can be changed later to further narrow your audience. You can add them either when you add a strategy to a toggle or at any point thereafter.

    In the strategy configuration screen for the strategy that you want to configure, use the "add constraint" button to add a strategy constraint.

    Segments

    info

    This can be done after you have created a strategy.

    API: add segments

    Use the API for adding segments to a strategy to add segments to your strategy.

    In the strategy configuration screen for the strategy that you want to configure, use the "select segments" dropdown to add segments.

    Add variants

    info

    This can be done at any point after you've created your toggle.

    API: add variants

    Use the update variants endpoint. The payload should be your desired variant configuration.

    Variants give you the ability to further target your users and split them into groups of your choosing, such as for A/B testing. On the toggle overview page, select the variants tab. Use the "new variant" button to add the variants that you want.


    1. Prior to Unleash 4.21, "create/edit variants" was a project-level permission instead of an environment-level permission.
    - - + + \ No newline at end of file diff --git a/how-to/how-to-create-personal-access-tokens.html b/how-to/how-to-create-personal-access-tokens.html index cd835436d5..83b1d1cefe 100644 --- a/how-to/how-to-create-personal-access-tokens.html +++ b/how-to/how-to-create-personal-access-tokens.html @@ -20,15 +20,15 @@ - - + +

    How to create Personal Access Tokens

    availability

    Personal access tokens are planned to be released in Unleash 4.17.

    Personal access tokens are a tool to enable a user to use the Admin API as themselves with their own set of permissions, rather than using an admin token. See how to use the Admin API for more information.

    Step 1: Navigate to the personal access tokens page

    Open the user profile pane in the admin UI and select the view user profile menu item (available at the URL /profile).

    The admin UI navigation &quot;user profile&quot; menu with the view user profile menu item selected.

    Select the "Personal Access Tokens" menu item.

    The user profile page with the &quot;Personal Access Tokens&quot; menu item highlighted.

    Step 2: Navigate to New Token

    Navigate to "New Token".

    The New Token element highlighted.

    Step 3: Fill in the create personal API token form and create it

    Give your token a description and optionally set an expiry date. By default the expiry date is set to 30 days.

    The New Token form with the description text box and create element highlighted.

    Step 4: Save your token

    Once your new token is created, the popup will display the new token details. You must save your token somewhere outside of Unleash, you won't be able to access it again.

    The token created popup with the &quot;Copy Token&quot; element highlighted.

    Your personal access token can now be used in place of an admin token.

    - - + + \ No newline at end of file diff --git a/how-to/how-to-create-project-api-tokens.html b/how-to/how-to-create-project-api-tokens.html index a9951c3ad7..94a349defe 100644 --- a/how-to/how-to-create-project-api-tokens.html +++ b/how-to/how-to-create-project-api-tokens.html @@ -20,15 +20,15 @@ - - + +

    How to create Project API Tokens

    Permissions

    Creating Project API tokens requires you to have the CREATE_PROJECT_API_TOKEN permission.

    Unleash SDKs use API tokens to authenticate to the Unleash API. Unleash supports different types of API tokens, each with different levels of access and privileges. Refer to the API tokens and client keys article for complete overview of the different token types.

    Step 1: Navigate to the API token creation form

    Navigate to the Project API access page in the admin UI (available at the URL /admin/projects/<project-name>/api-access). Use the navigation tab "Project Settings" and select "API access".

    The Project overview &quot;Project settings&quot; submenu with the API access item highlighted.

    On the API access page, use the "New API token" button to navigate to the token creation form.

    The Project API access page with the &quot;New API token&quot; button highlighted.

    Step 2: Fill in the API token form

    Project API token creation form.

    Fill in the form with the desired values for the token you want to create. Refer to the API tokens and client keys article for a detailed explanation of what all the fields mean.

    Using Project API tokens

    When you have created the Project API token, it will be listed on the Project API access page. If you have the required permissions to see the token (READ_PROJECT_API_TOKEN), you can copy it for easy use in your applications.

    API access token table with a &quot;copy token&quot; button highlighted.

    - - + + \ No newline at end of file diff --git a/how-to/how-to-create-service-accounts.html b/how-to/how-to-create-service-accounts.html index 92a1683e80..ca8dba01f4 100644 --- a/how-to/how-to-create-service-accounts.html +++ b/how-to/how-to-create-service-accounts.html @@ -20,15 +20,15 @@ - - + +

    How to create service accounts

    availability

    Service accounts is an enterprise feature available from Unleash 4.21 onwards.

    Service accounts enable Unleash admins to create accounts that act as users and respect the same set of permissions, however they do not have a password and cannot log in to the Unleash UI. Instead, they are intended to be used to access the Unleash API programatically.

    Step 1: Navigate to the service accounts page

    Navigate to the service accounts page in the admin UI (available at the URL /admin/service-accounts). Use the settings button in the navigation menu and select "service accounts".

    The admin UI navigation settings submenu with the Service accounts item highlighted.

    Step 2: Click the "new service account" button

    Use the "new service account" button to open the "new service account" form.

    The &quot;service accounts&quot; table with the &quot;new service account&quot; button highlighted.

    Step 3: Fill in the service account form

    Give your new service account a name. After leaving the name field, the username field is pre-filled with a suggestion based on the name you entered, but you can change it to whatever you like.

    Select a root role for your service account, which will define what your new service account will be allowed to do. The roles that you can assign to service accounts are the same ones that are available for regular users.

    The service account form filled with some example data, and the &quot;add service account&quot; button highlighted at the bottom.

    You can optionally generate a token for the new service account right away. Give your token a description and optionally set an expiry date. By default the expiry date is set to 30 days. The token will be generated when you submit the form.

    The service account form with the token section highlighted and the &quot;generate a token now&quot; option selected

    The token details with the &quot;copy token&quot; element highlighted.

    Managing service account tokens

    You can later manage service account tokens by editing the respective service account. This allows you to add new tokens to that service account or delete existing ones.

    The service account table with the &quot;edit&quot; button highlighted.

    The service account form with the tokens table highlighted

    - - + + \ No newline at end of file diff --git a/how-to/how-to-define-custom-context-fields.html b/how-to/how-to-define-custom-context-fields.html index b8de05bde8..c708a7d074 100644 --- a/how-to/how-to-define-custom-context-fields.html +++ b/how-to/how-to-define-custom-context-fields.html @@ -20,15 +20,15 @@ - - + +

    How to define custom context fields

    Availability

    Before Unleash 4.16, custom context fields were only available to Unleash Pro and Enterprise users. From 4.16 onwards, they're available to everyone. They were introduced in Unleash 3.2.28.

    This guide shows you how to create custom context field for the Unleash Context. You can use custom context fields for strategy constraints and for custom stickiness calculations. If there are standard Unleash Context fields missing from the context fields page, you can use the same steps to add them too.

    Step 1: Navigate to the context field creation form

    In the Unleash Admin UI, navigate to the context fields page:

    1. Click the "Configure" button in the top menu to open the configuration dropdown menu.

    2. Click the "Context fields" menu item.

      A visual representation of the tutorial steps described in the preceding paragraph, showing the interaction points in the admin UI in order.

    3. On the context fields page, click the "add new context field" button.

      The &quot;context fields&quot; page with the &quot;add new context field&quot; button highlighted.

    Step 2: Define the new context field

    Define the custom context field by filling out the form. You must at least the field a unique name. Everything else is optional. Refer to the custom context field reference guide for a full overview of the parameters and their functions and requirements.

    When you are satisfied with the context field's values, use the "create" button to submit the form and save the context field.

    A &quot;create context field&quot; form. It contains data for a custom context field called &quot;region&quot;. Its description is &quot;allows you to constrain on specific regions&quot; and its legal values are &quot;Africa&quot;, &quot;Asia&quot;, &quot;Europe&quot;, and &quot;North America&quot;. Its custom stickiness value is not shown.

    - - + + \ No newline at end of file diff --git a/how-to/how-to-download-login-history.html b/how-to/how-to-download-login-history.html index 616ee2be0b..4a1f1c054f 100644 --- a/how-to/how-to-download-login-history.html +++ b/how-to/how-to-download-login-history.html @@ -20,15 +20,15 @@ - - + +

    How to download your login history

    availability

    Login history is an enterprise feature available from Unleash 4.22 onwards.

    Login history enables Unleash admins to audit login events and their respective information, including whether they were successful or not.

    Step 1: Navigate to the login history page

    Navigate to the login history page in the admin UI (available at the URL /admin/logins). Use the settings button in the navigation menu and select "login history".

    The admin UI navigation settings submenu with the login history item highlighted.

    Step 2: Click the "Download login history" button

    Use the "download login history" button to proceed with the download of the login history as CSV.

    The login history table with the &quot;download login history&quot; button highlighted.

    - - + + \ No newline at end of file diff --git a/how-to/how-to-enable-openapi.html b/how-to/how-to-enable-openapi.html index 2e923fb30e..e7b2545ef0 100644 --- a/how-to/how-to-enable-openapi.html +++ b/how-to/how-to-enable-openapi.html @@ -20,8 +20,8 @@ - - + + @@ -29,7 +29,7 @@

    How to enable OpenAPI and the Swagger UI

    Availability

    OpenAPI schemas were introduced in v4.13.0 of Unleash and v0.10.0 of the Unleash proxy. Since v5.2.0 Unleash has OpenAPI enabled by default.

    Both Unleash and the Unleash proxy have included OpenAPI schemas and Swagger UIs for their APIs. The schemas can be used to get an overview over all API operations and to generate API clients using OpenAPI client generators. The Swagger UI lets you see and try out all the available API operations directly in your browser.

    To enable the OpenAPI documentation and the Swagger UI, you must start Unleash or the proxy with the correct configuration option. The following section shows you how. The methods are the same for both Unleash and the Unleash proxy, so the steps described in the next section will work for either.

    Location of the OpenAPI spec

    Once you enable OpenAPI, you can find the specification in JSON format at /docs/openapi.json and the swagger UI at /docs/openapi.

    For instance, if you're running the Unleash server locally at http://localhost:4242, then

    • the JSON specification will be at http://localhost:4242/docs/openapi.json
    • the Swagger UI will be at http://localhost:4242/docs/openapi

    Similarly, if you're running the Unleash proxy locally at http://localhost:3000 (so that the proxy endpoint is at http://localhost:3000/proxy), then

    • the JSON specification will be at http://localhost:3000/docs/openapi.json
    • the Swagger UI will be at http://localhost:3000/docs/openapi

    Step 1: enable OpenAPI

    The OpenAPI spec and the Swagger UI can be turned on either via environment variables or via configuration options. Configuration options take precedence over environment variables.

    If you are using Unleash v5.2.0, OpenAPI is enabled by default. You still need to enable it for Unleash proxy.

    Enable OpenAPI via environment variables

    To turn on OpenAPI via environment variables, set the ENABLE_OAS to true in the environment you're running the server in.

    Enable OpenAPI via an environment variable
    export ENABLE_OAS=true

    Enable OpenAPI via configuration options

    The configuration option for enabling OpenAPI and the swagger UI is enableOAS. Set this option to true.

    The following examples have been shortened to show the only the relevant configuration options. For more detailed instructions on how to run Unleash or the proxy, refer to how to run the Unleash proxy or the section on running Unleash via Node.js from the deployment section of the documentation.

    Enable OpenAPI for Unleash via configuration option
    const unleash = require('unleash-server');

    unleash
    .start({
    // ... Other options elided for brevity
    enableOAS: true,
    })
    .then((unleash) => {
    console.log(
    `Unleash started on http://localhost:${unleash.app.get('port')}`,
    );
    });
    - - + + \ No newline at end of file diff --git a/how-to/how-to-environment-import-export.html b/how-to/how-to-environment-import-export.html index 62e11e8794..bc86ace5ff 100644 --- a/how-to/how-to-environment-import-export.html +++ b/how-to/how-to-environment-import-export.html @@ -20,8 +20,8 @@ - - + + @@ -29,7 +29,7 @@

    Environment Import & Export

    Availability

    The environment import and export first appeared in Unleash 4.22.0.

    Environment export and import lets you copy feature configurations from one environment to another and even copy features from one Unleash instance to another.

    When exporting, you select a set of features and one environment to export the configuration from. The environment must be the same for all features.

    Then, when you import, you must select one environment and one project to import into. All features are imported into that project in that environment. If Unleash is unable to import the configuration safely, it will tell you why the import failed and what you need to do fix it (read more about import requirements.

    Import & Export items

    When you export features, the export will contain both feature-specific configuration and global configuration.

    On the project-level these items are exported:

    On the environment-level, these items are exported for the chosen environment:

    Additionally, these global configuration items are exported:

    Importantly, while references to segments are exported, the segments themselves are not exported. Consult the import requirements section for more information.

    Export

    You can export features either from a project search or the global feature search. Use the search functionality to narrow the results to the list of features you want to export and use the export button to select environment. All features included in your search results will be included in the export.

    Import

    Import is a 3 stage process designed to be efficient and error-resistant.

    Import stages

    • upload - you can upload previously exported JSON file or copy-paste export data from the exported JSON file into the code editor
    • validation - you will get feedback on any errors or warnings before you do the actual import. This makes sure your feature flags configurations. You will not be able to finish the import if you have errors. Warnings don't stop you from importing.
    • import - the actual import that creates a new configuration in the target environment or creates a [change request]/reference/change-requests.md) when the environment has change requests enabled

    The import UI. It has three stages: import, file, validate configuration, finish import.

    Import requirements

    Unleash will reject an import if it discovers conflicts between the data to be imported and what already exists on the Unleash instance. The import tool will tell you about why an import is rejected in these cases.

    The following sections describe requirements that must be met for Unleash not to stop the import job.

    Context fields

    When you import a custom context field with legal values defined:

    If a custom context field with the same name already exists in the target instance, then the pre-existing context field must have at least those legal values defined. In other words, the imported context field legal values must be a subset of the already existing context field's legal values.

    When you import custom context fields without legal values or custom context fields that don't already exist in the target instance, then there are no requirements.

    Custom context fields that don't already exist in the target instance will be created upon import.

    Segments

    If your import has segments, then a segment with the same name must already exist in the Unleash instance you are trying to import into. Only the name must be the same: the constraints in the segment can be different.

    Dependencies

    If your import has a parent feature dependency, then the parent feature must already exist in the target Unleash instance or be part of the import data. The existing feature is identified by name.

    Custom strategies

    If your import contains custom strategies, then custom strategies with the same names must already exist in the target Unleash instance. The custom strategy definitions (including strategy parameters) that exist in the target instance do not otherwise need to match the custom strategy definitions in the instance the import came from.

    Existing features

    If any of the features you import already exist in the target Unleash instance, then they must exist within the project you're importing into. If any features you are attempting to import exist in a different project on the target instance, the import will be rejected.

    Pending change requests

    The project and environment you are importing into must not have any pending change requests.

    Import warnings

    The import validation system will warn you about potential problems it finds in the import data. These warnings do not prevent you from importing the data, but you should read them carefully to ensure that Unleash does as you intend.

    The following sections list things that the import tool will warn you about.

    Archived features

    The import tool will not import any features that have already been archived on the target Unleash instance. Because features are identified by their name, that means that if a feature called feature-a has been created and archived on the target Unleash instance, then a feature with the same name (feature-a) will not be imported.

    If you permanently delete the archived feature-a from the target instance, then the new feature-a (in the import data) will be imported.

    Custom strategies

    Unleash will verify that any custom strategies you are trying to import have already been defined on the target instance. However, it only does this verification by name. It does not validate that the definitions are otherwise the same or that they have the same configuration parameters.

    Required permissions

    To import features, you will always require the update feature toggles permission. Additionally, depending on the actions your import job would trigger, you may also require any of the following permissions:

    • Create feature toggles: when the import would create new features
    • Update tag types: when the import would create new tag types
    • Create context fields: when the import would create new context fields
    • Create activation strategies: when the import would add activation strategies to a feature and change requests are disabled
    • Delete activation strategies: when import would remove existing activation strategies from a feature and change requests are disabled
    • Update variants: when the import would update variants and change requests are disabled
    • Update feature dependency: when the import would add feature dependencies and change requests are disabled

    Import and change requests

    If change requests are enabled for the target project and environment, the import will not be fully applied. Any new features will be created, but all feature configuration will be added to a new change request.

    If change requests are enabled, any permissions for Create activation strategies, Delete activation strategies and Update variants are not required.

    Environment import/export vs the instance import/export API

    Environment import/export has some similarities to the instance import/export API, but they serve different purposes.

    The instance import/export API was designed to export all feature toggles (optionally with strategies and projects) from one Unleash instance to another. When it was developed, Unleash had much fewer features than it does now. As such, the API lacks support for some of the more recent features in Unleash.

    On the other hand, the environment import/export feature was designed to export a selection of features based on search criteria. It can only export data from a single environment and only import it to a single environment. It also only supports importing into a single project (although it can export features from multiple projects).

    Further, the environment import/export comes with a much more stringent validation and will attempt to stop any corrupted data imports.

    - - + + \ No newline at end of file diff --git a/how-to/how-to-import-export.html b/how-to/how-to-import-export.html index da1fc59773..234995d927 100644 --- a/how-to/how-to-import-export.html +++ b/how-to/how-to-import-export.html @@ -20,8 +20,8 @@ - - + + @@ -30,7 +30,7 @@

    [Deprecated] Import & Export

    Availability

    The import and export API first appeared in Unleash 3.3.0.

    Deprecation notice

    Api admin state is deprecated from version 5. We recommend using the new Environment Import & Export.

    Unleash supports import and export of feature toggles and strategies at startup and during runtime. The main purpose of the import/export feature is to bootstrap new Unleash instances with feature toggles and their configuration. If you are looking for a granular way to keep seperate Unleash instances in sync we strongly recommend that you take a look at the Unleash Admin APIs.

    The import mechanism guarantees that:

    • all imported features will be non-archived
    • existing updates to strategies and features are included in the event history

    All import mechanisms support a drop parameter which will clean the database before import (all strategies and features will be removed).

    Dropping in production

    Be careful when using the drop parameter in production environments: cleaning the database could lead to unintended loss of data.

    Runtime import & export

    State Service

    Unleash returns a StateService when started, you can use this to import and export data at any time.

    const unleash = require('unleash-server');

    const { services } = await unleash.start({...});
    const { stateService } = services;

    const exportedData = await stateService.export({includeStrategies: false, includeFeatureToggles: true, includeTags: true, includeProjects: true});

    await stateService.import({data: exportedData, userName: 'import', dropBeforeImport: false});

    await stateService.importFile({file: 'exported-data.yml', userName: 'import', dropBeforeImport: true})

    If you want the database to be cleaned before import (all strategies and features will be removed), set the dropBeforeImport parameter.

    It is also possible to not override existing feature toggles (and strategies) by using the keepExisting parameter.

    API Export

    The api endpoint /api/admin/state/export will export feature-toggles and strategies as json by default. You can customize the export with query parameters:

    ParameterDefaultDescription
    formatjsonExport format, either json or yaml
    downloadfalseIf the exported data should be downloaded as a file
    featureTogglestrueInclude feature-toggles in the exported data
    strategiestrueInclude strategies in the exported data
    tagstrueInclude tagtypes, tags and feature_tags in the exported data
    projectstrueInclude projects in the exported data

    For example if you want to download just feature-toggles as yaml:

    Export features (and nothing else) as YAML.
    GET <unleash-url>/api/admin/state/export?format=yaml&featureToggles=1&strategies=0&tags=0&projects=0&download=1
    Authorization: <API-token>
    content-type: application/json

    API Import

    Importing environments in Unleash 4.19 and below

    This is only relevant if you use Unleash 4.19 or earlier:

    If you import an environment into an instance that already has that environment defined, Unleash will delete any API keys created specifically for that environment. This is to prevent unexpected access to the newly imported environments.

    You can import feature-toggles and strategies by POSTing to the /api/admin/state/import endpoint (keep in mind this will require authentication).\ You can either send the data as JSON in the POST-body or send a file parameter with multipart/form-data (YAML files are also accepted here).

    You can customize the import with query parameters:

    ParameterDefaultDescription
    dropfalseIf the database should be cleaned before import (see comment below)
    keeptrueIf true, the existing feature toggles and strategies will not be override

    If you want the database to be cleaned before import (all strategies and features will be removed), specify a drop query parameter.

    caution

    You should be cautious about using the drop query parameter in production environments.

    Example usage:

    Import data into Unleash.
    POST <unleash-url>/api/admin/state/import
    Authorization: <API-token>
    content-type: application/json

    {
    "version": 3,
    "features": [
    {
    "name": "a-feature-toggle",
    "enabled": true,
    "description": "#1 feature-toggle"
    }
    ]
    }

    Startup import

    You can import toggles and strategies on startup by using an import file in JSON or YAML format. As with other forms of imports, you can also choose to remove the current toggle and strategy configuration in the database before importing.

    Unleash lets you do this both via configuration parameters and environment variables. The relevant parameters/variables are:

    config parameterenvironment variabledefaultvalue
    importFileIMPORT_FILEnonepath to the configuration file
    dropBeforeImportIMPORT_DROP_BEFORE_IMPORTfalsewhether to clean the database before importing the file

    Importing files

    To import strategies and toggles from a file (called configuration.yml in the examples below), either

    • use the importFile parameter to point to the file (you can also passed this into the unleash.start() entry point)

      unleash-server --databaseUrl [...] \
      --importFile configuration.yml
    • set the IMPORT_FILE environment variable to the path of the file before starting Unleash

      IMPORT_FILE=configuration.yml

    Drop before import

    caution

    You should never use this in production environments.

    To remove pre-existing feature toggles and strategies in the database before importing the new ones, either:

    • add the dropBeforeImport flag to the unleash-server command (or to unleash.start())

      unleash-server --databaseUrl [...] \
      --importFile configuration.yml \
      --dropBeforeImport
    • set the IMPORT_DROP_BEFORE_IMPORT environment variable (note the leading IMPORT_) to true, t, or 1. The variable is case-sensitive.

      IMPORT_DROP_BEFORE_IMPORT=true
    - - + + \ No newline at end of file diff --git a/how-to/how-to-manage-public-invite-tokens.html b/how-to/how-to-manage-public-invite-tokens.html index 7d32274123..371525158c 100644 --- a/how-to/how-to-manage-public-invite-tokens.html +++ b/how-to/how-to-manage-public-invite-tokens.html @@ -20,8 +20,8 @@ - - + + @@ -29,7 +29,7 @@

    How to manage public invite tokens

    Public invite links let you invite new members to an Unleash instance. A key part of an invite link is the public invite token. This guide shows you how to use the Unleash admin UI to create, update, and delete public invite tokens. You can also manage public signup tokens via the Unleash API.

    Only Unleash instance admins have the necessary permissions to create and manage public invite tokens.

    Creating a token

    1. Navigate to the users page in Unleash and use the create invite link button

    The settings menu in the Unleash nav bar with the &quot;users&quot; link highlighted.

    The Unleash users page. There is a separate &quot;create invite link&quot; section above the list of users.

    1. Fill out the "create invite link" form and (optionally) copy the invite link. You can always get the link later. A short form with only one field: token expiry.

    An &quot;invite link created&quot; modal. It contains an invite link that can be copied and some info on how to use it.

    Updating/Deleting a token

    1. Follow the steps in the previous paragraph to navigate to the users page.
    2. When you have an active invite token, use the button labeled "update invite link".
    3. Use the form to edit the expiry for the token or to delete it entirely.
    - - + + \ No newline at end of file diff --git a/how-to/how-to-run-the-unleash-proxy.html b/how-to/how-to-run-the-unleash-proxy.html index dc57d9adc9..b574f73a2f 100644 --- a/how-to/how-to-run-the-unleash-proxy.html +++ b/how-to/how-to-run-the-unleash-proxy.html @@ -20,15 +20,15 @@ - - + +

    How to run the Unleash Proxy

    Placeholders

    Placeholders in the code samples below are delimited by angle brackets (i.e. <placeholder-name>). You will need to replace them with the values that are correct in your situation for the code samples to run properly.

    The Unleash Proxy provides a way for you to consume feature toggles in front-end clients, such as the JavaScript Proxy client and React Proxy client.

    Depending on your setup, the Proxy is most easily run in one of two ways, depending on your situation:

    If you're using a hosted version of Unleash, we can also run the proxy for you.

    Prerequisites

    This is what you need before you can run the proxy:

    • A running Unleash server to connect to. You'll need its API path (e.g. https://app.unleash-hosted.com/demo/api) to connect the proxy to it.
    • A client API token for the proxy to use.
    • If you're running the Proxy via Docker: the docker command line tool.
    • If you're running the Proxy as a Node.js app: Node.js and its command line tools.
    • A Proxy client key. This can be any arbitrary string (for instance: proxy-client-key). Use this key when connecting a client SDK to the Proxy.

    How to run the Proxy via Docker

    We provide a Docker image (available on on Docker Hub) that you can use to run the proxy.

    1. Pull the Proxy image

    Use the docker command to pull the Proxy image:

    Pull the Unleash Proxy docker image
    docker pull unleashorg/unleash-proxy

    2. Start the Proxy

    When running the Proxy, you'll need to provide it with at least the configuration options listed below. Check the reference docs for the full list of configuration options.

    Run the Unleash Proxy via Docker
    docker run \
    -e UNLEASH_PROXY_CLIENT_KEYS=<proxy-client-key> \
    -e UNLEASH_URL='<unleash-api-url>' \
    -e UNLEASH_API_TOKEN=<client-api-token> \
    -p 3000:3000 \
    unleashorg/unleash-proxy

    If the proxy starts up successfully, you should see the following output:

    Unleash-proxy is listening on port 3000!

    How to run the Proxy as a Node.js app

    To run the Proxy via Node.js, you'll have to create your own Node.js project and use the Unleash Proxy as a dependency.

    1. initialize the project

    If you don't already have an existing Node.js project, create one:

    npm init

    2. Install the Unleash Proxy package

    Install the Unleash Proxy as a dependency:

    npm install @unleash/proxy

    3. Configure and start the proxy

    Import the createApp function from @unleash/proxy and configure the proxy. You'll need to provide at least the configuration options highlighted below. Check the reference docs for the full list of configuration options.

    Here is an example of what running the Proxy as a Node.js app might look like:

    Sample app running the Unleash Proxy
    const port = 3000;

    const { createApp } = require('@unleash/proxy');

    const app = createApp({
    unleashUrl: '<unleash-api-url>',
    unleashApiToken: '<client-api-token>',
    clientKeys: ['<proxy-client-key>'],
    proxyPort: 3000,
    });

    app.listen(port, () =>
    console.log(`Unleash Proxy listening on http://localhost:${port}/proxy`),
    );

    Verify that the proxy is working

    When the proxy process has started up correctly, you can start querying its /proxy endpoint. If it's running correctly, you'll get back a JSON object with a list of toggles. The list may be empty if you haven't added any toggles for the corresponding project/environment yet.

    Request toggles from the Unleash Proxy
    GET <proxy-url>/proxy
    Authorization: <proxy-client-key>
    content-type: application/json
    - - + + \ No newline at end of file diff --git a/how-to/how-to-schedule-feature-releases.html b/how-to/how-to-schedule-feature-releases.html index 847f087b8e..31a2812fff 100644 --- a/how-to/how-to-schedule-feature-releases.html +++ b/how-to/how-to-schedule-feature-releases.html @@ -20,8 +20,8 @@ - - + + @@ -29,7 +29,7 @@

    How to schedule feature releases

    Placeholders

    Placeholders in the code samples below are delimited by angle brackets (i.e. <placeholder-name>). You will need to replace them with the values that are correct in your situation for the code samples to run properly.

    There's a whole host of reasons why you may want to schedule the release of a feature, such as:

    • to release a feature at a specific date and time (for a product launch, for instance)
    • to make a feature available only up until a specific moment (for a contest cutoff, for instance)
    • to make a feature available during a limited period (for a 24 hour flash sale, for instance)

    There's two distinct ways to do this, depending on which version of Unleash you are running:

    In this guide we'll schedule a feature for release at some point in time. The exact same logic applies if you want to make a feature available until some point in the future. Finally, if you want to only make a feature available during a limited time period, you can easily combine the two options.

    Prerequisites

    This guide assumes that you've got the following:

    • some basic experience with Unleash
    • a running instance of Unleash and connected clients (where applicable)
    • an existing feature toggle that you want to schedule the release for

    Schedule feature releases with strategy constraints

    Strategy contstraints are the easiest way to schedule feature releases (as long as your SDKs are up to date). You can use this approach with any strategy you want. The strategies will work just as they normally do, they just won't become active until the specified time. For example: with the standard strategy, the feature would become available to all your users at the specified time; with a gradual rollout, the rollout would start at the specified time.

    Step 1: Add an activation strategy with a date-based constraint

    Scheduling a release via the UI

    To schedule a feature release via the UI:

    1. Add the desired activation strategy to the feature
    2. Open the constraint creator by using the "Add constraint" button
    3. Add a date-based constraint by selecting the currentTime context field (step 1 in the below image), choosing the DATE_AFTER operator (step 2), and setting the point in time where you want the feature to be available from (step 3) A strategy constraint specifying that the activation strategy should be enabled at 12:00 AM, November 25th 2022. There are visual call-outs pointing to the relevant settings mentioned above.

    Scheduling a release via the API

    To add an activation strategy via the Admin API, use the feature's strategies endpoint to add a new strategy (see the API documentation for adding strategies to feature toggles for more details).

    The payload's "name" property should contain the name of the strategy to apply (see activation strategies reference documentation for all built-in strategies' modeling names).

    The "constraint" object should have the same format as described in the code sample below. The activation date must be in an RFC 3339-compatible format, e.g. "1990-12-31T23:59:60Z".

    Add a feature activation strategy with a scheduled activation time.
    POST <unleash-url>/api/admin/projects/<project-id>/features/environments/<environment>/strategies
    Authorization: <API-token>
    content-type: application/json

    {
    "name": "default",
    "constraints": [
    {
    "value": "<activation-date>",
    "operator": "DATE_AFTER",
    "contextName": "currentTime"
    }
    ]
    }

    The "operator" property in the code sample can be replaced with any of the other date and time-based operators according to your needs.

    Schedule feature releases with custom activation strategies

    To schedule feature releases without using strategy constraints, you can use custom activation strategies. This requires defining a custom strategy and then implementing that strategy in your SDKs. For detailed instructions on how to do either of these things, refer to their respective how-to guides:

    Step 1: Define and apply a custom activation strategy

    Define a strategy that takes a parameter that tells it when to activate (visit the custom activation strategy reference documentation for full details on definitions):

    1. Give the strategy a name. We'll use enableAfter.
    2. Give the strategy a required string parameter where the user can enter the time at which the feature should activate. Be sure to describe the format that the user must adhere to.
    3. Save the strategy

    Apply the strategy to the feature toggle you want to schedule.

    A custom activation strategy definition for a strategy called `enableAfter`. It takes a required parameter called `start time`: a string in a date format.

    Step 2: Implement the custom activation strategy in your clients

    In each of the client SDKs that will interact with your feature, implement the strategy (the implementation how-to guide has steps for all SDK types).

    - - + + \ No newline at end of file diff --git a/how-to/how-to-send-feature-updates-to-slack-deprecated.html b/how-to/how-to-send-feature-updates-to-slack-deprecated.html index eb563e1a4e..c7e3b1f690 100644 --- a/how-to/how-to-send-feature-updates-to-slack-deprecated.html +++ b/how-to/how-to-send-feature-updates-to-slack-deprecated.html @@ -20,15 +20,15 @@ - - + +

    Feature Updates To slack

    caution

    This guide is deprecated. If you're looking for ways to integrate with Slack, you should refer to the Slack integration guide instead.

    Event hook option is removed in Unleash v5

    Create a custom Slack WebHook url:

    1. Go to https://slack.com/apps/manage/custom-integrations
    2. Click Incoming WebHooks
    3. Click “Add Configuration”
    4. This is Slack's help page on how to do this: https://api.slack.com/messaging/webhooks
      • Choose a channel, follow the wizard, get the custom URL.

    Send data to Slack using an event hook function

    Using the eventHook option, create a function that will send the data you'd like into Slack when mutation events happen.

    const unleash = require('unleash-server');
    const axios = require('axios');

    function onEventHook(event, eventData) {
    const { createdBy: user, data } = eventData;
    let text = '';

    const unleashUrl = 'http://your.unleash.host.com';
    const feature = `<${unleashUrl}/#/features/strategies/${data.name}|${data.name}>`;

    switch (event) {
    case 'feature-created':
    case 'feature-updated': {
    const verb =
    event === 'feature-created' ? 'created a new' : 'updated the';
    text = `${user} ${verb} feature ${feature}\ndescription: ${
    data.description
    }\nenabled: ${data.enabled}\nstrategies: \`${JSON.stringify(
    data.strategies,
    )}\``;
    break;
    }
    case 'feature-archived':
    case 'feature-revived': {
    const verb = event === 'feature-archived' ? 'archived' : 'revived';
    text = `${user} ${verb} the feature ${feature}`;
    break;
    }
    default: {
    console.error(`Unknown event ${event}`);
    return;
    }
    }

    axios
    .post(
    'https://hooks.slack.com/services/THIS_IS_WHERE_THE_CUSTOM_URL_GOES',
    {
    username: 'Unleash',
    icon_emoji: ':unleash:', // if you added a custom emoji, otherwise you can remove this field.
    text: text,
    },
    )
    .then((res) => {
    console.log(`Slack post statusCode: ${res.status}. Text: ${text}`);
    })
    .catch((error) => {
    console.error(error);
    });
    }

    const options = {
    eventHook: onEventHook,
    };

    unleash.start(options).then((server) => {
    console.log(`Unleash started on http://localhost:${server.app.get('port')}`);
    });
    - - + + \ No newline at end of file diff --git a/how-to/how-to-set-up-group-sso-sync.html b/how-to/how-to-set-up-group-sso-sync.html index 983abde5df..434a6baf1a 100644 --- a/how-to/how-to-set-up-group-sso-sync.html +++ b/how-to/how-to-set-up-group-sso-sync.html @@ -20,15 +20,15 @@ - - + +

    How set up user group SSO syncing

    availability

    User group syncing was released in Unleash 4.18 and is available for enterprise customers.

    This guide takes you through how to configure your user groups to automatically populate users through Single Sign On (SSO). Refer to setting up Keycloak for user group sync for an end to end example. Note that the steps below require you to be logged in as an admin user.

    Step 1: Navigate to SSO configuration

    Navigate to the "Single sign-on" configuration page.

    The Unleash Admin UI with the steps highlighted to navigate to the Single sign-on configuration.

    Step 2: Enable Group Syncing

    Turn on "Enable Group Syncing" and enter a value a for "Group Field JSON Path". Refer to the User group SSO integration documentation for more information or to the how-to guide for integrating with Keycloak for a practical example.

    The value is the JSON path in the token response where your group properties are located, this is up to your SSO provider, a full example for Keycloak can be found here. Once you're happy, save your configuration.

    The Single sign-on configuration page with enable group syncing, group field JSON path and save inputs highlighted.

    Step 3: Navigate to a group

    Navigate to the group you want to sync users for.

    The Unleash Admin UI with the steps highlighted to navigate to groups and a highlighted group card.

    Step 4: Edit the group configuration

    Navigate to edit group.

    The group configuration screen with edit group highlighted.

    Step 5: Link SSO groups to your group

    Link as many SSO groups as you like to your group, these names should match the group name or ID sent by your SSO provider exactly. Save your group configuration, the next time a user belonging to one of these groups logs in, they'll be automatically added to this group.

    The edit group screen with SSO Group input and save highlighted.

    - - + + \ No newline at end of file diff --git a/how-to/how-to-setup-sso-keycloak-group-sync.html b/how-to/how-to-setup-sso-keycloak-group-sync.html index 70b43984f9..698e6b85c2 100644 --- a/how-to/how-to-setup-sso-keycloak-group-sync.html +++ b/how-to/how-to-setup-sso-keycloak-group-sync.html @@ -20,15 +20,15 @@ - - + +

    How to set up Keycloak and Unleash to sync user groups

    availability

    User group syncing was released in Unleash 4.18 and is available to enterprise customers.

    In this guide, we will setup OIDC Single Sign-On (SSO) in Keycloak and configure Unleash to automatically sync user group membership from Keycloak.

    Prerequisites

    The steps in this guide assume you have admin access to a running Unleash instance and to a running Keycloak instance.

    Keycloak Configuration

    Step 1: Navigate to Create Client

    Open the Keycloak admin dashboard, navigate to clients and select "Create Client".

    The Keycloak Admin UI with the steps highlighted to navigate to client configuration.

    Step 2: Create an Unleash Client

    Select "OpenID Connect" as the client type and give your client a name, then save your configuration.

    The Keycloak Admin UI with the client configuration open.

    Step 3: Set a redirect URI

    Set the redirect URI to:

    <base-url>/auth/oidc/callback

    For a hosted Unleash instance this becomes:

    https://<region>.app.unleash-hosted.com/<instance-name>/auth/oidc/callback

    Save your configuration.

    The Keycloak client configuration with redirect URIs highlighted.

    Step 4: Copy your client secret

    Navigate to "Credentials" and copy your client secret. You'll need to add this to the Unleash configuration later, so put it somewhere you'll be able to find it.

    The Keycloak credentials configuration with copy client secret highlighted.

    Step 5: Copy your OpenID endpoint configuration

    Navigate to your realm settings and copy the link to OpenID endpoint configuration. You'll need to add this to the Unleash configuration later.

    The Keycloak realm settings the OpenID endpoint configuration link highlighted.

    Step 6: Create a new Client Scope and Map Groups

    Navigate to the "Client Scopes" page and select "Create Client Scope".

    The Keycloak Client Scopes page with the Create Client Scope button highlighted.

    Give your new scope a name. Set the type to "Optional". Make sure the protocol is set to "OpenID Connect" and the "Include in Token Response" option is enabled. Save your new scope.

    The Keycloak Add Client Scope page with the Name, Type, Protocol and Include in Token Response fields highlighted.

    Navigate to the Mappers tab and select "Configure new Mapper".

    The Keycloak Client Scope details page with the Mappers tab and Configure new Mapper element highlighted.

    Select the Group Membership mapper.

    The Keycloak mapper popup with the Group Membership mapper highlighted.

    Give your mapper a claim name, this must match the "Group Field JSON Path" in Unleash, and turn off the "Full group path" option.

    The Keycloak mapper options screen with the Token Claim Name and Full Group Path elements highlighted.

    Unleash Configuration

    Step 1: Navigate to the Unleash SSO Configuration

    Log in to Unleash as an admin user and navigate to the SSO configuration. Input your Client Secret (copied in step 3 of the Keycloak configuration), your Discover URL (copied in step 4 of the Keycloak configuration), and the Client ID (from step 2 of the Keycloak configuration).

    The Unleash SSO configuration screen with Client ID, Client Secret and Discover URL highlighted.

    Step 2: Enable Group Syncing

    Turn on Group Syncing and set a value for "Group Field JSON Path". This must match the value in claim name in Keycloak exactly. Save your configuration.

    The Unleash SSO configuration screen with the Enable Group Syncing and Group Field JSON Path highlighted.

    Step 3: Enable Group Syncing for your Group

    Navigate to Groups and select the group that you want to sync.

    The Groups page with a group element highlighted.

    Edit the group.

    The Group page with the Edit group element highlighted.

    Add as many SSO groups as you like. These need to match the Keycloak groups exactly.

    The edit group page with the add SSO group element highlighted.

    Save your configuration. Once a user belonging to one of these Keycloak groups logs in through SSO, they'll be automatically added to this Unleash group.

    - - + + \ No newline at end of file diff --git a/how-to/how-to-synchronize-unleash-instances.html b/how-to/how-to-synchronize-unleash-instances.html index 6e48e45b09..73687d864f 100644 --- a/how-to/how-to-synchronize-unleash-instances.html +++ b/how-to/how-to-synchronize-unleash-instances.html @@ -20,8 +20,8 @@ - - + + @@ -32,7 +32,7 @@ Example: SOURCE_API_TOKEN="user:98b555423fa020a3e67267fb8462fdeea13a1d62e7ea61d5fe4f3022"
  • SOURCE_ENV: The environment name for the source instance. Only feature toggles matching this environment will be exported.
  • SOURCE_TAG: The tag to filter feature toggles for export.
  • Target Unleash Instance

    • TARGET_URL: The URL of the target Unleash API. Example: TARGET_URL="http://localhost:4242/api/admin/features-batch/import"
    • TARGET_API_TOKEN: The API token for the target Unleash instance. This can be a personal access token or a service account token with enough privileges to perform the import operation. Example: TARGET_API_TOKEN="user:98b555423fa020a3e67267fb8462fdeea13a1d62e7ea61d5fe4f3022"
    • TARGET_PROJECT: The project name for the target instance where the feature toggles will be imported.
    • TARGET_ENV: The environment name for the target instance.

    Process

    The script performs the following steps:

    1. Export feature toggles from the source instance based on the specified tag and environment.
    2. Import the exported feature toggles into the target instance under the specified project and environment.

    If change requests are enabled in the target project, the import process will go through the change request process, allowing you to review the changes before applying them.

    The script prints each step of the export and import process, providing feedback on the success or failure of each operation.

    Troubleshooting

    Here are some common issues you might encounter and how to resolve them:

    1. Make sure you use the correct URLs for the source and target instances.
    2. Ensure that the API tokens have the necessary permissions to perform export and import operations.
    3. Verify that the specified source and target environments exist.
    4. Check that the target project exists.
    5. If you have change requests enabled in the target project, ensure that there are no pending change requests for the same API token.
    Feedback wanted

    If you would like to give feedback on this feature, experience issues or have questions, please feel free to open an issue on GitHub.

    - - + + \ No newline at end of file diff --git a/how-to/how-to-use-custom-strategies.html b/how-to/how-to-use-custom-strategies.html index 3f16be973b..30ff826b14 100644 --- a/how-to/how-to-use-custom-strategies.html +++ b/how-to/how-to-use-custom-strategies.html @@ -20,15 +20,15 @@ - - + +

    How to use custom activation strategies

    tip

    Use custom activation strategies only for cases where strategy constraints don't give you enough control.

    The downside of using a custom activation strategy instead of constraints is that you need to distribute the code of your strategy with Unleash client SDK while on the other hand, strategy constraints work without any extra code or maintenance.

    This guide takes you through how to use custom activation strategies with Unleash. We'll go through how you define a custom strategy in the admin UI, how you add it to a toggle, and how you'd implement it in a client SDK.

    In this example we want to define an activation strategy offers a scheduled release of a feature toggle. This means that we want the feature toggle to be activated after a given date and time.

    Step 1: Define your custom strategy

    1. Navigate to the strategies view. Interact with the "Configure" button in the page header and then go to the "Strategies" link in the dropdown menu that appears.

      A visual guide for how to navigate to the strategies page in the Unleash admin UI. It shows the steps described in the preceding paragraph.

    2. Define your strategy. Use the "Add new strategy" button to open the strategy creation form. Fill in the form to define your strategy. Refer to the custom strategy reference documentation for a full list of options.

      A strategy creation form. It has fields labeled &quot;strategy name&quot; — &quot;TimeStamp&quot; — and &quot;description&quot; — &quot;activate toggle after a given timestamp&quot;. It also has fields for a parameter named &quot;enableAfter&quot;. The parameter is of type &quot;string&quot; and the parameter description is &quot;Expected format: YYYY-MM-DD HH:MM&quot;. The parameter is required.

    Step 2: Apply your custom strategy to a feature toggle

    Navigate to your feature toggle and apply the strategy you just created.

    The strategy configuration screen for the custom &quot;TimeStamp&quot; strategy from the previous step. The &quot;enableAfter&quot; field says &quot;2021-12-25 00:00&quot;.

    Step 3: Implement the strategy in your client SDK

    The steps to implement a custom strategy for your client depend on the kind of client SDK you're using:

    Option A: Implement the strategy for a server-side client SDK

    1. Implement the custom strategy in your client SDK. The exact way to do this will vary depending on the specific SDK you're using, so refer to the SDK's documentation. The example below shows an example of how you'd implement a custom strategy called "TimeStamp" for the Node.js client SDK.

      const { Strategy } = require('unleash-client');

      class TimeStampStrategy extends Strategy {
      constructor() {
      super('TimeStamp');
      }

      isEnabled(parameters, context) {
      return Date.parse(parameters.enableAfter) < Date.now();
      }
      }
    2. Register the custom strategy with the Unleash Client. When instantiating the Unleash Client, provide it with a list of the custom strategies you'd like to use — again: refer to your client SDK's docs for the specifics.

      Here's a full, working example for Node.js. Notice the strategies property being passed to the initialize function.

      const { Strategy, initialize, isEnabled } = require('unleash-client');

      class TimeStampStrategy extends Strategy {
      constructor() {
      super('TimeStamp');
      }

      isEnabled(parameters, context) {
      return Date.parse(parameters.enableAfter) < Date.now();
      }
      }

      const instance = initialize({
      url: 'https://unleash.example.com/api/',
      appName: 'unleash-demo',
      instanceId: '1',
      strategies: [new TimeStampStrategy()],
      });

      instance.on('ready', () => {
      setInterval(() => {
      console.log(isEnabled('demo.TimeStampRollout'));
      }, 1000);
      });

    Option B: Implement the strategy for a front-end client SDK

    Front-end client SDKs don't evaluate strategies directly, so you need to implement the custom strategy in the Unleash Proxy. Depending on how you run the Unleash Proxy, follow one of the below series of steps:

    With a containerized proxy

    Strategies are stored in separate JavaScript files and loaded into the container at startup. Refer to the Unleash Proxy documentation for a full overview of all the options.

    1. Create a strategies directory. Create a directory that Docker has access to where you can store your strategies. The next steps assume you called it strategies

    2. Initialize a Node.js project and install the Unleash Client:

      npm init -y && \
      npm install unleash-client
    3. Create a strategy file and implement your strategies. Remember to export your list of strategies. The next steps will assume you called the file timestamp.js. An example implementation looks like this:

      const { Strategy } = require('unleash-client');

      class TimeStampStrategy extends Strategy {
      constructor() {
      super('TimeStamp');
      }

      isEnabled(parameters, context) {
      return Date.parse(parameters.enableAfter) < Date.now();
      }
      }

      module.exports = [new TimeStampStrategy()]; // <- export strategies
    4. Mount the strategies directory and point the Unleash Proxy docker container at your strategies file. The highlighted lines below show the extra options you need to add. The following command assumes that your strategies directory is a direct subdirectory of your current working directory. Modify the rest of the command to suit your needs.

      docker run --name unleash-proxy --pull=always \
      -e UNLEASH_PROXY_CLIENT_KEYS=some-secret \
      -e UNLEASH_URL='http://unleash:4242/api/' \
      -e UNLEASH_API_TOKEN=${API_TOKEN} \
      -e UNLEASH_CUSTOM_STRATEGIES_FILE=/strategies/timestamp.js \
      --mount type=bind,source="$(pwd)"/strategies,target=/strategies \
      -p 3000:3000 --network unleash unleashorg/unleash-proxy

    When running the proxy with Node.js

    The Unleash Proxy accepts a customStrategies property as part of its initialization options. Use this to pass it initialized strategies.

    1. Install the unleash-client package. You'll need this to implement the custom strategy:

      npm install unleash-client
    2. Implement your strategy. You can import it from a different file or put it in the same file as the Proxy initialization. For instance, a TimeStampStrategy could look like this:

      const { Strategy } = require('unleash-client');

      class TimeStampStrategy extends Strategy {
      constructor() {
      super('TimeStamp');
      }

      isEnabled(parameters, context) {
      return Date.parse(parameters.enableAfter) < Date.now();
      }
      }
    3. Pass the strategy to the Proxy Client using the customStrategies option. A full code example:

      const { createApp } = require('@unleash/proxy');
      const { Strategy } = require('unleash-client');

      class TimeStampStrategy extends Strategy {
      constructor() {
      super('TimeStamp');
      }

      isEnabled(parameters, context) {
      return Date.parse(parameters.enableAfter) < Date.now();
      }
      }

      const port = 3000;

      const app = createApp({
      unleashUrl: 'https://app.unleash-hosted.com/demo/api/',
      unleashApiToken:
      '*:default.56907a2fa53c1d16101d509a10b78e36190b0f918d9f122d',
      clientKeys: ['proxy-secret', 'another-proxy-secret', 's1'],
      refreshInterval: 1000,
      customStrategies: [new TimeStampStrategy()],
      });

      app.listen(port, () =>
      // eslint-disable-next-line no-console
      console.log(`Unleash Proxy listening on http://localhost:${port}/proxy`),
      );
    - - + + \ No newline at end of file diff --git a/how-to/how-to-use-the-admin-api.html b/how-to/how-to-use-the-admin-api.html index 9dbcfd1057..85ff67b1c9 100644 --- a/how-to/how-to-use-the-admin-api.html +++ b/how-to/how-to-use-the-admin-api.html @@ -20,15 +20,15 @@ - - + +

    How to use the Admin API

    It is possible to integrate directly with the Admin API. In this guide we will explain all the steps to set it up.

    Step 1: Create API token

    You'll need either an admin token or a personal access token for this to work. To create one, follow the steps in the how to create API tokens guide or the how to create personal access tokens guide, respectively.

    Please note that it may take up to 60 seconds for the new key to propagate to all Unleash instances due to eager caching.

    note

    If you need an API token to use in a client SDK you should create a "client token" as these have fewer access rights.

    Step 2: Use Admin API

    Now that you have an access token with admin privileges we can use that to perform changes in our Unleash instance.

    In the example below we will use the Unleash Admin API to enable the “Demo” feature toggle using curl.

    curl -X POST -H "Content-Type: application/json" \
    -H "Authorization: some-token" \
    https://app.unleash-hosted.com/demo/api/admin/features/Demo/toggle/on

    Great success! We have now enabled the feature toggle. We can also verify that it was actually changed by the API user by navigating to the Event log (history) for this feature toggle.

    A feature toggle&#39;s event log showing that it was last updated by &quot;admin-api&quot;.

    API overview

    You can find the full documentation on everything the Unleash API supports in the Unleash API documentation.

    - - + + \ No newline at end of file diff --git a/how-to/misc.html b/how-to/misc.html index f53e6153a7..a44a72903e 100644 --- a/how-to/misc.html +++ b/how-to/misc.html @@ -20,15 +20,15 @@ - - + +

    How-to: general Unleash tasks

    Guides for how to perform general Unleash tasks.

    - - + + \ No newline at end of file diff --git a/how-to/proxy.html b/how-to/proxy.html index c49f05f2e4..223b861013 100644 --- a/how-to/proxy.html +++ b/how-to/proxy.html @@ -20,15 +20,15 @@ - - + + - - + + \ No newline at end of file diff --git a/how-to/sso.html b/how-to/sso.html index 8e13ba5da6..1648a9a5d6 100644 --- a/how-to/sso.html +++ b/how-to/sso.html @@ -20,15 +20,15 @@ - - + +

    How-to: Single Sign-On

    Single Sign-On how-to guides.

    📄️ [Deprecated] How to add SSO with Google

    Single Sign-on via the Google Authenticator provider has been removed in Unleash v5 (deprecated in v4). We recommend using OpenID Connect instead. If you're running a self hosted version of Unleash and you need to temporarily re-enable Google SSO, you can do so by setting the GOOGLEAUTHENABLED environment variable to true. If you're running a hosted version of Unleash, you'll need to reach out to us and ask us to re-enable the flag. Note that this code will be removed in a future release and this is not safe to depend on.

    - - + + \ No newline at end of file diff --git a/how-to/users-and-permissions.html b/how-to/users-and-permissions.html index 40cf8913d8..cf2c869fc4 100644 --- a/how-to/users-and-permissions.html +++ b/how-to/users-and-permissions.html @@ -20,15 +20,15 @@ - - + +
    - - + + \ No newline at end of file diff --git a/img/license-banner-expired.png b/img/license-banner-expired.png new file mode 100644 index 0000000000000000000000000000000000000000..1af308b06d6ccec862b1320555e363d1cf28ba28 GIT binary patch literal 23212 zcmeFZWmFv9(mzTF1cxC6NpKJD5ZowYz&yPp|5=t7=!(uKlb2r64bfhWrH?1_lOAT1rd_1_mDft?z*N;q81# zT(|mmdFQAkDFRbIN^tP@Lky@PZ7L@RL;Kc8gn0)thk^UoBX7r-w*v;|ebzge_ixvC z|GJk2`@jFf!)Lw!-#(1PzaAXV5gven5rUBx6IOA1cbpENhOPS8N9>{5DAUNB`{{iX z1k1`>oVh|#|2_VZjeZjrl!=yvHq4JiD$t2bhUwhbXV2jzgLinj!S&W={RKS}xI21q zn#Rp`nYx#zygiPi>w-dv6b17SO(>gGp?Z8e4x94bN0NVN%y3_kD@p&E1Sz?YA|xu^ zPAw5m%J09oHv(w?hZ6|HdWRI%z6!^qQbNUMVABvX3{XI#hL@h{({b(68@6> z@8dThCRO;iBu>eJKo)6|IV2&N=0d&aLI7SIsaVGx?r_03Jh;$4>Ho$(WcGJq0q_r3 z5Ic8i7^8LZ>%(Mv(g*RES#jSk;?a-}fl>rg^vUS|D&Y23F~6MkR;OBxSsn6q2>mlt+!cqwR ztHFbW^!Ot`rG2TQfd3^FO(qWS+vIZmRjS8GQ_3KdvOQShW94A^8_mQPGg^!Bk;5HHcDWSq6XqORYkbXUvPI-_(4ApyPxZxtEldr4OV#mai>^}U z(Jg#TYT@%^O~fQWK~7}A>K57@U%0!>c!m}AHuGD!rvez+1pYoy$O^+2mR?hBQZHA<8; zTU)1Sr!Nxe%dLNve#Mm6Ewm||zBij&umx}Not;woLF<3jZWsyNIxP2LwY0W8+u10J zKDL#JhcAE6IKUx;Z8+ysQaGC_p3_9;Q~qLAIYk!S5oDc2n0d(82bGD}3NMGG6E%sC zrdMzkqj_JT7`|Z44&L2Tn^WLgx)Uu&3*Ui2K0C z>9woxUmJc}_2|e%dKZkngO{-q^@22T;+{VKtV~3Fp2^l|y1Q0gK*~Y_A+Jf09<~XET1|z8-e_}-&M^O%XQexp?fEFH2QiWKQpV88T6)9vr==d zZPw#xDn}~hkM)Wwm!~T8($1FlgBXl0p&Vg8*A)v@z@x^R33^hY)urP2Y9(Q(gS4M)XtnOjlSNC^;%k=JM!22s zw@rv$qv*bH3{H6(xD08lieIVoxWr7o;E`L_+D1z6p5QP5_bc-*dM5K&`ksB<9u3== zi@vGlYuFcQ(lTI6A7hzsOFS-9)1D~zi=AK#Oi=853`gu?wmn^LY5a;gP|DZ2jsxme z|FkbsXlRY2(O2dy*C)fDWgNyvscVeDNGMC48yw8@Ehs(8o1&98-QA7L+e=iK$mz+| z>3P0t5ICoP41Rs0?UDJE(K;N79~mt~i=^;LmQpUiW(?CWIJ4%WiY&A4qYJvQpf5p{ zxk2iFad)b!ox8)R)w9u>QmcLi*@KZee82j~PsPo^ol^r?@MtErErSjo0ZDg39dfSr zh33~G6yb|8XDFPn6DBO4*mM7pCj3W9XKaemVr9vt>wx;5uE3i-te3cZ+7F!%C%Eg(;?~sFE zmgdv3o^dsnN`s!Cn*eIZ6SW-O$ zLnhLTm=3w1Lk%?#9cc{SD!+O1JWw{S7@$lE8Gs@vq$|c zR^@tx6npvD%nq4(66SlkcZCFibf*tkux(sS9LxDND6il zv8nX;$2;txCD)cwdcBCu05Kfv1*6IC@Y>tv?2NfW4qTlveIL%Hza{EiUUZh974nw~ zZp@>sxcQcw3e37b`YWHfrs`F=>^PAj~$7>r^%7))JUsu^w+Bpsz zBeC*UEb63tm+AVl~6rr3l%KJ4n01Bkjten(|l0c*p-Y1;_o zMN&nUuDn!FStf7g%ggHLDLHP;IcI$)eVpfC%@Yz!s$XsYq-&JqR{%mt5J4V3txmtv za%b%U@LN3`m^V+Z$r(%3kb%Aru(LXuEcurjoJuZDkRr8)SZk6E8;k+u9*_+f%V z5!AZHQ#0!$)*}G!f|ySu1&}PSEy@@CB-*cgWfE2#&;P~BG+(KCGUszzRlYK8XQjd8 zp{ucmd%8?kmSsu4VA+(0UT^|FE>A_rXDG(SD$fI%jj*bvloMZw;k)U9tI!t5If>l|@o)iUjkM`oW-KVthCeztftyxY)nYt;3;7lPl5E>Y{*e&=iZe4=dsqqdA32%*=rbGGbI^kUfNPJ9zmufzjMUf!XG>1J=Vlt z0pO;0mSs>w zmZ9TYtVL5NK1zWL9Y^vSOvuFo_%30B9Ax)jd%geK*6lqkIeLHhb} zy0lk@dQCS7I%H@mgoZbJqj$noe|#2lC+cVHCpX$(`wPLRVHK+-75WI=`}4e4MD;Q_ z!w&;Na=EgqmNf~pkGBc599sR2st;+edkB?Q8=3h62tw87o=o!iV$SFqI{O#G0Q-71<5FHA18TrI11TWL0c=fX9){%%e+VHs^;%@U@21{IC( zTpQ;~uoI__-pLK7lN821aN$I8pHE=04?WEBn!St*x0J?)9%mx(2VSR@9HVAS0^@WLws|XvDNNY@4@%$?`bUn(3*BW3A|Q0!T)c&g{N6}ydQglBCrE~4b6Qu=zbv#QU#N9`#s?_M2fYXF zI6AubUnJoa?>u!C+)yRGEk+5rmG4F8yTY$pcqp7FW*H;^Vmx2T%lShe0eik$E#H}s zz{eSmk&Z25OBsX$R%2@;gD=`GddJMKS*#=&!-k;5${3z%qGN3`?$(Q%e)A(=W!X1<$^l*xjU)sfpy2(&|81L0j-fUrlod*xSjUXE#lQHAU4r@ z#kBlAnD~jsb#v>>s$SbZXmsv?#pPfiAb#7{a_(oU6?XaY=mI$_DL`vApcn)GFW8p?ZyqG@I8-FH<>F z{C;Yg=$aqYqhpuh0nv${1tfxj}7_J@Q ziDXjrD@b9>huJb`IN-#&_QVB$Rt?_P0KC1Y;ys?49kF>n znK}!fD_FWrKJw-(we@QkaGf$eD7baf@JpQLM8cgV+FY^RTA}MKpUHwwJo2$R%zVEj zeQ#9f+;U9>Y9$IV(73lIXmZZvJ@dK@UVVKjv3_BGCP6d1_dM<5Y+Z5TKjI-rMti*b zlFaksonFOwaWwfHnwq=olWy`fHa5Vth92X9Z878%fj+%asCkQ&`k^)*qv3n;oH@1c zZ#P+=M?(m5=*Te3-AwtK>X$kmy+{nj4JhYAwIL0PtTR{IKG!)M}KT#40?`w|ysE=`|sCzcB@ka9h|) zT2ehLwcPyi`SH)^6x3|MIqonK@K54*I$JowOXD$L;NVP@Q4cx&_dHc*vK1TeKZGn! zS^2yYmkmpuR1bq-Ijea>e`DR*WQXafd(W!JfF9;??paMgsy!;1)LY zm+384*@N$XkuEvCWd=6je$wELhLXPYz-^+TjfPuvU2I}wXPp7@a9FwaWm26Xi2lxN z@D}NQ;9NUL&Jm6rn21@d_2v8o{=!G(9d;Rp zJRR5YQDM_yk1;=#3klxDKu;hy1hT&KehGxfy|4pGcfqc6W3#tu z;Z$%|Sz}HZE+B`8DSCGMkXP}i2uicQ zRx7cW zTJamki1cF{j((hgqPhp<-caf@^Pz!#}Bxir4;TUTfbK8Hm}P7vKUjBt`+FIb;PhCg=e5(r&78+ zViL|n4tV(!o%FLvS8&z&F=uJn1AoChhWfw*aPL0`YQ^Jp%Fw$pBK64o3AyGZs78C- z5pxv3w838QxItX+Eu5Qg;aqJYJJRampFj2JeoC$vIKD(Y5x#@W%ABs?9lT>{h!+s- z@Zg`gNcHo4xfF2qgFm5!0En_-bDdAT=ZPHadQ{++E6r=dBvFQmpGah9EImTKE?0LZ zjNc594sFT!-X5S;amaXQsE$#EuCd>7C21ovS)B0&?>s2XGS6i1k9ACiP4okp-RRR~ zQMsCJ(wgma%DuzpJ$R^IaqJ7U$z_Fc8iEp04+`d>RVg2E5+Y1aGtN0W4eD>85(wPuW&ATVKFkHt05xby`Uk;A%FKVRFnjI zz#~uM5#8PvFEl&8wtf#&rm6vWpZhM6P)4}>Q3s?gTpA$dhieQy6L(M7igMSFF|2^I zCs(|kdqDFC4X3yFGi)IEamYOsCcdM_dFGm#w*sr0Oxdbwx{W5TOE)WUEdQf#$&I-$3c??6lD zHNs)n$S2FLS+3f-wW%nCrDtz-=_W*Uwpp-ySzhCeD5CwQiD3U8V+L)g5F6M%FTpUi znORo@VVow_3Y@+`V}b$_CpbUS#Ex)Th2v6AkJZ|9@jacbV}!@Qqea$4I7bnQ_Exlq zp47fs3vxNWr?&FF6=%q<-%hwLLDvK2M=xdi|nx?v$Nz0+0$ zl4rSR5E9*cxV$!JsdQ{O%)XlC4qLEjids^I53fqx=&!)SU`OYrX&TcB%-ovqf?rgW zNtc|HT*vus&o0Bm6Mg9M0ybs=Cx*X%U1G>^(OZ1f)a8aYb~lGe;i@r|2Zly`6P^Oh zLw>5`(OMIVMe4P^n)-gQVd1;FF;^Y|*n1JQ2higEu`UyQPtHRGASb2o@<2Z}tIY7# zNa(H2l?rn6CKd@wU;3YABxyd?xQiLQZ~@Qe9>~LR5$W!im|pXWiyvqB&Bw9MkDb;U zYbS|y+O=Z06DN=_yboVPn*s61eJE{jVGC{oZpimON44L5&ukaJsOi{oCMsTT3WuZx(;q033gw`!pE4;#qdPFgz<5 zZ8;MnXM&QAmEZ}v;;(Wcyfpme?V;gWJBG{|nYD+`2QuiOUp6asJngzkY9i#lKr;4M zuNxp`_3or*H9M?Rs<0r(hR%w^3*-CLMb2vxOnl(b*1*nBEXxsDu{E8C zp10^HkM+`tI6c+5QdFOJ3rF5(F**hk9p}Xv=cY zPZ>_Nsq9bg0SqNG7RDi)ivdC=L(4>;^=u-6W=%jxwbL5QIdpQRgzt29>Pm*dLY1;2 zbP&;Jtg#(r7Te_?Kcpj#UrM^__Xf9Fz(Zk_v41z}YzAo&@Hz)sFxB;N?K`-GnW2sG zGE~xmyONEd`X8_q03i~nIX?*uMy!w_HNce802cuX3nTE`CI_+pizL*`vAj#h1-=G3 zkU5yCWp+k}$y%@d9KY`z@>BP-w>?9NNJcu=AQ8rrk)nSs?YehE-xt)4^Z8)hpQ!d| zE}BfFtITwD)`>q<&%8_{y z(>9V~rH3*fof6cP4@oCCC$CeY;Jo|kc;YVd?QSC&wQJ62!I&m=%gwF#{@S&oFK{qY zp~ab~0__hr-lVg>hxJ3Dt%lq1=irS^;=?BcE~re8D(!ClEE@#InXsIO*S5fu@26lX zDxiv`AQk$>Tsv~lAl(o&wP;r_Pzi+?-9FP=@Mnuu(uH(<#OXpW>tUP}tpBcB6Epnr zC~$MNoC%Ca6$-`j?NM1?(l67EoZ+5_Sl9_-8<56mBa~oi=&$CN_$Q7-)Jl1zF5h=y zYwfw&WNU?|aS4=QAljID7QmLsJUP!?+WiPma2VN^_CyGCgzH*o4#Ih62|;aw1#@XN zw_+G#an}jYNw7Yn(RUi??(YRWG0z08Ny4N`tmm~*!<0`zTKT`)rb^~$BPIdNSe)U+ zI&QD@fMYTWL^jVPe`f5|HhY$9Pg*#m9om(!Z|P;Z)_vLoA%EP}0z<0FkLUj-$+s1S zC_Cx!rsywhLektUo7oMI&N%daHk)aIQ;Qk}Pq#6r_ng01dU*@}XO^0q^)VHUah)F; zK~wuVOMleiybi4Pbv=BzirhPF0hj32B@vmF_{2Ty?NN*#w9JA>f9aI?oCo|d9F_M% z?X)<)!}B@_LZbSq;dzsfd$sI;)1sEit+TsYzq&@3C?5@8ZP2ioff|LBoy=FLsvQhA zCpn_$to|A9zogo|tm?9BF}GR`!k}n+bID9G$J$i_yN6)L;r9bR&zPNif0s(tA)wUAY(Y<8zI_id5CM4Ij#x&EtSa9-ZcO_b$s7) zB&1kQjO`kU&XE!PME5mvLpntsv4!ftVt~JEOk{O)T3!0uT#@PCmp%=~MLuU~Wq%h< zEnF~?I$!=Q_}#dsmG5Yq)WYB0KRK0!D(ZeiKcuDWpr5#Aojy-35P+$u4;@r3rAuWa zBd$MUv3c8od_t-x_k0pP`-=1wjx?W0+_Wy`qdN2Bs2!y@v+-*m;Ycb9y~CO6Ex_Bo zmw-75^cL;9BuNoqv35=Fis+&m{ngqWQO&0P3<8aIU6G>0QkU8@qat1Q$1~{3f~0!G z;%yqwZ5EQ|iAAN+C!I+5Ri=4#vK=Xm_CFKQzhjB}(1{Qfh~-pYVQKvfY46+!UJ;qe zw@_aJRKt6AO9Z=`q@u z#yE60_y-c>jJ4aT^jo<7?ArwBxlln~_J}ZO1s9*)tZxBnv!JsVX zCOuJ%gJw?+F~b_Ic>{FeIl~;cAW`1#@!qYgrwFb1y8u2zARc&DqVV*y1^+ubglcHk z!@R@B!#7p^ayz-K5kt&(-L5M*{I~nO%xT!8wK80&Hxs@FH7mr|JfIysxKF&dQwq8! z@FJ@V&Npnw=ijieNDjSgDJJ{6!%-$S3MbkmUP&5Tz@2n(!r#1(+RrQ>5irj%(B89r zP?KHv!@y9tX!C1enFv{DHp@$|`>1^jznmY;FjR7A5Ae17chia4Li{=K+S1&Se{bomeCu<5%YjQpfgRevnpQ~*Vw*e+CK4)zz zpxq5GV;=~~o?aJXm2}zII8&!Tzs{R?9$8|*-P|HlliPa?gy$*8J#`nugBmd4HMpv< zU)n5g{VTzMkbMhx3^p7HFQQ{cNSX6k$;Y)tk!hyG=ip4}udih;gWFB%fn~SBzs|=z zZN&!L2G_qtB5!l;SCiiXLK{0^>vN}8Jypt$MiD|V{`z*(t_hEN%CFcNJ_9CQGL-?i z#C+qKR9lI~d-%4xPm2ZGZCTF{ft6mF8v!D)tzUc%_BkoKhyk5GnNbran18;qUYiM& zDqR%A9aySen&5*qy?JIzSYAwgtAbiu9&T>XD)bE% zi%~Z}>@#h%$2~huc2~sta-|OLp&yWN7e*{;Yt&sBUEsw=vnatn9sQ7r|3~xMFdd@d zWCBunP#nFg{7^hmvaQ`1L3Y2{&6@tWSCAk<=?4};GM)2} z(R%?q;nBXH3*ZfZ2~?}89UWkW;^$$mjhK6;QdCHa)#7e9f0s{q61-wcoghaF{4PZ| z#00hO1F?pPky~wM7h4&WnTr(X=egrI$sybWZk`Jqwm3YLE!f06^X^ zAzVNkyaKQFD+>2j(18!J~%BJJ{Yn-KNh&RL#GQ zUU-@oDR;XAcE(1&c>2rnf zU^1I&TwGL82g)j9dM0v7Q>S4?uy$&ERGl+A9cs_LOjj3AZz~MvB&n5#v{O?8ycz&c z%~(7zi*JiiP*;7ZD-+868kb%*!hcd1nVmLPUBaO6bjlcBXT=fj*-nPltSHztjui54 z?XTmv5bmJIEE!zox_DYI^l2w%D`rIysR>>qlKaPScCT!VAt@}zCLn7x|7*v?2ltlw z?7!xLZuTAD#~_@J&%<`K{6B1=e6hdk-F9wkJyEf|DcT3&-Qj~Ji8Ks9g*(mPzY-;} zDC-^1PlOt{=oPRuMxo-Ew4-pAvuM;cmY7g0WxnMl8B66MtnA&=^ajSbLLGy>!swUs z9S?Z3-t|;bA8;zUHG@TdN_8 zA|Xh|KbhbpzBS!p(z_iKcz_D$JwbW}8eb7g5+4`WJIp67kDJY0X?1b2t8fgCK{)_F5i&r_L)w>el74$!8$gco5||cx z@=-YTU3klmLWNpF#<@^cB?`H^^A9}6YSA5OjVzEfbUQX5)Y4f=co|E}$mHry19!+S zntd)PVs&$T3Po-B*6wP?#JKB$lJ`$1qHj^d)-~E)kUs{6>mhNU#GMK>Dv38bu_b1C6fw{CBmO%1-d8JIpm*Vq zIZ`?J$vp=Ixd)B1&1|<2j@lGAn z4-dwYpxN&c4f_u@a%Uwv*%z^zcb-H6j_=XC!@_7E7-n$Y1PxI!Vj&;9!WHBROZU!T ztU#1I?EIl>HPbP0`Yq2PrW^3}aRPjINJeLY;#g|!GN%1f@&*v$s(C#beoQVtK0*MCSR%^x2q1_})YP~fZ z?ho-OM>kXPhb*po&5vJBR$LE?Z(92PpdER39)cVr1yI9(>`Z6N85f$7J*$$r2Va*# zj>HfVwRcNt{d<036RLWolcUzjlddQ&5r^%dspj)l}82n)2;J| z4}8W#B-c$uU5ld2VTES@R&u#357Z&Sg$r^ox#icZR+EBM#Gx*zeyvwo|C4v)dgz>? zf0%5|bX8Yq%ggE-guXjUe;DN_2%TKk#HYi;^jd#-V6V|-g^3fz)nFssxp7}A1zopy zun_>qvX?zaHp-w3cGQp6ccL`xu0dA|1y;)^TO9A9$Je~ziG*kOC+RbB9+O2Ic3H3_ zE90tt&^XLXEV9^DyV?!TUIfmhV4}2Bun`-1EEHEd4?Xy0_i8q*6J)<=!1L6;Xd2eP zc_Rqd8*I*#Z6jJl0)`12@L?k$NLY|U4G=^OC(Qe4fr&mc+K+O*Rd&52lhGGRLvdV- zV8_-xiT6!?S>ZCKf+d($=Cz)wuo4ZWNt)o?h{qTkn#0v4K|>*$Sv#ucVK~a9r8;ub zbjq5wMUdc)&T;^^*&&x{|8a4ia!*(o-ox^7DE@ZU-00*?WaLmu&I{e8W}9h7#iwU= zwLkG>{+}?{&o(~!o3K7Iz6*5+yf-{I1HnFOqBe3k!3%fik?c=y2>b`Lxy|@)I1iO*#_>+QGkIh$LJ{<4Vk2 z2du~5VSaR1+3SEuvZeb8ZiEUFBzlx{-1AFf2P?jNVx&7jy8;J2I-4G-WKw<{L4k<4 zvp5^Y$xU}W4d|{zVEV?)D3W(gU+U6jI68i!O2BecwLiN3!DM|$3_If_95XnRQcStQ zTJ;UN6#0MN%!H`$X&7a2*f^6ErUGL#3;Q_$nW%y>?FUK+i4dRse=vprgDm{dSt08U zr9?ko016&rg2l9K52NyIibC`eLhYvQB4A9@Hw(Fs`EL-%+<-SUjf#TFm;W2Je`6N^ za|eQf*-m@;udRxIYQdpzG(?;#HC6u+0t@q|OjZd0M1Vq}|82zdCe&#&{|uvjBk=zJ z+XOxdW_$FAr~&_9aJg6{bF@6 z)xp6*rR5ys0)$jFfnJxk%5*eZv+89aqBNQ6AN46p2+=?8SAe;>xx=A(%XX46dLj{j zqx3}1c?3WmtVb>w@ub>%zw7y#WpgjeXfpM`9Y_!o}1JpyXY{DQ=U7j+$4T;1s< z4LDQkvd>dy10~>AhD0;TG?}e;XQ&>jfHAus;_MGn#3z92H#ndY&sA5?l<3mhSv*IDO63$7ij~N zbK~cu{uL;4qR;1O0FOVV6dH4P8UhJ4rCg)b&RnPzm196`VFc}Xb`STO%_;!AP(f9DwNj~87WT>RCR&tuMgG5DPKC_Q2_cxnlR<`2ge z5c_ffGLQ7U4(z!v`bE9Hlsxx)@xd{z-b}LAIkFM|$YUiTn1FS^0$!I10X9^F+cT0~z+sNhhqTx)f2 zvYM%mM!5K1BjYA7Oiz2!|MxbZ=XLyQwoIF;kJvSzUcZ%=M z@?o_$Y=9UbJyNq57|d1H3-+9meS+bQE{Ac zLDG8OdlN*yJTpaG%oOuTR<<=cFS)Gso9BIhAnFs=plN-5DS&s`h*rv|v_&a2Hh@16 z{x5#@RRk7bZ!r@V%3Vlwqd(arbISqp5c(8^FZkkmSqGjm)MsE?{0+2)`Z+>Oq%G7R zYRsuC%idI;tiyp4^+K<?qn<%=?9O6AG#-E}y}UV0csI^ge<;_U9jMz^TRYIn~$`}o!^-RL*BL{)9gAY9A3R650?Yzn)ROKU}>nMGBOy=(%eIS z6R;@Z&}&3pTqTnr!C@BawKT4up0>TBBUk<)vYoDvK3c9&-W|(q`;j16hq3}O6pig! zZg3ZbBdvob{Cr^0u8l({59864V;PRC?(`mQYmK6e`q*E8xe z*A_E78zAD*c(3HI|M6x=mT;R$+-KUAxds{=nj0TQrW5sj=&3W@BvgJcCME@%0~|vw z3#o&?(i!ZiuLKP4C-f!E@rPU*2MZ!v+98^C?e;O-F1aXB$6)}^9c$Cn#KiEr2aYYX ztjwcX^E$<>IcivRG+!ZlwyE`30xf-?L-fN6zm&uug5;RS4w19q6sCPxt}dmR3^_R1 z<)+gXx`IrXM*g@{ZJj*>nfX;F+Qm5{aF{rPBe&kVxTD;54_O@lF1v4-+nx8a}9LJ{Er`axrTeV zAv8GLr|zd9d)ek9(UuUbKqVV%-7LM<)Nof^d%}V58r*--CLPx{b^Tgy9ca_sJYQ$s zw~I}IsO4+5!aP5Ae5shH^MW5U|M2(CPR3OWR@Fq%A=g?jeLuUe^r9@5c1KJmg+f`R^ zn)9ya#1=OVYuj)@0I}EMK*9D~1Th$^^oas}e~YnsU*!H~ z$@Vh$)wp!_Vd*W+c^BV8mIexvD;X8WumpaqQ z!D*f`sFD27H<9liIJ4ie;ZQSQ_PhGMx@udjmg^N)-JT;0(vw2J!Cmtl^JH zSf3LAw2E86TICaS5+GGC}*Ap2>X_jN4(*sug`+FN8blq(mt=z% z2qa_<{=V(;WN5-IG;r;Nc(QsknQJ6#t)KY3`QR=d#hk=j`;XX`+hw~58D~V6mz8^q zz2Yq78>fx%e7R18(n%wfQCdbO_C5Ny_D@Kn6B+I|pNrJN84G{#xJ8!a-M7?BHCP&s zDu=cMF$AMBC7vn3EjK9yF7t*H#qE;^TR#LieW|Z_NIeL6%`zQpn;#ZSF1F5ycPjdb zW6M_>d+(aeBOl!3nqPXM4*jH{BxZh(tEi<{&nC=8>j$%RXEk<#kcigDlctMHCcz=k z(>7{E{c#P1f_(l)96pN_J>unOrM(~ughaDhbcKOq{aES$+JyWGw0LtaJ^mL zbQ*IS;!S<(d)BJsy5ijXefERcy}sKyyl|@MXQsw|GH5*05{+_xKie1|xm*f+bmR5m zhlF?5F9&8BI69f%PSbf^IqCm=yFkQ3I7Vo08Lcha9Z8jJ>AdT+ z;^*8OBFu4?t3ub;+I)Sv4{~F-tHuOl4c9PYc~6_!c6CUtwt6FW{1J$lUx!WQq_E{Z5wuQG~0Yyu1@C!X%Tbi0f%oE6yfuYZo;(_*)sjD%kg5eBi}_qbpDTW z)sloFvWK5Kj{C0Xh6s=9dd_)SZS61h=Uc;)#EFvg4f46f756#5^_S(ma=y+6k?-X= z2X4qir_Em(9lXG?BeM)8O_@@7Tzt3xslg=CgoPIls#4Qlt(+$otlU?Kh_7#!uB3eU;Hsgm_#R zb2AtZZmE|fxB>$Tcd21M5Z;#1cXKpNGCk?i@zE{2Z|QoD?6BO2^qdcM;ZsX(i|X0j zTWQ+SfBl1|995fyg6^1%&4kI~(}O}KmMcl*-mTHx{4_aaTF)~0bcYp*5>jk<;Snk^ zWhTpoBj^2a#iH8jhxN5Rq+SvW!puk(4FBje(Lllce>|*TPS7Hq1X`NYr72mem zaB=i+B%g)Yn}=T-heVln51TdChpLD;Ic);*R{mcOOJkSH`ugAol22GxO^G z=zH(+dlj_UN!M8#%`Rd*K5GC9Zwup$LXGrez;F+Z|2Pw^m50|(xvAq z)nNI!@!{lnsV-5H$Xl%Ug+*;Vq(r5NY)s&`RGP@KBZn#I@p?{wmp&aW#(hmGH>v^!{c9l-GL?wPN!&_dhOp9{iK+#hCk!v&7s~C}E%E-iLWV$gd+TnS`&oSgX?~90CdsJJO%_VrN@}Ov0Mh|p1n z9&}yvrY=zjC)l>W^oIIzvgFRZhP_;@y}qcoKDM0>8;0VI@!zIl;<03A>o30L6=*Ep z1>x$UcJLP5a#MMya@M+W_n>eG(Z^8=63|b05X1q~Lldt_13^hS6iU9$`^Gj6q@p4H zAY!kdoBk8iMam;AygxMf(0}!{1r5G^Kmhq)-vXFjI$b7m-;S2(=G{AT%t4j>LqG$; z3ldLU8{fyKQ%iERALZV|v&Vab|JALyr2fWZ5wU4IzS#$8ptAWu%j&i(TN4M2BLz(1 z&}~~-a4FR=KA>pZh8kEeR!7S)wuYd7BSrj7)l+q9m%82dc-DI!(_(ZuTabFM=U9C#r_u6@wul$?Oan0OdXE2JKVhs1( z`QXU|gI@K#hU22~oB9K8$IgxAJ_24!@5~|_0@i~B`^38z`5OL{z4bRi)YyQ-adkuL=A`(W$=|WezfkU+VmmYG$}xN2lruuLbT;Ahv1- zp4C#;$(~hhQvKQMaaJhZN3TP?Z_Es-XUohW)EaeEDUTnB@7Zr zw3{U@v!vKO4YLKj?NofrJwM4{CsES+SG`>s9rLEEg4S7}=ocG2#B@9XtMtsMpU)SL z;fBDr=bKbaF&>wrug-_l(PkMwRHlmWH4^IsNCuZ0WXw9oB`gh1plA1HnL~c>-ouG5 zlfb&`g3t+vQ9LhANAeWABs5oxbq$n!>a(iL6k@Q=Px<4_&o_HaH3&;V1}(d~SH`Cr zMf?G{w?^XO;#kgmSaP{9kgTpvCVj6LadxyL1C9w5-ZsZ@7a04)Zmb`_2hhpTMx!&o z;$e+nzQ^%EQz;F`PW41L%)aT06chs$Bg}Y?9);qWZjlkdDbXP=9+{3MGw{zevh-)` zL25h6FlIM&{5cMTd+;F4qi9^qBs0ZoY}ZZbp=)ZI-+VdBYYM-Jo8jA0%8ssxmQdLC`qG4y@YNUY#*#~T z&RQeWU}%3uKJ6O|En=tbar2tDYCjh2rps5otjqaUpd)u|_qDT#}zO zjxzsFjjT(yV|SF+WO)wbBWz0QA(E_mb@_;n`%AsA7@*tDYW{}q+hoBsSw#kb6IEYF z8VY||poW=k@wks<44XTt%-m{YGo3@hNZ6IEy(wFUAcPEFASB?mePOU1ny5?P#z#kZ zb=$&CKJQxV6)e*+=YB0~DQFMeyD*o5JIJ5x!G`B59g_m;(MVg0O3b1dfB*P9Hk2-b z6c3L5N%*THeCs!9v6dK2tms~;WG6_zy6Jj&c>LZscs^J9bC&m>UXf0v&>+C5 ze!YA}(}~6j5X665RF@hRcxW1L;QoIQp^3 zH#iL2)von z>behd6sN^U=qvSAXem)CGYU*)!0UXNgfcs^WKRQJ-s=T$!h*#f?n zvaql}BpFAtsF`sZ+>l>sAHc9fPGNq_nHux9XM{kL_KX$k{F^Cu*T#*t8F| zfrIh1Lx&tCD~0Z@k(Z&WxlJwyw1&sjiZ(v+?J5}Srpb4~=e!AL4e}6TVWL&e=qVox zDZE^geOC$h-~V36W-)(kZ?>D&0%!}W1A64!98QM%NLodHUsp0B=9Jdy3Q65GA4*Pm zG7u{Ifbj^?(sI(%o5D(1Mpm^%!(x8WevYFoV6;J&NuVE5e28rQzv?*es3x|q52I40 zBuGa}0O>(OP>O;96_p~0g-#$Ks2~vtEp$Q$rAnkoi(LZ*>Ai(sL{NH{b_u^3sIM`;cr`>F-a|JSUQg>uNUmqsd6y75l zVahhwhpYA%KG~Mwnlix!`h&UM-17{hS@5t!GK80DL&-M}^%LMXW3+FWgDc+Jr;f`*Yq*?vY0mKLXR z)ELf)gKVSB^>kk!@6+jHjrNF9nsORenT4N-2)5lVXm~a1cDpFEtiUaZoz4*Zvb+>G zfj6a~G{sYYd%_{fM6vMv&ID-3CUuTn={d9g;A!Sh+rAFfuP+pKyauW-&SkJhPnMJ| z;qQs8Yu}ASwMVJI4Z8)0NLb-|5C#A6A&G~JZpTE~yBF0f#qtxVziM;%t=HfK*UTNL z=dy)j^F*yV1HY^)c^d_M4>1@F2B=E`F70MjPv3;dMr<$l4(r7LG>K{3c+&#CWNeOE zTg6JHV}fPAYiLRfB`cUM!R1F&N+UDGs&DeTfKWO|J~c!flJ4(AWuAxOtWwt3O+>dC zE_P^ziDd7L24wOn`OasD^N)>vrgy1@I51lBdhGW^%kzDLfrn^hZ^5F~l`Z81p1-qw z7h|tqo`$T{rpCB)^s$3hWj1fd)V7sTB9|de&eu)UY=QtLlTJgWtt`KV6$yd+2n%wv z@gdQDJ}uVh)gT7D?Zp=7iCzPkxoA{Jv2(mmzyIRVmWEWG*uR zKIT8?RB(CW;*%QNuq4~HsDdw-%=KYQZUrD#;;q6jHjU9Ko1m!KBoObN)-(PZg_Ton zvQGn}(GvL(;H+~3EI=JW5j0w7b2Gd{>FS*YqZ}qbNFW>M=D6el)WuV-3a*k(X#ffCwlXvjqMOGVP!+R3%C8%GA)ei%!kr(+B3Yc1Bs2a)wLC_=G<}N3EwUM z61K6`T(5_r8X@^Ei`I0@`N;Ebl%1oNNW?(=bWlW)lU3UA9#bG?ER|;uQdQ zRiZ5UVtvW-c*=y!Xdr1iyAF^-2e&pAk4yPX!QHrSbI9r9hu(ZfO}~OTQ#j1h1V5 zBp~SbL<_Whs|n5eQJMRE*?N@h&CxO6jhkXano=RFUXye@^hN*{c7RhcRZx`>xT*_w zJznBIY^3_5LlHOAQ0dI;4eu7j+w|(ZOG!c_)FWNTJy-mo&J!AgIfoVupDss~16f2s z@&hCZixH<_QcIysM%+KVku4U9l5Ifdum<7 zgHeMe-{vN6&1rX}5PMRFnaCTVQ#_*ChZ)-~?ubZB!^9|$XN?!DX;S=~*(Gae&)2LI zFVeFUvIxa9Zo@I^v&fv)C83SsFH5L6s^m*PS1p)w@zut0oJH~#Ac{$1%AxSR_-^oU zYW}@T23aV4?#dQAtE&M%Fubo zK5FqGJfn&&8Vmw33K}ui60W{Zi;ov+e<8nfO&U_e@c~Bcbe(B$O=#4e!Vbpv8Ry$8@$Y>-R%fJEDPTJ?coo&fiz(hI(>_xq+l6CTZuprm$8`GCvGism?h%MK_ktAh$5NNYp)D zM>gUQbw7o#KUq%Qbpe&ucxK5e11gck5=7N7gfg~D-Bttpto_-h0JI|@<4;C%V#}pl zm$nCNJ7qq3P*jOUllXaN&BX|@*f_&i*XYCrQ6o*!s-<%bwB3GR>psc~nSWzKDcEF` zW~uIg`ymvB?umxGbx&h7NfG$En_9`cw@NtR_ON09PJZXG|9ns&{fO5NK@1`vzfJM_twPlG>C zsnurZObBq&GWcV~@C;6>2SWvptkKSBLxu?FwBmbM=$k&`M9&inCMh^)INvWkf4^Y` z*BGVKx|z`Dm?sw|)X)P&{%p!o_>EHz?v}Tz5(u5XOKl*LjhiqK3;3?R{XnN3>(Is1 zEZCiGDKHMsyKw%acI1L-4aexui30-;R(_7Ilff$UY-OR18JRc2L~7P7;tp3QDI7#G zfC;r)xvS_l(dG3F_RB1ZHI{%xJ5f7s6pyK>*pud(8eLRKJbOhuhX{&K^X`s->fZCj zPBm96wgYH4bawoXdEXf>T4dFh(O@Auk*p@u1F-J8(eYE+x;8L*T0wcj*vel#;O4$A zL1rUTGUK6C^lI$C1^)?xKXx)S{{Y8RxiD{)jMc zoeE4o5iQ4fej5NHthfV%PA0@*hq3l!`*T}Qc6>|GKO>R30q~_)OQj86UzHsCFWKfT zx(Fb0+S`>1ef{hn-BmZ@1m9D-+2woI(d<0}z`|v+U5DUR05b#&%!}FLQ;Z_cb)oU->PzH_ zhPb#*i6F*fjK;}30OqvX1lOL|+Uv8YpDf#UUI+JKzmmt zIP;qp58Oc8OfTe5q4L+B{%=Qu2u&T!6sG`DO7^vQ;x+&9qT^2fQI7@@*6mA8?u`aJ zUht)UeBqt0syElFBQOQn^NCcR=gsSn)E;7TX6KURYrMzxN;4B)a_HUPoQF-5H_jSB z&IKP0^ems6AMg%8hpcYBk5N70!=|E?IytoW$?oT@)9g;uJwE8%9G@X$StH z7B!#bD+IIN)UTi)V?uI{+<0hB$&<0O1#e%s7KYB?D5|4O>36#AA%6SdGWQ8kodXd) zW0Y$lUNY)!0Jm%Q{kyOKI0rqi4Uukq(}K;RmLYq_vtPyEt^O*ilEl23MrIOh5H zH`$NX;l)gSfV55);Ij!jn!vJMdE11ffR3?dVL(!NxEdDYdGE`Qug*)g&v;*VpZo5;sTAcAm%X%zE=?4#dH}jVOj;vbp_K}(yrY2om&l`_} zZJYIPpojG0{xMpA_rO)hn&c7R%Dix+BEb>I?KgI=A8fLu&be{XcCfs*lo!gvtSNTM z0sg^*>5KGy=k9=QY@E+Zg{6#f?(|mM*Ojj|G4aF5iWl4%Ra{HU|1v6n{DRiXvKwXj zVCG?{Q(#`?Qu7?Z=fRxpM4f0&P>W0tqDRJ8lV!oVeQ9hY^pu$QtlQ_tKL50|vhDYQAL zFLvz1CjrI|^(Z?vp7*yF&Lir6qn7#}LE6$`HPU5LZojJMFJRP$h7xa0&X8o=r@o6P zL}Hf+((H6j%T%Q%=~c`nJ@3tV8LHwDZ&vgfYgmjRq$}7KlU4n3Px;TAg|VKV^mh46 z-v5UKzb~I{^ilj?)e-z&n=}_)m6p7D@zcAX;P-lZd+qvjEfey8!H1^y+f|6YqhY}t zpeKxHs`j0sjYFj_i4(DZgyw%+#{ac1YXT@`<{VX@a3ljp!!%7t<&4pqDd;1w(Kseb zZMI93&cYpf}LOsd;8-qx?eo`aW|+5 v=aEjgFjLLlz(6;spPzp}f{tIEI6PKAyI{n%m6Jhj6iuV8rGGW|vNh&Eu>0n9 literal 0 HcmV?d00001 diff --git a/img/license-banner-highuser.png b/img/license-banner-highuser.png new file mode 100644 index 0000000000000000000000000000000000000000..940a40a33bc5ec823e616b8e8e0eb66f6bc62069 GIT binary patch literal 30571 zcmeFZWmH?;*Dne*I24K&mtw`GxKrGn0Kp2B7713|i+hVp(El#~6ciME6=iu{6cmgM zWwWbj!5+-3AXuqC;ViS~GDN!pk0&MZv| z@QjA)-O@@W2?i+tQ@eCP>+^J4d9GH%lT<^OkXm>{QNVWpxg3Cl$0URHkBt^h0mfxP#meoI zaK?fC0p*`t?-MqD?h z^en3Vs|TC_Vr+VYc7%!OQUAGNF>8n6=&{k!GX0Cr$AoO_*JIWHNXI`N2vvhI)C;`5 zKm0S-lsnae$<)mmZ#DnL`yu}qO^=QC-;I}uZF{Nf2ss=CPsk-HM^=ABHSjrx)#AJ$ zKlosN)+_LzNQ9%l|A+~aM;u)G&b964RbTPG$X=!y&uBI)^2om9z52V5?0W(DA3OZC zy$YPM7s>!AcM?9GDshT}Y!XDvB}A$QRz=p}c%j&+qxkjUV?X`;nd+q+7> znq!JI4Xmgms0*ctE1KC3h1hN4-)MX(jYg7j+2KA$cC-VaC+Tt<#K2N<35aw89upU{ z@rOUz;QO1H%z9yyIB8d2E+N2Yv;4@OJR!m@)Fcb&US1DbQW|NFOOy-b=cuky7WMiB zcD)sukJ$|D$J_Ju7xTuDf@q_B#cp7GHkMPX^58}}U*>IG1>`es3-Hsddv){TS*JVI zU6{8q=_Gu=+xMbK_hg;lZnEST#laUocRx@}%+G4jQBzzN`35N%Ej2d%gJvrGpl-n7 zTvK*#FYeWx&4FybY$pFisiU!RtF2lH%JW~RxJ5rX-*wP;+DSZ2g4ztJp6k?pa-91% zFO7Dk$f8|Pdmm9cuPsz%K_sdHRvb~eR)`@CKYoGHBX1^KgyUt2_6=Q?{QBeKdrK&O#kyM*w|t~oX&?@+a5k*^L@}Q3V_v(fT~;XZqa5nr=i^N=E@X=Y zjFwR@%!$IDxBLO?N*^aTU-y1Wu{a1eoT$9gNHT7z zycIlLTC5;DV?-rVeu>sM0nw}=!u}vbOZeT*HT|Mzy15UReS6`hZ`RAUz}qfi9MB$v z^cBy?Sjmn!*e<~)hki<_QS&zmersP)O5x2*#Mm}hk$p7Lb6lV+#o>U(<4Qw~`SPgK z%?e`#U--U8b!oDok&xpXfch2Jh2T67U(K7IGa4~f(VS18YOUH{%wVgir})>ti@vq% z$7^2#GZaJmpaC7`4@L=e>2iDB$u>hb9{lgR{4+J+#og``I4T;SLeUB`HD5L@hns_Kf;FQ8?D>D@z z^9-xv(&#a3PU|YD^##g`2s5}VEmj&Ub0OR<3wt>1cpJaV4;F0t`#*J%r08!w6PmKkK%WAUO9&WHD@UOee8VXRCfJWRCL~nKy8^8MwTyo7kT^V>hl{ zB`_B8({G11LrA~5?OO2L?wi(nPA_&Fdce|=!jFzL8M?v!OibEOu{$TeV_?b4cA1E* ziAZ5$D}C>&+5!x-RScv!W;B9x{A+EZ&m~t;uGN>0G$xLOK9!L@i=WQHWC?(=T$hxL zhAwcT?Ud86*D|MjqlLa6`;Z8w!wBa|=Sn8u8T)wP2V%}K{F(+@D6R2t@ejYPkEo)1 z-^e^46k$S4Vl*5RAep~{mn&dr;mopN6b$I@ha~yNk7~z`P_b&fIIOX}c=ypnWtDcJ z&R&qB#@`qfCh-s(=)bM*6O>4b$q7y@H8;DNcBc1gc=LQl{pol*hOPuNBu2cxA63y( zl`r=DAd=AEQ(wsweMd*zUuZ)4ImpaIPLYSLw06s`}ItNZ_D3s7qR;GyFNelZ4DUNV?u(<}t;?a4bR`qdpIC zN%hA9XX}kd93BCF6$g21uYnvTHWSURd6@#HzWSAn!a{Eu@9uxU^!#Px zs1Y!E!$?|rOtwG6k=F7PJxxQ&_O&Yfl(Y!=5ihejF!afndJzY#E-39HD!v(nUh~tQ zZc}Z|rcWRk()k+K0a#K&U6LmK-C-BTbs7rmi+S+{-YZf8QWPV0r^q_Sfks@v-D2;` z{?Z4{O&5HV@0yk^n2<0NO}@meHJ*!9i1CeYfz`h#ff`ec7zZJUsO|z z9a4vy*9{_as|8PEn{(V(UMSI|IWO3cF#{6CidBa#?5!RZ036uoj)emjpbAZ=y7Ffs z(8t|&CJ8GhygRV6uZ%<7K6Yzz*%I}VjjE*2MB2rI+8t+Wp4G1$JqnBppT>R*=~wWQ zJUW;Tn050A!b$^vF3AZD(&FQjUAFoOsUahRmX`|Y94cr39S z@daSsc5yT;vOk7N?L}OXb0w**f_0bJ!m-tLZm>`L0LLnV4&@&DJ{t*dE80=-b1o5P zemDviG4pi&Z|2+QXF3Uz$Ka`2_&M^DnpXuppU3?x15U>;;7cqD_i?Sf(Lkfh$0x7{ zwypiD-C+00B1ch4;gi66E!btj!@N|&{kxb;(~3J=#8wy-zWK%A-npH1LHH+TqRv&8 z#`F@yQ~D$4)G{~`r@h$$bbCPPiAbYfFV)VfwAzYD$Jn#p$asP8;+gJ=e4_i!!|{P+ zu;tKq{s$EshR`Htb<>WC!0k(XFZHm#RqUHrvb&55=%60?1e@_T3mW2h;lQ`um(2F? zaQ4;w6=Q}+K5JQTKc^Mt9a8MnR7bbVyA~xp)3)XDgWPvDhF1?RZhoZHxbuoS7y(33hrA$cfWY*k;~2N^H@ws znnwj zi@k9P+D%?S+OglE>~drJvj=CR3W!Q?EiU!&izQoMsgL~!JkSf#{oX9h3_CW7ASPc} zENMn4L1VhmlpkW@X3?`7Rd7@gpS-Br65`wPcXvb=?eA+_ z8C2<@tuEE@%5f^Q3*!UHJDQCIGjO{?Z0ZM=%EWZ`%+V>zH_4o4yqeqNx3Zo`K7&=I zCb-k~E+3{!C%c$TMG)jr+AI!@g-@m2KbCRL+_B*&)Y=W9Qx%C9`2sd`-6yqg4yzTJ zYCV+o1AEAnhGZixcyM3eaay+rsiyH{a<)u4y$K)kDDvz%oG2LB(*5k|cfD5U=a^RG z^!mWlBv5Ii+4;IKnjc=xQ+yyxJ|tJ!6VsP$eFE{+=c;_R4xWd|rP)iSmlr4D_r;_q zfdlTjk%K3h`rdI>Y{4xR==!d(-ypah+%F$z44mN3QmalUJFzM5_+I z&aa4m4Vy?!RS6+?MNpFc-aICp@x`O zm{!JO4<%kSphVAO9WfLU4=5yMS?7E18L{m(=G>8`!i6RG`{x#EhBgZeDZLBHBsP}+ z0D)BFv+m6scF71!cqDeM!xYi$T+Mq+A`D_Q$WCLO0`A%C>vvAiF0B~}iQO7Ba~%O9 zn;~M3KO5>W%GI(CT0DpCbL4vyh13I=Bz%%oBI!tJJPL6?t+JpMQlrW9b_5Tm-Cs~* z0TMD35Txo@pz0lflO_}QG>4vv`GiN#`@Jx?6WDKJr-e)|7SBCt4- zVuSb&qU!BVQ)j7c`40u=hhiFmZwt?fi~PJkx?n1^5o5WoE~z()~K4trQIBWg&70R>Z%!q@)`Tt zr`s{*^-hf)a=bvOQ_Ip;OFVxBbGfiN^=}LDxT|8D!UPAdcefH9V6zPjJO^3E+4eQB zj$WImhX}Z*qc9x3d%ejmdc9wv%(+-^_HAA)JsO5a^P1SbR<_Xga*T+#0Bs@3;c$m| zT7`Eerr{F&oqtC8i6sqsO-pJ*p0^YPZL+xe0c|U|#v4TD(ipqDqFw=%F;TQbJ0bb4^-3xc^CERBxJbeY zld5sOx+D2s#=BN2HA^D$jSi|#bZ%#>rifKr@)v1Vg)~_wlAKrR)*SCi4!H*S^n9O^bP^8$oqxCz-{yJ_>*po%XnT0Qn-9>7%;s?(o1D*cwi>(RjTb~{r zmjkWs_{Ryo-#)e`tr)7|=FTnPOSlV|U^>gjP>4tk-_W_y1IPL4-1w*LjFgUJSXntWj7M4Hwyc)g|VQ|?v(1#mgb=BOX<*i-5*DT86Q7HW~h)&SaHHA}; z@AYuWw4$ZR=4_DasBexz!_wiQ3mR4FlJr_>oL8er?O4AMmdTore!%@y)~K zTLStd^V}Nx`XSzDcdTC6oMwdGLp(+WCGYlppd)!Fp>dSgJkJeEVsJ<#i-9$|?C%43 z2#xyvAEb9(zHd!b91QY1bz|~5VtR^N9Ak2PnKlV*34M6oTTm`&@|9Hw_2p>e*Ww+ubo2b!cnUIzVAeIwP>u~VW)=7rMFYL>OzO4@3% z)n_A5D?3w+zgAAo1VE{1IaAApctLwYV)B$wF~xS@hkoskWj-x>aTyWrC_3dY6YnRQ z&wTjZW@Vnm6o3iC3UjM@55MEaC5l{sEZWN~*uaKazPC&-yX(Opo&Usi;CC*-jc>%A zV}C;d^UXRPXyFf>Is@wmvY-;N$phGUNeh$v%9CWu0iY}IvhB6>$7zC$7q^d@p#)^4 zN^pWZ+=~G`PEX6X03B56A4@KJ$t7;Vnna+Rf!SHpp5J`eEAz8dL?!?nR~lS~MT z|=QS;5c9xJSdj@TT+-L=Az zBZj^FP4gSZByYgy14mLguc-ggREHy5cZd5`&nk*2!uUpdPFI1%gnnZGE`|!I)_Qa`>H$i zSEhw1t~jVWzHygt8*R^HFf$svHhEdfT%^or&W=MhNp@bW;@h?695p+27m<*iC@Pmo z%gj<#IbZgWlCeVL4bakpXL12>?Spf<R@$83&R39_@5l+JJWmVLC*N|i9< zx8OA|eBU(CjJNa1;f`T*vweF;=_6~u3{_G8$xR>unTD^8z#+*T+T?t)F@ z?`izxW#9uLgN`vv(ivSYjg)?R)Xpz={C-(1U#tv&>93kj)+%-i($bz*)|rEB-9drl zzE6@A65jjqnQJ$>7HXowpW^)Vdb7jb??yJR7|38x7wdaSv}COvslONfP6r;`7=5Gl zLjbLxmm>*dDQ^wj926SC{?7S>vth5Fk<47QoM!cB$n9PKm`yH&OLe!vw*wxC;%$S} zf@JA9_m?{{u}fXR-l~z|)P0KShIwH-$(E?VUY%?q-vAw%-6xL7p3D{ojTz`UP)EI(WDRh9oc%5u(w6qqD7(gARCNSl(oI#f-ba=*p z-c)pEHA9KoTgeHiV^oM8R;mUO5)Yerh$+o`wkfBL(0wKo(u;F>aW+1_xf#2nedWxv z`3Apa!uD&HM91|nL+e3<5|E(i{cU=YPV8sf?qBJP<$Y2;zIgA65*|diHR?Jla9%~` z{r>Q7*Q3o}_|~(I=RdA&X0fhTMWmqHEWTT5?#7ZIKz`>&8uZZ;dhvHvJCE)Su5f@K=?ob{<2`>QS!Z0R>Qf^RGetKMi_l@kZ5li7o1UHNA6MB&@w2>IK*+{xH=HtT}3Rr0naF{hz>tKB9Iqy zMRgw8!1NZr3r)pfi9ZmHjm_9h_gJV4V^YtxV#Gytre27?w7GfYEmcs+HXj~*Gk3c* zsL^Pz#_az&dD{>q=L_;9ykUoLI@HycT{M+mM9mBPef=I2jfrJtm?BRBK?xJ_dyfP5 zjuo3*9T1w6Sp2QipJ_7|PzWEp3|;(X2NplR=4@kO$e0!Gs3DN+Y|9Xtk;3;<26qZZ zo&Zn-&a<~T;))+}KDv{eaz9EJ$~@`kY4fs}s_(v|lK+^?E<(S$jQ$Zlj$VR61{7|) zYvc#2oOe$dfo6?7eT_5yu$RhhT_I6(&(Od0DFWgfD&4CUcZD3cPxaSg{^bnkuiQ6_ zt6m-Q_qG>VPNi4)fW`Dm<<*jSyi47_Mx?azG zds`!zWoZ^S2%N&ddI12MuH9SrG4H<&pqqA`2*~XWY~=1OqGPFl!fVdJC8v4l5Eww> z^9oO1dCQo6&H+ZUc}t$EroD80DwncD@a21B;t?mv1|S18nE&BE?w){9-yh(1l!4kcYq6dLlY>lwNv8>b(3r*O6|igaDAomVVxE95u6;* zXur^8orsxmV`ek<&H2?jQG)YQt!Dq-Iki*-29Rr_DpP%6r%G`u8-h?aAVb&4!o$E7&~V(4+-n! z?1x2Kteo7hhWt4Fj3dgr7e;>@-qXhbOKw);igOJ1!)_Ht=rn*m)91(x5GX+0&ULQu z3V1BR3_s>jW+GfBF@`BWJ@PtWRgdby!D9SDy7!)dOc#=YoAu$F56k>3-ABnWOVA zvhoWUA()BLaAaA*$spC$jSmOW7r9lk2t*X%@kR(Pq0yM;O}VlP%K_UCdN=73e)+-G-}yvV2L`Yo5>8b$x6o(-wd_#eKB7e9w{ z#F|^I^A0l5eiB|ism=(CT0SA>nO-b_r8&RbWt6T06*%yZB`&$^wpE8;=$IP&NwS=A zrVa@@4EG4Jr$d^Xu9*=O8!J1*X_O2n=%na;nP7`gU1zj%Ph0Y9IPE1{$9-jYU3kJh zs6pm8=oPME5GT7^8fu;*`W&jXH&m6!yN3;VpD$}lcewXyt%ueBMAD#-b#Am~1>RTM z&E8SWj2hqgfBJ9b_S#p!r071%e3Q@~7bb=M^M{$hxGC8lDO7q3)|hT@QH$q(HrKhm z3x0MN80~H0dIrr)KKRV>P6F$O%jT??41C;wfBfy+QD7!@KKOWB6EPq4)CG=FQ{rDd z-#j+a0#1_5+Zdp84kKc|C{7si7^Mqi5hj`Rh@;nK2umeUE92kdpEg?ol&$Pz&DA@F zs^44))@?tTkYh4#x4X-Qe%;y4#oLf0@Q4J~$th$n?U_hPm9CHse8NQ|LjzrVh@I~> zmiDQo@Ht@g9gKwYh z<3>-tOl8Kx2ZX7nrr6Bz0j3K zc&x2qvGtdo3sp?CM%-{9DxxM(0T?B%cF*_ulV3;5V1jU<&uqY8Jt_DUayYTJm+)9l za#61Ph6s-NO7c_gba{V;rj8=m?5nKx)2iVIhv|7x_Jg*tGi<)Ae%zVL77zDpg9itq zwhFpw7gDRsdQXfVg$m&hN8cT^5BdF}SvS1JsFfjCHCS9XYRzLxm7G|uG#yHfGemGj z5r$b(hCZ3QgDcDp6P1$*>X~<5Mp0w-l4PVFP$MJX6G=pT--S91et1kXC3q;W*Pwso z5Picoq_J&`b8;t_;sDb?&d9)}#arF$IP}9rBQ?Rsq0ry4=>{$u^B)a&8 z$4Z40UNl5$q>l&<9g%OQ|7u@1dd(j&#S<{w{+3ue1-&i^Cg^%on%$J_5ue|)m0XZM zm_TFdu~bB(+bVw>%kcfHf3eF?kAVa(MR{Y1Ajc7>9MecM@hd4@Kq^7(GXRzu(C=+_ zjIQ`q5l1;ADJ-Uz`-I)F%o!$T(($@or7i%jxZ?Zr{5_#<&t;|Y)#+eIM-`g#<%)`_ zkZl{G%?75E*Wn^ENS(%;Cf|~?!blhdS=Q5h1S0D${59iDdp2r+R$|c4&z5;c+8Z!% zIofW!X~POo0LEUU@2@m7pGc1Ja9ts*d`Kihep5`muO@InI^Z7$0B(Cj+@-$8jD|mc zzCpA(TQ--iwqqOyw;rYuVj_JrQyk$}yMrEy8%bXtYGsgZaTC=EGw zH0|{XzEs;e^n1zr>i9DeT+NctriHMDiZyc_oYP6is6j`nvl!eExz@8p)rlGeRBvl( zKS^wE>P8^EPL6%2wb2(MH_}sM6KP(!4MF<#Ac7O((|V^ zGo_u|#$wA%PNgyUO2AYLG`=i_uJe>Twx5~s%M#s6eQw}_{LV6uM(swt1+Aw=ty4}# zH#(<#wbK-k%yF5TO3zZqJE9r*o=P`Ld-=o2pSAwWdmtv3p?o8mbC$&O+T+}jr`tx( zek&2oWo_wmT%2hqh8`)wd=!VG&N&iuKt6=oiq5GlHHf^FV^Q^e5Th&hTbDM*PMXrn zj-~ZJ0<3&2Z;ZEJor`-ea;aLc#QpHd@Le$d`x;$e&lF6TXE~K$K_OE1!{2t8kSQ&V zjinolI^)SL2<>G=DX4poEYHrx_~hg=uqRdp!YH9>@wlnk>e<HeHKlnvtc@;Vef3l>7JX`rjo{-#jqVRJZBc(Bq2i`=9e#0= z+btT}T1C-+JZj8?OFu;yc!Wq`$D`E=5;(N!>YGi!as!^f=B3za2>}9(aGzQsj-SdR zfe$sUk&vE0S!HI_iJH&*pao9!t7*PIFV5c?+04UFqH{024U}CXUNU-h;4Z~9sgz^$ zQ5u=@0nJeOz6`uMMXkHkd|fHT&h3<*I$VOo`Jn*gjtyl`ZR= zz_V0>;{A9$WTUX0scPT1JjxS+P>f9<+rYJaH} zcro&W9ETWvimnQa0nl9V4JQ&x`7U|OzPKN$*V-c3HbJ9RF7=^MWCRk6AsADK>iDIo z4#aiwUEo_s32HTx8J+-s1VpN<&i(apXTOGeEzk54_VTBQ?1DP`89uOkMGL?Btu_B1 zQ-GiU7paG}$ljlV{PCuYn@`AAt!cpVkj4X!8<=3=?5o;eY&j6x!dN3@Z_}J2ndK`u z<0S;ItSs5?OPL2g>}{8chA4L8_$Mn6s$_-p+YNjtb#*xs2{SHJOP!RV8`1BN9D@Bz zM(EeAYRAkTzMEPA4{!6i{T^4EjH+)pCG##f4h9a0`k9(h=X`N~@|o7sxY7o@DIeS> zvi5!SsuCR+9k;J~Dms;Cs(A>1`7E67ol7zrA1bk`A)DLe?D$pvo3|jYfC7+Ec}o{J zdkjdwI=6Ykn9S4}S8ih^TS&k`LSv$psubR;cL9>w4|6Dm5Ekd18(P&y;|Q;CI0KMT z<&ovdM7Mc5NXguu(?Yyo^4<_d6?sj6hQ~D8&kM)4K=4nM9r^cjCT&%;y0}>9Jx$p{ z9M!6?W&PBr=COIsN!9BgJ1zp34LBrPr*Vb_T50Z+Bgw0bDx%*(E;T9wf(NPKRo`Wf zuX%JKbkwNm4Nk$yHc*yD`fb=@Z$h^eVOrgsUfDYEJ-t-wgVRgX4p){hJIj@p z=9hK(zUI0e?N1hKFwDSVNJ~>4r{7Q2;lJN*GanNe;qVCF-dfdNh5u{{pM}%oq`u8e z?ePGS*;k>qKZzO!DUi)&sjdO!kjOltlMXG+2dIS+&SniP^hGQ+)RYWSh#i7nY+q$4 zXlTdbG=dyr!7{{<85eaS zLIRz1rStpvoyco;!1<4p(N@3)2_Uva?V>n(Tu`c_kzU`@ZDx6;ZQ~{Xm7p_SUmNWe zU);sj9xmRXgqo;<=IP4)l2G+sM`%OFllHD%ypqk;!C2)C&uyx@Hc-`%uh9HU?72hK>Bruy^ zKhX)m2siDZpc2Zm(_U<0m|CV;AL6s8h$8>bnm_uGZ*XdqO4y#J@SLv0p{dc_?`iUD zr^O=1%~Khnr^5`Il3yK$<_K-$zPWyO)H_%R^ufHa>sTRn35}G(RFj@Uy~F2`>#XFE zAC+f7rqqXBc}fFi$1?||sET~ey8}CZUnoda`yO!wI^nJ#%qZPWU>0Gvyfzb&wW6yC{n0>ZZ;<1 zVGqe=8!I!=95RZZUy#WRoAv5yoFQfX`-+bCl>W+u>EG2E7oX-mTjUkK2%ZjTg?b$1 zo=lX$6?D~ROtYkRHySUZA(wie4F@u6SrdI;3RM`hL8&~G6}b|lqqR%9IYavsea)9P z%&hID{q~IdRm)#5jqTMR=tDR6$ccOE9nBpoiYV(vbYb>OYN=&5FopLXW2Xwl@!>jO za*eXyN7j#HDby28G^n3;Z8IKS#!U!HV@zal!1QO9Fz)sm#DIK%u z(!kA|NOy1dD@<}RW3^JX;`uproN5Csw&?|J-slQjyl5s%3?L_mJ*GN;ipElNczdSR z)k{|4Z~5^NE;naB&QB&*RJ3WR$vu~e*Jgb1tu(AYsFHXm`n|?Y(N}vsaa8=v9cm#+I_JB*g@R1Gm*yP&vLMv_XzOD1m;5@>gj_{e z@0y_f$B-AlLlUCR8&~U4@w&Dzw@2{>TIaB8_DfWV!6_{U5^!?_L#KvebI30|jqhcAJ#hBS#xVVTXdk3SjbMrl1f^+jAGA$(e z9w+>054q@b6Ju#!B+l^@uDbiQ?`=JK?Ud{USGIL2!Ff(t<0R7&YFVo=qvN_H!qRN$ z4FQUkbs9VPT$uu{t1W*-r>FsTgCAfogTJzT4?SK7j(qoAq`ULL*EN-r&^niE*o&Q7^wV3JPcas<$?(g~Ua*G|a* zu&GOnXG8l)I8>1V7P3XWr~5(`>(zFGv5)9sK|)}_^L&K%lo$P5)oamx3qqDI^AKwIq2Ku{BzO5cFKrA4H3$d+@t0H zqtVJ#;!2p}?jk78n>g)ik5pjmtCjGOtkNRAWJ3Fr{1Vh5awOw6dL;WK=Q_9a7Fk)7 zTAn$&6_rejEP-iN2z++0hEDHckiE=n_h~ZL z!hsoD1i93`n>-b}Y@Go5R2qvK!H)@yJA`8VR@yEG**WeXXiPdj9+~l&`I@oj5VRN) zyuOn%lQ{Bv2Ai8j20<7Pg*x^Ze*30sal_CC;FiGY)yam!%Q7Ra{Gu zr~~zI!vHTHM>x&`SU}cYcQPr$$E_YA!(V*Ge4zT|1p}}40#SW7l=RA_uI^61A&yy0 zOEj}ajba&7J|+iW2t(L7=R@e=(P(2lB8axFY&NH<4H04QI?5fwWF1lbOMkA(hSSeg zZGfLx4z9l>K8{QdcITU{9=J`BCbI!Y#O4TLTAfX~+wpW4zpZvP#agzC=II=lte|1S zwM)mk1DgFNqwN%jw755TuYpriKYeE{xu1^SJmL1m3T^6LS2dau>smLbS(>8nHHAHMVkE^8EJNv%FXmku27dVVEN+u8$&fR63 zGt-u10XmYZsM9L}c^G%R4yoCTmZmkAf(gBsF{rzjwZC{V+_<<<`!;_2i(#&Z_%5<$ z(01fXYjAOQrPr=yhia%lOM_~L>j}~$owx1TTu6LiP$Rc#DqqTG>%n3sREJgO^2k!J zmREhkVTXKwPq>B8_J|u@Z*6mQ_(=S8;QGB^3s<%wLV*uKlvnL?7(y{X^Ie-fT*C%g z(7Uu(BBcts7`Q&(AZI4%XFyie(;=&nCw4#(7N!q70FVxrp%M=|@C$H_?Yr;pD=1DQ z&@7rEY%6&O{}rka5=6GXy3RU;OqS^fXKv%7R`^}W&VWxFr2G3p0$XZbqHJLS0+`1* zKDxOGxj;1I03tR^vZrmW`;uiQ0HBcPyS?g=aY8oZ=EAABv!AuU z*}+oAi|0|fE~=qbTazXV#r(B~3i;E+Uj{$DUC9uPqHGWw`)0eHmxlvGXHU|ZYHV?* zih_P_zGyODrgAd@Q3^knCdUg|YVo2JjzZTHQ$xaNlv7d6Hj7mgc9ru9iGmf9NHZ^S z@rXDU z-XeIQY;k(7Q&@2>wWU+%NeokZZ=kDT_7xj^{HJs=O6_}}#zr2;RnmVDUtv%3J4qBH z2b3-8m{4S~7>Uv3S%4f?78#{sjZv7cgC@dO79X-Dsjb;Q0U)y?WS#w4Pw)^zs1>q5 zgFu};buS#n5U(k7Q~b+`#sj26Uh^?Q2#A7XIV%)Itd>mbO4g-j<%OdEg9)M!?yPsB zvfY|^K5Ptp5lBn)A*SWOKy^vXGUPskMArU9vzS;iOtp;3PLBaB z9Q?8_wQ{6Bx;MyILiDDyr38zOJN1zGfS z(6W=p^(;ggV3`TvU~Xa2(`Gc`LjSlH_*ab61D3Ys{KJ^2sXAH+pkmhMS)6H0zyG(Y zv;9Eu-*JbR&S-s?N85E5$dnp4DwYaKGKle3{jWH-Ri*r>70tsxulkQIgdVB1@4vf9 z|Lw*98-o_igRE9<2R0`E4Go4A-am~#3W+VF=U>$N>qwDN4P;wY3r0)kAv^px2th1H zB;XIN5&!V{wB)r+>8wu|2u*IYryZ348mdTx%)Goe`@VWF}QxUruqLXd1Y$a zGgIFuxc(c)p4y2wghSE}K$sufio(07@z3O7TGc}G?;pU1|09T4 ze^d_z&+A|Dj~L0n@o>L?!#}h?{6pzk`bXv=JKX=_pZl8mzu_MWQa#W*U4;Jy(SoIs z>TwThMB{~7Ox}C?3+5qf7T{Q^^@;!rXT_H zWqgu;w4AJ;yep`EY!DaQR~Df<@p_n8Z6c{|C|l2K5E z2LJ2@fSmp1Ed6puYb#F_=ght8_DgFFJPqf$TZZ>jmK40y`p57FXZ7`Cs-AQArKP2; zpQ))uXWAD3em`-od-shfLsP1$8g$xlw%Pf}bX#@9ZtN2mPn(E-#Nz;Aqf={^e}20U z=>Sg~aOx;#W!mV$%+Q$qLK1iRuNWd-Uy#D{I|tk6^%XV~yPsK(FN$XKOF89P4Mymvjy42451Nh_#Ux5Z0o)w*klOWUExq^sxDWn z=__{Q(#qBe!Bw9DO=?wjtg)NsNi03~IVANt?ix7#hWGkkAlnBCfkC%Kb3g2S_*If( zOu;rD&!umJizb+2K$d8FKMNH$Yh(#jSK5`B%dM+uk;wrax4X^-n~S|gJrsB7`;oYR zvTK$d^udB@uw3YmikdJCx(_-!_vVGppar_+%V^HIHX(4{m0Azu^w#{kQvX4_-vXi* ze6>9*Rq<2*q;kHgUQw?GIClNJ0Mt~=|GP@j+*r8gQ;|_s@3z2gU zU7Z@!|1&Wi-!>S+Kq3~NA9N@f*ZqqH<>oSqE8CWoDW|KrgyKt zI|}@;=rc3ww$hdurto*nK7B|)3JPaBJ-g4N%e7eq-O{dP?-A1ne$+7NAzUls75^AuSS9U*fwFljO>EGsDl zXjh;15GXf8{GViKitUQ*&O=vc%y0@Pi&UO9oloFyUhi1o$8`VI<<$O#C1cYi-?}u9 zfk&6SOng$GD>_%WbjhP;dFp(ItW^jw+bnRTwpNx#Gk3)n{B|c{7C!kOBO*0CNl*gA zx?qHnkmNTWpLeh9n61^Cr9i|J{)4&bWPaKO;`?m z(IPlqI$DYrJH$nsgj`+4*w6(Ym$lxWQZ1K|4IP=ac%;EGK(t>GO*@yd0o$1WM>p#G zb5vk6@Cn8%Kz9aT2f(Cl^IPk!a-z`MI2DdxOX~sl&Ebq^N2B|8!bg_)76c6uwc3|p zB&4?VRZN4klEzM2DtBHR-bAr^^j=MYmv4+({c@>A@yy?YwwO%$5*Z1hkv#WD3;yivM#TsZ1S zkG;RcyvZHU3-U35z>;QYXe-iPD)K`cf^McU3rmee)KjW>vC2f!&G|;EElgQNw&Rb| zBjfWd7HBd}{}}pf1CZ)ge&AO7NXn~WRYlFb9kj`4R%)syesr3XUgAWqb2L3!0h?; zqNAPHu;a&6lX@n9t=|=Ru2rAhn#0o0@-@DYv6fk~xLqt1(n@qpqq?P8-h~A#29=<29P*JfmF9Fs)4q08T znyz~{_k!XECbPSIfcESKgA~$CaIXPn(T|$&zG8-vVigX4a^{#S`ubt7o^FPIJ|dug zIZy(LaHS&O#XS{U_Wxnuu~)JrH6@S<%{T>F?z!7ke+wniFL+6Cz@5r&kRs>+!%#pj zx4*liW#A3ibWriSws;=Q`)7jWuX)NYuK$ts z;i~9`18jukZ1+pvkeSzVvGW*dGP${T*SQ~-mUG*+p}AqG&;Pan>XcGJk$aDC{uQHT zMN@ghhLgh%v|;0 zhjlnSV_d?uKi7GtK!LE0(gC4N>trViKPl}`cx_A&Cki9_ucog+Q zohqcvedkhFhgBX&Poxncdph^!8IE46@k~Hm=b92V_rq1tLwcn_wvbbj>qh@zzHF!s zY{YmjVF#9wBko1Xhs!vV)OY3h{IGQSNcGD zayGu}(X5+w<5I)vV*6U*mPGewJW}JZO&W2}?{O3YHpg|LTZp5ManDcU$c=hX4E?_3 zcxIhkw;$) z=>>zgYc4D#^M9voyDCZ=1y+X56>)}cW~Y{&;bCvHMRptGOR)y#x#ofcL(b?V7!GIP z9i!^ay98?m=9Y`E9P{|H&&%0%(v{2)-4Sl!)KCd>E??pQ)!tWzwYha|77E3RHN}gS z0;M=D(&E}uycCzL4h=oU*JdVvK zM{^bNDh{n+O=-WsOntmNp7vYsXU$AXG4y03RS9~N91(q!3DE7PsPtSlYSOw(Uf ztG3`I8zDzHLWW}gRs$w5k!a|Dc#k9kDe!vOvmzKdpsRso$!F$j-L(sY%)4Hactpl7 z17A(|>qo!dpoEF*xWm5LUGqas^>ZVzdIERu=cqxCZRBQYrF|x(jJ_=9$j`RRf6hji zIV_Q@UN;R$NF2mKIL$Qn{#=LZcJQF(2wl5#)h^MD9#EoWDIFFNl_JgCCR8j(**0|} zjtl;SO8DM#KJQ~kssg-kj=H$2Y@5d{(&^G$u*Wce=Bi0jDh@1(MCkj9_RkDXE^ebDru`n8mvYVH$v<8LF zrcJ*hI)C;)>bEIxCLg4+pMU4J{p@;#xL)M};3!rE^+CUf#EAZOG}jTKp*1ine$O#c zK=#9L9U6J0FiQ&EE=Z4tzVSSuM21-9s8U%l4Z-PdbR_LZ)N4VTuVYE}TNAiR?+0TI{Hcnl=7CLGyT=-6h)d>nb zUCytBhLMR$dYAB$6y1!nu|Q8}_4qD)mPJoV{mOQlQBjsxpCm;B987{l#XcWj9ITD9 zFn3LhRD{HtA6H!;Ei6#HKrzIy2iI>aEqXq)BYyI|$qrTU{?6*POb$b+z}Gt3+Pl}5 zx1?;%!-YI{VT3dVQWjd<3x#z$B@ZDRo`V(stV#NJ)2UC!nEC7Q;F$rUK4zq*H> z(pPR6A6+<&3d6NMpK=?{H*wy#``Tl>Wt)t0x_c+iw?mq2bk&!>>gVa;;8eMyB2KG`(7|Y z@KYb(xtx@PuMhJ*Pj_^7f74btUzzUGqtK<{Ksf1)fuoc|kpDE|CwkMHamNK7YTjv7 zzTB`m&ntF+ycKw_eYctA*V0yI(mHt6&)fUVlu{GD%@}$=+v3w0fy2(uBT_LO;Mj(N zg9}T$+(-vuZDMjk=T%#Med>3LOY-a*YdxzCUNE&hTd~ntDUc5v1+Xe7F^&?(w=SXF z-a@b2@vsYuqCt&&0@jnz`1!NgId9g>Yq}*ZhMShN9h=Z!<^fLN52Mmp$!rc_0aQsm zs(TaI*BL9qaX{<^3?-(w&vD%zmxVx4_-oXJa6|}&!OJqex@5gO->AyS>PuReNwtle zTB|=&FpxI4!`dT-G7qQCnU@w(H}FzJCSs98iFsfBsEViFmxvVfVqZ0n+hA^LSE9gG zFS8?9#I0VS$ZK;OP5sTBdBXzBEw1;OS2U@W>;5!bYxG?=l*M^1@7OZhK*+N7_AI>| z8r;CG3;l3Vc|Qz$eEzOjEjJXLjgt~hdGD|=rK%5E@~K;NETOtSoCz(u-5np~;wP_2Id2u&|K59X zH6OQm;w!BCh?vf{%H>*CHB0f#&3D2rTQ-&74l8sQ{4(lX=OiXoiYZ0AugTA8;3GW3nCqWl;jD< z(JrG5m_Pj5FTrp;RM<-g^dv|}V!?O>B!+^lg2gc45x5T@Jtom^mfQ*m`9?_M+6GlG zRJIUh(0?d&02L!~pgU+h5PN%-nv0FBSfMV`MR$c`XMDFp8O%$N3q7) zVv5}ii5r+!x-{?tm-La;86kX$0n+EJ3J*Hz4(-LxdAQ1@*D`d{Xc%^y;D{Kck{pLI z8Pt*VlAw`z_q018AF_zVU2_N{qRm6c!b0(x`+|g%Zyd$EKp;OPmj=|pd*UfYFDXK^ zyP45g)gvLyvg)43XKwqZTB#swszh&e63<4(481^AT zhDA$X>Rq0s?xVjUMqDPNGX5qoPqsbl6!>xCm}?+4U_^48gd+J#2qSdwxP*H)0?(_f zQ(!=*k*th^z15rGTI_I|VlK=n%r|D)4FWh#2^1@@9Wji@v#aX8*$H=w5L!HAR6>jM zO^!B8BOi6R12#2%Ju+1O`jZRzr8r|nAc^3QGedFb+AQQrSC|lTeDkHsKnN7-!8wq0 zdg8c}{kV#-j2}t|WDfcmxQMmJ^j0l>l&%_;aX%PiO9?bsqgU=9azXOlR>G?c*Zh2j zbHH)Ns8uUT;d9GyNrl}W&Hh^N*VvMGT^P7jmKh$&q)$rQ{P8wv?3>yZp8>&_WL0Qn znn(otd4K)KuO!Y5qWFV{*FxTDkh6|5erM6^;OihgV4Ea`;mx$*g4(0t5_E^t_dPda za*IBK$N=^tMKfF^JFXfMUY=qDY3adbAo*AVbR=A?9=C{Pz5ZlPM;7 zsSWJuk|(8g*he@R=$>B# zxE<|b^xjt0an&PY3d^&DWJe}Ar=4v`9DJ7gG8f8KV<~Ew-Lm^i2*p5F;(au8fDmqp zprC}R=joId!cXOHRQs(&5+&+}q<%S&AlfC)CV!kpU>iT1owSqNy-#NO4rK}yHC#eY8pu)#o3jwB7XUKzkZXBNtaRlgj73o9^6YS_c(Y;sTe~OT;ci~=6t#{ zmFTwQ1=kkQ4ZtG@#)@4PwY?%DcP1$&9jn`K*|c}cxQd|o1!F58&tA$UtiX?^4umlM zrg%Pz(9(w_*F9VZ65p|N4Zef}AmE#?cP_uZ`ECG5F^z&t&efH+4ah9FG~RjgxG&<> z+qt$uS7v$C!3JhKNM&fntuANaW6caHM%Y?E3r$wrE{Ty!E-m_ zzcekFR~f!g;`i|_R48cic*PSj`a;*Y^FzLfA~m+sYC!_9*t!Wbj!dug^a(0t%;cQ) zRj(2bxF>xiQeJ?rmZZ*Bq?}=$llUmETGlX4|SLlgjo6Te8m7!9?M|ps+lo)k-_HFC;8E*4bfLEoI4mz!aioY@7 zMB(YrdS3|4J2c<7Ut37cD|0n^G-Mf5VW)UIOj$e}@MmX(y6;TZt1BAezXEKxP+xvi z%5;djISR9R!gaRWpxUoU#LDra8@J-gTQpcgY_u+Mgwgv#MJ~#5%vU->ED~mv2bgRW zhw4B}hK3%O--O^ENLX2-WYD92_VOkZ4n_u^cGRQ&>E(tovXr<{Jy44|2MGh}GUWdD zz^@09>9w1ru0je$1-hiMb1{wp{caI*{8@~LqGDS|)4sT>OFq{bZOxjA1xi>jt!)zB z-?oeuRTU;|91%u#;8T7r8so?NErC*1(Zi8+T>f4rk#O+sYM5ID^5c0l&29y8~YvG~*I)epL z+y$~;&V}8bs-ianduzA1qjuUJ62F1d?T2Y=l{#;a_k0{je=G{Tel}+>D8<`S=+$-^ zmaZh6Eb|LA=7PmVNo@Shb=JC&iqsVijw?BiQJx=bvbtYUtONnGWU9>eu_GcY1LL*K z_z=(hK5@L|NU@!tR{nd#TC_e#5;;HCdM|NImN*!}lFAD<&^G9;$COHSc<>13lPU~8 zsFF0)w0hX|ne}AhxNQ_$dN~-NJZjOu1~WTcmfVL+9Z}r^kengdCpzWW>e@F=VzrH1 zhUF?Rky|--dA{*q0=hLv!p}_t!w>c^G&BzupQb*-`C|4t2@vHH9_3_MxnCt;)g;$V zJAG_E3t>j>jj(~knohq8Z(Sx7Voq!fhu9pws+ei=P|yumAW;pJp=4kF*S&z*mGZu9t)8W8X1p4 z!KQ0cd(gF*MAl624qPHMZVYE@^$9z(AaHZnDAw%PWW~ixHbB!aL*Ro3WdXJ6JRu>E zhbY-QSaHAC{537$VA1IMDTl?ltWjXda0*k)B+k5SPli646^TBjyW-aWQ=fRVc$U{e zX%i^ClO_Oy&GmXS8c!1$b6TL z*;~00BiKg!ytL03U9F@d5@>MX{ImG&oUDvnP12OzQ2(<48mw>heiyuYYSXQDHitnU zYq7AQ57$^B@QR}my=ECWDeO~Fl%)3im1;sseXvL&T)p^q2*x=8D_3Fh;tu~5Ll&bS z->>u@+@Ie*`ZqC-M6r!5fPn}b&paF_>IFN4PJF+8kiv3&hg@5xsg{HDbp-pz0-L!{ zdnvwl(uR<@?^$*u)4v{Z#`HNmR|WACAN_>5E)_AVU>AI2*V_Kh`j+oE9%ck5%_kTVqHx8sidmabVW9)Yef&i{Ax3rLjWg} z*2bi+pAw}0%;Yrz)?}z4>F`vL#00>+D=`4ki#%vS5-NWtWA2IG$#{R@chA1Y z8l4iw-7*ag2$IFock$r#?m+b+O)JD2&>QR4^C^G{y^lm$EHhmSz%GyC+aVqgS$ubw z=|J06`2~?v#pccR72Y$BORqY=u%`L*G2S6$cI#}~f5zr1_*bCjK&|p6$z?+|0WLwy z*i-yuJw_KBG@t@d$d`AK+HC*e8z7I#jd!ob>szHse`+QW>$1$$i^zAkIf(**tAIZ#DazYlRt5iywUNlN%~Rmbx(tEV6f+l`T`SYEoNc z>Dz`c?a&mOZhUh@vfcW`GB*NTMywsm+#SDb8Ro0VIv)X2ZcuoDa%#aHBbcjW2s4Og zuhNX;yDi-Z(~wt<Vaa_QC?k@~tSWokA zFE_J%$_kr2SYA!gi2reeT6fZO$nCct_}1T^&7om&O9~nSv3~qs-fK*Y(T=e{NhEe* zL?vLGs+cY?Jc$9Bs%TNu_9vspgjHM}k2WKjXWlsIBr8hH{6%3@mL3jrr!|eU?a#k; zG9NPy`c(X>IC{xrleYdMAqvrSJn~wK8wU3{OS4=H*7%Hjxps4j?R9qpGeSjy%7sm_ z&p(|}YWJ3?<;tOk;Y(da9WHcd?X#r&NZYoWpksZ#=90O6Wvm~?((YHHOy_REY zAcQ*^!svbTr*{W`)qRcpsr`dntk)pnCkF2;PNY%=M3K7Z1*1_1{b#Se7Ss7{SQ1%P zVl?oUNOzB>1N@0=Rpgo}u*}vP=h*qw^ikbi8UMZvu#r??Ml3YwXiN4hi!}B&WyOox zRxQ%BGPIa>&;pRYiAdj5Nv}i8x6?vaB`&!XZz4czj_EB}wpn56B3Pi-HgLH@zq7Sc z9MlXMc4vKEz7WQFnJsmi{}y#wyu941JX zzmV34R}Bo)TSI6Z>WvJTLw}H7pjTQ1%y0e+48KE{-CpQ=w>sbBcw>hPT+}E$XQ}Pr zjV_qS8{I%_$_InoEzwaEp$EK{(fiw4vbUv}vieJjl%X5iS* zj9QK2?upSVH#Fda9tKXIzohSg2;=s+&4a^?uD30-da7uIElPEck>N}+B&8T;o${<1 zX7jt*;&OO~NV4Aq#(!1#Jv8>wr_xD&3vI|8vovzBk-^!S9%xXvD>KqWRRuMzc81oktFU)i;2CKZYJp0jQH#phz zJYe$dg}qA`^Jz%C5ZnCWG$g~V|KS^l+IaZqiVJ@sKJN>MG*O#cosund{tF(lC@b#i ze%0U$qo1t-vHCThfcQZBB5CjJifw11^4IoS)wTFA)$hDa3Z1#!Ixgos=_aZhrhC?8^7ruYbRfrbVdGAoKDhQRQD~pgD`llb+Ei zRstp=ovJGV93=R+EA zv|!nZW|}!zGrSXw%HEW&g!)dOLt8$T`-%oIB&({eo9aYeQ^2B%QllRGZwvOwdkD!siq3>3z+UXqg{3+_*o^{M}68jhpD;vuyQ$@zR-2bfdC{_%NGUVxTmmeE(o7{ksbB=_Yd zBjGp`TBO~u3)8basC>B;FBXv{t5&6$E=2Ohfi7_m2YcbcB!!F03(6Ju93gu>{9+Y~ zETFD{8<8RAN=Zf{;*VWcy@t~EakPO#{!;R$^u(gU>gR2}QUj}JlY(FR)B~(1iHE%OD+4*=ichE8RqRLhGcHgE__HCn73V9oM= ze($@!6c5`9;2)l`%;2nB@J+8A-}dqbCXS-(azp?Q9nb!;Eoaa`F!K{wKbTdAvF#_3 z1N(Vf5OFGpq1PTi6OW|+OoeYyn{HnyHUW8`lHlSCVb|S?Mtn1`YrDEZ^INXhng;Lb zJo0`$grUQ|YlN1UKj>%p(%zQnd;jTn+%#~*4}NpQ*mg=jTZ@)dwQsn^?mKX8Qw&_w zz-u87k2gnCXdtbUF>g0g^RQU*XhqyBNJ&A^qet~Ok)+n}g4!+X6I(M9&wQ!f0Jne}W_?uTX>B=6z1m>p{fgaR!amjm3{m zD4Z9vfAr#ra*k($hypr&^7e!=}dK+v#aaxE0_D;P-nlOJNw((wAdd^y2`K5(*O%KhLg$|;|{Ay z+4Fv^&B`V=bUNjCL=&VR(F#f`e-Y2VQCaKte)DMQ6nXw#IAD_Xb{|{q*i7Vaj9{M% z=3BB9TLa!-gp7{>uoXYpqqLAX!k?URS&+Py(Z+rn%jI2VU(ax59Jyk z9{8Yn)7aw@;ZC7!uiM4j0VpkRaP5E#AsIomO{ zTq-bAMJ=#-K)UILN;Y*LVvmg;%5Y33GDba^&oTB%US*CPjNX2|3bO)J88{!wS|r>I zy&iePl{0U8q2J(aZ9V(@SDIqhYfI&#Y16@AVS&;mj1)It49IaDD@Vq1@&I8PKVQCm zM#I%chl$UcNxaufPO;8gDwYC+6jw>IAcRb3D53dGl{sc52S?lLV;HGg@XreG-AdgW z8~Lw8)PYW#ar@YIKI6vek4ftft6_v#QE z>GsK>mP-~jlx^>x3l2qMg~)FRQ?=}T^+QB%tE>T#^(Fm*FSI+zW;kCVMQ-PK@nrjq z_6N^d+wI)5W`WYVk)NKc?*>XUt$V6p+6ZAI67{YV_bi3{(AJJOYt}hpNkuWWDpH4- zje+EMfT@(;zZt+81iNrko^x=FgP$zTDJvF-DU~vb1vA5+yilV~k(Tt`U-C18kY&Ye z0y408lr(KG~D4IE=9W>rX7@wCc3&-B{h58za~RYq8!ak931;i{W+ORRY}=2um< zF3fZ0p3MXDmswB~A?S9Kz)r4M`B+-aoikNzhSzX^QGg>k2kih&@VH-W=qAGQ?o%gV zNqT#;#+~pa6>}$zyHWhk{lJs#*Hc{HPWIpn1o~i^o7MFxvB^JVMhtz>oET1uoMLZXO?)+U9RdMf?t^`mzdn#5QFLX+N7veK65*02L^97CdsK6p za*`wDj~N7&29vn}H~BXHAs;3m7_$~bACfW(cIJdcVfj7w!h6wq@!h7J*ZwH9g|5XS zZJ0JN1UW!z&t}|?>ZR8~Qq8!l)%EMU^#Kr@wa{R1#*b)c5H|gJ!~^9-7FoCLbbGj3 z)$QX=Vpw)PX~JG&y&B8$e|Y9TC_c!+V42u^(Ug(bn|Gh5FsEqR3 z{TR_BKg|8h^Vs-1uRjzEzERY3E!K0Jb|LI9W^6ES=L?Q|M#~TKdo_{>eR4(_H?*3Y z5I-~fgGDG(ykFIS2i-GeF9`cY^uo#aq?~ zHfd;=3b4ZA7Ol71mvT8~vO6&Zbf`RlFjvAT^@l@r%GykDxlozjsD2@2Cfcm=fvfrC zQyOaXDVrjajO^}p9jMm-)^wJZszsd`@IFdXbml@2g`nx_CZ0a zA+ui;o`)DU<(PaW(M?S52qP)u1Mw>$0N0R#lHKlhT+qsez3LpXA6y ziA~~W>#I-bd4@P5$1shz9`jIG63%p{8Cw_$S&oPm-kuFEdhIpC?hint6PK2I-|#=q za3n99nEQA(YW}kOCKOcqr1az4YW9Q8?J}N8pmAlLP4ic^{mtSzcjYYXf^wo;MSacd zN=TCH@4Z->n8i7lOrqpiC=1JDDtKAhX8&uuMw6pII1T(u;;2w}ppuW)+}8KIl-D0U zh|FkCRMzv&Hiqv#=RJ|b5Bp1Y&lkbHptTX;lEE4D8wn3PX8!7kMBU`M?G;%M8;kFX zP)roX2PFG&hOl6*?cx)(0Ka|q(BmkVCP2akKd%E>EDni0dE3lx0VK7qH2~`Q`1suT z>_x~@P3-315?%}6wVQlW{4VlqEsN7|SD%=(Ov98(HQK+m+T`SfS;GGT1|}9R3B%z& zpwGhdevMnY)a<#LT))YOyIznL#mN%A&%6s9ngxD^b(h{PX#0|I~%_y=t#& z-;nC*anY$y70A5>i&5C|vA>QN^DZ&hG?mf>k=&@t`svUfqE>LVc%!j_Ut9jZBfffEkkk&Oa~DEe zTIMRl&)%z~tI`Rgv2&g2(eXErU`Tjs_tWsm?Q~IZww}J8DsE*XWO5mJx$I%3Nq6J~ zQ}!F z8|;}>OaF=u_9^SX#pX#LJ|JB%%X|Oip%3%Wj!>G~cI^0ZevZ7RvwUxw>~=2UNI2V4 zk{{56qNQ`8QOnI2#Y6V}lNZqEQ-Mw^;9SZ#B$=7}tLFvN2Lf%arl~|1bdo2I{A^?S(7pU+` zCrLR__S%bzuuvlKCTb{JZr;@Eyd+(KW| z=?crx0JEKP>IJZ2H}2J%^CFlM*Af&83Mc04&|V&vWUgC}eUA_Sx3T;yQQbEQZq`N{ zO-5qHsibYfoW*ua_6Xx)QUkTWVdJr(qa(_~=Oy3o0oE4keshyC9e-9=fZMKv21@tQ zp=X^2c3CuzYIT_EdNy9s9*5Efa2!vERlG;DCE?ego4aweB)=I&nf=D#e0PR|EhRZQ zajJ1V+V<(crpUj?<$q2Z3IDcVnAfgvy7yX_+;>8r>B7q9X*vzdg^_*c+&t(K^ks|B z?boAuC+;6Em0O>dm!!M`}>14$VE`wjo~jwse$ z&0URq&E+tq&uP597WG_uqYodqfuddE+~OB;$B(TWF=VOub$fQgIeS)Kd1(sW?o7pYsl0>fqjZ|890;WT>DM}8a<~aB5OSV^2>BYT zi@SZ>7_Z21$*|!fJJxCHP;}(x$v3kYzB1ux*>2?ZKy`J@e{^l~I$yj%?78)P=Z{UX zReVVLPtSw(@n89F^-nGvfIPty+yc3*1ls)Etcg;W9QQ9pnCo^NRlzn01yh$NfK6 z4Hpt`ltiDFt;G-u#m?S#=GxDt7NP%S9{ZSf`Cn&QhedMmnD@t09WSPe-L9ryZL4%8 zu_c;z1QVFdL(hAQ)Lhim)d5N%!R|Z3rAWBje&i<5owpWr4J-xTe}403cQmx(|14ww znHQh)C3JJQr)A&2gIj&MZ2MzH>m@S#zDCZ=&TH~FOlEklM$OAX`|*O{^Koj0PN1hs z(wDFK69F2|0o%^~jmrP@Y$Ob-z5$E~Vz%f1HA((^pe6VTUh-%h&m{`~<24L87}y2R z#zJ2F=dk~Gf`6_?pEb$fDq0Wt#}siSK_AF~VesJ}=kNa?$bW|B|D}lB(4R6qGyh6E R{;wl(W19K$_$`ypbiE@>hw3qk$14-WwuYz_hQr_0+1`|Se(0i6vA0sXdx{BtZD>OW`S z5obgHXCK1mPe+dc=O+jV0SHMEL1hr+Nd}xJx+u|E1ArZQHj;lz1NZ35$?RyN` zEK1d5N;XRW$xg&H+^W#!Z`f5H&o>s3oa06fh7Zi?ZTIO97h2fB@v-`|&(%+G$C zy3ju@b2In1vT{3gGQ>b2{Iv<7BcMkcEYPOHNcjG>u|U3$#f2vofcjfgpb#KpIt!(*iR>o_4%DJ5b4umL~?bwaI{=#mgOLl2}mGFa2;(C!?YPbFe*w24$!p$+~w zZf3`zP|j8?#6BUWjIt!bNDy|z3pjymZFoEH_l@xbf4C;m=Mf&f3)z$W--FDCf8&rw zAa{~1iJv;a7sAbWrL~nBKC)x&!>dOhNiqw^wf6qN)W&)DR9x zt-ueW+8@}x`jAZgM863vZ1x+^b$BUQ%Uu}s=LeSRkH7S=+e0*-(FD4PHwbHR{E;d; zF;vZcI*ka2nMDLbJf()%?XJ;av#P2%6scV?VBLf@SW2!ggA||-)p0P-C-*~ML_!&v zUZT!~l!tVb>-$EsYKt-z)$Vu59_jXcQmO*r6%KU*EtT^K8D2b|F*~&`P>Sk;Cf?U! zl|NVb$AK`_6rHSX|M})(aEoY5X0I^P=a%Q)tW--C1ztB1huVl!hj^zFP1f8S4mZBJ z(gyHyqmIZ?h>QN`B7))?_TA#jNuYh0co%EPg>H#aNle+Gv+00whQm&N{ubA%dO;Y^ zbh*Zl9Y*~Lq8Qb}i2PDrQ_8B-x|^A2+Ua$0hwv!o*H-nFr{0=Sb3_V1Gp{kS_mPI< zAQ45&Mk#d1O%w6aR=n-vxU+Eg=Z(v++{bn;>x)(Ss?T2)nj$~Fu?sg9M#!U-k7c2H z=1p55;ghN6fW-OZPZR8POJB&VKD*h0GiFcj(^FY@2ClCIa?c?+z|Sb(RU3~5?Q=5s zH7iniR&)CWhvKB-W7v^x=^KP-+X7J)2n$8Dmdi9t>jjdLSXnC66_}wY_H2g$qN>v~ zMc&YU#aLC<4trm*I_>jl#}}~_(9(H)R_Z0aSLzvaokiNJk8oIG*4S;oJ5sz-AvT?b zxHzBg5PW`Z{wRdh8~p<>4!}snqN^}?Cki)AshgWWO&RxU$*s(dN~a_1Jl>M6dJ>ZM z>CBQN9acxDszV!CG$yF=QOy#Kf39^@aIgIqRrG7ld?s|cf2v&eCY#m{B6 ztp)iKX;K}sk3AJrC+-omd)m#7qBLGXQmpw#B@!!B;gUha4cpbQ;*wI4yQRwx3XE;w z+7|OAM-ok!l1WL$4uhMJ!~<3JW-v3e>QGov=Wrt(&2rcl=O1OKWPm(P)zJ&2^@aJ! z3w3V9vKj%jLFT=GXLSK&)wv5?^BHlann^I7 z6m75^d7??&Ym4wV(N4;si~gCW^~=D2_to>JG}|1NPS9P^?Zi1fB`(ga@Lj`RVan@| za_c3So>#_0X}=|Gw(oQvE+fSamI1qTr3mi$F3HEsZS1+D(t(y7xkv{3_9TGF3#_IA z^lSr9I@bhE7FnBNA3|bd)o7`6zbooOCs}??v~yWZwxfNQYp}@e5cMIUdGb> zk>1H4>*aAL@^f=HSeKI1!$q_lEb@iY^2t5rdI~MnirAXujRD(lUJK*$WBvmv0Qt=; zJr#PUn$JGeKyK#AMaX>A3em0#qy%Z(C>m_b9nE>D>MFQN-de|e$(XMH8c3zp_`^~f zW26x(j3wEDt6qoO&1FN#4-Olj)LUuQar}!mA?KH)%WmmipW!k%e6(|vwDMz+=t9k2 z7j|&+@O$oPMWx8_k}ccW^4Q&BkBc^E%boGjv>Ws}4{Y{C7mHHs-Avxgf!c!0k#9(? zvIXIoaauTR@n}kABe4BD1^wa1kE&}Xf?R^hKAAQ%QUjEQ6#4m5mSm{oBR}H>ulCG0 zFqKHN6s8gwt#7S&h{6)k2vq#neW;Oo4$RFROd?aE=u>&pmiXfwlXzK?7``BUecqY=qc zWc%M+kIok9kN9R7-x+o1=Ye0?p6Xaelhw4>k)tSkHh2 zv3e0H6-}u&PQ+PZany;V}dLj&iI*)$_##qBzSq`i7D#JBIuT= z>qpS9){yA&=W3h6vydM*Qx7{nbe^PK9&VDg%E^D-qgJ`O&q|HW;eEL+am*^Q?1Nwb zY-w6@WFFkAi>E{b?P%5=Q?HLrn16(i9-mp5uQk0ARXky|G~vt-#lRCV_DwdF!*H2_;>)x zA-y$tjny)d?PB|I=B6Rk1M(4eE)Bst!F1V~$dirGi_$BUYJV4dU~g{DQgx)rqMUHv zOYhMw6yS&=QT|tpW<g*I%~*(@F1;*010!@#i9*HChuv8HuzV2_ba7*{gJ5xp?>UI{ZXrwm#3pz632|F;CYZylUBZB zLhJVy=LDIT9rL02dSz<03VMpnyYip+&qx{^ruD~bdryF zmJeklZS6}jsMY8mgk4U%)EDVrzq-j4Q58)#aUc48S1oD}TOyU-eIdOv^{&`TCGd`A zv2go#ywEP}0kqNNS9`fPueR{OF`wCB!g?BAd2wGPSW#=V)ylh?Ep^RjF>?tk*W%RC z#GOzoSfIFid60U2WLAgk4@-(+%czX!IuJj4g;S%?D%JQb$XX)g%n7z6(4n_VznO!Z ztQ>jWE~2+dIM!L}3;$Fq0Av>Rxcc^Rxi>C%n#eFE3N!MOwFRw z{Fd>EACtd6clo@U(^ol=Nx{#)aMM+Mqjt@~=&1g>-aDFh#B?!8M{p?0(366hrN?$4 z#Pgze$R4=9!|ineMA}+m7WU{gwEUhLb<#kj zK)5UR8vXPMVE(&*mNX z2nAk%pQpBqGNH|u_4`+it#X14h)zgLCjk_PP-Go~A+v>|y^vOLH%)~py$- z^;-UhY5jKPQH8|Tc)l}2?cDh+@6{b(>73pw2Jnx}6PU|;=4{e>1oUi$WCVjbKCJ#@yud zF~yFV^M_cv_614B26r?aV-hk4AtP;$BQ0xNlL0C7)fXWurLjAHN;VTt)tMICS71tZ z-{gXlxP&wr?yOoPQ8|!JG?6};uq^R6Rp;w{^3T*uKR#W>9mlT~U((`f3r3XB`^(HX z8*rQwdnem0kb6_&f@yLts3l7l6bHFqoj7jE>6pG^-If2oxY8>rJRanPk=Z)+o<%Et zCfk{tKpt7^Y$qRG)X9D)44(`SEC{hTpw!0Chst*t zq^3K)TuDIy%SGp~;#Ol|W#wXS2bZ(A9{$NGd&t(?dV|x#58iUbkLFN*g*t=pKY;fM zmywinKk?YClbOST2sZsNg#s>k(#u2;%K3viwiL|v5bo|X7wVpKCy);zNA{>%iRBop zH?FZ+^Y?U>V3aBVu;aSfY=Y)A! zbV|s?3vggcS=_+)@O1F!`&??g#d^J{OW{NLtW{!xOlORv*gz?AaCsj!4+Amb$FxR) z!1=^F@j1V%aAlp__zA=rCu`6V8uR*v%yQvHi`h#8r0fARmhHs;hm3TqOs=gAEpy{QyyVBEx!K zWcK9x!k!K>+crRxZ9d+!t;lakvbH`0`Sv1*#-U*(U}WpJv^9 z*jEit1eMts@Lb57H|>1xcc|6762EbCXv{spbL=A@I?{$5G5R0{##(3zS5wH`i&lfe z0J~3AWhmpcl^tC_h$13O5k##YvS5+6+P~|e^YP~T>AO_(H&?t4^=>b$ zPWWaF0Jstp&-@vofIef6p|*Dmw|fE3g`a!ta*F=}D{GMVR*!9Y zek<(z+%S?jwV7((>G&I_?Hi&)6%tFMotH@&{UM>);+#dbJ^DWLr=9sM$9r6mW@sb; z$+g&HTU4rxpn|W+etgjCeMqca$t%_RK@@$aVth=vN9j^I%3iXA_m^b)hxDxW!n2<= zecT{;o0VFblge}tBw=uX-Cgu$jL47qdJ&J&?x}G)7E6!J%SRk?8jR<{yrKw?uVbgY zLf^|+V@8M5Y6DU6H=diUM;xBRMT|$tEMqU42=HSKzj~<4j%3pLP@Rdxd(&}WPd+fH zaFbQESONKU-Lx`^0OJ^=55PIdv|$yAi|`EA>0Mg&I)~x;z4{6*wl9jrmSNM@)3S+U z1Z6?LblRAvmCtYmwK<4r9NT4np9+#q@BR?mInk)Hca^M|P$XE|5dh2BdY4AHcQ)x+ zu=H%#j9XNVQ*C7?9X2dVAh1FkO-2V8odF4K<2_bcvmV0j4ULIkUrwBNb~wd4gR+;N z%PAV>cDz|l(jlz_=eAa%l_t`fI@ML_RlFVtwYba}%X4TI7rj%4Xz^L#%4spS#k zLZK7)=JeOU1vaSyr}Y+RH+YeEQJ741UyQtYh-FjON`kH6MbBYAGnh7VR_NjsH(Z*O z9XN;u_DS*_k1I@tpinrR2Qa|^KmnZ4U{yfGx$)*n@Loe(ZGqx&cat(mWPRd#haSm?zuyG1$crf{tgAd_Sl!|8#%bsAZWP^EXk+TgAdc=I?eJ^XIK?x2^J(}vz$Sf zFvDL8gxtAJe@s%V=d(=;brJORt=`(n_!cZ!kCqwt96y(b0NM`-&O;$yDU8^HT*c@; zmk)X*^8QRTNtaauU1j6x5KoB_ChNlK7p0H_r**Fj*-4VD&t`e;4lDiU~69y>TgAjy&{iM4z$;cQ%q1%hsjT`J#om8@|Uv)WKBU zw6h5eKf0dBu{*{-s(otetw&TUYV^@Ne^L=b8G5I0dSHFD%4Y#K-U|Hrv#C`ND$V^M zJaVOV?b@04Jh|&#@|q}KCST^evb~vXqbu} zokYEXzS??z|6;cDesren%VKM#SwbQyA08#9oNGd;M;7eW0AZp{a47V#$of3W|JhgT#&ZFY6LK0i2noz3Y}VA z*9K@p&p(sfN1Kj=hk|#g))Q>}|6{TZPb|OYnHo1(=Q(PpV7b5Dw~BZHd-9&?8+hPM z(DXvl0!ZDVwYtM_h4~Zuk6Ahy%QCdvI#vE=#$O!jS8`g3No=g=?fN z40m`rRx2m@t+~b}Z(xDZ=h=a-z+gk$YjNT2a^-?34D?s*S8!OT^dSqEs++8LPQQrE z$k<|sLTIo>U6Pdf<;a}tAs1ncwKlhSQozB4@vsUi2vKNSyXla?=i#yIksmQL?i0;i z;??V25Wj4V!67~xq#u$}A(2q3FV$Z1?CFQ(fv6p~uV)YLnUq!?amqQVyD;`MAu$q0 zN#S=k(#$w6~jfGf6Rr$Q?yy( z$GufQy`bP;P1UZKeqwxwK|BG9||lfRW}?DoR#|E55IBU%5ni3O+Y%S;}d*m4|r{)iXDadU5B6m$VChk zhL79L!A3T(o`RW%oe>hXSY;Ym}4Gtev z{NmC@f?8un0%nGKS7a7+khc#llZM0!x)`dGwYkYXqoJ9=Aw~hKANxK+-!LY^*}vJV~UV;WDHrI0xM%wZ(_?HMwF^HxEo} zU-dHqN>p~Df;I#Z9c;tjT}cNDF?=k=vwLmmm0;DunHt)-h$7?vq5wR*#ndIY&i?Xd zC6v7<*aL1=eDLUMzg{jjL%Dhps(Is|n8AK05c9uEQsfjR>K9H(h7-_yTW&e`#R zk8U*KmujplFyKzo6A}RX7=-yJ%1{eyogLIPn6H^>B5rtR{O4r&vk|{t`cxF_PGctz z?zu@ROq0sha)}ECgn!Yv=nWy==mrCW84b6D=A>^JTv4W}3xP17Ng{azcWe>9-<$dv#F(?n z4v6PaUC`{au-rm)QOJU*h&zVN$prr8x{YhXVS!2AEGoMYJ6_AYpg_ zw5q_5uEmke46-;bJ!RGPHxLsI7OW=;@MnVYYD1#)OeYA}zrtDWldyqdW4cv74H-Kp z%E+()`Hi7ah*(I-sl8u?u!s71(NUmvyyDp_huyi*3jTBWR&`m%UUcj0Q7_q4vm7XV z@f=uszK4dP3ZUJq*3A~QY?=jkEUdp4-cAr|dGyL~-5?^HTX$*%AbPz9@r+)+{~J1Q z{15y3%aFciKM*KDvj|n`Oc;q08YdICY`6>C1>g=>+yFg+F^`nZalw}(ezTB04C9QALrSLIw6TwxE~_g{X1ALFZjTEM~#eX+lE!pSPM z?gr}XR*y&U-5ecx(1M?;^WV>+Wj0TEyf>@%5GFlqfUBD?t5T`Qse^w@L`m`y_T2lp zQq;NcWNun~3(#b`Ub)qMZm4fwzhbGddlSXn$%B{u9sd;3mM@z#%5*yGZcCRM-CDZ1 z$3PGIB%iH51Vn0|>xwtjn@rZ9wAhI5J0iZ|m9Xpish3n0Y;#TC#6wcsbf$)BYsw=- ziLhCPDHU3Yp7=F+m5pzq;LNAHEpR(5k_q3STUfaV$Xcm>Ny!Hjn(l*XFy$tl^`J38 z`fL_6!ip8n*lM(oDzbN#<+HV5p_!6dS=G*D@|Eg-b1wduY~EY-QJ(DJO8 z+Itm%T}B`$qBtCTIda#7Brh`3vFqHHO#&ev8g z+n#_MHmrYQL2an(kPk_pr{0z_%4newbmtr4RLrOgae}@-GeCc<9 z4hHAyTGTgQkL9WtAx_bck~7UKT;uOJe=?i1I`(vb3%>n>$wUOpCjxyI*EAb{qT}7t zxU}}?K4rNc=g)!ql($W3qX%Aw3kS98<6|wXMam&VA9qWGL1^a?(Nw8i|Oh6gQsr&&B*erVsS@~mp91FWdNEB#GvD`Ho0X%>k zA8&15^W(gMIQ{9T>_eTBV|3k zfzTHpP0#`ha1yegvQdqQe0zn*WgwB2E7LMu9`y{Ni1BCb(y%t2K-0ko-SkZ*WNS{R zv}G=*lQ?-WmIRjzh6G;z-ae`;j)YtuI=w&kFvzJdpoP*MW#*mNQ-jsvqv@>X{d|wM zY@BU-3pw9HM?hPG;q#IJzO~KG7JkI26vIF@36F?Q1c{aUZBoN-wid9;aqOta&061D zm6qQBiRlp95k#9+(I*MBe3Yp;N87`fAj(0N9bULr?GULeEu;& zvle2mW(97{yKyUbhTmpuDz5}C!)39k!>knBwyYINXMY;FR^-*N>Y73UeV^KbQ-rO~aS= z{I&S@T3REc<0^|*t9U;$K7%&*gPl;}eUl3#8*svDCf=`poBv&)FoHiueEsxV=tEvG zeZ3V&4^pV=4xwK)DKlT|$nF5E7-%KMJ(e{POCOleN$)-9=9$Po3qJ??pvegqG*ye`h zx?ox7?RJl8#P=JI$5t!BGP=J+{kHlhkfoE*&Jr6ls#KNg>k*LqI&&>86Ujcs z*iPGX+KO2sYg3&^Q^-TJtyz99MX>Kl$hR?%Ui=Zw^wB=yP<(IN5p8c~u<*kkO(2qt zm*{H`&99FjnbH#Q=6cIsI)+UVqLbC|5BVwMn!FaRu2c_@U9fL_mhZ~k#>SD6za^DQ z1JaZ{37-GOZ7qUJ*A~|1wOsdhSdH5~oT1@ng~SpS(Vqj1 zCU2jkd07+mYQ}^^G?97ts12_Q%N9wc(h4@B)*ow;K;qf^KK+dz056QtpYTgPM-hp4 zW=_VD?SWdkv}8`qsd7QRO%RhTVx^>S45?P~WD=j0}fw{9WZa<@|hA)4CVIZo{gkh&4(y~F`Y*4?H31(CRhM|A@>ck@T=>@tic4?;v z*2=$bt;Q2_5>!)r#T~6c;IcIU%jyRuvUyobAM{^$HS+trAs;G7AA&$& z;OEeFGO<=7CSuY}LZMXiRU31+N;9X+sHt7_uaYb*#1BQ_0fTRq6%|VX)w#4C0nhq(4Wf=>(w`dejb@`F zU#^mKy*Ay(hXNI#ufNPAX{Bev%TmR&=>_B(BMx9lEt#_*#0dO`-ySoJ$U-Vlfsx1t zp!)j4L!xFznxrZP0-Jy?^jpiLnYqjubRx7OT{8p(QYgO& zS?THBFVd*p@iA7EYDA)~S-k2M%$(Ar9h5n9!M|OHJv0MFAOST0ETO(I2I9W>u{DOH zsS?=|sMY{8VUyVPa5X7zctj|a7D*bCKC4T;)j=t%7{-RAoTqVhkRKo4p*Af;MIxc9G6u(=jM*m;XD+%Sd>J?gpsTt<~8s~p# zp%cAz370pmrF!G?zqgp;H)I64#rUr;eE&BGXlA&lx|`-+ae8-F0uca!~gD$jsn#`iBrF%E3(*Q^1o02_YqP` z$giVce?{eJ{C!aIe+DJhKGgW%SN+c&0slWT&&Ct3`|IqpDN){VXYuZF)&r;V$?rcT*AJ9@mJ-NkpH8_^@e5` ze1fqoe-~C2eQcCK(*VFA#qjgHH`zW>sgHJ52dNs#Hn&8u4iLJ+w$HWL3Ez}zC zEMK|*pen+p*J*S4zS$4WZDX@ouR$L1sv{QtA~_KGbQ5zb&zAWYab}yLNKm(HeW$TX zlG<^RQcSU;QeqO9N}MmlNvUfJG}?=gFF$6gwKnA(ngRtB%QZ3>*?aDD+ph5Msb6 zQkzJvOd}}*-S{y`kL7{iN7(a+2Y!aVqartOLlX+Yz5mhfuwaZSTCU(C`22(%KhF2O z$k%W9SJ6vIKt)=VC#0%0ip(}@R&3AddMdS$XxV`Z5{eZI26{J)L#+VVDoyCSB zNADl!D3^!gSv#Dro<$@(&lGUGoO_;*{hyLZ%K;0;3at^iH?eFFk?h|_;^atgWp8G+ zs*<#9+P^L>mCZFUJ&c!pUG|jwUtT)K7sg8P1C%UrxB(zK9cr>^%{$XqIUsG@O8gx6 zLQ+3ace=wwwo-nNe6gj#SsF7d5vKA}qg)yWhitvU6t!)_rEx7Z#@a@0rDB6kgRl8A zaSi$Q{hp6q_+^&xO11vV?C^`@1YbxA0Kv8oBs1h9#IqFOj0|ihw+=Ai>0jm zkuZ@h2)$BBCY^}h(ipUuy*rvF-R5$#o%MzoOK zRn2kLJBSkDrBF^t5-0k?s83-&OmE=(3Wj3@8)1edTG@pP31@He`ont^rV2N@aZnY! zl?P_fvsy&d=U7N<$~c4V=^(s+2A zs@8Y}bAh=oEzW-BY zgVicnLo5pS09)->TxefTvZh|Xa*1NCN7K_J$~$XIj)!ROQ;vg$vZ8d4v6c;^aIM8# z@%kjT3P;|xyVDgK>7*|SLva*)8*VyTN2HD(g*>EIFv^tH-K8(yhFTacDT^G%ZhPuRIhkM>3?)-=nA3kYX`SNBj;APLN zQAOT{R-WhUDTR`xptt8Zf1)7)eG-5KkI?erOz4KSS)XEvZl0Le7t>5PKV1kKnV;R$ z;hZqM^=O+(D5IM*Z1gzR9ue)goY&DPI7GWdZ@k8drn)vc&X%o4Zft&MScLK=7S=7- z3yCygl`i3X?XpbR0p~n~R;`N%z2j%e0E1l=janl~&PTH~Z1`9!0q>DRcV+SSu@6CN zB9}8|T5>k659H3mnk1Z@oc_T(_4OUM=6JVDtJ1pIu&bU_r`zp#a8$R!Pl6*H+| za@T?hyeRjwTst^F-KlBs>Ajz@AB@6tJN9!`WeNNtvh^!Z9JwS@RcF=xs)FZaFKrzr zitCWF<@e>t4BU%FMzcYG*!T4Ga$W)=@(E}rj$RRI$P1KQJ$Luka$tjIVwL(A(=4w> zZO2hohw+s=kz+o~rl+3pFY$8x&&ru@k5~;5o{V^`F30L6iVb1Euczc(6#vP%{xCL{ z;#ZLz|B4tp3MG0s)v9kqqzx2aU_lRRM3N4J$HXcKF}B;Pek<~RzkVTe*J>a;gC1kh z-aX7J{7qmgwbHX4orxk#hUiV)#vQYo{gN)>IHgXxxl z&Q<^NqU0=xo>9X|*VAdimvkQ1@*4u#RsZ@4Wa?o!uga6WehKBc)KqF{xR3GT1Rly? zTF;5!6gwG0@N|@YjaCAs(oK%9cP(~OfW zR3!WJK(wzhlTo7!#P1-Lrud(xo4o6>=$D#olQzyiMk{|(pf55R)eK9Mi@7VMjb{P6 zx&0>^5+F+;9($2{)*$M%Y>!yLzR0(`xv;I zRUJQ$9R!xuyTT!Mv7)ekx@RbZtEms7g<#y3> zCr*w(76Fao$C7O)-AbG59s4_coh+Y=mKWZB-kk8C@K#R+^! zC9=$j)SvX}McOyckdkV)28LsZg!A9x*Hq(3Y*r^nD0^Rlhg)vjJ#&|tQWfVHa@o}V zE5FA^?G^_I%T6ul$1^b>ifEq85IwF=wXS5W%VR)^ZfD2!@q*Hx@Y;Niq(D*=NJvZ%B3FT>H&7?X8!;dXa~s z4ZY)|`5)A6+D_O`kgQO9UUI6CKSglPgLV?sQj43xW0|~KK<@`*^XcN)=@PBMDcFdI zB%rsXomcJ}x6^*MM!MZ50vGZCGKl+aah6SEyAgl4MPRl(KUcBD3z^{ifakipQH2cG z+-P5s>U-be^)#$unn9-P^_&p#U>?D9zaHTB_}(0c&5AMr7HRS#%lo!O+v!5on_fUk z6%0IU*~4FGbuPPZ@2JuZ45S0V`Mf-sVqFq9IXc52iQV@q6VHA(rb-^uEOYlJmeFmO{3=@+g%MzCMF}q&Xg4eju0IlC@C45 zq6PA>PfQ$vwYG@%Q}9YrY&~$WYksAi&1zMSDA`u?*f?`sfe^xrM;gcH{T+dBGN~f& z{fpk}X_ZD)S-lMS7DU>-S3j8->-Hk`ZTCx-s~=kSve0Cp`(WV#rMqKkWxP%9H-`mj zfj2lrUxipol_w~G#G8j@Rf#yZV1+8LyZf!^qnoo#uUifCiQJ&EV1AbcWS(`+>2~*V(M9Bm?*Yj%* zqROlS{|;lZnwAH;ACReS0E`9Y`|9On_@6x%*Iv8@aiM1Uarr!(@wtUFv2uepXDhTu zLXgGHxIIfm!n4fo%6&|S;^cTro-Q6RGv{2=TnuGpB9K7|&*#s`mpzE|`{qT^upeT2 zKVH2Il97>#0^N2#(f#CkUB0@UQPVZQKvgo8^`!)Mc)E2r2f}-*>e6JJ2NCH{bnG}VJ=KXzi3|lF|P*8%Mhl+Y!0W`GbC6& zs*5u!(rXMXVLFfy+UwCoh9u)j3LJ_587jhvLPe6;xXm{z-sf~C^nIAZO?-fgoT^0o zr^PpT1i`Z)OkGSF=f-#3@St{O?QGxQ}4kY-*NKv{V{o<(J8WetA&kg5S zHxX<|^ox5|1H+=P;`bxg$==PcpW#}0K1H**;K#x5y*)TBbXX#$v|@KjxemR2uQ2#= zFVmwEqn!MebHN8Y)9HK|`$6{9gy#J1U91@uHQ){=HMOL;XqjSWyK0z;!Hyan>gDy`#CRIWPMmvLrJ>>}Q+cr_(mM zfk+$*=u@2aEMNrH@>%C=hxXE*rkl%;I!$H!MjXpa#d#O=Mv|^K3J4xUi64F~`UP0o`QqMkSxU3B z5sW{(w=N4m-jHSkmh5!Jakcp!J8!+*tkYz32cx3_@598zw7vMMbF{`YQq7=I!#F*b z!Tm1nPQXg|Gpj>9_+d8frN&@#;PDgCzqu)uUIJ={;3InkQaPrDi~nG2&RfF!4Zo>* zZKE~7_1tyMJIPcBnBMu8%Nh5FeOipSd*q5aaak@+D65|<y>J)zfu=@TR2 zNOf$4=X^dZd8se_+VP)&n*%079eP}d`Q4|7j7?b%V=<b2HF=_tzfd>C_rL#Q6sXq)c1(pqnU zVUHutgb=NDZID2SH{g&E_8T_gVG0BzT5c8sCH&UO9%qUy8ESSaq!m~fVo&V)tIGiDvz`J?6Qs-@5KG@;GG3*K- zz#!7M4JP4+FO0;Nqn`R!)%mFTcymON%m=?Qg!Ps<6^Z&6%BD)W2V5_k6(s2y!ogJ4 zMsmcZ9jueg*IUd`^BSW770)LfZEzj!Dt~19smter5PWiOAFFff2{QfX1@_abIxmv^ zppX4=4%TOzKEna}#(1uN^2`W+79wath;>8dTJ1us=|BzHs4%@F;<@mC*)e@(f&iWX z>wi&nrGO>i6lOzW3R{E1F^27j!>4Gnb9e`J zf%M!!Bo>R}EYfAgdh#qdG2L@bqw>JcVfcK_Xmz~K$#WrWXZ!LF83+W#9=&MMQL^I` znFwt+R-{~*>2`578SnDt!^(X2lhDc*wg;S?BifH)d=f69#)U*PUdjyZ9?s%%OI^bc zNgY5$dx&VttD`HF2}v9C<#E9pyRXRz&W3z&~QXGNL*|oPqx2{|2rD>G?Pn*(J zneF+mXRDKYt$dozU8nW0zpv!7txzHZ22+yMF{PepDAW?m%W1o(UAW)!fqrkCd-W4@ z1H^2J4--h;@a`NgM_SX-@+A(ZPVV4fy&d#dJ3e&*AL99*mZ^3}4H8O73+?Z;0a zHjEEHBx7Jxe!d-Z(GR;Fki~IU1%2wF`8?%c)v;^Q&bEh{<%3c@A2N_)UQd_W^egwk zqjzn}zhf?H0YE%ag{8%GIS+Xzxc8W*>-CU;p}*lAX1-UH?jCkq$&4ku6Mb#7HE`7G zyo$!~687mXQdEh3b!lrdy-C;c_P)?2**-&?bHN!`xk6Ee&3e_yH1dJ*5v485m8Uij z28Tb#!O;>XaRP$9w~sDzsU_x#tM3W^&@L*}e^ngZClMB7!#nKwUv>9jV#x9R#oiPX>w2WOe;g>}R4& z4EdtiAeKeqZ00PT!XaA`G48BT<~;`1rA%y_O~fOb!rwg})XNN}x466o7@Hq?mSpy) zOX4!W->>1W#cMdOzGVx{S_Ch@uf{LubUUa(9FUCfR`uY4#z#K@FI~{6O6@gT>}-qO z-##RdCBqZN-@bBR>CzE*OOIo`a5UFWM&};5T#k$9X0A~iWH#2L zBkk5XP46mDKFD%asgZzEK_Cmk}9BoyHaoDdaGfW=hEbl|}(Qc6*Lo~}pB7y0L%MtpUvp-2U`-)SX zKv#GI9A}^gyxKhinJ=A)#`VeXhYoKsKF}&xl8$e4N6#i%_&cThjROQ*WtjPBMx3v{!K7?0kNbJizZa_Ut-}-PU@S=nxWR9W zl{>a**?lTy{-y!yPNZOYRR4pXDrv)7*z^`5UEwDN{EImB1Uu7$tMfp z*JMVebIahgF_5N!Sum(Nhe_llu)s25ENA9zdJgqJ1 zdO&ZsYxz=sbKc7YT_U(`hEhxi>^QYwS&svGZ(tDt`sv2>>eT{8=Bv3ao*bLPI>(Pl zM>i($Z!vAXOa_;Vt*_tBcohEz=oLHZEh<(dfN4-(+pBavFj1l%51RC@?p<_9wt1T0 ziawxPSZQ^h0%tV3+jpTmhgc?0av$I5FSeoJ_;fAJj51GrnHwUw#a(@|YadwGaa|>Q ztNS&??QyPFdj~vraS2sfBgA4WNp38J${MU&Hyb{E!_!YJ*;FUDbY@>dCMP-ClPV zf-JUA^)wH$^OnRXUX63>+FHOsVr;Y@07&>97q~|;rb$Rt9AQDL$q z9b1&imSrqS_Q)Vcr5Q{1Wn>*oW)Q|Uga+ksejnd+&h@*l@5#U4>-t>J-_P@W?(2T; z=ly;?x8R_lb!|H>C$v0j>tQY9=(w)3Z8l46L@2mzRpd?%N(%Pm@~LaNtL|X()I!K^ z09o9>-bnx5-Hoc2juwUNQ1~G689B5WKNORwBVxx&!cJB*ZI1igDXe9&l!8lQI^2+nq-UB8 zvDY|`yo7ZeAHY3`6c2UW)tokLI zY~i%YvnM*Q=cNI{i(D_?B_+;unNivpe0}?6+iXoPZb1oBF`$S}roCu~s$&Pp_!$7B<5^Yl@evN1ev6rvYA?+>=9wBu|lUv|FN7$vw(|D+R^KAbbp zM5%Z0N*U6%_+-h4)(3R){`#=6QQX7jL#yVX4xnUq9Fw@PhlcFz*Z&i|ld-XwC}`3r zCorU2Nc`zPVdkE{;IKlGMPQ5`kj zrCs0cqP+;U&g7A=H5(JVeXVT?KJBf*VkXN?1h6x}vuNjH0x6q7+12aK#Su7VB`4@1 z!}c@ikf3uq8zPyBni-NwZ;`mIRtPSzcB7qFiMzMwg#ir;lVSIAE7C)EYwrxj&^NVf zzuw$x#AnY+*Yx8fUXw=OlIuxQfMX*)9DVFWzIv(L_PLRi_m=RwDd*5QeTGh92zz41 ziA%W-!4c@79Pd{S8eX3qFc-S-@MQ#%X$W)DI!AzJh{+ z1rAge(!rsWQzXgYQ0DiYd|1yZ=MU9`xNFy-r>;x!)jq278Z6FbQHE9M9@ba{Xw%Da z+mGaOPHL*E+5^f9tvgA1)##Q!^U#wN`Od|QrH9Apt&KMfgy#im8Obv6&!tmbzV@md zI<~>hhQ~h52|GyjQp}j|f(5TD6td+}3E>kcs~Ry4?{rZz{apm?y~>KR;^cdf3OnTH zzIb%Ga&M#kae##iLTtK5Y;mw8k8qni4d`-!pD3_M>CZ$oqWn&etZ9`a2_GIX7?=D1 ze6hz=qy%KUE5so2`4=wzll)yOlVeICPphuB^d=r&(oVed<&W{n$;mY|4UK`d4+2KB z0v-6sF>)m_-na^L>K}%}pkd*?04IM8!Xa4&GS#iB^cms>jtXZpk~zKX9~{y82`%+% zu4u(}EHmd`#4MX#wKrSSlKDBG}J++k;WQd-ET7M zZ`}4CYI>PL674yZ+Y5^MW~Ek#91oDhuxV*bpIBDc^^$^YN(ynAzV$=Dgy}^5JZv{G zqepILN0ZEbBMLwhBKxBj=sFuc#sz|I#T6TH5JE>v z%_vfLs+sr3baO2798KGIeO5bYlEawXBs577h&eqLS)*RLl0!cJB-%@PuW>t42i3jO z-`IJSkE>68uvI!;iY6j#nx!s_t@_&K^(O>w@`68v+;DG>C}JReMtl~{SoiN z&*gP!`*lxmsYPLx{*M_YHLV+;1LsHGo0mpLp6C2bBv-5G|8T>y;$Mk_l`nl}jRiOK z-ib`huN?4!jU(&L1fOacY>%g=NCRHi`sDeCN5aK$i1yL!Jc7s?F)()eyiSLca&t8* z@Pgzrg{WYo-C`*~re%k@7?l~D4T{-I+-usV4w-Ol6a;O^kkGUsYJqbmY?VM6w6;t+ zKSXdP^oFgkrBM=@T;qgdr8oUO0%bn1`K4QnoeXS=7h58B99n$jdE(7Gq+$i#Ds(%JH*|}sjOj^$=ni8ktthaJu}v@bodq0R5SmvXP=XmN<5Km)SXnXZ z3{SBWuG}T~8=&^5g!JF#lMgf1AfAq79wA3n4BR~PAA7Mu!u-&0Jg@;MC-&N-hDud~ zKnv+L#W6?#X|wGoVL}`|fjXNbCSK|mCl+F%-)i#4QBd@P9!csATcxOxmueChk7U<{ zlL~^wjJlcObVDf?NE|5r6$h6&?orJN+Ahq})Tq07+FJM-x%z&sM`p<&_^Wqe>sRie z28w_P!m$|(S6)Dj8FRZ&G~17eTdr|uEq{iGR{F0F*k!yJ>4($xIZb1l=hSjRi5|B% zIcv;&s80uW16rn`b7!TGGblSo1zu$9%9~Wf0GRqVOU>Q3I@W9&+W|C&22+ebxETz6 zBn0N%BS#?dqx7HBNKgBML{! zbFDkPCO@EsUd9>t?(8~)6}rA{#Zz=yQ#6OjEHOkp_F_QU!BA5;&-Q(!yByIAi&W~T z6dw^T9V|`Fy!M9w*6`DAJ#5GYpui(MyRm}(A{CP$qoh%Mkp8tGb zT2|I{g|LeDlb^;dk$s;#p@>d7MPi$tI z@4XkD8*o3LXiub7TePe+kW~0_;52o~Y>Xabkawo9?T8s*^JUqw1nVzsb=;$d#`p79 z8Sr5NX!tM6>AyzkCFbKp4hrg}-my0W0Q@JmD{HJJw2pZ|}7q|M0@%KQU2Gm%9y{6fZGepeob+4m^u1eH|;~Nu|gk=Qq zoLOjOF9zSg^Taa04f|e~8T-x!m#OKSXanD~pagkFdXH6WJKSsC2bq8Q0o<-x>Boz= zr9I~{-c2ez0~KTs>8=lNQa*JEFsj*PjWYy_3JcrKW$&kzJL_7E|5RsVCG|2vr3B>+10|IeXcMb7_nW486<+%qn1aWOXmy-yFB MpR+bDKkE|yf0__(=Kufz literal 0 HcmV?d00001 diff --git a/img/license-overview.png b/img/license-overview.png new file mode 100644 index 0000000000000000000000000000000000000000..def533cc2ede25e82625587627dfa52522d614dd GIT binary patch literal 215308 zcmeFZcT|(zw>3&p5Tz-i0#Yo9bd@GSnuvf%5u`&vks5j@As`~4qBI4hH<3>0NJ&5e z=^YZOp|=2`CO}B|9)90B_pkHbabEwsV_e2aAtbQbd+oL6nse^wy{@(z(^<~5R8&+< z8tRYqsHo1hP*KrcJwpe)5(vsN0v@P6_0%3xmGoX+0lx9GG1jov(xMUqo}ZyQMa@Zd z`s6LZO@W&0|9SqH`Zg8qe_p4dq6&AQI`v=g(FPt*zM_HK$!GrWBW(utf4v)cEraI2 z-+itngZ96l(_KCJScKHKDByv?P2I$kis};o$&Fe=@8%{Il@gW4Bjx9Q)N7OUuP>^i z+P04A-pr@O+&n|4r28iC%GPt%^S!UB?_|88xt`@g$M}_1M_2iL#Y?}huO))1*)p!v z>E66i`bziBHydJ4Bbo@3AL9vco0E7v(t-Sy%_Y4P#5YEcVM8r{G7+b5&F?G~IuD_ic0aoO3yjD0FU91qp|?x^bE^-?(B=h2Oh`_Z># zR~%wDl9%K$mp}G(Co@eEe-n9AxdVR+G(JJkdZVlYf{Agoe6Oi!&iut4tKKwTG^hiC zhYLz26vr*v`Nn!-gFFJ>@9$p4I&IgY=IgaN5DfeH+~b>OhOSJbSGdmq#WertUwum5 zgG3zLB(9Cunwe5alGs<94znah4V|t5k${pTX`Jt`&CrAQZX$oRl2pO(PF0t*$MW;` z#nGY|16*7pUE}vtu8x*^R1_UJTRdWV#_@MCLKXb$Qv0F9d#3|3%htjSzr{_p_+GHB z?R)XHNVf-nH(2UZ*Uz`_^)NSXNAU7v5<+#b<+jOdhdX$6Lax|ftt1ct3~Ya@_QXf4 zuWJq(iMKy77Px=sZ-3RmnO8lQ*H^}6T%v3(34gu2UeJtVoa1-lFoHl4?bR!PJOB2l z2>7WRE48w<18ZFhe-q~?+vDa_rHaiJ200M@bg4v!~9q z2=u1ixbb(pwS`ZKPl4`%kLuqIUp<8?c)w-T^VQ#lZ15Fe^lh)(O8#>G-?Q`0tc~(KGt3efYbji@pzRoA(tu8#)`(RTDo0sq}z zRyBXE)PSdT>CZq_I{AN-bRBrgDdWPohr`5iN)>1>Jc+w0f8+S(p`7nVY)^8HI(p2z z{N;k9NrPX$i1}u6clxpC`Nrq><^O6c^VOaj+MoKClheUrp&g^It?(=o zKdikvQpmQWOx&0huCNOduG&oDyG! zwhFOgziSE0HZIi;7LOH>t@u9Q3_a!6$Cbi@XM?W*Txt>QZ zZ#=-37q2`HT9zM^7%UA~;FtQ=l65b(KI84xKNp-e^UEyD&b0Whcogiec*b}w`nW3%0Ivzf+uf%Y4sj+F%d`AAfXk7)4r259)Zk)z|S&L;L_!_E%Ml zn9Vo?myOW@cPMn$#`jA?r{zdAZF)lN1Tqx5Oe1d`!`Na}XlM|Iy{8T(Jg8CH%5n%8 zaE-9py%E*5ln!U_*;otto9P$49qm4*FzWRjqb9W&cep`y{u_SdB2&f61+OaV(HvAo4dLTj?yrm*p>_g(OiQre=YieEZjyq>`oR zC;U~jB^M1!QH(mCI2Y?d>GB~ZoppSWld!F z_gMMDM^92TpU}-hhVyReD9F~Sht3oi+O)jmT0kfs#|r9CRTyslTgWCT)W4cnv9LQt zrwN5p+HI=}m{fxzAw&AMf%gmpO*aESq+Oc?U?lq;iRR0}cN+E_O14rp$!oUd$G9V+ z7zB0?`CUNYu7R*Hpr)pg{g|o2P^T8kBL6@Snk?0)0*Gi6Ozu3o#f(oI}grV9{mkw~OVmU9LAEwj#%DhN$>K;zG+@9uk!@kap3m!Y(teYn&@+6|r~b3knr(z<(VuT|jN~+? zTr_|&4Hjw$zIT0K-dFrPzWF9h%k(*1=ws?E?Dy%3xeXu*CRhf(pyuV5O+pwomP~l! z&v1A(i*^iLg9iW0p{1QRt=Zu5u-$J88yu2QB_7Ony$??Ii=v{@5Gf`7>M0lZ44Q!c z0cB@ijSIAYzMhktb1=KP@@ySf{W&c;&HDY_k;#Ia0q}snA5=QIj#WY)ZQRXgH_von zc4e2mVj{#g5(Bui)&6A?V%5vg$R=E?k#qiN%bV+C98N_OdLw#t$8}UMNZ*_#e&$?d z!jrC#tGRDqsVCtTx5S{4(2@>QJ=?JTnPz&8h0eIxk+P;k?=I00o8N8eo$;dMXVG7O@_Zw$~x^urT*h3{PM36;ev4q z?Qwx|M_0ju+$AU&+Uu=TC_=9wxMC`Bo-0;?HRv}-t~!ai?bfeb^LLNO%7$v>WnOU~ zKenv;fJ&UQ7BsT(PMq+Y3wr60^=HGZ?B!{U=P?ekUNZjY-(a4L-(R!8VPTg{m2nk& zvH$Q$cXc?Q%S)2!9E_nw-0ZH*tWv{TNMe`f4s0rGA$WyMKXQ|v z_d^Z;w>bME6E%|yro_UssI-*shg4p1tETP```ayek%m(<1^NqBGwsVe2^QLakN7k1R**V!??Y%;n{MrP?X&GWYm2 zKGf7Zy#c|x?~0xz#S31$^Kxsdms7=wQ*xjK(Br88J!_ru=gdhsdxfnvy)5oZmQSAc z$%q<<$@DcHf~H^pqze{0O#&-&)y(prX@IxUB_4DbuXy~{a_N~jLwI|qXBmrVlK41D z{v~ELzmI*U8f0xpHT&x_#`b)_UlCij5_n=C>@48KXpwov$slB#m%pznb~t!{2j(M8 zm++RTzcyN!JmHJI5u2Qhg9|vGpmf3O(Lh7z;NVambWFLpa(D94|AH0E*k|x#ay6Ur zqnXG=#JEgC4{XPX3S6Gc=s%p|)Z!R_ObN1k#w+7jop+&yT?fVCC+YjtN-mR6sv!73 zc*H;kycDzSGyIHp6I|jBja?bay*%o<(8Bue`ShD};?YmyggTN& zAoFD64n@=Bo3k#~gv!kBGCJ&XdwYAdFrseo`O#Jz@4|R3HVpfov(&g_>GD(y7qi=W;uhKViY?g{gIfYj&PR`obsNUA|Cwc zn?@)T*$WpnaOlY#pbF^acueuVD;`I;m$Q2=wDDqv^(30r=9YI&R(+M4w+!!q?Zj#} zgthmOCFh)Ez%y=$Hl0`UkkVknwauiF%{saXTD*MbVWIclPBk{YlABz#7IN*Dc7Vup z3BfAozOuJ7#Vk)n?=2cuPu<#;a-iL^OL713Mch&>^L~flrkGvYLp}u##Vdzzsc4>^ z`yyhZN5{h9MAJCd-qY+F&!Vue#C-KWt9SC+*9IESwGxwx??t9nPY7$HpBuQfveGQv zex4pIeW7x&nCuELo(G;krF%14^JJM?hz1TpPB+PUNbB!`)=LjUOt#Ad*N0IgHCCTG5dF!Zpn1pPPTZCwxhK z1siORgtad;@2u^llsK<|svb6S_A*VB{4{v-^K1*}Y&D9PU{#828~1gJ7DuJn2d?2F zX3K5PBRnf;*0vn`9NQdgZiiMCdwN1gAQRwS&p}>fPkxaNe~zA*0yIAbfnW zSxqORk&-kh@jXGv;K}1C9?>>m&=gC42+D=i=d=G{Pb9a$1kF|733m?{qNi4Boi{Ei z_+riRlFzY%$Ump<=qEJ6%AM+mB@(AE=W={)9=(4WA;qW&lyJBDa^&k%DGHzMnQc!0 zjCi;yleDALqHs9b(P;`{UYFc|Y$sgvp*OIE-|fzaDr8=xDaLNvzpj|2=<)_U&&Mly zXo;<)=hb+c&oJc`PVX1V`l)T4x}RdKG%Ge=pY(R_rZqjY>^$qrpR&~MG_vzOpD2ZE zua&MbEElaVbcP0gcpQ3`{hMmUrNL}9mIDe2zX05HIyp3a@l&G5@ZCn+ct0i$$PW>A z+i=Y5@wd6R)EiOcc>s+He$!FdWA@yrUFvkc;lh@xiUCXVnKTB%cZRCWj5ws7z6R{< z$q0uOf{uipCvb`eI3uZX%L_u?OH?X#78F7;+q8v`izw6JcWp(xOP zLB0X6Wd(gcb&2~a;lM7v>+4F^f>F(v#$~jBLUQjkB#lPUX$YPcU=?cTT6tC_ zO0yMHplXOc`xE9~&GNgWvy;o~f*vOMzY3Dx(muwPW41L$K-BdC(j87D#(Vh6Va(h~c!oQ{TC2#$u0)G( zb6ea!O+WJWbLk8<#)#0Mm0>~rdt1wZtx(2B{~xdH?2;*tO<0~c#~bzd`BdMj7|&i= zwO!(LORm7Jb^n38|D2wFF|K8>MSSGEI?qZ>Eq);q}lKP(KMC=w+V}_cPGnW zQiQ5w;IrDa?$;sL6so*!1IegSdcl_5gT4~(J8O!q4t;? zWQklz*LaPqjKog_7-=f)IwfM;51xBGSYRNW4^$7J@_O6S*$;Au5JPQCKCofR)x#; zdM7z?!0%DtYs!uPC9<`3sRt6p7vNPAtdsJtByN8#uCt=WPP#h=o|u1k6cV5y>RO!l ze(V+EDY06GdgMtb$82(qy{S^JnPiglK$b~x$_ZizRm=JI5+iZ6Mvu!IF?FU06_Rxh zoK$_!*JaA{^I|s{+&~a4jVeRc9A>W@!v`t$@72;p5`Ko1+;vX!IhJ*`h-|B+PYXF# zLWRDOY4Y(ts9vY9wFb|-qnVlsVibXk-G>4352R)U+a!(`5-jIG1RCv!ghs|0zgId%V*;sob+o@# z{;K0bEh(+OhwY5mk7=63!>Q*$^?7zm^cf%uUO(S&GKA;yPOpSude^>(Wb$y3-!%}Y zV9Ftxl@D!gzo+<4`QKvX?QoCHEMfjw(HZuMtEe>^@peXK`w_BCYuwr<72hF&L))lM zhHV(v`*7!5a1H(Z^H1v0?e8>T6|8Eiz@0AzVlLEJpwl~`<6B1UY|rYe zrsnJR>DU#DNR_T;JVlh4p)013ZRmcZv2AEDx zGYp#liI{Y>R^e0F_jH~3p}hG7y?@#fQ^F7Q$Z@H1z6Y3AYvquo5ISDQmuAk=4gnvl zu525qOc-X27vx>rnqm$QZHl<9Q+DeI-y#O-hSjm&}ejK;oXg>XrN$p zuth1()Zg0Xk^6Fd=K7fJc(pU|qsZv*JxNI%R#NYd#}SPk0_g$QgO16*(_)L{vNj%J zi+b;#n)S8=#m;&>D%sUynk#k7B!sM$EG;F`wC)hQw)*N~Xv(JAOLe_riFZwAW0%hsgIF|IpNLQUatGdn<{n{GYj zrLb4LOX0HJg@Qa%dJu)9;KLUHY5+juR;td1$lmJ6XFYz+_hGCj|BpYZxTeSMQh9KY z^eXC*quOay&k<7~$|d9cg*WKvnQf57xH*=yzkwV%uC^^<0RG@`}}f3-n<=%3;?%dlt+uiY)z?8yZCAFz8Ck4C|UpQ&jef{6DDZ^=t&po>gy7!(>x z5h*_FfSJsxYL-ER(0eC5>q#J*XfrZcFDoU2XW>nI@QeJ>zWA8DUlpKd;HU5fovJH2(3G<*~NN zTlW4$qF}yg#>{Gy38fexL04CtMZ8l(ecLcKYoF6kdscB072Gi`1Ril&FNLx^`Iu?Ep zY9_+!(_+qP(02Qc5YGFD%h{?bTO`6d8B< zW*d7zoM>puA8}hBt5O{w_V7L~Tqu#l#~$n~nI)4EoZpA%#v;P9RZh=_Gjj_8(9#*( zdg?xxjEe{`^9|S*^Mt*2X$tmDC7hWu z!a3nQT9C85K5oPV-+h*NwEF6}dKQ%t!dRqq6r*+;OKRAcbQv_}@t;30yW6j_V}K!L zg-w$PIAIV~mVd+|1u{o@MkL`V00Xd{K7MRHg|kYVy1djFbRax-X*V!~R;jXqFa}Y6 zvA@00m7Ms47Ox0(Ua86?{w$07#4ca$ja<#(;tqHr?j2BZ&1^jP2z${pWy#WSYC^vl z3c4*+`nE2-03&PB7XyG|FolFYPNEhLCgEV*udKLX_%!4-+AS?+$Sk4|<7>r7ng6U4 zAI8MhDT58T0b1sZ$E)8Yp3U76D#7iVaz5}Ym4$ORWMbbpk#duJ2v!pl7*EtZx5$g?1wb z*$g<}-^z-GRoB>{AC|4^15RK_4G27e| z&0NBat_CAYFv-mQraKLEYgMctYF_#OlTT~=+#{}nVZ#sBXQUdzV7FnEnCs%gD8b+$ ze!b$Z8JC{7p-tg&$9_SDxxSuG6OM5KAWZsn6C&o2>Ue%m7uRvZG*z-&x?kR%2DoKc z1>aaTuYgHB(P%tf#6qJkFi*?$4J__U@II&14t?tdr!QZ^5zn%rfC{X}IZ;isjD%To z6Z;&t=k1PyH`O_#ZccNr9MBO|`ht4Ru}?QVQ7jYlLA+7c_U(##)-@)3WbSju{hk9C zwDuJ;1gAh}?Eg45A2MZC_&}`6AHBa&9@MxaZl79p7);N5=Y?OYKYTr@O7FOTdZ3ik z4LbW?ARwwVe@=1syH@f%3E!A-(s4(Z(Vg=q4Y8%iZ|Ou5>064ptDPrrH6Cq8_!la} zg-Qy)Xb$53{ejVy;w*_j zLBw_6a$hW10v8ZM$Mncf+y#q5Dzah^zoJpFhp>Y^jAhQt)(^Jv_LG)&FDNiy3dg4J zdgt*e7N&<$oKl^h!e+hQ+6d*l&K0-WQ}aiKXz3WEL(g5J3w1AE&jfo{ zJ0g3%2oOZ&3awn^d7BEa~RVt#zH)8*CcU56tvkSWNJkER^eLX zF#h|2gHp&4Xs#L)n-=f#CYm!0LzNfCHzY9#*KrrTJp4B7wp6JLyL=P?C`P;6iz{Hb zsNOf&T2DKSQbX9xUj4TaX|Eu13BYgknn#Sea&KnQK9+oNu*l_V8ZLxJL&aoR3uRfv@Bz9&G$5Dq0W)%cc#3hcl}Y<2B7IfKe;&HVW`y( zUs2a&5tkxhmQK<*zMRXJes?Vm)+55uQdBPN@awJRIbGj&ujTZx$rb&45>#~9TyTL< z^|1A<2d2IL>6JkI6f%oe0Z=b5R9fc2>}}samAF3-5RB)^9Cu&7mBl89Pdb&1X8fLQ zxXy!(X@Z4Pc#hx$_(gE01Q~|}4Z>7YbR-0mhp1W6#7Fqi!?$66aej9-?c z=2W#!x-vv{<^AzTsPurrT6rqnEV zlML&jt_+MVm?DEA#n=OSxtlM6O~)y4z3S&&7nz(l&3#%OL(iaZ!49VHq3rE_cYiP{#cwMZ-CZ!N8GT z(pZB5V95%a*_nFAX_26FK89$3Br22FI3?+XsIb?c6=&6;lB1~UATQluC<*5(V)G-6 zRfBN^n1K~H4vEP?8Yqr}i3Vezoh+UtQS!6|HJm7-g0x$4Lj0Hb9ZI5ExdlHw=DdkX zSC81p&cyrC&+`{ zU*f&q3Swaz_O(x6+AuU4f8SzkF>wgNt4w>`uV^}KAc|#B|2D0HByvltuU!hmaKL!E zNM9MgW97G95~ZD!M%9Ql^wm;76t+uZC_t*K{~g8a`fJ7zA$hUr=iO8e6WTOdI_5hS z$&P3K2&KabiOW&(eG11*}k6v>DcJWDggcy|hhm-_jN>55qnKq}f+y9mE6#x(*c^I%{>%%ZNH%rb$9 z@aU5#zGX~y;AZ;akf9R+a%dU}e84nC-&^h~l)I@2wEi*Xfd>m&6{?UkF58gL3+px5 z_}p};X|?nC;!3w9YhK9ZBIV1Vb4O;bNPbXLvyfw9dNAGQ#*ZCl3D3&jEQmPgaz*%z z_oY1XKH zk|_eUZjC4uk`9@0H?H?npCx{(>*IZD(co7$0qrWb4YMx1#stSb-Dq|Js??$}j@DA@ zs+?{2l?v1JQ>;R)2u#7=MQU>IpY-YsrilkfwgWhW2-onq8QncyY>*h_a4a{E$ur{E)L7p&-fc|pE1#Y*|Jog(r@`Iyu zu3YZP{zA_xEoyAtYHYn~#N4WFydU4r%{ zt*Xc?dT0Vl4If=534BR32*7B5i1Y$m`KoPPOFBs@~&=Jszu;*Q?PBAb5Ll| zC9g1|08h}b?)$tyM3abUr(Mv{W|ufJMf_dydQYkx=IGIoPx(f6h}cFQPK<33y_pF% zLqhlNRrIVXMHpH7k90@!6dZ%*O4&j1ogSHmgQawq0?G+Z@SBON=i}%hQ3dcu_B5R< z%mPbrvetBZ_9+N_woG+G?lrD5j$^>E;@Uv3*KZQCW@Z2fj!GCK&&xLvXaxd9TZIg| z{V1Y$LLp#(GhoIB!kg+JunN;(Xl7BUG?YcHvYjpA^@F37sZF0XVt{;*# zI`@+5u?VfeE}mNyzf3qevH`3({$XPma=j_U0RHJ^&sKbntqxJC0IJR zEs8L8;ZGzPNw*mB%dYe|TAncUa`aX-qqN55cNh~QY`hpgL9+(h39zz&@IuJ*9et;IP&|(DevlxX!Ka6^ zEAG2(JdQPZh$zh8+oK|qTgcBfwJ^b2SM5lGq26S94aJ#M`JwkCcR|3xLc(^hT&A5b z65v0y*f?V`Zgq4rlbr$*iZM$`3Ic%9B90*;Euga_OBtd5flj}eG_^dC9rA0_&Ks-#aN#f(dvy0Z!^^v3P{FdsM{pfeEhvYw0WZyDIS%ZC&};{@lLsaxrR zg#4k!3~x?#)J~+_PUsA+aH==KgrZc=TwdBY&uwgk4X<023{ae|qqZ%5KZF9|_WLtc zdTynv{>Y=B_4e^$`c@xz!tr=*MqVnG@m_siQxJ*U%fDz?3y8SD44|dqse1<^62gSg z%NF*Q>y~J~y$EsCZC@k#pgq`Kkq0TYL08pN>SxAM5~w8(TP~$8fFll9@oi052M{bUT>(Cy zlVG{sbN`tclw5E0G}uqx@LQ(eug!wT|D>hO-pB~nAP=}zU1BnZC>caZ>3cpqqnO0@ zVMsAp;vQ_zJ;{HR^CPQ++sN7Gwv{jTbir>Ko+eB4W6Nmm-e3goo$B=}3DIo`#BWQ1 zTc$3yI32XMdRs0AwiRZ+Fy(iMHk_1EfxbFeDp*;#IzAK^w79WN{;~AZ+;`Xc_Gd%A zg70yk88ui63>AtiE43iE)pBg;ozGRh)ao@nFp&OX=w9JInMo!6=({Bq_ZcU>W}7dY zG^!&S&4Zc0`q9$D_AUv7CT-TDoHV{?-ZlQOBM+@CMW2QVU5 z4*`~!1t{(LU+2Vk`v85(33B^S@iKB@3bt)4;a!+rQHA@6HTAk=#Y7h-3<41AY?Y*9 zgMUl#ekAt}mCh7^3*ZTBOGgjbay)Jc?6XsFcZMV$Z6Vq)T3 z?|84&y9WR>(*+PA`o`fp%kI^#SQnZASJc+`#MI-;Asva33&RiTK_}GhJ(son{_9nk z?LB?Ft0lUbZ%lI2^LwG(1@=F{v|CJ^l6NH7%`iEE^hMh%x#=J1KC8sJwUy!J7LLEa z3)@m*hlIh3B_QY`U4v#9vkKc!kSkzcaH9*d@sn-1%#QfwkF`Kp!8(BEG20 z)UHI7J0R=lQ)G4J+GM?v3sCDU`%VMnLFEU#sk&zvH7xUGzCH+Yj?6(o8we6KBnh((Tfu#GQ6c?bYP`)M!7g)N0SKVW#pCAJ{OKb*&zQ5uld1xSd@nQsNJ z**sSJ3H|kM8!^>#Yj@USlKmq8u|m??2Iv4q>W(jCU`ukp@Lc^&rF4b z;`MnaSL;KUyyN|uiXa1}joy(nhYQ~I)pks*XR4wIKLs!HB1S1H-1T|R5KKubgL!>w z0^KApz&8%)dgNtMH@aosgf;n6jt(-Dc3g-0bLqZxeGz0cxLpo(50q-jQ@E->q=L4R zAbWTPE@~%eckGe*DsqP)=g}*>)Sem6_gLgBW!B<;j_td_>khmKwb4#!xSe&}-~J#FCE{PIyJTVvuwn0nt4CULQ? zI2pt@_xqYmr;Z|d-mujjp7SIQXhQZ6mNsAbIp17TT2@YlIwOy^yRs^JAhoj}gkSA% z&e)n9I{@TtrOV`4LlAM&x2K8w_;=8yBHRIBLGgg=J)TwoVo1v?F3u(g<+-Plg>nBn0spICS z$0(CV$~2W|u64lvK=>_I#JCq#io(oI^aZhth;=}ufc3vvJl;$Via*HsRP@hBy7_u( zKp`mkjKW|rg8Er;?oVHctse~28TfnRzpn*QWYtjZ+Vd{;vL8EkYc- z7UoixZv_;`8XA^+!e!$tr>E>K_XGy@WS{=AbiQB_E5?f88?V7fe_-FS4b_s=E$0ae z@5{S=N&K2)&5qBx(E1TyQ~V`+BIriFesY~Q>5`CF$v8br-S0J?b^o_=kFBspe207; zawRSp1=ElVXRqJp{S&TR0c@oOWPQ^_t*2JIsoifMp{@kc1O~QS?RM$##Y)*(ZvfQn z;VWLi_mxTw(Ny1h%Zq~DAU^&r@q}vYcV)rdd`z zpiBptHR3eI>z|)U)@&4XA8rg+!k&8G)J{DJS6uU}7tCWO9&7E&)FBV(DciFV;++=0 zll&*HC9kJ;AIMnJ@#mr{lSO#G-T-{Du7d0-ZkgHOpraB35_bEcSN5{i&EWbNIL8ru ze;DV>pY@wt=yI-dDBu_lV`zr>$(Jt6t2LKp^YVyxC5W?3TqPjkzadP%P(k(wN9%`E z_4tkvBTLb-rO!IDCY;hvw}p*LBzuUp4N?By$=qNWQ}UpKKf^E3=+}@(TRN4*ekQK% zK%He@Q+kvi1L^UAd|&+!_zC6_Pxo)Yv#06sF$lX%RgPi{LEgCi1ZxxZd8&`Pv;r?> z1!~~yA$J20Wy^ocxj6rZ!G8dNrY4oyZ77Ragh#5Hu#?l-oIlM z#>u{vmAP&oGWXuiGkGgua~i^Jryga*D>c0#eXf4!?(ncj)tl|Qpl`J7MY_nLyT+G; zf`Mp3U#2=#c0?Dgkg{_khfDO$gbefSX_kczi=uNh6MpYyXeMww;g{ppx-gbO+|_U< z8#E%FsAelk9)Ry2Rhkdr&z>Ac`(y2R8iw>X!5vi$pO{zD2vK(YOu>;s9t}^Swy|Ct z6DB8$YND2bN>I&|Pa$v&(R`zxi1OpPmp?wpF%-_oasBfxt$PyIkmwTUPrD9g? zwq8{iy zi5C$hm{`e6T?l^y?dFiM{mRSqA=2OLst9Ow>rxc)mcHclrpP|&pa zz+AGuI$i<2Wi5z(;A{r<*N?7$MA<1aL(QU-lWU z-gj&2KEm4&j@2Uzey#%!5)B~iRO$QeznbH(xu`a00F&fCp5jVAMjiwW=IZbc{JdkA z>K*@pJUrwvYLk#?y#_c#(}Fe*0`}Hj%gf83UI7zFbOV0iL*-drT?LXB>^Cw&+3>mF zpL%9!NCm+TNNm0Zz!p99=|D{?7+;&$KJW5$7kT~H#w`PHGEy2waFSj4qTlU3iG=w| zY~G01{tcsg{0%PHt<(TS6e*9^ZLvF%t-KzT9dM!{bLR$&)ymT24GwEe^I~ISWd7V7 zCW{VJ(9%em6`%%b9daW9Q3(CL6mfbhmHF}q0Y+YebtcU}1RkqlMx<-l3nou9>h_R| z*rj#O`}&?j;k?Z+AL%NG)N`Jx`f#X9QoJ^B2-S5R3@p zT2t-rU_gf2Gs4xDQnrXES zG2SY{#^*)q zR_T1B*t910q)A5|t_mBSXHv6pZ)LZwrz=5`HsmmME?kSfslrC)iI~U7ki#w}43_m? zcL{ZAiADW~-byF&u}HdEDMrNliRF~g^GsAObf7`{+fT|dsqy$TaDFnPw+L_+nVp!n z8#dC0_jc?0cPjeMtPjuB?J+CFI+71VLXg*yy_M@g&-1za<5NHq45?@Ww!xC^Ehssq zcRjG=#4dKLa%uQ-sy}J1qR(gS;$EFPVlt0qw=c3RjMG{^@Y&dLMW2EM@jR1vIzUlu zQ;w&6<&glDU|eLC56KCVUiO_Wu9>=bZNIzwDF?_a509T>>p|z^wWMg1C&u>IhWh5e zecq<1Z@@KNIs7r|H3c34xpVU2yB1swe^^)Ky_%Y+DL-|wy=3VOskoTs&X{>#TP6*MXlYSauKnPi8zDHK{Y^cu0CqU$TP7D2LE@j8#EY z(>KmL5A+ERDm!J;Sv?w$`t?VrdEU@rR0#k6EE<-Jaig`^We_neW4Tp^9 zc3ZmZK5IQwaItnh1~B@6uG!$2*Voj5RZ`*9xm<6a|xFm3Ct>T)_cjJ%efll$ZCqbq{>%MVF4R8)312~xV zDN7<^Jfc^RjBTSNyT(IF1p9;=wLVh1aKHi9Ny69DBeK`&NvdL-1T)uU>I=}~Dfc$+ za`)!PfZ2SS+9fwNFp04t(CQKWSHPT7yPOr#Ex$WFq%B2C8l6e^P3Hy+!?!yGJeo7O z9?`olXYeu&Y6n1H_Lq5ALy_nz{FX?}go<}Y)E*f~FZQrlt^?F#wlK1MxOL*!lX`pQ#{+eL6n97-|4SeIV3##jos8sK>-aW({FXm0b zHy`?rv?9fsM=QZyho>%)DZH?WGQc2mv+WT20OuM#5>(_<#MaTNu824}b91-t>J89- zUd9uv+i7}8W-d?JmJEhxNnM;VSTD&Y7;t0OB@v! zx3P6&+an3qYn#)DUt;X(X-`)he8mr;6F%;{0km6emF)OUf%WMj6jHKWfv6)sauw&a z$L;w{JN~)4T#I}Cc;~7E`!_>V)ta7b7&U9Q@}IY*Z`tq#&b+~9-H)5C=z3!rJEYb; z7~JX{w$lXd*ap+C1WV-MV8Tzq(d1!e`cU_)^7IloB;HS3wfT@U*7xv%)FaCjdbRuh z>=^Hp7HTOCu3b_~;2hxFn$vk82B3~wz)ab8>M?GlSb!Vkb*F*{^5)d6rFKcT~16E_Imo0A= z8X=@fki4Dc?5Iofy#N$n{-t z+3!)%bh5(gZy}+KJoo-|Un+Whn7CXuRQgP1$7I!~@aYxBpTynIpU`GIoZBGM28^xF zX)3FhVUh1&!Eck{ zn5pW71zpSraIEX;{xd{9fV{s#Wa5$PrR8l z^-z|6d>vTMj)Hs2rZ6&E1lZa(R=&tMkKH^5d|C4iJH0!YbU?(f)j2e}FS6Y9f7pBP zXgJueZ#0oas*-|4Pee!r3DFr6HzJ}%^p@yd^iD(~LV^%2%7_r%=$(-$6TR0lM(<;c z!C)}o<$1p6eZRZE&N=I>b=JB6vzRsZRd&68`?oLnJZNe#zwzh*ui4AK?z^#}Y1ley zKNmdaRibR}EF!(Wz=*3&sz*t0Bn{O{dx0L5uiK$ZH!bcElB%$$>0hTHox8q+lB3;9 z#Q_kHfkC;pHQgRE)e7zHTeaGN@?>qMI9g!T$ zg2xh_qn-9T=3kbBHVI|O?zl9f)37Q>nY@dmZeERc9ba;Ca!TR|+vy<2s^r1xPdC*} zx{~kDdWo&z9L&0sD1V&3&9=|~V5HdcV!0@qC`sB?TgTGx`J|&+_Pl1>h0tx9awJy< z55hyzMMmcrk8;=}Iu-mn7hfI}dWxk0)Ky7?6Y@f20B`E|Cp{0z8D|^BPO;3VT~){) zl8zpzOxYE+A7eyMYkFKnEO;~l(6%sF`fz-%_EbHl)%AFUb{!=DdOp-Y0P>)pVzi92 zc5vKd{23!`F4J|#7;f){8q7`FWLL-TRamn@V_$f5-0(0(Yp#_oJFz+F!R1{=^9Y4& za|!B&zG&CxPxpBH_gp&KZP?6XEHHj7%@6Jgek{hO*#V#Dx9{;~w`R%yrdsLy7$;6w z7vHU`?zFS+tmP*XLB&JXv+8wv?o?&FyPd7_hNcAEJ+MDld-$tEftf2Hb>k&!%~jZJ z?wM!#nKk(iECd6G&p0cmb156!OI6G|?q0n56j&(?V# zy|6W!l#;tIE{gZBW_CPa)Vg4Mg* z$Kxwq(o{5iAA9l82wx30EWPCehwZZInviMHB68E?rA0nO*9s72-#7(RKA8^*0a4FP z_D(&OWKf5NI6gAY%qOp0@5uJ%8PbgiYq!>D?g@r-s%%e(gc|H@9M8LBQG`_QiZ!zV zYl`HhmkfIabWDs+$lS6xj_2+a4NLpHHCqvq)G%9d(kOn@Oh3j&aN%me+n*PeA4e)l?I(sWk1lb{lN3a zqGd#-o}nsT%);g8BE>=|_j}M6cYb-NQ({#iNCJ6V>qj{zOLA=rxM}(6rq0;A(tV`- zddf&psp>m-y^nky8|B{C!@Pp;!B$avE)3+xUwN2%%;F5t0o)sx$Dw~l7?WBE=o&Eo zm{;O-6k3a&>kDd69KVfQ*d)Bl#C>KT%aruLs*#u|sUDD_G)K2oSvQO~w(Yd^lOUf} z;=^gj(B&l=(z6u8peuE<8ctET_*D=j8A*t^Cx+bCt+P@FE{!Bbrv&MC6Pz zB@y10OnMsb`ueP(&kH$Ot$aL>{VE1}U#?eP8Y7)ynuzO2!Y3xU>dc)Y++yyZdwn_` zk&+BhnsMJZd@`iU?WhveQH%8n6&*$4y2wb3di}W4T*}hA-Y7YR*4!5*B_d(P$7RST zS8#H}3L=Dl7Fd^3w5j`rP-iHG|}-EQ*hd16*DZ~*`jIa%Pwkc8A8kb>VYl& z@d1@tkQrE*oesqW8JW>xojh1BzQS%`4UC}A^^gN93+oHL?YE0WGyO$5282B=5Cm?z$J*U5KhtuZgZaZ|T4RXWd0GVzoC8;r zYCJ!)#`*==-#_L?Z`Xe(wX6YSm4KBSq!VONZ$LrKNvfV5}?KoDEk<_8R+Hcdz$$~Mo9wm z6mM*=5M%BGQLwAo(7+Sa#!D^6uR=7P0Yaq76J&=6-0j`@t6q89f$NhRd(fQS&2u0KB zJoQ!U;2+5eEecjY18J?#GIPBNO}^2r0@_7MjX9uSSxoOLes*vE?D0lHuN>*3eRX4m z1b3v*Q9@nHRr^FHy<^>*;Z+^XESHnZHkpJr7amag>movD)QOl&R#1?+ciY-McPzV8 z9XNk1U|p$Wf}D9?q=Kp5=4@;;nz45RUjX4P!*`YDh*mz9SMrxbh~L_|2wST=MisRG zi`u=e-1~+UcLi++`i?`(wpP?c+49M!s-pqAEiMZQ^f9Tgi@1fGY7gsx3Q!>5m?)wx8Ab+beTX?(p|Fv#c)Ln@=fEvN+mWCS z=G>yI)+#QD&;YibdtKWj*04AzIR)TkQlyHs>d$Is$=<)?{{pt>viwT~tk200|3Rod zjJ(U`#xsgbyav@n_&JkAqj8sk983Q0xQ!TY!y8;NOBT$@ksa1mr~u01cSspeV<4Efip2Y`k%3t1z2=;I)TK| zh+_&X!|F^&AFKaTkd&;6Jb2+9^CjALYr~fMVI1sy3bER)0=t>VEQ;qNuZ|`?p6Khe z&Z}w48Cd!OQq)b?gMvs!DuI1=tUp7QjsK$aK;`W?QvG)?m3J4|cdKkrrTuUl?)gZb zb&k)I&QucNPIaTi8efjwDXSn-trwR6J3*R(@m7D9#ziKTH2!+)XAnKyj);|*K`k4u zT=Vsklj>Mrz=C4IG?r?|X27fJY8V>fMjuXr~wk;dVXu0p3Pf42v-SGkqAwpyBU!CWN zuA-A=_K;I3H*^#Ld57tO2UxZ8PKB_&#cWuMb);d)Li7Croco`FF#}Vt z64+6g*mpXx?@uY-=-ju&$W=?@0DxK$lg4>m6*G6S)$uua-iPQw1-`f|T;J4uwfd2P zoU-X%^_w9A=NYvBg^-q@;viHCXG!vnVfhyiy8$m1r2=dvN$W-11-YBW8Alzgqg^SS zJ-J2CdS}FqgGT)48Wi7`l5Z72Y9^ymzNe=t}RN4;xJ3FYvnZI4uCTo_SGO(fv_hjvcBp1x~m(x{tJn zZ&R65xDJ=P#QP=ozM@=ZxMvX#2;Z8|Jp#{32{t3e&AX6N$+RDqXm_gEZjOc9x}|Li ztz$J_iImNve-d~f@xM-;|DZ~~E9x5h1jQRr_Yg9)&_A{e`oQl#-?otc>ay5(FKUYYf zZ}0enS!Uw#q#I?^b>DUw)RP|a$w3&JwaCne8`RcB{4wyG`dmbX*iVI*hIZIo)pSzk0)DRG*E%^fxOtqllCwjuxG84L;}vvebnR zeBsYiqkj1MC8cv02YtiW2B#yk;yY?8w)3|a4*=@MczP@Qk>bqZ(&tm;2vL8o4I(ti zOUhxpnykj-PhZTa6Q#y&X1hxwbi6#CLOx1c_3_OCJB}!BxO*nzZX(qU#w^{t%k?NotRAkalkLBk; zx_+=wmF}ZIh3y&>VyLzkTz?osp04KN>Q1aU=xdF0Kq;(_)1$ zX0HiPQY1oaY-?`hxGpbg%d#?)30Mt;XXjP6+)LxX41ncjIE!dOz;5H8dZ|krntNZ$ z%+(zo$#d-{y(7BZ;p;A-NVw`I2A|vR0R~nL0ItSPYsY*5)Lx=wOk%E|-oHQB`@Ae& zI<;fCOYP6DCxDA|sZ{sApP|?4XYqQbUXs(5MgFqH;?&U1BQ}CN``vpRmoFU;{tSqV z%6_1}baX;g8Wv8U`Lf9!j09XJZEom{IXLuq(pU09X_V%xrA`;d`7fC1k|;O$nNi*- zbj*Hjbr3?z`$O>=M*}n6<_-$qS3AK8ihOKykV&$r0>6!WW=P`#XvC=31CYC!0&^hfl$Tsl5^vukag1>|$cb5_=H6M%M1_e8zzCUBnfZ&KJ3@AU>_mX7u^0#KKHI`KqVZ(F2t@#1l z5irlajV{wJyk71{ABfdobH5oEFLXY>rlhUdnTX79HFewiu?h(>x!K%%k?tc1)z$1O zO!Hecu-R`nY8t#usRW#y9<{uaJ02Fkns5D79kefK#);tZ8JizWIVheig$4d_X>#Yr zAj>v{MBG-y`E@Ju@?VYYf4EBPED6%s#EntwyDkXNTyDoXCW04?QAMpYy9|ps)3jtp z%)t$K=+X`lUz+Kmb(#WB8ZN<%g$ANFS*z8XE#WGGI_TmVuaeL-#Jdf>d$BB94>s8H z?Q)1GOA*Ar^OxQCqyXkAEH!;@@7f-CBr$2ZB+H)O!SPV}ycFw`qlv>6ibz4TvV(Tu zfjoT%%Zq@si7PHH&hh&jx`$zSub;s*TqX+1U}%Nbb}(*2;2euBa??1=pZLjs^Yfdc zgn@EZ>vaE)$Ck}rKx{&P~R<4 zBry!K`w$@6^TLZ#Xru76;?z>f?q?1teYs$@*ACu%=9gC`Iqm>B`8#2=&hW!NcI#F9QCctkWSCE!_3?caNFua%ulL>gS06HMV5_XM>q)1Zdnu4V3o2J67Xk-!f=-vUU-G z3Vvv2Nqm_w+Q-GkWj`rQp?Z>XBttDd37!dPuVQDaIj(PMit)U*8XPmjjBMz`b;U(k zV+R)a&*iZ{(ZlpSdS_GNl?FIt%k(}u%bK1^yMF(OjF@xV#=xs^sVOB&wZY|bh7Rv8?;vlbbo^F)pn2?Ipu<7~B zX4_~d+oq#t&&oU6D%@s$ygv{|ojR}tViEx>%zIgqAXFA3Nkh`bNnyG$8qUEQHz6g}j#@MS<9G`Hblh^620{t=V2g-N8v!Jyu zgesR_XNB4bOpISYqF1C%Q7fC`VZfp6yQl^mL5oMx$if7kZ&boheEHI{L_TfazOxlT zE}txM8ZjtN6tHDLVi$dGz$GT8m=`y73GnpK`!yG|pi&o(WA~mxW49YiD^#94Vxv4R z1uD;SVUOYDKMLvjIeBllLc^F`hj=`Yv!#amE{gG0=0cBON-fc8mD*@0IP6@>CCQ?$ z0^#|{0KZVX_pP0g8%41qPG*9a(I!lbK}YsSLY;iG1O$__D_A&Gu)K-~g&}qR54~S1ceX{^&VW z9>SSFd-D81_1EcRY_=ZR-sSqv=o0e=4)|!Y;tOXeia)}uvn-dD40n2j*B2MRg@^jZ z*!#8{bhZnj&R^t!pq8gD|N5MIE@ot91rxDwNn(@%jqHs-Tfr%6Ej8BDZXDU4ICL*i zD?O-#3WnFrUe3k%sJ(5oCR=$1*JtDN^E?$qvDMhd=5+7eF{fPI`1%62X^u+dmPXFcA~aVQeF5s?A6~2f z+R@+q;7>t)+50I%*liusU6j$PlWV5V8$5W+Tvq03u1j9Er+Us$@BT6u&RZd_aWK8o z;KdfL^E0aIHLl{-`l|BRedI2|`ZtnO#tIah&8(S}DMt9_z*yaDO#=obCye zCqDb0Lk*SEcWj<1Xbnm<+FE7mb+!J8^IQK4JzHNERDAz8GX2d0Y+~z*#i9{n$lIYw z#|9`H;A#RwzC~7gZCM6|8tHSNm>K2DkUqIn&HmS4`r@*@mOxu(03v#so}-vS-CoMi zXr{P4XLpQz$&!;A`hFOX8uMTXR9@5-n}jm9Phmz`^ezF6Ugy-A_;1CnJVbSI!RJQ3 z&fn<%NL&9x^3QtXJMiiYGugM|tI^&I`Gwa~rfh50(h@wEILoDW;rjyIwFDG?sU~i~ zadki!(I5ba*OrWi5`SwE{^EatKXu(YVOCuw?7N+At}|1^E+l{pJfVs+P&Ggsem^mJ zvHc{oyD*P1$=rh?-Wrs1BreP)dO75^m$3c$h5aJ|l24pSirqEP^JzWHEo{_!E^4%Q zoi4hh?HSbQ)2nk090V_Xdl^ERw_;Ll)N@||K;;lq@#rmseVT9A4!Lpkm&&o-LX%-- z@S*YT2Vx&b6&#h{`$ zslzUV_MwKlpKbT@JZU3K%RtWn`-WG&Jt%|p)TFZ?X|;C}o$;B0@Kl^GF$)QE!M$Jf zD5$R7zi+*mnG+Q2UjHg~|3_iT$Q<53<^Hj^nsZM6A98IQ^6IC06#N*Eouss|iL5SP zDOB*I0c~VH27M`~4k;g`M#>VtsO8&M9gFr`Nyp}#@SNYa!_K?w!ZO@p87E||N3K4q zbVDyA6p^N)HSI|ozVDg5l-Gy7doS~u#5)HMy;Mben9F(-k&;1Gd;-W{ZSi01{{Q}y zg#?*;NJz?bb4Pcs)0~$0GjL_K;E2-qDNB_yPn@pW<=&u%j>p8ng25HOZr1KCDf@ot zrl{+3Y>boR)Q2*J5FBTS+tlsF7eZvtAMOUME}Y2v_DP1|=Jks_G>Aq&#&3r*KL71X zk(Vr1w=5_ahl$T0jK*eJbk$Atb4%Lb!^*p5^{2#0E@lZYeiXDzYO%?=FJ?C5va|Ev z$wwwxnw*U}qzGN~#~)F0hE<&G2wtOPN%Ist2x-QsHs7_pMW*SMV=d$5-kMi6Wu-rgcS$G~ zwdeQWs;qm<$R|oEY5^azCw=S&1(3>g$i_auvLqc9WZhzv5O&gdAcr97;wO5zlb1@~ zIP+=!J?F@C^&I*&)!U5X94!i(K3Va$nyR#@blVH2Ft^}E&6X6${q#j1EJIndls8Iy z*tSgHXb|NZ`n*m&s0MPfdLj0V1Pwq?7f9Jt@0P$qDW!+ zzgK}@iS8eG{83p>a7wMZYK^u>&SJ@}ma5#KHJ-S=(6~Xrw=|)f$BE9a`3D9w2K~V)UUm zC_DvBcnaPM{EKMvS9j*WB=a8yvO1OZM*MM7@fXMPPGmGJHO?td4ZlL7gZig+qF}Q`8St!R#p+Z|RGNmURWlT7k4YtTG=Z zyLskRLwbWsSjjz6XQPoMVA2={SuBs@sNh+C|SkXQw?)=>FmEds=qs-f0XjC zcIY3a{0qDDk5c}Pn*M)FDIaO(XnmesJ-p`v*5|a=Rfv#K?q6P(KQi2`roiBR;;36a zyzhz!zRS)|=a4bG)Jv(B1K*|ma^mpTS?C}P87;ncxOg3eA)gMykfbB^hj(6f2hOX) zQs!@N+V8jTCU6Mp;ztAjv@Z8&XXQT%VnZjV-r&Eoi+|{5p3s6g#Dnad@}Y&TI1Rkl z>^GV}{MtVg^x(s(w?UY;cKmYEp@nt42m+CIMXp0T_lMfRxzC)wBYtRMr*8mZoA5$#*IO<=b3e4OKNW$n|0w34U7>#z^KVr3AI1EA8vB3z=Ko>E+zO_3`wthu z-yF=pEXF_H^N(VR|K2&N|9jq3_^5u>X6PnHG)?RN#5sv)buT%&xo;-CQvb7uzdtcE zQblJ!2@+3v2-kJgQ-;`|0Zr@{xkYmH~rM@3qXk-gAjl-F}b@#)wdDOl8F6F8becw5kan*Nv?a=M-asiA92 z!{&;z&9!Fy>vQ^@tjV@#ib2(GL``2^`#k@cPi^V@1#GQK6|_@~O%4;uZNK|kPB*9R zrkX)WV~Av1XiUyI($1%w+=8Y4e{u%D`xpQFmi)B$@7d*nZD_bHF`+}QB3D=xV%k3} z^g_9U7??J&QoFBlVWnC%;58{sT9Q;DJpQs92e2AVCt0e|M#c<5fA+n9ykP&`gAmEm z_ZF9CV>V18yl0*PVgoqcjfV0QCs9|hO#lE`Gr{B?ObdFf-h)<$DYfw8Y;3$cQGioKS1=zo;tdaw8q1|d*Gk6>St58Fx@-v_R!}>QYcwHz z*%9zdu$uj6)%;W2ZMS9&gKSGj#LU_~*>6Bx)N0DAYaxra>(~IV<^BHoL32dx=IljW zPb;BGslGNg)08yi-21)2XnWG#ZJ-sm&R^CvT4CSs+MA1YdTJ`Kz}UF% zPyWl=SyFLxm?ZY-)7vL|Dz*6PVP}6+eh! z%ef6aW}-Zi8_UD|wQEtZMsvw3^L5OiP&D@*=a`SZ?5G!3!hz&N3Ig;h{m72{hl><* zRas$VI8bH|xfli)^xV9xVvQ=X#tx+t#)nK}>Wd~;3kyTqIGsu&z-6a3{u1Zi{*!V+V+A5y-rJlLO-l>ov2UC1?H%BVH1%+;Twej(lfc^)O_-vQ);BaDhqy(# zJv#4BKoP7u*~R3cLSxPf-`ZtM5d7O)Lq>J9aofXUlpWqL$C2iKT(`KmtUHlc50!Ct zeLk^v1E{Uf+}7e9u?CF*NYxm`N>o`RP`Q*~R?8S^xMl5KY;*rczW5yoF%D)% zwk$EFV7R5~Y=aiqL}PV_my%-LrQ2e+pWJAub+D0~etEfiW9+U`!+BNJl3Ps@>#fTI z&2U;CZ{8*ecNxIE!nQerZ&=2KL3F}>aKwcNTPwzNZjUlI)q0NFPZIou1&ge9t9L#NH5V|sGSY(Htkyht0|SGx>pp_tA`hlaMBg@G zJn6R*KIAc{Pj6F5+CSM4rMV{>%>Ny{_s}**Dnkw+As_E*(@K{zU8uqq$hppJQZk9= z&;}_#SfDdl@3pz?yR+^#Iq{Yycn<2tLvCNBXlA-2UG!xO>q|(8m9Tal^&?3xGWo5~ zcgLv{i6%`t^nR5qVZ;6NUK`6aa!fAHf}{J9XL&os-{L0=+8(V5=vrFJ@2<6E^L47m zXyreY!kxE*f+**r^4Sqdv-(x}csYKq4av=+*Yl`s z&E+Dj@37QvZQP(!=iYNY=Wb=Iz_r5#BgyJ#rxjWEgGgJJHl@R*+RG;D8_kiD+UmrO z%&Do5a()aXhaqL7sxM!Q1}`;oaCr{4sh}Xt?prZ&N49eX(kEwPNFTK?*WRWi^)$wX zuh&v+sv=p7__r(W;=)eyWJ|2e-q0iUwEBhFC;56gw;V@PLFN;*wD(naXJ+k9G|rGT zF)qzoCz-ZTLq3DXMsG2c2He!}>lgL>7Kf47w3};hP{X~qCr3*EWmw9_d`0c5@Cg|i zTHCkxb?Wa#;j`L z8v|RW?Zgimfwz2Ion>KYL;Tx^4{%Pi-qdm|E+%c+#tyCALWnEqEwcodfwHMs)Lx+oUie3FDz01F^por!g_GmWUmflFL7WX+rHA7P~)2#wOG08D+>nH7B`T=)3 zZq&(G^KAW~o|bmSt(EGflW~xwERnu(Ye$UFlyAYnHc-|}%937Q7~J&?1EF(IYG^5zbp z1hY}8iZ0)3<0tVKIg)M+=kEYaWwFlX+LbLTiS==-?0P@U)ws3*Bx-l3G>+fA=^aN7bD(lwc23TFKt3*lGyx%%hIM7PEW==} z-%`6}2~j`{4B8v8Dq((6H@|GPArCc&Zshj=SQW6%+urNP^@Q-6p+hehShqii{Y^dQ zH`Z9EM9Sd!WZAG)r_S^{<^2j*zxKq%#jIHPDoOw>d=N8LzyX61K2MJ$eA+Blz1!cJ zxNKeTGC+g*XSyYC)0LrmRfcZnGc9Qa&cg_|>l0?2R#4O+}~S2bj!w@ zwQ3cb*j+bV8qK=3B%xmx))+F^=cS1GPQH^N0Z&S#;fl_i3l39;X%$?rbX}mqAl>Zx zHW)lrJ+{|uA`4v;K|=Hu#v{kipADBKDshLC+A-2nCH^+i%1ShzjDbWAqPF5ec+2ril{MfwuSVV3g^V3=lMdT>Zck^h!S|F#KnwO+;nfhlY3 zMldo|E%RMlshz8hz4P3A)T>hh_mhn(w?xMW?YmcmiJ-di-fMLVeI{HYzX?8Ku2HC5 zs%NGDmCOEc0rsWW2lobW@I^H0^IRdM^NUL~G>w(78hm!n<+>UCdmY$ksxCfqgQ6NtAFAwf13!r8 z^Q6nV7Id^m&#PDh-`ke1n{Wu#%hG#CAtlZ959e(@X>Pbq@r8~ z_pbU9km$TJeHWL0Pnb6Tl}o!HpWDCg(iQ=e= z@a~!G%Y-4uzXwdcp-T!=Q-WBwKd(MqRlLjWL@<7$W3G+pF0gD8MHs6owaZUcViF2v zMwMj{yw%oK?GYP{Ha65X+nO)z$LjWY*jzt!B#I$Gw?1u@Is@}5*xS5jDn&p&YXgPu zEhjPbxhYnGW*K&kJiUSr7yx!~^75jS_lxL+98Zuk)krkW!8io7)6=J>3CI#_79s|2 zh;di@_PBY4>I&cWM+yo9+SZP|QBm1e$dSWGhu5ejT}EC%{^_PYu&rPyEP3(NN_n)0 zyn{UBJ`0_>X1L*!$*dGul_A`Adnn^0(Zt^G#}>|`b=8mLrchd{?|S9DxgQZupTmDT zg2PZi*BD8`D10Yxf3HNK{qwZI>FU8UMJPl95TYeW^uz2)BMK`ugt28Dh_ISqzI(_v zKNgHtXFm!|7*haasbF926H8q>k%v5$k>hLEs0~#M`1WyirXO>vYlqE{O{#AV>GyUh zL#`i(+Qi%O34`Ich`TRZI~!Wo!WI()zRiCf3tf^JDvuh@4xx9`uaUkt`OM{>J?ZP_ zu+Xhrx2E24Fx&qku%Nd#WPEXQrHT~I1bB)%gGl>W5?%A7MHv~HVdLgA1kf4f_r7K+ z3Gjkz+J0QJiYZ39BCk4(Sf(Rv%fjB}mf3BU(e}A&bt*~2<;UO78Bzlb;{9`DX7M8$ zpf&qFc6ZRtcawOH&wsY!1o7_qyw>2n4IRSRI5pILs=iXEAdnbQDcR3?iy$jj7^1zU5UIsoxFj#YUJt71^ zwkopw`&6nv-`PDV*3Efit8jRzexXiPQ!T(T`yrUh5UC&=Ff}!G#*RraID`0UW@g3# z3}9$1+sDTbqoea@g%8qE;q6oyg2BG}qeTA~X~f7K?+ZEb#TG8UIpz0&&Hx4mOCK`* zf0CThIjalfFdADU=s$Tv>qE3*VQ^cD)IQ9@DE5xYI+rK+u7GN!wgs&9G286q&|F_-8W zu&{*5_(%sT1^|H7+yiW}c$?P7-E za7HQ5;oe-kbdIm_QKNNUoZWzolwqlrl{{|&&OrvKn7i|rvB1j1`Dtp48M4OQ-ThGm zO0t7g5zIw}+<1H$Ds8R!M+*1HrN4g{@w)iHdoK5sp32wv^cBjbV<39r&ZwkO`0twa z|D>QOg}gu=WUq@yWmGYVSUtUMSSy|c$jxQ-v4;g}pGv)|^0_Iwsm;*;z&2ybQ1l1c z$IJ{DzDygh4cG_rpj4rz<$e6)X4XX?Qr3n!puuIYGI zHd_`y=@)hN>!hce>SnI|uzvLuysG*BTPcr6YRXgZJql^%yYGG$MYkI{PkzWi&4Sk7 z}s1XFF2u{_#l#KBzb+H-@^IgWBdlM)c3r<#&;2mpEB7*(`fwTAa#ma7Ip+Rnq;> z`GeKUu`!=M$!b?A{a95LUZ^}=d=5~U(qzpKX5|CLnjPne(S!c~+C{FVaBSe>A=?^X zN@ZVsSU4*GO$=IAP0?UpznX@@4=dRocA*yiWTtVC-5_r#@qEE8-@gBRw}lidv=HraK8W|$t>B&u{#)cpbZ-x>ZbD@8 zn_|r`=jx+D)e@jvyz^C#9d=hv-Fv`l$Txl<3U{ffH(Sz`m_r{8@9{PEV@aN1Q!|8v zgVlK1c$v@09QIq1sgFmTbB{Qo?4?li%INu4G4$xvVFAX^0#2-5J^L?7%1r_$?)`zQ z_|FOa@t^#i|MLaHPk67Q((C3At9P7I{@y=dr9)+TmEf<*yj?#%KI{bpBCvur(ARJV z_aq+wde}X9Sx0_$TACKjd#o1UC6^SYKkPxU+WM!6gC#HrDSKUbe|Yrsn}bL{7XQua zuv$^<0RZWOq*F)!_-y|CW=lOk$kk@6GY_4s*&0Z(*0xKM9XYIRREY)I_8SrVtixv8 zQt_Z=%pM_O-=2u6!_A2eFWDPKHZA z)gS~b9y^`GlaI8kYJ6@edb_2iCB+_HJ4~X~g#Qo&;#~SBRd~z%0_0I`C#DKp>o8A8 z*`C)UVg*$#SojuCJFEEL5kNBQ@y$`|&XUE)zMH21&RMoOUx3jiW`2QBbadzqB`1~& z8fj@2n-LyQe#=l4P6CgNhAC&x%8cup`*69UeF(z%c=M(Rb_KzVF4_?CdOW+hUbP9;B(n`N3l$SNAEBOyYdM&8n-d3&X9=7i1O7R zXC>7oKUnJN>!V^@KhMykNuS@Sryn1wu8V>1&IkjnZ3>BYuZU&S^3*gy?7qLL_N_r( z(}1S;o1*nR9|CxP6|V5(S9oiuCr1eSmCl(c<$Z`I-mUV0OWqg1gzigfMI@$fx?)Be z0Y&jzfw(~K)$lfTKqF=hMqpScSPqvZ#Yb-l>a?D@PD=dX^(LN0QTdsO%ns#tej3&cJTKPAa6>}CC}wQ2c@>H z#x)fnn7tY{g!EY6GCDqF&~zdmKXp$wIkosY(XDA3`=MDA#l2l+(T=P9o?baP-Mh~V z+cn*t{sEm#$}h7v2N!ROOwNIGc%agYW5A?`KztYYhZt%N6FITVb3@+FERd0EX$%$`(LteO+w1<^bS68QJzh~8UQAzxZ>R<)n zJNt3rVJ+dde&M~Q54t`s6&t9=!P?D|hf~BK>~w|~H1&LCz$=@#54+&-_Pv#&=HS8e zKrPwstoS?Nz>MPtB1<(3S2XOW@0NK{yI=_SNxLh6^5)s*pg#wTROS3+NFf(?%(++b zgp>UlcB$l<6SO>9moIWCCcfQUTyt4-wI9v@JZV~)~KHEj&0Y3C36{r=g4x2IZ)$By)6 z<)ouTgb2w;NzZk`)bbpB>~5s`M5MYrsIwG9{3j{sWP>gn5D-g~VBXRPH6;mluT?Ox zIRw6ku`d8TjmkWQa-e-yl#I7#{Yiq)z;KCN1o}?FRH&F48i4nT3Rx;MCE^fbfbG|a zSu?#@;6$xA7?t_;gW#jR?X}Sy5&p4@T5zc~!t%i<_UI~K9r4#-yW`q6^sqL9TbtW= zGLnU)w>lGJXn^Vbxb0oz#^=YRNAJIm36oS-0B{paKlugl0l0m@_W}&}~BHwmuY(9z8PZRTRp_KK&YM zTiAS~8vp=h^{ZxOU^aF!4S3TRVpSTS*Z>sjd z0~id}uJCgN=_Z&)>D35&A!v2a4PC2r!zM8q1I=^5N)XKaHoSpW1xztHOrLkCn?GjN z3H2PwEO;@o2*0rrxMUwRt(kL!)XQ1%)O(iUoz-o85A5^?7Aup4`n4Jo|jmki%M zU%0DdG--P@-RrY?M_W;Xc?Z7Dse`yCniS`?GV__xHrndq2gACI79dJ$``nXq>_T`- zEjD&v;dwc!!0eT&lG&5lIfah|Qmx7Aqjfwp7xyH##sg-)pJcubSazki5t94e;$Bch zg?7~P?#)3%H=ph9FyE)94Z}d8TSb@h)I3m5of0#3$0(PJMxP~lj~`>T8d=MT-+Qly2f_lAU+kZ)=a`vpE*;UXG}MaN%D|g`J%yB<*Rp?81`Z=#d+jIfw;O z8akP`w+zolOYY*R*oi180O7CoLidYP`TP70h4Xa~VRl5-iv67>Vh${x_yyn$D@>sH zdA6`@3Yvw(41|skyY6IE&gd@g+bM- zbt~LpOLKb<>u2ceFvr`%9;5dpk;+xa6#$L7?0s9~vyD3^-ZB}qN4LF#Znv4Yta05o zWMWW)EKIj#dN7Fuf+iI!(>FAloM(B;pSisEn0&|4BA{ugwgM(h&1y^BxHEgA(x=VP zGcK*QYHTB~@W&e0(>VsxUOI_qdv6qskL$P3*bu!r7izfQ62p*q$7$c~Cmq$o)K0u1 z>05$RhltNN0m+55y=P7}5%>2{SYJ1nj!1lx$6S1w=v+%hC%~xdF_)A4XNS|h#J|qP z^yX$jPm9Q!7{;ldC^Sr%W5<6&i{s)KM{!{6mG3s@A{f&6O9-iVft@w(eKOqvOBS z&dpYOJ${yRrh3H%mM!~UVXp{n&tq9{Rl{TNyT{J!A-~<1E~(z56uw!WAolIz+(E?T ziwiOQq!&h-;q|jLNW}>gzXQBbhE4u=a?|f`Xl<64NSkHwD26=v7d`ihH!~mQm@V7T zyW{QlzD~^*ruufEMayw#P?=XG2I&)q%HFyDb6?qSM@>BpYq+*|w3AmYY}qeb-)=Z- zk9DbI%~Dq!%zJ6^A&hC-+f~9VqW7jbAlhS99bTMw-gehF)Des`*BtcRwk=~W)8!8Itm*&YS~XD`>4HoayrLwg zEjp6}-`_FKEgAegAa9;W7lkHG5N5zM_bl1t^Uh&uW1CWb$P$|A*wv3=-l@}~eiLtK zc{(LXdfVVMHS?-gDjhe}H3?##Dq{6YgL4%(GLYOwdXGYxtYIat8^R>yi+YM|!Ey19 zJ(z!9e{b9pp6FX^#TIwZ;YrlJ+3h)K=i~>zQCCb;@c!Pj{J~=tF`8jLJ7I9rS4O}9 zsUB~+w4(#{Ta{+X8R6mEtF(sklRf4NpZiUfS4#TX=!3I7B563AOU!K&1O>V0*@7bc z|H>%V-YI*h=xh*b2{mg4VLmx$coOefJ{fijlOWorR?d!CKYI1j(P`FasTO9uabetZ zZBS`NXvry1x?{5tH5(BxclykT)lx3`IZu=w5|b7GxZ<=g2!b{5L$wFVS%#C(T(UB- zXXX&z$jXLgy|R)l*-p=5c$|x`-fHCK5gK)urb$R5|LFn_&e>(!oER65kR zlF_^CyZV$@mB*79YgkpO=R#_v(yi0gq>-D3Tz~`L%6N0MT*ocZg`txG8Z)z=lMG2a zn%Yyg*7v>%Nj?nN1#KR>s6MZQ@s6Ql%R@gGN5r9EBA>$etFt_uryoDsp1f*!`y`V% z_mWpF7P{AFt{~ceFJ|<`R?`)9Mq%kpmEW!dLR080pEB79p!p`v4im*Y+QX&4k^h_) zLgrnU+IzW2#VN<=@e7S}kRh4+~BEe5-$K zZHW>>ZR5pSpTA|%xRBqa!&kh=BZhr)qO!DBL$dQjJjk4@7g^Pnt@q>W-dwEpe6Udg zb?|FZqGP!uY&4AVxP54fK z7XM=d1u09ix_AX4f;64Jx2m>tw~PW=^o4UC^UvQ#gNf@k9!?(+2qrf@7sxlN(&_8vLcW1iNq2jCHj$qN3i9q5 zYYx9$eV4~z=zGv(8D^O4w71fI9q!!-HkDWy+zHz$emrq^g6)ZU(D#Rb>8ROpmLy-uQtp$ z8GgEJyQSOk=4nw|&7Qliz&`Mnk+wD$XNnbl8`AAvdH z*lBqYj82b{PN>^Z{EYJx!#&&V80Q3K%ROeQIWPV`f}VOE>D2oasm=5`ARVQ* zMCmn%jMykjN2(*eO78>)1gSwFgdU1XfDi(N7LvesW%lzv$2-pKJ>&cRoqrxNN4W2G zm9^Hn&ULPLR!ill%SGoOB@zdoaavSQbfL>HcB+m2bjm?G-X==-4V9(XINjGRt=;k9 zxNmmKSS2nq@ce~wG_F&$_{uK1ao1>L(w?kLa9bzJP2xsnZ)_I|MlcELr>~E0n@}jp zNDDBJxFUV;1?2}%xj@Fa-vN4Zwc3B=gy$DI9Bx|O5tf~*VW>e2exgM(#j*GI| zB2|qcYN<)W8I-2x=jY6{Yd1!7UB_wbuCq(3bzYQ>P(P(ND0ehOyUwU$ldvHMnefo3 zEM)m7A3q7LUwv*MnRiYXx;hl~zonU;#L^M- zc+DuH$S|?Hi_OdA!L!q$6)&YP@~>xp*k)}J9)O>{6utjCWK=&&Nf{Dk(#2!sYaKg5 z4?f5@$MHGEHafGAipB2kY&xwJNClU8svl_b!iDhDtxg|J!}L!L7Zt2*$eg{QQw5q| zJerps#~YRvI5*frcW3!`?-$iBW}%#5@Ae9K19(>pbMxFk7|uzChTM%dCC|0+3oHJyYBW-K8DwaoDj@E(LoBcJZs- z9he!8#wbf;xah$NANUy#o@KSiLs2h|albs@+|!{J!%&sfO$}2)t)Dq`=Z5d3TM#th_j5r*ORm9_K5!{xfSN4C5Hd^D+L>30 z5!hfAEEg@&Jz=)7>8W1XztY|Y1|mLj%?de z#zvaL8Z*BhxK}!9znhf4Y4ZtwwMD77q6p>A(jfNc%CF9a6@~^qivw8_Qiz=hjYNKcRp4|j`)2o!H3me z(;RB|c+n+eA2=D)9od6P_oZ(Y^TRdA^J&dC(mH!|d5@+sEK0NRW{Q{zw)-88b(5H? z#u`|vWo1zv)mJs#>3FE_M1RB^_su~$T350q26vMfnB%8Q)mpgnS*{(#eXI56>IeEf zy7Lqec)EBBuUo^L*_-k(p&Zd*L(lAwUPSKRO}iX&&BTWG1&O*tUm-zliS}zeNBLie z9Ud7kdgIvP;SVO!YC)8WW#U;Wyu0XGo|LU5OXt9323k1{o9(&>hVSuxFoZ6OMqBuk z&?V-jz6$fw116Fz=JM?8oJZ4aC1-9;r1sYyM=^~d7o_g?sz-LLVEd|_$Fpw^dq8=cjr=GQQE=2#pStWS7r*FzP%#DQg{qouV ziR;)p@9WX~qXZF+MmnOs+&Yy(JbS8Uc=D$>P)jg(ka(f`OtosPrf0x#|(#~%=@fDU@BA1{{bP;pBuIrV>I-fNjKWe54Tfy%nNunsR{`stCg&2elyy~ilygK7t;>;cRB0!vS#=5~sURA3?`kX32Xofb0n*uxj@QKZjCVvjSWk-JLu>#G~!XU!0%KD)z)Gviy3Lw3@eRLPpCA769-ll zXup_=!fQLS&;dQW#0{z~r8bFQ_&##UCb21ZQFovuPmJyGgj@dA{3(XtXjLP%Fe>P> zXDDhhUdm)ZMes&3TSUGBN@xy&MUlts%k#=3GcE)ciCWM`F&?2i%Of{mENg)7>TFA}&U^RN!O$=?~<{Dr54Da0QaSqkE zJn2uq`nn$WHjhyPbN=k<&Y}P<84=#@aj9!!+3k={s|+nDli5(_D`=mf(EpE7c zqi$$Pzvtp;^>iqYhWs1xlGanU0K*iTAV%4)S|BK`@Xpe%E2cxc%iCzG zgN3D2Zf2g=YR8*`6A7pHyQRF-ai$&{!Z@tcwBC3XO&gj${p@A=Zaq^QWXY? zh0~Q|Jz-v@(`Q>BTqoI#pahLo1CbYvr@+jf8D-axa^kwCDjVjO*XNEHJg0hn=*<)h zxys{Oen-Z6^c9jGE56BAk9r~C7!*iQz#5bE-H(1&YjGTKFgoBKIH~A&LuuQw?xaeu z!e+f2fDLkGqu&NZD%qKsx)HsM{E1HC)@>yH+vWMukv5IxDIWRe3dE!@#Ne>WMA?|~ z^Vz(@o<-y_+lC%N3N{-H{_&5@uScq_==KfLM0{zk+oCcogA@s(y3 z;WQ67^$#n0nr9&#bBR$du!Diq>B`N^iKzlTS1X5q8H7~Jj{2C)+FJ1FbR1ihuXg+d zDY$OV;5dX1uoh`n^_)U9Oh_Z_9J*s{-sV1kSZs;in&pYNd8nZtw=lj10};uZ%@K8c zuk^3NEoNu;4fKsIWtri1JHi`|PA5LdB3f-}!I#wD5K=TGH_*+?HHq;DESi^7%C*S1 zf|+2D4KyZ7b=T%&cTbGGsrA#5XoG-{IH@#-miik_4f%H$hd;`3DGzXDBs@vSyFd~r z!=tb}oXD+DdDc4aZ#-E%BILFc+Q^FB3v~vksssC)4;nx)WxLzs7HRn#EuInqmC=j= zSlyDL09qT}-rUQqomvgisk5;I*!HZbj|M9in!$mTnwA z%IIz3lEqDb_!M@Wa?`_WIsCm)T15B11YJ z4W9$S?5uysr}%p<{e+sItPFqP>b-{Sl}=6Xaz}bDd9-gQhn#!VwAcIr5zVk-iRr(i zU^~U{d`{NavjLG4n{ymT)8^;*tOFpWYGKeglIzX(bG8>rCWzpu%{mY*QKS`#{o^-+S?~?Q325oeC0~lg2$R=m*J6 zeWy5W;o#bW(oPZA5F!x+g36)&Zbk9L?jbMwIV{p+F#asOxZ)Trqqxe>PGKbMEUSGr z$!1<9%`@X`{bf=?q3a87V^zzULy>3vBQ;k8RrkdP5?qzY&6kF+0^BP0*d@6OxRZ9FPI*#9;|O#K&h zZicnSL>rx<*@%^xlSa;e103zyyLY8=pOm@4u~ufHRgofXA;=|j%%>}U^k*pY2PJv%i9;M;z+Pg4Dq*wX7kF`Jle z`wVB~pa=_m1lrmS_D=WQSnr_>?%g-?90O5Wz+iWSH!Tw&hkqV6i zT?5+u5pekvg86Qp7XC{zRsskj0#P36KJXwxC0&!HkQ{21W-Tnmw6@^CdPi zhXWRucH$~n!-SS|^lRq)Ib}*Ys}4Xb#N5YRt72Jc-Xe0Az1J=XHgoi(-Ojev5|+qk zJ4#5f^sem6IY#M)khL=B!7)|6(7EoVA-`R5;8n#h<*A$9VU(^R*C7-UC5o~{&Jl?n zQY5&$-aXZ8PRE+ylaC}Lg+6AF*=AtftBCMGR#gqFx3|tD7JZnm0tLYZlXajE0=sv? z)tUSJ)s&`zYs!L{&ppAU8a^F=3;DN@_~Z|%%V>^w!m;?17ism&WMSjmqf`BHCRK8Z zAj>mI52VoL;$6gcX+gVV^zy?e7!=E$aeI^0iSY#3MJ21R5c-eXTu4+U0oG#Y0Lhz1bOi zr4z4jiyB$Fd(Ps9l5U1`hHeRrEG_h(dy>jyMGo36pX%nPPaO<@kce%>kI`6XA=pI? z*+_iUq7U%BumaDs56gTL<^A2VY=>ub1e^68f8cW(0rkPo4Fwm=t6t2c+?Zqs1h;a= zWT3%`f{(ydb{W`N8y-TcKTJJq`h5&*KLLM3g9W9qA1Yg1RzcCEI7_WwhfLE%*LSY% z)#w?S*fzCG$4Js$&6l=1c-LDhgJz?8SuQgse(nr!(P>^bu0-(r)_3<0l-|HS)+p^m z3qo`ch}@(e4V^V5zmDcbq_39`)o@0*7kaKHt3X(zHVH z21yl~LL#FmE&RIUnX1!H|KYzY1mA|%t_}Dv?xEa`WwdyF9gaQXeC-$=gucvT&n0$Qq z2HTJCxCOB?!X{x8yIz6eO6BRH9332IPhD`$ZEWth$v7I)aZR zE@%x3UbrWVIiDDoB!$c#u$C0DkBKN(@)F4}-br)s?~eWw8uhk{sx%mB%8`6@%|hOF zz=Mg9IClj{nGF(N7h-%JpP%g@7UC(NV3o{6kw$(r+Tqq?TIHNo_&D8}v5n9yf}3vY z-((cwcJwDIM;u2k%H-Ydz7N7<`5wdYE!nY@wiGe<~}C{J}gsx zzBx3t0;_Un-}Dn`9|)^7h%c<1%tH)s)r~sk%gDQ!Z!GmcA(*22=}kv6;(xs$CbZMtv=g_l(2fF|MU}aih7IFynLMc+HoY|fIiYT z&O8g%*BWA+HKFJ@)vb*iANT7roxE%7ue&_(&bi3gg4$^mXY;gnW>qv#OOGu|JH(X* zCw0*urh>k)R&&|HK02>zVB7~tzIkJ!X^bR+B39MRUcy88UBU9AXu&I6IVnLeswkh@sdCjjX8`G?8 z3~#`mTasN^1jT_0jJd)bT82D!prH?GAFxCvoPFIbXN2w5UnjhIkQDtMmM-bInh|Wr zy+%w4C=<8%>iPN}v4*hZ*doO4m#J^R_-uAuZ^7AuCFJzy6SFe0X)Fp|&9-7PST5+0%ArLfN{e`n3XTiba%O}p*eN`zDA1=RCSm{@SW!_Q1KD=tD=Sy!Nc+ti3iFrCnmkf{OwrrkNn)^n9W(uZ&LQ zeZHESv>7Btqf25+xw)Mpl{Rwn64JK<5C9oR$gOsqH@LZ+%aQT5&tUs!O5ivYWSKe`M-gQElDCsB=tTdhu zE2MNCD-RfDAx3N`LWquHa;l^nOlr6~+dHgpd`ee@0_nb6;XC)nFvXF8$?(x;(d*G*w_zs|H86PeOTmb+NEXzM-yIu7||JWk^l%dK5(&JcKdz@vN>jv!<{^z z>K{5OYg7zPL?6_5>-HfzTbcrO3`?HfA<1DPWe6v)gbK{=60fTd2#Ag};>8U-o&uLQZS2+|Nb~sD zTV7?2!{77^NZboqTegU((O-|!A z+vCrYzVN`9xVUPJs1^*z08G@_@@d(2mxtuIt4V5;8r#LPN1-BVZ>_clS8M_BDbyhh zlIrw4b95)qS}D%fx{(u zt=u?joio!IeR@@29I@9nU*q`}^75$eV(;#}3e`U)@DYG?vB_8=0TJHoM&_RAQb3T3 zCkGG|aN2Ocfk%dLV-{8Ey?<2A>Qp6*^t{VpKyiai%h1lIE zz125b`3g@vH)vEWcTznF`|L)c%OR;0z~~64Y4#Y;P^y;_mYEgyTG@f^+8>h^RDTOp z?nag$F@LnP-}k;b)U^g3hn}T_E~mi@2{tzPp;;4aRrYTRCwp10vb4O;a4=5HW<799 z?ZjrU0@mKUXVcEyvyS?D&Z0I8F+-TUFqnw@MH=2$# z;hy)tZ~KN};?6HDNNBE+<;q>H!!nA-0|t6<6J%d+Pu{f3p9dezM#9Yu&eN8?Hq#e~ zuet}ry-3p*?1L8bdiCfb)7bciMO#~t;uR_r142y${6-J8Inu7v7R!hu48dDCJ=)z^xp8d54%C)Nq*Vvvz`wBKcl_H66Iym{=o=n)LVgFV&P&UB= zRUO+3w4o@o)Jk2xt!!rdQ|B69U}I2buPNKFW^l&3MyS7*`b}UqEY!S$Kk)091%}bc zJ40j7TFjWX4@h)}4;46ha}-TThNNtuXOSl0;MF^Ae27rYJGb?0Ul+3)900)f!qRFx zAG>EwiAj`Wy~&*o#}5y`3T=Z$ zz~D@Dk+nN85xl#aL_nxzFZMbA5>IPiZRP1W%lDrBN1^1eV*$y`%4lAg)=fDb8F46p zt++!=1cDzNmiA6p&qmfNGP;8{>O_mnSDW^(TAF_3Q16!FIdmDj)dpD@v73WGeZRA_iHu(y7*Ll?dB3ettQ|Tp z7wH3kml91ctpz2 zCP;9^|G8V?Ft&Ev|LWeKnJCFR9B7D&(*h7_t4rREqzdZTt-|9G#=1(HqArly5F zO7r%bi7&l6_#oQCDtLb(*%7&Cos9RWNL0@PP(oOvahGc5pzl4Guv1aIGbbnUd?s z&^lgqSjii!uWdmM7d7HeHX%47QeBy{py&)~F~IQ)K?#l4r|LukXt+MLd(-nk#8G`} zrg&hkQ^I0-!91&-ZSZqGznnn`d;c{%ke-mYyat+a9Dq^Njt)~0yrAdwPR-`t$cU-{EfPbw_g zvU2ErGh}{^feP9w^P49*>k#*h%47A?g!jk@ez+Al6~aH~y}~TQemT=m4$~Gtj;A*` zZYZZozy)Pc0Sl_c6TK*Xi8#qJrv6L#cCs{z!WSvO zkR!sTmRc1TCg2TXCWDTjOe`tMO)i3c%4g~?G>4hNkKwDK??^Tb94-g7X_t}WlC!D# zG}WJTg3DL<<(uugIdTJGGfnDjWhwQAI8X4?XX%>K;A z(Po5(g4c7GNpN&XMfe(4C-+6MjIFzz%K-wGf8Rf5!Z_S(D``+PqZOKs9xZ8 zAtqnC<5Aybwl{?ehboFFp2hy=v6(Wop0}64wJxZ#kmB4>rdttOO1Fw|4>Vk23+6k0 z9}v3oPll`}sny`{rrS2?ZoMgJ4MQ01%+{^mUSG>$E3<;#)wJuKMfwF$i+8t+@f}LL z*GC{ylDZmCvq6_eu}o=gg`u)BW@E%Y%?&6emdiM#Adv)a&&#yWUV>wZyXXOy8pJLQ z%Xwj6g}$V#*O||sjpw;Waxd0BGMBe~U4Z&@>Z6q2?G}p(_TxCZ1dTLwT=OPKN(k}u zm(ou9jK(qvvyXd}`v|cpCy5(|@n!+)7xGs*8<(H0;vu`)14iz=la$HLm=v$cT;IFM z$mgpK+)MsTRXlz7VWoELGwW~0{mAiWB7(RAw-_8;*2dh+&$@rJFqzp)&pK&tC< zcJf_w=PbrGB<={({ek+J`38?Cbv3?meC}SfcGN{k-5OMMJUvki|FViZnMTEx>zMPF+uo+UOGJ zUC&&2%s?)F=)Fz}i;^7%^x6+L3qfnwZpXhCYjmh3jxgyZCoZAkcGoec>(`)hAvN%8 z`2fq=K&U78EBo}y5n^m${IHIkbY@o^x~62J7EDQ?ss8j~QL~sn9ZRxHnjcumP<1pi z4tVLPZYoCKc!pMPm}sk;C{l`{P`xnwdOW)?9bQ+eOm-Kh1$N@%%`N)H-v6O;82Y1x z7K6hJJ);gjHE;HtS@ z0}!?@XHJBxG3e>&QkbMhsLJMnaI+Fff_DTCNlB$0BBzg}L5yPRB);z%RAew7pUika)#12CoY2$Wf_zzZ;L-$yD4O zrx-l{#U?fYq*rqs_ufT_34@v29u-Q7=ixYki>xmf!XE zHm~tQZ^HLn z4-lf;KL1pBx_UF!KOgJao47IT9=S?7(rCWl?`u^&C3gUc@JqYp=LsFgQm6HGl?~#| z+&UJ<|4orY_~2p#L!ND$HYO3`<>1h{sgS(m3vviW5b@bEX+J|gV`p(6co5}pbCCne zWWY>t4dAO=s&tlHkI~oc&33=yPaDHpqzo_s&E2&-n*1+$alt3DDPa*xOT$rb@qmJb zTW#nm5vA=c`t$pKeH-OI65OfR2x$pK?L&Vef`W=b^L)rF{q3g%MBH~pM*NM20rt3x1tUxoXzL7gEoBPtDRoML~qZoJeMbaCYsW!ul) zbScv-!j_RWo2Y1vnK`U(y>h`qt*9L`=`-85BlFs>{Qb~d)zJdCEG>MjB{Q%~BSv0(%tzGLb7a(~TRW z&ha*TwcE<+0MpqSXAFGCYig||Rhdkq25P#iJYbA6YhI-k z9R;J{<=0nA)lRD=ScS3*%EklyFMko#wQ<`#St&_t-qwx2H4%{qhuJ;Nih7RhBPl2& zQG_lgys&?RP%qs~reaJ=Q}uO~Dl$Uwk2K{YvqJQDDnv(6LM^F3rN^^cG&JSsW7X^A zarFgaa_l@X6}k#?Z1MSOu{o8JwyS8KHG!u_eWEY}bYEaOv;F9cABD!cDjbUHO!m$A zdqS_rP+FjjdEFjX^z8d9&~Zu^E4oc?cQlrjm|bRH7v)UGDzA*K)?$!F3!mR~e_1EQ zQmaPY^5@TJhIYj_tl8RbDtQ&lKgt?#*pZ3%=)=OXjJz+{0~{xygXEq^lkVT0 z&^b^O5*%@ajJcGl3McB2{Z}(O0xfDm$5vkY7_aweiBlJm@~|s9kE6RKvRsRIbYLe! zS6@gsh-FNSOSwZ`Qu2*TUE&+)l8)&m2&(23f^D7LU#mC=I!2esyosj0nA zwmvJ%jLK`r2!770qa-`DSLyq4@!aSm6|}|Yk|a|s-S9UINdZgkT|gLQ`y19+^nO

    Welcome

    Welcome to Unleash's documentation, your one-stop shop for everything Unleash. Whether you're just getting started or have been using Unleash for years, you should be able to find answers to all your questions here.

    • Quick Start: Get up and running with Unleash in just a few steps.
    • Unleash Academy: Video Tutorials to onboard you to Unleash’s full suite of capabilities.
    • Feature Flag Best Practices: Maximize your effectiveness with feature flags, regardless of what technology you choose to use.
    • Feature Flag Tutorials: Our small but growing collection of tutorials on using feature flags with different technologies.
    • Understanding Unleash: Get a high-level view of Unleash's architecture, core functionalities, and concepts to leverage our platform to its full potential.
    • Using Unleash: Our comprehensive guides on APIs, SDKs, integrations, how-to guides, and troubleshooting will help you get things done with Unleash.
    • Contributing to Unleash: Join our community and contribute to the ongoing growth of Unleash.

    Dive in and start exploring—great things await. If you need a hand, our team is just a click away.

    Getting help

    Have questions that you can't find the answer to in these docs?

    💬 If you've got questions or want to chat with the team and other Unleash users join our Slack community or a GitHub Discussion. Our Slack tends to be more active, but you're welcome to use whatever works best for you.

    🤖 Ask our AI ‘intern’ (still in training, but not bad). The "Ask AI" button lives in the bottom right corner on every page. ↘

    🐦 You can also follow us on Twitter, LinkedIn and visit our website for ongoing updates and content.

    - - + + \ No newline at end of file diff --git a/quickstart.html b/quickstart.html index 679c975d4f..4f7c88fbff 100644 --- a/quickstart.html +++ b/quickstart.html @@ -20,15 +20,15 @@ - - + +

    Quick Start

    There are lots of options to get started with Unleash. If you're comfortable with Docker, this is the fastest way to get up and running. If that's not you, here are some additional ways to try Unleash

    1. Set up Unleash with Docker

    The easiest way to run unleash locally is using git and docker.

    git clone git@github.com:Unleash/unleash.git
    cd unleash
    docker compose up -d

    2. Log in to the Admin UI

    Then point your browser to localhost:4242 and log in using:

    username: admin
    password: unleash4all

    3. Create your first flag

    1. Navigate to the Feature toggles list
    2. Click 'New feature toggle'
    3. Give it a unique name, and click 'Create feature toggle'

    For a detailed guide on how to create a flag through the UI, you can follow this guide.

    4a. Connect a client-side SDK

    To try Unleash with a client-side technology, create a front-end token and use <your-unleash-instance>/api/frontend as the API URL.

    Now you can open your application code and connect through one of the client-side SDKs.

    The following example shows you how you could use the JavaScript SDK to connect to the Unleash demo frontend API:

    import { UnleashClient } from 'unleash-proxy-client';

    const unleash = new UnleashClient({
    url: 'https://<your-unleash-instance>/api/frontend',
    clientKey: '<your-token>',
    appName: '<your-app-name>',
    });

    unleash.on('synchronized', () => {
    // Unleash is ready to serve updated feature flags.

    // Check a feature flag
    if (unleash.isEnabled('some-toggle')) {
    // do cool new things when the flag is enabled
    }
    });

    4b. Connect a backend SDK

    To try Unleash with a server-side technology, create a client token and use <your-unleash-instance>/api as the API URL.

    Now you can open up your application code and create a connection to Unleash using one of our SDKs. Here's an example using the NodeJS SDK to connect to the Unleash demo instance:

    const { initialize } = require('unleash-client');
    const unleash = initialize({
    url: 'https://<your-unleash-instance>/api/',
    appName: '<your-app-name>',
    customHeaders: {
    Authorization: '<your-token>',
    },
    });

    unleash.on('synchronized', () => {
    // Unleash is ready to serve updated feature flags.

    if(unleash.isEnabled('some-toggle')){
    // do cool new things when the flag is enabled
    }
    });

    Additional Ways to Try Unleash

    Unleash Demo Instance

    For testing purposes we have set up a demo instance that you can use in order to test out different use-cases before setting up your own instance. You can find the demo instance here: https://app.unleash-hosted.com/demo/

    NOTE: This is a demo instance set up with the Enterprise version. More information on our different versions.

    If you don't have your own Unleash instance set up, you can use the Unleash demo instance. In that case, the details are:

    Client Side

    • API URL: https://app.unleash-hosted.com/demo/api/frontend
    • Frontend key: demo-app:default.bf8d2a449a025d1715a28f218dd66a40ef4dcc97b661398f7e05ba67

    Server Side

    • API URL: https://app.unleash-hosted.com/demo/api
    • Client key: 56907a2fa53c1d16101d509a10b78e36190b0f918d9f122d

    Curl command to test credentials and retrieve feature toggles:

    curl https://app.unleash-hosted.com/demo/api/client/features \
    -H "Authorization: 56907a2fa53c1d16101d509a10b78e36190b0f918d9f122d";

    Unleash Pro & Enterprise Instances

    You can run Unleash in the cloud by using our hosted offerings. Please see the plans page to learn more.

    Other Local Setup Options

    There are several more options to get started locally.

    - - + + \ No newline at end of file diff --git a/reference.html b/reference.html index 978dab68b5..88b4a48568 100644 --- a/reference.html +++ b/reference.html @@ -20,15 +20,15 @@ - - + +

    Unleash Concepts

    Documents describing the inner parts of Unleash.

    - - + + \ No newline at end of file diff --git a/reference/activation-strategies.html b/reference/activation-strategies.html index 9591f7039f..83518053bd 100644 --- a/reference/activation-strategies.html +++ b/reference/activation-strategies.html @@ -20,15 +20,15 @@ - - + +

    Activation Strategies

    It is powerful to be able to turn a feature on and off instantaneously, without redeploying the application. Activation strategies let you enable a feature only for a specified audience. Different strategies use different parameters. Predefined strategies are bundled with Unleash. The recommended strategy is the gradual rollout strategy with 100% rollout, which basically means that the feature should be enabled to everyone.

    Unleash comes with a number of built-in strategies (described below) that can be enhanced with constraints for fine-grained control. For more advanced use cases, where constraints do not fulfill your needs, you can add your own custom activation strategies. However, while activation strategies are defined on the server, the server does not implement the strategies. Instead, activation strategy implementation is done client-side. This means that it is the client that decides whether a feature should be enabled or not.

    All server-side client SDKs and the Unleash Proxy implement the default strategies (and allow you to add your own custom strategy implementations). The front-end client SDKs do not do the evaluation themselves, instead relying on the Unleash Proxy to take care of the implementation and evaluation.

    Some activation strategies require the client to provide the current Unleash context to the toggle evaluation function for the evaluation to be done correctly.

    The following activation strategies are bundled with Unleash and always available:

    Standard

    A basic strategy that means "active for everyone".

    This strategy has the following modelling name in the code:

    • default

    UserIDs

    Active for users with a userId defined in the userIds list. A typical use case is to enable a feature for a few specific devs or key persons before enabling the feature for everyone else. This strategy allows you to specify a list of user IDs that you want to expose the new feature for. (A user id may, of course, be an email if that is more appropriate in your system.)

    Parameters

    • userIds - List of user IDs you want the feature toggle to be enabled for

    This strategy has the following modelling name in the code:

    • userWithId

    Gradual Rollout

    A flexible rollout strategy which combines all gradual rollout strategies in to a single strategy. This strategy allows you to customize what parameter should be sticky, and defaults to userId or sessionId.

    Parameters

    • stickiness is used to define how we guarantee consistency for a gradual rollout. The same userId and the same rollout percentage should give predictable results. Configuration that should be supported:
      • default - Unleash chooses the first value present on the context in defined order userId, sessionId, random.
      • userId - guaranteed to be sticky on userId. If userId not present the behavior would be false
      • sessionId - guaranteed to be sticky on sessionId. If sessionId not present the behavior would be false.
      • random - no stickiness guaranteed. For every isEnabled call it will yield a random true/false based on the selected rollout percentage.
    • groupId is used to ensure that different toggles will hash differently for the same user. The groupId defaults to feature toggle name, but the user can override it to correlate rollout of multiple feature toggles.
    • rollout The percentage (0-100) you want to enable the feature toggle for.

    This strategy has the following modelling name in the code:

    • flexibleRollout

    Custom stickiness

    SDK compatibility

    Custom stickiness is supported by all of our SDKs except for the Rust SDK. You can always refer to the SDK compatibility table for the full overview.

    By enabling the stickiness option on a custom context field you can use the custom context field to group users with the gradual rollout strategy. This will guarantee a consistent behavior for specific values of this context field.

    IPs

    The remote address strategy activates a feature toggle for remote addresses defined in the IP list. We occasionally use this strategy to enable a feature only for IPs in our office network.

    Parameters

    • IPs - List of IPs to enable the feature for.

    This strategy has the following modelling name in the code:

    • remoteAddress

    Hostnames

    The application hostname strategy activates a feature toggle for client instances with a hostName in the hostNames list.

    Parameters

    • hostNames - List of hostnames to enable the feature toggle for.

    This strategy has the following modelling name in the code:

    • applicationHostname

    Multiple activation strategies

    You can apply as many activation strategies to a toggle as you want. When a toggle has multiple strategies, Unleash will check each strategy in isolation. If any one of the strategies would enable the toggle for the current user, then the toggle is enabled.

    As an example, consider a case where you want to roll a feature out to 75% of your users. However, you also want to make sure that you and your product lead get access to the feature. To achieve this, you would apply a gradual rollout strategy and set it to 75%. Additionally, you would add a user IDs strategy and add engineer@mycompany.com and productlead@mycompany.com.

    A feature toggle with two active strategies: a user ID strategy and a gradual rollout strategy. The strategies are configured as described in the preceding paragraph.

    Deprecated strategies

    gradualRolloutUserId (DEPRECATED from v4) - Use Gradual rollout instead

    The gradualRolloutUserId strategy gradually activates a feature toggle for logged-in users. Stickiness is based on the user ID. The strategy guarantees that the same user gets the same experience every time across devices. It also assures that a user which is among the first 10% will also be among the first 20% of the users. That way, we ensure the users get the same experience, even if we gradually increase the number of users exposed to a particular feature. To achieve this, we hash the user ID and normalize the hash value to a number between 1 and 100 with a simple modulo operator.

    hash_and_normalise

    Starting from v3.x all clients should use the 32-bit MurmurHash3 algorithm to normalize values. (issue 247)

    Parameters

    • percentage - The percentage (0-100) you want to enable the feature toggle for.
    • groupId - Used to define an activation group, which allows you to correlate rollout across feature toggles.

    gradualRolloutSessionId (DEPRECATED from v4) - Use Gradual rollout instead

    Similar to gradualRolloutUserId strategy, this strategy gradually activates a feature toggle, with the exception being that the stickiness is based on the session IDs. This makes it possible to target all users (not just logged-in users), guaranteeing that a user will get the same experience within a session.

    Parameters

    • percentage - The percentage (0-100) you want to enable the feature toggle for.
    • groupId - Used to define an activation group, which allows you to correlate rollout across feature toggles.

    gradualRolloutRandom (DEPRECATED from v4) - Use Gradual rollout instead

    The gradualRolloutRandom strategy randomly activates a feature toggle and has no stickiness. We have found this rollout strategy very useful in some scenarios, especially when we enable a feature which is not visible to the user. It is also the strategy we use to sample metrics and error reports.

    Parameters

    • percentage - The percentage (0-100) you want to enable the feature toggle for.
    - - + + \ No newline at end of file diff --git a/reference/api-tokens-and-client-keys.html b/reference/api-tokens-and-client-keys.html index 4b5b912fa4..f47a95a51b 100644 --- a/reference/api-tokens-and-client-keys.html +++ b/reference/api-tokens-and-client-keys.html @@ -20,8 +20,8 @@ - - + + @@ -30,7 +30,7 @@

    API Tokens and Client Keys

    For Unleash to be of any use, it requires at least a server and a consuming client. More advanced use cases may call for multiple clients, automated feature toggle updates, the Unleash proxy and Unleash proxy clients, and more. To facilitate communication between all these moving parts, Unleash uses a system of API tokens and client keys, all with a specific purpose in mind.

    This document details the three kinds of tokens and keys that you will need to fully connect any Unleash system:

    API tokens

    tip

    This section describes what API tokens are. For information on how to create them, refer to the how-to guide for creating API tokens.

    Use API tokens to connect to the Unleash server API. API tokens come in four distinct types:

    All types use the same format but have different intended uses. Admin and client tokens are secrets and should not be exposed to end users. Front-end tokens, on the other hand, are not secret.

    The parts of an API token

    Admin, client and front-end tokens contain the following pieces of information:

    NameDescription
    Token name (sometimes called "username")The token's name. Names are not required to be unique.
    TypeWhat kind of token it is: admin, client, or front-end.
    ProjectsWhat projects a token has access to.
    EnvironmentWhat environment the token has access to.

    Personal access tokens follow their own special format, and only contain an optional description for the token and an expiry date.

    API token visibility

    project-level visibility

    Project-level visibility and access to API tokens was introduced in Unleash 4.22.

    By default, only admin users can create API tokens, and only admins can see their values.

    However, any [client](#client-tokens client tokens) and front-end tokens that are applicable to a project, will also be visible to any members of that project that have the READ_PROJECT_API_TOKEN permission (all project members by default).

    Similarly, any project members with the CREATE_PROJECT_API_TOKEN permission can also create client and front-end tokens for that specific project (how to create project API tokens).

    Admin tokens

    Admin tokens grant full read and write access to all resources in the Unleash server API. Admin tokens have access to all projects, all environments, and all root resources (find out more about resources in the RBAC document).

    Use admin tokens to:

    • Automate Unleash behavior such as creating feature toggles, projects, etc.

    • Write custom Unleash UIs to replace the default Unleash admin UI. Do not use admin tokens for:

    • Client SDKs: You will not be able to read toggle data from multiple environments. Use client tokens instead.

    Support for scoped admin tokens with more fine-grained permissions is currently in the planning stage.

    Deprecation Notice We do not recommend using admin tokens anymore, they are not connected to any user, and as such is a lot harder to track.

    Personal access tokens

    Personal access tokens are a special form of admin tokens and grant access to the same resources that the user that created them has access to. These permissions are dynamic, so if a user's permissions change through addition of a custom role, the token will likewise have altered permissions.

    When using a personal access token to modify resources, the event log will list the token creator's name for that operation.

    Personal access tokens with a lifetime will stop working after the expiration date.

    Use personal access tokens to:

    • Provide more fine-grained permissions for automation than an admin token provides
    • Give temporary access to an automation tool
    On token expiration

    It is possible to set a token's expiration date to never. However, a token that doesn't expire brings with it a few security concerns. We recommend that you use tokens with expiration dates whenever possible.

    Do not use personal access tokens for:

    • Client SDKs: You will not be able to read toggle data from multiple environments. Use client tokens instead.
    • Write custom Unleash UIs: Personal access tokens may expire and their permissions may change. It's better to use admin tokens tokens instead.

    Client tokens

    Client tokens are intended for use in server-side client SDKs (including the Unleash Proxy) and grant the user permissions to:

    • Read feature toggle information
    • Register applications with the Unleash server
    • Send usage metrics

    When creating a client token, you can choose which projects it should be able to read data from. You can give it access to a specific list of projects or to all projects (including projects that don't exist yet). Prior to Unleash 4.10, a token could be valid only for a single project or all projects.

    Each client token is only valid for a single environment.

    Use client tokens:

    Do not use client tokens in:

    Front-end tokens

    Front-end tokens are used with front-end SDKs when used with the Unleash front-end API. They grant the user permission to:

    • Read the enabled toggled for a given context
    • Register applications with the Unleash server
    • Send usage metrics

    As with client tokens, front-end tokens can read data from one, multiple, or all existing projects.

    Each front-end token is only valid for a single environment.

    Use front-end tokens in:

    Do not use front-end tokens in:

    Format

    API tokens come in one of two formats. When we introduced environments in Unleash 4.3, we updated the format of the tokens to provide more human-readable information to the user. Both formats are still valid (you don't need to update a working token of the old format) and are described below.

    Version 1

    The first version of API tokens was a 64 character long hexadecimal string. Example:

    be44368985f7fb3237c584ef86f3d6bdada42ddbd63a019d26955178

    Version 2

    API tokens consist of three parts:

    1. Project(s)
    2. Environment
    3. Hash

    The parts are separated by two different separators: A colon (:) between the project(s) and the environment, and a full stop (.) between the environment and the hash.

    The project(s) part is one of:

    • The id of a specific project, for example: default. This indicates that the token is only valid for this project.
    • A pair of opening and closing square brackets: []. This indicates that the token is valid for a discrete list of projects. The list of projects is not shown in the token.
    • An asterisk: *. This indicates that the token is valid for all projects (current and future).

    The environment is the name of an environment on your Unleash server, such as development.

    The hash is 64 character long hexadecimal string.

    Personal access tokens do not contain project or environment information, since they mimic the user that created them. Instead, the token starts with the string user.

    Some example client tokens are:

    • A token with access to toggles in the "development" environment of a single project, "project-a":
      project-a:development.be44368985f7fb3237c584ef86f3d6bdada42ddbd63a019d26955178
    • A token with access to toggles in the "production" environment multiple projects:
      []:production.be44368985f7fb3237c584ef86f3d6bdada42ddbd63a019d26955178
    • A token with access to toggles in the "development" environment of all projects:
      *:development.be44368985f7fb3237c584ef86f3d6bdada42ddbd63a019d26955178
    • A personal access token:
      user:be7536c3a160ff15e3a92da45de531dd54bc1ae15d8455c0476f086b

    Proxy client keys

    Use proxy client keys to connect Proxy client SDKs (front-end SDKs) to the Unleash Proxy. As opposed to the API tokens, Proxy client keys are not considered secret and are safe to use on any clients (refer to the the proxy documentation for more about privacy). They do not let you connect to the Unleash server API.

    Proxy client keys are arbitrary strings that you must provide the Unleash proxy with on startup. They can be whatever you want and you create them yourself.

    Creating proxy client keys

    To designate a string as a proxy client key, add it to the clientKeys list when starting the proxy, as mentioned in the configuration section of the Unleash proxy documentation. Connecting clients should then specify the same string as their client key.

    Unleash does not generate proxy client keys for you. Because of this, they have no specific format.

    Use Proxy client keys to:

    Do not use Proxy client keys to:

    • Connect to the Unleash API. It will not work. Use an appropriate API token instead.
    - - + + \ No newline at end of file diff --git a/reference/api/legacy/unleash.html b/reference/api/legacy/unleash.html index 51f4267dcc..62542e7336 100644 --- a/reference/api/legacy/unleash.html +++ b/reference/api/legacy/unleash.html @@ -20,15 +20,15 @@ - - + +

    Legacy API Documentation

    caution

    The docs in this category are legacy documentation. You should prefer to use the Unleash OpenAPI docs instead whenever possible.

    Client API

    This describes the API provided to unleash-clients.

    Since v4.0.0 all operations require an API token with Client level access.

    With versions earlier than v4.0.0 and insecure authentication no authentication is required.

    Admin API (internal)

    The internal API used by the Admin UI (unleash-frontend). Since v4.0.0 all operations require an API token with Admin level access:

    With versions earlier than v4.0.0 and insecure authentication Basic Auth (with curl -u myemail@test.com:) is enough.

    System API's

    Content-Type

    All endpoints require application/json as content type, so if you're using curl remember to add

    -H "Content-Type: application/json"
    - - + + \ No newline at end of file diff --git a/reference/api/legacy/unleash/admin/addons.html b/reference/api/legacy/unleash/admin/addons.html index fb6d96978e..322fde13dc 100644 --- a/reference/api/legacy/unleash/admin/addons.html +++ b/reference/api/legacy/unleash/admin/addons.html @@ -20,15 +20,15 @@ - - + +

    /api/admin/addons

    In order to access the admin API endpoints you need to identify yourself. Unless you're using the none authentication method, you'll need to create an ADMIN token and add an Authorization header using the token.

    List integrations and providers

    GET https://unleash.host.com/api/admin/addons

    Returns a list of configured integrations and available integration providers.

    Example response:

    {
    "addons": [
    {
    "id": 30,
    "provider": "webhook",
    "enabled": true,
    "description": "post updates to slack",
    "parameters": {
    "url": "http://localhost:4242/webhook"
    },
    "events": ["feature-created", "feature-updated"]
    },
    {
    "id": 33,
    "provider": "slack",
    "enabled": true,
    "description": "default",
    "parameters": {
    "defaultChannel": "integration-demo-instance",
    "url": "https://hooks.slack.com/someurl"
    },
    "events": ["feature-created", "feature-updated"]
    }
    ],
    "providers": [
    {
    "name": "webhook",
    "displayName": "Webhook",
    "description": "Webhooks are a simple way to post messages from Unleash to third party services. Unleash make use of normal HTTP POST with a payload you may define yourself.",
    "parameters": [
    {
    "name": "url",
    "displayName": "Webhook URL",
    "type": "url",
    "required": true
    },
    {
    "name": "bodyTemplate",
    "displayName": "Body template",
    "description": "You may format the body using a mustache template. If you don't specify anything, the format will be similar to the /api/admin/events format",
    "type": "textfield",
    "required": false
    }
    ],
    "events": [
    "feature-created",
    "feature-updated",
    "feature-archived",
    "feature-revived"
    ]
    },
    {
    "name": "slack",
    "displayName": "Slack",
    "description": "Integrates Unleash with Slack.",
    "parameters": [
    {
    "name": "url",
    "displayName": "Slack webhook URL",
    "type": "url",
    "required": true
    },
    {
    "name": "defaultChannel",
    "displayName": "Default channel",
    "description": "Default channel to post updates to if not specified in the slack-tag",
    "type": "text",
    "required": true
    }
    ],
    "events": [
    "feature-created",
    "feature-updated",
    "feature-archived",
    "feature-revived"
    ],
    "tags": [
    {
    "name": "slack",
    "description": "Slack tag used by the slack integration to specify the slack channel.",
    "icon": "S"
    }
    ]
    }
    ]
    }

    Create a new integration configuration

    POST https://unleash.host.com/api/addons

    Creates an integration configuration for an integration provider.

    Body

    {
    "provider": "webhook",
    "description": "Optional description",
    "enabled": true,
    "parameters": {
    "url": "http://localhost:4242/webhook"
    },
    "events": ["feature-created", "feature-updated"]
    }

    Notes

    • provider must be a valid integration provider

    Update new integration configuration

    POST https://unleash.host.com/api/addons/:id

    Updates an integration configuration.

    Body

    {
    "provider": "webhook",
    "description": "Optional updated description",
    "enabled": true,
    "parameters": {
    "url": "http://localhost:4242/webhook"
    },
    "events": ["feature-created", "feature-updated"]
    }

    Notes

    • provider can not be changed.

    Delete an integration configuration

    DELETE https://unleash.host.com/api/admin/addons/:id

    Deletes the integration with id=id.

    - - + + \ No newline at end of file diff --git a/reference/api/legacy/unleash/admin/archive.html b/reference/api/legacy/unleash/admin/archive.html index 182b6715ce..42c2b3484c 100644 --- a/reference/api/legacy/unleash/admin/archive.html +++ b/reference/api/legacy/unleash/admin/archive.html @@ -20,15 +20,15 @@ - - + +

    /api/admin/archive

    In order to access the admin API endpoints you need to identify yourself. Unless you're using the none authentication method, you'll need to create an ADMIN token and add an Authorization header using the token.

    Fetch archived toggles

    GET http://unleash.host.com/api/admin/archive/features

    Used to fetch list of archived feature toggles

    Example response:

    {
    "version": 1,
    "features": [
    {
    "name": "Feature.A",
    "description": "lorem ipsum",
    "type": "release",
    "stale": false,
    "variants": [],
    "tags": [],
    "strategy": "default",
    "parameters": {}
    }
    ]
    }

    Revive feature toggle

    POST http://unleash.host.com/api/admin/archive/revive/:featureName

    Response: 200 OK - When feature toggle was successfully revived.

    Delete an archived feature toggle

    DELETE http://unleash.host.com/api/admin/archive/:featureName

    Will fully remove the feature toggle and associated configuration. Impossible to restore after this action.

    - - + + \ No newline at end of file diff --git a/reference/api/legacy/unleash/admin/context.html b/reference/api/legacy/unleash/admin/context.html index 4f3f4b5246..9209461a33 100644 --- a/reference/api/legacy/unleash/admin/context.html +++ b/reference/api/legacy/unleash/admin/context.html @@ -20,15 +20,15 @@ - - + +

    /api/admin/context

    The context feature is only available as part of Unleash Enterprise. In order to access the API programmatically you need to make sure you obtain a API token with admin permissions.

    List context fields defined in Unleash

    GET https://unleash.host.com/api/admin/context

    Returns a list of context fields defined in Unleash.

    Example response:

    [
    {
    "name": "appName",
    "description": "Allows you to constrain on application name",
    "stickiness": false,
    "sortOrder": 2,
    "createdAt": "2020-03-05T19:33:19.784Z"
    },
    {
    "name": "environment",
    "description": "Allows you to constrain on application environment",
    "stickiness": false,
    "sortOrder": 0,
    "legalValues": ["qa", "dev", "prod"],
    "createdAt": "2020-03-05T19:33:19.784Z"
    },
    {
    "name": "tenantId",
    "description": "Control rollout to your tenants",
    "stickiness": true,
    "sortOrder": 10,
    "legalValues": ["company-a, company-b"],
    "createdAt": "2020-03-05T19:33:19.784Z"
    },
    {
    "name": "userId",
    "description": "Allows you to constrain on userId",
    "stickiness": false,
    "sortOrder": 1,
    "createdAt": "2020-03-05T19:33:19.784Z"
    }
    ]

    Create a new context field

    POST https://unleash.host.com/api/admin/context

    Creates a new context field.

    Body

    {
    "name": "region",
    "description": "Control rollout based on region",
    "legalValues": ["asia", "eu", "europe"],
    "stickiness": true
    }

    Update a context field

    PUT https://unleash.host.com/api/context/:name

    Updates a new context field

    Body

    {
    "name": "region",
    "description": "Control rollout based on region",
    "legalValues": ["asia", "eu"],
    "stickiness": true
    }

    Delete a context field

    DELETE https://unleash.host.com/api/admin/context/:name

    Deletes the context field with name=name.

    - - + + \ No newline at end of file diff --git a/reference/api/legacy/unleash/admin/events.html b/reference/api/legacy/unleash/admin/events.html index fc8d9fb86e..f326e59d7a 100644 --- a/reference/api/legacy/unleash/admin/events.html +++ b/reference/api/legacy/unleash/admin/events.html @@ -20,15 +20,15 @@ - - + +

    /api/admin/events

    note

    In order to access the admin API endpoints you need to identify yourself. Unless you're using the none authentication method, you'll need to create an ADMIN token and add an Authorization header using the token.

    The Events API lets you retrieve events from your Unleash instance.

    Event endpoints

    Get all events

    Retrieve all events from the Unleash instance.
    GET <unleash-url>/api/admin/events
    Authorization: <API-token>
    content-type: application/json

    Query parameters

    Query parameterDescriptionRequired
    projectWhen applied, the endpoint will only return events from the given project.No

    When called without any query parameters, the endpoint will return the last 100 events from the Unleash instance. When called with a project query parameter, it will return only events related to that project, but it will return all the events, and not just the last 100.

    Get events by project

    Retrieve all events belonging to the given project.
    GET <unleash-url>/api/admin/events?project=<project-name>
    Authorization: <API-token>
    content-type: application/json

    Use the project query parameter to make the API return all events pertaining to the given project.

    Responses

    Responses
    200 OK

    The last 100 events from the Unleash server when called without a project query parameter.

    When called with a project query parameter: all events related to that project.

    Successful response; a list of events
    {
    "version": 1,
    "events": [
    {
    "id": 842,
    "type": "feature-environment-enabled",
    "createdBy": "user@company.com",
    "createdAt": "2022-05-12T08:49:49.521Z",
    "data": null,
    "preData": null,
    "tags": [],
    "featureName": "my-constrained-toggle",
    "project": "my-project",
    "environment": "development"
    },
    {
    "id": 841,
    "type": "feature-environment-disabled",
    "createdBy": "user@company.com",
    "createdAt": "2022-05-12T08:49:45.986Z",
    "data": null,
    "preData": null,
    "tags": [],
    "featureName": "my-constrained-toggle",
    "project": "my-project",
    "environment": "development"
    }
    ]
    }

    Get events for a specific toggle

    Retrieve all events related to the given toggle.
    GET <unleash-url>/api/admin/events/<toggle-name>
    Authorization: <API-token>
    content-type: application/json

    Fetch all events related to a specified toggle.

    Responses

    Responses
    200 OK

    The list of events related to the given toggle.

    Successful response; all events relating to the specified toggle
    {
    "toggleName": "my-constrained-toggle",
    "events": [
    {
    "id": 842,
    "type": "feature-environment-enabled",
    "createdBy": "user@company.com",
    "createdAt": "2022-05-12T08:49:49.521Z",
    "data": null,
    "preData": null,
    "tags": [],
    "featureName": "my-constrained-toggle",
    "project": "my-project",
    "environment": "development"
    },
    {
    "id": 841,
    "type": "feature-environment-disabled",
    "createdBy": "user@company.com",
    "createdAt": "2022-05-12T08:49:45.986Z",
    "data": null,
    "preData": null,
    "tags": [],
    "featureName": "my-constrained-toggle",
    "project": "my-project",
    "environment": "development"
    }
    ]
    }

    Event type description

    Content moved

    This section has been moved to a dedicated event type reference document.

    - - + + \ No newline at end of file diff --git a/reference/api/legacy/unleash/admin/feature-types.html b/reference/api/legacy/unleash/admin/feature-types.html index 96fb3aacac..c06397ded9 100644 --- a/reference/api/legacy/unleash/admin/feature-types.html +++ b/reference/api/legacy/unleash/admin/feature-types.html @@ -20,15 +20,15 @@ - - + +

    /api/admin/feature-types

    In order to access the admin API endpoints you need to identify yourself. Unless you're using the none authentication method, you'll need to create an ADMIN token and add an Authorization header using the token.

    Feature Types API

    GET: http://unleash.host.com/api/admin/feature-types

    Used to fetch all feature types defined in the unleash system.

    Response

    {
    "version": 1,
    "types": [
    {
    "id": "release",
    "name": "Release",
    "description": "Used to enable trunk-based development for teams practicing Continuous Delivery.",
    "lifetimeDays": 40
    },
    {
    "id": "experiment",
    "name": "Experiment",
    "description": "Used to perform multivariate or A/B testing.",
    "lifetimeDays": 40
    },
    {
    "id": "ops",
    "name": "Operational",
    "description": "Used to control operational aspects of the system behavior.",
    "lifetimeDays": 7
    },
    {
    "id": "killswitch",
    "name": "Kill switch",
    "description": "Used to to gracefully degrade system functionality.",
    "lifetimeDays": null
    },
    {
    "id": "permission",
    "name": "Permission",
    "description": "Used to change the features or product experience that certain users receive.",
    "lifetimeDays": null
    }
    ]
    }
    - - + + \ No newline at end of file diff --git a/reference/api/legacy/unleash/admin/features-v2.html b/reference/api/legacy/unleash/admin/features-v2.html index 648c6ea134..7013999c16 100644 --- a/reference/api/legacy/unleash/admin/features-v2.html +++ b/reference/api/legacy/unleash/admin/features-v2.html @@ -20,8 +20,8 @@ - - + + @@ -29,7 +29,7 @@

    /api/admin/projects/:projectId

    info

    In order to access the admin API endpoints you need to identify yourself. Unless you're using the none authentication method, you'll need to create an admin token and add an Authorization header using the token.

    Fetching Feature Toggles

    Available since Unleash v4.3

    In this document we will guide you on how you can work with feature toggles and their configuration. Please remember the following details:

    • All feature toggles exists inside a project.
    • A feature toggle exists across all environments.
    • A feature toggle can take different configuration, activation strategies, per environment.
    note

    This document lists HTTP request data and cURL and HTTPie command examples for all endpoints. Further examples use HTTPie.

    Get Project Overview

    Get a project overview
    GET <unleash-url>/api/admin/projects/:project-id
    Authorization: <API-token>
    content-type: application/json

    This endpoint will give you an general overview of a project. It will return essential details about a project, in addition it will return all feature toggles and high level environment details per feature toggle.

    Example Query

    http GET http://localhost:4242/api/admin/projects/default Authorization:$KEY

    Example response:

    {
    "description": "Default project",
    "features": [
    {
    "createdAt": "2021-08-31T08:00:33.335Z",
    "environments": [
    {
    "displayName": "Development",
    "enabled": false,
    "name": "development"
    },
    {
    "displayName": "Production",
    "enabled": false,
    "name": "production"
    }
    ],
    "lastSeenAt": null,
    "name": "demo",
    "stale": false,
    "type": "release"
    },
    {
    "createdAt": "2021-08-31T09:43:13.686Z",
    "environments": [
    {
    "displayName": "Development",
    "enabled": false,
    "name": "development"
    },
    {
    "displayName": "Production",
    "enabled": false,
    "name": "production"
    }
    ],
    "lastSeenAt": null,
    "name": "demo.test",
    "stale": false,
    "type": "release"
    }
    ],
    "health": 100,
    "members": 2,
    "name": "Default",
    "version": 1
    }

    From the results we can see that we have received two feature toggles, demo, demo.test, and other useful metadata about the project.

    Get All Feature Toggles

    Get all feature toggles in a project
    GET <unleash-url>/api/admin/projects/:projectId/features
    Authorization: <API-token>
    content-type: application/json

    This endpoint will return all feature toggles and high level environment details per feature toggle for a given projectId

    Example Query

    http GET http://localhost:4242/api/admin/projects/default/features \
    Authorization:$KEY

    Example response:

    {
    "features": [
    {
    "createdAt": "2021-08-31T08:00:33.335Z",
    "environments": [
    {
    "displayName": "Development",
    "enabled": false,
    "name": "development"
    },
    {
    "displayName": "Production",
    "enabled": false,
    "name": "production"
    }
    ],
    "lastSeenAt": null,
    "name": "demo",
    "stale": false,
    "type": "release"
    },
    {
    "createdAt": "2021-08-31T09:43:13.686Z",
    "environments": [
    {
    "displayName": "Development",
    "enabled": false,
    "name": "development"
    },
    {
    "displayName": "Production",
    "enabled": false,
    "name": "production"
    }
    ],
    "lastSeenAt": null,
    "name": "demo.test",
    "stale": false,
    "type": "release"
    }
    ],
    "version": 1
    }

    Create Feature Toggle

    Create a feature toggle with the specified details (example data)
    POST <unleash-url>/api/admin/projects/:projectId/features
    Authorization: <API-token>
    content-type: application/json

    {
    "name": "my-feature-toggle"
    }

    This endpoint will accept HTTP POST request to create a new feature toggle for a given projectId

    Toggle options

    This endpoint accepts the following toggle options:

    Property nameRequiredDescriptionExample value
    nameYesThe name of the feature toggle."my-feature-toggle"
    descriptionNoThe feature toggle's description. Defaults to an empty string."Turn my feature on!"
    impressionDataNoWhether to enable impression data for this toggle. Defaults to false.true
    typeNoThe type of toggle you want to create. Defaults to "release""release"

    Example Query

    echo '{"name": "demo2", "description": "A new feature toggle"}' | \
    http POST http://localhost:4242/api/admin/projects/default/features \
    Authorization:$KEY`

    Example response:

    HTTP/1.1 201 Created
    Access-Control-Allow-Origin: *
    Connection: keep-alive
    Content-Length: 159
    Content-Type: application/json; charset=utf-8
    Date: Tue, 07 Sep 2021 20:16:02 GMT
    ETag: W/"9f-4btEokgk0N74zuBVKKxws0IBu4w"
    Keep-Alive: timeout=60
    Vary: Accept-Encoding

    {
    "createdAt": "2021-09-07T20:16:02.614Z",
    "description": "A new feature toggle",
    "lastSeenAt": null,
    "name": "demo2",
    "project": "default",
    "stale": false,
    "type": "release",
    "variants": null
    }

    Possible Errors:

    • 409 Conflict - A toggle with that name already exists

    Get Feature Toggle

    Retrieve a named feature toggle
    GET <unleash-url>/api/admin/projects/:projectId/features/:featureName
    Authorization: <API-token>
    content-type: application/json

    This endpoint will return the feature toggles with the defined name and projectId. We will also see the list of environments and all activation strategies configured per environment.

    Example Query

    http GET http://localhost:4242/api/admin/projects/default/features/demo \
    Authorization:$KEY`

    Example response:

    {
    "archived": false,
    "createdAt": "2021-08-31T08:00:33.335Z",
    "description": null,
    "environments": [
    {
    "enabled": false,
    "name": "development",
    "strategies": [
    {
    "constraints": [],
    "id": "8eaa8abb-0e03-4dbb-a440-f3bf193917ad",
    "name": "default",
    "parameters": null
    }
    ]
    },
    {
    "enabled": false,
    "name": "production",
    "strategies": []
    }
    ],
    "lastSeenAt": null,
    "name": "demo",
    "project": "default",
    "stale": false,
    "type": "release",
    "variants": null
    }

    Possible Errors:

    • 404 Not Found - Could not find feature toggle with the provided name.

    Update Feature Toggle

    Update a feature toggle entry (example data)
    PUT <unleash-url>/api/admin/projects/:projectId/features/:featureName
    Authorization: <API-token>
    content-type: application/json

    {
    "name": "demo",
    "description": "An updated feature toggle description."
    }

    This endpoint will accept HTTP PUT request to update the feature toggle metadata.

    Example Query

    echo '{"name": "demo", "description": "An update feature toggle", "type": "kill-switch"}' | \
    http PUT http://localhost:4242/api/admin/projects/default/features/demo \
    Authorization:$KEY`

    Example response:

    {
    "createdAt": "2021-09-07T20:16:02.614Z",
    "description": "An update feature toggle",
    "lastSeenAt": null,
    "name": "demo",
    "project": "default",
    "stale": false,
    "type": "kill-switch",
    "variants": null
    }

    Some fields is not possible to change via this endpoint:

    • name
    • project
    • createdAt
    • lastSeen

    Patch Feature Toggle

    Patch a feature toggle (example data)
    PATCH <unleash-url>/api/admin/projects/:projectId/features/:featureName
    Authorization: <API-token>
    content-type: application/json

    [
    {
    "op": "replace",
    "path": "/description",
    "value": "patched description"
    }
    ]

    This endpoint will accept HTTP PATCH request to update the feature toggle metadata.

    Example Query

    echo '[{"op": "replace", "path": "/description", "value": "patched desc"}]' | \
    http PATCH http://localhost:4242/api/admin/projects/default/features/demo \
    Authorization:$KEY`

    Example response:

    {
    "createdAt": "2021-09-07T20:16:02.614Z",
    "description": "patched desc",
    "lastSeenAt": null,
    "name": "demo",
    "project": "default",
    "stale": false,
    "type": "release",
    "variants": null
    }

    Some fields is not possible to change via this endpoint:

    • name
    • project
    • createdAt
    • lastSeen

    Clone Feature Toggle

    Clone a feature toggle (example data)
    POST <unleash-url>/api/admin/projects/:projectId/features/:featureName/clone
    Authorization: <API-token>
    content-type: application/json

    {
    "name": "newToggleName"
    }

    This endpoint will accept HTTP POST request to clone an existing feature toggle with all strategies and variants. When cloning a toggle, you must provide a new name for it. You can not clone archived feature toggles. The newly created feature toggle will be disabled for all environments.

    Example Query

    echo '{ "name": "DemoNew" }' | \
    http POST http://localhost:4242/api/admin/projects/default/features/Demo/clone \
    Authorization:$KEY`

    Example response:

    HTTP/1.1 201 Created
    Access-Control-Allow-Origin: *
    Connection: keep-alive
    Content-Length: 260
    Content-Type: application/json; charset=utf-8
    Date: Wed, 06 Oct 2021 20:04:39 GMT
    ETag: W/"104-joC/gdjtJ29jZMxj91lIzR42Pmo"
    Keep-Alive: timeout=60
    Vary: Accept-Encoding

    {
    "createdAt": "2021-09-29T10:22:28.523Z",
    "description": "Some useful description",
    "lastSeenAt": null,
    "name": "DemoNew",
    "project": "default",
    "stale": false,
    "type": "release",
    "variants": [
    {
    "name": "blue",
    "overrides": [],
    "stickiness": "default",
    "weight": 1000,
    "weightType": "variable"
    }
    ]
    }

    Possible Errors:

    • 409 Conflict - A toggle with that name already exists

    Archive Feature Toggle

    Archive a named feature toggle
    DELETE <unleash-url>/api/admin/projects/:projectId/features/:featureName
    Authorization: <API-token>
    content-type: application/json

    This endpoint will accept HTTP DELETE requests to archive a feature toggle.

    Example Query

    http DELETE http://localhost:4242/api/admin/projects/default/features/demo \
    Authorization:$KEY`

    Example response:

    HTTP/1.1 202 Accepted
    Access-Control-Allow-Origin: *
    Connection: keep-alive
    Date: Wed, 08 Sep 2021 20:09:21 GMT
    Keep-Alive: timeout=60
    Transfer-Encoding: chunked

    Add strategy to Feature Toggle

    Add a new strategy to the named feature toggle in the named environment (example data)
    POST <unleash-url>/api/admin/projects/:projectId/features/:featureName/environments/:environment/strategies
    Authorization: <API-token>
    content-type: application/json

    {
    "name": "flexibleRollout",
    "parameters": {
    "rollout": 20,
    "groupId": "demo",
    "stickiness": "default"
    }
    }

    This endpoint will allow you to add a new strategy to a feature toggle in a given environment.

    Example Query

    echo '{"name": "flexibleRollout",
    "parameters": {
    "rollout": 20,
    "groupId": "demo",
    "stickiness": "default"
    }
    }' | \
    http POST \
    http://localhost:4242/api/admin/projects/default/features/demo/environments/production/strategies \
    Authorization:$KEY

    Example response:

    {
    "constraints": [],
    "id": "77bbe972-ffce-49b2-94d9-326593e2228e",
    "name": "flexibleRollout",
    "parameters": {
    "groupId": "demo",
    "rollout": 20,
    "stickiness": "default"
    }
    }

    Update strategy configuration

    Overwrite the specified strategy on the named feature toggle in the named environment (example data)
    PUT <unleash-url>/api/admin/projects/:projectId/features/:featureName/environments/:environment/strategies/:strategy-id
    Authorization: <API-token>
    content-type: application/json

    {
    "name": "flexibleRollout",
    "parameters": {
    "rollout": 25,
    "groupId": "demo",
    "stickiness": "default"
    }
    }

    Example Query

    echo '{"name": "flexibleRollout",
    "parameters": {
    "rollout": 25,
    "groupId": "demo",
    "stickiness": "default"
    }
    }' | \
    http PUT \
    http://localhost:4242/api/admin/projects/default/features/demo/environments/production/strategies/77bbe972-ffce-49b2-94d9-326593e2228e \
    Authorization:$KEY

    Example response:

    {
    "constraints": [],
    "id": "77bbe972-ffce-49b2-94d9-326593e2228e",
    "name": "flexibleRollout",
    "parameters": {
    "groupId": "demo",
    "rollout": 20,
    "stickiness": "default"
    }
    }

    Patch strategy configuration

    Patch update a strategy definition (example data)
    PATCH <unleash-url>/api/admin/projects/:project-id/features/:featureName/environments/:environment/strategies/:strategyId
    Authorization: <API-token>
    content-type: application/json

    [
    {
    "op": "replace",
    "path": "/parameters/rollout",
    "value": 50
    }
    ]

    Example Query

    echo '[{"op": "replace", "path": "/parameters/rollout", "value": 50}]' | \
    http PATCH \
    http://localhost:4242/api/admin/projects/default/features/demo/environments/production/strategies/ea5404e5-0c0d-488c-93b2-0a2200534827 \
    Authorization:$KEY

    Example response:

    {
    "constraints": [],
    "id": "ea5404e5-0c0d-488c-93b2-0a2200534827",
    "name": "flexibleRollout",
    "parameters": {
    "groupId": "demo",
    "rollout": 50,
    "stickiness": "default"
    }
    }

    Delete strategy from Feature Toggle

    Delete the strategy with the given ID
    DELETE <unleash-url>/api/admin/projects/:projectId/features/:featureName/environments/:environment/strategies/:strategyId
    Authorization: <API-token>
    content-type: application/json

    Example Query

    http DELETE http://localhost:4242/api/admin/projects/default/features/demo/environments/production/strategies/77bbe972-ffce-49b2-94d9-326593e2228e Authorization:$KEY

    Example response:

    HTTP/1.1 200 OK
    Access-Control-Allow-Origin: *
    Connection: keep-alive
    Content-Type: application/json; charset=utf-8
    Date: Tue, 07 Sep 2021 20:47:55 GMT
    Keep-Alive: timeout=60
    Transfer-Encoding: chunked
    Vary: Accept-Encoding

    Enabling and disabling toggles

    Enable Feature Toggle in an Environment

    Activate the named toggle in the given environment
    POST <unleash-url>/api/admin/projects/:projectId/features/:featureName/environments/:environment/on
    Authorization: <API-token>
    content-type: application/json

    Example Query

    http POST http://localhost:4242/api/admin/projects/default/features/demo/environments/development/on Authorization:$KEY --json

    Example response:

    HTTP/1.1 200 OK
    Access-Control-Allow-Origin: *
    Connection: keep-alive
    Date: Tue, 07 Sep 2021 20:49:51 GMT
    Keep-Alive: timeout=60
    Transfer-Encoding: chunked

    Possible Errors:

    • 409 Conflict - You can not enable the environment before it has strategies.

    Disable Feature Toggle in an Environment

    Disable the named toggle in the given environment
    POST <unleash-url>/api/admin/projects/:projectId/features/:featureName/environments/:environment/off
    Authorization: <API-token>
    content-type: application/json

    Example Query

    http POST http://localhost:4242/api/admin/projects/default/features/demo/environments/development/off Authorization:$KEY --json

    Example response:

    HTTP/1.1 200 OK
    Access-Control-Allow-Origin: *
    Connection: keep-alive
    Date: Tue, 07 Sep 2021 20:49:51 GMT
    Keep-Alive: timeout=60
    Transfer-Encoding: chunked

    Feature Variants

    Put variants for Feature Toggle

    caution

    From 4.21 variants are tied to an environment. Check our Open API docs for the up-to-date docs: https://docs.getunleash.io/reference/api/unleash/features

    This endpoint affects all environments at once. If you only want to update a single environment, use #update-variants-per-environment instead.

    Create (overwrite) variants for a feature toggle in all environments (example data)
    PUT <unleash-url>/api/admin/projects/:projectId/features/:featureName/variants
    Authorization: <API-token>
    content-type: application/json

    [
    {
    "name": "variant1",
    "weightType": "fix",
    "weight": 650,
    "payload": {
    "type": "json",
    "value": "{\"key1\": \"value\", \"key2\": 123}"
    },
    "stickiness": "userId",
    "overrides": [
    {
    "contextName": "userId",
    "values": [
    "1",
    "23"
    ]
    }
    ]
    },
    {
    "name": "variant2",
    "weightType": "variable",
    "weight": 123
    }
    ]

    This overwrites the current variants for the feature toggle specified in the :featureName parameter. The backend will validate the input for the following invariants

    • If there are variants, there needs to be at least one variant with weightType: variable
    • The sum of the weights of variants with weightType: fix must be below 1000 (< 1000)

    The backend will also distribute remaining weight up to 1000 after adding the variants with weightType: fix together amongst the variants of weightType: variable

    Example Query

    echo '[
    {
    "name": "variant1",
    "weightType": "fix",
    "weight": 650,
    "payload": {
    "type": "json",
    "value": "{\"key1\": \"value\", \"key2\": 123}"
    },
    "stickiness": "userId",
    "overrides": [{
    "contextName": "userId",
    "values": ["1", "23"]
    }]
    },
    {
    "name": "variant2",
    "weightType": "variable",
    "weight": 123
    }
    ]' | \
    http PUT http://localhost:4242/api/admin/projects/default/features/demo/variants Authorization:$KEY

    Example response:

    HTTP/1.1 200 OK
    Access-Control-Allow-Origin: *
    Connection: keep-alive
    Date: Tue, 23 Nov 2021 08:46:32 GMT
    Keep-Alive: timeout=60
    Transfer-Encoding: chunked
    Content-Type: application/json; charset=utf-8

    {
    "version": "1",
    "variants": [
    {
    "name": "variant2",
    "weightType": "variable",
    "weight": 350
    },
    {
    "name": "variant1",
    "weightType": "fix",
    "weight": 650
    }
    ]
    }

    PATCH variants for a feature toggle

    caution

    This API documentation is no longer maintained. You should use the OpenAPI docs for this endpoint instead.

    This endpoint affects all environments at once. If you only want to update a single environment, use the operation for updating variants per enviroment instead.

    Patch variants for a feature toggle in all environments (example data)
    PATCH <unleash-url>/api/admin/projects/:projectId/features/:featureName/variants
    Authorization: <API-token>
    content-type: application/json

    [
    {
    "op": "add",
    "path": "/1",
    "value": {
    "name": "new-variant",
    "weightType": "fix",
    "weight": 200
    }
    }
    ]

    Example Query

    echo '[{"op": "add", "path": "/1", "value": {
    "name": "new-variant",
    "weightType": "fix",
    "weight": 200
    }}]' | \
    http PATCH \
    http://localhost:4242/api/admin/projects/default/features/demo/variants \
    Authorization:$KEY

    Example Response

    {
    "version": "1",
    "variants": [
    {
    "name": "variant2",
    "weightType": "variable",
    "weight": 150
    },
    {
    "name": "new-variant",
    "weightType": "fix",
    "weight": 200
    },
    {
    "name": "variant1",
    "weightType": "fix",
    "weight": 650
    }
    ]
    }

    Manage project users and roles

    You can add and remove users to a project using the /api/admin/projects/:projectId/users/:userId/roles/:roleId endpoint. When adding or removing users, you must also provide the ID for the role to give them (when adding) or the ID of the role they currently have (when removing).

    Add a user to a project

    Add a user to a project (example data)
    POST <unleash-url>/api/admin/projects/:projectId/users/:userId/roles/:roleId
    Authorization: <API-token>
    content-type: application/json

    {
    "userId": 25,
    "projectId": "myProject",
    "roleId": "1"
    }

    This will add a user to a project and give the user a specified role within that project.

    URL parameters

    ParameterTypeDescriptionExample value
    userIdintegerThe ID of the user you want to add to the project.1
    projectIdstringThe id of the project to add the user to."MyCoolProject"
    roleIdintegerThe id of the role you want to assign to the new user in the project.7

    Responses

    Responses data
    200 OK

    The user was added to the project with the specified role. This response has no body.

    400 Bad Request

    The user already exists in the project and cannot be added again:

    [
    {
    "msg": "User already has access to project=<projectId>"
    }
    ]

    Example query

    The following query would add the user with ID 42 to the MyCoolProject project and give them the role with ID 13.

    http POST \
    http://localhost:4242/api/admin/projects/MyCoolProject/users/42/roles/13 \
    Authorization:$KEY

    Change a user's role in a project

    Update a user's role in a project
    PUT <unleash-url>/api/admin/projects/:projectId/users/:userId/roles/:roleId
    Authorization: <API-token>
    content-type: application/json

    {
    "userId": 25,
    "projectId": "myProject",
    "roleId": "3"
    }

    This will change the user's project role to the role specified by :roleId. If the user has not been added to the project, nothing happens.

    URL parameters

    ParameterTypeDescriptionExample value
    userIdintegerThe ID of the user whose role you want to update.1
    projectIdstringThe id of the relevant project."MyCoolProject"
    roleIdintegerThe role ID of the role you wish to assign the user.7

    Responses

    Responses data
    200 OK

    The user's role has been successfully changed. This response has no body.

    400 Bad Request

    You tried to change the role of the only user with the owner role in the project:

    [
    {
    "msg": "A project must have at least one owner."
    }
    ]

    Example query

    The following query would change the role of the user with ID 42 the role with ID 13 in the MyCoolProject project.

    http PUT \
    http://localhost:4242/api/admin/projects/MyCoolProject/users/42/roles/13 \
    Authorization:$KEY

    Remove a user from a project

    Delete a user from a project
    DELETE <unleash-url>/api/admin/projects/:projectId/users/:userId/roles/:roleId
    Authorization: <API-token>
    content-type: application/json

    This removes the specified role from the user in the project. Because users can only have one role in a project, this effectively removes the user from the project. The user must have the role indicated by the :roleId URL parameter for the request to succeed.

    URL parameters

    ParameterTypeDescriptionExample value
    userIdintegerThe ID of the user you want to remove from the project.1
    projectIdstringThe id of the project to remove the user from."MyCoolProject"
    roleIdintegerThe current role the of the user you want to remove from the project.7

    Responses

    Responses data
    200 OK

    The user no longer has the specified role in the project. If the user had this role prior to this API request, they will have been removed from the project. This response has no body.

    400 Bad Request

    You tried to remove the only user with the role owner in the project:

    [
    {
    "msg": "A project must have at least one owner."
    }
    ]

    Example query

    The following query would remove the user with ID 42 and role ID 13 from the MyCoolProject project.

    http DELETE \
    http://localhost:4242/api/admin/projects/MyCoolProject/users/42/roles/13 \
    Authorization:$KEY
    - - + + \ No newline at end of file diff --git a/reference/api/legacy/unleash/admin/features.html b/reference/api/legacy/unleash/admin/features.html index 56f2540acb..0b9b7d679c 100644 --- a/reference/api/legacy/unleash/admin/features.html +++ b/reference/api/legacy/unleash/admin/features.html @@ -20,15 +20,15 @@ - - + +

    /api/admin/features

    Deprecation notice

    Most of this API was removed in Unleash v5 (after being deprecated since Unleash v4.3). You should use the project-based API (/api/admin/projects/:projectId) instead.

    info

    In order to access the admin API endpoints you need to identify yourself. Unless you're using the none authentication method, you'll need to create an admin token and add an Authorization header using the token.

    Fetching Feature Toggles

    Deprecation notice

    This endpoint was removed in Unleash v5. Please use the project-based endpoint to fetch all toggles instead.

    GET: http://unleash.host.com/api/admin/features

    This endpoint is the one all admin ui should use to fetch all available feature toggles from the unleash-server. The response returns all active feature toggles and their current strategy configuration. A feature toggle will have at least one configured strategy. A strategy will have a name and parameters map.

    Example response:

    {
    "version": 2,
    "features": [
    {
    "name": "Feature.A",
    "description": "lorem ipsum",
    "type": "release",
    "enabled": false,
    "stale": false,
    "strategies": [
    {
    "name": "default",
    "parameters": {}
    }
    ],
    "variants": [
    {
    "name": "variant1",
    "weight": 50
    },
    {
    "name": "variant2",
    "weight": 50
    }
    ],
    "tags": [
    {
    "id": 1,
    "type": "simple",
    "value": "TeamRed"
    }
    ]
    },
    {
    "name": "Feature.B",
    "description": "lorem ipsum",
    "enabled": true,
    "stale": false,
    "strategies": [
    {
    "name": "ActiveForUserWithId",
    "parameters": {
    "userIdList": "123,221,998"
    }
    },
    {
    "name": "GradualRolloutRandom",
    "parameters": {
    "percentage": "10"
    }
    }
    ],
    "variants": [],
    "tags": []
    }
    ]
    }

    Filter feature toggles

    Supports three params for now

    • tag - filters for features tagged with tag
    • project - filters for features belonging to project
    • namePrefix - filters for features beginning with prefix

    For tag and project performs OR filtering if multiple arguments

    To filter for any feature tagged with a simple tag with value taga or a simple tag with value tagb use

    GET https://unleash.host.com/api/admin/features?tag[]=simple:taga&tag[]simple:tagb

    To filter for any feature belonging to project myproject use

    GET https://unleash.host.com/api/admin/features?project=myproject

    Response format is the same as api/admin/features

    Fetch specific feature toggle

    Removal notice

    This endpoint was removed in Unleash v5 (deprecated since v4). Please use the project-based endpoint to fetch specific toggles instead.

    GET: http://unleash.host.com/api/admin/features/:featureName

    Used to fetch details about a specific featureToggle. This is mostly provded to make it easy to debug the API and should not be used by the client implementations.

    {
    "name": "Feature.A",
    "description": "lorem ipsum..",
    "type": "release",
    "enabled": false,
    "stale": false,
    "strategies": [
    {
    "name": "default",
    "parameters": {}
    }
    ],
    "variants": [],
    "tags": []
    }

    Create a new Feature Toggle

    Removal notice

    This endpoint was removed in Unleash v5 (deprecated since v4). Please use the project-based endpoint to create feature toggles instead.

    POST: http://unleash.host.com/api/admin/features/

    Body:

    {
    "name": "Feature.A",
    "description": "lorem ipsum..",
    "type": "release",
    "enabled": false,
    "stale": false,
    "strategies": [
    {
    "name": "default",
    "parameters": {}
    }
    ]
    }

    Used by the admin-dashboard to create a new feature toggles.

    Notes:

    • name must be globally unique, otherwise you will get a 403-response.
    • type is optional. If not defined it defaults to release

    Returns 200-response if the feature toggle was created successfully.

    Update a Feature Toggle

    Removal notice

    This endpoint was removed in Unleash v5. Please use the project-based endpoint to update a feature toggle instead.

    PUT: http://unleash.host.com/api/admin/features/:toggleName

    Body:

    {
    "name": "Feature.A",
    "description": "lorem ipsum..",
    "type": "release",
    "enabled": false,
    "stale": false,
    "strategies": [
    {
    "name": "default",
    "parameters": {}
    }
    ],
    "variants": []
    }

    Used by the admin dashboard to update a feature toggles. The name has to match an existing features toggle.

    Returns 200-response if the feature toggle was updated successfully.

    Tag a Feature Toggle

    POST https://unleash.host.com/api/admin/features/:featureName/tags

    Used to tag a feature

    If the tuple (type, value) does not already exist, it will be added to the list of tags. Then Unleash will add a relation between the feature name and the tag.

    Body:

    {
    "type": "simple",
    "value": "Team-Green"
    }

    Success

    - Returns _201-CREATED_ if the feature was tagged successfully
    - Creates the tag if needed, then connects the tag to the existing feature

    Failures

    - Returns _404-NOT-FOUND_ if the `type` was not found

    Remove a tag from a Feature Toggle

    DELETE https://unleash.host.com/api/admin/features/:featureName/tags/:type/:value

    Removes the specified tag from the (type, value) tuple from the Feature Toggle's list of tags.

    Success

    - Returns _200-OK_

    Failures

    - Returns 404 if the tag does not exist
    - Returns 500 if the database could not be reached

    Archive a Feature Toggle

    Removal notice

    This endpoint was removed in v5. Please use the project-based endpoint to archive toggles instead.

    DELETE: http://unleash.host.com/api/admin/features/:toggleName

    Used to archive a feature toggle. A feature toggle can never be totally be deleted, but can be archived. This is a design decision to make sure that a old feature toggle does not suddenly reappear because someone else is re-using the same name.

    Enable a Feature Toggle

    Removal notice

    This endpoint was removed in v5. Please use the project-based endpoint to enable feature toggles instead.

    POST: http://unleash.host.com/api/admin/features/:featureName/toggle/on

    Used to enable a feature toggle. The :featureName must match an existing Feature Toggle. Returns 200-response if the feature toggle was enabled successfully.

    Body

    None

    Example response:

    {
    "name": "Feature.A",
    "description": "lorem ipsum..",
    "type": "release",
    "enabled": true,
    "strategies": [
    {
    "name": "default",
    "parameters": {}
    }
    ],
    "variants": [],
    "tags": []
    }

    Disable a Feature Toggle

    Removal notice

    This endpoint was removed in v5. Please use the project-based endpoint to disable feature toggles instead.

    POST: http://unleash.host.com/api/admin/features/:featureName/toggle/off

    Used to disable a feature toggle. The :featureName must match an existing Feature Toggle. Returns 200-response if the feature toggle was disabled successfully.

    Body

    None

    Example response:

    {
    "name": "Feature.A",
    "description": "lorem ipsum..",
    "type": "release",
    "enabled": false,
    "stale": false,
    "strategies": [
    {
    "name": "default",
    "parameters": {}
    }
    ],
    "variants": [],
    "tags": []
    }

    Mark a Feature Toggle as "stale"

    Removal notice

    This endpoint was removed in v5. Please use the project-based endpoint to patch a feature toggle and mark it as stale instead.

    POST: http://unleash.host.com/api/admin/features/:featureName/stale/on

    Used to mark a feature toggle as stale (deprecated). The :featureName must match an existing Feature Toggle. Returns 200-response if the feature toggle was enabled successfully.

    Body

    None

    Example response:

    {
    "name": "Feature.A",
    "description": "lorem ipsum..",
    "type": "release",
    "enabled": true,
    "stale": true,
    "strategies": [
    {
    "name": "default",
    "parameters": {}
    }
    ],
    "variants": [],
    "tags": []
    }

    Mark a Feature Toggle as "active"

    Removal notice

    This endpoint was removed in v5. Please use the project-based endpoint to patch a feature toggle and mark it as not stale instead.

    POST: http://unleash.host.com/api/admin/features/:featureName/stale/off

    Used to mark a feature toggle active (remove stale marking). The :featureName must match an existing Feature Toggle. Returns 200-response if the feature toggle was disabled successfully.

    Body

    None

    Example response:

    {
    "name": "Feature.A",
    "description": "lorem ipsum..",
    "type": "release",
    "enabled": false,
    "stale": false,
    "strategies": [
    {
    "name": "default",
    "parameters": {}
    }
    ],
    "variants": [],
    "tags": []
    }
    - - + + \ No newline at end of file diff --git a/reference/api/legacy/unleash/admin/metrics.html b/reference/api/legacy/unleash/admin/metrics.html index eb97b91f7d..ff328deb2f 100644 --- a/reference/api/legacy/unleash/admin/metrics.html +++ b/reference/api/legacy/unleash/admin/metrics.html @@ -20,15 +20,15 @@ - - + +

    /api/admin/metrics

    In order to access the admin API endpoints you need to identify yourself. Unless you're using the none authentication method, you'll need to create an ADMIN token and add an Authorization header using the token.

    This document describes the metrics endpoint for admin ui

    Seen-toggles

    Deprecation notice

    This endpoint has been deprecated

    GET http://unleash.host.com/api/admin/metrics/seen-toggles

    This enpoints returns a list of applications and what toogles unleash has seen for each application. It will only guarantee toggles reported by client applications within the last hour, but will in most cases remember seen toggles for applications longer.

    Example response:

    [
    {
    "appName": "demo-app",
    "seenToggles": ["add-feature-2", "toggle-2", "toggle-3"],
    "metricsCount": 127
    },
    {
    "appName": "demo-app-2",
    "seenToggles": ["add-feature-2", "toggle-2", "toggle-3"],
    "metricsCount": 21
    }
    ]

    Fields:

    • appName - Name of the application seen by unleash-server
    • seenToggles - array of toggles names seen by unleash-server for this application
    • metricsCount - number of metrics counted across all toggles for this application.

    Feature-Toggles metrics

    GET http://unleash.host.com/api/admin/metrics/feature-toggles

    This endpoint gives last minute and last hour metrics for all active toggles. This is based on metrics reported by client applications. Yes is the number of times a given feature toggle was evaluated to enabled in a client application, and no is the number it was evaluated to false.

    Example response:

    {
    "lastHour": {
    "add-feature-2": {
    "yes": 0,
    "no": 527
    },
    "toggle-2": {
    "yes": 265,
    "no": 85
    },
    "toggle-3": {
    "yes": 257,
    "no": 93
    }
    },
    "lastMinute": {
    "add-feature-2": {
    "yes": 0,
    "no": 527
    },
    "toggle-2": {
    "yes": 265,
    "no": 85
    },
    "toggle-3": {
    "yes": 257,
    "no": 93
    }
    }
    }

    Fields:

    • lastHour - Hour projection collected metrics for all feature toggles.
    • lastMinute - Minute projection collected metrics for all feature toggles.

    Applications

    GET http://unleash.host.com/api/admin/metrics/applications

    This endpoint returns a list of known applications (seen in the last day) and a link to follow for more details.

    {
    "applications": [
    {
    "appName": "another",
    "strategies": ["default", "other", "brother"],
    "createdAt": "2016-12-09T14:56:36.730Z",
    "links": {
    "appDetails": "/api/admin/applications/another"
    }
    },
    {
    "appName": "bow",
    "strategies": ["default", "other", "brother"],
    "createdAt": "2016-12-09T14:56:36.730Z",
    "links": {
    "appDetails": "/api/admin/applications/bow"
    }
    }
    ]
    }

    Query Params

    You can also specify the query param: strategyName, which will return all applications implementing the given strategy.

    GET http://unleash.host.com/api/admin/metrics/applications?strategyName=someStrategyName

    Application Details

    GET http://unleash.host.com/api/admin/metrics/applications/:appName

    This endpoint gives insight into details about a client application, such as instances, strategies implemented and seen toggles.

    {
    "appName": "demo-app",
    "instances": [
    {
    "instanceId": "generated-732038-17080",
    "clientIp": "::ffff:127.0.0.1",
    "lastSeen": "2016-11-30T17:32:04.265Z",
    "createdAt": "2016-11-30T17:31:08.914Z"
    },
    {
    "instanceId": "generated-639919-11185",
    "clientIp": "::ffff:127.0.0.1",
    "lastSeen": "2016-11-30T16:04:15.991Z",
    "createdAt": "2016-11-30T10:49:11.223Z"
    }
    ],
    "strategies": [
    {
    "appName": "demo-app",
    "strategies": ["default", "extra"]
    }
    ],
    "seenToggles": ["add-feature-2", "toggle-2", "toggle-3"]
    }

    Seen applications

    Deprecation notice

    This endpoint has been deprecated

    GET http://unleash.host.com/api/admin/metrics/seen-apps

    This endpoint gives insight into details about application seen per feature toggle.

    {
    "my-toggle": [
    {
    "appName": "my-app",
    "createdAt": "2016-12-28T10:39:24.966Z",
    "updatedAt": "2017-01-06T15:32:41.932Z",
    "description": "our main app",
    "strategies": [
    "gradualRolloutRandom",
    "abTest",
    "default",
    "betaUser",
    "userWithId",
    "byHostName",
    "gradualRolloutWithSessionId",
    "gradualRollout",
    "byRemoteAddr"
    ],
    "url": "http://example.com",
    "color": null,
    "icon": "terrain"
    },
    {
    "appName": "my-other-app",
    "createdAt": "2016-12-28T10:39:24.966Z",
    "updatedAt": "2017-01-06T15:32:41.932Z",
    "description": "our other app",
    "strategies": ["default"],
    "url": "http://example.com",
    "color": null,
    "icon": "desktop"
    }
    ]
    }
    - - + + \ No newline at end of file diff --git a/reference/api/legacy/unleash/admin/projects.html b/reference/api/legacy/unleash/admin/projects.html index 704c1c5a3f..228a537de3 100644 --- a/reference/api/legacy/unleash/admin/projects.html +++ b/reference/api/legacy/unleash/admin/projects.html @@ -20,15 +20,15 @@ - - + +

    /api/admin/projects

    The context feature is only available as part of Unleash Enterprise. In order to access the API programmatically you need to make sure you obtain an API token with admin permissions.

    List projects in Unleash

    GET https://unleash.host.com/api/admin/projects

    Returns a list of projects in Unleash.

    Example response:

    {
    "version": 1,
    "projects": [
    {
    "id": "default",
    "name": "Default",
    "description": "Default project",
    "createdAt": "2020-12-03T09:47:20.170Z"
    },
    {
    "id": "MyNewProject",
    "name": "MyNewProject",
    "description": "A test project",
    "createdAt": "2020-12-03T09:47:20.170Z"
    },
    {
    "id": "test",
    "name": "Test Project",
    "description": "Collection of test toggles",
    "createdAt": "2020-12-03T09:47:20.170Z"
    }
    ]
    }

    Create a new project

    POST https://unleash.host.com/api/admin/projects

    Creates a new project.

    Body

    {
    "id": "someId",
    "name": "Test Project",
    "description": "Some description"
    }

    Update a projects field

    PUT https://unleash.host.com/api/projects/:id

    Updates a project with id=id.

    Body

    {
    "id": "someId",
    "name": "Test Project",
    "description": "Some description"
    }

    Delete a projects field

    DELETE https://unleash.host.com/api/admin/projects/:id

    Deletes the project with id=id.

    - - + + \ No newline at end of file diff --git a/reference/api/legacy/unleash/admin/segments.html b/reference/api/legacy/unleash/admin/segments.html index 74f4ba6159..edfc22fefd 100644 --- a/reference/api/legacy/unleash/admin/segments.html +++ b/reference/api/legacy/unleash/admin/segments.html @@ -20,15 +20,15 @@ - - + +

    /api/admin/segments

    Availability

    Segments are available to Unleash Pro and Unleash Enterprise users since Unleash 4.13.

    note

    To use the admin API, you'll need to create and use an admin API token.

    The segments API lets you create, read, update, and delete segments.

    Get all segments

    Retrieve all segments that exist in this Unleash instance. Returns a list of segment objects.

    Retrieve all existing segments.
    GET <unleash-url>/api/admin/segments
    Authorization: <API-token>
    content-type: application/json
    Example responses

    200 OK

    [
    {
    "id": 1,
    "name": "my-segment",
    "description": "a segment description",
    "constraints": [],
    "createdBy": "user@example.com",
    "createdAt": "2022-04-01T14:02:25.491Z"
    }
    ]

    Create segment

    Create a new segment with the specified configuration.

    Create a new segment.
    POST <unleash-url>/api/admin/segments
    Authorization: <API-token>
    content-type: application/json

    {
    "name": "my-segment",
    "description": "a segment description",
    "constraints": []
    }
    Example responses

    201 Created

    The segment was successfully created. This response has no body.

    400 Bad Request

    A segment with the provided name already exists.

    Payload structure

    Use a JSON object with the following properties to create a new segment.

    PropertyTypeRequiredDescriptionExample value
    namestringYesThe name of the segment."mobile-users"
    descriptionstringNoA description of the segment."This segment is for users on mobile devices."
    constraintslist of constraint objectsYesThe constraints in this segment.[]

    Get segment by ID

    Retrieves the segment with the specified ID.

    Retrieve the segment with the provided ID.
    GET <unleash-url>/api/admin/segments/<segment-id>
    Authorization: <API-token>
    content-type: application/json
    Example responses

    200 OK

    {
    "id": 1,
    "name": "my-segment",
    "description": "a segment description",
    "constraints": [],
    "createdBy": "user@example.com",
    "createdAt": "2022-04-01T14:02:25.491Z"
    }

    404 Not Found

    No segment with the provided ID exists.

    Update an existing segment

    Replace the data of the specified segment with the provided payload.

    Update a segment with new data.
    PUT <unleash-url>/api/admin/segments/<segment-id>
    Authorization: <API-token>
    content-type: application/json

    {
    "name": "my-segment",
    "description": "this is a newly provided description.",
    "constraints": []
    }
    Example responses

    204 No Content

    The update was successful. This response has no body.

    404 Not Found

    No segment with the provided ID exists.

    Delete a segment

    Delete the request with the specified ID.

    Delete a segment.
    DELETE <unleash-url>/api/admin/segments/<segment-id>
    Authorization: <API-token>
    content-type: application/json
    Example responses

    204 No Content

    The segment was deleted successfully.

    404 Not Found

    No segment with the provided ID exists.

    409 Conflict

    The segment is being used by at least one strategy and can not be deleted. To delete the segment, first remove it from any strategies that use it.

    List strategies that use a specific segment

    Retrieve all strategies that use the specified segment. Returns a list of activation strategy objects.

    Retrieve all activation strategies that use the specified segment.
    GET <unleash-url>/api/admin/segments/<segment-id>/strategies
    Authorization: <API-token>
    content-type: application/json
    Example responses

    200 OK

    [
    {
    "id": "strategy-id",
    "featureName": "my-feature",
    "projectId": "my-project",
    "environment": "development",
    "strategyName": "my strategy",
    "parameters": {},
    "constraints": [],
    "createdAt": "2022-04-01T14:02:25.491Z"
    }
    ]

    404 Not Found

    No segment with the provided id exists.

    List segments applied to a specific strategy

    Retrieve all segments that are applied to the specified strategy. Returns a list of segment objects.

    Retrieve all segments that are used by the specified strategy.
    GET <unleash-url>/api/admin/segments/strategies/<strategy-id>
    Authorization: <API-token>
    content-type: application/json
    Example responses

    200 OK

    [
    {
    "id": 1,
    "name": "my-segment",
    "description": "a segment description",
    "constraints": [],
    "createdBy": "user@example.com",
    "createdAt": "2022-04-01T14:02:25.491Z"
    }
    ]

    404 Not Found

    No strategy with the provided id exists.

    Replace activation strategy segments

    Replace the segments applied to the specified activation strategy with the provided segment list.

    Replace the segments to the specified strategy.
    POST <unleash-url>/api/admin/segments/strategies
    Authorization: <API-token>
    content-type: application/json

    {
    "projectId": "my-project",
    "strategyId": "my-strategy",
    "environmentId": "development",
    "segmentIds": [
    61,
    62,
    63,
    64
    ]
    }

    Remove all segments from an activation strategy

    To remove all segments from an activation strategy, use this endpoint and provide an empty list of segmentIds. For instance, the following payload would remove all segments from the strategy "my-strategy".

    {
    "projectId": "my-project",
    "strategyId": "my-strategy",
    "environmentId": "development",
    "segmentIds": []
    }
    Example responses

    201 Created

    The strategy's list of segments was successfully updated.

    403 Forbidden

    You do not have access to edit this activation strategy.

    404 Not Found

    No strategy with the provided ID exists.

    Payload structure

    Use a JSON object with the following properties to update the list of applied segments.

    PropertyTypeRequiredDescriptionExample value
    projectIdstringYesThe ID of the feature toggle's project."my-project"
    strategyIdstringYesThe ID of the strategy."my-strategy"
    environmentIdstringYesThe ID of the environment."development"
    segmentIdslist of segment IDs (numbers)YesThe list of segment IDs to apply to the strategy.[]

    API types

    This section describes the data objects returned by the endpoints in the segments API. For information on a specific endpoint, refer to its specific description above.

    Segment

    Example

    {
    "id": 12054,
    "name": "segment name",
    "description": "segment description",
    "constraints": [],
    "createdBy": "you@example.com",
    "createdAt": "2022-05-23T15:45:22.000Z"
    }

    Description

    PropertyTypeRequiredDescriptionExample value
    idnumberYesThe segment's ID.546
    namestringYesThe segment's name"my-segment"
    descriptionstringNoAn optional description of the segment."segment description"
    constraintslist of constraint objectsYesThe list of constraint objects in the segment.[]
    createdBystringNoAn identifier for who created the segment."you@example.com"
    createdAttimestamp stringYesThe time when the segment was created. Format: YYYY-MM-DDThh:mm:ss.sTZD"2022-04-23T13:56:24.45+01:00"

    Constraint

    Example

    {
    "contextName": "appName",
    "operator": "STR_CONTAINS",
    "values": [],
    "inverted": false,
    "caseInsensitive": false
    }

    Description

    values and value

    Some constraint operators only support single values. If a constraint uses one of these operators, the payload will contain a value property with the correct value. However, for backwards compatibility reasons, the payload will also contain a values property. If the operator accepts multiple values, the value property will not be present. Visit the strategy constraints documentation for more information on what operators support what number of values.

    PropertyTypeRequiredDescriptionExample value
    contextNamestringYesThe name of the context field targeted by the constraint."myContextField"
    operatorstring, the name of one of the constraint operatorsYesThe operator to apply to the context field."DATE_BEFORE"
    valuesa list of stringsYesThe list of values to apply the constraint operator to.["value a", "value b"]
    valuestringNoThe value to apply the constraint operator to."15"
    invertedbooleanNoWhether the result of the constraint will be negated or not.false
    caseInsensitiveboolean stringNoWhether the constraint operator is case sensitive or not. Only applies to some string-based operators.false

    Activation strategy

    Example

    {
    "id": "64fbe72b-d107-4b26-b6b8-4fead08d286c",
    "environment": "development",
    "featureName": "my-feature",
    "projectId": "my-project",
    "strategyName": "flexibleRollout"
    }

    Description

    PropertyTypeRequiredDescriptionExample value
    idGUID stringNoThe ID of the strategy."64fbe72b-d107-4b26-b6b8-4fead08d286c"
    environmentstringYesThe name of the strategy's environment."development"
    featureNamestringYesThe name of the feature the strategy is applied to."my-feature"
    projectIdstringYesThe name of the current project."my-project"
    strategyNamestringYesThe name of the strategy."flexibleRollout"
    - - + + \ No newline at end of file diff --git a/reference/api/legacy/unleash/admin/state.html b/reference/api/legacy/unleash/admin/state.html index 7e1fc7775f..34e76d8603 100644 --- a/reference/api/legacy/unleash/admin/state.html +++ b/reference/api/legacy/unleash/admin/state.html @@ -20,8 +20,8 @@ - - + + @@ -31,7 +31,7 @@ You can customize the export with query parameters:

    ParameterDefaultDescription
    formatjsonExport format, either json or yaml
    downloadfalseIf the exported data should be downloaded as a file
    featureTogglestrueInclude feature-toggles in the exported data
    strategiestrueInclude strategies in the exported data

    Example response:

    GET /api/admin/state/export?format=yaml&featureToggles=1&strategies=1

    version: 1
    features:
    - name: Feature.A
    description: lorem ipsum
    enabled: false
    strategies:
    - name: default
    parameters: {}
    variants:
    - name: variant1
    weight: 50
    - name: variant2
    weight: 50
    - name: Feature.B
    description: lorem ipsum
    enabled: true
    strategies:
    - name: ActiveForUserWithId
    parameters:
    userIdList: '123,221,998'
    - name: GradualRolloutRandom
    parameters:
    percentage: '10'
    variants: []
    strategies:
    - name: country
    description: Enable feature for certain countries
    parameters:
    - name: countries
    type: list
    description: List of countries
    required: true

    Import Feature Toggles & Strategies

    POST: http://unleash.host.com/api/admin/state/import

    You can import feature-toggles and strategies by POSTing to the /api/admin/state/import endpoint.\ You can either send the data as JSON in the POST-body or send a file parameter with multipart/form-data (YAML files are also accepted here).

    Query Paramters

    • drop - Use this parameter if you want the database to be cleaned before import (all strategies and features will be removed).
    • keep - Use this query parameter if you want to keep all exiting feature toggle (and strategy) configurations as is (no override), and only insert missing feature toggles from the data provided.

    You should be careful using the drop parameter in production environments.

    Success: 202 Accepted\ Error: 400 Bad Request

    Example body:

    {
    "version": 1,
    "features": [
    {
    "name": "Feature.A",
    "description": "lorem ipsum",
    "enabled": false,
    "strategies": [
    {
    "name": "default",
    "parameters": {}
    }
    ],
    "variants": [
    {
    "name": "variant1",
    "weight": 50
    },
    {
    "name": "variant2",
    "weight": 50
    }
    ]
    },
    {
    "name": "Feature.B",
    "description": "lorem ipsum",
    "enabled": true,
    "strategies": [
    {
    "name": "ActiveForUserWithId",
    "parameters": {
    "userIdList": "123,221,998"
    }
    },
    {
    "name": "GradualRolloutRandom",
    "parameters": {
    "percentage": "10"
    }
    }
    ],
    "variants": []
    }
    ],
    "strategies": [
    {
    "name": "country",
    "description": "Enable feature for certain countries",
    "parameters": [
    {
    "name": "countries",
    "type": "list",
    "description": "List of countries",
    "required": true
    }
    ]
    }
    ]
    }
    - - + + \ No newline at end of file diff --git a/reference/api/legacy/unleash/admin/strategies.html b/reference/api/legacy/unleash/admin/strategies.html index 9863da5272..9ea9ea9513 100644 --- a/reference/api/legacy/unleash/admin/strategies.html +++ b/reference/api/legacy/unleash/admin/strategies.html @@ -20,15 +20,15 @@ - - + +

    /api/admin/strategies

    In order to access the admin API endpoints you need to identify yourself. Unless you're using the none authentication method, you'll need to create an ADMIN token and add an Authorization header using the token.

    Fetch Strategies

    GET: http://unleash.host.com/api/admin/strategies

    Used to fetch all defined strategies and their defined parameters.

    Response

    {
    "version": 1,
    "strategies": [
    {
    "name": "default",
    "description": "Default on/off strategy.",
    "parameters": []
    },
    {
    "name": "userWithId",
    "description": "Active for userId specified in the comma seperated 'userIds' parameter.",
    "parameters": [
    {
    "name": "userIds",
    "type": "list",
    "description": "List of unique userIds the feature should be active for.",
    "required": true
    }
    ]
    },
    {
    "name": "gradualRollout",
    "description": "Gradual rollout to logged in users",
    "parameters": [
    {
    "name": "percentage",
    "type": "percentage",
    "description": "How many percent should the new feature be active for.",
    "required": false
    },
    {
    "name": "group",
    "type": "string",
    "description": "Group key to use when hasing the userId. Makes sure that the same user get different value for different groups",
    "required": false
    }
    ]
    }
    ]
    }

    Create strategy

    POST: http://unleash.host.com/api/admin/strategies

    Body

    {
    "name": "gradualRollout",
    "description": "Gradual rollout to logged in users",
    "parameters": [
    {
    "name": "percentage",
    "type": "percentage",
    "description": "How many percent should the new feature be active for.",
    "required": false
    },
    {
    "name": "group",
    "type": "string",
    "description": "Group key to use when hasing the userId. Makes sure that the same user get different value for different groups",
    "required": false
    }
    ]
    },

    Used to create a new Strategy. Name is required and must be unique. It is also required to have a parameters array, but it can be empty.

    Update strategy

    PUT: http://unleash.host.com/api/admin/strategies/:name

    Body

    {
    "name": "gradualRollout",
    "description": "Gradual rollout to logged in users with updated desc",
    "parameters": [
    {
    "name": "percentage",
    "type": "percentage",
    "description": "How many percent should the new feature be active for.",
    "required": false
    },
    {
    "name": "group",
    "type": "string",
    "description": "Group key to use when hasing the userId. Makes sure that the same user get different value for different groups",
    "required": false
    }
    ]
    },

    Used to update a Strategy definition. Name can't be changed. PS! I can be dangerous to change an implemented strategy as the implementation also might need to be changed

    Deprecate strategy

    POST: https://unleash.host.com/api/admin/strategies/:name/deprecate

    Used to deprecate a strategy definition. This will set the deprecated flag to true. If the strategy is already deprecated, this will be a noop.

    Errors

    404 NOT FOUND - if :name does not exist

    Reactivate strategy

    POST: https://unleash.host.com/api/admin/strategies/:name/reactivate

    Used to reactivate a deprecated strategy definition. This will set the deprecated flag back to false. If the strategy is not deprecated this is a noop and will still return 200.

    Errors

    404 NOT FOUND - if :name does not exist

    - - + + \ No newline at end of file diff --git a/reference/api/legacy/unleash/admin/tags.html b/reference/api/legacy/unleash/admin/tags.html index c7fba87551..4a5b432802 100644 --- a/reference/api/legacy/unleash/admin/tags.html +++ b/reference/api/legacy/unleash/admin/tags.html @@ -20,15 +20,15 @@ - - + +

    /api/admin/tags

    In order to access the admin API endpoints you need to identify yourself. Unless you're using the none authentication method, you'll need to create an ADMIN token and add an Authorization header using the token.

    Create a new tag

    POST https://unleash.host.com/api/admin/tags

    Creates a new tag without connecting it to any other object, can be helpful to build an autocomplete list.

    Body

    {
    "value": "MyTag",
    "type": "simple"
    }

    Notes

    • type must exist in tag-types

    List tags

    GET https://unleash.host.com/api/admin/tags

    This endpoint is the one all admin UIs should use to fetch all available tags from the unleash_server. The response returns all tags.

    Example response:

    {
    "version": 1,
    "tags": [
    {
    "value": "Team-Red",
    "type": "simple"
    },
    {
    "value": "Team-Green",
    "type": "simple"
    },
    {
    "value": "DecemberExperiment",
    "type": "simple"
    },
    {
    "value": "#team-alert-channel",
    "type": "slack"
    }
    ]
    }

    List tags by type

    GET: https://unleash.host.com/api/admin/tags/:type

    Lists all tags of :type. If none exist, returns the empty list

    Example response to query for https://unleash.host.com/api/admin/tags/simple

    {
    "version": 1,
    "tags": [
    {
    "value": "Team-Red",
    "type": "simple"
    },
    {
    "value": "Team-Green",
    "type": "simple"
    },
    {
    "value": "DecemberExperiment",
    "type": "simple"
    }
    ]
    }

    Get a single tag

    GET https://unleash.host.com/api/admin/tags/:type/:value

    Gets the tag defined by the type, value tuple

    Delete a tag

    DELETE https://unleash.host.com/api/admin/tags/:type/:value

    Deletes the tag defined by the type, value tuple; all features tagged with this tag will lose the tag.

    Fetching Tag types

    GET: https://unleash.host.com/api/admin/tag-types

    Used to fetch all types the server knows about. This endpoint is the one all admin UI should use to fetch all available tag types from the unleash-server. The response returns all tag types. Any server will have at least one configured tag type (the simple type). A tag type will be a map of type, description, icon

    Example response:

    {
    "version": 1,
    "tagTypes": [
    {
    "name": "simple",
    "description": "Arbitrary tags. Used to simplify filtering of features",
    "icon": "#"
    }
    ]
    }

    Get a single tag type

    GET: https://unleash.host.com/api/admin/tag-types/simple

    Used to fetch details about a specific tag-type. This is mostly provided to make it easy to debug the API and should not be used by the client implementations.

    Example response:

    {
    "version": 1,
    "tagType": {
    "name": "simple",
    "description": "Some description",
    "icon": "Some icon",
    "createdAt": "2021-01-07T10:00:00Z"
    }
    }

    Create a new tag type

    POST: https://unleash.host.com/api/admin/tag-types

    Used to register a new tag type. This endpoint should be used to inform the server about a new type of tags.

    Body:

    {
    "name": "tagtype",
    "description": "Purpose of tag type",
    "icon": "Either an URL to icon or a simple prefix string for tag"
    }

    Notes:

    • if name is not unique, will return 409 CONFLICT, if you'd like to update an existing tag through admin-api look at Update tag type.

    Returns 201-CREATED if the tag type was created successfully

    Update tag type

    PUT: https://unleash.host.com/api/admin/tag-types/:typeName

    Body:

    {
    "description": "New description",
    "icon": "New icon"
    }

    Deleting a tag type

    DELETE: https://unleash.host.com/api/admin/tag-types/:typeName

    Returns 200 if the type was not in use and the type was deleted. If the type was in use, will return a 409 CONFLICT

    - - + + \ No newline at end of file diff --git a/reference/api/legacy/unleash/admin/user-admin.html b/reference/api/legacy/unleash/admin/user-admin.html index b5fdd9d79f..9cdc8622a6 100644 --- a/reference/api/legacy/unleash/admin/user-admin.html +++ b/reference/api/legacy/unleash/admin/user-admin.html @@ -20,15 +20,15 @@ - - + +

    /api/admin/user-admin

    In order to access the admin API endpoints you need to identify yourself. Unless you're using the none authentication method, you'll need to create an ADMIN token and add an Authorization header using the token.

    List all users

    GET https://unleash.host.com/api/admin/user-admin

    Will return all users and all available root roles for the Unleash instance.

    Body

    {
    "rootRoles": [
    {
    "description": "Users with the root admin role have superuser access to Unleash and can perform any operation within the unleash platform.",
    "id": 1,
    "name": "Admin",
    "project": null,
    "type": "root"
    },
    {
    "description": "Users with this role have access most features in Unleash, but can not manage users and roles in the root scope. If a user with a regular root role creates a project, they will become a project admin and receive superuser rights within the context of that project.",
    "id": 2,
    "name": "Editor",
    "project": null,
    "type": "root"
    },
    {
    "description": "Users with this role can only read root resources in Unleash. They may be added as collaborator to specific projects.",
    "id": 3,
    "name": "Viewer",
    "project": null,
    "type": "root"
    }
    ],
    "users": [
    {
    "createdAt": "2021-05-14T08:56:34.859Z",
    "email": "random-user@getunleash.ai",
    "id": 3,
    "imageUrl": "https://gravatar.com/avatar/3066e45cf3a09d9a4b51e08a3ac20749?size=42&default=retro",
    "inviteLink": "",
    "isAPI": false,
    "loginAttempts": 0,
    "rootRole": 1,
    "seenAt": null
    },
    {
    "createdAt": "2021-05-14T08:58:07.891Z",
    "email": "random-user2@getunleash.ai",
    "id": 4,
    "imageUrl": "https://gravatar.com/avatar/90047524992cd6ae8f66e249a7630d80?size=42&default=retro",
    "inviteLink": "",
    "isAPI": false,
    "loginAttempts": 0,
    "rootRole": 1,
    "seenAt": null
    }
    ]
    }

    Get a single user

    GET https://unleash.host.com/api/admin/user-admin/:id

    Will return a single user by id.

    Body

    {
    "createdAt": "2021-05-14T08:58:07.891Z",
    "email": "random-user2@getunleash.ai",
    "id": 4,
    "imageUrl": "https://gravatar.com/avatar/90047524992cd6ae8f66e249a7630d80?size=42&default=retro",
    "inviteLink": "",
    "isAPI": false,
    "loginAttempts": 0,
    "rootRole": 1,
    "seenAt": null
    }

    Search for users

    You can also search for users via the search API. It will preform a simple search based on name and email matching the given query. Requires minimum 2 characters.

    GET http://localhost:4242/api/admin/user-admin/search?q=iv

    Body

    [
    {
    "email": "iva2@some-mail.com",
    "id": 19,
    "imageUrl": "https://gravatar.com/avatar/6c795493735ff1864f17d47ec52cf0ec?size=42&default=retro"
    },
    {
    "email": "ivar@another.com",
    "id": 20,
    "imageUrl": "https://gravatar.com/avatar/f4b3e16a54bfbe824eb814479053bf88?size=42&default=retro"
    }
    ]

    Add a new user

    POST https://unleash.host.com/api/admin/user-admin

    Creates a new user with the given root role.

    Payload properties

    Requirements

    The payload must contain at least one of the name and email properties, though which one is up to you. For the user to be able to log in to the system, the user must have an email.

    Property nameRequiredDescriptionExample value(s)
    emailNoThe user's email address. Must be provided if username is not provided."user@getunleash.io"
    usernameNoThe user's username. Must be provided if email is not provided."Baz the Beholder"
    rootRoleYesThe role to assign to the user. Can be either the role's ID or its unique name.2, "Editor"
    sendEmailNoWhether to send a welcome email with a login link to the user or not. Defaults to true.false
    nameNoThe user's name (not the user's username)."Sam Seawright"

    Body

    {
    "email": "some-email@getunleash.io",
    "username": "Baz the Beholder",
    "rootRole": "Editor",
    "sendEmail": true
    }

    Return values:

    201: Created

    {
    "createdAt": "2021-05-18T10:28:23.067Z",
    "email": "some-email@getunleash.io",
    "emailSent": true,
    "id": 1337,
    "imageUrl": "https://gravatar.com/avatar/222f2ab70c039dda12e3d11acdcebd02?size=42&default=retro",
    "inviteLink": "http://localhost:4242/new-user?token=123",
    "isAPI": false,
    "loginAttempts": 0,
    "name": "Some Name",
    "rootRole": 2,
    "seenAt": null
    }

    400: Bad request

    [
    {
    "msg": "User already exists"
    }
    ]

    400: Bad request

    [
    {
    "msg": "You must specify username or email"
    }
    ]

    Update a user

    PUT https://unleash.host.com/api/admin/user-admin/:userId

    Updates user with new fields

    Body

    {
    "email": "some-email@getunleash.io",
    "name": "Some Name",
    "rootRole": 2
    }

    Notes

    • userId is required as a url path parameter.
    • All fields are optional. Only provided fields are updated.
    • Note that earlier versions of Unleash required either name or email to be set.

    Delete a user

    DELETE https://unleash.host.com/api/admin/user-admin/:userId

    Deletes the user with the given userId.

    Possible return values:

    • 200: OK - user was deleted
    • 404: NOT FOUND - No user with the provided userId was found

    Change password for a user

    POST https://unleash.host.com/api/admin/user-admin/:userId/change-password

    Body

    {
    "password": "k!5As3HquUrQ"
    }

    Return values:

    • 200 OK: Password was changed.
    • 400 Bad Request: Password was not changed. Unleash requires a strong password.
      • This means
        • minimum 10 characters long
        • contains at least one uppercase letter
        • contains at least one number
        • contains at least one special character (symbol)
    • Please see in the response body on how to improve the password.

    Validate password for a user

    You can use this endpoint to validate the strength of a given password. Unleash requires a strong password.

    • This means
      • minimum 10 characters long
      • contains at least one uppercase letter
      • contains at least one number
      • contains at least one special character (symbol)

    http POST http://localhost:4242/api/admin/user-admin/validate-password

    Body

    {
    "password": "some-simple"
    }
    • 200 OK: Password is strong enough for Unleash.
    • 400 Bad Request: Unleash requires a stronger password. Please see in the response body on how to improve the password.
    - - + + \ No newline at end of file diff --git a/reference/api/legacy/unleash/basic-auth.html b/reference/api/legacy/unleash/basic-auth.html index d9e4869e48..c268d85a97 100644 --- a/reference/api/legacy/unleash/basic-auth.html +++ b/reference/api/legacy/unleash/basic-auth.html @@ -20,15 +20,15 @@ - - + +

    Basic auth

    When using the insecure authentication method, identifying using basic auth against the API is enough. Since the insecure method doesn't require a password, it is enough to define the username when making HTTP requests.

    With curl

    Add the -u myemail@test.com flag to your curl command.

    With wget

    Add the --user=myemail@test.com flag to your wget command.

    - - + + \ No newline at end of file diff --git a/reference/api/legacy/unleash/client/features.html b/reference/api/legacy/unleash/client/features.html index 542fc46d2a..55c1614ed7 100644 --- a/reference/api/legacy/unleash/client/features.html +++ b/reference/api/legacy/unleash/client/features.html @@ -20,15 +20,15 @@ - - + +

    /api/client/features

    In order to access the client API endpoints you need to identify yourself. Unless you're using the none authentication method, you'll need to create a CLIENT token and add an Authorization header using the token.

    Fetching Feature Toggles

    GET: http://unleash.host.com/api/client/features

    HEADERS:

    • UNLEASH-APPNAME: appName
    • UNLEASH-INSTANCEID: instanceId

    This endpoint is the one all clients should use to fetch all available feature toggles from the unleash-server. The response returns all active feature toggles and their current strategy configuration. A feature toggle will have at least one configured strategy. A strategy will have a name and parameters map.

    Note: Clients should prefer the strategies property. Legacy properties (strategy & parameters) will be kept until version 2 of the format.

    This endpoint should never return anything besides a valid 20X or 304-response. It will also include an Etag-header. The value of this header can be used by clients as the value of the If-None-Match-header in the request to prevent a data transfer if the client already has the latest response locally.

    Example response:

    {
    "version": 1,
    "features": [
    {
    "name": "Feature.A",
    "type": "release",
    "enabled": false,
    "stale": false,
    "strategies": [
    {
    "name": "default",
    "parameters": {}
    }
    ],
    "strategy": "default",
    "parameters": {}
    },
    {
    "name": "Feature.B",
    "type": "killswitch",
    "enabled": true,
    "stale": false,
    "strategies": [
    {
    "name": "ActiveForUserWithId",
    "parameters": {
    "userIdList": "123,221,998"
    }
    },
    {
    "name": "GradualRolloutRandom",
    "parameters": {
    "percentage": "10"
    }
    }
    ],
    "strategy": "ActiveForUserWithId",
    "parameters": {
    "userIdList": "123,221,998"
    }
    }
    ]
    }

    Filter feature toggles

    Supports three params for now

    • tag - filters for features tagged with tag
    • project - filters for features belonging to project
    • namePrefix - filters for features beginning with prefix

    For tag and project performs OR filtering if multiple arguments

    To filter for any feature tagged with a simple tag with value taga or a simple tag with value tagb use

    GET https://unleash.host.com/api/client/features?tag[]=simple:taga&tag[]simple:tagb

    To filter for any feature belonging to project myproject use

    GET https://unleash.host.com/api/client/features?project=myproject

    Response format is the same as api/client/features

    Get specific feature toggle

    GET: http://unleash.host.com/api/client/features/:featureName

    Used to fetch details about a specific feature toggle. This is mainly provided to make it easy to debug the API and should not be used by the client implementations.

    Notice: You will not get a version property when fetching a specific feature toggle by name.

    {
    "name": "Feature.A",
    "type": "release",
    "enabled": false,
    "stale": false,
    "strategies": [
    {
    "name": "default",
    "parameters": {}
    }
    ],
    "strategy": "default",
    "parameters": {}
    }

    Strategy Constraints

    Availability

    Before Unleash 4.16, strategy constraints were only available to Unleash Pro and Enterprise users. From 4.16 onwards, they're available to everyone.

    Strategy definitions may also contain a constraints property. Strategy constraints is a feature in Unleash which work on context fields, which is defined as part of the Unleash Context. The purpose is to define a set of rules where all needs to be satisfied in order for the activation strategy to evaluate to true. A high level description of it is available online.

    Example response:

    The example shows strategy constraints in action. Constraints is a new field on the strategy-object. It is a list of constraints that need to be satisfied.

    In the example environment needs to be production AND userId must be either 123 OR 44 in order for the Unleash Client to evaluate the strategy, which in this scenario is “default” and will always evaluate to true.

    {
    "type": "release",
    "enabled": true,
    "stale": false,
    "name": "Demo",
    "strategies": [
    {
    "constraints": [
    {
    "contextName": "environment",
    "operator": "IN",
    "values": ["production"]
    },
    {
    "contextName": "userId",
    "operator": "IN",
    "values": ["123", "44"]
    }
    ],
    "name": "default",
    "parameters": {}
    }
    ]
    }
    • contextName - is the name of the field to look up on the unleash context.
    • values - is a list of values (string).
    • operator - is the logical action to take on the values Supported operator are:
      • IN - constraint is satisfied if one of the values in the list matches the value for this context field in the context.
      • NOT_IN - constraint is satisfied if NONE of the values is the list matches the value for this field in the context.

    Variants

    All feature toggles can also take an array of variants. You can read more about feature toggle variants.

    {
    "version": 1,
    "features": [
    {
    "name": "Demo",
    "type": "operational",
    "enabled": true,
    "stale": false,
    "strategies": [
    {
    "name": "default"
    }
    ],
    "variants": [
    {
    "name": "red",
    "weight": 500,
    "weightType": "variable",
    "payload": {
    "type": "string",
    "value": "something"
    },
    "overrides": [
    {
    "contextName": "userId",
    "values": ["123"]
    }
    ]
    },
    {
    "name": "blue",
    "weight": 500,
    "overrides": [],
    "weightType": "variable"
    }
    ]
    }
    ]
    }
    • payload - an optional object representing a payload to the variant. Takes two properties if present type and value.
    • overrides - an optional array of overrides. If any context field matches any of the the defined overrides it means that the variant should be selected.
    - - + + \ No newline at end of file diff --git a/reference/api/legacy/unleash/client/metrics.html b/reference/api/legacy/unleash/client/metrics.html index 74b4475f08..788efe1b9d 100644 --- a/reference/api/legacy/unleash/client/metrics.html +++ b/reference/api/legacy/unleash/client/metrics.html @@ -20,15 +20,15 @@ - - + +

    /api/client/metrics

    In order to access the client API endpoints you need to identify yourself. Unless you're using the none authentication method, you'll need to create a CLIENT token and add an Authorization header using the token.

    Send metrics

    POST: http://unleash.host.com/api/client/metrics

    Register a metrics payload with a timed bucket.

    {
    "appName": "appName",
    "instanceId": "instanceId",
    "bucket": {
    "start": "2016-11-03T07:16:43.572Z",
    "stop": "2016-11-03T07:16:53.572Z",
    "toggles": {
    "toggle-name-1": {
    "yes": 123,
    "no": 321
    },
    "toggle-name-2": {
    "yes": 111,
    "no": 0
    }
    }
    }
    }
    - - + + \ No newline at end of file diff --git a/reference/api/legacy/unleash/client/register.html b/reference/api/legacy/unleash/client/register.html index 519dcf0807..c59dbf77c9 100644 --- a/reference/api/legacy/unleash/client/register.html +++ b/reference/api/legacy/unleash/client/register.html @@ -20,15 +20,15 @@ - - + +

    /api/client/register

    In order to access the client API endpoints you need to identify yourself. Unless you're using the none authentication method, you'll need to create a CLIENT token and add an Authorization header using the token.

    Client registration

    POST: http://unleash.host.com/api/client/register

    Registers a client instance with the unleash server. The client should send all fields specified.

    {
    "appName": "appName",
    "instanceId": "instanceId",
    "sdkVersion": "unleash-client-java:2.2.0",
    "strategies": ["default", "some-strategy-1"],
    "started": "2016-11-03T07:16:43.572Z",
    "interval": 10000
    }

    Fields:

    • appName - Name of the application seen by unleash-server
    • instanceId - Instance id for this application (typically hostname, podId or similar)
    • sdkVersion - Optional field that describes the sdk version (name:version)
    • strategies - List of strategies implemented by this application
    • started - When this client started. Should be reported as ISO8601 time.
    • interval - At which interval, in milliseconds, will this client be expected to send metrics
    - - + + \ No newline at end of file diff --git a/reference/api/legacy/unleash/internal/health.html b/reference/api/legacy/unleash/internal/health.html index 7cee3566ca..37a41fe184 100644 --- a/reference/api/legacy/unleash/internal/health.html +++ b/reference/api/legacy/unleash/internal/health.html @@ -20,15 +20,15 @@ - - + +
    - - + + \ No newline at end of file diff --git a/reference/api/legacy/unleash/internal/prometheus.html b/reference/api/legacy/unleash/internal/prometheus.html index d7a8d399cf..fb5e51931e 100644 --- a/reference/api/legacy/unleash/internal/prometheus.html +++ b/reference/api/legacy/unleash/internal/prometheus.html @@ -20,15 +20,15 @@ - - + +

    Internal Backstage API

    GET http://unleash.host.com/internal-backstage/prometheus

    Unleash uses Prometheus internally to collect metrics. By default, the metrics are available at /internal-backstage/prometheus. You can disable this endpoint by setting the serverMetrics option to false.

    Note that it's not recommended to expose Prometheus metrics to the public as of the Prometheus pentest-report issue PRM-01-002. Thus, if you want to keep metrics enabled, you should block all external access to /internal-backstage/* on the network layer to keep your instance secure.

    Read more about Prometheus

    Annotations

    Unleash will automatically count all updates for all toggles under the metric name feature_toggle_update_total, and the toggle name is will be set as a label value. This information can be used to create annotations in grafana for everytime a feature toggle is changed.

    You can use this query in grafana to achieve this:

    delta(feature_toggle_update_total{toggle="Demo"}[1m]) != bool 0

    Another useful counter is the feature_toggle_usage_total which will give you the numbers for how many times a feature toggle has been evaluated to active or not.

    - - + + \ No newline at end of file diff --git a/reference/api/unleash.html b/reference/api/unleash.html index 74cd730883..dd87c1ffb4 100644 --- a/reference/api/unleash.html +++ b/reference/api/unleash.html @@ -20,15 +20,15 @@ - - + +
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/add-access-to-project.html b/reference/api/unleash/add-access-to-project.html index c8ed37af6a..96291e0b18 100644 --- a/reference/api/unleash/add-access-to-project.html +++ b/reference/api/unleash/add-access-to-project.html @@ -20,15 +20,15 @@ - - + +

    Configure project access

    POST /api/admin/projects/:projectId/access

    Configure project access for groups and single users. The provided users and groups will be given the roles specified in the payload.

    Request

    Path Parameters

    • projectId string required

    Body

    required

    projectAddAccessSchema

    • roles integer[] required

      A list of role IDs

    • groups integer[] required

      A list of group IDs

    • users integer[] required

      A list of user IDs

    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/add-change-request-comment.html b/reference/api/unleash/add-change-request-comment.html index 3a2c7c982e..509b077612 100644 --- a/reference/api/unleash/add-change-request-comment.html +++ b/reference/api/unleash/add-change-request-comment.html @@ -20,15 +20,15 @@ - - + +

    This endpoint will add a comment to a change request

    POST /api/admin/projects/:projectId/change-requests/:id/comments

    This endpoint will add a comment to a change request for the user making the request.

    Request

    Path Parameters

    • projectId string required
    • id string required

    Body

    required

    changeRequestAddCommentSchema

    • text string required

      The content of the comment.

    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/add-default-strategy-to-project-environment.html b/reference/api/unleash/add-default-strategy-to-project-environment.html index 061aad36ec..0d0db890c2 100644 --- a/reference/api/unleash/add-default-strategy-to-project-environment.html +++ b/reference/api/unleash/add-default-strategy-to-project-environment.html @@ -20,15 +20,15 @@ - - + +

    Set environment-default strategy

    POST /api/admin/projects/:projectId/environments/:environment/default-strategy

    Sets a default strategy for this environment. Unleash will use this strategy by default when enabling a toggle. Use the wild card "*" for :environment to add to all environments.

    Request

    Path Parameters

    • projectId string required
    • environment string required

    Body

    required

    createFeatureStrategySchema

    • name string required

      The name of the strategy type

    • title string nullable

      A descriptive title for the strategy

    • disabled boolean nullable

      A toggle to disable the strategy. defaults to false. Disabled strategies are not evaluated or returned to the SDKs

    • sortOrder number

      The order of the strategy in the list

    • constraints object[]

      A list of the constraints attached to the strategy. See https://docs.getunleash.io/reference/strategy-constraints

    • Array [
    • contextName string required

      The name of the context field that this constraint should apply to.

    • operator string required

      Possible values: [NOT_IN, IN, STR_ENDS_WITH, STR_STARTS_WITH, STR_CONTAINS, NUM_EQ, NUM_GT, NUM_GTE, NUM_LT, NUM_LTE, DATE_AFTER, DATE_BEFORE, SEMVER_EQ, SEMVER_GT, SEMVER_LT]

      The operator to use when evaluating this constraint. For more information about the various operators, refer to the strategy constraint operator documentation.

    • caseInsensitive boolean

      Default value: false

      Whether the operator should be case sensitive or not. Defaults to false (being case sensitive).

    • inverted boolean

      Default value: false

      Whether the result should be negated or not. If true, will turn a true result into a false result and vice versa.

    • values string[]

      The context values that should be used for constraint evaluation. Use this property instead of value for properties that accept multiple values.

    • value string

      The context value that should be used for constraint evaluation. Use this property instead of values for properties that only accept single values.

    • ]
    • variants object[]

      Strategy level variants

    • Array [
    • name string required

      The variant name. Must be unique for this feature toggle

    • weight integer required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is an integer between 0 and 1000. See the section on variant weights for more information

    • weightType string required

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000. Refer to the variant weight documentation.

    • stickiness string required

      The stickiness to use for distribution of this variant. Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • ]
    • parameters object

      A list of parameters for a strategy

    • property name* string
    • segments number[]

      Ids of segments to use for this strategy

    Responses

    createFeatureStrategySchema

    Schema
    • name string required

      The name of the strategy type

    • title string nullable

      A descriptive title for the strategy

    • disabled boolean nullable

      A toggle to disable the strategy. defaults to false. Disabled strategies are not evaluated or returned to the SDKs

    • sortOrder number

      The order of the strategy in the list

    • constraints object[]

      A list of the constraints attached to the strategy. See https://docs.getunleash.io/reference/strategy-constraints

    • Array [
    • contextName string required

      The name of the context field that this constraint should apply to.

    • operator string required

      Possible values: [NOT_IN, IN, STR_ENDS_WITH, STR_STARTS_WITH, STR_CONTAINS, NUM_EQ, NUM_GT, NUM_GTE, NUM_LT, NUM_LTE, DATE_AFTER, DATE_BEFORE, SEMVER_EQ, SEMVER_GT, SEMVER_LT]

      The operator to use when evaluating this constraint. For more information about the various operators, refer to the strategy constraint operator documentation.

    • caseInsensitive boolean

      Default value: false

      Whether the operator should be case sensitive or not. Defaults to false (being case sensitive).

    • inverted boolean

      Default value: false

      Whether the result should be negated or not. If true, will turn a true result into a false result and vice versa.

    • values string[]

      The context values that should be used for constraint evaluation. Use this property instead of value for properties that accept multiple values.

    • value string

      The context value that should be used for constraint evaluation. Use this property instead of values for properties that only accept single values.

    • ]
    • variants object[]

      Strategy level variants

    • Array [
    • name string required

      The variant name. Must be unique for this feature toggle

    • weight integer required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is an integer between 0 and 1000. See the section on variant weights for more information

    • weightType string required

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000. Refer to the variant weight documentation.

    • stickiness string required

      The stickiness to use for distribution of this variant. Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • ]
    • parameters object

      A list of parameters for a strategy

    • property name* string
    • segments number[]

      Ids of segments to use for this strategy

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/add-environment-to-project.html b/reference/api/unleash/add-environment-to-project.html index 256ee0ec39..8100908431 100644 --- a/reference/api/unleash/add-environment-to-project.html +++ b/reference/api/unleash/add-environment-to-project.html @@ -20,15 +20,15 @@ - - + +

    Add an environment to a project.

    POST /api/admin/projects/:projectId/environments

    This endpoint adds the provided environment to the specified project, with optional support for enabling and disabling change requests for the environment and project.

    Request

    Path Parameters

    • projectId string required

    Body

    required

    projectEnvironmentSchema

    • environment string required

      The environment to add to the project

    • changeRequestsEnabled boolean

      Whether change requests should be enabled or for this environment on the project or not

    • defaultStrategy object

      A default strategy to create for this environment on the project.

    • name string required

      The name of the strategy type

    • title string nullable

      A descriptive title for the strategy

    • disabled boolean nullable

      A toggle to disable the strategy. defaults to false. Disabled strategies are not evaluated or returned to the SDKs

    • sortOrder number

      The order of the strategy in the list

    • constraints object[]

      A list of the constraints attached to the strategy. See https://docs.getunleash.io/reference/strategy-constraints

    • Array [
    • contextName string required

      The name of the context field that this constraint should apply to.

    • operator string required

      Possible values: [NOT_IN, IN, STR_ENDS_WITH, STR_STARTS_WITH, STR_CONTAINS, NUM_EQ, NUM_GT, NUM_GTE, NUM_LT, NUM_LTE, DATE_AFTER, DATE_BEFORE, SEMVER_EQ, SEMVER_GT, SEMVER_LT]

      The operator to use when evaluating this constraint. For more information about the various operators, refer to the strategy constraint operator documentation.

    • caseInsensitive boolean

      Default value: false

      Whether the operator should be case sensitive or not. Defaults to false (being case sensitive).

    • inverted boolean

      Default value: false

      Whether the result should be negated or not. If true, will turn a true result into a false result and vice versa.

    • values string[]

      The context values that should be used for constraint evaluation. Use this property instead of value for properties that accept multiple values.

    • value string

      The context value that should be used for constraint evaluation. Use this property instead of values for properties that only accept single values.

    • ]
    • variants object[]

      Strategy level variants

    • Array [
    • name string required

      The variant name. Must be unique for this feature toggle

    • weight integer required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is an integer between 0 and 1000. See the section on variant weights for more information

    • weightType string required

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000. Refer to the variant weight documentation.

    • stickiness string required

      The stickiness to use for distribution of this variant. Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • ]
    • parameters object

      A list of parameters for a strategy

    • property name* string
    • segments number[]

      Ids of segments to use for this strategy

    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/add-favorite-feature.html b/reference/api/unleash/add-favorite-feature.html index 79c5c7416e..fb72fd17dd 100644 --- a/reference/api/unleash/add-favorite-feature.html +++ b/reference/api/unleash/add-favorite-feature.html @@ -20,15 +20,15 @@ - - + +

    Add feature to favorites

    POST /api/admin/projects/:projectId/features/:featureName/favorites

    This endpoint marks the feature in the url as favorite

    Request

    Path Parameters

    • projectId string required
    • featureName string required
    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/add-favorite-project.html b/reference/api/unleash/add-favorite-project.html index 69a2b1677a..135e3f7bd2 100644 --- a/reference/api/unleash/add-favorite-project.html +++ b/reference/api/unleash/add-favorite-project.html @@ -20,15 +20,15 @@ - - + +

    Add project to favorites

    POST /api/admin/projects/:projectId/favorites

    This endpoint marks the project in the url as favorite

    Request

    Path Parameters

    • projectId string required
    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/add-feature-dependency.html b/reference/api/unleash/add-feature-dependency.html index 4ea9a86b4e..40d6da210a 100644 --- a/reference/api/unleash/add-feature-dependency.html +++ b/reference/api/unleash/add-feature-dependency.html @@ -20,15 +20,15 @@ - - + +

    Add a feature dependency.

    POST /api/admin/projects/:projectId/features/:child/dependencies

    Add a dependency to a parent feature. Each environment will resolve corresponding dependency independently.

    Request

    Path Parameters

    • projectId string required
    • child string required

    Body

    required

    createDependentFeatureSchema

    • feature string required

      The name of the feature we depend on.

    • enabled boolean

      Whether the parent feature should be enabled. When false variants are ignored. true by default.

    • variants string[]

      The list of variants the parent feature should resolve to. Leave empty when you only want to check the enabled status.

    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/add-feature-strategy.html b/reference/api/unleash/add-feature-strategy.html index 65374707f5..b41515e6f0 100644 --- a/reference/api/unleash/add-feature-strategy.html +++ b/reference/api/unleash/add-feature-strategy.html @@ -20,15 +20,15 @@ - - + +

    Add a strategy to a feature toggle

    POST /api/admin/projects/:projectId/features/:featureName/environments/:environment/strategies

    Add a strategy to a feature toggle in the specified environment.

    Request

    Path Parameters

    • projectId string required
    • featureName string required
    • environment string required

    Body

    required

    createFeatureStrategySchema

    • name string required

      The name of the strategy type

    • title string nullable

      A descriptive title for the strategy

    • disabled boolean nullable

      A toggle to disable the strategy. defaults to false. Disabled strategies are not evaluated or returned to the SDKs

    • sortOrder number

      The order of the strategy in the list

    • constraints object[]

      A list of the constraints attached to the strategy. See https://docs.getunleash.io/reference/strategy-constraints

    • Array [
    • contextName string required

      The name of the context field that this constraint should apply to.

    • operator string required

      Possible values: [NOT_IN, IN, STR_ENDS_WITH, STR_STARTS_WITH, STR_CONTAINS, NUM_EQ, NUM_GT, NUM_GTE, NUM_LT, NUM_LTE, DATE_AFTER, DATE_BEFORE, SEMVER_EQ, SEMVER_GT, SEMVER_LT]

      The operator to use when evaluating this constraint. For more information about the various operators, refer to the strategy constraint operator documentation.

    • caseInsensitive boolean

      Default value: false

      Whether the operator should be case sensitive or not. Defaults to false (being case sensitive).

    • inverted boolean

      Default value: false

      Whether the result should be negated or not. If true, will turn a true result into a false result and vice versa.

    • values string[]

      The context values that should be used for constraint evaluation. Use this property instead of value for properties that accept multiple values.

    • value string

      The context value that should be used for constraint evaluation. Use this property instead of values for properties that only accept single values.

    • ]
    • variants object[]

      Strategy level variants

    • Array [
    • name string required

      The variant name. Must be unique for this feature toggle

    • weight integer required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is an integer between 0 and 1000. See the section on variant weights for more information

    • weightType string required

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000. Refer to the variant weight documentation.

    • stickiness string required

      The stickiness to use for distribution of this variant. Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • ]
    • parameters object

      A list of parameters for a strategy

    • property name* string
    • segments number[]

      Ids of segments to use for this strategy

    Responses

    featureStrategySchema

    Schema
    • id string

      A uuid for the feature strategy

    • name string required

      The name or type of strategy

    • title string nullable

      A descriptive title for the strategy

    • disabled boolean nullable

      A toggle to disable the strategy. defaults to false. Disabled strategies are not evaluated or returned to the SDKs

    • featureName string

      The name or feature the strategy is attached to

    • sortOrder number

      The order of the strategy in the list

    • segments number[]

      A list of segment ids attached to the strategy

    • constraints object[]

      A list of the constraints attached to the strategy. See https://docs.getunleash.io/reference/strategy-constraints

    • Array [
    • contextName string required

      The name of the context field that this constraint should apply to.

    • operator string required

      Possible values: [NOT_IN, IN, STR_ENDS_WITH, STR_STARTS_WITH, STR_CONTAINS, NUM_EQ, NUM_GT, NUM_GTE, NUM_LT, NUM_LTE, DATE_AFTER, DATE_BEFORE, SEMVER_EQ, SEMVER_GT, SEMVER_LT]

      The operator to use when evaluating this constraint. For more information about the various operators, refer to the strategy constraint operator documentation.

    • caseInsensitive boolean

      Default value: false

      Whether the operator should be case sensitive or not. Defaults to false (being case sensitive).

    • inverted boolean

      Default value: false

      Whether the result should be negated or not. If true, will turn a true result into a false result and vice versa.

    • values string[]

      The context values that should be used for constraint evaluation. Use this property instead of value for properties that accept multiple values.

    • value string

      The context value that should be used for constraint evaluation. Use this property instead of values for properties that only accept single values.

    • ]
    • variants object[]

      Strategy level variants

    • Array [
    • name string required

      The variant name. Must be unique for this feature toggle

    • weight integer required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is an integer between 0 and 1000. See the section on variant weights for more information

    • weightType string required

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000. Refer to the variant weight documentation.

    • stickiness string required

      The stickiness to use for distribution of this variant. Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • ]
    • parameters object

      A list of parameters for a strategy

    • property name* string
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/add-public-signup-token-user.html b/reference/api/unleash/add-public-signup-token-user.html index 1d6913af0a..77945ab079 100644 --- a/reference/api/unleash/add-public-signup-token-user.html +++ b/reference/api/unleash/add-public-signup-token-user.html @@ -20,15 +20,15 @@ - - + +

    Add a user via a signup token

    POST /invite/:token/signup

    Create a user with the viewer root role and link them to the provided signup token

    Request

    Path Parameters

    • token string required

    Body

    required

    createInvitedUserSchema

    • username string

      The user's username. Must be unique if provided.

    • email string required

      The invited user's email address

    • name string required

      The user's name

    • password string required

      The user's password

    Responses

    userSchema

    Schema
    • id integer required

      The user id

    • isAPI boolean deprecated

      (Deprecated): Used internally to know which operations the user should be allowed to perform

    • name string nullable

      Name of the user

    • email string

      Email of the user

    • username string nullable

      A unique username for the user

    • imageUrl string

      URL used for the userprofile image

    • inviteLink string

      If the user is actively inviting other users, this is the link that can be shared with other users

    • loginAttempts integer

      How many unsuccessful attempts at logging in has the user made

    • emailSent boolean

      Is the welcome email sent to the user or not

    • rootRole integer

      Which root role this user is assigned

    • seenAt date-time nullable

      The last time this user logged in

    • createdAt date-time

      The user was created at this time

    • accountType string

      A user is either an actual User or a Service Account

    • permissions string[]

      Deprecated

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/add-role-access-to-project.html b/reference/api/unleash/add-role-access-to-project.html index cd38fe8201..bc7916bb29 100644 --- a/reference/api/unleash/add-role-access-to-project.html +++ b/reference/api/unleash/add-role-access-to-project.html @@ -20,15 +20,15 @@ - - + +

    Configure project role access

    POST /api/admin/projects/:projectId/role/:roleId/access
    deprecated

    This endpoint has been deprecated and may be removed in future versions of the API.

    Configure project access for groups and single users. The provided users and groups will be given the role specified in the URL parameters. This endpoint is deprecated. Use /:projectId/access instead.

    Request

    Path Parameters

    • projectId string required
    • roleId string required

    Body

    required

    projectAddRoleAccessSchema

    • groups object[]required

      A list of groups IDs

    • Array [
    • id integer required

      A group ID

    • ]
    • users object[]required

      A list of user IDs

    • Array [
    • id integer required

      A user ID

    • ]
    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/add-role-to-user.html b/reference/api/unleash/add-role-to-user.html index 44aa671a2a..5b3384ac0d 100644 --- a/reference/api/unleash/add-role-to-user.html +++ b/reference/api/unleash/add-role-to-user.html @@ -20,15 +20,15 @@ - - + +

    Add user to project

    POST /api/admin/projects/:projectId/users/:userId/roles/:roleId
    deprecated

    This endpoint has been deprecated and may be removed in future versions of the API.

    Adds the specified user to a project with the provided role. This endpoint is deprecated. Use /:projectId/access instead.

    Request

    Path Parameters

    • projectId string required
    • userId string required
    • roleId string required
    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/add-tag-to-features.html b/reference/api/unleash/add-tag-to-features.html index aa1993c871..69688c922a 100644 --- a/reference/api/unleash/add-tag-to-features.html +++ b/reference/api/unleash/add-tag-to-features.html @@ -20,15 +20,15 @@ - - + +

    Adds a tag to the specified features

    PUT /api/admin/projects/:projectId/tags

    Add a tag to a list of features. Create tags if needed.

    Request

    Path Parameters

    • projectId string required

    Body

    required

    tagsBulkAddSchema

    • features string[] required

      Possible values: non-empty

      The list of features that will be affected by the tag changes.

    • tags objectrequired

      The tag changes to be applied to the features.

    • addedTags object[]required

      Tags to add to the feature.

    • Array [
    • value string required

      Possible values: >= 2 characters and <= 50 characters

      The value of the tag

    • type string required

      Possible values: >= 2 characters and <= 50 characters

      Default value: simple

      The type of the tag

    • ]
    • removedTags object[]required

      Tags to remove from the feature.

    • Array [
    • value string required

      Possible values: >= 2 characters and <= 50 characters

      The value of the tag

    • type string required

      Possible values: >= 2 characters and <= 50 characters

      Default value: simple

      The type of the tag

    • ]
    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/add-tag.html b/reference/api/unleash/add-tag.html index abddd84c62..430e88723d 100644 --- a/reference/api/unleash/add-tag.html +++ b/reference/api/unleash/add-tag.html @@ -20,15 +20,15 @@ - - + +

    Adds a tag to a feature.

    POST /api/admin/features/:featureName/tags

    Adds a tag to a feature if the feature and tag type exist in the system. The operation is idempotent, so adding an existing tag will result in a successful response.

    Request

    Path Parameters

    • featureName string required

    Body

    required

    tagSchema

    • value string required

      Possible values: >= 2 characters and <= 50 characters

      The value of the tag

    • type string required

      Possible values: >= 2 characters and <= 50 characters

      Default value: simple

      The type of the tag

    Responses

    The resource was successfully created.

    Response Headers
    • location string

      The location of the newly created resource.

    Schema
    • value string required

      Possible values: >= 2 characters and <= 50 characters

      The value of the tag

    • type string required

      Possible values: >= 2 characters and <= 50 characters

      Default value: simple

      The type of the tag

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/addons.html b/reference/api/unleash/addons.html index 4279f0c4bc..d71d8d2a38 100644 --- a/reference/api/unleash/addons.html +++ b/reference/api/unleash/addons.html @@ -20,15 +20,15 @@ - - + +
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/admin-ui.html b/reference/api/unleash/admin-ui.html index 5f336a184c..57c1e18ac5 100644 --- a/reference/api/unleash/admin-ui.html +++ b/reference/api/unleash/admin-ui.html @@ -20,15 +20,15 @@ - - + +
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/api-tokens.html b/reference/api/unleash/api-tokens.html index c54f8fc8fc..cad379f117 100644 --- a/reference/api/unleash/api-tokens.html +++ b/reference/api/unleash/api-tokens.html @@ -20,15 +20,15 @@ - - + +
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/archive-feature.html b/reference/api/unleash/archive-feature.html index f19e1da481..fc9d9aa1a1 100644 --- a/reference/api/unleash/archive-feature.html +++ b/reference/api/unleash/archive-feature.html @@ -20,15 +20,15 @@ - - + +

    Archive a feature toggle

    DELETE /api/admin/projects/:projectId/features/:featureName

    This endpoint archives the specified feature if the feature belongs to the specified project.

    Request

    Path Parameters

    • projectId string required
    • featureName string required
    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/archive-features.html b/reference/api/unleash/archive-features.html index 12fddb8334..5904ea460b 100644 --- a/reference/api/unleash/archive-features.html +++ b/reference/api/unleash/archive-features.html @@ -20,15 +20,15 @@ - - + +

    Archives a list of features

    POST /api/admin/projects/:projectId/archive

    This endpoint archives the specified features. Any features that are already archived or that don't exist are ignored. All existing features (whether already archived or not) that are provided must belong to the specified project.

    Request

    Path Parameters

    • projectId string required

    Body

    required

    batchFeaturesSchema

    • features string[] required

      List of feature toggle names

    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/archive.html b/reference/api/unleash/archive.html index 1e275de3ee..716f214aa9 100644 --- a/reference/api/unleash/archive.html +++ b/reference/api/unleash/archive.html @@ -20,15 +20,15 @@ - - + +
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/auth.html b/reference/api/unleash/auth.html index 4084778462..3f7e464eff 100644 --- a/reference/api/unleash/auth.html +++ b/reference/api/unleash/auth.html @@ -20,15 +20,15 @@ - - + +

    Auth

    Manage logins, passwords, etc.

    - - + + \ No newline at end of file diff --git a/reference/api/unleash/banners.html b/reference/api/unleash/banners.html index f493cf13a6..8d7f4e1b6c 100644 --- a/reference/api/unleash/banners.html +++ b/reference/api/unleash/banners.html @@ -20,15 +20,15 @@ - - + +
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/bulk-metrics.html b/reference/api/unleash/bulk-metrics.html index 406193f348..a264a77359 100644 --- a/reference/api/unleash/bulk-metrics.html +++ b/reference/api/unleash/bulk-metrics.html @@ -20,8 +20,8 @@ - - + + @@ -29,7 +29,7 @@

    Send metrics from Edge

    POST /edge/metrics

    This operation accepts batched metrics from Edge. Metrics will be inserted into Unleash's metrics storage

    Request

    Body

    required

    bulkMetricsSchema

    • applications object[]required

      A list of applications registered by an Unleash SDK

    • Array [
    • connectVia object[]

      A list of applications this app registration has been registered through. If connected directly to Unleash, this is an empty list. This can be used in later visualizations to tell how many levels of proxy or Edge instances our SDKs have connected through

    • Array [
    • appName string required
    • instanceId string required
    • ]
    • appName string required

      The name of the application that is evaluating toggles

    • environment string required

      Which environment the application is running in

    • instanceId string required

      A (somewhat) unique identifier for the application

    • interval number

      How often (in seconds) the application refreshes its features

    • started object

      The application started at

      oneOf
    • string date-time

      An RFC-3339-compliant timestamp.

    • strategies string[]

      Enabled strategies in the application

    • sdkVersion string

      The version the sdk is running. Typically :

    • ]
    • metrics object[]required

      a list of client usage metrics registered by downstream providers. (Typically Unleash Edge)

    • Array [
    • featureName string required

      Name of the feature checked by the SDK

    • appName string required

      The name of the application the SDK is being used in

    • environment string required

      Which environment the SDK is being used in

    • timestamp object

      The start of the time window these metrics are valid for. The window is 1 hour wide

      oneOf
    • string date-time

      An RFC-3339-compliant timestamp.

    • yes integer

      How many times the toggle evaluated to true

    • no integer

      How many times the toggle evaluated to false

    • variants object

      How many times each variant was returned

    • property name* integer
    • ]
    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/bulk-toggle-features-environment-off.html b/reference/api/unleash/bulk-toggle-features-environment-off.html index 342ba50fd4..00daa3df92 100644 --- a/reference/api/unleash/bulk-toggle-features-environment-off.html +++ b/reference/api/unleash/bulk-toggle-features-environment-off.html @@ -20,15 +20,15 @@ - - + +

    Bulk disable a list of features

    POST /api/admin/projects/:projectId/bulk_features/environments/:environment/off

    This endpoint disables multiple feature toggles.

    Request

    Path Parameters

    • projectId string required
    • environment string required

    Body

    required

    bulkToggleFeaturesSchema

    • features string[] required

      The features that we want to bulk toggle

    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/bulk-toggle-features-environment-on.html b/reference/api/unleash/bulk-toggle-features-environment-on.html index 114c643d54..289252cb6c 100644 --- a/reference/api/unleash/bulk-toggle-features-environment-on.html +++ b/reference/api/unleash/bulk-toggle-features-environment-on.html @@ -20,15 +20,15 @@ - - + +

    Bulk enable a list of features

    POST /api/admin/projects/:projectId/bulk_features/environments/:environment/on

    This endpoint enables multiple feature toggles.

    Request

    Path Parameters

    • projectId string required
    • environment string required

    Body

    required

    bulkToggleFeaturesSchema

    • features string[] required

      The features that we want to bulk toggle

    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/change-my-password.html b/reference/api/unleash/change-my-password.html index dee8af7b69..b3488a55e2 100644 --- a/reference/api/unleash/change-my-password.html +++ b/reference/api/unleash/change-my-password.html @@ -20,15 +20,15 @@ - - + +

    Change your own password

    POST /api/admin/user/change-password

    Requires specifying old password and confirming new password

    Request

    Body

    required

    passwordSchema

    • password string required

      The new password to change or validate.

    • oldPassword string

      The old password the user is changing. This field is for the non-admin users changing their own password.

    • confirmPassword string

      The confirmation of the new password. This field is for the non-admin users changing their own password.

    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/change-password.html b/reference/api/unleash/change-password.html index 339faecefa..9a0a499694 100644 --- a/reference/api/unleash/change-password.html +++ b/reference/api/unleash/change-password.html @@ -20,15 +20,15 @@ - - + +

    Changes a user password

    POST /auth/reset/password

    Allows users with a valid reset token to reset their password without remembering their old password

    Request

    Body

    required

    changePasswordSchema

    • token string required

      A reset token used to validate that the user is allowed to change the password.

    • password string required

      The new password for the user

    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/change-project.html b/reference/api/unleash/change-project.html index ff90c7afbd..4ec5c7e02e 100644 --- a/reference/api/unleash/change-project.html +++ b/reference/api/unleash/change-project.html @@ -20,15 +20,15 @@ - - + +

    Move feature to project

    POST /api/admin/projects/:projectId/features/:featureName/changeProject

    Moves the specified feature to the new project in the request schema. Requires you to have permissions to move the feature toggle in both projects. Features that are included in any active change requests can not be moved.

    Request

    Path Parameters

    • projectId string required
    • featureName string required

    Body

    required

    changeProjectSchema

    • newProjectId string required

      The project to move the feature toggle to.

    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/change-request.html b/reference/api/unleash/change-request.html index 3061c6e64e..e0a36ab957 100644 --- a/reference/api/unleash/change-request.html +++ b/reference/api/unleash/change-request.html @@ -20,8 +20,8 @@ - - + + @@ -30,7 +30,7 @@

    Create/Add change to a change request

    POST /api/admin/projects/:projectId/environments/:environment/change-requests

    Given a change request exists, this endpoint will attempt to add a change to an existing change request for the user. If a change request does not exist. It will attempt to create it.

    Request

    Path Parameters

    • projectId string required
    • environment string required

    Body

    required

    changeRequestOneOrManyCreateSchema

      oneOf
    • action string required

      Possible values: [updateSegment]

      The name of this action.

    • payload objectrequired

      Data used to create or update a segment

    • id integer required

      The ID of the segment to update.

    Responses

    changeRequestSchema

    Schema
      oneOf
    • id number required

      This change requests's ID.

    • title string

      A title describing the change request's content.

    • environment string required

      The environment in which the changes should be applied.

    • minApprovals number required

      The minimum number of approvals required before this change request can be applied.

    • project string required

      The project this change request belongs to.

    • features object[]required

      The list of features and their changes that relate to this change request.

    • Array [
    • name string required

      The name of the feature

    • conflict string

      A string describing the conflicts related to this change. Only present if there are any concflicts on the feature level.

    • changes object[]required

      List of changes inside change request. This list may be empty when listing all change requests for a project.

    • Array [
    • id number required

      The ID of this change.

    • action string required

      The kind of action that the change contains information about.

    • conflict string

      A description of the conflict caused by this change. Only present if there are any conflicts.

    • payload objectrequired

      The data required to perform this action.

      oneOf
    • string
    • createdBy object

      The user who created this change.

    • username string nullable

      The user's username.

    • imageUrl uri nullable

      The URL where the user's image can be found.

    • createdAt date-time

      When this change was suggested

    • ]
    • defaultChange object

      A description of a default change that will be applied with the change request to prevent invalid states.

      Default changes are changes that are applied in addition to explicit user-specified changes when a change request is applied. Any default changes are applied in the background and are not a real part of the change request.

    • action string required

      The kind of action this is.

    • payload object required

      The necessary data to perform this change.

    • ]
    • segments object[]required

      The list of segments and their changes that relate to this change request.

    • Array [
    • id number required

      The ID of this change.

    • action string required

      The kind of action that the change contains information about.

    • conflict string

      A description of the conflict caused by this change. Only present if there are any conflicts.

    • payload objectrequired

      The data required to perform this action.

      oneOf
    • string
    • createdBy object

      The user who created this change.

    • username string nullable

      The user's username.

    • imageUrl uri nullable

      The URL where the user's image can be found.

    • createdAt date-time

      When this change was suggested

    • name string required

      The current name of the segment

    • ]
    • approvals object[]

      A list of approvals that this change request has received.

    • Array [
    • createdBy objectrequired

      Information about the user who gave this approval.

    • id number

      The ID of the user who gave this approval.

    • username string

      The approving user's username.

    • imageUrl uri

      The URL where the user's image can be found.

    • createdAt date-time required

      When the approval was given.

    • ]
    • rejections object[]

      A list of rejections that this change request has received.

    • Array [
    • createdBy objectrequired

      Information about the user who gave this approval.

    • id number

      The ID of the user who gave this approval.

    • username string

      The approving user's username.

    • imageUrl uri

      The URL where the user's image can be found.

    • createdAt date-time required

      When the approval was given.

    • ]
    • comments object[]

      All comments that have been made on this change request.

    • Array [
    • id number

      The comment's ID. Unique per change request.

    • text string required

      The content of the comment.

    • createdBy objectrequired

      Information about the user who posted the comment

    • username string nullable

      The user's username.

    • imageUrl uri nullable

      The URL where the user's image can be found.

    • createdAt date-time required

      When the comment was made.

    • ]
    • createdBy objectrequired

      The user who created this change request.

    • username string nullable
    • imageUrl uri nullable

      The URL of the user's profile image.

    • createdAt date-time required

      When this change request was created.

    • state string required

      Possible values: [Draft, In review, Approved, Applied, Cancelled, Rejected]

      The current state of the change request.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/change-requests.html b/reference/api/unleash/change-requests.html index 21486ae9f4..81956419ee 100644 --- a/reference/api/unleash/change-requests.html +++ b/reference/api/unleash/change-requests.html @@ -20,15 +20,15 @@ - - + +

    Change Requests

    API for managing change requests.

    - - + + \ No newline at end of file diff --git a/reference/api/unleash/change-role-for-group.html b/reference/api/unleash/change-role-for-group.html index 70f2475471..0408abfd98 100644 --- a/reference/api/unleash/change-role-for-group.html +++ b/reference/api/unleash/change-role-for-group.html @@ -20,15 +20,15 @@ - - + +

    Update group's project role

    PUT /api/admin/projects/:projectId/groups/:groupId/roles/:roleId
    deprecated

    This endpoint has been deprecated and may be removed in future versions of the API.

    Updates the permissions that the group has within the given project. This endpoint is deprecated. Use /:projectId/users/:userId/roles instead.

    Request

    Path Parameters

    • projectId string required
    • groupId string required
    • roleId string required
    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/change-role-for-user.html b/reference/api/unleash/change-role-for-user.html index 991f3da99c..ff2adeedac 100644 --- a/reference/api/unleash/change-role-for-user.html +++ b/reference/api/unleash/change-role-for-user.html @@ -20,15 +20,15 @@ - - + +

    Update user's project role

    PUT /api/admin/projects/:projectId/users/:userId/roles/:roleId
    deprecated

    This endpoint has been deprecated and may be removed in future versions of the API.

    Replaces the given user's project role with the provided role. The user must already be a memeber of the project. This endpoint is deprecated. Use /:projectId/users/:userId/roles instead.

    Request

    Path Parameters

    • projectId string required
    • userId string required
    • roleId string required
    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/change-user-password.html b/reference/api/unleash/change-user-password.html index bd7b6129bc..469d213819 100644 --- a/reference/api/unleash/change-user-password.html +++ b/reference/api/unleash/change-user-password.html @@ -20,15 +20,15 @@ - - + +

    Change password for a user

    POST /api/admin/user-admin/:id/change-password

    Change password for a user as an admin

    Request

    Path Parameters

    • id string required

    Body

    required

    passwordSchema

    • password string required

      The new password to change or validate.

    • oldPassword string

      The old password the user is changing. This field is for the non-admin users changing their own password.

    • confirmPassword string

      The confirmation of the new password. This field is for the non-admin users changing their own password.

    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/check-dependencies-exist.html b/reference/api/unleash/check-dependencies-exist.html index a54688b99d..b6885fff3a 100644 --- a/reference/api/unleash/check-dependencies-exist.html +++ b/reference/api/unleash/check-dependencies-exist.html @@ -20,15 +20,15 @@ - - + +

    Check dependencies exist.

    GET /api/admin/projects/:projectId/dependencies

    Check if any dependencies exist in this Unleash instance

    Request

    Path Parameters

    • projectId string required
    Responses

    dependenciesExistSchema

    Schema
    • boolean

      true when any dependencies exist, false when no dependencies exist.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/check-license.html b/reference/api/unleash/check-license.html index 37b5e0825d..70e855b04e 100644 --- a/reference/api/unleash/check-license.html +++ b/reference/api/unleash/check-license.html @@ -20,15 +20,15 @@ - - + +
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/client.html b/reference/api/unleash/client.html index 8a6d238a73..8f5bd8bebc 100644 --- a/reference/api/unleash/client.html +++ b/reference/api/unleash/client.html @@ -20,15 +20,15 @@ - - + +
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/clone-environment.html b/reference/api/unleash/clone-environment.html index aa626eda32..abc6051549 100644 --- a/reference/api/unleash/clone-environment.html +++ b/reference/api/unleash/clone-environment.html @@ -20,15 +20,15 @@ - - + +

    Clones an environment

    POST /api/admin/environments/:name/clone

    Given an existing environment name and a set of options, this will create a copy of that environment

    Request

    Path Parameters

    • name string required
    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/clone-feature.html b/reference/api/unleash/clone-feature.html index 983a923b3c..5fa1ecc80e 100644 --- a/reference/api/unleash/clone-feature.html +++ b/reference/api/unleash/clone-feature.html @@ -20,15 +20,15 @@ - - + +

    Clone a feature toggle

    POST /api/admin/projects/:projectId/features/:featureName/clone

    Creates a copy of the specified feature toggle. The copy can be created in any project.

    Request

    Path Parameters

    • projectId string required
    • featureName string required

    Body

    required

    cloneFeatureSchema

    • name string required

      The name of the new feature

    • replaceGroupId boolean

      Whether to use the new feature name as its group ID or not. Group ID is used for calculating stickiness. Defaults to true.

    Responses

    featureSchema

    Schema
    • name string required

      Unique feature name

    • type string

      Type of the toggle e.g. experiment, kill-switch, release, operational, permission

    • description string nullable

      Detailed description of the feature

    • archived boolean

      true if the feature is archived

    • project string

      Name of the project the feature belongs to

    • enabled boolean

      true if the feature is enabled, otherwise false.

    • stale boolean

      true if the feature is stale based on the age and feature type, otherwise false.

    • favorite boolean

      true if the feature was favorited, otherwise false.

    • impressionData boolean

      true if the impression data collection is enabled for the feature, otherwise false.

    • createdAt date-time nullable

      The date the feature was created

    • archivedAt date-time nullable

      The date the feature was archived

    • lastSeenAt date-time nullable deprecated

      The date when metrics where last collected for the feature. This field is deprecated, use the one in featureEnvironmentSchema

    • environments object[]

      The list of environments where the feature can be used

    • Array [
    • name string required

      The name of the environment

    • featureName string

      The name of the feature

    • environment string

      The name of the environment

    • type string

      The type of the environment

    • enabled boolean required

      true if the feature is enabled for the environment, otherwise false.

    • sortOrder number

      The sort order of the feature environment in the feature environments list

    • variantCount number

      The number of defined variants

    • strategies object[]

      A list of activation strategies for the feature environment

    • Array [
    • id string

      A uuid for the feature strategy

    • name string required

      The name or type of strategy

    • title string nullable

      A descriptive title for the strategy

    • disabled boolean nullable

      A toggle to disable the strategy. defaults to false. Disabled strategies are not evaluated or returned to the SDKs

    • featureName string

      The name or feature the strategy is attached to

    • sortOrder number

      The order of the strategy in the list

    • segments number[]

      A list of segment ids attached to the strategy

    • constraints object[]

      A list of the constraints attached to the strategy. See https://docs.getunleash.io/reference/strategy-constraints

    • Array [
    • contextName string required

      The name of the context field that this constraint should apply to.

    • operator string required

      Possible values: [NOT_IN, IN, STR_ENDS_WITH, STR_STARTS_WITH, STR_CONTAINS, NUM_EQ, NUM_GT, NUM_GTE, NUM_LT, NUM_LTE, DATE_AFTER, DATE_BEFORE, SEMVER_EQ, SEMVER_GT, SEMVER_LT]

      The operator to use when evaluating this constraint. For more information about the various operators, refer to the strategy constraint operator documentation.

    • caseInsensitive boolean

      Default value: false

      Whether the operator should be case sensitive or not. Defaults to false (being case sensitive).

    • inverted boolean

      Default value: false

      Whether the result should be negated or not. If true, will turn a true result into a false result and vice versa.

    • values string[]

      The context values that should be used for constraint evaluation. Use this property instead of value for properties that accept multiple values.

    • value string

      The context value that should be used for constraint evaluation. Use this property instead of values for properties that only accept single values.

    • ]
    • variants object[]

      Strategy level variants

    • Array [
    • name string required

      The variant name. Must be unique for this feature toggle

    • weight integer required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is an integer between 0 and 1000. See the section on variant weights for more information

    • weightType string required

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000. Refer to the variant weight documentation.

    • stickiness string required

      The stickiness to use for distribution of this variant. Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • ]
    • parameters object

      A list of parameters for a strategy

    • property name* string
    • ]
    • variants object[]

      A list of variants for the feature environment

    • Array [
    • name string required

      The variants name. Is unique for this feature toggle

    • weight number required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is a number between 0 and 1000. See the section on variant weights for more information

    • weightType string

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000

    • stickiness string

      Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • overrides object[]

      Overrides assigning specific variants to specific users. The weighting system automatically assigns users to specific groups for you, but any overrides in this list will take precedence.

    • Array [
    • contextName string required

      The name of the context field used to determine overrides

    • values string[] required

      Which values that should be overriden

    • ]
    • ]
    • lastSeenAt date-time nullable

      The date when metrics where last collected for the feature environment

    • hasStrategies boolean

      Whether the feature has any strategies defined.

    • hasEnabledStrategies boolean

      Whether the feature has any enabled strategies defined.

    • ]
    • variants object[]deprecated

      The list of feature variants

    • Array [
    • name string required

      The variants name. Is unique for this feature toggle

    • weight number required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is a number between 0 and 1000. See the section on variant weights for more information

    • weightType string

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000

    • stickiness string

      Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • overrides object[]

      Overrides assigning specific variants to specific users. The weighting system automatically assigns users to specific groups for you, but any overrides in this list will take precedence.

    • Array [
    • contextName string required

      The name of the context field used to determine overrides

    • values string[] required

      Which values that should be overriden

    • ]
    • ]
    • strategies object[] deprecated

      This is a legacy field that will be deprecated

    • tags object[]nullable

      The list of feature tags

    • Array [
    • value string required

      Possible values: >= 2 characters and <= 50 characters

      The value of the tag

    • type string required

      Possible values: >= 2 characters and <= 50 characters

      Default value: simple

      The type of the tag

    • ]
    • children string[]

      The list of child feature names. This is an experimental field and may change.

    • dependencies object[]

      The list of parent dependencies. This is an experimental field and may change.

    • Array [
    • feature string required

      The name of the parent feature

    • enabled boolean

      Whether the parent feature is enabled or not

    • variants string[]

      The list of variants the parent feature should resolve to. Only valid when feature is enabled.

    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/context.html b/reference/api/unleash/context.html index 076dd2f020..a5361669ab 100644 --- a/reference/api/unleash/context.html +++ b/reference/api/unleash/context.html @@ -20,15 +20,15 @@ - - + +
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/create-addon.html b/reference/api/unleash/create-addon.html index b9bfb4a3a1..d11cec8c4a 100644 --- a/reference/api/unleash/create-addon.html +++ b/reference/api/unleash/create-addon.html @@ -20,8 +20,8 @@ - - + + @@ -37,7 +37,7 @@
  • webhook for webhooks
  • The provider you choose for your addon dictates what properties the parameters object needs. Refer to the documentation for each provider for more information.

  • description string

    A description of the addon.

  • enabled boolean required

    Whether the addon should be enabled or not.

  • parameters objectrequired

    Parameters for the addon provider. This object has different required and optional properties depending on the provider you choose. Consult the documentation for details.

  • events string[] required

    The event types that will trigger this specific addon.

  • projects string[]

    The projects that this addon will listen to events from. An empty list means it will listen to events from all projects.

  • environments string[]

    The list of environments that this addon will listen to events from. An empty list means it will listen to events from all environments.

  • Responses

    addonSchema

    Schema
    • id integer required

      Possible values: >= 1

      The addon's unique identifier.

    • provider string required

      The addon provider, such as "webhook" or "slack".

    • description string nullable required

      A description of the addon. null if no description exists.

    • enabled boolean required

      Whether the addon is enabled or not.

    • parameters objectrequired

      Parameters for the addon provider. This object has different required and optional properties depending on the provider you choose.

    • events string[] required

      The event types that trigger this specific addon.

    • projects string[]

      The projects that this addon listens to events from. An empty list means it listens to events from all projects.

    • environments string[]

      The list of environments that this addon listens to events from. An empty list means it listens to events from all environments.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/create-api-token.html b/reference/api/unleash/create-api-token.html index 2cde8295d6..702a126667 100644 --- a/reference/api/unleash/create-api-token.html +++ b/reference/api/unleash/create-api-token.html @@ -20,15 +20,15 @@ - - + +

    Create API token

    POST /api/admin/api-tokens

    Create an API token of a specific type: one of client, admin, frontend.

    Request

    Body

    required

    createApiTokenSchema

      oneOf
    • expiresAt date-time

      The time when this token should expire.

    • type string required

      Possible values: Value must match regular expression ^[Aa][Dd][Mm][Ii][Nn]$

      An admin token. Must be the string "admin" (not case sensitive).

    • tokenName string required

      The name of the token.

    Responses

    The resource was successfully created.

    Response Headers
    • location string

      The location of the newly created resource.

    Schema
    • secret string required

      The token used for authentication.

    • username string deprecated

      This property was deprecated in Unleash v5. Prefer the tokenName property instead.

    • tokenName string required

      A unique name for this particular token

    • type string required

      Possible values: [client, admin, frontend]

      The type of API token

    • environment string

      The environment the token has access to. * if it has access to all environments.

    • project string required

      The project this token belongs to.

    • projects string[] required

      The list of projects this token has access to. If the token has access to specific projects they will be listed here. If the token has access to all projects it will be represented as [*]

    • expiresAt date-time nullable

      The token's expiration date. NULL if the token doesn't have an expiration set.

    • createdAt date-time required

      When the token was created.

    • seenAt date-time nullable

      When the token was last seen/used to authenticate with. NULL if the token has not yet been used for authentication.

    • alias string nullable

      Alias is no longer in active use and will often be NULL. It's kept around as a way of allowing old proxy tokens created with the old metadata format to keep working.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/create-application.html b/reference/api/unleash/create-application.html index 611cb74980..021683cd54 100644 --- a/reference/api/unleash/create-application.html +++ b/reference/api/unleash/create-application.html @@ -20,15 +20,15 @@ - - + +

    Create an application to connect reported metrics

    POST /api/admin/metrics/applications/:appName

    Is used to report usage as well which sdk the application uses

    Request

    Path Parameters

    • appName string required

    Body

    required

    createApplicationSchema

    • appName string

      Name of the application

    • sdkVersion string

      Which SDK and version the application reporting uses. Typically represented as <identifier>:<version>

    • strategies string[]

      Which strategies the application has loaded. Useful when trying to figure out if your custom strategy has been loaded in the SDK

    • url string

      A link to reference the application reporting the metrics. Could for instance be a GitHub link to the repository of the application

    • color string

      Css color to be used to color the application's entry in the application list

    • icon string

      An URL to an icon file to be used for the applications's entry in the application list

    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/create-banner.html b/reference/api/unleash/create-banner.html index 2edf215f33..ee9ab134d8 100644 --- a/reference/api/unleash/create-banner.html +++ b/reference/api/unleash/create-banner.html @@ -20,15 +20,15 @@ - - + +

    Create a banner.

    POST /api/admin/banners

    Creates a new banner.

    Request

    Body

    required

    createBannerSchema

    • message string required

      The message to display to all users. Supports markdown.

    • enabled boolean

      Whether the banner should be displayed currently. If not specified, defaults to true.

    • variant string

      The variant of the banner. One of "info", "warning", "error", or "success". If not specified, defaults to "info".

    • sticky boolean

      Whether the banner should be sticky on the screen. If not specified, defaults to false.

    • icon string nullable

      The icon to display on the banner. Can be one of https://fonts.google.com/icons. If not specified, this will be the default icon for the variant. If "none", no icon will be displayed.

    • link string nullable

      The link to display on the banner. Can either be an absolute or a relative link (e.g. absolute: "https://example.com" or relative: "/admin/service-accounts"). If "dialog", will display a dialog when clicked. If not specified, no link will be displayed.

    • linkText string nullable

      The text to display on the link. If not specified, will be displayed as "More info".

    • dialogTitle string nullable

      The title to display on the dialog. If not specified, this will be the same as linkText.

    • dialog string nullable

      The markdown to display on the dialog. If not specified, no dialog will be displayed.

    Responses

    The resource was successfully created.

    Response Headers
    • location string

      The location of the newly created resource.

    Schema
    • id integer required

      Possible values: >= 1

      The banner's ID. Banner IDs are incrementing integers. In other words, a more recently created banner will always have a higher ID than an older one.

    • message string required

      The message to display to all users. Supports markdown.

    • enabled boolean

      Whether the banner should be displayed currently. If not specified, defaults to true.

    • variant string

      The variant of the banner. One of "info", "warning", "error", or "success". If not specified, defaults to "info".

    • sticky boolean

      Whether the banner should be sticky on the screen. If not specified, defaults to false.

    • icon string nullable

      The icon to display on the banner. Can be one of https://fonts.google.com/icons. If not specified, this will be the default icon for the variant. If "none", no icon will be displayed.

    • link string nullable

      The link to display on the banner. Can either be an absolute or a relative link (e.g. absolute: "https://example.com" or relative: "/admin/service-accounts"). If "dialog", will display a dialog when clicked. If not specified, no link will be displayed.

    • linkText string nullable

      The text to display on the link. If not specified, will be displayed as "More info".

    • dialogTitle string nullable

      The title to display on the dialog. If not specified, this will be the same as linkText.

    • dialog string nullable

      The markdown to display on the dialog. If not specified, no dialog will be displayed.

    • createdAt date-time required

      The date and time of when the banner was created.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/create-context-field.html b/reference/api/unleash/create-context-field.html index 2cda74e134..bd16e4e107 100644 --- a/reference/api/unleash/create-context-field.html +++ b/reference/api/unleash/create-context-field.html @@ -20,15 +20,15 @@ - - + +

    Create a context field

    POST /api/admin/context

    Endpoint that allows creation of custom context fields

    Request

    Body

    required

    createContextFieldSchema

    • description string

      A description of the context field

    • stickiness boolean

      true if this field should be available for use with custom stickiness, otherwise false

    • sortOrder integer

      How this context field should be sorted if no other sort order is selected

    • legalValues object[]

      A list of allowed values for this context field

    • Array [
    • value string required

      The valid value

    • description string

      Describes this specific legal value

    • ]
    • name string required

      The name of the context field.

    Responses

    The resource was successfully created.

    Response Headers
    • location string

      The location of the newly created resource.

    Schema
    • name string required

      The name of the context field

    • description string nullable

      The description of the context field.

    • stickiness boolean

      Does this context field support being used for stickiness calculations

    • sortOrder integer

      Used when sorting a list of context fields. Is also used as a tiebreaker if a list of context fields is sorted alphabetically.

    • createdAt date-time nullable

      When this context field was created

    • usedInFeatures integer nullable

      Number of projects where this context field is used in

    • usedInProjects integer nullable

      Number of projects where this context field is used in

    • legalValues object[]

      Allowed values for this context field schema. Can be used to narrow down accepted input

    • Array [
    • value string required

      The valid value

    • description string

      Describes this specific legal value

    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/create-environment.html b/reference/api/unleash/create-environment.html index cae6794c8b..fd0a7e2554 100644 --- a/reference/api/unleash/create-environment.html +++ b/reference/api/unleash/create-environment.html @@ -20,8 +20,8 @@ - - + + @@ -35,7 +35,7 @@
  • production
  • If you pass a string that is not one of the recognized values, Unleash will accept it, but it will carry no special semantics.

  • enabled boolean

    Newly created environments are enabled by default. Set this property to false to create the environment in a disabled state.

  • sortOrder integer

    Defines where in the list of environments to place this environment. The list uses an ascending sort, so lower numbers are shown first. You can change this value later.

  • Responses

    The resource was successfully created.

    Response Headers
    • location string

      The location of the newly created resource.

    Schema
    • name string required

      The name of the environment

    • type string required
    • enabled boolean required

      true if the environment is enabled for the project, otherwise false.

    • protected boolean required

      true if the environment is protected, otherwise false. A protected environment can not be deleted.

    • sortOrder integer required

      Priority of the environment in a list of environments, the lower the value, the higher up in the list the environment will appear. Needs to be an integer

    • projectCount integer nullable

      The number of projects with this environment

    • apiTokenCount integer nullable

      The number of API tokens for the project environment

    • enabledToggleCount integer nullable

      The number of enabled toggles for the project environment

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/create-feature.html b/reference/api/unleash/create-feature.html index 437473d0cf..1d9df9bc41 100644 --- a/reference/api/unleash/create-feature.html +++ b/reference/api/unleash/create-feature.html @@ -20,15 +20,15 @@ - - + +

    Add a new feature toggle

    POST /api/admin/projects/:projectId/features

    Create a new feature toggle in a specified project.

    Request

    Path Parameters

    • projectId string required

    Body

    required

    createFeatureSchema

    • name string required

      Unique feature name

    • type string

      The feature toggle's type. One of experiment, kill-switch, release, operational, or permission

    • description string nullable

      Detailed description of the feature

    • impressionData boolean

      true if the impression data collection is enabled for the feature, otherwise false.

    Responses

    featureSchema

    Schema
    • name string required

      Unique feature name

    • type string

      Type of the toggle e.g. experiment, kill-switch, release, operational, permission

    • description string nullable

      Detailed description of the feature

    • archived boolean

      true if the feature is archived

    • project string

      Name of the project the feature belongs to

    • enabled boolean

      true if the feature is enabled, otherwise false.

    • stale boolean

      true if the feature is stale based on the age and feature type, otherwise false.

    • favorite boolean

      true if the feature was favorited, otherwise false.

    • impressionData boolean

      true if the impression data collection is enabled for the feature, otherwise false.

    • createdAt date-time nullable

      The date the feature was created

    • archivedAt date-time nullable

      The date the feature was archived

    • lastSeenAt date-time nullable deprecated

      The date when metrics where last collected for the feature. This field is deprecated, use the one in featureEnvironmentSchema

    • environments object[]

      The list of environments where the feature can be used

    • Array [
    • name string required

      The name of the environment

    • featureName string

      The name of the feature

    • environment string

      The name of the environment

    • type string

      The type of the environment

    • enabled boolean required

      true if the feature is enabled for the environment, otherwise false.

    • sortOrder number

      The sort order of the feature environment in the feature environments list

    • variantCount number

      The number of defined variants

    • strategies object[]

      A list of activation strategies for the feature environment

    • Array [
    • id string

      A uuid for the feature strategy

    • name string required

      The name or type of strategy

    • title string nullable

      A descriptive title for the strategy

    • disabled boolean nullable

      A toggle to disable the strategy. defaults to false. Disabled strategies are not evaluated or returned to the SDKs

    • featureName string

      The name or feature the strategy is attached to

    • sortOrder number

      The order of the strategy in the list

    • segments number[]

      A list of segment ids attached to the strategy

    • constraints object[]

      A list of the constraints attached to the strategy. See https://docs.getunleash.io/reference/strategy-constraints

    • Array [
    • contextName string required

      The name of the context field that this constraint should apply to.

    • operator string required

      Possible values: [NOT_IN, IN, STR_ENDS_WITH, STR_STARTS_WITH, STR_CONTAINS, NUM_EQ, NUM_GT, NUM_GTE, NUM_LT, NUM_LTE, DATE_AFTER, DATE_BEFORE, SEMVER_EQ, SEMVER_GT, SEMVER_LT]

      The operator to use when evaluating this constraint. For more information about the various operators, refer to the strategy constraint operator documentation.

    • caseInsensitive boolean

      Default value: false

      Whether the operator should be case sensitive or not. Defaults to false (being case sensitive).

    • inverted boolean

      Default value: false

      Whether the result should be negated or not. If true, will turn a true result into a false result and vice versa.

    • values string[]

      The context values that should be used for constraint evaluation. Use this property instead of value for properties that accept multiple values.

    • value string

      The context value that should be used for constraint evaluation. Use this property instead of values for properties that only accept single values.

    • ]
    • variants object[]

      Strategy level variants

    • Array [
    • name string required

      The variant name. Must be unique for this feature toggle

    • weight integer required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is an integer between 0 and 1000. See the section on variant weights for more information

    • weightType string required

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000. Refer to the variant weight documentation.

    • stickiness string required

      The stickiness to use for distribution of this variant. Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • ]
    • parameters object

      A list of parameters for a strategy

    • property name* string
    • ]
    • variants object[]

      A list of variants for the feature environment

    • Array [
    • name string required

      The variants name. Is unique for this feature toggle

    • weight number required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is a number between 0 and 1000. See the section on variant weights for more information

    • weightType string

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000

    • stickiness string

      Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • overrides object[]

      Overrides assigning specific variants to specific users. The weighting system automatically assigns users to specific groups for you, but any overrides in this list will take precedence.

    • Array [
    • contextName string required

      The name of the context field used to determine overrides

    • values string[] required

      Which values that should be overriden

    • ]
    • ]
    • lastSeenAt date-time nullable

      The date when metrics where last collected for the feature environment

    • hasStrategies boolean

      Whether the feature has any strategies defined.

    • hasEnabledStrategies boolean

      Whether the feature has any enabled strategies defined.

    • ]
    • variants object[]deprecated

      The list of feature variants

    • Array [
    • name string required

      The variants name. Is unique for this feature toggle

    • weight number required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is a number between 0 and 1000. See the section on variant weights for more information

    • weightType string

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000

    • stickiness string

      Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • overrides object[]

      Overrides assigning specific variants to specific users. The weighting system automatically assigns users to specific groups for you, but any overrides in this list will take precedence.

    • Array [
    • contextName string required

      The name of the context field used to determine overrides

    • values string[] required

      Which values that should be overriden

    • ]
    • ]
    • strategies object[] deprecated

      This is a legacy field that will be deprecated

    • tags object[]nullable

      The list of feature tags

    • Array [
    • value string required

      Possible values: >= 2 characters and <= 50 characters

      The value of the tag

    • type string required

      Possible values: >= 2 characters and <= 50 characters

      Default value: simple

      The type of the tag

    • ]
    • children string[]

      The list of child feature names. This is an experimental field and may change.

    • dependencies object[]

      The list of parent dependencies. This is an experimental field and may change.

    • Array [
    • feature string required

      The name of the parent feature

    • enabled boolean

      Whether the parent feature is enabled or not

    • variants string[]

      The list of variants the parent feature should resolve to. Only valid when feature is enabled.

    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/create-feedback.html b/reference/api/unleash/create-feedback.html index 5891646df4..71d94a5a67 100644 --- a/reference/api/unleash/create-feedback.html +++ b/reference/api/unleash/create-feedback.html @@ -20,15 +20,15 @@ - - + +

    Send Unleash feedback

    POST /api/admin/feedback

    Sends feedback gathered from the Unleash UI to the Unleash server. Must be called with a token with an identifiable user (either from being sent from the UI or from using a PAT).

    Request

    Body

    required

    feedbackCreateSchema

    • neverShow boolean

      true if the user has asked never to see this feedback questionnaire again. Defaults to false.

    • feedbackId string required

      The name of the feedback session

    Responses

    feedbackResponseSchema

    Schema
    • userId integer

      The ID of the user that gave the feedback.

    • neverShow boolean

      true if the user has asked never to see this feedback questionnaire again.

    • given date-time nullable

      When this feedback was given

    • feedbackId string

      The name of the feedback session

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/create-group.html b/reference/api/unleash/create-group.html index 7b111a6a78..5aac9f3e70 100644 --- a/reference/api/unleash/create-group.html +++ b/reference/api/unleash/create-group.html @@ -20,15 +20,15 @@ - - + +

    Create a new group

    POST /api/admin/groups

    Create a new user group for Role-Based Access Control

    Request

    Body

    required

    createGroupSchema

    • name string required

      The name of the group

    • description string nullable

      A custom description of the group

    • mappingsSSO string[]

      A list of SSO groups that should map to this Unleash group

    • rootRole number nullable

      A role id that is used as the root role for all users in this group. This can be either the id of the Viewer, Editor or Admin role.

    • users object[]

      A list of users belonging to this group

    • Array [
    • user objectrequired

      A minimal user object

    • id integer required

      The user id

    • ]
    Responses

    The resource was successfully created.

    Response Headers
    • location string

      The location of the newly created resource.

    Schema
    • id integer

      The group id

    • name string required

      The name of the group

    • description string nullable

      A custom description of the group

    • mappingsSSO string[]

      A list of SSO groups that should map to this Unleash group

    • rootRole number nullable

      A role id that is used as the root role for all users in this group. This can be either the id of the Viewer, Editor or Admin role.

    • createdBy string nullable

      A user who created this group

    • createdAt date-time nullable

      When was this group created

    • users object[]

      A list of users belonging to this group

    • Array [
    • joinedAt date-time

      The date when the user joined the group

    • createdBy string nullable

      The username of the user who added this user to this group

    • user objectrequired

      An Unleash user

    • id integer required

      The user id

    • isAPI boolean deprecated

      (Deprecated): Used internally to know which operations the user should be allowed to perform

    • name string nullable

      Name of the user

    • email string

      Email of the user

    • username string nullable

      A unique username for the user

    • imageUrl string

      URL used for the userprofile image

    • inviteLink string

      If the user is actively inviting other users, this is the link that can be shared with other users

    • loginAttempts integer

      How many unsuccessful attempts at logging in has the user made

    • emailSent boolean

      Is the welcome email sent to the user or not

    • rootRole integer

      Which root role this user is assigned

    • seenAt date-time nullable

      The last time this user logged in

    • createdAt date-time

      The user was created at this time

    • accountType string

      A user is either an actual User or a Service Account

    • permissions string[]

      Deprecated

    • ]
    • projects string[]

      A list of projects where this group is used

    • userCount integer

      The number of users that belong to this group

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/create-pat.html b/reference/api/unleash/create-pat.html index f79c0add04..d9ab19f066 100644 --- a/reference/api/unleash/create-pat.html +++ b/reference/api/unleash/create-pat.html @@ -20,15 +20,15 @@ - - + +

    Create a new Personal Access Token.

    POST /api/admin/user/tokens

    Creates a new Personal Access Token for the current user.

    Request

    Body

    required

    patSchema

    • id integer

      Possible values: >= 1

      The unique identification number for this Personal Access Token. (This property is set by Unleash when the token is created and cannot be set manually: if you provide a value when creating a PAT, Unleash will ignore it.)

    • secret string

      The token used for authentication. (This property is set by Unleash when the token is created and cannot be set manually: if you provide a value when creating a PAT, Unleash will ignore it.)

    • expiresAt date-time

      The token's expiration date.

    • createdAt date-time

      When the token was created. (This property is set by Unleash when the token is created and cannot be set manually: if you provide a value when creating a PAT, Unleash will ignore it.)

    • seenAt date-time nullable

      When the token was last seen/used to authenticate with. null if it has not been used yet. (This property is set by Unleash when the token is created and cannot be set manually: if you provide a value when creating a PAT, Unleash will ignore it.)

    Responses

    The resource was successfully created.

    Response Headers
    • location string

      The location of the newly created resource.

    Schema
    • id integer

      Possible values: >= 1

      The unique identification number for this Personal Access Token. (This property is set by Unleash when the token is created and cannot be set manually: if you provide a value when creating a PAT, Unleash will ignore it.)

    • secret string

      The token used for authentication. (This property is set by Unleash when the token is created and cannot be set manually: if you provide a value when creating a PAT, Unleash will ignore it.)

    • expiresAt date-time

      The token's expiration date.

    • createdAt date-time

      When the token was created. (This property is set by Unleash when the token is created and cannot be set manually: if you provide a value when creating a PAT, Unleash will ignore it.)

    • seenAt date-time nullable

      When the token was last seen/used to authenticate with. null if it has not been used yet. (This property is set by Unleash when the token is created and cannot be set manually: if you provide a value when creating a PAT, Unleash will ignore it.)

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/create-project-api-token.html b/reference/api/unleash/create-project-api-token.html index b1d6d23802..4445f36e7d 100644 --- a/reference/api/unleash/create-project-api-token.html +++ b/reference/api/unleash/create-project-api-token.html @@ -20,15 +20,15 @@ - - + +

    Create a project API token.

    POST /api/admin/projects/:projectId/api-tokens

    Endpoint that allows creation of project API tokens for the specified project.

    Request

    Path Parameters

    • projectId string required

    Body

    required

    createApiTokenSchema

      oneOf
    • expiresAt date-time

      The time when this token should expire.

    • type string required

      Possible values: Value must match regular expression ^[Aa][Dd][Mm][Ii][Nn]$

      An admin token. Must be the string "admin" (not case sensitive).

    • tokenName string required

      The name of the token.

    Responses

    The resource was successfully created.

    Response Headers
    • location string

      The location of the newly created resource.

    Schema
    • secret string required

      The token used for authentication.

    • username string deprecated

      This property was deprecated in Unleash v5. Prefer the tokenName property instead.

    • tokenName string required

      A unique name for this particular token

    • type string required

      Possible values: [client, admin, frontend]

      The type of API token

    • environment string

      The environment the token has access to. * if it has access to all environments.

    • project string required

      The project this token belongs to.

    • projects string[] required

      The list of projects this token has access to. If the token has access to specific projects they will be listed here. If the token has access to all projects it will be represented as [*]

    • expiresAt date-time nullable

      The token's expiration date. NULL if the token doesn't have an expiration set.

    • createdAt date-time required

      When the token was created.

    • seenAt date-time nullable

      When the token was last seen/used to authenticate with. NULL if the token has not yet been used for authentication.

    • alias string nullable

      Alias is no longer in active use and will often be NULL. It's kept around as a way of allowing old proxy tokens created with the old metadata format to keep working.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/create-project.html b/reference/api/unleash/create-project.html index bfc73b9465..fe5ab24676 100644 --- a/reference/api/unleash/create-project.html +++ b/reference/api/unleash/create-project.html @@ -20,15 +20,15 @@ - - + +

    Create project

    POST /api/admin/projects

    Create a new Unleash project.

    Request

    Body

    required

    createProjectSchema

    • id string required

      Possible values: Value must match regular expression [A-Za-z0-9_~.-]+

      The project's identifier.

    • name string required

      Possible values: non-empty

      The project's name.

    • description string nullable

      The project's description.

    • mode string

      Possible values: [open, protected, private]

      A mode of the project affecting what actions are possible in this project

    • defaultStickiness string

      A default stickiness for the project affecting the default stickiness value for variants and Gradual Rollout strategy

    Responses

    The resource was successfully created.

    Response Headers
    • location string

      The location of the newly created resource.

    Schema
    • id string required

      Possible values: Value must match regular expression [A-Za-z0-9_~.-]+

      The project's identifier.

    • name string required

      Possible values: non-empty

      The project's name.

    • description string nullable

      The project's description.

    • featureLimit integer nullable

      A limit on the number of features allowed in the project. null if no limit.

    • mode string

      Possible values: [open, protected, private]

      A mode of the project affecting what actions are possible in this project

    • defaultStickiness string

      A default stickiness for the project affecting the default stickiness value for variants and Gradual Rollout strategy

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/create-public-signup-token.html b/reference/api/unleash/create-public-signup-token.html index 51c2ebcf92..2a6c9edd44 100644 --- a/reference/api/unleash/create-public-signup-token.html +++ b/reference/api/unleash/create-public-signup-token.html @@ -20,15 +20,15 @@ - - + +

    Create a public signup token

    POST /api/admin/invite-link/tokens

    Lets administrators create a invite link to share with colleagues. People that join using the public invite are assigned the Viewer role

    Request

    Body

    required

    publicSignupTokenCreateSchema

    • name string required

      The token's name.

    • expiresAt date-time required

      The token's expiration date.

    Responses

    The resource was successfully created.

    Response Headers
    • location string

      The location of the newly created resource.

    Schema
    • secret string required

      The actual value of the token. This is the part that is used by Unleash to create an invite link

    • url string nullable required

      The public signup link for the token. Users who follow this link will be taken to a signup page where they can create an Unleash user.

    • name string required

      The token's name. Only for displaying in the UI

    • enabled boolean required

      Whether the token is active. This property will always be false for a token that has expired.

    • expiresAt date-time required

      The time when the token will expire.

    • createdAt date-time required

      When the token was created.

    • createdBy string nullable required

      The creator's email or username

    • users object[]nullable

      Array of users that have signed up using the token.

    • Array [
    • id integer required

      The user id

    • isAPI boolean deprecated

      (Deprecated): Used internally to know which operations the user should be allowed to perform

    • name string nullable

      Name of the user

    • email string

      Email of the user

    • username string nullable

      A unique username for the user

    • imageUrl string

      URL used for the userprofile image

    • inviteLink string

      If the user is actively inviting other users, this is the link that can be shared with other users

    • loginAttempts integer

      How many unsuccessful attempts at logging in has the user made

    • emailSent boolean

      Is the welcome email sent to the user or not

    • rootRole integer

      Which root role this user is assigned

    • seenAt date-time nullable

      The last time this user logged in

    • createdAt date-time

      The user was created at this time

    • accountType string

      A user is either an actual User or a Service Account

    • permissions string[]

      Deprecated

    • ]
    • role objectrequired

      A role holds permissions to allow Unleash to decide what actions a role holder is allowed to perform

    • id integer required

      The role id

    • type string required

      A role can either be a global root role (applies to all projects) or a project role

    • name string required

      The name of the role

    • description string

      A more detailed description of the role and what use it's intended for

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/create-role.html b/reference/api/unleash/create-role.html index 96482287f0..18beb58f9a 100644 --- a/reference/api/unleash/create-role.html +++ b/reference/api/unleash/create-role.html @@ -20,15 +20,15 @@ - - + +

    Create a new role

    POST /api/admin/roles

    Create a new custom role for Role-Based Access Control

    Request

    Body

    required

    createRoleWithPermissionsSchema

      anyOf
    • name string required

      The name of the custom role

    • description string

      A more detailed description of the custom role and what use it's intended for

    • type string

      Possible values: [root-custom, custom]

      Custom root roles (type=root-custom) are root roles with a custom set of permissions. Custom project roles (type=custom) contain a specific set of permissions for project resources.

    • permissions object[]

      A list of permissions assigned to this role

    • Array [
    • name string required

      The name of the permission

    • environment string

      The environments of the permission if the permission is environment specific

    • ]
    Responses

    roleWithVersionSchema

    Schema
    • version integer required

      Possible values: >= 1

      The version of this schema

    • roles objectrequired

      A role holds permissions to allow Unleash to decide what actions a role holder is allowed to perform

    • id integer required

      The role id

    • type string required

      A role can either be a global root role (applies to all projects) or a project role

    • name string required

      The name of the role

    • description string

      A more detailed description of the role and what use it's intended for

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/create-segment.html b/reference/api/unleash/create-segment.html index 16d0ac8284..2f59632404 100644 --- a/reference/api/unleash/create-segment.html +++ b/reference/api/unleash/create-segment.html @@ -20,15 +20,15 @@ - - + +

    Create a new segment

    POST /api/admin/segments

    Creates a new segment using the payload provided

    Request

    Body

    required

    upsertSegmentSchema

    • name string required

      The name of the segment

    • description string nullable

      A description of what the segment is for

    • project string nullable

      The project the segment belongs to if any.

    • constraints object[]required

      The list of constraints that make up this segment

    • Array [
    • contextName string required

      The name of the context field that this constraint should apply to.

    • operator string required

      Possible values: [NOT_IN, IN, STR_ENDS_WITH, STR_STARTS_WITH, STR_CONTAINS, NUM_EQ, NUM_GT, NUM_GTE, NUM_LT, NUM_LTE, DATE_AFTER, DATE_BEFORE, SEMVER_EQ, SEMVER_GT, SEMVER_LT]

      The operator to use when evaluating this constraint. For more information about the various operators, refer to the strategy constraint operator documentation.

    • caseInsensitive boolean

      Default value: false

      Whether the operator should be case sensitive or not. Defaults to false (being case sensitive).

    • inverted boolean

      Default value: false

      Whether the result should be negated or not. If true, will turn a true result into a false result and vice versa.

    • values string[]

      The context values that should be used for constraint evaluation. Use this property instead of value for properties that accept multiple values.

    • value string

      The context value that should be used for constraint evaluation. Use this property instead of values for properties that only accept single values.

    • ]
    Responses

    The resource was successfully created.

    Response Headers
    • location string

      The location of the newly created resource.

    Schema
    • id integer required

      The ID of this segment

    • name string required

      The name of this segment

    • description string nullable

      The description for this segment

    • constraints object[]required

      The list of constraints that are used in this segment

    • Array [
    • contextName string required

      The name of the context field that this constraint should apply to.

    • operator string required

      Possible values: [NOT_IN, IN, STR_ENDS_WITH, STR_STARTS_WITH, STR_CONTAINS, NUM_EQ, NUM_GT, NUM_GTE, NUM_LT, NUM_LTE, DATE_AFTER, DATE_BEFORE, SEMVER_EQ, SEMVER_GT, SEMVER_LT]

      The operator to use when evaluating this constraint. For more information about the various operators, refer to the strategy constraint operator documentation.

    • caseInsensitive boolean

      Default value: false

      Whether the operator should be case sensitive or not. Defaults to false (being case sensitive).

    • inverted boolean

      Default value: false

      Whether the result should be negated or not. If true, will turn a true result into a false result and vice versa.

    • values string[]

      The context values that should be used for constraint evaluation. Use this property instead of value for properties that accept multiple values.

    • value string

      The context value that should be used for constraint evaluation. Use this property instead of values for properties that only accept single values.

    • ]
    • usedInFeatures integer nullable

      The number of feature flags that use this segment. The number also includes the any flags with pending change requests that would add this segment.

    • usedInProjects integer nullable

      The number of projects that use this segment. The number includes any projects with pending change requests that would add this segment.

    • project string nullable

      The project the segment belongs to. Only present if the segment is a project-specific segment.

    • createdBy string nullable

      The creator's email or username

    • createdAt date-time required

      When the segment was created

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/create-service-account-token.html b/reference/api/unleash/create-service-account-token.html index ebc70daaa9..84c9e2b8e4 100644 --- a/reference/api/unleash/create-service-account-token.html +++ b/reference/api/unleash/create-service-account-token.html @@ -20,15 +20,15 @@ - - + +

    Create a token for a service account.

    POST /api/admin/service-account/:id/token

    Creates a new token for the service account identified by the id.

    Request

    Path Parameters

    • id string required

    Body

    required

    patSchema

    • id integer

      Possible values: >= 1

      The unique identification number for this Personal Access Token. (This property is set by Unleash when the token is created and cannot be set manually: if you provide a value when creating a PAT, Unleash will ignore it.)

    • secret string

      The token used for authentication. (This property is set by Unleash when the token is created and cannot be set manually: if you provide a value when creating a PAT, Unleash will ignore it.)

    • expiresAt date-time

      The token's expiration date.

    • createdAt date-time

      When the token was created. (This property is set by Unleash when the token is created and cannot be set manually: if you provide a value when creating a PAT, Unleash will ignore it.)

    • seenAt date-time nullable

      When the token was last seen/used to authenticate with. null if it has not been used yet. (This property is set by Unleash when the token is created and cannot be set manually: if you provide a value when creating a PAT, Unleash will ignore it.)

    Responses

    The resource was successfully created.

    Response Headers
    • location string

      The location of the newly created resource.

    Schema
    • id integer

      Possible values: >= 1

      The unique identification number for this Personal Access Token. (This property is set by Unleash when the token is created and cannot be set manually: if you provide a value when creating a PAT, Unleash will ignore it.)

    • secret string

      The token used for authentication. (This property is set by Unleash when the token is created and cannot be set manually: if you provide a value when creating a PAT, Unleash will ignore it.)

    • expiresAt date-time

      The token's expiration date.

    • createdAt date-time

      When the token was created. (This property is set by Unleash when the token is created and cannot be set manually: if you provide a value when creating a PAT, Unleash will ignore it.)

    • seenAt date-time nullable

      When the token was last seen/used to authenticate with. null if it has not been used yet. (This property is set by Unleash when the token is created and cannot be set manually: if you provide a value when creating a PAT, Unleash will ignore it.)

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/create-service-account.html b/reference/api/unleash/create-service-account.html index da52928d32..53fc71539c 100644 --- a/reference/api/unleash/create-service-account.html +++ b/reference/api/unleash/create-service-account.html @@ -20,15 +20,15 @@ - - + +

    Create a service account.

    POST /api/admin/service-account

    Creates a new service account.

    Request

    Body

    required

    createServiceAccountSchema

    • username string

      The username of the service account

    • name string

      The name of the service account

    • rootRole integer required

      The id of the root role for the service account

    Responses

    The resource was successfully created.

    Response Headers
    • location string

      The location of the newly created resource.

    Schema
    • id number required

      The service account id

    • isAPI boolean deprecated

      Deprecated: for internal use only, should not be exposed through the API

    • name string

      The name of the service account

    • email string deprecated

      Deprecated: service accounts don't have emails associated with them

    • username string

      The service account username

    • imageUrl string

      The service account image url

    • inviteLink string deprecated

      Deprecated: service accounts cannot be invited via an invitation link

    • loginAttempts number deprecated

      Deprecated: service accounts cannot log in to Unleash

    • emailSent boolean deprecated

      Deprecated: internal use only

    • rootRole integer

      The root role id associated with the service account

    • seenAt date-time nullable deprecated

      Deprecated. This property is always null. To find out when a service account was last seen, check its tokens list and refer to each token's lastSeen property instead.

    • createdAt date-time

      The service account creation date

    • tokens object[]

      The list of tokens associated with the service account

    • Array [
    • id integer

      Possible values: >= 1

      The unique identification number for this Personal Access Token. (This property is set by Unleash when the token is created and cannot be set manually: if you provide a value when creating a PAT, Unleash will ignore it.)

    • secret string

      The token used for authentication. (This property is set by Unleash when the token is created and cannot be set manually: if you provide a value when creating a PAT, Unleash will ignore it.)

    • expiresAt date-time

      The token's expiration date.

    • createdAt date-time

      When the token was created. (This property is set by Unleash when the token is created and cannot be set manually: if you provide a value when creating a PAT, Unleash will ignore it.)

    • seenAt date-time nullable

      When the token was last seen/used to authenticate with. null if it has not been used yet. (This property is set by Unleash when the token is created and cannot be set manually: if you provide a value when creating a PAT, Unleash will ignore it.)

    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/create-strategy.html b/reference/api/unleash/create-strategy.html index daa4a50d9c..87c91ff4d4 100644 --- a/reference/api/unleash/create-strategy.html +++ b/reference/api/unleash/create-strategy.html @@ -20,15 +20,15 @@ - - + +

    Create a strategy

    POST /api/admin/strategies

    Creates a strategy type based on the supplied data.

    Request

    Body

    required

    createStrategySchema

    • name string required

      The name of the strategy type. Must be unique.

    • title string

      The title of the strategy

    • description string

      A description of the strategy type.

    • editable boolean

      Whether the strategy type is editable or not. Defaults to true.

    • deprecated boolean

      Whether the strategy type is deprecated or not. Defaults to false.

    • parameters object[]required

      The parameter list lets you pass arguments to your custom activation strategy. These will be made available to your custom strategy implementation.

    • Array [
    • name string required

      The name of the parameter

    • type string required

      Possible values: [string, percentage, list, number, boolean]

    • description string

      A description of this strategy parameter. Use this to indicate to the users what the parameter does.

    • required boolean

      Whether this parameter must be configured when using the strategy. Defaults to false

    • ]
    Responses

    The resource was successfully created.

    Response Headers
    • location string

      The location of the newly created resource.

    Schema
    • title string nullable

      An optional title for the strategy

    • name string required

      The name (type) of the strategy

    • displayName string nullable required

      A human friendly name for the strategy

    • description string nullable required

      A short description of the strategy

    • editable boolean required

      Whether the strategy can be edited or not. Strategies bundled with Unleash cannot be edited.

    • deprecated boolean required
    • parameters object[]required

      A list of relevant parameters for each strategy

    • Array [
    • name string
    • type string
    • description string
    • required boolean
    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/create-tag-type.html b/reference/api/unleash/create-tag-type.html index 1510c04410..b568ba5a1f 100644 --- a/reference/api/unleash/create-tag-type.html +++ b/reference/api/unleash/create-tag-type.html @@ -20,15 +20,15 @@ - - + +

    Create a tag type

    POST /api/admin/tag-types

    Create a new tag type.

    Request

    Body

    required

    tagTypeSchema

    • name string required

      The name of the tag type.

    • description string

      The description of the tag type.

    • icon string nullable

      The icon of the tag type.

    Responses

    The resource was successfully created.

    Response Headers
    • location string

      The location of the newly created resource.

    Schema
    • name string required

      The name of the tag type.

    • description string

      The description of the tag type.

    • icon string nullable

      The icon of the tag type.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/create-tag.html b/reference/api/unleash/create-tag.html index 3424a58696..8a22164247 100644 --- a/reference/api/unleash/create-tag.html +++ b/reference/api/unleash/create-tag.html @@ -20,15 +20,15 @@ - - + +

    Create a new tag.

    POST /api/admin/tags

    Create a new tag with the specified data.

    Request

    Body

    required

    tagSchema

    • value string required

      Possible values: >= 2 characters and <= 50 characters

      The value of the tag

    • type string required

      Possible values: >= 2 characters and <= 50 characters

      Default value: simple

      The type of the tag

    Responses

    The resource was successfully created.

    Response Headers
    • location string

      The location of the newly created resource.

    Schema
    • version integer required

      The version of the schema used to model the tag.

    • tag objectrequired

      Representation of a tag

    • value string required

      Possible values: >= 2 characters and <= 50 characters

      The value of the tag

    • type string required

      Possible values: >= 2 characters and <= 50 characters

      Default value: simple

      The type of the tag

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/create-user.html b/reference/api/unleash/create-user.html index 532e6f4fa2..8b4ad5c9c8 100644 --- a/reference/api/unleash/create-user.html +++ b/reference/api/unleash/create-user.html @@ -20,15 +20,15 @@ - - + +

    Create a new user

    POST /api/admin/user-admin

    Creates a new user with the given root role.

    Request

    Body

    required

    createUserSchema

    • username string

      The user's username. Must be provided if email is not provided.

    • email string

      The user's email address. Must be provided if username is not provided.

    • name string

      The user's name (not the user's username).

    • password string

      Password for the user

    • rootRole objectrequired

      The role to assign to the user. Can be either the role's ID or its unique name.

      oneOf
    • integer
    • sendEmail boolean

      Whether to send a welcome email with a login link to the user or not. Defaults to true.

    Responses

    The resource was successfully created.

    Response Headers
    • location string

      The location of the newly created resource.

    Schema
    • id integer required

      The user id

    • isAPI boolean deprecated

      (Deprecated): Used internally to know which operations the user should be allowed to perform

    • name string nullable

      Name of the user

    • email string

      Email of the user

    • username string nullable

      A unique username for the user

    • imageUrl string

      URL used for the userprofile image

    • inviteLink string

      If the user is actively inviting other users, this is the link that can be shared with other users

    • loginAttempts integer

      How many unsuccessful attempts at logging in has the user made

    • emailSent boolean

      Is the welcome email sent to the user or not

    • rootRole object

      Which root role this user is assigned. Usually a numeric role ID, but can be a string when returning newly created user with an explicit string role.

      oneOf
    • integer
    • seenAt date-time nullable

      The last time this user logged in

    • createdAt date-time

      The user was created at this time

    • accountType string

      A user is either an actual User or a Service Account

    • permissions string[]

      Deprecated

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/delete-addon.html b/reference/api/unleash/delete-addon.html index f348722231..1fa06f0407 100644 --- a/reference/api/unleash/delete-addon.html +++ b/reference/api/unleash/delete-addon.html @@ -20,15 +20,15 @@ - - + +

    Delete an addon

    DELETE /api/admin/addons/:id

    Delete the addon specified by the ID in the request path.

    Request

    Path Parameters

    • id string required
    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/delete-api-token.html b/reference/api/unleash/delete-api-token.html index cbf9413c5c..38cf05e143 100644 --- a/reference/api/unleash/delete-api-token.html +++ b/reference/api/unleash/delete-api-token.html @@ -20,15 +20,15 @@ - - + +

    Delete API token

    DELETE /api/admin/api-tokens/:token

    Deletes an existing API token. The token path parameter is the token's secret. If the token does not exist, this endpoint returns a 200 OK, but does nothing.

    Request

    Path Parameters

    • token string required
    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/delete-application.html b/reference/api/unleash/delete-application.html index 917531b7af..22ff2b9cef 100644 --- a/reference/api/unleash/delete-application.html +++ b/reference/api/unleash/delete-application.html @@ -20,15 +20,15 @@ - - + +

    Delete an application

    DELETE /api/admin/metrics/applications/:appName

    Delete the application specified in the request URL. Returns 200 OK if the application was successfully deleted or if it didn't exist

    Request

    Path Parameters

    • appName string required
    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/delete-banner.html b/reference/api/unleash/delete-banner.html index 516ad62803..6d95a71282 100644 --- a/reference/api/unleash/delete-banner.html +++ b/reference/api/unleash/delete-banner.html @@ -20,15 +20,15 @@ - - + +

    Delete a banner.

    DELETE /api/admin/banners/:id

    Deletes an existing banner identified by its id.

    Request

    Path Parameters

    • id string required
    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/delete-change-request.html b/reference/api/unleash/delete-change-request.html index 6a7a7d2bdc..5f5edd07f8 100644 --- a/reference/api/unleash/delete-change-request.html +++ b/reference/api/unleash/delete-change-request.html @@ -20,15 +20,15 @@ - - + +

    Deletes a change request by id

    DELETE /api/admin/projects/:projectId/change-requests/:id

    This endpoint will delete one change request if it matches the provided id.

    Request

    Path Parameters

    • projectId string required
    • id string required
    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/delete-change.html b/reference/api/unleash/delete-change.html index c72da25c41..cb2f2653b2 100644 --- a/reference/api/unleash/delete-change.html +++ b/reference/api/unleash/delete-change.html @@ -20,15 +20,15 @@ - - + +

    Discards a change from a change request by change id

    DELETE /api/admin/projects/:projectId/change-requests/:changeRequestId/changes/:changeId

    This endpoint will discard one change from a change request if it matches the provided id.

    Request

    Path Parameters

    • projectId string required
    • changeRequestId string required
    • changeId string required
    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/delete-context-field.html b/reference/api/unleash/delete-context-field.html index 4550c3cb38..6d77598849 100644 --- a/reference/api/unleash/delete-context-field.html +++ b/reference/api/unleash/delete-context-field.html @@ -20,15 +20,15 @@ - - + +
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/delete-feature-dependencies.html b/reference/api/unleash/delete-feature-dependencies.html index 36a2ec8830..3dee548530 100644 --- a/reference/api/unleash/delete-feature-dependencies.html +++ b/reference/api/unleash/delete-feature-dependencies.html @@ -20,15 +20,15 @@ - - + +

    Deletes feature dependencies.

    DELETE /api/admin/projects/:projectId/features/:child/dependencies

    Remove dependencies to all parent features.

    Request

    Path Parameters

    • projectId string required
    • child string required
    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/delete-feature-dependency.html b/reference/api/unleash/delete-feature-dependency.html index c32afd2708..11ea8d397e 100644 --- a/reference/api/unleash/delete-feature-dependency.html +++ b/reference/api/unleash/delete-feature-dependency.html @@ -20,15 +20,15 @@ - - + +

    Deletes a feature dependency.

    DELETE /api/admin/projects/:projectId/features/:child/dependencies/:parent

    Remove a dependency to a parent feature.

    Request

    Path Parameters

    • projectId string required
    • child string required
    • parent string required
    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/delete-feature-strategy.html b/reference/api/unleash/delete-feature-strategy.html index 5eedfa5d17..e87cd13f68 100644 --- a/reference/api/unleash/delete-feature-strategy.html +++ b/reference/api/unleash/delete-feature-strategy.html @@ -20,15 +20,15 @@ - - + +

    Delete a strategy from a feature toggle

    DELETE /api/admin/projects/:projectId/features/:featureName/environments/:environment/strategies/:strategyId

    Delete a strategy configuration from a feature toggle in the specified environment.

    Request

    Path Parameters

    • projectId string required
    • featureName string required
    • environment string required
    • strategyId string required
    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/delete-feature.html b/reference/api/unleash/delete-feature.html index 5e68056077..6645434031 100644 --- a/reference/api/unleash/delete-feature.html +++ b/reference/api/unleash/delete-feature.html @@ -20,15 +20,15 @@ - - + +

    Archives a feature

    DELETE /api/admin/archive/:featureName

    This endpoint archives the specified feature.

    Request

    Path Parameters

    • featureName string required
    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/delete-features.html b/reference/api/unleash/delete-features.html index 01fd5d8537..bd549c47aa 100644 --- a/reference/api/unleash/delete-features.html +++ b/reference/api/unleash/delete-features.html @@ -20,15 +20,15 @@ - - + +

    Deletes a list of features

    POST /api/admin/projects/:projectId/delete

    This endpoint deletes the specified features, that are in archive.

    Request

    Path Parameters

    • projectId string required

    Body

    required

    batchFeaturesSchema

    • features string[] required

      List of feature toggle names

    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/delete-group.html b/reference/api/unleash/delete-group.html index 527d4d20a2..b969b08091 100644 --- a/reference/api/unleash/delete-group.html +++ b/reference/api/unleash/delete-group.html @@ -20,15 +20,15 @@ - - + +

    Delete a single group

    DELETE /api/admin/groups/:groupId

    Delete a single user group by group id

    Request

    Path Parameters

    • groupId string required
    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/delete-pat.html b/reference/api/unleash/delete-pat.html index 4601e01e2a..d2361cf4cb 100644 --- a/reference/api/unleash/delete-pat.html +++ b/reference/api/unleash/delete-pat.html @@ -20,15 +20,15 @@ - - + +

    Delete a Personal Access Token.

    DELETE /api/admin/user/tokens/:id

    This endpoint allows for deleting a Personal Access Token belonging to the current user.

    Request

    Path Parameters

    • id string required
    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/delete-project-api-token.html b/reference/api/unleash/delete-project-api-token.html index a1f6f9c4d2..e6461edf05 100644 --- a/reference/api/unleash/delete-project-api-token.html +++ b/reference/api/unleash/delete-project-api-token.html @@ -20,15 +20,15 @@ - - + +

    Delete a project API token.

    DELETE /api/admin/projects/:projectId/api-tokens/:token

    This operation deletes the API token specified in the request URL. If the token doesn't exist, returns an OK response (status code 200).

    Request

    Path Parameters

    • projectId string required
    • token string required
    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/delete-project.html b/reference/api/unleash/delete-project.html index b93c3d6431..b073cc0034 100644 --- a/reference/api/unleash/delete-project.html +++ b/reference/api/unleash/delete-project.html @@ -20,15 +20,15 @@ - - + +

    Delete project

    DELETE /api/admin/projects/:projectId

    Permanently delete the provided project. All feature toggles in the project must be archived before you can delete it. This permanently deletes the project and its archived toggles. It can not be undone.

    Request

    Path Parameters

    • projectId string required
    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/delete-role.html b/reference/api/unleash/delete-role.html index 6b96601214..cfd0d05536 100644 --- a/reference/api/unleash/delete-role.html +++ b/reference/api/unleash/delete-role.html @@ -20,15 +20,15 @@ - - + +

    Delete a custom role

    DELETE /api/admin/roles/:roleId

    Delete a custom role by id. You cannot delete built-in roles or roles that are in use.

    Request

    Path Parameters

    • roleId string required
    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/delete-service-account-token.html b/reference/api/unleash/delete-service-account-token.html index cec656d14c..e9849bda86 100644 --- a/reference/api/unleash/delete-service-account-token.html +++ b/reference/api/unleash/delete-service-account-token.html @@ -20,15 +20,15 @@ - - + +

    Delete a token for a service account.

    DELETE /api/admin/service-account/:id/token/:tokenId

    Deletes a token for the service account identified both by the service account's id and the token's id.

    Request

    Path Parameters

    • id string required
    • tokenId string required
    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/delete-service-account.html b/reference/api/unleash/delete-service-account.html index 2b7ffad9c8..3036d03805 100644 --- a/reference/api/unleash/delete-service-account.html +++ b/reference/api/unleash/delete-service-account.html @@ -20,15 +20,15 @@ - - + +

    Delete a service account.

    DELETE /api/admin/service-account/:id

    Deletes an existing service account identified by its id.

    Request

    Path Parameters

    • id string required
    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/delete-tag-type.html b/reference/api/unleash/delete-tag-type.html index 1253e57221..2759847c62 100644 --- a/reference/api/unleash/delete-tag-type.html +++ b/reference/api/unleash/delete-tag-type.html @@ -20,15 +20,15 @@ - - + +

    Delete a tag type

    DELETE /api/admin/tag-types/:name

    Deletes a tag type. If any features have tags of this type, those tags will be deleted.

    Request

    Path Parameters

    • name string required
    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/delete-tag.html b/reference/api/unleash/delete-tag.html index 40f8e15c1c..4433c4e058 100644 --- a/reference/api/unleash/delete-tag.html +++ b/reference/api/unleash/delete-tag.html @@ -20,15 +20,15 @@ - - + +
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/delete-user.html b/reference/api/unleash/delete-user.html index 0b3589fcf6..763ba2dd42 100644 --- a/reference/api/unleash/delete-user.html +++ b/reference/api/unleash/delete-user.html @@ -20,15 +20,15 @@ - - + +

    Delete a user

    DELETE /api/admin/user-admin/:id

    Deletes the user with the given userId

    Request

    Path Parameters

    • id string required
    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/dependencies.html b/reference/api/unleash/dependencies.html index f814a27b69..c3fe42e8c0 100644 --- a/reference/api/unleash/dependencies.html +++ b/reference/api/unleash/dependencies.html @@ -20,15 +20,15 @@ - - + +
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/deprecate-strategy.html b/reference/api/unleash/deprecate-strategy.html index aa1e19a3f5..a2f2a50098 100644 --- a/reference/api/unleash/deprecate-strategy.html +++ b/reference/api/unleash/deprecate-strategy.html @@ -20,15 +20,15 @@ - - + +

    Deprecate a strategy

    POST /api/admin/strategies/:strategyName/deprecate

    Marks the specified strategy as deprecated.

    Request

    Path Parameters

    • strategyName string required
    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/disable-banner.html b/reference/api/unleash/disable-banner.html index ebf0570128..a3e868904a 100644 --- a/reference/api/unleash/disable-banner.html +++ b/reference/api/unleash/disable-banner.html @@ -20,15 +20,15 @@ - - + +

    Disables a banner.

    POST /api/admin/banners/:id/off

    Disables an existing banner, identified by its id.

    Request

    Path Parameters

    • id string required
    Responses

    bannerSchema

    Schema
    • id integer required

      Possible values: >= 1

      The banner's ID. Banner IDs are incrementing integers. In other words, a more recently created banner will always have a higher ID than an older one.

    • message string required

      The message to display to all users. Supports markdown.

    • enabled boolean

      Whether the banner should be displayed currently. If not specified, defaults to true.

    • variant string

      The variant of the banner. One of "info", "warning", "error", or "success". If not specified, defaults to "info".

    • sticky boolean

      Whether the banner should be sticky on the screen. If not specified, defaults to false.

    • icon string nullable

      The icon to display on the banner. Can be one of https://fonts.google.com/icons. If not specified, this will be the default icon for the variant. If "none", no icon will be displayed.

    • link string nullable

      The link to display on the banner. Can either be an absolute or a relative link (e.g. absolute: "https://example.com" or relative: "/admin/service-accounts"). If "dialog", will display a dialog when clicked. If not specified, no link will be displayed.

    • linkText string nullable

      The text to display on the link. If not specified, will be displayed as "More info".

    • dialogTitle string nullable

      The title to display on the dialog. If not specified, this will be the same as linkText.

    • dialog string nullable

      The markdown to display on the dialog. If not specified, no dialog will be displayed.

    • createdAt date-time required

      The date and time of when the banner was created.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/edge.html b/reference/api/unleash/edge.html index 58dc3ce500..788a476b05 100644 --- a/reference/api/unleash/edge.html +++ b/reference/api/unleash/edge.html @@ -20,15 +20,15 @@ - - + +
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/edit-change.html b/reference/api/unleash/edit-change.html index c1e9a090d1..fea81d75b5 100644 --- a/reference/api/unleash/edit-change.html +++ b/reference/api/unleash/edit-change.html @@ -20,8 +20,8 @@ - - + + @@ -29,7 +29,7 @@

    Edits a single change in a change request

    PUT /api/admin/projects/:projectId/change-requests/:changeRequestId/changes/:changeId

    This endpoint will edit one change from a change request if it matches the provided id. The edit removes previous change and inserts a new one. You should not rely on the changeId for subsequent edits and always check the most recent changeId.

    Request

    Path Parameters

    • projectId string required
    • changeRequestId string required
    • changeId string required

    Body

    required

    changeRequestCreateSchema

      oneOf
    • action string required

      Possible values: [updateSegment]

      The name of this action.

    • payload objectrequired

      Data used to create or update a segment

    • id integer required

      The ID of the segment to update.

    Responses

    changeRequestSchema

    Schema
      oneOf
    • id number required

      This change requests's ID.

    • title string

      A title describing the change request's content.

    • environment string required

      The environment in which the changes should be applied.

    • minApprovals number required

      The minimum number of approvals required before this change request can be applied.

    • project string required

      The project this change request belongs to.

    • features object[]required

      The list of features and their changes that relate to this change request.

    • Array [
    • name string required

      The name of the feature

    • conflict string

      A string describing the conflicts related to this change. Only present if there are any concflicts on the feature level.

    • changes object[]required

      List of changes inside change request. This list may be empty when listing all change requests for a project.

    • Array [
    • id number required

      The ID of this change.

    • action string required

      The kind of action that the change contains information about.

    • conflict string

      A description of the conflict caused by this change. Only present if there are any conflicts.

    • payload objectrequired

      The data required to perform this action.

      oneOf
    • string
    • createdBy object

      The user who created this change.

    • username string nullable

      The user's username.

    • imageUrl uri nullable

      The URL where the user's image can be found.

    • createdAt date-time

      When this change was suggested

    • ]
    • defaultChange object

      A description of a default change that will be applied with the change request to prevent invalid states.

      Default changes are changes that are applied in addition to explicit user-specified changes when a change request is applied. Any default changes are applied in the background and are not a real part of the change request.

    • action string required

      The kind of action this is.

    • payload object required

      The necessary data to perform this change.

    • ]
    • segments object[]required

      The list of segments and their changes that relate to this change request.

    • Array [
    • id number required

      The ID of this change.

    • action string required

      The kind of action that the change contains information about.

    • conflict string

      A description of the conflict caused by this change. Only present if there are any conflicts.

    • payload objectrequired

      The data required to perform this action.

      oneOf
    • string
    • createdBy object

      The user who created this change.

    • username string nullable

      The user's username.

    • imageUrl uri nullable

      The URL where the user's image can be found.

    • createdAt date-time

      When this change was suggested

    • name string required

      The current name of the segment

    • ]
    • approvals object[]

      A list of approvals that this change request has received.

    • Array [
    • createdBy objectrequired

      Information about the user who gave this approval.

    • id number

      The ID of the user who gave this approval.

    • username string

      The approving user's username.

    • imageUrl uri

      The URL where the user's image can be found.

    • createdAt date-time required

      When the approval was given.

    • ]
    • rejections object[]

      A list of rejections that this change request has received.

    • Array [
    • createdBy objectrequired

      Information about the user who gave this approval.

    • id number

      The ID of the user who gave this approval.

    • username string

      The approving user's username.

    • imageUrl uri

      The URL where the user's image can be found.

    • createdAt date-time required

      When the approval was given.

    • ]
    • comments object[]

      All comments that have been made on this change request.

    • Array [
    • id number

      The comment's ID. Unique per change request.

    • text string required

      The content of the comment.

    • createdBy objectrequired

      Information about the user who posted the comment

    • username string nullable

      The user's username.

    • imageUrl uri nullable

      The URL where the user's image can be found.

    • createdAt date-time required

      When the comment was made.

    • ]
    • createdBy objectrequired

      The user who created this change request.

    • username string nullable
    • imageUrl uri nullable

      The URL of the user's profile image.

    • createdAt date-time required

      When this change request was created.

    • state string required

      Possible values: [Draft, In review, Approved, Applied, Cancelled, Rejected]

      The current state of the change request.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/enable-banner.html b/reference/api/unleash/enable-banner.html index ff26669813..7e16fe3351 100644 --- a/reference/api/unleash/enable-banner.html +++ b/reference/api/unleash/enable-banner.html @@ -20,15 +20,15 @@ - - + +

    Enables a banner.

    POST /api/admin/banners/:id/on

    Enables an existing banner, identified by its id.

    Request

    Path Parameters

    • id string required
    Responses

    bannerSchema

    Schema
    • id integer required

      Possible values: >= 1

      The banner's ID. Banner IDs are incrementing integers. In other words, a more recently created banner will always have a higher ID than an older one.

    • message string required

      The message to display to all users. Supports markdown.

    • enabled boolean

      Whether the banner should be displayed currently. If not specified, defaults to true.

    • variant string

      The variant of the banner. One of "info", "warning", "error", or "success". If not specified, defaults to "info".

    • sticky boolean

      Whether the banner should be sticky on the screen. If not specified, defaults to false.

    • icon string nullable

      The icon to display on the banner. Can be one of https://fonts.google.com/icons. If not specified, this will be the default icon for the variant. If "none", no icon will be displayed.

    • link string nullable

      The link to display on the banner. Can either be an absolute or a relative link (e.g. absolute: "https://example.com" or relative: "/admin/service-accounts"). If "dialog", will display a dialog when clicked. If not specified, no link will be displayed.

    • linkText string nullable

      The text to display on the link. If not specified, will be displayed as "More info".

    • dialogTitle string nullable

      The title to display on the dialog. If not specified, this will be the same as linkText.

    • dialog string nullable

      The markdown to display on the dialog. If not specified, no dialog will be displayed.

    • createdAt date-time required

      The date and time of when the banner was created.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/environments.html b/reference/api/unleash/environments.html index c9b166cf8c..c6503f53d6 100644 --- a/reference/api/unleash/environments.html +++ b/reference/api/unleash/environments.html @@ -20,15 +20,15 @@ - - + +

    Environments

    Create, update, delete, enable or disable environments for this Unleash instance.

    - - + + \ No newline at end of file diff --git a/reference/api/unleash/events.html b/reference/api/unleash/events.html index 353c4c8296..9eb78de49b 100644 --- a/reference/api/unleash/events.html +++ b/reference/api/unleash/events.html @@ -20,15 +20,15 @@ - - + +
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/export-features.html b/reference/api/unleash/export-features.html index d8b7fcaf5d..b4b6166863 100644 --- a/reference/api/unleash/export-features.html +++ b/reference/api/unleash/export-features.html @@ -20,15 +20,15 @@ - - + +

    Export feature toggles from an environment

    POST /api/admin/features-batch/export

    Exports all features listed in the features property from the environment specified in the request body. If set to true, the downloadFile property will let you download a file with the exported data. Otherwise, the export data is returned directly as JSON. Refer to the documentation for more information about Unleash's export functionality.

    Request

    Body

    required

    exportQuerySchema

      anyOf
    • environment string required

      The environment to export from

    • downloadFile boolean

      Whether to return a downloadable file

    • features string[] required

      Possible values: non-empty

      Selects features to export by name. If the list is empty all features are returned.

    Responses

    exportResultSchema

    Schema
    • features object[]required

      All the exported features.

    • Array [
    • name string required

      Unique feature name

    • type string

      Type of the toggle e.g. experiment, kill-switch, release, operational, permission

    • description string nullable

      Detailed description of the feature

    • archived boolean

      true if the feature is archived

    • project string

      Name of the project the feature belongs to

    • enabled boolean

      true if the feature is enabled, otherwise false.

    • stale boolean

      true if the feature is stale based on the age and feature type, otherwise false.

    • favorite boolean

      true if the feature was favorited, otherwise false.

    • impressionData boolean

      true if the impression data collection is enabled for the feature, otherwise false.

    • createdAt date-time nullable

      The date the feature was created

    • archivedAt date-time nullable

      The date the feature was archived

    • lastSeenAt date-time nullable deprecated

      The date when metrics where last collected for the feature. This field is deprecated, use the one in featureEnvironmentSchema

    • environments object[]

      The list of environments where the feature can be used

    • Array [
    • name string required

      The name of the environment

    • featureName string

      The name of the feature

    • environment string

      The name of the environment

    • type string

      The type of the environment

    • enabled boolean required

      true if the feature is enabled for the environment, otherwise false.

    • sortOrder number

      The sort order of the feature environment in the feature environments list

    • variantCount number

      The number of defined variants

    • strategies object[]

      A list of activation strategies for the feature environment

    • Array [
    • id string

      A uuid for the feature strategy

    • name string required

      The name or type of strategy

    • title string nullable

      A descriptive title for the strategy

    • disabled boolean nullable

      A toggle to disable the strategy. defaults to false. Disabled strategies are not evaluated or returned to the SDKs

    • featureName string

      The name or feature the strategy is attached to

    • sortOrder number

      The order of the strategy in the list

    • segments number[]

      A list of segment ids attached to the strategy

    • constraints object[]

      A list of the constraints attached to the strategy. See https://docs.getunleash.io/reference/strategy-constraints

    • Array [
    • contextName string required

      The name of the context field that this constraint should apply to.

    • operator string required

      Possible values: [NOT_IN, IN, STR_ENDS_WITH, STR_STARTS_WITH, STR_CONTAINS, NUM_EQ, NUM_GT, NUM_GTE, NUM_LT, NUM_LTE, DATE_AFTER, DATE_BEFORE, SEMVER_EQ, SEMVER_GT, SEMVER_LT]

      The operator to use when evaluating this constraint. For more information about the various operators, refer to the strategy constraint operator documentation.

    • caseInsensitive boolean

      Default value: false

      Whether the operator should be case sensitive or not. Defaults to false (being case sensitive).

    • inverted boolean

      Default value: false

      Whether the result should be negated or not. If true, will turn a true result into a false result and vice versa.

    • values string[]

      The context values that should be used for constraint evaluation. Use this property instead of value for properties that accept multiple values.

    • value string

      The context value that should be used for constraint evaluation. Use this property instead of values for properties that only accept single values.

    • ]
    • variants object[]

      Strategy level variants

    • Array [
    • name string required

      The variant name. Must be unique for this feature toggle

    • weight integer required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is an integer between 0 and 1000. See the section on variant weights for more information

    • weightType string required

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000. Refer to the variant weight documentation.

    • stickiness string required

      The stickiness to use for distribution of this variant. Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • ]
    • parameters object

      A list of parameters for a strategy

    • property name* string
    • ]
    • variants object[]

      A list of variants for the feature environment

    • Array [
    • name string required

      The variants name. Is unique for this feature toggle

    • weight number required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is a number between 0 and 1000. See the section on variant weights for more information

    • weightType string

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000

    • stickiness string

      Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • overrides object[]

      Overrides assigning specific variants to specific users. The weighting system automatically assigns users to specific groups for you, but any overrides in this list will take precedence.

    • Array [
    • contextName string required

      The name of the context field used to determine overrides

    • values string[] required

      Which values that should be overriden

    • ]
    • ]
    • lastSeenAt date-time nullable

      The date when metrics where last collected for the feature environment

    • hasStrategies boolean

      Whether the feature has any strategies defined.

    • hasEnabledStrategies boolean

      Whether the feature has any enabled strategies defined.

    • ]
    • variants object[]deprecated

      The list of feature variants

    • Array [
    • name string required

      The variants name. Is unique for this feature toggle

    • weight number required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is a number between 0 and 1000. See the section on variant weights for more information

    • weightType string

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000

    • stickiness string

      Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • overrides object[]

      Overrides assigning specific variants to specific users. The weighting system automatically assigns users to specific groups for you, but any overrides in this list will take precedence.

    • Array [
    • contextName string required

      The name of the context field used to determine overrides

    • values string[] required

      Which values that should be overriden

    • ]
    • ]
    • strategies object[] deprecated

      This is a legacy field that will be deprecated

    • tags object[]nullable

      The list of feature tags

    • Array [
    • value string required

      Possible values: >= 2 characters and <= 50 characters

      The value of the tag

    • type string required

      Possible values: >= 2 characters and <= 50 characters

      Default value: simple

      The type of the tag

    • ]
    • children string[]

      The list of child feature names. This is an experimental field and may change.

    • dependencies object[]

      The list of parent dependencies. This is an experimental field and may change.

    • Array [
    • feature string required

      The name of the parent feature

    • enabled boolean

      Whether the parent feature is enabled or not

    • variants string[]

      The list of variants the parent feature should resolve to. Only valid when feature is enabled.

    • ]
    • ]
    • featureStrategies object[]required

      All strategy instances that are used by the exported features in the features list.

    • Array [
    • id string

      A uuid for the feature strategy

    • name string required

      The name or type of strategy

    • title string nullable

      A descriptive title for the strategy

    • disabled boolean nullable

      A toggle to disable the strategy. defaults to false. Disabled strategies are not evaluated or returned to the SDKs

    • featureName string

      The name or feature the strategy is attached to

    • sortOrder number

      The order of the strategy in the list

    • segments number[]

      A list of segment ids attached to the strategy

    • constraints object[]

      A list of the constraints attached to the strategy. See https://docs.getunleash.io/reference/strategy-constraints

    • Array [
    • contextName string required

      The name of the context field that this constraint should apply to.

    • operator string required

      Possible values: [NOT_IN, IN, STR_ENDS_WITH, STR_STARTS_WITH, STR_CONTAINS, NUM_EQ, NUM_GT, NUM_GTE, NUM_LT, NUM_LTE, DATE_AFTER, DATE_BEFORE, SEMVER_EQ, SEMVER_GT, SEMVER_LT]

      The operator to use when evaluating this constraint. For more information about the various operators, refer to the strategy constraint operator documentation.

    • caseInsensitive boolean

      Default value: false

      Whether the operator should be case sensitive or not. Defaults to false (being case sensitive).

    • inverted boolean

      Default value: false

      Whether the result should be negated or not. If true, will turn a true result into a false result and vice versa.

    • values string[]

      The context values that should be used for constraint evaluation. Use this property instead of value for properties that accept multiple values.

    • value string

      The context value that should be used for constraint evaluation. Use this property instead of values for properties that only accept single values.

    • ]
    • variants object[]

      Strategy level variants

    • Array [
    • name string required

      The variant name. Must be unique for this feature toggle

    • weight integer required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is an integer between 0 and 1000. See the section on variant weights for more information

    • weightType string required

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000. Refer to the variant weight documentation.

    • stickiness string required

      The stickiness to use for distribution of this variant. Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • ]
    • parameters object

      A list of parameters for a strategy

    • property name* string
    • ]
    • featureEnvironments object[]

      Environment-specific configuration for all the features in the features list. Includes data such as whether the feature is enabled in the selected export environment, whether there are any variants assigned, etc.

    • Array [
    • name string required

      The name of the environment

    • featureName string

      The name of the feature

    • environment string

      The name of the environment

    • type string

      The type of the environment

    • enabled boolean required

      true if the feature is enabled for the environment, otherwise false.

    • sortOrder number

      The sort order of the feature environment in the feature environments list

    • variantCount number

      The number of defined variants

    • strategies object[]

      A list of activation strategies for the feature environment

    • Array [
    • id string

      A uuid for the feature strategy

    • name string required

      The name or type of strategy

    • title string nullable

      A descriptive title for the strategy

    • disabled boolean nullable

      A toggle to disable the strategy. defaults to false. Disabled strategies are not evaluated or returned to the SDKs

    • featureName string

      The name or feature the strategy is attached to

    • sortOrder number

      The order of the strategy in the list

    • segments number[]

      A list of segment ids attached to the strategy

    • constraints object[]

      A list of the constraints attached to the strategy. See https://docs.getunleash.io/reference/strategy-constraints

    • Array [
    • contextName string required

      The name of the context field that this constraint should apply to.

    • operator string required

      Possible values: [NOT_IN, IN, STR_ENDS_WITH, STR_STARTS_WITH, STR_CONTAINS, NUM_EQ, NUM_GT, NUM_GTE, NUM_LT, NUM_LTE, DATE_AFTER, DATE_BEFORE, SEMVER_EQ, SEMVER_GT, SEMVER_LT]

      The operator to use when evaluating this constraint. For more information about the various operators, refer to the strategy constraint operator documentation.

    • caseInsensitive boolean

      Default value: false

      Whether the operator should be case sensitive or not. Defaults to false (being case sensitive).

    • inverted boolean

      Default value: false

      Whether the result should be negated or not. If true, will turn a true result into a false result and vice versa.

    • values string[]

      The context values that should be used for constraint evaluation. Use this property instead of value for properties that accept multiple values.

    • value string

      The context value that should be used for constraint evaluation. Use this property instead of values for properties that only accept single values.

    • ]
    • variants object[]

      Strategy level variants

    • Array [
    • name string required

      The variant name. Must be unique for this feature toggle

    • weight integer required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is an integer between 0 and 1000. See the section on variant weights for more information

    • weightType string required

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000. Refer to the variant weight documentation.

    • stickiness string required

      The stickiness to use for distribution of this variant. Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • ]
    • parameters object

      A list of parameters for a strategy

    • property name* string
    • ]
    • variants object[]

      A list of variants for the feature environment

    • Array [
    • name string required

      The variants name. Is unique for this feature toggle

    • weight number required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is a number between 0 and 1000. See the section on variant weights for more information

    • weightType string

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000

    • stickiness string

      Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • overrides object[]

      Overrides assigning specific variants to specific users. The weighting system automatically assigns users to specific groups for you, but any overrides in this list will take precedence.

    • Array [
    • contextName string required

      The name of the context field used to determine overrides

    • values string[] required

      Which values that should be overriden

    • ]
    • ]
    • lastSeenAt date-time nullable

      The date when metrics where last collected for the feature environment

    • hasStrategies boolean

      Whether the feature has any strategies defined.

    • hasEnabledStrategies boolean

      Whether the feature has any enabled strategies defined.

    • ]
    • contextFields object[]

      A list of all the context fields that are in use by any of the strategies in the featureStrategies list.

    • Array [
    • name string required

      The name of the context field

    • description string nullable

      The description of the context field.

    • stickiness boolean

      Does this context field support being used for stickiness calculations

    • sortOrder integer

      Used when sorting a list of context fields. Is also used as a tiebreaker if a list of context fields is sorted alphabetically.

    • createdAt date-time nullable

      When this context field was created

    • usedInFeatures integer nullable

      Number of projects where this context field is used in

    • usedInProjects integer nullable

      Number of projects where this context field is used in

    • legalValues object[]

      Allowed values for this context field schema. Can be used to narrow down accepted input

    • Array [
    • value string required

      The valid value

    • description string

      Describes this specific legal value

    • ]
    • ]
    • featureTags object[]

      A list of all the tags that have been applied to any of the features in the features list.

    • Array [
    • featureName string required

      The name of the feature this tag is applied to

    • tagType string

      The [type](https://docs.getunleash.io/reference/tags#tag-types tag types) of the tag

    • tagValue string required

      The value of the tag

    • type string deprecated

      The [type](https://docs.getunleash.io/reference/tags#tag-types tag types) of the tag. This property is deprecated and will be removed in a future version of Unleash. Superseded by the tagType property.

    • value string deprecated

      The value of the tag. This property is deprecated and will be removed in a future version of Unleash. Superseded by the tagValue property.

    • createdByUserId number nullable

      The id of the user who created this tag

    • ]
    • segments object[]

      A list of all the segments that are used by the strategies in the featureStrategies list.

    • Array [
    • id number required
    • name string required
    • ]
    • tagTypes object[]required

      A list of all of the tag types that are used in the featureTags list.

    • Array [
    • name string required

      The name of the tag type.

    • description string

      The description of the tag type.

    • icon string nullable

      The icon of the tag type.

    • ]
    • dependencies object[]

      A list of all the dependencies for features in features list.

    • Array [
    • feature string required

      The name of the child feature.

    • dependencies object[]required

      List of parent features for the child feature

    • Array [
    • feature string required

      The name of the feature we depend on.

    • enabled boolean

      Whether the parent feature should be enabled. When false variants are ignored. true by default.

    • variants string[]

      The list of variants the parent feature should resolve to. Leave empty when you only want to check the enabled status.

    • ]
    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/export.html b/reference/api/unleash/export.html index 76dc221521..70c0bb07c8 100644 --- a/reference/api/unleash/export.html +++ b/reference/api/unleash/export.html @@ -20,15 +20,15 @@ - - + +

    Export state (deprecated)

    GET /api/admin/state/export
    deprecated

    This endpoint has been deprecated and may be removed in future versions of the API.

    Exports the current state of the system. Deprecated in favor of /api/admin/features-batch/export

    Request

    Query Parameters

    • format string

      Possible values: [json, yaml]

      Default value: json

      Desired export format. Must be either json or yaml.

    • download any

      Whether exported data should be downloaded as a file.

    • strategies any

      Whether strategies should be included in the exported data.

    • featureToggles any

      Whether feature toggles should be included in the exported data.

    • projects any

      Whether projects should be included in the exported data.

    • tags any

      Whether tag types, tags, and feature_tags should be included in the exported data.

    • environments any

      Whether environments should be included in the exported data.

    Responses

    stateSchema

    Schema
    • version integer required

      The version of the schema used to describe the state

    • features object[]

      A list of features

    • Array [
    • name string required

      Unique feature name

    • type string

      Type of the toggle e.g. experiment, kill-switch, release, operational, permission

    • description string nullable

      Detailed description of the feature

    • archived boolean

      true if the feature is archived

    • project string

      Name of the project the feature belongs to

    • enabled boolean

      true if the feature is enabled, otherwise false.

    • stale boolean

      true if the feature is stale based on the age and feature type, otherwise false.

    • favorite boolean

      true if the feature was favorited, otherwise false.

    • impressionData boolean

      true if the impression data collection is enabled for the feature, otherwise false.

    • createdAt date-time nullable

      The date the feature was created

    • archivedAt date-time nullable

      The date the feature was archived

    • lastSeenAt date-time nullable deprecated

      The date when metrics where last collected for the feature. This field is deprecated, use the one in featureEnvironmentSchema

    • environments object[]

      The list of environments where the feature can be used

    • Array [
    • name string required

      The name of the environment

    • featureName string

      The name of the feature

    • environment string

      The name of the environment

    • type string

      The type of the environment

    • enabled boolean required

      true if the feature is enabled for the environment, otherwise false.

    • sortOrder number

      The sort order of the feature environment in the feature environments list

    • variantCount number

      The number of defined variants

    • strategies object[]

      A list of activation strategies for the feature environment

    • Array [
    • id string

      A uuid for the feature strategy

    • name string required

      The name or type of strategy

    • title string nullable

      A descriptive title for the strategy

    • disabled boolean nullable

      A toggle to disable the strategy. defaults to false. Disabled strategies are not evaluated or returned to the SDKs

    • featureName string

      The name or feature the strategy is attached to

    • sortOrder number

      The order of the strategy in the list

    • segments number[]

      A list of segment ids attached to the strategy

    • constraints object[]

      A list of the constraints attached to the strategy. See https://docs.getunleash.io/reference/strategy-constraints

    • Array [
    • contextName string required

      The name of the context field that this constraint should apply to.

    • operator string required

      Possible values: [NOT_IN, IN, STR_ENDS_WITH, STR_STARTS_WITH, STR_CONTAINS, NUM_EQ, NUM_GT, NUM_GTE, NUM_LT, NUM_LTE, DATE_AFTER, DATE_BEFORE, SEMVER_EQ, SEMVER_GT, SEMVER_LT]

      The operator to use when evaluating this constraint. For more information about the various operators, refer to the strategy constraint operator documentation.

    • caseInsensitive boolean

      Default value: false

      Whether the operator should be case sensitive or not. Defaults to false (being case sensitive).

    • inverted boolean

      Default value: false

      Whether the result should be negated or not. If true, will turn a true result into a false result and vice versa.

    • values string[]

      The context values that should be used for constraint evaluation. Use this property instead of value for properties that accept multiple values.

    • value string

      The context value that should be used for constraint evaluation. Use this property instead of values for properties that only accept single values.

    • ]
    • variants object[]

      Strategy level variants

    • Array [
    • name string required

      The variant name. Must be unique for this feature toggle

    • weight integer required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is an integer between 0 and 1000. See the section on variant weights for more information

    • weightType string required

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000. Refer to the variant weight documentation.

    • stickiness string required

      The stickiness to use for distribution of this variant. Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • ]
    • parameters object

      A list of parameters for a strategy

    • property name* string
    • ]
    • variants object[]

      A list of variants for the feature environment

    • Array [
    • name string required

      The variants name. Is unique for this feature toggle

    • weight number required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is a number between 0 and 1000. See the section on variant weights for more information

    • weightType string

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000

    • stickiness string

      Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • overrides object[]

      Overrides assigning specific variants to specific users. The weighting system automatically assigns users to specific groups for you, but any overrides in this list will take precedence.

    • Array [
    • contextName string required

      The name of the context field used to determine overrides

    • values string[] required

      Which values that should be overriden

    • ]
    • ]
    • lastSeenAt date-time nullable

      The date when metrics where last collected for the feature environment

    • hasStrategies boolean

      Whether the feature has any strategies defined.

    • hasEnabledStrategies boolean

      Whether the feature has any enabled strategies defined.

    • ]
    • variants object[]deprecated

      The list of feature variants

    • Array [
    • name string required

      The variants name. Is unique for this feature toggle

    • weight number required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is a number between 0 and 1000. See the section on variant weights for more information

    • weightType string

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000

    • stickiness string

      Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • overrides object[]

      Overrides assigning specific variants to specific users. The weighting system automatically assigns users to specific groups for you, but any overrides in this list will take precedence.

    • Array [
    • contextName string required

      The name of the context field used to determine overrides

    • values string[] required

      Which values that should be overriden

    • ]
    • ]
    • strategies object[] deprecated

      This is a legacy field that will be deprecated

    • tags object[]nullable

      The list of feature tags

    • Array [
    • value string required

      Possible values: >= 2 characters and <= 50 characters

      The value of the tag

    • type string required

      Possible values: >= 2 characters and <= 50 characters

      Default value: simple

      The type of the tag

    • ]
    • children string[]

      The list of child feature names. This is an experimental field and may change.

    • dependencies object[]

      The list of parent dependencies. This is an experimental field and may change.

    • Array [
    • feature string required

      The name of the parent feature

    • enabled boolean

      Whether the parent feature is enabled or not

    • variants string[]

      The list of variants the parent feature should resolve to. Only valid when feature is enabled.

    • ]
    • ]
    • strategies object[]

      A list of strategies

    • Array [
    • title string nullable

      An optional title for the strategy

    • name string required

      The name (type) of the strategy

    • displayName string nullable required

      A human friendly name for the strategy

    • description string nullable required

      A short description of the strategy

    • editable boolean required

      Whether the strategy can be edited or not. Strategies bundled with Unleash cannot be edited.

    • deprecated boolean required
    • parameters object[]required

      A list of relevant parameters for each strategy

    • Array [
    • name string
    • type string
    • description string
    • required boolean
    • ]
    • ]
    • tags object[]

      A list of tags

    • Array [
    • value string required

      Possible values: >= 2 characters and <= 50 characters

      The value of the tag

    • type string required

      Possible values: >= 2 characters and <= 50 characters

      Default value: simple

      The type of the tag

    • ]
    • tagTypes object[]

      A list of tag types

    • Array [
    • name string required

      The name of the tag type.

    • description string

      The description of the tag type.

    • icon string nullable

      The icon of the tag type.

    • ]
    • featureTags object[]

      A list of tags applied to features

    • Array [
    • featureName string required

      The name of the feature this tag is applied to

    • tagType string

      The [type](https://docs.getunleash.io/reference/tags#tag-types tag types) of the tag

    • tagValue string required

      The value of the tag

    • type string deprecated

      The [type](https://docs.getunleash.io/reference/tags#tag-types tag types) of the tag. This property is deprecated and will be removed in a future version of Unleash. Superseded by the tagType property.

    • value string deprecated

      The value of the tag. This property is deprecated and will be removed in a future version of Unleash. Superseded by the tagValue property.

    • createdByUserId number nullable

      The id of the user who created this tag

    • ]
    • projects object[]

      A list of projects

    • Array [
    • id string required

      The id of this project

    • name string required

      The name of this project

    • description string nullable

      Additional information about the project

    • health number

      An indicator of the project's health on a scale from 0 to 100

    • featureCount number

      The number of features this project has

    • memberCount number

      The number of members this project has

    • createdAt date-time

      When this project was created.

    • updatedAt date-time nullable

      When this project was last updated.

    • favorite boolean

      true if the project was favorited, otherwise false.

    • mode string

      Possible values: [open, protected, private]

      The project's collaboration mode. Determines whether non-project members can submit change requests or not.

    • defaultStickiness string

      A default stickiness for the project affecting the default stickiness value for variants and Gradual Rollout strategy

    • ]
    • featureStrategies object[]

      A list of feature strategies as applied to features

    • Array [
    • id string

      A uuid for the feature strategy

    • name string required

      The name or type of strategy

    • title string nullable

      A descriptive title for the strategy

    • disabled boolean nullable

      A toggle to disable the strategy. defaults to false. Disabled strategies are not evaluated or returned to the SDKs

    • featureName string

      The name or feature the strategy is attached to

    • sortOrder number

      The order of the strategy in the list

    • segments number[]

      A list of segment ids attached to the strategy

    • constraints object[]

      A list of the constraints attached to the strategy. See https://docs.getunleash.io/reference/strategy-constraints

    • Array [
    • contextName string required

      The name of the context field that this constraint should apply to.

    • operator string required

      Possible values: [NOT_IN, IN, STR_ENDS_WITH, STR_STARTS_WITH, STR_CONTAINS, NUM_EQ, NUM_GT, NUM_GTE, NUM_LT, NUM_LTE, DATE_AFTER, DATE_BEFORE, SEMVER_EQ, SEMVER_GT, SEMVER_LT]

      The operator to use when evaluating this constraint. For more information about the various operators, refer to the strategy constraint operator documentation.

    • caseInsensitive boolean

      Default value: false

      Whether the operator should be case sensitive or not. Defaults to false (being case sensitive).

    • inverted boolean

      Default value: false

      Whether the result should be negated or not. If true, will turn a true result into a false result and vice versa.

    • values string[]

      The context values that should be used for constraint evaluation. Use this property instead of value for properties that accept multiple values.

    • value string

      The context value that should be used for constraint evaluation. Use this property instead of values for properties that only accept single values.

    • ]
    • variants object[]

      Strategy level variants

    • Array [
    • name string required

      The variant name. Must be unique for this feature toggle

    • weight integer required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is an integer between 0 and 1000. See the section on variant weights for more information

    • weightType string required

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000. Refer to the variant weight documentation.

    • stickiness string required

      The stickiness to use for distribution of this variant. Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • ]
    • parameters object

      A list of parameters for a strategy

    • property name* string
    • ]
    • featureEnvironments object[]

      A list of feature environment configurations

    • Array [
    • name string required

      The name of the environment

    • featureName string

      The name of the feature

    • environment string

      The name of the environment

    • type string

      The type of the environment

    • enabled boolean required

      true if the feature is enabled for the environment, otherwise false.

    • sortOrder number

      The sort order of the feature environment in the feature environments list

    • variantCount number

      The number of defined variants

    • strategies object[]

      A list of activation strategies for the feature environment

    • Array [
    • id string

      A uuid for the feature strategy

    • name string required

      The name or type of strategy

    • title string nullable

      A descriptive title for the strategy

    • disabled boolean nullable

      A toggle to disable the strategy. defaults to false. Disabled strategies are not evaluated or returned to the SDKs

    • featureName string

      The name or feature the strategy is attached to

    • sortOrder number

      The order of the strategy in the list

    • segments number[]

      A list of segment ids attached to the strategy

    • constraints object[]

      A list of the constraints attached to the strategy. See https://docs.getunleash.io/reference/strategy-constraints

    • Array [
    • contextName string required

      The name of the context field that this constraint should apply to.

    • operator string required

      Possible values: [NOT_IN, IN, STR_ENDS_WITH, STR_STARTS_WITH, STR_CONTAINS, NUM_EQ, NUM_GT, NUM_GTE, NUM_LT, NUM_LTE, DATE_AFTER, DATE_BEFORE, SEMVER_EQ, SEMVER_GT, SEMVER_LT]

      The operator to use when evaluating this constraint. For more information about the various operators, refer to the strategy constraint operator documentation.

    • caseInsensitive boolean

      Default value: false

      Whether the operator should be case sensitive or not. Defaults to false (being case sensitive).

    • inverted boolean

      Default value: false

      Whether the result should be negated or not. If true, will turn a true result into a false result and vice versa.

    • values string[]

      The context values that should be used for constraint evaluation. Use this property instead of value for properties that accept multiple values.

    • value string

      The context value that should be used for constraint evaluation. Use this property instead of values for properties that only accept single values.

    • ]
    • variants object[]

      Strategy level variants

    • Array [
    • name string required

      The variant name. Must be unique for this feature toggle

    • weight integer required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is an integer between 0 and 1000. See the section on variant weights for more information

    • weightType string required

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000. Refer to the variant weight documentation.

    • stickiness string required

      The stickiness to use for distribution of this variant. Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • ]
    • parameters object

      A list of parameters for a strategy

    • property name* string
    • ]
    • variants object[]

      A list of variants for the feature environment

    • Array [
    • name string required

      The variants name. Is unique for this feature toggle

    • weight number required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is a number between 0 and 1000. See the section on variant weights for more information

    • weightType string

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000

    • stickiness string

      Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • overrides object[]

      Overrides assigning specific variants to specific users. The weighting system automatically assigns users to specific groups for you, but any overrides in this list will take precedence.

    • Array [
    • contextName string required

      The name of the context field used to determine overrides

    • values string[] required

      Which values that should be overriden

    • ]
    • ]
    • lastSeenAt date-time nullable

      The date when metrics where last collected for the feature environment

    • hasStrategies boolean

      Whether the feature has any strategies defined.

    • hasEnabledStrategies boolean

      Whether the feature has any enabled strategies defined.

    • ]
    • environments object[]

      A list of environments

    • Array [
    • name string required

      The name of the environment

    • type string required
    • enabled boolean required

      true if the environment is enabled for the project, otherwise false.

    • protected boolean required

      true if the environment is protected, otherwise false. A protected environment can not be deleted.

    • sortOrder integer required

      Priority of the environment in a list of environments, the lower the value, the higher up in the list the environment will appear. Needs to be an integer

    • projectCount integer nullable

      The number of projects with this environment

    • apiTokenCount integer nullable

      The number of API tokens for the project environment

    • enabledToggleCount integer nullable

      The number of enabled toggles for the project environment

    • ]
    • segments object[]

      A list of segments

    • Array [
    • id number required

      The segment's id.

    • name string

      The name of the segment.

    • constraints object[]required

      List of constraints that determine which users are part of the segment

    • Array [
    • contextName string required

      The name of the context field that this constraint should apply to.

    • operator string required

      Possible values: [NOT_IN, IN, STR_ENDS_WITH, STR_STARTS_WITH, STR_CONTAINS, NUM_EQ, NUM_GT, NUM_GTE, NUM_LT, NUM_LTE, DATE_AFTER, DATE_BEFORE, SEMVER_EQ, SEMVER_GT, SEMVER_LT]

      The operator to use when evaluating this constraint. For more information about the various operators, refer to the strategy constraint operator documentation.

    • caseInsensitive boolean

      Default value: false

      Whether the operator should be case sensitive or not. Defaults to false (being case sensitive).

    • inverted boolean

      Default value: false

      Whether the result should be negated or not. If true, will turn a true result into a false result and vice versa.

    • values string[]

      The context values that should be used for constraint evaluation. Use this property instead of value for properties that accept multiple values.

    • value string

      The context value that should be used for constraint evaluation. Use this property instead of values for properties that only accept single values.

    • ]
    • description string nullable

      The description of the segment.

    • createdAt date-time

      The time the segment was created as a RFC 3339-conformant timestamp.

    • createdBy string

      Which user created this segment

    • project string nullable

      The project the segment relates to, if applicable.

    • ]
    • featureStrategySegments object[]

      A list of segment/strategy pairings

    • Array [
    • segmentId integer required

      The ID of the segment

    • featureStrategyId string required

      The ID of the strategy

    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/feature-types.html b/reference/api/unleash/feature-types.html index ba19e0c320..9f0185507f 100644 --- a/reference/api/unleash/feature-types.html +++ b/reference/api/unleash/feature-types.html @@ -20,15 +20,15 @@ - - + +
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/features.html b/reference/api/unleash/features.html index 87017aa478..11d4570473 100644 --- a/reference/api/unleash/features.html +++ b/reference/api/unleash/features.html @@ -20,15 +20,15 @@ - - + +

    Features

    Create, update, and delete features toggles.

    - - + + \ No newline at end of file diff --git a/reference/api/unleash/frontend-api.html b/reference/api/unleash/frontend-api.html index 518f6f81ba..4be9e1808c 100644 --- a/reference/api/unleash/frontend-api.html +++ b/reference/api/unleash/frontend-api.html @@ -20,15 +20,15 @@ - - + +
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-access-overview.html b/reference/api/unleash/get-access-overview.html index 13c060c6ac..9b5c833aaa 100644 --- a/reference/api/unleash/get-access-overview.html +++ b/reference/api/unleash/get-access-overview.html @@ -20,15 +20,15 @@ - - + +

    Gets access overview

    GET /api/admin/access/overview

    Gets an overview of what access all users have to all projects and groups

    Request

    Responses

    accessOverviewSchema

    Schema
    • overview object[]

      A list of user access details

    • Array [
    • userId integer required

      The identifier for the user

    • createdAt date-time nullable

      When the user was created

    • userName string nullable

      The name of the user

    • lastSeen date-time nullable

      The last time the user logged in

    • accessibleProjects string[] required

      A list of project ids that this user has access to

    • groups string[] required

      A list of group names that this user is a member of

    • rootRole string required

      The name of the root role that this user has

    • userEmail string required

      The email address of the user

    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-addon.html b/reference/api/unleash/get-addon.html index 33785a22a4..3b0e9ebb47 100644 --- a/reference/api/unleash/get-addon.html +++ b/reference/api/unleash/get-addon.html @@ -20,15 +20,15 @@ - - + +

    Get a specific addon

    GET /api/admin/addons/:id

    Retrieve information about the addon whose ID matches the ID in the request URL.

    Request

    Path Parameters

    • id string required
    Responses

    addonSchema

    Schema
    • id integer required

      Possible values: >= 1

      The addon's unique identifier.

    • provider string required

      The addon provider, such as "webhook" or "slack".

    • description string nullable required

      A description of the addon. null if no description exists.

    • enabled boolean required

      Whether the addon is enabled or not.

    • parameters objectrequired

      Parameters for the addon provider. This object has different required and optional properties depending on the provider you choose.

    • events string[] required

      The event types that trigger this specific addon.

    • projects string[]

      The projects that this addon listens to events from. An empty list means it listens to events from all projects.

    • environments string[]

      The list of environments that this addon listens to events from. An empty list means it listens to events from all environments.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-addons.html b/reference/api/unleash/get-addons.html index 5b4e20fc1b..6276c42f2e 100644 --- a/reference/api/unleash/get-addons.html +++ b/reference/api/unleash/get-addons.html @@ -20,15 +20,15 @@ - - + +

    Get all addons and providers

    GET /api/admin/addons

    Retrieve all addons and providers that are defined on this Unleash instance.

    Request

    Responses

    addonsSchema

    Schema
    • addons object[]required

      All the addons that exist on this instance of Unleash.

    • Array [
    • id integer required

      Possible values: >= 1

      The addon's unique identifier.

    • provider string required

      The addon provider, such as "webhook" or "slack".

    • description string nullable required

      A description of the addon. null if no description exists.

    • enabled boolean required

      Whether the addon is enabled or not.

    • parameters objectrequired

      Parameters for the addon provider. This object has different required and optional properties depending on the provider you choose.

    • events string[] required

      The event types that trigger this specific addon.

    • projects string[]

      The projects that this addon listens to events from. An empty list means it listens to events from all projects.

    • environments string[]

      The list of environments that this addon listens to events from. An empty list means it listens to events from all environments.

    • ]
    • providers object[]required

      A list of all available addon providers, along with their parameters and descriptions.

    • Array [
    • name string required

      The name of the addon type. When creating new addons, this goes in the payload's type field.

    • displayName string required

      The addon type's name as it should be displayed in the admin UI.

    • documentationUrl string required

      A URL to where you can find more information about using this addon type.

    • description string required

      A description of the addon type.

    • howTo string

      A long description of how to use this addon type. This will be displayed on the top of configuration page. Can contain markdown.

    • tagTypes object[]

      A list of Unleash tag types that this addon uses. These tags will be added to the Unleash instance when an addon of this type is created.

    • Array [
    • name string required

      The name of the tag type.

    • description string

      The description of the tag type.

    • icon string nullable

      The icon of the tag type.

    • ]
    • parameters object[]

      The addon provider's parameters. Use these to configure an addon of this provider type. Items with required: true must be provided.

    • Array [
    • name string required

      The name of the parameter as it is used in code. References to this parameter should use this value.

    • displayName string required

      The name of the parameter as it is shown to the end user in the Admin UI.

    • type string required

      The type of the parameter. Corresponds roughly to HTML input field types. Multi-line inut fields are indicated as textfield (equivalent to the HTML textarea tag).

    • description string

      A description of the parameter. This should explain to the end user what the parameter is used for.

    • placeholder string

      The default value for this parameter. This value is used if no other value is provided.

    • required boolean required

      Whether this parameter is required or not. If a parameter is required, you must give it a value when you create the addon. If it is not required it can be left out. It may receive a default value in those cases.

    • sensitive boolean required

      Indicates whether this parameter is sensitive or not. Unleash will not return sensitive parameters to API requests. It will instead use a number of asterisks to indicate that a value is set, e.g. "******". The number of asterisks does not correlate to the parameter's value.

    • ]
    • events string[]

      All the event types that are available for this addon provider.

    • installation object

      The installation configuration for this addon type.

    • url string required

      A URL to where the addon configuration should redirect to install addons of this type.

    • title string

      The title of the installation configuration. This will be displayed to the user when installing addons of this type.

    • helpText string

      The help text of the installation configuration. This will be displayed to the user when installing addons of this type.

    • alerts object[]

      A list of alerts to display to the user when installing addons of this type.

    • Array [
    • type string required

      Possible values: [success, info, warning, error]

      The type of alert. This determines the color of the alert.

    • text string required

      The text of the alert. This is what will be displayed to the user.

    • ]
    • deprecated string

      This should be used to inform the user that this addon type is deprecated and should not be used. Deprecated addons will show a badge with this information on the UI.

    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-admin-count.html b/reference/api/unleash/get-admin-count.html index 987d02bfe2..f4d941dab3 100644 --- a/reference/api/unleash/get-admin-count.html +++ b/reference/api/unleash/get-admin-count.html @@ -20,15 +20,15 @@ - - + +

    Get total count of admin accounts

    GET /api/admin/user-admin/admin-count

    Get a total count of admins with password, without password and admin service accounts

    Request

    Responses

    adminCountSchema

    Schema
    • password number required

      Total number of admins that have a password set.

    • noPassword number required

      Total number of admins that do not have a password set. May be SSO, but may also be users that did not set a password yet.

    • service number required

      Total number of service accounts that have the admin root role.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-advanced-playground.html b/reference/api/unleash/get-advanced-playground.html index 260a62e1ce..53c97051d7 100644 --- a/reference/api/unleash/get-advanced-playground.html +++ b/reference/api/unleash/get-advanced-playground.html @@ -20,8 +20,8 @@ - - + + @@ -40,7 +40,7 @@ variant. If a feature is disabled or doesn't have any variants, you would get the disabled variant. Otherwise, you'll get one of the feature's defined variants.

  • name string required

    The variant's name. If there is no variant or if the toggle is disabled, this will be disabled

  • enabled boolean required

    Whether the variant is enabled or not. If the feature is disabled or if it doesn't have variants, this property will be false

  • payload object

    An optional payload attached to the variant.

  • type string required

    The format of the payload.

  • value string required

    The payload value stringified.

  • variants object[]required

    The feature variants.

  • Array [
  • name string required

    The variants name. Is unique for this feature toggle

  • weight number required

    Possible values: <= 1000

    The weight is the likelihood of any one user getting this variant. It is a number between 0 and 1000. See the section on variant weights for more information

  • weightType string

    Possible values: [variable, fix]

    Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000

  • stickiness string

    Stickiness is how Unleash guarantees that the same user gets the same variant every time

  • payload object

    Extra data configured for this variant

  • type string required

    Possible values: [json, csv, string, number]

    The type of the value. Commonly used types are string, number, json and csv.

  • value string required

    The actual value of payload

  • overrides object[]

    Overrides assigning specific variants to specific users. The weighting system automatically assigns users to specific groups for you, but any overrides in this list will take precedence.

  • Array [
  • contextName string required

    The name of the context field used to determine overrides

  • values string[] required

    Which values that should be overriden

  • ]
  • ]
  • ]
  • ]
  • Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-all-api-tokens.html b/reference/api/unleash/get-all-api-tokens.html index f887e16c8c..085b195e3f 100644 --- a/reference/api/unleash/get-all-api-tokens.html +++ b/reference/api/unleash/get-all-api-tokens.html @@ -20,15 +20,15 @@ - - + +

    Get API tokens

    GET /api/admin/api-tokens

    Retrieves all API tokens that exist in the Unleash instance.

    Request

    Responses

    apiTokensSchema

    Schema
    • tokens object[]required

      A list of Unleash API tokens.

    • Array [
    • secret string required

      The token used for authentication.

    • username string deprecated

      This property was deprecated in Unleash v5. Prefer the tokenName property instead.

    • tokenName string required

      A unique name for this particular token

    • type string required

      Possible values: [client, admin, frontend]

      The type of API token

    • environment string

      The environment the token has access to. * if it has access to all environments.

    • project string required

      The project this token belongs to.

    • projects string[] required

      The list of projects this token has access to. If the token has access to specific projects they will be listed here. If the token has access to all projects it will be represented as [*]

    • expiresAt date-time nullable

      The token's expiration date. NULL if the token doesn't have an expiration set.

    • createdAt date-time required

      When the token was created.

    • seenAt date-time nullable

      When the token was last seen/used to authenticate with. NULL if the token has not yet been used for authentication.

    • alias string nullable

      Alias is no longer in active use and will often be NULL. It's kept around as a way of allowing old proxy tokens created with the old metadata format to keep working.

    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-all-client-features.html b/reference/api/unleash/get-all-client-features.html index 689d487752..c3511681d3 100644 --- a/reference/api/unleash/get-all-client-features.html +++ b/reference/api/unleash/get-all-client-features.html @@ -20,15 +20,15 @@ - - + +

    Get all toggles (SDK)

    GET /api/client/features

    Returns the SDK configuration for all feature toggles that are available to the provided API key. Used by SDKs to configure local evaluation

    Request

    Responses

    clientFeaturesSchema

    Schema
    • version number required

      A version number for the format used in the response. Most Unleash instances now return version 2, which includes segments as a separate array

    • features object[]required

      A list of feature toggles with their configuration

    • Array [
    • name string required

      The unique name of a feature toggle. Is validated to be URL safe on creation

    • type string

      What kind of feature flag is this. Refer to the documentation on feature toggle types for more information

    • description string nullable

      A description of the toggle

    • enabled boolean required

      Whether the feature flag is enabled for the current API key or not. This is ANDed with the evaluation results of the strategies list, so if this is false, the evaluation result will always be false

    • stale boolean

      If this is true Unleash believes this feature toggle has been active longer than Unleash expects a toggle of this type to be active

    • impressionData boolean nullable

      Set to true if SDKs should trigger impression events when this toggle is evaluated

    • project string

      Which project this feature toggle belongs to

    • strategies object[]

      Evaluation strategies for this toggle. Each entry in this list will be evaluated and ORed together

    • Array [
    • id string

      A uuid for the feature strategy

    • name string required

      The name or type of strategy

    • title string nullable

      A descriptive title for the strategy

    • disabled boolean nullable

      A toggle to disable the strategy. defaults to false. Disabled strategies are not evaluated or returned to the SDKs

    • featureName string

      The name or feature the strategy is attached to

    • sortOrder number

      The order of the strategy in the list

    • segments number[]

      A list of segment ids attached to the strategy

    • constraints object[]

      A list of the constraints attached to the strategy. See https://docs.getunleash.io/reference/strategy-constraints

    • Array [
    • contextName string required

      The name of the context field that this constraint should apply to.

    • operator string required

      Possible values: [NOT_IN, IN, STR_ENDS_WITH, STR_STARTS_WITH, STR_CONTAINS, NUM_EQ, NUM_GT, NUM_GTE, NUM_LT, NUM_LTE, DATE_AFTER, DATE_BEFORE, SEMVER_EQ, SEMVER_GT, SEMVER_LT]

      The operator to use when evaluating this constraint. For more information about the various operators, refer to the strategy constraint operator documentation.

    • caseInsensitive boolean

      Default value: false

      Whether the operator should be case sensitive or not. Defaults to false (being case sensitive).

    • inverted boolean

      Default value: false

      Whether the result should be negated or not. If true, will turn a true result into a false result and vice versa.

    • values string[]

      The context values that should be used for constraint evaluation. Use this property instead of value for properties that accept multiple values.

    • value string

      The context value that should be used for constraint evaluation. Use this property instead of values for properties that only accept single values.

    • ]
    • variants object[]

      Strategy level variants

    • Array [
    • name string required

      The variant name. Must be unique for this feature toggle

    • weight integer required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is an integer between 0 and 1000. See the section on variant weights for more information

    • weightType string required

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000. Refer to the variant weight documentation.

    • stickiness string required

      The stickiness to use for distribution of this variant. Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • ]
    • parameters object

      A list of parameters for a strategy

    • property name* string
    • ]
    • variants object[]nullable

      Variants configured for this toggle

    • Array [
    • name string required

      The variants name. Is unique for this feature toggle

    • weight number required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is a number between 0 and 1000. See the section on variant weights for more information

    • weightType string

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000

    • stickiness string

      Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • overrides object[]

      Overrides assigning specific variants to specific users. The weighting system automatically assigns users to specific groups for you, but any overrides in this list will take precedence.

    • Array [
    • contextName string required

      The name of the context field used to determine overrides

    • values string[] required

      Which values that should be overriden

    • ]
    • ]
    • dependencies object[]

      Feature dependencies for this toggle

    • Array [
    • feature string required

      The name of the feature we depend on.

    • enabled boolean

      Whether the parent feature should be enabled. When false variants are ignored. true by default.

    • variants string[]

      The list of variants the parent feature should resolve to. Leave empty when you only want to check the enabled status.

    • ]
    • ]
    • segments object[]

      A list of Segments configured for this Unleash instance

    • Array [
    • id number required

      The segment's id.

    • name string

      The name of the segment.

    • constraints object[]required

      List of constraints that determine which users are part of the segment

    • Array [
    • contextName string required

      The name of the context field that this constraint should apply to.

    • operator string required

      Possible values: [NOT_IN, IN, STR_ENDS_WITH, STR_STARTS_WITH, STR_CONTAINS, NUM_EQ, NUM_GT, NUM_GTE, NUM_LT, NUM_LTE, DATE_AFTER, DATE_BEFORE, SEMVER_EQ, SEMVER_GT, SEMVER_LT]

      The operator to use when evaluating this constraint. For more information about the various operators, refer to the strategy constraint operator documentation.

    • caseInsensitive boolean

      Default value: false

      Whether the operator should be case sensitive or not. Defaults to false (being case sensitive).

    • inverted boolean

      Default value: false

      Whether the result should be negated or not. If true, will turn a true result into a false result and vice versa.

    • values string[]

      The context values that should be used for constraint evaluation. Use this property instead of value for properties that accept multiple values.

    • value string

      The context value that should be used for constraint evaluation. Use this property instead of values for properties that only accept single values.

    • ]
    • ]
    • query object

      A summary of filters and parameters sent to the endpoint. Used by the server to build the features and segments response

    • tag array[]

      Features tagged with one of these tags are included

    • project string[] deprecated

      Features that are part of these projects are included in this response. (DEPRECATED) - Handled by API tokens

    • namePrefix string

      Features are filtered to only include features whose name starts with this prefix

    • environment string deprecated

      Strategies for the feature toggle configured for this environment are included. (DEPRECATED) - Handled by API tokens

    • inlineSegmentConstraints boolean

      Set to true if requesting client does not support Unleash-Client-Specification 4.2.2 or newer. Modern SDKs will have this set to false, since they will be able to merge constraints and segments themselves

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-all-environments.html b/reference/api/unleash/get-all-environments.html index 0bcbeb1ffb..828f0e523a 100644 --- a/reference/api/unleash/get-all-environments.html +++ b/reference/api/unleash/get-all-environments.html @@ -20,15 +20,15 @@ - - + +

    Get all environments

    GET /api/admin/environments

    Retrieves all environments that exist in this Unleash instance.

    Request

    Responses

    environmentsSchema

    Schema
    • version integer required

      Version of the environments schema

    • environments object[]required

      List of environments

    • Array [
    • name string required

      The name of the environment

    • type string required
    • enabled boolean required

      true if the environment is enabled for the project, otherwise false.

    • protected boolean required

      true if the environment is protected, otherwise false. A protected environment can not be deleted.

    • sortOrder integer required

      Priority of the environment in a list of environments, the lower the value, the higher up in the list the environment will appear. Needs to be an integer

    • projectCount integer nullable

      The number of projects with this environment

    • apiTokenCount integer nullable

      The number of API tokens for the project environment

    • enabledToggleCount integer nullable

      The number of enabled toggles for the project environment

    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-all-feature-types.html b/reference/api/unleash/get-all-feature-types.html index 85a6c7a849..5b4c7ee6f1 100644 --- a/reference/api/unleash/get-all-feature-types.html +++ b/reference/api/unleash/get-all-feature-types.html @@ -20,15 +20,15 @@ - - + +

    Get all feature types

    GET /api/admin/feature-types

    Retrieves all feature types that exist in this Unleash instance, along with their descriptions and lifetimes.

    Request

    Responses

    featureTypesSchema

    Schema
    • version integer required

      Possible values: [1]

      The schema version used to describe the feature toggle types listed in the types property.

    • types object[]required

      The list of feature toggle types.

    • Array [
    • id string required

      The identifier of this feature toggle type.

    • name string required

      The display name of this feature toggle type.

    • description string required

      A description of what this feature toggle type is intended to be used for.

    • lifetimeDays integer nullable required

      How many days it takes before a feature toggle of this typed is flagged as potentially stale by Unleash. If this value is null, Unleash will never mark it as potentially stale.

    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-all-public-signup-tokens.html b/reference/api/unleash/get-all-public-signup-tokens.html index 811ec1c228..05fc33a725 100644 --- a/reference/api/unleash/get-all-public-signup-tokens.html +++ b/reference/api/unleash/get-all-public-signup-tokens.html @@ -20,15 +20,15 @@ - - + +

    Get public signup tokens

    GET /api/admin/invite-link/tokens

    Retrieves all existing public signup tokens.

    Request

    Responses

    publicSignupTokensSchema

    Schema
    • tokens object[]required

      An array of all the public signup tokens

    • Array [
    • secret string required

      The actual value of the token. This is the part that is used by Unleash to create an invite link

    • url string nullable required

      The public signup link for the token. Users who follow this link will be taken to a signup page where they can create an Unleash user.

    • name string required

      The token's name. Only for displaying in the UI

    • enabled boolean required

      Whether the token is active. This property will always be false for a token that has expired.

    • expiresAt date-time required

      The time when the token will expire.

    • createdAt date-time required

      When the token was created.

    • createdBy string nullable required

      The creator's email or username

    • users object[]nullable

      Array of users that have signed up using the token.

    • Array [
    • id integer required

      The user id

    • isAPI boolean deprecated

      (Deprecated): Used internally to know which operations the user should be allowed to perform

    • name string nullable

      Name of the user

    • email string

      Email of the user

    • username string nullable

      A unique username for the user

    • imageUrl string

      URL used for the userprofile image

    • inviteLink string

      If the user is actively inviting other users, this is the link that can be shared with other users

    • loginAttempts integer

      How many unsuccessful attempts at logging in has the user made

    • emailSent boolean

      Is the welcome email sent to the user or not

    • rootRole integer

      Which root role this user is assigned

    • seenAt date-time nullable

      The last time this user logged in

    • createdAt date-time

      The user was created at this time

    • accountType string

      A user is either an actual User or a Service Account

    • permissions string[]

      Deprecated

    • ]
    • role objectrequired

      A role holds permissions to allow Unleash to decide what actions a role holder is allowed to perform

    • id integer required

      The role id

    • type string required

      A role can either be a global root role (applies to all projects) or a project role

    • name string required

      The name of the role

    • description string

      A more detailed description of the role and what use it's intended for

    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-all-strategies.html b/reference/api/unleash/get-all-strategies.html index cedec2f097..52a4259d9a 100644 --- a/reference/api/unleash/get-all-strategies.html +++ b/reference/api/unleash/get-all-strategies.html @@ -20,15 +20,15 @@ - - + +

    Get all strategies

    GET /api/admin/strategies

    Retrieves all strategy types (predefined and custom strategies) that are defined on this Unleash instance.

    Request

    Responses

    strategiesSchema

    Schema
    • version integer required

      Possible values: [1]

      Version of the strategies schema

    • strategies object[]required

      List of strategies

    • Array [
    • title string nullable

      An optional title for the strategy

    • name string required

      The name (type) of the strategy

    • displayName string nullable required

      A human friendly name for the strategy

    • description string nullable required

      A short description of the strategy

    • editable boolean required

      Whether the strategy can be edited or not. Strategies bundled with Unleash cannot be edited.

    • deprecated boolean required
    • parameters object[]required

      A list of relevant parameters for each strategy

    • Array [
    • name string
    • type string
    • description string
    • required boolean
    • ]
    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-all-toggles.html b/reference/api/unleash/get-all-toggles.html index 116ff0d10b..b5b7817af8 100644 --- a/reference/api/unleash/get-all-toggles.html +++ b/reference/api/unleash/get-all-toggles.html @@ -20,15 +20,15 @@ - - + +

    Get all feature toggles (deprecated)

    GET /api/admin/features
    deprecated

    This endpoint has been deprecated and may be removed in future versions of the API.

    Gets all feature toggles with their full configuration. This endpoint is deprecated. You should use the project-based endpoint instead (/api/admin/projects/<project-id>/features).

    Request

    Responses

    featuresSchema

    Schema
    • version integer required

      The version of the feature's schema

    • features object[]required

      A list of features

    • Array [
    • name string required

      Unique feature name

    • type string

      Type of the toggle e.g. experiment, kill-switch, release, operational, permission

    • description string nullable

      Detailed description of the feature

    • archived boolean

      true if the feature is archived

    • project string

      Name of the project the feature belongs to

    • enabled boolean

      true if the feature is enabled, otherwise false.

    • stale boolean

      true if the feature is stale based on the age and feature type, otherwise false.

    • favorite boolean

      true if the feature was favorited, otherwise false.

    • impressionData boolean

      true if the impression data collection is enabled for the feature, otherwise false.

    • createdAt date-time nullable

      The date the feature was created

    • archivedAt date-time nullable

      The date the feature was archived

    • lastSeenAt date-time nullable deprecated

      The date when metrics where last collected for the feature. This field is deprecated, use the one in featureEnvironmentSchema

    • environments object[]

      The list of environments where the feature can be used

    • Array [
    • name string required

      The name of the environment

    • featureName string

      The name of the feature

    • environment string

      The name of the environment

    • type string

      The type of the environment

    • enabled boolean required

      true if the feature is enabled for the environment, otherwise false.

    • sortOrder number

      The sort order of the feature environment in the feature environments list

    • variantCount number

      The number of defined variants

    • strategies object[]

      A list of activation strategies for the feature environment

    • Array [
    • id string

      A uuid for the feature strategy

    • name string required

      The name or type of strategy

    • title string nullable

      A descriptive title for the strategy

    • disabled boolean nullable

      A toggle to disable the strategy. defaults to false. Disabled strategies are not evaluated or returned to the SDKs

    • featureName string

      The name or feature the strategy is attached to

    • sortOrder number

      The order of the strategy in the list

    • segments number[]

      A list of segment ids attached to the strategy

    • constraints object[]

      A list of the constraints attached to the strategy. See https://docs.getunleash.io/reference/strategy-constraints

    • Array [
    • contextName string required

      The name of the context field that this constraint should apply to.

    • operator string required

      Possible values: [NOT_IN, IN, STR_ENDS_WITH, STR_STARTS_WITH, STR_CONTAINS, NUM_EQ, NUM_GT, NUM_GTE, NUM_LT, NUM_LTE, DATE_AFTER, DATE_BEFORE, SEMVER_EQ, SEMVER_GT, SEMVER_LT]

      The operator to use when evaluating this constraint. For more information about the various operators, refer to the strategy constraint operator documentation.

    • caseInsensitive boolean

      Default value: false

      Whether the operator should be case sensitive or not. Defaults to false (being case sensitive).

    • inverted boolean

      Default value: false

      Whether the result should be negated or not. If true, will turn a true result into a false result and vice versa.

    • values string[]

      The context values that should be used for constraint evaluation. Use this property instead of value for properties that accept multiple values.

    • value string

      The context value that should be used for constraint evaluation. Use this property instead of values for properties that only accept single values.

    • ]
    • variants object[]

      Strategy level variants

    • Array [
    • name string required

      The variant name. Must be unique for this feature toggle

    • weight integer required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is an integer between 0 and 1000. See the section on variant weights for more information

    • weightType string required

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000. Refer to the variant weight documentation.

    • stickiness string required

      The stickiness to use for distribution of this variant. Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • ]
    • parameters object

      A list of parameters for a strategy

    • property name* string
    • ]
    • variants object[]

      A list of variants for the feature environment

    • Array [
    • name string required

      The variants name. Is unique for this feature toggle

    • weight number required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is a number between 0 and 1000. See the section on variant weights for more information

    • weightType string

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000

    • stickiness string

      Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • overrides object[]

      Overrides assigning specific variants to specific users. The weighting system automatically assigns users to specific groups for you, but any overrides in this list will take precedence.

    • Array [
    • contextName string required

      The name of the context field used to determine overrides

    • values string[] required

      Which values that should be overriden

    • ]
    • ]
    • lastSeenAt date-time nullable

      The date when metrics where last collected for the feature environment

    • hasStrategies boolean

      Whether the feature has any strategies defined.

    • hasEnabledStrategies boolean

      Whether the feature has any enabled strategies defined.

    • ]
    • variants object[]deprecated

      The list of feature variants

    • Array [
    • name string required

      The variants name. Is unique for this feature toggle

    • weight number required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is a number between 0 and 1000. See the section on variant weights for more information

    • weightType string

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000

    • stickiness string

      Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • overrides object[]

      Overrides assigning specific variants to specific users. The weighting system automatically assigns users to specific groups for you, but any overrides in this list will take precedence.

    • Array [
    • contextName string required

      The name of the context field used to determine overrides

    • values string[] required

      Which values that should be overriden

    • ]
    • ]
    • strategies object[] deprecated

      This is a legacy field that will be deprecated

    • tags object[]nullable

      The list of feature tags

    • Array [
    • value string required

      Possible values: >= 2 characters and <= 50 characters

      The value of the tag

    • type string required

      Possible values: >= 2 characters and <= 50 characters

      Default value: simple

      The type of the tag

    • ]
    • children string[]

      The list of child feature names. This is an experimental field and may change.

    • dependencies object[]

      The list of parent dependencies. This is an experimental field and may change.

    • Array [
    • feature string required

      The name of the parent feature

    • enabled boolean

      Whether the parent feature is enabled or not

    • variants string[]

      The list of variants the parent feature should resolve to. Only valid when feature is enabled.

    • ]
    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-api-tokens-by-name.html b/reference/api/unleash/get-api-tokens-by-name.html index 6042cf895a..a63455f190 100644 --- a/reference/api/unleash/get-api-tokens-by-name.html +++ b/reference/api/unleash/get-api-tokens-by-name.html @@ -20,15 +20,15 @@ - - + +

    Get API tokens by name

    GET /api/admin/api-tokens/:name

    Retrieves all API tokens that match a given token name. Because token names are not unique, this endpoint will always return a list. If no tokens with the provided name exist, the list will be empty. Otherwise, it will contain all the tokens with the given name.

    Request

    Path Parameters

    • name string required
    Responses

    apiTokensSchema

    Schema
    • tokens object[]required

      A list of Unleash API tokens.

    • Array [
    • secret string required

      The token used for authentication.

    • username string deprecated

      This property was deprecated in Unleash v5. Prefer the tokenName property instead.

    • tokenName string required

      A unique name for this particular token

    • type string required

      Possible values: [client, admin, frontend]

      The type of API token

    • environment string

      The environment the token has access to. * if it has access to all environments.

    • project string required

      The project this token belongs to.

    • projects string[] required

      The list of projects this token has access to. If the token has access to specific projects they will be listed here. If the token has access to all projects it will be represented as [*]

    • expiresAt date-time nullable

      The token's expiration date. NULL if the token doesn't have an expiration set.

    • createdAt date-time required

      When the token was created.

    • seenAt date-time nullable

      When the token was last seen/used to authenticate with. NULL if the token has not yet been used for authentication.

    • alias string nullable

      Alias is no longer in active use and will often be NULL. It's kept around as a way of allowing old proxy tokens created with the old metadata format to keep working.

    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-application.html b/reference/api/unleash/get-application.html index 1a147227dd..7907093369 100644 --- a/reference/api/unleash/get-application.html +++ b/reference/api/unleash/get-application.html @@ -20,15 +20,15 @@ - - + +

    Get application data

    GET /api/admin/metrics/applications/:appName

    Returns data about the specified application (appName). The data contains information on the name of the application, sdkVersion (which sdk reported these metrics, typically unleash-client-node:3.4.1 or unleash-client-java:7.1.0), as well as data about how to display this application in a list.

    Request

    Path Parameters

    • appName string required
    Responses

    applicationSchema

    Schema
    • appName string required

      Name of the application

    • sdkVersion string

      Which SDK and version the application reporting uses. Typically represented as <identifier>:<version>

    • strategies string[]

      Which strategies the application has loaded. Useful when trying to figure out if your custom strategy has been loaded in the SDK

    • description string

      Extra information added about the application reporting the metrics. Only present if added via the Unleash Admin interface

    • url string

      A link to reference the application reporting the metrics. Could for instance be a GitHub link to the repository of the application

    • color string

      The CSS color that is used to color the application's entry in the application list

    • icon string

      An URL to an icon file to be used for the applications's entry in the application list

    • usage object[]

      The list of projects the application has been using.

    • Array [
    • project string required

      Name of the project

    • environments string[] required

      Which environments have been accessed in this project.

    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-applications.html b/reference/api/unleash/get-applications.html index 8ea0d8f774..6c53dd0099 100644 --- a/reference/api/unleash/get-applications.html +++ b/reference/api/unleash/get-applications.html @@ -20,15 +20,15 @@ - - + +

    Get all applications

    GET /api/admin/metrics/applications

    Returns all applications registered with Unleash. Applications can be created via metrics reporting or manual creation

    Request

    Responses

    applicationsSchema

    Schema
    • applications object[]

      The list of applications that have connected to this Unleash instance.

    • Array [
    • appName string required

      Name of the application

    • sdkVersion string

      Which SDK and version the application reporting uses. Typically represented as <identifier>:<version>

    • strategies string[]

      Which strategies the application has loaded. Useful when trying to figure out if your custom strategy has been loaded in the SDK

    • description string

      Extra information added about the application reporting the metrics. Only present if added via the Unleash Admin interface

    • url string

      A link to reference the application reporting the metrics. Could for instance be a GitHub link to the repository of the application

    • color string

      The CSS color that is used to color the application's entry in the application list

    • icon string

      An URL to an icon file to be used for the applications's entry in the application list

    • usage object[]

      The list of projects the application has been using.

    • Array [
    • project string required

      Name of the project

    • environments string[] required

      Which environments have been accessed in this project.

    • ]
    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-archived-features-by-project-id.html b/reference/api/unleash/get-archived-features-by-project-id.html index c91828c40f..529f22e39e 100644 --- a/reference/api/unleash/get-archived-features-by-project-id.html +++ b/reference/api/unleash/get-archived-features-by-project-id.html @@ -20,15 +20,15 @@ - - + +

    Get archived features in project

    GET /api/admin/archive/features/:projectId
    deprecated

    This endpoint has been deprecated and may be removed in future versions of the API.

    Retrieves a list of archived features that belong to the provided project.

    Request

    Path Parameters

    • projectId string required
    Responses

    featuresSchema

    Schema
    • version integer required

      The version of the feature's schema

    • features object[]required

      A list of features

    • Array [
    • name string required

      Unique feature name

    • type string

      Type of the toggle e.g. experiment, kill-switch, release, operational, permission

    • description string nullable

      Detailed description of the feature

    • archived boolean

      true if the feature is archived

    • project string

      Name of the project the feature belongs to

    • enabled boolean

      true if the feature is enabled, otherwise false.

    • stale boolean

      true if the feature is stale based on the age and feature type, otherwise false.

    • favorite boolean

      true if the feature was favorited, otherwise false.

    • impressionData boolean

      true if the impression data collection is enabled for the feature, otherwise false.

    • createdAt date-time nullable

      The date the feature was created

    • archivedAt date-time nullable

      The date the feature was archived

    • lastSeenAt date-time nullable deprecated

      The date when metrics where last collected for the feature. This field is deprecated, use the one in featureEnvironmentSchema

    • environments object[]

      The list of environments where the feature can be used

    • Array [
    • name string required

      The name of the environment

    • featureName string

      The name of the feature

    • environment string

      The name of the environment

    • type string

      The type of the environment

    • enabled boolean required

      true if the feature is enabled for the environment, otherwise false.

    • sortOrder number

      The sort order of the feature environment in the feature environments list

    • variantCount number

      The number of defined variants

    • strategies object[]

      A list of activation strategies for the feature environment

    • Array [
    • id string

      A uuid for the feature strategy

    • name string required

      The name or type of strategy

    • title string nullable

      A descriptive title for the strategy

    • disabled boolean nullable

      A toggle to disable the strategy. defaults to false. Disabled strategies are not evaluated or returned to the SDKs

    • featureName string

      The name or feature the strategy is attached to

    • sortOrder number

      The order of the strategy in the list

    • segments number[]

      A list of segment ids attached to the strategy

    • constraints object[]

      A list of the constraints attached to the strategy. See https://docs.getunleash.io/reference/strategy-constraints

    • Array [
    • contextName string required

      The name of the context field that this constraint should apply to.

    • operator string required

      Possible values: [NOT_IN, IN, STR_ENDS_WITH, STR_STARTS_WITH, STR_CONTAINS, NUM_EQ, NUM_GT, NUM_GTE, NUM_LT, NUM_LTE, DATE_AFTER, DATE_BEFORE, SEMVER_EQ, SEMVER_GT, SEMVER_LT]

      The operator to use when evaluating this constraint. For more information about the various operators, refer to the strategy constraint operator documentation.

    • caseInsensitive boolean

      Default value: false

      Whether the operator should be case sensitive or not. Defaults to false (being case sensitive).

    • inverted boolean

      Default value: false

      Whether the result should be negated or not. If true, will turn a true result into a false result and vice versa.

    • values string[]

      The context values that should be used for constraint evaluation. Use this property instead of value for properties that accept multiple values.

    • value string

      The context value that should be used for constraint evaluation. Use this property instead of values for properties that only accept single values.

    • ]
    • variants object[]

      Strategy level variants

    • Array [
    • name string required

      The variant name. Must be unique for this feature toggle

    • weight integer required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is an integer between 0 and 1000. See the section on variant weights for more information

    • weightType string required

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000. Refer to the variant weight documentation.

    • stickiness string required

      The stickiness to use for distribution of this variant. Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • ]
    • parameters object

      A list of parameters for a strategy

    • property name* string
    • ]
    • variants object[]

      A list of variants for the feature environment

    • Array [
    • name string required

      The variants name. Is unique for this feature toggle

    • weight number required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is a number between 0 and 1000. See the section on variant weights for more information

    • weightType string

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000

    • stickiness string

      Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • overrides object[]

      Overrides assigning specific variants to specific users. The weighting system automatically assigns users to specific groups for you, but any overrides in this list will take precedence.

    • Array [
    • contextName string required

      The name of the context field used to determine overrides

    • values string[] required

      Which values that should be overriden

    • ]
    • ]
    • lastSeenAt date-time nullable

      The date when metrics where last collected for the feature environment

    • hasStrategies boolean

      Whether the feature has any strategies defined.

    • hasEnabledStrategies boolean

      Whether the feature has any enabled strategies defined.

    • ]
    • variants object[]deprecated

      The list of feature variants

    • Array [
    • name string required

      The variants name. Is unique for this feature toggle

    • weight number required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is a number between 0 and 1000. See the section on variant weights for more information

    • weightType string

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000

    • stickiness string

      Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • overrides object[]

      Overrides assigning specific variants to specific users. The weighting system automatically assigns users to specific groups for you, but any overrides in this list will take precedence.

    • Array [
    • contextName string required

      The name of the context field used to determine overrides

    • values string[] required

      Which values that should be overriden

    • ]
    • ]
    • strategies object[] deprecated

      This is a legacy field that will be deprecated

    • tags object[]nullable

      The list of feature tags

    • Array [
    • value string required

      Possible values: >= 2 characters and <= 50 characters

      The value of the tag

    • type string required

      Possible values: >= 2 characters and <= 50 characters

      Default value: simple

      The type of the tag

    • ]
    • children string[]

      The list of child feature names. This is an experimental field and may change.

    • dependencies object[]

      The list of parent dependencies. This is an experimental field and may change.

    • Array [
    • feature string required

      The name of the parent feature

    • enabled boolean

      Whether the parent feature is enabled or not

    • variants string[]

      The list of variants the parent feature should resolve to. Only valid when feature is enabled.

    • ]
    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-archived-features.html b/reference/api/unleash/get-archived-features.html index 48d6fa04e5..fe6d49783f 100644 --- a/reference/api/unleash/get-archived-features.html +++ b/reference/api/unleash/get-archived-features.html @@ -20,15 +20,15 @@ - - + +

    Get archived features

    GET /api/admin/archive/features
    deprecated

    This endpoint has been deprecated and may be removed in future versions of the API.

    Retrieve a list of all archived feature toggles.

    Request

    Responses

    featuresSchema

    Schema
    • version integer required

      The version of the feature's schema

    • features object[]required

      A list of features

    • Array [
    • name string required

      Unique feature name

    • type string

      Type of the toggle e.g. experiment, kill-switch, release, operational, permission

    • description string nullable

      Detailed description of the feature

    • archived boolean

      true if the feature is archived

    • project string

      Name of the project the feature belongs to

    • enabled boolean

      true if the feature is enabled, otherwise false.

    • stale boolean

      true if the feature is stale based on the age and feature type, otherwise false.

    • favorite boolean

      true if the feature was favorited, otherwise false.

    • impressionData boolean

      true if the impression data collection is enabled for the feature, otherwise false.

    • createdAt date-time nullable

      The date the feature was created

    • archivedAt date-time nullable

      The date the feature was archived

    • lastSeenAt date-time nullable deprecated

      The date when metrics where last collected for the feature. This field is deprecated, use the one in featureEnvironmentSchema

    • environments object[]

      The list of environments where the feature can be used

    • Array [
    • name string required

      The name of the environment

    • featureName string

      The name of the feature

    • environment string

      The name of the environment

    • type string

      The type of the environment

    • enabled boolean required

      true if the feature is enabled for the environment, otherwise false.

    • sortOrder number

      The sort order of the feature environment in the feature environments list

    • variantCount number

      The number of defined variants

    • strategies object[]

      A list of activation strategies for the feature environment

    • Array [
    • id string

      A uuid for the feature strategy

    • name string required

      The name or type of strategy

    • title string nullable

      A descriptive title for the strategy

    • disabled boolean nullable

      A toggle to disable the strategy. defaults to false. Disabled strategies are not evaluated or returned to the SDKs

    • featureName string

      The name or feature the strategy is attached to

    • sortOrder number

      The order of the strategy in the list

    • segments number[]

      A list of segment ids attached to the strategy

    • constraints object[]

      A list of the constraints attached to the strategy. See https://docs.getunleash.io/reference/strategy-constraints

    • Array [
    • contextName string required

      The name of the context field that this constraint should apply to.

    • operator string required

      Possible values: [NOT_IN, IN, STR_ENDS_WITH, STR_STARTS_WITH, STR_CONTAINS, NUM_EQ, NUM_GT, NUM_GTE, NUM_LT, NUM_LTE, DATE_AFTER, DATE_BEFORE, SEMVER_EQ, SEMVER_GT, SEMVER_LT]

      The operator to use when evaluating this constraint. For more information about the various operators, refer to the strategy constraint operator documentation.

    • caseInsensitive boolean

      Default value: false

      Whether the operator should be case sensitive or not. Defaults to false (being case sensitive).

    • inverted boolean

      Default value: false

      Whether the result should be negated or not. If true, will turn a true result into a false result and vice versa.

    • values string[]

      The context values that should be used for constraint evaluation. Use this property instead of value for properties that accept multiple values.

    • value string

      The context value that should be used for constraint evaluation. Use this property instead of values for properties that only accept single values.

    • ]
    • variants object[]

      Strategy level variants

    • Array [
    • name string required

      The variant name. Must be unique for this feature toggle

    • weight integer required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is an integer between 0 and 1000. See the section on variant weights for more information

    • weightType string required

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000. Refer to the variant weight documentation.

    • stickiness string required

      The stickiness to use for distribution of this variant. Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • ]
    • parameters object

      A list of parameters for a strategy

    • property name* string
    • ]
    • variants object[]

      A list of variants for the feature environment

    • Array [
    • name string required

      The variants name. Is unique for this feature toggle

    • weight number required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is a number between 0 and 1000. See the section on variant weights for more information

    • weightType string

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000

    • stickiness string

      Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • overrides object[]

      Overrides assigning specific variants to specific users. The weighting system automatically assigns users to specific groups for you, but any overrides in this list will take precedence.

    • Array [
    • contextName string required

      The name of the context field used to determine overrides

    • values string[] required

      Which values that should be overriden

    • ]
    • ]
    • lastSeenAt date-time nullable

      The date when metrics where last collected for the feature environment

    • hasStrategies boolean

      Whether the feature has any strategies defined.

    • hasEnabledStrategies boolean

      Whether the feature has any enabled strategies defined.

    • ]
    • variants object[]deprecated

      The list of feature variants

    • Array [
    • name string required

      The variants name. Is unique for this feature toggle

    • weight number required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is a number between 0 and 1000. See the section on variant weights for more information

    • weightType string

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000

    • stickiness string

      Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • overrides object[]

      Overrides assigning specific variants to specific users. The weighting system automatically assigns users to specific groups for you, but any overrides in this list will take precedence.

    • Array [
    • contextName string required

      The name of the context field used to determine overrides

    • values string[] required

      Which values that should be overriden

    • ]
    • ]
    • strategies object[] deprecated

      This is a legacy field that will be deprecated

    • tags object[]nullable

      The list of feature tags

    • Array [
    • value string required

      Possible values: >= 2 characters and <= 50 characters

      The value of the tag

    • type string required

      Possible values: >= 2 characters and <= 50 characters

      Default value: simple

      The type of the tag

    • ]
    • children string[]

      The list of child feature names. This is an experimental field and may change.

    • dependencies object[]

      The list of parent dependencies. This is an experimental field and may change.

    • Array [
    • feature string required

      The name of the parent feature

    • enabled boolean

      Whether the parent feature is enabled or not

    • variants string[]

      The list of variants the parent feature should resolve to. Only valid when feature is enabled.

    • ]
    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-banners.html b/reference/api/unleash/get-banners.html index 62d2cd8629..9dedc836a5 100644 --- a/reference/api/unleash/get-banners.html +++ b/reference/api/unleash/get-banners.html @@ -20,15 +20,15 @@ - - + +

    Get all banners.

    GET /api/admin/banners

    Returns a list of all configured banners.

    Request

    Responses

    bannersSchema

    Schema
    • banners object[]required

      A list of banners.

    • Array [
    • id integer required

      Possible values: >= 1

      The banner's ID. Banner IDs are incrementing integers. In other words, a more recently created banner will always have a higher ID than an older one.

    • message string required

      The message to display to all users. Supports markdown.

    • enabled boolean

      Whether the banner should be displayed currently. If not specified, defaults to true.

    • variant string

      The variant of the banner. One of "info", "warning", "error", or "success". If not specified, defaults to "info".

    • sticky boolean

      Whether the banner should be sticky on the screen. If not specified, defaults to false.

    • icon string nullable

      The icon to display on the banner. Can be one of https://fonts.google.com/icons. If not specified, this will be the default icon for the variant. If "none", no icon will be displayed.

    • link string nullable

      The link to display on the banner. Can either be an absolute or a relative link (e.g. absolute: "https://example.com" or relative: "/admin/service-accounts"). If "dialog", will display a dialog when clicked. If not specified, no link will be displayed.

    • linkText string nullable

      The text to display on the link. If not specified, will be displayed as "More info".

    • dialogTitle string nullable

      The title to display on the dialog. If not specified, this will be the same as linkText.

    • dialog string nullable

      The markdown to display on the dialog. If not specified, no dialog will be displayed.

    • createdAt date-time required

      The date and time of when the banner was created.

    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-base-users-and-groups.html b/reference/api/unleash/get-base-users-and-groups.html index 4830d3f1a2..e128b5e213 100644 --- a/reference/api/unleash/get-base-users-and-groups.html +++ b/reference/api/unleash/get-base-users-and-groups.html @@ -20,15 +20,15 @@ - - + +

    Get basic user and group information

    GET /api/admin/user-admin/access

    Get a subset of user and group information eligible even for non-admin users

    Request

    Responses

    usersGroupsBaseSchema

    Schema
    • groups object[]

      A list of user groups and their configuration.

    • Array [
    • id integer

      The group id

    • name string required

      The name of the group

    • description string nullable

      A custom description of the group

    • mappingsSSO string[]

      A list of SSO groups that should map to this Unleash group

    • rootRole number nullable

      A role id that is used as the root role for all users in this group. This can be either the id of the Viewer, Editor or Admin role.

    • createdBy string nullable

      A user who created this group

    • createdAt date-time nullable

      When was this group created

    • users object[]

      A list of users belonging to this group

    • Array [
    • joinedAt date-time

      The date when the user joined the group

    • createdBy string nullable

      The username of the user who added this user to this group

    • user objectrequired

      An Unleash user

    • id integer required

      The user id

    • isAPI boolean deprecated

      (Deprecated): Used internally to know which operations the user should be allowed to perform

    • name string nullable

      Name of the user

    • email string

      Email of the user

    • username string nullable

      A unique username for the user

    • imageUrl string

      URL used for the userprofile image

    • inviteLink string

      If the user is actively inviting other users, this is the link that can be shared with other users

    • loginAttempts integer

      How many unsuccessful attempts at logging in has the user made

    • emailSent boolean

      Is the welcome email sent to the user or not

    • rootRole integer

      Which root role this user is assigned

    • seenAt date-time nullable

      The last time this user logged in

    • createdAt date-time

      The user was created at this time

    • accountType string

      A user is either an actual User or a Service Account

    • permissions string[]

      Deprecated

    • ]
    • projects string[]

      A list of projects where this group is used

    • userCount integer

      The number of users that belong to this group

    • ]
    • users object[]

      A list of users.

    • Array [
    • id integer required

      The user id

    • isAPI boolean deprecated

      (Deprecated): Used internally to know which operations the user should be allowed to perform

    • name string nullable

      Name of the user

    • email string

      Email of the user

    • username string nullable

      A unique username for the user

    • imageUrl string

      URL used for the userprofile image

    • inviteLink string

      If the user is actively inviting other users, this is the link that can be shared with other users

    • loginAttempts integer

      How many unsuccessful attempts at logging in has the user made

    • emailSent boolean

      Is the welcome email sent to the user or not

    • rootRole integer

      Which root role this user is assigned

    • seenAt date-time nullable

      The last time this user logged in

    • createdAt date-time

      The user was created at this time

    • accountType string

      A user is either an actual User or a Service Account

    • permissions string[]

      Deprecated

    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-change-request.html b/reference/api/unleash/get-change-request.html index cadbb4e419..10eb54b23b 100644 --- a/reference/api/unleash/get-change-request.html +++ b/reference/api/unleash/get-change-request.html @@ -20,15 +20,15 @@ - - + +

    Retrieves one change request by id

    GET /api/admin/projects/:projectId/change-requests/:id

    This endpoint will retrieve one change request if it matches the provided id.

    Request

    Path Parameters

    • projectId string required
    • id string required
    Responses

    changeRequestSchema

    Schema
      oneOf
    • id number required

      This change requests's ID.

    • title string

      A title describing the change request's content.

    • environment string required

      The environment in which the changes should be applied.

    • minApprovals number required

      The minimum number of approvals required before this change request can be applied.

    • project string required

      The project this change request belongs to.

    • features object[]required

      The list of features and their changes that relate to this change request.

    • Array [
    • name string required

      The name of the feature

    • conflict string

      A string describing the conflicts related to this change. Only present if there are any concflicts on the feature level.

    • changes object[]required

      List of changes inside change request. This list may be empty when listing all change requests for a project.

    • Array [
    • id number required

      The ID of this change.

    • action string required

      The kind of action that the change contains information about.

    • conflict string

      A description of the conflict caused by this change. Only present if there are any conflicts.

    • payload objectrequired

      The data required to perform this action.

      oneOf
    • string
    • createdBy object

      The user who created this change.

    • username string nullable

      The user's username.

    • imageUrl uri nullable

      The URL where the user's image can be found.

    • createdAt date-time

      When this change was suggested

    • ]
    • defaultChange object

      A description of a default change that will be applied with the change request to prevent invalid states.

      Default changes are changes that are applied in addition to explicit user-specified changes when a change request is applied. Any default changes are applied in the background and are not a real part of the change request.

    • action string required

      The kind of action this is.

    • payload object required

      The necessary data to perform this change.

    • ]
    • segments object[]required

      The list of segments and their changes that relate to this change request.

    • Array [
    • id number required

      The ID of this change.

    • action string required

      The kind of action that the change contains information about.

    • conflict string

      A description of the conflict caused by this change. Only present if there are any conflicts.

    • payload objectrequired

      The data required to perform this action.

      oneOf
    • string
    • createdBy object

      The user who created this change.

    • username string nullable

      The user's username.

    • imageUrl uri nullable

      The URL where the user's image can be found.

    • createdAt date-time

      When this change was suggested

    • name string required

      The current name of the segment

    • ]
    • approvals object[]

      A list of approvals that this change request has received.

    • Array [
    • createdBy objectrequired

      Information about the user who gave this approval.

    • id number

      The ID of the user who gave this approval.

    • username string

      The approving user's username.

    • imageUrl uri

      The URL where the user's image can be found.

    • createdAt date-time required

      When the approval was given.

    • ]
    • rejections object[]

      A list of rejections that this change request has received.

    • Array [
    • createdBy objectrequired

      Information about the user who gave this approval.

    • id number

      The ID of the user who gave this approval.

    • username string

      The approving user's username.

    • imageUrl uri

      The URL where the user's image can be found.

    • createdAt date-time required

      When the approval was given.

    • ]
    • comments object[]

      All comments that have been made on this change request.

    • Array [
    • id number

      The comment's ID. Unique per change request.

    • text string required

      The content of the comment.

    • createdBy objectrequired

      Information about the user who posted the comment

    • username string nullable

      The user's username.

    • imageUrl uri nullable

      The URL where the user's image can be found.

    • createdAt date-time required

      When the comment was made.

    • ]
    • createdBy objectrequired

      The user who created this change request.

    • username string nullable
    • imageUrl uri nullable

      The URL of the user's profile image.

    • createdAt date-time required

      When this change request was created.

    • state string required

      Possible values: [Draft, In review, Approved, Applied, Cancelled, Rejected]

      The current state of the change request.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-change-requests-for-project.html b/reference/api/unleash/get-change-requests-for-project.html index 86740cc1c0..74258fdedf 100644 --- a/reference/api/unleash/get-change-requests-for-project.html +++ b/reference/api/unleash/get-change-requests-for-project.html @@ -20,15 +20,15 @@ - - + +

    Retrieves all change requests for a project

    GET /api/admin/projects/:projectId/change-requests

    This endpoint will retrieve all change requests regardless of status for a given project. There's an upper limit of last 300 change requests ordered by creation date.

    Request

    Path Parameters

    • projectId string required
    Responses

    changeRequestsSchema

    Schema
    • Array [
    • oneOf
    • id number required

      This change requests's ID.

    • title string

      A title describing the change request's content.

    • environment string required

      The environment in which the changes should be applied.

    • minApprovals number required

      The minimum number of approvals required before this change request can be applied.

    • project string required

      The project this change request belongs to.

    • features object[]required

      The list of features and their changes that relate to this change request.

    • Array [
    • name string required

      The name of the feature

    • conflict string

      A string describing the conflicts related to this change. Only present if there are any concflicts on the feature level.

    • changes object[]required

      List of changes inside change request. This list may be empty when listing all change requests for a project.

    • Array [
    • id number required

      The ID of this change.

    • action string required

      The kind of action that the change contains information about.

    • conflict string

      A description of the conflict caused by this change. Only present if there are any conflicts.

    • payload objectrequired

      The data required to perform this action.

      oneOf
    • string
    • createdBy object

      The user who created this change.

    • username string nullable

      The user's username.

    • imageUrl uri nullable

      The URL where the user's image can be found.

    • createdAt date-time

      When this change was suggested

    • ]
    • defaultChange object

      A description of a default change that will be applied with the change request to prevent invalid states.

      Default changes are changes that are applied in addition to explicit user-specified changes when a change request is applied. Any default changes are applied in the background and are not a real part of the change request.

    • action string required

      The kind of action this is.

    • payload object required

      The necessary data to perform this change.

    • ]
    • segments object[]required

      The list of segments and their changes that relate to this change request.

    • Array [
    • id number required

      The ID of this change.

    • action string required

      The kind of action that the change contains information about.

    • conflict string

      A description of the conflict caused by this change. Only present if there are any conflicts.

    • payload objectrequired

      The data required to perform this action.

      oneOf
    • string
    • createdBy object

      The user who created this change.

    • username string nullable

      The user's username.

    • imageUrl uri nullable

      The URL where the user's image can be found.

    • createdAt date-time

      When this change was suggested

    • name string required

      The current name of the segment

    • ]
    • approvals object[]

      A list of approvals that this change request has received.

    • Array [
    • createdBy objectrequired

      Information about the user who gave this approval.

    • id number

      The ID of the user who gave this approval.

    • username string

      The approving user's username.

    • imageUrl uri

      The URL where the user's image can be found.

    • createdAt date-time required

      When the approval was given.

    • ]
    • rejections object[]

      A list of rejections that this change request has received.

    • Array [
    • createdBy objectrequired

      Information about the user who gave this approval.

    • id number

      The ID of the user who gave this approval.

    • username string

      The approving user's username.

    • imageUrl uri

      The URL where the user's image can be found.

    • createdAt date-time required

      When the approval was given.

    • ]
    • comments object[]

      All comments that have been made on this change request.

    • Array [
    • id number

      The comment's ID. Unique per change request.

    • text string required

      The content of the comment.

    • createdBy objectrequired

      Information about the user who posted the comment

    • username string nullable

      The user's username.

    • imageUrl uri nullable

      The URL where the user's image can be found.

    • createdAt date-time required

      When the comment was made.

    • ]
    • createdBy objectrequired

      The user who created this change request.

    • username string nullable
    • imageUrl uri nullable

      The URL of the user's profile image.

    • createdAt date-time required

      When this change request was created.

    • state string required

      Possible values: [Draft, In review, Approved, Applied, Cancelled, Rejected]

      The current state of the change request.

    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-client-feature.html b/reference/api/unleash/get-client-feature.html index 69aa91fc38..685e58b597 100644 --- a/reference/api/unleash/get-client-feature.html +++ b/reference/api/unleash/get-client-feature.html @@ -20,15 +20,15 @@ - - + +

    Get a single feature toggle

    GET /api/client/features/:featureName

    Gets all the client data for a single toggle. Contains the exact same information about a toggle as the /api/client/features endpoint does, but only contains data about the specified toggle. All SDKs should use /api/client/features

    Request

    Path Parameters

    • featureName string required
    Responses

    clientFeatureSchema

    Schema
    • name string required

      The unique name of a feature toggle. Is validated to be URL safe on creation

    • type string

      What kind of feature flag is this. Refer to the documentation on feature toggle types for more information

    • description string nullable

      A description of the toggle

    • enabled boolean required

      Whether the feature flag is enabled for the current API key or not. This is ANDed with the evaluation results of the strategies list, so if this is false, the evaluation result will always be false

    • stale boolean

      If this is true Unleash believes this feature toggle has been active longer than Unleash expects a toggle of this type to be active

    • impressionData boolean nullable

      Set to true if SDKs should trigger impression events when this toggle is evaluated

    • project string

      Which project this feature toggle belongs to

    • strategies object[]

      Evaluation strategies for this toggle. Each entry in this list will be evaluated and ORed together

    • Array [
    • id string

      A uuid for the feature strategy

    • name string required

      The name or type of strategy

    • title string nullable

      A descriptive title for the strategy

    • disabled boolean nullable

      A toggle to disable the strategy. defaults to false. Disabled strategies are not evaluated or returned to the SDKs

    • featureName string

      The name or feature the strategy is attached to

    • sortOrder number

      The order of the strategy in the list

    • segments number[]

      A list of segment ids attached to the strategy

    • constraints object[]

      A list of the constraints attached to the strategy. See https://docs.getunleash.io/reference/strategy-constraints

    • Array [
    • contextName string required

      The name of the context field that this constraint should apply to.

    • operator string required

      Possible values: [NOT_IN, IN, STR_ENDS_WITH, STR_STARTS_WITH, STR_CONTAINS, NUM_EQ, NUM_GT, NUM_GTE, NUM_LT, NUM_LTE, DATE_AFTER, DATE_BEFORE, SEMVER_EQ, SEMVER_GT, SEMVER_LT]

      The operator to use when evaluating this constraint. For more information about the various operators, refer to the strategy constraint operator documentation.

    • caseInsensitive boolean

      Default value: false

      Whether the operator should be case sensitive or not. Defaults to false (being case sensitive).

    • inverted boolean

      Default value: false

      Whether the result should be negated or not. If true, will turn a true result into a false result and vice versa.

    • values string[]

      The context values that should be used for constraint evaluation. Use this property instead of value for properties that accept multiple values.

    • value string

      The context value that should be used for constraint evaluation. Use this property instead of values for properties that only accept single values.

    • ]
    • variants object[]

      Strategy level variants

    • Array [
    • name string required

      The variant name. Must be unique for this feature toggle

    • weight integer required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is an integer between 0 and 1000. See the section on variant weights for more information

    • weightType string required

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000. Refer to the variant weight documentation.

    • stickiness string required

      The stickiness to use for distribution of this variant. Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • ]
    • parameters object

      A list of parameters for a strategy

    • property name* string
    • ]
    • variants object[]nullable

      Variants configured for this toggle

    • Array [
    • name string required

      The variants name. Is unique for this feature toggle

    • weight number required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is a number between 0 and 1000. See the section on variant weights for more information

    • weightType string

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000

    • stickiness string

      Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • overrides object[]

      Overrides assigning specific variants to specific users. The weighting system automatically assigns users to specific groups for you, but any overrides in this list will take precedence.

    • Array [
    • contextName string required

      The name of the context field used to determine overrides

    • values string[] required

      Which values that should be overriden

    • ]
    • ]
    • dependencies object[]

      Feature dependencies for this toggle

    • Array [
    • feature string required

      The name of the feature we depend on.

    • enabled boolean

      Whether the parent feature should be enabled. When false variants are ignored. true by default.

    • variants string[]

      The list of variants the parent feature should resolve to. Leave empty when you only want to check the enabled status.

    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-context-field.html b/reference/api/unleash/get-context-field.html index 2ad489b0d6..38a76c7c49 100644 --- a/reference/api/unleash/get-context-field.html +++ b/reference/api/unleash/get-context-field.html @@ -20,15 +20,15 @@ - - + +

    Gets context field

    GET /api/admin/context/:contextField

    Returns specific context field identified by the name in the path

    Request

    Path Parameters

    • contextField string required
    Responses

    contextFieldSchema

    Schema
    • name string required

      The name of the context field

    • description string nullable

      The description of the context field.

    • stickiness boolean

      Does this context field support being used for stickiness calculations

    • sortOrder integer

      Used when sorting a list of context fields. Is also used as a tiebreaker if a list of context fields is sorted alphabetically.

    • createdAt date-time nullable

      When this context field was created

    • usedInFeatures integer nullable

      Number of projects where this context field is used in

    • usedInProjects integer nullable

      Number of projects where this context field is used in

    • legalValues object[]

      Allowed values for this context field schema. Can be used to narrow down accepted input

    • Array [
    • value string required

      The valid value

    • description string

      Describes this specific legal value

    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-context-fields.html b/reference/api/unleash/get-context-fields.html index c73c563486..83db4778c9 100644 --- a/reference/api/unleash/get-context-fields.html +++ b/reference/api/unleash/get-context-fields.html @@ -20,15 +20,15 @@ - - + +

    Gets configured context fields

    GET /api/admin/context

    Returns all configured Context fields that have been created.

    Request

    Responses

    contextFieldsSchema

    Schema
    • Array [
    • name string required

      The name of the context field

    • description string nullable

      The description of the context field.

    • stickiness boolean

      Does this context field support being used for stickiness calculations

    • sortOrder integer

      Used when sorting a list of context fields. Is also used as a tiebreaker if a list of context fields is sorted alphabetically.

    • createdAt date-time nullable

      When this context field was created

    • usedInFeatures integer nullable

      Number of projects where this context field is used in

    • usedInProjects integer nullable

      Number of projects where this context field is used in

    • legalValues object[]

      Allowed values for this context field schema. Can be used to narrow down accepted input

    • Array [
    • value string required

      The valid value

    • description string

      Describes this specific legal value

    • ]
    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-deprecated-project-overview.html b/reference/api/unleash/get-deprecated-project-overview.html index 9e9f62e8fc..8476848a1d 100644 --- a/reference/api/unleash/get-deprecated-project-overview.html +++ b/reference/api/unleash/get-deprecated-project-overview.html @@ -20,15 +20,15 @@ - - + +

    Get an overview of a project. (deprecated)

    GET /api/admin/projects/:projectId
    deprecated

    This endpoint has been deprecated and may be removed in future versions of the API.

    This endpoint returns an overview of the specified projects stats, project health, number of members, which environments are configured, and the features in the project.

    Request

    Path Parameters

    • projectId string required
    Responses

    deprecatedProjectOverviewSchema

    Schema
    • stats object

      Project statistics

    • avgTimeToProdCurrentWindow number required

      The average time from when a feature was created to when it was enabled in the "production" environment during the current window

    • createdCurrentWindow number required

      The number of feature toggles created during the current window

    • createdPastWindow number required

      The number of feature toggles created during the previous window

    • archivedCurrentWindow number required

      The number of feature toggles that were archived during the current window

    • archivedPastWindow number required

      The number of feature toggles that were archived during the previous window

    • projectActivityCurrentWindow number required

      The number of project events that occurred during the current window

    • projectActivityPastWindow number required

      The number of project events that occurred during the previous window

    • projectMembersAddedCurrentWindow number required

      The number of members that were added to the project during the current window

    • version integer required

      The schema version used to describe the project overview

    • name string required

      The name of this project

    • description string nullable

      Additional information about the project

    • defaultStickiness string

      A default stickiness for the project affecting the default stickiness value for variants and Gradual Rollout strategy

    • mode string

      Possible values: [open, protected, private]

      The project's collaboration mode. Determines whether non-project members can submit change requests or not.

    • featureLimit number nullable

      A limit on the number of features allowed in the project. Null if no limit.

    • featureNaming object

      Create a feature naming pattern

    • pattern string nullable required

      A JavaScript regular expression pattern, without the start and end delimiters. Optional flags are not allowed.

    • example string nullable

      An example of a feature name that matches the pattern. Must itself match the pattern supplied.

    • description string nullable

      A description of the pattern in a human-readable format. Will be shown to users when they create a new feature flag.

    • members number

      The number of members this project has

    • health number

      An indicator of the project's health on a scale from 0 to 100

    • environments object[]

      The environments that are enabled for this project

    • Array [
    • environment string required

      The environment to add to the project

    • changeRequestsEnabled boolean

      Whether change requests should be enabled or for this environment on the project or not

    • defaultStrategy object

      A default strategy to create for this environment on the project.

    • name string required

      The name of the strategy type

    • title string nullable

      A descriptive title for the strategy

    • disabled boolean nullable

      A toggle to disable the strategy. defaults to false. Disabled strategies are not evaluated or returned to the SDKs

    • sortOrder number

      The order of the strategy in the list

    • constraints object[]

      A list of the constraints attached to the strategy. See https://docs.getunleash.io/reference/strategy-constraints

    • Array [
    • contextName string required

      The name of the context field that this constraint should apply to.

    • operator string required

      Possible values: [NOT_IN, IN, STR_ENDS_WITH, STR_STARTS_WITH, STR_CONTAINS, NUM_EQ, NUM_GT, NUM_GTE, NUM_LT, NUM_LTE, DATE_AFTER, DATE_BEFORE, SEMVER_EQ, SEMVER_GT, SEMVER_LT]

      The operator to use when evaluating this constraint. For more information about the various operators, refer to the strategy constraint operator documentation.

    • caseInsensitive boolean

      Default value: false

      Whether the operator should be case sensitive or not. Defaults to false (being case sensitive).

    • inverted boolean

      Default value: false

      Whether the result should be negated or not. If true, will turn a true result into a false result and vice versa.

    • values string[]

      The context values that should be used for constraint evaluation. Use this property instead of value for properties that accept multiple values.

    • value string

      The context value that should be used for constraint evaluation. Use this property instead of values for properties that only accept single values.

    • ]
    • variants object[]

      Strategy level variants

    • Array [
    • name string required

      The variant name. Must be unique for this feature toggle

    • weight integer required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is an integer between 0 and 1000. See the section on variant weights for more information

    • weightType string required

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000. Refer to the variant weight documentation.

    • stickiness string required

      The stickiness to use for distribution of this variant. Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • ]
    • parameters object

      A list of parameters for a strategy

    • property name* string
    • segments number[]

      Ids of segments to use for this strategy

    • ]
    • features object[]

      The full list of features in this project (excluding archived features)

    • Array [
    • name string required

      Unique feature name

    • type string

      Type of the toggle e.g. experiment, kill-switch, release, operational, permission

    • description string nullable

      Detailed description of the feature

    • archived boolean

      true if the feature is archived

    • project string

      Name of the project the feature belongs to

    • enabled boolean

      true if the feature is enabled, otherwise false.

    • stale boolean

      true if the feature is stale based on the age and feature type, otherwise false.

    • favorite boolean

      true if the feature was favorited, otherwise false.

    • impressionData boolean

      true if the impression data collection is enabled for the feature, otherwise false.

    • createdAt date-time nullable

      The date the feature was created

    • archivedAt date-time nullable

      The date the feature was archived

    • lastSeenAt date-time nullable deprecated

      The date when metrics where last collected for the feature. This field is deprecated, use the one in featureEnvironmentSchema

    • environments object[]

      The list of environments where the feature can be used

    • Array [
    • name string required

      The name of the environment

    • featureName string

      The name of the feature

    • environment string

      The name of the environment

    • type string

      The type of the environment

    • enabled boolean required

      true if the feature is enabled for the environment, otherwise false.

    • sortOrder number

      The sort order of the feature environment in the feature environments list

    • variantCount number

      The number of defined variants

    • strategies object[]

      A list of activation strategies for the feature environment

    • Array [
    • id string

      A uuid for the feature strategy

    • name string required

      The name or type of strategy

    • title string nullable

      A descriptive title for the strategy

    • disabled boolean nullable

      A toggle to disable the strategy. defaults to false. Disabled strategies are not evaluated or returned to the SDKs

    • featureName string

      The name or feature the strategy is attached to

    • sortOrder number

      The order of the strategy in the list

    • segments number[]

      A list of segment ids attached to the strategy

    • constraints object[]

      A list of the constraints attached to the strategy. See https://docs.getunleash.io/reference/strategy-constraints

    • Array [
    • contextName string required

      The name of the context field that this constraint should apply to.

    • operator string required

      Possible values: [NOT_IN, IN, STR_ENDS_WITH, STR_STARTS_WITH, STR_CONTAINS, NUM_EQ, NUM_GT, NUM_GTE, NUM_LT, NUM_LTE, DATE_AFTER, DATE_BEFORE, SEMVER_EQ, SEMVER_GT, SEMVER_LT]

      The operator to use when evaluating this constraint. For more information about the various operators, refer to the strategy constraint operator documentation.

    • caseInsensitive boolean

      Default value: false

      Whether the operator should be case sensitive or not. Defaults to false (being case sensitive).

    • inverted boolean

      Default value: false

      Whether the result should be negated or not. If true, will turn a true result into a false result and vice versa.

    • values string[]

      The context values that should be used for constraint evaluation. Use this property instead of value for properties that accept multiple values.

    • value string

      The context value that should be used for constraint evaluation. Use this property instead of values for properties that only accept single values.

    • ]
    • variants object[]

      Strategy level variants

    • Array [
    • name string required

      The variant name. Must be unique for this feature toggle

    • weight integer required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is an integer between 0 and 1000. See the section on variant weights for more information

    • weightType string required

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000. Refer to the variant weight documentation.

    • stickiness string required

      The stickiness to use for distribution of this variant. Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • ]
    • parameters object

      A list of parameters for a strategy

    • property name* string
    • ]
    • variants object[]

      A list of variants for the feature environment

    • Array [
    • name string required

      The variants name. Is unique for this feature toggle

    • weight number required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is a number between 0 and 1000. See the section on variant weights for more information

    • weightType string

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000

    • stickiness string

      Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • overrides object[]

      Overrides assigning specific variants to specific users. The weighting system automatically assigns users to specific groups for you, but any overrides in this list will take precedence.

    • Array [
    • contextName string required

      The name of the context field used to determine overrides

    • values string[] required

      Which values that should be overriden

    • ]
    • ]
    • lastSeenAt date-time nullable

      The date when metrics where last collected for the feature environment

    • hasStrategies boolean

      Whether the feature has any strategies defined.

    • hasEnabledStrategies boolean

      Whether the feature has any enabled strategies defined.

    • ]
    • variants object[]deprecated

      The list of feature variants

    • Array [
    • name string required

      The variants name. Is unique for this feature toggle

    • weight number required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is a number between 0 and 1000. See the section on variant weights for more information

    • weightType string

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000

    • stickiness string

      Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • overrides object[]

      Overrides assigning specific variants to specific users. The weighting system automatically assigns users to specific groups for you, but any overrides in this list will take precedence.

    • Array [
    • contextName string required

      The name of the context field used to determine overrides

    • values string[] required

      Which values that should be overriden

    • ]
    • ]
    • strategies object[] deprecated

      This is a legacy field that will be deprecated

    • tags object[]nullable

      The list of feature tags

    • Array [
    • value string required

      Possible values: >= 2 characters and <= 50 characters

      The value of the tag

    • type string required

      Possible values: >= 2 characters and <= 50 characters

      Default value: simple

      The type of the tag

    • ]
    • children string[]

      The list of child feature names. This is an experimental field and may change.

    • dependencies object[]

      The list of parent dependencies. This is an experimental field and may change.

    • Array [
    • feature string required

      The name of the parent feature

    • enabled boolean

      Whether the parent feature is enabled or not

    • variants string[]

      The list of variants the parent feature should resolve to. Only valid when feature is enabled.

    • ]
    • ]
    • updatedAt date-time nullable

      When the project was last updated.

    • createdAt date-time nullable

      When the project was created.

    • favorite boolean

      true if the project was favorited, otherwise false.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-environment-feature-variants.html b/reference/api/unleash/get-environment-feature-variants.html index 49f417da77..94affedb8e 100644 --- a/reference/api/unleash/get-environment-feature-variants.html +++ b/reference/api/unleash/get-environment-feature-variants.html @@ -20,15 +20,15 @@ - - + +

    Get variants for a feature in an environment

    GET /api/admin/projects/:projectId/features/:featureName/environments/:environment/variants

    Returns the variants for a feature in a specific environment. If the feature has no variants it will return an empty array of variants

    Request

    Path Parameters

    • projectId string required
    • featureName string required
    • environment string required
    Responses

    featureVariantsSchema

    Schema
    • version integer required

      The version of the feature variants schema.

    • variants object[]required

      All variants defined for a specific feature toggle.

    • Array [
    • name string required

      The variants name. Is unique for this feature toggle

    • weight number required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is a number between 0 and 1000. See the section on variant weights for more information

    • weightType string

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000

    • stickiness string

      Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • overrides object[]

      Overrides assigning specific variants to specific users. The weighting system automatically assigns users to specific groups for you, but any overrides in this list will take precedence.

    • Array [
    • contextName string required

      The name of the context field used to determine overrides

    • values string[] required

      Which values that should be overriden

    • ]
    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-environment.html b/reference/api/unleash/get-environment.html index 66cfa99725..d95057a89b 100644 --- a/reference/api/unleash/get-environment.html +++ b/reference/api/unleash/get-environment.html @@ -20,15 +20,15 @@ - - + +

    Get the environment with `name`

    GET /api/admin/environments/:name

    Retrieves the environment with name if it exists in this Unleash instance

    Request

    Path Parameters

    • name string required
    Responses

    environmentSchema

    Schema
    • name string required

      The name of the environment

    • type string required
    • enabled boolean required

      true if the environment is enabled for the project, otherwise false.

    • protected boolean required

      true if the environment is protected, otherwise false. A protected environment can not be deleted.

    • sortOrder integer required

      Priority of the environment in a list of environments, the lower the value, the higher up in the list the environment will appear. Needs to be an integer

    • projectCount integer nullable

      The number of projects with this environment

    • apiTokenCount integer nullable

      The number of API tokens for the project environment

    • enabledToggleCount integer nullable

      The number of enabled toggles for the project environment

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-events-for-toggle.html b/reference/api/unleash/get-events-for-toggle.html index c91d3b0f91..4c3a3c0f43 100644 --- a/reference/api/unleash/get-events-for-toggle.html +++ b/reference/api/unleash/get-events-for-toggle.html @@ -20,15 +20,15 @@ - - + +

    Get all events related to a specific feature toggle.

    GET /api/admin/events/:featureName

    Returns all events related to the specified feature toggle. If the feature toggle does not exist, the list of events will be empty.

    Request

    Path Parameters

    • featureName string required
    Responses

    featureEventsSchema

    Schema
    • version integer

      Possible values: >= 1, [1]

      An API versioning number

    • toggleName string

      The name of the feature toggle these events relate to

    • events object[]required

      The list of events

    • Array [
    • id integer required

      Possible values: >= 1

      The ID of the event. An increasing natural number.

    • createdAt date-time required

      The time the event happened as a RFC 3339-conformant timestamp.

    • type string required

      Possible values: [application-created, feature-created, feature-deleted, feature-updated, feature-metadata-updated, feature-variants-updated, feature-environment-variants-updated, feature-project-change, feature-archived, feature-revived, feature-import, feature-tagged, feature-tag-import, feature-strategy-update, feature-strategy-add, feature-strategy-remove, feature-type-updated, strategy-order-changed, drop-feature-tags, feature-untagged, feature-stale-on, feature-stale-off, drop-features, feature-environment-enabled, feature-environment-disabled, strategy-created, strategy-deleted, strategy-deprecated, strategy-reactivated, strategy-updated, strategy-import, drop-strategies, context-field-created, context-field-updated, context-field-deleted, project-access-added, project-access-user-roles-updated, project-access-group-roles-updated, project-access-user-roles-deleted, project-access-group-roles-deleted, project-access-updated, project-created, project-updated, project-deleted, project-import, project-user-added, project-user-removed, project-user-role-changed, project-group-role-changed, project-group-added, project-group-removed, role-created, role-updated, role-deleted, drop-projects, tag-created, tag-deleted, tag-import, drop-tags, tag-type-created, tag-type-deleted, tag-type-updated, tag-type-import, drop-tag-types, addon-config-created, addon-config-updated, addon-config-deleted, db-pool-update, user-created, user-updated, user-deleted, drop-environments, environment-import, environment-created, environment-updated, environment-deleted, segment-created, segment-updated, segment-deleted, group-created, group-updated, group-deleted, group-user-added, group-user-removed, setting-created, setting-updated, setting-deleted, client-metrics, client-register, pat-created, pat-deleted, public-signup-token-created, public-signup-token-user-added, public-signup-token-updated, change-request-created, change-request-discarded, change-added, change-discarded, change-edited, change-request-rejected, change-request-approved, change-request-approval-added, change-request-cancelled, change-request-sent-to-review, scheduled-change-request-executed, change-request-applied, change-request-scheduled, change-request-scheduled-application-success, change-request-scheduled-application-failure, change-request-configuration-updated, api-token-created, api-token-updated, api-token-deleted, feature-favorited, feature-unfavorited, project-favorited, project-unfavorited, features-exported, features-imported, service-account-created, service-account-deleted, service-account-updated, feature-potentially-stale-on, feature-dependency-added, feature-dependency-removed, feature-dependencies-removed, banner-created, banner-updated, banner-deleted, project-environment-added, project-environment-removed, default-strategy-updated, segment-import, incoming-webhook-created, incoming-webhook-updated, incoming-webhook-deleted, incoming-webhook-token-created, incoming-webhook-token-updated, incoming-webhook-token-deleted]

      What type of event this is

    • createdBy string required

      Which user created this event

    • createdByUserId number nullable

      The is of the user that created this event

    • environment string nullable

      The feature toggle environment the event relates to, if applicable.

    • project string nullable

      The project the event relates to, if applicable.

    • featureName string nullable

      The name of the feature toggle the event relates to, if applicable.

    • data object nullable

      Extra associated data related to the event, such as feature toggle state, segment configuration, etc., if applicable.

    • preData object nullable

      Data relating to the previous state of the event's subject.

    • tags object[]nullable

      Any tags related to the event, if applicable.

    • Array [
    • value string required

      Possible values: >= 2 characters and <= 50 characters

      The value of the tag

    • type string required

      Possible values: >= 2 characters and <= 50 characters

      Default value: simple

      The type of the tag

    • ]
    • ]
    • totalEvents integer

      How many events are there for this feature toggle

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-events.html b/reference/api/unleash/get-events.html index 9092de4e48..19dff1685e 100644 --- a/reference/api/unleash/get-events.html +++ b/reference/api/unleash/get-events.html @@ -20,15 +20,15 @@ - - + +

    Get the most recent events from the Unleash instance or all events related to a project.

    GET /api/admin/events

    Returns the last 100 events from the Unleash instance when called without a query parameter. When called with a project parameter, returns all events for the specified project.

    If the provided project does not exist, the list of events will be empty.

    Request

    Query Parameters

    • project string

      The name of the project whose events you want to retrieve

    Responses

    eventsSchema

    Schema
    • version integer required

      Possible values: >= 1, [1]

      The api version of this response. A natural increasing number. Only increases if format changes

    • events object[]required

      The list of events

    • Array [
    • id integer required

      Possible values: >= 1

      The ID of the event. An increasing natural number.

    • createdAt date-time required

      The time the event happened as a RFC 3339-conformant timestamp.

    • type string required

      Possible values: [application-created, feature-created, feature-deleted, feature-updated, feature-metadata-updated, feature-variants-updated, feature-environment-variants-updated, feature-project-change, feature-archived, feature-revived, feature-import, feature-tagged, feature-tag-import, feature-strategy-update, feature-strategy-add, feature-strategy-remove, feature-type-updated, strategy-order-changed, drop-feature-tags, feature-untagged, feature-stale-on, feature-stale-off, drop-features, feature-environment-enabled, feature-environment-disabled, strategy-created, strategy-deleted, strategy-deprecated, strategy-reactivated, strategy-updated, strategy-import, drop-strategies, context-field-created, context-field-updated, context-field-deleted, project-access-added, project-access-user-roles-updated, project-access-group-roles-updated, project-access-user-roles-deleted, project-access-group-roles-deleted, project-access-updated, project-created, project-updated, project-deleted, project-import, project-user-added, project-user-removed, project-user-role-changed, project-group-role-changed, project-group-added, project-group-removed, role-created, role-updated, role-deleted, drop-projects, tag-created, tag-deleted, tag-import, drop-tags, tag-type-created, tag-type-deleted, tag-type-updated, tag-type-import, drop-tag-types, addon-config-created, addon-config-updated, addon-config-deleted, db-pool-update, user-created, user-updated, user-deleted, drop-environments, environment-import, environment-created, environment-updated, environment-deleted, segment-created, segment-updated, segment-deleted, group-created, group-updated, group-deleted, group-user-added, group-user-removed, setting-created, setting-updated, setting-deleted, client-metrics, client-register, pat-created, pat-deleted, public-signup-token-created, public-signup-token-user-added, public-signup-token-updated, change-request-created, change-request-discarded, change-added, change-discarded, change-edited, change-request-rejected, change-request-approved, change-request-approval-added, change-request-cancelled, change-request-sent-to-review, scheduled-change-request-executed, change-request-applied, change-request-scheduled, change-request-scheduled-application-success, change-request-scheduled-application-failure, change-request-configuration-updated, api-token-created, api-token-updated, api-token-deleted, feature-favorited, feature-unfavorited, project-favorited, project-unfavorited, features-exported, features-imported, service-account-created, service-account-deleted, service-account-updated, feature-potentially-stale-on, feature-dependency-added, feature-dependency-removed, feature-dependencies-removed, banner-created, banner-updated, banner-deleted, project-environment-added, project-environment-removed, default-strategy-updated, segment-import, incoming-webhook-created, incoming-webhook-updated, incoming-webhook-deleted, incoming-webhook-token-created, incoming-webhook-token-updated, incoming-webhook-token-deleted]

      What type of event this is

    • createdBy string required

      Which user created this event

    • createdByUserId number nullable

      The is of the user that created this event

    • environment string nullable

      The feature toggle environment the event relates to, if applicable.

    • project string nullable

      The project the event relates to, if applicable.

    • featureName string nullable

      The name of the feature toggle the event relates to, if applicable.

    • data object nullable

      Extra associated data related to the event, such as feature toggle state, segment configuration, etc., if applicable.

    • preData object nullable

      Data relating to the previous state of the event's subject.

    • tags object[]nullable

      Any tags related to the event, if applicable.

    • Array [
    • value string required

      Possible values: >= 2 characters and <= 50 characters

      The value of the tag

    • type string required

      Possible values: >= 2 characters and <= 50 characters

      Default value: simple

      The type of the tag

    • ]
    • ]
    • totalEvents integer

      The total count of events

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-feature-environment.html b/reference/api/unleash/get-feature-environment.html index 8208ec96fb..7ccdd61daa 100644 --- a/reference/api/unleash/get-feature-environment.html +++ b/reference/api/unleash/get-feature-environment.html @@ -20,15 +20,15 @@ - - + +

    Get a feature environment

    GET /api/admin/projects/:projectId/features/:featureName/environments/:environment

    Information about the enablement status and strategies for a feature toggle in specified environment.

    Request

    Path Parameters

    • projectId string required
    • featureName string required
    • environment string required
    Responses

    featureEnvironmentSchema

    Schema
    • name string required

      The name of the environment

    • featureName string

      The name of the feature

    • environment string

      The name of the environment

    • type string

      The type of the environment

    • enabled boolean required

      true if the feature is enabled for the environment, otherwise false.

    • sortOrder number

      The sort order of the feature environment in the feature environments list

    • variantCount number

      The number of defined variants

    • strategies object[]

      A list of activation strategies for the feature environment

    • Array [
    • id string

      A uuid for the feature strategy

    • name string required

      The name or type of strategy

    • title string nullable

      A descriptive title for the strategy

    • disabled boolean nullable

      A toggle to disable the strategy. defaults to false. Disabled strategies are not evaluated or returned to the SDKs

    • featureName string

      The name or feature the strategy is attached to

    • sortOrder number

      The order of the strategy in the list

    • segments number[]

      A list of segment ids attached to the strategy

    • constraints object[]

      A list of the constraints attached to the strategy. See https://docs.getunleash.io/reference/strategy-constraints

    • Array [
    • contextName string required

      The name of the context field that this constraint should apply to.

    • operator string required

      Possible values: [NOT_IN, IN, STR_ENDS_WITH, STR_STARTS_WITH, STR_CONTAINS, NUM_EQ, NUM_GT, NUM_GTE, NUM_LT, NUM_LTE, DATE_AFTER, DATE_BEFORE, SEMVER_EQ, SEMVER_GT, SEMVER_LT]

      The operator to use when evaluating this constraint. For more information about the various operators, refer to the strategy constraint operator documentation.

    • caseInsensitive boolean

      Default value: false

      Whether the operator should be case sensitive or not. Defaults to false (being case sensitive).

    • inverted boolean

      Default value: false

      Whether the result should be negated or not. If true, will turn a true result into a false result and vice versa.

    • values string[]

      The context values that should be used for constraint evaluation. Use this property instead of value for properties that accept multiple values.

    • value string

      The context value that should be used for constraint evaluation. Use this property instead of values for properties that only accept single values.

    • ]
    • variants object[]

      Strategy level variants

    • Array [
    • name string required

      The variant name. Must be unique for this feature toggle

    • weight integer required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is an integer between 0 and 1000. See the section on variant weights for more information

    • weightType string required

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000. Refer to the variant weight documentation.

    • stickiness string required

      The stickiness to use for distribution of this variant. Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • ]
    • parameters object

      A list of parameters for a strategy

    • property name* string
    • ]
    • variants object[]

      A list of variants for the feature environment

    • Array [
    • name string required

      The variants name. Is unique for this feature toggle

    • weight number required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is a number between 0 and 1000. See the section on variant weights for more information

    • weightType string

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000

    • stickiness string

      Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • overrides object[]

      Overrides assigning specific variants to specific users. The weighting system automatically assigns users to specific groups for you, but any overrides in this list will take precedence.

    • Array [
    • contextName string required

      The name of the context field used to determine overrides

    • values string[] required

      Which values that should be overriden

    • ]
    • ]
    • lastSeenAt date-time nullable

      The date when metrics where last collected for the feature environment

    • hasStrategies boolean

      Whether the feature has any strategies defined.

    • hasEnabledStrategies boolean

      Whether the feature has any enabled strategies defined.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-feature-strategies.html b/reference/api/unleash/get-feature-strategies.html index 0eb9bb2b1a..239be99564 100644 --- a/reference/api/unleash/get-feature-strategies.html +++ b/reference/api/unleash/get-feature-strategies.html @@ -20,15 +20,15 @@ - - + +

    Get feature toggle strategies

    GET /api/admin/projects/:projectId/features/:featureName/environments/:environment/strategies

    Get strategies defined for a feature toggle in the specified environment.

    Request

    Path Parameters

    • projectId string required
    • featureName string required
    • environment string required
    Responses

    featureStrategySchema

    Schema
    • id string

      A uuid for the feature strategy

    • name string required

      The name or type of strategy

    • title string nullable

      A descriptive title for the strategy

    • disabled boolean nullable

      A toggle to disable the strategy. defaults to false. Disabled strategies are not evaluated or returned to the SDKs

    • featureName string

      The name or feature the strategy is attached to

    • sortOrder number

      The order of the strategy in the list

    • segments number[]

      A list of segment ids attached to the strategy

    • constraints object[]

      A list of the constraints attached to the strategy. See https://docs.getunleash.io/reference/strategy-constraints

    • Array [
    • contextName string required

      The name of the context field that this constraint should apply to.

    • operator string required

      Possible values: [NOT_IN, IN, STR_ENDS_WITH, STR_STARTS_WITH, STR_CONTAINS, NUM_EQ, NUM_GT, NUM_GTE, NUM_LT, NUM_LTE, DATE_AFTER, DATE_BEFORE, SEMVER_EQ, SEMVER_GT, SEMVER_LT]

      The operator to use when evaluating this constraint. For more information about the various operators, refer to the strategy constraint operator documentation.

    • caseInsensitive boolean

      Default value: false

      Whether the operator should be case sensitive or not. Defaults to false (being case sensitive).

    • inverted boolean

      Default value: false

      Whether the result should be negated or not. If true, will turn a true result into a false result and vice versa.

    • values string[]

      The context values that should be used for constraint evaluation. Use this property instead of value for properties that accept multiple values.

    • value string

      The context value that should be used for constraint evaluation. Use this property instead of values for properties that only accept single values.

    • ]
    • variants object[]

      Strategy level variants

    • Array [
    • name string required

      The variant name. Must be unique for this feature toggle

    • weight integer required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is an integer between 0 and 1000. See the section on variant weights for more information

    • weightType string required

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000. Refer to the variant weight documentation.

    • stickiness string required

      The stickiness to use for distribution of this variant. Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • ]
    • parameters object

      A list of parameters for a strategy

    • property name* string
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-feature-strategy.html b/reference/api/unleash/get-feature-strategy.html index 15cd7b024f..61e670e362 100644 --- a/reference/api/unleash/get-feature-strategy.html +++ b/reference/api/unleash/get-feature-strategy.html @@ -20,15 +20,15 @@ - - + +

    Get a strategy configuration

    GET /api/admin/projects/:projectId/features/:featureName/environments/:environment/strategies/:strategyId

    Get a strategy configuration for an environment in a feature toggle.

    Request

    Path Parameters

    • projectId string required
    • featureName string required
    • environment string required
    • strategyId string required
    Responses

    featureStrategySchema

    Schema
    • id string

      A uuid for the feature strategy

    • name string required

      The name or type of strategy

    • title string nullable

      A descriptive title for the strategy

    • disabled boolean nullable

      A toggle to disable the strategy. defaults to false. Disabled strategies are not evaluated or returned to the SDKs

    • featureName string

      The name or feature the strategy is attached to

    • sortOrder number

      The order of the strategy in the list

    • segments number[]

      A list of segment ids attached to the strategy

    • constraints object[]

      A list of the constraints attached to the strategy. See https://docs.getunleash.io/reference/strategy-constraints

    • Array [
    • contextName string required

      The name of the context field that this constraint should apply to.

    • operator string required

      Possible values: [NOT_IN, IN, STR_ENDS_WITH, STR_STARTS_WITH, STR_CONTAINS, NUM_EQ, NUM_GT, NUM_GTE, NUM_LT, NUM_LTE, DATE_AFTER, DATE_BEFORE, SEMVER_EQ, SEMVER_GT, SEMVER_LT]

      The operator to use when evaluating this constraint. For more information about the various operators, refer to the strategy constraint operator documentation.

    • caseInsensitive boolean

      Default value: false

      Whether the operator should be case sensitive or not. Defaults to false (being case sensitive).

    • inverted boolean

      Default value: false

      Whether the result should be negated or not. If true, will turn a true result into a false result and vice versa.

    • values string[]

      The context values that should be used for constraint evaluation. Use this property instead of value for properties that accept multiple values.

    • value string

      The context value that should be used for constraint evaluation. Use this property instead of values for properties that only accept single values.

    • ]
    • variants object[]

      Strategy level variants

    • Array [
    • name string required

      The variant name. Must be unique for this feature toggle

    • weight integer required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is an integer between 0 and 1000. See the section on variant weights for more information

    • weightType string required

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000. Refer to the variant weight documentation.

    • stickiness string required

      The stickiness to use for distribution of this variant. Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • ]
    • parameters object

      A list of parameters for a strategy

    • property name* string
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-feature-usage-summary.html b/reference/api/unleash/get-feature-usage-summary.html index 6cf540088f..2031087a2b 100644 --- a/reference/api/unleash/get-feature-usage-summary.html +++ b/reference/api/unleash/get-feature-usage-summary.html @@ -20,15 +20,15 @@ - - + +

    Last hour of usage and a list of applications that have reported seeing this feature toggle

    GET /api/admin/client-metrics/features/:name

    Separate counts for yes (enabled), no (disabled), as well as how many times each variant was selected during the last hour

    Request

    Path Parameters

    • name string required
    Responses

    featureUsageSchema

    Schema
    • version integer required

      Possible values: >= 1

      The version of this schema

    • maturity string required

      The maturity level of this API (alpha, beta, stable, deprecated)

    • featureName string required

      The name of the feature

    • lastHourUsage object[]required

      Last hour statistics. Accumulated per feature per environment. Contains counts for evaluations to true (yes) and to false (no)

    • Array [
    • featureName string

      The name of the feature

    • appName string

      The name of the application the SDK is being used in

    • environment string required

      Which environment the SDK is being used in

    • timestamp objectrequired

      The start of the time window these metrics are valid for. The window is usually 1 hour wide

      oneOf
    • string date-time

      An RFC-3339-compliant timestamp.

    • yes integer required

      How many times the toggle evaluated to true

    • no integer required

      How many times the toggle evaluated to false

    • variants object

      How many times each variant was returned

    • property name* integer
    • ]
    • seenApplications string[] required

      A list of applications seen using this feature

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-feature-variants.html b/reference/api/unleash/get-feature-variants.html index 668b5cccec..17b4b321cf 100644 --- a/reference/api/unleash/get-feature-variants.html +++ b/reference/api/unleash/get-feature-variants.html @@ -20,15 +20,15 @@ - - + +

    Retrieve variants for a feature (deprecated)

    GET /api/admin/projects/:projectId/features/:featureName/variants
    deprecated

    This endpoint has been deprecated and may be removed in future versions of the API.

    (deprecated from 4.21) Retrieve the variants for the specified feature. From Unleash 4.21 onwards, this endpoint will attempt to choose a production-type environment as the source of truth. If more than one production environment is found, the first one will be used.

    Request

    Path Parameters

    • projectId string required
    • featureName string required
    Responses

    featureVariantsSchema

    Schema
    • version integer required

      The version of the feature variants schema.

    • variants object[]required

      All variants defined for a specific feature toggle.

    • Array [
    • name string required

      The variants name. Is unique for this feature toggle

    • weight number required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is a number between 0 and 1000. See the section on variant weights for more information

    • weightType string

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000

    • stickiness string

      Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • overrides object[]

      Overrides assigning specific variants to specific users. The weighting system automatically assigns users to specific groups for you, but any overrides in this list will take precedence.

    • Array [
    • contextName string required

      The name of the context field used to determine overrides

    • values string[] required

      Which values that should be overriden

    • ]
    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-feature.html b/reference/api/unleash/get-feature.html index 20b973b2cd..dae4bed877 100644 --- a/reference/api/unleash/get-feature.html +++ b/reference/api/unleash/get-feature.html @@ -20,15 +20,15 @@ - - + +

    Get a feature

    GET /api/admin/projects/:projectId/features/:featureName

    This endpoint returns the information about the requested feature if the feature belongs to the specified project.

    Request

    Path Parameters

    • projectId string required
    • featureName string required
    Responses

    featureSchema

    Schema
    • name string required

      Unique feature name

    • type string

      Type of the toggle e.g. experiment, kill-switch, release, operational, permission

    • description string nullable

      Detailed description of the feature

    • archived boolean

      true if the feature is archived

    • project string

      Name of the project the feature belongs to

    • enabled boolean

      true if the feature is enabled, otherwise false.

    • stale boolean

      true if the feature is stale based on the age and feature type, otherwise false.

    • favorite boolean

      true if the feature was favorited, otherwise false.

    • impressionData boolean

      true if the impression data collection is enabled for the feature, otherwise false.

    • createdAt date-time nullable

      The date the feature was created

    • archivedAt date-time nullable

      The date the feature was archived

    • lastSeenAt date-time nullable deprecated

      The date when metrics where last collected for the feature. This field is deprecated, use the one in featureEnvironmentSchema

    • environments object[]

      The list of environments where the feature can be used

    • Array [
    • name string required

      The name of the environment

    • featureName string

      The name of the feature

    • environment string

      The name of the environment

    • type string

      The type of the environment

    • enabled boolean required

      true if the feature is enabled for the environment, otherwise false.

    • sortOrder number

      The sort order of the feature environment in the feature environments list

    • variantCount number

      The number of defined variants

    • strategies object[]

      A list of activation strategies for the feature environment

    • Array [
    • id string

      A uuid for the feature strategy

    • name string required

      The name or type of strategy

    • title string nullable

      A descriptive title for the strategy

    • disabled boolean nullable

      A toggle to disable the strategy. defaults to false. Disabled strategies are not evaluated or returned to the SDKs

    • featureName string

      The name or feature the strategy is attached to

    • sortOrder number

      The order of the strategy in the list

    • segments number[]

      A list of segment ids attached to the strategy

    • constraints object[]

      A list of the constraints attached to the strategy. See https://docs.getunleash.io/reference/strategy-constraints

    • Array [
    • contextName string required

      The name of the context field that this constraint should apply to.

    • operator string required

      Possible values: [NOT_IN, IN, STR_ENDS_WITH, STR_STARTS_WITH, STR_CONTAINS, NUM_EQ, NUM_GT, NUM_GTE, NUM_LT, NUM_LTE, DATE_AFTER, DATE_BEFORE, SEMVER_EQ, SEMVER_GT, SEMVER_LT]

      The operator to use when evaluating this constraint. For more information about the various operators, refer to the strategy constraint operator documentation.

    • caseInsensitive boolean

      Default value: false

      Whether the operator should be case sensitive or not. Defaults to false (being case sensitive).

    • inverted boolean

      Default value: false

      Whether the result should be negated or not. If true, will turn a true result into a false result and vice versa.

    • values string[]

      The context values that should be used for constraint evaluation. Use this property instead of value for properties that accept multiple values.

    • value string

      The context value that should be used for constraint evaluation. Use this property instead of values for properties that only accept single values.

    • ]
    • variants object[]

      Strategy level variants

    • Array [
    • name string required

      The variant name. Must be unique for this feature toggle

    • weight integer required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is an integer between 0 and 1000. See the section on variant weights for more information

    • weightType string required

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000. Refer to the variant weight documentation.

    • stickiness string required

      The stickiness to use for distribution of this variant. Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • ]
    • parameters object

      A list of parameters for a strategy

    • property name* string
    • ]
    • variants object[]

      A list of variants for the feature environment

    • Array [
    • name string required

      The variants name. Is unique for this feature toggle

    • weight number required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is a number between 0 and 1000. See the section on variant weights for more information

    • weightType string

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000

    • stickiness string

      Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • overrides object[]

      Overrides assigning specific variants to specific users. The weighting system automatically assigns users to specific groups for you, but any overrides in this list will take precedence.

    • Array [
    • contextName string required

      The name of the context field used to determine overrides

    • values string[] required

      Which values that should be overriden

    • ]
    • ]
    • lastSeenAt date-time nullable

      The date when metrics where last collected for the feature environment

    • hasStrategies boolean

      Whether the feature has any strategies defined.

    • hasEnabledStrategies boolean

      Whether the feature has any enabled strategies defined.

    • ]
    • variants object[]deprecated

      The list of feature variants

    • Array [
    • name string required

      The variants name. Is unique for this feature toggle

    • weight number required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is a number between 0 and 1000. See the section on variant weights for more information

    • weightType string

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000

    • stickiness string

      Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • overrides object[]

      Overrides assigning specific variants to specific users. The weighting system automatically assigns users to specific groups for you, but any overrides in this list will take precedence.

    • Array [
    • contextName string required

      The name of the context field used to determine overrides

    • values string[] required

      Which values that should be overriden

    • ]
    • ]
    • strategies object[] deprecated

      This is a legacy field that will be deprecated

    • tags object[]nullable

      The list of feature tags

    • Array [
    • value string required

      Possible values: >= 2 characters and <= 50 characters

      The value of the tag

    • type string required

      Possible values: >= 2 characters and <= 50 characters

      Default value: simple

      The type of the tag

    • ]
    • children string[]

      The list of child feature names. This is an experimental field and may change.

    • dependencies object[]

      The list of parent dependencies. This is an experimental field and may change.

    • Array [
    • feature string required

      The name of the parent feature

    • enabled boolean

      Whether the parent feature is enabled or not

    • variants string[]

      The list of variants the parent feature should resolve to. Only valid when feature is enabled.

    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-features.html b/reference/api/unleash/get-features.html index 70b8fc4304..998e695f94 100644 --- a/reference/api/unleash/get-features.html +++ b/reference/api/unleash/get-features.html @@ -20,15 +20,15 @@ - - + +

    Get all features in a project

    GET /api/admin/projects/:projectId/features

    A list of all features for the specified project.

    Request

    Path Parameters

    • projectId string required
    Responses

    featuresSchema

    Schema
    • version integer required

      The version of the feature's schema

    • features object[]required

      A list of features

    • Array [
    • name string required

      Unique feature name

    • type string

      Type of the toggle e.g. experiment, kill-switch, release, operational, permission

    • description string nullable

      Detailed description of the feature

    • archived boolean

      true if the feature is archived

    • project string

      Name of the project the feature belongs to

    • enabled boolean

      true if the feature is enabled, otherwise false.

    • stale boolean

      true if the feature is stale based on the age and feature type, otherwise false.

    • favorite boolean

      true if the feature was favorited, otherwise false.

    • impressionData boolean

      true if the impression data collection is enabled for the feature, otherwise false.

    • createdAt date-time nullable

      The date the feature was created

    • archivedAt date-time nullable

      The date the feature was archived

    • lastSeenAt date-time nullable deprecated

      The date when metrics where last collected for the feature. This field is deprecated, use the one in featureEnvironmentSchema

    • environments object[]

      The list of environments where the feature can be used

    • Array [
    • name string required

      The name of the environment

    • featureName string

      The name of the feature

    • environment string

      The name of the environment

    • type string

      The type of the environment

    • enabled boolean required

      true if the feature is enabled for the environment, otherwise false.

    • sortOrder number

      The sort order of the feature environment in the feature environments list

    • variantCount number

      The number of defined variants

    • strategies object[]

      A list of activation strategies for the feature environment

    • Array [
    • id string

      A uuid for the feature strategy

    • name string required

      The name or type of strategy

    • title string nullable

      A descriptive title for the strategy

    • disabled boolean nullable

      A toggle to disable the strategy. defaults to false. Disabled strategies are not evaluated or returned to the SDKs

    • featureName string

      The name or feature the strategy is attached to

    • sortOrder number

      The order of the strategy in the list

    • segments number[]

      A list of segment ids attached to the strategy

    • constraints object[]

      A list of the constraints attached to the strategy. See https://docs.getunleash.io/reference/strategy-constraints

    • Array [
    • contextName string required

      The name of the context field that this constraint should apply to.

    • operator string required

      Possible values: [NOT_IN, IN, STR_ENDS_WITH, STR_STARTS_WITH, STR_CONTAINS, NUM_EQ, NUM_GT, NUM_GTE, NUM_LT, NUM_LTE, DATE_AFTER, DATE_BEFORE, SEMVER_EQ, SEMVER_GT, SEMVER_LT]

      The operator to use when evaluating this constraint. For more information about the various operators, refer to the strategy constraint operator documentation.

    • caseInsensitive boolean

      Default value: false

      Whether the operator should be case sensitive or not. Defaults to false (being case sensitive).

    • inverted boolean

      Default value: false

      Whether the result should be negated or not. If true, will turn a true result into a false result and vice versa.

    • values string[]

      The context values that should be used for constraint evaluation. Use this property instead of value for properties that accept multiple values.

    • value string

      The context value that should be used for constraint evaluation. Use this property instead of values for properties that only accept single values.

    • ]
    • variants object[]

      Strategy level variants

    • Array [
    • name string required

      The variant name. Must be unique for this feature toggle

    • weight integer required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is an integer between 0 and 1000. See the section on variant weights for more information

    • weightType string required

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000. Refer to the variant weight documentation.

    • stickiness string required

      The stickiness to use for distribution of this variant. Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • ]
    • parameters object

      A list of parameters for a strategy

    • property name* string
    • ]
    • variants object[]

      A list of variants for the feature environment

    • Array [
    • name string required

      The variants name. Is unique for this feature toggle

    • weight number required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is a number between 0 and 1000. See the section on variant weights for more information

    • weightType string

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000

    • stickiness string

      Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • overrides object[]

      Overrides assigning specific variants to specific users. The weighting system automatically assigns users to specific groups for you, but any overrides in this list will take precedence.

    • Array [
    • contextName string required

      The name of the context field used to determine overrides

    • values string[] required

      Which values that should be overriden

    • ]
    • ]
    • lastSeenAt date-time nullable

      The date when metrics where last collected for the feature environment

    • hasStrategies boolean

      Whether the feature has any strategies defined.

    • hasEnabledStrategies boolean

      Whether the feature has any enabled strategies defined.

    • ]
    • variants object[]deprecated

      The list of feature variants

    • Array [
    • name string required

      The variants name. Is unique for this feature toggle

    • weight number required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is a number between 0 and 1000. See the section on variant weights for more information

    • weightType string

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000

    • stickiness string

      Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • overrides object[]

      Overrides assigning specific variants to specific users. The weighting system automatically assigns users to specific groups for you, but any overrides in this list will take precedence.

    • Array [
    • contextName string required

      The name of the context field used to determine overrides

    • values string[] required

      Which values that should be overriden

    • ]
    • ]
    • strategies object[] deprecated

      This is a legacy field that will be deprecated

    • tags object[]nullable

      The list of feature tags

    • Array [
    • value string required

      Possible values: >= 2 characters and <= 50 characters

      The value of the tag

    • type string required

      Possible values: >= 2 characters and <= 50 characters

      Default value: simple

      The type of the tag

    • ]
    • children string[]

      The list of child feature names. This is an experimental field and may change.

    • dependencies object[]

      The list of parent dependencies. This is an experimental field and may change.

    • Array [
    • feature string required

      The name of the parent feature

    • enabled boolean

      Whether the parent feature is enabled or not

    • variants string[]

      The list of variants the parent feature should resolve to. Only valid when feature is enabled.

    • ]
    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-feedback.html b/reference/api/unleash/get-feedback.html index bca362f363..f2104a3377 100644 --- a/reference/api/unleash/get-feedback.html +++ b/reference/api/unleash/get-feedback.html @@ -20,15 +20,15 @@ - - + +

    Get all feedback events.

    GET /api/admin/feedback

    Get all feedback events.

    Request

    Responses

    feedbackListSchema

    Schema
    • Array [
    • id number required

      The unique identifier of the feedback.

    • createdAt date-time required

      The date and time when the feedback was provided.

    • category string required

      The category of the feedback.

    • userType string nullable required

      The type of user providing the feedback.

    • difficultyScore number nullable required

      A score indicating the difficulty experienced by the user.

    • positive string nullable required

      This field is for users to mention what they liked.

    • areasForImprovement string nullable required

      Details aspects of the service or product that could benefit from enhancements or modifications. Aids in pinpointing areas needing attention for improvement.

    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-frontend-features.html b/reference/api/unleash/get-frontend-features.html index 972a17d335..0401e7b9ba 100644 --- a/reference/api/unleash/get-frontend-features.html +++ b/reference/api/unleash/get-frontend-features.html @@ -20,15 +20,15 @@ - - + +

    Retrieve enabled feature toggles for the provided context.

    GET /api/frontend

    This endpoint returns the list of feature toggles that the proxy evaluates to enabled for the given context. Context values are provided as query parameters. If the Frontend API is disabled 404 is returned.

    Request

    Responses

    proxyFeaturesSchema

    Schema
    • toggles object[]required

      The actual features returned to the Frontend SDK

    • Array [
    • name string required

      Unique feature name.

    • enabled boolean required

      Always set to true.

    • impressionData boolean required

      true if the impression data collection is enabled for the feature, otherwise false.

    • variant object

      Variant details

    • name string required

      The variants name. Is unique for this feature toggle

    • enabled boolean required

      Whether the variant is enabled or not.

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string]

      The format of the payload.

    • value string required

      The payload value stringified.

    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-google-settings.html b/reference/api/unleash/get-google-settings.html index c011f9dc2b..73b2110410 100644 --- a/reference/api/unleash/get-google-settings.html +++ b/reference/api/unleash/get-google-settings.html @@ -20,15 +20,15 @@ - - + +

    Get Google auth settings

    GET /api/admin/auth/google/settings
    deprecated

    This endpoint has been deprecated and may be removed in future versions of the API.

    Returns the current settings for Google Authentication (deprecated, please use OpenID instead)

    Request

    Responses

    googleSettingsSchema

    Schema
    • enabled boolean

      Is google OIDC enabled

    • clientId string required

      The google client id, used to authenticate against google

    • clientSecret string required

      The client secret used to authenticate the OAuth session used to log the user in

    • unleashHostname string required

      Name of the host allowed to access the Google authentication flow

    • autoCreate boolean

      Should Unleash create users based on the emails coming back in the authentication reply from Google

    • emailDomains string

      A comma separated list of email domains that Unleash will auto create user accounts for.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-group.html b/reference/api/unleash/get-group.html index 9f77e41c14..26ce10df86 100644 --- a/reference/api/unleash/get-group.html +++ b/reference/api/unleash/get-group.html @@ -20,15 +20,15 @@ - - + +

    Get a single group

    GET /api/admin/groups/:groupId

    Get a single user group by group id

    Request

    Path Parameters

    • groupId string required
    Responses

    groupSchema

    Schema
    • id integer

      The group id

    • name string required

      The name of the group

    • description string nullable

      A custom description of the group

    • mappingsSSO string[]

      A list of SSO groups that should map to this Unleash group

    • rootRole number nullable

      A role id that is used as the root role for all users in this group. This can be either the id of the Viewer, Editor or Admin role.

    • createdBy string nullable

      A user who created this group

    • createdAt date-time nullable

      When was this group created

    • users object[]

      A list of users belonging to this group

    • Array [
    • joinedAt date-time

      The date when the user joined the group

    • createdBy string nullable

      The username of the user who added this user to this group

    • user objectrequired

      An Unleash user

    • id integer required

      The user id

    • isAPI boolean deprecated

      (Deprecated): Used internally to know which operations the user should be allowed to perform

    • name string nullable

      Name of the user

    • email string

      Email of the user

    • username string nullable

      A unique username for the user

    • imageUrl string

      URL used for the userprofile image

    • inviteLink string

      If the user is actively inviting other users, this is the link that can be shared with other users

    • loginAttempts integer

      How many unsuccessful attempts at logging in has the user made

    • emailSent boolean

      Is the welcome email sent to the user or not

    • rootRole integer

      Which root role this user is assigned

    • seenAt date-time nullable

      The last time this user logged in

    • createdAt date-time

      The user was created at this time

    • accountType string

      A user is either an actual User or a Service Account

    • permissions string[]

      Deprecated

    • ]
    • projects string[]

      A list of projects where this group is used

    • userCount integer

      The number of users that belong to this group

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-groups.html b/reference/api/unleash/get-groups.html index 2e90a0902a..aaf6472b85 100644 --- a/reference/api/unleash/get-groups.html +++ b/reference/api/unleash/get-groups.html @@ -20,15 +20,15 @@ - - + +

    Get a list of groups

    GET /api/admin/groups

    Get a list of user groups for Role-Based Access Control

    Request

    Responses

    groupsSchema

    Schema
    • groups object[]

      A list of groups

    • Array [
    • id integer

      The group id

    • name string required

      The name of the group

    • description string nullable

      A custom description of the group

    • mappingsSSO string[]

      A list of SSO groups that should map to this Unleash group

    • rootRole number nullable

      A role id that is used as the root role for all users in this group. This can be either the id of the Viewer, Editor or Admin role.

    • createdBy string nullable

      A user who created this group

    • createdAt date-time nullable

      When was this group created

    • users object[]

      A list of users belonging to this group

    • Array [
    • joinedAt date-time

      The date when the user joined the group

    • createdBy string nullable

      The username of the user who added this user to this group

    • user objectrequired

      An Unleash user

    • id integer required

      The user id

    • isAPI boolean deprecated

      (Deprecated): Used internally to know which operations the user should be allowed to perform

    • name string nullable

      Name of the user

    • email string

      Email of the user

    • username string nullable

      A unique username for the user

    • imageUrl string

      URL used for the userprofile image

    • inviteLink string

      If the user is actively inviting other users, this is the link that can be shared with other users

    • loginAttempts integer

      How many unsuccessful attempts at logging in has the user made

    • emailSent boolean

      Is the welcome email sent to the user or not

    • rootRole integer

      Which root role this user is assigned

    • seenAt date-time nullable

      The last time this user logged in

    • createdAt date-time

      The user was created at this time

    • accountType string

      A user is either an actual User or a Service Account

    • permissions string[]

      Deprecated

    • ]
    • projects string[]

      A list of projects where this group is used

    • userCount integer

      The number of users that belong to this group

    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-health.html b/reference/api/unleash/get-health.html index ebca847f52..4391c2181d 100644 --- a/reference/api/unleash/get-health.html +++ b/reference/api/unleash/get-health.html @@ -20,15 +20,15 @@ - - + +

    Get instance operational status

    GET /health

    This operation returns information about whether this Unleash instance is healthy and ready to serve requests or not. Typically used by your deployment orchestrator (e.g. Kubernetes, Docker Swarm, Mesos, et al.).

    Request

    Responses

    healthCheckSchema

    Schema
    • health string required

      Possible values: [GOOD, BAD]

      The state this Unleash instance is in. GOOD if everything is ok, BAD if the instance should be restarted

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-instance-admin-stats-csv.html b/reference/api/unleash/get-instance-admin-stats-csv.html index d8e52561a8..1d43a6c5ca 100644 --- a/reference/api/unleash/get-instance-admin-stats-csv.html +++ b/reference/api/unleash/get-instance-admin-stats-csv.html @@ -20,15 +20,15 @@ - - + +

    Instance usage statistics

    GET /api/admin/instance-admin/statistics/csv

    Provides statistics about various features of Unleash to allow for reporting of usage for self-hosted customers. The response contains data such as the number of users, groups, features, strategies, versions, etc.

    Request

    Responses

    instanceAdminStatsSchemaCsv

    Schema
    • string
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-instance-admin-stats.html b/reference/api/unleash/get-instance-admin-stats.html index 2209761590..e277c21bfc 100644 --- a/reference/api/unleash/get-instance-admin-stats.html +++ b/reference/api/unleash/get-instance-admin-stats.html @@ -20,15 +20,15 @@ - - + +

    Instance usage statistics

    GET /api/admin/instance-admin/statistics
    deprecated

    This endpoint has been deprecated and may be removed in future versions of the API.

    Provides statistics about various features of Unleash to allow for reporting of usage for self-hosted customers. The response contains data such as the number of users, groups, features, strategies, versions, etc.

    Request

    Responses

    instanceAdminStatsSchema

    Schema
    • instanceId string required

      A unique identifier for this instance. Generated by the database migration scripts at first run. Typically a UUID.

    • timestamp date-time nullable

      When these statistics were produced

    • versionOSS string

      The version of Unleash OSS that is bundled in this instance

    • versionEnterprise string

      The version of Unleash Enterprise that is bundled in this instance

    • users number

      The number of users this instance has

    • activeUsers object

      The number of active users in the last 7, 30 and 90 days

    • last7 number

      The number of active users in the last 7 days

    • last30 number

      The number of active users in the last 30 days

    • last60 number

      The number of active users in the last 60 days

    • last90 number

      The number of active users in the last 90 days

    • productionChanges object

      The number of changes to the production environment in the last 30, 60 and 90 days

    • last30 number

      The number of changes in production in the last 30 days

    • last60 number

      The number of changes in production in the last 60 days

    • last90 number

      The number of changes in production in the last 90 days

    • featureToggles number

      The number of feature-toggles this instance has

    • projects number

      The number of projects defined in this instance.

    • contextFields number

      The number of context fields defined in this instance.

    • roles number

      The number of roles defined in this instance

    • groups number

      The number of groups defined in this instance

    • environments number

      The number of environments defined in this instance

    • segments number

      The number of segments defined in this instance

    • strategies number

      The number of strategies defined in this instance

    • SAMLenabled boolean

      Whether or not SAML authentication is enabled for this instance

    • OIDCenabled boolean

      Whether or not OIDC authentication is enabled for this instance

    • clientApps object[]

      A count of connected applications in the last week, last month and all time since last restart

    • Array [
    • range string

      Possible values: [allTime, 30d, 7d]

      A description of a time range

    • count number

      The number of client applications that have been observed in this period

    • ]
    • featureExports number

      The number of export operations on this instance

    • featureImports number

      The number of import operations on this instance

    • sum string

      A SHA-256 checksum of the instance statistics to be used to verify that the data in this object has not been tampered with

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-login-history.html b/reference/api/unleash/get-login-history.html index d361501776..a128646fda 100644 --- a/reference/api/unleash/get-login-history.html +++ b/reference/api/unleash/get-login-history.html @@ -20,15 +20,15 @@ - - + +

    Get all login events.

    GET /api/admin/logins

    Returns all login events in the Unleash system. You can optionally get them in CSV format by specifying the Accept header as text/csv.

    Request

    Responses

    loginHistorySchema

    Schema
    • events object[]required

      A list of login events

    • Array [
    • id integer required

      Possible values: >= 1

      The event's ID. Event IDs are incrementing integers. In other words, a more recent event will always have a higher ID than an older event.

    • username string

      The username of the user that attempted to log in. Will return "Incorrectly configured provider" when attempting to log in using a misconfigured provider.

    • auth_type string

      The authentication type used to log in.

    • created_at date-time

      The date and time of when the login was attempted.

    • successful boolean

      Whether the login was successful or not.

    • ip string nullable

      The IP address of the client that attempted to log in.

    • failure_reason string nullable

      The reason for the login failure. This property is only present if the login was unsuccessful.

    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-maintenance.html b/reference/api/unleash/get-maintenance.html index 90ea0ed16a..121768c1c0 100644 --- a/reference/api/unleash/get-maintenance.html +++ b/reference/api/unleash/get-maintenance.html @@ -20,15 +20,15 @@ - - + +

    Get maintenance mode status

    GET /api/admin/maintenance

    Tells you whether maintenance mode is enabled or disabled

    Request

    Responses

    maintenanceSchema

    Schema
    • enabled boolean required

      Whether maintenance mode is enabled or not.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-me.html b/reference/api/unleash/get-me.html index 9bcda424fd..b3ebe13758 100644 --- a/reference/api/unleash/get-me.html +++ b/reference/api/unleash/get-me.html @@ -20,15 +20,15 @@ - - + +

    Get your own user details

    GET /api/admin/user

    Detailed information about the current user, user permissions and user feedback

    Request

    Responses

    meSchema

    Schema
    • user objectrequired

      An Unleash user

    • id integer required

      The user id

    • isAPI boolean deprecated

      (Deprecated): Used internally to know which operations the user should be allowed to perform

    • name string nullable

      Name of the user

    • email string

      Email of the user

    • username string nullable

      A unique username for the user

    • imageUrl string

      URL used for the userprofile image

    • inviteLink string

      If the user is actively inviting other users, this is the link that can be shared with other users

    • loginAttempts integer

      How many unsuccessful attempts at logging in has the user made

    • emailSent boolean

      Is the welcome email sent to the user or not

    • rootRole integer

      Which root role this user is assigned

    • seenAt date-time nullable

      The last time this user logged in

    • createdAt date-time

      The user was created at this time

    • accountType string

      A user is either an actual User or a Service Account

    • permissions string[]

      Deprecated

    • permissions object[]required

      User permissions for projects and environments

    • Array [
    • permission string required

      Project or environment permission name

    • project string

      The project this permission applies to

    • environment string

      The environment this permission applies to

    • ]
    • feedback object[]required

      User feedback information

    • Array [
    • userId integer

      The ID of the user that gave the feedback.

    • neverShow boolean

      true if the user has asked never to see this feedback questionnaire again.

    • given date-time nullable

      When this feedback was given

    • feedbackId string

      The name of the feedback session

    • ]
    • splash objectrequired

      Splash screen configuration

    • property name* boolean
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-notifications.html b/reference/api/unleash/get-notifications.html index 8a6d0370e6..a5eb7a1d45 100644 --- a/reference/api/unleash/get-notifications.html +++ b/reference/api/unleash/get-notifications.html @@ -20,15 +20,15 @@ - - + +

    Retrieves a list of notifications

    GET /api/admin/notifications

    A user may get relevant notifications from the projects they are a member of

    Request

    Responses

    notificationsSchema

    Schema
    • Array [
    • id number required

      The id of this notification

    • message string required

      The actual notification message

    • link string required

      The link to change request or feature toggle the notification refers to

    • notificationType required

      Possible values: [change-request, toggle]

      The type of the notification used e.g. for the graphical hints

    • createdBy objectrequired
    • username string nullable

      The name of the user who triggered the notification

    • imageUrl string nullable

      The avatar url of the user who triggered the notification

    • createdAt date-time required

      The date and time when the notification was created

    • readAt date-time nullable required

      The date and time when the notification was read or marked as read, otherwise null

    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-oidc-settings.html b/reference/api/unleash/get-oidc-settings.html index 03db55a4ba..7897beda51 100644 --- a/reference/api/unleash/get-oidc-settings.html +++ b/reference/api/unleash/get-oidc-settings.html @@ -20,8 +20,8 @@ - - + + @@ -29,7 +29,7 @@

    Get OIDC auth settings

    GET /api/admin/auth/oidc/settings

    Returns the current settings for OIDC Authentication

    Request

    Responses

    oidcSettingsSchema

    Schema
    • enabled boolean

      true if OpenID connect is turned on for this instance, otherwise false

    • discoverUrl string
    • clientId string required

      The OIDC client ID of this application.

    • secret string required

      Shared secret from OpenID server. Used to authenticate login requests

    • autoCreate boolean

      Auto create users based on email addresses from login tokens

    • enableSingleSignOut boolean

      Support Single sign out when user clicks logout in Unleash. If true user is signed out of all OpenID Connect sessions against the clientId they may have active

    • defaultRootRole string

      Possible values: [Viewer, Editor, Admin]

      Default role granted to users auto-created from email. Only relevant if autoCreate is true

    • emailDomains string

      Comma separated list of email domains that are automatically approved for an account in the server. Only relevant if autoCreate is true

    • acrValues string

      Authentication Context Class Reference, used to request extra values in the acr claim returned from the server. If multiple values are required, they should be space separated. Consult the OIDC reference for more information

    • idTokenSigningAlgorithm string

      Possible values: [RS256, RS384, RS512]

      The signing algorithm used to sign our token. Refer to the JWT signatures documentation for more information.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-open-change-requests-for-user.html b/reference/api/unleash/get-open-change-requests-for-user.html index 1392d12578..899b09164a 100644 --- a/reference/api/unleash/get-open-change-requests-for-user.html +++ b/reference/api/unleash/get-open-change-requests-for-user.html @@ -20,15 +20,15 @@ - - + +

    Retrieves pending change requests in configured environments

    GET /api/admin/projects/:projectId/change-requests/open

    This endpoint will retrieve the pending change requests in the configured environments for the project, for the current user performing the request.

    Request

    Path Parameters

    • projectId string required
    Responses

    changeRequestsSchema

    Schema
    • Array [
    • oneOf
    • id number required

      This change requests's ID.

    • title string

      A title describing the change request's content.

    • environment string required

      The environment in which the changes should be applied.

    • minApprovals number required

      The minimum number of approvals required before this change request can be applied.

    • project string required

      The project this change request belongs to.

    • features object[]required

      The list of features and their changes that relate to this change request.

    • Array [
    • name string required

      The name of the feature

    • conflict string

      A string describing the conflicts related to this change. Only present if there are any concflicts on the feature level.

    • changes object[]required

      List of changes inside change request. This list may be empty when listing all change requests for a project.

    • Array [
    • id number required

      The ID of this change.

    • action string required

      The kind of action that the change contains information about.

    • conflict string

      A description of the conflict caused by this change. Only present if there are any conflicts.

    • payload objectrequired

      The data required to perform this action.

      oneOf
    • string
    • createdBy object

      The user who created this change.

    • username string nullable

      The user's username.

    • imageUrl uri nullable

      The URL where the user's image can be found.

    • createdAt date-time

      When this change was suggested

    • ]
    • defaultChange object

      A description of a default change that will be applied with the change request to prevent invalid states.

      Default changes are changes that are applied in addition to explicit user-specified changes when a change request is applied. Any default changes are applied in the background and are not a real part of the change request.

    • action string required

      The kind of action this is.

    • payload object required

      The necessary data to perform this change.

    • ]
    • segments object[]required

      The list of segments and their changes that relate to this change request.

    • Array [
    • id number required

      The ID of this change.

    • action string required

      The kind of action that the change contains information about.

    • conflict string

      A description of the conflict caused by this change. Only present if there are any conflicts.

    • payload objectrequired

      The data required to perform this action.

      oneOf
    • string
    • createdBy object

      The user who created this change.

    • username string nullable

      The user's username.

    • imageUrl uri nullable

      The URL where the user's image can be found.

    • createdAt date-time

      When this change was suggested

    • name string required

      The current name of the segment

    • ]
    • approvals object[]

      A list of approvals that this change request has received.

    • Array [
    • createdBy objectrequired

      Information about the user who gave this approval.

    • id number

      The ID of the user who gave this approval.

    • username string

      The approving user's username.

    • imageUrl uri

      The URL where the user's image can be found.

    • createdAt date-time required

      When the approval was given.

    • ]
    • rejections object[]

      A list of rejections that this change request has received.

    • Array [
    • createdBy objectrequired

      Information about the user who gave this approval.

    • id number

      The ID of the user who gave this approval.

    • username string

      The approving user's username.

    • imageUrl uri

      The URL where the user's image can be found.

    • createdAt date-time required

      When the approval was given.

    • ]
    • comments object[]

      All comments that have been made on this change request.

    • Array [
    • id number

      The comment's ID. Unique per change request.

    • text string required

      The content of the comment.

    • createdBy objectrequired

      Information about the user who posted the comment

    • username string nullable

      The user's username.

    • imageUrl uri nullable

      The URL where the user's image can be found.

    • createdAt date-time required

      When the comment was made.

    • ]
    • createdBy objectrequired

      The user who created this change request.

    • username string nullable
    • imageUrl uri nullable

      The URL of the user's profile image.

    • createdAt date-time required

      When this change request was created.

    • state string required

      Possible values: [Draft, In review, Approved, Applied, Cancelled, Rejected]

      The current state of the change request.

    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-pats.html b/reference/api/unleash/get-pats.html index ddf4185182..4e4ad32292 100644 --- a/reference/api/unleash/get-pats.html +++ b/reference/api/unleash/get-pats.html @@ -20,15 +20,15 @@ - - + +

    Get all Personal Access Tokens for the current user.

    GET /api/admin/user/tokens

    Returns all of the Personal Access Tokens belonging to the current user.

    Request

    Responses

    patsSchema

    Schema
    • pats object[]

      A collection of Personal Access Tokens

    • Array [
    • id integer

      Possible values: >= 1

      The unique identification number for this Personal Access Token. (This property is set by Unleash when the token is created and cannot be set manually: if you provide a value when creating a PAT, Unleash will ignore it.)

    • secret string

      The token used for authentication. (This property is set by Unleash when the token is created and cannot be set manually: if you provide a value when creating a PAT, Unleash will ignore it.)

    • expiresAt date-time

      The token's expiration date.

    • createdAt date-time

      When the token was created. (This property is set by Unleash when the token is created and cannot be set manually: if you provide a value when creating a PAT, Unleash will ignore it.)

    • seenAt date-time nullable

      When the token was last seen/used to authenticate with. null if it has not been used yet. (This property is set by Unleash when the token is created and cannot be set manually: if you provide a value when creating a PAT, Unleash will ignore it.)

    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-pending-change-requests-for-feature.html b/reference/api/unleash/get-pending-change-requests-for-feature.html index c15b7f0c15..b70a4257ec 100644 --- a/reference/api/unleash/get-pending-change-requests-for-feature.html +++ b/reference/api/unleash/get-pending-change-requests-for-feature.html @@ -20,15 +20,15 @@ - - + +

    Retrieves all pending change requests referencing a feature in the project

    GET /api/admin/projects/:projectId/change-requests/pending/:featureName

    This endpoint will retrieve all pending change requests (change requests with a status of Draft | In review | Approved) referencing the given feature toggle name.

    Request

    Path Parameters

    • projectId string required
    • featureName string required
    Responses

    changeRequestsSchema

    Schema
    • Array [
    • oneOf
    • id number required

      This change requests's ID.

    • title string

      A title describing the change request's content.

    • environment string required

      The environment in which the changes should be applied.

    • minApprovals number required

      The minimum number of approvals required before this change request can be applied.

    • project string required

      The project this change request belongs to.

    • features object[]required

      The list of features and their changes that relate to this change request.

    • Array [
    • name string required

      The name of the feature

    • conflict string

      A string describing the conflicts related to this change. Only present if there are any concflicts on the feature level.

    • changes object[]required

      List of changes inside change request. This list may be empty when listing all change requests for a project.

    • Array [
    • id number required

      The ID of this change.

    • action string required

      The kind of action that the change contains information about.

    • conflict string

      A description of the conflict caused by this change. Only present if there are any conflicts.

    • payload objectrequired

      The data required to perform this action.

      oneOf
    • string
    • createdBy object

      The user who created this change.

    • username string nullable

      The user's username.

    • imageUrl uri nullable

      The URL where the user's image can be found.

    • createdAt date-time

      When this change was suggested

    • ]
    • defaultChange object

      A description of a default change that will be applied with the change request to prevent invalid states.

      Default changes are changes that are applied in addition to explicit user-specified changes when a change request is applied. Any default changes are applied in the background and are not a real part of the change request.

    • action string required

      The kind of action this is.

    • payload object required

      The necessary data to perform this change.

    • ]
    • segments object[]required

      The list of segments and their changes that relate to this change request.

    • Array [
    • id number required

      The ID of this change.

    • action string required

      The kind of action that the change contains information about.

    • conflict string

      A description of the conflict caused by this change. Only present if there are any conflicts.

    • payload objectrequired

      The data required to perform this action.

      oneOf
    • string
    • createdBy object

      The user who created this change.

    • username string nullable

      The user's username.

    • imageUrl uri nullable

      The URL where the user's image can be found.

    • createdAt date-time

      When this change was suggested

    • name string required

      The current name of the segment

    • ]
    • approvals object[]

      A list of approvals that this change request has received.

    • Array [
    • createdBy objectrequired

      Information about the user who gave this approval.

    • id number

      The ID of the user who gave this approval.

    • username string

      The approving user's username.

    • imageUrl uri

      The URL where the user's image can be found.

    • createdAt date-time required

      When the approval was given.

    • ]
    • rejections object[]

      A list of rejections that this change request has received.

    • Array [
    • createdBy objectrequired

      Information about the user who gave this approval.

    • id number

      The ID of the user who gave this approval.

    • username string

      The approving user's username.

    • imageUrl uri

      The URL where the user's image can be found.

    • createdAt date-time required

      When the approval was given.

    • ]
    • comments object[]

      All comments that have been made on this change request.

    • Array [
    • id number

      The comment's ID. Unique per change request.

    • text string required

      The content of the comment.

    • createdBy objectrequired

      Information about the user who posted the comment

    • username string nullable

      The user's username.

    • imageUrl uri nullable

      The URL where the user's image can be found.

    • createdAt date-time required

      When the comment was made.

    • ]
    • createdBy objectrequired

      The user who created this change request.

    • username string nullable
    • imageUrl uri nullable

      The URL of the user's profile image.

    • createdAt date-time required

      When this change request was created.

    • state string required

      Possible values: [Draft, In review, Approved, Applied, Cancelled, Rejected]

      The current state of the change request.

    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-pending-change-requests-for-user.html b/reference/api/unleash/get-pending-change-requests-for-user.html index eb99be1068..9ab0d764f7 100644 --- a/reference/api/unleash/get-pending-change-requests-for-user.html +++ b/reference/api/unleash/get-pending-change-requests-for-user.html @@ -20,15 +20,15 @@ - - + +

    Retrieves pending change requests in configured environments

    GET /api/admin/projects/:projectId/change-requests/pending

    This endpoint will retrieve the pending change requests in the configured environments for the project, for the current user performing the request.

    Request

    Path Parameters

    • projectId string required
    Responses

    changeRequestsSchema

    Schema
    • Array [
    • oneOf
    • id number required

      This change requests's ID.

    • title string

      A title describing the change request's content.

    • environment string required

      The environment in which the changes should be applied.

    • minApprovals number required

      The minimum number of approvals required before this change request can be applied.

    • project string required

      The project this change request belongs to.

    • features object[]required

      The list of features and their changes that relate to this change request.

    • Array [
    • name string required

      The name of the feature

    • conflict string

      A string describing the conflicts related to this change. Only present if there are any concflicts on the feature level.

    • changes object[]required

      List of changes inside change request. This list may be empty when listing all change requests for a project.

    • Array [
    • id number required

      The ID of this change.

    • action string required

      The kind of action that the change contains information about.

    • conflict string

      A description of the conflict caused by this change. Only present if there are any conflicts.

    • payload objectrequired

      The data required to perform this action.

      oneOf
    • string
    • createdBy object

      The user who created this change.

    • username string nullable

      The user's username.

    • imageUrl uri nullable

      The URL where the user's image can be found.

    • createdAt date-time

      When this change was suggested

    • ]
    • defaultChange object

      A description of a default change that will be applied with the change request to prevent invalid states.

      Default changes are changes that are applied in addition to explicit user-specified changes when a change request is applied. Any default changes are applied in the background and are not a real part of the change request.

    • action string required

      The kind of action this is.

    • payload object required

      The necessary data to perform this change.

    • ]
    • segments object[]required

      The list of segments and their changes that relate to this change request.

    • Array [
    • id number required

      The ID of this change.

    • action string required

      The kind of action that the change contains information about.

    • conflict string

      A description of the conflict caused by this change. Only present if there are any conflicts.

    • payload objectrequired

      The data required to perform this action.

      oneOf
    • string
    • createdBy object

      The user who created this change.

    • username string nullable

      The user's username.

    • imageUrl uri nullable

      The URL where the user's image can be found.

    • createdAt date-time

      When this change was suggested

    • name string required

      The current name of the segment

    • ]
    • approvals object[]

      A list of approvals that this change request has received.

    • Array [
    • createdBy objectrequired

      Information about the user who gave this approval.

    • id number

      The ID of the user who gave this approval.

    • username string

      The approving user's username.

    • imageUrl uri

      The URL where the user's image can be found.

    • createdAt date-time required

      When the approval was given.

    • ]
    • rejections object[]

      A list of rejections that this change request has received.

    • Array [
    • createdBy objectrequired

      Information about the user who gave this approval.

    • id number

      The ID of the user who gave this approval.

    • username string

      The approving user's username.

    • imageUrl uri

      The URL where the user's image can be found.

    • createdAt date-time required

      When the approval was given.

    • ]
    • comments object[]

      All comments that have been made on this change request.

    • Array [
    • id number

      The comment's ID. Unique per change request.

    • text string required

      The content of the comment.

    • createdBy objectrequired

      Information about the user who posted the comment

    • username string nullable

      The user's username.

    • imageUrl uri nullable

      The URL where the user's image can be found.

    • createdAt date-time required

      When the comment was made.

    • ]
    • createdBy objectrequired

      The user who created this change request.

    • username string nullable
    • imageUrl uri nullable

      The URL of the user's profile image.

    • createdAt date-time required

      When this change request was created.

    • state string required

      Possible values: [Draft, In review, Approved, Applied, Cancelled, Rejected]

      The current state of the change request.

    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-permissions.html b/reference/api/unleash/get-permissions.html index 7f50f3fec4..bbf9f19420 100644 --- a/reference/api/unleash/get-permissions.html +++ b/reference/api/unleash/get-permissions.html @@ -20,15 +20,15 @@ - - + +

    Gets available permissions

    GET /api/admin/permissions

    Returns a list of available permissions

    Request

    Responses

    adminPermissionsSchema

    Schema
    • permissions objectrequired

      Returns permissions available at all three levels (root|project|environment)

    • root object[]

      Permissions available at the root level, i.e. not connected to any specific project or environment

    • Array [
    • id integer required

      The identifier for this permission

    • name string required

      The name of this permission

    • displayName string required

      The name to display in listings of permissions

    • type string required

      What level this permission applies to. Either root, project or the name of the environment it applies to

    • environment string

      Which environment this permission applies to

    • ]
    • project object[]required

      Permissions available at the project level

    • Array [
    • id integer required

      The identifier for this permission

    • name string required

      The name of this permission

    • displayName string required

      The name to display in listings of permissions

    • type string required

      What level this permission applies to. Either root, project or the name of the environment it applies to

    • environment string

      Which environment this permission applies to

    • ]
    • environments object[]required

      A list of environments with available permissions per environment

    • Array [
    • name string required

      The name of the environment

    • permissions object[]required

      Permissions available for this environment

    • Array [
    • id integer required

      The identifier for this permission

    • name string required

      The name of this permission

    • displayName string required

      The name to display in listings of permissions

    • type string required

      What level this permission applies to. Either root, project or the name of the environment it applies to

    • environment string

      Which environment this permission applies to

    • ]
    • ]
    • version integer required

      Possible values: >= 1, [1, 2]

      The api version of this response. A natural increasing number. Only increases if format changes

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-playground.html b/reference/api/unleash/get-playground.html index 0cb57cd9c8..1ef5a41b4a 100644 --- a/reference/api/unleash/get-playground.html +++ b/reference/api/unleash/get-playground.html @@ -20,8 +20,8 @@ - - + + @@ -40,7 +40,7 @@ variant. If a feature is disabled or doesn't have any variants, you would get the disabled variant. Otherwise, you'll get one of thefeature's defined variants.

  • name string required

    The variant's name. If there is no variant or if the toggle is disabled, this will be disabled

  • enabled boolean required

    Whether the variant is enabled or not. If the feature is disabled or if it doesn't have variants, this property will be false

  • payload object

    An optional payload attached to the variant.

  • type string required

    The format of the payload.

  • value string required

    The payload value stringified.

  • variants object[]required

    The feature variants.

  • Array [
  • name string required

    The variants name. Is unique for this feature toggle

  • weight number required

    Possible values: <= 1000

    The weight is the likelihood of any one user getting this variant. It is a number between 0 and 1000. See the section on variant weights for more information

  • weightType string

    Possible values: [variable, fix]

    Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000

  • stickiness string

    Stickiness is how Unleash guarantees that the same user gets the same variant every time

  • payload object

    Extra data configured for this variant

  • type string required

    Possible values: [json, csv, string, number]

    The type of the value. Commonly used types are string, number, json and csv.

  • value string required

    The actual value of payload

  • overrides object[]

    Overrides assigning specific variants to specific users. The weighting system automatically assigns users to specific groups for you, but any overrides in this list will take precedence.

  • Array [
  • contextName string required

    The name of the context field used to determine overrides

  • values string[] required

    Which values that should be overriden

  • ]
  • ]
  • ]
  • Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-profile.html b/reference/api/unleash/get-profile.html index 914b6d77e8..e04fae6c47 100644 --- a/reference/api/unleash/get-profile.html +++ b/reference/api/unleash/get-profile.html @@ -20,15 +20,15 @@ - - + +

    Get your own user profile

    GET /api/admin/user/profile

    Detailed information about the current user root role and project membership

    Request

    Responses

    profileSchema

    Schema
    • rootRole objectrequired

      A role holds permissions to allow Unleash to decide what actions a role holder is allowed to perform

    • id integer required

      The role id

    • type string required

      A role can either be a global root role (applies to all projects) or a project role

    • name string required

      The name of the role

    • description string

      A more detailed description of the role and what use it's intended for

    • projects string[] required

      Which projects this user is a member of

    • features object[]required

      Deprecated, always returns empty array

    • Array [
    • name string required

      Unique feature name

    • type string

      Type of the toggle e.g. experiment, kill-switch, release, operational, permission

    • description string nullable

      Detailed description of the feature

    • archived boolean

      true if the feature is archived

    • project string

      Name of the project the feature belongs to

    • enabled boolean

      true if the feature is enabled, otherwise false.

    • stale boolean

      true if the feature is stale based on the age and feature type, otherwise false.

    • favorite boolean

      true if the feature was favorited, otherwise false.

    • impressionData boolean

      true if the impression data collection is enabled for the feature, otherwise false.

    • createdAt date-time nullable

      The date the feature was created

    • archivedAt date-time nullable

      The date the feature was archived

    • lastSeenAt date-time nullable deprecated

      The date when metrics where last collected for the feature. This field is deprecated, use the one in featureEnvironmentSchema

    • environments object[]

      The list of environments where the feature can be used

    • Array [
    • name string required

      The name of the environment

    • featureName string

      The name of the feature

    • environment string

      The name of the environment

    • type string

      The type of the environment

    • enabled boolean required

      true if the feature is enabled for the environment, otherwise false.

    • sortOrder number

      The sort order of the feature environment in the feature environments list

    • variantCount number

      The number of defined variants

    • strategies object[]

      A list of activation strategies for the feature environment

    • Array [
    • id string

      A uuid for the feature strategy

    • name string required

      The name or type of strategy

    • title string nullable

      A descriptive title for the strategy

    • disabled boolean nullable

      A toggle to disable the strategy. defaults to false. Disabled strategies are not evaluated or returned to the SDKs

    • featureName string

      The name or feature the strategy is attached to

    • sortOrder number

      The order of the strategy in the list

    • segments number[]

      A list of segment ids attached to the strategy

    • constraints object[]

      A list of the constraints attached to the strategy. See https://docs.getunleash.io/reference/strategy-constraints

    • Array [
    • contextName string required

      The name of the context field that this constraint should apply to.

    • operator string required

      Possible values: [NOT_IN, IN, STR_ENDS_WITH, STR_STARTS_WITH, STR_CONTAINS, NUM_EQ, NUM_GT, NUM_GTE, NUM_LT, NUM_LTE, DATE_AFTER, DATE_BEFORE, SEMVER_EQ, SEMVER_GT, SEMVER_LT]

      The operator to use when evaluating this constraint. For more information about the various operators, refer to the strategy constraint operator documentation.

    • caseInsensitive boolean

      Default value: false

      Whether the operator should be case sensitive or not. Defaults to false (being case sensitive).

    • inverted boolean

      Default value: false

      Whether the result should be negated or not. If true, will turn a true result into a false result and vice versa.

    • values string[]

      The context values that should be used for constraint evaluation. Use this property instead of value for properties that accept multiple values.

    • value string

      The context value that should be used for constraint evaluation. Use this property instead of values for properties that only accept single values.

    • ]
    • variants object[]

      Strategy level variants

    • Array [
    • name string required

      The variant name. Must be unique for this feature toggle

    • weight integer required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is an integer between 0 and 1000. See the section on variant weights for more information

    • weightType string required

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000. Refer to the variant weight documentation.

    • stickiness string required

      The stickiness to use for distribution of this variant. Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • ]
    • parameters object

      A list of parameters for a strategy

    • property name* string
    • ]
    • variants object[]

      A list of variants for the feature environment

    • Array [
    • name string required

      The variants name. Is unique for this feature toggle

    • weight number required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is a number between 0 and 1000. See the section on variant weights for more information

    • weightType string

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000

    • stickiness string

      Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • overrides object[]

      Overrides assigning specific variants to specific users. The weighting system automatically assigns users to specific groups for you, but any overrides in this list will take precedence.

    • Array [
    • contextName string required

      The name of the context field used to determine overrides

    • values string[] required

      Which values that should be overriden

    • ]
    • ]
    • lastSeenAt date-time nullable

      The date when metrics where last collected for the feature environment

    • hasStrategies boolean

      Whether the feature has any strategies defined.

    • hasEnabledStrategies boolean

      Whether the feature has any enabled strategies defined.

    • ]
    • variants object[]deprecated

      The list of feature variants

    • Array [
    • name string required

      The variants name. Is unique for this feature toggle

    • weight number required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is a number between 0 and 1000. See the section on variant weights for more information

    • weightType string

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000

    • stickiness string

      Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • overrides object[]

      Overrides assigning specific variants to specific users. The weighting system automatically assigns users to specific groups for you, but any overrides in this list will take precedence.

    • Array [
    • contextName string required

      The name of the context field used to determine overrides

    • values string[] required

      Which values that should be overriden

    • ]
    • ]
    • strategies object[] deprecated

      This is a legacy field that will be deprecated

    • tags object[]nullable

      The list of feature tags

    • Array [
    • value string required

      Possible values: >= 2 characters and <= 50 characters

      The value of the tag

    • type string required

      Possible values: >= 2 characters and <= 50 characters

      Default value: simple

      The type of the tag

    • ]
    • children string[]

      The list of child feature names. This is an experimental field and may change.

    • dependencies object[]

      The list of parent dependencies. This is an experimental field and may change.

    • Array [
    • feature string required

      The name of the parent feature

    • enabled boolean

      Whether the parent feature is enabled or not

    • variants string[]

      The list of variants the parent feature should resolve to. Only valid when feature is enabled.

    • ]
    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-project-access.html b/reference/api/unleash/get-project-access.html index 4d18ebe76c..5ba3f51a88 100644 --- a/reference/api/unleash/get-project-access.html +++ b/reference/api/unleash/get-project-access.html @@ -20,15 +20,15 @@ - - + +

    Get users and groups in project

    GET /api/admin/projects/:projectId/access

    Get all groups, users and their roles, and available roles for the given project.

    Request

    Path Parameters

    • projectId string required
    Responses

    projectAccessSchema

    Schema
    • groups object[]required

      A list of groups that have access to this project

    • Array [
    • name string

      The name of the group

    • id integer required

      The group's ID in the Unleash system

    • addedAt date-time

      When this group was added to the project

    • roleId integer

      The ID of the role this group has in the given project

    • roles integer[]

      A list of roles this user has in the given project

    • description string nullable

      A custom description of the group

    • mappingsSSO string[]

      A list of SSO groups that should map to this Unleash group

    • rootRole number nullable

      A role id that is used as the root role for all users in this group. This can be either the id of the Viewer, Editor or Admin role.

    • createdBy string nullable

      A user who created this group

    • createdAt date-time nullable

      When was this group created

    • users object[]

      A list of users belonging to this group

    • Array [
    • joinedAt date-time

      The date when the user joined the group

    • createdBy string nullable

      The username of the user who added this user to this group

    • user objectrequired

      An Unleash user

    • id integer required

      The user id

    • isAPI boolean deprecated

      (Deprecated): Used internally to know which operations the user should be allowed to perform

    • name string nullable

      Name of the user

    • email string

      Email of the user

    • username string nullable

      A unique username for the user

    • imageUrl string

      URL used for the userprofile image

    • inviteLink string

      If the user is actively inviting other users, this is the link that can be shared with other users

    • loginAttempts integer

      How many unsuccessful attempts at logging in has the user made

    • emailSent boolean

      Is the welcome email sent to the user or not

    • rootRole integer

      Which root role this user is assigned

    • seenAt date-time nullable

      The last time this user logged in

    • createdAt date-time

      The user was created at this time

    • accountType string

      A user is either an actual User or a Service Account

    • permissions string[]

      Deprecated

    • ]
    • ]
    • users object[]required

      A list of users and their roles within this project

    • Array [
    • isAPI boolean deprecated

      Whether this user is authenticated through Unleash tokens or logged in with a session

    • name string

      The name of the user

    • email string nullable

      The user's email address

    • id integer required

      The user's ID in the Unleash system

    • imageUrl uri nullable

      A URL pointing to the user's image.

    • addedAt date-time

      When this user was added to the project

    • roleId integer

      The ID of the role this user has in the given project

    • roles integer[]

      A list of roles this user has in the given project

    • ]
    • roles object[]required

      A list of roles that are available within this project.

    • Array [
    • id integer required

      The role id

    • type string required

      A role can either be a global root role (applies to all projects) or a project role

    • name string required

      The name of the role

    • description string

      A more detailed description of the role and what use it's intended for

    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-project-api-tokens.html b/reference/api/unleash/get-project-api-tokens.html index 26c42d5f16..9afa3a142b 100644 --- a/reference/api/unleash/get-project-api-tokens.html +++ b/reference/api/unleash/get-project-api-tokens.html @@ -20,15 +20,15 @@ - - + +

    Get api tokens for project.

    GET /api/admin/projects/:projectId/api-tokens

    Returns the project API tokens that have been created for this project.

    Request

    Path Parameters

    • projectId string required
    Responses

    apiTokensSchema

    Schema
    • tokens object[]required

      A list of Unleash API tokens.

    • Array [
    • secret string required

      The token used for authentication.

    • username string deprecated

      This property was deprecated in Unleash v5. Prefer the tokenName property instead.

    • tokenName string required

      A unique name for this particular token

    • type string required

      Possible values: [client, admin, frontend]

      The type of API token

    • environment string

      The environment the token has access to. * if it has access to all environments.

    • project string required

      The project this token belongs to.

    • projects string[] required

      The list of projects this token has access to. If the token has access to specific projects they will be listed here. If the token has access to all projects it will be represented as [*]

    • expiresAt date-time nullable

      The token's expiration date. NULL if the token doesn't have an expiration set.

    • createdAt date-time required

      When the token was created.

    • seenAt date-time nullable

      When the token was last seen/used to authenticate with. NULL if the token has not yet been used for authentication.

    • alias string nullable

      Alias is no longer in active use and will often be NULL. It's kept around as a way of allowing old proxy tokens created with the old metadata format to keep working.

    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-project-change-request-config.html b/reference/api/unleash/get-project-change-request-config.html index 38d302d514..762910b045 100644 --- a/reference/api/unleash/get-project-change-request-config.html +++ b/reference/api/unleash/get-project-change-request-config.html @@ -20,15 +20,15 @@ - - + +

    Retrieves change request configuration for a project

    GET /api/admin/projects/:projectId/change-requests/config

    Given a projectId, this endpoint will retrieve change request configuration for the project

    Request

    Path Parameters

    • projectId string required
    Responses

    changeRequestConfigSchema

    Schema
    • Array [
    • environment string required

      The environment that this configuration applies to.

    • type string required

      The type of the environment listed in environment.

    • changeRequestEnabled boolean required

      true if this environment has change requests enabled, otherwise false.

    • requiredApprovals number nullable required

      The number of approvals that are required for a change request to be fully approved and ready to be applied in this environment.

    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-project-dora.html b/reference/api/unleash/get-project-dora.html index 2f55d8a6d0..40ba458166 100644 --- a/reference/api/unleash/get-project-dora.html +++ b/reference/api/unleash/get-project-dora.html @@ -20,15 +20,15 @@ - - + +

    Get an overview project dora metrics.

    GET /api/admin/projects/:projectId/dora

    This endpoint returns an overview of the specified dora metrics

    Request

    Path Parameters

    • projectId string required
    Responses

    projectDoraMetricsSchema

    Schema
    • projectAverage number

      The average time it takes a feature toggle to be enabled in production. The measurement unit is days.

    • features object[]required

      An array of objects containing feature toggle name and timeToProduction values. The measurement unit of timeToProduction is days.

    • Array [
    • name string required

      The name of a feature toggle

    • timeToProduction number required

      The average number of days it takes a feature toggle to get into production

    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-project-environments.html b/reference/api/unleash/get-project-environments.html index bfa328fed4..e1aaaae9ab 100644 --- a/reference/api/unleash/get-project-environments.html +++ b/reference/api/unleash/get-project-environments.html @@ -20,15 +20,15 @@ - - + +

    Get the environments available to a project

    GET /api/admin/environments/project/:projectId

    Gets the environments that are available for this project. An environment is available for a project if enabled in the project configuration

    Request

    Path Parameters

    • projectId string required
    Responses

    environmentsProjectSchema

    Schema
    • version integer required

      Version of the environments schema

    • environments object[]required

      List of environments

    • Array [
    • name string required

      The name of the environment

    • type string required
    • enabled boolean required

      true if the environment is enabled for the project, otherwise false

    • protected boolean required

      true if the environment is protected, otherwise false. A protected environment can not be deleted.

    • sortOrder integer required

      Priority of the environment in a list of environments, the lower the value, the higher up in the list the environment will appear

    • projectApiTokenCount integer

      The number of client and front-end API tokens that have access to this project

    • projectEnabledToggleCount integer

      The number of features enabled in this environment for this project

    • defaultStrategy object

      The strategy configuration to add when enabling a feature environment by default

    • name string required

      The name of the strategy type

    • title string nullable

      A descriptive title for the strategy

    • disabled boolean nullable

      A toggle to disable the strategy. defaults to false. Disabled strategies are not evaluated or returned to the SDKs

    • sortOrder number

      The order of the strategy in the list

    • constraints object[]

      A list of the constraints attached to the strategy. See https://docs.getunleash.io/reference/strategy-constraints

    • Array [
    • contextName string required

      The name of the context field that this constraint should apply to.

    • operator string required

      Possible values: [NOT_IN, IN, STR_ENDS_WITH, STR_STARTS_WITH, STR_CONTAINS, NUM_EQ, NUM_GT, NUM_GTE, NUM_LT, NUM_LTE, DATE_AFTER, DATE_BEFORE, SEMVER_EQ, SEMVER_GT, SEMVER_LT]

      The operator to use when evaluating this constraint. For more information about the various operators, refer to the strategy constraint operator documentation.

    • caseInsensitive boolean

      Default value: false

      Whether the operator should be case sensitive or not. Defaults to false (being case sensitive).

    • inverted boolean

      Default value: false

      Whether the result should be negated or not. If true, will turn a true result into a false result and vice versa.

    • values string[]

      The context values that should be used for constraint evaluation. Use this property instead of value for properties that accept multiple values.

    • value string

      The context value that should be used for constraint evaluation. Use this property instead of values for properties that only accept single values.

    • ]
    • variants object[]

      Strategy level variants

    • Array [
    • name string required

      The variant name. Must be unique for this feature toggle

    • weight integer required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is an integer between 0 and 1000. See the section on variant weights for more information

    • weightType string required

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000. Refer to the variant weight documentation.

    • stickiness string required

      The stickiness to use for distribution of this variant. Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • ]
    • parameters object

      A list of parameters for a strategy

    • property name* string
    • segments number[]

      Ids of segments to use for this strategy

    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-project-health-report.html b/reference/api/unleash/get-project-health-report.html index 7a4944d8d8..a485591e1f 100644 --- a/reference/api/unleash/get-project-health-report.html +++ b/reference/api/unleash/get-project-health-report.html @@ -20,15 +20,15 @@ - - + +

    Get a health report for a project.

    GET /api/admin/projects/:projectId/health-report

    This endpoint returns a health report for the specified project. This data is used for the technical debt dashboard

    Request

    Path Parameters

    • projectId string required
    Responses

    healthReportSchema

    Schema
    • version integer required

      The project overview version.

    • name string required

      The project's name

    • description string nullable

      The project's description

    • defaultStickiness string required

      A default stickiness for the project affecting the default stickiness value for variants and Gradual Rollout strategy

    • mode string required

      Possible values: [open, protected, private]

      The project's collaboration mode. Determines whether non-project members can submit change requests or not.

    • featureLimit number nullable

      A limit on the number of features allowed in the project. Null if no limit.

    • members integer required

      The number of users/members in the project.

    • health integer required

      The overall health rating of the project.

    • environments object[]required

      An array containing the names of all the environments configured for the project.

    • Array [
    • environment string required

      The environment to add to the project

    • changeRequestsEnabled boolean

      Whether change requests should be enabled or for this environment on the project or not

    • defaultStrategy object

      A default strategy to create for this environment on the project.

    • name string required

      The name of the strategy type

    • title string nullable

      A descriptive title for the strategy

    • disabled boolean nullable

      A toggle to disable the strategy. defaults to false. Disabled strategies are not evaluated or returned to the SDKs

    • sortOrder number

      The order of the strategy in the list

    • constraints object[]

      A list of the constraints attached to the strategy. See https://docs.getunleash.io/reference/strategy-constraints

    • Array [
    • contextName string required

      The name of the context field that this constraint should apply to.

    • operator string required

      Possible values: [NOT_IN, IN, STR_ENDS_WITH, STR_STARTS_WITH, STR_CONTAINS, NUM_EQ, NUM_GT, NUM_GTE, NUM_LT, NUM_LTE, DATE_AFTER, DATE_BEFORE, SEMVER_EQ, SEMVER_GT, SEMVER_LT]

      The operator to use when evaluating this constraint. For more information about the various operators, refer to the strategy constraint operator documentation.

    • caseInsensitive boolean

      Default value: false

      Whether the operator should be case sensitive or not. Defaults to false (being case sensitive).

    • inverted boolean

      Default value: false

      Whether the result should be negated or not. If true, will turn a true result into a false result and vice versa.

    • values string[]

      The context values that should be used for constraint evaluation. Use this property instead of value for properties that accept multiple values.

    • value string

      The context value that should be used for constraint evaluation. Use this property instead of values for properties that only accept single values.

    • ]
    • variants object[]

      Strategy level variants

    • Array [
    • name string required

      The variant name. Must be unique for this feature toggle

    • weight integer required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is an integer between 0 and 1000. See the section on variant weights for more information

    • weightType string required

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000. Refer to the variant weight documentation.

    • stickiness string required

      The stickiness to use for distribution of this variant. Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • ]
    • parameters object

      A list of parameters for a strategy

    • property name* string
    • segments number[]

      Ids of segments to use for this strategy

    • ]
    • features object[]required

      An array containing an overview of all the features of the project and their individual status

    • Array [
    • name string required

      Unique feature name

    • type string

      Type of the toggle e.g. experiment, kill-switch, release, operational, permission

    • description string nullable

      Detailed description of the feature

    • archived boolean

      true if the feature is archived

    • project string

      Name of the project the feature belongs to

    • enabled boolean

      true if the feature is enabled, otherwise false.

    • stale boolean

      true if the feature is stale based on the age and feature type, otherwise false.

    • favorite boolean

      true if the feature was favorited, otherwise false.

    • impressionData boolean

      true if the impression data collection is enabled for the feature, otherwise false.

    • createdAt date-time nullable

      The date the feature was created

    • archivedAt date-time nullable

      The date the feature was archived

    • lastSeenAt date-time nullable deprecated

      The date when metrics where last collected for the feature. This field is deprecated, use the one in featureEnvironmentSchema

    • environments object[]

      The list of environments where the feature can be used

    • Array [
    • name string required

      The name of the environment

    • featureName string

      The name of the feature

    • environment string

      The name of the environment

    • type string

      The type of the environment

    • enabled boolean required

      true if the feature is enabled for the environment, otherwise false.

    • sortOrder number

      The sort order of the feature environment in the feature environments list

    • variantCount number

      The number of defined variants

    • strategies object[]

      A list of activation strategies for the feature environment

    • Array [
    • id string

      A uuid for the feature strategy

    • name string required

      The name or type of strategy

    • title string nullable

      A descriptive title for the strategy

    • disabled boolean nullable

      A toggle to disable the strategy. defaults to false. Disabled strategies are not evaluated or returned to the SDKs

    • featureName string

      The name or feature the strategy is attached to

    • sortOrder number

      The order of the strategy in the list

    • segments number[]

      A list of segment ids attached to the strategy

    • constraints object[]

      A list of the constraints attached to the strategy. See https://docs.getunleash.io/reference/strategy-constraints

    • Array [
    • contextName string required

      The name of the context field that this constraint should apply to.

    • operator string required

      Possible values: [NOT_IN, IN, STR_ENDS_WITH, STR_STARTS_WITH, STR_CONTAINS, NUM_EQ, NUM_GT, NUM_GTE, NUM_LT, NUM_LTE, DATE_AFTER, DATE_BEFORE, SEMVER_EQ, SEMVER_GT, SEMVER_LT]

      The operator to use when evaluating this constraint. For more information about the various operators, refer to the strategy constraint operator documentation.

    • caseInsensitive boolean

      Default value: false

      Whether the operator should be case sensitive or not. Defaults to false (being case sensitive).

    • inverted boolean

      Default value: false

      Whether the result should be negated or not. If true, will turn a true result into a false result and vice versa.

    • values string[]

      The context values that should be used for constraint evaluation. Use this property instead of value for properties that accept multiple values.

    • value string

      The context value that should be used for constraint evaluation. Use this property instead of values for properties that only accept single values.

    • ]
    • variants object[]

      Strategy level variants

    • Array [
    • name string required

      The variant name. Must be unique for this feature toggle

    • weight integer required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is an integer between 0 and 1000. See the section on variant weights for more information

    • weightType string required

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000. Refer to the variant weight documentation.

    • stickiness string required

      The stickiness to use for distribution of this variant. Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • ]
    • parameters object

      A list of parameters for a strategy

    • property name* string
    • ]
    • variants object[]

      A list of variants for the feature environment

    • Array [
    • name string required

      The variants name. Is unique for this feature toggle

    • weight number required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is a number between 0 and 1000. See the section on variant weights for more information

    • weightType string

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000

    • stickiness string

      Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • overrides object[]

      Overrides assigning specific variants to specific users. The weighting system automatically assigns users to specific groups for you, but any overrides in this list will take precedence.

    • Array [
    • contextName string required

      The name of the context field used to determine overrides

    • values string[] required

      Which values that should be overriden

    • ]
    • ]
    • lastSeenAt date-time nullable

      The date when metrics where last collected for the feature environment

    • hasStrategies boolean

      Whether the feature has any strategies defined.

    • hasEnabledStrategies boolean

      Whether the feature has any enabled strategies defined.

    • ]
    • variants object[]deprecated

      The list of feature variants

    • Array [
    • name string required

      The variants name. Is unique for this feature toggle

    • weight number required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is a number between 0 and 1000. See the section on variant weights for more information

    • weightType string

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000

    • stickiness string

      Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • overrides object[]

      Overrides assigning specific variants to specific users. The weighting system automatically assigns users to specific groups for you, but any overrides in this list will take precedence.

    • Array [
    • contextName string required

      The name of the context field used to determine overrides

    • values string[] required

      Which values that should be overriden

    • ]
    • ]
    • strategies object[] deprecated

      This is a legacy field that will be deprecated

    • tags object[]nullable

      The list of feature tags

    • Array [
    • value string required

      Possible values: >= 2 characters and <= 50 characters

      The value of the tag

    • type string required

      Possible values: >= 2 characters and <= 50 characters

      Default value: simple

      The type of the tag

    • ]
    • children string[]

      The list of child feature names. This is an experimental field and may change.

    • dependencies object[]

      The list of parent dependencies. This is an experimental field and may change.

    • Array [
    • feature string required

      The name of the parent feature

    • enabled boolean

      Whether the parent feature is enabled or not

    • variants string[]

      The list of variants the parent feature should resolve to. Only valid when feature is enabled.

    • ]
    • ]
    • updatedAt date-time nullable

      When the project was last updated.

    • createdAt date-time nullable

      When the project was last updated.

    • favorite boolean

      Indicates if the project has been marked as a favorite by the current user requesting the project health overview.

    • stats object

      Project statistics

    • avgTimeToProdCurrentWindow number required

      The average time from when a feature was created to when it was enabled in the "production" environment during the current window

    • createdCurrentWindow number required

      The number of feature toggles created during the current window

    • createdPastWindow number required

      The number of feature toggles created during the previous window

    • archivedCurrentWindow number required

      The number of feature toggles that were archived during the current window

    • archivedPastWindow number required

      The number of feature toggles that were archived during the previous window

    • projectActivityCurrentWindow number required

      The number of project events that occurred during the current window

    • projectActivityPastWindow number required

      The number of project events that occurred during the previous window

    • projectMembersAddedCurrentWindow number required

      The number of members that were added to the project during the current window

    • featureNaming object

      Create a feature naming pattern

    • pattern string nullable required

      A JavaScript regular expression pattern, without the start and end delimiters. Optional flags are not allowed.

    • example string nullable

      An example of a feature name that matches the pattern. Must itself match the pattern supplied.

    • description string nullable

      A description of the pattern in a human-readable format. Will be shown to users when they create a new feature flag.

    • potentiallyStaleCount number required

      The number of potentially stale feature toggles.

    • activeCount number required

      The number of active feature toggles.

    • staleCount number required

      The number of stale feature toggles.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-project-overview.html b/reference/api/unleash/get-project-overview.html index 8a7cc09a9c..6e79938e81 100644 --- a/reference/api/unleash/get-project-overview.html +++ b/reference/api/unleash/get-project-overview.html @@ -20,15 +20,15 @@ - - + +

    Get an overview of a project.

    GET /api/admin/projects/:projectId/overview

    This endpoint returns an overview of the specified projects stats, project health, number of members, which environments are configured, and the features types in the project.

    Request

    Path Parameters

    • projectId string required
    Responses

    projectOverviewSchema

    Schema
    • stats object

      Project statistics

    • avgTimeToProdCurrentWindow number required

      The average time from when a feature was created to when it was enabled in the "production" environment during the current window

    • createdCurrentWindow number required

      The number of feature toggles created during the current window

    • createdPastWindow number required

      The number of feature toggles created during the previous window

    • archivedCurrentWindow number required

      The number of feature toggles that were archived during the current window

    • archivedPastWindow number required

      The number of feature toggles that were archived during the previous window

    • projectActivityCurrentWindow number required

      The number of project events that occurred during the current window

    • projectActivityPastWindow number required

      The number of project events that occurred during the previous window

    • projectMembersAddedCurrentWindow number required

      The number of members that were added to the project during the current window

    • version integer required

      The schema version used to describe the project overview

    • name string required

      The name of this project

    • description string nullable

      Additional information about the project

    • defaultStickiness string

      A default stickiness for the project affecting the default stickiness value for variants and Gradual Rollout strategy

    • mode string

      Possible values: [open, protected, private]

      The project's collaboration mode. Determines whether non-project members can submit change requests or not.

    • featureLimit number nullable

      A limit on the number of features allowed in the project. Null if no limit.

    • featureNaming object

      Create a feature naming pattern

    • pattern string nullable required

      A JavaScript regular expression pattern, without the start and end delimiters. Optional flags are not allowed.

    • example string nullable

      An example of a feature name that matches the pattern. Must itself match the pattern supplied.

    • description string nullable

      A description of the pattern in a human-readable format. Will be shown to users when they create a new feature flag.

    • members number

      The number of members this project has

    • health number

      An indicator of the project's health on a scale from 0 to 100

    • environments object[]

      The environments that are enabled for this project

    • Array [
    • environment string required

      The environment to add to the project

    • changeRequestsEnabled boolean

      Whether change requests should be enabled or for this environment on the project or not

    • defaultStrategy object

      A default strategy to create for this environment on the project.

    • name string required

      The name of the strategy type

    • title string nullable

      A descriptive title for the strategy

    • disabled boolean nullable

      A toggle to disable the strategy. defaults to false. Disabled strategies are not evaluated or returned to the SDKs

    • sortOrder number

      The order of the strategy in the list

    • constraints object[]

      A list of the constraints attached to the strategy. See https://docs.getunleash.io/reference/strategy-constraints

    • Array [
    • contextName string required

      The name of the context field that this constraint should apply to.

    • operator string required

      Possible values: [NOT_IN, IN, STR_ENDS_WITH, STR_STARTS_WITH, STR_CONTAINS, NUM_EQ, NUM_GT, NUM_GTE, NUM_LT, NUM_LTE, DATE_AFTER, DATE_BEFORE, SEMVER_EQ, SEMVER_GT, SEMVER_LT]

      The operator to use when evaluating this constraint. For more information about the various operators, refer to the strategy constraint operator documentation.

    • caseInsensitive boolean

      Default value: false

      Whether the operator should be case sensitive or not. Defaults to false (being case sensitive).

    • inverted boolean

      Default value: false

      Whether the result should be negated or not. If true, will turn a true result into a false result and vice versa.

    • values string[]

      The context values that should be used for constraint evaluation. Use this property instead of value for properties that accept multiple values.

    • value string

      The context value that should be used for constraint evaluation. Use this property instead of values for properties that only accept single values.

    • ]
    • variants object[]

      Strategy level variants

    • Array [
    • name string required

      The variant name. Must be unique for this feature toggle

    • weight integer required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is an integer between 0 and 1000. See the section on variant weights for more information

    • weightType string required

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000. Refer to the variant weight documentation.

    • stickiness string required

      The stickiness to use for distribution of this variant. Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • ]
    • parameters object

      A list of parameters for a strategy

    • property name* string
    • segments number[]

      Ids of segments to use for this strategy

    • ]
    • featureTypeCounts object[]

      The number of features of each type that are in this project

    • Array [
    • type string required

      Type of the flag e.g. experiment, kill-switch, release, operational, permission

    • count number required

      Number of feature flags of this type

    • ]
    • updatedAt date-time nullable

      When the project was last updated.

    • createdAt date-time nullable

      When the project was created.

    • favorite boolean

      true if the project was favorited, otherwise false.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-project-users.html b/reference/api/unleash/get-project-users.html index 03e1a546ae..31cbfd804e 100644 --- a/reference/api/unleash/get-project-users.html +++ b/reference/api/unleash/get-project-users.html @@ -20,15 +20,15 @@ - - + +

    Get users in project

    GET /api/admin/projects/:projectId/users
    deprecated

    This endpoint has been deprecated and may be removed in future versions of the API.

    Get users belonging to a project together with their roles as well as a list of roles available to the project. This endpoint is deprecated. Use /:projectId/access instead.

    Request

    Path Parameters

    • projectId string required
    Responses

    projectUsersSchema

    Schema
    • users object[]required

      A list of users with access to this project and their role within it.

    • Array [
    • isAPI boolean deprecated

      Whether this user is authenticated through Unleash tokens or logged in with a session

    • name string

      The name of the user

    • email string nullable

      The user's email address

    • id integer required

      The user's ID in the Unleash system

    • imageUrl uri nullable

      A URL pointing to the user's image.

    • addedAt date-time

      When this user was added to the project

    • roleId integer

      The ID of the role this user has in the given project

    • roles integer[]

      A list of roles this user has in the given project

    • ]
    • roles object[]required

      A list of roles that are available for this project

    • Array [
    • id integer required

      The role id

    • type string required

      A role can either be a global root role (applies to all projects) or a project role

    • name string required

      The name of the role

    • description string

      A more detailed description of the role and what use it's intended for

    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-projects.html b/reference/api/unleash/get-projects.html index 9c86b4227d..e8f61db6f3 100644 --- a/reference/api/unleash/get-projects.html +++ b/reference/api/unleash/get-projects.html @@ -20,15 +20,15 @@ - - + +

    Get a list of all projects.

    GET /api/admin/projects

    This endpoint returns an list of all the projects in the Unleash instance.

    Request

    Responses

    projectsSchema

    Schema
    • version integer required

      The schema version used to represent the project data.

    • projects object[]required

      A list of projects in the Unleash instance

    • Array [
    • id string required

      The id of this project

    • name string required

      The name of this project

    • description string nullable

      Additional information about the project

    • health number

      An indicator of the project's health on a scale from 0 to 100

    • featureCount number

      The number of features this project has

    • memberCount number

      The number of members this project has

    • createdAt date-time

      When this project was created.

    • updatedAt date-time nullable

      When this project was last updated.

    • favorite boolean

      true if the project was favorited, otherwise false.

    • mode string

      Possible values: [open, protected, private]

      The project's collaboration mode. Determines whether non-project members can submit change requests or not.

    • defaultStickiness string

      A default stickiness for the project affecting the default stickiness value for variants and Gradual Rollout strategy

    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-public-signup-token.html b/reference/api/unleash/get-public-signup-token.html index 409e41cdaa..524f485731 100644 --- a/reference/api/unleash/get-public-signup-token.html +++ b/reference/api/unleash/get-public-signup-token.html @@ -20,15 +20,15 @@ - - + +

    Retrieve a token

    GET /api/admin/invite-link/tokens/:token

    Get information about a specific token. The :token part of the URL should be the token's secret.

    Request

    Path Parameters

    • token string required
    Responses

    publicSignupTokenSchema

    Schema
    • secret string required

      The actual value of the token. This is the part that is used by Unleash to create an invite link

    • url string nullable required

      The public signup link for the token. Users who follow this link will be taken to a signup page where they can create an Unleash user.

    • name string required

      The token's name. Only for displaying in the UI

    • enabled boolean required

      Whether the token is active. This property will always be false for a token that has expired.

    • expiresAt date-time required

      The time when the token will expire.

    • createdAt date-time required

      When the token was created.

    • createdBy string nullable required

      The creator's email or username

    • users object[]nullable

      Array of users that have signed up using the token.

    • Array [
    • id integer required

      The user id

    • isAPI boolean deprecated

      (Deprecated): Used internally to know which operations the user should be allowed to perform

    • name string nullable

      Name of the user

    • email string

      Email of the user

    • username string nullable

      A unique username for the user

    • imageUrl string

      URL used for the userprofile image

    • inviteLink string

      If the user is actively inviting other users, this is the link that can be shared with other users

    • loginAttempts integer

      How many unsuccessful attempts at logging in has the user made

    • emailSent boolean

      Is the welcome email sent to the user or not

    • rootRole integer

      Which root role this user is assigned

    • seenAt date-time nullable

      The last time this user logged in

    • createdAt date-time

      The user was created at this time

    • accountType string

      A user is either an actual User or a Service Account

    • permissions string[]

      Deprecated

    • ]
    • role objectrequired

      A role holds permissions to allow Unleash to decide what actions a role holder is allowed to perform

    • id integer required

      The role id

    • type string required

      A role can either be a global root role (applies to all projects) or a project role

    • name string required

      The name of the role

    • description string

      A more detailed description of the role and what use it's intended for

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-raw-feature-metrics.html b/reference/api/unleash/get-raw-feature-metrics.html index 432408eb17..e854a16fe2 100644 --- a/reference/api/unleash/get-raw-feature-metrics.html +++ b/reference/api/unleash/get-raw-feature-metrics.html @@ -20,15 +20,15 @@ - - + +

    Get feature metrics

    GET /api/admin/client-metrics/features/:name/raw

    Get usage metrics for a specific feature for the last 48 hours, grouped by hour

    Request

    Path Parameters

    • name string required
    Responses

    featureMetricsSchema

    Schema
    • version integer required

      Possible values: >= 1

      The version of this schema

    • maturity string required

      The maturity level of this API (alpha, beta, stable, deprecated)

    • data object[]required

      Metrics gathered per environment

    • Array [
    • featureName string

      The name of the feature

    • appName string

      The name of the application the SDK is being used in

    • environment string required

      Which environment the SDK is being used in

    • timestamp objectrequired

      The start of the time window these metrics are valid for. The window is usually 1 hour wide

      oneOf
    • string date-time

      An RFC-3339-compliant timestamp.

    • yes integer required

      How many times the toggle evaluated to true

    • no integer required

      How many times the toggle evaluated to false

    • variants object

      How many times each variant was returned

    • property name* integer
    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-requests-per-second.html b/reference/api/unleash/get-requests-per-second.html index 549a30db74..84c32b3af7 100644 --- a/reference/api/unleash/get-requests-per-second.html +++ b/reference/api/unleash/get-requests-per-second.html @@ -20,15 +20,15 @@ - - + +

    Gets usage data

    GET /api/admin/metrics/rps

    Gets usage data per app/endpoint from a prometheus compatible metrics endpoint

    Request

    Responses

    requestsPerSecondSegmentedSchema

    Schema
    • clientMetrics object

      Statistics for usage of Unleash, formatted so it can easily be used in a graph

    • status string

      Possible values: [success, failure]

      Whether the query against prometheus succeeded or failed

    • data object

      The query result from prometheus

    • resultType string

      Possible values: [matrix, vector, scalar, string]

      Prometheus compatible result type.

    • result object[]

      An array of values per metric. Each one represents a line in the graph labeled by its metric name

    • Array [
    • metric object

      A key value set representing the metric

    • appName string

      Name of the application this metric relates to

    • endpoint string

      Which endpoint has been accessed

    • values array[]

      An array of arrays. Each element of the array is an array of size 2 consisting of the 2 axis for the graph: in position zero the x axis represented as a number and position one the y axis represented as string

    • ]
    • adminMetrics object

      Statistics for usage of Unleash, formatted so it can easily be used in a graph

    • status string

      Possible values: [success, failure]

      Whether the query against prometheus succeeded or failed

    • data object

      The query result from prometheus

    • resultType string

      Possible values: [matrix, vector, scalar, string]

      Prometheus compatible result type.

    • result object[]

      An array of values per metric. Each one represents a line in the graph labeled by its metric name

    • Array [
    • metric object

      A key value set representing the metric

    • appName string

      Name of the application this metric relates to

    • endpoint string

      Which endpoint has been accessed

    • values array[]

      An array of arrays. Each element of the array is an array of size 2 consisting of the 2 axis for the graph: in position zero the x axis represented as a number and position one the y axis represented as string

    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-role-by-id.html b/reference/api/unleash/get-role-by-id.html index 02356f07a2..02722efc9d 100644 --- a/reference/api/unleash/get-role-by-id.html +++ b/reference/api/unleash/get-role-by-id.html @@ -20,15 +20,15 @@ - - + +

    Get a single role

    GET /api/admin/roles/:roleId

    Get a single role by role id

    Request

    Path Parameters

    • roleId string required
    Responses

    roleWithPermissionsSchema

    Schema
    • id number required

      The role id

    • type string required

      A role can either be a global root role, or a project role or a custom project role or a custom global root-custom role

    • name string required

      The name of the role

    • description string

      A more detailed description of the role and what use it's intended for

    • permissions object[]required

      A list of permissions assigned to this role

    • Array [
    • id integer required

      The identifier for this permission

    • name string required

      The name of this permission

    • displayName string required

      The name to display in listings of permissions

    • type string required

      What level this permission applies to. Either root, project or the name of the environment it applies to

    • environment string

      Which environment this permission applies to

    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-role-project-access.html b/reference/api/unleash/get-role-project-access.html index 1efc725cf6..edb34928da 100644 --- a/reference/api/unleash/get-role-project-access.html +++ b/reference/api/unleash/get-role-project-access.html @@ -20,15 +20,15 @@ - - + +

    Get project-role mappings

    GET /api/admin/projects/roles/:roleId/access

    For the provided role, retrieves a list of projects that use this role. For each project it also contains information about how the role used in that project, such as how many users, groups, or service accounts that use the role.

    Request

    Path Parameters

    • roleId string required
    Responses

    projectRoleUsageSchema

    Schema
    • projects object[]

      A collection of projects with counts of users and groups mapped to them with specified roles.

    • Array [
    • project string required

      The id of the project user and group count are counted for.

    • role integer

      Id of the role the user and group count are counted for.

    • userCount integer

      Number of users mapped to this project.

    • serviceAccountCount integer

      Number of service accounts mapped to this project.

    • groupCount integer

      Number of groups mapped to this project.

    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-roles.html b/reference/api/unleash/get-roles.html index d688cada83..d06b9680e5 100644 --- a/reference/api/unleash/get-roles.html +++ b/reference/api/unleash/get-roles.html @@ -20,15 +20,15 @@ - - + +

    Get a list of roles

    GET /api/admin/roles

    Get a list of project, root and custom roles for Role-Based Access Control

    Request

    Responses

    rolesWithVersionSchema

    Schema
    • version integer required

      Possible values: >= 1

      The version of this schema

    • roles object[]required

      A list of roles

    • Array [
    • id integer required

      The role id

    • type string required

      A role can either be a global root role (applies to all projects) or a project role

    • name string required

      The name of the role

    • description string

      A more detailed description of the role and what use it's intended for

    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-saml-settings.html b/reference/api/unleash/get-saml-settings.html index fe15965e41..a05adf8cad 100644 --- a/reference/api/unleash/get-saml-settings.html +++ b/reference/api/unleash/get-saml-settings.html @@ -20,15 +20,15 @@ - - + +

    Get SAML auth settings

    GET /api/admin/auth/saml/settings

    Returns the current settings for SAML authentication

    Request

    Responses

    samlSettingsSchema

    Schema
    • enabled boolean

      Is SAML authentication enabled

    • entityId string required

      The SAML 2.0 entity ID

    • signOnUrl string required

      Which URL to use for Single Sign On

    • certificate string required

      The X509 certificate used to validate requests

    • signOutUrl string

      Which URL to use for Single Sign Out

    • spCertificate string

      Signing certificate for sign out requests

    • autoCreate boolean

      Should Unleash create users based on the emails coming back in the authentication reply from the SAML server

    • emailDomains string

      A comma separated list of email domains that Unleash will auto create user accounts for.

    • defaultRootRole string

      Possible values: [Viewer, Editor, Admin]

      Assign this root role to auto created users

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-scheduled-change-requests.html b/reference/api/unleash/get-scheduled-change-requests.html index c624f7936e..a6fec4fb9d 100644 --- a/reference/api/unleash/get-scheduled-change-requests.html +++ b/reference/api/unleash/get-scheduled-change-requests.html @@ -20,15 +20,15 @@ - - + +

    Get scheduled change requests matching a query.

    GET /api/admin/projects/:projectId/change-requests/scheduled

    This endpoint retrieves basic information about all scheduled change requests that match the search criteria provided in the query parameters. The endpoint will return any change request that matches one or more of the provided parameters. If you use no query parameters, you'll get an empty list.

    For instance, to find all the scheduled change requests that either touch feature MyFeature or strategy 0D198067-7D55-460C-9EC7-DB86E3AE261A, you would use the query string feature=MyFeature&strategyId=0D198067-7D55-460C-9EC7-DB86E3AE261A.

    Request

    Path Parameters

    • projectId string required
    Responses

    changeRequestScheduledResultSchema

    Schema
    • Array [
    • id number required

      The change request id

    • environment string required

      The environment of the change request

    • title string

      The change request title

    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-segment.html b/reference/api/unleash/get-segment.html index 19771d782f..b2431b4bf0 100644 --- a/reference/api/unleash/get-segment.html +++ b/reference/api/unleash/get-segment.html @@ -20,15 +20,15 @@ - - + +

    Get a segment

    GET /api/admin/segments/:id

    Retrieves a segment based on its ID.

    Request

    Path Parameters

    • id string required
    Responses

    adminSegmentSchema

    Schema
    • id integer required

      The ID of this segment

    • name string required

      The name of this segment

    • description string nullable

      The description for this segment

    • constraints object[]required

      The list of constraints that are used in this segment

    • Array [
    • contextName string required

      The name of the context field that this constraint should apply to.

    • operator string required

      Possible values: [NOT_IN, IN, STR_ENDS_WITH, STR_STARTS_WITH, STR_CONTAINS, NUM_EQ, NUM_GT, NUM_GTE, NUM_LT, NUM_LTE, DATE_AFTER, DATE_BEFORE, SEMVER_EQ, SEMVER_GT, SEMVER_LT]

      The operator to use when evaluating this constraint. For more information about the various operators, refer to the strategy constraint operator documentation.

    • caseInsensitive boolean

      Default value: false

      Whether the operator should be case sensitive or not. Defaults to false (being case sensitive).

    • inverted boolean

      Default value: false

      Whether the result should be negated or not. If true, will turn a true result into a false result and vice versa.

    • values string[]

      The context values that should be used for constraint evaluation. Use this property instead of value for properties that accept multiple values.

    • value string

      The context value that should be used for constraint evaluation. Use this property instead of values for properties that only accept single values.

    • ]
    • usedInFeatures integer nullable

      The number of feature flags that use this segment. The number also includes the any flags with pending change requests that would add this segment.

    • usedInProjects integer nullable

      The number of projects that use this segment. The number includes any projects with pending change requests that would add this segment.

    • project string nullable

      The project the segment belongs to. Only present if the segment is a project-specific segment.

    • createdBy string nullable

      The creator's email or username

    • createdAt date-time required

      When the segment was created

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-segments-by-strategy-id.html b/reference/api/unleash/get-segments-by-strategy-id.html index eaf7ff60ae..c6bf46938f 100644 --- a/reference/api/unleash/get-segments-by-strategy-id.html +++ b/reference/api/unleash/get-segments-by-strategy-id.html @@ -20,15 +20,15 @@ - - + +

    Get strategy segments

    GET /api/admin/segments/strategies/:strategyId

    Retrieve all segments that are referenced by the specified strategy. Returns an empty list of segments if the strategy ID doesn't exist.

    Request

    Path Parameters

    • strategyId string required
    Responses

    segmentsSchema

    Schema
    • segments object[]

      A list of segments

    • Array [
    • id integer required

      The ID of this segment

    • name string required

      The name of this segment

    • description string nullable

      The description for this segment

    • constraints object[]required

      The list of constraints that are used in this segment

    • Array [
    • contextName string required

      The name of the context field that this constraint should apply to.

    • operator string required

      Possible values: [NOT_IN, IN, STR_ENDS_WITH, STR_STARTS_WITH, STR_CONTAINS, NUM_EQ, NUM_GT, NUM_GTE, NUM_LT, NUM_LTE, DATE_AFTER, DATE_BEFORE, SEMVER_EQ, SEMVER_GT, SEMVER_LT]

      The operator to use when evaluating this constraint. For more information about the various operators, refer to the strategy constraint operator documentation.

    • caseInsensitive boolean

      Default value: false

      Whether the operator should be case sensitive or not. Defaults to false (being case sensitive).

    • inverted boolean

      Default value: false

      Whether the result should be negated or not. If true, will turn a true result into a false result and vice versa.

    • values string[]

      The context values that should be used for constraint evaluation. Use this property instead of value for properties that accept multiple values.

    • value string

      The context value that should be used for constraint evaluation. Use this property instead of values for properties that only accept single values.

    • ]
    • usedInFeatures integer nullable

      The number of feature flags that use this segment. The number also includes the any flags with pending change requests that would add this segment.

    • usedInProjects integer nullable

      The number of projects that use this segment. The number includes any projects with pending change requests that would add this segment.

    • project string nullable

      The project the segment belongs to. Only present if the segment is a project-specific segment.

    • createdBy string nullable

      The creator's email or username

    • createdAt date-time required

      When the segment was created

    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-segments.html b/reference/api/unleash/get-segments.html index b0d6838c11..33e5905b76 100644 --- a/reference/api/unleash/get-segments.html +++ b/reference/api/unleash/get-segments.html @@ -20,15 +20,15 @@ - - + +

    Get all segments

    GET /api/admin/segments

    Retrieves all segments that exist in this Unleash instance.

    Request

    Responses

    segmentsSchema

    Schema
    • segments object[]

      A list of segments

    • Array [
    • id integer required

      The ID of this segment

    • name string required

      The name of this segment

    • description string nullable

      The description for this segment

    • constraints object[]required

      The list of constraints that are used in this segment

    • Array [
    • contextName string required

      The name of the context field that this constraint should apply to.

    • operator string required

      Possible values: [NOT_IN, IN, STR_ENDS_WITH, STR_STARTS_WITH, STR_CONTAINS, NUM_EQ, NUM_GT, NUM_GTE, NUM_LT, NUM_LTE, DATE_AFTER, DATE_BEFORE, SEMVER_EQ, SEMVER_GT, SEMVER_LT]

      The operator to use when evaluating this constraint. For more information about the various operators, refer to the strategy constraint operator documentation.

    • caseInsensitive boolean

      Default value: false

      Whether the operator should be case sensitive or not. Defaults to false (being case sensitive).

    • inverted boolean

      Default value: false

      Whether the result should be negated or not. If true, will turn a true result into a false result and vice versa.

    • values string[]

      The context values that should be used for constraint evaluation. Use this property instead of value for properties that accept multiple values.

    • value string

      The context value that should be used for constraint evaluation. Use this property instead of values for properties that only accept single values.

    • ]
    • usedInFeatures integer nullable

      The number of feature flags that use this segment. The number also includes the any flags with pending change requests that would add this segment.

    • usedInProjects integer nullable

      The number of projects that use this segment. The number includes any projects with pending change requests that would add this segment.

    • project string nullable

      The project the segment belongs to. Only present if the segment is a project-specific segment.

    • createdBy string nullable

      The creator's email or username

    • createdAt date-time required

      When the segment was created

    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-service-account-tokens.html b/reference/api/unleash/get-service-account-tokens.html index ec8defad22..2630ce89ad 100644 --- a/reference/api/unleash/get-service-account-tokens.html +++ b/reference/api/unleash/get-service-account-tokens.html @@ -20,15 +20,15 @@ - - + +

    List all tokens for a service account.

    GET /api/admin/service-account/:id/token

    Returns the list of all tokens for a service account identified by the id.

    Request

    Path Parameters

    • id string required
    Responses

    patsSchema

    Schema
    • pats object[]

      A collection of Personal Access Tokens

    • Array [
    • id integer

      Possible values: >= 1

      The unique identification number for this Personal Access Token. (This property is set by Unleash when the token is created and cannot be set manually: if you provide a value when creating a PAT, Unleash will ignore it.)

    • secret string

      The token used for authentication. (This property is set by Unleash when the token is created and cannot be set manually: if you provide a value when creating a PAT, Unleash will ignore it.)

    • expiresAt date-time

      The token's expiration date.

    • createdAt date-time

      When the token was created. (This property is set by Unleash when the token is created and cannot be set manually: if you provide a value when creating a PAT, Unleash will ignore it.)

    • seenAt date-time nullable

      When the token was last seen/used to authenticate with. null if it has not been used yet. (This property is set by Unleash when the token is created and cannot be set manually: if you provide a value when creating a PAT, Unleash will ignore it.)

    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-service-accounts.html b/reference/api/unleash/get-service-accounts.html index 874a5e116d..f7a62f8dbd 100644 --- a/reference/api/unleash/get-service-accounts.html +++ b/reference/api/unleash/get-service-accounts.html @@ -20,15 +20,15 @@ - - + +

    List service accounts.

    GET /api/admin/service-account

    Returns the list of all service accounts.

    Request

    Responses

    serviceAccountsSchema

    Schema
    • serviceAccounts object[]required

      A list of service accounts

    • Array [
    • id number required

      The service account id

    • isAPI boolean deprecated

      Deprecated: for internal use only, should not be exposed through the API

    • name string

      The name of the service account

    • email string deprecated

      Deprecated: service accounts don't have emails associated with them

    • username string

      The service account username

    • imageUrl string

      The service account image url

    • inviteLink string deprecated

      Deprecated: service accounts cannot be invited via an invitation link

    • loginAttempts number deprecated

      Deprecated: service accounts cannot log in to Unleash

    • emailSent boolean deprecated

      Deprecated: internal use only

    • rootRole integer

      The root role id associated with the service account

    • seenAt date-time nullable deprecated

      Deprecated. This property is always null. To find out when a service account was last seen, check its tokens list and refer to each token's lastSeen property instead.

    • createdAt date-time

      The service account creation date

    • tokens object[]

      The list of tokens associated with the service account

    • Array [
    • id integer

      Possible values: >= 1

      The unique identification number for this Personal Access Token. (This property is set by Unleash when the token is created and cannot be set manually: if you provide a value when creating a PAT, Unleash will ignore it.)

    • secret string

      The token used for authentication. (This property is set by Unleash when the token is created and cannot be set manually: if you provide a value when creating a PAT, Unleash will ignore it.)

    • expiresAt date-time

      The token's expiration date.

    • createdAt date-time

      When the token was created. (This property is set by Unleash when the token is created and cannot be set manually: if you provide a value when creating a PAT, Unleash will ignore it.)

    • seenAt date-time nullable

      When the token was last seen/used to authenticate with. null if it has not been used yet. (This property is set by Unleash when the token is created and cannot be set manually: if you provide a value when creating a PAT, Unleash will ignore it.)

    • ]
    • ]
    • rootRoles object[]

      A list of root roles that are referenced from service account objects in the serviceAccounts list

    • Array [
    • id integer required

      The role id

    • type string required

      A role can either be a global root role (applies to all projects) or a project role

    • name string required

      The name of the role

    • description string

      A more detailed description of the role and what use it's intended for

    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-simple-settings.html b/reference/api/unleash/get-simple-settings.html index 179de88031..b1447421f7 100644 --- a/reference/api/unleash/get-simple-settings.html +++ b/reference/api/unleash/get-simple-settings.html @@ -20,15 +20,15 @@ - - + +

    Get Simple auth settings

    GET /api/admin/auth/simple/settings

    Is simple authentication (username/password) enabled for this server

    Request

    Responses

    passwordAuthSchema

    Schema
    • enabled boolean

      Is username/password authentication enabled

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-strategies-by-context-field.html b/reference/api/unleash/get-strategies-by-context-field.html index f1fb899c59..380b5e28f9 100644 --- a/reference/api/unleash/get-strategies-by-context-field.html +++ b/reference/api/unleash/get-strategies-by-context-field.html @@ -20,15 +20,15 @@ - - + +

    Get strategies that use a context field

    GET /api/admin/context/:contextField/strategies

    Retrieves a list of all strategies that use the specified context field. If the context field doesn't exist, returns an empty list of strategies

    Request

    Path Parameters

    • contextField string required
    Responses

    contextFieldStrategiesSchema

    Schema
    • strategies object[]required

      List of strategies using the context field

    • Array [
    • id string required

      The ID of the strategy.

    • featureName string required

      The name of the feature that contains this strategy.

    • projectId string required

      The ID of the project that contains this feature.

    • environment string required

      The ID of the environment where this strategy is in.

    • strategyName string required

      The name of the strategy.

    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-strategies-by-segment-id.html b/reference/api/unleash/get-strategies-by-segment-id.html index 2f740344e0..16a32d3209 100644 --- a/reference/api/unleash/get-strategies-by-segment-id.html +++ b/reference/api/unleash/get-strategies-by-segment-id.html @@ -20,15 +20,15 @@ - - + +

    Get strategies that reference segment

    GET /api/admin/segments/:id/strategies

    Retrieve all strategies that reference the specified segment.

    Request

    Path Parameters

    • id string required
    Responses

    segmentStrategiesSchema

    Schema
    • strategies object[]required

      The list of strategies

    • Array [
    • id string required

      The ID of the strategy

    • featureName string required

      The name of the feature flag that this strategy belongs to.

    • projectId string required

      The ID of the project that the strategy belongs to.

    • environment string required

      The ID of the environment that the strategy belongs to.

    • strategyName string required

      The name of the strategy's type.

    • ]
    • changeRequestStrategies object[]

      A list of strategies that use this segment in active change requests.

    • Array [
    • id string

      The ID of the strategy. Not present on new strategies that haven't been added to the feature flag yet.

    • featureName string required

      The name of the feature flag that this strategy belongs to.

    • projectId string required

      The ID of the project that the strategy belongs to.

    • environment string required

      The ID of the environment that the strategy belongs to.

    • strategyName string required

      The name of the strategy's type.

    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-strategy.html b/reference/api/unleash/get-strategy.html index ea7dd4ecba..ccde7ed17f 100644 --- a/reference/api/unleash/get-strategy.html +++ b/reference/api/unleash/get-strategy.html @@ -20,15 +20,15 @@ - - + +

    Get a strategy definition

    GET /api/admin/strategies/:name

    Retrieves the definition of the strategy specified in the URL

    Request

    Path Parameters

    • name string required
    Responses

    strategySchema

    Schema
    • title string nullable

      An optional title for the strategy

    • name string required

      The name (type) of the strategy

    • displayName string nullable required

      A human friendly name for the strategy

    • description string nullable required

      A short description of the strategy

    • editable boolean required

      Whether the strategy can be edited or not. Strategies bundled with Unleash cannot be edited.

    • deprecated boolean required
    • parameters object[]required

      A list of relevant parameters for each strategy

    • Array [
    • name string
    • type string
    • description string
    • required boolean
    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-tag-type.html b/reference/api/unleash/get-tag-type.html index b5901710a0..48f472217b 100644 --- a/reference/api/unleash/get-tag-type.html +++ b/reference/api/unleash/get-tag-type.html @@ -20,15 +20,15 @@ - - + +

    Get a tag type

    GET /api/admin/tag-types/:name

    Get a tag type by name.

    Request

    Path Parameters

    • name string required
    Responses

    tagTypeSchema

    Schema
    • name string required

      The name of the tag type.

    • description string

      The description of the tag type.

    • icon string nullable

      The icon of the tag type.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-tag-types.html b/reference/api/unleash/get-tag-types.html index 0419bac830..15ebf4112f 100644 --- a/reference/api/unleash/get-tag-types.html +++ b/reference/api/unleash/get-tag-types.html @@ -20,15 +20,15 @@ - - + +

    Get all tag types

    GET /api/admin/tag-types

    Get a list of all available tag types.

    Request

    Responses

    tagTypesSchema

    Schema
    • version integer required

      The version of the schema used to model the tag types.

    • tagTypes object[]required

      The list of tag types.

    • Array [
    • name string required

      The name of the tag type.

    • description string

      The description of the tag type.

    • icon string nullable

      The icon of the tag type.

    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-tag.html b/reference/api/unleash/get-tag.html index dadbe76fef..dda5207e4a 100644 --- a/reference/api/unleash/get-tag.html +++ b/reference/api/unleash/get-tag.html @@ -20,15 +20,15 @@ - - + +

    Get a tag by type and value.

    GET /api/admin/tags/:type/:value

    Get a tag by type and value. Can be used to check whether a given tag already exists in Unleash or not.

    Request

    Path Parameters

    • type string required
    • value string required
    Responses

    tagWithVersionSchema

    Schema
    • version integer required

      The version of the schema used to model the tag.

    • tag objectrequired

      Representation of a tag

    • value string required

      Possible values: >= 2 characters and <= 50 characters

      The value of the tag

    • type string required

      Possible values: >= 2 characters and <= 50 characters

      Default value: simple

      The type of the tag

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-tags-by-type.html b/reference/api/unleash/get-tags-by-type.html index 698748e368..d3768b980f 100644 --- a/reference/api/unleash/get-tags-by-type.html +++ b/reference/api/unleash/get-tags-by-type.html @@ -20,15 +20,15 @@ - - + +

    List all tags of a given type.

    GET /api/admin/tags/:type

    List all tags of a given type. If the tag type does not exist it returns an empty list.

    Request

    Path Parameters

    • type string required
    Responses

    tagsSchema

    Schema
    • version integer required

      The version of the schema used to model the tags.

    • tags object[]required

      A list of tags.

    • Array [
    • value string required

      Possible values: >= 2 characters and <= 50 characters

      The value of the tag

    • type string required

      Possible values: >= 2 characters and <= 50 characters

      Default value: simple

      The type of the tag

    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-tags.html b/reference/api/unleash/get-tags.html index bc7908e1bb..9e2d1d2b0c 100644 --- a/reference/api/unleash/get-tags.html +++ b/reference/api/unleash/get-tags.html @@ -20,15 +20,15 @@ - - + +

    List all tags.

    GET /api/admin/tags

    List all tags available in Unleash.

    Request

    Responses

    tagsSchema

    Schema
    • version integer required

      The version of the schema used to model the tags.

    • tags object[]required

      A list of tags.

    • Array [
    • value string required

      Possible values: >= 2 characters and <= 50 characters

      The value of the tag

    • type string required

      Possible values: >= 2 characters and <= 50 characters

      Default value: simple

      The type of the tag

    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-telemetry-settings.html b/reference/api/unleash/get-telemetry-settings.html index 3ae53a0bdb..c6ce4b64ef 100644 --- a/reference/api/unleash/get-telemetry-settings.html +++ b/reference/api/unleash/get-telemetry-settings.html @@ -20,15 +20,15 @@ - - + +
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-ui-config.html b/reference/api/unleash/get-ui-config.html index e78424eb6e..56262d3ea7 100644 --- a/reference/api/unleash/get-ui-config.html +++ b/reference/api/unleash/get-ui-config.html @@ -20,15 +20,15 @@ - - + +

    Get UI configuration

    GET /api/admin/ui-config

    Retrieves the full configuration used to set up the Unleash Admin UI.

    Request

    Responses

    uiConfigSchema

    Schema
    • slogan string

      The slogan to display in the UI footer.

    • name string

      The name of this Unleash instance. Used to build the text in the footer.

    • version string required

      The current version of Unleash

    • environment string

      What kind of Unleash instance it is: Enterprise, Pro, or Open source

    • unleashUrl string required

      The URL of the Unleash instance.

    • baseUriPath string required

      The base URI path at which this Unleash instance is listening.

    • feedbackUriPath string

      The URI path at which the feedback endpoint is listening.

    • disablePasswordAuth boolean

      Whether password authentication should be disabled or not.

    • emailEnabled boolean

      Whether this instance can send out emails or not.

    • maintenanceMode boolean

      Whether maintenance mode is currently active or not.

    • segmentValuesLimit number

      The maximum number of values that can be used in a single segment.

    • strategySegmentsLimit number

      The maximum number of segments that can be applied to a single strategy.

    • networkViewEnabled boolean

      Whether to enable the Unleash network view or not.

    • frontendApiOrigins string[]

      The list of origins that the front-end API should accept requests from.

    • flags object

      Additional (largely experimental) features that are enabled in this Unleash instance.

    • anyOf
    • links object[]

      Relevant links to use in the UI.

    • authenticationType string

      Possible values: [open-source, demo, enterprise, hosted, custom, none]

      The type of authentication enabled for this Unleash instance

    • versionInfo objectrequired

      Detailed information about an Unleash version

    • current objectrequired

      The current version of Unleash.

    • oss string

      The OSS version used when building this Unleash instance, represented as a git revision belonging to the main Unleash git repo

    • enterprise string

      The Enterpris version of Unleash used to build this instance, represented as a git revision belonging to the Unleash Enterprise repository. Will be an empty string if no enterprise version was used,

    • latest objectrequired

      Information about the latest available Unleash releases. Will be an empty object if no data is available.

    • oss string

      The latest available OSS version of Unleash

    • enterprise string

      The latest available Enterprise version of Unleash

    • isLatest boolean required

      Whether the Unleash server is running the latest release (true) or if there are updates available (false)

    • instanceId string

      The instance identifier of the Unleash instance

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-user.html b/reference/api/unleash/get-user.html index 58c5f3ccda..dc7e095548 100644 --- a/reference/api/unleash/get-user.html +++ b/reference/api/unleash/get-user.html @@ -20,15 +20,15 @@ - - + +

    Get user

    GET /api/admin/user-admin/:id

    Will return a single user by id

    Request

    Path Parameters

    • id string required
    Responses

    userSchema

    Schema
    • id integer required

      The user id

    • isAPI boolean deprecated

      (Deprecated): Used internally to know which operations the user should be allowed to perform

    • name string nullable

      Name of the user

    • email string

      Email of the user

    • username string nullable

      A unique username for the user

    • imageUrl string

      URL used for the userprofile image

    • inviteLink string

      If the user is actively inviting other users, this is the link that can be shared with other users

    • loginAttempts integer

      How many unsuccessful attempts at logging in has the user made

    • emailSent boolean

      Is the welcome email sent to the user or not

    • rootRole integer

      Which root role this user is assigned

    • seenAt date-time nullable

      The last time this user logged in

    • createdAt date-time

      The user was created at this time

    • accountType string

      A user is either an actual User or a Service Account

    • permissions string[]

      Deprecated

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-users.html b/reference/api/unleash/get-users.html index ffa3e64662..70199349d9 100644 --- a/reference/api/unleash/get-users.html +++ b/reference/api/unleash/get-users.html @@ -20,15 +20,15 @@ - - + +

    Get all users and [root roles](https://docs.getunleash.io/reference/rbac#predefined-roles)

    GET /api/admin/user-admin

    Will return all users and all available root roles for the Unleash instance.

    Request

    Responses

    usersSchema

    Schema
    • users object[]required

      A list of users in the Unleash instance.

    • Array [
    • id integer required

      The user id

    • isAPI boolean deprecated

      (Deprecated): Used internally to know which operations the user should be allowed to perform

    • name string nullable

      Name of the user

    • email string

      Email of the user

    • username string nullable

      A unique username for the user

    • imageUrl string

      URL used for the userprofile image

    • inviteLink string

      If the user is actively inviting other users, this is the link that can be shared with other users

    • loginAttempts integer

      How many unsuccessful attempts at logging in has the user made

    • emailSent boolean

      Is the welcome email sent to the user or not

    • rootRole integer

      Which root role this user is assigned

    • seenAt date-time nullable

      The last time this user logged in

    • createdAt date-time

      The user was created at this time

    • accountType string

      A user is either an actual User or a Service Account

    • permissions string[]

      Deprecated

    • ]
    • rootRoles object[]

      A list of root roles in the Unleash instance.

    • Array [
    • id integer required

      The role id

    • type string required

      A role can either be a global root role (applies to all projects) or a project role

    • name string required

      The name of the role

    • description string

      A more detailed description of the role and what use it's intended for

    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/get-valid-tokens.html b/reference/api/unleash/get-valid-tokens.html index d28bbaac7c..f81768061f 100644 --- a/reference/api/unleash/get-valid-tokens.html +++ b/reference/api/unleash/get-valid-tokens.html @@ -20,15 +20,15 @@ - - + +

    Check which tokens are valid

    POST /edge/validate

    This operation accepts a list of tokens to validate. Unleash will validate each token you provide. For each valid token you provide, Unleash will return the token along with its type and which projects it has access to.

    Request

    Body

    required

    tokenStringListSchema

    • tokens string[] required

      Tokens that we want to get access information about

    Responses

    validatedEdgeTokensSchema

    Schema
    • tokens object[]required

      The list of Unleash token objects. Each object contains the token itself and some additional metadata.

    • Array [
    • projects string[] required

      The list of projects this token has access to. If the token has access to specific projects they will be listed here. If the token has access to all projects it will be represented as [*]

    • type string required

      Possible values: [client, admin, frontend]

      The API token's type. Unleash supports three different types of API tokens (ADMIN, CLIENT, FRONTEND). They all have varying access, so when validating a token it's important to know what kind you're dealing with

    • token string required

      The actual token value. Unleash API tokens are comprised of three parts. <project(s)>:.randomcharacters

    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/import-export.html b/reference/api/unleash/import-export.html index 0871100a23..684abdd38c 100644 --- a/reference/api/unleash/import-export.html +++ b/reference/api/unleash/import-export.html @@ -20,15 +20,15 @@ - - + +
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/import-toggles.html b/reference/api/unleash/import-toggles.html index 88e2c1c9f3..321a485b51 100644 --- a/reference/api/unleash/import-toggles.html +++ b/reference/api/unleash/import-toggles.html @@ -20,15 +20,15 @@ - - + +

    Import feature toggles

    POST /api/admin/features-batch/import

    Import feature toggles into a specific project and environment.

    Request

    Body

    required

    importTogglesSchema

    • project string required

      The exported project

    • environment string required

      The exported environment

    • data objectrequired

      The result of the export operation, providing you with the feature toggle definitions, strategy definitions and the rest of the elements relevant to the features (tags, environments etc.)

    • features object[]required

      All the exported features.

    • Array [
    • name string required

      Unique feature name

    • type string

      Type of the toggle e.g. experiment, kill-switch, release, operational, permission

    • description string nullable

      Detailed description of the feature

    • archived boolean

      true if the feature is archived

    • project string

      Name of the project the feature belongs to

    • enabled boolean

      true if the feature is enabled, otherwise false.

    • stale boolean

      true if the feature is stale based on the age and feature type, otherwise false.

    • favorite boolean

      true if the feature was favorited, otherwise false.

    • impressionData boolean

      true if the impression data collection is enabled for the feature, otherwise false.

    • createdAt date-time nullable

      The date the feature was created

    • archivedAt date-time nullable

      The date the feature was archived

    • lastSeenAt date-time nullable deprecated

      The date when metrics where last collected for the feature. This field is deprecated, use the one in featureEnvironmentSchema

    • environments object[]

      The list of environments where the feature can be used

    • Array [
    • name string required

      The name of the environment

    • featureName string

      The name of the feature

    • environment string

      The name of the environment

    • type string

      The type of the environment

    • enabled boolean required

      true if the feature is enabled for the environment, otherwise false.

    • sortOrder number

      The sort order of the feature environment in the feature environments list

    • variantCount number

      The number of defined variants

    • strategies object[]

      A list of activation strategies for the feature environment

    • Array [
    • id string

      A uuid for the feature strategy

    • name string required

      The name or type of strategy

    • title string nullable

      A descriptive title for the strategy

    • disabled boolean nullable

      A toggle to disable the strategy. defaults to false. Disabled strategies are not evaluated or returned to the SDKs

    • featureName string

      The name or feature the strategy is attached to

    • sortOrder number

      The order of the strategy in the list

    • segments number[]

      A list of segment ids attached to the strategy

    • constraints object[]

      A list of the constraints attached to the strategy. See https://docs.getunleash.io/reference/strategy-constraints

    • Array [
    • contextName string required

      The name of the context field that this constraint should apply to.

    • operator string required

      Possible values: [NOT_IN, IN, STR_ENDS_WITH, STR_STARTS_WITH, STR_CONTAINS, NUM_EQ, NUM_GT, NUM_GTE, NUM_LT, NUM_LTE, DATE_AFTER, DATE_BEFORE, SEMVER_EQ, SEMVER_GT, SEMVER_LT]

      The operator to use when evaluating this constraint. For more information about the various operators, refer to the strategy constraint operator documentation.

    • caseInsensitive boolean

      Default value: false

      Whether the operator should be case sensitive or not. Defaults to false (being case sensitive).

    • inverted boolean

      Default value: false

      Whether the result should be negated or not. If true, will turn a true result into a false result and vice versa.

    • values string[]

      The context values that should be used for constraint evaluation. Use this property instead of value for properties that accept multiple values.

    • value string

      The context value that should be used for constraint evaluation. Use this property instead of values for properties that only accept single values.

    • ]
    • variants object[]

      Strategy level variants

    • Array [
    • name string required

      The variant name. Must be unique for this feature toggle

    • weight integer required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is an integer between 0 and 1000. See the section on variant weights for more information

    • weightType string required

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000. Refer to the variant weight documentation.

    • stickiness string required

      The stickiness to use for distribution of this variant. Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • ]
    • parameters object

      A list of parameters for a strategy

    • property name* string
    • ]
    • variants object[]

      A list of variants for the feature environment

    • Array [
    • name string required

      The variants name. Is unique for this feature toggle

    • weight number required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is a number between 0 and 1000. See the section on variant weights for more information

    • weightType string

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000

    • stickiness string

      Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • overrides object[]

      Overrides assigning specific variants to specific users. The weighting system automatically assigns users to specific groups for you, but any overrides in this list will take precedence.

    • Array [
    • contextName string required

      The name of the context field used to determine overrides

    • values string[] required

      Which values that should be overriden

    • ]
    • ]
    • lastSeenAt date-time nullable

      The date when metrics where last collected for the feature environment

    • hasStrategies boolean

      Whether the feature has any strategies defined.

    • hasEnabledStrategies boolean

      Whether the feature has any enabled strategies defined.

    • ]
    • variants object[]deprecated

      The list of feature variants

    • Array [
    • name string required

      The variants name. Is unique for this feature toggle

    • weight number required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is a number between 0 and 1000. See the section on variant weights for more information

    • weightType string

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000

    • stickiness string

      Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • overrides object[]

      Overrides assigning specific variants to specific users. The weighting system automatically assigns users to specific groups for you, but any overrides in this list will take precedence.

    • Array [
    • contextName string required

      The name of the context field used to determine overrides

    • values string[] required

      Which values that should be overriden

    • ]
    • ]
    • strategies object[] deprecated

      This is a legacy field that will be deprecated

    • tags object[]nullable

      The list of feature tags

    • Array [
    • value string required

      Possible values: >= 2 characters and <= 50 characters

      The value of the tag

    • type string required

      Possible values: >= 2 characters and <= 50 characters

      Default value: simple

      The type of the tag

    • ]
    • children string[]

      The list of child feature names. This is an experimental field and may change.

    • dependencies object[]

      The list of parent dependencies. This is an experimental field and may change.

    • Array [
    • feature string required

      The name of the parent feature

    • enabled boolean

      Whether the parent feature is enabled or not

    • variants string[]

      The list of variants the parent feature should resolve to. Only valid when feature is enabled.

    • ]
    • ]
    • featureStrategies object[]required

      All strategy instances that are used by the exported features in the features list.

    • Array [
    • id string

      A uuid for the feature strategy

    • name string required

      The name or type of strategy

    • title string nullable

      A descriptive title for the strategy

    • disabled boolean nullable

      A toggle to disable the strategy. defaults to false. Disabled strategies are not evaluated or returned to the SDKs

    • featureName string

      The name or feature the strategy is attached to

    • sortOrder number

      The order of the strategy in the list

    • segments number[]

      A list of segment ids attached to the strategy

    • constraints object[]

      A list of the constraints attached to the strategy. See https://docs.getunleash.io/reference/strategy-constraints

    • Array [
    • contextName string required

      The name of the context field that this constraint should apply to.

    • operator string required

      Possible values: [NOT_IN, IN, STR_ENDS_WITH, STR_STARTS_WITH, STR_CONTAINS, NUM_EQ, NUM_GT, NUM_GTE, NUM_LT, NUM_LTE, DATE_AFTER, DATE_BEFORE, SEMVER_EQ, SEMVER_GT, SEMVER_LT]

      The operator to use when evaluating this constraint. For more information about the various operators, refer to the strategy constraint operator documentation.

    • caseInsensitive boolean

      Default value: false

      Whether the operator should be case sensitive or not. Defaults to false (being case sensitive).

    • inverted boolean

      Default value: false

      Whether the result should be negated or not. If true, will turn a true result into a false result and vice versa.

    • values string[]

      The context values that should be used for constraint evaluation. Use this property instead of value for properties that accept multiple values.

    • value string

      The context value that should be used for constraint evaluation. Use this property instead of values for properties that only accept single values.

    • ]
    • variants object[]

      Strategy level variants

    • Array [
    • name string required

      The variant name. Must be unique for this feature toggle

    • weight integer required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is an integer between 0 and 1000. See the section on variant weights for more information

    • weightType string required

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000. Refer to the variant weight documentation.

    • stickiness string required

      The stickiness to use for distribution of this variant. Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • ]
    • parameters object

      A list of parameters for a strategy

    • property name* string
    • ]
    • featureEnvironments object[]

      Environment-specific configuration for all the features in the features list. Includes data such as whether the feature is enabled in the selected export environment, whether there are any variants assigned, etc.

    • Array [
    • name string required

      The name of the environment

    • featureName string

      The name of the feature

    • environment string

      The name of the environment

    • type string

      The type of the environment

    • enabled boolean required

      true if the feature is enabled for the environment, otherwise false.

    • sortOrder number

      The sort order of the feature environment in the feature environments list

    • variantCount number

      The number of defined variants

    • strategies object[]

      A list of activation strategies for the feature environment

    • Array [
    • id string

      A uuid for the feature strategy

    • name string required

      The name or type of strategy

    • title string nullable

      A descriptive title for the strategy

    • disabled boolean nullable

      A toggle to disable the strategy. defaults to false. Disabled strategies are not evaluated or returned to the SDKs

    • featureName string

      The name or feature the strategy is attached to

    • sortOrder number

      The order of the strategy in the list

    • segments number[]

      A list of segment ids attached to the strategy

    • constraints object[]

      A list of the constraints attached to the strategy. See https://docs.getunleash.io/reference/strategy-constraints

    • Array [
    • contextName string required

      The name of the context field that this constraint should apply to.

    • operator string required

      Possible values: [NOT_IN, IN, STR_ENDS_WITH, STR_STARTS_WITH, STR_CONTAINS, NUM_EQ, NUM_GT, NUM_GTE, NUM_LT, NUM_LTE, DATE_AFTER, DATE_BEFORE, SEMVER_EQ, SEMVER_GT, SEMVER_LT]

      The operator to use when evaluating this constraint. For more information about the various operators, refer to the strategy constraint operator documentation.

    • caseInsensitive boolean

      Default value: false

      Whether the operator should be case sensitive or not. Defaults to false (being case sensitive).

    • inverted boolean

      Default value: false

      Whether the result should be negated or not. If true, will turn a true result into a false result and vice versa.

    • values string[]

      The context values that should be used for constraint evaluation. Use this property instead of value for properties that accept multiple values.

    • value string

      The context value that should be used for constraint evaluation. Use this property instead of values for properties that only accept single values.

    • ]
    • variants object[]

      Strategy level variants

    • Array [
    • name string required

      The variant name. Must be unique for this feature toggle

    • weight integer required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is an integer between 0 and 1000. See the section on variant weights for more information

    • weightType string required

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000. Refer to the variant weight documentation.

    • stickiness string required

      The stickiness to use for distribution of this variant. Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • ]
    • parameters object

      A list of parameters for a strategy

    • property name* string
    • ]
    • variants object[]

      A list of variants for the feature environment

    • Array [
    • name string required

      The variants name. Is unique for this feature toggle

    • weight number required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is a number between 0 and 1000. See the section on variant weights for more information

    • weightType string

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000

    • stickiness string

      Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • overrides object[]

      Overrides assigning specific variants to specific users. The weighting system automatically assigns users to specific groups for you, but any overrides in this list will take precedence.

    • Array [
    • contextName string required

      The name of the context field used to determine overrides

    • values string[] required

      Which values that should be overriden

    • ]
    • ]
    • lastSeenAt date-time nullable

      The date when metrics where last collected for the feature environment

    • hasStrategies boolean

      Whether the feature has any strategies defined.

    • hasEnabledStrategies boolean

      Whether the feature has any enabled strategies defined.

    • ]
    • contextFields object[]

      A list of all the context fields that are in use by any of the strategies in the featureStrategies list.

    • Array [
    • name string required

      The name of the context field

    • description string nullable

      The description of the context field.

    • stickiness boolean

      Does this context field support being used for stickiness calculations

    • sortOrder integer

      Used when sorting a list of context fields. Is also used as a tiebreaker if a list of context fields is sorted alphabetically.

    • createdAt date-time nullable

      When this context field was created

    • usedInFeatures integer nullable

      Number of projects where this context field is used in

    • usedInProjects integer nullable

      Number of projects where this context field is used in

    • legalValues object[]

      Allowed values for this context field schema. Can be used to narrow down accepted input

    • Array [
    • value string required

      The valid value

    • description string

      Describes this specific legal value

    • ]
    • ]
    • featureTags object[]

      A list of all the tags that have been applied to any of the features in the features list.

    • Array [
    • featureName string required

      The name of the feature this tag is applied to

    • tagType string

      The [type](https://docs.getunleash.io/reference/tags#tag-types tag types) of the tag

    • tagValue string required

      The value of the tag

    • type string deprecated

      The [type](https://docs.getunleash.io/reference/tags#tag-types tag types) of the tag. This property is deprecated and will be removed in a future version of Unleash. Superseded by the tagType property.

    • value string deprecated

      The value of the tag. This property is deprecated and will be removed in a future version of Unleash. Superseded by the tagValue property.

    • createdByUserId number nullable

      The id of the user who created this tag

    • ]
    • segments object[]

      A list of all the segments that are used by the strategies in the featureStrategies list.

    • Array [
    • id number required
    • name string required
    • ]
    • tagTypes object[]required

      A list of all of the tag types that are used in the featureTags list.

    • Array [
    • name string required

      The name of the tag type.

    • description string

      The description of the tag type.

    • icon string nullable

      The icon of the tag type.

    • ]
    • dependencies object[]

      A list of all the dependencies for features in features list.

    • Array [
    • feature string required

      The name of the child feature.

    • dependencies object[]required

      List of parent features for the child feature

    • Array [
    • feature string required

      The name of the feature we depend on.

    • enabled boolean

      Whether the parent feature should be enabled. When false variants are ignored. true by default.

    • variants string[]

      The list of variants the parent feature should resolve to. Leave empty when you only want to check the enabled status.

    • ]
    • ]
    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/import.html b/reference/api/unleash/import.html index 050f404dd1..64c8e2b0d2 100644 --- a/reference/api/unleash/import.html +++ b/reference/api/unleash/import.html @@ -20,15 +20,15 @@ - - + +

    Import state (deprecated)

    POST /api/admin/state/import
    deprecated

    This endpoint has been deprecated and may be removed in future versions of the API.

    Imports state into the system. Deprecated in favor of /api/admin/features-batch/import

    Request

    Body

    required

    stateSchema

    • version integer required

      The version of the schema used to describe the state

    • features object[]

      A list of features

    • Array [
    • name string required

      Unique feature name

    • type string

      Type of the toggle e.g. experiment, kill-switch, release, operational, permission

    • description string nullable

      Detailed description of the feature

    • archived boolean

      true if the feature is archived

    • project string

      Name of the project the feature belongs to

    • enabled boolean

      true if the feature is enabled, otherwise false.

    • stale boolean

      true if the feature is stale based on the age and feature type, otherwise false.

    • favorite boolean

      true if the feature was favorited, otherwise false.

    • impressionData boolean

      true if the impression data collection is enabled for the feature, otherwise false.

    • createdAt date-time nullable

      The date the feature was created

    • archivedAt date-time nullable

      The date the feature was archived

    • lastSeenAt date-time nullable deprecated

      The date when metrics where last collected for the feature. This field is deprecated, use the one in featureEnvironmentSchema

    • environments object[]

      The list of environments where the feature can be used

    • Array [
    • name string required

      The name of the environment

    • featureName string

      The name of the feature

    • environment string

      The name of the environment

    • type string

      The type of the environment

    • enabled boolean required

      true if the feature is enabled for the environment, otherwise false.

    • sortOrder number

      The sort order of the feature environment in the feature environments list

    • variantCount number

      The number of defined variants

    • strategies object[]

      A list of activation strategies for the feature environment

    • Array [
    • id string

      A uuid for the feature strategy

    • name string required

      The name or type of strategy

    • title string nullable

      A descriptive title for the strategy

    • disabled boolean nullable

      A toggle to disable the strategy. defaults to false. Disabled strategies are not evaluated or returned to the SDKs

    • featureName string

      The name or feature the strategy is attached to

    • sortOrder number

      The order of the strategy in the list

    • segments number[]

      A list of segment ids attached to the strategy

    • constraints object[]

      A list of the constraints attached to the strategy. See https://docs.getunleash.io/reference/strategy-constraints

    • Array [
    • contextName string required

      The name of the context field that this constraint should apply to.

    • operator string required

      Possible values: [NOT_IN, IN, STR_ENDS_WITH, STR_STARTS_WITH, STR_CONTAINS, NUM_EQ, NUM_GT, NUM_GTE, NUM_LT, NUM_LTE, DATE_AFTER, DATE_BEFORE, SEMVER_EQ, SEMVER_GT, SEMVER_LT]

      The operator to use when evaluating this constraint. For more information about the various operators, refer to the strategy constraint operator documentation.

    • caseInsensitive boolean

      Default value: false

      Whether the operator should be case sensitive or not. Defaults to false (being case sensitive).

    • inverted boolean

      Default value: false

      Whether the result should be negated or not. If true, will turn a true result into a false result and vice versa.

    • values string[]

      The context values that should be used for constraint evaluation. Use this property instead of value for properties that accept multiple values.

    • value string

      The context value that should be used for constraint evaluation. Use this property instead of values for properties that only accept single values.

    • ]
    • variants object[]

      Strategy level variants

    • Array [
    • name string required

      The variant name. Must be unique for this feature toggle

    • weight integer required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is an integer between 0 and 1000. See the section on variant weights for more information

    • weightType string required

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000. Refer to the variant weight documentation.

    • stickiness string required

      The stickiness to use for distribution of this variant. Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • ]
    • parameters object

      A list of parameters for a strategy

    • property name* string
    • ]
    • variants object[]

      A list of variants for the feature environment

    • Array [
    • name string required

      The variants name. Is unique for this feature toggle

    • weight number required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is a number between 0 and 1000. See the section on variant weights for more information

    • weightType string

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000

    • stickiness string

      Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • overrides object[]

      Overrides assigning specific variants to specific users. The weighting system automatically assigns users to specific groups for you, but any overrides in this list will take precedence.

    • Array [
    • contextName string required

      The name of the context field used to determine overrides

    • values string[] required

      Which values that should be overriden

    • ]
    • ]
    • lastSeenAt date-time nullable

      The date when metrics where last collected for the feature environment

    • hasStrategies boolean

      Whether the feature has any strategies defined.

    • hasEnabledStrategies boolean

      Whether the feature has any enabled strategies defined.

    • ]
    • variants object[]deprecated

      The list of feature variants

    • Array [
    • name string required

      The variants name. Is unique for this feature toggle

    • weight number required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is a number between 0 and 1000. See the section on variant weights for more information

    • weightType string

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000

    • stickiness string

      Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • overrides object[]

      Overrides assigning specific variants to specific users. The weighting system automatically assigns users to specific groups for you, but any overrides in this list will take precedence.

    • Array [
    • contextName string required

      The name of the context field used to determine overrides

    • values string[] required

      Which values that should be overriden

    • ]
    • ]
    • strategies object[] deprecated

      This is a legacy field that will be deprecated

    • tags object[]nullable

      The list of feature tags

    • Array [
    • value string required

      Possible values: >= 2 characters and <= 50 characters

      The value of the tag

    • type string required

      Possible values: >= 2 characters and <= 50 characters

      Default value: simple

      The type of the tag

    • ]
    • children string[]

      The list of child feature names. This is an experimental field and may change.

    • dependencies object[]

      The list of parent dependencies. This is an experimental field and may change.

    • Array [
    • feature string required

      The name of the parent feature

    • enabled boolean

      Whether the parent feature is enabled or not

    • variants string[]

      The list of variants the parent feature should resolve to. Only valid when feature is enabled.

    • ]
    • ]
    • strategies object[]

      A list of strategies

    • Array [
    • title string nullable

      An optional title for the strategy

    • name string required

      The name (type) of the strategy

    • displayName string nullable required

      A human friendly name for the strategy

    • description string nullable required

      A short description of the strategy

    • editable boolean required

      Whether the strategy can be edited or not. Strategies bundled with Unleash cannot be edited.

    • deprecated boolean required
    • parameters object[]required

      A list of relevant parameters for each strategy

    • Array [
    • name string
    • type string
    • description string
    • required boolean
    • ]
    • ]
    • tags object[]

      A list of tags

    • Array [
    • value string required

      Possible values: >= 2 characters and <= 50 characters

      The value of the tag

    • type string required

      Possible values: >= 2 characters and <= 50 characters

      Default value: simple

      The type of the tag

    • ]
    • tagTypes object[]

      A list of tag types

    • Array [
    • name string required

      The name of the tag type.

    • description string

      The description of the tag type.

    • icon string nullable

      The icon of the tag type.

    • ]
    • featureTags object[]

      A list of tags applied to features

    • Array [
    • featureName string required

      The name of the feature this tag is applied to

    • tagType string

      The [type](https://docs.getunleash.io/reference/tags#tag-types tag types) of the tag

    • tagValue string required

      The value of the tag

    • type string deprecated

      The [type](https://docs.getunleash.io/reference/tags#tag-types tag types) of the tag. This property is deprecated and will be removed in a future version of Unleash. Superseded by the tagType property.

    • value string deprecated

      The value of the tag. This property is deprecated and will be removed in a future version of Unleash. Superseded by the tagValue property.

    • createdByUserId number nullable

      The id of the user who created this tag

    • ]
    • projects object[]

      A list of projects

    • Array [
    • id string required

      The id of this project

    • name string required

      The name of this project

    • description string nullable

      Additional information about the project

    • health number

      An indicator of the project's health on a scale from 0 to 100

    • featureCount number

      The number of features this project has

    • memberCount number

      The number of members this project has

    • createdAt date-time

      When this project was created.

    • updatedAt date-time nullable

      When this project was last updated.

    • favorite boolean

      true if the project was favorited, otherwise false.

    • mode string

      Possible values: [open, protected, private]

      The project's collaboration mode. Determines whether non-project members can submit change requests or not.

    • defaultStickiness string

      A default stickiness for the project affecting the default stickiness value for variants and Gradual Rollout strategy

    • ]
    • featureStrategies object[]

      A list of feature strategies as applied to features

    • Array [
    • id string

      A uuid for the feature strategy

    • name string required

      The name or type of strategy

    • title string nullable

      A descriptive title for the strategy

    • disabled boolean nullable

      A toggle to disable the strategy. defaults to false. Disabled strategies are not evaluated or returned to the SDKs

    • featureName string

      The name or feature the strategy is attached to

    • sortOrder number

      The order of the strategy in the list

    • segments number[]

      A list of segment ids attached to the strategy

    • constraints object[]

      A list of the constraints attached to the strategy. See https://docs.getunleash.io/reference/strategy-constraints

    • Array [
    • contextName string required

      The name of the context field that this constraint should apply to.

    • operator string required

      Possible values: [NOT_IN, IN, STR_ENDS_WITH, STR_STARTS_WITH, STR_CONTAINS, NUM_EQ, NUM_GT, NUM_GTE, NUM_LT, NUM_LTE, DATE_AFTER, DATE_BEFORE, SEMVER_EQ, SEMVER_GT, SEMVER_LT]

      The operator to use when evaluating this constraint. For more information about the various operators, refer to the strategy constraint operator documentation.

    • caseInsensitive boolean

      Default value: false

      Whether the operator should be case sensitive or not. Defaults to false (being case sensitive).

    • inverted boolean

      Default value: false

      Whether the result should be negated or not. If true, will turn a true result into a false result and vice versa.

    • values string[]

      The context values that should be used for constraint evaluation. Use this property instead of value for properties that accept multiple values.

    • value string

      The context value that should be used for constraint evaluation. Use this property instead of values for properties that only accept single values.

    • ]
    • variants object[]

      Strategy level variants

    • Array [
    • name string required

      The variant name. Must be unique for this feature toggle

    • weight integer required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is an integer between 0 and 1000. See the section on variant weights for more information

    • weightType string required

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000. Refer to the variant weight documentation.

    • stickiness string required

      The stickiness to use for distribution of this variant. Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • ]
    • parameters object

      A list of parameters for a strategy

    • property name* string
    • ]
    • featureEnvironments object[]

      A list of feature environment configurations

    • Array [
    • name string required

      The name of the environment

    • featureName string

      The name of the feature

    • environment string

      The name of the environment

    • type string

      The type of the environment

    • enabled boolean required

      true if the feature is enabled for the environment, otherwise false.

    • sortOrder number

      The sort order of the feature environment in the feature environments list

    • variantCount number

      The number of defined variants

    • strategies object[]

      A list of activation strategies for the feature environment

    • Array [
    • id string

      A uuid for the feature strategy

    • name string required

      The name or type of strategy

    • title string nullable

      A descriptive title for the strategy

    • disabled boolean nullable

      A toggle to disable the strategy. defaults to false. Disabled strategies are not evaluated or returned to the SDKs

    • featureName string

      The name or feature the strategy is attached to

    • sortOrder number

      The order of the strategy in the list

    • segments number[]

      A list of segment ids attached to the strategy

    • constraints object[]

      A list of the constraints attached to the strategy. See https://docs.getunleash.io/reference/strategy-constraints

    • Array [
    • contextName string required

      The name of the context field that this constraint should apply to.

    • operator string required

      Possible values: [NOT_IN, IN, STR_ENDS_WITH, STR_STARTS_WITH, STR_CONTAINS, NUM_EQ, NUM_GT, NUM_GTE, NUM_LT, NUM_LTE, DATE_AFTER, DATE_BEFORE, SEMVER_EQ, SEMVER_GT, SEMVER_LT]

      The operator to use when evaluating this constraint. For more information about the various operators, refer to the strategy constraint operator documentation.

    • caseInsensitive boolean

      Default value: false

      Whether the operator should be case sensitive or not. Defaults to false (being case sensitive).

    • inverted boolean

      Default value: false

      Whether the result should be negated or not. If true, will turn a true result into a false result and vice versa.

    • values string[]

      The context values that should be used for constraint evaluation. Use this property instead of value for properties that accept multiple values.

    • value string

      The context value that should be used for constraint evaluation. Use this property instead of values for properties that only accept single values.

    • ]
    • variants object[]

      Strategy level variants

    • Array [
    • name string required

      The variant name. Must be unique for this feature toggle

    • weight integer required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is an integer between 0 and 1000. See the section on variant weights for more information

    • weightType string required

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000. Refer to the variant weight documentation.

    • stickiness string required

      The stickiness to use for distribution of this variant. Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • ]
    • parameters object

      A list of parameters for a strategy

    • property name* string
    • ]
    • variants object[]

      A list of variants for the feature environment

    • Array [
    • name string required

      The variants name. Is unique for this feature toggle

    • weight number required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is a number between 0 and 1000. See the section on variant weights for more information

    • weightType string

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000

    • stickiness string

      Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • overrides object[]

      Overrides assigning specific variants to specific users. The weighting system automatically assigns users to specific groups for you, but any overrides in this list will take precedence.

    • Array [
    • contextName string required

      The name of the context field used to determine overrides

    • values string[] required

      Which values that should be overriden

    • ]
    • ]
    • lastSeenAt date-time nullable

      The date when metrics where last collected for the feature environment

    • hasStrategies boolean

      Whether the feature has any strategies defined.

    • hasEnabledStrategies boolean

      Whether the feature has any enabled strategies defined.

    • ]
    • environments object[]

      A list of environments

    • Array [
    • name string required

      The name of the environment

    • type string required
    • enabled boolean required

      true if the environment is enabled for the project, otherwise false.

    • protected boolean required

      true if the environment is protected, otherwise false. A protected environment can not be deleted.

    • sortOrder integer required

      Priority of the environment in a list of environments, the lower the value, the higher up in the list the environment will appear. Needs to be an integer

    • projectCount integer nullable

      The number of projects with this environment

    • apiTokenCount integer nullable

      The number of API tokens for the project environment

    • enabledToggleCount integer nullable

      The number of enabled toggles for the project environment

    • ]
    • segments object[]

      A list of segments

    • Array [
    • id number required

      The segment's id.

    • name string

      The name of the segment.

    • constraints object[]required

      List of constraints that determine which users are part of the segment

    • Array [
    • contextName string required

      The name of the context field that this constraint should apply to.

    • operator string required

      Possible values: [NOT_IN, IN, STR_ENDS_WITH, STR_STARTS_WITH, STR_CONTAINS, NUM_EQ, NUM_GT, NUM_GTE, NUM_LT, NUM_LTE, DATE_AFTER, DATE_BEFORE, SEMVER_EQ, SEMVER_GT, SEMVER_LT]

      The operator to use when evaluating this constraint. For more information about the various operators, refer to the strategy constraint operator documentation.

    • caseInsensitive boolean

      Default value: false

      Whether the operator should be case sensitive or not. Defaults to false (being case sensitive).

    • inverted boolean

      Default value: false

      Whether the result should be negated or not. If true, will turn a true result into a false result and vice versa.

    • values string[]

      The context values that should be used for constraint evaluation. Use this property instead of value for properties that accept multiple values.

    • value string

      The context value that should be used for constraint evaluation. Use this property instead of values for properties that only accept single values.

    • ]
    • description string nullable

      The description of the segment.

    • createdAt date-time

      The time the segment was created as a RFC 3339-conformant timestamp.

    • createdBy string

      Which user created this segment

    • project string nullable

      The project the segment relates to, if applicable.

    • ]
    • featureStrategySegments object[]

      A list of segment/strategy pairings

    • Array [
    • segmentId integer required

      The ID of the segment

    • featureStrategyId string required

      The ID of the strategy

    • ]
    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/instance-admin.html b/reference/api/unleash/instance-admin.html index 647c930073..6f074e5a0d 100644 --- a/reference/api/unleash/instance-admin.html +++ b/reference/api/unleash/instance-admin.html @@ -20,15 +20,15 @@ - - + +
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/list-parent-options.html b/reference/api/unleash/list-parent-options.html index db45253545..0e7dde209c 100644 --- a/reference/api/unleash/list-parent-options.html +++ b/reference/api/unleash/list-parent-options.html @@ -20,15 +20,15 @@ - - + +

    List parent options.

    GET /api/admin/projects/:projectId/features/:child/parents

    List available parents who have no transitive dependencies.

    Request

    Path Parameters

    • projectId string required
    • child string required
    Responses

    parentFeatureOptionsSchema

    Schema
    • Array [
    • string
    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/list-tags.html b/reference/api/unleash/list-tags.html index 325a4422e7..21cd85e80a 100644 --- a/reference/api/unleash/list-tags.html +++ b/reference/api/unleash/list-tags.html @@ -20,15 +20,15 @@ - - + +

    Get all tags for a feature.

    GET /api/admin/features/:featureName/tags

    Retrieves all the tags for a feature name. If the feature does not exist it returns an empty list.

    Request

    Path Parameters

    • featureName string required
    Responses

    tagsSchema

    Schema
    • version integer required

      The version of the schema used to model the tags.

    • tags object[]required

      A list of tags.

    • Array [
    • value string required

      Possible values: >= 2 characters and <= 50 characters

      The value of the tag

    • type string required

      Possible values: >= 2 characters and <= 50 characters

      Default value: simple

      The type of the tag

    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/login.html b/reference/api/unleash/login.html index fbb6fc856e..f669953d29 100644 --- a/reference/api/unleash/login.html +++ b/reference/api/unleash/login.html @@ -20,15 +20,15 @@ - - + +

    Log in

    POST /auth/simple/login

    Logs in the user and creates an active session

    Request

    Body

    required

    loginSchema

    • username string required

      The username trying to log in

    • password string required

      The password of the user trying to log in

    Responses

    userSchema

    Schema
    • id integer required

      The user id

    • isAPI boolean deprecated

      (Deprecated): Used internally to know which operations the user should be allowed to perform

    • name string nullable

      Name of the user

    • email string

      Email of the user

    • username string nullable

      A unique username for the user

    • imageUrl string

      URL used for the userprofile image

    • inviteLink string

      If the user is actively inviting other users, this is the link that can be shared with other users

    • loginAttempts integer

      How many unsuccessful attempts at logging in has the user made

    • emailSent boolean

      Is the welcome email sent to the user or not

    • rootRole integer

      Which root role this user is assigned

    • seenAt date-time nullable

      The last time this user logged in

    • createdAt date-time

      The user was created at this time

    • accountType string

      A user is either an actual User or a Service Account

    • permissions string[]

      Deprecated

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/maintenance.html b/reference/api/unleash/maintenance.html index 9c2003e432..c92c640b3b 100644 --- a/reference/api/unleash/maintenance.html +++ b/reference/api/unleash/maintenance.html @@ -20,15 +20,15 @@ - - + + - - + + \ No newline at end of file diff --git a/reference/api/unleash/mark-notifications-as-read.html b/reference/api/unleash/mark-notifications-as-read.html index ce8497607b..0f7c7ebd21 100644 --- a/reference/api/unleash/mark-notifications-as-read.html +++ b/reference/api/unleash/mark-notifications-as-read.html @@ -20,15 +20,15 @@ - - + +
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/metrics.html b/reference/api/unleash/metrics.html index 432c29706a..88cadec084 100644 --- a/reference/api/unleash/metrics.html +++ b/reference/api/unleash/metrics.html @@ -20,15 +20,15 @@ - - + +
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/notifications.html b/reference/api/unleash/notifications.html index f7c30e0319..0c55521b29 100644 --- a/reference/api/unleash/notifications.html +++ b/reference/api/unleash/notifications.html @@ -20,15 +20,15 @@ - - + + - - + + \ No newline at end of file diff --git a/reference/api/unleash/operational.html b/reference/api/unleash/operational.html index b475c63896..4ff0ca4f75 100644 --- a/reference/api/unleash/operational.html +++ b/reference/api/unleash/operational.html @@ -20,15 +20,15 @@ - - + + - - + + \ No newline at end of file diff --git a/reference/api/unleash/overwrite-environment-feature-variants.html b/reference/api/unleash/overwrite-environment-feature-variants.html index b1eb854a37..baf3311a03 100644 --- a/reference/api/unleash/overwrite-environment-feature-variants.html +++ b/reference/api/unleash/overwrite-environment-feature-variants.html @@ -20,15 +20,15 @@ - - + +

    Create (overwrite) variants for a feature in an environment

    PUT /api/admin/projects/:projectId/features/:featureName/environments/:environment/variants

    This overwrites the current variants for the feature toggle in the :featureName parameter for the :environment parameter.

    The backend will validate the input for the following invariants:

    • If there are variants, there needs to be at least one variant with weightType: variable
    • The sum of the weights of variants with weightType: fix must be strictly less than 1000 (< 1000)

    The backend will also distribute remaining weight up to 1000 after adding the variants with weightType: fix together amongst the variants of weightType: variable

    Request

    Path Parameters

    • projectId string required
    • featureName string required
    • environment string required

    Body

    arrayrequired

    variantsSchema

    • Array [
    • name string required

      The variants name. Is unique for this feature toggle

    • weight number required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is a number between 0 and 1000. See the section on variant weights for more information

    • weightType string

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000

    • stickiness string

      Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • overrides object[]

      Overrides assigning specific variants to specific users. The weighting system automatically assigns users to specific groups for you, but any overrides in this list will take precedence.

    • Array [
    • contextName string required

      The name of the context field used to determine overrides

    • values string[] required

      Which values that should be overriden

    • ]
    • ]
    Responses

    featureVariantsSchema

    Schema
    • version integer required

      The version of the feature variants schema.

    • variants object[]required

      All variants defined for a specific feature toggle.

    • Array [
    • name string required

      The variants name. Is unique for this feature toggle

    • weight number required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is a number between 0 and 1000. See the section on variant weights for more information

    • weightType string

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000

    • stickiness string

      Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • overrides object[]

      Overrides assigning specific variants to specific users. The weighting system automatically assigns users to specific groups for you, but any overrides in this list will take precedence.

    • Array [
    • contextName string required

      The name of the context field used to determine overrides

    • values string[] required

      Which values that should be overriden

    • ]
    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/overwrite-feature-variants-on-environments.html b/reference/api/unleash/overwrite-feature-variants-on-environments.html index 85f8fb82eb..667d9cd2cb 100644 --- a/reference/api/unleash/overwrite-feature-variants-on-environments.html +++ b/reference/api/unleash/overwrite-feature-variants-on-environments.html @@ -20,15 +20,15 @@ - - + +

    Create (overwrite) variants for a feature toggle in multiple environments

    PUT /api/admin/projects/:projectId/features/:featureName/variants-batch

    This overwrites the current variants for the feature toggle in the :featureName parameter for the :environment parameter.

    Request

    Path Parameters

    • projectId string required
    • featureName string required

    Body

    required

    pushVariantsSchema

    • variants object[]

      The variants to write to the provided environments

    • Array [
    • name string required

      The variants name. Is unique for this feature toggle

    • weight number required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is a number between 0 and 1000. See the section on variant weights for more information

    • weightType string

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000

    • stickiness string

      Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • overrides object[]

      Overrides assigning specific variants to specific users. The weighting system automatically assigns users to specific groups for you, but any overrides in this list will take precedence.

    • Array [
    • contextName string required

      The name of the context field used to determine overrides

    • values string[] required

      Which values that should be overriden

    • ]
    • ]
    • environments string[]

      The enviromnents to write the provided variants to

    Responses

    featureVariantsSchema

    Schema
    • version integer required

      The version of the feature variants schema.

    • variants object[]required

      All variants defined for a specific feature toggle.

    • Array [
    • name string required

      The variants name. Is unique for this feature toggle

    • weight number required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is a number between 0 and 1000. See the section on variant weights for more information

    • weightType string

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000

    • stickiness string

      Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • overrides object[]

      Overrides assigning specific variants to specific users. The weighting system automatically assigns users to specific groups for you, but any overrides in this list will take precedence.

    • Array [
    • contextName string required

      The name of the context field used to determine overrides

    • values string[] required

      Which values that should be overriden

    • ]
    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/overwrite-feature-variants.html b/reference/api/unleash/overwrite-feature-variants.html index 63a33ddbbf..ab8a436bd4 100644 --- a/reference/api/unleash/overwrite-feature-variants.html +++ b/reference/api/unleash/overwrite-feature-variants.html @@ -20,15 +20,15 @@ - - + +

    Create (overwrite) variants for a feature toggle in all environments

    PUT /api/admin/projects/:projectId/features/:featureName/variants

    This overwrites the current variants for the feature specified in the :featureName parameter in all environments.

    The backend will validate the input for the following invariants

    • If there are variants, there needs to be at least one variant with weightType: variable
    • The sum of the weights of variants with weightType: fix must be strictly less than 1000 (< 1000)

    The backend will also distribute remaining weight up to 1000 after adding the variants with weightType: fix together amongst the variants of weightType: variable

    Request

    Path Parameters

    • projectId string required
    • featureName string required

    Body

    arrayrequired

    variantsSchema

    • Array [
    • name string required

      The variants name. Is unique for this feature toggle

    • weight number required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is a number between 0 and 1000. See the section on variant weights for more information

    • weightType string

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000

    • stickiness string

      Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • overrides object[]

      Overrides assigning specific variants to specific users. The weighting system automatically assigns users to specific groups for you, but any overrides in this list will take precedence.

    • Array [
    • contextName string required

      The name of the context field used to determine overrides

    • values string[] required

      Which values that should be overriden

    • ]
    • ]
    Responses

    featureVariantsSchema

    Schema
    • version integer required

      The version of the feature variants schema.

    • variants object[]required

      All variants defined for a specific feature toggle.

    • Array [
    • name string required

      The variants name. Is unique for this feature toggle

    • weight number required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is a number between 0 and 1000. See the section on variant weights for more information

    • weightType string

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000

    • stickiness string

      Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • overrides object[]

      Overrides assigning specific variants to specific users. The weighting system automatically assigns users to specific groups for you, but any overrides in this list will take precedence.

    • Array [
    • contextName string required

      The name of the context field used to determine overrides

    • values string[] required

      Which values that should be overriden

    • ]
    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/patch-environments-feature-variants.html b/reference/api/unleash/patch-environments-feature-variants.html index 34cf3ae75e..7d85a92785 100644 --- a/reference/api/unleash/patch-environments-feature-variants.html +++ b/reference/api/unleash/patch-environments-feature-variants.html @@ -20,15 +20,15 @@ - - + +

    Patch a feature's variants in an environment

    PATCH /api/admin/projects/:projectId/features/:featureName/environments/:environment/variants

    Apply a list of patches to the features environments in the specified environment. The patch objects should conform to the JSON-patch format (RFC 6902).

    Request

    Path Parameters

    • projectId string required
    • featureName string required
    • environment string required

    Body

    arrayrequired

    patchesSchema

    • Array [
    • path string required

      The path to the property to operate on

    • op string required

      Possible values: [add, remove, replace, copy, move]

      The kind of operation to perform

    • from string

      The target to move or copy from, if performing one of those operations

    • value

      The value to add or replace, if performing one of those operations

    • ]
    Responses

    featureVariantsSchema

    Schema
    • version integer required

      The version of the feature variants schema.

    • variants object[]required

      All variants defined for a specific feature toggle.

    • Array [
    • name string required

      The variants name. Is unique for this feature toggle

    • weight number required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is a number between 0 and 1000. See the section on variant weights for more information

    • weightType string

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000

    • stickiness string

      Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • overrides object[]

      Overrides assigning specific variants to specific users. The weighting system automatically assigns users to specific groups for you, but any overrides in this list will take precedence.

    • Array [
    • contextName string required

      The name of the context field used to determine overrides

    • values string[] required

      Which values that should be overriden

    • ]
    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/patch-feature-strategy.html b/reference/api/unleash/patch-feature-strategy.html index 7ae692f86a..df3986d229 100644 --- a/reference/api/unleash/patch-feature-strategy.html +++ b/reference/api/unleash/patch-feature-strategy.html @@ -20,15 +20,15 @@ - - + +

    Change specific properties of a strategy

    PATCH /api/admin/projects/:projectId/features/:featureName/environments/:environment/strategies/:strategyId

    Change specific properties of a strategy configuration in a feature toggle.

    Request

    Path Parameters

    • projectId string required
    • featureName string required
    • environment string required
    • strategyId string required

    Body

    arrayrequired

    patchesSchema

    • Array [
    • path string required

      The path to the property to operate on

    • op string required

      Possible values: [add, remove, replace, copy, move]

      The kind of operation to perform

    • from string

      The target to move or copy from, if performing one of those operations

    • value

      The value to add or replace, if performing one of those operations

    • ]
    Responses

    featureStrategySchema

    Schema
    • id string

      A uuid for the feature strategy

    • name string required

      The name or type of strategy

    • title string nullable

      A descriptive title for the strategy

    • disabled boolean nullable

      A toggle to disable the strategy. defaults to false. Disabled strategies are not evaluated or returned to the SDKs

    • featureName string

      The name or feature the strategy is attached to

    • sortOrder number

      The order of the strategy in the list

    • segments number[]

      A list of segment ids attached to the strategy

    • constraints object[]

      A list of the constraints attached to the strategy. See https://docs.getunleash.io/reference/strategy-constraints

    • Array [
    • contextName string required

      The name of the context field that this constraint should apply to.

    • operator string required

      Possible values: [NOT_IN, IN, STR_ENDS_WITH, STR_STARTS_WITH, STR_CONTAINS, NUM_EQ, NUM_GT, NUM_GTE, NUM_LT, NUM_LTE, DATE_AFTER, DATE_BEFORE, SEMVER_EQ, SEMVER_GT, SEMVER_LT]

      The operator to use when evaluating this constraint. For more information about the various operators, refer to the strategy constraint operator documentation.

    • caseInsensitive boolean

      Default value: false

      Whether the operator should be case sensitive or not. Defaults to false (being case sensitive).

    • inverted boolean

      Default value: false

      Whether the result should be negated or not. If true, will turn a true result into a false result and vice versa.

    • values string[]

      The context values that should be used for constraint evaluation. Use this property instead of value for properties that accept multiple values.

    • value string

      The context value that should be used for constraint evaluation. Use this property instead of values for properties that only accept single values.

    • ]
    • variants object[]

      Strategy level variants

    • Array [
    • name string required

      The variant name. Must be unique for this feature toggle

    • weight integer required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is an integer between 0 and 1000. See the section on variant weights for more information

    • weightType string required

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000. Refer to the variant weight documentation.

    • stickiness string required

      The stickiness to use for distribution of this variant. Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • ]
    • parameters object

      A list of parameters for a strategy

    • property name* string
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/patch-feature-variants.html b/reference/api/unleash/patch-feature-variants.html index 9690511d25..492cf87718 100644 --- a/reference/api/unleash/patch-feature-variants.html +++ b/reference/api/unleash/patch-feature-variants.html @@ -20,15 +20,15 @@ - - + +

    Apply a patch to a feature's variants (in all environments).

    PATCH /api/admin/projects/:projectId/features/:featureName/variants

    Apply a list of patches patch to the specified feature's variants. The patch objects should conform to the JSON-patch format (RFC 6902).

    ⚠️ Warning: This method is not atomic. If something fails in the middle of applying the patch, you can be left with a half-applied patch. We recommend that you instead patch variants on a per-environment basis, which is an atomic operation.

    Request

    Path Parameters

    • projectId string required
    • featureName string required

    Body

    arrayrequired

    patchesSchema

    • Array [
    • path string required

      The path to the property to operate on

    • op string required

      Possible values: [add, remove, replace, copy, move]

      The kind of operation to perform

    • from string

      The target to move or copy from, if performing one of those operations

    • value

      The value to add or replace, if performing one of those operations

    • ]
    Responses

    featureVariantsSchema

    Schema
    • version integer required

      The version of the feature variants schema.

    • variants object[]required

      All variants defined for a specific feature toggle.

    • Array [
    • name string required

      The variants name. Is unique for this feature toggle

    • weight number required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is a number between 0 and 1000. See the section on variant weights for more information

    • weightType string

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000

    • stickiness string

      Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • overrides object[]

      Overrides assigning specific variants to specific users. The weighting system automatically assigns users to specific groups for you, but any overrides in this list will take precedence.

    • Array [
    • contextName string required

      The name of the context field used to determine overrides

    • values string[] required

      Which values that should be overriden

    • ]
    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/patch-feature.html b/reference/api/unleash/patch-feature.html index 9b86ca6d62..b2394f4d24 100644 --- a/reference/api/unleash/patch-feature.html +++ b/reference/api/unleash/patch-feature.html @@ -20,15 +20,15 @@ - - + +

    Modify a feature toggle

    PATCH /api/admin/projects/:projectId/features/:featureName

    Change specific properties of a feature toggle.

    Request

    Path Parameters

    • projectId string required
    • featureName string required

    Body

    arrayrequired

    patchesSchema

    • Array [
    • path string required

      The path to the property to operate on

    • op string required

      Possible values: [add, remove, replace, copy, move]

      The kind of operation to perform

    • from string

      The target to move or copy from, if performing one of those operations

    • value

      The value to add or replace, if performing one of those operations

    • ]
    Responses

    featureSchema

    Schema
    • name string required

      Unique feature name

    • type string

      Type of the toggle e.g. experiment, kill-switch, release, operational, permission

    • description string nullable

      Detailed description of the feature

    • archived boolean

      true if the feature is archived

    • project string

      Name of the project the feature belongs to

    • enabled boolean

      true if the feature is enabled, otherwise false.

    • stale boolean

      true if the feature is stale based on the age and feature type, otherwise false.

    • favorite boolean

      true if the feature was favorited, otherwise false.

    • impressionData boolean

      true if the impression data collection is enabled for the feature, otherwise false.

    • createdAt date-time nullable

      The date the feature was created

    • archivedAt date-time nullable

      The date the feature was archived

    • lastSeenAt date-time nullable deprecated

      The date when metrics where last collected for the feature. This field is deprecated, use the one in featureEnvironmentSchema

    • environments object[]

      The list of environments where the feature can be used

    • Array [
    • name string required

      The name of the environment

    • featureName string

      The name of the feature

    • environment string

      The name of the environment

    • type string

      The type of the environment

    • enabled boolean required

      true if the feature is enabled for the environment, otherwise false.

    • sortOrder number

      The sort order of the feature environment in the feature environments list

    • variantCount number

      The number of defined variants

    • strategies object[]

      A list of activation strategies for the feature environment

    • Array [
    • id string

      A uuid for the feature strategy

    • name string required

      The name or type of strategy

    • title string nullable

      A descriptive title for the strategy

    • disabled boolean nullable

      A toggle to disable the strategy. defaults to false. Disabled strategies are not evaluated or returned to the SDKs

    • featureName string

      The name or feature the strategy is attached to

    • sortOrder number

      The order of the strategy in the list

    • segments number[]

      A list of segment ids attached to the strategy

    • constraints object[]

      A list of the constraints attached to the strategy. See https://docs.getunleash.io/reference/strategy-constraints

    • Array [
    • contextName string required

      The name of the context field that this constraint should apply to.

    • operator string required

      Possible values: [NOT_IN, IN, STR_ENDS_WITH, STR_STARTS_WITH, STR_CONTAINS, NUM_EQ, NUM_GT, NUM_GTE, NUM_LT, NUM_LTE, DATE_AFTER, DATE_BEFORE, SEMVER_EQ, SEMVER_GT, SEMVER_LT]

      The operator to use when evaluating this constraint. For more information about the various operators, refer to the strategy constraint operator documentation.

    • caseInsensitive boolean

      Default value: false

      Whether the operator should be case sensitive or not. Defaults to false (being case sensitive).

    • inverted boolean

      Default value: false

      Whether the result should be negated or not. If true, will turn a true result into a false result and vice versa.

    • values string[]

      The context values that should be used for constraint evaluation. Use this property instead of value for properties that accept multiple values.

    • value string

      The context value that should be used for constraint evaluation. Use this property instead of values for properties that only accept single values.

    • ]
    • variants object[]

      Strategy level variants

    • Array [
    • name string required

      The variant name. Must be unique for this feature toggle

    • weight integer required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is an integer between 0 and 1000. See the section on variant weights for more information

    • weightType string required

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000. Refer to the variant weight documentation.

    • stickiness string required

      The stickiness to use for distribution of this variant. Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • ]
    • parameters object

      A list of parameters for a strategy

    • property name* string
    • ]
    • variants object[]

      A list of variants for the feature environment

    • Array [
    • name string required

      The variants name. Is unique for this feature toggle

    • weight number required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is a number between 0 and 1000. See the section on variant weights for more information

    • weightType string

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000

    • stickiness string

      Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • overrides object[]

      Overrides assigning specific variants to specific users. The weighting system automatically assigns users to specific groups for you, but any overrides in this list will take precedence.

    • Array [
    • contextName string required

      The name of the context field used to determine overrides

    • values string[] required

      Which values that should be overriden

    • ]
    • ]
    • lastSeenAt date-time nullable

      The date when metrics where last collected for the feature environment

    • hasStrategies boolean

      Whether the feature has any strategies defined.

    • hasEnabledStrategies boolean

      Whether the feature has any enabled strategies defined.

    • ]
    • variants object[]deprecated

      The list of feature variants

    • Array [
    • name string required

      The variants name. Is unique for this feature toggle

    • weight number required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is a number between 0 and 1000. See the section on variant weights for more information

    • weightType string

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000

    • stickiness string

      Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • overrides object[]

      Overrides assigning specific variants to specific users. The weighting system automatically assigns users to specific groups for you, but any overrides in this list will take precedence.

    • Array [
    • contextName string required

      The name of the context field used to determine overrides

    • values string[] required

      Which values that should be overriden

    • ]
    • ]
    • strategies object[] deprecated

      This is a legacy field that will be deprecated

    • tags object[]nullable

      The list of feature tags

    • Array [
    • value string required

      Possible values: >= 2 characters and <= 50 characters

      The value of the tag

    • type string required

      Possible values: >= 2 characters and <= 50 characters

      Default value: simple

      The type of the tag

    • ]
    • children string[]

      The list of child feature names. This is an experimental field and may change.

    • dependencies object[]

      The list of parent dependencies. This is an experimental field and may change.

    • Array [
    • feature string required

      The name of the parent feature

    • enabled boolean

      Whether the parent feature is enabled or not

    • variants string[]

      The list of variants the parent feature should resolve to. Only valid when feature is enabled.

    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/personal-access-tokens.html b/reference/api/unleash/personal-access-tokens.html index aa351a9cf9..35ebf006c9 100644 --- a/reference/api/unleash/personal-access-tokens.html +++ b/reference/api/unleash/personal-access-tokens.html @@ -20,15 +20,15 @@ - - + +
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/playground.html b/reference/api/unleash/playground.html index a29339dcb7..5410fbd32e 100644 --- a/reference/api/unleash/playground.html +++ b/reference/api/unleash/playground.html @@ -20,15 +20,15 @@ - - + +
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/projects.html b/reference/api/unleash/projects.html index ad5534c23f..f839c2b957 100644 --- a/reference/api/unleash/projects.html +++ b/reference/api/unleash/projects.html @@ -20,15 +20,15 @@ - - + +

    Projects

    Create, update, and delete Unleash projects.

    - - + + \ No newline at end of file diff --git a/reference/api/unleash/provide-feedback.html b/reference/api/unleash/provide-feedback.html index 18ad239965..ad9a5bc54d 100644 --- a/reference/api/unleash/provide-feedback.html +++ b/reference/api/unleash/provide-feedback.html @@ -20,15 +20,15 @@ - - + +

    Submit user feedback

    POST /feedback

    Allows users to submit feedback.

    Request

    Body

    required

    provideFeedbackSchema

    • category string required

      The category of the feedback.

    • userType string nullable

      The type of user providing the feedback.

    • difficultyScore number nullable

      A score indicating the difficulty experienced by the user.

    • positive string nullable

      This field is for users to mention what they liked.

    • areasForImprovement string nullable

      Details aspects of the service or product that could benefit from enhancements or modifications. Aids in pinpointing areas needing attention for improvement.

    Responses

    feedbackSchema

    Schema
    • id number required

      The unique identifier of the feedback.

    • createdAt date-time required

      The date and time when the feedback was provided.

    • category string required

      The category of the feedback.

    • userType string nullable required

      The type of user providing the feedback.

    • difficultyScore number nullable required

      A score indicating the difficulty experienced by the user.

    • positive string nullable required

      This field is for users to mention what they liked.

    • areasForImprovement string nullable required

      Details aspects of the service or product that could benefit from enhancements or modifications. Aids in pinpointing areas needing attention for improvement.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/public-signup-tokens.html b/reference/api/unleash/public-signup-tokens.html index 7946793507..a56fcda924 100644 --- a/reference/api/unleash/public-signup-tokens.html +++ b/reference/api/unleash/public-signup-tokens.html @@ -20,15 +20,15 @@ - - + +
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/reactivate-strategy.html b/reference/api/unleash/reactivate-strategy.html index e6c2281a19..a5fe47e849 100644 --- a/reference/api/unleash/reactivate-strategy.html +++ b/reference/api/unleash/reactivate-strategy.html @@ -20,15 +20,15 @@ - - + +

    Reactivate a strategy

    POST /api/admin/strategies/:strategyName/reactivate

    Marks the specified strategy as not deprecated. If the strategy wasn't already deprecated, nothing changes.

    Request

    Path Parameters

    • strategyName string required
    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/read-license.html b/reference/api/unleash/read-license.html index 3d438c2f6b..ebe9495e21 100644 --- a/reference/api/unleash/read-license.html +++ b/reference/api/unleash/read-license.html @@ -20,15 +20,15 @@ - - + +

    Reads the Unleash license.

    GET /api/admin/license

    Reads the Unleash license. Only available for self-hosted Enterprise customers.

    Request

    Responses

    licenseReadSchema

    Schema
    • token string required

      The actual license token.

    • customer string

      Name of the customer that owns the license. This is the name of the company that purchased the license.

    • plan string

      Name of plan that the license is for.

    • seats number

      Number of seats in the license.

    • expireAt date-time

      Date when the license expires.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/register-client-application.html b/reference/api/unleash/register-client-application.html index c3d9d4f182..b06f79be39 100644 --- a/reference/api/unleash/register-client-application.html +++ b/reference/api/unleash/register-client-application.html @@ -20,15 +20,15 @@ - - + +

    Register a client SDK

    POST /api/client/register

    Register a client SDK with Unleash. SDKs call this endpoint on startup to tell Unleash about their existence. Used to track custom strategies in use as well as SDK versions.

    Request

    Body

    required

    clientApplicationSchema

    • appName string required

      An identifier for the app that uses the sdk, should be static across SDK restarts

    • instanceId string

      A unique identifier identifying the instance of the application running the SDK. Often changes based on execution environment. For instance: two pods in Kubernetes will have two different instanceIds

    • sdkVersion string

      An SDK version identifier. Usually formatted as "unleash-client-:"

    • environment string deprecated

      The SDK's configured 'environment' property. Deprecated. This property does not control which Unleash environment the SDK gets toggles for. To control Unleash environments, use the SDKs API key.

    • interval number required

      How often (in seconds) does the client refresh its toggles

    • started objectrequired

      Either an RFC-3339 timestamp or a unix timestamp in seconds

      oneOf
    • string date-time
    • strategies string[] required

      Which strategies the SDKs runtime knows about

    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/register-client-metrics.html b/reference/api/unleash/register-client-metrics.html index 59ab4070d3..e1552c5192 100644 --- a/reference/api/unleash/register-client-metrics.html +++ b/reference/api/unleash/register-client-metrics.html @@ -20,15 +20,15 @@ - - + +

    Register client usage metrics

    POST /api/client/metrics

    Registers usage metrics. Stores information about how many times each toggle was evaluated to enabled and disabled within a time frame. If provided, this operation will also store data on how many times each feature toggle's variants were displayed to the end user.

    Request

    Body

    required

    clientMetricsSchema

    • appName string required

      The name of the application that is evaluating toggles

    • instanceId string

      A (somewhat) unique identifier for the application

    • environment string

      Which environment the application is running in

    • bucket objectrequired

      Holds all metrics gathered over a window of time. Typically 1 hour wide

    • start objectrequired

      The start of the time window these metrics are valid for. The window is usually 1 hour wide

      oneOf
    • string date-time

      An RFC-3339-compliant timestamp.

    • stop objectrequired

      The end of the time window these metrics are valid for. The window is 1 hour wide

      oneOf
    • string date-time

      An RFC-3339-compliant timestamp.

    • toggles objectrequired

      an object containing feature names with yes/no plus variant usage

    • property name* object
    • yes number

      How many times the toggle evaluated to true

    • no integer

      How many times the toggle evaluated to false

    • variants object

      An object describing how many times each variant was returned. Variant names are used as properties, and the number of times they were exposed is the corresponding value (i.e. { [variantName]: number }).

    • property name* integer
    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/register-frontend-client.html b/reference/api/unleash/register-frontend-client.html index 238b9edf9d..d8380afc0d 100644 --- a/reference/api/unleash/register-frontend-client.html +++ b/reference/api/unleash/register-frontend-client.html @@ -20,15 +20,15 @@ - - + +

    Register a client SDK

    POST /api/frontend/client/register

    This is for future use. Currently Frontend client registration is not supported. Returning 200 for clients that expect this status code. If the Frontend API is disabled 404 is returned.

    Request

    Body

    required

    proxyClientSchema

    • appName string required

      Name of the application using Unleash

    • instanceId string

      Instance id for this application (typically hostname, podId or similar)

    • sdkVersion string

      Optional field that describes the sdk version (name:version)

    • environment string deprecated

      deprecated

    • interval number required

      At which interval, in milliseconds, will this client be expected to send metrics

    • started objectrequired

      When this client started. Should be reported as ISO8601 time.

      oneOf
    • string date-time
    • strategies string[] required

      List of strategies implemented by this application

    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/register-frontend-metrics.html b/reference/api/unleash/register-frontend-metrics.html index 1b77af9b59..c9b6bdd1f2 100644 --- a/reference/api/unleash/register-frontend-metrics.html +++ b/reference/api/unleash/register-frontend-metrics.html @@ -20,15 +20,15 @@ - - + +

    Register client usage metrics

    POST /api/frontend/client/metrics

    Registers usage metrics. Stores information about how many times each toggle was evaluated to enabled and disabled within a time frame. If provided, this operation will also store data on how many times each feature toggle's variants were displayed to the end user. If the Frontend API is disabled 404 is returned.

    Request

    Body

    required

    clientMetricsSchema

    • appName string required

      The name of the application that is evaluating toggles

    • instanceId string

      A (somewhat) unique identifier for the application

    • environment string

      Which environment the application is running in

    • bucket objectrequired

      Holds all metrics gathered over a window of time. Typically 1 hour wide

    • start objectrequired

      The start of the time window these metrics are valid for. The window is usually 1 hour wide

      oneOf
    • string date-time

      An RFC-3339-compliant timestamp.

    • stop objectrequired

      The end of the time window these metrics are valid for. The window is 1 hour wide

      oneOf
    • string date-time

      An RFC-3339-compliant timestamp.

    • toggles objectrequired

      an object containing feature names with yes/no plus variant usage

    • property name* object
    • yes number

      How many times the toggle evaluated to true

    • no integer

      How many times the toggle evaluated to false

    • variants object

      An object describing how many times each variant was returned. Variant names are used as properties, and the number of times they were exposed is the corresponding value (i.e. { [variantName]: number }).

    • property name* integer
    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/remove-environment-from-project.html b/reference/api/unleash/remove-environment-from-project.html index 5eb4c2ba0e..18d197caf3 100644 --- a/reference/api/unleash/remove-environment-from-project.html +++ b/reference/api/unleash/remove-environment-from-project.html @@ -20,15 +20,15 @@ - - + +

    Remove an environment from a project.

    DELETE /api/admin/projects/:projectId/environments/:environment

    This endpoint removes the specified environment from the project.

    Request

    Path Parameters

    • projectId string required
    • environment string required
    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/remove-environment.html b/reference/api/unleash/remove-environment.html index ca6d075e81..28735232cf 100644 --- a/reference/api/unleash/remove-environment.html +++ b/reference/api/unleash/remove-environment.html @@ -20,15 +20,15 @@ - - + +

    Deletes an environment by name

    DELETE /api/admin/environments/:name

    Given an existing environment by name, this endpoint will attempt to delete it

    Request

    Path Parameters

    • name string required
    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/remove-favorite-feature.html b/reference/api/unleash/remove-favorite-feature.html index d8472c4e63..23b8e95295 100644 --- a/reference/api/unleash/remove-favorite-feature.html +++ b/reference/api/unleash/remove-favorite-feature.html @@ -20,15 +20,15 @@ - - + +

    Remove feature from favorites

    DELETE /api/admin/projects/:projectId/features/:featureName/favorites

    This endpoint removes the feature in the url from favorites

    Request

    Path Parameters

    • projectId string required
    • featureName string required
    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/remove-favorite-project.html b/reference/api/unleash/remove-favorite-project.html index 8f52f25a51..e0c60563d6 100644 --- a/reference/api/unleash/remove-favorite-project.html +++ b/reference/api/unleash/remove-favorite-project.html @@ -20,15 +20,15 @@ - - + +

    Remove project from favorites

    DELETE /api/admin/projects/:projectId/favorites

    This endpoint removes the project in the url from favorites

    Request

    Path Parameters

    • projectId string required
    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/remove-group-access.html b/reference/api/unleash/remove-group-access.html index aa9cda6ade..baa71fbf9d 100644 --- a/reference/api/unleash/remove-group-access.html +++ b/reference/api/unleash/remove-group-access.html @@ -20,15 +20,15 @@ - - + +

    Remove project access for a group

    DELETE /api/admin/projects/:projectId/groups/:groupId/roles

    Removes project access for a group by removing all of its roles for the project.

    Request

    Path Parameters

    • projectId string required
    • groupId string required
    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/remove-role-for-user.html b/reference/api/unleash/remove-role-for-user.html index cd933867ce..03d79ea7f9 100644 --- a/reference/api/unleash/remove-role-for-user.html +++ b/reference/api/unleash/remove-role-for-user.html @@ -20,15 +20,15 @@ - - + +

    Removes role from user

    DELETE /api/admin/projects/:projectId/users/:userId/roles/:roleId
    deprecated

    This endpoint has been deprecated and may be removed in future versions of the API.

    Remove the specified project role from the specified user. This endpoint is deprecated. Use /:projectId/users/:userId/roles instead.

    Request

    Path Parameters

    • projectId string required
    • userId string required
    • roleId string required
    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/remove-role-from-group.html b/reference/api/unleash/remove-role-from-group.html index 16854b1372..a52fd0dd5d 100644 --- a/reference/api/unleash/remove-role-from-group.html +++ b/reference/api/unleash/remove-role-from-group.html @@ -20,15 +20,15 @@ - - + +

    Remove project group role

    DELETE /api/admin/projects/:projectId/groups/:groupId/roles/:roleId
    deprecated

    This endpoint has been deprecated and may be removed in future versions of the API.

    Removes a project role from a group. This endpoint is deprecated. Use /:projectId/groups/:groupId/roles instead.

    Request

    Path Parameters

    • projectId string required
    • groupId string required
    • roleId string required
    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/remove-segment.html b/reference/api/unleash/remove-segment.html index aa2220502e..7f04b52d09 100644 --- a/reference/api/unleash/remove-segment.html +++ b/reference/api/unleash/remove-segment.html @@ -20,15 +20,15 @@ - - + +

    Deletes a segment by id

    DELETE /api/admin/segments/:id

    Deletes a segment by its id, if not found returns a 409 error

    Request

    Path Parameters

    • id string required
    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/remove-strategy.html b/reference/api/unleash/remove-strategy.html index 53bfcbeb55..aaf1f20246 100644 --- a/reference/api/unleash/remove-strategy.html +++ b/reference/api/unleash/remove-strategy.html @@ -20,15 +20,15 @@ - - + +

    Delete a strategy

    DELETE /api/admin/strategies/:name

    Deletes the specified strategy definition

    Request

    Path Parameters

    • name string required
    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/remove-tag.html b/reference/api/unleash/remove-tag.html index 2129f85c80..7fbaed5717 100644 --- a/reference/api/unleash/remove-tag.html +++ b/reference/api/unleash/remove-tag.html @@ -20,15 +20,15 @@ - - + +

    Removes a tag from a feature.

    DELETE /api/admin/features/:featureName/tags/:type/:value

    Removes a tag from a feature. If the feature exists but the tag does not, it returns a successful response.

    Request

    Path Parameters

    • featureName string required
    • type string required
    • value string required
    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/remove-user-access.html b/reference/api/unleash/remove-user-access.html index 537ca7c0ae..e65e39b20a 100644 --- a/reference/api/unleash/remove-user-access.html +++ b/reference/api/unleash/remove-user-access.html @@ -20,15 +20,15 @@ - - + +

    Remove project access for a user

    DELETE /api/admin/projects/:projectId/users/:userId/roles

    Removes project access for a user by removing all of its roles for the project.

    Request

    Path Parameters

    • projectId string required
    • userId string required
    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/reset-user-password.html b/reference/api/unleash/reset-user-password.html index a26d2177f9..cec50b3001 100644 --- a/reference/api/unleash/reset-user-password.html +++ b/reference/api/unleash/reset-user-password.html @@ -20,15 +20,15 @@ - - + +

    Reset user password

    POST /api/admin/user-admin/reset-password

    Reset user password as an admin

    Request

    Body

    required

    idSchema

    • id string required

      User email

    Responses

    resetPasswordSchema

    Schema
    • resetPasswordUrl uri required

      A URL pointing to a location where the user can reset their password

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/revive-feature.html b/reference/api/unleash/revive-feature.html index f51872df44..891a805203 100644 --- a/reference/api/unleash/revive-feature.html +++ b/reference/api/unleash/revive-feature.html @@ -20,15 +20,15 @@ - - + +

    Revives a feature

    POST /api/admin/archive/revive/:featureName

    This endpoint revives the specified feature from archive.

    Request

    Path Parameters

    • featureName string required
    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/revive-features.html b/reference/api/unleash/revive-features.html index 5badf25bac..a628c9f21c 100644 --- a/reference/api/unleash/revive-features.html +++ b/reference/api/unleash/revive-features.html @@ -20,15 +20,15 @@ - - + +

    Revives a list of features

    POST /api/admin/projects/:projectId/revive

    This endpoint revives the specified features.

    Request

    Path Parameters

    • projectId string required

    Body

    required

    batchFeaturesSchema

    • features string[] required

      List of feature toggle names

    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/search-events.html b/reference/api/unleash/search-events.html index 14a9ec9964..d10c6d4d1b 100644 --- a/reference/api/unleash/search-events.html +++ b/reference/api/unleash/search-events.html @@ -20,15 +20,15 @@ - - + +

    Search for events

    POST /api/admin/events/search

    Allows searching for events matching the search criteria in the request body

    Request

    Body

    required

    searchEventsSchema

    • type string

      Possible values: [application-created, feature-created, feature-deleted, feature-updated, feature-metadata-updated, feature-variants-updated, feature-environment-variants-updated, feature-project-change, feature-archived, feature-revived, feature-import, feature-tagged, feature-tag-import, feature-strategy-update, feature-strategy-add, feature-strategy-remove, feature-type-updated, strategy-order-changed, drop-feature-tags, feature-untagged, feature-stale-on, feature-stale-off, drop-features, feature-environment-enabled, feature-environment-disabled, strategy-created, strategy-deleted, strategy-deprecated, strategy-reactivated, strategy-updated, strategy-import, drop-strategies, context-field-created, context-field-updated, context-field-deleted, project-access-added, project-access-user-roles-updated, project-access-group-roles-updated, project-access-user-roles-deleted, project-access-group-roles-deleted, project-access-updated, project-created, project-updated, project-deleted, project-import, project-user-added, project-user-removed, project-user-role-changed, project-group-role-changed, project-group-added, project-group-removed, role-created, role-updated, role-deleted, drop-projects, tag-created, tag-deleted, tag-import, drop-tags, tag-type-created, tag-type-deleted, tag-type-updated, tag-type-import, drop-tag-types, addon-config-created, addon-config-updated, addon-config-deleted, db-pool-update, user-created, user-updated, user-deleted, drop-environments, environment-import, environment-created, environment-updated, environment-deleted, segment-created, segment-updated, segment-deleted, group-created, group-updated, group-deleted, group-user-added, group-user-removed, setting-created, setting-updated, setting-deleted, client-metrics, client-register, pat-created, pat-deleted, public-signup-token-created, public-signup-token-user-added, public-signup-token-updated, change-request-created, change-request-discarded, change-added, change-discarded, change-edited, change-request-rejected, change-request-approved, change-request-approval-added, change-request-cancelled, change-request-sent-to-review, scheduled-change-request-executed, change-request-applied, change-request-scheduled, change-request-scheduled-application-success, change-request-scheduled-application-failure, change-request-configuration-updated, api-token-created, api-token-updated, api-token-deleted, feature-favorited, feature-unfavorited, project-favorited, project-unfavorited, features-exported, features-imported, service-account-created, service-account-deleted, service-account-updated, feature-potentially-stale-on, feature-dependency-added, feature-dependency-removed, feature-dependencies-removed, banner-created, banner-updated, banner-deleted, project-environment-added, project-environment-removed, default-strategy-updated, segment-import, incoming-webhook-created, incoming-webhook-updated, incoming-webhook-deleted, incoming-webhook-token-created, incoming-webhook-token-updated, incoming-webhook-token-deleted]

      Find events by event type (case-sensitive).

    • project string

      Find events by project ID (case-sensitive).

    • feature string

      Find events by feature toggle name (case-sensitive).

    • query string

      Find events by a free-text search query. The query will be matched against the event type, the username or email that created the event (if any), and the event data payload (if any).

    • limit integer

      Possible values: >= 1 and <= 100

      Default value: 100

      The maximum amount of events to return in the search result

    • offset integer

      Which event id to start listing from

    Responses

    eventsSchema

    Schema
    • version integer required

      Possible values: >= 1, [1]

      The api version of this response. A natural increasing number. Only increases if format changes

    • events object[]required

      The list of events

    • Array [
    • id integer required

      Possible values: >= 1

      The ID of the event. An increasing natural number.

    • createdAt date-time required

      The time the event happened as a RFC 3339-conformant timestamp.

    • type string required

      Possible values: [application-created, feature-created, feature-deleted, feature-updated, feature-metadata-updated, feature-variants-updated, feature-environment-variants-updated, feature-project-change, feature-archived, feature-revived, feature-import, feature-tagged, feature-tag-import, feature-strategy-update, feature-strategy-add, feature-strategy-remove, feature-type-updated, strategy-order-changed, drop-feature-tags, feature-untagged, feature-stale-on, feature-stale-off, drop-features, feature-environment-enabled, feature-environment-disabled, strategy-created, strategy-deleted, strategy-deprecated, strategy-reactivated, strategy-updated, strategy-import, drop-strategies, context-field-created, context-field-updated, context-field-deleted, project-access-added, project-access-user-roles-updated, project-access-group-roles-updated, project-access-user-roles-deleted, project-access-group-roles-deleted, project-access-updated, project-created, project-updated, project-deleted, project-import, project-user-added, project-user-removed, project-user-role-changed, project-group-role-changed, project-group-added, project-group-removed, role-created, role-updated, role-deleted, drop-projects, tag-created, tag-deleted, tag-import, drop-tags, tag-type-created, tag-type-deleted, tag-type-updated, tag-type-import, drop-tag-types, addon-config-created, addon-config-updated, addon-config-deleted, db-pool-update, user-created, user-updated, user-deleted, drop-environments, environment-import, environment-created, environment-updated, environment-deleted, segment-created, segment-updated, segment-deleted, group-created, group-updated, group-deleted, group-user-added, group-user-removed, setting-created, setting-updated, setting-deleted, client-metrics, client-register, pat-created, pat-deleted, public-signup-token-created, public-signup-token-user-added, public-signup-token-updated, change-request-created, change-request-discarded, change-added, change-discarded, change-edited, change-request-rejected, change-request-approved, change-request-approval-added, change-request-cancelled, change-request-sent-to-review, scheduled-change-request-executed, change-request-applied, change-request-scheduled, change-request-scheduled-application-success, change-request-scheduled-application-failure, change-request-configuration-updated, api-token-created, api-token-updated, api-token-deleted, feature-favorited, feature-unfavorited, project-favorited, project-unfavorited, features-exported, features-imported, service-account-created, service-account-deleted, service-account-updated, feature-potentially-stale-on, feature-dependency-added, feature-dependency-removed, feature-dependencies-removed, banner-created, banner-updated, banner-deleted, project-environment-added, project-environment-removed, default-strategy-updated, segment-import, incoming-webhook-created, incoming-webhook-updated, incoming-webhook-deleted, incoming-webhook-token-created, incoming-webhook-token-updated, incoming-webhook-token-deleted]

      What type of event this is

    • createdBy string required

      Which user created this event

    • createdByUserId number nullable

      The is of the user that created this event

    • environment string nullable

      The feature toggle environment the event relates to, if applicable.

    • project string nullable

      The project the event relates to, if applicable.

    • featureName string nullable

      The name of the feature toggle the event relates to, if applicable.

    • data object nullable

      Extra associated data related to the event, such as feature toggle state, segment configuration, etc., if applicable.

    • preData object nullable

      Data relating to the previous state of the event's subject.

    • tags object[]nullable

      Any tags related to the event, if applicable.

    • Array [
    • value string required

      Possible values: >= 2 characters and <= 50 characters

      The value of the tag

    • type string required

      Possible values: >= 2 characters and <= 50 characters

      Default value: simple

      The type of the tag

    • ]
    • ]
    • totalEvents integer

      The total count of events

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/search-features.html b/reference/api/unleash/search-features.html index ce5cb97705..192774d71f 100644 --- a/reference/api/unleash/search-features.html +++ b/reference/api/unleash/search-features.html @@ -20,15 +20,15 @@ - - + +

    Search and filter features

    GET /api/admin/search/features

    Search and filter by selected fields.

    Request

    Query Parameters

    • query string

      The search query for the feature name or tag

    • project string

      Possible values: Value must match regular expression ^(IS|IS_NOT|IS_ANY_OF|IS_NONE_OF):(.*?)(,([a-zA-Z0-9_]+))*$

      Id of the project where search and filter is performed. The project id can be specified with an operator. The supported operators are IS, IS_NOT, IS_ANY_OF, IS_NONE_OF.

    • state string

      Possible values: Value must match regular expression ^(IS|IS_NOT|IS_ANY_OF|IS_NONE_OF):(.*?)(,([a-zA-Z0-9_]+))*$

      The state of the feature active/stale. The state can be specified with an operator. The supported operators are IS, IS_NOT, IS_ANY_OF, IS_NONE_OF.

    • type string[]

      The list of feature types to filter by

    • tag string

      Possible values: Value must match regular expression ^(INCLUDE|DO_NOT_INCLUDE|INCLUDE_ALL_OF|INCLUDE_ANY_OF|EXCLUDE_IF_ANY_OF|EXCLUDE_ALL):(?:\s*[^,:]+:[^,:]+\s*)(?:,\s*[^,:]+:[^,:]+\s*)*$

      The list of feature tags to filter by. Feature tag has to specify a type and a value joined with a colon.

    • segment string

      Possible values: Value must match regular expression ^(INCLUDE|DO_NOT_INCLUDE|INCLUDE_ALL_OF|INCLUDE_ANY_OF|EXCLUDE_IF_ANY_OF|EXCLUDE_ALL):(.*?)(,([a-zA-Z0-9_]+))*$

      The list of segments with operators to filter by. The segment valid operators are INCLUDE, DO_NOT_INCLUDE, INCLUDE_ALL_OF, INCLUDE_ANY_OF, EXCLUDE_IF_ANY_OF, EXCLUDE_ALL.

    • status string[]

      The list of feature environment status to filter by. Feature environment has to specify a name and a status joined with a colon.

    • offset string

      The number of features to skip when returning a page. By default it is set to 0.

    • limit string

      The number of feature environments to return in a page. By default it is set to 50.

    • sortBy string

      The field to sort the results by. By default it is set to "createdAt".

    • sortOrder string

      The sort order for the sortBy. By default it is det to "asc".

    • favoritesFirst string

      The flag to indicate if the favorite features should be returned first. By default it is set to false.

    • createdAt string

      Possible values: Value must match regular expression ^(IS_BEFORE|IS_ON_OR_AFTER):\d{4}-\d{2}-\d{2}$

      The date the feature was created. The date can be specified with an operator. The supported operators are IS_BEFORE, IS_ON_OR_AFTER.

    Responses

    searchFeaturesSchema

    Schema
    • features object[]required

      The full list of features in this project matching search and filter criteria.

    • Array [
    • name string required

      Unique feature name

    • type string

      Type of the toggle e.g. experiment, kill-switch, release, operational, permission

    • description string nullable

      Detailed description of the feature

    • archived boolean

      true if the feature is archived

    • project string

      Name of the project the feature belongs to

    • enabled boolean

      true if the feature is enabled, otherwise false.

    • stale boolean

      true if the feature is stale based on the age and feature type, otherwise false.

    • favorite boolean

      true if the feature was favorited, otherwise false.

    • impressionData boolean

      true if the impression data collection is enabled for the feature, otherwise false.

    • createdAt date-time nullable

      The date the feature was created

    • archivedAt date-time nullable

      The date the feature was archived

    • lastSeenAt date-time nullable deprecated

      The date when metrics where last collected for the feature. This field is deprecated, use the one in featureEnvironmentSchema

    • environments object[]

      The list of environments where the feature can be used

    • Array [
    • name string required

      The name of the environment

    • featureName string

      The name of the feature

    • environment string

      The name of the environment

    • type string

      The type of the environment

    • enabled boolean required

      true if the feature is enabled for the environment, otherwise false.

    • sortOrder number

      The sort order of the feature environment in the feature environments list

    • variantCount number

      The number of defined variants

    • strategies object[]

      A list of activation strategies for the feature environment

    • Array [
    • id string

      A uuid for the feature strategy

    • name string required

      The name or type of strategy

    • title string nullable

      A descriptive title for the strategy

    • disabled boolean nullable

      A toggle to disable the strategy. defaults to false. Disabled strategies are not evaluated or returned to the SDKs

    • featureName string

      The name or feature the strategy is attached to

    • sortOrder number

      The order of the strategy in the list

    • segments number[]

      A list of segment ids attached to the strategy

    • constraints object[]

      A list of the constraints attached to the strategy. See https://docs.getunleash.io/reference/strategy-constraints

    • Array [
    • contextName string required

      The name of the context field that this constraint should apply to.

    • operator string required

      Possible values: [NOT_IN, IN, STR_ENDS_WITH, STR_STARTS_WITH, STR_CONTAINS, NUM_EQ, NUM_GT, NUM_GTE, NUM_LT, NUM_LTE, DATE_AFTER, DATE_BEFORE, SEMVER_EQ, SEMVER_GT, SEMVER_LT]

      The operator to use when evaluating this constraint. For more information about the various operators, refer to the strategy constraint operator documentation.

    • caseInsensitive boolean

      Default value: false

      Whether the operator should be case sensitive or not. Defaults to false (being case sensitive).

    • inverted boolean

      Default value: false

      Whether the result should be negated or not. If true, will turn a true result into a false result and vice versa.

    • values string[]

      The context values that should be used for constraint evaluation. Use this property instead of value for properties that accept multiple values.

    • value string

      The context value that should be used for constraint evaluation. Use this property instead of values for properties that only accept single values.

    • ]
    • variants object[]

      Strategy level variants

    • Array [
    • name string required

      The variant name. Must be unique for this feature toggle

    • weight integer required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is an integer between 0 and 1000. See the section on variant weights for more information

    • weightType string required

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000. Refer to the variant weight documentation.

    • stickiness string required

      The stickiness to use for distribution of this variant. Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • ]
    • parameters object

      A list of parameters for a strategy

    • property name* string
    • ]
    • variants object[]

      A list of variants for the feature environment

    • Array [
    • name string required

      The variants name. Is unique for this feature toggle

    • weight number required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is a number between 0 and 1000. See the section on variant weights for more information

    • weightType string

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000

    • stickiness string

      Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • overrides object[]

      Overrides assigning specific variants to specific users. The weighting system automatically assigns users to specific groups for you, but any overrides in this list will take precedence.

    • Array [
    • contextName string required

      The name of the context field used to determine overrides

    • values string[] required

      Which values that should be overriden

    • ]
    • ]
    • lastSeenAt date-time nullable

      The date when metrics where last collected for the feature environment

    • hasStrategies boolean

      Whether the feature has any strategies defined.

    • hasEnabledStrategies boolean

      Whether the feature has any enabled strategies defined.

    • ]
    • segments string[]

      The list of segments the feature is enabled for.

    • variants object[]deprecated

      The list of feature variants

    • Array [
    • name string required

      The variants name. Is unique for this feature toggle

    • weight number required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is a number between 0 and 1000. See the section on variant weights for more information

    • weightType string

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000

    • stickiness string

      Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • overrides object[]

      Overrides assigning specific variants to specific users. The weighting system automatically assigns users to specific groups for you, but any overrides in this list will take precedence.

    • Array [
    • contextName string required

      The name of the context field used to determine overrides

    • values string[] required

      Which values that should be overriden

    • ]
    • ]
    • strategies object[] deprecated

      This is a legacy field that will be deprecated

    • tags object[]nullable

      The list of feature tags

    • Array [
    • value string required

      Possible values: >= 2 characters and <= 50 characters

      The value of the tag

    • type string required

      Possible values: >= 2 characters and <= 50 characters

      Default value: simple

      The type of the tag

    • ]
    • children string[]

      The list of child feature names. This is an experimental field and may change.

    • dependencies object[]

      The list of parent dependencies. This is an experimental field and may change.

    • Array [
    • feature string required

      The name of the parent feature

    • enabled boolean

      Whether the parent feature is enabled or not

    • variants string[]

      The list of variants the parent feature should resolve to. Only valid when feature is enabled.

    • ]
    • ]
    • total number

      Total count of the features matching search and filter criteria

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/search-users.html b/reference/api/unleash/search-users.html index 50b0c39f26..11bf1ecd32 100644 --- a/reference/api/unleash/search-users.html +++ b/reference/api/unleash/search-users.html @@ -20,15 +20,15 @@ - - + +

    Search users

    GET /api/admin/user-admin/search

    It will preform a simple search based on name and email matching the given query. Requires minimum 2 characters

    Request

    Query Parameters

    • q string

      The pattern to search in the username or email

    Responses

    usersSchema

    Schema
    • users object[]required

      A list of users in the Unleash instance.

    • Array [
    • id integer required

      The user id

    • isAPI boolean deprecated

      (Deprecated): Used internally to know which operations the user should be allowed to perform

    • name string nullable

      Name of the user

    • email string

      Email of the user

    • username string nullable

      A unique username for the user

    • imageUrl string

      URL used for the userprofile image

    • inviteLink string

      If the user is actively inviting other users, this is the link that can be shared with other users

    • loginAttempts integer

      How many unsuccessful attempts at logging in has the user made

    • emailSent boolean

      Is the welcome email sent to the user or not

    • rootRole integer

      Which root role this user is assigned

    • seenAt date-time nullable

      The last time this user logged in

    • createdAt date-time

      The user was created at this time

    • accountType string

      A user is either an actual User or a Service Account

    • permissions string[]

      Deprecated

    • ]
    • rootRoles object[]

      A list of root roles in the Unleash instance.

    • Array [
    • id integer required

      The role id

    • type string required

      A role can either be a global root role (applies to all projects) or a project role

    • name string required

      The name of the role

    • description string

      A more detailed description of the role and what use it's intended for

    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/search.html b/reference/api/unleash/search.html index 50008c928e..366f9117e2 100644 --- a/reference/api/unleash/search.html +++ b/reference/api/unleash/search.html @@ -20,15 +20,15 @@ - - + + - - + + \ No newline at end of file diff --git a/reference/api/unleash/segments.html b/reference/api/unleash/segments.html index e21fee6209..7d4abe78a5 100644 --- a/reference/api/unleash/segments.html +++ b/reference/api/unleash/segments.html @@ -20,15 +20,15 @@ - - + +
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/send-reset-password-email.html b/reference/api/unleash/send-reset-password-email.html index fd9f7fa399..7a1eac860c 100644 --- a/reference/api/unleash/send-reset-password-email.html +++ b/reference/api/unleash/send-reset-password-email.html @@ -20,15 +20,15 @@ - - + +

    Reset password

    POST /auth/reset/password-email

    Requests a password reset email for the user. This email can be used to reset the password for a user that has forgotten their password

    Request

    Body

    required

    emailSchema

    • email string required

      The email address

    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/service-accounts.html b/reference/api/unleash/service-accounts.html index 67d11fb6e7..c335b849f7 100644 --- a/reference/api/unleash/service-accounts.html +++ b/reference/api/unleash/service-accounts.html @@ -20,15 +20,15 @@ - - + +
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/set-google-settings.html b/reference/api/unleash/set-google-settings.html index 5c2b6ce9de..2193a0553a 100644 --- a/reference/api/unleash/set-google-settings.html +++ b/reference/api/unleash/set-google-settings.html @@ -20,15 +20,15 @@ - - + +

    Set Google auth options

    POST /api/admin/auth/google/settings
    deprecated

    This endpoint has been deprecated and may be removed in future versions of the API.

    Updates the settings for Google Authentication (deprecated, please use OpenID instead)

    Request

    Body

    required

    googleSettingsSchema

    • enabled boolean

      Is google OIDC enabled

    • clientId string required

      The google client id, used to authenticate against google

    • clientSecret string required

      The client secret used to authenticate the OAuth session used to log the user in

    • unleashHostname string required

      Name of the host allowed to access the Google authentication flow

    • autoCreate boolean

      Should Unleash create users based on the emails coming back in the authentication reply from Google

    • emailDomains string

      A comma separated list of email domains that Unleash will auto create user accounts for.

    Responses

    googleSettingsSchema

    Schema
    • enabled boolean

      Is google OIDC enabled

    • clientId string required

      The google client id, used to authenticate against google

    • clientSecret string required

      The client secret used to authenticate the OAuth session used to log the user in

    • unleashHostname string required

      Name of the host allowed to access the Google authentication flow

    • autoCreate boolean

      Should Unleash create users based on the emails coming back in the authentication reply from Google

    • emailDomains string

      A comma separated list of email domains that Unleash will auto create user accounts for.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/set-oidc-settings.html b/reference/api/unleash/set-oidc-settings.html index d456b03d54..fc360b243f 100644 --- a/reference/api/unleash/set-oidc-settings.html +++ b/reference/api/unleash/set-oidc-settings.html @@ -20,8 +20,8 @@ - - + + @@ -30,7 +30,7 @@

    Set OIDC settings

    POST /api/admin/auth/oidc/settings

    Configure OpenID Connect as a login provider for Unleash.

    Request

    Body

    required

    oidcSettingsSchema

    • enabled boolean

      true if OpenID connect is turned on for this instance, otherwise false

    • discoverUrl string
    • clientId string required

      The OIDC client ID of this application.

    • secret string required

      Shared secret from OpenID server. Used to authenticate login requests

    • autoCreate boolean

      Auto create users based on email addresses from login tokens

    • enableSingleSignOut boolean

      Support Single sign out when user clicks logout in Unleash. If true user is signed out of all OpenID Connect sessions against the clientId they may have active

    • defaultRootRole string

      Possible values: [Viewer, Editor, Admin]

      Default role granted to users auto-created from email. Only relevant if autoCreate is true

    • emailDomains string

      Comma separated list of email domains that are automatically approved for an account in the server. Only relevant if autoCreate is true

    • acrValues string

      Authentication Context Class Reference, used to request extra values in the acr claim returned from the server. If multiple values are required, they should be space separated. Consult the OIDC reference for more information

    • idTokenSigningAlgorithm string

      Possible values: [RS256, RS384, RS512]

      The signing algorithm used to sign our token. Refer to the JWT signatures documentation for more information.

    Responses

    oidcSettingsSchema

    Schema
    • enabled boolean

      true if OpenID connect is turned on for this instance, otherwise false

    • discoverUrl string
    • clientId string required

      The OIDC client ID of this application.

    • secret string required

      Shared secret from OpenID server. Used to authenticate login requests

    • autoCreate boolean

      Auto create users based on email addresses from login tokens

    • enableSingleSignOut boolean

      Support Single sign out when user clicks logout in Unleash. If true user is signed out of all OpenID Connect sessions against the clientId they may have active

    • defaultRootRole string

      Possible values: [Viewer, Editor, Admin]

      Default role granted to users auto-created from email. Only relevant if autoCreate is true

    • emailDomains string

      Comma separated list of email domains that are automatically approved for an account in the server. Only relevant if autoCreate is true

    • acrValues string

      Authentication Context Class Reference, used to request extra values in the acr claim returned from the server. If multiple values are required, they should be space separated. Consult the OIDC reference for more information

    • idTokenSigningAlgorithm string

      Possible values: [RS256, RS384, RS512]

      The signing algorithm used to sign our token. Refer to the JWT signatures documentation for more information.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/set-project-access.html b/reference/api/unleash/set-project-access.html index bda2bf90b7..533492de2f 100644 --- a/reference/api/unleash/set-project-access.html +++ b/reference/api/unleash/set-project-access.html @@ -20,15 +20,15 @@ - - + +

    Set users and groups to roles in the current project

    PUT /api/admin/projects/:projectId/access

    Sets all groups, users and their roles for the given project, overriding any existing configuration.

    Request

    Path Parameters

    • projectId string required

    Body

    required

    projectAccessConfigurationSchema

    • roles object[]required

      A list of roles that are available within this project.

    • Array [
    • id integer

      Possible values: >= 1

      The id of the role.

    • groups integer[]

      A list of group ids that will be assigned this role

    • users integer[]

      A list of user ids that will be assigned this role

    • ]
    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/set-roles-for-group.html b/reference/api/unleash/set-roles-for-group.html index 79756ea2f4..ea416be095 100644 --- a/reference/api/unleash/set-roles-for-group.html +++ b/reference/api/unleash/set-roles-for-group.html @@ -20,15 +20,15 @@ - - + +

    Sets roles for group

    PUT /api/admin/projects/:projectId/groups/:groupId/roles

    Sets the roles a group has in the project.

    Request

    Path Parameters

    • projectId string required
    • groupId string required
    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/set-roles-for-user.html b/reference/api/unleash/set-roles-for-user.html index e5afbb0c1c..0afaf4119f 100644 --- a/reference/api/unleash/set-roles-for-user.html +++ b/reference/api/unleash/set-roles-for-user.html @@ -20,15 +20,15 @@ - - + +

    Sets roles for user

    PUT /api/admin/projects/:projectId/users/:userId/roles

    Sets the roles a user has in the project.

    Request

    Path Parameters

    • projectId string required
    • userId string required
    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/set-saml-settings.html b/reference/api/unleash/set-saml-settings.html index b9dc876908..0fe40a5b8f 100644 --- a/reference/api/unleash/set-saml-settings.html +++ b/reference/api/unleash/set-saml-settings.html @@ -20,15 +20,15 @@ - - + +

    Update SAML auth settings

    POST /api/admin/auth/saml/settings

    Updates the settings for SAML Authentication

    Request

    Body

    required

    samlSettingsSchema

    • enabled boolean

      Is SAML authentication enabled

    • entityId string required

      The SAML 2.0 entity ID

    • signOnUrl string required

      Which URL to use for Single Sign On

    • certificate string required

      The X509 certificate used to validate requests

    • signOutUrl string

      Which URL to use for Single Sign Out

    • spCertificate string

      Signing certificate for sign out requests

    • autoCreate boolean

      Should Unleash create users based on the emails coming back in the authentication reply from the SAML server

    • emailDomains string

      A comma separated list of email domains that Unleash will auto create user accounts for.

    • defaultRootRole string

      Possible values: [Viewer, Editor, Admin]

      Assign this root role to auto created users

    Responses

    samlSettingsSchema

    Schema
    • enabled boolean

      Is SAML authentication enabled

    • entityId string required

      The SAML 2.0 entity ID

    • signOnUrl string required

      Which URL to use for Single Sign On

    • certificate string required

      The X509 certificate used to validate requests

    • signOutUrl string

      Which URL to use for Single Sign Out

    • spCertificate string

      Signing certificate for sign out requests

    • autoCreate boolean

      Should Unleash create users based on the emails coming back in the authentication reply from the SAML server

    • emailDomains string

      A comma separated list of email domains that Unleash will auto create user accounts for.

    • defaultRootRole string

      Possible values: [Viewer, Editor, Admin]

      Assign this root role to auto created users

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/set-simple-settings.html b/reference/api/unleash/set-simple-settings.html index 41c0d098d9..7557d30ba3 100644 --- a/reference/api/unleash/set-simple-settings.html +++ b/reference/api/unleash/set-simple-settings.html @@ -20,15 +20,15 @@ - - + +

    Update Simple auth settings

    POST /api/admin/auth/simple/settings

    Enable or disable simple authentication (username/password)

    Request

    Body

    required

    passwordAuthSchema

    • enabled boolean

      Is username/password authentication enabled

    Responses

    passwordAuthSchema

    Schema
    • enabled boolean

      Is username/password authentication enabled

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/set-strategy-sort-order.html b/reference/api/unleash/set-strategy-sort-order.html index 340ac1f21f..5c51432a87 100644 --- a/reference/api/unleash/set-strategy-sort-order.html +++ b/reference/api/unleash/set-strategy-sort-order.html @@ -20,15 +20,15 @@ - - + +

    Set strategy sort order

    POST /api/admin/projects/:projectId/features/:featureName/environments/:environment/strategies/set-sort-order

    Set the sort order of the provided list of strategies.

    Request

    Path Parameters

    • projectId string required
    • featureName string required
    • environment string required

    Body

    arrayrequired

    setStrategySortOrderSchema

    • Array [
    • id string required

      The ID of the strategy

    • sortOrder number required

      The new sort order of the strategy

    • ]
    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/set-ui-config.html b/reference/api/unleash/set-ui-config.html index 937315e5ff..e191758928 100644 --- a/reference/api/unleash/set-ui-config.html +++ b/reference/api/unleash/set-ui-config.html @@ -20,15 +20,15 @@ - - + +
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/stale-features.html b/reference/api/unleash/stale-features.html index a1caec781f..1899411f9a 100644 --- a/reference/api/unleash/stale-features.html +++ b/reference/api/unleash/stale-features.html @@ -20,15 +20,15 @@ - - + +

    Mark features as stale / not stale

    POST /api/admin/projects/:projectId/stale

    This endpoint marks the provided list of features as either stale or not stale depending on the request body you send. Any provided features that don't exist are ignored.

    Request

    Path Parameters

    • projectId string required

    Body

    required

    batchStaleSchema

    • features string[] required

      A list of features to mark as (not) stale

    • stale boolean required

      Whether the list of features should be marked as stale or not stale. If true, the features will be marked as stale. If false, the features will be marked as not stale.

    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/strategies.html b/reference/api/unleash/strategies.html index fce78b7dd2..6c0ff85c67 100644 --- a/reference/api/unleash/strategies.html +++ b/reference/api/unleash/strategies.html @@ -20,15 +20,15 @@ - - + +
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/tags.html b/reference/api/unleash/tags.html index 805548bb47..a43dae42cd 100644 --- a/reference/api/unleash/tags.html +++ b/reference/api/unleash/tags.html @@ -20,15 +20,15 @@ - - + +
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/telemetry.html b/reference/api/unleash/telemetry.html index 8bab21ce44..ebf6383c55 100644 --- a/reference/api/unleash/telemetry.html +++ b/reference/api/unleash/telemetry.html @@ -20,15 +20,15 @@ - - + + - - + + \ No newline at end of file diff --git a/reference/api/unleash/toggle-environment-off.html b/reference/api/unleash/toggle-environment-off.html index c0dc00c783..cfd19b45a4 100644 --- a/reference/api/unleash/toggle-environment-off.html +++ b/reference/api/unleash/toggle-environment-off.html @@ -20,15 +20,15 @@ - - + +

    Toggle the environment with `name` off

    POST /api/admin/environments/:name/off

    Removes this environment from the list of available environments for projects to use

    Request

    Path Parameters

    • name string required
    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/toggle-environment-on.html b/reference/api/unleash/toggle-environment-on.html index 4a32b1c5fe..dbee53673f 100644 --- a/reference/api/unleash/toggle-environment-on.html +++ b/reference/api/unleash/toggle-environment-on.html @@ -20,15 +20,15 @@ - - + +

    Toggle the environment with `name` on

    POST /api/admin/environments/:name/on

    Makes it possible to enable this environment for a project. An environment must first be globally enabled using this endpoint before it can be enabled for a project

    Request

    Path Parameters

    • name string required
    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/toggle-feature-environment-off.html b/reference/api/unleash/toggle-feature-environment-off.html index a75eddabe5..f77558f88e 100644 --- a/reference/api/unleash/toggle-feature-environment-off.html +++ b/reference/api/unleash/toggle-feature-environment-off.html @@ -20,15 +20,15 @@ - - + +

    Disable a feature toggle

    POST /api/admin/projects/:projectId/features/:featureName/environments/:environment/off

    Disable a feature toggle in the specified environment.

    Request

    Path Parameters

    • projectId string required
    • featureName string required
    • environment string required
    Responses

    featureSchema

    Schema
    • name string required

      Unique feature name

    • type string

      Type of the toggle e.g. experiment, kill-switch, release, operational, permission

    • description string nullable

      Detailed description of the feature

    • archived boolean

      true if the feature is archived

    • project string

      Name of the project the feature belongs to

    • enabled boolean

      true if the feature is enabled, otherwise false.

    • stale boolean

      true if the feature is stale based on the age and feature type, otherwise false.

    • favorite boolean

      true if the feature was favorited, otherwise false.

    • impressionData boolean

      true if the impression data collection is enabled for the feature, otherwise false.

    • createdAt date-time nullable

      The date the feature was created

    • archivedAt date-time nullable

      The date the feature was archived

    • lastSeenAt date-time nullable deprecated

      The date when metrics where last collected for the feature. This field is deprecated, use the one in featureEnvironmentSchema

    • environments object[]

      The list of environments where the feature can be used

    • Array [
    • name string required

      The name of the environment

    • featureName string

      The name of the feature

    • environment string

      The name of the environment

    • type string

      The type of the environment

    • enabled boolean required

      true if the feature is enabled for the environment, otherwise false.

    • sortOrder number

      The sort order of the feature environment in the feature environments list

    • variantCount number

      The number of defined variants

    • strategies object[]

      A list of activation strategies for the feature environment

    • Array [
    • id string

      A uuid for the feature strategy

    • name string required

      The name or type of strategy

    • title string nullable

      A descriptive title for the strategy

    • disabled boolean nullable

      A toggle to disable the strategy. defaults to false. Disabled strategies are not evaluated or returned to the SDKs

    • featureName string

      The name or feature the strategy is attached to

    • sortOrder number

      The order of the strategy in the list

    • segments number[]

      A list of segment ids attached to the strategy

    • constraints object[]

      A list of the constraints attached to the strategy. See https://docs.getunleash.io/reference/strategy-constraints

    • Array [
    • contextName string required

      The name of the context field that this constraint should apply to.

    • operator string required

      Possible values: [NOT_IN, IN, STR_ENDS_WITH, STR_STARTS_WITH, STR_CONTAINS, NUM_EQ, NUM_GT, NUM_GTE, NUM_LT, NUM_LTE, DATE_AFTER, DATE_BEFORE, SEMVER_EQ, SEMVER_GT, SEMVER_LT]

      The operator to use when evaluating this constraint. For more information about the various operators, refer to the strategy constraint operator documentation.

    • caseInsensitive boolean

      Default value: false

      Whether the operator should be case sensitive or not. Defaults to false (being case sensitive).

    • inverted boolean

      Default value: false

      Whether the result should be negated or not. If true, will turn a true result into a false result and vice versa.

    • values string[]

      The context values that should be used for constraint evaluation. Use this property instead of value for properties that accept multiple values.

    • value string

      The context value that should be used for constraint evaluation. Use this property instead of values for properties that only accept single values.

    • ]
    • variants object[]

      Strategy level variants

    • Array [
    • name string required

      The variant name. Must be unique for this feature toggle

    • weight integer required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is an integer between 0 and 1000. See the section on variant weights for more information

    • weightType string required

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000. Refer to the variant weight documentation.

    • stickiness string required

      The stickiness to use for distribution of this variant. Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • ]
    • parameters object

      A list of parameters for a strategy

    • property name* string
    • ]
    • variants object[]

      A list of variants for the feature environment

    • Array [
    • name string required

      The variants name. Is unique for this feature toggle

    • weight number required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is a number between 0 and 1000. See the section on variant weights for more information

    • weightType string

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000

    • stickiness string

      Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • overrides object[]

      Overrides assigning specific variants to specific users. The weighting system automatically assigns users to specific groups for you, but any overrides in this list will take precedence.

    • Array [
    • contextName string required

      The name of the context field used to determine overrides

    • values string[] required

      Which values that should be overriden

    • ]
    • ]
    • lastSeenAt date-time nullable

      The date when metrics where last collected for the feature environment

    • hasStrategies boolean

      Whether the feature has any strategies defined.

    • hasEnabledStrategies boolean

      Whether the feature has any enabled strategies defined.

    • ]
    • variants object[]deprecated

      The list of feature variants

    • Array [
    • name string required

      The variants name. Is unique for this feature toggle

    • weight number required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is a number between 0 and 1000. See the section on variant weights for more information

    • weightType string

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000

    • stickiness string

      Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • overrides object[]

      Overrides assigning specific variants to specific users. The weighting system automatically assigns users to specific groups for you, but any overrides in this list will take precedence.

    • Array [
    • contextName string required

      The name of the context field used to determine overrides

    • values string[] required

      Which values that should be overriden

    • ]
    • ]
    • strategies object[] deprecated

      This is a legacy field that will be deprecated

    • tags object[]nullable

      The list of feature tags

    • Array [
    • value string required

      Possible values: >= 2 characters and <= 50 characters

      The value of the tag

    • type string required

      Possible values: >= 2 characters and <= 50 characters

      Default value: simple

      The type of the tag

    • ]
    • children string[]

      The list of child feature names. This is an experimental field and may change.

    • dependencies object[]

      The list of parent dependencies. This is an experimental field and may change.

    • Array [
    • feature string required

      The name of the parent feature

    • enabled boolean

      Whether the parent feature is enabled or not

    • variants string[]

      The list of variants the parent feature should resolve to. Only valid when feature is enabled.

    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/toggle-feature-environment-on.html b/reference/api/unleash/toggle-feature-environment-on.html index aa47757ac0..248cbddee9 100644 --- a/reference/api/unleash/toggle-feature-environment-on.html +++ b/reference/api/unleash/toggle-feature-environment-on.html @@ -20,15 +20,15 @@ - - + +

    Enable a feature toggle

    POST /api/admin/projects/:projectId/features/:featureName/environments/:environment/on

    Enable a feature toggle in the specified environment.

    Request

    Path Parameters

    • projectId string required
    • featureName string required
    • environment string required
    Responses

    featureSchema

    Schema
    • name string required

      Unique feature name

    • type string

      Type of the toggle e.g. experiment, kill-switch, release, operational, permission

    • description string nullable

      Detailed description of the feature

    • archived boolean

      true if the feature is archived

    • project string

      Name of the project the feature belongs to

    • enabled boolean

      true if the feature is enabled, otherwise false.

    • stale boolean

      true if the feature is stale based on the age and feature type, otherwise false.

    • favorite boolean

      true if the feature was favorited, otherwise false.

    • impressionData boolean

      true if the impression data collection is enabled for the feature, otherwise false.

    • createdAt date-time nullable

      The date the feature was created

    • archivedAt date-time nullable

      The date the feature was archived

    • lastSeenAt date-time nullable deprecated

      The date when metrics where last collected for the feature. This field is deprecated, use the one in featureEnvironmentSchema

    • environments object[]

      The list of environments where the feature can be used

    • Array [
    • name string required

      The name of the environment

    • featureName string

      The name of the feature

    • environment string

      The name of the environment

    • type string

      The type of the environment

    • enabled boolean required

      true if the feature is enabled for the environment, otherwise false.

    • sortOrder number

      The sort order of the feature environment in the feature environments list

    • variantCount number

      The number of defined variants

    • strategies object[]

      A list of activation strategies for the feature environment

    • Array [
    • id string

      A uuid for the feature strategy

    • name string required

      The name or type of strategy

    • title string nullable

      A descriptive title for the strategy

    • disabled boolean nullable

      A toggle to disable the strategy. defaults to false. Disabled strategies are not evaluated or returned to the SDKs

    • featureName string

      The name or feature the strategy is attached to

    • sortOrder number

      The order of the strategy in the list

    • segments number[]

      A list of segment ids attached to the strategy

    • constraints object[]

      A list of the constraints attached to the strategy. See https://docs.getunleash.io/reference/strategy-constraints

    • Array [
    • contextName string required

      The name of the context field that this constraint should apply to.

    • operator string required

      Possible values: [NOT_IN, IN, STR_ENDS_WITH, STR_STARTS_WITH, STR_CONTAINS, NUM_EQ, NUM_GT, NUM_GTE, NUM_LT, NUM_LTE, DATE_AFTER, DATE_BEFORE, SEMVER_EQ, SEMVER_GT, SEMVER_LT]

      The operator to use when evaluating this constraint. For more information about the various operators, refer to the strategy constraint operator documentation.

    • caseInsensitive boolean

      Default value: false

      Whether the operator should be case sensitive or not. Defaults to false (being case sensitive).

    • inverted boolean

      Default value: false

      Whether the result should be negated or not. If true, will turn a true result into a false result and vice versa.

    • values string[]

      The context values that should be used for constraint evaluation. Use this property instead of value for properties that accept multiple values.

    • value string

      The context value that should be used for constraint evaluation. Use this property instead of values for properties that only accept single values.

    • ]
    • variants object[]

      Strategy level variants

    • Array [
    • name string required

      The variant name. Must be unique for this feature toggle

    • weight integer required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is an integer between 0 and 1000. See the section on variant weights for more information

    • weightType string required

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000. Refer to the variant weight documentation.

    • stickiness string required

      The stickiness to use for distribution of this variant. Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • ]
    • parameters object

      A list of parameters for a strategy

    • property name* string
    • ]
    • variants object[]

      A list of variants for the feature environment

    • Array [
    • name string required

      The variants name. Is unique for this feature toggle

    • weight number required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is a number between 0 and 1000. See the section on variant weights for more information

    • weightType string

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000

    • stickiness string

      Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • overrides object[]

      Overrides assigning specific variants to specific users. The weighting system automatically assigns users to specific groups for you, but any overrides in this list will take precedence.

    • Array [
    • contextName string required

      The name of the context field used to determine overrides

    • values string[] required

      Which values that should be overriden

    • ]
    • ]
    • lastSeenAt date-time nullable

      The date when metrics where last collected for the feature environment

    • hasStrategies boolean

      Whether the feature has any strategies defined.

    • hasEnabledStrategies boolean

      Whether the feature has any enabled strategies defined.

    • ]
    • variants object[]deprecated

      The list of feature variants

    • Array [
    • name string required

      The variants name. Is unique for this feature toggle

    • weight number required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is a number between 0 and 1000. See the section on variant weights for more information

    • weightType string

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000

    • stickiness string

      Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • overrides object[]

      Overrides assigning specific variants to specific users. The weighting system automatically assigns users to specific groups for you, but any overrides in this list will take precedence.

    • Array [
    • contextName string required

      The name of the context field used to determine overrides

    • values string[] required

      Which values that should be overriden

    • ]
    • ]
    • strategies object[] deprecated

      This is a legacy field that will be deprecated

    • tags object[]nullable

      The list of feature tags

    • Array [
    • value string required

      Possible values: >= 2 characters and <= 50 characters

      The value of the tag

    • type string required

      Possible values: >= 2 characters and <= 50 characters

      Default value: simple

      The type of the tag

    • ]
    • children string[]

      The list of child feature names. This is an experimental field and may change.

    • dependencies object[]

      The list of parent dependencies. This is an experimental field and may change.

    • Array [
    • feature string required

      The name of the parent feature

    • enabled boolean

      Whether the parent feature is enabled or not

    • variants string[]

      The list of variants the parent feature should resolve to. Only valid when feature is enabled.

    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/toggle-maintenance.html b/reference/api/unleash/toggle-maintenance.html index 2ecca49826..deda270e8a 100644 --- a/reference/api/unleash/toggle-maintenance.html +++ b/reference/api/unleash/toggle-maintenance.html @@ -20,15 +20,15 @@ - - + +

    Enabled/disabled maintenance mode

    POST /api/admin/maintenance

    Lets administrators put Unleash into a mostly read-only mode. While Unleash is in maintenance mode, users can not change any configuration settings

    Request

    Body

    required

    toggleMaintenanceSchema

    • enabled boolean required

      true if you want to activate maintenance mode, false if you want to deactivate it.

    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/unleash-api.html b/reference/api/unleash/unleash-api.html index fc448e96ee..1efa6bbdf9 100644 --- a/reference/api/unleash/unleash-api.html +++ b/reference/api/unleash/unleash-api.html @@ -20,15 +20,15 @@ - - + + - - + + \ No newline at end of file diff --git a/reference/api/unleash/unstable.html b/reference/api/unleash/unstable.html index c506ec06a2..db6aef9674 100644 --- a/reference/api/unleash/unstable.html +++ b/reference/api/unleash/unstable.html @@ -20,15 +20,15 @@ - - + +
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/update-addon.html b/reference/api/unleash/update-addon.html index 7d90672223..4c03a98451 100644 --- a/reference/api/unleash/update-addon.html +++ b/reference/api/unleash/update-addon.html @@ -20,8 +20,8 @@ - - + + @@ -37,7 +37,7 @@
  • webhook for webhooks
  • The provider you choose for your addon dictates what properties the parameters object needs. Refer to the documentation for each provider for more information.

  • description string

    A description of the addon.

  • enabled boolean required

    Whether the addon should be enabled or not.

  • parameters objectrequired

    Parameters for the addon provider. This object has different required and optional properties depending on the provider you choose. Consult the documentation for details.

  • events string[] required

    The event types that will trigger this specific addon.

  • projects string[]

    The projects that this addon will listen to events from. An empty list means it will listen to events from all projects.

  • environments string[]

    The list of environments that this addon will listen to events from. An empty list means it will listen to events from all environments.

  • Responses

    addonSchema

    Schema
    • id integer required

      Possible values: >= 1

      The addon's unique identifier.

    • provider string required

      The addon provider, such as "webhook" or "slack".

    • description string nullable required

      A description of the addon. null if no description exists.

    • enabled boolean required

      Whether the addon is enabled or not.

    • parameters objectrequired

      Parameters for the addon provider. This object has different required and optional properties depending on the provider you choose.

    • events string[] required

      The event types that trigger this specific addon.

    • projects string[]

      The projects that this addon listens to events from. An empty list means it listens to events from all projects.

    • environments string[]

      The list of environments that this addon listens to events from. An empty list means it listens to events from all environments.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/update-api-token.html b/reference/api/unleash/update-api-token.html index 620df524f5..a732558cab 100644 --- a/reference/api/unleash/update-api-token.html +++ b/reference/api/unleash/update-api-token.html @@ -20,15 +20,15 @@ - - + +

    Update API token

    PUT /api/admin/api-tokens/:token

    Updates an existing API token with a new expiry date. The token path parameter is the token's secret. If the token does not exist, this endpoint returns a 200 OK, but does nothing.

    Request

    Path Parameters

    • token string required

    Body

    required

    updateApiTokenSchema

    • expiresAt date-time required

      The new time when this token should expire.

    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/update-banner.html b/reference/api/unleash/update-banner.html index 4da26df192..0ec92a29d9 100644 --- a/reference/api/unleash/update-banner.html +++ b/reference/api/unleash/update-banner.html @@ -20,15 +20,15 @@ - - + +

    Update a banner.

    PUT /api/admin/banners/:id

    Updates an existing banner identified by its id.

    Request

    Path Parameters

    • id string required

    Body

    required

    createBannerSchema

    • message string required

      The message to display to all users. Supports markdown.

    • enabled boolean

      Whether the banner should be displayed currently. If not specified, defaults to true.

    • variant string

      The variant of the banner. One of "info", "warning", "error", or "success". If not specified, defaults to "info".

    • sticky boolean

      Whether the banner should be sticky on the screen. If not specified, defaults to false.

    • icon string nullable

      The icon to display on the banner. Can be one of https://fonts.google.com/icons. If not specified, this will be the default icon for the variant. If "none", no icon will be displayed.

    • link string nullable

      The link to display on the banner. Can either be an absolute or a relative link (e.g. absolute: "https://example.com" or relative: "/admin/service-accounts"). If "dialog", will display a dialog when clicked. If not specified, no link will be displayed.

    • linkText string nullable

      The text to display on the link. If not specified, will be displayed as "More info".

    • dialogTitle string nullable

      The title to display on the dialog. If not specified, this will be the same as linkText.

    • dialog string nullable

      The markdown to display on the dialog. If not specified, no dialog will be displayed.

    Responses

    bannerSchema

    Schema
    • id integer required

      Possible values: >= 1

      The banner's ID. Banner IDs are incrementing integers. In other words, a more recently created banner will always have a higher ID than an older one.

    • message string required

      The message to display to all users. Supports markdown.

    • enabled boolean

      Whether the banner should be displayed currently. If not specified, defaults to true.

    • variant string

      The variant of the banner. One of "info", "warning", "error", or "success". If not specified, defaults to "info".

    • sticky boolean

      Whether the banner should be sticky on the screen. If not specified, defaults to false.

    • icon string nullable

      The icon to display on the banner. Can be one of https://fonts.google.com/icons. If not specified, this will be the default icon for the variant. If "none", no icon will be displayed.

    • link string nullable

      The link to display on the banner. Can either be an absolute or a relative link (e.g. absolute: "https://example.com" or relative: "/admin/service-accounts"). If "dialog", will display a dialog when clicked. If not specified, no link will be displayed.

    • linkText string nullable

      The text to display on the link. If not specified, will be displayed as "More info".

    • dialogTitle string nullable

      The title to display on the dialog. If not specified, this will be the same as linkText.

    • dialog string nullable

      The markdown to display on the dialog. If not specified, no dialog will be displayed.

    • createdAt date-time required

      The date and time of when the banner was created.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/update-change-request-state.html b/reference/api/unleash/update-change-request-state.html index b26f0f153a..14f4f1bd96 100644 --- a/reference/api/unleash/update-change-request-state.html +++ b/reference/api/unleash/update-change-request-state.html @@ -20,15 +20,15 @@ - - + +

    This endpoint will update the state of a change request

    PUT /api/admin/projects/:projectId/change-requests/:id/state

    This endpoint will update the state of a change request if the business rules allow it. The state can be one of the following: Draft, In review, Approved, Cancelled, Applied. In order to be approved, the change request must have at least one change and the number of approvals must be greater than or equal to the number of approvals required for the environment.

                    Once a change request has been approved, it can be applied. Once a change request has been applied, it cannot be changed. Once a change request has been cancelled, it cannot be changed. Any change to a change request in the state of Approved will result in the state being set to In Review and the number of approvals will be reset.

    Request

    Path Parameters

    • projectId string required
    • id string required
    Responses

    changeRequestStateSchema

    Schema
      oneOf
    • state string required

      Possible values: [Draft, In review, Approved, Applied, Cancelled, Rejected]

      The new desired state for the change request

    • comment string

      Any comments accompanying the state changed. Used when sending a draft to review.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/update-change-request-title.html b/reference/api/unleash/update-change-request-title.html index 7243b60b5e..0038fc56df 100644 --- a/reference/api/unleash/update-change-request-title.html +++ b/reference/api/unleash/update-change-request-title.html @@ -20,8 +20,8 @@ - - + + @@ -29,7 +29,7 @@

    This endpoint will update the custom title of a change request

    PUT /api/admin/projects/:projectId/change-requests/:id/title

    Change requests get a default title e.g. Change Request #1. This endpoint allows to make the title more meaningful and describe the intent behind the Change Request

    Request

    Path Parameters

    • projectId string required
    • id string required
    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/update-context-field.html b/reference/api/unleash/update-context-field.html index 1f71dd6748..d82d1a6422 100644 --- a/reference/api/unleash/update-context-field.html +++ b/reference/api/unleash/update-context-field.html @@ -20,15 +20,15 @@ - - + +

    Update an existing context field

    PUT /api/admin/context/:contextField

    Endpoint that allows updating a custom context field. Used to toggle stickiness and add/remove legal values for this context field

    Request

    Path Parameters

    • contextField string required

    Body

    required

    updateContextFieldSchema

    • description string

      A description of the context field

    • stickiness boolean

      true if this field should be available for use with custom stickiness, otherwise false

    • sortOrder integer

      How this context field should be sorted if no other sort order is selected

    • legalValues object[]

      A list of allowed values for this context field

    • Array [
    • value string required

      The valid value

    • description string

      Describes this specific legal value

    • ]
    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/update-environment.html b/reference/api/unleash/update-environment.html index e2e1370f3d..b5f8c4e319 100644 --- a/reference/api/unleash/update-environment.html +++ b/reference/api/unleash/update-environment.html @@ -20,15 +20,15 @@ - - + +

    Updates an environment by name

    PUT /api/admin/environments/update/:name

    Given an environment by name updates the environment with the given payload. Note that name, enabled and protected cannot be changed by this API

    Request

    Path Parameters

    • name string required

    Body

    required

    updateEnvironmentSchema

    • type string

      Updates the type of environment (i.e. development or production).

    • sortOrder integer

      Changes the sort order of this environment.

    Responses

    environmentSchema

    Schema
    • name string required

      The name of the environment

    • type string required
    • enabled boolean required

      true if the environment is enabled for the project, otherwise false.

    • protected boolean required

      true if the environment is protected, otherwise false. A protected environment can not be deleted.

    • sortOrder integer required

      Priority of the environment in a list of environments, the lower the value, the higher up in the list the environment will appear. Needs to be an integer

    • projectCount integer nullable

      The number of projects with this environment

    • apiTokenCount integer nullable

      The number of API tokens for the project environment

    • enabledToggleCount integer nullable

      The number of enabled toggles for the project environment

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/update-feature-strategy-segments.html b/reference/api/unleash/update-feature-strategy-segments.html index c6e0604aeb..256a49fa68 100644 --- a/reference/api/unleash/update-feature-strategy-segments.html +++ b/reference/api/unleash/update-feature-strategy-segments.html @@ -20,15 +20,15 @@ - - + +

    Update strategy segments

    POST /api/admin/segments/strategies

    Sets the segments of the strategy specified to be exactly the ones passed in the payload. Any segments that were used by the strategy before will be removed if they are not in the provided list of segments.

    Request

    Body

    required

    updateFeatureStrategySegmentsSchema

    • projectId string required

      The ID of the project that the strategy belongs to.

    • strategyId string required

      The ID of the strategy to update segments for.

    • environmentId string required

      The ID of the strategy environment.

    • segmentIds integer[] required

      The new list of segments (IDs) to use for this strategy. Any segments not in this list will be removed from the strategy.

    Responses

    The resource was successfully created.

    Response Headers
    • location string

      The location of the newly created resource.

    Schema
    • projectId string required

      The ID of the project that the strategy belongs to.

    • strategyId string required

      The ID of the strategy to update segments for.

    • environmentId string required

      The ID of the strategy environment.

    • segmentIds integer[] required

      The new list of segments (IDs) to use for this strategy. Any segments not in this list will be removed from the strategy.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/update-feature-strategy.html b/reference/api/unleash/update-feature-strategy.html index 84e04b282d..184d9eb158 100644 --- a/reference/api/unleash/update-feature-strategy.html +++ b/reference/api/unleash/update-feature-strategy.html @@ -20,15 +20,15 @@ - - + +

    Update a strategy

    PUT /api/admin/projects/:projectId/features/:featureName/environments/:environment/strategies/:strategyId

    Replace strategy configuration for a feature toggle in the specified environment.

    Request

    Path Parameters

    • projectId string required
    • featureName string required
    • environment string required
    • strategyId string required

    Body

    required

    updateFeatureStrategySchema

    • name string

      The name of the strategy type

    • sortOrder number

      The order of the strategy in the list in feature environment configuration

    • constraints object[]

      A list of the constraints attached to the strategy. See https://docs.getunleash.io/reference/strategy-constraints

    • Array [
    • contextName string required

      The name of the context field that this constraint should apply to.

    • operator string required

      Possible values: [NOT_IN, IN, STR_ENDS_WITH, STR_STARTS_WITH, STR_CONTAINS, NUM_EQ, NUM_GT, NUM_GTE, NUM_LT, NUM_LTE, DATE_AFTER, DATE_BEFORE, SEMVER_EQ, SEMVER_GT, SEMVER_LT]

      The operator to use when evaluating this constraint. For more information about the various operators, refer to the strategy constraint operator documentation.

    • caseInsensitive boolean

      Default value: false

      Whether the operator should be case sensitive or not. Defaults to false (being case sensitive).

    • inverted boolean

      Default value: false

      Whether the result should be negated or not. If true, will turn a true result into a false result and vice versa.

    • values string[]

      The context values that should be used for constraint evaluation. Use this property instead of value for properties that accept multiple values.

    • value string

      The context value that should be used for constraint evaluation. Use this property instead of values for properties that only accept single values.

    • ]
    • title string nullable

      A descriptive title for the strategy

    • disabled boolean nullable

      A toggle to disable the strategy. defaults to true. Disabled strategies are not evaluated or returned to the SDKs

    • parameters object

      A list of parameters for a strategy

    • property name* string
    Responses

    featureStrategySchema

    Schema
    • id string

      A uuid for the feature strategy

    • name string required

      The name or type of strategy

    • title string nullable

      A descriptive title for the strategy

    • disabled boolean nullable

      A toggle to disable the strategy. defaults to false. Disabled strategies are not evaluated or returned to the SDKs

    • featureName string

      The name or feature the strategy is attached to

    • sortOrder number

      The order of the strategy in the list

    • segments number[]

      A list of segment ids attached to the strategy

    • constraints object[]

      A list of the constraints attached to the strategy. See https://docs.getunleash.io/reference/strategy-constraints

    • Array [
    • contextName string required

      The name of the context field that this constraint should apply to.

    • operator string required

      Possible values: [NOT_IN, IN, STR_ENDS_WITH, STR_STARTS_WITH, STR_CONTAINS, NUM_EQ, NUM_GT, NUM_GTE, NUM_LT, NUM_LTE, DATE_AFTER, DATE_BEFORE, SEMVER_EQ, SEMVER_GT, SEMVER_LT]

      The operator to use when evaluating this constraint. For more information about the various operators, refer to the strategy constraint operator documentation.

    • caseInsensitive boolean

      Default value: false

      Whether the operator should be case sensitive or not. Defaults to false (being case sensitive).

    • inverted boolean

      Default value: false

      Whether the result should be negated or not. If true, will turn a true result into a false result and vice versa.

    • values string[]

      The context values that should be used for constraint evaluation. Use this property instead of value for properties that accept multiple values.

    • value string

      The context value that should be used for constraint evaluation. Use this property instead of values for properties that only accept single values.

    • ]
    • variants object[]

      Strategy level variants

    • Array [
    • name string required

      The variant name. Must be unique for this feature toggle

    • weight integer required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is an integer between 0 and 1000. See the section on variant weights for more information

    • weightType string required

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000. Refer to the variant weight documentation.

    • stickiness string required

      The stickiness to use for distribution of this variant. Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • ]
    • parameters object

      A list of parameters for a strategy

    • property name* string
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/update-feature-type-lifetime.html b/reference/api/unleash/update-feature-type-lifetime.html index 53b399101c..5d3a6a0f52 100644 --- a/reference/api/unleash/update-feature-type-lifetime.html +++ b/reference/api/unleash/update-feature-type-lifetime.html @@ -20,15 +20,15 @@ - - + +

    Update feature type lifetime

    PUT /api/admin/feature-types/:id/lifetime

    Updates the lifetime configuration for the specified feature toggle type. The expected lifetime is an integer representing the number of days before Unleash marks a feature toggle of that type as potentially stale. If set to null or 0, then feature toggles of that particular type will never be marked as potentially stale.

    When a feature toggle type's expected lifetime is changed, this will also cause any feature toggles of this type to be reevaluated for potential staleness.

    Request

    Path Parameters

    • id string required

    Body

    required

    updateFeatureTypeLifetimeSchema

    • lifetimeDays integer nullable required

      Possible values: <= 2147483647

      The new lifetime (in days) that you want to assign to the feature toggle type. If the value is null or 0, then the feature toggles of that type will never be marked as potentially stale. Otherwise, they will be considered potentially stale after the number of days indicated by this property.

    Responses

    featureTypeSchema

    Schema
    • id string required

      The identifier of this feature toggle type.

    • name string required

      The display name of this feature toggle type.

    • description string required

      A description of what this feature toggle type is intended to be used for.

    • lifetimeDays integer nullable required

      How many days it takes before a feature toggle of this typed is flagged as potentially stale by Unleash. If this value is null, Unleash will never mark it as potentially stale.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/update-feature.html b/reference/api/unleash/update-feature.html index d17f55a1c9..55d84b950d 100644 --- a/reference/api/unleash/update-feature.html +++ b/reference/api/unleash/update-feature.html @@ -20,15 +20,15 @@ - - + +

    Update a feature toggle

    PUT /api/admin/projects/:projectId/features/:featureName

    Updates the specified feature if the feature belongs to the specified project. Only the provided properties are updated; any feature properties left out of the request body are left untouched.

    Request

    Path Parameters

    • projectId string required
    • featureName string required

    Body

    required

    updateFeatureSchema

    • description string

      Detailed description of the feature

    • type string

      Type of the toggle e.g. experiment, kill-switch, release, operational, permission

    • stale boolean

      true if the feature is archived

    • archived boolean

      If true the feature toggle will be moved to the archive with a property archivedAt set to current time

    • impressionData boolean

      true if the impression data collection is enabled for the feature

    Responses

    featureSchema

    Schema
    • name string required

      Unique feature name

    • type string

      Type of the toggle e.g. experiment, kill-switch, release, operational, permission

    • description string nullable

      Detailed description of the feature

    • archived boolean

      true if the feature is archived

    • project string

      Name of the project the feature belongs to

    • enabled boolean

      true if the feature is enabled, otherwise false.

    • stale boolean

      true if the feature is stale based on the age and feature type, otherwise false.

    • favorite boolean

      true if the feature was favorited, otherwise false.

    • impressionData boolean

      true if the impression data collection is enabled for the feature, otherwise false.

    • createdAt date-time nullable

      The date the feature was created

    • archivedAt date-time nullable

      The date the feature was archived

    • lastSeenAt date-time nullable deprecated

      The date when metrics where last collected for the feature. This field is deprecated, use the one in featureEnvironmentSchema

    • environments object[]

      The list of environments where the feature can be used

    • Array [
    • name string required

      The name of the environment

    • featureName string

      The name of the feature

    • environment string

      The name of the environment

    • type string

      The type of the environment

    • enabled boolean required

      true if the feature is enabled for the environment, otherwise false.

    • sortOrder number

      The sort order of the feature environment in the feature environments list

    • variantCount number

      The number of defined variants

    • strategies object[]

      A list of activation strategies for the feature environment

    • Array [
    • id string

      A uuid for the feature strategy

    • name string required

      The name or type of strategy

    • title string nullable

      A descriptive title for the strategy

    • disabled boolean nullable

      A toggle to disable the strategy. defaults to false. Disabled strategies are not evaluated or returned to the SDKs

    • featureName string

      The name or feature the strategy is attached to

    • sortOrder number

      The order of the strategy in the list

    • segments number[]

      A list of segment ids attached to the strategy

    • constraints object[]

      A list of the constraints attached to the strategy. See https://docs.getunleash.io/reference/strategy-constraints

    • Array [
    • contextName string required

      The name of the context field that this constraint should apply to.

    • operator string required

      Possible values: [NOT_IN, IN, STR_ENDS_WITH, STR_STARTS_WITH, STR_CONTAINS, NUM_EQ, NUM_GT, NUM_GTE, NUM_LT, NUM_LTE, DATE_AFTER, DATE_BEFORE, SEMVER_EQ, SEMVER_GT, SEMVER_LT]

      The operator to use when evaluating this constraint. For more information about the various operators, refer to the strategy constraint operator documentation.

    • caseInsensitive boolean

      Default value: false

      Whether the operator should be case sensitive or not. Defaults to false (being case sensitive).

    • inverted boolean

      Default value: false

      Whether the result should be negated or not. If true, will turn a true result into a false result and vice versa.

    • values string[]

      The context values that should be used for constraint evaluation. Use this property instead of value for properties that accept multiple values.

    • value string

      The context value that should be used for constraint evaluation. Use this property instead of values for properties that only accept single values.

    • ]
    • variants object[]

      Strategy level variants

    • Array [
    • name string required

      The variant name. Must be unique for this feature toggle

    • weight integer required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is an integer between 0 and 1000. See the section on variant weights for more information

    • weightType string required

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000. Refer to the variant weight documentation.

    • stickiness string required

      The stickiness to use for distribution of this variant. Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • ]
    • parameters object

      A list of parameters for a strategy

    • property name* string
    • ]
    • variants object[]

      A list of variants for the feature environment

    • Array [
    • name string required

      The variants name. Is unique for this feature toggle

    • weight number required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is a number between 0 and 1000. See the section on variant weights for more information

    • weightType string

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000

    • stickiness string

      Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • overrides object[]

      Overrides assigning specific variants to specific users. The weighting system automatically assigns users to specific groups for you, but any overrides in this list will take precedence.

    • Array [
    • contextName string required

      The name of the context field used to determine overrides

    • values string[] required

      Which values that should be overriden

    • ]
    • ]
    • lastSeenAt date-time nullable

      The date when metrics where last collected for the feature environment

    • hasStrategies boolean

      Whether the feature has any strategies defined.

    • hasEnabledStrategies boolean

      Whether the feature has any enabled strategies defined.

    • ]
    • variants object[]deprecated

      The list of feature variants

    • Array [
    • name string required

      The variants name. Is unique for this feature toggle

    • weight number required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is a number between 0 and 1000. See the section on variant weights for more information

    • weightType string

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000

    • stickiness string

      Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • overrides object[]

      Overrides assigning specific variants to specific users. The weighting system automatically assigns users to specific groups for you, but any overrides in this list will take precedence.

    • Array [
    • contextName string required

      The name of the context field used to determine overrides

    • values string[] required

      Which values that should be overriden

    • ]
    • ]
    • strategies object[] deprecated

      This is a legacy field that will be deprecated

    • tags object[]nullable

      The list of feature tags

    • Array [
    • value string required

      Possible values: >= 2 characters and <= 50 characters

      The value of the tag

    • type string required

      Possible values: >= 2 characters and <= 50 characters

      Default value: simple

      The type of the tag

    • ]
    • children string[]

      The list of child feature names. This is an experimental field and may change.

    • dependencies object[]

      The list of parent dependencies. This is an experimental field and may change.

    • Array [
    • feature string required

      The name of the parent feature

    • enabled boolean

      Whether the parent feature is enabled or not

    • variants string[]

      The list of variants the parent feature should resolve to. Only valid when feature is enabled.

    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/update-feedback.html b/reference/api/unleash/update-feedback.html index 0b34725705..d69c5be20a 100644 --- a/reference/api/unleash/update-feedback.html +++ b/reference/api/unleash/update-feedback.html @@ -20,15 +20,15 @@ - - + +

    Update Unleash feedback

    PUT /api/admin/feedback/:id

    Updates the feedback with the provided ID. Only provided fields are updated. Fields left out are left untouched. Must be called with a token with an identifiable user (either from being sent from the UI or from using a PAT).

    Request

    Path Parameters

    • id string required

    Body

    required

    feedbackUpdateSchema

    • userId integer

      The ID of the user that gave the feedback.

    • neverShow boolean

      true if the user has asked never to see this feedback questionnaire again.

    • given date-time nullable

      When this feedback was given

    Responses

    feedbackResponseSchema

    Schema
    • userId integer

      The ID of the user that gave the feedback.

    • neverShow boolean

      true if the user has asked never to see this feedback questionnaire again.

    • given date-time nullable

      When this feedback was given

    • feedbackId string

      The name of the feedback session

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/update-group.html b/reference/api/unleash/update-group.html index 725444931a..b870a3200b 100644 --- a/reference/api/unleash/update-group.html +++ b/reference/api/unleash/update-group.html @@ -20,15 +20,15 @@ - - + +

    Update a group

    PUT /api/admin/groups/:groupId

    Update existing user group by group id. It overrides previous group details.

    Request

    Path Parameters

    • groupId string required

    Body

    required

    createGroupSchema

    • name string required

      The name of the group

    • description string nullable

      A custom description of the group

    • mappingsSSO string[]

      A list of SSO groups that should map to this Unleash group

    • rootRole number nullable

      A role id that is used as the root role for all users in this group. This can be either the id of the Viewer, Editor or Admin role.

    • users object[]

      A list of users belonging to this group

    • Array [
    • user objectrequired

      A minimal user object

    • id integer required

      The user id

    • ]
    Responses

    groupSchema

    Schema
    • id integer

      The group id

    • name string required

      The name of the group

    • description string nullable

      A custom description of the group

    • mappingsSSO string[]

      A list of SSO groups that should map to this Unleash group

    • rootRole number nullable

      A role id that is used as the root role for all users in this group. This can be either the id of the Viewer, Editor or Admin role.

    • createdBy string nullable

      A user who created this group

    • createdAt date-time nullable

      When was this group created

    • users object[]

      A list of users belonging to this group

    • Array [
    • joinedAt date-time

      The date when the user joined the group

    • createdBy string nullable

      The username of the user who added this user to this group

    • user objectrequired

      An Unleash user

    • id integer required

      The user id

    • isAPI boolean deprecated

      (Deprecated): Used internally to know which operations the user should be allowed to perform

    • name string nullable

      Name of the user

    • email string

      Email of the user

    • username string nullable

      A unique username for the user

    • imageUrl string

      URL used for the userprofile image

    • inviteLink string

      If the user is actively inviting other users, this is the link that can be shared with other users

    • loginAttempts integer

      How many unsuccessful attempts at logging in has the user made

    • emailSent boolean

      Is the welcome email sent to the user or not

    • rootRole integer

      Which root role this user is assigned

    • seenAt date-time nullable

      The last time this user logged in

    • createdAt date-time

      The user was created at this time

    • accountType string

      A user is either an actual User or a Service Account

    • permissions string[]

      Deprecated

    • ]
    • projects string[]

      A list of projects where this group is used

    • userCount integer

      The number of users that belong to this group

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/update-license.html b/reference/api/unleash/update-license.html index 5470f30864..833a1bacd2 100644 --- a/reference/api/unleash/update-license.html +++ b/reference/api/unleash/update-license.html @@ -20,15 +20,15 @@ - - + +

    Set a new Unleash license.

    POST /api/admin/license

    Set a new Unleash license. Only available for self-hosted Enterprise customers.

    Request

    Body

    required

    licenseUpdateSchema

    • token string required

      The actual license token.

    Responses

    The resource was successfully created.

    Response Headers
    • location string

      The location of the newly created resource.

    Schema
    • token string required

      The actual license token.

    • customer string

      Name of the customer that owns the license. This is the name of the company that purchased the license.

    • plan string

      Name of plan that the license is for.

    • seats number

      Number of seats in the license.

    • expireAt date-time

      Date when the license expires.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/update-project-change-request-config.html b/reference/api/unleash/update-project-change-request-config.html index 7074abc2d7..f989962aff 100644 --- a/reference/api/unleash/update-project-change-request-config.html +++ b/reference/api/unleash/update-project-change-request-config.html @@ -20,15 +20,15 @@ - - + +

    Updates change request configuration for an environment in the project

    PUT /api/admin/projects/:projectId/environments/:environment/change-requests/config

    This endpoint will change the change request configuration for a given environment, set it to either on/off and optionally configure the number of approvals needed. The minimum number of approvals is 1 and the maximum number is 10. If you provide a number higher than 10 or lower than 1, Unleash will clamp it to the allowed range.

    Request

    Path Parameters

    • projectId string required
    • environment string required

    Body

    required

    updateChangeRequestEnvironmentConfigSchema

    • changeRequestsEnabled boolean required

      true if change requests should be enabled, otherwise false.

    • requiredApprovals integer

      The number of approvals required before a change request can be applied in this environment.

    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/update-project-enterprise-settings.html b/reference/api/unleash/update-project-enterprise-settings.html index e3fe1a5541..08ddea168a 100644 --- a/reference/api/unleash/update-project-enterprise-settings.html +++ b/reference/api/unleash/update-project-enterprise-settings.html @@ -20,15 +20,15 @@ - - + +

    Update project enterprise settings

    PUT /api/admin/projects/:projectId/settings

    Update project enterprise settings with new values. Any fields not provided are ignored.

    Request

    Path Parameters

    • projectId string required

    Body

    required

    updateProjectEnterpriseSettingsSchema

    • mode string

      Possible values: [open, protected, private]

      A mode of the project affecting what actions are possible in this project

    • featureNaming object

      Create a feature naming pattern

    • pattern string nullable required

      A JavaScript regular expression pattern, without the start and end delimiters. Optional flags are not allowed.

    • example string nullable

      An example of a feature name that matches the pattern. Must itself match the pattern supplied.

    • description string nullable

      A description of the pattern in a human-readable format. Will be shown to users when they create a new feature flag.

    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/update-project.html b/reference/api/unleash/update-project.html index 49d38fc593..956eb73b6e 100644 --- a/reference/api/unleash/update-project.html +++ b/reference/api/unleash/update-project.html @@ -20,15 +20,15 @@ - - + +

    Update project

    PUT /api/admin/projects/:projectId

    Update a project with new configuration. Any fields not provided are ignored.

    Request

    Path Parameters

    • projectId string required

    Body

    required

    updateProjectSchema

    • name string required

      The new name of the project

    • description string

      A new description for the project

    • mode string

      Possible values: [open, protected, private]

      A mode of the project affecting what actions are possible in this project

    • defaultStickiness string

      A default stickiness for the project affecting the default stickiness value for variants and Gradual Rollout strategy

    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/update-public-signup-token.html b/reference/api/unleash/update-public-signup-token.html index 64567563b0..f14e89de47 100644 --- a/reference/api/unleash/update-public-signup-token.html +++ b/reference/api/unleash/update-public-signup-token.html @@ -20,15 +20,15 @@ - - + +

    Update a public signup token

    PUT /api/admin/invite-link/tokens/:token

    Update information about a specific token. The :token part of the URL should be the token's secret.

    Request

    Path Parameters

    • token string required

    Body

    required

    publicSignupTokenUpdateSchema

    • expiresAt date-time

      The token's expiration date.

    • enabled boolean

      Whether the token is active or not.

    Responses

    publicSignupTokenSchema

    Schema
    • secret string required

      The actual value of the token. This is the part that is used by Unleash to create an invite link

    • url string nullable required

      The public signup link for the token. Users who follow this link will be taken to a signup page where they can create an Unleash user.

    • name string required

      The token's name. Only for displaying in the UI

    • enabled boolean required

      Whether the token is active. This property will always be false for a token that has expired.

    • expiresAt date-time required

      The time when the token will expire.

    • createdAt date-time required

      When the token was created.

    • createdBy string nullable required

      The creator's email or username

    • users object[]nullable

      Array of users that have signed up using the token.

    • Array [
    • id integer required

      The user id

    • isAPI boolean deprecated

      (Deprecated): Used internally to know which operations the user should be allowed to perform

    • name string nullable

      Name of the user

    • email string

      Email of the user

    • username string nullable

      A unique username for the user

    • imageUrl string

      URL used for the userprofile image

    • inviteLink string

      If the user is actively inviting other users, this is the link that can be shared with other users

    • loginAttempts integer

      How many unsuccessful attempts at logging in has the user made

    • emailSent boolean

      Is the welcome email sent to the user or not

    • rootRole integer

      Which root role this user is assigned

    • seenAt date-time nullable

      The last time this user logged in

    • createdAt date-time

      The user was created at this time

    • accountType string

      A user is either an actual User or a Service Account

    • permissions string[]

      Deprecated

    • ]
    • role objectrequired

      A role holds permissions to allow Unleash to decide what actions a role holder is allowed to perform

    • id integer required

      The role id

    • type string required

      A role can either be a global root role (applies to all projects) or a project role

    • name string required

      The name of the role

    • description string

      A more detailed description of the role and what use it's intended for

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/update-role.html b/reference/api/unleash/update-role.html index ae69ddfe30..e69e22b325 100644 --- a/reference/api/unleash/update-role.html +++ b/reference/api/unleash/update-role.html @@ -20,15 +20,15 @@ - - + +

    Update a role

    PUT /api/admin/roles/:roleId

    Update a custom role by role id

    Request

    Path Parameters

    • roleId string required

    Body

    required

    createRoleWithPermissionsSchema

      anyOf
    • name string required

      The name of the custom role

    • description string

      A more detailed description of the custom role and what use it's intended for

    • type string

      Possible values: [root-custom, custom]

      Custom root roles (type=root-custom) are root roles with a custom set of permissions. Custom project roles (type=custom) contain a specific set of permissions for project resources.

    • permissions object[]

      A list of permissions assigned to this role

    • Array [
    • name string required

      The name of the permission

    • environment string

      The environments of the permission if the permission is environment specific

    • ]
    Responses

    roleWithVersionSchema

    Schema
    • version integer required

      Possible values: >= 1

      The version of this schema

    • roles objectrequired

      A role holds permissions to allow Unleash to decide what actions a role holder is allowed to perform

    • id integer required

      The role id

    • type string required

      A role can either be a global root role (applies to all projects) or a project role

    • name string required

      The name of the role

    • description string

      A more detailed description of the role and what use it's intended for

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/update-segment.html b/reference/api/unleash/update-segment.html index d21d5232f1..9d1c943b6d 100644 --- a/reference/api/unleash/update-segment.html +++ b/reference/api/unleash/update-segment.html @@ -20,15 +20,15 @@ - - + +

    Update segment by id

    PUT /api/admin/segments/:id

    Updates the content of the segment with the provided payload. Any fields not specified will be left untouched.

    Request

    Path Parameters

    • id string required
    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/update-service-account.html b/reference/api/unleash/update-service-account.html index 3882888c40..3acc416c28 100644 --- a/reference/api/unleash/update-service-account.html +++ b/reference/api/unleash/update-service-account.html @@ -20,15 +20,15 @@ - - + +

    Update a service account.

    PUT /api/admin/service-account/:id

    Updates an existing service account identified by its id.

    Request

    Path Parameters

    • id string required

    Body

    required

    updateServiceAccountSchema

    • name string

      The name of the service account

    • rootRole integer

      The id of the root role for the service account

    • property name* any

      Describes the properties required to update a service account

    Responses

    serviceAccountSchema

    Schema
    • id number required

      The service account id

    • isAPI boolean deprecated

      Deprecated: for internal use only, should not be exposed through the API

    • name string

      The name of the service account

    • email string deprecated

      Deprecated: service accounts don't have emails associated with them

    • username string

      The service account username

    • imageUrl string

      The service account image url

    • inviteLink string deprecated

      Deprecated: service accounts cannot be invited via an invitation link

    • loginAttempts number deprecated

      Deprecated: service accounts cannot log in to Unleash

    • emailSent boolean deprecated

      Deprecated: internal use only

    • rootRole integer

      The root role id associated with the service account

    • seenAt date-time nullable deprecated

      Deprecated. This property is always null. To find out when a service account was last seen, check its tokens list and refer to each token's lastSeen property instead.

    • createdAt date-time

      The service account creation date

    • tokens object[]

      The list of tokens associated with the service account

    • Array [
    • id integer

      Possible values: >= 1

      The unique identification number for this Personal Access Token. (This property is set by Unleash when the token is created and cannot be set manually: if you provide a value when creating a PAT, Unleash will ignore it.)

    • secret string

      The token used for authentication. (This property is set by Unleash when the token is created and cannot be set manually: if you provide a value when creating a PAT, Unleash will ignore it.)

    • expiresAt date-time

      The token's expiration date.

    • createdAt date-time

      When the token was created. (This property is set by Unleash when the token is created and cannot be set manually: if you provide a value when creating a PAT, Unleash will ignore it.)

    • seenAt date-time nullable

      When the token was last seen/used to authenticate with. null if it has not been used yet. (This property is set by Unleash when the token is created and cannot be set manually: if you provide a value when creating a PAT, Unleash will ignore it.)

    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/update-sort-order.html b/reference/api/unleash/update-sort-order.html index e0f878e787..d491ca614c 100644 --- a/reference/api/unleash/update-sort-order.html +++ b/reference/api/unleash/update-sort-order.html @@ -20,15 +20,15 @@ - - + +

    Update environment sort orders

    PUT /api/admin/environments/sort-order

    Updates sort orders for the named environments. Environments not specified are unaffected.

    Request

    Body

    required

    sortOrderSchema

    • property name* integer

      Sort order for the object whose ID is the key used for this property.

    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/update-splash-settings.html b/reference/api/unleash/update-splash-settings.html index 8a315db6e9..4f176fe241 100644 --- a/reference/api/unleash/update-splash-settings.html +++ b/reference/api/unleash/update-splash-settings.html @@ -20,15 +20,15 @@ - - + +

    Update splash settings

    POST /api/admin/splash/:id

    This operation updates splash settings for a user, indicating that they have seen a particualar splash screen.

    Request

    Path Parameters

    • id string required
    Responses

    splashResponseSchema

    Schema
    • userId integer required

      The ID of the user that was shown the splash screen.

    • splashId string required

      The ID of the splash screen that was shown.

    • seen boolean required

      Indicates whether the user has seen the splash screen or not.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/update-strategy.html b/reference/api/unleash/update-strategy.html index 5b7e92e1a3..e66323b1cc 100644 --- a/reference/api/unleash/update-strategy.html +++ b/reference/api/unleash/update-strategy.html @@ -20,15 +20,15 @@ - - + +

    Update a strategy type

    PUT /api/admin/strategies/:name

    Updates the specified strategy type. Any properties not specified in the request body are left untouched.

    Request

    Path Parameters

    • name string required

    Body

    required

    updateStrategySchema

    • description string

      A description of the strategy type.

    • parameters object[]required

      The parameter list lets you pass arguments to your custom activation strategy. These will be made available to your custom strategy implementation.

    • Array [
    • name string required

      The name of the parameter

    • type string required

      Possible values: [string, percentage, list, number, boolean]

    • description string

      A description of this strategy parameter. Use this to indicate to the users what the parameter does.

    • required boolean

      Whether this parameter must be configured when using the strategy. Defaults to false

    • ]
    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/update-tag-type.html b/reference/api/unleash/update-tag-type.html index 371d66e8d9..996b7f9ef9 100644 --- a/reference/api/unleash/update-tag-type.html +++ b/reference/api/unleash/update-tag-type.html @@ -20,15 +20,15 @@ - - + +

    Update a tag type

    PUT /api/admin/tag-types/:name

    Update the configuration for the specified tag type.

    Request

    Path Parameters

    • name string required

    Body

    required

    updateTagTypeSchema

    • description string

      The description of the tag type.

    • icon string

      The icon of the tag type.

    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/update-tags.html b/reference/api/unleash/update-tags.html index 423a9b6aeb..bef0f34365 100644 --- a/reference/api/unleash/update-tags.html +++ b/reference/api/unleash/update-tags.html @@ -20,15 +20,15 @@ - - + +

    Updates multiple tags for a feature.

    PUT /api/admin/features/:featureName/tags

    Receives a list of tags to add and a list of tags to remove that are mandatory but can be empty. All tags under addedTags are first added to the feature and then all tags under removedTags are removed from the feature.

    Request

    Path Parameters

    • featureName string required

    Body

    required

    updateTagsSchema

    • addedTags object[]required

      Tags to add to the feature.

    • Array [
    • value string required

      Possible values: >= 2 characters and <= 50 characters

      The value of the tag

    • type string required

      Possible values: >= 2 characters and <= 50 characters

      Default value: simple

      The type of the tag

    • ]
    • removedTags object[]required

      Tags to remove from the feature.

    • Array [
    • value string required

      Possible values: >= 2 characters and <= 50 characters

      The value of the tag

    • type string required

      Possible values: >= 2 characters and <= 50 characters

      Default value: simple

      The type of the tag

    • ]
    Responses

    The resource was successfully created.

    Response Headers
    • location string

      The location of the newly created resource.

    Schema
    • version integer required

      The version of the schema used to model the tags.

    • tags object[]required

      A list of tags.

    • Array [
    • value string required

      Possible values: >= 2 characters and <= 50 characters

      The value of the tag

    • type string required

      Possible values: >= 2 characters and <= 50 characters

      Default value: simple

      The type of the tag

    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/update-user.html b/reference/api/unleash/update-user.html index 1af0856d4d..03d77e78cf 100644 --- a/reference/api/unleash/update-user.html +++ b/reference/api/unleash/update-user.html @@ -20,15 +20,15 @@ - - + +

    Update a user

    PUT /api/admin/user-admin/:id

    Only the explicitly specified fields get updated.

    Request

    Path Parameters

    • id string required

    Body

    required

    updateUserSchema

    • email string

      The user's email address. Must be provided if username is not provided.

    • name string

      The user's name (not the user's username).

    • rootRole object

      The role to assign to the user. Can be either the role's ID or its unique name.

      oneOf
    • integer
    Responses

    createUserResponseSchema

    Schema
    • id integer required

      The user id

    • isAPI boolean deprecated

      (Deprecated): Used internally to know which operations the user should be allowed to perform

    • name string nullable

      Name of the user

    • email string

      Email of the user

    • username string nullable

      A unique username for the user

    • imageUrl string

      URL used for the userprofile image

    • inviteLink string

      If the user is actively inviting other users, this is the link that can be shared with other users

    • loginAttempts integer

      How many unsuccessful attempts at logging in has the user made

    • emailSent boolean

      Is the welcome email sent to the user or not

    • rootRole object

      Which root role this user is assigned. Usually a numeric role ID, but can be a string when returning newly created user with an explicit string role.

      oneOf
    • integer
    • seenAt date-time nullable

      The last time this user logged in

    • createdAt date-time

      The user was created at this time

    • accountType string

      A user is either an actual User or a Service Account

    • permissions string[]

      Deprecated

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/users.html b/reference/api/unleash/users.html index 2319f17201..5f716c8855 100644 --- a/reference/api/unleash/users.html +++ b/reference/api/unleash/users.html @@ -20,15 +20,15 @@ - - + +

    Users

    Manage users and passwords.

    - - + + \ No newline at end of file diff --git a/reference/api/unleash/validate-archive-features.html b/reference/api/unleash/validate-archive-features.html index 32d2774ed4..24f352e8fe 100644 --- a/reference/api/unleash/validate-archive-features.html +++ b/reference/api/unleash/validate-archive-features.html @@ -20,15 +20,15 @@ - - + +

    Validates archive features

    POST /api/admin/projects/:projectId/archive/validate

    This endpoint return info about the archive features impact.

    Request

    Path Parameters

    • projectId string required

    Body

    required

    batchFeaturesSchema

    • features string[] required

      List of feature toggle names

    Responses

    validateArchiveFeaturesSchema

    Schema
    • parentsWithChildFeatures string[] required

      List of parent features that would orphan child features that are not part of the archive operation

    • hasDeletedDependencies boolean required

      Whether any dependencies will be deleted as part of archive

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/validate-constraint.html b/reference/api/unleash/validate-constraint.html index 2a7d41b82b..10dfc11adb 100644 --- a/reference/api/unleash/validate-constraint.html +++ b/reference/api/unleash/validate-constraint.html @@ -20,15 +20,15 @@ - - + +

    Validate constraint

    POST /api/admin/constraints/validate

    Validates a constraint definition. Checks whether the context field exists and whether the applied configuration is valid. Additional properties are not allowed on data objects that you send to this endpoint.

    Request

    Body

    required

    constraintSchema

    • contextName string required

      The name of the context field that this constraint should apply to.

    • operator string required

      Possible values: [NOT_IN, IN, STR_ENDS_WITH, STR_STARTS_WITH, STR_CONTAINS, NUM_EQ, NUM_GT, NUM_GTE, NUM_LT, NUM_LTE, DATE_AFTER, DATE_BEFORE, SEMVER_EQ, SEMVER_GT, SEMVER_LT]

      The operator to use when evaluating this constraint. For more information about the various operators, refer to the strategy constraint operator documentation.

    • caseInsensitive boolean

      Default value: false

      Whether the operator should be case sensitive or not. Defaults to false (being case sensitive).

    • inverted boolean

      Default value: false

      Whether the result should be negated or not. If true, will turn a true result into a false result and vice versa.

    • values string[]

      The context values that should be used for constraint evaluation. Use this property instead of value for properties that accept multiple values.

    • value string

      The context value that should be used for constraint evaluation. Use this property instead of values for properties that only accept single values.

    Responses

    The constraint is valid

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/validate-environment-name.html b/reference/api/unleash/validate-environment-name.html index 6cce8f1c7f..06faae92e4 100644 --- a/reference/api/unleash/validate-environment-name.html +++ b/reference/api/unleash/validate-environment-name.html @@ -20,15 +20,15 @@ - - + +

    Validates if an environment name exists

    POST /api/admin/environments/validate

    Uses the name provided in the body of the request to validate if the given name exists or not

    Request

    Body

    required

    nameSchema

    • name string required

      The name of the represented object.

    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/validate-feature.html b/reference/api/unleash/validate-feature.html index 7fe2eedbb1..2736ca85d3 100644 --- a/reference/api/unleash/validate-feature.html +++ b/reference/api/unleash/validate-feature.html @@ -20,15 +20,15 @@ - - + +

    Validate a feature toggle name.

    POST /api/admin/features/validate

    Validates a feature toggle name: checks whether the name is URL-friendly and whether a feature with the given name already exists. Returns 200 if the feature name is compliant and unused.

    Request

    Body

    required

    validateFeatureSchema

    • name string required

      The feature name to validate.

    • projectId string nullable

      The id of the project that the feature flag will belong to. If the target project has a feature naming pattern defined, the name will be validated against that pattern.

    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/validate-import.html b/reference/api/unleash/validate-import.html index bdbd2fa249..62b1e3cad4 100644 --- a/reference/api/unleash/validate-import.html +++ b/reference/api/unleash/validate-import.html @@ -20,15 +20,15 @@ - - + +

    Validate feature import data

    POST /api/admin/features-batch/validate

    Validates a feature toggle data set. Checks whether the data can be imported into the specified project and environment. The returned value is an object that contains errors, warnings, and permissions required to perform the import, as described in the import documentation.

    Request

    Body

    required

    importTogglesSchema

    • project string required

      The exported project

    • environment string required

      The exported environment

    • data objectrequired

      The result of the export operation, providing you with the feature toggle definitions, strategy definitions and the rest of the elements relevant to the features (tags, environments etc.)

    • features object[]required

      All the exported features.

    • Array [
    • name string required

      Unique feature name

    • type string

      Type of the toggle e.g. experiment, kill-switch, release, operational, permission

    • description string nullable

      Detailed description of the feature

    • archived boolean

      true if the feature is archived

    • project string

      Name of the project the feature belongs to

    • enabled boolean

      true if the feature is enabled, otherwise false.

    • stale boolean

      true if the feature is stale based on the age and feature type, otherwise false.

    • favorite boolean

      true if the feature was favorited, otherwise false.

    • impressionData boolean

      true if the impression data collection is enabled for the feature, otherwise false.

    • createdAt date-time nullable

      The date the feature was created

    • archivedAt date-time nullable

      The date the feature was archived

    • lastSeenAt date-time nullable deprecated

      The date when metrics where last collected for the feature. This field is deprecated, use the one in featureEnvironmentSchema

    • environments object[]

      The list of environments where the feature can be used

    • Array [
    • name string required

      The name of the environment

    • featureName string

      The name of the feature

    • environment string

      The name of the environment

    • type string

      The type of the environment

    • enabled boolean required

      true if the feature is enabled for the environment, otherwise false.

    • sortOrder number

      The sort order of the feature environment in the feature environments list

    • variantCount number

      The number of defined variants

    • strategies object[]

      A list of activation strategies for the feature environment

    • Array [
    • id string

      A uuid for the feature strategy

    • name string required

      The name or type of strategy

    • title string nullable

      A descriptive title for the strategy

    • disabled boolean nullable

      A toggle to disable the strategy. defaults to false. Disabled strategies are not evaluated or returned to the SDKs

    • featureName string

      The name or feature the strategy is attached to

    • sortOrder number

      The order of the strategy in the list

    • segments number[]

      A list of segment ids attached to the strategy

    • constraints object[]

      A list of the constraints attached to the strategy. See https://docs.getunleash.io/reference/strategy-constraints

    • Array [
    • contextName string required

      The name of the context field that this constraint should apply to.

    • operator string required

      Possible values: [NOT_IN, IN, STR_ENDS_WITH, STR_STARTS_WITH, STR_CONTAINS, NUM_EQ, NUM_GT, NUM_GTE, NUM_LT, NUM_LTE, DATE_AFTER, DATE_BEFORE, SEMVER_EQ, SEMVER_GT, SEMVER_LT]

      The operator to use when evaluating this constraint. For more information about the various operators, refer to the strategy constraint operator documentation.

    • caseInsensitive boolean

      Default value: false

      Whether the operator should be case sensitive or not. Defaults to false (being case sensitive).

    • inverted boolean

      Default value: false

      Whether the result should be negated or not. If true, will turn a true result into a false result and vice versa.

    • values string[]

      The context values that should be used for constraint evaluation. Use this property instead of value for properties that accept multiple values.

    • value string

      The context value that should be used for constraint evaluation. Use this property instead of values for properties that only accept single values.

    • ]
    • variants object[]

      Strategy level variants

    • Array [
    • name string required

      The variant name. Must be unique for this feature toggle

    • weight integer required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is an integer between 0 and 1000. See the section on variant weights for more information

    • weightType string required

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000. Refer to the variant weight documentation.

    • stickiness string required

      The stickiness to use for distribution of this variant. Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • ]
    • parameters object

      A list of parameters for a strategy

    • property name* string
    • ]
    • variants object[]

      A list of variants for the feature environment

    • Array [
    • name string required

      The variants name. Is unique for this feature toggle

    • weight number required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is a number between 0 and 1000. See the section on variant weights for more information

    • weightType string

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000

    • stickiness string

      Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • overrides object[]

      Overrides assigning specific variants to specific users. The weighting system automatically assigns users to specific groups for you, but any overrides in this list will take precedence.

    • Array [
    • contextName string required

      The name of the context field used to determine overrides

    • values string[] required

      Which values that should be overriden

    • ]
    • ]
    • lastSeenAt date-time nullable

      The date when metrics where last collected for the feature environment

    • hasStrategies boolean

      Whether the feature has any strategies defined.

    • hasEnabledStrategies boolean

      Whether the feature has any enabled strategies defined.

    • ]
    • variants object[]deprecated

      The list of feature variants

    • Array [
    • name string required

      The variants name. Is unique for this feature toggle

    • weight number required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is a number between 0 and 1000. See the section on variant weights for more information

    • weightType string

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000

    • stickiness string

      Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • overrides object[]

      Overrides assigning specific variants to specific users. The weighting system automatically assigns users to specific groups for you, but any overrides in this list will take precedence.

    • Array [
    • contextName string required

      The name of the context field used to determine overrides

    • values string[] required

      Which values that should be overriden

    • ]
    • ]
    • strategies object[] deprecated

      This is a legacy field that will be deprecated

    • tags object[]nullable

      The list of feature tags

    • Array [
    • value string required

      Possible values: >= 2 characters and <= 50 characters

      The value of the tag

    • type string required

      Possible values: >= 2 characters and <= 50 characters

      Default value: simple

      The type of the tag

    • ]
    • children string[]

      The list of child feature names. This is an experimental field and may change.

    • dependencies object[]

      The list of parent dependencies. This is an experimental field and may change.

    • Array [
    • feature string required

      The name of the parent feature

    • enabled boolean

      Whether the parent feature is enabled or not

    • variants string[]

      The list of variants the parent feature should resolve to. Only valid when feature is enabled.

    • ]
    • ]
    • featureStrategies object[]required

      All strategy instances that are used by the exported features in the features list.

    • Array [
    • id string

      A uuid for the feature strategy

    • name string required

      The name or type of strategy

    • title string nullable

      A descriptive title for the strategy

    • disabled boolean nullable

      A toggle to disable the strategy. defaults to false. Disabled strategies are not evaluated or returned to the SDKs

    • featureName string

      The name or feature the strategy is attached to

    • sortOrder number

      The order of the strategy in the list

    • segments number[]

      A list of segment ids attached to the strategy

    • constraints object[]

      A list of the constraints attached to the strategy. See https://docs.getunleash.io/reference/strategy-constraints

    • Array [
    • contextName string required

      The name of the context field that this constraint should apply to.

    • operator string required

      Possible values: [NOT_IN, IN, STR_ENDS_WITH, STR_STARTS_WITH, STR_CONTAINS, NUM_EQ, NUM_GT, NUM_GTE, NUM_LT, NUM_LTE, DATE_AFTER, DATE_BEFORE, SEMVER_EQ, SEMVER_GT, SEMVER_LT]

      The operator to use when evaluating this constraint. For more information about the various operators, refer to the strategy constraint operator documentation.

    • caseInsensitive boolean

      Default value: false

      Whether the operator should be case sensitive or not. Defaults to false (being case sensitive).

    • inverted boolean

      Default value: false

      Whether the result should be negated or not. If true, will turn a true result into a false result and vice versa.

    • values string[]

      The context values that should be used for constraint evaluation. Use this property instead of value for properties that accept multiple values.

    • value string

      The context value that should be used for constraint evaluation. Use this property instead of values for properties that only accept single values.

    • ]
    • variants object[]

      Strategy level variants

    • Array [
    • name string required

      The variant name. Must be unique for this feature toggle

    • weight integer required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is an integer between 0 and 1000. See the section on variant weights for more information

    • weightType string required

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000. Refer to the variant weight documentation.

    • stickiness string required

      The stickiness to use for distribution of this variant. Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • ]
    • parameters object

      A list of parameters for a strategy

    • property name* string
    • ]
    • featureEnvironments object[]

      Environment-specific configuration for all the features in the features list. Includes data such as whether the feature is enabled in the selected export environment, whether there are any variants assigned, etc.

    • Array [
    • name string required

      The name of the environment

    • featureName string

      The name of the feature

    • environment string

      The name of the environment

    • type string

      The type of the environment

    • enabled boolean required

      true if the feature is enabled for the environment, otherwise false.

    • sortOrder number

      The sort order of the feature environment in the feature environments list

    • variantCount number

      The number of defined variants

    • strategies object[]

      A list of activation strategies for the feature environment

    • Array [
    • id string

      A uuid for the feature strategy

    • name string required

      The name or type of strategy

    • title string nullable

      A descriptive title for the strategy

    • disabled boolean nullable

      A toggle to disable the strategy. defaults to false. Disabled strategies are not evaluated or returned to the SDKs

    • featureName string

      The name or feature the strategy is attached to

    • sortOrder number

      The order of the strategy in the list

    • segments number[]

      A list of segment ids attached to the strategy

    • constraints object[]

      A list of the constraints attached to the strategy. See https://docs.getunleash.io/reference/strategy-constraints

    • Array [
    • contextName string required

      The name of the context field that this constraint should apply to.

    • operator string required

      Possible values: [NOT_IN, IN, STR_ENDS_WITH, STR_STARTS_WITH, STR_CONTAINS, NUM_EQ, NUM_GT, NUM_GTE, NUM_LT, NUM_LTE, DATE_AFTER, DATE_BEFORE, SEMVER_EQ, SEMVER_GT, SEMVER_LT]

      The operator to use when evaluating this constraint. For more information about the various operators, refer to the strategy constraint operator documentation.

    • caseInsensitive boolean

      Default value: false

      Whether the operator should be case sensitive or not. Defaults to false (being case sensitive).

    • inverted boolean

      Default value: false

      Whether the result should be negated or not. If true, will turn a true result into a false result and vice versa.

    • values string[]

      The context values that should be used for constraint evaluation. Use this property instead of value for properties that accept multiple values.

    • value string

      The context value that should be used for constraint evaluation. Use this property instead of values for properties that only accept single values.

    • ]
    • variants object[]

      Strategy level variants

    • Array [
    • name string required

      The variant name. Must be unique for this feature toggle

    • weight integer required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is an integer between 0 and 1000. See the section on variant weights for more information

    • weightType string required

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000. Refer to the variant weight documentation.

    • stickiness string required

      The stickiness to use for distribution of this variant. Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • ]
    • parameters object

      A list of parameters for a strategy

    • property name* string
    • ]
    • variants object[]

      A list of variants for the feature environment

    • Array [
    • name string required

      The variants name. Is unique for this feature toggle

    • weight number required

      Possible values: <= 1000

      The weight is the likelihood of any one user getting this variant. It is a number between 0 and 1000. See the section on variant weights for more information

    • weightType string

      Possible values: [variable, fix]

      Set to fix if this variant must have exactly the weight allocated to it. If the type is variable, the weight will adjust so that the total weight of all variants adds up to 1000

    • stickiness string

      Stickiness is how Unleash guarantees that the same user gets the same variant every time

    • payload object

      Extra data configured for this variant

    • type string required

      Possible values: [json, csv, string, number]

      The type of the value. Commonly used types are string, number, json and csv.

    • value string required

      The actual value of payload

    • overrides object[]

      Overrides assigning specific variants to specific users. The weighting system automatically assigns users to specific groups for you, but any overrides in this list will take precedence.

    • Array [
    • contextName string required

      The name of the context field used to determine overrides

    • values string[] required

      Which values that should be overriden

    • ]
    • ]
    • lastSeenAt date-time nullable

      The date when metrics where last collected for the feature environment

    • hasStrategies boolean

      Whether the feature has any strategies defined.

    • hasEnabledStrategies boolean

      Whether the feature has any enabled strategies defined.

    • ]
    • contextFields object[]

      A list of all the context fields that are in use by any of the strategies in the featureStrategies list.

    • Array [
    • name string required

      The name of the context field

    • description string nullable

      The description of the context field.

    • stickiness boolean

      Does this context field support being used for stickiness calculations

    • sortOrder integer

      Used when sorting a list of context fields. Is also used as a tiebreaker if a list of context fields is sorted alphabetically.

    • createdAt date-time nullable

      When this context field was created

    • usedInFeatures integer nullable

      Number of projects where this context field is used in

    • usedInProjects integer nullable

      Number of projects where this context field is used in

    • legalValues object[]

      Allowed values for this context field schema. Can be used to narrow down accepted input

    • Array [
    • value string required

      The valid value

    • description string

      Describes this specific legal value

    • ]
    • ]
    • featureTags object[]

      A list of all the tags that have been applied to any of the features in the features list.

    • Array [
    • featureName string required

      The name of the feature this tag is applied to

    • tagType string

      The [type](https://docs.getunleash.io/reference/tags#tag-types tag types) of the tag

    • tagValue string required

      The value of the tag

    • type string deprecated

      The [type](https://docs.getunleash.io/reference/tags#tag-types tag types) of the tag. This property is deprecated and will be removed in a future version of Unleash. Superseded by the tagType property.

    • value string deprecated

      The value of the tag. This property is deprecated and will be removed in a future version of Unleash. Superseded by the tagValue property.

    • createdByUserId number nullable

      The id of the user who created this tag

    • ]
    • segments object[]

      A list of all the segments that are used by the strategies in the featureStrategies list.

    • Array [
    • id number required
    • name string required
    • ]
    • tagTypes object[]required

      A list of all of the tag types that are used in the featureTags list.

    • Array [
    • name string required

      The name of the tag type.

    • description string

      The description of the tag type.

    • icon string nullable

      The icon of the tag type.

    • ]
    • dependencies object[]

      A list of all the dependencies for features in features list.

    • Array [
    • feature string required

      The name of the child feature.

    • dependencies object[]required

      List of parent features for the child feature

    • Array [
    • feature string required

      The name of the feature we depend on.

    • enabled boolean

      Whether the parent feature should be enabled. When false variants are ignored. true by default.

    • variants string[]

      The list of variants the parent feature should resolve to. Leave empty when you only want to check the enabled status.

    • ]
    • ]
    Responses

    importTogglesValidateSchema

    Schema
    • errors object[]required

      A list of errors that prevent the provided data from being successfully imported.

    • Array [
    • message string required

      The validation error message

    • affectedItems string[] required

      The items affected by this error message

    • ]
    • warnings object[]required

      A list of warnings related to the provided data.

    • Array [
    • message string required

      The validation error message

    • affectedItems string[] required

      The items affected by this error message

    • ]
    • permissions object[]

      Any additional permissions required to import the data. If the list is empty, you require no additional permissions beyond what your user already has.

    • Array [
    • message string required

      The validation error message

    • affectedItems string[] required

      The items affected by this error message

    • ]
    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/validate-password.html b/reference/api/unleash/validate-password.html index 87a4a9bc95..bbc95498bc 100644 --- a/reference/api/unleash/validate-password.html +++ b/reference/api/unleash/validate-password.html @@ -20,15 +20,15 @@ - - + +

    Validates password

    POST /auth/reset/validate-password

    Verifies that the password adheres to the Unleash password guidelines

    Request

    Body

    required

    validatePasswordSchema

    • password string required

      The password to validate

    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/validate-project.html b/reference/api/unleash/validate-project.html index 7dfcd1a3b6..38cdbf2e63 100644 --- a/reference/api/unleash/validate-project.html +++ b/reference/api/unleash/validate-project.html @@ -20,15 +20,15 @@ - - + +

    Validate project ID

    POST /api/admin/projects/validate

    Validate a project ID against Unleash's rules

    Request

    Body

    required

    validateProjectSchema

    • id string required

      The project ID to validate

    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/validate-public-signup-token.html b/reference/api/unleash/validate-public-signup-token.html index cf5a1092dd..4ee3cd667b 100644 --- a/reference/api/unleash/validate-public-signup-token.html +++ b/reference/api/unleash/validate-public-signup-token.html @@ -20,15 +20,15 @@ - - + +

    Validate signup token

    GET /invite/:token/validate

    Check whether the provided public sign-up token exists, has not expired and is enabled

    Request

    Path Parameters

    • token string required
    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/validate-role.html b/reference/api/unleash/validate-role.html index f3a773040f..da6e3bab5a 100644 --- a/reference/api/unleash/validate-role.html +++ b/reference/api/unleash/validate-role.html @@ -20,15 +20,15 @@ - - + +

    Validate a role

    POST /api/admin/roles/validate

    Check if the role matches schema and has a unique name

    Request

    Body

    required

    createRoleWithPermissionsSchema

      anyOf
    • name string required

      The name of the custom role

    • description string

      A more detailed description of the custom role and what use it's intended for

    • type string

      Possible values: [root-custom, custom]

      Custom root roles (type=root-custom) are root roles with a custom set of permissions. Custom project roles (type=custom) contain a specific set of permissions for project resources.

    • permissions object[]

      A list of permissions assigned to this role

    • Array [
    • name string required

      The name of the permission

    • environment string

      The environments of the permission if the permission is environment specific

    • ]
    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/validate-segment.html b/reference/api/unleash/validate-segment.html index 30897dba56..562a2ae588 100644 --- a/reference/api/unleash/validate-segment.html +++ b/reference/api/unleash/validate-segment.html @@ -20,15 +20,15 @@ - - + +

    Validates if a segment name exists

    POST /api/admin/segments/validate

    Uses the name provided in the body of the request to validate if the given name exists or not

    Request

    Body

    required

    nameSchema

    • name string required

      The name of the represented object.

    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/validate-tag-type.html b/reference/api/unleash/validate-tag-type.html index 23e8ae3671..9d0011b2e6 100644 --- a/reference/api/unleash/validate-tag-type.html +++ b/reference/api/unleash/validate-tag-type.html @@ -20,15 +20,15 @@ - - + +

    Validate a tag type

    POST /api/admin/tag-types/validate

    Validates whether if the body of the request is a valid tag and whether the a tag type with that name already exists or not. If a tag type with the same name exists, this operation will return a 409 status code.

    Request

    Body

    required

    tagTypeSchema

    • name string required

      The name of the tag type.

    • description string

      The description of the tag type.

    • icon string nullable

      The icon of the tag type.

    Responses

    validateTagTypeSchema

    Schema
    • valid boolean required

      Whether or not the tag type is valid.

    • tagType objectrequired

      A tag type.

    • name string required

      The name of the tag type.

    • description string

      The description of the tag type.

    • icon string nullable

      The icon of the tag type.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/validate-token.html b/reference/api/unleash/validate-token.html index adfe55ffd3..024eeb37e3 100644 --- a/reference/api/unleash/validate-token.html +++ b/reference/api/unleash/validate-token.html @@ -20,15 +20,15 @@ - - + +

    Validates a token

    GET /auth/reset/validate

    If the token is valid returns the user that owns the token

    Request

    Responses

    tokenUserSchema

    Schema
    • id integer required

      The user id

    • name string

      The name of the user

    • email string required

      The email of the user

    • token string required

      A token uniquely identifying a user

    • createdBy string nullable required

      A username or email identifying which user created this token

    • role objectrequired

      A role holds permissions to allow Unleash to decide what actions a role holder is allowed to perform

    • id integer required

      The role id

    • type string required

      A role can either be a global root role (applies to all projects) or a project role

    • name string required

      The name of the role

    • description string

      A more detailed description of the role and what use it's intended for

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/validate-user-password.html b/reference/api/unleash/validate-user-password.html index 87730272bb..8f843d4df7 100644 --- a/reference/api/unleash/validate-user-password.html +++ b/reference/api/unleash/validate-user-password.html @@ -20,15 +20,15 @@ - - + +

    Validate password for a user

    POST /api/admin/user-admin/validate-password

    Validate the password strength. Minimum 10 characters, uppercase letter, number, special character.

    Request

    Body

    required

    passwordSchema

    • password string required

      The new password to change or validate.

    • oldPassword string

      The old password the user is changing. This field is for the non-admin users changing their own password.

    • confirmPassword string

      The confirmation of the new password. This field is for the non-admin users changing their own password.

    Responses

    This response has no body.

    Loading...
    - - + + \ No newline at end of file diff --git a/reference/api/unleash/validate.html b/reference/api/unleash/validate.html index b7301f00f6..2a54d3fe5f 100644 --- a/reference/api/unleash/validate.html +++ b/reference/api/unleash/validate.html @@ -20,15 +20,15 @@ - - + +
    - - + + \ No newline at end of file diff --git a/reference/archived-toggles.html b/reference/archived-toggles.html index f6148fa417..27d60ecaa5 100644 --- a/reference/archived-toggles.html +++ b/reference/archived-toggles.html @@ -20,15 +20,15 @@ - - + +

    Archived Toggles

    You can archive a feature toggle when it is not needed anymore. You do this by clicking the "Archive" button on the feature toggle details view. When you archive a feature toggle, it will no longer be available to Client SDKs.

    The Unleash toggle view showing a focused &quot;archive feature toggle&quot; button, highlighted by a red arrow.

    Viewing archived toggles

    Archived toggles are displayed in two places:

    1. The global toggle archive
    2. The containing project's toggle archive

    Unleash keeps a list of all archived toggles across projects in the global archive. The global archive is accessible from the global feature list.

    Additionally, each project keeps a list of all of its archived toggles. That is, when you archive a toggle, Unleash adds it to the containing project's archive.

    Reviving a feature toggle

    If you want to re-use a feature toggle that you previously archived, you can revive in from the feature toggle archive. Click the "revive icon" to revive the toggle. Revived toggles will be in the disabled state when you re-enable them.

    A list of archived toggles. Each toggle displays its name and project it belongs to. Each toggle also has a &quot;revive&quot; button, as highlighted by a red arrow.

    Deleting a feature toggle

    caution

    We generally discourage deleting feature toggles. The reason is that feature toggle names in Unleash are used as identifiers, so if you were to delete a toggle and then create a new one with the same name, the new one would reactivate any code that uses the old toggle.

    The only way to fully delete a feature toggle in Unleash is by using the archive. An archived toggle can be deleted via the API or by using the "delete feature toggle" button in the global or project-level archive.

    A list of archived toggles, each with a button to delete the toggle permanently.

    - - + + \ No newline at end of file diff --git a/reference/banners.html b/reference/banners.html index 3a1a9c5512..5132014a82 100644 --- a/reference/banners.html +++ b/reference/banners.html @@ -20,15 +20,15 @@ - - + +

    Banners

    Availability

    Banners were introduced as a beta feature in Unleash 5.6 and are only available in Unleash Enterprise. We plan to make this feature generally available to all Enterprise users in Unleash 5.7.

    Banners allow you to configure and display internal messages that all users of your Unleash instance can see and interact with. They are displayed at the top of the Unleash UI, and can be configured to be interactive.

    Banners table

    A common use case could be to have some pre-configured banners that you can enable when you need to communicate something to your users. For example, you could have a banner that you enable when you're doing maintenance on your Unleash instance, and another banner that you enable when you're running a survey.

    In order to create and display a banner, you can follow the how to create and display banners guide.

    Banners can be enabled or disabled at any time. For more information on how to enable or disable a banner, see the section on displaying banners.

    OptionDescription
    EnabledWhether the banner is currently displayed to all users of your Unleash instance.

    Configuration

    Banners can be configured with the following options:

    OptionDescription
    TypeThe type of banner, which controls the banner's color and its icon, if using the default icon option.
    IconThe icon displayed on the banner. This can be the default for the banner type, a custom icon, or hidden by selecting "None".
    MessageThe banner's message. Supports Markdown.

    Custom icon

    To further personalize your banner, you can configure it with a custom icon.

    To use a custom icon in your banner:

    1. Select "Custom" in the icon dropdown.
    2. Enter the name of the desired Material Symbol.
      • For example, for the "Rocket Launch" icon, enter rocket_launch in the custom icon field.
    OptionDescription
    Custom iconThe custom icon to be displayed on the banner, using Material Symbols.

    You can set up an action for your banner:

    OptionDescription
    Banner actionThe action activated when a user interacts with the banner link. Defaults to "None". Options include a link or a dialog.

    When choosing the link action, a link will be displayed on the banner that directs users to a specified URL.

    The configured URL can be absolute, as in e.g. https://docs.getunleash.io/, or relative as in e.g. /admin/network. Absolute URLs will open in a new tab.

    OptionDescription
    URLThe URL to open when the user uses the banner link.
    TextThe text to display on the banner link.

    Dialog

    When opting for a dialog action, an interactable link appears on the banner which opens a dialog with additional information.

    OptionDescription
    TextThe text to display on the banner link.
    Dialog titleThe title to display on the dialog.
    Dialog contentThe content to display on the dialog. Supports Markdown.

    Sticky banner

    For added visibility, banners can be configured to be "sticky," ensuring they remain at the top of the Unleash UI, even after scrolling the page. This is useful for banners that you want to make sure that your users see and interact with.

    OptionDescription
    StickyWhether the banner is sticky on the screen when scrolling.
    - - + + \ No newline at end of file diff --git a/reference/change-requests.html b/reference/change-requests.html index 1576df9c5c..f73554f627 100644 --- a/reference/change-requests.html +++ b/reference/change-requests.html @@ -20,8 +20,8 @@ - - + + @@ -30,7 +30,7 @@

    Change Requests

    Availability

    The change requests feature is an enterprise-only feature that was introduced in Unleash 4.19.0. The change requests for segments was introduced in Unleash 5.4.0.

    Feature flagging is a powerful tool, and because it's so powerful, you sometimes need to practice caution. The ability to have complete control over your production environment comes at the cost of the potential to make mistakes in production. Change requests were introduced in version 4.19.0 to alleviate this fear. Change requests allow you to group changes together and apply them to production at the same time, instead of applying changes directly to production. This allows you to make multiple changes to feature toggles and their configuration and status (on/off) all at once, reducing the risk of errors in production.

    Our goal is developer efficiency, but we also recognize that we have users and customers in highly regulated industries, governed by law and strict requirements. Therefore, we have added a capability to change requests that will allow you to enforce the 4 eyes principle.

    Change request configuration

    The change request configuration can be set up per project, per environment. This means that you can have different change request configurations for different environments, such as production and development. This is useful because different environments may have different requirements, so you can customize the change request configuration to fit those requirements. However, this also means that you cannot change toggles across projects in the same change request.

    Currently there are two configuration options for change requests:

    • Enable change requests - This is a boolean value that enables or disables change requests for the project and environment.
    • Required approvals - This is an integer value that indicates how many approvals are required before a change request can be applied. Specific permissions are required to approve and apply change requests.

    The change request configuration can be set up in the project settings page:

    Change request configuration

    Change request flow

    Once a change request flow is configured for a project and environment, you can no longer directly change the status of a toggle. Instead, you will be asked to put your changes into a draft. The change request flow handles the following scenarios:

    • Updating the status of a toggle in the environment
    • Adding a strategy to the feature toggle in the environment
    • Updating a strategy of a feature toggle in the environment
    • Deleting a strategy from a feature toggle in the environment

    The flow can be summarized as follows:

    Change request flow

    Once a change is added to a draft, the draft needs to be completed before another change request can be opened. The draft is personal to the user that created the change request draft, until it is sent for review. Once changes are added to draft, the user will have a banner in the top of the screen indicating that a draft exists. The state of a change request can be one of the following:

    • Draft - The change request is in draft mode, and can be edited by the user that created the draft.
    • In review - The change request is in review mode, and can be edited by the user that created the draft. If editing occurs, all current approvals are revoked
    • Approved - The change request has been approved by the required number of users.
    • Scheduled - The change request has been scheduled and will be applied at the scheduled time (unless there are conflicts, as described in the section on scheduling change requests).
    • Applied - The change request has been applied to the environment. The feature toggle configuration is updated.
    • Cancelled - The change request has been cancelled by the change request author or by an admin.
    • Rejected - The change request has been rejected by the reviewer or by an admin.

    Change request banner

    Once a change request is sent to review by the user who created it, it becomes available for everyone in the change request tab in the project.

    From here, you can navigate to the change request overview page. This page will give you information about the changes the change request contains, the state the change request is in, and what action needs to be taken next.

    Change request banner

    From here, if you have the correct permissions, you can approve and schedule or apply the change request. Once applied, the changes will be live in production.

    Scheduled changes

    Availability

    Change request scheduling is currently in development and will be released in an upcoming version of Unleash. How the feature works (and as such, the contents of this subsection) can change before the feature is released.

    When a change request is approved, you can schedule it to be applied at a later time. This allows you to group changes together and apply them at a time that is convenient for you, such as during a maintenance window, or at a time when you know there will be less traffic to your application.

    Scheduled changes can be rescheduled, applied immediately, or rejected. They can not be edited or moved back to any of the previous states.

    Unleash will attempt to apply the changes at the scheduled time. However, if there are conflicts, the changes will not be applied and the change request will be marked as failed. Conflicts will occur if the change request contains changes that affect a flag that has been archived or a strategy that has been deleted.

    Be aware that if a strategy or variants affected by a scheduled change request are updated after the change request was scheduled, the application of the scheduled change request will overwrite those changes with the state in the scheduled change request.

    Unleash will warn you ahead of time if you make changes that conflict with a scheduled change request.

    If Unleash fails to apply a scheduled change request, the change request will remain in the scheduled state. You can reschedule it and try to apply it again later, or you can reject it.

    If a scheduled change request can not be applied, Unleash will send a notification to the person who scheduled it and to the person who created the change request.

    When a scheduled change request is applied, the person who scheduled it and the person who created it will each receive a notification.

    Edge cases: what happens when ...?

    If the user who scheduled a change request is deleted from the Unleash users list before the scheduled time, the changes will not be applied. Instead, the schedule will be put into a special suspended state. A change request with suspended schedule will not be applied at its scheduled time. A user with the required permission can reschedule, apply, or reject the change request. Any of these actions will put the change request back into the regular flow.

    If a change request has been scheduled and change requests are then disabled for the project and environment, the change request will still be applied according to schedule. To prevent this, you can reject the scheduled change request.

    Different ways to schedule changes

    Unleash currently offers two distinct ways to schedule changes. Each method has its own pros and cons, and you can also combine the methods for maximum flexibility.

    The first method is through scheduled change requests, as we have explained in the preceding sections. Scheduled change requests make it easy to see all the changes across multiple flags and strategies in one view and makes it easy to reschedule or reject them. However, because scheduled changes rely on flags and strategy configurations, conflicts can arise causing the schedule to fail.

    The second method uses Unleash's constraints and the DATE_AFTER operator to encode when changes should take effect. The pros of this method is that because these changes can be applied immediately, you won't run into any conflicts when they happen. The cons are that you'll need to apply the same constraints to all the parts that you want to change and that there is no easy way to see all the changes in one view. You also can not scheduled changes to a segment in this way. When using this option, we recommend that you use segments if you want to schedule multiple changes, so that their application time stays in sync.

    Another important distinction is how these changes affect your connected SDKs. If you use constraints (or segments), then any connected SDK will be aware of the schedule ahead of time. That means that even if the SDK can not connect to Unleash at the scheduled time, it will still activate the changes because it's encoded in its constraints. On the other hand, if you use change requests to schedule changes, SDKs must update their configuration after the scheduled time to be aware of the changes.

    Change request permissions

    Change requests have their own set of environment-specific permissions that can be applied to custom project roles. These permissions let users

    • approve/reject change requests
    • apply change requests
    • skip the change request flow

    None of the predefined roles have any change request permissions, so you must create your own project roles to take advantage of change requests. In other words, even a user with the project "owner" role can not approve or apply change requests.

    There is no permission to create change requests: Anyone can create change requests, even Unleash users with the root viewer role. Change requests don't cause any changes until approved and applied by someone with the correct permissions.

    You can prevent non-project members from submitting change requests by setting a protected project collaboration mode.

    Circumventing change requests

    The skip change requests permission allows users to bypass the change request flow. Users with this permission can change feature toggles directly (they are of course still limited by any other permissions they have).

    The skip change requests permission was designed to make it possible to quickly turn something off in the event that a feature release didn't go as expected or was causing issues.

    The skip change requests permission does not grant any other permissions, so to be allowed to do things as enabling/disabling a toggle, the user will still need the explicit permissions to do that too.

    In the UI non-admin users with skip change requests permission and explicit permission to perform the actual action will be able to make changes directly without change requests.

    Admin users will always see the change request UI so that they can test the change request flow. Admin users can however self-approve and self-apply their own changes.

    Change Request for segments

    Changes to project segments (as opposed to global segments) also go through the change request process. This is to prevent a backdoor in the change request process.

    Since projects segments are not environment specific and change requests are always environment specific we allow to attach segment change to any environment with change requests enabled. When you make changes though the Change Request UI it will automatically select first environment with change requests enabled, giving priority to production environments.

    Changes to segments can be only circumvented by admin users through the API calls.

    - - + + \ No newline at end of file diff --git a/reference/custom-activation-strategies.html b/reference/custom-activation-strategies.html index 47f7f2fe27..9a959b5ec0 100644 --- a/reference/custom-activation-strategies.html +++ b/reference/custom-activation-strategies.html @@ -20,15 +20,15 @@ - - + +

    Custom Activation Strategies

    tip

    This document is a reference for custom activation strategies. If you're looking for a guide on how to use them, see the how to use custom strategies guide.

    Custom activation strategies should be considered an advanced feature. In most cases strategy constraints provides enough control without the additional complexity.

    Custom activation strategies let you define your own activation strategies to use with Unleash. When the built-in activation strategies aren't enough, custom activation strategies are there to provide you with the flexibility you need.

    Custom activation strategies work exactly like the built-in activation strategies when working in the admin UI.

    Definition

    A strategy creation form. It has fields labeled &quot;strategy name&quot; — &quot;TimeStamp&quot; — and &quot;description&quot; — &quot;activate toggle after a given timestamp&quot;. It also has fields for a parameter named &quot;enableAfter&quot;. The parameter is of type &quot;string&quot; and the parameter description is &quot;Expected format: YYYY-MM-DD HH:MM&quot;. The parameter is required.

    You define custom activation strategies on your Unleash instance, either via the admin UI or via the API. A strategy contains:

    • A strategy name: You'll use this to refer to the strategy in the UI and in code. The strategy name should make it easy to understand what the strategy does. For instance, if a strategy uses contact numbers to determine whether a feature should be enabled, then ContactNumbers would be a good name.
    • An optional description: Use this to describe what the strategy should do.
    • An optional list of parameters: The parameter list lets you pass arguments to your custom activation strategy. These will be made available to your custom strategy implementation. How you interact with them differs between SDKs, but refer to the Node.js example in the how-to guide for a rough idea.

    The strategy name is the only required parameter, but adding a good description will make it easier to remember what a strategy should do. The list of parameters lets you pass data from the Unleash instance to the strategy implementation.

    Parameters

    A strategy with five parameters, one of each type.

    Parameters let you provide arguments to your strategy that it can access for evaluation. When creating a strategy, each parameter can be either required or optional. This marking is to help the user understand what they need to fill out; they can still save the strategy without filling out a required field.

    Each parameter consists of three parts:

    • a name: must be unique among the strategy's parameters.
    • an optional description: describe the purpose or format of the parameter.
    • a parameter type: dictates the kind of input field the user will see in the admin UI and the type of the value in the implementation code.

    Parameter types

    Each parameter has one of five different parameter types. A parameter's type impacts the kind of controls shown in the admin UI and how it's represented in code.

    The below table lists the types and how they're represented in the JSON payload returned from the Unleash server. When parsed, the exact types will vary based on your programming language's type system.

    All values have an empty string ("") as the default value. As such, if you don't interact with a control or otherwise set a value, the value will be an empty string.

    type namecode representationexample valueUI control
    stringstring"a string"A standard input field
    percentagea string representing a number between 0 and 100 (inclusive)"99"A value slider
    liststring (values are comma-separated)"one,two"A multi-input text field
    numberstring"123"A numeric text field
    booleana string: one of "true" or "false""true"An on/off toggle

    Implementation

    note

    If you have not implemented the strategy in your client SDK, the check will always return false because the client doesn't recognize the strategy.

    While custom strategies are defined on the Unleash server, they must be implemented on the client. All official Unleash client SDKs provide a way for you to implement custom strategies. You should refer to the individual SDK's documentation for specifics, but for an example, you can have a look at the Node.js client implementation section in the how to use custom strategies guide.

    The exact method for implementing custom strategies will vary between SDKs, but the server SDKs follow the same patterns. For front-end client SDKs (Android, JavaScript, React, iOS, Flutter), the custom activation strategy must be implemented in the Unleash Proxy.

    When implementing a strategy in your client, you will get access to the strategy's parameters and the Unleash Context. Again, refer to your specific SDK's documentation for more details.

    - - + + \ No newline at end of file diff --git a/reference/dependent-features.html b/reference/dependent-features.html index 249367cb26..8ed4d019fe 100644 --- a/reference/dependent-features.html +++ b/reference/dependent-features.html @@ -20,8 +20,8 @@ - - + + @@ -29,7 +29,7 @@

    Dependent Features

    Availability

    Dependent features were first introduced in Unleash 5.7 for Pro and Enterprise users.

    Overview

    Dependent features allow to define a child feature flag that depends on a parent feature flag.
    A feature flag can have only one parent dependency but multiple child flags can share the same parent. For a child flag to be activated, its parent dependency must be enabled.

    Parent dependency criteria

    • Project Association: Both parent and child flags should belong to the same project.
    • Single Level Dependency: The parent flag can’t have its own parent, ensuring a straightforward, single-level dependency.

    Managing dependencies

    Adding

    Introduce dependencies either through the UI or API, also applicable when copying a child feature with an existing parent dependency.

    A button for adding parent dependency.

    Removing

    Eliminate them through the UI or API. Dependencies are also removed when archiving a child feature. A parent feature can’t be removed if it would leave a child feature orphaned. To remove both, batch archive them. If Unleash confirms no other child features are using the parent, archiving proceeds.

    A button for deleting parent dependency.

    Permissions

    The Update feature dependency project permission, auto-assigned to admin and project members, allows managing dependencies.

    Metrics calculation

    Metrics are influenced solely by the evaluation of child features.

    Client SDK Support

    To make use of dependent feature, you need to use a compatible client. Client SDK with variant support:

    If you would like to give feedback on this feature, experience issues or have questions, please feel free to open an issue on GitHub.

    - - + + \ No newline at end of file diff --git a/reference/environments.html b/reference/environments.html index a4038264bf..e8ecfaa18c 100644 --- a/reference/environments.html +++ b/reference/environments.html @@ -20,15 +20,15 @@ - - + +

    Environments

    Availability

    Environments were released in Unleash v4.3.0.

    Environments is a new way to organize activation strategy configurations for feature toggles into separate environments. In Unleash, a feature lives across all your environments — after all, the goal is to get the new feature released as soon as possible — but it makes sense to configure the activation differently per environment. You might want the feature enabled for everyone in development, but only for yourself in production, for instance.

    Previously, Unleash Enterprise could use strategy constraints to control the rollout across environments. With the new environments feature, this is no longer necessary. Now all activation strategies belong to an explicit environment instead.

    Further, connected applications will use environment-scoped API keys to make sure they only download feature toggle configurations for the environment they are running in.

    Finally, metrics have also been upgraded to record the environment. This, in turn, means that Unleash can display usage metrics per environment.

    Despite this being a shift in how Unleash works, everything will continue to work exactly how it did for existing users. For backwards compatibility, we have created an environment named "default" that will contain all of the existing toggles and API keys. Read more about that in the migration section.

    A graph showing how environments work. Each project can have multiple features, and each feature can have different activation strategies in each of its environments.

    Environment types

    All environments in Unleash have a type. When you create a new environment, you must also assign it a type.

    The built-in environment types are:

    • Development
    • Test
    • Pre-production
    • Production

    The production environment type is special: Unleash will show additional confirmation prompts when you change something that could impact users in environments of this type. The built-in "production" environment is a production-type environment.

    The other environment types do not currently have any functionality associated with them. This may change in the future.

    Global and project-level environments

    Environments exist on a global level, so they are available to all projects within Unleash. However, every project might not need to use every environment. That's why you can also choose which of the global environments should be available within a project.

    How to start using environments

    In order to start using environments you need to be on Unleash v4.2 or higher.

    If you are on v4.2, you also need to have the environment feature enabled (if you are using Unleash Hosted, please reach out on contact@getunleash.io if you want to start using environments.

    If you are on v4.3 or later, environments are already enabled for you.

    Note that in order to enable an environment for a feature toggle, you must first add activation strategies for that environment. You cannot enable an environment without activation strategies.

    Step 1: Enable new environments for your Project

    Navigate to the project and choose the “environments” tab.

    A project view showing the Environments tab. The UI displays three environment toggles: &quot;default&quot;, &quot;development&quot;, and &quot;production&quot;. The &quot;default&quot; environment is enabled.

    Step 2: Configure activation strategies for the new environment

    From the “feature toggle view” you will now be able to configure activation strategies per environment. You can also enable and disable environments here. Remember that an environment must have activation strategies before you can enable it.

    A feature toggle strategies tab showing three different environments, of which one is active. The UI displays data about the currently selected environment,

    Step 3: Create environment specific API keys

    In order for the SDK to download the feature toggle configuration for the correct environment you will need to create an API token for a defined environment.

    An API key creation form. The form&#39;s fields are &quot;username&quot;, &quot;token type&quot;, &quot;project&quot;, and, crucially, &quot;environment&quot;. The development environment is selected.

    Cloning environments

    Availability

    Environment cloning was made available in Unleash 4.19.

    Unleash environments can be cloned. Cloning an environment creates a new environment based on the selected source environment. When cloning an environment, you select any number of projects whose feature toggle configurations will also be cloned. These projects will have identical configurations for the source and target environments. However, the environments can then be configured independently and will not stay in sync with each other.

    When cloning an environment, you must give the new environment

    • a name
    • an environment type
    • a list of projects to clone feature configurations in

    You can also clone user permissions into the new environment. When you do that, permissions in the new environment will be the same as in the environment you cloned from. If you don't clone permissions, only admins and project editors can make changes in the new environment. You can change permissions after creation by using custom project roles.

    In order to clone an environment, you can follow the how to clone environments guide.

    Once created, the new environment works just as any other environment.

    Migration

    To ease migration we have created a special environment called “default”. All existing activation strategies have been added to this environment. All existing Client API keys have also been scoped to work against the default environment to ensure zero disruption as part of the upgrade.

    If you're currently using strategy constraints together with the “environment” field on the Unleash Context, you should be aware that the new environment support works slightly differently. With environments, the SDK API will use the client's API key to determine which environment the client is configured for. The API then sends only strategies belonging to the client's environment. This means that you might not need the "environment" property of the Unleash Context anymore.

    A strategy constraint using the environment field of the unleash context.

    Integrations

    We have made some slight changes to events related to feature toggles: there's one deprecation and several new event types. Most of the new events contain project and environment data.

    To avoid missing important updates, you will also need to update your integration configuration to subscribe to the new events.

    Deprecated events:

    • FEATURE_UPDATE - not used after switching to environments

    New Events

    • FEATURE-METADATA-UPDATED - The feature toggle metadata was updated (across all environments).
    • FEATURE-STRATEGY-ADD¹ - An activation strategy was added to a feature toggle in an environment. The data will contain the updated strategy configuration.
    • FEATURE-STRATEGY-UPDATE¹ - An activation strategy was updated for a feature toggle. The data will contain the updated strategy configuration.
    • FEATURE-STRATEGY-REMOVE¹ - An activation strategy was removed for a feature toggle.
    • FEATURE-ENVIRONMENT-ENABLED¹ - Signals that a feature toggle has been enabled in a defined environment.
    • FEATURE-ENVIRONMENT-DISABLED¹ - Signals that a feature toggle has been disabled in a defined environment.
    • FEATURE-PROJECT-CHANGE¹ - The feature toggle was moved to a new project.
    1. These feature events will contain project and environment as part of the event metadata.

    API

    In order to support configuration per environment we had to rebuild our feature toggle admin API to account for environments as well. This means that we're making the following changes to the API:

    • /api/admin/features - deprecated (scheduled for removal in Unleash v5.0). The old feature toggles admin API still works, but strategy configuration will be assumed to target the “default” environment.
    • /api/admin/projects/:projectId/features - New feature API to be used for feature toggles which also adds support for environments. See the documentation to learn more.

    Plan Differences

    Open-Source (free)

    • Will get access to two preconfigured environments: “development” and “production”. Existing users of Unleash will also get an additional “default” environment to simplify the adoption of environments.
    • Will be possible to turn environments on/off for all projects.

    Pro (commercial)

    • Will get access to two preconfigured environments: “development” and “production”. Existing users of Unleash will also get an additional “default” environment to simplify the adoption of environments.
    • Will be possible to turn environments on/off for the default project.

    Enterprise (commercial)

    • Will get access to two preconfigured environments: “development” and “production”. Existing users of Unleash will also get an additional “default” environment to simplify the adoption of environments.
    • Will be possible to turn environments on/off for all projects
    • Will be allowed to update and remove environments.
    • Will be allowed to create new environments and clone existing environments.

    Rollout Plan

    • Unleash v4.2 will provide early access to environment support. This means that it can be enabled per customer via a feature flag.
    • Unleash v4.3 plans to provide general access to the environment support for all users of Unleash (Open-Source, Pro, Enterprise).
    - - + + \ No newline at end of file diff --git a/reference/event-log.html b/reference/event-log.html index f2f677e568..7dad2a6047 100644 --- a/reference/event-log.html +++ b/reference/event-log.html @@ -20,15 +20,15 @@ - - + +

    Event Log

    The event log lets you track changes in Unleash. It lists what changed, when it changed, and who performed the change.

    Feature toggle log

    Each feature toggle has its own event log. The event log is available under the "Event log" tab in the tab view.

    The event log for a feature toggle. The &quot;Event log&quot; tab is highlighted and the UI shows the most recent changes, including a JSON diff and the change details.

    Global Event Log

    Unleash also keeps an event log across all toggles and activation strategies, tracking all changes. You access the global event log via the “event log”, which you can find in the drawer menu. The global event log is only accessible by users with instance admin access.

    The global event log and how to get there. It shows a number of events and their changes as well as the navigation steps: use the admin menu and navigate to &quot;event history&quot;.

    - - + + \ No newline at end of file diff --git a/reference/event-types.html b/reference/event-types.html index 939fa3a7c7..bf5db2c70f 100644 --- a/reference/event-types.html +++ b/reference/event-types.html @@ -20,15 +20,15 @@ - - + +

    Event Types

    Unleash emits a large number of different events (described in more detail in the next sections). The exact fields an event contains varies from event to event, but they all conform to the following TypeScript interface before being transformed to JSON:

    interface IEvent {
    id: number;
    createdAt: Date;
    type: string;
    createdBy: string;
    project?: string;
    environment?: string;
    featureName?: string;
    data?: any;
    preData?: any;
    tags?: ITag[];
    }

    The event properties are described in short in the table below. For more info regarding specific event types, refer to the following sections.

    PropertyDescription
    createdAtThe time the event happened as a RFC 3339-conformant timestamp.
    dataExtra associated data related to the event, such as feature toggle state, segment configuration, etc., if applicable.
    environmentThe feature toggle environment the event relates to, if applicable.
    featureNameThe name of the feature toggle the event relates to, if applicable.
    idThe ID of the event. An increasing natural number.
    preDataData relating to the previous state of the event's subject.
    projectThe project the event relates to, if applicable.
    tagsAny tags related to the event, if applicable.
    typeThe event type, as described in the rest of this section.

    Feature toggle events

    These events pertain to feature toggles and their life cycle.

    feature-created

    This event fires when you create a feature. The data property contains the details for the new feature.

    example event: feature-created
    {
    "id": 899,
    "type": "feature-created",
    "createdBy": "user@company.com",
    "createdAt": "2022-05-31T13:32:20.560Z",
    "data": {
    "name": "new-feature",
    "description": "Toggle description",
    "type": "release",
    "project": "my-project",
    "stale": false,
    "variants": [],
    "createdAt": "2022-05-31T13:32:20.547Z",
    "lastSeenAt": null,
    "impressionData": true
    },
    "preData": null,
    "tags": [],
    "featureName": "new-feature",
    "project": "my-project",
    "environment": null
    }

    feature-updated

    Deprecation notice

    This event type was replaced by more granular event types in Unleash 4.3. From Unleash 4.3 onwards, you'll need to use the events listed later in this section instead.

    This event fires when a feature gets updated in some way. The data property contains the new state of the toggle. This is a legacy event, so it does not populate preData property.

    example event: feature-updated
    {
    "id": 899,
    "type": "feature-updated",
    "createdBy": "user@company.com",
    "createdAt": "2022-05-31T13:32:20.560Z",
    "data": {
    "name": "new-feature",
    "description": "Toggle description",
    "type": "release",
    "project": "my-project",
    "stale": false,
    "variants": [],
    "createdAt": "2022-05-31T13:32:20.547Z",
    "lastSeenAt": null,
    "impressionData": true
    },
    "preData": null,
    "tags": [],
    "featureName": "new-feature",
    "project": "my-project",
    "environment": null
    }

    feature-deleted

    This event fires when you delete a feature toggle. The preData property contains the deleted toggle data.

    example event: feature-deleted
    {
    "id": 903,
    "type": "feature-deleted",
    "createdBy": "admin-account",
    "createdAt": "2022-05-31T14:06:14.574Z",
    "data": null,
    "preData": {
    "name": "new-feature",
    "type": "experiment",
    "stale": false,
    "project": "my-project",
    "variants": [],
    "createdAt": "2022-05-31T13:32:20.547Z",
    "lastSeenAt": null,
    "description": "Toggle description",
    "impressionData": true
    },
    "tags": [],
    "featureName": "new-feature",
    "project": "my-project",
    "environment": null
    }

    feature-archived

    This event fires when you archive a toggle.

    example event: feature-archived
    {
    "id": 902,
    "type": "feature-archived",
    "createdBy": "user@company.com",
    "createdAt": "2022-05-31T14:04:38.661Z",
    "data": null,
    "preData": null,
    "tags": [],
    "featureName": "new-feature",
    "project": "my-project",
    "environment": null
    }

    feature-revived

    This event fires when you revive an archived feature toggle (when you take a toggle out from the archive).

    example-event: feature-revived
    {
    "id": 914,
    "type": "feature-revived",
    "createdBy": "user@company.com",
    "createdAt": "2022-06-01T09:57:10.719Z",
    "data": null,
    "preData": null,
    "tags": [],
    "featureName": "new-feature",
    "project": "my-other-project",
    "environment": null
    }

    feature-metadata-updated

    This event fires when a feature's metadata (its description, toggle type, or impression data settings) are changed. The data property contains the new toggle data. The preData property contains the toggle's previous data.

    The below example changes the toggle's type from release to experiment.

    example event: feature-metadata-updated
    {
    "id": 901,
    "type": "feature-metadata-updated",
    "createdBy": "user@company.com",
    "createdAt": "2022-05-31T13:35:25.244Z",
    "data": {
    "name": "new-feature",
    "description": "Toggle description",
    "type": "experiment",
    "project": "my-project",
    "stale": false,
    "variants": [],
    "createdAt": "2022-05-31T13:32:20.547Z",
    "lastSeenAt": null,
    "impressionData": true
    },
    "preData": {
    "name": "new-feature",
    "type": "release",
    "stale": false,
    "project": "my-project",
    "variants": [],
    "createdAt": "2022-05-31T13:32:20.547Z",
    "lastSeenAt": null,
    "description": "Toggle description",
    "impressionData": true
    },
    "tags": [],
    "featureName": "new-feature",
    "project": "my-project",
    "environment": null
    }

    feature-project-change

    This event fires when you move a feature from one project to another. The data property contains the names of the old and the new project.

    example event: feature-project-change
    {
    "id": 11,
    "type": "feature-project-change",
    "createdBy": "admin",
    "createdAt": "2022-06-03T11:09:41.444Z",
    "data": {
    "newProject": "default",
    "oldProject": "2"
    },
    "preData": null,
    "tags": [],
    "featureName": "feature",
    "project": "default",
    "environment": null
    }

    feature-import

    This event fires when you import a feature as part of an import process. The data property contains the feature data.

    example event: feature-import
    {
    "id": 26,
    "type": "feature-import",
    "createdBy": "import-API-token",
    "createdAt": "2022-06-03T11:30:40.570Z",
    "data": {
    "name": "feature",
    "description": "",
    "type": "release",
    "project": "default",
    "stale": false,
    "variants": [],
    "impressionData": false,
    "enabled": false,
    "archived": false
    },
    "preData": null,
    "tags": [],
    "featureName": null,
    "project": null,
    "environment": null
    }

    feature-tagged

    This event fires when you add a tag to a feature. The data property contains the new tag.

    example event: feature-tagged
    {
    "id": 897,
    "type": "feature-tagged",
    "createdBy": "user@company.com",
    "createdAt": "2022-05-31T13:06:31.047Z",
    "data": {
    "type": "simple",
    "value": "tag2"
    },
    "preData": null,
    "tags": [],
    "featureName": "example-feature-name",
    "project": null,
    "environment": null
    }

    feature-untagged

    This event fires when you remove a tag from a toggle. The data property contains the tag that was removed.

    example event: feature-untagged
    {
    "id": 893,
    "type": "feature-untagged",
    "createdBy": "user@company.com",
    "createdAt": "2022-05-31T12:58:10.241Z",
    "data": {
    "type": "simple",
    "value": "thisisatag"
    },
    "preData": null,
    "tags": [],
    "featureName": "example-feature-name",
    "project": null,
    "environment": null
    }

    feature-tag-import

    This event fires when you import a tagged feature as part of an import job. The data property contains the name of the feature and the tag.

    example event: feature-tag-import
    {
    "id": 43,
    "type": "feature-tag-import",
    "createdBy": "import-API-token",
    "createdAt": "2022-06-03T11:30:40.606Z",
    "data": {
    "featureName": "new-feature",
    "tag": {
    "type": "simple",
    "value": "tag1"
    }
    },
    "preData": null,
    "tags": [],
    "featureName": null,
    "project": null,
    "environment": null
    }

    feature-strategy-add

    This event fires when you add a strategy to a feature. The data property contains the configuration for the new strategy.

    example event: feature-strategy-add
    {
    "id": 919,
    "type": "feature-strategy-add",
    "createdBy": "user@company.com",
    "createdAt": "2022-06-01T10:03:08.290Z",
    "data": {
    "id": "3f4bf713-696c-43a4-8ce7-d6c607108858",
    "name": "flexibleRollout",
    "constraints": [],
    "parameters": {
    "groupId": "new-feature",
    "rollout": "67",
    "stickiness": "default"
    }
    },
    "preData": null,
    "tags": [],
    "featureName": "new-feature",
    "project": "my-other-project",
    "environment": "default"
    }

    feature-strategy-update

    This event fires when you update a feature strategy. The data property contains the new strategy configuration. The preData property contains the previous strategy configuration.

    example event: feature-strategy-update
    {
    "id": 920,
    "type": "feature-strategy-update",
    "createdBy": "user@company.com",
    "createdAt": "2022-06-01T10:03:11.549Z",
    "data": {
    "id": "3f4bf713-696c-43a4-8ce7-d6c607108858",
    "name": "flexibleRollout",
    "constraints": [],
    "parameters": {
    "groupId": "new-feature",
    "rollout": "32",
    "stickiness": "default"
    }
    },
    "preData": {
    "id": "3f4bf713-696c-43a4-8ce7-d6c607108858",
    "name": "flexibleRollout",
    "parameters": {
    "groupId": "new-feature",
    "rollout": "67",
    "stickiness": "default"
    },
    "constraints": []
    },
    "tags": [],
    "featureName": "new-feature",
    "project": "my-other-project",
    "environment": "default"
    }

    feature-strategy-remove

    This event fires when you remove a strategy from a feature. The preData contains the configuration of the strategy that was removed.

    example event: feature-strategy-remove
    {
    "id": 918,
    "type": "feature-strategy-remove",
    "createdBy": "user@company.com",
    "createdAt": "2022-06-01T10:03:00.229Z",
    "data": null,
    "preData": {
    "id": "9591090e-acb0-4088-8958-21faaeb7147d",
    "name": "default",
    "parameters": {},
    "constraints": []
    },
    "tags": [],
    "featureName": "new-feature",
    "project": "my-other-project",
    "environment": "default"
    }

    feature-stale-on

    This event fires when you mark a feature as stale.

    example event: feature-stale-on
    {
    "id": 926,
    "type": "feature-stale-on",
    "createdBy": "user@company.com",
    "createdAt": "2022-06-01T10:10:46.737Z",
    "data": null,
    "preData": null,
    "tags": [
    {
    "value": "tag",
    "type": "simple"
    },
    {
    "value": "tog",
    "type": "simple"
    }
    ],
    "featureName": "new-feature",
    "project": "my-other-project",
    "environment": null
    }

    feature-stale-off

    This event fires when you mark a stale feature as no longer being stale.

    example event: feature-stale-off
    {
    "id": 928,
    "type": "feature-stale-off",
    "createdBy": "user@company.com",
    "createdAt": "2022-06-01T10:10:52.790Z",
    "data": null,
    "preData": null,
    "tags": [
    {
    "value": "tag",
    "type": "simple"
    },
    {
    "value": "tog",
    "type": "simple"
    }
    ],
    "featureName": "new-feature",
    "project": "my-other-project",
    "environment": null
    }

    feature-environment-enabled

    This event fires when you enable an environment for a feature. The environment property contains the name of the environment.

    example event: feature-environment-enabled
    {
    "id": 930,
    "type": "feature-environment-enabled",
    "createdBy": "user@company.com",
    "createdAt": "2022-06-02T12:09:03.045Z",
    "data": null,
    "preData": null,
    "tags": [
    {
    "value": "tag",
    "type": "simple"
    },
    {
    "value": "tog",
    "type": "simple"
    }
    ],
    "featureName": "new-feature",
    "project": "my-other-project",
    "environment": "development"
    }

    feature-environment-disabled

    This event fires when you disable an environment for a feature. The environment property contains the name of the environment.

    example event: feature-environment-disabled
    {
    "id": 931,
    "type": "feature-environment-disabled",
    "createdBy": "user@company.com",
    "createdAt": "2022-06-02T12:09:04.469Z",
    "data": null,
    "preData": null,
    "tags": [
    {
    "value": "tag",
    "type": "simple"
    },
    {
    "value": "tog",
    "type": "simple"
    }
    ],
    "featureName": "new-feature",
    "project": "my-other-project",
    "environment": "development"
    }

    drop-features

    This event fires when you delete existing features as part of an import job. The data.name property will always be "all-features".

    example event: drop-features
    {
    "id": 25,
    "type": "drop-features",
    "createdBy": "import-API-token",
    "createdAt": "2022-06-03T11:30:40.563Z",
    "data": {
    "name": "all-features"
    },
    "preData": null,
    "tags": [],
    "featureName": null,
    "project": null,
    "environment": null
    }

    drop-feature-tags

    This event fires when you drop all existing tags as part of a configuration import. The data.name property will always be "all-feature-tags".

    example event: drop-feature-tags
    {
    "id": 36,
    "type": "drop-feature-tags",
    "createdBy": "import-API-token",
    "createdAt": "2022-06-03T11:30:40.596Z",
    "data": {
    "name": "all-feature-tags"
    },
    "preData": null,
    "tags": [],
    "featureName": null,
    "project": null,
    "environment": null
    }

    feature-potentially-stale-on

    Availability

    feature-potentially-stale-on events are currently in development and expected to be stabilized in one of the upcoming releases.

    This event fires when Unleash marks a feature toggle as potentially stale. Feature toggles are marked as potentially stale when they exceed the expected lifetime of their feature toggle type.

    Example event when my-feature is marked as potentially stale
    {
    "id": 561,
    "type": "feature-potentially-stale-on",
    "createdBy": "unleash-system",
    "createdAt": "2023-07-19T09:12:31.313Z",
    "data": null,
    "preData": null,
    "tags": [],
    "featureName": "helix",
    "project": "viridian-forest",
    "environment": null
    }

    Strategy events

    strategy-created

    This event fires when you create a strategy. The data property contains the strategy configuration.

    example event: strategy-created
    {
    "id": 932,
    "type": "strategy-created",
    "createdBy": "user@company.com",
    "createdAt": "2022-06-02T12:20:52.111Z",
    "data": {
    "name": "new-strategy",
    "description": "this strategy does ...",
    "parameters": [],
    "editable": true,
    "deprecated": false
    },
    "preData": null,
    "tags": [],
    "featureName": null,
    "project": null,
    "environment": null
    }

    strategy-updated

    This event fires when you change a strategy's configuration. The data property contains the new strategy configuration.

    example event: strategy-updated
    {
    "id": 933,
    "type": "strategy-updated",
    "createdBy": "user@company.com",
    "createdAt": "2022-06-02T12:21:23.741Z",
    "data": {
    "name": "new-strategy",
    "description": "this strategy does something else!",
    "parameters": [],
    "editable": true,
    "deprecated": false
    },
    "preData": null,
    "tags": [],
    "featureName": null,
    "project": null,
    "environment": null
    }

    strategy-deleted

    This event fires when you delete a strategy. The data property contains the name of the deleted strategy.

    example event: strategy-deleted
    {
    "id": 936,
    "type": "strategy-deleted",
    "createdBy": "user@company.com",
    "createdAt": "2022-06-02T12:22:01.302Z",
    "data": {
    "name": "new-strategy"
    },
    "preData": null,
    "tags": [],
    "featureName": null,
    "project": null,
    "environment": null
    }

    strategy-deprecated

    This event fires when you deprecate a strategy. The data property contains the name of the deprecated strategy.

    example event: strategy-deprecated
    {
    "id": 934,
    "type": "strategy-deprecated",
    "createdBy": "user@company.com",
    "createdAt": "2022-06-02T12:21:45.041Z",
    "data": {
    "name": "new-strategy"
    },
    "preData": null,
    "tags": [],
    "featureName": null,
    "project": null,
    "environment": null
    }

    strategy-reactivated

    This event fires when you bring reactivate a deprecated strategy. The data property contains the name of the reactivated strategy.

    example event: strategy-reactivated
    {
    "id": 935,
    "type": "strategy-reactivated",
    "createdBy": "user@company.com",
    "createdAt": "2022-06-02T12:21:49.010Z",
    "data": {
    "name": "new-strategy"
    },
    "preData": null,
    "tags": [],
    "featureName": null,
    "project": null,
    "environment": null
    }

    strategy-import

    This event fires when you import a strategy as part of an import job. The data property contains the strategy's configuration.

    example event: strategy-import
    {
    "id": 29,
    "type": "strategy-import",
    "createdBy": "import-API-token",
    "createdAt": "2022-06-03T11:30:40.583Z",
    "data": {
    "name": "gradualRolloutSessionId",
    "description": "Gradually activate feature toggle. Stickiness based on session id.",
    "parameters": [
    {
    "name": "percentage",
    "type": "percentage",
    "description": "",
    "required": false
    },
    {
    "name": "groupId",
    "type": "string",
    "description": "Used to define a activation groups, which allows you to correlate across feature toggles.",
    "required": true
    }
    ],
    "deprecated": true
    },
    "preData": null,
    "tags": [],
    "featureName": null,
    "project": null,
    "environment": null
    }

    drop-strategies

    This event fires when you delete existing strategies as part of an important job. The data.name property will always be "all-strategies".

    example event: drop-strategies
    {
    "id": 28,
    "type": "drop-strategies",
    "createdBy": "import-API-token",
    "createdAt": "2022-06-03T11:30:40.579Z",
    "data": {
    "name": "all-strategies"
    },
    "preData": null,
    "tags": [],
    "featureName": null,
    "project": null,
    "environment": null
    }

    Context field events

    context-field-created

    This event fires when you create a context field. The data property contains the context field configuration.

    example event: context-field-created
    {
    "id": 937,
    "type": "context-field-created",
    "createdBy": "user@company.com",
    "createdAt": "2022-06-02T13:17:17.499Z",
    "data": {
    "name": "new-context-field",
    "description": "this context field is for describing events",
    "legalValues": [],
    "stickiness": false
    },
    "preData": null,
    "tags": [],
    "featureName": null,
    "project": null,
    "environment": null
    }

    context-field-updated

    This event fires when you update a context field. The data property contains the new context field configuration.

    example event: context-field-updated
    {
    "id": 939,
    "type": "context-field-updated",
    "createdBy": "user@company.com",
    "createdAt": "2022-06-02T13:19:19.422Z",
    "data": {
    "name": "new-context-field",
    "description": "this context field is for describing events",
    "legalValues": [
    {
    "value": "0fcf7d07-276c-41e1-a207-e62876d9c949",
    "description": "Red team"
    },
    {
    "value": "176ab647-4d50-41bf-afe0-f8b856d9bbb9",
    "description": "Blue team"
    }
    ],
    "stickiness": false
    },
    "preData": null,
    "tags": [],
    "featureName": null,
    "project": null,
    "environment": null
    }

    context-field-deleted

    This event fires when you delete a context field. The data property contains the name of the deleted context field.

    example event: context-field-deleted
    {
    "id": 940,
    "type": "context-field-deleted",
    "createdBy": "user@company.com",
    "createdAt": "2022-06-02T13:20:41.386Z",
    "data": {
    "name": "new-context-field"
    },
    "preData": null,
    "tags": [],
    "featureName": null,
    "project": null,
    "environment": null
    }

    Project events

    project-created

    This event fires when you create a project. The data property contains the project configuration.

    example event: project-created
    {
    "id": 905,
    "type": "project-created",
    "createdBy": "user@company.com",
    "createdAt": "2022-05-31T14:16:14.498Z",
    "data": {
    "id": "my-other-project",
    "name": "my other project",
    "description": "a project for important work"
    },
    "preData": null,
    "tags": [],
    "featureName": null,
    "project": "my-other-project",
    "environment": null
    }

    project-updated

    This event fires when you update a project's configuration. The data property contains the new project configuration. The preData property contains the previous project configuration.

    example event: project-updated
    {
    "id": 941,
    "type": "project-updated",
    "createdBy": "user@company.com",
    "createdAt": "2022-06-02T13:23:55.025Z",
    "data": {
    "id": "my-other-project",
    "name": "my other project",
    "description": "a project for important work!"
    },
    "preData": {
    "id": "my-other-project",
    "name": "my other project",
    "health": 50,
    "createdAt": "2022-05-31T14:16:14.483Z",
    "updatedAt": "2022-06-02T12:30:48.095Z",
    "description": "a project for important work"
    },
    "tags": [],
    "featureName": null,
    "project": "my-other-project",
    "environment": null
    }

    project-deleted

    This event fires when you delete a project. The project property contains the name of the deleted project.

    example event: project-deleted
    {
    "id": 944,
    "type": "project-deleted",
    "createdBy": "user@company.com",
    "createdAt": "2022-06-02T13:25:53.820Z",
    "data": null,
    "preData": null,
    "tags": [],
    "featureName": null,
    "project": "my-other-project",
    "environment": null
    }

    project-import

    This event fires when you import a project. The data property contains the project's configuration details.

    example event: project-import
    {
    "id": 35,
    "type": "project-import",
    "createdBy": "import-API-token",
    "createdAt": "2022-06-03T11:30:40.591Z",
    "data": {
    "id": "default",
    "name": "Default",
    "description": "Default project",
    "createdAt": "2022-06-03T09:30:40.587Z",
    "health": 100,
    "updatedAt": "2022-06-03T11:30:40.587Z"
    },
    "preData": null,
    "tags": [],
    "featureName": null,
    "project": null,
    "environment": null
    }

    drop-projects

    This event fires when you delete existing projects as part of an import job. The data.name property will always be "all-projects".

    example event: drop-projects
    {
    "id": 33,
    "type": "drop-projects",
    "createdBy": "import-API-token",
    "createdAt": "2022-06-03T11:30:40.586Z",
    "data": {
    "name": "all-projects"
    },
    "preData": null,
    "tags": [],
    "featureName": null,
    "project": null,
    "environment": null
    }

    Tag events

    tag-created

    This event fires when you create a new tag. The data property contains the tag that was created.

    example event: tag-created
    {
    "id": 959,
    "type": "feature-tagged",
    "createdBy": "user@company.com",
    "createdAt": "2022-06-03T10:13:39.401Z",
    "data": {
    "type": "heartag",
    "value": "tag-value"
    },
    "preData": null,
    "tags": [],
    "featureName": "new-feature",
    "project": null,
    "environment": null
    }

    tag-deleted

    This event fires when you delete a tag. The data property contains the tag that was deleted.

    example event: tag-deleted
    {
    "id": 957,
    "type": "tag-deleted",
    "createdBy": "user@company.com",
    "createdAt": "2022-06-03T10:12:17.310Z",
    "data": {
    "type": "heartag",
    "value": "tag-value"
    },
    "preData": null,
    "tags": [],
    "featureName": null,
    "project": null,
    "environment": null
    }

    tag-import

    This event fires when you import a tag as part of an import job. The data property contains the imported tag.

    example event: tag-import
    {
    "id": 41,
    "type": "tag-import",
    "createdBy": "import-API-token",
    "createdAt": "2022-06-03T11:30:40.603Z",
    "data": {
    "type": "simple",
    "value": "tag1"
    },
    "preData": null,
    "tags": [],
    "featureName": null,
    "project": null,
    "environment": null
    }

    drop-tags

    This event fires when you delete existing tags as part of an import job. The data.name property will always be "all-tags".

    example event: drop-tags
    {
    "id": 37,
    "type": "drop-tags",
    "createdBy": "import-API-token",
    "createdAt": "2022-06-03T11:30:40.596Z",
    "data": {
    "name": "all-tags"
    },
    "preData": null,
    "tags": [],
    "featureName": null,
    "project": null,
    "environment": null
    }

    Tag type events

    tag-type-created

    This event fires when you create a new tag type. The data property contains the tag type configuration.

    example event: tag-type-created
    {
    "id": 945,
    "type": "tag-type-created",
    "createdBy": "user@company.com",
    "createdAt": "2022-06-02T13:27:01.235Z",
    "data": {
    "name": "new-tag-type",
    "description": "event testing"
    },
    "preData": null,
    "tags": [],
    "featureName": null,
    "project": null,
    "environment": null
    }

    tag-type-updated

    This event fires when you update a tag type. The data property contains the new tag type configuration.

    example event: tag-type-updated
    {
    "id": 946,
    "type": "tag-type-updated",
    "createdBy": "user@company.com",
    "createdAt": "2022-06-02T13:27:31.126Z",
    "data": {
    "name": "new-tag-type",
    "description": "This tag is for testing events."
    },
    "preData": null,
    "tags": [],
    "featureName": null,
    "project": null,
    "environment": null
    }

    tag-type-deleted

    This event fires when you delete a tag type. The data property contains the name of the deleted tag type.

    example event: tag-type-deleted
    {
    "id": 947,
    "type": "tag-type-deleted",
    "createdBy": "user@company.com",
    "createdAt": "2022-06-02T13:27:37.277Z",
    "data": {
    "name": "new-tag-type"
    },
    "preData": null,
    "tags": [],
    "featureName": null,
    "project": null,
    "environment": null
    }

    tag-type-import

    This event fires when you import a tag type as part of an import job. The data property contains the imported tag.

    example event: tag-type-import
    {
    "id": 40,
    "type": "tag-type-import",
    "createdBy": "import-API-token",
    "createdAt": "2022-06-03T11:30:40.599Z",
    "data": {
    "name": "custom-tag-type",
    "description": "custom tag type",
    "icon": null
    },
    "preData": null,
    "tags": [],
    "featureName": null,
    "project": null,
    "environment": null
    }

    drop-tag-types

    This event fires when you drop all existing tag types as part of a configuration import. The data.name property will always be "all-tag-types".

    example event: drop-tag-types
    {
    "id": 38,
    "type": "drop-tag-types",
    "createdBy": "import-API-token",
    "createdAt": "2022-06-03T11:30:40.596Z",
    "data": {
    "name": "all-tag-types"
    },
    "preData": null,
    "tags": [],
    "featureName": null,
    "project": null,
    "environment": null
    }

    Integration events

    addon-config-created

    This event fires when you create an integration configuration. The data property contains the provider type.

    example event: addon-config-created
    {
    "id": 960,
    "type": "addon-config-created",
    "createdBy": "user@company.com",
    "createdAt": "2022-06-03T10:15:45.040Z",
    "data": {
    "provider": "webhook"
    },
    "preData": null,
    "tags": [],
    "featureName": null,
    "project": null,
    "environment": null
    }

    addon-config-updated

    This event fires when you update an integration configuration. The data property contains the integration's ID and provider type.

    example event: addon-config-updated
    {
    "id": 961,
    "type": "addon-config-updated",
    "createdBy": "user@company.com",
    "createdAt": "2022-06-03T10:16:11.732Z",
    "data": {
    "id": "2",
    "provider": "webhook"
    },
    "preData": null,
    "tags": [],
    "featureName": null,
    "project": null,
    "environment": null
    }

    addon-config-deleted

    This event fires when you update an integration configuration. The data property contains the integration's ID.

    example event: addon-config-deleted
    {
    "id": 964,
    "type": "addon-config-deleted",
    "createdBy": "user@company.com",
    "createdAt": "2022-06-03T10:16:59.723Z",
    "data": {
    "id": "2"
    },
    "preData": null,
    "tags": [],
    "featureName": null,
    "project": null,
    "environment": null
    }

    User events

    user-created

    This event fires when you create a new user. The data property contains the user's information.

    example event: user-created
    {
    "id": 965,
    "type": "user-created",
    "createdBy": "user@company.com",
    "createdAt": "2022-06-03T10:23:47.713Z",
    "data": {
    "id": 44,
    "name": "New User Name",
    "email": "newuser@company.com"
    },
    "preData": null,
    "tags": [],
    "featureName": null,
    "project": null,
    "environment": null
    }

    user-updated

    This event fires when you update a user. The data property contains the updated user information; the preData property contains the previous state of the user's information.

    example event: user-updated
    {
    "id": 967,
    "type": "user-updated",
    "createdBy": "user@company.com",
    "createdAt": "2022-06-03T10:24:26.301Z",
    "data": {
    "id": 44,
    "name": "New User's Name",
    "email": "newuser@company.com"
    },
    "preData": {
    "id": 44,
    "name": "New User Name",
    "email": "newuser@company.com"
    },
    "tags": [],
    "featureName": null,
    "project": null,
    "environment": null
    }

    user-deleted

    This event fires when you delete a user. The preData property contains the deleted user's information.

    example event: user-deleted
    {
    "id": 968,
    "type": "user-deleted",
    "createdBy": "user@company.com",
    "createdAt": "2022-06-03T10:24:49.153Z",
    "data": null,
    "preData": {
    "id": 44,
    "name": "New User's Name",
    "email": "newuser@company.com"
    },
    "tags": [],
    "featureName": null,
    "project": null,
    "environment": null
    }

    Environment events

    environment-import

    This event fires when you import an environment (custom or otherwise) as part of an import job. The data property contains the configuration of the imported environment.

    example event: environment-import
    {
    "id": 24,
    "type": "environment-import",
    "createdBy": "import-API-token",
    "createdAt": "2022-06-03T11:30:40.557Z",
    "data": {
    "name": "custom-environment",
    "type": "test",
    "sortOrder": 9999,
    "enabled": true,
    "protected": false
    },
    "preData": null,
    "tags": [],
    "featureName": null,
    "project": null,
    "environment": null
    }

    drop-environments

    This event fires when you delete existing environments as part of an import job. The data.name property will always be "all-environments".

    example event: drop-environments
    {
    "id": 21,
    "type": "drop-environments",
    "createdBy": "import-API-token",
    "createdAt": "2022-06-03T11:30:40.549Z",
    "data": {
    "name": "all-projects"
    },
    "preData": null,
    "tags": [],
    "featureName": null,
    "project": null,
    "environment": null
    }

    Segment events

    segment-created

    This event fires when you create a segment. The data property contains the newly created segment.

    example event: segment-created
    {
    "id": 969,
    "type": "segment-created",
    "createdBy": "user@company.com",
    "createdAt": "2022-06-03T10:29:43.977Z",
    "data": {
    "id": 5,
    "name": "new segment",
    "description": "this segment is for events",
    "constraints": [
    {
    "values": ["appA", "appB", "appC"],
    "inverted": false,
    "operator": "IN",
    "contextName": "appName",
    "caseInsensitive": false
    }
    ],
    "createdBy": "user@company.com",
    "createdAt": "2022-06-03T10:29:43.974Z"
    },
    "preData": null,
    "tags": [],
    "featureName": null,
    "project": null,
    "environment": null
    }

    segment-updated

    This event fires when you update a segment's configuration. The data property contains the new segment configuration; the preData property contains the previous segment configuration.

    example event: segment-updated
    {
    "id": 970,
    "type": "segment-updated",
    "createdBy": "user@company.com",
    "createdAt": "2022-06-03T10:29:59.892Z",
    "data": {
    "id": 5,
    "name": "new segment",
    "description": "this segment is for events",
    "constraints": [],
    "createdBy": "user@company.com",
    "createdAt": "2022-06-03T10:29:43.974Z"
    },
    "preData": {
    "id": 5,
    "name": "new segment",
    "createdAt": "2022-06-03T10:29:43.974Z",
    "createdBy": "user@company.com",
    "constraints": [
    {
    "values": ["appA", "appB", "appC"],
    "inverted": false,
    "operator": "IN",
    "contextName": "appName",
    "caseInsensitive": false
    }
    ],
    "description": "this segment is for events"
    },
    "tags": [],
    "featureName": null,
    "project": null,
    "environment": null
    }

    segment-deleted

    This event fires when you delete a segment.

    example event: segment-deleted
    {
    "id": 971,
    "type": "segment-deleted",
    "createdBy": "user@company.com",
    "createdAt": "2022-06-03T10:30:08.128Z",
    "data": {},
    "preData": null,
    "tags": [],
    "featureName": null,
    "project": null,
    "environment": null
    }

    Suggest changes events

    suggest-change-created

    This event fires when you create a change-request draft.

    example event: suggest-change-created
    {
    "id": 971,
    "type": "suggest-change-created",
    "createdBy": "user@company.com",
    "createdAt": "2022-06-03T10:30:08.128Z",
    "data": {},
    "preData": null,
    "tags": [],
    "featureName": null,
    "project": null,
    "environment": null
    }
    - - + + \ No newline at end of file diff --git a/reference/feature-flag-naming-patterns.html b/reference/feature-flag-naming-patterns.html index a27e2c9fc2..5d1d76cb86 100644 --- a/reference/feature-flag-naming-patterns.html +++ b/reference/feature-flag-naming-patterns.html @@ -20,15 +20,15 @@ - - + +

    Feature Flag Naming Patterns

    Availability

    Feature flag naming patterns is an in-development, enterprise-only feature.

    A feature flag naming pattern is JavaScript regular expression that is used to validate the name of a feature flag before the flag can be created.

    The pattern is defined in the project settings and is enforced when creating a new feature flag. The pattern is also enforced when creating a new feature flag via the API.

    Patterns are implicitly anchored to the start and end of the string. This means that a pattern is matched against the entire new feature flag name, and not just any subset of it, as if the pattern was surrounded by ^ and $. In other words, the pattern [a-z]+ will be interpreted as ^[a-z]+$ and will match "somefeature", but will not match "some.other.feature".

    Feature flag naming patterns are defined on a per-project basis.

    In addition to the pattern itself, you can also define a an example and a description of the pattern. If defined, both the example and the description will be shown to the user when they are creating a new feature flag.

    Overview

    The naming pattern consists of three parts:

    Pattern (required)
    The regular expression that is used to validate the name of the feature flag. Must be a valid regular expression. Flags (such as case insensitivity) are not available.
    Example (optional)
    An example of a name that is valid according to the provided pattern. Note: the example must be valid against the described pattern for it to be saved.
    Description (optional)
    Any additional text that you would like to display to users to provide extra information. This can be anything that you think they would find useful and can be as long or short as you want.

    For instance, you might define a pattern that requires all feature flags to follow a specific pattern, such as ^(red|blue|green|yellow)\.[a-z-]+\.[0-9]+$. You could then provide an example of a valid feature flag name (for instance "blue.water-gun.64") and a description of what the pattern should reflect: "<team>.<feature>.<ticket>".

    - - + + \ No newline at end of file diff --git a/reference/feature-toggle-types.html b/reference/feature-toggle-types.html index 50281dc132..58c07d0b69 100644 --- a/reference/feature-toggle-types.html +++ b/reference/feature-toggle-types.html @@ -20,15 +20,15 @@ - - + +

    Feature Toggle Types

    This feature was introduced in Unleash v3.5.0.

    You can use feature toggles to support different use cases, each with their own specific needs. Heavily inspired by Pete Hodgson's article on feature toggles, Unleash introduced the concept of feature toggle types in version 3.5.0.

    A feature toggle's type affects only two things:

    1. It gives the toggle an appropriate icon
    2. The toggle's expected lifetime changes

    Aside from this, there are no differences between the toggle types and you can always change the type of a toggle after you have created it.

    Classifying feature toggles by their type makes it easier for you manage them: the toggles get different icons in the toggle list and you can sort the toggles by their types.

    Five feature toggles, each of a different type, showing the different icons that Unleash uses for each toggle type.

    A toggle's type also helps Unleash understand the toggle's expected lifetime.

    Feature toggle types

    Here's the list of the feature toggle types that Unleash supports together with their intended use case and expected lifetime:

    Feature toggle typeUsed to ...Expected lifetime
    ReleaseEnable trunk-based development for teams practicing Continuous Delivery.40 days
    ExperimentPerform multivariate or A/B testing.40 days
    OperationalControl operational aspects of the system's behavior.7 days
    Kill switchGracefully degrade system functionality. You can read about kill switch best practices on our blog.Permanent
    PermissionChange the features or product experience that certain users receive.Permanent

    Expected lifetimes

    info

    The ability to update a feature toggle type's expected lifetime is currently in development. We expect to release it in one of the upcoming releases.

    A feature toggle's expected lifetime is an indicator of how long Unleash expects toggles of that type to be around. Some feature toggles are meant to live for a few weeks as you work on new functionality, while others stick around for much longer. As a part of good code hygiene, you should clean up your feature toggles when they have served their purpose. This is further explored in the document on technical debt.

    Each feature toggle type in Unleash has an assigned expected lifetime, after which the system will consider this feature potentially stale. The reasoning behind each type's expected lifetime is detailed in this blog post on best practices for feature toggle lifetimes.

    Unleash admins can change the expected lifetime of Unleash's feature types from the Unleash configuration menu.

    Deprecating feature toggles

    You can mark feature toggles as stale. This is a way to deprecate a feature toggle without removing the active configuration for connected applications. Use this to signal that you should stop using the feature in your applications. Stale toggles will show as stale in the "technical debt dashboard".

    When you mark a toggle as stale, Unleash will emit an event. You can use an integration to integrate this with your systems, for instance to post a message in a Slack channel.

    Additionally, with some extra work, you can also use the stale property to:

    • Inform developers that a toggle is stale while they're developing.
    • Break a project build if the code contains stale feature toggles.
    • Send automatic PRs to remove usage of toggles that have served their purpose.
    - - + + \ No newline at end of file diff --git a/reference/feature-toggle-variants.html b/reference/feature-toggle-variants.html index 5fc55d7208..1ce5d1d162 100644 --- a/reference/feature-toggle-variants.html +++ b/reference/feature-toggle-variants.html @@ -20,8 +20,8 @@ - - + + @@ -29,7 +29,7 @@

    Feature Toggle Variants

    Availability

    Feature toggle variants were first introduced in Unleash 3.2. In Unleash 4.21, variants were updated to be environment-dependent, meaning the same feature could have different variant configurations in different environments.

    Every toggle in Unleash can have something called variants. Where feature toggles allow you to decide which users get access to a feature, toggle variants allow you to further split those users into segments. Say, for instance, that you're testing out a new feature, such as an alternate sign-up form. The feature toggle would expose the form only to a select group of users. The variants could decide whether the user sees a blue or a green submit button on that form.

    Variants facilitate A/B testing and experimentation by letting you create controlled and measurable experiments. Check our blog post on using Unleash for A/B/n experiments for some more insights into how you can set it up.

    What are variants?

    Whenever you create a feature toggle, you can assign it any number of variants. This is commonly done in cases where you want to serve your users different versions of a feature to see which performs better. A feature can have different variants in each of its environments.

    A variant has four components that define it:

    • a name:

      This must be unique among the toggle's variants in the specified environment. When working with a feature with variants in a client, you will typically use the variant's name to find out which variant it is.

    • a weight:

      The weight is the likelihood of any one user getting this specific variant. See the weights section for more info.

    • an optional payload:

      A variant can also have an associated payload. Use this to deliver more data or context. See the payload section for a more details.

    • an optional override

      Overrides let you specify that certain users (as identified either by their user ID or by another custom stickiness value) will always get this variant, regardless of the variant's weight.

    A form for adding new variants. It has fields for name, weight, payload, and overrides.

    Variant weight

    A variant's weight determines how likely it is that a user will receive that variant. It is a numeric value between 0 and 100 (inclusive) with one decimal's worth of precision.

    When you have multiple variants, the sum of all their weights must add up to exactly 100. Depending on the weight type, Unleash may automatically determine the weight of the new variant and balance it out with the other variants.

    Weight types and calculation

    There are two kinds of variant weight types: variable and fixed. Unleash requires you to always have at least one variable weight variant.

    The default weight type is variable. Variable weight variants will adjust their weight based on the number of other variable weight variants and whatever weight is not used up by fixed weight variants.

    Fixed weight variants have a set weight which will not change. All fixed weight variants' weights can not add up to more than 100.

    To calculate what the weight of a variable weight variant is, Unleash first subtracts the sum of fixed weights from 100 and then distributes the remaining weight evenly among the variable weight variants.

    For instance, if you have three variable weight variants and two fixed weight variants weighted at 25 and 15 respectively, Unleash will:

    1. Subtract the fixed weight from the total available: 100 - 40 = 60
    2. Divide the remainder by the number of variable weight variants: 60 / 3 = 20
    3. Assign each variable weight variant the same (up to rounding differences) weight: 20%

    In the example above, 60 divides cleanly by three. In cases where the remainder doesn't divide evenly among the variable weight variants, Unleash will distribute it as evenly as it can to one decimal's precision. If you have three variable weight variants, they will be weighted at 33.4, 33.3, and 33.3 respectively, so that it adds up to 100.0.

    Overrides

    note

    Overrides are intended to be used for one-off cases and during development and may not be suitable for other use cases.

    The weighting system automatically assigns users to a specific group for you. If you want to make sure that a specific user or group of users receives a certain variant, though, you can use the override functionality to achieve that.

    When adding an override, you choose a field from the Unleash Context and specify that if a context contains one of a given list of values, then the current variant should always activate.

    You can use both standard and custom context fields when creating overrides.

    Each variant can have multiple overrides, so you can use any number of context fields you want to create overrides. When using multiple overrides, each user only has to match one of them. In other words, if you use the following overrides, the users with IDs aa599 and aa65 and any users who use the client with ID 123abc will receive the specified variant:

    • userId: aa599, aa65
    • clientId: 123abc

    Note that if multiple variants use overrides that affect the same user, then Unleash can not guarantee which override will take effect. We recommend that you do not use multiple overrides that can conflict in this way, as it will probably not do what you expect.

    Variant payload

    Each variant can have an associated payload. Use this to add more context or data to a payload that you can access on the client, such as a customized message or other information.

    Unleash currently supports these payload types:

    • JSON
    • CSV
    • String

    Variant stickiness

    Variant stickiness is calculated on the received user and context, as described in the stickiness chapter. This ensures that the same user will consistently see the same variant barring overrides and weight changes. If no context data is provided, the traffic will be spread randomly for each request.

    How do I configure variants

    In the Unleash UI, you can configure variants by navigating to the feature view, and then choosing the 'Variants' tab.

    toggle_variants

    The disabled variant

    When a toggle has no variants or when a toggle is disabled for a user, Unleash will return it with variant data that looks like this:

    {
    "name": "disabled",
    "enabled": false
    }

    This is a fallback variant that Unleash uses to represent the lack of a variant.

    Note: The actual representation of the built-in fallback variant in the client SDK will vary slightly, to honor best practices in various languages.

    Client SDK Support

    To make use of toggle variants, you need to use a compatible client. Client SDK with variant support:

    If you would like to give feedback on this feature, experience issues or have questions, please feel free to open an issue on GitHub.

    - - + + \ No newline at end of file diff --git a/reference/feature-toggles.html b/reference/feature-toggles.html index db7d796f8b..3e13bf18bf 100644 --- a/reference/feature-toggles.html +++ b/reference/feature-toggles.html @@ -20,15 +20,15 @@ - - + +

    Feature Toggles

    Feature toggles are the central concept that we built Unleash around. In Unleash, feature toggles are organized within projects. Feature toggles can have different activation strategies for each of their project's environments, and can also be given variants to facilitate A/B testing.

    Configuration options

    Each feature toggle has the following configuration options

    OptionRequired?Default valueDescription
    nameYesN/AThe feature toggle's name. Must be URL-friendly according to section 2.3 of RFC 3986 and must be unique within your Unleash instance. Must be between 1 and 100 characters long, inclusive.
    feature toggle typeYesReleaseThe feature toggle's type.
    projectYesThe default project. When created from a project page in the admin UI, that project will be the default value instead.The project that should contain the feature toggle.
    descriptionNoN/AA description of the feature toggle's purpose.
    enable impression dataYesNoWhether to enable impression data for this toggle or not.

    Environments

    You probably won't want to use the same configuration to enable a toggle in development as you do in production. That's why feature toggles have different activation strategy configurations for each environment.

    You can enable and disable a toggle independently in each of the project's environments. When you disable a toggle in an environment, it will always evaluate to false in that environment. When you enable a toggle in an environment, the toggle will evaluate to true or false depending on its activation strategies.

    Refer to the documentation on environments for more details on how environments work.

    Activation strategies

    To enable a feature in an environment, you must assign it at least one activation strategy. A feature toggle's activation strategies determine whether the toggle gets enabled for a particular Unleash context (typically a user). When using multiple strategies in a single environment, only a single strategy needs to evaluate to true for the toggle to get enabled for a user. Whenever Unleash evaluates a toggle, it will evaluate strategies in the current environment until one of them resolves to true. If no strategies resolve to true, then the toggle's value is false.

    Refer to the activation strategies documentation for a detailed description of all the built-in strategies.

    Variants

    Variants adds another dimension of flexibility to feature toggles. Each feature toggle can be assigned any number of variants which will then get distributed amongst your users based on your choice of context field. You can find out more about variants in the variants docs.

    Creating toggles with payloads

    While variants are most often used for A/B testing and the like, you can also use variants to assign a constant payload to a toggle. If you give a toggle only a single variant and that variant has a payload, then all users variants will receive that payload.

    - - + + \ No newline at end of file diff --git a/reference/front-end-api.html b/reference/front-end-api.html index c9cb27070f..b755ade753 100644 --- a/reference/front-end-api.html +++ b/reference/front-end-api.html @@ -20,15 +20,15 @@ - - + +

    Front-end API access

    Availability

    The Unleash front-end API was released in Unleash 4.18. You can read more in the Unleash 4.18 release blog post.

    The Unleash front-end API offers a simplified workflow for connecting a client-side (front-end) applications to Unleash. It provides the exact same API as Unleash edge and the Unleash proxy - deprecated. The front-end API is a quick and easy way to add Unleash to single-page applications and mobile apps.

    Compared to using Unleash Edge, using the Unleash front-end API has both benefits and drawbacks. The benefits are:

    • You don't need to configure and run Unleash Edge. The front-end API is part of Unleash itself and not an external process. All clients will work exactly the same as they would with Unleash Edge.

    On the other hand, using the front-end API has the following drawbacks compared to using Unleash Edge:

    • It can't handle a large number of requests per second. Because the front-end API is part of Unleash, you can't scale it horizontally the way you can scale Unleash Edge.
    • It sends client details to your Unleash instance. Unleash only stores these details in its short-term runtime cache, but this can be a privacy issue for some use cases.

    These points make the Unleash front-end API best suited for development purposes and applications that don’t receive a lot of traffic, such as internal dashboards. However, because the API is identical to the Unleash Edge API, you can go from one to the other at any time. As such, you can start out by using the front-end API and switch to using Unleash Edge when you need it.

    Using the Unleash front-end API

    When using the front-end API in an SDK, there's three things you need to configure.

    Front-end API tokens

    As a client-side API, you should use a front-end API token to interact with it. Refer to the how to create API tokens guide for steps on how to create API tokens.

    Cross-origin resource sharing (CORS) configuration

    You need to allow traffic from your application domains to use the Unleash front-end API with web and hybrid mobile applications. You can update the front-end API CORS settings from the Unleash UI under admin > CORS or by using the API.

    API URL

    The client needs to point to the correct API endpoint. The front-end API is available at <your-unleash-instance>/api/frontend.

    API token

    You can create appropriate token, with type FRONTEND on <YOUR_UNLEASH_URL>/admin/api/create-token page or with a request to /api/admin/api-tokens. See our guide on how to create API tokens for more details.

    Refresh interval for tokens

    Internally, Unleash creates a new Unleash client for each token it receives. Each client is configured with the project and environment specified in the token.

    Each client updates its feature toggle configuration at a specified refresh interval plus a random offset between 0 and 10 seconds. By default, the refresh interval is set to 10 seconds. The random offset is used to stagger incoming requests to avoid a large number of clients all querying the database simultaneously. A new, random offset is used for every update.

    The refresh interval is specified in milliseconds and can be set by using the FRONTEND_API_REFRESH_INTERVAL_MS environment variable or by using the frontendApi.refreshIntervalInMs configuration option in code.

    - - + + \ No newline at end of file diff --git a/reference/impression-data.html b/reference/impression-data.html index 93dd888533..de54294016 100644 --- a/reference/impression-data.html +++ b/reference/impression-data.html @@ -20,15 +20,15 @@ - - + +

    Impression Data

    Availability

    The impression data feature was introduced in Unleash 4.7. It is available in the JavaScript-based proxy clients and in some server-side SDKs. Please refer to the SDK compatibility table for an overview of server-side SDKs that support it.

    Unleash can provide you with impression data about the toggles in your application. Impression data contains information about a specific feature toggle activation check: The client SDK will emit an impression event when it calls isEnabled or getVariant. Some front-end SDKs emit impression events only when a toggle is enabled.

    Front-end SDKs and disabled toggles

    Older versions of the front-end SDKs and other SDKs that connect the Unleash proxy or the Unleash front-end API would not emit impression events when a toggle is disabled.

    This is because impression data is a per-toggle setting and the Proxy and front-end API only transmit information about toggles that are enabled. As such, the SDK will never know that it should emit an impression event if a toggle is disabled.

    Some of the front-end SDKs now include a include a configuration property that lets you turn on impression data for all toggles regardless of whether they're enabled or not.

    Impression data was designed to make it easier for you to collect analytics data, perform A/B tests, and enrich experiments in your applications. It contains information about the feature toggle and the related Unleash Context.

    Impression data is opt-in on a per-toggle basis. Unleash will not emit impression events for toggles not marked as such. Once you've turned impression data on for a toggle, you can start listening for impression events in your client SDK.

    Impression event data

    There's two types of impression events you can listen for:

    The getVariant event contains all the information found in an isEnabled event in addition to extra data that's only relevant to getVariant calls.

    This table describes all the properties on the impression events:

    Property nameDescriptionEvent type
    eventTypeThe type of the event: isEnabled or getVariantAll
    eventIdA globally unique id (GUID) assigned to this event.All
    contextA representation of the current Unleash Context.All
    enabledWhether the toggle was enabled or not at when the client made the request.All
    featureNameThe name of the feature toggle.All
    variantThe name of the active variantgetVariant events only

    Example isEnabled event

    {
    eventType: 'isEnabled',
    eventId: '84b41a43-5ba0-47d8-b21f-a60a319607b0',
    context: {
    sessionId: 54085233,
    appName: 'my-webapp',
    environment: 'default'
    },
    enabled: true,
    featureName: 'my-feature-toggle',
    }

    Example getVariant event

    {
    eventType: 'getVariant',
    eventId: '84b41a43-5ba0-47d8-b21f-a60a319607b0',
    context: {
    sessionId: 54085233,
    appName: 'my-webapp',
    environment: 'default'
    },
    enabled: true,
    featureName: 'my-feature-toggle',
    variant: 'variantA'
    }

    Enabling impression data

    Impression data is strictly an opt-in feature and must be enabled on a per-toggle basis. You can enable and disable it both when you create a toggle and when you edit a toggle.

    You can enable impression data via the impression data toggle in the admin UI's toggle creation form. You can also go via the the API, using the impressionData option. For more detailed instructions, see the section on enabling impression data in the how-to guide for capturing impression data.

    A feature toggle creation form. At the end of the form is a heading that says &quot;Impression data&quot;, a short paragraph that describes the feature, and a toggle to opt in or out of it.

    Example setup

    The exact setup will vary depending on your client SDK. The below example configures the [Unleash Proxy client../reference/sdks/javascript-browser) to listen for impression events and log them to the console. If "my-feature-toggle" is configured to emit impression data, then it will trigger an impression event as soon as Unleash is ready.

    const unleash = new UnleashClient({
    url: 'https://eu.unleash-hosted.com/hosted/proxy',
    clientKey: 'your-proxy-key',
    appName: 'my-webapp',
    });

    unleash.start();

    unleash.on('ready', () => {
    unleash.isEnabled('my-feature-toggle');
    });

    unleash.on('impression', (event) => {
    // Capture the event here and pass it internal data lake or analytics provider
    console.log(event);
    });
    - - + + \ No newline at end of file diff --git a/reference/integrations.html b/reference/integrations.html index 026e76fd17..50f0042ef2 100644 --- a/reference/integrations.html +++ b/reference/integrations.html @@ -20,15 +20,15 @@ - - + +

    Integrations

    Availability

    Unleash integrations were introduced in Unleash v3.11.0.

    Integrations were previously known as addons.

    Unleash integrations allows you to extend Unleash with new functionality and to connect to external applications.

    Unleash has two types of integrations: Integrations that allow you to listen to changes in Unleash and trigger updates in other systems (for instance via webhooks or direct integrations) and integrations that communicate with Unleash (such as the Jira integrations).

    Official integrations

    Unleash currently supports the following integrations out of the box:

    • Datadog - Allows Unleash to post Updates to Datadog when a feature toggle is updated.
    • Jira Cloud - Allows you to create, view and manage Unleash feature flags directly from a Jira Cloud issue
    • Jira Server - Allows you to create and link Unleash feature flags directly from a Jira Server issue
    • Microsoft Teams - Allows Unleash to post updates to Microsoft Teams.
    • Slack App - The Unleash Slack App posts messages to the selected channels in your Slack workspace.
    • Webhook - A generic way to post messages from Unleash to third party services.
    Missing an integration? Request it!

    If you're looking for an integration that Unleash doesn't have at the moment, you can fill out this integration request form to register it with us.

    Deprecated integrations

    These integrations are deprecated and will be removed in a future release:

    • Slack - Allows Unleash to post updates to Slack. Please try the new Slack App integration instead.

    Community integrations

    Our wonderful community has also created the following integrations:

    Notes

    When updating or creating a new integration configuration it can take up to one minute before Unleash picks up the new config on all instances due to caching.

    Integration pages

    - - + + \ No newline at end of file diff --git a/reference/integrations/datadog.html b/reference/integrations/datadog.html index fb3a25bcf0..4baa1f253e 100644 --- a/reference/integrations/datadog.html +++ b/reference/integrations/datadog.html @@ -20,15 +20,15 @@ - - + +

    Datadog

    This feature was introduced in Unleash v4.0.0.

    The Datadog integration allows Unleash to post Updates to Datadog when a feature toggle is updated. To set up this integration, you need to set up a webhook connector for your channel. You can follow Submitting events to Datadog on how to do that.

    The Datadog integration will perform a single retry if the HTTP POST against the Datadog Webhook URL fails (either a 50x or network error). Duplicate events may happen, and you should never assume events always comes in order.

    Configuration

    Events

    You can choose to trigger updates for the following events:

    • feature-created
    • feature-updated (*)
    • feature-metadata-updated
    • feature-project-change
    • feature-archived
    • feature-revived
    • feature-strategy-update
    • feature-strategy-add
    • feature-strategy-remove
    • feature-stale-on
    • feature-stale-off
    • feature-environment-enabled
    • feature-environment-disabled
    • feature-environment-variants-updated
    • feature-potentially-stale-on

    *) Deprecated, and will not be used after transition to environments in Unleash v4.3

    Parameters

    Unleash Datadog integration takes the following parameters.

    • Datadog API key - This is a required property. The API key to use to authenticate with Datadog.

    • Datadog Source Type Name - This is an optional property. Sets source_type_name parameter to be included in Datadog events. List of valid api source values

    • Extra HTTP Headers - This is an optional property. Used to set the additional headers when Unleash communicates with Datadog.

    Example:

    {
    "SOME_CUSTOM_HTTP_HEADER": "SOME_VALUE",
    "SOME_OTHER_CUSTOM_HTTP_HEADER": "SOME_OTHER_VALUE"
    }
    Body template availability

    The body template property is available from Unleash 5.6 onwards.

    • Body template - This is an optional property. The template is used to override the body template used by Unleash when performing the HTTP POST. You can format your message using a Mustache template. Refer to the Unleash event types reference to find out which event properties you have access to in the template.

    Example:

    {
    "event": "{{event.type}}",
    "createdBy": "{{event.createdBy}}",
    "featureToggle": "{{event.data.name}}",
    "timestamp": "{{event.data.createdAt}}"
    }

    If you don't specify anything Unleash will send a formatted markdown body.

    Example:

    username created feature toggle (featurename)[http://your.url/projects/projectname/features/featurename] in project *projectname*

    Tags

    Datadog's incoming webhooks are app specific. You will be able to create multiple integrations to support messaging on different apps.

    - - + + \ No newline at end of file diff --git a/reference/integrations/jira-cloud-plugin-installation.html b/reference/integrations/jira-cloud-plugin-installation.html index 50ec3147b9..bfda307b08 100644 --- a/reference/integrations/jira-cloud-plugin-installation.html +++ b/reference/integrations/jira-cloud-plugin-installation.html @@ -20,15 +20,15 @@ - - + +

    Jira Cloud Integration - Installation

    Prerequisites

    Jira Cloud

    For Jira Data Center, check out the Jira Server plugin

    Unleash

    • Unleash v4 or higher

    Required access levels

    Unleash

    You will need an Unleash admin user to configure the access tokens needed to connect the plugin to Unleash.

    This plugin requires an Unleash Pro or an Unleash Enterprise instance.

    We recommend using a service account token for communicating with Unleash. Service accounts are also required to integrate with Change Requests

    Jira

    You will need a Jira admin user.

    Installation

    Start by visiting the Landing page for the Unleash Enterprise For Jira plugin on Atlassian marketplace.

    Atlassian marketplace landing page for the Unleash Enterprise For Jira plugin.

    1. Click the yellow button in the top right corner labeled "Get it now"
    2. Choose which site to install the app to.
    3. Click the blue "Install app" button located at the bottom right.

    Configuring the plugin

    After the plugin has been installed, each project's settings page in Jira will have a menu entry link called "Unleash Project Settings" under the "Apps" menu section.

    Jira Cloud project settings page with the apps menu open. The link to Unleash project settings is highlighted.

    Following that link takes you to the "Unleash Project Settings" configuration page. This is where you specify the connection details (Unleash server URL and access token) for the Unleash server to be used with this particular project.

    Jira Cloud: Unleash project settings. A form with inputs for Unleash URL and an Unleash auth token.

    Once you have configured the connection for the Unleash server, your users should be ready to use the Jira Cloud plugin

    - - + + \ No newline at end of file diff --git a/reference/integrations/jira-cloud-plugin-usage.html b/reference/integrations/jira-cloud-plugin-usage.html index 89303abe9b..ed4ddca63b 100644 --- a/reference/integrations/jira-cloud-plugin-usage.html +++ b/reference/integrations/jira-cloud-plugin-usage.html @@ -20,8 +20,8 @@ - - + + @@ -32,7 +32,7 @@ from Jira.

    Change Requests

    The plugin respects Unleash's change requests. If change requests are turned on in the connected project and the selected environment, the plugin will ask whether you want to create a change request or not.

    If you already have an active change request for that project and that environment, the changes will be added to your existing change request.

    If you confirm that you would like to open a change request, then the plugin will create one for you and present a confirmation dialog.

    When the Change Request has been reviewed and applied in Unleash, the toggle will show the requested state after the next refresh of the issue and toggle status page.

    Disconnecting toggle from Issue

    If a toggle is no longer relevant for your Jira Issue, you can disconnect it using the "disconnect toggle" button. This button is only available if your user has edit permissions for the Jira issue.

    Jira Cloud: issue with a connected toggle. The &#39;disconnect toggle&#39; button (highlighted) is displayed next to the toggle&#39;s name.

    The toggle will be disconnected immediately. However, the plugin will not delete the toggle from Unleash, so you can still reconnect your Jira issue to the same toggle using the "Connect to existing toggle" functionality

    - - + + \ No newline at end of file diff --git a/reference/integrations/jira-server-plugin-installation.html b/reference/integrations/jira-server-plugin-installation.html index f3a73149a7..0c6ce03fa9 100644 --- a/reference/integrations/jira-server-plugin-installation.html +++ b/reference/integrations/jira-server-plugin-installation.html @@ -20,8 +20,8 @@ - - + + @@ -36,7 +36,7 @@ server. Since this is a destructive operation, our plugin will ask for confirmation that you're sure you want to do this.

    A plugin deletion confirmation dialog. It gives you two options: &quot;Delete connection&quot;, and &quot;Cancel&quot;.

    You cannot delete a server that has toggles connected to issues. Instead, you'll get a warning dialog telling you that you'll need to disconnect the toggles from their issues first.

    A warning dialog telling you that you can&#39;t delete a server.

    - - + + \ No newline at end of file diff --git a/reference/integrations/jira-server-plugin-usage.html b/reference/integrations/jira-server-plugin-usage.html index 29dc2b4591..1188d98cee 100644 --- a/reference/integrations/jira-server-plugin-usage.html +++ b/reference/integrations/jira-server-plugin-usage.html @@ -20,8 +20,8 @@ - - + + @@ -33,7 +33,7 @@ from Jira.

    Jira Server - Toggle status

    Disconnecting toggle from Issue

    If a toggle is no longer relevant for your Jira Issue, you can disconnect it using the Disconnect toggle button ( provided your user has edit rights on the issue)

    Jira Server - Disconnect toggle

    Once you click the button, you'll need to confirm the dialog that opens up.

    Jira Server - Disconnect toggle dialog

    If confirmed, the toggle will be disconnected immediately. However, the plugin will not delete the toggle from Unleash, so you can still reconnect your Jira issue to the same toggle using the "Connect to existing toggle" functionality

    - - + + \ No newline at end of file diff --git a/reference/integrations/slack-app.html b/reference/integrations/slack-app.html index f7e8ca9123..daed33ab2d 100644 --- a/reference/integrations/slack-app.html +++ b/reference/integrations/slack-app.html @@ -20,15 +20,15 @@ - - + +

    Slack App

    Availability

    The Slack App integration was introduced in Unleash 5.5.

    The Slack App integration posts messages to a specified set of channels in your Slack workspace. The channels can be public or private, and can be specified on a per-flag basis by using Slack tags.

    Installation

    To install the Slack App integration, follow these steps:

    1. Navigate to the integrations page in the Unleash admin UI (available at the URL /integrations) and select "configure" on the Slack App integration.
    2. On the integration configuration form, use the "install & connect" button.
    3. A new tab will open, asking you to select the Slack workspace where you'd like to install the app.
    4. After successful installation of the Unleash Slack App in your chosen Slack workspace, you'll be automatically redirected to a page displaying a newly generated access token.
    5. Copy this access token and paste it into the Access token field within the integration settings.

    By default, the Unleash Slack App is granted access to public channels. If you want the app to post messages to private channels, you'll need to manually invite it to each of those channels.

    Configuration

    The configuration settings allow you to choose the events you're interested in and whether you want to filter them by projects and environments. You can configure a comma-separated list of channels to post the configured events to. These channels are always notified, regardless of the event type or the presence of Slack tags.

    Events

    You can choose to trigger updates for the following events:

    • addon-config-created
    • addon-config-deleted
    • addon-config-updated
    • api-token-created
    • api-token-deleted
    • change-added
    • change-discarded
    • change-edited
    • change-request-applied
    • change-request-approval-added
    • change-request-approved
    • change-request-cancelled
    • change-request-created
    • change-request-discarded
    • change-request-rejected
    • change-request-sent-to-review
    • context-field-created
    • context-field-deleted
    • context-field-updated
    • feature-archived
    • feature-created
    • feature-deleted
    • feature-environment-disabled
    • feature-environment-enabled
    • feature-environment-variants-updated
    • feature-metadata-updated
    • feature-potentially-stale-on
    • feature-project-change
    • feature-revived
    • feature-stale-off
    • feature-stale-on
    • feature-strategy-add
    • feature-strategy-remove
    • feature-strategy-update
    • feature-tagged
    • feature-untagged
    • group-created
    • group-deleted
    • group-updated
    • project-created
    • project-deleted
    • segment-created
    • segment-deleted
    • segment-updated
    • service-account-created
    • service-account-deleted
    • service-account-updated
    • user-created
    • user-deleted
    • user-updated

    Parameters

    The Unleash Slack App integration takes the following parameters.

    • Access token - This is the only required property. After successful installation of the Unleash Slack App in your chosen Slack workspace, you'll be automatically redirected to a page displaying a newly generated access token. You should copy this access token and paste it into this field.
    • Channels - A comma-separated list of channels to post the configured events to. These channels are always notified, regardless of the event type or the presence of a Slack tag.

    Tags

    Besides the configured channels, you can choose to notify other channels by tagging your feature flags with Slack-specific tags. For instance, if you want the Unleash Slack App to send notifications to the #general channel, add a Slack-type tag with the value "general" (or "#general"; both will work) to your flag. This will ensure that any configured events related to that feature flag will notify the tagged channel in addition to any channels configured on the integration-level.

    To exclusively use tags for determining notification channels, you can leave the "channels" field blank in the integration configuration. Since you can have multiple configurations for the integration, you're free to mix and match settings to meet your precise needs. Before posting a message, all channels for that event, both configured and tagged, are combined and duplicates are removed.

    We have defined two Slack tags for the "new-payment-system" flag. In this example Unleash will post updates to the #notifications and #random channels, along with any channels defined in the integration configuration.
    - - + + \ No newline at end of file diff --git a/reference/integrations/slack.html b/reference/integrations/slack.html index 953df9fc0f..fea17bb696 100644 --- a/reference/integrations/slack.html +++ b/reference/integrations/slack.html @@ -20,15 +20,15 @@ - - + +

    Slack (deprecated)

    Deprecation notice

    This Slack integration is deprecated and will be removed in a future release. We recommend using the new Slack App integration instead.

    This feature was introduced in Unleash v3.11.0.

    The Slack integration allows Unleash to post Updates when a feature toggle is updated. To set up Slack, you need to configure an incoming Slack webhook URL. You can follow Sending messages using Incoming Webhooks on how to do that. You can also choose to create a slack app for Unleash, which will provide you with additional functionality to control how Unleash communicates messages on your Slack workspace.

    The Slack integration will perform a single retry if the HTTP POST against the Slack Webhook URL fails (either a 50x or network error). Duplicate events may happen. You should never assume events always comes in order.

    Configuration

    Events

    You can choose to trigger updates for the following events:

    • feature-created
    • feature-updated (*)
    • feature-metadata-updated
    • feature-project-change
    • feature-archived
    • feature-revived
    • feature-strategy-update
    • feature-strategy-add
    • feature-strategy-remove
    • feature-stale-on
    • feature-stale-off
    • feature-environment-enabled
    • feature-environment-disabled

    *) Deprecated, and will not be used after transition to environments in Unleash v4.3

    Parameters

    Unleash Slack integration takes the following parameters.

    • Slack Webhook URL - This is the only required property. If you are using a Slack Application you must also make sure your application is allowed to post to the channel you want to post to.
    • Username - Used to override the username used to post the update to a Slack channel.
    • Emoji Icon - Used to override the emoji icon used to post the update to a Slack channel.
    • Default channel - Where to post the message if the feature toggles has not overridden the channel via the slack tags.

    Global configuration

    • Unleash URL - The slack plugin uses the server.unleashUrl property to create the link back to Unleash in the posts. This can be set using the UNLEASH_URL environment variable or the server.unleashUrl property when starting the server from node.

    Tags

    The Slack integration also defined the Tag type "slack". You may use this tag to override which Slack channel Unleash should post updates to for this feature toggle.

    Slack Tags

    In the picture you can see we have defined two slack tags for the "new-payment-system" toggle. In this example Unleash will post updates to the #notifications and #random channel.

    - - + + \ No newline at end of file diff --git a/reference/integrations/teams.html b/reference/integrations/teams.html index 8da904c94e..0e1c649ae9 100644 --- a/reference/integrations/teams.html +++ b/reference/integrations/teams.html @@ -20,15 +20,15 @@ - - + +

    Microsoft Teams

    This feature was introduced in Unleash v4.0.0.

    The MicrosoftTeams integration allows Unleash to post Updates when a feature toggle is updated. To set up this integration, you need to set up a webhook connector for your channel. You can follow Creating an Incoming Webhook for a channel on how to do that.

    The Microsoft Teams integration will perform a single retry if the HTTP POST against the Microsoft Teams Webhook URL fails (either a 50x or network error). Duplicate events may happen, and you should never assume events always comes in order.

    Configuration

    Events

    You can choose to trigger updates for the following events:

    • feature-created
    • feature-updated (*)
    • feature-metadata-updated
    • feature-project-change
    • feature-archived
    • feature-revived
    • feature-strategy-update
    • feature-strategy-add
    • feature-strategy-remove
    • feature-stale-on
    • feature-stale-off
    • feature-environment-enabled
    • feature-environment-disabled

    *) Deprecated, and will not be used after transition to environments in Unleash v4.3

    Parameters

    Unleash Microsoft Teams integration takes the following parameters.

    • Microsoft Teams Webhook URL - This is the only required property.

    Tags

    Microsoft teams's incoming webhooks are channel specific. You will be able to create multiple integrations to support messaging on multiple channels.

    - - + + \ No newline at end of file diff --git a/reference/integrations/webhook.html b/reference/integrations/webhook.html index 0a98a5a701..0a785b3c12 100644 --- a/reference/integrations/webhook.html +++ b/reference/integrations/webhook.html @@ -20,8 +20,8 @@ - - + + @@ -30,7 +30,7 @@

    Webhook

    This feature was introduced in Unleash v3.11.0.

    The Webhook Integration introduces a generic way to post messages from Unleash to third party services. Unleash allows you to define a webhook which listens for changes in Unleash and posts them to third party services.

    The webhook will perform a single retry if the HTTP POST call fails (either a 50x or network error). Duplicate events may happen, and you should never assume events always comes in order.

    Configuration

    Events

    You can choose to trigger updates for the following events (we might add more event types in the future):

    • feature-created
    • feature-updated (*)
    • feature-metadata-updated
    • feature-project-change
    • feature-archived
    • feature-revived
    • feature-strategy-update
    • feature-strategy-add
    • feature-strategy-remove
    • feature-stale-on
    • feature-stale-off
    • feature-environment-enabled
    • feature-environment-disabled

    *) Deprecated, and will not be used after transition to environments in Unleash v4.3

    Parameters

    Unleash Webhook integration takes the following parameters.

    Webhook URL This is the only required property. If you are using a Slack Application you must also make sure your application is allowed to post the channel you want to post to.

    Content-Type Used to set the content-type header used when unleash performs an HTTP POST to the defined endpoint.

    Body template Used to override the body template used by Unleash when performing the HTTP POST. You may format you message using a Mustache template. You will have the Unleash event format available in the rendering context.

    Example:

    {
    "event": "{{event.type}}",
    "createdBy": "{{event.createdBy}}",
    "featureToggle": "{{event.data.name}}",
    "timestamp": "{{event.data.createdAt}}"
    }

    If you don't specify anything Unleash will use the Unleash event format.

    Custom SSL certificates

    If your webhook endpoint uses a custom SSL certificate, you will need to start Unleash with the NODE_EXTRA_CA_CERTS environment variable set. It needs to point to your custom certificate file in pem format.

    For more information, see the official Node.js documentation on setting extra certificate files.

    - - + + \ No newline at end of file diff --git a/reference/login-history.html b/reference/login-history.html index 212c066592..ed9c193fb3 100644 --- a/reference/login-history.html +++ b/reference/login-history.html @@ -20,15 +20,15 @@ - - + +

    Login History

    Availability

    Login history is an enterprise feature available from Unleash 4.22 onwards.

    Unleash's login history lets you track login events in your Unleash instance, and whether the attempts were successful in logging in or not.

    Login history table

    For each login event, it lists:

    • Created: When it happened
    • Username: The username that was used
    • Authentication: The authentication type that was used
    • IP address: The IP address that made the attempt
    • Success: Whether the attempt was successful or not
    • Failure reason: If the attempt was not successful, the reason why

    You can see the failure reason by hovering over the "False" badge in the "Success" column.

    Login history table failure reason

    Use the login history to:

    • Audit login events in your Unleash instance
    • Identify failed login attempts and investigate the cause
    • Debug misconfigured authentication providers

    The login history is mutable: You can remove individual login events or clear the entire history by deleting all of them.

    Finally, the login history can be downloaded (how do I download my Unleash login history) for external backups, audits, and the like.

    Retention

    Events in the login history are retained for 336 hours (14 days).

    Events older than the retention period are automatically deleted, and you won't be able to recover them. If you would like to collect login event information past the retention period, we suggest periodically downloading the login history.

    - - + + \ No newline at end of file diff --git a/reference/maintenance-mode.html b/reference/maintenance-mode.html index e57de975e4..e033f61e0c 100644 --- a/reference/maintenance-mode.html +++ b/reference/maintenance-mode.html @@ -20,8 +20,8 @@ - - + + @@ -30,7 +30,7 @@

    Maintenance Mode

    Availability

    Maintenance mode was introduced in Unleash 4.22.0.

    Unleash maintenance mode is a feature that lets administrators put Unleash into a mostly read-only mode. While Unleash is in maintenance mode:

    • Unleash users can not change any configuration settings
    • Unleash's APIs will not allow you to persist any changes

    However, any metrics sent to Unleash from client SDKs are still processed, so this mode does not have any effect on client SDKs.

    Maintenance mode is particularly useful during important events when any accidental or deliberate changes the feature/strategy configurations could lead to issues.

    Maintenance mode configuration

    Maintenance mode is controlled from the "maintenance" section of the Unleash admin configuration page.

    Maintenance mode being toggled on

    When maintenance mode is enabled, a warning banner appears at the top of the Unleash dashboard, indicating that any changes made during this period will not be saved and may result in errors.

    Maintenance mode banner when maintenance mod is toggled on

    Maintenance mode and scheduled jobs

    When maintenance mode is enabled all internal jobs performed by Unleash such as updating metrics and statistics are paused. When maintenance mode is toggled back to disabled later, all scheduled jobs are resumed.

    Maintenance mode and read-only DB user

    When maintenance mode is enabled most DB operations are suspended so you can use even read-only DB user. There's one exception though. Unleash DB user role needs a DELETE and UPDATE permission on the unleash_session table.

    GRANT DELETE ON unleash_session TO my_db_role;
    GRANT UPDATE ON unleash_session TO my_db_role;
    - - + + \ No newline at end of file diff --git a/reference/network-view.html b/reference/network-view.html index e7a46e2101..2e1c5e78be 100644 --- a/reference/network-view.html +++ b/reference/network-view.html @@ -20,15 +20,15 @@ - - + +

    Network View

    Availability

    The network view was released in Unleash 4.21. It is available to Pro and Enterprise users.

    You must configure the data source to activate the feature.

    The Unleash admin UI contains a network view as part of the admin configuration pages. This network view was designed to give you an overview and understanding of incoming requests to your Unleash instance. It makes it possible to pinpoint sources of sudden request surges, and can therefore also help you identify issues with SDK setups1.

    The network view offers two different visualizations of the same data, known as the network overview and the network traffic views.

    Because Unleash doesn't store this kind of data itself, the network view requires you to configure an external data source for the network overview to work. The network view is only available if you tell Unleash where it can find the data (refer to the data source section).

    The network view is intended to provide a simple and Unleash-centric overview that serves basic use cases. If you need detailed metrics and connection graphs, you may be better off using specialized network monitoring tools.

    Applications

    Both the network overview and the network traffic diagrams show you applications that have made requests to the Unleash instance. An application is defined as anything that sends requests to the Unleash client API, the Unleash front-end API, the Unleash admin API, or any other API that Unleash exposes. This includes Unleash SDKs, Unleash Edge, the Unleash proxy, and even the admin UI.

    "unknown" applications

    Requests that come from Unleash SDKs and other official Unleash applications will always have an application name defined. But you might sometimes see some applications listed as "unknown" in the diagrams.

    This happens when Unleash receives requests that don't contain an application name header (UNLEASH_APPNAME). This can happen, for instance, if you make HTTP requests from the command line to test Unleash connections or if you write your own HTTP client for Unleash that doesn't provide an application name.

    Because Unleash can't separate these based on their application names, all "unknown" clients will get lumped together as one application in the overview.

    Network overview

    The network overview is a diagram that shows the Unleash instance and known applications that have connected to it within the last five minutes. Unknown applications are not shown.

    Each application shown on the diagram has:

    • An application name
    • the average number of requests per second (req/s) that we have registered over the last five minutes.
    The network overview shows applications that have recently made requests to Unleash. In this figure, it's showing three different instances of the Unleash proxy connected to Unleash. Each instance has an average of 20 req/s.

    Network traffic

    The network traffic diagram is a line chart that presents requests that have used the most network traffic over the last six hours, grouped by client and base URL for the request. For legibility, this chart only shows the ten groups that have caused the most traffic over the last six hours.

    Unleash aggregates requests by client (using application name) and base URL. Base URLs are batched together using the first two path segments following the /api part of the URL. In essence, that means:

    1. Separate requests by API: Admin API requests are separate from client API requests.
    2. Within each of these groups, group all requests by their next URL path segment. For instance: /client/features and /client/features/feature-a are grouped together, while /client/register and /admin/features are separate groups.
    The network traffic chart plots req/s on the Y axis and time on the X axis. Requests are bundled per endpoint per application.

    Data source

    Prometheus and other sources

    The network view was written to be used with Prometheus and is therefore compatible with Prometheus' API.

    Other services that offer the same capabilities and the same API may work as substitutes, but we make no guarantees.

    This section will refer to the external source as Prometheus for simplicity.

    The network view uses an external Prometheus-like API to create diagrams. Because of this, Unleash will not enable the network view feature unless you set the PROMETHEUS_API environment variable.

    The PROMETHEUS_API environment variable should point to the base path of the Prometheus installation, and Prometheus should be configured to get its data from Unleash's internal backstage API. This can for example be done via a scraping job2:

    Scraping job for Unleash metrics
      - job_name: unleash_internal_metrics
    metrics_path: /internal-backstage/prometheus
    static_configs:
    - targets: ['unleash-url']

    This setup means that there is a mutual dependency between Unleash and Prometheus, where Prometheus regularly fetches data from Unleash's backstage API and Unleash fetches and displays this data when you use the network view. This diagram provides a visual representation of that.


    1. For instance: when using Unleash in an API setting, a common mistake is to instantiate a new SDK for every request instead of sharing a single instance across requests. This would be visible in the network overview graph as a large number of requests from the same app.
    2. How to set up Prometheus to collect metrics from that API is outside of the scope of this document.
    - - + + \ No newline at end of file diff --git a/reference/notifications.html b/reference/notifications.html index 815fc24f3e..7543c38a58 100644 --- a/reference/notifications.html +++ b/reference/notifications.html @@ -20,15 +20,15 @@ - - + +

    Notifications

    Availability

    Notifications were introduced in Unleash 4.22.0 for pro and enteprise customers.

    Unleash's notifications give you updates when certain events occur in projects that you are part of. The notifications are accessible from the notifications button in the navigation section of the admin UI.

    Your notifications will only contain updates from other members; you will not get notifications about actions you perform yourself.

    The notifications overview. It&#39;s telling the user about a feature that was enabled in production and that there is a change request ready for review.

    From within the notifications list, you can:

    • navigate to the features or change request that the notification is for
    • mark all notifications as read
    • filter out unread notifications

    Notifications can not be disabled.

    Notification events

    The following actions in your projects will trigger notifications for your project members:

    • Creating a feature
    • Archiving a feature
    • Enabling a feature in an environment
    • Submitting a change request
    • Approving change request
    • Rejecting change request
    • Applying change request
    - - + + \ No newline at end of file diff --git a/reference/playground.html b/reference/playground.html index 28cfe36a8c..70b5e82e09 100644 --- a/reference/playground.html +++ b/reference/playground.html @@ -20,15 +20,15 @@ - - + +

    Playground

    Availability

    The Unleash playground is available in all Unleash versions from Unleash 4.14 onwards. Unleash 5.3 introduced a more advanced playground that allows you to query multiple environments and multiple values for a single context value at the same time.

    The Unleash Playground form and an indication of where in the nav menu it is located.

    The unleash playground is a part of the Unleash Admin UI and an accompanying API. The playground lets you see which of your instance's feature toggles would be enabled for a given Unleash context. It has multiple uses, such as:

    • Understanding how Unleash evaluates strategies, constraints, and segments.
    • Debugging Unleash contexts and toggles behaving differently from what you expect.
    • Tailoring a set of strategies for given contexts.

    Each feature toggle will contain info on whether it was enabled or not and which variant it was assigned. Later versions of the playground also contain detailed evaluation results to help you understand exactly why the feature was enabled or disabled.

    Configuring the query

    This section describes what information the playground needs to evaluate your Unleash context against your existing features.

    Environments and projects

    The playground needs to know which environments and which projects to use when evaluating features. The playground UI will default to using the first environment in your list of instance environments and all projects.

    You can select as many of your environments as you want. All environments must be environments that exist in your Unleash instance. Prior to Unleash 5.3, features could only be evaluated against a single environment at a time.

    The playground will only evaluate features belonging to the projects you specify. The projects parameter can either be a list of projects or all projects.

    The Unleash context

    The Unleash context represents a client SDK's configuration and used for evaluating your features. You can construct your own context or use a JSON version of a context from a client.

    Multiple values for a single context field

    Availability

    The ability to specifiy multiple values for a single context field was introduced in Unleash 5.3.

    You can specify multiple values for a single context field by separating the values with a comma. For instance: "value1, value2".

    When you specify multiple values for context field, each value will be used to populate one variant of the response.

    Implicit context fields

    You can add any fields you want to the context used for the evaluation, and you can also leave out any fields you want. However, there are some fields that will be set for you if don't provide them:

    • appName: Unleash clients all require an appName to start up. If you do not provide an appName, the playground will assign a default value to this property instead.
    • currentTime: The currentTime property of the Unleash context gets auto-populated with the current time. You can override this manually by providing a value of your own.

    The response

    Playground results: a table of feature names and whether they&#39;re enabled or not.

    The playground's response contains a list of all the feature toggles that have been evaluated based on your configured environments, projects and context. The full response will contain results for all combinations of context fields and all environments that you selected.

    In the UI, the playground displays the features in a table. Each row of the table corresponds to a single feature. The table has a separate column for each of the environments that you selected for your query.

    Each row contains a feature and columns for the selected environments. In this screenshot, the "development" and "production" environments have been selected.

    Because you can add multiple values for each context field, each feature-environment cell contains the number of combinations that evaluated to true and false for the feature in the given environment. This can be expanded into a more detailed overview over what combinations of context fields evaluated to true and false along with any variants.

    A small table showing the detailed evaluation for a feature in the "development" environment. The provided context contained three values for the "userId" property, so the table contains three rows, showing all different combinations of the context.

    As with all of Unleash's client SDKs, the playground respects stickiness. The stickiness algorithm guarantees that you'll always get the same variants and the same gradual rollout results if you provide the same context, as long as you provide the context field used for calculating stickiness.

    The diff view

    You can compare how a feature evaluates in different environments. If you select more than one environment for your playground query, the table will have an additional "Diff" column in each row. Using the "preview diff" button, you can open a table that gives an overview over how the feature evaluated for each context combination in each environment.

    A table with three rows. Each row contains a context combination and results in the form of `true` and `false` each environment. In this case, it shows that when the "userId" context field is "2", then the feature is enabled in development, but not in production.

    Detailed evaluation results

    Availability

    Detailed evaluation results were added to the playground in Unleash 4.15.

    Detailed strategy evaluation results. A list of strategies and their constraints along with indicators of whether each one is `true` or `false`.

    Each feature in the response contains information about all of its evaluated strategies. Each of a feature's strategies lists all of the strategies constraints and segments and how it all evaluated (as best as the playground can, as mentioned in the unknown strategies section).

    In addition to the results of individual strategies, each strategy is also assigned an overall strategy evaluation result: one of true, false, and unknown. The rules for the overall result are:

    • If at least one strategy evaluates to true, then the overall result is true.
    • If all strategies evaluate to false, the overall result is false.
    • If one or more strategies evaluate to unknown and all other strategies evaluate to false, then the overall result is unknown.

    Unknown strategies

    Not all strategies can be correctly evaluated by the playground. Strategies that cannot be fully evaluated will be given an evaluation result status of 'unknown'.

    There's currently two cases where the playground can't evaluate the strategy:

    1. The strategy is a custom strategy and the playground doesn't have an implementation for the strategy.
    2. The strategy is the 'Application Hostname' strategy.

    Even if the playground doesn't recognize or otherwise can't evaluate a specific strategy, it may still evaluate the overall strategy result to false (and be certain that it is the right result). If a strategy has constraints or segments that are not satisfied, the playground knows that the strategy result wouldn't be true, regardless of the actual strategy implementation. As such, if a strategy can't be evaluated, it can be either unknown or false.

    Custom Strategies

    The playground does not have any implementations for custom strategies and adding custom strategy implementations to Unleash is not currently supported. Because of this, custom strategies will never be evaluated as true.

    The Application Hostname strategy

    The application hostname strategy is a little different from the other built-in strategies: it depends on external state and not on the Unleash context to evaluate. In short, the strategy checks its application environment to get the hostname, so the Unleash context has no effect on the result of this strategy.

    Because the application hostname strategy depends on external state, the playground can't guarantee whether the strategy would be true or false in a client. Further, the result in the playground would most likely be different from what you'd see in a client. Because of this, the playground does not evaluate this strategy.

    - - + + \ No newline at end of file diff --git a/reference/project-collaboration-mode.html b/reference/project-collaboration-mode.html index 13418a2218..4b01bc24d4 100644 --- a/reference/project-collaboration-mode.html +++ b/reference/project-collaboration-mode.html @@ -20,15 +20,15 @@ - - + +

    Project Collaboration Mode

    Availability

    The project collaboration mode is an enterprise-only feature that was introduced in Unleash 4.22.0.

    A project's collaboration mode specifies who can submit change requests. There are two collaboration modes:

    Open collaboration mode

    Anyone can submit change requests in an open project, regardless of their permissions within the project and globally. This is the default collaboration mode for projects.

    The open collaboration mode is the default in Unleash and is how all projects worked in Unleash before the introduction of collaboration modes.

    Protected collaboration mode

    Only admins and project members can submit change requests in a protected project. Other users can not submit change requests.

    Private collaboration mode

    Private collaboration mode renders the project invisible to all viewers who are not project members. This means that viewers cannot see the project in the project list, nor can they access the project's features or locate the project anywhere within Unleash.

    Editors and admins can still see private projects.

    Change requests

    When you change a project's collaboration mode from open to protected, users who do not have the right permissions will lose the ability to create new change requests.

    However, existing change requests created by these users will not be deleted. Any users with open change requests will still be able to cancel the change request, but they will not be able to update the change request.

    Project collaboration mode setting

    The collaboration mode for a project can be set at the time of creation and modified at any subsequent time, found within the 'Enterprise Settings' section.

    Project creation form with a collaboration mode field and corresponding explanation.

    Since the collaboration mode is an integral feature of the project, the options to modify it can be found on the 'Project Settings' page.

    The project-level header section with the &quot;edit project&quot; button highlighted.

    Pre-existing projects

    Projects that were created in earlier versions of Unleash (before the release of project collaboration modes) get the open mode when they are migrated to a version of Unleash with project collaboration modes.

    To change the project collaboration mode for an existing project you have to edit the project.

    - - + + \ No newline at end of file diff --git a/reference/projects.html b/reference/projects.html index 6b317fc8a5..f45b032b1d 100644 --- a/reference/projects.html +++ b/reference/projects.html @@ -20,15 +20,15 @@ - - + +

    Projects

    feature availability

    All users get access to projects, but only pro and enterprise users can create, update, or delete them.

    This document explains how Unleash uses projects, including how to create and maintain them.

    The default project

    All users get access to the default project. You cannot delete this project. You can, however, rename it if you're using the pro or enterprise version of Unleash.

    Understanding purpose of projects

    Projects are a way to organize your feature toggles within Unleash. Within a large organization, having multiple feature toggles, staying on top of the feature toggles might become a challenge. Every feature toggle will be part of a project. Projects can be linked to a development team or to functional modules within the software.

    A common pattern is to organize the feature toggles according to key areas of the application, e.g. “Basic user process” and “Advanced user process”. This is illustrated below.

    A diagram with two boxes labeled &quot;Basic user process&quot; and &quot;Advanced user process&quot;, respectively. The former contains features &quot;New login&quot; and &quot;Winter theme enablement&quot;, the latter &quot;New in-app purchase&quot; and &quot;Updated invoice repository&quot;.

    Projects and environments

    You can configure which environments should be available within a project. By default, all globally available environments are available. You can only enable/disable a feature toggle for the environments you configure for a project.

    Within the admin UI, the projects are available under the "environments" tab of the project page. Changing them requires the project owner role.

    Creating a new project

    When you log into Unleash for the first time, there is a Default project already created. All feature toggles are included in the Default project, unless explicitly set to a different one.

    From the top-line menu – click on “Projects”

    The Unleash admin UI with the &quot;Projects&quot; nav link in the top bar highlighted.

    The UI shows the currently available projects. To create a new project, use the “new project” button.

    A list of projects. There is a button saying &quot;Add new project&quot;.

    The configuration of a new Project is now available. the following input is available to create the new Project.

    A project creation form. The &quot;Create&quot; button is highlighted.

    ItemDescription
    Project IdId for this Project
    Project nameThe name of the Project.
    DescriptionA short description of the project
    ModeThe project collaboration mode
    Default StickinessThe default stickiness for the project. This setting controls the default stickiness value for variants and for the gradual rollout strategy.

    Deleting an existing project

    To keep your feature toggles clean, removing deprecated projects is important. From the overview of Projects –

    1. In the top right of the project card, find the project menu represented by three vertical dots.

    A list of projects. Each project has three vertical dots — a kebab menu — next to it.

    1. Click on Delete Project

    A list of projects. Each project has three vertical dots — a kebab menu — next to it, which exposes a menu with the &quot;Edit project&quot; and &quot;Delete project&quot; options when interacted with.

    Filter feature toggles on projects

    When browsing the feature toggles in Unleash, you might want to filter the view by looking only at the ones included in the project of interest. This is possible from the Feature toggle overview.

    From the UI top navigation menu, choose "Feature toggles".

    The Unleash Admin UI navigation menu with the &quot;Feature toggles&quot; option highlighted by a red arrow.

    The list of features toggles can be filtered on the project of your choice. By default, all feature toggles are listed in the view. You can use the search to filter to a specific project or even for multiple projects in the same time if you need.

    The feature toggle list with toggles scoped to the &quot;fintech&quot; project. The filter is activated by using a form control.

    In the search you can type "project:specific-name" to filter that project only.

    The feature toggle list with an overlay listing all the projects available. You can select a project and the list will update with the toggles belonging to that project.

    The view will now be updated with the filtered feature toggles.

    Assigning project to a new feature toggle

    When you create a new feature toggle, you can choose which project to create it in. The default project is whatever project you are currently configuring.

    A form to create a toggle. An arrow points to an input labeled &quot;project&quot;.

    All available projects are available from the drop-down menu.

    A form to create a toggle. The &quot;project&quot; input is expanded to show projects you can create the toggle in.

    Change project for an existing feature toggle

    If you want to change which project a feature toggle belongs to, you can change that from the feature toggle's configuration page. Under the settings tab, choose the project option and choose the new project from the dropdown menu.

    A feature toggle&#39;s settings tab. The project setting shows a dropdown to change projects.

    Project default strategy

    Availability

    The project default strategy feature is generally available starting with Unleash 5.2.0.

    You can define default strategies for each of a project's environments. The default strategy for an environment will be added to a feature when you enable it in an environment if and only if the feature has no active strategies defined.

    All default project strategies use the gradual rollout activation strategy. By default, the rollout 100%. You can customize the strategies by changing the rollout percentage and adding constraints and segments as you would for any other strategy.

    Configuration

    Custom strategies are configured from each project's project settings tab.

    The default strategy configuration page is available from the project settings tab.

    The default strategies screen lists a strategy for each of the project's environments

    Each strategy can be individually configured with the corresponding edit button.

    Project overview

    The project overview gives statistics for a project, including:

    • the number of all changes/events in the past 30 days compared to previous 30 days
    • the average time from when a feature was created to when it was enabled in the "production" environment. This value is calculated for all features in the project lifetime.
    • the number of features created in the past 30 days compared to previous 30 days
    • the number of features archived in the past 30 days compared to previous 30 days

    Project overview with 4 statistics for total changes, average time to production, features created and features archived.

    - - + + \ No newline at end of file diff --git a/reference/public-signup.html b/reference/public-signup.html index b89d036d90..30f6bf8107 100644 --- a/reference/public-signup.html +++ b/reference/public-signup.html @@ -20,15 +20,15 @@ - - + +

    Public Invite Links

    Public invite links let you invite team members to your Unleash instance. Any user with an invite link can sign up to Unleash instance that created the link. The user will get the viewer role (refer to the predefined roles_ section of the RBAC document for more information on roles).

    User who follow the invite link are taken directly to the Unleash sign-up page, where they can create an account.

    Only Unleash instance admins can create public invite links.

    An Unleash signup form for new users

    Public sign-up tokens

    The most important part of a public sign-up link is the sign-up token. The token is added as the invite query parameter to the invite link.

    Each token has an expiry date. After this expiry date, the token will stop working and users can no longer sign up using an invite link with that token.

    Creating, updating, and deleting tokens

    You can create, update and delete tokens via the Unleash Admin UI or via the Unleash API.

    A token is active as soon as it's created and stops working as soon as it's deleted or expired.

    You can only have one active invite token at a time. If you already have an active token, you must delete it to create a new one.

    - - + + \ No newline at end of file diff --git a/reference/rbac.html b/reference/rbac.html index 1632c0430e..45282ebf9e 100644 --- a/reference/rbac.html +++ b/reference/rbac.html @@ -20,8 +20,8 @@ - - + + @@ -68,7 +68,7 @@ within the admin UI. The API endpoints have been superseded by the create/overwrite environment variants (PUT) and update environment variants (PATCH) endpoints, respectively. - - + + \ No newline at end of file diff --git a/reference/sdks.html b/reference/sdks.html index d3282989b5..ef4cfb7b0d 100644 --- a/reference/sdks.html +++ b/reference/sdks.html @@ -20,15 +20,15 @@ - - + +

    SDK overview

    In order to connect your application to Unleash you will need a client SDK (software developer kit) for your programming language and an API token. The SDK will handle connecting to the Unleash server instance and retrieving feature toggles based on your configuration. All versions of Unleash (OSS, Pro, and Enterprise) use the same client SDKs.

    Unleash provides official client SDKs for a number of programming language. Additionally, our community have developed and contributed SDKs for other languages. So if you can't find your favorite language in the list of official SDKs, check out the list of clients written by our fantastic community.

    Official SDKs

    Server-side SDKs:

    Server-side clients run on your server and communicate directly with your Unleash instance. We provide these official clients:

    Client-side SDKs

    Client-side SDKs can connect to the Unleash Proxy or to the Unleash front-end API, but not to the regular Unleash client API.

    Server-side SDK compatibility table

    The below table shows what features the various server-side SDKs support. Note that certain features make sense only for some clients due to how the programming language works or due to how the client works.

    Legend:

    • ✅: Implemented
    • ⭕: Not yet implemented, but we're looking into it
    • ❌: Not implemented, not planned
    • N/A: Not applicable to this SDK
    note

    If you see an item marked with a ❌ that you would find useful, feel free to reach out to us (on Slack, for instance) with your use case. It may not be something we can prioritize right now, but if you'd like to contribute it back to the community, we'd love to help you build it.

    CapabilityJavaNode.jsGoPythonRuby.NETPHPRust
    Category: Initialization
    Async initialization
    Can block until synchronized
    Default refresh interval10s15s15s15s15s30s30s15s
    Default metrics interval60s60s60s60s60s60s30s15s
    Context providerN/AN/AN/AN/AN/A
    Global fallback function
    Toggle Query: namePrefix
    Toggle Query: tags
    Toggle Query: project_nameN/A
    Category: Custom Headers
    static
    function✅ (4.3)
    Category: Built-in strategies
    Standard
    Gradual rollout
    Gradual rollout: custom stickiness
    UserID
    IP
    IP: CIDR syntax
    Hostname
    Category: Custom strategies
    Basic support
    Category: Strategy constraints
    Basic support (IN, NOT_IN operators)
    Advanced support (Semver, date, numeric and extended string operators) (introduced in)✅ (5.1)✅ (3.12)✅ (3.3)✅ (5.1)✅ (4.2)✅ (2.1)✅ (1.3.1)
    Category: Unleash Context
    Static fields (environment, appName)
    Defined fields
    Custom properties
    Category: isEnabled
    Can take context
    Override fallback value
    Fallback function
    Category: Variants
    Basic support
    Custom fallback variant
    Custom weight
    Custom stickiness
    Strategy Variants
    Category: Local backup
    File based backup
    Category: Usage metrics
    Can disable metrics
    Client registration
    Basic usage metrics (yes/no)
    Impression data
    Category: Bootstrap (beta)
    Bootstrap from file
    Custom Bootstrap implementation

    Community SDKs ❤️

    Here's some of the fantastic work our community has done to make Unleash work in even more contexts. If you still can't find your favorite language, let us know and we'd love to help you create the client for it!

    Implement your own SDK

    If you can't find an SDK that fits your need, you can also develop your own SDK. To make implementation easier, check out these resources:

    • Unleash Client Specifications - Used by all official SDKs to make sure they behave correctly across different language implementations. This lets us verify that a gradual rollout to 10% of the users would affect the same users regardless of which SDK you're using.
    • Client SDK overview - A brief, overall guide of the Unleash Architecture and important aspects of the SDK role in it all.

    Client-side SDK behavior

    The following section details the behavior of frontend / client-side SDKs when initializing and fetching flags with respect to network connectivity.

    When the SDK is initialized in the application, an in memory repository is setup and synchronized against the frontent API using the configured token and context. Note that the frontend API is hosted by either the Unleash Proxy/Edge or the upstream Unleash instance directly.

    1. All feature flag evaluation is performed by the Proxy/Edge or Unleash instance. A payload of all enabled flags and their variants (if applicable) is returned as a single request. Disabled flags are not included.

    2. When a page inside the application requests a feature flag, the SDK will return the flag state from memory. No network connection to the frontend API is performed.

    3. The SDK periodically syncs with the frontend API to retrieve the latest set of enabled toggles

    Working offline

    Once they have been initialized, all Unleash clients will continue to work perfectly well without an internet connection or in the event that the Unleash Server has an outage.

    Because the SDKs and the Unleash Proxy/Edge cache their feature toggle states locally and only communicate with the Unleash server (in the case of the server-side SDKs and the Proxy) or the Proxy/Edge (in the case of front-end SDKs) at predetermined intervals, a broken connection only means that they won't get any new updates.

    Unless the SDK supports bootstrapping, it will need to connect to Unleash at startup to get its initial feature toggle data set. If the SDK doesn't have a feature toggle data set available, all toggles will fall back to evaluating as disabled or as the specified default value (in SDKs that support that).

    Bootstrapping

    By default, all SDKs reach out to the Unleash Server at startup to fetch their toggle configuration. Additionally, some of the server-side SDKs and the Proxy (see the above compatibility table) also support bootstrapping, which allows them to get their toggle configuration from a file, the environment, or other local resources. These SDKs can work without any network connection whatsoever.

    Bootstrapping is also supported by the following front-end client SDKs:

    - - + + \ No newline at end of file diff --git a/reference/sdks/android-proxy.html b/reference/sdks/android-proxy.html index 869604134a..f0faf86201 100644 --- a/reference/sdks/android-proxy.html +++ b/reference/sdks/android-proxy.html @@ -20,8 +20,8 @@ - - + + @@ -34,8 +34,8 @@ You can configure the pollInterval and a listener that gets notified when toggles are updated in the background thread. If you set the poll interval to 0, the SDK will fetch once, but not set up polling.

    The listener is a no-argument lambda that gets called by the RefreshPolicy for every poll that

    1. Does not return 304 - Not Modified
    2. Does not return a list of toggles that's exactly the same as the one we've already stored in local cache. Just in case the ETag/If-None-Match fails.

    Example usage is equal to the Example setup above

    val context = UnleashContext.newBuilder()
    .appName("Your AppName")
    .userId("However you resolve your userid")
    .sessionId("However you resolve your session id")
    .build()
    val config = UnleashConfig.newBuilder()
    .proxyUrl("URL to your front-end API or proxy")
    .clientKey("your front-end API token or proxy client key")
    .pollingMode(PollingModes.autoPoll(60) { // poll interval in seconds
    featuresUpdated()
    })
    .build()
    val client = UnleashClient(unleashConfig = config, unleashContext = context)
    FilePolling (since v0.2)

    The name FilePolling can be a tad misleading, since this policy doesn't actually poll, it simply loads a file of toggles from disk on startup, and uses that to answer all client calls. Useful when your app might have limited internet connectivity, you'd like to run tests with a known toggle state or you simply do not want background activity.

    The following example shows how to use it, provided the file to use is located at /tmp/proxyresponse.json

    val toggles = File("/tmp/proxyresponse.json")
    val pollingMode = PollingModes.fileMode(toggles)
    val context = UnleashContext.newBuilder()
    .appName("Your AppName")
    .userId("However you resolve your userid")
    .sessionId("However you resolve your session id")
    .build()
    val config = UnleashConfig.newBuilder()
    .proxyUrl("URL to your front-end API or proxy") // These two don't matter for FilePolling,
    .clientKey("front-end API token / proxy client key") // since the client never speaks to the proxy
    .pollingMode(pollingMode)
    .build()
    val client = UnleashClient(unleashConfig = config, unleashContext = context)

    Metrics (since v0.2)

    If you'd like the client to post metrics to the proxy so the admin interface can be updated, add a call to enableMetrics().

    NB Only supported by SDK version >=26

    val config = UnleashConfig
    .newBuilder()
    .appName()
    .userId()
    .sessionId()
    .enableMetrics()
    .build()

    The default configuration configures a daemon to report metrics once every minute, this can be altered using the metricsInterval(long milliseconds) method on the builder, so if you'd rather see us post in 5 minutes intervals you could do

    UnleashConfig().newBuilder().metricsInterval(300000) // Every 5 minutes

    Example main activity

    In the sample app -we use this to update the text on the first view

     this@MainActivity.runOnUiThread {
    val firstFragmentText = findViewById<TextView>(R.id.textview_first)
    firstFragmentText.text = "Variant ${unleashClient.getVariant("unleash_android_sdk_demo").name}"
    }

    This content was generated on

    - - +we use this to update the text on the first view

     this@MainActivity.runOnUiThread {
    val firstFragmentText = findViewById<TextView>(R.id.textview_first)
    firstFragmentText.text = "Variant ${unleashClient.getVariant("unleash_android_sdk_demo").name}"
    }

    This content was generated on

    + + \ No newline at end of file diff --git a/reference/sdks/dotnet.html b/reference/sdks/dotnet.html index 60ac892e69..efc530d30d 100644 --- a/reference/sdks/dotnet.html +++ b/reference/sdks/dotnet.html @@ -20,8 +20,8 @@ - - + + @@ -37,8 +37,8 @@ HttpClientFactory that inherits from DefaultHttpClientFactory, and override the method CreateHttpClientInstance. Then configure UnleashSettings to use your custom HttpClientFactory.

    internal class CustomHttpClientFactory : DefaultHttpClientFactory
    {
    protected override HttpClient CreateHttpClientInstance(Uri unleashApiUri)
    {
    var messageHandler = new CustomHttpMessageHandler();
    var httpClient = new HttpClient(messageHandler)
    {
    BaseAddress = apiUri,
    Timeout = TimeSpan.FromSeconds(5)
    };
    }
    }

    var settings = new UnleashSettings
    {
    AppName = "dotnet-test",
    //...
    HttpClientFactory = new CustomHttpClientFactory()
    };

    Dynamic custom HTTP headers

    If you need custom http headers that change during the lifetime of the client, a provider can be defined via the UnleashSettings.

    Public Class CustomHttpHeaderProvider
    Implements IUnleashCustomHttpHeaderProvider

    Public Function GetCustomHeaders() As Dictionary(Of String, String) Implements IUnleashCustomHttpHeaderProvider.GetCustomHeaders
    Dim token = ' Acquire or refresh a token
    Return New Dictionary(Of String, String) From
    {{"Authorization", "Bearer " & token}}
    End Function
    End Class

    ' ...

    Dim unleashSettings As New UnleashSettings()
    unleashSettings.AppName = "dotnet-test"
    unleashSettings.InstanceTag = "instance z"
    ' add the custom http header provider to the settings
    unleashSettings.UnleashCustomHttpHeaderProvider = New CustomHttpHeaderProvider()
    unleashSettings.UnleashApi = new Uri("http://unleash.herokuapp.com/api/")
    unleashSettings.UnleashContextProvider = New AspNetContextProvider()
    Dim unleash = New DefaultUnleash(unleashSettings)

    Logging

    By default Unleash-client uses LibLog to integrate with the currently configured logger for your application. The supported loggers are:

    • Serilog
    • NLog
    • Log4Net
    • EntLib
    • Loupe

    Custom logger integration

    To plug in your own logger you can implement the ILogProvider interface, and register it with Unleash:

    Unleash.Logging.LogProvider.SetCurrentLogProvider(new CustomLogProvider());
    var settings = new UnleashSettings()
    //...

    The GetLogger method is responsible for returning a delegate to be used for logging, and your logging integration should be placed inside that delegate:

    using System;
    using Unleash.Logging;

    namespace Unleash.Demo.CustomLogging
    {
    public class CustomLogProvider : ILogProvider
    {
    public Logger GetLogger(string name)
    {
    return (logLevel, messageFunc, exception, formatParameters) =>
    {
    // Plug in your logging code here

    return true;
    };
    }

    public IDisposable OpenMappedContext(string key, object value, bool destructure = false)
    {
    return new EmptyIDisposable();
    }

    public IDisposable OpenNestedContext(string message)
    {
    return new EmptyIDisposable();
    }
    }

    public class EmptyIDisposable : IDisposable
    {
    public void Dispose()
    {
    }
    }
    }

    Local backup

    By default unleash-client fetches the feature toggles from unleash-server every 20s, and stores the result in temporary .json file which is located in System.IO.Path.GetTempPath() directory. This means that if the unleash-server becomes unavailable, the unleash-client will still be able to toggle the features based on the values stored in .json file. As a result of this, the second argument of IsEnabled will be returned in two cases:

    • When .json file does not exists
    • When the named feature toggle does not exist in .json file

    The backup file name will follow this pattern: {fileNameWithoutExtension}-{AppName}-{InstanceTag}-{SdkVersion}.{extension}, where InstanceTag is either what you configure on UnleashSettings during startup, or a formatted string with a random component following this pattern: {Dns.GetHostName()}-generated-{Guid.NewGuid()}.

    You can configure InstanceTag like this:

    var settings = new UnleashSettings()
    {
    AppName = "dotnet-test",
    UnleashApi = new Uri("http://unleash.herokuapp.com/api/"),
    // Set an instance tag for consistent backup file naming
    InstanceTag = "CustomInstanceTag",
    UnleashContextProvider = new AspNetContextProvider(),
    CustomHttpHeaders = new Dictionary<string, string>()
    {
    {"Authorization", "API token" }
    }
    };

    Bootstrapping

    • Unleash supports bootstrapping from a JSON string.
    • Configure your own custom provider implementing the IToggleBootstrapProvider interface's single method ToggleCollection Read(). -This should return a ToggleCollection. The UnleashSettings.JsonSerializer can be used to deserialize a JSON string in the same format returned from /api/client/features.
    • Example bootstrap files can be found in the json files located in tests/Unleash.Tests/App_Data
    • Our assumption is this can be use for applications deployed to ephemeral containers or more locked down file systems where Unleash's need to write the backup file is not desirable or possible.
    • Loading with bootstrapping defaults to override feature toggles loaded from Local Backup, this override can be switched off by setting the UnleashSettings.ToggleOverride property to false

    Configuring with the UnleashSettings:

    var settings = new UnleashSettings()
    {
    AppName = "dotnet-test",
    UnleashApi = new Uri("http://unleash.herokuapp.com/api/"),
    CustomHttpHeaders = new Dictionary()
    {
    {"Authorization","API token" }
    },
    ToggleOverride = false, // Defaults to true
    ToggleBootstrapProvider = new MyToggleBootstrapProvider() // A toggle bootstrap provider implementing IToggleBootstrapProvider here
    };

    Provided Bootstrappers

    • Two ToggleBootstrapProviders are provided
    • These are found in the Unleash.Utilities:

    ToggleBootstrapFileProvider

    • Unleash comes with a ToggleBootstrapFileProvider which implements the IToggleBootstrapProvider interface.
    • Configure with UnleashSettings helper method:
    settings.UseBootstrapFileProvider("./path/to/file.json");

    ToggleBootstrapUrlProvider

    • Unleash also comes with a ToggleBootstrapUrlProvider which implements the IToggleBootstrapProvider interface.

    • Fetches JSON from a webaddress using HttpMethod.Get

    • Configure with UnleashSettings helper method:

    var shouldThrowOnError = true; // Throws for 500, 404, etc
    var customHeaders = new Dictionary<string, string>()
    {
    { "Authorization", "Bearer ABCdefg123" } // Or whichever set of headers would be required to GET this file
    }; // Defaults to null
    settings.UseBootstrapUrlProvider("://domain.top/path/to/file.json", shouldThrowOnError, customHeaders);

    Json Serialization

    The unleash client is dependant on a json serialization library. If your application already have Newtonsoft.Json >= 9.0.1 installed, everything should work out of the box. If not, you will get an error message during startup telling you to implement an 'IJsonSerializer' interface, which needs to be added to the configuration.

    With Newtonsoft.Json version 7.0.0.0, the following implementation can be used. For older versions, consider to upgrade.

    var settings = new UnleashSettings()
    {
    AppName = "dotnet-test",
    UnleashApi = new Uri("http://unleash.herokuapp.com/api/"),
    JsonSerializer = new NewtonsoftJson7Serializer()
    };

    public class NewtonsoftJson7Serializer : IJsonSerializer
    {
    private readonly Encoding utf8 = Encoding.UTF8;

    private static readonly JsonSerializer Serializer = new JsonSerializer()
    {
    ContractResolver = new CamelCaseExceptDictionaryKeysResolver()
    };

    public T Deserialize<T>(Stream stream)
    {
    using (var streamReader = new StreamReader(stream, utf8))
    using (var textReader = new JsonTextReader(streamReader))
    {
    return Serializer.Deserialize<T>(textReader);
    }
    }

    public void Serialize<T>(Stream stream, T instance)
    {
    using (var writer = new StreamWriter(stream, utf8, 1024 * 4, leaveOpen: true))
    using (var jsonWriter = new JsonTextWriter(writer))
    {
    Serializer.Serialize(jsonWriter, instance);

    jsonWriter.Flush();
    stream.Position = 0;
    }
    }

    class CamelCaseExceptDictionaryKeysResolver : CamelCasePropertyNamesContractResolver
    {
    protected override JsonDictionaryContract CreateDictionaryContract(Type objectType)
    {
    var contract = base.CreateDictionaryContract(objectType);

    contract.DictionaryKeyResolver = propertyName =>
    {
    return propertyName;
    };

    return contract;
    }
    }
    }

    The server api needs camel cased json, but not for certain dictionary keys. The implementation can be naively validated by the JsonSerializerTester.Assert function. (Work in progress).

    Run unleash server with Docker locally

    The Unleash team have made a separate project which runs unleash server inside docker. Please see unleash-docker for more details.

    Development

    Visual Studio 2017 / Code

    Cakebuild

    Other information

    • Check out our guide for more information on how to build and scale feature flag systems

    This content was generated on

    - - +This should return a ToggleCollection. The UnleashSettings.JsonSerializer can be used to deserialize a JSON string in the same format returned from /api/client/features.
  • Example bootstrap files can be found in the json files located in tests/Unleash.Tests/App_Data
  • Our assumption is this can be use for applications deployed to ephemeral containers or more locked down file systems where Unleash's need to write the backup file is not desirable or possible.
  • Loading with bootstrapping defaults to override feature toggles loaded from Local Backup, this override can be switched off by setting the UnleashSettings.ToggleOverride property to false
  • Configuring with the UnleashSettings:

    var settings = new UnleashSettings()
    {
    AppName = "dotnet-test",
    UnleashApi = new Uri("http://unleash.herokuapp.com/api/"),
    CustomHttpHeaders = new Dictionary()
    {
    {"Authorization","API token" }
    },
    ToggleOverride = false, // Defaults to true
    ToggleBootstrapProvider = new MyToggleBootstrapProvider() // A toggle bootstrap provider implementing IToggleBootstrapProvider here
    };

    Provided Bootstrappers

    • Two ToggleBootstrapProviders are provided
    • These are found in the Unleash.Utilities:

    ToggleBootstrapFileProvider

    • Unleash comes with a ToggleBootstrapFileProvider which implements the IToggleBootstrapProvider interface.
    • Configure with UnleashSettings helper method:
    settings.UseBootstrapFileProvider("./path/to/file.json");

    ToggleBootstrapUrlProvider

    • Unleash also comes with a ToggleBootstrapUrlProvider which implements the IToggleBootstrapProvider interface.

    • Fetches JSON from a webaddress using HttpMethod.Get

    • Configure with UnleashSettings helper method:

    var shouldThrowOnError = true; // Throws for 500, 404, etc
    var customHeaders = new Dictionary<string, string>()
    {
    { "Authorization", "Bearer ABCdefg123" } // Or whichever set of headers would be required to GET this file
    }; // Defaults to null
    settings.UseBootstrapUrlProvider("://domain.top/path/to/file.json", shouldThrowOnError, customHeaders);

    Json Serialization

    The unleash client is dependant on a json serialization library. If your application already have Newtonsoft.Json >= 9.0.1 installed, everything should work out of the box. If not, you will get an error message during startup telling you to implement an 'IJsonSerializer' interface, which needs to be added to the configuration.

    With Newtonsoft.Json version 7.0.0.0, the following implementation can be used. For older versions, consider to upgrade.

    var settings = new UnleashSettings()
    {
    AppName = "dotnet-test",
    UnleashApi = new Uri("http://unleash.herokuapp.com/api/"),
    JsonSerializer = new NewtonsoftJson7Serializer()
    };

    public class NewtonsoftJson7Serializer : IJsonSerializer
    {
    private readonly Encoding utf8 = Encoding.UTF8;

    private static readonly JsonSerializer Serializer = new JsonSerializer()
    {
    ContractResolver = new CamelCaseExceptDictionaryKeysResolver()
    };

    public T Deserialize<T>(Stream stream)
    {
    using (var streamReader = new StreamReader(stream, utf8))
    using (var textReader = new JsonTextReader(streamReader))
    {
    return Serializer.Deserialize<T>(textReader);
    }
    }

    public void Serialize<T>(Stream stream, T instance)
    {
    using (var writer = new StreamWriter(stream, utf8, 1024 * 4, leaveOpen: true))
    using (var jsonWriter = new JsonTextWriter(writer))
    {
    Serializer.Serialize(jsonWriter, instance);

    jsonWriter.Flush();
    stream.Position = 0;
    }
    }

    class CamelCaseExceptDictionaryKeysResolver : CamelCasePropertyNamesContractResolver
    {
    protected override JsonDictionaryContract CreateDictionaryContract(Type objectType)
    {
    var contract = base.CreateDictionaryContract(objectType);

    contract.DictionaryKeyResolver = propertyName =>
    {
    return propertyName;
    };

    return contract;
    }
    }
    }

    The server api needs camel cased json, but not for certain dictionary keys. The implementation can be naively validated by the JsonSerializerTester.Assert function. (Work in progress).

    Run unleash server with Docker locally

    The Unleash team have made a separate project which runs unleash server inside docker. Please see unleash-docker for more details.

    Development

    Visual Studio 2017 / Code

    Cakebuild

    Other information

    • Check out our guide for more information on how to build and scale feature flag systems

    This content was generated on

    + + \ No newline at end of file diff --git a/reference/sdks/flutter.html b/reference/sdks/flutter.html index db3800e284..25f3360652 100644 --- a/reference/sdks/flutter.html +++ b/reference/sdks/flutter.html @@ -20,8 +20,8 @@ - - + + @@ -32,8 +32,8 @@ This makes it super simple to use Unleash from any Flutter app.

    How to use the client as a module

    Step 1: Installation

    flutter pub add unleash_proxy_client_flutter

    Step 2: Initialize the SDK


    💡 TIP: As a client-side SDK, this SDK requires you to connect to either an Unleash proxy or to the Unleash front-end API. Refer to the connection options section for more information.


    Configure the client according to your needs. The following example provides only the required options. Refer to the section on available options for the full list.

    import 'package:unleash_proxy_client_flutter/unleash_proxy_client_flutter.dart';

    final unleash = UnleashClient(
    url: Uri.parse('https://<your-unleash-instance>/api/frontend'),
    clientKey: '<your-client-side-token>',
    appName: 'my-app');

    Connection options

    To connect this SDK to your Unleash instance's front-end API, use the URL to your Unleash instance's front-end API (<unleash-url>/api/frontend) as the url parameter. For the clientKey parameter, use a FRONTEND token generated from your Unleash instance. Refer to the how to create API tokens guide for the necessary steps.

    To connect this SDK to the Unleash proxy, use the proxy's URL and a proxy client key. The configuration section of the Unleash proxy docs contains more info on how to configure client keys for your proxy.

    Step 3: Let the client synchronize

    You should wait for the client's ready or initialized events before you start working with it. Before it's ready, the client might not report the correct state for your features.

    unleash.on('ready', (_) {
    if (unleash.isEnabled('proxy.demo')) {
    print('proxy.demo is enabled');
    } else {
    print('proxy.demo is disabled');
    }
    });

    The difference between the events is explained below.

    Step 4: Check feature toggle states

    Once the client is ready, you can start checking features in your application. Use the isEnabled method to check the state of any feature you want:

    unleash.isEnabled('proxy.demo');

    You can use the getVariant method to get the variant of an enabled feature that has variants. If the feature is disabled or if it has no variants, then you will get back the disabled variant

    final variant = unleash.getVariant('proxy.demo');

    if (variant.name == 'blue') {
    // something with variant blue...
    }

    You can also access the payload associated with the variant:

    final variant = unleash.getVariant('proxy.demo');
    final payload = variant.payload;

    if (payload != null) {
    // do something with the payload
    // print(payload "${payload.type} ${payload.value}");
    }

    Updating the Unleash context

    The Unleash context is used to evaluate features against attributes of a the current user. To update and configure the Unleash context in this SDK, use the updateContext and setContextField methods.

    The context you set in your app will be passed along to the Unleash proxy or the front-end API as query parameters for feature evaluation.

    The updateContext method will replace the entire (mutable part) of the Unleash context with the data that you pass in.

    The setContextField method only acts on the property that you choose. It does not affect any other properties of the Unleash context.

    // Used to set the context fields, shared with the Unleash Proxy. This 
    // method will replace the entire (mutable part) of the Unleash Context.
    unleash.updateContext(UnleashContext(userId: '1233'));

    // Used to update a single field on the Unleash Context.
    unleash.setContextField('userId', '4141');

    Available options

    The Unleash SDK takes the following options:

    optionrequireddefaultdescription
    urlyesn/aThe Unleash Proxy URL to connect to. E.g.: https://examples.com/proxy
    clientKeyyesn/aThe Unleash Proxy Secret to be used
    appNameyesn/aThe name of the application using this SDK. Will be used as part of the metrics sent to Unleash Proxy. Will also be part of the Unleash Context.
    refreshIntervalno30How often, in seconds, the SDK should check for updated toggle configuration. If set to 0 will disable checking for updates
    disableRefreshnofalseIf set to true, the client will not check for updated toggle configuration
    metricsIntervalno30How often, in seconds, the SDK should send usage metrics back to Unleash Proxy
    disableMetricsnofalseSet this option to true if you want to disable usage metrics
    storageProvidernoSharedPreferencesStorageProviderAllows you to inject a custom storeProvider
    bootstrapno[]Allows you to bootstrap the cached feature toggle configuration.
    bootstrapOverridenotrueShould the bootstrap automatically override cached data in the local-storage. Will only be used if bootstrap is not an empty array.
    headerNamenoAuthorizationProvides possiblity to specify custom header that is passed to Unleash / Unleash Proxy with the clientKey
    customHeadersno{}Additional headers to use when making HTTP requests to the Unleash proxy. In case of name collisions with the default headers, the customHeaders value will be used.
    impressionDataAllnofalseAllows you to trigger "impression" events for all getToggle and getVariant invocations. This is particularly useful for "disabled" feature toggles that are not visible to frontend SDKs.
    fetchernohttp.getAllows you to define your own fetcher. Can be used to add certificate pinning or additional http behavior.
    posternohttp.postAllows you to define your own poster. Can be used to add certificate pinning or additional http behavior.

    Listen for updates via the events_emitter

    The client is also an event emitter. This means that your code can subscribe to updates from the client. This is a neat way to update your app when toggle state updates.

    unleash.on('update', (_) {
    final myToggle = unleash.isEnabled('proxy.demo');
    //do something useful
    });

    Available events:

    • error - emitted when an error occurs on init, or when fetch function fails, or when fetch receives a non-ok response object. The error object is sent as payload.
    • initialized - emitted after the SDK has read local cached data in the storageProvider.
    • ready - emitted after the SDK has successfully started and performed the initial fetch towards the Unleash Proxy.
    • update - emitted every time the Unleash Proxy return a new feature toggle configuration. The SDK will emit this event as part of the initial fetch from the SDK.

    PS! Please remember that you should always register your event listeners before your call unleash.start(). If you register them after you have started the SDK you risk loosing important events.

    SessionId - Important note!

    You may provide a custom session id via the "context". If you do not provide a sessionId this SDK will create a random session id, which will also be stored in the provided storage. By always having a consistent sessionId available ensures that even "anonymous" users will get a consistent experience when feature toggles is evaluated, in combination with a gradual (percentage based) rollout.

    Stop the SDK

    You can stop the Unleash client by calling the stop method. Once the client has been stopped, it will no longer check for updates or send metrics to the server.

    A stopped client can be restarted.

    unleash.stop();

    Bootstrap

    Now it is possible to bootstrap the SDK with your own feature toggle configuration when you don't want to make an API call.

    This is also useful if you require the toggles to be in a certain state immediately after initializing the SDK.

    How to use it ?

    Add a bootstrap attribute when create a new UnleashClient.
    -There's also a bootstrapOverride attribute which is by default is true.

    final unleash = UnleashClient(
    url: Uri.parse('https://app.unleash-hosted.com/demo/api/proxy'),
    clientKey: 'proxy-123',
    appName: 'my-app',
    bootstrapOverride: false,
    bootstrap: {
    'demoApp.step4': ToggleConfig(
    enabled: true,
    impressionData: false,
    variant: Variant(enabled: true, name: 'blue'))
    });

    NOTES: ⚠️

    • If bootstrapOverride is true (by default), any local cached data will be overridden with the bootstrap specified.
    • If bootstrapOverride is false any local cached data will not be overridden unless the local cache is empty.

    Useful commands development

    • flutter test
    • dart format lib test
    • flutter analyze lib test
    • dart pub publish

    This content was generated on

    - - +There's also a bootstrapOverride attribute which is by default is true.

    final unleash = UnleashClient(
    url: Uri.parse('https://app.unleash-hosted.com/demo/api/proxy'),
    clientKey: 'proxy-123',
    appName: 'my-app',
    bootstrapOverride: false,
    bootstrap: {
    'demoApp.step4': ToggleConfig(
    enabled: true,
    impressionData: false,
    variant: Variant(enabled: true, name: 'blue'))
    });

    NOTES: ⚠️

    • If bootstrapOverride is true (by default), any local cached data will be overridden with the bootstrap specified.
    • If bootstrapOverride is false any local cached data will not be overridden unless the local cache is empty.

    Useful commands development

    • flutter test
    • dart format lib test
    • flutter analyze lib test
    • dart pub publish

    This content was generated on

    + + \ No newline at end of file diff --git a/reference/sdks/go.html b/reference/sdks/go.html index 6a53410f28..3c83d3be6e 100644 --- a/reference/sdks/go.html +++ b/reference/sdks/go.html @@ -20,8 +20,8 @@ - - + + @@ -38,8 +38,8 @@ you can add the following to your apps go.mod:

        replace github.com/Unleash/unleash-client-go/v3 => ../unleash-client-go/

    Steps to release

    • Update the clientVersion in client.go
    • Tag the repository with the new tag

    Adding client specifications

    In order to make sure the unleash clients uphold their contract, we have defined a set of client specifications that define this contract. These are used to make sure that each unleash client at any time adhere to the contract, and define a set of functionality that is core to unleash. You can view -the client specifications here.

    In order to make the tests run please do the following steps.

    // in repository root
    // testdata is gitignored
    mkdir testdata
    cd testdata
    git clone https://github.com/Unleash/client-specification.git

    Requirements:

    • make
    • golint (go get -u golang.org/x/lint/golint)

    Run tests:

    make

    Run lint check:

    make lint

    Run code-style checks:(currently failing)

    make strict-check

    Run race-tests:

    make test-race

    Benchmarking

    You can benchmark feature toggle evaluation by running:

    go test -run=^$ -bench=BenchmarkFeatureToggleEvaluation -benchtime=10s

    Here's an example of how the output could look like:

    goos: darwin
    goarch: arm64
    pkg: github.com/Unleash/unleash-client-go/v3
    BenchmarkFeatureToggleEvaluation-8 Final Estimated Operations Per Day: 101.131 billion (1.011315e+11)
    13635154 854.3 ns/op
    PASS
    ok github.com/Unleash/unleash-client-go/v3 13.388s

    In this example the benchmark was run on a MacBook Pro (M1 Pro, 2021) with 16GB RAM.

    We can see a result of 854.3 ns/op, which means around 101.131 billion feature toggle evaluations per day.

    Note: The benchmark is run with a single CPU core, no parallelism.

    Design philsophy

    This feature flag SDK is designed according to our design philosophy. You can read more about that here.


    This content was generated on

    - - +the client specifications here.

    In order to make the tests run please do the following steps.

    // in repository root
    // testdata is gitignored
    mkdir testdata
    cd testdata
    git clone https://github.com/Unleash/client-specification.git

    Requirements:

    • make
    • golint (go get -u golang.org/x/lint/golint)

    Run tests:

    make

    Run lint check:

    make lint

    Run code-style checks:(currently failing)

    make strict-check

    Run race-tests:

    make test-race

    Benchmarking

    You can benchmark feature toggle evaluation by running:

    go test -run=^$ -bench=BenchmarkFeatureToggleEvaluation -benchtime=10s

    Here's an example of how the output could look like:

    goos: darwin
    goarch: arm64
    pkg: github.com/Unleash/unleash-client-go/v3
    BenchmarkFeatureToggleEvaluation-8 Final Estimated Operations Per Day: 101.131 billion (1.011315e+11)
    13635154 854.3 ns/op
    PASS
    ok github.com/Unleash/unleash-client-go/v3 13.388s

    In this example the benchmark was run on a MacBook Pro (M1 Pro, 2021) with 16GB RAM.

    We can see a result of 854.3 ns/op, which means around 101.131 billion feature toggle evaluations per day.

    Note: The benchmark is run with a single CPU core, no parallelism.

    Design philsophy

    This feature flag SDK is designed according to our design philosophy. You can read more about that here.


    This content was generated on

    + + \ No newline at end of file diff --git a/reference/sdks/ios-proxy.html b/reference/sdks/ios-proxy.html index cb278e885a..bd17fdbacb 100644 --- a/reference/sdks/ios-proxy.html +++ b/reference/sdks/ios-proxy.html @@ -20,16 +20,16 @@ - - + +

    iOS

    Generated content

    This document was generated from the README in the iOS GitHub repository.

    tip

    To connect to Unleash from a client-side context, you'll need to use the Unleash front-end API (how do I create an API token?) or the Unleash proxy (how do I create client keys?).

    unleash-proxy-client-swift

    The unleash-proxy-client-swift makes it easy for native applications and other swift platforms to connect to the unleash proxy. The proxy will evaluate a feature toggle for a given context and return a list of feature flags relevant for the provided context.

    The unleash-proxy-client-swift will then cache these toggles in a map in memory and refresh the configuration at a configurable interval, making queries against the toggle configuration extremely fast.

    Requirements

    • MacOS: 12.15
    • iOS: 12

    Installation

    Follow the following steps in order to install the unleash-proxy-client-swift:

    1. In your Xcode project go to File -> Swift Packages -> Add Package Dependency
    2. Supply the link to this repository
    3. Set the appropriate package constraints (typically up to next major version)
    4. Let Xcode find and install the necessary packages

    Once you're done, you should see SwiftEventBus and UnleashProxyClientSwift listed as dependencies in the file explorer of your project.

    Usage

    In order to get started you need to import and instantiate the unleash client:

    iOS >= 13

    import SwiftUI
    import UnleashProxyClientSwift

    // Setup Unleash in the context where it makes most sense

    var unleash = UnleashProxyClientSwift.UnleashClient(unleashUrl: "https://<unleash-instance>/api/frontend", clientKey: "<client-side-api-token>", refreshInterval: 15, appName: "test", context: ["userId": "c3b155b0-5ebe-4a20-8386-e0cab160051e"])

    unleash.start()

    iOS >= 12

    import SwiftUI
    import UnleashProxyClientSwift

    // Setup Unleash in the context where it makes most sense

    var unleash = UnleashProxyClientSwift.UnleashClientBase(unleashUrl: "https://<unleash-instance>/api/frontend", clientKey: "<client-side-api-token>", refreshInterval: 15, appName: "test", context: ["userId": "c3b155b0-5ebe-4a20-8386-e0cab160051e"])

    unleash.start()

    In the example above we import the UnleashProxyClientSwift and instantiate the client. You need to provide the following parameters:

    • unleashUrl: the full url to either the Unleash front-end API OR an Unleash proxy [String]
    • clientKey: either an client-side API token if you use the front-end API (how) or a proxy client key if you use the proxy [String]
    • refreshInterval: the polling interval in seconds [Int]. Set to 0to only poll once and disable a periodic polling
    • appName: the application name identifier [String]
    • context: the context parameters except from appName and environment which should be specified explicitly in the init [[String: String]]

    Running unleash.start() will make the first request against the proxy and retrieve the feature toggle configuration, and set up the polling interval in the background.

    NOTE: While waiting to boot up the configuration may not be available, which means that asking for a feature toggle may result in a false if the configuration has not loaded. In the event that you need to be certain that the configuration is loaded we emit an event you can subscribe to, once the configuration is loaded. See more in the Events section.

    Once the configuration is loaded you can ask against the cache for a given feature toggle:

    if unleash.isEnabled(name: "ios") {
    // do something
    } else {
    // do something else
    }

    You can also set up variants and use them in a similar fashion:

    var variant = unleash.getVariant(name: "ios")
    if variant.enabled {
    // do something
    } else {
    // do something else
    }

    Available options

    The Unleash SDK takes the following options:

    optionrequireddefaultdescription
    unleashUrlyesn/aThe Unleash Proxy URL to connect to. E.g.: https://examples.com/proxy
    clientKeyyesn/aThe Unleash Proxy Secret to be used
    appNamenounleash-swift-clientThe name of the application using this SDK. Will be used as part of the metrics sent to Unleash Proxy. Will also be part of the Unleash Context.
    environmentnodefaultThe name of the environment using this SDK. Will be used as part of the metrics sent to Unleash Proxy. Will also be part of the Unleash Context.
    refreshIntervalno15How often, in seconds, the SDK should check for updated toggle configuration. If set to 0 will disable checking for updates
    metricsIntervalno30How often, in seconds, the SDK should send usage metrics back to Unleash Proxy
    disableMetricsnofalseSet this option to true if you want to disable usage metrics
    contextno[:]The initial context parameters except from appName and `environment which are specified as top level fields

    Update context

    In order to update the context you can use the following method:

    var context: [String: String] = [:]
    context["userId"] = "c3b155b0-5ebe-4a20-8386-e0cab160051e"
    unleash.updateContext(context: context)

    This will stop and start the polling interval in order to renew polling with new context values.

    You can use any of the predefined fields. If you need to support -custom properties pass them as the second argument:

    var context: [String: String] = [:]
    context["userId"] = "c3b155b0-5ebe-4a20-8386-e0cab160051e"
    var properties: [String: String] = [:]
    properties["customKey"] = "customValue";
    unleash.updateContext(context: context, properties: properties)

    Events

    The proxy client emits events that you can subscribe to. The following events are available:

    • "ready"
    • "update"
    • "sent" (metrics sent)
    • "error" (metrics sending error)

    Usage them in the following manner:

    func handleReady() {
    // do this when unleash is ready
    }

    unleash.subscribe(name: "ready", callback: handleReady)

    func handleUpdate() {
    // do this when unleash is updated
    }

    unleash.subscribe(name: "update", callback: handleUpdate)

    The ready event is fired once the client has received it's first set of feature toggles and cached it in memory. Every subsequent event will be an update event that is triggered if there is a change in the feature toggle configuration.

    Releasing

    Note: To release the package you'll need to have CocoaPods installed.

    First, you'll need to add a tag. Releasing the tag is enough for the Swift package manager, but it's polite to also ensure CocoaPods users can also consume the code.

    git tag -a 0.0.4 -m "v0.0.4"

    Please make sure that that tag is pushed to remote.

    The next few commands assume that you have CocoaPods installed and available on your shell.

    First, validate your session with cocoapods with the following command:

    pod trunk register <email> "Your name"

    The email that owns this package is the general unleash team email. Cocoapods will send a link to this email, click it to validate your shell session.

    Bump the version number of the package, you can find this in UnleashProxyClientSwift.podspec, we use SemVer for this project. Once that's committed and merged to main:

    Linting the podspec is always a good idea:

    pod spec lint UnleashProxyClientSwift.podspec

    Once that succeeds, you can do the actual release:

    pod trunk push UnleashProxyClientSwift.podspec --allow-warnings

    Testing

    In order to test this package you can run the swift test command. To test thread safety, run swift test with:

    swift test --sanitize=thread

    This will give you warnings in the console when you have any data races.


    This content was generated on

    - - +custom properties pass them as the second argument:

    var context: [String: String] = [:]
    context["userId"] = "c3b155b0-5ebe-4a20-8386-e0cab160051e"
    var properties: [String: String] = [:]
    properties["customKey"] = "customValue";
    unleash.updateContext(context: context, properties: properties)

    Events

    The proxy client emits events that you can subscribe to. The following events are available:

    • "ready"
    • "update"
    • "sent" (metrics sent)
    • "error" (metrics sending error)

    Usage them in the following manner:

    func handleReady() {
    // do this when unleash is ready
    }

    unleash.subscribe(name: "ready", callback: handleReady)

    func handleUpdate() {
    // do this when unleash is updated
    }

    unleash.subscribe(name: "update", callback: handleUpdate)

    The ready event is fired once the client has received it's first set of feature toggles and cached it in memory. Every subsequent event will be an update event that is triggered if there is a change in the feature toggle configuration.

    Releasing

    Note: To release the package you'll need to have CocoaPods installed.

    First, you'll need to add a tag. Releasing the tag is enough for the Swift package manager, but it's polite to also ensure CocoaPods users can also consume the code.

    git tag -a 0.0.4 -m "v0.0.4"

    Please make sure that that tag is pushed to remote.

    The next few commands assume that you have CocoaPods installed and available on your shell.

    First, validate your session with cocoapods with the following command:

    pod trunk register <email> "Your name"

    The email that owns this package is the general unleash team email. Cocoapods will send a link to this email, click it to validate your shell session.

    Bump the version number of the package, you can find this in UnleashProxyClientSwift.podspec, we use SemVer for this project. Once that's committed and merged to main:

    Linting the podspec is always a good idea:

    pod spec lint UnleashProxyClientSwift.podspec

    Once that succeeds, you can do the actual release:

    pod trunk push UnleashProxyClientSwift.podspec --allow-warnings

    Testing

    In order to test this package you can run the swift test command. To test thread safety, run swift test with:

    swift test --sanitize=thread

    This will give you warnings in the console when you have any data races.


    This content was generated on

    + + \ No newline at end of file diff --git a/reference/sdks/java.html b/reference/sdks/java.html index 0c4fe004fd..88cae9f4bd 100644 --- a/reference/sdks/java.html +++ b/reference/sdks/java.html @@ -20,8 +20,8 @@ - - + + @@ -54,8 +54,8 @@ Unleash do come with a FakeUnleash implementation for doing this.

    Some examples on how to use it below:

    // example 1: everything on
    FakeUnleash fakeUnleash = new FakeUnleash();
    fakeUnleash.enableAll();

    assertThat(fakeUnleash.isEnabled("unknown"), is(true));
    assertThat(fakeUnleash.isEnabled("unknown2"), is(true));

    // example 2
    FakeUnleash fakeUnleash = new FakeUnleash();
    fakeUnleash.enable("t1", "t2");

    assertThat(fakeUnleash.isEnabled("t1"), is(true));
    assertThat(fakeUnleash.isEnabled("t2"), is(true));
    assertThat(fakeUnleash.isEnabled("unknown"), is(false));

    // example 3: variants
    FakeUnleash fakeUnleash = new FakeUnleash();
    fakeUnleash.enable("t1", "t2");
    fakeUnleash.setVariant("t1", new Variant("a", (String) null, true));

    assertThat(fakeUnleash.getVariant("t1").getName(), is("a"));

    Se more in FakeUnleashTest.java

    Development

    Build:

    mvn clean install

    Jacoco coverage reports:

    mvn jacoco:report

    The generated report will be available at target/site/jacoco/index.html

    Formatting

    Releasing

    Deployment

    • You'll need an account with Sonatype's JIRA - https://issues.sonatype.org
    • In addition your account needs access to publish under io.getunleash

    GPG signing

    • You'll need gpg installed and a configured gpg key for signing the artifacts

    Example settings.xml

    • In ~/.m2/settings.xml put
    <settings>
    ...
    <profiles>
    ...
    <profile>
    <id>ossrh</id>
    <activation>
    <activeByDefault>true</activeByDefault>
    </activation>
    <properties>
    <gpg.executable>gpg</gpg.executable> <!-- Where to find gpg -->
    <gpg.passphrase>[PASSPHRASE FOR YOUR GPG KEY]</gpg.passphrase>
    </properties>
    </profile>
    </profiles>
    ...
    <servers>
    ...
    <server>
    <id>sonatype-nexus-snapshots</id>
    <username>[YOUR_SONATYPE_JIRA_USERNAME]</username>
    <password>[YOUR_SONATYPE_JIRA_PASSWORD]</password>
    </server>
    <server>
    <id>ossrh</id>
    <username>[YOUR_SONATYPE_JIRA_USERNAME]</username>
    <password>[YOUR_SONATYPE_JIRA_PASSWORD]</password>
    </server>
    <server>
    <id>sonatype-nexus-staging</id>
    <username>[YOUR_SONATYPE_JIRA_USERNAME]</username>
    <password>[YOUR_SONATYPE_JIRA_PASSWORD]</password>
    </server>
    </servers>
    </settings>

    More information

    Configuration options

    The UnleashConfig$Builder class (created via UnleashConfig.builder()) exposes a set of builder methods to configure your Unleash client. The available options are listed below with a description of what they do. For the full signatures, take a look at the UnleashConfig class definition.

    Method nameDescriptionRequiredDefault value
    apiKeyThe api key to use for authenticating against the Unleash API.Yesnull
    appNameThe name of the application as shown in the Unleash UI. Registered applications are listed on the Applications page.Yesnull
    backupFileThe path to the file where local backups get stored.NoSynthesized from your system's java.io.tmpdir and your appName: "<java.io.tmpdir>/unleash-<appName>-repo.json"
    customHttpHeaderAdd a custom HTTP header to the list of HTTP headers that will the client sends to the Unleash API. Each method call will add a new header. Note: in most cases, you'll need to use this method to provide an API token.NoN/A
    customHttpHeadersProviderAdd a custom HTTP header provider. Useful for dynamic custom HTTP headers.Nonull
    disablePollingA boolean indicating whether the client should poll the unleash api for updates to toggles.
    disableMetricsA boolean indicating whether the client should disable sending usage metrics to the Unleash server.Nofalse
    enableProxyAuthenticationByJvmPropertiesEnable support for using JVM properties for HTTP proxy authentication.Nofalse
    environmentThe value to set for the Unleash context's environment property. Not the same as Unleash's environments.Nonull
    fallbackStrategyA strategy implementation that the client can use if it doesn't recognize the strategy type returned from the server.Nonull
    fetchTogglesIntervalHow often (in seconds) the client should check for toggle updates. Set to 0 if you want to only check once.No10
    instanceIdA unique(-ish) identifier for your instance. Typically a hostname, pod id or something similar. Unleash uses this to separate metrics from the client SDKs with the same appName.Yesnull
    namePrefixIf provided, the client will only fetch toggles whose name starts with the provided value.Nonull
    projectNameIf provided, the client will only fetch toggles from the specified project. (This can also be achieved with an API token).Nonull
    proxyA Proxy object. Use this to configure a third-party proxy that sits between your client and the Unleash server.Nonull
    scheduledExecutorA custom executor to control timing and running of tasks (such as fetching toggles, sending metrics).NoUnleashScheduledExecutorImpl
    sendMetricsIntervalHow often (in seconds) the client should send metrics to the Unleash server. Ignored if you disable metrics with the disableMetrics method.No60
    subscriberRegister a subscriber to Unleash client events.Nonull
    synchronousFetchOnInitialisationWhether the client should fetch toggle configuration synchronously (in a blocking manner) on initialisation.Nofalse
    toggleBootstrapProviderAdd a bootstrap provider (must implement the ToggleBootstrapProvider interface)No
    unleashAPIThe URL of the Unleash API.Yesnull
    unleashContextProviderAn Unleash context provider used to configure Unleash.Nonull
    unleashFeatureFetcherFactoryA factory providing a FeatureFetcher implementation.NoHttpFeatureFetcher::new
    unleashMetricsSenderFactoryA factory providing a MetricSender implementation.NoDefaultHttpMetricsSender::new

    When you have set all the desired options, initialize the configuration with the build method. You can then pass the configuration to the Unleash client constructor. -As an example:


    UnleashConfig config = UnleashConfig.builder()
    .appName("your app name")
    .instanceId("instance id")
    .unleashAPI("http://unleash.herokuapp.com/api/")
    .apiKey("API token")
    // ... more configuration options
    .build();

    Unleash unleash = new DefaultUnleash(config);

    Other information

    • Check out our guide for more information on how to build and scale feature flag systems

    This content was generated on

    - - +As an example:


    UnleashConfig config = UnleashConfig.builder()
    .appName("your app name")
    .instanceId("instance id")
    .unleashAPI("http://unleash.herokuapp.com/api/")
    .apiKey("API token")
    // ... more configuration options
    .build();

    Unleash unleash = new DefaultUnleash(config);

    Other information

    • Check out our guide for more information on how to build and scale feature flag systems

    This content was generated on

    + + \ No newline at end of file diff --git a/reference/sdks/javascript-browser.html b/reference/sdks/javascript-browser.html index bfc4af923d..98a61b3490 100644 --- a/reference/sdks/javascript-browser.html +++ b/reference/sdks/javascript-browser.html @@ -20,8 +20,8 @@ - - + + @@ -33,8 +33,8 @@ This is a neat way to update a single page app when toggle state updates.

    unleash.on('update', () => {
    const myToggle = unleash.isEnabled('proxy.demo');
    //do something useful
    });

    Available events:

    • error - emitted when an error occurs on init, or when fetch function fails, or when fetch receives a non-ok response object. The error object is sent as payload.
    • initialized - emitted after the SDK has read local cached data in the storageProvider.
    • ready - emitted after the SDK has successfully started and performed the initial fetch towards the Unleash Proxy.
    • update - emitted every time the Unleash Proxy return a new feature toggle configuration. The SDK will emit this event as part of the initial fetch from the SDK.
    • recovered - emitted when the SDK has recovered from an error. This event will only be emitted if the SDK has previously emitted an error.
    • sent - emitted when the SDK has successfully sent metrics to Unleash.

    PS! Please remember that you should always register your event listeners before your call unleash.start(). If you register them after you have started the SDK you risk loosing important events.

    SessionId - Important note!

    You may provide a custom session id via the "context". If you do not provide a sessionId this SDK will create a random session id, which will also be stored in the provided storage (local storage). By always having a consistent sessionId available ensures that even "anonymous" users will get a consistent experience when feature toggles is evaluated, in combination with a gradual (percentage based) rollout.

    Stop the SDK

    You can stop the Unleash client by calling the stop method. Once the client has been stopped, it will no longer check for updates or send metrics to the server.

    A stopped client can be restarted.

    unleash.stop()

    Custom store

    This SDK can work with React Native storage @react-native-async-storage/async-storage or react-native-shared-preferences and many more to backup feature toggles locally. This is useful for bootstrapping the SDK the next time the user comes back to your application.

    You can provide your own storage implementation.

    Examples:

    import SharedPreferences from 'react-native-shared-preferences';
    import { UnleashClient } from 'unleash-proxy-client';

    const unleash = new UnleashClient({
    url: 'https://eu.unleash-hosted.com/hosted/proxy',
    clientKey: 'your-proxy-key',
    appName: 'my-webapp',
    storageProvider: {
    save: (name: string, data: any) => SharedPreferences.setItem(name, data),
    get: (name: string) => SharedPreferences.getItem(name, (val) => val)
    },
    });
    import AsyncStorage from '@react-native-async-storage/async-storage';
    import { UnleashClient } from 'unleash-proxy-client';

    const PREFIX = 'unleash:repository';

    const unleash = new UnleashClient({
    url: 'https://eu.unleash-hosted.com/hosted/proxy',
    clientKey: 'your-proxy-key',
    appName: 'my-webapp',
    storageProvider: {
    save: (name: string, data: any) => {
    const repo = JSON.stringify(data);
    const key = `${PREFIX}:${name}`;
    return AsyncStorage.setItem(key, repo);
    },
    get: (name: string) => {
    const key = `${PREFIX}:${name}`;
    const data = await AsyncStorage.getItem(key);
    return data ? JSON.parse(data) : undefined;
    }
    },
    });

    How to use in node.js

    This SDK can also be used in node.js applications (from v1.4.0). Please note that you will need to provide a valid "fetch" implementation. Only ECMAScript modules is exported from this package.

    import fetch from 'node-fetch';
    import { UnleashClient, InMemoryStorageProvider } from 'unleash-proxy-client';

    const unleash = new UnleashClient({
    url: 'https://app.unleash-hosted.com/demo/proxy',
    clientKey: 'proxy-123',
    appName: 'nodejs-proxy',
    storageProvider: new InMemoryStorageProvider(),
    fetch,
    });

    await unleash.start();
    const isEnabled = unleash.isEnabled('proxy.demo');
    console.log(isEnabled);

    index.mjs

    How to use the client via CDN.

    <html>
    <head>
    <script src="https://unpkg.com/unleash-proxy-client@latest/build/main.min.js" type="text/javascript"></script>
    <script type="text/javascript">
    var config = {url: 'https://app.unleash-hosted.com/demo/proxy', clientKey: 'proxy-123', appName: 'web'};
    var client = new unleash.UnleashClient(config);
    client.updateContext({userId: '1233'})

    client.on('update', () => {
    console.log(client.isEnabled('proxy.demo'));
    });
    client.start();
    </script>
    </head>
    </html>

    Bootstrap

    Now it is possible to bootstrap the SDK with your own feature toggle configuration when you don't want to make an API call.

    This is also useful if you require the toggles to be in a certain state immediately after initializing the SDK.

    How to use it ?

    Add a bootstrap attribute when create a new UnleashClient.
    There's also a bootstrapOverride attribute which is by default is true.

    import { UnleashClient } from 'unleash-proxy-client';

    const unleash = new UnleashClient({
    url: 'https://app.unleash-hosted.com/demo/proxy',
    clientKey: 'proxy-123',
    appName: 'nodejs-proxy',
    bootstrap: [{
    "enabled": true,
    "name": "demoApp.step4",
    "variant": {
    "enabled": true,
    "name": "blue",
    "feature_enabled": true,
    }
    }],
    bootstrapOverride: false
    });

    NOTES: ⚠️ If bootstrapOverride is true (by default), any local cached data will be overridden with the bootstrap specified.
    -If bootstrapOverride is false any local cached data will not be overridden unless the local cache is empty.


    This content was generated on

    - - +If bootstrapOverride is false any local cached data will not be overridden unless the local cache is empty.


    This content was generated on

    + + \ No newline at end of file diff --git a/reference/sdks/next-js.html b/reference/sdks/next-js.html index 671c3fd930..417f14421e 100644 --- a/reference/sdks/next-js.html +++ b/reference/sdks/next-js.html @@ -20,8 +20,8 @@ - - + + @@ -37,8 +37,8 @@ You can find out more about direct Front-end API access in our documentation, including a guide on how to setup a client-side SDK key.

    Important: Hooks and provider are only available in @unleash/nextjs/client.

    import type { AppProps } from "next/app";
    import { FlagProvider } from "@unleash/nextjs/client";

    export default function App({ Component, pageProps }: AppProps) {
    return (
    <FlagProvider>
    <Component {...pageProps} />
    </FlagProvider>
    );
    }

    With <FlagProvider /> in place you can now use hooks like: useFlag, useVariant, or useFlagsStatus to block rendering until flags are ready.

    import { useFlag } from "@unleash/nextjs/client";

    const YourComponent = () => {
    const isEnabled = useFlag("nextjs-example");

    return <>{isEnabled ? "ENABLED" : "DISABLED"}</>;
    };

    Optionally, you can configure FlagProvider with the config prop. It will take priority over environment variables.

    <FlagProvider
    config={{
    url: "http://localhost:4242/api/frontend", // replaces NEXT_PUBLIC_UNLEASH_FRONTEND_API_URL
    clientKey: "<Frontend_API_token>", // replaces NEXT_PUBLIC_UNLEASH_FRONTEND_API_TOKEN
    appName: "nextjs", // replaces NEXT_PUBLIC_UNLEASH_APP_NAME

    refreshInterval: 15, // additional client configuration
    // see https://github.com/Unleash/unleash-proxy-client-js#available-options
    }}
    >

    If you only plan to use Unleash client-side React SDK now also works with Next.js. Check documentation there for more examples.

    D). Static Site Generation, optimized performance (SSG)

    With same access as in the client-side example above you can resolve Unleash feature flags when building static pages.

    import {
    flagsClient,
    getDefinitions,
    evaluateFlags,
    getFrontendFlags,
    type IVariant,
    } from "@unleash/nextjs";
    import type { GetStaticProps, NextPage } from "next";

    type Data = {
    isEnabled: boolean;
    variant: IVariant;
    };

    const ExamplePage: NextPage<Data> = ({ isEnabled, variant }) => (
    <>
    Flag status: {isEnabled ? "ENABLED" : "DISABLED"}
    <br />
    Variant: {variant.name}
    </>
    );

    export const getStaticProps: GetStaticProps<Data> = async (_ctx) => {
    /* Using server-side SDK: */
    const definitions = await getDefinitions();
    const context = {}; // optional, see https://docs.getunleash.io/reference/unleash-context
    const { toggles } = evaluateFlags(definitions, context);

    /* Or with the proxy/front-end API */
    // const { toggles } = await getFrontendFlags({ context });

    const flags = flagsClient(toggles);

    return {
    props: {
    isEnabled: flags.isEnabled("nextjs-example"),
    variant: flags.getVariant("nextjs-example"),
    },
    };
    };

    export default ExamplePage;

    The same approach will work for ISR (Incremental Static Regeneration).

    Both getDefinitions() and getFrontendFlags() can take arguments overriding URL, token and other request parameters.

    E). Server Side Rendering (SSR)

    import {
    flagsClient,
    evaluateFlags,
    getDefinitions,
    type IVariant,
    } from "@unleash/nextjs";
    import type { GetServerSideProps, NextPage } from "next";

    type Data = {
    isEnabled: boolean;
    };

    const ExamplePage: NextPage<Data> = ({ isEnabled }) => (
    <>Flag status: {isEnabled ? "ENABLED" : "DISABLED"}</>
    );

    export const getServerSideProps: GetServerSideProps<Data> = async (ctx) => {
    const sessionId =
    ctx.req.cookies["unleash-session-id"] ||
    `${Math.floor(Math.random() * 1_000_000_000)}`;
    ctx.res.setHeader("set-cookie", `unleash-session-id=${sessionId}; path=/;`);

    const context = {
    sessionId, // needed for stickiness
    // userId: "123" // etc
    };

    const { toggles } = await getFrontendFlags({ context }); // Use Proxy/Frontend API
    const flags = flagsClient(toggles);

    return {
    props: {
    isEnabled: flags.isEnabled("nextjs-example"),
    },
    };
    };

    export default ExamplePage;

    F). Bootstrapping / rehydration

    You can bootstrap Unleash React SDK to have values loaded from the start. Initial value can be customized server-side.

    import App, { AppContext, type AppProps } from "next/app";
    import {
    FlagProvider,
    getFrontendFlags,
    type IMutableContext,
    type IToggle,
    } from "@unleash/nextjs";

    type Data = {
    toggles: IToggle[];
    context: IMutableContext;
    };

    export default function CustomApp({
    Component,
    pageProps,
    toggles,
    context,
    }: AppProps & Data) {
    return (
    <FlagProvider
    config={{
    bootstrap: toggles,
    context,
    }}
    >
    <Component {...pageProps} />
    </FlagProvider>
    );
    }

    CustomApp.getInitialProps = async (ctx: AppContext) => {
    const context = {
    userId: "123",
    };

    const { toggles } = await getFrontendFlags(); // use Unleash Proxy

    return {
    ...(await App.getInitialProps(ctx)),
    bootstrap: toggles,
    context, // pass context along so client can refetch correct values
    };
    };

    ⚗️ CLI (experimental)

    You can use unleash [action] [options] in your package.json scripts section, or with:

    npx @unleash/nextjs

    For the CLI to work, the following environment variables must be configured with appropriate values:

    • UNLEASH_SERVER_API_URL
    • UNLEASH_SERVER_API_TOKEN

    The CLI will attempt to read environment values from any .env files if they're present. You can also set the variables directly when invoking the interface, as in the CLI usage example.

    Usage

    • get-definitions <outputFile.json> Download feature flags definitions for bootstrapping (offline use) of server-side SDK.
    • generate-types [options] <outputFile.ts> Generate types and typed functions from feature flags defined in an Unleash instance. -It will also generate strictly typed versions of useFlag, useVariant, useFlags and flagsClient (unless --typesOnly is used).
      • -t, --typesOnly don't include typed versions of functions exported from @unleash/nextjs (default: false)
      • -b, --bootstrap <sourceFile.json> load definitions from a file instead of fetching definitions - work offline
    • -V Output the version number

    Example

    Try it now

    UNLEASH_SERVER_API_URL=https://app.unleash-hosted.com/demo/api \
    UNLEASH_SERVER_API_TOKEN=test-server:default.8a090f30679be7254af997864d66b86e44dcfc5291916adff72a0fb5 \
    npx @unleash/nextjs generate-types ./unleash.ts

    Known limitation

    • In current interation server-side SDK does not support metrics.
    • When used server-side, this SDK does not support the "Hostname" and "IP" strategies. Use custom context fields and constraints instead.

    This content was generated on

    - - +It will also generate strictly typed versions of useFlag, useVariant, useFlags and flagsClient (unless --typesOnly is used).
    • -t, --typesOnly don't include typed versions of functions exported from @unleash/nextjs (default: false)
    • -b, --bootstrap <sourceFile.json> load definitions from a file instead of fetching definitions - work offline
  • -V Output the version number
  • Example

    Try it now

    UNLEASH_SERVER_API_URL=https://app.unleash-hosted.com/demo/api \
    UNLEASH_SERVER_API_TOKEN=test-server:default.8a090f30679be7254af997864d66b86e44dcfc5291916adff72a0fb5 \
    npx @unleash/nextjs generate-types ./unleash.ts

    Known limitation

    • In current interation server-side SDK does not support metrics.
    • When used server-side, this SDK does not support the "Hostname" and "IP" strategies. Use custom context fields and constraints instead.

    This content was generated on

    + + \ No newline at end of file diff --git a/reference/sdks/node.html b/reference/sdks/node.html index 2946c3c9f0..0a532594dd 100644 --- a/reference/sdks/node.html +++ b/reference/sdks/node.html @@ -20,8 +20,8 @@ - - + + @@ -84,8 +84,8 @@ offline, from a browser environment or implement your own caching layer. See example.

    Unleash depends on a ready event of the repository you pass in. Be sure that you emit the event after you've initialized unleash.

    Design philosophy

    This feature flag SDK is designed according to our design philosophy. You can -read more about that here.


    This content was generated on

    - - +read more about that here.


    This content was generated on

    + + \ No newline at end of file diff --git a/reference/sdks/php.html b/reference/sdks/php.html index 4475cd4304..1bba1b4a7b 100644 --- a/reference/sdks/php.html +++ b/reference/sdks/php.html @@ -20,8 +20,8 @@ - - + + @@ -83,8 +83,8 @@ the code.

    <?php

    use Unleash\Client\UnleashBuilder;

    $unleash = UnleashBuilder::create()
    ->withAppName('Some App Name')
    ->withAppUrl('https://somewhere.com')
    ->withInstanceId('some-instance-id')
    ->withMetricsEnabled(false) // turn off metric sending
    ->withMetricsEnabled(true) // turn on metric sending
    ->withMetricsInterval(10_000) // interval in milliseconds (10 seconds)
    ->withMetricsCacheHandler(new Psr16Cache(new RedisAdapter())) // use custom cache handler for metrics, defaults to standard cache handler
    ->build();

    // the metric will be collected but not sent immediately
    $unleash->isEnabled('test');
    sleep(10);
    // now the metrics will get sent
    $unleash->isEnabled('test');

    Custom headers via middleware

    While middlewares for http client are not natively supported by this SDK, you can pass your own http client which supports them.

    The most popular http client, guzzle, supports them out of the box and here's an example of how to pass custom headers automatically (for more information visit official guzzle documentation on middlewares):

    <?php

    use GuzzleHttp\Client;
    use GuzzleHttp\Handler\CurlHandler;
    use GuzzleHttp\HandlerStack;
    use GuzzleHttp\Middleware;
    use Psr\Http\Message\RequestInterface;
    use Unleash\Client\UnleashBuilder;

    // any callable is valid, it may be a function reference, anonymous function or an invokable class

    // example invokable class
    final class AddHeaderMiddleware
    {
    public function __construct(
    private readonly string $headerName,
    private readonly string $value,
    ) {
    }

    public function __invoke(RequestInterface $request): RequestInterface
    {
    return $request->withHeader($this->headerName, $this->value);
    }
    }

    // example anonymous function
    $addHeaderMiddleware = fn (string $headerName, string $headerValue)
    => fn(RequestInterface $request)
    => $request->withHeader($headerName, $headerValue);

    // create a handler stack that holds information about all middlewares
    $stack = HandlerStack::create(new CurlHandler());
    // mapRequest is a helper that simplifies modifying request
    $stack->push(Middleware::mapRequest(new AddHeaderMiddleware('X-My-Header', 'Some-Value')));
    // or with lambda
    $stack->push(Middleware::mapRequest($addHeaderMiddleware('X-My-Header2', 'Some-Value')));
    // assign the stack with middlewares as a handler
    $httpClient = new Client(['handler' => $stack]);

    $unleash = UnleashBuilder::create()
    ->withHttpClient($httpClient) // assign the custom http client
    ->withAppName('My-App')
    ->withInstanceId('My-Instance')
    ->withAppUrl('http://localhost:4242')
    ->build();

    // now every http request will have X-My-Header header with value Some-Value
    $unleash->isEnabled('some-feature');

    Constraints

    Constraints are supported by this SDK and will be handled correctly by Unleash::isEnabled() if present.

    GitLab specifics

    • In GitLab you have to use the provided instance id, you cannot create your own.
    • No authorization header is necessary.
    • Instead of app name you need to specify the GitLab environment.
      • For this purpose you can use withGitlabEnvironment() method in builder, it's an alias to withAppName() but -communicates the intent better.
    • GitLab doesn't use registration system, you can set the SDK to disable automatic registration and save one http call.
    • GitLab doesn't read metrics, you can set the SDK to disable sending them and save some http calls.
    <?php

    use Unleash\Client\UnleashBuilder;

    $gitlabUnleash = UnleashBuilder::createForGitlab()
    ->withInstanceId('H9sU9yVHVAiWFiLsH2Mo') // generated in GitLab
    ->withAppUrl('https://git.example.com/api/v4/feature_flags/unleash/1')
    ->withGitlabEnvironment('Production')
    ->build();

    // the above is equivalent to
    $gitlabUnleash = UnleashBuilder::create()
    ->withInstanceId('H9sU9yVHVAiWFiLsH2Mo')
    ->withAppUrl('https://git.example.com/api/v4/feature_flags/unleash/1')
    ->withGitlabEnvironment('Production')
    ->withAutomaticRegistrationEnabled(false)
    ->withMetricsEnabled(false)
    ->build();

    Check out our guide for more information on how to build and scale feature flag systems


    This content was generated on

    - - +communicates the intent better.
  • GitLab doesn't use registration system, you can set the SDK to disable automatic registration and save one http call.
  • GitLab doesn't read metrics, you can set the SDK to disable sending them and save some http calls.
  • <?php

    use Unleash\Client\UnleashBuilder;

    $gitlabUnleash = UnleashBuilder::createForGitlab()
    ->withInstanceId('H9sU9yVHVAiWFiLsH2Mo') // generated in GitLab
    ->withAppUrl('https://git.example.com/api/v4/feature_flags/unleash/1')
    ->withGitlabEnvironment('Production')
    ->build();

    // the above is equivalent to
    $gitlabUnleash = UnleashBuilder::create()
    ->withInstanceId('H9sU9yVHVAiWFiLsH2Mo')
    ->withAppUrl('https://git.example.com/api/v4/feature_flags/unleash/1')
    ->withGitlabEnvironment('Production')
    ->withAutomaticRegistrationEnabled(false)
    ->withMetricsEnabled(false)
    ->build();

    Check out our guide for more information on how to build and scale feature flag systems


    This content was generated on

    + + \ No newline at end of file diff --git a/reference/sdks/python.html b/reference/sdks/python.html index 021c6746e1..70d86cab91 100644 --- a/reference/sdks/python.html +++ b/reference/sdks/python.html @@ -20,16 +20,16 @@ - - + +

    Python

    Generated content

    This document was generated from the README in the Python GitHub repository.

    tip

    To connect to Unleash, you'll need your Unleash API url (e.g. https://<your-unleash>/api) and a server-side API token (how do I create an API token?).

    unleash-client-python

    Coverage Status PyPI version PyPI - Python Version License: MIT

    This is the Python client for Unleash. It implements Client Specifications 1.0 and checks compliance based on spec in unleash/client-specifications

    What it supports:

    • Default activation strategies using 32-bit Murmurhash3
    • Custom strategies
    • Full client lifecycle:
      • Client registers with Unleash server
      • Client periodically fetches feature toggles and stores to on-disk cache
      • Client periodically sends metrics to Unleash Server
    • Tested on Linux (Ubuntu), OSX, and Windows

    Check out the project documentation and the changelog.

    Installation

    Check out the package on Pypi!

    pip install UnleashClient

    For Flask Users

    If you're looking into running Unleash from Flask, you might want to take a look at Flask-Unleash, the Unleash Flask extension. The extension builds upon this SDK to reduce the amount of boilerplate code you need to write to integrate with Flask. Of course, if you'd rather use this package directly, that will work too.

    Usage

    Initialization

    from UnleashClient import UnleashClient

    client = UnleashClient(
    url="https://unleash.herokuapp.com",
    app_name="my-python-app",
    custom_headers={'Authorization': '<API token>'})

    client.initialize_client()

    For more information about configuring UnleashClient, check out the project reference docs!

    Checking if a feature is enabled

    A check of a simple toggle:

    client.is_enabled("my_toggle")

    To supply application context, use the second positional argument:

    app_context = {"userId": "test@email.com"}
    client.is_enabled("user_id_toggle", app_context)

    Fallback function and default values

    You can specify a fallback function for cases where the client doesn't recognize the toggle by using the fallback_function keyword argument:

    def custom_fallback(feature_name: str, context: dict) -> bool:
    return True

    client.is_enabled("my_toggle", fallback_function=custom_fallback)

    You can also use the fallback_function argument to replace the obsolete default_value keyword argument by using a lambda that ignores its inputs. Whatever the lambda returns will be used as the default value.

    client.is_enabled("my_toggle", fallback_function=lambda feature_name, context: True)

    The fallback function must accept the feature name and context as positional arguments in that order.

    The client will evaluate the fallback function only if an exception occurs when calling the is_enabled() method. This happens when the client can't find the feature flag. The client may also throw other, general exceptions.

    For more information about usage, see the Usage documentation.

    Getting a variant

    Checking for a variant:

    context = {'userId': '2'}  # Context must have userId, sessionId, or remoteAddr.  If none are present, distribution will be random.

    variant = client.get_variant("variant_toggle", context)

    print(variant)
    > {
    > "name": "variant1",
    > "payload": {
    > "type": "string",
    > "value": "val1"
    > },
    > "enabled": True
    > }

    For more information about variants, see the Variant documentation.

    Developing

    For development, you'll need to setup the environment to run the tests. This repository is using -tox to run the test suite to test against multiple versions of Python. Running the tests is as simple as running this command in the makefile:

    tox -e py311

    This command will take care of downloading the client specifications and putting them in the correct place in the repository, and install all the dependencies you need.

    However, there are some caveats to this method. There is no easy way to run a single test, and running the entire test suite can be slow.

    Manual setup

    First, make sure you have pip or pip3 installed.

    Then setup your viritual environment:

    Linux & Mac:

    python3 -m venv venv
    source venv/bin/activate

    Windows + cmd:

    python -m venv venv
    venv\Scripts\activate.bat

    Powershell:

    python -m venv venv
    venv\Scripts\activate.bat

    Once you've done your setup, run:

    pip install -r requirements.txt

    Run the get-spec script to download the client specifications tests:

    ./scripts/get-spec.sh

    Now you can run the tests by running pytest in the root directory.

    In order to run a single test, run the following command:

    pytest testfile.py::function_name

    # example: pytest tests/unit_tests/test_client.py::test_consistent_results

    Linting

    In order to lint all the files you can run the following command:

    make fmt

    This content was generated on

    - - +tox to run the test suite to test against multiple versions of Python. Running the tests is as simple as running this command in the makefile:

    tox -e py311

    This command will take care of downloading the client specifications and putting them in the correct place in the repository, and install all the dependencies you need.

    However, there are some caveats to this method. There is no easy way to run a single test, and running the entire test suite can be slow.

    Manual setup

    First, make sure you have pip or pip3 installed.

    Then setup your viritual environment:

    Linux & Mac:

    python3 -m venv venv
    source venv/bin/activate

    Windows + cmd:

    python -m venv venv
    venv\Scripts\activate.bat

    Powershell:

    python -m venv venv
    venv\Scripts\activate.bat

    Once you've done your setup, run:

    pip install -r requirements.txt

    Run the get-spec script to download the client specifications tests:

    ./scripts/get-spec.sh

    Now you can run the tests by running pytest in the root directory.

    In order to run a single test, run the following command:

    pytest testfile.py::function_name

    # example: pytest tests/unit_tests/test_client.py::test_consistent_results

    Linting

    In order to lint all the files you can run the following command:

    make fmt

    This content was generated on

    + + \ No newline at end of file diff --git a/reference/sdks/react.html b/reference/sdks/react.html index ff243428a2..fc90294fdf 100644 --- a/reference/sdks/react.html +++ b/reference/sdks/react.html @@ -20,8 +20,8 @@ - - + + @@ -30,8 +30,8 @@

    React

    Generated content

    This document was generated from the README in the React GitHub repository.

    tip

    To connect to Unleash from a client-side context, you'll need to use the Unleash front-end API (how do I create an API token?) or the Unleash proxy (how do I create client keys?).

    Installation

    npm install @unleash/proxy-client-react unleash-proxy-client
    # or
    yarn add @unleash/proxy-client-react unleash-proxy-client

    How to use

    Initialize the client

    Prepare Unleash Proxy secret or Frontend API Access token.

    Import the provider like this in your entrypoint file (typically index.js/ts):

    import { createRoot } from 'react-dom/client';
    import { FlagProvider } from '@unleash/proxy-client-react';

    const config = {
    url: '<unleash-url>/api/frontend', // Your front-end API URL or the Unleash proxy's URL (https://<proxy-url>/proxy)
    clientKey: '<your-token>', // A client-side API token OR one of your proxy's designated client keys (previously known as proxy secrets)
    refreshInterval: 15, // How often (in seconds) the client should poll the proxy for updates
    appName: 'your-app-name', // The name of your application. It's only used for identifying your application
    };

    const root = createRoot(document.getElementById('root'));

    root.render(
    <React.StrictMode>
    <FlagProvider config={config}>
    <App />
    </FlagProvider>
    </React.StrictMode>
    );

    Connection options

    To connect this SDK to your Unleash instance's front-end API, use the URL to your Unleash instance's front-end API (<unleash-url>/api/frontend) as the url parameter. For the clientKey parameter, use a FRONTEND token generated from your Unleash instance. Refer to the how to create API tokens guide for the necessary steps.

    To connect this SDK to the Unleash proxy, use the proxy's URL and a proxy client key. The configuration section of the Unleash proxy docs contains more info on how to configure client keys for your proxy.

    Check feature toggle status

    To check if a feature is enabled:

    import { useFlag } from '@unleash/proxy-client-react';

    const TestComponent = () => {
    const enabled = useFlag('travel.landing');

    if (enabled) {
    return <SomeComponent />;
    }
    return <AnotherComponent />;
    };

    export default TestComponent;

    Check variants

    To check variants:

    import { useVariant } from '@unleash/proxy-client-react';

    const TestComponent = () => {
    const variant = useVariant('travel.landing');

    if (variant.enabled && variant.name === 'SomeComponent') {
    return <SomeComponent />;
    } else if (variant.enabled && variant.name === 'AnotherComponent') {
    return <AnotherComponent />;
    }
    return <DefaultComponent />;
    };

    export default TestComponent;

    Defer rendering until flags fetched

    useFlagsStatus retrieves the ready state and error events. Follow the following steps in order to delay rendering until the flags have been fetched.

    import { useFlagsStatus } from '@unleash/proxy-client-react';

    const MyApp = () => {
    const { flagsReady, flagsError } = useFlagsStatus();

    if (!flagsReady) {
    return <Loading />;
    }
    return <MyComponent error={flagsError} />;
    };

    Updating context

    Initial context can be specified on a FlagProvider config.context property.

    <FlagProvider config={{ ...config, context: { userId: 123 }}>

    This code sample shows you how to update the unleash context dynamically:

    import { useUnleashContext, useFlag } from '@unleash/proxy-client-react';

    const MyComponent = ({ userId }) => {
    const variant = useFlag('my-toggle');
    const updateContext = useUnleashContext();

    useEffect(() => {
    // context is updated with userId
    updateContext({ userId });
    }, [userId]);

    // OR if you need to perform an action right after new context is applied
    useEffect(() => {
    async function run() {
    // Can wait for the new flags to pull in from the different context
    await updateContext({ userId });
    console.log('new flags loaded for', userId);
    }
    run();
    }, [userId]);
    };

    Advanced use cases

    Deferring client start

    By default, the Unleash client will start polling the Proxy for toggles immediately when the FlagProvider component renders. You can prevent it by setting startClient prop to false. This is useful when you'd like to for example bootstrap the client and work offline.

    Deferring the client start gives you more fine-grained control over when to start fetching the feature toggle configuration. This could be handy in cases where you need to get some other context data from the server before fetching toggles, for instance.

    To start the client, use the client's start method. The below snippet of pseudocode will defer polling until the end of the asyncProcess function.

    const client = new UnleashClient({
    /* ... */
    });

    const MyAppComponent = () => {
    useEffect(() => {
    const asyncProcess = async () => {
    // do async work ...
    client.start();
    };
    asyncProcess();
    }, []);

    return (
    // Pass client as `unleashClient` and set `startClient` to `false`
    <FlagProvider unleashClient={client} startClient={false}>
    <App />
    </FlagProvider>
    );
    };

    Use unleash client directly

    import { useUnleashContext, useUnleashClient } from '@unleash/proxy-client-react'

    const MyComponent = ({ userId }) => {
    const client = useUnleashClient();

    const login = () => {
    // login user
    if (client.isEnabled("new-onboarding")) {
    // Send user to new onboarding flow
    } else (
    // send user to old onboarding flow
    )
    }

    return <LoginForm login={login}/>
    }

    Usage with class components

    Since this library uses hooks you have to implement a wrapper to use with class components. Beneath you can find an example of how to use this library with class components, using a custom wrapper:

    import React from 'react';
    import {
    useFlag,
    useUnleashClient,
    useUnleashContext,
    useVariant,
    useFlagsStatus,
    } from '@unleash/proxy-client-react';

    interface IUnleashClassFlagProvider {
    render: (props: any) => React.ReactNode;
    flagName: string;
    }

    export const UnleashClassFlagProvider = ({
    render,
    flagName,
    }: IUnleashClassFlagProvider) => {
    const enabled = useFlag(flagName);
    const variant = useVariant(flagName);
    const client = useUnleashClient();

    const updateContext = useUnleashContext();
    const { flagsReady, flagsError } = useFlagsStatus();

    const isEnabled = () => {
    return enabled;
    };

    const getVariant = () => {
    return variant;
    };

    const getClient = () => {
    return client;
    };

    const getUnleashContextSetter = () => {
    return updateContext;
    };

    const getFlagsStatus = () => {
    return { flagsReady, flagsError };
    };

    return (
    <>
    {render({
    isEnabled,
    getVariant,
    getClient,
    getUnleashContextSetter,
    getFlagsStatus,
    })}
    </>
    );
    };

    Wrap your components like so:

    <UnleashClassFlagProvider
    flagName="demoApp.step1"
    render={({ isEnabled, getClient }) => (
    <MyClassComponent isEnabled={isEnabled} getClient={getClient} />
    )}
    />

    React Native

    IMPORTANT: This no longer comes included in the unleash-proxy-client-js library. You will need to install the storage adapter for your preferred storage solution.

    Because React Native doesn't run in a web browser, it doesn't have access to the localStorage API. Instead, you need to tell Unleash to use your specific storage provider. The most common storage provider for React Native is AsyncStorage. -To configure it, add the following property to your configuration object:

    const config = {
    storageProvider: {
    save: (name, data) => AsyncStorage.setItem(name, JSON.stringify(data)),
    get: async (name) => {
    const data = await AsyncStorage.getItem(name);
    return data ? JSON.parse(data) : undefined;
    },
    },
    };

    Migration guide

    Upgrade path from v1 -> v2

    If you were previously using the built in Async storage used in the unleash-proxy-client-js, this no longer comes bundled with the library. You will need to install the storage adapter for your preferred storage solution. Otherwise there are no breaking changes.

    Upgrade path from v2 -> v3

    Previously the unleash client was bundled as dependency directly in this library. It's now changed to a peer dependency and listed as an external.

    In v2 there was only one distribution based on the fact that webpack polyfilled the necessary features in v4. This is no longer the case in webpack v5. We now provide two distribution builds, one for the server and one for the client - and use the browser field in the npm package to hint module builders about which version to use. The default dist/index.js file points to the node version, while the web build is located at dist/index.browser.js

    Upgrading should be as easy as running yarn again with the new version, but we made the made bump regardless to be safe. Note: If you are not able to resolve the peer dependency on unleash-proxy-client you might need to run npm install unleash-proxy-client

    Upgrade path from v3 -> v4

    startClient option has been simpilfied. Now it will also work if you don't pass custom client with it, and in SSR (when typeof window === 'undefined') it defaults to false.

    Note on v4.0.0:

    The major release is driven by Node14 end of life and represents no other changes. From this version onwards we do not guarantee that this library will work server side with Node 14.

    Design philosophy

    This feature flag SDK is designed according to our design philosophy. You can read more about that here.


    This content was generated on

    - - +To configure it, add the following property to your configuration object:

    const config = {
    storageProvider: {
    save: (name, data) => AsyncStorage.setItem(name, JSON.stringify(data)),
    get: async (name) => {
    const data = await AsyncStorage.getItem(name);
    return data ? JSON.parse(data) : undefined;
    },
    },
    };

    Migration guide

    Upgrade path from v1 -> v2

    If you were previously using the built in Async storage used in the unleash-proxy-client-js, this no longer comes bundled with the library. You will need to install the storage adapter for your preferred storage solution. Otherwise there are no breaking changes.

    Upgrade path from v2 -> v3

    Previously the unleash client was bundled as dependency directly in this library. It's now changed to a peer dependency and listed as an external.

    In v2 there was only one distribution based on the fact that webpack polyfilled the necessary features in v4. This is no longer the case in webpack v5. We now provide two distribution builds, one for the server and one for the client - and use the browser field in the npm package to hint module builders about which version to use. The default dist/index.js file points to the node version, while the web build is located at dist/index.browser.js

    Upgrading should be as easy as running yarn again with the new version, but we made the made bump regardless to be safe. Note: If you are not able to resolve the peer dependency on unleash-proxy-client you might need to run npm install unleash-proxy-client

    Upgrade path from v3 -> v4

    startClient option has been simpilfied. Now it will also work if you don't pass custom client with it, and in SSR (when typeof window === 'undefined') it defaults to false.

    Note on v4.0.0:

    The major release is driven by Node14 end of life and represents no other changes. From this version onwards we do not guarantee that this library will work server side with Node 14.

    Design philosophy

    This feature flag SDK is designed according to our design philosophy. You can read more about that here.


    This content was generated on

    + + \ No newline at end of file diff --git a/reference/sdks/ruby.html b/reference/sdks/ruby.html index 1aca135935..fc2de0baa5 100644 --- a/reference/sdks/ruby.html +++ b/reference/sdks/ruby.html @@ -20,8 +20,8 @@ - - + + @@ -46,8 +46,8 @@ In order for strategy to work correctly it should support two methods name and is_enabled?

    class MyCustomStrategy
    def name
    'myCustomStrategy'
    end

    def is_enabled?(params = {}, context = nil)
    true
    end
    end

    Unleash.configure do |config|
    config.strategies.add(MyCustomStrategy.new)
    end

    Development

    After checking out the repo, run bin/setup to install dependencies. Then, run rake spec to run the tests. You can also run bin/console for an interactive prompt that will allow you to experiment.

    This SDK is also built against the Unleash Client Specification tests. -To run the Ruby SDK against this test suite, you'll need to have a copy on your machine, you can clone the repository directly using:

    git clone --depth 5 --branch v5.0.2 https://github.com/Unleash/client-specification.git client-specification

    After doing this, rake spec will also run the client specification tests.

    To install this gem onto your local machine, run bundle exec rake install.

    Releasing

    Choose a new version number following Semantic Versioning semantics and then:

    Contributing

    Bug reports and pull requests are welcome on GitHub at https://github.com/unleash/unleash-client-ruby.

    Be sure to run both bundle exec rspec and bundle exec rubocop in your branch before creating a pull request.

    Please include tests with any pull requests, to avoid regressions.

    Check out our guide for more information on how to build and scale feature flag systems


    This content was generated on

    - - +To run the Ruby SDK against this test suite, you'll need to have a copy on your machine, you can clone the repository directly using:

    git clone --depth 5 --branch v5.0.2 https://github.com/Unleash/client-specification.git client-specification

    After doing this, rake spec will also run the client specification tests.

    To install this gem onto your local machine, run bundle exec rake install.

    Releasing

    Choose a new version number following Semantic Versioning semantics and then:

    Contributing

    Bug reports and pull requests are welcome on GitHub at https://github.com/unleash/unleash-client-ruby.

    Be sure to run both bundle exec rspec and bundle exec rubocop in your branch before creating a pull request.

    Please include tests with any pull requests, to avoid regressions.

    Check out our guide for more information on how to build and scale feature flag systems


    This content was generated on

    + + \ No newline at end of file diff --git a/reference/sdks/rust.html b/reference/sdks/rust.html index 7d44c44e43..c0661635fe 100644 --- a/reference/sdks/rust.html +++ b/reference/sdks/rust.html @@ -20,8 +20,8 @@ - - + + @@ -44,8 +44,8 @@ a new API token at http://localhost:4242/admin/api/create-token for user admin, type Client.

    Then run the test suite:

    UNLEASH_API_URL=http://127.0.0.1:4242/api \
    UNLEASH_APP_NAME=fred UNLEASH_INSTANCE_ID=test \
    UNLEASH_CLIENT_SECRET="<tokenvalue>" \
    cargo test --features functional -- --nocapture

    or similar. The functional test suite looks for a manually setup set of features. E.g. log into the Unleash UI on port 4242 and create a feature called -default.


    This content was generated on

    - - +default.


    This content was generated on

    + + \ No newline at end of file diff --git a/reference/sdks/svelte.html b/reference/sdks/svelte.html index 57b5a52fa6..97ecb25aeb 100644 --- a/reference/sdks/svelte.html +++ b/reference/sdks/svelte.html @@ -20,16 +20,16 @@ - - + +

    Svelte

    Generated content

    This document was generated from the README in the Svelte GitHub repository.

    tip

    To connect to Unleash from a client-side context, you'll need to use the Unleash front-end API (how do I create an API token?) or the Unleash proxy (how do I create client keys?).

    Installation

    npm install @unleash/proxy-client-svelte
    # or
    yarn add @unleash/proxy-client-svelte

    How to use

    Initialize the client

    Depending on your needs and specific use-case, prepare one of:

    And a respective frontend token (or, if you're using the Unleash Proxy, one of your proxy's designated client keys, previously known as proxy secrets).

    Import the provider like this in your entrypoint file (typically index.svelte):

    <script lang="ts">
    import { FlagProvider } from '@unleash/proxy-client-svelte';

    const config = {
    url: '<unleash-url>/api/frontend', // Your Front-end API, Unleash Edge or Unleash Proxy URL
    clientKey: '<your-token>', // Front-end API token (or proxy client key)
    refreshInterval: 15, // How often (in seconds) the client should poll the proxy for updates
    appName: 'your-app-name' // The name of your application. It's only used for identifying your application
    };
    </script>

    <FlagProvider {config}>
    <App />
    </FlagProvider>

    Connection options

    To connect this SDK to your Unleash instance's front-end API, use the URL to your Unleash instance's front-end API (<unleash-url>/api/frontend) as the url parameter. For the clientKey parameter, use a FRONTEND token generated from your Unleash instance. Refer to the how to create API tokens guide for the necessary steps.

    To connect this SDK to the Unleash Edge, use the URL to your Unleash Edge instance as the url parameter. For the clientKey parameter, use a FRONTEND token generated from your Unleash Edge instance. Refer to the how to create API tokens guide for the necessary steps. Ensure that your Unleash Edge instance is correctly configured to have access to the feature toggles your FRONTEND token is requesting.

    To connect this SDK to the Unleash proxy, use the proxy's URL and a proxy client key. The configuration section of the Unleash proxy docs contains more info on how to configure client keys for your proxy.

    Check feature toggle status

    To check if a feature is enabled:

    <script lang="ts">
    import { useFlag } from '@unleash/proxy-client-svelte';

    const enabled = useFlag('travel.landing');
    </script>

    {#if $enabled}
    <SomeComponent />
    {:else}
    <AnotherComponent />
    {/if}

    Check variants

    To check variants:

    <script lang="ts">
    import { useVariant } from '@unleash/proxy-client-svelte';

    const variant = useVariant('travel.landing');
    </script>

    {#if variant.enabled && variant.name === 'SomeComponent'}
    <SomeComponent />
    {:else if variant.enabled && variant.name === 'AnotherComponent'}
    <AnotherComponent />
    {:else}
    <DefaultComponent />
    {/if}

    Defer rendering until flags fetched

    useFlagsStatus retrieves the ready state and error events. -Follow the following steps in order to delay rendering until the flags have been fetched.

    <script lang="ts">
    import { useFlagsStatus } from '@unleash/proxy-client-svelte';

    const { flagsReady, flagsError } = useFlagsStatus();
    </script>

    {#if !$flagsReady}
    <Loading />
    {:else}
    <MyComponent error={flagsError} />
    {/if}

    Updating context

    Initial context can be specified on a FlagProvider config.context property.

    <FlagProvider config={{ ...config, context: { userId: 123 }}>

    This code sample shows you how to update the unleash context dynamically:

    <script lang="ts">
    import { useUnleashContext, useFlag } from '@unleash/proxy-client-svelte';

    export let userId;

    const toggle = useFlag('my-toggle');
    const updateContext = useUnleashContext();

    $: {
    // context is updated with userId
    updateContext({ userId });
    }

    // OR if you need to perform an action right after new context is applied
    $: {
    async function run() {
    // Can wait for the new flags to pull in from the different context
    await updateContext({ userId });
    console.log('new flags loaded for', userId);
    }
    run();
    }
    </script>

    Advanced use cases

    Deferring client start

    By default, the Unleash client will start polling for toggles immediately when the FlagProvider component renders. You can prevent it by setting startClient prop to false. This is useful when you'd like to for example bootstrap the client and work offline.

    Deferring the client start gives you more fine-grained control over when to start fetching the feature toggle configuration. This could be handy in cases where you need to get some other context data from the server before fetching toggles, for instance.

    To start the client, use the client's start method. The below snippet of pseudocode will defer polling until the end of the asyncProcess function.

    <script lang="ts">
    const client = new UnleashClient({
    /* ... */
    });

    onMount(async () => {
    // do async work ...
    client.start();
    });
    </script>

    <FlagProvider unleashClient={client} startClient={false}>
    <App />
    </FlagProvider>

    Use unleash client directly

    <script lang="ts">
    import { useUnleashContext, useUnleashClient } from '@unleash/proxy-client-svelte';

    export let userId;

    const client = useUnleashClient();

    const login = () => {
    // login user
    if (client.isEnabled('new-onboarding')) {
    // Send user to new onboarding flow
    } else {
    // send user to old onboarding flow
    }
    };
    </script>

    <LoginForm {login} />

    This content was generated on

    - - +Follow the following steps in order to delay rendering until the flags have been fetched.

    <script lang="ts">
    import { useFlagsStatus } from '@unleash/proxy-client-svelte';

    const { flagsReady, flagsError } = useFlagsStatus();
    </script>

    {#if !$flagsReady}
    <Loading />
    {:else}
    <MyComponent error={flagsError} />
    {/if}

    Updating context

    Initial context can be specified on a FlagProvider config.context property.

    <FlagProvider config={{ ...config, context: { userId: 123 }}>

    This code sample shows you how to update the unleash context dynamically:

    <script lang="ts">
    import { useUnleashContext, useFlag } from '@unleash/proxy-client-svelte';

    export let userId;

    const toggle = useFlag('my-toggle');
    const updateContext = useUnleashContext();

    $: {
    // context is updated with userId
    updateContext({ userId });
    }

    // OR if you need to perform an action right after new context is applied
    $: {
    async function run() {
    // Can wait for the new flags to pull in from the different context
    await updateContext({ userId });
    console.log('new flags loaded for', userId);
    }
    run();
    }
    </script>

    Advanced use cases

    Deferring client start

    By default, the Unleash client will start polling for toggles immediately when the FlagProvider component renders. You can prevent it by setting startClient prop to false. This is useful when you'd like to for example bootstrap the client and work offline.

    Deferring the client start gives you more fine-grained control over when to start fetching the feature toggle configuration. This could be handy in cases where you need to get some other context data from the server before fetching toggles, for instance.

    To start the client, use the client's start method. The below snippet of pseudocode will defer polling until the end of the asyncProcess function.

    <script lang="ts">
    const client = new UnleashClient({
    /* ... */
    });

    onMount(async () => {
    // do async work ...
    client.start();
    });
    </script>

    <FlagProvider unleashClient={client} startClient={false}>
    <App />
    </FlagProvider>

    Use unleash client directly

    <script lang="ts">
    import { useUnleashContext, useUnleashClient } from '@unleash/proxy-client-svelte';

    export let userId;

    const client = useUnleashClient();

    const login = () => {
    // login user
    if (client.isEnabled('new-onboarding')) {
    // Send user to new onboarding flow
    } else {
    // send user to old onboarding flow
    }
    };
    </script>

    <LoginForm {login} />

    This content was generated on

    + + \ No newline at end of file diff --git a/reference/sdks/vue.html b/reference/sdks/vue.html index 4120508a4d..ae23d7689e 100644 --- a/reference/sdks/vue.html +++ b/reference/sdks/vue.html @@ -20,16 +20,16 @@ - - + +

    Vue

    Generated content

    This document was generated from the README in the Vue GitHub repository.

    tip

    To connect to Unleash from a client-side context, you'll need to use the Unleash front-end API (how do I create an API token?) or the Unleash proxy (how do I create client keys?).

    proxy-client-vue

    PoC for a Vue SDK for Unleash based on the official proxy-client-react.

    Usage note

    This library is meant to be used with Unleash Edge, the Unleash front-end API , or the Unleash proxy.

    It is not compatible with the Unleash client API.

    Installation

    npm install @unleash/proxy-client-vue
    // or
    yarn add @unleash/proxy-client-vue

    Initialization

    Using config:

    import { createApp } from 'vue'
    import { plugin as unleashPlugin } from '@unleash/proxy-client-vue'
    // import the root component App from a single-file component.
    import App from './App.vue'

    const config = {
    url: 'https://HOSTNAME/proxy',
    clientKey: 'PROXYKEY',
    refreshInterval: 15,
    appName: 'your-app-name',
    }

    const app = createApp(App)
    app.use(unleashPlugin, { config })
    app.mount('#app')

    Or use the FlagProvider component like this in your entrypoint file (typically App.vue):

    import { FlagProvider } from '@unleash/proxy-client-vue'

    const config = {
    url: 'https://UNLEASH-INSTANCE/api/frontend',
    clientKey: 'CLIENT—SIDE—API—TOKEN',
    refreshInterval: 15,
    appName: 'your-app-name',
    }

    <template>
    <FlagProvider :config="config">
    <App />
    </FlagProvider>
    </template>

    Initializing your own client

    import { createApp } from 'vue'
    import { plugin as unleashPlugin } from '@unleash/proxy-client-vue'
    // import the root component App from a single-file component.
    import App from './App.vue'

    const config = {
    url: 'https://HOSTNAME/proxy',
    clientKey: 'PROXYKEY',
    refreshInterval: 15,
    appName: 'your-app-name',
    }

    const client = new UnleashClient(config)

    const app = createApp(App)
    app.use(unleashPlugin, { unleashClient: client })
    app.mount('#app')

    Or, using FlagProvider:

    import { FlagProvider, UnleashClient } from '@unleash/proxy-client-vue'

    const config = {
    url: 'https://UNLEASH-INSTANCE/api/frontend',
    clientKey: 'CLIENT—SIDE—API—TOKEN',
    refreshInterval: 15,
    appName: 'your-app-name',
    }

    const client = new UnleashClient(config)

    <template>
    <FlagProvider :unleash-client="client">
    <App />
    </FlagProvider>
    </template>

    Deferring client start

    By default, the Unleash client will start polling the Proxy for toggles immediately when the FlagProvider component renders. You can delay the polling by:

    • setting the startClient prop to false
    • passing a client instance to the FlagProvider
    <template>
    <FlagProvider :unleash-client="client" :start-client="false">
    <App />
    </FlagProvider>
    </template>

    Deferring the client start gives you more fine-grained control over when to start fetching the feature toggle configuration. This could be handy in cases where you need to get some other context data from the server before fetching toggles, for instance.

    To start the client, use the client's start method. The below snippet of pseudocode will defer polling until the end of the asyncProcess function.

    const client = new UnleashClient({
    /* ... */
    })

    onMounted(() => {
    const asyncProcess = async () => {
    // do async work ...
    client.start()
    }
    asyncProcess()
    })

    <template>
    <FlagProvider :unleash-client="client" :start-client="false">
    <App />
    </FlagProvider>
    </template>

    Usage

    Check feature toggle status

    To check if a feature is enabled:

    <script setup>
    import { useFlag } from '@unleash/proxy-client-vue'

    const enabled = useFlag('travel.landing')
    </script>

    <template>
    <SomeComponent v-if="enabled" />
    <AnotherComponent v-else />
    </template>

    Check variants

    To check variants:

    <script setup>
    import { useVariant } from '@unleash/proxy-client-vue'

    const variant = useVariant('travel.landing')
    </script>

    <template>
    <SomeComponent v-if="variant.enabled && variant.name === 'SomeComponent'" />
    <AnotherComponent v-else-if="variant.enabled && variant.name === 'AnotherComponent" />
    <DefaultComponent v-else />
    </template>

    Defer rendering until flags fetched

    useFlagsStatus retrieves the ready state and error events. -Follow the following steps in order to delay rendering until the flags have been fetched.

    import { useFlagsStatus } from '@unleash/proxy-client-vue'

    const { flagsReady, flagsError } = useFlagsStatus()

    <Loading v-if="!flagsReady" />
    <MyComponent v-else error={flagsError} />

    Updating context

    Follow the following steps in order to update the unleash context:

    import { useUnleashContext, useFlag } from '@unleash/proxy-client-vue'

    const props = defineProps<{
    userId: string
    }>()

    const { userId } = toRefs(props)

    const updateContext = useUnleashContext()

    onMounted(() => {
    updateContext({ userId })
    })

    watch(userId, () => {
    async function run() {
    await updateContext({ userId: userId.value })
    console.log('new flags loaded for', userId.value)
    }
    run()
    })

    This content was generated on

    - - +Follow the following steps in order to delay rendering until the flags have been fetched.

    import { useFlagsStatus } from '@unleash/proxy-client-vue'

    const { flagsReady, flagsError } = useFlagsStatus()

    <Loading v-if="!flagsReady" />
    <MyComponent v-else error={flagsError} />

    Updating context

    Follow the following steps in order to update the unleash context:

    import { useUnleashContext, useFlag } from '@unleash/proxy-client-vue'

    const props = defineProps<{
    userId: string
    }>()

    const { userId } = toRefs(props)

    const updateContext = useUnleashContext()

    onMounted(() => {
    updateContext({ userId })
    })

    watch(userId, () => {
    async function run() {
    await updateContext({ userId: userId.value })
    console.log('new flags loaded for', userId.value)
    }
    run()
    })

    This content was generated on

    + + \ No newline at end of file diff --git a/reference/segments.html b/reference/segments.html index 6683dc8030..02724e2178 100644 --- a/reference/segments.html +++ b/reference/segments.html @@ -20,8 +20,8 @@ - - + + @@ -29,7 +29,7 @@

    Segments

    Availability

    Segments are available to Unleash Pro and Unleash Enterprise users since Unleash 4.13 and was made available for Open Source from Unleash v5.5.

    A segment is a reusable collection of strategy constraints. Like with strategy constraints, you apply segments to feature flag activation strategies.

    You can apply the same segment to multiple activation strategies. If you update the segment, the changes will affect every strategy that uses that segment.

    Segments let you create user groups based on data available in the Unleash context. These groups can be as simple or as complex as you want to make them. You could, for example, use segments to target:

    • Users in a specific region
    • Users on a specific device type
    • Users who signed up before a specific point in time
    • ... Or any combination of the above.

    Because segments stay in sync across strategies, any changes will propagate to all the activation strategies that use them. This also makes them ideal for use cases such as activating or deactivating multiple feature flags at the same time. In other words, you can use segments to

    • release one or more new features at a specified time
    • create events with start and end times and guarantee that features will only be active during that period

    Segments can be global or scoped to a specific project. If a segment is scoped to a single project, it can only be used in that project and it adheres to that project's change request settings.

    Structure and evaluation

    Segments are collections of strategy constraints. To satisfy a segment, all the constraints in the collection must be satisfied.

    If an activation strategy has a segment and additional constraints applied, the segment and the strategies must all be satisfied. Similarly, if an activation strategy has multiple segments, then they must must all be satisfied.

    Segment limits

    In theory, you could create segments with a thousand constraints, each with a million values. But this wouldn't scale well, so there are limitations in place to stop you from doing this. Unleash enforces the following limits on use of segments:

    • If you're on a Pro plan

      A segment can have at most 250 values specified across all of its constraints. That means that if you add a constraint that uses 10 values, you will have 240 more values to use for any other constraints you add to the same segment.

    • If you're on an Enterprise plan

      A segment can have at most 1000 values specified across all of its constraints. That means that if you add a constraint that uses 70 values, you will have 930 more values to use for any other constraints you add to the same segment.

    By default, you can apply at most 5 segments to any one strategy. Separate strategies (even on the same feature) do not count towards the same total, so you can have two strategies with 5 segments each.

    You can configure segment limits with environment variables.

    A note on large segments

    Segments are just constraints, so any limitations that apply to constraints also apply to segments.

    This means that if you want to add a hundred different user IDs to one of your constraints, you are most likely better off thinking about finding another way to solve this problem. That may be using a different abstraction or finding another pattern that you can use instead. Refer to the section on constraint limitations for a more thorough explanation or to the topic guide on using Unleash with large constraints for a more thorough .

    Creating, updating, and deleting segments

    info

    Currently in development, Unleash will soon count segments as being in use when they are used in change requests.

    Segments can be created, edited, and deleted from the segments page in the admin UI or via the API (see the segments API documentation).

    A segment that is in use cannot be deleted. If you'd like to delete a segment that is in use, you must first remove the segment from all the activation strategies that are currently using it. If a segment is in use by an archived flag, you must either unarchive the flag and remove the segment or delete the flag entirely before the segment can be deleted.

    A segment that is in use in multiple projects can not be turned into a project-level segment. Further, a segment that is in use in project A cannot be turned into a project-level segment in project B.

    The Segments page, listing two existing segments: &quot;Mobile users&quot; and &quot;Users in the APAC region&quot;. The navigation menu with the Segments page link is opened and highlighted to provide navigation help.

    - - + + \ No newline at end of file diff --git a/reference/service-accounts.html b/reference/service-accounts.html index b742d6d1e2..4d945ed446 100644 --- a/reference/service-accounts.html +++ b/reference/service-accounts.html @@ -20,15 +20,15 @@ - - + +

    Service Accounts

    Availability

    Service accounts is an enterprise feature available from Unleash 4.21 onwards.

    Service accounts are accounts that act as normal Unleash users and that respect the same set of permissions, but that don't represent real users. These accounts do not have a password and cannot log in to the Unleash UI. Instead, they are intended to be used to access the Unleash API programmatically, providing integrations an identity.

    Service account table

    Use service accounts to:

    • Provide a user-like identity to an integration or automation and manage it within Unleash
    • Give access to the Unleash API without giving access to the Unleash UI
    • Provide more fine-grained permissions than an admin token provides

    In order to create a service account, you can follow the how to create service accounts guide.

    Service account tokens

    Service account tokens allow service accounts to use the Admin API as themselves with their own set of permissions, rather than using an admin token. See how to use the Admin API for more information.

    These tokens act just like personal access tokens for the service accounts, except that they are managed by Unleash admins.

    When using a service account token to modify resources, the event log will display the service account name for that operation.

    Service account tokens can be managed by editing the respective service account:

    Service account tokens

    - - + + \ No newline at end of file diff --git a/reference/sso.html b/reference/sso.html index d469d7a1f2..bd89b0bfec 100644 --- a/reference/sso.html +++ b/reference/sso.html @@ -20,15 +20,15 @@ - - + +

    Single Sign-On

    The Single-Sign-On capability is only available for customers on the Enterprise subscription. Check out the Unleash plans for details.

    Unleash Enterprise supports SAML 2.0, OpenID Connect and Google Authentication. In addition, Unleash supports username/password authentication out of the box.

    Before you start

    In order to configure Single-Sign-On you will need to log in to the Unleash instance with a user that have "Admin" role. If you are self-hosting Unleash then a default user will be automatically created the first time you start unleash:

    • username: admin
    • password: unleash4all (or admin if you started with Unleash v3).

    Guides

    Unleash enterprise supports multiple authentication providers.

    - - + + \ No newline at end of file diff --git a/reference/stickiness.html b/reference/stickiness.html index ee8d725abf..606f4fb02f 100644 --- a/reference/stickiness.html +++ b/reference/stickiness.html @@ -20,15 +20,15 @@ - - + +

    Stickiness

    Stickiness is how Unleash guarantees that the same user gets the same features every time. Stickiness is useful in any scenario where you want to either show a feature to only a subset of users or give users a variant of a feature.

    Calculation

    By default, Unleash calculates stickiness based on the user id and the group id. If the user id is unavailable, it falls back to using the session id instead. It hashes these values to a number between 0 and 100 using the MurmurHash hash function. This number is what determines whether a user will see a specific feature or variant. Because the process is deterministic, the same user will always get the same number.

    If both the user id and the session id is unavailable, the calculation returns a random value and stickiness is not guaranteed.

    Consistency

    Because the number assigned to a user won't change, Unleash also guarantees that the a user will keep seeing the same features even if certain other parameters change.

    For instance: When using the gradual rollout activation strategy, any user whose number is less than or equal to the rollout percentage will see the feature. This means that the same users will keep seeing the feature even as you increase the percentage of your user base that sees the feature.

    Custom stickiness

    info

    Custom stickiness is available starting from Unleash Enterprise v4.

    When using the gradual rollout strategy or feature toggle variants, you can use parameters other than the user id to calculate stickiness. More specifically, you can use any field, custom or otherwise, of the Unleash Context as long as you have enabled custom stickiness for these fields.

    SDK compatibility

    Custom stickiness is supported by all of our SDKs except for the Rust SDK. You can always refer to the SDK compatibility table for the full overview.

    Enabling custom stickiness

    To enable custom stickiness on a field, navigate to the Create Context screen in the UI and find the field you want to enable. There's a "Custom stickiness" option at the bottom of the form. Enable this toggle and update the context field by pressing the "Update" button.

    The Create Context screen in the Unleash UI. There&#39;s a toggle at the bottom to control custom stickiness.

    Project default stickiness

    Availability

    Project default stickiness was introduced in Unleash v5.

    Each project in Unleash can have its own default stickiness context field. Whenever you add a gradual rollout strategy or variants to a feature in that project, Unleash will use the configured context field as the initial value.

    Only context fields that have the custom stickiness option turned on can be used as default stickiness fields.

    If you don't specify a default custom stickiness, the project will use the "default" stickiness option, which works as described in the calculation section.

    You can configure project default stickiness when you create a project or by editing the project later.

    The Edit Project screen.  There is a dropdown for setting the default stickiness

    - - + + \ No newline at end of file diff --git a/reference/strategy-constraints.html b/reference/strategy-constraints.html index 573dad42c8..9cd10cd49c 100644 --- a/reference/strategy-constraints.html +++ b/reference/strategy-constraints.html @@ -20,8 +20,8 @@ - - + + @@ -36,7 +36,7 @@ In that case the strategy will be evaluated from DATE_AFTER and until DATE_BEFORE.

    Date and time operators only support single values.

    Nametrue if currentTime is ...
    DATE_AFTERafter the provided date
    DATE_BEFOREbefore the provided date

    String operators

    String operators differ from the other categories in two different ways:

    • all operators accept multiple values
    • most operators also consider letter case and can be set to be case-sensitive or case-insensitive
    Nametrue if <context-field> ...Supports case-insensitivityAvailable since
    INis equal to any of the provided valuesNov3.3
    NOT_INis not equal to any of the provided valuesNov3.3
    STR_CONTAINScontains any of the provided stringsYesv4.9
    STR_ENDS_WITHends with any of the provided stringsYesv4.9
    STR_STARTS_WITHstarts with any of the provided stringsYesv4.9

    Versioning (SemVer) operators

    The SemVer operators are used to compare version numbers such as application versions, dependency versions, etc.

    The SemVer input must follow a few rules:

    • The value you enter must start with and contain at least major, minor, and patch versions: Example: 1.2.3
    • Optionally, you can also add pre-release version information by adding a hyphen and series of dot separated identifiers after the patch version. Example: 1.2.3-rc.2

    Versions with pre-release indicators (e.g. 4.8.0-rc.2) are considered less than versions without (e.g. 4.8.0) in accordance with the SemVer specification, item 11.

    You can read more about SemVer in the full SemVer specification.

    SemVer operators only support single values.

    Nametrue if <context-field> is ...
    SEMVER_EQequal to the provided value
    SEMVER_GTstrictly greater than the provided value
    SEMVER_LTstrictly less than the provided value

    Additionally, you can use negation to get less than or equal to and greater than or equal to functionality:

    EffectHowtrue if <context-field> is ...
    Greater than or equal toNegate SEMVER_LTgreater than or equal to the provided value
    Less than or equal toNegate SEMVER_GTless than or equal to the provided value

    "Not less than 2.0.0" is the same as "greater than or equal to 2.0.0". The same applies for less than or equal: "Not greater than 1.9.5." is the same as "less than or equal to 1.9.5".

    Interacting with strategy constraints in the client SDKs

    note

    This section gives a brief overview over to use the client SDKs to interact with strategy constraints. The exact steps will vary depending on which client you are using, so make sure to consult the documentation for your specific client SDK.

    Strategy constraints require the Unleash Context to work. All official Unleash client SDKs support the option to pass dynamic context values to the isEnabled function (or the SDK's equivalent).

    If the strategy constraint uses a standard Unleash Context field, set the context field to the value you wish to give it.

    If the strategy constraint uses a custom context field, use the Unleash Context's properties field. Use the name of the custom context field as a key and set the value to your desired string.

    If you set a context field to a value that the SDKs cannot parse correctly for a chosen constraint operator, the strategy constraint will evaluate to false. In other words: if you have a strategy constraint operator that expects a number, such as NUM_GT, but you set the corresponding context field to a string value, then the expression will be false: "some string" is not greater than 5. This value can still be negated as explained in the section on negating values.

    Constraint limitations (or "how many user IDs can I add to a constraint")

    tip

    Explore the content in this subsection in more depth in the topic guide on using Unleash with large constraints.

    When using a constraint operator that accepts a list of values, it might be tempting to add a large number of values to that list. However, we advise you not to do that: Unleash is not a database, and is not intended to store large amounts of data. Instead you should try and find a different way to achieve what you want.

    For instance, instead of adding hundreds of user ids to the constraint value list, think about what properties those users share. Are they beta testers? Are they premium members? Are they employees?

    Can you map their common feature into an Unleash context property instead and set the constraint on that? If they're beta testers, how about using a betaTester property? And likewise, for premium members, you could check to see if their membership is premium? And if they're employees, maybe you're better off checking whether their user ID ends with @yourcompany.tld?

    The reason why you should try and keep value lists small has to do with Unleash's evaluation model: Because Unleash's server-side SDKs fetch the full feature toggle configuration from Unleash, every value that you add to that constraint value list will increase the payload size. For small numbers, this isn't an issue, but as the list grows, so will the payload, and so will the time and processing power used by the SDK to evaluate the feature.

    Incompatibilities and undefined behavior

    It's important that you use an up-to-date client SDK if you're using the advanced constraint operators introduced in Unleash 4.9. If your client SDK does not support the new operators, we cannot guarantee how it'll react. As a result, you may see different behavior across applications.

    If you use the new constraints with old SDKs, here's how it'll affect some of the SDKs (the list is not exhaustive):

    • The Node.js and Go client SDKs will ignore the new constraints completely: the constraints will not affect the toggle's status.
    • The Python client SDK will evaluate the toggle to false, as it cannot evaluate the constraint successfully.
    • The .NET, Ruby, and PHP SDKs raise exceptions if the provided operator is not IN or NOT_IN.

    Please inspect the SDK compatibility table to see which version of your preferred SDK introduced support for this feature.

    After Unleash 4.9, we updated the Unleash client specification. Going forward, any constraint that a client does not recognize, must be evaluated as false

    [Deprecated]: Constrain on a specific environment

    Before Unleash 4.3, using strategy constraints was the recommended way to have different toggle configurations per environment. Now that Unleash has environment support built in, we no longer recommend you use strategy constraints for this. Instead, see the environments documentation.


    1. userScore is not a default Unleash field, but can be added as a custom context field.
    - - + + \ No newline at end of file diff --git a/reference/strategy-variants.html b/reference/strategy-variants.html index 3d7542f7ef..70204d4471 100644 --- a/reference/strategy-variants.html +++ b/reference/strategy-variants.html @@ -20,8 +20,8 @@ - - + + @@ -37,7 +37,7 @@ broad activation strategy with multiple percentage based variants.

    In the example below we configure fixed title for the internal users based on the clientId constraint. In the second strategy we split titles between all other users based on the 50%/50% split. strategy_variants example

    Client SDK Support

    To make use of strategy variants, you need to use a compatible client. Client SDK with variant support:

    If you would like to give feedback on this feature, experience issues or have questions, please feel free to open an issue on GitHub.

    - - + + \ No newline at end of file diff --git a/reference/tags.html b/reference/tags.html index 341c286c8b..ace9f00812 100644 --- a/reference/tags.html +++ b/reference/tags.html @@ -20,15 +20,15 @@ - - + +

    Tags

    This feature was introduced in Unleash 3.11.0

    Do you want to filter your features to avoid having to see all features belonging to other teams than your own? Do you want to write a plugin that only gets notified about changes to features that your plugin knows how to handle?

    Say hello to Typed tags

    Unleash supports tagging features with an arbitrary number of tags. This eases filtering the list of tags to only those features that are tagged with the tag you're interested in.

    How does it work?

    Unleash will allow users to tag any feature with any number of tags. When viewing a feature, the UI will/may display all tags connected to that feature.

    When adding a new tag, a dropdown will show you which type of tag you're about to add. Our first type; simple are meant to be used for filtering features. Show only features that have a tag of MyTeam.

    Tag types

    Types can be anything, and their purpose is to add some semantics to the tag itself.

    Some tag types will be defined by plugins (e.g. the slack plugin can define the Slack-type, to make it easy to specify which Slack channels to post updates for a specific feature toggle to).

    Other tags can be defined by the user to give semantic logic to the management of the UI. It could be that you want to use tag functionality to specify which products a feature toggle belongs to, or to which teams.

    - - + + \ No newline at end of file diff --git a/reference/technical-debt.html b/reference/technical-debt.html index 4ed59d2bd0..4e01e9b62f 100644 --- a/reference/technical-debt.html +++ b/reference/technical-debt.html @@ -20,15 +20,15 @@ - - + +

    Technical Debt

    At Unleash we care deeply about code quality. Technical debt creeps up over time and slowly builds to the point where it really starts to hurt. At that point it's too late. Feature toggles that have outlived their feature and are not cleaned up represent technical debt that you should remove from your code.

    Stale and potentially stale toggles

    When a toggle is no longer useful, we say that it has become stale. A stale toggle is a toggle that has served its purpose and that you should remove from the code base. For a toggle to become stale, you have to explicitly mark it as such. You can mark a toggle as stale in the technical debt dashboard.

    Unleash also has a concept of potentially stale toggles. These are toggles that have lived longer than what Unleash expects them to based on their feature toggle type. However, Unleash can't know for sure whether a toggle is actually stale or not, so it's up to you to make the decision on whether to mark it as stale or to keep it as an active toggle.

    A toggle being (potentially) stale, does not affect how it performs in your application; it's only there to make it easier for you to manage your toggles.

    The technical debt dashboard

    In order to assist with removing unused feature toggles, Unleash provides project health dashboards for each project. The health dashboard is listed as one of a project's tabs.

    Three UI elements describing the health rating of the project. The first card has info on the project, including its name. The second is the &quot;report card&quot;, containing the project&#39;s overall health rating, a toggle report, and potential actions. The last card is a list of all the project&#39;s toggles with data on when it was last seen, when it was created, when it expired, its status and a report.

    The dashboard includes a health report card, and a list of toggles that can be filtered on different parameters.

    Report card

    The project&#39;s health report card. It lists the project&#39;s health rating and when it was last updated; a toggle report containing the number of active toggles in the project; and potential actions, in this case asking the user to review potentially stale toggles.

    The report card includes some statistics of your application. It lists the overall amount of your active toggles, the overall amount of stale toggles, and lastly, the toggles that Unleash believes should be stale. This calculation is performed on the basis of toggle types:

    • Release - Used to enable trunk-based development for teams practicing Continuous Delivery. Expected lifetime 40 days
    • Experiment - Used to perform multivariate or A/B testing. Expected lifetime 40 days
    • Operational - Used to control operational aspects of the system's behavior. Expected lifetime 7 days
    • Kill switch - Used to to gracefully degrade system functionality. (permanent)
    • Permission - Used to change the features or product experience that certain users receive. (permanent)

    If your toggle exceeds the expected lifetime of it's toggle type it will be marked as potentially stale.

    One thing to note is that the report card and corresponding list are showing stats related to the currently selected project. If you have more than one project, you will be provided with a project selector in order to swap between the projects.

    Health rating

    Unleash calculates a project's health rating based on the project's total number of active toggles and how many of those active toggles are stale or potentially stale. When you archive a toggle, it no longer counts towards your project's health rating.

    The health rating updates once every hour, so there may be some lag if you have recently added, removed, or changed the status of a toggle.

    Toggle list

    A table of the toggles in the current project with their health reports. The table has the following columns: name, last seen, created, expired, status, and report.

    The toggle list gives an overview over all of your toggles and their status. In this list you can sort the toggles by their name, last seen, created, expired, status and report. This will allow you to quickly get an overview over which toggles may be worth deprecating and removing from the code.

    - - + + \ No newline at end of file diff --git a/reference/unleash-context.html b/reference/unleash-context.html index 6d7179fa94..006da748be 100644 --- a/reference/unleash-context.html +++ b/reference/unleash-context.html @@ -20,8 +20,8 @@ - - + + @@ -29,7 +29,7 @@

    Unleash Context

    The Unleash Context contains information relating to the current feature toggle request. Unleash uses this context to evaluate activation strategies and strategy constraints and to calculate toggle stickiness. The Unleash Context is an important feature of all the Unleash client SDKs.

    Structure

    You can group the Unleash Context fields into two separate groups based on how they work in the client SDKs: static and dynamic context fields.

    Static fields' values remain constant throughout an application's lifetime. You'll typically set these when you initialize the client SDK.

    Dynamic fields, however, can change with every request. You'll typically provide these when checking whether a toggle is enabled in your client.

    All fields are optional, but some strategies depend on certain fields being present. For instance, the UserIDs strategy requires that the userId field is present on the Context.

    The below table gives a brief overview over what the fields' intended usage is, their lifetime, and their type. Note that the exact type can vary between programming languages and implementations. Be sure to consult your specific client SDK for more information on its implementation of the Unleash Context.

    field nametypelifetimedescription
    appNamestringstaticthe name of the application
    environment1stringstaticthe environment the app is running in
    userIdstringdynamican identifier for the current user
    sessionIdstringdynamican identifier for the current session
    remoteAddressstringdynamicthe app's IP address
    propertiesMap<string, string>dynamica key-value store of any data you want
    currentTime2DateTime/stringdynamicA DateTime (or similar) data class instance or a string in an RFC3339-compatible format. Defaults to the current time if not set by the user.

    The properties field

    The properties field is different from the others. You can use the properties field to provide arbitrary data to custom strategies or to strategy constraints. The properties field is also where you add values for custom context fields.

    A note on properties and constraints

    Some SDK implementations of the Unleash Context allow for the values in the properties map to be of other types than a string type. Using non-string types as values may cause issues if you're using the property in a constraint. Because the Unleash Admin UI accepts any string as input for constraint checking, the SDKs must also assume that the value is a string.

    As an example: You've created a custom field called groupId. You know group IDs will always be numeric. You then create a constraint on a strategy that says the user must be in group 123456. If you were to set the property groupId to the number 123456 in the properties field on the SDK side, the constraint check would fail, because in most languages the number 123456 is not equal to the string 123456 (i.e. 123456 != "123456").

    Custom context fields

    Availability

    Before Unleash 4.16, custom context fields were only available to Unleash Pro and Enterprise users. From 4.16 onwards, they're available to everyone. They were introduced in Unleash 3.2.28.

    Custom context fields allow you to extend the Unleash Context with more data that is applicable to your situation. Each context field definition consists of a name and an optional description. Additionally, you can choose to define a set of legal values, and you can choose whether or not the context field can be used in custom stickiness calculations for the gradual rollout strategy and for feature toggle variants.

    When interacting with custom context fields in code, they must be accessed via the Unleash Context's properties map, using the context field's name as the key.

    Creating and updating custom context fields

    You can create as many custom context fields as you wish. Refer to "how to define custom context fields" for information on how you define your own custom context fields.

    You can update custom context fields after they have been created. You can change everything about the definition except for the name.

    By using the legal values option when creating a context field, you can create a set of valid options for a context field's values. If a context field has a defined set of legal values, the Unleash Admin UI will only allow users to enter one or more of the specified values. If a context field doesn't have any defined legal values, the user can enter whatever they want.

    Using a custom context field called region as an example: if you define the field's legal values as Africa, Asia, Europe, and North America, then you would only be allowed to use one or more of those four values when using the custom context field as a strategy constraint.

    A strategy constraint form with a constraint set to &quot;region&quot;. The &quot;values&quot; input is a dropdown menu containing the options &quot;Africa&quot;, &quot;Asia&quot;, &quot;Europe&quot;, and &quot;North America&quot;, as defined in the preceding paragraph.

    Custom stickiness

    SDK compatibility

    Custom stickiness is supported by all of our SDKs except for the Rust SDK. You can always refer to the SDK compatibility table for the full overview.

    Any context field can be used to calculate custom stickiness. However, you need to explicitly tell Unleash that you want a field to be used for custom stickiness for it to be possible. You can enable this functionality either when you create the context field or at any later point. For steps on how to do this, see the How to define custom context fields guide.


    1. Check the strategy constraints: advanced support row of the compatibility table for an overview of which SDKs provide the currentTime property.
    2. If you're on Unleash 4.3 or higher, you'll probably want to use the environments feature instead of relying on the environment context field when working with environments.
    - - + + \ No newline at end of file diff --git a/reference/unleash-edge.html b/reference/unleash-edge.html index fe89688d88..24f8874875 100644 --- a/reference/unleash-edge.html +++ b/reference/unleash-edge.html @@ -20,8 +20,8 @@ - - + + @@ -42,8 +42,8 @@ However, tokens in <project>:<environment>.<secret> format will still filter by project.

    Metrics

    ❗ Note: For Unleash to correctly register SDK usage metrics sent from Edge instances, your Unleash instance must be v4.22 or newer.

    Since Edge is designed to avoid overloading its upstream, Edge gathers and accumulates usage metrics from SDKs for a set interval (METRICS_INTERVAL_SECONDS) before posting a batch upstream. This reduces load on Unleash instances down to a single call every interval, instead of every single client posting to Unleash for updating metrics. Unleash instances running on versions older than 4.22 are not able to handle the batch format posted by Edge, which means you won't see any metrics from clients connected to an Edge instance until you're able to update to 4.22 or newer.

    Compatibility

    Unleash Edge adheres to Semantic Versioning (SemVer) on the API and CLI layers. If you're using Unleash Edge as a library in your projects, be cautious, internal codebase changes, which might occur in any version release (including minor and patch versions), could potentially break your implementation.

    Performance

    Unleash Edge will scale linearly with CPU. There are k6 benchmarks in the benchmark folder. We've already got some initial numbers from hey.

    Do note that the number of requests Edge can handle does depend on the total size of your toggle response. That is, Edge is faster if you only have 10 toggles with 1 strategy each, than it will be with 1000 toggles with multiple strategies on each. Benchmarks here were run with data fetched from the Unleash demo instance (roughly 100kB (350 features / 200 strategies)) as well as against a small dataset of 5 features with one strategy on each.

    Edge was started using -docker run --cpus="<cpu>" --memory=128M -p 3063:3063 -e UPSTREAM_URL=<upstream> -e TOKENS="<client token>" unleashorg/unleash-edge:edge -w <number of cpus rounded up to closest integer> edge

    Then we run hey against the proxy endpoint, evaluating toggles

    Large Dataset (350 features (100kB))

    $ hey -z 10s -H "Authorization: <frontend token>" http://localhost:3063/api/frontend`
    CPUMemoryRPSEndpointp95Data transferred
    0.16.7 Mi600/api/frontend103ms76Mi
    16.7 Mi6900/api/frontend7.4ms866Mi
    49.525300/api/frontend2.4ms3.2Gi
    81540921/api/frontend1.6ms5.26Gi

    and against our client features endpoint.

    $ hey -z 10s -H "Authorization: <client token>" http://localhost:3063/api/client/features
    CPUMemory observedRPSEndpointp95Data transferred
    0.111 Mi309/api/client/features199ms300 Mi
    111 Mi3236/api/client/features16ms3 Gi
    411 Mi12815/api/client/features4.5ms14 Gi
    817 Mi23207/api/client/features2.7ms26 Gi

    Small Dataset (5 features (2kB))

    $ hey -z 10s -H "Authorization: <frontend token>" http://localhost:3063/api/frontend`
    CPUMemoryRPSEndpointp95Data transferred
    0.14.3 Mi3673/api/frontend93ms9Mi
    16.7 Mi39000/api/frontend1.6ms80Mi
    46.9 Mi100020/api/frontend600μs252Mi
    812.5 Mi141090/api/frontend600μs324Mi

    and against our client features endpoint.

    $ hey -z 10s -H "Authorization: <client token>" http://localhost:3063/api/client/features
    CPUMemory observedRPSEndpointp95Data transferred
    0.14 Mi3298/api/client/features92ms64 Mi
    14 Mi32360/api/client/features2ms527Mi
    411 Mi95838/api/client/features600μs2.13 Gi
    817 Mi129381/api/client/features490μs2.87 Gi

    Why choose Unleash Edge over the Unleash Proxy?

    Edge offers a superset of the same feature set as the Unleash Proxy and we've made sure it offers the same security and privacy features.

    However, there are a few notable differences between the Unleash Proxy and Unleash Edge:

    • Unleash Edge is built to be light and fast, it handles an order of magnitude more requests per second than the Unleash Proxy can, while using two orders of magnitude less memory.
    • All your Unleash environments can be handled by a single instance, no more running multiple instances of the Unleash Proxy to handle both your development and production environments.
    • Backend SDKs can connect to Unleash Edge without turning on experimental feature flags.
    • Unleash Edge is smart enough to dynamically resolve the tokens you use to connect to it against the upstream Unleash instance. This means you don't have to worry about knowing in advance what tokens your SDKs use - if you want to swap out the Unleash token your SDK uses, this can be done without ever restarting or worrying about Unleash Edge. Unleash Edge will only collect and cache data for the environments and projects you use.

    Debugging

    You can adjust the RUST_LOG environment variable to see more verbose log output. For example, in order to get logs originating directly from Edge but not its dependencies, you can raise the default log level from error to warning and set Edge to debug, like this:

    RUST_LOG="warn,unleash-edge=debug" ./unleash-edge #<command>

    See more about available logging and log levels at https://docs.rs/env_logger/latest/env_logger/#enabling-logging

    Development

    See our Contributors guide as well as our development-guide


    This content was generated on

    - - +docker run --cpus="<cpu>" --memory=128M -p 3063:3063 -e UPSTREAM_URL=<upstream> -e TOKENS="<client token>" unleashorg/unleash-edge:edge -w <number of cpus rounded up to closest integer> edge

    Then we run hey against the proxy endpoint, evaluating toggles

    Large Dataset (350 features (100kB))

    $ hey -z 10s -H "Authorization: <frontend token>" http://localhost:3063/api/frontend`
    CPUMemoryRPSEndpointp95Data transferred
    0.16.7 Mi600/api/frontend103ms76Mi
    16.7 Mi6900/api/frontend7.4ms866Mi
    49.525300/api/frontend2.4ms3.2Gi
    81540921/api/frontend1.6ms5.26Gi

    and against our client features endpoint.

    $ hey -z 10s -H "Authorization: <client token>" http://localhost:3063/api/client/features
    CPUMemory observedRPSEndpointp95Data transferred
    0.111 Mi309/api/client/features199ms300 Mi
    111 Mi3236/api/client/features16ms3 Gi
    411 Mi12815/api/client/features4.5ms14 Gi
    817 Mi23207/api/client/features2.7ms26 Gi

    Small Dataset (5 features (2kB))

    $ hey -z 10s -H "Authorization: <frontend token>" http://localhost:3063/api/frontend`
    CPUMemoryRPSEndpointp95Data transferred
    0.14.3 Mi3673/api/frontend93ms9Mi
    16.7 Mi39000/api/frontend1.6ms80Mi
    46.9 Mi100020/api/frontend600μs252Mi
    812.5 Mi141090/api/frontend600μs324Mi

    and against our client features endpoint.

    $ hey -z 10s -H "Authorization: <client token>" http://localhost:3063/api/client/features
    CPUMemory observedRPSEndpointp95Data transferred
    0.14 Mi3298/api/client/features92ms64 Mi
    14 Mi32360/api/client/features2ms527Mi
    411 Mi95838/api/client/features600μs2.13 Gi
    817 Mi129381/api/client/features490μs2.87 Gi

    Why choose Unleash Edge over the Unleash Proxy?

    Edge offers a superset of the same feature set as the Unleash Proxy and we've made sure it offers the same security and privacy features.

    However, there are a few notable differences between the Unleash Proxy and Unleash Edge:

    • Unleash Edge is built to be light and fast, it handles an order of magnitude more requests per second than the Unleash Proxy can, while using two orders of magnitude less memory.
    • All your Unleash environments can be handled by a single instance, no more running multiple instances of the Unleash Proxy to handle both your development and production environments.
    • Backend SDKs can connect to Unleash Edge without turning on experimental feature flags.
    • Unleash Edge is smart enough to dynamically resolve the tokens you use to connect to it against the upstream Unleash instance. This means you don't have to worry about knowing in advance what tokens your SDKs use - if you want to swap out the Unleash token your SDK uses, this can be done without ever restarting or worrying about Unleash Edge. Unleash Edge will only collect and cache data for the environments and projects you use.

    Debugging

    You can adjust the RUST_LOG environment variable to see more verbose log output. For example, in order to get logs originating directly from Edge but not its dependencies, you can raise the default log level from error to warning and set Edge to debug, like this:

    RUST_LOG="warn,unleash-edge=debug" ./unleash-edge #<command>

    See more about available logging and log levels at https://docs.rs/env_logger/latest/env_logger/#enabling-logging

    Development

    See our Contributors guide as well as our development-guide


    This content was generated on

    + + \ No newline at end of file diff --git a/reference/unleash-proxy.html b/reference/unleash-proxy.html index c3c20def48..87b12a2b5b 100644 --- a/reference/unleash-proxy.html +++ b/reference/unleash-proxy.html @@ -20,15 +20,15 @@ - - + +
    -

    Unleash Proxy

    Generated content

    This document was generated from the README in the Unleash Proxy GitHub repository.

    tip

    Looking for how to run the Unleash proxy? Check out the how to run the Unleash proxy guide!

    Build & Tests npm Docker Pulls

    The Unleash Proxy

    The Unleash proxy offers a way to use Unleash in client-side applications, such as single-page and native apps.

    Note: You might want to consider using Unleash Edge instead.

    The Unleash proxy sits between the Unleash API and your client-side SDK and does the evaluation of feature toggles for your client-side SDK. This way, you can keep your configuration private and secure, while still allowing your client-side apps to use Unleash's features.

    The proxy offers three important features:

    • Performance: The caches all features in memory and can run close to your end-users. A single instance can able to handle thousands of requests per second, and you can easily scale it by adding additional instances.
    • Security: The proxy evaluates the features for the user on the server-side and by default only exposes results for features that are enabled for the specific user. No feature toggle configuration is ever shared with the user.
    • Privacy: If you run the proxy yourself, Unleash will never get any data on your end-users: no user ids, no IPs, no nothing.
    Client-side apps connect to the Unleash proxy, which in turn connects to the Unleash API. The proxy itself uses the Unleash Node.js SDK to evaluate features. The SDK syncs with Unleash in the background. Local evaluation on the proxy provides privacy.

    A note on privacy and the proxy

    Why would you not want to expose your Unleash configuration to your end-users?

    The way Unleash works, you can add all kinds of data to feature strategies and constraints. For instance, you might show a feature only to a specific subset of user IDs; or you might have a brand new and unannounced new feature with a revealing name.

    If you just sent the regular Unleash client payload to your client-side apps, all of this — the user IDs and the new feature names — would be exposed to your users.

    Single page apps work in the context of a specific user. The proxy allows you to only provide data that relates to that one user: The proxy will default to only returning the evaluated toggles that should be enabled for that specific user in that specific context.

    API

    The Unleash proxy exposes a simple API for consumption by client-side SDKs.

    OpenAPI integration


    ℹ️ Availability

    The OpenAPI integration is available in versions 0.9 and later of the Unleash proxy.


    The proxy can expose a runtime-generated OpenAPI JSON spec and a corresponding OpenAPI UI for its API. The OpenAPI UI page is an interactive page where you can discover and test the API endpoints the proxy exposes. The JSON spec can be used to generate an OpenAPI client with OpenAPI tooling such as the OpenAPI generator.

    To enable the JSON spec and UI, set ENABLE_OAS (environment variable) or enableOAS (in-code configuration variable) to true.

    The spec and UI can then be found at <base url>/docs/openapi.json and <base url>/docs/openapi respectively.

    You can refer to the how to enable the OpenAPI spec guide for more detailed information on how to configure it.

    GET /proxy

    The primary proxy API operation. This endpoint accepts an Unleash context encoded as query parameters, and will return all toggles that are evaluated as true for the provided context.

    When sending GET requests to the Unleash proxy's /proxy endpoint, the request should contain the current Unleash context as query parameters. The proxy will return all enabled toggles for the provided context.

    Payload

    The GET /proxy operation returns information about toggles enabled for the current user. The payload is a JSON object with a toggles property, which contains a list of toggles.

    {
    "toggles": [
    {
    "name": "demo",
    "enabled": true,
    "variant": {
    "name": "disabled",
    "enabled": false
    }
    },
    {
    "name": "demoApp.step1",
    "enabled": true,
    "variant": {
    "name": "disabled",
    "enabled": false
    }
    }
    ]
    }

    Toggle data

    The data for a toggle without variants looks like this:

    {
    "name": "basic-toggle",
    "enabled": true,
    "variant": {
    "name": "disabled",
    "enabled": false
    }
    }
    • name: the name of the feature.
    • enabled: whether the toggle is enabled or not. Will always be true.
    • variant: describes whether the toggle has variants and, if it does, what variant is active for this user. If a toggle doesn't have any variants, it will always be {"name": "disabled", "enabled": false}.

    ℹ️ The "disabled" variant

    Unleash uses a fallback variant called "disabled" to indicate that a toggle has no variants. However, you are free to create a variant called "disabled" yourself. In that case you can tell them apart by checking the variant's enabled property: if the toggle has no variants, enabled will be false. If the toggle is the "disabled" variant that you created, it will have enabled set to true.


    If a toggle has variants, then the variant object can also contain an optional payload property. The payload will contain data about the variant's payload: what type it is, and what the content is. To learn more about variants and their payloads, check the feature toggle variants documentation.

    Variant toggles without payloads look will have their name listed and the enabled property set to true:

    {
    "name": "toggle-with-variants",
    "enabled": true,
    "variant": {
    "name": "simple",
    "enabled": true
    }
    }

    If the variant has a payload, the optional payload property will list the payload's type and it's content in a stringified form:

    {
    "name": "toggle-with-variants",
    "enabled": true,
    "variant": {
    "name": "with-payload-string",
    "payload": {
    "type": "string",
    "value": "this string is the variant's payload"
    },
    "enabled": true
    }
    }

    For the variant property:

    • name: is the name of the variant, as shown in the Admin UI.
    • enabled: indicates whether the variant is enabled or not. If the toggle has variants, this is always true.
    • payload (optional): Only present if the variant has a payload. Describes the payload's type and content.

    If the variant has a payload, the payload object contains:

    • type: the type of the variant's payload
    • value: the value of the variant's payload

    The value will always be the payload's content as a string, escaped as necessary. For instance, a variant with a JSON payload would look like this:

    {
    "name": "toggle-with-variants",
    "enabled": true,
    "variant": {
    "name": "with-payload-json",
    "payload": {
    "type": "json",
    "value": "{\"description\": \"this is data delivered as a json string\"}"
    },
    "enabled": true
    }
    }

    POST /proxy

    The proxy also offers a POST API used to evaluate toggles This can be used to evaluate a list of know toggle names or to retrieve all enabled toggles for a given context.

    Evaluate list of known toggles

    This method will allow you to send a list of toggle names together with an Unleash Context and evaluate them accordingly. It will return enablement of all provided toggles.

    URL: POST https://proxy-host.server/proxy

    Content Type: application/json

    Body:

    {
    "toggles": ["demoApp.step1"],
    "context": {
    "appName": "someApp",
    "sessionId": "233312AFF22"
    }
    }

    Result:

    HTTP/1.1 200 OK
    Access-Control-Allow-Origin: *
    Access-Control-Expose-Headers: ETag
    Cache-control: public, max-age=2
    Connection: keep-alive
    Content-Length: 122
    Content-Type: application/json; charset=utf-8
    Date: Wed, 30 Nov 2022 14:46:48 GMT
    ETag: W/"7a-RMKUyY0BWIhjahpVPWnNdXyDw6I"
    Keep-Alive: timeout=5
    Vary: Accept-Encoding

    {
    "toggles": [
    {
    "enabled": false,
    "impressionData": true,
    "name": "demoApp.step1",
    "variant": {
    "enabled": false,
    "name": "disabled"
    }
    }
    ]
    }

    Evaluate all enabled toggles

    This method will allow you to get all enabled toggles for a given context.

    URL: POST https://proxy-host.server/proxy

    Content Type: application/json

    Body:

    {
    "context": {
    "appName": "someApp",
    "sessionId": "233312AFF22"
    }
    }

    Result:

    HTTP/1.1 200 OK
    Access-Control-Allow-Origin: *
    Access-Control-Expose-Headers: ETag
    Cache-control: public, max-age=2
    Connection: keep-alive
    Content-Length: 465
    Content-Type: application/json; charset=utf-8
    Date: Wed, 30 Nov 2022 14:48:55 GMT
    ETag: W/"1d1-dm6tkvMpkx42mZmojNSNKmHid1M"
    Keep-Alive: timeout=5
    Vary: Accept-Encoding

    {
    "toggles": [
    {
    "enabled": true,
    "impressionData": false,
    "name": "demoApp.step2",
    "variant": {
    "enabled": true,
    "name": "userWithId",
    "payload": {
    "type": "string",
    "value": "90732934"
    }
    }
    },
    {
    "enabled": true,
    "impressionData": false,
    "name": "demoApp.step3",
    "variant": {
    "enabled": true,
    "name": "C",
    "payload": {
    "type": "string",
    "value": "hello"
    }
    }
    },
    {
    "enabled": true,
    "impressionData": true,
    "name": "demoApp.step4",
    "variant": {
    "enabled": true,
    "name": "Orange",
    "payload": {
    "type": "string",
    "value": "orange"
    }
    }
    }
    ]
    }


    GET /proxy/all Return enabled and disabled toggles:

    By default, the proxy only returns enabled toggles. However, in certain use cases, you might want it to return all toggles, regardless of whether they're enabled or disabled. The /proxy/all endpoint does this.

    Because returning all toggles regardless of their state is a potential security vulnerability, the endpoint has to be explicitly enabled. To enable it, set the enableAllEndpoint configuration option or the ENABLE_ALL_ENDPOINT environment variable to true.

    The response payload follows the same format as the GET /proxy response payload.

    GET /proxy/health: Health endpoint

    The proxy will try to synchronize with the Unleash API at startup, until it has successfully done that the proxy will return HTTP 503 - Not Ready for all request. You can use the health endpoint to validate that the proxy is ready to receive requests:

    curl http://localhost:3000/proxy/health -I

    If the proxy is ready, the response should look a little something like this:

    HTTP/1.1 200 OK
    Access-Control-Allow-Origin: *
    Access-Control-Expose-Headers: ETag
    Content-Type: text/html; charset=utf-8
    Content-Length: 2
    ETag: W/"2-eoX0dku9ba8cNUXvu/DyeabcC+s"
    Date: Fri, 04 Jun 2021 10:38:27 GMT
    Connection: keep-alive
    Keep-Alive: timeout=5

    Configuration options

    The Proxy has a large number of configuration options that you can use to adjust it to your specific use case. The following table lists all the available options.


    ℹ️ Required configuration

    You must configure these three variables for the proxy to start successfully:

    • unleashUrl / UNLEASH_URL

    • unleashApiToken / UNLEASH_API_TOKEN

    • clientKeys / UNLEASH_PROXY_CLIENT_KEYS


    OptionEnvironment VariableDefault valueRequiredDescription
    unleashUrlUNLEASH_URLn/ayesAPI Url to the Unleash instance to connect to
    unleashApiTokenUNLEASH_API_TOKENn/ayesAPI token (client) needed to connect to Unleash API.
    clientKeysUNLEASH_PROXY_CLIENT_KEYSn/ayesList of client keys that the proxy should accept. When querying the proxy, Proxy SDKs must set the request's client keys header to one of these values. The default client keys header is Authorization.
    proxySecretsUNLEASH_PROXY_SECRETSn/anoDeprecated alias for clientKeys. Please use clientKeys instead.
    n/aPORT or PROXY_PORT3000noThe port where the proxy should listen.
    proxyBasePathPROXY_BASE_PATH""noThe base path to run the proxy from. "/proxy" will be added at the end. For instance, if proxyBasePath is "base/path", the proxy will run at /base/path/proxy.
    unleashAppNameUNLEASH_APP_NAME"unleash-proxy"noApp name to used when registering with Unleash
    unleashInstanceIdUNLEASH_INSTANCE_IDgeneratednoUnleash instance id to used when registering with Unleash
    refreshIntervalUNLEASH_FETCH_INTERVAL5000noHow often the proxy should query Unleash for updates, defined in ms.
    metricsIntervalUNLEASH_METRICS_INTERVAL30000noHow often the proxy should send usage metrics back to Unleash, defined in ms.
    metricsJitterUNLEASH_METRICS_JITTER0noAdds jitter to the metrics interval to avoid multiple instances sending metrics at the same time, defined in ms.
    environmentUNLEASH_ENVIRONMENTundefinednoIf set this will be the environment used by the proxy in the Unleash Context. It will not be possible for proxy SDKs to override the environment if set.
    projectNameUNLEASH_PROJECT_NAMEundefinednoThe projectName (id) to fetch feature toggles for. The proxy will only return know about feature toggles that belongs to the project, if specified.
    loggern/aSimpleLoggernoRegister a custom logger.
    useJsonLoggerJSON_LOGGERfalsenoSets the default logger to log as one-line JSON. This has no effect if a custom logger is provided.
    logLevelLOG_LEVEL "warn"noUsed to set logLevel. Supported options: "debug", "info", "warn", "error" and "fatal"
    customStrategiesUNLEASH_CUSTOM_STRATEGIES_FILE[]noUse this option to inject implementation of custom activation strategies. If you are using UNLEASH_CUSTOM_STRATEGIES_FILE you need to provide a valid path to a javascript file which exports an array of custom activation strategies and the SDK will automatically load these
    trustProxyTRUST_PROXY falsenoBy enabling the trustProxy option, Unleash Proxy will have knowledge that it's sitting behind a proxy and that the X-Forwarded-* header fields may be trusted, which otherwise may be easily spoofed. The proxy will automatically enrich the ip address in the Unleash Context. Can either be true/false (Trust all proxies), trust only given IP/CIDR (e.g. '127.0.0.1') as a string. May be a list of comma separated values (e.g. '127.0.0.1,192.168.1.1/24'
    namePrefixUNLEASH_NAME_PREFIXundefinednoUsed to filter features by using prefix when requesting backend values.
    tagsUNLEASH_TAGSundefinednoUsed to filter features by using tags set for features. Format should be tagName:tagValue,tagName2:tagValue2
    clientKeysHeaderNameCLIENT_KEY_HEADER_NAME"authorization"noThe name of the HTTP header to use for client keys. Incoming requests must set the value of this header to one of the Proxy's clientKeys to be authorized successfully.
    storageProvidern/aundefinednoRegister a custom storage provider. Refer to the section on custom storage providers in the Node.js SDK's readme for more information.
    enableOASENABLE_OASfalsenoSet to true to expose the proxy's OpenAPI spec at /docs/openapi.json and an interactive OpenAPI UI at /docs/openapi. Read more in the OpenAPI section.
    enableAllEndpointENABLE_ALL_ENDPOINTfalsenoSet to true to expose the /proxy/all endpoint. Refer to the section on returning all toggles for more info.
    corsn/an/anoPass custom options for CORS module
    cors.allowedHeadersCORS_ALLOWED_HEADERSn/anoHeaders to allow for CORS
    cors.credentialsCORS_CREDENTIALSfalsenoAllow credentials in CORS requests
    cors.exposedHeadersCORS_EXPOSED_HEADERS"ETag"noHeaders to expose for CORS
    cors.maxAgeCORS_MAX_AGE172800noMaximum number of seconds to cache CORS results
    cors.methodsCORS_METHODS"GET, POST"noMethods to allow for CORS
    cors.optionsSuccessStatusCORS_OPTIONS_SUCCESS_STATUS204noIf the Options call passes CORS, which http status to return
    cors.originCORS_ORIGIN*noOrigin URL or list of comma separated list of URLs to whitelist for CORS
    cors.preflightContinueCORS_PREFLIGHT_CONTINUEfalseno
    httpOptions.rejectUnauthorizedHTTP_OPTIONS_REJECT_UNAUTHORIZEDtruenoIf true, unleash-proxy will automatically reject connections to unleash server with invalid certificates

    Experimental configuration options

    Some functionality is under validation and introduced as experimental. This allows us to test new functionality early and quickly. Experimental options can change or be removed at any point in the future and are explicitly not part of the proxy's semantic versioning.

    OptionEnvironment VariableDefault valueRequiredDescription
    expBootstrapn/an/anoWhere the Proxy can bootstrap configuration from. See Node.js SDK for details.
    expBootstrap.urlEXP_BOOTSTRAP_URLn/anoUrl where the Proxy can bootstrap configuration from. See Node.js SDK for details.
    expBootstrap.urlHeaders.AuthorizationEXP_BOOTSTRAP_AUTHORIZATIONn/anoAuthorization header value to be used when bootstrapping
    expServerSideSdkConfig.tokensEXP_SERVER_SIDE_SDK_CONFIG_TOKENSn/anoAPI tokens that can be used by Server SDKs (and proxies) to read feature toggle configuration from this Proxy instance.
    OptionEnvironment VariableDefault valueRequiredDescription
    expBootstrapn/an/anoWhere the Proxy can bootstrap configuration from. See Node.js SDK for details.
    expBootstrap.urlEXP_BOOTSTRAP_URLn/anoUrl where the Proxy can bootstrap configuration from. See Node.js SDK for details.
    expBootstrap.urlHeaders.AuthorizationEXP_BOOTSTRAP_AUTHORIZATIONn/anoAuthorization header value to be used when bootstrapping
    expServerSideSdkConfig.tokensEXP_SERVER_SIDE_SDK_CONFIG_TOKENSn/anoAPI tokens that can be used by Server SDKs (and proxies) to read feature toggle configuration from this Proxy instance.
    expCustomEnrichersEXP_CUSTOM_ENRICHERS_FILE[]noUse this option to inject implementation of custom context enrichers. If you are using EXP_CUSTOM_ENRICHERS_FILE you need to provide a valid path to a javascript file which exports an array of custom context enrichers and the SDK will automatically load these

    Internal SDK capabilities

    The proxy uses the Node.js SDK internally. When you start the proxy it initializes a client for you using any configuration option you provided on startup. You can also provide your own client.

    The client that the proxy uses supports all the SDK features that the Node.js SDK supports, as listed in the Unleash server-side compatibility overview. The proxy has supported advanced strategy constraints since version 0.8. Because the proxy uses the Node SDK internally,

    Run The Unleash Proxy

    The Unleash proxy is a small stateless HTTP application you run. The only requirement is that it needs to be able to talk with the Unleash API (either Unleash OSS or Unleash Hosted).

    Run with Docker

    The easies way to run Unleash is via Docker. We have published a docker image on docker hub.

    Step 1: Pull

    docker pull unleashorg/unleash-proxy

    Step 2: Start

    docker run \
    -e UNLEASH_PROXY_CLIENT_KEYS=some-secret \
    -e UNLEASH_URL=https://app.unleash-hosted.com/demo/api/ \
    -e UNLEASH_API_TOKEN=56907a2fa53c1d16101d509a10b78e36190b0f918d9f122d \
    -p 3000:3000 \
    unleashorg/unleash-proxy

    You should see the following output:

    Unleash-proxy is listening on port 3000!

    Step 3: verify

    In order to verify the proxy you can use curl and see that you get a few evaluated feature toggles back:

    curl http://localhost:3000/proxy -H "Authorization: some-secret"

    Expected output would be something like:

    {
    "toggles": [
    {
    "name": "demo",
    "enabled": true,
    "variant": {
    "name": "disabled",
    "enabled": false
    }
    },
    {
    "name": "demoApp.step1",
    "enabled": true,
    "variant": {
    "name": "disabled",
    "enabled": false
    }
    }
    ]
    }

    Health endpoint

    The proxy will try to synchronize with the Unleash API at startup, until it has successfully done that the proxy will return HTTP 503 - Not Read? for all request. You can use the health endpoint to validate that the proxy is ready to recieve requests:

    curl http://localhost:3000/proxy/health -I
    HTTP/1.1 200 OK
    Access-Control-Allow-Origin: *
    Access-Control-Expose-Headers: ETag
    Content-Type: text/html; charset=utf-8
    Content-Length: 2
    ETag: W/"2-eoX0dku9ba8cNUXvu/DyeabcC+s"
    Date: Fri, 04 Jun 2021 10:38:27 GMT
    Connection: keep-alive
    Keep-Alive: timeout=5

    Run with Node.js:

    STEP 1: Install dependency

    npm install @unleash/proxy

    STEP 2: use in your code

    const port = 3000;

    const { createApp } = require('@unleash/proxy');

    const app = createApp({
    unleashUrl: 'https://app.unleash-hosted.com/demo/api/',
    unleashApiToken: '56907a2fa53c1d16101d509a10b78e36190b0f918d9f122d',
    clientKeys: ['proxy-secret', 'another-proxy-secret', 's1'],
    refreshInterval: 1000,
    // logLevel: 'info',
    // projectName: 'order-team',
    // environment: 'development',
    });

    app.listen(port, () =>
    // eslint-disable-next-line no-console
    console.log(`Unleash Proxy listening on http://localhost:${port}/proxy`),
    );

    How to connect to the Proxy?

    Unleash offers several different client-side SDKs for a number of use cases, and the community has created even more. These SDKs connect to the proxy and integrate with well with their designated language/framework. They also sync with the proxy at periodic intervals and post usage metrics back to Unleash via the proxy.

    For applications where there is no appropriate client-side SDK or where you simply want to avoid the dependency, you could also get away with using a simple HTTP-client if you just want to get the list of active features.

    The proxy is also ideal fit for serverless functions such as AWS Lambda. In that scenario the proxy can run on a small container near the serverless function, preferably in the same VPC, giving the lambda extremely fast access to feature flags, at a predictable cost.

    Custom activation strategies


    ℹ️ Limitations for hosted proxies

    Custom activation strategies can not be used with the Unleash-hosted proxy available to Pro and Enterprise customers.


    The Unleash Proxy can load custom activation strategies for client-side SDKs. For a step-by-step guide, refer to the how to use custom strategies guide.

    To load custom strategies, use either of these two options:

    • the customStrategies option: use this if you're running the Unleash Proxy via Node directly.
    • the UNLEASH_CUSTOM_STRATEGIES_FILE environment variable: use this if you're running the proxy as a container.

    Both options take a list of file paths to JavaScript files that export custom strategy implementations.

    Custom activation strategy files format

    Each strategy file must export a list of instantiated strategies. A file can export as many strategies as you'd like.

    Here's an example file that exports two custom strategies:

    const { Strategy } = require('unleash-client');

    class MyCustomStrategy extends Strategy {
    // ... strategy implementation
    }

    class MyOtherCustomStrategy extends Strategy {
    // ... strategy implementation
    }

    // export strategies
    module.exports = [new MyCustomStrategy(), new MyOtherCustomStrategy()];

    Refer the custom activation strategy documentation for more details on how to implement a custom activation strategy.


    This content was generated on

    - - +

    Unleash Proxy

    Generated content

    This document was generated from the README in the Unleash Proxy GitHub repository.

    tip

    Looking for how to run the Unleash proxy? Check out the how to run the Unleash proxy guide!

    Build & Tests npm Docker Pulls

    The Unleash Proxy

    The Unleash proxy offers a way to use Unleash in client-side applications, such as single-page and native apps.

    Note: You might want to consider using Unleash Edge instead.

    The Unleash proxy sits between the Unleash API and your client-side SDK and does the evaluation of feature toggles for your client-side SDK. This way, you can keep your configuration private and secure, while still allowing your client-side apps to use Unleash's features.

    The proxy offers three important features:

    • Performance: The caches all features in memory and can run close to your end-users. A single instance can able to handle thousands of requests per second, and you can easily scale it by adding additional instances.
    • Security: The proxy evaluates the features for the user on the server-side and by default only exposes results for features that are enabled for the specific user. No feature toggle configuration is ever shared with the user.
    • Privacy: If you run the proxy yourself, Unleash will never get any data on your end-users: no user ids, no IPs, no nothing.
    Client-side apps connect to the Unleash proxy, which in turn connects to the Unleash API. The proxy itself uses the Unleash Node.js SDK to evaluate features. The SDK syncs with Unleash in the background. Local evaluation on the proxy provides privacy.

    A note on privacy and the proxy

    Why would you not want to expose your Unleash configuration to your end-users?

    The way Unleash works, you can add all kinds of data to feature strategies and constraints. For instance, you might show a feature only to a specific subset of user IDs; or you might have a brand new and unannounced new feature with a revealing name.

    If you just sent the regular Unleash client payload to your client-side apps, all of this — the user IDs and the new feature names — would be exposed to your users.

    Single page apps work in the context of a specific user. The proxy allows you to only provide data that relates to that one user: The proxy will default to only returning the evaluated toggles that should be enabled for that specific user in that specific context.

    API

    The Unleash proxy exposes a simple API for consumption by client-side SDKs.

    OpenAPI integration


    ℹ️ Availability

    The OpenAPI integration is available in versions 0.9 and later of the Unleash proxy.


    The proxy can expose a runtime-generated OpenAPI JSON spec and a corresponding OpenAPI UI for its API. The OpenAPI UI page is an interactive page where you can discover and test the API endpoints the proxy exposes. The JSON spec can be used to generate an OpenAPI client with OpenAPI tooling such as the OpenAPI generator.

    To enable the JSON spec and UI, set ENABLE_OAS (environment variable) or enableOAS (in-code configuration variable) to true.

    The spec and UI can then be found at <base url>/docs/openapi.json and <base url>/docs/openapi respectively.

    You can refer to the how to enable the OpenAPI spec guide for more detailed information on how to configure it.

    GET /proxy

    The primary proxy API operation. This endpoint accepts an Unleash context encoded as query parameters, and will return all toggles that are evaluated as true for the provided context.

    When sending GET requests to the Unleash proxy's /proxy endpoint, the request should contain the current Unleash context as query parameters. The proxy will return all enabled toggles for the provided context.

    Payload

    The GET /proxy operation returns information about toggles enabled for the current user. The payload is a JSON object with a toggles property, which contains a list of toggles.

    {
    "toggles": [
    {
    "name": "demo",
    "enabled": true,
    "variant": {
    "name": "disabled",
    "enabled": false
    }
    },
    {
    "name": "demoApp.step1",
    "enabled": true,
    "variant": {
    "name": "disabled",
    "enabled": false
    }
    }
    ]
    }

    Toggle data

    The data for a toggle without variants looks like this:

    {
    "name": "basic-toggle",
    "enabled": true,
    "variant": {
    "name": "disabled",
    "enabled": false
    }
    }
    • name: the name of the feature.
    • enabled: whether the toggle is enabled or not. Will always be true.
    • variant: describes whether the toggle has variants and, if it does, what variant is active for this user. If a toggle doesn't have any variants, it will always be {"name": "disabled", "enabled": false}.

    ℹ️ The "disabled" variant

    Unleash uses a fallback variant called "disabled" to indicate that a toggle has no variants. However, you are free to create a variant called "disabled" yourself. In that case you can tell them apart by checking the variant's enabled property: if the toggle has no variants, enabled will be false. If the toggle is the "disabled" variant that you created, it will have enabled set to true.


    If a toggle has variants, then the variant object can also contain an optional payload property. The payload will contain data about the variant's payload: what type it is, and what the content is. To learn more about variants and their payloads, check the feature toggle variants documentation.

    Variant toggles without payloads look will have their name listed and the enabled property set to true:

    {
    "name": "toggle-with-variants",
    "enabled": true,
    "variant": {
    "name": "simple",
    "enabled": true
    }
    }

    If the variant has a payload, the optional payload property will list the payload's type and it's content in a stringified form:

    {
    "name": "toggle-with-variants",
    "enabled": true,
    "variant": {
    "name": "with-payload-string",
    "payload": {
    "type": "string",
    "value": "this string is the variant's payload"
    },
    "enabled": true
    }
    }

    For the variant property:

    • name: is the name of the variant, as shown in the Admin UI.
    • enabled: indicates whether the variant is enabled or not. If the toggle has variants, this is always true.
    • payload (optional): Only present if the variant has a payload. Describes the payload's type and content.

    If the variant has a payload, the payload object contains:

    • type: the type of the variant's payload
    • value: the value of the variant's payload

    The value will always be the payload's content as a string, escaped as necessary. For instance, a variant with a JSON payload would look like this:

    {
    "name": "toggle-with-variants",
    "enabled": true,
    "variant": {
    "name": "with-payload-json",
    "payload": {
    "type": "json",
    "value": "{\"description\": \"this is data delivered as a json string\"}"
    },
    "enabled": true
    }
    }

    POST /proxy

    The proxy also offers a POST API used to evaluate toggles This can be used to evaluate a list of know toggle names or to retrieve all enabled toggles for a given context.

    Evaluate list of known toggles

    This method will allow you to send a list of toggle names together with an Unleash Context and evaluate them accordingly. It will return enablement of all provided toggles.

    URL: POST https://proxy-host.server/proxy

    Content Type: application/json

    Body:

    {
    "toggles": ["demoApp.step1"],
    "context": {
    "appName": "someApp",
    "sessionId": "233312AFF22"
    }
    }

    Result:

    HTTP/1.1 200 OK
    Access-Control-Allow-Origin: *
    Access-Control-Expose-Headers: ETag
    Cache-control: public, max-age=2
    Connection: keep-alive
    Content-Length: 122
    Content-Type: application/json; charset=utf-8
    Date: Wed, 30 Nov 2022 14:46:48 GMT
    ETag: W/"7a-RMKUyY0BWIhjahpVPWnNdXyDw6I"
    Keep-Alive: timeout=5
    Vary: Accept-Encoding

    {
    "toggles": [
    {
    "enabled": false,
    "impressionData": true,
    "name": "demoApp.step1",
    "variant": {
    "enabled": false,
    "name": "disabled"
    }
    }
    ]
    }

    Evaluate all enabled toggles

    This method will allow you to get all enabled toggles for a given context.

    URL: POST https://proxy-host.server/proxy

    Content Type: application/json

    Body:

    {
    "context": {
    "appName": "someApp",
    "sessionId": "233312AFF22"
    }
    }

    Result:

    HTTP/1.1 200 OK
    Access-Control-Allow-Origin: *
    Access-Control-Expose-Headers: ETag
    Cache-control: public, max-age=2
    Connection: keep-alive
    Content-Length: 465
    Content-Type: application/json; charset=utf-8
    Date: Wed, 30 Nov 2022 14:48:55 GMT
    ETag: W/"1d1-dm6tkvMpkx42mZmojNSNKmHid1M"
    Keep-Alive: timeout=5
    Vary: Accept-Encoding

    {
    "toggles": [
    {
    "enabled": true,
    "impressionData": false,
    "name": "demoApp.step2",
    "variant": {
    "enabled": true,
    "name": "userWithId",
    "payload": {
    "type": "string",
    "value": "90732934"
    }
    }
    },
    {
    "enabled": true,
    "impressionData": false,
    "name": "demoApp.step3",
    "variant": {
    "enabled": true,
    "name": "C",
    "payload": {
    "type": "string",
    "value": "hello"
    }
    }
    },
    {
    "enabled": true,
    "impressionData": true,
    "name": "demoApp.step4",
    "variant": {
    "enabled": true,
    "name": "Orange",
    "payload": {
    "type": "string",
    "value": "orange"
    }
    }
    }
    ]
    }


    GET /proxy/all Return enabled and disabled toggles:

    By default, the proxy only returns enabled toggles. However, in certain use cases, you might want it to return all toggles, regardless of whether they're enabled or disabled. The /proxy/all endpoint does this.

    Because returning all toggles regardless of their state is a potential security vulnerability, the endpoint has to be explicitly enabled. To enable it, set the enableAllEndpoint configuration option or the ENABLE_ALL_ENDPOINT environment variable to true.

    The response payload follows the same format as the GET /proxy response payload.

    GET /proxy/health: Health endpoint

    The proxy will try to synchronize with the Unleash API at startup, until it has successfully done that the proxy will return HTTP 503 - Not Ready for all request. You can use the health endpoint to validate that the proxy is ready to receive requests:

    curl http://localhost:3000/proxy/health -I

    If the proxy is ready, the response should look a little something like this:

    HTTP/1.1 200 OK
    Access-Control-Allow-Origin: *
    Access-Control-Expose-Headers: ETag
    Content-Type: text/html; charset=utf-8
    Content-Length: 2
    ETag: W/"2-eoX0dku9ba8cNUXvu/DyeabcC+s"
    Date: Fri, 04 Jun 2021 10:38:27 GMT
    Connection: keep-alive
    Keep-Alive: timeout=5

    Configuration options

    The Proxy has a large number of configuration options that you can use to adjust it to your specific use case. The following table lists all the available options.


    ℹ️ Required configuration

    You must configure these three variables for the proxy to start successfully:

    • unleashUrl / UNLEASH_URL

    • unleashApiToken / UNLEASH_API_TOKEN

    • clientKeys / UNLEASH_PROXY_CLIENT_KEYS


    OptionEnvironment VariableDefault valueRequiredDescription
    unleashUrlUNLEASH_URLn/ayesAPI Url to the Unleash instance to connect to
    unleashApiTokenUNLEASH_API_TOKENn/ayesAPI token (client) needed to connect to Unleash API.
    clientKeysUNLEASH_PROXY_CLIENT_KEYSn/ayesList of client keys that the proxy should accept. When querying the proxy, Proxy SDKs must set the request's client keys header to one of these values. The default client keys header is Authorization.
    proxySecretsUNLEASH_PROXY_SECRETSn/anoDeprecated alias for clientKeys. Please use clientKeys instead.
    n/aPORT or PROXY_PORT3000noThe port where the proxy should listen.
    proxyBasePathPROXY_BASE_PATH""noThe base path to run the proxy from. "/proxy" will be added at the end. For instance, if proxyBasePath is "base/path", the proxy will run at /base/path/proxy.
    unleashAppNameUNLEASH_APP_NAME"unleash-proxy"noApp name to used when registering with Unleash
    unleashInstanceIdUNLEASH_INSTANCE_IDgeneratednoUnleash instance id to used when registering with Unleash
    refreshIntervalUNLEASH_FETCH_INTERVAL5000noHow often the proxy should query Unleash for updates, defined in ms.
    metricsIntervalUNLEASH_METRICS_INTERVAL30000noHow often the proxy should send usage metrics back to Unleash, defined in ms.
    metricsJitterUNLEASH_METRICS_JITTER0noAdds jitter to the metrics interval to avoid multiple instances sending metrics at the same time, defined in ms.
    environmentUNLEASH_ENVIRONMENTundefinednoIf set this will be the environment used by the proxy in the Unleash Context. It will not be possible for proxy SDKs to override the environment if set.
    projectNameUNLEASH_PROJECT_NAMEundefinednoThe projectName (id) to fetch feature toggles for. The proxy will only return know about feature toggles that belongs to the project, if specified.
    loggern/aSimpleLoggernoRegister a custom logger.
    useJsonLoggerJSON_LOGGERfalsenoSets the default logger to log as one-line JSON. This has no effect if a custom logger is provided.
    logLevelLOG_LEVEL "warn"noUsed to set logLevel. Supported options: "debug", "info", "warn", "error" and "fatal"
    customStrategiesUNLEASH_CUSTOM_STRATEGIES_FILE[]noUse this option to inject implementation of custom activation strategies. If you are using UNLEASH_CUSTOM_STRATEGIES_FILE you need to provide a valid path to a javascript file which exports an array of custom activation strategies and the SDK will automatically load these
    trustProxyTRUST_PROXY falsenoBy enabling the trustProxy option, Unleash Proxy will have knowledge that it's sitting behind a proxy and that the X-Forwarded-* header fields may be trusted, which otherwise may be easily spoofed. The proxy will automatically enrich the ip address in the Unleash Context. Can either be true/false (Trust all proxies), trust only given IP/CIDR (e.g. '127.0.0.1') as a string. May be a list of comma separated values (e.g. '127.0.0.1,192.168.1.1/24'
    namePrefixUNLEASH_NAME_PREFIXundefinednoUsed to filter features by using prefix when requesting backend values.
    tagsUNLEASH_TAGSundefinednoUsed to filter features by using tags set for features. Format should be tagName:tagValue,tagName2:tagValue2
    clientKeysHeaderNameCLIENT_KEY_HEADER_NAME"authorization"noThe name of the HTTP header to use for client keys. Incoming requests must set the value of this header to one of the Proxy's clientKeys to be authorized successfully.
    storageProvidern/aundefinednoRegister a custom storage provider. Refer to the section on custom storage providers in the Node.js SDK's readme for more information.
    enableOASENABLE_OASfalsenoSet to true to expose the proxy's OpenAPI spec at /docs/openapi.json and an interactive OpenAPI UI at /docs/openapi. Read more in the OpenAPI section.
    enableAllEndpointENABLE_ALL_ENDPOINTfalsenoSet to true to expose the /proxy/all endpoint. Refer to the section on returning all toggles for more info.
    corsn/an/anoPass custom options for CORS module
    cors.allowedHeadersCORS_ALLOWED_HEADERSn/anoHeaders to allow for CORS
    cors.credentialsCORS_CREDENTIALSfalsenoAllow credentials in CORS requests
    cors.exposedHeadersCORS_EXPOSED_HEADERS"ETag"noHeaders to expose for CORS
    cors.maxAgeCORS_MAX_AGE172800noMaximum number of seconds to cache CORS results
    cors.methodsCORS_METHODS"GET, POST"noMethods to allow for CORS
    cors.optionsSuccessStatusCORS_OPTIONS_SUCCESS_STATUS204noIf the Options call passes CORS, which http status to return
    cors.originCORS_ORIGIN*noOrigin URL or list of comma separated list of URLs to whitelist for CORS
    cors.preflightContinueCORS_PREFLIGHT_CONTINUEfalseno
    httpOptions.rejectUnauthorizedHTTP_OPTIONS_REJECT_UNAUTHORIZEDtruenoIf true, unleash-proxy will automatically reject connections to unleash server with invalid certificates

    Experimental configuration options

    Some functionality is under validation and introduced as experimental. This allows us to test new functionality early and quickly. Experimental options can change or be removed at any point in the future and are explicitly not part of the proxy's semantic versioning.

    OptionEnvironment VariableDefault valueRequiredDescription
    expBootstrapn/an/anoWhere the Proxy can bootstrap configuration from. See Node.js SDK for details.
    expBootstrap.urlEXP_BOOTSTRAP_URLn/anoUrl where the Proxy can bootstrap configuration from. See Node.js SDK for details.
    expBootstrap.urlHeaders.AuthorizationEXP_BOOTSTRAP_AUTHORIZATIONn/anoAuthorization header value to be used when bootstrapping
    expServerSideSdkConfig.tokensEXP_SERVER_SIDE_SDK_CONFIG_TOKENSn/anoAPI tokens that can be used by Server SDKs (and proxies) to read feature toggle configuration from this Proxy instance.
    OptionEnvironment VariableDefault valueRequiredDescription
    expBootstrapn/an/anoWhere the Proxy can bootstrap configuration from. See Node.js SDK for details.
    expBootstrap.urlEXP_BOOTSTRAP_URLn/anoUrl where the Proxy can bootstrap configuration from. See Node.js SDK for details.
    expBootstrap.urlHeaders.AuthorizationEXP_BOOTSTRAP_AUTHORIZATIONn/anoAuthorization header value to be used when bootstrapping
    expServerSideSdkConfig.tokensEXP_SERVER_SIDE_SDK_CONFIG_TOKENSn/anoAPI tokens that can be used by Server SDKs (and proxies) to read feature toggle configuration from this Proxy instance.
    expCustomEnrichersEXP_CUSTOM_ENRICHERS_FILE[]noUse this option to inject implementation of custom context enrichers. If you are using EXP_CUSTOM_ENRICHERS_FILE you need to provide a valid path to a javascript file which exports an array of custom context enrichers and the SDK will automatically load these

    Internal SDK capabilities

    The proxy uses the Node.js SDK internally. When you start the proxy it initializes a client for you using any configuration option you provided on startup. You can also provide your own client.

    The client that the proxy uses supports all the SDK features that the Node.js SDK supports, as listed in the Unleash server-side compatibility overview. The proxy has supported advanced strategy constraints since version 0.8. Because the proxy uses the Node SDK internally,

    Run The Unleash Proxy

    The Unleash proxy is a small stateless HTTP application you run. The only requirement is that it needs to be able to talk with the Unleash API (either Unleash OSS or Unleash Hosted).

    Run with Docker

    The easies way to run Unleash is via Docker. We have published a docker image on docker hub.

    Step 1: Pull

    docker pull unleashorg/unleash-proxy

    Step 2: Start

    docker run \
    -e UNLEASH_PROXY_CLIENT_KEYS=some-secret \
    -e UNLEASH_URL=https://app.unleash-hosted.com/demo/api/ \
    -e UNLEASH_API_TOKEN=56907a2fa53c1d16101d509a10b78e36190b0f918d9f122d \
    -p 3000:3000 \
    unleashorg/unleash-proxy

    You should see the following output:

    Unleash-proxy is listening on port 3000!

    Step 3: verify

    In order to verify the proxy you can use curl and see that you get a few evaluated feature toggles back:

    curl http://localhost:3000/proxy -H "Authorization: some-secret"

    Expected output would be something like:

    {
    "toggles": [
    {
    "name": "demo",
    "enabled": true,
    "variant": {
    "name": "disabled",
    "enabled": false
    }
    },
    {
    "name": "demoApp.step1",
    "enabled": true,
    "variant": {
    "name": "disabled",
    "enabled": false
    }
    }
    ]
    }

    Health endpoint

    The proxy will try to synchronize with the Unleash API at startup, until it has successfully done that the proxy will return HTTP 503 - Not Read? for all request. You can use the health endpoint to validate that the proxy is ready to recieve requests:

    curl http://localhost:3000/proxy/health -I
    HTTP/1.1 200 OK
    Access-Control-Allow-Origin: *
    Access-Control-Expose-Headers: ETag
    Content-Type: text/html; charset=utf-8
    Content-Length: 2
    ETag: W/"2-eoX0dku9ba8cNUXvu/DyeabcC+s"
    Date: Fri, 04 Jun 2021 10:38:27 GMT
    Connection: keep-alive
    Keep-Alive: timeout=5

    Run with Node.js:

    STEP 1: Install dependency

    npm install @unleash/proxy

    STEP 2: use in your code

    const port = 3000;

    const { createApp } = require('@unleash/proxy');

    const app = createApp({
    unleashUrl: 'https://app.unleash-hosted.com/demo/api/',
    unleashApiToken: '56907a2fa53c1d16101d509a10b78e36190b0f918d9f122d',
    clientKeys: ['proxy-secret', 'another-proxy-secret', 's1'],
    refreshInterval: 1000,
    // logLevel: 'info',
    // projectName: 'order-team',
    // environment: 'development',
    });

    app.listen(port, () =>
    // eslint-disable-next-line no-console
    console.log(`Unleash Proxy listening on http://localhost:${port}/proxy`),
    );

    How to connect to the Proxy?

    Unleash offers several different client-side SDKs for a number of use cases, and the community has created even more. These SDKs connect to the proxy and integrate with well with their designated language/framework. They also sync with the proxy at periodic intervals and post usage metrics back to Unleash via the proxy.

    For applications where there is no appropriate client-side SDK or where you simply want to avoid the dependency, you could also get away with using a simple HTTP-client if you just want to get the list of active features.

    The proxy is also ideal fit for serverless functions such as AWS Lambda. In that scenario the proxy can run on a small container near the serverless function, preferably in the same VPC, giving the lambda extremely fast access to feature flags, at a predictable cost.

    Custom activation strategies


    ℹ️ Limitations for hosted proxies

    Custom activation strategies can not be used with the Unleash-hosted proxy available to Pro and Enterprise customers.


    The Unleash Proxy can load custom activation strategies for client-side SDKs. For a step-by-step guide, refer to the how to use custom strategies guide.

    To load custom strategies, use either of these two options:

    • the customStrategies option: use this if you're running the Unleash Proxy via Node directly.
    • the UNLEASH_CUSTOM_STRATEGIES_FILE environment variable: use this if you're running the proxy as a container.

    Both options take a list of file paths to JavaScript files that export custom strategy implementations.

    Custom activation strategy files format

    Each strategy file must export a list of instantiated strategies. A file can export as many strategies as you'd like.

    Here's an example file that exports two custom strategies:

    const { Strategy } = require('unleash-client');

    class MyCustomStrategy extends Strategy {
    // ... strategy implementation
    }

    class MyOtherCustomStrategy extends Strategy {
    // ... strategy implementation
    }

    // export strategies
    module.exports = [new MyCustomStrategy(), new MyOtherCustomStrategy()];

    Refer the custom activation strategy documentation for more details on how to implement a custom activation strategy.


    This content was generated on

    + + \ No newline at end of file diff --git a/reference/whats-new-v4.html b/reference/whats-new-v4.html index c7c47e7507..49cae8241f 100644 --- a/reference/whats-new-v4.html +++ b/reference/whats-new-v4.html @@ -20,8 +20,8 @@ - - + + @@ -29,7 +29,7 @@

    What's new in v4?

    Version 4 of Unleash brings a lot of improvements to Unleash. In this document we will highlight some of the things that has been added.

    Upgrade with ease

    Unleash can either be hosted by us or self-hosted. If you have a managed Unleash Enterprise instance you are automatically upgraded to version 4. If you manage Unleash yourself (either Open-Source or Enterprise Self-hosted) we recommend reading the migration guide.

    PS! The first time you access Unleash v4 from a self-hosted instance you will need to login with the default admin user:

    • username: admin
    • password: unleash4all

    (We recommend changing the password of the default user from the admin section.)

    Role-Based Access Control

    With Role-Based Access Control you can now assign groups to users in order to control access. You can control access to root resources in addition to controlling access to projects. Please be aware that all existing users will become "Owner" of all existing projects as part of the migration from v3 to v4.

    Read more

    New Integrations

    Integrations make it easy to integrate Unleash with other systems. In version 4 we bring two new integrations to Unleash:

    Improved UX

    Unleash v4 includes a new implementation of the frontend based on Material-ui. This will make it much easier for us to improve the Unleash Admin UI and our ambition is to make it intuitive to use even for non-developers. The improved UX is made available in Unleash Open-Source and Unleash Enterprise.

    New SSO Option

    In version 4 we added support for OpenID Connect as part of the Unleash Enterprise offering. OpenID Connect is a simple identity layer on top of the OAuth 2.0 protocol. It allows Clients to verify the identity of the End-User based on the authentication performed by an Authorization Server, as well as to obtain basic profile information about the End-User in an interoperable and REST-like manner.

    User Management

    In version 4 we improved the User Management and made it available for Unleash Open-Source and Unleash Enterprise. Starting in v4 all users accessing Unleash needs to exist in Unleash in order to gain access (because they need to have the proper permission from RBAC.)

    Read more

    API access

    In version 4 we improved the API Access and made it available for Unleash Open-Source and Unleash Enterprise. Starting from Unleash v4 we require all SDKs to use an access token in order to connect to Unleash.

    Read more

    Custom stickiness

    In Unleash Enterprise v4 you can configure stickiness when you are doing a gradual rollout with the "gradual rollout" strategy (previously known as "flexible rollout") or together with feature toggle variants. This means that you can now have consistent behavior based on any field available on the Unleash context.

    - - + + \ No newline at end of file diff --git a/search.html b/search.html index 1d8afb05e4..2f80039dee 100644 --- a/search.html +++ b/search.html @@ -20,15 +20,15 @@ - - + +

    Search the documentation

    - - + + \ No newline at end of file diff --git a/sitemap.xml b/sitemap.xml index fa1f473314..006766ea56 100644 --- a/sitemap.xml +++ b/sitemap.xml @@ -1 +1 @@ -https://docs.getunleash.io/searchweekly0.5https://docs.getunleash.io/weekly0.5https://docs.getunleash.io/client-specificationweekly0.5https://docs.getunleash.io/contributingweekly0.5https://docs.getunleash.io/contributing/ADRsweekly0.5https://docs.getunleash.io/contributing/ADRs/back-end/breaking-db-changesweekly0.5https://docs.getunleash.io/contributing/ADRs/back-end/namingweekly0.5https://docs.getunleash.io/contributing/ADRs/back-end/POST-PUT-api-payloadweekly0.5https://docs.getunleash.io/contributing/ADRs/back-end/preferred-exportweekly0.5https://docs.getunleash.io/contributing/ADRs/back-end/specificity-db-columnsweekly0.5https://docs.getunleash.io/contributing/ADRs/front-end/component-namingweekly0.5https://docs.getunleash.io/contributing/ADRs/front-end/handling-tablesweekly0.5https://docs.getunleash.io/contributing/ADRs/front-end/interface-namingweekly0.5https://docs.getunleash.io/contributing/ADRs/front-end/preferred-component-props-usageweekly0.5https://docs.getunleash.io/contributing/ADRs/front-end/preferred-data-fetching-methodweekly0.5https://docs.getunleash.io/contributing/ADRs/front-end/preferred-data-mutation-methodweekly0.5https://docs.getunleash.io/contributing/ADRs/front-end/preferred-exportweekly0.5https://docs.getunleash.io/contributing/ADRs/front-end/preferred-folder-structureweekly0.5https://docs.getunleash.io/contributing/ADRs/front-end/preferred-form-architectureweekly0.5https://docs.getunleash.io/contributing/ADRs/front-end/preferred-function-typeweekly0.5https://docs.getunleash.io/contributing/ADRs/front-end/preferred-styles-import-placementweekly0.5https://docs.getunleash.io/contributing/ADRs/front-end/preferred-styling-methodweekly0.5https://docs.getunleash.io/contributing/ADRs/front-end/sdk-generatorweekly0.5https://docs.getunleash.io/contributing/ADRs/overarching/domain-languageweekly0.5https://docs.getunleash.io/contributing/ADRs/overarching/separation-request-response-schemasweekly0.5https://docs.getunleash.io/contributing/backend/overviewweekly0.5https://docs.getunleash.io/contributing/developer-guideweekly0.5https://docs.getunleash.io/contributing/frontend/overviewweekly0.5https://docs.getunleash.io/feature-flag-tutorialsweekly0.5https://docs.getunleash.io/feature-flag-tutorials/flutter/a-b-testingweekly0.5https://docs.getunleash.io/feature-flag-tutorials/nextjs/implementing-feature-flagsweekly0.5https://docs.getunleash.io/feature-flag-tutorials/reactweekly0.5https://docs.getunleash.io/feature-flag-tutorials/react/examplesweekly0.5https://docs.getunleash.io/how-toweekly0.5https://docs.getunleash.io/how-to/apiweekly0.5https://docs.getunleash.io/how-to/envweekly0.5https://docs.getunleash.io/how-to/how-to-add-feature-flag-naming-patternsweekly0.5https://docs.getunleash.io/how-to/how-to-add-sso-azure-samlweekly0.5https://docs.getunleash.io/how-to/how-to-add-sso-googleweekly0.5https://docs.getunleash.io/how-to/how-to-add-sso-open-id-connectweekly0.5https://docs.getunleash.io/how-to/how-to-add-sso-samlweekly0.5https://docs.getunleash.io/how-to/how-to-add-sso-saml-keycloakweekly0.5https://docs.getunleash.io/how-to/how-to-add-strategy-constraintsweekly0.5https://docs.getunleash.io/how-to/how-to-add-users-to-unleashweekly0.5https://docs.getunleash.io/how-to/how-to-capture-impression-dataweekly0.5https://docs.getunleash.io/how-to/how-to-clone-environmentsweekly0.5https://docs.getunleash.io/how-to/how-to-create-and-assign-custom-project-rolesweekly0.5https://docs.getunleash.io/how-to/how-to-create-and-assign-custom-root-rolesweekly0.5https://docs.getunleash.io/how-to/how-to-create-and-display-bannersweekly0.5https://docs.getunleash.io/how-to/how-to-create-and-manage-user-groupsweekly0.5https://docs.getunleash.io/how-to/how-to-create-api-tokensweekly0.5https://docs.getunleash.io/how-to/how-to-create-feature-togglesweekly0.5https://docs.getunleash.io/how-to/how-to-create-personal-access-tokensweekly0.5https://docs.getunleash.io/how-to/how-to-create-project-api-tokensweekly0.5https://docs.getunleash.io/how-to/how-to-create-service-accountsweekly0.5https://docs.getunleash.io/how-to/how-to-define-custom-context-fieldsweekly0.5https://docs.getunleash.io/how-to/how-to-download-login-historyweekly0.5https://docs.getunleash.io/how-to/how-to-enable-openapiweekly0.5https://docs.getunleash.io/how-to/how-to-environment-import-exportweekly0.5https://docs.getunleash.io/how-to/how-to-import-exportweekly0.5https://docs.getunleash.io/how-to/how-to-manage-public-invite-tokensweekly0.5https://docs.getunleash.io/how-to/how-to-run-the-unleash-proxyweekly0.5https://docs.getunleash.io/how-to/how-to-schedule-feature-releasesweekly0.5https://docs.getunleash.io/how-to/how-to-send-feature-updates-to-slack-deprecatedweekly0.5https://docs.getunleash.io/how-to/how-to-set-up-group-sso-syncweekly0.5https://docs.getunleash.io/how-to/how-to-setup-sso-keycloak-group-syncweekly0.5https://docs.getunleash.io/how-to/how-to-synchronize-unleash-instancesweekly0.5https://docs.getunleash.io/how-to/how-to-use-custom-strategiesweekly0.5https://docs.getunleash.io/how-to/how-to-use-the-admin-apiweekly0.5https://docs.getunleash.io/how-to/miscweekly0.5https://docs.getunleash.io/how-to/proxyweekly0.5https://docs.getunleash.io/how-to/ssoweekly0.5https://docs.getunleash.io/how-to/users-and-permissionsweekly0.5https://docs.getunleash.io/quickstartweekly0.5https://docs.getunleash.io/referenceweekly0.5https://docs.getunleash.io/reference/activation-strategiesweekly0.5https://docs.getunleash.io/reference/api-tokens-and-client-keysweekly0.5https://docs.getunleash.io/reference/api/legacy/unleashweekly0.5https://docs.getunleash.io/reference/api/legacy/unleash/admin/addonsweekly0.5https://docs.getunleash.io/reference/api/legacy/unleash/admin/archiveweekly0.5https://docs.getunleash.io/reference/api/legacy/unleash/admin/contextweekly0.5https://docs.getunleash.io/reference/api/legacy/unleash/admin/eventsweekly0.5https://docs.getunleash.io/reference/api/legacy/unleash/admin/feature-typesweekly0.5https://docs.getunleash.io/reference/api/legacy/unleash/admin/featuresweekly0.5https://docs.getunleash.io/reference/api/legacy/unleash/admin/features-v2weekly0.5https://docs.getunleash.io/reference/api/legacy/unleash/admin/metricsweekly0.5https://docs.getunleash.io/reference/api/legacy/unleash/admin/projectsweekly0.5https://docs.getunleash.io/reference/api/legacy/unleash/admin/segmentsweekly0.5https://docs.getunleash.io/reference/api/legacy/unleash/admin/stateweekly0.5https://docs.getunleash.io/reference/api/legacy/unleash/admin/strategiesweekly0.5https://docs.getunleash.io/reference/api/legacy/unleash/admin/tagsweekly0.5https://docs.getunleash.io/reference/api/legacy/unleash/admin/user-adminweekly0.5https://docs.getunleash.io/reference/api/legacy/unleash/basic-authweekly0.5https://docs.getunleash.io/reference/api/legacy/unleash/client/featuresweekly0.5https://docs.getunleash.io/reference/api/legacy/unleash/client/metricsweekly0.5https://docs.getunleash.io/reference/api/legacy/unleash/client/registerweekly0.5https://docs.getunleash.io/reference/api/legacy/unleash/internal/healthweekly0.5https://docs.getunleash.io/reference/api/legacy/unleash/internal/prometheusweekly0.5https://docs.getunleash.io/reference/api/unleashweekly0.5https://docs.getunleash.io/reference/api/unleash/add-access-to-projectweekly0.5https://docs.getunleash.io/reference/api/unleash/add-change-request-commentweekly0.5https://docs.getunleash.io/reference/api/unleash/add-default-strategy-to-project-environmentweekly0.5https://docs.getunleash.io/reference/api/unleash/add-environment-to-projectweekly0.5https://docs.getunleash.io/reference/api/unleash/add-favorite-featureweekly0.5https://docs.getunleash.io/reference/api/unleash/add-favorite-projectweekly0.5https://docs.getunleash.io/reference/api/unleash/add-feature-dependencyweekly0.5https://docs.getunleash.io/reference/api/unleash/add-feature-strategyweekly0.5https://docs.getunleash.io/reference/api/unleash/add-public-signup-token-userweekly0.5https://docs.getunleash.io/reference/api/unleash/add-role-access-to-projectweekly0.5https://docs.getunleash.io/reference/api/unleash/add-role-to-userweekly0.5https://docs.getunleash.io/reference/api/unleash/add-tagweekly0.5https://docs.getunleash.io/reference/api/unleash/add-tag-to-featuresweekly0.5https://docs.getunleash.io/reference/api/unleash/addonsweekly0.5https://docs.getunleash.io/reference/api/unleash/admin-uiweekly0.5https://docs.getunleash.io/reference/api/unleash/api-tokensweekly0.5https://docs.getunleash.io/reference/api/unleash/archiveweekly0.5https://docs.getunleash.io/reference/api/unleash/archive-featureweekly0.5https://docs.getunleash.io/reference/api/unleash/archive-featuresweekly0.5https://docs.getunleash.io/reference/api/unleash/authweekly0.5https://docs.getunleash.io/reference/api/unleash/bannersweekly0.5https://docs.getunleash.io/reference/api/unleash/bulk-metricsweekly0.5https://docs.getunleash.io/reference/api/unleash/bulk-toggle-features-environment-offweekly0.5https://docs.getunleash.io/reference/api/unleash/bulk-toggle-features-environment-onweekly0.5https://docs.getunleash.io/reference/api/unleash/change-my-passwordweekly0.5https://docs.getunleash.io/reference/api/unleash/change-passwordweekly0.5https://docs.getunleash.io/reference/api/unleash/change-projectweekly0.5https://docs.getunleash.io/reference/api/unleash/change-requestweekly0.5https://docs.getunleash.io/reference/api/unleash/change-requestsweekly0.5https://docs.getunleash.io/reference/api/unleash/change-role-for-groupweekly0.5https://docs.getunleash.io/reference/api/unleash/change-role-for-userweekly0.5https://docs.getunleash.io/reference/api/unleash/change-user-passwordweekly0.5https://docs.getunleash.io/reference/api/unleash/check-dependencies-existweekly0.5https://docs.getunleash.io/reference/api/unleash/check-licenseweekly0.5https://docs.getunleash.io/reference/api/unleash/clientweekly0.5https://docs.getunleash.io/reference/api/unleash/clone-environmentweekly0.5https://docs.getunleash.io/reference/api/unleash/clone-featureweekly0.5https://docs.getunleash.io/reference/api/unleash/contextweekly0.5https://docs.getunleash.io/reference/api/unleash/create-addonweekly0.5https://docs.getunleash.io/reference/api/unleash/create-api-tokenweekly0.5https://docs.getunleash.io/reference/api/unleash/create-applicationweekly0.5https://docs.getunleash.io/reference/api/unleash/create-bannerweekly0.5https://docs.getunleash.io/reference/api/unleash/create-context-fieldweekly0.5https://docs.getunleash.io/reference/api/unleash/create-environmentweekly0.5https://docs.getunleash.io/reference/api/unleash/create-featureweekly0.5https://docs.getunleash.io/reference/api/unleash/create-feedbackweekly0.5https://docs.getunleash.io/reference/api/unleash/create-groupweekly0.5https://docs.getunleash.io/reference/api/unleash/create-patweekly0.5https://docs.getunleash.io/reference/api/unleash/create-projectweekly0.5https://docs.getunleash.io/reference/api/unleash/create-project-api-tokenweekly0.5https://docs.getunleash.io/reference/api/unleash/create-public-signup-tokenweekly0.5https://docs.getunleash.io/reference/api/unleash/create-roleweekly0.5https://docs.getunleash.io/reference/api/unleash/create-segmentweekly0.5https://docs.getunleash.io/reference/api/unleash/create-service-accountweekly0.5https://docs.getunleash.io/reference/api/unleash/create-service-account-tokenweekly0.5https://docs.getunleash.io/reference/api/unleash/create-strategyweekly0.5https://docs.getunleash.io/reference/api/unleash/create-tagweekly0.5https://docs.getunleash.io/reference/api/unleash/create-tag-typeweekly0.5https://docs.getunleash.io/reference/api/unleash/create-userweekly0.5https://docs.getunleash.io/reference/api/unleash/delete-addonweekly0.5https://docs.getunleash.io/reference/api/unleash/delete-api-tokenweekly0.5https://docs.getunleash.io/reference/api/unleash/delete-applicationweekly0.5https://docs.getunleash.io/reference/api/unleash/delete-bannerweekly0.5https://docs.getunleash.io/reference/api/unleash/delete-changeweekly0.5https://docs.getunleash.io/reference/api/unleash/delete-change-requestweekly0.5https://docs.getunleash.io/reference/api/unleash/delete-context-fieldweekly0.5https://docs.getunleash.io/reference/api/unleash/delete-featureweekly0.5https://docs.getunleash.io/reference/api/unleash/delete-feature-dependenciesweekly0.5https://docs.getunleash.io/reference/api/unleash/delete-feature-dependencyweekly0.5https://docs.getunleash.io/reference/api/unleash/delete-feature-strategyweekly0.5https://docs.getunleash.io/reference/api/unleash/delete-featuresweekly0.5https://docs.getunleash.io/reference/api/unleash/delete-groupweekly0.5https://docs.getunleash.io/reference/api/unleash/delete-patweekly0.5https://docs.getunleash.io/reference/api/unleash/delete-projectweekly0.5https://docs.getunleash.io/reference/api/unleash/delete-project-api-tokenweekly0.5https://docs.getunleash.io/reference/api/unleash/delete-roleweekly0.5https://docs.getunleash.io/reference/api/unleash/delete-service-accountweekly0.5https://docs.getunleash.io/reference/api/unleash/delete-service-account-tokenweekly0.5https://docs.getunleash.io/reference/api/unleash/delete-tagweekly0.5https://docs.getunleash.io/reference/api/unleash/delete-tag-typeweekly0.5https://docs.getunleash.io/reference/api/unleash/delete-userweekly0.5https://docs.getunleash.io/reference/api/unleash/dependenciesweekly0.5https://docs.getunleash.io/reference/api/unleash/deprecate-strategyweekly0.5https://docs.getunleash.io/reference/api/unleash/disable-bannerweekly0.5https://docs.getunleash.io/reference/api/unleash/edgeweekly0.5https://docs.getunleash.io/reference/api/unleash/edit-changeweekly0.5https://docs.getunleash.io/reference/api/unleash/enable-bannerweekly0.5https://docs.getunleash.io/reference/api/unleash/environmentsweekly0.5https://docs.getunleash.io/reference/api/unleash/eventsweekly0.5https://docs.getunleash.io/reference/api/unleash/exportweekly0.5https://docs.getunleash.io/reference/api/unleash/export-featuresweekly0.5https://docs.getunleash.io/reference/api/unleash/feature-typesweekly0.5https://docs.getunleash.io/reference/api/unleash/featuresweekly0.5https://docs.getunleash.io/reference/api/unleash/frontend-apiweekly0.5https://docs.getunleash.io/reference/api/unleash/get-access-overviewweekly0.5https://docs.getunleash.io/reference/api/unleash/get-addonweekly0.5https://docs.getunleash.io/reference/api/unleash/get-addonsweekly0.5https://docs.getunleash.io/reference/api/unleash/get-admin-countweekly0.5https://docs.getunleash.io/reference/api/unleash/get-advanced-playgroundweekly0.5https://docs.getunleash.io/reference/api/unleash/get-all-api-tokensweekly0.5https://docs.getunleash.io/reference/api/unleash/get-all-client-featuresweekly0.5https://docs.getunleash.io/reference/api/unleash/get-all-environmentsweekly0.5https://docs.getunleash.io/reference/api/unleash/get-all-feature-typesweekly0.5https://docs.getunleash.io/reference/api/unleash/get-all-public-signup-tokensweekly0.5https://docs.getunleash.io/reference/api/unleash/get-all-strategiesweekly0.5https://docs.getunleash.io/reference/api/unleash/get-all-togglesweekly0.5https://docs.getunleash.io/reference/api/unleash/get-api-tokens-by-nameweekly0.5https://docs.getunleash.io/reference/api/unleash/get-applicationweekly0.5https://docs.getunleash.io/reference/api/unleash/get-applicationsweekly0.5https://docs.getunleash.io/reference/api/unleash/get-archived-featuresweekly0.5https://docs.getunleash.io/reference/api/unleash/get-archived-features-by-project-idweekly0.5https://docs.getunleash.io/reference/api/unleash/get-bannersweekly0.5https://docs.getunleash.io/reference/api/unleash/get-base-users-and-groupsweekly0.5https://docs.getunleash.io/reference/api/unleash/get-change-requestweekly0.5https://docs.getunleash.io/reference/api/unleash/get-change-requests-for-projectweekly0.5https://docs.getunleash.io/reference/api/unleash/get-client-featureweekly0.5https://docs.getunleash.io/reference/api/unleash/get-context-fieldweekly0.5https://docs.getunleash.io/reference/api/unleash/get-context-fieldsweekly0.5https://docs.getunleash.io/reference/api/unleash/get-deprecated-project-overviewweekly0.5https://docs.getunleash.io/reference/api/unleash/get-environmentweekly0.5https://docs.getunleash.io/reference/api/unleash/get-environment-feature-variantsweekly0.5https://docs.getunleash.io/reference/api/unleash/get-eventsweekly0.5https://docs.getunleash.io/reference/api/unleash/get-events-for-toggleweekly0.5https://docs.getunleash.io/reference/api/unleash/get-featureweekly0.5https://docs.getunleash.io/reference/api/unleash/get-feature-environmentweekly0.5https://docs.getunleash.io/reference/api/unleash/get-feature-strategiesweekly0.5https://docs.getunleash.io/reference/api/unleash/get-feature-strategyweekly0.5https://docs.getunleash.io/reference/api/unleash/get-feature-usage-summaryweekly0.5https://docs.getunleash.io/reference/api/unleash/get-feature-variantsweekly0.5https://docs.getunleash.io/reference/api/unleash/get-featuresweekly0.5https://docs.getunleash.io/reference/api/unleash/get-feedbackweekly0.5https://docs.getunleash.io/reference/api/unleash/get-frontend-featuresweekly0.5https://docs.getunleash.io/reference/api/unleash/get-google-settingsweekly0.5https://docs.getunleash.io/reference/api/unleash/get-groupweekly0.5https://docs.getunleash.io/reference/api/unleash/get-groupsweekly0.5https://docs.getunleash.io/reference/api/unleash/get-healthweekly0.5https://docs.getunleash.io/reference/api/unleash/get-instance-admin-statsweekly0.5https://docs.getunleash.io/reference/api/unleash/get-instance-admin-stats-csvweekly0.5https://docs.getunleash.io/reference/api/unleash/get-login-historyweekly0.5https://docs.getunleash.io/reference/api/unleash/get-maintenanceweekly0.5https://docs.getunleash.io/reference/api/unleash/get-meweekly0.5https://docs.getunleash.io/reference/api/unleash/get-notificationsweekly0.5https://docs.getunleash.io/reference/api/unleash/get-oidc-settingsweekly0.5https://docs.getunleash.io/reference/api/unleash/get-open-change-requests-for-userweekly0.5https://docs.getunleash.io/reference/api/unleash/get-patsweekly0.5https://docs.getunleash.io/reference/api/unleash/get-pending-change-requests-for-featureweekly0.5https://docs.getunleash.io/reference/api/unleash/get-pending-change-requests-for-userweekly0.5https://docs.getunleash.io/reference/api/unleash/get-permissionsweekly0.5https://docs.getunleash.io/reference/api/unleash/get-playgroundweekly0.5https://docs.getunleash.io/reference/api/unleash/get-profileweekly0.5https://docs.getunleash.io/reference/api/unleash/get-project-accessweekly0.5https://docs.getunleash.io/reference/api/unleash/get-project-api-tokensweekly0.5https://docs.getunleash.io/reference/api/unleash/get-project-change-request-configweekly0.5https://docs.getunleash.io/reference/api/unleash/get-project-doraweekly0.5https://docs.getunleash.io/reference/api/unleash/get-project-environmentsweekly0.5https://docs.getunleash.io/reference/api/unleash/get-project-health-reportweekly0.5https://docs.getunleash.io/reference/api/unleash/get-project-overviewweekly0.5https://docs.getunleash.io/reference/api/unleash/get-project-usersweekly0.5https://docs.getunleash.io/reference/api/unleash/get-projectsweekly0.5https://docs.getunleash.io/reference/api/unleash/get-public-signup-tokenweekly0.5https://docs.getunleash.io/reference/api/unleash/get-raw-feature-metricsweekly0.5https://docs.getunleash.io/reference/api/unleash/get-requests-per-secondweekly0.5https://docs.getunleash.io/reference/api/unleash/get-role-by-idweekly0.5https://docs.getunleash.io/reference/api/unleash/get-role-project-accessweekly0.5https://docs.getunleash.io/reference/api/unleash/get-rolesweekly0.5https://docs.getunleash.io/reference/api/unleash/get-saml-settingsweekly0.5https://docs.getunleash.io/reference/api/unleash/get-scheduled-change-requestsweekly0.5https://docs.getunleash.io/reference/api/unleash/get-segmentweekly0.5https://docs.getunleash.io/reference/api/unleash/get-segmentsweekly0.5https://docs.getunleash.io/reference/api/unleash/get-segments-by-strategy-idweekly0.5https://docs.getunleash.io/reference/api/unleash/get-service-account-tokensweekly0.5https://docs.getunleash.io/reference/api/unleash/get-service-accountsweekly0.5https://docs.getunleash.io/reference/api/unleash/get-simple-settingsweekly0.5https://docs.getunleash.io/reference/api/unleash/get-strategies-by-context-fieldweekly0.5https://docs.getunleash.io/reference/api/unleash/get-strategies-by-segment-idweekly0.5https://docs.getunleash.io/reference/api/unleash/get-strategyweekly0.5https://docs.getunleash.io/reference/api/unleash/get-tagweekly0.5https://docs.getunleash.io/reference/api/unleash/get-tag-typeweekly0.5https://docs.getunleash.io/reference/api/unleash/get-tag-typesweekly0.5https://docs.getunleash.io/reference/api/unleash/get-tagsweekly0.5https://docs.getunleash.io/reference/api/unleash/get-tags-by-typeweekly0.5https://docs.getunleash.io/reference/api/unleash/get-telemetry-settingsweekly0.5https://docs.getunleash.io/reference/api/unleash/get-ui-configweekly0.5https://docs.getunleash.io/reference/api/unleash/get-userweekly0.5https://docs.getunleash.io/reference/api/unleash/get-usersweekly0.5https://docs.getunleash.io/reference/api/unleash/get-valid-tokensweekly0.5https://docs.getunleash.io/reference/api/unleash/importweekly0.5https://docs.getunleash.io/reference/api/unleash/import-exportweekly0.5https://docs.getunleash.io/reference/api/unleash/import-togglesweekly0.5https://docs.getunleash.io/reference/api/unleash/instance-adminweekly0.5https://docs.getunleash.io/reference/api/unleash/list-parent-optionsweekly0.5https://docs.getunleash.io/reference/api/unleash/list-tagsweekly0.5https://docs.getunleash.io/reference/api/unleash/loginweekly0.5https://docs.getunleash.io/reference/api/unleash/maintenanceweekly0.5https://docs.getunleash.io/reference/api/unleash/mark-notifications-as-readweekly0.5https://docs.getunleash.io/reference/api/unleash/metricsweekly0.5https://docs.getunleash.io/reference/api/unleash/notificationsweekly0.5https://docs.getunleash.io/reference/api/unleash/operationalweekly0.5https://docs.getunleash.io/reference/api/unleash/overwrite-environment-feature-variantsweekly0.5https://docs.getunleash.io/reference/api/unleash/overwrite-feature-variantsweekly0.5https://docs.getunleash.io/reference/api/unleash/overwrite-feature-variants-on-environmentsweekly0.5https://docs.getunleash.io/reference/api/unleash/patch-environments-feature-variantsweekly0.5https://docs.getunleash.io/reference/api/unleash/patch-featureweekly0.5https://docs.getunleash.io/reference/api/unleash/patch-feature-strategyweekly0.5https://docs.getunleash.io/reference/api/unleash/patch-feature-variantsweekly0.5https://docs.getunleash.io/reference/api/unleash/personal-access-tokensweekly0.5https://docs.getunleash.io/reference/api/unleash/playgroundweekly0.5https://docs.getunleash.io/reference/api/unleash/projectsweekly0.5https://docs.getunleash.io/reference/api/unleash/provide-feedbackweekly0.5https://docs.getunleash.io/reference/api/unleash/public-signup-tokensweekly0.5https://docs.getunleash.io/reference/api/unleash/reactivate-strategyweekly0.5https://docs.getunleash.io/reference/api/unleash/read-licenseweekly0.5https://docs.getunleash.io/reference/api/unleash/register-client-applicationweekly0.5https://docs.getunleash.io/reference/api/unleash/register-client-metricsweekly0.5https://docs.getunleash.io/reference/api/unleash/register-frontend-clientweekly0.5https://docs.getunleash.io/reference/api/unleash/register-frontend-metricsweekly0.5https://docs.getunleash.io/reference/api/unleash/remove-environmentweekly0.5https://docs.getunleash.io/reference/api/unleash/remove-environment-from-projectweekly0.5https://docs.getunleash.io/reference/api/unleash/remove-favorite-featureweekly0.5https://docs.getunleash.io/reference/api/unleash/remove-favorite-projectweekly0.5https://docs.getunleash.io/reference/api/unleash/remove-group-accessweekly0.5https://docs.getunleash.io/reference/api/unleash/remove-role-for-userweekly0.5https://docs.getunleash.io/reference/api/unleash/remove-role-from-groupweekly0.5https://docs.getunleash.io/reference/api/unleash/remove-segmentweekly0.5https://docs.getunleash.io/reference/api/unleash/remove-strategyweekly0.5https://docs.getunleash.io/reference/api/unleash/remove-tagweekly0.5https://docs.getunleash.io/reference/api/unleash/remove-user-accessweekly0.5https://docs.getunleash.io/reference/api/unleash/reset-user-passwordweekly0.5https://docs.getunleash.io/reference/api/unleash/revive-featureweekly0.5https://docs.getunleash.io/reference/api/unleash/revive-featuresweekly0.5https://docs.getunleash.io/reference/api/unleash/searchweekly0.5https://docs.getunleash.io/reference/api/unleash/search-eventsweekly0.5https://docs.getunleash.io/reference/api/unleash/search-featuresweekly0.5https://docs.getunleash.io/reference/api/unleash/search-usersweekly0.5https://docs.getunleash.io/reference/api/unleash/segmentsweekly0.5https://docs.getunleash.io/reference/api/unleash/send-reset-password-emailweekly0.5https://docs.getunleash.io/reference/api/unleash/service-accountsweekly0.5https://docs.getunleash.io/reference/api/unleash/set-google-settingsweekly0.5https://docs.getunleash.io/reference/api/unleash/set-oidc-settingsweekly0.5https://docs.getunleash.io/reference/api/unleash/set-project-accessweekly0.5https://docs.getunleash.io/reference/api/unleash/set-roles-for-groupweekly0.5https://docs.getunleash.io/reference/api/unleash/set-roles-for-userweekly0.5https://docs.getunleash.io/reference/api/unleash/set-saml-settingsweekly0.5https://docs.getunleash.io/reference/api/unleash/set-simple-settingsweekly0.5https://docs.getunleash.io/reference/api/unleash/set-strategy-sort-orderweekly0.5https://docs.getunleash.io/reference/api/unleash/set-ui-configweekly0.5https://docs.getunleash.io/reference/api/unleash/stale-featuresweekly0.5https://docs.getunleash.io/reference/api/unleash/strategiesweekly0.5https://docs.getunleash.io/reference/api/unleash/tagsweekly0.5https://docs.getunleash.io/reference/api/unleash/telemetryweekly0.5https://docs.getunleash.io/reference/api/unleash/toggle-environment-offweekly0.5https://docs.getunleash.io/reference/api/unleash/toggle-environment-onweekly0.5https://docs.getunleash.io/reference/api/unleash/toggle-feature-environment-offweekly0.5https://docs.getunleash.io/reference/api/unleash/toggle-feature-environment-onweekly0.5https://docs.getunleash.io/reference/api/unleash/toggle-maintenanceweekly0.5https://docs.getunleash.io/reference/api/unleash/unleash-apiweekly0.5https://docs.getunleash.io/reference/api/unleash/unstableweekly0.5https://docs.getunleash.io/reference/api/unleash/update-addonweekly0.5https://docs.getunleash.io/reference/api/unleash/update-api-tokenweekly0.5https://docs.getunleash.io/reference/api/unleash/update-bannerweekly0.5https://docs.getunleash.io/reference/api/unleash/update-change-request-stateweekly0.5https://docs.getunleash.io/reference/api/unleash/update-change-request-titleweekly0.5https://docs.getunleash.io/reference/api/unleash/update-context-fieldweekly0.5https://docs.getunleash.io/reference/api/unleash/update-environmentweekly0.5https://docs.getunleash.io/reference/api/unleash/update-featureweekly0.5https://docs.getunleash.io/reference/api/unleash/update-feature-strategyweekly0.5https://docs.getunleash.io/reference/api/unleash/update-feature-strategy-segmentsweekly0.5https://docs.getunleash.io/reference/api/unleash/update-feature-type-lifetimeweekly0.5https://docs.getunleash.io/reference/api/unleash/update-feedbackweekly0.5https://docs.getunleash.io/reference/api/unleash/update-groupweekly0.5https://docs.getunleash.io/reference/api/unleash/update-licenseweekly0.5https://docs.getunleash.io/reference/api/unleash/update-projectweekly0.5https://docs.getunleash.io/reference/api/unleash/update-project-change-request-configweekly0.5https://docs.getunleash.io/reference/api/unleash/update-project-enterprise-settingsweekly0.5https://docs.getunleash.io/reference/api/unleash/update-public-signup-tokenweekly0.5https://docs.getunleash.io/reference/api/unleash/update-roleweekly0.5https://docs.getunleash.io/reference/api/unleash/update-segmentweekly0.5https://docs.getunleash.io/reference/api/unleash/update-service-accountweekly0.5https://docs.getunleash.io/reference/api/unleash/update-sort-orderweekly0.5https://docs.getunleash.io/reference/api/unleash/update-splash-settingsweekly0.5https://docs.getunleash.io/reference/api/unleash/update-strategyweekly0.5https://docs.getunleash.io/reference/api/unleash/update-tag-typeweekly0.5https://docs.getunleash.io/reference/api/unleash/update-tagsweekly0.5https://docs.getunleash.io/reference/api/unleash/update-userweekly0.5https://docs.getunleash.io/reference/api/unleash/usersweekly0.5https://docs.getunleash.io/reference/api/unleash/validateweekly0.5https://docs.getunleash.io/reference/api/unleash/validate-archive-featuresweekly0.5https://docs.getunleash.io/reference/api/unleash/validate-constraintweekly0.5https://docs.getunleash.io/reference/api/unleash/validate-environment-nameweekly0.5https://docs.getunleash.io/reference/api/unleash/validate-featureweekly0.5https://docs.getunleash.io/reference/api/unleash/validate-importweekly0.5https://docs.getunleash.io/reference/api/unleash/validate-passwordweekly0.5https://docs.getunleash.io/reference/api/unleash/validate-projectweekly0.5https://docs.getunleash.io/reference/api/unleash/validate-public-signup-tokenweekly0.5https://docs.getunleash.io/reference/api/unleash/validate-roleweekly0.5https://docs.getunleash.io/reference/api/unleash/validate-segmentweekly0.5https://docs.getunleash.io/reference/api/unleash/validate-tag-typeweekly0.5https://docs.getunleash.io/reference/api/unleash/validate-tokenweekly0.5https://docs.getunleash.io/reference/api/unleash/validate-user-passwordweekly0.5https://docs.getunleash.io/reference/archived-togglesweekly0.5https://docs.getunleash.io/reference/bannersweekly0.5https://docs.getunleash.io/reference/change-requestsweekly0.5https://docs.getunleash.io/reference/custom-activation-strategiesweekly0.5https://docs.getunleash.io/reference/dependent-featuresweekly0.5https://docs.getunleash.io/reference/environmentsweekly0.5https://docs.getunleash.io/reference/event-logweekly0.5https://docs.getunleash.io/reference/event-typesweekly0.5https://docs.getunleash.io/reference/feature-flag-naming-patternsweekly0.5https://docs.getunleash.io/reference/feature-toggle-typesweekly0.5https://docs.getunleash.io/reference/feature-toggle-variantsweekly0.5https://docs.getunleash.io/reference/feature-togglesweekly0.5https://docs.getunleash.io/reference/front-end-apiweekly0.5https://docs.getunleash.io/reference/impression-dataweekly0.5https://docs.getunleash.io/reference/integrationsweekly0.5https://docs.getunleash.io/reference/integrations/datadogweekly0.5https://docs.getunleash.io/reference/integrations/jira-cloud-plugin-installationweekly0.5https://docs.getunleash.io/reference/integrations/jira-cloud-plugin-usageweekly0.5https://docs.getunleash.io/reference/integrations/jira-server-plugin-installationweekly0.5https://docs.getunleash.io/reference/integrations/jira-server-plugin-usageweekly0.5https://docs.getunleash.io/reference/integrations/slackweekly0.5https://docs.getunleash.io/reference/integrations/slack-appweekly0.5https://docs.getunleash.io/reference/integrations/teamsweekly0.5https://docs.getunleash.io/reference/integrations/webhookweekly0.5https://docs.getunleash.io/reference/login-historyweekly0.5https://docs.getunleash.io/reference/maintenance-modeweekly0.5https://docs.getunleash.io/reference/network-viewweekly0.5https://docs.getunleash.io/reference/notificationsweekly0.5https://docs.getunleash.io/reference/playgroundweekly0.5https://docs.getunleash.io/reference/project-collaboration-modeweekly0.5https://docs.getunleash.io/reference/projectsweekly0.5https://docs.getunleash.io/reference/public-signupweekly0.5https://docs.getunleash.io/reference/rbacweekly0.5https://docs.getunleash.io/reference/sdksweekly0.5https://docs.getunleash.io/reference/sdks/android-proxyweekly0.5https://docs.getunleash.io/reference/sdks/dotnetweekly0.5https://docs.getunleash.io/reference/sdks/flutterweekly0.5https://docs.getunleash.io/reference/sdks/goweekly0.5https://docs.getunleash.io/reference/sdks/ios-proxyweekly0.5https://docs.getunleash.io/reference/sdks/javaweekly0.5https://docs.getunleash.io/reference/sdks/javascript-browserweekly0.5https://docs.getunleash.io/reference/sdks/next-jsweekly0.5https://docs.getunleash.io/reference/sdks/nodeweekly0.5https://docs.getunleash.io/reference/sdks/phpweekly0.5https://docs.getunleash.io/reference/sdks/pythonweekly0.5https://docs.getunleash.io/reference/sdks/reactweekly0.5https://docs.getunleash.io/reference/sdks/rubyweekly0.5https://docs.getunleash.io/reference/sdks/rustweekly0.5https://docs.getunleash.io/reference/sdks/svelteweekly0.5https://docs.getunleash.io/reference/sdks/vueweekly0.5https://docs.getunleash.io/reference/segmentsweekly0.5https://docs.getunleash.io/reference/service-accountsweekly0.5https://docs.getunleash.io/reference/ssoweekly0.5https://docs.getunleash.io/reference/stickinessweekly0.5https://docs.getunleash.io/reference/strategy-constraintsweekly0.5https://docs.getunleash.io/reference/strategy-variantsweekly0.5https://docs.getunleash.io/reference/tagsweekly0.5https://docs.getunleash.io/reference/technical-debtweekly0.5https://docs.getunleash.io/reference/unleash-contextweekly0.5https://docs.getunleash.io/reference/unleash-edgeweekly0.5https://docs.getunleash.io/reference/unleash-proxyweekly0.5https://docs.getunleash.io/reference/whats-new-v4weekly0.5https://docs.getunleash.io/topicsweekly0.5https://docs.getunleash.io/topics/a-b-testingweekly0.5https://docs.getunleash.io/topics/feature-flag-migration/business-case-feature-flag-migrationweekly0.5https://docs.getunleash.io/topics/feature-flag-migration/feature-flag-migration-best-practicesweekly0.5https://docs.getunleash.io/topics/feature-flag-migration/feature-flag-migration-scopeweekly0.5https://docs.getunleash.io/topics/feature-flag-migration/how-to-execute-feature-flag-migrationweekly0.5https://docs.getunleash.io/topics/feature-flag-migration/onbording-users-to-feature-flag-serviceweekly0.5https://docs.getunleash.io/topics/feature-flag-migration/planning-feature-flag-migrationweekly0.5https://docs.getunleash.io/topics/feature-flags/availability-over-consistencyweekly0.5https://docs.getunleash.io/topics/feature-flags/democratize-feature-flag-accessweekly0.5https://docs.getunleash.io/topics/feature-flags/enable-traceabilityweekly0.5https://docs.getunleash.io/topics/feature-flags/evaluate-flags-close-to-userweekly0.5https://docs.getunleash.io/topics/feature-flags/feature-flag-best-practicesweekly0.5https://docs.getunleash.io/topics/feature-flags/limit-payloadsweekly0.5https://docs.getunleash.io/topics/feature-flags/never-expose-piiweekly0.5https://docs.getunleash.io/topics/feature-flags/prioritize-uxweekly0.5https://docs.getunleash.io/topics/feature-flags/runtime-controlweekly0.5https://docs.getunleash.io/topics/feature-flags/scale-horizontallyweekly0.5https://docs.getunleash.io/topics/feature-flags/short-lived-feature-flagsweekly0.5https://docs.getunleash.io/topics/feature-flags/unique-namesweekly0.5https://docs.getunleash.io/understanding-unleashweekly0.5https://docs.getunleash.io/understanding-unleash/data-collectionweekly0.5https://docs.getunleash.io/understanding-unleash/managing-constraintsweekly0.5https://docs.getunleash.io/understanding-unleash/proxy-hostingweekly0.5https://docs.getunleash.io/understanding-unleash/the-anatomy-of-unleashweekly0.5https://docs.getunleash.io/understanding-unleash/unleash-overviewweekly0.5https://docs.getunleash.io/unleash-academy/advanced-for-devsweekly0.5https://docs.getunleash.io/unleash-academy/foundationalweekly0.5https://docs.getunleash.io/unleash-academy/introductionweekly0.5https://docs.getunleash.io/unleash-academy/managing-unleash-for-devopsweekly0.5https://docs.getunleash.io/using-unleashweekly0.5https://docs.getunleash.io/using-unleash/deployweekly0.5https://docs.getunleash.io/using-unleash/deploy/configuring-unleashweekly0.5https://docs.getunleash.io/using-unleash/deploy/configuring-unleash-v3weekly0.5https://docs.getunleash.io/using-unleash/deploy/database-backupweekly0.5https://docs.getunleash.io/using-unleash/deploy/database-setupweekly0.5https://docs.getunleash.io/using-unleash/deploy/email-serviceweekly0.5https://docs.getunleash.io/using-unleash/deploy/getting-startedweekly0.5https://docs.getunleash.io/using-unleash/deploy/google-auth-hookweekly0.5https://docs.getunleash.io/using-unleash/deploy/google-auth-v3weekly0.5https://docs.getunleash.io/using-unleash/deploy/securing-unleashweekly0.5https://docs.getunleash.io/using-unleash/deploy/securing-unleash-v3weekly0.5https://docs.getunleash.io/using-unleash/deploy/upgrading-unleashweekly0.5https://docs.getunleash.io/using-unleash/troubleshootingweekly0.5https://docs.getunleash.io/using-unleash/troubleshooting/corsweekly0.5https://docs.getunleash.io/using-unleash/troubleshooting/email-serviceweekly0.5https://docs.getunleash.io/using-unleash/troubleshooting/feature-not-availableweekly0.5https://docs.getunleash.io/using-unleash/troubleshooting/flag-exposureweekly0.5https://docs.getunleash.io/using-unleash/troubleshooting/flag-not-returnedweekly0.5 \ No newline at end of file +https://docs.getunleash.io/searchweekly0.5https://docs.getunleash.io/weekly0.5https://docs.getunleash.io/client-specificationweekly0.5https://docs.getunleash.io/contributingweekly0.5https://docs.getunleash.io/contributing/ADRsweekly0.5https://docs.getunleash.io/contributing/ADRs/back-end/breaking-db-changesweekly0.5https://docs.getunleash.io/contributing/ADRs/back-end/namingweekly0.5https://docs.getunleash.io/contributing/ADRs/back-end/POST-PUT-api-payloadweekly0.5https://docs.getunleash.io/contributing/ADRs/back-end/preferred-exportweekly0.5https://docs.getunleash.io/contributing/ADRs/back-end/specificity-db-columnsweekly0.5https://docs.getunleash.io/contributing/ADRs/front-end/component-namingweekly0.5https://docs.getunleash.io/contributing/ADRs/front-end/handling-tablesweekly0.5https://docs.getunleash.io/contributing/ADRs/front-end/interface-namingweekly0.5https://docs.getunleash.io/contributing/ADRs/front-end/preferred-component-props-usageweekly0.5https://docs.getunleash.io/contributing/ADRs/front-end/preferred-data-fetching-methodweekly0.5https://docs.getunleash.io/contributing/ADRs/front-end/preferred-data-mutation-methodweekly0.5https://docs.getunleash.io/contributing/ADRs/front-end/preferred-exportweekly0.5https://docs.getunleash.io/contributing/ADRs/front-end/preferred-folder-structureweekly0.5https://docs.getunleash.io/contributing/ADRs/front-end/preferred-form-architectureweekly0.5https://docs.getunleash.io/contributing/ADRs/front-end/preferred-function-typeweekly0.5https://docs.getunleash.io/contributing/ADRs/front-end/preferred-styles-import-placementweekly0.5https://docs.getunleash.io/contributing/ADRs/front-end/preferred-styling-methodweekly0.5https://docs.getunleash.io/contributing/ADRs/front-end/sdk-generatorweekly0.5https://docs.getunleash.io/contributing/ADRs/overarching/domain-languageweekly0.5https://docs.getunleash.io/contributing/ADRs/overarching/separation-request-response-schemasweekly0.5https://docs.getunleash.io/contributing/backend/overviewweekly0.5https://docs.getunleash.io/contributing/developer-guideweekly0.5https://docs.getunleash.io/contributing/frontend/overviewweekly0.5https://docs.getunleash.io/feature-flag-tutorialsweekly0.5https://docs.getunleash.io/feature-flag-tutorials/flutter/a-b-testingweekly0.5https://docs.getunleash.io/feature-flag-tutorials/nextjs/implementing-feature-flagsweekly0.5https://docs.getunleash.io/feature-flag-tutorials/reactweekly0.5https://docs.getunleash.io/feature-flag-tutorials/react/examplesweekly0.5https://docs.getunleash.io/how-toweekly0.5https://docs.getunleash.io/how-to/apiweekly0.5https://docs.getunleash.io/how-to/envweekly0.5https://docs.getunleash.io/how-to/how-to-add-feature-flag-naming-patternsweekly0.5https://docs.getunleash.io/how-to/how-to-add-sso-azure-samlweekly0.5https://docs.getunleash.io/how-to/how-to-add-sso-googleweekly0.5https://docs.getunleash.io/how-to/how-to-add-sso-open-id-connectweekly0.5https://docs.getunleash.io/how-to/how-to-add-sso-samlweekly0.5https://docs.getunleash.io/how-to/how-to-add-sso-saml-keycloakweekly0.5https://docs.getunleash.io/how-to/how-to-add-strategy-constraintsweekly0.5https://docs.getunleash.io/how-to/how-to-add-users-to-unleashweekly0.5https://docs.getunleash.io/how-to/how-to-capture-impression-dataweekly0.5https://docs.getunleash.io/how-to/how-to-clone-environmentsweekly0.5https://docs.getunleash.io/how-to/how-to-create-and-assign-custom-project-rolesweekly0.5https://docs.getunleash.io/how-to/how-to-create-and-assign-custom-root-rolesweekly0.5https://docs.getunleash.io/how-to/how-to-create-and-display-bannersweekly0.5https://docs.getunleash.io/how-to/how-to-create-and-manage-user-groupsweekly0.5https://docs.getunleash.io/how-to/how-to-create-api-tokensweekly0.5https://docs.getunleash.io/how-to/how-to-create-feature-togglesweekly0.5https://docs.getunleash.io/how-to/how-to-create-personal-access-tokensweekly0.5https://docs.getunleash.io/how-to/how-to-create-project-api-tokensweekly0.5https://docs.getunleash.io/how-to/how-to-create-service-accountsweekly0.5https://docs.getunleash.io/how-to/how-to-define-custom-context-fieldsweekly0.5https://docs.getunleash.io/how-to/how-to-download-login-historyweekly0.5https://docs.getunleash.io/how-to/how-to-enable-openapiweekly0.5https://docs.getunleash.io/how-to/how-to-environment-import-exportweekly0.5https://docs.getunleash.io/how-to/how-to-import-exportweekly0.5https://docs.getunleash.io/how-to/how-to-manage-public-invite-tokensweekly0.5https://docs.getunleash.io/how-to/how-to-run-the-unleash-proxyweekly0.5https://docs.getunleash.io/how-to/how-to-schedule-feature-releasesweekly0.5https://docs.getunleash.io/how-to/how-to-send-feature-updates-to-slack-deprecatedweekly0.5https://docs.getunleash.io/how-to/how-to-set-up-group-sso-syncweekly0.5https://docs.getunleash.io/how-to/how-to-setup-sso-keycloak-group-syncweekly0.5https://docs.getunleash.io/how-to/how-to-synchronize-unleash-instancesweekly0.5https://docs.getunleash.io/how-to/how-to-use-custom-strategiesweekly0.5https://docs.getunleash.io/how-to/how-to-use-the-admin-apiweekly0.5https://docs.getunleash.io/how-to/miscweekly0.5https://docs.getunleash.io/how-to/proxyweekly0.5https://docs.getunleash.io/how-to/ssoweekly0.5https://docs.getunleash.io/how-to/users-and-permissionsweekly0.5https://docs.getunleash.io/quickstartweekly0.5https://docs.getunleash.io/referenceweekly0.5https://docs.getunleash.io/reference/activation-strategiesweekly0.5https://docs.getunleash.io/reference/api-tokens-and-client-keysweekly0.5https://docs.getunleash.io/reference/api/legacy/unleashweekly0.5https://docs.getunleash.io/reference/api/legacy/unleash/admin/addonsweekly0.5https://docs.getunleash.io/reference/api/legacy/unleash/admin/archiveweekly0.5https://docs.getunleash.io/reference/api/legacy/unleash/admin/contextweekly0.5https://docs.getunleash.io/reference/api/legacy/unleash/admin/eventsweekly0.5https://docs.getunleash.io/reference/api/legacy/unleash/admin/feature-typesweekly0.5https://docs.getunleash.io/reference/api/legacy/unleash/admin/featuresweekly0.5https://docs.getunleash.io/reference/api/legacy/unleash/admin/features-v2weekly0.5https://docs.getunleash.io/reference/api/legacy/unleash/admin/metricsweekly0.5https://docs.getunleash.io/reference/api/legacy/unleash/admin/projectsweekly0.5https://docs.getunleash.io/reference/api/legacy/unleash/admin/segmentsweekly0.5https://docs.getunleash.io/reference/api/legacy/unleash/admin/stateweekly0.5https://docs.getunleash.io/reference/api/legacy/unleash/admin/strategiesweekly0.5https://docs.getunleash.io/reference/api/legacy/unleash/admin/tagsweekly0.5https://docs.getunleash.io/reference/api/legacy/unleash/admin/user-adminweekly0.5https://docs.getunleash.io/reference/api/legacy/unleash/basic-authweekly0.5https://docs.getunleash.io/reference/api/legacy/unleash/client/featuresweekly0.5https://docs.getunleash.io/reference/api/legacy/unleash/client/metricsweekly0.5https://docs.getunleash.io/reference/api/legacy/unleash/client/registerweekly0.5https://docs.getunleash.io/reference/api/legacy/unleash/internal/healthweekly0.5https://docs.getunleash.io/reference/api/legacy/unleash/internal/prometheusweekly0.5https://docs.getunleash.io/reference/api/unleashweekly0.5https://docs.getunleash.io/reference/api/unleash/add-access-to-projectweekly0.5https://docs.getunleash.io/reference/api/unleash/add-change-request-commentweekly0.5https://docs.getunleash.io/reference/api/unleash/add-default-strategy-to-project-environmentweekly0.5https://docs.getunleash.io/reference/api/unleash/add-environment-to-projectweekly0.5https://docs.getunleash.io/reference/api/unleash/add-favorite-featureweekly0.5https://docs.getunleash.io/reference/api/unleash/add-favorite-projectweekly0.5https://docs.getunleash.io/reference/api/unleash/add-feature-dependencyweekly0.5https://docs.getunleash.io/reference/api/unleash/add-feature-strategyweekly0.5https://docs.getunleash.io/reference/api/unleash/add-public-signup-token-userweekly0.5https://docs.getunleash.io/reference/api/unleash/add-role-access-to-projectweekly0.5https://docs.getunleash.io/reference/api/unleash/add-role-to-userweekly0.5https://docs.getunleash.io/reference/api/unleash/add-tagweekly0.5https://docs.getunleash.io/reference/api/unleash/add-tag-to-featuresweekly0.5https://docs.getunleash.io/reference/api/unleash/addonsweekly0.5https://docs.getunleash.io/reference/api/unleash/admin-uiweekly0.5https://docs.getunleash.io/reference/api/unleash/api-tokensweekly0.5https://docs.getunleash.io/reference/api/unleash/archiveweekly0.5https://docs.getunleash.io/reference/api/unleash/archive-featureweekly0.5https://docs.getunleash.io/reference/api/unleash/archive-featuresweekly0.5https://docs.getunleash.io/reference/api/unleash/authweekly0.5https://docs.getunleash.io/reference/api/unleash/bannersweekly0.5https://docs.getunleash.io/reference/api/unleash/bulk-metricsweekly0.5https://docs.getunleash.io/reference/api/unleash/bulk-toggle-features-environment-offweekly0.5https://docs.getunleash.io/reference/api/unleash/bulk-toggle-features-environment-onweekly0.5https://docs.getunleash.io/reference/api/unleash/change-my-passwordweekly0.5https://docs.getunleash.io/reference/api/unleash/change-passwordweekly0.5https://docs.getunleash.io/reference/api/unleash/change-projectweekly0.5https://docs.getunleash.io/reference/api/unleash/change-requestweekly0.5https://docs.getunleash.io/reference/api/unleash/change-requestsweekly0.5https://docs.getunleash.io/reference/api/unleash/change-role-for-groupweekly0.5https://docs.getunleash.io/reference/api/unleash/change-role-for-userweekly0.5https://docs.getunleash.io/reference/api/unleash/change-user-passwordweekly0.5https://docs.getunleash.io/reference/api/unleash/check-dependencies-existweekly0.5https://docs.getunleash.io/reference/api/unleash/check-licenseweekly0.5https://docs.getunleash.io/reference/api/unleash/clientweekly0.5https://docs.getunleash.io/reference/api/unleash/clone-environmentweekly0.5https://docs.getunleash.io/reference/api/unleash/clone-featureweekly0.5https://docs.getunleash.io/reference/api/unleash/contextweekly0.5https://docs.getunleash.io/reference/api/unleash/create-addonweekly0.5https://docs.getunleash.io/reference/api/unleash/create-api-tokenweekly0.5https://docs.getunleash.io/reference/api/unleash/create-applicationweekly0.5https://docs.getunleash.io/reference/api/unleash/create-bannerweekly0.5https://docs.getunleash.io/reference/api/unleash/create-context-fieldweekly0.5https://docs.getunleash.io/reference/api/unleash/create-environmentweekly0.5https://docs.getunleash.io/reference/api/unleash/create-featureweekly0.5https://docs.getunleash.io/reference/api/unleash/create-feedbackweekly0.5https://docs.getunleash.io/reference/api/unleash/create-groupweekly0.5https://docs.getunleash.io/reference/api/unleash/create-patweekly0.5https://docs.getunleash.io/reference/api/unleash/create-projectweekly0.5https://docs.getunleash.io/reference/api/unleash/create-project-api-tokenweekly0.5https://docs.getunleash.io/reference/api/unleash/create-public-signup-tokenweekly0.5https://docs.getunleash.io/reference/api/unleash/create-roleweekly0.5https://docs.getunleash.io/reference/api/unleash/create-segmentweekly0.5https://docs.getunleash.io/reference/api/unleash/create-service-accountweekly0.5https://docs.getunleash.io/reference/api/unleash/create-service-account-tokenweekly0.5https://docs.getunleash.io/reference/api/unleash/create-strategyweekly0.5https://docs.getunleash.io/reference/api/unleash/create-tagweekly0.5https://docs.getunleash.io/reference/api/unleash/create-tag-typeweekly0.5https://docs.getunleash.io/reference/api/unleash/create-userweekly0.5https://docs.getunleash.io/reference/api/unleash/delete-addonweekly0.5https://docs.getunleash.io/reference/api/unleash/delete-api-tokenweekly0.5https://docs.getunleash.io/reference/api/unleash/delete-applicationweekly0.5https://docs.getunleash.io/reference/api/unleash/delete-bannerweekly0.5https://docs.getunleash.io/reference/api/unleash/delete-changeweekly0.5https://docs.getunleash.io/reference/api/unleash/delete-change-requestweekly0.5https://docs.getunleash.io/reference/api/unleash/delete-context-fieldweekly0.5https://docs.getunleash.io/reference/api/unleash/delete-featureweekly0.5https://docs.getunleash.io/reference/api/unleash/delete-feature-dependenciesweekly0.5https://docs.getunleash.io/reference/api/unleash/delete-feature-dependencyweekly0.5https://docs.getunleash.io/reference/api/unleash/delete-feature-strategyweekly0.5https://docs.getunleash.io/reference/api/unleash/delete-featuresweekly0.5https://docs.getunleash.io/reference/api/unleash/delete-groupweekly0.5https://docs.getunleash.io/reference/api/unleash/delete-patweekly0.5https://docs.getunleash.io/reference/api/unleash/delete-projectweekly0.5https://docs.getunleash.io/reference/api/unleash/delete-project-api-tokenweekly0.5https://docs.getunleash.io/reference/api/unleash/delete-roleweekly0.5https://docs.getunleash.io/reference/api/unleash/delete-service-accountweekly0.5https://docs.getunleash.io/reference/api/unleash/delete-service-account-tokenweekly0.5https://docs.getunleash.io/reference/api/unleash/delete-tagweekly0.5https://docs.getunleash.io/reference/api/unleash/delete-tag-typeweekly0.5https://docs.getunleash.io/reference/api/unleash/delete-userweekly0.5https://docs.getunleash.io/reference/api/unleash/dependenciesweekly0.5https://docs.getunleash.io/reference/api/unleash/deprecate-strategyweekly0.5https://docs.getunleash.io/reference/api/unleash/disable-bannerweekly0.5https://docs.getunleash.io/reference/api/unleash/edgeweekly0.5https://docs.getunleash.io/reference/api/unleash/edit-changeweekly0.5https://docs.getunleash.io/reference/api/unleash/enable-bannerweekly0.5https://docs.getunleash.io/reference/api/unleash/environmentsweekly0.5https://docs.getunleash.io/reference/api/unleash/eventsweekly0.5https://docs.getunleash.io/reference/api/unleash/exportweekly0.5https://docs.getunleash.io/reference/api/unleash/export-featuresweekly0.5https://docs.getunleash.io/reference/api/unleash/feature-typesweekly0.5https://docs.getunleash.io/reference/api/unleash/featuresweekly0.5https://docs.getunleash.io/reference/api/unleash/frontend-apiweekly0.5https://docs.getunleash.io/reference/api/unleash/get-access-overviewweekly0.5https://docs.getunleash.io/reference/api/unleash/get-addonweekly0.5https://docs.getunleash.io/reference/api/unleash/get-addonsweekly0.5https://docs.getunleash.io/reference/api/unleash/get-admin-countweekly0.5https://docs.getunleash.io/reference/api/unleash/get-advanced-playgroundweekly0.5https://docs.getunleash.io/reference/api/unleash/get-all-api-tokensweekly0.5https://docs.getunleash.io/reference/api/unleash/get-all-client-featuresweekly0.5https://docs.getunleash.io/reference/api/unleash/get-all-environmentsweekly0.5https://docs.getunleash.io/reference/api/unleash/get-all-feature-typesweekly0.5https://docs.getunleash.io/reference/api/unleash/get-all-public-signup-tokensweekly0.5https://docs.getunleash.io/reference/api/unleash/get-all-strategiesweekly0.5https://docs.getunleash.io/reference/api/unleash/get-all-togglesweekly0.5https://docs.getunleash.io/reference/api/unleash/get-api-tokens-by-nameweekly0.5https://docs.getunleash.io/reference/api/unleash/get-applicationweekly0.5https://docs.getunleash.io/reference/api/unleash/get-applicationsweekly0.5https://docs.getunleash.io/reference/api/unleash/get-archived-featuresweekly0.5https://docs.getunleash.io/reference/api/unleash/get-archived-features-by-project-idweekly0.5https://docs.getunleash.io/reference/api/unleash/get-bannersweekly0.5https://docs.getunleash.io/reference/api/unleash/get-base-users-and-groupsweekly0.5https://docs.getunleash.io/reference/api/unleash/get-change-requestweekly0.5https://docs.getunleash.io/reference/api/unleash/get-change-requests-for-projectweekly0.5https://docs.getunleash.io/reference/api/unleash/get-client-featureweekly0.5https://docs.getunleash.io/reference/api/unleash/get-context-fieldweekly0.5https://docs.getunleash.io/reference/api/unleash/get-context-fieldsweekly0.5https://docs.getunleash.io/reference/api/unleash/get-deprecated-project-overviewweekly0.5https://docs.getunleash.io/reference/api/unleash/get-environmentweekly0.5https://docs.getunleash.io/reference/api/unleash/get-environment-feature-variantsweekly0.5https://docs.getunleash.io/reference/api/unleash/get-eventsweekly0.5https://docs.getunleash.io/reference/api/unleash/get-events-for-toggleweekly0.5https://docs.getunleash.io/reference/api/unleash/get-featureweekly0.5https://docs.getunleash.io/reference/api/unleash/get-feature-environmentweekly0.5https://docs.getunleash.io/reference/api/unleash/get-feature-strategiesweekly0.5https://docs.getunleash.io/reference/api/unleash/get-feature-strategyweekly0.5https://docs.getunleash.io/reference/api/unleash/get-feature-usage-summaryweekly0.5https://docs.getunleash.io/reference/api/unleash/get-feature-variantsweekly0.5https://docs.getunleash.io/reference/api/unleash/get-featuresweekly0.5https://docs.getunleash.io/reference/api/unleash/get-feedbackweekly0.5https://docs.getunleash.io/reference/api/unleash/get-frontend-featuresweekly0.5https://docs.getunleash.io/reference/api/unleash/get-google-settingsweekly0.5https://docs.getunleash.io/reference/api/unleash/get-groupweekly0.5https://docs.getunleash.io/reference/api/unleash/get-groupsweekly0.5https://docs.getunleash.io/reference/api/unleash/get-healthweekly0.5https://docs.getunleash.io/reference/api/unleash/get-instance-admin-statsweekly0.5https://docs.getunleash.io/reference/api/unleash/get-instance-admin-stats-csvweekly0.5https://docs.getunleash.io/reference/api/unleash/get-login-historyweekly0.5https://docs.getunleash.io/reference/api/unleash/get-maintenanceweekly0.5https://docs.getunleash.io/reference/api/unleash/get-meweekly0.5https://docs.getunleash.io/reference/api/unleash/get-notificationsweekly0.5https://docs.getunleash.io/reference/api/unleash/get-oidc-settingsweekly0.5https://docs.getunleash.io/reference/api/unleash/get-open-change-requests-for-userweekly0.5https://docs.getunleash.io/reference/api/unleash/get-patsweekly0.5https://docs.getunleash.io/reference/api/unleash/get-pending-change-requests-for-featureweekly0.5https://docs.getunleash.io/reference/api/unleash/get-pending-change-requests-for-userweekly0.5https://docs.getunleash.io/reference/api/unleash/get-permissionsweekly0.5https://docs.getunleash.io/reference/api/unleash/get-playgroundweekly0.5https://docs.getunleash.io/reference/api/unleash/get-profileweekly0.5https://docs.getunleash.io/reference/api/unleash/get-project-accessweekly0.5https://docs.getunleash.io/reference/api/unleash/get-project-api-tokensweekly0.5https://docs.getunleash.io/reference/api/unleash/get-project-change-request-configweekly0.5https://docs.getunleash.io/reference/api/unleash/get-project-doraweekly0.5https://docs.getunleash.io/reference/api/unleash/get-project-environmentsweekly0.5https://docs.getunleash.io/reference/api/unleash/get-project-health-reportweekly0.5https://docs.getunleash.io/reference/api/unleash/get-project-overviewweekly0.5https://docs.getunleash.io/reference/api/unleash/get-project-usersweekly0.5https://docs.getunleash.io/reference/api/unleash/get-projectsweekly0.5https://docs.getunleash.io/reference/api/unleash/get-public-signup-tokenweekly0.5https://docs.getunleash.io/reference/api/unleash/get-raw-feature-metricsweekly0.5https://docs.getunleash.io/reference/api/unleash/get-requests-per-secondweekly0.5https://docs.getunleash.io/reference/api/unleash/get-role-by-idweekly0.5https://docs.getunleash.io/reference/api/unleash/get-role-project-accessweekly0.5https://docs.getunleash.io/reference/api/unleash/get-rolesweekly0.5https://docs.getunleash.io/reference/api/unleash/get-saml-settingsweekly0.5https://docs.getunleash.io/reference/api/unleash/get-scheduled-change-requestsweekly0.5https://docs.getunleash.io/reference/api/unleash/get-segmentweekly0.5https://docs.getunleash.io/reference/api/unleash/get-segmentsweekly0.5https://docs.getunleash.io/reference/api/unleash/get-segments-by-strategy-idweekly0.5https://docs.getunleash.io/reference/api/unleash/get-service-account-tokensweekly0.5https://docs.getunleash.io/reference/api/unleash/get-service-accountsweekly0.5https://docs.getunleash.io/reference/api/unleash/get-simple-settingsweekly0.5https://docs.getunleash.io/reference/api/unleash/get-strategies-by-context-fieldweekly0.5https://docs.getunleash.io/reference/api/unleash/get-strategies-by-segment-idweekly0.5https://docs.getunleash.io/reference/api/unleash/get-strategyweekly0.5https://docs.getunleash.io/reference/api/unleash/get-tagweekly0.5https://docs.getunleash.io/reference/api/unleash/get-tag-typeweekly0.5https://docs.getunleash.io/reference/api/unleash/get-tag-typesweekly0.5https://docs.getunleash.io/reference/api/unleash/get-tagsweekly0.5https://docs.getunleash.io/reference/api/unleash/get-tags-by-typeweekly0.5https://docs.getunleash.io/reference/api/unleash/get-telemetry-settingsweekly0.5https://docs.getunleash.io/reference/api/unleash/get-ui-configweekly0.5https://docs.getunleash.io/reference/api/unleash/get-userweekly0.5https://docs.getunleash.io/reference/api/unleash/get-usersweekly0.5https://docs.getunleash.io/reference/api/unleash/get-valid-tokensweekly0.5https://docs.getunleash.io/reference/api/unleash/importweekly0.5https://docs.getunleash.io/reference/api/unleash/import-exportweekly0.5https://docs.getunleash.io/reference/api/unleash/import-togglesweekly0.5https://docs.getunleash.io/reference/api/unleash/instance-adminweekly0.5https://docs.getunleash.io/reference/api/unleash/list-parent-optionsweekly0.5https://docs.getunleash.io/reference/api/unleash/list-tagsweekly0.5https://docs.getunleash.io/reference/api/unleash/loginweekly0.5https://docs.getunleash.io/reference/api/unleash/maintenanceweekly0.5https://docs.getunleash.io/reference/api/unleash/mark-notifications-as-readweekly0.5https://docs.getunleash.io/reference/api/unleash/metricsweekly0.5https://docs.getunleash.io/reference/api/unleash/notificationsweekly0.5https://docs.getunleash.io/reference/api/unleash/operationalweekly0.5https://docs.getunleash.io/reference/api/unleash/overwrite-environment-feature-variantsweekly0.5https://docs.getunleash.io/reference/api/unleash/overwrite-feature-variantsweekly0.5https://docs.getunleash.io/reference/api/unleash/overwrite-feature-variants-on-environmentsweekly0.5https://docs.getunleash.io/reference/api/unleash/patch-environments-feature-variantsweekly0.5https://docs.getunleash.io/reference/api/unleash/patch-featureweekly0.5https://docs.getunleash.io/reference/api/unleash/patch-feature-strategyweekly0.5https://docs.getunleash.io/reference/api/unleash/patch-feature-variantsweekly0.5https://docs.getunleash.io/reference/api/unleash/personal-access-tokensweekly0.5https://docs.getunleash.io/reference/api/unleash/playgroundweekly0.5https://docs.getunleash.io/reference/api/unleash/projectsweekly0.5https://docs.getunleash.io/reference/api/unleash/provide-feedbackweekly0.5https://docs.getunleash.io/reference/api/unleash/public-signup-tokensweekly0.5https://docs.getunleash.io/reference/api/unleash/reactivate-strategyweekly0.5https://docs.getunleash.io/reference/api/unleash/read-licenseweekly0.5https://docs.getunleash.io/reference/api/unleash/register-client-applicationweekly0.5https://docs.getunleash.io/reference/api/unleash/register-client-metricsweekly0.5https://docs.getunleash.io/reference/api/unleash/register-frontend-clientweekly0.5https://docs.getunleash.io/reference/api/unleash/register-frontend-metricsweekly0.5https://docs.getunleash.io/reference/api/unleash/remove-environmentweekly0.5https://docs.getunleash.io/reference/api/unleash/remove-environment-from-projectweekly0.5https://docs.getunleash.io/reference/api/unleash/remove-favorite-featureweekly0.5https://docs.getunleash.io/reference/api/unleash/remove-favorite-projectweekly0.5https://docs.getunleash.io/reference/api/unleash/remove-group-accessweekly0.5https://docs.getunleash.io/reference/api/unleash/remove-role-for-userweekly0.5https://docs.getunleash.io/reference/api/unleash/remove-role-from-groupweekly0.5https://docs.getunleash.io/reference/api/unleash/remove-segmentweekly0.5https://docs.getunleash.io/reference/api/unleash/remove-strategyweekly0.5https://docs.getunleash.io/reference/api/unleash/remove-tagweekly0.5https://docs.getunleash.io/reference/api/unleash/remove-user-accessweekly0.5https://docs.getunleash.io/reference/api/unleash/reset-user-passwordweekly0.5https://docs.getunleash.io/reference/api/unleash/revive-featureweekly0.5https://docs.getunleash.io/reference/api/unleash/revive-featuresweekly0.5https://docs.getunleash.io/reference/api/unleash/searchweekly0.5https://docs.getunleash.io/reference/api/unleash/search-eventsweekly0.5https://docs.getunleash.io/reference/api/unleash/search-featuresweekly0.5https://docs.getunleash.io/reference/api/unleash/search-usersweekly0.5https://docs.getunleash.io/reference/api/unleash/segmentsweekly0.5https://docs.getunleash.io/reference/api/unleash/send-reset-password-emailweekly0.5https://docs.getunleash.io/reference/api/unleash/service-accountsweekly0.5https://docs.getunleash.io/reference/api/unleash/set-google-settingsweekly0.5https://docs.getunleash.io/reference/api/unleash/set-oidc-settingsweekly0.5https://docs.getunleash.io/reference/api/unleash/set-project-accessweekly0.5https://docs.getunleash.io/reference/api/unleash/set-roles-for-groupweekly0.5https://docs.getunleash.io/reference/api/unleash/set-roles-for-userweekly0.5https://docs.getunleash.io/reference/api/unleash/set-saml-settingsweekly0.5https://docs.getunleash.io/reference/api/unleash/set-simple-settingsweekly0.5https://docs.getunleash.io/reference/api/unleash/set-strategy-sort-orderweekly0.5https://docs.getunleash.io/reference/api/unleash/set-ui-configweekly0.5https://docs.getunleash.io/reference/api/unleash/stale-featuresweekly0.5https://docs.getunleash.io/reference/api/unleash/strategiesweekly0.5https://docs.getunleash.io/reference/api/unleash/tagsweekly0.5https://docs.getunleash.io/reference/api/unleash/telemetryweekly0.5https://docs.getunleash.io/reference/api/unleash/toggle-environment-offweekly0.5https://docs.getunleash.io/reference/api/unleash/toggle-environment-onweekly0.5https://docs.getunleash.io/reference/api/unleash/toggle-feature-environment-offweekly0.5https://docs.getunleash.io/reference/api/unleash/toggle-feature-environment-onweekly0.5https://docs.getunleash.io/reference/api/unleash/toggle-maintenanceweekly0.5https://docs.getunleash.io/reference/api/unleash/unleash-apiweekly0.5https://docs.getunleash.io/reference/api/unleash/unstableweekly0.5https://docs.getunleash.io/reference/api/unleash/update-addonweekly0.5https://docs.getunleash.io/reference/api/unleash/update-api-tokenweekly0.5https://docs.getunleash.io/reference/api/unleash/update-bannerweekly0.5https://docs.getunleash.io/reference/api/unleash/update-change-request-stateweekly0.5https://docs.getunleash.io/reference/api/unleash/update-change-request-titleweekly0.5https://docs.getunleash.io/reference/api/unleash/update-context-fieldweekly0.5https://docs.getunleash.io/reference/api/unleash/update-environmentweekly0.5https://docs.getunleash.io/reference/api/unleash/update-featureweekly0.5https://docs.getunleash.io/reference/api/unleash/update-feature-strategyweekly0.5https://docs.getunleash.io/reference/api/unleash/update-feature-strategy-segmentsweekly0.5https://docs.getunleash.io/reference/api/unleash/update-feature-type-lifetimeweekly0.5https://docs.getunleash.io/reference/api/unleash/update-feedbackweekly0.5https://docs.getunleash.io/reference/api/unleash/update-groupweekly0.5https://docs.getunleash.io/reference/api/unleash/update-licenseweekly0.5https://docs.getunleash.io/reference/api/unleash/update-projectweekly0.5https://docs.getunleash.io/reference/api/unleash/update-project-change-request-configweekly0.5https://docs.getunleash.io/reference/api/unleash/update-project-enterprise-settingsweekly0.5https://docs.getunleash.io/reference/api/unleash/update-public-signup-tokenweekly0.5https://docs.getunleash.io/reference/api/unleash/update-roleweekly0.5https://docs.getunleash.io/reference/api/unleash/update-segmentweekly0.5https://docs.getunleash.io/reference/api/unleash/update-service-accountweekly0.5https://docs.getunleash.io/reference/api/unleash/update-sort-orderweekly0.5https://docs.getunleash.io/reference/api/unleash/update-splash-settingsweekly0.5https://docs.getunleash.io/reference/api/unleash/update-strategyweekly0.5https://docs.getunleash.io/reference/api/unleash/update-tag-typeweekly0.5https://docs.getunleash.io/reference/api/unleash/update-tagsweekly0.5https://docs.getunleash.io/reference/api/unleash/update-userweekly0.5https://docs.getunleash.io/reference/api/unleash/usersweekly0.5https://docs.getunleash.io/reference/api/unleash/validateweekly0.5https://docs.getunleash.io/reference/api/unleash/validate-archive-featuresweekly0.5https://docs.getunleash.io/reference/api/unleash/validate-constraintweekly0.5https://docs.getunleash.io/reference/api/unleash/validate-environment-nameweekly0.5https://docs.getunleash.io/reference/api/unleash/validate-featureweekly0.5https://docs.getunleash.io/reference/api/unleash/validate-importweekly0.5https://docs.getunleash.io/reference/api/unleash/validate-passwordweekly0.5https://docs.getunleash.io/reference/api/unleash/validate-projectweekly0.5https://docs.getunleash.io/reference/api/unleash/validate-public-signup-tokenweekly0.5https://docs.getunleash.io/reference/api/unleash/validate-roleweekly0.5https://docs.getunleash.io/reference/api/unleash/validate-segmentweekly0.5https://docs.getunleash.io/reference/api/unleash/validate-tag-typeweekly0.5https://docs.getunleash.io/reference/api/unleash/validate-tokenweekly0.5https://docs.getunleash.io/reference/api/unleash/validate-user-passwordweekly0.5https://docs.getunleash.io/reference/archived-togglesweekly0.5https://docs.getunleash.io/reference/bannersweekly0.5https://docs.getunleash.io/reference/change-requestsweekly0.5https://docs.getunleash.io/reference/custom-activation-strategiesweekly0.5https://docs.getunleash.io/reference/dependent-featuresweekly0.5https://docs.getunleash.io/reference/environmentsweekly0.5https://docs.getunleash.io/reference/event-logweekly0.5https://docs.getunleash.io/reference/event-typesweekly0.5https://docs.getunleash.io/reference/feature-flag-naming-patternsweekly0.5https://docs.getunleash.io/reference/feature-toggle-typesweekly0.5https://docs.getunleash.io/reference/feature-toggle-variantsweekly0.5https://docs.getunleash.io/reference/feature-togglesweekly0.5https://docs.getunleash.io/reference/front-end-apiweekly0.5https://docs.getunleash.io/reference/impression-dataweekly0.5https://docs.getunleash.io/reference/integrationsweekly0.5https://docs.getunleash.io/reference/integrations/datadogweekly0.5https://docs.getunleash.io/reference/integrations/jira-cloud-plugin-installationweekly0.5https://docs.getunleash.io/reference/integrations/jira-cloud-plugin-usageweekly0.5https://docs.getunleash.io/reference/integrations/jira-server-plugin-installationweekly0.5https://docs.getunleash.io/reference/integrations/jira-server-plugin-usageweekly0.5https://docs.getunleash.io/reference/integrations/slackweekly0.5https://docs.getunleash.io/reference/integrations/slack-appweekly0.5https://docs.getunleash.io/reference/integrations/teamsweekly0.5https://docs.getunleash.io/reference/integrations/webhookweekly0.5https://docs.getunleash.io/reference/login-historyweekly0.5https://docs.getunleash.io/reference/maintenance-modeweekly0.5https://docs.getunleash.io/reference/network-viewweekly0.5https://docs.getunleash.io/reference/notificationsweekly0.5https://docs.getunleash.io/reference/playgroundweekly0.5https://docs.getunleash.io/reference/project-collaboration-modeweekly0.5https://docs.getunleash.io/reference/projectsweekly0.5https://docs.getunleash.io/reference/public-signupweekly0.5https://docs.getunleash.io/reference/rbacweekly0.5https://docs.getunleash.io/reference/sdksweekly0.5https://docs.getunleash.io/reference/sdks/android-proxyweekly0.5https://docs.getunleash.io/reference/sdks/dotnetweekly0.5https://docs.getunleash.io/reference/sdks/flutterweekly0.5https://docs.getunleash.io/reference/sdks/goweekly0.5https://docs.getunleash.io/reference/sdks/ios-proxyweekly0.5https://docs.getunleash.io/reference/sdks/javaweekly0.5https://docs.getunleash.io/reference/sdks/javascript-browserweekly0.5https://docs.getunleash.io/reference/sdks/next-jsweekly0.5https://docs.getunleash.io/reference/sdks/nodeweekly0.5https://docs.getunleash.io/reference/sdks/phpweekly0.5https://docs.getunleash.io/reference/sdks/pythonweekly0.5https://docs.getunleash.io/reference/sdks/reactweekly0.5https://docs.getunleash.io/reference/sdks/rubyweekly0.5https://docs.getunleash.io/reference/sdks/rustweekly0.5https://docs.getunleash.io/reference/sdks/svelteweekly0.5https://docs.getunleash.io/reference/sdks/vueweekly0.5https://docs.getunleash.io/reference/segmentsweekly0.5https://docs.getunleash.io/reference/service-accountsweekly0.5https://docs.getunleash.io/reference/ssoweekly0.5https://docs.getunleash.io/reference/stickinessweekly0.5https://docs.getunleash.io/reference/strategy-constraintsweekly0.5https://docs.getunleash.io/reference/strategy-variantsweekly0.5https://docs.getunleash.io/reference/tagsweekly0.5https://docs.getunleash.io/reference/technical-debtweekly0.5https://docs.getunleash.io/reference/unleash-contextweekly0.5https://docs.getunleash.io/reference/unleash-edgeweekly0.5https://docs.getunleash.io/reference/unleash-proxyweekly0.5https://docs.getunleash.io/reference/whats-new-v4weekly0.5https://docs.getunleash.io/topicsweekly0.5https://docs.getunleash.io/topics/a-b-testingweekly0.5https://docs.getunleash.io/topics/feature-flag-migration/business-case-feature-flag-migrationweekly0.5https://docs.getunleash.io/topics/feature-flag-migration/feature-flag-migration-best-practicesweekly0.5https://docs.getunleash.io/topics/feature-flag-migration/feature-flag-migration-scopeweekly0.5https://docs.getunleash.io/topics/feature-flag-migration/how-to-execute-feature-flag-migrationweekly0.5https://docs.getunleash.io/topics/feature-flag-migration/onbording-users-to-feature-flag-serviceweekly0.5https://docs.getunleash.io/topics/feature-flag-migration/planning-feature-flag-migrationweekly0.5https://docs.getunleash.io/topics/feature-flags/availability-over-consistencyweekly0.5https://docs.getunleash.io/topics/feature-flags/democratize-feature-flag-accessweekly0.5https://docs.getunleash.io/topics/feature-flags/enable-traceabilityweekly0.5https://docs.getunleash.io/topics/feature-flags/evaluate-flags-close-to-userweekly0.5https://docs.getunleash.io/topics/feature-flags/feature-flag-best-practicesweekly0.5https://docs.getunleash.io/topics/feature-flags/limit-payloadsweekly0.5https://docs.getunleash.io/topics/feature-flags/never-expose-piiweekly0.5https://docs.getunleash.io/topics/feature-flags/prioritize-uxweekly0.5https://docs.getunleash.io/topics/feature-flags/runtime-controlweekly0.5https://docs.getunleash.io/topics/feature-flags/scale-horizontallyweekly0.5https://docs.getunleash.io/topics/feature-flags/short-lived-feature-flagsweekly0.5https://docs.getunleash.io/topics/feature-flags/unique-namesweekly0.5https://docs.getunleash.io/understanding-unleashweekly0.5https://docs.getunleash.io/understanding-unleash/data-collectionweekly0.5https://docs.getunleash.io/understanding-unleash/managing-constraintsweekly0.5https://docs.getunleash.io/understanding-unleash/proxy-hostingweekly0.5https://docs.getunleash.io/understanding-unleash/the-anatomy-of-unleashweekly0.5https://docs.getunleash.io/understanding-unleash/unleash-overviewweekly0.5https://docs.getunleash.io/unleash-academy/advanced-for-devsweekly0.5https://docs.getunleash.io/unleash-academy/foundationalweekly0.5https://docs.getunleash.io/unleash-academy/introductionweekly0.5https://docs.getunleash.io/unleash-academy/managing-unleash-for-devopsweekly0.5https://docs.getunleash.io/using-unleashweekly0.5https://docs.getunleash.io/using-unleash/deployweekly0.5https://docs.getunleash.io/using-unleash/deploy/configuring-unleashweekly0.5https://docs.getunleash.io/using-unleash/deploy/configuring-unleash-v3weekly0.5https://docs.getunleash.io/using-unleash/deploy/database-backupweekly0.5https://docs.getunleash.io/using-unleash/deploy/database-setupweekly0.5https://docs.getunleash.io/using-unleash/deploy/email-serviceweekly0.5https://docs.getunleash.io/using-unleash/deploy/getting-startedweekly0.5https://docs.getunleash.io/using-unleash/deploy/google-auth-hookweekly0.5https://docs.getunleash.io/using-unleash/deploy/google-auth-v3weekly0.5https://docs.getunleash.io/using-unleash/deploy/license-keysweekly0.5https://docs.getunleash.io/using-unleash/deploy/securing-unleashweekly0.5https://docs.getunleash.io/using-unleash/deploy/securing-unleash-v3weekly0.5https://docs.getunleash.io/using-unleash/deploy/upgrading-unleashweekly0.5https://docs.getunleash.io/using-unleash/troubleshootingweekly0.5https://docs.getunleash.io/using-unleash/troubleshooting/corsweekly0.5https://docs.getunleash.io/using-unleash/troubleshooting/email-serviceweekly0.5https://docs.getunleash.io/using-unleash/troubleshooting/feature-not-availableweekly0.5https://docs.getunleash.io/using-unleash/troubleshooting/flag-exposureweekly0.5https://docs.getunleash.io/using-unleash/troubleshooting/flag-not-returnedweekly0.5 \ No newline at end of file diff --git a/topics.html b/topics.html index 8bf894800a..6326894b5e 100644 --- a/topics.html +++ b/topics.html @@ -20,15 +20,15 @@ - - + + - - + + \ No newline at end of file diff --git a/topics/a-b-testing.html b/topics/a-b-testing.html index c2200356ca..09ad57cb14 100644 --- a/topics/a-b-testing.html +++ b/topics/a-b-testing.html @@ -20,8 +20,8 @@ - - + + @@ -32,7 +32,7 @@ It allows you to capture events whenever a feature toggle is checked in your applications. The event contains all the information about the toggle and the current context, so you can pass everything onto your third-party analytics provider, such as Google Analytics or Posthog. This makes Unleash even more useful as an A/B testing tool and makes it much easier to correlate events and variants with feature toggles and Unleash context.

    Summary

    A/B testing allows you to run experiments on your users and improve your product by using real, proven metrics. It's used by some of the world's most popular businesses and can help you get ahead of competitors (and stay on top). We at Unleash want to help you, so we've baked in some tools to let you do A/B testing right out the gate to make it as smooth as possible to get started.

    So what are you waiting for? Find out what you want to improve next and get testing!

    - - + + \ No newline at end of file diff --git a/topics/feature-flag-migration/business-case-feature-flag-migration.html b/topics/feature-flag-migration/business-case-feature-flag-migration.html index 9b4e453b1e..478ad2f4a4 100644 --- a/topics/feature-flag-migration/business-case-feature-flag-migration.html +++ b/topics/feature-flag-migration/business-case-feature-flag-migration.html @@ -20,15 +20,15 @@ - - + +

    Make the business case for feature flag migration

    Once you have scoped your migration, you need to make a business case. Even the most well planned migrations take effort, meaning time, money, and energy dedicated to a project. If you don’t have the proper buy-in, you risk being under-resourced or worse, being unable to complete the migration at all.

    When building a business case, you want to be clear on what pain the feature flag migration is solving and the happy end state once the migration is complete.

    To structure your thinking, ask yourself:

    • What practices related to feature deployments, debugging and rollbacks are overburdening teams today and driving down productivity?
    • What specific deficiencies are there in the current platform
    • What business outcomes are you looking to drive?
    • After the migration, what does "better" look like?

    Use our Feature Flag Migration template to fill in details about your business case.

    - - + + \ No newline at end of file diff --git a/topics/feature-flag-migration/feature-flag-migration-best-practices.html b/topics/feature-flag-migration/feature-flag-migration-best-practices.html index 741c6a42a0..c6435939f4 100644 --- a/topics/feature-flag-migration/feature-flag-migration-best-practices.html +++ b/topics/feature-flag-migration/feature-flag-migration-best-practices.html @@ -20,15 +20,15 @@ - - + +

    Best Practices for Migrating from a Homegrown Feature Management Solution

    Many large organizations have an existing feature management solution that they have outgrown and plan to migrate to a feature flag service.

    This guide outlines Best Practices for feature flag migrations. Approaching the migration from your current feature flag solution to Unleash the right way will save you time, money, and a lot of headaches.

    Based on our work with organizations having millions of flags and thousands of users, there are five phases of a feature flag migration:

    1. Defining the scope of the feature flag migration
    2. Make the business case for feature flag migration
    3. Planning Feature Flag Migration
    4. Migration Execution
    5. Onboarding users

    This guide provides a summary of each topic as well as a detailed Feature Flag Migration template that you can use to plan your migration.

    - - + + \ No newline at end of file diff --git a/topics/feature-flag-migration/feature-flag-migration-scope.html b/topics/feature-flag-migration/feature-flag-migration-scope.html index a4dcdd4725..eb3d32728b 100644 --- a/topics/feature-flag-migration/feature-flag-migration-scope.html +++ b/topics/feature-flag-migration/feature-flag-migration-scope.html @@ -20,15 +20,15 @@ - - + +

    Defining the scope of the feature flag migration

    Scoping a feature flag migration properly is the most significant task you can do to ensure the success of your project.

    Based on experiences working with dozens of large enterprises migrating homegrown systems to Unleash, we recommend two best practices when scoping your feature flag migration.

    1- Separate the migration of old flags from the existing system from new flags created in Unleash.

    The older the system, the more existing flags there are. It might take weeks or months to hunt down the developer responsible for an old flag in an obscure part of the code base. In the meantime, hundreds of developers are trying to create new flags today. By separating these concerns, you can get to the "happy end state" for your development team faster, and burn down your flag migrations over time.

    So you should end up with two separate tracks as part of your project scope.

    1. Build the new platform around the "better" target state - ideal use cases and ways of working that enable greater levels of developer efficiency

    In parallel, the second track:

    1. Clean up stale feature flags in the current platform. You should decide strategically on what should be migrated and what should be cleaned up. Many old flags can simply be deleted rather than migrated.

    2- Do not make end-to-end app modernization a dependency of your feature flag migration

    Organizations who adopt feature flags see improvements in all key operational metrics for DevOps: Lead time to changes, mean-time-to-recovery, deployment frequency, and change failure rate. So it is natural as part of a feature flag migration to also improve other parts of the software development lifecycle. From the perspective of feature flag migration, this is a mistake.

    Making feature flag migration dependent on breaking down mission-critical monolithic applications into microservices, for example, will slow down your feature flag migration.

    Rather, enable feature flags for all codebases, independent of your state of modernization. Even monolithic applications can benefit from feature flags in some instances. When this monolith is broken down, the accrued benefits will be even greater, and you will ship your new feature management system a lot faster.

    Use our Feature Flag Migration template to fill in details about your project scope.

    - - + + \ No newline at end of file diff --git a/topics/feature-flag-migration/how-to-execute-feature-flag-migration.html b/topics/feature-flag-migration/how-to-execute-feature-flag-migration.html index c99cc0611c..09f4b66edb 100644 --- a/topics/feature-flag-migration/how-to-execute-feature-flag-migration.html +++ b/topics/feature-flag-migration/how-to-execute-feature-flag-migration.html @@ -20,15 +20,15 @@ - - + +

    Migration Execution

    Now that we have completed the planning, below are some of our Best Practices for carrying out the migration based on our experience.

    First, it can help to break the migration down into an order of activities, for example:

    • Minimum Viable Product (MVP) launch
      • Platform implemented, passed security/change management requirements, and available for developer use
      • Rollout to the highest priority groups of users
      • Matching use cases of the legacy platform
      • Legacy system fallback available
    • Medium-term
      • Rollout to additional groups; adoption of further, less critical use cases
      • Sunset of legacy system
    • Longer term
      • Adoption of new use cases

    For each activity, plan for the Level of Effort (LoE), or the number of hours/days the task will take the assigned resource or group to fulfill.

    Next up is risk handling. Are there any perceived risks to the timelines that could be addressed upfront?

    • Have the teams involved with the migration committed to set hours for working the migration tasks upfront, have had migration project success criteria and their tasks communicated to them, and Q&A fulfilled?
    • How long are various sign-offs by any relevant groups expected to take?
      • E.g. Change Advisory Board, Security Controls, hardening checks, etc
      • Plan to exceed each team’s documentation requirements to ensure fewer Requests for Information

    Every step of the way, it can help to conduct reviews and look-backs at each rollout stage as well as what lies ahead.

    Use our Feature Flag Migration template to fill in details about your project plan execution.

    - - + + \ No newline at end of file diff --git a/topics/feature-flag-migration/onbording-users-to-feature-flag-service.html b/topics/feature-flag-migration/onbording-users-to-feature-flag-service.html index 54d51ea3ce..29acdbc42f 100644 --- a/topics/feature-flag-migration/onbording-users-to-feature-flag-service.html +++ b/topics/feature-flag-migration/onbording-users-to-feature-flag-service.html @@ -20,15 +20,15 @@ - - + +

    Onboarding users

    Finally, after the migration has been completed and everyone has celebrated, you need to onboard team members onto the platform.

    Unleash Customer Success is here to help, whether your developers are seasoned feature flag management experts or new to the concepts - we will deliver tailored, white-glove training to accommodate use cases and developer skill levels.

    In addition, for those who prefer a self-paced approach, a wealth of content is available on our YouTube channel, website, and documentation portal to get your teams going quickly.

    - - + + \ No newline at end of file diff --git a/topics/feature-flag-migration/planning-feature-flag-migration.html b/topics/feature-flag-migration/planning-feature-flag-migration.html index aa431eddf7..a06ffff06e 100644 --- a/topics/feature-flag-migration/planning-feature-flag-migration.html +++ b/topics/feature-flag-migration/planning-feature-flag-migration.html @@ -20,15 +20,15 @@ - - + +

    Planning Feature Flag Migration

    When planning your feature flag migration, it is essential to focus on four key areas:

    • Use cases
    • Core feature flag setup
    • Key stakeholders
    • System architecture

    Use Cases

    A key requirement is to understand how feature flags will be used so that you can set up the proper data model, user permissions, and system architecture.

    Below are some of the most common use cases for feature flags. Start by selecting those that are in scope for your initial rollout. It is common to start with some, but not all, of these and roll out the remaining in the future.

    Use cases for feature flags:

    • Operational or Permission flags (also known as "Kill Switches")
    • Gradual rollouts
    • Canary releases
    • A/B testing / Experimentation

    Core Feature Flag setup

    This planning phase is all about understanding the anatomy of the existing feature flag setup so it can be moved to Unleash.

    Key questions include:

    • How many flags are there?
    • How many are active?
    • Do the inactive flags need to be migrated, or can they be removed entirely, simplifying migration?

    Once you have an understanding of what needs to be migrated, you should plan for how flags will be organized in the future. Picking the right organizing principle is critical for access controls. Unleash supports Application, Project, Environment, and Role-based access, so pick the option that makes the most logical sense for your organization.

    Our experience tells us that organizing flags by the development teams that work on them is the best approach. For example:

    • By application or microservice
      • E.g. Shopping Cart, Website, Mobile app
    • By Projects
      • E.g. Logistics, Finance
    • By organization hierarchy
      • E.g. Frontend, backend, platform teams

    Key stakeholders

    When planning your migration, it is important to understand who will be managing the feature flag system and who will be using the feature flag system on a day-to-day basis. Additionally, you need to know who will be responsible for key migration tasks.

    From our experience, looping all key stakeholders into the project early on means that all eventualities can be planned for in advance, reducing the risk of project implementation delays due to unforeseen sign-off requirements. Decision makers can help assign and gather resources needed for the migration, as well as advise on the correct business processes that need to be followed at the various project stages.

    System Architecture

    You will also need to plan how you set up Unleash itself as part of your migration planning process. Unleash is extremely flexible with lots of hosting and security configuration options to align with the unique requirements of large enterprises.

    How is our current feature flag architecture set up?

    This part is key to understanding how Unleash needs to be implemented. This plays a part in resource planning for both personnel and infrastructure cost allocation. For instance

    • What languages and frameworks are our front end and backend using?
    • Where are our applications hosted?
    • Where are end users of the application based geographically?

    Security and organizational policy requirements

    You will also want to pay attention to Security & Organisational Policy requirements.

    For example, do you need to keep sensitive context inside your firewall perimeter?

    Often the answer to this question defines whether you will run Unleash in a hosted, or self-hosted fashion.

    Many customers prefer a hosted solution if their security policy will allow it because it reduces overhead on SRE teams and infrastructure. Unleash offers a single-tenant hosted solution, so even security-conscious customers can sometimes opt for a hosted solution.

    If that is not an option, Unleash instances need to be managed by customer SRE / DevOps teams. If this is the direction you are going, you should plan for it in this phase of the project.

    Other areas of system architecture to investigate during the planning phase are:

    • Data protection
      • Do we have to comply with HIPPA, SOC-2, ISO 27001, GDPR, etc?
    • How do we authenticate and manage user access & RBAC/roles?
    • Do we have any Change Management policies we need to adhere to?
    • Do we consume feature flag data, such as events, in any other systems downstream?
      • For example, Jira Cloud for issue management, Datadog for real-time telemetry, Slack or Microsoft Teams for notifications or Google Analytics for user interactions.

    Use our Feature Flag Migration template to fill in details about your project planning.

    - - + + \ No newline at end of file diff --git a/topics/feature-flags/availability-over-consistency.html b/topics/feature-flags/availability-over-consistency.html index 8fc51bd25c..09ae0bdee4 100644 --- a/topics/feature-flags/availability-over-consistency.html +++ b/topics/feature-flags/availability-over-consistency.html @@ -20,15 +20,15 @@ - - + +

    6. Design for failure. Favor availability over consistency.

    Your feature flag system should not be able to take down your main application under any circumstance, including network disruptions. Follow these patterns to achieve fault tolerance for your feature flag system.

    Zero dependencies: Your application's availability should have zero dependencies on the availability of your feature flag system. Robust feature flag systems avoid relying on real-time flag evaluations because the unavailability of the feature flag system will cause application downtime, outages, degraded performance, or even a complete failure of your application.

    Graceful degradation: If the system goes down, it should not disrupt the user experience or cause unexpected behavior. Feature flagging should gracefully degrade in the absence of the Feature Flag Control service, ensuring that users can continue to use the application without disruption.

    Resilient Architecture Patterns:

    • Bootstrapping SDKs with Data: Feature flagging SDKs used within your application should be designed to work with locally cached data, even when the network connection to the Feature Flag Control service is unavailable. The SDKs can bootstrap with the last known feature flag configuration or default values to ensure uninterrupted functionality.

    • Local Cache: Maintaining a local cache of feature flag configuration helps reduce network round trips and dependency on external services. The local cache can be periodically synchronized with the central Feature Flag Control service when it's available. This approach minimizes the impact of network failures or service downtime on your application.

    • Evaluate Locally: Whenever possible, the SDKs or application components should be able to evaluate feature flags locally without relying on external services. This ensures that feature flag evaluations continue even when the feature flagging service is temporarily unavailable.

    • Availability Over Consistency: As the CAP theorem teaches us, in distributed systems, prioritizing availability over strict consistency can be a crucial design choice. This means that, in the face of network partitions or downtime of external services, your application should favor maintaining its availability rather than enforcing perfectly consistent feature flag configuration caches. Eventually consistent systems can tolerate temporary inconsistencies in flag evaluations without compromising availability. In CAP theorem parlance, a feature flagging system should aim for AP over CP.

    By implementing these resilient architecture patterns, your feature flagging system can continue to function effectively even in the presence of downtime or network disruptions in the feature flagging service. This ensures that your main application remains stable, available, and resilient to potential issues in the feature flagging infrastructure, ultimately leading to a better user experience and improved reliability.

    - - + + \ No newline at end of file diff --git a/topics/feature-flags/democratize-feature-flag-access.html b/topics/feature-flags/democratize-feature-flag-access.html index 0b983d1663..00e4404c6a 100644 --- a/topics/feature-flags/democratize-feature-flag-access.html +++ b/topics/feature-flags/democratize-feature-flag-access.html @@ -20,15 +20,15 @@ - - + +

    9. Choose open by default. Democratize feature flag access.

    Allowing engineers, product owners, and even technical support to have open access to a feature flagging system is essential for effective development, debugging, and decision-making. These groups should have access to the system, along with access to the codebase and visibility into configuration changes:

    1. Debugging and Issue Resolution:

      • Code Access: Engineers should have access to the codebase where feature flags are implemented. This access enables them to quickly diagnose and fix issues related to feature flags when they arise. Without code access, debugging becomes cumbersome, and troubleshooting becomes slower, potentially leading to extended downtimes or performance problems.
    2. Visibility into Configuration:

      • Configuration Transparency: Engineers, product owners, and even technical support should be able to view the feature toggle configuration. This transparency provides insights into which features are currently active, what conditions trigger them, and how they impact the application's behavior. It helps understand the system's state and behavior, which is crucial for making informed decisions.

      • Change History: Access to a history of changes made to feature flags, including who made the changes and when, is invaluable. This audit trail allows teams to track changes to the system's behavior over time. It aids in accountability and can be instrumental in troubleshooting when unexpected behavior arises after a change.

      • Correlating Changes with Metrics: Engineers and product owners often need to correlate feature flag changes with production application metrics. This correlation helps them understand how feature flags affect user behavior, performance, and system health. It's essential for making data-driven decisions about feature rollouts, optimizations, or rollbacks.

    3. Collaboration:

      • Efficient Communication: Open access fosters efficient communication between engineers and the rest of the organization. When it's open by default, everyone can see the feature flagging system and its changes, and have more productive discussions about feature releases, experiments, and their impact on the user experience.
    4. Empowering Product Decisions:

      • Product Owner Involvement: Product owners play a critical role in defining feature flags' behavior and rollout strategies based on user needs and business goals. Allowing them to access the feature flagging system empowers them to make real-time decisions about feature releases, rollbacks, or adjustments without depending solely on engineering resources.
    5. Security and Compliance:

      • Security Audits: Users of a feature flag system should be part of corporate access control groups such as SSO. Sometimes, additional controls are necessary, such as feature flag approvals using the four-eyes principle.

    Access control and visibility into feature flag changes are essential for security and compliance purposes. It helps track and audit who has made changes to the system, which can be crucial in maintaining data integrity and adhering to regulatory requirements.

    - - + + \ No newline at end of file diff --git a/topics/feature-flags/enable-traceability.html b/topics/feature-flags/enable-traceability.html index 7012e6de72..f378b2c392 100644 --- a/topics/feature-flags/enable-traceability.html +++ b/topics/feature-flags/enable-traceability.html @@ -20,15 +20,15 @@ - - + +

    11. Enable traceability. Make it easy to understand flag evaluation.

    Developer experience is a critical factor to consider when implementing a feature flag solution. A positive developer experience enhances the efficiency of the development process and contributes to the overall success and effectiveness of feature flagging. One crucial aspect of developer experience is ensuring the testability of the SDK and providing tools for developers to understand how and why feature flags are evaluated. This is important because:

    1. Ease of Testing and Debugging:

      • Faster Development Cycles: A feature flagging solution with a testable SDK allows developers to quickly test and iterate on new features. They can easily turn flags on or off, simulate different conditions, and observe the results without needing extensive code changes or redeployments.

      • Rapid Issue Resolution: When issues or unexpected behavior arise, a testable SDK enables developers to pinpoint the problem more efficiently. They can examine the flag configurations, log feature flag decisions, and troubleshoot issues more precisely.

    2. Visibility into Flag Behaviour:

      • Understanding User Experience: Developers need tools to see and understand how feature flags affect the user experience. This visibility helps them gauge the impact of flag changes and make informed decisions about when to roll out features to different user segments. Debugging a feature flag with multiple inputs simultaneously makes it easy for developers to compare the results and quickly figure out how a feature flag evaluates in different scenarios with multiple input values.

      • Enhanced Collaboration: Feature flagging often involves cross-functional teams, including developers, product managers, and QA testers. Providing tools with a clear view of flag behavior fosters effective collaboration and communication among team members.

    3. Transparency and Confidence:

      • Confidence in Flag Decisions: A transparent feature flagging solution empowers developers to make data-driven decisions. They can see why a particular flag evaluates to a certain value, which is crucial for making informed choices about feature rollouts and experimentation.

      • Reduced Risk: When developers clearly understand of why flags evaluate the way they do, they are less likely to make unintentional mistakes that could lead to unexpected issues in production.

    4. Effective Monitoring and Metrics:

      • Tracking Performance: A testable SDK should provide developers with the ability to monitor the performance of feature flags in real time. This includes tracking metrics related to flag evaluations, user engagement, and the impact of flag changes.

      • Data-Driven Decisions: Developers can use this data to evaluate the success of new features, conduct A/B tests, and make informed decisions about optimizations.

      • Usage metrics: A feature flag system should provide insight on an aggregated level about the usage of feature flags. This is helpful for developers so that they can easily assess that everything works as expected.

    5. Documentation and Training:

      • Onboarding and Training: The entire feature flag solution, including API, UI, and the SDKs, requires clear and comprehensive documentation, along with easy-to-understand examples, in order to simplify the onboarding process for new developers. It also supports the ongoing training of new team members, ensuring that everyone can effectively use the feature flagging solution.
    - - + + \ No newline at end of file diff --git a/topics/feature-flags/evaluate-flags-close-to-user.html b/topics/feature-flags/evaluate-flags-close-to-user.html index e692672cbc..538fcbd880 100644 --- a/topics/feature-flags/evaluate-flags-close-to-user.html +++ b/topics/feature-flags/evaluate-flags-close-to-user.html @@ -20,15 +20,15 @@ - - + +

    3. Evaluate flags as close to the user as possible. Reduce latency.

    Feature flags should be evaluated as close to your users as possible, and the evaluation should always happen server side as discussed in Principle 2. In addition to security and privacy benefits, performing evaluation as close as possible to your users has multiple benefits:

    1. Performance Efficiency:

      a. Reduced Latency: Network roundtrips introduce latency, which can slow down your application's response time. Local evaluation eliminates the need for these roundtrips, resulting in faster feature flag decisions. Users will experience a more responsive application, which can be critical for maintaining a positive user experience.

      b. Offline Functionality: Applications often need to function offline or in low-connectivity environments. Local evaluation ensures that feature flags can still be used and decisions can be made without relying on a network connection. This is especially important for mobile apps or services in remote locations.

    2. Cost Savings:

      a. Reduced Bandwidth Costs: Local evaluation reduces the amount of data transferred between your application and the feature flag service. This can lead to significant cost savings, particularly if you have a large user base or high traffic volume.

    3. Offline Development and Testing:

      a. Development and Testing: Local evaluation is crucial for local development and testing environments where a network connection to the feature flag service might not be readily available. Developers can work on feature flag-related code without needing constant access to the service, streamlining the development process.

    4. Resilience:

      a. Service Outages: If the feature flag service experiences downtime or outages, local evaluation allows your application to continue functioning without interruptions. This is important for maintaining service reliability and ensuring your application remains available even when the service is down.

    In summary, this principle emphasizes the importance of optimizing performance while protecting end-user privacy by evaluating feature flags as close to the end user as possible. Done right, this also leads to a highly available feature flag system that scales with your applications.

    - - + + \ No newline at end of file diff --git a/topics/feature-flags/feature-flag-best-practices.html b/topics/feature-flags/feature-flag-best-practices.html index b3710ea7b0..6908d68e23 100644 --- a/topics/feature-flags/feature-flag-best-practices.html +++ b/topics/feature-flags/feature-flag-best-practices.html @@ -20,15 +20,15 @@ - - + +

    11 Principles for building and scaling feature flag systems

    Feature flags, sometimes called feature toggles or feature switches, are a software development technique that allows engineering teams to decouple the release of new functionality from software deployments. With feature flags, developers can turn specific features or code segments on or off at runtime, without the need for a code deployment or rollback. Organizations who adopt feature flags see improvements in all key operational metrics for DevOps: Lead time to changes, mean-time-to-recovery, deployment frequency, and change failure rate.

    There are 11 principles for building a large-scale feature flag system. These principles have their roots in distributed systems architecture and pay particular attention to security, privacy, and scale that is required by most enterprise systems. If you follow these principles, your feature flag system is less likely to break under load and will be easier to evolve and maintain.

    These principles are:

    1. Enable run-time control. Control flags dynamically, not using config files.
    2. Never expose PII. Follow the principle of least privilege.
    3. Evaluate flags as close to the user as possible. Reduce latency.
    4. Scale Horizontally. Decouple reading and writing flags.
    5. Limit payloads. Feature flag payload should be as small as possible.
    6. Design for failure. Favor availability over consistency.
    7. Make feature flags short-lived. Do not confuse flags with application configuration.
    8. Use unique names across all applications. Enforce naming conventions.
    9. Choose open by default. Democratize feature flag access.
    10. Do no harm. Prioritize consistent user experience.
    11. Enable traceability. Make it easy to understand flag evaluation

    Background

    Feature flags have become a central part of the DevOps toolbox along with Git, CI/CD and microservices. You can write modern software without all of these things, but it sure is a lot harder, and a lot less fun.

    And just like the wrong Git repo design can cause interminable headaches, getting the details wrong when first building a feature flag system can be very costly.

    This set of principles for building a large-scale feature management platform is the result of thousands of hours of work building and scaling Unleash, an open-source feature management solution used by thousands of organizations.

    Before Unleash was a community and a company, it was an internal project, started by one dev, for one company. As the community behind Unleash grew, patterns and anti-patterns of large-scale feature flag systems emerged. Our community quickly discovered that these are important principles for anyone who wanted to avoid spending weekends debugging the production system that is supposed to make debugging in production easier.

    “Large scale” means the ability to support millions of flags served to end-users with minimal latency or impact on application uptime or performance. That is the type of system most large enterprises are building today and the type of feature flag system that this guide focuses on.

    Our motivation for writing these principles is to share what we’ve learned building a large-scale feature flag solution with other architects and engineers solving similar challenges. Unleash is open-source, and so are these principles. Have something to contribute? Open a PR or discussion on our Github.

    - - + + \ No newline at end of file diff --git a/topics/feature-flags/limit-payloads.html b/topics/feature-flags/limit-payloads.html index 2a89977e7a..985d97a4a4 100644 --- a/topics/feature-flags/limit-payloads.html +++ b/topics/feature-flags/limit-payloads.html @@ -20,15 +20,15 @@ - - + +

    5. Limit payloads. Feature flag payload should be as small as possible.

    Minimizing the size of feature flag payloads is a critical aspect of maintaining the efficiency and performance of a feature flag system. The configuration of your feature flags can vary in size depending on the complexity of your targeting rules. For instance, if you have a targeting engine that determines whether a feature flag should be active or inactive based on individual user IDs, you might be tempted to include all these user IDs within the configuration payload. While this approach may work fine for a small user base, it can become unwieldy when dealing with a large number of users.

    If you find yourself facing this challenge, your instinct might be to store this extensive user information directly in the feature flagging system. However, this can also run into scaling problems. A more efficient approach is to categorize these users into logical groupings at a different layer and then use these group identifiers when you evaluate flags within your feature flagging system. For example, you can group users based on their subscription plan or geographical location. Find a suitable parameter for grouping users, and employ those group parameters as targeting rules in your feature flagging solution.

    Imposing limitations on payloads is crucial for scaling a feature flag system:

    1. Reduced Network Load:

      • Large payloads, especially for feature flag evaluations, can lead to increased network traffic between the application and the feature flagging service. This can overwhelm the network and cause bottlenecks, leading to slow response times and degraded system performance. Limiting payloads helps reduce the amount of data transferred over the network, alleviating this burden. Even small numbers become large when multiplied by millions.
    2. Faster Evaluation:

      • Smaller payloads reduce latency which means quicker transmission and evaluation. Speed is essential when evaluating feature flags, especially for real-time decisions that impact user experiences. Limiting payloads ensures evaluations occur faster, allowing your application to respond promptly to feature flag changes.
    3. Improved Memory Efficiency:

      • Feature flagging systems often store flag configurations in memory for quick access during runtime. Larger payloads consume more memory, potentially causing memory exhaustion and system crashes. By limiting payloads, you ensure that the system remains memory-efficient, reducing the risk of resource-related issues.
    4. Scalability:

      • Scalability is a critical concern for modern applications, especially those experiencing rapid growth. Feature flagging solutions need to scale horizontally to accommodate increased workloads. Smaller payloads require fewer resources for processing, making it easier to scale your system horizontally.
    5. Lower Infrastructure Costs:

      • When payloads are limited, the infrastructure required to support the feature flagging system can be smaller and less costly. This saves on infrastructure expenses and simplifies the management and maintenance of the system.
    1. Reliability:

      • A feature flagging system that consistently delivers small, manageable payloads is more likely to be reliable. It reduces the risk of network failures, timeouts, and other issues when handling large data transfers. Reliability is paramount for mission-critical applications.
    2. Ease of Monitoring and Debugging:

      • Smaller payloads are easier to monitor and debug. When issues arise, it's simpler to trace problems and identify their root causes when dealing with smaller, more manageable data sets.
    - - + + \ No newline at end of file diff --git a/topics/feature-flags/never-expose-pii.html b/topics/feature-flags/never-expose-pii.html index 0ab2e10753..6120e45101 100644 --- a/topics/feature-flags/never-expose-pii.html +++ b/topics/feature-flags/never-expose-pii.html @@ -20,15 +20,15 @@ - - + +

    2. Never expose PII. Follow the principle of least privilege.

    To keep things simple, you may be tempted to evaluate the feature flags in your Feature Flag Control Service. Don’t. Your Feature Flag Control Service should only handle the configuration for your feature flags and pass this configuration down to SDKs connecting from your applications.

    The primary rationale behind this practice is that feature flags often require contextual data for accurate evaluation. This may include user IDs, email addresses, or geographical locations that influence whether a flag should be toggled on or off. Safeguarding this sensitive information from external exposure is paramount. This information may include Personally Identifiable Information (PII), which must remain confined within the boundaries of your application, following the data security principle of least privilege (PoLP).

    feature-flag-server-side-evaluation

    For client-side applications where the code resides on the user's machine, such as in the browser or on mobile devices, you’ll want to take a different approach. You can’t evaluate on the client side because it raises significant security concerns by exposing potentially sensitive information such as API keys, flag data, and flag configurations. Placing these critical elements on the client side increases the risk of unauthorized access, tampering, or data breaches.

    Instead of performing client-side evaluation, a more secure and maintainable approach is to conduct feature flag evaluation within a self-hosted environment. Doing so can safeguard sensitive elements like API keys and flag configurations from potential client-side exposure. This strategy involves a server-side evaluation of feature flags, where the server makes decisions based on user and application parameters and then securely passes down the evaluated results to the frontend without any configuration leaking.

    feature-flag-architecture-client-side

    Here’s how you can architect your solution to minimize PII or configuration leakage:

    1. Server-Side Components:

    In Principle 1, we proposed a set of architectural principles and components to set up a Feature Flag Control Service. The same architecture patterns apply here, with additional suggestions for achieving local evaluation. Refer to Principle 1 for patterns to set up a feature flagging service.

    Feature Flag Evaluation Service: If you need to use feature flags on the client side, where code is delivered to users' devices, you’ll need an evaluation server that can evaluate feature flags and pass evaluated results down to the SDK in the client application.

    1. SDKs:

    SDKs will make it more comfortable to work with feature flags. Depending on the context of your infrastructure, you need different types of SDKs to talk to your feature flagging service. For the server side, you’ll need SDKs that can talk directly to the feature flagging service and fetch the configuration.

    The server-side SDKs should implement logic to evaluate feature flags based on the configuration received from the Feature Flag Control Service and the application-specific context. Local evaluation ensures that decisions are made quickly without relying on network roundtrips.

    For client-side feature flags, you’ll need a different type of SDK. These SDKs will send the context to the Feature Flag Evaluation Service and receive the evaluated results. These results should be stored in memory and used when doing a feature flag lookup in the client-side application. By keeping the evaluated results for a specific context in memory in the client-side application, you avoid network roundtrips every time your application needs to check the status of a feature flag. It achieves the same level of performance as a server-side SDK, but the content stored in memory is different and limited to evaluated results on the client.

    The benefits of this approach include:

    Privacy Protection:

    a. Data Minimization: By evaluating feature flags in this way, you minimize the amount of data that needs to be sent to the Feature Flag Control Service. This can be crucial for protecting user privacy, as less user-specific data is transmitted over the network.

    b. Reduced Data Exposure: Sensitive information about your users or application's behavior is less likely to be exposed to potential security threats. Data breaches or leaks can be mitigated by limiting the transmission of sensitive data.

    - - + + \ No newline at end of file diff --git a/topics/feature-flags/prioritize-ux.html b/topics/feature-flags/prioritize-ux.html index 9fecf5d8f8..a556b0bbcc 100644 --- a/topics/feature-flags/prioritize-ux.html +++ b/topics/feature-flags/prioritize-ux.html @@ -20,15 +20,15 @@ - - + +

    10. Do no harm. Prioritize consistent user experience.

    Feature flagging solutions are indispensable tools in modern software development, enabling teams to manage feature releases and experiment with new functionality. However, one aspect that is absolutely non-negotiable in any feature flag solution is the need to ensure a consistent user experience. This isn't a luxury; it's a fundamental requirement. Feature flagging solutions must prioritize consistency and guarantee the same user experience every time, especially with percentage-based gradual rollouts.

    Why Consistency is Paramount:

    1. User Trust: Consistency breeds trust. When users interact with an application, they form expectations about how it behaves. Any sudden deviations can erode trust and lead to a sense of unreliability.

    2. Reduced Friction: Consistency reduces friction. Users shouldn't have to relearn how to use an app every time they open it. A consistent experience reduces the cognitive load on users, enabling them to engage effortlessly.

    3. Quality Assurance: Maintaining a consistent experience makes quality assurance more manageable. It's easier to test and monitor when you have a reliable benchmark for the user experience.

    4. Support and Feedback: Inconsistent experiences lead to confused users, increased support requests, and muddied user feedback. Consistency ensures that user issues are easier to identify and address.

    5. Brand Integrity: A consistent experience reflects positively on your brand. It demonstrates professionalism and commitment to user satisfaction, enhancing your brand's reputation.

    Strategies for Consistency in Percentage-Based Gradual Rollouts:

    1. User Hashing: Assign users to consistent groups using a secure hashing algorithm based on unique identifiers like user IDs or emails. This ensures that the same user consistently falls into the same group.

    2. Segmentation Control: Provide controls within the feature flagging tool to allow developers to segment users logically. For instance, segment by location, subscription type, or any relevant criteria to ensure similar user experiences.

    3. Fallback Mechanisms: Include fallback mechanisms in your architecture. If a user encounters issues or inconsistencies, the system can automatically switch them to a stable version or feature state.

    4. Logging and Monitoring: Implement robust logging and monitoring. Continuously track which users are in which groups and what version of the feature they are experiencing. Monitor for anomalies or deviations and consider building automated processes to disable features that may be misbehaving.

    5. Transparent Communication: Clearly communicate the gradual rollout to users. Use in-app notifications, tooltips, or changelogs to inform users about changes, ensuring they know what to expect.

    In summary, consistency is a cornerstone of effective feature flagging solutions. When designing an architecture for percentage-based gradual rollouts, prioritize mechanisms that guarantee the same user gets the same experience every time. This isn't just about good software practice; it's about respecting your users and upholding their trust in your application. By implementing these strategies, you can create a feature flagging solution that empowers your development process and delights your users with a dependable and consistent experience.

    - - + + \ No newline at end of file diff --git a/topics/feature-flags/runtime-control.html b/topics/feature-flags/runtime-control.html index 689052ee80..db1637d525 100644 --- a/topics/feature-flags/runtime-control.html +++ b/topics/feature-flags/runtime-control.html @@ -20,15 +20,15 @@ - - + +

    1. Enable run-time control. Control flags dynamically, not using config files.

    A scalable feature management system evaluates flags at runtime. Flags are dynamic, not static. If you need to restart your application to turn on a flag, you are using configuration, not feature flags.

    A large-scale feature flag system that enables runtime control should have at minimum the following components:

    1. Feature Flag Control Service: Use a centralized feature flag service that acts as the control plane for your feature flags. This service will handle flag configuration. The scope of this service should reflect the boundaries of your organization.

    Independent business units or product lines should potentially have their own instances, while business units or product lines that work closely together should most likely use the same instance in order to facilitate collaboration. This will always be a contextual decision based on your organization and how you organize the work, but keep in mind that you’d like to keep the management of the flags as simple as possible to avoid the complexity of cross-instance synchronization of feature flag configuration.

    2. Database or Data Store: Use a robust and scalable database or data store to store feature flag configurations. Popular choices include SQL or NoSQL databases or key-value stores. Ensure that this store is highly available and reliable.

    3. API Layer: Develop an API layer that exposes endpoints for your application to interact with the Feature Flag Control Service. This API should allow your application to request feature flag configurations.

    4. Feature Flag SDK: Build or integrate a feature flag SDK into your application. This SDK should provide an easy-to-use interface for fetching flag configurations and evaluating feature flags at runtime. When evaluating feature flags in your application, the call to the SDK should query the local cache, and the SDK should ask the central service for updates in the background.

    Build SDK bindings for each relevant language in your organization. Make sure that the SDKs uphold a standard contract governed by a set of feature flag client specifications that documents what functionality each SDK should support.

    feature-flag-scalable-architecture

    5. Continuously Updated: Implement update mechanisms in your application so that changes to feature flag configurations are reflected without requiring application restarts or redeployments. The SDK should handle subscriptions or polling to the feature flag service for updates.

    - - + + \ No newline at end of file diff --git a/topics/feature-flags/scale-horizontally.html b/topics/feature-flags/scale-horizontally.html index 4fe637f2c6..ad2080eca9 100644 --- a/topics/feature-flags/scale-horizontally.html +++ b/topics/feature-flags/scale-horizontally.html @@ -20,15 +20,15 @@ - - + +

    4. Scale Horizontally. Decouple reading and writing flags.

    Separating the reading and writing of feature flags into distinct APIs is a critical architectural decision for building a scalable and efficient feature flag system, particularly when considering horizontal scaling. This separation provides several benefits:

    feature-flag-horizontal-scaling

    1. Horizontal Scaling:

      • By separating read and write APIs, you can horizontally scale each component independently. This enables you to add more servers or containers to handle increased traffic for reading feature flags, writing updates, or both, depending on the demand.
    2. Caching Efficiency:

      • Feature flag systems often rely on caching to improve response times for flag evaluations. Separating read and write APIs allows you to optimize caching strategies independently. For example, you can cache read operations more aggressively to minimize latency during flag evaluations while still ensuring that write operations maintain consistency across the system.
    3. Granular Access Control:

      • Separation of read and write APIs simplifies access control and permissions management. You can apply different security measures and access controls to the two APIs. This helps ensure that only authorized users or systems can modify feature flags, reducing the risk of accidental or unauthorized changes.
    4. Better Monitoring and Troubleshooting:

      • Monitoring and troubleshooting become more straightforward when read and write operations are separated. It's easier to track and analyze the performance of each API independently. When issues arise, you can isolate the source of the problem more quickly and apply targeted fixes or optimizations.
    5. Flexibility and Maintenance:

      • Separation of concerns makes your system more flexible and maintainable. Changes or updates to one API won't directly impact the other, reducing the risk of unintended consequences. This separation allows development teams to work on each API separately, facilitating parallel development and deployment cycles.
    6. Load Balancing:

      • Load balancing strategies can be tailored to the specific needs of the read and write APIs. You can distribute traffic and resources accordingly to optimize performance and ensure that neither API becomes a bottleneck under heavy loads.
    - - + + \ No newline at end of file diff --git a/topics/feature-flags/short-lived-feature-flags.html b/topics/feature-flags/short-lived-feature-flags.html index 175153b87b..f58bd6572b 100644 --- a/topics/feature-flags/short-lived-feature-flags.html +++ b/topics/feature-flags/short-lived-feature-flags.html @@ -20,15 +20,15 @@ - - + +

    7. Make feature flags short-lived. Do not confuse flags with application configuration.

    Feature flags have a lifecycle shorter than an application lifecycle. The most common use case for feature flags is to protect new functionality. That means that when the roll-out of new functionality is complete, the feature flag should be removed from the code and archived. If there were old code paths that the new functionality replaces, those should also be cleaned up and removed.

    Feature flags should not be used for static application configuration. Application configuration is expected to be consistent, long-lived, and read when launching an application. Using feature flags to configure an application can lead to inconsistencies between different instances of the same application. Feature flags, on the other hand, are designed to be short-lived, dynamic, and changed at runtime. They are expected to be read and updated at runtime and favor availability over consistency.

    To succeed with feature flags in a large organization, you should:

    • Use flag expiration dates: By setting expiration dates for your feature flags, you make it easier to keep track of old feature flags that are no longer needed. A proper feature flag solution will inform you about potentially expired flags.

    • Treat feature flags like technical debt.: You must plan to clean up old feature branches in sprint or project planning, the same way you plan to clean up technical debt in your code. Feature flags add complexity to your code. You’ll need to know what code paths the feature flag enables, and while the feature flag lives, the context of it needs to be maintained and known within the organization. If you don’t clean up feature flags, eventually, you may lose the context surrounding it if enough time passes and/or personnel changes happen. As time passes, you will find it hard to remove flags, or to operate them effectively.

    • Archive old flags: Feature flags that are no longer in use should be archived after their usage has been removed from the codebase. The archive serves as an important audit log of feature flags that are no longer in use, and allows you to revive them if you need to install an older version of your application.

    There are valid exceptions to short-lived feature flags. In general, you should try to limit the amount of long-lived feature flags. Some examples include:

    • Kill-switches - these work like an inverted feature flag and are used to gracefully disable part of a system with known weak spots.
    • Internal flags used to enable additional debugging, tracing, and metrics at runtime, which are too costly to run all the time. These can be enabled by software engineers while debugging issues.
    - - + + \ No newline at end of file diff --git a/topics/feature-flags/unique-names.html b/topics/feature-flags/unique-names.html index 646132ce03..7d7bf4f615 100644 --- a/topics/feature-flags/unique-names.html +++ b/topics/feature-flags/unique-names.html @@ -20,15 +20,15 @@ - - + +

    8. Use unique names across all applications. Enforce naming conventions.

    All flags served by the same Feature Flag Control service should have unique names across the entire cluster to avoid inconsistencies and errors.

    • Avoid zombies: Uniqueness should be controlled using a global list of feature flag names. This prevents the reuse of old flag names to protect new features. Using old names can lead to accidental exposure of old features, still protected with the same feature flag name.
    • Naming convention enforcement: Ideally, unique names are enforced at creation time. In a large organization, it is impossible for all developers to know all flags used. Enforcing a naming convention makes naming easier, ensures consistency, and provides an easy way to check for uniqueness.

    Unique naming has the following advantages:

    • Flexibility over time: Large enterprise systems are not static. Over time, monoliths are split into microservices, microservices are merged into larger microservices, and applications change responsibility. This means that the way flags are grouped will change over time, and a unique name for the entire organization ensures that you keep the option to reorganize your flags to match the changing needs of your organization.
    • Prevent conflicts: If two applications use the same Feature Flag name it can be impossible to know which flag is controlling which applications. This can lead to accidentally flipping the wrong flag, even if they are separated into different namespaces (projects, workspaces etc).
    • Easier to manage: It's easier to know what a flag is used for and where it is being used when it has a unique name. E.g. It will be easier to search across multiple code bases to find references for a feature flag when it has a unique identifier across the entire organization.
    • Enables collaboration: When a feature flag has a unique name in the organization, it simplifies collaboration across teams, products and applications. It ensures that we all talk about the same feature.
    - - + + \ No newline at end of file diff --git a/understanding-unleash.html b/understanding-unleash.html index ba3f20903c..a6a464c12d 100644 --- a/understanding-unleash.html +++ b/understanding-unleash.html @@ -20,15 +20,15 @@ - - + +

    Understanding Unleash

    Documentation on how Unleash works, high-level architecture and important concepts.

    📄️ Unleash introductory overview

    One of the most important aspects of the architecture to understand is that feature toggles are evaluated in a client SDKs which runs as part of your application. This makes toggle evaluations super-fast (we're talking nano-seconds), scalable and resilient against network disturbances. In order to achieve this Unleash compromises a small update-delay when you change your toggle configurations until it is fully propagated to your application (in terms of seconds and is configurable).

    - - + + \ No newline at end of file diff --git a/understanding-unleash/data-collection.html b/understanding-unleash/data-collection.html index 1f311ed7a1..841392d1e0 100644 --- a/understanding-unleash/data-collection.html +++ b/understanding-unleash/data-collection.html @@ -20,15 +20,15 @@ - - + +

    Data collection

    info

    At Unleash, we prioritize the privacy and security of our users' data. This document provides an overview of the data collected when running Unleash. We explain the purpose of data collection and provide instructions on managing data collection settings.

    What data is collected

    When running Unleash, we collect the following data:

    Version and Instance ID: A unique identifier and version for your Unleash instance. This ID allows us to track usage statistics and measure the adoption of Unleash across different installations and helps us ensure that you're using the latest version with the most up-to-date features and security enhancements.

    Feature Usage Data: Starting from Unleash 5.3, we collect additional data related to feature usage in Unleash.

    This includes the following data points:

    • The number of active feature toggles in use
    • The total number of users in the system
    • The total number of projects in the system
    • The number of custom context fields defined and in use
    • The number of user groups defined in the system
    • The number of custom roles defined in the system
    • The number of environments defined in the system
    • The number of segments in active use by feature toggles
    • The number of custom strategies defined and in use
    • The number of feature exports/imports made

    Please note that all collected data is anonymous, and we only collect usage counts. This data helps us understand how features are used in Unleash, enabling us to prioritize important features and make informed decisions about deprecating features that are no longer relevant to our users.

    Please note that we do not collect personally identifiable information (PII) through Unleash.

    Managing data collection settings

    We understand that privacy preferences may vary among our users. While the data collected by Unleash is limited and anonymous, we provide options to manage data collection settings:

    Disabling All Telemetry: If you have previously disabled the version telemetry by setting the environment variable CHECK_VERSION to anything other than "true", "t" or "1" both the version telemetry and the feature telemetry will be disabled. This respects your choice to opt out of all telemetry data if you had previously disabled it.

    Turning Off Feature Telemetry: To disable the collection of the new telemetry data while still allowing the version telemetry, set the environment variable SEND_TELEMETRY to anything other than "true", "t" or "1" before starting Unleash. This will ensure that the new telemetry data is not sent, but the version information is still sent.

    We respect your privacy choices, and we will continue to honor your decision regarding telemetry. If you have any questions or concerns about managing data collection settings or privacy, please reach out to our support team for assistance.

    - - + + \ No newline at end of file diff --git a/understanding-unleash/managing-constraints.html b/understanding-unleash/managing-constraints.html index 086e1e72c5..1350c29b05 100644 --- a/understanding-unleash/managing-constraints.html +++ b/understanding-unleash/managing-constraints.html @@ -20,15 +20,15 @@ - - + +

    Managing constraints

    info

    In this explanatory guide, we will discuss how best to deal with large and complex constraints, such as when you want to constrain a feature to a list of 150 user IDs.

    Unleash offers several ways to limit feature exposure to a specified audience, such as the User IDs strategy, strategy constraints, and segments. Each of these options make it easy to add some sort of user identifier to the list of users who should get access to a feature.

    Because of their availability and ease of use with smaller lists, it can be tempting to just keep adding identifiers to those lists. However, once you start approaching a hundred elements, we recommend that you find another way to manage these IDs. In fact, it's probably better to stop well before you get that far.

    The rest of this document explains why this is generally a bad idea and how we suggest you solve your problems instead. However, as always there are exceptional cases where this is the solution.

    How large is too large? How complex is too complex? It depends. However, smaller and simpler is generally better.

    The cost of data

    First, let's talk a bit about Unleash's architecture: Your Unleash instance is where you define all your features, their strategies, and constraints (and a lot more). However, the Unleash instance itself does not do any of the feature evaluation1. That responsibility is delegated to the Unleash SDKs (or Edge / the Unleash proxy).

    For the SDKs (Edge/proxy included) to evaluate a feature, it needs to know everything about that feature in a specific environment. This includes all strategies and their constraints. This means that the Unleash instance must transmit all information about this feature (and all other features) as a response to an API call.

    As the number of elements in a constraint list (such as number of unique user IDs) grows, so does the size of the HTTP response from the Unleash instance. More data to transmit, means more bandwidth used and longer response times. More data to parse (on the SDK side) means more time spent processing and more data to store and look up on the client.

    To fetch feature configuration, Unleash SDKs run a polling loop in the background. With normal-sized configurations this isn't an issue, but as you add more and more complex constraints, this can eventually overload your network and slow down your SDK. And the more SDKs that connect to your Unleash instance, the worse the problem gets.

    In other words: it's not good.

    Rethinking your options

    Okay, so putting all these IDs in Unleash isn't good. But how do you manage features for these 124 special cases you have?

    The first thing to think about would be whether you can group these special cases in some way. Maybe you already have an Unleash context field that covers the same amount of users. In that case, you can constrain on that instead. If you don't have a context field that matches these users, then you might need to create one.

    Further, if you have a lot of special cases and require complex constraint logic to model it correctly, this probably reflects some logic that is specific to your domain. It's also likely that this same logic is used elsewhere in your system external to Unleash. Modeling this logic in multiple places can quickly lead to breakage, and we recommend having a single source of truth for cases like this.

    Say, for instance that you have a VIP program where only your top 100 customers get access, and you want to use Unleash to manage access to these exclusive features. In that case, you probably have that list of 100 customers stored outside of Unleash (and if you don't, you definitely should: Unleash is not a data store). To solve this without using a constraint with 100 customer IDs, you could add a cusomerType field to the Unleash context, and only expose the features to users whose customerType is VIP.

    The external storage trick

    Continuing the customerType example above, how would you resolve a user's customerType in your application?

    If you have customers, then you need to store those customers' data somewhere. This is usually in a database. In some cases, you might already have all the information you need available when you first get the customer information (when they log in). If that's the case, you should be able to take the necessary info and populate the Unleash context with it directly.

    But what if it's more ephemeral or what if it's not available together with the customer data directly? Maybe it's something that changes dynamically?

    An option is to set up an external data store that handles this information specifically. As an example, consider a Redis instance that has a list of VIP customers. You could have an application connect to it and receive the latest state of VIP customers whenever you check a feature with Unleash. For an example of this, check out the Unleash and Redis example.

    Why doesn't Unleash support complex constraint setups out the box?

    The Unleash SDKs are designed to fast and unobtrusive. This means that resolving a large set of constraints at runtime results in one of two problems: either the SDK needs to resolve very large amounts of data, which can put pressure on your network or it needs to make a potentially slow network call to resolve the segment. Both of these are undesirable for the health of your application.


    1. Well, except for in the case of the front-end API. But even then, the size of the data to transmit matters.
    - - + + \ No newline at end of file diff --git a/understanding-unleash/proxy-hosting.html b/understanding-unleash/proxy-hosting.html index a38888433e..dabcd0dc76 100644 --- a/understanding-unleash/proxy-hosting.html +++ b/understanding-unleash/proxy-hosting.html @@ -20,8 +20,8 @@ - - + + @@ -33,7 +33,7 @@ Client API: For backend SDKs, Edge returns entire toggle payload for scope of token (project/environment)

    Start Edge & populate toggle cache

    • This initial command will populate toggle cache on startup using the client token specified in the environment variable:

    docker run -p 3063:3063 -e TOKENS='CLIENT_API_TOKEN' -e UPSTREAM_URL='UPSTREAM_URL' unleashorg/unleash-edge:v8.1 edge

    The following can be used to pass new tokens to Edge for different project/environment scopes:

    curl --location --request GET 'http://0.0.0.0:3063/api/client/features' \ --header 'Content-Type: application/json' \ --header 'Authorization: NEW_TOKEN' \ --data-raw ''

    On SDKs

    • Point frontend/client SDKs to Edge endpoints
      • Backend SDKs: /api/client
      • Frontend SDKs: /api/frontend, /api/proxy

    You host everything

    Availability

    This setup is only available open-source and Enterprise customers.

    An architecture diagram of using the proxy in a single-region, self-hosted setup.

    You host everything yourself. Everything runs where and how you configure it to.

    To configure Edge and the SDKs, follow steps in the previous section on Unleash hosts the API, you host Edge

    As you might expect, doing everything yourself is going to give you the most flexibility, but it's also going to be the most work. If you're already hosting Unleash yourself, though, this shouldn't be any more difficult than the previous section.

    As described in the section on tradeoffs between self-hosted and Unleash-hosted setups, running Edge yourself gives you a number of benefits.

    Multi-region

    An architecture diagram of using the proxy in a multi-region, self-hosted setup.

    You can also use Edge for a multi-region setup. You can run Edge in a different region to the API and still connect to the API. Because Edge runs closer to your applications it still provides you benefits in terms of quicker response times. Everything should be configured as described in the you host everything section or the Unleash hosts the API, you host Edge section. You can add as many Edge instances in as many extra regions as you want.

    Legacy: Unleash hosts the API, you host the Proxy

    Recommendation

    This approach is no longer recommended. You should use Unleash Edge instead of the Unleash proxy. If you are an existing Proxy user, see our Edge migration guide for a guide on how to migrate. Please take note of the section covering features Edge does not currently support in the same linked document before planning a migration.

    Availability

    This setup is only available to Pro and Enterprise customers.

    An architecture diagram of using the proxy in a setup where Unleash hosts the API and you host the proxy.

    You host the proxy yourself. It runs in a standalone container alongside your other applications in your cloud or hosting setup. Unleash manages the Unleash API, the admin UI, and the backing database in our AWS setup: the API and the UI run together in a Kubernetes deployment and connect to an Amazon RDS database.

    You'll need to configure the proxy and the proxy client SDKs.

    For the proxy, configure:

    For the proxy client SDK, configure:

    • One of the proxy client keys that you configured for the proxy.
    • The proxy's endpoint. This will depend on where and how you're hosting the proxy, but will typically end in "/proxy"

    This setup requires a little more setup on your part than the Unleash-hosted proxy does, but it offers all the benefits described in the section on tradeoffs between self-hosted and Unleash-hosted setups.

    - - + + \ No newline at end of file diff --git a/understanding-unleash/the-anatomy-of-unleash.html b/understanding-unleash/the-anatomy-of-unleash.html index 2109f64837..e21b77b213 100644 --- a/understanding-unleash/the-anatomy-of-unleash.html +++ b/understanding-unleash/the-anatomy-of-unleash.html @@ -20,15 +20,15 @@ - - + +

    The Anatomy of Unleash

    This guide's purpose is to give you a conceptual overview of how Unleash works. It covers the various components that exist within an Unleash system and how they interact with each other and with external applications. The diagrams are intended to help you understand the fundamental building blocks, such as projects, environments, variants and, of course, feature toggles.

    The end of this guide presents a short use case, explaining how you might configure Unleash to start working with feature toggles.

    The root level

    Some things in Unleash are configured and defined on the root level. These options apply across the entire Unleash instance. The most important root configuration options for day-to-day operations are:

    Projects

    Projects contain feature toggles and their configurations, and a set of active environments.

    All Unleash instances must have at least one project at any given time. New instances get a project called “Default”.

    Pro and Enterprise customers can create, rename, and delete projects as they wish (as long as there is always at least one project). Open-source users, on the other hand, only get access to the Default project.

    A square labeled 'project' containing another square, labeled 'environment'.
    Unleash projects contain one or more environments.

    Environments and project environments

    Feature toggles can be activated or deactivated independently in different environments. For instance, a feature toggle can be active in the development environment, and deactivated in the production environment. Even if their configuration is otherwise identical, deactivated feature toggles will never be considered enabled.

    Environments in Unleash let you change how a feature toggle works in your application’s different environments. For instance, while you are developing a feature, it’s likely that you’ll want it to be available in your development environment, but not in your production environment: environments let you do that. You might also want to enable a feature for only some users in your development environment, but no users in your production environment: environments let you do that.

    Environments exist on two different levels within Unleash. The set of all available environments is defined on the root level. Additionally, each project can choose which of these root environments should be available on the project level. The set of environments available to any given project is always a subset of the set of all available environments at the root level.

    Each project must always have at least one active environment.

    Enterprise users can create and remove environments. Open-source and Pro customers get access to two environments: development and production.

    Environments are adjacent to feature toggles in Unleash: neither one contains the other, but they come together to let you define activation strategies.

    You can use different activation strategies and constraints in different environments. For instance, you can show a feature only to select user IDs in development, but roll it out to 25% of your user base in production.
    Environments and API keys

    When connecting an SDK to Unleash, it's the API key that decides which environment to fetch features for. For legacy reasons, all Unleash SDKs accept a configuration option called environment, but this does not affect the environment at all. It is an Unleash context field and a holdover from before Unleash had native environments.

    Features (feature toggles)

    Feature toggles are at the heart of Unleash’s functionality. Feature toggles belong to projects and live next to project environments. In and of itself, a feature toggle doesn’t do anything. You must assign activation strategies to it for it to start taking effect.

    When creating a feature toggle, you must assign a unique (across your Unleash instance) name, a feature toggle type, a project it belongs to, and an optional description. Everything except for the name can be changed later.

    A hierarchy showing a project containing an environment containing a feature toggle configuration.
    Feature toggle states are evaluated independently in each environment.

    Activation strategies

    A hierarchy displaying an environment containing a feature toggle configuration with an activation strategy.
    Activation strategies are applied to feature toggles on a per-environment basis and decide whether a feature is enabled or not.

    Activation strategies (or just strategies for short) are the part of feature toggles that tell Unleash who should get a feature. An activation strategy is assigned to one feature toggle in one environment.

    When you check a feature toggle in an application, the following decides the result:

    1. Is the toggle active in the current environment? If not, it will be disabled.
    2. If the toggle is active in the current environment, the toggle’s strategies decide the result. As long as at least one of a toggle’s strategies resolve to true for the current context (user or application), then the toggle will be considered enabled. In other words, if you have a hundred strategies and ninety-nine of them resolve to false, but one of them resolves to true, then the toggle is enabled.

    Activation strategies tie feature toggles and environments together. When you assign an activation strategy to a feature toggle, you do so in one environment at a time. You can assign the same strategy to the same toggle in different environments, but they will be different instances of the same strategy, and do not stay in sync. Unleash also lets you copy strategies from one environment to another.

    Unleash comes with a number of strategies built in (refer the activation strategies documentation for more information on those). You can also create your own custom activation strategies if you need them. All strategies can be further augmented by strategy constraints.

    Feature toggles exist across environments and can have different activation strategies in each environment.
    Feature toggle activation strategies can be copied between environments. You can also create new strategies in each environment.

    Strategy constraints

    Strategy constraints (or just constraints) help you fine-tune your strategies. They are an extra layer of prerequisites that help you narrow the audience of a strategy down. Strategy constraints are applied to activation strategies.

    For example, if you wanted to roll a feature out to 50% of users with a specific email domain (such as “@mycompany.com”), then strategy constraints would let you target only users with that email domain.

    Constraints can also be used for more general purposes, such as timing feature releases or releasing features in specific regions.

    An activation strategy can have as many constraints as you want. When an activation strategy has multiple constraints, then every constraint must be satisfied for the strategy to be evaluated. So if you have two constraints: one that says users must have an “@mycompany.com” email address and one that says users must have signed up for a beta program, then the strategy would only be evaluated for users with @mycompany.com emails that have signed up for the program.

    Strategies and constraints

    Feature toggle strategies are permissive: As long as one strategy resolves to true, the feature is considered enabled. On the other hand, constrains are restrictive: for a given strategy, all constraints must be met for it to resolve to true.

    We can exemplify this difference with the logical operators AND and OR:

    • For a feature toggle, if Strategy1 OR Strategy2 OR .. OR StrategyN is true, then the feature is enabled.
    • For a strategy, it can be evaluated if and only if Constraint1 AND Constraint2 AND .. AND ConstraintN are met.

    Note that even if all the constraints are met, the strategy itself might not resolve to true: that will depend on the strategy and the provided context.

    You can define constraints on whatever properties you want in your Unleash context.

    Constraints are applied to individual strategies and do not stay in sync with each other. When you need to have the same constraints applied to multiple strategies and need those constraints to stay in sync, use segments.

    A hierarchy drawing showing a constraint applied to an activation strategy.
    Constraints can be applied to strategies, allowing you to narrow a feature's audience.

    Segments

    Segments add extra functionality on top of strategy constraints. A segment is a reusable collection of strategy constraints with a name and an optional description. When you apply a segment to a strategy, the strategy will be evaluated as if all of the segment's constraints were applied to it.

    Segments let you apply a set of constraints to multiple strategies and keep the constraints in sync between those strategies. Whenever you apply a segment to a strategy, you essentially create a reference to that segment. This means that whenever you change the segment by adding, removing, or changing constraints, this change propagates to all the strategies that reference this segment.

    You can apply multiple segments to a strategy. Much like with constraints, every segment needs every constraint to be satisfied for the strategy to be evaluated. If you also have other constraints on the strategy, then those must also be satisfied.

    Segments are only available to Pro and Enterprise users.

    Segments are reusable lists of constraints that can be applied to a strategy instead of or in addition to constraints.

    Variants and feature toggle payloads

    By default, a feature toggle in Unleash only tells you whether a feature is enabled or disabled, but you can also add more information to your toggles by using feature toggle variants. Variants also allow you to run A/B testing experiments.

    Feature toggles are designed to let you decide which users get access to a feature. Variants are designed to let you decide which version of the feature a user gets access to. For instance, if user A is part of your beta testing program and gets access to a new beta feature, then you can use variants to decide whether they should get the red version or the green version of the feature.

    When you create new variants for a feature, they must be given a name and a weighting indicating how many users should see this particular variant of the feature. They can also be given a payload.

    You can use the variant payload to attach arbitrary data to a variant. Variants can have different kinds of payloads.

    A feature toggle can have as many variants as you want.

    Variants and environments

    Prior to 4.21, variants were independent of environments. In other words: if you're on 4.19 or lower, you’ll always have the exact same variants with the exact same weightings and the exact same payloads in all environments.

    Before Unleash 4.21, feature toggle variants were the same for all environments.

    As of version 4.21, a feature can have different variants in different environments. For instance, a development environment might have no variants, while a production environment has 2 variants. Payloads, weightings and anything else can also differ between environments.

    From Unleash 4.21 on, a feature toggle can have different in each environment.

    Use case: changing website colors

    Using the concepts we have looked at in the previous sections, let’s create a hypothetical case and see how Unleash would solve it.

    Problem statement: You have an existing website with a red color scheme, but you’re feeling a bit adventurous and would like to try and see if changing it to a blue color scheme would be better.

    Current state: You have an existing website that gets server-side rendered and you have a newly created instance of Unleash.

    Configuring Unleash for development

    Assuming you have a brand new Unleash instance, you already have the “Default” project and the “Development” and “Production” environments available. That’s going to be all you need for now.

    First things first, in the Default project, you create a new feature toggle, called “new-color-scheme” (toggle names have to be URL-friendly, so no spaces allowed!).

    Because you’d like to see the new color scheme while you’re developing it, you assign a “standard” strategy to the new-color-scheme toggle in the development environment and turn it on.

    In your application

    You configure an Unleash SDK for your server to communicate with Unleash. When rendering the page, you check the state of the new-color-scheme feature and render a different stylesheet based on the results.

    In pseudocode (loosely based on the Node.js SDK), that might look like this:

    if (unleash.isEnabled(new-color-scheme”)) {
    // load stylesheet with new color scheme
    } else {
    // load stylesheet with old color scheme
    }

    And with that, the new color scheme is now live in your development environment. Because there aren’t any strategies defined in the production environment yet, the feature is not active, and everything is as it was.

    Rolling out the feature to users

    When you’re happy with the new color scheme, you decide to start rolling it out to users. But you want it to go out to only a small number of users at first, so that you can get some feedback while rolling out.

    You decide to add a gradual rollout strategy to the new-color-scheme feature in the production environment. Because you want to start small, you set the rollout percentage to 5%.

    As soon as you enable the production environment, the feature gets rolled out to 5% of your users (assuming you’ve deployed the code to production).

    Adding variants

    While you were developing the new color scheme, you also dabbled a bit with other colors in addition to blue: green and purple might be nice too! So you decide to create two extra color schemes that you’re happy with. But you’d like to hear what your users think too, so you need to roll it out to them somehow.

    You decide to use feature toggle variants to differentiate between the different themes, creating three variants: blue, green, and purple. You want each of them to roll out to the same number of users, so you leave them equally weighted.


    const theme = unleash.getVariant(new-color-scheme”).name;

    if (theme === “green”) {
    // load stylesheet with green color scheme
    } else if (theme === “blue”) {
    // load stylesheet with blue color scheme
    } else if (theme === “purple”) {
    // load stylesheet with purple color scheme
    } else {
    // load stylesheet with old color scheme
    }

    Now users that are included in the gradual rollout will get one of the three themes. Users that aren’t included get the old theme.

    - - + + \ No newline at end of file diff --git a/understanding-unleash/unleash-overview.html b/understanding-unleash/unleash-overview.html index fc35c58afc..eb1a667453 100644 --- a/understanding-unleash/unleash-overview.html +++ b/understanding-unleash/unleash-overview.html @@ -20,15 +20,15 @@ - - + +

    Unleash introductory overview

    One of the most important aspects of the architecture to understand is that feature toggles are evaluated in a client SDKs which runs as part of your application. This makes toggle evaluations super-fast (we're talking nano-seconds), scalable and resilient against network disturbances. In order to achieve this Unleash compromises a small update-delay when you change your toggle configurations until it is fully propagated to your application (in terms of seconds and is configurable).

    If you want more details you can read about our unique architecture.

    Unleash Server

    Before you can connect your application to Unleash you need a Unleash server. You have a few options available:

    1. Unleash Open-Source
    2. Unleash Enterprise

    System Overview

    A visual overview of an Unleash system as described in the following paragraph.

    • The Unleash API - The Unleash instance. This is where you create feature toggles, configure activation strategies, and parameters, etc. The service holding all feature toggles and their configurations. Configurations declare which activation strategies to use and which parameters they should get.
    • The Unleash admin UI - The bundled web interface for interacting with the Unleash instance. Manage toggles, define strategies, look at metrics, and much more. Use the UI to create feature toggles, manage project access roles, create API tokens, and more.
    • Unleash SDKs - Unleash SDKs integrate into your applications and get feature configurations from the Unleash API. Use them to check whether features are enabled or disabled and to send metrics to the Unleash API. See all our SDKs
    • The Unleash Edge - The Unleash Edge sits between front-end and native applications and the Unleash API, it can also sit between server-side SDKs and the Unleash API as well. You can scale it independently of the Unleash API to handle large request rates without causing issues for the Unleash API. Edge has all endpoints for the client API, frontend API, and proxy API.

    To be super fast (we're talking nano-seconds), the client SDK caches all feature toggles and their current configuration in memory. The activation strategies are also implemented in the SDK. This makes it really fast to check if a toggle is on or off because it is just a simple function operating on local state, without the need to poll data from the database.

    - - + + \ No newline at end of file diff --git a/unleash-academy/advanced-for-devs.html b/unleash-academy/advanced-for-devs.html index b77b64c28c..10694c55d6 100644 --- a/unleash-academy/advanced-for-devs.html +++ b/unleash-academy/advanced-for-devs.html @@ -20,8 +20,8 @@ - - + + @@ -30,7 +30,7 @@

    Advanced for Developers

    info

    This Unleash Academy course is for all developer roles working with Unleash, after Foundational content has been reviewed.

    note

    Estimated time burden: 45 minutes


    Learning Objectives

    Unlock the full potential of Unleash, streamline your development process and make data-driven decisions with confidence. Learn how Unleash can support your business needs and understand advanced usage like collaboration, data insights and A/B testing, auto-remediation, and integration into third party tools.

    You’ll master advanced use cases and implement best practices - this course will empower you to maximize Unleash's capabilities for your projects.


    Course Detail

    Embedded Player

    The full course is shown above.
    Click the icon in the top right corner of the embedded player to view your progress as you work through the videos.
    Options to go full screen, view the playlist on YouTube or share are also enabled.

    - - + + \ No newline at end of file diff --git a/unleash-academy/foundational.html b/unleash-academy/foundational.html index 59ef34c3b3..99c84f837a 100644 --- a/unleash-academy/foundational.html +++ b/unleash-academy/foundational.html @@ -20,8 +20,8 @@ - - + + @@ -31,7 +31,7 @@ An understanding of Unleash anatomy and architecture and how the different systems connect together.


    Intro to Feature Flags/Toggles & Unleash

    • Feature flags - What they are & why to use them
    • What is Unleash?

    Anatomy of Unleash

    • Covering the various components that exist within the Unleash system and how they interact with each other and with external applications. Components that include but not limited to: projects, environments, variants, feature toggles --> strategies, tokens, tags, context

    Architecture overview

    • Unleash API, UI, SDKs, Proxy --> Edge (Proxy 2.0)

    How to use Unleash

    • Projects and Environments
    • Toggle creation and targeting
    • Strategies, constraints, segments
    • Variants (and environments), Strategy Variants
    • Introduction to A/B Testing
    • API tokens (Personal Access Tokens, client, frontend)

    Course Detail

    Embedded Player

    The full course is shown above.
    Click the icon in the top right corner of the embedded player to view your progress as you work through the videos.
    Options to go full screen, view the playlist on YouTube or share are also enabled.

    - - + + \ No newline at end of file diff --git a/unleash-academy/introduction.html b/unleash-academy/introduction.html index 8c695ddf92..a7974f9c17 100644 --- a/unleash-academy/introduction.html +++ b/unleash-academy/introduction.html @@ -20,8 +20,8 @@ - - + + @@ -29,7 +29,7 @@

    Unleash Academy

    Introduction

    Unleash Academy is your go-to portal for complimentary self-paced training materials around the Unleash platform.

    Whether you are new to feature flags or a seasoned feature management user and whatever your organizational role, our materials will help onboard you to the full suite of capabilities the Unleash platform has to offer, refresh your existing knowledge, broaden your knowledge to new use cases, and much more!

    Any Unleash user is welcomed - from Open Source, Pro to Enterprise.

    Content is carefully curated to ensure the best, most relevant targeted learning experience. Review the next section on how to use the content to get started today!

    Learning Paths and How to use this content

    Content is built around Learning Paths that are based on common user roles and personas using Unleash today. This helps ensure that you maximize value from the time you spend with Unleash Academy by reviewing content that will help you achieve your goals with Unleash.

    1. Start by identifying the persona most closely associated with your day to day responsibilities.
    Example titles are provided for additional guidance:

    • Developer
      • E.g Software Engineer, Software Developer
    • DevOps / Admin
      • Platform Engineer, Site Reliability Engineer, DevOps Engineer, Systems Administrator, Cloud Infrastructure Consultant
    • Product Owner
      • Product Manager, Product Analyst
    • People Leader
      • Manager, Executive
    note

    All roles working with Unleash start with the Foundational training. Then, role dependent courses are offered thereafter.

    2. Now check the course directory or the graphic below to find out which courses apply to your persona!

    Course Directory

    Directory by Persona


    Select the tab that corresponds to your persona. A course list is shown - plan to complete the courses in the displayed order, noting the estimated completion times.

    info

    See also the following visualization of the learning paths

    Course order for Developer, DevOps, Product and Leader personas
    Learning Paths organized by Persona

    Directory by Course

    - - + + \ No newline at end of file diff --git a/unleash-academy/managing-unleash-for-devops.html b/unleash-academy/managing-unleash-for-devops.html index 3a2fce7268..eaeadbc738 100644 --- a/unleash-academy/managing-unleash-for-devops.html +++ b/unleash-academy/managing-unleash-for-devops.html @@ -20,8 +20,8 @@ - - + + @@ -30,7 +30,7 @@

    Managing Unleash for DevOps/Admins

    info

    This Unleash Academy course is for all DevOps and Admin roles working with Unleash, after Foundational content has been reviewed.

    note

    Estimated time burden: 45 minutes


    Learning Objectives

    Understand how to deploy Unleash in a secure and performant manner through Edge, maintain and manage users, groups, permissions, and RBAC.


    Course Detail

    Embedded Player

    The full course is shown above.
    Click the icon in the top right corner of the embedded player to view your progress as you work through the videos.
    Options to go full screen, view the playlist on YouTube or share are also enabled.

    - - + + \ No newline at end of file diff --git a/using-unleash.html b/using-unleash.html index c4f5344158..6b38520080 100644 --- a/using-unleash.html +++ b/using-unleash.html @@ -20,15 +20,15 @@ - - + + - - + + + \ No newline at end of file diff --git a/using-unleash/deploy.html b/using-unleash/deploy.html index b1c2ba1772..ef92ff26a5 100644 --- a/using-unleash/deploy.html +++ b/using-unleash/deploy.html @@ -20,15 +20,15 @@ - - + +
    -
    - - +

    Self-Hosting Unleash

    All you need to learn how to deploy and manage your own Unleash instance.

    + + \ No newline at end of file diff --git a/using-unleash/deploy/configuring-unleash-v3.html b/using-unleash/deploy/configuring-unleash-v3.html index 9744fe0b5c..828480cdc7 100644 --- a/using-unleash/deploy/configuring-unleash-v3.html +++ b/using-unleash/deploy/configuring-unleash-v3.html @@ -20,15 +20,15 @@ - - + +

    Configuring Unleash v3

    This is the guide on how to configure Unleash v3 self-hosted. If you are using Unleash v4 you should checkout configuring Unleash

    In order to customize "anything" in Unleash you need to use Unleash from Node.js:

    const unleash = require('unleash-server');

    const unleashOptions = {
    db: {
    user: 'unleash_user',
    password: 'passord',
    host: 'localhost',
    port: 5432,
    database: 'unleash',
    ssl: false,
    pool: {
    min: 0,
    max: 4,
    idleTimeoutMillis: 30000,
    },
    },
    enableRequestLogger: true,
    };

    unleash.start(unleashOptions);

    Available Unleash options include:

    • db - The database configuration object taking the following properties:
      • user - the database username (DATABASE_USERNAME)
      • password - the database password (DATABASE_PASSWORD)
      • host - the database hostname (DATABASE_HOST)
      • port - the database port defaults to 5432 (DATABASE_PORT)
      • database - the database name to be used (DATABASE_NAME)
      • ssl - an object describing ssl options, see https://node-postgres.com/features/ssl (DATABASE_SSL, as a stringified json object)
      • version - the postgres database version. Used to connect a non-standard database. Defaults to undefined, which let the underlying adapter to detect the version automatically. (DATABASE_VERSION)
      • pool - an object describing pool options, see https://knexjs.org/guide/#pool. We support the following four fields:
        • min - minimum connections in connections pool (defaults to 0) (DATABASE_POOL_MIN)
        • max - maximum connections in connections pool (defaults to 4) (DATABASE_POOL_MAX)
        • idleTimeoutMillis - time in milliseconds a connection must be idle before being marked as a candidate for eviction (defaults to 30000) (DATABASE_POOL_IDLE_TIMEOUT_MS)
        • afterCreate - a callback for for configuring active connections, see https://knexjs.org/guide/#aftercreate. This is incompatible with the ALLOW_NON_STANDARD_DB_DATES environment variable, which will override this property to support non-standard Postgres date formats. If you've set your Postgres instance to use a date style other than ISO, DMY then you'll need to set the ALLOW_NON_STANDARD_DB_DATES environment variable to true. Setting the environment variable should be preferred over writing your own callback.
    • databaseUrl - (deprecated) the postgres database url to connect to. Only used if db object is not specified, and overrides the db object and any environment variables that change parts of it (like DATABASE_SSL). Should include username/password. This value may also be set via the DATABASE_URL environment variable. Alternatively, if you would like to read the database url from a file, you may set the DATABASE_URL_FILE environment variable with the full file path. The contents of the file must be the database url exactly.
    • databaseSchema - the postgres database schema to use. Defaults to 'public'. (DATABASE_SCHEMA)
    • port - which port the unleash-server should bind to. If port is omitted or is 0, the operating system will assign an arbitrary unused port. Will be ignored if pipe is specified. This value may also be set via the HTTP_PORT environment variable
    • host - which host the unleash-server should bind to. If host is omitted, the server will accept connections on the unspecified IPv6 address (::) when IPv6 is available, or the unspecified IPv4 address (0.0.0.0) otherwise. This value may also be set via the HTTP_HOST environment variable
    • pipe - parameter to identify IPC endpoints. See https://nodejs.org/api/net.html#net_identifying_paths_for_ipc_connections for more details
    • enableLegacyRoutes (boolean) - allows you to turn on/off support for legacy routes to support older clients. Disabled by default. Will be removed in 4.x.
    • serverMetrics (boolean) - use this option to turn on/off prometheus metrics.
    • preHook (function) - this is a hook if you need to provide any middlewares to express before unleash adds any. Express app instance is injected as first argument.
    • preRouterHook (function) - use this to register custom express middlewares before the unleash specific routers are added. This is typically how you would register custom middlewares to handle authentication.
    • adminAuthentication (string) - use this when implementing custom admin authentication securing-unleash. Possible values are:
      • none - will disable authentication altogether
      • unsecure - (default) will use simple cookie based authentication. UI will require the user to specify an email in order to use unleash.
      • custom - use this when you implement your own custom authentication logic.
    • ui (object) - Set of UI specific overrides. You may set the following keys: headerBackground, environment, slogan.
    • getLogger (function) - Used to register a custom log provider.
    • eventHook (function(event, data)) - If provided, this function will be invoked whenever a feature is mutated. The possible values for event are 'feature-created', 'feature-updated', 'feature-archived', 'feature-revived'. The data argument contains information about the mutation. Its fields are type (string) - the event type (same as event); createdBy (string) - the user who performed the mutation; data - the contents of the change. The contents in data differs based on the event type; For 'feature-archived' and 'feature-revived', the only field will be name - the name of the feature. For 'feature-created' and 'feature-updated' the data follows this schema defined in the source code. See an api here.
    • baseUriPath (string) - use to register a base path for all routes on the application. For example /my/unleash/base (note the starting /). Defaults to /. Can also be configured through the environment variable BASE_URI_PATH.
    • unleashUrl (string) - Used to specify the official URL this instance of Unleash can be accessed at for an end user. Can also be configured through the environment variable UNLEASH_URL.
    • secureHeaders (boolean) - use this to enable security headers (HSTS, CSP, etc) when serving Unleash from HTTPS. Can also be configured through the environment variable SECURE_HEADERS.
    • checkVersion - the checkVersion object deciding where to check for latest version
      • url - The url to check version (Defaults to https://version.unleash.run) - Overridable with (UNLEASH_VERSION_URL)
      • enable - Whether version checking is enabled (defaults to true) - Overridable with (CHECK_VERSION) (if anything other than true, does not check)

    Disabling Auto-Start

    If you're using Unleash as part of a larger express app, you can disable the automatic server start by calling server.create. It takes the same options as server.start, but will not begin listening for connections.

    const unleash = require('unleash-server');
    // ... const app = express();

    unleash
    .create({
    databaseUrl: 'postgres://unleash_user:password@localhost:5432/unleash',
    port: 4242,
    })
    .then((result) => {
    app.use(result.app);
    console.log(`Unleash app generated and attached to parent application`);
    });

    Securing Unleash

    You can integrate Unleash with your authentication provider (OAuth 2.0). Read more about securing unleash.

    How do I configure the log output?

    By default, unleash uses log4js to log important information. It is possible to swap out the logger provider (only when using Unleash programmatically). You do this by providing an implementation of the getLogger function as This enables filtering of log levels and easy redirection of output streams.

    function getLogger(name) {
    // do something with the name
    return {
    debug: console.log,
    info: console.log,
    warn: console.log,
    error: console.error,
    };
    }

    The logger interface with its debug, info, warn and error methods expects format string support as seen in debug or the JavaScript console object. Many commonly used logging implementations cover this API, e.g., bunyan, pino or winston.

    Database pooling connection timeouts

    • Please be aware of the default values of connection pool about idle session handling.
    • If you have a network component which closes idle sessions on tcp layer, please ensure, that the connection pool idleTimeoutMillis setting is lower than the timespan as the network component will close the idle connection.
    - - + + \ No newline at end of file diff --git a/using-unleash/deploy/configuring-unleash.html b/using-unleash/deploy/configuring-unleash.html index 358f1fc592..dbbbb70a9f 100644 --- a/using-unleash/deploy/configuring-unleash.html +++ b/using-unleash/deploy/configuring-unleash.html @@ -20,16 +20,16 @@ - - + +
    -

    Configuring Unleash

    This is the guide on how to configure Unleash v4 self-hosted. If you are still using Unleash v3 you should checkout configuring Unleash v3

    Must configure

    Database

    In order for Unleash server to work, you need a running database and its connection details. See the database configuration section for the available options and configuration details.

    Nice to configure

    Unleash URL

    • Configured with UNLEASH_URL Should be set to the public discoverable URL of your instance, so if your instance is accessed by your users at https://unleash.mysite.com use that. If you're deploying this to a subpath, include the subpath in this. So https://mysite.com/unleash will also be correct.
    • Used to create
      • Reset password URLs
      • Welcome link for new users
      • Links in events for our Slack, Microsoft Teams and Datadog integrations

    Email server details

    Used to send reset-password mails and welcome mails when onboarding new users.
    NOTE - If this is not configured, you will not be able to allow your users to reset their own passwords.

    For more details, see here

    Further customization

    In order to customize "anything" in Unleash you need to use Unleash from Node.js or start the docker image with environment variables.

    const unleash = require('unleash-server');

    const unleashOptions = {
    db: {
    user: 'unleash_user',
    password: 'password',
    host: 'localhost',
    port: 5432,
    database: 'unleash',
    ssl: false,
    pool: {
    min: 0,
    max: 4,
    idleTimeoutMillis: 30000,
    },
    },
    enableRequestLogger: true,
    };

    unleash.start(unleashOptions);

    Available Unleash options include:

    • authentication - (object) - An object for configuring/implementing custom admin authentication

      • enableApiToken / AUTH_ENABLE_API_TOKEN: boolean — Should unleash require API tokens for access? Defaults to true.

      • type / AUTH_TYPE: string — What kind of authentication to use. Possible values

        • open-source - Sign in with username and password. This is the default value.
        • custom - If implementing your own authentication hook, use this
        • none - Turn off authentication all together
        • demo - Only requires an email to sign in (was default in v3)
      • customAuthHandler: function (app: any, config: IUnleashConfig): void — custom express middleware handling authentication. Used when type is set to custom. Can not be set via environment variables.

      • initialAdminUser: { username: string, password: string} | null — whether to create an admin user with default password - Defaults to using admin and unleash4all as the username and password. Can not be overridden by setting the UNLEASH_DEFAULT_ADMIN_USERNAME and UNLEASH_DEFAULT_ADMIN_PASSWORD environment variables.

        • initApiTokens / INIT_ADMIN_API_TOKENS, INIT_CLIENT_API_TOKENS, and INIT_FRONTEND_API_TOKENS: ApiTokens[] — Array of API tokens to create on startup. The tokens will only be created if the database doesn't already contain any API tokens. Example:
        [
        {
        environment: '*',
        project: '*',
        secret: '*:*.964a287e1b728cb5f4f3e0120df92cb5',
        type: ApiTokenType.ADMIN,
        username: 'some-user',
        },
        ];

        The tokens can be of any API token type. Note that admin tokens must target all environments and projects (i.e. use '*' for environments and project and start the secret with *:*.).

        You can also use the environment variables INIT_ADMIN_API_TOKENS, INIT_CLIENT_API_TOKENS, and INIT_FRONTEND_API_TOKENS. All variables require a comma-separated list of API tokens to initialize (for instance *:*.some-random-string, *:*.some-other-token). The tokens found in INIT_ADMIN_API_TOKENS, INIT_CLIENT_API_TOKENS, and INIT_FRONTEND_API_TOKENS will be created as admin, client, and frontend tokens respectively, and Unleash will assign usernames automatically.

    • databaseUrl - (deprecated) the postgres database url to connect to. Only used if db object is not specified, and overrides the db object and any environment variables that change parts of it (like DATABASE_SSL). Should include username/password. This value may also be set via the DATABASE_URL environment variable. Alternatively, if you would like to read the database url from a file, you may set the DATABASE_URL_FILE environment variable with the full file path. The contents of the file must be the database url exactly.

    • db - The database configuration object. See the database configuration section for a full overview of the available properties.

    • disableLegacyFeaturesApi (boolean) - whether to disable the legacy features API. Defaults to false (DISABLE_LEGACY_FEATURES_API). Introduced in Unleash 4.6.

    • email - the email object configuring an SMTP server for sending welcome mails and password reset mails

      • host - The server URL to your SMTP server
      • port - Which port the SMTP server is running on. Defaults to 465 (Secure SMTP)
      • secure (boolean) - Whether to use SMTPS or not.
      • sender - Which email should be set as sender of mails being sent from Unleash?
      • smtpuser - Username for your SMTP server
      • smtppass - Password for your SMTP server
    • eventHook (function(event, data)) - (deprecated in Unleash 4.3 in favor of the Webhook integration. Removed in Unleash 5) If provided, this function will be invoked whenever a feature is mutated. The possible values for event are 'feature-created', 'feature-archived' and 'feature-revived'. The data argument contains information about the mutation. Its fields are type (string) - the event type (same as event); createdBy (string) - the user who performed the mutation; data - the contents of the change. The contents in data differs based on the event type; For 'feature-archived' and 'feature-revived', the only field will be name - the name of the feature. For 'feature-created' the data follows a schema defined in the code here. See an api here.

    • getLogger (function) - Used to register a custom log provider.

    • logLevel (debug | info | warn | error | fatal) - The lowest level to log at, also configurable using environment variable LOG_LEVEL.

    • enableRequestLogger (boolean) - use this to enable logging for requested urls and response codes (default: false).

    • preHook (function) - this is a hook if you need to provide any middlewares to express before unleash adds any. Express app instance is injected as first argument.

    • preRouterHook (function) - use this to register custom express middlewares before the unleash specific routers are added.

    • secureHeaders (boolean) - use this to enable security headers (HSTS, CSP, etc) when serving Unleash from HTTPS. Can also be configured through the environment variable SECURE_HEADERS.

    • additionalCspAllowedDomains (CspAllowedDomains) - use this when you want to enable security headers but have additional domains you need to allow traffic to. You can set the following environment variables:

      • CSP_ALLOWED_DEFAULT to allow new defaultSrc (comma separated list)
      • CSP_ALLOWED_FONT to allow new fontSrc (comma separated list)
      • CSP_ALLOWED_STYLE to allow new styleSrc (comma separated list)
      • CSP_ALLOWED_SCRIPT to allow new scriptSrc (comma separated list)
      • CSP_ALLOWED_IMG to allow new imgSrc (comma separated list)
      • CSP_ALLOWED_CONNECT to allow new connectSrc (comma separated list)
      • CSP_ALLOWED_FRAME to allow new frameSrc (comma separated listed)
      • CSP_ALLOWED_MEDIA to allow new mediaSrc (comma separated list)
      • CSP_ALLOWED_OBJECT to allow new objectSrc (comma separated list)
    • server - The server config object taking the following properties

      • port - which port the unleash-server should bind to. If port is omitted or is 0, the operating system will assign an arbitrary unused port. Will be ignored if pipe is specified. This value may also be set via the HTTP_PORT environment variable
      • host - which host the unleash-server should bind to. If host is omitted, the server will accept connections on the unspecified IPv6 address (::) when IPv6 is available, or the unspecified IPv4 address (0.0.0.0) otherwise. This value may also be set via the HTTP_HOST environment variable
      • pipe - parameter to identify IPC endpoints. See https://nodejs.org/api/net.html#net_identifying_paths_for_ipc_connections for more details
      • serverMetrics (boolean) - use this option to turn on/off prometheus metrics.
      • baseUriPath (string) - use to register a base path for all routes on the application. For example /my/unleash/base (note the starting /). Defaults to /. Can also be configured through the environment variable BASE_URI_PATH.
      • unleashUrl (string) - Used to specify the official URL this instance of Unleash can be accessed at for an end user. Can also be configured through the environment variable UNLEASH_URL.
      • gracefulShutdownEnable: (boolean) - Used to control if Unleash should shutdown gracefully (close connections, stop tasks,). Defaults to true. GRACEFUL_SHUTDOWN_ENABLE
      • gracefulShutdownTimeout: (number) - Used to control the timeout, in milliseconds, for shutdown Unleash gracefully. Will kill all connections regardless if this timeout is exceeded. Defaults to 1000ms GRACEFUL_SHUTDOWN_TIMEOUT
    • session - The session config object takes the following options:

      • ttlHours (number) - The number of hours a user session is allowed to live before a new sign-in is required. Defaults to 48 hours. SESSION_TTL_HOURS
      • clearSiteDataOnLogout (boolean) - When true, a logout action will return a Clear Site Data response header instructing the browser to clear all cookies on the same domain Unleash is running on. If disabled unleash will only destroy and clear the session cookie. Defaults to true. SESSION_CLEAR_SITE_DATA_ON_LOGOUT
      • cookieName - Name of the cookies used to hold the session id. Defaults to 'unleash-session'.
    • ui (object) - Set of UI specific overrides. You may set the following keys: environment, slogan.

    • versionCheck - the object deciding where to check for latest version

      • url - The url to check version (Defaults to https://version.unleash.run) - Overridable with (UNLEASH_VERSION_URL)
      • enable - Whether version checking is enabled (defaults to true) - Overridable with (CHECK_VERSION) (if anything other than true, does not check)
    • environmentEnableOverrides - A list of environment names to force enable at startup. This is feature should be used with caution. When passed a list, this will enable each environment in that list and disable all other environments. You can't use this to disable all environments, passing an empty list will do nothing. If one of the given environments is not already enabled on startup then it will also enable projects and toggles for that environment. Note that if one of the passed environments doesn't already exist this will do nothing aside from log a warning.

    • clientFeatureCaching - configuring memoization of the /api/client/features endpoint

      • enabled - set to true by default - Overridable with (CLIENT_FEATURE_CACHING_ENABLED)
      • maxAge - the time to cache features, set to 600 milliseconds by default - Overridable with (CLIENT_FEATURE_CACHING_MAXAGE) ) (accepts milliseconds)
    • frontendApi - Configuration options for the Unleash front-end API.

      • refreshIntervalInMs - how often (in milliseconds) front-end clients should refresh their data from the cache. Overridable with the FRONTEND_API_REFRESH_INTERVAL_MS environment variable.
    • accessControlMaxAge - You can configure the max-age of the Access-Control-Max-Age header. Defaults to 86400 seconds. Overridable with the ACCESS_CONTROL_MAX_AGE environment variable.

    • responseTimeWithAppNameKillSwitch - use this to disable metrics with app names. This is enabled by default but may increase the cardinality of metrics causing Unleash memory usage to grow if your app name is randomly generated (which is not recommended). Overridable with the UNLEASH_RESPONSE_TIME_WITH_APP_NAME_KILL_SWITCH environment variable.

    • disableCompression - Disables Express compression middleware, useful when configuring the application in a serverless environment. Defaults to false. Overridable with the SERVER_DISABLE_COMPRESSION environment variable.

    • keepAliveTimeout - Use this to tweak connection keepalive timeout in seconds. Useful for hosted situations where you need to make sure your connections are closed before terminating the instance. Defaults to 15. Overridable with the SERVER_KEEPALIVE_TIMEOUT environment variable. +

      Configuring Unleash

      This is the guide on how to configure Unleash v4 self-hosted. If you are still using Unleash v3 you should checkout configuring Unleash v3

      Must configure

      Database

      In order for Unleash server to work, you need a running database and its connection details. See the database configuration section for the available options and configuration details.

      Nice to configure

      Unleash URL

      • Configured with UNLEASH_URL Should be set to the public discoverable URL of your instance, so if your instance is accessed by your users at https://unleash.mysite.com use that. If you're deploying this to a subpath, include the subpath in this. So https://mysite.com/unleash will also be correct.
      • Used to create
        • Reset password URLs
        • Welcome link for new users
        • Links in events for our Slack, Microsoft Teams and Datadog integrations

      Email server details

      Used to send reset-password mails and welcome mails when onboarding new users.
      NOTE - If this is not configured, you will not be able to allow your users to reset their own passwords.

      For more details, see here

      Further customization

      In order to customize "anything" in Unleash you need to use Unleash from Node.js or start the docker image with environment variables.

      const unleash = require('unleash-server');

      const unleashOptions = {
      db: {
      user: 'unleash_user',
      password: 'password',
      host: 'localhost',
      port: 5432,
      database: 'unleash',
      ssl: false,
      pool: {
      min: 0,
      max: 4,
      idleTimeoutMillis: 30000,
      },
      },
      enableRequestLogger: true,
      };

      unleash.start(unleashOptions);

      Available Unleash options include:

      • authentication - (object) - An object for configuring/implementing custom admin authentication

        • enableApiToken / AUTH_ENABLE_API_TOKEN: boolean — Should unleash require API tokens for access? Defaults to true.

        • type / AUTH_TYPE: string — What kind of authentication to use. Possible values

          • open-source - Sign in with username and password. This is the default value.
          • custom - If implementing your own authentication hook, use this
          • none - Turn off authentication all together
          • demo - Only requires an email to sign in (was default in v3)
        • customAuthHandler: function (app: any, config: IUnleashConfig): void — custom express middleware handling authentication. Used when type is set to custom. Can not be set via environment variables.

        • initialAdminUser: { username: string, password: string} | null — whether to create an admin user with default password - Defaults to using admin and unleash4all as the username and password. Can not be overridden by setting the UNLEASH_DEFAULT_ADMIN_USERNAME and UNLEASH_DEFAULT_ADMIN_PASSWORD environment variables.

          • initApiTokens / INIT_ADMIN_API_TOKENS, INIT_CLIENT_API_TOKENS, and INIT_FRONTEND_API_TOKENS: ApiTokens[] — Array of API tokens to create on startup. The tokens will only be created if the database doesn't already contain any API tokens. Example:
          [
          {
          environment: '*',
          project: '*',
          secret: '*:*.964a287e1b728cb5f4f3e0120df92cb5',
          type: ApiTokenType.ADMIN,
          username: 'some-user',
          },
          ];

          The tokens can be of any API token type. Note that admin tokens must target all environments and projects (i.e. use '*' for environments and project and start the secret with *:*.).

          You can also use the environment variables INIT_ADMIN_API_TOKENS, INIT_CLIENT_API_TOKENS, and INIT_FRONTEND_API_TOKENS. All variables require a comma-separated list of API tokens to initialize (for instance *:*.some-random-string, *:*.some-other-token). The tokens found in INIT_ADMIN_API_TOKENS, INIT_CLIENT_API_TOKENS, and INIT_FRONTEND_API_TOKENS will be created as admin, client, and frontend tokens respectively, and Unleash will assign usernames automatically.

      • databaseUrl - (deprecated) the postgres database url to connect to. Only used if db object is not specified, and overrides the db object and any environment variables that change parts of it (like DATABASE_SSL). Should include username/password. This value may also be set via the DATABASE_URL environment variable. Alternatively, if you would like to read the database url from a file, you may set the DATABASE_URL_FILE environment variable with the full file path. The contents of the file must be the database url exactly.

      • db - The database configuration object. See the database configuration section for a full overview of the available properties.

      • disableLegacyFeaturesApi (boolean) - whether to disable the legacy features API. Defaults to false (DISABLE_LEGACY_FEATURES_API). Introduced in Unleash 4.6.

      • email - the email object configuring an SMTP server for sending welcome mails and password reset mails

        • host - The server URL to your SMTP server
        • port - Which port the SMTP server is running on. Defaults to 465 (Secure SMTP)
        • secure (boolean) - Whether to use SMTPS or not.
        • sender - Which email should be set as sender of mails being sent from Unleash?
        • smtpuser - Username for your SMTP server
        • smtppass - Password for your SMTP server
      • eventHook (function(event, data)) - (deprecated in Unleash 4.3 in favor of the Webhook integration. Removed in Unleash 5) If provided, this function will be invoked whenever a feature is mutated. The possible values for event are 'feature-created', 'feature-archived' and 'feature-revived'. The data argument contains information about the mutation. Its fields are type (string) - the event type (same as event); createdBy (string) - the user who performed the mutation; data - the contents of the change. The contents in data differs based on the event type; For 'feature-archived' and 'feature-revived', the only field will be name - the name of the feature. For 'feature-created' the data follows a schema defined in the code here. See an api here.

      • getLogger (function) - Used to register a custom log provider.

      • logLevel (debug | info | warn | error | fatal) - The lowest level to log at, also configurable using environment variable LOG_LEVEL.

      • enableRequestLogger (boolean) - use this to enable logging for requested urls and response codes (default: false).

      • preHook (function) - this is a hook if you need to provide any middlewares to express before unleash adds any. Express app instance is injected as first argument.

      • preRouterHook (function) - use this to register custom express middlewares before the unleash specific routers are added.

      • secureHeaders (boolean) - use this to enable security headers (HSTS, CSP, etc) when serving Unleash from HTTPS. Can also be configured through the environment variable SECURE_HEADERS.

      • additionalCspAllowedDomains (CspAllowedDomains) - use this when you want to enable security headers but have additional domains you need to allow traffic to. You can set the following environment variables:

        • CSP_ALLOWED_DEFAULT to allow new defaultSrc (comma separated list)
        • CSP_ALLOWED_FONT to allow new fontSrc (comma separated list)
        • CSP_ALLOWED_STYLE to allow new styleSrc (comma separated list)
        • CSP_ALLOWED_SCRIPT to allow new scriptSrc (comma separated list)
        • CSP_ALLOWED_IMG to allow new imgSrc (comma separated list)
        • CSP_ALLOWED_CONNECT to allow new connectSrc (comma separated list)
        • CSP_ALLOWED_FRAME to allow new frameSrc (comma separated listed)
        • CSP_ALLOWED_MEDIA to allow new mediaSrc (comma separated list)
        • CSP_ALLOWED_OBJECT to allow new objectSrc (comma separated list)
      • server - The server config object taking the following properties

        • port - which port the unleash-server should bind to. If port is omitted or is 0, the operating system will assign an arbitrary unused port. Will be ignored if pipe is specified. This value may also be set via the HTTP_PORT environment variable
        • host - which host the unleash-server should bind to. If host is omitted, the server will accept connections on the unspecified IPv6 address (::) when IPv6 is available, or the unspecified IPv4 address (0.0.0.0) otherwise. This value may also be set via the HTTP_HOST environment variable
        • pipe - parameter to identify IPC endpoints. See https://nodejs.org/api/net.html#net_identifying_paths_for_ipc_connections for more details
        • serverMetrics (boolean) - use this option to turn on/off prometheus metrics.
        • baseUriPath (string) - use to register a base path for all routes on the application. For example /my/unleash/base (note the starting /). Defaults to /. Can also be configured through the environment variable BASE_URI_PATH.
        • unleashUrl (string) - Used to specify the official URL this instance of Unleash can be accessed at for an end user. Can also be configured through the environment variable UNLEASH_URL.
        • gracefulShutdownEnable: (boolean) - Used to control if Unleash should shutdown gracefully (close connections, stop tasks,). Defaults to true. GRACEFUL_SHUTDOWN_ENABLE
        • gracefulShutdownTimeout: (number) - Used to control the timeout, in milliseconds, for shutdown Unleash gracefully. Will kill all connections regardless if this timeout is exceeded. Defaults to 1000ms GRACEFUL_SHUTDOWN_TIMEOUT
      • session - The session config object takes the following options:

        • ttlHours (number) - The number of hours a user session is allowed to live before a new sign-in is required. Defaults to 48 hours. SESSION_TTL_HOURS
        • clearSiteDataOnLogout (boolean) - When true, a logout action will return a Clear Site Data response header instructing the browser to clear all cookies on the same domain Unleash is running on. If disabled unleash will only destroy and clear the session cookie. Defaults to true. SESSION_CLEAR_SITE_DATA_ON_LOGOUT
        • cookieName - Name of the cookies used to hold the session id. Defaults to 'unleash-session'.
      • ui (object) - Set of UI specific overrides. You may set the following keys: environment, slogan.

      • versionCheck - the object deciding where to check for latest version

        • url - The url to check version (Defaults to https://version.unleash.run) - Overridable with (UNLEASH_VERSION_URL)
        • enable - Whether version checking is enabled (defaults to true) - Overridable with (CHECK_VERSION) (if anything other than true, does not check)
      • environmentEnableOverrides - A list of environment names to force enable at startup. This is feature should be used with caution. When passed a list, this will enable each environment in that list and disable all other environments. You can't use this to disable all environments, passing an empty list will do nothing. If one of the given environments is not already enabled on startup then it will also enable projects and toggles for that environment. Note that if one of the passed environments doesn't already exist this will do nothing aside from log a warning.

      • clientFeatureCaching - configuring memoization of the /api/client/features endpoint

        • enabled - set to true by default - Overridable with (CLIENT_FEATURE_CACHING_ENABLED)
        • maxAge - the time to cache features, set to 600 milliseconds by default - Overridable with (CLIENT_FEATURE_CACHING_MAXAGE) ) (accepts milliseconds)
      • frontendApi - Configuration options for the Unleash front-end API.

        • refreshIntervalInMs - how often (in milliseconds) front-end clients should refresh their data from the cache. Overridable with the FRONTEND_API_REFRESH_INTERVAL_MS environment variable.
      • accessControlMaxAge - You can configure the max-age of the Access-Control-Max-Age header. Defaults to 86400 seconds. Overridable with the ACCESS_CONTROL_MAX_AGE environment variable.

      • responseTimeWithAppNameKillSwitch - use this to disable metrics with app names. This is enabled by default but may increase the cardinality of metrics causing Unleash memory usage to grow if your app name is randomly generated (which is not recommended). Overridable with the UNLEASH_RESPONSE_TIME_WITH_APP_NAME_KILL_SWITCH environment variable.

      • disableCompression - Disables Express compression middleware, useful when configuring the application in a serverless environment. Defaults to false. Overridable with the SERVER_DISABLE_COMPRESSION environment variable.

      • keepAliveTimeout - Use this to tweak connection keepalive timeout in seconds. Useful for hosted situations where you need to make sure your connections are closed before terminating the instance. Defaults to 15. Overridable with the SERVER_KEEPALIVE_TIMEOUT environment variable. You can also set the environment variable ENABLED_ENVIRONMENTS to a comma delimited string of environment names to override environments.

      • metricsRateLimiting - Use the following to tweak the rate limits for /api/client/register, /api/client/metrics, /api/frontend/register and /api/frontend/metrics POST endpoints

        • clientMetricsMaxPerMinute - How many requests per minute per IP is allowed against POST /api/client/metrics before returning 429. Set to 6000 by default (100rps) - Overridable with REGISTER_CLIENT_RATE_LIMIT_PER_MINUTE environment variable
        • clientRegisterMaxPerMinute - How many requests per minute per IP is allowed against POST /api/client/register before returning 429. Set to 6000 by default (100rps) - Overridable with CLIENT_METRICS_RATE_LIMIT_PER_MINUTE environment variable
        • frontendMetricsMaxPerMinute - How many requests per minute per IP is allowed against POST /api/frontend/metrics before returning 429. Set to 6000 by default (100rps) - Overridable with FRONTEND_METRICS_RATE_LIMIT_PER_MINUTE environment variable
        • frontendRegisterMaxPerMinute - How many requests per minute per IP is allowed against POST /api/frontend/register before returning 429. Set to 6000 by default (100rps) - Overridable with REGISTER_FRONTEND_RATE_LIMIT_PER_MINUTE environment variable
      • rateLimiting - Use the following to tweak the rate limits for POST /auth/simple (Password-based login) and POST /api/admin/user-admin (Creating users)

        • simpleLoginMaxPerMinute - How many requests per minute per IP is allowed against POST /auth/simple before returning 429. Set to 10 by default - Overridable with SIMPLE_LOGIN_LIMIT_PER_MINUTE environment variable
        • createUserMaxPerMinute - How many requests per minute per IP is allowed against POST /api/admin/user-admin before returning 429. Set to 20 by default - Overridable with CREATE_USER_RATE_LIMIT_PER_MINUTE environment variable

      Disabling Auto-Start

      If you're using Unleash as part of a larger express app, you can disable the automatic server start by calling server.create. It takes the same options as server.start, but will not begin listening for connections.

      const express = require('express');
      const unleash = require('unleash-server');
      const app = express();

      const start = async () => {
      const instance = await unleash.create({
      databaseUrl: 'postgres://unleash_user:password@localhost:5432/unleash',
      });
      app.use(instance.app);
      console.log(`Unleash app generated and attached to parent application`);
      };

      start();

      Graceful shutdown

      PS! Unleash will listen for the SIGINT signal and automatically trigger graceful shutdown of Unleash.

      If you need to stop Unleash (close database connections, and stop running Unleash tasks) you may use the stop function. Be aware that it is not possible to restart the Unleash instance after stopping it, but you can create a new instance of Unleash.

      const express = require('express');
      const unleash = require('unleash-server');
      const app = express();

      const start = async () => {
      const instance = await unleash.start({
      databaseUrl: 'postgres://unleash_user:password@localhost:5432/unleash',
      port: 4242,
      });

      //Sometime later
      await instance.stop();
      console.log('Unleash has now stopped');
      };

      start();

      Segment limits

      caution

      Changing segment limits could have a negative impact on the performance of Unleash SDKs and cause network congestion. Think twice before changing these values.

      Some facets of the segments feature can be customized via environment variables. This lets you change the segment limits that Unleash uses.

      UNLEASH_STRATEGY_SEGMENTS_LIMIT controls the maximum number of segments that can be applied to a single strategy. The default is 5.

      UNLEASH_SEGMENT_VALUES_LIMIT controls the maximum number of values that you can assign across a segment's constraints. The default is 1000 (Since v5.1.0; it was 100 before v5.1.0)

      Securing Unleash

      You can integrate Unleash with your authentication provider (OAuth 2.0). Read more about securing unleash.

      How do I configure the log output?

      By default, unleash uses log4js to log important information. It is possible to swap out the logger provider (only when using Unleash programmatically). You do this by providing an implementation of the getLogger function as This enables filtering of log levels and easy redirection of output streams.

      function getLogger(name) {
      // do something with the name
      return {
      debug: console.log,
      info: console.log,
      warn: console.log,
      error: console.error,
      };
      }

      The logger interface with its debug, info, warn and error methods expects format string support as seen in debug or the JavaScript console object. Many commonly used logging implementations cover this API, e.g., bunyan, pino or winston.

      Database configuration

      info

      In-code configuration values take precedence over environment values: If you provide a database username both via environment variables and in code with the Unleash options object, Unleash will use the in-code username.

      You cannot run the Unleash server without a database. You must provide Unleash with database connection details for it to start correctly.

      The available options are listed in the table below. Options can be specified either via JavaScript (only when starting Unleash via code) or via environment variables. The "property name" column below gives the name of the property on the Unleash options' db object.

      Property nameEnvironment variableDefault valueDescription
      userDATABASE_USERNAMEunleash_userThe database username.
      passwordDATABASE_PASSWORDpasswordThe database password.
      hostDATABASE_HOSTlocalhostThe database hostname.
      portDATABASE_PORT5432The database port.
      databaseDATABASE_NAMEunleashThe name of the database.
      sslDATABASE_SSLN/AAn object describing SSL options. In code, provide a regular JavaScript object. When using the environment variable, provide a stringified JSON object.
      poolN/A (use the below variables)An object describing database pool options. With environment variables, use the options below directly.
      pool.minDATABASE_POOL_MIN0The minimum number of connections in the connection pool.
      pool.maxDATABASE_POOL_MAX4The maximum number of connections in the connection pool.
      pool.idleTimeoutMillisDATABASE_POOL_IDLE_TIMEOUT_MS30000The amount of time (in milliseconds) that a connection must be idle for before it is marked as a candidate for eviction.
      applicationNameDATABASE_APPLICATION_NAMEunleashThe name of the application that created this Client instance.
      schemaDATABASE_SCHEMApublicThe schema to use in the database.

      Alternatively, you can use a single-host libpq connection string to connect to the database. You can provide it directly or from a file by using one of the below options. In JavaScript, these are top-level properties of the root configuration object, not the db object.

      Property nameEnvironment variableDefault valueDescription
      databaseUrlDATABASE_URLN/AA string that matches a single-host libpq connection string, such as postgres://USER:PASSWORD@HOST:PORT/DATABASE. Unleash does not support using multiple hosts.
      databaseUrlFileDATABASE_URL_FILEN/AThe path to a file that contains a libpq connection string.

      Below is an example JavaScript configuration object.

      const unleashOptions = {
      databaseUrl: 'postgres:/USER:PASSWORD@HOST:PORT/DATABASE',
      databaseUrlFile: '/path/to/file',
      db: {
      user: 'unleash_user',
      password: 'password',
      host: 'localhost',
      port: 5432,
      database: 'unleash',
      ssl: false,
      pool: {
      min: 0,
      max: 4,
      idleTimeoutMillis: 30000,
      },
      },
      };

      db.ssl vs DATABASE_SSL options

      When initializing Unleash from code, you'll provide the db.ssl option as a JavaScript object. As such, any functions will get evaluated before the object is used for configuration. When using the DATABASE_SSL environment variable, you must provide the value as a stringified JSON object. The object will get parsed before being used for configuration, but no further evaluation will take place.

      If you want to read content from a file, you should either initialize Unleash via JavaScript or manually interpolate the required values into the environment variable:

      Reading from the file system in JavaScript
      const unleashOptions = {
      db: {
      // other options omitted for brevity
      ssl: {
      ca: fs.readFileSync('/path/to/server-certificates/root.crt').toString(),
      },
      },
      };

      Enabling self-signed certificates

      To use self-signed certificates, you should set the SSL property rejectUnauthorized to false and set the ca property to the value of the certificate:

      Enable self-signed certificates
      const unleashOptions = {
      db: {
      // other options omitted for brevity
      ssl: {
      rejectUnauthorized: false,
      ca: fs.readFileSync('/path/to/server-certificates/root.crt').toString(),
      },
      },
      };

      Visit the node-postgres library's SSL section for more information.

      Supported Postgres SSL modes

      Unleash builds directly on the node-postgres library, so we support all the same SSL modes that they support. As of version 8 (released on February 25th 2020), node-postgres no longer supports all sslmodes. For this reason, Unleash cannot support all of Postgres' SSL modes. Specifically, Unleash does not support sslmode=prefer. Instead you must know whether you require an SSL connection ahead of time.

      Troubleshooting: database pooling connection timeouts

      • Check the default values of connection pool about idle session handling.

      • If you have a network component which closes idle sessions on the TCP layer, make sure that the connection pool's idleTimeoutMillis setting is lower than the timespan setting. If it isn't, then the network component will close the connection.

      Proxying requests from Unleash

      You can configure proxy services that intercept all outgoing requests from Unleash. This lets you use the Microsoft Teams or the Webhook integration for external services, even if the internet can only be reached via a proxy on your machine or container (for example if restricted by a firewall or similiar).

      As an example, here's how you could do it using the node-global-proxy package:

      const proxy = require("node-global-proxy").default;

      proxy.setConfig({
      http: "http://user:password@url:8080", //proxy adress, replace values as needed
      //https: "https://user:password@url:1080", //if a https proxy is needed
      });

      proxy.start(); //this starts the proxy, after this call all requests will be proxied

      Using above code-snippet, every outgoing request from unleash or its integrations will be subsequently routed through set proxy. If the proxy routing needs to be bypassed or stopped, its possible to stop it by using

      proxy.stop();

    - - + + \ No newline at end of file diff --git a/using-unleash/deploy/database-backup.html b/using-unleash/deploy/database-backup.html index bbd6e80a7b..42eb018ea3 100644 --- a/using-unleash/deploy/database-backup.html +++ b/using-unleash/deploy/database-backup.html @@ -20,15 +20,15 @@ - - + + - - +
    + + \ No newline at end of file diff --git a/using-unleash/deploy/database-setup.html b/using-unleash/deploy/database-setup.html index f043ba8cbe..2e44d78644 100644 --- a/using-unleash/deploy/database-setup.html +++ b/using-unleash/deploy/database-setup.html @@ -20,15 +20,15 @@ - - + +
    -

    Database Setup

    To run Unleash you need to have PostgreSQL database (PostgreSQL v14.x or newer).

    1. Create a local unleash databases in PostgreSQL
    $ psql postgres <<SQL
    CREATE USER unleash_user WITH PASSWORD 'password';
    CREATE DATABASE unleash;
    GRANT ALL PRIVILEGES ON DATABASE unleash to unleash_user;
    SQL

    You will now have a PostgreSQL database with the following configuration:

    • database name: unleash
    • username: unleash_user
    • password: password
    • host: localhost (assuming you started it locally)
    • port: 5432 (assuming you are using the default PostgreSQL port)
    - - +

    Database Setup

    To run Unleash you need to have PostgreSQL database (PostgreSQL v14.x or newer).

    1. Create a local unleash databases in PostgreSQL
    $ psql postgres <<SQL
    CREATE USER unleash_user WITH PASSWORD 'password';
    CREATE DATABASE unleash;
    GRANT ALL PRIVILEGES ON DATABASE unleash to unleash_user;
    SQL

    You will now have a PostgreSQL database with the following configuration:

    • database name: unleash
    • username: unleash_user
    • password: password
    • host: localhost (assuming you started it locally)
    • port: 5432 (assuming you are using the default PostgreSQL port)
    + + \ No newline at end of file diff --git a/using-unleash/deploy/email-service.html b/using-unleash/deploy/email-service.html index 7babf33f77..065311e2ab 100644 --- a/using-unleash/deploy/email-service.html +++ b/using-unleash/deploy/email-service.html @@ -20,15 +20,15 @@ - - + +
    -

    Email service

    New since Unleash v4.0.0 is an email service allowing us to send reset password and welcome mails to new users. In order for this to work you'll need to tell unleash what SMTP service you'd like to send mails from.

    If the service is not configured you'll see a log line every time you add a new user saying

    [2021-05-07T12:59:04.572] [WARN] routes/user-controller.ts - email
    was not sent to the user because email configuration is lacking

    Configuring

    Depending on your deploy case there are different ways of configuring this service. Full documentation of all configuration possibilities is available here

    Docker

    With docker, we configure the mail service via environment variables.

    You'll want to at least include EMAIL_HOST, EMAIL_USER, EMAIL_PASSWORD and EMAIL_SENDER

    Environment variables:

    • EMAIL_HOST - Your SMTP server address
    • EMAIL_PORT - Your SMTP server port - defaults to 567
    • EMAIL_SECURE - whether to use SMTPS - set to false or true - defaults to false,
    • EMAIL_USER - the username to authenticate against your SMTP server
    • EMAIL_PASSWORD - the password for your SMTP user
    • EMAIL_SENDER - which address should reset-password mails and welcome mails be sent from - defaults to noreply@unleash-hosted.com which is probably not what you want.

    Node

    With node, we can configure this when calling Unleash's start method.

    const unleash = require('unleash-server');

    unleash.start({
    email: {
    host: 'myhost',
    smtpuser: 'username',
    smtppass: 'password',
    sender: 'noreply@mycompany.com',
    },
    });

    Troubleshooting

    For troubleshooting tips, please refer to the email service troubleshooting guide.

    - - +

    Email service

    New since Unleash v4.0.0 is an email service allowing us to send reset password and welcome mails to new users. In order for this to work you'll need to tell unleash what SMTP service you'd like to send mails from.

    If the service is not configured you'll see a log line every time you add a new user saying

    [2021-05-07T12:59:04.572] [WARN] routes/user-controller.ts - email
    was not sent to the user because email configuration is lacking

    Configuring

    Depending on your deploy case there are different ways of configuring this service. Full documentation of all configuration possibilities is available here

    Docker

    With docker, we configure the mail service via environment variables.

    You'll want to at least include EMAIL_HOST, EMAIL_USER, EMAIL_PASSWORD and EMAIL_SENDER

    Environment variables:

    • EMAIL_HOST - Your SMTP server address
    • EMAIL_PORT - Your SMTP server port - defaults to 567
    • EMAIL_SECURE - whether to use SMTPS - set to false or true - defaults to false,
    • EMAIL_USER - the username to authenticate against your SMTP server
    • EMAIL_PASSWORD - the password for your SMTP user
    • EMAIL_SENDER - which address should reset-password mails and welcome mails be sent from - defaults to noreply@unleash-hosted.com which is probably not what you want.

    Node

    With node, we can configure this when calling Unleash's start method.

    const unleash = require('unleash-server');

    unleash.start({
    email: {
    host: 'myhost',
    smtpuser: 'username',
    smtppass: 'password',
    sender: 'noreply@mycompany.com',
    },
    });

    Troubleshooting

    For troubleshooting tips, please refer to the email service troubleshooting guide.

    + + \ No newline at end of file diff --git a/using-unleash/deploy/getting-started.html b/using-unleash/deploy/getting-started.html index 7f44c8447e..6a3560ef68 100644 --- a/using-unleash/deploy/getting-started.html +++ b/using-unleash/deploy/getting-started.html @@ -20,15 +20,15 @@ - - + +
    -

    Getting Started

    This section only applies if you plan to self-host Unleash. If you are looking for our hosted solution you should head over to www.getunleash.io

    Requirements

    You will need:

    Start Unleash server

    Whichever option you choose to start Unleash, you must specify a database URI (it can be set in the environment variable DATABASE_URL). If your database server is not set up to support SSL you'll also need to set the environment variable DATABASE_SSL to false


    Once the server has started, you will see the message:

    Unleash started on http://localhost:4242

    To run multiple replicas of Unleash simply point all instances to the same database.

    Unleash v4: The first time Unleash starts it will create a default user which you can use to sign-in to you Unleash instance and add more users with:

    • username: admin
    • password: unleash4all

    If you'd like the default admin user to be created with a different username and password, you may define the following environment variables when running Unleash:

    • UNLEASH_DEFAULT_ADMIN_USERNAME
    • UNLEASH_DEFAULT_ADMIN_PASSWORD

    The way of defining these variables may vary depending on how you run Unleash.

    Option 1 - use Docker

    Useful links:

    Steps:

    1. Create a network by running docker network create unleash
    2. Start a postgres database:
    docker run -e POSTGRES_PASSWORD=some_password \
    -e POSTGRES_USER=unleash_user -e POSTGRES_DB=unleash \
    --network unleash --name postgres postgres
    1. Start Unleash via docker:
    docker run -p 4242:4242 \
    -e DATABASE_HOST=postgres -e DATABASE_NAME=unleash \
    -e DATABASE_USERNAME=unleash_user -e DATABASE_PASSWORD=some_password \
    -e DATABASE_SSL=false \
    --network unleash --pull=always unleashorg/unleash-server

    Option 2 - use Docker-compose

    Steps:

    1. Clone the Unleash repository.
    2. Run docker compose up -d in repository root folder.

    Option 3 - from Node.js

    1. Create a new folder/directory on your development computer.

    2. From a terminal/bash shell, install the dependencies:

      npm init
      npm install unleash-server --save
    3. Create a file called server.js, paste the following into it and save.

      const unleash = require('unleash-server');

      unleash
      .start({
      db: {
      ssl: false,
      host: 'localhost',
      port: 5432,
      database: 'unleash',
      user: 'unleash_user',
      password: 'password',
      },
      server: {
      port: 4242,
      },
      })
      .then((unleash) => {
      console.log(
      `Unleash started on http://localhost:${unleash.app.get('port')}`,
      );
      });
    4. Run server.js:

      node server.js

    Create an api token for your client

    Test your server and create a sample API call

    Once the Unleash server has started, go to localhost:4242 in your browser. If you see an empty list of feature toggles, try creating one with curl from a terminal/bash shell:

    curl --location -H "Authorization: <apitoken from previous step>" \
    --request POST 'http://localhost:4242/api/admin/features' \
    --header 'Content-Type: application/json' --data-raw '{\
    "name": "Feature.A",\
    "description": "Dolor sit amet.",\
    "type": "release",\
    "enabled": false,\
    "stale": false,\
    "strategies": [\
    {\
    "name": "default",\
    "parameters": {}\
    }\
    ]\
    }'

    Version check

    • Unleash checks that it uses the latest version by making a call to https://version.unleash.run.
      • This is a cloud function storing instance id to our database for statistics.
    • This request includes a unique instance id for your server.
    • If you do not wish to check for upgrades define the environment variable CHECK_VERSION to anything else other than true before starting, and Unleash won't make any calls
      • export CHECK_VERSION=false
    - - +

    Getting Started

    This section only applies if you plan to self-host Unleash. If you are looking for our hosted solution you should head over to www.getunleash.io

    Requirements

    You will need:

    Start Unleash server

    Whichever option you choose to start Unleash, you must specify a database URI (it can be set in the environment variable DATABASE_URL). If your database server is not set up to support SSL you'll also need to set the environment variable DATABASE_SSL to false


    Once the server has started, you will see the message:

    Unleash started on http://localhost:4242

    To run multiple replicas of Unleash simply point all instances to the same database.

    Unleash v4: The first time Unleash starts it will create a default user which you can use to sign-in to you Unleash instance and add more users with:

    • username: admin
    • password: unleash4all

    If you'd like the default admin user to be created with a different username and password, you may define the following environment variables when running Unleash:

    • UNLEASH_DEFAULT_ADMIN_USERNAME
    • UNLEASH_DEFAULT_ADMIN_PASSWORD

    The way of defining these variables may vary depending on how you run Unleash.

    Option 1 - use Docker

    Useful links:

    Steps:

    1. Create a network by running docker network create unleash
    2. Start a postgres database:
    docker run -e POSTGRES_PASSWORD=some_password \
    -e POSTGRES_USER=unleash_user -e POSTGRES_DB=unleash \
    --network unleash --name postgres postgres
    1. Start Unleash via docker:
    docker run -p 4242:4242 \
    -e DATABASE_HOST=postgres -e DATABASE_NAME=unleash \
    -e DATABASE_USERNAME=unleash_user -e DATABASE_PASSWORD=some_password \
    -e DATABASE_SSL=false \
    --network unleash --pull=always unleashorg/unleash-server

    Option 2 - use Docker-compose

    Steps:

    1. Clone the Unleash repository.
    2. Run docker compose up -d in repository root folder.

    Option 3 - from Node.js

    1. Create a new folder/directory on your development computer.

    2. From a terminal/bash shell, install the dependencies:

      npm init
      npm install unleash-server --save
    3. Create a file called server.js, paste the following into it and save.

      const unleash = require('unleash-server');

      unleash
      .start({
      db: {
      ssl: false,
      host: 'localhost',
      port: 5432,
      database: 'unleash',
      user: 'unleash_user',
      password: 'password',
      },
      server: {
      port: 4242,
      },
      })
      .then((unleash) => {
      console.log(
      `Unleash started on http://localhost:${unleash.app.get('port')}`,
      );
      });
    4. Run server.js:

      node server.js

    Create an api token for your client

    Test your server and create a sample API call

    Once the Unleash server has started, go to localhost:4242 in your browser. If you see an empty list of feature toggles, try creating one with curl from a terminal/bash shell:

    curl --location -H "Authorization: <apitoken from previous step>" \
    --request POST 'http://localhost:4242/api/admin/features' \
    --header 'Content-Type: application/json' --data-raw '{\
    "name": "Feature.A",\
    "description": "Dolor sit amet.",\
    "type": "release",\
    "enabled": false,\
    "stale": false,\
    "strategies": [\
    {\
    "name": "default",\
    "parameters": {}\
    }\
    ]\
    }'

    Version check

    • Unleash checks that it uses the latest version by making a call to https://version.unleash.run.
      • This is a cloud function storing instance id to our database for statistics.
    • This request includes a unique instance id for your server.
    • If you do not wish to check for upgrades define the environment variable CHECK_VERSION to anything else other than true before starting, and Unleash won't make any calls
      • export CHECK_VERSION=false
    + + \ No newline at end of file diff --git a/using-unleash/deploy/google-auth-hook.html b/using-unleash/deploy/google-auth-hook.html index 3c12f95352..065c0d3000 100644 --- a/using-unleash/deploy/google-auth-hook.html +++ b/using-unleash/deploy/google-auth-hook.html @@ -20,15 +20,15 @@ - - + +
    -

    Google Auth Hook

    You can also find the complete source code for this guide in the unleash-examples project.

    This part of the tutorial shows how to create a sign-in flow for users and integrate with Unleash server project. The implementation assumes that I am working in localhost with 4242 port.

    If you are still using Unleash v3 you need to follow the google-auth-hook-v3

    This is a simple index.ts server file.

    const unleash = require('unleash-server');

    unleash.start(options).then((unleash) => {
    console.log(`Unleash started on http://localhost:${unleash.app.get('port')}`);
    });

    Creating a web application client ID

    1. Go to the credentials section in the Google Cloud Platform Console.

    2. Click OAuth consent screen. Type a product name. Fill in any relevant optional fields. Click Save.

    3. Click Create credentials > OAuth client ID.

    4. Under Application type, select Web Application.

    5. Type the Name.

    6. Under Authorized redirect URIs enter the following URLs, one at a time.

    http://localhost:4242/api/auth/callback
    1. Click Create.

    2. Copy the CLIENT ID and CLIENT SECRET and save them for later use.

    Add dependencies

    Add two dependencies @passport-next/passport and @passport-next/passport-google-oauth2 inside index.ts file

    const unleash = require('unleash-server');
    const passport = require('@passport-next/passport');
    const GoogleOAuth2Strategy =
    require('@passport-next/passport-google-oauth2').Strategy;

    Add googleAdminAuth() function and other options. Make sure to also accept the services argument to get access to the userService.

    function googleAdminAuth(app, config, services) {
    const { baseUriPath } = config.server;
    const { userService } = services;
    }

    let options = {
    authentication: {
    type: 'custom',
    customAuthHandler: googleAdminAuth,
    },
    };

    unleash.start(options).then((instance) => {
    console.log(
    `Unleash started on http://localhost:${instance.app.get('port')}`,
    );
    });

    In googleAdminAuth function: Configure the Google strategy for use by Passport.js

    OAuth 2-based strategies require a verify function which receives the credential (accessToken) for accessing the Google API on the user's behalf, along with the user's profile. The function must invoke cb with a user object, which will be set at req.user in route handlers after authentication.

    const GOOGLE_CLIENT_ID = '...';
    const GOOGLE_CLIENT_SECRET = '...';
    const GOOGLE_CALLBACK_URL = 'http://localhost:4242/api/auth/callback';

    function googleAdminAuth(app, config, services) {
    const { baseUriPath } = config.server;
    const { userService } = services;

    passport.use(
    new GoogleOAuth2Strategy(
    {
    clientID: GOOGLE_CLIENT_ID,
    clientSecret: GOOGLE_CLIENT_SECRET,
    callbackURL: GOOGLE_CALLBACK_URL,
    },
    async function (accessToken, refreshToken, profile, cb) {
    // Extract the minimal profile information we need from the profile object
    // and connect with Unleash
    const email = profile.emails[0].value;
    const user = await userService.loginUserWithoutPassword(email, true);
    cb(null, user);
    },
    ),
    );
    }

    In googleAdminAuth function

    Configure passport package.

    function googleAdminAuth(app, config, services) {
    // ...
    app.use(passport.initialize());
    app.use(passport.session());
    passport.serializeUser((user, done) => done(null, user));
    passport.deserializeUser((user, done) => done(null, user));
    // ...
    }

    Implement a preRouter hook for /auth/google/login. It's necessary for login with Google.

    function googleAdminAuth(app, config, services) {
    // ...
    app.get(
    '/auth/google/login',
    passport.authenticate('google', { scope: ['email'] }),
    );
    // ...
    }

    Implement a preRouter hook for /api/auth/callback. It's a callback when the login is executed.

    function googleAdminAuth(app, config, services) {
    // ...
    app.get(
    '/api/auth/callback',
    passport.authenticate('google', {
    failureRedirect: '/api/admin/error-login',
    }),
    (req, res) => {
    // Successful authentication, redirect to your app.
    res.redirect('/');
    },
    );
    // ...
    }

    Implement a preRouter hook for /api/.

    function googleAdminAuth(app, config, services) {
    // ...
    app.use('/api/', (req, res, next) => {
    if (req.user) {
    next();
    } else {
    // Instruct unleash-frontend to pop-up auth dialog
    return res
    .status('401')
    .json(
    new unleash.AuthenticationRequired({
    path: '/auth/google/login',
    type: 'custom',
    message: `You have to identify yourself in order to use Unleash. Click the button and follow the instructions.`,
    }),
    )
    .end();
    }
    });
    // ...
    }

    The complete code

    You can also find the complete source code for this guide in the unleash-examples project.

    The index.ts server file.

    'use strict';

    const unleash = require('unleash-server');
    const passport = require('@passport-next/passport');
    const GoogleOAuth2Strategy = require('@passport-next/passport-google-oauth2');

    const GOOGLE_CLIENT_ID = '...';
    const GOOGLE_CLIENT_SECRET = '...';
    const GOOGLE_CALLBACK_URL = 'http://localhost:4242/api/auth/callback';

    function googleAdminAuth(app, config, services) {
    const { baseUriPath } = config.server;
    const { userService } = services;

    passport.use(
    new GoogleOAuth2Strategy(
    {
    clientID: GOOGLE_CLIENT_ID,
    clientSecret: GOOGLE_CLIENT_SECRET,
    callbackURL: GOOGLE_CALLBACK_URL,
    },
    async (accessToken, refreshToken, profile, cb) => {
    const email = profile.emails[0].value;
    const user = await userService.loginUserWithoutPassword(email, true);
    cb(null, user);
    },
    ),
    );

    app.use(passport.initialize());
    app.use(passport.session());
    passport.serializeUser((user, done) => done(null, user));
    passport.deserializeUser((user, done) => done(null, user));

    app.get(
    '/auth/google/login',
    passport.authenticate('google', { scope: ['email'] }),
    );
    app.get(
    '/api/auth/callback',
    passport.authenticate('google', {
    failureRedirect: '/api/admin/error-login',
    }),
    (req, res) => {
    res.redirect('/');
    },
    );

    app.use('/api/', (req, res, next) => {
    if (req.user) {
    next();
    } else {
    return res
    .status('401')
    .json(
    new unleash.AuthenticationRequired({
    path: '/auth/google/login',
    type: 'custom',
    message: `You have to identify yourself in order to use Unleash. Click the button and follow the instructions.`,
    }),
    )
    .end();
    }
    });
    }

    const options = {
    authentication: {
    type: 'custom',
    customAuthHandler: googleAdminAuth,
    },
    };

    unleash.start(options);
    - - +

    Google Auth Hook

    You can also find the complete source code for this guide in the unleash-examples project.

    This part of the tutorial shows how to create a sign-in flow for users and integrate with Unleash server project. The implementation assumes that I am working in localhost with 4242 port.

    If you are still using Unleash v3 you need to follow the google-auth-hook-v3

    This is a simple index.ts server file.

    const unleash = require('unleash-server');

    unleash.start(options).then((unleash) => {
    console.log(`Unleash started on http://localhost:${unleash.app.get('port')}`);
    });

    Creating a web application client ID

    1. Go to the credentials section in the Google Cloud Platform Console.

    2. Click OAuth consent screen. Type a product name. Fill in any relevant optional fields. Click Save.

    3. Click Create credentials > OAuth client ID.

    4. Under Application type, select Web Application.

    5. Type the Name.

    6. Under Authorized redirect URIs enter the following URLs, one at a time.

    http://localhost:4242/api/auth/callback
    1. Click Create.

    2. Copy the CLIENT ID and CLIENT SECRET and save them for later use.

    Add dependencies

    Add two dependencies @passport-next/passport and @passport-next/passport-google-oauth2 inside index.ts file

    const unleash = require('unleash-server');
    const passport = require('@passport-next/passport');
    const GoogleOAuth2Strategy =
    require('@passport-next/passport-google-oauth2').Strategy;

    Add googleAdminAuth() function and other options. Make sure to also accept the services argument to get access to the userService.

    function googleAdminAuth(app, config, services) {
    const { baseUriPath } = config.server;
    const { userService } = services;
    }

    let options = {
    authentication: {
    type: 'custom',
    customAuthHandler: googleAdminAuth,
    },
    };

    unleash.start(options).then((instance) => {
    console.log(
    `Unleash started on http://localhost:${instance.app.get('port')}`,
    );
    });

    In googleAdminAuth function: Configure the Google strategy for use by Passport.js

    OAuth 2-based strategies require a verify function which receives the credential (accessToken) for accessing the Google API on the user's behalf, along with the user's profile. The function must invoke cb with a user object, which will be set at req.user in route handlers after authentication.

    const GOOGLE_CLIENT_ID = '...';
    const GOOGLE_CLIENT_SECRET = '...';
    const GOOGLE_CALLBACK_URL = 'http://localhost:4242/api/auth/callback';

    function googleAdminAuth(app, config, services) {
    const { baseUriPath } = config.server;
    const { userService } = services;

    passport.use(
    new GoogleOAuth2Strategy(
    {
    clientID: GOOGLE_CLIENT_ID,
    clientSecret: GOOGLE_CLIENT_SECRET,
    callbackURL: GOOGLE_CALLBACK_URL,
    },
    async function (accessToken, refreshToken, profile, cb) {
    // Extract the minimal profile information we need from the profile object
    // and connect with Unleash
    const email = profile.emails[0].value;
    const user = await userService.loginUserWithoutPassword(email, true);
    cb(null, user);
    },
    ),
    );
    }

    In googleAdminAuth function

    Configure passport package.

    function googleAdminAuth(app, config, services) {
    // ...
    app.use(passport.initialize());
    app.use(passport.session());
    passport.serializeUser((user, done) => done(null, user));
    passport.deserializeUser((user, done) => done(null, user));
    // ...
    }

    Implement a preRouter hook for /auth/google/login. It's necessary for login with Google.

    function googleAdminAuth(app, config, services) {
    // ...
    app.get(
    '/auth/google/login',
    passport.authenticate('google', { scope: ['email'] }),
    );
    // ...
    }

    Implement a preRouter hook for /api/auth/callback. It's a callback when the login is executed.

    function googleAdminAuth(app, config, services) {
    // ...
    app.get(
    '/api/auth/callback',
    passport.authenticate('google', {
    failureRedirect: '/api/admin/error-login',
    }),
    (req, res) => {
    // Successful authentication, redirect to your app.
    res.redirect('/');
    },
    );
    // ...
    }

    Implement a preRouter hook for /api/.

    function googleAdminAuth(app, config, services) {
    // ...
    app.use('/api/', (req, res, next) => {
    if (req.user) {
    next();
    } else {
    // Instruct unleash-frontend to pop-up auth dialog
    return res
    .status('401')
    .json(
    new unleash.AuthenticationRequired({
    path: '/auth/google/login',
    type: 'custom',
    message: `You have to identify yourself in order to use Unleash. Click the button and follow the instructions.`,
    }),
    )
    .end();
    }
    });
    // ...
    }

    The complete code

    You can also find the complete source code for this guide in the unleash-examples project.

    The index.ts server file.

    'use strict';

    const unleash = require('unleash-server');
    const passport = require('@passport-next/passport');
    const GoogleOAuth2Strategy = require('@passport-next/passport-google-oauth2');

    const GOOGLE_CLIENT_ID = '...';
    const GOOGLE_CLIENT_SECRET = '...';
    const GOOGLE_CALLBACK_URL = 'http://localhost:4242/api/auth/callback';

    function googleAdminAuth(app, config, services) {
    const { baseUriPath } = config.server;
    const { userService } = services;

    passport.use(
    new GoogleOAuth2Strategy(
    {
    clientID: GOOGLE_CLIENT_ID,
    clientSecret: GOOGLE_CLIENT_SECRET,
    callbackURL: GOOGLE_CALLBACK_URL,
    },
    async (accessToken, refreshToken, profile, cb) => {
    const email = profile.emails[0].value;
    const user = await userService.loginUserWithoutPassword(email, true);
    cb(null, user);
    },
    ),
    );

    app.use(passport.initialize());
    app.use(passport.session());
    passport.serializeUser((user, done) => done(null, user));
    passport.deserializeUser((user, done) => done(null, user));

    app.get(
    '/auth/google/login',
    passport.authenticate('google', { scope: ['email'] }),
    );
    app.get(
    '/api/auth/callback',
    passport.authenticate('google', {
    failureRedirect: '/api/admin/error-login',
    }),
    (req, res) => {
    res.redirect('/');
    },
    );

    app.use('/api/', (req, res, next) => {
    if (req.user) {
    next();
    } else {
    return res
    .status('401')
    .json(
    new unleash.AuthenticationRequired({
    path: '/auth/google/login',
    type: 'custom',
    message: `You have to identify yourself in order to use Unleash. Click the button and follow the instructions.`,
    }),
    )
    .end();
    }
    });
    }

    const options = {
    authentication: {
    type: 'custom',
    customAuthHandler: googleAdminAuth,
    },
    };

    unleash.start(options);
    + + \ No newline at end of file diff --git a/using-unleash/deploy/google-auth-v3.html b/using-unleash/deploy/google-auth-v3.html index a7294b5661..d9f55e1d0a 100644 --- a/using-unleash/deploy/google-auth-v3.html +++ b/using-unleash/deploy/google-auth-v3.html @@ -20,15 +20,15 @@ - - + +

    Google Auth v3

    You can also find the complete source code for this guide in the unleash-examples project.

    This part of the tutorial shows how to create a sign-in flow for users and integrate with Unleash server project. The implementation assumes that I am working in localhost with 4242 port.

    This is a simple index.ts server file.

    const unleash = require('unleash-server');

    unleash.start(options).then((unleash) => {
    console.log(`Unleash started on http://localhost:${unleash.app.get('port')}`);
    });

    Creating a web application client ID

    1. Go to the credentials section in the Google Cloud Platform Console.

    2. Click OAuth consent screen. Type a product name. Fill in any relevant optional fields. Click Save.

    3. Click Create credentials > OAuth client ID.

    4. Under Application type, select Web Application.

    5. Type the Name.

    6. Under Authorized redirect URIs enter the following URLs, one at a time.

    http://localhost:4242/api/auth/callback
    1. Click Create.

    2. Copy the CLIENT ID and CLIENT SECRET and save them for later use.

    Add dependencies

    Add two dependencies @passport-next/passport and @passport-next/passport-google-oauth2 inside index.ts file

    const unleash = require('unleash-server');
    const passport = require('@passport-next/passport');
    const GoogleOAuth2Strategy =
    require('@passport-next/passport-google-oauth2').Strategy;

    Configure the Google strategy for use by Passport.js

    OAuth 2-based strategies require a verify function which receives the credential (accessToken) for accessing the Google API on the user's behalf, along with the user's profile. The function must invoke cb with a user object, which will be set at req.user in route handlers after authentication.

    const GOOGLE_CLIENT_ID = '...';
    const GOOGLE_CLIENT_SECRET = '...';
    const GOOGLE_CALLBACK_URL = 'http://localhost:4242/api/auth/callback';

    passport.use(
    new GoogleOAuth2Strategy(
    {
    clientID: GOOGLE_CLIENT_ID,
    clientSecret: GOOGLE_CLIENT_SECRET,
    callbackURL: GOOGLE_CALLBACK_URL,
    },
    function (accessToken, refreshToken, profile, cb) {
    // Extract the minimal profile information we need from the profile object
    // and connect with Unleash to get name and email.
    cb(
    null,
    new unleash.User({
    name: profile.displayName,
    email: profile.emails[0].value,
    }),
    );
    },
    ),
    );

    Add googleAdminAuth() function and other options

    function googleAdminAuth(app) {}

    let options = {
    adminAuthentication: 'custom',
    preRouterHook: googleAdminAuth,
    };

    unleash.start(options).then((instance) => {
    console.log(
    `Unleash started on http://localhost:${instance.app.get('port')}`,
    );
    });

    In googleAdminAuth function

    Configure passport package.

    function googleAdminAuth(app) {
    app.use(passport.initialize());
    app.use(passport.session());
    passport.serializeUser((user, done) => done(null, user));
    passport.deserializeUser((user, done) => done(null, user));
    // ...
    }

    Implement a preRouter hook for /auth/google/login. It's necessary for login with Google.

    function googleAdminAuth(app) {
    // ...
    app.get(
    '/auth/google/login',
    passport.authenticate('google', { scope: ['email'] }),
    );
    // ...
    }

    Implement a preRouter hook for /api/auth/callback. It's a callback when the login is executed.

    function googleAdminAuth(app) {
    // ...
    app.get(
    '/api/auth/callback',
    passport.authenticate('google', {
    failureRedirect: '/api/admin/error-login',
    }),
    (req, res) => {
    // Successful authentication, redirect to your app.
    res.redirect('/');
    },
    );
    // ...
    }

    Implement a preRouter hook for /api/admin.

    function googleAdminAuth(app) {
    // ...
    app.use('/api/admin/', (req, res, next) => {
    if (req.user) {
    next();
    } else {
    // Instruct unleash-frontend to pop-up auth dialog
    return res
    .status('401')
    .json(
    new unleash.AuthenticationRequired({
    path: '/auth/google/login',
    type: 'custom',
    message: `You have to identify yourself in order to use Unleash. Click the button and follow the instructions.`,
    }),
    )
    .end();
    }
    });
    // ...
    }

    The complete code

    The index.ts server file.

    'use strict';

    const unleash = require('unleash-server');
    const passport = require('@passport-next/passport');
    const GoogleOAuth2Strategy = require('@passport-next/passport-google-oauth2');

    const GOOGLE_CLIENT_ID = '...';
    const GOOGLE_CLIENT_SECRET = '...';
    const GOOGLE_CALLBACK_URL = 'http://localhost:4242/api/auth/callback';

    passport.use(
    new GoogleOAuth2Strategy(
    {
    clientID: GOOGLE_CLIENT_ID,
    clientSecret: GOOGLE_CLIENT_SECRET,
    callbackURL: GOOGLE_CALLBACK_URL,
    },
    (accessToken, refreshToken, profile, cb) => {
    cb(
    null,
    new unleash.User({
    name: profile.displayName,
    email: profile.emails[0].value,
    }),
    );
    },
    ),
    );

    function googleAdminAuth(app) {
    app.use(passport.initialize());
    app.use(passport.session());
    passport.serializeUser((user, done) => done(null, user));
    passport.deserializeUser((user, done) => done(null, user));

    app.get(
    '/auth/google/login',
    passport.authenticate('google', { scope: ['email'] }),
    );
    app.get(
    '/api/auth/callback',
    passport.authenticate('google', {
    failureRedirect: '/api/admin/error-login',
    }),
    (req, res) => {
    res.redirect('/');
    },
    );

    app.use('/api/admin/', (req, res, next) => {
    if (req.user) {
    next();
    } else {
    return res
    .status('401')
    .json(
    new unleash.AuthenticationRequired({
    path: '/auth/google/login',
    type: 'custom',
    message: `You have to identify yourself in order to use Unleash. Click the button and follow the instructions.`,
    }),
    )
    .end();
    }
    });
    }

    const options = {
    adminAuthentication: 'custom',
    preRouterHook: googleAdminAuth,
    };

    unleash.start(options).then((instance) => {
    console.log(
    `Unleash started on http://localhost:${instance.app.get('port')}`,
    );
    });
    - - + + \ No newline at end of file diff --git a/using-unleash/deploy/license-keys.htm/index.html b/using-unleash/deploy/license-keys.htm/index.html new file mode 100644 index 0000000000..1ae7c80950 --- /dev/null +++ b/using-unleash/deploy/license-keys.htm/index.html @@ -0,0 +1,11 @@ + + + + + + + + + \ No newline at end of file diff --git a/using-unleash/deploy/license-keys.html b/using-unleash/deploy/license-keys.html new file mode 100644 index 0000000000..4f0cb5b4df --- /dev/null +++ b/using-unleash/deploy/license-keys.html @@ -0,0 +1,38 @@ + + + + + +License Keys for Unleash Enterprise (BETA) | Unleash + + + + + + + + + + + + + + + + + + + + + + +
    +

    License Keys for Unleash Enterprise (BETA)

    This document describes all aspects and workflows of license key management in Unleash 5.8 and above.
    +Note that this is beta functionality and as such, these steps should only be followed under the direction of Unleash Customer Success.

    Warning

    License key management applies only to self-hosted, Enterprise versions of Unleash.

    • For Unleash hosted (Pro/Enterprise), license management is not necessary and is handled automatically.
    • For Unleash Open Source, no license is necessary and these steps do not apply.

    Areas of license key management that will be covered include:


    License policy

    License keys are leveraged for self-hosted Unleash Enterprise instance(s) to be compliant with the following aspects of a customer agreement:

    • Seat count (# of licensed users)
    • Contract duration (expiration date)

    These parameters are encoded into the license key. Therefore, installation of a new license key per self-hosted instance will be required should changes to any of the above occur (for example, increased seat count or contract renewal).

    Separate license keys are required per Unleash instance, for example, a dev or production instance would have separate keys.

    danger

    Reusing Unleash keys across multiple installations is prohibited under the terms of your license.

    Enforcement

    The Unleash UI will display banners in any of the following non-compliant scenarios:

    • No installed/missing license key
      Banner showing a missing license
    • Expired license key
      Banner showing an expired license
    • Enroled user count higher than the number of seats supported by the license
      Banner showing a high user count

    Obtaining a new or updated license

    New Customers

    If you are a new customer, your account representative will provide required license key(s) as part of your self-hosted instance onboarding.

    Existing Customers

    If you are an existing customer and are making changes to your agreement (changing seat count or the contract expiration), your account representative can be contacted to obtain required license key(s).

    Alternatively, reach out to sales@getunleash.io


    Installing a new or replacing an existing license

    Once you have a new or changed key, two methods are available to install a license on an enterprise self-hosted instance:

    1) Through the admin UI, logged in as an admin privileged user
    +2) Specifying a environment variable upon container deployment

    The process is the same whether you are licensing your self-hosted instance for the first time.

    note

    After updating a license key, a manual browser refresh may be required for any displayed licensing related banners to be removed from your session.


    Admin UI

    info

    This area of the UI can also be used to verify the currently installed key details

    1) Log in to the Unleash UI as an admin privileged user
    +2) Navigate to the admin menu (gear icon) -> License
    +3) If a key is already installed, the following details will be shown:

    • Customer name
    • Plan
    • Seat count
    • License expiration date

    4) Paste the new license key and click Update license key

    License overview screen


    Environment Variable

    License key installation or update through an environment variable on container deployment is also possible, using the format UNLEASH_LICENSE.

    For security, if using helm charts or docker compose to deploy Unleash, a best practice would be to setup the license key as a secret and reference it in the helm configuration values.yaml or docker-compose.yml.

    info

    The default image references in the Helm chart point to open-source Unleash resources. To use Unleash Enterprise, you must manually update the image references to point to the appropriate unleash-enterprise images

    + + + + \ No newline at end of file diff --git a/using-unleash/deploy/license-keys.html.html b/using-unleash/deploy/license-keys.html.html new file mode 100644 index 0000000000..1ae7c80950 --- /dev/null +++ b/using-unleash/deploy/license-keys.html.html @@ -0,0 +1,11 @@ + + + + + + + + + \ No newline at end of file diff --git a/using-unleash/deploy/securing-unleash-v3.html b/using-unleash/deploy/securing-unleash-v3.html index b952b38e2b..9bd3ad16c3 100644 --- a/using-unleash/deploy/securing-unleash-v3.html +++ b/using-unleash/deploy/securing-unleash-v3.html @@ -20,15 +20,15 @@ - - + +

    Securing Unleash v3

    This guide is only relevant if you are using Unleash Open-Source. The Enterprise edition does already ship with a secure setup and multiple SSO options.

    The Unleash API is split into two different paths: /api/client and /api/admin. This makes it easy to have different authentication strategy for the admin interface and the client-api used by the applications integrating with Unleash.

    General settings

    Unleash uses an encrypted cookie to maintain a user session. This allows users to be logged in across multiple instances of Unleash. To protect this cookie, Unleash will automatically generate a secure token the first time you start Unleash.

    Securing the Admin API

    To secure the Admin API, you have to tell Unleash that you are using a custom admin authentication and implement your authentication logic as a preHook.

    const unleash = require('unleash-server');
    const myCustomAdminAuth = require('./auth-hook');

    unleash
    .start({
    databaseUrl: 'postgres://unleash_user:passord@localhost:5432/unleash',
    adminAuthentication: 'custom',
    preRouterHook: myCustomAdminAuth,
    })
    .then((unleash) => {
    console.log(
    `Unleash started on http://localhost:${unleash.app.get('port')}`,
    );
    });

    Additionally, you can trigger the admin interface to prompt the user to sign in by configuring your middleware to return a 401 status on protected routes. The response body must contain a message and a path used to redirect the user to the proper login route.

    {
    "message": "You must be logged in to use Unleash",
    "path": "/custom/login"
    }

    Examples of custom authentication hooks:

    Securing the Client API

    A common way to support client access is to use pre-shared secrets. This can be solved by having clients send a shared key in an HTTP header with every client request to the Unleash API. All official Unleash clients should support this.

    In the Java client this would look like this:

    UnleashConfig unleashConfig = UnleashConfig.builder()
    .appName("my-app")
    .instanceId("my-instance-1")
    .unleashAPI(unleashAPI)
    .customHttpHeader("Authorization", "12312Random")
    .build();

    On the Unleash server side, you need to implement a preRouter hook which verifies that all calls to /api/client include this pre-shared key in the defined header. This could look something like this.

    const unleash = require('unleash-server');
    const sharedSecret = '12312Random';

    unleash
    .start({
    databaseUrl: 'postgres://unleash_user:passord@localhost:5432/unleash',
    preRouterHook: (app) => {
    app.use('/api/client', (req, res, next) => {
    if (req.header('authorization') !== sharedSecret) {
    res.sendStatus(401);
    } else {
    next();
    }
    });
    },
    })
    .then((unleash) => {
    console.log(
    `Unleash started on http://localhost:${unleash.app.get('port')}`,
    );
    });

    client-auth-unleash.js

    - - + + \ No newline at end of file diff --git a/using-unleash/deploy/securing-unleash.html b/using-unleash/deploy/securing-unleash.html index 9d817a6549..8ae5bc7b86 100644 --- a/using-unleash/deploy/securing-unleash.html +++ b/using-unleash/deploy/securing-unleash.html @@ -20,15 +20,15 @@ - - + +
    -

    Securing Unleash

    If you are still using Unleash v3 you need to follow the securing-unleash-v3

    This guide is only relevant if you are using Unleash Open-Source. The Enterprise edition does already ship with multiple SSO options, such as SAML 2.0, OpenID Connect.

    Unleash Open-Source v4 comes with username/password authentication out of the box. In addition Unleash v4 also comes with API token support, to make it easy to handle access tokens for Client SDKs and programmatic access to the Unleash APIs.

    Password requirements

    Unleash requires a strong password.

    • minimum 10 characters long
    • contains at least one uppercase letter
    • contains at least one number
    • contains at least one special character (symbol)

    Implementing Custom Authentication

    If you do not wish to use the built-in username/password authentication you can add a customAuthHandler

    To secure the Admin API, you have to tell Unleash that you are using a custom admin authentication and implement your authentication logic as a preHook.

    const unleash = require('unleash-server');
    const myCustomAdminAuth = require('./auth-hook');

    unleash
    .start({
    databaseUrl: 'postgres://unleash_user:password@localhost:5432/unleash',
    authentication: {
    type: 'custom',
    customAuthHandler: myCustomAdminAuth,
    },
    })
    .then((unleash) => {
    console.log(
    `Unleash started on http://localhost:${unleash.app.get('port')}`,
    );
    });

    Additionally, you can trigger the admin interface to prompt the user to sign in by configuring your middleware to return a 401 status on protected routes. The response body must contain a message and a path used to redirect the user to the proper login route.

    {
    "message": "You must be logged in to use Unleash",
    "path": "/custom/login"
    }

    Examples of custom authentication hooks:

    - - +

    Securing Unleash

    If you are still using Unleash v3 you need to follow the securing-unleash-v3

    This guide is only relevant if you are using Unleash Open-Source. The Enterprise edition does already ship with multiple SSO options, such as SAML 2.0, OpenID Connect.

    Unleash Open-Source v4 comes with username/password authentication out of the box. In addition Unleash v4 also comes with API token support, to make it easy to handle access tokens for Client SDKs and programmatic access to the Unleash APIs.

    Password requirements

    Unleash requires a strong password.

    • minimum 10 characters long
    • contains at least one uppercase letter
    • contains at least one number
    • contains at least one special character (symbol)

    Implementing Custom Authentication

    If you do not wish to use the built-in username/password authentication you can add a customAuthHandler

    To secure the Admin API, you have to tell Unleash that you are using a custom admin authentication and implement your authentication logic as a preHook.

    const unleash = require('unleash-server');
    const myCustomAdminAuth = require('./auth-hook');

    unleash
    .start({
    databaseUrl: 'postgres://unleash_user:password@localhost:5432/unleash',
    authentication: {
    type: 'custom',
    customAuthHandler: myCustomAdminAuth,
    },
    })
    .then((unleash) => {
    console.log(
    `Unleash started on http://localhost:${unleash.app.get('port')}`,
    );
    });

    Additionally, you can trigger the admin interface to prompt the user to sign in by configuring your middleware to return a 401 status on protected routes. The response body must contain a message and a path used to redirect the user to the proper login route.

    {
    "message": "You must be logged in to use Unleash",
    "path": "/custom/login"
    }

    Examples of custom authentication hooks:

    + + \ No newline at end of file diff --git a/using-unleash/deploy/upgrading-unleash.html b/using-unleash/deploy/upgrading-unleash.html index 4fdd0c579b..c1e8fe18bb 100644 --- a/using-unleash/deploy/upgrading-unleash.html +++ b/using-unleash/deploy/upgrading-unleash.html @@ -20,15 +20,15 @@ - - + +
    -

    Upgrading Unleash

    Generally, the intention is that unleash-server should always provide support for clients one major version lower than the current one. This should make it possible to upgrade unleash gradually.

    Upgrading directly from v3.x to v5.x

    Ivar Østhus, Unleash CTO and Co-Founder, demonstrates how to update Unleash 3.x to Unleash 5.x in just a few minutes with no downtime. You can also watch this on YouTube with a transcript.

    Upgrading from v4.x to v5.x

    Unleash v5 was released on April 27th, 2023. It contains a few breaking changes.

    Requires Node.js version 18+

    Unleash v5 drops support Node.js versions below 18, which is the current active LTS at the time of release. Unleash v4 officially supported Node.js v14 and v16, but both of these will reach end of life in 2023.

    The Google Authenticator provider for SSO has been removed

    The Google Authenticator provider is now hidden by default. We recommend using OpenID Connect instead.

    However, if you are running a self hosted version of Unleash and you need to temporarily re-enable Google SSO, you can do so by setting the GOOGLE_AUTH_ENABLED environment variable to true. If you're running a hosted version of Unleash, you'll need to reach out to us and ask us to re-enable the flag. However, the ability to do this will be removed in a future release and this is not safe to depend on.

    This provider was deprecated in v4.

    Default database password

    The Unleash default database password is now password instead of passord. Turns out that the Norwegian word for password is too similar to the English word, and that people would think it was a typo.

    This should only impact dev builds and initial setup. You should never use the default password in any production environments.

    The /api/admin/features API is gone

    Most of the old features API was deprecated in v4.3 and superseded by the project API instead. In v5, the deprecated parts have been completely removed. The only operations on that API that are still active are the operations to add or remove a tag from a feature toggle.

    Error message structure

    Some of Unleash's API error messages have changed their structure. Specifically, this applies to error messages generated by our OpenAPI validation layer. However, only their structure has changed (and they now contain more human-friendly messages); the error codes should still be the same.

    Previously, they would look like this:

    {
    "error": "Request validation failed",
    "validation": [
    {
    "keyword": "type",
    "dataPath": ".body.parameters",
    "schemaPath": "#/components/schemas/addonCreateUpdateSchema/properties/parameters/type",
    "params": {
    "type": "object"
    },
    "message": "should be object"
    }
    ]
    }

    Now they look like this instead, and are more in line with the rest of Unleash's error messages.

    {
    "id": "37a1765f-a5a0-4371-8aa2-341f331579f9",
    "name": "ValidationError",
    "message": "Request validation failed: the payload you provided doesn't conform to the schema. Check the `details` property for a list of errors that we found.",
    "details": [
    {
    "description": "The .parameters property should be object. You sent [].",
    "path": "parameters"
    }
    ]
    }

    As such, if you're relying on the specifics of the error structure for those API errors, you might need to update your handling.

    Upgrading from v3.x to v4.x

    Before you upgrade we strongly recommend that you take a full database backup, to make sure you can downgrade to version 3.

    You can also read the highlights of what's new in v4.

    1. All API calls now require a token.

    If you are upgrading from Unleash Open-Source v3 client SDKs did not need to use an API token in order to connect to Unleash-server. Starting from v4 we have back-ported the API token handling for Enterprise in to the Open-Source version. This means that all client SDKs now need to use a client token in order to connect to Unleash.

    Read more in the API token documentation.

    2. Configuring Unleash

    We have done a lot of changes to the options you can pass in to Unleash. If you are manually configuring Unleash you should have a look on the updated configuring Unleash documentation

    3. Role-based Access Control (RBAC)

    We have implemented RBAC in Unleash v4. This has totally changed the permission system in Unleash.

    Required actions: If you have implemented "custom authentication" for your users you will need to make changes to your integration:

    • extendedPermissions option has been removed. You can no longer specify custom permission per-user basis. All "logged_in users" must belong to a "root" role. This can be "Admin", "Editor" or "Viewer". This is taken care of when you create new users via userService.
    • All "logged-in users" needs to be defined in Unleash and have a unique ID. This can be achieved by calling "createUser" on "userService".

    Code example:

    const user = userService.loginUserWithoutPassword(
    'some@getunleash.io',
    false, // autoCreateUser. Set to true if you want to create users on the fly.
    );

    // The user needs to be set on the current active session
    req.session.user = user;

    4. Legacy v2 routes removed

    Only relevant if you use the enableLegacyRoutes option.

    In v2 you could query feature toggles on /api/features. This was deprecated in v4 and we introduced two different endpoints (/api/admin/features and /api/client/features) to be able to optimize performance and security. In v3 you could still enable the legacy routes via the enableLegacyRoutes option. This was removed in v4.

    5. Unleash CLI has been removed

    Unleash no longer ships with a binary that allows you to start Unleash directly from the command line. From v4 you need to either use Unleash via docker or programmatically.

    Read more in our getting started documentation

    Upgrading from v2.x to v3.x

    The notable change introduced in Unleash v3.x is a strict separation of API paths for client requests and admin requests. This makes it easier to implement different authentication mechanisms for the admin UI and all unleash-clients. You can read more about securing unleash.

    The recommended approach is to first upgrade the unleash-server to v3 (which still supports v2 clients). After this is done, you should upgrade all your clients to v3.

    After upgrading all your clients, you should consider turning off legacy routes, used by v2 clients. To do this, set the configuration option enableLegacyRoutes to false as described in the page on configuring Unleash v3.

    Upgrading from v1.0 to v2.0

    Caveat 1: Not used db-migrate to migrate the Unleash database?

    In FINN we used liquibase, for internal reasons, to migrate our database. Because unleash from version 2.0 migrates the database internally, with db-migrate, you need to make sure that all previous migrations for version 1 exist, so that Unleash does not try to create already existing tables.

    How to check?

    If you don't have a "migrations" table with 7 unique migrations you are affected by this.

    How to fix?

    Before starting unleash version 2 you have to run the SQL located under scripts/fix-migrations-version-1.sql

    Caveat 2: databaseUrl (not databaseUri)

    Using Unleash as a library and injecting your own config? Then you should know that we changed the databaseUri config param name to databaseUrl. This is to make sure the param is aligned with the environment variable DATABASE_URL and avoid multiple names for the same config param.

    - - +

    Upgrading Unleash

    Generally, the intention is that unleash-server should always provide support for clients one major version lower than the current one. This should make it possible to upgrade unleash gradually.

    Upgrading directly from v3.x to v5.x

    Ivar Østhus, Unleash CTO and Co-Founder, demonstrates how to update Unleash 3.x to Unleash 5.x in just a few minutes with no downtime. You can also watch this on YouTube with a transcript.

    Upgrading from v4.x to v5.x

    Unleash v5 was released on April 27th, 2023. It contains a few breaking changes.

    Requires Node.js version 18+

    Unleash v5 drops support Node.js versions below 18, which is the current active LTS at the time of release. Unleash v4 officially supported Node.js v14 and v16, but both of these will reach end of life in 2023.

    The Google Authenticator provider for SSO has been removed

    The Google Authenticator provider is now hidden by default. We recommend using OpenID Connect instead.

    However, if you are running a self hosted version of Unleash and you need to temporarily re-enable Google SSO, you can do so by setting the GOOGLE_AUTH_ENABLED environment variable to true. If you're running a hosted version of Unleash, you'll need to reach out to us and ask us to re-enable the flag. However, the ability to do this will be removed in a future release and this is not safe to depend on.

    This provider was deprecated in v4.

    Default database password

    The Unleash default database password is now password instead of passord. Turns out that the Norwegian word for password is too similar to the English word, and that people would think it was a typo.

    This should only impact dev builds and initial setup. You should never use the default password in any production environments.

    The /api/admin/features API is gone

    Most of the old features API was deprecated in v4.3 and superseded by the project API instead. In v5, the deprecated parts have been completely removed. The only operations on that API that are still active are the operations to add or remove a tag from a feature toggle.

    Error message structure

    Some of Unleash's API error messages have changed their structure. Specifically, this applies to error messages generated by our OpenAPI validation layer. However, only their structure has changed (and they now contain more human-friendly messages); the error codes should still be the same.

    Previously, they would look like this:

    {
    "error": "Request validation failed",
    "validation": [
    {
    "keyword": "type",
    "dataPath": ".body.parameters",
    "schemaPath": "#/components/schemas/addonCreateUpdateSchema/properties/parameters/type",
    "params": {
    "type": "object"
    },
    "message": "should be object"
    }
    ]
    }

    Now they look like this instead, and are more in line with the rest of Unleash's error messages.

    {
    "id": "37a1765f-a5a0-4371-8aa2-341f331579f9",
    "name": "ValidationError",
    "message": "Request validation failed: the payload you provided doesn't conform to the schema. Check the `details` property for a list of errors that we found.",
    "details": [
    {
    "description": "The .parameters property should be object. You sent [].",
    "path": "parameters"
    }
    ]
    }

    As such, if you're relying on the specifics of the error structure for those API errors, you might need to update your handling.

    Upgrading from v3.x to v4.x

    Before you upgrade we strongly recommend that you take a full database backup, to make sure you can downgrade to version 3.

    You can also read the highlights of what's new in v4.

    1. All API calls now require a token.

    If you are upgrading from Unleash Open-Source v3 client SDKs did not need to use an API token in order to connect to Unleash-server. Starting from v4 we have back-ported the API token handling for Enterprise in to the Open-Source version. This means that all client SDKs now need to use a client token in order to connect to Unleash.

    Read more in the API token documentation.

    2. Configuring Unleash

    We have done a lot of changes to the options you can pass in to Unleash. If you are manually configuring Unleash you should have a look on the updated configuring Unleash documentation

    3. Role-based Access Control (RBAC)

    We have implemented RBAC in Unleash v4. This has totally changed the permission system in Unleash.

    Required actions: If you have implemented "custom authentication" for your users you will need to make changes to your integration:

    • extendedPermissions option has been removed. You can no longer specify custom permission per-user basis. All "logged_in users" must belong to a "root" role. This can be "Admin", "Editor" or "Viewer". This is taken care of when you create new users via userService.
    • All "logged-in users" needs to be defined in Unleash and have a unique ID. This can be achieved by calling "createUser" on "userService".

    Code example:

    const user = userService.loginUserWithoutPassword(
    'some@getunleash.io',
    false, // autoCreateUser. Set to true if you want to create users on the fly.
    );

    // The user needs to be set on the current active session
    req.session.user = user;

    4. Legacy v2 routes removed

    Only relevant if you use the enableLegacyRoutes option.

    In v2 you could query feature toggles on /api/features. This was deprecated in v4 and we introduced two different endpoints (/api/admin/features and /api/client/features) to be able to optimize performance and security. In v3 you could still enable the legacy routes via the enableLegacyRoutes option. This was removed in v4.

    5. Unleash CLI has been removed

    Unleash no longer ships with a binary that allows you to start Unleash directly from the command line. From v4 you need to either use Unleash via docker or programmatically.

    Read more in our getting started documentation

    Upgrading from v2.x to v3.x

    The notable change introduced in Unleash v3.x is a strict separation of API paths for client requests and admin requests. This makes it easier to implement different authentication mechanisms for the admin UI and all unleash-clients. You can read more about securing unleash.

    The recommended approach is to first upgrade the unleash-server to v3 (which still supports v2 clients). After this is done, you should upgrade all your clients to v3.

    After upgrading all your clients, you should consider turning off legacy routes, used by v2 clients. To do this, set the configuration option enableLegacyRoutes to false as described in the page on configuring Unleash v3.

    Upgrading from v1.0 to v2.0

    Caveat 1: Not used db-migrate to migrate the Unleash database?

    In FINN we used liquibase, for internal reasons, to migrate our database. Because unleash from version 2.0 migrates the database internally, with db-migrate, you need to make sure that all previous migrations for version 1 exist, so that Unleash does not try to create already existing tables.

    How to check?

    If you don't have a "migrations" table with 7 unique migrations you are affected by this.

    How to fix?

    Before starting unleash version 2 you have to run the SQL located under scripts/fix-migrations-version-1.sql

    Caveat 2: databaseUrl (not databaseUri)

    Using Unleash as a library and injecting your own config? Then you should know that we changed the databaseUri config param name to databaseUrl. This is to make sure the param is aligned with the environment variable DATABASE_URL and avoid multiple names for the same config param.

    + + \ No newline at end of file diff --git a/using-unleash/troubleshooting.html b/using-unleash/troubleshooting.html index 907364251c..8641ba48aa 100644 --- a/using-unleash/troubleshooting.html +++ b/using-unleash/troubleshooting.html @@ -20,15 +20,15 @@ - - + +
    -

    How-to: troubleshooting

    Troubleshooting common problems. If you want to suggest new items, please phrase the title as a concrete problem

    - - +

    How-to: troubleshooting

    Troubleshooting common problems. If you want to suggest new items, please phrase the title as a concrete problem

    + + \ No newline at end of file diff --git a/using-unleash/troubleshooting/cors.html b/using-unleash/troubleshooting/cors.html index fd02208c36..a91e42d06d 100644 --- a/using-unleash/troubleshooting/cors.html +++ b/using-unleash/troubleshooting/cors.html @@ -20,15 +20,15 @@ - - + +

    My requests are being blocked by CORS

    1. Make sure you've configured CORS access in Unleash admin UI settings as defined in the Unleash CORS Policy docs. These settings can be changed in the Unleash Dashboard under Settings -> CORS Origins or by using the API. Allowing all origins (using a single asterisk) will address this matter and is a great starting point when troubleshooting the behavior.
    2. When receiving "No 'Access-Control-Policy' header is present on the requested resource", using the command curl -I https://<host>/<endpoint> will allow us to verify that the response includes the header Access-Control-Allow-Origin: *.
    - - + + \ No newline at end of file diff --git a/using-unleash/troubleshooting/email-service.html b/using-unleash/troubleshooting/email-service.html index 9dc836ae3c..abceebc48d 100644 --- a/using-unleash/troubleshooting/email-service.html +++ b/using-unleash/troubleshooting/email-service.html @@ -20,15 +20,15 @@ - - + +

    The email service is not working correctly on my self-hosted Unleash instance

    When setting up your self-hosted Unleash instance, one of the available options is to configure an email service that will allow Unleash to send reset password and welcome emails to users.

    Here's how to troubleshoot some common issues related to the email service.

    Configuration

    The most common issues arise from misconfiguration. Please refer to the following documentation for guidance:

    You should double check that the details in your configuration look correct.

    Invalid URL error

    Make sure that the UNLEASH_URL variable is correctly set to a valid URL. This should be set to the public discoverable URL of your Unleash instance, and it should include the protocol (http or https).

    Examples:

    • Subdomain: https://unleash.mysite.com
    • Subpath: https://mysite.com/unleash

    SMTP TLS port

    Please double check that you're trying to reach your SMTP server on the TLS port, typically 587.

    Custom SSL certificate

    If you're using your own SMTP server which uses a custom SSL certificate, you will need to tell Unleash to trust that certificate. You can do this by setting the NODE_EXTRA_CA_CERTS variable to the path of the certificate file.

    This is usually done by mounting the custom certificate in a volume, and then setting NODE_EXTRA_CA_CERTS to the absolute path in the container where it can find the certificate. For example, if you mount it to /var/certs, you should set NODE_EXTRA_CA_CERTS to something like /var/certs/mycert.crt.

    - - + + \ No newline at end of file diff --git a/using-unleash/troubleshooting/feature-not-available.html b/using-unleash/troubleshooting/feature-not-available.html index 38b9a06d83..ba94415ed7 100644 --- a/using-unleash/troubleshooting/feature-not-available.html +++ b/using-unleash/troubleshooting/feature-not-available.html @@ -20,15 +20,15 @@ - - + +

    I don't see a documented Unleash feature in my admin UI

    Occasionally, users might come across a situation where a documented Unleash feature isn't visible in their admin UI. Here's how to troubleshoot this issue.

    You can usually find availability information in the feature's documentation page, displayed in a box like this:

    Availability

    Cool new feature was introduced as a beta feature in Unleash 5.5 and is only available in Unleash Enterprise. We plan to make this feature generally available to all Enterprise users in Unleash 5.6.

    1. Check that the feature is available in your current Unleash version. For example, Service accounts are available in Unleash 4.21 and later. If you're running a previous version, you'll need to update your Unleash instance.
    2. Make sure the feature is available for your plan, as you may need to upgrade your plan to access the feature.
    3. If this is a beta feature, it may not be enabled for your Unleash instance. Here's how you can enable it:
      • If you have a hosted Unleash instance and you'd like early access to the new feature, reach out to us so we can enable it for you.
      • If you're running a self-hosted Unleash instance, make sure you've enabled the feature in your Unleash configuration. Usually, this involves setting the correct environment variable. You can check the current flags and respective environment variables in your version's src/lib/types/experimental.ts. Setting this variable may look something like UNLEASH_EXPERIMENTAL_NEW_FEATURE=true.

    If you've followed the above steps and still can't access the feature, please contact us for further assistance.

    If you're currently using a beta feature, please reach out to us! We would be thrilled if you could provide some feedback.

    - - + + \ No newline at end of file diff --git a/using-unleash/troubleshooting/flag-exposure.html b/using-unleash/troubleshooting/flag-exposure.html index 23dd2722b4..1d696cc542 100644 --- a/using-unleash/troubleshooting/flag-exposure.html +++ b/using-unleash/troubleshooting/flag-exposure.html @@ -20,15 +20,15 @@ - - + +

    My feature flag is enabled but all/some of our users are not exposed to it

    To confirm how users will have flags resolved, follow these steps:

    1. Ensure your application is waiting for the ready event: It could be that frontend clients are calling isEnabled('feature-flag') before they have the response from the server. In this case, you should defer isEnabled calls until the client has emitted the ready event.
    2. The Unleash Playground was developed with this particular use case in mind. An access token can be used along with context values (passed in via the UI) to see how a flag will be resolved.
    3. When using a gradual rollout strategy, be mindful of the stickiness value. When evaluating a flag, if the provided context does not include the field used in the stickiness configuration, the gradual rollout strategy will be evaluated to false.
    - - + + \ No newline at end of file diff --git a/using-unleash/troubleshooting/flag-not-returned.html b/using-unleash/troubleshooting/flag-not-returned.html index 789f9ed58f..1eeb3a87c0 100644 --- a/using-unleash/troubleshooting/flag-not-returned.html +++ b/using-unleash/troubleshooting/flag-not-returned.html @@ -20,15 +20,15 @@ - - + +

    My feature flag is not returned in the Frontend API/Edge/Proxy

    By default, these endpoints will not return feature flags that are not enabled. This is mainly to save on bandwidth but it makes it a bit difficult to debug when features are not being returned.

    The first thing to look into is to validate that the feature is well configured and then check the token used from the SDK because it determines the set of accessible features. Last, verify that the context you're providing contains all the required data.

    1. Check that the feature is properly enabled:
      1. Verify that the feature has a strategy associated to it that will return true for your context (ref: add a strategy)
      2. Verify that the feature has been enabled in the environment used by the client application (ref: enabling a feature flag)
    2. Check that your token is of the right type. To connect to the Frontend API, Edge or Proxy, you need to use a Front-end token
    3. Check that your token has access to the feature flag. The token access configuration is immutable post-creation and defines the set of features that the token can access. The different parts of a token determine what projects and environment can be accessed:
      1. Access to all projects (current and future) - Tokens with a leading asterisk will provide access to all projects in a particular environment. For example, the token *:production:xyz123etc... will provide access to flags in the production environment of all projects.
      2. Access to a discrete list of projects - Tokens with a leading set of square brackets (empty) will be given access to a subset of projects in a particular environment. The token will look similar to the following: []:production:xyz123etc.... Which projects the token has access to can be found on the API Tokens page in the Unleash admin UI.
      3. Single project access - Tokens that lead with a project name are bound to the specified project and environment. For example, my_fullstack_app:production:xyz123etc... will only have access to flags in the "my_fullstack_app" project as set in the production environment.
    4. When using a gradual rollout strategy, be mindful of the stickiness value. When evaluating a flag, if the provided context does not include the field used in the stickiness configuration, the gradual rollout strategy will be evaluated to false and therefore it will not be returned by the API.
    5. Feature activation strategies may have constraints, segments, and rules that can be combined in different ways that can lead to complex scenarios. Try using the Playground to verify that the feature is properly configured and responding as expected.
    - - + + \ No newline at end of file

    hJL)y-!9})qyD2`BYDac{yJ=akWdlI0_|=n!WaI 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 bu1qs$e%3ozUan(DpT`dk9w6_(eEYuuzAmYE literal 0 HcmV?d00001 diff --git a/index.html b/index.html index ecc0448746..9bdc18c66b 100644 --- a/index.html +++ b/index.html @@ -20,15 +20,15 @@ - - + +

    hJL)y-!9})qyD2`BYDac{yJ=akWdlI0_|=n!WaI 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 bu1qs$e%3ozUan(DpT`dk9w6_(eEYuuzAmYE literal 0 HcmV?d00001 diff --git a/assets/js/0833ee16.e8794ef8.js b/assets/js/0833ee16.3310d1d2.js similarity index 98% rename from assets/js/0833ee16.e8794ef8.js rename to assets/js/0833ee16.3310d1d2.js index ce1ae0c6f9..45a9780f08 100644 --- a/assets/js/0833ee16.e8794ef8.js +++ b/assets/js/0833ee16.3310d1d2.js @@ -1 +1 @@ -"use strict";(self.webpackChunkwebsitev_2=self.webpackChunkwebsitev_2||[]).push([[14583],{75590:(e,t,n)=>{n.r(t),n.d(t,{assets:()=>i,contentTitle:()=>l,default:()=>d,frontMatter:()=>r,metadata:()=>s,toc:()=>p});var a=n(87462),o=(n(67294),n(3905));const r={title:"Svelte",slug:"/reference/sdks/svelte",custom_edit_url:"https://github.com/Unleash/proxy-client-svelte/edit/main/README.md"},l=void 0,s={unversionedId:"generated/sdks/client-side/svelte",id:"generated/sdks/client-side/svelte",title:"Svelte",description:"This document was generated from the README in the Svelte GitHub repository.",source:"@site/docs/generated/sdks/client-side/svelte.md",sourceDirName:"generated/sdks/client-side",slug:"/reference/sdks/svelte",permalink:"/reference/sdks/svelte",draft:!1,editUrl:"https://github.com/Unleash/proxy-client-svelte/edit/main/README.md",tags:[],version:"current",frontMatter:{title:"Svelte",slug:"/reference/sdks/svelte",custom_edit_url:"https://github.com/Unleash/proxy-client-svelte/edit/main/README.md"},sidebar:"documentation",previous:{title:"React",permalink:"/reference/sdks/react"},next:{title:"Vue",permalink:"/reference/sdks/vue"}},i={},p=[{value:"Initialize the client",id:"initialize-the-client",level:2},{value:"Connection options",id:"connection-options",level:3},{value:"Check feature toggle status",id:"check-feature-toggle-status",level:2},{value:"Check variants",id:"check-variants",level:2},{value:"Defer rendering until flags fetched",id:"defer-rendering-until-flags-fetched",level:2},{value:"Updating context",id:"updating-context",level:2},{value:"Deferring client start",id:"deferring-client-start",level:2},{value:"Use unleash client directly",id:"use-unleash-client-directly",level:2}],c={toc:p};function d(e){let{components:t,...n}=e;return(0,o.kt)("wrapper",(0,a.Z)({},c,n,{components:t,mdxType:"MDXLayout"}),(0,o.kt)("admonition",{title:"Generated content",type:"info"},(0,o.kt)("p",{parentName:"admonition"},"This document was generated from the README in the ",(0,o.kt)("a",{parentName:"p",href:"https://github.com/Unleash/proxy-client-svelte"},"Svelte GitHub repository"),".")),(0,o.kt)("admonition",{type:"tip"},(0,o.kt)("p",{parentName:"admonition"},"To connect to Unleash from a client-side context, you'll need to use the ",(0,o.kt)("a",{parentName:"p",href:"/reference/front-end-api"},"Unleash front-end API")," (",(0,o.kt)("a",{parentName:"p",href:"/how-to/how-to-create-api-tokens"},"how do I create an API token?"),") or the ",(0,o.kt)("a",{parentName:"p",href:"/reference/unleash-proxy"},"Unleash proxy")," (",(0,o.kt)("a",{parentName:"p",href:"/reference/api-tokens-and-client-keys#proxy-client-keys"},"how do I create client keys?"),").")),(0,o.kt)("h1",{id:"installation"},"Installation"),(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-bash"},"npm install @unleash/proxy-client-svelte\n# or\nyarn add @unleash/proxy-client-svelte\n")),(0,o.kt)("h1",{id:"how-to-use"},"How to use"),(0,o.kt)("h2",{id:"initialize-the-client"},"Initialize the client"),(0,o.kt)("p",null,"Depending on your needs and specific use-case, prepare one of:"),(0,o.kt)("ul",null,(0,o.kt)("li",{parentName:"ul"},(0,o.kt)("a",{parentName:"li",href:"https://docs.getunleash.io/reference/front-end-api"},"Front-end API")),(0,o.kt)("li",{parentName:"ul"},(0,o.kt)("a",{parentName:"li",href:"https://docs.getunleash.io/reference/unleash-edge"},"Unleash Edge")),(0,o.kt)("li",{parentName:"ul"},(0,o.kt)("a",{parentName:"li",href:"https://docs.getunleash.io/reference/unleash-proxy"},"Unleash Proxy"))),(0,o.kt)("p",null,"And a respective frontend token (or, if you're using the Unleash Proxy, one of your proxy's designated client keys, previously known as proxy secrets)."),(0,o.kt)("p",null,"Import the provider like this in your entrypoint file (typically index.svelte):"),(0,o.kt)("pre",null,(0,o.kt)("code",{parentName:"pre",className:"language-svelte"}," - - + +