/** * Copyright (c) 2008, http://www.snakeyaml.org * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package org.pyyaml; import java.util.ArrayList; import java.util.List; import java.util.Map; import org.yaml.snakeyaml.error.Mark; import org.yaml.snakeyaml.nodes.Tag; import org.yaml.snakeyaml.scanner.Scanner; import org.yaml.snakeyaml.scanner.ScannerImpl; import org.yaml.snakeyaml.tokens.AliasToken; import org.yaml.snakeyaml.tokens.AnchorToken; import org.yaml.snakeyaml.tokens.DirectiveToken; import org.yaml.snakeyaml.tokens.DocumentStartToken; import org.yaml.snakeyaml.tokens.FlowEntryToken; import org.yaml.snakeyaml.tokens.FlowMappingEndToken; import org.yaml.snakeyaml.tokens.FlowMappingStartToken; import org.yaml.snakeyaml.tokens.FlowSequenceEndToken; import org.yaml.snakeyaml.tokens.FlowSequenceStartToken; import org.yaml.snakeyaml.tokens.KeyToken; import org.yaml.snakeyaml.tokens.ScalarToken; import org.yaml.snakeyaml.tokens.StreamEndToken; import org.yaml.snakeyaml.tokens.StreamStartToken; import org.yaml.snakeyaml.tokens.TagToken; import org.yaml.snakeyaml.tokens.TagTuple; import org.yaml.snakeyaml.tokens.Token; import org.yaml.snakeyaml.tokens.ValueToken; public class CanonicalScanner implements Scanner { private static final String DIRECTIVE = "%YAML 1.1"; private final static Map<Character, Integer> QUOTE_CODES = ScannerImpl.ESCAPE_CODES; private final static Map<Character, String> QUOTE_REPLACES = ScannerImpl.ESCAPE_REPLACEMENTS; private String data; private int index; public ArrayList<Token> tokens; private boolean scanned; private Mark mark; public CanonicalScanner(String data) { this.data = data + "\0"; this.index = 0; this.tokens = new ArrayList<Token>(); this.scanned = false; this.mark = new Mark("test", 0, 0, 0, data, 0); } public boolean checkToken(Token.ID... choices) { if (!scanned) { scan(); } if (!tokens.isEmpty()) { if (choices.length == 0) { return true; } Token first = this.tokens.get(0); for (Token.ID choice : choices) { if (first.getTokenId() == choice) { return true; } } } return false; } public Token peekToken() { if (!scanned) { scan(); } if (!tokens.isEmpty()) { return this.tokens.get(0); } return null; } public Token getToken() { if (!scanned) { scan(); } return this.tokens.remove(0); } public Token getToken(Token.ID choice) { Token token = getToken(); if (choice != null && token.getTokenId() != choice) { throw new CanonicalException("unexpected token " + token); } return token; } private void scan() { this.tokens.add(new StreamStartToken(mark, mark)); boolean stop = false; while (!stop) { findToken(); char ch = data.charAt(index); switch (ch) { case '\0': tokens.add(new StreamEndToken(mark, mark)); stop = true; break; case '%': tokens.add(scanDirective()); break; case '-': if ("---".equals(data.substring(index, index + 3))) { index += 3; tokens.add(new DocumentStartToken(mark, mark)); } break; case '[': index++; tokens.add(new FlowSequenceStartToken(mark, mark)); break; case '{': index++; tokens.add(new FlowMappingStartToken(mark, mark)); break; case ']': index++; tokens.add(new FlowSequenceEndToken(mark, mark)); break; case '}': index++; tokens.add(new FlowMappingEndToken(mark, mark)); break; case '?': index++; tokens.add(new KeyToken(mark, mark)); break; case ':': index++; tokens.add(new ValueToken(mark, mark)); break; case ',': index++; tokens.add(new FlowEntryToken(mark, mark)); break; case '*': tokens.add(scanAlias()); break; case '&': tokens.add(scanAlias()); break; case '!': tokens.add(scanTag()); break; case '"': tokens.add(scanScalar()); break; default: throw new CanonicalException("invalid token"); } } scanned = true; } private Token scanDirective() { String chunk1 = data.substring(index, index + DIRECTIVE.length()); char chunk2 = data.charAt(index + DIRECTIVE.length()); if (DIRECTIVE.equals(chunk1) && "\n\0".indexOf(chunk2) != -1) { index += DIRECTIVE.length(); List<Integer> implicit = new ArrayList<Integer>(2); implicit.add(new Integer(1)); implicit.add(new Integer(1)); return new DirectiveToken<Integer>("YAML", implicit, mark, mark); } else { throw new CanonicalException("invalid directive"); } } private Token scanAlias() { boolean isTokenClassAlias; if (data.charAt(index) == '*') { isTokenClassAlias = true; } else { isTokenClassAlias = false; } index++; int start = index; while (", \n\0".indexOf(data.charAt(index)) == -1) { index++; } String value = data.substring(start, index); Token token; if (isTokenClassAlias) { token = new AliasToken(value, mark, mark); } else { token = new AnchorToken(value, mark, mark); } return token; } private Token scanTag() { index++; int start = index; while (" \n\0".indexOf(data.charAt(index)) == -1) { index++; } String value = data.substring(start, index); if (value.length() == 0) { value = "!"; } else if (value.charAt(0) == '!') { value = Tag.PREFIX + value.substring(1); } else if (value.charAt(0) == '<' && value.charAt(value.length() - 1) == '>') { value = value.substring(1, value.length() - 1); } else { value = "!" + value; } return new TagToken(new TagTuple("", value), mark, mark); } private Token scanScalar() { index++; StringBuilder chunks = new StringBuilder(); int start = index; boolean ignoreSpaces = false; while (data.charAt(index) != '"') { if (data.charAt(index) == '\\') { ignoreSpaces = false; chunks.append(data.substring(start, index)); index++; char ch = data.charAt(index); index++; if (ch == '\n') { ignoreSpaces = true; } else if (QUOTE_CODES.keySet().contains(ch)) { int length = QUOTE_CODES.get(ch); int code = Integer.parseInt(data.substring(index, index + length), 16); chunks.append(String.valueOf((char) code)); index += length; } else { if (!QUOTE_REPLACES.keySet().contains(ch)) { throw new CanonicalException("invalid escape code"); } chunks.append(QUOTE_REPLACES.get(ch)); } start = index; } else if (data.charAt(index) == '\n') { chunks.append(data.substring(start, index)); chunks.append(" "); index++; start = index; ignoreSpaces = true; } else if (ignoreSpaces && data.charAt(index) == ' ') { index++; start = index; } else { ignoreSpaces = false; index++; } } chunks.append(data.substring(start, index)); index++; return new ScalarToken(chunks.toString(), mark, mark, false); } private void findToken() { boolean found = false; while (!found) { while (" \t".indexOf(data.charAt(index)) != -1) { index++; } if (data.charAt(index) == '#') { while (data.charAt(index) != '\n') { index++; } } if (data.charAt(index) == '\n') { index++; } else { found = true; } } } }