@speleotica/frcsdata 4.0.0 → 4.0.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (42) hide show
  1. package/FrcsPlotFile.js +5 -1
  2. package/FrcsPlotShot.js +5 -1
  3. package/FrcsShot.js +4 -5
  4. package/FrcsSurveyFile.js +5 -1
  5. package/FrcsTrip.js +5 -1
  6. package/FrcsTripSummary.js +5 -1
  7. package/FrcsTripSummaryFile.js +5 -1
  8. package/es/FrcsPlotFile.js +5 -1
  9. package/es/FrcsPlotShot.js +5 -1
  10. package/es/FrcsShot.js +5 -6
  11. package/es/FrcsSurveyFile.js +5 -1
  12. package/es/FrcsTrip.js +5 -1
  13. package/es/FrcsTripSummary.js +5 -1
  14. package/es/FrcsTripSummaryFile.js +5 -1
  15. package/es/formatFrcsShot.js +5 -19
  16. package/es/formatFrcsShot.spec.js +8 -17
  17. package/es/index.js +21 -33
  18. package/es/node/index.js +2 -10
  19. package/es/node/parseFrcsPlotFile.spec.js +1 -5
  20. package/es/node/parseFrcsSurveyFile.spec.js +2 -6
  21. package/es/node/parseFrcsTripSummaryFile.spec.js +1 -5
  22. package/es/parseFrcsPlotFile.js +1 -26
  23. package/es/parseFrcsSurveyFile.js +45 -95
  24. package/es/parseFrcsTripSummaryFile.js +1 -12
  25. package/es/string/index.js +2 -9
  26. package/es/string/parseFrcsPlotFile.spec.js +1 -5
  27. package/es/string/parseFrcsSurveyFile.spec.js +8 -14
  28. package/formatFrcsShot.js +5 -19
  29. package/formatFrcsShot.spec.js +4 -15
  30. package/index.js +20 -32
  31. package/node/index.js +1 -9
  32. package/node/parseFrcsPlotFile.spec.js +165 -180
  33. package/node/parseFrcsSurveyFile.spec.js +306 -321
  34. package/node/parseFrcsTripSummaryFile.spec.js +83 -106
  35. package/package.json +15 -15
  36. package/parseFrcsPlotFile.js +224 -318
  37. package/parseFrcsSurveyFile.js +411 -452
  38. package/parseFrcsTripSummaryFile.js +102 -142
  39. package/string/index.js +10 -27
  40. package/string/parseFrcsPlotFile.spec.js +165 -180
  41. package/string/parseFrcsSurveyFile.spec.js +598 -689
  42. package/yarn.lock +1285 -702
@@ -4,106 +4,86 @@ Object.defineProperty(exports, "__esModule", {
4
4
  value: true
5
5
  });
6
6
  exports.default = parseFrcsSurveyFile;
7
-
8
7
  var _parseSegment = require("parse-segment");
9
-
10
8
  var _unitized = require("@speleotica/unitized");
11
-
12
9
  var _FrcsShot = require("./FrcsShot");
13
-
14
10
  function parseNumber(s, unit) {
15
11
  const value = parseFloat(s);
16
12
  if (isNaN(value)) return null;
17
13
  return new _unitized.UnitizedNumber(value, unit);
18
14
  }
19
-
20
15
  function parseAzimuth(s, unit) {
21
16
  const parsed = parseNumber(s, unit);
22
17
  return (parsed === null || parsed === void 0 ? void 0 : parsed.get(_unitized.Angle.degrees)) === 360 ? _unitized.Unitize.degrees(0) : parsed;
23
18
  }
24
-
25
19
  function parseKind(kind) {
26
20
  switch (kind) {
27
21
  case 'H':
28
22
  return _FrcsShot.FrcsShotKind.Horizontal;
29
-
30
23
  case 'D':
31
24
  return _FrcsShot.FrcsShotKind.Diagonal;
32
-
33
25
  default:
34
26
  return _FrcsShot.FrcsShotKind.Normal;
35
27
  }
36
28
  }
37
-
38
29
  function parseLengthUnit(unit) {
39
30
  switch (unit) {
40
31
  case 'FI':
41
32
  return _unitized.Length.inches;
42
-
33
+ case 'FF':
43
34
  case 'FT':
44
35
  return _unitized.Length.feet;
45
-
36
+ case 'MT':
46
37
  case 'M ':
47
38
  return _unitized.Length.meters;
48
39
  }
49
-
50
40
  return null;
51
41
  }
52
-
53
42
  function parseAngleUnit(unit) {
54
43
  switch (unit) {
55
44
  case 'D':
56
45
  return _unitized.Angle.degrees;
57
-
58
46
  case 'G':
59
47
  return _unitized.Angle.gradians;
60
-
61
48
  case 'M':
62
49
  return _unitized.Angle.milsNATO;
63
50
  }
64
-
65
51
  return null;
66
- } // determines if a cell contains a valid station name.
67
-
52
+ }
68
53
 
54
+ // determines if a cell contains a valid station name.
69
55
  function isValidStation(s) {
70
56
  return /^\s*\S+\s*$/.test(s);
71
- } // determines if a cell contains a valid unsigned integer.
72
-
73
-
57
+ }
58
+ // determines if a cell contains a valid unsigned integer.
74
59
  function isValidUInt(s) {
75
60
  return /^\s*[0-9]+\s*$/.test(s);
76
- } // determines if a cell contains a valid float.
77
-
78
-
61
+ }
62
+ // determines if a cell contains a valid float.
79
63
  function isValidFloat(s) {
80
64
  return /^\s*[-+]?([0-9]+(\.[0-9]*)?|\.[0-9]+)\s*$/.test(s);
81
- } // determines if a cell contains a valid unsigned float or whitespace.
82
-
83
-
65
+ }
66
+ // determines if a cell contains a valid unsigned float or whitespace.
84
67
  function isValidOptFloat(s) {
85
68
  return /^\s*[-+]?[0-9]*(\.[0-9]*)?\s*$/.test(s);
86
- } // determines if a cell contains a valid unsigned float or whitespace.
87
-
88
-
69
+ }
70
+ // determines if a cell contains a valid unsigned float or whitespace.
89
71
  function isValidOptUFloat(s) {
90
72
  return /^\s*[0-9]*(\.[0-9]*)?\s*$/.test(s);
91
- } // determines if a cell contains a valid unsigned float.
92
-
93
-
73
+ }
74
+ // determines if a cell contains a valid unsigned float.
94
75
  function isValidUFloat(s) {
95
76
  return /^\s*([0-9]+(\.[0-9]*)?|\.[0-9]+)\s*$/.test(s);
96
- } // determines if a cell contains a valid inclination or whitespace.
97
-
98
-
77
+ }
78
+ // determines if a cell contains a valid inclination or whitespace.
99
79
  function isValidOptInclination(s) {
100
80
  return /^\s*[-+]?[0-9]*(\.[0-9]*)?\s*$/.test(s);
101
81
  }
102
-
103
82
  function parseLrud(s, unit) {
104
83
  const value = parseFloat(s);
105
84
  return !Number.isFinite(value) || value < 0 ? null : new _unitized.UnitizedNumber(value, unit);
106
85
  }
86
+
107
87
  /**
108
88
  * Parses a raw cdata.fr survey file. These look like so:
109
89
  *
@@ -149,9 +129,8 @@ FI B DD
149
129
  ...</pre>
150
130
  *
151
131
  */
