pkg/dockerscript: a simple shell-like syntax to express docker operations
Docker-DCO-1.1-Signed-off-by: Solomon Hykes <solomon@docker.com> (github: shykes)
This commit is contained in:
		
							parent
							
								
									358c2f3ca4
								
							
						
					
					
						commit
						d188c734e9
					
				
					 2 changed files with 675 additions and 1 deletions
				
			
		|  | @ -3,7 +3,7 @@ package main | |||
| import ( | ||||
| 	"fmt" | ||||
| 	"os" | ||||
| 	"text/scanner" | ||||
| 	scanner "github.com/dotcloud/docker/pkg/dockerscript" | ||||
| 	"strings" | ||||
| ) | ||||
| 
 | ||||
|  |  | |||
							
								
								
									
										674
									
								
								dockerscript/scanner.go
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										674
									
								
								dockerscript/scanner.go
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,674 @@ | |||
| // Copyright 2009 The Go Authors. All rights reserved. | ||||
| // Use of this source code is governed by a BSD-style | ||||
| // license that can be found in the LICENSE file. | ||||
| 
 | ||||
| // Package scanner provides a scanner and tokenizer for UTF-8-encoded text. | ||||
| // It takes an io.Reader providing the source, which then can be tokenized | ||||
| // through repeated calls to the Scan function.  For compatibility with | ||||
| // existing tools, the NUL character is not allowed. If the first character | ||||
| // in the source is a UTF-8 encoded byte order mark (BOM), it is discarded. | ||||
| // | ||||
| // By default, a Scanner skips white space and Go comments and recognizes all | ||||
| // literals as defined by the Go language specification.  It may be | ||||
| // customized to recognize only a subset of those literals and to recognize | ||||
| // different white space characters. | ||||
| // | ||||
| // Basic usage pattern: | ||||
| // | ||||
| //	var s scanner.Scanner | ||||
| //	s.Init(src) | ||||
| //	tok := s.Scan() | ||||
| //	for tok != scanner.EOF { | ||||
| //		// do something with tok | ||||
| //		tok = s.Scan() | ||||
| //	} | ||||
| // | ||||
| package scanner | ||||
| 
 | ||||
| import ( | ||||
| 	"bytes" | ||||
| 	"fmt" | ||||
| 	"io" | ||||
| 	"os" | ||||
| 	"unicode" | ||||
| 	"unicode/utf8" | ||||
| ) | ||||
| 
 | ||||
| // TODO(gri): Consider changing this to use the new (token) Position package. | ||||
| 
 | ||||
| // A source position is represented by a Position value. | ||||
| // A position is valid if Line > 0. | ||||
| type Position struct { | ||||
| 	Filename string // filename, if any | ||||
| 	Offset   int    // byte offset, starting at 0 | ||||
| 	Line     int    // line number, starting at 1 | ||||
| 	Column   int    // column number, starting at 1 (character count per line) | ||||
| } | ||||
| 
 | ||||
| // IsValid returns true if the position is valid. | ||||
| func (pos *Position) IsValid() bool { return pos.Line > 0 } | ||||
| 
 | ||||
| func (pos Position) String() string { | ||||
| 	s := pos.Filename | ||||
| 	if pos.IsValid() { | ||||
| 		if s != "" { | ||||
| 			s += ":" | ||||
| 		} | ||||
| 		s += fmt.Sprintf("%d:%d", pos.Line, pos.Column) | ||||
| 	} | ||||
| 	if s == "" { | ||||
| 		s = "???" | ||||
| 	} | ||||
| 	return s | ||||
| } | ||||
| 
 | ||||
| // Predefined mode bits to control recognition of tokens. For instance, | ||||
| // to configure a Scanner such that it only recognizes (Go) identifiers, | ||||
| // integers, and skips comments, set the Scanner's Mode field to: | ||||
| // | ||||
| //	ScanIdents | ScanInts | SkipComments | ||||
| // | ||||
| const ( | ||||
| 	ScanIdents     = 1 << -Ident | ||||
| 	ScanInts       = 1 << -Int | ||||
| 	ScanFloats     = 1 << -Float // includes Ints | ||||
| 	ScanChars      = 1 << -Char | ||||
| 	ScanStrings    = 1 << -String | ||||
| 	ScanRawStrings = 1 << -RawString | ||||
| 	ScanComments   = 1 << -Comment | ||||
| 	SkipComments   = 1 << -skipComment // if set with ScanComments, comments become white space | ||||
| 	GoTokens       = ScanIdents | ScanFloats | ScanChars | ScanStrings | ScanRawStrings | ScanComments | SkipComments | ||||
| ) | ||||
| 
 | ||||
