Files
SqlSugar/Src/Asp.NetCore2/MongoDb.Ado.data/ExecuteNonQueryItemsAsync/BulkWriteHandlerAsync.cs

103 lines
3.8 KiB
C#
Raw Normal View History

2025-06-22 10:44:05 +08:00
using MongoDB.Bson.Serialization;
using MongoDB.Bson;
using MongoDB.Driver;
using System;
using System.Collections.Generic;
using System.Text;
using System.Threading.Tasks;
using System.Threading;
namespace MongoDb.Ado.data
{
public class BulkWriteHandlerAsync : IMongoOperationHandlerAsync
{
2025-07-09 12:06:43 +08:00
public HandlerContext context { get; set; }
2025-06-22 10:44:05 +08:00
public CancellationToken token { get; set; }
public string operation { get; set; }
public async Task<int> HandleAsync(IMongoCollection<BsonDocument> collection, string json)
{
var documents = ParseJsonArray(json);
var bulkOps = new List<WriteModel<BsonDocument>>();
2025-08-09 10:43:17 +08:00
// 新增逻辑如果只有一个文档且update只包含$set
if (documents.Count == 1)
{
var doc = documents[0];
var filter = doc["filter"].AsBsonDocument;
var update = doc["update"].AsBsonDocument;
if (IsUpateBySql(update))
{
return await HandlePipelineUpdate(collection, filter, update);
}
}
2025-06-22 10:44:05 +08:00
foreach (var doc in documents)
{
var filter = doc["filter"].AsBsonDocument;
var update = doc["update"].AsBsonDocument;
var op = new UpdateManyModel<BsonDocument>(filter, update);
bulkOps.Add(op);
}
2025-06-24 19:20:39 +08:00
if (bulkOps.Count == 0) return 0;
2025-07-11 17:20:42 +08:00
if (context.IsAnyServerSession)
{
var result = await collection.BulkWriteAsync(context.ServerSession,bulkOps);
return (int)result.ModifiedCount;
}
else
{
var result = await collection.BulkWriteAsync(bulkOps);
return (int)result.ModifiedCount;
}
2025-06-22 10:44:05 +08:00
}
private List<BsonDocument> ParseJsonArray(string json)
{
if (json.TrimStart().StartsWith("["))
return BsonSerializer.Deserialize<List<BsonDocument>>(json);
return new List<BsonDocument> { BsonDocument.Parse(json) };
}
2025-08-09 10:43:17 +08:00
2025-08-09 10:45:45 +08:00
private async Task<int> HandlePipelineUpdate(IMongoCollection<BsonDocument> collection, BsonDocument filter, BsonDocument update)
2025-08-09 10:43:17 +08:00
{
// 构造pipeline update
// 构造pipeline update不写死循环现有的$set值
var setDoc = update["$set"].AsBsonDocument;
var setPipelineDoc = new BsonDocument();
foreach (var element in setDoc.Elements)
{
// 检查值是否为BsonDocument且包含操作符如$add否则直接赋值
if (element.Value.IsBsonDocument && element.Value.AsBsonDocument.GetElement(0).Name.StartsWith("$"))
{
setPipelineDoc[element.Name] = element.Value;
}
else
{
setPipelineDoc[element.Name] = element.Value;
}
}
var updatePipeline = new[]
{
new BsonDocument("$set", setPipelineDoc)
};
var pipelineUpdate = new PipelineUpdateDefinition<BsonDocument>(updatePipeline);
2025-08-09 10:45:45 +08:00
if (context.IsAnyServerSession)
{
var result = await collection.UpdateManyAsync(context.ServerSession,filter, pipelineUpdate);
return (int)result.ModifiedCount;
}
else
{
var result = await collection.UpdateManyAsync(filter, pipelineUpdate);
return (int)result.ModifiedCount;
}
2025-08-09 10:43:17 +08:00
}
private static bool IsUpateBySql(BsonDocument update)
{
return update.ElementCount == 1 && update.Contains("$set");
}
2025-06-22 10:44:05 +08:00
}
}