/* * Copyright 2014 the original author or authors. * * 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 org.gradle.api.internal.artifacts.ivyservice.ivyresolve; import org.gradle.api.InvalidUserCodeException; import org.gradle.api.artifacts.ComponentMetadata; import org.gradle.api.artifacts.ComponentSelection; import org.gradle.api.artifacts.ivy.IvyModuleDescriptor; import org.gradle.api.internal.artifacts.ComponentSelectionInternal; import org.gradle.api.specs.Spec; import org.gradle.api.specs.Specs; import org.gradle.internal.rules.SpecRuleAction; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import java.util.ArrayList; import java.util.Collection; import java.util.Collections; import java.util.List; public class ComponentSelectionRulesProcessor { private static final Logger LOGGER = LoggerFactory.getLogger(ComponentSelectionRulesProcessor.class); private final Spec<SpecRuleAction<? super ComponentSelection>> withNoInputs = new Spec<SpecRuleAction<? super ComponentSelection>>() { public boolean isSatisfiedBy(SpecRuleAction<? super ComponentSelection> element) { return element.getAction().getInputTypes().isEmpty(); } }; private final Spec<SpecRuleAction<? super ComponentSelection>> withInputs = Specs.negate(withNoInputs); public void apply(ComponentSelectionInternal selection, Collection<SpecRuleAction<? super ComponentSelection>> specRuleActions, MetadataProvider metadataProvider) { if (processRules(specRuleActions, withNoInputs, selection, metadataProvider)) { processRules(specRuleActions, withInputs, selection, metadataProvider); } } private boolean processRules(Collection<SpecRuleAction<? super ComponentSelection>> specRuleActions, Spec<SpecRuleAction<? super ComponentSelection>> filter, ComponentSelectionInternal selection, MetadataProvider metadataProvider) { for (SpecRuleAction<? super ComponentSelection> rule : specRuleActions) { if (filter.isSatisfiedBy(rule)) { processRule(rule, selection, metadataProvider); if (selection.isRejected()) { LOGGER.info("Selection of {} rejected by component selection rule: {}", selection.getCandidate().getDisplayName(), selection.getRejectionReason()); return false; } } } return true; } private void processRule(SpecRuleAction<? super ComponentSelection> rule, ComponentSelection selection, MetadataProvider metadataProvider) { if (!rule.getSpec().isSatisfiedBy(selection)) { return; } List<Object> inputValues = getInputValues(rule.getAction().getInputTypes(), metadataProvider); if (inputValues == null) { // Broken meta-data, bail return; } if (inputValues.contains(null)) { // If any of the input values are not available for this selection, ignore the rule return; } try { rule.getAction().execute(selection, inputValues); } catch (Exception e) { throw new InvalidUserCodeException(String.format("There was an error while evaluating a component selection rule for %s.", selection.getCandidate().getDisplayName()), e); } } private List<Object> getInputValues(List<Class<?>> inputTypes, MetadataProvider metadataProvider) { if (inputTypes.size() == 0) { return Collections.emptyList(); } if (!metadataProvider.resolve()) { return null; } List<Object> inputs = new ArrayList<Object>(inputTypes.size()); for (Class<?> inputType : inputTypes) { if (inputType == ComponentMetadata.class) { inputs.add(metadataProvider.getComponentMetadata()); continue; } if (inputType == IvyModuleDescriptor.class) { inputs.add(metadataProvider.getIvyModuleDescriptor()); continue; } // We've already validated the inputs: should never get here. throw new IllegalStateException(); } return inputs; } }