File: SemanticTreeDumper.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 (219 lines) | stat: -rw-r--r-- 7,889 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
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
/*
 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
*/

/**
 A ``SemanticWalker`` that dumps a textual representation of a ``Semantic`` tree for debugging.
 
 - Note: This type is utilized by a public ``Semantic.dump()`` method available on all semantic nodes.
 */
struct SemanticTreeDumper: SemanticWalker {
    /// The resulting string built up during dumping.
    var result = ""
    
    /// The current path in the tree so far, used for printing edges
    /// in the dumped tree.
    private var path = [Semantic]()
    
    private mutating func dump(_ semantic: Semantic, customDescription: String? = nil) {
        if !path.isEmpty {
            result += "\n"
        }
        path.append(semantic)
        result += indentationPrefix
        result += "\(type(of: semantic))"
        if let directiveConvertible = semantic as? DirectiveConvertible,
            let range = directiveConvertible.originalMarkup.range {
            let start = range.lowerBound
            let end = range.upperBound
            result += " @\(start.line):\(start.column)-\(end.line):\(end.column)"
        }
        if let customDescription {
            result += " \(customDescription)"
        }
        increasingDepth(semantic)
    }
    
    private var indentation: String {
        return String(repeating: " ", count: path.count * 3)
    }
    
    private func parentOfNode(at index: Int) -> Semantic? {
        let parentIndex = index - 1
        guard parentIndex >= 0, path.indices.contains(parentIndex) else {
            return nil
        }
        return path[parentIndex]
    }
    
    /**
     Add an indentation prefix for a semantic node using the current ``path``.
     - parameter semantic: The ``Semantic`` node about to be printed
     */
    private var indentationPrefix: String {
        var prefix = ""
        var suffix = ""
        for (depth, node) in path.enumerated().reversed() {
            guard let parent = parentOfNode(at: depth) else {
                continue
            }
            if let lastChild = parent.children.last, lastChild == node, depth == path.count - 1 {
                suffix.append("└─ ")
            } else if let leaf = path.last, leaf == node {
                suffix.append("├─ ")
            } else if let lastChild = parent.children.last, lastChild != node {
                prefix.append("  │")
            } else {
                prefix.append("   ")
            }
        }
        return prefix.reversed() + suffix
    }
    
    /**
     Push `node` to the current path and descend into the children, popping `node` from the path when returning.
     
     - parameter node: The parent node you're descending into.
     */
    private mutating func increasingDepth(_ node: Semantic) {
        descendIntoChildren(of: node)
        path.removeLast()
    }
    
    mutating func visitCode(_ code: Code) {
        dump(code, customDescription: "fileReference: \(code.fileReference) fileName: '\(code.fileName)' shouldResetDiff: \(code.shouldResetDiff) preview: \(String(describing: code.preview))")
    }
    
    mutating func visitSteps(_ steps: Steps) {
        dump(steps)
    }
    
    mutating func visitStep(_ step: Step) {
        dump(step)
    }
    
    mutating func visitTutorialSection(_ tutorialSection: TutorialSection) {
        dump(tutorialSection)
    }
    
    mutating func visitTutorial(_ tutorial: Tutorial) {
        let projectFiles = tutorial.projectFiles.map { $0.path } ?? "nil"
        dump(tutorial, customDescription: "projectFiles: \(projectFiles)")
    }
    
    mutating func visitIntro(_ intro: Intro) {
        dump(intro, customDescription: "title: '\(intro.title)'")
    }
    
    mutating func visitXcodeRequirement(_ requirement: XcodeRequirement) {
        let description = "title: \(requirement.title.singleQuoted) destination: \(requirement.destination.absoluteString.singleQuoted)"
        dump(requirement, customDescription: description)
    }
    
    mutating func visitAssessments(_ assessments: Assessments) {
        dump(assessments)
    }
    
    mutating func visitMultipleChoice(_ multipleChoice: MultipleChoice) {
        dump(multipleChoice, customDescription: "title: '\(multipleChoice.questionPhrasing)'")
    }
    
    mutating func visitJustification(_ justification: Justification) {
        if let reaction = justification.reaction {
            dump(justification, customDescription: "reaction: '\(reaction)'")
        } else {
            dump(justification)
        }
    }
    
    mutating func visitChoice(_ choice: Choice) {
        dump(choice, customDescription: "isCorrect: \(choice.isCorrect)")
    }
    mutating func visitMarkupContainer(_ markupContainer: MarkupContainer) {
        let description: String
        switch markupContainer.elements.count {
        case 0:
            description = "(empty)"
        case 1:
            description = "(1 element)"
        default:
            description = "(\(markupContainer.elements.count) elements)"
        }
        dump(markupContainer, customDescription: description)
    }
        
    mutating func visitTechnology(_ technology: Technology) {
        dump(technology, customDescription: "name: '\(technology.name)'")
    }
    
    mutating func visitContentAndMedia(_ contentAndMedia: ContentAndMedia) -> () {
        dump(contentAndMedia, customDescription: "mediaPosition: '\(contentAndMedia.mediaPosition.rawValue)'")
    }
    
    mutating func visitVolume(_ volume: Volume) -> () {
        dump(volume, customDescription: volume.name.map { "name: '\($0)'" })
    }
    
    mutating func visitChapter(_ chapter: Chapter) -> () {
        dump(chapter, customDescription: "name: '\(chapter.name)'")
    }
    
    mutating func visitImageMedia(_ imageMedia: ImageMedia) -> () {
        dump(imageMedia, customDescription: "source: '\(imageMedia.source)' altText: \(imageMedia.altText.map { "'\($0)'" } ?? "nil")")
    }
    
    mutating func visitVideoMedia(_ videoMedia: VideoMedia) -> () {
        dump(videoMedia, customDescription: "source: '\(videoMedia.source)' poster: '\(String(describing: videoMedia.poster))'")
    }
    
    mutating func visitTutorialReference(_ tutorialReference: TutorialReference) -> () {
        dump(tutorialReference, customDescription: "tutorial: '\(tutorialReference.topic)'")
    }

    mutating func visitResources(_ resources: Resources) {
        dump(resources)
    }
    
    mutating func visitTile(_ tile: Tile) {
        let description = "identifier: \(tile.identifier) title: \(tile.title.singleQuoted) destination: \(tile.destination?.absoluteString.singleQuoted ?? "nil")"
        dump(tile, customDescription: description)
    }
    
    mutating func visitTutorialArticle(_ article: TutorialArticle) {
        var descriptionComponents: [String] = []
        if let title = article.intro?.title {
            descriptionComponents.append("title: '\(title)'")
        }
        if let time = article.durationMinutes {
            descriptionComponents.append("time: '\(time)'")
        }
        
        if !descriptionComponents.isEmpty {
            dump(article, customDescription: descriptionComponents.joined(separator: " "))
        } else {
            dump(article)
        }
    }
    
    mutating func visitArticle(_ article: Article) {
        dump(article)
    }
    
    mutating func visitStack(_ stack: Stack) {
        dump(stack)
    }

    mutating func visitSymbol(_ symbol: Symbol) {
        dump(symbol)
    }
    
    mutating func visitDeprecationSummary(_ summary: DeprecationSummary) {
        dump(summary)
    }
}