diff --git a/.env.template b/.env.template index 46c84eb..234b9d3 100644 --- a/.env.template +++ b/.env.template @@ -1,7 +1,6 @@ NODE_ENV=development SESSION_KEY=dette-er-en-lang-og-sikker-streng-for-signering-av-session-cookies AUTH_REDIRECT_URI=http://localhost:8080/oauth2/callback -INTERNARBEIDSFLATEDECORATOR_HOST=internarbeidsflatedecorator-q1.dev.intern.nav.no MODIACONTEXTHOLDER_HOST=modiacontextholder.dev.intern.nav.no MODIACONTEXTHOLDER_AAD_APP_CLIENT_ID=dev-gcp.personoversikt.modiacontextholder FASTLEGEREST_HOST=https://fastlegerest.intern.dev.nav.no diff --git a/.nais/naiserator-dev.yaml b/.nais/naiserator-dev.yaml index 05375dc..37af269 100644 --- a/.nais/naiserator-dev.yaml +++ b/.nais/naiserator-dev.yaml @@ -31,12 +31,8 @@ spec: value: production - name: REDIS_HOST value: finnfastlege-redis - - name: FINNFASTLEGE_URL - value: "http://finnfastlege" - name: AUTH_REDIRECT_URI value: "https://finnfastlege.intern.dev.nav.no/oauth2/callback" - - name: INTERNARBEIDSFLATEDECORATOR_HOST - value: "internarbeidsflatedecorator-q1.dev-fss-pub.nais.io" - name: FASTLEGEREST_AAD_APP_CLIENT_ID value: "dev-gcp.teamsykefravr.fastlegerest" - name: SYFOPERSON_AAD_APP_CLIENT_ID @@ -63,7 +59,6 @@ spec: outbound: external: - host: "login.microsoftonline.com" - - host: "internarbeidsflatedecorator-q1.dev-fss-pub.nais.io" rules: - application: fastlegerest - application: istilgangskontroll diff --git a/.nais/naiserator-prod.yaml b/.nais/naiserator-prod.yaml index ea8a573..939f9b5 100644 --- a/.nais/naiserator-prod.yaml +++ b/.nais/naiserator-prod.yaml @@ -31,12 +31,8 @@ spec: value: production - name: REDIS_HOST value: finnfastlege-redis - - name: FINNFASTLEGE_URL - value: "http://finnfastlege" - name: AUTH_REDIRECT_URI value: "https://finnfastlege.intern.nav.no/oauth2/callback" - - name: INTERNARBEIDSFLATEDECORATOR_HOST - value: "internarbeidsflatedecorator.prod-fss-pub.nais.io" - name: FASTLEGEREST_AAD_APP_CLIENT_ID value: "prod-gcp.teamsykefravr.fastlegerest" - name: SYFOPERSON_AAD_APP_CLIENT_ID @@ -63,7 +59,6 @@ spec: outbound: external: - host: "login.microsoftonline.com" - - host: "internarbeidsflatedecorator.prod-fss-pub.nais.io" rules: - application: fastlegerest - application: istilgangskontroll diff --git a/public/index.html b/public/index.html index 400c6e1..13800fc 100644 --- a/public/index.html +++ b/public/index.html @@ -6,16 +6,16 @@ <% if (process.env.NODE_ENV !== 'development') { %> - + <% } %> <% if (process.env.NODE_ENV === 'development') { %> - + <% } %> diff --git a/server/authUtils.ts b/server/authUtils.ts index 0631df3..301c41a 100644 --- a/server/authUtils.ts +++ b/server/authUtils.ts @@ -34,7 +34,7 @@ async function initJWKSet() { _remoteJWKSet = await createRemoteJWKSet(new URL(Config.auth.jwksUri)); } -const retrieveToken = async ( +const retrieveAndValidateToken = async ( req: Request, azureAdIssuer: OpenIdClient.Issuer ): Promise => { @@ -95,7 +95,7 @@ export const getOrRefreshOnBehalfOfToken = async ( req: Request, clientId: string ): Promise => { - const token = await retrieveToken(req, issuer); + const token = await retrieveAndValidateToken(req, issuer); if (!token) { throw Error( "Could not get on-behalf-of token because the token was undefined" @@ -104,6 +104,7 @@ export const getOrRefreshOnBehalfOfToken = async ( if (req.session.tokenCache === undefined) { req.session.tokenCache = {}; } + let cachedOboToken = req.session.tokenCache[clientId]; if (cachedOboToken && isNotExpired(cachedOboToken)) { return cachedOboToken.token; diff --git a/server/config.ts b/server/config.ts index ec6ad27..627a3b7 100644 --- a/server/config.ts +++ b/server/config.ts @@ -43,16 +43,6 @@ export interface ExternalAppConfig { export const server = { host: envVar({ name: "HOST", defaultValue: "localhost" }), port: Number.parseInt(envVar({ name: "PORT", defaultValue: "8080" })), - finnfastlegeUrl: envVar({ - name: "FINNFASTLEGE_URL", - defaultValue: "http://localhost:8080", - }), - - frontendDir: envVar({ - name: "FRONTEND_DIR", - defaultValue: path.join(__dirname, "frontend"), - }), - sessionKey: envVar({ name: "SESSION_KEY" }), sessionCookieName: envVar({ name: "SESSION_COOKIE_NAME", @@ -109,10 +99,6 @@ export const auth = { responseMode: "query", tokenEndpointAuthSigningAlg: "RS256", - internarbeidsflatedecoratorHost: envVar({ - name: "INTERNARBEIDSFLATEDECORATOR_HOST", - }), - modiacontextholder: { applicationName: "modiacontextholder", clientId: envVar({ @@ -121,6 +107,7 @@ export const auth = { host: envVar({ name: "MODIACONTEXTHOLDER_HOST", }), + removePathPrefix: true, }, fastlegerest: { diff --git a/server/proxy.ts b/server/proxy.ts index 5162ee8..6e5f5c0 100644 --- a/server/proxy.ts +++ b/server/proxy.ts @@ -168,28 +168,5 @@ export const setupProxy = ( ); } ); - - router.use( - "/internarbeidsflatedecorator", - expressHttpProxy(Config.auth.internarbeidsflatedecoratorHost, { - https: true, - proxyReqPathResolver: (req) => { - return `/internarbeidsflatedecorator${req.url}`; - }, - proxyErrorHandler: (err, res, next) => { - console.log( - `Error in proxy for internarbeidsflatedecorator ${err.message}, ${err.code}` - ); - if (err && err.code === "ECONNREFUSED") { - console.log("proxyErrorHandler: Got ECONNREFUSED"); - return res - .status(503) - .send({ message: `Could not contact internarbeidsflatedecorator` }); - } - next(err); - }, - }) - ); - return router; }; diff --git a/src/api/axios.ts b/src/api/axios.ts index 0a4a57d..8639cce 100644 --- a/src/api/axios.ts +++ b/src/api/axios.ts @@ -75,3 +75,23 @@ export const get = ( } }); }; + +export const post = ( + url: string, + // eslint-disable-next-line @typescript-eslint/no-explicit-any + data: Record | Record[], + personIdent?: string +): Promise => { + return axios + .post(url, data, { + headers: defaultRequestHeaders(personIdent), + }) + .then((response) => response.data) + .catch(function (error) { + if (axios.isAxiosError(error)) { + handleAxiosError(error); + } else { + throw new ApiErrorException(generalError(error), error.code); + } + }); +}; diff --git a/src/api/constants.ts b/src/api/constants.ts index fb4430d..ce47fd6 100644 --- a/src/api/constants.ts +++ b/src/api/constants.ts @@ -1,3 +1,4 @@ +export const MODIACONTEXTHOLDER_ROOT = "/modiacontextholder/api"; export const SYFOPERSON_ROOT = "/syfoperson/api/v2"; export const FASTLEGEREST_ROOT = "/fastlegerest/api/v2"; export const TILGANGSKONTROLL_AD_PATH = diff --git a/src/data/modiacontext/useAktivBruker.ts b/src/data/modiacontext/useAktivBruker.ts new file mode 100644 index 0000000..60c7714 --- /dev/null +++ b/src/data/modiacontext/useAktivBruker.ts @@ -0,0 +1,14 @@ +import { post } from "@/api/axios"; +import { MODIACONTEXTHOLDER_ROOT } from "@/api/constants"; +import { useMutation } from "@tanstack/react-query"; + +const NY_AKTIV_BRUKER = "NY_AKTIV_BRUKER"; + +export const useAktivBruker = () => + useMutation({ + mutationFn: (fnr: string) => + post(`${MODIACONTEXTHOLDER_ROOT}/context`, { + verdi: fnr, + eventType: NY_AKTIV_BRUKER, + }), + }); diff --git a/src/decorator/Decorator.tsx b/src/decorator/Decorator.tsx index 787b313..1b1e203 100644 --- a/src/decorator/Decorator.tsx +++ b/src/decorator/Decorator.tsx @@ -3,16 +3,23 @@ import NAVSPA from "@navikt/navspa"; import { DecoratorProps } from "./decoratorProps"; import decoratorConfig from "./decoratorconfig"; import { fullNaisUrlDefault } from "@/utils/miljoUtil"; +import { useAktivBruker } from "@/data/modiacontext/useAktivBruker"; const InternflateDecorator = NAVSPA.importer( - "internarbeidsflatefs" + "internarbeidsflate-decorator-v3" ); const Decorator = () => { + const aktivBruker = useAktivBruker(); + const handlePersonsokSubmit = (nyttFnr: string) => { - const host = "syfomodiaperson"; - const path = `/sykefravaer/${nyttFnr}`; - window.location.href = fullNaisUrlDefault(host, path); + aktivBruker.mutate(nyttFnr, { + onSuccess: () => { + const host = "syfomodiaperson"; + const path = `/sykefravaer`; + window.location.href = fullNaisUrlDefault(host, path); + }, + }); }; const config = useCallback(decoratorConfig, [handlePersonsokSubmit])( diff --git a/src/decorator/decoratorProps.ts b/src/decorator/decoratorProps.ts index c10492a..538a8d0 100644 --- a/src/decorator/decoratorProps.ts +++ b/src/decorator/decoratorProps.ts @@ -1,48 +1,67 @@ -interface TogglesConfig { - visVeileder: boolean; +export interface DecoratorProps { + enhet?: string | undefined; // Konfigurasjon av enhet-kontekst + accessToken?: string | undefined; // Manuell innsending av JWT, settes som Authorization-header. Om null sendes cookies vha credentials: 'include' + fnr?: string | undefined; // Konfigurasjon av fødselsnummer-kontekst + userKey?: string | undefined; // Om man ikke ønsker å bruke fnr i urler, kan andre apper kalle contextholder for å generere en midlertidig kode. Hvis App A skal navigere til App B som har dekoratøren, må App A først sende en post request til /fnr-code/generate med {fnr: string} i bodyen, dette returnerer {fnr: string, code: string} til App A. App A kan så navigere til App B og sende med denne koden. App B kan så sende den koden inn til dekoratøren i userKey propen og så henter dekoratøren fnr for den koden fra contextholderen. + enableHotkeys?: boolean | undefined; // Aktivere hurtigtaster + fetchActiveEnhetOnMount?: boolean | undefined; // Om enhet er undefined fra container appen, og denne er satt til true, henter den sist aktiv enhet og bruker denne. + fetchActiveUserOnMount?: boolean | undefined; // Om fnr er undefined fra container appen, og denne er satt til true for at den skal hente siste aktiv fnr. + onEnhetChanged: () => void; // Kalles når enheten endres + onFnrChanged: (fnr?: string | null) => void; // Kalles når fnr enheten endres + onLinkClick?: (link: { text: string; url: string }) => void; // Kan brukes for å legge til callbacks ved klikk på lenker i menyen. Merk at callbacken ikke kan awaites og man må selv håndtere at siden lukkes. Nyttig for å f.eks tracke navigasjon events i amplitude + appName: string; // Navn på applikasjonen + hotkeys?: Hotkey[]; // Konfigurasjon av hurtigtaster + markup?: Markup; // Egen HTML + showEnheter: boolean; // Vis enheter + showSearchArea: boolean; // Vis søkefelt + showHotkeys: boolean; // Vis hurtigtaster + environment: Environment; // Miljø som skal brukes. + urlFormat: UrlFormat; // URL format + proxy?: string | undefined; // Manuell overstyring av urlene til BFFs. Gjør alle kall til relativt path hvis true, og bruker verdien som domene om satt til en string. Default: false } -interface Markup { - etterSokefelt?: string; +export interface Markup { + etterSokefelt?: string; // Gir muligheten for sende inn egen html som blir en del av dekoratøren } -export interface ControlledContextvalue extends BaseContextvalue { - value?: string; -} -interface UncontrolledContextvalue extends BaseContextvalue { - initialValue?: string; +export interface Enhet { + readonly enhetId: string; + readonly navn: string; } -interface BaseContextvalue { - display: T; - onChange(value?: string): void; - skipModal?: boolean; - ignoreWsEvents?: boolean; -} +// Miljø +export type Environment = + | "q0" + | "q1" + | "q2" + | "q3" + | "q4" + | "prod" + | "local" + | "mock"; -export type Contextvalue = - | ControlledContextvalue - | UncontrolledContextvalue; +export type UrlFormat = "LOCAL" | "NAV_NO" | "ANSATT"; // UrlFormat. Brukes om proxy ikke er satt & i url til websocket. -export enum EnhetDisplay { - ENHET = "ENHET", - ENHET_VALG = "ENHET_VALG", +export interface HotkeyObject { + char: string; + altKey?: boolean; + ctrlKey?: boolean; + metaKey?: boolean; + shiftKey?: boolean; } -export enum FnrDisplay { - SOKEFELT = "SOKEFELT", +export interface HotkeyDescription { + key: HotkeyObject; + description: string; + forceOverride?: boolean; } -type EnhetContextvalue = Contextvalue; -type FnrContextvalue = Contextvalue; -type ProxyConfig = boolean | string; +export interface ActionHotKey extends HotkeyDescription { + action(event: KeyboardEvent): void; +} -export interface DecoratorProps { - appname: string; - fnr?: FnrContextvalue; - enhet?: EnhetContextvalue; - toggles?: TogglesConfig; - markup?: Markup; - useProxy?: ProxyConfig; - accessToken?: string; +export interface DocumentingHotKey extends HotkeyDescription { + documentationOnly: boolean; } + +export type Hotkey = ActionHotKey | DocumentingHotKey; diff --git a/src/decorator/decoratorconfig.ts b/src/decorator/decoratorconfig.ts index 13f97e2..c0ca6f5 100644 --- a/src/decorator/decoratorconfig.ts +++ b/src/decorator/decoratorconfig.ts @@ -1,34 +1,43 @@ -import { DecoratorProps, EnhetDisplay, FnrDisplay } from "./decoratorProps"; +import { DecoratorProps, Environment, UrlFormat } from "./decoratorProps"; +import { erAnsattDev, erDev, erLokal, erProd } from "@/utils/miljoUtil"; -const RESET_VALUE = "\u0000"; - -const decoratorconfig = (setFnr: (fnr: string) => void): DecoratorProps => { +const decoratorConfig = (setFnr: (fnr: string) => void): DecoratorProps => { return { - appname: "Sykefraværsoppfølging", - fnr: { - initialValue: RESET_VALUE, - display: FnrDisplay.SOKEFELT, - ignoreWsEvents: true, - skipModal: true, - onChange: (value) => { - if (value) { - setFnr(value); - } - }, - }, - enhet: { - initialValue: undefined, - display: EnhetDisplay.ENHET_VALG, - onChange(): void { - /* Do nothing */ - }, - skipModal: true, + appName: "Sykefraværsoppfølging", + fetchActiveEnhetOnMount: false, + onEnhetChanged: () => { + // do nothing }, - toggles: { - visVeileder: true, + onFnrChanged: (fnr?: string | null) => { + if (fnr) { + setFnr(fnr); + } }, - useProxy: true, + showEnheter: true, + showSearchArea: true, + showHotkeys: false, + environment: getEnvironment(), + urlFormat: getUrlFormat(), + proxy: "/modiacontextholder", }; }; -export default decoratorconfig; +const getEnvironment = (): Environment => { + if (erProd()) { + return "prod"; + } else if (erDev()) { + return "q2"; + } else { + return "local"; + } +}; + +const getUrlFormat = (): UrlFormat => { + if (erAnsattDev()) { + return "ANSATT"; + } else if (erLokal()) { + return "LOCAL"; + } else return "NAV_NO"; +}; + +export default decoratorConfig; diff --git a/src/faro.ts b/src/faro.ts index 9c1ec11..4c50ae8 100644 --- a/src/faro.ts +++ b/src/faro.ts @@ -1,11 +1,11 @@ import { getWebInstrumentations, initializeFaro } from "@grafana/faro-react"; import { TracingInstrumentation } from "@grafana/faro-web-tracing"; -import { erLokal, erPreProd } from "@/utils/miljoUtil"; +import { erLokal, erDev } from "@/utils/miljoUtil"; const getUrl = () => { if (erLokal()) { return "/collect"; - } else if (erPreProd()) { + } else if (erDev()) { return "https://telemetry.ekstern.dev.nav.no/collect"; } else { return "https://telemetry.nav.no/collect"; diff --git a/src/utils/fnrValideringUtil.ts b/src/utils/fnrValideringUtil.ts index f32c0b5..23478e7 100644 --- a/src/utils/fnrValideringUtil.ts +++ b/src/utils/fnrValideringUtil.ts @@ -1,4 +1,4 @@ -import { erPreProd } from "@/utils/miljoUtil"; +import { erDev } from "@/utils/miljoUtil"; const kontrollRekke1 = [3, 7, 6, 1, 8, 9, 4, 5, 2]; const kontrollRekke2 = [5, 4, 3, 2, 7, 6, 5, 4, 3, 2]; @@ -35,7 +35,7 @@ const erGyldigSkatteetatenTestdato = (dag: number, maned: number) => { const erGyldigTestdato = (dag: number, maned: number) => { return ( - erPreProd() && + erDev() && (erGyldigNavTestdato(dag, maned) || erGyldigSkatteetatenTestdato(dag, maned)) ); diff --git a/src/utils/miljoUtil.ts b/src/utils/miljoUtil.ts index 911a27d..a26c335 100644 --- a/src/utils/miljoUtil.ts +++ b/src/utils/miljoUtil.ts @@ -1,16 +1,30 @@ -export const erPreProd = () => { +export const erDev = (): boolean => { return ( + window.location.href.indexOf("dev.intern.nav.no") > -1 || window.location.href.indexOf("intern.dev.nav.no") > -1 || - window.location.href.indexOf("dev.intern.nav.no") > -1 + erAnsattDev() ); }; -export const erLokal = () => { +export const erAnsattDev = (): boolean => { + return window.location.href.indexOf("ansatt.dev.nav.no") > -1; +}; + +export const erLokal = (): boolean => { return window.location.host.indexOf("localhost") > -1; }; +export function erProd(): boolean { + return window.location.href.indexOf("finnfastlege.intern.nav.no") > -1; +} -export const finnNaisUrlDefault = () => { - return erPreProd() ? ".dev.intern.nav.no" : ".intern.nav.no"; +const finnNaisUrlDefault = (): string => { + if (erAnsattDev()) { + return ".ansatt.dev.nav.no"; + } else if (erDev()) { + return ".intern.dev.nav.no"; + } else { + return ".intern.nav.no"; + } }; export const fullNaisUrlDefault = (host: string, path: string) => {