package jdk.jfr.internal.management;
import java.io.IOException;
import java.io.InputStream;
import java.time.Instant;
import java.util.Date;
import java.util.HashMap;
import java.util.Map;
import java.util.Timer;
import java.util.concurrent.TimeUnit;
import jdk.jfr.Recording;
import jdk.jfr.internal.consumer.FinishedStream;
import jdk.jfr.internal.consumer.OngoingStream;
public final class StreamManager {
public static final long TIME_OUT = TimeUnit.MINUTES.toMillis(200);
public static final int DEFAULT_BLOCK_SIZE = 50000;
private final Map<Long, EventByteStream> streams = new HashMap<>();
private Timer timer;
public synchronized EventByteStream getStream(long streamIdentifer) {
EventByteStream stream = streams.get(streamIdentifer);
if (stream == null) {
throw new IllegalArgumentException("Unknown stream identifier " + streamIdentifer);
}
return stream;
}
public synchronized EventByteStream createOngoing(Recording recording, int blockSize, Instant startTime, Instant endTime) {
long startTimeNanos = 0;
long endTimeNanos = Long.MAX_VALUE;
if (!startTime.equals(Instant.MIN)) {
startTimeNanos = startTime.getEpochSecond() * 1_000_000_000L;
startTimeNanos += startTime.getNano();
}
if (!endTime.equals(Instant.MAX)) {
endTimeNanos = endTime.getEpochSecond() * 1_000_000_000L;
endTimeNanos+= endTime.getNano();
}
EventByteStream stream = EventByteStream.newOngoingStream(recording, blockSize, startTimeNanos, endTimeNanos);
streams.put(stream.getId(), stream);
scheduleAbort(stream, System.currentTimeMillis() + TIME_OUT);
return stream;
}
public synchronized EventByteStream create(InputStream is, int blockSize) {
EventByteStream stream = EventByteStream.newFinishedStream(is, blockSize);
streams.put(stream.getId(), stream);
scheduleAbort(stream, System.currentTimeMillis() + TIME_OUT);
return stream;
}
public synchronized void destroy(EventByteStream stream) {
try {
stream.close();
} catch (IOException e) {
}
streams.remove(stream.getId());
if (streams.isEmpty()) {
timer.cancel();
timer = null;
}
}
public synchronized void scheduleAbort(EventByteStream s, long when) {
if (timer == null) {
timer = new Timer(true);
}
timer.schedule(new StreamCleanupTask(this, s), new Date(when + TIME_OUT));
}
}