| // The result of Scan is one of the following tokens or a Unicode character. | ||||
| const ( | ||||
| 	EOF = -(iota + 1) | ||||
| 	Ident | ||||
| 	Int | ||||
| 	Float | ||||
| 	Char | ||||
| 	String | ||||
| 	RawString | ||||
| 	Comment | ||||
| 	skipComment | ||||
| ) | ||||
| 
 | ||||
| var tokenString = map[rune]string{ | ||||
| 	EOF:       "EOF", | ||||
| 	Ident:     "Ident", | ||||
| 	Int:       "Int", | ||||
| 	Float:     "Float", | ||||
| 	Char:      "Char", | ||||
| 	String:    "String", | ||||
| 	RawString: "RawString", | ||||
| 	Comment:   "Comment", | ||||
| } | ||||
| 
 | ||||
| // TokenString returns a printable string for a token or Unicode character. | ||||
| func TokenString(tok rune) string { | ||||
| 	if s, found := tokenString[tok]; found { | ||||
| 		return s | ||||
| 	} | ||||
| 	return fmt.Sprintf("%q", string(tok)) | ||||
| } | ||||
| 
 | ||||
| // GoWhitespace is the default value for the Scanner's Whitespace field. | ||||
| // Its value selects Go's white space characters. | ||||
| const GoWhitespace = 1<<'\t' | 1<<'\n' | 1<<'\r' | 1<<' ' | ||||
| 
 | ||||
| const bufLen = 1024 // at least utf8.UTFMax | ||||
| 
 | ||||
| // A Scanner implements reading of Unicode characters and tokens from an io.Reader. | ||||
| type Scanner struct { | ||||
| 	// Input | ||||
| 	src io.Reader | ||||
| 
 | ||||
| 	// Source buffer | ||||
| 	srcBuf [bufLen + 1]byte // +1 for sentinel for common case of s.next() | ||||
| 	srcPos int              // reading position (srcBuf index) | ||||
| 	srcEnd int              // source end (srcBuf index) | ||||
| 
 | ||||
| 	// Source position | ||||
| 	srcBufOffset int // byte offset of srcBuf[0] in source | ||||
| 	line         int // line count | ||||
| 	column       int // character count | ||||
| 	lastLineLen  int // length of last line in characters (for correct column reporting) | ||||
| 	lastCharLen  int // length of last character in bytes | ||||
| 
 | ||||
| 	// Token text buffer | ||||
| 	// Typically, token text is stored completely in srcBuf, but in general | ||||
| 	// the token text's head may be buffered in tokBuf while the token text's | ||||
| 	// tail is stored in srcBuf. | ||||
| 	tokBuf bytes.Buffer // token text head that is not in srcBuf anymore | ||||
| 	tokPos int          // token text tail position (srcBuf index); valid if >= 0 | ||||
| 	tokEnd int          // token text tail end (srcBuf index) | ||||
| 
 | ||||
| 	// One character look-ahead | ||||
| 	ch rune // character before current srcPos | ||||
| 
 | ||||
| 	// Error is called for each error encountered. If no Error | ||||
| 	// function is set, the error is reported to os.Stderr. | ||||
| 	Error func(s *Scanner, msg string) | ||||
| 
 | ||||
| 	// ErrorCount is incremented by one for each error encountered. | ||||
| 	ErrorCount int | ||||
| 
 | ||||
| 	// The Mode field controls which tokens are recognized. For instance, | ||||
| 	// to recognize Ints, set the ScanInts bit in Mode. The field may be | ||||
| 	// changed at any time. | ||||
| 	Mode uint | ||||
| 
 | ||||
| 	// The Whitespace field controls which characters are recognized | ||||
| 	// as white space. To recognize a character ch <= ' ' as white space, | ||||
| 	// set the ch'th bit in Whitespace (the Scanner's behavior is undefined | ||||
| 	// for values ch > ' '). The field may be changed at any time. | ||||
| 	Whitespace uint64 | ||||
| 
 | ||||
| 	// Start position of most recently scanned token; set by Scan. | ||||
| 	// Calling Init or Next invalidates the position (Line == 0). | ||||
| 	// The Filename field is always left untouched by the Scanner. | ||||
| 	// If an error is reported (via Error) and Position is invalid, | ||||
| 	// the scanner is not inside a token. Call Pos to obtain an error | ||||
| 	// position in that case. | ||||
| 	Position | ||||
| } | ||||
| 
 | ||||
