com.github.erosb.jsonsKema.CompositeSchemaBuilder.kt Maven / Gradle / Ivy
package com.github.erosb.jsonsKema
import java.net.URI
typealias SchemaSupplier = (JsonPointer) -> Schema
internal fun callingSourceLocation(pointer: JsonPointer): SourceLocation {
val elem =
Thread.currentThread().stackTrace.find {
it.methodName != "getStackTrace" &&
!it.className.startsWith("com.github.erosb.jsonsKema.")
}
return elem?.let {
SourceLocation(
pointer = pointer,
documentSource = URI("classpath://" + elem.className),
lineNumber = elem.lineNumber,
position = 0,
)
} ?: UnknownSource
}
abstract class SchemaBuilder {
companion object {
private fun type(typeValue: String): SchemaSupplier =
{ ptr ->
TypeSchema(
JsonString(typeValue, callingSourceLocation(ptr + "type")),
callingSourceLocation(ptr + "type"),
)
}
@JvmStatic
fun typeString(): CompositeSchemaBuilder = empty().type("string")
@JvmStatic
fun typeObject(): CompositeSchemaBuilder = empty().type("object")
@JvmStatic
fun typeArray(): CompositeSchemaBuilder = empty().type("array")
@JvmStatic
fun typeNumber(): CompositeSchemaBuilder = empty().type("number")
@JvmStatic
fun typeInteger(): CompositeSchemaBuilder = empty().type("integer")
@JvmStatic
fun typeBoolean(): CompositeSchemaBuilder = empty().type("boolean")
@JvmStatic
fun typeNull(): CompositeSchemaBuilder = empty().type("null")
@JvmStatic
fun empty(): CompositeSchemaBuilder = CompositeSchemaBuilder()
@JvmStatic
fun falseSchema(): SchemaBuilder = FalseSchemaBuilder()
@JvmStatic
fun trueSchema(): SchemaBuilder = TrueSchemaBuilder()
@JvmStatic
fun ifSchema(ifSchema: SchemaBuilder) = empty().ifSchema(ifSchema)
@JvmStatic
fun allOf(subschemas: List): CompositeSchemaBuilder = empty().allOf(subschemas)
@JvmStatic
fun allOf(vararg subschemas: SchemaBuilder): CompositeSchemaBuilder = empty().allOf(*subschemas)
@JvmStatic
fun oneOf(subschemas: List): CompositeSchemaBuilder = empty().oneOf(subschemas)
@JvmStatic
fun oneOf(vararg subschemas: SchemaBuilder): CompositeSchemaBuilder = empty().oneOf(*subschemas)
@JvmStatic
fun anyOf(subschemas: List): CompositeSchemaBuilder = empty().anyOf(subschemas)
@JvmStatic
fun anyOf(vararg subschemas: SchemaBuilder): CompositeSchemaBuilder = empty().anyOf(*subschemas)
@JvmStatic
fun not(notSchema: SchemaBuilder): CompositeSchemaBuilder = empty().not(notSchema)
@JvmStatic
fun constSchema(constant: IJsonValue): CompositeSchemaBuilder = empty().constSchema(constant)
@JvmStatic
fun enumSchema(vararg enumValues: IJsonValue) = enumSchema(enumValues.toList())
@JvmStatic
fun enumSchema(enumValues: List): CompositeSchemaBuilder = empty().enumSchema(enumValues)
}
protected var ptr: JsonPointer = JsonPointer()
open fun buildAt(parentPointer: JsonPointer): Schema {
val origPtr = this.ptr
this.ptr = parentPointer
try {
return build()
} finally {
this.ptr = origPtr
}
}
abstract fun build(): Schema
fun buildAt(loc: SourceLocation) = buildAt(loc.pointer)
}
class FalseSchemaBuilder(
private val origLocation: SourceLocation = callingSourceLocation(JsonPointer()),
) : SchemaBuilder() {
override fun build(): Schema = FalseSchema(origLocation.withPointer(ptr + Keyword.FALSE.value))
}
class TrueSchemaBuilder(
private val origLocation: SourceLocation = callingSourceLocation(JsonPointer()),
) : SchemaBuilder() {
override fun build(): Schema = TrueSchema(origLocation.withPointer(ptr + Keyword.TRUE.value))
}
class CompositeSchemaBuilder internal constructor() : SchemaBuilder() {
private val subschemas: MutableMap = mutableMapOf()// subschemas.toMutableList()
private val propertySchemas = mutableMapOf()
private val patternPropertySchemas = mutableMapOf()
private var unevaluatedPropertiesSchema: SchemaSupplier? = null
private var unevaluatedItemsSchema: SchemaSupplier? = null
private var ifSchema: SchemaSupplier? = null
private var thenSchema: SchemaSupplier? = null
private var elseSchema: SchemaSupplier? = null
private val regexFactory = JavaUtilRegexpFactory()
private var prefixSchemasCount: Int = 0
fun minLength(minLength: Int) =
appendSupplier(Keyword.MIN_LENGTH) { loc ->
MinLengthSchema(minLength, loc)
}
fun maxLength(maxLength: Int) =
appendSupplier(Keyword.MAX_LENGTH) { loc ->
MaxLengthSchema(maxLength, loc)
}
override fun build(): Schema {
val ifSchema = this.ifSchema
if (ifSchema != null) {
subschemas[Keyword.IF] = { loc ->
IfThenElseSchema(
ifSchema(ptr),
thenSchema?.invoke(loc),
elseSchema?.invoke(loc),
callingSourceLocation(loc),
)
}
}
return CompositeSchema(
subschemas =
subschemas.values
.map { it(ptr) }
.toSet(),
location = callingSourceLocation(ptr),
propertySchemas = propertySchemas.mapValues { it.value(ptr) },
patternPropertySchemas =
patternPropertySchemas
.map {
regexFactory.createHandler(it.key) to it.value(ptr)
}.toMap(),
unevaluatedPropertiesSchema = unevaluatedPropertiesSchema?.invoke(ptr),
unevaluatedItemsSchema = unevaluatedItemsSchema?.invoke(ptr),
)
}
fun property(
propertyName: String,
schema: CompositeSchemaBuilder,
): CompositeSchemaBuilder {
propertySchemas[propertyName] = { ptr ->
schema.buildAt(ptr + Keyword.PROPERTIES.value + propertyName)
}
return this
}
fun minItems(minItems: Int): CompositeSchemaBuilder = appendSupplier(Keyword.MIN_ITEMS) { loc -> MinItemsSchema(minItems, loc) }
fun maxItems(maxItems: Int): CompositeSchemaBuilder = appendSupplier(Keyword.MAX_ITEMS) { loc -> MaxItemsSchema(maxItems, loc) }
private fun createSupplier(
kw: Keyword,
baseSchemaFn: (SourceLocation) -> Schema,
): SchemaSupplier {
val callingLocation = callingSourceLocation(JsonPointer())
return { ptr ->
baseSchemaFn(callingLocation.withPointer(ptr + kw.value))
}
}
private fun appendSupplier(
kw: Keyword,
baseSchemaFn: (SourceLocation) -> Schema,
): CompositeSchemaBuilder {
subschemas[kw] = createSupplier(kw, baseSchemaFn)
return this
}
fun items(itemsSchema: SchemaBuilder): CompositeSchemaBuilder =
appendSupplier(Keyword.ITEMS) { loc ->
ItemsSchema(itemsSchema.buildAt(loc), prefixSchemasCount, loc)
}
fun contains(containedSchema: SchemaBuilder) =
appendSupplier(Keyword.CONTAINS) { loc ->
ContainsSchema(containedSchema.buildAt(loc), 1, null, loc)
}
fun minContains(
minContains: Int,
containedSchema: SchemaBuilder,
) = appendSupplier(Keyword.MIN_CONTAINS) { loc ->
ContainsSchema(containedSchema.buildAt(loc), minContains, null, loc)
}
fun maxContains(
maxContains: Int,
containedSchema: SchemaBuilder,
) = appendSupplier(Keyword.MAX_CONTAINS) { loc ->
ContainsSchema(containedSchema.buildAt(loc), 0, maxContains, loc)
}
fun uniqueItems() = appendSupplier(Keyword.UNIQUE_ITEMS) { loc -> UniqueItemsSchema(true, loc) }
fun minProperties(minProperties: Int) = appendSupplier(Keyword.MIN_PROPERTIES) { loc -> MinPropertiesSchema(minProperties, loc) }
fun maxProperties(maxProperties: Int) = appendSupplier(Keyword.MAX_PROPERTIES) { loc -> MaxPropertiesSchema(maxProperties, loc) }
fun propertyNames(propertyNameSchema: SchemaBuilder) =
appendSupplier(Keyword.PROPERTY_NAMES) { loc ->
PropertyNamesSchema(
propertyNameSchema.buildAt(loc),
loc,
)
}
fun required(vararg requiredProperties: String) =
appendSupplier(Keyword.REQUIRED) { loc -> RequiredSchema(requiredProperties.toList(), loc) }
fun dependentRequired(dependentRequired: Map>) =
appendSupplier(Keyword.DEPENDENT_REQUIRED) { loc ->
DependentRequiredSchema(dependentRequired, loc)
}
fun readOnly(readOnly: Boolean) = if (readOnly) appendSupplier(Keyword.READ_ONLY) { loc -> ReadOnlySchema(loc) } else this
fun writeOnly(writeOnly: Boolean) = if (writeOnly) appendSupplier(Keyword.WRITE_ONLY) { loc -> WriteOnlySchema(loc) } else this
fun pattern(regexp: String) =
appendSupplier(Keyword.PATTERN) { loc ->
PatternSchema(regexFactory.createHandler(regexp), loc)
}
fun patternProperties(patternProps: Map): CompositeSchemaBuilder {
patternProps.forEach { (pattern, builder) ->
patternPropertySchemas[pattern] = { loc -> builder.buildAt(loc) }
}
return this
}
fun unevaluatedProperties(schema: SchemaBuilder): CompositeSchemaBuilder {
unevaluatedPropertiesSchema =
createSupplier(Keyword.UNEVALUATED_PROPERTIES) { loc ->
UnevaluatedPropertiesSchema(schema.buildAt(loc), loc)
}
return this
}
fun unevaluatedItems(schema: SchemaBuilder): CompositeSchemaBuilder {
unevaluatedItemsSchema =
createSupplier(Keyword.UNEVALUATED_ITEMS) { loc ->
UnevaluatedItemsSchema(schema.buildAt(loc), loc)
}
return this
}
fun ifSchema(ifSchema: SchemaBuilder): CompositeSchemaBuilder {
this.ifSchema = createSupplier(Keyword.IF) { loc -> ifSchema.buildAt(loc) }
return this
}
fun thenSchema(thenSchema: SchemaBuilder): CompositeSchemaBuilder {
this.thenSchema = createSupplier(Keyword.THEN) { loc -> thenSchema.buildAt(loc) }
return this
}
fun elseSchema(elseSchema: SchemaBuilder): CompositeSchemaBuilder {
this.elseSchema = createSupplier(Keyword.ELSE) { loc -> elseSchema.buildAt(loc) }
return this
}
fun minimum(minimum: Number) = appendSupplier(Keyword.MINIMUM) { loc -> MinimumSchema(minimum, loc) }
fun maximum(minimum: Number) = appendSupplier(Keyword.MAXIMUM) { loc -> MaximumSchema(minimum, loc) }
fun allOf(subschemas: List) =
appendSupplier(Keyword.ALL_OF) { loc ->
AllOfSchema(subschemas.map { it.buildAt(loc) }, loc)
}
fun allOf(vararg subschemas: SchemaBuilder) = allOf(subschemas.toList())
fun oneOf(subschemas: List) =
appendSupplier(Keyword.ONE_OF) { loc ->
OneOfSchema(subschemas.map { it.buildAt(loc) }, loc)
}
fun oneOf(vararg subschemas: SchemaBuilder) = oneOf(subschemas.toList())
fun anyOf(subschemas: List) =
appendSupplier(Keyword.ANY_OF) { loc ->
AnyOfSchema(subschemas.map { it.buildAt(loc) }, loc)
}
fun anyOf(vararg subschemas: SchemaBuilder) = anyOf(subschemas.toList())
fun not(negatedSchema: SchemaBuilder) =
appendSupplier(Keyword.NOT) { loc ->
NotSchema(negatedSchema.buildAt(loc), loc)
}
fun constSchema(constant: IJsonValue) =
appendSupplier(Keyword.CONST) { loc ->
ConstSchema(constant, loc)
}
fun enumSchema(enumValues: List) =
appendSupplier(Keyword.ENUM) { loc ->
EnumSchema(enumValues, loc)
}
fun exclusiveMinimum(exclMinimum: Int) =
appendSupplier(Keyword.EXCLUSIVE_MINIMUM) { loc ->
ExclusiveMinimumSchema(exclMinimum, loc)
}
fun exclusiveMaximum(exclMaximum: Int) =
appendSupplier(Keyword.EXCLUSIVE_MAXIMUM) { loc ->
ExclusiveMaximumSchema(exclMaximum, loc)
}
fun multipleOf(denominator: Int) =
appendSupplier(Keyword.MULTIPLE_OF) { loc ->
MultipleOfSchema(denominator, loc)
}
fun format(formatName: String) = appendSupplier(Keyword.FORMAT) { loc -> FormatSchema(formatName, loc) }
fun dependentSchemas(dependentSchemas: Map) =
appendSupplier(Keyword.DEPENDENT_SCHEMAS) { loc ->
DependentSchemasSchema(dependentSchemas.mapValues { it.value.buildAt(loc) }, loc)
}
fun additionalProperties(additionalPropertiesSchema: SchemaBuilder) =
appendSupplier(Keyword.ADDITIONAL_PROPERTIES) { loc ->
AdditionalPropertiesSchema(
additionalPropertiesSchema.buildAt(loc),
propertySchemas.keys.toList(),
patternPropertySchemas.keys.toList().map {
regexFactory.createHandler(it)
},
loc,
)
}
fun prefixItems(prefixSchemas: List): CompositeSchemaBuilder {
prefixSchemasCount = prefixSchemas.size
return appendSupplier(Keyword.PREFIX_ITEMS) { loc ->
PrefixItemsSchema(prefixSchemas.map { it.buildAt(loc) }, loc)
}
}
fun type(type: String) = appendSupplier(Keyword.TYPE) { loc -> TypeSchema(JsonString(type), loc)}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy