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

org.jboss.vfs.util.PathTokenizer Maven / Gradle / Ivy

There is a newer version: 3.3.2.Final
Show newest version
/*
* JBoss, Home of Professional Open Source
* Copyright 2006, JBoss Inc., and individual contributors as indicated
* by the @authors tag.
*
 * 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.jboss.vfs.util;

import static org.jboss.vfs.VFSMessages.MESSAGES;

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

import org.jboss.vfs.VFSMessages;

/**
 * PathTokenizer.
 *
 * @author Adrian Brock
 * @author Ales Justin
 * @author David M. Lloyd
 * @version $Revision: 1.1 $
 */
@SuppressWarnings({"StringEquality"})
public class PathTokenizer {

    /**
     * The reverse path const
     */
    private static final String CURRENT_PATH = ".";

    /**
     * The reverse path const
     */
    private static final String REVERSE_PATH = "..";

    /**
     * Token states
     */
    private static final int STATE_INITIAL = 0;
    private static final int STATE_NORMAL = 1;
    private static final int STATE_MAYBE_CURRENT_PATH = 2;
    private static final int STATE_MAYBE_REVERSE_PATH = 3;

    /**
     * Utility class
     */
    private PathTokenizer() {
    }

    /**
     * Get the remaining path from some tokens
     *
     * @param tokens the tokens
     * @param i      the current location
     * @param end    the end index
     * @return the remaining path
     * @throws IllegalArgumentException for null tokens or i is out of range
     */
    protected static String getRemainingPath(List tokens, int i, int end) {
        if (tokens == null) {
            throw MESSAGES.nullArgument("tokens");
        }
        if (i < 0 || i >= end) { throw new IllegalArgumentException("i is not in the range of tokens: 0-" + (end - 1)); }
        if (i == end - 1) { return tokens.get(end - 1); }
        StringBuilder buffer = new StringBuilder();
        for (; i < end - 1; ++i) {
            buffer.append(tokens.get(i));
            buffer.append("/");
        }
        buffer.append(tokens.get(end - 1));
        return buffer.toString();
    }

    /**
     * Get the tokens that comprise this path.
     *
     * @param path the path
     * @return the tokens or null if the path is empty
     * @throws IllegalArgumentException if the path is null
     */
    public static List getTokens(String path) {
        if (path == null) {
            throw MESSAGES.nullArgument("path");
        }
        List list = new ArrayList();
        getTokens(list, path);
        return list;
    }

    /**
     * Get the tokens that comprise this path and append them to the list.
     *
     * @param path the path
     * @return the tokens or null if the path is empty
     * @throws IllegalArgumentException if the path is null
     */
    public static void getTokens(List list, String path) {
        int start = -1, length = path.length(), state = STATE_INITIAL;
        char ch;
        for (int index = 0; index < length; index++) {
            ch = path.charAt(index);
            switch (ch) {
                case '/': {
                    switch (state) {
                        case STATE_INITIAL: {
                            // skip extra leading /
                            continue;
                        }
                        case STATE_MAYBE_CURRENT_PATH: {
                            // it's '.'
                            list.add(CURRENT_PATH);
                            state = STATE_INITIAL;
                            continue;
                        }
                        case STATE_MAYBE_REVERSE_PATH: {
                            // it's '..'
                            list.add(REVERSE_PATH);
                            state = STATE_INITIAL;
                            continue;
                        }
                        case STATE_NORMAL: {
                            // it's just a normal path segment
                            list.add(path.substring(start, index));
                            state = STATE_INITIAL;
                            continue;
                        }
                    }
                    continue;
                }
                case '.': {
                    switch (state) {
                        case STATE_INITIAL: {
                            // . is the first char; might be a special token
                            state = STATE_MAYBE_CURRENT_PATH;
                            start = index;
                            continue;
                        }
                        case STATE_MAYBE_CURRENT_PATH: {
                            // the second . in a row...
                            state = STATE_MAYBE_REVERSE_PATH;
                            continue;
                        }
                        case STATE_MAYBE_REVERSE_PATH: {
                            // the third . in a row, guess it's just a weird path name
                            state = STATE_NORMAL;
                            continue;
                        }
                    }
                    continue;
                }
                default: {
                    switch (state) {
                        case STATE_INITIAL: {
                            state = STATE_NORMAL;
                            start = index;
                            continue;
                        }
                        case STATE_MAYBE_CURRENT_PATH:
                        case STATE_MAYBE_REVERSE_PATH: {
                            state = STATE_NORMAL;
                        }
                    }
                }
            }
        }
        // handle the last token
        switch (state) {
            case STATE_INITIAL: {
                // trailing /
                break;
            }
            case STATE_MAYBE_CURRENT_PATH: {
                list.add(CURRENT_PATH);
                break;
            }
            case STATE_MAYBE_REVERSE_PATH: {
                list.add(REVERSE_PATH);
                break;
            }
            case STATE_NORMAL: {
                list.add(path.substring(start));
                break;
            }
        }
        return;
    }

    /**
     * Get the remaining path from some tokens
     *
     * @param tokens the tokens
     * @param i      the current location
     * @return the remaining path
     * @throws IllegalArgumentException for null tokens or i is out of range
     */
    public static String getRemainingPath(List tokens, int i) {
        if (tokens == null) {
            throw MESSAGES.nullArgument("tokens");
        }
        return getRemainingPath(tokens, i, tokens.size());
    }

    /**
     * Apply any . or .. paths in the path param.
     *
     * @param path the path
     * @return simple path, containing no . or .. paths
     */
    public static String applySpecialPaths(String path) throws IllegalArgumentException {
        List tokens = getTokens(path);
        if (tokens == null) { return null; }
        int i = 0;
        for (int j = 0; j < tokens.size(); j++) {
            String token = tokens.get(j);
            if (isCurrentToken(token)) { continue; } else if (isReverseToken(token)) { i--; } else { tokens.set(i++, token); }
            if (i < 0) {
                throw VFSMessages.MESSAGES.onRootPath();
            }
        }
        return getRemainingPath(tokens, 0, i);
    }

    /**
     * Apply any . or .. paths in the pathTokens parameter, returning the minimal token list.
     *
     * @param pathTokens the path tokens
     * @return the simple path tokens
     * @throws IllegalArgumentException if reverse path goes over the top path
     */
    public static List applySpecialPaths(List pathTokens) throws IllegalArgumentException {
        final ArrayList newTokens = new ArrayList();
        for (String pathToken : pathTokens) {
            if (isCurrentToken(pathToken)) { continue; } else if (isReverseToken(pathToken)) {
                final int size = newTokens.size();
                if (size == 0) {
                    throw VFSMessages.MESSAGES.onRootPath();
                }
                newTokens.remove(size - 1);
            } else { newTokens.add(pathToken); }
        }
        return newTokens;
    }

    /**
     * Is current token.
     *
     * @param token the token to check
     * @return true if token matches current path token
     */
    public static boolean isCurrentToken(String token) {
        return CURRENT_PATH == token;
    }

    /**
     * Is reverse token.
     *
     * @param token the token to check
     * @return true if token matches reverse path token
     */
    public static boolean isReverseToken(String token) {
        return REVERSE_PATH == token;
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy