Skip to content

Commit 452fbd9

Browse files
committedNov 1, 2016
Used LazyForwardingEquality to implement FormatterStep.Strict, which guarantees correct up-to-date checks.
1 parent 6113458 commit 452fbd9

File tree

2 files changed

+115
-5
lines changed

2 files changed

+115
-5
lines changed
 

‎src/main/java/com/diffplug/gradle/spotless/FormatterStep.java

+105-3
Original file line numberDiff line numberDiff line change
@@ -18,6 +18,7 @@
1818
import java.io.File;
1919
import java.io.Serializable;
2020
import java.util.Objects;
21+
import java.util.function.BiFunction;
2122
import java.util.function.Predicate;
2223
import java.util.function.Supplier;
2324

@@ -45,7 +46,7 @@ public interface FormatterStep extends Serializable {
4546
* @return The formatted content, guaranteed to only have unix-style newlines
4647
* @throws Throwable when the formatter steps experiences a problem
4748
*/
48-
String format(String raw, File file) throws Throwable;
49+
String format(String rawUnix, File file) throws Throwable;
4950

5051
/**
5152
* Returns a new FormatterStep which will only apply its changes
@@ -55,7 +56,98 @@ default FormatterStep filterByFile(Predicate<File> filter) {
5556
return new FilterByFileFormatterStep(this, filter);
5657
}
5758

59+
/**
60+
* Implements a FormatterStep in a strict way which guarantees correct and lazy implementation
61+
* of up-to-date checks. This maximizes performance for cases where the FormatterStep is not
62+
* actually needed (e.g. don't load eclipse setting file unless this step is actually running)
63+
* while also ensuring that gradle can detect changes in a step's settings to determine that
64+
* it needs to rerun a format.
65+
*/
66+
abstract class Strict<Key extends Serializable> extends LazyForwardingEquality<Key> implements FormatterStep {
Has a conversation. Original line has a conversation.
67+
private static final long serialVersionUID = 1L;
68+
69+
/**
70+
* Implements the formatting function strictly in terms
71+
* of the input data and the result of {@link #calculateKey()}.
72+
*/
73+
protected abstract String format(Key key, String rawUnix, File file);
74+
75+
@Override
76+
public final String format(String rawUnix, File file) {
77+
return format(key(), rawUnix, file);
78+
}
79+
}
80+
81+
/**
82+
* @param name
83+
* The name of the formatter step
84+
* @param keySupplier
85+
* If the rule has any state, this supplier will calculate it lazily
86+
* @param formatter
87+
* A pure function which calculates a formatted string from an unformatted
88+
* string, using only the state supplied by keySupplier and nowhere else.
89+
* @return A FormatterStep
90+
*/
91+
public static <Key extends Serializable> FormatterStep createLazy(
92+
String name,
93+
Throwing.Specific.Supplier<Key, Exception> keySupplier,
94+
BiFunction<Key, String, String> formatter) {
95+
return new Strict<Key>() {
96+
private static final long serialVersionUID = 1L;
97+
98+
@Override
99+
public String getName() {
100+
return name;
101+
}
102+
103+
@Override
104+
protected Key calculateKey() throws Exception {
105+
return keySupplier.get();
106+
}
107+
108+
@Override
109+
protected String format(Key key, String rawUnix, File file) {
110+
return formatter.apply(key, rawUnix);
111+
}
112+
};
113+
}
114+
115+
/**
116+
* @param name
117+
* The name of the formatter step
118+
* @param key
119+
* If the rule has any state, this key must contain all of it
120+
* @param formatter
121+
* A pure function which calculates a formatted string from an unformatted
122+
* string, using only the state supplied by key and nowhere else.
123+
* @return A FormatterStep
124+
*/
125+
public static <Key extends Serializable> FormatterStep create(
126+
String name,
127+
Key key,
128+
BiFunction<Key, String, String> formatter) {
129+
return new Strict<Key>() {
130+
private static final long serialVersionUID = 1L;
131+
132+
@Override
133+
public String getName() {
134+
return name;
135+
}
136+
137+
@Override
138+
protected Key calculateKey() throws Exception {
139+
return key;
140+
}
141+
142+
@Override
143+
protected String format(Key key, String rawUnix, File file) {
144+
return formatter.apply(key, rawUnix);
145+
}
146+
};
147+
}
148+
58149
/** A FormatterStep which doesn't depend on the input file. */
150+
@Deprecated
59151
class FileIndependent implements FormatterStep {
60152
private final String name;
61153
private final Throwing.Function<String, String> formatter;
@@ -96,12 +188,22 @@ public int hashCode() {
96188
private static final long serialVersionUID = 1L;
97189
}
98190

99-
/** Creates a FormatterStep from the given function. */
191+
/**
192+
* Creates a FormatterStep from the given function.
193+
*
194+
* @deprecated This method makes it hard to check up-to-dateness, use {@link FormatterStep#create(String, Serializable, BiFunction)} instead.
195+
*/
196+
@Deprecated
100197
static FormatterStep create(String name, Throwing.Function<String, String> formatter) {
101198
return new FileIndependent(name, formatter);
102199
}
103200

104-
/** Creates a FormatterStep lazily from the given formatterSupplier function. */
201+
/**
202+
* Creates a FormatterStep lazily from the given formatterSupplier function.
203+
*
204+
* @deprecated This method makes it hard to check up-to-dateness, use {@link FormatterStep#createLazy(String, com.diffplug.common.base.Throwing.Specific.Supplier, BiFunction)} instead.
205+
*/
206+
@Deprecated
105207
static FormatterStep createLazy(String name, Throwing.Supplier<Throwing.Function<String, String>> formatterSupplier) {
106208
// wrap the supplier as a regular Supplier (not a Throwing.Supplier)
107209
Supplier<Throwing.Function<String, String>> rethrowFormatterSupplier = Errors.rethrow().wrap(formatterSupplier);

‎src/main/java/com/diffplug/gradle/spotless/LazyForwardingEquality.java

+10-2
Original file line numberDiff line numberDiff line change
@@ -25,6 +25,8 @@
2525
import javax.annotation.Nonnull;
2626
import javax.annotation.Nullable;
2727

28+
import com.diffplug.common.base.Errors;
29+
2830
/**
2931
* Implements equality, hashcode, and serialization entirely in terms
3032
* of a lazily-computed key.
@@ -38,9 +40,11 @@ public abstract class LazyForwardingEquality<T extends Serializable> implements
3840
/**
3941
* This function is guaranteed to be called at most once.
4042
* If the key is never required, then it will never be called at all.
43+
*
44+
* Throws exception because it's likely that there will be some IO going on.
4145
*/
4246
@Nonnull
43-
protected abstract T calculateKey();
47+
protected abstract T calculateKey() throws Exception;
4448

4549
/** Returns the underlying key, possibly triggering a call to {{@link #calculateKey()}. */
4650
@Nonnull
@@ -49,7 +53,11 @@ protected final T key() {
4953
if (key == null) {
5054
synchronized (this) {
5155
if (key == null) {
52-
key = calculateKey();
56+
try {
57+
key = calculateKey();
58+
} catch (Exception e) {
59+
throw Errors.asRuntime(e);
60+
}
5361
}
5462
}
5563
}

0 commit comments

Comments
 (0)
Please sign in to comment.