XmlDataDocument.cs 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630
  1. //
  2. // mcs/class/System.Data/System.Xml/XmlDataDocument.cs
  3. //
  4. // Purpose: Provides a W3C XML DOM Document to interact with
  5. // relational data in a DataSet
  6. //
  7. // class: XmlDataDocument
  8. // assembly: System.Data.dll
  9. // namespace: System.Xml
  10. //
  11. // Author:
  12. // Daniel Morgan <[email protected]>
  13. // Ville Palo <[email protected]>
  14. //
  15. // (c)copyright 2002 Daniel Morgan
  16. //
  17. // XmlDataDocument is included within the Mono Class Library.
  18. //
  19. using System;
  20. using System.Data;
  21. using System.IO;
  22. using System.Text;
  23. using System.Xml.XPath;
  24. using System.Collections;
  25. using System.Globalization;
  26. namespace System.Xml {
  27. public class XmlDataDocument : XmlDocument {
  28. #region Fields
  29. private DataSet dataSet;
  30. private bool isReadOnly = false;
  31. private int dataRowID = 1;
  32. private ArrayList dataRowIDList = new ArrayList ();
  33. // this is needed for inserting new row to datatable via xml
  34. private Hashtable TempTable = new Hashtable ();
  35. #endregion // Fields
  36. #region Constructors
  37. public XmlDataDocument() {
  38. dataSet = new DataSet();
  39. this.NodeChanged += new XmlNodeChangedEventHandler (OnNodeChanged);
  40. //this.NodeChanging += new XmlNodeChangedEventHandler (OnXmlDataColumnChanged);
  41. this.NodeInserting += new XmlNodeChangedEventHandler (OnNodeInserting);
  42. this.NodeRemoved += new XmlNodeChangedEventHandler (OnNodeRemoved);
  43. this.NodeInserted += new XmlNodeChangedEventHandler (OnNodeInserted);
  44. }
  45. public XmlDataDocument(DataSet dataset) {
  46. this.dataSet = dataset;
  47. this.NodeChanged += new XmlNodeChangedEventHandler (OnNodeChanged);
  48. //this.NodeChanging += new XmlNodeChangedEventHandler (OnXmlDataColumnChanged);
  49. this.NodeInserting += new XmlNodeChangedEventHandler (OnNodeInserting);
  50. this.NodeRemoved += new XmlNodeChangedEventHandler (OnNodeRemoved);
  51. this.NodeInserted += new XmlNodeChangedEventHandler (OnNodeInserted);
  52. }
  53. #endregion // Constructors
  54. #region Public Properties
  55. public override string BaseURI {
  56. [MonoTODO]
  57. get {
  58. // TODO: why are we overriding?
  59. return base.BaseURI;
  60. }
  61. }
  62. public DataSet DataSet {
  63. [MonoTODO]
  64. get {
  65. return dataSet;
  66. }
  67. }
  68. // override inheritted method from XmlDocument
  69. public override string InnerXml {
  70. [MonoTODO]
  71. get {
  72. throw new NotImplementedException();
  73. }
  74. [MonoTODO]
  75. set {
  76. throw new NotImplementedException();
  77. }
  78. }
  79. public override bool IsReadOnly {
  80. [MonoTODO]
  81. get {
  82. return isReadOnly;
  83. }
  84. }
  85. // Item indexer
  86. public override XmlElement this[string name] {
  87. [MonoTODO]
  88. get {
  89. throw new NotImplementedException();
  90. }
  91. }
  92. // Item indexer
  93. public override XmlElement this[string localname, string ns] {
  94. [MonoTODO]
  95. get {
  96. throw new NotImplementedException();
  97. }
  98. }
  99. public override string LocalName {
  100. [MonoTODO]
  101. get {
  102. throw new NotImplementedException();
  103. }
  104. }
  105. public override string Name {
  106. [MonoTODO]
  107. get {
  108. return base.Name;
  109. }
  110. }
  111. public override XmlDocument OwnerDocument {
  112. [MonoTODO]
  113. get {
  114. return null;
  115. }
  116. }
  117. #endregion // Public Properties
  118. #region Public Methods
  119. [MonoTODO]
  120. public override XmlNode CloneNode(bool deep)
  121. {
  122. throw new NotImplementedException();
  123. }
  124. #region overloaded CreateElement methods
  125. [MonoTODO ("why this is override?")]
  126. public override XmlElement CreateElement(string prefix,
  127. string localName, string namespaceURI)
  128. {
  129. if ((localName == null) || (localName == String.Empty))
  130. throw new ArgumentException ("The local name for elements or attributes cannot be null" +
  131. "or an empty string.");
  132. string pref = prefix != null ? prefix : String.Empty;
  133. return base.CreateElement (pref, localName, namespaceURI != null ? namespaceURI : String.Empty);
  134. }
  135. #endregion // overloaded CreateElement Methods
  136. // will not be supported
  137. public override XmlEntityReference CreateEntityReference(string name)
  138. {
  139. throw new NotSupportedException();
  140. }
  141. // will not be supported
  142. public override XmlElement GetElementById(string elemId)
  143. {
  144. throw new NotSupportedException();
  145. }
  146. // get the XmlElement associated with the DataRow
  147. [MonoTODO ("Exceptions")]
  148. public XmlElement GetElementFromRow(DataRow r)
  149. {
  150. if (r.XmlRowID == 0) // datarow was not in xmldatadocument
  151. throw new Exception ();
  152. int elementRow = dataRowIDList.IndexOf (r.XmlRowID);
  153. return (XmlElement)GetElementsByTagName (r.Table.TableName) [elementRow];
  154. }
  155. // get the DataRow associated with the XmlElement
  156. [MonoTODO ("Exceptions")]
  157. public DataRow GetRowFromElement(XmlElement e)
  158. {
  159. XmlElement node = e;
  160. if (node == null)
  161. return null;
  162. XPathNavigator nodeNavigator = node.CreateNavigator ();
  163. int c = GetElementsByTagName (node.Name).Count;
  164. if (c == 0)
  165. return null;
  166. // FIXME: I dont know which way it should be but this work on linux.
  167. // could be also GetElementsByTagName (args.OldParent.Name) []
  168. XmlNodeList nodeList = GetElementsByTagName (node.Name);
  169. int i = 0;
  170. bool isSame = false;
  171. while (i < c && !isSame) {
  172. XPathNavigator docNavigator = nodeList [i].CreateNavigator ();
  173. isSame = docNavigator.IsSamePosition (nodeNavigator);
  174. docNavigator = nodeList [i].CreateNavigator ();
  175. if (!isSame)
  176. i++;
  177. }
  178. if (!isSame)
  179. return null;
  180. if (i >= dataRowIDList.Count)
  181. return null;
  182. // now we know rownum
  183. int xmlrowid = (int)dataRowIDList [i];
  184. if (xmlrowid <= 0)
  185. return null;
  186. DataTable dt = DataSet.Tables [node.Name];
  187. DataRow row = null;
  188. if (dt == null)
  189. return null;
  190. foreach (DataRow r in dt.Rows) {
  191. if (xmlrowid == r.XmlRowID) {
  192. row = r;
  193. }
  194. }
  195. return row;
  196. }
  197. #region overload Load methods
  198. public override void Load(Stream inStream) {
  199. Load (new XmlTextReader (inStream));
  200. }
  201. public override void Load(string filename) {
  202. Load (new XmlTextReader (filename));
  203. }
  204. public override void Load(TextReader txtReader) {
  205. Load (new XmlTextReader (txtReader));
  206. }
  207. public override void Load(XmlReader reader) {
  208. // dont listen these events
  209. RemoveXmlDocumentListeners ();
  210. DataTable dt = null;
  211. // For reading xml to XmlDocument
  212. XmlTextReader textReader = new XmlTextReader (
  213. reader.BaseURI);
  214. if (reader.NodeType != XmlNodeType.Element)
  215. reader.MoveToContent ();
  216. // TODO: Findout which exception should be throwen
  217. if (reader.NodeType != XmlNodeType.Element) {
  218. throw new Exception ();
  219. }
  220. if (dataSet.DataSetName != reader.Name) {
  221. throw new Exception ();
  222. }
  223. // read to next element
  224. while (reader.Read () && reader.NodeType != XmlNodeType.Element);
  225. do {
  226. // Find right table from tablecollection
  227. for (int i = 0; i < DataSet.Tables.Count && dt == null; i++) {
  228. if (reader.Name == DataSet.Tables [i].TableName) {
  229. dt = DataSet.Tables [i];
  230. dt.ColumnChanged += new DataColumnChangeEventHandler (OnDataTableColumnChanged);
  231. dt.RowDeleted += new DataRowChangeEventHandler (OnDataTableRowDeleted);
  232. dt.RowChanged += new DataRowChangeEventHandler (OnDataTableRowChanged);
  233. }
  234. }
  235. // TODO: Findout what kind of exception
  236. if (dt == null)
  237. throw new Exception (); // there were no correct table
  238. // Read rows to table
  239. DataRow tempRow = dt.NewRow ();
  240. while ((reader.NodeType != XmlNodeType.EndElement ||
  241. reader.Name != dt.TableName) && reader.Read()) {
  242. switch (reader.NodeType) {
  243. case XmlNodeType.Element:
  244. // Add column to DataRow
  245. LoadRow (reader, ref tempRow);
  246. break;
  247. default:
  248. break;
  249. }
  250. }
  251. // Every row must have unique id.
  252. tempRow.XmlRowID = dataRowID;
  253. dataRowIDList.Add (dataRowID);
  254. dt.Rows.Add (tempRow);
  255. dataRowID++;
  256. } while (reader.Read ());
  257. base.Load (textReader);
  258. AddXmlDocumentListeners ();
  259. }
  260. #endregion // overloaded Load methods
  261. [MonoTODO]
  262. public override void WriteContentTo(XmlWriter xw) {
  263. base.WriteContentTo (xw);
  264. }
  265. [MonoTODO]
  266. public override void WriteTo(XmlWriter w) {
  267. base.WriteTo (w);
  268. }
  269. #endregion // Public Methods
  270. #region Protected Methods
  271. //FIXME: how do you handle this?
  272. //[MonoTODO]
  273. //protected internal override XPathNavigator CreateNavigator(XmlNode node) {
  274. // throw new NotImplementedException();
  275. //}
  276. [MonoTODO]
  277. public new XPathNavigator CreateNavigator() {
  278. throw new NotImplementedException();
  279. }
  280. #endregion // Protected Methods
  281. #region DataSet event handlers
  282. // Invoked when XmlNode is changed colum is changed
  283. [MonoTODO]
  284. private void OnNodeChanged (object sender, XmlNodeChangedEventArgs args)
  285. {
  286. if (args.Node == null)
  287. return;
  288. DataRow row = GetRowFromElement ((XmlElement)args.Node);
  289. if (row == null)
  290. return;
  291. if (row [args.Node.Name].ToString () != args.Node.InnerText)
  292. row [args.Node.Name] = args.Node.InnerText;
  293. }
  294. // Invoked when XmlNode is removed
  295. [MonoTODO]
  296. private void OnNodeRemoved (object sender, XmlNodeChangedEventArgs args)
  297. {
  298. if (args.OldParent == null)
  299. return;
  300. DataRow row = GetRowFromElement ((XmlElement)args.OldParent);
  301. if (row == null) {
  302. return ;
  303. }
  304. // Dont trig event again
  305. row.Table.ColumnChanged -= new DataColumnChangeEventHandler (OnDataTableColumnChanged);
  306. row [args.Node.Name] = null;
  307. // if all columns are "nulled" we can remove the row.
  308. // FIXME: This is rather slow, try to find faster way
  309. bool allNulls = true;
  310. foreach (DataColumn dc in row.Table.Columns) {
  311. if (row [dc.ColumnName] != DBNull.Value) {
  312. allNulls = false;
  313. break;
  314. }
  315. }
  316. if (allNulls) {
  317. dataRowIDList.Remove (row.XmlRowID);
  318. row.Delete ();
  319. }
  320. row.Table.ColumnChanged += new DataColumnChangeEventHandler (OnDataTableColumnChanged);
  321. }
  322. private void OnNodeInserting (object sender, XmlNodeChangedEventArgs args) {
  323. if (DataSet.EnforceConstraints)
  324. throw new InvalidOperationException (Locale.GetText ("Please set DataSet.EnforceConstraints == false " +
  325. "before trying to edit XmlDataDocument using " +
  326. "XML operations."));
  327. }
  328. private void OnNodeInserted (object sender, XmlNodeChangedEventArgs args)
  329. {
  330. // this is table element
  331. if (DataSet.Tables.Contains (args.NewParent.Name)) {
  332. Hashtable ht = null;
  333. if (TempTable.ContainsKey (args.NewParent.Name)) {
  334. // if TempTable contains table name, get it and remove it from hashtable
  335. // so we can later add it :)
  336. ht = TempTable [args.NewParent.Name] as Hashtable;
  337. TempTable.Remove (args.NewParent.Name);
  338. }
  339. else
  340. ht = new Hashtable ();
  341. ht.Add (args.Node.Name, args.Node.InnerText);
  342. TempTable.Add (args.NewParent.Name, ht);
  343. }
  344. else if (DataSet.Tables.Contains (args.Node.Name)) {
  345. // if nodes name is same as some table in the list is is time to
  346. // add row to datatable
  347. DataTable dt = DataSet.Tables [args.Node.Name];
  348. DataRow row = dt.NewRow ();
  349. Hashtable ht = TempTable [args.Node.Name] as Hashtable;
  350. IDictionaryEnumerator enumerator = ht.GetEnumerator ();
  351. while (enumerator.MoveNext ()) {
  352. row [enumerator.Key.ToString ()] = enumerator.Value.ToString ();
  353. }
  354. dt.RowChanged -= new DataRowChangeEventHandler (OnDataTableRowChanged);
  355. DataSet.Tables [args.Node.Name].Rows.Add (row);
  356. dt.RowChanged += new DataRowChangeEventHandler (OnDataTableRowChanged);
  357. }
  358. }
  359. [MonoTODO]
  360. private void OnDataTableColumnChanged(object sender,
  361. DataColumnChangeEventArgs eventArgs)
  362. {
  363. RemoveXmlDocumentListeners ();
  364. // row is not yet in datatable
  365. if (eventArgs.Row.XmlRowID == 0)
  366. return;
  367. // TODO: Here should be some kind of error checking.
  368. GetElementsByTagName (eventArgs.Column.ToString ()) [dataRowIDList.IndexOf (
  369. eventArgs.Row.XmlRowID)].InnerText = eventArgs.ProposedValue.ToString ();
  370. AddXmlDocumentListeners ();
  371. }
  372. [MonoTODO]
  373. private void OnDataTableRowDeleted(object sender,
  374. DataRowChangeEventArgs eventArgs)
  375. {
  376. DataRow deletedRow = null;
  377. deletedRow = eventArgs.Row;
  378. if (eventArgs.Row.XmlRowID == 0)
  379. return;
  380. int rowIndex = dataRowIDList.IndexOf (eventArgs.Row.XmlRowID);
  381. if (rowIndex == -1 || eventArgs.Row.XmlRowID == 0 ||
  382. rowIndex > GetElementsByTagName (deletedRow.Table.TableName).Count - 1)
  383. return;
  384. // Remove element from xmldocument and row indexlist
  385. // FIXME: this is one way to do this, but i hope someday i find out much better way.
  386. XmlNode p = GetElementsByTagName (deletedRow.Table.TableName) [rowIndex].ParentNode;
  387. if (p != null) {
  388. p.RemoveChild (GetElementsByTagName (deletedRow.Table.TableName) [rowIndex]);
  389. dataRowIDList.RemoveAt (rowIndex);
  390. }
  391. }
  392. [MonoTODO]
  393. private void OnDataTableRowChanged(object sender, DataRowChangeEventArgs eventArgs)
  394. {
  395. switch (eventArgs.Action) {
  396. case DataRowAction.Delete:
  397. OnDataTableRowDeleted (sender, eventArgs);
  398. break;
  399. case DataRowAction.Add:
  400. OnDataTableRowAdded (eventArgs);
  401. break;
  402. case DataRowAction.Rollback:
  403. OnDataTableRowRollback (eventArgs);
  404. break;
  405. default:
  406. break;
  407. }
  408. }
  409. // Added
  410. [MonoTODO]
  411. private void OnDataTableRowAdded (DataRowChangeEventArgs args)
  412. {
  413. RemoveXmlDocumentListeners ();
  414. // If XmlRowID is != 0 then it is already added
  415. if (args.Row.XmlRowID != 0)
  416. return;
  417. // Create row element. Row's name same as TableName
  418. DataRow row = args.Row;
  419. row.XmlRowID = dataRowID;
  420. dataRowID++;
  421. XmlElement element = CreateElement (args.Row.Table.TableName);
  422. DocumentElement.AppendChild (element);
  423. XmlElement rowElement = null;
  424. for (int i = 0; i < row.Table.Columns.Count; i++) {
  425. rowElement = CreateElement (row.Table.Columns [i].ToString ());
  426. rowElement.InnerText = (string)row [i];
  427. element.AppendChild (rowElement);
  428. }
  429. AddXmlDocumentListeners ();
  430. }
  431. // Rollback
  432. [MonoTODO]
  433. private void OnDataTableRowRollback (DataRowChangeEventArgs args)
  434. {
  435. RemoveXmlDocumentListeners ();
  436. DataRow row = args.Row;
  437. int rowid = dataRowIDList.IndexOf (row.XmlRowID);
  438. // find right element in xmldocument
  439. if (rowid == 0 || rowid >= GetElementsByTagName (row.Table.TableName).Count)
  440. return;
  441. XmlNode node = GetElementsByTagName (row.Table.TableName) [rowid];
  442. int rowValue = 0;
  443. for (int i = 0; i < node.ChildNodes.Count; i++) {
  444. XmlNode child = node.ChildNodes [i];
  445. if (child.NodeType != XmlNodeType.Whitespace) {
  446. child.InnerText = (string)row [rowValue++];
  447. }
  448. }
  449. AddXmlDocumentListeners ();
  450. }
  451. #endregion // DataSet event handlers
  452. #region Private methods
  453. [MonoTODO]
  454. private void LoadRow (XmlReader reader, ref DataRow row)
  455. {
  456. // dt.Rows.Add (LoadRow (reader, dt.NewRow ()));
  457. // This method returns DataRow filled by values
  458. // from xmldocument
  459. string rowname = reader.Name;
  460. string column = "";
  461. if (reader.NodeType == XmlNodeType.Element)
  462. column = reader.Name;
  463. reader.Read ();
  464. if (reader.NodeType == XmlNodeType.Text) {
  465. string val = reader.Value;
  466. if (row.Table.Columns.Contains (column))
  467. row [column] = val;
  468. }
  469. }
  470. private void RemoveXmlDocumentListeners ()
  471. {
  472. this.NodeInserting -= new XmlNodeChangedEventHandler (OnNodeInserting);
  473. this.NodeInserted -= new XmlNodeChangedEventHandler (OnNodeInserted);
  474. }
  475. private void AddXmlDocumentListeners ()
  476. {
  477. this.NodeInserting += new XmlNodeChangedEventHandler (OnNodeInserting);
  478. this.NodeInserted += new XmlNodeChangedEventHandler (OnNodeInserted);
  479. }
  480. #endregion // Private methods
  481. }
  482. }