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.config; 017 018import io.github.ascopes.jct.compilers.JctCompiler; 019import io.github.ascopes.jct.ex.JctIllegalInputException; 020import io.github.ascopes.jct.filemanagers.AnnotationProcessorDiscovery; 021import io.github.ascopes.jct.filemanagers.JctFileManager; 022import java.util.Map; 023import javax.tools.StandardLocation; 024import org.slf4j.Logger; 025import org.slf4j.LoggerFactory; 026 027/** 028 * Configurer for a file manager that makes annotation processors in the classpath accessible to the 029 * annotation processor path. 030 * 031 * <p>If annotation processor discovery is disabled for dependencies, this will be skipped. 032 * 033 * @author Ashley Scopes 034 * @since 0.0.1 035 */ 036public final class JctFileManagerAnnotationProcessorClassPathConfigurer 037 implements JctFileManagerConfigurer { 038 039 private static final Logger log = LoggerFactory 040 .getLogger(JctFileManagerAnnotationProcessorClassPathConfigurer.class); 041 042 private static final Map<StandardLocation, StandardLocation> INHERITED_AP_PATHS = Map.of( 043 // https://stackoverflow.com/q/53084037 044 // Seems that javac will always use the classpath to implement this behaviour, and never 045 // the module path. Let's keep this simple and mimic this behaviour. If someone complains 046 // about it being problematic in the future, then I am open to change how this works to 047 // keep it sensible. 048 // (from -> to) 049 StandardLocation.CLASS_PATH, StandardLocation.ANNOTATION_PROCESSOR_PATH 050 ); 051 052 private final JctCompiler compiler; 053 054 /** 055 * Initialise the configurer with the desired compiler. 056 * 057 * @param compiler the compiler to wrap. 058 */ 059 public JctFileManagerAnnotationProcessorClassPathConfigurer(JctCompiler compiler) { 060 this.compiler = compiler; 061 } 062 063 @Override 064 public JctFileManager configure(JctFileManager fileManager) { 065 log.debug("Configuring annotation processor discovery mechanism"); 066 067 switch (compiler.getAnnotationProcessorDiscovery()) { 068 case ENABLED: 069 log.trace("Annotation processor discovery is enabled, ensuring empty location exists"); 070 071 INHERITED_AP_PATHS.values().forEach(fileManager::createEmptyLocation); 072 073 return fileManager; 074 075 case INCLUDE_DEPENDENCIES: 076 log.trace("Annotation processor discovery is enabled, copying classpath dependencies " 077 + "into the annotation processor path"); 078 079 INHERITED_AP_PATHS.forEach(fileManager::copyContainers); 080 INHERITED_AP_PATHS.values().forEach(fileManager::createEmptyLocation); 081 082 return fileManager; 083 084 default: 085 throw new JctIllegalInputException("Cannot configure annotation processor discovery"); 086 } 087 } 088 089 @Override 090 public boolean isEnabled() { 091 return compiler.getAnnotationProcessorDiscovery() != AnnotationProcessorDiscovery.DISABLED; 092 } 093}