blob: 9833174066e7e3ac03771d6f490c22034b3cc6b9 [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.collect.ImmutableMap;
import com.google.common.collect.ImmutableSet;
import com.google.devtools.build.lib.packages.Attribute.ConfigurationTransition;
import com.google.devtools.build.lib.syntax.FragmentClassNameResolver;
import java.util.Collection;
import java.util.Collections;
import java.util.LinkedHashMap;
import java.util.LinkedHashSet;
import java.util.Map;
import java.util.Set;
/**
* Policy used to express the set of configuration fragments which are legal for a rule or aspect
* to access.
*/
public final class ConfigurationFragmentPolicy {
/**
* How to handle the case if the configuration is missing fragments that are required according
* to the rule class.
*/
public enum MissingFragmentPolicy {
/**
* Some rules are monolithic across languages, and we want them to continue to work even when
* individual languages are disabled. Use this policy if the rule implementation is handling
* missing fragments.
*/
IGNORE,
/**
* Use this policy to generate fail actions for the target rather than failing the analysis
* outright. Again, this is used when rules are monolithic across languages, but we still need
* to analyze the dependent libraries. (Instead of this mechanism, consider annotating
* attributes as unused if certain fragments are unavailable.)
*/
CREATE_FAIL_ACTIONS,
/**
* Use this policy to fail the analysis of that target with an error message; this is the
* default.
*/
FAIL_ANALYSIS;
}
/**
* Builder to construct a new ConfigurationFragmentPolicy.
*/
public static final class Builder {
private Set<Class<?>> requiredConfigurationFragments = new LinkedHashSet<>();
private Map<ConfigurationTransition, ImmutableSet<String>> requiredConfigurationFragmentNames =
new LinkedHashMap<>();
private MissingFragmentPolicy missingFragmentPolicy = MissingFragmentPolicy.FAIL_ANALYSIS;
private FragmentClassNameResolver fragmentNameResolver;
/**
* Declares that the implementation of the associated rule class requires the given
* configuration fragments to be present in the configuration. The value is inherited by
* subclasses.
*/
public Builder requiresConfigurationFragments(Collection<Class<?>> configurationFragments) {
requiredConfigurationFragments.addAll(configurationFragments);
return this;
}
/**
* Declares that the implementation of the associated rule class requires the given
* configuration fragments to be present in the configuration. The value is inherited by
* subclasses.
*/
public Builder requiresConfigurationFragments(Class<?>... configurationFragments) {
Collections.addAll(requiredConfigurationFragments, configurationFragments);
return this;
}
/**
* Sets the policy for the case where the configuration is missing required fragments (see
* {@link #requiresConfigurationFragments}).
*/
public Builder setMissingFragmentPolicy(MissingFragmentPolicy missingFragmentPolicy) {
this.missingFragmentPolicy = missingFragmentPolicy;
return this;
}
/**
* Declares the configuration fragments that are required by this rule.
*
* <p>In contrast to {@link #requiresConfigurationFragments(Class...)}, this method a) takes the
* names of fragments instead of their classes and b) distinguishes whether the fragments can be
* accessed in host (HOST) or target (NONE) configuration.
*/
public Builder requiresConfigurationFragments(
FragmentClassNameResolver fragmentNameResolver,
Map<ConfigurationTransition, ImmutableSet<String>> configurationFragmentNames) {
requiredConfigurationFragmentNames.putAll(configurationFragmentNames);
this.fragmentNameResolver = fragmentNameResolver;
return this;
}
public ConfigurationFragmentPolicy build() {
return new ConfigurationFragmentPolicy(
ImmutableSet.copyOf(requiredConfigurationFragments),
ImmutableMap.copyOf(requiredConfigurationFragmentNames),
fragmentNameResolver,
missingFragmentPolicy);
}
}
/**
* The set of required configuration fragments; this should list all fragments that can be
* accessed by the rule implementation.
*/
private final ImmutableSet<Class<?>> requiredConfigurationFragments;
/**
* A dictionary that maps configurations (NONE for target configuration, HOST for host
* configuration) to lists of names of required configuration fragments.
*/
private final ImmutableMap<ConfigurationTransition, ImmutableSet<String>>
requiredConfigurationFragmentNames;
/**
* Used to resolve the names of fragments in order to compare them to values in {@link
* #requiredConfigurationFragmentNames}
*/
private final FragmentClassNameResolver fragmentNameResolver;
/**
* What to do during analysis if a configuration fragment is missing.
*/
private final MissingFragmentPolicy missingFragmentPolicy;
private ConfigurationFragmentPolicy(
ImmutableSet<Class<?>> requiredConfigurationFragments,
ImmutableMap<ConfigurationTransition, ImmutableSet<String>>
requiredConfigurationFragmentNames,
FragmentClassNameResolver fragmentNameResolver,
MissingFragmentPolicy missingFragmentPolicy) {
this.requiredConfigurationFragments = requiredConfigurationFragments;
this.requiredConfigurationFragmentNames = requiredConfigurationFragmentNames;
this.fragmentNameResolver = fragmentNameResolver;
this.missingFragmentPolicy = missingFragmentPolicy;
}
/**
* The set of required configuration fragments; this contains all fragments that can be
* accessed by the rule implementation.
*/
public Set<Class<?>> getRequiredConfigurationFragments() {
return requiredConfigurationFragments;
}
/**
* Checks if the configuration fragment may be accessed (i.e., if it's declared) in the specified
* configuration (target or host).
*/
public boolean isLegalConfigurationFragment(
Class<?> configurationFragment, ConfigurationTransition config) {
return requiredConfigurationFragments.contains(configurationFragment)
|| hasLegalFragmentName(configurationFragment, config);
}
public boolean isLegalConfigurationFragment(Class<?> configurationFragment) {
// NONE means target configuration.
return isLegalConfigurationFragment(configurationFragment, ConfigurationTransition.NONE);
}
/**
* Checks whether the name of the given fragment class was declared as required in the
* specified configuration (target or host).
*/
private boolean hasLegalFragmentName(
Class<?> configurationFragment, ConfigurationTransition config) {
if (fragmentNameResolver == null) {
return false;
}
String name = fragmentNameResolver.resolveName(configurationFragment);
ImmutableSet<String> fragmentNames = requiredConfigurationFragmentNames.get(config);
return (name != null && fragmentNames != null && fragmentNames.contains(name));
}
/**
* Whether to fail analysis if any of the required configuration fragments are missing.
*/
public MissingFragmentPolicy getMissingFragmentPolicy() {
return missingFragmentPolicy;
}
}