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

org.simpleframework.http.parse.PathParser Maven / Gradle / Ivy

/*
 * PathParser.java February 2001
 *
 * Copyright (C) 2001, Niall Gallagher 
 *
 * 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.simpleframework.http.parse;

import org.simpleframework.http.Path;
import org.simpleframework.util.parse.Parser;

/**
 * This is used to parse a path given as part of a URI. This will  read the
 * path, normalize it, and break it up into its components. The normalization
 * of the path is the conversion of the path given into it's actual path by
 * removing the references to the parent directories and to the current dir.
 * 

* If the path that this represents is /usr/bin/../etc/./README * then the actual path, normalized, is /usr/etc/README. Once * the path has been normalized it is possible to acquire the segments as * an array of strings, which allows simple manipulation of the path. *

* Although RFC 2396 defines the path within a URI to have parameters this * does not extract those parameters this will simply normalize the path and * include the path parameters in the path. If the path is to be converted * into a OS specific file system path that has the parameters extracted * then the AddressParser should be used. * * @author Niall Gallagher */ public class PathParser extends Parser implements Path{ /** * Used to store the individual path segments. */ private TokenList list; /** * Used to store consumed name characters. */ private Token name; /** * Used to store consumed file extension. */ private Token ext; /** * Used to store the highest directory path. */ private Token dir; /** * Used to store consumed normalized path name. */ private Token path; /** * The default constructor will create a PathParser that * contains no specifics. The instance will return null * for all the get methods. The PathParser's get methods * may be populated by using the parse method. */ public PathParser() { this.list = new TokenList(); this.ext = new Token(); this.dir = new Token(); this.path = new Token(); this.name = new Token(); } /** * This is primarily a convineance constructor. This will parse the * String given to extract the specifics. This could be * achived by calling the default no-arg constructor and then using * the instance to invoke the parse method on that * String to extract the parts. * * @param path a String containing a path value */ public PathParser(String path){ this(); parse(path); } /** * This will parse the path in such a way that it ensures that at no * stage there are trailing back references, using path normalization. * The need to remove the back references is so that this * PathParser will create the same String * path given a set of paths that have different back references. For * example the paths /path/../path and /path * are the same path but different String's. *

* This will NOT parse an immediate back reference as this signifies * a path that cannot exist. So a path such as /../ will * result in a null for all methods. Paths such as ../bin * will not be allowed. */ protected void parse() { normalize(); path(); segments(); name(); extension(); } /** * This will initialize the parser so that it is in a ready state. * This allows the parser to be used to parse many paths. This will * clear the parse buffer objects and reset the offset to point to * the start of the char buffer. The count variable is reset by the * Parser.parse method. */ protected void init() { list.clear(); ext.clear(); dir.clear(); name.clear(); path.clear(); off = 0; } /** * This will return the extension that the file name contains. * For example a file name file.en_US.extension * will produce an extension of extension. This * will return null if the path contains no file extension. * * @return this will return the extension this path contains */ public String getExtension() { return ext.toString(); } /** * This will return the full name of the file without the path. * As regargs the definition of the path in RFC 2396 the name * would be considered the last path segment. So if the path * was /usr/README the name is README. * Also for directorys the name of the directory in the last * path segment is returned. This returns the name without any * of the path parameters. As RFC 2396 defines the path to have * path parameters after the path segments. * * @return this will return the name of the file in the path */ public String getName(){ return name.toString(); } /** * This will return the normalized path. The normalized path is * the path without any references to its parent or itself. So * if the path to be parsed is /usr/../etc/./ the * path is /etc/. If the path that this represents * is a path with an immediate back reference then this will * return null. This is the path with all its information even * the parameter information if it was defined in the path. * * @return this returns the normalize path without * ../ or ./ */ public String getPath() { return path.toString(); } /** * This will return the normalized path from the specified path * segment. This allows various path parts to be acquired in an * efficient means what does not require copy operations of the * use of substring invocations. Of particular * interest is the extraction of context based paths. This is * the path with all its information even the parameter * information if it was defined in the path. * * @param from this is the segment offset to get the path for * * @return this returns the normalize path without * ../ or ./ */ public String getPath(int from) { return list.segment(from); } /** * This will return the normalized path from the specified path * segment. This allows various path parts to be acquired in an * efficient means what does not require copy operations of the * use of substring invocations. Of particular * interest is the extraction of context based paths. This is * the path with all its information even the parameter * information if it was defined in the path. * * @param from this is the segment offset to get the path for * @param count this is the number of path segments to include * * @return this returns the normalize path without * ../ or ./ */ public String getPath(int from, int count) { return list.segment(from, count); } /** * This will return the highest directory that exists within * the path. This is used to that files within the same path * can be acquired. An example of that this would do given * the path /pub/./bin/README would be to return * the highest directory path /pub/bin/. The "/" * character will allways be the last character in the path. * * @return this method will return the highest directory */ public String getDirectory(){ return dir.toString(); } /** * This method is used to break the path into individual parts * called segments, see RFC 2396. This can be used as an easy * way to compare paths and to examine the directory tree that * the path points to. For example, if an path was broken from * the string /usr/bin/../etc then the segments * returned would be usr and etc as * the path is normalized before the segments are extracted. * * @return return all the path segments within the directory */ public String[] getSegments(){ return list.list(); } /** * This will return the path as it is relative to the issued * path. This in effect will chop the start of this path if * it's start matches the highest directory of the given path * as of getDirectory. This is useful if paths * that are relative to a specific location are required. To * illustrate what this method will do the following example * is provided. If this object represented the path string * /usr/share/rfc/rfc2396.txt and the issued * path was /usr/share/text.txt then this will * return the path string /rfc/rfc2396.txt. * * @param path the path prefix to acquire a relative path * * @return returns a path relative to the one it is given * otherwize this method will return null */ public String getRelative(String path){ return getRelative(new PathParser(path)); } /** * This is used by the getRelative(String) to * normalize the path string and determine if it contains a * highest directory which is shared with the path that is * represented by this object. If the path has leading back * references, such as ../, then the result of * this is null. The returned path begins with a '/'. * * @param path the path prefix to acquire a relative path * * @return returns a path relative to the one it is given * otherwize this method will return null */ private String getRelative(PathParser path){ char[] text = path.buf; int off = path.dir.off; int len = path.dir.len; return getRelative(text, off, len); } /** * This will return the path as it is relative to the issued * path. This in effect will chop the start of this path if * it's start matches the highest directory of the given path * as of getDirectory. This is useful if paths * that are relative to a specific location are required. To * illustrate what this method will do the following example * is provided. If this object represented the path string * /usr/share/rfc/rfc2396.txt and the issued * path was /usr/share/text.txt then this will * return the path string /rfc/rfc2396.txt. * * @param text the path prefix to acquire a relative path * @param off this is the offset within the text to read * @param len this is the number of characters in the path * * @return returns a path relative to the one it is given * otherwize this method will return null */ private String getRelative(char[] text, int off, int len){ int size = path.len - len + 1; /* '/' */ int pos = path.off + len - 1; for(int i = 0; i < len; i++){ if(text[off++] != buf[path.off+i]){ return null; } } if(pos < 0) { /* ../ */ return null; } return new String(buf,pos,size); } /** * This will extract the path of the given String * after it has been normalized. If the path can not be normalized * then the count is set to -1 and the path cannot be extracted. * When this happens then the path parameter is null. */ private void path() { if(count > 0){ path.len = count; path.off = 0; } } /** * This will simply read the characters from the end of the * buffer until it encounters the first peroid character. When * this is read it will store the file extension and remove the * characters from the buffer. */ private void extension() { int pos = off + count; /* index.html[]*/ int len = 0; while(pos-1 >= off) { /* index.htm[l]*/ if(buf[--pos]=='.'){ /* index[.]html*/ ext.off = pos+1; ext.len = len; count = pos; break; } len++; } } /** * This wil extract each individual segment from the path and * also extract the highest directory. The path segments are * basically the strings delimited by the '/' character of a * normalized path. As well as extracting the path segments * this will also extract the directory of path, that is, the * the path up to the last occurance of the '/' character. */ private void segments() { int pos = count - 1; int len = 1; if(count > 0){ if(buf[pos] == '/'){ /* /pub/bin[/] */ dir.len = pos+1; dir.off = 0; pos--; /* /pub/bi[n]/ */ } while(pos >= off){ if(buf[pos] == '/'){ /* /pub[/]bin/*/ if(dir.len == 0){ dir.len = pos+1; /* [/] is 0*/ dir.off = 0; } list.add(pos+1,len-1); len = 0; } len++; pos--; } } } /** * The normalization of the path is the conversion of the path * given into it's actual path by removing the references to * the parent directorys and to the current dir. So if the path * given was /usr/bin/../etc/./README then the actual * path, the normalized path, is /usr/etc/README. *

* This method ensures the if there are an illegal number of back * references that the path will be evaluated as empty. This can * evaluate any path configuration, this includes any references * like ../ or /.. within the path. * This will also remove empty segments like //. */ private void normalize(){ int size = count + off; int pos = off; for(off = count = 0; pos < size; pos++) { buf[count++] = buf[pos]; if(buf[pos] == '/') { if(count -1 > 0){ if(buf[count -2] == '/') /* [/]/./path/ */ count--; /* /[/]./path/ */ } } else if(buf[pos] == '.') { /* //[.]/path/ */ if(count -1 > 0) { /* /[/]./path/ */ if(buf[count - 2] !='/') /* /[/]./path./ */ continue; /* /path.[/] */ } if(pos + 2 > size){ /* /path/[.] */ count--; } else { if(buf[pos + 1] =='/'){ /* /.[/]path */ pos++;/* /[/]. */ count--; /* /.[/]path */ } if(buf[pos] !='.'){ /* /.[/]path */ continue; } if(pos + 2< size){ if(buf[pos + 2]!='/') /* /..[p]ath */ continue; /* /[.].path */ } if(count - 2 > 0) { for(count -= 2; count - 1 > 0;){ /* /path[/]..*/ if(buf[count - 1]=='/') { /* [/]path/..*/ break; } count--; } }else { /* /../ */ count = 0; off = 0; break; } pos += 2; /* /path/.[.]/ */ } } } } /** * This will extract the full name of the file without the path. * As regards the definition of the path in RFC 2396 the name * would be considered the last path segment. So if the path * was /usr/README the name is README. * Also for directorys the name of the directory in the last * path segment is returned. This returns the name without any * of the path parameters. As RFC 2396 defines the path to have * path parameters after the path segments. So the path for the * directory "/usr/bin;param=value/;param=value" would result * in the name "bin". If the path given was "/" then there will * be nothing in the buffer because extract will * have removed it. */ private void name(){ int pos = count; int len = 0; while(pos-- > off) { /* /usr/bin/;para[m] */ if(buf[pos]==';'){ /* /usr/bin/[;]param */ if(buf[pos-1]=='/'){ /* /usr/bin[/];param */ pos--; /* /usr/bin[/];param */ } len = 0; /* /usr/bin[/]*/ }else if(buf[pos]=='/'){ /* /usr[/]bin*/ off = pos + 1; /* /usr/[b]in*/ count = len; /* [b]in */ break; }else{ len++; } } name.len = count; name.off = off; } /** * This will return the normalized path. The normalized path is * the path without any references to its parent or itself. So * if the path to be parsed is /usr/../etc/./ the * path is /etc/. If the path that this represents * is a path with an immediate back reference then this will * return null. This is the path with all its information even * the parameter information if it was defined in the path. * * @return this returns the normalize path without * ../ or ./ */ public String toString(){ return getPath(); } /** * This is used so that the PathParser can speed * up the parsing of the data. Rather than using a buffer like * a ParseBuffer or worse a StringBuffer * this just keeps an index into the character array from the * start and end of the token. Also this enables a cache to be * kept so that a String does not need to be made * again after the first time it is created. */ private class Token { /** * Provides a quick retrieval of the token value. */ public String value; /** * Offset within the buffer that the token starts. */ public int off; /** * Length of the region that the token consumes. */ public int len; /** * If the Token is to be reused this will clear * all previous data. Clearing the buffer allows it to be * reused if there is a new URI to be parsed. This ensures * that a null is returned if the token length is zero. */ public void clear() { value = null; len = 0; } /** * This method will convert the Token into it's * String equivelant. This will firstly check * to see if there is a value, for the string representation, * if there is the value is returned, otherwise the region * is converted into a String and returned. * * @return this returns a value representing the token */ public String toString() { if(value != null) { return value; } if(len > 0) { value = new String(buf,off,len); } return value; } } /** * The TokenList class is used to store a list of * tokens. This provides an add method which can * be used to store an offset and length of a token within * the buffer. Once the tokens have been added to they can be * examined, in the order they were added, using the provided * list method. This has a scalable capacity. */ private class TokenList { /** * This is used to cache the segments that are created. */ private String[] cache; /** * Contains the offsets and lengths of the tokens. */ private int[] list; /** * Determines the write offset into the array. */ private int count; /** * Constructor for the TokenList is used to * create a scalable list to store tokens. The initial * list is created with an array of sixteen ints, which * is enough to store eight tokens. */ private TokenList(){ list = new int[16]; } /** * This is used to acquire the path from the segment that * is specified. This provides an efficient means to get * the path without having to perform expensive copy of * substring operations. * * @param from this is the path segment to get the path * * @return the string that is the path segment created */ public String segment(int from) { int total = count / 2; int left = total - from; return segment(from, left); } /** * This is used to acquire the path from the segment that * is specified. This provides an efficient means to get * the path without having to perform expensive copy of * substring operations. * * @param from this is the path segment to get the path * @param total this is the number of segments to use * * @return the string that is the path segment created */ public String segment(int from, int total) { int last = list[0] + list[1] + 1; if(from + total < count / 2) { last = offset(from + total); } int start = offset(from); int length = last - start; return new String(buf, start-1, length); } /** * This is used to acquire the offset within the buffer * of the specified segment. This allows a path to be * created that is constructed from a given segment. * * @param segment this is the segment offset to use * * @return this returns the offset start for the segment */ private int offset(int segment) { int last = count - 2; int shift = segment * 2; int index = last - shift; return list[index]; } /** * This is used to add a new token to the list. Tokens * will be available from the list method in * the order it was added, so the first to be added will * at index zero and the last with be in the last index. * * @param off this is the read offset within the buffer * @param len the number of characters within the token */ public void add(int off, int len){ if(count+1 > list.length) { resize(count *2); } list[count++] = off; list[count++] = len; } /** * This is used to retrieve the list of tokens inserted * to this list using the add method. The * indexes of the tokens represents the order that the * tokens were added to the list. * * @return returns an ordered list of token strings */ public String[] list(){ if(cache == null) { cache = build(); } return cache; } /** * This is used to retrieve the list of tokens inserted * to this list using the add method. The * indexes of the tokens represents the order that the * tokens were added to the list. * * @return returns an ordered list of token strings */ private String[] build(){ String[] value = new String[count/2]; for(int i =0, j = count/2; i< count; i+=2){ int index = j - (i/2) - 1; int off = list[i]; int size = list[i + 1]; value[index] = new String(buf, off, size); } return value; } /** * This is used to clear all tokens previously stored * in the list. This is required so that initialization * of the parser with the init method can * ensure that there are no tokens from previous data. */ public void clear(){ cache =null; count =0; } /** * Scales the internal array used should the number of * tokens exceed the initial capacity. This will just * copy across the ints used to represent the token. * * @param size length the capacity is to increase to */ private void resize(int size){ int[] copy = new int[size]; System.arraycopy(list,0,copy,0,count); list = copy; } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy