forked from openrs2/openrs2
This is closer to the client's implementation, which ignores the gzip header/trailer entirely and just decompresses the inner DEFLATE stream. While Jagex always produce valid gzip files, some cache editors in the private server scene do not set the checksum/length correctly in the trailer. I'm planning to add an option to use the lax gzip implementation to support reading these caches. Signed-off-by: Graham <gpe@openrs2.org>bzip2
parent
689fe7c372
commit
db43c73085
@ -0,0 +1,22 @@ |
|||||||
|
package org.openrs2.compress.cli.gzip |
||||||
|
|
||||||
|
import com.github.ajalt.clikt.core.CliktCommand |
||||||
|
import com.github.ajalt.clikt.parameters.options.option |
||||||
|
import com.github.ajalt.clikt.parameters.types.defaultStdin |
||||||
|
import com.github.ajalt.clikt.parameters.types.defaultStdout |
||||||
|
import com.github.ajalt.clikt.parameters.types.inputStream |
||||||
|
import com.github.ajalt.clikt.parameters.types.outputStream |
||||||
|
import org.openrs2.compress.gzip.Gzip |
||||||
|
|
||||||
|
public class GunzipLaxCommand : CliktCommand(name = "gunzip-lax") { |
||||||
|
private val input by option().inputStream().defaultStdin() |
||||||
|
private val output by option().outputStream(truncateExisting = true).defaultStdout() |
||||||
|
|
||||||
|
override fun run() { |
||||||
|
Gzip.createLaxInputStream(input).use { input -> |
||||||
|
output.use { output -> |
||||||
|
input.copyTo(output) |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,116 @@ |
|||||||
|
package org.openrs2.compress.gzip |
||||||
|
|
||||||
|
import java.io.DataInputStream |
||||||
|
import java.io.IOException |
||||||
|
import java.io.InputStream |
||||||
|
import java.util.zip.Inflater |
||||||
|
|
||||||
|
public class GzipLaxInputStream( |
||||||
|
private val input: InputStream |
||||||
|
) : InputStream() { |
||||||
|
private val inflater = Inflater(true) |
||||||
|
private val buffer = ByteArray(4096) |
||||||
|
private var checkedTrailer = false |
||||||
|
|
||||||
|
init { |
||||||
|
val dataInput = DataInputStream(input) |
||||||
|
if (dataInput.readUnsignedShort() != HEADER_MAGIC) { |
||||||
|
throw IOException("Invalid GZIP header magic") |
||||||
|
} else if (dataInput.readUnsignedByte() != METHOD_DEFLATE) { |
||||||
|
throw IOException("Unsupported compression method") |
||||||
|
} |
||||||
|
|
||||||
|
val flags = dataInput.readUnsignedByte() |
||||||
|
dataInput.skip(6) |
||||||
|
|
||||||
|
if ((flags and FLAG_EXTRA) != 0) { |
||||||
|
dataInput.skip(dataInput.readUnsignedShort().toLong()) |
||||||
|
} |
||||||
|
|
||||||
|
if ((flags and FLAG_NAME) != 0) { |
||||||
|
while (dataInput.readUnsignedByte() != 0) { |
||||||
|
// empty |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
if ((flags and FLAG_COMMENT) != 0) { |
||||||
|
while (dataInput.readUnsignedByte() != 0) { |
||||||
|
// empty |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
if ((flags and FLAG_HEADER_CRC) != 0) { |
||||||
|
dataInput.skip(2) |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
override fun read(): Int { |
||||||
|
val n = read(buffer, 0, 1) |
||||||
|
return if (n < 0) { |
||||||
|
-1 |
||||||
|
} else { |
||||||
|
buffer[0].toInt() and 0xFF |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
override fun read(b: ByteArray, off: Int, len: Int): Int { |
||||||
|
while (true) { |
||||||
|
val n = inflater.inflate(b, off, len) |
||||||
|
if (n != 0) { |
||||||
|
return n |
||||||
|
} |
||||||
|
|
||||||
|
when { |
||||||
|
inflater.finished() -> { |
||||||
|
checkTrailer() |
||||||
|
return -1 |
||||||
|
} |
||||||
|
inflater.needsInput() -> fill() |
||||||
|
inflater.needsDictionary() -> throw IOException("Dictionaries not supported") |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
override fun close() { |
||||||
|
inflater.end() |
||||||
|
input.close() |
||||||
|
} |
||||||
|
|
||||||
|
private fun fill() { |
||||||
|
val n = input.read(buffer, 0, buffer.size) |
||||||
|
if (n < 0) { |
||||||
|
throw IOException("Compressed data truncated") |
||||||
|
} |
||||||
|
inflater.setInput(buffer, 0, n) |
||||||
|
} |
||||||
|
|
||||||
|
private fun checkTrailer() { |
||||||
|
if (checkedTrailer) { |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
checkedTrailer = true |
||||||
|
|
||||||
|
var len = TRAILER_LEN |
||||||
|
len -= inflater.remaining |
||||||
|
|
||||||
|
if (len < 0) { |
||||||
|
throw IOException("Compressed data overflow") |
||||||
|
} else if (input.skip(len) != len) { |
||||||
|
throw IOException("GZIP trailer missing") |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
private companion object { |
||||||
|
private const val HEADER_MAGIC = 0x1F8B |
||||||
|
|
||||||
|
private const val METHOD_DEFLATE = 8 |
||||||
|
|
||||||
|
private const val FLAG_HEADER_CRC = 0x2 |
||||||
|
private const val FLAG_EXTRA = 0x4 |
||||||
|
private const val FLAG_NAME = 0x8 |
||||||
|
private const val FLAG_COMMENT = 0x10 |
||||||
|
|
||||||
|
private const val TRAILER_LEN = 8L |
||||||
|
} |
||||||
|
} |
Loading…
Reference in new issue