152
-
153
-
154
- async function parseFrcsSurveyFile(file, // eslint-disable-line @typescript-eslint/no-explicit-any
132
+ async function parseFrcsSurveyFile(file,
133
+ // eslint-disable-line @typescript-eslint/no-explicit-any
155
134
  lines) {
156
135
  let cave = null;
157
136
  let location = null;
@@ -168,20 +147,16 @@ lines) {
168
147
  let trip = null;
169
148
  let inBlockComment = false;
170
149
  let section;
171
-
172
150
  function getComment() {
173
151
  if (!commentLines) return null;
174
152
  const comment = commentLines.join('\n').trim();
175
153
  commentLines = null;
176
154
  return comment || null;
177
155
  }
178
-
179
156
  let lineNumber = 0;
180
-
181
157
  for await (const line of lines) {
182
158
  let errored = false;
183
159
  const thisLineNumber = lineNumber;
184
-
185
160
  const error = (message, startColumn, endColumn) => {
186
161
  errored = true;
187
162
  errors.push(new _parseSegment.SegmentParseError(message, new _parseSegment.Segment({
@@ -191,34 +166,25 @@ lines) {
191
166
  startCol: 0
192
167
  }).substring(startColumn, endColumn)));
193
168
  };
194
-
195
169
  const validate = (startColumn, endColumn, fieldName, validator) => {
196
170
  const field = line.substring(startColumn, endColumn);
197
-
198
171
  if (!validator(field)) {
199
172
  error((field.trim() ? 'Invalid ' : 'Missing ') + fieldName, startColumn, endColumn);
200
173
  }
201
-
202
174
  return field;
203
175
  };
204
-
205
176
  lineNumber++;
206
-
207
177
  if (lineNumber === 1) {
208
178
  const match = /^\s*([^,]+)(,(.*))?/.exec(line);
209
-
210
179
  if (match) {
211
180
  cave = match[1].trim();
212
-
213
181
  if (match[3]) {
214
182
  location = match[3].trim();
215
183
  }
216
184
  }
217
185
  }
218
-
219
186
  if (line.charAt(0) === ' ' && line.charAt(1) === '*') {
220
187
  inTripComment = !inTripComment;
221
-
222
188
  if (inTripComment) {
223
189
  section = undefined;
224
190
  tripComment = [];
@@ -231,7 +197,6 @@ lines) {
231
197
  tripName = line && line.trim();
232
198
  } else if (lineNumber === tripCommentStartLine + 2) {
233
199
  const match = /^(.+?)\s*[-.]\s*(\d+)\/(\d+)\/(\d+)$/.exec(line && line.trim());
234
-
235
200
  if (match) {
236
201
  let k = 1;
237
202
  const team = match[k++];
@@ -244,9 +209,7 @@ lines) {
244
209
  } else if (lineNumber > 1) {
245
210
  tripComment.push(line);
246
211
  }
247
-
248
212
  const match = /^\*\*\*([^*])\*\*\*/.exec(line);
249
-
250
213
  if (match) {
251
214
  section = match[1].trim();
252
215
  }
@@ -262,39 +225,30 @@ lines) {
262
225
  // FT CC DD
263
226
  // 01234567
264
227
  let distanceUnit = parseLengthUnit(line.slice(0, 2));
265
-
266
228
  if (!distanceUnit) {
267
229
  distanceUnit = _unitized.Length.feet;
268
230
  error('Invalid distance unit', 0, 2);
269
231
  }
270
-
271
232
  let azimuthUnit = parseAngleUnit(line[6]);
272
-
273
233
  if (!azimuthUnit) {
274
234
  azimuthUnit = _unitized.Angle.degrees;
275
235
  error('Invalid azimuth unit', 6, 7);
276
236
  }
277
-
278
237
  let inclinationUnit = parseAngleUnit(line[7]);
279
-
280
238
  if (!inclinationUnit) {
281
239
  inclinationUnit = _unitized.Angle.degrees;
282
240
  error('Invalid inclination unit', 7, 8);
283
241
  }
284
-
285
242
  const backsightAzimuthCorrected = line[3] === 'C';
286
243
  const backsightInclinationCorrected = line[4] === 'C';
287
244
  const hasBacksightAzimuth = line[3] !== ' ' && line[3] !== '-';
288
245
  const hasBacksightInclination = line[4] !== ' ' && line[4] !== '-';
289
-
290
246
  if (!/[-CB ]/.test(line[3])) {
291
247
  error('Invalid backsight azimuth type', 3, 4);
292
248
  }
293
-
294
249
  if (!/[-CB ]/.test(line[4])) {
295
250
  error('Invalid backsight inclination type', 4, 5);
296
251
  }
297
-
298
252
  trip = {
299
253
  header: {
300
254
  name: tripName || '',
@@ -325,16 +279,19 @@ lines) {
325
279
  distanceUnit
326
280
  } = trip.header;
327
281
  const inches = distanceUnit === _unitized.Length.inches;
328
- if (inches) distanceUnit = _unitized.Length.feet; // rigorously check the values in all the columns to make sure this
282
+ if (inches) distanceUnit = _unitized.Length.feet;
283
+
284
+ // rigorously check the values in all the columns to make sure this
329
285
  // is really a survey shot line, just in case any stray comments are
330
286
  // not properly delimited.
331
- // from station name
332
287
 
288
+ // from station name
333
289
  if (!/\S/.test(line.substring(5, 10))) continue;
334
- const fromStr = validate(5, 10, 'from station', isValidStation); // Sadly I have found negative LRUD values in Chip's format and apparently
290
+ const fromStr = validate(5, 10, 'from station', isValidStation);
291
+
292
+ // Sadly I have found negative LRUD values in Chip's format and apparently
335
293
  // his program doesn't fail on them, so I have to accept them here
336
294
  // isValidOptFloat instead of isValidOptUFloat
337
-
338
295
  const lStr = validate(40, 43, 'left', isValidOptFloat);
339
296
  const rStr = validate(43, 46, 'right', isValidOptFloat);
340
297
  const uStr = validate(46, 49, 'up', isValidOptFloat);
@@ -343,10 +300,10 @@ lines) {
343
300
  const up = parseLrud(uStr, distanceUnit);
344
301
  const down = parseLrud(dStr, distanceUnit);
345
302
  const left = parseLrud(lStr, distanceUnit);
346
- const right = parseLrud(rStr, distanceUnit); // to station name
303
+ const right = parseLrud(rStr, distanceUnit);
347
304
 
305
+ // to station name
348
306
  const toStr = line.substring(0, 5);
349
-
350
307
  if (!toStr.trim()) {
351
308
  const shot = {
352
309
  from: fromStr.trim(),
@@ -369,14 +326,11 @@ lines) {
369
326
  shots.push(shot);
370
327
  continue;
371
328
  }
372
-
373
329
  if (!isValidStation(toStr)) {
374
330
  error('Invalid station name', 0, 5);
375
331
  }
376
-
377
332
  const fromLrudMatch = new RegExp(`^\\s+${fromStr.trim().replace(/[.*+?^${}()|[\]\\]/g, '\\$&')}((\\s+(\\d+(\\.\\d*)?|\\.\\d+)){4})`).exec(line.substring(52));
378
333
  let fromLruds;
379
-
380
334
  if (fromLrudMatch) {
381
335
  const [left, right, up, down] = fromLrudMatch[1].trim().split(/\s+/g).map(s => parseLrud(s, distanceUnit));
382
336
  fromLruds = {
@@ -385,9 +339,9 @@ lines) {
385
339
  up,
386
340
  down
387
341
  };
388
- } // azimuth and inclination
389
-
342
+ }
390
343
 
344
+ // azimuth and inclination
391
345
  const azmFsStr = validate(19, 25, 'azimuth', isValidOptUFloat);
392
346
  const azmBsStr = validate(25, 30, 'azimuth', isValidOptUFloat);
393
347
  const incFsStr = line.substring(30, 35);
@@ -400,26 +354,27 @@ lines) {
400
354
  let frontsightInclination;
401
355
  let backsightInclination;
402
356
  let excludeDistance;
403
- let isSplay; // parse distance
357
+ let isSplay;
404
358
 
359
+ // parse distance
405
360
  if (inches) {
406
361
  const feetStr = line.substring(10, 14);
407
- const inchesStr = line.substring(14, 17); // feet and inches are not both optional
408
-
362
+ const inchesStr = line.substring(14, 17);
363
+ // feet and inches are not both optional
409
364
  if (!isValidUInt(feetStr) && !isValidUInt(inchesStr)) {
410
365
  const invalid = feetStr.trim() || inchesStr.trim();
411
366
  error(invalid ? 'Invalid distance' : 'Missing distance', 10, 17);
412
367
  continue;
413
- } // sometimes inches are omitted, hence the || 0...I'm assuming it's possible
414
- // for feet to be omitted as well
415
-
368
+ }
416
369
 
370
+ // sometimes inches are omitted, hence the || 0...I'm assuming it's possible
371
+ // for feet to be omitted as well
417
372
  distance = _unitized.Unitize.inches(parseFloat(inchesStr) || 0).add(_unitized.Unitize.feet(parseFloat(feetStr) || 0));
418
- kind = parseKind(line[17]); // NOTE there are two columns around here that can contain a *.
373
+ kind = parseKind(line[17]);
374
+ // NOTE there are two columns around here that can contain a *.
419
375
  // I think they might represent different values, but thisis confused by
420
376
  // the fact that for ft/in shots, if there is a D or H flag it occupies the
421
377
  // first column that can contain a * for decimal feet shots
422
-
423
378
  excludeDistance = line[18] === '*' || line[18] === 's';
424
379
  isSplay = line[18] === 's';
425
380
  } else {
@@ -430,15 +385,14 @@ lines) {
430
385
  excludeDistance = line[17] === '*' || line[17] === 's';
431
386
  isSplay = line[17] === 's';
432
387
  }
433
-
434
388
  if (kind !== _FrcsShot.FrcsShotKind.Normal) {
435
389
  validate(30, 35, 'vertical-distance', isValidFloat);
436
- } // convert horizontal and diagonal shots to standard
390
+ }
391
+
392
+ // convert horizontal and diagonal shots to standard
437
393
  // in this case incFs is the vertical offset between stations
438
394
  // fortunately it appears we can always count on incFs being specified
439
395
  // and incBs not being specified for these types of shots
440
-
441
-
442
396
  if (kind === _FrcsShot.FrcsShotKind.Horizontal) {
443
397
  // distance is horizontal offset and incFsStr is vertical offset
444
398
  horizontalDistance = distance;
@@ -457,21 +411,18 @@ lines) {
457
411
  backsightInclination = null;
458
412
  } else {
459
413
  // frontsight inclination
460
- validate(30, 35, 'inclination', isValidOptInclination); // frontsight inclination
461
-
414
+ validate(30, 35, 'inclination', isValidOptInclination);
415
+ // frontsight inclination
462
416
  validate(35, 40, 'inclination', isValidOptInclination);
463
417
  frontsightInclination = parseNumber(incFsStr, inclinationUnit);
464
418
  backsightInclination = parseNumber(incBsStr, inclinationUnit);
465
419
  }
466
-
467
420
  if (errored) continue;
468
421
  const frontsightAzimuth = parseAzimuth(azmFsStr, azimuthUnit);
469
422
  const backsightAzimuth = parseAzimuth(azmBsStr, azimuthUnit);
470
-
471
423
  if (!frontsightInclination && !backsightInclination) {
472
424
  frontsightInclination = _unitized.Unitize.degrees(0);
473
425
  }
474
-
475
426
  const shot = {
476
427
  from: fromStr.trim(),
477
428
  to: toStr.trim(),
@@ -497,7 +448,6 @@ lines) {
497
448
  shots.push(shot);
498
449
  }
499
450
  }
500
-
501
451
  return {
502
452
  cave,
503
453
  location,
@@ -505,4 +455,4 @@ lines) {
505
455
  errors
506
456
  };
507
457
  }
508
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../src/parseFrcsSurveyFile.ts"],"names":["parseNumber","s","unit","value","parseFloat","isNaN","UnitizedNumber","parseAzimuth","parsed","get","Angle","degrees","Unitize","parseKind","kind","FrcsShotKind","Horizontal","Diagonal","Normal","parseLengthUnit","Length","inches","feet","meters","parseAngleUnit","gradians","milsNATO","isValidStation","test","isValidUInt","isValidFloat","isValidOptFloat","isValidOptUFloat","isValidUFloat","isValidOptInclination","parseLrud","Number","isFinite","parseFrcsSurveyFile","file","lines","cave","location","trips","errors","tripName","tripTeam","tripDate","inTripComment","tripCommentStartLine","tripCommentEndLine","tripComment","commentLines","trip","inBlockComment","section","getComment","comment","join","trim","lineNumber","line","errored","thisLineNumber","error","message","startColumn","endColumn","push","SegmentParseError","Segment","source","startLine","startCol","substring","validate","fieldName","validator","field","match","exec","charAt","undefined","k","team","month","parseInt","day","year","Date","split","indexOf","part","distanceUnit","slice","azimuthUnit","inclinationUnit","backsightAzimuthCorrected","backsightInclinationCorrected","hasBacksightAzimuth","hasBacksightInclination","header","name","date","shots","fromStr","lStr","rStr","uStr","dStr","up","down","left","right","toStr","shot","from","to","distance","frontsightAzimuth","backsightAzimuth","frontsightInclination","backsightInclination","fromLruds","excludeDistance","fromLrudMatch","RegExp","replace","map","azmFsStr","azmBsStr","incFsStr","incBsStr","horizontalDistance","verticalDistance","isSplay","feetStr","inchesStr","invalid","add","h","v","Math","sqrt","atan2","d","asin","toLruds"],"mappings":"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","sourcesContent":["import { FrcsSurveyFile } from './FrcsSurveyFile'\nimport { FrcsTrip } from './FrcsTrip'\nimport { Segment, SegmentParseError } from 'parse-segment'\nimport {\n  Angle,\n  Length,\n  Unit,\n  UnitizedNumber,\n  UnitType,\n  Unitize,\n} from '@speleotica/unitized'\nimport { FrcsShot, FrcsShotKind } from './FrcsShot'\n\nfunction parseNumber<T extends UnitType<T>>(\n  s: string,\n  unit: Unit<T>\n): UnitizedNumber<T> | null {\n  const value = parseFloat(s)\n  if (isNaN(value)) return null\n  return new UnitizedNumber(value, unit)\n}\n\nfunction parseAzimuth(\n  s: string,\n  unit: Unit<Angle>\n): UnitizedNumber<Angle> | null {\n  const parsed = parseNumber(s, unit)\n  return parsed?.get(Angle.degrees) === 360 ? Unitize.degrees(0) : parsed\n}\n\nfunction parseKind(kind: string): FrcsShotKind {\n  switch (kind) {\n    case 'H':\n      return FrcsShotKind.Horizontal\n    case 'D':\n      return FrcsShotKind.Diagonal\n    default:\n      return FrcsShotKind.Normal\n  }\n}\n\nfunction parseLengthUnit(unit: string): Unit<Length> | null {\n  switch (unit) {\n    case 'FI':\n      return Length.inches\n    case 'FT':\n      return Length.feet\n    case 'M ':\n      return Length.meters\n  }\n  return null\n}\nfunction parseAngleUnit(unit: string): Unit<Angle> | null {\n  switch (unit) {\n    case 'D':\n      return Angle.degrees\n    case 'G':\n      return Angle.gradians\n    case 'M':\n      return Angle.milsNATO\n  }\n  return null\n}\n\n// determines if a cell contains a valid station name.\nfunction isValidStation(s: string): boolean {\n  return /^\\s*\\S+\\s*$/.test(s)\n}\n// determines if a cell contains a valid unsigned integer.\nfunction isValidUInt(s: string): boolean {\n  return /^\\s*[0-9]+\\s*$/.test(s)\n}\n// determines if a cell contains a valid float.\nfunction isValidFloat(s: string): boolean {\n  return /^\\s*[-+]?([0-9]+(\\.[0-9]*)?|\\.[0-9]+)\\s*$/.test(s)\n}\n// determines if a cell contains a valid unsigned float or whitespace.\nfunction isValidOptFloat(s: string): boolean {\n  return /^\\s*[-+]?[0-9]*(\\.[0-9]*)?\\s*$/.test(s)\n}\n// determines if a cell contains a valid unsigned float or whitespace.\nfunction isValidOptUFloat(s: string): boolean {\n  return /^\\s*[0-9]*(\\.[0-9]*)?\\s*$/.test(s)\n}\n// determines if a cell contains a valid unsigned float.\nfunction isValidUFloat(s: string): boolean {\n  return /^\\s*([0-9]+(\\.[0-9]*)?|\\.[0-9]+)\\s*$/.test(s)\n}\n// determines if a cell contains a valid inclination or whitespace.\nfunction isValidOptInclination(s: string): boolean {\n  return /^\\s*[-+]?[0-9]*(\\.[0-9]*)?\\s*$/.test(s)\n}\nfunction parseLrud<T extends UnitType<T>>(\n  s: string,\n  unit: Unit<Length>\n): UnitizedNumber<Length> | null {\n  const value = parseFloat(s)\n  return !Number.isFinite(value) || value < 0\n    ? null\n    : new UnitizedNumber(value, unit)\n}\n\n/**\n * Parses a raw cdata.fr survey file.  These look like so:\n *\n<pre>      Fisher Ridge Cave System, Hart Co., KY\nENTRANCE DROPS, JOE'S \"I LOVE MY WIFE TRAVERSE\", TRICKY TRAVERSE\nPETER QUICK, KEITH ORTIZ   -  2-15-81\nThis File has Crumps test connected.  11/20/12\n *\nFT C  DD    A\n      AE20     0                          1  3  0  2\n*      %FS\n*     AE20     0        0        0        Bug-can't put before so put after-so can't make 2 fixed 10/28/12\n AE19 AE20   9.3    60.0  60.0-36.0       2 12  0 20\n AE18 AE19  24.5     0.0   0.0-90.0       6 10 25  0\n AE17 AE18   8.0   350.5 350.5 17.0       3  5  0  0\n AE16 AE17   6.7     0.0   0.0-90.0       3  5  6  1\n AE15 AE16  12.6    70.5  71.0-18.0       4  0  2  1\n AE14 AE15  10.0    21.5  20.0  6.0       5  5  0  3\n AE13 AE14  26.8   288.0 286.0-50.0       0  7 20  5\n*\n*SHORT CANYON AT THE BASE OF THE SECOND DROP\n AE12 AE13  20.7   236.0 236.0 34.0       3  5  4  4\n AE11 AE12  12.4   210.0 210.0 35.0       7  4  5  1\n AE10 AE13  25.7    40.0  40.0 -9.0       2  2  3  6\n*\n*AE10 AT JOE'S \" I LOVE MY WIFE TRAVERSE \"\n  AE9 AE10  17.8    32.5  31.0 23.0       4  5 20 15\n  AE1  AE9  13.7    82.0  82.0-13.0\n   A1  AE1  34.3    46.0  48.0-17.5\n*\n*SURVEY TO DOME NEAR THE ENTRANCE DOME (ABOVE THE SECOND DROP)\n  AD1 AE15   8.0   200.0 200.0  0.0       3  1  1  1\n  AD2  AD1  17.7   161.0 161.0  7.0       1  4 25  1\n  AD3  AD2  10.4   180.0 180.0 50.0       4  1 15  5\n *\nTRICKY TRAVERSE AND THEN FIRST SURVEY IN UPPER CROWLWAY\nDAN CROWL, KEITH ORTIZ, CHIP HOPPER, PETER QUICK, LARRY BEAN    14 FEB 1981\n *\nFI B  DD\n   A2   A1  48 10  292.0 110.0-42.0       5 10 35  5\n   A3   A2  12  5  333.5 153.5 35.0       3  1 15  5\n   A4   A3   4  2    0.0   0.0 90.0       3  1 10 10\n...</pre>\n *\n */\nexport default async function parseFrcsSurveyFile(\n  file: any, // eslint-disable-line @typescript-eslint/no-explicit-any\n  lines: AsyncIterable<string>\n): Promise<FrcsSurveyFile> {\n  let cave: string | null = null\n  let location: string | null = null\n  const trips: Array<FrcsTrip> = []\n  const errors: Array<SegmentParseError> = []\n\n  let tripName\n  let tripTeam\n  let tripDate\n  let inTripComment = true\n  let tripCommentStartLine = 1\n  let tripCommentEndLine = -1\n  let tripComment: Array<string> = []\n  let commentLines: Array<string> | null = null\n  let trip: FrcsTrip | null = null\n  let inBlockComment = false\n  let section\n\n  function getComment(): string | null {\n    if (!commentLines) return null\n    const comment = commentLines.join('\\n').trim()\n    commentLines = null\n    return comment || null\n  }\n\n  let lineNumber = 0\n\n  for await (const line of lines) {\n    let errored = false\n\n    const thisLineNumber = lineNumber\n\n    const error = (\n      message: string,\n      startColumn: number,\n      endColumn: number\n    ): void => {\n      errored = true\n      errors.push(\n        new SegmentParseError(\n          message,\n          new Segment({\n            value: line,\n            source: file,\n            startLine: thisLineNumber,\n            startCol: 0,\n          }).substring(startColumn, endColumn)\n        )\n      )\n    }\n\n    const validate = (\n      startColumn: number,\n      endColumn: number,\n      fieldName: string,\n      validator: (value: string) => boolean\n    ): string => {\n      const field = line.substring(startColumn, endColumn)\n      if (!validator(field)) {\n        error(\n          (field.trim() ? 'Invalid ' : 'Missing ') + fieldName,\n          startColumn,\n          endColumn\n        )\n      }\n      return field\n    }\n\n    lineNumber++\n\n    if (lineNumber === 1) {\n      const match = /^\\s*([^,]+)(,(.*))?/.exec(line)\n      if (match) {\n        cave = match[1].trim()\n        if (match[3]) {\n          location = match[3].trim()\n        }\n      }\n    }\n\n    if (line.charAt(0) === ' ' && line.charAt(1) === '*') {\n      inTripComment = !inTripComment\n      if (inTripComment) {\n        section = undefined\n        tripComment = []\n        tripCommentStartLine = lineNumber\n      } else {\n        tripCommentEndLine = lineNumber\n      }\n    } else if (inTripComment) {\n      if (lineNumber === tripCommentStartLine + 1) {\n        tripName = line && line.trim()\n      } else if (lineNumber === tripCommentStartLine + 2) {\n        const match = /^(.+?)\\s*[-.]\\s*(\\d+)\\/(\\d+)\\/(\\d+)$/.exec(\n          line && line.trim()\n        )\n        if (match) {\n          let k = 1\n          const team = match[k++]\n          const month = parseInt(match[k++])\n          const day = parseInt(match[k++])\n          const year = parseInt(match[k++])\n          tripDate = new Date(year < 70 ? year + 2000 : year, month - 1, day)\n          tripTeam = team.split(\n            team.indexOf(';') >= 0 ? /\\s*;\\s*/g : /\\s*,\\s*/g\n          )\n        }\n      } else if (lineNumber > 1) {\n        tripComment.push(line)\n      }\n      const match = /^\\*\\*\\*([^*])\\*\\*\\*/.exec(line)\n      if (match) {\n        section = match[1].trim()\n      }\n    } else if (line.charAt(0) === '*') {\n      inBlockComment = !inBlockComment\n      if (inBlockComment) commentLines = []\n      else if (commentLines) {\n        const part = line.substring(1).trim()\n        if (part && commentLines) commentLines.push(part)\n      }\n    } else if (inBlockComment) {\n      if (commentLines) commentLines.push(line)\n    } else if (lineNumber === tripCommentEndLine + 1) {\n      // FT CC DD\n      // 01234567\n      let distanceUnit = parseLengthUnit(line.slice(0, 2))\n      if (!distanceUnit) {\n        distanceUnit = Length.feet\n        error('Invalid distance unit', 0, 2)\n      }\n      let azimuthUnit = parseAngleUnit(line[6])\n      if (!azimuthUnit) {\n        azimuthUnit = Angle.degrees\n        error('Invalid azimuth unit', 6, 7)\n      }\n      let inclinationUnit = parseAngleUnit(line[7])\n      if (!inclinationUnit) {\n        inclinationUnit = Angle.degrees\n        error('Invalid inclination unit', 7, 8)\n      }\n      const backsightAzimuthCorrected = line[3] === 'C'\n      const backsightInclinationCorrected = line[4] === 'C'\n      const hasBacksightAzimuth = line[3] !== ' ' && line[3] !== '-'\n      const hasBacksightInclination = line[4] !== ' ' && line[4] !== '-'\n\n      if (!/[-CB ]/.test(line[3])) {\n        error('Invalid backsight azimuth type', 3, 4)\n      }\n      if (!/[-CB ]/.test(line[4])) {\n        error('Invalid backsight inclination type', 4, 5)\n      }\n\n      trip = {\n        header: {\n          name: tripName || '',\n          comment: (tripComment && tripComment.join('\\n')) || null,\n          section,\n          date: tripDate,\n          team: tripTeam,\n          distanceUnit,\n          azimuthUnit,\n          inclinationUnit,\n          backsightAzimuthCorrected,\n          backsightInclinationCorrected,\n          hasBacksightAzimuth,\n          hasBacksightInclination,\n        },\n        shots: [],\n      }\n      trips.push(trip)\n    } else if (trip) {\n      const {\n        shots,\n        header: { azimuthUnit, inclinationUnit },\n      } = trip\n      let { distanceUnit } = trip.header\n\n      const inches = distanceUnit === Length.inches\n      if (inches) distanceUnit = Length.feet\n\n      // rigorously check the values in all the columns to make sure this\n      // is really a survey shot line, just in case any stray comments are\n      // not properly delimited.\n\n      // from station name\n      if (!/\\S/.test(line.substring(5, 10))) continue\n      const fromStr = validate(5, 10, 'from station', isValidStation)\n\n      // Sadly I have found negative LRUD values in Chip's format and apparently\n      // his program doesn't fail on them, so I have to accept them here\n      // isValidOptFloat instead of isValidOptUFloat\n      const lStr = validate(40, 43, 'left', isValidOptFloat)\n      const rStr = validate(43, 46, 'right', isValidOptFloat)\n      const uStr = validate(46, 49, 'up', isValidOptFloat)\n      const dStr = validate(49, 52, 'down', isValidOptFloat)\n\n      if (errored) continue\n\n      const up = parseLrud(uStr, distanceUnit)\n      const down = parseLrud(dStr, distanceUnit)\n      const left = parseLrud(lStr, distanceUnit)\n      const right = parseLrud(rStr, distanceUnit)\n\n      // to station name\n      const toStr = line.substring(0, 5)\n      if (!toStr.trim()) {\n        const shot: FrcsShot = {\n          from: fromStr.trim(),\n          to: null,\n          kind: FrcsShotKind.Normal,\n          distance: new UnitizedNumber(0, distanceUnit),\n          frontsightAzimuth: null,\n          backsightAzimuth: null,\n          frontsightInclination: null,\n          backsightInclination: null,\n          fromLruds: {\n            left,\n            right,\n            up,\n            down,\n          },\n          excludeDistance: true,\n          comment: getComment(),\n        }\n        shots.push(shot)\n        continue\n      }\n      if (!isValidStation(toStr)) {\n        error('Invalid station name', 0, 5)\n      }\n\n      const fromLrudMatch = new RegExp(\n        `^\\\\s+${fromStr\n          .trim()\n          .replace(\n            /[.*+?^${}()|[\\]\\\\]/g,\n            '\\\\$&'\n          )}((\\\\s+(\\\\d+(\\\\.\\\\d*)?|\\\\.\\\\d+)){4})`\n      ).exec(line.substring(52))\n      let fromLruds\n      if (fromLrudMatch) {\n        const [left, right, up, down] = fromLrudMatch[1]\n          .trim()\n          .split(/\\s+/g)\n          .map(s => parseLrud(s, distanceUnit))\n        fromLruds = { left, right, up, down }\n      }\n\n      // azimuth and inclination\n      const azmFsStr = validate(19, 25, 'azimuth', isValidOptUFloat)\n      const azmBsStr = validate(25, 30, 'azimuth', isValidOptUFloat)\n      const incFsStr = line.substring(30, 35)\n      const incBsStr = line.substring(35, 40)\n\n      if (errored) continue\n\n      let kind: FrcsShotKind\n      let distance: UnitizedNumber<Length>\n      let horizontalDistance: UnitizedNumber<Length> | undefined\n      let verticalDistance: UnitizedNumber<Length> | undefined\n      let frontsightInclination: UnitizedNumber<Angle> | null\n      let backsightInclination: UnitizedNumber<Angle> | null\n      let excludeDistance: boolean\n      let isSplay: boolean\n\n      // parse distance\n      if (inches) {\n        const feetStr = line.substring(10, 14)\n        const inchesStr = line.substring(14, 17)\n        // feet and inches are not both optional\n        if (!isValidUInt(feetStr) && !isValidUInt(inchesStr)) {\n          const invalid = feetStr.trim() || inchesStr.trim()\n          error(invalid ? 'Invalid distance' : 'Missing distance', 10, 17)\n          continue\n        }\n\n        // sometimes inches are omitted, hence the || 0...I'm assuming it's possible\n        // for feet to be omitted as well\n        distance = Unitize.inches(parseFloat(inchesStr) || 0).add(\n          Unitize.feet(parseFloat(feetStr) || 0)\n        )\n\n        kind = parseKind(line[17])\n        // NOTE there are two columns around here that can contain a *.\n        // I think they might represent different values, but thisis confused by\n        // the fact that for ft/in shots, if there is a D or H flag it occupies the\n        // first column that can contain a * for decimal feet shots\n        excludeDistance = line[18] === '*' || line[18] === 's'\n        isSplay = line[18] === 's'\n      } else {\n        // decimal feet are not optional\n        const feetStr = validate(10, 16, 'distance', isValidUFloat)\n        distance = new UnitizedNumber(parseFloat(feetStr), distanceUnit)\n        kind = parseKind(line[16])\n        excludeDistance = line[17] === '*' || line[17] === 's'\n        isSplay = line[17] === 's'\n      }\n\n      if (kind !== FrcsShotKind.Normal) {\n        validate(30, 35, 'vertical-distance', isValidFloat)\n      }\n\n      // convert horizontal and diagonal shots to standard\n      // in this case incFs is the vertical offset between stations\n      // fortunately it appears we can always count on incFs being specified\n      // and incBs not being specified for these types of shots\n      if (kind === FrcsShotKind.Horizontal) {\n        // distance is horizontal offset and incFsStr is vertical offset\n        horizontalDistance = distance\n        const h = horizontalDistance.get(distanceUnit)\n        const v = parseFloat(incFsStr)\n        verticalDistance = new UnitizedNumber(v, distanceUnit)\n        distance = new UnitizedNumber(Math.sqrt(h * h + v * v), distanceUnit)\n        frontsightInclination = Angle.atan2(\n          verticalDistance,\n          horizontalDistance\n        )\n        backsightInclination = null\n      } else if (kind === FrcsShotKind.Diagonal) {\n        // distance is as usual, but incFsStr is vertical offset\n        const d = distance.get(distanceUnit)\n        const v = parseFloat(incFsStr)\n        verticalDistance = new UnitizedNumber(v, distanceUnit)\n        frontsightInclination = Angle.asin(v / d)\n        backsightInclination = null\n      } else {\n        // frontsight inclination\n        validate(30, 35, 'inclination', isValidOptInclination)\n        // frontsight inclination\n        validate(35, 40, 'inclination', isValidOptInclination)\n        frontsightInclination = parseNumber(incFsStr, inclinationUnit)\n        backsightInclination = parseNumber(incBsStr, inclinationUnit)\n      }\n      if (errored) continue\n\n      const frontsightAzimuth = parseAzimuth(azmFsStr, azimuthUnit)\n      const backsightAzimuth = parseAzimuth(azmBsStr, azimuthUnit)\n\n      if (!frontsightInclination && !backsightInclination) {\n        frontsightInclination = Unitize.degrees(0)\n      }\n\n      const shot: FrcsShot = {\n        from: fromStr.trim(),\n        to: toStr.trim(),\n        kind,\n        distance,\n        frontsightAzimuth,\n        backsightAzimuth,\n        frontsightInclination,\n        backsightInclination,\n        toLruds: {\n          left,\n          right,\n          up,\n          down,\n        },\n        excludeDistance,\n        comment: getComment(),\n      }\n      if (isSplay) shot.isSplay = true\n      if (fromLruds) shot.fromLruds = fromLruds\n      if (horizontalDistance) shot.horizontalDistance = horizontalDistance\n      if (verticalDistance) shot.verticalDistance = verticalDistance\n      shots.push(shot)\n    }\n  }\n\n  return {\n    cave,\n    location,\n    trips,\n    errors,\n  }\n}\n"]}
458
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_parseSegment","require","_unitized","_FrcsShot","parseNumber","s","unit","value","parseFloat","isNaN","UnitizedNumber","parseAzimuth","parsed","get","Angle","degrees","Unitize","parseKind","kind","FrcsShotKind","Horizontal","Diagonal","Normal","parseLengthUnit","Length","inches","feet","meters","parseAngleUnit","gradians","milsNATO","isValidStation","test","isValidUInt","isValidFloat","isValidOptFloat","isValidOptUFloat","isValidUFloat","isValidOptInclination","parseLrud","Number","isFinite","parseFrcsSurveyFile","file","lines","cave","location","trips","errors","tripName","tripTeam","tripDate","inTripComment","tripCommentStartLine","tripCommentEndLine","tripComment","commentLines","trip","inBlockComment","section","getComment","comment","join","trim","lineNumber","line","errored","thisLineNumber","error","message","startColumn","endColumn","push","SegmentParseError","Segment","source","startLine","startCol","substring","validate","fieldName","validator","field","match","exec","charAt","undefined","k","team","month","parseInt","day","year","Date","split","indexOf","part","distanceUnit","slice","azimuthUnit","inclinationUnit","backsightAzimuthCorrected","backsightInclinationCorrected","hasBacksightAzimuth","hasBacksightInclination","header","name","date","shots","fromStr","lStr","rStr","uStr","dStr","up","down","left","right","toStr","shot","from","to","distance","frontsightAzimuth","backsightAzimuth","frontsightInclination","backsightInclination","fromLruds","excludeDistance","fromLrudMatch","RegExp","replace","map","azmFsStr","azmBsStr","incFsStr","incBsStr","horizontalDistance","verticalDistance","isSplay","feetStr","inchesStr","invalid","add","h","v","Math","sqrt","atan2","d","asin","toLruds"],"sources":["../src/parseFrcsSurveyFile.ts"],"sourcesContent":["import { FrcsSurveyFile } from './FrcsSurveyFile'\nimport { FrcsTrip } from './FrcsTrip'\nimport { Segment, SegmentParseError } from 'parse-segment'\nimport {\n  Angle,\n  Length,\n  Unit,\n  UnitizedNumber,\n  UnitType,\n  Unitize,\n} from '@speleotica/unitized'\nimport { FrcsShot, FrcsShotKind } from './FrcsShot'\n\nfunction parseNumber<T extends UnitType<T>>(\n  s: string,\n  unit: Unit<T>\n): UnitizedNumber<T> | null {\n  const value = parseFloat(s)\n  if (isNaN(value)) return null\n  return new UnitizedNumber(value, unit)\n}\n\nfunction parseAzimuth(\n  s: string,\n  unit: Unit<Angle>\n): UnitizedNumber<Angle> | null {\n  const parsed = parseNumber(s, unit)\n  return parsed?.get(Angle.degrees) === 360 ? Unitize.degrees(0) : parsed\n}\n\nfunction parseKind(kind: string): FrcsShotKind {\n  switch (kind) {\n    case 'H':\n      return FrcsShotKind.Horizontal\n    case 'D':\n      return FrcsShotKind.Diagonal\n    default:\n      return FrcsShotKind.Normal\n  }\n}\n\nfunction parseLengthUnit(unit: string): Unit<Length> | null {\n  switch (unit) {\n    case 'FI':\n      return Length.inches\n    case 'FF':\n    case 'FT':\n      return Length.feet\n    case 'MT':\n    case 'M ':\n      return Length.meters\n  }\n  return null\n}\nfunction parseAngleUnit(unit: string): Unit<Angle> | null {\n  switch (unit) {\n    case 'D':\n      return Angle.degrees\n    case 'G':\n      return Angle.gradians\n    case 'M':\n      return Angle.milsNATO\n  }\n  return null\n}\n\n// determines if a cell contains a valid station name.\nfunction isValidStation(s: string): boolean {\n  return /^\\s*\\S+\\s*$/.test(s)\n}\n// determines if a cell contains a valid unsigned integer.\nfunction isValidUInt(s: string): boolean {\n  return /^\\s*[0-9]+\\s*$/.test(s)\n}\n// determines if a cell contains a valid float.\nfunction isValidFloat(s: string): boolean {\n  return /^\\s*[-+]?([0-9]+(\\.[0-9]*)?|\\.[0-9]+)\\s*$/.test(s)\n}\n// determines if a cell contains a valid unsigned float or whitespace.\nfunction isValidOptFloat(s: string): boolean {\n  return /^\\s*[-+]?[0-9]*(\\.[0-9]*)?\\s*$/.test(s)\n}\n// determines if a cell contains a valid unsigned float or whitespace.\nfunction isValidOptUFloat(s: string): boolean {\n  return /^\\s*[0-9]*(\\.[0-9]*)?\\s*$/.test(s)\n}\n// determines if a cell contains a valid unsigned float.\nfunction isValidUFloat(s: string): boolean {\n  return /^\\s*([0-9]+(\\.[0-9]*)?|\\.[0-9]+)\\s*$/.test(s)\n}\n// determines if a cell contains a valid inclination or whitespace.\nfunction isValidOptInclination(s: string): boolean {\n  return /^\\s*[-+]?[0-9]*(\\.[0-9]*)?\\s*$/.test(s)\n}\nfunction parseLrud<T extends UnitType<T>>(\n  s: string,\n  unit: Unit<Length>\n): UnitizedNumber<Length> | null {\n  const value = parseFloat(s)\n  return !Number.isFinite(value) || value < 0\n    ? null\n    : new UnitizedNumber(value, unit)\n}\n\n/**\n * Parses a raw cdata.fr survey file.  These look like so:\n *\n<pre>      Fisher Ridge Cave System, Hart Co., KY\nENTRANCE DROPS, JOE'S \"I LOVE MY WIFE TRAVERSE\", TRICKY TRAVERSE\nPETER QUICK, KEITH ORTIZ   -  2-15-81\nThis File has Crumps test connected.  11/20/12\n *\nFT C  DD    A\n      AE20     0                          1  3  0  2\n*      %FS\n*     AE20     0        0        0        Bug-can't put before so put after-so can't make 2 fixed 10/28/12\n AE19 AE20   9.3    60.0  60.0-36.0       2 12  0 20\n AE18 AE19  24.5     0.0   0.0-90.0       6 10 25  0\n AE17 AE18   8.0   350.5 350.5 17.0       3  5  0  0\n AE16 AE17   6.7     0.0   0.0-90.0       3  5  6  1\n AE15 AE16  12.6    70.5  71.0-18.0       4  0  2  1\n AE14 AE15  10.0    21.5  20.0  6.0       5  5  0  3\n AE13 AE14  26.8   288.0 286.0-50.0       0  7 20  5\n*\n*SHORT CANYON AT THE BASE OF THE SECOND DROP\n AE12 AE13  20.7   236.0 236.0 34.0       3  5  4  4\n AE11 AE12  12.4   210.0 210.0 35.0       7  4  5  1\n AE10 AE13  25.7    40.0  40.0 -9.0       2  2  3  6\n*\n*AE10 AT JOE'S \" I LOVE MY WIFE TRAVERSE \"\n  AE9 AE10  17.8    32.5  31.0 23.0       4  5 20 15\n  AE1  AE9  13.7    82.0  82.0-13.0\n   A1  AE1  34.3    46.0  48.0-17.5\n*\n*SURVEY TO DOME NEAR THE ENTRANCE DOME (ABOVE THE SECOND DROP)\n  AD1 AE15   8.0   200.0 200.0  0.0       3  1  1  1\n  AD2  AD1  17.7   161.0 161.0  7.0       1  4 25  1\n  AD3  AD2  10.4   180.0 180.0 50.0       4  1 15  5\n *\nTRICKY TRAVERSE AND THEN FIRST SURVEY IN UPPER CROWLWAY\nDAN CROWL, KEITH ORTIZ, CHIP HOPPER, PETER QUICK, LARRY BEAN    14 FEB 1981\n *\nFI B  DD\n   A2   A1  48 10  292.0 110.0-42.0       5 10 35  5\n   A3   A2  12  5  333.5 153.5 35.0       3  1 15  5\n   A4   A3   4  2    0.0   0.0 90.0       3  1 10 10\n...</pre>\n *\n */\nexport default async function parseFrcsSurveyFile(\n  file: any, // eslint-disable-line @typescript-eslint/no-explicit-any\n  lines: AsyncIterable<string>\n): Promise<FrcsSurveyFile> {\n  let cave: string | null = null\n  let location: string | null = null\n  const trips: Array<FrcsTrip> = []\n  const errors: Array<SegmentParseError> = []\n\n  let tripName\n  let tripTeam\n  let tripDate\n  let inTripComment = true\n  let tripCommentStartLine = 1\n  let tripCommentEndLine = -1\n  let tripComment: Array<string> = []\n  let commentLines: Array<string> | null = null\n  let trip: FrcsTrip | null = null\n  let inBlockComment = false\n  let section\n\n  function getComment(): string | null {\n    if (!commentLines) return null\n    const comment = commentLines.join('\\n').trim()\n    commentLines = null\n    return comment || null\n  }\n\n  let lineNumber = 0\n\n  for await (const line of lines) {\n    let errored = false\n\n    const thisLineNumber = lineNumber\n\n    const error = (\n      message: string,\n      startColumn: number,\n      endColumn: number\n    ): void => {\n      errored = true\n      errors.push(\n        new SegmentParseError(\n          message,\n          new Segment({\n            value: line,\n            source: file,\n            startLine: thisLineNumber,\n            startCol: 0,\n          }).substring(startColumn, endColumn)\n        )\n      )\n    }\n\n    const validate = (\n      startColumn: number,\n      endColumn: number,\n      fieldName: string,\n      validator: (value: string) => boolean\n    ): string => {\n      const field = line.substring(startColumn, endColumn)\n      if (!validator(field)) {\n        error(\n          (field.trim() ? 'Invalid ' : 'Missing ') + fieldName,\n          startColumn,\n          endColumn\n        )\n      }\n      return field\n    }\n\n    lineNumber++\n\n    if (lineNumber === 1) {\n      const match = /^\\s*([^,]+)(,(.*))?/.exec(line)\n      if (match) {\n        cave = match[1].trim()\n        if (match[3]) {\n          location = match[3].trim()\n        }\n      }\n    }\n\n    if (line.charAt(0) === ' ' && line.charAt(1) === '*') {\n      inTripComment = !inTripComment\n      if (inTripComment) {\n        section = undefined\n        tripComment = []\n        tripCommentStartLine = lineNumber\n      } else {\n        tripCommentEndLine = lineNumber\n      }\n    } else if (inTripComment) {\n      if (lineNumber === tripCommentStartLine + 1) {\n        tripName = line && line.trim()\n      } else if (lineNumber === tripCommentStartLine + 2) {\n        const match = /^(.+?)\\s*[-.]\\s*(\\d+)\\/(\\d+)\\/(\\d+)$/.exec(\n          line && line.trim()\n        )\n        if (match) {\n          let k = 1\n          const team = match[k++]\n          const month = parseInt(match[k++])\n          const day = parseInt(match[k++])\n          const year = parseInt(match[k++])\n          tripDate = new Date(year < 70 ? year + 2000 : year, month - 1, day)\n          tripTeam = team.split(\n            team.indexOf(';') >= 0 ? /\\s*;\\s*/g : /\\s*,\\s*/g\n          )\n        }\n      } else if (lineNumber > 1) {\n        tripComment.push(line)\n      }\n      const match = /^\\*\\*\\*([^*])\\*\\*\\*/.exec(line)\n      if (match) {\n        section = match[1].trim()\n      }\n    } else if (line.charAt(0) === '*') {\n      inBlockComment = !inBlockComment\n      if (inBlockComment) commentLines = []\n      else if (commentLines) {\n        const part = line.substring(1).trim()\n        if (part && commentLines) commentLines.push(part)\n      }\n    } else if (inBlockComment) {\n      if (commentLines) commentLines.push(line)\n    } else if (lineNumber === tripCommentEndLine + 1) {\n      // FT CC DD\n      // 01234567\n      let distanceUnit = parseLengthUnit(line.slice(0, 2))\n      if (!distanceUnit) {\n        distanceUnit = Length.feet\n        error('Invalid distance unit', 0, 2)\n      }\n      let azimuthUnit = parseAngleUnit(line[6])\n      if (!azimuthUnit) {\n        azimuthUnit = Angle.degrees\n        error('Invalid azimuth unit', 6, 7)\n      }\n      let inclinationUnit = parseAngleUnit(line[7])\n      if (!inclinationUnit) {\n        inclinationUnit = Angle.degrees\n        error('Invalid inclination unit', 7, 8)\n      }\n      const backsightAzimuthCorrected = line[3] === 'C'\n      const backsightInclinationCorrected = line[4] === 'C'\n      const hasBacksightAzimuth = line[3] !== ' ' && line[3] !== '-'\n      const hasBacksightInclination = line[4] !== ' ' && line[4] !== '-'\n\n      if (!/[-CB ]/.test(line[3])) {\n        error('Invalid backsight azimuth type', 3, 4)\n      }\n      if (!/[-CB ]/.test(line[4])) {\n        error('Invalid backsight inclination type', 4, 5)\n      }\n\n      trip = {\n        header: {\n          name: tripName || '',\n          comment: (tripComment && tripComment.join('\\n')) || null,\n          section,\n          date: tripDate,\n          team: tripTeam,\n          distanceUnit,\n          azimuthUnit,\n          inclinationUnit,\n          backsightAzimuthCorrected,\n          backsightInclinationCorrected,\n          hasBacksightAzimuth,\n          hasBacksightInclination,\n        },\n        shots: [],\n      }\n      trips.push(trip)\n    } else if (trip) {\n      const {\n        shots,\n        header: { azimuthUnit, inclinationUnit },\n      } = trip\n      let { distanceUnit } = trip.header\n\n      const inches = distanceUnit === Length.inches\n      if (inches) distanceUnit = Length.feet\n\n      // rigorously check the values in all the columns to make sure this\n      // is really a survey shot line, just in case any stray comments are\n      // not properly delimited.\n\n      // from station name\n      if (!/\\S/.test(line.substring(5, 10))) continue\n      const fromStr = validate(5, 10, 'from station', isValidStation)\n\n      // Sadly I have found negative LRUD values in Chip's format and apparently\n      // his program doesn't fail on them, so I have to accept them here\n      // isValidOptFloat instead of isValidOptUFloat\n      const lStr = validate(40, 43, 'left', isValidOptFloat)\n      const rStr = validate(43, 46, 'right', isValidOptFloat)\n      const uStr = validate(46, 49, 'up', isValidOptFloat)\n      const dStr = validate(49, 52, 'down', isValidOptFloat)\n\n      if (errored) continue\n\n      const up = parseLrud(uStr, distanceUnit)\n      const down = parseLrud(dStr, distanceUnit)\n      const left = parseLrud(lStr, distanceUnit)\n      const right = parseLrud(rStr, distanceUnit)\n\n      // to station name\n      const toStr = line.substring(0, 5)\n      if (!toStr.trim()) {\n        const shot: FrcsShot = {\n          from: fromStr.trim(),\n          to: null,\n          kind: FrcsShotKind.Normal,\n          distance: new UnitizedNumber(0, distanceUnit),\n          frontsightAzimuth: null,\n          backsightAzimuth: null,\n          frontsightInclination: null,\n          backsightInclination: null,\n          fromLruds: {\n            left,\n            right,\n            up,\n            down,\n          },\n          excludeDistance: true,\n          comment: getComment(),\n        }\n        shots.push(shot)\n        continue\n      }\n      if (!isValidStation(toStr)) {\n        error('Invalid station name', 0, 5)\n      }\n\n      const fromLrudMatch = new RegExp(\n        `^\\\\s+${fromStr\n          .trim()\n          .replace(\n            /[.*+?^${}()|[\\]\\\\]/g,\n            '\\\\$&'\n          )}((\\\\s+(\\\\d+(\\\\.\\\\d*)?|\\\\.\\\\d+)){4})`\n      ).exec(line.substring(52))\n      let fromLruds\n      if (fromLrudMatch) {\n        const [left, right, up, down] = fromLrudMatch[1]\n          .trim()\n          .split(/\\s+/g)\n          .map(s => parseLrud(s, distanceUnit))\n        fromLruds = { left, right, up, down }\n      }\n\n      // azimuth and inclination\n      const azmFsStr = validate(19, 25, 'azimuth', isValidOptUFloat)\n      const azmBsStr = validate(25, 30, 'azimuth', isValidOptUFloat)\n      const incFsStr = line.substring(30, 35)\n      const incBsStr = line.substring(35, 40)\n\n      if (errored) continue\n\n      let kind: FrcsShotKind\n      let distance: UnitizedNumber<Length>\n      let horizontalDistance: UnitizedNumber<Length> | undefined\n      let verticalDistance: UnitizedNumber<Length> | undefined\n      let frontsightInclination: UnitizedNumber<Angle> | null\n      let backsightInclination: UnitizedNumber<Angle> | null\n      let excludeDistance: boolean\n      let isSplay: boolean\n\n      // parse distance\n      if (inches) {\n        const feetStr = line.substring(10, 14)\n        const inchesStr = line.substring(14, 17)\n        // feet and inches are not both optional\n        if (!isValidUInt(feetStr) && !isValidUInt(inchesStr)) {\n          const invalid = feetStr.trim() || inchesStr.trim()\n          error(invalid ? 'Invalid distance' : 'Missing distance', 10, 17)\n          continue\n        }\n\n        // sometimes inches are omitted, hence the || 0...I'm assuming it's possible\n        // for feet to be omitted as well\n        distance = Unitize.inches(parseFloat(inchesStr) || 0).add(\n          Unitize.feet(parseFloat(feetStr) || 0)\n        )\n\n        kind = parseKind(line[17])\n        // NOTE there are two columns around here that can contain a *.\n        // I think they might represent different values, but thisis confused by\n        // the fact that for ft/in shots, if there is a D or H flag it occupies the\n        // first column that can contain a * for decimal feet shots\n        excludeDistance = line[18] === '*' || line[18] === 's'\n        isSplay = line[18] === 's'\n      } else {\n        // decimal feet are not optional\n        const feetStr = validate(10, 16, 'distance', isValidUFloat)\n        distance = new UnitizedNumber(parseFloat(feetStr), distanceUnit)\n        kind = parseKind(line[16])\n        excludeDistance = line[17] === '*' || line[17] === 's'\n        isSplay = line[17] === 's'\n      }\n\n      if (kind !== FrcsShotKind.Normal) {\n        validate(30, 35, 'vertical-distance', isValidFloat)\n      }\n\n      // convert horizontal and diagonal shots to standard\n      // in this case incFs is the vertical offset between stations\n      // fortunately it appears we can always count on incFs being specified\n      // and incBs not being specified for these types of shots\n      if (kind === FrcsShotKind.Horizontal) {\n        // distance is horizontal offset and incFsStr is vertical offset\n        horizontalDistance = distance\n        const h = horizontalDistance.get(distanceUnit)\n        const v = parseFloat(incFsStr)\n        verticalDistance = new UnitizedNumber(v, distanceUnit)\n        distance = new UnitizedNumber(Math.sqrt(h * h + v * v), distanceUnit)\n        frontsightInclination = Angle.atan2(\n          verticalDistance,\n          horizontalDistance\n        )\n        backsightInclination = null\n      } else if (kind === FrcsShotKind.Diagonal) {\n        // distance is as usual, but incFsStr is vertical offset\n        const d = distance.get(distanceUnit)\n        const v = parseFloat(incFsStr)\n        verticalDistance = new UnitizedNumber(v, distanceUnit)\n        frontsightInclination = Angle.asin(v / d)\n        backsightInclination = null\n      } else {\n        // frontsight inclination\n        validate(30, 35, 'inclination', isValidOptInclination)\n        // frontsight inclination\n        validate(35, 40, 'inclination', isValidOptInclination)\n        frontsightInclination = parseNumber(incFsStr, inclinationUnit)\n        backsightInclination = parseNumber(incBsStr, inclinationUnit)\n      }\n      if (errored) continue\n\n      const frontsightAzimuth = parseAzimuth(azmFsStr, azimuthUnit)\n      const backsightAzimuth = parseAzimuth(azmBsStr, azimuthUnit)\n\n      if (!frontsightInclination && !backsightInclination) {\n        frontsightInclination = Unitize.degrees(0)\n      }\n\n      const shot: FrcsShot = {\n        from: fromStr.trim(),\n        to: toStr.trim(),\n        kind,\n        distance,\n        frontsightAzimuth,\n        backsightAzimuth,\n        frontsightInclination,\n        backsightInclination,\n        toLruds: {\n          left,\n          right,\n          up,\n          down,\n        },\n        excludeDistance,\n        comment: getComment(),\n      }\n      if (isSplay) shot.isSplay = true\n      if (fromLruds) shot.fromLruds = fromLruds\n      if (horizontalDistance) shot.horizontalDistance = horizontalDistance\n      if (verticalDistance) shot.verticalDistance = verticalDistance\n      shots.push(shot)\n    }\n  }\n\n  return {\n    cave,\n    location,\n    trips,\n    errors,\n  }\n}\n"],"mappings":";;;;;;AAEA,IAAAA,aAAA,GAAAC,OAAA;AACA,IAAAC,SAAA,GAAAD,OAAA;AAQA,IAAAE,SAAA,GAAAF,OAAA;AAEA,SAASG,WAAWA,CAClBC,CAAS,EACTC,IAAa,EACa;EAC1B,MAAMC,KAAK,GAAGC,UAAU,CAACH,CAAC,CAAC;EAC3B,IAAII,KAAK,CAACF,KAAK,CAAC,EAAE,OAAO,IAAI;EAC7B,OAAO,IAAIG,wBAAc,CAACH,KAAK,EAAED,IAAI,CAAC;AACxC;AAEA,SAASK,YAAYA,CACnBN,CAAS,EACTC,IAAiB,EACa;EAC9B,MAAMM,MAAM,GAAGR,WAAW,CAACC,CAAC,EAAEC,IAAI,CAAC;EACnC,OAAO,CAAAM,MAAM,aAANA,MAAM,uBAANA,MAAM,CAAEC,GAAG,CAACC,eAAK,CAACC,OAAO,CAAC,MAAK,GAAG,GAAGC,iBAAO,CAACD,OAAO,CAAC,CAAC,CAAC,GAAGH,MAAM;AACzE;AAEA,SAASK,SAASA,CAACC,IAAY,EAAgB;EAC7C,QAAQA,IAAI;IACV,KAAK,GAAG;MACN,OAAOC,sBAAY,CAACC,UAAU;IAChC,KAAK,GAAG;MACN,OAAOD,sBAAY,CAACE,QAAQ;IAC9B;MACE,OAAOF,sBAAY,CAACG,MAAM;EAC9B;AACF;AAEA,SAASC,eAAeA,CAACjB,IAAY,EAAuB;EAC1D,QAAQA,IAAI;IACV,KAAK,IAAI;MACP,OAAOkB,gBAAM,CAACC,MAAM;IACtB,KAAK,IAAI;IACT,KAAK,IAAI;MACP,OAAOD,gBAAM,CAACE,IAAI;IACpB,KAAK,IAAI;IACT,KAAK,IAAI;MACP,OAAOF,gBAAM,CAACG,MAAM;EACxB;EACA,OAAO,IAAI;AACb;AACA,SAASC,cAAcA,CAACtB,IAAY,EAAsB;EACxD,QAAQA,IAAI;IACV,KAAK,GAAG;MACN,OAAOQ,eAAK,CAACC,OAAO;IACtB,KAAK,GAAG;MACN,OAAOD,eAAK,CAACe,QAAQ;IACvB,KAAK,GAAG;MACN,OAAOf,eAAK,CAACgB,QAAQ;EACzB;EACA,OAAO,IAAI;AACb;;AAEA;AACA,SAASC,cAAcA,CAAC1B,CAAS,EAAW;EAC1C,OAAO,aAAa,CAAC2B,IAAI,CAAC3B,CAAC,CAAC;AAC9B;AACA;AACA,SAAS4B,WAAWA,CAAC5B,CAAS,EAAW;EACvC,OAAO,gBAAgB,CAAC2B,IAAI,CAAC3B,CAAC,CAAC;AACjC;AACA;AACA,SAAS6B,YAAYA,CAAC7B,CAAS,EAAW;EACxC,OAAO,2CAA2C,CAAC2B,IAAI,CAAC3B,CAAC,CAAC;AAC5D;AACA;AACA,SAAS8B,eAAeA,CAAC9B,CAAS,EAAW;EAC3C,OAAO,gCAAgC,CAAC2B,IAAI,CAAC3B,CAAC,CAAC;AACjD;AACA;AACA,SAAS+B,gBAAgBA,CAAC/B,CAAS,EAAW;EAC5C,OAAO,2BAA2B,CAAC2B,IAAI,CAAC3B,CAAC,CAAC;AAC5C;AACA;AACA,SAASgC,aAAaA,CAAChC,CAAS,EAAW;EACzC,OAAO,sCAAsC,CAAC2B,IAAI,CAAC3B,CAAC,CAAC;AACvD;AACA;AACA,SAASiC,qBAAqBA,CAACjC,CAAS,EAAW;EACjD,OAAO,gCAAgC,CAAC2B,IAAI,CAAC3B,CAAC,CAAC;AACjD;AACA,SAASkC,SAASA,CAChBlC,CAAS,EACTC,IAAkB,EACa;EAC/B,MAAMC,KAAK,GAAGC,UAAU,CAACH,CAAC,CAAC;EAC3B,OAAO,CAACmC,MAAM,CAACC,QAAQ,CAAClC,KAAK,CAAC,IAAIA,KAAK,GAAG,CAAC,GACvC,IAAI,GACJ,IAAIG,wBAAc,CAACH,KAAK,EAAED,IAAI,CAAC;AACrC;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACe,eAAeoC,mBAAmBA,CAC/CC,IAAS;AAAE;AACXC,KAA4B,EACH;EACzB,IAAIC,IAAmB,GAAG,IAAI;EAC9B,IAAIC,QAAuB,GAAG,IAAI;EAClC,MAAMC,KAAsB,GAAG,EAAE;EACjC,MAAMC,MAAgC,GAAG,EAAE;EAE3C,IAAIC,QAAQ;EACZ,IAAIC,QAAQ;EACZ,IAAIC,QAAQ;EACZ,IAAIC,aAAa,GAAG,IAAI;EACxB,IAAIC,oBAAoB,GAAG,CAAC;EAC5B,IAAIC,kBAAkB,GAAG,CAAC,CAAC;EAC3B,IAAIC,WAA0B,GAAG,EAAE;EACnC,IAAIC,YAAkC,GAAG,IAAI;EAC7C,IAAIC,IAAqB,GAAG,IAAI;EAChC,IAAIC,cAAc,GAAG,KAAK;EAC1B,IAAIC,OAAO;EAEX,SAASC,UAAUA,CAAA,EAAkB;IACnC,IAAI,CAACJ,YAAY,EAAE,OAAO,IAAI;IAC9B,MAAMK,OAAO,GAAGL,YAAY,CAACM,IAAI,CAAC,IAAI,CAAC,CAACC,IAAI,CAAC,CAAC;IAC9CP,YAAY,GAAG,IAAI;IACnB,OAAOK,OAAO,IAAI,IAAI;EACxB;EAEA,IAAIG,UAAU,GAAG,CAAC;EAElB,WAAW,MAAMC,IAAI,IAAIrB,KAAK,EAAE;IAC9B,IAAIsB,OAAO,GAAG,KAAK;IAEnB,MAAMC,cAAc,GAAGH,UAAU;IAEjC,MAAMI,KAAK,GAAGA,CACZC,OAAe,EACfC,WAAmB,EACnBC,SAAiB,KACR;MACTL,OAAO,GAAG,IAAI;MACdlB,MAAM,CAACwB,IAAI,CACT,IAAIC,+BAAiB,CACnBJ,OAAO,EACP,IAAIK,qBAAO,CAAC;QACVnE,KAAK,EAAE0D,IAAI;QACXU,MAAM,EAAEhC,IAAI;QACZiC,SAAS,EAAET,cAAc;QACzBU,QAAQ,EAAE;MACZ,CAAC,CAAC,CAACC,SAAS,CAACR,WAAW,EAAEC,SAAS,CACrC,CACF,CAAC;IACH,CAAC;IAED,MAAMQ,QAAQ,GAAGA,CACfT,WAAmB,EACnBC,SAAiB,EACjBS,SAAiB,EACjBC,SAAqC,KAC1B;MACX,MAAMC,KAAK,GAAGjB,IAAI,CAACa,SAAS,CAACR,WAAW,EAAEC,SAAS,CAAC;MACpD,IAAI,CAACU,SAAS,CAACC,KAAK,CAAC,EAAE;QACrBd,KAAK,CACH,CAACc,KAAK,CAACnB,IAAI,CAAC,CAAC,GAAG,UAAU,GAAG,UAAU,IAAIiB,SAAS,EACpDV,WAAW,EACXC,SACF,CAAC;MACH;MACA,OAAOW,KAAK;IACd,CAAC;IAEDlB,UAAU,EAAE;IAEZ,IAAIA,UAAU,KAAK,CAAC,EAAE;MACpB,MAAMmB,KAAK,GAAG,qBAAqB,CAACC,IAAI,CAACnB,IAAI,CAAC;MAC9C,IAAIkB,KAAK,EAAE;QACTtC,IAAI,GAAGsC,KAAK,CAAC,CAAC,CAAC,CAACpB,IAAI,CAAC,CAAC;QACtB,IAAIoB,KAAK,CAAC,CAAC,CAAC,EAAE;UACZrC,QAAQ,GAAGqC,KAAK,CAAC,CAAC,CAAC,CAACpB,IAAI,CAAC,CAAC;QAC5B;MACF;IACF;IAEA,IAAIE,IAAI,CAACoB,MAAM,CAAC,CAAC,CAAC,KAAK,GAAG,IAAIpB,IAAI,CAACoB,MAAM,CAAC,CAAC,CAAC,KAAK,GAAG,EAAE;MACpDjC,aAAa,GAAG,CAACA,aAAa;MAC9B,IAAIA,aAAa,EAAE;QACjBO,OAAO,GAAG2B,SAAS;QACnB/B,WAAW,GAAG,EAAE;QAChBF,oBAAoB,GAAGW,UAAU;MACnC,CAAC,MAAM;QACLV,kBAAkB,GAAGU,UAAU;MACjC;IACF,CAAC,MAAM,IAAIZ,aAAa,EAAE;MACxB,IAAIY,UAAU,KAAKX,oBAAoB,GAAG,CAAC,EAAE;QAC3CJ,QAAQ,GAAGgB,IAAI,IAAIA,IAAI,CAACF,IAAI,CAAC,CAAC;MAChC,CAAC,MAAM,IAAIC,UAAU,KAAKX,oBAAoB,GAAG,CAAC,EAAE;QAClD,MAAM8B,KAAK,GAAG,sCAAsC,CAACC,IAAI,CACvDnB,IAAI,IAAIA,IAAI,CAACF,IAAI,CAAC,CACpB,CAAC;QACD,IAAIoB,KAAK,EAAE;UACT,IAAII,CAAC,GAAG,CAAC;UACT,MAAMC,IAAI,GAAGL,KAAK,CAACI,CAAC,EAAE,CAAC;UACvB,MAAME,KAAK,GAAGC,QAAQ,CAACP,KAAK,CAACI,CAAC,EAAE,CAAC,CAAC;UAClC,MAAMI,GAAG,GAAGD,QAAQ,CAACP,KAAK,CAACI,CAAC,EAAE,CAAC,CAAC;UAChC,MAAMK,IAAI,GAAGF,QAAQ,CAACP,KAAK,CAACI,CAAC,EAAE,CAAC,CAAC;UACjCpC,QAAQ,GAAG,IAAI0C,IAAI,CAACD,IAAI,GAAG,EAAE,GAAGA,IAAI,GAAG,IAAI,GAAGA,IAAI,EAAEH,KAAK,GAAG,CAAC,EAAEE,GAAG,CAAC;UACnEzC,QAAQ,GAAGsC,IAAI,CAACM,KAAK,CACnBN,IAAI,CAACO,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,GAAG,UAAU,GAAG,UACxC,CAAC;QACH;MACF,CAAC,MAAM,IAAI/B,UAAU,GAAG,CAAC,EAAE;QACzBT,WAAW,CAACiB,IAAI,CAACP,IAAI,CAAC;MACxB;MACA,MAAMkB,KAAK,GAAG,qBAAqB,CAACC,IAAI,CAACnB,IAAI,CAAC;MAC9C,IAAIkB,KAAK,EAAE;QACTxB,OAAO,GAAGwB,KAAK,CAAC,CAAC,CAAC,CAACpB,IAAI,CAAC,CAAC;MAC3B;IACF,CAAC,MAAM,IAAIE,IAAI,CAACoB,MAAM,CAAC,CAAC,CAAC,KAAK,GAAG,EAAE;MACjC3B,cAAc,GAAG,CAACA,cAAc;MAChC,IAAIA,cAAc,EAAEF,YAAY,GAAG,EAAE,MAChC,IAAIA,YAAY,EAAE;QACrB,MAAMwC,IAAI,GAAG/B,IAAI,CAACa,SAAS,CAAC,CAAC,CAAC,CAACf,IAAI,CAAC,CAAC;QACrC,IAAIiC,IAAI,IAAIxC,YAAY,EAAEA,YAAY,CAACgB,IAAI,CAACwB,IAAI,CAAC;MACnD;IACF,CAAC,MAAM,IAAItC,cAAc,EAAE;MACzB,IAAIF,YAAY,EAAEA,YAAY,CAACgB,IAAI,CAACP,IAAI,CAAC;IAC3C,CAAC,MAAM,IAAID,UAAU,KAAKV,kBAAkB,GAAG,CAAC,EAAE;MAChD;MACA;MACA,IAAI2C,YAAY,GAAG1E,eAAe,CAAC0C,IAAI,CAACiC,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC;MACpD,IAAI,CAACD,YAAY,EAAE;QACjBA,YAAY,GAAGzE,gBAAM,CAACE,IAAI;QAC1B0C,KAAK,CAAC,uBAAuB,EAAE,CAAC,EAAE,CAAC,CAAC;MACtC;MACA,IAAI+B,WAAW,GAAGvE,cAAc,CAACqC,IAAI,CAAC,CAAC,CAAC,CAAC;MACzC,IAAI,CAACkC,WAAW,EAAE;QAChBA,WAAW,GAAGrF,eAAK,CAACC,OAAO;QAC3BqD,KAAK,CAAC,sBAAsB,EAAE,CAAC,EAAE,CAAC,CAAC;MACrC;MACA,IAAIgC,eAAe,GAAGxE,cAAc,CAACqC,IAAI,CAAC,CAAC,CAAC,CAAC;MAC7C,IAAI,CAACmC,eAAe,EAAE;QACpBA,eAAe,GAAGtF,eAAK,CAACC,OAAO;QAC/BqD,KAAK,CAAC,0BAA0B,EAAE,CAAC,EAAE,CAAC,CAAC;MACzC;MACA,MAAMiC,yBAAyB,GAAGpC,IAAI,CAAC,CAAC,CAAC,KAAK,GAAG;MACjD,MAAMqC,6BAA6B,GAAGrC,IAAI,CAAC,CAAC,CAAC,KAAK,GAAG;MACrD,MAAMsC,mBAAmB,GAAGtC,IAAI,CAAC,CAAC,CAAC,KAAK,GAAG,IAAIA,IAAI,CAAC,CAAC,CAAC,KAAK,GAAG;MAC9D,MAAMuC,uBAAuB,GAAGvC,IAAI,CAAC,CAAC,CAAC,KAAK,GAAG,IAAIA,IAAI,CAAC,CAAC,CAAC,KAAK,GAAG;MAElE,IAAI,CAAC,QAAQ,CAACjC,IAAI,CAACiC,IAAI,CAAC,CAAC,CAAC,CAAC,EAAE;QAC3BG,KAAK,CAAC,gCAAgC,EAAE,CAAC,EAAE,CAAC,CAAC;MAC/C;MACA,IAAI,CAAC,QAAQ,CAACpC,IAAI,CAACiC,IAAI,CAAC,CAAC,CAAC,CAAC,EAAE;QAC3BG,KAAK,CAAC,oCAAoC,EAAE,CAAC,EAAE,CAAC,CAAC;MACnD;MAEAX,IAAI,GAAG;QACLgD,MAAM,EAAE;UACNC,IAAI,EAAEzD,QAAQ,IAAI,EAAE;UACpBY,OAAO,EAAGN,WAAW,IAAIA,WAAW,CAACO,IAAI,CAAC,IAAI,CAAC,IAAK,IAAI;UACxDH,OAAO;UACPgD,IAAI,EAAExD,QAAQ;UACdqC,IAAI,EAAEtC,QAAQ;UACd+C,YAAY;UACZE,WAAW;UACXC,eAAe;UACfC,yBAAyB;UACzBC,6BAA6B;UAC7BC,mBAAmB;UACnBC;QACF,CAAC;QACDI,KAAK,EAAE;MACT,CAAC;MACD7D,KAAK,CAACyB,IAAI,CAACf,IAAI,CAAC;IAClB,CAAC,MAAM,IAAIA,IAAI,EAAE;MACf,MAAM;QACJmD,KAAK;QACLH,MAAM,EAAE;UAAEN,WAAW;UAAEC;QAAgB;MACzC,CAAC,GAAG3C,IAAI;MACR,IAAI;QAAEwC;MAAa,CAAC,GAAGxC,IAAI,CAACgD,MAAM;MAElC,MAAMhF,MAAM,GAAGwE,YAAY,KAAKzE,gBAAM,CAACC,MAAM;MAC7C,IAAIA,MAAM,EAAEwE,YAAY,GAAGzE,gBAAM,CAACE,IAAI;;MAEtC;MACA;MACA;;MAEA;MACA,IAAI,CAAC,IAAI,CAACM,IAAI,CAACiC,IAAI,CAACa,SAAS,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,EAAE;MACvC,MAAM+B,OAAO,GAAG9B,QAAQ,CAAC,CAAC,EAAE,EAAE,EAAE,cAAc,EAAEhD,cAAc,CAAC;;MAE/D;MACA;MACA;MACA,MAAM+E,IAAI,GAAG/B,QAAQ,CAAC,EAAE,EAAE,EAAE,EAAE,MAAM,EAAE5C,eAAe,CAAC;MACtD,MAAM4E,IAAI,GAAGhC,QAAQ,CAAC,EAAE,EAAE,EAAE,EAAE,OAAO,EAAE5C,eAAe,CAAC;MACvD,MAAM6E,IAAI,GAAGjC,QAAQ,CAAC,EAAE,EAAE,EAAE,EAAE,IAAI,EAAE5C,eAAe,CAAC;MACpD,MAAM8E,IAAI,GAAGlC,QAAQ,CAAC,EAAE,EAAE,EAAE,EAAE,MAAM,EAAE5C,eAAe,CAAC;MAEtD,IAAI+B,OAAO,EAAE;MAEb,MAAMgD,EAAE,GAAG3E,SAAS,CAACyE,IAAI,EAAEf,YAAY,CAAC;MACxC,MAAMkB,IAAI,GAAG5E,SAAS,CAAC0E,IAAI,EAAEhB,YAAY,CAAC;MAC1C,MAAMmB,IAAI,GAAG7E,SAAS,CAACuE,IAAI,EAAEb,YAAY,CAAC;MAC1C,MAAMoB,KAAK,GAAG9E,SAAS,CAACwE,IAAI,EAAEd,YAAY,CAAC;;MAE3C;MACA,MAAMqB,KAAK,GAAGrD,IAAI,CAACa,SAAS,CAAC,CAAC,EAAE,CAAC,CAAC;MAClC,IAAI,CAACwC,KAAK,CAACvD,IAAI,CAAC,CAAC,EAAE;QACjB,MAAMwD,IAAc,GAAG;UACrBC,IAAI,EAAEX,OAAO,CAAC9C,IAAI,CAAC,CAAC;UACpB0D,EAAE,EAAE,IAAI;UACRvG,IAAI,EAAEC,sBAAY,CAACG,MAAM;UACzBoG,QAAQ,EAAE,IAAIhH,wBAAc,CAAC,CAAC,EAAEuF,YAAY,CAAC;UAC7C0B,iBAAiB,EAAE,IAAI;UACvBC,gBAAgB,EAAE,IAAI;UACtBC,qBAAqB,EAAE,IAAI;UAC3BC,oBAAoB,EAAE,IAAI;UAC1BC,SAAS,EAAE;YACTX,IAAI;YACJC,KAAK;YACLH,EAAE;YACFC;UACF,CAAC;UACDa,eAAe,EAAE,IAAI;UACrBnE,OAAO,EAAED,UAAU,CAAC;QACtB,CAAC;QACDgD,KAAK,CAACpC,IAAI,CAAC+C,IAAI,CAAC;QAChB;MACF;MACA,IAAI,CAACxF,cAAc,CAACuF,KAAK,CAAC,EAAE;QAC1BlD,KAAK,CAAC,sBAAsB,EAAE,CAAC,EAAE,CAAC,CAAC;MACrC;MAEA,MAAM6D,aAAa,GAAG,IAAIC,MAAM,CAC7B,QAAOrB,OAAO,CACZ9C,IAAI,CAAC,CAAC,CACNoE,OAAO,CACN,qBAAqB,EACrB,MACF,CAAE,qCACN,CAAC,CAAC/C,IAAI,CAACnB,IAAI,CAACa,SAAS,CAAC,EAAE,CAAC,CAAC;MAC1B,IAAIiD,SAAS;MACb,IAAIE,aAAa,EAAE;QACjB,MAAM,CAACb,IAAI,EAAEC,KAAK,EAAEH,EAAE,EAAEC,IAAI,CAAC,GAAGc,aAAa,CAAC,CAAC,CAAC,CAC7ClE,IAAI,CAAC,CAAC,CACN+B,KAAK,CAAC,MAAM,CAAC,CACbsC,GAAG,CAAC/H,CAAC,IAAIkC,SAAS,CAAClC,CAAC,EAAE4F,YAAY,CAAC,CAAC;QACvC8B,SAAS,GAAG;UAAEX,IAAI;UAAEC,KAAK;UAAEH,EAAE;UAAEC;QAAK,CAAC;MACvC;;MAEA;MACA,MAAMkB,QAAQ,GAAGtD,QAAQ,CAAC,EAAE,EAAE,EAAE,EAAE,SAAS,EAAE3C,gBAAgB,CAAC;MAC9D,MAAMkG,QAAQ,GAAGvD,QAAQ,CAAC,EAAE,EAAE,EAAE,EAAE,SAAS,EAAE3C,gBAAgB,CAAC;MAC9D,MAAMmG,QAAQ,GAAGtE,IAAI,CAACa,SAAS,CAAC,EAAE,EAAE,EAAE,CAAC;MACvC,MAAM0D,QAAQ,GAAGvE,IAAI,CAACa,SAAS,CAAC,EAAE,EAAE,EAAE,CAAC;MAEvC,IAAIZ,OAAO,EAAE;MAEb,IAAIhD,IAAkB;MACtB,IAAIwG,QAAgC;MACpC,IAAIe,kBAAsD;MAC1D,IAAIC,gBAAoD;MACxD,IAAIb,qBAAmD;MACvD,IAAIC,oBAAkD;MACtD,IAAIE,eAAwB;MAC5B,IAAIW,OAAgB;;MAEpB;MACA,IAAIlH,MAAM,EAAE;QACV,MAAMmH,OAAO,GAAG3E,IAAI,CAACa,SAAS,CAAC,EAAE,EAAE,EAAE,CAAC;QACtC,MAAM+D,SAAS,GAAG5E,IAAI,CAACa,SAAS,CAAC,EAAE,EAAE,EAAE,CAAC;QACxC;QACA,IAAI,CAAC7C,WAAW,CAAC2G,OAAO,CAAC,IAAI,CAAC3G,WAAW,CAAC4G,SAAS,CAAC,EAAE;UACpD,MAAMC,OAAO,GAAGF,OAAO,CAAC7E,IAAI,CAAC,CAAC,IAAI8E,SAAS,CAAC9E,IAAI,CAAC,CAAC;UAClDK,KAAK,CAAC0E,OAAO,GAAG,kBAAkB,GAAG,kBAAkB,EAAE,EAAE,EAAE,EAAE,CAAC;UAChE;QACF;;QAEA;QACA;QACApB,QAAQ,GAAG1G,iBAAO,CAACS,MAAM,CAACjB,UAAU,CAACqI,SAAS,CAAC,IAAI,CAAC,CAAC,CAACE,GAAG,CACvD/H,iBAAO,CAACU,IAAI,CAAClB,UAAU,CAACoI,OAAO,CAAC,IAAI,CAAC,CACvC,CAAC;QAED1H,IAAI,GAAGD,SAAS,CAACgD,IAAI,CAAC,EAAE,CAAC,CAAC;QAC1B;QACA;QACA;QACA;QACA+D,eAAe,GAAG/D,IAAI,CAAC,EAAE,CAAC,KAAK,GAAG,IAAIA,IAAI,CAAC,EAAE,CAAC,KAAK,GAAG;QACtD0E,OAAO,GAAG1E,IAAI,CAAC,EAAE,CAAC,KAAK,GAAG;MAC5B,CAAC,MAAM;QACL;QACA,MAAM2E,OAAO,GAAG7D,QAAQ,CAAC,EAAE,EAAE,EAAE,EAAE,UAAU,EAAE1C,aAAa,CAAC;QAC3DqF,QAAQ,GAAG,IAAIhH,wBAAc,CAACF,UAAU,CAACoI,OAAO,CAAC,EAAE3C,YAAY,CAAC;QAChE/E,IAAI,GAAGD,SAAS,CAACgD,IAAI,CAAC,EAAE,CAAC,CAAC;QAC1B+D,eAAe,GAAG/D,IAAI,CAAC,EAAE,CAAC,KAAK,GAAG,IAAIA,IAAI,CAAC,EAAE,CAAC,KAAK,GAAG;QACtD0E,OAAO,GAAG1E,IAAI,CAAC,EAAE,CAAC,KAAK,GAAG;MAC5B;MAEA,IAAI/C,IAAI,KAAKC,sBAAY,CAACG,MAAM,EAAE;QAChCyD,QAAQ,CAAC,EAAE,EAAE,EAAE,EAAE,mBAAmB,EAAE7C,YAAY,CAAC;MACrD;;MAEA;MACA;MACA;MACA;MACA,IAAIhB,IAAI,KAAKC,sBAAY,CAACC,UAAU,EAAE;QACpC;QACAqH,kBAAkB,GAAGf,QAAQ;QAC7B,MAAMsB,CAAC,GAAGP,kBAAkB,CAAC5H,GAAG,CAACoF,YAAY,CAAC;QAC9C,MAAMgD,CAAC,GAAGzI,UAAU,CAAC+H,QAAQ,CAAC;QAC9BG,gBAAgB,GAAG,IAAIhI,wBAAc,CAACuI,CAAC,EAAEhD,YAAY,CAAC;QACtDyB,QAAQ,GAAG,IAAIhH,wBAAc,CAACwI,IAAI,CAACC,IAAI,CAACH,CAAC,GAAGA,CAAC,GAAGC,CAAC,GAAGA,CAAC,CAAC,EAAEhD,YAAY,CAAC;QACrE4B,qBAAqB,GAAG/G,eAAK,CAACsI,KAAK,CACjCV,gBAAgB,EAChBD,kBACF,CAAC;QACDX,oBAAoB,GAAG,IAAI;MAC7B,CAAC,MAAM,IAAI5G,IAAI,KAAKC,sBAAY,CAACE,QAAQ,EAAE;QACzC;QACA,MAAMgI,CAAC,GAAG3B,QAAQ,CAAC7G,GAAG,CAACoF,YAAY,CAAC;QACpC,MAAMgD,CAAC,GAAGzI,UAAU,CAAC+H,QAAQ,CAAC;QAC9BG,gBAAgB,GAAG,IAAIhI,wBAAc,CAACuI,CAAC,EAAEhD,YAAY,CAAC;QACtD4B,qBAAqB,GAAG/G,eAAK,CAACwI,IAAI,CAACL,CAAC,GAAGI,CAAC,CAAC;QACzCvB,oBAAoB,GAAG,IAAI;MAC7B,CAAC,MAAM;QACL;QACA/C,QAAQ,CAAC,EAAE,EAAE,EAAE,EAAE,aAAa,EAAEzC,qBAAqB,CAAC;QACtD;QACAyC,QAAQ,CAAC,EAAE,EAAE,EAAE,EAAE,aAAa,EAAEzC,qBAAqB,CAAC;QACtDuF,qBAAqB,GAAGzH,WAAW,CAACmI,QAAQ,EAAEnC,eAAe,CAAC;QAC9D0B,oBAAoB,GAAG1H,WAAW,CAACoI,QAAQ,EAAEpC,eAAe,CAAC;MAC/D;MACA,IAAIlC,OAAO,EAAE;MAEb,MAAMyD,iBAAiB,GAAGhH,YAAY,CAAC0H,QAAQ,EAAElC,WAAW,CAAC;MAC7D,MAAMyB,gBAAgB,GAAGjH,YAAY,CAAC2H,QAAQ,EAAEnC,WAAW,CAAC;MAE5D,IAAI,CAAC0B,qBAAqB,IAAI,CAACC,oBAAoB,EAAE;QACnDD,qBAAqB,GAAG7G,iBAAO,CAACD,OAAO,CAAC,CAAC,CAAC;MAC5C;MAEA,MAAMwG,IAAc,GAAG;QACrBC,IAAI,EAAEX,OAAO,CAAC9C,IAAI,CAAC,CAAC;QACpB0D,EAAE,EAAEH,KAAK,CAACvD,IAAI,CAAC,CAAC;QAChB7C,IAAI;QACJwG,QAAQ;QACRC,iBAAiB;QACjBC,gBAAgB;QAChBC,qBAAqB;QACrBC,oBAAoB;QACpByB,OAAO,EAAE;UACPnC,IAAI;UACJC,KAAK;UACLH,EAAE;UACFC;QACF,CAAC;QACDa,eAAe;QACfnE,OAAO,EAAED,UAAU,CAAC;MACtB,CAAC;MACD,IAAI+E,OAAO,EAAEpB,IAAI,CAACoB,OAAO,GAAG,IAAI;MAChC,IAAIZ,SAAS,EAAER,IAAI,CAACQ,SAAS,GAAGA,SAAS;MACzC,IAAIU,kBAAkB,EAAElB,IAAI,CAACkB,kBAAkB,GAAGA,kBAAkB;MACpE,IAAIC,gBAAgB,EAAEnB,IAAI,CAACmB,gBAAgB,GAAGA,gBAAgB;MAC9D9B,KAAK,CAACpC,IAAI,CAAC+C,IAAI,CAAC;IAClB;EACF;EAEA,OAAO;IACL1E,IAAI;IACJC,QAAQ;IACRC,KAAK;IACLC;EACF,CAAC;AACH"}