Skip to content

Latest commit

 

History

History
158 lines (114 loc) · 8.1 KB

README.md

File metadata and controls

158 lines (114 loc) · 8.1 KB

Zipkin.net Instrumentations

Zipkin is a distributed logging framework, that allows to analyze the behavior of complex applications, specially those based in microservices, with access to different databases from different components. This project provides different Zipkin instrumentations for common use .NET frameworks based on the Zipkin.net library. All the instrumentations allow automatic logging to Zipkin of different parts of your application activity, according to your framework of choice:

  • Owin: automatic logging of every incoming HTTP request to an Owin web interface.
  • Ado: automatic logging of every database query.
  • NancyFx: automatic logging of every incoming HTTP request to a NancyFx web application.
  • RestSharp: automatic logging of every outgoing HTTP request via the RestSharp library.

Setting Up The Logging Into Zipkin

The Zipkin instrumentations included in this project are based on the Zipkin.net library. In order to start logging traces into Zipkin you have to create a Zipkin bootstrapper instance and start it (as explained in the Zipkin.net library documentation):

	new Zipkin.ZipkinBootstrapper("my-service")
            .ZipkinAt("localhost") // where's the zipkin server?
            .WithSampleRate(1.0) // 0.1 means trace 10% of calls
            .Start(); 
}

If your application is an Owin web application and you are using the Zipkin4Owin library, you don't need to manually start the ZipkinBootstrapper, because Zipkin4Owin already does it for you. For the rest of instrumentations, you need to manually start the Zipkin boostrapper.

Logging Owin HTTP Requests Into Zipkin

In order to automatically log all the HTTP requests received by your Owin-based web application you have to reference the Zipkin4Owin Nuget package, Zipkin4Owin, in your project and setup the Zipkin4Owin Owin middleware during the Owin start-up via the UseZipkinTracer extension method:

using Zipkin4Owin;

...

public void Configuration(IAppBuilder app)
{
    if (zipkinEnabled)
    {
        app.UseZipkinTracer(new Zipkin.ZipkinBootstrapper("my-service")
            .ZipkinAt("localhost") // where's the zipkin server?
            .WithSampleRate(1.0)); // 0.1 means trace 10% of calls
    }

    app.Use...
}

As you can notice in the code, the Zipkin4Owin Owin middleware accepts a ZipkinBootstrapper, which carries the configuration for the Zipkin logger to start. Zipkin4Owin starts this bootstrapper for you at start up, so you don't need to do anything else in order to start logging messages.

Logging Database Queries Into Zipkin

In order to automatically log all your ADO.Net database queries into Zipkin you have to reference the Zipkin4Ado Nuget package, Zipkin4Ado, in your project and wrap your database connection (DbConnection) into a ZipkinDbConnection.

DbConnection connection = new SqlConnection(_connectionString);
if (zipkinEnabled)
	connection = new Zipkin4Ado.ZipkinDbConnection(connection);

Having the database connection wrapped into a ZipkinDbConnection it is possible to create and execute database commands, that are automatically logged into Zipkin:

var dbCommand = connection.CreateCommand();
dbCommand.CommandText = "SELECT 1";
dbCommand.Execute();

Be careful to create the database commands via the wrapped ZipkinDbConnection because otherwise they will not be logged into Zipkin:

var dbCommand = new SqlCommand("SELECT 1")
dbCommand.Connection = connection;
dbCommand.ExecuteReader(); // THIS WILL NOT BE LOGGED

In order to start up the Zipkin logging you need to start and configure the Zipkin bootstrapper somewhere at the start of your application, as documented in the Zipkin.net library:

new Zipkin.ZipkinBootstrapper("my-service")
			.ZipkinAt("localhost") // where's the zipkin server?
			.WithSampleRate(0.1)   // 0.1 means trace 10% of calls
			.Start();

The idea of Zipkin4Ado is based on the amazing project Glimpse, which also has a database connection wrapper for automatically logging database activity. It is compatible with micro-ORMs like Dapper, PetaPoco or NPoco.

Logging NancyFx HTTP Requests Into Zipkin

In order to automatically log all the HTTP requests received by your NancyFx web application you have to reference the Zipkin4Nancy Nuget package, Zipkin4Nancy, create an instance of a ZipkinNancyTracer and invoke the tracer in your BeforeRequest, AfterRequest and OnError pipelines at your NancyFx bootstrapper:

using Zipkin4Nancy;

...
	var zipkinNancyTracer = new ZipkinNancyTracer();
    pipelines.OnError += (ctx, ex) =>
    {
        ...
        zipkinNancyTracer.ZipkinTraceOnError(ctx, ex);
		...
	};

    pipelines.BeforeRequest.AddItemToEndOfPipeline((ctx) =>
    {
        ...
	    zipkinNancyTracer.ZipkinTraceBeforeRequest(ctx);
	    ...
    });

    pipelines.AfterRequest.AddItemToEndOfPipeline((ctx) =>
    {
        ...
        zipkinNancyTracer.ZipkinTraceAfterRequest(ctx);
    });
}

This will register all the incoming HTTP requests, including the HTTP URL, HTTP method, host and response status code.

For Nancy 1.4.x you should depend on the Zipkin4Nancy 1.4.x package. Nancy 2.x is not yet supported.

Logging RestSharp HTTP Requests Into Zipkin

The Zipkin instrumentation for RestSharp serves 2 purposes:

  • Automatically logging every request made via RestSharp into Zipkin. Currently logging is only available in synchronous operations: Execute, ExecuteAsGet, etc.
  • Propagating the Zipkin traces information to the callee via standard Zipkin B3 HTTP headers.

In order to start logging and propagating Zipkin trace information in your RestSharp calls you have to reference the Zipkin4RestSharp Nuget package, Zipkin4RestSharp, in your project and wrap you IRestClient into a ZipkinRestClient instance:

using Zipkin4RestSharp;

...
    var restClient = new RestClient("http://example.com");
    if (zipkinEnabled)
    {
        restClient = new ZipkinRestClient(restClient);
    }

    var request = new RestRequest("resource/id", Method.GET);
    IRestResponse response = client.Execute(request);

As in the case of Zipkin4Owin, Zipkin4RestSharp uses the standard Zipkin B3 HTTP headers for propagating the Zipkin traces information to the callee, in order to co-relate logs from different services taking part in the processing of the same request. This way, every HTTP request performed by via ZipkinRestClient will be injected the X-B3-TraceId, X-B3-SpanId and X-B3-ParentSpanId headers containing information of the current Zipkin trace.

Trace Propagation

In order to co-relate requests between different services in the processing of a single requests to your system, Zipkin has defined a series of standard HTTP headers (B3 headers). Zipkin4Owin and Zipkin4Nancy automatically detect trace information in those HTTP request headers so that:

  • If the appropriate headers are provided, the logged HTTP request will be logged as a child of the specified span, part of the same trace.
  • If no headers are provided, the logged HTTP request will be a new span and trace.

In addition, Zipkin4Restshap will detect already existing trace and span identifiers, and will include it into B3 headers, in order to propagate the trace and span information to the destination.

The following are the expected standard Zipkin B3 HTTP headers:

  • X-B3-TraceId
  • X-B3-SpanId
  • X-B3-ParentSpanId

If you are not familiar with the concept of a trace, span and parent span, it is advisable to have a look at http://zipkin.io.