FileSqlServerSync/Server/Common/ConnectPipeline.cs

195 lines
6.5 KiB
C#
Raw Permalink Normal View History

using System.Net.WebSockets;
using System.Text;
using System.Text.Json;
namespace Common;
public abstract class AbsPipeLine(bool isAES)
{
/// <summary>
/// pipeLine工作函数生效期间永久阻塞
/// </summary>
/// <param name="receiveCb">回调函数从pipeline接收到的消息</param>
/// <param name="addr">连接的远程地址</param>
/// <returns>第一次返回连接信息第二次当pipeline 被断开时返回,此时可能是正常完成断开,或异常发生断开</returns>
public abstract IAsyncEnumerable<int> Work(Func<byte[], bool> receiveCb, string addr = "");
protected Func<byte[], bool> ReceiveMsg = (byte[] a) =>
{
return true;
};
/// <summary>
/// 监听pipeline 消息由Work 函数调用
/// </summary>
/// <param name="receiveCb">消息回调</param>
/// <returns></returns>
protected abstract Task Listen(Func<byte[], bool> receiveCb);
/// <summary>
/// 关闭连接
/// </summary>
/// <param name="CloseReason">关闭原因</param>
/// <returns></returns>
public abstract Task Close(string? CloseReason);
/// <summary>
/// 向管道中发送消息
/// </summary>
/// <param name="msg"></param>
/// <returns></returns>
public abstract Task SendMsg(SyncMsg msg);
/// <summary>
/// 打包文件上传
/// </summary>
/// <param name="url">上传地址</param>
/// <param name="filePath">上传的文件路径</param>
/// <param name="progressCb">上传进度回调(现在没有回调)</param>
/// <returns>上传完成时返回</returns>/
public abstract Task UploadFile(string url, string filePath, Func<double, bool> progressCb);
/// <summary>
/// 管道消息是否使用AES加密
/// </summary>
protected readonly bool IsAES = isAES;
}
public class WebSocPipeLine<TSocket>(TSocket socket, bool isAES) : AbsPipeLine(isAES)
where TSocket : WebSocket
{
public readonly TSocket Socket = socket;
2024-09-22 05:27:52 +00:00
public override async Task UploadFile(
string url,
string filePath,
2024-09-22 05:27:52 +00:00
Func<double, bool> progressCb
)
{
using var client = new HttpClient();
using var content = new MultipartFormDataContent();
using var fileStream = new FileStream(filePath, FileMode.Open);
// TODO 上传进度回调
2024-10-29 08:16:31 +00:00
var progress = new Progress<double>(
(current) =>
{
progressCb(current);
}
);
var fileContent = new ProgressStreamContent(fileStream, progress);
content.Add(fileContent, "file", Path.GetFileName(filePath));
var it = await client.PostAsync("http://" + url + "/UploadFile", content);
if (it.StatusCode != System.Net.HttpStatusCode.OK)
2024-09-22 05:27:52 +00:00
{
throw new Exception(it.Content.ReadAsStringAsync().Result);
2024-09-22 05:27:52 +00:00
}
}
public override async IAsyncEnumerable<int> Work(Func<byte[], bool> receiveCb, string addr = "")
{
if (Socket is ClientWebSocket CSocket)
{
//连接失败会抛出异常
await CSocket.ConnectAsync(new Uri("ws://" + addr), CancellationToken.None);
yield return 0;
}
// 从controller 来,这个已经连接上了
else if (Socket is WebSocket)
{
yield return 0;
}
await Listen(receiveCb);
yield return 1;
}
protected override async Task Listen(Func<byte[], bool> receiveCb)
{
//warning 最大支持10MB这由需要同步的文件数量大小决定 UTF-8 每个字符汉字视为4个字节数字1个 英文字母2个。1MB=256KB*425万个字符能描述就行
var buffer = new byte[10 * 1024 * 1024];
while (Socket.State == WebSocketState.Open)
{
var receiveResult = await Socket.ReceiveAsync(
new ArraySegment<byte>(buffer),
CancellationToken.None
);
if (receiveResult.MessageType == WebSocketMessageType.Close)
{
throw new Exception(receiveResult.CloseStatusDescription);
}
else
{
var nbuffer = new byte[receiveResult.Count];
System.Buffer.BlockCopy(buffer, 0, nbuffer, 0, receiveResult.Count);
if (IsAES)
{
var nnbuffer = AESHelper.DecryptStringFromBytes_Aes(nbuffer);
receiveCb(Encoding.UTF8.GetBytes(nnbuffer));
}
else
{
receiveCb(nbuffer);
}
}
}
}
public override async Task Close(string? CloseReason)
{
if (Socket.State == WebSocketState.Open)
{
//CloseReason 最大不能超过123bytes.https://developer.mozilla.org/en-US/docs/Web/API/WebSocket/close
//若传递超过这个限制此处表现WebSocket将会卡住无法关闭。
if (Encoding.UTF8.GetBytes(CloseReason ?? "").Length > 120)
{
await SendMsg(
new SyncMsg
{
Type = SyncMsgType.Error,
Step = SyncProcessStep.Close,
Body = CloseReason ?? ""
}
);
await Socket.CloseAsync(
WebSocketCloseStatus.NormalClosure,
"查看上一条错误信息!",
CancellationToken.None
);
}
else
{
await Socket.CloseAsync(
WebSocketCloseStatus.NormalClosure,
CloseReason,
CancellationToken.None
);
}
}
}
public override async Task SendMsg(SyncMsg msg)
{
string msgStr = JsonSerializer.Serialize(msg);
var it = AESHelper.EncryptStringToBytes_Aes(msgStr);
if (IsAES)
{
// 加密后的字符使用Binary 发送,加密通常不会发送到最前端,通常是 js 写的websocket
await Socket.SendAsync(
new ArraySegment<byte>(it),
WebSocketMessageType.Binary,
true,
CancellationToken.None
);
}
else
{
await Socket.SendAsync(
new ArraySegment<byte>(Encoding.UTF8.GetBytes(msgStr)),
WebSocketMessageType.Text,
true,
CancellationToken.None
);
}
}
}