之前不是写了篇收集报错日志上传ftp服务器的文章,那时候因为某些原因,我只有权限上传到ftp服务器上,但是在压测的时候发现很多问题,连接打不开,因为ftp有权限设置,还有连接人数上限,这就尴尬了,这就导致很多报错信息没有收到。这完全不符合需求,因此,需要后台PHP来处理。
PHP来处理就非常简单了,只需要上传字符串就行。性能上也优化了,皆大欢喜。代码就很简单,不说什么了。
/// <summary>
/// 上传错误日志
/// </summary>
public void UploadErrorToPHP(string error)
{
if (!Main.bIsCollectLog) return;
StringBuilder sb = new StringBuilder();
sb.Append(ErrorUploadUrl + "&error=" + error);
sb.Append("&uname=" + account);
System.DateTime nowTime = System.DateTime.Now;
var filename = "Unity_" + (nowTime.ToString("HH_mm_ss"));
sb.Append("&filename=" + filename);
#if UNITY_EDITOR
string system = "PC";
#elif UNITY_ANDROID
string system = "Android";
#elif UNITY_IOS
string system = "IOS";
#endif
sb.Append("&system=" + system);
RequestHttp(sb.ToString(), requetTimeOut);
}
private string RequestHttp(string url, int Timeout = 100000)
{
string requestPath = url;
string result = "";
HttpWebRequest httpWebReq = null;
Stream stream = null;
StreamReader reader = null;
try
{
httpWebReq = (HttpWebRequest)WebRequest.Create(requestPath);
//Debug.LogError(" httpWebReq.Timeout: " + httpWebReq.Timeout);
httpWebReq.Timeout = Timeout;
httpWebReq.KeepAlive = false;
stream = httpWebReq.GetResponse().GetResponseStream();
reader = new StreamReader(stream, Encoding.UTF8);
string line = string.Empty;
while (line != null)
{
line = reader.ReadLine();
result = result + line + "\n";
}
result = Decrypt(result);
reader.Close();
stream.Close();
}
catch (Exception e)
{
Debug.Log("RequestHttp In Error. url:" + url + "--" + e.Message);
}
finally
{
if (stream != null)
stream.Close();
if (reader != null)
reader.Close();
if (httpWebReq != null)
httpWebReq.Abort();
}
return result;
}
有了报错日志,需求又增加了,想要获取崩溃日志。但是这边又不想用Bugly这些插件,会影响性能,之前测试过,有降帧。所以只好在Java层上在实现一套上传崩溃日志功能,因为现在安卓版本在测试,所以安卓功能优先,就意味下次我还得处理一套iOS的功能,尴尬了。
之前有写过怎么获取java层的单例,这边也不再多说一遍,还是在之前写的安卓通用类上加了个接口。
开启接口
/**
* 开启java层上传异常信息
*/
public void InitCatchException()
{
if(ec != null)
return;
ec = new exceptioncatch();
ec.Init();
}
/**
* 修改java层上传异常信息的账号名
*/
public void SetAccount(String account)
{
if(ec != null)
{
ec.SetAccount(account);
}
}
需要注意的是要调用SystemClock.sleep(requetTimeOut)这个接口延迟一段时间关闭进程,不然还没上传数据到PHP就被关闭了。这个问题就导致我调了3,4个小时。在使用一段时间后,发现几乎没有收集到Java层的报错,之前写了个Android Demo测试,可以捕获到异常的,没有用Unity来测试,故意写个异常出来,Unity会捕获变成Unity层的错误,然后就调用了C#的上传报错日志接口。只好在网上继续搜索查找,然后找到了一个重要原因:setUncaughtExceptionHandler只对应注册的线程起作用,对应的线程,这就尴尬了,果然不熟悉Android和Java,后果太严重了。也查找到了setDefaultUncaughtExceptionHandler则是在所有线程上都起作用。换了接口后,一个下午就收集到了Java上的报错信息。又是一个坑被我趟过。
public class exceptioncatch implements Thread.UncaughtExceptionHandler {
private Thread.UncaughtExceptionHandler defaultThread;
private final String ErrorUploadUrl = "http://xxxxxxxxxxxxxx.php?";
private String account = "Login";
private int requetTimeOut = 5000;
public void Init()
{
Thread.setDefaultUncaughtExceptionHandler(this);//所有线程上都起作用
//Thread.currentThread().setUncaughtExceptionHandler(this);//只对应注册的线程起作用
defaultThread = Thread.getDefaultUncaughtExceptionHandler();
Log.e("ZP", "启动异常捕获");
}
public void SetAccount(String accoutStr)
{
account = accoutStr;
}
@Override
public void uncaughtException(Thread t, Throwable ex)
{
Log.e("ZP", "捕捉到了异常");
StringWriter sw = new StringWriter();
PrintWriter pw = new PrintWriter(sw);
ex.printStackTrace(pw);
pw.close();
UploadErrorToPHP(sw.toString());
SystemClock.sleep(requetTimeOut);
defaultThread.uncaughtException(t, ex);
}
private void UploadErrorToPHP(String error)
{
Date date = new Date();
SimpleDateFormat sdf = new SimpleDateFormat("(yyyy-mm-dd:HH:mm:ss)");
error = "[Java异常" + sdf.format(date) + "]" + error;
StringBuffer sb = new StringBuffer();
sb.append(ErrorUploadUrl + "error=" + error);
sb.append("&uname=" + account);
sdf = new SimpleDateFormat("HH_mm_ss");
String filename = "Java_" + sdf.format(date);
sb.append("&filename=" + filename);
sb.append("&system=Java");
RequestHttp(sb.toString());
}
private void RequestHttp(String requestUrl)
{
urlThread tempUrlThread = new urlThread();
tempUrlThread.SetRequestUrl(requestUrl);
Thread tempThread = new Thread(tempUrlThread);
tempThread.start();
}
上传到PHP
public class urlThread implements Runnable {
private String requestUrl;
public void SetRequestUrl(String requestUrl)
{
this.requestUrl = requestUrl;
}
@Override
public void run() {
Log.e("ZP","开始上传报错:" + requestUrl);
if(requestUrl != null && !requestUrl.isEmpty()) {
try {
URL url = new URL(requestUrl);
//返回一个URLConnection对象,它表示到URL所引用的远程对象的连接
URLConnection connection = url.openConnection();
connection.connect();
InputStream is = connection.getInputStream();
Log.e("ZP","完成上传报错");
} catch (Exception e) {
e.printStackTrace();
Log.e("ZP","上传exception:" + e.toString());
}
finally {
requestUrl = null;
}
}
}
}
Java我不是很熟悉,很多我都是按照C#的逻辑来实现的,找几个类似C#的方法来实现的,比如java的url我就是照着c#的webrequest来的,总计为了实现这个功能踩了还是蛮多的坑,记录下来防止以后继续踩坑。突然发现自己在Unity上写功能的时间少了很多,都是在弄这种奇奇怪怪的需求。