2009-04-16 21:00:42 -07:00
|
|
|
/*
|
2012-11-02 15:50:11 +00:00
|
|
|
* Copyright (c) 2009, 2011, Oracle and/or its affiliates. All rights reserved.
|
2009-04-16 21:00:42 -07:00
|
|
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
|
|
|
*
|
|
|
|
* This code is free software; you can redistribute it and/or modify it
|
|
|
|
* under the terms of the GNU General Public License version 2 only, as
|
2010-05-25 15:58:33 -07:00
|
|
|
* published by the Free Software Foundation. Oracle designates this
|
2009-04-16 21:00:42 -07:00
|
|
|
* particular file as subject to the "Classpath" exception as provided
|
2010-05-25 15:58:33 -07:00
|
|
|
* by Oracle in the LICENSE file that accompanied this code.
|
2009-04-16 21:00:42 -07:00
|
|
|
*
|
|
|
|
* This code 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
|
|
|
|
* version 2 for more details (a copy is included in the LICENSE file that
|
|
|
|
* accompanied this code).
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License version
|
|
|
|
* 2 along with this work; if not, write to the Free Software Foundation,
|
|
|
|
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
|
|
*
|
2010-05-25 15:58:33 -07:00
|
|
|
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
|
|
|
|
* or visit www.oracle.com if you need additional information or have any
|
|
|
|
* questions.
|
2009-04-16 21:00:42 -07:00
|
|
|
*/
|
|
|
|
|
|
|
|
package java.util.zip;
|
|
|
|
|
|
|
|
import java.nio.ByteBuffer;
|
|
|
|
import java.nio.CharBuffer;
|
|
|
|
import java.nio.charset.Charset;
|
2011-05-03 16:32:47 -07:00
|
|
|
import java.nio.charset.StandardCharsets;
|
2009-04-16 21:00:42 -07:00
|
|
|
import java.nio.charset.CharsetDecoder;
|
|
|
|
import java.nio.charset.CharsetEncoder;
|
|
|
|
import java.nio.charset.CoderResult;
|
|
|
|
import java.nio.charset.CodingErrorAction;
|
|
|
|
import java.util.Arrays;
|
2011-05-02 11:42:52 -07:00
|
|
|
import sun.nio.cs.ArrayDecoder;
|
|
|
|
import sun.nio.cs.ArrayEncoder;
|
2009-04-16 21:00:42 -07:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Utility class for zipfile name and comment decoding and encoding
|
|
|
|
*/
|
|
|
|
|
|
|
|
final class ZipCoder {
|
|
|
|
|
|
|
|
String toString(byte[] ba, int length) {
|
|
|
|
CharsetDecoder cd = decoder().reset();
|
|
|
|
int len = (int)(length * cd.maxCharsPerByte());
|
|
|
|
char[] ca = new char[len];
|
|
|
|
if (len == 0)
|
|
|
|
return new String(ca);
|
2011-05-02 11:42:52 -07:00
|
|
|
// UTF-8 only for now. Other ArrayDeocder only handles
|
|
|
|
// CodingErrorAction.REPLACE mode. ZipCoder uses
|
|
|
|
// REPORT mode.
|
|
|
|
if (isUTF8 && cd instanceof ArrayDecoder) {
|
|
|
|
int clen = ((ArrayDecoder)cd).decode(ba, 0, length, ca);
|
|
|
|
if (clen == -1) // malformed
|
|
|
|
throw new IllegalArgumentException("MALFORMED");
|
|
|
|
return new String(ca, 0, clen);
|
|
|
|
}
|
2009-04-16 21:00:42 -07:00
|
|
|
ByteBuffer bb = ByteBuffer.wrap(ba, 0, length);
|
|
|
|
CharBuffer cb = CharBuffer.wrap(ca);
|
|
|
|
CoderResult cr = cd.decode(bb, cb, true);
|
|
|
|
if (!cr.isUnderflow())
|
|
|
|
throw new IllegalArgumentException(cr.toString());
|
|
|
|
cr = cd.flush(cb);
|
|
|
|
if (!cr.isUnderflow())
|
|
|
|
throw new IllegalArgumentException(cr.toString());
|
|
|
|
return new String(ca, 0, cb.position());
|
|
|
|
}
|
|
|
|
|
|
|
|
String toString(byte[] ba) {
|
|
|
|
return toString(ba, ba.length);
|
|
|
|
}
|
|
|
|
|
|
|
|
byte[] getBytes(String s) {
|
|
|
|
CharsetEncoder ce = encoder().reset();
|
|
|
|
char[] ca = s.toCharArray();
|
|
|
|
int len = (int)(ca.length * ce.maxBytesPerChar());
|
|
|
|
byte[] ba = new byte[len];
|
|
|
|
if (len == 0)
|
|
|
|
return ba;
|
2011-05-02 11:42:52 -07:00
|
|
|
// UTF-8 only for now. Other ArrayDeocder only handles
|
|
|
|
// CodingErrorAction.REPLACE mode.
|
|
|
|
if (isUTF8 && ce instanceof ArrayEncoder) {
|
|
|
|
int blen = ((ArrayEncoder)ce).encode(ca, 0, ca.length, ba);
|
|
|
|
if (blen == -1) // malformed
|
|
|
|
throw new IllegalArgumentException("MALFORMED");
|
|
|
|
return Arrays.copyOf(ba, blen);
|
|
|
|
}
|
2009-04-16 21:00:42 -07:00
|
|
|
ByteBuffer bb = ByteBuffer.wrap(ba);
|
|
|
|
CharBuffer cb = CharBuffer.wrap(ca);
|
|
|
|
CoderResult cr = ce.encode(cb, bb, true);
|
|
|
|
if (!cr.isUnderflow())
|
|
|
|
throw new IllegalArgumentException(cr.toString());
|
|
|
|
cr = ce.flush(bb);
|
|
|
|
if (!cr.isUnderflow())
|
|
|
|
throw new IllegalArgumentException(cr.toString());
|
|
|
|
if (bb.position() == ba.length) // defensive copy?
|
|
|
|
return ba;
|
|
|
|
else
|
|
|
|
return Arrays.copyOf(ba, bb.position());
|
|
|
|
}
|
|
|
|
|
|
|
|
// assume invoked only if "this" is not utf8
|
|
|
|
byte[] getBytesUTF8(String s) {
|
2011-05-02 11:42:52 -07:00
|
|
|
if (isUTF8)
|
2009-04-16 21:00:42 -07:00
|
|
|
return getBytes(s);
|
|
|
|
if (utf8 == null)
|
2011-05-03 16:32:47 -07:00
|
|
|
utf8 = new ZipCoder(StandardCharsets.UTF_8);
|
2009-04-16 21:00:42 -07:00
|
|
|
return utf8.getBytes(s);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
String toStringUTF8(byte[] ba, int len) {
|
2011-05-02 11:42:52 -07:00
|
|
|
if (isUTF8)
|
2009-04-16 21:00:42 -07:00
|
|
|
return toString(ba, len);
|
|
|
|
if (utf8 == null)
|
2011-05-03 16:32:47 -07:00
|
|
|
utf8 = new ZipCoder(StandardCharsets.UTF_8);
|
2009-04-16 21:00:42 -07:00
|
|
|
return utf8.toString(ba, len);
|
|
|
|
}
|
|
|
|
|
|
|
|
boolean isUTF8() {
|
2011-05-02 11:42:52 -07:00
|
|
|
return isUTF8;
|
2009-04-16 21:00:42 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
private Charset cs;
|
|
|
|
private CharsetDecoder dec;
|
|
|
|
private CharsetEncoder enc;
|
2011-05-02 11:42:52 -07:00
|
|
|
private boolean isUTF8;
|
2009-04-16 21:00:42 -07:00
|
|
|
private ZipCoder utf8;
|
|
|
|
|
|
|
|
private ZipCoder(Charset cs) {
|
|
|
|
this.cs = cs;
|
2011-05-03 16:32:47 -07:00
|
|
|
this.isUTF8 = cs.name().equals(StandardCharsets.UTF_8.name());
|
2009-04-16 21:00:42 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static ZipCoder get(Charset charset) {
|
|
|
|
return new ZipCoder(charset);
|
|
|
|
}
|
|
|
|
|
|
|
|
private CharsetDecoder decoder() {
|
|
|
|
if (dec == null) {
|
|
|
|
dec = cs.newDecoder()
|
|
|
|
.onMalformedInput(CodingErrorAction.REPORT)
|
|
|
|
.onUnmappableCharacter(CodingErrorAction.REPORT);
|
|
|
|
}
|
|
|
|
return dec;
|
|
|
|
}
|
|
|
|
|
|
|
|
private CharsetEncoder encoder() {
|
|
|
|
if (enc == null) {
|
|
|
|
enc = cs.newEncoder()
|
|
|
|
.onMalformedInput(CodingErrorAction.REPORT)
|
|
|
|
.onUnmappableCharacter(CodingErrorAction.REPORT);
|
|
|
|
}
|
|
|
|
return enc;
|
|
|
|
}
|
|
|
|
}
|