AxibugEmuOnline/AxibugEmuOnline.Client/Assets/Script/AppMain/AxiHttp/AxiHttp.cs
2024-12-24 23:36:49 +08:00

1204 lines
30 KiB
C#
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

using System.Collections.Generic;
using System;
using System.Diagnostics;
using System.IO;
using System.IO.Compression;
using System.Net;
using System.Net.Security;
using System.Net.Sockets;
using System.Security.Authentication;
using System.Security.Cryptography.X509Certificates;
using System.Text;
using System.Text.RegularExpressions;
using System.Threading;
public static class PSVThread
{
static AutoResetEvent autoEvent = new AutoResetEvent(false);
static Queue<Action> qActs = new Queue<Action>();
static Queue<Action> qWork = new Queue<Action>();
public static void DoTask(Action act)
{
#if UNITY_PSP2
AddSingleTask(act);
#else
ThreadPool.QueueUserWorkItem(new WaitCallback((state) => act.Invoke()));
#endif
}
#if UNITY_PSP2
static Thread psvThread = new Thread(Loop);
static bool bSingleInit = false;
static void SingleInit()
{
if (bSingleInit) return;
psvThread.Start();
bSingleInit = true;
}
static void AddSingleTask(Action act)
{
SingleInit();
lock (qActs)
{
qActs.Enqueue(act);
}
autoEvent.Set();
}
static void Loop()
{
while (autoEvent.WaitOne())
{
lock (qActs)
{
while (qActs.Count > 0) { qWork.Enqueue(qActs.Dequeue()); }
}
while (qWork.Count > 0)
{
Action act = qWork.Dequeue();
try
{
act.Invoke();
}
catch (Exception ex)
{
UnityEngine.Debug.Log(ex.ToString());
}
}
}
}
#endif
}
public static class AxiHttp
{
public const char T = '\n';
public const string CT = "\r\n";
public const string CTRL = "\r\n\r\n";
public const string Content_Length_Str = "content-length: ";
public const string Content_Length_Str_M = "Content-Length: ";
public const string Content_Length = "content-length";
public const string Content_Encoding = "content-encoding";
public const string Transfer_Encoding = "transfer-encoding";
public const string Connection = "connection";
public static long index = 0;
static int singlePkgMaxRead = 1024;
public class WaitAxiRequest : UnityEngine.CustomYieldInstruction
{
public AxiRespInfo mReqAsync;
public WaitAxiRequest(AxiRespInfo reqAsync)
{
mReqAsync = reqAsync;
}
~WaitAxiRequest()
{
mReqAsync = null;
}
public override bool keepWaiting
{
get { return !mReqAsync.isDone; }
}
}
public static void Log(string log)
{
UnityEngine.Debug.Log(log);
//Console.WriteLine(log);
}
static Dictionary<string, IPAddress> dictIP2Address = new Dictionary<string, IPAddress>();
public class AxiRespInfo
{
public bool isDone = false;
public AxiDownLoadMode downloadMode = AxiDownLoadMode.NotDownLoad;
public string Err = null;
public string host = "";//host主机头
public string url = "";//pathAndQuery
public int port = 80;
public string requestRaw = "";
public string encoding = "";
public string header = "";
public string text { get { return body; } }
public string body = "";
public string reuqestBody = "";
public string reuqestHeader = "";
public Dictionary<string, string> headers = new Dictionary<string, string>();
public string response = "";
//public string gzip = "";
public bool isGzip = false;
public int length = 0;
public int code = 0;
public int location = 0;
public int runTime = 0;//获取网页消耗时间,毫秒
public int sleepTime = 0;//休息时间
public string cookies = "";
public bool bTimeOut = false;
public int NeedloadedLenght;
public int loadedLenght;
public byte[] data { get { return bodyRaw; } }
public byte[] bodyRaw;
public string fileName;
public float DownLoadPr =>
NeedloadedLenght <= 0 ? -1 : (float)loadedLenght / NeedloadedLenght;
public BinaryWriter binaryWriter;
}
public static IPAddress GetDnsIP(string str)
{
if (!dictIP2Address.ContainsKey(str))
{
try
{
IPAddress ip = Dns.GetHostEntry(str).AddressList[0];
dictIP2Address[str] = ip;
}
catch
{
return null;
}
}
return dictIP2Address[str];
}
public enum AxiDownLoadMode
{
NotDownLoad = 0,
DownLoadBytes = 1,
DownloadToBinaryWriter = 2
}
public static AxiRespInfo AxiRequest(string url)
{
AxiRespInfo respInfo = new AxiRespInfo();
respInfo.downloadMode = AxiDownLoadMode.NotDownLoad;
SendAxiRequest(url, ref respInfo);
return respInfo;
}
public static WaitAxiRequest AxiRequestAsync(string url)
{
AxiRespInfo respInfo = new AxiRespInfo();
respInfo.downloadMode = AxiDownLoadMode.NotDownLoad;
WaitAxiRequest respAsync = new WaitAxiRequest(respInfo);
//Task task = new Task(() => SendAxiRequest(url, ref respInfo));
//task.Start()
PSVThread.DoTask(() => SendAxiRequest(url, ref respInfo));
return respAsync;
}
public static AxiRespInfo AxiDownload(string url)
{
AxiRespInfo respInfo = new AxiRespInfo();
respInfo.downloadMode = AxiDownLoadMode.DownLoadBytes;
SendAxiRequest(url, ref respInfo);
return respInfo;
}
public static AxiRespInfo AxiDownloadAsync(string url)
{
AxiRespInfo respInfo = new AxiRespInfo();
respInfo.downloadMode = AxiDownLoadMode.DownLoadBytes;
//Task task = new Task(() => SendAxiRequest(url, ref respInfo));
//task.Start();
PSVThread.DoTask(() => SendAxiRequest(url, ref respInfo));
return respInfo;
}
static void SendAxiRequest(string url, ref AxiRespInfo respinfo, int timeout = 1000 * 1000, string encoding = "UTF-8")
{
if (url.ToLower().StartsWith("https://"))
SendAxiRequestHttps(url, ref respinfo, timeout, encoding);// SendAxiRequestHttps(url, ref respinfo, timeout, encoding);
else
SendAxiRequestHttp(url, ref respinfo, timeout, encoding);
}
static void SendAxiRequestHttp(string url, ref AxiRespInfo respinfo, int timeout, string encoding)
{
Log("SendAxiRequestHttp");
respinfo.url = url;
Stopwatch sw = new Stopwatch();
sw.Start();
respinfo.loadedLenght = 0;
try
{
string strURI = url;
string strHost = "";
string strIP = "";
int port = 0;
string strRelativePath = "";
bool bSSL = false;
bool foward_302 = true;
if (!ParseURI(strURI, ref bSSL, ref strHost, ref strIP, ref port, ref strRelativePath))
{
Log("ParseURI False");
respinfo.Err = "ParseURI False";
respinfo.code = 0;
respinfo.isDone = true;
return;
}
var ip = GetDnsIP(strHost);
var ipEndPoint = new IPEndPoint(ip, port);
using (Socket client = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp))
using (MemoryStream memoryStream = new MemoryStream())
{
client.Connect(ipEndPoint);
if (!client.Connected)
{
client.Close();
sw.Stop();
respinfo.code = 0;
respinfo.isDone = true;
return;
}
//string requestRaw = $"GET {strRelativePath} HTTP/1.1\r\nHost: {strHost}\r\nConnection: Close\r\n\r\n";
string request = $"GET {strURI} HTTP/1.1\r\nHost: {strHost}\r\nConnection: Close\r\n\r\n";
checkContentLength(ref respinfo, ref request);
respinfo.requestRaw = request;
byte[] temp_responseBody = new byte[singlePkgMaxRead];
byte[] buffer = Encoding.ASCII.GetBytes(request);
client.Send(buffer);
string tmp = "";
int len = 0;
StringBuilder sb = new StringBuilder();
do
{
byte[] responseHeader = new byte[1];
len = client.Receive(responseHeader, 1, SocketFlags.None);
if (len == 1)
{
char c = (char)responseHeader[0];
sb.Append(c);
if (c.Equals(T))
{
tmp = String.Concat(sb[sb.Length - 4], sb[sb.Length - 3], sb[sb.Length - 2], c);
}
}
} while (!tmp.Equals(CTRL)
&& sw.ElapsedMilliseconds < timeout
);
respinfo.header = sb.ToString().Replace(CTRL, "");
string[] headers = Regex.Split(respinfo.header, CT);
if (headers != null && headers.Length > 0)
{
//处理header
doHeader(ref respinfo, ref headers);
}
//自动修正编码
if (!String.IsNullOrEmpty(respinfo.encoding))
{
encoding = respinfo.encoding;
}
Encoding encod = Encoding.GetEncoding(encoding);
//302 301跳转
if ((respinfo.code == 302 || respinfo.code == 301) && foward_302)
{
StringBuilder rsb = new StringBuilder(respinfo.requestRaw);
int urlStart = respinfo.requestRaw.IndexOf(" ") + 1;
int urlEnd = respinfo.requestRaw.IndexOf(" HTTP");
if (urlStart != -1 && urlEnd != -1)
{
url = respinfo.requestRaw.Substring(urlStart, urlEnd - urlStart);
rsb.Remove(urlStart, url.Length);
String location = respinfo.headers["location"];
if (!respinfo.headers["location"].StartsWith("/") && !respinfo.headers["location"].StartsWith("http"))
{
location = Tools.getCurrentPath(url) + location;
}
rsb.Insert(urlStart, location);
//return sendHTTPRequest(count, host, port, payload, rsb.ToString(), timeout, encoding, false);
client.Close();
sw.Stop();
SendAxiRequest(url, ref respinfo, timeout, encoding);
return;
}
}
//根据请求头解析
if (respinfo.headers.ContainsKey(Content_Length))
{
Log("User Head");
int length = int.Parse(respinfo.headers[Content_Length]);
respinfo.NeedloadedLenght = length;
// while (respinfo.loadedLenght < length
// && sw.ElapsedMilliseconds < timeout
// )
//{
// int readsize = length - respinfo.loadedLenght;
// len = client.Receive(temp_responseBody, respinfo.loadedLenght, readsize, SocketFlags.None);
// if (len > 0)
// {
// respinfo.loadedLenght += len;
// }
//}
while (respinfo.loadedLenght < length
&& sw.ElapsedMilliseconds < timeout
)
{
//len = client.Receive(temp_responseBody, respinfo.loadedLenght, readsize, SocketFlags.None);
int readsize = length - respinfo.loadedLenght;
readsize = Math.Min(readsize, singlePkgMaxRead);
len = client.Receive(temp_responseBody, 0, readsize, SocketFlags.None);
if (len > 0)
{
memoryStream.Write(temp_responseBody, 0, len);
respinfo.loadedLenght += len;
}
}
}
//解析chunked传输
else if (respinfo.headers.ContainsKey(Transfer_Encoding))
{
Log("User chunked");
//读取长度
int chunkedSize = 0;
byte[] chunkedByte = new byte[1];
//读取总长度
respinfo.loadedLenght = 0;
do
{
string ctmp = "";
do
{
len = client.Receive(chunkedByte, 1, SocketFlags.None);
ctmp += Encoding.UTF8.GetString(chunkedByte);
} while ((ctmp.IndexOf(CT) == -1)
&& (sw.ElapsedMilliseconds < timeout)
);
chunkedSize = Tools.convertToIntBy16(ctmp.Replace(CT, ""));
//chunked的结束0\r\n\r\n是结束标志单个chunked块\r\n结束
if (ctmp.Equals(CT))
{
continue;
}
if (chunkedSize == 0)
{
//结束了
break;
}
//int onechunkLen = 0;
//while (onechunkLen < chunkedSize
// && sw.ElapsedMilliseconds < timeout
// )
//{
// len = client.Receive(responseBody, respinfo.loadedLenght, chunkedSize - onechunkLen, SocketFlags.None);
// if (len > 0)
// {
// onechunkLen += len;
// respinfo.loadedLenght += len;
// }
//}
int onechunkLen = 0;
while (onechunkLen < chunkedSize
&& sw.ElapsedMilliseconds < timeout
)
{
//len = client.Receive(responseBody, respinfo.loadedLenght, chunkedSize - onechunkLen, SocketFlags.None);
int readsize = chunkedSize - onechunkLen;
readsize = Math.Min(readsize, singlePkgMaxRead);
len = client.Receive(temp_responseBody, 0, readsize, SocketFlags.None);
if (len > 0)
{
memoryStream.Write(temp_responseBody, 0, len);
onechunkLen += len;
respinfo.loadedLenght += len;
}
}
//判断
} while (sw.ElapsedMilliseconds < timeout);
}
//connection close方式或未知body长度
else
{
Log("connection close or Unknow bodylenght");
while (sw.ElapsedMilliseconds < timeout)
{
if (client.Poll(timeout, SelectMode.SelectRead))
{
if (client.Available > 0)
{
//len = client.Receive(responseBody, respinfo.loadedLenght, (1024 * 200) - respinfo.loadedLenght, SocketFlags.None);
int readsize = (1024 * 200) - respinfo.loadedLenght;
readsize = Math.Min(readsize, singlePkgMaxRead);
len = client.Receive(temp_responseBody, 0, readsize, SocketFlags.None);
if (len > 0)
{
memoryStream.Write(temp_responseBody, 0, len);
respinfo.loadedLenght += len;
}
}
else
{
break;
}
}
}
}
byte[] responseBody = memoryStream.ToArray();
//如果是下载
if (respinfo.downloadMode > AxiDownLoadMode.NotDownLoad)
{
//判断是否gzip
if (respinfo.headers.ContainsKey(Content_Encoding))
{
respinfo.bodyRaw = unGzipBytes(responseBody, respinfo.loadedLenght);
}
else
{
respinfo.bodyRaw = responseBody;
}
// 使用Uri类解析URL
Uri uri = new Uri(url);
respinfo.fileName = Path.GetFileName(uri.LocalPath);
}
else
{
//判断是否gzip
if (respinfo.headers.ContainsKey(Content_Encoding))
{
respinfo.body = unGzip(responseBody, respinfo.loadedLenght, encod);
}
else
{
respinfo.body = encod.GetString(responseBody, 0, respinfo.loadedLenght);
}
}
client.Close();
}
}
catch (Exception ex)
{
respinfo.Err = $"ex : {ex.ToString()}";
}
finally
{
sw.Stop();
respinfo.length = respinfo.loadedLenght;
respinfo.runTime = (int)sw.ElapsedMilliseconds;
respinfo.bTimeOut = sw.ElapsedMilliseconds >= timeout;
//if (socket != null)
//{
// clientSocket.Close();
//}
respinfo.isDone = true;
}
}
static void SendAxiRequestHttps(string url, ref AxiRespInfo respinfo, int timeout, string encoding)
{
respinfo.url = url;
Stopwatch sw = new Stopwatch();
sw.Start();
respinfo.loadedLenght = 0;
TcpClient client = null;
try
{
string strURI = url;
string strHost = "";
string strIP = "";
int port = 0;
string strRelativePath = "";
bool bSSL = false;
bool foward_302 = true;
if (!ParseURI(strURI, ref bSSL, ref strHost, ref strIP, ref port, ref strRelativePath))
{
Log("ParseURI False");
respinfo.Err = "ParseURI False";
respinfo.code = 0;
respinfo.isDone = true;
return;
}
//var ip = Dns.GetHostEntry(strHost).AddressList[0];
//var ipEndPoint = new IPEndPoint(ip, port);
//using (Socket client = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp))
//using (TcpClient tcpclient = new TcpClient())
using (MemoryStream memoryStream = new MemoryStream())
{
//client.Connect(ipEndPoint);
TimeOutSocket tos = new TimeOutSocket();
client = tos.Connect(strHost, port, timeout);
if (!client.Connected)
{
client.Close();
sw.Stop();
respinfo.code = 0;
respinfo.isDone = true;
return;
}
SslStream ssl = null;
//string requestRaw = $"GET {strRelativePath} HTTP/1.1\r\nHost: {strHost}\r\nConnection: Close\r\n\r\n";
string request = $"GET {strURI} HTTP/1.1\r\nHost: {strHost}\r\nConnection: Close\r\n\r\n";
ssl = new SslStream(client.GetStream(), false, new RemoteCertificateValidationCallback(ValidateServerCertificate));
SslProtocols protocol = SslProtocols.Ssl3 | SslProtocols.Ssl2 | SslProtocols.Tls;
ssl.AuthenticateAsClient(strHost, null, protocol, false);
if (ssl.IsAuthenticated)
{
checkContentLength(ref respinfo, ref request);
respinfo.requestRaw = request;
byte[] requestByte = Encoding.UTF8.GetBytes(request);
ssl.Write(requestByte);
ssl.Flush();
}
checkContentLength(ref respinfo, ref request);
respinfo.requestRaw = request;
byte[] temp_responseBody = new byte[singlePkgMaxRead];
//byte[] buffer = Encoding.ASCII.GetBytes(requestRaw);
//client.Send(buffer);
string tmp = "";
int len = 0;
StringBuilder sb = new StringBuilder();
do
{
byte[] responseHeader = new byte[1];
int read = ssl.ReadByte();
char c = (char)read;
sb.Append(c);
if (c.Equals(T))
{
tmp = String.Concat(sb[sb.Length - 4], sb[sb.Length - 3], sb[sb.Length - 2], c);
}
} while (!tmp.Equals(CTRL)
&& sw.ElapsedMilliseconds < timeout
);
respinfo.header = sb.ToString().Replace(CTRL, "");
string[] headers = Regex.Split(respinfo.header, CT);
if (headers != null && headers.Length > 0)
{
//处理header
doHeader(ref respinfo, ref headers);
}
//自动修正编码
if (!String.IsNullOrEmpty(respinfo.encoding))
{
encoding = respinfo.encoding;
}
Encoding encod = Encoding.GetEncoding(encoding);
//302 301跳转
if ((respinfo.code == 302 || respinfo.code == 301) && foward_302)
{
int urlStart = respinfo.requestRaw.IndexOf(" ");
int urlEnd = respinfo.requestRaw.IndexOf(" HTTP");
if (urlStart != -1 && urlEnd != -1)
{
url = respinfo.requestRaw.Substring(urlStart + 1, urlEnd - urlStart - 1);
if (!respinfo.headers["location"].StartsWith("/") && !respinfo.headers["location"].StartsWith("https"))
{
respinfo.requestRaw = respinfo.requestRaw.Replace(url, Tools.getCurrentPath(url) + respinfo.headers["location"]);
}
else
{
respinfo.requestRaw = respinfo.requestRaw.Replace(url, respinfo.headers["location"]);
}
//return sendHTTPRequest(count, host, port, payload, rsb.ToString(), timeout, encoding, false);
client.Close();
sw.Stop();
SendAxiRequest(url, ref respinfo, timeout, encoding);
return;
}
}
//根据请求头解析
if (respinfo.headers.ContainsKey(Content_Length))
{
Log("Use Head");
int length = int.Parse(respinfo.headers[Content_Length]);
respinfo.NeedloadedLenght = length;
while (respinfo.loadedLenght < length && sw.ElapsedMilliseconds < timeout)
{
//len = ssl.Read(responseBody, respinfo.loadedLenght, length - respinfo.loadedLenght);
int readsize = length - respinfo.loadedLenght;
readsize = Math.Min(readsize, singlePkgMaxRead);
len = ssl.Read(temp_responseBody, 0, readsize);
if (len > 0)
{
memoryStream.Write(temp_responseBody, 0, len);
respinfo.loadedLenght += len;
}
}
}
//解析chunked传输
else if (respinfo.headers.ContainsKey(Transfer_Encoding))
{
Log("User chunked");
//读取长度
int chunkedSize = 0;
byte[] chunkedByte = new byte[1];
//读取总长度
respinfo.loadedLenght = 0;
do
{
string ctmp = "";
do
{
len = ssl.Read(chunkedByte, 0, 1);
ctmp += Encoding.UTF8.GetString(chunkedByte);
} while (ctmp.IndexOf(CT) == -1 && sw.ElapsedMilliseconds < timeout);
chunkedSize = Tools.convertToIntBy16(ctmp.Replace(CT, ""));
//chunked的结束0\r\n\r\n是结束标志单个chunked块\r\n结束
if (ctmp.Equals(CT))
{
continue;
}
if (chunkedSize == 0)
{
//结束了
break;
}
int onechunkLen = 0;
while (onechunkLen < chunkedSize && sw.ElapsedMilliseconds < timeout)
{
//len = ssl.Read(responseBody, respinfo.loadedLenght, chunkedSize - onechunkLen);
int readsize = chunkedSize - onechunkLen;
readsize = Math.Min(readsize, singlePkgMaxRead);
len = ssl.Read(temp_responseBody, 0, readsize);
if (len > 0)
{
memoryStream.Write(temp_responseBody, 0, len);
onechunkLen += len;
respinfo.loadedLenght += len;
}
}
//判断
} while (sw.ElapsedMilliseconds < timeout);
}
//connection close方式或未知body长度
else
{
Log("connection close or Unknow bodylenght");
while (sw.ElapsedMilliseconds < timeout)
{
while (sw.ElapsedMilliseconds < timeout)
{
if (client.Client.Poll(timeout, SelectMode.SelectRead))
{
if (client.Available > 0)
{
//len = ssl.Read(responseBody, respinfo.loadedLenght, (1024 * 200) - respinfo.loadedLenght);
int readsize = (1024 * 200) - respinfo.loadedLenght;
readsize = Math.Min(readsize, singlePkgMaxRead);
len = ssl.Read(temp_responseBody, 0, readsize);
if (len > 0)
{
memoryStream.Write(temp_responseBody, 0, len);
respinfo.loadedLenght += len;
}
}
else
{
break;
}
}
}
}
}
byte[] responseBody = memoryStream.ToArray();
//如果是下载
if (respinfo.downloadMode > AxiDownLoadMode.NotDownLoad)
{
//判断是否gzip
if (respinfo.isGzip)
{
respinfo.bodyRaw = unGzipBytes(responseBody, respinfo.loadedLenght);
}
else
{
respinfo.bodyRaw = responseBody;
}
// 使用Uri类解析URL
Uri uri = new Uri(url);
respinfo.fileName = Path.GetFileName(uri.LocalPath);
}
else
{
//判断是否gzip
if (respinfo.isGzip)
{
respinfo.body = unGzip(responseBody, respinfo.loadedLenght, encod);
}
else
{
respinfo.body = encod.GetString(responseBody, 0, respinfo.loadedLenght);
}
}
}
}
catch (Exception ex)
{
respinfo.Err = $"ex : {ex.ToString()}";
}
finally
{
client?.Close();
sw.Stop();
respinfo.length = respinfo.loadedLenght;
respinfo.runTime = (int)sw.ElapsedMilliseconds;
respinfo.bTimeOut = sw.ElapsedMilliseconds >= timeout;
//if (socket != null)
//{
// clientSocket.Close();
//}
respinfo.isDone = true;
}
if (client != null)
client.Dispose();
}
private static void doHeader(ref AxiRespInfo respinfo, ref string[] headers)
{
for (int i = 0; i < headers.Length; i++)
{
if (i == 0)
{
respinfo.code = Tools.convertToInt(headers[i].Split(' ')[1]);
}
else
{
String[] kv = Regex.Split(headers[i], ": ");
String key = kv[0].ToLower();
if (!respinfo.headers.ContainsKey(key))
{
//自动识别编码
if ("content-type".Equals(key))
{
String hecnode = getHTMLEncoding(kv[1], "");
if (!String.IsNullOrEmpty(hecnode))
{
respinfo.encoding = hecnode;
}
}
if (kv.Length > 1)
{
respinfo.headers.Add(key, kv[1]);
}
else
{
respinfo.headers.Add(key, "");
}
}
}
respinfo.isGzip = respinfo.headers.ContainsKey(Content_Encoding);
}
}
public static string unGzip(byte[] data, int len, Encoding encoding)
{
string str = "";
MemoryStream ms = new MemoryStream(data, 0, len);
GZipStream gs = new GZipStream(ms, CompressionMode.Decompress);
MemoryStream outbuf = new MemoryStream();
byte[] block = new byte[1024];
try
{
while (true)
{
int bytesRead = gs.Read(block, 0, block.Length);
if (bytesRead <= 0)
{
break;
}
else
{
outbuf.Write(block, 0, bytesRead);
}
}
str = encoding.GetString(outbuf.ToArray());
}
catch (Exception e)
{
Log("解压Gzip发生异常----" + e.Message);
}
finally
{
outbuf.Close();
gs.Close();
ms.Close();
}
return str;
}
public static byte[] unGzipBytes(byte[] data, int len)
{
MemoryStream ms = new MemoryStream(data, 0, len);
GZipStream gs = new GZipStream(ms, CompressionMode.Decompress);
MemoryStream outbuf = new MemoryStream();
byte[] block = new byte[1024];
byte[] result;
try
{
while (true)
{
int bytesRead = gs.Read(block, 0, block.Length);
if (bytesRead <= 0)
{
break;
}
else
{
outbuf.Write(block, 0, bytesRead);
}
}
result = outbuf.ToArray();
}
catch (Exception e)
{
Log("解压Gzip发生异常----" + e.Message);
result = new byte[0];
}
finally
{
outbuf.Close();
gs.Close();
ms.Close();
}
return result;
}
public static string getHTMLEncoding(string header, string body)
{
if (string.IsNullOrEmpty(header) && string.IsNullOrEmpty(body))
{
return "";
}
body = body.ToUpper();
Match m = Regex.Match(header, @"charset\b\s*=\s*""?(?<charset>[^""]*)", RegexOptions.IgnoreCase);
if (m.Success)
{
return m.Groups["charset"].Value.ToUpper();
}
else
{
if (string.IsNullOrEmpty(body))
{
return "";
}
m = Regex.Match(body, @"charset\b\s*=\s*""?(?<charset>[^""]*)", RegexOptions.IgnoreCase);
if (m.Success)
{
return m.Groups["charset"].Value.ToUpper();
}
}
return "";
}
private static void checkContentLength(ref AxiRespInfo respinfo, ref string request)
{
//重新计算并设置Content-length
int sindex = request.IndexOf(CTRL);
respinfo.reuqestHeader = request;
if (sindex != -1)
{
respinfo.reuqestHeader = request.Substring(0, sindex);
respinfo.reuqestBody = request.Substring(sindex + 4, request.Length - sindex - 4);
int contentLength = Encoding.UTF8.GetBytes(respinfo.reuqestBody).Length;
String newContentLength = Content_Length_Str_M + contentLength;
if (request.IndexOf(Content_Length_Str_M) != -1)
{
request = Regex.Replace(request, Content_Length_Str_M + "\\d+", newContentLength);
}
else
{
request = request.Insert(sindex, "\r\n" + newContentLength);
}
}
else
{
request = Regex.Replace(request, Content_Length_Str + "\\d+", Content_Length_Str_M + "0");
request += CTRL;
}
}
private static bool ValidateServerCertificate(object sender, X509Certificate certificate, X509Chain chain, SslPolicyErrors sslPolicyErrors)
{
return true;
}
public static bool ParseURI(string strURI, ref bool bIsSSL, ref string strHost, ref string strIP, ref int Port, ref string strRelativePath)
{
string strAddressRet;
string strPortRet;
string strRelativePathRet;
string strIPRet;
/*string strProtocol = strURI.Substring(0, 7);
if (strProtocol != "http://"
||
strProtocol != "https://")
return false;*/
if (!strURI.ToLower().StartsWith("http://") || strURI.ToLower().StartsWith("https://"))
return false;
bIsSSL = strURI.ToLower().StartsWith("https://");
string strLeft = strURI.Substring(7, strURI.Length - 7);
int nIndexPort = strLeft.IndexOf(':');
if (nIndexPort == -1)
{
if (bIsSSL)
strPortRet = "443";
else
strPortRet = "80";
int nIndexRelative = strLeft.IndexOf('/');
if (nIndexRelative != -1)
{
strAddressRet = strLeft.Substring(0, nIndexRelative);
strRelativePathRet = strLeft.Substring(nIndexRelative, strLeft.Length - nIndexRelative);
}
else
return false;
}
else
{
strAddressRet = strLeft.Substring(0, nIndexPort);
int nIndexRelative = strLeft.IndexOf('/');
if (nIndexRelative != -1)
{
strPortRet = strLeft.Substring(nIndexPort + 1, nIndexRelative - (nIndexPort + 1));
strRelativePathRet = strLeft.Substring(nIndexRelative, strLeft.Length - nIndexRelative);
}
else
return false;
}
strHost = strAddressRet;
try
{
//IPHostEntry hostinfo = Dns.GetHostEntry(strAddressRet);
//IPAddress[] aryIP = hostinfo.AddressList;
//strIPRet = aryIP[0].ToString();
strIPRet = GetDnsIP(strAddressRet).ToString();
}
catch
{
return false;
}
strIP = strIPRet;
Port = int.Parse(strPortRet);
strRelativePath = UrlEncode(strRelativePathRet);
return true;
}
public static string UrlEncode(string str)
{
string sb = "";
List<char> filter = new List<char>() { '!', '#', '$', '&', '\'', '(', ')', '*', '+', ',', '-', '.', '/', ':', ';', '=', '?', '@', '_', '~' };
byte[] byStr = System.Text.Encoding.UTF8.GetBytes(str); //System.Text.Encoding.Default.GetBytes(str)
for (int i = 0; i < byStr.Length; i++)
{
if (filter.Contains((char)byStr[i]))
sb += (char)byStr[i];
else if ((char)byStr[i] >= 'a' && (char)byStr[i] <= 'z')
sb += (char)byStr[i];
else if ((char)byStr[i] >= 'A' && (char)byStr[i] <= 'Z')
sb += (char)byStr[i];
else if ((char)byStr[i] >= '0' && (char)byStr[i] <= '9')
sb += (char)byStr[i];
else
sb += (@"%" + Convert.ToString(byStr[i], 16));
}
return sb;
}
class Tools
{
public static long currentMillis()
{
return (long)(DateTime.UtcNow - new DateTime(1970, 1, 1, 0, 0, 0, DateTimeKind.Utc)).TotalMilliseconds;
}
/// <summary>
/// 将16进制转换成10进制
/// </summary>
/// <param name="str">16进制字符串</param>
/// <returns></returns>
public static int convertToIntBy16(String str)
{
try
{
//Log($"convertToIntBy16 str- {str} lenght->{str.Length}");
if (str.Length == 0)
return 0;
return Convert.ToInt32(str, 16);
}
catch (Exception e)
{
Log($"convertToIntBy16 - {e.ToString()}");
}
return 0;
}
public static String getCurrentPath(String url)
{
int index = url.LastIndexOf("/");
if (index != -1)
{
return url.Substring(0, index) + "/";
}
else
{
return "";
}
}
/// <summary>
/// 将字符串转换成数字错误返回0
/// </summary>
/// <param name="strs">字符串</param>
/// <returns></returns>
public static int convertToInt(String str)
{
try
{
return int.Parse(str);
}
catch (Exception e)
{
Log("info:-" + e.Message);
}
return 0;
}
}
class TimeOutSocket
{
private bool IsConnectionSuccessful = false;
private Exception socketexception = null;
private ManualResetEvent TimeoutObject = new ManualResetEvent(false);
public int useTime = 0;
public TcpClient Connect(string host, int port, int timeoutMSec)
{
Stopwatch sw = new Stopwatch();
sw.Start();
TimeoutObject.Reset();
socketexception = null;
TcpClient tcpclient = new TcpClient();
//IPHostEntry hostinfo = Dns.GetHostEntry("emu.axibug.com");
//IPAddress[] aryIP = hostinfo.AddressList;
//host = aryIP[0].ToString();
Log($"BeginConnect {host}:{port} timeoutMSec=>{timeoutMSec}");
tcpclient.BeginConnect(host, port, new AsyncCallback(CallBackMethod), tcpclient);
if (TimeoutObject.WaitOne(timeoutMSec, false))
{
if (IsConnectionSuccessful)
{
sw.Stop();
useTime = (int)sw.ElapsedMilliseconds;
return tcpclient;
}
else
{
throw socketexception;
}
}
else
{
tcpclient.Close();
throw new TimeoutException("TimeOut Exception");
}
}
private void CallBackMethod(IAsyncResult asyncresult)
{
try
{
IsConnectionSuccessful = false;
TcpClient tcpclient = asyncresult.AsyncState as TcpClient;
if (tcpclient.Client != null)
{
tcpclient.EndConnect(asyncresult);
IsConnectionSuccessful = true;
}
}
catch (Exception ex)
{
Log($"CallBackMethod - {ex.ToString()}");
IsConnectionSuccessful = false;
socketexception = ex;
}
finally
{
TimeoutObject.Set();
}
}
}
}