JsonTableDataSource.cs 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470
  1. using FastReport.Data.JsonConnection.JsonParser;
  2. using FastReport.Utils;
  3. using System;
  4. using System.Collections;
  5. using System.Collections.Generic;
  6. using System.ComponentModel;
  7. namespace FastReport.Data.JsonConnection
  8. {
  9. /// <summary>
  10. /// JsonTableDataSource present a json array object
  11. /// </summary>
  12. public class JsonTableDataSource : TableDataSource
  13. {
  14. #region Private Fields
  15. private JsonArray _json;
  16. private bool updateSchema;
  17. private bool simpleStructure;
  18. private string tableData;
  19. #endregion Private Fields
  20. #region Public Properties
  21. /// <summary>
  22. /// Gets or sets value for force update schema on init schema
  23. /// </summary>
  24. [Browsable(false)]
  25. public bool UpdateSchema
  26. {
  27. get
  28. {
  29. return updateSchema;
  30. }
  31. set
  32. {
  33. updateSchema = value;
  34. }
  35. }
  36. /// <summary>
  37. /// Get or sets simplify mode for array types
  38. /// </summary>
  39. [Browsable(false)]
  40. public bool SimpleStructure
  41. {
  42. get
  43. {
  44. return simpleStructure;
  45. }
  46. set
  47. {
  48. simpleStructure = value;
  49. }
  50. }
  51. /// <inheritdoc />
  52. [Browsable(false)]
  53. public override string TableData
  54. {
  55. get
  56. {
  57. if (String.IsNullOrEmpty(tableData))
  58. {
  59. tableData = Json.ToString();
  60. }
  61. return tableData;
  62. }
  63. set
  64. {
  65. tableData = value;
  66. }
  67. }
  68. #endregion Public Properties
  69. #region Internal Properties
  70. internal JsonArray Json
  71. {
  72. get
  73. {
  74. if (_json == null)
  75. {
  76. if (StoreData && !String.IsNullOrEmpty(tableData))
  77. {
  78. _json = JsonBase.FromString(tableData) as JsonArray;
  79. }
  80. else
  81. {
  82. _json = GetJson(Parent, this) as JsonArray;
  83. }
  84. if (_json == null)
  85. _json = new JsonArray();
  86. }
  87. return _json;
  88. }
  89. set
  90. {
  91. _json = value;
  92. }
  93. }
  94. #endregion Internal Properties
  95. #region Private Properties
  96. private int CurrentIndex
  97. {
  98. get
  99. {
  100. if (currentRow is int)
  101. return (int)currentRow;
  102. if (CurrentRowNo < Rows.Count)
  103. {
  104. object result = Rows[CurrentRowNo];
  105. if (result is int)
  106. return (int)result;
  107. }
  108. return CurrentRowNo;
  109. }
  110. }
  111. #endregion Private Properties
  112. #region Public Constructors
  113. /// <inheritdoc/>
  114. public JsonTableDataSource()
  115. {
  116. DataType = typeof(JsonArray);
  117. }
  118. #endregion Public Constructors
  119. #region Public Methods
  120. /// <inheritdoc/>
  121. public override void InitializeComponent()
  122. {
  123. base.InitializeComponent();
  124. Json = null;
  125. }
  126. /// <inheritdoc/>
  127. public override void InitSchema()
  128. {
  129. if (Columns.Count == 0 || UpdateSchema && !StoreData)
  130. {
  131. if (Connection is JsonDataSourceConnection)
  132. {
  133. JsonDataSourceConnection con = Connection as JsonDataSourceConnection;
  134. InitSchema(this, con.JsonSchema, con.SimpleStructure);
  135. }
  136. }
  137. UpdateSchema = false;
  138. }
  139. /// <inheritdoc/>
  140. public override void LoadData(ArrayList rows)
  141. {
  142. Json = null;
  143. // JSON is calculated property, no problem with null
  144. if (rows != null && Json != null)
  145. {
  146. rows.Clear();
  147. int count = Json.Count;
  148. for (int i = 0; i < count; i++)
  149. {
  150. rows.Add(i);
  151. }
  152. }
  153. }
  154. #endregion Public Methods
  155. #region Internal Methods
  156. internal static object GetJsonBaseByColumn(Base parentColumn, Column column)
  157. {
  158. if (parentColumn is JsonTableDataSource)
  159. {
  160. JsonTableDataSource jsonTableDataSource = parentColumn as JsonTableDataSource;
  161. if (jsonTableDataSource.SimpleStructure)
  162. {
  163. if (!String.IsNullOrEmpty(column.PropName))
  164. {
  165. var obj = (parentColumn as JsonTableDataSource).Json[(parentColumn as JsonTableDataSource).CurrentIndex];
  166. return (obj as JsonBase)[column.PropName];
  167. }
  168. }
  169. else
  170. {
  171. switch (column.PropName)
  172. {
  173. case "item":
  174. return (parentColumn as JsonTableDataSource).Json[(parentColumn as JsonTableDataSource).CurrentIndex];
  175. }
  176. JsonTableDataSource source = column as JsonTableDataSource;
  177. return source.Json;
  178. }
  179. }
  180. if (parentColumn is Column && !String.IsNullOrEmpty(column.PropName))
  181. {
  182. object json = GetJsonBaseByColumn(parentColumn.Parent, parentColumn as Column);
  183. if (json is JsonBase)
  184. {
  185. return (json as JsonBase)[column.PropName];
  186. }
  187. }
  188. return null;
  189. }
  190. internal static object GetValueByColumn(Base parentColumn, Column column)
  191. {
  192. if (parentColumn is JsonTableDataSource)
  193. {
  194. switch (column.PropName)
  195. {
  196. case "index":
  197. return (parentColumn as JsonTableDataSource).CurrentIndex;
  198. case "array":
  199. return (parentColumn as JsonTableDataSource).Json;
  200. }
  201. }
  202. object json = GetJsonBaseByColumn(parentColumn, column);
  203. return json;
  204. }
  205. internal static void InitSchema(Column table, JsonSchema schema, bool simpleStructure)
  206. {
  207. List<Column> saveColumns = new List<Column>();
  208. switch (schema.Type)
  209. {
  210. case "object":
  211. foreach (KeyValuePair<string, JsonSchema> kv in schema.Properties)
  212. {
  213. if (kv.Value.Type == "object")
  214. {
  215. Column c = new Column();
  216. c.Name = kv.Key;
  217. c.Alias = kv.Key;
  218. c.PropName = kv.Key;
  219. c.DataType = kv.Value.DataType;
  220. c = UpdateColumn(table, c, saveColumns);
  221. InitSchema(c, kv.Value, simpleStructure);
  222. }
  223. else if (kv.Value.Type == "array")
  224. {
  225. Column c = new JsonTableDataSource();
  226. c.Name = kv.Key;
  227. c.Alias = kv.Key;
  228. c.PropName = kv.Key;
  229. c.DataType = kv.Value.DataType;
  230. c = UpdateColumn(table, c, saveColumns);
  231. InitSchema(c, kv.Value, simpleStructure);
  232. }
  233. else
  234. {
  235. Column c = new Column();
  236. c.Name = kv.Key;
  237. c.Alias = kv.Key;
  238. c.PropName = kv.Key;
  239. c.DataType = kv.Value.DataType;
  240. c.SetBindableControlType(c.DataType);
  241. UpdateColumn(table, c, saveColumns);
  242. }
  243. }
  244. break;
  245. case "array":
  246. JsonSchema items = schema.Items;
  247. bool simpleArray = false;
  248. if (table is JsonTableDataSource)
  249. {
  250. JsonTableDataSource jsonTableDataSource = table as JsonTableDataSource;
  251. simpleArray = jsonTableDataSource.SimpleStructure =
  252. simpleStructure & items.Type == "object";
  253. }
  254. if (simpleArray)
  255. {
  256. // remake schema in simplify mode
  257. InitSchema(table, items, simpleStructure);
  258. // and return, no need to clear column data
  259. // in this case this method has no control to columns
  260. return;
  261. }
  262. {
  263. Column c = new Column();
  264. c.Name = "index";
  265. c.Alias = "index";
  266. c.PropName = "index";
  267. c.DataType = typeof(int);
  268. UpdateColumn(table, c, saveColumns);
  269. }
  270. {
  271. Column c;
  272. bool iSchema = false;
  273. if (items.Type == "object")
  274. {
  275. c = new Column();
  276. iSchema = true;
  277. }
  278. else if (items.Type == "array")
  279. {
  280. c = new JsonTableDataSource();
  281. iSchema = true;
  282. }
  283. else
  284. {
  285. c = new Column();
  286. }
  287. c.Name = "item";
  288. c.Alias = "item";
  289. c.PropName = "item";
  290. c.DataType = items.DataType;
  291. c = UpdateColumn(table, c, saveColumns);
  292. if (iSchema)
  293. InitSchema(c, items, simpleStructure);
  294. }
  295. {
  296. Column c = new Column();
  297. c.Name = "array";
  298. c.Alias = "array";
  299. c.PropName = "array";
  300. c.DataType = typeof(JsonBase);
  301. UpdateColumn(table, c, saveColumns);
  302. }
  303. break;
  304. }
  305. for (int i = 0; i < table.Columns.Count; i++)
  306. {
  307. if (!(table.Columns[i].Calculated || saveColumns.Contains(table.Columns[i])))
  308. {
  309. table.Columns.RemoveAt(i);
  310. i--;
  311. }
  312. }
  313. }
  314. internal object GetJson(Base parentColumn, Column column)
  315. {
  316. if(parentColumn is IJsonProviderSourceConnection)
  317. {
  318. return (parentColumn as IJsonProviderSourceConnection).GetJson(this);
  319. }
  320. //if (parentColumn is ESDataSourceConnection)
  321. // parentColumn = (parentColumn as ESDataSourceConnection).GetParentJTDSByName(column.Name);
  322. if (parentColumn is JsonTableDataSource)
  323. {
  324. //if (parentColumn.Parent is ESDataSourceConnection)
  325. // parentColumn.Parent = (parentColumn.Parent as ESDataSourceConnection).GetParentJTDSByName(parentColumn.Name);
  326. // Here it’s completely fail,
  327. // I commented out the code with ElasticSearch,
  328. // when you bring ElasticSearch into a separate plugin,
  329. // then this will need to be corrected
  330. JsonTableDataSource source = parentColumn as JsonTableDataSource;
  331. if (source.SimpleStructure)
  332. {
  333. object parentJson = source.Json[source.CurrentRowNo];
  334. if (parentJson is JsonBase && !String.IsNullOrEmpty(column.PropName))
  335. {
  336. return (parentJson as JsonBase)[column.PropName];
  337. }
  338. }
  339. else
  340. {
  341. return source.Json[source.CurrentRowNo] as object;
  342. }
  343. }
  344. else if (parentColumn is Column)
  345. {
  346. object parentJson = GetJson(parentColumn.Parent, parentColumn as Column);
  347. if (parentJson is JsonBase && !String.IsNullOrEmpty(column.PropName))
  348. {
  349. return (parentJson as JsonBase)[column.PropName];
  350. }
  351. }
  352. return null;
  353. }
  354. #endregion Internal Methods
  355. #region Protected Methods
  356. /// <inheritdoc/>
  357. protected override object GetValue(Column column)
  358. {
  359. return GetValueByColumn(column.Parent, column);
  360. }
  361. /// <inheritdoc/>
  362. protected override object GetValue(string alias)
  363. {
  364. // TODO TEST
  365. Column column = this;
  366. string[] colAliases = alias.Split('.');
  367. foreach (string colAlias in colAliases)
  368. {
  369. column = column.Columns.FindByAlias(colAlias);
  370. if (column == null)
  371. return null;
  372. }
  373. return GetValueByColumn(column.Parent, column);
  374. }
  375. #endregion Protected Methods
  376. #region Private Methods
  377. private static Column UpdateColumn(Column table, Column c, List<Column> list)
  378. {
  379. foreach (Column child in table.Columns)
  380. {
  381. if (child.PropName == c.PropName)
  382. {
  383. child.DataType = c.DataType;
  384. list.Add(child);
  385. return child;
  386. }
  387. }
  388. table.AddChild(c);
  389. list.Add(c);
  390. return c;
  391. }
  392. #endregion Private Methods
  393. /// <inheritdoc/>
  394. public override void Serialize(FRWriter writer)
  395. {
  396. base.Serialize(writer);
  397. if (SimpleStructure)
  398. {
  399. writer.WriteBool("SimpleStructure", SimpleStructure);
  400. }
  401. }
  402. }
  403. }