using System.Collections.Generic; using System.Linq; using Microsoft.VisualStudio.TestTools.UnitTesting; using Moq; using ZoneCodeGenerator.Domain; using ZoneCodeGenerator.Parsing; using ZoneCodeGenerator.Parsing.C_Header; using ZoneCodeGenerator.Parsing.C_Header.Blocks; using ZoneCodeGenerator.Parsing.C_Header.Tests; using ZoneCodeGenerator.Parsing.Testing; namespace ZoneCodeGeneratorTests.Parsing.C_Header.Tests { [TestClass] public class TestStructTest { private Mock parserStateMock; private List typesOfState; private Namespace currentNamespace; private BlockStruct pushedStruct; private Mock lexerMock; private int tokenOffset; private List tokens; [TestInitialize] public void Setup() { currentNamespace = new Namespace(); parserStateMock = new Mock(); typesOfState = new List(); parserStateMock.SetupGet(state => state.CurrentNamespace) .Returns(() => currentNamespace); parserStateMock.Setup(state => state.PushBlock(It.IsAny())) .Callback((Block block) => pushedStruct = block as BlockStruct); parserStateMock.Setup(state => state.FindType(It.IsAny())) .Returns((string query) => typesOfState.FirstOrDefault(type => type.FullName.Equals(query))); pushedStruct = null; tokenOffset = 0; tokens = new List(); lexerMock = new Mock(); lexerMock.Setup(lexer => lexer.PeekToken(It.IsAny())) .Returns((int index) => tokens.ElementAtOrDefault(index + tokenOffset)); lexerMock.Setup(lexer => lexer.NextToken()) .Returns(() => tokens.ElementAtOrDefault(tokenOffset++)); lexerMock.Setup(lexer => lexer.SkipTokens(It.IsAny())) .Callback((int count) => tokenOffset += count); } [TestMethod] public void EnsureAcceptsSimpleStructStatement() { tokens.AddRange(new List { "struct", "{" }); var test = new TestStruct(); Assert.AreEqual(TokenTestResult.Match, test.PerformTest(parserStateMock.Object, lexerMock.Object)); Assert.AreEqual(2, test.ConsumedTokenCount); Assert.IsNotNull(pushedStruct); Assert.IsFalse(pushedStruct.IsTypedef); } [TestMethod] public void EnsureAcceptsNamedStructStatement() { tokens.AddRange(new List { "struct", "test_struct", "{" }); var test = new TestStruct(); Assert.AreEqual(TokenTestResult.Match, test.PerformTest(parserStateMock.Object, lexerMock.Object)); Assert.AreEqual(3, test.ConsumedTokenCount); Assert.IsNotNull(pushedStruct); Assert.IsFalse(pushedStruct.IsTypedef); Assert.AreEqual("test_struct", pushedStruct.Name); } [TestMethod] public void EnsureCanBeTypedef() { tokens.AddRange(new List { "typedef", "struct", "test_struct", "{" }); var test = new TestStruct(); Assert.AreEqual(TokenTestResult.Match, test.PerformTest(parserStateMock.Object, lexerMock.Object)); Assert.AreEqual(4, test.ConsumedTokenCount); Assert.IsNotNull(pushedStruct); Assert.IsTrue(pushedStruct.IsTypedef); Assert.AreEqual("test_struct", pushedStruct.Name); } [TestMethod] public void EnsureAcceptsCustomSpecifiedAlignment() { tokens.AddRange(new List { "struct", "__declspec", "(", "align", "(", "64", ")", ")", "test_struct", "{" }); var test = new TestStruct(); Assert.AreEqual(TokenTestResult.Match, test.PerformTest(parserStateMock.Object, lexerMock.Object)); Assert.AreEqual(10, test.ConsumedTokenCount); Assert.IsNotNull(pushedStruct); Assert.IsFalse(pushedStruct.IsTypedef); Assert.AreEqual("test_struct", pushedStruct.Name); Assert.AreEqual(64, pushedStruct.CustomAlignment); } [TestMethod] public void EnsureCanSpecifyParent() { tokens.AddRange(new List { "struct", "test_struct", ":", "test", ":", ":", "parent", "{" }); var test = new TestStruct(); var parent = new DataTypeStruct("test", "parent", 4); var parentEntry = new Variable("test_int", new TypeDeclaration(DataTypeBaseType.INT, new List())); parent.Members.Add(parentEntry); typesOfState.Add(parent); Assert.AreEqual(TokenTestResult.Match, test.PerformTest(parserStateMock.Object, lexerMock.Object)); Assert.AreEqual(8, test.ConsumedTokenCount); Assert.IsNotNull(pushedStruct); Assert.IsFalse(pushedStruct.IsTypedef); Assert.AreEqual("test_struct", pushedStruct.Name); Assert.AreEqual(1, pushedStruct.Variables.Count); Assert.AreEqual(parentEntry, pushedStruct.Variables[0]); } [TestMethod] public void EnsureFailsWhenParentTypeCannotBeFound() { tokens.AddRange(new List { "struct", "test_struct", ":", "test", ":", ":", "parent", "{" }); var test = new TestStruct(); Assert.ThrowsException( () => test.PerformTest(parserStateMock.Object, lexerMock.Object)); } [TestMethod] public void EnsureFailsWhenParentTypeIsNotDataTypeWithMembers() { tokens.AddRange(new List { "struct", "test_struct", ":", "test", ":", ":", "parent", "{" }); var test = new TestStruct(); var testEnum = new DataTypeEnum("test", "parent", DataTypeBaseType.INT); typesOfState.Add(testEnum); Assert.ThrowsException( () => test.PerformTest(parserStateMock.Object, lexerMock.Object)); } } }