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.containers; 017 018import io.github.ascopes.jct.filemanagers.PathFileObject; 019import io.github.ascopes.jct.workspaces.PathRoot; 020import java.io.Closeable; 021import java.io.IOException; 022import java.lang.module.ModuleFinder; 023import java.nio.file.Path; 024import java.util.Collection; 025import java.util.Set; 026import javax.tools.FileObject; 027import javax.tools.JavaFileManager.Location; 028import javax.tools.JavaFileObject; 029import javax.tools.JavaFileObject.Kind; 030import org.jspecify.annotations.Nullable; 031 032/** 033 * Container that wraps a file path source of some description. 034 * 035 * <p>Closing this resource will close any internally held resources that were opened internally. 036 * Already-opened resources passed to the implementation will not be closed. 037 * 038 * @author Ashley Scopes 039 */ 040public interface Container extends Closeable { 041 042 /** 043 * Determine if this container contains the path file object. 044 * 045 * @param fileObject the file object to look up. 046 * @return {@code true} if the object exists in this container, or {@code false} otherwise. 047 */ 048 boolean contains(PathFileObject fileObject); 049 050 /** 051 * Find the physical path to a given string file path. 052 * 053 * <p>Examples: 054 * 055 * <pre><code> 056 * // Using platform-specific separators ('/' for in-memory file systems). 057 * container.getFile("foo/bar/baz.txt"); 058 * 059 * // Letting JCT infer the correct path separators to use. 060 * container.getFile("foo", "bar", "baz.txt"); 061 * </code></pre> 062 * 063 * @param fragments parts of the file name to find. 064 * @return the path if the file exists, or null if it does not exist. 065 * @throws NullPointerException if any of the path fragments are {@code null}. 066 * @throws IllegalArgumentException if there are no path fragments provided. 067 */ 068 @Nullable 069 Path getFile(String... fragments); 070 071 /** 072 * Get a {@link FileObject} for reading, if it exists. 073 * 074 * <p>If the file does not exist, {@code null} is returned. 075 * 076 * @param packageName the package name of the file. 077 * @param relativeName the relative name of the file in the package. 078 * @return the file object, or null if it does not exist. 079 */ 080 @Nullable 081 PathFileObject getFileForInput(String packageName, String relativeName); 082 083 /** 084 * Get a {@link FileObject} for writing. 085 * 086 * <p>If the container is read-only, {@code null} is returned. 087 * 088 * @param packageName the package name of the file. 089 * @param relativeName the relative name of the file in the package. 090 * @return the file object, or null if this container is read-only. 091 */ 092 @Nullable 093 PathFileObject getFileForOutput(String packageName, String relativeName); 094 095 /** 096 * Get the inner path root of the container. 097 * 098 * <p>This will usually be the same as the {@link #getPathRoot() path root}, but for archives, 099 * this will point to the root directory of the archive for the virtual file system it is loaded 100 * in, rather than the location of the archive on the original file system. 101 * 102 * <p>It is worth noting that this operation may result in the archive being loaded into memory 103 * eagerly if it uses lazy loading, due to the need to load the archive to be able to determine an 104 * accurate handle to the inner root directory. 105 * 106 * @return the path root. 107 * @since 0.0.6 108 */ 109 PathRoot getInnerPathRoot(); 110 111 /** 112 * Get a {@link JavaFileObject} for reading, if it exists. 113 * 114 * <p>If the file does not exist, {@code null} is returned. 115 * 116 * @param className the binary name of the class to open. 117 * @param kind the kind of file to open. 118 * @return the file object, or null if it does not exist. 119 */ 120 @Nullable 121 PathFileObject getJavaFileForInput(String className, Kind kind); 122 123 /** 124 * Get a {@link JavaFileObject} for writing. 125 * 126 * <p>If the container is read-only, {@code null} is returned. 127 * 128 * @param className the binary name of the class to open. 129 * @param kind the kind of file to open. 130 * @return the file object, or null if this container is read-only. 131 */ 132 @Nullable 133 PathFileObject getJavaFileForOutput(String className, Kind kind); 134 135 /** 136 * Get the location of this container. 137 * 138 * @return the location. 139 */ 140 Location getLocation(); 141 142 /** 143 * Get a module finder for this container. 144 * 145 * <p>Note that this will not detect modules that are not yet compiled. 146 * 147 * @return the module finder for this container, or {@code null} if not relevant to the 148 * implementation. 149 */ 150 @Nullable 151 ModuleFinder getModuleFinder(); 152 153 /** 154 * Get the name of this container for human consumption. 155 * 156 * @return the container name. 157 */ 158 String getName(); 159 160 /** 161 * Get the path root of the container. 162 * 163 * @return the path root. 164 */ 165 PathRoot getPathRoot(); 166 167 /** 168 * Infer the binary name of a given Java file object. 169 * 170 * @param javaFileObject the Java file object to infer the binary name of. 171 * @return the name, or null if the file does not exist in this container. 172 */ 173 @Nullable 174 String inferBinaryName(PathFileObject javaFileObject); 175 176 /** 177 * List all files within this container. 178 * 179 * @return all files within the container, relative to the container base path. 180 * @throws IOException if the file system failed to be read. 181 */ 182 Collection<Path> listAllFiles() throws IOException; 183 184 /** 185 * List all the file objects that match the given criteria in this group. 186 * 187 * <p>The results are filled into a given collection, since this call may be made many times 188 * per compilation, and this reduces the memory overhead needed in such cases. 189 * 190 * @param packageName the package name to look in. 191 * @param kinds the kinds of file to look for. Set to {@code Set.of(Kind.OTHER)} to find all 192 * types of file. 193 * @param recurse {@code true} to recurse subpackages, {@code false} to only consider the 194 * given package. 195 * @param collection the collection to fill. 196 * @throws IOException if the file system failed to be read. 197 */ 198 void listFileObjects( 199 String packageName, 200 Set<? extends Kind> kinds, 201 boolean recurse, 202 Collection<JavaFileObject> collection 203 ) throws IOException; 204}