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

io.undertow.websockets.jsr.Encoding Maven / Gradle / Ivy

/*
 * JBoss, Home of Professional Open Source.
 * Copyright 2014 Red Hat, Inc., and individual contributors
 * as indicated by the @author tags.
 *
 * 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 io.undertow.websockets.jsr;

import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.Closeable;
import java.io.IOException;
import java.io.StringReader;
import java.io.StringWriter;
import java.nio.ByteBuffer;
import java.util.List;
import java.util.Map;

import javax.websocket.DecodeException;
import javax.websocket.Decoder;
import javax.websocket.EncodeException;
import javax.websocket.Encoder;

import io.undertow.servlet.api.InstanceHandle;

/**
 * Manages all encoders and decoders for an endpoint instance
 *
 * @author Stuart Douglas
 */
public class Encoding implements Closeable {


    private final Map, List>> binaryEncoders;
    private final Map, List>> binaryDecoders;
    private final Map, List>> textEncoders;
    private final Map, List>> textDecoders;

    public Encoding(final Map, List>> binaryEncoders, final Map, List>> binaryDecoders, final Map, List>> textEncoders, final Map, List>> textDecoders) {
        this.binaryEncoders = binaryEncoders;
        this.binaryDecoders = binaryDecoders;
        this.textEncoders = textEncoders;
        this.textDecoders = textDecoders;
    }


    public boolean canEncodeText(final Class type) {
        if(textEncoders.containsKey(type)) {
            return true;
        }
        for(Class key : textEncoders.keySet()) {
            if(key.isAssignableFrom(type)) {
                return true;
            }
        }
        if (EncodingFactory.isPrimitiveOrBoxed(type)) {
            Class primType = boxedType(type);
            return !binaryEncoders.containsKey(primType) && !binaryEncoders.containsKey(Object.class); //don't use a built in coding if a user supplied binary one is present
        }
        return false;
    }


    public boolean canDecodeText(final Class type) {
        if(textDecoders.containsKey(type)) {
            return true;
        }
        if (EncodingFactory.isPrimitiveOrBoxed(type)) {
            Class primType = boxedType(type);
            return !binaryDecoders.containsKey(primType) && !binaryEncoders.containsKey(Object.class); //don't use a built in coding if a user supplied binary one is present
        }
        return false;
    }


    public boolean canEncodeBinary(final Class type) {
        if(binaryEncoders.containsKey(type)) {
            return true;
        }

        for(Class key : binaryEncoders.keySet()) {
            if(key.isAssignableFrom(type)) {
                return true;
            }
        }
        return false;
    }


    public boolean canDecodeBinary(final Class type) {
        return binaryDecoders.containsKey(type);
    }


    public Object decodeText(final Class targetType, final String message) throws DecodeException {
        if (EncodingFactory.isPrimitiveOrBoxed(targetType)) {
            return decodePrimitive(targetType, message);
        }
        List> decoders = textDecoders.get(targetType);
        if (decoders != null) {
            for (InstanceHandle decoderHandle : decoders) {
                Decoder decoder = decoderHandle.getInstance();
                if (decoder instanceof Decoder.Text) {
                    if (((Decoder.Text) decoder).willDecode(message)) {
                        return ((Decoder.Text) decoder).decode(message);
                    }
                } else {
                    try {
                        return ((Decoder.TextStream) decoder).decode(new StringReader(message));
                    } catch (IOException e) {
                        throw new DecodeException(message, "Could not decode string", e);
                    }
                }
            }
        }
        throw new DecodeException(message, "Could not decode string");
    }

    private Object decodePrimitive(final Class targetType, final String message) throws DecodeException {
        if (targetType == Boolean.class || targetType == boolean.class) {
            return Boolean.valueOf(message);
        } else if (targetType == Character.class || targetType == char.class) {
            return message.charAt(0);
        } else if (targetType == Byte.class || targetType == byte.class) {
            return Byte.valueOf(message);
        } else if (targetType == Short.class || targetType == short.class) {
            return Short.valueOf(message);
        } else if (targetType == Integer.class || targetType == int.class) {
            return Integer.valueOf(message);
        } else if (targetType == Long.class || targetType == long.class) {
            return Long.valueOf(message);
        } else if (targetType == Float.class || targetType == float.class) {
            return Float.valueOf(message);
        } else if (targetType == Double.class || targetType == double.class) {
            return Double.valueOf(message);
        }
        return null; // impossible
    }

