org.gradle.api.file.RelativePath Maven / Gradle / Ivy
/*
* Copyright 2010 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.file;
import org.apache.commons.lang.StringUtils;
import org.gradle.api.internal.cache.StringInterner;
import org.gradle.util.CollectionUtils;
import java.io.File;
import java.io.Serializable;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.ListIterator;
/**
* Represents a relative path from some base directory to a file. Used in file copying to represent both a source
* and target file path when copying files.
*
* {@code RelativePath} instances are immutable.
*/
public class RelativePath implements Serializable, Comparable {
private static final StringInterner PATH_SEGMENT_STRING_INTERNER = new StringInterner();
private static final String FILE_PATH_SEPARATORS = File.separatorChar != '/' ? ("/" + File.separator) : File.separator;
private final boolean endsWithFile;
private final String[] segments;
/**
* Creates a {@code RelativePath}.
*
* @param endsWithFile - if true, the path ends with a file, otherwise a directory
*/
public RelativePath(boolean endsWithFile, String... segments) {
this(endsWithFile, null, segments);
}
private RelativePath(boolean endsWithFile, RelativePath parentPath, String... childSegments) {
this.endsWithFile = endsWithFile;
int targetOffsetForChildSegments;
if (parentPath != null) {
String[] sourceSegments = parentPath.getSegments();
segments = new String[sourceSegments.length + childSegments.length];
copySegments(segments, sourceSegments, sourceSegments.length);
targetOffsetForChildSegments = sourceSegments.length;
} else {
segments = new String[childSegments.length];
targetOffsetForChildSegments = 0;
}
copyAndInternSegments(segments, targetOffsetForChildSegments, childSegments);
}
private static void copySegments(String[] target, String[] source) {
copySegments(target, source, target.length);
}
private static void copySegments(String[] target, String[] source, int length) {
// No String instance interning is needed since Strings are from other
// RelativePath instances which contain only interned String instances
System.arraycopy(source, 0, target, 0, length);
}
private static void copyAndInternSegments(String[] target, int targetOffset, String[] source) {
for (int i = 0; i < source.length; i++) {
target[targetOffset + i] = internPathSegment(source[i]);
}
}
private static String internPathSegment(String sample) {
// Intern all String instances added to RelativePath instances to minimize memory use
// by de-duplicating all path segment String instances
return PATH_SEGMENT_STRING_INTERNER.intern(sample);
}
public String[] getSegments() {
return segments;
}
public ListIterator segmentIterator() {
ArrayList content = new ArrayList(Arrays.asList(segments));
return content.listIterator();
}
public boolean isFile() {
return endsWithFile;
}
public String getPathString() {
return CollectionUtils.join("/", segments);
}
public File getFile(File baseDir) {
return new File(baseDir, getPathString());
}
public String getLastName() {
if (segments.length > 0) {
return segments[segments.length - 1];
} else {
return null;
}
}
@Override
public boolean equals(Object o) {
if (this == o) {
return true;
}
if (o == null || getClass() != o.getClass()) {
return false;
}
RelativePath that = (RelativePath) o;
if (endsWithFile != that.endsWithFile) {
return false;
}
if (!Arrays.equals(segments, that.segments)) {
return false;
}
return true;
}
@Override
public int hashCode() {
int result = endsWithFile ? 1 : 0;
result = 31 * result + Arrays.hashCode(segments);
return result;
}
@Override
public String toString() {
return getPathString();
}
/**
* Returns the parent of this path.
*
* @return The parent of this path, or null if this is the root path.
*/
public RelativePath getParent() {
if (segments.length == 0) {
return null;
}
String[] parentSegments = new String[segments.length - 1];
copySegments(parentSegments, segments);
return new RelativePath(false, parentSegments);
}
public static RelativePath parse(boolean isFile, String path) {
return parse(isFile, null, path);
}
public static RelativePath parse(boolean isFile, RelativePath parent, String path) {
String[] names = StringUtils.split(path, FILE_PATH_SEPARATORS);
return new RelativePath(isFile, parent, names);
}
/**
* Returns a copy of this path, with the last name replaced with the given name.
*
* @param name The name.
* @return The path.
*/
public RelativePath replaceLastName(String name) {
String[] newSegments = new String[segments.length];
copySegments(newSegments, segments, segments.length - 1);
newSegments[segments.length - 1] = internPathSegment(name);
return new RelativePath(endsWithFile, newSegments);
}
/**
* Appends the given path to the end of this path.
*
* @param other The path to append
* @return The new path
*/
public RelativePath append(RelativePath other) {
return new RelativePath(other.endsWithFile, this, other.segments);
}
/**
*
Appends the given path to the end of this path.
*
* @param other The path to append
* @return The new path
*/
public RelativePath plus(RelativePath other) {
return append(other);
}
/**
* Appends the given names to the end of this path.
*
* @param segments The names to append.
* @param endsWithFile when true, the new path refers to a file.
* @return The new path.
*/
public RelativePath append(boolean endsWithFile, String... segments) {
return new RelativePath(endsWithFile, this, segments);
}
/**
* Prepends the given names to the start of this path.
*
* @param segments The names to prepend
* @return The new path.
*/
public RelativePath prepend(String... segments) {
return new RelativePath(false, segments).append(this);
}
@Override
public int compareTo(RelativePath o) {
int len1 = segments.length;
int len2 = o.segments.length;
if (len1 != len2) {
return len1 - len2;
}
int lim = Math.min(len1, len2);
String v1[] = segments;
String v2[] = o.segments;
int k = 0;
while (k < lim) {
String c1 = v1[k];
String c2 = v2[k];
int compareResult = c1 == c2 ? 0 : c1.compareTo(c2);
if (compareResult != 0) {
return compareResult;
}
k++;
}
return 0;
}
}