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.junit; 017 018import io.github.ascopes.jct.workspaces.Workspace; 019import io.github.ascopes.jct.workspaces.Workspaces; 020import java.lang.reflect.Field; 021import java.lang.reflect.Modifier; 022import java.util.ArrayList; 023import java.util.List; 024import org.jspecify.annotations.Nullable; 025import org.junit.jupiter.api.extension.AfterAllCallback; 026import org.junit.jupiter.api.extension.AfterEachCallback; 027import org.junit.jupiter.api.extension.BeforeAllCallback; 028import org.junit.jupiter.api.extension.BeforeEachCallback; 029import org.junit.jupiter.api.extension.Extension; 030import org.junit.jupiter.api.extension.ExtensionContext; 031import org.slf4j.Logger; 032import org.slf4j.LoggerFactory; 033 034/** 035 * JUnit5 extension that will manage the lifecycle of {@link Managed}-annotated {@link Workspace} 036 * fields within JUnit5 test classes. 037 * 038 * <pre><code> 039 * {@literal @ExtendWith(JctExtension.class)} 040 * class MyTest { 041 * {@literal @Managed} 042 * Workspace workspace; 043 * 044 * {@literal @JavacCompilerTest} 045 * void myTest(JctCompiler compiler) { 046 * // Given 047 * workspace 048 * .createSourcePathPackage() 049 * ...; 050 * 051 * // When 052 * var compilation = compiler.compile(workspace); 053 * 054 * // Then 055 * ... 056 * } 057 * } 058 * </code></pre> 059 * 060 * @author Ashley Scopes 061 * @since 0.4.0 062 */ 063public final class JctExtension 064 implements Extension, BeforeEachCallback, BeforeAllCallback, AfterEachCallback, AfterAllCallback { 065 066 private static final Logger log = LoggerFactory.getLogger(JctExtension.class); 067 068 /** 069 * Initialise this extension. 070 * 071 * <p>You shouldn't ever need to call this directly. See the class description for an example 072 * of how to use this. 073 */ 074 public JctExtension() { 075 // Nothing to do. 076 } 077 078 @Override 079 public void beforeAll(ExtensionContext context) throws Exception { 080 for (var field : getManagedWorkspaceFields(context.getRequiredTestClass(), true)) { 081 initWorkspaceForField(field, null); 082 } 083 } 084 085 @Override 086 public void beforeEach(ExtensionContext context) throws Exception { 087 for (var instance : context.getRequiredTestInstances().getAllInstances()) { 088 for (var field : getManagedWorkspaceFields(instance.getClass(), false)) { 089 initWorkspaceForField(field, instance); 090 } 091 } 092 } 093 094 @Override 095 public void afterAll(ExtensionContext context) throws Exception { 096 for (var field : getManagedWorkspaceFields(context.getRequiredTestClass(), true)) { 097 closeWorkspaceForField(field, null); 098 } 099 } 100 101 @Override 102 public void afterEach(ExtensionContext context) throws Exception { 103 for (var instance : context.getRequiredTestInstances().getAllInstances()) { 104 for (var field : getManagedWorkspaceFields(instance.getClass(), false)) { 105 closeWorkspaceForField(field, instance); 106 } 107 } 108 } 109 110 private List<Field> getManagedWorkspaceFields(Class<?> clazz, boolean wantStatic) { 111 var fields = new ArrayList<Field>(); 112 113 Class<?> currentClass = clazz; 114 115 do { 116 for (var field : currentClass.getDeclaredFields()) { 117 var isWorkspace = field.getType().equals(Workspace.class); 118 var isManaged = field.isAnnotationPresent(Managed.class); 119 var isDesiredScope = Modifier.isStatic(field.getModifiers()) == wantStatic; 120 121 if (isWorkspace && isManaged && isDesiredScope) { 122 field.setAccessible(true); 123 fields.add(field); 124 } 125 } 126 127 // Only recurse if we are checking instance scope. We don't manage annotated fields 128 // in superclasses that are static as we cannot guarantee they are not shared with a 129 // different class running in parallel. 130 currentClass = wantStatic 131 ? null 132 : currentClass.getSuperclass(); 133 134 } while (currentClass != null); 135 136 return fields; 137 } 138 139 private void initWorkspaceForField(Field field, @Nullable Object instance) throws Exception { 140 log.atTrace() 141 .setMessage("Initialising workspace for field in {}: {} {} on instance {}") 142 .addArgument(() -> field.getDeclaringClass().getSimpleName()) 143 .addArgument(() -> field.getType().getSimpleName()) 144 .addArgument(field::getName) 145 .addArgument(instance) 146 .log(); 147 148 var managedWorkspace = field.getAnnotation(Managed.class); 149 var workspace = Workspaces.newWorkspace(managedWorkspace.pathStrategy()); 150 field.set(instance, workspace); 151 } 152 153 private void closeWorkspaceForField(Field field, @Nullable Object instance) throws Exception { 154 log.atTrace() 155 .setMessage("Closing workspace for field in {}: {} {} on instance {}") 156 .addArgument(() -> field.getDeclaringClass().getSimpleName()) 157 .addArgument(() -> field.getType().getSimpleName()) 158 .addArgument(field::getName) 159 .addArgument(instance) 160 .log(); 161 162 var workspace = (Workspace) field.get(instance); 163 workspace.close(); 164 } 165}