| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383 |
- //
- // System.Web.Services.Protocols.DiscoveryClientProtocol.cs
- //
- // Author:
- // Dave Bettin ([email protected])
- // Lluis Sanchez Gual ([email protected])
- //
- // Copyright (C) Dave Bettin, 2002
- //
- //
- // Permission is hereby granted, free of charge, to any person obtaining
- // a copy of this software and associated documentation files (the
- // "Software"), to deal in the Software without restriction, including
- // without limitation the rights to use, copy, modify, merge, publish,
- // distribute, sublicense, and/or sell copies of the Software, and to
- // permit persons to whom the Software is furnished to do so, subject to
- // the following conditions:
- //
- // The above copyright notice and this permission notice shall be
- // included in all copies or substantial portions of the Software.
- //
- // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
- // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
- // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
- // NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
- // LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
- // OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
- // WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
- //
- using System.Collections;
- using System.IO;
- using System.Web.Services.Protocols;
- using System.Web.Services.Description;
- using System.Xml;
- using System.Xml.Schema;
- using System.Xml.Serialization;
- using System.Net;
- using System.Text.RegularExpressions;
- namespace System.Web.Services.Discovery {
- public class DiscoveryClientProtocol : HttpWebClientProtocol {
- #region Fields
- private IList additionalInformation = new ArrayList ();
- private DiscoveryClientDocumentCollection documents = new DiscoveryClientDocumentCollection();
- private DiscoveryExceptionDictionary errors = new DiscoveryExceptionDictionary();
- private DiscoveryClientReferenceCollection references = new DiscoveryClientReferenceCollection();
- #endregion // Fields
- #region Constructors
- public DiscoveryClientProtocol ()
- {
- }
-
- #endregion // Constructors
- #region Properties
- public IList AdditionalInformation {
- get { return additionalInformation; }
- }
-
- public DiscoveryClientDocumentCollection Documents {
- get { return documents; }
- }
-
- public DiscoveryExceptionDictionary Errors {
- get { return errors; }
- }
- public DiscoveryClientReferenceCollection References {
- get { return references; }
- }
-
- #endregion // Properties
- #region Methods
- public DiscoveryDocument Discover (string url)
- {
- Stream stream = Download (ref url);
- XmlTextReader reader = new XmlTextReader (url, stream);
- reader.XmlResolver = null;
- if (!DiscoveryDocument.CanRead (reader))
- throw new InvalidOperationException ("The url '" + url + "' does not point to a valid discovery document");
-
- DiscoveryDocument doc = DiscoveryDocument.Read (reader);
- reader.Close ();
- documents.Add (url, doc);
- AddDiscoReferences (doc);
- return doc;
- }
- public DiscoveryDocument DiscoverAny (string url)
- {
- try
- {
- string contentType = null;
- Stream stream = Download (ref url, ref contentType);
-
- if (contentType.IndexOf ("text/html") != -1)
- {
- // Look for an alternate url
-
- StreamReader sr = new StreamReader (stream);
- string str = sr.ReadToEnd ();
-
- string rex = "link\\s*rel\\s*=\\s*[\"']?alternate[\"']?\\s*";
- rex += "type\\s*=\\s*[\"']?text/xml[\"']?\\s*href\\s*=\\s*(?:\"(?<1>[^\"]*)\"|'(?<1>[^']*)'|(?<1>\\S+))";
- Regex rob = new Regex (rex, RegexOptions.IgnoreCase);
- Match m = rob.Match (str);
- if (!m.Success)
- throw new InvalidOperationException ("The HTML document does not contain Web service discovery information");
-
- if (url.StartsWith ("/"))
- {
- Uri uri = new Uri (url);
- url = uri.GetLeftPart (UriPartial.Authority) + m.Groups[1];
- }
- else
- {
- int i = url.LastIndexOf ('/');
- if (i == -1)
- throw new InvalidOperationException ("The HTML document does not contain Web service discovery information");
- Uri tmp = new Uri (url);
- tmp = new Uri (tmp, m.Groups [1].ToString ());
- url = tmp.ToString ();
- }
- stream = Download (ref url);
- }
-
- XmlTextReader reader = new XmlTextReader (url, stream);
- reader.XmlResolver = null;
- reader.MoveToContent ();
- DiscoveryDocument doc;
- DiscoveryReference refe = null;
-
- if (DiscoveryDocument.CanRead (reader))
- {
- doc = DiscoveryDocument.Read (reader);
- documents.Add (url, doc);
- refe = new DiscoveryDocumentReference ();
- AddDiscoReferences (doc);
- }
- else if (ServiceDescription.CanRead (reader))
- {
- ServiceDescription wsdl = ServiceDescription.Read (reader);
- documents.Add (url, wsdl);
- doc = new DiscoveryDocument ();
- refe = new ContractReference ();
- doc.References.Add (refe);
- refe.Url = url;
- ((ContractReference)refe).ResolveInternal (this, wsdl);
- }
- else
- {
- XmlSchema schema = XmlSchema.Read (reader, null);
- documents.Add (url, schema);
- doc = new DiscoveryDocument ();
- refe = new SchemaReference ();
- ((SchemaReference)refe).ResolveInternal (this, schema);
- doc.References.Add (refe);
- }
-
- refe.ClientProtocol = this;
- refe.Url = url;
- references.Add (url, refe);
-
- reader.Close ();
- return doc;
- }
- catch (DiscoveryException ex) {
- throw ex.Exception;
- }
- }
-
- void AddDiscoReferences (DiscoveryDocument doc)
- {
- foreach (DiscoveryReference re in doc.References)
- {
- re.ClientProtocol = this;
- references.Add (re.Url, re);
- }
-
- if (doc.AdditionalInfo != null) {
- foreach (object info in doc.AdditionalInfo)
- additionalInformation.Add (info);
- }
- }
-
- public Stream Download (ref string url)
- {
- string contentType = null;
- return Download (ref url, ref contentType);
- }
-
- public Stream Download (ref string url, ref string contentType)
- {
- if (url.StartsWith ("http://") || url.StartsWith ("https://"))
- {
- WebRequest request = GetWebRequest (new Uri(url));
- WebResponse resp = request.GetResponse ();
- contentType = resp.ContentType;
- return resp.GetResponseStream ();
- }
- else if (url.StartsWith ("file://"))
- {
- WebRequest request = WebRequest.Create (new Uri (url));
- WebResponse resp = request.GetResponse ();
- contentType = resp.ContentType;
- return resp.GetResponseStream ();
- }
- else
- {
- string ext = Path.GetExtension (url).ToLower();
- if (ext == ".wsdl" || ext == ".xsd")
- {
- contentType = "text/xml";
- return new FileStream (url, FileMode.Open, FileAccess.Read);
- }
- else
- throw new InvalidOperationException ("Unrecognized file type '" + url + "'. Extension must be one of .wsdl or .xsd");
- }
- }
- #if NET_2_0
- [System.Runtime.InteropServices.ComVisible (false)]
- #endif
- [Obsolete ("This method will be removed from a future version. The method call is no longer required for resource discovery", false)]
- public void LoadExternals ()
- {
- }
- public DiscoveryClientResultCollection ReadAll (string topLevelFilename)
- {
- StreamReader sr = new StreamReader (topLevelFilename);
- XmlSerializer ser = new XmlSerializer (typeof (DiscoveryClientResultsFile));
- DiscoveryClientResultsFile resfile = (DiscoveryClientResultsFile) ser.Deserialize (sr);
- sr.Close ();
-
- string basePath = Path.GetDirectoryName (topLevelFilename);
-
- foreach (DiscoveryClientResult dcr in resfile.Results)
- {
- Type type = Type.GetType (dcr.ReferenceTypeName);
- DiscoveryReference dr = (DiscoveryReference) Activator.CreateInstance (type);
- dr.Url = dcr.Url;
- FileStream fs = new FileStream (Path.Combine (basePath, dcr.Filename), FileMode.Open, FileAccess.Read);
- Documents.Add (dr.Url, dr.ReadDocument (fs));
- fs.Close ();
- References.Add (dr.Url, dr);
- }
- return resfile.Results;
- }
- public void ResolveAll ()
- {
- ArrayList list = new ArrayList (References.Values);
- foreach (DiscoveryReference re in list)
- {
- try
- {
- if (re is DiscoveryDocumentReference)
- ((DiscoveryDocumentReference)re).ResolveAll ();
- else
- re.Resolve ();
- }
- catch (DiscoveryException ex)
- {
- Errors [ex.Url] = ex.Exception;
- }
- catch (Exception ex)
- {
- Errors [re.Url] = ex;
- }
- }
- }
-
- public void ResolveOneLevel ()
- {
- ArrayList list = new ArrayList (References.Values);
- foreach (DiscoveryReference re in list)
- re.Resolve ();
- }
-
- public DiscoveryClientResultCollection WriteAll (string directory, string topLevelFilename)
- {
- DiscoveryClientResultsFile resfile = new DiscoveryClientResultsFile();
-
- foreach (DiscoveryReference re in References.Values)
- {
- object doc = Documents [re.Url];
- if (doc == null) continue;
-
- string fileName = FindValidName (resfile, re.DefaultFilename);
- resfile.Results.Add (new DiscoveryClientResult (re.GetType(), re.Url, fileName));
-
- string filepath = Path.Combine (directory, fileName);
- FileStream fs = new FileStream (filepath, FileMode.Create, FileAccess.Write);
- re.WriteDocument (doc, fs);
- fs.Close ();
- }
-
- StreamWriter sw = new StreamWriter (Path.Combine (directory, topLevelFilename));
- XmlSerializer ser = new XmlSerializer (typeof (DiscoveryClientResultsFile));
- ser.Serialize (sw, resfile);
- sw.Close ();
- return resfile.Results;
- }
-
- string FindValidName (DiscoveryClientResultsFile resfile, string baseName)
- {
- string name = baseName;
- int id = 0;
- bool found;
- do
- {
- found = false;
- foreach (DiscoveryClientResult res in resfile.Results)
- {
- if (name == res.Filename) {
- found = true; break;
- }
- }
- if (found)
- name = Path.GetFileNameWithoutExtension (baseName) + (++id) + Path.GetExtension (baseName);
- }
- while (found);
-
- return name;
- }
-
- #endregion // Methods
-
- #region Classes
-
- public sealed class DiscoveryClientResultsFile {
-
- #region Fields
-
- private DiscoveryClientResultCollection results;
- #endregion // Fields
- #region Contructors
-
- public DiscoveryClientResultsFile ()
- {
- results = new DiscoveryClientResultCollection ();
- }
-
- #endregion // Constructors
-
- #region Properties
-
- public DiscoveryClientResultCollection Results {
- get { return results; }
- }
-
- #endregion // Properties
- }
-
- #endregion // Classes
- }
-
- internal class DiscoveryException : Exception
- {
- public string Url;
- public Exception Exception;
-
- public DiscoveryException (string url, Exception origin)
- {
- Url = url;
- Exception = origin;
- }
- }
- }
|