Skip to content

Commit 638a7cd

Browse files
aragoscopybara-github
authored andcommitted
Transform options by parsing result.
Allows an existing FragmentOptions to be the base for a new one which is distinguished by a set of flags provided through an option parser (for example from the command line or a file). Step 1/N towards the platforms mapping functionality for #6426 RELNOTES: None. PiperOrigin-RevId: 238022069
1 parent 0b1ad3d commit 638a7cd

File tree

2 files changed

+153
-35
lines changed

2 files changed

+153
-35
lines changed

src/main/java/com/google/devtools/build/lib/analysis/config/BuildOptions.java

+72
Original file line numberDiff line numberDiff line change
@@ -39,7 +39,9 @@
3939
import com.google.devtools.common.options.OptionsBase;
4040
import com.google.devtools.common.options.OptionsParser;
4141
import com.google.devtools.common.options.OptionsParsingException;
42+
import com.google.devtools.common.options.OptionsParsingResult;
4243
import com.google.devtools.common.options.OptionsProvider;
44+
import com.google.devtools.common.options.ParsedOptionDescription;
4345
import com.google.protobuf.ByteString;
4446
import com.google.protobuf.CodedInputStream;
4547
import com.google.protobuf.CodedOutputStream;
@@ -312,6 +314,76 @@ public BuildOptions applyDiff(OptionsDiffForReconstruction optionsDiff) {
312314
return builder.build();
313315
}
314316

