2020-05-30 00:34:53 +02:00
|
|
|
// Copyright The OpenTelemetry 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.
|
|
|
|
|
2020-11-16 19:30:54 +02:00
|
|
|
package otel
|
2020-05-30 00:34:53 +02:00
|
|
|
|
|
|
|
import (
|
2020-06-01 19:38:05 +02:00
|
|
|
"bytes"
|
2020-05-30 00:34:53 +02:00
|
|
|
"errors"
|
2021-08-06 19:05:32 +02:00
|
|
|
"io/ioutil"
|
2020-06-01 19:38:05 +02:00
|
|
|
"log"
|
2021-11-29 18:27:59 +02:00
|
|
|
"os"
|
2020-05-30 00:34:53 +02:00
|
|
|
"testing"
|
|
|
|
|
|
|
|
"github.com/stretchr/testify/suite"
|
|
|
|
)
|
|
|
|
|
2021-11-29 18:27:59 +02:00
|
|
|
type testErrCatcher []string
|
2020-06-01 19:38:05 +02:00
|
|
|
|
2021-11-29 18:27:59 +02:00
|
|
|
func (l *testErrCatcher) Write(p []byte) (int, error) {
|
2020-06-01 19:38:05 +02:00
|
|
|
msg := bytes.TrimRight(p, "\n")
|
|
|
|
(*l) = append(*l, string(msg))
|
|
|
|
return len(msg), nil
|
|
|
|
}
|
|
|
|
|
2021-11-29 18:27:59 +02:00
|
|
|
func (l *testErrCatcher) Reset() {
|
|
|
|
*l = testErrCatcher([]string{})
|
2020-06-01 19:38:05 +02:00
|
|
|
}
|
2020-05-30 00:34:53 +02:00
|
|
|
|
2021-11-29 18:27:59 +02:00
|
|
|
func (l *testErrCatcher) Got() []string {
|
2020-06-01 19:38:05 +02:00
|
|
|
return []string(*l)
|
2020-05-30 00:34:53 +02:00
|
|
|
}
|
|
|
|
|
2021-08-06 19:05:32 +02:00
|
|
|
func causeErr(text string) {
|
|
|
|
Handle(errors.New(text))
|
|
|
|
}
|
|
|
|
|
2020-05-30 00:34:53 +02:00
|
|
|
type HandlerTestSuite struct {
|
|
|
|
suite.Suite
|
|
|
|
|
2021-08-06 19:05:32 +02:00
|
|
|
origHandler ErrorHandler
|
2021-11-29 18:27:59 +02:00
|
|
|
errCatcher *testErrCatcher
|
2020-05-30 00:34:53 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func (s *HandlerTestSuite) SetupSuite() {
|
2021-11-29 18:27:59 +02:00
|
|
|
s.errCatcher = new(testErrCatcher)
|
|
|
|
s.origHandler = globalErrorHandler.eh
|
2021-08-06 19:05:32 +02:00
|
|
|
|
2021-11-29 18:27:59 +02:00
|
|
|
globalErrorHandler.setDelegate(&errLogger{l: log.New(s.errCatcher, "", 0)})
|
2020-05-30 00:34:53 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func (s *HandlerTestSuite) TearDownSuite() {
|
2021-11-29 18:27:59 +02:00
|
|
|
globalErrorHandler.setDelegate(s.origHandler)
|
2020-05-30 00:34:53 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func (s *HandlerTestSuite) SetupTest() {
|
2021-11-29 18:27:59 +02:00
|
|
|
s.errCatcher.Reset()
|
2020-05-30 00:34:53 +02:00
|
|
|
}
|
|
|
|
|
2021-08-06 19:05:32 +02:00
|
|
|
func (s *HandlerTestSuite) TearDownTest() {
|
2021-11-29 18:27:59 +02:00
|
|
|
globalErrorHandler.setDelegate(&errLogger{l: log.New(s.errCatcher, "", 0)})
|
2021-08-06 19:05:32 +02:00
|
|
|
}
|
|
|
|
|
2020-06-01 18:28:27 +02:00
|
|
|
func (s *HandlerTestSuite) TestGlobalHandler() {
|
2020-06-01 19:38:05 +02:00
|
|
|
errs := []string{"one", "two"}
|
2020-11-16 19:30:54 +02:00
|
|
|
GetErrorHandler().Handle(errors.New(errs[0]))
|
2020-06-01 19:38:05 +02:00
|
|
|
Handle(errors.New(errs[1]))
|
2021-11-29 18:27:59 +02:00
|
|
|
s.Assert().Equal(errs, s.errCatcher.Got())
|
2020-05-30 00:34:53 +02:00
|
|
|
}
|
|
|
|
|
2021-08-06 19:05:32 +02:00
|
|
|
func (s *HandlerTestSuite) TestDelegatedHandler() {
|
|
|
|
eh := GetErrorHandler()
|
|
|
|
|
2021-11-29 18:27:59 +02:00
|
|
|
newErrLogger := new(testErrCatcher)
|
|
|
|
SetErrorHandler(&errLogger{l: log.New(newErrLogger, "", 0)})
|
2021-08-06 19:05:32 +02:00
|
|
|
|
|
|
|
errs := []string{"TestDelegatedHandler"}
|
|
|
|
eh.Handle(errors.New(errs[0]))
|
|
|
|
s.Assert().Equal(errs, newErrLogger.Got())
|
|
|
|
}
|
|
|
|
|
2020-05-30 00:34:53 +02:00
|
|
|
func (s *HandlerTestSuite) TestNoDropsOnDelegate() {
|
2021-08-06 19:05:32 +02:00
|
|
|
causeErr("")
|
2021-11-29 18:27:59 +02:00
|
|
|
s.Require().Len(s.errCatcher.Got(), 1)
|
2020-05-30 00:34:53 +02:00
|
|
|
|
|
|
|
// Change to another Handler. We are testing this is loss-less.
|
2021-11-29 18:27:59 +02:00
|
|
|
newErrLogger := new(testErrCatcher)
|
|
|
|
secondary := &errLogger{
|
2020-06-01 19:38:05 +02:00
|
|
|
l: log.New(newErrLogger, "", 0),
|
|
|
|
}
|
2020-08-05 19:31:42 +02:00
|
|
|
SetErrorHandler(secondary)
|
2020-05-30 00:34:53 +02:00
|
|
|
|
2021-08-06 19:05:32 +02:00
|
|
|
causeErr("")
|
2021-11-29 18:27:59 +02:00
|
|
|
s.Assert().Len(s.errCatcher.Got(), 1, "original Handler used after delegation")
|
2021-04-21 21:17:02 +02:00
|
|
|
s.Assert().Len(newErrLogger.Got(), 1, "new Handler not used after delegation")
|
2020-05-30 00:34:53 +02:00
|
|
|
}
|
|
|
|
|
2021-08-06 19:05:32 +02:00
|
|
|
func (s *HandlerTestSuite) TestAllowMultipleSets() {
|
2021-11-29 18:27:59 +02:00
|
|
|
notUsed := new(testErrCatcher)
|
2021-08-06 19:05:32 +02:00
|
|
|
|
2021-11-29 18:27:59 +02:00
|
|
|
secondary := &errLogger{l: log.New(notUsed, "", 0)}
|
2021-08-06 19:05:32 +02:00
|
|
|
SetErrorHandler(secondary)
|
2021-11-29 18:27:59 +02:00
|
|
|
s.Require().Same(GetErrorHandler(), globalErrorHandler, "set changed globalErrorHandler")
|
|
|
|
s.Require().Same(globalErrorHandler.eh, secondary, "new Handler not set")
|
2021-08-06 19:05:32 +02:00
|
|
|
|
2021-11-29 18:27:59 +02:00
|
|
|
tertiary := &errLogger{l: log.New(notUsed, "", 0)}
|
2021-08-06 19:05:32 +02:00
|
|
|
SetErrorHandler(tertiary)
|
2021-11-29 18:27:59 +02:00
|
|
|
s.Require().Same(GetErrorHandler(), globalErrorHandler, "set changed globalErrorHandler")
|
|
|
|
s.Assert().Same(globalErrorHandler.eh, tertiary, "user Handler not overridden")
|
2021-08-06 19:05:32 +02:00
|
|
|
}
|
|
|
|
|
2020-05-30 00:34:53 +02:00
|
|
|
func TestHandlerTestSuite(t *testing.T) {
|
|
|
|
suite.Run(t, new(HandlerTestSuite))
|
|
|
|
}
|
2021-08-06 19:05:32 +02:00
|
|
|
|
2021-11-29 18:27:59 +02:00
|
|
|
func TestHandlerRace(t *testing.T) {
|
|
|
|
go SetErrorHandler(&errLogger{log.New(os.Stderr, "", 0)})
|
2022-05-19 22:15:07 +02:00
|
|
|
go Handle(errors.New("error"))
|
2021-11-29 18:27:59 +02:00
|
|
|
}
|
|
|
|
|
2021-08-06 19:05:32 +02:00
|
|
|
func BenchmarkErrorHandler(b *testing.B) {
|
2021-11-29 18:27:59 +02:00
|
|
|
primary := &errLogger{l: log.New(ioutil.Discard, "", 0)}
|
|
|
|
secondary := &errLogger{l: log.New(ioutil.Discard, "", 0)}
|
|
|
|
tertiary := &errLogger{l: log.New(ioutil.Discard, "", 0)}
|
2021-08-06 19:05:32 +02:00
|
|
|
|
2021-11-29 18:27:59 +02:00
|
|
|
globalErrorHandler.setDelegate(primary)
|
2021-08-06 19:05:32 +02:00
|
|
|
|
2022-05-19 22:15:07 +02:00
|
|
|
err := errors.New("benchmark error handler")
|
2021-08-06 19:05:32 +02:00
|
|
|
|
|
|
|
b.ReportAllocs()
|
|
|
|
b.ResetTimer()
|
|
|
|
for i := 0; i < b.N; i++ {
|
|
|
|
GetErrorHandler().Handle(err)
|
|
|
|
Handle(err)
|
|
|
|
|
|
|
|
SetErrorHandler(secondary)
|
|
|
|
GetErrorHandler().Handle(err)
|
|
|
|
Handle(err)
|
|
|
|
|
|
|
|
SetErrorHandler(tertiary)
|
|
|
|
GetErrorHandler().Handle(err)
|
|
|
|
Handle(err)
|
|
|
|
|
2021-11-29 18:27:59 +02:00
|
|
|
globalErrorHandler.setDelegate(primary)
|
2021-08-06 19:05:32 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
reset()
|
|
|
|
}
|
|
|
|
|
|
|
|
var eh ErrorHandler
|
|
|
|
|
|
|
|
func BenchmarkGetDefaultErrorHandler(b *testing.B) {
|
|
|
|
b.ReportAllocs()
|
|
|
|
for i := 0; i < b.N; i++ {
|
|
|
|
eh = GetErrorHandler()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func BenchmarkGetDelegatedErrorHandler(b *testing.B) {
|
2021-11-29 18:27:59 +02:00
|
|
|
SetErrorHandler(&errLogger{l: log.New(ioutil.Discard, "", 0)})
|
2021-08-06 19:05:32 +02:00
|
|
|
|
|
|
|
b.ReportAllocs()
|
|
|
|
b.ResetTimer()
|
|
|
|
for i := 0; i < b.N; i++ {
|
|
|
|
eh = GetErrorHandler()
|
|
|
|
}
|
|
|
|
|
|
|
|
reset()
|
|
|
|
}
|
|
|
|
|
|
|
|
func BenchmarkDefaultErrorHandlerHandle(b *testing.B) {
|
2021-11-29 18:27:59 +02:00
|
|
|
globalErrorHandler.setDelegate(
|
|
|
|
&errLogger{l: log.New(ioutil.Discard, "", 0)},
|
|
|
|
)
|
2021-08-06 19:05:32 +02:00
|
|
|
|
|
|
|
eh := GetErrorHandler()
|
2022-05-19 22:15:07 +02:00
|
|
|
err := errors.New("benchmark default error handler handle")
|
2021-08-06 19:05:32 +02:00
|
|
|
|
|
|
|
b.ReportAllocs()
|
|
|
|
b.ResetTimer()
|
|
|
|
for i := 0; i < b.N; i++ {
|
|
|
|
eh.Handle(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
reset()
|
|
|
|
}
|
|
|
|
|
|
|
|
func BenchmarkDelegatedErrorHandlerHandle(b *testing.B) {
|
|
|
|
eh := GetErrorHandler()
|
2021-11-29 18:27:59 +02:00
|
|
|
SetErrorHandler(&errLogger{l: log.New(ioutil.Discard, "", 0)})
|
2022-05-19 22:15:07 +02:00
|
|
|
err := errors.New("benchmark delegated error handler handle")
|
2021-08-06 19:05:32 +02:00
|
|
|
|
|
|
|
b.ReportAllocs()
|
|
|
|
b.ResetTimer()
|
|
|
|
for i := 0; i < b.N; i++ {
|
|
|
|
eh.Handle(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
reset()
|
|
|
|
}
|
|
|
|
|
|
|
|
func BenchmarkSetErrorHandlerDelegation(b *testing.B) {
|
2021-11-29 18:27:59 +02:00
|
|
|
alt := &errLogger{l: log.New(ioutil.Discard, "", 0)}
|
2021-08-06 19:05:32 +02:00
|
|
|
|
|
|
|
b.ReportAllocs()
|
|
|
|
b.ResetTimer()
|
|
|
|
for i := 0; i < b.N; i++ {
|
|
|
|
SetErrorHandler(alt)
|
|
|
|
|
|
|
|
reset()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func reset() {
|
|
|
|
globalErrorHandler = defaultErrorHandler()
|
|
|
|
}
|