10 changed files with 492 additions and 15 deletions
@ -0,0 +1,231 @@
@@ -0,0 +1,231 @@
|
||||
// Copyright (c) AlphaSierraPapa for the SharpDevelop Team (for details please see \doc\copyright.txt)
|
||||
// This code is distributed under MIT X11 license (for details please see \doc\license.txt)
|
||||
|
||||
using System; |
||||
using System.Collections.Generic; |
||||
using System.Diagnostics; |
||||
using System.IO; |
||||
using ICSharpCode.NRefactory.TypeSystem; |
||||
|
||||
namespace ICSharpCode.NRefactory.Documentation |
||||
{ |
||||
/// <summary>
|
||||
/// Provides xml documentation from a binary cache file.
|
||||
/// This allows providing XML documentation without having to read the whole documentation into memory.
|
||||
/// </summary>
|
||||
public class BinaryDocumentationProvider : IDisposable, IDocumentationProvider |
||||
{ |
||||
struct IndexEntry |
||||
{ |
||||
public readonly int HashCode; |
||||
public readonly int FileLocation; |
||||
|
||||
public IndexEntry(int HashCode, int FileLocation) |
||||
{ |
||||
this.HashCode = HashCode; |
||||
this.FileLocation = FileLocation; |
||||
} |
||||
} |
||||
|
||||
#region Save binary files
|
||||
// FILE FORMAT FOR BINARY DOCUMENTATION
|
||||
// long magic = 0x4244636f446c6d58 (identifies file type = 'XmlDocDB')
|
||||
const long magic = 0x4244636f446c6d58; |
||||
// short version = 3 (file version)
|
||||
const short version = 3; |
||||
// long fileDate (last change date of xml file in DateTime ticks)
|
||||
// int testHashCode = magicTestString.GetHashCode() // (check if hash-code implementation is compatible)
|
||||
const string magicTestString = "HashMe-XmlDocDB"; |
||||
// int entryCount (count of entries)
|
||||
// int indexPointer (points to location where index starts in the file)
|
||||
// {
|
||||
// string key (documentation key as length-prefixed string)
|
||||
// string docu (xml documentation as length-prefixed string)
|
||||
// }
|
||||
// indexPointer points to the start of the following section:
|
||||
// {
|
||||
// int hashcode
|
||||
// int index (index where the docu string starts in the file)
|
||||
// }
|
||||
|
||||
|
||||
/// <summary>
|
||||
/// Saves the xml documentation into a on-disk database file.
|
||||
/// </summary>
|
||||
/// <param name="fileName">Filename of the database</param>
|
||||
/// <param name="fileDate">Last-modified date of the .xml file</param>
|
||||
/// <param name="xmlDocumentation">The xml documentation that should be written to disk.</param>
|
||||
public static void Save(string fileName, DateTime fileDate, IEnumerable<KeyValuePair<string, string>> xmlDocumentation) |
||||
{ |
||||
if (fileName == null) |
||||
throw new ArgumentNullException("fileName"); |
||||
if (xmlDocumentation == null) |
||||
throw new ArgumentNullException("xmlDocumentation"); |
||||
using (FileStream fs = new FileStream(fileName, FileMode.Create, FileAccess.Write, FileShare.None)) { |
||||
using (BinaryWriter w = new BinaryWriter(fs)) { |
||||
w.Write(magic); |
||||
w.Write(version); |
||||
w.Write(fileDate.Ticks); |
||||
w.Write(magicTestString.GetHashCode()); |
||||
|
||||
List<IndexEntry> index = new List<IndexEntry>(); |
||||
int indexLengthPos = (int)fs.Position; |
||||
w.Write(0); // skip 4 bytes for index length
|
||||
w.Write(0); // skip 4 bytes for index pointer
|
||||
|
||||
int i = 0; |
||||
foreach (KeyValuePair<string, string> p in xmlDocumentation) { |
||||
index.Add(new IndexEntry(p.Key.GetHashCode(), (int)fs.Position)); |
||||
w.Write(p.Key); |
||||
w.Write(p.Value.Trim()); |
||||
i += 1; |
||||
} |
||||
|
||||
index.Sort((a,b) => a.HashCode.CompareTo(b.HashCode)); |
||||
|
||||
int indexStart = (int)fs.Position; |
||||
foreach (IndexEntry entry in index) { |
||||
w.Write(entry.HashCode); |
||||
w.Write(entry.FileLocation); |
||||
} |
||||
w.Seek(indexLengthPos, SeekOrigin.Begin); |
||||
w.Write(index.Count); // write index length
|
||||
w.Write(indexStart); // write index count
|
||||
} |
||||
} |
||||
} |
||||
#endregion
|
||||
|
||||
BinaryReader loader; |
||||
FileStream fs; |
||||
|
||||
Dictionary<string, string> xmlDescription = new Dictionary<string, string>(); |
||||
IndexEntry[] index; // SORTED array of index entries
|
||||
|
||||
const int cacheLength = 50; // number of strings to cache when working in file-mode
|
||||
Queue<string> keyCacheQueue = new Queue<string>(cacheLength); |
||||
|
||||
#region Load binary files
|
||||
private BinaryDocumentationProvider() {} |
||||
|
||||
/// <summary>
|
||||
/// Loads binary documentation.
|
||||
/// </summary>
|
||||
/// <remarks>
|
||||
/// Don't forget to dispose the BinaryDocumentationProvider.
|
||||
/// </remarks>
|
||||
/// <param name="fileName">The name of the binary cache file.</param>
|
||||
/// <param name="fileDate">The file date of the original XML file. Loading will fail if the cached data was generated
|
||||
/// from a different file date than the original XML file.</param>
|
||||
/// <returns>
|
||||
/// The BinaryDocumentationProvider representing the file's content; or null if loading failed.
|
||||
/// </returns>
|
||||
public static BinaryDocumentationProvider Load(string fileName, DateTime fileDate) |
||||
{ |
||||
BinaryDocumentationProvider doc = new BinaryDocumentationProvider(); |
||||
try { |
||||
doc.fs = new FileStream(fileName, FileMode.Open, FileAccess.Read, FileShare.Read | FileShare.Delete); |
||||
int len = (int)doc.fs.Length; |
||||
BinaryReader loader = doc.loader = new BinaryReader(doc.fs); |
||||
if (loader.ReadInt64() != magic) { |
||||
Debug.WriteLine("Cannot load XmlDoc: wrong magic"); |
||||
return null; |
||||
} |
||||
if (loader.ReadInt16() != version) { |
||||
Debug.WriteLine("Cannot load XmlDoc: wrong version"); |
||||
return null; |
||||
} |
||||
if (loader.ReadInt64() != fileDate.Ticks) { |
||||
Debug.WriteLine("Not loading XmlDoc: file changed since cache was created"); |
||||
return null; |
||||
} |
||||
int count = loader.ReadInt32(); |
||||
int indexStartPosition = loader.ReadInt32(); // go to start of index
|
||||
if (indexStartPosition <= 0 || indexStartPosition >= len) { |
||||
Debug.WriteLine("XmlDoc: Cannot find index, cache invalid!"); |
||||
return null; |
||||
} |
||||
doc.fs.Position = indexStartPosition; |
||||
IndexEntry[] index = new IndexEntry[count]; |
||||
for (int i = 0; i < index.Length; i++) { |
||||
index[i] = new IndexEntry(loader.ReadInt32(), loader.ReadInt32()); |
||||
} |
||||
doc.index = index; |
||||
return doc; |
||||
} catch (IOException ex) { |
||||
Debug.WriteLine("Cannot load from cache" + ex.ToString()); |
||||
return null; |
||||
} |
||||
} |
||||
|
||||
string LoadDocumentation(string key) |
||||
{ |
||||
if (keyCacheQueue.Count > cacheLength - 1) { |
||||
xmlDescription.Remove(keyCacheQueue.Dequeue()); |
||||
} |
||||
|
||||
int hashcode = key.GetHashCode(); |
||||
|
||||
// use binary search to find the item
|
||||
string resultDocu = null; |
||||
|
||||
int m = Array.BinarySearch(index, new IndexEntry(hashcode, 0)); |
||||
if (m >= 0) { |
||||
// correct hash code found.
|
||||
// possibly there are multiple items with the same hash, so go to the first.
|
||||
while (--m >= 0 && index[m].HashCode == hashcode); |
||||
// go through all items that have the correct hash
|
||||
while (++m < index.Length && index[m].HashCode == hashcode) { |
||||
fs.Position = index[m].FileLocation; |
||||
string keyInFile = loader.ReadString(); |
||||
if (keyInFile == key) { |
||||
//LoggingService.Debug("Got XML documentation for " + key);
|
||||
resultDocu = loader.ReadString(); |
||||
break; |
||||
} else { |
||||
// this is a harmless hash collision, just continue reading
|
||||
Debug.WriteLine("Found " + keyInFile + " instead of " + key); |
||||
} |
||||
} |
||||
} |
||||
|
||||
keyCacheQueue.Enqueue(key); |
||||
xmlDescription.Add(key, resultDocu); |
||||
|
||||
return resultDocu; |
||||
} |
||||
#endregion
|
||||
|
||||
public string GetDocumentation(string key) |
||||
{ |
||||
lock (xmlDescription) { |
||||
if (index == null) |
||||
throw new ObjectDisposedException("BinaryDocumentationProvider"); |
||||
string result; |
||||
if (xmlDescription.TryGetValue(key, out result)) |
||||
return result; |
||||
return LoadDocumentation(key); |
||||
} |
||||
} |
||||
|
||||
public string GetDocumentation(IEntity entity) |
||||
{ |
||||
return GetDocumentation(XmlDocumentationProvider.GetDocumentationKey(entity)); |
||||
} |
||||
|
||||
public void Dispose() |
||||
{ |
||||
lock (xmlDescription) { |
||||
if (loader != null) { |
||||
loader.Close(); |
||||
fs.Close(); |
||||
} |
||||
xmlDescription.Clear(); |
||||
index = null; |
||||
keyCacheQueue = null; |
||||
loader = null; |
||||
fs = null; |
||||
} |
||||
} |
||||
} |
||||
} |
||||
@ -0,0 +1,179 @@
@@ -0,0 +1,179 @@
|
||||
// Copyright (c) AlphaSierraPapa for the SharpDevelop Team (for details please see \doc\copyright.txt)
|
||||
// This code is distributed under MIT X11 license (for details please see \doc\license.txt)
|
||||
|
||||
using System; |
||||
using System.Collections.Generic; |
||||
using System.Diagnostics; |
||||
using System.IO; |
||||
using System.Xml; |
||||
|
||||
using ICSharpCode.NRefactory.TypeSystem; |
||||
|
||||
namespace ICSharpCode.NRefactory.Documentation |
||||
{ |
||||
/// <summary>
|
||||
/// Provides documentation from an .xml file (as generated by the Microsoft C# compiler).
|
||||
/// </summary>
|
||||
public class XmlDocumentationProvider : IDocumentationProvider |
||||
{ |
||||
readonly IDictionary<string, string> xmlDocumentation; |
||||
|
||||
#region Load From File
|
||||
public XmlDocumentationProvider(string fileName) |
||||
{ |
||||
if (fileName == null) |
||||
throw new ArgumentNullException("fileName"); |
||||
this.xmlDocumentation = new Dictionary<string, string>(); |
||||
using (XmlTextReader xmlReader = new XmlTextReader(fileName)) { |
||||
xmlReader.MoveToContent(); |
||||
if (string.IsNullOrEmpty(xmlReader.GetAttribute("redirect"))) { |
||||
ReadXmlDoc(xmlReader); |
||||
} else { |
||||
string redirectionTarget = GetRedirectionTarget(xmlReader.GetAttribute("redirect")); |
||||
if (redirectionTarget != null) { |
||||
Debug.WriteLine("XmlDoc " + fileName + " is redirecting to " + redirectionTarget); |
||||
using (XmlTextReader redirectedXmlReader = new XmlTextReader(redirectionTarget)) { |
||||
ReadXmlDoc(redirectedXmlReader); |
||||
} |
||||
} else { |
||||
Debug.WriteLine("XmlDoc " + fileName + " is redirecting to " + xmlReader.GetAttribute("redirect") + ", but that file was not found."); |
||||
} |
||||
} |
||||
} |
||||
} |
||||
|
||||
static string GetRedirectionTarget(string target) |
||||
{ |
||||
string programFilesDir = Environment.GetFolderPath(Environment.SpecialFolder.ProgramFiles); |
||||
programFilesDir = AppendDirectorySeparator(programFilesDir); |
||||
|
||||
string corSysDir = System.Runtime.InteropServices.RuntimeEnvironment.GetRuntimeDirectory(); |
||||
corSysDir = AppendDirectorySeparator(corSysDir); |
||||
|
||||
return LookupLocalizedXmlDoc(target.Replace("%PROGRAMFILESDIR%", programFilesDir) |
||||
.Replace("%CORSYSDIR%", corSysDir)); |
||||
} |
||||
|
||||
static string AppendDirectorySeparator(string dir) |
||||
{ |
||||
if (dir.EndsWith("\\", StringComparison.Ordinal) || dir.EndsWith("/", StringComparison.Ordinal)) |
||||
return dir; |
||||
else |
||||
return dir + Path.DirectorySeparatorChar; |
||||
} |
||||
|
||||
internal static string LookupLocalizedXmlDoc(string fileName) |
||||
{ |
||||
string xmlFileName = Path.ChangeExtension(fileName, ".xml"); |
||||
string currentCulture = System.Threading.Thread.CurrentThread.CurrentUICulture.TwoLetterISOLanguageName; |
||||
string localizedXmlDocFile = GetLocalizedName(xmlFileName, currentCulture); |
||||
|
||||
Debug.WriteLine("Try find XMLDoc @" + localizedXmlDocFile); |
||||
if (File.Exists(localizedXmlDocFile)) { |
||||
return localizedXmlDocFile; |
||||
} |
||||
Debug.WriteLine("Try find XMLDoc @" + xmlFileName); |
||||
if (File.Exists(xmlFileName)) { |
||||
return xmlFileName; |
||||
} |
||||
if (currentCulture != "en") { |
||||
string englishXmlDocFile = GetLocalizedName(xmlFileName, "en"); |
||||
Debug.WriteLine("Try find XMLDoc @" + englishXmlDocFile); |
||||
if (File.Exists(englishXmlDocFile)) { |
||||
return englishXmlDocFile; |
||||
} |
||||
} |
||||
return null; |
||||
} |
||||
|
||||
static string GetLocalizedName(string fileName, string language) |
||||
{ |
||||
string localizedXmlDocFile = Path.GetDirectoryName(fileName); |
||||
localizedXmlDocFile = Path.Combine(localizedXmlDocFile, language); |
||||
localizedXmlDocFile = Path.Combine(localizedXmlDocFile, Path.GetFileName(fileName)); |
||||
return localizedXmlDocFile; |
||||
} |
||||
#endregion
|
||||
|
||||
#region Load from XmlReader
|
||||
public XmlDocumentationProvider(XmlReader reader) |
||||
{ |
||||
if (reader == null) |
||||
throw new ArgumentNullException("reader"); |
||||
this.xmlDocumentation = new Dictionary<string, string>(); |
||||
ReadXmlDoc(reader); |
||||
} |
||||
|
||||
public XmlDocumentationProvider(IDictionary<string, string> dictionary) |
||||
{ |
||||
if (dictionary == null) |
||||
throw new ArgumentNullException("dictionary"); |
||||
this.xmlDocumentation = dictionary; |
||||
} |
||||
|
||||
void ReadXmlDoc(XmlReader reader) |
||||
{ |
||||
while (reader.Read()) { |
||||
if (reader.IsStartElement()) { |
||||
switch (reader.LocalName) { |
||||
case "members": |
||||
ReadMembersSection(reader); |
||||
break; |
||||
} |
||||
} |
||||
} |
||||
} |
||||
|
||||
void ReadMembersSection(XmlReader reader) |
||||
{ |
||||
while (reader.Read()) { |
||||
switch (reader.NodeType) { |
||||
case XmlNodeType.EndElement: |
||||
if (reader.LocalName == "members") { |
||||
return; |
||||
} |
||||
break; |
||||
case XmlNodeType.Element: |
||||
if (reader.LocalName == "member") { |
||||
string memberAttr = reader.GetAttribute(0); |
||||
string innerXml = reader.ReadInnerXml(); |
||||
xmlDocumentation[memberAttr] = innerXml; |
||||
} |
||||
break; |
||||
} |
||||
} |
||||
} |
||||
#endregion
|
||||
|
||||
/// <summary>
|
||||
/// Gets all entries in the documentation file.
|
||||
/// </summary>
|
||||
public IDictionary<string, string> XmlDocumentation { |
||||
get { return xmlDocumentation; } |
||||
} |
||||
|
||||
public string GetDocumentation(string key) |
||||
{ |
||||
if (key == null) |
||||
throw new ArgumentNullException("key"); |
||||
|
||||
string result; |
||||
if (xmlDocumentation.TryGetValue(key, out result)) |
||||
return result; |
||||
else |
||||
return null; |
||||
} |
||||
|
||||
public string GetDocumentation(IEntity entity) |
||||
{ |
||||
return GetDocumentation(GetDocumentationKey(entity)); |
||||
} |
||||
|
||||
public static string GetDocumentationKey(IEntity entity) |
||||
{ |
||||
if (entity == null) |
||||
throw new ArgumentNullException("entity"); |
||||
throw new NotImplementedException(); |
||||
} |
||||
} |
||||
} |
||||
@ -0,0 +1,19 @@
@@ -0,0 +1,19 @@
|
||||
// Copyright (c) AlphaSierraPapa for the SharpDevelop Team (for details please see \doc\copyright.txt)
|
||||
// This code is distributed under MIT X11 license (for details please see \doc\license.txt)
|
||||
|
||||
using System; |
||||
using System.IO; |
||||
|
||||
namespace ICSharpCode.NRefactory.TypeSystem |
||||
{ |
||||
/// <summary>
|
||||
/// Provides XML documentation for members.
|
||||
/// </summary>
|
||||
public interface IDocumentationProvider |
||||
{ |
||||
/// <summary>
|
||||
/// Gets the XML documentation for the specified entity.
|
||||
/// </summary>
|
||||
string GetDocumentation(IEntity entity); |
||||
} |
||||
} |
||||
Loading…
Reference in new issue