io.cequence.openaiscala.JsonFormats.scala Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of openai-scala-core_3 Show documentation
Show all versions of openai-scala-core_3 Show documentation
Core module of OpenAI Scala client
The newest version!
package io.cequence.openaiscala
import io.cequence.openaiscala.domain.AssistantTool.FunctionTool
import io.cequence.openaiscala.domain.Batch._
import io.cequence.openaiscala.domain.ChunkingStrategy.StaticChunkingStrategy
import io.cequence.openaiscala.domain.FineTune.WeightsAndBiases
import io.cequence.openaiscala.domain.ThreadAndRun.Content.ContentBlock.ImageDetail
//import io.cequence.openaiscala.domain.RunTool.{CodeInterpreterTool, FileSearchTool, FunctionTool}
import io.cequence.openaiscala.domain.Run.TruncationStrategy
import io.cequence.openaiscala.domain.ToolChoice.EnforcedTool
import io.cequence.openaiscala.domain.StepDetail.{MessageCreation, ToolCalls}
import io.cequence.openaiscala.domain.response.AssistantToolResourceResponse.{
CodeInterpreterResourcesResponse,
FileSearchResourcesResponse
}
import io.cequence.openaiscala.domain.response.ResponseFormat.{
JsonObjectResponse,
StringResponse,
TextResponse
}
import io.cequence.openaiscala.domain.response._
import io.cequence.openaiscala.domain.settings.JsonSchemaDef
import io.cequence.openaiscala.domain.{ThreadMessageFile, _}
import io.cequence.wsclient.JsonUtil
import io.cequence.wsclient.JsonUtil.{enumFormat, snakeEnumFormat}
import play.api.libs.functional.syntax._
import play.api.libs.json.Json.toJson
import play.api.libs.json.JsonNaming.SnakeCase
import play.api.libs.json.{Format, JsValue, Json, _}
import java.{util => ju}
object JsonFormats {
private implicit lazy val dateFormat: Format[ju.Date] = JsonUtil.SecDateFormat
implicit lazy val permissionFormat: Format[Permission] = Json.format[Permission]
implicit lazy val modelSpecFormat: Format[ModelInfo] = {
val reads: Reads[ModelInfo] = (
(__ \ "id").read[String] and
(__ \ "created").read[ju.Date] and
(__ \ "owned_by").read[String] and
(__ \ "root").readNullable[String] and
(__ \ "parent").readNullable[String] and
(__ \ "permission").read[Seq[Permission]].orElse(Reads.pure(Nil))
)(ModelInfo.apply _)
val writes: Writes[ModelInfo] = Json.writes[ModelInfo]
Format(reads, writes)
}
implicit lazy val usageInfoFormat: Format[UsageInfo] = Json.format[UsageInfo]
private implicit lazy val stringDoubleMapFormat: Format[Map[String, Double]] =
JsonUtil.StringDoubleMapFormat
private implicit lazy val stringStringMapFormat: Format[Map[String, String]] =
JsonUtil.StringStringMapFormat
implicit lazy val logprobsInfoFormat: Format[LogprobsInfo] =
Json.format[LogprobsInfo]
implicit lazy val textCompletionChoiceInfoFormat: Format[TextCompletionChoiceInfo] =
Json.format[TextCompletionChoiceInfo]
implicit lazy val textCompletionFormat: Format[TextCompletionResponse] =
Json.format[TextCompletionResponse]
implicit lazy val chatRoleFormat: Format[ChatRole] = enumFormat[ChatRole](
ChatRole.User,
ChatRole.System,
ChatRole.Assistant,
ChatRole.Function,
ChatRole.Tool
)
implicit lazy val contentWrites: Writes[Content] = Writes[Content] {
_ match {
case c: TextContent =>
Json.obj("type" -> "text", "text" -> c.text)
case c: ImageURLContent =>
Json.obj("type" -> "image_url", "image_url" -> Json.obj("url" -> c.url))
}
}
implicit lazy val contentReads: Reads[Content] = Reads[Content] { (json: JsValue) =>
(json \ "type").validate[String].flatMap {
case "text" => (json \ "text").validate[String].map(TextContent.apply)
case "image_url" =>
(json \ "image_url" \ "url").validate[String].map(ImageURLContent.apply)
case _ => JsError("Invalid type")
}
}
implicit val functionCallSpecFormat: Format[FunctionCallSpec] =
Json.format[FunctionCallSpec]
implicit val systemMessageFormat: Format[SystemMessage] = Json.format[SystemMessage]
implicit val userMessageFormat: Format[UserMessage] = Json.format[UserMessage]
implicit val userSeqMessageFormat: Format[UserSeqMessage] = Json.format[UserSeqMessage]
implicit val toolMessageFormat: Format[ToolMessage] = Json.format[ToolMessage]
implicit val assistantMessageFormat: Format[AssistantMessage] = Json.format[AssistantMessage]
implicit val assistantToolMessageReads: Reads[AssistantToolMessage] = (
(__ \ "content").readNullable[String] and
(__ \ "name").readNullable[String] and
(__ \ "tool_calls").readNullable[JsArray]
) {
(
content,
name,
tool_calls
) =>
val idToolCalls = tool_calls.getOrElse(JsArray()).value.toSeq.map { toolCall =>
val callId = (toolCall \ "id").as[String]
val callType = (toolCall \ "type").as[String]
val call: ToolCallSpec = callType match {
case "function" => (toolCall \ "function").as[FunctionCallSpec]
case _ => throw new Exception(s"Unknown tool call type: $callType")
}
(callId, call)
}
AssistantToolMessage(content, name, idToolCalls)
}
implicit lazy val assistantFunMessageFormat: Format[AssistantFunMessage] =
Json.format[AssistantFunMessage]
implicit lazy val funMessageFormat: Format[FunMessage] = Json.format[FunMessage]
implicit lazy val messageSpecFormat: Format[MessageSpec] = Json.format[MessageSpec]
implicit lazy val chatCompletionToolFormat: Format[FunctionTool] = {
// use just here for FunctionSpec
implicit lazy val stringAnyMapFormat: Format[Map[String, Any]] =
JsonUtil.StringAnyMapFormat
Json.format[FunctionTool]
}
// val assistantsFunctionSpecFormat: Format[FunctionTool] = {
// implicit lazy val stringAnyMapFormat: Format[Map[String, Any]] =
// JsonUtil.StringAnyMapFormat
//
// val assistantsFunctionSpecWrites: Writes[FunctionSpec] = new Writes[FunctionSpec] {
// def writes(fs: FunctionSpec): JsValue = Json.obj(
// "type" -> "function",
// "function" -> Json.obj(
// "name" -> fs.name,
// "description" -> fs.description,
// "parameters" -> fs.parameters
// )
// )
// }
//
// val assistantsFunctionSpecReads: Reads[FunctionSpec] = (
// (JsPath \ "function" \ "name").read[String] and
// (JsPath \ "function" \ "description").readNullable[String] and
// (JsPath \ "function" \ "parameters").read[Map[String, Any]]
// )(FunctionSpec.apply _)
//
// Format(assistantsFunctionSpecReads, assistantsFunctionSpecWrites)
// }
implicit lazy val messageAttachmentToolFormat: Format[MessageAttachmentTool] = {
val typeDiscriminatorKey = "type"
Format[MessageAttachmentTool](
(json: JsValue) => {
(json \ typeDiscriminatorKey).validate[String].flatMap {
case "code_interpreter" => JsSuccess(MessageAttachmentTool.CodeInterpreterSpec)
case "file_search" => JsSuccess(MessageAttachmentTool.FileSearchSpec)
case _ => JsError("Unknown type")
}
},
{ (tool: MessageAttachmentTool) =>
tool match {
case MessageAttachmentTool.CodeInterpreterSpec =>
Json.obj(typeDiscriminatorKey -> "code_interpreter")
case MessageAttachmentTool.FileSearchSpec =>
Json.obj(typeDiscriminatorKey -> "file_search")
}
}
)
}
implicit lazy val assistantToolFormat: Format[AssistantTool] = {
val typeDiscriminatorKey = "type"
implicit val mapFormat = JsonUtil.StringAnyMapFormat
lazy val assistantFunctionToolFormat: Format[AssistantTool.FunctionTool] = {
val reads = Reads[AssistantTool.FunctionTool] { json =>
for {
name <- (json \ "name").validate[String]
description <- (json \ "description").validateOpt[String]
parameters <- (json \ "parameters").validate[Map[String, Any]](mapFormat)
strict <- (json \ "strict").validateOpt[Boolean]
} yield AssistantTool.FunctionTool(name, description, parameters, strict)
}
val writes = Json.writes[AssistantTool.FunctionTool]
Format(reads, writes)
}
lazy val assistantFileSearchToolFormat: Format[AssistantTool.FileSearchTool] = {
implicit val config: JsonConfiguration = JsonConfiguration(SnakeCase)
Json.format[AssistantTool.FileSearchTool]
}
Format[AssistantTool](
(json: JsValue) => {
(json \ typeDiscriminatorKey).validate[String].flatMap {
case "code_interpreter" => JsSuccess(AssistantTool.CodeInterpreterTool)
case "file_search" =>
(json \ "file_search")
.validate[AssistantTool.FileSearchTool](assistantFileSearchToolFormat)
case "function" =>
(json \ "function")
.validate[AssistantTool.FunctionTool](assistantFunctionToolFormat)
case _ => JsError("Unknown type")
}
},
{ (tool: AssistantTool) =>
val typeField = Json.obj {
val discriminatorValue = tool match {
case AssistantTool.CodeInterpreterTool => "code_interpreter"
case AssistantTool.FileSearchTool(_) => "file_search"
case AssistantTool.FunctionTool(_, _, _, _) => "function"
}
typeDiscriminatorKey -> discriminatorValue
}
val customFields = tool match {
case AssistantTool.CodeInterpreterTool => JsObject.empty
case fileSearchTool: AssistantTool.FileSearchTool =>
JsObject(
Seq(
"file_search" -> Json
.toJson(fileSearchTool)(assistantFileSearchToolFormat)
.as[JsObject]
)
)
case functionTool: AssistantTool.FunctionTool =>
JsObject(
Seq(
"function" -> Json
.toJson(functionTool)(assistantFunctionToolFormat)
.as[JsObject]
)
)
}
typeField ++ customFields
}
)
}
implicit val messageReads: Reads[BaseMessage] = Reads { (json: JsValue) =>
val role = (json \ "role").as[ChatRole]
(json \ "name").asOpt[String]
val message: BaseMessage = role match {
case ChatRole.System => json.as[SystemMessage]
case ChatRole.User =>
json.asOpt[UserMessage] match {
case Some(userMessage) => userMessage
case None => json.as[UserSeqMessage]
}
case ChatRole.Tool => json.as[ToolMessage]
case ChatRole.Assistant =>
// if contains tool_calls, then it is AssistantToolMessage
(json \ "tool_calls").asOpt[JsArray] match {
case Some(_) => json.as[AssistantToolMessage]
case None =>
json.asOpt[AssistantMessage] match {
case Some(assistantMessage) => assistantMessage
case None => json.as[AssistantFunMessage]
}
}
case ChatRole.Function => json.as[FunMessage]
}
JsSuccess(message)
}
implicit lazy val messageWrites: Writes[BaseMessage] = Writes { (message: BaseMessage) =>
def optionalJsObject(
fieldName: String,
value: Option[JsValue]
) =
value.map(x => Json.obj(fieldName -> x)).getOrElse(Json.obj())
val role = Json.obj("role" -> toJson(message.role))
val name = optionalJsObject("name", message.nameOpt.map(JsString(_)))
val json = message match {
case m: SystemMessage => toJson(m)
case m: UserMessage => toJson(m)
case m: UserSeqMessage => Json.obj("content" -> toJson(m.content))
case m: AssistantMessage => toJson(m)
case m: AssistantToolMessage =>
val calls = m.tool_calls.map { case (callId, call) =>
call match {
case c: FunctionCallSpec =>
Json.obj("id" -> callId, "type" -> "function", "function" -> toJson(c))
}
}
optionalJsObject(
"tool_calls",
if (calls.nonEmpty) Some(JsArray(calls)) else None
) ++ optionalJsObject(
"content",
m.content.map(JsString(_))
)
case m: AssistantFunMessage => toJson(m)
case m: ToolMessage => toJson(m)
case m: FunMessage => toJson(m)
case m: MessageSpec => toJson(m)
}
json.as[JsObject] ++ role ++ name
}
implicit lazy val assistantToolOutputFormat: Format[AssistantToolOutput] =
Json.format[AssistantToolOutput]
implicit lazy val chatCompletionToolWrites: Writes[ChatCompletionTool] =
Writes[ChatCompletionTool] {
_ match {
case x: FunctionTool =>
Json.obj("type" -> "function", "function" -> Json.toJson(x))
}
}
implicit lazy val topLogprobInfoormat: Format[TopLogprobInfo] = {
val reads: Reads[TopLogprobInfo] = (
(__ \ "token").read[String] and
(__ \ "logprob").read[Double] and
(__ \ "bytes").read[Seq[Short]].orElse(Reads.pure(Nil))
)(TopLogprobInfo.apply _)
val writes: Writes[TopLogprobInfo] = Json.writes[TopLogprobInfo]
Format(reads, writes)
}
implicit val byteArrayReads: Reads[Seq[Byte]] = new Reads[Seq[Byte]] {
/**
* Parses a JSON representation of a `Seq[Byte]` into a `JsResult[Seq[Byte]]`. This method
* expects the JSON to be an array of numbers, where each number represents a valid byte
* value (between -128 and 127, inclusive). If the JSON structure is correct and all
* numbers are valid byte values, it returns a `JsSuccess` containing the sequence of
* bytes. Otherwise, it returns a `JsError` detailing the parsing issue encountered.
*
* @param json
* The `JsValue` to be parsed, expected to be a `JsArray` of `JsNumber`.
* @return
* A `JsResult[Seq[Byte]]` which is either a `JsSuccess` containing the parsed sequence
* of bytes, or a `JsError` with parsing error details.
*/
def reads(json: JsValue): JsResult[Seq[Byte]] = json match {
case JsArray(elements) =>
try {
JsSuccess(elements.map {
case JsNumber(n) if n.isValidInt => n.toIntExact.toByte
case _ => throw new RuntimeException("Invalid byte value")
}.toIndexedSeq)
} catch {
case e: Exception => JsError("Error parsing byte array: " + e.getMessage)
}
case _ => JsError("Expected JSON array for byte array")
}
}
implicit lazy val logprobInfoFormat: Format[LogprobInfo] =
Json.format[LogprobInfo]
implicit lazy val logprobsFormat: Format[Logprobs] =
Json.format[Logprobs]
implicit lazy val chatCompletionChoiceInfoFormat: Format[ChatCompletionChoiceInfo] =
Json.format[ChatCompletionChoiceInfo]
implicit lazy val chatCompletionResponseFormat: Format[ChatCompletionResponse] =
Json.format[ChatCompletionResponse]
implicit lazy val chatToolCompletionChoiceInfoReads: Reads[ChatToolCompletionChoiceInfo] =
Json.reads[ChatToolCompletionChoiceInfo]
implicit lazy val chatToolCompletionResponseReads: Reads[ChatToolCompletionResponse] =
Json.reads[ChatToolCompletionResponse]
implicit lazy val chatFunCompletionChoiceInfoFormat: Format[ChatFunCompletionChoiceInfo] =
Json.format[ChatFunCompletionChoiceInfo]
implicit lazy val chatFunCompletionResponseFormat: Format[ChatFunCompletionResponse] =
Json.format[ChatFunCompletionResponse]
implicit lazy val chatChunkMessageFormat: Format[ChunkMessageSpec] =
Json.format[ChunkMessageSpec]
implicit lazy val chatCompletionChoiceChunkInfoFormat
: Format[ChatCompletionChoiceChunkInfo] =
Json.format[ChatCompletionChoiceChunkInfo]
implicit lazy val chatCompletionChunkResponseFormat: Format[ChatCompletionChunkResponse] =
Json.format[ChatCompletionChunkResponse]
implicit lazy val textEditChoiceInfoFormat: Format[TextEditChoiceInfo] =
Json.format[TextEditChoiceInfo]
implicit lazy val textEditFormat: Format[TextEditResponse] =
Json.format[TextEditResponse]
implicit lazy val imageFormat: Format[ImageInfo] = Json.format[ImageInfo]
implicit lazy val embeddingInfoFormat: Format[EmbeddingInfo] =
Json.format[EmbeddingInfo]
implicit lazy val embeddingUsageInfoFormat: Format[EmbeddingUsageInfo] =
Json.format[EmbeddingUsageInfo]
implicit lazy val embeddingFormat: Format[EmbeddingResponse] =
Json.format[EmbeddingResponse]
implicit lazy val fileStatisticsFormat: Format[FileStatistics] = Json.format[FileStatistics]
implicit lazy val fileInfoFormat: Format[FileInfo] = Json.format[FileInfo]
implicit lazy val fineTuneEventFormat: Format[FineTuneEvent] = {
implicit lazy val stringAnyMapFormat: Format[Map[String, Any]] =
JsonUtil.StringAnyMapFormat
Json.format[FineTuneEvent]
}
implicit lazy val eitherIntStringFormat: Format[Either[Int, String]] =
JsonUtil.eitherFormat[Int, String]
implicit lazy val fineTuneMetricsFormat: Format[Metrics] = Json.format[Metrics]
implicit lazy val fineTuneCheckpointFormat: Format[FineTuneCheckpoint] =
Json.format[FineTuneCheckpoint]
implicit lazy val fineTuneHyperparamsFormat: Format[FineTuneHyperparams] =
Json.format[FineTuneHyperparams]
implicit lazy val fineTuneErrorFormat: Format[Option[FineTuneError]] =
new Format[Option[FineTuneError]] {
def reads(json: JsValue): JsResult[Option[FineTuneError]] = json match {
case JsObject(underlying) if underlying.isEmpty => JsSuccess(None)
case JsNull => JsSuccess(None)
case _ => Json.reads[FineTuneError].reads(json).map(Some(_))
}
def writes(o: Option[FineTuneError]): JsValue = o match {
case None => JsObject.empty
case Some(error) => Json.writes[FineTuneError].writes(error)
}
}
implicit lazy val fineTuneIntegrationFormat: Format[FineTune.Integration] = {
val typeDiscriminatorKey = "type"
val weightsAndBiasesType = "wandb"
implicit val weightsAndBiasesIntegrationFormat: Format[WeightsAndBiases] =
Json.format[WeightsAndBiases]
Format[FineTune.Integration](
(json: JsValue) => {
(json \ typeDiscriminatorKey).validate[String].flatMap { case `weightsAndBiasesType` =>
(json \ weightsAndBiasesType)
.validate[WeightsAndBiases](weightsAndBiasesIntegrationFormat)
}
},
{ (integration: FineTune.Integration) =>
val commonJson = Json.obj {
val discriminatorValue = integration match {
case _: WeightsAndBiases => weightsAndBiasesType
}
typeDiscriminatorKey -> discriminatorValue
}
integration match {
case integration: WeightsAndBiases =>
commonJson ++ JsObject(
Seq(
weightsAndBiasesType -> weightsAndBiasesIntegrationFormat.writes(integration)
)
)
}
}
)
}
implicit val fineTuneJobFormat: Format[FineTuneJob] = (
(__ \ "id").format[String] and
(__ \ "model").format[String] and
(__ \ "created_at").format[ju.Date] and
(__ \ "finished_at").formatNullable[ju.Date] and
(__ \ "fine_tuned_model").formatNullable[String] and
(__ \ "organization_id").format[String] and
(__ \ "status").format[String] and
(__ \ "training_file").format[String] and
(__ \ "validation_file").formatNullable[String] and
(__ \ "result_files").format[Seq[String]] and
(__ \ "trained_tokens").formatNullable[Int] and
(__ \ "error").format[Option[FineTuneError]](fineTuneErrorFormat) and
(__ \ "hyperparameters").format[FineTuneHyperparams] and
(__ \ "integrations").formatNullable[Seq[FineTune.Integration]] and
(__ \ "seed").format[Int]
)(
FineTuneJob.apply,
// somehow FineTuneJob.unapply is not working in Scala3
(x: FineTuneJob) =>
(
x.id,
x.model,
x.created_at,
x.finished_at,
x.fine_tuned_model,
x.organization_id,
x.status,
x.training_file,
x.validation_file,
x.result_files,
x.trained_tokens,
x.error,
x.hyperparameters,
x.integrations,
x.seed
)
)
implicit lazy val moderationCategoriesFormat: Format[ModerationCategories] = (
(__ \ "hate").format[Boolean] and
(__ \ "hate/threatening").format[Boolean] and
(__ \ "self-harm").format[Boolean] and
(__ \ "sexual").format[Boolean] and
(__ \ "sexual/minors").format[Boolean] and
(__ \ "violence").format[Boolean] and
(__ \ "violence/graphic").format[Boolean]
)(
ModerationCategories.apply,
// somehow ModerationCategories.unapply is not working in Scala3
(x: ModerationCategories) =>
(
x.hate,
x.hate_threatening,
x.self_harm,
x.sexual,
x.sexual_minors,
x.violence,
x.violence_graphic
)
)
// somehow ModerationCategoryScores.unapply is not working in Scala3
implicit lazy val moderationCategoryScoresFormat: Format[ModerationCategoryScores] = (
(__ \ "hate").format[Double] and
(__ \ "hate/threatening").format[Double] and
(__ \ "self-harm").format[Double] and
(__ \ "sexual").format[Double] and
(__ \ "sexual/minors").format[Double] and
(__ \ "violence").format[Double] and
(__ \ "violence/graphic").format[Double]
)(
ModerationCategoryScores.apply,
{ (x: ModerationCategoryScores) =>
(
x.hate,
x.hate_threatening,
x.self_harm,
x.sexual,
x.sexual_minors,
x.violence,
x.violence_graphic
)
}
)
implicit lazy val moderationResultFormat: Format[ModerationResult] =
Json.format[ModerationResult]
implicit lazy val moderationFormat: Format[ModerationResponse] =
Json.format[ModerationResponse]
implicit lazy val threadMessageFormat: Format[ThreadMessage] =
Json.format[ThreadMessage]
implicit lazy val assistantToolResourceResponsesFormat
: Reads[Seq[AssistantToolResourceResponse]] = Reads { json =>
val codeInterpreter = (json \ "code_interpreter" \ "file_ids")
.asOpt[Seq[FileId]]
.map(CodeInterpreterResourcesResponse.apply)
val fileSearch = (json \ "file_search" \ "vector_store_ids")
.asOpt[Seq[String]]
.map(FileSearchResourcesResponse.apply)
Seq(codeInterpreter, fileSearch).flatten match {
case Nil => JsError("Expected code_interpreter or file_search response")
case responses => JsSuccess(responses)
}
}
implicit lazy val assistantToolResourceResponsesWrites
: Writes[Seq[AssistantToolResourceResponse]] = Writes { items =>
items.map {
case c: CodeInterpreterResourcesResponse =>
Json.obj("code_interpreter" -> Json.obj("file_ids" -> c.file_ids))
case f: FileSearchResourcesResponse =>
Json.obj("file_search" -> Json.obj("vector_store_ids" -> f.vector_store_ids))
}.foldLeft(Json.obj())(_ ++ _)
}
implicit lazy val assistantToolResourceCodeInterpreterResourceWrites
: Writes[AssistantToolResource.CodeInterpreterResources] =
Writes { c =>
Json.obj("code_interpreter" -> Json.obj("file_ids" -> c.fileIds))
}
implicit lazy val assistantToolResourceFileSearchResourceWrites
: Writes[AssistantToolResource.FileSearchResources] =
Writes { f =>
assert(
f.vectorStoreIds.isEmpty || f.vectorStores.isEmpty,
"Only one of vector_store_ids or vector_stores should be provided."
)
val vectorStoreIdsJson =
if (f.vectorStoreIds.nonEmpty) Json.obj("vector_store_ids" -> f.vectorStoreIds)
else Json.obj()
val vectorStoresJson =
if (f.vectorStores.nonEmpty) Json.obj("vector_stores" -> f.vectorStores)
else Json.obj()
Json.obj("file_search" -> (vectorStoreIdsJson ++ vectorStoresJson))
}
implicit lazy val assistantToolResourceWrites: Writes[AssistantToolResource] = Writes {
case AssistantToolResource(Some(codeInterpreter), _) =>
Json.toJson(codeInterpreter)(assistantToolResourceCodeInterpreterResourceWrites)
case AssistantToolResource(_, Some(fileSearch)) =>
Json.toJson(fileSearch)(assistantToolResourceFileSearchResourceWrites)
case _ => Json.obj()
}
implicit lazy val codeInterpreterResourcesReads
: Reads[AssistantToolResource.CodeInterpreterResources] = {
implicit val config: JsonConfiguration = JsonConfiguration(JsonNaming.SnakeCase)
Json.reads[AssistantToolResource.CodeInterpreterResources]
}
implicit lazy val fileSearchResourcesReads
: Reads[AssistantToolResource.FileSearchResources] = {
implicit val config: JsonConfiguration = JsonConfiguration(JsonNaming.SnakeCase)
(
(__ \ "vector_store_ids").readNullable[Seq[String]].map(_.getOrElse(Seq.empty)) and
(__ \ "vector_stores")
.readNullable[Seq[AssistantToolResource.VectorStore]]
.map(_.getOrElse(Seq.empty))
)(AssistantToolResource.FileSearchResources.apply _)
}
implicit lazy val assistantToolResourceReads: Reads[AssistantToolResource] = (
(__ \ "code_interpreter").readNullable[AssistantToolResource.CodeInterpreterResources] and
(__ \ "file_search").readNullable[AssistantToolResource.FileSearchResources]
)(
(
codeInterpreter,
fileSearch
) => AssistantToolResource(codeInterpreter, fileSearch)
)
implicit lazy val threadAndRunCodeInterpreterResourceWrites
: Writes[ThreadAndRunToolResource.CodeInterpreterResource] = {
implicit val config: JsonConfiguration = JsonConfiguration(JsonNaming.SnakeCase)
Json.writes[ThreadAndRunToolResource.CodeInterpreterResource]
}
implicit lazy val threadAndRunCodeInterpreterResourceReads
: Reads[ThreadAndRunToolResource.CodeInterpreterResource] = {
implicit val config: JsonConfiguration = JsonConfiguration(JsonNaming.SnakeCase)
Json.reads[ThreadAndRunToolResource.CodeInterpreterResource]
}
implicit lazy val threadAndRunFileSearchResourceWrites
: Writes[ThreadAndRunToolResource.FileSearchResource] =
Json.writes[ThreadAndRunToolResource.FileSearchResource]
implicit lazy val threadAndRunFileSearchResourceReads
: Reads[ThreadAndRunToolResource.FileSearchResource] =
Json.reads[ThreadAndRunToolResource.FileSearchResource]
implicit lazy val threadAndRunToolResourceWrites: Writes[ThreadAndRunToolResource] = {
implicit val config: JsonConfiguration = JsonConfiguration(JsonNaming.SnakeCase)
Json.writes[ThreadAndRunToolResource]
}
implicit lazy val threadReads: Reads[Thread] =
(
(__ \ "id").read[String] and
(__ \ "created_at").read[ju.Date] and
(__ \ "tool_resources")
.read[Seq[AssistantToolResourceResponse]]
.orElse(Reads.pure(Nil)) and
(__ \ "metadata").read[Map[String, String]].orElse(Reads.pure(Map()))
)(Thread.apply _)
implicit val fileIdFormat: Format[FileId] = Format(
Reads.StringReads.map(FileId.apply),
Writes[FileId](fileId => JsString(fileId.file_id))
)
implicit lazy val threadMessageContentTypeFormat: Format[ThreadMessageContentType] =
enumFormat[ThreadMessageContentType](
ThreadMessageContentType.image_file,
ThreadMessageContentType.text
)
implicit lazy val fileAnnotationTypeFormat: Format[FileAnnotationType] =
enumFormat[FileAnnotationType](
FileAnnotationType.file_citation,
FileAnnotationType.file_path
)
implicit lazy val fileAnnotationFormat: Format[FileAnnotation] =
Json.format[FileAnnotation]
implicit lazy val fileCitationFormat: Format[FileCitation] =
Json.format[FileCitation]
implicit lazy val threadMessageTextFormat: Format[ThreadMessageText] =
Json.format[ThreadMessageText]
implicit lazy val threadMessageContentFormat: Format[ThreadMessageContent] =
Json.format[ThreadMessageContent]
implicit lazy val threadFullMessageReads: Reads[ThreadFullMessage] =
(
(__ \ "id").read[String] and
(__ \ "created_at").read[ju.Date] and
(__ \ "thread_id").read[String] and
(__ \ "role").read[ChatRole].orElse(Reads.pure(ChatRole.User)) and
(__ \ "content").read[Seq[ThreadMessageContent]].orElse(Reads.pure(Nil)) and
(__ \ "assistant_id").readNullable[String] and
(__ \ "run_id").readNullable[String] and
(__ \ "attachments").read[Seq[Attachment]].orElse(Reads.pure(Nil)) and
(__ \ "metadata").read[Map[String, String]].orElse(Reads.pure(Map()))
)(ThreadFullMessage.apply _)
implicit lazy val threadFullMessageWrites: Writes[ThreadFullMessage] =
Json.writes[ThreadFullMessage]
implicit lazy val threadMessageFileFormat: Format[ThreadMessageFile] =
Json.format[ThreadMessageFile]
implicit lazy val assistantToolResourceVectorStoreFormat
: Format[AssistantToolResource.VectorStore] = {
implicit val stringStringMapFormat: Format[Map[String, String]] =
JsonUtil.StringStringMapFormat
(
(__ \ "file_ids").format[Seq[FileId]] and
(__ \ "metadata").format[Map[String, String]] and
(__ \ "chunking_strategy").formatNullable[ChunkingStrategy]
)(
AssistantToolResource.VectorStore.apply,
unlift(AssistantToolResource.VectorStore.unapply)
)
}
implicit lazy val codeInterpreterResourcesResponseFormat
: Format[CodeInterpreterResourcesResponse] =
Json.format[CodeInterpreterResourcesResponse]
implicit lazy val fileSearchResourcesResponseFormat: Format[FileSearchResourcesResponse] =
Json.format[FileSearchResourcesResponse]
implicit lazy val responseFormatFormat: Format[ResponseFormat] = {
def error(json: JsValue) = JsError(
s"Expected String response, JSON Object response, or Text response format, but got $json"
)
Format(
Reads {
case JsString("auto") => JsSuccess(StringResponse)
case JsObject(fields) =>
fields
.get("type")
.map {
case JsString("json_object") => JsSuccess(JsonObjectResponse)
case JsString("text") => JsSuccess(TextResponse)
case json => error(json)
}
.getOrElse(error(JsObject(fields)))
case json => error(json)
},
Writes {
case StringResponse => JsString("auto")
case JsonObjectResponse => Json.obj("type" -> "json_object")
case TextResponse => Json.obj("type" -> "text")
}
)
}
implicit lazy val attachmentFormat: Format[Attachment] = (
(__ \ "file_id").formatNullable[FileId] and
(__ \ "tools").format[Seq[MessageAttachmentTool]]
)(Attachment.apply, unlift(Attachment.unapply))
implicit lazy val assistantReads: Reads[Assistant] = (
(__ \ "id").read[String] and
(__ \ "created_at").read[ju.Date] and
(__ \ "name").readNullable[String] and
(__ \ "description").readNullable[String] and
(__ \ "model").read[String] and
(__ \ "instructions").readNullable[String] and
(__ \ "tools").read[Seq[AssistantTool]] and
(__ \ "tool_resources")
.read[Seq[AssistantToolResourceResponse]]
.orElse(Reads.pure(Nil)) and
(__ \ "metadata").read[Map[String, String]].orElse(Reads.pure(Map())) and
(__ \ "temperature").readNullable[Double].orElse(Reads.pure(None)) and
(__ \ "top_p").readNullable[Double].orElse(Reads.pure(None)) and
(__ \ "response_format").read[ResponseFormat]
)(Assistant.apply _)
// implicit lazy val assistantWrites: Writes[Assistant] =
// Json.writes[Assistant]
implicit lazy val batchEndPointFormat: Format[BatchEndpoint] = enumFormat[BatchEndpoint](
BatchEndpoint.`/v1/chat/completions`,
BatchEndpoint.`/v1/embeddings`
)
implicit lazy val completionWindowFormat: Format[CompletionWindow] =
enumFormat[CompletionWindow](
CompletionWindow.`24h`
)
implicit lazy val batchProcessingErrorFormat: Format[BatchProcessingError] =
Json.format[BatchProcessingError]
implicit lazy val batchProcessingErrorsFormat: Format[BatchProcessingErrors] =
Json.format[BatchProcessingErrors]
implicit lazy val batchFormat: Format[Batch] = Json.format[Batch]
implicit lazy val batchInputFormat: Format[BatchRow] = Json.format[BatchRow]
implicit lazy val chatCompletionBatchResponseFormat: Format[ChatCompletionBatchResponse] =
Json.format[ChatCompletionBatchResponse]
implicit lazy val embeddingBatchResponseFormat: Format[EmbeddingBatchResponse] =
Json.format[EmbeddingBatchResponse]
implicit lazy val batchResponseFormat: Format[BatchResponse] = {
val reads: Reads[BatchResponse] = Reads { json =>
chatCompletionBatchResponseFormat
.reads(json)
.orElse(embeddingBatchResponseFormat.reads(json))
}
val writes: Writes[BatchResponse] = Writes {
case chatCompletionResponse: ChatCompletionResponse =>
chatCompletionResponseFormat.writes(chatCompletionResponse)
case embeddingResponse: EmbeddingResponse =>
embeddingFormat.writes(embeddingResponse)
}
Format(reads, writes)
}
implicit lazy val batchErrorFormat: Format[BatchError] = Json.format[BatchError]
implicit lazy val createBatchResponseFormat: Format[CreateBatchResponse] =
Json.format[CreateBatchResponse]
implicit lazy val createBatchResponsesFormat: Format[CreateBatchResponses] =
Json.format[CreateBatchResponses]
implicit lazy val fileCountsFormat: Format[FileCounts] = {
implicit val config: JsonConfiguration = JsonConfiguration(SnakeCase)
Json.format[FileCounts]
}
implicit lazy val vectorStoreFormat: Format[VectorStore] =
Json.format[VectorStore]
implicit lazy val vectorStoreFileFormat: Format[VectorStoreFile] = {
implicit val config: JsonConfiguration = JsonConfiguration(SnakeCase)
Json.format[VectorStoreFile]
}
implicit lazy val vectorStoreFileStatusFormat: Format[VectorStoreFileStatus] = {
import VectorStoreFileStatus._
enumFormat(Cancelled, Completed, InProgress, Failed)
}
implicit lazy val lastErrorFormat: Format[LastError] = Json.format[LastError]
implicit lazy val lastErrorCodeFormat: Format[LastErrorCode] = {
import LastErrorCode._
snakeEnumFormat(ServerError, RateLimitExceeded)
}
implicit lazy val chunkingStrategyAutoFormat
: Format[ChunkingStrategy.AutoChunkingStrategy.type] =
Json.format[ChunkingStrategy.AutoChunkingStrategy.type]
implicit lazy val chunkingStrategyStaticFormat
: Format[ChunkingStrategy.StaticChunkingStrategy.type] =
Json.format[ChunkingStrategy.StaticChunkingStrategy.type]
val chunkingStrategyFormatReads: Reads[ChunkingStrategy] =
(
(__ \ "max_chunk_size_tokens").readNullable[Int] and
(__ \ "chunk_overlap_tokens").readNullable[Int]
)(
(
maxChunkSizeTokens: Option[Int],
chunkOverlapTokens: Option[Int]
) => StaticChunkingStrategy(maxChunkSizeTokens, chunkOverlapTokens)
)
implicit lazy val chunkingStrategyFormat: Format[ChunkingStrategy] = {
val reads: Reads[ChunkingStrategy] = Reads { json =>
import ChunkingStrategy._
(json \ "type").validate[String].flatMap {
case "auto" => JsSuccess(AutoChunkingStrategy)
case "static" =>
(json.validate[ChunkingStrategy](chunkingStrategyFormatReads))
case "" => JsSuccess(AutoChunkingStrategy)
case _ => JsError("Unknown chunking strategy type")
}
}
val writes: Writes[ChunkingStrategy] = Writes {
case ChunkingStrategy.AutoChunkingStrategy => Json.obj("type" -> "auto")
case ChunkingStrategy.StaticChunkingStrategy(maxChunkSizeTokens, chunkOverlapTokens) =>
Json.obj(
"type" -> "static",
"max_chunk_size_tokens" -> maxChunkSizeTokens,
"chunk_overlap_tokens" -> chunkOverlapTokens
)
}
Format(reads, writes)
}
implicit lazy val runReasonFormat: Format[Run.Reason] = Json.format[Run.Reason]
implicit lazy val lastRunErrorCodeFormat: Format[Run.LastErrorCode] = {
import Run.LastErrorCode._
snakeEnumFormat(ServerError, RateLimitExceeded, InvalidPrompt)
}
implicit lazy val truncationStrategyTypeFormat: Format[Run.TruncationStrategyType] = {
import Run.TruncationStrategyType._
snakeEnumFormat(Auto, LastMessages)
}
implicit lazy val runStatusFormat: Format[RunStatus] = {
import RunStatus._
snakeEnumFormat(
Queued,
InProgress,
RequiresAction,
Cancelling,
Cancelled,
Failed,
Completed,
Incomplete,
Expired
)
}
implicit lazy val runFormat: Format[Run] = Json.format[Run]
implicit lazy val functionToolFormat: Format[RunTool.FunctionTool] =
Json.format[RunTool.FunctionTool]
implicit lazy val runToolFormat: Format[RunTool] = {
val runToolWrites: Writes[RunTool] = Writes {
case RunTool.CodeInterpreterTool => Json.obj("type" -> "code_interpreter")
case RunTool.FileSearchTool => Json.obj("type" -> "file_search")
case RunTool.FunctionTool(name) =>
Json.obj("type" -> "function", "function" -> Json.obj("name" -> name))
}
val runToolReads: Reads[RunTool] = Reads { json =>
(json \ "type").validate[String].flatMap {
case "code_interpreter" => JsSuccess(RunTool.CodeInterpreterTool)
case "file_search" => JsSuccess(RunTool.FileSearchTool)
case "function" =>
(json \ "function" \ "name").validate[String].map(RunTool.FunctionTool.apply)
case _ => JsError("Unknown type")
}
}
Format(runToolReads, runToolWrites)
}
// implicit lazy val forcableToolFormat: Format[ForcableTool] = {
// val reades: Reads[ForcableTool] = Reads { json =>
// (json \ "type").validate[String].flatMap {
// case "code_interpreter" => JsSuccess(CodeInterpreterSpec)
// case "file_search" => JsSuccess(FileSearchSpec)
// case "function" => json.validate[FunctionSpec]
// case unsupportedType =>
// JsError(s"Unsupported type of a forceable tool: $unsupportedType")
// }
// }
//
// val writes: Writes[ForcableTool] = Writes {
// case CodeInterpreterSpec => Json.obj("type" -> "code_interpreter")
// case FileSearchSpec => Json.obj("type" -> "file_search")
// case functionTool: FunctionSpec =>
// Json.toJson(functionTool).as[JsObject] + ("type" -> JsString("function"))
// }
//
// Format(reades, writes)
// }
implicit val toolChoiceFormat: Format[ToolChoice] = {
import ToolChoice._
val reads: Reads[ToolChoice] = Reads { json =>
json.validate[String].flatMap {
case "none" => JsSuccess(None)
case "auto" => JsSuccess(Auto)
case "required" => JsSuccess(Required)
case _ => runToolFormat.reads(json).map(EnforcedTool.apply)
}
}
val writes: Writes[ToolChoice] = Writes {
case None => JsString("none")
case Auto => JsString("auto")
case Required => JsString("required")
case EnforcedTool(runTool) => runToolFormat.writes(runTool)
}
Format(reads, writes)
}
implicit lazy val runResponseFormat: Format[RunResponse] = Json.format[RunResponse]
implicit lazy val toolCallFormat: Format[ToolCall] = Json.format[ToolCall]
implicit lazy val submitToolOutputsFormat: Format[SubmitToolOutputs] =
Json.format[SubmitToolOutputs]
implicit lazy val requiredActionFormat: Format[RequiredAction] = Json.format[RequiredAction]
// implicit lazy val runStepLastErrorFormat: Format[RunStep.LastError] =
// Json.format[RunStep.LastError]
//
// implicit lazy val runStepLastErrorCodeFormat: Format[RunStep.LastErrorCode] = {
// import RunStep.LastErrorCode._
// snakeEnumFormat[RunStep.LastErrorCode](ServerError, RateLimitExceeded)
// }
//
// implicit lazy val runStepLastErrorFormat: Format[RunStep.LastError] = {
// format[RunStep.LastError]
// }
implicit lazy val runStepFormat: Format[RunStep] = {
implicit val jsonConfig: JsonConfiguration = JsonConfiguration(SnakeCase)
Json.format[RunStep]
}
implicit val messageCreationReads: Reads[MessageCreation] =
(__ \ "message_creation" \ "message_id").read[String].map(MessageCreation.apply)
implicit val messageCreationWrites: Writes[MessageCreation] = Writes { messageCreation =>
Json.obj("message_creation" -> Json.obj("message_id" -> messageCreation.messageId))
}
implicit val messageCreationFormat: Format[MessageCreation] =
Format(messageCreationReads, messageCreationWrites)
implicit val toolCallsFormat: Format[ToolCalls] = Json.format[ToolCalls]
implicit val stepDetailFormat: Format[StepDetail] = {
implicit val jsonConfig: JsonConfiguration = JsonConfiguration(SnakeCase)
implicit val stepDetailReads: Reads[StepDetail] = Reads[StepDetail] { json =>
(json \ "type").as[String] match {
case "message_creation" => messageCreationFormat.reads(json)
case "tool_calls" => toolCallsFormat.reads(json)
}
}
implicit val stepDetailWrites: Writes[StepDetail] = Writes[StepDetail] {
case mc: MessageCreation =>
messageCreationFormat.writes(mc).as[JsObject] + ("type" -> JsString("MessageCreation"))
case tc: ToolCalls =>
toolCallsFormat.writes(tc).as[JsObject] + ("type" -> JsString("ToolCalls"))
}
Format(stepDetailReads, stepDetailWrites)
}
implicit lazy val truncationStrategyWrites: Writes[TruncationStrategy] =
Json.format[TruncationStrategy]
implicit lazy val threadWrites: Writes[Thread] = Json.writes[Thread]
implicit lazy val threadAndRunRoleWrites: Writes[ThreadAndRunRole] =
Writes {
case ChatRole.Assistant => JsString("assistant")
case ChatRole.User => JsString("user")
}
implicit lazy val theadAndRunImageFileDetailsWrites
: Writes[ThreadAndRun.Content.ContentBlock.ImageFileDetail] = Writes {
case ImageDetail.Low => JsString("low")
case ImageDetail.High => JsString("high")
}
implicit lazy val threadAndRunContentBlockWrites
: Writes[ThreadAndRun.Content.ContentBlock] = {
implicit val mapFormat = JsonUtil.StringAnyMapFormat
Writes {
case ThreadAndRun.Content.ContentBlock.TextBlock(text) =>
Json.toJson("type" -> "text", "text" -> text)
case ThreadAndRun.Content.ContentBlock.ImageFileBlock(fileId, detail) =>
Json.toJson(
"type" -> "image_file",
"image_file" -> Json.obj("file_id" -> fileId, "detail" -> Json.toJson(detail))
)
}
}
implicit lazy val threadAndRunContentWrites: Writes[ThreadAndRun.Content] = Writes {
case ThreadAndRun.Content.SingleString(text) => JsString(text)
case ThreadAndRun.Content.ContentBlocks(tool) => JsArray(tool.map(Json.toJson(_)))
}
implicit lazy val theadAndRunMessageWrites: Writes[ThreadAndRun.Message] = {
implicit val mapFormat = JsonUtil.StringAnyMapFormat
Writes { case message =>
Json.obj(
"content" -> Json.toJson(message.content),
"role" -> Json.toJson(message.role),
"attachments" -> Json.toJson(message.attachments),
"metadata" -> Json.toJson(message.metadata)
)
}
}
implicit lazy val threadAndRunWrites: Writes[ThreadAndRun] = {
// snake case naming strategy
implicit val config: JsonConfiguration = JsonConfiguration(SnakeCase)
implicit val mapFormat = JsonUtil.StringAnyMapFormat
Json.writes[ThreadAndRun]
}
implicit lazy val jsonTypeFormat: Format[JsonType] = enumFormat[JsonType](
JsonType.Object,
JsonType.String,
JsonType.Number,
JsonType.Boolean,
JsonType.Null,
JsonType.Array
)
implicit lazy val jsonSchemaWrites: Writes[JsonSchema] = {
implicit val stringWrites = Json.writes[JsonSchema.String]
implicit val numberWrites = Json.writes[JsonSchema.Number]
implicit val booleanWrites = Json.writes[JsonSchema.Boolean]
// implicit val nullWrites = Json.writes[JsonSchema.Null]
def writesAux(o: JsonSchema): JsValue = {
val typeValueJson = o.`type`.toString
val json: JsObject = o match {
case c: JsonSchema.String =>
val json = Json.toJson(c).as[JsObject]
if ((json \ "enum").asOpt[Seq[String]].exists(_.isEmpty)) json - "enum" else json
case c: JsonSchema.Number =>
Json.toJson(c).as[JsObject]
case c: JsonSchema.Boolean =>
Json.toJson(c).as[JsObject]
case _: JsonSchema.Null =>
Json.obj()
case c: JsonSchema.Object =>
Json.obj(
"properties" -> JsObject(
c.properties.map { case (key, value) => (key, writesAux(value)) }
),
"required" -> c.required
)
case c: JsonSchema.Array =>
Json.obj(
"items" -> writesAux(c.items)
)
}
json ++ Json.obj("type" -> typeValueJson)
}
(o: JsonSchema) => writesAux(o)
}
implicit lazy val jsonSchemaReads: Reads[JsonSchema] = new Reads[JsonSchema] {
implicit val stringReads: Reads[JsonSchema.String] = (
(__ \ "description").readNullable[String] and
(__ \ "enum").readWithDefault[Seq[String]](Nil)
)(JsonSchema.String.apply _)
implicit val numberReads: Reads[JsonSchema.Number] = Json.reads[JsonSchema.Number]
implicit val booleanReads: Reads[JsonSchema.Boolean] = Json.reads[JsonSchema.Boolean]
// implicit val nullReads = Json.reads[JsonSchema.Null]
def readsAux(o: JsValue): JsResult[JsonSchema] = {
(o \ "type")
.asOpt[JsonType]
.map {
case JsonType.String =>
Json.fromJson[JsonSchema.String](o)
case JsonType.Number =>
Json.fromJson[JsonSchema.Number](o)
case JsonType.Boolean =>
Json.fromJson[JsonSchema.Boolean](o)
case JsonType.Null =>
JsSuccess(JsonSchema.Null())
case JsonType.Object =>
(o \ "properties")
.asOpt[JsObject]
.map { propertiesJson =>
val propertiesResults = propertiesJson.fields.map { case (key, jsValue) =>
(key, readsAux(jsValue))
}.toMap
val propertiesErrors = propertiesResults.collect { case (_, JsError(errors)) =>
errors
}
val properties = propertiesResults.collect { case (key, JsSuccess(value, _)) =>
(key, value)
}
val required = (o \ "required").asOpt[Seq[String]].getOrElse(Nil)
if (propertiesErrors.isEmpty)
JsSuccess(JsonSchema.Object(properties, required))
else
JsError(propertiesErrors.reduce(_ ++ _))
}
.getOrElse(
JsError("Object schema must have a 'properties' field.")
)
case JsonType.Array =>
(o \ "items")
.asOpt[JsObject]
.map { itemsJson =>
readsAux(itemsJson).map { items =>
JsonSchema.Array(items)
}
}
.getOrElse(
JsError("Array schema must have an 'items' field.")
)
}
.getOrElse(
JsError("Schema must have a 'type' field.")
)
}
override def reads(json: JsValue): JsResult[JsonSchema] = readsAux(json)
}
implicit lazy val jsonSchemaFormat: Format[JsonSchema] =
Format(jsonSchemaReads, jsonSchemaWrites)
implicit lazy val eitherJsonSchemaReads: Reads[Either[JsonSchema, Map[String, Any]]] = {
implicit val stringAnyMapFormat: Format[Map[String, Any]] =
JsonUtil.StringAnyMapFormat
Reads[Either[JsonSchema, Map[String, Any]]] { (json: JsValue) =>
json
.validate[JsonSchema]
.map(Left(_))
.orElse(
json.validate[Map[String, Any]].map(Right(_))
)
}
}
implicit lazy val eitherJsonSchemaWrites: Writes[Either[JsonSchema, Map[String, Any]]] = {
implicit val stringAnyMapFormat: Format[Map[String, Any]] =
JsonUtil.StringAnyMapFormat
Writes[Either[JsonSchema, Map[String, Any]]] {
case Left(schema) => Json.toJson(schema)
case Right(map) => Json.toJson(map)
}
}
implicit lazy val eitherJsonSchemaFormat: Format[Either[JsonSchema, Map[String, Any]]] =
Format(eitherJsonSchemaReads, eitherJsonSchemaWrites)
implicit val jsonSchemaDefFormat: Format[JsonSchemaDef] = Json.format[JsonSchemaDef]
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy