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

net.sourceforge.urin.Path Maven / Gradle / Ivy

There is a newer version: 5.2
Show newest version
/*
 * Copyright 2012 Mark Slater
 *
 * 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 net.sourceforge.urin;

import java.util.ArrayList;
import java.util.Collections;
import java.util.List;

import static java.util.Arrays.asList;
import static net.sourceforge.urin.Segment.segment;

/**
 * An iterable of {@code Segment}s.
 * {@code Path}s can either be absolute (starting with '/'), or rootless (not starting with '/').
 * 

* Immutable and threadsafe. * * @see RFC 3986 - Path */ public abstract class Path implements Iterable> { /** * Factory method for creating rootless {@code Path}s from {@code String}s. * * @param firstSegment a {@code String} representing the first segment. * @param segments any further segments. * @return a {@code Path} representing the given {@code String}s. */ public static Path rootlessPath(final String firstSegment, final String... segments) { final List> segmentList = new ArrayList<>(segments.length + 1); segmentList.add(segment(firstSegment)); for (String segment : segments) { segmentList.add(segment(segment)); } return rootlessPath(segmentList); } /** * Factory method for creating empty rootless {@code Path}s. * * @return an empty {@code Path}. */ public static Path rootlessPath() { return rootlessPath(Collections.>emptyList()); } /** * Factory method for creating rootless {@code Path}s from {@code Segment}s. * * @param segments {@code Segment}s that will be represented by this {@code Path}. * @return a {@code Path} representing the given {@code Segment}s. */ @SafeVarargs public static Path rootlessPath(final Segment... segments) { return rootlessPath(asList(segments)); } /** * Factory method for creating rootless {@code Path}s from an {@code Iterable} of {@code Segment}s. * * @param segments {@code Iterable} of {@code Segment}s that will be represented by this {@code Path}. * @return a {@code Path} representing the given {@code Segment}s. */ public static Path rootlessPath(final Iterable> segments) { if (segments.iterator().hasNext()) { return new RootlessPath<>(segments); } else { return new EmptyPath<>(); } } /** * Factory method for creating {@code AbsolutePath}s from {@code String}s. * * @param firstSegment a {@code String} representing the first segment. * @param segments any further segments. * @return a {@code AbsolutePath} representing the given {@code String}s. */ public static AbsolutePath path(final String firstSegment, final String... segments) { final List> segmentList = new ArrayList<>(segments.length + 1); segmentList.add(segment(firstSegment)); for (String segment : segments) { segmentList.add(segment(segment)); } return path(segmentList); } /** * Factory method for creating an empty {@code AbsolutePath}. * * @return an empty {@code AbsolutePath}. */ public static AbsolutePath path() { return path(Collections.>emptyList()); } /** * Factory method for creating {@code AbsolutePath}s from {@code Segment}s. * * @param segments {@code Segment}s that will be represented by this {@code AbsolutePath}. * @return a {@code AbsolutePath} representing the given {@code Segment}s. */ @SafeVarargs public static AbsolutePath path(final Segment... segments) { return path(asList(segments)); } /** * Factory method for creating {@code AbsolutePath}s from an {@code Iterable} of {@code Segment}s. * * @param segments {@code Iterable} of {@code Segment}s that will be represented by this {@code AbsolutePath}. * @return a {@code AbsolutePath} representing the given {@code Segment}s. */ public static AbsolutePath path(final Iterable> segments) { return new AbsolutePath<>(segments); } static Path parseRootlessPath(final String rawPath, final MakingDecoder, ?, String> segmentMakingDecoder) throws ParseException { return rootlessPath(rawPath == null || "".equals(rawPath) ? new ArrayList>() : new ArrayList>() {{ for (String segmentString : rawPath.split("/")) { add(Segment.parse(segmentString, segmentMakingDecoder)); } }}); } static AbsolutePath parsePath(final String rawPath, final MakingDecoder, ?, String> segmentMakingDecoder) throws ParseException { return path("/".equals(rawPath) ? new ArrayList>() : new ArrayList>() {{ boolean isFirst = true; for (String segmentString : rawPath.split("/", -1)) { if (!isFirst) { add(Segment.parse(segmentString, segmentMakingDecoder)); } isFirst = false; } }}); } Path() { } abstract boolean firstPartIsSuppliedButIsEmpty(); abstract String asString(final PrefixWithDotSegmentCriteria prefixWithDotSegmentCriteria); abstract boolean isEmpty(); abstract boolean firstPartIsSuppliedButContainsColon(); abstract Path resolveRelativeTo(final Path basePath); abstract Path replaceLastSegmentWith(final Iterable> segments); /** * Indicates whether this path is absolute (begins with '/') or not. * * @return whether this path is absolute (begins with '/') or not. */ public abstract boolean isAbsolute(); /** * Returns the list of {@code Segment}s that this path represents. * Note that this {@code List} contains the same elements as returned by calling {@code iterator()} on {@code this}. * * @return the list of {@code Segment}s that this path represents. */ public abstract List> segments(); static enum PrefixWithDotSegmentCriteria { NEVER_PREFIX_WITH_DOT_SEGMENT { @Override boolean matches(final Path path) { return false; } }, PREFIX_WITH_DOT_SEGMENT_IF_FIRST_CONTAINS_COLON { @Override boolean matches(final Path path) { return path.firstPartIsSuppliedButContainsColon(); } }, PREFIX_WITH_DOT_SEGMENT_IF_FIRST_IS_EMPTY { @Override boolean matches(final Path path) { return path.firstPartIsSuppliedButIsEmpty(); } }, PREFIX_WITH_DOT_SEGMENT_IF_FIRST_IS_EMPTY_OR_CONTAINS_COLON { @Override boolean matches(final Path path) { return path.firstPartIsSuppliedButIsEmpty() || path.firstPartIsSuppliedButContainsColon(); } }; abstract boolean matches(final Path path); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy