org.springframework.cloud.sleuth.instrument.web.TraceRestTemplateInterceptorTests.java Source code

Java tutorial

Introduction

Here is the source code for org.springframework.cloud.sleuth.instrument.web.TraceRestTemplateInterceptorTests.java

Source

/*
 * Copyright 2013-2018 the original author or authors.
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *      http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

package org.springframework.cloud.sleuth.instrument.web;

import java.util.Arrays;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

import brave.Span;
import brave.Tracer;
import brave.Tracing;
import brave.http.HttpTracing;
import brave.sampler.Sampler;
import brave.propagation.StrictScopeDecorator;
import brave.propagation.ThreadLocalCurrentTraceContext;
import brave.spring.web.TracingClientHttpRequestInterceptor;
import org.apache.commons.lang3.StringUtils;
import org.junit.After;
import org.junit.Assert;
import org.junit.Before;
import org.junit.Test;
import org.springframework.cloud.sleuth.util.ArrayListSpanReporter;
import org.springframework.cloud.sleuth.util.SpanUtil;
import org.springframework.http.HttpHeaders;
import org.springframework.http.client.ClientHttpRequestInterceptor;
import org.springframework.test.web.client.MockMvcClientHttpRequestFactory;
import org.springframework.test.web.servlet.MockMvc;
import org.springframework.test.web.servlet.setup.MockMvcBuilders;
import org.springframework.web.bind.annotation.RequestHeader;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
import org.springframework.web.client.RestTemplate;

import static org.assertj.core.api.BDDAssertions.then;

/**
 * @author Dave Syer
 *
 */
public class TraceRestTemplateInterceptorTests {

    private TestController testController = new TestController();

    private MockMvc mockMvc = MockMvcBuilders.standaloneSetup(this.testController).build();

    private RestTemplate template = new RestTemplate(new MockMvcClientHttpRequestFactory(this.mockMvc));

    ArrayListSpanReporter reporter = new ArrayListSpanReporter();

    Tracing tracing = Tracing.newBuilder().currentTraceContext(
            ThreadLocalCurrentTraceContext.newBuilder().addScopeDecorator(StrictScopeDecorator.create()).build())
            .spanReporter(this.reporter).build();

    Tracer tracer = this.tracing.tracer();

    TraceKeys traceKeys = new TraceKeys();

    @Before
    public void setup() {
        setInterceptors(HttpTracing.create(this.tracing));
    }

    private void setInterceptors(HttpTracing httpTracing) {
        this.template.setInterceptors(Arrays
                .<ClientHttpRequestInterceptor>asList(TracingClientHttpRequestInterceptor.create(httpTracing)));
    }

    @After
    public void clean() {
        Tracing.current().close();
    }

    @Test
    public void headersAddedWhenNoTracingWasPresent() {
        @SuppressWarnings("unchecked")
        Map<String, String> headers = this.template.getForEntity("/", Map.class).getBody();

        then(headers.get("X-B3-TraceId")).isNotNull();
        then(headers.get("X-B3-SpanId")).isNotNull();
    }

    @Test
    public void headersAddedWhenTracing() {
        Span span = this.tracer.nextSpan().name("new trace");
        Map<String, String> headers;

        try (Tracer.SpanInScope ws = this.tracer.withSpanInScope(span.start())) {
            headers = this.template.getForEntity("/", Map.class).getBody();
        } finally {
            span.finish();
        }

        then(headers.get("X-B3-TraceId")).isEqualTo(SpanUtil.idToHex(span.context().traceId()));
        then(headers.get("X-B3-SpanId")).isNotEqualTo(SpanUtil.idToHex(span.context().spanId()));
        then(headers.get("X-B3-ParentSpanId")).isEqualTo(SpanUtil.idToHex(span.context().spanId()));
    }

    // Issue #290
    @Test
    public void requestHeadersAddedWhenTracing() {
        setInterceptors(HttpTracing.newBuilder(this.tracing)
                .clientParser(new SleuthHttpClientParser(this.traceKeys)).build());
        Span span = this.tracer.nextSpan().name("new trace");

        try (Tracer.SpanInScope ws = this.tracer.withSpanInScope(span.start())) {
            this.template.getForEntity("/foo?a=b", Map.class);
        } finally {
            span.finish();
        }

        List<zipkin2.Span> spans = reporter.getSpans();
        then(spans).isNotEmpty();
        then(spans.get(0).tags()).containsEntry("http.url", "/foo?a=b").containsEntry("http.path", "/foo")
                .containsEntry("http.method", "GET");
    }

