OLD | NEW |
(Empty) | |
| 1 // Copyright (c) 2014, the Dart project authors. Please see the AUTHORS file |
| 2 // for details. All rights reserved. Use of this source code is governed by a |
| 3 // BSD-style license that can be found in the LICENSE file. |
| 4 |
| 5 library glob.single_component; |
| 6 |
| 7 import 'package:path/path.dart' as p; |
| 8 import 'package:string_scanner/string_scanner.dart'; |
| 9 |
| 10 import 'ast.dart'; |
| 11 import 'utils.dart'; |
| 12 |
| 13 const _HYPHEN = 0x2D; |
| 14 |
| 15 /// A parser for globs. |
| 16 class Parser { |
| 17 /// The scanner used to scan the source. |
| 18 final StringScanner _scanner; |
| 19 |
| 20 /// The path context for the glob. |
| 21 final p.Context _context; |
| 22 |
| 23 Parser(String component, this._context) |
| 24 : _scanner = new StringScanner(component); |
| 25 |
| 26 /// Parses an entire glob. |
| 27 SequenceNode parse() => _parseSequence(); |
| 28 |
| 29 /// Parses a [SequenceNode]. |
| 30 /// |
| 31 /// If [inOptions] is true, this is parsing within an [OptionsNode]. |
| 32 SequenceNode _parseSequence({bool inOptions: false}) { |
| 33 var nodes = []; |
| 34 |
| 35 if (_scanner.isDone) { |
| 36 _scanner.error('expected a glob.', position: 0, length: 0); |
| 37 } |
| 38 |
| 39 while (!_scanner.isDone) { |
| 40 if (inOptions && (_scanner.matches(',') || _scanner.matches('}'))) break; |
| 41 nodes.add(_parseNode(inOptions: inOptions)); |
| 42 } |
| 43 |
| 44 return new SequenceNode(nodes); |
| 45 } |
| 46 |
| 47 /// Parses an [AstNode]. |
| 48 /// |
| 49 /// If [inOptions] is true, this is parsing within an [OptionsNode]. |
| 50 AstNode _parseNode({bool inOptions: false}) { |
| 51 var star = _parseStar(); |
| 52 if (star != null) return star; |
| 53 |
| 54 var anyChar = _parseAnyChar(); |
| 55 if (anyChar != null) return anyChar; |
| 56 |
| 57 var range = _parseRange(); |
| 58 if (range != null) return range; |
| 59 |
| 60 var options = _parseOptions(); |
| 61 if (options != null) return options; |
| 62 |
| 63 return _parseLiteral(inOptions: inOptions); |
| 64 } |
| 65 |
| 66 /// Tries to parse a [StarNode] or a [DoubleStarNode]. |
| 67 /// |
| 68 /// Returns `null` if there's not one to parse. |
| 69 AstNode _parseStar() { |
| 70 if (!_scanner.scan('*')) return null; |
| 71 return _scanner.scan('*') ? new DoubleStarNode(_context) : new StarNode(); |
| 72 } |
| 73 |
| 74 /// Tries to parse an [AnyCharNode]. |
| 75 /// |
| 76 /// Returns `null` if there's not one to parse. |
| 77 AstNode _parseAnyChar() { |
| 78 if (!_scanner.scan('?')) return null; |
| 79 return new AnyCharNode(); |
| 80 } |
| 81 |
| 82 /// Tries to parse an [RangeNode]. |
| 83 /// |
| 84 /// Returns `null` if there's not one to parse. |
| 85 AstNode _parseRange() { |
| 86 if (!_scanner.scan('[')) return null; |
| 87 if (_scanner.matches(']')) _scanner.error('unexpected "]".'); |
| 88 var negated = _scanner.scan('!') || _scanner.scan('^'); |
| 89 |
| 90 var ranges = []; |
| 91 while (!_scanner.scan(']')) { |
| 92 var start = _scanner.position; |
| 93 // Allow a backslash to escape a character. |
| 94 _scanner.scan('\\'); |
| 95 var char = _scanner.readChar(); |
| 96 |
| 97 if (_scanner.scan('-')) { |
| 98 if (_scanner.matches(']')) { |
| 99 ranges.add(new Range.singleton(char)); |
| 100 ranges.add(new Range.singleton(_HYPHEN)); |
| 101 continue; |
| 102 } |
| 103 |
| 104 // Allow a backslash to escape a character. |
| 105 _scanner.scan('\\'); |
| 106 |
| 107 var end = _scanner.readChar(); |
| 108 if (end < char) { |
| 109 _scanner.error("Range out of order.", |
| 110 position: start, |
| 111 length: _scanner.position - start); |
| 112 } |
| 113 ranges.add(new Range(char, end)); |
| 114 } else { |
| 115 ranges.add(new Range.singleton(char)); |
| 116 } |
| 117 } |
| 118 |
| 119 return new RangeNode(ranges, negated: negated); |
| 120 } |
| 121 |
| 122 /// Tries to parse an [OptionsNode]. |
| 123 /// |
| 124 /// Returns `null` if there's not one to parse. |
| 125 AstNode _parseOptions() { |
| 126 if (!_scanner.scan('{')) return null; |
| 127 if (_scanner.matches('}')) _scanner.error('unexpected "}".'); |
| 128 |
| 129 var options = []; |
| 130 do { |
| 131 options.add(_parseSequence(inOptions: true)); |
| 132 } while (_scanner.scan(',')); |
| 133 |
| 134 // Don't allow single-option blocks. |
| 135 if (options.length == 1) _scanner.expect(','); |
| 136 _scanner.expect('}'); |
| 137 |
| 138 return new OptionsNode(options); |
| 139 } |
| 140 |
| 141 /// Parses a [LiteralNode]. |
| 142 AstNode _parseLiteral({bool inOptions: false}) { |
| 143 // If we're in an options block, we want to stop parsing as soon as we hit a |
| 144 // comma. Otherwise, commas are fair game for literals. |
| 145 var regExp = new RegExp( |
| 146 inOptions ? r'[^*{[?\\}\],()]*' : r'[^*{[?\\}\]()]*'); |
| 147 |
| 148 _scanner.scan(regExp); |
| 149 var buffer = new StringBuffer()..write(_scanner.lastMatch[0]); |
| 150 |
| 151 while (_scanner.scan('\\')) { |
| 152 buffer.writeCharCode(_scanner.readChar()); |
| 153 _scanner.scan(regExp); |
| 154 buffer.write(_scanner.lastMatch[0]); |
| 155 } |
| 156 |
| 157 for (var char in const [']', '(', ')']) { |
| 158 if (_scanner.matches(char)) _scanner.error('unexpected "$char"'); |
| 159 } |
| 160 if (!inOptions && _scanner.matches('}')) _scanner.error('unexpected "}"'); |
| 161 |
| 162 return new LiteralNode(buffer.toString(), _context); |
| 163 } |
| 164 } |
OLD | NEW |