JsonModel is a set of utilities built on top of the Swift Codable protocol to allow for polymorphic serialization and documentation using a subset of JSON Schema draft 7.
See the unit tests and the ResultModel
target for examples for how to use this
library to support polymorphic serialization.
Moved the results protocols and objects into a separate target within the JsonModel
library. To migrate to this version, you will need to import ResultModel
anywhere
that you reference ResultData
model objects.
- Added property wrappers that can be used in polymorphic serialization.
- Deprecated
PolymorphicSerializer
and replaced withGenericPolymorphicSerializer
Note: Polymorphic encoding using the static typeName
defined by the PolymorphicStaticTyped
protocol is not currently supported for encoding root objects, and is therefore not
used by any of the SerializableResultData
model objects defined within this library.
A root object can be encoded and decoded using the PolymorphicValue
as a wrapper or
by defining the typeName
as a read/write instance property.
For example,
public protocol GooProtocol {
var value: Int { get }
}
public struct FooObject : Codable, PolymorphicStaticTyped, GooProtocol {
public static var typeName: String { "foo" }
public let value: Int
public init(value: Int = 0) {
self.value = value
}
}
public struct MooObject : Codable, PolymorphicTyped, GooProtocol {
private enum CodingKeys : String, CodingKey {
case typeName = "type", goos
}
public private(set) var typeName: String = "moo"
public var value: Int {
goos.count
}
@PolymorphicArray public var goos: [GooProtocol]
public init(goos: [GooProtocol] = []) {
self.goos = goos
}
}
public struct RaguObject : Codable, PolymorphicStaticTyped, GooProtocol {
public static let typeName: String = "ragu"
public let value: Int
@PolymorphicValue public private(set) var goo: GooProtocol
public init(value: Int, goo: GooProtocol) {
self.value = value
self.goo = goo
}
}
open class GooFactory : SerializationFactory {
public let gooSerializer = GenericPolymorphicSerializer<GooProtocol>([
MooObject(),
FooObject(),
])
public required init() {
super.init()
self.registerSerializer(gooSerializer)
gooSerializer.add(typeOf: RaguObject.self)
}
}
In this example, MooObject
can be directly serialized because the typeName
is a read/write
instance property. Decoding can be handled like this:
let factory = GooFactory()
let decoder = factory.createJSONDecoder()
let json = """
{
"type" : "moo",
"goos" : [
{ "type" : "foo", "value" : 2 },
{ "type" : "moo", "goos" : [{ "type" : "foo", "value" : 5 }] }
]
}
""".data(using: .utf8)!
let decodedObject = try decoder.decode(MooObject.self, from: json)
And because the root object does not use a static typeName
, can be encoded as follows:
let encoder = JSONEncoder()
let encodedData = try encoder.encode(decodedObject)
Whereas RaguObject
must be wrapped:
let factory = GooFactory()
let decoder = factory.createJSONDecoder()
let encoder = factory.createJSONEncoder()
let json = """
{
"type" : "ragu",
"value" : 7,
"goo" : { "type" : "foo", "value" : 2 }
}
""".data(using: .utf8)!
let decodedObject = try decoder.decode(PolymorphicValue<GooProtocol>.self, from: json)
let encodedData = try encoder.encode(decodedObject)
Added support for additional Json Schema Draft 7 generated documentation.
Moved from Sage-Bionetworks to BridgeDigitalHealth as the owning organization.
Added @Serializable
macro for defining the boilerplate code used to support
polymorphic serialization using the GenericPolymorphicSerializer
and the
SerializationFactory
. This uses the same naming, where possible, as Kotlin
serialization in an effort to improve readability when moving between Kotlin
and Swift.
@Serializable
: member macro- declares a struct or class to be
Codable
- adds CodingKeys
- implements init and encode if needed
- declares a struct or class to be
@SerialName(_ name: String)
: peer macro- when defined on a property, adds custom CodingKey
- when defined on a struct or final class, adds
typeName
property
@Transient
: peer macro- does not encode property
@Polymorphic
: peer macro- encode/decode properties using the serialization factory
Moved definitions for the ResultData
protocol used by Sage Bionetworks
into this library to simplify the dependency chain for the libraries and
frameworks used by our organization.
Added ResultModel library with a placeholder file so that libraries that depend
on the ResultData
protocol can support both a version of the library where the
actual model is defined using import JsonModel
and version
2 where the model for results is defined using import ResultModel
.
.package(name: "JsonModel",
url: "https://github.com/BridgeDigitalHealth/JsonModel-Swift.git",
"1.6.0"..<"3.0.0"),
JsonModel is available under the BSD license:
Copyright (c) 2017-2023, Sage Bionetworks All rights reserved.
Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met:
- Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer.
- Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution.
- Neither the name of Sage Bionetworks nor the names of any contributors may be used to endorse or promote products derived from this software without specific prior written permission.
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL SAGE BIONETWORKS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.