You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					
					
						
							156 lines
						
					
					
						
							6.1 KiB
						
					
					
				
			
		
		
	
	
							156 lines
						
					
					
						
							6.1 KiB
						
					
					
				// Copyright (c) AlphaSierraPapa for the SharpDevelop Team | 
						|
//  | 
						|
// 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.IO; | 
						|
using ICSharpCode.NRefactory.Editor; | 
						|
using ICSharpCode.NRefactory.TypeSystem; | 
						|
using NUnit.Framework; | 
						|
 | 
						|
namespace ICSharpCode.NRefactory.CSharp.Parser | 
						|
{ | 
						|
	/// <summary> | 
						|
	/// Test fixture that parses NRefactory itself, | 
						|
	/// ensures that there are no parser crashes while doing so, | 
						|
	/// and that the returned positions are consistent. | 
						|
	/// </summary> | 
						|
	[TestFixture] | 
						|
	public class ParseSelfTests | 
						|
	{ | 
						|
		string[] fileNames; | 
						|
		 | 
						|
		[TestFixtureSetUp] | 
						|
		public void SetUp() | 
						|
		{ | 
						|
			string path = Path.GetFullPath (Path.Combine ("..", "..", "..")); | 
						|
			if (!File.Exists(Path.Combine(path, "NRefactory.sln"))) | 
						|
				throw new InvalidOperationException("Test cannot find the NRefactory source code in " + path); | 
						|
			fileNames = Directory.GetFiles(path, "*.cs", SearchOption.AllDirectories); | 
						|
		} | 
						|
		 | 
						|
		[Test] | 
						|
		public void GenerateTypeSystem() | 
						|
		{ | 
						|
			IProjectContent pc = new CSharpProjectContent(); | 
						|
			CSharpParser parser = new CSharpParser(); | 
						|
			parser.GenerateTypeSystemMode = true; | 
						|
			foreach (string fileName in fileNames) { | 
						|
				SyntaxTree syntaxTree; | 
						|
				using (var fs = new FileStream(fileName, FileMode.Open, FileAccess.Read, FileShare.Read, 4096, FileOptions.SequentialScan)) { | 
						|
					syntaxTree = parser.Parse(fs, fileName); | 
						|
				} | 
						|
				var unresolvedFile = syntaxTree.ToTypeSystem(); | 
						|
				foreach (var td in unresolvedFile.GetAllTypeDefinitions()) { | 
						|
					Assert.AreSame(unresolvedFile, td.UnresolvedFile); | 
						|
					foreach (var member in td.Members) { | 
						|
						Assert.AreSame(unresolvedFile, member.UnresolvedFile); | 
						|
						Assert.AreSame(td, member.DeclaringTypeDefinition); | 
						|
					} | 
						|
				} | 
						|
				pc = pc.AddOrUpdateFiles(unresolvedFile); | 
						|
			} | 
						|
		} | 
						|
		 | 
						|
		#region ParseAndCheckPositions | 
						|
		string currentFileName; | 
						|
		ReadOnlyDocument currentDocument; | 
						|
		 | 
						|
		[Test, Ignore("Positions still are incorrect in several cases")] | 
						|
		public void ParseAndCheckPositions() | 
						|
		{ | 
						|
			CSharpParser parser = new CSharpParser(); | 
						|
			foreach (string fileName in fileNames) { | 
						|
				this.currentDocument = new ReadOnlyDocument(File.ReadAllText(fileName)); | 
						|
				SyntaxTree syntaxTree = parser.Parse(currentDocument, fileName); | 
						|
				if (parser.HasErrors) | 
						|
					continue; | 
						|
				this.currentFileName = fileName; | 
						|
				CheckPositionConsistency(syntaxTree); | 
						|
				CheckMissingTokens(syntaxTree); | 
						|
			} | 
						|
		} | 
						|
 | 
						|
		void PrintNode (AstNode node) | 
						|
		{ | 
						|
			Console.WriteLine ("Parent:" + node.GetType ()); | 
						|
			Console.WriteLine ("Children:"); | 
						|
			foreach (var c in node.Children) | 
						|
				Console.WriteLine (c.GetType () +" at:"+ c.StartLocation +"-"+ c.EndLocation + " Role: "+ c.Role); | 
						|
			Console.WriteLine ("----"); | 
						|
		} | 
						|
		 | 
						|
		void CheckPositionConsistency (AstNode node) | 
						|
		{ | 
						|
			string comment = "(" + node.GetType ().Name + " at " + node.StartLocation + " in " + currentFileName + ")"; | 
						|
			var pred = node.StartLocation <= node.EndLocation; | 
						|
			if (!pred) | 
						|
				PrintNode (node); | 
						|
			Assert.IsTrue(pred, "StartLocation must be before EndLocation " + comment); | 
						|
			var prevNodeEnd = node.StartLocation; | 
						|
			var prevNode = node; | 
						|
			for (AstNode child = node.FirstChild; child != null; child = child.NextSibling) { | 
						|
				bool assertion = child.StartLocation >= prevNodeEnd; | 
						|
				if (!assertion) { | 
						|
					PrintNode (prevNode); | 
						|
					PrintNode (node); | 
						|
					 | 
						|
				} | 
						|
				Assert.IsTrue(assertion, currentFileName + ": Child " + child.GetType () +" (" + child.StartLocation  + ")" +" must start after previous sibling " + prevNode.GetType () + "(" + prevNode.StartLocation + ")"); | 
						|
				CheckPositionConsistency(child); | 
						|
				prevNodeEnd = child.EndLocation; | 
						|
				prevNode = child; | 
						|
			} | 
						|
			Assert.IsTrue(prevNodeEnd <= node.EndLocation, "Last child must end before parent node ends " + comment); | 
						|
		} | 
						|
		 | 
						|
		void CheckMissingTokens(AstNode node) | 
						|
		{ | 
						|
			if (node is CSharpTokenNode) { | 
						|
				Assert.IsNull(node.FirstChild, "Token nodes should not have children"); | 
						|
			} else { | 
						|
				var prevNodeEnd = node.StartLocation; | 
						|
				var prevNode = node; | 
						|
				for (AstNode child = node.FirstChild; child != null; child = child.NextSibling) { | 
						|
					CheckWhitespace(prevNode, prevNodeEnd, child, child.StartLocation); | 
						|
					CheckMissingTokens(child); | 
						|
					prevNode = child; | 
						|
					prevNodeEnd = child.EndLocation; | 
						|
				} | 
						|
				CheckWhitespace(prevNode, prevNodeEnd, node, node.EndLocation); | 
						|
			} | 
						|
		} | 
						|
		 | 
						|
		void CheckWhitespace(AstNode startNode, TextLocation whitespaceStart, AstNode endNode, TextLocation whitespaceEnd) | 
						|
		{ | 
						|
			if (whitespaceStart == whitespaceEnd || startNode == endNode) | 
						|
				return; | 
						|
			int start = currentDocument.GetOffset(whitespaceStart.Line, whitespaceStart.Column); | 
						|
			int end = currentDocument.GetOffset(whitespaceEnd.Line, whitespaceEnd.Column); | 
						|
			string text = currentDocument.GetText(start, end - start); | 
						|
			bool assertion = string.IsNullOrWhiteSpace(text); | 
						|
			if (!assertion) { | 
						|
				if (startNode.Parent != endNode.Parent) | 
						|
					PrintNode (startNode.Parent); | 
						|
				PrintNode (endNode.Parent); | 
						|
			} | 
						|
			Assert.IsTrue(assertion, "Expected whitespace between " + startNode.GetType () +":" + whitespaceStart + " and " + endNode.GetType () + ":" + whitespaceEnd | 
						|
			              + ", but got '" + text + "' (in " + currentFileName + " parent:" + startNode.Parent.GetType () +")"); | 
						|
		} | 
						|
		#endregion | 
						|
	} | 
						|
}
 | 
						|
 |