diff --git a/ddtrace/contrib/bottle/trace.py b/ddtrace/contrib/bottle/trace.py index 36e678be293..c67595ca3bd 100644 --- a/ddtrace/contrib/bottle/trace.py +++ b/ddtrace/contrib/bottle/trace.py @@ -1,5 +1,5 @@ # 3p -from bottle import response, request +from bottle import response, request, HTTPError # stdlib import ddtrace @@ -47,6 +47,11 @@ def wrapped(*args, **kwargs): code = 0 try: return callback(*args, **kwargs) + except HTTPError as e: + # you can interrupt flows using abort(status_code, 'message')... + # we need to respect the defined status_code. + code = e.status_code + raise except Exception: # bottle doesn't always translate unhandled exceptions, so # we mark it here. diff --git a/tests/contrib/bottle/test.py b/tests/contrib/bottle/test.py index bf3eb80f6be..079c7b93ec7 100644 --- a/tests/contrib/bottle/test.py +++ b/tests/contrib/bottle/test.py @@ -94,6 +94,29 @@ def hi(): assert s.get_tag('http.method') == 'GET' assert s.get_tag(http.URL) == 'http://localhost:80/hi' + def test_abort(self): + @self.app.route('/hi') + def hi(): + raise bottle.abort(420, 'Enhance Your Calm') + self._trace_app(self.tracer) + + # make a request + try: + resp = self.app.get('/hi') + assert resp.status_int == 420 + except Exception: + pass + + spans = self.tracer.writer.pop() + assert len(spans) == 1 + s = spans[0] + assert s.name == 'bottle.request' + assert s.service == 'bottle-app' + assert s.resource == 'GET /hi' + assert s.get_tag('http.status_code') == '420' + assert s.get_tag('http.method') == 'GET' + assert s.get_tag(http.URL) == 'http://localhost:80/hi' + def test_bottle_global_tracer(self): # without providing a Tracer instance, it should work @self.app.route('/home/')