Skip to content

Commit 33093c7

Browse files
authored
Make the Equo-based steps round-trip serializable (#1950 towards #1274)
2 parents 26fbd03 + 99eebec commit 33093c7

File tree

7 files changed

+73
-95
lines changed

7 files changed

+73
-95
lines changed

lib-extra/src/main/java/com/diffplug/spotless/extra/EquoBasedStepBuilder.java

+45-31
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright 2016-2023 DiffPlug
2+
* Copyright 2016-2024 DiffPlug
33
*
44
* Licensed under the Apache License, Version 2.0 (the "License");
55
* you may not use this file except in compliance with the License.
@@ -34,7 +34,7 @@
3434
import com.diffplug.spotless.FormatterStep;
3535
import com.diffplug.spotless.JarState;
3636
import com.diffplug.spotless.Provisioner;
37-
import com.diffplug.spotless.ThrowingEx;
37+
import com.diffplug.spotless.SerializedFunction;
3838

3939
import dev.equo.solstice.NestedJars;
4040
import dev.equo.solstice.p2.P2ClientCache;
@@ -48,19 +48,19 @@
4848
public abstract class EquoBasedStepBuilder {
4949
private final String formatterName;
5050
private final Provisioner mavenProvisioner;
51-
private final ThrowingEx.Function<State, FormatterFunc> stateToFormatter;
51+
private final SerializedFunction<State, FormatterFunc> stateToFormatter;
5252
private String formatterVersion;
5353
private Iterable<File> settingsFiles = new ArrayList<>();
5454
private Map<String, String> p2Mirrors = Map.of();
5555

5656
/** @deprecated if you use this constructor you *must* call {@link #setVersion(String)} before calling {@link #build()} */
5757
@Deprecated
58-
public EquoBasedStepBuilder(String formatterName, Provisioner mavenProvisioner, ThrowingEx.Function<State, FormatterFunc> stateToFormatter) {
58+
public EquoBasedStepBuilder(String formatterName, Provisioner mavenProvisioner, SerializedFunction<State, FormatterFunc> stateToFormatter) {
5959
this(formatterName, mavenProvisioner, null, stateToFormatter);
6060
}
6161

6262
/** Initialize valid default configuration, taking latest version */
63-
public EquoBasedStepBuilder(String formatterName, Provisioner mavenProvisioner, @Nullable String defaultVersion, ThrowingEx.Function<State, FormatterFunc> stateToFormatter) {
63+
public EquoBasedStepBuilder(String formatterName, Provisioner mavenProvisioner, @Nullable String defaultVersion, SerializedFunction<State, FormatterFunc> stateToFormatter) {
6464
this.formatterName = formatterName;
6565
this.mavenProvisioner = mavenProvisioner;
6666
this.formatterVersion = defaultVersion;
@@ -83,11 +83,6 @@ public void setP2Mirrors(Collection<P2Mirror> p2Mirrors) {
8383
this.p2Mirrors = p2Mirrors.stream().collect(toMap(P2Mirror::getPrefix, P2Mirror::getUrl));
8484
}
8585

86-
/** Returns the FormatterStep (whose state will be calculated lazily). */
87-
public FormatterStep build() {
88-
return FormatterStep.createLazy(formatterName, this::get, stateToFormatter);
89-
}
90-
9186
protected abstract P2Model model(String version);
9287

9388
protected void addPlatformRepo(P2Model model, String version) {
@@ -107,26 +102,28 @@ protected void addPlatformRepo(P2Model model, String version) {
107102
}
108103
}
109104

110-
/** Creates the state of the configuration. */
111-
EquoBasedStepBuilder.State get() throws Exception {
112-
P2QueryResult query;
113-
try {
114-
query = createModelWithMirrors().query(P2ClientCache.PREFER_OFFLINE, P2QueryCache.ALLOW);
115-
} catch (Exception x) {
116-
throw new IOException("Failed to load " + formatterName + ": " + x, x);
117-
}
118-
var classpath = new ArrayList<File>();
119-
var mavenDeps = new ArrayList<String>();
120-
mavenDeps.add("dev.equo.ide:solstice:1.7.4");
121-
mavenDeps.add("com.diffplug.durian:durian-swt.os:4.2.0");
122-
mavenDeps.addAll(query.getJarsOnMavenCentral());
123-
classpath.addAll(mavenProvisioner.provisionWithTransitives(false, mavenDeps));
124-
classpath.addAll(query.getJarsNotOnMavenCentral());
125-
for (var nested : NestedJars.inFiles(query.getJarsNotOnMavenCentral()).extractAllNestedJars()) {
126-
classpath.add(nested.getValue());
127-
}
128-
var jarState = JarState.preserveOrder(classpath);
129-
return new State(formatterVersion, jarState, FileSignature.signAsList(settingsFiles));
105+
/** Returns the FormatterStep (whose state will be calculated lazily). */
106+
public FormatterStep build() {
107+
var roundtrippableState = new EquoStep(formatterVersion, FileSignature.promise(settingsFiles), JarState.promise(() -> {
108+
P2QueryResult query;
109+
try {
110+
query = createModelWithMirrors().query(P2ClientCache.PREFER_OFFLINE, P2QueryCache.ALLOW);
111+
} catch (Exception x) {
112+
throw new IOException("Failed to load " + formatterName + ": " + x, x);
113+
}
114+
var classpath = new ArrayList<File>();
115+
var mavenDeps = new ArrayList<String>();
116+
mavenDeps.add("dev.equo.ide:solstice:1.7.5");
117+
mavenDeps.add("com.diffplug.durian:durian-swt.os:4.2.0");
118+
mavenDeps.addAll(query.getJarsOnMavenCentral());
119+
classpath.addAll(mavenProvisioner.provisionWithTransitives(false, mavenDeps));
120+
classpath.addAll(query.getJarsNotOnMavenCentral());
121+
for (var nested : NestedJars.inFiles(query.getJarsNotOnMavenCentral()).extractAllNestedJars()) {
122+
classpath.add(nested.getValue());
123+
}
124+
return JarState.preserveOrder(classpath);
125+
}));
126+
return FormatterStep.create(formatterName, roundtrippableState, EquoStep::state, stateToFormatter);
130127
}
131128

132129
private P2Model createModelWithMirrors() {
@@ -152,12 +149,29 @@ private P2Model createModelWithMirrors() {
152149
return model;
153150
}
154151

152+
static class EquoStep implements Serializable {
153+
private static final long serialVersionUID = 1;
154+
private final String semanticVersion;
155+
private final FileSignature.Promised settingsPromise;
156+
private final JarState.Promised jarPromise;
157+
158+
EquoStep(String semanticVersion, FileSignature.Promised settingsPromise, JarState.Promised jarPromise) {
159+
this.semanticVersion = semanticVersion;
160+
this.settingsPromise = settingsPromise;
161+
this.jarPromise = jarPromise;
162+
}
163+
164+
private State state() {
165+
return new State(semanticVersion, jarPromise.get(), settingsPromise.get());
166+
}
167+
}
168+
155169
/**
156170
* State of Eclipse configuration items, providing functionality to derived information
157171
* based on the state.
158172
*/
159173
public static class State implements Serializable {
160-
private static final long serialVersionUID = 584400372246020995L;
174+
private static final long serialVersionUID = 1;
161175
final String semanticVersion;
162176
final JarState jarState;
163177
final FileSignature settingsFiles;

lib-extra/src/test/java/com/diffplug/spotless/extra/cpp/EclipseCdtFormatterStepTest.java

+5-6
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright 2016-2023 DiffPlug
2+
* Copyright 2016-2024 DiffPlug
33
*
44
* Licensed under the Apache License, Version 2.0 (the "License");
55
* you may not use this file except in compliance with the License.
@@ -24,17 +24,16 @@
2424
import com.diffplug.spotless.extra.eclipse.EquoResourceHarness;
2525

2626
class EclipseCdtFormatterStepTest extends EquoResourceHarness {
27-
private final static String INPUT = "#include <a.h>;\nint main(int argc, \nchar *argv[]) {}";
28-
private final static String EXPECTED = "#include <a.h>;\nint main(int argc, char *argv[]) {\n}\n";
29-
3027
public EclipseCdtFormatterStepTest() {
31-
super(EclipseCdtFormatterStep.createBuilder(TestProvisioner.mavenCentral()), INPUT, EXPECTED);
28+
super(EclipseCdtFormatterStep.createBuilder(TestProvisioner.mavenCentral()));
3229
}
3330

3431
@ParameterizedTest
3532
@MethodSource
3633
void formatWithVersion(String version) throws Exception {
37-
assertFormatted(version);
34+
harnessFor(version).test("main.c",
35+
"#include <a.h>;\nint main(int argc, \nchar *argv[]) {}",
36+
"#include <a.h>;\nint main(int argc, char *argv[]) {\n}\n");
3837
}
3938

4039
private static Stream<String> formatWithVersion() {
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright 2016-2023 DiffPlug
2+
* Copyright 2016-2024 DiffPlug
33
*
44
* Licensed under the Apache License, Version 2.0 (the "License");
55
* you may not use this file except in compliance with the License.
@@ -15,14 +15,12 @@
1515
*/
1616
package com.diffplug.spotless.extra.eclipse;
1717

18-
import static org.assertj.core.api.Assertions.assertThat;
19-
2018
import java.io.File;
2119
import java.util.Arrays;
2220

2321
import com.diffplug.spotless.FormatterStep;
24-
import com.diffplug.spotless.LineEnding;
2522
import com.diffplug.spotless.ResourceHarness;
23+
import com.diffplug.spotless.StepHarnessWithFile;
2624
import com.diffplug.spotless.extra.EquoBasedStepBuilder;
2725

2826
/**
@@ -43,57 +41,19 @@
4341
*/
4442
public class EquoResourceHarness extends ResourceHarness {
4543
private final EquoBasedStepBuilder stepBuilder;
46-
private final String fileName;
47-
private final String input;
48-
private final String expected;
49-
50-
/**
51-
* Create harness to be used for several versions of the formatter step
52-
* @param builder Eclipse Formatter step builder
53-
* @param unformatted Simple unformatted input
54-
* @param formatted Expected formatted output
55-
*/
56-
public EquoResourceHarness(EquoBasedStepBuilder builder, String unformatted, String formatted) {
57-
this(builder, "someSourceFile", unformatted, formatted);
58-
}
5944

6045
/**
6146
* Create harness to be used for several versions of the formatter step
6247
* @param builder Eclipse Formatter step builder
63-
* @param sourceFileName File name of the source file
64-
* @param unformatted Simple unformatted input
65-
* @param formatted Expected formatted output
6648
*/
67-
public EquoResourceHarness(EquoBasedStepBuilder builder, String sourceFileName, String unformatted, String formatted) {
49+
public EquoResourceHarness(EquoBasedStepBuilder builder) {
6850
stepBuilder = builder;
69-
fileName = sourceFileName;
70-
input = unformatted;
71-
expected = formatted;
7251
}
7352

74-
/**
75-
* Assert that formatting input results in expected output
76-
* @param formatterVersion Formatter version
77-
* @param settingsFiles Formatter settings
78-
* @return Formatted string
79-
*/
80-
protected String assertFormatted(String formatterVersion, File... settingsFiles) throws Exception {
81-
String output = format(formatterVersion, settingsFiles);
82-
assertThat(output).isEqualTo(expected);
83-
return output;
84-
}
85-
86-
/**
87-
* Formatting input results and returns output
88-
* @param formatterVersion Formatter version
89-
* @param settingsFiles Formatter settings
90-
* @return Formatted string
91-
*/
92-
protected String format(String formatterVersion, File... settingsFiles) throws Exception {
93-
File inputFile = setFile(fileName).toContent(input);
53+
protected StepHarnessWithFile harnessFor(String formatterVersion, File... settingsFiles) throws Exception {
9454
stepBuilder.setVersion(formatterVersion);
9555
stepBuilder.setPreferences(Arrays.asList(settingsFiles));
9656
FormatterStep step = stepBuilder.build();
97-
return LineEnding.toUnix(step.format(input, inputFile));
57+
return StepHarnessWithFile.forStep(this, step);
9858
}
9959
}

lib-extra/src/test/java/com/diffplug/spotless/extra/groovy/GrEclipseFormatterStepTest.java

+4-3
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright 2016-2023 DiffPlug
2+
* Copyright 2016-2024 DiffPlug
33
*
44
* Licensed under the Apache License, Version 2.0 (the "License");
55
* you may not use this file except in compliance with the License.
@@ -28,13 +28,14 @@ public class GrEclipseFormatterStepTest extends EquoResourceHarness {
2828
private final static String EXPECTED = "class F{\n\tdef m(){}\n}";
2929

3030
public GrEclipseFormatterStepTest() {
31-
super(GrEclipseFormatterStep.createBuilder(TestProvisioner.mavenCentral()), INPUT, EXPECTED);
31+
super(GrEclipseFormatterStep.createBuilder(TestProvisioner.mavenCentral()));
3232
}
3333

3434
@ParameterizedTest
3535
@MethodSource
3636
void formatWithVersion(String version) throws Exception {
37-
assertFormatted(version);
37+
harnessFor(version).test("test.groovy",
38+
"class F{ def m(){} }", "class F{\n\tdef m(){}\n}");
3839
}
3940

4041
private static Stream<String> formatWithVersion() {

lib-extra/src/test/java/com/diffplug/spotless/extra/java/EclipseJdtFormatterStepTest.java

+8-10
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright 2016-2023 DiffPlug
2+
* Copyright 2016-2024 DiffPlug
33
*
44
* Licensed under the Apache License, Version 2.0 (the "License");
55
* you may not use this file except in compliance with the License.
@@ -15,7 +15,6 @@
1515
*/
1616
package com.diffplug.spotless.extra.java;
1717

18-
import java.io.File;
1918
import java.util.stream.Stream;
2019

2120
import org.junit.jupiter.api.Nested;
@@ -28,21 +27,20 @@
2827
import com.diffplug.spotless.extra.eclipse.EquoResourceHarness;
2928

3029
class EclipseJdtFormatterStepTest extends EquoResourceHarness {
31-
private final static String INPUT = "package p; class C{}";
32-
private final static String EXPECTED = "package p;\nclass C {\n}";
33-
3430
private static EquoBasedStepBuilder createBuilder() {
3531
return EclipseJdtFormatterStep.createBuilder(TestProvisioner.mavenCentral());
3632
}
3733

3834
public EclipseJdtFormatterStepTest() {
39-
super(createBuilder(), INPUT, EXPECTED);
35+
super(createBuilder());
4036
}
4137

4238
@ParameterizedTest
4339
@MethodSource
4440
void formatWithVersion(String version) throws Exception {
45-
assertFormatted(version);
41+
harnessFor(version).test("test.java",
42+
"package p; class C{}",
43+
"package p;\nclass C {\n}");
4644
}
4745

4846
private static Stream<String> formatWithVersion() {
@@ -53,13 +51,13 @@ private static Stream<String> formatWithVersion() {
5351
@Nested
5452
class NewFormatInterface extends EquoResourceHarness {
5553
public NewFormatInterface() {
56-
super(createBuilder(), "module-info.java", getTestResource("java/eclipse/ModuleInfoUnformatted.test"), getTestResource("java/eclipse/ModuleInfoFormatted.test"));
54+
super(createBuilder());
5755
}
5856

5957
@Test
6058
void formatModuleInfo() throws Exception {
61-
File settingsFile = createTestFile("java/eclipse/ModuleInfo.prefs");
62-
assertFormatted("4.11", settingsFile);
59+
harnessFor("4.11", createTestFile("java/eclipse/ModuleInfo.prefs"))
60+
.testResource("module-info.java", "java/eclipse/ModuleInfoUnformatted.test", "java/eclipse/ModuleInfoFormatted.test");
6361
}
6462
}
6563
}

lib/src/main/java/com/diffplug/spotless/FileSignature.java

+4
Original file line numberDiff line numberDiff line change
@@ -121,6 +121,10 @@ public Promised asPromise() {
121121
return new Promised(files, this);
122122
}
123123

124+
public static Promised promise(Iterable<File> files) {
125+
return new Promised(MoreIterables.toNullHostileList(files), null);
126+
}
127+
124128
/** Returns all of the files in this signature, throwing an exception if there are more or less than 1 file. */
125129
public Collection<File> files() {
126130
return Collections.unmodifiableList(files);

testlib/src/main/java/com/diffplug/spotless/StepHarnessBase.java

+2
Original file line numberDiff line numberDiff line change
@@ -43,6 +43,8 @@ protected StepHarnessBase(Formatter formatter) {
4343
supportsRoundTrip = true;
4444
} else if (onlyStepName.equals("diktat")) {
4545
supportsRoundTrip = true;
46+
} else if (onlyStepName.equals("eclipse jdt formatter") || onlyStepName.equals("eclipse cdt formatter") || onlyStepName.equals("eclipse groovy formatter")) {
47+
supportsRoundTrip = true;
4648
}
4749
}
4850
}

0 commit comments

Comments
 (0)