/*
 * JBoss, Home of Professional Open Source
 *
 * Copyright 2010 Red Hat, Inc. and/or its affiliates.
 *
 * 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 org.xnio.streams;

import static org.xnio._private.Messages.msg;

import java.io.IOException;
import java.io.OutputStream;
import java.io.UnsupportedEncodingException;
import java.io.Writer;
import java.nio.ByteBuffer;
import java.nio.CharBuffer;
import java.nio.charset.Charset;
import java.nio.charset.CharsetDecoder;
import java.nio.charset.CoderResult;
import java.nio.charset.CodingErrorAction;

An output stream which decodes bytes into a character writer.
/** * An output stream which decodes bytes into a character writer. */
public final class WriterOutputStream extends OutputStream { private final Writer writer; private final CharsetDecoder decoder; private final ByteBuffer byteBuffer; private final char[] chars; private volatile boolean closed;
Construct a new instance.
Params:
  • writer – the writer to decode into
/** * Construct a new instance. * * @param writer the writer to decode into */
public WriterOutputStream(final Writer writer) { this(writer, Charset.defaultCharset()); }
Construct a new instance.
Params:
  • writer – the writer to decode into
  • decoder – the charset decoder to use
/** * Construct a new instance. * * @param writer the writer to decode into * @param decoder the charset decoder to use */
public WriterOutputStream(final Writer writer, final CharsetDecoder decoder) { this(writer, decoder, 1024); }
Construct a new instance.
Params:
  • writer – the writer to decode into
  • decoder – the charset decoder to use
  • bufferSize – the buffer size to use
/** * Construct a new instance. * * @param writer the writer to decode into * @param decoder the charset decoder to use * @param bufferSize the buffer size to use */
public WriterOutputStream(final Writer writer, final CharsetDecoder decoder, int bufferSize) { if (writer == null) { throw msg.nullParameter("writer"); } if (decoder == null) { throw msg.nullParameter("decoder"); } if (bufferSize < 1) { throw msg.parameterOutOfRange("bufferSize"); } this.writer = writer; this.decoder = decoder; byteBuffer = ByteBuffer.allocate(bufferSize); chars = new char[(int) ((float)bufferSize * decoder.maxCharsPerByte() + 0.5f)]; }
Construct a new instance.
Params:
  • writer – the writer to decode into
  • charset – the character set to use
/** * Construct a new instance. * * @param writer the writer to decode into * @param charset the character set to use */
public WriterOutputStream(final Writer writer, final Charset charset) { this(writer, getDecoder(charset)); }
Construct a new instance.
Params:
  • writer – the writer to decode into
  • charsetName – the character set name to use
Throws:
/** * Construct a new instance. * * @param writer the writer to decode into * @param charsetName the character set name to use * @throws UnsupportedEncodingException if the character set name is unknown */
public WriterOutputStream(final Writer writer, final String charsetName) throws UnsupportedEncodingException { this(writer, Streams.getCharset(charsetName)); } private static CharsetDecoder getDecoder(final Charset charset) { final CharsetDecoder decoder = charset.newDecoder(); decoder.onMalformedInput(CodingErrorAction.REPLACE); decoder.onUnmappableCharacter(CodingErrorAction.REPLACE); decoder.replaceWith("?"); return decoder; }
{@inheritDoc}
/** {@inheritDoc} */
public void write(final int b) throws IOException { if (closed) throw msg.streamClosed(); final ByteBuffer byteBuffer = this.byteBuffer; if (! byteBuffer.hasRemaining()) { doFlush(false); } byteBuffer.put((byte) b); }
{@inheritDoc}
/** {@inheritDoc} */
public void write(final byte[] b, int off, int len) throws IOException { if (closed) throw msg.streamClosed(); final ByteBuffer byteBuffer = this.byteBuffer; // todo Correct first, fast later while (len > 0) { final int r = byteBuffer.remaining(); if (r == 0) { doFlush(false); continue; } final int c = Math.min(len, r); byteBuffer.put(b, off, c); len -= c; off += c; } } private void doFlush(final boolean eof) throws IOException { final CharBuffer charBuffer = CharBuffer.wrap(chars); final ByteBuffer byteBuffer = this.byteBuffer; final CharsetDecoder decoder = this.decoder; byteBuffer.flip(); try { while (byteBuffer.hasRemaining()) { final CoderResult result = decoder.decode(byteBuffer, charBuffer, eof); if (result.isOverflow()) { writer.write(chars, 0, charBuffer.position()); charBuffer.clear(); continue; } if (result.isUnderflow()) { final int p = charBuffer.position(); if (p > 0) { writer.write(chars, 0, p); } return; } if (result.isError()) { if (result.isMalformed()) { throw msg.malformedInput(); } if (result.isUnmappable()) { throw msg.unmappableCharacter(); } throw msg.characterDecodingProblem(); } } } finally { byteBuffer.compact(); } }
{@inheritDoc}
/** {@inheritDoc} */
public void flush() throws IOException { if (closed) throw msg.streamClosed(); doFlush(false); writer.flush(); }
{@inheritDoc}
/** {@inheritDoc} */
public void close() throws IOException { closed = true; doFlush(true); byteBuffer.clear(); writer.close(); }
Get the string representation of this object.
Returns:the string
/** * Get the string representation of this object. * * @return the string */
public String toString() { return "Output stream writing to " + writer; } }