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.diagnostics; 017 018import static java.util.Collections.unmodifiableList; 019import static java.util.Objects.requireNonNull; 020 021import io.github.ascopes.jct.utils.ToStringBuilder; 022import java.time.Instant; 023import java.util.List; 024import java.util.Locale; 025import javax.tools.Diagnostic; 026import javax.tools.JavaFileObject; 027import org.jspecify.annotations.Nullable; 028 029/** 030 * A wrapper around a {@link Diagnostic} which contains additional information about where the 031 * diagnostic was reported. 032 * 033 * @param <S> the file type. 034 * @author Ashley Scopes 035 * @since 0.0.1 036 */ 037public class TraceDiagnostic<S extends JavaFileObject> implements Diagnostic<S> { 038 039 private final Instant timestamp; 040 private final long threadId; 041 private final @Nullable String threadName; 042 private final List<StackTraceElement> stackTrace; 043 private final Diagnostic<? extends S> original; 044 045 /** 046 * Initialize this diagnostic. 047 * 048 * @param timestamp the timestamp. 049 * @param threadId the thread ID. 050 * @param threadName the thread name, or {@code null} if not known. 051 * @param stackTrace the stacktrace. 052 * @param original the original diagnostic that was reported. 053 */ 054 public TraceDiagnostic( 055 Instant timestamp, 056 long threadId, 057 @Nullable String threadName, 058 List<StackTraceElement> stackTrace, 059 Diagnostic<? extends S> original 060 ) { 061 this.timestamp = requireNonNull(timestamp, "timestamp"); 062 this.threadId = threadId; 063 this.threadName = threadName; 064 this.stackTrace = unmodifiableList(requireNonNull(stackTrace, "stackTrace")); 065 this.original = requireNonNull(original, "original"); 066 } 067 068 @Override 069 public Kind getKind() { 070 return original.getKind(); 071 } 072 073 @Nullable 074 @Override 075 public S getSource() { 076 return original.getSource(); 077 } 078 079 @Override 080 public long getPosition() { 081 return original.getPosition(); 082 } 083 084 @Override 085 public long getStartPosition() { 086 return original.getStartPosition(); 087 } 088 089 @Override 090 public long getEndPosition() { 091 return original.getEndPosition(); 092 } 093 094 @Override 095 public long getLineNumber() { 096 return original.getLineNumber(); 097 } 098 099 @Override 100 public long getColumnNumber() { 101 return original.getColumnNumber(); 102 } 103 104 @Nullable 105 @Override 106 public String getCode() { 107 return original.getCode(); 108 } 109 110 @Override 111 public String getMessage(@Nullable Locale locale) { 112 return original.getMessage(locale); 113 } 114 115 /** 116 * Get the timestamp that the diagnostic was created at. 117 * 118 * @return the diagnostic timestamp. 119 */ 120 public Instant getTimestamp() { 121 return timestamp; 122 } 123 124 /** 125 * Get the thread ID for the thread that created this diagnostic. 126 * 127 * @return the thread ID. 128 */ 129 public long getThreadId() { 130 return threadId; 131 } 132 133 /** 134 * Get the thread name for the thread that created this diagnostic. 135 * 136 * @return the thread name, if known, or else {@code null}. 137 */ 138 @Nullable 139 public String getThreadName() { 140 return threadName; 141 } 142 143 /** 144 * Get the stacktrace of where the diagnostic was written from. 145 * 146 * @return the stacktrace, in an unmodifiable list. 147 */ 148 public List<StackTraceElement> getStackTrace() { 149 return stackTrace; 150 } 151 152 @Override 153 public String toString() { 154 return new ToStringBuilder(this) 155 .attribute("timestamp", timestamp) 156 .attribute("threadId", threadId) 157 .attribute("threadName", threadName) 158 .attribute("kind", original.getKind()) 159 .attribute("code", original.getCode()) 160 .attribute("column", original.getColumnNumber()) 161 .attribute("line", original.getLineNumber()) 162 .attribute("message", original.getMessage(Locale.ROOT)) 163 .toString(); 164 } 165}