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.filemanagers; 017 018import io.github.ascopes.jct.containers.ModuleContainerGroup; 019import io.github.ascopes.jct.containers.OutputContainerGroup; 020import io.github.ascopes.jct.containers.PackageContainerGroup; 021import io.github.ascopes.jct.workspaces.PathRoot; 022import io.github.ascopes.jct.workspaces.Workspace; 023import java.io.IOException; 024import java.util.Collection; 025import java.util.Iterator; 026import java.util.Set; 027import javax.tools.FileObject; 028import javax.tools.JavaFileManager; 029import javax.tools.JavaFileObject; 030import javax.tools.JavaFileObject.Kind; 031import javax.tools.StandardLocation; 032import org.jspecify.annotations.Nullable; 033 034/** 035 * Extension around a {@link JavaFileManager} that allows adding of {@link PathRoot} objects to the 036 * manager. 037 * 038 * <p>This component is responsible for bridging the gap between a {@link Workspace} and 039 * a {@link javax.tools.JavaCompiler} when performing a compilation, and thus includes a number of 040 * required operations that the compiler will query the file system with. In addition, this 041 * interface also defines a number of additional functionalities that are useful for querying and 042 * verifying the outcome of compilations within tests. 043 * 044 * @author Ashley Scopes 045 * @since 0.0.1 046 */ 047public interface JctFileManager extends JavaFileManager { 048 049 /** 050 * Add a package-oriented path to a given location. 051 * 052 * <p>To add a module, first obtain the module location using 053 * {@link #getLocationForModule(Location, String)}, and pass that result to this call. 054 * 055 * @param location the location to use. 056 * @param path the path to add. 057 * @see #getLocationForModule(Location, String) 058 */ 059 void addPath(Location location, PathRoot path); 060 061 /** 062 * Add a collection of package-oriented paths to a given location. 063 * 064 * <p>To add a module, first obtain the module location using 065 * {@link #getLocationForModule(Location, String)}, and pass that result to this call. 066 * 067 * @param location the location to use. 068 * @param paths the paths to add. 069 * @see #getLocationForModule(Location, String) 070 */ 071 void addPaths(Location location, Collection<? extends PathRoot> paths); 072 073 /** 074 * Close this file manager. 075 * 076 * <p>If the file manager is already closed, this will have no effect. 077 * 078 * <p>JCT File managers may choose to not implement this operation if not applicable. 079 * 080 * @throws IOException if an IO error occurs during closure. 081 */ 082 @Override 083 void close() throws IOException; 084 085 /** 086 * Determine if the given file object resides in the given location. 087 * 088 * <p>If package-oriented, then the file must exist in the given location directly. 089 * 090 * <p>If module-oriented, then the file must exist in one of the modules within the location. 091 * This implies that {@link #getLocationForModule(Location, JavaFileObject)} would also return 092 * a valid non-null value. 093 * 094 * @param location the location. 095 * @param fileObject the file object. 096 * @return {@code true} if the file exists in the given location, or {@code false} otherwise. 097 * @throws IOException if an IO error occurs during the lookup. 098 */ 099 @Override 100 boolean contains(Location location, FileObject fileObject) throws IOException; 101 102 /** 103 * Copy all containers from the first location to the second location. 104 * 105 * @param from the first location. 106 * @param to the second location. 107 */ 108 void copyContainers(Location from, Location to); 109 110 /** 111 * Register an empty container for the given location to indicate to the compiler that the feature 112 * exists, but has no configured paths. 113 * 114 * <p>This is needed to coerce the behaviour for annotation processing in some cases. 115 * 116 * <p>If the location already exists, then do not do anything. 117 * 118 * <p>If the location is an output location, then this operation does not make any sense, since 119 * an empty location cannot have files output to it. In this case, you will likely get an 120 * exception. 121 * 122 * <p>Likewise, this operation does not make sense for module locations within a module-oriented 123 * location group, so this operation will fail with an error for those inputs as well. 124 * 125 * @param location the location to apply an empty container for. 126 * @throws IllegalArgumentException if the location is an output location or a module location. 127 */ 128 void createEmptyLocation(Location location); 129 130 /** 131 * Flush this file manager. 132 * 133 * <p>If the file manager is already closed, this will have no effect. 134 * 135 * <p>JCT File managers may choose to not implement this operation if not applicable. 136 * 137 * @throws IOException if an IO error occurs during flushing. 138 */ 139 @Override 140 void flush() throws IOException; 141 142 /** 143 * Get the class loader for loading classes for the given location. 144 * 145 * @param location the location to fetch a class loader for. 146 * @return a class loader, or {@code null} if loading classes for the given location is not 147 * supported. 148 * @throws SecurityException if a class loader can not be created in the current security 149 * context 150 * @throws IllegalStateException if {@link #close} has been called and this file manager cannot 151 * be reopened 152 * @throws IllegalArgumentException if the location is a module-oriented location 153 */ 154 @Nullable 155 @Override 156 ClassLoader getClassLoader(Location location); 157 158 /** 159 * Get the associated effective release. 160 * 161 * @return the effective release. 162 * @since 0.0.1 163 */ 164 String getEffectiveRelease(); 165 166 /** 167 * Get a file for input operations. 168 * 169 * @param location a package-oriented location. 170 * @param packageName a package name. 171 * @param relativeName a relative name. 172 * @return the file object, or {@code null} if not found or not available. 173 * @throws IOException if an IO error occurs reading the file details. 174 * @throws IllegalArgumentException if the location is not a package-oriented location. 175 */ 176 @Nullable 177 @Override 178 FileObject getFileForInput( 179 Location location, 180 String packageName, 181 String relativeName 182 ) throws IOException; 183 184 /** 185 * Get a file for output operations. 186 * 187 * @param location an output location. 188 * @param packageName a package name. 189 * @param relativeName a relative name. 190 * @param sibling a file object to be used as hint for placement, might be {@code null}. 191 * @return the file object, or {@code null} if not found or not available. 192 * @throws IOException if an IO error occurs reading the file details. 193 * @throws IllegalArgumentException if the location is not an output location. 194 */ 195 @Nullable 196 @Override 197 FileObject getFileForOutput( 198 Location location, 199 String packageName, 200 String relativeName, 201 @Nullable FileObject sibling 202 ) throws IOException; 203 204 /** 205 * Get a Java file for input operations. 206 * 207 * @param location a package-oriented location. 208 * @param className the name of a class. 209 * @param kind the kind of file. 210 * @return the file object, or {@code null} if not found or not available. 211 * @throws IOException if an IO error occurs reading the file details. 212 * @throws IllegalArgumentException if the location is not a package-oriented location. 213 */ 214 @Nullable 215 @Override 216 JavaFileObject getJavaFileForInput( 217 Location location, 218 String className, 219 Kind kind 220 ) throws IOException; 221 222 /** 223 * Get a Java file for output operations. 224 * 225 * @param location an output location. 226 * @param className the name of a class. 227 * @param kind the kind of file. 228 * @param sibling a file object to be used as hint for placement, might be {@code null}. 229 * @return the file object, or {@code null} if not found or not available. 230 * @throws IOException if an IO error occurs reading the file details. 231 * @throws IllegalArgumentException if the location is not an output location. 232 */ 233 @Nullable 234 @Override 235 JavaFileObject getJavaFileForOutput( 236 Location location, 237 String className, 238 Kind kind, 239 @Nullable FileObject sibling 240 ) throws IOException; 241 242 /** 243 * Get the location for a named module within the given location. 244 * 245 * @param location the module-oriented location. 246 * @param moduleName the name of the module to be found. 247 * @return the location of the named module, or {@code null} if not found. 248 * @throws IOException if an IO error occurs during resolution. 249 * @throws IllegalArgumentException if the location is neither an output location nor a 250 * module-oriented location. 251 */ 252 @Nullable 253 @Override 254 Location getLocationForModule(Location location, String moduleName) throws IOException; 255 256 /** 257 * Get the location for the module holding a given file object within the given location. 258 * 259 * @param location the module-oriented location. 260 * @param fileObject the file object to resolve the module for. 261 * @return the location, or {@code null} if not found. 262 * @throws IOException if an IO error occurs during resolution. 263 * @throws IllegalArgumentException if the location is neither an output location nor a 264 * module-oriented location. 265 */ 266 @Nullable 267 @Override 268 Location getLocationForModule(Location location, JavaFileObject fileObject) throws IOException; 269 270 /** 271 * Get the container group for the given package-oriented location. 272 * 273 * @param location the package oriented location. 274 * @return the container group, or null if one does not exist. 275 */ 276 @Nullable 277 PackageContainerGroup getPackageContainerGroup(Location location); 278 279 /** 280 * Get a collection of all package container groups in this file manager. 281 * 282 * @return the package container groups. 283 */ 284 Collection<PackageContainerGroup> getPackageContainerGroups(); 285 286 /** 287 * Get the container group for the given module-oriented location. 288 * 289 * @param location the module oriented location. 290 * @return the container group, or null if one does not exist. 291 */ 292 @Nullable 293 ModuleContainerGroup getModuleContainerGroup(Location location); 294 295 /** 296 * Get a collection of all module container groups in this file manager. 297 * 298 * @return the module container groups. 299 */ 300 Collection<ModuleContainerGroup> getModuleContainerGroups(); 301 302 /** 303 * Get the container group for the given output-oriented location. 304 * 305 * @param location the output oriented location. 306 * @return the container group, or null if one does not exist. 307 */ 308 @Nullable 309 OutputContainerGroup getOutputContainerGroup(Location location); 310 311 /** 312 * Get a collection of all output container groups in this file manager. 313 * 314 * @return the output container groups. 315 */ 316 Collection<OutputContainerGroup> getOutputContainerGroups(); 317 318 /** 319 * Handles one option. If {@code current} is an option to this file manager it will consume any 320 * arguments to that option from {@code remaining} and return true, otherwise return false. 321 * 322 * @param current current option. 323 * @param remaining remaining options. 324 * @return {@code true} if this option was handled by this file manager, {@code false} otherwise. 325 * @throws IllegalArgumentException if this option to this file manager is used incorrectly. 326 * @throws IllegalStateException if {@link #close} has been called and this file manager cannot 327 * be reopened. 328 */ 329 boolean handleOption(String current, Iterator<String> remaining); 330 331 /** 332 * Determines if a location is known to this file manager. 333 * 334 * @param location a location. 335 * @return true if the location is known to this file manager. 336 */ 337 boolean hasLocation(Location location); 338 339 /** 340 * Infers a binary name of a file object based on a package-oriented location. The binary name 341 * returned might not be a valid binary name according to the Java Language Specification. 342 * 343 * @param location a location. 344 * @param file a file object. 345 * @return a binary name or {@code null} the file object is not found in the given location 346 * @throws IllegalArgumentException if the location is a module-oriented location. 347 * @throws IllegalStateException if {@link #close} has been called and this file manager cannot 348 * be reopened. 349 */ 350 @Nullable 351 @Override 352 String inferBinaryName(Location location, JavaFileObject file); 353 354 /** 355 * Infer the name of the module from its location, as returned by {@code getLocationForModule} or 356 * {@code listModuleLocations}. 357 * 358 * @param location a package-oriented location representing a module. 359 * @return the name of the module, or {@code null} if the name cannot be resolved or does not 360 * exist. 361 * @throws IOException if an I/O error occurred during resolution. 362 * @throws IllegalArgumentException if the location is not one known to this file manager. 363 */ 364 @Nullable 365 @Override 366 String inferModuleName(Location location) throws IOException; 367 368 /** 369 * Determine if the two file objects are the same file. 370 * 371 * @param a a file object, can be {@code null}. 372 * @param b a file object, can be {@code null}. 373 * @return {@code true} if both arguments are non-null and refer to the same logical file, or 374 * {@code false} otherwise. 375 */ 376 @Override 377 boolean isSameFile(@Nullable FileObject a, @Nullable FileObject b); 378 379 /** 380 * List all file objects in the given location matching the given criteria. 381 * 382 * @param location the location to search in. 383 * @param packageName the package name to search in, or {@code ""} to search in the root 384 * location. 385 * @param kinds the kinds of file to return, or {@link Set#of} {@code (} {@link Kind#OTHER} 386 * {@code )} to find all files. 387 * @param recurse {@code true} to recurse into subpackages, {@code false} to only check the 388 * current package. 389 * @return a collection of unique file objects that were found. 390 * @throws IOException if an IO error occurs reading the file system. 391 * @throws IllegalArgumentException if the location is a module-oriented location 392 * @throws IllegalStateException if {@link #close} has been called and this file manager cannot 393 * be reopened 394 */ 395 @Override 396 Set<JavaFileObject> list( 397 Location location, 398 String packageName, 399 Set<Kind> kinds, 400 boolean recurse 401 ) throws IOException; 402 403 /** 404 * List all module-based locations for the given module-oriented or output location. 405 * 406 * @param location the module-oriented/output location for which to list the modules. 407 * @return the iterable of sets of modules. 408 * @throws IOException if an IO error occurs during resolution. 409 * @throws IllegalArgumentException if the location is not a module-oriented or output location. 410 */ 411 @Override 412 Iterable<Set<Location>> listLocationsForModules(Location location) throws IOException; 413 414 /// 415 /// Default helper overrides 416 /// 417 418 /** 419 * Get the location holding the {@link StandardLocation#CLASS_OUTPUT class outputs}. 420 * 421 * @return the location, or {@code null} if the location is not present in the file manager. 422 * @since 0.1.0 423 */ 424 @Nullable 425 default OutputContainerGroup getClassOutputGroup() { 426 return getOutputContainerGroup(StandardLocation.CLASS_OUTPUT); 427 } 428 429 /** 430 * Get the location holding the {@link StandardLocation#SOURCE_OUTPUT source outputs}. 431 * 432 * @return the location, or {@code null} if the location is not present in the file manager. 433 * @since 0.1.0 434 */ 435 @Nullable 436 default OutputContainerGroup getSourceOutputGroup() { 437 return getOutputContainerGroup(StandardLocation.SOURCE_OUTPUT); 438 } 439 440 /** 441 * Get the location holding the {@link StandardLocation#CLASS_PATH class path}. 442 * 443 * @return the location, or {@code null} if the location is not present in the file manager. 444 * @since 0.1.0 445 */ 446 @Nullable 447 default PackageContainerGroup getClassPathGroup() { 448 return getPackageContainerGroup(StandardLocation.CLASS_PATH); 449 } 450 451 /** 452 * Get the location holding the {@link StandardLocation#SOURCE_PATH source path}. 453 * 454 * @return the location, or {@code null} if the location is not present in the file manager. 455 * @since 0.1.0 456 */ 457 @Nullable 458 default PackageContainerGroup getSourcePathGroup() { 459 return getPackageContainerGroup(StandardLocation.SOURCE_PATH); 460 } 461 462 /** 463 * Get the location holding the 464 * {@link StandardLocation#ANNOTATION_PROCESSOR_PATH annotation processor path}. 465 * 466 * @return the location, or {@code null} if the location is not present in the file manager. 467 * @since 0.1.0 468 */ 469 @Nullable 470 default PackageContainerGroup getAnnotationProcessorPathGroup() { 471 return getPackageContainerGroup(StandardLocation.ANNOTATION_PROCESSOR_PATH); 472 } 473 474 /** 475 * Get the location holding the 476 * {@link StandardLocation#ANNOTATION_PROCESSOR_MODULE_PATH annotation processor module path}. 477 * 478 * @return the location, or {@code null} if the location is not present in the file manager. 479 * @since 0.1.0 480 */ 481 @Nullable 482 default ModuleContainerGroup getAnnotationProcessorModulePathGroup() { 483 return getModuleContainerGroup(StandardLocation.ANNOTATION_PROCESSOR_MODULE_PATH); 484 } 485 486 /** 487 * Get the location holding the {@link StandardLocation#MODULE_SOURCE_PATH module source path}. 488 * 489 * @return the location, or {@code null} if the location is not present in the file manager. 490 * @since 0.1.0 491 */ 492 @Nullable 493 default ModuleContainerGroup getModuleSourcePathGroup() { 494 return getModuleContainerGroup(StandardLocation.MODULE_SOURCE_PATH); 495 } 496 497 /** 498 * Get the location holding the {@link StandardLocation#MODULE_PATH module path}. 499 * 500 * @return the location, or {@code null} if the location is not present in the file manager. 501 * @since 0.1.0 502 */ 503 @Nullable 504 default ModuleContainerGroup getModulePathGroup() { 505 return getModuleContainerGroup(StandardLocation.MODULE_PATH); 506 } 507}