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

org.javacc.mojo.UrlUtils Maven / Gradle / Ivy

package org.javacc.mojo;

/*
 * Licensed to the Apache Software Foundation (ASF) under one
 * or more contributor license agreements.  See the NOTICE file 
 * distributed with this work for additional information
 * regarding copyright ownership.  The ASF licenses this file
 * to you 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.
 */

import java.io.File;
import java.net.MalformedURLException;
import java.net.URL;
import java.nio.ByteBuffer;
import java.nio.charset.Charset;

/**
 * Assists in handling of URLs.
 * 
 * @author Benjamin Bentmann
 * @version $Id$
 */
class UrlUtils
{

    /**
     * The UTF-8 character set, used to decode octets in URLs.
     */
    private static final Charset UTF8 = Charset.forName( "UTF-8" );

    /**
     * The protocol prefix for "jar:" URLs.
     */
    private static final String JAR = "jar:";

    /**
     * The protocol prefix for "file:" URLs.
     */
    private static final String FILE = "file:";

    /**
     * The protocol prefix for "jar:file:" URLs.
     */
    private static final String JAR_FILE = JAR + FILE;

    /**
     * Gets the absolute filesystem path to the class path root for the specified resource. The root is either a JAR
     * file or a directory with loose class files. If the URL does not use a supported protocol, an exception will be
     * thrown.
     * 
     * @param url The URL to the resource, may be null.
     * @param resource The name of the resource, must not be null.
     * @return The absolute filesystem path to the class path root of the resource or null if the input
     *         URL was null.
     */
    public static File getResourceRoot( URL url, String resource )
    {
        String path = null;
        if ( url != null )
        {
            String spec = url.toExternalForm();
            if ( ( JAR_FILE ).regionMatches( true, 0, spec, 0, JAR_FILE.length() ) )
            {
                URL jar;
                try
                {
                    jar = new URL( spec.substring( JAR.length(), spec.lastIndexOf( "!/" ) ) );
                }
                catch ( MalformedURLException e )
                {
                    throw new IllegalArgumentException( "Invalid JAR URL: " + url + ", " + e.getMessage() );
                }
                path = decodeUrl( jar.getPath() );
            }
            else if ( FILE.regionMatches( true, 0, spec, 0, FILE.length() ) )
            {
                path = decodeUrl( url.getPath() );
                path = path.substring( 0, path.length() - resource.length() );
            }
            else
            {
                throw new IllegalArgumentException( "Invalid class path URL: " + url );
            }
        }
        return ( path != null ) ? new File( path ) : null;
    }

    /**
     * Decodes the specified URL as per RFC 3986, i.e. transforms percent-encoded octets to characters by decoding with
     * the UTF-8 character set. This function is primarily intended for usage with {@link java.net.URL} which
     * unfortunately does not enforce proper URLs. As such, this method will leniently accept invalid characters or
     * malformed percent-encoded octets and simply pass them literally through to the result string. Except for rare
     * edge cases, this will make unencoded URLs pass through unaltered.
     * 
     * @param url The URL to decode, may be null.
     * @return The decoded URL or null if the input was null.
     */
    public static String decodeUrl( String url )
    {
        String decoded = url;
        if ( url != null && url.indexOf( '%' ) >= 0 )
        {
            int n = url.length();
            StringBuffer buffer = new StringBuffer();
            ByteBuffer bytes = ByteBuffer.allocate( n );
            for ( int i = 0; i < n; )
            {
                if ( url.charAt( i ) == '%' )
                {
                    try
                    {
                        do
                        {
                            byte octet = (byte) Integer.parseInt( url.substring( i + 1, i + 3 ), 16 );
                            bytes.put( octet );
                            i += 3;
                        }
                        while ( i < n && url.charAt( i ) == '%' );
                        continue;
                    }
                    catch ( RuntimeException e )
                    {
                        // malformed percent-encoded octet, fall through and append characters literally
                    }
                    finally
                    {
                        if ( bytes.position() > 0 )
                        {
                            bytes.flip();
                            buffer.append( UTF8.decode( bytes ).toString() );
                            bytes.clear();
                        }
                    }
                }
                buffer.append( url.charAt( i++ ) );
            }
            decoded = buffer.toString();
        }
        return decoded;
    }

}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy