blob: 02d2ca6951e05fd56a07cabd1d20b05579c47216 [file] [log] [blame]
// Copyright (C) 2013 The Android Open Source Project
//
// 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.gerrit.server.query.change;
import com.google.auto.value.AutoValue;
import com.google.common.base.Splitter;
import com.google.gerrit.index.query.QueryParseException;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
/**
* This class is used to extract comma separated values in a predicate.
*
* <p>If tags for the values are present (e.g. "branch=jb_2.3,vote=approved") then the args are
* placed in a map that maps tag to value (e.g., "branch" to "jb_2.3"). If no tag is present (e.g.
* "jb_2.3,approved") then the args are placed into a positional list. Args may be mixed so some may
* appear in the map and others in the positional list (e.g. "vote=approved,jb_2.3).
*/
public class PredicateArgs {
private static final Pattern SPLIT_PATTERN = Pattern.compile("(>|>=|=|<|<=)([^=].*)$");
public List<String> positional;
public Map<String, ValOp> keyValue;
enum Operator {
EQUAL("="),
GREATER_EQUAL(">="),
GREATER(">"),
LESS_EQUAL("<="),
LESS("<");
final String op;
Operator(String op) {
this.op = op;
}
}
@AutoValue
public abstract static class ValOp {
abstract String value();
abstract Operator operator();
static ValOp create(String value, Operator operator) {
return new AutoValue_PredicateArgs_ValOp(value, operator);
}
}
/**
* Parses query arguments into {@link #keyValue} and/or {@link #positional}..
*
* <p>Labels for these arguments should be kept in ChangeQueryBuilder as {@code ARG_ID_[argument
* name]}.
*
* @param args arguments to be parsed
*/
public PredicateArgs(String args) throws QueryParseException {
positional = new ArrayList<>();
keyValue = new HashMap<>();
for (String arg : Splitter.on(',').split(args)) {
Matcher m = SPLIT_PATTERN.matcher(arg);
if (!m.find()) {
positional.add(arg);
} else if (m.groupCount() == 2) {
String key = arg.substring(0, m.start());
String op = m.group(1);
String val = m.group(2);
if (!keyValue.containsKey(key)) {
keyValue.put(key, ValOp.create(val, getOperator(op)));
} else {
throw new QueryParseException("Duplicate key " + key);
}
} else {
throw new QueryParseException("Invalid arg " + arg);
}
}
}
private Operator getOperator(String operator) {
switch (operator) {
case "<":
return Operator.LESS;
case "<=":
return Operator.LESS_EQUAL;
case "=":
return Operator.EQUAL;
case ">=":
return Operator.GREATER_EQUAL;
case ">":
return Operator.GREATER;
default:
throw new IllegalArgumentException("Invalid Operator " + operator);
}
}
}