    @Test
    public void notSampledHeaderAddedWhenNotExportable() {
        Tracing tracing = Tracing.newBuilder()
                .currentTraceContext(ThreadLocalCurrentTraceContext.newBuilder()
                        .addScopeDecorator(StrictScopeDecorator.create()).build())
                .spanReporter(this.reporter).sampler(Sampler.NEVER_SAMPLE).build();
        this.template.setInterceptors(Arrays.<ClientHttpRequestInterceptor>asList(
                TracingClientHttpRequestInterceptor.create(HttpTracing.create(tracing))));

        Span span = tracing.tracer().nextSpan().name("new trace");
        Map<String, String> headers;

        try (Tracer.SpanInScope ws = tracing.tracer().withSpanInScope(span.start())) {
            headers = this.template.getForEntity("/", Map.class).getBody();
        } finally {
            span.finish();
        }

        then(reporter.getSpans()).isEmpty();
    }

    // issue #198
    @Test
    public void spanRemovedFromThreadUponException() {
        Span span = this.tracer.nextSpan().name("new trace");

        try (Tracer.SpanInScope ws = this.tracer.withSpanInScope(span.start())) {
            this.template.getForEntity("/exception", Map.class).getBody();
            Assert.fail("should throw an exception");
        } catch (RuntimeException e) {
            then(e).hasMessage("500 Internal Server Error");
        } finally {
            span.finish();
        }

        then(this.tracer.currentSpan()).isNull();
    }

    @Test
    public void createdSpanNameHasOnlyPrintableAsciiCharactersForNonEncodedURIWithNonAsciiChars() {
        setInterceptors(HttpTracing.newBuilder(this.tracing)
                .clientParser(new SleuthHttpClientParser(this.traceKeys)).build());
        Span span = this.tracer.nextSpan().name("new trace");

        try (Tracer.SpanInScope ws = this.tracer.withSpanInScope(span.start())) {
            this.template.getForEntity("/cas~fs~", Map.class).getBody();
        } catch (Exception e) {

        } finally {
            span.finish();
        }

        List<zipkin2.Span> spans = reporter.getSpans();
        then(spans).hasSize(2);
        String spanName = spans.get(0).name();
        then(spanName).isEqualTo("http:/cas~fs~%c3%a5%cb%86%e2%80%99");
        then(StringUtils.isAsciiPrintable(spanName));
    }

    @Test
    public void willShortenTheNameOfTheSpan() {
        setInterceptors(HttpTracing.newBuilder(this.tracing)
                .clientParser(new SleuthHttpClientParser(this.traceKeys)).build());
        Span span = this.tracer.nextSpan().name("new trace");

        try (Tracer.SpanInScope ws = this.tracer.withSpanInScope(span.start())) {
            this.template.getForEntity("/" + bigName(), Map.class).getBody();
        } catch (Exception e) {

        } finally {
            span.finish();
        }

        List<zipkin2.Span> spans = reporter.getSpans();
        then(spans).isNotEmpty();
        String spanName = spans.get(0).name();
        then(spanName).hasSize(50);
        then(StringUtils.isAsciiPrintable(spanName));
    }

    private String bigName() {
        StringBuilder sb = new StringBuilder();
        for (int i = 0; i < 60; i++) {
            sb.append("a");
        }
        return sb.toString();
    }

    @RestController
    public class TestController {

        Span span;

        @RequestMapping("/")
        public Map<String, String> home(@RequestHeader HttpHeaders headers) {
            this.span = TraceRestTemplateInterceptorTests.this.tracer.currentSpan();
            Map<String, String> map = new HashMap<String, String>();
            addHeaders(map, headers, "X-B3-SpanId", "X-B3-TraceId", "X-B3-ParentSpanId");
            return map;
        }

        @RequestMapping("/foo")
        public void foo() {
        }

        @RequestMapping("/exception")
        public Map<String, String> exception() {
            throw new RuntimeException("foo");
        }

        private void addHeaders(Map<String, String> map, HttpHeaders headers, String... names) {
            if (headers != null) {
                for (String name : names) {
                    String value = headers.getFirst(name);
                    if (value != null) {
                        map.put(name, value);
                    }
                }
            }
        }

    }

}