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

com.android.okhttp.internal.FaultRecoveringOutputStream Maven / Gradle / Ivy

There is a newer version: 1.2.9
Show newest version
/*
 * Copyright (C) 2013 Square, Inc.
 *
 * 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.android.okhttp.internal;

import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.OutputStream;

import static com.android.okhttp.internal.Util.checkOffsetAndCount;

/**
 * An output stream wrapper that recovers from failures in the underlying stream
 * by replacing it with another stream. This class buffers a fixed amount of
 * data under the assumption that failures occur early in a stream's life.
 * If a failure occurs after the buffer has been exhausted, no recovery is
 * attempted.
 *
 * 

Subclasses must override {@link #replacementStream} which will request a * replacement stream each time an {@link IOException} is encountered on the * current stream. */ public abstract class FaultRecoveringOutputStream extends AbstractOutputStream { private final int maxReplayBufferLength; /** Bytes to transmit on the replacement stream, or null if no recovery is possible. */ private ByteArrayOutputStream replayBuffer; private OutputStream out; /** * @param maxReplayBufferLength the maximum number of successfully written * bytes to buffer so they can be replayed in the event of an error. * Failure recoveries are not possible once this limit has been exceeded. */ public FaultRecoveringOutputStream(int maxReplayBufferLength, OutputStream out) { if (maxReplayBufferLength < 0) throw new IllegalArgumentException(); this.maxReplayBufferLength = maxReplayBufferLength; this.replayBuffer = new ByteArrayOutputStream(maxReplayBufferLength); this.out = out; } @Override public final void write(byte[] buffer, int offset, int count) throws IOException { if (closed) throw new IOException("stream closed"); checkOffsetAndCount(buffer.length, offset, count); while (true) { try { out.write(buffer, offset, count); if (replayBuffer != null) { if (count + replayBuffer.size() > maxReplayBufferLength) { // Failure recovery is no longer possible once we overflow the replay buffer. replayBuffer = null; } else { // Remember the written bytes to the replay buffer. replayBuffer.write(buffer, offset, count); } } return; } catch (IOException e) { if (!recover(e)) throw e; } } } @Override public final void flush() throws IOException { if (closed) { return; // don't throw; this stream might have been closed on the caller's behalf } while (true) { try { out.flush(); return; } catch (IOException e) { if (!recover(e)) throw e; } } } @Override public final void close() throws IOException { if (closed) { return; } while (true) { try { out.close(); closed = true; return; } catch (IOException e) { if (!recover(e)) throw e; } } } /** * Attempt to replace {@code out} with another equivalent stream. Returns true * if a suitable replacement stream was found. */ private boolean recover(IOException e) { if (replayBuffer == null) { return false; // Can't recover because we've dropped data that we would need to replay. } while (true) { OutputStream replacementStream = null; try { replacementStream = replacementStream(e); if (replacementStream == null) { return false; } replaceStream(replacementStream); return true; } catch (IOException replacementStreamFailure) { // The replacement was also broken. Loop to ask for another replacement. Util.closeQuietly(replacementStream); e = replacementStreamFailure; } } } /** * Returns true if errors in the underlying stream can currently be recovered. */ public boolean isRecoverable() { return replayBuffer != null; } /** * Replaces the current output stream with {@code replacementStream}, writing * any replay bytes to it if they exist. The current output stream is closed. */ public final void replaceStream(OutputStream replacementStream) throws IOException { if (!isRecoverable()) { throw new IllegalStateException(); } if (this.out == replacementStream) { return; // Don't replace a stream with itself. } replayBuffer.writeTo(replacementStream); Util.closeQuietly(out); out = replacementStream; } /** * Returns a replacement output stream to recover from {@code e} thrown by the * previous stream. Returns a new OutputStream if recovery was successful, in * which case all previously-written data will be replayed. Returns null if * the failure cannot be recovered. */ protected abstract OutputStream replacementStream(IOException e) throws IOException; }





© 2015 - 2024 Weber Informatics LLC | Privacy Policy