1 // Copyright 2017 The Bazel Authors. All rights reserved. 2 // 3 // Licensed under the Apache License, Version 2.0 (the "License"); 4 // you may not use this file except in compliance with the License. 5 // You may obtain a copy of the License at 6 // 7 // http://www.apache.org/licenses/LICENSE-2.0 8 // 9 // Unless required by applicable law or agreed to in writing, software 10 // distributed under the License is distributed on an "AS IS" BASIS, 11 // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 12 // See the License for the specific language governing permissions and 13 // limitations under the License. 14 package com.google.devtools.common.options; 15 16 import static java.nio.charset.StandardCharsets.UTF_8; 17 18 import java.io.IOException; 19 import java.io.PushbackReader; 20 import java.io.Reader; 21 import java.nio.file.FileSystem; 22 import java.nio.file.Files; 23 import java.nio.file.Path; 24 import java.util.ArrayList; 25 import java.util.List; 26 27 /** 28 * A {@link ParamsFilePreProcessor} that processes a parameter file using the {@code 29 * com.google.devtools.build.lib.actions.ParameterFile.ParameterFileType.SHELL_QUOTED} format. This 30 * format assumes each parameter is separated by whitespace and is quoted using singe quotes 31 * ({@code '}) if it contains any special characters or is an empty string. 32 */ 33 public class ShellQuotedParamsFilePreProcessor extends ParamsFilePreProcessor { 34 35 public ShellQuotedParamsFilePreProcessor(FileSystem fs) { 36 super(fs); 37 } 38 39 @Override 40 protected List<String> parse(Path paramsFile) throws IOException { 41 List<String> args = new ArrayList<>(); 42 try (ShellQuotedReader reader = 43 new ShellQuotedReader(Files.newBufferedReader(paramsFile, UTF_8))) { 44 String arg; 45 while ((arg = reader.readArg()) != null) { 46 args.add(arg); 47 } 48 } 49 return args; 50 } 51 52 private static class ShellQuotedReader implements AutoCloseable { 53 54 private final PushbackReader reader; 55 private int position = -1; 56 57 public ShellQuotedReader(Reader reader) { 58 this.reader = new PushbackReader(reader, 10); 59 } 60 61 private char read() throws IOException { 62 int value = reader.read(); 63 position++; 64 return (char) value; 65 } 66 67 private void unread(char value) throws IOException { 68 reader.unread(value); 69 position--; 70 } 71 72 private boolean hasNext() throws IOException { 73 char value = read(); 74 boolean hasNext = value != (char) -1; 75 unread(value); 76 return hasNext; 77 } 78 79 @Override 80 public void close() throws IOException { 81 reader.close(); 82 } 83 84 public String readArg() throws IOException { 85 if (!hasNext()) { 86 return null; 87 } 88 89 StringBuilder arg = new StringBuilder(); 90 91 int quoteStart = -1; 92 boolean quoted = false; 93 char current; 94 95 while ((current = read()) != (char) -1) { 96 if (quoted) { 97 if (current == '\'') { 98 StringBuilder escapedQuoteRemainder = 99 new StringBuilder().append(read()).append(read()).append(read()); 100 if (escapedQuoteRemainder.toString().equals("\\''")) { 101 arg.append("'"); 102 } else { 103 for (char c : escapedQuoteRemainder.reverse().toString().toCharArray()) { 104 unread(c); 105 } 106 quoted = false; 107 quoteStart = -1; 108 } 109 } else { 110 arg.append(current); 111 } 112 } else { 113 if (current == '\'') { 114 quoted = true; 115 quoteStart = position; 116 } else if (current == '\r') { 117 char next = read(); 118 if (next == '\n') { 119 return arg.toString(); 120 } else { 121 unread(next); 122 return arg.toString(); 123 } 124 } else if (Character.isWhitespace(current)) { 125 return arg.toString(); 126 } else { 127 arg.append(current); 128 } 129 } 130 } 131 if (quoted) { 132 throw new IOException( 133 String.format(UNFINISHED_QUOTE_MESSAGE_FORMAT, "'", quoteStart)); 134 } 135 return arg.toString(); 136 } 137 } 138 } 139