changeset 253:34df34841225 v3 v3.0.1-beta

Implab.Diagnostics drafts
author cin
date Mon, 12 Feb 2018 17:03:37 +0300 (2018-02-12)
parents 6f4630d0bcd9
children 12c00235b105
files Implab.Test/UnitTest1.cs Implab/AbstractEvent.cs Implab/Diagnostics/ActivityScope.cs Implab/Diagnostics/LogicalOperation.cs Implab/Diagnostics/LogicalOperationScope.cs Implab/Diagnostics/SimpleTraceListener.cs Implab/Diagnostics/Trace.cs Implab/Diagnostics/TraceEventCodes.cs Implab/Implab.csproj Implab/ValueEventArgs.cs
diffstat 10 files changed, 274 insertions(+), 48 deletions(-) [+]
line wrap: on
line diff
--- a/Implab.Test/UnitTest1.cs	Mon Feb 12 07:24:31 2018 +0300
+++ b/Implab.Test/UnitTest1.cs	Mon Feb 12 17:03:37 2018 +0300
@@ -4,34 +4,26 @@
 using Implab.Diagnostics;
 using Xunit;
 
-namespace Implab.Test
-{
+namespace Implab.Test {
+    using System.Threading.Tasks;
     using static Trace<UnitTest1>;
-    public class UnitTest1
-    {
+    public class UnitTest1 {
         [Fact]
-        public void Test1()
-        {
-            var listener = new TextWriterTraceListener(Console.Out);
+        public async Task Test1() {
+            var listener = new SimpleTraceListener(Console.Out);
+
             var source = TraceSource;
             source.Switch.Level = SourceLevels.All;
 
             source.Listeners.Add(listener);
-            Trace.Listeners.Add(listener);
 
-            Trace.WriteLine("Hello!");
-            StartLogicalOperation();
-
-            Trace.WriteLine("Inner");
-            foreach(var x in Trace.CorrelationManager.LogicalOperationStack)
-                Trace.WriteLine($"-{x}");
-            Log("source event");
-
-            listener.IndentLevel = 1;
-
-            source.TraceData(TraceEventType.Start, 1, DateTime.Now);
-            
-            StopLogicalOperation();
+            using (var op = LogicalOperation(nameof(Test1)))
+            using (LogicalOperation("InnerOperation")){
+                await Task.Yield();
+                Log("Inner");
+                await Task.Yield();
+                Log("source event");
+            }
         }
     }
 }
--- a/Implab/AbstractEvent.cs	Mon Feb 12 07:24:31 2018 +0300
+++ b/Implab/AbstractEvent.cs	Mon Feb 12 17:03:37 2018 +0300
@@ -62,7 +62,7 @@
             if (TransitionalState != Interlocked.CompareExchange(ref m_state, ResolvedState, TransitionalState))
                 throw new InvalidOperationException("Can't complete transition when the object isn't in the transitional state");
 #else
-            m_state = state;
+            m_state = ResolvedState;
 #endif
             Signal();
         }
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/Implab/Diagnostics/ActivityScope.cs	Mon Feb 12 17:03:37 2018 +0300
@@ -0,0 +1,29 @@
+using System;
+using System.Diagnostics;
+
+namespace Implab.Diagnostics {
+    public class ActivityScope : IDisposable {
+        readonly TraceSource m_source;
+
+        readonly Guid m_prevId;
+
+        readonly string m_activity;
+
+        readonly int m_code;
+
+        internal ActivityScope(TraceSource source, Guid prevId, int code, string activity) {
+            m_source = source;
+            m_prevId = prevId;
+            m_code = code;
+            m_activity = activity;
+        }
+
+
+        public void Dispose() {
+            if (Trace.CorrelationManager.ActivityId != m_prevId)
+                m_source.TraceTransfer(m_code, "Transfer", m_prevId);
+            m_source.TraceEvent(TraceEventType.Stop, 0, m_activity);
+            Trace.CorrelationManager.ActivityId = m_prevId;
+        }
+    }
+}
\ No newline at end of file
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/Implab/Diagnostics/LogicalOperation.cs	Mon Feb 12 17:03:37 2018 +0300
@@ -0,0 +1,17 @@
+using System;
+using System.Diagnostics;
+
+namespace Implab.Diagnostics {
+    public class LogicalOperation {
+        public Stopwatch OperationStopwatch { get; private set; }
+
+        public string Name { get; private set; }
+
+        internal LogicalOperation(string name) {
+            Name = string.IsNullOrEmpty(name) ? "<unnamed>" : name;
+            OperationStopwatch = Stopwatch.StartNew();
+        }
+
+        public override string ToString() => Name;
+    }
+}
\ No newline at end of file
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/Implab/Diagnostics/LogicalOperationScope.cs	Mon Feb 12 17:03:37 2018 +0300
@@ -0,0 +1,21 @@
+using System;
+using System.Diagnostics;
+
+namespace Implab.Diagnostics {
+    public class LogicalOperationScope : IDisposable {
+        readonly TraceSource m_source;
+
+        readonly LogicalOperation m_operation;
+
+        internal LogicalOperationScope(TraceSource source, LogicalOperation operation) {
+            m_source = source;
+            m_operation = operation;
+        }
+
+        public void Dispose() {
+            m_operation.OperationStopwatch.Stop();
+            Trace.CorrelationManager.StopLogicalOperation();
+            m_source.TraceData(TraceEventType.Information, TraceEventCodes.StopLogicalOperation, m_operation);
+        }
+    }
+}
\ No newline at end of file
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/Implab/Diagnostics/SimpleTraceListener.cs	Mon Feb 12 17:03:37 2018 +0300
@@ -0,0 +1,114 @@
+using System;
+using System.Diagnostics;
+using System.IO;
+
+namespace Implab.Diagnostics {
+    public class SimpleTraceListener : TextWriterTraceListener {
+        public SimpleTraceListener() {
+        }
+
+        public SimpleTraceListener(Stream stream) : base(stream) {
+        }
+
+        public SimpleTraceListener(TextWriter writer) : base(writer) {
+        }
+
+        public SimpleTraceListener(string fileName) : base(fileName) {
+        }
+
+        public SimpleTraceListener(Stream stream, string name) : base(stream, name) {
+        }
+
+        public SimpleTraceListener(TextWriter writer, string name) : base(writer, name) {
+        }
+
+        public SimpleTraceListener(string fileName, string name) : base(fileName, name) {
+        }
+
+        public override void TraceData(TraceEventCache eventCache, string source, TraceEventType eventType, int id, object data) {
+            switch (id) {
+                case TraceEventCodes.StartLogicalOperation:
+                    TraceEvent(eventCache, source, eventType, id, "+{0}", data);
+                    break;
+                case TraceEventCodes.StopLogicalOperation:
+                    TraceEvent(eventCache, source, eventType, id, FormatStopLogicalOperation(data));
+                    break;
+                default:
+                    TraceEvent(eventCache, source, eventType, id, data?.ToString());
+                    break;
+            }
+        }
+
+        string FormatStopLogicalOperation(object data) {
+            if (data is LogicalOperation op) {
+                return string.Format("-{0} ({1})", op, FormatTimespan(op.OperationStopwatch.Elapsed));
+            } else {
+                return data?.ToString();
+            }
+        }
+
+        string FormatTimespan(TimeSpan value) {
+            if (value.TotalSeconds < 10) {
+                return value.Milliseconds.ToString() + "ms";
+            } else if (value.TotalSeconds < 30) {
+                return string.Format("{0:0.###}s", value.TotalSeconds);
+            } else {
+                return value.ToString();
+            }
+        }
+
+        public override void TraceData(TraceEventCache eventCache, string source, TraceEventType eventType, int id, params object[] data) {
+            var prev = IndentLevel;
+            IndentLevel += eventCache.LogicalOperationStack.Count;
+            try {
+                base.TraceData(eventCache, source, eventType, id, data);
+            } finally {
+                IndentLevel = prev;
+            }
+        }
+
+        public override void TraceEvent(TraceEventCache eventCache, string source, TraceEventType eventType, int id) {
+            var prev = IndentLevel;
+            IndentLevel += eventCache.LogicalOperationStack.Count;
+            try {
+                base.TraceEvent(eventCache, source, eventType, id);
+            } finally {
+                IndentLevel = prev;
+            }
+        }
+
+        public override void TraceEvent(TraceEventCache eventCache, string source, TraceEventType eventType, int id, string format, params object[] args) {
+            TraceEvent(eventCache, source, eventType, id, String.Format(format, args));
+        }
+
+        public override void TraceEvent(TraceEventCache eventCache, string source, TraceEventType eventType, int id, string message) {
+            var prev = IndentLevel;
+            IndentLevel += eventCache.LogicalOperationStack.Count;
+            try {
+                LogicalOperation operation = null;
+                if (eventCache.LogicalOperationStack.Count > 0)
+                    operation = eventCache.LogicalOperationStack.Peek() as LogicalOperation;
+
+                if (operation != null) {
+                    base.TraceData(eventCache, source, eventType, id, FormatTimespan(operation.OperationStopwatch.Elapsed) + ": " + message);
+                } else {
+                    base.TraceData(eventCache, source, eventType, id, message);
+                }
+            } finally {
+                IndentLevel = prev;
+            }
+        }
+
+        public override void TraceTransfer(TraceEventCache eventCache, string source, int id, string message, Guid relatedActivityId) {
+            var prev = IndentLevel;
+            IndentLevel += eventCache.LogicalOperationStack.Count;
+            try {
+                base.TraceTransfer(eventCache, source, id, message, relatedActivityId);
+            } finally {
+                IndentLevel = prev;
+            }
+        }
+
+
+    }
+}
\ No newline at end of file
--- a/Implab/Diagnostics/Trace.cs	Mon Feb 12 07:24:31 2018 +0300
+++ b/Implab/Diagnostics/Trace.cs	Mon Feb 12 17:03:37 2018 +0300
@@ -29,7 +29,7 @@
         /// <param name="name">Name.</param>
         [Conditional("TRACE")]
         public static void StartLogicalOperation(string name) {
-            Trace.CorrelationManager.StartLogicalOperation();
+            Trace.CorrelationManager.StartLogicalOperation(name);
         }
 
         /// <summary>
@@ -47,7 +47,7 @@
         /// <param name="arguments">Arguments.</param>
         [Conditional("TRACE")]
         public static void Log(string format, params object[] arguments) {
-            TraceSource.TraceEvent(TraceEventType.Information, 1, format, arguments);
+            TraceSource.TraceEvent(TraceEventType.Information, 0, format, arguments);
         }
 
         /// <summary>
@@ -57,17 +57,71 @@
         /// <param name="arguments">Arguments.</param>
         [Conditional("TRACE")]
         public static void Warn(string format, params object[] arguments) {
-            TraceSource.TraceEvent(TraceEventType.Warning, 1, format, arguments);
+            TraceSource.TraceEvent(TraceEventType.Warning, 0, format, arguments);
         }
 
         [Conditional("TRACE")]
         public static void Error(string format, params object[] arguments) {
-            TraceSource.TraceEvent(TraceEventType.Error, 1, format, arguments);
+            TraceSource.TraceEvent(TraceEventType.Error, 0, format, arguments);
         }
 
         [Conditional("TRACE")]
         public static void Error(Exception err) {
-            TraceSource.TraceData(TraceEventType.Error, 1, err);
+            TraceSource.TraceData(TraceEventType.Error, 0, err);
+        }
+
+        /// <summary>
+        /// This method save the current activity, and transfers to the specified activity,
+        /// emits <see cref="TraceEventType.Start"/> and returns a scope of the new
+        /// activity.
+        /// </summary>
+        /// <param name="activityName">The name of the new activity/</param>
+        /// <param name="activityId">The identifier of the activity to which 
+        /// the control will be transferred</param>
+        /// <returns>A scope of the new activity, dispose it to transfer
+        /// the control back to the original activity.</returns>
+        public static ActivityScope TransferActivity(string activityName, Guid activityId) {
+            var prev = Trace.CorrelationManager.ActivityId;
+
+            TraceSource.TraceTransfer(0, "Transfer", activityId);
+            Trace.CorrelationManager.ActivityId = activityId;
+            TraceSource.TraceEvent(TraceEventType.Start, 0, activityName);
+
+            return new ActivityScope(TraceSource, prev, 0, activityName);
+        }
+
+        /// <summary>
+        /// Emits <see cref="TraceEventType.Start"/> and returns a scope of the
+        /// activity.
+        /// </summary>
+        /// <param name="activityName">The name of the activity to start</param>
+        /// <returns>A scope of the new activity, dispose it to emit
+        /// <see cref="TraceEventType.Stop"/> for the current activity.</returns>
+        public static ActivityScope StartActivity(string activityName) {
+            if (Trace.CorrelationManager.ActivityId == Guid.Empty)
+                Trace.CorrelationManager.ActivityId = Guid.NewGuid();
+
+            var prev = Trace.CorrelationManager.ActivityId;
+            
+            TraceSource.TraceEvent(TraceEventType.Start, 0, activityName);
+            return new ActivityScope(TraceSource, prev, 0, activityName);
+        }
+
+        /// <summary>
+        /// Creates new <see cref="LogicalOperation(string)"/> and calls
+        /// to <see cref="CorrelationManager.StartLogicalOperation(object)"/>
+        /// passing the created operation as identity. Calls
+        /// <see cref="TraceSource.TraceData(TraceEventType, int, object)"/>
+        /// to notify listeners on operation start.
+        /// </summary>
+        /// <param name="name">The name of the logical operation.</param>
+        /// <returns>Logical operation scope, disposing it will stop
+        /// logical operation and notify trace listeners.</returns>
+        public static LogicalOperationScope LogicalOperation(string name) {
+            var operation = new LogicalOperation(name);
+            TraceSource.TraceData(TraceEventType.Information, TraceEventCodes.StartLogicalOperation, operation);
+            Trace.CorrelationManager.StartLogicalOperation(operation);
+            return new LogicalOperationScope(TraceSource, operation);
         }
     }
 }
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/Implab/Diagnostics/TraceEventCodes.cs	Mon Feb 12 17:03:37 2018 +0300
@@ -0,0 +1,10 @@
+namespace Implab.Diagnostics {
+    public class TraceEventCodes {
+        public const int EventCodesBase = 1024;
+      
+        public const int StartLogicalOperation = EventCodesBase + 1;
+
+        public const int StopLogicalOperation = EventCodesBase + 2;
+
+    }
+}
\ No newline at end of file
--- a/Implab/Implab.csproj	Mon Feb 12 07:24:31 2018 +0300
+++ b/Implab/Implab.csproj	Mon Feb 12 17:03:37 2018 +0300
@@ -1,6 +1,16 @@
 <Project Sdk="Microsoft.NET.Sdk">
 
   <PropertyGroup>
+    <Authors>Sergey Smirnov</Authors>
+    <Title>Implab library</Title>
+    <Description>Provides some helper clesses like XML serialization helpers, JSON XML reader,
+    JSON pull-parser, ECMA-style promises, lightweight synchonization routines Signal
+    and SharedLock, Trace helpers on top of System.Diagnostics, ObjectPool etc.
+    </Description>
+    <Copyright>2012-2018 Sergey Smirnov</Copyright>
+    <LicenseUrl>https://opensource.org/licenses/BSD-2-Clause</LicenseUrl>
+    <ProjectUrl>https://implab.org</ProjectUrl>
+    <RepositoryUrl>https://hg.implab.org/pub/ImplabNet/</RepositoryUrl>
     <TargetFrameworks>netstandard2.0;net45</TargetFrameworks>
     <FrameworkPathOverride Condition="'$(TargetFramework)'=='net45' and '$(OSTYPE)'=='linux'">/usr/lib/mono/4.5/</FrameworkPathOverride>
   </PropertyGroup>
--- a/Implab/ValueEventArgs.cs	Mon Feb 12 07:24:31 2018 +0300
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,21 +0,0 @@
-using System;
-using System.Collections.Generic;
-using System.Linq;
-using System.Text;
-
-namespace Implab
-{
-    [Serializable]
-    public class ValueEventArgs<T>: EventArgs
-    {
-        public ValueEventArgs(T value)
-        {
-            this.Value = value;
-        }
-        public T Value
-        {
-            get;
-            private set;
-        }
-    }
-}