![JAR search and dependency download from the Maven repository](/logo.png)
run.qontract.core.pattern.DictionaryPattern.kt Maven / Gradle / Ivy
package run.qontract.core.pattern
import run.qontract.core.Resolver
import run.qontract.core.Result
import run.qontract.core.mismatchResult
import run.qontract.core.value.JSONArrayValue
import run.qontract.core.value.JSONObjectValue
import run.qontract.core.value.StringValue
import run.qontract.core.value.Value
data class DictionaryPattern(val keyPattern: Pattern, val valuePattern: Pattern, override val typeAlias: String? = null) : Pattern {
override fun matches(sampleData: Value?, resolver: Resolver): Result {
if(sampleData !is JSONObjectValue)
return mismatchResult("JSON object", sampleData)
sampleData.jsonObject.forEach { (key, value) ->
try {
val parsedKey = keyPattern.parse(key, resolver)
when (val result = resolver.matchesPattern(null, keyPattern, parsedKey)) {
is Result.Failure -> return result.breadCrumb(key)
}
} catch(e: ContractException) {
return e.failure().breadCrumb(key)
}
try {
val parsedValue = when (value) {
is StringValue -> valuePattern.parse(value.string, resolver)
else -> value
}
when (val result = resolver.matchesPattern(null, valuePattern, parsedValue)) {
is Result.Failure -> return result.breadCrumb(key)
}
} catch(e: ContractException) {
return e.failure().breadCrumb(key)
}
}
return Result.Success()
}
override fun generate(resolver: Resolver): Value {
return JSONObjectValue((1..randomNumber(10)).fold(emptyMap()) { obj, _ ->
val key = keyPattern.generate(resolver).toStringValue()
val value = valuePattern.generate(resolver)
obj.plus(key to value)
})
}
override fun newBasedOn(row: Row, resolver: Resolver): List {
val newValuePatterns = valuePattern.newBasedOn(Row(), resolver)
return newValuePatterns.map {
DictionaryPattern(keyPattern, it)
}
}
override fun parse(value: String, resolver: Resolver): Value = parsedJSON(value)
override fun encompasses(otherPattern: Pattern, thisResolver: Resolver, otherResolver: Resolver, typeStack: TypeStack): Result =
when (otherPattern) {
is ExactValuePattern -> otherPattern.fitsWithin(listOf(this), otherResolver, thisResolver, typeStack)
!is DictionaryPattern -> Result.Failure("Expected dictionary type, got ${otherPattern.typeName}")
else -> {
listOf(
{ biggerEncompassesSmaller(this.keyPattern, otherPattern.keyPattern, thisResolver, otherResolver, typeStack) },
{ biggerEncompassesSmaller(this.valuePattern, otherPattern.valuePattern, thisResolver, otherResolver, typeStack) }
).asSequence().map { it.invoke() }.firstOrNull { it is Result.Failure } ?: Result.Success()
}
}
override fun listOf(valueList: List, resolver: Resolver): Value {
return JSONArrayValue(valueList)
}
override val typeName: String = "object with key type ${keyPattern.typeName} and value type ${valuePattern.typeName}"
override val pattern: Any = "(${withoutPatternDelimiters(keyPattern.pattern.toString())}:${withoutPatternDelimiters(valuePattern.pattern.toString())})"
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy