1 /* 2 * Copyright (C) 2007 The Guava Authors 3 * Copyright (C) 2014, Sasa Zivkov <sasa.zivkov@sap.com>, SAP AG 4 * and other copyright owners as documented in the project's IP log. 5 * 6 * This program and the accompanying materials are made available 7 * under the terms of the Eclipse Distribution License v1.0 which 8 * accompanies this distribution, is reproduced below, and is 9 * available at http://www.eclipse.org/org/documents/edl-v10.php 10 * 11 * All rights reserved. 12 * 13 * Redistribution and use in source and binary forms, with or 14 * without modification, are permitted provided that the following 15 * conditions are met: 16 * 17 * - Redistributions of source code must retain the above copyright 18 * notice, this list of conditions and the following disclaimer. 19 * 20 * - Redistributions in binary form must reproduce the above 21 * copyright notice, this list of conditions and the following 22 * disclaimer in the documentation and/or other materials provided 23 * with the distribution. 24 * 25 * - Neither the name of the Eclipse Foundation, Inc. nor the 26 * names of its contributors may be used to endorse or promote 27 * products derived from this software without specific prior 28 * written permission. 29 * 30 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND 31 * CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, 32 * INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES 33 * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 34 * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR 35 * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, 36 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT 37 * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; 38 * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER 39 * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, 40 * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) 41 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF 42 * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 43 */ 44 45 package org.eclipse.jgit.util.io; 46 47 import java.io.FilterInputStream; 48 import java.io.IOException; 49 import java.io.InputStream; 50 51 import org.eclipse.jgit.internal.JGitText; 52 53 /** 54 * Wraps a {@link java.io.InputStream}, limiting the number of bytes which can 55 * be read. 56 * 57 * This class was copied and modifed from the Google Guava 16.0. Differently 58 * from the original Guava code, when a caller tries to read from this stream 59 * past the given limit and the wrapped stream hasn't yet reached its EOF this 60 * class will call the limitExceeded method instead of returning EOF. 61 * 62 * @since 3.3 63 */ 64 public abstract class LimitedInputStream extends FilterInputStream { 65 66 private long left; 67 /** Max number of bytes to be read from the wrapped stream */ 68 protected final long limit; 69 private long mark = -1; 70 71 /** 72 * Create a new LimitedInputStream 73 * 74 * @param in an InputStream 75 * @param limit max number of bytes to read from the InputStream 76 */ 77 protected LimitedInputStream(InputStream in, long limit) { 78 super(in); 79 left = limit; 80 this.limit = limit; 81 } 82 83 /** {@inheritDoc} */ 84 @Override 85 public int available() throws IOException { 86 return (int) Math.min(in.available(), left); 87 } 88 89 // it's okay to mark even if mark isn't supported, as reset won't work 90 /** {@inheritDoc} */ 91 @Override 92 public synchronized void mark(int readLimit) { 93 in.mark(readLimit); 94 mark = left; 95 } 96 97 /** {@inheritDoc} */ 98 @Override 99 public int read() throws IOException { 100 if (left == 0) { 101 if (in.available() == 0) 102 return -1; 103 else 104 limitExceeded(); 105 } 106 107 int result = in.read(); 108 if (result != -1) 109 --left; 110 return result; 111 } 112 113 /** {@inheritDoc} */ 114 @Override 115 public int read(byte[] b, int off, int len) throws IOException { 116 if (left == 0) { 117 if (in.available() == 0) 118 return -1; 119 else 120 limitExceeded(); 121 } 122 123 len = (int) Math.min(len, left); 124 int result = in.read(b, off, len); 125 if (result != -1) 126 left -= result; 127 return result; 128 } 129 130 /** {@inheritDoc} */ 131 @Override 132 public synchronized void reset() throws IOException { 133 if (!in.markSupported()) 134 throw new IOException(JGitText.get().unsupportedMark); 135 136 if (mark == -1) 137 throw new IOException(JGitText.get().unsetMark); 138 139 in.reset(); 140 left = mark; 141 } 142 143 /** {@inheritDoc} */ 144 @Override 145 public long skip(long n) throws IOException { 146 n = Math.min(n, left); 147 long skipped = in.skip(n); 148 left -= skipped; 149 return skipped; 150 } 151 152 /** 153 * Called when trying to read past the given {@link #limit} and the wrapped 154 * InputStream {@link #in} hasn't yet reached its EOF 155 * 156 * @throws java.io.IOException 157 * subclasses can throw an {@link java.io.IOException} when the 158 * limit is exceeded. The throws java.io.IOException will be 159 * forwarded back to the caller of the read method which read 160 * the stream past the limit. 161 */ 162 protected abstract void limitExceeded() throws IOException; 163 }