SelectTable.cs 31 KB

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