SelectTable.cs 31 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835
  1. // 大名科技(天津)有限公司版权所有 电话:18020030720 QQ:515096995
  2. //
  3. // 此源代码遵循位于源代码树根目录中的 LICENSE 文件的许可证
  4. namespace Admin.NET.Core.Service;
  5. using AspectCore.Extensions.Reflection;
  6. using Newtonsoft.Json.Linq;
  7. using SqlSugar;
  8. using System;
  9. using System.Collections.Generic;
  10. using System.Dynamic;
  11. using System.Linq;
  12. using System.Text.RegularExpressions;
  13. /// <summary>
  14. ///
  15. /// </summary>
  16. public class SelectTable : ISingleton
  17. {
  18. private readonly IdentityService _identitySvc;
  19. private readonly TableMapper _tableMapper;
  20. private readonly ISqlSugarClient db;
  21. /// <summary>
  22. ///
  23. /// </summary>
  24. /// <param name="identityService"></param>
  25. /// <param name="tableMapper"></param>
  26. /// <param name="dbClient"></param>
  27. public SelectTable(IdentityService identityService, TableMapper tableMapper, ISqlSugarClient dbClient)
  28. {
  29. _identitySvc = identityService;
  30. _tableMapper = tableMapper;
  31. db = dbClient;
  32. }
  33. /// <summary>
  34. /// 判断表名是否正确
  35. /// </summary>
  36. /// <param name="table"></param>
  37. /// <returns></returns>
  38. public bool IsTable(string table)
  39. {
  40. return db.DbMaintenance.GetTableInfoList().Any(it => it.Name.Equals(table, StringComparison.CurrentCultureIgnoreCase));
  41. }
  42. /// <summary>
  43. /// 判断表的列名是否正确
  44. /// </summary>
  45. /// <param name="table"></param>
  46. /// <param name="col"></param>
  47. /// <returns></returns>
  48. public bool IsCol(string table, string col)
  49. {
  50. return db.DbMaintenance.GetColumnInfosByTableName(table).Any(it => it.DbColumnName.Equals(col, StringComparison.CurrentCultureIgnoreCase));
  51. }
  52. /// <summary>
  53. /// 动态调用方法
  54. /// </summary>
  55. /// <param name="funcname"></param>
  56. /// <param name="param"></param>
  57. /// <param name="types"></param>
  58. /// <returns></returns>
  59. public object ExecFunc(string funcname, object[] param, Type[] types)
  60. {
  61. var method = typeof(FuncList).GetMethod(funcname);
  62. var reflector = method.GetReflector();
  63. var result = reflector.Invoke(new FuncList(), param);
  64. return result;
  65. }
  66. private string ToSql(string subtable, int page, int count, int query, string json)
  67. {
  68. JObject values = JObject.Parse(json);
  69. page = values["page"] == null ? page : int.Parse(values["page"].ToString());
  70. count = values["count"] == null ? count : int.Parse(values["count"].ToString());
  71. query = values["query"] == null ? query : int.Parse(values["query"].ToString());
  72. values.Remove("page");
  73. values.Remove("count");
  74. subtable = _tableMapper.GetTableName(subtable);
  75. var tb = sugarQueryable(subtable, "*", values, null);
  76. var xx = tb.Skip((page - 1) * count).Take(10).ToSql();
  77. return xx.Key;
  78. }
  79. /// <summary>
  80. ///
  81. /// </summary>
  82. /// <param name="subtable"></param>
  83. /// <param name="page"></param>
  84. /// <param name="count"></param>
  85. /// <param name="query"></param>
  86. /// <param name="json"></param>
  87. /// <param name="dd"></param>
  88. /// <returns></returns>
  89. /// <exception cref="Exception"></exception>
  90. public Tuple<dynamic, int> GetTableData(string subtable, int page, int count, int query, string json, JObject dd)
  91. {
  92. var role = _identitySvc.GetSelectRole(subtable);
  93. if (!role.Item1)//没有权限返回异常
  94. {
  95. throw new Exception(role.Item2);
  96. }
  97. string selectrole = role.Item2;
  98. subtable = _tableMapper.GetTableName(subtable);
  99. JObject values = JObject.Parse(json);
  100. page = values["page"] == null ? page : int.Parse(values["page"].ToString());
  101. count = values["count"] == null ? count : int.Parse(values["count"].ToString());
  102. query = values["query"] == null ? query : int.Parse(values["query"].ToString());
  103. values.Remove("page");
  104. values.Remove("count");
  105. var tb = sugarQueryable(subtable, selectrole, values, dd);
  106. if (query == 1)//1-总数
  107. return new Tuple<dynamic, int>(new List<object>(), tb.Count());
  108. else
  109. {
  110. if (count > 0)
  111. {
  112. int total = 0;
  113. if (query == 0)//0-对象
  114. return new Tuple<dynamic, int>(tb.ToPageList(page, count), total);
  115. else
  116. //2-以上全部
  117. return new Tuple<dynamic, int>(tb.ToPageList(page, count, ref total), total);
  118. }
  119. else
  120. {
  121. if (query == 0)
  122. return new Tuple<dynamic, int>(tb.ToList(), 0);
  123. else
  124. return new Tuple<dynamic, int>(tb.ToList(), tb.Count());
  125. }
  126. }
  127. }
  128. /// <summary>
  129. ///
  130. /// </summary>
  131. /// <param name="subtable"></param>
  132. /// <param name="json"></param>
  133. /// <param name="dd"></param>
  134. /// <returns></returns>
  135. public dynamic GetFirstData(string subtable, string json, JObject dd)
  136. {
  137. var role = _identitySvc.GetSelectRole(subtable);
  138. if (!role.Item1)//没有权限返回异常
  139. {
  140. throw new Exception(role.Item2);
  141. }
  142. string selectrole = role.Item2;
  143. subtable = _tableMapper.GetTableName(subtable);
  144. JObject values = JObject.Parse(json);
  145. values.Remove("page");
  146. values.Remove("count");
  147. var tb = sugarQueryable(subtable, selectrole, values, dd).First();
  148. var dic = (IDictionary<string, object>)tb;
  149. foreach (var item in values.Properties().Where(it => it.Name.EndsWith("()")))
  150. {
  151. if (!item.Value.IsNullOrEmpty())
  152. {
  153. string func = item.Value.ToString().Substring(0, item.Value.ToString().IndexOf("("));
  154. string param = item.Value.ToString().Substring(item.Value.ToString().IndexOf("(") + 1).TrimEnd(')');
  155. var types = new List<Type>();
  156. var paramss = new List<object>();
  157. foreach (var va in param.Split(','))
  158. {
  159. types.Add(typeof(object));
  160. paramss.Add(tb.Where(it => it.Key.Equals(va)).Select(i => i.Value));
  161. }
  162. dic[item.Name] = ExecFunc(func, paramss.ToArray(), types.ToArray());
  163. }
  164. }
  165. return tb;
  166. }
  167. /// <summary>
  168. /// 解析并查询
  169. /// </summary>
  170. /// <param name="queryJson"></param>
  171. /// <returns></returns>
  172. public JObject Query(string queryJson)
  173. {
  174. JObject resultObj = new();
  175. try
  176. {
  177. JObject queryJobj = JObject.Parse(queryJson);
  178. resultObj = Query(queryJobj);
  179. }
  180. catch (Exception ex)
  181. {
  182. resultObj.Add("code", "500");
  183. resultObj.Add("msg", ex.Message);
  184. }
  185. return resultObj;
  186. }
  187. /// <summary>
  188. /// 单表查询
  189. /// </summary>
  190. /// <param name="queryObj"></param>
  191. /// <param name="nodeName">返回数据的节点名称 默认为 infos</param>
  192. /// <returns></returns>
  193. public JObject QuerySingle(JObject queryObj, string nodeName = "infos")
  194. {
  195. JObject resultObj = new();
  196. resultObj.Add("code", "200");
  197. resultObj.Add("msg", "success");
  198. try
  199. {
  200. int total = 0;
  201. foreach (var item in queryObj)
  202. {
  203. string key = item.Key.Trim();
  204. if (key.EndsWith("[]"))
  205. {
  206. total = QuerySingleList(resultObj, item, nodeName);
  207. }
  208. else if (key.Equals("func"))
  209. {
  210. ExecFunc(resultObj, item);
  211. }
  212. else if (key.Equals("total@"))
  213. {
  214. resultObj.Add("total", total);
  215. }
  216. }
  217. }
  218. catch (Exception ex)
  219. {
  220. resultObj["code"] = "500";
  221. resultObj["msg"] = ex.Message;
  222. }
  223. return resultObj;
  224. }
  225. /// <summary>
  226. /// 获取查询语句
  227. /// </summary>
  228. /// <param name="queryObj"></param>
  229. /// <returns></returns>
  230. public string ToSql(JObject queryObj)
  231. {
  232. foreach (var item in queryObj)
  233. {
  234. string key = item.Key.Trim();
  235. if (key.EndsWith("[]"))
  236. {
  237. return ToSql(item);
  238. }
  239. }
  240. return string.Empty;
  241. }
  242. /// <summary>
  243. /// 解析并查询
  244. /// </summary>
  245. /// <param name="queryObj"></param>
  246. /// <returns></returns>
  247. public JObject Query(JObject queryObj)
  248. {
  249. JObject resultObj = new();
  250. resultObj.Add("code", "200");
  251. resultObj.Add("msg", "success");
  252. try
  253. {
  254. int total = 0;
  255. foreach (var item in queryObj)
  256. {
  257. string key = item.Key.Trim();
  258. if (key.Equals("[]"))
  259. {
  260. total = QueryMoreList(resultObj, item);
  261. }
  262. else if (key.EndsWith("[]"))
  263. {
  264. total = QuerySingleList(resultObj, item);
  265. }
  266. else if (key.Equals("func"))
  267. {
  268. ExecFunc(resultObj, item);
  269. }
  270. else if (key.Equals("total@"))
  271. {
  272. resultObj.Add("total", total);
  273. }
  274. else
  275. {
  276. var template = GetFirstData(key, item.Value.ToString(), resultObj);
  277. if (template != null)
  278. {
  279. resultObj.Add(key, JToken.FromObject(template));
  280. }
  281. }
  282. }
  283. }
  284. catch (Exception ex)
  285. {
  286. resultObj["code"] = "500";
  287. resultObj["msg"] = ex.Message;
  288. }
  289. return resultObj;
  290. }
  291. //单表查询,返回的数据在指定的NodeName节点
  292. private int QuerySingleList(JObject resultObj, KeyValuePair<string, JToken> item, string nodeName)
  293. {
  294. string key = item.Key.Trim();
  295. var jb = JObject.Parse(item.Value.ToString());
  296. int page = jb["page"] == null ? 0 : int.Parse(jb["page"].ToString());
  297. int count = jb["count"] == null ? 10 : int.Parse(jb["count"].ToString());
  298. int query = jb["query"] == null ? 0 : int.Parse(jb["query"].ToString());
  299. int total = 0;
  300. jb.Remove("page"); jb.Remove("count"); jb.Remove("query");
  301. var htt = new JArray();
  302. foreach (var t in jb)
  303. {
  304. var datas = GetTableData(t.Key, page, count, query, t.Value.ToString(), null);
  305. if (query > 0)
  306. {
  307. total = datas.Item2;
  308. }
  309. foreach (var data in datas.Item1)
  310. {
  311. htt.Add(JToken.FromObject(data));
  312. }
  313. }
  314. if (!string.IsNullOrEmpty(nodeName))
  315. {
  316. resultObj.Add(nodeName, htt);
  317. }
  318. else
  319. resultObj.Add(key, htt);
  320. return total;
  321. }
  322. private string ToSql(KeyValuePair<string, JToken> item)
  323. {
  324. string key = item.Key.Trim();
  325. var jb = JObject.Parse(item.Value.ToString());
  326. int page = jb["page"] == null ? 0 : int.Parse(jb["page"].ToString());
  327. int count = jb["count"] == null ? 10 : int.Parse(jb["count"].ToString());
  328. int query = jb["query"] == null ? 0 : int.Parse(jb["query"].ToString());
  329. jb.Remove("page"); jb.Remove("count"); jb.Remove("query");
  330. var htt = new JArray();
  331. foreach (var t in jb)
  332. {
  333. return ToSql(t.Key, page, count, query, t.Value.ToString());
  334. }
  335. return string.Empty;
  336. }
  337. //单表查询
  338. private int QuerySingleList(JObject resultObj, KeyValuePair<string, JToken> item)
  339. {
  340. string key = item.Key.Trim();
  341. return QuerySingleList(resultObj, item, key);
  342. }
  343. //多列表查询
  344. private int QueryMoreList(JObject resultObj, KeyValuePair<string, JToken> item)
  345. {
  346. int total = 0;
  347. var jb = JObject.Parse(item.Value.ToString());
  348. var page = jb["page"] == null ? 0 : int.Parse(jb["page"].ToString());
  349. var count = jb["count"] == null ? 10 : int.Parse(jb["count"].ToString());
  350. var query = jb["query"] == null ? 0 : int.Parse(jb["query"].ToString());
  351. jb.Remove("page"); jb.Remove("count"); jb.Remove("query");
  352. var htt = new JArray();
  353. List<string> tables = new(), where = new();
  354. foreach (var t in jb)
  355. {
  356. tables.Add(t.Key); where.Add(t.Value.ToString());
  357. }
  358. if (tables.Count > 0)
  359. {
  360. string table = tables[0];
  361. var temp = GetTableData(table, page, count, query, where[0], null);
  362. if (query > 0)
  363. {
  364. total = temp.Item2;
  365. }
  366. foreach (var dd in temp.Item1)
  367. {
  368. var zht = new JObject();
  369. zht.Add(table, JToken.FromObject(dd));
  370. for (int i = 1; i < tables.Count; i++)
  371. {
  372. string subtable = tables[i];
  373. if (subtable.EndsWith("[]"))
  374. {
  375. subtable = subtable.TrimEnd("[]".ToCharArray());
  376. var jbb = JObject.Parse(where[i]);
  377. page = jbb["page"] == null ? 0 : int.Parse(jbb["page"].ToString());
  378. count = jbb["count"] == null ? 0 : int.Parse(jbb["count"].ToString());
  379. var lt = new JArray();
  380. foreach (var d in GetTableData(subtable, page, count, query, jbb[subtable].ToString(), zht).Item1)
  381. {
  382. lt.Add(JToken.FromObject(d));
  383. }
  384. zht.Add(tables[i], lt);
  385. }
  386. else
  387. {
  388. var ddf = GetFirstData(subtable, where[i].ToString(), zht);
  389. if (ddf != null)
  390. {
  391. zht.Add(subtable, JToken.FromObject(ddf));
  392. }
  393. }
  394. }
  395. htt.Add(zht);
  396. }
  397. }
  398. if (query != 1)
  399. {
  400. resultObj.Add("[]", htt);
  401. }
  402. return total;
  403. }
  404. private void ExecFunc(JObject resultObj, KeyValuePair<string, JToken> item)
  405. {
  406. JObject jb = JObject.Parse(item.Value.ToString());
  407. Type type = typeof(FuncList);
  408. var dataJObj = new JObject();
  409. foreach (var f in jb)
  410. {
  411. var types = new List<Type>();
  412. var param = new List<object>();
  413. foreach (var va in JArray.Parse(f.Value.ToString()))
  414. {
  415. types.Add(typeof(object));
  416. param.Add(va);
  417. }
  418. dataJObj.Add(f.Key, JToken.FromObject(ExecFunc(f.Key, param.ToArray(), types.ToArray())));
  419. }
  420. resultObj.Add("func", dataJObj);
  421. }
  422. private ISugarQueryable<ExpandoObject> sugarQueryable(string subtable, string selectrole, JObject values, JObject dd)
  423. {
  424. if (!IsTable(subtable))
  425. {
  426. throw new Exception($"表名{subtable}不正确!");
  427. }
  428. var tb = db.Queryable(subtable, "tb");
  429. if (!values["@column"].IsNullOrEmpty())
  430. {
  431. ProcessColumn(subtable, selectrole, values, tb);
  432. }
  433. else
  434. {
  435. tb.Select(selectrole);
  436. }
  437. List<IConditionalModel> conModels = new();
  438. if (!values["identity"].IsNullOrEmpty())
  439. {
  440. conModels.Add(new ConditionalModel() { FieldName = values["identity"].ToString(), ConditionalType = ConditionalType.Equal, FieldValue = _identitySvc.GetUserIdentity() });
  441. }
  442. foreach (var va in values)
  443. {
  444. string vakey = va.Key.Trim();
  445. string fieldValue = va.Value.ToString();
  446. if (vakey.EndsWith("$"))//模糊查询
  447. {
  448. FuzzyQuery(subtable, conModels, va);
  449. }
  450. else if (vakey.EndsWith("{}"))//逻辑运算
  451. {
  452. ConditionQuery(subtable, conModels, va);
  453. }
  454. else if (vakey.EndsWith("%"))//between查询
  455. {
  456. ConditionBetween(subtable, conModels, va);
  457. }
  458. else if (vakey.EndsWith("@") && dd != null) // 关联上一个table
  459. {
  460. string[] str = fieldValue.Split('/');
  461. string value = string.Empty;
  462. if (str.Length == 3)
  463. {
  464. value = dd[str[1]][str[2]].ToString();
  465. }
  466. else if (str.Length == 2)
  467. {
  468. value = dd[str[0]][str[1]].ToString();
  469. }
  470. conModels.Add(new ConditionalModel() { FieldName = vakey.TrimEnd('@'), ConditionalType = ConditionalType.Equal, FieldValue = value });
  471. }
  472. else if (IsCol(subtable, vakey)) //其他where条件
  473. {
  474. conModels.Add(new ConditionalModel() { FieldName = vakey, ConditionalType = ConditionalType.Equal, FieldValue = fieldValue });
  475. }
  476. }
  477. tb.Where(conModels);
  478. //排序
  479. ProcessOrder(subtable, values, tb);
  480. //分组
  481. ProcessGroup(subtable, values, tb);
  482. //Having
  483. ProcessHaving(values, tb);
  484. return tb;
  485. }
  486. //处理字段重命名 "@column":"toId:parentId",对应SQL是toId AS parentId,将查询的字段toId变为parentId返回
  487. private void ProcessColumn(string subtable, string selectrole, JObject values, ISugarQueryable<ExpandoObject> tb)
  488. {
  489. var str = new System.Text.StringBuilder(100);
  490. foreach (var item in values["@column"].ToString().Split(','))
  491. {
  492. string[] ziduan = item.Split(':');
  493. string colName = ziduan[0];
  494. var ma = new Regex(@"\((\w+)\)").Match(colName);
  495. //处理max,min这样的函数
  496. if (ma.Success && ma.Groups.Count > 1)
  497. {
  498. colName = ma.Groups[1].Value;
  499. }
  500. //判断列表是否有权限 sum(1),sum(*),Count(1)这样的值直接有效
  501. if (colName == "*" || int.TryParse(colName, out int colNumber) || (IsCol(subtable, colName) && _identitySvc.ColIsRole(colName, selectrole.Split(','))))
  502. {
  503. if (ziduan.Length > 1)
  504. {
  505. if (ziduan[1].Length > 20)
  506. {
  507. throw new Exception("别名不能超过20个字符");
  508. }
  509. str.Append(ziduan[0] + " as " + ReplaceSQLChar(ziduan[1]) + ",");
  510. }
  511. else
  512. str.Append(ziduan[0] + ",");
  513. }
  514. }
  515. if (string.IsNullOrEmpty(str.ToString()))
  516. {
  517. throw new Exception($"表名{subtable}没有可查询的字段!");
  518. }
  519. tb.Select(str.ToString().TrimEnd(','));
  520. }
  521. // "@having":"function0(...)?value0;function1(...)?value1;function2(...)?value2...",
  522. // SQL函数条件,一般和 @group一起用,函数一般在 @column里声明
  523. private void ProcessHaving(JObject values, ISugarQueryable<ExpandoObject> tb)
  524. {
  525. if (!values["@having"].IsNullOrEmpty())
  526. {
  527. List<IConditionalModel> hw = new();
  528. List<string> havingItems = new();
  529. if (values["@having"].HasValues)
  530. {
  531. havingItems = values["@having"].Select(p => p.ToString()).ToList();
  532. }
  533. else
  534. {
  535. havingItems.Add(values["@having"].ToString());
  536. }
  537. foreach (var item in havingItems)
  538. {
  539. string and = item.ToString();
  540. var model = new ConditionalModel();
  541. if (and.Contains(">="))
  542. {
  543. model.FieldName = and.Split(new string[] { ">=" }, StringSplitOptions.RemoveEmptyEntries)[0];
  544. model.ConditionalType = ConditionalType.GreaterThanOrEqual;
  545. model.FieldValue = and.Split(new string[] { ">=" }, StringSplitOptions.RemoveEmptyEntries)[1];
  546. }
  547. else if (and.Contains("<="))
  548. {
  549. model.FieldName = and.Split(new string[] { "<=" }, StringSplitOptions.RemoveEmptyEntries)[0];
  550. model.ConditionalType = ConditionalType.LessThanOrEqual;
  551. model.FieldValue = and.Split(new string[] { "<=" }, StringSplitOptions.RemoveEmptyEntries)[1];
  552. }
  553. else if (and.Contains('>'))
  554. {
  555. model.FieldName = and.Split(new string[] { ">" }, StringSplitOptions.RemoveEmptyEntries)[0];
  556. model.ConditionalType = ConditionalType.GreaterThan;
  557. model.FieldValue = and.Split(new string[] { ">" }, StringSplitOptions.RemoveEmptyEntries)[1];
  558. }
  559. else if (and.Contains('<'))
  560. {
  561. model.FieldName = and.Split(new string[] { "<" }, StringSplitOptions.RemoveEmptyEntries)[0];
  562. model.ConditionalType = ConditionalType.LessThan;
  563. model.FieldValue = and.Split(new string[] { "<" }, StringSplitOptions.RemoveEmptyEntries)[1];
  564. }
  565. else if (and.Contains("!="))
  566. {
  567. model.FieldName = and.Split(new string[] { "!=" }, StringSplitOptions.RemoveEmptyEntries)[0];
  568. model.ConditionalType = ConditionalType.NoEqual;
  569. model.FieldValue = and.Split(new string[] { "!=" }, StringSplitOptions.RemoveEmptyEntries)[1];
  570. }
  571. else if (and.Contains('='))
  572. {
  573. model.FieldName = and.Split(new string[] { "=" }, StringSplitOptions.RemoveEmptyEntries)[0];
  574. model.ConditionalType = ConditionalType.Equal;
  575. model.FieldValue = and.Split(new string[] { "=" }, StringSplitOptions.RemoveEmptyEntries)[1];
  576. }
  577. hw.Add(model);
  578. }
  579. tb.Having(string.Join(",", havingItems));
  580. }
  581. }
  582. //"@group":"column0,column1...",分组方式。如果 @column里声明了Table的id,则id也必须在 @group中声明;其它情况下必须满足至少一个条件:
  583. //1.分组的key在 @column里声明
  584. //2.Table主键在 @group中声明
  585. private void ProcessGroup(string subtable, JObject values, ISugarQueryable<ExpandoObject> tb)
  586. {
  587. if (!values["@group"].IsNullOrEmpty())
  588. {
  589. var str = new System.Text.StringBuilder(100);
  590. foreach (var and in values["@group"].ToString().Split(','))
  591. {
  592. if (IsCol(subtable, and))
  593. {
  594. str.Append(and + ",");
  595. }
  596. }
  597. tb.GroupBy(str.ToString().TrimEnd(','));
  598. }
  599. }
  600. //处理排序 "@order":"name-,id"查询按 name降序、id默认顺序 排序的User数组
  601. private void ProcessOrder(string subtable, JObject values, ISugarQueryable<ExpandoObject> tb)
  602. {
  603. if (!values["@order"].IsNullOrEmpty())
  604. {
  605. foreach (var item in values["@order"].ToString().Split(','))
  606. {
  607. string col = item.Replace("-", "").Replace("+", "");
  608. if (IsCol(subtable, col))
  609. {
  610. if (item.EndsWith("-"))
  611. {
  612. tb.OrderBy($"{col} desc");
  613. }
  614. else if (item.EndsWith("+"))
  615. {
  616. tb.OrderBy($"{col} asc");
  617. }
  618. else
  619. {
  620. tb.OrderBy($"{col}");
  621. }
  622. }
  623. }
  624. }
  625. }
  626. //条件查询 "key{}":"条件0,条件1...",条件为任意SQL比较表达式字符串,非Number类型必须用''包含条件的值,如'a'
  627. //&, |, ! 逻辑运算符,对应数据库 SQL 中的 AND, OR, NOT。
  628. // 横或纵与:同一字段的值内条件默认 | 或连接,不同字段的条件默认 & 与连接。
  629. // ① & 可用于"key&{}":"条件"等
  630. // ② | 可用于"key|{}":"条件", "key|{}":[] 等,一般可省略
  631. // ③ ! 可单独使用,如"key!":Object,也可像&,|一样配合其他功能符使用
  632. private void ConditionQuery(string subtable, List<IConditionalModel> conModels, KeyValuePair<string, JToken> va)
  633. {
  634. string vakey = va.Key.Trim();
  635. string field = vakey.TrimEnd("{}".ToCharArray());
  636. if (va.Value.HasValues)
  637. {
  638. List<string> inValues = new();
  639. foreach (var cm in va.Value)
  640. {
  641. inValues.Add(cm.ToString());
  642. }
  643. conModels.Add(new ConditionalModel() { FieldName = field, ConditionalType = field.EndsWith("!") ? ConditionalType.NotIn : ConditionalType.In, FieldValue = string.Join(",", inValues) });
  644. }
  645. else
  646. {
  647. var ddt = new List<KeyValuePair<WhereType, ConditionalModel>>();
  648. foreach (var and in va.Value.ToString().Split(','))
  649. {
  650. var model = new ConditionalModel();
  651. model.FieldName = field.TrimEnd("&".ToCharArray());//处理&()的查询方式
  652. if (and.StartsWith(">="))
  653. {
  654. model.ConditionalType = ConditionalType.GreaterThanOrEqual;
  655. model.FieldValue = and.TrimStart(">=".ToCharArray());
  656. }
  657. else if (and.StartsWith("<="))
  658. {
  659. model.ConditionalType = ConditionalType.LessThanOrEqual;
  660. model.FieldValue = and.TrimStart("<=".ToCharArray());
  661. }
  662. else if (and.StartsWith(">"))
  663. {
  664. model.ConditionalType = ConditionalType.GreaterThan;
  665. model.FieldValue = and.TrimStart('>');
  666. }
  667. else if (and.StartsWith("<"))
  668. {
  669. model.ConditionalType = ConditionalType.LessThan;
  670. model.FieldValue = and.TrimStart('<');
  671. }
  672. ddt.Add(new KeyValuePair<WhereType, ConditionalModel>((field.EndsWith("&") ? WhereType.And : WhereType.Or), model));
  673. }
  674. conModels.Add(new ConditionalCollections() { ConditionalList = ddt });
  675. }
  676. }
  677. //"key%":"start,end" => "key%":["start,end"],其中 start 和 end 都只能为 Boolean, Number, String 中的一种,如 "2017-01-01,2019-01-01" ,["1,90000", "82001,100000"] ,可用于连续范围内的筛选
  678. private void ConditionBetween(string subtable, List<IConditionalModel> conModels, KeyValuePair<string, JToken> va)
  679. {
  680. string vakey = va.Key.Trim();
  681. string field = vakey.TrimEnd("%".ToCharArray());
  682. List<string> inValues = new();
  683. if (va.Value.HasValues)
  684. {
  685. foreach (var cm in va.Value)
  686. {
  687. inValues.Add(cm.ToString());
  688. }
  689. }
  690. else
  691. {
  692. inValues.Add(va.Value.ToString());
  693. }
  694. for (var i = 0; i < inValues.Count; i++)
  695. {
  696. var fileds = inValues[i].Split(',');
  697. if (fileds.Length == 2)
  698. {
  699. var ddt = new List<KeyValuePair<WhereType, ConditionalModel>>();
  700. var leftCondition = new ConditionalModel()
  701. {
  702. FieldName = field,
  703. ConditionalType = ConditionalType.GreaterThanOrEqual,
  704. FieldValue = fileds[0]
  705. };
  706. ddt.Add(new KeyValuePair<WhereType, ConditionalModel>(i == 0 ? WhereType.And : WhereType.Or, leftCondition));
  707. var rightCondition = new ConditionalModel()
  708. {
  709. FieldName = field,
  710. ConditionalType = ConditionalType.LessThanOrEqual,
  711. FieldValue = fileds[1]
  712. };
  713. ddt.Add(new KeyValuePair<WhereType, ConditionalModel>(WhereType.And, rightCondition));
  714. conModels.Add(new ConditionalCollections() { ConditionalList = ddt });
  715. }
  716. }
  717. }
  718. //模糊搜索 "key$":"SQL搜索表达式" => "key$":["SQL搜索表达式"],任意SQL搜索表达式字符串,如 %key%(包含key), key%(以key开始), %k%e%y%(包含字母k,e,y) 等,%表示任意字符
  719. private void FuzzyQuery(string subtable, List<IConditionalModel> conModels, KeyValuePair<string, JToken> va)
  720. {
  721. string vakey = va.Key.Trim();
  722. string fieldValue = va.Value.ToString();
  723. var conditionalType = ConditionalType.Like;
  724. if (IsCol(subtable, vakey.TrimEnd('$')))
  725. {
  726. //支持三种like查询
  727. if (fieldValue.StartsWith("%") && fieldValue.EndsWith("%"))
  728. {
  729. conditionalType = ConditionalType.Like;
  730. }
  731. else if (fieldValue.StartsWith("%"))
  732. {
  733. conditionalType = ConditionalType.LikeRight;
  734. }
  735. else if (fieldValue.EndsWith("%"))
  736. {
  737. conditionalType = ConditionalType.LikeLeft;
  738. }
  739. conModels.Add(new ConditionalModel() { FieldName = vakey.TrimEnd('$'), ConditionalType = conditionalType, FieldValue = fieldValue.TrimEnd("%".ToArray()).TrimStart("%".ToArray()) });
  740. }
  741. }
  742. public static string ReplaceSQLChar(string str)
  743. {
  744. if (str == String.Empty)
  745. return String.Empty;
  746. str = str.Replace("'", "");
  747. str = str.Replace(";", "");
  748. str = str.Replace(",", "");
  749. str = str.Replace("?", "");
  750. str = str.Replace("<", "");
  751. str = str.Replace(">", "");
  752. str = str.Replace("(", "");
  753. str = str.Replace(")", "");
  754. str = str.Replace("@", "");
  755. str = str.Replace("=", "");
  756. str = str.Replace("+", "");
  757. str = str.Replace("*", "");
  758. str = str.Replace("&", "");
  759. str = str.Replace("#", "");
  760. str = str.Replace("%", "");
  761. str = str.Replace("$", "");
  762. str = str.Replace("\"", "");
  763. //删除与数据库相关的词
  764. str = Regex.Replace(str, "delete from", "", RegexOptions.IgnoreCase);
  765. str = Regex.Replace(str, "drop table", "", RegexOptions.IgnoreCase);
  766. str = Regex.Replace(str, "truncate", "", RegexOptions.IgnoreCase);
  767. str = Regex.Replace(str, "xp_cmdshell", "", RegexOptions.IgnoreCase);
  768. str = Regex.Replace(str, "exec master", "", RegexOptions.IgnoreCase);
  769. str = Regex.Replace(str, "net localgroup administrators", "", RegexOptions.IgnoreCase);
  770. str = Regex.Replace(str, "net user", "", RegexOptions.IgnoreCase);
  771. str = Regex.Replace(str, "-", "", RegexOptions.IgnoreCase);
  772. str = Regex.Replace(str, "truncate", "", RegexOptions.IgnoreCase);
  773. return str;
  774. }
  775. }