com.google.api.tools.framework.aspects.http.HttpTemplateParser Maven / Gradle / Ivy
/*
* Copyright (C) 2016 Google Inc.
*
* 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 com.google.api.tools.framework.aspects.http;
import com.google.api.tools.framework.aspects.http.model.HttpAttribute.FieldSegment;
import com.google.api.tools.framework.aspects.http.model.HttpAttribute.LiteralSegment;
import com.google.api.tools.framework.aspects.http.model.HttpAttribute.PathSegment;
import com.google.api.tools.framework.aspects.http.model.HttpAttribute.WildcardSegment;
import com.google.api.tools.framework.model.Diag;
import com.google.api.tools.framework.model.DiagCollector;
import com.google.api.tools.framework.model.Location;
import com.google.common.collect.FluentIterable;
import com.google.common.collect.ImmutableList;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
/**
* HTTP path template parser.
*/
public class HttpTemplateParser {
private static final Pattern LITERAL_PATTERN = Pattern.compile("[^/*}{=]+");
private static final Pattern TOKEN_PATTERN =
Pattern.compile(String.format("%s|[/*}{=]", LITERAL_PATTERN));
private static final Pattern CUSTOM_VERB_PATTERN =
Pattern.compile(String.format("(? parse() {
ImmutableList path = parse(true, false);
if (!pathStartedWithSlash) {
addError("effective path must start with leading '/'.");
}
if (current != null) {
addError("unrecognized input at '%s'.", current);
}
if (configVersion > 0 && CUSTOM_VERB_PATTERN_ILLEGAL.matcher(template).find()) {
addError("invalid token '/:' before the custom verb.");
}
if (hadErrors) {
return null;
}
if (customVerb != null) {
return FluentIterable.from(path).append(new LiteralSegment(customVerb, true)).toList();
}
return path;
}
private ImmutableList parse(boolean firstSegment, boolean subPath) {
if (lookingAt("/")) {
getCurrentAndShift();
if (firstSegment) {
firstSegment = false;
pathStartedWithSlash = true;
} else {
addError("leading '/' only allowed for first segment of path.");
}
}
ImmutableList.Builder segments = ImmutableList.builder();
while (true) {
String token = getCurrentAndShift();
if (token == null) {
break;
}
switch (token) {
case "*":
if (lookingAt("*")) {
getCurrentAndShift();
segments.add(new WildcardSegment(true));
} else {
segments.add(new WildcardSegment(false));
}
break;
case "{":
segments.add(parseField(firstSegment, subPath));
break;
default:
segments.add(new LiteralSegment(token));
break;
}
// No longer processing the first segment.
firstSegment = false;
if (!lookingAt("/")) {
break;
} else {
getCurrentAndShift();
}
}
return segments.build();
}
private FieldSegment parseField(boolean firstSegment, boolean subPath) {
String fieldPath = getCurrentAndShift();
if (lookingAt("=")) {
getCurrentAndShift();
if (subPath) {
addError("cannot have fields in nested paths.");
}
ImmutableList segments = parse(firstSegment, true);
expectAndShift("}");
return new FieldSegment(fieldPath, segments);
} else {
expectAndShift("}");
return new FieldSegment(fieldPath, ImmutableList.of());
}
}
private boolean lookingAt(String token) {
return token.equals(current);
}
private String getCurrentAndShift() {
if (current == null) {
addError(String.format("unexpected end of input '%s'.", prev == null ? "" : prev));
}
prev = current;
current = tokens.find() ? tokens.group().trim() : null;
return prev;
}
private void expectAndShift(String token) {
if (!lookingAt(token)) {
addError("expected '%s', looking at %s.", token,
current == null ? "end of input" : "'" + current + "'");
} else {
getCurrentAndShift();
}
}
private void addError(String message, Object... params) {
diagCollector.addDiag(Diag.error(location, "In path template '" + template + "': " + message,
params));
hadErrors = true;
}
}
© 2015 - 2024 Weber Informatics LLC | Privacy Policy