org.gradle.api.tasks.util.internal.PatternSpecFactory Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of gradle-test-kit Show documentation
Show all versions of gradle-test-kit Show documentation
Gradle 6.2.1 API redistribution.
/*
* Copyright 2015 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.tasks.util.internal;
import org.apache.tools.ant.DirectoryScanner;
import org.gradle.api.InvalidUserCodeException;
import org.gradle.api.file.FileTreeElement;
import org.gradle.api.internal.file.RelativePathSpec;
import org.gradle.api.internal.file.pattern.PatternMatcher;
import org.gradle.api.internal.file.pattern.PatternMatcherFactory;
import org.gradle.api.specs.Spec;
import org.gradle.api.specs.Specs;
import org.gradle.api.tasks.util.PatternSet;
import org.gradle.internal.file.excludes.FileSystemDefaultExcludesListener;
import org.gradle.internal.service.scopes.Scope;
import org.gradle.internal.service.scopes.ServiceScope;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collection;
import java.util.Comparator;
import java.util.EnumMap;
import java.util.HashSet;
import java.util.List;
import java.util.Map;
import java.util.Set;
/**
* The basic implementation for converting {@link PatternSet}s to {@link Spec}s.
* This implementation only caches the default exclude patterns, as these are always
* used, no matter which other includes and excludes a {@link PatternSet} has. For an
* implementation that caches all other patterns as well, see {@link CachingPatternSpecFactory}.
*/
@ServiceScope(Scope.Global.class)
public class PatternSpecFactory implements FileSystemDefaultExcludesListener {
public static final PatternSpecFactory INSTANCE = new PatternSpecFactory();
private Set previousDefaultExcludes = new HashSet();
private final Map> defaultExcludeSpecCache = new EnumMap<>(CaseSensitivity.class);
@Override
public void onDefaultExcludesChanged(List excludes) {
setDefaultExcludesFromSettings(excludes.toArray(new String[0]));
}
public Spec createSpec(PatternSet patternSet) {
return Specs.intersect(createIncludeSpec(patternSet), Specs.negate(createExcludeSpec(patternSet)));
}
public Spec createIncludeSpec(PatternSet patternSet) {
Set> includeSpecs = patternSet.getIncludeSpecsView();
List> allIncludeSpecs = new ArrayList<>(1 + includeSpecs.size());
Set includes = patternSet.getIncludesView();
if (!includes.isEmpty()) {
allIncludeSpecs.add(createSpec(includes, true, patternSet.isCaseSensitive()));
}
allIncludeSpecs.addAll(includeSpecs);
return Specs.union(allIncludeSpecs);
}
public Spec createExcludeSpec(PatternSet patternSet) {
Set> excludeSpecs = patternSet.getExcludeSpecsView();
List> allExcludeSpecs = new ArrayList<>(2 + excludeSpecs.size());
Set excludes = patternSet.getExcludesView();
if (!excludes.isEmpty()) {
allExcludeSpecs.add(createSpec(excludes, false, patternSet.isCaseSensitive()));
}
allExcludeSpecs.add(getDefaultExcludeSpec(CaseSensitivity.forCaseSensitive(patternSet.isCaseSensitive())));
allExcludeSpecs.addAll(excludeSpecs);
return Specs.union(allExcludeSpecs);
}
private synchronized Spec getDefaultExcludeSpec(CaseSensitivity caseSensitivity) {
Set defaultExcludes = new HashSet(Arrays.asList(DirectoryScanner.getDefaultExcludes()));
if (defaultExcludeSpecCache.isEmpty()) {
updateDefaultExcludeSpecCache(defaultExcludes);
} else if (invalidChangeOfExcludes(defaultExcludes)) {
failOnChangedDefaultExcludes(previousDefaultExcludes, defaultExcludes);
}
return defaultExcludeSpecCache.get(caseSensitivity);
}
private boolean invalidChangeOfExcludes(Set defaultExcludes) {
return !previousDefaultExcludes.equals(defaultExcludes);
}
private void failOnChangedDefaultExcludes(Set excludesFromSettings, Set newDefaultExcludes) {
List sortedExcludesFromSettings = new ArrayList(excludesFromSettings);
sortedExcludesFromSettings.sort(Comparator.naturalOrder());
List sortedNewExcludes = new ArrayList(newDefaultExcludes);
sortedNewExcludes.sort(Comparator.naturalOrder());
throw new InvalidUserCodeException(String.format("Cannot change default excludes during the build. They were changed from %s to %s. Configure default excludes in the settings script instead.", sortedExcludesFromSettings, sortedNewExcludes));
}
public synchronized void setDefaultExcludesFromSettings(String[] excludesFromSettings) {
Set excludesFromSettingsSet = new HashSet(Arrays.asList(excludesFromSettings));
if (!previousDefaultExcludes.equals(excludesFromSettingsSet)) {
updateDefaultExcludeSpecCache(excludesFromSettingsSet);
}
}
private void updateDefaultExcludeSpecCache(Set defaultExcludes) {
previousDefaultExcludes = defaultExcludes;
for (CaseSensitivity caseSensitivity : CaseSensitivity.values()) {
defaultExcludeSpecCache.put(caseSensitivity, createSpec(defaultExcludes, false, caseSensitivity.isCaseSensitive()));
}
}
protected Spec createSpec(Collection patterns, boolean include, boolean caseSensitive) {
if (patterns.isEmpty()) {
return include ? Specs.satisfyAll() : Specs.satisfyNone();
}
PatternMatcher matcher = PatternMatcherFactory.getPatternsMatcher(include, caseSensitive, patterns);
return new RelativePathSpec(matcher);
}
private enum CaseSensitivity {
CASE_SENSITIVE(true),
CASE_INSENSITIVE(false);
CaseSensitivity(boolean caseSensitive) {
this.caseSensitive = caseSensitive;
}
public static CaseSensitivity forCaseSensitive(boolean caseSensitive) {
return caseSensitive
? CASE_SENSITIVE
: CASE_INSENSITIVE;
}
private final boolean caseSensitive;
public boolean isCaseSensitive() {
return caseSensitive;
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy