~/posts/3leq2g4jgbo2w
cd ..

OpenTracing for ASP.NET MVC and WebAPI

3 min read

Preface - I really like what Microsoft is doing with .NET Core and ASP.NET Core.

However, the horror they've unleashed upon the world in the form of ASP.NET MVC and WebAPI is a sin that will take more than a few moons to wash away. That said, quite a few people are still building software using this stuff and I got curious how you'd do instrumentation of it via OpenTracing. This post is the result of several hours of hacking towards that end.

Action Filters For Fun And Profit

It's actually pretty straightforward, assuming you know what to Google and can handle the absolute state of documentation that's available. At a high level, here's how it works. ASP.NET - similar to Java Servlets - provides Action Filters which are simple lifecycle hooks into the HTTP request pipeline. There's four interfaces you can target if you want to be more specific, but a fairly trivial implementation of a Logger can be done like so:

public class CustomLogger : ActionFilterAttribute
{
public override void OnActionExecuting(ActionExecutingContext filterContext)
{
Debug.WriteLine($"executing controller: {filterContext.RouteData.Values["controller"]}");
// etc etc...
}

public ovveride void OnResultExecuted(ActionExecutingContext filterContext)
{
Debug.WriteLine($"result complete in controller: {filterContext.RouteData.Values["controller"]}");
// etc etc...
}
}
public class CustomLogger : ActionFilterAttribute
{
public override void OnActionExecuting(ActionExecutingContext filterContext)
{
Debug.WriteLine($"executing controller: {filterContext.RouteData.Values["controller"]}");
// etc etc...
}

public ovveride void OnResultExecuted(ActionExecutingContext filterContext)
{
Debug.WriteLine($"result complete in controller: {filterContext.RouteData.Values["controller"]}");
// etc etc...
}
}

Pretty straightforward, like I said. There's also OnActionExecuted and OnResultExecuted which are called after and before a controller action, and controller action result, respectively.

So you'd think it'd be pretty easy, right? OpenTracing provides a handy GlobalTracer singleton, so create a TracingFilter...

public class TracingFilter : ActionFilterAttribute
{
public override void OnActionExecuting(ActionExecutingContext filterContext)
{
var routeValues = filterContext.RouteData.Values;
var scope = GlobalTracer.Instance.BuildSpan($"{routeValues["controller"]}").StartActive();
scope.Span.SetTag("action", routeValues["action"].ToString());
}

public override void OnResultExecuted(ResultExecutedContext filterContext)
{
var scope = GlobalTracer.Instance.ScopeManager.Active;
scope.Span.Finish();
}
}
public class TracingFilter : ActionFilterAttribute
{
public override void OnActionExecuting(ActionExecutingContext filterContext)
{
var routeValues = filterContext.RouteData.Values;
var scope = GlobalTracer.Instance.BuildSpan($"{routeValues["controller"]}").StartActive();
scope.Span.SetTag("action", routeValues["action"].ToString());
}

public override void OnResultExecuted(ResultExecutedContext filterContext)
{
var scope = GlobalTracer.Instance.ScopeManager.Active;
scope.Span.Finish();
}
}

Then in your RegisterGlobalFilters method, do a quick filters.Add(new TracingFilter()), register a Tracer, and away you go! Right?

Wrong.

Well, half-right.

That Sounds Like Me, Yeah.

Assuming you're only using MVC, you're right. So you'll see spans for, say, GETting your index page, but not for any of your API routes. Why? Because there's two ActionFilterAttributes. The one we just did is System.Web.Mvc.ActionFilterAttribute. Want your WebAPI traced too? Time to create a System.Web.Http.Filters.ActionFilterAttribute. You can tell them apart by the extremely different method signatures, as seen here -

public class WebApiTracingFilter : ActionFilterAttribute
{
public override void OnActionExecuting(HttpActionContext actionContext)
{
var scope = GlobalTracer.Instance.BuildSpan(actionContext.ControllerContext.ControllerDescriptor.ControllerName).StartActive();
scope.Span.SetTag("action", actionContext.ActionDescriptor.ActionName);
}

public override void OnActionExecuted(HttpActionExecutedContext actionExecutedContext)
{
var scope = GlobalTracer.Instance.ScopeManager.Active;
scope.Span.Finish();
}
}
public class WebApiTracingFilter : ActionFilterAttribute
{
public override void OnActionExecuting(HttpActionContext actionContext)
{
var scope = GlobalTracer.Instance.BuildSpan(actionContext.ControllerContext.ControllerDescriptor.ControllerName).StartActive();
scope.Span.SetTag("action", actionContext.ActionDescriptor.ActionName);
}

public override void OnActionExecuted(HttpActionExecutedContext actionExecutedContext)
{
var scope = GlobalTracer.Instance.ScopeManager.Active;
scope.Span.Finish();
}
}

Yeah, that took me a few minutes and this StackOverflow answer to puzzle out. c'est la vie.

That said, this is pretty much the hard part. Since you've got spans being automagically started and finished whenever the request pipeline hits, you can implicitly utilize those parent spans inside a controller to create children:

[WebApiTracingFilter]
public class ValuesController : ApiController
{
public IEnumerable<string> Get()
{
var returnValue = getCurrentTime();
return new string[] { returnValue };
}

private string getCurrentTime()
{
using (var scope = GlobalTracer.Instance.BuildSpan("getCurrentTime").StartActive())
{
return DateTime.Now.ToShortDateString();
}
}

// and so forth...
}
[WebApiTracingFilter]
public class ValuesController : ApiController
{
public IEnumerable<string> Get()
{
var returnValue = getCurrentTime();
return new string[] { returnValue };
}

private string getCurrentTime()
{
using (var scope = GlobalTracer.Instance.BuildSpan("getCurrentTime").StartActive())
{
return DateTime.Now.ToShortDateString();
}
}

// and so forth...
}

You can also get fancy with your OnActionExecuted/OnResultExecuted filters by checking for exceptions coming in and adding stack traces to your span logs.

If you'd like to check out the complete sample project I made, it's on GitHub.

$date -r October 13, 2018