blob: 03daf7a5b3742ffea01114f9cd036dd8faeee99e [file] [log] [blame]
/**
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You under the Apache License, Version 2.0
* (the "License"); you may not use this file except in compliance with
* the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.apache.geronimo.microprofile.reporter.storage.plugins.tracing;
import static java.util.Optional.ofNullable;
import static java.util.stream.Collectors.toList;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.lang.reflect.ParameterizedType;
import java.util.Collection;
import java.util.Collections;
import java.util.Map;
import javax.annotation.PostConstruct;
import javax.enterprise.context.ApplicationScoped;
import io.opentracing.Span;
@ApplicationScoped
public class SpanMapper {
private boolean active;
private Method getSpanId;
private Method getTraceId;
private Method getParentId;
private Method getName;
private Method getTimestamp;
private Method getDuration;
private Method getKind;
private Method getTags;
private Method getLogs;
private Method logGetTimestampMicros;
private Method logGetFields;
@PostConstruct
private void init() {
try {
final Class<?> spanImpl = Thread.currentThread().getContextClassLoader()
.loadClass("org.apache.geronimo.microprofile.opentracing.common.impl.SpanImpl");
getTraceId = spanImpl.getMethod("getTraceId");
getSpanId = spanImpl.getMethod("getId");
getParentId = spanImpl.getMethod("getParentId");
getName = spanImpl.getMethod("getName");
getTimestamp = spanImpl.getMethod("getTimestamp");
getDuration = spanImpl.getMethod("getDuration");
getKind = spanImpl.getMethod("getKind");
getTags = spanImpl.getMethod("getTags");
getLogs = spanImpl.getMethod("getLogs");
final Class<?> logType = Class.class
.cast(ParameterizedType.class.cast(getLogs.getGenericReturnType()).getActualTypeArguments()[0]);
logGetTimestampMicros = logType.getMethod("getTimestampMicros");
logGetFields = logType.getMethod("getFields");
active = true;
} catch (final Exception e) {
active = false;
}
}
public SpanEntry map(final Span span) {
if (!active) {
return null;
}
try {
final Collection<SpanEntry.LogEntry> logs = ofNullable((Collection<?>) getLogs.invoke(span))
.map(it -> it.stream().map(log -> {
try {
return new SpanEntry.LogEntry(Long.class.cast(logGetTimestampMicros.invoke(log)), Map.class.cast(logGetFields.invoke(log)));
} catch (final IllegalAccessException e) {
throw new IllegalStateException(e);
} catch (final InvocationTargetException e) {
throw new IllegalStateException(e.getTargetException());
}
}).collect(toList()))
.orElseGet(Collections::emptyList);
return new SpanEntry(
stringify(getSpanId.invoke(span)),
stringify(getTraceId.invoke(span)),
stringify(getParentId.invoke(span)),
stringify(getName.invoke(span)),
Long.class.cast(getTimestamp.invoke(span)),
Long.class.cast(getDuration.invoke(span)),
stringify(getKind.invoke(span)),
Map.class.cast(getTags.invoke(span)), logs);
} catch (final IllegalAccessException e) {
throw new IllegalStateException(e);
} catch (final InvocationTargetException e) {
throw new IllegalStateException(e.getTargetException());
}
}
private static String stringify(final Object value) {
return value == null ? null : String.valueOf(value);
}
}