001/*
002 * Copyright (C) 2022 - 2025, the original author or authors.
003 *
004 * Licensed under the Apache License, Version 2.0 (the "License");
005 * you may not use this file except in compliance with the License.
006 * You may obtain a copy of the License at
007 *
008 *    http://www.apache.org/licenses/LICENSE-2.0
009 *
010 * Unless required by applicable law or agreed to in writing, software
011 * distributed under the License is distributed on an "AS IS" BASIS,
012 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
013 * See the License for the specific language governing permissions and
014 * limitations under the License.
015 */
016package io.github.ascopes.jct.assertions;
017
018import static java.util.Objects.requireNonNull;
019
020import io.github.ascopes.jct.compilers.JctCompilation;
021import io.github.ascopes.jct.containers.ContainerGroup;
022import io.github.ascopes.jct.repr.TraceDiagnosticListRepresentation;
023import java.util.Collection;
024import javax.tools.Diagnostic.Kind;
025import javax.tools.JavaFileManager.Location;
026import javax.tools.StandardLocation;
027import org.assertj.core.api.AbstractAssert;
028import org.assertj.core.api.StringAssert;
029import org.jspecify.annotations.Nullable;
030
031/**
032 * Assertions that apply to a {@link JctCompilation}.
033 *
034 * @author Ashley Scopes
035 * @since 0.0.1
036 */
037public final class JctCompilationAssert
038    extends AbstractAssert<JctCompilationAssert, JctCompilation> {
039
040  /**
041   * Initialize this compilation assertion.
042   *
043   * @param value the value to assert on.
044   */
045  @SuppressWarnings("DataFlowIssue")
046  public JctCompilationAssert(@Nullable JctCompilation value) {
047    super(value, JctCompilationAssert.class);
048  }
049
050  /**
051   * Assert that the arguments passed to the compiler were the expected values.
052   *
053   * @return a list assertion object to perform assertions on the arguments with.
054   * @throws AssertionError if the compilation was null.
055   */
056  public TypeAwareListAssert<String, StringAssert> arguments() {
057    isNotNull();
058
059    var arguments = actual.getArguments();
060
061    return new TypeAwareListAssert<>(arguments, StringAssert::new);
062  }
063
064  /**
065   * Assert that the compilation was successful.
066   *
067   * @return this assertion object.
068   * @throws AssertionError if the compilation was null, or if the compilation was not successful.
069   */
070  public JctCompilationAssert isSuccessful() {
071    isNotNull();
072
073    if (actual.isFailure()) {
074      throw failWithDiagnostics(
075          // If we have error diagnostics, add them to the error message to provide helpful
076          // debugging information. If we are treating warnings as errors, then we want to include
077          // those in this as well.
078          actual.isFailOnWarnings()
079              ? DiagnosticKindAssert.WARNING_AND_ERROR_DIAGNOSTIC_KINDS
080              : DiagnosticKindAssert.ERROR_DIAGNOSTIC_KINDS,
081          "Expected a successful compilation, but it failed."
082      );
083    }
084
085    return myself;
086  }
087
088  /**
089   * Assert that the compilation was successful and had no warnings.
090   *
091   * <p>If warnings were treated as errors by the compiler, then this is identical to calling
092   * {@link #isSuccessful()}.
093   *
094   * @return this assertion object.
095   * @throws AssertionError if the compilation was null, if the compilation was not successful, or
096   *                        if the compilation was successful but had one or more warning
097   *                        diagnostics.
098   */
099  public JctCompilationAssert isSuccessfulWithoutWarnings() {
100    isSuccessful();
101    diagnostics().hasNoErrorsOrWarnings();
102    return myself;
103  }
104
105  /**
106   * Assert that the compilation was a failure.
107   *
108   * @return this assertion object.
109   * @throws AssertionError if the compilation was null, or if the compilation succeeded.
110   */
111  public JctCompilationAssert isFailure() {
112    isNotNull();
113
114    // If we fail due to failOnWarnings, we expect the compiler itself to have failed the
115    // build because of this. If the compiler ignores this flag and succeeds, then this method will
116    // follow that behaviour and treat the compilation as a success.
117
118    if (actual.isSuccessful()) {
119      // If we have any warnings, we should show them in the error message as it might be useful
120      // to the user.
121      throw failWithDiagnostics(
122          DiagnosticKindAssert.WARNING_AND_ERROR_DIAGNOSTIC_KINDS,
123          "Expected compilation to fail, but it succeeded."
124      );
125    }
126
127    return myself;
128  }
129
130  /**
131   * Get assertions for diagnostics.
132   *
133   * @return assertions for the diagnostics.
134   * @throws AssertionError if the compilation was null.
135   */
136  public TraceDiagnosticListAssert diagnostics() {
137    isNotNull();
138    return new TraceDiagnosticListAssert(actual.getDiagnostics());
139  }
140
141  /**
142   * Perform assertions on the given package group, if it has been configured.
143   *
144   * <p>If not configured, this will return assertions on a {@code null} value instead.
145   *
146   * @param location the location to configure.
147   * @return the assertions to perform.
148   * @throws AssertionError           if the compilation was null, or no group for the location was
149   *                                  found.
150   * @throws IllegalArgumentException if the location was
151   *                                  {@link Location#isModuleOrientedLocation() module-oriented} or
152   *                                  {@link Location#isOutputLocation() an output location}.
153   * @throws NullPointerException     if the provided location object is null.
154   */
155  public PackageContainerGroupAssert packageGroup(Location location) {
156    requireNonNull(location, "location must not be null");
157
158    if (location.isModuleOrientedLocation()) {
159      throw new IllegalArgumentException(
160          "Expected location " + location + " to not be module-oriented"
161      );
162    }
163
164    if (location.isOutputLocation()) {
165      throw new IllegalArgumentException(
166          "Expected location " + location + " to not be an output location"
167      );
168    }
169
170    isNotNull();
171
172    var group = actual.getFileManager().getPackageContainerGroup(location);
173    assertLocationExists(location, group);
174    return new PackageContainerGroupAssert(group);
175  }
176
177  /**
178   * Perform assertions on the given module group, if it has been configured.
179   *
180   * <p>If not configured, the value being asserted on will be {@code null} in value.
181   *
182   * @param location the location to configure.
183   * @return the assertions to perform.
184   * @throws AssertionError           if the compilation was null, or no group for the location was
185   *                                  found.
186   * @throws IllegalArgumentException if the location is not
187   *                                  {@link Location#isModuleOrientedLocation() module-oriented}.
188   * @throws NullPointerException     if the provided location object is null.
189   */
190  public ModuleContainerGroupAssert moduleGroup(Location location) {
191    requireNonNull(location, "location must not be null");
192
193    if (!location.isModuleOrientedLocation()) {
194      throw new IllegalArgumentException(
195          "Expected location " + location.getName() + " to be module-oriented"
196      );
197    }
198
199    if (location.isOutputLocation()) {
200      throw new IllegalArgumentException(
201          "Expected location " + location.getName() + " to not be an output location"
202      );
203    }
204
205    isNotNull();
206
207    var group = actual.getFileManager().getModuleContainerGroup(location);
208    assertLocationExists(location, group);
209    return new ModuleContainerGroupAssert(group);
210  }
211
212  /**
213   * Perform assertions on the given output group, if it has been configured.
214   *
215   * <p>If not configured, the value being asserted on will be {@code null} in value.
216   *
217   * @param location the location to configure.
218   * @return the assertions to perform.
219   * @throws AssertionError           if the compilation was null, or no group for the location was
220   *                                  found.
221   * @throws IllegalArgumentException if the location is not
222   *                                  {@link Location#isOutputLocation() an output location}.
223   * @throws NullPointerException     if the provided location object is null.
224   */
225  public OutputContainerGroupAssert outputGroup(Location location) {
226    requireNonNull(location, "location must not be null");
227
228    if (!location.isOutputLocation()) {
229      throw new IllegalArgumentException(
230          "Expected location " + location.getName() + " to be an output location"
231      );
232    }
233
234    isNotNull();
235
236    var group = actual.getFileManager().getOutputContainerGroup(location);
237    assertLocationExists(location, group);
238    return new OutputContainerGroupAssert(group);
239  }
240
241  /**
242   * Get assertions on the path containing class package outputs, if it exists.
243   *
244   * <p>If not configured, the value being asserted on will be {@code null} in value.
245   *
246   * @return the assertions to perform on the class package outputs.
247   * @throws AssertionError if the compilation was null, or no group for the location was found.
248   * @since 0.6.4
249   */
250  public PackageContainerGroupAssert classOutputPackages() {
251    return outputGroup(StandardLocation.CLASS_OUTPUT).packages();
252  }
253
254  /**
255   * Get assertions on the path containing class module outputs, if it exists.
256   *
257   * <p>If not configured, the value being asserted on will be {@code null} in value.
258   *
259   * @return the assertions to perform on the class module outputs.
260   * @throws AssertionError if the compilation was null, or no group for the location was found.
261   * @since 0.6.4
262   */
263  public ModuleContainerGroupAssert classOutputModules() {
264    return outputGroup(StandardLocation.CLASS_OUTPUT).modules();
265  }
266
267  /**
268   * Get assertions on the path containing generated source package outputs, if it exists.
269   *
270   * <p>If not configured, the value being asserted on will be {@code null} in value.
271   *
272   * @return the assertions to perform on the source package outputs.
273   * @throws AssertionError if the compilation was null, or no group for the location was found.
274   * @since 0.6.4
275   */
276  public PackageContainerGroupAssert sourceOutputPackages() {
277    return outputGroup(StandardLocation.SOURCE_OUTPUT).packages();
278  }
279
280  /**
281   * Get assertions on the path containing generated source module outputs, if it exists.
282   *
283   * <p>If not configured, the value being asserted on will be {@code null} in value.
284   *
285   * @return the assertions to perform on the source module outputs.
286   * @throws AssertionError if the compilation was null, or no group for the location was found.
287   * @since 0.6.4
288   */
289  public ModuleContainerGroupAssert sourceOutputModules() {
290    return outputGroup(StandardLocation.SOURCE_OUTPUT).modules();
291  }
292
293  /**
294   * Get assertions on the path containing the class path, if it exists.
295   *
296   * <p>If not configured, the value being asserted on will be {@code null} in value.
297   *
298   * @return the assertions to perform on the class path.
299   * @throws AssertionError if the compilation was null, or no group for the location was found.
300   * @since 0.6.4
301   */
302  public PackageContainerGroupAssert classPathPackages() {
303    return packageGroup(StandardLocation.CLASS_PATH);
304  }
305
306  /**
307   * Get assertions on the path containing the source path, if it exists.
308   *
309   * <p>If not configured, the value being asserted on will be {@code null} in value.
310   *
311   * @return the assertions to perform on the source path.
312   * @throws AssertionError if the compilation was null, or no group for the location was found.
313   * @since 0.6.4
314   */
315  public PackageContainerGroupAssert sourcePathPackages() {
316    return packageGroup(StandardLocation.SOURCE_PATH);
317  }
318
319  /**
320   * Get assertions on the path containing the source path, if it exists.
321   *
322   * <p>If not configured, the value being asserted on will be {@code null} in value.
323   *
324   * @return the assertions to perform on the source path.
325   * @throws AssertionError if the compilation was null, or no group for the location was found.
326   * @since 0.6.4
327   */
328  public ModuleContainerGroupAssert moduleSourcePathModules() {
329    return moduleGroup(StandardLocation.MODULE_SOURCE_PATH);
330  }
331
332  /**
333   * Get assertions on the path containing the module path, if it exists.
334   *
335   * <p>If not configured, the value being asserted on will be {@code null} in value.
336   *
337   * @return the assertions to perform on the module path.
338   * @throws AssertionError if the compilation was null, or no group for the location was found.
339   * @since 0.6.4
340   */
341  public ModuleContainerGroupAssert modulePathModules() {
342    return moduleGroup(StandardLocation.MODULE_PATH);
343  }
344
345  private AssertionError failWithDiagnostics(
346      Collection<? extends Kind> kindsToDisplay,
347      String message
348  ) {
349    var diagnostics = actual
350        .getDiagnostics()
351        .stream()
352        .filter(diagnostic -> kindsToDisplay.contains(diagnostic.getKind()))
353        .toList();
354
355    return failure(
356        "%s\n\nDiagnostics:\n%s",
357        message,
358        TraceDiagnosticListRepresentation.getInstance().toStringOf(diagnostics)
359    );
360  }
361
362  private void assertLocationExists(Location location, @Nullable ContainerGroup group) {
363    if (group == null) {
364      throw failure("No location named %s exists", location.getName());
365    }
366  }
367}