/Source/HydroDesktop.WebServices/HISCentralClient.cs

# · C# · 706 lines · 473 code · 110 blank · 123 comment · 74 complexity · 4a1c037dcf8c6f808426aaa6a6029001 MD5 · raw file

  1. #region Namespace
  2. using System;
  3. using System.Collections.Generic;
  4. using System.Text;
  5. using System.Reflection;
  6. using System.Web.Services.Description;
  7. using System.Data;
  8. using HydroDesktop.Interfaces.ObjectModel;
  9. using System.Diagnostics;
  10. using System.Globalization;
  11. #endregion
  12. namespace HydroDesktop.WebServices
  13. {
  14. /// <summary>
  15. /// Calls to the HIS central web services
  16. /// </summary>
  17. public class HISCentralClient : WebServiceClientBase
  18. {
  19. Stopwatch watch1 = new Stopwatch();
  20. Stopwatch watch2 = new Stopwatch();
  21. /// <summary>
  22. /// Creates a new instance of HIS central web service client
  23. /// </summary>
  24. /// <param name="_assemblyxURL">The url of the server with HIS Central web services</param>
  25. public HISCentralClient(string asmxURL) :
  26. base(asmxURL)
  27. {
  28. if (_webService != null)
  29. {
  30. if (_webService.GetType().GetMethod("GetSearchableConcepts") == null)
  31. {
  32. throw new System.Net.WebException("The Service doesn't have the required method GetSearchableConcepts");
  33. }
  34. }
  35. }
  36. #region Private Variables
  37. //the list of unique site / variable combinations
  38. private List<string> _uniqueCodeList = new List<string>();
  39. #endregion
  40. /// <summary>
  41. /// Returns a list of mapped variables
  42. /// </summary>
  43. /// <returns></returns>
  44. public DataTable GetMappedVariables()
  45. {
  46. if (_assembly == null) return null;
  47. PropertyInfo[] properties = GetPropertyInfo("MappedVariable");
  48. DataTable table = CreateResultTable(properties, "MappedVariable");
  49. object result = null;
  50. object[] param = new object[2];
  51. param[0] = new string[] { "" };
  52. param[1] = new string[] { "" };
  53. result = CallWebMethod("GetMappedVariables", param);
  54. WriteResults(result, properties, table);
  55. return table;
  56. }
  57. /// <summary>
  58. /// Returns a string array of searchable concept names
  59. /// </summary>
  60. /// <returns></returns>
  61. public string[] GetSearchableConcepts()
  62. {
  63. if (_assembly == null) return null;
  64. object result = null;
  65. result = CallWebMethod("GetSearchableConcepts", null);
  66. Array resultArr = result as Array;
  67. if (resultArr != null)
  68. {
  69. string[] result2 = new string[resultArr.Length];
  70. for (int i = 0; i < resultArr.Length; i++)
  71. {
  72. result2[i] = (string)resultArr.GetValue(i);
  73. }
  74. return result2;
  75. }
  76. return null;
  77. }
  78. /// <summary>
  79. /// Gets the ontology tree, in the form of a DataTable
  80. /// </summary>
  81. /// <returns>A data table with following three columns:
  82. /// ID
  83. /// ParentID
  84. /// Keyword
  85. /// </returns>
  86. public DataTable GetOntologyTreeTable()
  87. {
  88. if (_assembly == null) return null;
  89. object result = null;
  90. DataTable tbl = new DataTable();
  91. tbl.Columns.Add(new DataColumn("ID", typeof(int)));
  92. tbl.Columns.Add(new DataColumn("ParentID", typeof(int)));
  93. tbl.Columns.Add(new DataColumn("Keyword", typeof(string)));
  94. PropertyInfo[] properties = GetPropertyInfo("OntologyNode");
  95. //the top-level keyword is always 'hydrosphere'
  96. object[] param = new object[1] { "hydrosphere" };
  97. result = CallWebMethod("getOntologyTree", param);
  98. ReadTree2(result, properties, 1, tbl);
  99. return tbl;
  100. }
  101. /// <summary>
  102. /// Returns all valid pairs of conceptCode, conceptKeyword
  103. /// from the ontology concept tree
  104. /// </summary>
  105. /// <returns></returns>
  106. public Dictionary<string, string> GetOntologyTree()
  107. {
  108. if (_assembly == null) return null;
  109. object result = null;
  110. PropertyInfo[] properties = GetPropertyInfo("OntologyNode");
  111. object[] param = new object[1] { "hydrosphere" };
  112. // Finally, Invoke the web service method
  113. result = CallWebMethod("getOntologyTree", param);
  114. Dictionary<string, string> keywordLookup = new Dictionary<string, string>();
  115. ReadTree(result, properties, keywordLookup);
  116. return keywordLookup;
  117. }
  118. private void ReadTree(object node, PropertyInfo[] properties, Dictionary<string, string> resultList)
  119. {
  120. if (resultList == null) return;
  121. string keyword = properties[0].GetValue(node, null).ToString();
  122. keyword = keyword.Replace("\\c", ",");
  123. string code = properties[1].GetValue(node, null).ToString();
  124. Array childNodes = properties[2].GetValue(node, null) as Array;
  125. //if there are no child nodes, we'll add the keyword
  126. if (resultList.ContainsKey(keyword) == false && keyword.IndexOf("other") < 0)
  127. {
  128. if (childNodes == null)
  129. {
  130. resultList.Add(keyword.ToString(), code.ToString());
  131. }
  132. else if (childNodes.Length == 0)
  133. {
  134. resultList.Add(keyword.ToString(), code.ToString());
  135. }
  136. }
  137. if (childNodes == null) return;
  138. if (childNodes.Length == 0) return;
  139. foreach (object obj in childNodes)
  140. {
  141. ReadTree(obj, properties, resultList);
  142. }
  143. }
  144. private void ReadTree2(object node, PropertyInfo[] properties, int parentID, DataTable resultTable)
  145. {
  146. string keyword = properties[0].GetValue(node, null).ToString();
  147. keyword = keyword.Replace("\\c", ",");
  148. int code = Convert.ToInt32(properties[1].GetValue(node, null));
  149. //add the keyword to table
  150. DataRow newRow = resultTable.NewRow();
  151. newRow["ID"] = code;
  152. newRow["ParentID"] = parentID;
  153. newRow["Keyword"] = keyword;
  154. resultTable.Rows.Add(newRow);
  155. Array childNodes = properties[2].GetValue(node, null) as Array;
  156. if (childNodes == null) return;
  157. if (childNodes.Length == 0) return; //exit if there are no child nodes
  158. //otherwise, add the child keywords
  159. foreach (object obj in childNodes)
  160. {
  161. ReadTree2(obj, properties, code, resultTable);
  162. }
  163. }
  164. /// <summary>
  165. /// Returns the series catalog for the specified region (latitude | longitude bounding box)
  166. /// </summary>
  167. /// <param name="latLongBox">the latitude/longitude bounding box</param>
  168. /// <param name="keywords">the array of search keywords (not included in search criteria if null)</param>
  169. /// <param name="beginDate">the begin date</param>
  170. /// <param name="endDate">the end date</param>
  171. /// <param name="networkIDs">the ServiceIDs. These are obtained by calling GetServicesInBox() function</param>
  172. /// <returns>a list o all matching data series entries</returns>
  173. public IList<SeriesMetadata> GetSeriesCatalogForBox(Box latLongBox, string[] keywords, DateTime beginDate, DateTime endDate, int[] networkIDs)
  174. {
  175. if (_assembly == null) return null;
  176. _uniqueCodeList.Clear();
  177. //if no keywords are specified - set them to an array with one empty string
  178. //if no keywords are specified, set the keywords[] parameter to an array with one empty string
  179. if (keywords == null)
  180. {
  181. keywords = new string[]{ String.Empty };
  182. }
  183. if (keywords.Length == 0)
  184. {
  185. keywords = new string[]{ String.Empty };
  186. }
  187. //if no service IDs are specified - set them to all available services within region
  188. //if (networkIDs == null)
  189. //{
  190. // IList<DataServiceInfo> networks = GetServicesInBox(latLongBox.xmin, latLongBox.ymin, latLongBox.xmax, latLongBox.ymax);
  191. // networkIDs = new int[networks.Count];
  192. // for (int i = 0; i < networks.Count; i++)
  193. // {
  194. // networkIDs[i] = networks[i].HISCentralID;
  195. // }
  196. //}
  197. // TODO: why not use MinValue and MaxValue
  198. //if no begin date specified - set it to a very early date
  199. if (beginDate == null) beginDate = new DateTime(1900, 1, 1);
  200. //if no end date specified - set it to current date
  201. if (endDate == null) endDate = DateTime.Now.Date;
  202. PropertyInfo[] resultProperties = GetPropertyInfo("SeriesRecord");
  203. DataTable table = CreateResultTable(resultProperties, "SeriesRecord");
  204. IList<SeriesMetadata> lst = new List<SeriesMetadata>();
  205. object boxObj = InitializeBox(latLongBox);
  206. object[] param = new object[5];
  207. MethodInfo mi = _webService.GetType().GetMethod("GetSeriesCatalogForBox");
  208. for (int i = 0; i < keywords.Length; i++)
  209. {
  210. try
  211. {
  212. param[0] = boxObj;
  213. param[1] = keywords[i];
  214. param[2] = networkIDs;
  215. param[3] = DateToString(beginDate);
  216. param[4] = DateToString(endDate);
  217. // Invoke the web service method
  218. watch1.Start();
  219. object result = mi.Invoke(_webService, param);
  220. watch1.Stop();
  221. // Process the result
  222. watch2.Start();
  223. WriteDataSeriesResults(result, resultProperties, lst);
  224. watch2.Stop();
  225. }
  226. finally
  227. {
  228. //System.Windows.Forms.MessageBox.Show(ex.Message + " " + ex.InnerException.Message);
  229. }
  230. }
  231. return lst;
  232. }
  233. /// <summary>
  234. /// Returns the series catalog for the specified region (latitude | longitude bounding box)
  235. /// </summary>
  236. /// <param name="latLongBox">the latitude / </param>
  237. /// <param name="keyword"></param>
  238. /// <param name="beginDate"></param>
  239. /// <param name="endDate"></param>
  240. /// <returns>a list of all matching data series entries</returns>
  241. public IList<SeriesMetadata> GetSeriesCatalogForBox(Box latLongBox, string[] keywords, DateTime beginDate, DateTime endDate)
  242. {
  243. if (_assembly == null) return null;
  244. _uniqueCodeList.Clear();
  245. PropertyInfo[] resultProperties = GetPropertyInfo("SeriesRecord");
  246. List<SeriesMetadata> lst = new List<SeriesMetadata>();
  247. object[] param = new object[8];
  248. object result = null;
  249. MethodInfo mi = _webService.GetType().GetMethod("GetSeriesCatalogForBox2");
  250. //if no keywords are specified - set them to an array with one empty string
  251. if (keywords == null)
  252. {
  253. keywords = new string[] { string.Empty };
  254. }
  255. //if no begin date specified - set it to a very early date
  256. if (beginDate == null) beginDate = new DateTime(1900, 1, 1);
  257. //if no end date specified - set it to current date
  258. if (endDate == null) endDate = DateTime.Now.Date;
  259. //if no keywords are specified, set the keywords[] parameter to an array with one empty string
  260. if (keywords == null)
  261. {
  262. keywords = new string[]{ String.Empty };
  263. }
  264. if (keywords.Length == 0)
  265. {
  266. keywords = new string[]{ String.Empty };
  267. }
  268. for (int i = 0; i < keywords.Length; i++)
  269. {
  270. try
  271. {
  272. param[0] = latLongBox.XMin;
  273. param[1] = latLongBox.XMax;
  274. param[2] = latLongBox.YMin;
  275. param[3] = latLongBox.YMax;
  276. param[4] = keywords[i];
  277. param[5] = "";
  278. param[6] = DateToString(beginDate);
  279. param[7] = DateToString(endDate);
  280. // Invoke the web service method
  281. watch1.Start();
  282. result = mi.Invoke(_webService, param);
  283. watch1.Stop();
  284. // Process the result
  285. watch2.Start();
  286. WriteDataSeriesResults(result, resultProperties, lst);
  287. watch2.Stop();
  288. }
  289. catch (Exception ex)
  290. {
  291. throw new Exception("Error accessing the HIS Central Web service", ex);
  292. }
  293. }
  294. return lst;
  295. }
  296. /// <summary>
  297. /// Retrieves information about CUAHSI WaterOneFlow web services available in the specified region
  298. /// (latitude / longitude bounding box).
  299. /// </summary>
  300. /// <param name="latLongBox"></param>
  301. /// <returns></returns>
  302. public IList<DataServiceInfo> GetServicesInBox(double xmin, double ymin, double xmax, double ymax)
  303. {
  304. object[] param = new object[4];
  305. param[0] = xmin;
  306. param[1] = ymin;
  307. param[2] = xmax;
  308. param[3] = ymax;
  309. object result = null;
  310. result = CallWebMethod("GetServicesInBox2", param);
  311. // Process the result
  312. PropertyInfo[] properties = GetPropertyInfo("ServiceInfo");
  313. DataTable table = CreateResultTable(properties, "ServiceInfo");
  314. WriteResults(result, properties, table);
  315. return TableToWebServiceInfo(table);
  316. }
  317. /// <summary>
  318. /// Retrieves all sites within a geographic region that belong to the specified network
  319. /// (web service) and measure the specified variable
  320. /// </summary>
  321. /// <param name="latLongBox"></param>
  322. /// <param name="keyword"></param>
  323. /// <param name="networks"></param>
  324. /// <returns></returns>
  325. public DataTable GetSitesInBox(Box latLongBox, string keyword, int[] networks)
  326. {
  327. PropertyInfo[] properties = GetPropertyInfo("Site");
  328. DataTable table = CreateResultTable(properties, "Site");
  329. object boxObj = InitializeBox(latLongBox);
  330. object[] param = new object[3];
  331. param[0] = boxObj;
  332. param[1] = keyword;
  333. param[2] = networks;
  334. object result = null;
  335. result = CallWebMethod("GetSitesInBox", param);
  336. //MethodInfo mi = _webService.GetType().GetMethod("GetSitesInBox");
  337. //result = mi.Invoke(_webService, param);
  338. // Process the result
  339. WriteResults(result, properties, table);
  340. return table;
  341. }
  342. /// <summary>
  343. /// Gets all sites within the specified latitude / longitude bounding box.
  344. /// </summary>
  345. /// <param name="latLongBox"></param>
  346. /// <returns></returns>
  347. public DataTable GetAllSitesInBox(Box latLongBox)
  348. {
  349. //DataTable networkTable = GetServicesInBox(latLongBox);
  350. //int[] networkIDs = new int[networkTable.Rows.Count];
  351. PropertyInfo[] properties = GetPropertyInfo("Site");
  352. DataTable table = CreateResultTable(properties, "Site");
  353. object boxObj = InitializeBox(latLongBox);
  354. object[] param = new object[3];
  355. param[0] = boxObj;
  356. param[1] = "";
  357. param[2] = null;
  358. object result = null;
  359. result = CallWebMethod("GetSitesInBox", param);
  360. //MethodInfo mi = _webService.GetType().GetMethod("GetSitesInBox");
  361. //result = mi.Invoke(_webService, param);
  362. // Process the result
  363. WriteResults(result, properties, table);
  364. return table;
  365. }
  366. /// <summary>
  367. /// Returns all sites within the region, regardless of the web service or variable
  368. /// </summary>
  369. /// <param name="latLongBox"></param>
  370. /// <returns></returns>
  371. public DataTable GetAllSitesInBox2(Box latLongBox)
  372. {
  373. //first we get all available concepts
  374. string[] keywords = GetSearchableConcepts();
  375. DataTable resultTable = null;
  376. foreach (string keyword in keywords)
  377. {
  378. DataTable tmpTable = GetAllSitesInBox(latLongBox);
  379. if (resultTable == null)
  380. {
  381. resultTable = tmpTable.Clone();
  382. }
  383. else
  384. {
  385. foreach (DataRow row in tmpTable.Rows)
  386. {
  387. DataRow newRow = resultTable.NewRow();
  388. newRow.ItemArray = row.ItemArray;
  389. resultTable.Rows.Add(newRow);
  390. }
  391. }
  392. }
  393. return resultTable;
  394. }
  395. /// <summary>
  396. /// Retrieves information about CUAHSI WaterOneFlow web services available at the HIS Central
  397. /// </summary>
  398. /// <returns></returns>
  399. public IList<DataServiceInfo> GetWaterOneFlowServiceInfo()
  400. {
  401. // check the dynamic assembly
  402. if (_assembly == null) return null;
  403. PropertyInfo[] properties = GetPropertyInfo("ServiceInfo");
  404. DataTable table = CreateResultTable(properties, "ServiceInfo");
  405. object result = null;
  406. result = CallWebMethod("GetWaterOneFlowServiceInfo", null);
  407. // Process the result
  408. WriteResults(result, properties, table);
  409. return TableToWebServiceInfo(table);
  410. }
  411. #region Private Methods
  412. private string DateToString(DateTime dat)
  413. {
  414. return dat.ToString("MM/dd/yyyy");
  415. }
  416. // writes the results to a list of data series objects (which is a theme)
  417. private void WriteDataSeriesResults(Object resultObj, PropertyInfo[] properties, IList<SeriesMetadata> lst)
  418. {
  419. if (resultObj == null) return;
  420. //we need to get the data service information as well.
  421. IList<DataServiceInfo> servicesList = GetServicesInBox(-179, -89, 179, 89);
  422. Dictionary<string, DataServiceInfo> servicesLookup = new Dictionary<string, DataServiceInfo>();
  423. foreach (DataServiceInfo servInfo in servicesList)
  424. {
  425. string url = servInfo.EndpointURL.ToLower().Trim();
  426. if (!servicesLookup.ContainsKey(url))
  427. {
  428. servicesLookup.Add(url, servInfo);
  429. }
  430. }
  431. //to prevent duplicate search results - each site code / variable code combination
  432. //is added to the 'uniqueCode' list
  433. Array arr = resultObj as Array;
  434. if (arr != null)
  435. {
  436. foreach (object obj in arr)
  437. {
  438. SeriesMetadata sm = new SeriesMetadata();
  439. sm.Site = new Site();
  440. sm.Variable = new Variable();
  441. sm.DataService = new DataServiceInfo();
  442. foreach (PropertyInfo pi in properties)
  443. {
  444. object val = pi.GetValue(obj, null);
  445. if (val != null)
  446. {
  447. string propertyName = pi.Name.ToLower();
  448. switch(propertyName)
  449. {
  450. case "location":
  451. sm.Site.Code = val.ToString();
  452. break;
  453. case "varcode":
  454. sm.Variable.Code = val.ToString();
  455. break;
  456. case "servcode":
  457. string servCode = val.ToString();
  458. sm.Site.NetworkPrefix = servCode;
  459. sm.Variable.VocabularyPrefix = servCode;
  460. sm.DataService.ServiceCode = servCode;
  461. break;
  462. case "latitude":
  463. sm.Site.Latitude = Convert.ToDouble(val);
  464. break;
  465. case "longitude":
  466. sm.Site.Longitude = Convert.ToDouble(val);
  467. break;
  468. case "sitename":
  469. sm.Site.Name = Convert.ToString(val);
  470. break;
  471. case "varname":
  472. sm.Variable.Name = Convert.ToString(val);
  473. break;
  474. case "servurl":
  475. string url = Convert.ToString(val).ToLower().Trim();
  476. if (url.EndsWith("?wsdl"))
  477. {
  478. url = url.Replace("?wsdl", "");
  479. }
  480. sm.DataService.EndpointURL = url;
  481. break;
  482. case "valuecount":
  483. sm.ValueCount = Convert.ToInt32(val);
  484. break;
  485. case "begindate":
  486. sm.BeginDateTime = Convert.ToDateTime(val, CultureInfo.GetCultureInfo("en-US"));
  487. break;
  488. case "enddate":
  489. sm.EndDateTime = Convert.ToDateTime(val, CultureInfo.GetCultureInfo("en-US"));
  490. break;
  491. case "datatype":
  492. sm.Variable.DataType = val.ToString();
  493. break;
  494. case "valuetype":
  495. sm.Variable.ValueType = val.ToString();
  496. break;
  497. case "samplemedium":
  498. sm.Variable.SampleMedium = val.ToString();
  499. break;
  500. case "timeunits":
  501. sm.Variable.TimeUnit = new Unit(val.ToString(), "time", val.ToString());
  502. break;
  503. case "gencategory":
  504. sm.Variable.GeneralCategory = val.ToString();
  505. break;
  506. case "timesupport":
  507. sm.Variable.TimeSupport = Convert.ToDouble(val);
  508. break;
  509. }
  510. }
  511. }
  512. if (String.IsNullOrEmpty(sm.Variable.Code))
  513. {
  514. continue;
  515. }
  516. string compoundCode = sm.Site.Code + "|" + sm.Variable.Code;
  517. if (_uniqueCodeList.Contains(compoundCode) == false)
  518. {
  519. if (servicesLookup.ContainsKey(sm.DataService.EndpointURL))
  520. {
  521. sm.Source.Organization = ((DataServiceInfo)servicesLookup[sm.DataService.EndpointURL]).ServiceName;
  522. sm.Source.Description = ((DataServiceInfo)servicesLookup[sm.DataService.EndpointURL]).ServiceTitle;
  523. }
  524. lst.Add(sm);
  525. _uniqueCodeList.Add(compoundCode);
  526. }
  527. }
  528. }
  529. }
  530. //initializes the 'box' object that is used by the web services
  531. private object InitializeBox(Box box)
  532. {
  533. // Create the 'Box' object using the dynamic assembly and set its properties
  534. object boxObj = _assembly.CreateInstance("Box");
  535. Type boxType = boxObj.GetType();
  536. PropertyInfo[] boxProperties = boxType.GetProperties();
  537. boxProperties[0].SetValue(boxObj, box.XMin, null);
  538. boxProperties[1].SetValue(boxObj, box.XMax, null);
  539. boxProperties[2].SetValue(boxObj, box.YMin, null);
  540. boxProperties[3].SetValue(boxObj, box.YMax, null);
  541. return boxObj;
  542. }
  543. //converts a data table to a list of WaterOneFlowServiceInfo objects
  544. private IList<DataServiceInfo> TableToWebServiceInfo(DataTable table)
  545. {
  546. List<DataServiceInfo> resultList = new List<DataServiceInfo>();
  547. foreach (DataRow row in table.Rows)
  548. {
  549. string endpointURL = row["servURL"].ToString();
  550. endpointURL = endpointURL.ToLower();
  551. if (endpointURL.EndsWith("?wsdl"))
  552. endpointURL = endpointURL.Replace("?wsdl", "");
  553. string title = row["Title"].ToString();
  554. DataServiceInfo servInfo = new DataServiceInfo(endpointURL, title);
  555. servInfo.DescriptionURL = row["ServiceDescriptionURL"].ToString();
  556. servInfo.ContactName = row["organization"].ToString();
  557. servInfo.ContactEmail = row["orgwebsite"].ToString();
  558. servInfo.Citation = row["citation"].ToString();
  559. servInfo.Abstract = (string)row["aabstract"];
  560. servInfo.ValueCount = Convert.ToInt32(row["valuecount"]);
  561. servInfo.SiteCount = Convert.ToInt32(row["sitecount"]);
  562. servInfo.HISCentralID = Convert.ToInt32(row["ServiceID"]);
  563. servInfo.EastLongitude = Convert.ToDouble(row["minx"]);
  564. servInfo.SouthLatitude = Convert.ToDouble(row["miny"]);
  565. servInfo.WestLongitude = Convert.ToDouble(row["maxx"]);
  566. servInfo.NorthLatitude = Convert.ToDouble(row["maxy"]);
  567. servInfo.ServiceCode = Convert.ToString(row["NetworkName"]);
  568. servInfo.ServiceName = Convert.ToString(row["organization"]);
  569. resultList.Add(servInfo);
  570. }
  571. return resultList;
  572. }
  573. //creates the empty data table for retrieving a list of web services
  574. private DataTable CreateWebServiceInfoTable()
  575. {
  576. DataTable tbl = new DataTable();
  577. tbl.Columns.Add("ServiceID", typeof(int));
  578. tbl.Columns.Add("ServiceTitle", typeof(string));
  579. tbl.Columns.Add("ServiceName", typeof(string));
  580. tbl.Columns.Add("ServiceCode", typeof(string));
  581. tbl.Columns.Add("ServiceVersion", typeof(string));
  582. tbl.Columns.Add("ServiceProtocol", typeof(string));
  583. tbl.Columns.Add("ServiceEndpointURL", typeof(string));
  584. tbl.Columns.Add("ServiceDescriptionURL", typeof(string));
  585. tbl.Columns.Add("NorthLatitude", typeof(string));
  586. tbl.Columns.Add("SouthLatitude", typeof(string));
  587. tbl.Columns.Add("EastLongitude", typeof(string));
  588. tbl.Columns.Add("WestLongitude", typeof(string));
  589. tbl.Columns.Add("Abstract", typeof(string));
  590. tbl.Columns.Add("ContactName", typeof(string));
  591. tbl.Columns.Add("Citation", typeof(string));
  592. return tbl;
  593. }
  594. #endregion
  595. }
  596. }