eleme.openapi.sdk.media.trace.Reporter Maven / Gradle / Ivy
The newest version!
package eleme.openapi.sdk.media.trace;
import eleme.openapi.sdk.media.MediaConfiguration;
import eleme.openapi.sdk.media.__Version__;
import eleme.openapi.sdk.media.utils.IpUtils;
import java.io.File;
import java.net.URL;
import java.util.ArrayList;
import java.util.List;
import java.util.Map;
import java.util.Queue;
import java.util.Random;
import java.util.Timer;
import java.util.TimerTask;
import java.util.concurrent.ConcurrentLinkedQueue;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicLong;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;
import static eleme.openapi.sdk.utils.StringUtils.isEmpty;
/**
* 不保证不丢数据,尽可能上报
* Created by huamulou on 15/12/9.
*/
public class Reporter {
/**
* 默认会对上传结果汇报,用于上传性能跟踪和优化
*/
private volatile boolean close = false; //是否暂时关闭跟踪, 可以恢复
private volatile boolean shutdown = false; //是否关闭整个类, 无法恢复
private Timer timer; //定时任务, 定时触发汇报线程
private Thread worker; //汇报线程
private volatile long lastLaunchTime; //只有单线程修改, 上次汇报线程启动的时间
private final byte[] lock; //汇报线程锁
private Queue queue; //需要汇报的队列
private ReportClient reportService; //汇报的http服务
private DeviceId deviceId; //机器id, 用于uploadId
private String reportStartTime; //本类开始的时间, 用于uploadId的一部分
private AtomicLong reportSequence; //上传序号, 用于uploadId
private boolean inited = false; //reporter是否已经初始化
private boolean needDebugLog = false; //是否开启debuglog
private CountDownLatch workerExitCountDownLatch; //用于graceful地退出
private static final String DEFAULT_DEVICE_ID_FILE_NAME = "ams_device"; //机器id存储的文件名
private Lock queueLock = new ReentrantLock();
private volatile int queueSize = 0;
private static Reporter instance = new Reporter();
public static Reporter sharedInstance() {
if (!instance.inited) {
synchronized (instance) {
if (!instance.inited) {
instance.init();
}
}
}
return instance;
}
private String getLocalPath() {
String path;
URL base = getClass().getClassLoader().getResource("");
if (base == null) {
base = getClass().getClassLoader().getResource("/");
}
if (base == null) {
path = System.getProperty("user.dir");
} else {
path = base.getPath();
}
path = path.endsWith(File.separator) ? path : path + File.separator;
D("device id path is " + path);
return path;
}
private Reporter() {
this.deviceId = new DeviceId(getLocalPath() + DEFAULT_DEVICE_ID_FILE_NAME);
this.timer = new Timer("ams-report-timer");
this.lock = new byte[1];
Random random = new Random();
this.queue = new ConcurrentLinkedQueue();
this.lastLaunchTime = 0;
this.reportStartTime = Numbers.toString(Long.parseLong(System.currentTimeMillis() / 1000 + "" + random.nextInt(100000)), Numbers.MAX_RADIX);
this.reportSequence = new AtomicLong(0);
this.workerExitCountDownLatch = new CountDownLatch(1);
this.reportService = new ReportClient();
Runtime.getRuntime().addShutdownHook(new Thread(new Runnable() {
@Override
public void run() {
shutdown();
}
}));
}
public void init() {
this.timer.schedule(new TimerTask() {
@Override
public void run() {
if (!shutdown) { //如果close, 照样触发. close的时候只要不接消息就好了.
D("try trigger notify");
if (lastLaunchTime == 0) {
D("try trigger notify success");
notifyAllWaiters();
} else {
if (System.currentTimeMillis() - lastLaunchTime > 15 * 1000) {
D("try trigger notify success");
notifyAllWaiters();
}
}
}
}
}, 10 * 1000, 5 * 1000);
this.createAndStartDispatchThread();
this.inited = true;
}
public void shutdown() {
D("try shutdown gracefully");
this.shutdown = true;
try {
this.notifyAllWaiters();
this.workerExitCountDownLatch.await(10, TimeUnit.SECONDS);
D("shutdown gracefully success");
} catch (InterruptedException e) {
//
D("interrupt at shutdown", e);
}
this.timer.cancel();
D("timer cancel success");
}
private static class WorkerThreadExceptionHandler implements Thread.UncaughtExceptionHandler {
final Reporter reporter;
public WorkerThreadExceptionHandler(final Reporter reporter) {
this.reporter = reporter;
}
@Override
public void uncaughtException(final Thread t, final Throwable e) {
if (!this.reporter.isShutdown())
this.reporter.createAndStartDispatchThread();
}
}
private void createAndStartDispatchThread() {
this.worker = createDispatchThread();
this.worker.setName("ams-report-worker");
this.worker.setUncaughtExceptionHandler(new WorkerThreadExceptionHandler(this));
this.worker.start();
}
protected Thread createDispatchThread() {
return new Thread(new Runnable() {
@Override
public void run() {
while (true) {
D("entering running loop.");
if (shutdown && queue.isEmpty()) {
D("break by shutdown or queue size empty.");//没有同步, 可能丢数据
break;
}
try {
waitOnLock();
if (System.currentTimeMillis() - lastLaunchTime < 5 * 1000 && !shutdown) {
continue;
}
lastLaunchTime = System.currentTimeMillis();
while (true && !queue.isEmpty()) {
D("send report loop.");
int count = 0;
List reportList = new ArrayList();
while (!queue.isEmpty() && count < 100) { //每次发送不超过100的报告
count++;
String report = getReport();
if (report == null) {
break;
}
reportList.add(report);
}
try {
if (!reportList.isEmpty()) {
D("send {0} reports.", count);
reportService.report(reportList);
}
} catch (Exception e) {
D("exception happened when send report. message: " + e.getMessage(), e);
if (shutdown) {
D("break by shutdown and exception");
break;
} else {
for (String report : reportList) {
addReport(report);
}
D("sleep 1 minute before retry");
Thread.sleep(1 * 60 * 1000); //线程在睡觉的时候可能导致提交的数据丢失 TODO
}
}
}
} catch (InterruptedException e) {
//如果interrupt,忽略
D("running loop interrupted", e);
}
}
workerExitCountDownLatch.countDown();
}
});
}
public void setTraceOn(boolean traceOn) {
D("set trace on: " + traceOn);
close = !traceOn;
}
public enum ArgsFactor {
CODE("code"), MESSAGE("message"), UPLOAD_SIZE("uploadSize"), TOTAL_TIME("totalTime"),
DNS("dns"), OPEN_CONN("openConn"), FIRST_PACKET_ARRIVE("firstPacketArrive");
String desc;
ArgsFactor(String desc) {
this.desc = desc;
}
}
public enum ReportFactor {
VERSION,
OS,
OS_VERSION,
SDK_TYPE,
SDK_VERSION,
SESSION_ID,
RECORD_DATE,
EVENT_ID,
LOCAL_IP,
REQUEST_ID,
APPKEY,
ARGS
}
public enum OP {
UPLOAD("upload"), BLOCK_INIT("blockInit"), BLOCK_UPLOAD("blockUpload"), BLOCK_COMPLETE("blockComplete"), BLOCK_CANCEL("blockCancel"),
DNS("dns"), OPEN_CONN("openConn"), FIRST_PACKET_ARRIVE("firstPacketArrive");
public String desc;
OP(String desc) {
this.desc = desc;
}
}
/**
* 每次记录都要先start,这样同时对线程的脏数据进行清理
*/
public void start() {
if (close || shutdown) return;
Profiler.clear();
Profiler.setStarted(true);
}
public void addKV(String key, String value) {
if (isEmpty(key) || isEmpty(value)) {
return;
} else {
Profiler.getKVMap().put(key, value);
}
}
/**
* 对Profiler方法的封装 start
* @param op op
*/
public void enter(OP op) {
if (close || shutdown) return;
if (!Profiler.getStarted()) return;
Profiler.Factor factor = Profiler.getParameter(op.name());
factor = factor == null ? new Profiler.Factor(-1) : factor;
factor.setStart(System.currentTimeMillis());
Profiler.setParameter(op.name(), factor);
D("start trace of op {0}", op.name());
}
public void release(OP op) {
if (close || shutdown) return;
if (!Profiler.getStarted()) return;
Profiler.Factor factor = Profiler.getParameter(op.name());
if (factor != null)
factor.setEnd(System.currentTimeMillis());
}
private String getAk() {
if (MediaConfiguration.getLastConfiguration() == null) {
return null;
}
return MediaConfiguration.getLastConfiguration().getAk();
}
public void finish(String requestId, String uploadId, String message, String code, long uploadSize) {
if (close || shutdown) return;
D("finish trace");
ReportStore reportNew = new ReportStore();
reportNew.addFactor(ReportFactor.SDK_VERSION, __Version__.SDK_VERSION);
reportNew.addFactor(ReportFactor.SDK_TYPE, __Version__.SDK_TYPE);
reportNew.addFactor(ReportFactor.OS, System.getProperty("os.name"));
reportNew.addFactor(ReportFactor.OS_VERSION, System.getProperty("os.version"));
reportNew.addFactor(ReportFactor.APPKEY, getAk());
reportNew.addFactor(ReportFactor.RECORD_DATE, System.currentTimeMillis());
reportNew.addFactor(ReportFactor.LOCAL_IP, IpUtils.getIpAddress());
reportNew.addFactor(ReportFactor.REQUEST_ID, requestId);
reportNew.addFactor(ReportFactor.SESSION_ID, uploadId);
reportNew.addFactor(ReportFactor.VERSION, "1.0");
reportNew.addArgs(ArgsFactor.CODE, code);
reportNew.addArgs(ArgsFactor.MESSAGE, message);
reportNew.addArgs(ArgsFactor.UPLOAD_SIZE, uploadSize);
if (Profiler.getMap() != null) {
for (Map.Entry e : Profiler.getMap().entrySet()) {
if (e.getKey() == null || e.getValue() == null) continue;
OP op = OP.valueOf(e.getKey());
switch (op) {
case UPLOAD:
reportNew.addArgs(ArgsFactor.TOTAL_TIME, e.getValue().getInterval());
reportNew.addFactor(ReportFactor.EVENT_ID, op.desc);
break;
case BLOCK_INIT:
reportNew.addArgs(ArgsFactor.TOTAL_TIME, e.getValue().getInterval());
reportNew.addFactor(ReportFactor.EVENT_ID, op.desc);
break;
case BLOCK_UPLOAD:
reportNew.addArgs(ArgsFactor.TOTAL_TIME, e.getValue().getInterval());
reportNew.addFactor(ReportFactor.EVENT_ID, op.desc);
break;
case BLOCK_COMPLETE:
reportNew.addArgs(ArgsFactor.TOTAL_TIME, e.getValue().getInterval());
reportNew.addFactor(ReportFactor.EVENT_ID, op.desc);
break;
case BLOCK_CANCEL:
reportNew.addArgs(ArgsFactor.TOTAL_TIME, e.getValue().getInterval());
reportNew.addFactor(ReportFactor.EVENT_ID, op.desc);
break;
default:
reportNew.addArgs(op.desc, e.getValue().getInterval());
break;
}
}
}
if (Profiler.getKVMap() != null) {
for (Map.Entry e : Profiler.getKVMap().entrySet()) {
reportNew.addArgs(e.getKey(), e.getValue());
}
}
D("add report: {0}", reportNew.getReport());
Profiler.clear();
this.addReport(reportNew.getReport());
}
public String getSessionId() {
this.reportSequence.incrementAndGet();
return this.deviceId.getValue() + "-" + this.reportStartTime + "-" + reportSequence.get();
}
private void addReport(String report) {
if (close || shutdown) return;
if (report == null) return;
//size方法需要遍历,速度比较慢
queueLock.lock();
try {
if (queueSize > 10240) {
return; //如果内存累积过多,直接丢弃
}
boolean addResult = queue.add(report);
if (addResult)
queueSize++;
if (queueSize > 100) {
notifyAllWaiters();
}
} finally {
queueLock.unlock();
}
}
private String getReport() {
queueLock.lock();
try {
String report = queue.poll();
if (report != null) {
queueSize--;
}
return report;
} finally {
queueLock.unlock();
}
}
private void waitOnLock() throws InterruptedException {
synchronized (lock) {
lock.wait();
}
}
private void notifyAllWaiters() {
synchronized (lock) {
lock.notifyAll();
}
}
private void D(String pattern, Object... args) {
}
private void D(String arg) {
}
private void D(String arg, Throwable t) {
}
public boolean isClose() {
return close;
}
public void setClose(boolean close) {
this.close = close;
}
public boolean isShutdown() {
return shutdown;
}
public void setShutdown(boolean shutdown) {
this.shutdown = shutdown;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy