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.workspaces;
017
018import io.github.ascopes.jct.workspaces.impl.AbstractManagedDirectory;
019import io.github.ascopes.jct.workspaces.impl.RamDirectoryImpl;
020import io.github.ascopes.jct.workspaces.impl.TempDirectoryImpl;
021import java.io.File;
022import java.nio.file.Path;
023import java.util.function.Function;
024import javax.annotation.processing.Filer;
025
026/**
027 * Strategy to use for creating new test directories.
028 *
029 * <p>This is used to define whether to use a totally isolated in-memory file system,
030 * or whether to use temporary directories on the default file system.
031 *
032 * @author Ashley Scopes
033 * @since 0.0.1
034 */
035public enum PathStrategy {
036  /**
037   * Use RAM-based directories for any created directories.
038   *
039   * <p>This is faster as everything remains in-memory. It also prevents the risk of resources
040   * not being tidied up correctly if the JVM is suddenly shut down. Test directories are also kept
041   * isolated from other tests that may be running in parallel, and isolated from the host operating
042   * system.
043   *
044   * <p>If any annotation processors rely on being run in the {@link File default file system}
045   * rather than using {@link Path the NIO path API} or {@link Filer filers} directly, then they
046   * will not be compatible with reading files from this implementation of test directory. In this
047   * situation, users should opt to use {@link #TEMP_DIRECTORIES} instead.
048   *
049   * <p>Some non-Javac compiler implementations (such as ECJ) may also have some difficulties
050   * dealing with these paths.
051   */
052  RAM_DIRECTORIES(RamDirectoryImpl::newRamDirectory),
053
054  /**
055   * Use OS-level temporary directories for any created directories.
056   *
057   * <p>This will write files to the OS temporary directory. These files will be deleted once the
058   * owning workspace is closed, but may be missed if the JVM is forcefully terminated or crashes.
059   *
060   * <p>There are fewer guarantees of speed and isolation compared to using
061   * {@link #RAM_DIRECTORIES}. However, you do gain the ability to set a breakpoint and inspect the
062   * contents of the directory in a file explorer.
063   *
064   * <p>Since the temporary directories are usually created on the
065   * {@link File default file system}, they are compatible with any annotation processors or
066   * compiler implementations that expect to be run on the default file system only.
067   *
068   * <p>Some restrictions regarding file naming may be present depending on the platform that
069   * is in use, such as file name lengths on Windows. See your system documentation for more
070   * details.
071   */
072  TEMP_DIRECTORIES(TempDirectoryImpl::newTempDirectory);
073
074  private final Function<String, AbstractManagedDirectory> constructor;
075
076  PathStrategy(Function<String, AbstractManagedDirectory> constructor) {
077    this.constructor = constructor;
078  }
079
080  /**
081   * Create a new instance of the test directory type with the given name.
082   *
083   * <p>Note that calling this directly will return an object that you will have to manually
084   * manage the lifetime for. Failing to do so will result in resources being leaked.
085   *
086   * <p><strong>Users should not call this method by default unless they know what they are
087   * doing.</strong> As a result of this constraint, this method is not part of the public API and
088   * may be subject to change without notice.
089   *
090   * @param name the name to use.
091   * @return the new test directory.
092   */
093  public ManagedDirectory newInstance(String name) {
094    return constructor.apply(name);
095  }
096
097  /**
098   * Determine the default strategy to fall back onto.
099   *
100   * <p>This will be {@link #RAM_DIRECTORIES} by default, but this may be subject to change between
101   * minor versions without notice, so do not rely on this if you are testing code that may be
102   * sensitive to the type of file system being used.
103   *
104   * @return the path strategy to use by default.
105   */
106  public static PathStrategy defaultStrategy() {
107    return RAM_DIRECTORIES;
108  }
109}