Back to project page wototoplayer.
The source code is released under:
Copyright (c) 2015, Chris Greenhalgh All rights reserved. Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met...
If you think the Android project wototoplayer listed in this page is inappropriate, such as containing malicious code/tools or violating the copyright, please email info at java2s dot com, thanks.
/* * Copyright (C) 2013 Square, Inc.//from w ww . j av a2 s.c om * * 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.squareup.okhttp.internal; import java.io.ByteArrayOutputStream; import java.io.IOException; import java.io.OutputStream; import static com.squareup.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. * * <p>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; }