All Downloads are FREE. Search and download functionalities are using the official Maven repository.

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; } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy