Skip to content

Commit

Permalink
Write Properties files in a reproducible way
Browse files Browse the repository at this point in the history
  • Loading branch information
Zlika committed Jan 10, 2025
1 parent 4ca8978 commit 92e4fbc
Show file tree
Hide file tree
Showing 13 changed files with 131 additions and 93 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -40,6 +40,7 @@
import org.codehaus.plexus.component.annotations.Requirement;
import org.codehaus.plexus.logging.Logger;
import org.eclipse.equinox.internal.p2.repository.AuthenticationFailedException;
import org.eclipse.tycho.ReproducibleUtils;

@Component(role = HttpCache.class)
public class SharedHttpCacheStorage implements HttpCache {
Expand Down Expand Up @@ -391,12 +392,9 @@ protected void updateHeader(Headers response, int code) throws IOException, File
header.put(key, value.stream().collect(Collectors.joining(",")));
}
}
FileUtils.forceMkdir(file.getParentFile());
try (OutputStream out = new BufferedOutputStream(new FileOutputStream(headerFile))) {
// we store the header here, this might be a 404 response or (permanent)
// redirect we probably need to work with later on
header.store(out, null);
}
// we store the header here, this might be a 404 response or (permanent)
// redirect we probably need to work with later on
ReproducibleUtils.storeProperties(header, headerFile.toPath());
}

private synchronized Date pareHttpDate(String input) {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -12,12 +12,9 @@
*******************************************************************************/
package org.eclipse.sisu.equinox.launching.internal;

import java.io.BufferedOutputStream;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.OutputStream;
import java.util.ArrayList;
import java.util.Collection;
import java.util.HashMap;
Expand Down Expand Up @@ -45,6 +42,7 @@
import org.eclipse.sisu.equinox.launching.EquinoxInstallation;
import org.eclipse.sisu.equinox.launching.EquinoxInstallationDescription;
import org.eclipse.sisu.equinox.launching.EquinoxInstallationFactory;
import org.eclipse.tycho.ReproducibleUtils;
import org.eclipse.tycho.TychoConstants;
import org.osgi.framework.Constants;

Expand Down Expand Up @@ -148,12 +146,8 @@ public EquinoxInstallation createInstallation(EquinoxInstallationDescription des
}

File configIni = new File(location, TychoConstants.CONFIG_INI_PATH);
ReproducibleUtils.storeProperties(p, configIni.toPath());
File configurationLocation = configIni.getParentFile();
configurationLocation.mkdirs();
try (OutputStream fos = new BufferedOutputStream(new FileOutputStream(configIni))) {
p.store(fos, null);
}

return new DefaultEquinoxInstallation(description, location, configurationLocation);
} catch (IOException e) {
throw new RuntimeException("Exception creating test eclipse runtime", e);
Expand Down Expand Up @@ -210,9 +204,7 @@ private String createDevProperties(File location, Map<String, String> devEntries
File file = new File(location, "dev.properties");
Properties properties = new Properties();
properties.putAll(devEntries);
try (OutputStream os = new BufferedOutputStream(new FileOutputStream(file))) {
properties.store(os, null);
}
ReproducibleUtils.storeProperties(properties, file.toPath());
return file.toURI().toURL().toExternalForm();
}

Expand Down
55 changes: 55 additions & 0 deletions tycho-api/src/main/java/org/eclipse/tycho/ReproducibleUtils.java
Original file line number Diff line number Diff line change
@@ -0,0 +1,55 @@
/*******************************************************************************
* 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;

import java.io.BufferedOutputStream;
import java.io.ByteArrayOutputStream;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.OutputStream;
import java.nio.charset.StandardCharsets;
import java.nio.file.Files;
import java.nio.file.Path;
import java.util.Properties;
import java.util.stream.Collectors;

/**
* Utility methods for reproducible builds.
*/
public class ReproducibleUtils {
private ReproducibleUtils() {
}

/**
* Writes the property list to the output stream in a reproducible way. The java.util.Properties
* class writes the lines in a non-reproducible order, adds a non-reproducible timestamp and
* uses platform-dependent new line characters.
*
* @param properties
* the properties object to write to the file.
* @param file
* the file to write to. All the missing parent directories are also created.
* @throws IOException
* if writing the property list to the specified output stream throws an
* IOException.
*/
public static void storeProperties(Properties properties, Path file) throws IOException {
final Path folder = file.getParent();
if (folder != null) {
Files.createDirectories(folder);
}
final ByteArrayOutputStream baos = new ByteArrayOutputStream();
properties.store(baos, null);
final String content = baos.toString(StandardCharsets.ISO_8859_1).lines().filter(line -> !line.startsWith("#"))
.sorted().collect(Collectors.joining("\n", "", "\n"));
try (OutputStream os = new BufferedOutputStream(new FileOutputStream(file.toFile()))) {
os.write(content.getBytes(StandardCharsets.ISO_8859_1));
}
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -14,12 +14,9 @@

import static org.eclipse.tycho.p2.repository.BundleConstants.BUNDLE_ID;

import java.io.BufferedOutputStream;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.OutputStream;
import java.util.HashMap;
import java.util.LinkedHashMap;
import java.util.Map;
Expand All @@ -29,6 +26,7 @@
import org.eclipse.core.runtime.IStatus;
import org.eclipse.core.runtime.Status;
import org.eclipse.equinox.p2.core.ProvisionException;
import org.eclipse.tycho.ReproducibleUtils;
import org.eclipse.tycho.TychoConstants;
import org.eclipse.tycho.p2.repository.MavenRepositoryCoordinates;
import org.eclipse.tycho.p2.repository.RepositoryReader;
Expand Down Expand Up @@ -151,7 +149,7 @@ private void store() throws ProvisionException {
}

try {
writeProperties(outputProperties, mapFile);
ReproducibleUtils.storeProperties(outputProperties, mapFile.toPath());
} catch (IOException e) {
String message = "I/O error while writing repository to " + mapFile;
int code = ProvisionException.REPOSITORY_FAILED_WRITE;
Expand All @@ -160,10 +158,4 @@ private void store() throws ProvisionException {
}

}

private static void writeProperties(Properties properties, File outputFile) throws IOException {
try (OutputStream outputStream = new BufferedOutputStream(new FileOutputStream(outputFile))) {
properties.store(outputStream, null);
}
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -13,12 +13,9 @@
*******************************************************************************/
package org.eclipse.tycho.p2resolver;

import java.io.BufferedOutputStream;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.OutputStream;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.HashMap;
Expand Down Expand Up @@ -69,6 +66,7 @@
import org.eclipse.tycho.OptionalResolutionAction;
import org.eclipse.tycho.PackagingType;
import org.eclipse.tycho.ReactorProject;
import org.eclipse.tycho.ReproducibleUtils;
import org.eclipse.tycho.TargetEnvironment;
import org.eclipse.tycho.TychoConstants;
import org.eclipse.tycho.core.osgitools.BundleReader;
Expand Down Expand Up @@ -483,13 +481,7 @@ static void writeArtifactLocations(File outputFile, Map<String, File> artifactLo
}
}

writeProperties(outputProperties, outputFile);
}

private static void writeProperties(Properties properties, File outputFile) throws IOException {
try (OutputStream outputStream = new BufferedOutputStream(new FileOutputStream(outputFile))) {
properties.store(outputStream, null);
}
ReproducibleUtils.storeProperties(outputProperties, outputFile.toPath());
}

/**
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -17,7 +17,6 @@
import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.OutputStream;
import java.net.HttpURLConnection;
import java.net.URI;
import java.net.URLConnection;
Expand Down Expand Up @@ -72,6 +71,7 @@
import org.eclipse.tycho.ArtifactType;
import org.eclipse.tycho.BuildDirectory;
import org.eclipse.tycho.DependencySeed;
import org.eclipse.tycho.ReproducibleUtils;
import org.eclipse.tycho.core.shared.StatusTool;
import org.eclipse.tycho.p2.repository.GAV;
import org.eclipse.tycho.p2.repository.RepositoryLayoutHelper;
Expand Down Expand Up @@ -523,9 +523,8 @@ private void writeP2Index(File repositoryDestination) throws FacadeException {
properties.setProperty("version", "1");
properties.setProperty("artifact.repository.factory.order", "artifacts.xml,!");
properties.setProperty("metadata.repository.factory.order", "content.xml,!");
try (OutputStream stream = new BufferedOutputStream(
new FileOutputStream(new File(repositoryDestination, P2_INDEX_FILE)))) {
properties.store(stream, null);
try {
ReproducibleUtils.storeProperties(properties, new File(repositoryDestination, P2_INDEX_FILE).toPath());
} catch (IOException e) {
throw new FacadeException("writing index file failed", e);
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,7 @@
package org.eclipse.tycho.test.reproducible;

import java.io.IOException;
import java.nio.charset.StandardCharsets;
import java.nio.file.FileSystem;
import java.nio.file.FileSystems;
import java.nio.file.Files;
Expand All @@ -25,26 +26,40 @@
import org.junit.Assert;
import org.junit.Test;

/**
* Tests that the build artifacts produced by Tycho are reproducible.
*/
public class ReproducibleBuildTest 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());
Verifier verifier;

/**
* Check that the build is reproducible.
* Run the maven integration tests related to reproducible builds.
*
* @throws Exception
*/
@Test
public void test() throws Exception {
Verifier verifier = getVerifier("reproducible-build");
public void testReproducible() throws Exception {
verifier = getVerifier("reproducible-build");
verifier.executeGoals(List.of("clean", "verify"));
verifier.verifyErrorFreeLog();

// 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.
checkArchiveTimestamps();
testBuildQualifier();
testPropertiesFiles();
}

/**
* Checks 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.
*/
private void checkArchiveTimestamps() throws Exception {
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");
Expand All @@ -55,11 +70,8 @@ public void test() throws Exception {
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");

// Check that the build qualifier uses the timestamp specified in the
// "project.build.outputTimestamp" property of the pom file.
checkBuildQualifier(verifier.getBasedir()
+ "/reproducible.buildqualifier/target/reproducible.buildqualifier-1.0.0-SNAPSHOT.jar");
checkTimestamps(
verifier.getBasedir() + "/reproducible.repository/target/products/main.product.id-linux.gtk.x86.zip");
}

private void checkTimestamps(String file) throws IOException {
Expand All @@ -72,11 +84,33 @@ private void checkTimestamps(String file) throws IOException {
}
}

private void checkBuildQualifier(String file) throws IOException {
/**
* Checks that the build qualifier uses the timestamp specified in the
* "project.build.outputTimestamp" property of the pom file.
*
* @throws IOException
*/
private void testBuildQualifier() throws IOException {
final String file = verifier.getBasedir()
+ "/reproducible.buildqualifier/target/reproducible.buildqualifier-1.0.0-SNAPSHOT.jar";
try (FileSystem fileSystem = FileSystems.newFileSystem(Path.of(file))) {
Path manifest = fileSystem.getPath("META-INF/MANIFEST.MF");
List<String> lines = Files.readAllLines(manifest);
final Path manifest = fileSystem.getPath("META-INF/MANIFEST.MF");
final List<String> lines = Files.readAllLines(manifest);
Assert.assertTrue(lines.stream().anyMatch(l -> l.equals("Bundle-Version: 1.0.0.202301010000")));
}
}

/**
* Checks that the generated properties files are reproducible.
*
* @throws IOException
*/
private void testPropertiesFiles() throws IOException {
final String file = verifier.getBasedir() + "/reproducible.bundle/target/reproducible.bundle-1.0.0-sources.jar";
try (FileSystem fileSystem = FileSystems.newFileSystem(Path.of(file))) {
final Path propFile = fileSystem.getPath("OSGI-INF/l10n/bundle-src.properties");
final String content = Files.readString(propFile, StandardCharsets.ISO_8859_1);
Assert.assertEquals("bundleName=Reproducible-bundle Source\n" + "bundleVendor=unknown\n", content);
}
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -72,6 +72,7 @@
import org.eclipse.equinox.p2.core.IProvisioningAgent;
import org.eclipse.equinox.p2.repository.artifact.IArtifactRepositoryManager;
import org.eclipse.tycho.PackagingType;
import org.eclipse.tycho.ReproducibleUtils;
import org.eclipse.tycho.TychoConstants;
import org.eclipse.tycho.core.PGPService;
import org.eclipse.tycho.p2maven.tools.TychoFeaturesAndBundlesPublisherApplication;
Expand Down Expand Up @@ -484,9 +485,7 @@ protected File createMavenAdvice(Artifact artifact) throws MojoExecutionExceptio
addProvidesAndProperty(properties, TychoConstants.PROP_EXTENSION, artifact.getType(), cnt++);
addProvidesAndProperty(properties, TychoConstants.PROP_CLASSIFIER, artifact.getClassifier(), cnt++);
addProvidesAndProperty(properties, "maven-scope", artifact.getScope(), cnt++);
try (OutputStream os = new BufferedOutputStream(new FileOutputStream(p2))) {
properties.store(os, null);
}
ReproducibleUtils.storeProperties(properties, p2.toPath());
return p2;
} catch (IOException e) {
throw new MojoExecutionException("failed to generate p2.inf", e);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -12,13 +12,10 @@
*******************************************************************************/
package org.eclipse.tycho.packaging.reverseresolve;

import java.io.BufferedOutputStream;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.nio.file.Files;
import java.security.MessageDigest;
import java.util.List;
Expand All @@ -36,6 +33,7 @@
import org.codehaus.plexus.component.annotations.Component;
import org.codehaus.plexus.component.annotations.Requirement;
import org.codehaus.plexus.logging.Logger;
import org.eclipse.tycho.ReproducibleUtils;
import org.eclipse.tycho.core.shared.MavenContext;
import org.eclipse.tycho.p2.repository.GAV;
import org.eclipse.tycho.p2.repository.RepositoryLayoutHelper;
Expand Down Expand Up @@ -160,9 +158,7 @@ private void cacheResult(File cacheFile, Dependency dependency) {
properties.setProperty(KEY_ARTIFACT_ID, dependency.getArtifactId());
properties.setProperty(KEY_VERSION, dependency.getVersion());
properties.setProperty(KEY_TYPE, dependency.getType());
try (OutputStream stream = new BufferedOutputStream(new FileOutputStream(cacheFile))) {
properties.store(stream, null);
}
ReproducibleUtils.storeProperties(properties, cacheFile.toPath());
} catch (IOException e) {
// can't create cache file then...
}
Expand Down
Loading

0 comments on commit 92e4fbc

Please sign in to comment.