diff --git a/CHANGELOG.md b/CHANGELOG.md index 82045a5d..e477d0a7 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -2,6 +2,32 @@ ##### Breaking +* None. + +##### Enhancements + +* Yams is able to encode and decode Anchors via YamlAnchorProviding, and + YamlAnchorCoding. + [Adora Lynch](https://github.com/lynchsft) + [#125](https://github.com/jpsim/Yams/issues/125) + +* Yams is able to encode and decode Tags via YamlTagProviding + and YamlTagCoding. + [Adora Lynch](https://github.com/lynchsft) + [#265](https://github.com/jpsim/Yams/issues/265) + +* Yams is able to detect redundant structes and automaticaly + alias them during encoding via RedundancyAliasingStrategy + [Adora Lynch](https://github.com/lynchsft) + +##### Bug Fixes + +* None. + +## 5.2.0 + +##### Breaking + * Swift 5.7 or later is now required to build Yams. [JP Simard](https://github.com/jpsim) diff --git a/Sources/Yams/Anchor.swift b/Sources/Yams/Anchor.swift new file mode 100644 index 00000000..6e63d256 --- /dev/null +++ b/Sources/Yams/Anchor.swift @@ -0,0 +1,36 @@ +// +// Anchor.swift +// Yams +// +// Created by Adora Lynch on 8/9/24. +// Copyright (c) 2024 Yams. All rights reserved. + +import Foundation + +public final class Anchor: RawRepresentable, ExpressibleByStringLiteral, Codable, Hashable { + + public static let permittedCharacters = CharacterSet.lowercaseLetters + .union(.uppercaseLetters) + .union(.decimalDigits) + .union(.init(charactersIn: "-_")) + + public static func is_cyamlAlpha(_ string: String) -> Bool { + Anchor.permittedCharacters.isSuperset(of: .init(charactersIn: string)) + } + + + public let rawValue: String + + public init(rawValue: String) { + self.rawValue = rawValue + } + + public init(stringLiteral value: String) { + rawValue = value + } +} + +extension Anchor: CustomStringConvertible { + public var description: String { rawValue } +} + diff --git a/Sources/Yams/Constructor.swift b/Sources/Yams/Constructor.swift index 4120202c..e93ed2c5 100644 --- a/Sources/Yams/Constructor.swift +++ b/Sources/Yams/Constructor.swift @@ -54,6 +54,8 @@ public final class Constructor { return result } return [Any].construct_seq(from: sequence) + case .alias(_): + preconditionFailure("Aliases should be resolved before construction") } } diff --git a/Sources/Yams/Decoder.swift b/Sources/Yams/Decoder.swift index ae0b7dba..ffd294b8 100644 --- a/Sources/Yams/Decoder.swift +++ b/Sources/Yams/Decoder.swift @@ -48,8 +48,17 @@ public class YAMLDecoder { from yaml: String, userInfo: [CodingUserInfoKey: Any] = [:]) throws -> T where T: Swift.Decodable { do { - let node = try Parser(yaml: yaml, resolver: Resolver([.merge]), encoding: encoding).singleRoot() ?? "" - return try self.decode(type, from: node, userInfo: userInfo) + let parser = try Parser(yaml: yaml, resolver: Resolver([.merge]), encoding: encoding) + // ^ the parser holds the references to Anchors while parsing, + return try withExtendedLifetime(parser) { + //^ so we hold an explicit reference to the parser during decoding + let node = try parser.singleRoot() ?? "" + // ^ nodes only have weak references to Anchors (the Anchors would disappear if not held by the parser) + return try self.decode(type, from: node, userInfo: userInfo) + // ^ if the decoded type or contained types are YamlAnchorCoding, + // those types have taken ownership of Anchors. + // Otherwise the Anchors are deallocated when this function exits just like Tag and Mark + } } catch let error as DecodingError { throw error } catch { @@ -129,6 +138,8 @@ private struct _Decoder: Decoder { throw _typeMismatch(at: codingPath, expectation: Node.Scalar.self, reality: mapping) case .sequence(let sequence): throw _typeMismatch(at: codingPath, expectation: Node.Scalar.self, reality: sequence) + case .alias(let alias): + throw _typeMismatch(at: codingPath, expectation: Node.Scalar.self, reality: alias) } } } @@ -140,7 +151,41 @@ private struct _KeyedDecodingContainer: KeyedDecodingContainerPr init(decoder: _Decoder, wrapping mapping: Node.Mapping) { self.decoder = decoder - self.mapping = mapping + + let keys = mapping.keys + + let decodeAnchor: Anchor? + let decodeTag: Tag? + + if let anchor = mapping.anchor, keys.contains(.anchorKeyNode) == false { + decodeAnchor = anchor + } else { + decodeAnchor = nil + } + + if mapping.tag.name != .implicit && keys.contains(.tagKeyNode) == false { + decodeTag = mapping.tag + } else { + decodeTag = nil + } + + switch (decodeAnchor, decodeTag) { + case (nil, nil): + self.mapping = mapping + case (let anchor?, nil): + var mutableMapping = mapping + mutableMapping[.anchorKeyNode] = .scalar(.init(anchor.rawValue)) + self.mapping = mutableMapping + case (nil, let tag?): + var mutableMapping = mapping + mutableMapping[.tagKeyNode] = .scalar(.init(tag.name.rawValue)) + self.mapping = mutableMapping + case let (anchor?, tag?): + var mutableMapping = mapping + mutableMapping[.anchorKeyNode] = .scalar(.init(anchor.rawValue)) + mutableMapping[.tagKeyNode] = .scalar(.init(tag.name.rawValue)) + self.mapping = mutableMapping + } } // MARK: - Swift.KeyedDecodingContainerProtocol Methods diff --git a/Sources/Yams/Emitter.swift b/Sources/Yams/Emitter.swift index 6eeee9aa..565e1162 100644 --- a/Sources/Yams/Emitter.swift +++ b/Sources/Yams/Emitter.swift @@ -99,7 +99,8 @@ public func dump( sortKeys: Bool = false, sequenceStyle: Node.Sequence.Style = .any, mappingStyle: Node.Mapping.Style = .any, - newLineScalarStyle: Node.Scalar.Style = .any) throws -> String { + newLineScalarStyle: Node.Scalar.Style = .any, + redundancyAliasingStrategy: RedundancyAliasingStrategy? = nil) throws -> String { return try serialize( node: object.represented(), canonical: canonical, @@ -113,7 +114,8 @@ public func dump( sortKeys: sortKeys, sequenceStyle: sequenceStyle, mappingStyle: mappingStyle, - newLineScalarStyle: newLineScalarStyle + newLineScalarStyle: newLineScalarStyle, + redundancyAliasingStrategy: redundancyAliasingStrategy ) } @@ -148,7 +150,8 @@ public func serialize( sortKeys: Bool = false, sequenceStyle: Node.Sequence.Style = .any, mappingStyle: Node.Mapping.Style = .any, - newLineScalarStyle: Node.Scalar.Style = .any) throws -> String + newLineScalarStyle: Node.Scalar.Style = .any, + redundancyAliasingStrategy: RedundancyAliasingStrategy? = nil) throws -> String where Nodes: Sequence, Nodes.Iterator.Element == Node { let emitter = Emitter( canonical: canonical, @@ -162,7 +165,8 @@ public func serialize( sortKeys: sortKeys, sequenceStyle: sequenceStyle, mappingStyle: mappingStyle, - newLineScalarStyle: newLineScalarStyle + newLineScalarStyle: newLineScalarStyle, + redundancyAliasingStrategy: redundancyAliasingStrategy ) try emitter.open() try nodes.forEach(emitter.serialize) @@ -201,7 +205,8 @@ public func serialize( sortKeys: Bool = false, sequenceStyle: Node.Sequence.Style = .any, mappingStyle: Node.Mapping.Style = .any, - newLineScalarStyle: Node.Scalar.Style = .any) throws -> String { + newLineScalarStyle: Node.Scalar.Style = .any, + redundancyAliasingStrategy: RedundancyAliasingStrategy? = nil) throws -> String { return try serialize( nodes: [node], canonical: canonical, @@ -215,7 +220,8 @@ public func serialize( sortKeys: sortKeys, sequenceStyle: sequenceStyle, mappingStyle: mappingStyle, - newLineScalarStyle: newLineScalarStyle + newLineScalarStyle: newLineScalarStyle, + redundancyAliasingStrategy: redundancyAliasingStrategy ) } @@ -246,10 +252,10 @@ public final class Emitter { public var allowUnicode: Bool = false /// Set the preferred line break. public var lineBreak: LineBreak = .ln - - // internal since we don't know if these should be exposed. - var explicitStart: Bool = false - var explicitEnd: Bool = false + /// Set to emit an explicit document start marker. + public var explicitStart: Bool = false + /// Set to emit an explicit document end marker. + public var explicitEnd: Bool = false /// The `%YAML` directive value or nil. public var version: (major: Int, minor: Int)? @@ -265,6 +271,49 @@ public final class Emitter { /// Set the style for scalars that include newlines public var newLineScalarStyle: Node.Scalar.Style = .any + + /// Redundancy aliasing strategy to use when encoding. Defaults to nil + public var redundancyAliasingStrategy: RedundancyAliasingStrategy? + + /// Create `Emitter.Options` with the specified values. + /// + /// - parameter canonical: Set if the output should be in the "canonical" format described in the YAML + /// specification. + /// - parameter indent: Set the indentation value. + /// - parameter width: Set the preferred line width. -1 means unlimited. + /// - parameter allowUnicode: Set if unescaped non-ASCII characters are allowed. + /// - parameter lineBreak: Set the preferred line break. + /// - parameter explicitStart: Explicit document start `---`. + /// - parameter explicitEnd: Explicit document end `...`. + /// - parameter version: The `%YAML` directive value or nil. + /// - parameter sortKeys: Set if emitter should sort keys in lexicographic order. + /// - parameter sequenceStyle: Set the style for sequences (arrays / lists) + /// - parameter mappingStyle: Set the style for mappings (dictionaries) + /// - parameter newLineScalarStyle: Set the style for newline-containing scalars + /// - parameter redundancyAliasingStrategy: Set the strategy for identifying redundant structures and automatically aliasing them + public init(canonical: Bool = false, indent: Int = 0, width: Int = 0, allowUnicode: Bool = false, + lineBreak: Emitter.LineBreak = .ln, + explicitStart: Bool = false, + explicitEnd: Bool = false, + version: (major: Int, minor: Int)? = nil, + sortKeys: Bool = false, sequenceStyle: Node.Sequence.Style = .any, + mappingStyle: Node.Mapping.Style = .any, + newLineScalarStyle: Node.Scalar.Style = .any, + redundancyAliasingStrategy: RedundancyAliasingStrategy? = nil) { + self.canonical = canonical + self.indent = indent + self.width = width + self.allowUnicode = allowUnicode + self.lineBreak = lineBreak + self.explicitStart = explicitStart + self.explicitEnd = explicitEnd + self.version = version + self.sortKeys = sortKeys + self.sequenceStyle = sequenceStyle + self.mappingStyle = mappingStyle + self.newLineScalarStyle = newLineScalarStyle + self.redundancyAliasingStrategy = redundancyAliasingStrategy + } } /// Configuration options to use when emitting YAML. @@ -288,6 +337,8 @@ public final class Emitter { /// - parameter sortKeys: Set if emitter should sort keys in lexicographic order. /// - parameter sequenceStyle: Set the style for sequences (arrays / lists) /// - parameter mappingStyle: Set the style for mappings (dictionaries) + /// - parameter newLineScalarStyle: Set the style for newline-containing scalars + /// - parameter redundancyAliasingStrategy: Set the strategy for identifying redundant structures and automatically aliasing them public init(canonical: Bool = false, indent: Int = 0, width: Int = 0, @@ -299,7 +350,8 @@ public final class Emitter { sortKeys: Bool = false, sequenceStyle: Node.Sequence.Style = .any, mappingStyle: Node.Mapping.Style = .any, - newLineScalarStyle: Node.Scalar.Style = .any) { + newLineScalarStyle: Node.Scalar.Style = .any, + redundancyAliasingStrategy: RedundancyAliasingStrategy? = nil) { options = Options(canonical: canonical, indent: indent, width: width, @@ -311,7 +363,8 @@ public final class Emitter { sortKeys: sortKeys, sequenceStyle: sequenceStyle, mappingStyle: mappingStyle, - newLineScalarStyle: newLineScalarStyle) + newLineScalarStyle: newLineScalarStyle, + redundancyAliasingStrategy: redundancyAliasingStrategy) // configure emitter yaml_emitter_initialize(&emitter) yaml_emitter_set_output(&self.emitter, { pointer, buffer, size in @@ -413,38 +466,10 @@ public final class Emitter { } } -// MARK: - Options Initializer +//// MARK: - Options Initializer extension Emitter.Options { - /// Create `Emitter.Options` with the specified values. - /// - /// - parameter canonical: Set if the output should be in the "canonical" format described in the YAML - /// specification. - /// - parameter indent: Set the indentation value. - /// - parameter width: Set the preferred line width. -1 means unlimited. - /// - parameter allowUnicode: Set if unescaped non-ASCII characters are allowed. - /// - parameter lineBreak: Set the preferred line break. - /// - parameter explicitStart: Explicit document start `---`. - /// - parameter explicitEnd: Explicit document end `...`. - /// - parameter version: The `%YAML` directive value or nil. - /// - parameter sortKeys: Set if emitter should sort keys in lexicographic order. - /// - parameter sequenceStyle: Set the style for sequences (arrays / lists) - /// - parameter mappingStyle: Set the style for mappings (dictionaries) - public init(canonical: Bool = false, indent: Int = 0, width: Int = 0, allowUnicode: Bool = false, - lineBreak: Emitter.LineBreak = .ln, version: (major: Int, minor: Int)? = nil, - sortKeys: Bool = false, sequenceStyle: Node.Sequence.Style = .any, - mappingStyle: Node.Mapping.Style = .any, newLineScalarStyle: Node.Scalar.Style = .any) { - self.canonical = canonical - self.indent = indent - self.width = width - self.allowUnicode = allowUnicode - self.lineBreak = lineBreak - self.version = version - self.sortKeys = sortKeys - self.sequenceStyle = sequenceStyle - self.mappingStyle = mappingStyle - self.newLineScalarStyle = newLineScalarStyle - } + } // MARK: Implementation Details @@ -461,8 +486,17 @@ extension Emitter { case .scalar(let scalar): try serializeScalar(scalar) case .sequence(let sequence): try serializeSequence(sequence) case .mapping(let mapping): try serializeMapping(mapping) + case .alias(let alias): try serializeAlias(alias) } } + + private func serializeAlias(_ alias: Node.Alias) throws { + var event = yaml_event_t() + let anchor = alias.anchor.rawValue + yaml_alias_event_initialize(&event, + anchor) + try emit(&event) + } private func serializeScalar(_ scalar: Node.Scalar) throws { var value = scalar.string.utf8CString, tag = scalar.resolvedTag.name.rawValue.utf8CString @@ -472,7 +506,7 @@ extension Emitter { tag.withUnsafeMutableBytes { tag in yaml_scalar_event_initialize( &event, - nil, + scalar.anchor?.rawValue, tag.baseAddress?.assumingMemoryBound(to: UInt8.self), value.baseAddress?.assumingMemoryBound(to: UInt8.self), Int32(value.count - 1), @@ -492,7 +526,7 @@ extension Emitter { _ = tag.withUnsafeMutableBytes { tag in yaml_sequence_start_event_initialize( &event, - nil, + sequence.anchor?.rawValue, tag.baseAddress?.assumingMemoryBound(to: UInt8.self), implicit, sequenceStyle) @@ -511,7 +545,7 @@ extension Emitter { _ = tag.withUnsafeMutableBytes { tag in yaml_mapping_start_event_initialize( &event, - nil, + mapping.anchor?.rawValue, tag.baseAddress?.assumingMemoryBound(to: UInt8.self), implicit, mappingStyle) diff --git a/Sources/Yams/Encoder.swift b/Sources/Yams/Encoder.swift index fd17bac3..39b58ac3 100644 --- a/Sources/Yams/Encoder.swift +++ b/Sources/Yams/Encoder.swift @@ -28,10 +28,17 @@ public class YAMLEncoder { /// - throws: `EncodingError` if something went wrong while encoding. public func encode(_ value: T, userInfo: [CodingUserInfoKey: Any] = [:]) throws -> String { do { - let encoder = _Encoder(userInfo: userInfo, sequenceStyle: options.sequenceStyle, - mappingStyle: options.mappingStyle, newlineScalarStyle: options.newLineScalarStyle) + var finalUserInfo = userInfo + if let aliasingStrategy = options.redundancyAliasingStrategy { + finalUserInfo[.redundancyAliasingStrategyKey] = aliasingStrategy + } + let encoder = _Encoder(userInfo: finalUserInfo, + sequenceStyle: options.sequenceStyle, + mappingStyle: options.mappingStyle, + newlineScalarStyle: options.newLineScalarStyle) var container = encoder.singleValueContainer() try container.encode(value) + try options.redundancyAliasingStrategy?.releaseAnchorReferences() return try serialize(node: encoder.node, options: options) } catch let error as EncodingError { throw error @@ -165,7 +172,15 @@ private struct _KeyedEncodingContainer: KeyedEncodingContainerPr var codingPath: [CodingKey] { return encoder.codingPath } func encodeNil(forKey key: Key) throws { encoder.mapping[key.stringValue] = .null } func encode(_ value: T, forKey key: Key) throws where T: YAMLEncodable { try encoder(for: key).encode(value) } - func encode(_ value: T, forKey key: Key) throws where T: Encodable { try encoder(for: key).encode(value) } + func encode(_ value: T, forKey key: Key) throws where T: Encodable { + if let anchor = value as? Anchor, key.stringValue == Node.anchorKeyNode.string { + encoder.node = encoder.node.setting(anchor: anchor) + } else if let tag = value as? Tag, key.stringValue == Node.tagKeyNode.string { + encoder.node = encoder.node.setting(tag: tag) + } else { + try encoder(for: key).encode(value) + } + } func nestedContainer(keyedBy type: NestedKey.Type, forKey key: Key) -> KeyedEncodingContainer { @@ -225,21 +240,49 @@ extension _Encoder: SingleValueEncodingContainer { func encode(_ value: T) throws where T: YAMLEncodable { assertCanEncodeNewValue() - node = value.box() - if let stringValue = value as? (any StringProtocol), stringValue.contains("\n") { - node.scalar?.style = newlineScalarStyle + try encode(yamlEncodable: value) + } + + private func encode(yamlEncodable encodable: YAMLEncodable) throws { + func encodeNode() { + node = encodable.box() + if let stringValue = encodable as? (any StringProtocol), stringValue.contains("\n") { + node.scalar?.style = newlineScalarStyle + } + } + if let redundancyAliasingStrategy = userInfo[.redundancyAliasingStrategyKey] as? RedundancyAliasingStrategy { + switch try redundancyAliasingStrategy.alias(for: encodable) { + case .none: + encodeNode() + case let .anchor(anchor): + encodeNode() + self.node = self.node.setting(anchor: anchor) + case let .alias(anchor): + self.node = .alias(.init(anchor)) + } + } else { + encodeNode() } } func encode(_ value: T) throws where T: Encodable { assertCanEncodeNewValue() if let encodable = value as? YAMLEncodable { - node = encodable.box() - if let stringValue = value as? (any StringProtocol), stringValue.contains("\n") { - node.scalar?.style = newlineScalarStyle - } + try encode(yamlEncodable: encodable) } else { - try value.encode(to: self) + if let redundancyAliasingStrategy = userInfo[.redundancyAliasingStrategyKey] as? RedundancyAliasingStrategy { + switch try redundancyAliasingStrategy.alias(for: value) { + case .none: + try value.encode(to: self) + case let .anchor(anchor): + try value.encode(to: self) + self.node = self.node.setting(anchor: anchor) + case let .alias(anchor): + self.node = .alias(.init(anchor)) + } + } else { + try value.encode(to: self) + } } } diff --git a/Sources/Yams/Node.Alias.swift b/Sources/Yams/Node.Alias.swift new file mode 100644 index 00000000..1c3585bb --- /dev/null +++ b/Sources/Yams/Node.Alias.swift @@ -0,0 +1,58 @@ +// +// Node.Alias.swift +// Yams +// +// Created by Adora Lynch on 8/19/24. +// Copyright (c) 2024 Yams. All rights reserved. +// + +import Foundation + +// MARK: Node+Alias + +extension Node { + /// Scalar node. + public struct Alias { + /// The anchor for this alias. + public var anchor: Anchor + /// This node's tag (its type). + public var tag: Tag + /// The location for this node. + public var mark: Mark? + + /// Create a `Node.Alias` using the specified parameters. + /// + /// - parameter tag: This scalar's `Tag`. + /// - parameter mark: This scalar's `Mark`. + public init(_ anchor: Anchor, _ tag: Tag = .implicit, _ mark: Mark? = nil) { + self.anchor = anchor + self.tag = tag + self.mark = mark + } + } +} + +extension Node.Alias: Comparable { + /// :nodoc: + public static func < (lhs: Node.Alias, rhs: Node.Alias) -> Bool { + lhs.anchor.rawValue < rhs.anchor.rawValue + } +} + +extension Node.Alias: Equatable { + /// :nodoc: + public static func == (lhs: Node.Alias, rhs: Node.Alias) -> Bool { + lhs.anchor == rhs.anchor + } +} + +extension Node.Alias: Hashable { + /// :nodoc: + public func hash(into hasher: inout Hasher) { + hasher.combine(anchor) + } +} + +extension Node.Alias: TagResolvable { + static let defaultTagName = Tag.Name.implicit +} diff --git a/Sources/Yams/Node.Mapping.swift b/Sources/Yams/Node.Mapping.swift index a6feafc0..e9635fb8 100644 --- a/Sources/Yams/Node.Mapping.swift +++ b/Sources/Yams/Node.Mapping.swift @@ -16,6 +16,8 @@ extension Node { public var style: Style /// This mapping's `Mark`. public var mark: Mark? + /// The anchor for this node. + public weak var anchor: Anchor? /// The style to use when emitting a `Mapping`. public enum Style: UInt32 { @@ -33,11 +35,12 @@ extension Node { /// - parameter tag: This mapping's `Tag`. /// - parameter style: The style to use when emitting this `Mapping`. /// - parameter mark: This mapping's `Mark`. - public init(_ pairs: [(Node, Node)], _ tag: Tag = .implicit, _ style: Style = .any, _ mark: Mark? = nil) { + public init(_ pairs: [(Node, Node)], _ tag: Tag = .implicit, _ style: Style = .any, _ mark: Mark? = nil, _ anchor: Anchor? = nil) { self.pairs = pairs.map { Pair($0.0, $0.1) } self.tag = tag self.style = style self.mark = mark + self.anchor = anchor } } @@ -166,7 +169,7 @@ extension Node.Mapping { index += 1 } } - return Node.Mapping(merge + pairs, tag, style) + return Node.Mapping(merge + pairs, tag, style, nil, anchor) } } diff --git a/Sources/Yams/Node.Scalar.swift b/Sources/Yams/Node.Scalar.swift index 662e8b4c..68640f0a 100644 --- a/Sources/Yams/Node.Scalar.swift +++ b/Sources/Yams/Node.Scalar.swift @@ -23,6 +23,8 @@ extension Node { public var style: Style /// The location for this node. public var mark: Mark? + /// The anchor for this node. + public weak var anchor: Anchor? /// The style to use when emitting a `Scalar`. public enum Style: UInt32 { @@ -48,11 +50,12 @@ extension Node { /// - parameter tag: This scalar's `Tag`. /// - parameter style: The style to use when emitting this `Scalar`. /// - parameter mark: This scalar's `Mark`. - public init(_ string: String, _ tag: Tag = .implicit, _ style: Style = .any, _ mark: Mark? = nil) { + public init(_ string: String, _ tag: Tag = .implicit, _ style: Style = .any, _ mark: Mark? = nil, _ anchor: Anchor? = nil) { self.string = string self.tag = tag self.style = style self.mark = mark + self.anchor = anchor } } diff --git a/Sources/Yams/Node.Sequence.swift b/Sources/Yams/Node.Sequence.swift index 571c9aee..c74f039e 100644 --- a/Sources/Yams/Node.Sequence.swift +++ b/Sources/Yams/Node.Sequence.swift @@ -18,6 +18,8 @@ extension Node { public var style: Style /// The location for this node. public var mark: Mark? + /// The anchor for this node. + public weak var anchor: Anchor? /// The style to use when emitting a `Sequence`. public enum Style: UInt32 { @@ -35,11 +37,12 @@ extension Node { /// - parameter tag: This sequence's `Tag`. /// - parameter style: The style to use when emitting this `Sequence`. /// - parameter mark: This sequence's `Mark`. - public init(_ nodes: [Node], _ tag: Tag = .implicit, _ style: Style = .any, _ mark: Mark? = nil) { + public init(_ nodes: [Node], _ tag: Tag = .implicit, _ style: Style = .any, _ mark: Mark? = nil, _ anchor: Anchor? = nil) { self.nodes = nodes self.tag = tag self.style = style self.mark = mark + self.anchor = anchor } } diff --git a/Sources/Yams/Node.swift b/Sources/Yams/Node.swift index 87728960..c848da43 100644 --- a/Sources/Yams/Node.swift +++ b/Sources/Yams/Node.swift @@ -16,6 +16,8 @@ public enum Node: Hashable { case mapping(Mapping) /// Sequence node. case sequence(Sequence) + /// Alias node. + case alias(Alias) } extension Node { @@ -24,8 +26,8 @@ extension Node { /// - parameter string: String value for this node. /// - parameter tag: Tag for this node. /// - parameter style: Style to use when emitting this node. - public init(_ string: String, _ tag: Tag = .implicit, _ style: Scalar.Style = .any) { - self = .scalar(.init(string, tag, style)) + public init(_ string: String, _ tag: Tag = .implicit, _ style: Scalar.Style = .any, _ anchor: Anchor? = nil) { + self = .scalar(.init(string, tag, style, nil, anchor)) } /// Create a `Node.mapping` with a sequence of node pairs, tag & scalar style. @@ -33,8 +35,8 @@ extension Node { /// - parameter pairs: Pairs of nodes to use for this node. /// - parameter tag: Tag for this node. /// - parameter style: Style to use when emitting this node. - public init(_ pairs: [(Node, Node)], _ tag: Tag = .implicit, _ style: Mapping.Style = .any) { - self = .mapping(.init(pairs, tag, style)) + public init(_ pairs: [(Node, Node)], _ tag: Tag = .implicit, _ style: Mapping.Style = .any, _ anchor: Anchor? = nil) { + self = .mapping(.init(pairs, tag, style, nil, anchor)) } /// Create a `Node.sequence` with a sequence of nodes, tag & scalar style. @@ -42,8 +44,8 @@ extension Node { /// - parameter nodes: Sequence of nodes to use for this node. /// - parameter tag: Tag for this node. /// - parameter style: Style to use when emitting this node. - public init(_ nodes: [Node], _ tag: Tag = .implicit, _ style: Sequence.Style = .any) { - self = .sequence(.init(nodes, tag, style)) + public init(_ nodes: [Node], _ tag: Tag = .implicit, _ style: Sequence.Style = .any, _ anchor: Anchor? = nil) { + self = .sequence(.init(nodes, tag, style, nil, anchor)) } } @@ -58,6 +60,7 @@ extension Node { case let .scalar(scalar): return scalar.resolvedTag case let .mapping(mapping): return mapping.resolvedTag case let .sequence(sequence): return sequence.resolvedTag + case let .alias(alias): return alias.resolvedTag } } @@ -67,6 +70,17 @@ extension Node { case let .scalar(scalar): return scalar.mark case let .mapping(mapping): return mapping.mark case let .sequence(sequence): return sequence.mark + case let .alias(alias): return alias.mark + } + } + + /// The anchor for this node. + public var anchor: Anchor? { + switch self { + case let .scalar(scalar): return scalar.anchor + case let .mapping(mapping): return mapping.anchor + case let .sequence(sequence): return sequence.anchor + case let .alias(alias): return alias.anchor } } @@ -139,7 +153,7 @@ extension Node { public subscript(node: Node) -> Node? { get { switch self { - case .scalar: return nil + case .scalar, .alias: return nil case let .mapping(mapping): return mapping[node] case let .sequence(sequence): @@ -150,7 +164,7 @@ extension Node { set { guard let newValue = newValue else { return } switch self { - case .scalar: return + case .scalar, .alias: return case .mapping(var mapping): mapping[node] = newValue self = .mapping(mapping) @@ -290,4 +304,38 @@ extension Node { } return false } + + func setting(anchor: Anchor) -> Self { + switch self { + case var .mapping(mapping): + mapping.anchor = anchor + return .mapping(mapping) + case var .sequence(sequence): + sequence.anchor = anchor + return .sequence(sequence) + case var .scalar(scalar): + scalar.anchor = anchor + return .scalar(scalar) + case var .alias(alias): + alias.anchor = anchor + return .alias(alias) + } + } + + func setting(tag: Tag) -> Self { + switch self { + case var .mapping(mapping): + mapping.tag = tag + return .mapping(mapping) + case var .sequence(sequence): + sequence.tag = tag + return .sequence(sequence) + case var .scalar(scalar): + scalar.tag = tag + return .scalar(scalar) + case var .alias(alias): + alias.tag = tag + return .alias(alias) + } + } } diff --git a/Sources/Yams/Parser.swift b/Sources/Yams/Parser.swift index 41a4d74a..233c7fd6 100644 --- a/Sources/Yams/Parser.swift +++ b/Sources/Yams/Parser.swift @@ -254,7 +254,9 @@ public final class Parser { // MARK: - Private Members - private var anchors = [String: Node]() + private var _anchorMap = [Anchor: Node]() + private var _anchorList = [Anchor]() + private var anchors: [Anchor: Node] { _anchorMap } private var parser = yaml_parser_t() private enum Buffer { @@ -263,6 +265,20 @@ public final class Parser { case utf16(Data) } private var buffer: Buffer + + // MARK: – Pivate Mutators + private func register(anchor: Anchor?, to node: Node) { + if let anchor { + _anchorList.append(anchor) + // We keep a list (not a set) of all anchors encountered + // because yaml anchors are allowed to shadow one another. + // + // The map will keep the latest reference as expected + // but without the list the map will release reference to + // one of the Anchor instances whenever duplicates are encountered. + _anchorMap[anchor] = node + } + } } // MARK: Implementation Details @@ -324,10 +340,9 @@ private extension Parser { } func loadScalar(from event: Event) throws -> Node { - let node = Node.scalar(.init(event.scalarValue, tag(event.scalarTag), event.scalarStyle, event.startMark)) - if let anchor = event.scalarAnchor { - anchors[anchor] = node - } + let anchor = event.scalarAnchor + let node = Node.scalar(.init(event.scalarValue, tag(event.scalarTag), event.scalarStyle, event.startMark, anchor)) + register(anchor: anchor, to: node) return node } @@ -338,10 +353,9 @@ private extension Parser { array.append(try loadNode(from: event)) event = try parse() } - let node = Node.sequence(.init(array, tag(firstEvent.sequenceTag), event.sequenceStyle, firstEvent.startMark)) - if let anchor = firstEvent.sequenceAnchor { - anchors[anchor] = node - } + let anchor = firstEvent.sequenceAnchor + let node = Node.sequence(.init(array, tag(firstEvent.sequenceTag), event.sequenceStyle, firstEvent.startMark, anchor)) + register(anchor: anchor, to: node) return node } @@ -355,10 +369,9 @@ private extension Parser { pairs.append((key, value)) event = try parse() } - let node = Node.mapping(.init(pairs, tag(firstEvent.mappingTag), event.mappingStyle, firstEvent.startMark)) - if let anchor = firstEvent.mappingAnchor { - anchors[anchor] = node - } + let anchor = firstEvent.mappingAnchor + let node = Node.mapping(.init(pairs, tag(firstEvent.mappingTag), event.mappingStyle, firstEvent.startMark, anchor)) + register(anchor: anchor, to: node) return node } @@ -378,13 +391,13 @@ private class Event { } // alias - var aliasAnchor: String? { - return string(from: event.data.alias.anchor) + var aliasAnchor: Anchor? { + return string(from: event.data.alias.anchor).map(Anchor.init(stringLiteral: )) } // scalar - var scalarAnchor: String? { - return string(from: event.data.scalar.anchor) + var scalarAnchor: Anchor? { + return string(from: event.data.scalar.anchor).map(Anchor.init(stringLiteral: )) } var scalarStyle: Node.Scalar.Style { // swiftlint:disable:next force_unwrapping @@ -405,8 +418,8 @@ private class Event { } // sequence - var sequenceAnchor: String? { - return string(from: event.data.sequence_start.anchor) + var sequenceAnchor: Anchor? { + return string(from: event.data.sequence_start.anchor).map(Anchor.init(stringLiteral: )) } var sequenceStyle: Node.Sequence.Style { // swiftlint:disable:next force_unwrapping @@ -418,8 +431,8 @@ private class Event { } // mapping - var mappingAnchor: String? { - return string(from: event.data.scalar.anchor) + var mappingAnchor: Anchor? { + return string(from: event.data.mapping_start.anchor).map(Anchor.init(stringLiteral: )) } var mappingStyle: Node.Mapping.Style { // swiftlint:disable:next force_unwrapping diff --git a/Sources/Yams/RedundancyAliasingStrategy.swift b/Sources/Yams/RedundancyAliasingStrategy.swift new file mode 100644 index 00000000..d802847e --- /dev/null +++ b/Sources/Yams/RedundancyAliasingStrategy.swift @@ -0,0 +1,117 @@ +// +// RedundancyAliasingStrategy.swift +// Yams +// +// Created by Adora Lynch on 8/15/24. +// Copyright (c) 2024 Yams. All rights reserved. +// + +import Foundation + +public enum RedundancyAliasingOutcome { + case anchor(Anchor) + case alias(Anchor) + case none +} + +/// A class-bound protocol which implements a strategy for detecting aliasable values in a YAML document. +/// Implementations should return RedundancyAliasingOutcome.anchor(...) for the first occurrence of a value. +/// Subsequent occurrences of the same value (where same-ness is defined by the implementation) should +/// return RedundancyAliasingOutcome.alias(...) where the contained Anchor has the same value as the previously +/// returned RedundancyAliasingOutcome.anchor(...). Its the identity of the Anchor values returned that ultimately +/// informs the YAML encoder when to use aliases. +/// N,B. It is essential that implementations release all references to Anchors which are created by this type +/// when releaseAnchorReferences() is called by the Encoder. After this call the implementation will no longer be +/// referenced by the Encoder and will itself be released. +public protocol RedundancyAliasingStrategy: AnyObject { + + /// Implementations should return RedundancyAliasingOutcome.anchor(...) for the first occurrence of a value. + /// Subsequent occurrences of the same value (where same-ness is defined by the implementation) should + /// return RedundancyAliasingOutcome.alias(...) where the contained Anchor has the same value as the previously + /// returned RedundancyAliasingOutcome.anchor(...). Its the identity of the Anchor values returned that ultimately + /// informs the YAML encoder when to use aliases. + func alias(for encodable: any Encodable) throws -> RedundancyAliasingOutcome + + /// It is essential that implementations release all references to Anchors which are created by this type + /// when releaseAnchorReferences() is called by the Encoder. After this call, the implementation will no longer be + /// referenced by the Encoder and will itself be released. + + func releaseAnchorReferences() throws +} + +/// An implementation of RedundancyAliasingStrategy that defines alias-ability by Hashable-Equality. +/// i.e. if two values are Hashable-Equal, they will be aliased in the resultant YML document. +public class HashableAliasingStrategy: RedundancyAliasingStrategy { + private var hashesToAliases: [AnyHashable: Anchor] = [:] + + let uniqueAliasProvider = UniqueAliasProvider() + + public init() {} + + public func alias(for encodable: any Encodable) throws -> RedundancyAliasingOutcome { + guard let hashable = encodable as? any Hashable & Encodable else { + return .none + } + return try alias(for: hashable) + } + + private func alias(for hashable: any Hashable & Encodable) throws -> RedundancyAliasingOutcome { + let anyHashable = AnyHashable(hashable) + if let existing = hashesToAliases[anyHashable] { + return .alias(existing) + } else { + let newAlias = uniqueAliasProvider.uniqueAlias(for: hashable) + hashesToAliases[anyHashable] = newAlias + return .anchor(newAlias) + } + } + + public func releaseAnchorReferences() throws { + hashesToAliases.removeAll() + } +} + +/// An implementation of RedundancyAliasingStrategy that defines alias-ability by the coded representation of the values. +/// i.e. if two values encode to exactly the same, they will be aliased in the resultant YML document even if the values themselves are of different types +public class StrictEncodableAliasingStrategy: RedundancyAliasingStrategy { + private var codedToAliases: [String: Anchor] = [:] + + let uniqueAliasProvider = UniqueAliasProvider() + + public init() {} + + private let encoder = YAMLEncoder() + + public func alias(for encodable: any Encodable) throws -> RedundancyAliasingOutcome { + let coded = try encoder.encode(encodable) + if let existing = codedToAliases[coded] { + return .alias(existing) + } else { + let newAlias = uniqueAliasProvider.uniqueAlias(for: encodable) + codedToAliases[coded] = newAlias + return .anchor(newAlias) + } + } + + public func releaseAnchorReferences() throws { + codedToAliases.removeAll() + } +} + +class UniqueAliasProvider { + private var counter = 0 + + func uniqueAlias(for encodable: any Encodable) -> Anchor { + if let anchorProviding = encodable as? YamlAnchorProviding, + let anchor = anchorProviding.yamlAnchor { + return anchor + } else { + counter += 1 + return Anchor(rawValue: String(counter)) + } + } +} + +extension CodingUserInfoKey { + internal static let redundancyAliasingStrategyKey = Self(rawValue: "redundancyAliasingStrategy")! +} diff --git a/Sources/Yams/Resolver.swift b/Sources/Yams/Resolver.swift index 40abb2da..4cf8d743 100644 --- a/Sources/Yams/Resolver.swift +++ b/Sources/Yams/Resolver.swift @@ -48,6 +48,8 @@ public final class Resolver { return resolveTag(of: mapping) case let .sequence(sequence): return resolveTag(of: sequence) + case let .alias(alias): + return resolveTag(of: alias) } } diff --git a/Sources/Yams/Tag.swift b/Sources/Yams/Tag.swift index fd2e1dbd..47e774a5 100644 --- a/Sources/Yams/Tag.swift +++ b/Sources/Yams/Tag.swift @@ -85,6 +85,24 @@ extension Tag: Hashable { } } +extension Tag: RawRepresentable { + public convenience init?(rawValue: String) { + self.init(stringLiteral: rawValue) + } + public var rawValue: String { + name.rawValue + } +} + +extension Tag: Codable {} + +extension Tag: ExpressibleByStringLiteral { + /// :nodoc: + public convenience init(stringLiteral value: String) { + self.init(.init(rawValue: value)) + } +} + extension Tag.Name: ExpressibleByStringLiteral { /// :nodoc: public init(stringLiteral value: String) { @@ -92,6 +110,8 @@ extension Tag.Name: ExpressibleByStringLiteral { } } +extension Tag.Name: Codable {} + // http://www.yaml.org/spec/1.2/spec.html#Schema extension Tag.Name { // Special diff --git a/Sources/Yams/YamlAnchorProviding.swift b/Sources/Yams/YamlAnchorProviding.swift new file mode 100644 index 00000000..dfc4e899 --- /dev/null +++ b/Sources/Yams/YamlAnchorProviding.swift @@ -0,0 +1,45 @@ +// +// YamlAnchorProviding.swift +// Yams +// +// Created by Adora Lynch on 8/15/24. +// Copyright (c) 2024 Yams. All rights reserved. +// + +import Foundation + +/// Types that conform to YamlAnchorProviding and Encodable can optionally dictate the name of +/// a yaml anchor when they are encoded with YAMLEncoder +public protocol YamlAnchorProviding { + var yamlAnchor: Anchor? { get } +} + +/// YamlAnchorCoding refines YamlAnchorProviding. +/// Types that conform to YamlAnchorCoding and Decodable can decode yaml anchors +/// from source documents into `Anchor` values for reference or modification in memory. +public protocol YamlAnchorCoding: YamlAnchorProviding { + var yamlAnchor: Anchor? { get set } +} + +internal extension Node { + static let anchorKeyNode: Self = .scalar(.init(YamlAnchorFunctionNameProvider().getName())) +} + +private final class YamlAnchorFunctionNameProvider: YamlAnchorProviding { + + fileprivate var functionName: StaticString? + + var yamlAnchor: Anchor? { + functionName = #function + return nil + } + + func getName() -> StaticString { + _ = yamlAnchor + return functionName! + } + + func getName() -> String { + String(describing: getName() as StaticString) + } +} diff --git a/Sources/Yams/YamlTagProviding.swift b/Sources/Yams/YamlTagProviding.swift new file mode 100644 index 00000000..0f2a2d63 --- /dev/null +++ b/Sources/Yams/YamlTagProviding.swift @@ -0,0 +1,43 @@ +// +// YamlTagProviding.swift +// +// +// Created by Adora Lynch on 9/5/24. +// Copyright (c) 2024 Yams. All rights reserved. +// + +/// Types that conform to YamlTagProviding and Encodable can optionally dictate the name of +/// a yaml tag when they are encoded with YAMLEncoder +public protocol YamlTagProviding { + var yamlTag: Tag? { get } +} + +/// YamlTagCoding refines YamlTagProviding. +/// Types that conform to YamlTagCoding and Decodable can decode yaml tags +/// from source documents into `Tag` values for reference or modification in memory. +public protocol YamlTagCoding: YamlTagProviding { + var yamlTag: Tag? { get set } +} + +internal extension Node { + static let tagKeyNode: Self = .scalar(.init(YamlTagFunctionNameProvider().getName())) +} + +private final class YamlTagFunctionNameProvider: YamlTagProviding { + + fileprivate var functionName: StaticString? + + var yamlTag: Tag? { + functionName = #function + return nil + } + + func getName() -> StaticString { + _ = yamlTag + return functionName! + } + + func getName() -> String { + String(describing: getName() as StaticString) + } +} diff --git a/Tests/YamsTests/AnchorCodingTests.swift b/Tests/YamsTests/AnchorCodingTests.swift new file mode 100644 index 00000000..b3ddc43b --- /dev/null +++ b/Tests/YamsTests/AnchorCodingTests.swift @@ -0,0 +1,298 @@ +// +// AnchorEncodingTests.swift +// Yams +// +// Created by Adora Lynch on 8/9/24. +// Copyright (c) 2024 Yams. All rights reserved. +// + +import XCTest +import Yams + +class AnchorCodingTests: XCTestCase { + + /// Test the encoding of a yaml anchor using a type that conforms to YamlAnchorProviding + func testYamlAnchorProviding_valuePresent() throws { + let simpleStruct = SimpleWithAnchor(nested: .init(stringValue: "it's a value"), intValue: 52) + + _testRoundTrip(of: simpleStruct, + expectedYAML:""" + &simple + nested: + stringValue: it's a value + intValue: 52 + + """ ) // ^ the Yams.Anchor is encoded as a yaml anchor + } + + /// Test the encoding of a a type that does not conform to YamlAnchorProviding but none the less declares a coding member with the same name + func testStringTypeAnchorName_valuePresent() throws { + let simpleStruct = SimpleWithStringTypeAnchorName(nested: .init(stringValue: "it's a value"), + intValue: 52, + yamlAnchor: "but typed as a string") + + _testRoundTrip(of: simpleStruct, + expectedYAML:""" + nested: + stringValue: it's a value + intValue: 52 + yamlAnchor: but typed as a string + + """ ) // ^ the member is _not_ treated as an anchor + } + + /// Nothing interesting happens when a type does not conform to YamlAnchorProviding none the less declares a coding member with the same name but that value is nil + func testStringTypeAnchorName_valueNotPresent() throws { + let expectedStruct = SimpleWithStringTypeAnchorName(nested: .init(stringValue: "it's a value"), + intValue: 52, + yamlAnchor: nil) + _testRoundTrip(of: expectedStruct, + expectedYAML: """ + nested: + stringValue: it's a value + intValue: 52 + + """) + } + + /// This test documents some undesirable behavior, but in an unlikely circumstance. + /// If the decoded type does not conform to YamlAnchorProviding it can still have a coding key called `yamlAnchor` + /// If Yams tries to decode such a type AND the document has a nil value for `yamlAnchor` AND the parent context is a mapping AND that mapping has an actual anchor (in the document) + /// THEN Yams wrongly tries to decode the anchor as the declared type of key `yamlAnchor`. + /// If that declared type can be decoded from a scalar string value (like String and RawRepresentable where RawValue == String) then the decoding will actually succeed. + /// Which effectively injects an unexpected value into the decoded type. + func testStringTypeAnchorName_withAnchorPresent_valueNil() throws { + let expectedStruct = SimpleWithStringTypeAnchorName(nested: .init(stringValue: "it's a value"), + intValue: 52, + yamlAnchor: nil) + let decoder = YAMLDecoder() + let data = """ + &AnActualAnchor + nested: + stringValue: it's a value + intValue: 52 + + """.data(using: .utf8)! + + let decodedStruct = try decoder.decode(SimpleWithStringTypeAnchorName.self, from: data) + + let fixBulletin = "YESS!!! YOU FIXED IT! See \(#file):\(#line) for explanation." + + // begin assertions of known-but-undesirable behavior + XCTAssertNotEqual(decodedStruct, expectedStruct, fixBulletin) // We wish this was equal + XCTAssertEqual(decodedStruct.yamlAnchor, "AnActualAnchor", fixBulletin) // we wish .yamlAnchor was nil + // end assertions of known-but-undesirable behavior + + + // Check the remainder of the properties that the above confusion did not involve + XCTAssertEqual(decodedStruct.nested, expectedStruct.nested) + XCTAssertEqual(decodedStruct.intValue, expectedStruct.intValue) + } +} + +class AnchorAliasingTests: XCTestCase { + + /// CYaml library does not detect identical values and automatically alias them. + func testCyamlDoesNotAutoAlias_noAnchor() throws { + let simpleNoAnchor = SimpleWithoutAnchor(nested: .init(stringValue: "it's a value"), intValue: 52) + let differentTypesOneAnchor = SimplePair(first: simpleNoAnchor, + second: simpleNoAnchor) + + _testRoundTrip(of: differentTypesOneAnchor, + expectedYAML:""" + first: + nested: + stringValue: it's a value + intValue: 52 + second: + nested: + stringValue: it's a value + intValue: 52 + + """ ) + } + + /// CYaml library does not detect identical values and automatically alias them even if the first occurrence has an anchor. + func testCyamlDoesNotAutoAlias_uniqueAnchor() throws { + let simpleStruct = SimpleWithAnchor(nested: .init(stringValue: "it's a value"), intValue: 52) + let simpleNoAnchor = SimpleWithoutAnchor(nested: .init(stringValue: "it's a value"), intValue: 52) + let differentTypesOneAnchor = SimplePair(first: simpleStruct, + second: simpleNoAnchor) + + _testRoundTrip(of: differentTypesOneAnchor, + expectedYAML:""" + first: &simple + nested: + stringValue: it's a value + intValue: 52 + second: + nested: + stringValue: it's a value + intValue: 52 + + """ ) + } + + /// CYaml library does not detect identical values and automatically alias them even if they have identical anchors. + // This one is not a shortcoming of CYaml. The yaml spec requires that nodes can shadow earlier anchors. + func testCyamlDoesNotAutoAlias_duplicateAnchor() throws { + let simpleStruct = SimpleWithAnchor(nested: .init(stringValue: "it's a value"), intValue: 52) + let duplicatedStructPair = SimplePair(first: simpleStruct, second: simpleStruct) + + _testRoundTrip(of: duplicatedStructPair, + expectedYAML:""" + first: &simple + nested: + stringValue: it's a value + intValue: 52 + second: &simple + nested: + stringValue: it's a value + intValue: 52 + + """ ) + } + + + /// If types conform to YamlAnchorProviding and are Hashable-Equal then HashableAliasingStrategy aliases them + func testEncoderAutoAlias_Hashable_duplicateAnchor() throws { + let simpleStruct = SimpleWithAnchor(nested: .init(stringValue: "it's a value"), intValue: 52) + let duplicatedStructArray = [simpleStruct, simpleStruct] + + let options = YAMLEncoder.Options(redundancyAliasingStrategy: HashableAliasingStrategy()) + _testRoundTrip(of: duplicatedStructArray, + with: options, + expectedYAML:""" + - &simple + nested: + stringValue: it's a value + intValue: 52 + - *simple + + """ ) + } + + /// If types do NOT conform to YamlAnchorProviding and are Hashable-Equal then HashableAliasingStrategy aliases them + func testEncoderAutoAlias_Hashable_noAnchors() throws { + let simpleStruct = SimpleWithoutAnchor(nested: .init(stringValue: "it's a value"), intValue: 52) + let duplicatedStructArray = [simpleStruct, simpleStruct] // zero specified anchor + + let options = YAMLEncoder.Options(redundancyAliasingStrategy: HashableAliasingStrategy()) + _testRoundTrip(of: duplicatedStructArray, + with: options, + expectedYAML:""" + - &2 + nested: + stringValue: it's a value + intValue: 52 + - *2 + + """ ) + } + + /// If types conform to YamlAnchorProviding and are NOT Hashable-Equal then HashableAliasingStrategy does not alias them + /// even though their members may still be Hashable-Equal and therefor maybe aliased. + func testEncoderAutoAlias_Hashable_uniqueAnchor() throws { + let differentTypesOneAnchors = SimplePair(first: SimpleWithAnchor(nested: .init(stringValue: "it's a value"), intValue: 52), + second: SimpleWithoutAnchor(nested: .init(stringValue: "it's a value"), intValue: 52)) + + let options = YAMLEncoder.Options(redundancyAliasingStrategy: HashableAliasingStrategy()) + _testRoundTrip(of: differentTypesOneAnchors, + with: options, + expectedYAML:""" + first: &simple + nested: &2 + stringValue: it's a value + intValue: &4 52 + second: + nested: *2 + intValue: *4 + + """ ) + } + + /// If types conform to YamlAnchorProviding and are NOT Hashable-Equal then HashableAliasingStrategy does not alias them + /// even though their members may still be Hashable-Equal and therefor maybe aliased. + /// Note particularly that the to Simple* values here have exactly the same encoded representation, they're just different types and thus not Hashable-Equal + func testEncoderAutoAlias_Hashable_noAnchor() throws { + let differentTypesNoAnchors = SimplePair(first: SimpleWithoutAnchor2(nested: .init(stringValue: "it's a value"), intValue: 52), + second: SimpleWithoutAnchor(nested: .init(stringValue: "it's a value"), intValue: 52)) + + let options = YAMLEncoder.Options(redundancyAliasingStrategy: HashableAliasingStrategy()) + _testRoundTrip(of: differentTypesNoAnchors, + with: options, + expectedYAML:""" + first: + nested: &3 + stringValue: it's a value + intValue: &5 52 + second: + nested: *3 + intValue: *5 + + """ ) + } + + /// If types conform to YamlAnchorProviding and have exactly the same encoded representation then StrictEncodableAliasingStrategy alias them + /// even though they are encoded and decoded from different types. + func testEncoderAutoAlias_StrictEncodable_NoAnchors() throws { + let differentTypesNoAnchors = SimplePair(first: SimpleWithoutAnchor2(nested: .init(stringValue: "it's a value"), intValue: 52), + second: SimpleWithoutAnchor(nested: .init(stringValue: "it's a value"), intValue: 52)) + + var options = YAMLEncoder.Options() + options.redundancyAliasingStrategy = StrictEncodableAliasingStrategy() + _testRoundTrip(of: differentTypesNoAnchors, + with: options, + expectedYAML:""" + first: &2 + nested: + stringValue: it's a value + intValue: 52 + second: *2 + + """ ) + } + + /// A type used to contain values used during testing + private struct SimplePair: Hashable, Codable { + let first: First + let second: Second + } + +} + +// MARK: - Types used for Anchor encoding tests. + +fileprivate struct NestedStruct: Codable, Hashable { + let stringValue: String +} +fileprivate protocol SimpleProtocol: Codable, Hashable { + var nested: NestedStruct { get } + var intValue: Int { get } +} + +fileprivate struct SimpleWithAnchor: SimpleProtocol, YamlAnchorProviding { + let nested: NestedStruct + let intValue: Int + var yamlAnchor: Anchor? = "simple" +} + +fileprivate struct SimpleWithoutAnchor: SimpleProtocol { + let nested: NestedStruct + let intValue: Int +} + +fileprivate struct SimpleWithoutAnchor2: SimpleProtocol { + let nested: NestedStruct + let intValue: Int + var unrelatedValue: String? +} + +fileprivate struct SimpleWithStringTypeAnchorName: SimpleProtocol { + let nested: NestedStruct + let intValue: Int + var yamlAnchor: String? = "StringTypeAnchor" +} + + + diff --git a/Tests/YamsTests/AnchorTolerancesTests.swift b/Tests/YamsTests/AnchorTolerancesTests.swift new file mode 100644 index 00000000..60d126cb --- /dev/null +++ b/Tests/YamsTests/AnchorTolerancesTests.swift @@ -0,0 +1,162 @@ +// +// AnchorTolerancesTests.swift +// Yams +// +// Created by Adora Lynch on 9/18/24. +// Copyright (c) 2024 Yams. All rights reserved. +// + +import XCTest +import Yams + +class AnchorTolerancesTests: XCTestCase { + + struct Example: Codable, Hashable { + var myCustomAnchorDeclaration: Anchor + var extraneousValue: Int + } + + /// Any type that is Encodable and contains an `Anchor`value but with a coding key different from YamlAnchorProviding + /// will not encode to a yaml anchor + /// This may be unexpected + func testAnchorEncoding_undeclaredBehavior() throws { + let expectedYAML = """ + myCustomAnchorDeclaration: I-did-it-myyyyy-way + extraneousValue: 3 + + """ + + let value = Example(myCustomAnchorDeclaration: "I-did-it-myyyyy-way", + extraneousValue: 3) + + let encoder = YAMLEncoder() + let producedYAML = try encoder.encode(value) + XCTAssertEqual(producedYAML, expectedYAML, "Produced YAML not identical to expected YAML.") + } + + /// Any type that is Encodable and contains an `Anchor`value with the same coding key as YamlAnchorProviding + /// will encode to a yaml anchor even though the type does not conform to YamlAnchorProviding + /// This may be unexpected + func testAnchorEncoding_undeclaredBehavior_7() throws { + struct Example: Codable, Hashable { + var yamlAnchor: Anchor + var extraneousValue: Int + } + + let expectedYAML = """ + &I-did-it-myyyyy-way + extraneousValue: 3 + + """ + + let value = Example(yamlAnchor: "I-did-it-myyyyy-way", + extraneousValue: 3) + + let encoder = YAMLEncoder() + let producedYAML = try encoder.encode(value) + XCTAssertEqual(producedYAML, expectedYAML, "Produced YAML not identical to expected YAML.") + } + + /// Any type that is Decodable and contains an `Anchor` value but with a coding key different from YamlAnchorProviding + /// will not decode an anchor from the text representation. + /// In this case a key not found error will be thrown during decoding + /// This may be unexpected + func testAnchorDecoding_undeclaredBehavior_1() throws { + let sourceYAML = """ + &a-different-tag + extraneousValue: 3 + + """ + let decoder = YAMLDecoder() + XCTAssertThrowsError(try decoder.decode(Example.self, from: sourceYAML)) + // error is ^^ key not found, "myCustomAnchorDeclaration" + } + + /// Any type that is Decodable and contains an `Anchor` value but with a coding key different from YamlAnchorProviding + /// will not decode an anchor from the text representation. + /// In this case the decoding is successful and the anchor is respected by the parser. + /// This may be unexpected + func testAnchorDecoding_undeclaredBehavior_6() throws { + struct Example: Codable, Hashable { + var myCustomAnchorDeclaration: Anchor? + var extraneousValue: Int + } + let sourceYAML = """ + &a-different-tag + extraneousValue: 3 + + """ + + let expectedValue = Example(myCustomAnchorDeclaration: nil, + extraneousValue: 3) + + let decoder = YAMLDecoder() + let decodedValue = try decoder.decode(Example.self, from: sourceYAML) + XCTAssertEqual(decodedValue, expectedValue, "\(Example.self) did not round-trip to an equal value.") + } + + /// Any type that is Decodable and contains an `Anchor` value with the same coding key as YamlAnchorProviding + /// will decode an anchor from the text representation even though the type does not conform to YamlAnchorCoding + /// This may be unexpected + func testAnchorDecoding_undeclaredBehavior_8() throws { + struct Example: Codable, Hashable { + var yamlAnchor: Anchor? + var extraneousValue: Int + } + let sourceYAML = """ + &a-different-tag + extraneousValue: 3 + + """ + + let expectedValue = Example(yamlAnchor: "a-different-tag", + extraneousValue: 3) + + let decoder = YAMLDecoder() + let decodedValue = try decoder.decode(Example.self, from: sourceYAML) + XCTAssertEqual(decodedValue, expectedValue, "\(Example.self) did not round-trip to an equal value.") + } + + /// Any type that is Decodable and contains an `Anchor` value but with a coding key different from YamlAnchorProviding + /// will not decode an anchor from the text representation. + /// In this case the decoding is successful and the anchor is respected by the parser. + /// This is expected behavior, but in a strange situation. + func testAnchorDecoding_undeclaredBehavior_3() throws { + let sourceYAML = """ + &a-different-tag + extraneousValue: 3 + myCustomAnchorDeclaration: deliver-us-from-evil + + """ + let expectedValue = Example(myCustomAnchorDeclaration: "deliver-us-from-evil", + extraneousValue: 3) + + let decoder = YAMLDecoder() + let decodedValue = try decoder.decode(Example.self, from: sourceYAML) + XCTAssertEqual(decodedValue, expectedValue, "\(Example.self) did not round-trip to an equal value.") + + } + + /// Any type that is Decodable and contains an `Anchor` value but with a coding key different from YamlAnchorProviding + /// will not decode an anchor from the text representation. + /// In this case the decoding is successful even though and the `Anchor` was initialized with unsupported characters. + /// The anchor is respected by the parser. + /// This is expected behavior, but in a strange situation. + func testAnchorDecoding_undeclaredBehavior_2() throws { + let sourceYAML = """ + &a-different-tag + extraneousValue: 3 + myCustomAnchorDeclaration: "deliver us from |()evil" + + """ + + let expectedValue = Example(myCustomAnchorDeclaration: "deliver us from |()evil", + extraneousValue: 3) + + let decoder = YAMLDecoder() + let decodedValue = try decoder.decode(Example.self, from: sourceYAML) + XCTAssertEqual(decodedValue, expectedValue, "\(Example.self) did not round-trip to an equal value.") + + } + +} diff --git a/Tests/YamsTests/EncoderTests.swift b/Tests/YamsTests/EncoderTests.swift index e5a06dd2..284cfacf 100644 --- a/Tests/YamsTests/EncoderTests.swift +++ b/Tests/YamsTests/EncoderTests.swift @@ -405,35 +405,6 @@ class EncoderTests: XCTestCase { // swiftlint:disable:this type_body_length // MARK: - Helper Functions - private func _testRoundTrip(of value: T, - with options: YAMLEncoder.Options = .init(), - expectedYAML yamlString: String? = nil, - file: StaticString = #file, - line: UInt = #line) where T: Codable, T: Equatable { - do { - let encoder = YAMLEncoder() - encoder.options = options - let producedYAML = try encoder.encode(value) - - if let expectedYAML = yamlString { - XCTAssertEqual(producedYAML, expectedYAML, "Produced YAML not identical to expected YAML.", - file: (file), line: line) - } - - let decoder = YAMLDecoder() - let decoded = try decoder.decode(T.self, from: producedYAML) - XCTAssertEqual(decoded, value, "\(T.self) did not round-trip to an equal value.", - file: (file), line: line) - - } catch let error as EncodingError { - XCTFail("Failed to encode \(T.self) from YAML by error: \(error)", file: (file), line: line) - } catch let error as DecodingError { - XCTFail("Failed to decode \(T.self) from YAML by error: \(error)", file: (file), line: line) - } catch { - XCTFail("Rout trip test of \(T.self) failed with error: \(error)", file: (file), line: line) - } - } - private func _testDecode(of type: T.Type, from string: String, expectedValue value: T?, @@ -468,6 +439,35 @@ class EncoderTests: XCTestCase { // swiftlint:disable:this type_body_length } } +internal func _testRoundTrip(of value: T, + with options: YAMLEncoder.Options = .init(), + expectedYAML yamlString: String? = nil, + file: StaticString = #file, + line: UInt = #line) where T: Codable, T: Equatable { + do { + let encoder = YAMLEncoder() + encoder.options = options + let producedYAML = try encoder.encode(value) + + if let expectedYAML = yamlString { + XCTAssertEqual(producedYAML, expectedYAML, "Produced YAML not identical to expected YAML.", + file: (file), line: line) + } + + let decoder = YAMLDecoder() + let decoded = try decoder.decode(T.self, from: producedYAML) + XCTAssertEqual(decoded, value, "\(T.self) did not round-trip to an equal value.", + file: (file), line: line) + + } catch let error as EncodingError { + XCTFail("Failed to encode \(T.self) from YAML by error: \(error)", file: (file), line: line) + } catch let error as DecodingError { + XCTFail("Failed to decode \(T.self) from YAML by error: \(error)", file: (file), line: line) + } catch { + XCTFail("Rout trip test of \(T.self) failed with error: \(error)", file: (file), line: line) + } +} + // MARK: - Helper Global Functions public func expectEqual( _ expected: T, _ actual: T, diff --git a/Tests/YamsTests/TagCodingTests.swift b/Tests/YamsTests/TagCodingTests.swift new file mode 100644 index 00000000..f9bc55be --- /dev/null +++ b/Tests/YamsTests/TagCodingTests.swift @@ -0,0 +1,240 @@ +// +// TagCodingTests.swift +// Yams +// +// Created by Adora Lynch on 9/18/24. +// Copyright (c) 2024 Yams. All rights reserved. +// + +import XCTest +import Yams + +class TagCodingTests: XCTestCase { + + /// Test the encoding of a yaml tag using a type that conforms to YamlTagProviding + func testYamlTagProviding_valuePresent() throws { + let simpleStruct = SimpleWithTag(nested: .init(stringValue: "it's a value"), intValue: 52) + + _testRoundTrip(of: simpleStruct, + expectedYAML:""" + ! + nested: + stringValue: it's a value + intValue: 52 + + """ ) // ^ the Yams.Tag is encoded as a yaml tag + } + + /// Test the encoding of a a type that does not conform to YamlTagProviding but none the less declares a coding member with the same name + func testStringTypeTagName_valuePresent() throws { + let simpleStruct = SimpleWithStringTypeTagName(nested: .init(stringValue: "it's a value"), + intValue: 52, + yamlTag: "but typed as a string") + + _testRoundTrip(of: simpleStruct, + expectedYAML:""" + nested: + stringValue: it's a value + intValue: 52 + yamlTag: but typed as a string + + """ ) // ^ the member is _not_ treated as an tag + } + + /// Nothing interesting happens when a type does not conform to YamlTagProviding none the less declares a coding member with the same name but that value is nil + func testStringTypeTagName_valueNotPresent() throws { + let expectedStruct = SimpleWithStringTypeTagName(nested: .init(stringValue: "it's a value"), + intValue: 52, + yamlTag: nil) + _testRoundTrip(of: expectedStruct, + expectedYAML: """ + nested: + stringValue: it's a value + intValue: 52 + + """) + } + + /// This test documents some undesirable behavior, but in an unlikely circumstance. + /// If the decoded type does not conform to YamlTagProviding it can still have a coding key called `yamlTag` + /// If Yams tries to decode such a type AND the document has a nil value for `yamlTag` AND the parent context is a mapping AND that mapping has an actual tag (in the document) + /// THEN Yams wrongly tries to decode the tag as the declared type of key `yamlTag`. + /// If that declared type can be decoded from a scalar string value (like String and RawRepresentable where RawValue == String) then the decoding will actually succeed. + /// Which effectively injects an unexpected value into the decoded type. + func testStringTypeTagName_withTagPresent_valueNil() throws { + let expectedStruct = SimpleWithStringTypeTagName(nested: .init(stringValue: "it's a value"), + intValue: 52, + yamlTag: nil) + let decoder = YAMLDecoder() + let data = """ + ! + nested: + stringValue: it's a value + intValue: 52 + + """.data(using: .utf8)! + + let decodedStruct = try decoder.decode(SimpleWithStringTypeTagName.self, from: data) + + let fixBulletin = "YESS!!! YOU FIXED IT! See \(#file):\(#line) for explanation." + + // begin assertions of known-but-undesirable behavior + XCTAssertNotEqual(decodedStruct, expectedStruct, fixBulletin) // We wish this was equal + XCTAssertEqual(decodedStruct.yamlTag, "An:Actual:Tag", fixBulletin) // we wish .yamlTag was nil + // end assertions of known-but-undesirable behavior + + + // Check the remainder of the properties that the above confusion did not involve + XCTAssertEqual(decodedStruct.nested, expectedStruct.nested) + XCTAssertEqual(decodedStruct.intValue, expectedStruct.intValue) + } +} + +class TagWithAnchorCodingTests: XCTestCase { + + /// If types conform to YamlTagProviding and are Hashable-Equal then HashableAliasingStrategy aliases them + func testEncoderAutoAlias_Hashable_duplicateValue_commonTag() throws { + let simpleStruct = SimpleWithTag(nested: .init(stringValue: "it's a value"), intValue: 52) + let duplicatedStructArray = [simpleStruct, simpleStruct] + + let options = YAMLEncoder.Options(redundancyAliasingStrategy: HashableAliasingStrategy()) + _testRoundTrip(of: duplicatedStructArray, + with: options, + expectedYAML:""" + - &2 ! + nested: + stringValue: it's a value + intValue: 52 + - *2 + + """ ) + } + + /// If types conform to YamlTagProviding and are NOT Hashable-Equal then HashableAliasingStrategy does not alias them + /// even though their members may still be Hashable-Equal and therefor maybe aliased. + func testEncoderAutoAlias_Hashable_uniqueTag() throws { + let differentTypesOneTags = SimplePair(first: SimpleWithTag(nested: .init(stringValue: "it's a value"), intValue: 52), + second: SimpleWithoutTag(nested: .init(stringValue: "it's a value"), intValue: 52)) + + let options = YAMLEncoder.Options(redundancyAliasingStrategy: HashableAliasingStrategy()) + _testRoundTrip(of: differentTypesOneTags, + with: options, + expectedYAML:""" + first: ! + nested: &3 + stringValue: it's a value + intValue: &5 52 + second: + nested: *3 + intValue: *5 + + """ ) + } + + /// If types conform to YamlTagProviding can declare to have the same tag and still be NOT Hashable-Equal then HashableAliasingStrategy does not alias them + /// even though their members may still be Hashable-Equal and therefor maybe aliased. + func testEncoderAutoAlias_Hashable_distinctValues_commonTag() throws { + let differentTypesOneTags = SimplePair(first: SimpleWithTag(nested: .init(stringValue: "it's a value"), intValue: 52), + second: SimpleWithTag2(nested: .init(stringValue: "it's a value"), intValue: 52)) + + let options = YAMLEncoder.Options(redundancyAliasingStrategy: HashableAliasingStrategy()) + _testRoundTrip(of: differentTypesOneTags, + with: options, + expectedYAML:""" + first: ! + nested: &3 + stringValue: it's a value + intValue: &5 52 + second: ! + nested: *3 + intValue: *5 + + """ ) + } + + /// If different types conform to YamlTagProviding they can declare to have the same tag and further, have exactly the same encoded representation. + /// In thisi case StrictEncodableAliasingStrategy will still alias them even though they are encoded and decoded from different types. + func testEncoderAutoAlias_StrictEncodable_distinctValues_commonTag() throws { + let differentTypesOneTags = SimplePair(first: SimpleWithTag(nested: .init(stringValue: "it's a value"), intValue: 52), + second: SimpleWithTag2(nested: .init(stringValue: "it's a value"), intValue: 52)) + + var options = YAMLEncoder.Options() + options.redundancyAliasingStrategy = StrictEncodableAliasingStrategy() + _testRoundTrip(of: differentTypesOneTags, + with: options, + expectedYAML:""" + first: &2 ! + nested: + stringValue: it's a value + intValue: 52 + second: *2 + + """ ) + } + + /// If types conform to YamlTagProviding and YamlAnchorProviding, both are respected. + func testEncoderAutoAlias_Hashable_commonTagAndAnchor() throws { + let simpleStruct = SimpleWithTagAndAnchor(nested: .init(stringValue: "it's a value"), intValue: 52) + let duplicatedStructArray = [simpleStruct, simpleStruct] + + let options = YAMLEncoder.Options(redundancyAliasingStrategy: HashableAliasingStrategy()) + _testRoundTrip(of: duplicatedStructArray, + with: options, + expectedYAML:""" + - &simple-Anchor ! + nested: + stringValue: it's a value + intValue: 52 + - *simple-Anchor + + """ ) + } + + /// A type used to contain values used during testing + private struct SimplePair: Hashable, Codable { + let first: First + let second: Second + } + +} +// MARK: - Types used for Tag encoding tests. + +fileprivate struct NestedStruct: Codable, Hashable { + let stringValue: String +} +fileprivate protocol SimpleProtocol: Codable, Hashable { + var nested: NestedStruct { get } + var intValue: Int { get } +} + +fileprivate struct SimpleWithTag: SimpleProtocol, YamlTagProviding { + let nested: NestedStruct + let intValue: Int + var yamlTag: Tag? = "simple" +} + +fileprivate struct SimpleWithTag2: SimpleProtocol, YamlTagProviding { + let nested: NestedStruct + let intValue: Int + var yamlTag: Tag? = "simple" +} + +fileprivate struct SimpleWithoutTag: SimpleProtocol { + let nested: NestedStruct + let intValue: Int +} + +fileprivate struct SimpleWithStringTypeTagName: SimpleProtocol { + let nested: NestedStruct + let intValue: Int + var yamlTag: String? = "StringTypeTag" +} + +fileprivate struct SimpleWithTagAndAnchor: SimpleProtocol, YamlTagProviding, YamlAnchorProviding { + let nested: NestedStruct + let intValue: Int + var yamlTag: Tag? = "simple:Tag" + var yamlAnchor: Anchor? = "simple-Anchor" +} + + diff --git a/Tests/YamsTests/TagTolerancesTests.swift b/Tests/YamsTests/TagTolerancesTests.swift new file mode 100644 index 00000000..62a73e1a --- /dev/null +++ b/Tests/YamsTests/TagTolerancesTests.swift @@ -0,0 +1,179 @@ +// +// TagTolerancesTests.swift +// Yams +// +// Created by Adora Lynch on 9/18/24. +// Copyright (c) 2024 Yams. All rights reserved. +// + +import XCTest +import Yams + +class TagTolerancesTests: XCTestCase { + + struct Example: Codable, Hashable { + var myCustomTagDeclaration: Tag + var extraneousValue: Int + } + + /// Any type that is Encodable and contains an `Tag`value but with a coding key different from YamlTagProviding + /// will not encode to a yaml tag + /// This may be unexpected + func testTagEncoding_undeclaredBehavior() throws { + let expectedYAML = """ + myCustomTagDeclaration: I-did-it-myyyyy-way + extraneousValue: 3 + + """ + + let value = Example(myCustomTagDeclaration: "I-did-it-myyyyy-way", + extraneousValue: 3) + + let encoder = YAMLEncoder() + let producedYAML = try encoder.encode(value) + XCTAssertEqual(producedYAML, expectedYAML, "Produced YAML not identical to expected YAML.") + } + + /// Any type that is Encodable and contains an `Tag`value with the same coding key as YamlTagProviding + /// will encode to a yaml tag even though the type does not conform to YamlTagProviding + /// This may be unexpected + func testTagEncoding_undeclaredBehavior_7() throws { + struct Example: Codable, Hashable { + var yamlTag: Tag + var extraneousValue: Int + } + let expectedYAML = """ + ! + extraneousValue: 3 + + """ + + let value = Example(yamlTag: "I-did-it-myyyyy-way", + extraneousValue: 3) + + let encoder = YAMLEncoder() + let producedYAML = try encoder.encode(value) + XCTAssertEqual(producedYAML, expectedYAML, "Produced YAML not identical to expected YAML.") + } + + /// Tags are oddly permissive, but some characters do get escaped + /// This may be unexpected + func testTagEncoding_undeclaredBehavior_4() throws { + struct Example: Codable, Hashable, YamlTagProviding { + var yamlTag: Tag? + var extraneousValue: Int + } + + let expectedYAML = """ + ! + extraneousValue: 3 + + """ + + let value = Example(yamlTag: "I-did-it-[]-*-|-!-()way", + extraneousValue: 3) + + let encoder = YAMLEncoder() + let producedYAML = try encoder.encode(value) + XCTAssertEqual(producedYAML, expectedYAML, "Produced YAML not identical to expected YAML.") + } + + /// Any type that is Decodable and contains an `Tag` value but with a coding key different from YamlTagProviding + /// will not decode an tag from the text representation. + /// In this case a key not found error will be thrown during decoding + /// This may be unexpected + func testTagDecoding_undeclaredBehavior_1() throws { + let sourceYAML = """ + ! + extraneousValue: 3 + + """ + let decoder = YAMLDecoder() + XCTAssertThrowsError(try decoder.decode(Example.self, from: sourceYAML)) + // error is ^^ key not found, "myCustomTagDeclaration" + } + + /// Any type that is Decodable and contains an `Tag` value but with a coding key different from YamlTagProviding + /// will not decode an tag from the text representation. + /// This may be unexpected + func testTagDecoding_undeclaredBehavior_6() throws { + struct Example: Codable, Hashable { + var myCustomTagDeclaration: Tag? + var extraneousValue: Int + } + let sourceYAML = """ + ! + extraneousValue: 3 + + """ + + let expectedValue = Example(myCustomTagDeclaration: nil, + extraneousValue: 3) + + let decoder = YAMLDecoder() + let decodedValue = try decoder.decode(Example.self, from: sourceYAML) + XCTAssertEqual(decodedValue, expectedValue, "\(Example.self) did not round-trip to an equal value.") + } + + /// Any type that is Decodable and contains an `Tag` value with the same coding key as YamlTagProviding + /// will decode an tag from the text representatio even though the type does not conform to YamlTagCoding. + /// This may be unexpected + func testTagDecoding_undeclaredBehavior_8() throws { + struct Example: Codable, Hashable { + var yamlTag: Tag? + var extraneousValue: Int + } + let sourceYAML = """ + ! + extraneousValue: 3 + + """ + + let expectedValue = Example(yamlTag: "a-different-tag", + extraneousValue: 3) + + let decoder = YAMLDecoder() + let decodedValue = try decoder.decode(Example.self, from: sourceYAML) + XCTAssertEqual(decodedValue, expectedValue, "\(Example.self) did not round-trip to an equal value.") + } + + /// Any type that is Decodable and contains an `Tag` value but with a coding key different from YamlTagProviding + /// will not decode an tag from the text representation. + /// This is expected behavior, but in a strange situation. + func testTagDecoding_undeclaredBehavior_3() throws { + let sourceYAML = """ + ! + extraneousValue: 3 + myCustomTagDeclaration: deliver-us-from-evil + + """ + let expectedValue = Example(myCustomTagDeclaration: "deliver-us-from-evil", + extraneousValue: 3) + + let decoder = YAMLDecoder() + let decodedValue = try decoder.decode(Example.self, from: sourceYAML) + XCTAssertEqual(decodedValue, expectedValue, "\(Example.self) did not round-trip to an equal value.") + + } + + /// Any type that is Decodable and contains an `Tag` value but with a coding key different from YamlTagProviding + /// will not decode an tag from the text representation. + /// This is expected behavior, but in a strange situation. + func testTagDecoding_undeclaredBehavior_2() throws { + let sourceYAML = """ + ! + extraneousValue: 3 + myCustomTagDeclaration: "deliver us from |()evil" + + """ + + let expectedValue = Example(myCustomTagDeclaration: "deliver us from |()evil", + extraneousValue: 3) + + let decoder = YAMLDecoder() + let decodedValue = try decoder.decode(Example.self, from: sourceYAML) + XCTAssertEqual(decodedValue, expectedValue, "\(Example.self) did not round-trip to an equal value.") + + } + +}