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}