2013-05-29 19:50:47 -07:00
|
|
|
/*
|
2015-02-28 13:17:13 +01:00
|
|
|
* Copyright (c) 2013, 2015, Oracle and/or its affiliates. All rights reserved.
|
2013-05-29 19:50:47 -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
|
|
|
|
* published by the Free Software Foundation.
|
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @test
|
2015-07-22 21:11:38 -07:00
|
|
|
* @bug 4759491 6303183 7012868 8015666 8023713 8068790 8076641 8075526 8130914
|
2013-05-29 19:50:47 -07:00
|
|
|
* @summary Test ZOS and ZIS timestamp in extra field correctly
|
|
|
|
*/
|
|
|
|
|
|
|
|
import java.io.*;
|
2013-08-08 12:03:04 -07:00
|
|
|
import java.nio.file.Files;
|
|
|
|
import java.nio.file.Path;
|
|
|
|
import java.nio.file.Paths;
|
|
|
|
import java.nio.file.attribute.FileTime;
|
2013-08-28 09:46:55 -07:00
|
|
|
import java.util.Arrays;
|
2013-05-29 19:50:47 -07:00
|
|
|
import java.util.TimeZone;
|
|
|
|
import java.util.concurrent.TimeUnit;
|
|
|
|
import java.util.zip.ZipEntry;
|
2013-08-08 12:03:04 -07:00
|
|
|
import java.util.zip.ZipFile;
|
2013-05-29 19:50:47 -07:00
|
|
|
import java.util.zip.ZipInputStream;
|
|
|
|
import java.util.zip.ZipOutputStream;
|
|
|
|
|
|
|
|
public class TestExtraTime {
|
|
|
|
|
|
|
|
public static void main(String[] args) throws Throwable{
|
|
|
|
|
|
|
|
File src = new File(System.getProperty("test.src", "."), "TestExtraTime.java");
|
|
|
|
if (src.exists()) {
|
2013-08-08 12:03:04 -07:00
|
|
|
long time = src.lastModified();
|
|
|
|
FileTime mtime = FileTime.from(time, TimeUnit.MILLISECONDS);
|
|
|
|
FileTime atime = FileTime.from(time + 300000, TimeUnit.MILLISECONDS);
|
|
|
|
FileTime ctime = FileTime.from(time - 300000, TimeUnit.MILLISECONDS);
|
|
|
|
TimeZone tz = TimeZone.getTimeZone("Asia/Shanghai");
|
|
|
|
|
2013-08-28 09:46:55 -07:00
|
|
|
for (byte[] extra : new byte[][] { null, new byte[] {1, 2, 3}}) {
|
|
|
|
test(mtime, null, null, null, extra);
|
2015-07-22 21:43:33 +00:00
|
|
|
|
2013-08-28 09:46:55 -07:00
|
|
|
// ms-dos 1980 epoch problem
|
|
|
|
test(FileTime.from(10, TimeUnit.MILLISECONDS), null, null, null, extra);
|
2015-07-22 21:43:33 +00:00
|
|
|
// negative epoch time
|
|
|
|
test(FileTime.from(-100, TimeUnit.DAYS), null, null, null, extra);
|
|
|
|
|
2013-08-28 09:46:55 -07:00
|
|
|
// non-default tz
|
|
|
|
test(mtime, null, null, tz, extra);
|
2013-08-08 12:03:04 -07:00
|
|
|
|
2013-08-28 09:46:55 -07:00
|
|
|
test(mtime, atime, null, null, extra);
|
|
|
|
test(mtime, null, ctime, null, extra);
|
|
|
|
test(mtime, atime, ctime, null, extra);
|
2013-08-08 12:03:04 -07:00
|
|
|
|
2013-08-28 09:46:55 -07:00
|
|
|
test(mtime, atime, null, tz, extra);
|
|
|
|
test(mtime, null, ctime, tz, extra);
|
|
|
|
test(mtime, atime, ctime, tz, extra);
|
|
|
|
}
|
2013-05-29 19:50:47 -07:00
|
|
|
}
|
2015-02-21 13:46:24 +01:00
|
|
|
|
|
|
|
testNullHandling();
|
2015-04-06 13:41:10 -07:00
|
|
|
testTagOnlyHandling();
|
2015-02-28 13:17:13 +01:00
|
|
|
testTimeConversions();
|
2013-05-29 19:50:47 -07:00
|
|
|
}
|
|
|
|
|
2013-08-08 12:03:04 -07:00
|
|
|
static void test(FileTime mtime, FileTime atime, FileTime ctime,
|
2013-08-28 09:46:55 -07:00
|
|
|
TimeZone tz, byte[] extra) throws Throwable {
|
2013-08-08 12:03:04 -07:00
|
|
|
System.out.printf("--------------------%nTesting: [%s]/[%s]/[%s]%n",
|
|
|
|
mtime, atime, ctime);
|
2013-05-29 19:50:47 -07:00
|
|
|
TimeZone tz0 = TimeZone.getDefault();
|
|
|
|
if (tz != null) {
|
|
|
|
TimeZone.setDefault(tz);
|
|
|
|
}
|
|
|
|
ByteArrayOutputStream baos = new ByteArrayOutputStream();
|
|
|
|
ZipOutputStream zos = new ZipOutputStream(baos);
|
2013-08-28 09:46:55 -07:00
|
|
|
ZipEntry ze = new ZipEntry("TestExtraTime.java");
|
|
|
|
ze.setExtra(extra);
|
2013-08-08 12:03:04 -07:00
|
|
|
ze.setLastModifiedTime(mtime);
|
|
|
|
if (atime != null)
|
|
|
|
ze.setLastAccessTime(atime);
|
|
|
|
if (ctime != null)
|
|
|
|
ze.setCreationTime(ctime);
|
2013-05-29 19:50:47 -07:00
|
|
|
zos.putNextEntry(ze);
|
|
|
|
zos.write(new byte[] { 1,2 ,3, 4});
|
2013-08-28 09:46:55 -07:00
|
|
|
|
|
|
|
// append an extra entry to help check if the length and data
|
|
|
|
// of the extra field are being correctly written (in previous
|
|
|
|
// entry).
|
|
|
|
if (extra != null) {
|
|
|
|
ze = new ZipEntry("TestExtraEntry");
|
|
|
|
zos.putNextEntry(ze);
|
|
|
|
}
|
2013-05-29 19:50:47 -07:00
|
|
|
zos.close();
|
|
|
|
if (tz != null) {
|
|
|
|
TimeZone.setDefault(tz0);
|
|
|
|
}
|
2013-08-08 12:03:04 -07:00
|
|
|
// ZipInputStream
|
2013-05-29 19:50:47 -07:00
|
|
|
ZipInputStream zis = new ZipInputStream(
|
|
|
|
new ByteArrayInputStream(baos.toByteArray()));
|
|
|
|
ze = zis.getNextEntry();
|
|
|
|
zis.close();
|
2013-08-28 09:46:55 -07:00
|
|
|
check(mtime, atime, ctime, ze, extra);
|
2013-05-29 19:50:47 -07:00
|
|
|
|
2013-08-08 12:03:04 -07:00
|
|
|
// ZipFile
|
|
|
|
Path zpath = Paths.get(System.getProperty("test.dir", "."),
|
2013-08-28 09:46:55 -07:00
|
|
|
"TestExtraTime.zip");
|
2013-08-08 12:03:04 -07:00
|
|
|
Files.copy(new ByteArrayInputStream(baos.toByteArray()), zpath);
|
|
|
|
ZipFile zf = new ZipFile(zpath.toFile());
|
2013-08-28 09:46:55 -07:00
|
|
|
ze = zf.getEntry("TestExtraTime.java");
|
2013-08-08 12:03:04 -07:00
|
|
|
// ZipFile read entry from cen, which does not have a/ctime,
|
|
|
|
// for now.
|
2013-08-28 09:46:55 -07:00
|
|
|
check(mtime, null, null, ze, extra);
|
2013-08-08 12:03:04 -07:00
|
|
|
zf.close();
|
|
|
|
Files.delete(zpath);
|
|
|
|
}
|
2013-05-29 19:50:47 -07:00
|
|
|
|
2013-08-08 12:03:04 -07:00
|
|
|
static void check(FileTime mtime, FileTime atime, FileTime ctime,
|
2013-08-28 09:46:55 -07:00
|
|
|
ZipEntry ze, byte[] extra) {
|
2013-08-08 12:03:04 -07:00
|
|
|
/*
|
|
|
|
System.out.printf(" mtime [%tc]: [%tc]/[%tc]%n",
|
|
|
|
mtime.to(TimeUnit.MILLISECONDS),
|
|
|
|
ze.getTime(),
|
|
|
|
ze.getLastModifiedTime().to(TimeUnit.MILLISECONDS));
|
|
|
|
*/
|
|
|
|
if (mtime.to(TimeUnit.SECONDS) !=
|
|
|
|
ze.getLastModifiedTime().to(TimeUnit.SECONDS))
|
|
|
|
throw new RuntimeException("Timestamp: storing mtime failed!");
|
|
|
|
if (atime != null &&
|
|
|
|
atime.to(TimeUnit.SECONDS) !=
|
|
|
|
ze.getLastAccessTime().to(TimeUnit.SECONDS))
|
|
|
|
throw new RuntimeException("Timestamp: storing atime failed!");
|
|
|
|
if (ctime != null &&
|
|
|
|
ctime.to(TimeUnit.SECONDS) !=
|
|
|
|
ze.getCreationTime().to(TimeUnit.SECONDS))
|
|
|
|
throw new RuntimeException("Timestamp: storing ctime failed!");
|
2013-08-28 09:46:55 -07:00
|
|
|
if (extra != null) {
|
|
|
|
// if extra data exists, the current implementation put it at
|
|
|
|
// the end of the extra data array (implementation detail)
|
|
|
|
byte[] extra1 = ze.getExtra();
|
|
|
|
if (extra1 == null || extra1.length < extra.length ||
|
|
|
|
!Arrays.equals(Arrays.copyOfRange(extra1,
|
|
|
|
extra1.length - extra.length,
|
|
|
|
extra1.length),
|
|
|
|
extra)) {
|
|
|
|
throw new RuntimeException("Timestamp: storing extra field failed!");
|
|
|
|
}
|
|
|
|
}
|
2013-05-29 19:50:47 -07:00
|
|
|
}
|
2015-02-21 13:46:24 +01:00
|
|
|
|
|
|
|
static void testNullHandling() {
|
|
|
|
ZipEntry ze = new ZipEntry("TestExtraTime.java");
|
|
|
|
try {
|
|
|
|
ze.setLastAccessTime(null);
|
|
|
|
throw new RuntimeException("setLastAccessTime(null) should throw NPE");
|
|
|
|
} catch (NullPointerException ignored) {
|
|
|
|
// pass
|
|
|
|
}
|
|
|
|
try {
|
|
|
|
ze.setCreationTime(null);
|
|
|
|
throw new RuntimeException("setCreationTime(null) should throw NPE");
|
|
|
|
} catch (NullPointerException ignored) {
|
|
|
|
// pass
|
|
|
|
}
|
|
|
|
try {
|
|
|
|
ze.setLastModifiedTime(null);
|
|
|
|
throw new RuntimeException("setLastModifiedTime(null) should throw NPE");
|
|
|
|
} catch (NullPointerException ignored) {
|
|
|
|
// pass
|
|
|
|
}
|
|
|
|
}
|
2015-02-28 13:17:13 +01:00
|
|
|
|
|
|
|
// verify that setting and getting any time is possible as per the intent
|
|
|
|
// of 4759491
|
|
|
|
static void testTimeConversions() {
|
|
|
|
// Sample across the entire range
|
|
|
|
long step = Long.MAX_VALUE / 100L;
|
|
|
|
testTimeConversions(Long.MIN_VALUE, Long.MAX_VALUE - step, step);
|
|
|
|
|
|
|
|
// Samples through the near future
|
|
|
|
long currentTime = System.currentTimeMillis();
|
|
|
|
testTimeConversions(currentTime, currentTime + 1_000_000, 10_000);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void testTimeConversions(long from, long to, long step) {
|
|
|
|
ZipEntry ze = new ZipEntry("TestExtraTime.java");
|
|
|
|
for (long time = from; time <= to; time += step) {
|
|
|
|
ze.setTime(time);
|
|
|
|
FileTime lastModifiedTime = ze.getLastModifiedTime();
|
|
|
|
if (lastModifiedTime.toMillis() != time) {
|
|
|
|
throw new RuntimeException("setTime should make getLastModifiedTime " +
|
|
|
|
"return the specified instant: " + time +
|
|
|
|
" got: " + lastModifiedTime.toMillis());
|
|
|
|
}
|
|
|
|
if (ze.getTime() != time) {
|
|
|
|
throw new RuntimeException("getTime after setTime, expected: " +
|
|
|
|
time + " got: " + ze.getTime());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2015-04-06 13:41:10 -07:00
|
|
|
|
|
|
|
static void check(ZipEntry ze, byte[] extra) {
|
|
|
|
if (extra != null) {
|
|
|
|
byte[] extra1 = ze.getExtra();
|
|
|
|
if (extra1 == null || extra1.length < extra.length ||
|
|
|
|
!Arrays.equals(Arrays.copyOfRange(extra1,
|
|
|
|
extra1.length - extra.length,
|
|
|
|
extra1.length),
|
|
|
|
extra)) {
|
|
|
|
throw new RuntimeException("Timestamp: storing extra field failed!");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void testTagOnlyHandling() throws Throwable {
|
|
|
|
ByteArrayOutputStream baos = new ByteArrayOutputStream();
|
|
|
|
byte[] extra = new byte[] { 0x0a, 0, 4, 0, 0, 0, 0, 0 };
|
|
|
|
try (ZipOutputStream zos = new ZipOutputStream(baos)) {
|
|
|
|
ZipEntry ze = new ZipEntry("TestExtraTime.java");
|
|
|
|
ze.setExtra(extra);
|
|
|
|
zos.putNextEntry(ze);
|
|
|
|
zos.write(new byte[] { 1,2 ,3, 4});
|
|
|
|
}
|
|
|
|
try (ZipInputStream zis = new ZipInputStream(
|
|
|
|
new ByteArrayInputStream(baos.toByteArray()))) {
|
|
|
|
ZipEntry ze = zis.getNextEntry();
|
|
|
|
check(ze, extra);
|
|
|
|
}
|
|
|
|
Path zpath = Paths.get(System.getProperty("test.dir", "."),
|
|
|
|
"TestExtraTime.zip");
|
|
|
|
Files.copy(new ByteArrayInputStream(baos.toByteArray()), zpath);
|
|
|
|
try (ZipFile zf = new ZipFile(zpath.toFile())) {
|
|
|
|
ZipEntry ze = zf.getEntry("TestExtraTime.java");
|
|
|
|
check(ze, extra);
|
|
|
|
} finally {
|
|
|
|
Files.delete(zpath);
|
|
|
|
}
|
|
|
|
}
|
2013-05-29 19:50:47 -07:00
|
|
|
}
|