完善通用接口输入参数

This commit is contained in:
2023-11-21 00:19:45 +08:00
parent 079eec500b
commit 70de358f97
17 changed files with 375 additions and 226 deletions

View File

@@ -3,8 +3,10 @@
// https://git.tuotong-tech.com/tnb/tnb.server //
/////////////////////////////////////////////////////////////////////////////////
using JNPF;
using JNPF.DependencyInjection;
using JNPF.DynamicApiController;
using Microsoft.AspNetCore.Http;
using Tnb.Vengine.DataAccess;
using Tnb.Vengine.Domain;
@@ -30,7 +32,7 @@ public class BaseAppService : IDynamicApiController, ITransient
{
if (!_models.ContainsKey(id))
{
_models[id] = await _dataAccess.GetVmodelAsync(id, false);
_models[id] = await _dataAccess.GetVmodelAsync(id);
}
return _models[id];
}
@@ -42,7 +44,7 @@ public class BaseAppService : IDynamicApiController, ITransient
var key = areaCode + "/" + vmCode;
if (!_models.ContainsKey(key))
{
_models[key] = await _dataAccess.GetVmodelAsync(areaCode, vmCode, false);
_models[key] = await _dataAccess.GetVmodelAsync(areaCode, vmCode);
}
return _models[key];
}

View File

@@ -3,6 +3,7 @@
// https://git.tuotong-tech.com/tnb/tnb-server //
/////////////////////////////////////////////////////////////////////////////////
using JNPF;
using JNPF.Common.Security;
using Mapster;
using Microsoft.AspNetCore.Authorization;
@@ -40,10 +41,10 @@ public class VengineAppService : BaseAppService, IVengineAppService
public async Task<dynamic?> GetAsync(string vmid, [FromQuery] VmGetInput input)
{
var vm = await GetVmodelAsync(vmid);
var arg = input.Adapt<VmQueryInput>();
arg.AddQueryParaIf(!string.IsNullOrEmpty(input.id), vm.GetPrimary().code, input.id!);
if (!string.IsNullOrEmpty(input.id)) input.q.Add(vm.GetPrimary().code, input.id);
input.LoadFromHttpContext(App.HttpContext);
var ls = await ListAsync(vmid, arg);
var ls = await ListAsync(vmid, input.Adapt<VmQueryInput>());
return ls.items.FirstOrDefault();
}
@@ -51,9 +52,10 @@ public class VengineAppService : BaseAppService, IVengineAppService
/// 获取多条 数据列表
/// </summary>
[HttpGet("api/[area]/[controller]/{vmid}/get-list")]
public async Task<VmPagedOutput> GetListAsync(string vmid, [FromQuery] VmGetListInput input)
public async Task<VmPagedOutput> GetListAsync(string vmid, [FromQuery] VmQueryInput input)
{
return await ListAsync(vmid, input.Adapt<VmQueryInput>());
input.LoadFromHttpContext(App.HttpContext);
return await ListAsync(vmid, input);
}
/// <summary>
@@ -110,10 +112,10 @@ public class VengineAppService : BaseAppService, IVengineAppService
public async Task<dynamic?> GetAsync(string areaCode, string vmCode, [FromQuery] VmGetInput input)
{
var vm = await GetVmodelAsync(areaCode, vmCode);
var arg = input.Adapt<VmQueryInput>();
arg.AddQueryParaIf(!string.IsNullOrEmpty(input.id), vm.GetPrimary().code, input.id!);
if (!string.IsNullOrEmpty(input.id)) input.q.Add(vm.GetPrimary().code, input.id);
input.LoadFromHttpContext(App.HttpContext);
var ls = await ListAsync(areaCode, vmCode, arg);
var ls = await ListAsync(areaCode, vmCode, input.Adapt<VmQueryInput>());
return ls.items.FirstOrDefault();
}
@@ -121,9 +123,10 @@ public class VengineAppService : BaseAppService, IVengineAppService
/// 获取多条 数据列表
/// </summary>
[HttpGet("api/{areaCode}/{vmCode}/get-list")]
public async Task<VmPagedOutput> GetListAsync(string areaCode, string vmCode, [FromQuery] VmGetListInput input)
public async Task<VmPagedOutput> GetListAsync(string areaCode, string vmCode, [FromQuery] VmQueryInput input)
{
return await ListAsync(areaCode, vmCode, input.Adapt<VmQueryInput>());
input.LoadFromHttpContext(App.HttpContext);
return await ListAsync(areaCode, vmCode, input);
}
/// <summary>

