Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Implement Workitem support in TRX logger #2666

Merged
2 commits merged into from
Dec 9, 2020
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
Expand Up @@ -16,6 +16,7 @@ internal interface ITestElement
TestExecId ParentExecutionId { get; set; }
TestListCategoryId CategoryId { get; set; }
TestCategoryItemCollection TestCategories { get; }
WorkItemCollection WorkItems { get; set; }
TestType TestType { get; }
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -30,6 +30,7 @@ internal abstract class TestElement : ITestElement, IXmlTestStore
protected TestExecId executionId;
protected TestExecId parentExecutionId;
protected TestCategoryItemCollection testCategories;
protected WorkItemCollection workItems;
protected TestListCategoryId catId;

public TestElement(Guid id, string name, string adapter)
Expand Down Expand Up @@ -161,6 +162,20 @@ public TestCategoryItemCollection TestCategories
}
}

/// <summary>
/// Gets or sets the work items.
/// </summary>
public WorkItemCollection WorkItems
{
get { return this.workItems; }

set
{
EqtAssert.ParameterNotNull(value, "value");
this.workItems = value;
}
}

/// <summary>
/// Gets the adapter name.
/// </summary>
Expand Down Expand Up @@ -230,6 +245,8 @@ public virtual void Save(System.Xml.XmlElement element, XmlTestStoreParameters p
if (this.parentExecutionId != null)
h.SaveGuid(element, "Execution/@parentId", this.parentExecutionId.Id);

h.SaveObject(this.workItems, element, "Workitems", parameters);

XmlTestStoreParameters testIdParameters = XmlTestStoreParameters.GetParameters();
testIdParameters[TestId.IdLocationKey] = "@id";
h.SaveObject(this.id, element, testIdParameters);
Expand All @@ -245,6 +262,7 @@ private void Initialize()
this.executionId = TestExecId.Empty;
this.parentExecutionId = TestExecId.Empty;
this.testCategories = new TestCategoryItemCollection();
this.workItems = new WorkItemCollection();
this.isRunnable = true;
this.catId = TestListCategoryId.Uncategorized;
}
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,243 @@
// Copyright (c) Microsoft Corporation. All rights reserved.
// Licensed under the MIT license. See LICENSE file in the project root for full license information.

namespace Microsoft.TestPlatform.Extensions.TrxLogger.ObjectModel
{
using System;
using System.Globalization;
using System.Text;
using System.Xml;
using Microsoft.TestPlatform.Extensions.TrxLogger.Utility;
using Microsoft.TestPlatform.Extensions.TrxLogger.XML;

#region WorkItem
/// <summary>
/// Stores an int which represents a workitem
/// </summary>
internal sealed class WorkItem : IXmlTestStore
{
#region Fields
[StoreXmlField(Location = ".")]
private int id = 0;

#endregion

#region Constructors
/// <summary>
/// Create a new item with the workitem set
/// </summary>
/// <param name="workitemId">The workitem.</param>
public WorkItem(int workitemId)
{
this.id = workitemId;
}

#endregion

#region Properties/Methods
/// <summary>
/// Gets the id for this WorkItem
/// </summary>
public int Id
{
get
{
return this.id;
}
}

#endregion

#region Methods - overrides
/// <summary>
/// Compare the values of the items
/// </summary>
/// <param name="other">Value being compared to.</param>
/// <returns>True if the values are the same and false otherwise.</returns>
public override bool Equals(object other)
{
WorkItem otherItem = other as WorkItem;
if (otherItem == null)
{
return false;
}
return this.id == otherItem.id;
}

/// <summary>
/// Convert the workitem to a hashcode
/// </summary>
/// <returns>Hashcode of the workitem.</returns>
public override int GetHashCode()
{
return this.id.GetHashCode();
}

/// <summary>
/// Convert the workitem to a string
/// </summary>
/// <returns>The workitem.</returns>
public override string ToString()
{
return this.id.ToString(CultureInfo.InvariantCulture);
}
#endregion

#region IXmlTestStore Members

/// <summary>
/// Saves the class under the XmlElement.
/// </summary>
/// <param name="element"> XmlElement element </param>
/// <param name="parameters"> XmlTestStoreParameters parameters</param>
public void Save(System.Xml.XmlElement element, XmlTestStoreParameters parameters)
{
new XmlPersistence().SaveSingleFields(element, this, parameters);
}

#endregion
}
#endregion

#region WorkItemCollection
/// <summary>
/// A collection of ints represent the workitems
/// </summary>
internal sealed class WorkItemCollection : EqtBaseCollection<WorkItem>
{
#region Constructors
/// <summary>
/// Creates an empty WorkItemCollection.
/// </summary>
public WorkItemCollection()
{
}

/// <summary>
/// Create a new WorkItemCollection based on the int array.
/// </summary>
/// <param name="items">Add these items to the collection.</param>
public WorkItemCollection(int[] items)
{
EqtAssert.ParameterNotNull(items, "items");
foreach (int i in items)
{
this.Add(new WorkItem(i));
}
}

#endregion

#region Methods
/// <summary>
/// Adds the workitem.
/// </summary>
/// <param name="item">WorkItem to be added.</param>
public void Add(int item)
{
this.Add(new WorkItem(item));
}

/// <summary>
/// Adds the workitem.
/// </summary>
/// <param name="item">WorkItem to be added.</param>
public override void Add(WorkItem item)
{
EqtAssert.ParameterNotNull(item, "item");
base.Add(item);
}

/// <summary>
/// Convert the WorkItemCollection to a string.
/// each item is separated by a comma (,)
/// </summary>
/// <returns></returns>
public override string ToString()
{
StringBuilder returnString = new StringBuilder();
if (this.Count > 0)
{
returnString.Append(",");
foreach (WorkItem item in this)
{
returnString.Append(item);
returnString.Append(",");
}
}

return returnString.ToString();
}

/// <summary>
/// Convert the WorkItemCollection to an array of ints.
/// </summary>
/// <returns>Array of ints containing the workitems.</returns>
public int[] ToArray()
{
int[] result = new int[this.Count];

int i = 0;
foreach (WorkItem item in this)
{
result[i++] = item.Id;
}

return result;
}

/// <summary>
/// Compare the collection items
/// </summary>
/// <param name="obj">other collection</param>
/// <returns>true if the collections contain the same items</returns>
public override bool Equals(object obj)
{
WorkItemCollection other = obj as WorkItemCollection;
bool result = false;

if (other == null)
{
result = false;
}
else if (object.ReferenceEquals(this, other))
{
result = true;
}
else if (this.Count != other.Count)
{
result = false;
}
else
{
foreach (WorkItem item in this)
{
if (!other.Contains(item))
{
result = false;
break;
}
}
}

return result;
}

/// <summary>
/// Return the hash code of this collection
/// </summary>
/// <returns>The hashcode.</returns>
public override int GetHashCode()
{
return base.GetHashCode();
}

public override void Save(XmlElement element, XmlTestStoreParameters parameters)
{
XmlPersistence xmlPersistence = new XmlPersistence();
xmlPersistence.SaveHashtable(this.container, element, ".", ".", null, "Workitem", parameters);
}
#endregion
}
#endregion
}
Original file line number Diff line number Diff line change
Expand Up @@ -67,6 +67,13 @@ public ITestElement ToTestElement(
testElement.TestCategories.Add(testCategory);
}

