| /* |
| * Licensed to the Apache Software Foundation (ASF) under one or more |
| * contributor license agreements. See the NOTICE file distributed with |
| * this work for additional information regarding copyright ownership. |
| * The ASF licenses this file to You 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.apache.jackrabbit.core.data; |
| |
| import java.io.File; |
| import java.io.FileDescriptor; |
| import java.io.FileInputStream; |
| import java.io.FileNotFoundException; |
| import java.io.IOException; |
| |
| import org.apache.commons.io.input.AutoCloseInputStream; |
| |
| /** |
| * This input stream delays opening the file until the first byte is read, and |
| * closes and discards the underlying stream as soon as the end of input has |
| * been reached or when the stream is explicitly closed. |
| */ |
| public class LazyFileInputStream extends AutoCloseInputStream { |
| |
| /** |
| * The file descriptor to use. |
| */ |
| protected final FileDescriptor fd; |
| |
| /** |
| * The file to read from. |
| */ |
| protected final File file; |
| |
| /** |
| * True if the input stream was opened. It is also set to true if the stream |
| * was closed without reading (to avoid opening the file after the stream |
| * was closed). |
| */ |
| protected boolean opened; |
| |
| /** |
| * Creates a new <code>LazyFileInputStream</code> for the given file. If the |
| * file is unreadable, a FileNotFoundException is thrown. |
| * The file is not opened until the first byte is read from the stream. |
| * |
| * @param file the file |
| * @throws java.io.FileNotFoundException |
| */ |
| public LazyFileInputStream(File file) |
| throws FileNotFoundException { |
| super(null); |
| if (!file.canRead()) { |
| throw new FileNotFoundException(file.getPath()); |
| } |
| this.file = file; |
| this.fd = null; |
| } |
| |
| /** |
| * Creates a new <code>LazyFileInputStream</code> for the given file |
| * descriptor. |
| * The file is not opened until the first byte is read from the stream. |
| * |
| * @param fdObj |
| */ |
| public LazyFileInputStream(FileDescriptor fd) { |
| super(null); |
| this.file = null; |
| this.fd = fd; |
| } |
| |
| /** |
| * Creates a new <code>LazyFileInputStream</code> for the given file. If the |
| * file is unreadable, a FileNotFoundException is thrown. |
| * |
| * @param name |
| * @throws java.io.FileNotFoundException |
| */ |
| public LazyFileInputStream(String name) throws FileNotFoundException { |
| this(new File(name)); |
| } |
| |
| /** |
| * Open the stream if required. |
| * |
| * @throws java.io.IOException |
| */ |
| protected void open() throws IOException { |
| if (!opened) { |
| opened = true; |
| if (fd != null) { |
| in = new FileInputStream(fd); |
| } else { |
| in = new FileInputStream(file); |
| } |
| } |
| } |
| |
| public int read() throws IOException { |
| open(); |
| return super.read(); |
| } |
| |
| public int available() throws IOException { |
| open(); |
| return super.available(); |
| } |
| |
| public void close() throws IOException { |
| // make sure the file is not opened afterwards |
| opened = true; |
| |
| // only close the file if it was in fact opened |
| if (in != null) { |
| super.close(); |
| } |
| } |
| |
| public synchronized void reset() throws IOException { |
| open(); |
| super.reset(); |
| } |
| |
| public boolean markSupported() { |
| try { |
| open(); |
| } catch (IOException e) { |
| throw new IllegalStateException(e.toString()); |
| } |
| return super.markSupported(); |
| } |
| |
| public synchronized void mark(int readlimit) { |
| try { |
| open(); |
| } catch (IOException e) { |
| throw new IllegalStateException(e.toString()); |
| } |
| super.mark(readlimit); |
| } |
| |
| public long skip(long n) throws IOException { |
| open(); |
| return super.skip(n); |
| } |
| |
| public int read(byte[] b) throws IOException { |
| open(); |
| return super.read(b, 0, b.length); |
| } |
| |
| public int read(byte[] b, int off, int len) throws IOException { |
| open(); |
| return super.read(b, off, len); |
| } |
| |
| } |