Skip to content

Commit 51760a6

Browse files
committed
feat: support objects and lists in requestmapping claims
* should also ensure that params such as scope are subject to templating * formatting and linting
1 parent 8ad715c commit 51760a6

File tree

4 files changed

+239
-181
lines changed

4 files changed

+239
-181
lines changed
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,31 @@
1+
package no.nav.security.mock.oauth2.extensions
2+
3+
/**
4+
* Replaces all template values denoted with ${key} in a map with the corresponding values from the templates map.
5+
*
6+
* @param templates a map of template values
7+
* @return a new map with all template values replaced
8+
*/
9+
fun Map<String, Any>.replaceValues(templates: Map<String, Any>): Map<String, Any> {
10+
fun replaceTemplateString(
11+
value: String,
12+
templates: Map<String, Any>,
13+
): String {
14+
val regex = Regex("""\$\{(\w+)\}""")
15+
return regex.replace(value) { matchResult ->
16+
val key = matchResult.groupValues[1]
17+
templates[key]?.toString() ?: matchResult.value
18+
}
19+
}
20+
21+
fun replaceValue(value: Any): Any {
22+
return when (value) {
23+
is String -> replaceTemplateString(value, templates)
24+
is List<*> -> value.map { it?.let { replaceValue(it) } }
25+
is Map<*, *> -> value.mapValues { v -> v.value?.let { replaceValue(it) } }
26+
else -> value
27+
}
28+
}
29+
30+
return this.mapValues { replaceValue(it.value) }
31+
}

src/main/kotlin/no/nav/security/mock/oauth2/token/OAuth2TokenCallback.kt

+46-99
Original file line numberDiff line numberDiff line change
@@ -3,15 +3,11 @@ package no.nav.security.mock.oauth2.token
33
import com.nimbusds.jose.JOSEObjectType
44
import com.nimbusds.oauth2.sdk.GrantType
55
import com.nimbusds.oauth2.sdk.TokenRequest
6-
import kotlinx.serialization.json.Json
7-
import kotlinx.serialization.json.JsonArray
8-
import kotlinx.serialization.json.JsonObject
9-
import kotlinx.serialization.json.JsonPrimitive
106
import no.nav.security.mock.oauth2.extensions.clientIdAsString
117
import no.nav.security.mock.oauth2.extensions.grantType
8+
import no.nav.security.mock.oauth2.extensions.replaceValues
129
import no.nav.security.mock.oauth2.extensions.scopesWithoutOidcScopes
1310
import no.nav.security.mock.oauth2.extensions.tokenExchangeGrantOrNull
14-
import no.nav.security.mock.oauth2.http.objectMapper
1511
import java.time.Duration
1612
import java.util.*
1713