var workItems = GetCustomPropertyValueFromTestCase(rockSteadyTestCase, "WorkItemIds")
.Select(workItem => int.Parse(workItem));
foreach (int workItem in workItems)
{
testElement.WorkItems.Add(workItem);
}

return testElement;
}

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -736,13 +736,11 @@ public void CustomTrxFileNameShouldConstructFromLogFileParameter()
Assert.AreEqual(Path.Combine(TrxLoggerTests.DefaultTestRunDirectory, TrxLoggerTests.DefaultLogFileNameParameterValue), this.testableTrxLogger.trxFile, "Wrong Trx file name");
}



/// <summary>
/// Unit test for reading TestCategories from the TestCase which is part of test result.
/// </summary>
[TestMethod]
public void GetCustomPropertyValueFromTestCaseShouldReadCategoyrAttributesFromTestCase()
public void GetCustomPropertyValueFromTestCaseShouldReadCategoryAttributesFromTestCase()
{
ObjectModel.TestCase testCase1 = CreateTestCase("TestCase1");
TestProperty testProperty = TestProperty.Register("MSTestDiscoverer.TestCategory", "String array property", string.Empty, string.Empty, typeof(string[]), null, TestPropertyAttributes.Hidden, typeof(TestObject));
Expand All @@ -759,6 +757,24 @@ public void GetCustomPropertyValueFromTestCaseShouldReadCategoyrAttributesFromTe
CollectionAssert.AreEqual(listCategoriesExpected, listCategoriesActual);
}

[TestMethod]
public void GetCustomPropertyValueFromTestCaseShouldReadWorkItemAttributesFromTestCase()
{
ObjectModel.TestCase testCase1 = CreateTestCase("TestCase1");
TestProperty testProperty = TestProperty.Register("WorkItemIds", "String array property", string.Empty, string.Empty, typeof(string[]), null, TestPropertyAttributes.Hidden, typeof(TestObject));

testCase1.SetPropertyValue(testProperty, new[] { "99999", "0" });

var converter = new Converter(new Mock<IFileHelper>().Object, new TrxFileHelper());
List<string> listWorkItemsActual = converter.GetCustomPropertyValueFromTestCase(testCase1, "WorkItemIds");

List<string> listWorkItemsExpected = new List<string>();
listWorkItemsExpected.Add("99999");
listWorkItemsExpected.Add("0");

CollectionAssert.AreEqual(listWorkItemsExpected, listWorkItemsActual);
}

[TestMethod]
public void CRLFCharactersShouldGetRetainedInTrx()
{
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -98,6 +98,22 @@ public void ToTestElementShouldAssignTestCategoryOfUnitTestElement()
CollectionAssert.AreEqual(expected, unitTestElement.TestCategories.ToArray().OrderByDescending(x => x.ToString()).ToArray());
}

[TestMethod]
public void ToTestElementShouldAssignWorkitemOfUnitTestElement()
{
TestPlatformObjectModel.TestCase testCase = CreateTestCase("TestCase1");
TestPlatformObjectModel.TestResult result = new TestPlatformObjectModel.TestResult(testCase);
TestProperty testProperty = TestProperty.Register("WorkItemIds", "String array property", string.Empty, string.Empty, typeof(string[]), null, TestPropertyAttributes.Hidden, typeof(TestObject));

testCase.SetPropertyValue(testProperty, new[] { "3", "99999", "0" });

var unitTestElement = this.converter.ToTestElement(testCase.Id, Guid.Empty, Guid.Empty, testCase.DisplayName, TrxLoggerConstants.UnitTestType, testCase);

int[] expected = new[] { 0, 3, 99999 };

CollectionAssert.AreEquivalent(expected, unitTestElement.WorkItems.ToArray());
}

/// <summary>
/// Unit test for regression when there's no test categories.
/// </summary>
Expand Down