diff --git a/tycho-its/projects/reproducible-archive-timestamps/pom.xml b/tycho-its/projects/reproducible-archive-timestamps/pom.xml
new file mode 100644
index 0000000000..258f4e8b20
--- /dev/null
+++ b/tycho-its/projects/reproducible-archive-timestamps/pom.xml
@@ -0,0 +1,93 @@
+
+
+ 4.0.0
+
+ reproducible.archive.timestamps
+ reproducible.archive.timestamps.parent
+ 1.0.0
+ pom
+
+
+ reproducible.bundle
+ reproducible.bundle.feature
+ reproducible.iu
+ reproducible.repository
+
+
+
+ 2023-01-01T00:00:00Z
+ http://download.eclipse.org/releases/latest
+
+
+
+
+ repo
+ p2
+ ${target-platform}
+
+
+
+
+
+
+ org.eclipse.tycho
+ tycho-maven-plugin
+ ${tycho-version}
+ true
+
+
+
+ org.eclipse.tycho
+ target-platform-configuration
+ ${tycho-version}
+
+
+
+ linux
+ gtk
+ x86
+
+
+
+
+
+
+ org.eclipse.tycho
+ tycho-packaging-plugin
+ ${tycho-version}
+
+
+
+ org.eclipse.tycho
+ tycho-source-plugin
+ ${tycho-version}
+
+
+ plugin-source
+
+ plugin-source
+ feature-source
+
+
+
+
+
+
+ org.eclipse.tycho
+ tycho-p2-plugin
+ ${tycho-version}
+
+
+ attach-p2-metadata
+ package
+
+ p2-metadata
+
+
+
+
+
+
+
+
diff --git a/tycho-its/projects/reproducible-archive-timestamps/reproducible.bundle.feature/build.properties b/tycho-its/projects/reproducible-archive-timestamps/reproducible.bundle.feature/build.properties
new file mode 100644
index 0000000000..64f93a9f0b
--- /dev/null
+++ b/tycho-its/projects/reproducible-archive-timestamps/reproducible.bundle.feature/build.properties
@@ -0,0 +1 @@
+bin.includes = feature.xml
diff --git a/tycho-its/projects/reproducible-archive-timestamps/reproducible.bundle.feature/feature.xml b/tycho-its/projects/reproducible-archive-timestamps/reproducible.bundle.feature/feature.xml
new file mode 100644
index 0000000000..c6ae9e0a46
--- /dev/null
+++ b/tycho-its/projects/reproducible-archive-timestamps/reproducible.bundle.feature/feature.xml
@@ -0,0 +1,14 @@
+
+
+
+
+
+
diff --git a/tycho-its/projects/reproducible-archive-timestamps/reproducible.bundle.feature/pom.xml b/tycho-its/projects/reproducible-archive-timestamps/reproducible.bundle.feature/pom.xml
new file mode 100644
index 0000000000..c56f2de8d5
--- /dev/null
+++ b/tycho-its/projects/reproducible-archive-timestamps/reproducible.bundle.feature/pom.xml
@@ -0,0 +1,13 @@
+
+
+ 4.0.0
+
+
+ reproducible.archive.timestamps
+ reproducible.archive.timestamps.parent
+ 1.0.0
+
+
+ reproducible.bundle.feature
+ eclipse-feature
+
diff --git a/tycho-its/projects/reproducible-archive-timestamps/reproducible.bundle/META-INF/MANIFEST.MF b/tycho-its/projects/reproducible-archive-timestamps/reproducible.bundle/META-INF/MANIFEST.MF
new file mode 100644
index 0000000000..103b8ed14a
--- /dev/null
+++ b/tycho-its/projects/reproducible-archive-timestamps/reproducible.bundle/META-INF/MANIFEST.MF
@@ -0,0 +1,5 @@
+Manifest-Version: 1.0
+Bundle-ManifestVersion: 2
+Bundle-Name: Reproducible-bundle
+Bundle-SymbolicName: reproducible.bundle
+Bundle-Version: 1.0.0
diff --git a/tycho-its/projects/reproducible-archive-timestamps/reproducible.bundle/build.properties b/tycho-its/projects/reproducible-archive-timestamps/reproducible.bundle/build.properties
new file mode 100644
index 0000000000..34d2e4d2da
--- /dev/null
+++ b/tycho-its/projects/reproducible-archive-timestamps/reproducible.bundle/build.properties
@@ -0,0 +1,4 @@
+source.. = src/
+output.. = bin/
+bin.includes = META-INF/,\
+ .
diff --git a/tycho-its/projects/reproducible-archive-timestamps/reproducible.bundle/custom/META-INF/MANIFEST.MF b/tycho-its/projects/reproducible-archive-timestamps/reproducible.bundle/custom/META-INF/MANIFEST.MF
new file mode 100644
index 0000000000..f3d666f1a8
--- /dev/null
+++ b/tycho-its/projects/reproducible-archive-timestamps/reproducible.bundle/custom/META-INF/MANIFEST.MF
@@ -0,0 +1,5 @@
+Manifest-Version: 1.0
+Bundle-ManifestVersion: 2
+Bundle-Name: Reproducible-bundle
+Bundle-SymbolicName: reproducible.bundle.attached
+Bundle-Version: 1.0.0
diff --git a/tycho-its/projects/reproducible-archive-timestamps/reproducible.bundle/custom/META-INF/meta-test.txt b/tycho-its/projects/reproducible-archive-timestamps/reproducible.bundle/custom/META-INF/meta-test.txt
new file mode 100644
index 0000000000..e69de29bb2
diff --git a/tycho-its/projects/reproducible-archive-timestamps/reproducible.bundle/custom/test.txt b/tycho-its/projects/reproducible-archive-timestamps/reproducible.bundle/custom/test.txt
new file mode 100644
index 0000000000..e69de29bb2
diff --git a/tycho-its/projects/reproducible-archive-timestamps/reproducible.bundle/pom.xml b/tycho-its/projects/reproducible-archive-timestamps/reproducible.bundle/pom.xml
new file mode 100644
index 0000000000..4adfcab7fb
--- /dev/null
+++ b/tycho-its/projects/reproducible-archive-timestamps/reproducible.bundle/pom.xml
@@ -0,0 +1,44 @@
+
+
+ 4.0.0
+
+
+ reproducible.archive.timestamps
+ reproducible.archive.timestamps.parent
+ 1.0.0
+
+
+ reproducible.bundle
+ eclipse-plugin
+
+
+
+
+ org.eclipse.tycho.extras
+ tycho-custom-bundle-plugin
+ ${tycho-version}
+
+
+ attached
+ package
+
+ custom-bundle
+
+
+ ${project.basedir}/custom
+ attached
+
+
+ ${project.build.outputDirectory}
+
+ **/*.class
+
+
+
+
+
+
+
+
+
+
diff --git a/tycho-its/projects/reproducible-archive-timestamps/reproducible.bundle/src/reproducible/bundle/PublicClass.java b/tycho-its/projects/reproducible-archive-timestamps/reproducible.bundle/src/reproducible/bundle/PublicClass.java
new file mode 100644
index 0000000000..72e2ea4820
--- /dev/null
+++ b/tycho-its/projects/reproducible-archive-timestamps/reproducible.bundle/src/reproducible/bundle/PublicClass.java
@@ -0,0 +1,6 @@
+package reproducible.bundle;
+
+public class PublicClass
+{
+
+}
diff --git a/tycho-its/projects/reproducible-archive-timestamps/reproducible.iu/p2iu.xml b/tycho-its/projects/reproducible-archive-timestamps/reproducible.iu/p2iu.xml
new file mode 100644
index 0000000000..dd2b0d1151
--- /dev/null
+++ b/tycho-its/projects/reproducible-archive-timestamps/reproducible.iu/p2iu.xml
@@ -0,0 +1,18 @@
+
+
+
+
+
+
+
+
+
+
+ unzip(source:@artifact, target:${installFolder});
+
+
+ cleanupzip(source:@artifact, target:${installFolder});
+
+
+
+
diff --git a/tycho-its/projects/reproducible-archive-timestamps/reproducible.iu/pom.xml b/tycho-its/projects/reproducible-archive-timestamps/reproducible.iu/pom.xml
new file mode 100644
index 0000000000..bbeb688b88
--- /dev/null
+++ b/tycho-its/projects/reproducible-archive-timestamps/reproducible.iu/pom.xml
@@ -0,0 +1,13 @@
+
+
+ 4.0.0
+
+
+ reproducible.archive.timestamps
+ reproducible.archive.timestamps.parent
+ 1.0.0
+
+
+ reproducible.iu
+ p2-installable-unit
+
diff --git a/tycho-its/projects/reproducible-archive-timestamps/reproducible.iu/src/main/resources/myFile.txt b/tycho-its/projects/reproducible-archive-timestamps/reproducible.iu/src/main/resources/myFile.txt
new file mode 100644
index 0000000000..e69de29bb2
diff --git a/tycho-its/projects/reproducible-archive-timestamps/reproducible.repository/main.p2.inf b/tycho-its/projects/reproducible-archive-timestamps/reproducible.repository/main.p2.inf
new file mode 100644
index 0000000000..ca9789476c
--- /dev/null
+++ b/tycho-its/projects/reproducible-archive-timestamps/reproducible.repository/main.p2.inf
@@ -0,0 +1,3 @@
+requires.0.namespace=org.eclipse.equinox.p2.iu
+requires.0.name=reproducible.iu
+requires.0.range=[$version$,$version$]
diff --git a/tycho-its/projects/reproducible-archive-timestamps/reproducible.repository/main.product b/tycho-its/projects/reproducible-archive-timestamps/reproducible.repository/main.product
new file mode 100644
index 0000000000..ac537f33ba
--- /dev/null
+++ b/tycho-its/projects/reproducible-archive-timestamps/reproducible.repository/main.product
@@ -0,0 +1,17 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/tycho-its/projects/reproducible-archive-timestamps/reproducible.repository/pom.xml b/tycho-its/projects/reproducible-archive-timestamps/reproducible.repository/pom.xml
new file mode 100644
index 0000000000..cfcadb0953
--- /dev/null
+++ b/tycho-its/projects/reproducible-archive-timestamps/reproducible.repository/pom.xml
@@ -0,0 +1,60 @@
+
+
+ 4.0.0
+
+
+ reproducible.archive.timestamps
+ reproducible.archive.timestamps.parent
+ 1.0.0
+
+
+ reproducible.repository
+ eclipse-repository
+
+
+
+
+ org.eclipse.tycho
+ tycho-p2-repository-plugin
+ ${tycho-version}
+
+
+ assemble-maven-repository
+ verify
+
+ assemble-maven-repository
+
+
+
+
+
+ org.eclipse.tycho
+ tycho-p2-director-plugin
+ ${tycho-version}
+
+
+ materialize-products
+
+ materialize-products
+
+
+
+ archive-products
+
+ archive-products
+
+
+
+
+
+ zip
+ zip
+ zip
+
+
+
+
+
+
+
diff --git a/tycho-its/src/test/java/org/eclipse/tycho/test/reproducible/ReproducibleArchiveTimestampsTest.java b/tycho-its/src/test/java/org/eclipse/tycho/test/reproducible/ReproducibleArchiveTimestampsTest.java
new file mode 100644
index 0000000000..e870382c64
--- /dev/null
+++ b/tycho-its/src/test/java/org/eclipse/tycho/test/reproducible/ReproducibleArchiveTimestampsTest.java
@@ -0,0 +1,65 @@
+/*******************************************************************************
+ * This program and the accompanying materials
+ * are made available under the terms of the Eclipse Public License 2.0
+ * which accompanies this distribution, and is available at
+ * https://www.eclipse.org/legal/epl-2.0/
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ *******************************************************************************/
+package org.eclipse.tycho.test.reproducible;
+
+import java.io.IOException;
+import java.time.Instant;
+import java.time.LocalDateTime;
+import java.time.OffsetDateTime;
+import java.util.List;
+import java.util.zip.ZipEntry;
+import java.util.zip.ZipFile;
+
+import org.apache.maven.it.Verifier;
+import org.eclipse.tycho.test.AbstractTychoIntegrationTest;
+import org.junit.Assert;
+import org.junit.Test;
+
+public class ReproducibleArchiveTimestampsTest extends AbstractTychoIntegrationTest {
+ // The ZipEntry.getLastModifiedTime() method uses the default timezone to
+ // convert date and time fields to Instant, so we also use the default timezone
+ // for the expected timestamp here.
+ private static final String EXPECTED_TIMESTAMP_STRING = "2023-01-01T00:00:00";
+ private static final Instant EXPECTED_TIMESTAMP_INSTANT = LocalDateTime.parse(EXPECTED_TIMESTAMP_STRING)
+ .toInstant(OffsetDateTime.now().getOffset());
+
+ /**
+ * Check that the timestamp of the files inside the produced archives is equal
+ * to the one specified in the "project.build.outputTimestamp" property of the
+ * pom file.
+ */
+ @Test
+ public void test() throws Exception {
+ Verifier verifier = getVerifier("reproducible-archive-timestamps");
+ verifier.executeGoals(List.of("clean", "verify"));
+ verifier.verifyErrorFreeLog();
+
+ // Check timestamps of files in archives
+ checkTimestamps(verifier.getBasedir() + "/reproducible.bundle/target/reproducible.bundle-1.0.0.jar");
+ checkTimestamps(verifier.getBasedir() + "/reproducible.bundle/target/reproducible.bundle-1.0.0-attached.jar");
+ checkTimestamps(verifier.getBasedir() + "/reproducible.bundle/target/reproducible.bundle-1.0.0-sources.jar");
+ checkTimestamps(
+ verifier.getBasedir() + "/reproducible.bundle.feature/target/reproducible.bundle.feature-1.0.0.jar");
+ checkTimestamps(verifier.getBasedir()
+ + "/reproducible.bundle.feature/target/reproducible.bundle.feature-1.0.0-sources-feature.jar");
+ checkTimestamps(verifier.getBasedir() + "/reproducible.iu/target/reproducible.iu-1.0.0.zip");
+ checkTimestamps(verifier.getBasedir() + "/reproducible.repository/target/reproducible.repository-1.0.0.zip");
+ checkTimestamps(verifier.getBasedir() + "/reproducible.repository/target/p2-site.zip");
+ }
+
+ private void checkTimestamps(String file) throws IOException {
+ try (ZipFile zip = new ZipFile(file)) {
+ final var entries = zip.entries();
+ while (entries.hasMoreElements()) {
+ final ZipEntry entry = entries.nextElement();
+ Assert.assertEquals(EXPECTED_TIMESTAMP_INSTANT, entry.getLastModifiedTime().toInstant());
+ }
+ }
+ }
+}
diff --git a/tycho-p2-director-plugin/src/main/java/org/eclipse/tycho/plugins/p2/director/ProductArchiverMojo.java b/tycho-p2-director-plugin/src/main/java/org/eclipse/tycho/plugins/p2/director/ProductArchiverMojo.java
index 9ebde9ad07..aec193db23 100644
--- a/tycho-p2-director-plugin/src/main/java/org/eclipse/tycho/plugins/p2/director/ProductArchiverMojo.java
+++ b/tycho-p2-director-plugin/src/main/java/org/eclipse/tycho/plugins/p2/director/ProductArchiverMojo.java
@@ -255,6 +255,9 @@ private void createCommonsCompressTarGz(File productArchive, File sourceDir) thr
TarGzArchiver archiver = new TarGzArchiver();
archiver.setStoreCreationTimeAttribute(storeCreationTime);
archiver.setLog(getLog());
+ // configure for Reproducible Builds based on outputTimestamp value
+ MavenArchiver.parseBuildOutputTimestamp(outputTimestamp).map(FileTime::from)
+ .ifPresent(modifiedTime -> archiver.configureReproducibleBuild(modifiedTime));
archiver.addDirectory(sourceDir);
archiver.setDestFile(productArchive);
archiver.createArchive();
diff --git a/tycho-p2-director-plugin/src/main/java/org/eclipse/tycho/plugins/tar/TarGzArchiver.java b/tycho-p2-director-plugin/src/main/java/org/eclipse/tycho/plugins/tar/TarGzArchiver.java
index 7c879ac6ae..19f3402824 100644
--- a/tycho-p2-director-plugin/src/main/java/org/eclipse/tycho/plugins/tar/TarGzArchiver.java
+++ b/tycho-p2-director-plugin/src/main/java/org/eclipse/tycho/plugins/tar/TarGzArchiver.java
@@ -22,6 +22,7 @@
import java.nio.file.Files;
import java.nio.file.LinkOption;
import java.nio.file.Path;
+import java.nio.file.attribute.FileTime;
import java.nio.file.attribute.PosixFileAttributeView;
import java.nio.file.attribute.PosixFileAttributes;
import java.util.ArrayList;
@@ -52,6 +53,7 @@ public class TarGzArchiver {
private List sourceDirs = new ArrayList<>();
private Log log = new SystemStreamLog();
private boolean storeCreationTime;
+ private FileTime outputTimestamp = null;
public TarGzArchiver() {
}
@@ -68,6 +70,10 @@ public void setStoreCreationTimeAttribute(boolean storeCreationTime) {
this.storeCreationTime = storeCreationTime;
}
+ public void configureReproducibleBuild(FileTime timestamp) {
+ this.outputTimestamp = timestamp;
+ }
+
public void addDirectory(File directory) {
this.sourceDirs.add(directory);
}
@@ -131,9 +137,15 @@ private TarArchiveEntry createTarEntry(File tarRootDir, File source) throws IOEx
if (attrs != null) {
tarEntry.setMode(FilePermissionHelper.toOctalFileMode(attrs.permissions()));
}
- tarEntry.setModTime(source.lastModified());
+ if (outputTimestamp == null) {
+ tarEntry.setModTime(source.lastModified());
+ } else {
+ tarEntry.setModTime(outputTimestamp);
+ }
if (!storeCreationTime) { // GNU tar cannot handle 'LIBARCHIVE.creationtime' attributes and emits a lot of warnings on it
tarEntry.setCreationTime(null);
+ } else if (outputTimestamp != null) {
+ tarEntry.setCreationTime(outputTimestamp);
}
return tarEntry;
}