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; 017 018import java.io.IOException; 019import java.io.InputStream; 020import java.io.OutputStream; 021import java.io.Reader; 022import java.io.Writer; 023import java.net.URI; 024import java.nio.file.NoSuchFileException; 025import java.nio.file.Path; 026import javax.lang.model.element.Modifier; 027import javax.lang.model.element.NestingKind; 028import javax.tools.FileObject; 029import javax.tools.JavaFileManager.Location; 030import javax.tools.JavaFileObject; 031import org.jspecify.annotations.Nullable; 032 033/** 034 * Interface for a path-based {@link JavaFileObject} that points to a path on a file system 035 * somewhere. 036 * 037 * <p>This object will always use UTF-8 encoding when obtaining readers or writers. 038 * 039 * <p>No access level or nesting kind information is provided by this implementation. 040 * 041 * @author Ashley Scopes 042 * @since 1.0.0 043 */ 044public interface PathFileObject extends JavaFileObject { 045 046 /** 047 * Timestamp that is returned if the file object has never been modified. 048 */ 049 long NOT_MODIFIED = 0L; 050 051 /** 052 * Attempt to delete the file. 053 * 054 * @return {@code true} if the file was deleted, or {@code false} if it was not deleted. 055 */ 056 @Override 057 boolean delete(); 058 059 /** 060 * Determine if this object equals another. 061 * 062 * @param other the other object to compare with. 063 * @return {@code true} if the other object is also a {@link FileObject} and has the same 064 * {@link #toUri() URI} as this object, or {@code false} otherwise. 065 */ 066 @Override 067 boolean equals(@Nullable Object other); 068 069 /** 070 * Get the absolute path of this file object. 071 * 072 * @return the full path. 073 */ 074 Path getAbsolutePath(); 075 076 /** 077 * Get the class access level, where appropriate. 078 * 079 * <p>In most implementations, this class will always return {@code null}, since this 080 * information is not readily available without preloading the file in question and parsing it 081 * first. 082 * 083 * <p>At the time of writing, the OpenJDK implementations of the JavaFileObject class 084 * do not provide an implementation for this method either. 085 * 086 * @return the modifier for the access level, or {@code null} if the information is not available. 087 */ 088 @Nullable 089 @Override 090 default Modifier getAccessLevel() { 091 return null; 092 } 093 094 /** 095 * Get the inferred binary name of the file object. 096 * 097 * @return the inferred binary name. 098 */ 099 String getBinaryName(); 100 101 /** 102 * Read the character content of the file into memory and decode it using the default character 103 * set. 104 * 105 * @param ignoreEncodingErrors ignore encoding errors if {@code true}, or throw them otherwise. 106 * @return the character content, encoded as UTF-8. 107 * @throws IOException if an IO error occurs. 108 */ 109 @Override 110 String getCharContent(boolean ignoreEncodingErrors) throws IOException; 111 112 /** 113 * Get the kind of the file. 114 * 115 * @return the inferred file kind. 116 */ 117 @Override 118 Kind getKind(); 119 120 /** 121 * Determine when the file was last modified. 122 * 123 * @return the timestamp in milliseconds since UNIX epoch that the file was last modified at, or 124 * {@link #NOT_MODIFIED} if unmodified, if the information is not available, or if an error 125 * occurred obtaining the information. 126 */ 127 @Override 128 long getLastModified(); 129 130 /** 131 * Get the location of this path file object. 132 * 133 * @return the location. 134 */ 135 Location getLocation(); 136 137 /** 138 * Get the file name as a string. 139 * 140 * @return the name of the file. 141 */ 142 @Override 143 String getName(); 144 145 /** 146 * Determine the class nesting kind, where appropriate. 147 * 148 * <p>In most implementations, this method will always return {@code null}, since this 149 * information is not readily available without preloading the file in question and parsing it 150 * first. 151 * 152 * <p>At the time of writing, the OpenJDK implementations of the JavaFileObject class 153 * do not provide an implementation for this method either. 154 * 155 * @return the nesting kind, or {@code null} if the information is not available. 156 */ 157 @Nullable 158 @Override 159 default NestingKind getNestingKind() { 160 return null; 161 } 162 163 /** 164 * Get the relative path of this file object. 165 * 166 * @return the path of this file object. 167 */ 168 Path getRelativePath(); 169 170 /** 171 * Get the root path that the package containing this file is nested within. 172 * 173 * @return the root path. 174 */ 175 Path getRootPath(); 176 177 /** 178 * Determine the hash code for this object. 179 * 180 * @return the hash code for the object. 181 */ 182 @Override 183 int hashCode(); 184 185 /** 186 * Determine if a given simple name and file kind are compatible with this file object. 187 * 188 * <p>This will perform a case-sensitive check, regardless of the platform that it runs on. 189 * 190 * @param simpleName the simple name of the class to compare to this file. 191 * @param kind the kind of the class to compare to this file. 192 * @return {@code true} if the simple name and kind are compatible with the current file object 193 * name, or {@code false} if not. 194 */ 195 @Override 196 boolean isNameCompatible(String simpleName, Kind kind); 197 198 /** 199 * Open an input stream into this file. 200 * 201 * <p>This input stream must be closed once finished with, otherwise resources will be leaked. 202 * 203 * <p>The returned implementation will always be buffered when appropriate. 204 * 205 * @return an input stream. 206 * @throws NoSuchFileException if the file does not exist. 207 * @throws IOException if an IO error occurs. 208 */ 209 @Override 210 InputStream openInputStream() throws IOException; 211 212 /** 213 * Open an output stream to this file. 214 * 215 * <p>This will create the file first if it does not already exist. If it does exist, then this 216 * will overwrite the file and truncate it. The parent directories will also be created if they do 217 * not exist. 218 * 219 * <p>This output stream must be closed once finished with, otherwise resources will be leaked. 220 * 221 * <p>The returned implementation will always be buffered when appropriate. 222 * 223 * @return an output stream. 224 * @throws IOException if an IO error occurs. 225 */ 226 @Override 227 OutputStream openOutputStream() throws IOException; 228 229 /** 230 * Open a reader to this file using the default charset (UTF-8). 231 * 232 * <p>This reader must be closed once finished with, otherwise resources will be leaked. 233 * 234 * <p>The returned implementation will always be buffered when appropriate. 235 * 236 * @param ignoreEncodingErrors {@code true} to suppress encoding errors, or {@code false} to throw 237 * them to the caller. 238 * @return a reader. 239 * @throws NoSuchFileException if the file does not exist. 240 * @throws IOException if an IO error occurs. 241 */ 242 @Override 243 Reader openReader(boolean ignoreEncodingErrors) throws IOException; 244 245 /** 246 * Open a writer to this file using the default charset (UTF-8). 247 * 248 * <p>This will create the file first if it does not already exist. If it does exist, this will 249 * first overwrite the file and truncate it. The parent directories will also be created if they 250 * do not exist. 251 * 252 * <p>This input stream must be closed once finished with, otherwise resources will be leaked. 253 * 254 * <p>The returned implementation will always be buffered when appropriate. 255 * 256 * @return a writer. 257 * @throws IOException if an IO error occurs. 258 */ 259 @Override 260 Writer openWriter() throws IOException; 261 262 /** 263 * Get a string representation of this object. 264 * 265 * @return a string representation of this object. 266 */ 267 @Override 268 String toString(); 269 270 /** 271 * Determine the URI for this file. 272 * 273 * @return the URI for this file object. 274 */ 275 @Override 276 URI toUri(); 277}