| // Init initializes a Scanner with a new source and returns s. | ||||
| // Error is set to nil, ErrorCount is set to 0, Mode is set to GoTokens, | ||||
| // and Whitespace is set to GoWhitespace. | ||||
| func (s *Scanner) Init(src io.Reader) *Scanner { | ||||
| 	s.src = src | ||||
| 
 | ||||
| 	// initialize source buffer | ||||
| 	// (the first call to next() will fill it by calling src.Read) | ||||
| 	s.srcBuf[0] = utf8.RuneSelf // sentinel | ||||
| 	s.srcPos = 0 | ||||
| 	s.srcEnd = 0 | ||||
| 
 | ||||
| 	// initialize source position | ||||
| 	s.srcBufOffset = 0 | ||||
| 	s.line = 1 | ||||
| 	s.column = 0 | ||||
| 	s.lastLineLen = 0 | ||||
| 	s.lastCharLen = 0 | ||||
| 
 | ||||
| 	// initialize token text buffer | ||||
| 	// (required for first call to next()). | ||||
| 	s.tokPos = -1 | ||||
| 
 | ||||
| 	// initialize one character look-ahead | ||||
| 	s.ch = -1 // no char read yet | ||||
| 
 | ||||
| 	// initialize public fields | ||||
| 	s.Error = nil | ||||
| 	s.ErrorCount = 0 | ||||
| 	s.Mode = GoTokens | ||||
| 	s.Whitespace = GoWhitespace | ||||
| 	s.Line = 0 // invalidate token position | ||||
| 
 | ||||
| 	return s | ||||
| } | ||||
| 
 | ||||