View File

@@ -27,11 +27,10 @@ namespace Tnb.Vengine.AppService;
[Authorize]
//[ApiDescriptionSettings(Tag = ModuleConst.Tag, Area = ModuleConst.Area, Order = 1102)]
[Route("api/[area]/[controller]/[action]")]
public class VengineAppService<TEntity, TGetInput, TGetOutput, TQueryInput, TGetListInput, TGetListOutput, TCreateInput, TUpdateInput> : BaseAppService
public class VengineAppService<TEntity, TGetInput, TGetOutput, TQueryInput, TQueryOutput, TCreateInput, TUpdateInput> : BaseAppService
where TEntity : Entity
where TGetInput : VmGetInput
where TQueryInput : VmQueryInput
where TGetListInput : VmGetListInput
where TCreateInput : VmEditInput
where TUpdateInput : VmEditInput
{
@@ -93,10 +92,10 @@ public class VengineAppService<TEntity, TGetInput, TGetOutput, TQueryInput, TGet
public virtual async Task<TGetOutput> GetAsync([FromQuery] TGetInput input)
{
var vm = await GetVmodelAsync();
TQueryInput arg = input.Adapt<TQueryInput>();
arg.AddQueryParaIf(!string.IsNullOrEmpty(input.id), vm.GetPrimary().code, input.id!);
if (!string.IsNullOrEmpty(input.id)) input.q.Add(vm.GetPrimary().code, input.id);
input.LoadFromHttpContext(App.HttpContext);
var data = (await ListAsync(arg)).items.FirstOrDefault();
var data = (await ListAsync(input.Adapt<TQueryInput>())).items.FirstOrDefault();
return data == null ? default! : data.Adapt<TGetOutput>();
}
@@ -104,20 +103,21 @@ public class VengineAppService<TEntity, TGetInput, TGetOutput, TQueryInput, TGet
/// 获取多条 数据列表
/// </summary>
[HttpGet]
public virtual async Task<PagedOutput<TGetListOutput>> GetListAsync([FromQuery] TGetListInput input)
public virtual async Task<PagedOutput<TQueryOutput>> GetListAsync([FromQuery] TQueryInput input)
{
return await ListAsync(input.Adapt<TQueryInput>());
input.LoadFromHttpContext(App.HttpContext);
return await ListAsync(input);
}
/// <summary>
/// 获取多条 数据列表
/// </summary>
[HttpPost]
public virtual async Task<PagedOutput<TGetListOutput>> ListAsync([FromBody] TQueryInput input)
public virtual async Task<PagedOutput<TQueryOutput>> ListAsync([FromBody] TQueryInput input)
{
var vm = await GetVmodelAsync();
var ls = await _dataAccess.QueryDataAsync(vm, input);
return ls.ToPagedOutput<TGetListOutput>();
return ls.ToPagedOutput<TQueryOutput>();
}
/// <summary>
@@ -155,22 +155,31 @@ public class VengineAppService<TEntity, TGetInput, TGetOutput, TQueryInput, TGet
}
#region
public class VengineAppService<TEntity, TGetInput, TGetOutput, TQueryInput, TGetListInput, TGetListOutput> :
VengineAppService<TEntity, TGetInput, TGetOutput, TQueryInput, TGetListInput, TGetListOutput, VmEditInput, VmEditInput>
public class VengineAppService<TEntity, TGetInput, TGetOutput, TQueryInput, TQueryOutput, TCreateInput> :
VengineAppService<TEntity, TGetInput, TGetOutput, TQueryInput, TQueryOutput, TCreateInput, VmEditInput>
where TEntity : Entity
where TGetInput : VmGetInput
where TQueryInput : VmQueryInput
where TCreateInput : VmEditInput
{
public VengineAppService(IDataAccess da) : base(da)
{
}
}
public class VengineAppService<TEntity, TGetInput, TGetOutput, TQueryInput, TQueryOutput> :
VengineAppService<TEntity, TGetInput, TGetOutput, TQueryInput, TQueryOutput, VmEditInput, VmEditInput>
where TEntity : Entity
where TGetInput : VmGetInput
where TGetListInput : VmGetListInput
where TQueryInput : VmQueryInput
{
public VengineAppService(IDataAccess da) : base(da)
{
}
}
public class VengineAppService<TEntity, TGetInput, TGetOutput, TQueryInput, TGetListInput> :
VengineAppService<TEntity, TGetInput, TGetOutput, TQueryInput, TGetListInput, dynamic, VmEditInput, VmEditInput>
public class VengineAppService<TEntity, TGetInput, TGetOutput, TQueryInput> :
VengineAppService<TEntity, TGetInput, TGetOutput, TQueryInput, dynamic, VmEditInput, VmEditInput>
where TEntity : Entity
where TGetInput : VmGetInput
where TGetListInput : VmGetListInput
where TQueryInput : VmQueryInput
{
public VengineAppService(IDataAccess da) : base(da)
@@ -178,7 +187,7 @@ public class VengineAppService<TEntity, TGetInput, TGetOutput, TQueryInput, TGet
}
}
public class VengineAppService<TEntity, TGetInput, TGetOutput> :
VengineAppService<TEntity, TGetInput, TGetOutput, VmQueryInput, VmGetListInput, dynamic, VmEditInput, VmEditInput>
VengineAppService<TEntity, TGetInput, TGetOutput, VmQueryInput, dynamic, VmEditInput, VmEditInput>
where TEntity : Entity
where TGetInput : VmGetInput
{
@@ -186,8 +195,18 @@ public class VengineAppService<TEntity, TGetInput, TGetOutput> :
{
}
}
public class VengineAppService<TEntity, TGetInput> :
VengineAppService<TEntity, TGetInput, dynamic, VmQueryInput, dynamic, VmEditInput, VmEditInput>
where TEntity : Entity
where TGetInput : VmGetInput
{
public VengineAppService(IDataAccess da) : base(da)
{
}
}
public class VengineAppService<TEntity> :
VengineAppService<TEntity, VmGetInput, dynamic, VmQueryInput, VmGetListInput, dynamic, VmEditInput, VmEditInput>
VengineAppService<TEntity, VmGetInput, dynamic, VmQueryInput, dynamic, VmEditInput, VmEditInput>
where TEntity : Entity
{
public VengineAppService(IDataAccess da) : base(da)

View File

@@ -5,11 +5,15 @@
using JNPF;
using JNPF.Common.Core.Manager;
using JNPF.Common.Extension;
using JNPF.Common.Manager;
using JNPF.Common.Security;
using JNPF.ViewEngine;
using Mapster;
using Microsoft.AspNetCore.Http;
using Microsoft.AspNetCore.Mvc;
using Newtonsoft.Json;
using Org.BouncyCastle.Crypto;
using SqlSugar;
using Tnb.Core;
using Tnb.Vengine.DataAccess;
@@ -23,7 +27,7 @@ namespace Tnb.Vengine.AppService;
/// </summary>
[ApiDescriptionSettings(Tag = ModuleConst.Tag, Area = ModuleConst.Area, KeepVerb = true, Order = 1102)]
//[Route("api/[area]/[controller]/[action]")]
public class VmodelAppService : VengineAppService<Vmodel, VmodelGetInput, Vmodel, VmQueryInput, VmGetListInput, Vmodel, VmodelCreateInput, VmodelUpdateInput>, IVmodelAppService
public class VmodelAppService : VengineAppService<Vmodel, VmodelGetInput, Vmodel, VmQueryInput, Vmodel, VmodelCreateInput>, IVmodelAppService
{
private readonly IViewEngine _viewEngine;
private readonly ICacheManager _cache;
@@ -39,41 +43,113 @@ public class VmodelAppService : VengineAppService<Vmodel, VmodelGetInput, Vmodel
_user = user;
}
protected override Task<Vmodel> GetVmodelAsync()
{
var vm = new Vmodel
{
//id = "30285088677397",
areaCode = "tnb",
vmCode = "vmodel",
vmName = "视图模型",
dbCode = "",
tableName = "sys_vmodel",
dbProps = new List<VmDbProp> {
new VmDbProp{field="id",csType="string",length=50,digit=0,ordinal=1,required=true,pkey=true,fuzzy=false,defValue=null,descrip=null,code="id",name="主键标识"},
new VmDbProp{field="area_code",csType="string",length=50,digit=0,ordinal=2,required=true,pkey=false,fuzzy=false,defValue=null,descrip=null,code="areaCode",name="模块代码"},
new VmDbProp{field="vm_code",csType="string",length=100,digit=0,ordinal=3,required=true,pkey=false,fuzzy=true,defValue=null,descrip=null,code="vmCode",name="视图代码"},
new VmDbProp{field="vm_name",csType="string",length=100,digit=0,ordinal=4,required=true,pkey=false,fuzzy=true,defValue=null,descrip=null,code="vmName",name="视图名称"},
new VmDbProp{field="db_code",csType="string",length=50,digit=0,ordinal=5,required=false,pkey=false,fuzzy=false,defValue=null,descrip=null,code="dbCode",name="数据库连接"},
new VmDbProp{field="table_name",csType="string",length=50,digit=0,ordinal=6,required=true,pkey=false,fuzzy=true,defValue=null,descrip=null,code="tableName",name="主表名称"},
new VmDbProp{field="db_props",csType="string",length=0,digit=0,ordinal=7,required=true,pkey=false,fuzzy=false,defValue=null,descrip=null,code="dbProps",name="表字段属性"},
new VmDbProp{field="nav_props",csType="string",length=0,digit=0,ordinal=8,required=false,pkey=false,fuzzy=false,defValue=null,descrip=null,code="navProps",name="导航属性"},
new VmDbProp{field="cal_props",csType="string",length=0,digit=0,ordinal=9,required=false,pkey=false,fuzzy=false,defValue=null,descrip=null,code="calProps",name="计算配置"},
new VmDbProp{field="ordinal",csType="int",length=0,digit=0,ordinal=10,required=true,pkey=false,fuzzy=false,defValue=null,descrip=null,code="ordinal",name="排序"},
new VmDbProp{field="soft_delete",csType="short",length=0,digit=0,ordinal=11,required=true,pkey=false,fuzzy=false,defValue=null,descrip=null,code="softDelete",name="软删除"},
new VmDbProp{field="enabled",csType="short",length=0,digit=0,ordinal=12,required=true,pkey=false,fuzzy=false,defValue=null,descrip=null,code="enabled",name="是否激活"},
new VmDbProp{field="deleted",csType="short",length=0,digit=0,ordinal=13,required=true,pkey=false,fuzzy=false,defValue=null,descrip=null,code="deleted",name="是否删除"},
new VmDbProp{field="descrip",csType="string",length=250,digit=0,ordinal=14,required=false,pkey=false,fuzzy=false,defValue=null,descrip=null,code="descrip",name="描述"},
new VmDbProp{field="create_time",csType="DateTime",length=0,digit=0,ordinal=15,required=true,pkey=false,fuzzy=false,defValue=null,descrip=null,code="createTime",name="创建时间"},
new VmDbProp{field="create_id",csType="string",length=50,digit=0,ordinal=16,required=false,pkey=false,fuzzy=false,defValue=null,descrip=null,code="createId",name="创建人"},
new VmDbProp{field="modify_time",csType="DateTime",length=0,digit=0,ordinal=17,required=false,pkey=false,fuzzy=false,defValue=null,descrip=null,code="modifyTime",name="修改时间"},
new VmDbProp{field="modify_id",csType="string",length=50,digit=0,ordinal=18,required=false,pkey=false,fuzzy=false,defValue=null,descrip=null,code="modifyId",name="修改人"}
},
navProps = [],
calProps = [],
ordinal = 0,
softDelete = 0,
enabled = 1,
deleted = 0,
descrip = null
};
return Task.FromResult(vm);
}
/// <summary>
/// 获取一条 数据信息, q参数无效
/// </summary>
public override async Task<Vmodel> GetAsync(VmodelGetInput input)
{
input.LoadFromHttpContext(App.HttpContext);
var query = _db.Queryable<Vmodel>().Where(a => a.deleted == 0);
Vmodel? vm = null;
if (!string.IsNullOrEmpty(input.id))
{
vm = await _dataAccess.GetVmodelAsync(input.id, input.drill);
vm = await _dataAccess.GetVmodelAsync(input.id);
}
else if (!string.IsNullOrEmpty(input.areaCode) && !string.IsNullOrEmpty(input.vmCode))
else if (input.q.ContainsKey("areaCode") && input.q.ContainsKey("vmCode"))
{
vm = await _dataAccess.GetVmodelAsync(input.areaCode, input.vmCode, input.drill);
vm = await _dataAccess.GetVmodelAsync(input.q["areaCode"].ToString()!, input.q["vmCode"].ToString()!);
}
ThrowIf.IsNull(vm, "输入参数有误, id 和 areaCode,vmCode 不可同时为空");
if (input.drill)
{
List<string> vmids = vm.navProps.Select(a => a.vmid).Distinct().ToList();
if (vmids.Count > 0)
{
List<Vmodel> ls = await _db.Queryable<Vmodel>().Where(a => vmids.Contains(a.id)).ToListAsync();
Dictionary<string, Vmodel> navs = ls.ToDictionary(a => a.id);
foreach (VmNavProp navProp in vm.navProps)
{
navProp.naviModel = navs.GetOrDefault(navProp.vmid);
}
}
}
//var serializer = new Newtonsoft.Json.JsonSerializer();
//var stringWriter = new StringWriter();
//using (var writer = new Newtonsoft.Json.JsonTextWriter(stringWriter))
//{
// writer.QuoteName = false;
// serializer.Serialize(writer, vm);
//}
//var json = stringWriter.ToString();
return vm;
}
/// <summary>
/// 获取多条 数据列表
/// </summary>
public override async Task<PagedOutput<Vmodel>> GetListAsync(VmGetListInput input)
public override async Task<PagedOutput<Vmodel>> GetListAsync(VmQueryInput input)
{
return await ListAsync(input.Adapt<VmQueryInput>());
input.LoadFromHttpContext(App.HttpContext);
return await ListAsync(input);
}
[NonAction]
public override async Task<PagedOutput<Vmodel>> ListAsync(VmQueryInput input)
{
var q = _db.Queryable<Vmodel>().Where(a => a.deleted == 0)
.WhereIF(!string.IsNullOrEmpty(input.k), a => a.vmCode.Contains(input.k!) || a.vmName.Contains(input.k!));
RefAsync<int> total = 0;
var data = await q.OrderBy(input.sort).ToPageListAsync((input.pnum - 1) * input.psize, input.psize, total);
return PagedOutput.Create(total, data);
PagedOutput<Vmodel> result = new PagedOutput<Vmodel>();
var vm = await GetVmodelAsync();
var ls = await _dataAccess.QueryDataAsync(vm, input);
foreach (var item in ls.items)
{
item["dbProps"] = JsonConvert.DeserializeObject<List<VmDbProp>>(item["dbProps"].ToString()!)!;
item["navProps"] = JsonConvert.DeserializeObject<List<VmNavProp>>(item["navProps"].ToString()!)!;
item["calProps"] = JsonConvert.DeserializeObject<List<VmCalProp>>(item["calProps"].ToString()!)!;
}
return ls.ToPagedOutput<Vmodel>();
}
/// <summary>
@@ -95,20 +171,30 @@ public class VmodelAppService : VengineAppService<Vmodel, VmodelGetInput, Vmodel
/// <summary>
/// 更新 数据
/// </summary>
public override async Task<dynamic> UpdateAsync(VmodelUpdateInput input)
public override async Task<dynamic> UpdateAsync(VmEditInput input)
{
ThrowIf.IsNull(input.data);
ThrowIf.IsNullOrEmpty(input.data.id, $"更新数据时主键(id)不可为空");
Vmodel vm = input.data.Adapt<Vmodel>();
vm.areaCode = vm.areaCode.ToKebab();
vm.vmCode = vm.vmCode.ToKebab();
vm.navProps.ForEach(a => a.naviModel = null);
vm.modifyTime = DateTime.Now;
vm.modifyId = _user.UserId;
await _db.Updateable(vm).WhereColumns(a => a.id).ExecuteCommandAsync();
await _cache.DelAsync(_dataAccess.GetVmodelCacheKey(vm.id));
return vm;
var ret = await base.UpdateAsync(input);
if (input.data != null)
{
var id = ret as string;
await _cache.DelAsync(_dataAccess.GetVmodelCacheKey(id!));
}
else if (input.items != null)
{
var ids = ret as List<string>;
foreach (var item in ids!)
{
await _cache.DelAsync(_dataAccess.GetVmodelCacheKey(item));
}
}
return ret;
//vm.areaCode = vm.areaCode.ToKebab();
//vm.vmCode = vm.vmCode.ToKebab();
//vm.navProps.ForEach(a => a.naviModel = null);
//vm.modifyTime = DateTime.Now;
//vm.modifyId = _user.UserId;
//await _db.Updateable(vm).WhereColumns(a => a.id).ExecuteCommandAsync();
//return vm;
}
/// <summary>
@@ -116,8 +202,21 @@ public class VmodelAppService : VengineAppService<Vmodel, VmodelGetInput, Vmodel
/// </summary>
public override async Task<dynamic> DeleteAsync(VmDeleteInput input)
{
var ret = await _db.Updateable<Vmodel>().SetColumns(a => a.deleted, 1).Where(a => a.id == input.id).ExecuteCommandAsync();
await _cache.DelAsync(_dataAccess.GetVmodelCacheKey(input.id!));
List<string> ids = new();
if (input.id != null)
{
ids.Add(input.id);
}
else if (input.ids != null)
{
ids.AddRange(input.ids);
}
var ret = await _db.Updateable<Vmodel>().SetColumns(a => a.deleted, 1).Where(a => ids.Contains(a.id)).ExecuteCommandAsync();
foreach (var item in ids)
{
await _cache.DelAsync(_dataAccess.GetVmodelCacheKey(item));
}
return ret;
}
@@ -242,4 +341,5 @@ public class VmodelAppService : VengineAppService<Vmodel, VmodelGetInput, Vmodel
return code;
}
}
}