    public Object decodeBinary(final Class targetType, final byte[] bytes) throws DecodeException {
        List> decoders = binaryDecoders.get(targetType);
        if (decoders != null) {
            for (InstanceHandle decoderHandle : decoders) {
                Decoder decoder = decoderHandle.getInstance();
                if (decoder instanceof Decoder.Binary) {
                    if (((Decoder.Binary) decoder).willDecode(ByteBuffer.wrap(bytes))) {
                        return ((Decoder.Binary) decoder).decode(ByteBuffer.wrap(bytes));
                    }
                } else {
                    try {
                        return ((Decoder.BinaryStream) decoder).decode(new ByteArrayInputStream(bytes));
                    } catch (IOException e) {
                        throw new DecodeException(ByteBuffer.wrap(bytes), "Could not decode binary", e);
                    }
                }
            }
        }
        throw new DecodeException(ByteBuffer.wrap(bytes), "Could not decode binary");
    }

    public String encodeText(final Object o) throws EncodeException {
        List> encoders = textEncoders.get(o.getClass());
        if(encoders == null) {
            for(Map.Entry, List>> entry : textEncoders.entrySet()) {
                if(entry.getKey().isAssignableFrom(o.getClass())) {
                    encoders = entry.getValue();
                    break;
                }
            }
        }
        if (encoders != null) {
            for (InstanceHandle decoderHandle : encoders) {
                Encoder decoder = decoderHandle.getInstance();
                if (decoder instanceof Encoder.Text) {
                    return ((Encoder.Text) decoder).encode(o);
                } else {
                    try {
                        StringWriter out = new StringWriter();
                        ((Encoder.TextStream) decoder).encode(o, out);
                        return out.toString();
                    } catch (IOException e) {
                        throw new EncodeException(o, "Could not encode text", e);
                    }
                }
            }
        }

        if (EncodingFactory.isPrimitiveOrBoxed(o.getClass())) {
            return o.toString();
        }
        throw new EncodeException(o, "Could not encode text");
    }

    public ByteBuffer encodeBinary(final Object o) throws EncodeException {
        List> encoders = binaryEncoders.get(o.getClass());

        if(encoders == null) {
            for(Map.Entry, List>> entry : binaryEncoders.entrySet()) {
                if(entry.getKey().isAssignableFrom(o.getClass())) {
                    encoders = entry.getValue();
                    break;
                }
            }
        }
        if (encoders != null) {
            for (InstanceHandle decoderHandle : encoders) {
                Encoder decoder = decoderHandle.getInstance();
                if (decoder instanceof Encoder.Binary) {
                    return ((Encoder.Binary) decoder).encode(o);
                } else {
                    try {
                        ByteArrayOutputStream out = new ByteArrayOutputStream();
                        ((Encoder.BinaryStream) decoder).encode(o, out);
                        return ByteBuffer.wrap(out.toByteArray());
                    } catch (IOException e) {
                        throw new EncodeException(o, "Could not encode binary", e);
                    }
                }
            }
        }
        throw new EncodeException(o, "Could not encode binary");
    }

    @Override
    public void close() {
        for (Map.Entry, List>> entry : binaryDecoders.entrySet()) {
            for (InstanceHandle val : entry.getValue()) {
                val.getInstance().destroy();
                val.release();
            }
        }
        for (Map.Entry, List>> entry : textDecoders.entrySet()) {
            for (InstanceHandle val : entry.getValue()) {
                val.getInstance().destroy();
                val.release();
            }
        }
        for (Map.Entry, List>> entry : binaryEncoders.entrySet()) {
            for (InstanceHandle val : entry.getValue()) {
                val.getInstance().destroy();
                val.release();
            }
        }
        for (Map.Entry, List>> entry : textEncoders.entrySet()) {
            for (InstanceHandle val : entry.getValue()) {
                val.getInstance().destroy();
                val.release();
            }
        }
    }

    private static Class boxedType(final Class targetType) {
        if (targetType == Boolean.class || targetType == boolean.class) {
            return Boolean.class;
        } else if (targetType == Character.class || targetType == char.class) {
            return Character.class;
        } else if (targetType == Byte.class || targetType == byte.class) {
            return Byte.class;
        } else if (targetType == Short.class || targetType == short.class) {
            return Short.class;
        } else if (targetType == Integer.class || targetType == int.class) {
            return Integer.class;
        } else if (targetType == Long.class || targetType == long.class) {
            return Long.class;
        } else if (targetType == Float.class || targetType == float.class) {
            return Float.class;
        } else if (targetType == Double.class || targetType == double.class) {
            return Double.class;
        }
        return targetType;
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy