1 Star 3 Fork 1

apiumc/WebVPN

加入 Gitee
与超过 1400万 开发者一起发现、参与优秀开源项目,私有仓库也完全免费 :)
免费加入
文件
克隆/下载
HttpRepalcer.cs 38.78 KB
一键复制 编辑 原始数据 按行查看 历史
apiumc 提交于 2025-01-10 18:28 +08:00 . Change
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128
using System;
using UMC.Net;
using System.Text;
namespace UMC.ITME
{
public class HttpReplacer
{
static byte[] strHttps = UTF8Encoding.ASCII.GetBytes("https://");
static byte[] strHttp = UTF8Encoding.ASCII.GetBytes("http://");
static byte[] httpsEncode = UTF8Encoding.ASCII.GetBytes(Uri.EscapeDataString("https://"));
static byte[] httpEncode = UTF8Encoding.ASCII.GetBytes(Uri.EscapeDataString("http://"));
static byte[] strEncodePort = UTF8Encoding.ASCII.GetBytes("%3A");
public static byte[] HtmlHeadEnd = UTF8Encoding.ASCII.GetBytes("</head>");
static byte[] Equals16 = SiteConfig.Hex(new byte[] { 61 });
static byte[] Point16 = SiteConfig.Hex(new byte[] { 46 });
byte[] nowScheme;
byte[] nowSchemeEncode;
NetWriteData _writer;
SiteConfig.ReplaceSetting replaceSetting;
int _srcIndex, _destIndex;
// string _headKey;
HttpProxy _proxy;
int _maxLength;
int _minLength, _minHexLength;
byte[] buffer;
bool isFind = false;
bool isPort = false;
bool isEncodePort = false;
int bufferSize = 0;
int bufferStart = 0;
// bool isHead;
Func<bool, bool> _check;
public HttpReplacer(HttpProxy proxy, NetWriteData writer, SiteConfig.ReplaceSetting setting, int srcIndex, int destIndex)
{
this._proxy = proxy;
this._writer = writer;
this.replaceSetting = setting;
this._destIndex = destIndex;
this._srcIndex = srcIndex;
var hLength = proxy.Site.MainHost.Hosts[srcIndex].Length;
_minLength = proxy.Site.IsUnion ? Math.Min(hLength, proxy.Site.SuffixHost.Hosts[srcIndex].Length) : hLength;
if (replaceSetting != null)
{
_check = proxy.Site.IsUnion ? CheckUnionSetting : CheckSetting;
foreach (var hv in replaceSetting.Domains.Span)
{
var item = hv.Hosts[srcIndex];
if (item.Length > hLength)
{
hLength = item.Length;
}
if (item.Length < _minLength)
{
_minLength = item.Length;
}
}
}
else
{
_check = proxy.Site.IsUnion ? EndUnionWith : Check;
}
if (proxy.Context.Scheme.Length > 4)
{
this.nowScheme = strHttps;
this.nowSchemeEncode = httpsEncode;
}
else
{
this.nowScheme = strHttp;
this.nowSchemeEncode = httpEncode;
}
this._maxLength = hLength * 4 + 8;
this.buffer = new byte[_maxLength];
this._minHexLength = _minLength * 4;
}
int Read(NetReadData reader)
{
if (bufferSize == _maxLength)
{
_writer(buffer, 0, 1);
Array.Copy(buffer, 1, buffer, 0, bufferSize - 1);
bufferSize--;
if (bufferStart > 0)
{
bufferStart--;
}
}
return reader(buffer, bufferSize, 1);
}
static bool EndsWith(byte[] buffer, int bl, Span<byte> end)
{
return EndsWith(buffer, 0, bl, end);
}
static bool EndsWith(byte[] buffer, int offset, int size, Span<byte> end)
{
var el = end.Length;
if (el <= size)
{
offset = offset + size - el;
for (var i = 0; i < el; i++)
{
if (end[i] != buffer[offset + i])
{
return false;
}
}
return true;
}
return false;
}
static bool EndsWithIgnoreCase(byte[] buffer, int offset, int size, Span<byte> end)
{
var el = end.Length;
if (el <= size)
{
offset = offset + size - el;
for (var i = 0; i < el; i++)
{
if (Utility.IgnoreCase(end[i], buffer[offset + i]) == false)
{
return false;
}
}
return true;
}
return false;
}
byte lastByte;
bool isHead = true;
public void Html(StringValue jsKey, NetReadData reader)
{
byte value;
int len;
while ((len = Read(reader)) > 0)
{
value = buffer[bufferSize];
bufferSize++;
switch (value)
{
case 0:
if (len > 1)
bufferSize--;
goto case 20;
case 20:
default:
is16 = false;
len16 = 0;
bufferStart = 0;
isEncodePort = false;
isPort = false;
_writer(buffer, 0, bufferSize);
lastByte = buffer[bufferSize - 1];
bufferSize = 0;
isFind = false;
continue;
case 60://<
if (isHead)
{
continue;
}
else
{
goto case 20;
}
case 62://>
if (isHead && EndsWith(buffer, bufferSize, HtmlHeadEnd))
{
is16 = false;
isHead = false;
bufferSize = bufferSize - HtmlHeadEnd.Length;
_writer(buffer, 0, bufferSize);
var bData = UTF8Encoding.ASCII.GetBytes($"<script src=\"/UMC.Conf/{_proxy.Site.Root}/{_proxy.MD5(String.Empty)}/{HttpProxy.MD5(jsKey)}.js\"></script>\r\n</head>");
_writer(bData, 0, bData.Length);
bufferSize = 0;
bufferStart = 0;
continue;
}
goto case 20;
case 58://':':
if (isFind)
{
is16 = false;
isFind = false;
isPort = true;
bufferSize--;
continue;
}
else
{
bufferStart = bufferSize;
goto case 122;
}
case 37://'%':
if (isFind)
{
is16 = false;
isPort = false;
isFind = false;
isEncodePort = true;
continue;
}
else
{
bufferStart = bufferSize;
goto case 122;
}
case 48://0
case 49://1
case 50://2
case 51://3
case 52://4
case 53://5
case 54://6
case 55://7
case 56://8
case 57://9
if (isPort)
{
bufferSize--;
continue;
}
goto case 100;
case 65://A
case 66://B
case 67://C
case 68://D
case 69://E
case 70://F
if (is16)
{
if (len16 == 2 || len16 == 3)
{
len16++;
if (len16 == 4)
{
if (_minHexLength > bufferSize)
{
continue;
}
break;
}
else
{
continue;
}
}
else
{
len16 = 0;
}
}
bufferStart = bufferSize;
goto case 122;
case 47://“/”
bufferStart = bufferSize;
goto case 122;
case 92://"\"
if (is16 && len16 == 4)
{
len16 = 1;
}
else
{
is16 = true;
len16 = 1;
bufferStart = 0;
isEncodePort = false;
isPort = false;
_writer(buffer, 0, bufferSize - 1);
bufferSize = 1;
buffer[0] = value;
isFind = false;
}
continue;
// break;
case 120://x
if (is16)
{
if (len16 == 1)
{
len16++;
continue;
}
else
{
len16 = 0;
}
}
goto case 122;
// break;
case 97://a
case 98://b
case 99://c
case 100://d
case 101://e
case 102://f
if (is16)
{
if (len16 == 2 || len16 == 3)
{
len16++;
if (len16 == 4)
{
if (_minHexLength > bufferSize)
{
continue;
}
break;
}
else
{
continue;
}
}
else
{
len16 = 0;
}
}
goto case 122;
case 45://-
case 46://.
// case 47://“/”
// case 97://a
// case 98://b
// case 99://c
// case 100://d
// case 101://e
// case 102://f
case 103://g
case 104://h
case 105://i
case 106://j
case 107://k
case 108://l
case 109://m
case 110://n
case 111://o
case 112://p
case 113://q
case 114://r
case 115://s
case 116://t
case 117://u
case 118://v
case 119://w
case 121://y
case 122://z
is16 = false;
if (isEncodePort && bufferSize == 3)
{
isEncodePort = false;
if (EndsWith(buffer, bufferSize, strEncodePort))
{
isPort = true;
bufferSize = 0;
bufferStart = 0;
}
continue;
}
isPort = false;
if (_minLength > bufferSize - bufferStart)
{
continue;
}
break;
}
isFind = _check(is16);
}
}
bool is16 = false;
int len16 = 0;
public void Append(HttpAppender appender, NetReadData reader)
{
byte value;
int len;
while (appender.IsStop == false && (len = Read(reader)) > 0)
{
value = buffer[bufferSize];
bufferSize++;
switch (value)
{
case 0:
if (len > 1)
bufferSize--;
goto case 20;
case 20:
default:
is16 = false;
len16 = 0;
bufferStart = 0;
isEncodePort = false;
isPort = false;
_writer(buffer, 0, bufferSize);
lastByte = buffer[bufferSize - 1];
bufferSize = 0;
isFind = false;
continue;
case 10:
if (appender.Index == appender.AppendIndex)
{
appender.Append(_writer);
}
appender.Index++;
goto case 20;
// break;
case 58://':':
if (isFind)
{
is16 = false;
isFind = false;
isPort = true;
bufferSize--;
continue;
}
else
{
bufferStart = bufferSize;
goto case 122;
}
case 37://'%':
if (isFind)
{
is16 = false;
isPort = false;
isFind = false;
isEncodePort = true;
continue;
}
else
{
bufferStart = bufferSize;
goto case 122;
}
case 48://0
case 49://1
case 50://2
case 51://3
case 52://4
case 53://5
case 54://6
case 55://7
case 56://8
case 57://9
if (isPort)
{
bufferSize--;
continue;
}
goto case 100;
case 65://A
case 66://B
case 67://C
case 68://D
case 69://E
case 70://F
if (is16)
{
if (len16 == 2 || len16 == 3)
{
len16++;
if (len16 == 4)
{
if (_minHexLength > bufferSize)
{
continue;
}
break;
}
else
{
continue;
}
}
else
{
len16 = 0;
}
}
bufferStart = bufferSize;
goto case 122;
case 47://“/”
bufferStart = bufferSize;
goto case 122;
case 92://"\"
if (is16 && len16 == 4)
{
len16 = 1;
}
else
{
is16 = true;
len16 = 1;
bufferStart = 0;
isEncodePort = false;
isPort = false;
_writer(buffer, 0, bufferSize - 1);
bufferSize = 1;
buffer[0] = value;
isFind = false;
}
continue;
// break;
case 120://x
if (is16)
{
if (len16 == 1)
{
len16++;
continue;
}
else
{
len16 = 0;
}
}
goto case 122;
// break;
case 97://a
case 98://b
case 99://c
case 100://d
case 101://e
case 102://f
if (is16)
{
if (len16 == 2 || len16 == 3)
{
len16++;
if (len16 == 4)
{
if (_minHexLength > bufferSize)
{
continue;
}
break;
}
else
{
continue;
}
}
else
{
len16 = 0;
}
}
goto case 122;
case 45://-
case 46://.
// case 47://“/”
// case 97://a
// case 98://b
// case 99://c
// case 100://d
// case 101://e
// case 102://f
case 103://g
case 104://h
case 105://i
case 106://j
case 107://k
case 108://l
case 109://m
case 110://n
case 111://o
case 112://p
case 113://q
case 114://r
case 115://s
case 116://t
case 117://u
case 118://v
case 119://w
case 121://y
case 122://z
is16 = false;
if (isEncodePort && bufferSize == 3)
{
isEncodePort = false;
if (EndsWith(buffer, bufferSize, strEncodePort))
{
isPort = true;
bufferSize = 0;
bufferStart = 0;
}
continue;
}
isPort = false;
if (_minLength > bufferSize - bufferStart)
{
continue;
}
break;
}
isFind = _check(is16);
}
}
public void Replace(NetReadData reader)
{
byte value;
int len;
while ((len = Read(reader)) > 0)
{
value = buffer[bufferSize];
bufferSize++;
switch (value)
{
case 0:
if (len > 1)
bufferSize--;
goto case 20;
case 20:
default:
is16 = false;
len16 = 0;
bufferStart = 0;
isEncodePort = false;
isPort = false;
_writer(buffer, 0, bufferSize);
lastByte = buffer[bufferSize - 1];
bufferSize = 0;
isFind = false;
continue;
case 58://':':
if (isFind)
{
is16 = false;
isFind = false;
isPort = true;
bufferSize--;
continue;
}
else
{
bufferStart = bufferSize;
goto case 122;
}
case 37://'%':
if (isFind)
{
is16 = false;
isPort = false;
isFind = false;
isEncodePort = true;
continue;
}
else
{
bufferStart = bufferSize;
goto case 122;
}
case 48://0
case 49://1
case 50://2
case 51://3
case 52://4
case 53://5
case 54://6
case 55://7
case 56://8
case 57://9
if (isPort)
{
bufferSize--;
continue;
}
goto case 100;
case 65://A
case 66://B
case 67://C
case 68://D
case 69://E
case 70://F
if (is16)
{
if (len16 == 2 || len16 == 3)
{
len16++;
if (len16 == 4)
{
if (_minHexLength > bufferSize)
{
continue;
}
break;
}
else
{
continue;
}
}
else
{
len16 = 0;
}
}
bufferStart = bufferSize;
goto case 122;
case 47://“/”
bufferStart = bufferSize;
goto case 122;
case 92://"\"
if (is16 && len16 == 4)
{
len16 = 1;
}
else
{
is16 = true;
len16 = 1;
bufferStart = 0;
isEncodePort = false;
isPort = false;
_writer(buffer, 0, bufferSize - 1);
bufferSize = 1;
buffer[0] = value;
isFind = false;
}
continue;
// break;
case 120://x
if (is16)
{
if (len16 == 1)
{
len16++;
continue;
}
else
{
len16 = 0;
}
}
goto case 122;
// break;
case 97://a
case 98://b
case 99://c
case 100://d
case 101://e
case 102://f
if (is16)
{
if (len16 == 2 || len16 == 3)
{
len16++;
if (len16 == 4)
{
if (_minHexLength > bufferSize)
{
continue;
}
break;
}
else
{
continue;
}
}
else
{
len16 = 0;
}
}
goto case 122;
case 45://-
case 46://.
// case 47://“/”
// case 97://a
// case 98://b
// case 99://c
// case 100://d
// case 101://e
// case 102://f
case 103://g
case 104://h
case 105://i
case 106://j
case 107://k
case 108://l
case 109://m
case 110://n
case 111://o
case 112://p
case 113://q
case 114://r
case 115://s
case 116://t
case 117://u
case 118://v
case 119://w
case 121://y
case 122://z
is16 = false;
if (isEncodePort && bufferSize == 3)
{
isEncodePort = false;
if (EndsWith(buffer, bufferSize, strEncodePort))
{
isPort = true;
bufferSize = 0;
bufferStart = 0;
}
continue;
}
isPort = false;
if (_minLength > bufferSize - bufferStart)
{
continue;
}
break;
}
isFind = _check(is16);
}
}
bool Check(bool is16)
{
return EndWith(this._proxy.Site.MainHost, is16);
}
bool CheckSetting(bool is16)
{
foreach (var hem in replaceSetting.Domains.Span)
{
if (EndWith(hem, is16))
{
return true;
}
}
return EndWith(this._proxy.Site.MainHost, is16);
}
bool CheckUnionSetting(bool is16)
{
foreach (var hem in replaceSetting.Domains.Span)
{
if (EndWith(hem, is16))
{
return true;
}
}
return EndUnionWith(is16);
}
public const byte BytePOINT = 46;
bool EndUnionWith(bool is16)
{
var hem = this._proxy.Site.SuffixHost;
var khost = hem.Hosts[_srcIndex];
if (is16)
{
var hosthex = hem.Hex[_srcIndex];
if (EndsWithIgnoreCase(buffer, 0, bufferSize, hosthex))
{
bufferStart = 0;
bufferSize = bufferSize - hosthex.Length;
if (bufferSize >= 4)
{
if (Utility.EqualIgnoreCase(buffer.AsSpan(bufferSize - 4, 4), Point16))
{
var dest = _proxy.Site.MainHost.Hex[_destIndex];
if (bufferSize >= 8)
{
var bs = buffer.AsSpan(bufferSize - 8, 4);
if (Utility.EqualIgnoreCase(bs, Equals16))
{
_writer(buffer, 0, bufferSize - 4);
int start = Utility.FindIgnoreCase(dest, Point16);
_writer(dest, start, dest.Length - start);
}
else
{
_writer(buffer, 0, bufferSize - 4);
_writer(dest, 0, dest.Length);
}
}
else
{
int start = Utility.FindIgnoreCase(dest, Point16);
_writer(dest, start, dest.Length - start);
}
bufferSize = 0;
return true;
}
}
_writer(buffer, 0, bufferSize);
_writer(hem.Hex[_destIndex], 0, hem.Hex[_destIndex].Length);
bufferSize = 0;
return true;
}
else
{
return false;
}
}
else if (EndsWith(buffer, bufferStart, bufferSize - bufferStart, khost))
{
bufferStart = 0;
bufferSize = bufferSize - khost.Length;
if (bufferSize > 0)
{
var b = buffer[bufferSize - 1];
var kmhost = _proxy.Site.MainHost.Hosts[_srcIndex];
if (b == kmhost[kmhost.Length - khost.Length - 1])
{
bufferSize--;
hem = _proxy.Site.MainHost;
if (b == BytePOINT)
{
if (bufferSize == 0)
{
switch (lastByte)
{
case 42:
case 39:
case 61:
case 34:
var dest2 = hem.Hosts[_destIndex];
int index = Array.IndexOf(dest2, BytePOINT);// dest.IndexOf(46)
if (index > 0)
{
_writer(dest2, index, dest2.Length - index);
}
else
{
_writer(dest2, 0, dest2.Length);
}
bufferSize = 0;
return true;
}
}
}
}
}
else
{
switch (lastByte)
{
// case 42:
// case 39:
case 61:
// case 34:
var dest2 = _proxy.Site.MainHost.Hosts[_destIndex];
int index = Array.IndexOf(dest2, BytePOINT);// dest.IndexOf(46)
if (index > 0)
{
_writer(dest2, index + 1, dest2.Length - index - 1);
}
else
{
_writer(dest2, 0, dest2.Length);
}
bufferSize = 0;
return true;
}
}
if (EndsWith(buffer, bufferSize, strHttps))
{
bufferSize = bufferSize - strHttps.Length;
_writer(buffer, 0, bufferSize);
_writer(nowScheme, 0, nowScheme.Length);
}
else if (EndsWith(buffer, bufferSize, strHttp))
{
bufferSize = bufferSize - strHttp.Length;
_writer(buffer, 0, bufferSize);
_writer(nowScheme, 0, nowScheme.Length);
}
else if (EndsWith(buffer, bufferSize, httpsEncode))
{
bufferSize = bufferSize - httpsEncode.Length;
_writer(buffer, 0, bufferSize);
_writer(nowSchemeEncode, 0, nowSchemeEncode.Length);
}
else if (EndsWith(buffer, bufferSize, httpEncode))
{
bufferSize = bufferSize - httpEncode.Length;
_writer(buffer, 0, bufferSize);
_writer(nowSchemeEncode, 0, nowSchemeEncode.Length);
}
else
{
_writer(buffer, 0, bufferSize);
}
var dest = hem.Hosts[_destIndex];
_writer(dest, 0, dest.Length);
bufferSize = 0;
return true;
}
return false;
}
bool EndWith(SiteConfig.HostConvert hem, bool is16)
{
var khost = hem.Hosts[_srcIndex];
if (is16)
{
var hosthex = hem.Hex[_srcIndex];
if (EndsWithIgnoreCase(buffer, 0, bufferSize, hosthex))
{
bufferStart = 0;
bufferSize = bufferSize - hosthex.Length;
_writer(buffer, 0, bufferSize);
_writer(hem.Hex[_destIndex], 0, hem.Hex[_destIndex].Length);
bufferSize = 0;
return true;
}
else
{
return false;
}
}
else if (EndsWith(buffer, bufferStart, bufferSize - bufferStart, khost))
{
bufferStart = 0;
bufferSize = bufferSize - khost.Length;
if (EndsWith(buffer, bufferSize, strHttps))
{
bufferSize = bufferSize - strHttps.Length;
_writer(buffer, 0, bufferSize);
_writer(nowScheme, 0, nowScheme.Length);
}
else if (EndsWith(buffer, bufferSize, strHttp))
{
bufferSize = bufferSize - strHttp.Length;
_writer(buffer, 0, bufferSize);
_writer(nowScheme, 0, nowScheme.Length);
}
else if (EndsWith(buffer, bufferSize, httpsEncode))
{
bufferSize = bufferSize - httpsEncode.Length;
_writer(buffer, 0, bufferSize);
_writer(nowSchemeEncode, 0, nowSchemeEncode.Length);
}
else if (EndsWith(buffer, bufferSize, httpEncode))
{
bufferSize = bufferSize - httpEncode.Length;
_writer(buffer, 0, bufferSize);
_writer(nowSchemeEncode, 0, nowSchemeEncode.Length);
}
else
{
_writer(buffer, 0, bufferSize);
}
var dest = hem.Hosts[_destIndex];
_writer(dest, 0, dest.Length);
bufferSize = 0;
return true;
}
return false;
}
public void Flush()
{
if (bufferSize > 0)
{
_writer(buffer, 0, bufferSize);
}
}
}
}
Loading...
马建仓 AI 助手
尝试更多
代码解读
代码找茬
代码优化
C#
1
https://gitee.com/apiumc/WebVPN.git
git@gitee.com:apiumc/WebVPN.git
apiumc
WebVPN
WebVPN
master

搜索帮助