2021-12-11 15:27:08 +00:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2021, Oracle and/or its affiliates. All rights reserved.
|
|
|
|
* 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
|
2022-01-05 16:19:00 +00:00
|
|
|
* @requires vm.bits == 64
|
2021-12-11 15:27:08 +00:00
|
|
|
* @bug 8276766
|
|
|
|
* @summary Test jar --date source date of entries and that jars are
|
|
|
|
* reproducible
|
|
|
|
* @modules jdk.jartool
|
|
|
|
* @run testng/othervm ReproducibleJar
|
|
|
|
*/
|
|
|
|
|
|
|
|
import org.testng.Assert;
|
|
|
|
import org.testng.annotations.AfterMethod;
|
|
|
|
import org.testng.annotations.BeforeMethod;
|
|
|
|
import org.testng.annotations.Test;
|
|
|
|
import org.testng.annotations.DataProvider;
|
|
|
|
|
|
|
|
import java.io.File;
|
|
|
|
import java.io.IOException;
|
|
|
|
import java.io.PrintWriter;
|
|
|
|
import java.nio.file.Files;
|
|
|
|
import java.nio.file.attribute.FileTime;
|
|
|
|
import java.util.Date;
|
|
|
|
import java.util.TimeZone;
|
|
|
|
import java.util.spi.ToolProvider;
|
|
|
|
import java.time.LocalDateTime;
|
|
|
|
import java.time.ZonedDateTime;
|
|
|
|
import java.time.ZoneId;
|
|
|
|
import java.time.ZoneOffset;
|
|
|
|
import java.time.format.DateTimeFormatter;
|
|
|
|
import java.time.Instant;
|
|
|
|
import java.util.concurrent.TimeUnit;
|
|
|
|
|
|
|
|
public class ReproducibleJar {
|
|
|
|
private static final ToolProvider JAR_TOOL = ToolProvider.findFirst("jar")
|
|
|
|
.orElseThrow(() ->
|
|
|
|
new RuntimeException("jar tool not found")
|
|
|
|
);
|
|
|
|
|
|
|
|
// ZipEntry's mod date has 2 seconds precision: give extra time to
|
|
|
|
// allow for e.g. rounding/truncation and networked/samba drives.
|
|
|
|
private static final long PRECISION = 10000L;
|
|
|
|
|
|
|
|
private static final TimeZone TZ = TimeZone.getDefault();
|
|
|
|
private static final boolean DST = TZ.inDaylightTime(new Date());
|
|
|
|
private static final String UNIX_2038_ROLLOVER_TIME = "2038-01-19T03:14:07Z";
|
|
|
|
private static final Instant UNIX_2038_ROLLOVER = Instant.parse(UNIX_2038_ROLLOVER_TIME);
|
|
|
|
private static final File DIR_OUTER = new File("outer");
|
|
|
|
private static final File DIR_INNER = new File(DIR_OUTER, "inner");
|
|
|
|
private static final File FILE_INNER = new File(DIR_INNER, "foo.txt");
|
|
|
|
private static final File JAR_FILE_SOURCE_DATE1 = new File("JarEntryTimeSourceDate1.jar");
|
|
|
|
private static final File JAR_FILE_SOURCE_DATE2 = new File("JarEntryTimeSourceDate2.jar");
|
|
|
|
|
|
|
|
// Valid --date values for jar
|
|
|
|
@DataProvider
|
|
|
|
private Object[][] validSourceDates() {
|
|
|
|
return new Object[][]{
|
|
|
|
{"1980-01-01T00:00:02+00:00"},
|
|
|
|
{"1986-06-24T01:02:03+00:00"},
|
|
|
|
{"2022-03-15T00:00:00+00:00"},
|
|
|
|
{"2022-03-15T00:00:00+06:00"},
|
|
|
|
{"2021-12-25T09:30:00-08:00[America/Los_Angeles]"},
|
|
|
|
{"2021-12-31T23:59:59Z"},
|
|
|
|
{"2024-06-08T14:24Z"},
|
|
|
|
{"2026-09-24T16:26-05:00"},
|
|
|
|
{"2038-11-26T06:06:06+00:00"},
|
|
|
|
{"2098-02-18T00:00:00-08:00"},
|
|
|
|
{"2099-12-31T23:59:59+00:00"}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
// Invalid --date values for jar
|
|
|
|
@DataProvider
|
|
|
|
private Object[][] invalidSourceDates() {
|
|
|
|
return new Object[][]{
|
|
|
|
{"1976-06-24T01:02:03+00:00"},
|
|
|
|
{"1980-01-01T00:00:01+00:00"},
|
|
|
|
{"2100-01-01T00:00:00+00:00"},
|
|
|
|
{"2138-02-18T00:00:00-11:00"},
|
|
|
|
{"2006-04-06T12:38:00"},
|
|
|
|
{"2012-08-24T16"}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
@BeforeMethod
|
|
|
|
public void runBefore() throws IOException {
|
|
|
|
runAfter();
|
|
|
|
createOuterInnerDirs();
|
|
|
|
}
|
|
|
|
|
|
|
|
@AfterMethod
|
|
|
|
public void runAfter() {
|
|
|
|
cleanup(DIR_INNER);
|
|
|
|
cleanup(DIR_OUTER);
|
|
|
|
JAR_FILE_SOURCE_DATE1.delete();
|
|
|
|
JAR_FILE_SOURCE_DATE2.delete();
|
|
|
|
TimeZone.setDefault(TZ);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Test jar tool with various valid --date <timestamps>
|
|
|
|
*/
|
|
|
|
@Test(dataProvider = "validSourceDates")
|
|
|
|
public void testValidSourceDate(String sourceDate) {
|
|
|
|
if (isInTransition()) return;
|
|
|
|
|
|
|
|
// Test --date source date
|
|
|
|
Assert.assertEquals(JAR_TOOL.run(System.out, System.err,
|
|
|
|
"--create",
|
|
|
|
"--file", JAR_FILE_SOURCE_DATE1.getName(),
|
|
|
|
"--date", sourceDate,
|
|
|
|
DIR_OUTER.getName()), 0);
|
|
|
|
Assert.assertTrue(JAR_FILE_SOURCE_DATE1.exists());
|
|
|
|
|
|
|
|
// Extract JAR_FILE_SOURCE_DATE1 and check last modified values
|
|
|
|
Assert.assertEquals(JAR_TOOL.run(System.out, System.err,
|
|
|
|
"--extract",
|
|
|
|
"--file", JAR_FILE_SOURCE_DATE1.getName()), 0);
|
|
|
|
Assert.assertTrue(DIR_OUTER.exists());
|
|
|
|
Assert.assertTrue(DIR_INNER.exists());
|
|
|
|
Assert.assertTrue(FILE_INNER.exists());
|
|
|
|
LocalDateTime expectedLdt = ZonedDateTime.parse(sourceDate,
|
|
|
|
DateTimeFormatter.ISO_DATE_TIME)
|
|
|
|
.withZoneSameInstant(ZoneOffset.UTC)
|
|
|
|
.toLocalDateTime();
|
|
|
|
System.out.format("Checking jar entries local date time for --date %s, is %s%n",
|
|
|
|
sourceDate, expectedLdt);
|
|
|
|
long sourceDateEpochMillis = TimeUnit.MILLISECONDS.convert(
|
|
|
|
expectedLdt.toEpochSecond(ZoneId.systemDefault().getRules()
|
|
|
|
.getOffset(expectedLdt)), TimeUnit.SECONDS);
|
|
|
|
checkFileTime(DIR_OUTER.lastModified(), sourceDateEpochMillis);
|
|
|
|
checkFileTime(DIR_INNER.lastModified(), sourceDateEpochMillis);
|
|
|
|
checkFileTime(FILE_INNER.lastModified(), sourceDateEpochMillis);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Test jar tool with various invalid --date <timestamps>
|
|
|
|
*/
|
|
|
|
@Test(dataProvider = "invalidSourceDates")
|
|
|
|
public void testInvalidSourceDate(String sourceDate) {
|
|
|
|
// Negative Tests --date out of range or wrong format source date
|
|
|
|
Assert.assertNotEquals(JAR_TOOL.run(System.out, System.err,
|
|
|
|
"--create",
|
|
|
|
"--file", JAR_FILE_SOURCE_DATE1.getName(),
|
|
|
|
"--date", sourceDate,
|
|
|
|
DIR_OUTER.getName()), 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Test jar produces deterministic reproducible output
|
|
|
|
*/
|
|
|
|
@Test(dataProvider = "validSourceDates")
|
|
|
|
public void testJarsReproducible(String sourceDate) throws IOException {
|
|
|
|
// Test jars are reproducible across timezones
|
|
|
|
TimeZone tzAsia = TimeZone.getTimeZone("Asia/Shanghai");
|
|
|
|
TimeZone tzLA = TimeZone.getTimeZone("America/Los_Angeles");
|
|
|
|
TimeZone.setDefault(tzAsia);
|
|
|
|
Assert.assertEquals(JAR_TOOL.run(System.out, System.err,
|
|
|
|
"--create",
|
|
|
|
"--file", JAR_FILE_SOURCE_DATE1.getName(),
|
|
|
|
"--date", sourceDate,
|
|
|
|
DIR_OUTER.getName()), 0);
|
|
|
|
Assert.assertTrue(JAR_FILE_SOURCE_DATE1.exists());
|
|
|
|
|
|
|
|
try {
|
|
|
|
// Sleep 5 seconds to ensure jar timestamps might be different if they could be
|
|
|
|
Thread.sleep(5000);
|
|
|
|
} catch (InterruptedException ex) {
|
|
|
|
}
|
|
|
|
|
|
|
|
TimeZone.setDefault(tzLA);
|
|
|
|
Assert.assertEquals(JAR_TOOL.run(System.out, System.err,
|
|
|
|
"--create",
|
|
|
|
"--file", JAR_FILE_SOURCE_DATE2.getName(),
|
|
|
|
"--date", sourceDate,
|
|
|
|
DIR_OUTER.getName()), 0);
|
|
|
|
Assert.assertTrue(JAR_FILE_SOURCE_DATE2.exists());
|
|
|
|
|
|
|
|
// Check jars are identical
|
|
|
|
Assert.assertEquals(Files.readAllBytes(JAR_FILE_SOURCE_DATE1.toPath()),
|
|
|
|
Files.readAllBytes(JAR_FILE_SOURCE_DATE2.toPath()));
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Create the standard directory structure used by the test:
|
|
|
|
* outer/
|
|
|
|
* inner/
|
|
|
|
* foo.txt
|
|
|
|
*/
|
|
|
|
static void createOuterInnerDirs() throws IOException {
|
|
|
|
Assert.assertTrue(DIR_OUTER.mkdir());
|
|
|
|
Assert.assertTrue(DIR_INNER.mkdir());
|
|
|
|
try (PrintWriter pw = new PrintWriter(FILE_INNER)) {
|
|
|
|
pw.println("hello, world");
|
|
|
|
}
|
|
|
|
Assert.assertTrue(DIR_OUTER.exists());
|
|
|
|
Assert.assertTrue(DIR_INNER.exists());
|
|
|
|
Assert.assertTrue(FILE_INNER.exists());
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Check the extracted and original millis since Epoch file times are
|
|
|
|
* within the zip precision time period.
|
|
|
|
*/
|
|
|
|
static void checkFileTime(long now, long original) {
|
|
|
|
if (isTimeSettingChanged()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Math.abs(now - original) > PRECISION) {
|
|
|
|
// If original time is after UNIX 2038 32bit rollover
|
|
|
|
// and the now time is exactly the rollover time, then assume
|
|
|
|
// running on a file system that only supports to 2038 (e.g.XFS) and pass test
|
|
|
|
if (FileTime.fromMillis(original).toInstant().isAfter(UNIX_2038_ROLLOVER) &&
|
|
|
|
FileTime.fromMillis(now).toInstant().equals(UNIX_2038_ROLLOVER)) {
|
|
|
|
System.out.println("Checking file time after Unix 2038 rollover," +
|
|
|
|
" and extracted file time is " + UNIX_2038_ROLLOVER_TIME + ", " +
|
|
|
|
" Assuming restricted file system, pass file time check.");
|
|
|
|
} else {
|
|
|
|
throw new AssertionError("checkFileTime failed," +
|
|
|
|
" extracted to " + FileTime.fromMillis(now) +
|
|
|
|
", expected to be close to " + FileTime.fromMillis(original));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Has the timezone or DST changed during the test?
|
|
|
|
*/
|
|
|
|
private static boolean isTimeSettingChanged() {
|
|
|
|
TimeZone currentTZ = TimeZone.getDefault();
|
|
|
|
boolean currentDST = currentTZ.inDaylightTime(new Date());
|
|
|
|
if (!currentTZ.equals(TZ) || currentDST != DST) {
|
|
|
|
System.out.println("Timezone or DST has changed during " +
|
|
|
|
"ReproducibleJar testcase execution. Test skipped");
|
|
|
|
return true;
|
|
|
|
} else {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Is the Zone currently within the transition change period?
|
|
|
|
*/
|
|
|
|
private static boolean isInTransition() {
|
|
|
|
var inTransition = false;
|
|
|
|
var date = new Date();
|
|
|
|
var defZone = ZoneId.systemDefault();
|
|
|
|
if (defZone.getRules().getTransition(
|
|
|
|
date.toInstant().atZone(defZone).toLocalDateTime()) != null) {
|
|
|
|
System.out.println("ReproducibleJar testcase being run during Zone offset transition. Test skipped.");
|
|
|
|
inTransition = true;
|
|
|
|
}
|
|
|
|
return inTransition;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Remove the directory and its contents
|
|
|
|
*/
|
|
|
|
static void cleanup(File dir) {
|
|
|
|
File[] x = dir.listFiles();
|
|
|
|
if (x != null) {
|
|
|
|
for (File f : x) {
|
|
|
|
f.delete();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
dir.delete();
|
|
|
|
}
|
|
|
|
}
|