View File

@@ -13,18 +13,9 @@ namespace Tnb.Vengine.AppService;
public record VmodelCreateData(string areaCode, string vmCode, string vmName, string? dbCode, string tableName, List<VmDbProp> dbProps, List<VmNavPropCreateData> navProps, List<VmCalProp> calProps, int ordinal, short softDelete, short enabled, string? descrip);
public record VmNavPropCreateData(string code, string name, string vmid, eNavigateType navType, string refCode, string refProp, string fkProp);
public record VmodelUpdateData(string id, string areaCode, string vmCode, string vmName, string? dbCode, string tableName, List<VmDbProp> dbProps, List<VmNavPropCreateData> navProps, List<VmCalProp> calProps, int ordinal, short softDelete, short enabled, string? descrip);
public class VmodelGetInput : VmGetInput
{
/// <summary>
/// 模块code, areaCode和vmCode为一组
/// </summary>
public string? areaCode { get; set; }
/// <summary>
/// 模型code, areaCode和vmCode为一组
/// </summary>
public string? vmCode { get; set; }
/// <summary>
/// 是否钻取子模型
/// </summary>
@@ -40,10 +31,6 @@ public class VmodelCreateInput : VmEditInput
return input;
}
}
public class VmodelUpdateInput : VmEditInput
{
public new VmodelUpdateData? data { get; set; }
}
public class VmodelCreateFromTableInput
{
public string? dbCode { get; set; }
@@ -65,17 +52,6 @@ public class VmodelPageCreateInput : VmEditInput
}
}
public class VmodelPageUpdateInput : VmEditInput
{
public new VmodelPage? data { get; set; }
public override VmEditInput ToEditInput()
{
VmEditInput input = this;
input.data = DObject.FromObject(data);
return input;
}
}
public class VmodelPageCreateFromVmodelInput
{
public Guid? viewId { get; set; }

View File

@@ -19,7 +19,7 @@ namespace Tnb.Vengine.AppService;
/// 视图模型服务类
/// </summary>
[ApiDescriptionSettings(Tag = ModuleConst.Tag, Area = ModuleConst.Area, Order = 1104, KeepVerb = true)]
public class VmodelPageAppService : VengineAppService<VmodelPage, VmGetInput, VmodelPage, VmQueryInput, VmGetListInput, VmodelPage, VmodelPageCreateInput, VmodelPageUpdateInput>, IVmodelPageAppService
public class VmodelPageAppService : VengineAppService<VmodelPage, VmGetInput, VmodelPage, VmQueryInput, VmodelPage, VmodelPageCreateInput>, IVmodelPageAppService
{
/// <summary>
/// 构造函数