| // next reads and returns the next Unicode character. It is designed such | ||||
| // that only a minimal amount of work needs to be done in the common ASCII | ||||
| // case (one test to check for both ASCII and end-of-buffer, and one test | ||||
| // to check for newlines). | ||||
| func (s *Scanner) next() rune { | ||||
| 	ch, width := rune(s.srcBuf[s.srcPos]), 1 | ||||
| 
 | ||||
| 	if ch >= utf8.RuneSelf { | ||||
| 		// uncommon case: not ASCII or not enough bytes | ||||
| 		for s.srcPos+utf8.UTFMax > s.srcEnd && !utf8.FullRune(s.srcBuf[s.srcPos:s.srcEnd]) { | ||||
| 			// not enough bytes: read some more, but first | ||||
| 			// save away token text if any | ||||
| 			if s.tokPos >= 0 { | ||||
| 				s.tokBuf.Write(s.srcBuf[s.tokPos:s.srcPos]) | ||||
| 				s.tokPos = 0 | ||||
| 				// s.tokEnd is set by Scan() | ||||
| 			} | ||||
| 			// move unread bytes to beginning of buffer | ||||
| 			copy(s.srcBuf[0:], s.srcBuf[s.srcPos:s.srcEnd]) | ||||
| 			s.srcBufOffset += s.srcPos | ||||
| 			// read more bytes | ||||
| 			// (an io.Reader must return io.EOF when it reaches | ||||
| 			// the end of what it is reading - simply returning | ||||
| 			// n == 0 will make this loop retry forever; but the | ||||
| 			// error is in the reader implementation in that case) | ||||
| 			i := s.srcEnd - s.srcPos | ||||
| 			n, err := s.src.Read(s.srcBuf[i:bufLen]) | ||||
| 			s.srcPos = 0 | ||||
| 			s.srcEnd = i + n | ||||
| 			s.srcBuf[s.srcEnd] = utf8.RuneSelf // sentinel | ||||
| 			if err != nil { | ||||
| 				if s.srcEnd == 0 { | ||||
| 					if s.lastCharLen > 0 { | ||||
| 						// previous character was not EOF | ||||
| 						s.column++ | ||||
| 					} | ||||
| 					s.lastCharLen = 0 | ||||
| 					return EOF | ||||
| 				} | ||||
| 				if err != io.EOF { | ||||
| 					s.error(err.Error()) | ||||
| 				} | ||||
| 				// If err == EOF, we won't be getting more | ||||
| 				// bytes; break to avoid infinite loop. If | ||||
| 				// err is something else, we don't know if | ||||
| 				// we can get more bytes; thus also break. | ||||
| 				break | ||||
| 			} | ||||
| 		} | ||||
| 		// at least one byte | ||||
| 		ch = rune(s.srcBuf[s.srcPos]) | ||||
| 		if ch >= utf8.RuneSelf { | ||||
| 			// uncommon case: not ASCII | ||||
| 			ch, width = utf8.DecodeRune(s.srcBuf[s.srcPos:s.srcEnd]) | ||||
| 			if ch == utf8.RuneError && width == 1 { | ||||
| 				// advance for correct error position | ||||
| 				s.srcPos += width | ||||
| 				s.lastCharLen = width | ||||
| 				s.column++ | ||||
| 				s.error("illegal UTF-8 encoding") | ||||
| 				return ch | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	// advance | ||||
| 	s.srcPos += width | ||||
| 	s.lastCharLen = width | ||||
| 	s.column++ | ||||
| 
 | ||||
| 	// special situations | ||||
| 	switch ch { | ||||
| 	case 0: | ||||
| 		// for compatibility with other tools | ||||
| 		s.error("illegal character NUL") | ||||
| 	case '\n': | ||||
| 		s.line++ | ||||
| 		s.lastLineLen = s.column | ||||
| 		s.column = 0 | ||||
| 	} | ||||
| 
 | ||||
| 	return ch | ||||
| } | ||||
| 
 | ||||
| // Next reads and returns the next Unicode character. | ||||
| // It returns EOF at the end of the source. It reports | ||||
| // a read error by calling s.Error, if not nil; otherwise | ||||
| // it prints an error message to os.Stderr. Next does not | ||||
| // update the Scanner's Position field; use Pos() to | ||||
| // get the current position. | ||||
| func (s *Scanner) Next() rune { | ||||
| 	s.tokPos = -1 // don't collect token text | ||||
| 	s.Line = 0    // invalidate token position | ||||
| 	ch := s.Peek() | ||||
| 	s.ch = s.next() | ||||
| 	return ch | ||||
| } | ||||
| 
 | ||||
| // Peek returns the next Unicode character in the source without advancing | ||||
| // the scanner. It returns EOF if the scanner's position is at the last | ||||
| // character of the source. | ||||
| func (s *Scanner) Peek() rune { | ||||
| 	if s.ch < 0 { | ||||
| 		// this code is only run for the very first character | ||||
| 		s.ch = s.next() | ||||
| 		if s.ch == '\uFEFF' { | ||||
| 			s.ch = s.next() // ignore BOM | ||||
| 		} | ||||
| 	} | ||||
| 	return s.ch | ||||
| } | ||||
| 
 | ||||
| func (s *Scanner) error(msg string) { | ||||
| 	s.ErrorCount++ | ||||
| 	if s.Error != nil { | ||||
| 		s.Error(s, msg) | ||||
| 		return | ||||
| 	} | ||||
| 	pos := s.Position | ||||
| 	if !pos.IsValid() { | ||||
| 		pos = s.Pos() | ||||
| 	} | ||||
| 	fmt.Fprintf(os.Stderr, "%s: %s\n", pos, msg) | ||||
| } | ||||
| 
 | ||||
| func (s *Scanner) scanIdentifier() rune { | ||||
| 	ch := s.next() // read character after first '_' or letter | ||||
| 	for ch == '_' || unicode.IsLetter(ch) || unicode.IsDigit(ch) { | ||||
| 		ch = s.next() | ||||
| 	} | ||||
| 	return ch | ||||
| } | ||||
| 
 | ||||
| func digitVal(ch rune) int { | ||||
| 	switch { | ||||
| 	case '0' <= ch && ch <= '9': | ||||
| 		return int(ch - '0') | ||||
| 	case 'a' <= ch && ch <= 'f': | ||||
| 		return int(ch - 'a' + 10) | ||||
| 	case 'A' <= ch && ch <= 'F': | ||||
| 		return int(ch - 'A' + 10) | ||||
| 	} | ||||
| 	return 16 // larger than any legal digit val | ||||
| } | ||||
| 
 | ||||
| func isDecimal(ch rune) bool { return '0' <= ch && ch <= '9' } | ||||
| 
 | ||||
| func (s *Scanner) scanMantissa(ch rune) rune { | ||||
| 	for isDecimal(ch) { | ||||
| 		ch = s.next() | ||||
| 	} | ||||
| 	return ch | ||||
| } | ||||
| 
 | ||||
| func (s *Scanner) scanFraction(ch rune) rune { | ||||
| 	if ch == '.' { | ||||
| 		ch = s.scanMantissa(s.next()) | ||||
| 	} | ||||
| 	return ch | ||||
| } | ||||
| 
 | ||||
| func (s *Scanner) scanExponent(ch rune) rune { | ||||
| 	if ch == 'e' || ch == 'E' { | ||||
| 		ch = s.next() | ||||
| 		if ch == '-' || ch == '+' { | ||||
| 			ch = s.next() | ||||
| 		} | ||||
| 		ch = s.scanMantissa(ch) | ||||
| 	} | ||||
| 	return ch | ||||
| } | ||||
| 
 | ||||
| func (s *Scanner) scanNumber(ch rune) (rune, rune) { | ||||
| 	// isDecimal(ch) | ||||
| 	if ch == '0' { | ||||
| 		// int or float | ||||
| 		ch = s.next() | ||||
| 		if ch == 'x' || ch == 'X' { | ||||
| 			// hexadecimal int | ||||
| 			ch = s.next() | ||||
| 			hasMantissa := false | ||||
| 			for digitVal(ch) < 16 { | ||||
| 				ch = s.next() | ||||
| 				hasMantissa = true | ||||
| 			} | ||||
| 			if !hasMantissa { | ||||
| 				s.error("illegal hexadecimal number") | ||||
| 			} | ||||
| 		} else { | ||||
| 			// octal int or float | ||||
| 			has8or9 := false | ||||
| 			for isDecimal(ch) { | ||||
| 				if ch > '7' { | ||||
| 					has8or9 = true | ||||
| 				} | ||||
| 				ch = s.next() | ||||
| 			} | ||||
| 			if s.Mode&ScanFloats != 0 && (ch == '.' || ch == 'e' || ch == 'E') { | ||||
| 				// float | ||||
| 				ch = s.scanFraction(ch) | ||||
| 				ch = s.scanExponent(ch) | ||||
| 				return Float, ch | ||||
| 			} | ||||
| 			// octal int | ||||
| 			if has8or9 { | ||||
| 				s.error("illegal octal number") | ||||
| 			} | ||||
| 		} | ||||
| 		return Int, ch | ||||
| 	} | ||||
| 	// decimal int or float | ||||
| 	ch = s.scanMantissa(ch) | ||||
| 	if s.Mode&ScanFloats != 0 && (ch == '.' || ch == 'e' || ch == 'E') { | ||||
| 		// float | ||||
| 		ch = s.scanFraction(ch) | ||||
| 		ch = s.scanExponent(ch) | ||||
| 		return Float, ch | ||||
| 	} | ||||
| 	return Int, ch | ||||
| } | ||||
| 
 | ||||
| func (s *Scanner) scanDigits(ch rune, base, n int) rune { | ||||
| 	for n > 0 && digitVal(ch) < base { | ||||
| 		ch = s.next() | ||||
| 		n-- | ||||
| 	} | ||||
| 	if n > 0 { | ||||
| 		s.error("illegal char escape") | ||||
| 	} | ||||
| 	return ch | ||||
| } | ||||
| 
 | ||||
| func (s *Scanner) scanEscape(quote rune) rune { | ||||
| 	ch := s.next() // read character after '/' | ||||
| 	switch ch { | ||||
| 	case 'a', 'b', 'f', 'n', 'r', 't', 'v', '\\', quote: | ||||
| 		// nothing to do | ||||
| 		ch = s.next() | ||||
| 	case '0', '1', '2', '3', '4', '5', '6', '7': | ||||
| 		ch = s.scanDigits(ch, 8, 3) | ||||
| 	case 'x': | ||||
| 		ch = s.scanDigits(s.next(), 16, 2) | ||||
| 	case 'u': | ||||
| 		ch = s.scanDigits(s.next(), 16, 4) | ||||
| 	case 'U': | ||||
| 		ch = s.scanDigits(s.next(), 16, 8) | ||||
| 	default: | ||||
| 		s.error("illegal char escape") | ||||
| 	} | ||||
| 	return ch | ||||
| } | ||||
| 
 | ||||
| func (s *Scanner) scanString(quote rune) (n int) { | ||||
| 	ch := s.next() // read character after quote | ||||
| 	for ch != quote { | ||||
| 		if ch == '\n' || ch < 0 { | ||||
| 			s.error("literal not terminated") | ||||
| 			return | ||||
| 		} | ||||
| 		if ch == '\\' { | ||||
| 			ch = s.scanEscape(quote) | ||||
| 		} else { | ||||
| 			ch = s.next() | ||||
| 		} | ||||
| 		n++ | ||||
| 	} | ||||
| 	return | ||||
| } | ||||
| 
 | ||||
| func (s *Scanner) scanRawString() { | ||||
| 	ch := s.next() // read character after '`' | ||||
| 	for ch != '`' { | ||||
| 		if ch < 0 { | ||||
| 			s.error("literal not terminated") | ||||
| 			return | ||||
| 		} | ||||
| 		ch = s.next() | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| func (s *Scanner) scanChar() { | ||||
| 	if s.scanString('\'') != 1 { | ||||
| 		s.error("illegal char literal") | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| func (s *Scanner) scanComment(ch rune) rune { | ||||
| 	// ch == '/' || ch == '*' | ||||
| 	if ch == '/' { | ||||
| 		// line comment | ||||
| 		ch = s.next() // read character after "//" | ||||
| 		for ch != '\n' && ch >= 0 { | ||||
| 			ch = s.next() | ||||
| 		} | ||||
| 		return ch | ||||
| 	} | ||||
| 
 | ||||
| 	// general comment | ||||
| 	ch = s.next() // read character after "/*" | ||||
| 	for { | ||||
| 		if ch < 0 { | ||||
| 			s.error("comment not terminated") | ||||
| 			break | ||||
| 		} | ||||
| 		ch0 := ch | ||||
| 		ch = s.next() | ||||
| 		if ch0 == '*' && ch == '/' { | ||||
| 			ch = s.next() | ||||
| 			break | ||||
| 		} | ||||
| 	} | ||||
| 	return ch | ||||
| } | ||||
| 
 | ||||
| // Scan reads the next token or Unicode character from source and returns it. | ||||
| // It only recognizes tokens t for which the respective Mode bit (1<<-t) is set. | ||||
| // It returns EOF at the end of the source. It reports scanner errors (read and | ||||
| // token errors) by calling s.Error, if not nil; otherwise it prints an error | ||||
| // message to os.Stderr. | ||||
| func (s *Scanner) Scan() rune { | ||||
| 	ch := s.Peek() | ||||
| 
 | ||||
| 	// reset token text position | ||||
| 	s.tokPos = -1 | ||||
| 	s.Line = 0 | ||||
| 
 | ||||
| redo: | ||||
| 	// skip white space | ||||
| 	for s.Whitespace&(1<<uint(ch)) != 0 { | ||||
| 		ch = s.next() | ||||
| 	} | ||||
| 
 | ||||
| 	// start collecting token text | ||||
| 	s.tokBuf.Reset() | ||||
| 	s.tokPos = s.srcPos - s.lastCharLen | ||||
| 
 | ||||
| 	// set token position | ||||
| 	// (this is a slightly optimized version of the code in Pos()) | ||||
| 	s.Offset = s.srcBufOffset + s.tokPos | ||||
| 	if s.column > 0 { | ||||
| 		// common case: last character was not a '\n' | ||||
| 		s.Line = s.line | ||||
| 		s.Column = s.column | ||||
| 	} else { | ||||
| 		// last character was a '\n' | ||||
| 		// (we cannot be at the beginning of the source | ||||
| 		// since we have called next() at least once) | ||||
| 		s.Line = s.line - 1 | ||||
| 		s.Column = s.lastLineLen | ||||
| 	} | ||||
| 
 | ||||
| 	// determine token value | ||||
| 	tok := ch | ||||
| 	switch { | ||||
| 	case unicode.IsLetter(ch) || ch == '_': | ||||
| 		if s.Mode&ScanIdents != 0 { | ||||
| 			tok = Ident | ||||
| 			ch = s.scanIdentifier() | ||||
| 		} else { | ||||
| 			ch = s.next() | ||||
| 		} | ||||
| 	case isDecimal(ch): | ||||
| 		if s.Mode&(ScanInts|ScanFloats) != 0 { | ||||
| 			tok, ch = s.scanNumber(ch) | ||||
| 		} else { | ||||
| 			ch = s.next() | ||||
| 		} | ||||
| 	default: | ||||
| 		switch ch { | ||||
| 		case '"': | ||||
| 			if s.Mode&ScanStrings != 0 { | ||||
| 				s.scanString('"') | ||||
| 				tok = String | ||||
| 			} | ||||
| 			ch = s.next() | ||||
| 		case '\'': | ||||
| 			if s.Mode&ScanChars != 0 { | ||||
| 				s.scanChar() | ||||
| 				tok = Char | ||||
| 			} | ||||
| 			ch = s.next() | ||||
| 		case '.': | ||||
| 			ch = s.next() | ||||
| 			if isDecimal(ch) && s.Mode&ScanFloats != 0 { | ||||
| 				tok = Float | ||||
| 				ch = s.scanMantissa(ch) | ||||
| 				ch = s.scanExponent(ch) | ||||
| 			} | ||||
| 		case '/': | ||||
| 			ch = s.next() | ||||
| 			if (ch == '/' || ch == '*') && s.Mode&ScanComments != 0 { | ||||
| 				if s.Mode&SkipComments != 0 { | ||||
| 					s.tokPos = -1 // don't collect token text | ||||
| 					ch = s.scanComment(ch) | ||||
| 					goto redo | ||||
| 				} | ||||
| 				ch = s.scanComment(ch) | ||||
| 				tok = Comment | ||||
| 			} | ||||
| 		case '`': | ||||
| 			if s.Mode&ScanRawStrings != 0 { | ||||
| 				s.scanRawString() | ||||
| 				tok = String | ||||
| 			} | ||||
| 			ch = s.next() | ||||
| 		default: | ||||
| 			ch = s.next() | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	// end of token text | ||||
| 	s.tokEnd = s.srcPos - s.lastCharLen | ||||
| 
 | ||||
| 	s.ch = ch | ||||
| 	return tok | ||||
| } | ||||
| 
 | ||||
| // Pos returns the position of the character immediately after | ||||
| // the character or token returned by the last call to Next or Scan. | ||||
| func (s *Scanner) Pos() (pos Position) { | ||||
| 	pos.Filename = s.Filename | ||||
| 	pos.Offset = s.srcBufOffset + s.srcPos - s.lastCharLen | ||||
| 	switch { | ||||
| 	case s.column > 0: | ||||
| 		// common case: last character was not a '\n' | ||||
| 		pos.Line = s.line | ||||
| 		pos.Column = s.column | ||||
| 	case s.lastLineLen > 0: | ||||
| 		// last character was a '\n' | ||||
| 		pos.Line = s.line - 1 | ||||
| 		pos.Column = s.lastLineLen | ||||
| 	default: | ||||
| 		// at the beginning of the source | ||||
| 		pos.Line = 1 | ||||
| 		pos.Column = 1 | ||||
| 	} | ||||
| 	return | ||||
| } | ||||
| 
 | ||||
| // TokenText returns the string corresponding to the most recently scanned token. | ||||
| // Valid after calling Scan(). | ||||
| func (s *Scanner) TokenText() string { | ||||
| 	if s.tokPos < 0 { | ||||
| 		// no token text | ||||
| 		return "" | ||||
| 	} | ||||
| 
 | ||||
| 	if s.tokEnd < 0 { | ||||
| 		// if EOF was reached, s.tokEnd is set to -1 (s.srcPos == 0) | ||||
| 		s.tokEnd = s.tokPos | ||||
| 	} | ||||
| 
 | ||||
| 	if s.tokBuf.Len() == 0 { | ||||
| 		// common case: the entire token text is still in srcBuf | ||||
| 		return string(s.srcBuf[s.tokPos:s.tokEnd]) | ||||
| 	} | ||||
| 
 | ||||
| 	// part of the token text was saved in tokBuf: save the rest in | ||||
| 	// tokBuf as well and return its content | ||||
| 	s.tokBuf.Write(s.srcBuf[s.tokPos:s.tokEnd]) | ||||
| 	s.tokPos = s.tokEnd // ensure idempotency of TokenText() call | ||||
| 	return s.tokBuf.String() | ||||
| } | ||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue