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.assertions;
017
018import static io.github.ascopes.jct.utils.IoExceptionUtils.uncheckedIo;
019import static io.github.ascopes.jct.utils.IterableUtils.requireAtLeastOne;
020import static io.github.ascopes.jct.utils.IterableUtils.requireNonNullValues;
021import static io.github.ascopes.jct.utils.StringUtils.quoted;
022import static io.github.ascopes.jct.utils.StringUtils.quotedIterable;
023import static java.util.function.Predicate.not;
024import static org.assertj.core.api.Assertions.assertThat;
025
026import io.github.ascopes.jct.containers.PackageContainerGroup;
027import io.github.ascopes.jct.repr.LocationRepresentation;
028import io.github.ascopes.jct.utils.StringUtils;
029import java.nio.file.Path;
030import java.util.ArrayList;
031import java.util.List;
032import java.util.Objects;
033import java.util.Optional;
034import java.util.Set;
035import java.util.stream.Collectors;
036import java.util.stream.StreamSupport;
037import org.assertj.core.api.AbstractPathAssert;
038import org.assertj.core.description.TextDescription;
039import org.assertj.core.error.MultipleAssertionsError;
040import org.jspecify.annotations.Nullable;
041
042/**
043 * Assertions for package container groups.
044 *
045 * @author Ashley Scopes
046 * @since 0.0.1
047 */
048public final class PackageContainerGroupAssert
049    extends AbstractContainerGroupAssert<PackageContainerGroupAssert, PackageContainerGroup> {
050
051  /**
052   * Initialize the container group assertions.
053   *
054   * @param containerGroup the container group to assert upon.
055   */
056  public PackageContainerGroupAssert(@Nullable PackageContainerGroup containerGroup) {
057    super(containerGroup, PackageContainerGroupAssert.class);
058  }
059
060  /**
061   * Assert that all given files exist.
062   *
063   * @param paths paths to check for.
064   * @return this object for further call chaining.
065   * @throws AssertionError           if the container group is null, or if any of the files do not
066   *                                  exist.
067   * @throws NullPointerException     if any of the paths are null.
068   * @throws IllegalArgumentException if no fragments are provided.
069   */
070  public PackageContainerGroupAssert allFilesExist(String... paths) {
071    requireNonNullValues(paths, "paths");
072    requireAtLeastOne(paths, "paths");
073
074    allFilesExist(List.of(paths));
075    return this;
076  }
077
078  /**
079   * Assert that all given files exist.
080   *
081   * @param paths paths to check for.
082   * @return this object for further call chaining.
083   * @throws AssertionError       if the container group is null, or if any of the files do not
084   *                              exist.
085   * @throws NullPointerException if any of the paths are null.
086   */
087  public PackageContainerGroupAssert allFilesExist(Iterable<String> paths) {
088    requireNonNullValues(paths, "paths");
089
090    isNotNull();
091
092    var errors = new ArrayList<AssertionError>();
093
094    for (var path : paths) {
095      try {
096        fileExists(path);
097      } catch (AssertionError ex) {
098        errors.add(ex);
099      }
100    }
101
102    if (errors.isEmpty()) {
103      return this;
104    }
105
106    throw new MultipleAssertionsError(
107        new TextDescription(
108            "Expected all paths in %s to exist but one or more did not",
109            quotedIterable(paths)
110        ),
111        errors
112    );
113  }
114
115  /**
116   * Get assertions to perform on the class loader associated with this container group.
117   *
118   * @return the assertions to perform.
119   * @throws AssertionError if the container group is null.
120   */
121  public ClassLoaderAssert classLoader() {
122    isNotNull();
123
124    return new ClassLoaderAssert(actual.getClassLoader());
125  }
126
127  /**
128   * Assert that the given file does not exist.
129   *
130   * <pre><code>
131   *   // Using platform-specific separators.
132   *   assertions.fileDoesNotExist("foo/bar/baz.txt")...;
133   *
134   *   // Letting JCT infer the correct path separators to use (recommended).
135   *   assertions.fileDoesNotExist("foo", "bar", "baz.txt");
136   * </code></pre>
137   *
138   * @param fragments any additional parts of the path.
139   * @return this assertion object for further assertions.
140   * @throws AssertionError           if the file exists, or if the container group is null.
141   * @throws NullPointerException     if any of the fragments are null.
142   * @throws IllegalArgumentException if no fragments are provided.
143   */
144  public PackageContainerGroupAssert fileDoesNotExist(String... fragments) {
145    requireNonNullValues(fragments, "fragments");
146    requireAtLeastOne(fragments, "fragments");
147
148    isNotNull();
149
150    var actualFile = actual.getFile(fragments);
151
152    if (actualFile == null) {
153      return this;
154    }
155
156    throw failure(
157        "Expected path %s to not exist in %s but it was found at %s",
158        quotedUserProvidedPath(List.of(fragments)),
159        LocationRepresentation.getInstance().toStringOf(actual.getLocation()),
160        quoted(actualFile)
161    );
162  }
163
164  /**
165   * Assert that the given file exists.
166   *
167   * <pre><code>
168   *   // Letting JCT infer the correct path separators to use (recommended).
169   *   assertions.fileExists("foo", "bar", "baz.txt");
170   *
171   *   // Using platform-specific separators (more likely to produce unexpected results).
172   *   assertions.fileExists("foo/bar/baz.txt")...;
173   * </code></pre>
174   *
175   * <p>If the file does not exist, then this object will attempt to find the
176   * closest matches and list them in an error message along with the assertion error.
177   *
178   * @param fragments parts of the path.
179   * @return assertions to perform on the path of the file that exists.
180   * @throws AssertionError           if the file does not exist, or if the container group is
181   *                                  null.
182   * @throws NullPointerException     if any of the fragments are null.
183   * @throws IllegalArgumentException if no fragments are provided.
184   */
185  public AbstractPathAssert<?> fileExists(String... fragments) {
186    requireNonNullValues(fragments, "fragments");
187    requireAtLeastOne(fragments, "fragments");
188
189    isNotNull();
190
191    var actualFile = actual.getFile(fragments);
192
193    if (actualFile != null) {
194      // File exists with this path. Hooray, lets return assertions on it.
195      return assertThat(actualFile);
196    }
197
198    var expected = List.of(fragments);
199    var message = StringUtils.resultNotFoundWithFuzzySuggestions(
200        fuzzySafePath(expected),
201        quotedUserProvidedPath(expected),
202        listAllUniqueFilesForAllContainers(),
203        this::fuzzySafePath,
204        this::quotedUserProvidedPath,
205        "file with relative path"
206    );
207
208    throw failure(message);
209  }
210
211  private Set<Path> listAllUniqueFilesForAllContainers() {
212    return uncheckedIo(actual::listAllFiles)
213        .keySet()
214        .stream()
215        // Make all the files relative to their roots.
216        .flatMap(container -> uncheckedIo(container::listAllFiles)
217            .stream()
218            .map(container.getInnerPathRoot().getPath()::relativize))
219        // Filter out the inner path root itself, preventing few confusing issues with zero-length
220        // file names. In ZIP paths this can also be null, so we have to check for both "" and null
221        // here.
222        .filter(this::fileNameIsPresent)
223        // Remove duplicates (don't think this can ever happen but this is just to be safe).
224        .collect(Collectors.toSet());
225  }
226
227  private <T> String quotedUserProvidedPath(Iterable<T> parts) {
228    return StreamSupport
229        .stream(parts.spliterator(), false)
230        .map(Objects::toString)
231        .collect(Collectors.collectingAndThen(
232            Collectors.joining("/"),
233            StringUtils::quoted
234        ));
235  }
236
237  private <T> String fuzzySafePath(Iterable<T> parts) {
238    // Join on null bytes as we don't ever use those in normal path names. This way, we ignore
239    // file-system and OS-specific path separators creating ambiguities (like how Windows uses
240    // backslashes rather than forward slashes to delimit paths).
241    return StreamSupport
242        .stream(parts.spliterator(), false)
243        .map(Objects::toString)
244        .collect(Collectors.joining("\0"));
245  }
246
247  private boolean fileNameIsPresent(@Nullable Path path) {
248    // Path can be null if no path elements exist in ZipPath implementations.
249    return Optional
250        .ofNullable(path)
251        .map(Path::getFileName)
252        .map(Path::toString)
253        .filter(not(String::isBlank))
254        .isPresent();
255  }
256}