From 9ea6f9667ea5919c0c7d777e33902bdf40bb69a0 Mon Sep 17 00:00:00 2001 From: freddieptf Date: Tue, 19 Nov 2024 13:51:51 +0300 Subject: [PATCH 1/5] something like feature flags --- README.md | 1 + src/config/chis-ke/config.json | 36 ++++++++++++++++- src/config/index.ts | 5 ++- src/lib/remote-place-resolver.ts | 2 + src/liquid/app/nav.html | 24 +++++++---- src/liquid/place/directive_1_get_started.html | 16 ++++++-- src/liquid/place/list.html | 40 ++++++++++--------- src/liquid/place/list_lazy.html | 22 +++++----- src/routes/add-place.ts | 7 ++++ src/routes/move.ts | 4 ++ 10 files changed, 114 insertions(+), 43 deletions(-) diff --git a/README.md b/README.md index 73f4975f..930e2947 100644 --- a/README.md +++ b/README.md @@ -48,6 +48,7 @@ To use the User Management Tool with your CHT project, you'll need to create a n `contact_types.contact_properties` | Array | Defines the attributes which are collected and set on the user's primary contact doc. See [ConfigProperty](#ConfigProperty). `contact_types.deactivate_users_on_replace` | boolean | Controls what should happen to the defunct contact and user documents when a user is replaced. When `false`, the contact and user account will be deleted. When `true`, the contact will be unaltered and the user account will be assigned the role `deactivated`. This allows for account restoration. `contact_types.hint` | string | Provide a brief hint or description to clarify the expected input for the property. +`contact_types.feature_flags` | Array | A list of features to enable for this contact type. Acceptable values are `create`, `replace-contact` and `move`. All features are enabled by default `logoBase64` | Image in base64 | Logo image for your project #### ConfigProperty diff --git a/src/config/chis-ke/config.json b/src/config/chis-ke/config.json index e528a4d4..cce3e50e 100644 --- a/src/config/chis-ke/config.json +++ b/src/config/chis-ke/config.json @@ -188,7 +188,6 @@ "friendly": "West Pokot", "domain": "westpokot.echis.go.ke" }, - { "friendly": "Staging (chis-staging.health.go.ke)", "domain": "chis-staging.health.go.ke" @@ -337,6 +336,41 @@ "required": true } ] + }, + { + "name": "e_household", + "friendly": "Household", + "contact_type": "f_client", + "user_role": [], + "place_properties": [], + "contact_properties": [], + "feature_flags": [ + "move" + ], + "replacement_property": { + "friendly_name": "", + "property_name": "replacement", + "type": "name", + "required": true + }, + "hierarchy": [ + { + "friendly_name": "CHU", + "property_name": "CHU", + "contact_type": "c_community_health_unit", + "type": "name", + "required": true, + "level": 2 + }, + { + "friendly_name": "CHP Area", + "property_name": "CHP", + "contact_type": "d_community_health_volunteer_area", + "type": "name", + "required": true, + "level": 1 + } + ] } ], "logoBase64": "" diff --git a/src/config/index.ts b/src/config/index.ts index 69b32141..e7ed8667 100644 --- a/src/config/index.ts +++ b/src/config/index.ts @@ -19,13 +19,14 @@ export type ContactType = { contact_type: string; contact_friendly?: string; user_role: string[]; - username_from_place: boolean; + username_from_place?: boolean; hierarchy: HierarchyConstraint[]; replacement_property: ContactProperty; place_properties: ContactProperty[]; contact_properties: ContactProperty[]; - deactivate_users_on_replace: boolean; + deactivate_users_on_replace?: boolean; hint?: string; + feature_flags?: string[]; }; export type HierarchyConstraint = { diff --git a/src/lib/remote-place-resolver.ts b/src/lib/remote-place-resolver.ts index e7d10a57..487a0ae6 100644 --- a/src/lib/remote-place-resolver.ts +++ b/src/lib/remote-place-resolver.ts @@ -5,6 +5,7 @@ import { RemotePlace, ChtApi } from './cht-api'; import { Config, ContactType, HierarchyConstraint } from '../config'; import { Validation } from './validation'; import RemotePlaceCache from './remote-place-cache'; +import assert from 'assert'; type RemotePlaceMap = { [key: string]: RemotePlace }; @@ -111,6 +112,7 @@ export default class RemotePlaceResolver { function getFuzzFunction(place: Place, hierarchyLevel: HierarchyConstraint, contactType: ContactType) { const fuzzingProperty = hierarchyLevel.level === 0 ? contactType.replacement_property : hierarchyLevel; + assert(fuzzingProperty); if (fuzzingProperty.type === 'generated') { throw Error(`Invalid configuration: hierarchy properties cannot be of type "generated".`); } diff --git a/src/liquid/app/nav.html b/src/liquid/app/nav.html index 533ef2d8..6602082c 100644 --- a/src/liquid/app/nav.html +++ b/src/liquid/app/nav.html @@ -19,12 +19,20 @@ @@ -35,7 +43,9 @@ diff --git a/src/liquid/place/directive_1_get_started.html b/src/liquid/place/directive_1_get_started.html index a926790a..f3ef4b79 100644 --- a/src/liquid/place/directive_1_get_started.html +++ b/src/liquid/place/directive_1_get_started.html @@ -12,10 +12,18 @@ add Add {{contactType.friendly}} diff --git a/src/liquid/place/list.html b/src/liquid/place/list.html index 1cfc0afa..3931f6fc 100644 --- a/src/liquid/place/list.html +++ b/src/liquid/place/list.html @@ -1,23 +1,25 @@
{% for contactType in contactTypes %} -
-

{{contactType.friendly}}

- {% if contactType.places.length > 0 %} - - {% include "components/table_header.html" contactType=contactType %} - - {% for place in contactType.places %} - {% include "components/place_item.html" %} - {% endfor%} - -
- {% else %} -
- No Results -
- {% endif %} + {% if contactType.feature_flags == undefined or contactType.feature_flags contains "create" or contactType.feature_flags contains "replace-contact" %} +
+

{{contactType.friendly}}

+ {% if contactType.places.length > 0 %} + + {% include "components/table_header.html" contactType=contactType %} + + {% for place in contactType.places %} + {% include "components/place_item.html" %} + {% endfor%} + +
+ {% else %} +
+ No Results +
+ {% endif %} + {%endif%} {% endfor %}
\ No newline at end of file diff --git a/src/liquid/place/list_lazy.html b/src/liquid/place/list_lazy.html index 1f2e8f69..7a9255b4 100644 --- a/src/liquid/place/list_lazy.html +++ b/src/liquid/place/list_lazy.html @@ -1,15 +1,17 @@
{% for contactType in contactTypes %} -
-

{{contactType.friendly}}

-
- - {% include "components/table_header.html" contactType=contactType %} -
-
- Loading data -
-
+ {% if contactType.feature_flags == undefined or contactType.feature_flags contains "create" or contactType.feature_flags contains "replace-contact" %} +
+

{{contactType.friendly}}

+
+ + {% include "components/table_header.html" contactType=contactType %} +
+
+ Loading data +
+
+ {%endif%} {% endfor %}
\ No newline at end of file diff --git a/src/routes/add-place.ts b/src/routes/add-place.ts index 1d18459c..d162246f 100644 --- a/src/routes/add-place.ts +++ b/src/routes/add-place.ts @@ -17,6 +17,13 @@ export default async function addPlace(fastify: FastifyInstance) { ? Config.getContactType(queryParams.type) : contactTypes[contactTypes.length - 1]; const op = queryParams.op || 'new'; + if (contactType.feature_flags) { + if ((op === 'new' && !contactType.feature_flags.includes('create')) || + (op === 'replace' && !contactType.feature_flags.includes('replace-contact'))) { + resp.status(404); + return; + } + } const tmplData = { view: 'add', logo: Config.getLogoBase64(), diff --git a/src/routes/move.ts b/src/routes/move.ts index 0a2dbf77..0d7716fb 100644 --- a/src/routes/move.ts +++ b/src/routes/move.ts @@ -13,6 +13,10 @@ export default async function sessionCache(fastify: FastifyInstance) { const contactTypes = Config.contactTypes(); const contactType = Config.getContactType(placeType); + if (contactType.feature_flags && !contactType.feature_flags.includes('move')) { + resp.code(404).type("text/html").send("Not Found"); + return; + } const tmplData = { view: 'move', op: 'move', From 5b4aed5b02bb4aa32bad0bd2a01a283e0b2b10f8 Mon Sep 17 00:00:00 2001 From: freddieptf Date: Tue, 19 Nov 2024 14:03:51 +0300 Subject: [PATCH 2/5] fix lint --- src/routes/add-place.ts | 2 +- src/routes/move.ts | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/src/routes/add-place.ts b/src/routes/add-place.ts index d162246f..de54fae0 100644 --- a/src/routes/add-place.ts +++ b/src/routes/add-place.ts @@ -20,7 +20,7 @@ export default async function addPlace(fastify: FastifyInstance) { if (contactType.feature_flags) { if ((op === 'new' && !contactType.feature_flags.includes('create')) || (op === 'replace' && !contactType.feature_flags.includes('replace-contact'))) { - resp.status(404); + resp.code(404).type('text/html').send('Not Found'); return; } } diff --git a/src/routes/move.ts b/src/routes/move.ts index 0d7716fb..e7c93ac7 100644 --- a/src/routes/move.ts +++ b/src/routes/move.ts @@ -14,7 +14,7 @@ export default async function sessionCache(fastify: FastifyInstance) { const contactType = Config.getContactType(placeType); if (contactType.feature_flags && !contactType.feature_flags.includes('move')) { - resp.code(404).type("text/html").send("Not Found"); + resp.code(404).type('text/html').send('Not Found'); return; } const tmplData = { From 03727cf1263b60346f24564b7ff80d64db137215 Mon Sep 17 00:00:00 2001 From: freddieptf Date: Tue, 26 Nov 2024 03:50:29 +0300 Subject: [PATCH 3/5] address feedback --- src/config/config-factory.ts | 44 ++++++++++++++++++++++++++++++++---- src/config/index.ts | 7 ++++-- src/routes/add-place.ts | 5 ++-- src/routes/move.ts | 3 ++- 4 files changed, 49 insertions(+), 10 deletions(-) diff --git a/src/config/config-factory.ts b/src/config/config-factory.ts index dc6a7169..40bb12a1 100644 --- a/src/config/config-factory.ts +++ b/src/config/config-factory.ts @@ -4,11 +4,43 @@ import kenyaConfig from './chis-ke'; import togoConfig from './chis-tg'; import civConfig from './chis-civ'; +export enum Feature { + Create = 'create', + ReplaceContact = 'replace-contact', + Move = 'move', +} + +const parseConfig = (c: any): PartnerConfig => { + return { + config: { + ...c.config, + contact_types: c.config.contact_types.map((t: any) => { + return { + ...t, + feature_flags: t.feature_flags?.map((v: string) => { + if ((Object.values(Feature) as string[]).indexOf(v) === -1) { + throw new Error( + 'invalid feature flag: ' + + v + + '. Acceptable values are [' + + Object.values(Feature).join(' | ') + + ']' + ); + } + return v as Feature; + }), + }; + }), + }, + mutate: c.mutate, + }; +}; + const CONFIG_MAP: { [key: string]: PartnerConfig } = { - 'CHIS-KE': kenyaConfig, - 'CHIS-UG': ugandaConfig, - 'CHIS-TG': togoConfig, - 'CHIS-CIV': civConfig + 'CHIS-KE': parseConfig(kenyaConfig), + 'CHIS-UG': parseConfig(ugandaConfig), + 'CHIS-TG': parseConfig(togoConfig), + 'CHIS-CIV': parseConfig(civConfig), }; export default function getConfigByKey(key: string = 'CHIS-KE'): PartnerConfig { @@ -17,7 +49,9 @@ export default function getConfigByKey(key: string = 'CHIS-KE'): PartnerConfig { const result = CONFIG_MAP[usingKey]; if (!result) { const available = JSON.stringify(Object.keys(CONFIG_MAP)); - throw Error(`Failed to start: Cannot find configuration "${usingKey}". Configurations available are ${available}`); + throw Error( + `Failed to start: Cannot find configuration '${usingKey}'. Configurations available are ${available}` + ); } return result; diff --git a/src/config/index.ts b/src/config/index.ts index e7ed8667..0a1bc85d 100644 --- a/src/config/index.ts +++ b/src/config/index.ts @@ -1,6 +1,6 @@ import _ from 'lodash'; import { ChtApi, PlacePayload } from '../lib/cht-api'; -import getConfigByKey from './config-factory'; +import getConfigByKey, { Feature } from './config-factory'; export type ConfigSystem = { domains: AuthenticationInfo[]; @@ -26,7 +26,7 @@ export type ContactType = { contact_properties: ContactProperty[]; deactivate_users_on_replace?: boolean; hint?: string; - feature_flags?: string[]; + feature_flags?: Feature[]; }; export type HierarchyConstraint = { @@ -123,6 +123,9 @@ export class Config { } public static hasMultipleRoles(contactType: ContactType): boolean { + if (contactType.feature_flags?.length === 1 && contactType.feature_flags.includes(Feature.Move)) { + return false; + } if (!contactType.user_role.length || contactType.user_role.some(role => !role.trim())) { throw Error(`unvalidatable config: 'user_role' property is empty or contains empty strings`); } diff --git a/src/routes/add-place.ts b/src/routes/add-place.ts index de54fae0..2f7e19f3 100644 --- a/src/routes/add-place.ts +++ b/src/routes/add-place.ts @@ -7,6 +7,7 @@ import SessionCache from '../services/session-cache'; import RemotePlaceResolver from '../lib/remote-place-resolver'; import { UploadManager } from '../services/upload-manager'; import RemotePlaceCache from '../lib/remote-place-cache'; +import { Feature } from '../config/config-factory'; export default async function addPlace(fastify: FastifyInstance) { fastify.get('/add-place', async (req, resp) => { @@ -18,8 +19,8 @@ export default async function addPlace(fastify: FastifyInstance) { : contactTypes[contactTypes.length - 1]; const op = queryParams.op || 'new'; if (contactType.feature_flags) { - if ((op === 'new' && !contactType.feature_flags.includes('create')) || - (op === 'replace' && !contactType.feature_flags.includes('replace-contact'))) { + if ((op === 'new' && !contactType.feature_flags.includes(Feature.Create)) || + (op === 'replace' && !contactType.feature_flags.includes(Feature.ReplaceContact))) { resp.code(404).type('text/html').send('Not Found'); return; } diff --git a/src/routes/move.ts b/src/routes/move.ts index e7c93ac7..f0e77d75 100644 --- a/src/routes/move.ts +++ b/src/routes/move.ts @@ -5,6 +5,7 @@ import { ChtApi } from '../lib/cht-api'; import { FastifyInstance } from 'fastify'; import MoveLib from '../lib/move'; import SessionCache from '../services/session-cache'; +import { Feature } from '../config/config-factory'; export default async function sessionCache(fastify: FastifyInstance) { fastify.get('/move/:placeType', async (req, resp) => { @@ -13,7 +14,7 @@ export default async function sessionCache(fastify: FastifyInstance) { const contactTypes = Config.contactTypes(); const contactType = Config.getContactType(placeType); - if (contactType.feature_flags && !contactType.feature_flags.includes('move')) { + if (contactType.feature_flags && !contactType.feature_flags.includes(Feature.Move)) { resp.code(404).type('text/html').send('Not Found'); return; } From edef0950528a60fb5cc7e61559d49bc3012cc75b Mon Sep 17 00:00:00 2001 From: freddieptf Date: Tue, 26 Nov 2024 22:07:30 +0300 Subject: [PATCH 4/5] feedback --- README.md | 4 +- src/config/chis-ke/config.json | 7 ++-- src/config/config-factory.ts | 40 ++----------------- src/config/index.ts | 12 +++--- src/lib/remote-place-resolver.ts | 1 - src/liquid/app/nav.html | 10 ++--- src/liquid/place/directive_1_get_started.html | 8 ++-- src/liquid/place/list.html | 2 +- src/liquid/place/list_lazy.html | 2 +- src/routes/add-place.ts | 11 ++--- src/routes/move.ts | 3 +- 11 files changed, 34 insertions(+), 66 deletions(-) diff --git a/README.md b/README.md index 930e2947..5f1a7c77 100644 --- a/README.md +++ b/README.md @@ -48,7 +48,9 @@ To use the User Management Tool with your CHT project, you'll need to create a n `contact_types.contact_properties` | Array | Defines the attributes which are collected and set on the user's primary contact doc. See [ConfigProperty](#ConfigProperty). `contact_types.deactivate_users_on_replace` | boolean | Controls what should happen to the defunct contact and user documents when a user is replaced. When `false`, the contact and user account will be deleted. When `true`, the contact will be unaltered and the user account will be assigned the role `deactivated`. This allows for account restoration. `contact_types.hint` | string | Provide a brief hint or description to clarify the expected input for the property. -`contact_types.feature_flags` | Array | A list of features to enable for this contact type. Acceptable values are `create`, `replace-contact` and `move`. All features are enabled by default +`contact_types.can_create` | boolean | Optionally disable/enable creating places of this type. Defaults to true. +`contact_types.can_replace_contact` | boolean | Optionally disable/enable replacing contacts for places of this type. Defaults to true. +`contact_types.can_move` | boolean | Optionally disable/enable moving places of this type. Defaults to true. `logoBase64` | Image in base64 | Logo image for your project #### ConfigProperty diff --git a/src/config/chis-ke/config.json b/src/config/chis-ke/config.json index cce3e50e..9dd73672 100644 --- a/src/config/chis-ke/config.json +++ b/src/config/chis-ke/config.json @@ -344,9 +344,10 @@ "user_role": [], "place_properties": [], "contact_properties": [], - "feature_flags": [ - "move" - ], + "username_from_place": false, + "deactivate_users_on_replace": false, + "can_create": false, + "can_replace_contact": false, "replacement_property": { "friendly_name": "", "property_name": "replacement", diff --git a/src/config/config-factory.ts b/src/config/config-factory.ts index 40bb12a1..14b05399 100644 --- a/src/config/config-factory.ts +++ b/src/config/config-factory.ts @@ -4,43 +4,11 @@ import kenyaConfig from './chis-ke'; import togoConfig from './chis-tg'; import civConfig from './chis-civ'; -export enum Feature { - Create = 'create', - ReplaceContact = 'replace-contact', - Move = 'move', -} - -const parseConfig = (c: any): PartnerConfig => { - return { - config: { - ...c.config, - contact_types: c.config.contact_types.map((t: any) => { - return { - ...t, - feature_flags: t.feature_flags?.map((v: string) => { - if ((Object.values(Feature) as string[]).indexOf(v) === -1) { - throw new Error( - 'invalid feature flag: ' + - v + - '. Acceptable values are [' + - Object.values(Feature).join(' | ') + - ']' - ); - } - return v as Feature; - }), - }; - }), - }, - mutate: c.mutate, - }; -}; - const CONFIG_MAP: { [key: string]: PartnerConfig } = { - 'CHIS-KE': parseConfig(kenyaConfig), - 'CHIS-UG': parseConfig(ugandaConfig), - 'CHIS-TG': parseConfig(togoConfig), - 'CHIS-CIV': parseConfig(civConfig), + 'CHIS-KE': kenyaConfig, + 'CHIS-UG': ugandaConfig, + 'CHIS-TG': togoConfig, + 'CHIS-CIV': civConfig, }; export default function getConfigByKey(key: string = 'CHIS-KE'): PartnerConfig { diff --git a/src/config/index.ts b/src/config/index.ts index 0a1bc85d..8e4c8568 100644 --- a/src/config/index.ts +++ b/src/config/index.ts @@ -1,6 +1,6 @@ import _ from 'lodash'; import { ChtApi, PlacePayload } from '../lib/cht-api'; -import getConfigByKey, { Feature } from './config-factory'; +import getConfigByKey from './config-factory'; export type ConfigSystem = { domains: AuthenticationInfo[]; @@ -19,14 +19,16 @@ export type ContactType = { contact_type: string; contact_friendly?: string; user_role: string[]; - username_from_place?: boolean; + username_from_place: boolean; hierarchy: HierarchyConstraint[]; replacement_property: ContactProperty; place_properties: ContactProperty[]; contact_properties: ContactProperty[]; - deactivate_users_on_replace?: boolean; + deactivate_users_on_replace: boolean; hint?: string; - feature_flags?: Feature[]; + can_create?: boolean; + can_replace_contact?: boolean; + can_move?: boolean; }; export type HierarchyConstraint = { @@ -123,7 +125,7 @@ export class Config { } public static hasMultipleRoles(contactType: ContactType): boolean { - if (contactType.feature_flags?.length === 1 && contactType.feature_flags.includes(Feature.Move)) { + if (contactType.can_move && (contactType.can_create === false && contactType.can_replace_contact === false)) { return false; } if (!contactType.user_role.length || contactType.user_role.some(role => !role.trim())) { diff --git a/src/lib/remote-place-resolver.ts b/src/lib/remote-place-resolver.ts index 487a0ae6..76605c94 100644 --- a/src/lib/remote-place-resolver.ts +++ b/src/lib/remote-place-resolver.ts @@ -112,7 +112,6 @@ export default class RemotePlaceResolver { function getFuzzFunction(place: Place, hierarchyLevel: HierarchyConstraint, contactType: ContactType) { const fuzzingProperty = hierarchyLevel.level === 0 ? contactType.replacement_property : hierarchyLevel; - assert(fuzzingProperty); if (fuzzingProperty.type === 'generated') { throw Error(`Invalid configuration: hierarchy properties cannot be of type "generated".`); } diff --git a/src/liquid/app/nav.html b/src/liquid/app/nav.html index 6602082c..0f5b9b65 100644 --- a/src/liquid/app/nav.html +++ b/src/liquid/app/nav.html @@ -19,18 +19,18 @@