blob: 7e0799d22a035ba05ceaeb8704511f90d7e153ab [file] [log] [blame]
// Copyright 2015 The Bazel Authors. All rights reserved.
//
// 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.devtools.build.lib.packages;
import com.google.common.base.Preconditions;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.Iterables;
import com.google.devtools.build.lib.skyframe.serialization.autocodec.AutoCodec;
import com.google.devtools.build.lib.skylarkinterface.SkylarkModule;
import com.google.devtools.build.lib.syntax.Dict;
import com.google.devtools.build.lib.syntax.EvalException;
import com.google.devtools.build.lib.syntax.HasBinary;
import com.google.devtools.build.lib.syntax.Printer;
import com.google.devtools.build.lib.syntax.SkylarkType;
import com.google.devtools.build.lib.syntax.Starlark;
import com.google.devtools.build.lib.syntax.StarlarkValue;
import com.google.devtools.build.lib.syntax.TokenKind;
import java.util.Arrays;
import java.util.List;
import java.util.Objects;
/**
* An attribute value consisting of a concatenation of native types and selects, e.g:
*
* <pre>
* rule(
* name = 'myrule',
* deps =
* [':defaultdep']
* + select({
* 'a': [':adep'],
* 'b': [':bdep'],})
* + select({
* 'c': [':cdep'],
* 'd': [':ddep'],})
* )
* </pre>
*/
@SkylarkModule(
name = "select",
doc = "A selector between configuration-dependent entities.",
documented = false)
@AutoCodec
public final class SelectorList implements StarlarkValue, HasBinary {
// TODO(adonovan): combine Selector{List,Value} and BuildType.SelectorList.
// We don't need three classes for the same concept
private final Class<?> type;
private final List<Object> elements;
@AutoCodec.VisibleForSerialization
SelectorList(Class<?> type, List<Object> elements) {
this.type = type;
this.elements = elements;
}
/**
* Returns an ordered list of the elements in this expression. Each element may be a native type
* or a select.
*/
List<Object> getElements() {
return elements;
}
/** Returns the native type contained by this expression. */
Class<?> getType() {
return type;
}
/** Implementation of the Starlark {@code select} function exposed to BUILD and .bzl files. */
public static Object select(Dict<?, ?> dict, String noMatchError) throws EvalException {
if (dict.isEmpty()) {
throw Starlark.errorf(
"select({}) with an empty dictionary can never resolve because it includes no conditions"
+ " to match");
}
for (Object key : dict.keySet()) {
if (!(key instanceof String)) {
throw Starlark.errorf(
"select: got %s for dict key, want a label string", Starlark.type(key));
}
}
return SelectorList.of(new SelectorValue(dict, noMatchError));
}
/** Creates a "wrapper" list that consists of a single select. */
static SelectorList of(SelectorValue selector) {
return new SelectorList(selector.getType(), ImmutableList.of(selector));
}
/**
* Creates a list that concatenates two values, where each value may be a native type, a select
* over that type, or a selector list over that type.
*
* @throws EvalException if the values don't have the same underlying type
*/
static SelectorList concat(Object x, Object y) throws EvalException {
return of(Arrays.asList(x, y));
}
@Override
public SelectorList binaryOp(TokenKind op, Object that, boolean thisLeft) throws EvalException {
if (op == TokenKind.PLUS) {
return thisLeft ? concat(this, that) : concat(that, this);
}
return null;
}
/**
* Creates a list from the given sequence of values, which must be non-empty. Each value may be a
* native type, a select over that type, or a selector list over that type.
*
* @throws EvalException if all values don't have the same underlying type
*/
static SelectorList of(Iterable<?> values) throws EvalException {
Preconditions.checkArgument(!Iterables.isEmpty(values));
ImmutableList.Builder<Object> elements = ImmutableList.builder();
Object firstValue = null;
for (Object value : values) {
if (value instanceof SelectorList) {
elements.addAll(((SelectorList) value).getElements());
} else {
elements.add(value);
}
if (firstValue == null) {
firstValue = value;
}
if (!canConcatenate(getNativeType(firstValue), getNativeType(value))) {
throw Starlark.errorf(
"'+' operator applied to incompatible types (%s, %s)",
getTypeName(firstValue), getTypeName(value));
}
}
return new SelectorList(getNativeType(firstValue), elements.build());
}
private static final Class<?> NATIVE_LIST_TYPE = List.class;
private static String getTypeName(Object x) {
if (x instanceof SelectorList) {
return "select of " + SkylarkType.of(((SelectorList) x).getType());
} else if (x instanceof SelectorValue) {
return "select of " + SkylarkType.of(((SelectorValue) x).getType());
} else {
return Starlark.type(x);
}
}
private static Class<?> getNativeType(Object value) {
if (value instanceof SelectorList) {
return ((SelectorList) value).getType();
} else if (value instanceof SelectorValue) {
return ((SelectorValue) value).getType();
} else {
return value.getClass();
}
}
private static boolean isListType(Class<?> type) {
return NATIVE_LIST_TYPE.isAssignableFrom(type);
}
private static boolean canConcatenate(Class<?> type1, Class<?> type2) {
if (type1 == type2) {
return true;
} else if (isListType(type1) && isListType(type2)) {
return true;
} else {
return false;
}
}
@Override
public String toString() {
return Starlark.repr(this);
}
@Override
public void repr(Printer printer) {
printer.printList(elements, "", " + ", "", null);
}
@Override
public int hashCode() {
return Objects.hash(type, elements);
}
@Override
public boolean equals(Object other) {
if (this == other) {
return true;
}
if (!(other instanceof SelectorList)) {
return false;
}
SelectorList that = (SelectorList) other;
return Objects.equals(this.type, that.type) && Objects.equals(this.elements, that.elements);
}
}