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

org.firebirdsql.gds.impl.wire.XdrOutputStream Maven / Gradle / Ivy

There is a newer version: 4.0.10.java8
Show newest version
/*
 * Firebird Open Source JavaEE Connector - JDBC Driver
 *
 * Distributable under LGPL license.
 * You may obtain a copy of the License at http://www.gnu.org/copyleft/lgpl.html
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * LGPL License for more details.
 *
 * This file was created by members of the firebird development team.
 * All individual contributions remain the Copyright (C) of those
 * individuals.  Contributors to this file are either listed here or
 * can be obtained from a source control history command.
 *
 * All rights reserved.
 */
/*
 * The Original Code is the Firebird Java GDS implementation.
 *
 * The Initial Developer of the Original Code is Alejandro Alberola.
 * Portions created by Alejandro Alberola are Copyright (C) 2001
 * Boix i Oltra, S.L. All Rights Reserved.
 */
package org.firebirdsql.gds.impl.wire;

import org.firebirdsql.encodings.Encoding;
import org.firebirdsql.gds.ISCConstants;
import org.firebirdsql.gds.ParameterBuffer;

import javax.crypto.Cipher;
import javax.crypto.CipherOutputStream;
import java.io.BufferedOutputStream;
import java.io.IOException;
import java.io.OutputStream;
import java.util.Arrays;

