mirror of
https://github.com/apple/swift-foundation.git
synced 2025-05-21 21:09:57 +08:00
large integers are valid JSON, and we accept loss of precision when converting them to Double.
1585 lines
69 KiB
Swift
1585 lines
69 KiB
Swift
//===----------------------------------------------------------------------===//
|
|
//
|
|
// This source file is part of the Swift.org open source project
|
|
//
|
|
// Copyright (c) 2014 - 2017 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
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#if canImport(Darwin)
|
|
import Darwin
|
|
#elseif canImport(Glibc)
|
|
import Glibc
|
|
#endif
|
|
|
|
#if FOUNDATION_FRAMEWORK
|
|
@_implementationOnly import _CShims
|
|
#else
|
|
package import _CShims
|
|
#endif
|
|
|
|
/// A marker protocol used to determine whether a value is a `String`-keyed `Dictionary`
|
|
/// containing `Decodable` values (in which case it should be exempt from key conversion strategies).
|
|
///
|
|
/// The marker protocol also provides access to the type of the `Decodable` values,
|
|
/// which is needed for the implementation of the key conversion strategy exemption.
|
|
private protocol _JSONStringDictionaryDecodableMarker {
|
|
static var elementType: Decodable.Type { get }
|
|
}
|
|
|
|
extension Dictionary : _JSONStringDictionaryDecodableMarker where Key == String, Value: Decodable {
|
|
static var elementType: Decodable.Type { return Value.self }
|
|
}
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
// JSON Decoder
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
/// `JSONDecoder` facilitates the decoding of JSON into semantic `Decodable` types.
|
|
// NOTE: older overlays had Foundation.JSONDecoder as the ObjC name.
|
|
// The two must coexist, so it was renamed. The old name must not be
|
|
// used in the new runtime. _TtC10Foundation13__JSONDecoder is the
|
|
// mangled name for Foundation.__JSONDecoder.
|
|
#if FOUNDATION_FRAMEWORK
|
|
@_objcRuntimeName(_TtC10Foundation13__JSONDecoder)
|
|
#endif
|
|
@available(macOS 10.10, iOS 8.0, watchOS 2.0, tvOS 9.0, *)
|
|
open class JSONDecoder {
|
|
// MARK: Options
|
|
|
|
/// The strategy to use for decoding `Date` values.
|
|
public enum DateDecodingStrategy : Sendable {
|
|
/// Defer to `Date` for decoding. This is the default strategy.
|
|
case deferredToDate
|
|
|
|
/// Decode the `Date` as a UNIX timestamp from a JSON number.
|
|
case secondsSince1970
|
|
|
|
/// Decode the `Date` as UNIX millisecond timestamp from a JSON number.
|
|
case millisecondsSince1970
|
|
|
|
#if FOUNDATION_FRAMEWORK && !NO_FORMATTERS
|
|
// TODO: Reenable once DateFormatStyle has been moved: https://github.com/apple/swift-foundation/issues/46
|
|
/// Decode the `Date` as an ISO-8601-formatted string (in RFC 3339 format).
|
|
@available(macOS 10.12, iOS 10.0, watchOS 3.0, tvOS 10.0, *)
|
|
case iso8601
|
|
|
|
/// Decode the `Date` as a string parsed by the given formatter.
|
|
case formatted(DateFormatter)
|
|
#endif // FOUNDATION_FRAMEWORK
|
|
|
|
/// Decode the `Date` as a custom value decoded by the given closure.
|
|
@preconcurrency
|
|
case custom(@Sendable (_ decoder: Decoder) throws -> Date)
|
|
}
|
|
|
|
/// The strategy to use for decoding `Data` values.
|
|
public enum DataDecodingStrategy : Sendable {
|
|
/// Defer to `Data` for decoding.
|
|
case deferredToData
|
|
|
|
/// Decode the `Data` from a Base64-encoded string. This is the default strategy.
|
|
case base64
|
|
|
|
/// Decode the `Data` as a custom value decoded by the given closure.
|
|
@preconcurrency
|
|
case custom(@Sendable (_ decoder: Decoder) throws -> Data)
|
|
}
|
|
|
|
/// The strategy to use for non-JSON-conforming floating-point values (IEEE 754 infinity and NaN).
|
|
public enum NonConformingFloatDecodingStrategy : Sendable {
|
|
/// Throw upon encountering non-conforming values. This is the default strategy.
|
|
case `throw`
|
|
|
|
/// Decode the values from the given representation strings.
|
|
case convertFromString(positiveInfinity: String, negativeInfinity: String, nan: String)
|
|
}
|
|
|
|
/// The strategy to use for automatically changing the value of keys before decoding.
|
|
public enum KeyDecodingStrategy : Sendable {
|
|
/// Use the keys specified by each type. This is the default strategy.
|
|
case useDefaultKeys
|
|
|
|
/// Convert from "snake_case_keys" to "camelCaseKeys" before attempting to match a key with the one specified by each type.
|
|
///
|
|
/// The conversion to upper case uses `Locale.system`, also known as the ICU "root" locale. This means the result is consistent regardless of the current user's locale and language preferences.
|
|
///
|
|
/// Converting from snake case to camel case:
|
|
/// 1. Capitalizes the word starting after each `_`
|
|
/// 2. Removes all `_`
|
|
/// 3. Preserves starting and ending `_` (as these are often used to indicate private variables or other metadata).
|
|
/// For example, `one_two_three` becomes `oneTwoThree`. `_one_two_three_` becomes `_oneTwoThree_`.
|
|
///
|
|
/// - Note: Using a key decoding strategy has a nominal performance cost, as each string key has to be inspected for the `_` character.
|
|
case convertFromSnakeCase
|
|
|
|
/// Provide a custom conversion from the key in the encoded JSON to the keys specified by the decoded types.
|
|
/// The full path to the current decoding position is provided for context (in case you need to locate this key within the payload). The returned key is used in place of the last component in the coding path before decoding.
|
|
/// If the result of the conversion is a duplicate key, then only one value will be present in the container for the type to decode from.
|
|
@preconcurrency
|
|
case custom(@Sendable (_ codingPath: [CodingKey]) -> CodingKey)
|
|
|
|
fileprivate static func _convertFromSnakeCase(_ stringKey: String) -> String {
|
|
guard !stringKey.isEmpty else { return stringKey }
|
|
|
|
// Find the first non-underscore character
|
|
guard let firstNonUnderscore = stringKey.firstIndex(where: { $0 != "_" }) else {
|
|
// Reached the end without finding an _
|
|
return stringKey
|
|
}
|
|
|
|
// Find the last non-underscore character
|
|
var lastNonUnderscore = stringKey.index(before: stringKey.endIndex)
|
|
while lastNonUnderscore > firstNonUnderscore && stringKey[lastNonUnderscore] == "_" {
|
|
stringKey.formIndex(before: &lastNonUnderscore)
|
|
}
|
|
|
|
let keyRange = firstNonUnderscore...lastNonUnderscore
|
|
let leadingUnderscoreRange = stringKey.startIndex..<firstNonUnderscore
|
|
let trailingUnderscoreRange = stringKey.index(after: lastNonUnderscore)..<stringKey.endIndex
|
|
|
|
let components = stringKey[keyRange].split(separator: "_")
|
|
let joinedString: String
|
|
if components.count == 1 {
|
|
// No underscores in key, leave the word as is - maybe already camel cased
|
|
joinedString = String(stringKey[keyRange])
|
|
} else {
|
|
joinedString = ([components[0].lowercased()] + components[1...].map { $0.capitalized }).joined()
|
|
}
|
|
|
|
// Do a cheap isEmpty check before creating and appending potentially empty strings
|
|
let result: String
|
|
if (leadingUnderscoreRange.isEmpty && trailingUnderscoreRange.isEmpty) {
|
|
result = joinedString
|
|
} else if (!leadingUnderscoreRange.isEmpty && !trailingUnderscoreRange.isEmpty) {
|
|
// Both leading and trailing underscores
|
|
result = String(stringKey[leadingUnderscoreRange]) + joinedString + String(stringKey[trailingUnderscoreRange])
|
|
} else if (!leadingUnderscoreRange.isEmpty) {
|
|
// Just leading
|
|
result = String(stringKey[leadingUnderscoreRange]) + joinedString
|
|
} else {
|
|
// Just trailing
|
|
result = joinedString + String(stringKey[trailingUnderscoreRange])
|
|
}
|
|
return result
|
|
}
|
|
}
|
|
|
|
/// The strategy to use in decoding dates. Defaults to `.deferredToDate`.
|
|
open var dateDecodingStrategy: DateDecodingStrategy {
|
|
get {
|
|
optionsLock.lock()
|
|
defer { optionsLock.unlock() }
|
|
return options.dateDecodingStrategy
|
|
}
|
|
_modify {
|
|
optionsLock.lock()
|
|
var value = options.dateDecodingStrategy
|
|
defer {
|
|
options.dateDecodingStrategy = value
|
|
optionsLock.unlock()
|
|
}
|
|
yield &value
|
|
}
|
|
set {
|
|
optionsLock.lock()
|
|
defer { optionsLock.unlock() }
|
|
options.dateDecodingStrategy = newValue
|
|
}
|
|
}
|
|
|
|
/// The strategy to use in decoding binary data. Defaults to `.base64`.
|
|
open var dataDecodingStrategy: DataDecodingStrategy {
|
|
get {
|
|
optionsLock.lock()
|
|
defer { optionsLock.unlock() }
|
|
return options.dataDecodingStrategy
|
|
}
|
|
_modify {
|
|
optionsLock.lock()
|
|
var value = options.dataDecodingStrategy
|
|
defer {
|
|
options.dataDecodingStrategy = value
|
|
optionsLock.unlock()
|
|
}
|
|
yield &value
|
|
}
|
|
set {
|
|
optionsLock.lock()
|
|
defer { optionsLock.unlock() }
|
|
options.dataDecodingStrategy = newValue
|
|
}
|
|
}
|
|
|
|
/// The strategy to use in decoding non-conforming numbers. Defaults to `.throw`.
|
|
open var nonConformingFloatDecodingStrategy: NonConformingFloatDecodingStrategy {
|
|
get {
|
|
optionsLock.lock()
|
|
defer { optionsLock.unlock() }
|
|
return options.nonConformingFloatDecodingStrategy
|
|
}
|
|
_modify {
|
|
optionsLock.lock()
|
|
var value = options.nonConformingFloatDecodingStrategy
|
|
defer {
|
|
options.nonConformingFloatDecodingStrategy = value
|
|
optionsLock.unlock()
|
|
}
|
|
yield &value
|
|
}
|
|
set {
|
|
optionsLock.lock()
|
|
defer { optionsLock.unlock() }
|
|
options.nonConformingFloatDecodingStrategy = newValue
|
|
}
|
|
}
|
|
|
|
/// The strategy to use for decoding keys. Defaults to `.useDefaultKeys`.
|
|
open var keyDecodingStrategy: KeyDecodingStrategy {
|
|
get {
|
|
optionsLock.lock()
|
|
defer { optionsLock.unlock() }
|
|
return options.keyDecodingStrategy
|
|
}
|
|
_modify {
|
|
optionsLock.lock()
|
|
var value = options.keyDecodingStrategy
|
|
defer {
|
|
options.keyDecodingStrategy = value
|
|
optionsLock.unlock()
|
|
}
|
|
yield &value
|
|
}
|
|
set {
|
|
optionsLock.lock()
|
|
defer { optionsLock.unlock() }
|
|
options.keyDecodingStrategy = newValue
|
|
}
|
|
}
|
|
|
|
/// Contextual user-provided information for use during decoding.
|
|
open var userInfo: [CodingUserInfoKey : Any] {
|
|
get {
|
|
optionsLock.lock()
|
|
defer { optionsLock.unlock() }
|
|
return options.userInfo
|
|
}
|
|
_modify {
|
|
optionsLock.lock()
|
|
var value = options.userInfo
|
|
defer {
|
|
options.userInfo = value
|
|
optionsLock.unlock()
|
|
}
|
|
yield &value
|
|
}
|
|
set {
|
|
optionsLock.lock()
|
|
defer { optionsLock.unlock() }
|
|
options.userInfo = newValue
|
|
}
|
|
}
|
|
|
|
/// Set to `true` to allow parsing of JSON5. Defaults to `false`.
|
|
@available(macOS 12.0, iOS 15.0, tvOS 15.0, watchOS 8.0, *)
|
|
open var allowsJSON5: Bool {
|
|
get {
|
|
options.json5
|
|
}
|
|
set {
|
|
options.json5 = newValue
|
|
}
|
|
}
|
|
|
|
private let assumesTopLevelDictionaryKey = CodingUserInfoKey(rawValue: "_NSAssumesTopLevelDictionaryJSON5")!
|
|
|
|
/// Set to `true` to assume the data is a top level Dictionary (no surrounding "{ }" required). Defaults to `false`. Compatible with both JSON5 and non-JSON5 mode.
|
|
@available(macOS 12.0, iOS 15.0, tvOS 15.0, watchOS 8.0, *)
|
|
open var assumesTopLevelDictionary: Bool {
|
|
get {
|
|
userInfo[assumesTopLevelDictionaryKey] as? Bool ?? false
|
|
}
|
|
set {
|
|
userInfo[assumesTopLevelDictionaryKey] = newValue
|
|
}
|
|
}
|
|
|
|
/// Options set on the top-level encoder to pass down the decoding hierarchy.
|
|
fileprivate struct _Options {
|
|
var dateDecodingStrategy: DateDecodingStrategy = .deferredToDate
|
|
var dataDecodingStrategy: DataDecodingStrategy = .base64
|
|
var nonConformingFloatDecodingStrategy: NonConformingFloatDecodingStrategy = .throw
|
|
var keyDecodingStrategy: KeyDecodingStrategy = .useDefaultKeys
|
|
var userInfo: [CodingUserInfoKey : Any] = [:]
|
|
var json5: Bool = false
|
|
}
|
|
|
|
/// The options set on the top-level decoder.
|
|
fileprivate var options = _Options()
|
|
fileprivate let optionsLock = LockedState<Void>()
|
|
|
|
// MARK: - Constructing a JSON Decoder
|
|
|
|
/// Initializes `self` with default strategies.
|
|
public init() {}
|
|
|
|
private var scannerOptions : JSONScanner.Options {
|
|
.init(assumesTopLevelDictionary: self.assumesTopLevelDictionary)
|
|
}
|
|
|
|
private var json5ScannerOptions : JSON5Scanner.Options {
|
|
.init(assumesTopLevelDictionary: self.assumesTopLevelDictionary)
|
|
}
|
|
|
|
// MARK: - Decoding Values
|
|
|
|
/// Decodes a top-level value of the given type from the given JSON representation.
|
|
///
|
|
/// - parameter type: The type of the value to decode.
|
|
/// - parameter data: The data to decode from.
|
|
/// - returns: A value of the requested type.
|
|
/// - throws: `DecodingError.dataCorrupted` if values requested from the payload are corrupted, or if the given data is not valid JSON.
|
|
/// - throws: An error if any value throws an error during decoding.
|
|
open func decode<T: Decodable>(_ type: T.Type, from data: Data) throws -> T {
|
|
try _decode({
|
|
try $0.unwrap($1, as: type, for: .root, _JSONKey?.none)
|
|
}, from: data)
|
|
}
|
|
|
|
@available(FoundationPreview 0.1, *)
|
|
open func decode<T: DecodableWithConfiguration>(_ type: T.Type, from data: Data, configuration: T.DecodingConfiguration) throws -> T {
|
|
try _decode({
|
|
try $0.unwrap($1, as: type, configuration: configuration, for: .root, _JSONKey?.none)
|
|
}, from: data)
|
|
}
|
|
|
|
@available(FoundationPreview 0.1, *)
|
|
open func decode<T, C>(_ type: T.Type, from data: Data, configuration: C.Type) throws -> T where T : DecodableWithConfiguration, C : DecodingConfigurationProviding, T.DecodingConfiguration == C.DecodingConfiguration {
|
|
try decode(type, from: data, configuration: C.decodingConfiguration)
|
|
}
|
|
|
|
private func _decode<T>(_ unwrap: (JSONDecoderImpl, JSONMap.Value) throws -> T, from data: Data) throws -> T {
|
|
do {
|
|
return try Self.withUTF8Representation(of: data) { utf8Buffer in
|
|
|
|
var impl: JSONDecoderImpl
|
|
let topValue: JSONMap.Value
|
|
do {
|
|
// JSON5 is implemented with a separate scanner to allow regular JSON scanning to achieve higher performance without compromising for `allowsJSON5` checks throughout.
|
|
// Since the resulting JSONMap is identical, the decoder implementation is mostly shared between the two, with only a few branches to handle different methods of parsing strings and numbers. Strings and numbers are not completely parsed until decoding time.
|
|
let map: JSONMap
|
|
if allowsJSON5 {
|
|
var scanner = JSON5Scanner(bytes: utf8Buffer, options: self.json5ScannerOptions)
|
|
map = try scanner.scan()
|
|
} else {
|
|
var scanner = JSONScanner(bytes: utf8Buffer, options: self.scannerOptions)
|
|
map = try scanner.scan()
|
|
}
|
|
topValue = map.loadValue(at: 0)!
|
|
impl = JSONDecoderImpl(userInfo: self.userInfo, from: map, codingPathNode: .root, options: self.options)
|
|
}
|
|
impl.push(value: topValue) // This is something the old implementation did and apps started relying on. Weird.
|
|
let result = try unwrap(impl, topValue)
|
|
let uniquelyReferenced = isKnownUniquelyReferenced(&impl)
|
|
impl.takeOwnershipOfBackingDataIfNeeded(selfIsUniquelyReferenced: uniquelyReferenced)
|
|
return result
|
|
}
|
|
} catch let error as JSONError {
|
|
#if FOUNDATION_FRAMEWORK
|
|
let underlyingError: Error? = error.nsError
|
|
#else
|
|
let underlyingError: Error? = nil
|
|
#endif
|
|
throw DecodingError.dataCorrupted(DecodingError.Context(codingPath: [], debugDescription: "The given data was not valid JSON.", underlyingError: underlyingError))
|
|
} catch {
|
|
throw error
|
|
}
|
|
}
|
|
|
|
// Input: Data of any encoding specified by RFC4627 section 3, with or without BOM.
|
|
// Output: The closure is invoked with a UInt8 buffer containing the valid UTF-8 representation. If the input contained a BOM, that BOM will be excluded in the resulting buffer.
|
|
// If the input cannot be fully decoded by the detected encoding or cannot be converted to UTF-8, the function will throw a JSONError.cannotConvertEntireInputDataToUTF8 error.
|
|
// If the input is detected to already be UTF-8, the Data's buffer will be passed through without copying.
|
|
static func withUTF8Representation<T>(of jsonData: Data, _ closure: (BufferView<UInt8>) throws -> T ) throws -> T {
|
|
return try jsonData.withBufferView {
|
|
[length = jsonData.count] bytes in
|
|
assert(bytes.count == length)
|
|
// RFC4627 section 3
|
|
// The first two characters of a JSON text will always be ASCII. We can determine encoding by looking at the first four bytes.
|
|
let byte0 = (length > 0) ? bytes[uncheckedOffset: 0] : nil
|
|
let byte1 = (length > 1) ? bytes[uncheckedOffset: 1] : nil
|
|
let byte2 = (length > 2) ? bytes[uncheckedOffset: 2] : nil
|
|
let byte3 = (length > 3) ? bytes[uncheckedOffset: 3] : nil
|
|
|
|
// Check for explicit BOM first, then check the first two bytes. Note that if there is a BOM, we have to create our string without it.
|
|
// This isn't strictly part of the JSON spec but it's useful to do anyway.
|
|
let sourceEncoding : String._Encoding
|
|
let bomLength : Int
|
|
switch (byte0, byte1, byte2, byte3) {
|
|
case (0, 0, 0xFE, 0xFF):
|
|
sourceEncoding = .utf32BigEndian
|
|
bomLength = 4
|
|
case (0xFE, 0xFF, 0, 0):
|
|
sourceEncoding = .utf32LittleEndian
|
|
bomLength = 4
|
|
case (0xFE, 0xFF, _, _):
|
|
sourceEncoding = .utf16BigEndian
|
|
bomLength = 2
|
|
case (0xFF, 0xFE, _, _):
|
|
sourceEncoding = .utf16LittleEndian
|
|
bomLength = 2
|
|
case (0xEF, 0xBB, 0xBF, _):
|
|
sourceEncoding = .utf8
|
|
bomLength = 3
|
|
case let (0, 0, 0, .some(nz)) where nz != 0:
|
|
sourceEncoding = .utf32BigEndian
|
|
bomLength = 0
|
|
case let (0, .some(nz1), 0, .some(nz2)) where nz1 != 0 && nz2 != 0:
|
|
sourceEncoding = .utf16BigEndian
|
|
bomLength = 0
|
|
case let (.some(nz), 0, 0, 0) where nz != 0:
|
|
sourceEncoding = .utf32LittleEndian
|
|
bomLength = 0
|
|
case let (.some(nz1), 0, .some(nz2), 0) where nz1 != 0 && nz2 != 0:
|
|
sourceEncoding = .utf16LittleEndian
|
|
bomLength = 0
|
|
|
|
// These cases technically aren't specified by RFC4627, since it only covers cases where the input has at least 4 octets. However, when parsing JSON with fragments allowed, it's possible to have a valid UTF-16 input that is a single digit, which is 2 octets. To properly support these inputs, we'll extend the pattern described above for 4 octets of UTF-16.
|
|
case let (0, .some(nz), nil, nil) where nz != 0:
|
|
sourceEncoding = .utf16BigEndian
|
|
bomLength = 0
|
|
case let (.some(nz), 0, nil, nil) where nz != 0:
|
|
sourceEncoding = .utf16LittleEndian
|
|
bomLength = 0
|
|
|
|
default:
|
|
sourceEncoding = .utf8
|
|
bomLength = 0
|
|
}
|
|
let postBOMBuffer = bytes.dropFirst(bomLength)
|
|
if sourceEncoding == .utf8 {
|
|
return try closure(postBOMBuffer)
|
|
} else {
|
|
guard var string = String(bytes: postBOMBuffer, encoding: sourceEncoding) else {
|
|
throw JSONError.cannotConvertEntireInputDataToUTF8
|
|
}
|
|
return try string.withUTF8 {
|
|
// String never passes an empty buffer with a `nil` `baseAddress`.
|
|
try closure(BufferView(unsafeBufferPointer: $0)!)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// MARK: - JSONDecoderImpl
|
|
|
|
// NOTE: older overlays called this class _JSONDecoder. The two must
|
|
// coexist without a conflicting ObjC class name, so it was renamed.
|
|
// The old name must not be used in the new runtime.
|
|
fileprivate class JSONDecoderImpl {
|
|
var values: [JSONMap.Value] = []
|
|
let userInfo: [CodingUserInfoKey: Any]
|
|
var jsonMap: JSONMap
|
|
let options: JSONDecoder._Options
|
|
|
|
var codingPathNode: _JSONCodingPathNode
|
|
public var codingPath: [CodingKey] {
|
|
codingPathNode.path
|
|
}
|
|
|
|
var topValue : JSONMap.Value { self.values.last! }
|
|
func push(value: __owned JSONMap.Value) {
|
|
self.values.append(value)
|
|
}
|
|
func popValue() {
|
|
self.values.removeLast()
|
|
}
|
|
|
|
init(userInfo: [CodingUserInfoKey: Any], from map: JSONMap, codingPathNode: _JSONCodingPathNode, options: JSONDecoder._Options) {
|
|
self.userInfo = userInfo
|
|
self.codingPathNode = codingPathNode
|
|
self.jsonMap = map
|
|
self.options = options
|
|
}
|
|
|
|
@inline(__always)
|
|
func withBuffer<T>(for region: JSONMap.Region, perform closure: (_ jsonBytes: BufferView<UInt8>, _ fullSource: BufferView<UInt8>) throws -> T) rethrows -> T {
|
|
try jsonMap.withBuffer(for: region, perform: closure)
|
|
}
|
|
|
|
// This JSONDecoderImpl may have multiple references if an init(from: Decoder) implementation allows the Decoder (this object) to escape, or if a container escapes.
|
|
// The JSONMap might have multiple references if a superDecoder, which creates a different JSONDecoderImpl instance but references the same JSONMap, is allowed to escape.
|
|
// In either case, we need to copy-in the input buffer since it's about to go out of scope.
|
|
func takeOwnershipOfBackingDataIfNeeded(selfIsUniquelyReferenced: Bool) {
|
|
if !selfIsUniquelyReferenced || !isKnownUniquelyReferenced(&jsonMap) {
|
|
jsonMap.copyInBuffer()
|
|
}
|
|
}
|
|
}
|
|
|
|
extension JSONDecoderImpl: Decoder {
|
|
func container<Key: CodingKey>(keyedBy _: Key.Type) throws -> KeyedDecodingContainer<Key> {
|
|
switch topValue {
|
|
case let .object(region):
|
|
let container = try KeyedContainer<Key>(
|
|
impl: self,
|
|
codingPathNode: codingPathNode,
|
|
region: region
|
|
)
|
|
return KeyedDecodingContainer(container)
|
|
case .null:
|
|
throw DecodingError.valueNotFound([String: Any].self, DecodingError.Context(
|
|
codingPath: self.codingPath,
|
|
debugDescription: "Cannot get keyed decoding container -- found null value instead"
|
|
))
|
|
default:
|
|
throw DecodingError.typeMismatch([String: Any].self, DecodingError.Context(
|
|
codingPath: self.codingPath,
|
|
debugDescription: "Expected to decode \([String: Any].self) but found \(topValue.debugDataTypeDescription) instead."
|
|
))
|
|
}
|
|
}
|
|
|
|
func unkeyedContainer() throws -> UnkeyedDecodingContainer {
|
|
switch topValue {
|
|
case let .array(region):
|
|
return UnkeyedContainer(
|
|
impl: self,
|
|
codingPathNode: codingPathNode,
|
|
region: region
|
|
)
|
|
case .null:
|
|
throw DecodingError.valueNotFound([Any].self, DecodingError.Context(
|
|
codingPath: self.codingPath,
|
|
debugDescription: "Cannot get unkeyed decoding container -- found null value instead"
|
|
))
|
|
default:
|
|
throw DecodingError.typeMismatch([Any].self, DecodingError.Context(
|
|
codingPath: self.codingPath,
|
|
debugDescription: "Expected to decode \([Any].self) but found \(topValue.debugDataTypeDescription) instead."
|
|
))
|
|
}
|
|
}
|
|
|
|
func singleValueContainer() throws -> SingleValueDecodingContainer {
|
|
return self
|
|
}
|
|
|
|
// MARK: Special case handling
|
|
|
|
@inline(__always)
|
|
func checkNotNull<T>(_ value: JSONMap.Value, expectedType: T.Type, for codingPathNode: _JSONCodingPathNode, _ additionalKey: (some CodingKey)? = nil) throws {
|
|
if case .null = value {
|
|
throw DecodingError.valueNotFound(expectedType, DecodingError.Context(
|
|
codingPath: codingPathNode.path(with: additionalKey),
|
|
debugDescription: "Cannot get unkeyed decoding container -- found null value instead"
|
|
))
|
|
}
|
|
}
|
|
|
|
// Instead of creating a new JSONDecoderImpl for passing to methods that take Decoder arguments, wrap the access in this method, which temporarily mutates this JSONDecoderImpl instance with the nested value and its coding path.
|
|
@inline(__always)
|
|
func with<T>(value: JSONMap.Value, path: _JSONCodingPathNode?, perform closure: () throws -> T) rethrows -> T {
|
|
let oldPath = self.codingPathNode
|
|
if let path {
|
|
self.codingPathNode = path
|
|
}
|
|
self.push(value: value)
|
|
|
|
defer {
|
|
if path != nil {
|
|
self.codingPathNode = oldPath
|
|
}
|
|
self.popValue()
|
|
}
|
|
|
|
return try closure()
|
|
}
|
|
|
|
func unwrap<T: Decodable>(_ mapValue: JSONMap.Value, as type: T.Type, for codingPathNode: _JSONCodingPathNode, _ additionalKey: (some CodingKey)? = nil) throws -> T {
|
|
if type == Date.self {
|
|
return try self.unwrapDate(from: mapValue, for: codingPathNode, additionalKey) as! T
|
|
}
|
|
if type == Data.self {
|
|
return try self.unwrapData(from: mapValue, for: codingPathNode, additionalKey) as! T
|
|
}
|
|
#if FOUNDATION_FRAMEWORK // TODO: Reenable once URL and Decimal are moved
|
|
if type == URL.self {
|
|
return try self.unwrapURL(from: mapValue, for: codingPathNode, additionalKey) as! T
|
|
}
|
|
if type == Decimal.self {
|
|
return try self.unwrapDecimal(from: mapValue, for: codingPathNode, additionalKey) as! T
|
|
}
|
|
#endif// FOUNDATION_FRAMEWORK
|
|
if T.self is _JSONStringDictionaryDecodableMarker.Type {
|
|
return try self.unwrapDictionary(from: mapValue, as: type, for: codingPathNode, additionalKey)
|
|
}
|
|
|
|
return try self.with(value: mapValue, path: codingPathNode.pushing(additionalKey)) {
|
|
try type.init(from: self)
|
|
}
|
|
}
|
|
|
|
func unwrap<T: DecodableWithConfiguration>(_ mapValue: JSONMap.Value, as type: T.Type, configuration: T.DecodingConfiguration, for codingPathNode: _JSONCodingPathNode, _ additionalKey: (some CodingKey)? = nil) throws -> T {
|
|
try self.with(value: mapValue, path: codingPathNode.pushing(additionalKey)) {
|
|
try type.init(from: self, configuration: configuration)
|
|
}
|
|
}
|
|
|
|
private func unwrapDate<K: CodingKey>(from mapValue: JSONMap.Value, for codingPathNode: _JSONCodingPathNode, _ additionalKey: K? = nil) throws -> Date {
|
|
try checkNotNull(mapValue, expectedType: Date.self, for: codingPathNode, additionalKey)
|
|
|
|
switch self.options.dateDecodingStrategy {
|
|
case .deferredToDate:
|
|
return try self.with(value: mapValue, path: codingPathNode.pushing(additionalKey)) {
|
|
try Date(from: self)
|
|
}
|
|
|
|
case .secondsSince1970:
|
|
let double = try self.unwrapFloatingPoint(from: mapValue, as: Double.self, for: codingPathNode, additionalKey)
|
|
return Date(timeIntervalSince1970: double)
|
|
|
|
case .millisecondsSince1970:
|
|
let double = try self.unwrapFloatingPoint(from: mapValue, as: Double.self, for: codingPathNode, additionalKey)
|
|
return Date(timeIntervalSince1970: double / 1000.0)
|
|
#if FOUNDATION_FRAMEWORK && !NO_FORMATTERS
|
|
// TODO: Reenable once DateFormatStyle has been moved
|
|
case .iso8601:
|
|
let string = try self.unwrapString(from: mapValue, for: codingPathNode, additionalKey)
|
|
guard let date = try? Date.ISO8601FormatStyle().parse(string) else {
|
|
throw DecodingError.dataCorrupted(DecodingError.Context(codingPath: self.codingPath, debugDescription: "Expected date string to be ISO8601-formatted."))
|
|
}
|
|
return date
|
|
|
|
case .formatted(let formatter):
|
|
let string = try self.unwrapString(from: mapValue, for: codingPathNode, additionalKey)
|
|
guard let date = formatter.date(from: string) else {
|
|
throw DecodingError.dataCorrupted(DecodingError.Context(codingPath: codingPathNode.path(with: additionalKey), debugDescription: "Date string does not match format expected by formatter."))
|
|
}
|
|
return date
|
|
#endif // FOUNDATION_FRAMEWORK
|
|
case .custom(let closure):
|
|
return try self.with(value: mapValue, path: codingPathNode.pushing(additionalKey)) {
|
|
try closure(self)
|
|
}
|
|
}
|
|
}
|
|
|
|
private func unwrapData(from mapValue: JSONMap.Value, for codingPathNode: _JSONCodingPathNode, _ additionalKey: (some CodingKey)? = nil) throws -> Data {
|
|
try checkNotNull(mapValue, expectedType: Data.self, for: codingPathNode, additionalKey)
|
|
|
|
switch self.options.dataDecodingStrategy {
|
|
case .deferredToData:
|
|
return try self.with(value: mapValue, path: codingPathNode.pushing(additionalKey)) {
|
|
try Data(from: self)
|
|
}
|
|
|
|
case .base64:
|
|
let string = try self.unwrapString(from: mapValue, for: codingPathNode, additionalKey)
|
|
guard let data = Data(base64Encoded: string) else {
|
|
throw DecodingError.dataCorrupted(DecodingError.Context(codingPath: codingPathNode.path(with: additionalKey), debugDescription: "Encountered Data is not valid Base64."))
|
|
}
|
|
|
|
return data
|
|
|
|
case .custom(let closure):
|
|
return try self.with(value: mapValue, path: codingPathNode.pushing(additionalKey)) {
|
|
try closure(self)
|
|
}
|
|
}
|
|
}
|
|
|
|
#if FOUNDATION_FRAMEWORK // TODO: Reenable once URL and Decimal has been moved
|
|
private func unwrapURL(from mapValue: JSONMap.Value, for codingPathNode: _JSONCodingPathNode, _ additionalKey: (some CodingKey)? = nil) throws -> URL {
|
|
try checkNotNull(mapValue, expectedType: URL.self, for: codingPathNode, additionalKey)
|
|
|
|
let string = try self.unwrapString(from: mapValue, for: codingPathNode, additionalKey)
|
|
guard let url = URL(string: string) else {
|
|
throw DecodingError.dataCorrupted(DecodingError.Context(codingPath: codingPathNode.path(with: additionalKey),
|
|
debugDescription: "Invalid URL string."))
|
|
}
|
|
return url
|
|
}
|
|
|
|
private func unwrapDecimal(from mapValue: JSONMap.Value, for codingPathNode: _JSONCodingPathNode, _ additionalKey: (some CodingKey)? = nil) throws -> Decimal {
|
|
try checkNotNull(mapValue, expectedType: Decimal.self, for: codingPathNode, additionalKey)
|
|
|
|
guard case .number(let region, let hasExponent) = mapValue else {
|
|
throw DecodingError.typeMismatch(Decimal.self, DecodingError.Context(codingPath: codingPathNode.path(with: additionalKey), debugDescription: ""))
|
|
}
|
|
|
|
return try withBuffer(for: region) { numberBuffer, fullSource in
|
|
if options.json5 {
|
|
let (digitsStartPtr, isHex, isSpecialJSON5DoubleValue) = try JSON5Scanner.prevalidateJSONNumber(from: numberBuffer, fullSource: fullSource)
|
|
|
|
// Use our integer parsers for hex data, because the underlying strtod() implementation of T(prevalidatedBuffer:) is too permissive (e.g. it accepts decimals and 'p' exponents) which otherwise would require prevalidation of the entire string before calling it.
|
|
if isHex {
|
|
if numberBuffer.first! == UInt8(ascii: "-") {
|
|
guard let int = Int64(prevalidatedJSON5Buffer: numberBuffer, isHex: isHex), let decimal = Decimal(exactly: int) else {
|
|
throw JSONError.numberIsNotRepresentableInSwift(parsed: String(decoding: numberBuffer, as: UTF8.self))
|
|
}
|
|
return decimal
|
|
} else {
|
|
guard let int = UInt64(prevalidatedJSON5Buffer: numberBuffer, isHex: isHex), let decimal = Decimal(exactly: int) else {
|
|
throw JSONError.numberIsNotRepresentableInSwift(parsed: String(decoding: numberBuffer, as: UTF8.self))
|
|
}
|
|
return decimal
|
|
}
|
|
} else if isSpecialJSON5DoubleValue {
|
|
// Decimal itself doesn't have support for Infinity values yet. Even the part of the old NSJSONSerialization implementation that would try to reinterpret an NaN or Infinity value as an NSDecimalNumber did not have very predictable behavior.
|
|
// TODO: Proper handling of Infinity and NaN Decimal values.
|
|
return Decimal.quietNaN
|
|
} else {
|
|
let numberString = String(decoding: numberBuffer, as: UTF8.self)
|
|
if let decimal = Decimal(entire: numberString) {
|
|
return decimal
|
|
}
|
|
throw JSON5Scanner.validateNumber(from: numberBuffer.suffix(from: digitsStartPtr), fullSource: fullSource)
|
|
}
|
|
|
|
} else {
|
|
let digitsStartPtr = try JSONScanner.prevalidateJSONNumber(from: numberBuffer, hasExponent: hasExponent, fullSource: fullSource)
|
|
let numberString = String(decoding: numberBuffer, as: UTF8.self)
|
|
if let decimal = Decimal(entire: numberString) {
|
|
return decimal
|
|
}
|
|
throw JSONScanner.validateNumber(from: numberBuffer.suffix(from: digitsStartPtr), fullSource: fullSource)
|
|
}
|
|
}
|
|
}
|
|
#endif // FOUNDATION_FRAMEWORK
|
|
|
|
private func unwrapDictionary<T: Decodable>(from mapValue: JSONMap.Value, as type: T.Type, for codingPathNode: _JSONCodingPathNode, _ additionalKey: (some CodingKey)? = nil) throws -> T {
|
|
try checkNotNull(mapValue, expectedType: [String:Any].self, for: codingPathNode, additionalKey)
|
|
|
|
guard let dictType = type as? (_JSONStringDictionaryDecodableMarker & Decodable).Type else {
|
|
preconditionFailure("Must only be called if T implements __JSONStringDictionaryDecodableMarker")
|
|
}
|
|
|
|
guard case let .object(region) = mapValue else {
|
|
throw DecodingError.typeMismatch([String: Any].self, DecodingError.Context(
|
|
codingPath: codingPathNode.path(with: additionalKey),
|
|
debugDescription: "Expected to decode \([String: Any].self) but found \(mapValue.debugDataTypeDescription) instead."
|
|
))
|
|
}
|
|
|
|
var result = [String: Any]()
|
|
result.reserveCapacity(region.count / 2)
|
|
|
|
let dictCodingPathNode = codingPathNode.pushing(additionalKey)
|
|
|
|
var iter = jsonMap.makeObjectIterator(from: region.startOffset)
|
|
while let (keyValue, value) = iter.next() {
|
|
// Failing to unwrap a string here is impossible, as scanning already guarantees that dictionary keys are strings.
|
|
let key = try! self.unwrapString(from: keyValue, for: dictCodingPathNode, _JSONKey?.none)
|
|
let value = try self.unwrap(value, as: dictType.elementType, for: dictCodingPathNode, _JSONKey(stringValue: key)!)
|
|
result[key]._setIfNil(to: value)
|
|
}
|
|
|
|
return result as! T
|
|
}
|
|
|
|
private func unwrapString(from value: JSONMap.Value, for codingPathNode: _JSONCodingPathNode, _ additionalKey: (some CodingKey)? = nil) throws -> String {
|
|
try checkNotNull(value, expectedType: String.self, for: codingPathNode, additionalKey)
|
|
|
|
guard case .string(let region, let isSimple) = value else {
|
|
throw self.createTypeMismatchError(type: String.self, for: codingPathNode.path(with: additionalKey), value: value)
|
|
}
|
|
return try withBuffer(for: region) { stringBuffer, fullSource in
|
|
if isSimple {
|
|
guard let result = String._tryFromUTF8(stringBuffer) else {
|
|
throw JSONError.cannotConvertInputStringDataToUTF8(location: .sourceLocation(at: stringBuffer.startIndex, fullSource: fullSource))
|
|
}
|
|
return result
|
|
}
|
|
if options.json5 {
|
|
return try JSON5Scanner.stringValue(from: stringBuffer, fullSource: fullSource)
|
|
} else {
|
|
return try JSONScanner.stringValue(from: stringBuffer, fullSource: fullSource)
|
|
}
|
|
}
|
|
}
|
|
|
|
func isTrueZero(_ buffer: BufferView<UInt8>) -> Bool {
|
|
var remainingBuffer = buffer
|
|
|
|
// Non-zero numbers are allowed after 'e'/'E'. Since the format is already validated at this stage, we can stop scanning as soon as we see one.
|
|
let nonZeroRange = UInt8(ascii: "1") ... UInt8(ascii: "9")
|
|
|
|
@inline(__always)
|
|
func check(_ off: Int) -> Bool? {
|
|
switch remainingBuffer[uncheckedOffset: off] {
|
|
case nonZeroRange: return false
|
|
case UInt8(ascii: "e"), UInt8(ascii: "E"): return true
|
|
default: return nil
|
|
}
|
|
}
|
|
|
|
// Manual loop unrolling.
|
|
while remainingBuffer.count >= 4 {
|
|
if let res = check(0) { return res }
|
|
if let res = check(1) { return res }
|
|
if let res = check(2) { return res }
|
|
if let res = check(3) { return res }
|
|
|
|
remainingBuffer = remainingBuffer.dropFirst(4)
|
|
}
|
|
|
|
// Process any remaining bytes in the same way.
|
|
switch remainingBuffer.count {
|
|
case 3:
|
|
if let res = check(2) { return res }
|
|
fallthrough
|
|
case 2:
|
|
if let res = check(1) { return res }
|
|
fallthrough
|
|
case 1:
|
|
if let res = check(0) { return res }
|
|
break
|
|
default:
|
|
break
|
|
}
|
|
|
|
return true
|
|
}
|
|
|
|
private func unwrapFloatingPoint<T: PrevalidatedJSONNumberBufferConvertible & BinaryFloatingPoint>(
|
|
from value: JSONMap.Value,
|
|
as type: T.Type,
|
|
for codingPathNode: _JSONCodingPathNode, _ additionalKey: (some CodingKey)? = nil) throws -> T
|
|
{
|
|
try checkNotNull(value, expectedType: type, for: codingPathNode, additionalKey)
|
|
|
|
// We are always willing to return the number as a Double:
|
|
// * If the original value was integral, it is guaranteed to fit in a Double;
|
|
// we are willing to lose precision past 2^53 if you encoded a UInt64 but requested a Double
|
|
// * If it was a Float or Double, you will get back the precise value
|
|
// * If it was Decimal, you will get back the nearest approximation
|
|
|
|
if case .number(let region, let hasExponent) = value {
|
|
return try withBuffer(for: region) { numberBuffer, fullSource in
|
|
if options.json5 {
|
|
let (digitsStartPtr, isHex, isSpecialJSON5DoubleValue) = try JSON5Scanner.prevalidateJSONNumber(from: numberBuffer, fullSource: fullSource)
|
|
|
|
// Use our integer parsers for hex data, because the underlying strtod() implementation of T(prevalidatedBuffer:) is too permissive (e.g. it accepts decimals and 'p' exponents) which otherwise would require prevalidation of the entire string before calling it.
|
|
if isHex {
|
|
if numberBuffer.first.unsafelyUnwrapped == UInt8(ascii: "-") {
|
|
guard let int = Int64(prevalidatedJSON5Buffer: numberBuffer, isHex: isHex), let float = T(exactly: int) else {
|
|
throw JSONError.numberIsNotRepresentableInSwift(parsed: String(decoding: numberBuffer, as: UTF8.self))
|
|
}
|
|
return float
|
|
} else {
|
|
guard let int = UInt64(prevalidatedJSON5Buffer: numberBuffer, isHex: isHex), let float = T(exactly: int) else {
|
|
throw JSONError.numberIsNotRepresentableInSwift(parsed: String(decoding: numberBuffer, as: UTF8.self))
|
|
}
|
|
return float
|
|
}
|
|
} // else, fall through to the T(prevalidatedBuffer:) invocation, which is otherwise compatible with JSON5 after our pre-validation.
|
|
|
|
if let floatingPoint = T(prevalidatedBuffer: numberBuffer) {
|
|
// Check for overflow/underflow, which can result in "rounding" to infinity or zero.
|
|
// While strtod does set ERANGE in the either case, we don't rely on it because setting errno to 0 first and then check the result is surprisingly expensive. For values "rounded" to infinity, we reject those out of hand, unless it's an explicit JSON5 infinity/nan value. For values "rounded" down to zero, we perform check for any non-zero digits in the input, which turns out to be much faster.
|
|
if floatingPoint.isFinite {
|
|
guard floatingPoint != 0 || isTrueZero(numberBuffer) else {
|
|
throw JSONError.numberIsNotRepresentableInSwift(parsed: String(decoding: numberBuffer, as: UTF8.self))
|
|
}
|
|
return floatingPoint
|
|
} else {
|
|
if options.json5, isSpecialJSON5DoubleValue {
|
|
return floatingPoint
|
|
} else {
|
|
throw JSONError.numberIsNotRepresentableInSwift(parsed: String(decoding: numberBuffer, as: UTF8.self))
|
|
}
|
|
}
|
|
}
|
|
|
|
// We failed to parse the number. Is that because it was malformed?
|
|
throw JSON5Scanner.validateNumber(from: numberBuffer.suffix(from: digitsStartPtr), fullSource: fullSource)
|
|
} else {
|
|
let digitsStartPtr = try JSONScanner.prevalidateJSONNumber(from: numberBuffer, hasExponent: hasExponent, fullSource: fullSource)
|
|
|
|
if let floatingPoint = T(prevalidatedBuffer: numberBuffer) {
|
|
// Check for overflow (which results in an infinite result), or rounding to zero.
|
|
// While strtod does set ERANGE in the either case, we don't rely on it because setting errno to 0 first and then check the result is surprisingly expensive. For values "rounded" to infinity, we reject those out of hand. For values "rounded" down to zero, we perform check for any non-zero digits in the input, which turns out to be much faster.
|
|
if floatingPoint.isFinite {
|
|
guard floatingPoint != 0 || isTrueZero(numberBuffer) else {
|
|
throw JSONError.numberIsNotRepresentableInSwift(parsed: String(decoding: numberBuffer, as: UTF8.self))
|
|
}
|
|
return floatingPoint
|
|
} else {
|
|
throw JSONError.numberIsNotRepresentableInSwift(parsed: String(decoding: numberBuffer, as: UTF8.self))
|
|
}
|
|
}
|
|
|
|
throw JSONScanner.validateNumber(from: numberBuffer.suffix(from: digitsStartPtr), fullSource: fullSource)
|
|
}
|
|
}
|
|
}
|
|
|
|
if case .string(let region, let isSimple) = value, isSimple,
|
|
case .convertFromString(var posInfString, var negInfString, var nanString) =
|
|
self.options.nonConformingFloatDecodingStrategy
|
|
{
|
|
let result = withBuffer(for: region) { (stringBuffer, _) -> T? in
|
|
stringBuffer.withUnsafeRawPointer { (ptr, count) -> T? in
|
|
func bytesAreEqual(_ b: UnsafeBufferPointer<UInt8>) -> Bool {
|
|
count == b.count && memcmp(ptr, b.baseAddress!, b.count) == 0
|
|
}
|
|
if posInfString.withUTF8(bytesAreEqual(_:)) { return T.infinity }
|
|
if negInfString.withUTF8(bytesAreEqual(_:)) { return -T.infinity }
|
|
if nanString.withUTF8(bytesAreEqual(_:)) { return T.nan }
|
|
return nil
|
|
}
|
|
}
|
|
if let result { return result }
|
|
}
|
|
|
|
throw self.createTypeMismatchError(type: type, for: codingPathNode.path(with: additionalKey), value: value)
|
|
}
|
|
|
|
private func unwrapFixedWidthInteger<T: FixedWidthInteger>(
|
|
from value: JSONMap.Value,
|
|
as type: T.Type,
|
|
for codingPathNode: _JSONCodingPathNode, _ additionalKey: (some CodingKey)? = nil) throws -> T
|
|
{
|
|
try checkNotNull(value, expectedType: type, for: codingPathNode, additionalKey)
|
|
|
|
guard case .number(let region, let hasExponent) = value else {
|
|
throw self.createTypeMismatchError(type: type, for: codingPathNode.path(with: additionalKey), value: value)
|
|
}
|
|
return try withBuffer(for: region) { numberBuffer, fullSource in
|
|
let digitBeginning: BufferViewIndex<UInt8>
|
|
if options.json5 {
|
|
let isHex : Bool
|
|
let isSpecialFloatValue: Bool
|
|
(digitBeginning, isHex, isSpecialFloatValue) = try JSON5Scanner.prevalidateJSONNumber(from: numberBuffer, fullSource: fullSource)
|
|
|
|
// This is the fast pass. Number directly convertible to desired integer type.
|
|
if let integer = T(prevalidatedJSON5Buffer: numberBuffer, isHex: isHex) {
|
|
return integer
|
|
}
|
|
|
|
// NaN and Infinity values are not representable as Integers.
|
|
if isSpecialFloatValue {
|
|
throw JSONError.numberIsNotRepresentableInSwift(parsed: String(decoding: numberBuffer, as: UTF8.self))
|
|
}
|
|
} else {
|
|
digitBeginning = try JSONScanner.prevalidateJSONNumber(from: numberBuffer, hasExponent: hasExponent, fullSource: fullSource)
|
|
|
|
// This is the fast pass. Number directly convertible to Integer.
|
|
if let integer = T(prevalidatedBuffer: numberBuffer) {
|
|
return integer
|
|
}
|
|
}
|
|
|
|
return try _slowpath_unwrapFixedWidthInteger(as: type, numberBuffer: numberBuffer, fullSource: fullSource, digitBeginning: digitBeginning, for: codingPathNode, additionalKey)
|
|
}
|
|
}
|
|
|
|
private func _slowpath_unwrapFixedWidthInteger<T: FixedWidthInteger>(as type: T.Type, numberBuffer: BufferView<UInt8>, fullSource: BufferView<UInt8>, digitBeginning: BufferViewIndex<UInt8>, for codingPathNode: _JSONCodingPathNode, _ additionalKey: (some CodingKey)?) throws -> T {
|
|
// This is the slow path... If the fast path has failed. For example for "34.0" as an integer, we try to parse as either a Decimal or a Double and then convert back, losslessly.
|
|
if let double = Double(prevalidatedBuffer: numberBuffer) {
|
|
guard let value = T(exactly: double) else {
|
|
throw JSONError.numberIsNotRepresentableInSwift(parsed: String(decoding: numberBuffer, as: UTF8.self))
|
|
}
|
|
return value
|
|
}
|
|
|
|
let number = String(decoding: numberBuffer, as: Unicode.ASCII.self)
|
|
#if FOUNDATION_FRAMEWORK // TODO: Reenable once Decimal is moved
|
|
if let decimal = Decimal(entire: number) {
|
|
guard let value = T(decimal) else {
|
|
throw JSONError.numberIsNotRepresentableInSwift(parsed: String(decoding: numberBuffer, as: UTF8.self))
|
|
}
|
|
return value
|
|
}
|
|
#endif // FOUNDATION_FRAMEWORK
|
|
// Maybe it was just an unreadable sequence?
|
|
if options.json5 {
|
|
throw JSON5Scanner.validateNumber(from: numberBuffer.suffix(from: digitBeginning), fullSource: fullSource)
|
|
} else {
|
|
throw JSONScanner.validateNumber(from: numberBuffer.suffix(from: digitBeginning), fullSource: fullSource)
|
|
}
|
|
}
|
|
|
|
private func createTypeMismatchError(type: Any.Type, for path: [CodingKey], value: JSONMap.Value) -> DecodingError {
|
|
return DecodingError.typeMismatch(type, .init(
|
|
codingPath: path,
|
|
debugDescription: "Expected to decode \(type) but found \(value.debugDataTypeDescription) instead."
|
|
))
|
|
}
|
|
}
|
|
|
|
#if FOUNDATION_FRAMEWORK // TODO: Reenable once Decimal is moved
|
|
extension FixedWidthInteger {
|
|
init?(_ decimal: Decimal) {
|
|
let isNegative = decimal._isNegative != 0
|
|
if decimal._length == 0 && isNegative {
|
|
return nil
|
|
}
|
|
if isNegative {
|
|
guard Self.isSigned else {
|
|
return nil
|
|
}
|
|
}
|
|
|
|
var d : UInt64 = 0
|
|
for i in (0..<decimal._length).reversed() {
|
|
let overflow1: Bool
|
|
let overflow2: Bool
|
|
(d, overflow1) = d.multipliedReportingOverflow(by: 65536)
|
|
(d, overflow2) = d.addingReportingOverflow(UInt64(decimal[i]))
|
|
guard !overflow1 && !overflow2 else {
|
|
return nil
|
|
}
|
|
}
|
|
if (decimal._exponent < 0) {
|
|
for _ in 0 ..< -decimal._exponent {
|
|
let overflow: Bool
|
|
(d, overflow) = d.dividedReportingOverflow(by: 10)
|
|
guard !overflow else {
|
|
return nil
|
|
}
|
|
}
|
|
} else {
|
|
for _ in 0 ..< decimal._exponent {
|
|
let overflow: Bool
|
|
(d, overflow) = d.multipliedReportingOverflow(by: 10)
|
|
guard !overflow else {
|
|
return nil
|
|
}
|
|
}
|
|
}
|
|
if isNegative {
|
|
guard let signedAndSized = Self(exactly: d) else {
|
|
return nil
|
|
}
|
|
self = signedAndSized * -1
|
|
} else {
|
|
guard let sized = Self(exactly: d) else {
|
|
return nil
|
|
}
|
|
self = sized
|
|
}
|
|
}
|
|
}
|
|
|
|
extension Decimal {
|
|
init?(entire string: String) {
|
|
let scan = Scanner(string: string)
|
|
guard let decimal = scan.scanDecimal() else {
|
|
return nil
|
|
}
|
|
guard scan.isAtEnd else {
|
|
return nil
|
|
}
|
|
self = decimal
|
|
}
|
|
}
|
|
#endif // FOUNDATION_FRAMEWORK
|
|
|
|
extension JSONDecoderImpl : SingleValueDecodingContainer {
|
|
func decodeNil() -> Bool {
|
|
switch topValue {
|
|
case .null:
|
|
return true
|
|
default:
|
|
return false
|
|
}
|
|
}
|
|
|
|
func decode(_: Bool.Type) throws -> Bool {
|
|
guard case .bool(let bool) = self.topValue else {
|
|
throw self.createTypeMismatchError(type: Bool.self, for: self.codingPath, value: self.topValue)
|
|
}
|
|
|
|
return bool
|
|
}
|
|
|
|
func decode(_: String.Type) throws -> String {
|
|
try self.unwrapString(from: self.topValue, for: self.codingPathNode, _JSONKey?.none)
|
|
}
|
|
|
|
func decode(_: Double.Type) throws -> Double {
|
|
try decodeFloatingPoint()
|
|
}
|
|
|
|
func decode(_: Float.Type) throws -> Float {
|
|
try decodeFloatingPoint()
|
|
}
|
|
|
|
func decode(_: Int.Type) throws -> Int {
|
|
try decodeFixedWidthInteger()
|
|
}
|
|
|
|
func decode(_: Int8.Type) throws -> Int8 {
|
|
try decodeFixedWidthInteger()
|
|
}
|
|
|
|
func decode(_: Int16.Type) throws -> Int16 {
|
|
try decodeFixedWidthInteger()
|
|
}
|
|
|
|
func decode(_: Int32.Type) throws -> Int32 {
|
|
try decodeFixedWidthInteger()
|
|
}
|
|
|
|
func decode(_: Int64.Type) throws -> Int64 {
|
|
try decodeFixedWidthInteger()
|
|
}
|
|
|
|
func decode(_: UInt.Type) throws -> UInt {
|
|
try decodeFixedWidthInteger()
|
|
}
|
|
|
|
func decode(_: UInt8.Type) throws -> UInt8 {
|
|
try decodeFixedWidthInteger()
|
|
}
|
|
|
|
func decode(_: UInt16.Type) throws -> UInt16 {
|
|
try decodeFixedWidthInteger()
|
|
}
|
|
|
|
func decode(_: UInt32.Type) throws -> UInt32 {
|
|
try decodeFixedWidthInteger()
|
|
}
|
|
|
|
func decode(_: UInt64.Type) throws -> UInt64 {
|
|
try decodeFixedWidthInteger()
|
|
}
|
|
|
|
func decode<T: Decodable>(_ type: T.Type) throws -> T {
|
|
try self.unwrap(self.topValue, as: type, for: codingPathNode, _JSONKey?.none)
|
|
}
|
|
|
|
@inline(__always) private func decodeFixedWidthInteger<T: FixedWidthInteger>() throws -> T {
|
|
try self.unwrapFixedWidthInteger(from: self.topValue, as: T.self, for: codingPathNode, _JSONKey?.none)
|
|
}
|
|
|
|
@inline(__always) private func decodeFloatingPoint<T: PrevalidatedJSONNumberBufferConvertible & BinaryFloatingPoint>() throws -> T {
|
|
try self.unwrapFloatingPoint(from: self.topValue, as: T.self, for: codingPathNode, _JSONKey?.none)
|
|
}
|
|
}
|
|
|
|
extension JSONDecoderImpl {
|
|
struct KeyedContainer<K: CodingKey>: KeyedDecodingContainerProtocol {
|
|
typealias Key = K
|
|
|
|
let impl: JSONDecoderImpl
|
|
let codingPathNode: _JSONCodingPathNode
|
|
let dictionary: [String:JSONMap.Value]
|
|
|
|
static func stringify(objectRegion: JSONMap.Region, using impl: JSONDecoderImpl, codingPathNode: _JSONCodingPathNode, keyDecodingStrategy: JSONDecoder.KeyDecodingStrategy) throws -> [String:JSONMap.Value] {
|
|
var result = [String:JSONMap.Value]()
|
|
result.reserveCapacity(objectRegion.count / 2)
|
|
|
|
var iter = impl.jsonMap.makeObjectIterator(from: objectRegion.startOffset)
|
|
switch keyDecodingStrategy {
|
|
case .useDefaultKeys:
|
|
while let (keyValue, value) = iter.next() {
|
|
// Failing to unwrap a string here is impossible, as scanning already guarantees that dictionary keys are strings.
|
|
let key = try! impl.unwrapString(from: keyValue, for: codingPathNode, _JSONKey?.none)
|
|
result[key]._setIfNil(to: value)
|
|
}
|
|
case .convertFromSnakeCase:
|
|
while let (keyValue, value) = iter.next() {
|
|
// Failing to unwrap a string here is impossible, as scanning already guarantees that dictionary keys are strings.
|
|
let key = try! impl.unwrapString(from: keyValue, for: codingPathNode, _JSONKey?.none)
|
|
|
|
// Convert the snake case keys in the container to camel case.
|
|
// If we hit a duplicate key after conversion, then we'll use the first one we saw.
|
|
// Effectively an undefined behavior with JSON dictionaries.
|
|
result[JSONDecoder.KeyDecodingStrategy._convertFromSnakeCase(key)]._setIfNil(to: value)
|
|
}
|
|
case .custom(let converter):
|
|
let codingPathForCustomConverter = codingPathNode.path
|
|
while let (keyValue, value) = iter.next() {
|
|
// Failing to unwrap a string here is impossible, as scanning already guarantees that dictionary keys are strings.
|
|
let key = try! impl.unwrapString(from: keyValue, for: codingPathNode, _JSONKey?.none)
|
|
|
|
var pathForKey = codingPathForCustomConverter
|
|
pathForKey.append(_JSONKey(stringValue: key)!)
|
|
result[converter(pathForKey).stringValue]._setIfNil(to: value)
|
|
}
|
|
}
|
|
|
|
return result
|
|
}
|
|
|
|
init(impl: JSONDecoderImpl, codingPathNode: _JSONCodingPathNode, region: JSONMap.Region) throws {
|
|
self.impl = impl
|
|
self.codingPathNode = codingPathNode
|
|
self.dictionary = try Self.stringify(objectRegion: region, using: impl, codingPathNode: codingPathNode, keyDecodingStrategy: impl.options.keyDecodingStrategy)
|
|
}
|
|
|
|
public var codingPath : [CodingKey] {
|
|
codingPathNode.path
|
|
}
|
|
|
|
var allKeys: [K] {
|
|
self.dictionary.keys.compactMap { K(stringValue: $0) }
|
|
}
|
|
|
|
func contains(_ key: K) -> Bool {
|
|
dictionary.keys.contains(key.stringValue)
|
|
}
|
|
|
|
func decodeNil(forKey key: K) throws -> Bool {
|
|
guard case .null = try getValue(forKey: key) else {
|
|
return false
|
|
}
|
|
return true
|
|
}
|
|
|
|
func decode(_ type: Bool.Type, forKey key: K) throws -> Bool {
|
|
let value = try getValue(forKey: key)
|
|
|
|
guard case .bool(let bool) = value else {
|
|
throw createTypeMismatchError(type: type, forKey: key, value: value)
|
|
}
|
|
|
|
return bool
|
|
}
|
|
|
|
func decode(_ type: String.Type, forKey key: K) throws -> String {
|
|
let value = try getValue(forKey: key)
|
|
return try impl.unwrapString(from: value, for: self.codingPathNode, key)
|
|
}
|
|
|
|
func decode(_: Double.Type, forKey key: K) throws -> Double {
|
|
try decodeFloatingPoint(key: key)
|
|
}
|
|
|
|
func decode(_: Float.Type, forKey key: K) throws -> Float {
|
|
try decodeFloatingPoint(key: key)
|
|
}
|
|
|
|
func decode(_: Int.Type, forKey key: K) throws -> Int {
|
|
try decodeFixedWidthInteger(key: key)
|
|
}
|
|
|
|
func decode(_: Int8.Type, forKey key: K) throws -> Int8 {
|
|
try decodeFixedWidthInteger(key: key)
|
|
}
|
|
|
|
func decode(_: Int16.Type, forKey key: K) throws -> Int16 {
|
|
try decodeFixedWidthInteger(key: key)
|
|
}
|
|
|
|
func decode(_: Int32.Type, forKey key: K) throws -> Int32 {
|
|
try decodeFixedWidthInteger(key: key)
|
|
}
|
|
|
|
func decode(_: Int64.Type, forKey key: K) throws -> Int64 {
|
|
try decodeFixedWidthInteger(key: key)
|
|
}
|
|
|
|
func decode(_: UInt.Type, forKey key: K) throws -> UInt {
|
|
try decodeFixedWidthInteger(key: key)
|
|
}
|
|
|
|
func decode(_: UInt8.Type, forKey key: K) throws -> UInt8 {
|
|
try decodeFixedWidthInteger(key: key)
|
|
}
|
|
|
|
func decode(_: UInt16.Type, forKey key: K) throws -> UInt16 {
|
|
try decodeFixedWidthInteger(key: key)
|
|
}
|
|
|
|
func decode(_: UInt32.Type, forKey key: K) throws -> UInt32 {
|
|
try decodeFixedWidthInteger(key: key)
|
|
}
|
|
|
|
func decode(_: UInt64.Type, forKey key: K) throws -> UInt64 {
|
|
try decodeFixedWidthInteger(key: key)
|
|
}
|
|
|
|
func decode<T: Decodable>(_ type: T.Type, forKey key: K) throws -> T {
|
|
try self.impl.unwrap(try getValue(forKey: key), as: type, for: codingPathNode, key)
|
|
}
|
|
|
|
func nestedContainer<NestedKey: CodingKey>(keyedBy type: NestedKey.Type, forKey key: K) throws -> KeyedDecodingContainer<NestedKey> {
|
|
let value = try getValue(forKey: key)
|
|
return try impl.with(value: value, path: codingPathNode.pushing(key)) {
|
|
try impl.container(keyedBy: type)
|
|
}
|
|
}
|
|
|
|
func nestedUnkeyedContainer(forKey key: K) throws -> UnkeyedDecodingContainer {
|
|
let value = try getValue(forKey: key)
|
|
return try impl.with(value: value, path: codingPathNode.pushing(key)) {
|
|
try impl.unkeyedContainer()
|
|
}
|
|
}
|
|
|
|
func superDecoder() throws -> Decoder {
|
|
return decoderForKeyNoThrow(_JSONKey.super)
|
|
}
|
|
|
|
func superDecoder(forKey key: K) throws -> Decoder {
|
|
return decoderForKeyNoThrow(key)
|
|
}
|
|
|
|
private func decoderForKeyNoThrow(_ key: some CodingKey) -> JSONDecoderImpl {
|
|
let value: JSONMap.Value
|
|
do {
|
|
value = try getValue(forKey: key)
|
|
} catch {
|
|
// if there no value for this key then return a null value
|
|
value = .null
|
|
}
|
|
let impl = JSONDecoderImpl(userInfo: self.impl.userInfo, from: self.impl.jsonMap, codingPathNode: self.codingPathNode.pushing(key), options: self.impl.options)
|
|
impl.push(value: value)
|
|
return impl
|
|
}
|
|
|
|
@inline(__always) private func getValue(forKey key: some CodingKey) throws -> JSONMap.Value {
|
|
guard let value = dictionary[key.stringValue] else {
|
|
throw DecodingError.keyNotFound(key, .init(
|
|
codingPath: self.codingPath,
|
|
debugDescription: "No value associated with key \(key) (\"\(key.stringValue)\")."
|
|
))
|
|
}
|
|
return value
|
|
}
|
|
|
|
private func createTypeMismatchError(type: Any.Type, forKey key: K, value: JSONMap.Value) -> DecodingError {
|
|
return DecodingError.typeMismatch(type, .init(
|
|
codingPath: self.codingPathNode.path(with: key), debugDescription: "Expected to decode \(type) but found \(value.debugDataTypeDescription) instead."
|
|
))
|
|
}
|
|
|
|
@inline(__always) private func decodeFixedWidthInteger<T: FixedWidthInteger>(key: Self.Key) throws -> T {
|
|
let value = try getValue(forKey: key)
|
|
return try self.impl.unwrapFixedWidthInteger(from: value, as: T.self, for: codingPathNode, key)
|
|
}
|
|
|
|
@inline(__always) private func decodeFloatingPoint<T: PrevalidatedJSONNumberBufferConvertible & BinaryFloatingPoint>(key: K) throws -> T {
|
|
let value = try getValue(forKey: key)
|
|
return try self.impl.unwrapFloatingPoint(from: value, as: T.self, for: codingPathNode, key)
|
|
}
|
|
}
|
|
}
|
|
|
|
extension JSONDecoderImpl {
|
|
struct UnkeyedContainer: UnkeyedDecodingContainer {
|
|
let impl: JSONDecoderImpl
|
|
var valueIterator: JSONMap.ArrayIterator
|
|
var peekedValue: JSONMap.Value?
|
|
let count: Int?
|
|
|
|
var isAtEnd: Bool { self.currentIndex >= (self.count!) }
|
|
var currentIndex = 0
|
|
|
|
init(impl: JSONDecoderImpl, codingPathNode: _JSONCodingPathNode, region: JSONMap.Region) {
|
|
self.impl = impl
|
|
self.codingPathNode = codingPathNode
|
|
self.valueIterator = impl.jsonMap.makeArrayIterator(from: region.startOffset)
|
|
self.count = region.count
|
|
}
|
|
|
|
let codingPathNode: _JSONCodingPathNode
|
|
public var codingPath: [CodingKey] {
|
|
codingPathNode.path
|
|
}
|
|
|
|
@inline(__always)
|
|
var currentIndexKey : _JSONKey {
|
|
.init(index: currentIndex)
|
|
}
|
|
|
|
@inline(__always)
|
|
var currentCodingPath: [CodingKey] {
|
|
codingPathNode.path(with: currentIndexKey)
|
|
}
|
|
|
|
private mutating func advanceToNextValue() {
|
|
currentIndex += 1
|
|
peekedValue = nil
|
|
}
|
|
|
|
mutating func decodeNil() throws -> Bool {
|
|
let value = try self.peekNextValue(ofType: Never.self)
|
|
switch value {
|
|
case .null:
|
|
advanceToNextValue()
|
|
return true
|
|
default:
|
|
// The protocol states:
|
|
// If the value is not null, does not increment currentIndex.
|
|
return false
|
|
}
|
|
}
|
|
|
|
mutating func decode(_ type: Bool.Type) throws -> Bool {
|
|
let value = try self.peekNextValue(ofType: Bool.self)
|
|
guard case .bool(let bool) = value else {
|
|
throw impl.createTypeMismatchError(type: type, for: self.currentCodingPath, value: value)
|
|
}
|
|
|
|
advanceToNextValue()
|
|
return bool
|
|
}
|
|
|
|
mutating func decode(_ type: String.Type) throws -> String {
|
|
let value = try self.peekNextValue(ofType: String.self)
|
|
let string = try impl.unwrapString(from: value, for: codingPathNode, currentIndexKey)
|
|
advanceToNextValue()
|
|
return string
|
|
}
|
|
|
|
mutating func decode(_: Double.Type) throws -> Double {
|
|
try decodeFloatingPoint()
|
|
}
|
|
|
|
mutating func decode(_: Float.Type) throws -> Float {
|
|
try decodeFloatingPoint()
|
|
}
|
|
|
|
mutating func decode(_: Int.Type) throws -> Int {
|
|
try decodeFixedWidthInteger()
|
|
}
|
|
|
|
mutating func decode(_: Int8.Type) throws -> Int8 {
|
|
try decodeFixedWidthInteger()
|
|
}
|
|
|
|
mutating func decode(_: Int16.Type) throws -> Int16 {
|
|
try decodeFixedWidthInteger()
|
|
}
|
|
|
|
mutating func decode(_: Int32.Type) throws -> Int32 {
|
|
try decodeFixedWidthInteger()
|
|
}
|
|
|
|
mutating func decode(_: Int64.Type) throws -> Int64 {
|
|
try decodeFixedWidthInteger()
|
|
}
|
|
|
|
mutating func decode(_: UInt.Type) throws -> UInt {
|
|
try decodeFixedWidthInteger()
|
|
}
|
|
|
|
mutating func decode(_: UInt8.Type) throws -> UInt8 {
|
|
try decodeFixedWidthInteger()
|
|
}
|
|
|
|
mutating func decode(_: UInt16.Type) throws -> UInt16 {
|
|
try decodeFixedWidthInteger()
|
|
}
|
|
|
|
mutating func decode(_: UInt32.Type) throws -> UInt32 {
|
|
try decodeFixedWidthInteger()
|
|
}
|
|
|
|
mutating func decode(_: UInt64.Type) throws -> UInt64 {
|
|
try decodeFixedWidthInteger()
|
|
}
|
|
|
|
mutating func decode<T: Decodable>(_ type: T.Type) throws -> T {
|
|
let value = try self.peekNextValue(ofType: type)
|
|
let result = try impl.unwrap(value, as: type, for: codingPathNode, currentIndexKey)
|
|
|
|
advanceToNextValue()
|
|
return result
|
|
}
|
|
|
|
mutating func nestedContainer<NestedKey: CodingKey>(keyedBy type: NestedKey.Type) throws -> KeyedDecodingContainer<NestedKey> {
|
|
let value = try self.peekNextValue(ofType: KeyedDecodingContainer<NestedKey>.self)
|
|
let container = try impl.with(value: value, path: codingPathNode.pushing(currentIndexKey)) {
|
|
try impl.container(keyedBy: type)
|
|
}
|
|
|
|
advanceToNextValue()
|
|
return container
|
|
}
|
|
|
|
mutating func nestedUnkeyedContainer() throws -> UnkeyedDecodingContainer {
|
|
let value = try self.peekNextValue(ofType: UnkeyedDecodingContainer.self)
|
|
let container = try impl.with(value: value, path: codingPathNode.pushing(currentIndexKey)) {
|
|
try impl.unkeyedContainer()
|
|
}
|
|
|
|
advanceToNextValue()
|
|
return container
|
|
}
|
|
|
|
mutating func superDecoder() throws -> Decoder {
|
|
let decoder = try decoderForNextElement(ofType: Decoder.self)
|
|
advanceToNextValue()
|
|
return decoder
|
|
}
|
|
|
|
private mutating func decoderForNextElement<T>(ofType type: T.Type) throws -> JSONDecoderImpl {
|
|
let value = try self.peekNextValue(ofType: type)
|
|
let impl = JSONDecoderImpl(
|
|
userInfo: self.impl.userInfo,
|
|
from: self.impl.jsonMap,
|
|
codingPathNode: self.codingPathNode.pushing(_JSONKey(index: self.currentIndex)),
|
|
options: self.impl.options
|
|
)
|
|
impl.push(value: value)
|
|
return impl
|
|
}
|
|
|
|
@inline(__always)
|
|
private mutating func peekNextValue<T>(ofType type: T.Type) throws -> JSONMap.Value {
|
|
if let value = peekedValue {
|
|
return value
|
|
}
|
|
guard let nextValue = valueIterator.next() else {
|
|
var message = "Unkeyed container is at end."
|
|
if T.self == UnkeyedContainer.self {
|
|
message = "Cannot get nested unkeyed container -- unkeyed container is at end."
|
|
}
|
|
if T.self == Decoder.self {
|
|
message = "Cannot get superDecoder() -- unkeyed container is at end."
|
|
}
|
|
|
|
var path = self.codingPath
|
|
path.append(_JSONKey(index: self.currentIndex))
|
|
|
|
throw DecodingError.valueNotFound(
|
|
type,
|
|
.init(codingPath: path,
|
|
debugDescription: message,
|
|
underlyingError: nil))
|
|
}
|
|
peekedValue = nextValue
|
|
return nextValue
|
|
}
|
|
|
|
@inline(__always) private mutating func decodeFixedWidthInteger<T: FixedWidthInteger>() throws -> T {
|
|
let value = try self.peekNextValue(ofType: T.self)
|
|
let key = _JSONKey(index: self.currentIndex)
|
|
let result = try self.impl.unwrapFixedWidthInteger(from: value, as: T.self, for: codingPathNode, key)
|
|
advanceToNextValue()
|
|
return result
|
|
}
|
|
|
|
@inline(__always) private mutating func decodeFloatingPoint<T: PrevalidatedJSONNumberBufferConvertible & BinaryFloatingPoint>() throws -> T {
|
|
let value = try self.peekNextValue(ofType: T.self)
|
|
let key = _JSONKey(index: self.currentIndex)
|
|
let result = try self.impl.unwrapFloatingPoint(from: value, as: T.self, for: codingPathNode, key)
|
|
advanceToNextValue()
|
|
return result
|
|
}
|
|
}
|
|
}
|
|
|
|
// This is a workaround for the lack of a "set value only if absent" function for Dictionary.
|
|
extension Optional {
|
|
fileprivate mutating func _setIfNil(to value: Wrapped) {
|
|
guard _fastPath(self == nil) else { return }
|
|
self = value
|
|
}
|
|
}
|
|
|
|
@available(macOS 13.0, iOS 16.0, tvOS 16.0, watchOS 9.0, *)
|
|
extension JSONDecoder : @unchecked Sendable {}
|