317+
/**
318+
* Applies any options set in the parsing result on top of these options, returning the resulting
319+
* build options.
320+
*
321+
* <p>To preserve fragment trimming, this method will not expand the set of included native
322+
* fragments. If the parsing result contains native options whose owning fragment is not part of
323+
* these options they will be ignored (i.e. not set on the resulting options). Starlark options
324+
* are not affected by this restriction.
325+
*
326+
* @param parsingResult any options that are being modified
327+
* @return the new options after applying the parsing result to the original options
328+
* @throws OptionsParsingException if a value in the parsing result cannot in fact be parsed
329+
*/
330+
public BuildOptions applyParsingResult(OptionsParsingResult parsingResult)
331+
throws OptionsParsingException {
332+
Map<Class<? extends FragmentOptions>, FragmentOptions> modifiedFragments =
333+
toModifiedFragments(parsingResult);
334+
335+
BuildOptions.Builder builder = builder();
336+
for (Map.Entry<Class<? extends FragmentOptions>, FragmentOptions> classAndFragment :
337+
fragmentOptionsMap.entrySet()) {
338+
Class<? extends FragmentOptions> fragmentClass = classAndFragment.getKey();
339+
if (modifiedFragments.containsKey(fragmentClass)) {
340+
builder.addFragmentOptions(modifiedFragments.get(fragmentClass));
341+
} else {
342+
builder.addFragmentOptions(classAndFragment.getValue());
343+
}
344+
}
345+
346+
Map<Label, Object> starlarkOptions = new HashMap<>(skylarkOptionsMap);
347+
Map<Label, Object> parsedStarlarkOptions =
348+
labelizeStarlarkOptions(parsingResult.getStarlarkOptions());
349+
for (Map.Entry<Label, Object> starlarkOption : parsedStarlarkOptions.entrySet()) {
350+
starlarkOptions.put(starlarkOption.getKey(), starlarkOption.getValue());
351+
}
352+
builder.addStarlarkOptions(starlarkOptions);
353+
return builder.build();
354+
}
355+
356+
private Map<Class<? extends FragmentOptions>, FragmentOptions> toModifiedFragments(
357+
OptionsParsingResult parsingResult) throws OptionsParsingException {
358+
Map<Class<? extends FragmentOptions>, FragmentOptions> replacedOptions = new HashMap<>();
359+
for (ParsedOptionDescription parsedOption : parsingResult.asListOfExplicitOptions()) {
360+
OptionDefinition optionDefinition = parsedOption.getOptionDefinition();
361+
362+
// All options obtained from an options parser are guaranteed to have been defined in an
363+
// FragmentOptions class.
364+
@SuppressWarnings("unchecked")
365+
Class<? extends FragmentOptions> fragmentOptionClass =
366+
(Class<? extends FragmentOptions>) optionDefinition.getField().getDeclaringClass();
367+
368+
FragmentOptions originalFragment = fragmentOptionsMap.get(fragmentOptionClass);
369+
if (originalFragment == null) {
370+
// Preserve trimming by ignoring fragments not present in the original options.
371+
continue;
372+
}
373+
FragmentOptions newOptions =
374+
replacedOptions.computeIfAbsent(
375+
fragmentOptionClass,
376+
(Class<? extends FragmentOptions> k) -> originalFragment.clone());
377+
try {
378+
optionDefinition.getField().set(newOptions, parsedOption.getConvertedValue());
379+
} catch (IllegalAccessException e) {
380+
throw new IllegalStateException("Couldn't set " + optionDefinition.getField(), e);
381+
}
382+
}
383+
384+
return replacedOptions;
385+
}
386+
315387
/** Creates a builder object for BuildOptions */
316388
public static Builder builder() {
317389
return new Builder();

src/test/java/com/google/devtools/build/lib/analysis/config/BuildOptionsTest.java

+81-35
Original file line numberDiff line numberDiff line change
@@ -39,20 +39,21 @@
3939
/**
4040
* A test for {@link BuildOptions}.
4141
*
42-
* Currently this tests native options and skylark options completely
43-
* separately since these two types of options do not interact. In the future when we begin to
44-
* migrate native options to skylark options, the format of this test class will need to accommodate
45-
* that overlap.
42+
* <p>Currently this tests native options and skylark options completely separately since these two
43+
* types of options do not interact. In the future when we begin to migrate native options to
44+
* skylark options, the format of this test class will need to accommodate that overlap.
4645
*/
4746
@RunWith(JUnit4.class)
4847
public class BuildOptionsTest {
49-
private static final ImmutableList<Class<? extends FragmentOptions>> TEST_OPTIONS =
48+
private static final ImmutableList<Class<? extends FragmentOptions>> BUILD_CONFIG_OPTIONS =
5049
ImmutableList.of(BuildConfiguration.Options.class);
5150

5251
@Test
5352
public void optionSetCaching() {
54-
BuildOptions a = BuildOptions.of(TEST_OPTIONS, OptionsParser.newOptionsParser(TEST_OPTIONS));
55-
BuildOptions b = BuildOptions.of(TEST_OPTIONS, OptionsParser.newOptionsParser(TEST_OPTIONS));
53+
BuildOptions a =
54+
BuildOptions.of(BUILD_CONFIG_OPTIONS, OptionsParser.newOptionsParser(BUILD_CONFIG_OPTIONS));
55+
BuildOptions b =
56+
BuildOptions.of(BUILD_CONFIG_OPTIONS, OptionsParser.newOptionsParser(BUILD_CONFIG_OPTIONS));
5657
// The cache keys of the OptionSets must be equal even if these are
5758
// different objects, if they were created with the same options (no options in this case).
5859
assertThat(b.toString()).isEqualTo(a.toString());
@@ -62,26 +63,27 @@ public void optionSetCaching() {
6263
@Test
6364
public void cachingSpecialCases() throws Exception {
6465
// You can add options here to test that their string representations are good.
65-
String[] options = new String[] { "--run_under=//run_under" };
66-
BuildOptions a = BuildOptions.of(TEST_OPTIONS, options);
67-
BuildOptions b = BuildOptions.of(TEST_OPTIONS, options);
66+
String[] options = new String[] {"--run_under=//run_under"};
67+
BuildOptions a = BuildOptions.of(BUILD_CONFIG_OPTIONS, options);
68+
BuildOptions b = BuildOptions.of(BUILD_CONFIG_OPTIONS, options);
6869
assertThat(b.toString()).isEqualTo(a.toString());
6970
}
7071

7172
@Test
7273
public void optionsEquality() throws Exception {
73-
String[] options1 = new String[] { "--compilation_mode=opt" };
74-
String[] options2 = new String[] { "--compilation_mode=dbg" };
74+
String[] options1 = new String[] {"--compilation_mode=opt"};
75+
String[] options2 = new String[] {"--compilation_mode=dbg"};
7576
// Distinct instances with the same values are equal:
76-
assertThat(BuildOptions.of(TEST_OPTIONS, options1))
77-
.isEqualTo(BuildOptions.of(TEST_OPTIONS, options1));
77+
assertThat(BuildOptions.of(BUILD_CONFIG_OPTIONS, options1))
78+
.isEqualTo(BuildOptions.of(BUILD_CONFIG_OPTIONS, options1));
7879
// Same fragments, different values aren't equal:
7980
assertThat(
80-
BuildOptions.of(TEST_OPTIONS, options1).equals(BuildOptions.of(TEST_OPTIONS, options2)))
81+
BuildOptions.of(BUILD_CONFIG_OPTIONS, options1)
82+
.equals(BuildOptions.of(BUILD_CONFIG_OPTIONS, options2)))
8183
.isFalse();
8284
// Same values, different fragments aren't equal:
8385
assertThat(
84-
BuildOptions.of(TEST_OPTIONS, options1)
86+
BuildOptions.of(BUILD_CONFIG_OPTIONS, options1)
8587
.equals(
8688
BuildOptions.of(
8789
ImmutableList.<Class<? extends FragmentOptions>>of(
@@ -92,9 +94,9 @@ public void optionsEquality() throws Exception {
9294

9395
@Test
9496
public void optionsDiff() throws Exception {
95-
BuildOptions one = BuildOptions.of(TEST_OPTIONS, "--compilation_mode=opt", "cpu=k8");
96-
BuildOptions two = BuildOptions.of(TEST_OPTIONS, "--compilation_mode=dbg", "cpu=k8");
97-
BuildOptions three = BuildOptions.of(TEST_OPTIONS, "--compilation_mode=dbg", "cpu=k8");
97+
BuildOptions one = BuildOptions.of(BUILD_CONFIG_OPTIONS, "--compilation_mode=opt", "cpu=k8");
98+
BuildOptions two = BuildOptions.of(BUILD_CONFIG_OPTIONS, "--compilation_mode=dbg", "cpu=k8");
99+
BuildOptions three = BuildOptions.of(BUILD_CONFIG_OPTIONS, "--compilation_mode=dbg", "cpu=k8");
98100

99101
OptionsDiff diffOneTwo = BuildOptions.diff(one, two);
100102
OptionsDiff diffTwoThree = BuildOptions.diff(two, three);
@@ -111,23 +113,22 @@ public void optionsDiff() throws Exception {
111113
public void optionsDiff_differentFragments() throws Exception {
112114
BuildOptions one =
113115
BuildOptions.of(ImmutableList.<Class<? extends FragmentOptions>>of(CppOptions.class));
114-
BuildOptions two = BuildOptions.of(TEST_OPTIONS);
116+
BuildOptions two = BuildOptions.of(BUILD_CONFIG_OPTIONS);
115117

116118
OptionsDiff diff = BuildOptions.diff(one, two);
117119

118120
assertThat(diff.areSame()).isFalse();
119121
assertThat(diff.getExtraFirstFragmentClassesForTesting()).containsExactly(CppOptions.class);
120122
assertThat(
121-
diff.getExtraSecondFragmentsForTesting()
122-
.stream()
123+
diff.getExtraSecondFragmentsForTesting().stream()
123124
.map(Object::getClass)
124125
.collect(Collectors.toSet()))
125-
.containsExactlyElementsIn(TEST_OPTIONS);
126+
.containsExactlyElementsIn(BUILD_CONFIG_OPTIONS);
126127
}
127128

128129
@Test
129130
public void optionsDiff_nullOptionsThrow() throws Exception {
130-
BuildOptions one = BuildOptions.of(TEST_OPTIONS, "--compilation_mode=opt", "cpu=k8");
131+
BuildOptions one = BuildOptions.of(BUILD_CONFIG_OPTIONS, "--compilation_mode=opt", "cpu=k8");
131132
BuildOptions two = null;
132133
IllegalArgumentException e =
133134
assertThrows(IllegalArgumentException.class, () -> BuildOptions.diff(one, two));
@@ -149,8 +150,8 @@ public void optionsDiff_nullSecondValue() throws Exception {
149150

150151
@Test
151152
public void optionsDiff_differentBaseThrowException() throws Exception {
152-
BuildOptions one = BuildOptions.of(TEST_OPTIONS, "--compilation_mode=opt", "cpu=k8");
153-
BuildOptions two = BuildOptions.of(TEST_OPTIONS, "--compilation_mode=dbg", "cpu=k8");
153+
BuildOptions one = BuildOptions.of(BUILD_CONFIG_OPTIONS, "--compilation_mode=opt", "cpu=k8");
154+
BuildOptions two = BuildOptions.of(BUILD_CONFIG_OPTIONS, "--compilation_mode=dbg", "cpu=k8");
154155
BuildOptions three = BuildOptions.of(ImmutableList.of(CppOptions.class), "--compiler=gcc");
155156
OptionsDiffForReconstruction diffForReconstruction =
156157
BuildOptions.diffForReconstruction(one, two);
@@ -163,8 +164,8 @@ public void optionsDiff_differentBaseThrowException() throws Exception {
163164

164165
@Test
165166
public void optionsDiff_getEmptyAndApplyEmpty() throws Exception {
166-
BuildOptions one = BuildOptions.of(TEST_OPTIONS, "--compilation_mode=opt", "cpu=k8");
167-
BuildOptions two = BuildOptions.of(TEST_OPTIONS, "--compilation_mode=opt", "cpu=k8");
167+
BuildOptions one = BuildOptions.of(BUILD_CONFIG_OPTIONS, "--compilation_mode=opt", "cpu=k8");
168+
BuildOptions two = BuildOptions.of(BUILD_CONFIG_OPTIONS, "--compilation_mode=opt", "cpu=k8");
168169
OptionsDiffForReconstruction diffForReconstruction =
169170
BuildOptions.diffForReconstruction(one, two);
170171
BuildOptions reconstructed = one.applyDiff(diffForReconstruction);
@@ -173,8 +174,8 @@ public void optionsDiff_getEmptyAndApplyEmpty() throws Exception {
173174

174175
@Test
175176
public void applyDiff_nativeOptions() throws Exception {
176-
BuildOptions one = BuildOptions.of(TEST_OPTIONS, "--compilation_mode=opt", "cpu=k8");
177-
BuildOptions two = BuildOptions.of(TEST_OPTIONS, "--compilation_mode=dbg", "cpu=k8");
177+
BuildOptions one = BuildOptions.of(BUILD_CONFIG_OPTIONS, "--compilation_mode=opt", "cpu=k8");
178+
BuildOptions two = BuildOptions.of(BUILD_CONFIG_OPTIONS, "--compilation_mode=dbg", "cpu=k8");
178179
BuildOptions reconstructedTwo = one.applyDiff(BuildOptions.diffForReconstruction(one, two));
179180
assertThat(reconstructedTwo).isEqualTo(two);
180181
assertThat(reconstructedTwo).isNotSameAs(two);
@@ -271,14 +272,14 @@ public void applyDiff_extraStarlarkOptions() throws Exception {
271272
BuildOptions two = BuildOptions.of(ImmutableMap.of(flagNameTwo, flagValue));
272273

273274
BuildOptions reconstructedTwo = one.applyDiff(BuildOptions.diffForReconstruction(one, two));
274-
275+
275276
assertThat(reconstructedTwo).isEqualTo(two);
276277
assertThat(reconstructedTwo).isNotSameAs(two);
277278
}
278279

279280
private static ImmutableList.Builder<Class<? extends FragmentOptions>> makeOptionsClassBuilder() {
280281
return ImmutableList.<Class<? extends FragmentOptions>>builder()
281-
.addAll(TEST_OPTIONS)
282+
.addAll(BUILD_CONFIG_OPTIONS)
282283
.add(CppOptions.class);
283284
}
284285

@@ -322,8 +323,8 @@ public void codecStability() throws Exception {
322323

323324
@Test
324325
public void repeatedCodec() throws Exception {
325-
BuildOptions one = BuildOptions.of(TEST_OPTIONS, "--compilation_mode=opt", "cpu=k8");
326-
BuildOptions two = BuildOptions.of(TEST_OPTIONS, "--compilation_mode=dbg", "cpu=k8");
326+
BuildOptions one = BuildOptions.of(BUILD_CONFIG_OPTIONS, "--compilation_mode=opt", "cpu=k8");
327+
BuildOptions two = BuildOptions.of(BUILD_CONFIG_OPTIONS, "--compilation_mode=dbg", "cpu=k8");
327328
OptionsDiffForReconstruction diff = BuildOptions.diffForReconstruction(one, two);
328329
BuildOptions.OptionsDiffCache cache = new BuildOptions.FingerprintingKDiffToByteStringCache();
329330
assertThat(TestUtils.toBytes(diff, ImmutableMap.of(BuildOptions.OptionsDiffCache.class, cache)))
@@ -334,11 +335,56 @@ public void repeatedCodec() throws Exception {
334335
@Test
335336
public void testMultiValueOptionImmutability() throws Exception {
336337
BuildOptions options =
337-
BuildOptions.of(TEST_OPTIONS, OptionsParser.newOptionsParser(TEST_OPTIONS));
338+
BuildOptions.of(BUILD_CONFIG_OPTIONS, OptionsParser.newOptionsParser(BUILD_CONFIG_OPTIONS));
338339
BuildConfiguration.Options coreOptions = options.get(Options.class);
339340
assertThrows(
340341
UnsupportedOperationException.class,
341342
() ->
342343
coreOptions.commandLineBuildVariables.add(new AbstractMap.SimpleEntry<>("foo", "bar")));
343344
}
345+
346+
@Test
347+
public void parsingResultTransform() throws Exception {
348+
BuildOptions original = BuildOptions.of(BUILD_CONFIG_OPTIONS, "--cpu=foo");
349+
350+
OptionsParser parser = OptionsParser.newOptionsParser(BUILD_CONFIG_OPTIONS);
351+
parser.parse("--cpu=bar");
352+
parser.setStarlarkOptions(ImmutableMap.of("//custom:flag", "hello"));
353+
354+
BuildOptions modified = original.applyParsingResult(parser);
355+
356+
assertThat(original.get(BuildConfiguration.Options.class).cpu)
357+
.isNotEqualTo(modified.get(BuildConfiguration.Options.class).cpu);
358+
assertThat(modified.get(BuildConfiguration.Options.class).cpu).isEqualTo("bar");
359+
assertThat(modified.getStarlarkOptions().get(Label.parseAbsoluteUnchecked("//custom:flag")))
360+
.isEqualTo("hello");
361+
}
362+
363+
@Test
364+
public void parsingResultTransformNativeIgnored() throws Exception {
365+
ImmutableList.Builder<Class<? extends FragmentOptions>> fragmentClassesBuilder =
366+
ImmutableList.<Class<? extends FragmentOptions>>builder()
367+
.add(BuildConfiguration.Options.class);
368+
369+
BuildOptions original = BuildOptions.of(fragmentClassesBuilder.build());
370+
371+
fragmentClassesBuilder.add(CppOptions.class);
372+
373+
OptionsParser parser = OptionsParser.newOptionsParser(fragmentClassesBuilder.build());
374+
parser.parse("--cxxopt=bar");
375+
376+
BuildOptions modified = original.applyParsingResult(parser);
377+
378+
assertThat(modified.contains(CppOptions.class)).isFalse();
379+
}
380+
381+
@Test
382+
public void parsingResultTransformIllegalStarlarkLabel() throws Exception {
383+
BuildOptions original = BuildOptions.of(BUILD_CONFIG_OPTIONS);
384+
385+
OptionsParser parser = OptionsParser.newOptionsParser(BUILD_CONFIG_OPTIONS);
386+
parser.setStarlarkOptions(ImmutableMap.of("@@@", "hello"));
387+
388+
assertThrows(IllegalArgumentException.class, () -> original.applyParsingResult(parser));
389+
}
344390
}

0 commit comments

Comments
 (0)