MetaDataExporter.cs 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656
  1. //
  2. // System.Runtime.Remoting.MetadataServices.MetaDataExporter
  3. //
  4. // Authors:
  5. // Lluis Sanchez Gual ([email protected])
  6. //
  7. // (C) 2003 Novell, Inc
  8. //
  9. using System.Collections;
  10. using System.IO;
  11. using System.Text;
  12. using System.Xml;
  13. using System.Reflection;
  14. using System.Net;
  15. using System.Runtime.Remoting;
  16. using System.Runtime.Remoting.Metadata;
  17. using System.Runtime.Serialization;
  18. namespace System.Runtime.Remoting.MetadataServices
  19. {
  20. internal class MetaDataExporter
  21. {
  22. public void ExportTypes (ServiceType[] servicetypes, SdlType sdltype, XmlTextWriter tw)
  23. {
  24. if (sdltype == SdlType.Sdl) // Obsolete, we don't support this
  25. throw new NotSupportedException ();
  26. if (servicetypes.Length == 0) return;
  27. Type maint = servicetypes [0].ObjectType;
  28. Hashtable dataTypes = new Hashtable ();
  29. ArrayList services = new ArrayList ();
  30. FindTypes (servicetypes, dataTypes, services);
  31. if (services.Count > 0)
  32. maint = ((ServiceType) services[0]).ObjectType;
  33. string serviceNs = GetXmlNamespace (maint, null);
  34. tw.Formatting = Formatting.Indented;
  35. tw.WriteStartElement ("definitions", MetaData.WsdlNamespace);
  36. tw.WriteAttributeString ("name", maint.Name);
  37. tw.WriteAttributeString ("targetNamespace", serviceNs);
  38. tw.WriteAttributeString ("xmlns", MetaData.XmlnsNamespace, MetaData.WsdlNamespace);
  39. tw.WriteAttributeString ("xmlns", "tns", MetaData.XmlnsNamespace, serviceNs);
  40. tw.WriteAttributeString ("xmlns", "xsd", MetaData.XmlnsNamespace, MetaData.SchemaNamespace);
  41. tw.WriteAttributeString ("xmlns", "xsi", MetaData.XmlnsNamespace, MetaData.SchemaInstanceNamespace);
  42. tw.WriteAttributeString ("xmlns", "suds", MetaData.XmlnsNamespace, MetaData.SudsNamespace);
  43. tw.WriteAttributeString ("xmlns", "wsdl", MetaData.XmlnsNamespace, MetaData.WsdlNamespace);
  44. tw.WriteAttributeString ("xmlns", "soapenc", MetaData.XmlnsNamespace, MetaData.SoapEncodingNamespace);
  45. tw.WriteAttributeString ("xmlns", "soap", MetaData.XmlnsNamespace, MetaData.SoapNamespace);
  46. int nums = 0;
  47. foreach (DictionaryEntry entry in dataTypes)
  48. {
  49. string ns = (string) entry.Key;
  50. if (tw.LookupPrefix (ns) != null) continue;
  51. tw.WriteAttributeString ("xmlns", "ns"+nums, MetaData.XmlnsNamespace, ns);
  52. nums++;
  53. }
  54. // Schema
  55. if (dataTypes.Count > 0)
  56. {
  57. tw.WriteStartElement ("types", MetaData.WsdlNamespace);
  58. foreach (DictionaryEntry entry in dataTypes)
  59. {
  60. SchemaInfo sinfo = (SchemaInfo) entry.Value;
  61. if (sinfo == null || sinfo.Types.Count == 0) continue;
  62. tw.WriteStartElement ("s", "schema", MetaData.SchemaNamespace);
  63. tw.WriteAttributeString ("targetNamespace", (string) entry.Key);
  64. tw.WriteAttributeString ("elementFormDefault", "unqualified");
  65. tw.WriteAttributeString ("attributeFormDefault", "unqualified");
  66. foreach (string ns in sinfo.Imports)
  67. {
  68. if (ns == (string) entry.Key) continue;
  69. tw.WriteStartElement ("import", MetaData.SchemaNamespace);
  70. tw.WriteAttributeString ("namespace", ns);
  71. tw.WriteEndElement ();
  72. }
  73. foreach (Type type in sinfo.Types)
  74. WriteDataTypeSchema (tw, type);
  75. tw.WriteEndElement ();
  76. }
  77. tw.WriteEndElement ();
  78. }
  79. // Bindings
  80. /* foreach (ServiceType st in servicetypes)
  81. WriteServiceBinding (tw, st);
  82. */
  83. foreach (ServiceType st in services)
  84. WriteServiceBinding (tw, st, dataTypes);
  85. // Service element
  86. tw.WriteStartElement ("service", MetaData.WsdlNamespace);
  87. if (services.Count > 0)
  88. tw.WriteAttributeString ("name", GetServiceName (maint));
  89. else
  90. tw.WriteAttributeString ("name", "Service");
  91. foreach (ServiceType st in services)
  92. {
  93. WriteServiceType (tw, st);
  94. }
  95. tw.WriteEndElement ();
  96. // Closing
  97. tw.WriteEndElement ();
  98. tw.Flush ();
  99. }
  100. void WriteServiceType (XmlTextWriter tw, ServiceType st)
  101. {
  102. tw.WriteStartElement ("port", MetaData.WsdlNamespace);
  103. tw.WriteAttributeString ("name", GetPortName (st.ObjectType));
  104. tw.WriteAttributeString ("binding", "tns:" + GetBindingName (st.ObjectType));
  105. if (st.Url != null)
  106. {
  107. tw.WriteStartElement ("soap","address", MetaData.SoapNamespace);
  108. tw.WriteAttributeString ("location", st.Url);
  109. tw.WriteEndElement ();
  110. }
  111. tw.WriteEndElement ();
  112. }
  113. void WriteServiceBinding (XmlTextWriter tw, ServiceType st, Hashtable dataTypes)
  114. {
  115. Type type = st.ObjectType;
  116. string typeName = type.Name;
  117. MethodInfo[] mets = type.GetMethods (BindingFlags.Public | BindingFlags.Instance | BindingFlags.DeclaredOnly);
  118. bool isService = IsService (type);
  119. // Messages
  120. if (isService)
  121. {
  122. foreach (MethodInfo met in mets)
  123. {
  124. if (met.DeclaringType.Assembly == typeof(object).Assembly) continue;
  125. ParameterInfo[] pars = met.GetParameters ();
  126. tw.WriteStartElement ("message", MetaData.WsdlNamespace);
  127. tw.WriteAttributeString ("name", typeName + "." + met.Name + "Input");
  128. foreach (ParameterInfo par in pars)
  129. {
  130. if (!par.ParameterType.IsByRef)
  131. WritePart (tw, par.Name, par.ParameterType, type);
  132. }
  133. tw.WriteEndElement (); // message
  134. tw.WriteStartElement ("message", MetaData.WsdlNamespace);
  135. tw.WriteAttributeString ("name", typeName + "." + met.Name + "Output");
  136. if (met.ReturnType != typeof(void))
  137. WritePart (tw, "return", met.ReturnType, type);
  138. foreach (ParameterInfo par in pars)
  139. {
  140. if (par.ParameterType.IsByRef || par.IsOut)
  141. WritePart (tw, par.Name, par.ParameterType, type);
  142. }
  143. tw.WriteEndElement (); // message
  144. }
  145. }
  146. // Port type
  147. tw.WriteStartElement ("portType", MetaData.WsdlNamespace);
  148. tw.WriteAttributeString ("name", typeName + "PortType");
  149. if (isService)
  150. {
  151. foreach (MethodInfo met in mets)
  152. {
  153. if (met.DeclaringType.Assembly == typeof(object).Assembly) continue;
  154. tw.WriteStartElement ("operation", MetaData.WsdlNamespace);
  155. tw.WriteAttributeString ("name", met.Name);
  156. StringBuilder sb = new StringBuilder ();
  157. ParameterInfo[] pars = met.GetParameters ();
  158. foreach (ParameterInfo par in pars)
  159. {
  160. if (sb.Length != 0) sb.Append (" ");
  161. sb.Append (par.Name);
  162. }
  163. tw.WriteAttributeString ("parameterOrder", sb.ToString ());
  164. tw.WriteStartElement ("input", MetaData.WsdlNamespace);
  165. tw.WriteAttributeString ("name", met.Name + "Request");
  166. tw.WriteAttributeString ("message", "tns:" + typeName + "." + met.Name + "Input");
  167. tw.WriteEndElement ();
  168. tw.WriteStartElement ("output", MetaData.WsdlNamespace);
  169. tw.WriteAttributeString ("name", met.Name + "Response");
  170. tw.WriteAttributeString ("message", "tns:" + typeName + "." + met.Name + "Output");
  171. tw.WriteEndElement ();
  172. tw.WriteEndElement (); // operation
  173. }
  174. }
  175. tw.WriteEndElement (); // portType
  176. // Binding
  177. tw.WriteStartElement ("binding", MetaData.WsdlNamespace);
  178. tw.WriteAttributeString ("name", typeName + "Binding");
  179. tw.WriteAttributeString ("type", "tns:" + typeName + "PortType");
  180. tw.WriteStartElement ("soap", "binding", MetaData.SoapNamespace);
  181. tw.WriteAttributeString ("style", "rpc");
  182. tw.WriteAttributeString ("transport", "http://schemas.xmlsoap.org/soap/http");
  183. tw.WriteEndElement ();
  184. WriteTypeSuds (tw, type);
  185. SchemaInfo sinfo = (SchemaInfo) dataTypes [GetXmlNamespace (type,null)];
  186. if (sinfo != null && !sinfo.SudsGenerated)
  187. {
  188. foreach (Type dt in sinfo.Types)
  189. WriteTypeSuds (tw, dt);
  190. sinfo.SudsGenerated = true;
  191. }
  192. if (isService)
  193. {
  194. foreach (MethodInfo met in mets)
  195. {
  196. if (met.DeclaringType.Assembly == typeof(object).Assembly) continue;
  197. tw.WriteStartElement ("operation", MetaData.WsdlNamespace);
  198. tw.WriteAttributeString ("name", met.Name);
  199. tw.WriteStartElement ("soap", "operation", MetaData.SoapNamespace);
  200. tw.WriteAttributeString ("soapAction", GetSoapAction (met));
  201. tw.WriteEndElement ();
  202. tw.WriteStartElement ("suds", "method", MetaData.SudsNamespace);
  203. tw.WriteAttributeString ("attributes", "public");
  204. tw.WriteEndElement ();
  205. tw.WriteStartElement ("input", MetaData.WsdlNamespace);
  206. tw.WriteAttributeString ("name", met.Name + "Request");
  207. WriteMessageBindingBody (tw, type);
  208. tw.WriteEndElement ();
  209. tw.WriteStartElement ("output", MetaData.WsdlNamespace);
  210. tw.WriteAttributeString ("name", met.Name + "Response");
  211. WriteMessageBindingBody (tw, type);
  212. tw.WriteEndElement ();
  213. tw.WriteEndElement (); // operation
  214. }
  215. }
  216. tw.WriteEndElement (); // binding
  217. }
  218. void WriteTypeSuds (XmlTextWriter tw, Type type)
  219. {
  220. if (type.IsArray || type.IsEnum)
  221. {
  222. return;
  223. }
  224. else if (type.IsInterface)
  225. {
  226. tw.WriteStartElement ("suds", "interface", MetaData.SudsNamespace);
  227. tw.WriteAttributeString ("type", GetQualifiedXmlType (tw, type, null));
  228. foreach (Type interf in type.GetInterfaces ()) {
  229. tw.WriteStartElement ("suds","extends", MetaData.SudsNamespace);
  230. tw.WriteAttributeString ("type", GetQualifiedXmlType (tw, interf, null));
  231. tw.WriteEndElement ();
  232. }
  233. }
  234. else if (type.IsValueType)
  235. {
  236. tw.WriteStartElement ("suds", "struct", MetaData.SudsNamespace);
  237. tw.WriteAttributeString ("type", GetQualifiedXmlType (tw, type, null));
  238. if (type.BaseType != typeof(ValueType))
  239. tw.WriteAttributeString ("extends", GetQualifiedXmlType (tw, type.BaseType, null));
  240. }
  241. else
  242. {
  243. tw.WriteStartElement ("suds", "class", MetaData.SudsNamespace);
  244. tw.WriteAttributeString ("type", GetQualifiedXmlType (tw, type, null));
  245. if (IsService (type))
  246. {
  247. if (type.IsMarshalByRef)
  248. tw.WriteAttributeString ("rootType", "MarshalByRefObject");
  249. else
  250. tw.WriteAttributeString ("rootType", "Delegate");
  251. if (type.BaseType != typeof(MarshalByRefObject))
  252. tw.WriteAttributeString ("extends", GetQualifiedXmlType (tw, type.BaseType, null));
  253. if (type.IsMarshalByRef) {
  254. foreach (Type interf in type.GetInterfaces ()) {
  255. tw.WriteStartElement ("suds","implements", MetaData.SudsNamespace);
  256. tw.WriteAttributeString ("type", GetQualifiedXmlType (tw, interf, null));
  257. tw.WriteEndElement ();
  258. }
  259. }
  260. }
  261. else if (typeof(ISerializable).IsAssignableFrom (type))
  262. tw.WriteAttributeString ("rootType", "ISerializable");
  263. }
  264. tw.WriteEndElement (); // suds
  265. }
  266. void WriteMessageBindingBody (XmlTextWriter tw, Type t)
  267. {
  268. tw.WriteStartElement ("soap", "body", MetaData.SoapNamespace);
  269. tw.WriteAttributeString ("use", "encoded");
  270. tw.WriteAttributeString ("encodingStyle", MetaData.SoapEncodingNamespace);
  271. tw.WriteAttributeString ("namespace", GetXmlNamespace (t, null));
  272. tw.WriteEndElement ();
  273. }
  274. void WritePart (XmlTextWriter tw, string name, Type t, Type containerType)
  275. {
  276. tw.WriteStartElement ("part", MetaData.WsdlNamespace);
  277. tw.WriteAttributeString ("name", name);
  278. tw.WriteAttributeString ("type", GetQualifiedXmlType (tw, t, containerType));
  279. tw.WriteEndElement ();
  280. }
  281. void WriteDataTypeSchema (XmlTextWriter tw, Type type)
  282. {
  283. if (type.IsArray)
  284. WriteArraySchema (tw, type);
  285. else if (type.IsEnum)
  286. WriteEnumSchema (tw, type);
  287. else
  288. WriteClassSchema (tw, type);
  289. }
  290. void WriteArraySchema (XmlTextWriter tw, Type type)
  291. {
  292. tw.WriteStartElement ("complexType", MetaData.SchemaNamespace);
  293. tw.WriteAttributeString ("name", GetXmlType (type));
  294. tw.WriteStartElement ("complexContent", MetaData.SchemaNamespace);
  295. tw.WriteStartElement ("restriction", MetaData.SchemaNamespace);
  296. tw.WriteAttributeString ("base", GetQualifiedName (tw, MetaData.SoapEncodingNamespace, "Array"));
  297. tw.WriteStartElement ("attribute", MetaData.SchemaNamespace);
  298. tw.WriteAttributeString ("ref", GetQualifiedName (tw, MetaData.SoapEncodingNamespace, "arrayType"));
  299. string arrayType = "";
  300. while (type.IsArray)
  301. {
  302. arrayType = arrayType + "[" + new string (',', type.GetArrayRank()-1) + "]";
  303. type = type.GetElementType ();
  304. }
  305. arrayType = GetQualifiedXmlType (tw, type, null) + arrayType;
  306. tw.WriteAttributeString ("wsdl", "arrayType", MetaData.WsdlNamespace, arrayType);
  307. tw.WriteEndElement (); // attribute
  308. tw.WriteEndElement (); // restriction
  309. tw.WriteEndElement (); // complexContent
  310. tw.WriteEndElement (); // complexType
  311. }
  312. void WriteEnumSchema (XmlTextWriter tw, Type type)
  313. {
  314. tw.WriteStartElement ("simpleType", MetaData.SchemaNamespace);
  315. tw.WriteAttributeString ("name", GetXmlType (type));
  316. tw.WriteAttributeString ("suds", "enumType", MetaData.SudsNamespace, GetQualifiedXmlType (tw, type.UnderlyingSystemType, null));
  317. tw.WriteStartElement ("restriction", MetaData.SchemaNamespace);
  318. tw.WriteAttributeString ("base", "xsd:string");
  319. foreach (string name in Enum.GetNames (type))
  320. {
  321. tw.WriteStartElement ("enumeration", MetaData.SchemaNamespace);
  322. tw.WriteAttributeString ("value", name);
  323. tw.WriteEndElement ();
  324. }
  325. tw.WriteEndElement (); // restriction
  326. tw.WriteEndElement (); // simpleType
  327. }
  328. void WriteClassSchema (XmlTextWriter tw, Type type)
  329. {
  330. tw.WriteStartElement ("element", MetaData.SchemaNamespace);
  331. tw.WriteAttributeString ("name", type.Name);
  332. tw.WriteAttributeString ("type", GetQualifiedXmlType (tw, type, null));
  333. tw.WriteEndElement ();
  334. tw.WriteStartElement ("complexType", MetaData.SchemaNamespace);
  335. tw.WriteAttributeString ("name", GetXmlType (type));
  336. if (type.BaseType != null && type.BaseType != typeof(object) && type.BaseType != typeof(ValueType))
  337. tw.WriteAttributeString ("base", GetQualifiedXmlType (tw, type.BaseType, null));
  338. FieldInfo[] fields = type.GetFields (BindingFlags.Public | BindingFlags.NonPublic | BindingFlags.Instance | BindingFlags.DeclaredOnly);
  339. // Element fields
  340. bool elemsStart = false;
  341. foreach (FieldInfo fi in fields)
  342. {
  343. SoapFieldAttribute att = (SoapFieldAttribute) InternalRemotingServices.GetCachedSoapAttribute (fi);
  344. if (att.UseAttribute) continue;
  345. if (!elemsStart) { tw.WriteStartElement ("all", MetaData.SchemaNamespace); elemsStart = true; }
  346. tw.WriteStartElement ("element", MetaData.SchemaNamespace);
  347. tw.WriteAttributeString ("name", att.XmlElementName);
  348. tw.WriteAttributeString ("type", GetQualifiedXmlType (tw, fi.FieldType, type));
  349. tw.WriteEndElement ();
  350. }
  351. if (elemsStart) tw.WriteEndElement (); // all
  352. // Attribute fields
  353. foreach (FieldInfo fi in fields)
  354. {
  355. SoapFieldAttribute att = (SoapFieldAttribute) InternalRemotingServices.GetCachedSoapAttribute (fi);
  356. if (!att.UseAttribute) continue;
  357. tw.WriteStartElement ("attribute", MetaData.SchemaNamespace);
  358. tw.WriteAttributeString ("name", att.XmlElementName);
  359. tw.WriteAttributeString ("type", GetQualifiedXmlType (tw, fi.FieldType, type));
  360. tw.WriteEndElement ();
  361. }
  362. tw.WriteEndElement (); // complexType
  363. }
  364. ArrayList FindServices (ServiceType[] servicetypes)
  365. {
  366. ArrayList list = new ArrayList ();
  367. foreach (ServiceType st in servicetypes)
  368. if (IsService (st.ObjectType)) list.Add (st);
  369. return list;
  370. }
  371. string GetSoapAction (MethodInfo mb)
  372. {
  373. return SoapServices.GetSoapActionFromMethodBase (mb);
  374. }
  375. string GetXmlNamespace (Type t, Type containerType)
  376. {
  377. string name, ns;
  378. if (t.IsArray)
  379. {
  380. return GetXmlNamespace (containerType, null);
  381. }
  382. if (SoapServices.GetXmlTypeForInteropType (t, out name, out ns))
  383. return ns;
  384. SoapTypeAttribute att = (SoapTypeAttribute) InternalRemotingServices.GetCachedSoapAttribute (t);
  385. return att.XmlNamespace;
  386. }
  387. string GetQualifiedName (XmlTextWriter tw, string namspace, string localName)
  388. {
  389. return tw.LookupPrefix (namspace) + ":" + localName;
  390. }
  391. string GetQualifiedXmlType (XmlTextWriter tw, Type type, Type containerType)
  392. {
  393. string name, ns;
  394. if (type.IsArray)
  395. {
  396. name = GetXmlType (type);
  397. ns = GetXmlNamespace (type, containerType);
  398. }
  399. else
  400. {
  401. name = GetXsdType (type);
  402. if (name != null) return "xsd:" + name;
  403. if (!SoapServices.GetXmlTypeForInteropType (type, out name, out ns))
  404. {
  405. SoapTypeAttribute att = (SoapTypeAttribute) InternalRemotingServices.GetCachedSoapAttribute (type);
  406. name = att.XmlTypeName;
  407. ns = att.XmlNamespace;
  408. }
  409. }
  410. return GetQualifiedName (tw, ns, name);
  411. }
  412. string GetXmlType (Type type)
  413. {
  414. if (type.IsArray)
  415. {
  416. string itemType = GetXmlType (type.GetElementType ());
  417. itemType = "ArrayOf" + char.ToUpper (itemType[0]) + itemType.Substring (1);
  418. if (type.GetArrayRank () > 1) itemType += type.GetArrayRank ();
  419. return itemType;
  420. }
  421. else
  422. {
  423. string name = null, ns;
  424. name = GetXsdType (type);
  425. if (name != null) return name;
  426. if (SoapServices.GetXmlTypeForInteropType (type, out name, out ns))
  427. return name;
  428. SoapTypeAttribute att = (SoapTypeAttribute) InternalRemotingServices.GetCachedSoapAttribute (type);
  429. return att.XmlTypeName;
  430. }
  431. }
  432. string GetServiceName (Type t)
  433. {
  434. return t.Name + "Service";
  435. }
  436. string GetPortName (Type t)
  437. {
  438. return t.Name + "Port";
  439. }
  440. string GetBindingName (Type t)
  441. {
  442. return t.Name + "Binding";
  443. }
  444. void FindTypes (ServiceType[] servicetypes, Hashtable dataTypes, ArrayList services)
  445. {
  446. Hashtable types = new Hashtable ();
  447. ArrayList mbrTypes = new ArrayList();
  448. foreach (ServiceType st in servicetypes)
  449. FindDataTypes (st.ObjectType, null, dataTypes, mbrTypes);
  450. foreach (Type mbrType in mbrTypes)
  451. {
  452. ServiceType stFound = null;
  453. foreach (ServiceType st in servicetypes)
  454. if (mbrType == st.ObjectType) stFound = st;
  455. if (stFound != null) services.Add (stFound);
  456. else services.Add (new ServiceType (mbrType));
  457. }
  458. }
  459. void FindDataTypes (Type t, Type containerType, Hashtable types, ArrayList services)
  460. {
  461. if (IsSystemType (t))
  462. {
  463. string ns = GetXmlNamespace (t, null);
  464. types [ns] = null;
  465. return;
  466. }
  467. if (!IsService (t))
  468. {
  469. if (!t.IsSerializable) return;
  470. string ns = GetXmlNamespace (t, containerType);
  471. SchemaInfo sinfo = (SchemaInfo) types [ns];
  472. if (sinfo == null)
  473. {
  474. sinfo = new SchemaInfo ();
  475. types [ns] = sinfo;
  476. }
  477. if (sinfo.Types.Contains (t)) return;
  478. sinfo.Types.Add (t);
  479. if (t.IsArray) return;
  480. FieldInfo[] fields = t.GetFields (BindingFlags.Public | BindingFlags.NonPublic | BindingFlags.Instance | BindingFlags.DeclaredOnly);
  481. foreach (FieldInfo fi in fields)
  482. {
  483. string fns = GetXmlNamespace (fi.FieldType, t);
  484. if (!sinfo.Imports.Contains (fns)) sinfo.Imports.Add (fns);
  485. FindDataTypes (fi.FieldType, t, types, services);
  486. }
  487. }
  488. else
  489. {
  490. if (services.Contains (t)) return;
  491. services.Add (t);
  492. foreach (MethodInfo met in t.GetMethods (BindingFlags.Public | BindingFlags.Instance | BindingFlags.DeclaredOnly))
  493. {
  494. ParameterInfo[] pars = met.GetParameters ();
  495. foreach (ParameterInfo par in pars)
  496. FindDataTypes (par.ParameterType, t, types, services);
  497. FindDataTypes (met.ReturnType, t, types, services);
  498. }
  499. }
  500. }
  501. bool IsService (Type t)
  502. {
  503. return t.IsMarshalByRef || t.IsInterface || typeof(Delegate).IsAssignableFrom (t);
  504. }
  505. bool IsSystemType (Type t)
  506. {
  507. return t.FullName.StartsWith ("System.") && !t.IsArray;
  508. }
  509. static string GetXsdType (Type type)
  510. {
  511. if (type.IsEnum) return null;
  512. switch (Type.GetTypeCode (type))
  513. {
  514. case TypeCode.Boolean: return "boolean";
  515. case TypeCode.Byte: return "unsignedByte";
  516. case TypeCode.Char: return "char";
  517. case TypeCode.DateTime: return "dateTime";
  518. case TypeCode.Decimal: return "decimal";
  519. case TypeCode.Double: return "double";
  520. case TypeCode.Int16: return "short";
  521. case TypeCode.Int32: return "int";
  522. case TypeCode.Int64: return "long";
  523. case TypeCode.SByte: return "byte";
  524. case TypeCode.Single: return "float";
  525. case TypeCode.UInt16: return "unsignedShort";
  526. case TypeCode.UInt32: return "unsignedInt";
  527. case TypeCode.UInt64: return "unsignedLong";
  528. case TypeCode.String: return "string";
  529. }
  530. if (type == typeof (TimeSpan))
  531. return "duration";
  532. if (type == typeof (object))
  533. return "anyType";
  534. return null;
  535. }
  536. }
  537. class SchemaInfo
  538. {
  539. public ArrayList Types = new ArrayList ();
  540. public ArrayList Imports = new ArrayList ();
  541. public bool SudsGenerated;
  542. }
  543. }