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

org.gradle.util.Path Maven / Gradle / Ivy

There is a newer version: 8.11.1
Show newest version
/*
 * Copyright 2007 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.util;

import com.google.common.base.Strings;
import org.apache.commons.lang.StringUtils;
import org.gradle.api.InvalidUserDataException;
import org.gradle.util.internal.GUtil;

import javax.annotation.Nullable;
import java.util.Arrays;
import java.util.Comparator;

/**
 * Represents a path in Gradle.
 */
public class Path implements Comparable {
    public static final Path ROOT = new Path(new String[0], true);

    private static final Comparator STRING_COMPARATOR = GUtil.caseInsensitive();
    public static final String SEPARATOR = ":";

    public static Path path(@Nullable String path) {
        if (Strings.isNullOrEmpty(path)) {
            throw new InvalidUserDataException("A path must be specified!");
        }
        if (path.equals(SEPARATOR)) {
            return ROOT;
        } else {
            return parsePath(path);
        }
    }

    private static Path parsePath(String path) {
        String[] segments = StringUtils.split(path, SEPARATOR);
        boolean absolute = path.startsWith(SEPARATOR);
        return new Path(segments, absolute);
    }

    private final String[] segments;
    private final boolean absolute;
    private String fullPath;

    private Path(String[] segments, boolean absolute) {
        this.segments = segments;
        this.absolute = absolute;
    }

    @Override
    public String toString() {
        return getPath();
    }

    /**
     * Appends the supplied path to this path, returning the new path.
     * The resulting path with be absolute or relative based on the path being appended _to_.
     * It makes no difference if the _appended_ path is absolute or relative.
     *
     * 
     * path(':a:b').append(path(':c:d')) == path(':a:b:c:d')
     * path(':a:b').append(path('c:d')) == path(':a:b:c:d')
     * path('a:b').append(path(':c:d')) == path('a:b:c:d')
     * path('a:b').append(path('c:d')) == path('a:b:c:d')
     * 
*/ public Path append(Path path) { if (path.segments.length == 0) { return this; } String[] concat = new String[segments.length + path.segments.length]; System.arraycopy(segments, 0, concat, 0, segments.length); System.arraycopy(path.segments, 0, concat, segments.length, path.segments.length); return new Path(concat, absolute); } public String getPath() { if (fullPath == null) { fullPath = createFullPath(); } return fullPath; } private String createFullPath() { StringBuilder path = new StringBuilder(); if (absolute) { path.append(SEPARATOR); } for (int i = 0; i < segments.length; i++) { if (i > 0) { path.append(SEPARATOR); } String segment = segments[i]; path.append(segment); } return path.toString(); } @Override public boolean equals(@Nullable Object o) { if (this == o) { return true; } if (o == null || getClass() != o.getClass()) { return false; } Path path = (Path) o; if (absolute != path.absolute) { return false; } return Arrays.equals(segments, path.segments); } @Override public int hashCode() { int result = Arrays.hashCode(segments); result = 31 * result + (absolute ? 1 : 0); return result; } @Override public int compareTo(Path other) { if (absolute && !other.absolute) { return 1; } if (!absolute && other.absolute) { return -1; } for (int i = 0; i < Math.min(segments.length, other.segments.length); i++) { int diff = STRING_COMPARATOR.compare(segments[i], other.segments[i]); if (diff != 0) { return diff; } } int lenDiff = segments.length - other.segments.length; if (lenDiff > 0) { return 1; } if (lenDiff < 0) { return -1; } return 0; } /** * Returns the parent of this path, or null if this path has no parent. * * @return The parent of this path. */ @Nullable public Path getParent() { if (segments.length == 0) { return null; } if (segments.length == 1) { return absolute ? ROOT : null; } String[] parentPath = new String[segments.length - 1]; System.arraycopy(segments, 0, parentPath, 0, parentPath.length); return new Path(parentPath, absolute); } /** * Returns the base name of this path, or null if this path is the root path. * * @return The base name, */ @Nullable public String getName() { if (segments.length == 0) { return null; } return segments[segments.length - 1]; } /** * Creates a child of this path with the given name. */ public Path child(String name) { String[] childSegments = new String[segments.length + 1]; System.arraycopy(segments, 0, childSegments, 0, segments.length); childSegments[segments.length] = name; return new Path(childSegments, absolute); } /** * Resolves the given name relative to this path. If an absolute path is provided, it is returned. */ public String absolutePath(String path) { return absolutePath(path(path)).getPath(); } public Path absolutePath(Path path) { if (path.absolute) { return path; } return append(path); } public boolean isAbsolute() { return absolute; } /** * Calculates a path relative to this path. If the given path is not a child of this path, it is returned unmodified. */ public String relativePath(String path) { return relativePath(path(path)).getPath(); } public Path relativePath(Path path) { if (path.absolute != absolute) { return path; } if (path.segments.length < segments.length) { return path; } for (int i = 0; i < segments.length; i++) { if (!path.segments[i].equals(segments[i])) { return path; } } if (path.segments.length == segments.length) { return path; } String[] newSegments = Arrays.copyOfRange(path.segments, segments.length, path.segments.length); return new Path(newSegments, false); } public int segmentCount() { return segments.length; } public Path removeFirstSegments(int n) { if (n == 0) { return this; } else if (n < 0 || n >= segments.length) { throw new IllegalArgumentException("Cannot remove " + n + " segments from path " + getPath()); } return new Path(Arrays.copyOfRange(segments, n, segments.length), absolute); } public String segment(int index) { if (index < 0 || index >= segments.length) { throw new IllegalArgumentException("Segment index " + index + " is invalid for path " + getPath()); } return segments[index]; } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy