Monday, 13 October 2014

Date and time plugin for Visual Studio Web Test

A Visual Studio web performance test required the current date and time to be written to a context parameter for inserting into a request. Writing the current date and time to a string is straightforward and it is simple to write a string into a content parameter within the web test plugin.

using Microsoft.VisualStudio.TestTools.WebTesting;

namespace Plugins
{
    [System.ComponentModel.DisplayName(
        "Save current date time to context parameter")]
    [System.ComponentModel.Description(
        "Saves date time of call to a context parameter")]
    public class DateTimeToContext : WebTestRequestPlugin
    {
        [System.ComponentModel.DisplayName("Parameter name")]
        [System.ComponentModel.Description("Name of context parameter.")]
        public string ParameterName { get; set; }

        [System.ComponentModel.DisplayName("Date-time format")]
        [System.ComponentModel.DefaultValue("yyyy-MM-dd HH:mm:ss.fff")]
        public string DateTimeFormat { get; set; }

        public override void PreRequest(object sender, PreRequestEventArgs e)
        {
            string now 
                = string.IsNullOrWhiteSpace(DateTimeFormat) 
                ? System.DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss.fff")
                : System.DateTime.Now.ToString(DateTimeFormat);

            e.WebTest.AddCommentToResult(string.Format(
                "Setting context parameter '{0}' to '{1}'", ParameterName, now));
            e.WebTest.Context[ParameterName] = now;
        }
    }
} 
I like to add calls of e.WebTest.AddCommentToResult(...) at significant places in plugins, particularly when developing a plugin or a web test. In this case it simply reports the values that are set into the web test log.

Friday, 11 April 2014

Logging web test details to a text file

My next approach to logging was to by writing direct to text files. This worked fine and I was able to write large number of lines without obvious problems. I was concerned that opening and closing the files within each test case would be too much load, so I arranged to open the text file once per agent computer. A C# lock on the output file synchronises the multiple writers coming from the many test cases.

The code is written to have only one instance, so its contructor is private. The main class data items are the lock, the actual instance of the class and the output stream.

    public class SingletonWriter
    {
        private static object myLock = new object();
        private static SingletonWriter mySingleton = null;

        StreamWriter outStream;


The constructor generates the file name and creates the corresponding output stream.

        private SingletonWriter(string AgentId)
        {
            string streamName

                = "TestStatusAsComment."
                + System.DateTime.Now.ToString("yyyy-MM-dd.HHmm.")
                + AgentId + ".log";

            outStream = new StreamWriter(streamName, true);

                // true means append to an existing file.
            outStream.WriteLine("Start of output");
            outStream.Flush();
        }


Writeline is the main interface. It needs to access some parts of the web test details, it writes a simple string. After each write the stream is flushed.

        public static void WriteLine(PostWebTestEventArgs e,
                                     string line)
        {
            lock ( myLock ) {
                if ( mySingleton == null ) {
                    string AgentId =

                       ContextReader.GetContextField(
                           e, "$AgentId", "00");
                    mySingleton = new SingletonWriter(AgentId);
                }

                mySingleton.outStream.WriteLine(line);
                mySingleton.outStream.Flush();
            }
        }

Closing the stream allows one test run to generate multiple output files, if the created filenames are different. The stream creation above will append to an existing file rather than overwrite it.

        public static void Close()
        {
            lock ( myLock ) {
                if ( mySingleton != null ) {
                    mySingleton.outStream.WriteLine(

                       "End of output");
                    mySingleton.outStream.Close();
                    mySingleton = null;
                }
            }
        }
    }



Having got a write line routine it is a simple matter to use it in a plugin. The code below writes some details of the test case and the date, plus the contents of one context variable. An extra property is added to allow output from all test cases, or just those that are not passing at the time of the call.

public class WriteTestStatusAsComment : WebTestPlugin
{
    [System.ComponentModel.DisplayName("Write status for all tests")]
    [System.ComponentModel.Description("Only write for failing tests when false.")]
    [System.ComponentModel.DefaultValue("false")]
    public bool WriteStatusForAllTests { get; set; }

    [System.ComponentModel.Category("Additional data")]
    [System.ComponentModel.Description("Include value of this context variable, if any")]
    public string Context1 { get; set; }

    private StringBuilder sb;

    public override void PostWebTest(object sender, PostWebTestEventArgs e)
    {
        if ( WriteStatusForAllTests || e.WebTest.Outcome != Outcome.Pass ) {
            sb = new StringBuilder();

            sb.Append(System.DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss.fff "));
            sb.Append(e.WebTest.Outcome.ToString());
            sb.Append(" Iter: ");
            sb.Append(ContextReader.GetContextField(e, "$WebTestIteration"));
            sb.Append(" UserId: ");
            sb.Append(ContextReader.GetContextField(e, "$WebTestUserId"));
            sb.Append(' ');
            sb.Append(e.WebTest.Name);

            AddContext(e, Context1);

            string message = sb.ToString();
            e.WebTest.AddCommentToResult(message);

            SingletonWriter.WriteLine(e, message);
        }
    }
}