@@ -31,49 +27,49 @@ interface OAuth2TokenCallback {
3127

3228
// TODO: for JwtBearerGrant and TokenExchange should be able to ovverride sub, make sub nullable and return some default
3329
open class DefaultOAuth2TokenCallback
34-
@JvmOverloads
35-
constructor(
36-
private val issuerId: String = "default",
37-
private val subject: String = UUID.randomUUID().toString(),
38-
private val typeHeader: String = JOSEObjectType.JWT.type,
39-
// needs to be nullable in order to know if a list has explicitly been set, empty list should be a allowable value
40-
private val audience: List<String>? = null,
41-
private val claims: Map<String, Any> = emptyMap(),
42-
private val expiry: Long = 3600,
43-
) : OAuth2TokenCallback {
44-
override fun issuerId(): String = issuerId
45-
46-
override fun subject(tokenRequest: TokenRequest): String {
47-
return when (GrantType.CLIENT_CREDENTIALS) {
48-
tokenRequest.grantType() -> tokenRequest.clientIdAsString()
49-
else -> subject
30+
@JvmOverloads
31+
constructor(
32+
private val issuerId: String = "default",
33+
private val subject: String = UUID.randomUUID().toString(),
34+
private val typeHeader: String = JOSEObjectType.JWT.type,
35+
// needs to be nullable in order to know if a list has explicitly been set, empty list should be a allowable value
36+
private val audience: List<String>? = null,
37+
private val claims: Map<String, Any> = emptyMap(),
38+
private val expiry: Long = 3600,
39+
) : OAuth2TokenCallback {
40+
override fun issuerId(): String = issuerId
41+
42+
override fun subject(tokenRequest: TokenRequest): String {
43+
return when (GrantType.CLIENT_CREDENTIALS) {
44+
tokenRequest.grantType() -> tokenRequest.clientIdAsString()
45+
else -> subject
46+
}
5047
}
51-
}
5248

53-
override fun typeHeader(tokenRequest: TokenRequest): String {
54-
return typeHeader
55-
}
56-
57-
override fun audience(tokenRequest: TokenRequest): List<String> {
58-
val audienceParam = tokenRequest.tokenExchangeGrantOrNull()?.audience
59-
return when {
60-
audience != null -> audience
61-
audienceParam != null -> audienceParam
62-
tokenRequest.scope != null -> tokenRequest.scopesWithoutOidcScopes()
63-
else -> listOf("default")
49+
override fun typeHeader(tokenRequest: TokenRequest): String {
50+
return typeHeader
6451
}
65-
}
6652

67-
override fun addClaims(tokenRequest: TokenRequest): Map<String, Any> =
68-
mutableMapOf<String, Any>(
69-
"tid" to issuerId,
70-
).apply {
71-
putAll(claims)
72-
put("azp", tokenRequest.clientIdAsString())
53+
override fun audience(tokenRequest: TokenRequest): List<String> {
54+
val audienceParam = tokenRequest.tokenExchangeGrantOrNull()?.audience
55+
return when {
56+
audience != null -> audience
57+
audienceParam != null -> audienceParam
58+
tokenRequest.scope != null -> tokenRequest.scopesWithoutOidcScopes()
59+
else -> listOf("default")
60+
}
7361
}
7462

75-
override fun tokenExpiry(): Long = expiry
76-
}
63+
override fun addClaims(tokenRequest: TokenRequest): Map<String, Any> =
64+
mutableMapOf<String, Any>(
65+
"tid" to issuerId,
66+
).apply {
67+
putAll(claims)
68+
put("azp", tokenRequest.clientIdAsString())
69+
}
70+
71+
override fun tokenExpiry(): Long = expiry
72+
}
7773

7874
data class RequestMappingTokenCallback(
7975
val issuerId: String,
@@ -94,54 +90,14 @@ data class RequestMappingTokenCallback(
9490

9591
private fun List<RequestMapping>.getClaims(tokenRequest: TokenRequest): Map<String, Any> {
9692
val claims = firstOrNull { it.isMatch(tokenRequest) }?.claims ?: emptyMap()
97-
98-
// TODO: hack choose first element. Rewrite to support multiple elements and custom objects
99-
val params = (tokenRequest.toHTTPRequest().bodyAsFormParameters.map {
100-
it.key to it.value.first()
101-
}).toMap() + mapOf("clientId" to tokenRequest.clientIdAsString())
102-
103-
return claims.mapValues { (_, value) ->
104-
val v = objectMapper.writeValueAsString(value)
105-
val jsonElement = Json.parseToJsonElement(v)
106-
when (jsonElement) {
107-
is JsonPrimitive ->
108-
if (jsonElement.isString) {
109-
replaceVariables(jsonElement.content, params)
110-
} else {
111-
jsonElement.content
112-
}
113-
114-
is JsonObject -> {
115-
jsonElement.mapValues { (_, value) ->
116-
if (value is JsonPrimitive) {
117-
replaceVariables(value.content, params)
118-
} else if (value is JsonArray)
119-
value.map { element ->
120-
if (element is JsonPrimitive) {
121-
replaceVariables(element.content, params)
122-
} else {
123-
element
124-
}
125-
}
126-
else {
127-
value
128-
}
129-
}
130-
}
131-
132-
is JsonArray -> {
133-
jsonElement.map { element ->
134-
if (element is JsonPrimitive) {
135-
replaceVariables(element.content, params)
136-
} else {
137-
element
138-
}
139-
}
140-
}
141-
142-
else -> value
143-
}
144-
}
93+
val templateParams = tokenRequest.toHTTPRequest().bodyAsFormParameters.mapValues { it.value.joinToString(separator = " ") }
94+
95+
// in case client_id is not set as form param but as basic auth, we add it to the template params in two different formats for backwards compatibility
96+
return claims.replaceValues(
97+
templateParams +
98+
mapOf("clientId" to tokenRequest.clientIdAsString()) +
99+
mapOf("client_id" to tokenRequest.clientIdAsString()),
100+
)
145101
}
146102

147103
private inline fun <reified T> List<RequestMapping>.getClaimOrNull(
@@ -150,15 +106,6 @@ data class RequestMappingTokenCallback(
150106
): T? = getClaims(tokenRequest)[key] as? T
151107

152108
private fun List<RequestMapping>.getTypeHeader(tokenRequest: TokenRequest) = firstOrNull { it.isMatch(tokenRequest) }?.typeHeader ?: JOSEObjectType.JWT.type
153-
154-
private fun replaceVariables(
155-
input: String,
156-
replacements: Map<String, String>,
157-
): String {
158-
return replacements.entries.fold(input) { acc, (key, value) ->
159-
acc.replace("\${$key}", value)
160-
}
161-
}
162109
}
163110

164111
data class RequestMapping(
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,73 @@
1+
package no.nav.security.mock.oauth2.extensions
2+
3+
import io.kotest.assertions.asClue
4+
import io.kotest.matchers.shouldBe
5+
import org.junit.jupiter.api.Test
6+
7+
class TemplateTest {
8+
@Test
9+
fun `template values in map should be replaced`() {
10+
val templates =
11+
mapOf(
12+
"templateVal1" to "val1",
13+
"templateVal2" to "val2",
14+
"templateListVal" to "listVal1",
15+
)
16+
17+
mapOf(
18+
"object1" to mapOf("key1" to "\${templateVal1}"),
19+
"object2" to "\${templateVal2}",
20+
"nestedObject" to mapOf("nestedKey" to mapOf("nestedKeyAgain" to "\${templateVal2}")),
21+
"list1" to listOf("\${templateListVal}"),
22+
).replaceValues(templates).asClue {
23+
it["object1"] shouldBe mapOf("key1" to "val1")
24+
it["list1"] shouldBe listOf("listVal1")
25+
println(it)
26+
}
27+
}
28+
29+
// Example usage
30+
fun main() {
31+
val jsonData: Map<String, Any> =
32+
mapOf(
33+
"myobject" to
34+
mapOf(
35+
"participantId" to "\${someTemlateVal}",
36+
"actAs" to listOf("\${templateVal1}, \${templateVal2}"),
37+
"readAs" to listOf("\${templateVal2}"),
38+
),
39+
"myobject2" to "someValue",
40+
)
41+
42+
val templates =
43+
mapOf(
44+
"someTemlateVal" to "participant123",
45+
"templateVal1" to "actor123",
46+
"templateVal2" to "reader123",
47+
)
48+
49+
val replacedData = jsonData.replaceValues(templates)
50+
println(replacedData)
51+
}
52+
53+
/*fun replaceTemplates(data: Map<String, Any>, templates: Map<String, String>): Map<String, Any> {
54+
fun replaceValue(value: Any): Any {
55+
return when (value) {
56+
is String -> replaceTemplateString(value, templates)
57+
is List<*> -> value.map { replaceValue(it) }
58+
is Map<*, *> -> value.mapValues { replaceValue(it.value) }
59+
else -> value
60+
}
61+
}
62+
63+
fun replaceTemplateString(value: String, templates: Map<String, String>): String {
64+
val regex = Regex("""\$\{(\w+)\}""")
65+
return regex.replace(value) { matchResult ->
66+
val key = matchResult.groupValues[1]
67+
templates[key] ?: matchResult.value
68+
}
69+
}
70+
71+
return data.mapValues { replaceValue(it.value) }
72+
}*/
73+
}

0 commit comments

Comments
 (0)