diff options
Diffstat (limited to 'src/api/wix/WixToolset.Data/SourceLineNumber.cs')
-rw-r--r-- | src/api/wix/WixToolset.Data/SourceLineNumber.cs | 275 |
1 files changed, 275 insertions, 0 deletions
diff --git a/src/api/wix/WixToolset.Data/SourceLineNumber.cs b/src/api/wix/WixToolset.Data/SourceLineNumber.cs new file mode 100644 index 00000000..970f17e7 --- /dev/null +++ b/src/api/wix/WixToolset.Data/SourceLineNumber.cs | |||
@@ -0,0 +1,275 @@ | |||
1 | // Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. | ||
2 | |||
3 | namespace WixToolset.Data | ||
4 | { | ||
5 | using System; | ||
6 | using System.IO; | ||
7 | using System.Text; | ||
8 | using System.Xml; | ||
9 | using System.Xml.Linq; | ||
10 | using SimpleJson; | ||
11 | |||
12 | /// <summary> | ||
13 | /// Represents file name and line number for source file | ||
14 | /// </summary> | ||
15 | public sealed class SourceLineNumber | ||
16 | { | ||
17 | /// <summary> | ||
18 | /// Constructor for a source with no line information. | ||
19 | /// </summary> | ||
20 | /// <param name="fileName">File name of the source.</param> | ||
21 | public SourceLineNumber(string fileName) | ||
22 | { | ||
23 | this.FileName = fileName; | ||
24 | } | ||
25 | |||
26 | /// <summary> | ||
27 | /// Constructor for a source with line information. | ||
28 | /// </summary> | ||
29 | /// <param name="fileName">File name of the source.</param> | ||
30 | /// <param name="lineNumber">Line number of the source.</param> | ||
31 | public SourceLineNumber(string fileName, int lineNumber) | ||
32 | { | ||
33 | this.FileName = fileName; | ||
34 | this.LineNumber = lineNumber; | ||
35 | } | ||
36 | |||
37 | /// <summary> | ||
38 | /// Constructor for a source with a parent and no line information. | ||
39 | /// </summary> | ||
40 | /// <param name="fileName">File name of the source.</param> | ||
41 | /// <param name="parent">Parent of this source line number</param> | ||
42 | public SourceLineNumber(string fileName, SourceLineNumber parent) | ||
43 | { | ||
44 | this.FileName = fileName; | ||
45 | this.Parent = parent; | ||
46 | } | ||
47 | |||
48 | /// <summary> | ||
49 | /// Constructor for a source with a parent and line information. | ||
50 | /// </summary> | ||
51 | /// <param name="fileName">File name of the source.</param> | ||
52 | /// <param name="parent">Parent of this source line number</param> | ||
53 | /// <param name="lineNumber">Line number of the source.</param> | ||
54 | public SourceLineNumber(string fileName, SourceLineNumber parent, int lineNumber) | ||
55 | { | ||
56 | this.FileName = fileName; | ||
57 | this.Parent = parent; | ||
58 | this.LineNumber = lineNumber; | ||
59 | } | ||
60 | |||
61 | /// <summary> | ||
62 | /// Gets the file name of the source. | ||
63 | /// </summary> | ||
64 | /// <value>File name for the source.</value> | ||
65 | public string FileName { get; } | ||
66 | |||
67 | /// <summary> | ||
68 | /// Gets or sets the line number of the source. | ||
69 | /// </summary> | ||
70 | /// <value>Line number of the source.</value> | ||
71 | public int? LineNumber { get; set; } | ||
72 | |||
73 | /// <summary> | ||
74 | /// Gets or sets the parent source line number that included this source line number. | ||
75 | /// </summary> | ||
76 | public SourceLineNumber Parent { get; private set; } | ||
77 | |||
78 | /// <summary> | ||
79 | /// Gets the file name and line information. | ||
80 | /// </summary> | ||
81 | /// <value>File name and line information.</value> | ||
82 | public string QualifiedFileName => this.LineNumber.HasValue ? String.Concat(this.FileName, "*", this.LineNumber) : this.FileName; | ||
83 | |||
84 | internal static SourceLineNumber Deserialize(JsonObject jsonObject) | ||
85 | { | ||
86 | var fileName = jsonObject.GetValueOrDefault<string>("file"); | ||
87 | var lineNumber = jsonObject.GetValueOrDefault("line", null); | ||
88 | |||
89 | var parentJson = jsonObject.GetValueOrDefault<JsonObject>("parent"); | ||
90 | var parent = (parentJson == null) ? null : SourceLineNumber.Deserialize(parentJson); | ||
91 | |||
92 | return new SourceLineNumber(fileName) | ||
93 | { | ||
94 | LineNumber = lineNumber, | ||
95 | Parent = parent | ||
96 | }; | ||
97 | } | ||
98 | |||
99 | internal JsonObject Serialize() | ||
100 | { | ||
101 | var jsonObject = new JsonObject | ||
102 | { | ||
103 | { "file", this.FileName }, | ||
104 | { "line", this.LineNumber } | ||
105 | }; | ||
106 | |||
107 | if (this.Parent != null) | ||
108 | { | ||
109 | var parentJson = this.Parent.Serialize(); | ||
110 | jsonObject.Add("parent", parentJson); | ||
111 | } | ||
112 | |||
113 | return jsonObject; | ||
114 | } | ||
115 | |||
116 | /// <summary> | ||
117 | /// Creates a source line number from an encoded string. | ||
118 | /// </summary> | ||
119 | /// <param name="encodedSourceLineNumbers">Encoded string to parse.</param> | ||
120 | public static SourceLineNumber CreateFromEncoded(string encodedSourceLineNumbers) | ||
121 | { | ||
122 | var linesSplitIndex = encodedSourceLineNumbers.IndexOf('|'); | ||
123 | |||
124 | // The most common case is that there is a single encoded line, | ||
125 | // so optimize for that case. | ||
126 | if (linesSplitIndex < 0) | ||
127 | { | ||
128 | return DecodeSourceLineNumber(encodedSourceLineNumbers, 0, -1); | ||
129 | } | ||
130 | else // decode the multiple lines. | ||
131 | { | ||
132 | var startLine = 0; | ||
133 | |||
134 | SourceLineNumber first = null; | ||
135 | SourceLineNumber parent = null; | ||
136 | while (startLine < encodedSourceLineNumbers.Length) | ||
137 | { | ||
138 | var source = DecodeSourceLineNumber(encodedSourceLineNumbers, startLine, linesSplitIndex - 1); | ||
139 | |||
140 | if (null != parent) | ||
141 | { | ||
142 | parent.Parent = source; | ||
143 | } | ||
144 | |||
145 | parent = source; | ||
146 | if (null == first) | ||
147 | { | ||
148 | first = parent; | ||
149 | } | ||
150 | |||
151 | if (linesSplitIndex < 0) | ||
152 | { | ||
153 | break; | ||
154 | } | ||
155 | |||
156 | startLine = linesSplitIndex + 1; | ||
157 | linesSplitIndex = encodedSourceLineNumbers.IndexOf('|', startLine); | ||
158 | } | ||
159 | |||
160 | return first; | ||
161 | } | ||
162 | } | ||
163 | |||
164 | /// <summary> | ||
165 | /// Creates a source line number from a URI. | ||
166 | /// </summary> | ||
167 | /// <param name="uri">Uri to convert into source line number</param> | ||
168 | public static SourceLineNumber CreateFromUri(string uri) | ||
169 | { | ||
170 | if (String.IsNullOrEmpty(uri)) | ||
171 | { | ||
172 | return null; | ||
173 | } | ||
174 | |||
175 | // make the local path look like a normal local path | ||
176 | var localPath = new Uri(uri).LocalPath; | ||
177 | localPath = localPath.TrimStart(Path.AltDirectorySeparatorChar).Replace(Path.AltDirectorySeparatorChar, Path.DirectorySeparatorChar); | ||
178 | |||
179 | return new SourceLineNumber(localPath); | ||
180 | } | ||
181 | |||
182 | /// <summary> | ||
183 | /// Creates a source line number from an XObject. | ||
184 | /// </summary> | ||
185 | /// <param name="node">XML node to create source line number from.</param> | ||
186 | /// <param name="offset">Optional line number offset into XML file not already included in the line information.</param> | ||
187 | public static SourceLineNumber CreateFromXObject(XObject node, int offset = 0) | ||
188 | { | ||
189 | var result = CreateFromUri(node.BaseUri); | ||
190 | if (null != result && node is IXmlLineInfo lineInfo) | ||
191 | { | ||
192 | result.LineNumber = lineInfo.LineNumber + offset; | ||
193 | } | ||
194 | |||
195 | return result; | ||
196 | } | ||
197 | |||
198 | /// <summary> | ||
199 | /// Get the source line information for the current element. Typically this information | ||
200 | /// is set by the precompiler for each element that it encounters. | ||
201 | /// </summary> | ||
202 | /// <param name="node">Element to get source line information for.</param> | ||
203 | /// <returns> | ||
204 | /// The source line number used to author the element being processed or | ||
205 | /// null if the preprocessor did not process the element or the node is | ||
206 | /// not an element. | ||
207 | /// </returns> | ||
208 | public static SourceLineNumber GetFromXAnnotation(XObject node) | ||
209 | { | ||
210 | return node.Annotation<SourceLineNumber>(); | ||
211 | } | ||
212 | |||
213 | /// <summary> | ||
214 | /// Returns the SourceLineNumber and parents encoded as a string. | ||
215 | /// </summary> | ||
216 | public string GetEncoded() | ||
217 | { | ||
218 | var sb = new StringBuilder(this.QualifiedFileName); | ||
219 | |||
220 | for (var parent = this.Parent; null != parent; parent = parent.Parent) | ||
221 | { | ||
222 | sb.Append("|"); | ||
223 | sb.Append(parent.QualifiedFileName); | ||
224 | } | ||
225 | |||
226 | return sb.ToString(); | ||
227 | } | ||
228 | |||
229 | /// <summary> | ||
230 | /// Determines if two SourceLineNumbers are equivalent. | ||
231 | /// </summary> | ||
232 | /// <param name="obj">Object to compare.</param> | ||
233 | /// <returns>True if SourceLineNumbers are equivalent.</returns> | ||
234 | public override bool Equals(object obj) | ||
235 | { | ||
236 | return obj is SourceLineNumber other && | ||
237 | this.LineNumber.HasValue == other.LineNumber.HasValue && | ||
238 | (!this.LineNumber.HasValue || this.LineNumber == other.LineNumber) && | ||
239 | this.FileName.Equals(other.FileName, StringComparison.OrdinalIgnoreCase) && | ||
240 | (null == this.Parent && null == other.Parent || this.Parent.Equals(other.Parent)); | ||
241 | } | ||
242 | |||
243 | /// <summary> | ||
244 | /// Serves as a hash code for a particular type. | ||
245 | /// </summary> | ||
246 | /// <returns>The hash code.</returns> | ||
247 | public override int GetHashCode() | ||
248 | { | ||
249 | return this.GetEncoded().GetHashCode(); | ||
250 | } | ||
251 | |||
252 | /// <summary> | ||
253 | /// Shows a string representation of a source line number. | ||
254 | /// </summary> | ||
255 | /// <returns>String representation of a source line number.</returns> | ||
256 | public override string ToString() | ||
257 | { | ||
258 | return this.LineNumber.HasValue && !String.IsNullOrEmpty(this.FileName) ? String.Concat(this.FileName, "(", this.LineNumber, ")") : this.FileName ?? String.Empty; | ||
259 | } | ||
260 | |||
261 | private static SourceLineNumber DecodeSourceLineNumber(string encoded, int startIndex, int endIndex) | ||
262 | { | ||
263 | if (endIndex < 0) | ||
264 | { | ||
265 | endIndex = encoded.Length - 1; | ||
266 | } | ||
267 | |||
268 | var count = endIndex - startIndex; | ||
269 | var filenameSplitIndex = encoded.LastIndexOf('*', endIndex - 1, count); | ||
270 | return (filenameSplitIndex < 0) ? new SourceLineNumber(encoded) : | ||
271 | new SourceLineNumber(encoded.Substring(startIndex, filenameSplitIndex - startIndex), | ||
272 | Convert.ToInt32(encoded.Substring(filenameSplitIndex + 1, endIndex - filenameSplitIndex))); | ||
273 | } | ||
274 | } | ||
275 | } | ||