1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83
|
//===----------------------------------------------------------------------===//
//
// This source file is part of the Swift.org open source project
//
// Copyright (c) 2023 Apple Inc. and the Swift project authors
// Licensed under Apache License v2.0 with Runtime Library Exception
//
// See https://swift.org/LICENSE.txt for license information
// See https://swift.org/CONTRIBUTORS.txt for the list of Swift project authors
//
//===----------------------------------------------------------------------===//
struct _BPlistDecodingFormat : PlistDecodingFormat {
typealias Map = BPlistMap
static func container<Key>(keyedBy type: Key.Type, for value: BPlistMap.Value, referencing decoder: _PlistDecoder<_BPlistDecodingFormat>, codingPathNode: _CodingPathNode) throws -> KeyedDecodingContainer<Key> where Key : CodingKey {
switch value {
case let .dict(dict):
let iter = Map.DictionaryIterator.init(iter: dict.makeIterator())
let container = try _PlistKeyedDecodingContainer<Key, Self>(referencing: decoder, codingPathNode: codingPathNode, iterator: iter, count: dict.count)
return KeyedDecodingContainer(container)
case .nativeNull, .sentinelNull:
throw DecodingError.valueNotFound([String: Any].self, DecodingError.Context(codingPath: decoder.codingPath, debugDescription: "Cannot get keyed decoding container -- found null value instead"))
default:
throw DecodingError._typeMismatch(at: decoder.codingPath, expectation: [String : Any].self, reality: value)
}
}
static func unkeyedContainer(for value: BPlistMap.Value, referencing decoder: _PlistDecoder<_BPlistDecodingFormat>, codingPathNode: _CodingPathNode) throws -> UnkeyedDecodingContainer {
switch value {
case let .array(array):
let iter = Map.ArrayIterator(iter: array.makeIterator())
return _PlistUnkeyedDecodingContainer(referencing: decoder, codingPathNode: codingPathNode, iterator: iter, count: array.count)
case .nativeNull, .sentinelNull:
throw DecodingError.valueNotFound([Any].self, DecodingError.Context(codingPath: decoder.codingPath, debugDescription: "Cannot get unkeyed decoding container -- found null value instead"))
default:
throw DecodingError._typeMismatch(at: decoder.codingPath, expectation: [Any].self, reality: value)
}
}
static func valueIsNull(_ mapValue: BPlistMap.Value) -> Bool {
switch mapValue {
case .nativeNull, .sentinelNull: return true
default: return false
}
}
static func unwrapBool(from mapValue: BPlistMap.Value, for codingPathNode: _CodingPathNode, _ additionalKey: (some CodingKey)?) throws -> Bool {
guard case let .boolean(value) = mapValue else {
throw DecodingError._typeMismatch(at: codingPathNode.path(byAppending: additionalKey), expectation: Bool.self, reality: mapValue)
}
return value
}
static func unwrapDate(from mapValue: BPlistMap.Value, in map: BPlistMap, for codingPathNode: _CodingPathNode, _ additionalKey: (some CodingKey)?) throws -> Date {
return try mapValue.dateValue(in: map, for: codingPathNode, additionalKey)
}
static func unwrapData(from mapValue: BPlistMap.Value, in map: BPlistMap, for codingPathNode: _CodingPathNode, _ additionalKey: (some CodingKey)?) throws -> Data {
return try mapValue.dataValue(in: map, for: codingPathNode, additionalKey)
}
static func unwrapString(from mapValue: BPlistMap.Value, in map: BPlistMap, for codingPathNode: _CodingPathNode, _ additionalKey: (some CodingKey)?) throws -> String {
guard case let .string(region, isAscii) = mapValue else {
throw DecodingError._typeMismatch(at: codingPathNode.path(byAppending: additionalKey), expectation: String.self, reality: mapValue)
}
let result = map.withBuffer(for: region) { buffer, _ in
String(bytes: buffer, encoding: isAscii ? .ascii : .utf16BigEndian)
}
guard let result else {
throw DecodingError._dataCorrupted("Unable to read string", for: codingPathNode.appending(additionalKey))
}
return result
}
static func unwrapFloatingPoint<T : BinaryFloatingPoint>(from mapValue: BPlistMap.Value, in map: BPlistMap, for codingPathNode: _CodingPathNode, _ additionalKey: (some CodingKey)?) throws -> T {
try mapValue.realValue(in: map, as: T.self, for: codingPathNode, additionalKey)
}
static func unwrapFixedWidthInteger<T>(from mapValue: BPlistMap.Value, in map: BPlistMap, for codingPathNode: _CodingPathNode, _ additionalKey: (some CodingKey)?) throws -> T where T : FixedWidthInteger {
try mapValue.integerValue(in: map, as: T.self, for: codingPathNode, additionalKey)
}
}
|