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 java.io.File;
019import java.io.InputStream;
020import java.net.URL;
021import java.nio.ByteBuffer;
022import java.nio.charset.Charset;
023import java.nio.file.Path;
024import java.util.List;
025
026/**
027 * Chainable builder for creating individual files.
028 *
029 * @author Ashley Scopes
030 * @since 0.0.1
031 */
032@SuppressWarnings({"unused", "UnusedReturnValue"})
033public interface FileBuilder {
034
035  /**
036   * Take the directory, represented by the given {@link ManagedDirectory}, and convert it into a
037   * JAR file, that will be written to the file being created.
038   *
039   * @param directory the managed directory to use.
040   * @return the root managed directory for further configuration.
041   * @since 0.3.0
042   */
043  ManagedDirectory asJarFrom(PathRoot directory);
044
045  /**
046   * Take the directory, represented by the given path, and convert it into a JAR file, that will be
047   * written to the file being created.
048   *
049   * <pre><code>
050   *   ManagedDirectory someDirectory = ...;
051   *   someDirectory
052   *       .createFile("first.jar")
053   *       .asJarFrom(firstClassOutputs);
054   *   var secondCompilation = secondCompiler.compile(secondWorkspace);
055   *   ...
056   * </code></pre>
057   *
058   * @param directory the directory to use.
059   * @return the root managed directory for further configuration.
060   * @since 0.3.0
061   */
062  ManagedDirectory asJarFrom(Path directory);
063
064  /**
065   * Copy a resource from the given class loader into the file system.
066   *
067   * <pre><code>
068   *   var classLoader = getClass().getClassLoader();
069   *
070   *   directory
071   *       .createFile("org", "example", "HelloWorld.class")
072   *       .copiedFromClassPath(classLoader, "org/example/HelloWorld.class");
073   * </code></pre>
074   *
075   * @param classLoader the class loader to use.
076   * @param resource    the resource to copy.
077   * @return the root managed directory for further configuration.
078   */
079  ManagedDirectory copiedFromClassPath(ClassLoader classLoader, String resource);
080
081  /**
082   * Copy a resource from the class loader on the current thread into the file system.
083   *
084   * <pre><code>
085   *   directory
086   *       .createFile("org", "example", "HelloWorld.class")
087   *       .copiedFromClassPath("org/example/HelloWorld.class");
088   * </code></pre>
089   *
090   * @param resource the resource to copy.
091   * @return the root managed directory for further configuration.
092   */
093  ManagedDirectory copiedFromClassPath(String resource);
094
095  /**
096   * Copy the contents from the given file into the file system.
097   *
098   * <pre><code>
099   *   var file = new File("src/test/resources/code/HelloWorld.java");
100   *
101   *   directory
102   *       .createFile("org", "example", "HelloWorld.java")
103   *       .copiedFromFile(file);
104   * </code></pre>
105   *
106   * @param file the file to read.
107   * @return the root managed directory for further configuration.
108   */
109  ManagedDirectory copiedFromFile(File file);
110
111  /**
112   * Copy the contents from the given path into the file system.
113   *
114   * <pre><code>
115   *   var file = Path.of("src", "test", "resources", "code", "HelloWorld.java");
116   *
117   *   directory
118   *       .createFile("org", "example", "HelloWorld.java")
119   *       .copiedFromFile(file);
120   * </code></pre>
121   *
122   * @param file the file to read.
123   * @return the root managed directory for further configuration.
124   */
125  ManagedDirectory copiedFromFile(Path file);
126
127  /**
128   * Copy the contents from the given URL into the file system.
129   *
130   * <pre><code>
131   *   var url = URI.create("file:///code/org/example/HelloWorld.java").toURI();
132   *
133   *   directory
134   *       .createFile("org", "example", "HelloWorld.java")
135   *       .copiedFromUrl(url);
136   * </code></pre>
137   *
138   * @param url the URL to read.
139   * @return the root managed directory for further configuration.
140   */
141  ManagedDirectory copiedFromUrl(URL url);
142
143  /**
144   * Copy the contents from the given input stream into the file system.
145   *
146   * <p>The input stream will be closed when reading completes.
147   *
148   * <pre><code>
149   *   try (var is = getClass().getResourceAsStream("code/examples/HelloWorld.java")) {
150   *     directory
151   *         .createFile("org", "example", "HelloWorld.java")
152   *         .fromInputStream(is);
153   *   }
154   * </code></pre>
155   *
156   * @param inputStream the input stream to read.
157   * @return the root managed directory for further configuration.
158   */
159  ManagedDirectory fromInputStream(InputStream inputStream);
160
161  /**
162   * Create the file but do not put anything in it.
163   *
164   * <p>The resultant file will be 0 bytes long.
165   *
166   * <pre><code>
167   *   directory
168   *       .createFile(".gitkeep")
169   *       .thatIsEmpty();
170   * </code></pre>
171   *
172   * @return the root managed directory for further configuration.
173   */
174  ManagedDirectory thatIsEmpty();
175
176  /**
177   * Create the file with the given byte contents.
178   *
179   * <pre><code>
180   *   var classFile = new byte[]{
181   *      -54, -2, -70, -66, 0, 0, 0, 63, 0, 29, 10, 0, 2, 0, 3, 7, 0, 4, 12, 0, 5, 0, 6, 1, 0,
182   *      16, 106, 97, 118, 97, 47, 108, 97, 110, 103, 47, 79, 98, 106, 101, 99, 116, 1, 0, 6,
183   *      60, 105, 110, 105, 116, 62, 1, 0, 3, 40, 41, 86, 9, 0, 8, 0, 9, 7, 0, 10, 12, 0, 11,
184   *      0, 12, 1, 0, 16, 106, 97, 118, 97, 47, 108, 97, 110, 103, 47, 83, 121, 115, 116, 101,
185   *      109, 1, 0, 3, 111, 117, 116, 1, 0, 21, 76, 106, 97, 118, 97, 47, 105, 111, 47, 80, 114,
186   *      105, 110, 116, 83, 116, 114, 101, 97, 109, 59, 8, 0, 14, 1, 0, 13, 72, 101, 108, 108,
187   *      111, 44, 32, 87, 111, 114, 108, 100, 33, 10, 0, 16, 0, 17, 7, 0, 18, 12, 0, 19, 0, 20,
188   *      1, 0, 19, 106, 97, 118, 97, 47, 105, 111, 47, 80, 114, 105, 110, 116, 83, 116, 114, 101,
189   *      97, 109, 1, 0, 7, 112, 114, 105, 110, 116, 108, 110, 1, 0, 21, 40, 76, 106, 97, 118, 97,
190   *      47, 108, 97, 110, 103, 47, 83, 116, 114, 105, 110, 103, 59, 41, 86, 7, 0, 22, 1, 0, 10,
191   *      72, 101, 108, 108, 111, 87, 111, 114, 108, 100, 1, 0, 4, 67, 111, 100, 101, 1, 0, 15, 76,
192   *      105, 110, 101, 78, 117, 109, 98, 101, 114, 84, 97, 98, 108, 101, 1, 0, 4, 109, 97, 105,
193   *      110, 1, 0, 22, 40, 91, 76, 106, 97, 118, 97, 47, 108, 97, 110, 103, 47, 83, 116, 114,
194   *      105, 110, 103, 59, 41, 86, 1, 0, 10, 83, 111, 117, 114, 99, 101, 70, 105, 108, 101, 1,
195   *      0, 15, 72, 101, 108, 108, 111, 87, 111, 114, 108, 100, 46, 106, 97, 118, 97, 0, 33, 0,
196   *      21, 0, 2, 0, 0, 0, 0, 0, 2, 0, 1, 0, 5, 0, 6, 0, 1, 0, 23, 0, 0, 0, 29, 0, 1, 0, 1, 0,
197   *      0, 0, 5, 42, -73, 0, 1, -79, 0, 0, 0, 1, 0, 24, 0, 0, 0, 6, 0, 1, 0, 0, 0, 1, 0, 9, 0,
198   *      25, 0, 26, 0, 1, 0, 23, 0, 0, 0, 37, 0, 2, 0, 1, 0, 0, 0, 9, -78, 0, 7, 18, 13, -74, 0,
199   *      15, -79, 0, 0, 0, 1, 0, 24, 0, 0, 0, 10, 0, 2, 0, 0, 0, 3, 0, 8, 0, 4, 0, 1, 0, 27, 0,
200   *      0, 0, 2, 0, 28
201   *   };
202   *
203   *   directory
204   *       .createFile("HelloWorld.class")
205   *       .withContents(classFile);
206   * </code></pre>
207   *
208   * @param contents the bytes to write.
209   * @return the root managed directory for further configuration.
210   * @see #withContents(ByteBuffer)
211   */
212  ManagedDirectory withContents(byte[] contents);
213
214  /**
215   * An overload of {@link #withContents(byte[])} that consumes a NIO byte buffer.
216   *
217   * @param buffer the byte buffer to consume.
218   * @return the managed directory.
219   * @since 4.0.0
220   */
221  default ManagedDirectory withContents(ByteBuffer buffer) {
222    var array = new byte[buffer.remaining()];
223    buffer.get(array);
224    return withContents(array);
225  }
226
227  /**
228   * Create the file with the given contents.
229   *
230   * <pre><code>
231   *   directory
232   *      .createFile("org", "example", "HelloWorld.java")
233   *      .withContents(StandardCharsets.US_ASCII, List.of(
234   *        "package org.example;",
235   *        "",
236   *        "public class HelloWorld {",
237   *        "  public static void main(String[] args) {",
238   *        "    System.out.println(\"Hello, World!\");",
239   *        "  }",
240   *        "}"
241   *      ));
242   * </code></pre>
243   *
244   * @param charset the character encoding to use.
245   * @param lines   the lines to write.
246   * @return the root managed directory for further configuration.
247   * @see #withContents(Charset, String...)
248   * @see #withContents(String...)
249   * @see #withContents(List)
250   * @see #withContents(byte[])
251   * @since 4.0.0
252   */
253  ManagedDirectory withContents(Charset charset, List<String> lines);
254
255  /**
256   * Create the file with the given contents.
257   *
258   * <pre><code>
259   *   directory
260   *      .createFile("org", "example", "HelloWorld.java")
261   *      .withContents(StandardCharsets.US_ASCII, """
262   *        package org.example;
263   *
264   *        public class HelloWorld {
265   *          public static void main(String[] args) {
266   *            System.out.println("Hello, World!");
267   *          }
268   *        }
269   *      """);
270   * </code></pre>
271   *
272   * <p>If the Java language level of your tests does not support multi-line strings, you can
273   * alternatively pass each line of text to write as a separate string. These will be written to
274   * the file using line-feed {@code '\n'} separators. For example:
275   *
276   * <pre><code>
277   *   directory
278   *      .createFile("org", "example", "HelloWorld.java")
279   *      .withContents(
280   *        StandardCharsets.US_ASCII,
281   *        "package org.example;",
282   *        "",
283   *        "public class HelloWorld {",
284   *        "  public static void main(String[] args) {",
285   *        "    System.out.println(\"Hello, World!\");",
286   *        "  }",
287   *        "}"
288   *      );
289   * </code></pre>
290   *
291   * @param charset the character encoding to use.
292   * @param lines   the lines to write.
293   * @return the root managed directory for further configuration.
294   * @see #withContents(List)
295   * @see #withContents(Charset, List)
296   * @see #withContents(String...)
297   * @see #withContents(byte[])
298   */
299  default ManagedDirectory withContents(Charset charset, String... lines) {
300    return withContents(charset, List.of(lines));
301  }
302
303  /**
304   * Create the file with the given contents as UTF-8.
305   *
306   * <p>If you are using multi-line strings, an example of usage would be:
307   *
308   * <pre><code>
309   *   directory
310   *      .createFile("org", "example", "HelloWorld.java")
311   *      .withContents(List.of(
312   *        "package org.example;",
313   *        "",
314   *        "public class HelloWorld {",
315   *        "  public static void main(String[] args) {",
316   *        "    System.out.println(\"Hello, World!\");",
317   *        "  }",
318   *        "}"
319   *      ));
320   * </code></pre>
321   *
322   * @param lines the lines to write using the default charset.
323   * @return the root managed directory for further configuration.
324   * @see #withContents(Charset, String...)
325   * @see #withContents(String...)
326   * @see #withContents(Charset, List)
327   * @see #withContents(byte[])
328   * @since 4.0.0
329   */
330  ManagedDirectory withContents(List<String> lines);
331
332  /**
333   * Create the file with the given contents as UTF-8.
334   *
335   * <p>If you are using multi-line strings, an example of usage would be:
336   *
337   * <pre><code>
338   *   directory
339   *      .createFile("org", "example", "HelloWorld.java")
340   *      .withContents("""
341   *        package org.example;
342   *
343   *        public class HelloWorld {
344   *          public static void main(String[] args) {
345   *            System.out.println("Hello, World!");
346   *          }
347   *        }
348   *      """);
349   * </code></pre>
350   *
351   * <p>If the Java language level of your tests does not support multi-line strings, you can
352   * alternatively pass each line of text to write as a separate string. These will be written to
353   * the file using line-feed {@code '\n'} separators. For example:
354   *
355   * <pre><code>
356   *   directory
357   *      .createFile("org", "example", "HelloWorld.java")
358   *      .withContents(
359   *        "package org.example;",
360   *        "",
361   *        "public class HelloWorld {",
362   *        "  public static void main(String[] args) {",
363   *        "    System.out.println(\"Hello, World!\");",
364   *        "  }",
365   *        "}"
366   *      );
367   * </code></pre>
368   *
369   * @param lines the lines to write using the default charset.
370   * @return the root managed directory for further configuration.
371   * @see #withContents(Charset, String...)
372   * @see #withContents(List)
373   * @see #withContents(Charset, List)
374   * @see #withContents(byte[])
375   */
376  default ManagedDirectory withContents(String... lines) {
377    return withContents(List.of(lines));
378  }
379}