Skip to content

Commit f396d6e

Browse files
committed
TokenX endepunkt for personalia
1 parent 85ab870 commit f396d6e

File tree

13 files changed

+186
-21
lines changed

13 files changed

+186
-21
lines changed

build.gradle.kts

+2
Original file line numberDiff line numberDiff line change
@@ -45,6 +45,7 @@ dependencies {
4545
implementation(TmsKtorTokenSupport.tokendingsExchange)
4646
implementation(TmsKtorTokenSupport.idportenSidecar)
4747
implementation(TmsKtorTokenSupport.azureExchange)
48+
implementation(TmsKtorTokenSupport.tokenXValidation)
4849
implementation(Logstash.logbackEncoder)
4950
implementation(Micrometer.registryPrometheus)
5051
implementation(Prometheus.common)
@@ -67,6 +68,7 @@ dependencies {
6768
testImplementation(Jjwt.api)
6869
testImplementation(Mockk.mockk)
6970
testImplementation(TmsKtorTokenSupport.idportenSidecarMock)
71+
testImplementation(TmsKtorTokenSupport.tokenXValidationMock)
7072
testImplementation(TmsCommonLib.testutils)
7173
}
7274

nais/dev-gcp/nais.yaml

+3
Original file line numberDiff line numberDiff line change
@@ -43,6 +43,9 @@ spec:
4343
cpu: "20m"
4444
memory: 128Mi
4545
accessPolicy:
46+
inbound:
47+
rules:
48+
- application: tms-min-side-proxy
4649
outbound:
4750
external:
4851
- host: veilarboppfolging.dev-fss-pub.nais.io

nais/prod-gcp/nais.yaml

+3
Original file line numberDiff line numberDiff line change
@@ -41,6 +41,9 @@ spec:
4141
cpu: "50m"
4242
memory: 256Mi
4343
accessPolicy:
44+
inbound:
45+
rules:
46+
- application: tms-min-side-proxy
4447
outbound:
4548
external:
4649
- host: veilarboppfolging.prod-fss-pub.nais.io

src/main/kotlin/no/nav/tms/min/side/proxy/Application.kt

+10
Original file line numberDiff line numberDiff line change
@@ -16,6 +16,7 @@ import io.ktor.server.engine.embeddedServer
1616
import io.ktor.server.netty.Netty
1717
import no.nav.tms.common.util.config.StringEnvVar
1818
import no.nav.tms.min.side.proxy.personalia.NavnFetcher
19+
import no.nav.tms.min.side.proxy.personalia.PersonaliaFetcher
1920
import no.nav.tms.token.support.azure.exchange.AzureServiceBuilder
2021
import no.nav.tms.token.support.tokendings.exchange.TokendingsServiceBuilder
2122

@@ -93,6 +94,14 @@ data class AppConfiguration(
9394
pdlBehandlingsnummer,
9495
tokendingsService
9596
)
97+
98+
val personaliaFetcher = PersonaliaFetcher(
99+
httpClient,
100+
pdlApiUrl,
101+
pdlApiClientId,
102+
pdlBehandlingsnummer,
103+
tokendingsService
104+
)
96105
}
97106

