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 java.util.Objects.requireNonNull;
019import static org.assertj.core.api.Assertions.assertThat;
020
021import io.github.ascopes.jct.diagnostics.TraceDiagnostic;
022import io.github.ascopes.jct.repr.TraceDiagnosticRepresentation;
023import java.util.Locale;
024import javax.tools.JavaFileObject;
025import org.assertj.core.api.AbstractAssert;
026import org.assertj.core.api.AbstractInstantAssert;
027import org.assertj.core.api.AbstractLongAssert;
028import org.assertj.core.api.AbstractStringAssert;
029import org.jspecify.annotations.Nullable;
030
031/**
032 * Assertions for an individual {@link TraceDiagnostic trace diagnostic}.
033 *
034 * @author Ashley Scopes
035 * @since 0.0.1
036 */
037public final class TraceDiagnosticAssert
038    extends AbstractAssert<TraceDiagnosticAssert, TraceDiagnostic<? extends JavaFileObject>> {
039
040  /**
041   * Initialize this assertion type.
042   *
043   * @param value the value to assert on.
044   */
045  @SuppressWarnings("DataFlowIssue")
046  public TraceDiagnosticAssert(@Nullable TraceDiagnostic<? extends JavaFileObject> value) {
047    super(value, TraceDiagnosticAssert.class);
048    info.useRepresentation(TraceDiagnosticRepresentation.getInstance());
049  }
050
051  /**
052   * Get assertions for the kind of the diagnostic.
053   *
054   * @return the assertions for the diagnostic kind.
055   * @throws AssertionError if the diagnostic is null.
056   */
057  public DiagnosticKindAssert kind() {
058    isNotNull();
059
060    return new DiagnosticKindAssert(actual.getKind());
061  }
062
063  /**
064   * Get assertions for the source of the diagnostic.
065   *
066   * <p>If no source is present, then the value in the returned assertions may be {@code null}.
067   *
068   * @return the assertions for the source of the diagnostic.
069   * @throws AssertionError if the diagnostic is null.
070   */
071  public JavaFileObjectAssert source() {
072    isNotNull();
073
074    return new JavaFileObjectAssert(actual.getSource());
075  }
076
077  /**
078   * Get assertions for the position of the diagnostic.
079   *
080   * <p>The value may be -1 if no information is available.
081   *
082   * @return the assertions for the position of the diagnostic.
083   * @throws AssertionError if the diagnostic is null.
084   */
085  public AbstractLongAssert<?> position() {
086    isNotNull();
087
088    return assertThat(actual.getPosition()).describedAs("position");
089  }
090
091  /**
092   * Get assertions for the start position of the diagnostic.
093   *
094   * <p>The value may be -1 if no information is available.
095   *
096   * @return the assertions for the start position of the diagnostic.
097   * @throws AssertionError if the diagnostic is null.
098   */
099  public AbstractLongAssert<?> startPosition() {
100    isNotNull();
101
102    return assertThat(actual.getStartPosition()).describedAs("start position");
103  }
104
105  /**
106   * Get assertions for the end position of the diagnostic.
107   *
108   * <p>The value may be -1 if no information is available.
109   *
110   * @return the assertions for the end position of the diagnostic.
111   * @throws AssertionError if the diagnostic is null.
112   */
113  public AbstractLongAssert<?> endPosition() {
114    isNotNull();
115
116    return assertThat(actual.getEndPosition()).describedAs("end position");
117  }
118
119  /**
120   * Get assertions for the line number of the diagnostic.
121   *
122   * <p>The value may be -1 if no information is available.
123   *
124   * @return the assertions for the line number of the diagnostic.
125   * @throws AssertionError if the diagnostic is null.
126   */
127  public AbstractLongAssert<?> lineNumber() {
128    isNotNull();
129
130    return assertThat(actual.getLineNumber()).describedAs("line number");
131  }
132
133  /**
134   * Get assertions for the column number of the diagnostic.
135   *
136   * <p>The value may be -1 if no information is available.
137   *
138   * @return the assertions for the column number of the diagnostic.
139   * @throws AssertionError if the diagnostic is null.
140   */
141  public AbstractLongAssert<?> columnNumber() {
142    isNotNull();
143
144    return assertThat(actual.getColumnNumber()).describedAs("column number");
145  }
146
147  /**
148   * Get assertions for the code of the diagnostic.
149   *
150   * @return the assertions for the code of the diagnostic.
151   * @throws AssertionError if the diagnostic is null.
152   */
153  public AbstractStringAssert<?> code() {
154    isNotNull();
155
156    return assertThat(actual.getCode()).describedAs("code");
157  }
158
159  /**
160   * Get assertions for the message of the diagnostic, assuming the default locale.
161   *
162   * @return the assertions for the message of the diagnostic.
163   * @throws AssertionError if the diagnostic is null.
164   */
165  public AbstractStringAssert<?> message() {
166    isNotNull();
167
168    return assertThat(actual.getMessage(null));
169  }
170
171  /**
172   * Get assertions for the message of the diagnostic.
173   *
174   * @param locale the locale to use.
175   * @return the assertions for the message of the diagnostic.
176   * @throws AssertionError if the diagnostic is null.
177   */
178  public AbstractStringAssert<?> message(Locale locale) {
179    requireNonNull(locale, "locale must not be null");
180
181    isNotNull();
182
183    return assertThat(actual.getMessage(locale));
184  }
185
186  /**
187   * Get assertions for the timestamp of the diagnostic.
188   *
189   * @return the assertions for the timestamp of the diagnostic.
190   * @throws AssertionError if the diagnostic is null.
191   */
192  public AbstractInstantAssert<?> timestamp() {
193    isNotNull();
194
195    return assertThat(actual.getTimestamp());
196  }
197
198  /**
199   * Get assertions for the thread ID of the thread that reported the diagnostic to the compiler.
200   *
201   * @return the assertions for the thread ID.
202   * @throws AssertionError if the diagnostic is null.
203   */
204  public AbstractLongAssert<?> threadId() {
205    isNotNull();
206
207    return assertThat(actual.getThreadId());
208  }
209
210  /**
211   * Get assertions for the thread name of the thread that reported the diagnostic.
212   *
213   * <p>This may not be present in some situations, in which case the returned assertions will be
214   * performed on a null value instead.
215   *
216   * @return the assertions for the thread name.
217   * @throws AssertionError if the diagnostic is null.
218   */
219  public AbstractStringAssert<?> threadName() {
220    isNotNull();
221
222    return assertThat(actual.getThreadName());
223  }
224
225  /**
226   * Get assertions for the stack trace of the location the diagnostic was reported to.
227   *
228   * @return the assertions for the stack trace.
229   * @throws AssertionError if the diagnostic is null.
230   */
231  public StackTraceAssert stackTrace() {
232    isNotNull();
233
234    return new StackTraceAssert(actual.getStackTrace());
235  }
236}