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 static io.github.ascopes.jct.utils.IterableUtils.requireNonNullValues;
019import static java.util.Objects.requireNonNull;
020
021import io.github.ascopes.jct.compilers.impl.JctCompilationFactoryImpl;
022import io.github.ascopes.jct.compilers.impl.JctCompilationImpl;
023import io.github.ascopes.jct.ex.JctCompilerException;
024import io.github.ascopes.jct.filemanagers.AnnotationProcessorDiscovery;
025import io.github.ascopes.jct.filemanagers.JctFileManagerFactory;
026import io.github.ascopes.jct.filemanagers.JctFileManagers;
027import io.github.ascopes.jct.filemanagers.LoggingMode;
028import io.github.ascopes.jct.workspaces.Workspace;
029import java.io.IOException;
030import java.nio.charset.Charset;
031import java.util.ArrayList;
032import java.util.Collection;
033import java.util.List;
034import java.util.Locale;
035import java.util.Set;
036import javax.annotation.processing.Processor;
037import org.jspecify.annotations.Nullable;
038
039/**
040 * Common functionality for a compiler that can be overridden and that produces a
041 * {@link JctCompilationImpl} as the compilation result.
042 *
043 * <p>Implementations should extend this class and override anything they require.
044 * In most cases, you should not need to override anything other than the constructor.
045 *
046 * <p>This class is <strong>not thread-safe</strong>.
047 *
048 * <p>If you wish to create a common set of configuration settings for instances of
049 * this class, you should consider writing a custom {@link JctCompilerConfigurer} object to apply
050 * the desired operations, and then apply it to instances of this class using
051 * {@link #configure(JctCompilerConfigurer)}.
052 *
053 * @author Ashley Scopes
054 * @since 0.0.1
055 */
056public abstract class AbstractJctCompiler implements JctCompiler {
057
058  private final List<Processor> annotationProcessors;
059  private final List<String> annotationProcessorOptions;
060  private final List<String> compilerOptions;
061
062  private String name;
063  private boolean showWarnings;
064  private boolean showDeprecationWarnings;
065  private boolean failOnWarnings;
066  private CompilationMode compilationMode;
067  private Locale locale;
068  private Charset logCharset;
069  private boolean verbose;
070  private boolean previewFeatures;
071  private LoggingMode diagnosticLoggingMode;
072  private boolean fixJvmModulePathMismatch;
073  private boolean inheritClassPath;
074  private boolean inheritModulePath;
075  private boolean inheritSystemModulePath;
076  private LoggingMode fileManagerLoggingMode;
077  private AnnotationProcessorDiscovery annotationProcessorDiscovery;
078  private Set<DebuggingInfo> debuggingInfo;
079  private boolean parameterInfoEnabled;
080
081  private @Nullable String release;
082  private @Nullable String source;
083  private @Nullable String target;
084
085  /**
086   * Initialize this compiler.
087   *
088   * @param defaultName the printable default name to use for the compiler.
089   */
090  protected AbstractJctCompiler(String defaultName) {
091    name = requireNonNull(defaultName, "name");
092
093    annotationProcessors = new ArrayList<>();
094    annotationProcessorOptions = new ArrayList<>();
095    compilerOptions = new ArrayList<>();
096    showWarnings = DEFAULT_SHOW_WARNINGS;
097    showDeprecationWarnings = DEFAULT_SHOW_DEPRECATION_WARNINGS;
098    failOnWarnings = DEFAULT_FAIL_ON_WARNINGS;
099    compilationMode = DEFAULT_COMPILATION_MODE;
100    locale = DEFAULT_LOCALE;
101    logCharset = DEFAULT_LOG_CHARSET;
102    previewFeatures = DEFAULT_PREVIEW_FEATURES;
103    verbose = DEFAULT_VERBOSE;
104    diagnosticLoggingMode = DEFAULT_DIAGNOSTIC_LOGGING_MODE;
105    fixJvmModulePathMismatch = DEFAULT_FIX_JVM_MODULE_PATH_MISMATCH;
106    inheritClassPath = DEFAULT_INHERIT_CLASS_PATH;
107    inheritModulePath = DEFAULT_INHERIT_MODULE_PATH;
108    inheritSystemModulePath = DEFAULT_INHERIT_SYSTEM_MODULE_PATH;
109    fileManagerLoggingMode = DEFAULT_FILE_MANAGER_LOGGING_MODE;
110    annotationProcessorDiscovery = DEFAULT_ANNOTATION_PROCESSOR_DISCOVERY;
111    debuggingInfo = DEFAULT_DEBUGGING_INFO;
112    parameterInfoEnabled = DEFAULT_PARAMETER_INFO_ENABLED;
113
114    // If none of these are overridden then we assume the defaults instead.
115    release = null;
116    source = null;
117    target = null;
118  }
119
120  @Override
121  public JctCompilation compile(Workspace workspace) {
122    return performCompilation(workspace, null);
123  }
124
125  @Override
126  public JctCompilation compile(Workspace workspace, Collection<String> classNames) {
127    // There is no reason to invoke this overload with null values, so
128    // prevent this.
129    requireNonNullValues(classNames, "classNames");
130    return performCompilation(workspace, classNames);
131  }
132
133  @Override
134  public final <E extends Exception> AbstractJctCompiler configure(
135      JctCompilerConfigurer<E> configurer
136  ) throws E {
137    requireNonNull(configurer, "configurer");
138    configurer.configure(this);
139    return this;
140  }
141
142  @Override
143  public String getName() {
144    return name;
145  }
146
147  @Override
148  public AbstractJctCompiler name(String name) {
149    requireNonNull(name, "name");
150    this.name = name;
151    return this;
152  }
153
154  @Override
155  public boolean isVerbose() {
156    return verbose;
157  }
158
159  @Override
160  public AbstractJctCompiler verbose(boolean enabled) {
161    verbose = enabled;
162    return this;
163  }
164
165  @Override
166  public boolean isPreviewFeatures() {
167    return previewFeatures;
168  }
169
170  @Override
171  public AbstractJctCompiler previewFeatures(boolean enabled) {
172    previewFeatures = enabled;
173    return this;
174  }
175
176  @Override
177  public boolean isShowWarnings() {
178    return showWarnings;
179  }
180
181  @Override
182  public AbstractJctCompiler showWarnings(boolean enabled) {
183    showWarnings = enabled;
184    return this;
185  }
186
187  @Override
188  public boolean isShowDeprecationWarnings() {
189    return showDeprecationWarnings;
190  }
191
192  @Override
193  public AbstractJctCompiler showDeprecationWarnings(boolean enabled) {
194    showDeprecationWarnings = enabled;
195    return this;
196  }
197
198  @Override
199  public boolean isFailOnWarnings() {
200    return failOnWarnings;
201  }
202
203  @Override
204  public AbstractJctCompiler failOnWarnings(boolean enabled) {
205    failOnWarnings = enabled;
206    return this;
207  }
208
209  @Override
210  public CompilationMode getCompilationMode() {
211    return compilationMode;
212  }
213
214  @Override
215  public AbstractJctCompiler compilationMode(CompilationMode compilationMode) {
216    this.compilationMode = compilationMode;
217    return this;
218  }
219
220  @Override
221  public List<String> getAnnotationProcessorOptions() {
222    return List.copyOf(annotationProcessorOptions);
223  }
224
225  @Override
226  public AbstractJctCompiler addAnnotationProcessorOptions(
227      Iterable<String> annotationProcessorOptions
228  ) {
229    requireNonNullValues(annotationProcessorOptions, "annotationProcessorOptions");
230    annotationProcessorOptions.forEach(this.annotationProcessorOptions::add);
231    return this;
232  }
233
234  @Override
235  public List<Processor> getAnnotationProcessors() {
236    return List.copyOf(annotationProcessors);
237  }
238
239  @Override
240  public AbstractJctCompiler addAnnotationProcessors(
241      Iterable<? extends Processor> annotationProcessors
242  ) {
243    requireNonNullValues(annotationProcessors, "annotationProcessors");
244    annotationProcessors.forEach(this.annotationProcessors::add);
245
246    return this;
247  }
248
249  @Override
250  public List<String> getCompilerOptions() {
251    return List.copyOf(compilerOptions);
252  }
253
254  @Override
255  public AbstractJctCompiler addCompilerOptions(Iterable<String> compilerOptions) {
256    requireNonNullValues(compilerOptions, "compilerOptions");
257    compilerOptions.forEach(this.compilerOptions::add);
258    return this;
259  }
260
261  @Override
262  public String getEffectiveRelease() {
263    if (release != null) {
264      return release;
265    }
266
267    if (target != null) {
268      return target;
269    }
270
271    return getDefaultRelease();
272  }
273
274  @Nullable
275  @Override
276  public String getRelease() {
277    return release;
278  }
279
280  @Override
281  public AbstractJctCompiler release(@Nullable String release) {
282    this.release = release;
283
284    if (release != null) {
285      source = null;
286      target = null;
287    }
288
289    return this;
290  }
291
292  @Nullable
293  @Override
294  public String getSource() {
295    return source;
296  }
297
298  @Override
299  public AbstractJctCompiler source(@Nullable String source) {
300    this.source = source;
301    if (source != null) {
302      release = null;
303    }
304    return this;
305  }
306
307  @Nullable
308  @Override
309  public String getTarget() {
310    return target;
311  }
312
313  @Override
314  public AbstractJctCompiler target(@Nullable String target) {
315    this.target = target;
316    if (target != null) {
317      release = null;
318    }
319    return this;
320  }
321
322  @Override
323  public boolean isFixJvmModulePathMismatch() {
324    return fixJvmModulePathMismatch;
325  }
326
327  @Override
328  public AbstractJctCompiler fixJvmModulePathMismatch(boolean fixJvmModulePathMismatch) {
329    this.fixJvmModulePathMismatch = fixJvmModulePathMismatch;
330    return this;
331  }
332
333  @Override
334  public boolean isInheritClassPath() {
335    return inheritClassPath;
336  }
337
338  @Override
339  public AbstractJctCompiler inheritClassPath(boolean inheritClassPath) {
340    this.inheritClassPath = inheritClassPath;
341    return this;
342  }
343
344  @Override
345  public boolean isInheritModulePath() {
346    return inheritModulePath;
347  }
348
349  @Override
350  public AbstractJctCompiler inheritModulePath(boolean inheritModulePath) {
351    this.inheritModulePath = inheritModulePath;
352    return this;
353  }
354
355  @Override
356  public boolean isInheritSystemModulePath() {
357    return inheritSystemModulePath;
358  }
359
360  @Override
361  public AbstractJctCompiler inheritSystemModulePath(boolean inheritSystemModulePath) {
362    this.inheritSystemModulePath = inheritSystemModulePath;
363    return this;
364  }
365
366  @Override
367  public Locale getLocale() {
368    return locale;
369  }
370
371  @Override
372  public AbstractJctCompiler locale(Locale locale) {
373    requireNonNull(locale, "locale");
374    this.locale = locale;
375    return this;
376  }
377
378  @Override
379  public Charset getLogCharset() {
380    return logCharset;
381  }
382
383  @Override
384  public AbstractJctCompiler logCharset(Charset logCharset) {
385    requireNonNull(logCharset, "logCharset");
386    this.logCharset = logCharset;
387    return this;
388  }
389
390  @Override
391  public LoggingMode getFileManagerLoggingMode() {
392    return fileManagerLoggingMode;
393  }
394
395  @Override
396  public AbstractJctCompiler fileManagerLoggingMode(LoggingMode fileManagerLoggingMode) {
397    requireNonNull(fileManagerLoggingMode, "fileManagerLoggingMode");
398    this.fileManagerLoggingMode = fileManagerLoggingMode;
399    return this;
400  }
401
402  @Override
403  public LoggingMode getDiagnosticLoggingMode() {
404    return diagnosticLoggingMode;
405  }
406
407  @Override
408  public AbstractJctCompiler diagnosticLoggingMode(LoggingMode diagnosticLoggingMode) {
409    requireNonNull(diagnosticLoggingMode, "diagnosticLoggingMode");
410    this.diagnosticLoggingMode = diagnosticLoggingMode;
411    return this;
412  }
413
414  @Override
415  public AnnotationProcessorDiscovery getAnnotationProcessorDiscovery() {
416    return annotationProcessorDiscovery;
417  }
418
419  @Override
420  public AbstractJctCompiler annotationProcessorDiscovery(
421      AnnotationProcessorDiscovery annotationProcessorDiscovery
422  ) {
423    requireNonNull(annotationProcessorDiscovery, 
424        "annotationProcessorDiscovery");
425    this.annotationProcessorDiscovery = annotationProcessorDiscovery;
426    return this;
427  }
428
429  @Override
430  public Set<DebuggingInfo> getDebuggingInfo() {
431    return debuggingInfo;
432  }
433
434  @Override
435  public JctCompiler debuggingInfo(Set<DebuggingInfo> debuggingInfo) {
436    requireNonNullValues(debuggingInfo, "debuggingInfo");
437    this.debuggingInfo = Set.copyOf(debuggingInfo);
438    return this;
439  }
440
441  @Override
442  public boolean isParameterInfoEnabled() {
443    return parameterInfoEnabled;
444  }
445
446  @Override
447  public JctCompiler parameterInfoEnabled(boolean parameterInfoEnabled) {
448    this.parameterInfoEnabled = parameterInfoEnabled;
449    return this;
450  }
451
452  /**
453   * Get the string representation of the compiler.
454   *
455   * @return the string representation of the compiler.
456   */
457  @Override
458  public final String toString() {
459    // This returns the compiler name to simplify parameterization naming in @JavacCompilerTest
460    // parameterized tests.
461    return name;
462  }
463
464  /**
465   * Get the flag builder factory to use for building flags.
466   *
467   * @return the factory.
468   */
469  public abstract JctFlagBuilderFactory getFlagBuilderFactory();
470
471  /**
472   * Get the JSR-199 compiler factory to use for initialising an internal compiler.
473   *
474   * @return the factory.
475   */
476  public abstract Jsr199CompilerFactory getCompilerFactory();
477
478  /**
479   * Get the file manager factory to use for building AbstractJctCompiler file manager during
480   * compilation.
481   *
482   * <p>Since v1.1.0, this method has provided a default implementation. Before this, it was
483   * abstract. The default implementation calls
484   * {@link JctFileManagers#newJctFileManagerFactory(JctCompiler)}.
485   *
486   * @return the factory.
487   */
488  public JctFileManagerFactory getFileManagerFactory() {
489    return JctFileManagers.newJctFileManagerFactory(this);
490  }
491
492  /**
493   * Get the compilation factory to use for building a compilation.
494   *
495   * <p>By default, this uses a common internal implementation that is designed to work with
496   * compilers that have interfaces the same as, and behave the same as Javac.
497   *
498   * <p>Some obscure compiler implementations with potentially satanic rituals for initialising
499   * and configuring components correctly may need to provide a custom implementation here instead.
500   * In this case, this method should be overridden. Base classes are not provided for you to extend
501   * in this case as this is usually not something you want to be doing. Instead, you should
502   * implement {@link JctCompilationFactory} directly.
503   *
504   * @return the compilation factory.
505   */
506  public JctCompilationFactory getCompilationFactory() {
507    return new JctCompilationFactoryImpl(this);
508  }
509
510  /**
511   * {@inheritDoc}
512   *
513   * @return the default release version to use when no version is specified by the user.
514   */
515  @Override
516  public abstract String getDefaultRelease();
517
518  /**
519   * Build the list of flags from this compiler object using the flag builder.
520   *
521   * <p>Implementations should not need to override this unless there is a special edge case
522   * that needs configuring differently. This is exposed to assist in these kinds of cases.
523   *
524   * @param flagBuilder the flag builder to apply the flag configuration to.
525   * @return the string flags to use.
526   */
527  protected List<String> buildFlags(JctFlagBuilder flagBuilder) {
528    return flagBuilder
529        .annotationProcessorOptions(annotationProcessorOptions)
530        .showDeprecationWarnings(showDeprecationWarnings)
531        .failOnWarnings(failOnWarnings)
532        .compilerOptions(compilerOptions)
533        .previewFeatures(previewFeatures)
534        .release(release)
535        .source(source)
536        .target(target)
537        .verbose(verbose)
538        .showWarnings(showWarnings)
539        .debuggingInfo(debuggingInfo)
540        .parameterInfoEnabled(parameterInfoEnabled)
541        .build();
542  }
543
544  @SuppressWarnings("ThrowFromFinallyBlock")
545  private JctCompilation performCompilation(
546        Workspace workspace, 
547        @Nullable Collection<String> classNames
548  ) {
549    var fileManagerFactory = getFileManagerFactory();
550    var flagBuilderFactory = getFlagBuilderFactory();
551    var compilerFactory = getCompilerFactory();
552    var compilationFactory = getCompilationFactory();
553    var flags = buildFlags(flagBuilderFactory.createFlagBuilder());
554    var compiler = compilerFactory.createCompiler();
555    var fileManager = fileManagerFactory.createFileManager(workspace);
556
557    // Any internal exceptions should be rethrown as a JctCompilerException by the
558    // compilation factory, so there is nothing else to worry about here.
559    // Likewise, do not catch IOException on the compilation process, as it may hide
560    // bugs.
561    //
562    // The try-finally-try-catch-rethrow ensures we only catch IOExceptions during
563    // the file manager closure, where it is a bug.
564    
565    try {
566      return compilationFactory.createCompilation(flags, fileManager, compiler, classNames);
567    } finally {
568      try {
569        fileManager.close();
570      } catch (IOException ex) {
571        throw new JctCompilerException(
572            "Failed to close file manager. This is probably a bug, so please report it.",
573            ex
574        );
575      }
576    }
577  }
578}