You've already forked linux-packaging-mono
							
							
		
			
				
	
	
		
			184 lines
		
	
	
		
			6.2 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			184 lines
		
	
	
		
			6.2 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| //
 | |
| // msitomsx.cs: Microsoft Internal XML to Microsoft XML Documentation
 | |
| //
 | |
| // Arguably this doesn't belong in mdoc, but I'd rather not do some
 | |
| // stand-alone tool either, especially since the primary reason it exists is
 | |
| // to facilitate generating ECMA documentation via mdoc-update and
 | |
| // mdoc-update-ecma-xml...
 | |
| //
 | |
| // Author:
 | |
| //   Jonathan Pryor  <jpryor@novell.com>
 | |
| //
 | |
| // Copyright (c) 2010 Novell, Inc. (http://www.novell.com)
 | |
| //
 | |
| // 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;
 | |
| using System.Collections.Generic;
 | |
| using System.IO;
 | |
| using System.Reflection;
 | |
| using System.Text;
 | |
| using System.Xml;
 | |
| using System.Xml.Linq;
 | |
| using System.Xml.Xsl;
 | |
| 
 | |
| using Mono.Options;
 | |
| 
 | |
| namespace Mono.Documentation {
 | |
| 
 | |
| 	class MsidocToMsxdocConverter : MDocCommand {
 | |
| 
 | |
| 		XslCompiledTransform msiToMsxTransform = new XslCompiledTransform ();
 | |
| 
 | |
| 		public MsidocToMsxdocConverter ()
 | |
| 		{
 | |
| 			using (var r = XmlReader.Create (
 | |
| 						Assembly.GetExecutingAssembly ().GetManifestResourceStream ("msitomsx.xsl")))
 | |
| 				msiToMsxTransform.Load (r);
 | |
| 		}
 | |
| 
 | |
| 		public override void Run (IEnumerable<string> args)
 | |
| 		{
 | |
| 			var types = new List<string> ();
 | |
| 			string outdir = null;
 | |
| 
 | |
| 			var options = new OptionSet () {
 | |
| 				{ "o|out=", 
 | |
| 					"{DIRECTORY} to create Microsoft XML assembly.xml documentation files.",
 | |
| 					v => outdir = v },
 | |
| 				{ "library=",
 | |
| 					"Ignored for compatibility with update-ecma-xml.",
 | |
| 					v => {} },
 | |
| 				{ "type=",
 | |
| 					"The full {TYPE} name of a type to copy into the output file.",
 | |
| 					v => types.Add (v) },
 | |
| 			};
 | |
| 			var sources = Parse (options, args, "export-ecma-xml", 
 | |
| 					"[OPTIONS]+ DIRECTORIES",
 | |
| 					"Convert Microsoft internal XML documentation within DIRECTORIES into\n" +
 | |
| 					"Microsoft XML documentation.\n" +
 | |
| 					"\n" +
 | |
| 					"Microsoft internal XML documentation is a custom XML dialect (yay) with\n" +
 | |
| 					"a specific directory structure:\n" +
 | |
| 					"\n" +
 | |
| 					"\tROOT/dotted.namespace/asset.xml\n" +
 | |
| 					"\n" +
 | |
| 					"where ROOT is a directory that can be passed as one of the DIRECTORIES\n" +
 | |
| 					"arguments to x-msitomsx, dotted.namespace is the lowercased namespace\n" +
 | |
| 					"in dotted form, e.g. 'system.threading', and asset.xml is the name of \n" +
 | |
| 					"the \"asset\" being documented.  The asset.xml basename is a GUID, and\n" +
 | |
| 					"the file contains type/namespace/etc. documentation, one file per asset.\n" +
 | |
| 					"\n" +
 | |
| 					"There is no separation of assemblies in this format, only namespaces.\n" +
 | |
| 					"Types from all assemblies are intermixed with each other.\n" +
 | |
| 					"\n" +
 | |
| 					"The output of x-msitomsx will be a set of files in the --out directory,\n" +
 | |
| 					"one file per assembly (normal 'csc /doc' convention).  For example,\n" +
 | |
| 					"\n" +
 | |
| 					"\tmdoc x-msitomsx -o docs import-root --type System.String\n" +
 | |
| 					"\n" +
 | |
| 					"will create the file 'docs/mscorlib.dll' which will contain the converted\n" +
 | |
| 					"documentation for the System.String type.");
 | |
| 			if (sources == null)
 | |
| 				return;
 | |
| 			if (sources.Count == 0)
 | |
| 				Error ("No directories specified.");
 | |
| 			if (outdir == null)
 | |
| 				Error ("No output directory specified.  Please use --out=DIRECTORY.");
 | |
| 
 | |
| 			types.Sort ();
 | |
| 
 | |
| 			Dictionary<string, XDocument> docs = Convert (sources, types);
 | |
| 			foreach (KeyValuePair<string, XDocument> e in docs) {
 | |
| 				using (var o = CreateWriter (Path.Combine (outdir, e.Key + ".xml")))
 | |
| 					e.Value.WriteTo (o);
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		private Dictionary<string, XDocument> Convert (List<string> sources, List<string> types)
 | |
| 		{
 | |
| 			var docs = new Dictionary<string, XDocument> ();
 | |
| 
 | |
| 			foreach (var source in sources) {
 | |
| 				foreach (var dir in Directory.GetDirectories (source)) {
 | |
| 					foreach (var file in Directory.GetFiles (dir, "*.xml")) {
 | |
| 						ConvertDocs (docs, types, file);
 | |
| 					}
 | |
| 				}
 | |
| 			}
 | |
| 
 | |
| 			return docs;
 | |
| 		}
 | |
| 
 | |
| 		private void ConvertDocs (Dictionary<string, XDocument> docs, List<string> types, string file)
 | |
| 		{
 | |
| 			var doc = LoadFile (file);
 | |
| 			var type = doc.Root.Element ("members").Element ("member").Attribute ("name").Value;
 | |
| 
 | |
| 			if (type.StartsWith ("N:"))
 | |
| 				return;
 | |
| 
 | |
| 			if (!type.StartsWith ("T:"))
 | |
| 				throw new InvalidOperationException ("File '" + file + "' doesn't contain type documentation, it contains docs for: " + type);
 | |
| 
 | |
| 			type = type.Substring (2);
 | |
| 			if (types.Count > 0 && types.BinarySearch (type) < 0)
 | |
| 				return;
 | |
| 
 | |
| 			var assembly = doc.Root.Element ("assembly").Element ("name").Value;
 | |
| 			XDocument asmdocs;
 | |
| 			if (!docs.TryGetValue (assembly, out asmdocs)) {
 | |
| 				docs.Add (assembly, 
 | |
| 						asmdocs = new XDocument (
 | |
| 							new XElement ("doc", 
 | |
| 								new XElement ("assembly",
 | |
| 									new XElement ("name", assembly)),
 | |
| 								new XElement ("members"))));
 | |
| 			}
 | |
| 
 | |
| 			var import = new XDocument ();
 | |
| 			msiToMsxTransform.Transform (doc.CreateReader (), import.CreateWriter ());
 | |
| 
 | |
| 			asmdocs.Root.Element ("members").Add (import.Root.Element ("members").Elements ("member"));
 | |
| 		}
 | |
| 
 | |
| 		static XDocument LoadFile (string file)
 | |
| 		{
 | |
| 			using (XmlReader r = XmlReader.Create (file))
 | |
| 				return XDocument.Load (r);
 | |
| 		}
 | |
| 
 | |
| 		static XmlWriter CreateWriter (string file)
 | |
| 		{
 | |
| 			var settings = new XmlWriterSettings {
 | |
| 				Encoding            = new UTF8Encoding (false),
 | |
| 				Indent              = true,
 | |
| 				IndentChars         = "    ",
 | |
| 				NewLineChars        = "\r\n",
 | |
| 				OmitXmlDeclaration  = true,
 | |
| 			};
 | |
| 
 | |
| 			return XmlWriter.Create (file, settings);
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 |