From 7b4693e078cd8b626dba15a3ecab5233d9e9e7e8 Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?Sn=C3=A6r=20Seljan=20=C3=9E=C3=B3roddsson?=
<112904566+snaerseljan@users.noreply.github.com>
Date: Tue, 26 Nov 2024 10:39:57 +0000
Subject: [PATCH 01/12] fix(admin-portal): Fix bff hooks to not use old auth
hooks (#17018)
---
.../CreateDelegation/CreateDelegation.tsx | 49 +++++++++----------
.../DelegationAdmin.tsx | 25 +++++-----
.../delegation-admin/src/screens/Root.tsx | 20 ++++----
.../delegations/DelegationViewModal.tsx | 10 ++--
4 files changed, 51 insertions(+), 53 deletions(-)
diff --git a/libs/portals/admin/delegation-admin/src/screens/CreateDelegation/CreateDelegation.tsx b/libs/portals/admin/delegation-admin/src/screens/CreateDelegation/CreateDelegation.tsx
index 489fdf9c84f3..7e25f55962b6 100644
--- a/libs/portals/admin/delegation-admin/src/screens/CreateDelegation/CreateDelegation.tsx
+++ b/libs/portals/admin/delegation-admin/src/screens/CreateDelegation/CreateDelegation.tsx
@@ -1,54 +1,53 @@
-import React, { useEffect, useState } from 'react'
import {
- Box,
- Stack,
AlertMessage,
+ Box,
Checkbox,
+ DatePicker,
GridColumn,
GridRow,
+ Icon,
Input,
Select,
- DatePicker,
+ Stack,
toast,
- Icon,
} from '@island.is/island-ui/core'
-import { BackButton } from '@island.is/portals/admin/core'
import { useLocale } from '@island.is/localization'
+import { BackButton } from '@island.is/portals/admin/core'
+import React, { useEffect, useState } from 'react'
+import { Identity } from '@island.is/api/schema'
import { IntroHeader, m as coreMessages } from '@island.is/portals/core'
-import { m } from '../../lib/messages'
-import { DelegationAdminPaths } from '../../lib/paths'
-import NumberFormat from 'react-number-format'
+import {
+ DelegationsFormFooter,
+ useDynamicShadow,
+} from '@island.is/portals/shared-modules/delegations'
+import { useUserInfo } from '@island.is/react-spa/bff'
+import { replaceParams } from '@island.is/react-spa/shared'
+import { AuthDelegationType } from '@island.is/shared/types'
+import { maskString, unmaskString } from '@island.is/shared/utils'
+import cn from 'classnames'
import startOfDay from 'date-fns/startOfDay'
+import kennitala from 'kennitala'
+import debounce from 'lodash/debounce'
+import NumberFormat from 'react-number-format'
import {
Form,
- redirect,
useActionData,
useNavigate,
useSearchParams,
useSubmit,
} from 'react-router-dom'
+import { CreateDelegationConfirmModal } from '../../components/CreateDelegationConfirmModal'
+import { FORM_ERRORS } from '../../constants/errors'
+import { m } from '../../lib/messages'
+import { DelegationAdminPaths } from '../../lib/paths'
import { CreateDelegationResult } from './CreateDelegation.action'
import * as styles from './CreateDelegation.css'
import { useIdentityLazyQuery } from './CreateDelegation.generated'
-import debounce from 'lodash/debounce'
-import cn from 'classnames'
-import {
- DelegationsFormFooter,
- useDynamicShadow,
-} from '@island.is/portals/shared-modules/delegations'
-import { CreateDelegationConfirmModal } from '../../components/CreateDelegationConfirmModal'
-import { Identity } from '@island.is/api/schema'
-import kennitala from 'kennitala'
-import { maskString, unmaskString } from '@island.is/shared/utils'
-import { useAuth } from '@island.is/auth/react'
-import { replaceParams } from '@island.is/react-spa/shared'
-import { FORM_ERRORS } from '../../constants/errors'
-import { AuthDelegationType } from '@island.is/shared/types'
const CreateDelegationScreen = () => {
const { formatMessage } = useLocale()
- const { userInfo } = useAuth()
+ const userInfo = useUserInfo()
const navigate = useNavigate()
const [searchParams, setSearchParams] = useSearchParams()
diff --git a/libs/portals/admin/delegation-admin/src/screens/DelegationAdminDetails/DelegationAdmin.tsx b/libs/portals/admin/delegation-admin/src/screens/DelegationAdminDetails/DelegationAdmin.tsx
index daea025576c2..fa47e3700ce1 100644
--- a/libs/portals/admin/delegation-admin/src/screens/DelegationAdminDetails/DelegationAdmin.tsx
+++ b/libs/portals/admin/delegation-admin/src/screens/DelegationAdminDetails/DelegationAdmin.tsx
@@ -1,24 +1,23 @@
-import { Button, GridColumn, Box, Stack, Tabs } from '@island.is/island-ui/core'
-import { BackButton } from '@island.is/portals/admin/core'
-import { useLocale } from '@island.is/localization'
-import { useLoaderData, useNavigate } from 'react-router-dom'
-import { DelegationAdminResult } from './DelegationAdmin.loader'
-import { DelegationAdminPaths } from '../../lib/paths'
-import { formatNationalId, IntroHeader } from '@island.is/portals/core'
-import { m } from '../../lib/messages'
-import React from 'react'
-import DelegationList from '../../components/DelegationList'
import { AuthCustomDelegation } from '@island.is/api/schema'
-import { DelegationsEmptyState } from '@island.is/portals/shared-modules/delegations'
-import { useAuth } from '@island.is/auth/react'
import { AdminPortalScope } from '@island.is/auth/scopes'
+import { Box, Button, GridColumn, Stack, Tabs } from '@island.is/island-ui/core'
+import { useLocale } from '@island.is/localization'
+import { BackButton } from '@island.is/portals/admin/core'
+import { IntroHeader, formatNationalId } from '@island.is/portals/core'
+import { DelegationsEmptyState } from '@island.is/portals/shared-modules/delegations'
+import { useUserInfo } from '@island.is/react-spa/bff'
import { maskString } from '@island.is/shared/utils'
+import { useLoaderData, useNavigate } from 'react-router-dom'
+import DelegationList from '../../components/DelegationList'
+import { m } from '../../lib/messages'
+import { DelegationAdminPaths } from '../../lib/paths'
+import { DelegationAdminResult } from './DelegationAdmin.loader'
const DelegationAdminScreen = () => {
const { formatMessage } = useLocale()
const navigate = useNavigate()
const delegationAdmin = useLoaderData() as DelegationAdminResult
- const { userInfo } = useAuth()
+ const userInfo = useUserInfo()
const hasAdminAccess = userInfo?.scopes.includes(
AdminPortalScope.delegationSystemAdmin,
diff --git a/libs/portals/admin/delegation-admin/src/screens/Root.tsx b/libs/portals/admin/delegation-admin/src/screens/Root.tsx
index 3dd3600dd6c8..d7e8866b9f3f 100644
--- a/libs/portals/admin/delegation-admin/src/screens/Root.tsx
+++ b/libs/portals/admin/delegation-admin/src/screens/Root.tsx
@@ -1,20 +1,20 @@
-import { useLocale } from '@island.is/localization'
-import { IntroHeader } from '@island.is/portals/core'
-import { m } from '../lib/messages'
-import { Form, Outlet, useActionData, useNavigate } from 'react-router-dom'
+import { AdminPortalScope } from '@island.is/auth/scopes'
import {
AsyncSearchInput,
+ Box,
Button,
GridColumn,
GridRow,
- Box,
} from '@island.is/island-ui/core'
-import React, { useEffect, useState } from 'react'
+import { useLocale } from '@island.is/localization'
+import { IntroHeader } from '@island.is/portals/core'
+import { useUserInfo } from '@island.is/react-spa/bff'
import { useSubmitting } from '@island.is/react-spa/shared'
-import { GetDelegationForNationalIdResult } from './Root.action'
+import { useEffect, useState } from 'react'
+import { Form, Outlet, useActionData, useNavigate } from 'react-router-dom'
+import { m } from '../lib/messages'
import { DelegationAdminPaths } from '../lib/paths'
-import { useAuth } from '@island.is/auth/react'
-import { AdminPortalScope } from '@island.is/auth/scopes'
+import { GetDelegationForNationalIdResult } from './Root.action'
const Root = () => {
const [focused, setFocused] = useState(false)
@@ -24,7 +24,7 @@ const Root = () => {
const { isSubmitting, isLoading } = useSubmitting()
const [error, setError] = useState({ hasError: false, message: '' })
const navigate = useNavigate()
- const { userInfo } = useAuth()
+ const userInfo = useUserInfo()
const hasAdminAccess = userInfo?.scopes.includes(
AdminPortalScope.delegationSystemAdmin,
diff --git a/libs/portals/shared-modules/delegations/src/components/delegations/DelegationViewModal.tsx b/libs/portals/shared-modules/delegations/src/components/delegations/DelegationViewModal.tsx
index 277823485167..aaa02e0972ae 100644
--- a/libs/portals/shared-modules/delegations/src/components/delegations/DelegationViewModal.tsx
+++ b/libs/portals/shared-modules/delegations/src/components/delegations/DelegationViewModal.tsx
@@ -1,6 +1,5 @@
-import { useEffect } from 'react'
-import { useAuth } from '@island.is/auth/react'
import { Box } from '@island.is/island-ui/core'
+import { useEffect } from 'react'
import { useLocale } from '@island.is/localization'
import { formatNationalId } from '@island.is/portals/core'
@@ -9,10 +8,11 @@ import { IdentityCard } from '../IdentityCard/IdentityCard'
import { AccessListContainer } from '../access/AccessList/AccessListContainer/AccessListContainer'
import { useAuthScopeTreeLazyQuery } from '../access/AccessList/AccessListContainer/AccessListContainer.generated'
-import { m } from '../../lib/messages'
-import format from 'date-fns/format'
import { AuthDelegationScope, AuthDelegationType } from '@island.is/api/schema'
+import { useUserInfo } from '@island.is/react-spa/bff'
+import format from 'date-fns/format'
import isValid from 'date-fns/isValid'
+import { m } from '../../lib/messages'
type DelegationViewModalProps = {
delegation?: {
@@ -45,7 +45,7 @@ export const DelegationViewModal = ({
...rest
}: DelegationViewModalProps) => {
const { formatMessage, lang } = useLocale()
- const { userInfo } = useAuth()
+ const userInfo = useUserInfo()
const isOutgoing = direction === 'outgoing'
const [getAuthScopeTree, { data: scopeTreeData, loading: scopeTreeLoading }] =
useAuthScopeTreeLazyQuery()
From c815d9622a7068300908ba9ec117d0ca8b2b7ba5 Mon Sep 17 00:00:00 2001
From: albinagu <47886428+albinagu@users.noreply.github.com>
Date: Tue, 26 Nov 2024 10:55:01 +0000
Subject: [PATCH 02/12] fix(efs): taxfree value spouse percentage (#17002)
* fix(efs): taxfree value spouse percentage
* privateTransfer document section
* description - select estate
---------
Co-authored-by: kodiakhq[bot] <49736102+kodiakhq[bot]@users.noreply.github.com>
---
.../src/fields/HeirsRepeater/index.tsx | 8 ++++
.../sections/applicationTypeSelection.ts | 3 +-
.../src/forms/sections/deceased.ts | 4 +-
.../src/forms/sections/heirs.ts | 37 +++++++++++--------
.../inheritance-report/src/lib/messages.ts | 11 ++++--
5 files changed, 41 insertions(+), 22 deletions(-)
diff --git a/libs/application/templates/inheritance-report/src/fields/HeirsRepeater/index.tsx b/libs/application/templates/inheritance-report/src/fields/HeirsRepeater/index.tsx
index 639878598c91..768a0c81c0b4 100644
--- a/libs/application/templates/inheritance-report/src/fields/HeirsRepeater/index.tsx
+++ b/libs/application/templates/inheritance-report/src/fields/HeirsRepeater/index.tsx
@@ -213,9 +213,17 @@ export const HeirsRepeater: FC<
: valueToNumber(getValueViaPath(answers, 'netPropertyForExchange'))
const inheritanceValue = netPropertyForExchange * percentage
+ const customSpouseSharePercentage = getValueViaPath(
+ answers,
+ 'customShare.customSpouseSharePercentage',
+ )
+ const withCustomPercentage =
+ (100 - Number(customSpouseSharePercentage)) * 2
const taxFreeInheritanceValue = isSpouse
? inheritanceValue
+ : customSpouseSharePercentage
+ ? (withCustomPercentage / 100) * inheritanceTaxFreeLimit * percentage
: inheritanceTaxFreeLimit * percentage
const taxableInheritanceValue = inheritanceValue - taxFreeInheritanceValue
diff --git a/libs/application/templates/inheritance-report/src/forms/sections/applicationTypeSelection.ts b/libs/application/templates/inheritance-report/src/forms/sections/applicationTypeSelection.ts
index 10ffd1e7f6fd..2bff9470a59a 100644
--- a/libs/application/templates/inheritance-report/src/forms/sections/applicationTypeSelection.ts
+++ b/libs/application/templates/inheritance-report/src/forms/sections/applicationTypeSelection.ts
@@ -31,7 +31,8 @@ export const preSelection = buildSection({
}),
buildMultiField({
id: 'estate',
- title: m.applicationName,
+ title: m.selectEstate,
+ description: m.selectEstateDescription,
condition: (answers) => answers.applicationFor === ESTATE_INHERITANCE,
children: [
buildSelectField({
diff --git a/libs/application/templates/inheritance-report/src/forms/sections/deceased.ts b/libs/application/templates/inheritance-report/src/forms/sections/deceased.ts
index b679c626a58a..501dfc9f920c 100644
--- a/libs/application/templates/inheritance-report/src/forms/sections/deceased.ts
+++ b/libs/application/templates/inheritance-report/src/forms/sections/deceased.ts
@@ -84,7 +84,7 @@ export const deceased = buildSection({
}),
buildDescriptionField({
id: 'space3',
- space: 'gutter',
+ space: 'containerGutter',
title: '',
}),
buildRadioField({
@@ -123,7 +123,7 @@ export const deceased = buildSection({
),
buildDescriptionField({
id: 'space4',
- space: 'gutter',
+ space: 'containerGutter',
title: '',
}),
buildRadioField({
diff --git a/libs/application/templates/inheritance-report/src/forms/sections/heirs.ts b/libs/application/templates/inheritance-report/src/forms/sections/heirs.ts
index fd82a48babbf..5f6f38eb78eb 100644
--- a/libs/application/templates/inheritance-report/src/forms/sections/heirs.ts
+++ b/libs/application/templates/inheritance-report/src/forms/sections/heirs.ts
@@ -160,6 +160,27 @@ export const heirs = buildSection({
}),
],
}),
+ buildSubSection({
+ id: 'privateTransfer',
+ title: m.fileUploadPrivateTransfer,
+ children: [
+ buildMultiField({
+ id: 'heirsAdditionalInfo',
+ title: m.fileUploadPrivateTransfer,
+ description: m.uploadPrivateTransferUserGuidelines,
+ children: [
+ buildFileUploadField({
+ id: 'heirsAdditionalInfoPrivateTransferFiles',
+ uploadAccept: '.pdf, .doc, .docx, .jpg, .jpeg, .png, .xls, .xlsx',
+ uploadDescription: m.uploadPrivateTransferDescription,
+ title: '',
+ uploadHeader: '',
+ uploadMultiple: false,
+ }),
+ ],
+ }),
+ ],
+ }),
buildSubSection({
id: 'heirsAdditionalInfo',
title: m.heirAdditionalInfo,
@@ -183,22 +204,6 @@ export const heirs = buildSection({
rows: 4,
maxLength: 1800,
}),
- buildDescriptionField({
- id: 'heirsAdditionalInfoFilesPrivateTitle',
- title: m.fileUploadPrivateTransfer,
- description: m.uploadPrivateTransferUserGuidelines,
- titleVariant: 'h5',
- space: 'containerGutter',
- marginBottom: 'smallGutter',
- }),
- buildFileUploadField({
- id: 'heirsAdditionalInfoPrivateTransferFiles',
- uploadAccept: '.pdf, .doc, .docx, .jpg, .jpeg, .png, .xls, .xlsx',
- uploadDescription: m.uploadPrivateTransferDescription,
- title: '',
- uploadHeader: '',
- uploadMultiple: false,
- }),
buildDescriptionField({
id: 'heirsAdditionalInfoFilesOtherDocumentsTitle',
title: m.fileUploadOtherDocuments,
diff --git a/libs/application/templates/inheritance-report/src/lib/messages.ts b/libs/application/templates/inheritance-report/src/lib/messages.ts
index 0f079f314435..c13308ad57bc 100644
--- a/libs/application/templates/inheritance-report/src/lib/messages.ts
+++ b/libs/application/templates/inheritance-report/src/lib/messages.ts
@@ -67,9 +67,14 @@ export const m = defineMessages({
description: '',
},
// Application begin
- applicationName: {
- id: 'ir.application:applicationName',
- defaultMessage: 'Erfðafjárskýrsla eftir andlát',
+ selectEstate: {
+ id: 'ir.application:selectEstate',
+ defaultMessage: 'Veldu dánarbú',
+ description: '',
+ },
+ selectEstateDescription: {
+ id: 'ir.application:selectEstateDescription',
+ defaultMessage: 'Lorem ipsum dolor sit amet, consectetur adipiscing elit.',
description: '',
},
applicationNamePrepaid: {
From 41721f33cbd8e726c7805ba5e6440984dddfd2f1 Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?J=C3=B3nas=20G=2E=20Sigur=C3=B0sson?=
Date: Tue, 26 Nov 2024 11:19:21 +0000
Subject: [PATCH 03/12] feat(application-system): new shared component
buildFieldsRepeaterField (#16871)
* feat: start of fieldsRepeaterField
* feat: Make a copy of table repeater and update naming
* feat: make sure to throw away answers when repeater item is moved.
* .
* feat: additional options for fieldsRepeater
* chore: remove console.log
* feat: handle edge case going back and forth
* chore: revert back changes to the example form
* feat: update storybook
---------
Co-authored-by: kodiakhq[bot] <49736102+kodiakhq[bot]@users.noreply.github.com>
---
.../application/core/src/lib/fieldBuilders.ts | 43 ++++
libs/application/core/src/lib/messages.ts | 5 +
libs/application/types/src/lib/Fields.ts | 46 +++-
.../FieldsRepeaterFormField.stories.mdx | 128 ++++++++++
.../FieldsRepeaterFormField.tsx | 205 ++++++++++++++++
.../FieldsRepeaterItem.tsx | 223 ++++++++++++++++++
.../TableRepeaterItem.tsx | 6 +-
.../src/lib/TableRepeaterFormField/utils.ts | 8 +-
libs/application/ui-fields/src/lib/index.ts | 1 +
9 files changed, 654 insertions(+), 11 deletions(-)
create mode 100644 libs/application/ui-fields/src/lib/FieldsRepeaterFormField/FieldsRepeaterFormField.stories.mdx
create mode 100644 libs/application/ui-fields/src/lib/FieldsRepeaterFormField/FieldsRepeaterFormField.tsx
create mode 100644 libs/application/ui-fields/src/lib/FieldsRepeaterFormField/FieldsRepeaterItem.tsx
diff --git a/libs/application/core/src/lib/fieldBuilders.ts b/libs/application/core/src/lib/fieldBuilders.ts
index 2f1a86f6a03c..dabe437861a0 100644
--- a/libs/application/core/src/lib/fieldBuilders.ts
+++ b/libs/application/core/src/lib/fieldBuilders.ts
@@ -46,6 +46,7 @@ import {
SliderField,
MaybeWithApplication,
MaybeWithApplicationAndFieldAndLocale,
+ FieldsRepeaterField,
} from '@island.is/application/types'
import { Locale } from '@island.is/shared/types'
import { Colors } from '@island.is/island-ui/theme'
@@ -867,6 +868,48 @@ export const buildTableRepeaterField = (
}
}
+export const buildFieldsRepeaterField = (
+ data: Omit,
+): FieldsRepeaterField => {
+ const {
+ fields,
+ table,
+ title,
+ titleVariant,
+ formTitle,
+ formTitleVariant,
+ formTitleNumbering,
+ marginTop,
+ marginBottom,
+ removeItemButtonText,
+ addItemButtonText,
+ saveItemButtonText,
+ minRows,
+ maxRows,
+ } = data
+
+ return {
+ ...extractCommonFields(data),
+ children: undefined,
+ type: FieldTypes.FIELDS_REPEATER,
+ component: FieldComponents.FIELDS_REPEATER,
+ fields,
+ table,
+ title,
+ titleVariant,
+ formTitle,
+ formTitleVariant,
+ formTitleNumbering,
+ marginTop,
+ marginBottom,
+ removeItemButtonText,
+ addItemButtonText,
+ saveItemButtonText,
+ minRows,
+ maxRows,
+ }
+}
+
export const buildStaticTableField = (
data: Omit<
StaticTableField,
diff --git a/libs/application/core/src/lib/messages.ts b/libs/application/core/src/lib/messages.ts
index 3190a3542b1a..65cb2f5cdd67 100644
--- a/libs/application/core/src/lib/messages.ts
+++ b/libs/application/core/src/lib/messages.ts
@@ -41,6 +41,11 @@ export const coreMessages = defineMessages({
defaultMessage: 'Bæta við',
description: 'Add button',
},
+ buttonRemove: {
+ id: 'application.system:button.remove',
+ defaultMessage: 'Fjarlægja',
+ description: 'Remove button',
+ },
buttonCancel: {
id: 'application.system:button.cancel',
defaultMessage: 'Hætta við',
diff --git a/libs/application/types/src/lib/Fields.ts b/libs/application/types/src/lib/Fields.ts
index 9ea37be69132..c6763dbb9141 100644
--- a/libs/application/types/src/lib/Fields.ts
+++ b/libs/application/types/src/lib/Fields.ts
@@ -64,7 +64,7 @@ export type TagVariant =
| 'mint'
| 'disabled'
-export type TableRepeaterFields =
+export type RepeaterFields =
| 'input'
| 'select'
| 'radio'
@@ -82,8 +82,8 @@ type TableRepeaterOptions =
activeField?: Record,
) => RepeaterOption[] | [])
-export type TableRepeaterItem = {
- component: TableRepeaterFields
+export type RepeaterItem = {
+ component: RepeaterFields
/**
* Defaults to true
*/
@@ -253,6 +253,7 @@ export enum FieldTypes {
NATIONAL_ID_WITH_NAME = 'NATIONAL_ID_WITH_NAME',
ACTION_CARD_LIST = 'ACTION_CARD_LIST',
TABLE_REPEATER = 'TABLE_REPEATER',
+ FIELDS_REPEATER = 'FIELDS_REPEATER',
HIDDEN_INPUT = 'HIDDEN_INPUT',
HIDDEN_INPUT_WITH_WATCHED_VALUE = 'HIDDEN_INPUT_WITH_WATCHED_VALUE',
FIND_VEHICLE = 'FIND_VEHICLE',
@@ -289,6 +290,7 @@ export enum FieldComponents {
NATIONAL_ID_WITH_NAME = 'NationalIdWithNameFormField',
ACTION_CARD_LIST = 'ActionCardListFormField',
TABLE_REPEATER = 'TableRepeaterFormField',
+ FIELDS_REPEATER = 'FieldsRepeaterFormField',
HIDDEN_INPUT = 'HiddenInputFormField',
FIND_VEHICLE = 'FindVehicleFormField',
VEHICLE_RADIO = 'VehicleRadioFormField',
@@ -639,7 +641,7 @@ export type TableRepeaterField = BaseField & {
marginTop?: ResponsiveProp
marginBottom?: ResponsiveProp
titleVariant?: TitleVariants
- fields: Record
+ fields: Record
/**
* Maximum rows that can be added to the table.
* When the maximum is reached, the button to add a new row is disabled.
@@ -659,6 +661,41 @@ export type TableRepeaterField = BaseField & {
format?: Record string | StaticText>
}
}
+
+export type FieldsRepeaterField = BaseField & {
+ readonly type: FieldTypes.FIELDS_REPEATER
+ component: FieldComponents.FIELDS_REPEATER
+ titleVariant?: TitleVariants
+ formTitle?: StaticText
+ formTitleVariant?: TitleVariants
+ formTitleNumbering?: 'prefix' | 'suffix' | 'none'
+ removeItemButtonText?: StaticText
+ addItemButtonText?: StaticText
+ saveItemButtonText?: StaticText
+ marginTop?: ResponsiveProp
+ marginBottom?: ResponsiveProp
+ fields: Record
+ /**
+ * Maximum rows that can be added to the table.
+ * When the maximum is reached, the button to add a new row is disabled.
+ */
+ minRows?: number
+ maxRows?: number
+ table?: {
+ /**
+ * List of strings to render,
+ * if not provided it will be auto generated from the fields
+ */
+ header?: StaticText[]
+ /**
+ * List of field id's to render,
+ * if not provided it will be auto generated from the fields
+ */
+ rows?: string[]
+ format?: Record string | StaticText>
+ }
+}
+
export interface FindVehicleField extends InputField {
readonly type: FieldTypes.FIND_VEHICLE
component: FieldComponents.FIND_VEHICLE
@@ -786,6 +823,7 @@ export type Field =
| NationalIdWithNameField
| ActionCardListField
| TableRepeaterField
+ | FieldsRepeaterField
| HiddenInputWithWatchedValueField
| HiddenInputField
| FindVehicleField
diff --git a/libs/application/ui-fields/src/lib/FieldsRepeaterFormField/FieldsRepeaterFormField.stories.mdx b/libs/application/ui-fields/src/lib/FieldsRepeaterFormField/FieldsRepeaterFormField.stories.mdx
new file mode 100644
index 000000000000..081dc5c3a1dc
--- /dev/null
+++ b/libs/application/ui-fields/src/lib/FieldsRepeaterFormField/FieldsRepeaterFormField.stories.mdx
@@ -0,0 +1,128 @@
+import {
+ Meta,
+ Story,
+ Canvas,
+ ArgsTable,
+ Description,
+ Source,
+} from '@storybook/addon-docs'
+import { dedent } from 'ts-dedent'
+
+import { FieldsRepeaterFormField } from './FieldsRepeaterFormField'
+
+export const createMockApplication = (data = {}) => ({
+ id: '123',
+ assignees: [],
+ state: data.state || 'draft',
+ applicant: '111111-3000',
+ typeId: data.typeId || 'ExampleForm',
+ modified: new Date(),
+ created: new Date(),
+ attachments: {},
+ answers: data.answers || {},
+ externalData: data.externalData || {},
+})
+
+
+
+# TableRepeaterFormField
+
+### Usage in a template
+
+You can create a FieldsRepeaterFormField using the following function `FieldsTableRepeaterField`.
+Validation should be done via zod schema.
+
+
+
+The previous configuration object will result in the following component:
+
+
+
+You can also use this field into a custom component by using `` with the configuration object defined above.
+
+# Props
+
+## FieldsRepeaterFormField
+
+
diff --git a/libs/application/ui-fields/src/lib/FieldsRepeaterFormField/FieldsRepeaterFormField.tsx b/libs/application/ui-fields/src/lib/FieldsRepeaterFormField/FieldsRepeaterFormField.tsx
new file mode 100644
index 000000000000..9edc6dd1890e
--- /dev/null
+++ b/libs/application/ui-fields/src/lib/FieldsRepeaterFormField/FieldsRepeaterFormField.tsx
@@ -0,0 +1,205 @@
+import { Fragment, useEffect, useState } from 'react'
+import {
+ coreMessages,
+ formatText,
+ formatTextWithLocale,
+ getValueViaPath,
+} from '@island.is/application/core'
+import {
+ FieldBaseProps,
+ FieldsRepeaterField,
+} from '@island.is/application/types'
+import {
+ AlertMessage,
+ Box,
+ Button,
+ GridRow,
+ Stack,
+ Text,
+} from '@island.is/island-ui/core'
+import { useLocale } from '@island.is/localization'
+import { FieldDescription } from '@island.is/shared/form-fields'
+import { useFieldArray, useFormContext, useWatch } from 'react-hook-form'
+import { Item } from './FieldsRepeaterItem'
+import { Locale } from '@island.is/shared/types'
+import isEqual from 'lodash/isEqual'
+
+interface Props extends FieldBaseProps {
+ field: FieldsRepeaterField
+}
+
+export const FieldsRepeaterFormField = ({
+ application,
+ field: data,
+ showFieldName,
+ error,
+}: Props) => {
+ const {
+ id,
+ fields: rawItems,
+ description,
+ marginTop = 6,
+ marginBottom,
+ title,
+ titleVariant = 'h2',
+ formTitle,
+ formTitleVariant = 'h4',
+ formTitleNumbering = 'suffix',
+ removeItemButtonText = coreMessages.buttonRemove,
+ addItemButtonText = coreMessages.buttonAdd,
+ minRows = 1,
+ maxRows,
+ } = data
+
+ const { control, getValues, setValue } = useFormContext()
+ const answers = getValues()
+ const numberOfItemsInAnswers = getValueViaPath>(
+ answers,
+ id,
+ )?.length
+ const [numberOfItems, setNumberOfItems] = useState(
+ Math.max(numberOfItemsInAnswers ?? 0, minRows),
+ )
+ const [updatedApplication, setUpdatedApplication] = useState(application)
+
+ useEffect(() => {
+ if (!isEqual(application, updatedApplication)) {
+ setUpdatedApplication({
+ ...application,
+ answers: { ...answers },
+ })
+ }
+ }, [answers])
+
+ const items = Object.keys(rawItems).map((key) => ({
+ id: key,
+ ...rawItems[key],
+ }))
+
+ const { formatMessage, lang: locale } = useLocale()
+
+ const { fields, remove } = useFieldArray({
+ control: control,
+ name: id,
+ })
+
+ const values = useWatch({ name: data.id, control: control })
+
+ const handleNewItem = () => {
+ setNumberOfItems(numberOfItems + 1)
+ }
+
+ const handleRemoveItem = () => {
+ if (numberOfItems > (numberOfItemsInAnswers || 0)) {
+ setNumberOfItems(numberOfItems - 1)
+ } else if (numberOfItems === numberOfItemsInAnswers) {
+ setValue(id, answers[id].slice(0, -1))
+ setNumberOfItems(numberOfItems - 1)
+ } else if (
+ numberOfItemsInAnswers &&
+ numberOfItems < numberOfItemsInAnswers
+ ) {
+ const difference = numberOfItems - numberOfItemsInAnswers
+ setValue(id, answers[id].slice(0, difference))
+ setNumberOfItems(numberOfItems)
+ }
+
+ remove(numberOfItems - 1)
+ }
+
+ const repeaterFields = (index: number) =>
+ items.map((item) => (
+
+ ))
+
+ return (
+
+ {showFieldName && (
+
+ {formatTextWithLocale(
+ title,
+ application,
+ locale as Locale,
+ formatMessage,
+ )}
+
+ )}
+ {description && (
+
+ )}
+
+
+
+ {Array.from({ length: numberOfItems }).map((_i, i) => (
+
+ {(formTitleNumbering !== 'none' || formTitle) && (
+
+
+ {formTitleNumbering === 'prefix' ? `${i + 1}. ` : ''}
+ {formTitle &&
+ formatTextWithLocale(
+ formTitle,
+ application,
+ locale as Locale,
+ formatMessage,
+ )}
+ {formTitleNumbering === 'suffix' ? ` ${i + 1}` : ''}
+
+
+ )}
+ {repeaterFields(i)}
+
+ ))}
+
+
+ {numberOfItems > minRows && (
+
+
+
+ )}
+
+
+
+ {error && typeof error === 'string' && fields.length === 0 && (
+
+
+
+ )}
+
+
+ )
+}
diff --git a/libs/application/ui-fields/src/lib/FieldsRepeaterFormField/FieldsRepeaterItem.tsx b/libs/application/ui-fields/src/lib/FieldsRepeaterFormField/FieldsRepeaterItem.tsx
new file mode 100644
index 000000000000..4fd041e97550
--- /dev/null
+++ b/libs/application/ui-fields/src/lib/FieldsRepeaterFormField/FieldsRepeaterItem.tsx
@@ -0,0 +1,223 @@
+import { formatText, getValueViaPath } from '@island.is/application/core'
+import { Application, RepeaterItem } from '@island.is/application/types'
+import { GridColumn, Text } from '@island.is/island-ui/core'
+import { useLocale } from '@island.is/localization'
+import { useEffect, useRef } from 'react'
+import { useFormContext } from 'react-hook-form'
+import isEqual from 'lodash/isEqual'
+import {
+ CheckboxController,
+ DatePickerController,
+ InputController,
+ RadioController,
+ SelectController,
+ PhoneInputController,
+} from '@island.is/shared/form-fields'
+import { NationalIdWithName } from '@island.is/application/ui-components'
+
+interface ItemFieldProps {
+ application: Application
+ error?: string
+ item: RepeaterItem & { id: string }
+ dataId: string
+ index: number
+ values: Array>
+}
+
+const componentMapper = {
+ input: InputController,
+ select: SelectController,
+ checkbox: CheckboxController,
+ date: DatePickerController,
+ radio: RadioController,
+ nationalIdWithName: NationalIdWithName,
+ phone: PhoneInputController,
+}
+
+export const Item = ({
+ application,
+ error,
+ item,
+ dataId,
+ index,
+ values,
+}: ItemFieldProps) => {
+ const { formatMessage } = useLocale()
+ const { setValue, control, clearErrors } = useFormContext()
+ const prevWatchedValuesRef = useRef()
+
+ const {
+ component,
+ id: itemId,
+ backgroundColor = 'blue',
+ label = '',
+ placeholder = '',
+ options,
+ width = 'full',
+ condition,
+ readonly = false,
+ disabled = false,
+ updateValueObj,
+ defaultValue,
+ ...props
+ } = item
+ const isHalfColumn = component !== 'radio' && width === 'half'
+ const isThirdColumn = component !== 'radio' && width === 'third'
+ const span = isHalfColumn ? '1/2' : isThirdColumn ? '1/3' : '1/1'
+ const Component = componentMapper[component]
+ const id = `${dataId}[${index}].${itemId}`
+ const activeValues = index >= 0 && values ? values[index] : undefined
+
+ let watchedValues: string | (string | undefined)[] | undefined
+ if (updateValueObj) {
+ const watchedValuesId =
+ typeof updateValueObj.watchValues === 'function'
+ ? updateValueObj.watchValues(activeValues)
+ : updateValueObj.watchValues
+
+ if (watchedValuesId) {
+ if (Array.isArray(watchedValuesId)) {
+ watchedValues = watchedValuesId.map((value) => {
+ return activeValues?.[`${value}`]
+ })
+ } else {
+ watchedValues = activeValues?.[`${watchedValuesId}`]
+ }
+ }
+ }
+
+ useEffect(() => {
+ // We need to deep compare the watched values to avoid unnecessary re-renders
+ if (
+ watchedValues &&
+ !isEqual(prevWatchedValuesRef.current, watchedValues)
+ ) {
+ prevWatchedValuesRef.current = watchedValues
+ if (
+ updateValueObj &&
+ watchedValues &&
+ (Array.isArray(watchedValues)
+ ? !watchedValues.every((value) => value === undefined)
+ : true)
+ ) {
+ const finalValue = updateValueObj.valueModifier(
+ application,
+ activeValues,
+ )
+ setValue(id, finalValue)
+ }
+ }
+ }, [watchedValues, updateValueObj, activeValues, setValue, id])
+
+ const getFieldError = (id: string) => {
+ /**
+ * Errors that occur in a field-array have incorrect typing
+ * This hack is needed to get the correct type
+ */
+ const errorList = error as unknown as Record[] | undefined
+ const errors = errorList?.[index]
+ return errors?.[id]
+ }
+
+ const getDefaultValue = (
+ item: RepeaterItem,
+ application: Application,
+ activeField?: Record,
+ ) => {
+ const { defaultValue } = item
+
+ if (defaultValue === undefined) {
+ return undefined
+ }
+
+ return defaultValue(application, activeField)
+ }
+
+ let translatedOptions: any = []
+ if (typeof options === 'function') {
+ translatedOptions = options(application, activeValues)
+ } else {
+ translatedOptions =
+ options?.map((option) => ({
+ ...option,
+ label: formatText(option.label, application, formatMessage),
+ ...(option.tooltip && {
+ tooltip: formatText(option.tooltip, application, formatMessage),
+ }),
+ })) ?? []
+ }
+
+ let Readonly: boolean | undefined
+ if (typeof readonly === 'function') {
+ Readonly = readonly(application, activeValues)
+ } else {
+ Readonly = readonly
+ }
+
+ let Disabled: boolean | undefined
+ if (typeof disabled === 'function') {
+ Disabled = disabled(application, activeValues)
+ } else {
+ Disabled = disabled
+ }
+
+ let DefaultValue: any
+ if (component === 'input') {
+ DefaultValue = getDefaultValue(item, application, activeValues)
+ }
+ if (component === 'select') {
+ DefaultValue =
+ getValueViaPath(application.answers, id) ??
+ getDefaultValue(item, application, activeValues)
+ }
+ if (component === 'radio') {
+ DefaultValue =
+ (getValueViaPath(application.answers, id) as string[]) ??
+ getDefaultValue(item, application, activeValues)
+ }
+ if (component === 'checkbox') {
+ DefaultValue =
+ (getValueViaPath(application.answers, id) as string[]) ??
+ getDefaultValue(item, application, activeValues)
+ }
+ if (component === 'date') {
+ DefaultValue =
+ (getValueViaPath(application.answers, id) as string) ??
+ getDefaultValue(item, application, activeValues)
+ }
+
+ if (condition && !condition(application, activeValues)) {
+ return null
+ }
+
+ return (
+
+ {component === 'radio' && label && (
+
+ {formatText(label, application, formatMessage)}
+
+ )}
+ {
+ if (error) {
+ clearErrors(id)
+ }
+ }}
+ application={application}
+ defaultValue={DefaultValue}
+ {...props}
+ />
+
+ )
+}
diff --git a/libs/application/ui-fields/src/lib/TableRepeaterFormField/TableRepeaterItem.tsx b/libs/application/ui-fields/src/lib/TableRepeaterFormField/TableRepeaterItem.tsx
index dc5c01ae8309..5388065f176f 100644
--- a/libs/application/ui-fields/src/lib/TableRepeaterFormField/TableRepeaterItem.tsx
+++ b/libs/application/ui-fields/src/lib/TableRepeaterFormField/TableRepeaterItem.tsx
@@ -1,5 +1,5 @@
import { formatText, getValueViaPath } from '@island.is/application/core'
-import { Application, TableRepeaterItem } from '@island.is/application/types'
+import { Application, RepeaterItem } from '@island.is/application/types'
import { GridColumn, Text } from '@island.is/island-ui/core'
import { useLocale } from '@island.is/localization'
import { useEffect, useRef } from 'react'
@@ -18,7 +18,7 @@ import { NationalIdWithName } from '@island.is/application/ui-components'
interface ItemFieldProps {
application: Application
error?: string
- item: TableRepeaterItem & { id: string }
+ item: RepeaterItem & { id: string }
dataId: string
activeIndex: number
values: Array>
@@ -121,7 +121,7 @@ export const Item = ({
}
const getDefaultValue = (
- item: TableRepeaterItem,
+ item: RepeaterItem,
application: Application,
activeField?: Record,
) => {
diff --git a/libs/application/ui-fields/src/lib/TableRepeaterFormField/utils.ts b/libs/application/ui-fields/src/lib/TableRepeaterFormField/utils.ts
index f1b0c79ee077..7843b128e023 100644
--- a/libs/application/ui-fields/src/lib/TableRepeaterFormField/utils.ts
+++ b/libs/application/ui-fields/src/lib/TableRepeaterFormField/utils.ts
@@ -1,9 +1,9 @@
+import { RepeaterItem } from '@island.is/application/types'
import { coreMessages } from '@island.is/application/core'
-import { TableRepeaterItem } from '@island.is/application/types'
type Item = {
id: string
-} & TableRepeaterItem
+} & RepeaterItem
export type Value = { [key: string]: T }
@@ -42,7 +42,7 @@ const handleNationalIdWithNameItem = (
return newValues
}
-export const buildDefaultTableHeader = (items: Array) =>
+export const buildDefaultTableHeader = (items: Array) =>
items
.map((item) =>
// nationalIdWithName is a special case where the value is an object of name and nationalId
@@ -53,7 +53,7 @@ export const buildDefaultTableHeader = (items: Array) =>
.flat(2)
export const buildDefaultTableRows = (
- items: Array,
+ items: Array,
) =>
items
.map((item) =>
diff --git a/libs/application/ui-fields/src/lib/index.ts b/libs/application/ui-fields/src/lib/index.ts
index c3e3f9724309..5ca103c7e046 100644
--- a/libs/application/ui-fields/src/lib/index.ts
+++ b/libs/application/ui-fields/src/lib/index.ts
@@ -24,6 +24,7 @@ export { NationalIdWithNameFormField } from './NationalIdWithNameFormField/Natio
export { HiddenInputFormField } from './HiddenInputFormField/HiddenInputFormField'
export { ActionCardListFormField } from './ActionCardListFormField/ActionCardListFormField'
export { TableRepeaterFormField } from './TableRepeaterFormField/TableRepeaterFormField'
+export { FieldsRepeaterFormField } from './FieldsRepeaterFormField/FieldsRepeaterFormField'
export { FindVehicleFormField } from './FindVehicleFormField/FindVehicleFormField'
export { VehicleRadioFormField } from './VehicleRadioFormField/VehicleRadioFormField'
export { StaticTableFormField } from './StaticTableFormField/StaticTableFormField'
From 050a2ebbe1cc13b3af0d1da200f4554b2dbb8bc6 Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?Magnea=20R=C3=BAn=20Vignisd=C3=B3ttir?=
Date: Tue, 26 Nov 2024 11:34:05 +0000
Subject: [PATCH 04/12] fix(portals-admin): Stop redirecting when only one
result (#17020)
causes loop when going back now that we have query in url
Co-authored-by: kodiakhq[bot] <49736102+kodiakhq[bot]@users.noreply.github.com>
---
.../src/screens/Companies/Companies.action.ts | 14 --------------
.../src/screens/Companies/Companies.tsx | 4 +---
.../src/screens/Users/Users.action.ts | 16 ----------------
.../service-desk/src/screens/Users/Users.tsx | 4 +---
4 files changed, 2 insertions(+), 36 deletions(-)
diff --git a/libs/portals/admin/service-desk/src/screens/Companies/Companies.action.ts b/libs/portals/admin/service-desk/src/screens/Companies/Companies.action.ts
index 6110e892e1b6..106e56adce3c 100644
--- a/libs/portals/admin/service-desk/src/screens/Companies/Companies.action.ts
+++ b/libs/portals/admin/service-desk/src/screens/Companies/Companies.action.ts
@@ -68,20 +68,6 @@ export const CompaniesAction: WrappedActionFn =
throw res.error
}
- const companies = res.data?.companyRegistryCompanies?.data
-
- // Redirect to Procurers screen if only one company is found
- if (companies?.length === 1) {
- return redirect(
- replaceParams({
- href: ServiceDeskPaths.Company,
- params: {
- nationalId: companies[0].nationalId,
- },
- }),
- )
- }
-
return {
errors: null,
data: res.data.companyRegistryCompanies,
diff --git a/libs/portals/admin/service-desk/src/screens/Companies/Companies.tsx b/libs/portals/admin/service-desk/src/screens/Companies/Companies.tsx
index 2955edb6e8fe..e15a2c5115f6 100644
--- a/libs/portals/admin/service-desk/src/screens/Companies/Companies.tsx
+++ b/libs/portals/admin/service-desk/src/screens/Companies/Companies.tsx
@@ -79,14 +79,12 @@ const Companies = () => {
buttonProps={{
type: 'submit',
disabled: !searchInput,
- onClick: (e) => {
- e.preventDefault()
+ onClick: () => {
if (searchInput) {
setSearchParams((params) => {
params.set('q', searchInput)
return params
})
- submit(formRef.current)
}
},
}}
diff --git a/libs/portals/admin/service-desk/src/screens/Users/Users.action.ts b/libs/portals/admin/service-desk/src/screens/Users/Users.action.ts
index 4d723cad3321..13454e204aaa 100644
--- a/libs/portals/admin/service-desk/src/screens/Users/Users.action.ts
+++ b/libs/portals/admin/service-desk/src/screens/Users/Users.action.ts
@@ -78,22 +78,6 @@ export const UsersAction: WrappedActionFn =
throw res.error
}
- const { data: respData, totalCount } = res.data.UserProfileAdminProfiles
-
- if (totalCount === 1) {
- return redirect(
- replaceParams({
- href: ServiceDeskPaths.User,
- params: {
- nationalId: (await maskString(
- respData[0].nationalId,
- userInfo.profile.nationalId,
- )) as string,
- },
- }),
- )
- }
-
return {
data: res.data.UserProfileAdminProfiles,
errors: null,
diff --git a/libs/portals/admin/service-desk/src/screens/Users/Users.tsx b/libs/portals/admin/service-desk/src/screens/Users/Users.tsx
index 694c3357a959..642110a7ad76 100644
--- a/libs/portals/admin/service-desk/src/screens/Users/Users.tsx
+++ b/libs/portals/admin/service-desk/src/screens/Users/Users.tsx
@@ -87,14 +87,12 @@ const Users = () => {
buttonProps={{
type: 'submit',
disabled: searchInput.length === 0,
- onClick: (e) => {
- e.preventDefault()
+ onClick: () => {
if (searchInput) {
setSearchParams((params) => {
params.set('q', searchInput)
return params
})
- submit(formRef.current)
}
},
}}
From ffb273ca62323c1b265a5db39797f83d158c3157 Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?R=C3=BAnar=20Vestmann?=
<43557895+RunarVestmann@users.noreply.github.com>
Date: Tue, 26 Nov 2024 14:12:55 +0000
Subject: [PATCH 05/12] feat(web): Organization parent subpage (#17022)
* First draft
* Stop fetching organization page twice
* Handle if there is only a single child link
* Update apps/web/screens/Organization/OrganizationNews/OrganizationNewsArticle.tsx
Co-authored-by: coderabbitai[bot] <136622811+coderabbitai[bot]@users.noreply.github.com>
* Add errorHandler to backend endpoint
* Remove question mark
* Filter out invalid child links
---------
Co-authored-by: coderabbitai[bot] <136622811+coderabbitai[bot]@users.noreply.github.com>
Co-authored-by: kodiakhq[bot] <49736102+kodiakhq[bot]@users.noreply.github.com>
---
apps/web/pages/s/[...slugs]/index.tsx | 99 ++++---
.../OrganizationSubPageGenericListItem.tsx | 12 +-
.../OrganizationEventArticle.tsx | 43 +--
.../OrganizationNewsArticle.tsx | 47 ++--
.../Organization/Standalone/ParentSubpage.tsx | 251 ++++++++++++++++++
apps/web/screens/Organization/SubPage.tsx | 48 +++-
apps/web/screens/queries/Organization.tsx | 14 +
libs/cms/src/lib/cms.contentful.service.ts | 26 +-
libs/cms/src/lib/cms.resolver.ts | 10 +
.../dto/getOrganizationParentSubpage.input.ts | 18 ++
.../models/organizationParentSubpage.model.ts | 48 ++++
11 files changed, 526 insertions(+), 90 deletions(-)
create mode 100644 apps/web/screens/Organization/Standalone/ParentSubpage.tsx
create mode 100644 libs/cms/src/lib/dto/getOrganizationParentSubpage.input.ts
create mode 100644 libs/cms/src/lib/models/organizationParentSubpage.model.ts
diff --git a/apps/web/pages/s/[...slugs]/index.tsx b/apps/web/pages/s/[...slugs]/index.tsx
index 8c2cbde982af..5632dbe156ba 100644
--- a/apps/web/pages/s/[...slugs]/index.tsx
+++ b/apps/web/pages/s/[...slugs]/index.tsx
@@ -31,6 +31,9 @@ import PublishedMaterial, {
import StandaloneHome, {
type StandaloneHomeProps,
} from '@island.is/web/screens/Organization/Standalone/Home'
+import StandaloneParentSubpage, {
+ StandaloneParentSubpageProps,
+} from '@island.is/web/screens/Organization/Standalone/ParentSubpage'
import SubPage, {
type SubPageProps,
} from '@island.is/web/screens/Organization/SubPage'
@@ -42,6 +45,7 @@ import { getServerSidePropsWrapper } from '@island.is/web/utils/getServerSidePro
enum PageType {
FRONTPAGE = 'frontpage',
STANDALONE_FRONTPAGE = 'standalone-frontpage',
+ STANDALONE_PARENT_SUBPAGE = 'standalone-parent-subpage',
SUBPAGE = 'subpage',
ALL_NEWS = 'news',
PUBLISHED_MATERIAL = 'published-material',
@@ -55,6 +59,9 @@ enum PageType {
const pageMap: Record> = {
[PageType.FRONTPAGE]: (props) => ,
[PageType.STANDALONE_FRONTPAGE]: (props) => ,
+ [PageType.STANDALONE_PARENT_SUBPAGE]: (props) => (
+
+ ),
[PageType.SUBPAGE]: (props) => ,
[PageType.ALL_NEWS]: (props) => ,
[PageType.PUBLISHED_MATERIAL]: (props) => ,
@@ -79,6 +86,10 @@ interface Props {
type: PageType.STANDALONE_FRONTPAGE
props: StandaloneHomeProps
}
+ | {
+ type: PageType.STANDALONE_PARENT_SUBPAGE
+ props: StandaloneParentSubpageProps
+ }
| {
type: PageType.SUBPAGE
props: {
@@ -135,32 +146,32 @@ Component.getProps = async (context) => {
const slugs = context.query.slugs as string[]
const locale = context.locale || 'is'
- // Frontpage
- if (slugs.length === 1) {
- const {
- data: { getOrganizationPage: organizationPage },
- } = await context.apolloClient.query({
- query: GET_ORGANIZATION_PAGE_QUERY,
- variables: {
- input: {
- slug: slugs[0],
- lang: locale,
- },
+ const {
+ data: { getOrganizationPage: organizationPage },
+ } = await context.apolloClient.query({
+ query: GET_ORGANIZATION_PAGE_QUERY,
+ variables: {
+ input: {
+ slug: slugs[0],
+ lang: locale,
},
- })
+ },
+ })
- if (!organizationPage) {
- throw new CustomNextError(404)
- }
+ if (!organizationPage) {
+ throw new CustomNextError(404, 'Organization page was not found')
+ }
+
+ const modifiedContext = { ...context, organizationPage }
+
+ const STANDALONE_THEME = 'standalone'
- if (organizationPage.theme === 'standalone') {
+ if (slugs.length === 1) {
+ if (organizationPage.theme === STANDALONE_THEME) {
return {
page: {
type: PageType.STANDALONE_FRONTPAGE,
- props: await StandaloneHome.getProps({
- ...context,
- organizationPage,
- }),
+ props: await StandaloneHome.getProps(modifiedContext),
},
}
}
@@ -168,7 +179,7 @@ Component.getProps = async (context) => {
return {
page: {
type: PageType.FRONTPAGE,
- props: await Home.getProps({ ...context, organizationPage }),
+ props: await Home.getProps(modifiedContext),
},
}
}
@@ -179,7 +190,7 @@ Component.getProps = async (context) => {
return {
page: {
type: PageType.ALL_NEWS,
- props: await OrganizationNewsList.getProps(context),
+ props: await OrganizationNewsList.getProps(modifiedContext),
},
}
}
@@ -187,7 +198,7 @@ Component.getProps = async (context) => {
return {
page: {
type: PageType.ALL_EVENTS,
- props: await OrganizationEventList.getProps(context),
+ props: await OrganizationEventList.getProps(modifiedContext),
},
}
}
@@ -195,7 +206,7 @@ Component.getProps = async (context) => {
return {
page: {
type: PageType.PUBLISHED_MATERIAL,
- props: await PublishedMaterial.getProps(context),
+ props: await PublishedMaterial.getProps(modifiedContext),
},
}
}
@@ -204,7 +215,7 @@ Component.getProps = async (context) => {
return {
page: {
type: PageType.ALL_NEWS,
- props: await OrganizationNewsList.getProps(context),
+ props: await OrganizationNewsList.getProps(modifiedContext),
},
}
}
@@ -212,7 +223,7 @@ Component.getProps = async (context) => {
return {
page: {
type: PageType.ALL_EVENTS,
- props: await OrganizationEventList.getProps(context),
+ props: await OrganizationEventList.getProps(modifiedContext),
},
}
}
@@ -220,18 +231,25 @@ Component.getProps = async (context) => {
return {
page: {
type: PageType.PUBLISHED_MATERIAL,
- props: await PublishedMaterial.getProps(context),
+ props: await PublishedMaterial.getProps(modifiedContext),
},
}
}
}
- // Subpage
- const props = await SubPage.getProps(context)
+ if (organizationPage.theme === STANDALONE_THEME) {
+ return {
+ page: {
+ type: PageType.STANDALONE_PARENT_SUBPAGE,
+ props: await StandaloneParentSubpage.getProps(modifiedContext),
+ },
+ }
+ }
+
return {
page: {
type: PageType.SUBPAGE,
- props,
+ props: await SubPage.getProps(modifiedContext),
},
}
}
@@ -242,7 +260,7 @@ Component.getProps = async (context) => {
return {
page: {
type: PageType.NEWS_DETAILS,
- props: await OrganizationNewsArticle.getProps(context),
+ props: await OrganizationNewsArticle.getProps(modifiedContext),
},
}
}
@@ -250,7 +268,7 @@ Component.getProps = async (context) => {
return {
page: {
type: PageType.EVENT_DETAILS,
- props: await OrganizationEventArticle.getProps(context),
+ props: await OrganizationEventArticle.getProps(modifiedContext),
},
}
}
@@ -259,7 +277,7 @@ Component.getProps = async (context) => {
return {
page: {
type: PageType.NEWS_DETAILS,
- props: await OrganizationNewsArticle.getProps(context),
+ props: await OrganizationNewsArticle.getProps(modifiedContext),
},
}
}
@@ -267,16 +285,27 @@ Component.getProps = async (context) => {
return {
page: {
type: PageType.EVENT_DETAILS,
- props: await OrganizationEventArticle.getProps(context),
+ props: await OrganizationEventArticle.getProps(modifiedContext),
},
}
}
}
+ if (organizationPage.theme === STANDALONE_THEME) {
+ return {
+ page: {
+ type: PageType.STANDALONE_PARENT_SUBPAGE,
+ props: await StandaloneParentSubpage.getProps(modifiedContext),
+ },
+ }
+ }
+
return {
page: {
type: PageType.GENERIC_LIST_ITEM,
- props: await OrganizationSubPageGenericListItem.getProps(context),
+ props: await OrganizationSubPageGenericListItem.getProps(
+ modifiedContext,
+ ),
},
}
}
diff --git a/apps/web/screens/GenericList/OrganizationSubPageGenericListItem.tsx b/apps/web/screens/GenericList/OrganizationSubPageGenericListItem.tsx
index 7bba1474eef6..062fca30e94b 100644
--- a/apps/web/screens/GenericList/OrganizationSubPageGenericListItem.tsx
+++ b/apps/web/screens/GenericList/OrganizationSubPageGenericListItem.tsx
@@ -1,10 +1,11 @@
import { useMemo } from 'react'
import { useRouter } from 'next/router'
+import { Query } from '@island.is/web/graphql/schema'
import { useLinkResolver } from '@island.is/web/hooks'
import { useI18n } from '@island.is/web/i18n'
-import { type LayoutProps, withMainLayout } from '@island.is/web/layouts/main'
-import { Screen } from '@island.is/web/types'
+import { type LayoutProps } from '@island.is/web/layouts/main'
+import { Screen, ScreenContext } from '@island.is/web/types'
import SubPage, { type SubPageProps } from '../Organization/SubPage'
import GenericListItemPage, {
@@ -19,8 +20,13 @@ export interface OrganizationSubPageGenericListItemProps {
genericListItemProps: GenericListItemPageProps
}
+type OrganizationSubPageGenericListItemScreenContext = ScreenContext & {
+ organizationPage?: Query['getOrganizationPage']
+}
+
const OrganizationSubPageGenericListItem: Screen<
- OrganizationSubPageGenericListItemProps
+ OrganizationSubPageGenericListItemProps,
+ OrganizationSubPageGenericListItemScreenContext
> = (props) => {
const { organizationPage, subpage } = props.parentProps.componentProps
const router = useRouter()
diff --git a/apps/web/screens/Organization/OrganizationEvents/OrganizationEventArticle.tsx b/apps/web/screens/Organization/OrganizationEvents/OrganizationEventArticle.tsx
index 760d40330a9e..c8295301cdfa 100644
--- a/apps/web/screens/Organization/OrganizationEvents/OrganizationEventArticle.tsx
+++ b/apps/web/screens/Organization/OrganizationEvents/OrganizationEventArticle.tsx
@@ -41,7 +41,7 @@ import { useWindowSize } from '@island.is/web/hooks/useViewport'
import { useI18n } from '@island.is/web/i18n'
import { useDateUtils } from '@island.is/web/i18n/useDateUtils'
import { withMainLayout } from '@island.is/web/layouts/main'
-import type { Screen } from '@island.is/web/types'
+import type { Screen, ScreenContext } from '@island.is/web/types'
import { CustomNextError } from '@island.is/web/units/errors'
import { extractNamespaceFromOrganization } from '@island.is/web/utils/extractNamespaceFromOrganization'
import { getOrganizationSidebarNavigationItems } from '@island.is/web/utils/organization'
@@ -141,12 +141,14 @@ export interface OrganizationEventArticleProps {
locale: Locale
}
-const OrganizationEventArticle: Screen = ({
- organizationPage,
- event,
- namespace,
- locale,
-}) => {
+type OrganizationEventArticleScreenContext = ScreenContext & {
+ organizationPage?: Query['getOrganizationPage']
+}
+
+const OrganizationEventArticle: Screen<
+ OrganizationEventArticleProps,
+ OrganizationEventArticleScreenContext
+> = ({ organizationPage, event, namespace, locale }) => {
const n = useNamespace(namespace)
const router = useRouter()
@@ -285,19 +287,26 @@ const OrganizationEventArticle: Screen = ({
>
)
}
-OrganizationEventArticle.getProps = async ({ apolloClient, query, locale }) => {
+OrganizationEventArticle.getProps = async ({
+ apolloClient,
+ query,
+ locale,
+ organizationPage: _organizationPage,
+}) => {
const [organizationPageSlug, _, eventSlug] = query.slugs as string[]
const [organizationPageResponse, eventResponse, namespace] =
await Promise.all([
- apolloClient.query({
- query: GET_ORGANIZATION_PAGE_QUERY,
- variables: {
- input: {
- slug: organizationPageSlug,
- lang: locale as Locale,
- },
- },
- }),
+ !_organizationPage
+ ? apolloClient.query({
+ query: GET_ORGANIZATION_PAGE_QUERY,
+ variables: {
+ input: {
+ slug: organizationPageSlug,
+ lang: locale as Locale,
+ },
+ },
+ })
+ : { data: { getOrganizationPage: _organizationPage } },
apolloClient.query({
query: GET_SINGLE_EVENT_QUERY,
variables: {
diff --git a/apps/web/screens/Organization/OrganizationNews/OrganizationNewsArticle.tsx b/apps/web/screens/Organization/OrganizationNews/OrganizationNewsArticle.tsx
index d5da623eb539..24bf3f060b60 100644
--- a/apps/web/screens/Organization/OrganizationNews/OrganizationNewsArticle.tsx
+++ b/apps/web/screens/Organization/OrganizationNews/OrganizationNewsArticle.tsx
@@ -28,7 +28,7 @@ import {
GET_ORGANIZATION_PAGE_QUERY,
GET_SINGLE_NEWS_ITEM_QUERY,
} from '@island.is/web/screens/queries'
-import { Screen } from '@island.is/web/types'
+import type { Screen, ScreenContext } from '@island.is/web/types'
import { CustomNextError } from '@island.is/web/units/errors'
import { extractNamespaceFromOrganization } from '@island.is/web/utils/extractNamespaceFromOrganization'
@@ -39,12 +39,14 @@ export interface OrganizationNewsArticleProps {
locale: Locale
}
-const OrganizationNewsArticle: Screen = ({
- newsItem,
- namespace,
- organizationPage,
- locale,
-}) => {
+type OrganizationNewsArticleScreenContext = ScreenContext & {
+ organizationPage?: Query['getOrganizationPage']
+}
+
+const OrganizationNewsArticle: Screen<
+ OrganizationNewsArticleProps,
+ OrganizationNewsArticleScreenContext
+> = ({ newsItem, namespace, organizationPage, locale }) => {
const router = useRouter()
const { linkResolver } = useLinkResolver()
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
@@ -163,22 +165,27 @@ const OrganizationNewsArticle: Screen = ({
)
}
-OrganizationNewsArticle.getProps = async ({ apolloClient, locale, query }) => {
+OrganizationNewsArticle.getProps = async ({
+ apolloClient,
+ locale,
+ query,
+ organizationPage: _organizationPage,
+}) => {
const [organizationPageSlug, _, newsSlug] = query.slugs as string[]
- const organizationPage = (
- await Promise.resolve(
- apolloClient.query({
- query: GET_ORGANIZATION_PAGE_QUERY,
- variables: {
- input: {
- slug: organizationPageSlug,
- lang: locale as Locale,
+ const organizationPage = !_organizationPage
+ ? (
+ await apolloClient.query({
+ query: GET_ORGANIZATION_PAGE_QUERY,
+ variables: {
+ input: {
+ slug: organizationPageSlug,
+ lang: locale as Locale,
+ },
},
- },
- }),
- )
- ).data?.getOrganizationPage
+ })
+ ).data?.getOrganizationPage
+ : _organizationPage
if (!organizationPage) {
throw new CustomNextError(
diff --git a/apps/web/screens/Organization/Standalone/ParentSubpage.tsx b/apps/web/screens/Organization/Standalone/ParentSubpage.tsx
new file mode 100644
index 000000000000..ad4e08f629c9
--- /dev/null
+++ b/apps/web/screens/Organization/Standalone/ParentSubpage.tsx
@@ -0,0 +1,251 @@
+import { useRouter } from 'next/router'
+
+import {
+ Breadcrumbs,
+ GridColumn,
+ GridContainer,
+ GridRow,
+ Stack,
+ TableOfContents,
+ Text,
+} from '@island.is/island-ui/core'
+import {
+ ContentLanguage,
+ OrganizationPage,
+ OrganizationParentSubpage,
+ OrganizationSubpage,
+ Query,
+ QueryGetNamespaceArgs,
+ QueryGetOrganizationPageArgs,
+ QueryGetOrganizationParentSubpageArgs,
+ QueryGetOrganizationSubpageArgs,
+} from '@island.is/web/graphql/schema'
+import { useLinkResolver } from '@island.is/web/hooks'
+import { useI18n } from '@island.is/web/i18n'
+import { StandaloneLayout } from '@island.is/web/layouts/organization/standalone'
+import type { Screen, ScreenContext } from '@island.is/web/types'
+import { CustomNextError } from '@island.is/web/units/errors'
+
+import {
+ GET_NAMESPACE_QUERY,
+ GET_ORGANIZATION_PAGE_QUERY,
+ GET_ORGANIZATION_PARENT_SUBPAGE_QUERY,
+ GET_ORGANIZATION_SUBPAGE_QUERY,
+} from '../../queries'
+import { SubPageContent } from '../SubPage'
+
+type StandaloneParentSubpageScreenContext = ScreenContext & {
+ organizationPage?: Query['getOrganizationPage']
+}
+
+export interface StandaloneParentSubpageProps {
+ organizationPage: OrganizationPage
+ subpage: OrganizationSubpage
+ tableOfContentHeadings: {
+ headingId: string
+ headingTitle: string
+ label: string
+ href: string
+ }[]
+ selectedHeadingId: string
+ parentSubpage: OrganizationParentSubpage
+ namespace: Record
+}
+
+const StandaloneParentSubpage: Screen<
+ StandaloneParentSubpageProps,
+ StandaloneParentSubpageScreenContext
+> = ({
+ organizationPage,
+ parentSubpage,
+ selectedHeadingId,
+ subpage,
+ tableOfContentHeadings,
+ namespace,
+}) => {
+ const router = useRouter()
+ const { activeLocale } = useI18n()
+ const { linkResolver } = useLinkResolver()
+
+ return (
+
+
+
+
+
+
+ {parentSubpage.childLinks.length > 1 && (
+
+
+ {parentSubpage.title}
+
+ {
+ const href = tableOfContentHeadings.find(
+ (heading) => heading.headingId === headingId,
+ )?.href
+ if (href) {
+ router.push(href)
+ }
+ }}
+ tableOfContentsTitle={
+ namespace?.['standaloneTableOfContentsTitle'] ??
+ activeLocale === 'is'
+ ? 'Efnisyfirlit'
+ : 'Table of contents'
+ }
+ selectedHeadingId={selectedHeadingId}
+ />
+
+ )}
+
+
+
+ 1 ? 'h2' : 'h1'
+ }
+ />
+
+
+ )
+}
+
+StandaloneParentSubpage.getProps = async ({
+ apolloClient,
+ locale,
+ query,
+ organizationPage,
+}) => {
+ const [organizationPageSlug, parentSubpageSlug, subpageSlug] = (query.slugs ??
+ []) as string[]
+
+ const [
+ {
+ data: { getOrganizationPage },
+ },
+ {
+ data: { getOrganizationParentSubpage },
+ },
+ namespace,
+ ] = await Promise.all([
+ !organizationPage
+ ? apolloClient.query({
+ query: GET_ORGANIZATION_PAGE_QUERY,
+ variables: {
+ input: {
+ slug: organizationPageSlug,
+ lang: locale as ContentLanguage,
+ },
+ },
+ })
+ : {
+ data: { getOrganizationPage: organizationPage },
+ },
+ apolloClient.query({
+ query: GET_ORGANIZATION_PARENT_SUBPAGE_QUERY,
+ variables: {
+ input: {
+ organizationPageSlug: organizationPageSlug as string,
+ slug: parentSubpageSlug as string,
+ lang: locale as ContentLanguage,
+ },
+ },
+ }),
+ apolloClient
+ .query({
+ query: GET_NAMESPACE_QUERY,
+ variables: {
+ input: {
+ namespace: 'OrganizationPages',
+ lang: locale,
+ },
+ },
+ })
+ .then((variables) =>
+ variables.data.getNamespace?.fields
+ ? JSON.parse(variables.data.getNamespace.fields)
+ : {},
+ ),
+ ])
+
+ if (!getOrganizationPage) {
+ throw new CustomNextError(404, 'Organization page not found')
+ }
+
+ if (!getOrganizationParentSubpage) {
+ throw new CustomNextError(404, 'Organization parent subpage was not found')
+ }
+
+ let selectedIndex = 0
+
+ if (subpageSlug) {
+ const index = getOrganizationParentSubpage.childLinks.findIndex(
+ (link) => link.href.split('/').pop() === subpageSlug,
+ )
+ if (index >= 0) {
+ selectedIndex = index
+ } else {
+ throw new CustomNextError(
+ 404,
+ 'Subpage belonging to an organization parent subpage was not found',
+ )
+ }
+ }
+
+ const subpage = (
+ await apolloClient.query({
+ query: GET_ORGANIZATION_SUBPAGE_QUERY,
+ variables: {
+ input: {
+ organizationSlug: organizationPageSlug,
+ slug: getOrganizationParentSubpage.childLinks[selectedIndex].href
+ .split('/')
+ .pop() as string,
+ lang: locale as ContentLanguage,
+ },
+ },
+ })
+ ).data.getOrganizationSubpage
+
+ if (!subpage) {
+ throw new CustomNextError(
+ 404,
+ 'Subpage belonging to an organization parent subpage was not found',
+ )
+ }
+
+ const tableOfContentHeadings = getOrganizationParentSubpage.childLinks.map(
+ (link) => ({
+ headingId: link.href,
+ headingTitle: link.label,
+ label: link.label,
+ href: link.href,
+ }),
+ )
+ const selectedHeadingId = tableOfContentHeadings[selectedIndex].headingId
+
+ return {
+ organizationPage: getOrganizationPage,
+ parentSubpage: getOrganizationParentSubpage,
+ subpage,
+ tableOfContentHeadings,
+ selectedHeadingId,
+ namespace,
+ }
+}
+
+export default StandaloneParentSubpage
diff --git a/apps/web/screens/Organization/SubPage.tsx b/apps/web/screens/Organization/SubPage.tsx
index 6d1dc7697cb3..6d81780073b2 100644
--- a/apps/web/screens/Organization/SubPage.tsx
+++ b/apps/web/screens/Organization/SubPage.tsx
@@ -43,7 +43,7 @@ import { extractNamespaceFromOrganization } from '@island.is/web/utils/extractNa
import { webRichText } from '@island.is/web/utils/richText'
import { safelyExtractPathnameFromUrl } from '@island.is/web/utils/safelyExtractPathnameFromUrl'
-import { Screen } from '../../types'
+import { Screen, ScreenContext } from '../../types'
import {
GET_NAMESPACE_QUERY,
GET_ORGANIZATION_PAGE_QUERY,
@@ -61,11 +61,14 @@ export interface SubPageProps {
customContentfulIds?: (string | undefined)[]
}
-const SubPageContent = ({
+export const SubPageContent = ({
subpage,
namespace,
organizationPage,
-}: Pick) => {
+ subpageTitleVariant = 'h1',
+}: Pick & {
+ subpageTitleVariant?: 'h1' | 'h2'
+}) => {
const n = useNamespace(namespace)
const { activeLocale } = useI18n()
const content = (
@@ -132,7 +135,10 @@ const SubPageContent = ({
>
<>
-
+
{subpage?.title}
@@ -208,7 +214,11 @@ const SubPageContent = ({
)
}
-const SubPage: Screen = ({
+type SubPageScreenContext = ScreenContext & {
+ organizationPage?: Query['getOrganizationPage']
+}
+
+const SubPage: Screen = ({
organizationPage,
subpage,
namespace,
@@ -357,7 +367,13 @@ const renderSlices = (
}
}
-SubPage.getProps = async ({ apolloClient, locale, query, req }) => {
+SubPage.getProps = async ({
+ apolloClient,
+ locale,
+ query,
+ req,
+ organizationPage,
+}) => {
const pathname = safelyExtractPathnameFromUrl(req.url)
const { slug, subSlug } = getSlugAndSubSlug(query, pathname)
@@ -370,15 +386,19 @@ SubPage.getProps = async ({ apolloClient, locale, query, req }) => {
},
namespace,
] = await Promise.all([
- apolloClient.query({
- query: GET_ORGANIZATION_PAGE_QUERY,
- variables: {
- input: {
- slug: slug as string,
- lang: locale as ContentLanguage,
+ !organizationPage
+ ? apolloClient.query({
+ query: GET_ORGANIZATION_PAGE_QUERY,
+ variables: {
+ input: {
+ slug: slug as string,
+ lang: locale as ContentLanguage,
+ },
+ },
+ })
+ : {
+ data: { getOrganizationPage: organizationPage },
},
- },
- }),
apolloClient.query({
query: GET_ORGANIZATION_SUBPAGE_QUERY,
variables: {
diff --git a/apps/web/screens/queries/Organization.tsx b/apps/web/screens/queries/Organization.tsx
index f5a6cd3c5e92..fea842378a77 100644
--- a/apps/web/screens/queries/Organization.tsx
+++ b/apps/web/screens/queries/Organization.tsx
@@ -416,3 +416,17 @@ export const EMAIL_SIGNUP_MUTATION = gql`
}
}
`
+
+export const GET_ORGANIZATION_PARENT_SUBPAGE_QUERY = gql`
+ query GetOrganizationParentSubpageQuery(
+ $input: GetOrganizationParentSubpageInput!
+ ) {
+ getOrganizationParentSubpage(input: $input) {
+ title
+ childLinks {
+ label
+ href
+ }
+ }
+ }
+`
diff --git a/libs/cms/src/lib/cms.contentful.service.ts b/libs/cms/src/lib/cms.contentful.service.ts
index f54e8a4794c9..4488f25f5201 100644
--- a/libs/cms/src/lib/cms.contentful.service.ts
+++ b/libs/cms/src/lib/cms.contentful.service.ts
@@ -83,10 +83,11 @@ import { LifeEventPage, mapLifeEventPage } from './models/lifeEventPage.model'
import { GetGenericTagBySlugInput } from './dto/getGenericTagBySlug.input'
import { GetGenericTagsInTagGroupsInput } from './dto/getGenericTagsInTagGroups.input'
import { Grant, mapGrant } from './models/grant.model'
-import { GrantList } from './models/grantList.model'
import { mapManual } from './models/manual.model'
import { mapServiceWebPage } from './models/serviceWebPage.model'
import { mapEvent } from './models/event.model'
+import { GetOrganizationParentSubpageInput } from './dto/getOrganizationParentSubpage.input'
+import { mapOrganizationParentSubpage } from './models/organizationParentSubpage.model'
const errorHandler = (name: string) => {
return (error: Error) => {
@@ -1151,4 +1152,27 @@ export class CmsContentfulService {
return (result.items as types.IGenericTag[]).map(mapGenericTag)
}
+
+ async getOrganizationParentSubpage(input: GetOrganizationParentSubpageInput) {
+ const params = {
+ content_type: 'organizationParentSubpage',
+ 'fields.slug': input.slug,
+ 'fields.organizationPage.sys.contentType.sys.id': 'organizationPage',
+ 'fields.organizationPage.fields.slug': input.organizationPageSlug,
+ limit: 1,
+ }
+
+ const response = await this.contentfulRepository
+ .getLocalizedEntries(
+ input.lang,
+ params,
+ )
+ .catch(errorHandler('getOrganizationParentSubpage'))
+
+ return (
+ (response.items as types.IOrganizationParentSubpage[]).map(
+ mapOrganizationParentSubpage,
+ )[0] ?? null
+ )
+ }
}
diff --git a/libs/cms/src/lib/cms.resolver.ts b/libs/cms/src/lib/cms.resolver.ts
index 462b5e4531ec..8c82719d6bdc 100644
--- a/libs/cms/src/lib/cms.resolver.ts
+++ b/libs/cms/src/lib/cms.resolver.ts
@@ -126,6 +126,8 @@ import { Grant } from './models/grant.model'
import { GetGrantsInput } from './dto/getGrants.input'
import { GetSingleGrantInput } from './dto/getSingleGrant.input'
import { GrantList } from './models/grantList.model'
+import { OrganizationParentSubpage } from './models/organizationParentSubpage.model'
+import { GetOrganizationParentSubpageInput } from './dto/getOrganizationParentSubpage.input'
const defaultCache: CacheControlOptions = { maxAge: CACHE_CONTROL_MAX_AGE }
@@ -707,6 +709,14 @@ export class CmsResolver {
): Promise {
return this.cmsElasticsearchService.getTeamMembers(input)
}
+
+ @CacheControl(defaultCache)
+ @Query(() => OrganizationParentSubpage, { nullable: true })
+ getOrganizationParentSubpage(
+ @Args('input') input: GetOrganizationParentSubpageInput,
+ ): Promise {
+ return this.cmsContentfulService.getOrganizationParentSubpage(input)
+ }
}
@Resolver(() => LatestNewsSlice)
diff --git a/libs/cms/src/lib/dto/getOrganizationParentSubpage.input.ts b/libs/cms/src/lib/dto/getOrganizationParentSubpage.input.ts
new file mode 100644
index 000000000000..c45e5ef9fd33
--- /dev/null
+++ b/libs/cms/src/lib/dto/getOrganizationParentSubpage.input.ts
@@ -0,0 +1,18 @@
+import { Field, InputType } from '@nestjs/graphql'
+import { IsString } from 'class-validator'
+import { ElasticsearchIndexLocale } from '@island.is/content-search-index-manager'
+
+@InputType()
+export class GetOrganizationParentSubpageInput {
+ @Field()
+ @IsString()
+ slug!: string
+
+ @Field()
+ @IsString()
+ organizationPageSlug!: string
+
+ @Field(() => String)
+ @IsString()
+ lang: ElasticsearchIndexLocale = 'is'
+}
diff --git a/libs/cms/src/lib/models/organizationParentSubpage.model.ts b/libs/cms/src/lib/models/organizationParentSubpage.model.ts
new file mode 100644
index 000000000000..9c9a4d927059
--- /dev/null
+++ b/libs/cms/src/lib/models/organizationParentSubpage.model.ts
@@ -0,0 +1,48 @@
+import { CacheField } from '@island.is/nest/graphql'
+import { Field, ID, ObjectType } from '@nestjs/graphql'
+import { IOrganizationParentSubpage } from '../generated/contentfulTypes'
+import { getOrganizationPageUrlPrefix } from '@island.is/shared/utils'
+
+@ObjectType()
+class OrganizationSubpageLink {
+ @Field()
+ label!: string
+
+ @Field()
+ href!: string
+}
+
+@ObjectType()
+export class OrganizationParentSubpage {
+ @Field(() => ID)
+ id!: string
+
+ @Field()
+ title!: string
+
+ @CacheField(() => [OrganizationSubpageLink])
+ childLinks!: OrganizationSubpageLink[]
+}
+
+export const mapOrganizationParentSubpage = ({
+ sys,
+ fields,
+}: IOrganizationParentSubpage): OrganizationParentSubpage => {
+ return {
+ id: sys.id,
+ title: fields.title,
+ childLinks:
+ fields.pages
+ ?.filter(
+ (page) =>
+ Boolean(page.fields.organizationPage?.fields?.slug) &&
+ Boolean(page.fields.slug),
+ )
+ .map((page) => ({
+ label: page.fields.title,
+ href: `/${getOrganizationPageUrlPrefix(sys.locale)}/${
+ page.fields.organizationPage.fields.slug
+ }/${fields.slug}/${page.fields.slug}`,
+ })) ?? [],
+ }
+}
From 331545656d7a9797fef305175462d62f107c8b64 Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?J=C3=B3nas=20G=2E=20Sigur=C3=B0sson?=
Date: Tue, 26 Nov 2024 15:00:35 +0000
Subject: [PATCH 06/12] feat(app-sys): Shared display field (#17007)
* feat: start of display field
* feat: display field
* fix: revert changes to another template
* feat: simplify useEffect logic and undo example form changes
* chore: undo example form changes
---------
Co-authored-by: kodiakhq[bot] <49736102+kodiakhq[bot]@users.noreply.github.com>
---
.../application/core/src/lib/fieldBuilders.ts | 32 ++++++++
libs/application/types/src/lib/Fields.ts | 18 ++++-
.../lib/DisplayFormField/DisplayFormField.tsx | 80 +++++++++++++++++++
libs/application/ui-fields/src/lib/index.ts | 1 +
4 files changed, 130 insertions(+), 1 deletion(-)
create mode 100644 libs/application/ui-fields/src/lib/DisplayFormField/DisplayFormField.tsx
diff --git a/libs/application/core/src/lib/fieldBuilders.ts b/libs/application/core/src/lib/fieldBuilders.ts
index dabe437861a0..2c77343d5aca 100644
--- a/libs/application/core/src/lib/fieldBuilders.ts
+++ b/libs/application/core/src/lib/fieldBuilders.ts
@@ -46,6 +46,7 @@ import {
SliderField,
MaybeWithApplication,
MaybeWithApplicationAndFieldAndLocale,
+ DisplayField,
FieldsRepeaterField,
} from '@island.is/application/types'
import { Locale } from '@island.is/shared/types'
@@ -1009,3 +1010,34 @@ export const buildSliderField = (
saveAsString,
}
}
+
+export const buildDisplayField = (
+ data: Omit,
+): DisplayField => {
+ const {
+ title,
+ titleVariant,
+ label,
+ variant,
+ marginTop,
+ marginBottom,
+ value,
+ suffix,
+ rightAlign,
+ } = data
+ return {
+ ...extractCommonFields(data),
+ title,
+ titleVariant,
+ label,
+ variant,
+ marginTop,
+ marginBottom,
+ type: FieldTypes.DISPLAY,
+ component: FieldComponents.DISPLAY,
+ children: undefined,
+ value,
+ suffix,
+ rightAlign,
+ }
+}
diff --git a/libs/application/types/src/lib/Fields.ts b/libs/application/types/src/lib/Fields.ts
index c6763dbb9141..e5425ca66464 100644
--- a/libs/application/types/src/lib/Fields.ts
+++ b/libs/application/types/src/lib/Fields.ts
@@ -16,7 +16,7 @@ import {
StaticText,
} from './Form'
import { ApolloClient } from '@apollo/client'
-import { Application } from './Application'
+import { Application, FormValue } from './Application'
import { CallToAction } from './StateMachine'
import { Colors, theme } from '@island.is/island-ui/theme'
import { Condition } from './Condition'
@@ -262,6 +262,7 @@ export enum FieldTypes {
ACCORDION = 'ACCORDION',
BANK_ACCOUNT = 'BANK_ACCOUNT',
SLIDER = 'SLIDER',
+ DISPLAY = 'DISPLAY',
}
export enum FieldComponents {
@@ -298,6 +299,7 @@ export enum FieldComponents {
ACCORDION = 'AccordionFormField',
BANK_ACCOUNT = 'BankAccountFormField',
SLIDER = 'SliderFormField',
+ DISPLAY = 'DisplayFormField',
}
export interface CheckboxField extends InputField {
@@ -796,6 +798,19 @@ export interface SliderField extends BaseField {
saveAsString?: boolean
}
+export interface DisplayField extends BaseField {
+ readonly type: FieldTypes.DISPLAY
+ component: FieldComponents.DISPLAY
+ marginTop?: ResponsiveProp
+ marginBottom?: ResponsiveProp
+ titleVariant?: TitleVariants
+ suffix?: MessageDescriptor | string
+ rightAlign?: boolean
+ variant?: TextFieldVariant
+ label?: MessageDescriptor | string
+ value: (answers: FormValue) => string
+}
+
export type Field =
| CheckboxField
| CustomField
@@ -832,3 +847,4 @@ export type Field =
| AccordionField
| BankAccountField
| SliderField
+ | DisplayField
diff --git a/libs/application/ui-fields/src/lib/DisplayFormField/DisplayFormField.tsx b/libs/application/ui-fields/src/lib/DisplayFormField/DisplayFormField.tsx
new file mode 100644
index 000000000000..ae6faba48fed
--- /dev/null
+++ b/libs/application/ui-fields/src/lib/DisplayFormField/DisplayFormField.tsx
@@ -0,0 +1,80 @@
+import { formatTextWithLocale } from '@island.is/application/core'
+import { DisplayField, FieldBaseProps } from '@island.is/application/types'
+import { Box, Text } from '@island.is/island-ui/core'
+import { useLocale } from '@island.is/localization'
+import { InputController } from '@island.is/shared/form-fields'
+import { useEffect, useState } from 'react'
+import { useFormContext } from 'react-hook-form'
+import { Locale } from '@island.is/shared/types'
+
+interface Props extends FieldBaseProps {
+ field: DisplayField
+}
+
+export const DisplayFormField = ({ field, application }: Props) => {
+ const {
+ value,
+ id,
+ title,
+ titleVariant = 'h4',
+ label,
+ variant,
+ suffix,
+ rightAlign = false,
+ } = field
+ const { watch, setValue } = useFormContext()
+ const allValues = watch()
+ const { formatMessage, lang: locale } = useLocale()
+ const [displayValue, setDisplayValue] = useState(allValues[id])
+
+ useEffect(() => {
+ const newDisplayValue = value(allValues)
+ setDisplayValue(newDisplayValue)
+ setValue(id, newDisplayValue)
+ }, [allValues])
+
+ return (
+
+ {title ? (
+
+ {formatTextWithLocale(
+ title,
+ application,
+ locale as Locale,
+ formatMessage,
+ )}
+
+ ) : null}
+
+
+
+ )
+}
diff --git a/libs/application/ui-fields/src/lib/index.ts b/libs/application/ui-fields/src/lib/index.ts
index 5ca103c7e046..d796e89afbe9 100644
--- a/libs/application/ui-fields/src/lib/index.ts
+++ b/libs/application/ui-fields/src/lib/index.ts
@@ -31,3 +31,4 @@ export { StaticTableFormField } from './StaticTableFormField/StaticTableFormFiel
export { AccordionFormField } from './AccordionFormField/AccordionFormField'
export { BankAccountFormField } from './BankAccountFormField/BankAccountFormField'
export { SliderFormField } from './SliderFormField/SliderFormField'
+export { DisplayFormField } from './DisplayFormField/DisplayFormField'
From 3d9f462018281efe3e0422033e7e505c6b571a72 Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?R=C3=BAnar=20Vestmann?=
<43557895+RunarVestmann@users.noreply.github.com>
Date: Tue, 26 Nov 2024 15:20:46 +0000
Subject: [PATCH 07/12] feat(contentful-apps): Refactor list search
functionality into a reusable component (#17025)
* EntryListSearch component added
* Update query
---------
Co-authored-by: kodiakhq[bot] <49736102+kodiakhq[bot]@users.noreply.github.com>
---
.../components/EntryListSearch.tsx | 155 ++++++++++++++++++
.../GenericListEditor/GenericListEditor.tsx | 133 ++-------------
.../fields/generic-tag-group-items-field.tsx | 130 ++-------------
3 files changed, 179 insertions(+), 239 deletions(-)
create mode 100644 apps/contentful-apps/components/EntryListSearch.tsx
diff --git a/apps/contentful-apps/components/EntryListSearch.tsx b/apps/contentful-apps/components/EntryListSearch.tsx
new file mode 100644
index 000000000000..90efc883c5ab
--- /dev/null
+++ b/apps/contentful-apps/components/EntryListSearch.tsx
@@ -0,0 +1,155 @@
+import { useRef, useState } from 'react'
+import { useDebounce } from 'react-use'
+import {
+ CollectionProp,
+ EntryProps,
+ KeyValueMap,
+ QueryOptions,
+} from 'contentful-management'
+import {
+ Box,
+ EntryCard,
+ Pagination,
+ Spinner,
+ Stack,
+ Text,
+ TextInput,
+} from '@contentful/f36-components'
+import { useCMA, useSDK } from '@contentful/react-apps-toolkit'
+
+import { DEFAULT_LOCALE } from '../constants'
+
+const SEARCH_DEBOUNCE_TIME_IN_MS = 300
+
+interface EntryListSearchProps {
+ contentTypeId: string
+ contentTypeLabel: string
+ contentTypeTitleField: string
+ itemsPerPage?: number
+ onEntryClick?: (entry: EntryProps) => void
+ query?: QueryOptions
+}
+
+export const EntryListSearch = ({
+ itemsPerPage = 4,
+ contentTypeId,
+ contentTypeLabel,
+ contentTypeTitleField,
+ onEntryClick,
+ query,
+}: EntryListSearchProps) => {
+ const sdk = useSDK()
+ const cma = useCMA()
+
+ const searchValueRef = useRef('')
+ const [searchValue, setSearchValue] = useState('')
+ const [listItemResponse, setListItemResponse] =
+ useState>>()
+ const [loading, setLoading] = useState(false)
+ const [page, setPage] = useState(0)
+ const pageRef = useRef(0)
+ const [counter, setCounter] = useState(0)
+
+ const skip = itemsPerPage * page
+
+ useDebounce(
+ async () => {
+ setLoading(true)
+ try {
+ const response = await cma.entry.getMany({
+ query: {
+ content_type: contentTypeId,
+ limit: itemsPerPage,
+ skip,
+ [`fields.${contentTypeTitleField}[match]`]: searchValue,
+ 'sys.archivedAt[exists]': false,
+ ...query,
+ },
+ })
+ if (
+ searchValueRef.current === searchValue &&
+ pageRef.current === page
+ ) {
+ setListItemResponse(response)
+ }
+ } finally {
+ setLoading(false)
+ }
+ },
+ SEARCH_DEBOUNCE_TIME_IN_MS,
+ [page, searchValue, counter],
+ )
+
+ return (
+
+ {
+ searchValueRef.current = ev.target.value
+ setSearchValue(ev.target.value)
+ setPage(0)
+ pageRef.current = 0
+ }}
+ />
+
+
+
+
+
+ {listItemResponse?.items?.length > 0 && (
+ <>
+
+
+ {listItemResponse.items.map((item) => (
+ {
+ if (onEntryClick) {
+ onEntryClick(item)
+ return
+ }
+
+ sdk.navigator
+ .openEntry(item.sys.id, {
+ slideIn: { waitForClose: true },
+ })
+ .then(() => {
+ setCounter((c) => c + 1)
+ })
+ }}
+ />
+ ))}
+
+
+ {
+ pageRef.current = newPage
+ setPage(newPage)
+ }}
+ />
+ >
+ )}
+
+ {listItemResponse?.items?.length === 0 && (
+
+ No entry was found
+
+ )}
+
+ )
+}
diff --git a/apps/contentful-apps/components/editors/lists/GenericListEditor/GenericListEditor.tsx b/apps/contentful-apps/components/editors/lists/GenericListEditor/GenericListEditor.tsx
index 0740aff1d59d..2eac9ec947d2 100644
--- a/apps/contentful-apps/components/editors/lists/GenericListEditor/GenericListEditor.tsx
+++ b/apps/contentful-apps/components/editors/lists/GenericListEditor/GenericListEditor.tsx
@@ -1,27 +1,14 @@
-import { useMemo, useRef, useState } from 'react'
-import { useDebounce } from 'react-use'
-import { CollectionProp, EntryProps, KeyValueMap } from 'contentful-management'
+import { useMemo, useState } from 'react'
import dynamic from 'next/dynamic'
import { EditorExtensionSDK } from '@contentful/app-sdk'
-import {
- Box,
- Button,
- EntryCard,
- FormControl,
- Pagination,
- Spinner,
- Stack,
- Text,
- TextInput,
-} from '@contentful/f36-components'
+import { Box, Button, FormControl } from '@contentful/f36-components'
import { PlusIcon } from '@contentful/f36-icons'
import { useCMA, useSDK } from '@contentful/react-apps-toolkit'
+import { EntryListSearch } from '../../../../components/EntryListSearch'
import { mapLocalesToFieldApis } from '../../utils'
-const SEARCH_DEBOUNCE_TIME_IN_MS = 300
const LIST_ITEM_CONTENT_TYPE_ID = 'genericListItem'
-const LIST_ITEMS_PER_PAGE = 4
const createLocaleToFieldMapping = (sdk: EditorExtensionSDK) => {
return {
@@ -58,51 +45,11 @@ export const GenericListEditor = () => {
const sdk = useSDK()
const cma = useCMA()
- const searchValueRef = useRef('')
- const [searchValue, setSearchValue] = useState('')
- const [listItemResponse, setListItemResponse] =
- useState>>()
- const [loading, setLoading] = useState(false)
- const [page, setPage] = useState(0)
- const pageRef = useRef(0)
-
/** Counter that's simply used to refresh the list when an item gets created */
- const [counter, setCounter] = useState(0)
-
- const skip = LIST_ITEMS_PER_PAGE * page
+ const [_, setCounter] = useState(0)
const defaultLocale = sdk.locales.default
- useDebounce(
- async () => {
- setLoading(true)
- try {
- const response = await cma.entry.getMany({
- environmentId: sdk.ids.environment,
- spaceId: sdk.ids.space,
- query: {
- content_type: LIST_ITEM_CONTENT_TYPE_ID,
- limit: LIST_ITEMS_PER_PAGE,
- skip,
- 'fields.internalTitle[match]': searchValue,
- 'fields.genericList.sys.id': sdk.entry.getSys().id,
- 'sys.archivedAt[exists]': false,
- },
- })
- if (
- searchValueRef.current === searchValue &&
- pageRef.current === page
- ) {
- setListItemResponse(response)
- }
- } finally {
- setLoading(false)
- }
- },
- SEARCH_DEBOUNCE_TIME_IN_MS,
- [page, searchValue, counter],
- )
-
const createListItem = async () => {
const cardIntro = {}
@@ -189,70 +136,14 @@ export const GenericListEditor = () => {
}>Add item
-
- {
- searchValueRef.current = ev.target.value
- setSearchValue(ev.target.value)
- setPage(0)
- pageRef.current = 0
- }}
- />
-
-
-
-
-
- {listItemResponse?.items?.length > 0 && (
- <>
-
-
- {listItemResponse.items.map((item) => (
- {
- sdk.navigator
- .openEntry(item.sys.id, {
- slideIn: { waitForClose: true },
- })
- .then(() => {
- setCounter((c) => c + 1)
- })
- }}
- />
- ))}
-
-
- {
- pageRef.current = newPage
- setPage(newPage)
- }}
- />
- >
- )}
-
- {listItemResponse?.items?.length === 0 && (
-
- No item was found
-
- )}
-
+
)
}
diff --git a/apps/contentful-apps/pages/fields/generic-tag-group-items-field.tsx b/apps/contentful-apps/pages/fields/generic-tag-group-items-field.tsx
index 6a577e389b9d..bdd4a8f9a83e 100644
--- a/apps/contentful-apps/pages/fields/generic-tag-group-items-field.tsx
+++ b/apps/contentful-apps/pages/fields/generic-tag-group-items-field.tsx
@@ -1,37 +1,17 @@
-import { useEffect, useRef, useState } from 'react'
-import { useDebounce } from 'react-use'
+import { useEffect, useState } from 'react'
import type { FieldExtensionSDK } from '@contentful/app-sdk'
-import {
- Box,
- Button,
- EntryCard,
- Pagination,
- Spinner,
- Stack,
- Text,
- TextInput,
-} from '@contentful/f36-components'
+import { Box, Button } from '@contentful/f36-components'
import { PlusIcon } from '@contentful/f36-icons'
import { useCMA, useSDK } from '@contentful/react-apps-toolkit'
-const LIST_ITEMS_PER_PAGE = 4
-const SEARCH_DEBOUNCE_TIME_IN_MS = 300
+import { EntryListSearch } from '../../components/EntryListSearch'
const GenericTagGroupItemsField = () => {
- const [page, setPage] = useState(0)
- const pageRef = useRef(0)
- const [searchValue, setSearchValue] = useState('')
- const searchValueRef = useRef('')
- const [listItemResponse, setListItemResponse] = useState(null)
- const [isLoading, setIsLoading] = useState(false)
-
- const [counter, setCounter] = useState(0)
+ const [_, setCounter] = useState(0)
const sdk = useSDK()
const cma = useCMA()
- const skip = LIST_ITEMS_PER_PAGE * page
-
const createGenericTag = async () => {
const tag = await cma.entry.create(
{
@@ -62,35 +42,6 @@ const GenericTagGroupItemsField = () => {
})
}
- useDebounce(
- async () => {
- setIsLoading(true)
- try {
- const response = await cma.entry.getMany({
- query: {
- content_type: 'genericTag',
- limit: LIST_ITEMS_PER_PAGE,
- skip,
- 'fields.internalTitle[match]': searchValue,
- 'fields.genericTagGroup.sys.id': sdk.entry.getSys().id,
- 'sys.archivedAt[exists]': false,
- },
- })
-
- if (
- searchValueRef.current === searchValue &&
- pageRef.current === page
- ) {
- setListItemResponse(response)
- }
- } finally {
- setIsLoading(false)
- }
- },
- SEARCH_DEBOUNCE_TIME_IN_MS,
- [page, searchValue, counter],
- )
-
useEffect(() => {
sdk.window.startAutoResizer()
return () => {
@@ -108,71 +59,14 @@ const GenericTagGroupItemsField = () => {
}>Create tag
-
- {
- searchValueRef.current = ev.target.value
- setSearchValue(ev.target.value)
- setPage(0)
- pageRef.current = 0
- }}
- />
-
-
-
-
-
- {listItemResponse?.items?.length > 0 && (
- <>
-
-
- {listItemResponse.items.map((item) => (
- {
- sdk.navigator
- .openEntry(item.sys.id, {
- slideIn: { waitForClose: true },
- })
- .then(() => {
- setCounter((c) => c + 1)
- })
- }}
- />
- ))}
-
-
- {
- pageRef.current = newPage
- setPage(newPage)
- }}
- />
- >
- )}
-
- {listItemResponse?.items?.length === 0 && (
-
- No item was found
-
- )}
-
+
)
}
From ca7fe1a623a233b48316e7ae88062bef231a5626 Mon Sep 17 00:00:00 2001
From: valurefugl <65780958+valurefugl@users.noreply.github.com>
Date: Tue, 26 Nov 2024 16:30:27 +0000
Subject: [PATCH 08/12] feat(ids-api): Add alive check to legal representative
delegation type (#16284)
* Refactor alive status check into separate service. Add check to legal repr.
* Fix typo.
* Handle error when deleting from index.
* Refactor error handling when deleting from index.
* Enable using national registry v3 to check if deceased.
* Remove import.
* Make user optional.
* Add v3 config to other apis.
* chore: charts update dirty files
* chore: nx format:write update dirty files
* Fix feature flag.
* Preserve name fallback. Fix tests in public-api.
* Handle undefined input.
* Use Set instead of uniq.
* Single pass when merging.
---------
Co-authored-by: andes-it
Co-authored-by: kodiakhq[bot] <49736102+kodiakhq[bot]@users.noreply.github.com>
---
.../auth/admin-api/infra/auth-admin-api.ts | 11 +-
.../auth/admin-api/src/app/app.module.ts | 4 +-
.../auth/delegation-api/src/app/app.module.ts | 2 +
.../me-delegations.access-incoming.spec.ts | 364 ++--
...personal-representative.controller.spec.ts | 1153 +++++------
.../test/delegations-filters.spec.ts | 314 +--
.../test/delegations.controller.spec.ts | 465 ++---
.../infra/personal-representative.ts | 11 +-
.../src/app/app.module.ts | 6 +-
.../auth/public-api/infra/auth-public-api.ts | 11 +-
.../auth/public-api/src/app/app.module.ts | 4 +-
.../actorDelegations.controller.spec.ts | 1719 +++++++++--------
charts/identity-server/values.dev.yaml | 15 +
charts/identity-server/values.prod.yaml | 15 +
charts/identity-server/values.staging.yaml | 15 +
.../services-auth-admin-api/values.dev.yaml | 5 +
.../services-auth-admin-api/values.prod.yaml | 5 +
.../values.staging.yaml | 5 +
.../values.dev.yaml | 5 +
.../values.prod.yaml | 5 +
.../values.staging.yaml | 5 +
.../services-auth-public-api/values.dev.yaml | 5 +
.../services-auth-public-api/values.prod.yaml | 5 +
.../values.staging.yaml | 5 +
libs/auth-api-lib/src/index.ts | 1 +
.../lib/delegations/alive-status.service.ts | 192 ++
.../delegations-incoming-custom.service.ts | 205 +-
...gations-incoming-representative.service.ts | 86 +-
.../delegations-incoming.service.ts | 136 +-
.../src/lib/delegations/delegations.module.ts | 8 +-
.../national-registry-v3-feature.service.ts | 17 +
libs/feature-flags/src/lib/features.ts | 3 +
32 files changed, 2670 insertions(+), 2132 deletions(-)
create mode 100644 libs/auth-api-lib/src/lib/delegations/alive-status.service.ts
create mode 100644 libs/auth-api-lib/src/lib/delegations/national-registry-v3-feature.service.ts
diff --git a/apps/services/auth/admin-api/infra/auth-admin-api.ts b/apps/services/auth/admin-api/infra/auth-admin-api.ts
index 0fc25a62d5d4..cf773f0dbd94 100644
--- a/apps/services/auth/admin-api/infra/auth-admin-api.ts
+++ b/apps/services/auth/admin-api/infra/auth-admin-api.ts
@@ -4,7 +4,12 @@ import {
service,
ServiceBuilder,
} from '../../../../../infra/src/dsl/dsl'
-import { Base, Client, RskProcuring } from '../../../../../infra/src/dsl/xroad'
+import {
+ Base,
+ Client,
+ NationalRegistryAuthB2C,
+ RskProcuring,
+} from '../../../../../infra/src/dsl/xroad'
const REDIS_NODE_CONFIG = {
dev: json([
@@ -84,8 +89,10 @@ export const serviceSetup = (): ServiceBuilder<'services-auth-admin-api'> => {
'/k8s/xroad/client/NATIONAL-REGISTRY/IDENTITYSERVER_SECRET',
SYSLUMENN_USERNAME: '/k8s/services-auth/SYSLUMENN_USERNAME',
SYSLUMENN_PASSWORD: '/k8s/services-auth/SYSLUMENN_PASSWORD',
+ NATIONAL_REGISTRY_B2C_CLIENT_SECRET:
+ '/k8s/services-auth/NATIONAL_REGISTRY_B2C_CLIENT_SECRET',
})
- .xroad(Base, Client, RskProcuring)
+ .xroad(Base, Client, RskProcuring, NationalRegistryAuthB2C)
.ingress({
primary: {
host: {
diff --git a/apps/services/auth/admin-api/src/app/app.module.ts b/apps/services/auth/admin-api/src/app/app.module.ts
index dc56c6fe0343..5f947672c654 100644
--- a/apps/services/auth/admin-api/src/app/app.module.ts
+++ b/apps/services/auth/admin-api/src/app/app.module.ts
@@ -9,8 +9,10 @@ import {
import { AuthModule } from '@island.is/auth-nest-tools'
import { RskRelationshipsClientConfig } from '@island.is/clients-rsk-relationships'
import { NationalRegistryClientConfig } from '@island.is/clients/national-registry-v2'
+import { NationalRegistryV3ClientConfig } from '@island.is/clients/national-registry-v3'
import { CompanyRegistryConfig } from '@island.is/clients/rsk/company-registry'
import { SyslumennClientConfig } from '@island.is/clients/syslumenn'
+import { ZendeskServiceConfig } from '@island.is/clients/zendesk'
import { AuditModule } from '@island.is/nest/audit'
import { IdsClientConfig, XRoadConfig } from '@island.is/nest/config'
import { FeatureFlagConfig } from '@island.is/nest/feature-flags'
@@ -31,7 +33,6 @@ import { ProvidersModule } from './v2/providers/providers.module'
import { ScopesModule } from './v2/scopes/scopes.module'
import { ClientSecretsModule } from './v2/secrets/client-secrets.module'
import { TenantsModule } from './v2/tenants/tenants.module'
-import { ZendeskServiceConfig } from '@island.is/clients/zendesk'
@Module({
imports: [
@@ -61,6 +62,7 @@ import { ZendeskServiceConfig } from '@island.is/clients/zendesk'
DelegationConfig,
RskRelationshipsClientConfig,
NationalRegistryClientConfig,
+ NationalRegistryV3ClientConfig,
CompanyRegistryConfig,
FeatureFlagConfig,
XRoadConfig,
diff --git a/apps/services/auth/delegation-api/src/app/app.module.ts b/apps/services/auth/delegation-api/src/app/app.module.ts
index ac10af9d95f1..71b347827378 100644
--- a/apps/services/auth/delegation-api/src/app/app.module.ts
+++ b/apps/services/auth/delegation-api/src/app/app.module.ts
@@ -9,6 +9,7 @@ import {
import { AuthModule } from '@island.is/auth-nest-tools'
import { RskRelationshipsClientConfig } from '@island.is/clients-rsk-relationships'
import { NationalRegistryClientConfig } from '@island.is/clients/national-registry-v2'
+import { NationalRegistryV3ClientConfig } from '@island.is/clients/national-registry-v3'
import { CompanyRegistryConfig } from '@island.is/clients/rsk/company-registry'
import { SyslumennClientConfig } from '@island.is/clients/syslumenn'
import { AuditModule } from '@island.is/nest/audit'
@@ -48,6 +49,7 @@ import { ScopesModule } from './scopes/scopes.module'
FeatureFlagConfig,
IdsClientConfig,
NationalRegistryClientConfig,
+ NationalRegistryV3ClientConfig,
RskRelationshipsClientConfig,
CompanyRegistryConfig,
XRoadConfig,
diff --git a/apps/services/auth/delegation-api/src/app/delegations/test/me-delegations.access-incoming.spec.ts b/apps/services/auth/delegation-api/src/app/delegations/test/me-delegations.access-incoming.spec.ts
index 6f5c526d1188..377a7771410b 100644
--- a/apps/services/auth/delegation-api/src/app/delegations/test/me-delegations.access-incoming.spec.ts
+++ b/apps/services/auth/delegation-api/src/app/delegations/test/me-delegations.access-incoming.spec.ts
@@ -8,9 +8,11 @@ import {
Delegation,
DelegationScope,
DelegationsIndexService,
+ NationalRegistryV3FeatureService,
} from '@island.is/auth-api-lib'
import { RskRelationshipsClient } from '@island.is/clients-rsk-relationships'
import { NationalRegistryClientService } from '@island.is/clients/national-registry-v2'
+import { NationalRegistryV3ClientService } from '@island.is/clients/national-registry-v3'
import { CompanyRegistryClientService } from '@island.is/clients/rsk/company-registry'
import {
expectMatchingDelegations,
@@ -23,189 +25,219 @@ import { accessIncomingTestCases } from '../../../../test/access-incoming-test-c
import { setupWithAuth } from '../../../../test/setup'
import { filterExpectedDelegations } from './utils'
+const fromName: string = faker.name.findName()
+
describe('MeDelegationsController', () => {
- describe.each(Object.keys(accessIncomingTestCases))(
- 'Incoming Access with test case: %s',
- (caseName) => {
- const testCase = accessIncomingTestCases[caseName]
- let app: TestApp
- let server: request.SuperTest
- let factory: FixtureFactory
- let delegations: Delegation[] = []
- const fromName = faker.name.findName()
- let delegationIndexService: DelegationsIndexService
-
- beforeAll(async () => {
- // Arrange
- app = await setupWithAuth({
- user: testCase.user,
- customScopeRules: testCase.customScopeRules,
- })
- server = request(app.getHttpServer())
- delegationIndexService = app.get(DelegationsIndexService)
- const rskRelationshipsClientService = app.get(RskRelationshipsClient)
- const nationalRegistryClientService = app.get(
- NationalRegistryClientService,
- )
- const companyRegistryClientService = app.get(
- CompanyRegistryClientService,
- )
- jest
- .spyOn(nationalRegistryClientService, 'getCustodyChildren')
- .mockImplementation(async () => [])
- jest
- .spyOn(nationalRegistryClientService, 'getIndividual')
- .mockImplementation(async (nationalId: string) =>
- createNationalRegistryUser({
- nationalId,
- name: fromName,
- }),
- )
- jest
- .spyOn(rskRelationshipsClientService, 'getIndividualRelationships')
- .mockImplementation(async () => null)
- jest
- .spyOn(companyRegistryClientService, 'getCompany')
- .mockImplementation(async () => null)
- jest
- .spyOn(delegationIndexService, 'indexDelegations')
- .mockImplementation()
- jest
- .spyOn(delegationIndexService, 'indexCustomDelegations')
- .mockImplementation()
-
- factory = new FixtureFactory(app)
- await Promise.all(
- testCase.domains.map((domain) => factory.createDomain(domain)),
- )
- await Promise.all(
- (testCase.accessTo ?? []).map((scope) =>
- factory.createApiScopeUserAccess({
- nationalId: testCase.user.nationalId,
- scope,
- }),
- ),
- )
- })
-
- beforeEach(async () => {
- if (delegations) {
- await Delegation.destroy({
- where: {
- id: { [Op.in]: delegations.map((delegation) => delegation.id) },
- },
+ describe.each([false, true])(
+ 'national registry v3 featureflag: %s',
+ (featureFlag) => {
+ describe.each(Object.keys(accessIncomingTestCases))(
+ 'Incoming Access with test case: %s',
+ (caseName) => {
+ const testCase = accessIncomingTestCases[caseName]
+ let app: TestApp
+ let server: request.SuperTest
+ let factory: FixtureFactory
+ let delegations: Delegation[] = []
+ let delegationIndexService: DelegationsIndexService
+
+ beforeAll(async () => {
+ // Arrange
+ app = await setupWithAuth({
+ user: testCase.user,
+ customScopeRules: testCase.customScopeRules,
+ })
+ server = request(app.getHttpServer())
+ delegationIndexService = app.get(DelegationsIndexService)
+ const rskRelationshipsClientService = app.get(
+ RskRelationshipsClient,
+ )
+ const nationalRegistryClientService = app.get(
+ NationalRegistryClientService,
+ )
+ const nationalRegistryV3ClientService = app.get(
+ NationalRegistryV3ClientService,
+ )
+ const companyRegistryClientService = app.get(
+ CompanyRegistryClientService,
+ )
+ const nationalRegistryV3FeatureService = app.get(
+ NationalRegistryV3FeatureService,
+ )
+ jest
+ .spyOn(nationalRegistryClientService, 'getCustodyChildren')
+ .mockImplementation(async () => [])
+ jest
+ .spyOn(nationalRegistryClientService, 'getIndividual')
+ .mockImplementation(async (nationalId: string) =>
+ createNationalRegistryUser({
+ nationalId,
+ name: fromName,
+ }),
+ )
+ jest
+ .spyOn(nationalRegistryV3ClientService, 'getAllDataIndividual')
+ .mockImplementation(async (nationalId: string) => {
+ return { kennitala: nationalId, nafn: fromName }
+ })
+ jest
+ .spyOn(
+ rskRelationshipsClientService,
+ 'getIndividualRelationships',
+ )
+ .mockImplementation(async () => null)
+ jest
+ .spyOn(companyRegistryClientService, 'getCompany')
+ .mockImplementation(async () => null)
+ jest
+ .spyOn(delegationIndexService, 'indexDelegations')
+ .mockImplementation()
+ jest
+ .spyOn(delegationIndexService, 'indexCustomDelegations')
+ .mockImplementation()
+ jest
+ .spyOn(nationalRegistryV3FeatureService, 'getValue')
+ .mockImplementation(async () => featureFlag)
+ factory = new FixtureFactory(app)
+ await Promise.all(
+ testCase.domains.map((domain) => factory.createDomain(domain)),
+ )
+ await Promise.all(
+ (testCase.accessTo ?? []).map((scope) =>
+ factory.createApiScopeUserAccess({
+ nationalId: testCase.user.nationalId,
+ scope,
+ }),
+ ),
+ )
+ })
+
+ beforeEach(async () => {
+ if (delegations) {
+ await Delegation.destroy({
+ where: {
+ id: {
+ [Op.in]: delegations.map((delegation) => delegation.id),
+ },
+ },
+ })
+ }
+
+ delegations = await Promise.all(
+ (testCase.delegations ?? []).map((delegation) =>
+ factory.createCustomDelegation({
+ toNationalId: testCase.user.nationalId,
+ fromName,
+ ...delegation,
+ }),
+ ),
+ )
})
- }
-
- delegations = await Promise.all(
- (testCase.delegations ?? []).map((delegation) =>
- factory.createCustomDelegation({
- toNationalId: testCase.user.nationalId,
- fromName,
- ...delegation,
- }),
- ),
- )
- })
-
- it('GET /v1/me/delegations?direction=incoming filters delegations', async () => {
- // Arrange
- const expectedDelegations = filterExpectedDelegations(
- delegations,
- testCase.expected,
- )
-
- // Act
- const res = await server.get('/v1/me/delegations?direction=incoming')
-
- // Assert
- expect(delegationIndexService.indexDelegations).toHaveBeenCalled()
- expect(res.status).toEqual(200)
- expectMatchingDelegations(res.body, expectedDelegations)
- })
-
- if (testCase.expected.length > 0) {
- it.each(testCase.expected)(
- 'GET /v1/me/delegation/:id finds delegation in domain $name',
- async (domain) => {
+
+ it('GET /v1/me/delegations?direction=incoming filters delegations', async () => {
// Arrange
- const expectedDelegation = filterExpectedDelegations(
+ const expectedDelegations = filterExpectedDelegations(
delegations,
testCase.expected,
- ).find((delegation) => delegation.domainName === domain.name)
- assert(expectedDelegation)
+ )
// Act
const res = await server.get(
- `/v1/me/delegations/${expectedDelegation.id}`,
+ '/v1/me/delegations?direction=incoming',
)
// Assert
+ expect(delegationIndexService.indexDelegations).toHaveBeenCalled()
expect(res.status).toEqual(200)
- expectMatchingDelegations(res.body, expectedDelegation)
- },
- )
-
- it.each(testCase.expected)(
- 'DELETE /v1/me/delegations/:id removes access to delegation',
- async (domain) => {
- // Arrange
- const delegationScopeModel = app.get(
- getModelToken(DelegationScope),
- )
- const delegationModel = app.get(
- getModelToken(Delegation),
- )
- const delegation = delegations.find(
- (delegation) =>
- delegation.domainName === domain.name &&
- delegation.toNationalId === testCase.user.nationalId,
- )
- assert(delegation)
+ expectMatchingDelegations(res.body, expectedDelegations)
+ })
- // Act
- const res = await server.delete(
- `/v1/me/delegations/${delegation.id}`,
+ if (testCase.expected.length > 0) {
+ it.each(testCase.expected)(
+ 'GET /v1/me/delegation/:id finds delegation in domain $name',
+ async (domain) => {
+ // Arrange
+ const expectedDelegation = filterExpectedDelegations(
+ delegations,
+ testCase.expected,
+ ).find((delegation) => delegation.domainName === domain.name)
+ assert(expectedDelegation)
+
+ // Act
+ const res = await server.get(
+ `/v1/me/delegations/${expectedDelegation.id}`,
+ )
+
+ // Assert
+ expect(res.status).toEqual(200)
+ expectMatchingDelegations(res.body, expectedDelegation)
+ },
)
- // Assert
- expect(res.status).toEqual(204)
- expect(
- delegationIndexService.indexCustomDelegations,
- ).toHaveBeenCalled()
-
- const delegationAfter = await delegationModel.findByPk(
- delegation.id,
+ it.each(testCase.expected)(
+ 'DELETE /v1/me/delegations/:id removes access to delegation',
+ async (domain) => {
+ // Arrange
+ const delegationScopeModel = app.get(
+ getModelToken(DelegationScope),
+ )
+ const delegationModel = app.get(
+ getModelToken(Delegation),
+ )
+ const delegation = delegations.find(
+ (delegation) =>
+ delegation.domainName === domain.name &&
+ delegation.toNationalId === testCase.user.nationalId,
+ )
+ assert(delegation)
+
+ // Act
+ const res = await server.delete(
+ `/v1/me/delegations/${delegation.id}`,
+ )
+
+ // Assert
+ expect(res.status).toEqual(204)
+ expect(
+ delegationIndexService.indexCustomDelegations,
+ ).toHaveBeenCalled()
+
+ const delegationAfter = await delegationModel.findByPk(
+ delegation.id,
+ )
+ expect(delegationAfter).toBeNull()
+
+ const scopesAfter = await delegationScopeModel.findAll({
+ where: {
+ delegationId: delegation.id,
+ scopeName:
+ delegation.delegationScopes?.map(
+ (scope) => scope.scopeName,
+ ) ?? [],
+ },
+ })
+ expect(scopesAfter).toHaveLength(0)
+ },
)
- expect(delegationAfter).toBeNull()
-
- const scopesAfter = await delegationScopeModel.findAll({
- where: {
- delegationId: delegation.id,
- scopeName:
- delegation.delegationScopes?.map(
- (scope) => scope.scopeName,
- ) ?? [],
+ }
+
+ if (testCase.expected.length === 0 && delegations.length > 0) {
+ it.each(delegations)(
+ 'GET /v1/me/delegation/:id returns no content response for $name',
+ async (delegation) => {
+ // Act
+ const res = await server.get(
+ `/v1/me/delegations/${delegation.id}`,
+ )
+
+ // Assert
+ expect(res.status).toEqual(204)
+ expect(res.body).toMatchObject({})
},
- })
- expect(scopesAfter).toHaveLength(0)
- },
- )
- }
-
- if (testCase.expected.length === 0 && delegations.length > 0) {
- it.each(delegations)(
- 'GET /v1/me/delegation/:id returns no content response for $name',
- async (delegation) => {
- // Act
- const res = await server.get(`/v1/me/delegations/${delegation.id}`)
-
- // Assert
- expect(res.status).toEqual(204)
- expect(res.body).toMatchObject({})
- },
- )
- }
+ )
+ }
+ },
+ )
},
)
})
diff --git a/apps/services/auth/ids-api/src/app/delegations/delegations-personal-representative.controller.spec.ts b/apps/services/auth/ids-api/src/app/delegations/delegations-personal-representative.controller.spec.ts
index f0050b0d9e9e..5c07ba95334d 100644
--- a/apps/services/auth/ids-api/src/app/delegations/delegations-personal-representative.controller.spec.ts
+++ b/apps/services/auth/ids-api/src/app/delegations/delegations-personal-representative.controller.spec.ts
@@ -13,6 +13,7 @@ import {
Domain,
InactiveReason,
MergedDelegationDTO,
+ NationalRegistryV3FeatureService,
PersonalRepresentative,
PersonalRepresentativeDelegationTypeModel,
PersonalRepresentativeRight,
@@ -22,6 +23,7 @@ import {
UNKNOWN_NAME,
} from '@island.is/auth-api-lib'
import { NationalRegistryClientService } from '@island.is/clients/national-registry-v2'
+import { NationalRegistryV3ClientService } from '@island.is/clients/national-registry-v3'
import {
createClient,
createDomain,
@@ -54,374 +56,190 @@ import {
} from '../../../test/stubs/personalRepresentativeStubs'
describe('Personal Representative DelegationsController', () => {
- describe('Given a user is authenticated', () => {
- let app: TestApp
- let factory: FixtureFactory
- let server: request.SuperTest
- let apiScopeModel: typeof ApiScope
- let clientModel: typeof Client
- let prScopePermission: typeof PersonalRepresentativeScopePermission
- let apiScopeDelegationTypeModel: typeof ApiScopeDelegationType
- let prModel: typeof PersonalRepresentative
- let prRightsModel: typeof PersonalRepresentativeRight
- let prRightTypeModel: typeof PersonalRepresentativeRightType
- let prTypeModel: typeof PersonalRepresentativeType
- let prDelegationTypeModel: typeof PersonalRepresentativeDelegationTypeModel
- let delegationTypeModel: typeof DelegationTypeModel
- let nationalRegistryApi: NationalRegistryClientService
- let delegationProviderModel: typeof DelegationProviderModel
- let delegationIndexService: DelegationsIndexService
-
- const client = createClient({
- clientId: '@island.is/webapp',
- })
-
- const scopeValid1 = 'scope/valid1'
- const scopeValid2 = 'scope/valid2'
- const scopeValid1and2 = 'scope/valid1and2'
- const scopeUnactiveType = 'scope/unactiveType'
- const scopeOutdated = 'scope/outdated'
- const disabledScope = 'disabledScope'
-
- client.allowedScopes = Object.values([
- scopeValid1,
- scopeValid2,
- scopeValid1and2,
- scopeUnactiveType,
- scopeOutdated,
- disabledScope,
- ]).map((s) => ({
- clientId: client.clientId,
- scopeName: s,
- }))
-
- const userNationalId = getFakeNationalId()
-
- const user = createCurrentUser({
- nationalId: userNationalId,
- scope: [defaultScopes.testUserHasAccess.name],
- client: client.clientId,
- })
-
- const domain = createDomain()
-
- beforeAll(async () => {
- app = await setupWithAuth({
- user,
- })
- server = request(app.getHttpServer())
-
- prTypeModel = app.get(
- getModelToken(PersonalRepresentativeType),
- )
-
- await prTypeModel.create(personalRepresentativeType)
-
- const domainModel = app.get(getModelToken(Domain))
- await domainModel.create(domain)
-
- apiScopeModel = app.get(getModelToken(ApiScope))
- prModel = app.get(
- getModelToken(PersonalRepresentative),
- )
- prRightsModel = app.get(
- getModelToken(PersonalRepresentativeRight),
- )
- prRightTypeModel = app.get(
- getModelToken(PersonalRepresentativeRightType),
- )
- prScopePermission = app.get(
- getModelToken(PersonalRepresentativeScopePermission),
- )
- apiScopeDelegationTypeModel = app.get(
- getModelToken(ApiScopeDelegationType),
- )
- prDelegationTypeModel = app.get<
- typeof PersonalRepresentativeDelegationTypeModel
- >(getModelToken(PersonalRepresentativeDelegationTypeModel))
- delegationTypeModel = app.get(
- getModelToken(DelegationTypeModel),
- )
- delegationProviderModel = app.get(
- getModelToken(DelegationProviderModel),
- )
- clientModel = app.get(getModelToken(Client))
- nationalRegistryApi = app.get(NationalRegistryClientService)
- delegationIndexService = app.get(DelegationsIndexService)
- factory = new FixtureFactory(app)
- })
-
- const createDelegationTypeAndProvider = async (rightCode: string[]) => {
- const newDelegationProvider = await delegationProviderModel.create({
- id: AuthDelegationProvider.PersonalRepresentativeRegistry,
- name: 'Talsmannagrunnur',
- description: 'Talsmannagrunnur',
- delegationTypes: [],
- })
-
- await delegationTypeModel.bulkCreate(
- rightCode.map((code) => {
- return {
- id: getPersonalRepresentativeDelegationType(code),
- providerId: newDelegationProvider.id,
- name: getPersonalRepresentativeDelegationType(code),
- description: `Personal representative delegation type for right type ${code}`,
- }
- }),
- )
- }
-
- afterAll(async () => {
- await app.cleanUp()
- })
-
- describe('and given we have 3 valid, 1 not yet active and 1 outdate right types', () => {
- type rightsTypeStatus = 'valid' | 'unactivated' | 'outdated'
- type rightsType = [code: string, status: rightsTypeStatus]
- const rightsTypes: rightsType[] = [
- ['valid1', 'valid'],
- ['valid2', 'valid'],
- ['unactivated', 'unactivated'],
- ['outdated', 'outdated'],
- ]
-
- beforeAll(async () => {
- await prRightTypeModel.bulkCreate(
- rightsTypes.map(([code, status]) => {
- switch (status) {
- case 'valid':
- return getPersonalRepresentativeRightType(code)
- case 'unactivated':
- return getPersonalRepresentativeRightType(
- code,
- faker.date.soon(7),
- )
- case 'outdated':
- return getPersonalRepresentativeRightType(
- code,
- faker.date.recent(5),
- faker.date.recent(),
- )
- }
- }),
- )
- await createDelegationTypeAndProvider(rightsTypes.map(([code]) => code))
-
- client.supportedDelegationTypes = delegationTypes
- await factory.createClient(client)
- })
-
- afterAll(async () => {
- await prRightTypeModel.destroy({
- where: {},
- cascade: true,
- truncate: true,
- force: true,
+ describe.each([false, true])(
+ 'national registry v3 featureflag: %s',
+ (featureFlag) => {
+ describe('Given a user is authenticated', () => {
+ let app: TestApp
+ let factory: FixtureFactory
+ let server: request.SuperTest
+ let apiScopeModel: typeof ApiScope
+ let clientModel: typeof Client
+ let prScopePermission: typeof PersonalRepresentativeScopePermission
+ let apiScopeDelegationTypeModel: typeof ApiScopeDelegationType
+ let prModel: typeof PersonalRepresentative
+ let prRightsModel: typeof PersonalRepresentativeRight
+ let prRightTypeModel: typeof PersonalRepresentativeRightType
+ let prTypeModel: typeof PersonalRepresentativeType
+ let prDelegationTypeModel: typeof PersonalRepresentativeDelegationTypeModel
+ let delegationTypeModel: typeof DelegationTypeModel
+ let nationalRegistryApi: NationalRegistryClientService
+ let nationalRegistryV3Api: NationalRegistryV3ClientService
+ let delegationProviderModel: typeof DelegationProviderModel
+ let delegationIndexService: DelegationsIndexService
+
+ const client = createClient({
+ clientId: '@island.is/webapp',
})
- await delegationTypeModel.destroy({
- where: {},
- cascade: true,
- truncate: true,
- force: true,
- })
- await delegationProviderModel.destroy({
- where: {},
- cascade: true,
- truncate: true,
- force: true,
+
+ const scopeValid1 = 'scope/valid1'
+ const scopeValid2 = 'scope/valid2'
+ const scopeValid1and2 = 'scope/valid1and2'
+ const scopeUnactiveType = 'scope/unactiveType'
+ const scopeOutdated = 'scope/outdated'
+ const disabledScope = 'disabledScope'
+
+ client.allowedScopes = Object.values([
+ scopeValid1,
+ scopeValid2,
+ scopeValid1and2,
+ scopeUnactiveType,
+ scopeOutdated,
+ disabledScope,
+ ]).map((s) => ({
+ clientId: client.clientId,
+ scopeName: s,
+ }))
+
+ const userNationalId = getFakeNationalId()
+
+ const user = createCurrentUser({
+ nationalId: userNationalId,
+ scope: [defaultScopes.testUserHasAccess.name],
+ client: client.clientId,
})
- })
- describe.each([
- [1, 0, 0, 2, 1],
- [2, 0, 0, 1, 0],
- [0, 0, 0, 0, 0],
- [0, 1, 0, 0, 1],
- [0, 0, 1, 0, 0],
- [0, 1, 1, 0, 2],
- [1, 1, 1, 0, 0],
- ])(
- 'and given user has %d active representees with valid rights, %d active representees with outdated rights and %d active representees with unactivated',
- (
- valid: number,
- outdated: number,
- unactivated: number,
- deceased: number,
- nationalRegistryErrors: number,
- ) => {
- let nationalRegistryApiSpy: jest.SpyInstance
- const validRepresentedPersons: NameIdTuple[] = []
- const outdatedRepresentedPersons: NameIdTuple[] = []
- const unactivatedRepresentedPersons: NameIdTuple[] = []
- const errorNationalIdsRepresentedPersons: NameIdTuple[] = []
- const deceasedNationalIds = times(deceased, getFakeNationalId)
- const errorNationalIds = times(
- nationalRegistryErrors,
- getFakeNationalId,
+ const domain = createDomain()
+
+ beforeAll(async () => {
+ app = await setupWithAuth({
+ user,
+ })
+ server = request(app.getHttpServer())
+
+ prTypeModel = app.get(
+ getModelToken(PersonalRepresentativeType),
)
- beforeAll(async () => {
- for (let i = 0; i < valid; i++) {
- const representedPerson: NameIdTuple = [
- getFakeName(),
- getFakeNationalId(),
- ]
- const relationship = getPersonalRepresentativeRelationship(
- userNationalId,
- representedPerson[1],
- )
- validRepresentedPersons.push(representedPerson)
- await prModel.create(relationship)
- await prRightsModel.create(
- getPersonalRepresentativeRights('valid1', relationship.id),
- )
- await prDelegationTypeModel.create({
- personalRepresentativeId: relationship.id,
- delegationTypeId:
- getPersonalRepresentativeDelegationType('valid1'),
- })
- }
-
- for (let i = 0; i < outdated; i++) {
- const representedPerson: NameIdTuple = [
- getFakeName(),
- getFakeNationalId(),
- ]
- const relationship = getPersonalRepresentativeRelationship(
- userNationalId,
- representedPerson[1],
- )
- outdatedRepresentedPersons.push(representedPerson)
- await prModel.create(relationship)
- await prRightsModel.create(
- getPersonalRepresentativeRights('outdated', relationship.id),
- )
- await prDelegationTypeModel.create({
- personalRepresentativeId: relationship.id,
- delegationTypeId:
- getPersonalRepresentativeDelegationType('outdated'),
- })
- }
-
- for (let i = 0; i < unactivated; i++) {
- const representedPerson: NameIdTuple = [
- getFakeName(),
- getFakeNationalId(),
- ]
- const relationship = getPersonalRepresentativeRelationship(
- userNationalId,
- representedPerson[1],
- )
- unactivatedRepresentedPersons.push(representedPerson)
- await prModel.create(relationship)
- await prRightsModel.create(
- getPersonalRepresentativeRights('unactivated', relationship.id),
- )
- await prDelegationTypeModel.create({
- personalRepresentativeId: relationship.id,
- delegationTypeId:
- getPersonalRepresentativeDelegationType('unactivated'),
- })
- }
-
- for (let i = 0; i < deceased; i++) {
- const representedPerson: NameIdTuple = [
- getFakeName(),
- deceasedNationalIds[i],
- ]
- const relationship = getPersonalRepresentativeRelationship(
- userNationalId,
- representedPerson[1],
- )
+ await prTypeModel.create(personalRepresentativeType)
- await prModel.create(relationship)
- await prRightsModel.create(
- getPersonalRepresentativeRights('valid1', relationship.id),
- )
- await prDelegationTypeModel.create({
- personalRepresentativeId: relationship.id,
- delegationTypeId:
- getPersonalRepresentativeDelegationType('valid1'),
- })
- }
-
- for (let i = 0; i < nationalRegistryErrors; i++) {
- const representedPerson: NameIdTuple = [
- UNKNOWN_NAME,
- errorNationalIds[i],
- ]
- const relationship = getPersonalRepresentativeRelationship(
- userNationalId,
- representedPerson[1],
- )
+ const domainModel = app.get(getModelToken(Domain))
+ await domainModel.create(domain)
- errorNationalIdsRepresentedPersons.push(representedPerson)
- // Create Personal Representative model which will have nationalIdRepresentedPerson throw an error
- // when national registry api getIndividual is called
- await prModel.create(relationship)
- await prRightsModel.create(
- getPersonalRepresentativeRights('valid1', relationship.id),
- )
- await prDelegationTypeModel.create({
- personalRepresentativeId: relationship.id,
- delegationTypeId:
- getPersonalRepresentativeDelegationType('valid1'),
- })
- }
-
- const nationalRegistryUsers = [
- ...validRepresentedPersons.map(([name, nationalId]) =>
- createNationalRegistryUser({ name, nationalId }),
- ),
- ...outdatedRepresentedPersons.map(([name, nationalId]) =>
- createNationalRegistryUser({ name, nationalId }),
- ),
- ...unactivatedRepresentedPersons.map(([name, nationalId]) =>
- createNationalRegistryUser({ name, nationalId }),
- ),
- ]
+ apiScopeModel = app.get(getModelToken(ApiScope))
+ prModel = app.get(
+ getModelToken(PersonalRepresentative),
+ )
+ prRightsModel = app.get(
+ getModelToken(PersonalRepresentativeRight),
+ )
+ prRightTypeModel = app.get(
+ getModelToken(PersonalRepresentativeRightType),
+ )
+ prScopePermission = app.get<
+ typeof PersonalRepresentativeScopePermission
+ >(getModelToken(PersonalRepresentativeScopePermission))
+ apiScopeDelegationTypeModel = app.get(
+ getModelToken(ApiScopeDelegationType),
+ )
+ prDelegationTypeModel = app.get<
+ typeof PersonalRepresentativeDelegationTypeModel
+ >(getModelToken(PersonalRepresentativeDelegationTypeModel))
+ delegationTypeModel = app.get(
+ getModelToken(DelegationTypeModel),
+ )
+ delegationProviderModel = app.get(
+ getModelToken(DelegationProviderModel),
+ )
+ clientModel = app.get(getModelToken(Client))
+ nationalRegistryApi = app.get(NationalRegistryClientService)
+ nationalRegistryV3Api = app.get(NationalRegistryV3ClientService)
+ delegationIndexService = app.get(DelegationsIndexService)
+ const nationalRegistryV3FeatureService = app.get(
+ NationalRegistryV3FeatureService,
+ )
+ jest
+ .spyOn(nationalRegistryV3FeatureService, 'getValue')
+ .mockImplementation(async () => featureFlag)
+ factory = new FixtureFactory(app)
+ })
- nationalRegistryApiSpy = jest
- .spyOn(nationalRegistryApi, 'getIndividual')
- .mockImplementation(async (id) => {
- if (deceasedNationalIds.includes(id)) {
- return null
- }
+ const createDelegationTypeAndProvider = async (rightCode: string[]) => {
+ const newDelegationProvider = await delegationProviderModel.create({
+ id: AuthDelegationProvider.PersonalRepresentativeRegistry,
+ name: 'Talsmannagrunnur',
+ description: 'Talsmannagrunnur',
+ delegationTypes: [],
+ })
- if (
- errorNationalIds.find(
- (errorNationalId) => id === errorNationalId,
- )
- ) {
- throw new Error('National registry error')
- }
+ await delegationTypeModel.bulkCreate(
+ rightCode.map((code) => {
+ return {
+ id: getPersonalRepresentativeDelegationType(code),
+ providerId: newDelegationProvider.id,
+ name: getPersonalRepresentativeDelegationType(code),
+ description: `Personal representative delegation type for right type ${code}`,
+ }
+ }),
+ )
+ }
- const user = nationalRegistryUsers.find(
- (u) =>
- u?.nationalId === id &&
- // Make sure we don't return a user that has been marked as deceased
- !deceasedNationalIds.includes(u?.nationalId),
- )
+ afterAll(async () => {
+ await app.cleanUp()
+ })
- return user ?? null
- })
+ describe('and given we have 3 valid, 1 not yet active and 1 outdate right types', () => {
+ type rightsTypeStatus = 'valid' | 'unactivated' | 'outdated'
+ type rightsType = [code: string, status: rightsTypeStatus]
+ const rightsTypes: rightsType[] = [
+ ['valid1', 'valid'],
+ ['valid2', 'valid'],
+ ['unactivated', 'unactivated'],
+ ['outdated', 'outdated'],
+ ]
+
+ beforeAll(async () => {
+ await prRightTypeModel.bulkCreate(
+ rightsTypes.map(([code, status]) => {
+ switch (status) {
+ case 'valid':
+ return getPersonalRepresentativeRightType(code)
+ case 'unactivated':
+ return getPersonalRepresentativeRightType(
+ code,
+ faker.date.soon(7),
+ )
+ case 'outdated':
+ return getPersonalRepresentativeRightType(
+ code,
+ faker.date.recent(5),
+ faker.date.recent(),
+ )
+ }
+ }),
+ )
+ await createDelegationTypeAndProvider(
+ rightsTypes.map(([code]) => code),
+ )
+
+ client.supportedDelegationTypes = delegationTypes
+ await factory.createClient(client)
})
afterAll(async () => {
- jest.clearAllMocks()
- await prRightsModel.destroy({
+ await prRightTypeModel.destroy({
where: {},
cascade: true,
truncate: true,
force: true,
})
- await prModel.destroy({
+ await delegationTypeModel.destroy({
where: {},
cascade: true,
truncate: true,
force: true,
})
- await prDelegationTypeModel.destroy({
+ await delegationProviderModel.destroy({
where: {},
cascade: true,
truncate: true,
@@ -429,215 +247,409 @@ describe('Personal Representative DelegationsController', () => {
})
})
- describe('when user calls GET /v2/delegations', () => {
- const path = '/v2/delegations'
- let response: request.Response
- let body: MergedDelegationDTO[]
+ describe.each([
+ [1, 0, 0, 2, 1],
+ [2, 0, 0, 1, 0],
+ [0, 0, 0, 0, 0],
+ [0, 1, 0, 0, 1],
+ [0, 0, 1, 0, 0],
+ [0, 1, 1, 0, 2],
+ [1, 1, 1, 0, 0],
+ ])(
+ 'and given user has %d active representees with valid rights, %d active representees with outdated rights and %d active representees with unactivated',
+ (
+ valid: number,
+ outdated: number,
+ unactivated: number,
+ deceased: number,
+ nationalRegistryErrors: number,
+ ) => {
+ let nationalRegistryApiSpy: jest.SpyInstance
+ let nationalRegistryV3ApiSpy: jest.SpyInstance
+ const validRepresentedPersons: NameIdTuple[] = []
+ const outdatedRepresentedPersons: NameIdTuple[] = []
+ const unactivatedRepresentedPersons: NameIdTuple[] = []
+ const errorNationalIdsRepresentedPersons: NameIdTuple[] = []
+ const deceasedNationalIds = times(deceased, getFakeNationalId)
+ const errorNationalIds = times(
+ nationalRegistryErrors,
+ getFakeNationalId,
+ )
- beforeAll(async () => {
- response = await server.get(path)
- body = response.body
- })
+ beforeAll(async () => {
+ for (let i = 0; i < valid; i++) {
+ const representedPerson: NameIdTuple = [
+ getFakeName(),
+ getFakeNationalId(),
+ ]
+ const relationship = getPersonalRepresentativeRelationship(
+ userNationalId,
+ representedPerson[1],
+ )
+ validRepresentedPersons.push(representedPerson)
+ await prModel.create(relationship)
+ await prRightsModel.create(
+ getPersonalRepresentativeRights('valid1', relationship.id),
+ )
+ await prDelegationTypeModel.create({
+ personalRepresentativeId: relationship.id,
+ delegationTypeId:
+ getPersonalRepresentativeDelegationType('valid1'),
+ })
+ }
- it('should have a an OK return status', () => {
- expect(response.status).toEqual(200)
- })
+ for (let i = 0; i < outdated; i++) {
+ const representedPerson: NameIdTuple = [
+ getFakeName(),
+ getFakeNationalId(),
+ ]
+ const relationship = getPersonalRepresentativeRelationship(
+ userNationalId,
+ representedPerson[1],
+ )
+ outdatedRepresentedPersons.push(representedPerson)
+ await prModel.create(relationship)
+ await prRightsModel.create(
+ getPersonalRepresentativeRights(
+ 'outdated',
+ relationship.id,
+ ),
+ )
+ await prDelegationTypeModel.create({
+ personalRepresentativeId: relationship.id,
+ delegationTypeId:
+ getPersonalRepresentativeDelegationType('outdated'),
+ })
+ }
- it(`should return ${valid} ${
- valid === 1 ? 'item' : 'items'
- } `, () => {
- expect(body).toHaveLength(valid + nationalRegistryErrors)
- })
+ for (let i = 0; i < unactivated; i++) {
+ const representedPerson: NameIdTuple = [
+ getFakeName(),
+ getFakeNationalId(),
+ ]
+ const relationship = getPersonalRepresentativeRelationship(
+ userNationalId,
+ representedPerson[1],
+ )
+ unactivatedRepresentedPersons.push(representedPerson)
+ await prModel.create(relationship)
+ await prRightsModel.create(
+ getPersonalRepresentativeRights(
+ 'unactivated',
+ relationship.id,
+ ),
+ )
+ await prDelegationTypeModel.create({
+ personalRepresentativeId: relationship.id,
+ delegationTypeId:
+ getPersonalRepresentativeDelegationType('unactivated'),
+ })
+ }
- it('should have the nationalId of the user as the representer', () => {
- expect(
- body.every((d) => d.toNationalId === userNationalId),
- ).toBeTruthy()
- })
+ for (let i = 0; i < deceased; i++) {
+ const representedPerson: NameIdTuple = [
+ getFakeName(),
+ deceasedNationalIds[i],
+ ]
+ const relationship = getPersonalRepresentativeRelationship(
+ userNationalId,
+ representedPerson[1],
+ )
- it('should only have the nationalId of the valid representees', () => {
- expect(body.map((d) => d.fromNationalId).sort()).toEqual(
- [
- ...validRepresentedPersons.map(([_, id]) => id),
- ...errorNationalIdsRepresentedPersons.map(([_, id]) => id),
- ].sort(),
- )
- })
+ await prModel.create(relationship)
+ await prRightsModel.create(
+ getPersonalRepresentativeRights('valid1', relationship.id),
+ )
+ await prDelegationTypeModel.create({
+ personalRepresentativeId: relationship.id,
+ delegationTypeId:
+ getPersonalRepresentativeDelegationType('valid1'),
+ })
+ }
- it(`should only have ${
- valid + nationalRegistryErrors === 1 ? 'name' : 'names'
- } of the valid represented ${
- valid + nationalRegistryErrors === 1 ? 'person' : 'persons'
- }`, () => {
- expect(body.map((d) => d.fromName).sort()).toEqual([
- ...validRepresentedPersons.map(([name, _]) => name).sort(),
- ...errorNationalIdsRepresentedPersons.map(([name]) => name),
- ])
- })
+ for (let i = 0; i < nationalRegistryErrors; i++) {
+ const representedPerson: NameIdTuple = [
+ UNKNOWN_NAME,
+ errorNationalIds[i],
+ ]
+ const relationship = getPersonalRepresentativeRelationship(
+ userNationalId,
+ representedPerson[1],
+ )
- it(`should have fetched the ${
- valid + deceased + nationalRegistryErrors === 1 ? 'name' : 'names'
- } of the valid represented ${
- valid + deceased + nationalRegistryErrors === 1
- ? 'person'
- : 'persons'
- } from nationalRegistryApi`, () => {
- expect(nationalRegistryApiSpy).toHaveBeenCalledTimes(
- valid + deceased + nationalRegistryErrors,
- )
- })
+ errorNationalIdsRepresentedPersons.push(representedPerson)
+ // Create Personal Representative model which will have nationalIdRepresentedPerson throw an error
+ // when national registry api getIndividual is called
+ await prModel.create(relationship)
+ await prRightsModel.create(
+ getPersonalRepresentativeRights('valid1', relationship.id),
+ )
+ await prDelegationTypeModel.create({
+ personalRepresentativeId: relationship.id,
+ delegationTypeId:
+ getPersonalRepresentativeDelegationType('valid1'),
+ })
+ }
- it('should have the delegation type claims of PersonalRepresentative', () => {
- expect(
- body.every(
- (d) =>
- d.types[0] === AuthDelegationType.PersonalRepresentative,
- ),
- ).toBeTruthy()
- })
+ const nationalRegistryUsers = [
+ ...validRepresentedPersons.map(([name, nationalId]) =>
+ createNationalRegistryUser({ name, nationalId }),
+ ),
+ ...outdatedRepresentedPersons.map(([name, nationalId]) =>
+ createNationalRegistryUser({ name, nationalId }),
+ ),
+ ...unactivatedRepresentedPersons.map(([name, nationalId]) =>
+ createNationalRegistryUser({ name, nationalId }),
+ ),
+ ]
+
+ nationalRegistryApiSpy = jest
+ .spyOn(nationalRegistryApi, 'getIndividual')
+ .mockImplementation(async (id) => {
+ if (deceasedNationalIds.includes(id)) {
+ return null
+ }
+
+ if (
+ errorNationalIds.find(
+ (errorNationalId) => id === errorNationalId,
+ )
+ ) {
+ throw new Error('National registry error')
+ }
+
+ const user = nationalRegistryUsers.find(
+ (u) =>
+ u?.nationalId === id &&
+ // Make sure we don't return a user that has been marked as deceased
+ !deceasedNationalIds.includes(u?.nationalId),
+ )
+
+ return user ?? null
+ })
+
+ nationalRegistryV3ApiSpy = jest
+ .spyOn(nationalRegistryV3Api, 'getAllDataIndividual')
+ .mockImplementation(async (id) => {
+ if (
+ errorNationalIds.find(
+ (errorNationalId) => id === errorNationalId,
+ )
+ ) {
+ throw new Error('National registry error')
+ }
+
+ if (deceasedNationalIds.includes(id)) {
+ return {
+ kennitala: id,
+ afdrif: 'LÉST',
+ }
+ }
+
+ const user = nationalRegistryUsers.find(
+ (u) => u?.nationalId === id,
+ )
+
+ return user
+ ? {
+ kennitala: id,
+ nafn: user?.name,
+ afdrif: null,
+ }
+ : null
+ })
+ })
- it('should have made prModels inactive for deceased persons', async () => {
- // Arrange
- const expectedModels = await prModel.findAll({
- where: {
- nationalIdRepresentedPerson: deceasedNationalIds,
- inactive: true,
- inactiveReason: InactiveReason.DECEASED_PARTY,
- },
+ afterAll(async () => {
+ jest.clearAllMocks()
+ await prRightsModel.destroy({
+ where: {},
+ cascade: true,
+ truncate: true,
+ force: true,
+ })
+ await prModel.destroy({
+ where: {},
+ cascade: true,
+ truncate: true,
+ force: true,
+ })
+ await prDelegationTypeModel.destroy({
+ where: {},
+ cascade: true,
+ truncate: true,
+ force: true,
+ })
})
- // Assert
- expect(expectedModels.length).toEqual(deceased)
+ describe('when user calls GET /v2/delegations', () => {
+ const path = '/v2/delegations'
+ let response: request.Response
+ let body: MergedDelegationDTO[]
- expectedModels.forEach((model) => {
- expect(model.inactive).toEqual(true)
- expect(model.inactiveReason).toEqual(
- InactiveReason.DECEASED_PARTY,
- )
- })
- })
+ beforeAll(async () => {
+ response = await server.get(path)
+ body = response.body
+ })
- it('should return delegation if national registry api getIndividual throws an error', async () => {
- // Arrange
- const expectedModels = await prModel.findAll({
- where: {
- nationalIdRepresentedPerson: errorNationalIds,
- },
- })
+ it('should have a an OK return status', () => {
+ expect(response.status).toEqual(200)
+ })
- // Assert
- expect(expectedModels.length).toEqual(errorNationalIds.length)
- })
+ it(`should return ${valid} ${
+ valid === 1 ? 'item' : 'items'
+ } `, () => {
+ expect(body).toHaveLength(valid + nationalRegistryErrors)
+ })
- it('should index delegations', () => {
- expect(delegationIndexService.indexDelegations).toHaveBeenCalled()
- })
- })
- },
- )
-
- describe('and given we have a combination of scopes for personal representative', () => {
- type scopesType = [name: string, enabled: boolean, rightTypes: string[]]
- const scopes: scopesType[] = [
- [scopeValid1, true, ['valid1']],
- [scopeValid2, true, ['valid2']],
- [scopeValid1and2, true, ['valid1', 'valid2']],
- [scopeUnactiveType, true, ['unactivated']],
- [scopeOutdated, true, ['outdated']],
- [disabledScope, false, ['valid1']],
- ]
+ it('should have the nationalId of the user as the representer', () => {
+ expect(
+ body.every((d) => d.toNationalId === userNationalId),
+ ).toBeTruthy()
+ })
- beforeAll(async () => {
- const apiScopes = scopes.flatMap(([name, enabled, types]) => ({
- name: name,
- enabled,
- domainName: domain.name,
- supportedDelegationTypes: types.map((rt) =>
- getPersonalRepresentativeDelegationType(rt),
- ),
- }))
-
- await Promise.all(
- apiScopes.map((scope) => factory.createApiScope(scope)),
- )
+ it('should only have the nationalId of the valid representees', () => {
+ expect(body.map((d) => d.fromNationalId).sort()).toEqual(
+ [
+ ...validRepresentedPersons.map(([_, id]) => id),
+ ...errorNationalIdsRepresentedPersons.map(
+ ([_, id]) => id,
+ ),
+ ].sort(),
+ )
+ })
- await prScopePermission.bulkCreate(
- scopes.flatMap(([name, _, types]) =>
- types.map((rt) => getScopePermission(rt, name)),
- ),
- )
- })
+ it(`should only have ${
+ valid + nationalRegistryErrors === 1 ? 'name' : 'names'
+ } of the valid represented ${
+ valid + nationalRegistryErrors === 1 ? 'person' : 'persons'
+ }`, () => {
+ expect(body.map((d) => d.fromName).sort()).toEqual([
+ ...validRepresentedPersons.map(([name, _]) => name).sort(),
+ ...errorNationalIdsRepresentedPersons.map(([name]) => name),
+ ])
+ })
- afterAll(async () => {
- await prScopePermission.destroy({
- where: {},
- cascade: true,
- truncate: true,
- force: true,
- })
- await apiScopeDelegationTypeModel.destroy({
- where: {},
- cascade: true,
- truncate: true,
- force: true,
- })
- await apiScopeModel.destroy({
- where: {},
- cascade: true,
- truncate: true,
- force: true,
- })
- })
+ it(`should have fetched the ${
+ valid + deceased + nationalRegistryErrors === 1
+ ? 'name'
+ : 'names'
+ } of the valid represented ${
+ valid + deceased + nationalRegistryErrors === 1
+ ? 'person'
+ : 'persons'
+ } from nationalRegistryApi`, () => {
+ featureFlag
+ ? expect(nationalRegistryV3ApiSpy).toHaveBeenCalledTimes(
+ valid + deceased + nationalRegistryErrors,
+ )
+ : expect(nationalRegistryApiSpy).toHaveBeenCalledTimes(
+ valid + deceased + nationalRegistryErrors,
+ )
+ })
+
+ it('should have the delegation type claims of PersonalRepresentative', () => {
+ expect(
+ body.every(
+ (d) =>
+ d.types[0] ===
+ AuthDelegationType.PersonalRepresentative,
+ ),
+ ).toBeTruthy()
+ })
+
+ it('should have made prModels inactive for deceased persons', async () => {
+ // Arrange
+ const expectedModels = await prModel.findAll({
+ where: {
+ nationalIdRepresentedPerson: deceasedNationalIds,
+ inactive: true,
+ inactiveReason: InactiveReason.DECEASED_PARTY,
+ },
+ })
+
+ // Assert
+ expect(expectedModels.length).toEqual(deceased)
+
+ expectedModels.forEach((model) => {
+ expect(model.inactive).toEqual(true)
+ expect(model.inactiveReason).toEqual(
+ InactiveReason.DECEASED_PARTY,
+ )
+ })
+ })
- describe.each([
- [['valid1'], [scopeValid1, scopeValid1and2]],
- [['valid2'], [scopeValid2, scopeValid1and2]],
- [
- ['valid1', 'valid2'],
- [scopeValid1, scopeValid2, scopeValid1and2],
- ],
- [[], []],
- // [['unactivated'], []],
- // [['outdated'], []],
- ])(
- 'and given user is representing persons with rights %p',
- (rights, expected) => {
- const representeeNationalId = getFakeNationalId()
+ it('should return delegation if national registry api getIndividual throws an error', async () => {
+ // Arrange
+ const expectedModels = await prModel.findAll({
+ where: {
+ nationalIdRepresentedPerson: errorNationalIds,
+ },
+ })
+
+ // Assert
+ expect(expectedModels.length).toEqual(errorNationalIds.length)
+ })
+
+ it('should index delegations', () => {
+ expect(
+ delegationIndexService.indexDelegations,
+ ).toHaveBeenCalled()
+ })
+ })
+ },
+ )
+
+ describe('and given we have a combination of scopes for personal representative', () => {
+ type scopesType = [
+ name: string,
+ enabled: boolean,
+ rightTypes: string[],
+ ]
+ const scopes: scopesType[] = [
+ [scopeValid1, true, ['valid1']],
+ [scopeValid2, true, ['valid2']],
+ [scopeValid1and2, true, ['valid1', 'valid2']],
+ [scopeUnactiveType, true, ['unactivated']],
+ [scopeOutdated, true, ['outdated']],
+ [disabledScope, false, ['valid1']],
+ ]
beforeAll(async () => {
- const relationship = getPersonalRepresentativeRelationship(
- userNationalId,
- representeeNationalId,
+ const apiScopes = scopes.flatMap(([name, enabled, types]) => ({
+ name: name,
+ enabled,
+ domainName: domain.name,
+ supportedDelegationTypes: types.map((rt) =>
+ getPersonalRepresentativeDelegationType(rt),
+ ),
+ }))
+
+ await Promise.all(
+ apiScopes.map((scope) => factory.createApiScope(scope)),
)
- await prModel.create(relationship)
- await prRightsModel.bulkCreate(
- rights.map((r) =>
- getPersonalRepresentativeRights(r, relationship.id),
+ await prScopePermission.bulkCreate(
+ scopes.flatMap(([name, _, types]) =>
+ types.map((rt) => getScopePermission(rt, name)),
),
)
- await prDelegationTypeModel.bulkCreate(
- rights.map((r) => ({
- personalRepresentativeId: relationship.id,
- delegationTypeId: getPersonalRepresentativeDelegationType(r),
- })),
- )
})
afterAll(async () => {
- await prRightsModel.destroy({
+ await prScopePermission.destroy({
where: {},
cascade: true,
truncate: true,
force: true,
})
- await prModel.destroy({
+ await apiScopeDelegationTypeModel.destroy({
where: {},
cascade: true,
truncate: true,
force: true,
})
- await prDelegationTypeModel.destroy({
+ await apiScopeModel.destroy({
where: {},
cascade: true,
truncate: true,
@@ -645,34 +657,95 @@ describe('Personal Representative DelegationsController', () => {
})
})
- describe('when user calls GET /delegations/scopes', () => {
- const path = '/delegations/scopes'
- let response: request.Response
- let body: string[]
+ describe.each([
+ [['valid1'], [scopeValid1, scopeValid1and2]],
+ [['valid2'], [scopeValid2, scopeValid1and2]],
+ [
+ ['valid1', 'valid2'],
+ [scopeValid1, scopeValid2, scopeValid1and2],
+ ],
+ [[], []],
+ // [['unactivated'], []],
+ // [['outdated'], []],
+ ])(
+ 'and given user is representing persons with rights %p',
+ (rights, expected) => {
+ const representeeNationalId = getFakeNationalId()
+
+ beforeAll(async () => {
+ const relationship = getPersonalRepresentativeRelationship(
+ userNationalId,
+ representeeNationalId,
+ )
- beforeAll(async () => {
- response = await server.get(`${path}`).query({
- fromNationalId: representeeNationalId,
- delegationType: rights.map((r) =>
- getPersonalRepresentativeDelegationType(r),
- ),
+ await prModel.create(relationship)
+ await prRightsModel.bulkCreate(
+ rights.map((r) =>
+ getPersonalRepresentativeRights(r, relationship.id),
+ ),
+ )
+ await prDelegationTypeModel.bulkCreate(
+ rights.map((r) => ({
+ personalRepresentativeId: relationship.id,
+ delegationTypeId:
+ getPersonalRepresentativeDelegationType(r),
+ })),
+ )
})
- body = response.body
- })
- it('should have a an OK return status', () => {
- expect(response.status).toEqual(200)
- })
+ afterAll(async () => {
+ await prRightsModel.destroy({
+ where: {},
+ cascade: true,
+ truncate: true,
+ force: true,
+ })
+ await prModel.destroy({
+ where: {},
+ cascade: true,
+ truncate: true,
+ force: true,
+ })
+ await prDelegationTypeModel.destroy({
+ where: {},
+ cascade: true,
+ truncate: true,
+ force: true,
+ })
+ })
- it(`should return ${
- expected.length === 0 ? 'no scopes' : JSON.stringify(expected)
- }`, () => {
- expect(body.sort()).toEqual(expected.sort())
- })
- })
- },
- )
+ describe('when user calls GET /delegations/scopes', () => {
+ const path = '/delegations/scopes'
+ let response: request.Response
+ let body: string[]
+
+ beforeAll(async () => {
+ response = await server.get(`${path}`).query({
+ fromNationalId: representeeNationalId,
+ delegationType: rights.map((r) =>
+ getPersonalRepresentativeDelegationType(r),
+ ),
+ })
+ body = response.body
+ })
+
+ it('should have a an OK return status', () => {
+ expect(response.status).toEqual(200)
+ })
+
+ it(`should return ${
+ expected.length === 0
+ ? 'no scopes'
+ : JSON.stringify(expected)
+ }`, () => {
+ expect(body.sort()).toEqual(expected.sort())
+ })
+ })
+ },
+ )
+ })
+ })
})
- })
- })
+ },
+ )
})
diff --git a/apps/services/auth/ids-api/src/app/delegations/test/delegations-filters.spec.ts b/apps/services/auth/ids-api/src/app/delegations/test/delegations-filters.spec.ts
index 526bf4262a4a..eb6c5a885989 100644
--- a/apps/services/auth/ids-api/src/app/delegations/test/delegations-filters.spec.ts
+++ b/apps/services/auth/ids-api/src/app/delegations/test/delegations-filters.spec.ts
@@ -4,9 +4,13 @@ import faker from 'faker'
import { Sequelize } from 'sequelize-typescript'
import request from 'supertest'
-import { MergedDelegationDTO } from '@island.is/auth-api-lib'
+import {
+ MergedDelegationDTO,
+ NationalRegistryV3FeatureService,
+} from '@island.is/auth-api-lib'
import { RskRelationshipsClient } from '@island.is/clients-rsk-relationships'
import { NationalRegistryClientService } from '@island.is/clients/national-registry-v2'
+import { NationalRegistryV3ClientService } from '@island.is/clients/national-registry-v3'
import { FixtureFactory } from '@island.is/services/auth/testing'
import {
AuthDelegationProvider,
@@ -23,166 +27,190 @@ import { testCases } from './delegations-filters-test-cases'
import { user } from './delegations-filters-types'
describe('DelegationsController', () => {
- let sequelize: Sequelize
- let app: TestApp
- let server: request.SuperTest
- let factory: FixtureFactory
- let nationalRegistryApi: NationalRegistryClientService
- let rskApi: RskRelationshipsClient
- beforeAll(async () => {
- app = await setupWithAuth({
- user: user,
- })
- sequelize = await app.resolve(getConnectionToken() as Type)
-
- server = request(app.getHttpServer())
-
- nationalRegistryApi = app.get(NationalRegistryClientService)
- jest
- .spyOn(nationalRegistryApi, 'getIndividual')
- .mockImplementation(async (nationalId: string) =>
- createNationalRegistryUser({
- nationalId,
- name: faker.name.findName(),
- }),
- )
- rskApi = app.get(RskRelationshipsClient)
-
- factory = new FixtureFactory(app)
- })
-
- afterAll(async () => {
- await app.cleanUp()
- })
-
- describe.each(Object.keys(testCases))(
- 'Delegation filtering with test case: %s',
- (caseName) => {
- const testCase = testCases[caseName]
- testCase.user = user
- const path = '/v2/delegations'
-
+ describe.each([false, true])(
+ 'national registry v3 featureflag: %s',
+ (featureFlag) => {
+ let sequelize: Sequelize
+ let app: TestApp
+ let server: request.SuperTest
+ let factory: FixtureFactory
+ let nationalRegistryApi: NationalRegistryClientService
+ let nationalRegistryV3Api: NationalRegistryV3ClientService
+ let rskApi: RskRelationshipsClient
beforeAll(async () => {
- await truncate(sequelize)
-
- await Promise.all(
- testCase.domains.map((domain) => factory.createDomain(domain)),
- )
+ app = await setupWithAuth({
+ user: user,
+ })
+ sequelize = await app.resolve(getConnectionToken() as Type)
- await factory.createClient(testCase.client)
-
- await Promise.all(
- testCase.clientAllowedScopes.map((scope) =>
- factory.createClientAllowedScope(scope),
- ),
- )
+ server = request(app.getHttpServer())
- await Promise.all(
- testCase.apiScopes.map((scope) => factory.createApiScope(scope)),
- )
-
- await Promise.all(
- testCase.apiScopeUserAccess.map((access) =>
- factory.createApiScopeUserAccess(access),
- ),
- )
-
- await Promise.all(
- testCase.customDelegations.map((delegation) =>
- factory.createCustomDelegation(delegation),
- ),
- )
-
- await Promise.all(
- testCase.fromLegalRepresentative.map((nationalId) =>
- factory.createDelegationIndexRecord({
- fromNationalId: nationalId,
- toNationalId: testCase.user.nationalId,
- type: AuthDelegationType.LegalRepresentative,
- provider: AuthDelegationProvider.DistrictCommissionersRegistry,
+ nationalRegistryApi = app.get(NationalRegistryClientService)
+ jest
+ .spyOn(nationalRegistryApi, 'getIndividual')
+ .mockImplementation(async (nationalId: string) =>
+ createNationalRegistryUser({
+ nationalId,
+ name: faker.name.findName(),
}),
- ),
+ )
+ nationalRegistryV3Api = app.get(NationalRegistryV3ClientService)
+ jest
+ .spyOn(nationalRegistryV3Api, 'getAllDataIndividual')
+ .mockImplementation(async (nationalId: string) => {
+ const user = createNationalRegistryUser({
+ nationalId: nationalId,
+ })
+
+ return { kennitala: user.nationalId, nafn: user.name }
+ })
+ rskApi = app.get(RskRelationshipsClient)
+
+ const nationalRegistryV3FeatureService = app.get(
+ NationalRegistryV3FeatureService,
)
-
jest
- .spyOn(nationalRegistryApi, 'getCustodyChildren')
- .mockImplementation(async () => testCase.fromChildren)
+ .spyOn(nationalRegistryV3FeatureService, 'getValue')
+ .mockImplementation(async () => featureFlag)
- jest
- .spyOn(rskApi, 'getIndividualRelationships')
- .mockImplementation(async () => testCase.procuration)
+ factory = new FixtureFactory(app)
})
- let res: request.Response
-
- it(`GET ${path} returns correct filtered delegations`, async () => {
- res = await server.get(path)
-
- expect(res.status).toEqual(200)
- expect(res.body).toHaveLength(testCase.expectedFrom.length)
- expect(
- res.body.map((d: MergedDelegationDTO) => d.fromNationalId).sort(),
- ).toEqual(testCase.expectedFrom.sort())
- if (testCase.expectedTypes) {
- expect(res.body[0].types.sort()).toEqual(
- testCase.expectedTypes.sort(),
- )
- }
+ afterAll(async () => {
+ await app.cleanUp()
})
- },
- )
-
- describe('verify', () => {
- const testCase = testCases['legalRepresentative1']
- testCase.user = user
- const path = '/v1/delegations/verify'
- beforeAll(async () => {
- await truncate(sequelize)
-
- await Promise.all(
- testCase.domains.map((domain) => factory.createDomain(domain)),
+ describe.each(Object.keys(testCases))(
+ 'Delegation filtering with test case: %s',
+ (caseName) => {
+ const testCase = testCases[caseName]
+ testCase.user = user
+ const path = '/v2/delegations'
+
+ beforeAll(async () => {
+ await truncate(sequelize)
+
+ await Promise.all(
+ testCase.domains.map((domain) => factory.createDomain(domain)),
+ )
+
+ await factory.createClient(testCase.client)
+
+ await Promise.all(
+ testCase.clientAllowedScopes.map((scope) =>
+ factory.createClientAllowedScope(scope),
+ ),
+ )
+
+ await Promise.all(
+ testCase.apiScopes.map((scope) => factory.createApiScope(scope)),
+ )
+
+ await Promise.all(
+ testCase.apiScopeUserAccess.map((access) =>
+ factory.createApiScopeUserAccess(access),
+ ),
+ )
+
+ await Promise.all(
+ testCase.customDelegations.map((delegation) =>
+ factory.createCustomDelegation(delegation),
+ ),
+ )
+
+ await Promise.all(
+ testCase.fromLegalRepresentative.map((nationalId) =>
+ factory.createDelegationIndexRecord({
+ fromNationalId: nationalId,
+ toNationalId: testCase.user.nationalId,
+ type: AuthDelegationType.LegalRepresentative,
+ provider:
+ AuthDelegationProvider.DistrictCommissionersRegistry,
+ }),
+ ),
+ )
+
+ jest
+ .spyOn(nationalRegistryApi, 'getCustodyChildren')
+ .mockImplementation(async () => testCase.fromChildren)
+
+ jest
+ .spyOn(rskApi, 'getIndividualRelationships')
+ .mockImplementation(async () => testCase.procuration)
+ })
+
+ let res: request.Response
+
+ it(`GET ${path} returns correct filtered delegations`, async () => {
+ res = await server.get(path)
+
+ expect(res.status).toEqual(200)
+ expect(res.body).toHaveLength(testCase.expectedFrom.length)
+ expect(
+ res.body.map((d: MergedDelegationDTO) => d.fromNationalId).sort(),
+ ).toEqual(testCase.expectedFrom.sort())
+ if (testCase.expectedTypes) {
+ expect(res.body[0].types.sort()).toEqual(
+ testCase.expectedTypes.sort(),
+ )
+ }
+ })
+ },
)
- await factory.createClient(testCase.client)
+ describe('verify', () => {
+ const testCase = testCases['legalRepresentative1']
+ testCase.user = user
+ const path = '/v1/delegations/verify'
- await Promise.all(
- testCase.clientAllowedScopes.map((scope) =>
- factory.createClientAllowedScope(scope),
- ),
- )
+ beforeAll(async () => {
+ await truncate(sequelize)
- await Promise.all(
- testCase.apiScopes.map((scope) => factory.createApiScope(scope)),
- )
+ await Promise.all(
+ testCase.domains.map((domain) => factory.createDomain(domain)),
+ )
- await factory.createDelegationIndexRecord({
- fromNationalId: nonExistingLegalRepresentativeNationalId,
- toNationalId: testCase.user.nationalId,
- type: AuthDelegationType.LegalRepresentative,
- provider: AuthDelegationProvider.DistrictCommissionersRegistry,
- })
- })
+ await factory.createClient(testCase.client)
- let res: request.Response
- it(`POST ${path} returns verified response`, async () => {
- res = await server.post(path).send({
- fromNationalId: testCase.fromLegalRepresentative[0],
- delegationTypes: [AuthDelegationType.LegalRepresentative],
- })
+ await Promise.all(
+ testCase.clientAllowedScopes.map((scope) =>
+ factory.createClientAllowedScope(scope),
+ ),
+ )
- expect(res.status).toEqual(200)
- expect(res.body.verified).toEqual(true)
- })
+ await Promise.all(
+ testCase.apiScopes.map((scope) => factory.createApiScope(scope)),
+ )
- it(`POST ${path} returns non-verified response`, async () => {
- res = await server.post(path).send({
- fromNationalId: nonExistingLegalRepresentativeNationalId,
- delegationTypes: [AuthDelegationType.LegalRepresentative],
+ await factory.createDelegationIndexRecord({
+ fromNationalId: nonExistingLegalRepresentativeNationalId,
+ toNationalId: testCase.user.nationalId,
+ type: AuthDelegationType.LegalRepresentative,
+ provider: AuthDelegationProvider.DistrictCommissionersRegistry,
+ })
+ })
+
+ let res: request.Response
+ it(`POST ${path} returns verified response`, async () => {
+ res = await server.post(path).send({
+ fromNationalId: testCase.fromLegalRepresentative[0],
+ delegationTypes: [AuthDelegationType.LegalRepresentative],
+ })
+
+ expect(res.status).toEqual(200)
+ expect(res.body.verified).toEqual(true)
+ })
+
+ it(`POST ${path} returns non-verified response`, async () => {
+ res = await server.post(path).send({
+ fromNationalId: nonExistingLegalRepresentativeNationalId,
+ delegationTypes: [AuthDelegationType.LegalRepresentative],
+ })
+
+ expect(res.status).toEqual(200)
+ expect(res.body.verified).toEqual(false)
+ })
})
-
- expect(res.status).toEqual(200)
- expect(res.body.verified).toEqual(false)
- })
- })
+ },
+ )
})
diff --git a/apps/services/auth/ids-api/src/app/delegations/test/delegations.controller.spec.ts b/apps/services/auth/ids-api/src/app/delegations/test/delegations.controller.spec.ts
index 4abd4987bd31..f2dee16513d6 100644
--- a/apps/services/auth/ids-api/src/app/delegations/test/delegations.controller.spec.ts
+++ b/apps/services/auth/ids-api/src/app/delegations/test/delegations.controller.spec.ts
@@ -1,7 +1,7 @@
import { getModelToken } from '@nestjs/sequelize'
+import addDays from 'date-fns/addDays'
import request from 'supertest'
import { uuid } from 'uuidv4'
-import addDays from 'date-fns/addDays'
import {
ApiScope,
@@ -12,8 +12,10 @@ import {
DelegationScope,
DelegationTypeModel,
Domain,
+ NationalRegistryV3FeatureService,
} from '@island.is/auth-api-lib'
import { NationalRegistryClientService } from '@island.is/clients/national-registry-v2'
+import { NationalRegistryV3ClientService } from '@island.is/clients/national-registry-v3'
import {
createClient,
createDomain,
@@ -33,246 +35,273 @@ import { defaultScopes, setupWithAuth } from '../../../../test/setup'
import { getFakeNationalId } from '../../../../test/stubs/genericStubs'
describe('DelegationsController', () => {
- describe('Given a user is authenticated', () => {
- let app: TestApp
- let factory: FixtureFactory
- let server: request.SuperTest
-
- let apiScopeModel: typeof ApiScope
- let apiScopeDelegationTypeModel: typeof ApiScopeDelegationType
- let delegationDelegationTypeModel: typeof DelegationDelegationType
- let delegationModel: typeof Delegation
- let delegationTypeModel: typeof DelegationTypeModel
- let nationalRegistryApi: NationalRegistryClientService
- let delegationProviderModel: typeof DelegationProviderModel
- let delegationScopesModel: typeof DelegationScope
-
- const client = createClient({
- clientId: '@island.is/webapp',
- })
-
- const scopeValid1 = 'scope/valid1'
- const scopeValid2 = 'scope/valid2'
- const scopeValid1and2 = 'scope/valid1and2'
- const scopeUnactiveType = 'scope/unactiveType'
- const scopeOutdated = 'scope/outdated'
- const disabledScope = 'disabledScope'
-
- client.allowedScopes = Object.values([
- scopeValid1,
- scopeValid2,
- scopeValid1and2,
- scopeUnactiveType,
- scopeOutdated,
- disabledScope,
- ]).map((s) => ({
- clientId: client.clientId,
- scopeName: s,
- }))
-
- const userNationalId = getFakeNationalId()
-
- const user = createCurrentUser({
- nationalId: userNationalId,
- scope: [defaultScopes.testUserHasAccess.name],
- client: client.clientId,
- })
-
- const domain = createDomain()
-
- beforeAll(async () => {
- app = await setupWithAuth({
- user,
- })
- server = request(app.getHttpServer())
-
- const domainModel = app.get(getModelToken(Domain))
- await domainModel.create(domain)
-
- apiScopeModel = app.get(getModelToken(ApiScope))
-
- apiScopeDelegationTypeModel = app.get(
- getModelToken(ApiScopeDelegationType),
- )
- delegationTypeModel = app.get(
- getModelToken(DelegationTypeModel),
- )
- delegationProviderModel = app.get(
- getModelToken(DelegationProviderModel),
- )
- delegationScopesModel = app.get(
- getModelToken(DelegationScope),
- )
- delegationModel = app.get(getModelToken(Delegation))
- delegationDelegationTypeModel = app.get(
- getModelToken(DelegationDelegationType),
- )
- nationalRegistryApi = app.get(NationalRegistryClientService)
- factory = new FixtureFactory(app)
- })
-
- afterAll(async () => {
- await app.cleanUp()
- })
-
- describe('GET with general mandate delegation type', () => {
- const representeeNationalId = getFakeNationalId()
- let nationalRegistryApiSpy: jest.SpyInstance
- const scopeNames = [
- 'api-scope/generalMandate1',
- 'api-scope/generalMandate2',
- 'api-scope/generalMandate3',
- ]
-
- beforeAll(async () => {
- client.supportedDelegationTypes = [
- AuthDelegationType.GeneralMandate,
- AuthDelegationType.LegalGuardian,
- ]
- await factory.createClient(client)
-
- const delegations = await delegationModel.create({
- id: uuid(),
- fromDisplayName: 'Test',
- fromNationalId: representeeNationalId,
- toNationalId: userNationalId,
- toName: 'Test',
+ describe.each([false, true])(
+ 'national registry v3 featureflag: %s',
+ (featureFlag) => {
+ describe('Given a user is authenticated', () => {
+ let app: TestApp
+ let factory: FixtureFactory
+ let server: request.SuperTest
+
+ let apiScopeModel: typeof ApiScope
+ let apiScopeDelegationTypeModel: typeof ApiScopeDelegationType
+ let delegationDelegationTypeModel: typeof DelegationDelegationType
+ let delegationModel: typeof Delegation
+ let delegationTypeModel: typeof DelegationTypeModel
+ let nationalRegistryApi: NationalRegistryClientService
+ let nationalRegistryV3Api: NationalRegistryV3ClientService
+ let delegationProviderModel: typeof DelegationProviderModel
+ let delegationScopesModel: typeof DelegationScope
+
+ const client = createClient({
+ clientId: '@island.is/webapp',
+ })
+
+ const scopeValid1 = 'scope/valid1'
+ const scopeValid2 = 'scope/valid2'
+ const scopeValid1and2 = 'scope/valid1and2'
+ const scopeUnactiveType = 'scope/unactiveType'
+ const scopeOutdated = 'scope/outdated'
+ const disabledScope = 'disabledScope'
+
+ client.allowedScopes = Object.values([
+ scopeValid1,
+ scopeValid2,
+ scopeValid1and2,
+ scopeUnactiveType,
+ scopeOutdated,
+ disabledScope,
+ ]).map((s) => ({
+ clientId: client.clientId,
+ scopeName: s,
+ }))
+
+ const userNationalId = getFakeNationalId()
+
+ const user = createCurrentUser({
+ nationalId: userNationalId,
+ scope: [defaultScopes.testUserHasAccess.name],
+ client: client.clientId,
})
- await delegationProviderModel.create({
- id: AuthDelegationProvider.Custom,
- name: 'Custom',
- description: 'Custom',
+ const domain = createDomain()
+
+ beforeAll(async () => {
+ app = await setupWithAuth({
+ user,
+ })
+ server = request(app.getHttpServer())
+
+ const domainModel = app.get(getModelToken(Domain))
+ await domainModel.create(domain)
+
+ apiScopeModel = app.get(getModelToken(ApiScope))
+
+ apiScopeDelegationTypeModel = app.get(
+ getModelToken(ApiScopeDelegationType),
+ )
+ delegationTypeModel = app.get(
+ getModelToken(DelegationTypeModel),
+ )
+ delegationProviderModel = app.get(
+ getModelToken(DelegationProviderModel),
+ )
+ delegationScopesModel = app.get(
+ getModelToken(DelegationScope),
+ )
+ delegationModel = app.get(
+ getModelToken(Delegation),
+ )
+ delegationDelegationTypeModel = app.get<
+ typeof DelegationDelegationType
+ >(getModelToken(DelegationDelegationType))
+ nationalRegistryApi = app.get(NationalRegistryClientService)
+ nationalRegistryV3Api = app.get(NationalRegistryV3ClientService)
+ const nationalRegistryV3FeatureService = app.get(
+ NationalRegistryV3FeatureService,
+ )
+ jest
+ .spyOn(nationalRegistryV3FeatureService, 'getValue')
+ .mockImplementation(async () => featureFlag)
+ factory = new FixtureFactory(app)
})
- await delegationDelegationTypeModel.create({
- delegationId: delegations.id,
- delegationTypeId: AuthDelegationType.GeneralMandate,
+ afterAll(async () => {
+ await app.cleanUp()
})
- await apiScopeModel.bulkCreate(
- scopeNames.map((name) => ({
- name,
- domainName: domain.name,
- enabled: true,
- description: `${name}: description`,
- displayName: `${name}: display name`,
- })),
- )
-
- // set 2 of 3 scopes to have general mandate delegation type
- await apiScopeDelegationTypeModel.bulkCreate([
- {
- apiScopeName: scopeNames[0],
- delegationType: AuthDelegationType.GeneralMandate,
- },
- {
- apiScopeName: scopeNames[1],
- delegationType: AuthDelegationType.GeneralMandate,
- },
- ])
-
- nationalRegistryApiSpy = jest
- .spyOn(nationalRegistryApi, 'getIndividual')
- .mockImplementation(async (id) => {
- const user = createNationalRegistryUser({
- nationalId: representeeNationalId,
+ describe('GET with general mandate delegation type', () => {
+ const representeeNationalId = getFakeNationalId()
+ let nationalRegistryApiSpy: jest.SpyInstance
+ let nationalRegistryV3ApiSpy: jest.SpyInstance
+ const scopeNames = [
+ 'api-scope/generalMandate1',
+ 'api-scope/generalMandate2',
+ 'api-scope/generalMandate3',
+ ]
+
+ beforeAll(async () => {
+ client.supportedDelegationTypes = [
+ AuthDelegationType.GeneralMandate,
+ AuthDelegationType.LegalGuardian,
+ ]
+ await factory.createClient(client)
+
+ const delegations = await delegationModel.create({
+ id: uuid(),
+ fromDisplayName: 'Test',
+ fromNationalId: representeeNationalId,
+ toNationalId: userNationalId,
+ toName: 'Test',
+ })
+
+ await delegationProviderModel.create({
+ id: AuthDelegationProvider.Custom,
+ name: 'Custom',
+ description: 'Custom',
+ })
+
+ await delegationDelegationTypeModel.create({
+ delegationId: delegations.id,
+ delegationTypeId: AuthDelegationType.GeneralMandate,
})
- return user ?? null
+ await apiScopeModel.bulkCreate(
+ scopeNames.map((name) => ({
+ name,
+ domainName: domain.name,
+ enabled: true,
+ description: `${name}: description`,
+ displayName: `${name}: display name`,
+ })),
+ )
+
+ // set 2 of 3 scopes to have general mandate delegation type
+ await apiScopeDelegationTypeModel.bulkCreate([
+ {
+ apiScopeName: scopeNames[0],
+ delegationType: AuthDelegationType.GeneralMandate,
+ },
+ {
+ apiScopeName: scopeNames[1],
+ delegationType: AuthDelegationType.GeneralMandate,
+ },
+ ])
+
+ nationalRegistryApiSpy = jest
+ .spyOn(nationalRegistryApi, 'getIndividual')
+ .mockImplementation(async (id) => {
+ const user = createNationalRegistryUser({
+ nationalId: representeeNationalId,
+ })
+
+ return user ?? null
+ })
+
+ nationalRegistryV3ApiSpy = jest
+ .spyOn(nationalRegistryV3Api, 'getAllDataIndividual')
+ .mockImplementation(async () => {
+ const user = createNationalRegistryUser({
+ nationalId: representeeNationalId,
+ })
+
+ return { kennitala: user.nationalId, nafn: user.name }
+ })
})
- })
- afterAll(async () => {
- await app.cleanUp()
- nationalRegistryApiSpy.mockClear()
- })
+ afterAll(async () => {
+ await app.cleanUp()
+ nationalRegistryApiSpy.mockClear()
+ nationalRegistryV3ApiSpy.mockClear()
+ })
- it('should return mergedDelegationDTO with the generalMandate', async () => {
- const response = await server.get('/v2/delegations')
+ it('should return mergedDelegationDTO with the generalMandate', async () => {
+ const response = await server.get('/v2/delegations')
- expect(response.status).toEqual(200)
- expect(response.body).toHaveLength(1)
- })
+ expect(response.status).toEqual(200)
+ expect(response.body).toHaveLength(1)
+ })
- it('should get all general mandate scopes', async () => {
- const response = await server.get('/delegations/scopes').query({
- fromNationalId: representeeNationalId,
- delegationType: [AuthDelegationType.GeneralMandate],
- })
+ it('should get all general mandate scopes', async () => {
+ const response = await server.get('/delegations/scopes').query({
+ fromNationalId: representeeNationalId,
+ delegationType: [AuthDelegationType.GeneralMandate],
+ })
- expect(response.status).toEqual(200)
- expect(response.body).toEqual([scopeNames[0], scopeNames[1]])
- })
+ expect(response.status).toEqual(200)
+ expect(response.body).toEqual([scopeNames[0], scopeNames[1]])
+ })
- it('should only return valid general mandates', async () => {
- const newNationalId = getFakeNationalId()
- const newDelegation = await delegationModel.create({
- id: uuid(),
- fromDisplayName: 'Test',
- fromNationalId: newNationalId,
- toNationalId: userNationalId,
- toName: 'Test',
- })
+ it('should only return valid general mandates', async () => {
+ const newNationalId = getFakeNationalId()
+ const newDelegation = await delegationModel.create({
+ id: uuid(),
+ fromDisplayName: 'Test',
+ fromNationalId: newNationalId,
+ toNationalId: userNationalId,
+ toName: 'Test',
+ })
- await delegationDelegationTypeModel.create({
- delegationId: newDelegation.id,
- delegationTypeId: AuthDelegationType.GeneralMandate,
- validTo: addDays(new Date(), -2),
- })
+ await delegationDelegationTypeModel.create({
+ delegationId: newDelegation.id,
+ delegationTypeId: AuthDelegationType.GeneralMandate,
+ validTo: addDays(new Date(), -2),
+ })
- const response = await server.get('/delegations/scopes').query({
- fromNationalId: newNationalId,
- delegationType: [AuthDelegationType.GeneralMandate],
- })
+ const response = await server.get('/delegations/scopes').query({
+ fromNationalId: newNationalId,
+ delegationType: [AuthDelegationType.GeneralMandate],
+ })
- expect(response.status).toEqual(200)
- expect(response.body).toEqual([])
- })
+ expect(response.status).toEqual(200)
+ expect(response.body).toEqual([])
+ })
- it('should return all general mandate scopes and other preset scopes', async () => {
- const newDelegation = await delegationModel.create({
- id: uuid(),
- fromDisplayName: 'Test',
- fromNationalId: representeeNationalId,
- domainName: domain.name,
- toNationalId: userNationalId,
- toName: 'Test',
- })
+ it('should return all general mandate scopes and other preset scopes', async () => {
+ const newDelegation = await delegationModel.create({
+ id: uuid(),
+ fromDisplayName: 'Test',
+ fromNationalId: representeeNationalId,
+ domainName: domain.name,
+ toNationalId: userNationalId,
+ toName: 'Test',
+ })
- await delegationTypeModel.create({
- id: AuthDelegationType.Custom,
- name: 'custom',
- description: 'custom',
- providerId: AuthDelegationProvider.Custom,
- })
+ await delegationTypeModel.create({
+ id: AuthDelegationType.Custom,
+ name: 'custom',
+ description: 'custom',
+ providerId: AuthDelegationProvider.Custom,
+ })
- await delegationScopesModel.create({
- id: uuid(),
- delegationId: newDelegation.id,
- scopeName: scopeNames[2],
- // set valid from as yesterday and valid to as tomorrow
- validFrom: addDays(new Date(), -1),
- validTo: addDays(new Date(), 1),
- })
+ await delegationScopesModel.create({
+ id: uuid(),
+ delegationId: newDelegation.id,
+ scopeName: scopeNames[2],
+ // set valid from as yesterday and valid to as tomorrow
+ validFrom: addDays(new Date(), -1),
+ validTo: addDays(new Date(), 1),
+ })
- await apiScopeDelegationTypeModel.create({
- apiScopeName: scopeNames[2],
- delegationType: AuthDelegationType.LegalGuardian,
- })
+ await apiScopeDelegationTypeModel.create({
+ apiScopeName: scopeNames[2],
+ delegationType: AuthDelegationType.LegalGuardian,
+ })
- const response = await server.get('/delegations/scopes').query({
- fromNationalId: representeeNationalId,
- delegationType: [
- AuthDelegationType.GeneralMandate,
- AuthDelegationType.LegalGuardian,
- ],
- })
+ const response = await server.get('/delegations/scopes').query({
+ fromNationalId: representeeNationalId,
+ delegationType: [
+ AuthDelegationType.GeneralMandate,
+ AuthDelegationType.LegalGuardian,
+ ],
+ })
- expect(response.status).toEqual(200)
- expect(response.body).toEqual(expect.arrayContaining(scopeNames))
- expect(response.body).toHaveLength(scopeNames.length)
+ expect(response.status).toEqual(200)
+ expect(response.body).toEqual(expect.arrayContaining(scopeNames))
+ expect(response.body).toHaveLength(scopeNames.length)
+ })
+ })
})
- })
- })
+ },
+ )
})
diff --git a/apps/services/auth/personal-representative/infra/personal-representative.ts b/apps/services/auth/personal-representative/infra/personal-representative.ts
index 180b75ec9891..42928e86db9f 100644
--- a/apps/services/auth/personal-representative/infra/personal-representative.ts
+++ b/apps/services/auth/personal-representative/infra/personal-representative.ts
@@ -1,5 +1,10 @@
import { json, service, ServiceBuilder } from '../../../../../infra/src/dsl/dsl'
-import { Base, Client, RskProcuring } from '../../../../../infra/src/dsl/xroad'
+import {
+ Base,
+ Client,
+ NationalRegistryAuthB2C,
+ RskProcuring,
+} from '../../../../../infra/src/dsl/xroad'
const REDIS_NODE_CONFIG = {
dev: json([
@@ -62,8 +67,10 @@ export const serviceSetup =
'/k8s/services-auth/IDENTITY_SERVER_CLIENT_SECRET',
SYSLUMENN_USERNAME: '/k8s/services-auth/SYSLUMENN_USERNAME',
SYSLUMENN_PASSWORD: '/k8s/services-auth/SYSLUMENN_PASSWORD',
+ NATIONAL_REGISTRY_B2C_CLIENT_SECRET:
+ '/k8s/services-auth/NATIONAL_REGISTRY_B2C_CLIENT_SECRET',
})
- .xroad(Base, Client, RskProcuring)
+ .xroad(Base, Client, RskProcuring, NationalRegistryAuthB2C)
.ingress({
primary: {
host: {
diff --git a/apps/services/auth/personal-representative/src/app/app.module.ts b/apps/services/auth/personal-representative/src/app/app.module.ts
index e46d8b36eeed..0c89f4dd3a5a 100644
--- a/apps/services/auth/personal-representative/src/app/app.module.ts
+++ b/apps/services/auth/personal-representative/src/app/app.module.ts
@@ -8,8 +8,10 @@ import {
import { AuthModule } from '@island.is/auth-nest-tools'
import { RskRelationshipsClientConfig } from '@island.is/clients-rsk-relationships'
import { NationalRegistryClientConfig } from '@island.is/clients/national-registry-v2'
+import { NationalRegistryV3ClientConfig } from '@island.is/clients/national-registry-v3'
import { CompanyRegistryConfig } from '@island.is/clients/rsk/company-registry'
import { SyslumennClientConfig } from '@island.is/clients/syslumenn'
+import { ZendeskServiceConfig } from '@island.is/clients/zendesk'
import { AuditModule } from '@island.is/nest/audit'
import {
ConfigModule,
@@ -17,12 +19,11 @@ import {
XRoadConfig,
} from '@island.is/nest/config'
import { FeatureFlagConfig } from '@island.is/nest/feature-flags'
-import { ZendeskServiceConfig } from '@island.is/clients/zendesk'
import { environment } from '../environments'
-import { PersonalRepresentativeTypesModule } from './modules/personalRepresentativeTypes/personalRepresentativeTypes.module'
import { AccessLogsModule } from './modules/accessLogs/accessLogs.module'
import { PersonalRepresentativesModule } from './modules/personalRepresentatives/personalRepresentatives.module'
+import { PersonalRepresentativeTypesModule } from './modules/personalRepresentativeTypes/personalRepresentativeTypes.module'
import { RightTypesModule } from './modules/rightTypes/rightTypes.module'
@Module({
@@ -38,6 +39,7 @@ import { RightTypesModule } from './modules/rightTypes/rightTypes.module'
DelegationConfig,
IdsClientConfig,
NationalRegistryClientConfig,
+ NationalRegistryV3ClientConfig,
RskRelationshipsClientConfig,
CompanyRegistryConfig,
XRoadConfig,
diff --git a/apps/services/auth/public-api/infra/auth-public-api.ts b/apps/services/auth/public-api/infra/auth-public-api.ts
index b27663150b40..609588c46271 100644
--- a/apps/services/auth/public-api/infra/auth-public-api.ts
+++ b/apps/services/auth/public-api/infra/auth-public-api.ts
@@ -1,5 +1,10 @@
import { json, service, ServiceBuilder } from '../../../../../infra/src/dsl/dsl'
-import { Base, Client, RskProcuring } from '../../../../../infra/src/dsl/xroad'
+import {
+ Base,
+ Client,
+ NationalRegistryAuthB2C,
+ RskProcuring,
+} from '../../../../../infra/src/dsl/xroad'
const REDIS_NODE_CONFIG = {
dev: json([
@@ -86,8 +91,10 @@ export const serviceSetup = (): ServiceBuilder<'services-auth-public-api'> => {
'/k8s/xroad/client/NATIONAL-REGISTRY/IDENTITYSERVER_SECRET',
SYSLUMENN_USERNAME: '/k8s/services-auth/SYSLUMENN_USERNAME',
SYSLUMENN_PASSWORD: '/k8s/services-auth/SYSLUMENN_PASSWORD',
+ NATIONAL_REGISTRY_B2C_CLIENT_SECRET:
+ '/k8s/services-auth/NATIONAL_REGISTRY_B2C_CLIENT_SECRET',
})
- .xroad(Base, Client, RskProcuring)
+ .xroad(Base, Client, RskProcuring, NationalRegistryAuthB2C)
.ingress({
primary: {
host: {
diff --git a/apps/services/auth/public-api/src/app/app.module.ts b/apps/services/auth/public-api/src/app/app.module.ts
index e3719da7e4b9..7fe7ba43de60 100644
--- a/apps/services/auth/public-api/src/app/app.module.ts
+++ b/apps/services/auth/public-api/src/app/app.module.ts
@@ -9,8 +9,10 @@ import {
import { AuthModule } from '@island.is/auth-nest-tools'
import { RskRelationshipsClientConfig } from '@island.is/clients-rsk-relationships'
import { NationalRegistryClientConfig } from '@island.is/clients/national-registry-v2'
+import { NationalRegistryV3ClientConfig } from '@island.is/clients/national-registry-v3'
import { CompanyRegistryConfig } from '@island.is/clients/rsk/company-registry'
import { SyslumennClientConfig } from '@island.is/clients/syslumenn'
+import { ZendeskServiceConfig } from '@island.is/clients/zendesk'
import { AuditModule } from '@island.is/nest/audit'
import {
ConfigModule,
@@ -23,7 +25,6 @@ import { ProblemModule } from '@island.is/nest/problem'
import { environment } from '../environments'
import { DelegationsModule } from './modules/delegations/delegations.module'
import { PasskeysModule } from './modules/passkeys/passkeys.module'
-import { ZendeskServiceConfig } from '@island.is/clients/zendesk'
@Module({
imports: [
@@ -42,6 +43,7 @@ import { ZendeskServiceConfig } from '@island.is/clients/zendesk'
FeatureFlagConfig,
IdsClientConfig,
NationalRegistryClientConfig,
+ NationalRegistryV3ClientConfig,
RskRelationshipsClientConfig,
CompanyRegistryConfig,
XRoadConfig,
diff --git a/apps/services/auth/public-api/src/app/modules/delegations/actorDelegations.controller.spec.ts b/apps/services/auth/public-api/src/app/modules/delegations/actorDelegations.controller.spec.ts
index d57f16dea37d..a7ad9fb47b1e 100644
--- a/apps/services/auth/public-api/src/app/modules/delegations/actorDelegations.controller.spec.ts
+++ b/apps/services/auth/public-api/src/app/modules/delegations/actorDelegations.controller.spec.ts
@@ -1,8 +1,8 @@
import { getModelToken } from '@nestjs/sequelize'
+import addYears from 'date-fns/addYears'
+import kennitala from 'kennitala'
import times from 'lodash/times'
import request from 'supertest'
-import kennitala from 'kennitala'
-import addYears from 'date-fns/addYears'
import {
ClientDelegationType,
@@ -14,6 +14,7 @@ import {
DelegationScope,
DelegationTypeModel,
MergedDelegationDTO,
+ NationalRegistryV3FeatureService,
PersonalRepresentative,
PersonalRepresentativeDelegationTypeModel,
PersonalRepresentativeRight,
@@ -26,6 +27,7 @@ import {
IndividualDto,
NationalRegistryClientService,
} from '@island.is/clients/national-registry-v2'
+import { NationalRegistryV3ClientService } from '@island.is/clients/national-registry-v3'
import {
createClient,
createDelegation,
@@ -153,923 +155,976 @@ beforeAll(() => {
})
describe('ActorDelegationsController', () => {
- describe('with auth', () => {
- let app: TestApp
- let server: request.SuperTest
- let delegationModel: typeof Delegation
- let delegationDelegationTypeModel: typeof DelegationDelegationType
- let clientDelegationTypeModel: typeof ClientDelegationType
- let nationalRegistryApi: NationalRegistryClientService
-
- beforeAll(async () => {
- // TestApp setup with auth and database
- app = await setupWithAuth({
- user,
- userName,
- nationalRegistryUser,
- client: {
- props: client,
- scopes: Scopes.slice(0, 4).map((s) => s.name),
- },
- })
- server = request(app.getHttpServer())
-
- // Get reference on Delegation and Client models to seed DB
- delegationModel = app.get(getModelToken(Delegation))
- clientDelegationTypeModel = app.get(
- getModelToken(ClientDelegationType),
- )
- delegationDelegationTypeModel = app.get(
- getModelToken(DelegationDelegationType),
- )
- nationalRegistryApi = app.get(NationalRegistryClientService)
- })
-
- beforeEach(async () => {
- return await clientDelegationTypeModel.bulkCreate(
- delegationTypes.map((type) => ({
- clientId: client.clientId,
- delegationType: type,
- })),
- {
- updateOnDuplicate: ['modified'],
- },
- )
- })
-
- afterAll(async () => {
- await app.cleanUp()
- })
-
- afterEach(async () => {
- await delegationModel.destroy({
- where: {},
- cascade: true,
- truncate: true,
- force: true,
- })
- })
-
- describe('GET /actor/delegations', () => {
- let nationalRegistryApiSpy: jest.SpyInstance
- const path = '/v1/actor/delegations'
- const query = '?direction=incoming'
- const deceasedNationalIds = times(3, () => createNationalId('person'))
- const nationalRegistryUsers = [
- nationalRegistryUser,
- ...Object.values(mockDelegations).map((delegation) =>
- createNationalRegistryUser({
- nationalId: delegation.fromNationalId,
- }),
- ),
- ...times(10, () =>
- createNationalRegistryUser({
- name: getFakeName(),
- nationalId: createNationalId('person'),
- }),
- ),
- ]
-
- beforeAll(async () => {
- nationalRegistryApiSpy = jest
- .spyOn(nationalRegistryApi, 'getIndividual')
- .mockImplementation(async (id) => {
- if (deceasedNationalIds.includes(id)) {
- return null
- }
+ describe.each([false, true])(
+ 'national registry v3 featureflag: %s',
+ (featureFlag) => {
+ describe('with auth', () => {
+ let app: TestApp
+ let server: request.SuperTest
+ let delegationModel: typeof Delegation
+ let delegationDelegationTypeModel: typeof DelegationDelegationType
+ let clientDelegationTypeModel: typeof ClientDelegationType
+ let nationalRegistryApi: NationalRegistryClientService
+ let nationalRegistryV3Api: NationalRegistryV3ClientService
- const user = nationalRegistryUsers.find((u) => u?.nationalId === id)
-
- return user ?? null
+ beforeAll(async () => {
+ // TestApp setup with auth and database
+ app = await setupWithAuth({
+ user,
+ userName,
+ nationalRegistryUser,
+ client: {
+ props: client,
+ scopes: Scopes.slice(0, 4).map((s) => s.name),
+ },
})
- })
-
- it('should return only valid delegations', async () => {
- // Arrange
- await createDelegationModels(
- delegationModel,
- Object.values(mockDelegations),
- )
- const expectedModels = await findExpectedMergedDelegationModels(
- delegationModel,
- [
- mockDelegations.incoming.id,
- mockDelegations.incomingBothValidAndNotAllowed.id,
- mockDelegations.incomingWithOtherDomain.id,
- ],
- [Scopes[0].name],
- )
-
- // Act
- const res = await server.get(`${path}${query}`)
-
- // Assert
- expect(res.status).toEqual(200)
- expect(res.body).toHaveLength(3)
- expectMatchingMergedDelegations(
- res.body,
- updateDelegationFromNameToPersonName(
- expectedModels,
- nationalRegistryUsers,
- ),
- )
- })
+ server = request(app.getHttpServer())
- it('should return only delegations with scopes the client has access to', async () => {
- // Arrange
- await createDelegationModels(delegationModel, [
- mockDelegations.incomingWithOtherDomain,
- ])
- const expectedModel = await findExpectedMergedDelegationModels(
- delegationModel,
- mockDelegations.incomingWithOtherDomain.id,
- [Scopes[0].name],
- )
-
- // Act
- const res = await server.get(`${path}${query}`)
-
- // Assert
- expect(res.status).toEqual(200)
- expect(res.body).toHaveLength(1)
- expectMatchingMergedDelegations(
- res.body[0],
- updateDelegationFromNameToPersonName(
- expectedModel,
- nationalRegistryUsers,
- ),
- )
- })
-
- it('should return custom delegations and general mandate when the delegationTypes filter has both types and delegation exists for both', async () => {
- // Arrange
- const delegation = createDelegation({
- fromNationalId: nationalRegistryUser.nationalId,
- toNationalId: user.nationalId,
- scopes: [],
+ // Get reference on Delegation and Client models to seed DB
+ delegationModel = app.get(
+ getModelToken(Delegation),
+ )
+ clientDelegationTypeModel = app.get(
+ getModelToken(ClientDelegationType),
+ )
+ delegationDelegationTypeModel = app.get<
+ typeof DelegationDelegationType
+ >(getModelToken(DelegationDelegationType))
+ nationalRegistryApi = app.get(NationalRegistryClientService)
+ nationalRegistryV3Api = app.get(NationalRegistryV3ClientService)
})
- await delegationModel.create(delegation)
-
- await delegationDelegationTypeModel.create({
- delegationId: delegation.id,
- delegationTypeId: AuthDelegationType.GeneralMandate,
+ beforeEach(async () => {
+ return await clientDelegationTypeModel.bulkCreate(
+ delegationTypes.map((type) => ({
+ clientId: client.clientId,
+ delegationType: type,
+ })),
+ {
+ updateOnDuplicate: ['modified'],
+ },
+ )
})
- await createDelegationModels(delegationModel, [
- mockDelegations.incomingWithOtherDomain,
- ])
-
- // Act
- const res = await server.get(
- `${path}${query}&delegationTypes=${AuthDelegationType.Custom}&delegationTypes=${AuthDelegationType.GeneralMandate}`,
- )
-
- // Assert
- expect(res.status).toEqual(200)
- expect(res.body).toHaveLength(2)
- expect(
- res.body
- .map((d: MergedDelegationDTO) => d.types)
- .flat()
- .sort(),
- ).toEqual(
- [AuthDelegationType.Custom, AuthDelegationType.GeneralMandate].sort(),
- )
- })
-
- it('should return a merged object with both Custom and GeneralMandate types', async () => {
- // Arrange
- const delegation = createDelegation({
- fromNationalId:
- mockDelegations.incomingWithOtherDomain.fromNationalId,
- toNationalId: user.nationalId,
- domainName: null,
- scopes: [],
+ afterAll(async () => {
+ await app.cleanUp()
})
- await delegationModel.create(delegation)
-
- await delegationDelegationTypeModel.create({
- delegationId: delegation.id,
- delegationTypeId: AuthDelegationType.GeneralMandate,
+ afterEach(async () => {
+ await delegationModel.destroy({
+ where: {},
+ cascade: true,
+ truncate: true,
+ force: true,
+ })
})
- await createDelegationModels(delegationModel, [
- mockDelegations.incomingWithOtherDomain,
- ])
-
- // Act
- const res = await server.get(
- `${path}${query}&delegationTypes=${AuthDelegationType.Custom}&delegationTypes=${AuthDelegationType.GeneralMandate}`,
- )
-
- // Assert
- expect(res.status).toEqual(200)
- expect(res.body).toHaveLength(1)
- expect(
- res.body
- .map((d: MergedDelegationDTO) => d.types)
- .flat()
- .sort(),
- ).toEqual(
- [AuthDelegationType.Custom, AuthDelegationType.GeneralMandate].sort(),
- )
- })
-
- it('should return only delegations related to the provided otherUser national id', async () => {
- // Arrange
- await createDelegationModels(delegationModel, [
- mockDelegations.incoming,
- ])
- const expectedModel = await findExpectedMergedDelegationModels(
- delegationModel,
- mockDelegations.incoming.id,
- [Scopes[0].name],
- )
-
- // Act
- const res = await server.get(
- `${path}${query}&delegationTypes=${AuthDelegationType.Custom}&otherUser=${mockDelegations.incoming.fromNationalId}`,
- )
-
- // Assert
- expect(res.status).toEqual(200)
- expect(res.body).toHaveLength(1)
- expectMatchingMergedDelegations(
- res.body[0],
- updateDelegationFromNameToPersonName(
- expectedModel,
- nationalRegistryUsers,
- ),
- )
- })
-
- it('should return only delegations related to the provided otherUser national id without the general mandate since there is none', async () => {
- // Arrange
- await createDelegationModels(delegationModel, [
- mockDelegations.incoming,
- ])
- const expectedModel = await findExpectedMergedDelegationModels(
- delegationModel,
- mockDelegations.incoming.id,
- [Scopes[0].name],
- )
-
- // Act
- const res = await server.get(
- `${path}${query}&delegationTypes=${AuthDelegationType.Custom}&delegationTypes${AuthDelegationType.GeneralMandate}&otherUser=${mockDelegations.incoming.fromNationalId}`,
- )
-
- // Assert
- expect(res.status).toEqual(200)
- expect(res.body).toHaveLength(1)
- expectMatchingMergedDelegations(
- res.body[0],
- updateDelegationFromNameToPersonName(
- expectedModel,
- nationalRegistryUsers,
- ),
- )
- })
-
- it('should return empty array when provided otherUser national id is not related to any delegation', async () => {
- // Arrange
- const unrelatedNationalId = createNationalId('person')
+ describe('GET /actor/delegations', () => {
+ let nationalRegistryApiSpy: jest.SpyInstance
+ let nationalRegistryV3ApiSpy: jest.SpyInstance
+ const path = '/v1/actor/delegations'
+ const query = '?direction=incoming'
+ const deceasedNationalIds = times(3, () => createNationalId('person'))
+ const nationalRegistryUsers = [
+ nationalRegistryUser,
+ ...Object.values(mockDelegations).map((delegation) =>
+ createNationalRegistryUser({
+ nationalId: delegation.fromNationalId,
+ }),
+ ),
+ ...times(10, () =>
+ createNationalRegistryUser({
+ name: getFakeName(),
+ nationalId: createNationalId('person'),
+ }),
+ ),
+ ]
- // Act
- const res = await server.get(
- `${path}${query}&delegationTypes=${AuthDelegationType.Custom}&otherUser=${unrelatedNationalId}`,
- )
-
- // Assert
- expect(res.status).toEqual(200)
- expect(res.body).toHaveLength(0)
- })
-
- it('should return custom delegations and not deceased delegations, when the delegationTypes filter is custom type', async () => {
- // Arrange
- await createDelegationModels(delegationModel, [
- mockDelegations.incoming,
- createDelegation({
- fromNationalId: deceasedNationalIds[0],
- toNationalId: user.nationalId,
- scopes: [Scopes[0].name],
- today,
- }),
- createDelegation({
- fromNationalId: deceasedNationalIds[1],
- toNationalId: user.nationalId,
- scopes: [Scopes[0].name],
- today,
- }),
- ])
-
- // We expect the first model to be returned, but not the second or third since they are tied to a deceased person
- const expectedModel = await findExpectedMergedDelegationModels(
- delegationModel,
- mockDelegations.incoming.id,
- [Scopes[0].name],
- )
+ beforeAll(async () => {
+ nationalRegistryApiSpy = jest
+ .spyOn(nationalRegistryApi, 'getIndividual')
+ .mockImplementation(async (id) => {
+ if (deceasedNationalIds.includes(id)) {
+ return null
+ }
+
+ const user = nationalRegistryUsers.find(
+ (u) => u?.nationalId === id,
+ )
+
+ return user ?? null
+ })
+ const nationalRegistryV3FeatureService = app.get(
+ NationalRegistryV3FeatureService,
+ )
+ jest
+ .spyOn(nationalRegistryV3FeatureService, 'getValue')
+ .mockImplementation(async () => featureFlag)
+ nationalRegistryV3ApiSpy = jest
+ .spyOn(nationalRegistryV3Api, 'getAllDataIndividual')
+ .mockImplementation(async (id) => {
+ if (deceasedNationalIds.includes(id)) {
+ return {
+ kennitala: id,
+ afdrif: 'LÉST',
+ }
+ }
+
+ const user = nationalRegistryUsers.find(
+ (u) => u?.nationalId === id,
+ )
+
+ return user
+ ? {
+ kennitala: id,
+ nafn: user?.name,
+ afdrif: null,
+ }
+ : null
+ })
+ })
- // Act
- const res = await server.get(
- `${path}${query}&delegationTypes=${AuthDelegationType.Custom}`,
- )
+ it('should return only valid delegations', async () => {
+ // Arrange
+ await createDelegationModels(
+ delegationModel,
+ Object.values(mockDelegations),
+ )
+ const expectedModels = await findExpectedMergedDelegationModels(
+ delegationModel,
+ [
+ mockDelegations.incoming.id,
+ mockDelegations.incomingBothValidAndNotAllowed.id,
+ mockDelegations.incomingWithOtherDomain.id,
+ ],
+ [Scopes[0].name],
+ )
+
+ // Act
+ const res = await server.get(`${path}${query}`)
+
+ // Assert
+ expect(res.status).toEqual(200)
+ expect(res.body).toHaveLength(3)
+ expectMatchingMergedDelegations(
+ res.body,
+ updateDelegationFromNameToPersonName(
+ expectedModels,
+ nationalRegistryUsers,
+ ),
+ )
+ })
- // Assert
- expect(res.status).toEqual(200)
- expect(res.body).toHaveLength(1)
- expectMatchingMergedDelegations(
- res.body[0],
- updateDelegationFromNameToPersonName(
- expectedModel,
- nationalRegistryUsers,
- ),
- )
-
- // Verify
- const expectedModifiedModels = await delegationModel.findAll({
- where: {
- toNationalId: user.nationalId,
- },
- include: [
- {
- model: DelegationScope,
- as: 'delegationScopes',
- },
- ],
- })
+ it('should return only delegations with scopes the client has access to', async () => {
+ // Arrange
+ await createDelegationModels(delegationModel, [
+ mockDelegations.incomingWithOtherDomain,
+ ])
+ const expectedModel = await findExpectedMergedDelegationModels(
+ delegationModel,
+ mockDelegations.incomingWithOtherDomain.id,
+ [Scopes[0].name],
+ )
+
+ // Act
+ const res = await server.get(`${path}${query}`)
+
+ // Assert
+ expect(res.status).toEqual(200)
+ expect(res.body).toHaveLength(1)
+ expectMatchingMergedDelegations(
+ res.body[0],
+ updateDelegationFromNameToPersonName(
+ expectedModel,
+ nationalRegistryUsers,
+ ),
+ )
+ })
- expect(expectedModifiedModels.length).toEqual(1)
- })
+ it('should return custom delegations and general mandate when the delegationTypes filter has both types and delegation exists for both', async () => {
+ // Arrange
+ const delegation = createDelegation({
+ fromNationalId: nationalRegistryUser.nationalId,
+ toNationalId: user.nationalId,
+ scopes: [],
+ })
- it('should not mix up companies and individuals when processing deceased delegations [BUG]', async () => {
- // Arrange
- const incomingCompany = createDelegation({
- fromNationalId: createNationalId('company'),
- toNationalId: user.nationalId,
- scopes: [Scopes[0].name],
- today,
- })
- await createDelegationModels(delegationModel, [
- // The order of these is important to trigger the previous bug.
- incomingCompany,
- mockDelegations.incoming,
- ])
-
- // We expect both models to be returned.
- const expectedModels = await findExpectedMergedDelegationModels(
- delegationModel,
- [mockDelegations.incoming.id, incomingCompany.id],
- [Scopes[0].name],
- )
+ await delegationModel.create(delegation)
- // Act
- const res = await server.get(
- `${path}${query}&delegationTypes=${AuthDelegationType.Custom}`,
- )
+ await delegationDelegationTypeModel.create({
+ delegationId: delegation.id,
+ delegationTypeId: AuthDelegationType.GeneralMandate,
+ })
- // Assert
- expect(res.status).toEqual(200)
- expect(res.body).toHaveLength(2)
- expectMatchingMergedDelegations(
- res.body,
- updateDelegationFromNameToPersonName(
- expectedModels,
- nationalRegistryUsers,
- ),
- )
- })
+ await createDelegationModels(delegationModel, [
+ mockDelegations.incomingWithOtherDomain,
+ ])
- it('should return delegations which only has scopes with special scope rules [BUG]', async () => {
- // Arrange
- const specialDelegation = createDelegation({
- fromNationalId: nationalRegistryUser.nationalId,
- toNationalId: user.nationalId,
- scopes: [Scopes[3].name],
- today,
- })
- await createDelegationModels(delegationModel, [specialDelegation])
+ // Act
+ const res = await server.get(
+ `${path}${query}&delegationTypes=${AuthDelegationType.Custom}&delegationTypes=${AuthDelegationType.GeneralMandate}`,
+ )
- // We expect the delegation to be returned.
- const expectedModels = await findExpectedMergedDelegationModels(
- delegationModel,
- [specialDelegation.id],
- )
+ // Assert
+ expect(res.status).toEqual(200)
+ expect(res.body).toHaveLength(2)
+ expect(
+ res.body
+ .map((d: MergedDelegationDTO) => d.types)
+ .flat()
+ .sort(),
+ ).toEqual(
+ [
+ AuthDelegationType.Custom,
+ AuthDelegationType.GeneralMandate,
+ ].sort(),
+ )
+ })
- // Act
- const res = await server.get(
- `${path}${query}&delegationTypes=${AuthDelegationType.Custom}`,
- )
+ it('should return a merged object with both Custom and GeneralMandate types', async () => {
+ // Arrange
+ const delegation = createDelegation({
+ fromNationalId:
+ mockDelegations.incomingWithOtherDomain.fromNationalId,
+ toNationalId: user.nationalId,
+ domainName: null,
+ scopes: [],
+ })
- // Assert
- expect(res.status).toEqual(200)
- expectMatchingMergedDelegations(
- res.body,
- updateDelegationFromNameToPersonName(
- expectedModels,
- nationalRegistryUsers,
- ),
- )
- })
+ await delegationModel.create(delegation)
- it('should return delegations when the delegationTypes filter is empty', async () => {
- // Arrange
- await createDelegationModels(delegationModel, [
- mockDelegations.incomingWithOtherDomain,
- ])
- const expectedModel = await findExpectedMergedDelegationModels(
- delegationModel,
- mockDelegations.incomingWithOtherDomain.id,
- [Scopes[0].name],
- )
+ await delegationDelegationTypeModel.create({
+ delegationId: delegation.id,
+ delegationTypeId: AuthDelegationType.GeneralMandate,
+ })
- // Act
- const res = await server.get(`${path}${query}&delegationTypes=`)
+ await createDelegationModels(delegationModel, [
+ mockDelegations.incomingWithOtherDomain,
+ ])
- // Assert
- expect(res.status).toEqual(200)
- expect(res.body).toHaveLength(1)
- expectMatchingMergedDelegations(
- res.body[0],
- updateDelegationFromNameToPersonName(
- expectedModel,
- nationalRegistryUsers,
- ),
- )
- })
+ // Act
+ const res = await server.get(
+ `${path}${query}&delegationTypes=${AuthDelegationType.Custom}&delegationTypes=${AuthDelegationType.GeneralMandate}`,
+ )
- it('should not return custom delegations when the delegationTypes filter does not include custom type', async () => {
- // Arrange
- await createDelegationModels(delegationModel, [
- mockDelegations.incomingWithOtherDomain,
- ])
-
- // Act
- const res = await server.get(
- `${path}${query}&delegationTypes=${AuthDelegationType.ProcurationHolder}`,
- )
+ // Assert
+ expect(res.status).toEqual(200)
+ expect(res.body).toHaveLength(1)
+ expect(
+ res.body
+ .map((d: MergedDelegationDTO) => d.types)
+ .flat()
+ .sort(),
+ ).toEqual(
+ [
+ AuthDelegationType.Custom,
+ AuthDelegationType.GeneralMandate,
+ ].sort(),
+ )
+ })
- // Assert
- expect(res.status).toEqual(200)
- expect(res.body).toHaveLength(0)
- })
+ it('should return only delegations related to the provided otherUser national id', async () => {
+ // Arrange
+ await createDelegationModels(delegationModel, [
+ mockDelegations.incoming,
+ ])
+ const expectedModel = await findExpectedMergedDelegationModels(
+ delegationModel,
+ mockDelegations.incoming.id,
+ [Scopes[0].name],
+ )
+
+ // Act
+ const res = await server.get(
+ `${path}${query}&delegationTypes=${AuthDelegationType.Custom}&otherUser=${mockDelegations.incoming.fromNationalId}`,
+ )
+
+ // Assert
+ expect(res.status).toEqual(200)
+ expect(res.body).toHaveLength(1)
+ expectMatchingMergedDelegations(
+ res.body[0],
+ updateDelegationFromNameToPersonName(
+ expectedModel,
+ nationalRegistryUsers,
+ ),
+ )
+ })
- it('should return no delegation when the client does not have access to any scope', async () => {
- // Arrange
- await createDelegationModels(delegationModel, [
- mockDelegations.incomingOnlyOtherDomain,
- ])
+ it('should return only delegations related to the provided otherUser national id without the general mandate since there is none', async () => {
+ // Arrange
+ await createDelegationModels(delegationModel, [
+ mockDelegations.incoming,
+ ])
+ const expectedModel = await findExpectedMergedDelegationModels(
+ delegationModel,
+ mockDelegations.incoming.id,
+ [Scopes[0].name],
+ )
+
+ // Act
+ const res = await server.get(
+ `${path}${query}&delegationTypes=${AuthDelegationType.Custom}&delegationTypes${AuthDelegationType.GeneralMandate}&otherUser=${mockDelegations.incoming.fromNationalId}`,
+ )
+
+ // Assert
+ expect(res.status).toEqual(200)
+ expect(res.body).toHaveLength(1)
+ expectMatchingMergedDelegations(
+ res.body[0],
+ updateDelegationFromNameToPersonName(
+ expectedModel,
+ nationalRegistryUsers,
+ ),
+ )
+ })
- // Act
- const res = await server.get(`${path}${query}`)
+ it('should return empty array when provided otherUser national id is not related to any delegation', async () => {
+ // Arrange
+ const unrelatedNationalId = createNationalId('person')
- // Assert
- expect(res.status).toEqual(200)
- expect(res.body).toHaveLength(0)
- })
+ // Act
+ const res = await server.get(
+ `${path}${query}&delegationTypes=${AuthDelegationType.Custom}&otherUser=${unrelatedNationalId}`,
+ )
- it('should return 400 BadRequest if required query paramter is missing', async () => {
- // Act
- const res = await server.get(path)
+ // Assert
+ expect(res.status).toEqual(200)
+ expect(res.body).toHaveLength(0)
+ })
- // Assert
- expect(res.status).toEqual(400)
- expect(res.body).toMatchObject({
- status: 400,
- type: 'https://httpstatuses.org/400',
- title: 'Bad Request',
- detail:
- "'direction' can only be set to incoming for the /actor alias",
- })
- })
+ it('should return custom delegations and not deceased delegations, when the delegationTypes filter is custom type', async () => {
+ // Arrange
+ await createDelegationModels(delegationModel, [
+ mockDelegations.incoming,
+ createDelegation({
+ fromNationalId: deceasedNationalIds[0],
+ toNationalId: user.nationalId,
+ scopes: [Scopes[0].name],
+ today,
+ }),
+ createDelegation({
+ fromNationalId: deceasedNationalIds[1],
+ toNationalId: user.nationalId,
+ scopes: [Scopes[0].name],
+ today,
+ }),
+ ])
+
+ // We expect the first model to be returned, but not the second or third since they are tied to a deceased person
+ const expectedModel = await findExpectedMergedDelegationModels(
+ delegationModel,
+ mockDelegations.incoming.id,
+ [Scopes[0].name],
+ )
+
+ // Act
+ const res = await server.get(
+ `${path}${query}&delegationTypes=${AuthDelegationType.Custom}`,
+ )
+
+ // Assert
+ expect(res.status).toEqual(200)
+ expect(res.body).toHaveLength(1)
+ expectMatchingMergedDelegations(
+ res.body[0],
+ updateDelegationFromNameToPersonName(
+ expectedModel,
+ nationalRegistryUsers,
+ ),
+ )
+
+ // Verify
+ const expectedModifiedModels = await delegationModel.findAll({
+ where: {
+ toNationalId: user.nationalId,
+ },
+ include: [
+ {
+ model: DelegationScope,
+ as: 'delegationScopes',
+ },
+ ],
+ })
- it('should not return delegation with no scope longer allowed for delegation', async () => {
- // Arrange
- await createDelegationModels(delegationModel, [
- mockDelegations.incomingWithNoAllowed,
- ])
+ expect(expectedModifiedModels.length).toEqual(1)
+ })
- // Act
- const res = await server.get(`${path}${query}`)
+ it('should not mix up companies and individuals when processing deceased delegations [BUG]', async () => {
+ // Arrange
+ const incomingCompany = createDelegation({
+ fromNationalId: createNationalId('company'),
+ toNationalId: user.nationalId,
+ scopes: [Scopes[0].name],
+ today,
+ })
+ await createDelegationModels(delegationModel, [
+ // The order of these is important to trigger the previous bug.
+ incomingCompany,
+ mockDelegations.incoming,
+ ])
+
+ // We expect both models to be returned.
+ const expectedModels = await findExpectedMergedDelegationModels(
+ delegationModel,
+ [mockDelegations.incoming.id, incomingCompany.id],
+ [Scopes[0].name],
+ )
+
+ // Act
+ const res = await server.get(
+ `${path}${query}&delegationTypes=${AuthDelegationType.Custom}`,
+ )
+
+ // Assert
+ expect(res.status).toEqual(200)
+ expect(res.body).toHaveLength(2)
+ expectMatchingMergedDelegations(
+ res.body,
+ updateDelegationFromNameToPersonName(
+ expectedModels,
+ nationalRegistryUsers,
+ ),
+ )
+ })
- // Assert
- expect(res.status).toEqual(200)
- expect(res.body).toHaveLength(0)
- })
+ it('should return delegations which only has scopes with special scope rules [BUG]', async () => {
+ // Arrange
+ const specialDelegation = createDelegation({
+ fromNationalId: nationalRegistryUser.nationalId,
+ toNationalId: user.nationalId,
+ scopes: [Scopes[3].name],
+ today,
+ })
+ await createDelegationModels(delegationModel, [specialDelegation])
+
+ // We expect the delegation to be returned.
+ const expectedModels = await findExpectedMergedDelegationModels(
+ delegationModel,
+ [specialDelegation.id],
+ )
+
+ // Act
+ const res = await server.get(
+ `${path}${query}&delegationTypes=${AuthDelegationType.Custom}`,
+ )
+
+ // Assert
+ expect(res.status).toEqual(200)
+ expectMatchingMergedDelegations(
+ res.body,
+ updateDelegationFromNameToPersonName(
+ expectedModels,
+ nationalRegistryUsers,
+ ),
+ )
+ })
- it('should not return delegation when client does not support custom delegations', async () => {
- // Arrange
- await createDelegationModels(delegationModel, [
- mockDelegations.incoming,
- ])
- await clientDelegationTypeModel.destroy({
- where: {
- clientId: client.clientId,
- delegationType: AuthDelegationType.Custom,
- },
- })
+ it('should return delegations when the delegationTypes filter is empty', async () => {
+ // Arrange
+ await createDelegationModels(delegationModel, [
+ mockDelegations.incomingWithOtherDomain,
+ ])
+ const expectedModel = await findExpectedMergedDelegationModels(
+ delegationModel,
+ mockDelegations.incomingWithOtherDomain.id,
+ [Scopes[0].name],
+ )
+
+ // Act
+ const res = await server.get(`${path}${query}&delegationTypes=`)
+
+ // Assert
+ expect(res.status).toEqual(200)
+ expect(res.body).toHaveLength(1)
+ expectMatchingMergedDelegations(
+ res.body[0],
+ updateDelegationFromNameToPersonName(
+ expectedModel,
+ nationalRegistryUsers,
+ ),
+ )
+ })
- // Act
- const res = await server.get(`${path}${query}`)
+ it('should not return custom delegations when the delegationTypes filter does not include custom type', async () => {
+ // Arrange
+ await createDelegationModels(delegationModel, [
+ mockDelegations.incomingWithOtherDomain,
+ ])
- // Assert
- expect(res.status).toEqual(200)
- expect(res.body).toHaveLength(0)
- })
+ // Act
+ const res = await server.get(
+ `${path}${query}&delegationTypes=${AuthDelegationType.ProcurationHolder}`,
+ )
- describe('with legal guardian delegations', () => {
- let getForsja: jest.SpyInstance
- let clientInstance: any
-
- const mockForKt = (kt: string): void => {
- jest.spyOn(kennitala, 'info').mockReturnValue({
- kt,
- age: 16,
- birthday: addYears(Date.now(), -15),
- birthdayReadable: '',
- type: 'person',
- valid: true,
+ // Assert
+ expect(res.status).toEqual(200)
+ expect(res.body).toHaveLength(0)
})
- jest.spyOn(clientInstance, 'getIndividual').mockResolvedValueOnce({
- nationalId: kt,
- name: nationalRegistryUser.name,
- } as IndividualDto)
-
- jest
- .spyOn(clientInstance, 'getCustodyChildren')
- .mockResolvedValueOnce([kt])
- }
-
- beforeEach(() => {
- clientInstance = app.get(NationalRegistryClientService)
- getForsja = jest
- .spyOn(clientInstance, 'getCustodyChildren')
- .mockResolvedValue([nationalRegistryUser.nationalId])
- })
-
- afterAll(() => {
- getForsja.mockRestore()
- })
+ it('should return no delegation when the client does not have access to any scope', async () => {
+ // Arrange
+ await createDelegationModels(delegationModel, [
+ mockDelegations.incomingOnlyOtherDomain,
+ ])
- it('should return delegations', async () => {
- const kt = '1111089030'
-
- // Arrange
- mockForKt(kt)
-
- const expectedDelegation = DelegationDTOMapper.toMergedDelegationDTO({
- fromName: nationalRegistryUser.name,
- fromNationalId: kt,
- provider: AuthDelegationProvider.NationalRegistry,
- toNationalId: user.nationalId,
- type: [
- AuthDelegationType.LegalGuardian,
- AuthDelegationType.LegalGuardianMinor,
- ],
- } as Omit & { type: AuthDelegationType | AuthDelegationType[] })
-
- // Act
- const res = await server.get(`${path}${query}`)
-
- // Assert
- expect(res.status).toEqual(200)
- expect(res.body).toHaveLength(1)
- expect(res.body[0]).toEqual(expectedDelegation)
- })
+ // Act
+ const res = await server.get(`${path}${query}`)
- it('should not return delegations when client does not support legal guardian delegations', async () => {
- await clientDelegationTypeModel.destroy({
- where: {
- clientId: client.clientId,
- delegationType: AuthDelegationType.LegalGuardian,
- },
+ // Assert
+ expect(res.status).toEqual(200)
+ expect(res.body).toHaveLength(0)
})
- // Act
- const res = await server.get(`${path}${query}`)
-
- // Assert
- expect(res.status).toEqual(200)
- expect(res.body).toHaveLength(0)
- })
-
- it('should return a legal guardian delegation since the type is included in the delegationTypes filter', async () => {
- // Act
- const res = await server.get(
- `${path}${query}&delegationTypes=${AuthDelegationType.Custom}&delegationTypes=${AuthDelegationType.LegalGuardian}`,
- )
+ it('should return 400 BadRequest if required query paramter is missing', async () => {
+ // Act
+ const res = await server.get(path)
+
+ // Assert
+ expect(res.status).toEqual(400)
+ expect(res.body).toMatchObject({
+ status: 400,
+ type: 'https://httpstatuses.org/400',
+ title: 'Bad Request',
+ detail:
+ "'direction' can only be set to incoming for the /actor alias",
+ })
+ })
- // Assert
- expect(res.status).toEqual(200)
- expect(res.body).toHaveLength(1)
- expect(res.body[0].types[0]).toEqual(AuthDelegationType.LegalGuardian)
- })
+ it('should not return delegation with no scope longer allowed for delegation', async () => {
+ // Arrange
+ await createDelegationModels(delegationModel, [
+ mockDelegations.incomingWithNoAllowed,
+ ])
- it('should not return a legal guardian delegation since the type is not included in the delegationTypes filter', async () => {
- // Act
- const res = await server.get(
- `${path}${query}&delegationTypes=${AuthDelegationType.Custom}`,
- )
+ // Act
+ const res = await server.get(`${path}${query}`)
- // Assert
- expect(res.status).toEqual(200)
- expect(res.body).toHaveLength(0)
- })
- })
+ // Assert
+ expect(res.status).toEqual(200)
+ expect(res.body).toHaveLength(0)
+ })
- describe('with procuring delegations', () => {
- let getIndividualRelationships: jest.SpyInstance
- beforeAll(() => {
- const client = app.get(RskRelationshipsClient)
- getIndividualRelationships = jest
- .spyOn(client, 'getIndividualRelationships')
- .mockResolvedValue({
- name: nationalRegistryUser.name,
- nationalId: nationalRegistryUser.nationalId,
- relationships: [
- {
- nationalId: nationalRegistryUser.nationalId,
- name: nationalRegistryUser.name,
- },
- ],
+ it('should not return delegation when client does not support custom delegations', async () => {
+ // Arrange
+ await createDelegationModels(delegationModel, [
+ mockDelegations.incoming,
+ ])
+ await clientDelegationTypeModel.destroy({
+ where: {
+ clientId: client.clientId,
+ delegationType: AuthDelegationType.Custom,
+ },
})
- })
- afterAll(() => {
- getIndividualRelationships.mockRestore()
- })
+ // Act
+ const res = await server.get(`${path}${query}`)
- it('should return delegations', async () => {
- // Arrange
- const expectedDelegation = {
- fromName: nationalRegistryUser.name,
- fromNationalId: nationalRegistryUser.nationalId,
- provider: 'fyrirtaekjaskra',
- toNationalId: user.nationalId,
- type: 'ProcurationHolder',
- } as DelegationDTO
- // Act
- const res = await server.get(`${path}${query}`)
-
- // Assert
- expect(res.status).toEqual(200)
- expect(res.body).toHaveLength(1)
- expect(res.body[0]).toEqual(
- DelegationDTOMapper.toMergedDelegationDTO(expectedDelegation),
- )
- })
-
- it('should not return delegations when client does not support procuring holder delegations', async () => {
- // Arrange
- await clientDelegationTypeModel.destroy({
- where: {
- clientId: client.clientId,
- delegationType: AuthDelegationType.ProcurationHolder,
- },
+ // Assert
+ expect(res.status).toEqual(200)
+ expect(res.body).toHaveLength(0)
})
- // Act
- const res = await server.get(`${path}${query}`)
+ describe('with legal guardian delegations', () => {
+ let getForsja: jest.SpyInstance
+ let clientInstance: any
+
+ const mockForKt = (kt: string): void => {
+ jest.spyOn(kennitala, 'info').mockReturnValue({
+ kt,
+ age: 16,
+ birthday: addYears(Date.now(), -15),
+ birthdayReadable: '',
+ type: 'person',
+ valid: true,
+ })
+
+ jest
+ .spyOn(clientInstance, 'getIndividual')
+ .mockResolvedValueOnce({
+ nationalId: kt,
+ name: nationalRegistryUser.name,
+ } as IndividualDto)
- // Assert
- expect(res.status).toEqual(200)
- expect(res.body).toHaveLength(0)
- })
+ jest
+ .spyOn(clientInstance, 'getCustodyChildren')
+ .mockResolvedValueOnce([kt])
+ }
- it('should return a procuring holder delegation since the type is included in the delegationTypes filter', async () => {
- // Act
- const res = await server.get(
- `${path}${query}&delegationTypes=${AuthDelegationType.Custom}&delegationTypes=${AuthDelegationType.ProcurationHolder}&delegationTypes=${AuthDelegationType.PersonalRepresentative}`,
- )
+ beforeEach(() => {
+ clientInstance = app.get(NationalRegistryClientService)
+ getForsja = jest
+ .spyOn(clientInstance, 'getCustodyChildren')
+ .mockResolvedValue([nationalRegistryUser.nationalId])
+ })
- // Assert
- expect(res.status).toEqual(200)
- expect(res.body).toHaveLength(1)
- expect(res.body[0].types[0]).toEqual(
- AuthDelegationType.ProcurationHolder,
- )
- })
+ afterAll(() => {
+ getForsja.mockRestore()
+ })
- it('should not return a procuring holder delegation since the type is not included in the delegationTypes filter', async () => {
- // Act
- const res = await server.get(
- `${path}${query}&delegationTypes=${AuthDelegationType.Custom}&delegationTypes=${AuthDelegationType.LegalGuardian}&delegationTypes=${AuthDelegationType.PersonalRepresentative}`,
- )
+ it('should return delegations', async () => {
+ const kt = '1111089030'
+
+ // Arrange
+ mockForKt(kt)
+
+ const expectedDelegation =
+ DelegationDTOMapper.toMergedDelegationDTO({
+ fromName: nationalRegistryUser.name,
+ fromNationalId: kt,
+ provider: AuthDelegationProvider.NationalRegistry,
+ toNationalId: user.nationalId,
+ type: [
+ AuthDelegationType.LegalGuardian,
+ AuthDelegationType.LegalGuardianMinor,
+ ],
+ } as Omit & { type: AuthDelegationType | AuthDelegationType[] })
+
+ // Act
+ const res = await server.get(`${path}${query}`)
+
+ // Assert
+ expect(res.status).toEqual(200)
+ expect(res.body).toHaveLength(1)
+ expect(res.body[0]).toEqual(expectedDelegation)
+ })
- // Assert
- expect(res.status).toEqual(200)
- expect(res.body).toHaveLength(0)
- })
- })
+ it('should not return delegations when client does not support legal guardian delegations', async () => {
+ await clientDelegationTypeModel.destroy({
+ where: {
+ clientId: client.clientId,
+ delegationType: AuthDelegationType.LegalGuardian,
+ },
+ })
- describe('when user is a personal representative with one representee', () => {
- let prModel: typeof PersonalRepresentative
- let prRightsModel: typeof PersonalRepresentativeRight
- let prRightTypeModel: typeof PersonalRepresentativeRightType
- let prTypeModel: typeof PersonalRepresentativeType
- let delegationTypeModel: typeof DelegationTypeModel
- let delegationProviderModel: typeof DelegationProviderModel
- let prDelegationTypeModel: typeof PersonalRepresentativeDelegationTypeModel
+ // Act
+ const res = await server.get(`${path}${query}`)
- beforeAll(async () => {
- prTypeModel = app.get(
- 'PersonalRepresentativeTypeRepository',
- )
- prModel = app.get(
- 'PersonalRepresentativeRepository',
- )
- prRightTypeModel = app.get(
- 'PersonalRepresentativeRightTypeRepository',
- )
- prRightsModel = app.get(
- 'PersonalRepresentativeRightRepository',
- )
- delegationTypeModel = app.get(
- getModelToken(DelegationTypeModel),
- )
- delegationProviderModel = app.get(
- getModelToken(DelegationProviderModel),
- )
- prDelegationTypeModel = app.get<
- typeof PersonalRepresentativeDelegationTypeModel
- >(getModelToken(PersonalRepresentativeDelegationTypeModel))
-
- const prType = await prTypeModel.create({
- code: 'prTypeCode',
- name: 'prTypeName',
- description: 'prTypeDescription',
- })
+ // Assert
+ expect(res.status).toEqual(200)
+ expect(res.body).toHaveLength(0)
+ })
- const pr = await prModel.create({
- nationalIdPersonalRepresentative: user.nationalId,
- nationalIdRepresentedPerson: nationalRegistryUser.nationalId,
- personalRepresentativeTypeCode: prType.code,
- contractId: '1',
- externalUserId: '1',
- })
+ it('should return a legal guardian delegation since the type is included in the delegationTypes filter', async () => {
+ // Act
+ const res = await server.get(
+ `${path}${query}&delegationTypes=${AuthDelegationType.Custom}&delegationTypes=${AuthDelegationType.LegalGuardian}`,
+ )
+
+ // Assert
+ expect(res.status).toEqual(200)
+ expect(res.body).toHaveLength(1)
+ expect(res.body[0].types[0]).toEqual(
+ AuthDelegationType.LegalGuardian,
+ )
+ })
- const dt = await delegationTypeModel.create({
- id: getPersonalRepresentativeDelegationType('prRightType'),
- providerId: AuthDelegationProvider.PersonalRepresentativeRegistry,
- name: `Personal Representative: prRightType`,
- description: `Personal representative delegation type for right type prRightType`,
- })
+ it('should not return a legal guardian delegation since the type is not included in the delegationTypes filter', async () => {
+ // Act
+ const res = await server.get(
+ `${path}${query}&delegationTypes=${AuthDelegationType.Custom}`,
+ )
- const prRightType = await prRightTypeModel.create({
- code: 'prRightType',
- description: 'prRightTypeDescription',
+ // Assert
+ expect(res.status).toEqual(200)
+ expect(res.body).toHaveLength(0)
+ })
})
- const prr = await prRightsModel.create({
- rightTypeCode: prRightType.code,
- personalRepresentativeId: pr.id,
- })
+ describe('with procuring delegations', () => {
+ let getIndividualRelationships: jest.SpyInstance
+ beforeAll(() => {
+ const client = app.get(RskRelationshipsClient)
+ getIndividualRelationships = jest
+ .spyOn(client, 'getIndividualRelationships')
+ .mockResolvedValue({
+ name: nationalRegistryUser.name,
+ nationalId: nationalRegistryUser.nationalId,
+ relationships: [
+ {
+ nationalId: nationalRegistryUser.nationalId,
+ name: nationalRegistryUser.name,
+ },
+ ],
+ })
+ })
- await prDelegationTypeModel.create({
- id: prr.id,
- delegationTypeId: dt.id,
- personalRepresentativeId: pr.id,
- })
- })
+ afterAll(() => {
+ getIndividualRelationships.mockRestore()
+ })
- describe('when fetched', () => {
- let response: request.Response
- let body: MergedDelegationDTO[]
+ it('should return delegations', async () => {
+ // Arrange
+ const expectedDelegation = {
+ fromName: nationalRegistryUser.name,
+ fromNationalId: nationalRegistryUser.nationalId,
+ provider: 'fyrirtaekjaskra',
+ toNationalId: user.nationalId,
+ type: 'ProcurationHolder',
+ } as DelegationDTO
+ // Act
+ const res = await server.get(`${path}${query}`)
+
+ // Assert
+ expect(res.status).toEqual(200)
+ expect(res.body).toHaveLength(1)
+ expect(res.body[0]).toEqual(
+ DelegationDTOMapper.toMergedDelegationDTO(expectedDelegation),
+ )
+ })
- beforeAll(async () => {
- response = await server.get(`${path}${query}`)
- body = response.body
- })
+ it('should not return delegations when client does not support procuring holder delegations', async () => {
+ // Arrange
+ await clientDelegationTypeModel.destroy({
+ where: {
+ clientId: client.clientId,
+ delegationType: AuthDelegationType.ProcurationHolder,
+ },
+ })
- it('should have a an OK return status', () => {
- expect(response.status).toEqual(200)
- })
+ // Act
+ const res = await server.get(`${path}${query}`)
- it('should return a single entity', () => {
- expect(body.length).toEqual(1)
- })
+ // Assert
+ expect(res.status).toEqual(200)
+ expect(res.body).toHaveLength(0)
+ })
- it('should have the nationalId of the user as the representer', () => {
- expect(
- body.some((d) => d.toNationalId === user.nationalId),
- ).toBeTruthy()
- })
+ it('should return a procuring holder delegation since the type is included in the delegationTypes filter', async () => {
+ // Act
+ const res = await server.get(
+ `${path}${query}&delegationTypes=${AuthDelegationType.Custom}&delegationTypes=${AuthDelegationType.ProcurationHolder}&delegationTypes=${AuthDelegationType.PersonalRepresentative}`,
+ )
+
+ // Assert
+ expect(res.status).toEqual(200)
+ expect(res.body).toHaveLength(1)
+ expect(res.body[0].types[0]).toEqual(
+ AuthDelegationType.ProcurationHolder,
+ )
+ })
- it('should have the nationalId of the correct representee', () => {
- expect(
- body.some(
- (d) => d.fromNationalId === nationalRegistryUser.nationalId,
- ),
- ).toBeTruthy()
- })
+ it('should not return a procuring holder delegation since the type is not included in the delegationTypes filter', async () => {
+ // Act
+ const res = await server.get(
+ `${path}${query}&delegationTypes=${AuthDelegationType.Custom}&delegationTypes=${AuthDelegationType.LegalGuardian}&delegationTypes=${AuthDelegationType.PersonalRepresentative}`,
+ )
- it('should have the name of the correct representee', () => {
- expect(
- body.some((d) => d.fromName === nationalRegistryUser.name),
- ).toBeTruthy()
+ // Assert
+ expect(res.status).toEqual(200)
+ expect(res.body).toHaveLength(0)
+ })
})
- it('should have the delegation type claim of PersonalRepresentative', () => {
- expect(
- body.some(
- (d) => d.types[0] === AuthDelegationType.PersonalRepresentative,
- ),
- ).toBeTruthy()
- })
- })
+ describe('when user is a personal representative with one representee', () => {
+ let prModel: typeof PersonalRepresentative
+ let prRightsModel: typeof PersonalRepresentativeRight
+ let prRightTypeModel: typeof PersonalRepresentativeRightType
+ let prTypeModel: typeof PersonalRepresentativeType
+ let delegationTypeModel: typeof DelegationTypeModel
+ let delegationProviderModel: typeof DelegationProviderModel
+ let prDelegationTypeModel: typeof PersonalRepresentativeDelegationTypeModel
+
+ beforeAll(async () => {
+ prTypeModel = app.get(
+ 'PersonalRepresentativeTypeRepository',
+ )
+ prModel = app.get(
+ 'PersonalRepresentativeRepository',
+ )
+ prRightTypeModel = app.get<
+ typeof PersonalRepresentativeRightType
+ >('PersonalRepresentativeRightTypeRepository')
+ prRightsModel = app.get(
+ 'PersonalRepresentativeRightRepository',
+ )
+ delegationTypeModel = app.get(
+ getModelToken(DelegationTypeModel),
+ )
+ delegationProviderModel = app.get(
+ getModelToken(DelegationProviderModel),
+ )
+ prDelegationTypeModel = app.get<
+ typeof PersonalRepresentativeDelegationTypeModel
+ >(getModelToken(PersonalRepresentativeDelegationTypeModel))
+
+ const prType = await prTypeModel.create({
+ code: 'prTypeCode',
+ name: 'prTypeName',
+ description: 'prTypeDescription',
+ })
+
+ const pr = await prModel.create({
+ nationalIdPersonalRepresentative: user.nationalId,
+ nationalIdRepresentedPerson: nationalRegistryUser.nationalId,
+ personalRepresentativeTypeCode: prType.code,
+ contractId: '1',
+ externalUserId: '1',
+ })
+
+ const dt = await delegationTypeModel.create({
+ id: getPersonalRepresentativeDelegationType('prRightType'),
+ providerId:
+ AuthDelegationProvider.PersonalRepresentativeRegistry,
+ name: `Personal Representative: prRightType`,
+ description: `Personal representative delegation type for right type prRightType`,
+ })
+
+ const prRightType = await prRightTypeModel.create({
+ code: 'prRightType',
+ description: 'prRightTypeDescription',
+ })
+
+ const prr = await prRightsModel.create({
+ rightTypeCode: prRightType.code,
+ personalRepresentativeId: pr.id,
+ })
+
+ await prDelegationTypeModel.create({
+ id: prr.id,
+ delegationTypeId: dt.id,
+ personalRepresentativeId: pr.id,
+ })
+ })
- it('should not return delegations when client does not support personal representative delegations', async () => {
- // Prepare
- await clientDelegationTypeModel.destroy({
- where: {
- clientId: client.clientId,
- delegationType: AuthDelegationType.PersonalRepresentative,
- },
- })
+ describe('when fetched', () => {
+ let response: request.Response
+ let body: MergedDelegationDTO[]
+
+ beforeAll(async () => {
+ response = await server.get(`${path}${query}`)
+ body = response.body
+ })
+
+ it('should have a an OK return status', () => {
+ expect(response.status).toEqual(200)
+ })
+
+ it('should return a single entity', () => {
+ expect(body.length).toEqual(1)
+ })
+
+ it('should have the nationalId of the user as the representer', () => {
+ expect(
+ body.some((d) => d.toNationalId === user.nationalId),
+ ).toBeTruthy()
+ })
+
+ it('should have the nationalId of the correct representee', () => {
+ expect(
+ body.some(
+ (d) => d.fromNationalId === nationalRegistryUser.nationalId,
+ ),
+ ).toBeTruthy()
+ })
+
+ it('should have the name of the correct representee', () => {
+ expect(
+ body.some((d) => d.fromName === nationalRegistryUser.name),
+ ).toBeTruthy()
+ })
+
+ it('should have the delegation type claim of PersonalRepresentative', () => {
+ expect(
+ body.some(
+ (d) =>
+ d.types[0] === AuthDelegationType.PersonalRepresentative,
+ ),
+ ).toBeTruthy()
+ })
+ })
- // Act
- const res = await server.get(`${path}${query}`)
+ it('should not return delegations when client does not support personal representative delegations', async () => {
+ // Prepare
+ await clientDelegationTypeModel.destroy({
+ where: {
+ clientId: client.clientId,
+ delegationType: AuthDelegationType.PersonalRepresentative,
+ },
+ })
- // Assert
- expect(res.status).toEqual(200)
- expect(res.body).toHaveLength(0)
- })
+ // Act
+ const res = await server.get(`${path}${query}`)
- it('should return a personal representative delegation since the type is included in the delegationTypes filter', async () => {
- // Act
- const res = await server.get(
- `${path}${query}&delegationTypes=${AuthDelegationType.Custom}&delegationTypes=${AuthDelegationType.ProcurationHolder}&delegationTypes=${AuthDelegationType.PersonalRepresentative}`,
- )
+ // Assert
+ expect(res.status).toEqual(200)
+ expect(res.body).toHaveLength(0)
+ })
- // Assert
- expect(res.status).toEqual(200)
- expect(res.body).toHaveLength(1)
- expect(res.body[0].types[0]).toEqual(
- AuthDelegationType.PersonalRepresentative,
- )
- })
+ it('should return a personal representative delegation since the type is included in the delegationTypes filter', async () => {
+ // Act
+ const res = await server.get(
+ `${path}${query}&delegationTypes=${AuthDelegationType.Custom}&delegationTypes=${AuthDelegationType.ProcurationHolder}&delegationTypes=${AuthDelegationType.PersonalRepresentative}`,
+ )
+
+ // Assert
+ expect(res.status).toEqual(200)
+ expect(res.body).toHaveLength(1)
+ expect(res.body[0].types[0]).toEqual(
+ AuthDelegationType.PersonalRepresentative,
+ )
+ })
- it('should not return a personal representative delegation since the type is not included in the delegationTypes filter', async () => {
- // Act
- const res = await server.get(
- `${path}${query}&delegationTypes=${AuthDelegationType.Custom}&delegationTypes=${AuthDelegationType.LegalGuardian}&delegationTypes=${AuthDelegationType.ProcurationHolder}`,
- )
+ it('should not return a personal representative delegation since the type is not included in the delegationTypes filter', async () => {
+ // Act
+ const res = await server.get(
+ `${path}${query}&delegationTypes=${AuthDelegationType.Custom}&delegationTypes=${AuthDelegationType.LegalGuardian}&delegationTypes=${AuthDelegationType.ProcurationHolder}`,
+ )
- // Assert
- expect(res.status).toEqual(200)
- expect(res.body).toHaveLength(0)
- })
+ // Assert
+ expect(res.status).toEqual(200)
+ expect(res.body).toHaveLength(0)
+ })
- afterAll(async () => {
- await prRightsModel.destroy({
- where: {},
- cascade: true,
- truncate: true,
- force: true,
- })
- await prRightTypeModel.destroy({
- where: {},
- cascade: true,
- truncate: true,
- force: true,
- })
- await prModel.destroy({
- where: {},
- cascade: true,
- truncate: true,
- force: true,
- })
- await prTypeModel.destroy({
- where: {},
- cascade: true,
- truncate: true,
- force: true,
- })
- await prDelegationTypeModel.destroy({
- where: {},
- cascade: true,
- truncate: true,
- force: true,
- })
- await delegationTypeModel.destroy({
- where: {},
- cascade: true,
- truncate: true,
- force: true,
- })
- await delegationProviderModel.destroy({
- where: {},
- cascade: true,
- truncate: true,
- force: true,
+ afterAll(async () => {
+ await prRightsModel.destroy({
+ where: {},
+ cascade: true,
+ truncate: true,
+ force: true,
+ })
+ await prRightTypeModel.destroy({
+ where: {},
+ cascade: true,
+ truncate: true,
+ force: true,
+ })
+ await prModel.destroy({
+ where: {},
+ cascade: true,
+ truncate: true,
+ force: true,
+ })
+ await prTypeModel.destroy({
+ where: {},
+ cascade: true,
+ truncate: true,
+ force: true,
+ })
+ await prDelegationTypeModel.destroy({
+ where: {},
+ cascade: true,
+ truncate: true,
+ force: true,
+ })
+ await delegationTypeModel.destroy({
+ where: {},
+ cascade: true,
+ truncate: true,
+ force: true,
+ })
+ await delegationProviderModel.destroy({
+ where: {},
+ cascade: true,
+ truncate: true,
+ force: true,
+ })
+ })
})
})
})
- })
- })
+ },
+ )
describe('without auth and permission', () => {
it.each`
diff --git a/charts/identity-server/values.dev.yaml b/charts/identity-server/values.dev.yaml
index e409d5e63565..37227fdaaedd 100644
--- a/charts/identity-server/values.dev.yaml
+++ b/charts/identity-server/values.dev.yaml
@@ -227,6 +227,10 @@ services-auth-admin-api:
IDENTITY_SERVER_ISSUER_URL: 'https://identity-server.dev01.devland.is'
IDENTITY_SERVER_ISSUER_URL_LIST: '["https://identity-server.dev01.devland.is","https://identity-server.staging01.devland.is","https://innskra.island.is"]'
LOG_LEVEL: 'info'
+ NATIONAL_REGISTRY_B2C_CLIENT_ID: '6cf94113-d326-4e4d-b97c-1fea12d2f5e1'
+ NATIONAL_REGISTRY_B2C_ENDPOINT: 'https://skraidentitydev.b2clogin.com/skraidentitydev.onmicrosoft.com/b2c_1_midlun_flow/oauth2/v2.0/token'
+ NATIONAL_REGISTRY_B2C_PATH: 'IS-DEV/GOV/10001/SKRA-Cloud-Protected/Midlun-v1'
+ NATIONAL_REGISTRY_B2C_SCOPE: 'https://skraidentitydev.onmicrosoft.com/midlun/.default'
NODE_OPTIONS: '--max-old-space-size=691 -r dd-trace/init'
SERVERSIDE_FEATURES_ON: ''
SYSLUMENN_HOST: 'https://api.syslumenn.is/staging'
@@ -297,6 +301,7 @@ services-auth-admin-api:
CONFIGCAT_SDK_KEY: '/k8s/configcat/CONFIGCAT_SDK_KEY'
DB_PASS: '/k8s/servicesauth/DB_PASSWORD'
IDENTITY_SERVER_CLIENT_SECRET: '/k8s/services-auth/IDENTITY_SERVER_CLIENT_SECRET'
+ NATIONAL_REGISTRY_B2C_CLIENT_SECRET: '/k8s/services-auth/NATIONAL_REGISTRY_B2C_CLIENT_SECRET'
NATIONAL_REGISTRY_IDS_CLIENT_SECRET: '/k8s/xroad/client/NATIONAL-REGISTRY/IDENTITYSERVER_SECRET'
SYSLUMENN_PASSWORD: '/k8s/services-auth/SYSLUMENN_PASSWORD'
SYSLUMENN_USERNAME: '/k8s/services-auth/SYSLUMENN_USERNAME'
@@ -605,6 +610,10 @@ services-auth-personal-representative:
IDENTITY_SERVER_CLIENT_ID: '@island.is/clients/auth-api'
IDENTITY_SERVER_ISSUER_URL: 'https://identity-server.dev01.devland.is'
LOG_LEVEL: 'info'
+ NATIONAL_REGISTRY_B2C_CLIENT_ID: '6cf94113-d326-4e4d-b97c-1fea12d2f5e1'
+ NATIONAL_REGISTRY_B2C_ENDPOINT: 'https://skraidentitydev.b2clogin.com/skraidentitydev.onmicrosoft.com/b2c_1_midlun_flow/oauth2/v2.0/token'
+ NATIONAL_REGISTRY_B2C_PATH: 'IS-DEV/GOV/10001/SKRA-Cloud-Protected/Midlun-v1'
+ NATIONAL_REGISTRY_B2C_SCOPE: 'https://skraidentitydev.onmicrosoft.com/midlun/.default'
NODE_OPTIONS: '--max-old-space-size=460 -r dd-trace/init'
SERVERSIDE_FEATURES_ON: ''
SYSLUMENN_HOST: 'https://api.syslumenn.is/staging'
@@ -670,6 +679,7 @@ services-auth-personal-representative:
CONFIGCAT_SDK_KEY: '/k8s/configcat/CONFIGCAT_SDK_KEY'
DB_PASS: '/k8s/servicesauth/DB_PASSWORD'
IDENTITY_SERVER_CLIENT_SECRET: '/k8s/services-auth/IDENTITY_SERVER_CLIENT_SECRET'
+ NATIONAL_REGISTRY_B2C_CLIENT_SECRET: '/k8s/services-auth/NATIONAL_REGISTRY_B2C_CLIENT_SECRET'
SYSLUMENN_PASSWORD: '/k8s/services-auth/SYSLUMENN_PASSWORD'
SYSLUMENN_USERNAME: '/k8s/services-auth/SYSLUMENN_USERNAME'
ZENDESK_CONTACT_FORM_EMAIL: '/k8s/api/ZENDESK_CONTACT_FORM_EMAIL'
@@ -752,6 +762,10 @@ services-auth-public-api:
IDENTITY_SERVER_CLIENT_ID: '@island.is/clients/auth-api'
IDENTITY_SERVER_ISSUER_URL: 'https://identity-server.dev01.devland.is'
LOG_LEVEL: 'info'
+ NATIONAL_REGISTRY_B2C_CLIENT_ID: '6cf94113-d326-4e4d-b97c-1fea12d2f5e1'
+ NATIONAL_REGISTRY_B2C_ENDPOINT: 'https://skraidentitydev.b2clogin.com/skraidentitydev.onmicrosoft.com/b2c_1_midlun_flow/oauth2/v2.0/token'
+ NATIONAL_REGISTRY_B2C_PATH: 'IS-DEV/GOV/10001/SKRA-Cloud-Protected/Midlun-v1'
+ NATIONAL_REGISTRY_B2C_SCOPE: 'https://skraidentitydev.onmicrosoft.com/midlun/.default'
NODE_OPTIONS: '--max-old-space-size=345 -r dd-trace/init'
PASSKEY_CORE_ALLOWED_ORIGINS: '["https://island.is","android:apk-key-hash:JgPeo_F6KYk-ngRa26tO2SsAtMiTBQCc7WtSgN-jRX0","android:apk-key-hash:EsLTUu5kaY7XPmMl2f7nbq4amu-PNzdYu3FecNf90wU"]'
PASSKEY_CORE_CHALLENGE_TTL_MS: '120000'
@@ -832,6 +846,7 @@ services-auth-public-api:
CONFIGCAT_SDK_KEY: '/k8s/configcat/CONFIGCAT_SDK_KEY'
DB_PASS: '/k8s/servicesauth/DB_PASSWORD'
IDENTITY_SERVER_CLIENT_SECRET: '/k8s/services-auth/IDENTITY_SERVER_CLIENT_SECRET'
+ NATIONAL_REGISTRY_B2C_CLIENT_SECRET: '/k8s/services-auth/NATIONAL_REGISTRY_B2C_CLIENT_SECRET'
NATIONAL_REGISTRY_IDS_CLIENT_SECRET: '/k8s/xroad/client/NATIONAL-REGISTRY/IDENTITYSERVER_SECRET'
SYSLUMENN_PASSWORD: '/k8s/services-auth/SYSLUMENN_PASSWORD'
SYSLUMENN_USERNAME: '/k8s/services-auth/SYSLUMENN_USERNAME'
diff --git a/charts/identity-server/values.prod.yaml b/charts/identity-server/values.prod.yaml
index 7a6fc66e89f3..1ceb0720407e 100644
--- a/charts/identity-server/values.prod.yaml
+++ b/charts/identity-server/values.prod.yaml
@@ -224,6 +224,10 @@ services-auth-admin-api:
IDENTITY_SERVER_ISSUER_URL: 'https://innskra.island.is'
IDENTITY_SERVER_ISSUER_URL_LIST: '["https://innskra.island.is"]'
LOG_LEVEL: 'info'
+ NATIONAL_REGISTRY_B2C_CLIENT_ID: '8271bbc2-d8de-480f-8540-ea43fc40b7ae'
+ NATIONAL_REGISTRY_B2C_ENDPOINT: 'https://skraidentity.b2clogin.com/skraidentity.onmicrosoft.com/b2c_1_midlun_flow/oauth2/v2.0/token'
+ NATIONAL_REGISTRY_B2C_PATH: 'IS/GOV/6503760649/SKRA-Cloud-Protected/Midlun-v1'
+ NATIONAL_REGISTRY_B2C_SCOPE: 'https://skraidentity.onmicrosoft.com/midlun/.default'
NODE_OPTIONS: '--max-old-space-size=691 -r dd-trace/init'
SERVERSIDE_FEATURES_ON: 'driving-license-use-v1-endpoint-for-v2-comms'
SYSLUMENN_HOST: 'https://api.syslumenn.is/api'
@@ -294,6 +298,7 @@ services-auth-admin-api:
CONFIGCAT_SDK_KEY: '/k8s/configcat/CONFIGCAT_SDK_KEY'
DB_PASS: '/k8s/servicesauth/DB_PASSWORD'
IDENTITY_SERVER_CLIENT_SECRET: '/k8s/services-auth/IDENTITY_SERVER_CLIENT_SECRET'
+ NATIONAL_REGISTRY_B2C_CLIENT_SECRET: '/k8s/services-auth/NATIONAL_REGISTRY_B2C_CLIENT_SECRET'
NATIONAL_REGISTRY_IDS_CLIENT_SECRET: '/k8s/xroad/client/NATIONAL-REGISTRY/IDENTITYSERVER_SECRET'
SYSLUMENN_PASSWORD: '/k8s/services-auth/SYSLUMENN_PASSWORD'
SYSLUMENN_USERNAME: '/k8s/services-auth/SYSLUMENN_USERNAME'
@@ -602,6 +607,10 @@ services-auth-personal-representative:
IDENTITY_SERVER_CLIENT_ID: '@island.is/clients/auth-api'
IDENTITY_SERVER_ISSUER_URL: 'https://innskra.island.is'
LOG_LEVEL: 'info'
+ NATIONAL_REGISTRY_B2C_CLIENT_ID: '8271bbc2-d8de-480f-8540-ea43fc40b7ae'
+ NATIONAL_REGISTRY_B2C_ENDPOINT: 'https://skraidentity.b2clogin.com/skraidentity.onmicrosoft.com/b2c_1_midlun_flow/oauth2/v2.0/token'
+ NATIONAL_REGISTRY_B2C_PATH: 'IS/GOV/6503760649/SKRA-Cloud-Protected/Midlun-v1'
+ NATIONAL_REGISTRY_B2C_SCOPE: 'https://skraidentity.onmicrosoft.com/midlun/.default'
NODE_OPTIONS: '--max-old-space-size=460 -r dd-trace/init'
SERVERSIDE_FEATURES_ON: 'driving-license-use-v1-endpoint-for-v2-comms'
SYSLUMENN_HOST: 'https://api.syslumenn.is/api'
@@ -667,6 +676,7 @@ services-auth-personal-representative:
CONFIGCAT_SDK_KEY: '/k8s/configcat/CONFIGCAT_SDK_KEY'
DB_PASS: '/k8s/servicesauth/DB_PASSWORD'
IDENTITY_SERVER_CLIENT_SECRET: '/k8s/services-auth/IDENTITY_SERVER_CLIENT_SECRET'
+ NATIONAL_REGISTRY_B2C_CLIENT_SECRET: '/k8s/services-auth/NATIONAL_REGISTRY_B2C_CLIENT_SECRET'
SYSLUMENN_PASSWORD: '/k8s/services-auth/SYSLUMENN_PASSWORD'
SYSLUMENN_USERNAME: '/k8s/services-auth/SYSLUMENN_USERNAME'
ZENDESK_CONTACT_FORM_EMAIL: '/k8s/api/ZENDESK_CONTACT_FORM_EMAIL'
@@ -749,6 +759,10 @@ services-auth-public-api:
IDENTITY_SERVER_CLIENT_ID: '@island.is/clients/auth-api'
IDENTITY_SERVER_ISSUER_URL: 'https://innskra.island.is'
LOG_LEVEL: 'info'
+ NATIONAL_REGISTRY_B2C_CLIENT_ID: '8271bbc2-d8de-480f-8540-ea43fc40b7ae'
+ NATIONAL_REGISTRY_B2C_ENDPOINT: 'https://skraidentity.b2clogin.com/skraidentity.onmicrosoft.com/b2c_1_midlun_flow/oauth2/v2.0/token'
+ NATIONAL_REGISTRY_B2C_PATH: 'IS/GOV/6503760649/SKRA-Cloud-Protected/Midlun-v1'
+ NATIONAL_REGISTRY_B2C_SCOPE: 'https://skraidentity.onmicrosoft.com/midlun/.default'
NODE_OPTIONS: '--max-old-space-size=345 -r dd-trace/init'
PASSKEY_CORE_ALLOWED_ORIGINS: '["https://island.is","android:apk-key-hash:JgPeo_F6KYk-ngRa26tO2SsAtMiTBQCc7WtSgN-jRX0","android:apk-key-hash:EsLTUu5kaY7XPmMl2f7nbq4amu-PNzdYu3FecNf90wU"]'
PASSKEY_CORE_CHALLENGE_TTL_MS: '120000'
@@ -829,6 +843,7 @@ services-auth-public-api:
CONFIGCAT_SDK_KEY: '/k8s/configcat/CONFIGCAT_SDK_KEY'
DB_PASS: '/k8s/servicesauth/DB_PASSWORD'
IDENTITY_SERVER_CLIENT_SECRET: '/k8s/services-auth/IDENTITY_SERVER_CLIENT_SECRET'
+ NATIONAL_REGISTRY_B2C_CLIENT_SECRET: '/k8s/services-auth/NATIONAL_REGISTRY_B2C_CLIENT_SECRET'
NATIONAL_REGISTRY_IDS_CLIENT_SECRET: '/k8s/xroad/client/NATIONAL-REGISTRY/IDENTITYSERVER_SECRET'
SYSLUMENN_PASSWORD: '/k8s/services-auth/SYSLUMENN_PASSWORD'
SYSLUMENN_USERNAME: '/k8s/services-auth/SYSLUMENN_USERNAME'
diff --git a/charts/identity-server/values.staging.yaml b/charts/identity-server/values.staging.yaml
index 57c2727f4daa..a08cbe1f20a0 100644
--- a/charts/identity-server/values.staging.yaml
+++ b/charts/identity-server/values.staging.yaml
@@ -227,6 +227,10 @@ services-auth-admin-api:
IDENTITY_SERVER_ISSUER_URL: 'https://identity-server.staging01.devland.is'
IDENTITY_SERVER_ISSUER_URL_LIST: '["https://identity-server.staging01.devland.is","https://innskra.island.is"]'
LOG_LEVEL: 'info'
+ NATIONAL_REGISTRY_B2C_CLIENT_ID: '6cf94113-d326-4e4d-b97c-1fea12d2f5e1'
+ NATIONAL_REGISTRY_B2C_ENDPOINT: 'https://skraidentitydev.b2clogin.com/skraidentitydev.onmicrosoft.com/b2c_1_midlun_flow/oauth2/v2.0/token'
+ NATIONAL_REGISTRY_B2C_PATH: 'IS-TEST/GOV/10001/SKRA-Cloud-Protected/Midlun-v1'
+ NATIONAL_REGISTRY_B2C_SCOPE: 'https://skraidentitydev.onmicrosoft.com/midlun/.default'
NODE_OPTIONS: '--max-old-space-size=691 -r dd-trace/init'
SERVERSIDE_FEATURES_ON: ''
SYSLUMENN_HOST: 'https://api.syslumenn.is/staging'
@@ -297,6 +301,7 @@ services-auth-admin-api:
CONFIGCAT_SDK_KEY: '/k8s/configcat/CONFIGCAT_SDK_KEY'
DB_PASS: '/k8s/servicesauth/DB_PASSWORD'
IDENTITY_SERVER_CLIENT_SECRET: '/k8s/services-auth/IDENTITY_SERVER_CLIENT_SECRET'
+ NATIONAL_REGISTRY_B2C_CLIENT_SECRET: '/k8s/services-auth/NATIONAL_REGISTRY_B2C_CLIENT_SECRET'
NATIONAL_REGISTRY_IDS_CLIENT_SECRET: '/k8s/xroad/client/NATIONAL-REGISTRY/IDENTITYSERVER_SECRET'
SYSLUMENN_PASSWORD: '/k8s/services-auth/SYSLUMENN_PASSWORD'
SYSLUMENN_USERNAME: '/k8s/services-auth/SYSLUMENN_USERNAME'
@@ -605,6 +610,10 @@ services-auth-personal-representative:
IDENTITY_SERVER_CLIENT_ID: '@island.is/clients/auth-api'
IDENTITY_SERVER_ISSUER_URL: 'https://identity-server.staging01.devland.is'
LOG_LEVEL: 'info'
+ NATIONAL_REGISTRY_B2C_CLIENT_ID: '6cf94113-d326-4e4d-b97c-1fea12d2f5e1'
+ NATIONAL_REGISTRY_B2C_ENDPOINT: 'https://skraidentitydev.b2clogin.com/skraidentitydev.onmicrosoft.com/b2c_1_midlun_flow/oauth2/v2.0/token'
+ NATIONAL_REGISTRY_B2C_PATH: 'IS-TEST/GOV/10001/SKRA-Cloud-Protected/Midlun-v1'
+ NATIONAL_REGISTRY_B2C_SCOPE: 'https://skraidentitydev.onmicrosoft.com/midlun/.default'
NODE_OPTIONS: '--max-old-space-size=460 -r dd-trace/init'
SERVERSIDE_FEATURES_ON: ''
SYSLUMENN_HOST: 'https://api.syslumenn.is/staging'
@@ -670,6 +679,7 @@ services-auth-personal-representative:
CONFIGCAT_SDK_KEY: '/k8s/configcat/CONFIGCAT_SDK_KEY'
DB_PASS: '/k8s/servicesauth/DB_PASSWORD'
IDENTITY_SERVER_CLIENT_SECRET: '/k8s/services-auth/IDENTITY_SERVER_CLIENT_SECRET'
+ NATIONAL_REGISTRY_B2C_CLIENT_SECRET: '/k8s/services-auth/NATIONAL_REGISTRY_B2C_CLIENT_SECRET'
SYSLUMENN_PASSWORD: '/k8s/services-auth/SYSLUMENN_PASSWORD'
SYSLUMENN_USERNAME: '/k8s/services-auth/SYSLUMENN_USERNAME'
ZENDESK_CONTACT_FORM_EMAIL: '/k8s/api/ZENDESK_CONTACT_FORM_EMAIL'
@@ -752,6 +762,10 @@ services-auth-public-api:
IDENTITY_SERVER_CLIENT_ID: '@island.is/clients/auth-api'
IDENTITY_SERVER_ISSUER_URL: 'https://identity-server.staging01.devland.is'
LOG_LEVEL: 'info'
+ NATIONAL_REGISTRY_B2C_CLIENT_ID: '6cf94113-d326-4e4d-b97c-1fea12d2f5e1'
+ NATIONAL_REGISTRY_B2C_ENDPOINT: 'https://skraidentitydev.b2clogin.com/skraidentitydev.onmicrosoft.com/b2c_1_midlun_flow/oauth2/v2.0/token'
+ NATIONAL_REGISTRY_B2C_PATH: 'IS-TEST/GOV/10001/SKRA-Cloud-Protected/Midlun-v1'
+ NATIONAL_REGISTRY_B2C_SCOPE: 'https://skraidentitydev.onmicrosoft.com/midlun/.default'
NODE_OPTIONS: '--max-old-space-size=345 -r dd-trace/init'
PASSKEY_CORE_ALLOWED_ORIGINS: '["https://island.is","android:apk-key-hash:JgPeo_F6KYk-ngRa26tO2SsAtMiTBQCc7WtSgN-jRX0","android:apk-key-hash:EsLTUu5kaY7XPmMl2f7nbq4amu-PNzdYu3FecNf90wU"]'
PASSKEY_CORE_CHALLENGE_TTL_MS: '120000'
@@ -832,6 +846,7 @@ services-auth-public-api:
CONFIGCAT_SDK_KEY: '/k8s/configcat/CONFIGCAT_SDK_KEY'
DB_PASS: '/k8s/servicesauth/DB_PASSWORD'
IDENTITY_SERVER_CLIENT_SECRET: '/k8s/services-auth/IDENTITY_SERVER_CLIENT_SECRET'
+ NATIONAL_REGISTRY_B2C_CLIENT_SECRET: '/k8s/services-auth/NATIONAL_REGISTRY_B2C_CLIENT_SECRET'
NATIONAL_REGISTRY_IDS_CLIENT_SECRET: '/k8s/xroad/client/NATIONAL-REGISTRY/IDENTITYSERVER_SECRET'
SYSLUMENN_PASSWORD: '/k8s/services-auth/SYSLUMENN_PASSWORD'
SYSLUMENN_USERNAME: '/k8s/services-auth/SYSLUMENN_USERNAME'
diff --git a/charts/services/services-auth-admin-api/values.dev.yaml b/charts/services/services-auth-admin-api/values.dev.yaml
index f2b108d4a7ce..a792c06696f7 100644
--- a/charts/services/services-auth-admin-api/values.dev.yaml
+++ b/charts/services/services-auth-admin-api/values.dev.yaml
@@ -29,6 +29,10 @@ env:
IDENTITY_SERVER_ISSUER_URL: 'https://identity-server.dev01.devland.is'
IDENTITY_SERVER_ISSUER_URL_LIST: '["https://identity-server.dev01.devland.is","https://identity-server.staging01.devland.is","https://innskra.island.is"]'
LOG_LEVEL: 'info'
+ NATIONAL_REGISTRY_B2C_CLIENT_ID: '6cf94113-d326-4e4d-b97c-1fea12d2f5e1'
+ NATIONAL_REGISTRY_B2C_ENDPOINT: 'https://skraidentitydev.b2clogin.com/skraidentitydev.onmicrosoft.com/b2c_1_midlun_flow/oauth2/v2.0/token'
+ NATIONAL_REGISTRY_B2C_PATH: 'IS-DEV/GOV/10001/SKRA-Cloud-Protected/Midlun-v1'
+ NATIONAL_REGISTRY_B2C_SCOPE: 'https://skraidentitydev.onmicrosoft.com/midlun/.default'
NODE_OPTIONS: '--max-old-space-size=691 -r dd-trace/init'
SERVERSIDE_FEATURES_ON: ''
SYSLUMENN_HOST: 'https://api.syslumenn.is/staging'
@@ -99,6 +103,7 @@ secrets:
CONFIGCAT_SDK_KEY: '/k8s/configcat/CONFIGCAT_SDK_KEY'
DB_PASS: '/k8s/servicesauth/DB_PASSWORD'
IDENTITY_SERVER_CLIENT_SECRET: '/k8s/services-auth/IDENTITY_SERVER_CLIENT_SECRET'
+ NATIONAL_REGISTRY_B2C_CLIENT_SECRET: '/k8s/services-auth/NATIONAL_REGISTRY_B2C_CLIENT_SECRET'
NATIONAL_REGISTRY_IDS_CLIENT_SECRET: '/k8s/xroad/client/NATIONAL-REGISTRY/IDENTITYSERVER_SECRET'
SYSLUMENN_PASSWORD: '/k8s/services-auth/SYSLUMENN_PASSWORD'
SYSLUMENN_USERNAME: '/k8s/services-auth/SYSLUMENN_USERNAME'
diff --git a/charts/services/services-auth-admin-api/values.prod.yaml b/charts/services/services-auth-admin-api/values.prod.yaml
index f392a1400099..cc6563a905ed 100644
--- a/charts/services/services-auth-admin-api/values.prod.yaml
+++ b/charts/services/services-auth-admin-api/values.prod.yaml
@@ -29,6 +29,10 @@ env:
IDENTITY_SERVER_ISSUER_URL: 'https://innskra.island.is'
IDENTITY_SERVER_ISSUER_URL_LIST: '["https://innskra.island.is"]'
LOG_LEVEL: 'info'
+ NATIONAL_REGISTRY_B2C_CLIENT_ID: '8271bbc2-d8de-480f-8540-ea43fc40b7ae'
+ NATIONAL_REGISTRY_B2C_ENDPOINT: 'https://skraidentity.b2clogin.com/skraidentity.onmicrosoft.com/b2c_1_midlun_flow/oauth2/v2.0/token'
+ NATIONAL_REGISTRY_B2C_PATH: 'IS/GOV/6503760649/SKRA-Cloud-Protected/Midlun-v1'
+ NATIONAL_REGISTRY_B2C_SCOPE: 'https://skraidentity.onmicrosoft.com/midlun/.default'
NODE_OPTIONS: '--max-old-space-size=691 -r dd-trace/init'
SERVERSIDE_FEATURES_ON: 'driving-license-use-v1-endpoint-for-v2-comms'
SYSLUMENN_HOST: 'https://api.syslumenn.is/api'
@@ -99,6 +103,7 @@ secrets:
CONFIGCAT_SDK_KEY: '/k8s/configcat/CONFIGCAT_SDK_KEY'
DB_PASS: '/k8s/servicesauth/DB_PASSWORD'
IDENTITY_SERVER_CLIENT_SECRET: '/k8s/services-auth/IDENTITY_SERVER_CLIENT_SECRET'
+ NATIONAL_REGISTRY_B2C_CLIENT_SECRET: '/k8s/services-auth/NATIONAL_REGISTRY_B2C_CLIENT_SECRET'
NATIONAL_REGISTRY_IDS_CLIENT_SECRET: '/k8s/xroad/client/NATIONAL-REGISTRY/IDENTITYSERVER_SECRET'
SYSLUMENN_PASSWORD: '/k8s/services-auth/SYSLUMENN_PASSWORD'
SYSLUMENN_USERNAME: '/k8s/services-auth/SYSLUMENN_USERNAME'
diff --git a/charts/services/services-auth-admin-api/values.staging.yaml b/charts/services/services-auth-admin-api/values.staging.yaml
index 1ca982a2abb7..e67c4abfceac 100644
--- a/charts/services/services-auth-admin-api/values.staging.yaml
+++ b/charts/services/services-auth-admin-api/values.staging.yaml
@@ -29,6 +29,10 @@ env:
IDENTITY_SERVER_ISSUER_URL: 'https://identity-server.staging01.devland.is'
IDENTITY_SERVER_ISSUER_URL_LIST: '["https://identity-server.staging01.devland.is","https://innskra.island.is"]'
LOG_LEVEL: 'info'
+ NATIONAL_REGISTRY_B2C_CLIENT_ID: '6cf94113-d326-4e4d-b97c-1fea12d2f5e1'
+ NATIONAL_REGISTRY_B2C_ENDPOINT: 'https://skraidentitydev.b2clogin.com/skraidentitydev.onmicrosoft.com/b2c_1_midlun_flow/oauth2/v2.0/token'
+ NATIONAL_REGISTRY_B2C_PATH: 'IS-TEST/GOV/10001/SKRA-Cloud-Protected/Midlun-v1'
+ NATIONAL_REGISTRY_B2C_SCOPE: 'https://skraidentitydev.onmicrosoft.com/midlun/.default'
NODE_OPTIONS: '--max-old-space-size=691 -r dd-trace/init'
SERVERSIDE_FEATURES_ON: ''
SYSLUMENN_HOST: 'https://api.syslumenn.is/staging'
@@ -99,6 +103,7 @@ secrets:
CONFIGCAT_SDK_KEY: '/k8s/configcat/CONFIGCAT_SDK_KEY'
DB_PASS: '/k8s/servicesauth/DB_PASSWORD'
IDENTITY_SERVER_CLIENT_SECRET: '/k8s/services-auth/IDENTITY_SERVER_CLIENT_SECRET'
+ NATIONAL_REGISTRY_B2C_CLIENT_SECRET: '/k8s/services-auth/NATIONAL_REGISTRY_B2C_CLIENT_SECRET'
NATIONAL_REGISTRY_IDS_CLIENT_SECRET: '/k8s/xroad/client/NATIONAL-REGISTRY/IDENTITYSERVER_SECRET'
SYSLUMENN_PASSWORD: '/k8s/services-auth/SYSLUMENN_PASSWORD'
SYSLUMENN_USERNAME: '/k8s/services-auth/SYSLUMENN_USERNAME'
diff --git a/charts/services/services-auth-personal-representative/values.dev.yaml b/charts/services/services-auth-personal-representative/values.dev.yaml
index 973eb041bc32..aaca50f4c0c0 100644
--- a/charts/services/services-auth-personal-representative/values.dev.yaml
+++ b/charts/services/services-auth-personal-representative/values.dev.yaml
@@ -28,6 +28,10 @@ env:
IDENTITY_SERVER_CLIENT_ID: '@island.is/clients/auth-api'
IDENTITY_SERVER_ISSUER_URL: 'https://identity-server.dev01.devland.is'
LOG_LEVEL: 'info'
+ NATIONAL_REGISTRY_B2C_CLIENT_ID: '6cf94113-d326-4e4d-b97c-1fea12d2f5e1'
+ NATIONAL_REGISTRY_B2C_ENDPOINT: 'https://skraidentitydev.b2clogin.com/skraidentitydev.onmicrosoft.com/b2c_1_midlun_flow/oauth2/v2.0/token'
+ NATIONAL_REGISTRY_B2C_PATH: 'IS-DEV/GOV/10001/SKRA-Cloud-Protected/Midlun-v1'
+ NATIONAL_REGISTRY_B2C_SCOPE: 'https://skraidentitydev.onmicrosoft.com/midlun/.default'
NODE_OPTIONS: '--max-old-space-size=460 -r dd-trace/init'
SERVERSIDE_FEATURES_ON: ''
SYSLUMENN_HOST: 'https://api.syslumenn.is/staging'
@@ -93,6 +97,7 @@ secrets:
CONFIGCAT_SDK_KEY: '/k8s/configcat/CONFIGCAT_SDK_KEY'
DB_PASS: '/k8s/servicesauth/DB_PASSWORD'
IDENTITY_SERVER_CLIENT_SECRET: '/k8s/services-auth/IDENTITY_SERVER_CLIENT_SECRET'
+ NATIONAL_REGISTRY_B2C_CLIENT_SECRET: '/k8s/services-auth/NATIONAL_REGISTRY_B2C_CLIENT_SECRET'
SYSLUMENN_PASSWORD: '/k8s/services-auth/SYSLUMENN_PASSWORD'
SYSLUMENN_USERNAME: '/k8s/services-auth/SYSLUMENN_USERNAME'
ZENDESK_CONTACT_FORM_EMAIL: '/k8s/api/ZENDESK_CONTACT_FORM_EMAIL'
diff --git a/charts/services/services-auth-personal-representative/values.prod.yaml b/charts/services/services-auth-personal-representative/values.prod.yaml
index f9f74f8d92c1..e65c0aafe6fa 100644
--- a/charts/services/services-auth-personal-representative/values.prod.yaml
+++ b/charts/services/services-auth-personal-representative/values.prod.yaml
@@ -28,6 +28,10 @@ env:
IDENTITY_SERVER_CLIENT_ID: '@island.is/clients/auth-api'
IDENTITY_SERVER_ISSUER_URL: 'https://innskra.island.is'
LOG_LEVEL: 'info'
+ NATIONAL_REGISTRY_B2C_CLIENT_ID: '8271bbc2-d8de-480f-8540-ea43fc40b7ae'
+ NATIONAL_REGISTRY_B2C_ENDPOINT: 'https://skraidentity.b2clogin.com/skraidentity.onmicrosoft.com/b2c_1_midlun_flow/oauth2/v2.0/token'
+ NATIONAL_REGISTRY_B2C_PATH: 'IS/GOV/6503760649/SKRA-Cloud-Protected/Midlun-v1'
+ NATIONAL_REGISTRY_B2C_SCOPE: 'https://skraidentity.onmicrosoft.com/midlun/.default'
NODE_OPTIONS: '--max-old-space-size=460 -r dd-trace/init'
SERVERSIDE_FEATURES_ON: 'driving-license-use-v1-endpoint-for-v2-comms'
SYSLUMENN_HOST: 'https://api.syslumenn.is/api'
@@ -93,6 +97,7 @@ secrets:
CONFIGCAT_SDK_KEY: '/k8s/configcat/CONFIGCAT_SDK_KEY'
DB_PASS: '/k8s/servicesauth/DB_PASSWORD'
IDENTITY_SERVER_CLIENT_SECRET: '/k8s/services-auth/IDENTITY_SERVER_CLIENT_SECRET'
+ NATIONAL_REGISTRY_B2C_CLIENT_SECRET: '/k8s/services-auth/NATIONAL_REGISTRY_B2C_CLIENT_SECRET'
SYSLUMENN_PASSWORD: '/k8s/services-auth/SYSLUMENN_PASSWORD'
SYSLUMENN_USERNAME: '/k8s/services-auth/SYSLUMENN_USERNAME'
ZENDESK_CONTACT_FORM_EMAIL: '/k8s/api/ZENDESK_CONTACT_FORM_EMAIL'
diff --git a/charts/services/services-auth-personal-representative/values.staging.yaml b/charts/services/services-auth-personal-representative/values.staging.yaml
index 21ec99d0547a..16fadeb06ab7 100644
--- a/charts/services/services-auth-personal-representative/values.staging.yaml
+++ b/charts/services/services-auth-personal-representative/values.staging.yaml
@@ -28,6 +28,10 @@ env:
IDENTITY_SERVER_CLIENT_ID: '@island.is/clients/auth-api'
IDENTITY_SERVER_ISSUER_URL: 'https://identity-server.staging01.devland.is'
LOG_LEVEL: 'info'
+ NATIONAL_REGISTRY_B2C_CLIENT_ID: '6cf94113-d326-4e4d-b97c-1fea12d2f5e1'
+ NATIONAL_REGISTRY_B2C_ENDPOINT: 'https://skraidentitydev.b2clogin.com/skraidentitydev.onmicrosoft.com/b2c_1_midlun_flow/oauth2/v2.0/token'
+ NATIONAL_REGISTRY_B2C_PATH: 'IS-TEST/GOV/10001/SKRA-Cloud-Protected/Midlun-v1'
+ NATIONAL_REGISTRY_B2C_SCOPE: 'https://skraidentitydev.onmicrosoft.com/midlun/.default'
NODE_OPTIONS: '--max-old-space-size=460 -r dd-trace/init'
SERVERSIDE_FEATURES_ON: ''
SYSLUMENN_HOST: 'https://api.syslumenn.is/staging'
@@ -93,6 +97,7 @@ secrets:
CONFIGCAT_SDK_KEY: '/k8s/configcat/CONFIGCAT_SDK_KEY'
DB_PASS: '/k8s/servicesauth/DB_PASSWORD'
IDENTITY_SERVER_CLIENT_SECRET: '/k8s/services-auth/IDENTITY_SERVER_CLIENT_SECRET'
+ NATIONAL_REGISTRY_B2C_CLIENT_SECRET: '/k8s/services-auth/NATIONAL_REGISTRY_B2C_CLIENT_SECRET'
SYSLUMENN_PASSWORD: '/k8s/services-auth/SYSLUMENN_PASSWORD'
SYSLUMENN_USERNAME: '/k8s/services-auth/SYSLUMENN_USERNAME'
ZENDESK_CONTACT_FORM_EMAIL: '/k8s/api/ZENDESK_CONTACT_FORM_EMAIL'
diff --git a/charts/services/services-auth-public-api/values.dev.yaml b/charts/services/services-auth-public-api/values.dev.yaml
index 4398430d7eb2..9b4cc091ee46 100644
--- a/charts/services/services-auth-public-api/values.dev.yaml
+++ b/charts/services/services-auth-public-api/values.dev.yaml
@@ -28,6 +28,10 @@ env:
IDENTITY_SERVER_CLIENT_ID: '@island.is/clients/auth-api'
IDENTITY_SERVER_ISSUER_URL: 'https://identity-server.dev01.devland.is'
LOG_LEVEL: 'info'
+ NATIONAL_REGISTRY_B2C_CLIENT_ID: '6cf94113-d326-4e4d-b97c-1fea12d2f5e1'
+ NATIONAL_REGISTRY_B2C_ENDPOINT: 'https://skraidentitydev.b2clogin.com/skraidentitydev.onmicrosoft.com/b2c_1_midlun_flow/oauth2/v2.0/token'
+ NATIONAL_REGISTRY_B2C_PATH: 'IS-DEV/GOV/10001/SKRA-Cloud-Protected/Midlun-v1'
+ NATIONAL_REGISTRY_B2C_SCOPE: 'https://skraidentitydev.onmicrosoft.com/midlun/.default'
NODE_OPTIONS: '--max-old-space-size=345 -r dd-trace/init'
PASSKEY_CORE_ALLOWED_ORIGINS: '["https://island.is","android:apk-key-hash:JgPeo_F6KYk-ngRa26tO2SsAtMiTBQCc7WtSgN-jRX0","android:apk-key-hash:EsLTUu5kaY7XPmMl2f7nbq4amu-PNzdYu3FecNf90wU"]'
PASSKEY_CORE_CHALLENGE_TTL_MS: '120000'
@@ -108,6 +112,7 @@ secrets:
CONFIGCAT_SDK_KEY: '/k8s/configcat/CONFIGCAT_SDK_KEY'
DB_PASS: '/k8s/servicesauth/DB_PASSWORD'
IDENTITY_SERVER_CLIENT_SECRET: '/k8s/services-auth/IDENTITY_SERVER_CLIENT_SECRET'
+ NATIONAL_REGISTRY_B2C_CLIENT_SECRET: '/k8s/services-auth/NATIONAL_REGISTRY_B2C_CLIENT_SECRET'
NATIONAL_REGISTRY_IDS_CLIENT_SECRET: '/k8s/xroad/client/NATIONAL-REGISTRY/IDENTITYSERVER_SECRET'
SYSLUMENN_PASSWORD: '/k8s/services-auth/SYSLUMENN_PASSWORD'
SYSLUMENN_USERNAME: '/k8s/services-auth/SYSLUMENN_USERNAME'
diff --git a/charts/services/services-auth-public-api/values.prod.yaml b/charts/services/services-auth-public-api/values.prod.yaml
index cedab910898a..13e897e62429 100644
--- a/charts/services/services-auth-public-api/values.prod.yaml
+++ b/charts/services/services-auth-public-api/values.prod.yaml
@@ -28,6 +28,10 @@ env:
IDENTITY_SERVER_CLIENT_ID: '@island.is/clients/auth-api'
IDENTITY_SERVER_ISSUER_URL: 'https://innskra.island.is'
LOG_LEVEL: 'info'
+ NATIONAL_REGISTRY_B2C_CLIENT_ID: '8271bbc2-d8de-480f-8540-ea43fc40b7ae'
+ NATIONAL_REGISTRY_B2C_ENDPOINT: 'https://skraidentity.b2clogin.com/skraidentity.onmicrosoft.com/b2c_1_midlun_flow/oauth2/v2.0/token'
+ NATIONAL_REGISTRY_B2C_PATH: 'IS/GOV/6503760649/SKRA-Cloud-Protected/Midlun-v1'
+ NATIONAL_REGISTRY_B2C_SCOPE: 'https://skraidentity.onmicrosoft.com/midlun/.default'
NODE_OPTIONS: '--max-old-space-size=345 -r dd-trace/init'
PASSKEY_CORE_ALLOWED_ORIGINS: '["https://island.is","android:apk-key-hash:JgPeo_F6KYk-ngRa26tO2SsAtMiTBQCc7WtSgN-jRX0","android:apk-key-hash:EsLTUu5kaY7XPmMl2f7nbq4amu-PNzdYu3FecNf90wU"]'
PASSKEY_CORE_CHALLENGE_TTL_MS: '120000'
@@ -108,6 +112,7 @@ secrets:
CONFIGCAT_SDK_KEY: '/k8s/configcat/CONFIGCAT_SDK_KEY'
DB_PASS: '/k8s/servicesauth/DB_PASSWORD'
IDENTITY_SERVER_CLIENT_SECRET: '/k8s/services-auth/IDENTITY_SERVER_CLIENT_SECRET'
+ NATIONAL_REGISTRY_B2C_CLIENT_SECRET: '/k8s/services-auth/NATIONAL_REGISTRY_B2C_CLIENT_SECRET'
NATIONAL_REGISTRY_IDS_CLIENT_SECRET: '/k8s/xroad/client/NATIONAL-REGISTRY/IDENTITYSERVER_SECRET'
SYSLUMENN_PASSWORD: '/k8s/services-auth/SYSLUMENN_PASSWORD'
SYSLUMENN_USERNAME: '/k8s/services-auth/SYSLUMENN_USERNAME'
diff --git a/charts/services/services-auth-public-api/values.staging.yaml b/charts/services/services-auth-public-api/values.staging.yaml
index af05b8333772..ba19f2a7a3bf 100644
--- a/charts/services/services-auth-public-api/values.staging.yaml
+++ b/charts/services/services-auth-public-api/values.staging.yaml
@@ -28,6 +28,10 @@ env:
IDENTITY_SERVER_CLIENT_ID: '@island.is/clients/auth-api'
IDENTITY_SERVER_ISSUER_URL: 'https://identity-server.staging01.devland.is'
LOG_LEVEL: 'info'
+ NATIONAL_REGISTRY_B2C_CLIENT_ID: '6cf94113-d326-4e4d-b97c-1fea12d2f5e1'
+ NATIONAL_REGISTRY_B2C_ENDPOINT: 'https://skraidentitydev.b2clogin.com/skraidentitydev.onmicrosoft.com/b2c_1_midlun_flow/oauth2/v2.0/token'
+ NATIONAL_REGISTRY_B2C_PATH: 'IS-TEST/GOV/10001/SKRA-Cloud-Protected/Midlun-v1'
+ NATIONAL_REGISTRY_B2C_SCOPE: 'https://skraidentitydev.onmicrosoft.com/midlun/.default'
NODE_OPTIONS: '--max-old-space-size=345 -r dd-trace/init'
PASSKEY_CORE_ALLOWED_ORIGINS: '["https://island.is","android:apk-key-hash:JgPeo_F6KYk-ngRa26tO2SsAtMiTBQCc7WtSgN-jRX0","android:apk-key-hash:EsLTUu5kaY7XPmMl2f7nbq4amu-PNzdYu3FecNf90wU"]'
PASSKEY_CORE_CHALLENGE_TTL_MS: '120000'
@@ -108,6 +112,7 @@ secrets:
CONFIGCAT_SDK_KEY: '/k8s/configcat/CONFIGCAT_SDK_KEY'
DB_PASS: '/k8s/servicesauth/DB_PASSWORD'
IDENTITY_SERVER_CLIENT_SECRET: '/k8s/services-auth/IDENTITY_SERVER_CLIENT_SECRET'
+ NATIONAL_REGISTRY_B2C_CLIENT_SECRET: '/k8s/services-auth/NATIONAL_REGISTRY_B2C_CLIENT_SECRET'
NATIONAL_REGISTRY_IDS_CLIENT_SECRET: '/k8s/xroad/client/NATIONAL-REGISTRY/IDENTITYSERVER_SECRET'
SYSLUMENN_PASSWORD: '/k8s/services-auth/SYSLUMENN_PASSWORD'
SYSLUMENN_USERNAME: '/k8s/services-auth/SYSLUMENN_USERNAME'
diff --git a/libs/auth-api-lib/src/index.ts b/libs/auth-api-lib/src/index.ts
index c31273fdae15..6c4d00f2b791 100644
--- a/libs/auth-api-lib/src/index.ts
+++ b/libs/auth-api-lib/src/index.ts
@@ -61,6 +61,7 @@ export * from './lib/delegations/utils/scopes'
export * from './lib/delegations/admin/delegation-admin-custom.service'
export * from './lib/delegations/constants/names'
export * from './lib/delegations/constants/zendesk'
+export * from './lib/delegations/national-registry-v3-feature.service'
// Resources module
export * from './lib/resources/resources.module'
diff --git a/libs/auth-api-lib/src/lib/delegations/alive-status.service.ts b/libs/auth-api-lib/src/lib/delegations/alive-status.service.ts
new file mode 100644
index 000000000000..e1e4aeceb170
--- /dev/null
+++ b/libs/auth-api-lib/src/lib/delegations/alive-status.service.ts
@@ -0,0 +1,192 @@
+import { Inject, Injectable, Logger } from '@nestjs/common'
+import * as kennitala from 'kennitala'
+
+import { NationalRegistryClientService } from '@island.is/clients/national-registry-v2'
+import { NationalRegistryV3ClientService } from '@island.is/clients/national-registry-v3'
+import { CompanyRegistryClientService } from '@island.is/clients/rsk/company-registry'
+import { LOGGER_PROVIDER } from '@island.is/logging'
+
+import { UNKNOWN_NAME } from './constants/names'
+import { partitionWithIndex } from './utils/partitionWithIndex'
+
+export type NameInfo = {
+ nationalId: string
+ name: string
+}
+
+type IdentityInfo = NameInfo & { isDeceased: boolean }
+
+const decesead = 'LÉST'
+
+@Injectable()
+export class AliveStatusService {
+ constructor(
+ private readonly nationalRegistryClient: NationalRegistryClientService,
+ private readonly nationalRegistryV3Client: NationalRegistryV3ClientService,
+ private readonly companyRegistryClient: CompanyRegistryClientService,
+ @Inject(LOGGER_PROVIDER)
+ private logger: Logger,
+ ) {}
+
+ /**
+ * Divides nationalIds into alive and deceased
+ * - Makes calls for every nationalId to NationalRegistry to check if the person exists.
+ * - Divides the nationalIds into alive and deceased, based on
+ * 1. All companies will be divided into alive.
+ * 2. If the person exists in NationalRegistry, then the person is alive.
+ */
+ public async getStatus(
+ nameInfos: NameInfo[],
+ useNationalRegistryV3: boolean,
+ ): Promise<{
+ aliveNationalIds: string[]
+ deceasedNationalIds: string[]
+ aliveNameInfo: NameInfo[]
+ }> {
+ if (nameInfos.length === 0) {
+ return {
+ aliveNationalIds: [],
+ deceasedNationalIds: [],
+ aliveNameInfo: [],
+ }
+ }
+ let identitiesValuesNoError: IdentityInfo[] = []
+ try {
+ const identities = await (
+ await Promise.allSettled(
+ this.getIdentities(nameInfos, useNationalRegistryV3),
+ )
+ ).map((promiseResult) =>
+ promiseResult.status === 'fulfilled'
+ ? promiseResult.value
+ : new Error('Error getting identity'),
+ )
+
+ identitiesValuesNoError = identities.filter(this.isNotError)
+
+ const deceasedNationalIds = identitiesValuesNoError
+ .filter((identity) => identity.isDeceased)
+ .map((identity) => identity.nationalId)
+
+ return {
+ aliveNationalIds: nameInfos
+ .filter((info) => !deceasedNationalIds.includes(info.nationalId))
+ .map((info) => info.nationalId),
+ deceasedNationalIds: deceasedNationalIds,
+ aliveNameInfo: identitiesValuesNoError,
+ }
+ } catch (error) {
+ this.logger.error(`Error getting live status.`, error)
+
+ // We do not want to fail the whole request if we cannot get the live status.
+ // Therefore, we return all nationalIds as alive.
+ return {
+ aliveNationalIds: nameInfos.map((info) => info.nationalId),
+ deceasedNationalIds: [],
+ aliveNameInfo: identitiesValuesNoError,
+ }
+ }
+ }
+
+ private getIdentities(
+ nameInfos: NameInfo[],
+ useNationalRegistryV3: boolean,
+ ): Promise[] {
+ const [companies, individuals] = partitionWithIndex(nameInfos, (nameInfo) =>
+ kennitala.isCompany(nameInfo.nationalId),
+ )
+
+ const companyPromises = companies.map((nameInfo) =>
+ this.getCompanyIdentity(nameInfo),
+ )
+
+ const individualPromises = individuals.map((nameInfo) =>
+ this.getIndividualIdentity(nameInfo, useNationalRegistryV3),
+ )
+
+ return [...companyPromises, ...individualPromises]
+ }
+
+ private getCompanyIdentity(companyInfo: NameInfo): Promise {
+ // All companies will be divided into alive
+ return this.companyRegistryClient
+ .getCompany(companyInfo.nationalId)
+ .then((company) => {
+ if (company && this.isNotError(company)) {
+ return {
+ nationalId: company.nationalId,
+ name: company.name,
+ isDeceased: false,
+ }
+ } else {
+ return {
+ nationalId: companyInfo.nationalId,
+ name: companyInfo.name ?? UNKNOWN_NAME,
+ isDeceased: false,
+ }
+ }
+ })
+ }
+
+ private async getIndividualIdentity(
+ individualInfo: NameInfo,
+ useNationalRegistryV3: boolean,
+ ): Promise {
+ if (useNationalRegistryV3) {
+ return await this.nationalRegistryV3Client
+ .getAllDataIndividual(individualInfo.nationalId)
+ .then((individual) => {
+ if (
+ individual &&
+ this.isNotError(individual) &&
+ individual?.kennitala &&
+ individual?.kennitala !== null
+ ) {
+ return {
+ nationalId: individual.kennitala,
+ name: individual.nafn ?? individualInfo.name ?? UNKNOWN_NAME,
+ isDeceased: individual.afdrif === decesead,
+ }
+ } else {
+ // Pass through although Þjóðskrá API throws an error
+ return {
+ nationalId: individualInfo.nationalId,
+ name: individualInfo.name ?? UNKNOWN_NAME,
+ isDeceased: false,
+ }
+ }
+ })
+ } else {
+ return await this.getIndividualIdentityV2(individualInfo)
+ }
+ }
+
+ private getIndividualIdentityV2(
+ individualInfo: NameInfo,
+ ): Promise {
+ return this.nationalRegistryClient
+ .getIndividual(individualInfo.nationalId)
+ .then((individual) => {
+ if (individual === null) {
+ return {
+ nationalId: individualInfo.nationalId,
+ name: individualInfo.name ?? UNKNOWN_NAME,
+ isDeceased: true,
+ }
+ } else {
+ return {
+ nationalId: individual?.nationalId ?? individualInfo.nationalId,
+ name: individual?.name ?? individualInfo.name ?? UNKNOWN_NAME,
+ isDeceased: false,
+ }
+ }
+ })
+ }
+
+ /**
+ * Checks if item is not an instance of Error
+ */
+ private isNotError(item: T | Error): item is T {
+ return item instanceof Error === false
+ }
+}
diff --git a/libs/auth-api-lib/src/lib/delegations/delegations-incoming-custom.service.ts b/libs/auth-api-lib/src/lib/delegations/delegations-incoming-custom.service.ts
index d51a25e54cf9..ee60fb61c692 100644
--- a/libs/auth-api-lib/src/lib/delegations/delegations-incoming-custom.service.ts
+++ b/libs/auth-api-lib/src/lib/delegations/delegations-incoming-custom.service.ts
@@ -1,18 +1,12 @@
-import { Inject, Injectable, Logger } from '@nestjs/common'
-import { InjectModel } from '@nestjs/sequelize'
+import { Inject, Injectable } from '@nestjs/common'
import { ConfigType } from '@nestjs/config'
+import { InjectModel } from '@nestjs/sequelize'
+import startOfDay from 'date-fns/startOfDay'
import * as kennitala from 'kennitala'
import uniqBy from 'lodash/uniqBy'
import { Op } from 'sequelize'
-import startOfDay from 'date-fns/startOfDay'
import { User } from '@island.is/auth-nest-tools'
-import {
- IndividualDto,
- NationalRegistryClientService,
-} from '@island.is/clients/national-registry-v2'
-import { CompanyRegistryClientService } from '@island.is/clients/rsk/company-registry'
-import { LOGGER_PROVIDER } from '@island.is/logging'
import { AuditService } from '@island.is/nest/audit'
import { AuthDelegationType } from '@island.is/shared/types'
import { isDefined } from '@island.is/shared/utils'
@@ -20,17 +14,18 @@ import { isDefined } from '@island.is/shared/utils'
import { ApiScopeDelegationType } from '../resources/models/api-scope-delegation-type.model'
import { ApiScopeUserAccess } from '../resources/models/api-scope-user-access.model'
import { ApiScope } from '../resources/models/api-scope.model'
+import { AliveStatusService, NameInfo } from './alive-status.service'
import { UNKNOWN_NAME } from './constants/names'
+import { DelegationConfig } from './DelegationConfig'
import { ApiScopeInfo } from './delegations-incoming.service'
import { DelegationDTO } from './dto/delegation.dto'
import { MergedDelegationDTO } from './dto/merged-delegation.dto'
+import { DelegationDelegationType } from './models/delegation-delegation-type.model'
import { DelegationScope } from './models/delegation-scope.model'
import { Delegation } from './models/delegation.model'
+import { NationalRegistryV3FeatureService } from './national-registry-v3-feature.service'
import { DelegationValidity } from './types/delegationValidity'
-import { partitionWithIndex } from './utils/partitionWithIndex'
import { getScopeValidityWhereClause } from './utils/scopes'
-import { DelegationDelegationType } from './models/delegation-delegation-type.model'
-import { DelegationConfig } from './DelegationConfig'
type FindAllValidIncomingOptions = {
nationalId: string
@@ -38,11 +33,6 @@ type FindAllValidIncomingOptions = {
validity?: DelegationValidity
}
-type FromNameInfo = {
- nationalId: string
- name: string
-}
-
/**
* Service class for incoming delegations.
* This class supports domain based delegations.
@@ -54,13 +44,11 @@ export class DelegationsIncomingCustomService {
private delegationModel: typeof Delegation,
@InjectModel(ApiScopeUserAccess)
private apiScopeUserAccessModel: typeof ApiScopeUserAccess,
- private nationalRegistryClient: NationalRegistryClientService,
- private companyRegistryClient: CompanyRegistryClientService,
- @Inject(LOGGER_PROVIDER)
- private logger: Logger,
+ private aliveStatusService: AliveStatusService,
@Inject(DelegationConfig.KEY)
private delegationConfig: ConfigType,
private auditService: AuditService,
+ private readonly nationalRegistryV3FeatureService: NationalRegistryV3FeatureService,
) {}
async findAllValidIncoming(
@@ -70,6 +58,7 @@ export class DelegationsIncomingCustomService {
validity = DelegationValidity.NOW,
}: FindAllValidIncomingOptions,
useMaster = false,
+ user?: User,
): Promise {
const { delegations, fromNameInfo } = await this.findAllIncoming(
{
@@ -78,6 +67,7 @@ export class DelegationsIncomingCustomService {
domainName,
},
useMaster,
+ user,
)
const delegationDTOs = delegations.map((d) => d.toDTO())
@@ -95,6 +85,7 @@ export class DelegationsIncomingCustomService {
async findAllValidGeneralMandate(
{ nationalId }: FindAllValidIncomingOptions,
useMaster = false,
+ user?: User,
): Promise {
const { delegations, fromNameInfo } =
await this.findAllIncomingGeneralMandates(
@@ -102,6 +93,7 @@ export class DelegationsIncomingCustomService {
nationalId,
},
useMaster,
+ user,
)
return delegations.map((delegation) => {
@@ -133,10 +125,14 @@ export class DelegationsIncomingCustomService {
return []
}
- const { delegations, fromNameInfo } = await this.findAllIncoming({
- nationalId: user.nationalId,
- validity: DelegationValidity.NOW,
- })
+ const { delegations, fromNameInfo } = await this.findAllIncoming(
+ {
+ nationalId: user.nationalId,
+ validity: DelegationValidity.NOW,
+ },
+ false,
+ user,
+ )
const validDelegations = delegations
.map((d) => {
@@ -219,9 +215,13 @@ export class DelegationsIncomingCustomService {
}
const { delegations, fromNameInfo } =
- await this.findAllIncomingGeneralMandates({
- nationalId: user.nationalId,
- })
+ await this.findAllIncomingGeneralMandates(
+ {
+ nationalId: user.nationalId,
+ },
+ false,
+ user,
+ )
const mergedDelegationDTOs = uniqBy(
delegations.map((d) =>
@@ -243,7 +243,8 @@ export class DelegationsIncomingCustomService {
private async findAllIncomingGeneralMandates(
{ nationalId }: FindAllValidIncomingOptions,
useMaster = false,
- ): Promise<{ delegations: Delegation[]; fromNameInfo: FromNameInfo[] }> {
+ user?: User,
+ ): Promise<{ delegations: Delegation[]; fromNameInfo: NameInfo[] }> {
const startOfToday = startOfDay(new Date())
const delegations = await this.delegationModel.findAll({
@@ -268,10 +269,22 @@ export class DelegationsIncomingCustomService {
})
// Check live status, i.e. dead or alive for delegations
- const { aliveDelegations, deceasedDelegations, fromNameInfo } =
- await this.getLiveStatusFromDelegations(delegations)
+ const isNationalRegistryV3DeceasedStatusEnabled =
+ await this.nationalRegistryV3FeatureService.getValue(user)
+
+ const { aliveNationalIds, deceasedNationalIds, aliveNameInfo } =
+ await this.aliveStatusService.getStatus(
+ delegations.map((d) => ({
+ nationalId: d.fromNationalId,
+ name: d.fromDisplayName,
+ })),
+ isNationalRegistryV3DeceasedStatusEnabled,
+ )
- if (deceasedDelegations.length > 0) {
+ if (deceasedNationalIds.length > 0) {
+ const deceasedDelegations = delegations.filter((d) =>
+ deceasedNationalIds.includes(d.fromNationalId),
+ )
// Delete all deceased delegations by deleting them and their scopes.
const deletePromises = deceasedDelegations.map((delegation) =>
delegation.destroy(),
@@ -286,7 +299,12 @@ export class DelegationsIncomingCustomService {
})
}
- return { delegations: aliveDelegations, fromNameInfo }
+ return {
+ delegations: delegations.filter((d) =>
+ aliveNationalIds.includes(d.fromNationalId),
+ ),
+ fromNameInfo: aliveNameInfo,
+ }
}
private async findAllIncoming(
@@ -298,7 +316,8 @@ export class DelegationsIncomingCustomService {
validity: DelegationValidity
},
useMaster = false,
- ): Promise<{ delegations: Delegation[]; fromNameInfo: FromNameInfo[] }> {
+ user?: User,
+ ): Promise<{ delegations: Delegation[]; fromNameInfo: NameInfo[] }> {
let whereOptions = getScopeValidityWhereClause(validity)
if (domainName) whereOptions = { ...whereOptions, domainName: domainName }
@@ -336,10 +355,22 @@ export class DelegationsIncomingCustomService {
})
// Check live status, i.e. dead or alive for delegations
- const { aliveDelegations, deceasedDelegations, fromNameInfo } =
- await this.getLiveStatusFromDelegations(delegations)
+ const isNationalRegistryV3DeceasedStatusEnabled =
+ await this.nationalRegistryV3FeatureService.getValue(user)
+
+ const { aliveNationalIds, deceasedNationalIds, aliveNameInfo } =
+ await this.aliveStatusService.getStatus(
+ delegations.map((d) => ({
+ nationalId: d.fromNationalId,
+ name: d.fromDisplayName,
+ })),
+ isNationalRegistryV3DeceasedStatusEnabled,
+ )
- if (deceasedDelegations.length > 0) {
+ if (deceasedNationalIds.length > 0) {
+ const deceasedDelegations = delegations.filter((d) =>
+ deceasedNationalIds.includes(d.fromNationalId),
+ )
// Delete all deceased delegations by deleting them and their scopes.
const deletePromises = deceasedDelegations.map((delegation) =>
delegation.destroy(),
@@ -354,7 +385,12 @@ export class DelegationsIncomingCustomService {
})
}
- return { delegations: aliveDelegations, fromNameInfo }
+ return {
+ delegations: delegations.filter((d) =>
+ aliveNationalIds.includes(d.fromNationalId),
+ ),
+ fromNameInfo: aliveNameInfo,
+ }
}
private checkIfScopeIsValid(
@@ -389,103 +425,14 @@ export class DelegationsIncomingCustomService {
return false
}
- /**
- * Divides delegations into alive and deceased delegations
- * - Makes calls for every delegation to NationalRegistry to check if the person exists.
- * - Divides the delegations into alive and deceased delegations, based on
- * 1. All companies will be divided into alive delegations.
- * 2. If the person exists in NationalRegistry, then the delegation is alive.
- */
- private async getLiveStatusFromDelegations(
- delegations: Delegation[],
- ): Promise<{
- aliveDelegations: Delegation[]
- deceasedDelegations: Delegation[]
- fromNameInfo: FromNameInfo[]
- }> {
- if (delegations.length === 0) {
- return {
- aliveDelegations: [],
- deceasedDelegations: [],
- fromNameInfo: [],
- }
- }
-
- const delegationsPromises = delegations.map(({ fromNationalId }) =>
- kennitala.isCompany(fromNationalId)
- ? this.companyRegistryClient
- .getCompany(fromNationalId)
- .catch(this.handlerGetError)
- : this.nationalRegistryClient
- .getIndividual(fromNationalId)
- .catch(this.handlerGetError),
- )
-
- try {
- // Check if delegations is linked to a person, i.e. not deceased
- const identities = await Promise.all(delegationsPromises)
- const identitiesValuesNoError = identities
- .filter(this.isNotError)
- .filter(isDefined)
- .map((identity) => ({
- nationalId: identity.nationalId,
- name: identity.name ?? UNKNOWN_NAME,
- }))
-
- // Divide delegations into alive or deceased delegations.
- const [aliveDelegations, deceasedDelegations] = partitionWithIndex(
- delegations,
- ({ fromNationalId }, index) =>
- // All companies will be divided into aliveDelegations
- kennitala.isCompany(fromNationalId) ||
- // Pass through although Þjóðskrá API throws an error since it is not required to view the delegation.
- identities[index] instanceof Error ||
- // Make sure we can match the person to the delegation, i.e. not deceased
- (identities[index] as IndividualDto)?.nationalId === fromNationalId,
- )
-
- return {
- aliveDelegations,
- deceasedDelegations,
- fromNameInfo: identitiesValuesNoError,
- }
- } catch (error) {
- this.logger.error(
- `Error getting live status from delegations. Delegations: ${delegations.map(
- (d) => d.id,
- )}`,
- error,
- )
-
- // We do not want to fail the whole request if we cannot get the live status from delegations.
- // Therefore, we return all delegations as alive delegations.
- return {
- aliveDelegations: delegations,
- deceasedDelegations: [],
- fromNameInfo: [],
- }
- }
- }
-
- private handlerGetError(error: null | Error) {
- return error
- }
-
- /**
- * Checks if item is not an instance of Error
- */
- private isNotError(item: T | Error): item is T {
- return item instanceof Error === false
- }
-
/**
* Finds person by nationalId.
*/
private getPersonByNationalId(
- identities: Array,
+ identities: Array | undefined,
nationalId: string,
) {
- return identities.find((identity) => identity?.nationalId === nationalId)
+ return identities?.find((identity) => identity?.nationalId === nationalId)
}
private async findAccessControlList(
diff --git a/libs/auth-api-lib/src/lib/delegations/delegations-incoming-representative.service.ts b/libs/auth-api-lib/src/lib/delegations/delegations-incoming-representative.service.ts
index b4380ad4aef1..42d9ccb0a540 100644
--- a/libs/auth-api-lib/src/lib/delegations/delegations-incoming-representative.service.ts
+++ b/libs/auth-api-lib/src/lib/delegations/delegations-incoming-representative.service.ts
@@ -1,9 +1,6 @@
import { Inject, Logger } from '@nestjs/common'
-import {
- IndividualDto,
- NationalRegistryClientService,
-} from '@island.is/clients/national-registry-v2'
+import { User } from '@island.is/auth-nest-tools'
import { LOGGER_PROVIDER } from '@island.is/logging'
import { AuditService } from '@island.is/nest/audit'
import {
@@ -14,10 +11,11 @@ import { isDefined } from '@island.is/shared/utils'
import { PersonalRepresentativeDTO } from '../personal-representative/dto/personal-representative.dto'
import { PersonalRepresentativeService } from '../personal-representative/services/personalRepresentative.service'
+import { AliveStatusService } from './alive-status.service'
import { UNKNOWN_NAME } from './constants/names'
import { ApiScopeInfo } from './delegations-incoming.service'
import { DelegationDTO } from './dto/delegation.dto'
-import { partitionWithIndex } from './utils/partitionWithIndex'
+import { NationalRegistryV3FeatureService } from './national-registry-v3-feature.service'
type FindAllIncomingOptions = {
nationalId: string
@@ -28,10 +26,11 @@ type FindAllIncomingOptions = {
export class DelegationsIncomingRepresentativeService {
constructor(
private prService: PersonalRepresentativeService,
- private nationalRegistryClient: NationalRegistryClientService,
+ private aliveStatusService: AliveStatusService,
@Inject(LOGGER_PROVIDER)
private logger: Logger,
private auditService: AuditService,
+ private readonly nationalRegistryV3FeatureService: NationalRegistryV3FeatureService,
) {}
async findAllIncoming(
@@ -41,6 +40,7 @@ export class DelegationsIncomingRepresentativeService {
requireApiScopes,
}: FindAllIncomingOptions,
useMaster = false,
+ user?: User,
): Promise {
if (
requireApiScopes &&
@@ -95,42 +95,35 @@ export class DelegationsIncomingRepresentativeService {
}
})
- const personPromises = personalRepresentatives.map(
- ({ nationalIdRepresentedPerson }) =>
- this.nationalRegistryClient
- .getIndividual(nationalIdRepresentedPerson)
- .catch(this.handlerGetIndividualError),
- )
+ const isNationalRegistryV3DeceasedStatusEnabled =
+ await this.nationalRegistryV3FeatureService.getValue(user)
- const persons = await Promise.all(personPromises)
- const personsValues = persons.filter((person) => person !== undefined)
- const personsValuesNoError = personsValues.filter(this.isNotError)
-
- // Divide personal representatives into alive or deceased.
- const [alive, deceased] = partitionWithIndex(
- personalRepresentatives,
- ({ nationalIdRepresentedPerson }, index) =>
- // Pass through although Þjóðskrá API throws an error since it is not required to view the personal representative.
- persons[index] instanceof Error ||
- // Make sure we can match the person to the personal representatives, i.e. not deceased
- (persons[index] as IndividualDto)?.nationalId ===
- nationalIdRepresentedPerson,
- )
+ const { aliveNationalIds, deceasedNationalIds, aliveNameInfo } =
+ await this.aliveStatusService.getStatus(
+ personalRepresentatives.map((d) => ({
+ nationalId: d.nationalIdRepresentedPerson,
+ name: UNKNOWN_NAME,
+ })),
+ isNationalRegistryV3DeceasedStatusEnabled,
+ )
- if (deceased.length > 0) {
+ if (deceasedNationalIds.length > 0) {
+ const deceased = personalRepresentatives.filter((pr) =>
+ deceasedNationalIds.includes(pr.nationalIdRepresentedPerson),
+ )
await this.makePersonalRepresentativesInactive(deceased)
}
- return alive
- .map((pr) => {
- const person = this.getPersonByNationalId(
- personsValuesNoError,
- pr.nationalIdRepresentedPerson,
- )
+ const alive = personalRepresentatives.filter((pr) =>
+ aliveNationalIds.includes(pr.nationalIdRepresentedPerson),
+ )
- return toDelegationDTO(person?.name ?? UNKNOWN_NAME, pr)
- })
- .filter(isDefined)
+ return alive.map((pr) => {
+ const person = aliveNameInfo.find(
+ (n) => n.nationalId === pr.nationalIdRepresentedPerson,
+ )
+ return toDelegationDTO(person?.name ?? UNKNOWN_NAME, pr)
+ })
} catch (error) {
this.logger.error('Error in findAllRepresentedPersons', error)
}
@@ -154,25 +147,4 @@ export class DelegationsIncomingRepresentativeService {
system: true,
})
}
-
- private handlerGetIndividualError(error: null | Error) {
- return error
- }
-
- /**
- * Finds person by nationalId.
- */
- private getPersonByNationalId(
- persons: Array,
- nationalId: string,
- ) {
- return persons.find((person) => person?.nationalId === nationalId)
- }
-
- /**
- * Checks if item is not an instance of Error
- */
- private isNotError(item: T | Error): item is T {
- return item instanceof Error === false
- }
}
diff --git a/libs/auth-api-lib/src/lib/delegations/delegations-incoming.service.ts b/libs/auth-api-lib/src/lib/delegations/delegations-incoming.service.ts
index 1792164ddb48..b2e0775d3b48 100644
--- a/libs/auth-api-lib/src/lib/delegations/delegations-incoming.service.ts
+++ b/libs/auth-api-lib/src/lib/delegations/delegations-incoming.service.ts
@@ -2,10 +2,6 @@ import { BadRequestException, Injectable } from '@nestjs/common'
import { InjectModel } from '@nestjs/sequelize'
import { User } from '@island.is/auth-nest-tools'
-import {
- IndividualDto,
- NationalRegistryClientService,
-} from '@island.is/clients/national-registry-v2'
import { SyslumennService } from '@island.is/clients/syslumenn'
import { logger } from '@island.is/logging'
import { FeatureFlagService, Features } from '@island.is/nest/feature-flags'
@@ -19,6 +15,7 @@ import { ClientDelegationType } from '../clients/models/client-delegation-type.m
import { Client } from '../clients/models/client.model'
import { ApiScopeDelegationType } from '../resources/models/api-scope-delegation-type.model'
import { ApiScope } from '../resources/models/api-scope.model'
+import { AliveStatusService, NameInfo } from './alive-status.service'
import { UNKNOWN_NAME } from './constants/names'
import { DelegationDTOMapper } from './delegation-dto.mapper'
import { DelegationProviderService } from './delegation-provider.service'
@@ -27,8 +24,10 @@ import { DelegationsIncomingCustomService } from './delegations-incoming-custom.
import { DelegationsIncomingRepresentativeService } from './delegations-incoming-representative.service'
import { DelegationsIncomingWardService } from './delegations-incoming-ward.service'
import { DelegationsIndexService } from './delegations-index.service'
+import { DelegationRecordDTO } from './dto/delegation-index.dto'
import { DelegationDTO } from './dto/delegation.dto'
import { MergedDelegationDTO } from './dto/merged-delegation.dto'
+import { NationalRegistryV3FeatureService } from './national-registry-v3-feature.service'
type ClientDelegationInfo = Pick<
Client,
@@ -66,9 +65,10 @@ export class DelegationsIncomingService {
private delegationsIncomingWardService: DelegationsIncomingWardService,
private delegationsIndexService: DelegationsIndexService,
private delegationProviderService: DelegationProviderService,
- private nationalRegistryClient: NationalRegistryClientService,
+ private aliveStatusService: AliveStatusService,
private readonly featureFlagService: FeatureFlagService,
private readonly syslumennService: SyslumennService,
+ private readonly nationalRegistryV3FeatureService: NationalRegistryV3FeatureService,
) {}
async findAllValid(
@@ -96,22 +96,34 @@ export class DelegationsIncomingService {
)
delegationPromises.push(
- this.delegationsIncomingCustomService.findAllValidIncoming({
- nationalId: user.nationalId,
- domainName,
- }),
+ this.delegationsIncomingCustomService.findAllValidIncoming(
+ {
+ nationalId: user.nationalId,
+ domainName,
+ },
+ false,
+ user,
+ ),
)
delegationPromises.push(
- this.delegationsIncomingCustomService.findAllValidGeneralMandate({
- nationalId: user.nationalId,
- }),
+ this.delegationsIncomingCustomService.findAllValidGeneralMandate(
+ {
+ nationalId: user.nationalId,
+ },
+ false,
+ user,
+ ),
)
delegationPromises.push(
- this.delegationsIncomingRepresentativeService.findAllIncoming({
- nationalId: user.nationalId,
- }),
+ this.delegationsIncomingRepresentativeService.findAllIncoming(
+ {
+ nationalId: user.nationalId,
+ },
+ false,
+ user,
+ ),
)
const delegationSets = await Promise.all(delegationPromises)
@@ -211,11 +223,15 @@ export class DelegationsIncomingService {
) {
delegationPromises.push(
this.delegationsIncomingRepresentativeService
- .findAllIncoming({
- nationalId: user.nationalId,
- clientAllowedApiScopes,
- requireApiScopes: client.requireApiScopes,
- })
+ .findAllIncoming(
+ {
+ nationalId: user.nationalId,
+ clientAllowedApiScopes,
+ requireApiScopes: client.requireApiScopes,
+ },
+ false,
+ user,
+ )
.then((ds) =>
ds.map((d) => DelegationDTOMapper.toMergedDelegationDTO(d)),
),
@@ -335,11 +351,64 @@ export class DelegationsIncomingService {
clientAllowedApiScopes,
requireApiScopes,
)
- const merged = records.map((d) =>
- DelegationDTOMapper.recordToMergedDelegationDTO(d),
- )
- await Promise.all(merged.map((d) => this.updateName(d)))
+ const isNationalRegistryV3DeceasedStatusEnabled =
+ await this.nationalRegistryV3FeatureService.getValue(user)
+
+ const { aliveNationalIds, deceasedNationalIds, aliveNameInfo } =
+ await this.aliveStatusService.getStatus(
+ Array.from(
+ new Set(
+ records.map((d) => ({
+ nationalId: d.fromNationalId,
+ name: UNKNOWN_NAME,
+ })),
+ ),
+ ),
+ isNationalRegistryV3DeceasedStatusEnabled,
+ )
+
+ if (deceasedNationalIds.length > 0) {
+ const deceasedDelegations = records.filter((d) =>
+ deceasedNationalIds.includes(d.fromNationalId),
+ )
+ // Delete all deceased delegations from index
+ const deletePromises = deceasedDelegations.map((delegation) => {
+ this.delegationsIndexService.removeDelegationRecord(
+ {
+ fromNationalId: delegation.fromNationalId,
+ toNationalId: delegation.toNationalId,
+ type: delegation.type,
+ provider: AuthDelegationProvider.DistrictCommissionersRegistry,
+ },
+ user,
+ )
+ })
+
+ const results = await Promise.allSettled(deletePromises)
+ results.forEach((result, index) => {
+ if (result.status === 'rejected') {
+ logger.error('Failed to remove delegation record', {
+ error: result.reason,
+ delegation: deceasedDelegations[index],
+ })
+ }
+ })
+ }
+
+ const aliveNationalIdSet = new Set(aliveNationalIds)
+ const merged = records.reduce(
+ (acc: MergedDelegationDTO[], d: DelegationRecordDTO) => {
+ if (aliveNationalIdSet.has(d.fromNationalId)) {
+ acc.push({
+ ...DelegationDTOMapper.recordToMergedDelegationDTO(d),
+ fromName: this.getNameFromNameInfo(d.fromNationalId, aliveNameInfo),
+ })
+ }
+ return acc
+ },
+ [],
+ )
return merged
}
@@ -381,17 +450,12 @@ export class DelegationsIncomingService {
})
}
- private async updateName(
- mergedDelegation: MergedDelegationDTO,
- ): Promise {
- try {
- const fromIndividual: IndividualDto | null =
- await this.nationalRegistryClient.getIndividual(
- mergedDelegation.fromNationalId,
- )
- mergedDelegation.fromName = fromIndividual?.name ?? UNKNOWN_NAME
- } catch (error) {
- mergedDelegation.fromName = UNKNOWN_NAME
- }
+ private getNameFromNameInfo(
+ nationalId: string,
+ nameInfo: NameInfo[],
+ ): string {
+ return (
+ nameInfo.find((n) => n.nationalId === nationalId)?.name ?? UNKNOWN_NAME
+ )
}
}
diff --git a/libs/auth-api-lib/src/lib/delegations/delegations.module.ts b/libs/auth-api-lib/src/lib/delegations/delegations.module.ts
index 21aae6fe43e4..031423b76886 100644
--- a/libs/auth-api-lib/src/lib/delegations/delegations.module.ts
+++ b/libs/auth-api-lib/src/lib/delegations/delegations.module.ts
@@ -3,10 +3,11 @@ import { SequelizeModule } from '@nestjs/sequelize'
import { RskRelationshipsClientModule } from '@island.is/clients-rsk-relationships'
import { NationalRegistryClientModule } from '@island.is/clients/national-registry-v2'
+import { NationalRegistryV3ClientModule } from '@island.is/clients/national-registry-v3'
import { CompanyRegistryClientModule } from '@island.is/clients/rsk/company-registry'
import { SyslumennClientModule } from '@island.is/clients/syslumenn'
-import { FeatureFlagModule } from '@island.is/nest/feature-flags'
import { ZendeskModule } from '@island.is/clients/zendesk'
+import { FeatureFlagModule } from '@island.is/nest/feature-flags'
import { ClientAllowedScope } from '../clients/models/client-allowed-scope.model'
import { Client } from '../clients/models/client.model'
@@ -19,6 +20,7 @@ import { ResourcesModule } from '../resources/resources.module'
import { UserIdentitiesModule } from '../user-identities/user-identities.module'
import { UserSystemNotificationModule } from '../user-notification'
import { DelegationAdminCustomService } from './admin/delegation-admin-custom.service'
+import { AliveStatusService } from './alive-status.service'
import { DelegationProviderService } from './delegation-provider.service'
import { DelegationScopeService } from './delegation-scope.service'
import { IncomingDelegationsCompanyService } from './delegations-incoming-company.service'
@@ -37,12 +39,14 @@ import { DelegationScope } from './models/delegation-scope.model'
import { DelegationTypeModel } from './models/delegation-type.model'
import { Delegation } from './models/delegation.model'
import { NamesService } from './names.service'
+import { NationalRegistryV3FeatureService } from './national-registry-v3-feature.service'
@Module({
imports: [
ResourcesModule,
PersonalRepresentativeModule,
NationalRegistryClientModule,
+ NationalRegistryV3ClientModule,
RskRelationshipsClientModule,
CompanyRegistryClientModule,
ZendeskModule,
@@ -79,6 +83,8 @@ import { NamesService } from './names.service'
DelegationsIndexService,
DelegationProviderService,
DelegationAdminCustomService,
+ AliveStatusService,
+ NationalRegistryV3FeatureService,
],
exports: [
DelegationsService,
diff --git a/libs/auth-api-lib/src/lib/delegations/national-registry-v3-feature.service.ts b/libs/auth-api-lib/src/lib/delegations/national-registry-v3-feature.service.ts
new file mode 100644
index 000000000000..e750dbc12126
--- /dev/null
+++ b/libs/auth-api-lib/src/lib/delegations/national-registry-v3-feature.service.ts
@@ -0,0 +1,17 @@
+import { Injectable } from '@nestjs/common'
+
+import { User } from '@island.is/auth-nest-tools'
+import { FeatureFlagService, Features } from '@island.is/nest/feature-flags'
+
+@Injectable()
+export class NationalRegistryV3FeatureService {
+ constructor(private readonly featureFlagService: FeatureFlagService) {}
+
+ getValue(user?: User): Promise {
+ return this.featureFlagService.getValue(
+ Features.isNationalRegistryV3DeceasedStatusEnabled,
+ false,
+ user,
+ )
+ }
+}
diff --git a/libs/feature-flags/src/lib/features.ts b/libs/feature-flags/src/lib/features.ts
index ae839cdd1690..fc77efa64398 100644
--- a/libs/feature-flags/src/lib/features.ts
+++ b/libs/feature-flags/src/lib/features.ts
@@ -106,6 +106,9 @@ export enum Features {
// General mandate delegation type
isGeneralMandateDelegationEnabled = 'isGeneralMandateDelegationEnabled',
+
+ // Should auth api use national registry v3 for checking deceased status
+ isNationalRegistryV3DeceasedStatusEnabled = 'isNationalRegistryV3DeceasedStatusEnabled',
}
export enum ServerSideFeature {
From f66ba3cbe94f6108dcabf2b0c0edb73f8d43c173 Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?=C3=8Dvar=20Oddsson?=
Date: Tue, 26 Nov 2024 23:03:26 +0000
Subject: [PATCH 09/12] feat(j-s): Enable public prosecutors to properly handle
fines (#17023)
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
* Checkpoint
* Refactor AlertMessage
* Format date
* Cleanup
* Cleanup
* Merge
* Add key
* Refactor
* Remove console.log
* Merge
* Merge
* Add new BlueBox for FINES
* Move code around
* Add tests
* Fix tests
* Show appeal deadline in BlueBoxWithDate
* Swap out words based on whether or not there is a fine
* Swap out words based on whether or not there is a fine
* Show FINES to prison admins
* Revert island-ui changes
* Cleanup
* Cleanup
* Fix tests
* Fix tests
* Add tests
* Fix lint
* Checkpoint
* Fix dates
* Fix dates
* Fix texts
* Add tests
* Refactoring
* Refactoring
* Use constants
---------
Co-authored-by: kodiakhq[bot] <49736102+kodiakhq[bot]@users.noreply.github.com>
Co-authored-by: Guðjón Guðjónsson
---
.../interceptors/case.transformer.spec.ts | 48 +++++++++++
.../case/interceptors/case.transformer.ts | 21 +++--
.../app/modules/case/filters/case.filter.ts | 4 +-
.../app/modules/case/filters/cases.filter.ts | 7 +-
.../case/filters/test/cases.filter.spec.ts | 7 +-
.../test/prisonAdminUserFilter.spec.ts | 1 +
.../BlueBoxWithIcon/BlueBoxWithDate.spec.tsx | 80 +++++++++++++++++++
.../BlueBoxWithDate.strings.ts | 13 +++
.../BlueBoxWithIcon/BlueBoxWithDate.tsx | 77 +++++++++++-------
.../InfoCard/useInfoCardItems.strings.ts | 10 ++-
.../components/InfoCard/useInfoCardItems.tsx | 6 ++
.../web/src/components/index.ts | 1 +
.../Indictments/Overview/Overview.strings.ts | 4 +-
.../Indictments/Overview/Overview.tsx | 14 ++--
.../Tables/CasesReviewed.strings.ts | 6 ++
.../PublicProsecutor/Tables/CasesReviewed.tsx | 31 ++++---
.../ReviewDecision/ReviewDecision.css.ts | 9 ++-
.../ReviewDecision/ReviewDecision.strings.ts | 27 +++++--
.../ReviewDecision/ReviewDecision.tsx | 34 +++++---
.../IndictmentOverview/IndictmentOverview.tsx | 7 ++
libs/judicial-system/types/src/index.ts | 1 +
.../types/src/lib/indictmentCase.ts | 11 ++-
22 files changed, 342 insertions(+), 77 deletions(-)
create mode 100644 apps/judicial-system/web/src/components/BlueBoxWithIcon/BlueBoxWithDate.spec.tsx
diff --git a/apps/judicial-system/api/src/app/modules/case/interceptors/case.transformer.spec.ts b/apps/judicial-system/api/src/app/modules/case/interceptors/case.transformer.spec.ts
index 5c76205ba8d7..f73423c0f6d4 100644
--- a/apps/judicial-system/api/src/app/modules/case/interceptors/case.transformer.spec.ts
+++ b/apps/judicial-system/api/src/app/modules/case/interceptors/case.transformer.spec.ts
@@ -731,6 +731,54 @@ describe('getIndictmentInfo', () => {
indictmentVerdictAppealDeadlineExpired: false,
})
})
+
+ it('should return correct indictment info when the indictment ruling decision is FINE and the appeal deadline is not expired', () => {
+ const rulingDate = new Date().toISOString()
+ const defendants = [
+ {
+ serviceRequirement: ServiceRequirement.NOT_REQUIRED,
+ verdictViewDate: undefined,
+ } as Defendant,
+ ]
+
+ const indictmentInfo = getIndictmentInfo(
+ CaseIndictmentRulingDecision.FINE,
+ rulingDate,
+ defendants,
+ )
+
+ expect(indictmentInfo).toEqual({
+ indictmentAppealDeadline: new Date(
+ new Date(rulingDate).setDate(new Date(rulingDate).getDate() + 3),
+ ).toISOString(),
+ indictmentVerdictViewedByAll: true,
+ indictmentVerdictAppealDeadlineExpired: false,
+ })
+ })
+
+ it('should return correct indictment info when the indictment ruling decision is FINE and the appeal deadline is expired', () => {
+ const rulingDate = '2024-05-26T21:51:19.156Z'
+ const defendants = [
+ {
+ serviceRequirement: ServiceRequirement.NOT_REQUIRED,
+ verdictViewDate: undefined,
+ } as Defendant,
+ ]
+
+ const indictmentInfo = getIndictmentInfo(
+ CaseIndictmentRulingDecision.FINE,
+ rulingDate,
+ defendants,
+ )
+
+ expect(indictmentInfo).toEqual({
+ indictmentAppealDeadline: new Date(
+ new Date(rulingDate).setDate(new Date(rulingDate).getDate() + 3),
+ ).toISOString(),
+ indictmentVerdictViewedByAll: true,
+ indictmentVerdictAppealDeadlineExpired: true,
+ })
+ })
})
describe('getIndictmentDefendantsInfo', () => {
diff --git a/apps/judicial-system/api/src/app/modules/case/interceptors/case.transformer.ts b/apps/judicial-system/api/src/app/modules/case/interceptors/case.transformer.ts
index bf8139e14014..f01909d284f1 100644
--- a/apps/judicial-system/api/src/app/modules/case/interceptors/case.transformer.ts
+++ b/apps/judicial-system/api/src/app/modules/case/interceptors/case.transformer.ts
@@ -2,6 +2,7 @@ import {
CaseAppealDecision,
CaseIndictmentRulingDecision,
EventType,
+ FINE_APPEAL_WINDOW_DAYS,
getIndictmentVerdictAppealDeadlineStatus,
getStatementDeadline,
isRequestCase,
@@ -151,6 +152,8 @@ export const getIndictmentInfo = (
eventLog?: EventLog[],
): IndictmentInfo => {
const indictmentInfo: IndictmentInfo = {}
+ const isFine = rulingDecision === CaseIndictmentRulingDecision.FINE
+ const isRuling = rulingDecision === CaseIndictmentRulingDecision.RULING
if (!rulingDate) {
return indictmentInfo
@@ -158,13 +161,16 @@ export const getIndictmentInfo = (
const theRulingDate = new Date(rulingDate)
indictmentInfo.indictmentAppealDeadline = new Date(
- theRulingDate.getTime() + getDays(VERDICT_APPEAL_WINDOW_DAYS),
+ theRulingDate.getTime() +
+ getDays(isFine ? FINE_APPEAL_WINDOW_DAYS : VERDICT_APPEAL_WINDOW_DAYS),
).toISOString()
const verdictInfo = defendants?.map<[boolean, Date | undefined]>(
(defendant) => [
- rulingDecision === CaseIndictmentRulingDecision.RULING,
- defendant.serviceRequirement === ServiceRequirement.NOT_REQUIRED
+ isRuling || isFine,
+ isFine
+ ? theRulingDate
+ : defendant.serviceRequirement === ServiceRequirement.NOT_REQUIRED
? new Date()
: defendant.verdictViewDate
? new Date(defendant.verdictViewDate)
@@ -173,7 +179,7 @@ export const getIndictmentInfo = (
)
const [indictmentVerdictViewedByAll, indictmentVerdictAppealDeadlineExpired] =
- getIndictmentVerdictAppealDeadlineStatus(verdictInfo)
+ getIndictmentVerdictAppealDeadlineStatus(verdictInfo, isFine)
indictmentInfo.indictmentVerdictViewedByAll = indictmentVerdictViewedByAll
indictmentInfo.indictmentVerdictAppealDeadlineExpired =
indictmentVerdictAppealDeadlineExpired
@@ -189,6 +195,8 @@ export const getIndictmentDefendantsInfo = (theCase: Case) => {
return theCase.defendants?.map((defendant) => {
const serviceRequired =
defendant.serviceRequirement === ServiceRequirement.REQUIRED
+ const isFine =
+ theCase.indictmentRulingDecision === CaseIndictmentRulingDecision.FINE
const { verdictViewDate } = defendant
@@ -196,7 +204,10 @@ export const getIndictmentDefendantsInfo = (theCase: Case) => {
const verdictAppealDeadline = baseDate
? new Date(
- new Date(baseDate).getTime() + getDays(VERDICT_APPEAL_WINDOW_DAYS),
+ new Date(baseDate).getTime() +
+ getDays(
+ isFine ? FINE_APPEAL_WINDOW_DAYS : VERDICT_APPEAL_WINDOW_DAYS,
+ ),
).toISOString()
: undefined
diff --git a/apps/judicial-system/backend/src/app/modules/case/filters/case.filter.ts b/apps/judicial-system/backend/src/app/modules/case/filters/case.filter.ts
index e4200b4dd5ee..36dbfdc8671c 100644
--- a/apps/judicial-system/backend/src/app/modules/case/filters/case.filter.ts
+++ b/apps/judicial-system/backend/src/app/modules/case/filters/case.filter.ts
@@ -277,7 +277,9 @@ const canPrisonAdminUserAccessCase = (
// Check case indictment ruling decision access
if (
- theCase.indictmentRulingDecision !== CaseIndictmentRulingDecision.RULING
+ theCase.indictmentRulingDecision !==
+ CaseIndictmentRulingDecision.RULING &&
+ theCase.indictmentRulingDecision !== CaseIndictmentRulingDecision.FINE
) {
return false
}
diff --git a/apps/judicial-system/backend/src/app/modules/case/filters/cases.filter.ts b/apps/judicial-system/backend/src/app/modules/case/filters/cases.filter.ts
index 34ae100c239d..bc5f2f9f3a5f 100644
--- a/apps/judicial-system/backend/src/app/modules/case/filters/cases.filter.ts
+++ b/apps/judicial-system/backend/src/app/modules/case/filters/cases.filter.ts
@@ -211,7 +211,12 @@ const getPrisonAdminUserCasesQueryFilter = (): WhereOptions => {
{
type: indictmentCases,
state: CaseState.COMPLETED,
- indictment_ruling_decision: CaseIndictmentRulingDecision.RULING,
+ indictment_ruling_decision: {
+ [Op.or]: [
+ CaseIndictmentRulingDecision.RULING,
+ CaseIndictmentRulingDecision.FINE,
+ ],
+ },
indictment_review_decision: IndictmentCaseReviewDecision.ACCEPT,
id: {
[Op.in]: Sequelize.literal(`
diff --git a/apps/judicial-system/backend/src/app/modules/case/filters/test/cases.filter.spec.ts b/apps/judicial-system/backend/src/app/modules/case/filters/test/cases.filter.spec.ts
index cb6a69ebaf7c..4d6027e5d5b4 100644
--- a/apps/judicial-system/backend/src/app/modules/case/filters/test/cases.filter.spec.ts
+++ b/apps/judicial-system/backend/src/app/modules/case/filters/test/cases.filter.spec.ts
@@ -388,7 +388,12 @@ describe('getCasesQueryFilter', () => {
{
type: indictmentCases,
state: CaseState.COMPLETED,
- indictment_ruling_decision: CaseIndictmentRulingDecision.RULING,
+ indictment_ruling_decision: {
+ [Op.or]: [
+ CaseIndictmentRulingDecision.RULING,
+ CaseIndictmentRulingDecision.FINE,
+ ],
+ },
indictment_review_decision: IndictmentCaseReviewDecision.ACCEPT,
id: {
[Op.in]: Sequelize.literal(`
diff --git a/apps/judicial-system/backend/src/app/modules/case/filters/test/prisonAdminUserFilter.spec.ts b/apps/judicial-system/backend/src/app/modules/case/filters/test/prisonAdminUserFilter.spec.ts
index 570de65e3f81..9d147a4ca169 100644
--- a/apps/judicial-system/backend/src/app/modules/case/filters/test/prisonAdminUserFilter.spec.ts
+++ b/apps/judicial-system/backend/src/app/modules/case/filters/test/prisonAdminUserFilter.spec.ts
@@ -132,6 +132,7 @@ describe.each(prisonSystemRoles)('prison admin user %s', (role) => {
(state) => {
const accessibleCaseIndictmentRulingDecisions = [
CaseIndictmentRulingDecision.RULING,
+ CaseIndictmentRulingDecision.FINE,
]
describe.each(
diff --git a/apps/judicial-system/web/src/components/BlueBoxWithIcon/BlueBoxWithDate.spec.tsx b/apps/judicial-system/web/src/components/BlueBoxWithIcon/BlueBoxWithDate.spec.tsx
new file mode 100644
index 000000000000..48ac69307bbc
--- /dev/null
+++ b/apps/judicial-system/web/src/components/BlueBoxWithIcon/BlueBoxWithDate.spec.tsx
@@ -0,0 +1,80 @@
+import faker from 'faker'
+import { render, screen } from '@testing-library/react'
+
+import {
+ CaseIndictmentRulingDecision,
+ CaseType,
+ Defendant,
+} from '../../graphql/schema'
+import { mockCase } from '../../utils/mocks'
+import {
+ ApolloProviderWrapper,
+ FormContextWrapper,
+ IntlProviderWrapper,
+} from '../../utils/testHelpers'
+import BlueBoxWithDate from './BlueBoxWithDate'
+
+jest.mock('next/router', () => ({
+ useRouter() {
+ return {
+ pathname: '',
+ query: {
+ id: 'test_id',
+ },
+ }
+ },
+}))
+
+describe('BlueBoxWithDate', () => {
+ const name = faker.name.firstName()
+ const rulingDate = new Date().toISOString()
+
+ const mockDefendant: Defendant = {
+ name,
+ id: faker.datatype.uuid(),
+ }
+
+ it('renders correctly when ruling decision is FINE', () => {
+ render(
+
+
+
+
+
+
+ ,
+ )
+
+ expect(screen.getByText('Viðurlagaákvörðun')).toBeInTheDocument()
+ expect(screen.getByText(name)).toBeInTheDocument()
+ })
+
+ it('renders correctly when ruling decision is RULING', () => {
+ render(
+
+
+
+
+
+
+ ,
+ )
+
+ expect(screen.getByText('Birting dóms')).toBeInTheDocument()
+ expect(screen.getByText(name)).toBeInTheDocument()
+ })
+})
diff --git a/apps/judicial-system/web/src/components/BlueBoxWithIcon/BlueBoxWithDate.strings.ts b/apps/judicial-system/web/src/components/BlueBoxWithIcon/BlueBoxWithDate.strings.ts
index a68972df8229..a3f57cbebd4c 100644
--- a/apps/judicial-system/web/src/components/BlueBoxWithIcon/BlueBoxWithDate.strings.ts
+++ b/apps/judicial-system/web/src/components/BlueBoxWithIcon/BlueBoxWithDate.strings.ts
@@ -70,4 +70,17 @@ export const strings = defineMessages({
description:
'Notaður sem texti í valmynd fyrir aðgerðina að senda mál til fullnustu',
},
+ indictmentRulingDecisionFine: {
+ id: 'judicial.system.core:blue_box_with_date.indictment_ruling_decision_fine',
+ defaultMessage: 'Viðurlagaákvörðun',
+ description:
+ 'Notaður sem titill í svæði þar sem kærufrestur viðurlagaákvörðunar er tekinn fram',
+ },
+ fineAppealDeadline: {
+ id: 'judicial.system.core:blue_box_with_date.fine_appeal_deadline',
+ defaultMessage:
+ 'Kærufrestur Ríkissaksóknara {appealDeadlineIsInThePast, select, true {var} other {er}} til {appealDeadline}',
+ description:
+ 'Notaður sem titill í svæði þar sem kærufrestur viðurlagaákvörðunar er tekinn fram',
+ },
})
diff --git a/apps/judicial-system/web/src/components/BlueBoxWithIcon/BlueBoxWithDate.tsx b/apps/judicial-system/web/src/components/BlueBoxWithIcon/BlueBoxWithDate.tsx
index 6a404390e8c9..50ecd41a0c44 100644
--- a/apps/judicial-system/web/src/components/BlueBoxWithIcon/BlueBoxWithDate.tsx
+++ b/apps/judicial-system/web/src/components/BlueBoxWithIcon/BlueBoxWithDate.tsx
@@ -18,8 +18,8 @@ import { VERDICT_APPEAL_WINDOW_DAYS } from '@island.is/judicial-system/types'
import { errors } from '@island.is/judicial-system-web/messages'
import {
+ CaseIndictmentRulingDecision,
Defendant,
- IndictmentCaseReviewDecision,
ServiceRequirement,
} from '../../graphql/schema'
import { formatDateForServer, useDefendants } from '../../utils/hooks'
@@ -32,12 +32,11 @@ import * as styles from './BlueBoxWithIcon.css'
interface Props {
defendant: Defendant
- indictmentReviewDecision?: IndictmentCaseReviewDecision | null
icon?: IconMapIcon
}
const BlueBoxWithDate: FC = (props) => {
- const { defendant, indictmentReviewDecision, icon } = props
+ const { defendant, icon } = props
const { formatMessage } = useIntl()
const [dates, setDates] = useState<{
verdictViewDate?: Date
@@ -52,6 +51,9 @@ const BlueBoxWithDate: FC = (props) => {
const { workingCase, setWorkingCase } = useContext(FormContext)
const router = useRouter()
+ const isFine =
+ workingCase.indictmentRulingDecision === CaseIndictmentRulingDecision.FINE
+
const serviceRequired =
defendant.serviceRequirement === ServiceRequirement.REQUIRED
@@ -133,34 +135,45 @@ const BlueBoxWithDate: FC = (props) => {
const textItems = useMemo(() => {
const texts = []
- if (serviceRequired) {
+ if (isFine) {
texts.push(
- formatMessage(strings.defendantVerdictViewedDate, {
- date: formatDate(dates.verdictViewDate ?? defendant.verdictViewDate),
+ formatMessage(strings.fineAppealDeadline, {
+ appealDeadlineIsInThePast: defendant.isVerdictAppealDeadlineExpired,
+ appealDeadline: formatDate(defendant.verdictAppealDeadline),
}),
)
- }
-
- texts.push(
- formatMessage(appealExpirationInfo.message, {
- appealExpirationDate: appealExpirationInfo.date,
- }),
- )
+ } else {
+ if (serviceRequired) {
+ texts.push(
+ formatMessage(strings.defendantVerdictViewedDate, {
+ date: formatDate(
+ dates.verdictViewDate ?? defendant.verdictViewDate,
+ ),
+ }),
+ )
+ }
- if (defendant.verdictAppealDate) {
texts.push(
- formatMessage(strings.defendantAppealDate, {
- date: formatDate(defendant.verdictAppealDate),
+ formatMessage(appealExpirationInfo.message, {
+ appealExpirationDate: appealExpirationInfo.date,
}),
)
- }
- if (defendant.sentToPrisonAdminDate && defendant.isSentToPrisonAdmin) {
- texts.push(
- formatMessage(strings.sendToPrisonAdminDate, {
- date: formatDate(defendant.sentToPrisonAdminDate),
- }),
- )
+ if (defendant.verdictAppealDate) {
+ texts.push(
+ formatMessage(strings.defendantAppealDate, {
+ date: formatDate(defendant.verdictAppealDate),
+ }),
+ )
+ }
+
+ if (defendant.sentToPrisonAdminDate && defendant.isSentToPrisonAdmin) {
+ texts.push(
+ formatMessage(strings.sendToPrisonAdminDate, {
+ date: formatDate(defendant.sentToPrisonAdminDate),
+ }),
+ )
+ }
}
return texts
@@ -169,10 +182,13 @@ const BlueBoxWithDate: FC = (props) => {
appealExpirationInfo.message,
dates.verdictViewDate,
defendant.isSentToPrisonAdmin,
+ defendant.isVerdictAppealDeadlineExpired,
defendant.sentToPrisonAdminDate,
defendant.verdictAppealDate,
+ defendant.verdictAppealDeadline,
defendant.verdictViewDate,
formatMessage,
+ isFine,
serviceRequired,
])
@@ -205,7 +221,9 @@ const BlueBoxWithDate: FC = (props) => {
@@ -217,7 +235,7 @@ const BlueBoxWithDate: FC = (props) => {
- {(!serviceRequired || defendant.verdictViewDate) &&
+ {(!serviceRequired || defendant.verdictViewDate || isFine) &&
textItems.map((text, index) => (
= (props) => {
{defendant.verdictAppealDate ||
- defendant.isVerdictAppealDeadlineExpired ? null : !serviceRequired ||
- defendant.verdictViewDate ? (
+ defendant.isVerdictAppealDeadlineExpired ||
+ isFine ? null : !serviceRequired || defendant.verdictViewDate ? (
= (props) => {
variant="text"
onClick={handleSendToPrisonAdmin}
size="small"
- disabled={!indictmentReviewDecision || !defendant.verdictViewDate}
+ disabled={
+ !workingCase.indictmentReviewDecision ||
+ (!isFine && !defendant.verdictViewDate)
+ }
>
{formatMessage(strings.sendToPrisonAdmin)}
diff --git a/apps/judicial-system/web/src/components/InfoCard/useInfoCardItems.strings.ts b/apps/judicial-system/web/src/components/InfoCard/useInfoCardItems.strings.ts
index 57689f0fa174..cbe3bdb2b903 100644
--- a/apps/judicial-system/web/src/components/InfoCard/useInfoCardItems.strings.ts
+++ b/apps/judicial-system/web/src/components/InfoCard/useInfoCardItems.strings.ts
@@ -28,14 +28,16 @@ export const strings = defineMessages({
description: 'Notaður sem titill á "ákvörðun" hluta af yfirliti ákæru.',
},
reviewTagAppealed: {
- id: 'judicial.system.core:info_card_indictment.review_tag_appealed_v1',
- defaultMessage: 'Áfrýja dómi',
+ id: 'judicial.system.core:info_card_indictment.review_tag_appealed_v2',
+ defaultMessage:
+ 'Áfrýja {isFine, select, true {viðurlagaákvörðun} other {dómi}}',
description:
'Notað sem texti á tagg fyrir "Áfrýjun" tillögu í yfirliti ákæru.',
},
reviewTagAccepted: {
- id: 'judicial.system.core:info_card_indictment.review_tag_completed_v1',
- defaultMessage: 'Una dómi',
+ id: 'judicial.system.core:info_card_indictment.review_tag_completed_v2',
+ defaultMessage:
+ 'Una {isFine, select, true {viðurlagaákvörðun} other {dómi}}',
description: 'Notað sem texti á tagg fyrir "Una" tillögu í yfirliti ákæru.',
},
indictmentReviewedDateTitle: {
diff --git a/apps/judicial-system/web/src/components/InfoCard/useInfoCardItems.tsx b/apps/judicial-system/web/src/components/InfoCard/useInfoCardItems.tsx
index 1a0c7ccf70c3..a2112beb8c66 100644
--- a/apps/judicial-system/web/src/components/InfoCard/useInfoCardItems.tsx
+++ b/apps/judicial-system/web/src/components/InfoCard/useInfoCardItems.tsx
@@ -15,6 +15,7 @@ import { core } from '@island.is/judicial-system-web/messages'
import { requestCourtDate } from '@island.is/judicial-system-web/messages'
import {
Case,
+ CaseIndictmentRulingDecision,
CaseType,
IndictmentCaseReviewDecision,
} from '@island.is/judicial-system-web/src/graphql/schema'
@@ -279,6 +280,11 @@ const useInfoCardItems = () => {
IndictmentCaseReviewDecision.ACCEPT
? strings.reviewTagAccepted
: strings.reviewTagAppealed,
+ {
+ isFine:
+ workingCase.indictmentRulingDecision ===
+ CaseIndictmentRulingDecision.FINE,
+ },
),
],
}
diff --git a/apps/judicial-system/web/src/components/index.ts b/apps/judicial-system/web/src/components/index.ts
index adf54284e0c6..5dc0a2ff8240 100644
--- a/apps/judicial-system/web/src/components/index.ts
+++ b/apps/judicial-system/web/src/components/index.ts
@@ -1,6 +1,7 @@
export { CourtCaseInfo, ProsecutorCaseInfo } from './CaseInfo/CaseInfo'
export { default as AccordionListItem } from './AccordionListItem/AccordionListItem'
export { default as BlueBox } from './BlueBox/BlueBox'
+export { default as BlueBoxWithDate } from './BlueBoxWithIcon/BlueBoxWithDate'
export { default as CaseDates } from './CaseDates/CaseDates'
export { default as CaseFile } from './CaseFile/CaseFile'
export { default as CaseFileList } from './CaseFileList/CaseFileList'
diff --git a/apps/judicial-system/web/src/routes/PublicProsecutor/Indictments/Overview/Overview.strings.ts b/apps/judicial-system/web/src/routes/PublicProsecutor/Indictments/Overview/Overview.strings.ts
index bda2b5b7b75f..3418534bf794 100644
--- a/apps/judicial-system/web/src/routes/PublicProsecutor/Indictments/Overview/Overview.strings.ts
+++ b/apps/judicial-system/web/src/routes/PublicProsecutor/Indictments/Overview/Overview.strings.ts
@@ -22,9 +22,9 @@ export const strings = defineMessages({
description: 'Notaður sem titill á yfirliti ákæru.',
},
reviewerSubtitle: {
- id: 'judicial.system.core:public_prosecutor.indictments.overview.reviewer_subtitle',
+ id: 'judicial.system.core:public_prosecutor.indictments.overview.reviewer_subtitle_v2',
defaultMessage:
- 'Frestur til að áfrýja dómi rennur út {indictmentAppealDeadline}',
+ 'Frestur til að {isFine, select, true {kæra viðurlagaákvörðun} other {áfrýja dómi}} {appealDeadlineIsInThePast, select, true {rann} other {rennur}} út {indictmentAppealDeadline}',
description: 'Notaður sem undirtitill á yfirliti ákæru.',
},
reviewerAssignedModalTitle: {
diff --git a/apps/judicial-system/web/src/routes/PublicProsecutor/Indictments/Overview/Overview.tsx b/apps/judicial-system/web/src/routes/PublicProsecutor/Indictments/Overview/Overview.tsx
index 6ac0b378751e..36abf477db77 100644
--- a/apps/judicial-system/web/src/routes/PublicProsecutor/Indictments/Overview/Overview.tsx
+++ b/apps/judicial-system/web/src/routes/PublicProsecutor/Indictments/Overview/Overview.tsx
@@ -8,6 +8,7 @@ import { formatDate } from '@island.is/judicial-system/formatters'
import { core, titles } from '@island.is/judicial-system-web/messages'
import {
BlueBox,
+ BlueBoxWithDate,
CourtCaseInfo,
FormContentContainer,
FormContext,
@@ -23,8 +24,8 @@ import {
// useIndictmentsLawsBroken, NOTE: Temporarily hidden while list of laws broken is not complete
UserContext,
} from '@island.is/judicial-system-web/src/components'
-import BlueBoxWithDate from '@island.is/judicial-system-web/src/components/BlueBoxWithIcon/BlueBoxWithDate'
import { useProsecutorSelectionUsersQuery } from '@island.is/judicial-system-web/src/components/ProsecutorSelection/prosecutorSelectionUsers.generated'
+import { CaseIndictmentRulingDecision } from '@island.is/judicial-system-web/src/graphql/schema'
import { useCase } from '@island.is/judicial-system-web/src/utils/hooks'
import { strings } from './Overview.strings'
@@ -102,11 +103,7 @@ export const Overview = () => {
{workingCase.defendants?.map((defendant) => (
-
+
))}
@@ -130,9 +127,14 @@ export const Overview = () => {
description={
{fm(strings.reviewerSubtitle, {
+ isFine:
+ workingCase.indictmentRulingDecision ===
+ CaseIndictmentRulingDecision.FINE,
indictmentAppealDeadline: formatDate(
workingCase.indictmentAppealDeadline,
),
+ appealDeadlineIsInThePast:
+ workingCase.indictmentVerdictAppealDeadlineExpired,
})}
}
diff --git a/apps/judicial-system/web/src/routes/PublicProsecutor/Tables/CasesReviewed.strings.ts b/apps/judicial-system/web/src/routes/PublicProsecutor/Tables/CasesReviewed.strings.ts
index 85cb3bb24b15..130b6c5b87da 100644
--- a/apps/judicial-system/web/src/routes/PublicProsecutor/Tables/CasesReviewed.strings.ts
+++ b/apps/judicial-system/web/src/routes/PublicProsecutor/Tables/CasesReviewed.strings.ts
@@ -18,6 +18,12 @@ export const strings = defineMessages({
description:
'Notað sem texti á tagg fyrir "Unun" tillögu í yfirlesin mál málalista',
},
+ reviewTagFineAppealed: {
+ id: 'judicial.system.core:public_prosecutor.tables.cases_reviewed.review_tag_fine_appealed',
+ defaultMessage: 'Kært',
+ description:
+ 'Notað sem texti á tagg fyrir "Kært" tillögu í yfirlesin mál málalista',
+ },
infoContainerMessage: {
id: 'judicial.system.core:public_prosecutor.tables.cases_reviewed.info_container_message',
defaultMessage: 'Engin yfirlesin mál.',
diff --git a/apps/judicial-system/web/src/routes/PublicProsecutor/Tables/CasesReviewed.tsx b/apps/judicial-system/web/src/routes/PublicProsecutor/Tables/CasesReviewed.tsx
index e6a4b98ab6c2..3462a41a096e 100644
--- a/apps/judicial-system/web/src/routes/PublicProsecutor/Tables/CasesReviewed.tsx
+++ b/apps/judicial-system/web/src/routes/PublicProsecutor/Tables/CasesReviewed.tsx
@@ -4,6 +4,7 @@ import { AnimatePresence } from 'framer-motion'
import { Tag, Text } from '@island.is/island-ui/core'
import { capitalize } from '@island.is/judicial-system/formatters'
+import { CaseIndictmentRulingDecision } from '@island.is/judicial-system/types'
import { core, tables } from '@island.is/judicial-system-web/messages'
import { SectionHeading } from '@island.is/judicial-system-web/src/components'
import { useContextMenu } from '@island.is/judicial-system-web/src/components/ContextMenu/ContextMenu'
@@ -31,13 +32,19 @@ const CasesReviewed: FC = ({ loading, cases }) => {
const { formatMessage } = useIntl()
const { openCaseInNewTabMenuItem } = useContextMenu()
- const decisionMapping = {
- [IndictmentCaseReviewDecision.ACCEPT]: formatMessage(
- strings.reviewTagAccepted,
- ),
- [IndictmentCaseReviewDecision.APPEAL]: formatMessage(
- strings.reviewTagAppealed,
- ),
+ const indictmentReviewDecisionMapping = (
+ reviewDecision: IndictmentCaseReviewDecision,
+ isFine: boolean,
+ ) => {
+ if (reviewDecision === IndictmentCaseReviewDecision.ACCEPT) {
+ return formatMessage(strings.reviewTagAccepted)
+ } else if (reviewDecision === IndictmentCaseReviewDecision.APPEAL) {
+ return formatMessage(
+ isFine ? strings.reviewTagFineAppealed : strings.reviewTagAppealed,
+ )
+ } else {
+ return null
+ }
}
const getVerdictViewTag = (row: CaseListEntry) => {
@@ -49,7 +56,9 @@ const CasesReviewed: FC = ({ loading, cases }) => {
row.defendants.some((defendant) => defendant.isSentToPrisonAdmin),
)
- if (someDefendantIsSentToPrisonAdmin) {
+ if (row.indictmentRulingDecision === CaseIndictmentRulingDecision.FINE) {
+ return null
+ } else if (someDefendantIsSentToPrisonAdmin) {
variant = 'red'
message = strings.tagVerdictViewSentToPrisonAdmin
} else if (!row.indictmentVerdictViewedByAll) {
@@ -112,7 +121,11 @@ const CasesReviewed: FC = ({ loading, cases }) => {
cell: (row) => (
{row.indictmentReviewDecision &&
- decisionMapping[row.indictmentReviewDecision]}
+ indictmentReviewDecisionMapping(
+ row.indictmentReviewDecision,
+ row.indictmentRulingDecision ===
+ CaseIndictmentRulingDecision.FINE,
+ )}
),
},
diff --git a/apps/judicial-system/web/src/routes/PublicProsecutor/components/ReviewDecision/ReviewDecision.css.ts b/apps/judicial-system/web/src/routes/PublicProsecutor/components/ReviewDecision/ReviewDecision.css.ts
index f3b85cfcbefe..f0211d02ca05 100644
--- a/apps/judicial-system/web/src/routes/PublicProsecutor/components/ReviewDecision/ReviewDecision.css.ts
+++ b/apps/judicial-system/web/src/routes/PublicProsecutor/components/ReviewDecision/ReviewDecision.css.ts
@@ -4,7 +4,12 @@ import { theme } from '@island.is/island-ui/theme'
export const gridRow = style({
display: 'grid',
- gridTemplateColumns: '1.6fr 1fr',
+ gridTemplateColumns: '1fr auto',
gridGap: theme.spacing[1],
- marginBottom: theme.spacing[1],
+
+ '@media': {
+ [`screen and (max-width: ${theme.breakpoints.lg}px)`]: {
+ gridTemplateColumns: '1fr',
+ },
+ },
})
diff --git a/apps/judicial-system/web/src/routes/PublicProsecutor/components/ReviewDecision/ReviewDecision.strings.ts b/apps/judicial-system/web/src/routes/PublicProsecutor/components/ReviewDecision/ReviewDecision.strings.ts
index 796a7372f889..c9250aa3fd48 100644
--- a/apps/judicial-system/web/src/routes/PublicProsecutor/components/ReviewDecision/ReviewDecision.strings.ts
+++ b/apps/judicial-system/web/src/routes/PublicProsecutor/components/ReviewDecision/ReviewDecision.strings.ts
@@ -2,14 +2,14 @@ import { defineMessages } from 'react-intl'
export const strings = defineMessages({
title: {
- id: 'judicial.system.core:public_prosecutor.indictments.review_decision.title',
- defaultMessage: 'Ákvörðun um áfrýjun',
+ id: 'judicial.system.core:public_prosecutor.indictments.review_decision.title_v1',
+ defaultMessage: 'Ákvörðun um {isFine, select, true {kæru} other {áfrýjun}}',
description: 'Notaður sem titill á ákvörðum um áfrýjun boxi fyrir ákæru.',
},
subtitle: {
- id: 'judicial.system.core:public_prosecutor.indictments.review_decision.subtitle',
+ id: 'judicial.system.core:public_prosecutor.indictments.review_decision.subtitle_v1',
defaultMessage:
- 'Frestur til að áfrýja dómi rennur út {indictmentAppealDeadline}',
+ 'Frestur til að {isFine, select, true {kæra viðurlagaákvörðun} other {áfrýja dómi}} {appealDeadlineIsInThePast, select, true {rann} other {rennur}} út {indictmentAppealDeadline}',
description:
'Notaður sem undirtitill á ákvörðum um áfrýjun boxi fyrir ákæru.',
},
@@ -24,6 +24,18 @@ export const strings = defineMessages({
defaultMessage: 'Una héraðsdómi',
description: 'Notaður sem texti fyrir "Una héraðsdómi" radio takka.',
},
+ appealFineToCourtOfAppeals: {
+ id: 'judicial.system.core:public_prosecutor.indictments.review_decision.appeal_fine_to_court_of_appeals',
+ defaultMessage: 'Kæra viðurlagaákvörðun til Landsréttar',
+ description:
+ 'Notaður sem texti fyrir "Kæra viðurlagaákvörðun til Landsréttar" radio takka.',
+ },
+ acceptFineDecision: {
+ id: 'judicial.system.core:public_prosecutor.indictments.review_decision.accept_fine_decision',
+ defaultMessage: 'Una viðurlagaákvörðun',
+ description:
+ 'Notaður sem texti fyrir "Kæra viðurlagaákvörðun" radio takka.',
+ },
reviewModalTitle: {
id: 'judicial.system.core:indictments_review.title',
defaultMessage: 'Staðfesta ákvörðun',
@@ -31,11 +43,16 @@ export const strings = defineMessages({
},
reviewModalText: {
id: 'judicial.system.core:indictments_review.modal_text',
-
defaultMessage:
'Ertu viss um að þú viljir {reviewerDecision, select, ACCEPT {una héraðsdómi} APPEAL {áfrýja héraðsdómi til Landsréttar} other {halda áfram}}?',
description: 'Notaður sem texti í yfirlitsglugga um yfirlit ákæru.',
},
+ reviewModalTextFine: {
+ id: 'judicial.system.core:indictments_review.modal_text_fine',
+ defaultMessage:
+ 'Ertu viss um að þú viljir {reviewerDecision, select, ACCEPT {una viðurlagaákvörðun} APPEAL {kæra viðurlagaákvörðun til Landsréttar} other {halda áfram}}?',
+ description: 'Notaður sem texti í yfirlitsglugga um yfirlit ákæru.',
+ },
reviewModalPrimaryButtonText: {
id: 'judicial.system.core:indictments_review.modal_primary_button_text',
defaultMessage: 'Staðfesta',
diff --git a/apps/judicial-system/web/src/routes/PublicProsecutor/components/ReviewDecision/ReviewDecision.tsx b/apps/judicial-system/web/src/routes/PublicProsecutor/components/ReviewDecision/ReviewDecision.tsx
index ff25436bea93..119153159b66 100644
--- a/apps/judicial-system/web/src/routes/PublicProsecutor/components/ReviewDecision/ReviewDecision.tsx
+++ b/apps/judicial-system/web/src/routes/PublicProsecutor/components/ReviewDecision/ReviewDecision.tsx
@@ -21,25 +21,28 @@ import * as styles from './ReviewDecision.css'
interface Props {
caseId: string
indictmentAppealDeadline?: string
+ indictmentAppealDeadlineIsInThePast?: boolean
modalVisible?: boolean
setModalVisible: Dispatch>
+ isFine: boolean
onSelect?: () => void
}
export const ReviewDecision: FC = (props) => {
- const { user } = useContext(UserContext)
- const router = useRouter()
- const { formatMessage: fm } = useIntl()
- const { updateCase } = useCase()
-
const {
caseId,
indictmentAppealDeadline,
+ indictmentAppealDeadlineIsInThePast,
modalVisible,
setModalVisible,
+ isFine,
onSelect,
} = props
+ const { user } = useContext(UserContext)
+ const router = useRouter()
+ const { formatMessage: fm } = useIntl()
+ const { updateCase } = useCase()
const [indictmentReviewDecision, setIndictmentReviewDecision] = useState<
IndictmentCaseReviewDecision | undefined
>(undefined)
@@ -58,11 +61,15 @@ export const ReviewDecision: FC = (props) => {
const options = [
{
- label: fm(strings.appealToCourtOfAppeals),
+ label: fm(
+ isFine
+ ? strings.appealFineToCourtOfAppeals
+ : strings.appealToCourtOfAppeals,
+ ),
value: IndictmentCaseReviewDecision.APPEAL,
},
{
- label: fm(strings.acceptDecision),
+ label: fm(isFine ? strings.acceptFineDecision : strings.acceptDecision),
value: IndictmentCaseReviewDecision.ACCEPT,
},
]
@@ -74,11 +81,13 @@ export const ReviewDecision: FC = (props) => {
return (
{fm(strings.subtitle, {
+ isFine,
indictmentAppealDeadline: formatDate(indictmentAppealDeadline),
+ appealDeadlineIsInThePast: indictmentAppealDeadlineIsInThePast,
})}
}
@@ -107,9 +116,12 @@ export const ReviewDecision: FC = (props) => {
{modalVisible && (
setModalVisible(false)}
diff --git a/apps/judicial-system/web/src/routes/Shared/IndictmentOverview/IndictmentOverview.tsx b/apps/judicial-system/web/src/routes/Shared/IndictmentOverview/IndictmentOverview.tsx
index 83b140c09ed3..d5d2a465c780 100644
--- a/apps/judicial-system/web/src/routes/Shared/IndictmentOverview/IndictmentOverview.tsx
+++ b/apps/judicial-system/web/src/routes/Shared/IndictmentOverview/IndictmentOverview.tsx
@@ -285,8 +285,15 @@ const IndictmentOverview: FC = () => {
indictmentAppealDeadline={
workingCase.indictmentAppealDeadline ?? ''
}
+ indictmentAppealDeadlineIsInThePast={
+ workingCase.indictmentVerdictAppealDeadlineExpired ?? false
+ }
modalVisible={modalVisible}
setModalVisible={setModalVisible}
+ isFine={
+ workingCase.indictmentRulingDecision ===
+ CaseIndictmentRulingDecision.FINE
+ }
onSelect={() => setIsReviewDecisionSelected(true)}
/>
)}
diff --git a/libs/judicial-system/types/src/index.ts b/libs/judicial-system/types/src/index.ts
index 2a7cf9901626..140f6c9bc92a 100644
--- a/libs/judicial-system/types/src/index.ts
+++ b/libs/judicial-system/types/src/index.ts
@@ -102,6 +102,7 @@ export {
export {
getIndictmentVerdictAppealDeadlineStatus,
VERDICT_APPEAL_WINDOW_DAYS,
+ FINE_APPEAL_WINDOW_DAYS,
} from './lib/indictmentCase'
export type {
diff --git a/libs/judicial-system/types/src/lib/indictmentCase.ts b/libs/judicial-system/types/src/lib/indictmentCase.ts
index d751d60616b5..f823acbdf4b4 100644
--- a/libs/judicial-system/types/src/lib/indictmentCase.ts
+++ b/libs/judicial-system/types/src/lib/indictmentCase.ts
@@ -1,6 +1,7 @@
const DAYS_TO_MILLISECONDS = 24 * 60 * 60 * 1000
export const VERDICT_APPEAL_WINDOW_DAYS = 28
-const MILLISECONDS_TO_EXPIRY = VERDICT_APPEAL_WINDOW_DAYS * DAYS_TO_MILLISECONDS
+export const FINE_APPEAL_WINDOW_DAYS = 3
+const getDays = (days: number) => days * DAYS_TO_MILLISECONDS
/*
This function takes an array of verdict info tuples:
@@ -12,6 +13,7 @@ const MILLISECONDS_TO_EXPIRY = VERDICT_APPEAL_WINDOW_DAYS * DAYS_TO_MILLISECONDS
*/
export const getIndictmentVerdictAppealDeadlineStatus = (
verdictInfo?: [boolean, Date | undefined][],
+ isFine?: boolean,
): [boolean, boolean] => {
if (
!verdictInfo ||
@@ -34,5 +36,10 @@ export const getIndictmentVerdictAppealDeadlineStatus = (
new Date(0),
)
- return [true, Date.now() > newestViewDate.getTime() + MILLISECONDS_TO_EXPIRY]
+ return [
+ true,
+ Date.now() >
+ newestViewDate.getTime() +
+ getDays(isFine ? FINE_APPEAL_WINDOW_DAYS : VERDICT_APPEAL_WINDOW_DAYS),
+ ]
}
From b05c3c67eb27f4248a9a82e4feb2952844697d9c Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?=C3=9E=C3=B3r=C3=B0ur=20H?=
Date: Wed, 27 Nov 2024 09:22:16 +0000
Subject: [PATCH 10/12] fix(regulations-admin): Fixing bugs in editor and admin
flow (#17027)
* Pristine handling and error on save handling
* Remove guess for mentioned lawchapters. Sort search results by best match
* Update guesswork for preset changereg
* Fix prist hook
---------
Co-authored-by: kodiakhq[bot] <49736102+kodiakhq[bot]@users.noreply.github.com>
---
.../src/components/EditBasics.tsx | 23 ++++++----
.../src/components/LawChaptersSelect.tsx | 25 +++++-----
.../src/components/impacts/EditChange.tsx | 19 ++++++--
.../impacts/ImpactAmendingSelection.tsx | 29 +++++++++---
.../regulations-admin/src/lib/messages.ts | 5 ++
.../src/utils/formatAmendingRegulation.ts | 46 +++++++++++++------
.../src/utils/formatAmendingUtils.ts | 46 +++++++++++++++++--
.../regulations-admin/src/utils/hooks.ts | 43 +++++++++++++++++
8 files changed, 187 insertions(+), 49 deletions(-)
diff --git a/libs/portals/admin/regulations-admin/src/components/EditBasics.tsx b/libs/portals/admin/regulations-admin/src/components/EditBasics.tsx
index 304a1e004f20..b2868d63fe0f 100644
--- a/libs/portals/admin/regulations-admin/src/components/EditBasics.tsx
+++ b/libs/portals/admin/regulations-admin/src/components/EditBasics.tsx
@@ -29,6 +29,7 @@ import { ReferenceText } from './impacts/ReferenceText'
import { DraftChangeForm, DraftImpactForm } from '../state/types'
import { makeDraftAppendixForm } from '../state/makeFields'
import { hasAnyChange } from '../utils/formatAmendingUtils'
+import { usePristineRegulations } from '../utils/hooks'
const updateText =
'Ósamræmi er í texta stofnreglugerðar og breytingareglugerðar. Texti breytingareglugerðar þarf að samræmast breytingum sem gerðar hafa verið á stofnreglugerð, eigi breytingarnar að færast inn með réttum hætti.'
@@ -38,12 +39,13 @@ export const EditBasics = () => {
const { draft, actions, ministries } = useDraftingState()
const [editorKey, setEditorKey] = useState('initial')
const [titleError, setTitleError] = useState(undefined)
- const [hasUpdated, setHasUpdated] = useState(false)
const [hasUpdatedAppendix, setHasUpdatedAppendix] = useState(false)
const [references, setReferences] = useState()
const [isModalVisible, setIsModalVisible] = useState(true)
const [hasConfirmed, setHasConfirmed] = useState(false)
const [hasSeenModal, setHasSeenModal] = useState(false)
+ const { removePristineRegulation, isPristineRegulation } =
+ usePristineRegulations()
const { text, appendixes } = draft
const { updateState } = actions
@@ -120,13 +122,6 @@ export const EditBasics = () => {
// eslint-disable-next-line react-hooks/exhaustive-deps
}, [draft.impacts])
- useEffect(() => {
- if (!hasUpdatedAppendix && hasUpdated) {
- updateAppendixes()
- setHasUpdatedAppendix(true)
- }
- }, [hasUpdated, hasUpdatedAppendix])
-
const updateAppendixes = () => {
// FORMAT AMENDING REGULATION APPENDIXES
const impactArray = Object.values(draft.impacts).flat()
@@ -167,9 +162,17 @@ export const EditBasics = () => {
const additionString = additions.join('') as HTMLText
updateState('text', additionString)
- setHasUpdated(true)
+ // Update appendixes
+ if (!hasUpdatedAppendix) {
+ updateAppendixes()
+ setHasUpdatedAppendix(true)
+ }
+
+ // Remove from session storage
+ removePristineRegulation(draft.id)
}
+ const shouldShowModal = isPristineRegulation(draft.id)
return (
<>
@@ -328,7 +331,7 @@ export const EditBasics = () => {
)}
- {!hasUpdated ? (
+ {shouldShowModal ? (
{
const chaptersField = draft.lawChapters
const activeChapters = chaptersField.value
- const { data: mentionedList /*, loading , error */ } =
- useRegulationListQuery(draft.mentioned)
+ // const { data: mentionedList /*, loading , error */ } =
+ // useRegulationListQuery(draft.mentioned)
const lawChaptersOptions = useLawChapterOptions(
lawChapters.list,
@@ -47,16 +47,17 @@ export const LawChaptersSelect = () => {
)
// Auto fill lawChapters if there are mentions and no lawchapters present
- useEffect(() => {
- if (mentionedList?.length && !draft.lawChapters.value.length) {
- mentionedList.forEach((mention) => {
- mention.lawChapters?.forEach((ch) => {
- actions.updateLawChapterProp('add', ch.slug)
- })
- })
- }
- // eslint-disable-next-line react-hooks/exhaustive-deps
- }, [mentionedList])
+ // REMOVE FOR NOW. NOT SHOWING CORRECT RESULTS. MIGHT WANT TO REVISIT
+ // useEffect(() => {
+ // if (mentionedList?.length && !draft.lawChapters.value.length) {
+ // mentionedList.forEach((mention) => {
+ // mention.lawChapters?.forEach((ch) => {
+ // actions.updateLawChapterProp('add', ch.slug)
+ // })
+ // })
+ // }
+ // // eslint-disable-next-line react-hooks/exhaustive-deps
+ // }, [mentionedList])
return (
diff --git a/libs/portals/admin/regulations-admin/src/components/impacts/EditChange.tsx b/libs/portals/admin/regulations-admin/src/components/impacts/EditChange.tsx
index aad98a8ca232..443c1545203b 100644
--- a/libs/portals/admin/regulations-admin/src/components/impacts/EditChange.tsx
+++ b/libs/portals/admin/regulations-admin/src/components/impacts/EditChange.tsx
@@ -7,6 +7,7 @@ import {
GridRow,
GridColumn,
Text,
+ toast,
} from '@island.is/island-ui/core'
import { useEffect, useState, useMemo } from 'react'
import {
@@ -51,7 +52,10 @@ import {
updateFieldValue,
validateImpact,
} from '../../state/validations'
-import { useGetRegulationHistory } from '../../utils/hooks'
+import {
+ useGetRegulationHistory,
+ usePristineRegulations,
+} from '../../utils/hooks'
import { DraftRegulationChange } from '@island.is/regulations/admin'
import { useLocale } from '@island.is/localization'
import { cleanTitle } from '@island.is/regulations-tools/cleanTitle'
@@ -74,6 +78,7 @@ export const EditChange = (props: EditChangeProp) => {
const today = useMemo(() => new Date(), [])
const [minDate, setMinDate] = useState()
const [showChangeForm, setShowChangeForm] = useState(false)
+ const { addPristineRegulation } = usePristineRegulations()
// Target regulation for impact
const [activeRegulation, setActiveRegulation] = useState<
@@ -280,7 +285,12 @@ export const EditChange = (props: EditChangeProp) => {
}
return { success: true, error: undefined }
})
+ .then(() => {
+ addPristineRegulation(draft.id)
+ closeModal(true)
+ })
.catch((error) => {
+ toast.error(t(msg.errorOnSaveReg))
return { success: false, error: error as Error }
})
} else {
@@ -306,12 +316,15 @@ export const EditChange = (props: EditChangeProp) => {
}
return { success: true, error: undefined }
})
+ .then(() => {
+ addPristineRegulation(draft.id)
+ closeModal(true)
+ })
.catch((error) => {
+ toast.error(t(msg.errorOnSaveReg))
return { success: false, error: error as Error }
})
}
-
- closeModal(true)
}
const localActions = {
diff --git a/libs/portals/admin/regulations-admin/src/components/impacts/ImpactAmendingSelection.tsx b/libs/portals/admin/regulations-admin/src/components/impacts/ImpactAmendingSelection.tsx
index ec8ab460d01d..109679e96c38 100644
--- a/libs/portals/admin/regulations-admin/src/components/impacts/ImpactAmendingSelection.tsx
+++ b/libs/portals/admin/regulations-admin/src/components/impacts/ImpactAmendingSelection.tsx
@@ -2,7 +2,11 @@ import { useEffect, useState } from 'react'
import { impactMsgs } from '../../lib/messages'
import { useLocale } from '@island.is/localization'
-import { RegulationOptionList, RegulationType } from '@island.is/regulations'
+import {
+ RegulationOption,
+ RegulationOptionList,
+ RegulationType,
+} from '@island.is/regulations'
import { DraftImpactName } from '@island.is/regulations/admin'
import { AsyncSearch, Option, Text } from '@island.is/island-ui/core'
@@ -39,10 +43,24 @@ export const ImpactAmendingSelection = ({
RegulationSearchListQuery,
{
onCompleted: (data) => {
- setResults(
- (data.getRegulationsOptionSearch as RegulationOptionList) ||
- undefined,
- )
+ // The search results need to be reordered so that the objects with titles containing the input string are first
+ const results = data.getRegulationsOptionSearch
+ const reorderedResults = value
+ ? [
+ // Objects with titles containing the input string
+ ...results.filter((obj: RegulationOption) =>
+ obj.title.toLowerCase().includes(value.toLowerCase()),
+ ),
+ // The rest of the objects
+ ...results.filter(
+ (obj: RegulationOption) =>
+ !obj.title.toLowerCase().includes(value.toLowerCase()),
+ ),
+ ]
+ : results // If value is undefined, keep the original order
+
+ setIsLoading(false)
+ setResults((reorderedResults as RegulationOptionList) || undefined)
},
fetchPolicy: 'no-cache',
},
@@ -64,7 +82,6 @@ export const ImpactAmendingSelection = ({
variables: { input: { q: value, iA: false, iR: false } },
})
}
- setIsLoading(false)
},
600,
[value],
diff --git a/libs/portals/admin/regulations-admin/src/lib/messages.ts b/libs/portals/admin/regulations-admin/src/lib/messages.ts
index e49a79d65275..1483376cd6af 100644
--- a/libs/portals/admin/regulations-admin/src/lib/messages.ts
+++ b/libs/portals/admin/regulations-admin/src/lib/messages.ts
@@ -571,6 +571,11 @@ export const errorMsgs = defineMessages({
id: 'ap.regulations-admin:title-is-too-long',
defaultMessage: 'Titill reglugerðar er of langur.',
},
+ errorOnSaveReg: {
+ id: 'ap.regulations-admin:error-on-save-reg',
+ defaultMessage:
+ 'Villa kom upp við að vista texta. Vinsamlegast afritið texta og reynið aftur síðar.',
+ },
})
export const homeMessages = defineMessages({
diff --git a/libs/portals/admin/regulations-admin/src/utils/formatAmendingRegulation.ts b/libs/portals/admin/regulations-admin/src/utils/formatAmendingRegulation.ts
index d42974eb109f..651408bdf618 100644
--- a/libs/portals/admin/regulations-admin/src/utils/formatAmendingRegulation.ts
+++ b/libs/portals/admin/regulations-admin/src/utils/formatAmendingRegulation.ts
@@ -9,11 +9,15 @@ import flatten from 'lodash/flatten'
import uniq from 'lodash/uniq'
import {
allSameDay,
+ containsAnyTitleClass,
extractArticleTitleDisplay,
formatDate,
+ getArticleTitleType,
+ getArticleTypeText,
getTextWithSpaces,
groupElementsByArticleTitleFromDiv,
hasAnyChange,
+ hasSubtitle,
isGildisTaka,
removeRegPrefix,
updateAppendixWording,
@@ -203,13 +207,21 @@ export const formatAmendingRegBody = (
const testGroup: {
arr: HTMLText[]
original?: HTMLText[]
- title: string
+ titleObject: {
+ text: string
+ hasSubtitle: boolean
+ type: 'article' | 'chapter' | 'subchapter'
+ }
isDeletion?: boolean
isAddition?: boolean
} = {
arr: [],
original: [],
- title: '',
+ titleObject: {
+ text: '',
+ hasSubtitle: false,
+ type: 'article',
+ },
isDeletion: undefined,
isAddition: undefined,
}
@@ -218,16 +230,19 @@ export const formatAmendingRegBody = (
let pushHtml = '' as HTMLText
let isParagraph = false
- let isArticleTitle = false
+ let isSectionTitle = false
let isNumberList = false
let isLetterList = false
- if (element.classList.contains('article__title')) {
+ const containsAnyTitle = containsAnyTitleClass(element)
+ if (containsAnyTitle) {
const clone = element.cloneNode(true)
const textContent = getTextWithSpaces(clone)
articleTitle = extractArticleTitleDisplay(textContent)
- testGroup.title = articleTitle
- isArticleTitle = true
+ testGroup.titleObject.hasSubtitle = hasSubtitle(textContent)
+ testGroup.titleObject.type = getArticleTitleType(element)
+ testGroup.titleObject.text = articleTitle
+ isSectionTitle = true
paragraph = 0 // Reset paragraph count for the new article
} else if (element.nodeName.toLowerCase() === 'p') {
paragraph++
@@ -249,7 +264,7 @@ export const formatAmendingRegBody = (
const elementType =
isLetterList || isNumberList
? 'lidur'
- : isArticleTitle
+ : isSectionTitle
? 'greinTitle'
: undefined
@@ -275,7 +290,7 @@ export const formatAmendingRegBody = (
// Paragraph was deleted
pushHtml =
`${paragraph}. mgr. ${articleTitle} ${regNameDisplay} fellur brott.
` as HTMLText
- } else if (isArticleTitle) {
+ } else if (isSectionTitle) {
// Title was deleted
pushHtml =
`Fyrirsögn ${articleTitle} ${regNameDisplay} fellur brott.
` as HTMLText
@@ -304,7 +319,7 @@ export const formatAmendingRegBody = (
paragraph - 1
}. mgr. ${articleTitle} ${regNameDisplay} kemur ný málsgrein sem orðast svo:
${newText}
` as HTMLText)
: (`Á undan 1. mgr. ${articleTitle} ${regNameDisplay} kemur ný málsgrein svohljóðandi:
${newText}
` as HTMLText)
- } else if (isArticleTitle) {
+ } else if (isSectionTitle) {
// Title was added
testGroup.original?.push(`${newText}
` as HTMLText)
pushHtml =
@@ -337,7 +352,7 @@ export const formatAmendingRegBody = (
// Change detected. Not additon, not deletion.
testGroup.isDeletion = false
testGroup.isAddition = false
- if (isArticleTitle) {
+ if (isSectionTitle) {
// Title was changed
pushHtml =
`Fyrirsögn ${articleTitle} ${regNameDisplay} breytist og orðast svo:
${newText}
` as HTMLText
@@ -365,7 +380,7 @@ export const formatAmendingRegBody = (
}
})
if (testGroup.isDeletion === true) {
- const articleTitleNumber = testGroup.title
+ const articleTitleNumber = testGroup.titleObject.text
additionArray.push([
`${articleTitleNumber} ${regNameDisplay} fellur brott.
` as HTMLText,
@@ -376,7 +391,7 @@ export const formatAmendingRegBody = (
if (prevArticle.length > 0) {
prevArticleTitle = prevArticle[0]?.innerText
}
- const articleTitleNumber = testGroup.title
+ const articleTitleNumber = testGroup.titleObject.text
const originalTextArray = testGroup.original?.length
? flatten(testGroup.original)
: []
@@ -395,8 +410,13 @@ export const formatAmendingRegBody = (
: ''
}
+ const articleTypeText = getArticleTypeText(testGroup.titleObject.type)
additionArray.push([
- `Á eftir ${prevArticleTitleNumber} ${regNameDisplay} kemur ný grein, ${articleTitleNumber}, ásamt fyrirsögn, svohljóðandi:
${articleDisplayText}` as HTMLText,
+ `Á eftir ${prevArticleTitleNumber} ${regNameDisplay} kemur ${articleTypeText}, ${
+ articleTitleNumber ? articleTitleNumber + ',' : ''
+ }${
+ testGroup.titleObject.hasSubtitle ? ' ásamt fyrirsögn,' : ''
+ } svohljóðandi:
${articleDisplayText}` as HTMLText,
])
} else {
additionArray.push(testGroup.arr)
diff --git a/libs/portals/admin/regulations-admin/src/utils/formatAmendingUtils.ts b/libs/portals/admin/regulations-admin/src/utils/formatAmendingUtils.ts
index f67c3b420f81..3108a49dff83 100644
--- a/libs/portals/admin/regulations-admin/src/utils/formatAmendingUtils.ts
+++ b/libs/portals/admin/regulations-admin/src/utils/formatAmendingUtils.ts
@@ -11,10 +11,8 @@ export const groupElementsByArticleTitleFromDiv = (
Array.from(div.children).forEach((child) => {
const element = child as HTMLElement
- if (
- element.classList.contains('article__title') ||
- element.classList.contains('chapter__title')
- ) {
+ const containsAnyTitle = containsAnyTitleClass(element)
+ if (containsAnyTitle) {
if (currentGroup.length > 0) {
result.push(currentGroup)
}
@@ -31,15 +29,53 @@ export const groupElementsByArticleTitleFromDiv = (
return result
}
+const titleArray = ['article__title', 'subchapter__title', 'chapter__title']
+export type ArticleTitleType = 'article' | 'chapter' | 'subchapter'
+
+export const containsAnyTitleClass = (element: HTMLElement): boolean => {
+ return titleArray.some((title) => element.classList.contains(title))
+}
+
+export const getArticleTypeText = (titleType?: ArticleTitleType): string => {
+ switch (titleType) {
+ case 'chapter':
+ return 'nýr kafli'
+ case 'subchapter':
+ return 'nýr undirkafli'
+ default:
+ return 'ný grein'
+ }
+}
+
+export const getArticleTitleType = (element: HTMLElement): ArticleTitleType => {
+ if (element.classList.contains('subchapter__title')) {
+ return 'subchapter'
+ } else if (element.classList.contains('chapter__title')) {
+ return 'chapter'
+ } else {
+ return 'article'
+ }
+}
+
/**
* Extracts article title number (e.g., '1. gr.' or '1. gr. a') from a string, allowing for Icelandic characters.
*/
+const extractRegex =
+ /^\d+\. gr\.(?: [\p{L}]\.)?(?= |$)|^(?:\d+|[IVXLCDM]+)\.?\s*Kafli(?=\b| |$)/iu
+
export const extractArticleTitleDisplay = (title: string): string => {
- const grMatch = title.match(/^\d+\. gr\.(?: [\p{L}])?(?= |$)/u)
+ const grMatch = title.match(extractRegex)
const articleTitleDisplay = grMatch ? grMatch[0] : title
return articleTitleDisplay
}
+export const hasSubtitle = (title: string): boolean => {
+ const grMatch = title.match(extractRegex)
+ const articleTitleDisplay = grMatch ? grMatch[0] : title
+ const hasSubText = title.trim() !== articleTitleDisplay.trim()
+ return hasSubText
+}
+
export const getTextWithSpaces = (element: Node): string => {
let result = ''
diff --git a/libs/portals/admin/regulations-admin/src/utils/hooks.ts b/libs/portals/admin/regulations-admin/src/utils/hooks.ts
index 753a62100bc5..45eed9f313d8 100644
--- a/libs/portals/admin/regulations-admin/src/utils/hooks.ts
+++ b/libs/portals/admin/regulations-admin/src/utils/hooks.ts
@@ -105,3 +105,46 @@ export const useAffectedRegulations = (
mentionedOptions,
}
}
+
+export const usePristineRegulations = () => {
+ const P_REG_KEY = 'PRISTINE_REGULATIONS'
+
+ const [pristineRegulations, setPristineRegulations] = useState(
+ () => {
+ const storedRegulations = sessionStorage.getItem(P_REG_KEY)
+ if (!storedRegulations) return []
+
+ try {
+ return JSON.parse(storedRegulations)
+ } catch (e) {
+ console.error('Failed to parse pristine regulations:', e)
+ return []
+ }
+ },
+ )
+
+ const addPristineRegulation = (regId: string) => {
+ if (!pristineRegulations.includes(regId)) {
+ const updatedRegulations = [...pristineRegulations, regId]
+ setPristineRegulations(updatedRegulations)
+ sessionStorage.setItem(P_REG_KEY, JSON.stringify(updatedRegulations))
+ }
+ }
+
+ const removePristineRegulation = (regId: string) => {
+ const updatedRegulations = pristineRegulations.filter(
+ (id: string) => id !== regId,
+ )
+ setPristineRegulations(updatedRegulations)
+ sessionStorage.setItem(P_REG_KEY, JSON.stringify(updatedRegulations))
+ }
+
+ const isPristineRegulation = (regId: string) =>
+ pristineRegulations.includes(regId)
+
+ return {
+ addPristineRegulation,
+ removePristineRegulation,
+ isPristineRegulation,
+ }
+}
From fe303dfdd7586c591f8cfe38f8d6fc831eb16d80 Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?=C3=8Dvar=20Oddsson?=
Date: Wed, 27 Nov 2024 10:24:17 +0000
Subject: [PATCH 11/12] fix(j-s): Text in infocard (#17038)
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
* Checkpoint
* Refactor AlertMessage
* Format date
* Cleanup
* Cleanup
* Merge
* Add key
* Refactor
* Remove console.log
* Merge
* Merge
* Fix text
* Fix text
* Update apps/judicial-system/web/src/components/InfoCard/useInfoCardItems.strings.ts
---------
Co-authored-by: kodiakhq[bot] <49736102+kodiakhq[bot]@users.noreply.github.com>
Co-authored-by: Guðjón Guðjónsson
Co-authored-by: unakb
---
.../web/src/components/InfoCard/useInfoCardItems.strings.ts | 4 ++--
1 file changed, 2 insertions(+), 2 deletions(-)
diff --git a/apps/judicial-system/web/src/components/InfoCard/useInfoCardItems.strings.ts b/apps/judicial-system/web/src/components/InfoCard/useInfoCardItems.strings.ts
index cbe3bdb2b903..1cff30e2a52b 100644
--- a/apps/judicial-system/web/src/components/InfoCard/useInfoCardItems.strings.ts
+++ b/apps/judicial-system/web/src/components/InfoCard/useInfoCardItems.strings.ts
@@ -28,9 +28,9 @@ export const strings = defineMessages({
description: 'Notaður sem titill á "ákvörðun" hluta af yfirliti ákæru.',
},
reviewTagAppealed: {
- id: 'judicial.system.core:info_card_indictment.review_tag_appealed_v2',
+ id: 'judicial.system.core:info_card_indictment.review_tag_appealed_v3',
defaultMessage:
- 'Áfrýja {isFine, select, true {viðurlagaákvörðun} other {dómi}}',
+ '{isFine, select, true {Kæra viðurlagaákvörðun} other {Áfrýja dómi}}',
description:
'Notað sem texti á tagg fyrir "Áfrýjun" tillögu í yfirliti ákæru.',
},
From c33e7a2738c3dc1cd84346e09a7f5e3ceaca3de3 Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?R=C3=BAnar=20Vestmann?=
<43557895+RunarVestmann@users.noreply.github.com>
Date: Wed, 27 Nov 2024 11:00:43 +0000
Subject: [PATCH 12/12] feat(web): Standalone organization pages - Read alert
banner field from organization page (#17031)
Co-authored-by: kodiakhq[bot] <49736102+kodiakhq[bot]@users.noreply.github.com>
---
.../standalone/components/AlertBanner.tsx | 56 +++++++++++++++++++
.../organization/standalone/standalone.tsx | 2 +
2 files changed, 58 insertions(+)
create mode 100644 apps/web/layouts/organization/standalone/components/AlertBanner.tsx
diff --git a/apps/web/layouts/organization/standalone/components/AlertBanner.tsx b/apps/web/layouts/organization/standalone/components/AlertBanner.tsx
new file mode 100644
index 000000000000..32d09cec0150
--- /dev/null
+++ b/apps/web/layouts/organization/standalone/components/AlertBanner.tsx
@@ -0,0 +1,56 @@
+import Cookies from 'js-cookie'
+
+import { AlertBanner, AlertBannerVariants } from '@island.is/island-ui/core'
+import { stringHash } from '@island.is/shared/utils'
+import { OrganizationPage } from '@island.is/web/graphql/schema'
+import { linkResolver, LinkType } from '@island.is/web/hooks'
+import { useI18n } from '@island.is/web/i18n'
+
+interface StandaloneAlertBannerProps {
+ alertBanner: OrganizationPage['alertBanner'] | null | undefined
+}
+
+export const StandaloneAlertBanner = ({
+ alertBanner,
+}: StandaloneAlertBannerProps) => {
+ const { activeLocale } = useI18n()
+
+ if (!alertBanner) {
+ return null
+ }
+
+ const bannerId = `standalone-alert-${stringHash(
+ JSON.stringify(alertBanner ?? {}),
+ )}`
+
+ if (Cookies.get(bannerId) || !alertBanner.showAlertBanner) {
+ return null
+ }
+
+ return (
+ {
+ if (alertBanner.dismissedForDays !== 0) {
+ Cookies.set(bannerId, 'hide', {
+ expires: alertBanner.dismissedForDays,
+ })
+ }
+ }}
+ closeButtonLabel={activeLocale === 'is' ? 'Loka' : 'Close'}
+ link={
+ alertBanner.linkTitle && alertBanner.link?.type && alertBanner.link.slug
+ ? {
+ href: linkResolver(alertBanner.link.type as LinkType, [
+ alertBanner.link.slug,
+ ]).href,
+ title: alertBanner.linkTitle,
+ }
+ : undefined
+ }
+ />
+ )
+}
diff --git a/apps/web/layouts/organization/standalone/standalone.tsx b/apps/web/layouts/organization/standalone/standalone.tsx
index 9007c3db63a1..5053000cbff6 100644
--- a/apps/web/layouts/organization/standalone/standalone.tsx
+++ b/apps/web/layouts/organization/standalone/standalone.tsx
@@ -15,6 +15,7 @@ import { useLinkResolver } from '@island.is/web/hooks'
import { useI18n } from '@island.is/web/i18n'
import { getBackgroundStyle } from '@island.is/web/utils/organization'
+import { StandaloneAlertBanner } from './components/AlertBanner'
import { Header, HeaderProps } from './components/Header'
import { Navigation, NavigationProps } from './components/Navigation'
@@ -204,6 +205,7 @@ export const StandaloneLayout = ({
}
/>
+