/**
 * An XdrOutputStream writes data in XDR format to an
 * underlying java.io.OutputStream.
 * 

* This class is not thread-safe. *

* * @author Alejandro Alberola * @author David Jencks * @author Mark Rotteveel * @version 1.0 */ public final class XdrOutputStream extends OutputStream { private static final int BUF_SIZE = 32767; public static final int SPACE_BYTE = 0x20; public static final int NULL_BYTE = 0x0; private static final int TEXT_PAD_LENGTH = BUF_SIZE; private static final byte[] TEXT_PAD = createPadding(BUF_SIZE, SPACE_BYTE); private static final int ZERO_PAD_LENGTH = 3; private static final byte[] ZERO_PADDING = new byte[ZERO_PAD_LENGTH]; private final boolean buffered; private OutputStream out; private OutputStream directStream; private final OutputStream rawStream; // TODO In a lot of cases the padding written in this class should be NULL_BYTE instead of SPACE_BYTE /** * Create a new instance of XdrOutputStream with buffering. * * @param out * The underlying OutputStream to write to */ public XdrOutputStream(OutputStream out) { this(out, true); } /** * Create a new instance of XdrOutputStream with either a buffered or an unbuffered stream. * * @param out * The underlying OutputStream to write to * @param buffered * {@code true} Uses buffering (like {@link #XdrOutputStream(java.io.OutputStream)} or * {@code false} writes directly to provided {@code OutputStream}. */ public XdrOutputStream(OutputStream out, boolean buffered) { this.buffered = buffered; this.directStream = out; this.rawStream = out; if (buffered) { this.out = new BufferedOutputStream(out, BUF_SIZE); } else { this.out = out; } } /** * Writes the 0x00 alignment for the specified length. This padding is calculated as * (4 - length) & 3. * * @param length * The length of the previously written buffer to pad * @throws IOException * if an error occurs while writing to the * underlying output stream */ public void writeAlignment(int length) throws IOException { out.write(ZERO_PADDING, 0, (4 - length) & 3); } /** * Writes zero padding of the specified length * * @param length Length to write * @throws IOException if an error occurs while writing to the underlying output stream * @see #writePadding(int, int) */ public void writeZeroPadding(int length) throws IOException { byte[] padding = length <= ZERO_PAD_LENGTH ? ZERO_PADDING : new byte[length]; out.write(padding, 0, length); } /** * Writes space ({@code 0x20}) padding of the specified length * * @param length Length to write * @throws IOException if an error occurs while writing to the underlying output stream * @see #writePadding(int, int) */ public void writeSpacePadding(int length) throws IOException { byte[] padding = length <= TEXT_PAD_LENGTH ? TEXT_PAD : createPadding(length, SPACE_BYTE); out.write(padding, 0, length); } /** * Writes padding for the specified length of the specified padding byte. *

* Prefer using the more specific {@link #writeZeroPadding(int)} and {@link #writeZeroPadding(int)}. *

* * @param length * Length of padding to write * @param padByte * Padding byte to use * @throws IOException * if an error occurs while writing to the underlying output stream * @see #writeSpacePadding(int) * @see #writeZeroPadding(int) */ public void writePadding(int length, int padByte) throws IOException { final byte[] padding; if (padByte == SPACE_BYTE && length <= TEXT_PAD.length) { padding = TEXT_PAD; } else if (padByte == NULL_BYTE && length <= ZERO_PADDING.length) { padding = ZERO_PADDING; } else { padding = createPadding(length, (byte) padByte); } out.write(padding, 0, length); } /** * Creates a padding array. * * @param length Length of array * @param padByte Byte value for filling the array * @return Array filled with {@code padByte} */ private static byte[] createPadding(int length, int padByte) { byte[] padding = new byte[length]; Arrays.fill(padding, (byte) padByte); return padding; } /** * Write a byte buffer to the underlying output stream in * XDR format. * * @param buffer The byte buffer to be written * @throws IOException if an error occurs while writing to the * underlying output stream */ public void writeBuffer(byte[] buffer) throws IOException { if (buffer == null) writeInt(0); else { int len = buffer.length; writeInt(len); write(buffer, 0, len, (4 - len) & 3); } } /** * Write content of the specified string using the specified encoding. */ public void writeString(String s, Encoding encoding) throws IOException { if (encoding != null) { byte[] buffer = encoding.encodeToCharset(s); writeBuffer(buffer); } else { // TODO Remove this option (always require encoding) writeBuffer(s.getBytes()); } } /** * Write an Xdrable to this output stream. * * @param type Type of the Xdrable to be written, * e.g. {@link ISCConstants#isc_tpb_version3} * @param item The object to be written * @throws IOException if an error occurs while writing to the * underlying output stream */ public void writeTyped(int type, Xdrable item) throws IOException { int size; if (item == null) { writeInt(1); write(type); //e.g. isc_tpb_version3 size = 1; } else { size = item.getLength() + 1; writeInt(size); write(type); //e.g. isc_tpb_version3 item.write(this); } writeAlignment(size); } public void writeTyped(ParameterBuffer parameterBuffer) throws IOException { writeTyped(parameterBuffer.getType(), parameterBuffer.toXdrable()); } private final byte writeBuffer[] = new byte[8]; /** * Write a long value to the underlying stream in XDR format. * * @param v The long value to be written * @throws IOException if an error occurs while writing to the * underlying output stream */ public void writeLong(long v) throws IOException { final byte[] buffer = writeBuffer; buffer[0] = (byte) (v >>> 56); buffer[1] = (byte) (v >>> 48); buffer[2] = (byte) (v >>> 40); buffer[3] = (byte) (v >>> 32); buffer[4] = (byte) (v >>> 24); buffer[5] = (byte) (v >>> 16); buffer[6] = (byte) (v >>> 8); buffer[7] = (byte) v; out.write(buffer, 0, 8); } /** * Write an int value to the underlying stream in XDR format. * * @param v The int value to be written * @throws IOException if an error occurs while writing to the * underlying output stream */ public void writeInt(int v) throws IOException { out.write((v >>> 24) & 0xFF); out.write((v >>> 16) & 0xFF); out.write((v >>> 8) & 0xFF); out.write(v & 0xFF); } /** * Write a byte buffer to the underlying output stream * in XDR format. * * @param b The byte buffer to be written * @param offset The start offset in the buffer * @param len The number of bytes to write * @param pad The number of (blank) padding bytes to write * @throws IOException if an error occurs while writing to the * underlying output stream */ public void write(byte[] b, int offset, int len, int pad) throws IOException { out.write(b, offset, len); // TODO We shouldn't always pad with spaces writeSpacePadding(pad); } /** * Write a single byte to the underlying output stream in * XDR format. * * @param b The value to be written, will be truncated to a byte * @throws IOException if an error occurs while writing to the * underlying output stream */ @Override public void write(int b) throws IOException { out.write(b); } /** * Writes len bytes from the specified byte array * starting at offset off to this output stream as defined by {@link java.io.OutputStream#write(byte[], int, int)}. *

* Important: do not confuse this method with {@link #write(byte[], int, int, int)} which originally had * the signature of this method. *

* * @param b The data * @param off The start offset in the data * @param len The number of bytes to write * @throws IOException if an error occurs while writing to the * underlying output stream */ @Override public void write(byte b[], int off, int len) throws IOException { out.write(b, off, len); } /** * Flush all buffered data to the underlying output stream. * * @throws IOException if an error occurs while writing to the * underlying output stream */ @Override public void flush() throws IOException { out.flush(); } /** * Close this stream and the underlying output stream. * * @throws IOException if an error occurs while closing the * underlying stream */ @Override public void close() throws IOException { out.close(); } public void setCipher(Cipher cipher) throws IOException { if (directStream instanceof CipherOutputStream) { throw new IOException("Output stream already encrypted"); } directStream = new CipherOutputStream(rawStream, cipher); if (buffered) { out = new BufferedOutputStream(directStream, BUF_SIZE); } else { out = directStream; } } /** * Writes directly to the {@code OutputStream} of the underlying socket. * * @param data * Data to write * @throws IOException * For errors writing to the socket. */ public final void writeDirect(byte[] data) throws IOException { directStream.write(data); directStream.flush(); } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy