Mercurial > jhg
view src/org/tmatesoft/hg/internal/InflaterDataAccess.java @ 218:047b1dec7a04
Issue 7: Correctly handle manifest and changelog with different number of (or non-matching) revisions
author | Artem Tikhomirov <tikhomirov.artem@gmail.com> |
---|---|
date | Tue, 17 May 2011 03:42:33 +0200 |
parents | b413b16d10a5 |
children | 31f67be94e71 |
line wrap: on
line source
/* * Copyright (c) 2011 TMate Software Ltd * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; version 2 of the License. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * For information on how to redistribute this software under * the terms of a license other than GNU General Public License * contact TMate Software at support@hg4j.com */ package org.tmatesoft.hg.internal; import java.io.EOFException; import java.io.IOException; import java.util.zip.DataFormatException; import java.util.zip.Inflater; import java.util.zip.ZipException; import org.tmatesoft.hg.core.HgBadStateException; /** * DataAccess counterpart for InflaterInputStream. * XXX is it really needed to be subclass of FilterDataAccess? * * @author Artem Tikhomirov * @author TMate Software Ltd. */ public class InflaterDataAccess extends FilterDataAccess { private final Inflater inflater; private final byte[] buffer; private final byte[] singleByte = new byte[1]; private int decompressedPos = 0; private int decompressedLength; public InflaterDataAccess(DataAccess dataAccess, int offset, int compressedLength) { this(dataAccess, offset, compressedLength, -1, new Inflater(), 512); } public InflaterDataAccess(DataAccess dataAccess, int offset, int compressedLength, int actualLength) { this(dataAccess, offset, compressedLength, actualLength, new Inflater(), 512); } public InflaterDataAccess(DataAccess dataAccess, int offset, int compressedLength, int actualLength, Inflater inflater, int bufSize) { super(dataAccess, offset, compressedLength); this.inflater = inflater; this.decompressedLength = actualLength; buffer = new byte[bufSize]; } @Override public InflaterDataAccess reset() throws IOException { super.reset(); inflater.reset(); decompressedPos = 0; return this; } @Override protected int available() { return length() - decompressedPos; } @Override public boolean isEmpty() { // can't use super.available() <= 0 because even when 0 < super.count < 6(?) // decompressedPos might be already == length() return available() <= 0; } @Override public int length() { if (decompressedLength != -1) { return decompressedLength; } decompressedLength = 0; // guard to avoid endless loop in case length() would get invoked from below. int c = 0; try { int oldPos = decompressedPos; byte[] dummy = new byte[buffer.length]; int toRead; while ((toRead = super.available()) > 0) { if (toRead > buffer.length) { toRead = buffer.length; } super.readBytes(buffer, 0, toRead); inflater.setInput(buffer, 0, toRead); try { while (!inflater.needsInput()) { c += inflater.inflate(dummy, 0, dummy.length); } } catch (DataFormatException ex) { throw new HgBadStateException(ex); } } decompressedLength = c + oldPos; reset(); seek(oldPos); return decompressedLength; } catch (IOException ex) { decompressedLength = -1; // better luck next time? throw new HgBadStateException(ex); // XXX perhaps, checked exception } } @Override public void seek(int localOffset) throws IOException { if (localOffset < 0 /* || localOffset >= length() */) { throw new IllegalArgumentException(); } if (localOffset >= decompressedPos) { skip((int) (localOffset - decompressedPos)); } else { reset(); skip((int) localOffset); } } @Override public void skip(int bytes) throws IOException { if (bytes < 0) { bytes += decompressedPos; if (bytes < 0) { throw new IOException("Underflow. Rewind past start of the slice."); } reset(); // fall-through } while (!isEmpty() && bytes > 0) { readByte(); bytes--; } if (bytes != 0) { throw new IOException("Underflow. Rewind past end of the slice"); } } @Override public byte readByte() throws IOException { readBytes(singleByte, 0, 1); return singleByte[0]; } @Override public void readBytes(byte[] b, int off, int len) throws IOException { try { int n; while (len > 0) { while ((n = inflater.inflate(b, off, len)) == 0) { // FIXME few last bytes (checksum?) may be ignored by inflater, thus inflate may return 0 in // perfectly legal conditions (when all data already expanded, but there are still some bytes // in the input stream if (inflater.finished() || inflater.needsDictionary()) { throw new EOFException(); } if (inflater.needsInput()) { // fill: int toRead = super.available(); if (toRead > buffer.length) { toRead = buffer.length; } super.readBytes(buffer, 0, toRead); inflater.setInput(buffer, 0, toRead); } } off += n; len -= n; decompressedPos += n; if (len == 0) { return; // filled } } } catch (DataFormatException e) { String s = e.getMessage(); throw new ZipException(s != null ? s : "Invalid ZLIB data format"); } } }