98107
fun ObjectMapper.jsonConfig() {
@@ -110,6 +119,7 @@ fun ApplicationEngineEnvironmentBuilder.envConfig(appConfig: AppConfiguration) {
110119
contentFetcher = appConfig.contentFecther,
111120
externalContentFetcher = appConfig.externalContentFetcher,
112121
navnFetcher = appConfig.navnFetcher,
122+
personaliaFetcher = appConfig.personaliaFetcher,
113123
unleash = setupUnleash(
114124
unleashApiUrl = appConfig.unleashServerApiUrl,
115125
unleashApiKey = appConfig.unleashServerApiToken,

src/main/kotlin/no/nav/tms/min/side/proxy/personalia/NavnFetcher.kt

+2-2
Original file line numberDiff line numberDiff line change
@@ -78,7 +78,7 @@ class NavnFetcher(
7878

7979
class HentNavnException(message: String, cause: Exception? = null): Exception(message, cause)
8080

81-
private class HentNavn(ident: String) {
81+
class HentNavn(ident: String) {
8282
val query = """
8383
query(${'$'}ident: ID!) {
8484
hentPerson(ident: ${'$'}ident) {
@@ -102,7 +102,7 @@ fun String.compactJson(): String =
102102
.replace("\n", " ")
103103
.replace("\\s+".toRegex(), " ")
104104

105-
private data class HentNavnResponse(
105+
data class HentNavnResponse(
106106
val data: HentNavnData?,
107107
val errors: List<Map<String, Any>>?
108108
) {
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,78 @@
1+
package no.nav.tms.min.side.proxy.personalia
2+
3+
import com.github.benmanes.caffeine.cache.Caffeine
4+
import io.github.oshai.kotlinlogging.KotlinLogging
5+
import io.ktor.client.*
6+
import io.ktor.client.call.*
7+
import io.ktor.client.request.*
8+
import io.ktor.http.*
9+
import kotlinx.coroutines.Dispatchers
10+
import kotlinx.coroutines.runBlocking
11+
import no.nav.tms.token.support.tokendings.exchange.TokendingsService
12+
import no.nav.tms.token.support.tokenx.validation.user.TokenXUser
13+
import java.time.Duration
14+
15+
class PersonaliaFetcher(
16+
private val client: HttpClient,
17+
private val pdlUrl: String,
18+
private val pdlClientId: String,
19+
private val pdlBehandlingsnummer: String,
20+
private val tokendingsService: TokendingsService
21+
) {
22+
23+
private val cache = Caffeine.newBuilder()
24+
.maximumSize(10000)
25+
.expireAfterWrite(Duration.ofMinutes(5))
26+
.build<String, String>()
27+
28+
private val log = KotlinLogging.logger {}
29+
private val securelog = KotlinLogging.logger("secureLog")
30+
31+
fun getNavn(user: TokenXUser): String {
32+
return cache.get(user.ident) {
33+
fetchNavn(user)
34+
}
35+
}
36+
37+
private fun fetchNavn(user: TokenXUser): String = runBlocking(Dispatchers.IO) {
38+
tokendingsService.exchangeToken(user.tokenString, pdlClientId)
39+
.let { token -> queryForNavn(user.ident, token) }
40+
.let { response -> checkForErrors(response) }
41+
.hentPerson.fullnavn
42+
}
43+
44+
private suspend fun queryForNavn(ident: String, token: String): HentNavnResponse {
45+
val response = client.post {
46+
url(pdlUrl)
47+
header(HttpHeaders.Authorization, "Bearer $token")
48+
header("Behandlingsnummer", pdlBehandlingsnummer)
49+
header("Tema", "GEN")
50+
contentType(ContentType.Application.Json)
51+
setBody(HentNavn(ident))
52+
}
53+
54+
if (!response.status.isSuccess()) {
55+
throw HentNavnException("Fikk http-feil fra PDL")
56+
}
57+
58+
return try {
59+
response.body()
60+
} catch (e: Exception) {
61+
securelog.error(e) { "Klarer ikke tolke svar fra PDL." }
62+
throw HentNavnException("Klarte ikke tolke svar fra PDL", e)
63+
}
64+
}
65+
66+
private fun checkForErrors(response: HentNavnResponse): HentNavnResponse.HentNavnData {
67+
68+
response.errors?.let { errors ->
69+
if (errors.isNotEmpty()) {
70+
log.warn { "Feil i GraphQL-responsen: $errors" }
71+
throw HentNavnException("Feil i responsen under henting av navn")
72+
}
73+
}
74+
75+
return response.data?: throw HentNavnException("Ingen data i graphql-svar.")
76+
}
77+
}
78+

src/main/kotlin/no/nav/tms/min/side/proxy/personalia/navnRoutes.kt

+3-3
Original file line numberDiff line numberDiff line change
@@ -26,10 +26,10 @@ fun Route.navnRoutes(navnFetcher: NavnFetcher) {
2626
}
2727
}
2828

29-
private data class Navn(val navn: String)
30-
private data class Ident(val ident: String)
29+
data class Navn(val navn: String)
30+
data class Ident(val ident: String)
3131

32-
private data class NavnAndIdent(
32+
data class NavnAndIdent(
3333
val navn: String?,
3434
val ident: String
3535
)
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,21 @@
1+
package no.nav.tms.min.side.proxy.personalia
2+
3+
import io.ktor.server.application.*
4+
import io.ktor.server.response.*
5+
import io.ktor.server.routing.*
6+
import io.ktor.util.pipeline.*
7+
import no.nav.tms.token.support.tokenx.validation.user.TokenXUserFactory
8+
9+
fun Route.personaliaRoutes(personaliaFetcher: PersonaliaFetcher) {
10+
get("/personalia") {
11+
try {
12+
personaliaFetcher.getNavn(user)
13+
.let { navn -> call.respond(NavnAndIdent(navn, user.ident)) }
14+
} catch (e: HentNavnException) {
15+
call.respond(NavnAndIdent(navn = null, ident = user.ident))
16+
}
17+
}
18+
}
19+
20+
private val PipelineContext<Unit, ApplicationCall>.user
21+
get() = TokenXUserFactory.createTokenXUser(call)

src/main/kotlin/no/nav/tms/min/side/proxy/proxyApi.kt

+18-3
Original file line numberDiff line numberDiff line change
@@ -18,13 +18,13 @@ import io.micrometer.prometheus.PrometheusMeterRegistry
1818
import io.github.oshai.kotlinlogging.KotlinLogging
1919
import io.ktor.serialization.jackson.*
2020
import no.nav.tms.common.metrics.installTmsMicrometerMetrics
21-
import no.nav.tms.min.side.proxy.personalia.HentNavnException
22-
import no.nav.tms.min.side.proxy.personalia.NavnFetcher
23-
import no.nav.tms.min.side.proxy.personalia.navnRoutes
2421
import no.nav.tms.token.support.idporten.sidecar.IdPortenLogin
2522
import no.nav.tms.token.support.idporten.sidecar.LevelOfAssurance.SUBSTANTIAL
2623
import no.nav.tms.token.support.idporten.sidecar.idPorten
2724
import no.nav.tms.common.observability.ApiMdc
25+
import no.nav.tms.min.side.proxy.personalia.*
26+
import no.nav.tms.token.support.tokenx.validation.TokenXAuthenticator
27+
import no.nav.tms.token.support.tokenx.validation.tokenX
2828

2929
private val log = KotlinLogging.logger {}
3030
private val securelog = KotlinLogging.logger("secureLog")
@@ -34,15 +34,26 @@ fun Application.proxyApi(
3434
contentFetcher: ContentFetcher,
3535
externalContentFetcher: ExternalContentFetcher,
3636
navnFetcher: NavnFetcher,
37+
personaliaFetcher: PersonaliaFetcher,
3738
idportenAuthInstaller: Application.() -> Unit = {
3839
authentication {
3940
idPorten {
4041
setAsDefault = true
4142
levelOfAssurance = SUBSTANTIAL
4243
}
44+
tokenX {
45+
setAsDefault = false
46+
}
4347
}
4448
install(IdPortenLogin)
4549
},
50+
tokenXAuthInstaller: Application.() -> Unit = {
51+
authentication {
52+
tokenX {
53+
setAsDefault = false
54+
}
55+
}
56+
},
4657
unleash: Unleash
4758
) {
4859
val collectorRegistry = PrometheusMeterRegistry(PrometheusConfig.DEFAULT)
@@ -86,6 +97,7 @@ fun Application.proxyApi(
8697
}
8798

8899
idportenAuthInstaller()
100+
tokenXAuthInstaller()
89101

90102
installTmsMicrometerMetrics {
91103
installMicrometerPlugin = true
@@ -120,6 +132,9 @@ fun Application.proxyApi(
120132
)
121133
}
122134
}
135+
authenticate(TokenXAuthenticator.name) {
136+
personaliaRoutes(personaliaFetcher)
137+
}
123138
}
124139

125140
configureShutdownHook(contentFetcher)

src/test/kotlin/no/nav/tms/min/side/proxy/GetRoutesTest.kt

+15-5
Original file line numberDiff line numberDiff line change
@@ -49,7 +49,8 @@ class GetRoutesTest {
4949
mockApi(
5050
contentFetcher = contentFecther(proxyHttpClient),
5151
externalContentFetcher = externalContentFetcher(proxyHttpClient),
52-
navnFetcher = mockk()
52+
navnFetcher = mockk(),
53+
personaliaFetcher = mockk()
5354
)
5455

5556
initExternalServices(
@@ -96,7 +97,8 @@ class GetRoutesTest {
9697
mockApi(
9798
contentFetcher = contentFecther(proxyHttpClient),
9899
externalContentFetcher = externalContentFetcher(proxyHttpClient),
99-
navnFetcher = mockk()
100+
navnFetcher = mockk(),
101+
personaliaFetcher = mockk()
100102
)
101103

102104
initExternalServices(
@@ -129,7 +131,8 @@ class GetRoutesTest {
129131
mockApi(
130132
contentFetcher = contentFecther(proxyHttpClient),
131133
externalContentFetcher = externalContentFetcher(proxyHttpClient),
132-
navnFetcher = mockk()
134+
navnFetcher = mockk(),
135+
personaliaFetcher = mockk()
133136
)
134137

135138
client.get("/internal/isAlive").status shouldBe HttpStatusCode.OK
@@ -148,7 +151,8 @@ class GetRoutesTest {
148151
contentFetcher = contentFecther(proxyHttpClient),
149152
externalContentFetcher = externalContentFetcher(proxyHttpClient),
150153
unleash = unleash,
151-
navnFetcher = mockk()
154+
navnFetcher = mockk(),
155+
personaliaFetcher = mockk()
152156
)
153157

154158
client.get("/featuretoggles").assert {
@@ -159,7 +163,12 @@ class GetRoutesTest {
159163

160164
@Test
161165
fun authPing() = testApplication {
162-
mockApi(contentFetcher = mockk(), externalContentFetcher = mockk(), navnFetcher = mockk())
166+
mockApi(
167+
contentFetcher = mockk(),
168+
externalContentFetcher = mockk(),
169+
navnFetcher = mockk(),
170+
personaliaFetcher = mockk()
171+
)
163172
client.get("/authPing").status shouldBe HttpStatusCode.OK
164173
}
165174

@@ -170,6 +179,7 @@ class GetRoutesTest {
170179
contentFetcher = mockk(),
171180
externalContentFetcher = mockk(),
172181
navnFetcher = mockk(),
182+
personaliaFetcher = mockk(),
173183
levelOfAssurance = LevelOfAssurance.SUBSTANTIAL
174184
)
175185

src/test/kotlin/no/nav/tms/min/side/proxy/PostRoutesTest.kt

+2-1
Original file line numberDiff line numberDiff line change
@@ -21,7 +21,8 @@ class PostRoutesTest {
2121
mockApi(
2222
contentFetcher = contentFecther(proxyHttpClient),
2323
externalContentFetcher = externalContentFetcher(proxyHttpClient),
24-
navnFetcher = mockk()
24+
navnFetcher = mockk(),
25+
personaliaFetcher = mockk()
2526
)
2627

2728
initExternalServices(

0 commit comments

Comments
 (0)