This should fix #76928. This fix incorporates ideas from a patch
[mono.git] / mcs / class / System.Web.Services / System.Web.Services.Protocols / SoapDocumentationHandler.cs
1 //
2 // System.Web.Services.Protocols.SoapDocumentationHandler.cs
3 //
4 // Author:
5 //   Lluis Sanchez Gual (lluis@ximian.com)
6 //
7 // Copyright (C) Ximian, Inc. 2003
8 //
9
10 //
11 // Permission is hereby granted, free of charge, to any person obtaining
12 // a copy of this software and associated documentation files (the
13 // "Software"), to deal in the Software without restriction, including
14 // without limitation the rights to use, copy, modify, merge, publish,
15 // distribute, sublicense, and/or sell copies of the Software, and to
16 // permit persons to whom the Software is furnished to do so, subject to
17 // the following conditions:
18 // 
19 // The above copyright notice and this permission notice shall be
20 // included in all copies or substantial portions of the Software.
21 // 
22 // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
23 // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
24 // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
25 // NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
26 // LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
27 // OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
28 // WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
29 //
30
31 using System;
32 using System.Web;
33 using System.IO;
34 using System.Globalization;
35 using System.Xml;
36 using System.Text;
37 using System.Xml.Serialization;
38 using System.Xml.Schema;
39 using System.Web.Services.Description;
40 using System.Web.Services.Discovery;
41 using System.Web.Services.Configuration;
42 using System.Configuration;
43 using System.CodeDom;
44 using System.CodeDom.Compiler;
45 using Microsoft.CSharp;
46 using System.Web.UI;
47
48 namespace System.Web.Services.Protocols
49 {
50         internal class SoapDocumentationHandler: WebServiceHandler
51         {
52                 SoapTypeStubInfo _typeStubInfo;
53                 ServiceDescriptionCollection _descriptions;
54                 XmlSchemas _schemas;
55                 string _url;
56                 IHttpHandler _pageHandler = null;
57
58                 public SoapDocumentationHandler (Type type, HttpContext context): base (type)
59                 {
60                         _url = context.Request.Url.ToString();
61                         int i = _url.LastIndexOf ('?');
62                         if (i != -1) _url = _url.Substring (0,i);
63                         _typeStubInfo = (SoapTypeStubInfo) TypeStubManager.GetTypeStub (ServiceType, "Soap");
64                         
65                         HttpRequest req = context.Request;
66                         string key = null;
67                         if (req.QueryString.Count == 1) {
68                                 key = req.QueryString.GetKey (0);
69                                 if (key == null)
70                                         key = req.QueryString [0];
71
72                                 if (key != null)
73                                         key = key.ToLower (CultureInfo.InvariantCulture);
74                         }
75                                 
76                         if (key == "wsdl" || key == "schema" || key == "code" || key == "disco")
77                                 return;
78                                 
79 #if NET_2_0 && CONFIGURATION_2_0
80                         string help = WebServicesSection.Instance.WsdlHelpGenerator.Href;
81                         string path = Path.GetDirectoryName (ConfigurationManager.OpenMachineConfiguration().FilePath);
82 #else
83                         string help = WSConfig.Instance.WsdlHelpPage;
84                         string path = Path.GetDirectoryName (WSConfig.Instance.ConfigFilePath);
85 #endif
86                         string appPath = AppDomain.CurrentDomain.GetData (".appPath").ToString ();
87                         string vpath;
88                         if (path.StartsWith (appPath)) {
89                                 vpath = path.Substring (appPath.Length);
90                                 vpath = vpath.Replace ("\\", "/");
91                         } else {
92                                 vpath = "/";
93                         }
94
95                         if (vpath.EndsWith ("/"))
96                                 vpath += help;
97                         else
98                                 vpath += "/" + help;
99
100                         string physPath = Path.Combine (path, help);
101                         
102                         if (!File.Exists (physPath))
103                                 throw new InvalidOperationException ("Documentation page '" + physPath + "' not found");
104
105                         _pageHandler = PageParser.GetCompiledPageInstance (vpath, physPath, context);
106                                 
107                 }
108
109                 internal IHttpHandler PageHandler {
110                         get { return _pageHandler; }
111                 }
112
113                 public override bool IsReusable 
114                 {
115                         get { return false; }
116                 }
117
118                 public override void ProcessRequest (HttpContext context)
119                 {
120                         if (_pageHandler != null)
121                         {
122                                 context.Items["wsdls"] = GetDescriptions ();
123                                 context.Items["schemas"] = GetSchemas ();
124                                 _pageHandler.ProcessRequest (context);
125                         }
126                         else
127                         {
128                                 HttpRequest req = context.Request;
129                                 string key = req.QueryString.GetKey (0);
130                                 if (key == null)
131                                         key = req.QueryString [0];
132
133                                 if (key != null)
134                                         key = key.ToLower (CultureInfo.InvariantCulture);
135
136                                 if (key  == "wsdl") GenerateWsdlDocument (context, req.QueryString ["wsdl"]);
137                                 else if (key == "schema") GenerateSchema (context, req.QueryString ["schema"]);
138                                 else if (key == "code") GenerateCode (context, req.QueryString ["code"]);
139                                 else if (key == "disco") GenerateDiscoDocument (context);
140                                 else throw new Exception ("This should never happen");
141                         }
142                 }
143
144                 void GenerateWsdlDocument (HttpContext context, string wsdlId)
145                 {
146                         int di = 0;
147                         if (wsdlId != null && wsdlId != "") di = int.Parse (wsdlId);
148                         
149                         context.Response.ContentType = "text/xml; charset=utf-8";
150                         XmlTextWriter xtw = new XmlTextWriter (context.Response.OutputStream, new UTF8Encoding (false));
151                         xtw.Formatting = Formatting.Indented;
152                         GetDescriptions() [di].Write (xtw);
153                 }
154                 
155                 void GenerateDiscoDocument (HttpContext context)
156                 {
157                         ServiceDescriptionCollection descs = GetDescriptions ();
158                         
159                         DiscoveryDocument doc = new DiscoveryDocument ();
160                         ContractReference cref = new ContractReference ();
161                         cref.Ref = _url + "?wsdl";
162                         cref.DocRef = _url;
163                         doc.References.Add (cref);
164                         
165                         foreach (ServiceDescription desc in descs)
166                                 foreach (Service ser in desc.Services)
167                                         foreach (Port port in ser.Ports)
168                                         {
169                                                 SoapAddressBinding sab = port.Extensions.Find (typeof(SoapAddressBinding)) as SoapAddressBinding;
170                                                 if (sab != null)
171                                                 {
172                                                         System.Web.Services.Discovery.SoapBinding dsb = new System.Web.Services.Discovery.SoapBinding ();
173                                                         dsb.Address = sab.Location;
174                                                         dsb.Binding = port.Binding;
175                                                         doc.AdditionalInfo.Add (dsb);
176                                                 }
177                                         }
178
179                         context.Response.ContentType = "text/xml; charset=utf-8";
180                         XmlTextWriter xtw = new XmlTextWriter (context.Response.OutputStream, new UTF8Encoding (false));
181                         xtw.Formatting = Formatting.Indented;
182                         doc.Write (xtw);
183                 }
184                 
185                 void GenerateSchema (HttpContext context, string schemaId)
186                 {
187                         int di = 0;
188                         if (schemaId != null && schemaId != "") di = int.Parse (schemaId);
189                         
190                         context.Response.ContentType = "text/xml; charset=utf-8";
191                         XmlTextWriter xtw = new XmlTextWriter (context.Response.OutputStream, new UTF8Encoding (false));
192                         xtw.Formatting = Formatting.Indented;
193                         GetSchemas() [di].Write (xtw);
194                 }
195                 
196                 void GenerateCode (HttpContext context, string langId)
197                 {
198                         context.Response.ContentType = "text/plain; charset=utf-8";
199                         CodeNamespace codeNamespace = new CodeNamespace();
200                         CodeCompileUnit codeUnit = new CodeCompileUnit();
201                         
202                         codeUnit.Namespaces.Add (codeNamespace);
203
204                         ServiceDescriptionImporter importer = new ServiceDescriptionImporter();
205                         
206                         foreach (ServiceDescription sd in GetDescriptions ())
207                                 importer.AddServiceDescription(sd, null, null);
208
209                         foreach (XmlSchema sc in GetSchemas())
210                                 importer.Schemas.Add (sc);
211
212                         importer.Import(codeNamespace, codeUnit);
213                         
214                         if (langId == null || langId == "") langId = "cs";
215                         CodeDomProvider provider = GetProvider (langId);
216                         ICodeGenerator generator = provider.CreateGenerator();
217                         CodeGeneratorOptions options = new CodeGeneratorOptions();
218                         
219                         generator.GenerateCodeFromCompileUnit(codeUnit, context.Response.Output, options);
220                 }
221                 
222                 private CodeDomProvider GetProvider(string langId)
223                 {
224                         // FIXME these should be loaded dynamically using reflection
225                         CodeDomProvider provider;
226                         
227                         switch (langId.ToUpper())
228                         {
229                             case "CS":
230                                     provider = new CSharpCodeProvider();
231                                     break;
232                             
233                             default:
234                                     throw new Exception("Unknown language: " + langId);
235                         }
236
237                         return provider;
238                 }
239                 
240                 internal ServiceDescriptionCollection GetDescriptions ()
241                 {
242                         if (_descriptions == null)
243                         {
244                                 ServiceDescriptionReflector reflector = new ServiceDescriptionReflector ();
245                                 reflector.Reflect (ServiceType,_url);
246                                 _schemas = reflector.Schemas;
247                                 _descriptions = reflector.ServiceDescriptions;
248                         }
249                         return _descriptions;
250                 }
251                 
252                 internal XmlSchemas GetSchemas()
253                 {
254                         GetDescriptions();
255                         return _schemas;
256                 }
257         }
258 }