/*
 * 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.
 */

#ifndef _DECAF_NET_MALFORMEDURLEXCEPTION_H_
#define _DECAF_NET_MALFORMEDURLEXCEPTION_H_

#include <decaf/util/Config.h>
#include <decaf/io/IOException.h>

namespace decaf{
namespace net{

    class DECAF_API MalformedURLException  : public io::IOException {
    public:

        /**
         * Default Constructor
         */
        MalformedURLException() throw() {}

        /**
         * Conversion Constructor from some other Exception
         * @param An exception that should become this type of Exception
         */
        MalformedURLException( const Exception& ex ) throw()
        : io::IOException()
        {
            *(Exception*)this = ex;
        }

        /**
         * Copy Constructor
         */
        MalformedURLException( const MalformedURLException& ex ) throw()
        : io::IOException()
        {
            *(Exception*)this = ex;
        }

        /**
         * Constructor - Initializes the file name and line number where
         * this message occurred.  Sets the message to report, using an
         * optional list of arguments to parse into the message
         * @param file name where exception occurs
         * @param line number where the exception occurred.
         * @param cause The exception that was the cause for this one to be thrown.
         * @param message to report
         * @param list of primitives that are formatted into the message
         */
        MalformedURLException( const char* file, const int lineNumber,
                               const std::exception* cause,
                               const char* msg, ... )
        throw() : io::IOException( cause )
        {
            va_list vargs;
            va_start( vargs, msg );
            buildMessage( msg, vargs );

            // Set the first mark for this exception.
            setMark( file, lineNumber );
        }

        /**
         * Constructor
         * @param cause Pointer to the exception that caused this one to
         * be thrown, the object is cloned caller retains ownership.
         */
        MalformedURLException( const std::exception* cause )
            throw() : io::IOException( cause ) {}

        /**
         * Constructor - Initializes the file name and line number where
         * this message occurred.  Sets the message to report, using an
         * optional list of arguments to parse into the message
         * @param file name where exception occurs
         * @param line number where the exception occurred.
         * @param message to report
         * @param list of primitives that are formatted into the message
         */
        MalformedURLException( const char* file, const int lineNumber,
                               const char* msg, ... ) throw ()
        : io::IOException()
        {
            va_list vargs ;
            va_start( vargs, msg );
            buildMessage( msg, vargs );

            // Set the first mark for this exception.
            setMark( file, lineNumber );
        }

        /**
         * Clones this exception.  This is useful for cases where you need
         * to preserve the type of the original exception as well as the message.
         * All subclasses should override.
         */
        virtual MalformedURLException* clone() const {
            return new MalformedURLException( *this );
        }

        /**
         * Destructor
         */
        virtual ~MalformedURLException() throw() {}

    };

}}

#endif /*_DECAF_NET_MALFORMEDURLEXCEPTION_H_*/
