com.github.jinahya.codec.commons.BinaryDecoderProxy.java Source code

Java tutorial

Introduction

Here is the source code for com.github.jinahya.codec.commons.BinaryDecoderProxy.java

Source

/*
 * Copyright 2012 Jin Kwon <jinahya at gmail.com>.
 *
 * 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 com.github.jinahya.codec.commons;

import java.lang.reflect.Method;

/**
 * Abstract class for proxies of {@link org.apache.commons.codec.BinaryDecoder}.
 *
 * @author Jin Kwon <jinahya at gmail.com>
 * @param <T> decoder type parameter
 */
public abstract class BinaryDecoderProxy<T> extends DecoderProxy<T> {

    private static final String DECODER_NAME = "org.apache.commons.codec.BinaryDecoder";

    /**
     * the class of {@link org.apache.commons.codec.BinaryDecoder}.
     */
    private static final Class<?> DECODER;

    static {
        try {
            DECODER = Class.forName(DECODER_NAME);
        } catch (final ClassNotFoundException cnfe) {
            throw new InstantiationError(cnfe.getMessage());
        }
    }

    /**
     * the method of {@code decode([b)[b}.
     */
    private static final Method DECODE;

    static {
        try {
            DECODE = DECODER.getMethod("decode", byte[].class);
        } catch (final NoSuchMethodException nsme) {
            throw new InstantiationError(nsme.getMessage());
        }
    }

    /**
     * Creates a new proxy instance.
     *
     * @param <P> proxy type parameter.
     * @param <T> decoder type parameter
     * @param proxyType proxy type
     * @param decoderType decoder type
     * @param decoder decoder
     *
     * @return a new (proxy) instance.
     */
    protected static <P extends AbstractDecoderProxy<T>, T> Object newInstance(final Class<P> proxyType,
            final Class<T> decoderType, final T decoder) {

        if (proxyType == null) {
            throw new NullPointerException("proxyType");
        }

        if (!BinaryDecoderProxy.class.isAssignableFrom(proxyType)) {
            throw new IllegalArgumentException(
                    "proxyType(" + proxyType + ") is not assignable to " + BinaryDecoderProxy.class);
        }

        return newInstance(DECODER.getClassLoader(), new Class<?>[] { DECODER }, proxyType, decoderType, decoder);
    }

    /**
     * Creates a new instance.
     *
     * @param decoder the decoder to delegate.
     */
    protected BinaryDecoderProxy(final T decoder) {

        super(decoder);
    }

    @Override
    public Object invoke(final Object proxy, final Method method, final Object[] args) throws Throwable {

        if (DECODE.equals(method)) {
            return decode((byte[]) args[0]);
        }

        return super.invoke(proxy, method, args);
    }

    @Override
    protected Object decode(final Object source) throws Throwable {

        try {
            return decode((byte[]) source);
        } catch (final ClassCastException cce) {
            throw newDecoderException(cce);
        }
    }

    /**
     * Decodes given {@code source}.
     *
     * @param source data to decode.
     *
     * @return decoded result
     *
     * @throws Throwable if an error occurs.
     */
    protected abstract byte[] decode(final byte[] source) throws Throwable;

}