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.compilers; 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.diagnostics.TraceDiagnostic; 022import io.github.ascopes.jct.filemanagers.JctFileManager; 023import java.util.List; 024import java.util.Set; 025import javax.tools.JavaFileObject; 026import javax.tools.StandardLocation; 027import org.jspecify.annotations.Nullable; 028 029/** 030 * The result of a compilation. 031 * 032 * <p>This provides access to a number of useful pieces of information 033 * including the file manager used for the compilation, compiler logs, 034 * and diagnostics. 035 * 036 * @author Ashley Scopes 037 * @since 0.0.1 038 */ 039public interface JctCompilation { 040 041 /** 042 * Get the command line arguments that were passed to the compiler. 043 * 044 * @return the command line arguments. 045 * @since 0.5.0 046 */ 047 List<String> getArguments(); 048 049 /** 050 * Determine if warnings were treated as errors. 051 * 052 * @return {@code true} if warnings were treated as errors, or {@code false} otherwise. 053 */ 054 boolean isFailOnWarnings(); 055 056 /** 057 * Determine if the compilation was successful or not. 058 * 059 * @return {@code true} if successful, or {@code false} if not successful. 060 */ 061 boolean isSuccessful(); 062 063 /** 064 * Determine if the compilation was a failure or not. 065 * 066 * @return {@code true} if not successful, or {@code false} if successful. 067 */ 068 default boolean isFailure() { 069 return !isSuccessful(); 070 } 071 072 /** 073 * Get the lines of output produced by the compiler, if any were captured. 074 * 075 * <p>This is separate to diagnostics. 076 * 077 * @return the lines of output. 078 */ 079 List<String> getOutputLines(); 080 081 /** 082 * Get the compilation units used in the compilation. 083 * 084 * @return the compilation units. 085 */ 086 Set<JavaFileObject> getCompilationUnits(); 087 088 /** 089 * Get the diagnostics that were reported by the compilation, in the order that they 090 * were reported. 091 * 092 * @return the diagnostics 093 */ 094 List<TraceDiagnostic<JavaFileObject>> getDiagnostics(); 095 096 /** 097 * Get the file manager that was used to store and manage files. 098 * 099 * <p>This can be used to obtain a classloader for any compiled sources, 100 * which can then be used to reflectively test what the compiler produced. 101 * 102 * @return the file manager. 103 */ 104 JctFileManager getFileManager(); 105 106 /** 107 * Get the output container group for class outputs. 108 * 109 * <p>This usually consists of any {@code *.class} files produced by the compiler, 110 * and is equivalent to {@code target/classes} in a Maven project. 111 * 112 * @return the output container group, or {@code null} if it does not exist. 113 */ 114 @Nullable 115 default OutputContainerGroup getClassOutputs() { 116 return getFileManager().getOutputContainerGroup(StandardLocation.CLASS_OUTPUT); 117 } 118 119 /** 120 * Get the output container group for source outputs. 121 * 122 * <p>This consists of any generated source code created by annotation processors, 123 * and is equivalent to {@code target/generated-sources} in a Maven project. 124 * 125 * @return the output container group, or {@code null} if it does not exist. 126 */ 127 @Nullable 128 default OutputContainerGroup getSourceOutputs() { 129 return getFileManager().getOutputContainerGroup(StandardLocation.SOURCE_OUTPUT); 130 } 131 132 /** 133 * Get the package container group for the class path. 134 * 135 * <p>This represents the class path used for compilation. 136 * 137 * @return the package container group, or {@code null} if it does not exist. 138 */ 139 @Nullable 140 default PackageContainerGroup getClassPath() { 141 return getFileManager().getPackageContainerGroup(StandardLocation.CLASS_PATH); 142 } 143 144 /** 145 * Get the package container group for the source path. 146 * 147 * <p>This is equivalent to {@code src/main/java} and {@code src/main/resources} 148 * in a Maven project. 149 * 150 * @return the package container group, or {@code null} if it does not exist. 151 */ 152 @Nullable 153 default PackageContainerGroup getSourcePath() { 154 return getFileManager().getPackageContainerGroup(StandardLocation.SOURCE_PATH); 155 } 156 157 /** 158 * Get the package container group for the annotation processor path. 159 * 160 * @return the package container group, or {@code null} if it does not exist. 161 */ 162 @Nullable 163 default PackageContainerGroup getAnnotationProcessorPath() { 164 return getFileManager().getPackageContainerGroup(StandardLocation.ANNOTATION_PROCESSOR_PATH); 165 } 166 167 /** 168 * Get the module container group for the annotation processor module path. 169 * 170 * @return the module container group, or {@code null} if it does not exist. 171 */ 172 @Nullable 173 default ModuleContainerGroup getAnnotationProcessorModulePath() { 174 return getFileManager() 175 .getModuleContainerGroup(StandardLocation.ANNOTATION_PROCESSOR_MODULE_PATH); 176 } 177 178 /** 179 * Get the package container group for the platform class path (a.k.a. the bootstrap class path). 180 * 181 * <p>You generally do not need to use this. The platform class path mechanism has been mostly 182 * replaced by the use of the system modules path as of Java 11. It is simply provided for 183 * backwards compatibility. 184 * 185 * @return the package container group, or {@code null} if it does not exist. 186 */ 187 @Nullable 188 default PackageContainerGroup getPlatformClassPath() { 189 return getFileManager().getPackageContainerGroup(StandardLocation.PLATFORM_CLASS_PATH); 190 } 191 192 /** 193 * Get the output container group for the native header file outputs. 194 * 195 * <p>If you invoke {@code javac} with the {@code -H} flag, then this represents the 196 * directory that C/C++ header file stubs for JNI are written to. 197 * 198 * @return the output container group, or {@code null} if it does not exist. 199 */ 200 @Nullable 201 default OutputContainerGroup getNativeHeaderOutputs() { 202 return getFileManager().getOutputContainerGroup(StandardLocation.NATIVE_HEADER_OUTPUT); 203 } 204 205 /** 206 * Get the module container group for the module source path. 207 * 208 * <p>Many build tools do not provide a direct equivalent of this mechanism as of now, but 209 * this is a source path introduced in Java 9 that allows specifying multiple named JPMS 210 * modules to compile under a single compilation invocation. 211 * 212 * <p>For example, you may use this in a project to compile an API module, a default 213 * implementation module, and a module containing unit tests all together. 214 * 215 * @return the module container group, or {@code null} if it does not exist. 216 */ 217 @Nullable 218 default ModuleContainerGroup getModuleSourcePath() { 219 return getFileManager().getModuleContainerGroup(StandardLocation.MODULE_SOURCE_PATH); 220 } 221 222 /** 223 * Get the module container group for the upgrade module path. 224 * 225 * <p>You generally will not need to use this, as this is a mechanism used to upgrade 226 * modules in-place incrementally with fixes without redistributing the entire application. 227 * 228 * @return the module container group, or {@code null} if it does not exist. 229 */ 230 @Nullable 231 default ModuleContainerGroup getUpgradeModulePath() { 232 return getFileManager().getModuleContainerGroup(StandardLocation.UPGRADE_MODULE_PATH); 233 } 234 235 /** 236 * Get the module container group for all system modules that are part of the JDK distribution. 237 * 238 * <p>This will usually just point to the Java standard library. 239 * 240 * @return the module container group, or {@code null} if it does not exist. 241 */ 242 @Nullable 243 default ModuleContainerGroup getSystemModules() { 244 return getFileManager().getModuleContainerGroup(StandardLocation.SYSTEM_MODULES); 245 } 246 247 /** 248 * Get the module container group for the module path. 249 * 250 * <p>This is equivalent to the class path, but holds any JPMS modules. 251 * 252 * @return the module container group, or {@code null} if it does not exist. 253 */ 254 @Nullable 255 default ModuleContainerGroup getModulePath() { 256 return getFileManager().getModuleContainerGroup(StandardLocation.MODULE_PATH); 257 } 258 259 /** 260 * Get the module container group for the patch module path. 261 * 262 * <p>You generally will not need to use this. It consists of patchable module sources that 263 * can be used to inject additional classes into a module. This can be used for cases like 264 * unit tests where you wish to embed the unit test classes into the existing application 265 * module to exploit features such as package private access. 266 * 267 * @return the module container group, or {@code null} if it does not exist. 268 */ 269 @Nullable 270 default ModuleContainerGroup getPatchModulePath() { 271 return getFileManager().getModuleContainerGroup(StandardLocation.PATCH_MODULE_PATH); 272 } 273}