Skip to content

A set of utilities to help with polymorphic serialization and documentation.

License

Notifications You must be signed in to change notification settings

syoung-smallwisdom/JsonModel-Swift

 
 

Repository files navigation

JsonModel

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.

Version 2

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.

Version 2.1

  • Added property wrappers that can be used in polymorphic serialization.
  • Deprecated PolymorphicSerializer and replaced with GenericPolymorphicSerializer

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)

Version 2.2

Added support for additional Json Schema Draft 7 generated documentation.

Version 2.3

Moved from Sage-Bionetworks to BridgeDigitalHealth as the owning organization.

Version 2.4

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
  • @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

Version 1

Version 1.1

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.

Version 1.6

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"),

License

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.

About

A set of utilities to help with polymorphic serialization and documentation.

Resources

License

Stars

Watchers

Forks

Packages

No packages published

Languages

  • Swift 100.0%