blob: e67acae12421daf8dbca398c543cd8c278fbd1e7 [file] [log] [blame]
// Copyright 2018 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.rules.config;
import static com.google.devtools.build.lib.analysis.skylark.FunctionTransitionUtil.COMMAND_LINE_OPTION_PREFIX;
import com.google.common.collect.Sets;
import com.google.devtools.build.lib.analysis.config.StarlarkDefinedConfigTransition;
import com.google.devtools.build.lib.analysis.skylark.StarlarkTransition.Settings;
import com.google.devtools.build.lib.cmdline.Label;
import com.google.devtools.build.lib.skylarkbuildapi.config.ConfigGlobalLibraryApi;
import com.google.devtools.build.lib.skylarkbuildapi.config.ConfigurationTransitionApi;
import com.google.devtools.build.lib.syntax.BaseFunction;
import com.google.devtools.build.lib.syntax.Dict;
import com.google.devtools.build.lib.syntax.EvalException;
import com.google.devtools.build.lib.syntax.Sequence;
import com.google.devtools.build.lib.syntax.Starlark;
import com.google.devtools.build.lib.syntax.StarlarkSemantics;
import com.google.devtools.build.lib.syntax.StarlarkThread;
import java.util.HashSet;
import java.util.List;
import java.util.Map;
/**
* Implementation of {@link ConfigGlobalLibraryApi}.
*
* <p>A collection of top-level Starlark functions pertaining to configuration.
*/
public class ConfigGlobalLibrary implements ConfigGlobalLibraryApi {
@Override
public ConfigurationTransitionApi transition(
BaseFunction implementation,
Sequence<?> inputs, // <String> expected
Sequence<?> outputs, // <String> expected
StarlarkThread thread)
throws EvalException {
StarlarkSemantics semantics = thread.getSemantics();
List<String> inputsList = inputs.getContents(String.class, "inputs");
List<String> outputsList = outputs.getContents(String.class, "outputs");
validateBuildSettingKeys(
inputsList, Settings.INPUTS, semantics.experimentalStarlarkConfigTransitions());
validateBuildSettingKeys(
outputsList, Settings.OUTPUTS, semantics.experimentalStarlarkConfigTransitions());
return StarlarkDefinedConfigTransition.newRegularTransition(
implementation, inputsList, outputsList, semantics, thread);
}
@Override
public ConfigurationTransitionApi analysisTestTransition(
Dict<?, ?> changedSettings, // <String, String> expected
StarlarkThread thread)
throws EvalException {
Map<String, Object> changedSettingsMap =
changedSettings.getContents(String.class, Object.class, "changed_settings dict");
validateBuildSettingKeys(changedSettingsMap.keySet(), Settings.OUTPUTS, true);
return StarlarkDefinedConfigTransition.newAnalysisTestTransition(
changedSettingsMap, thread.getCallerLocation());
}
private void validateBuildSettingKeys(
Iterable<String> optionKeys,
Settings keyErrorDescriptor,
boolean starlarkTransitionsEnabled)
throws EvalException {
HashSet<String> processedOptions = Sets.newHashSet();
String singularErrorDescriptor = keyErrorDescriptor == Settings.INPUTS ? "input" : "output";
for (String optionKey : optionKeys) {
if (!optionKey.startsWith(COMMAND_LINE_OPTION_PREFIX)) {
if (!starlarkTransitionsEnabled) {
throw Starlark.errorf(
"transitions on Starlark-defined build settings is experimental and "
+ "disabled by default. This API is in development and subject to change at any"
+ "time. Use --experimental_starlark_config_transitions to use this experimental "
+ "API.");
}
try {
Label.parseAbsoluteUnchecked(optionKey);
} catch (IllegalArgumentException e) {
throw Starlark.errorf(
"invalid transition %s '%s'. If this is intended as a native option, "
+ "it must begin with //command_line_option: %s",
singularErrorDescriptor, optionKey, e.getMessage());
}
} else {
String optionName = optionKey.substring(COMMAND_LINE_OPTION_PREFIX.length());
if (optionName.startsWith("experimental_") || optionName.startsWith("incompatible_")) {
throw Starlark.errorf(
"Invalid transition %s '%s'. Cannot transition on --experimental_* or "
+ "--incompatible_* options",
singularErrorDescriptor, optionKey);
}
}
if (!processedOptions.add(optionKey)) {
throw Starlark.errorf("duplicate transition %s '%s'", singularErrorDescriptor, optionKey);
}
}
}
}