File: JSONPointer.swift

package info (click to toggle)
swiftlang 6.0.3-2
  • links: PTS, VCS
  • area: main
  • in suites: forky, sid, trixie
  • size: 2,519,992 kB
  • sloc: cpp: 9,107,863; ansic: 2,040,022; asm: 1,135,751; python: 296,500; objc: 82,456; f90: 60,502; lisp: 34,951; pascal: 19,946; sh: 18,133; perl: 7,482; ml: 4,937; javascript: 4,117; makefile: 3,840; awk: 3,535; xml: 914; fortran: 619; cs: 573; ruby: 573
file content (122 lines) | stat: -rw-r--r-- 4,990 bytes parent folder | download
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
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
/*
 This source file is part of the Swift.org open source project

 Copyright (c) 2021 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 Swift project authors
*/

import Foundation

/// A pointer to a specific value in a JSON document.
///
/// For more information, see [RFC6901](https://datatracker.ietf.org/doc/html/rfc6901).
public struct JSONPointer: Codable, CustomStringConvertible, Equatable {
    /// The path components of the pointer.
    ///
    /// The path components of the pointer are not escaped.
    public var pathComponents: [String]
    
    public var description: String {
        "/\(pathComponents.map(Self.escape).joined(separator: "/"))"
    }
    
    /// Creates a JSON Pointer given its path components.
    ///
    /// The components are assumed to be properly escaped per [RFC6901](https://datatracker.ietf.org/doc/html/rfc6901).
    public init(pathComponents: some Sequence<String>) {
        self.pathComponents = Array(pathComponents)
    }
    
    /// Returns the pointer with the first path component removed.
    public func removingFirstPathComponent() -> JSONPointer {
        JSONPointer(pathComponents: pathComponents.dropFirst())
    }
    
    func prependingPathComponents(_ components: [String]) -> JSONPointer {
        JSONPointer(pathComponents: components + pathComponents)
    }
    
    /// An enum representing characters that need escaping in JSON Pointer values.
    ///
    /// The characters that need to be escaped in JSON Pointer values are defined in
    /// [RFC6901](https://datatracker.ietf.org/doc/html/rfc6901).
    public enum EscapedCharacters: String, CaseIterable {
        /// The tilde character.
        ///
        /// This character is encoded as `~0` in JSON Pointer.
        case tilde = "~"
        
        /// The forward slash character.
        ///
        /// This character is encoded as `~1` in JSON Pointer.
        case forwardSlash = "/"
        
        /// The escaped character.
        public var escaped: String {
            switch self {
            case .tilde: return "~0"
            case .forwardSlash: return "~1"
            }
        }
    }
    
    /// Creates a JSON pointer given a coding path.
    ///
    /// Use this initializer when creating JSON pointers during encoding. This initializer escapes components as defined by
    /// [RFC6901](https://datatracker.ietf.org/doc/html/rfc6901).
    public init(from codingPath: [CodingKey]) {
        self.pathComponents = codingPath.map { component in
            if let intValue = component.intValue {
                // If the coding key is an index into an array, emit the index as a string.
                return "\(intValue)"
            } else {
                // Otherwise, emit the property name, escaped per the JSON Pointer specification.
                return component.stringValue
            }
        }
    }
    
    public func encode(to encoder: Encoder) throws {
        var container = encoder.singleValueContainer()
        try container.encode(description)
    }
    
    public init(from decoder: Decoder) throws {
        let container = try decoder.singleValueContainer()
        let stringValue = try container.decode(String.self)
        self.pathComponents = stringValue.removingLeadingSlash.components(separatedBy: "/").map(Self.unescape)
    }
    
    /// Escapes a path component of a JSON pointer.
    static func escape(_ pointerPathComponents: String) -> String {
        applyEscaping(pointerPathComponents, shouldUnescape: false)
    }
    
    /// Unescaped a path component of a JSON pointer.
    static func unescape(_ pointerPathComponents: String) -> String {
        applyEscaping(pointerPathComponents, shouldUnescape: true)
    }
    
    /// Applies an escaping operation to the path component of a JSON pointer.
    /// - Parameters:
    ///   - pointerPathComponent: The path component to escape.
    ///   - shouldUnescape: Whether this function should unescape or escape the path component.
    /// - Returns: The escaped value if `shouldUnescape` is false, otherwise the escaped value.
    private static func applyEscaping(_ pointerPathComponent: String, shouldUnescape: Bool) -> String {
        EscapedCharacters.allCases
            .reduce(pointerPathComponent) { partialResult, characterThatNeedsEscaping in
                partialResult
                    .replacingOccurrences(
                        of: characterThatNeedsEscaping[
                            keyPath: shouldUnescape ? \EscapedCharacters.escaped : \EscapedCharacters.rawValue
                        ],
                        with: characterThatNeedsEscaping[
                            keyPath: shouldUnescape ? \EscapedCharacters.rawValue : \EscapedCharacters.escaped
                        ]
                    )
            }
    }
}