diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index 3f20bf5..caa32eb 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -6,30 +6,11 @@ jobs: runs-on: ubuntu-latest strategy: matrix: - go: [ '1.15', '1.14' ] + go: [ '1.23', '1.22', '1.21' ] name: Go ${{ matrix.go }} test steps: - - name: Checkout repository - uses: actions/checkout@v2 - - name: Install Go stable version - if: matrix.go != 'tip' - uses: actions/setup-go@v2 - with: - go-version: ${{ matrix.go }} - - name: Install Go tip - if: matrix.go == 'tip' - run: | - git clone --depth=1 https://go.googlesource.com/go $HOME/gotip - cd $HOME/gotip/src - ./make.bash - echo "::set-env name=GOROOT::$HOME/gotip" - echo "::add-path::$HOME/gotip/bin" - echo "::add-path::$(go env GOPATH)/bin" + - uses: actions/checkout@692973e3d937129bcbf40652eb9f2f61becf3332 # v4.1.7 + - uses: actions/setup-go@0a12ed9d6a96ab950c8f026ed9f722fe0da7ef32 # v5.0.2 - name: Test run: go test -v -race ./... - - name: Upload code coverage to codecov - if: matrix.go == '1.15' - uses: codecov/codecov-action@v1 - with: - file: ./coverage.out diff --git a/.github/workflows/lint.yml b/.github/workflows/lint.yml index 9fcc424..e8794e7 100644 --- a/.github/workflows/lint.yml +++ b/.github/workflows/lint.yml @@ -1,11 +1,17 @@ name: lint -on: [push, pull_request] +on: + push: + jobs: golangci: - name: lint + name: golangci-lint runs-on: ubuntu-latest steps: - - uses: actions/checkout@v2 - - uses: golangci/golangci-lint-action@v2 + - uses: actions/checkout@692973e3d937129bcbf40652eb9f2f61becf3332 # v4.1.7 + - uses: actions/setup-go@0a12ed9d6a96ab950c8f026ed9f722fe0da7ef32 # v5.0.2 with: - version: v1.34.1 + go-version-file: "go.mod" + - name: golangci-lint + uses: golangci/golangci-lint-action@aaa42aa0628b4ae2578232a66b541047968fac86 # v6.1.0 + with: + args: "--timeout 5m0s" diff --git a/.golangci.yml b/.golangci.yml index 3a8ad73..f7f07de 100644 --- a/.golangci.yml +++ b/.golangci.yml @@ -1,19 +1,21 @@ run: - skip-dirs: internal linters-settings: govet: enable-all: true disable: - shadow + - fieldalignment linters: enable-all: true disable: + - cyclop + - depguard - dupl - exhaustive - - exhaustivestruct - errorlint + - err113 - funlen - gci - gochecknoglobals @@ -23,30 +25,71 @@ linters: - gocyclo - godot - godox - - goerr113 - gofumpt - gomnd - gosec + - govet + - inamedparam # oh, sod off + - ireturn # No, I _LIKE_ returning interfaces - lll + - maintidx # Do this in code review - makezero + - mnd - nakedret - nestif - nlreturn + - nonamedreturns # visit this back later - paralleltest + - perfsprint + - tagliatelle + - testifylint # TODO: revisit when we have the chance - testpackage - - thelper - - unconvert + - thelper # Tests are fine + - varnamelen # Short names are ok - wrapcheck - wsl issues: exclude-rules: + # not needed + - path: /*.go + text: "ST1003: should not use underscores in package names" + linters: + - stylecheck + - path: /*.go + text: "don't use an underscore in package name" + linters: + - revive + - path: /*.go + linters: + - contextcheck + - exhaustruct + - path: /main.go + linters: + - errcheck + - path: internal/codegen/codegen.go + linters: + - errcheck + - path: internal/jwxtest/jwxtest.go + linters: + - errcheck + - errchkjson + - forcetypeassert - path: /*_test.go linters: - errcheck + - errchkjson + - forcetypeassert - path: /*_example_test.go linters: - forbidigo + - path: cmd/jwx/jwx.go + linters: + - forbidigo + - path: /*_test.go + text: "var-naming: " + litners: + - revive # Maximum issues count per one linter. Set to 0 to disable. Default is 50. max-issues-per-linter: 0 diff --git a/buffered.go b/buffered.go index 5fc6dac..e492b2e 100644 --- a/buffered.go +++ b/buffered.go @@ -11,16 +11,16 @@ import ( // NewBuffered creates a new Buffered client. // Options may be one of the following: // -// * fluent.WithAddress -// * fluent.WithBufferLimit -// * fluent.WithDialTimeout -// * fluent.WithJSONMarshaler -// * fluent.WithMaxConnAttempts -// * fluent.WithMsgpackMarshaler -// * fluent.WithNetwork -// * fluent.WithTagPrefix -// * fluent.WithWriteThreshold -// * fluent.WithWriteQueueSize +// - fluent.WithAddress +// - fluent.WithBufferLimit +// - fluent.WithDialTimeout +// - fluent.WithJSONMarshaler +// - fluent.WithMaxConnAttempts +// - fluent.WithMsgpackMarshaler +// - fluent.WithNetwork +// - fluent.WithTagPrefix +// - fluent.WithWriteThreshold +// - fluent.WithWriteQueueSize // // Please see their respective documentation for details. func NewBuffered(options ...Option) (client *Buffered, err error) { @@ -37,6 +37,7 @@ func NewBuffered(options ...Option) (client *Buffered, err error) { ctx, cancel := context.WithCancel(context.Background()) var subsecond bool + //nolint:forcetypeassert for _, opt := range options { switch opt.Ident() { case identSubSecond{}: @@ -62,17 +63,16 @@ func NewBuffered(options ...Option) (client *Buffered, err error) { // If you would like to specify options to `Post()`, you may pass them at the end of // the method. Currently you can use the following: // -// fluent.WithContext: specify context.Context to use -// fluent.WithTimestamp: allows you to set arbitrary timestamp values -// fluent.WithSyncAppend: allows you to verify if the append was successful +// fluent.WithContext: specify context.Context to use +// fluent.WithTimestamp: allows you to set arbitrary timestamp values +// fluent.WithSyncAppend: allows you to verify if the append was successful // // If fluent.WithSyncAppend is provide and is true, the following errors // may be returned: // -// 1. If the current underlying pending buffer is is not large enough to -// hold this new data, an error will be returned -// 2. If the marshaling into msgpack/json failed, it is returned -// +// 1. If the current underlying pending buffer is not large enough to +// hold this new data, an error will be returned +// 2. If the marshaling into msgpack/json failed, it is returned func (c *Buffered) Post(tag string, v interface{}, options ...Option) (err error) { if pdebug.Enabled { g := pdebug.Marker("fluent.Buffered.Post").BindError(&err) @@ -90,6 +90,7 @@ func (c *Buffered) Post(tag string, v interface{}, options ...Option) (err error var subsecond = c.subsecond var t time.Time var ctx = context.Background() + //nolint:forcetypeassert for _, opt := range options { switch opt.Ident() { case identTimestamp{}: @@ -102,6 +103,7 @@ func (c *Buffered) Post(tag string, v interface{}, options ...Option) (err error if pdebug.Enabled { pdebug.Printf("client: using user-supplied context") } + //nolint:fatcontext ctx = opt.Value().(context.Context) } } @@ -220,6 +222,7 @@ func (c *Buffered) Ping(tag string, record interface{}, options ...Option) (err var ctx = context.Background() var subsecond bool var t time.Time + //nolint:forcetypeassert for _, opt := range options { switch opt.Ident() { case identSubSecond{}: @@ -230,6 +233,7 @@ func (c *Buffered) Ping(tag string, record interface{}, options ...Option) (err if pdebug.Enabled { pdebug.Printf("client: using user-supplied context") } + //nolint:fatcontext ctx = opt.Value().(context.Context) } } diff --git a/errors.go b/errors.go index b41dd33..0a2b1c8 100644 --- a/errors.go +++ b/errors.go @@ -1,7 +1,8 @@ package fluent -type bufferFullErr struct{} -type bufferFuller interface { +//nolint:errname +type errBufferFull struct{} +type errBufferFuller interface { BufferFull() bool } type causer interface { @@ -9,12 +10,12 @@ type causer interface { } // Just need one instance -var bufferFullErrInstance bufferFullErr +var errBufferFullInstance errBufferFull // IsBufferFull returns true if the error is a BufferFull error func IsBufferFull(e error) bool { for e != nil { - if berr, ok := e.(bufferFuller); ok { + if berr, ok := e.(errBufferFuller); ok { return berr.BufferFull() } @@ -25,10 +26,10 @@ func IsBufferFull(e error) bool { return false } -func (e *bufferFullErr) BufferFull() bool { +func (e *errBufferFull) BufferFull() bool { return true } -func (e *bufferFullErr) Error() string { +func (e *errBufferFull) Error() string { return `buffer full` } diff --git a/fluent.go b/fluent.go index bf06331..3be5c82 100644 --- a/fluent.go +++ b/fluent.go @@ -9,6 +9,7 @@ package fluent // respectively. func New(options ...Option) (Client, error) { var buffered = true + //nolint:forcetypeassert for _, opt := range options { switch opt.Ident() { case identBuffered{}: diff --git a/fluent_example_test.go b/fluent_example_test.go index 82f22cf..4e54fbd 100644 --- a/fluent_example_test.go +++ b/fluent_example_test.go @@ -55,7 +55,7 @@ func ExamplePing() { return } - ctx, cancel := context.WithCancel(context.Background()) + ctx, cancel := context.WithTimeout(context.Background(), 5*time.Second) defer cancel() // Goroutine to wait for errors @@ -75,6 +75,7 @@ func ExamplePing() { }() go fluent.Ping(ctx, client, "ping", "hostname", fluent.WithPingResultChan(errorCh)) - // Do what you need with your main program... + + // OUTPUT: } diff --git a/fluent_test.go b/fluent_test.go index 4849684..00a45a1 100644 --- a/fluent_test.go +++ b/fluent_test.go @@ -5,7 +5,6 @@ import ( "encoding/json" "fmt" "io" - "io/ioutil" "net" "os" "path/filepath" @@ -35,7 +34,7 @@ type server struct { } func newServer(useJSON bool) (*server, error) { - dir, err := ioutil.TempDir("", "sock-") + dir, err := os.MkdirTemp("", "sock-") if err != nil { return nil, errors.Wrap(err, `failed to create temporary directory`) } @@ -252,12 +251,11 @@ func (s *server) Run(ctx context.Context) { func TestConnectOnStart(t *testing.T) { for _, buffered := range []bool{true, false} { - buffered := buffered t.Run(fmt.Sprintf("failure case, buffered=%t", buffered), func(t *testing.T) { // find a port that is not available (this may be timing dependent) var dialer net.Dialer - var port int = 22412 - for i := 0; i < 1000; i++ { + port := 22412 + for range 1000 { ctx, cancel := context.WithTimeout(context.Background(), 100*time.Millisecond) conn, err := dialer.DialContext(ctx, `net`, fmt.Sprintf(`127.0.0.1:%d`, port)) cancel() @@ -295,7 +293,6 @@ func TestConnectOnStart(t *testing.T) { <-s.Ready() for _, buffered := range []bool{true, false} { - buffered := buffered t.Run(fmt.Sprintf("normal case, buffered=%t", buffered), func(t *testing.T) { client, err := fluent.New( fluent.WithNetwork(s.Network), @@ -490,7 +487,6 @@ func (msg *badmsgpack) EncodeMsgpack(_ *msgpack.Encoder) error { func TestPostSync(t *testing.T) { for _, syncAppend := range []bool{true, false} { - syncAppend := syncAppend t.Run("sync="+strconv.FormatBool(syncAppend), func(t *testing.T) { s, err := newServer(false) if !assert.NoError(t, err, "newServer should succeed") { @@ -548,8 +544,8 @@ func TestPostSync(t *testing.T) { } type Payload struct { - Foo string `msgpack:"foo" json:"foo"` - Bar string `msgpack:"bar" json:"bar"` + Foo string `json:"foo" msgpack:"foo"` + Bar string `json:"bar" msgpack:"bar"` } func TestPostRoundtrip(t *testing.T) { @@ -562,7 +558,6 @@ func TestPostRoundtrip(t *testing.T) { } for _, buffered := range []bool{true, false} { - buffered := buffered t.Run(fmt.Sprintf("buffered=%t", buffered), func(t *testing.T) { var options []fluent.Option if !buffered { @@ -672,13 +667,12 @@ func TestPostRoundtrip(t *testing.T) { func TestPing(t *testing.T) { for _, buffered := range []bool{true, false} { - buffered := buffered t.Run(fmt.Sprintf("buffered=%t", buffered), func(t *testing.T) { t.Run("Ping with no server", func(t *testing.T) { // find a port that is not available (this may be timing dependent) var dialer net.Dialer - var port int = 22412 - for i := 0; i < 1000; i++ { + port := 22412 + for range 1000 { ctx, cancel := context.WithTimeout(context.Background(), 100*time.Millisecond) conn, err := dialer.DialContext(ctx, `net`, fmt.Sprintf(`127.0.0.1:%d`, port)) cancel() diff --git a/go.mod b/go.mod index 7445574..504cf40 100644 --- a/go.mod +++ b/go.mod @@ -1,12 +1,19 @@ module github.com/lestrrat-go/fluent-client -go 1.16 +go 1.22.7 require ( - github.com/lestrrat-go/backoff/v2 v2.0.3 + github.com/lestrrat-go/backoff/v2 v2.0.8 github.com/lestrrat-go/msgpack v1.0.0 - github.com/lestrrat-go/option v0.0.0-20210103042652-6f1ecfceda35 - github.com/lestrrat-go/pdebug v0.0.0-20200204225717-4d6bd78da58d + github.com/lestrrat-go/option v1.0.1 + github.com/lestrrat-go/pdebug v0.0.0-20210111095411-35b07dbf089b github.com/pkg/errors v0.9.1 - github.com/stretchr/testify v1.6.1 + github.com/stretchr/testify v1.7.0 +) + +require ( + github.com/davecgh/go-spew v1.1.1 // indirect + github.com/lestrrat-go/bufferpool v0.0.0-20210118235918-2deb6a84c94c // indirect + github.com/pmezard/go-difflib v1.0.0 // indirect + gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c // indirect ) diff --git a/go.sum b/go.sum index 8346162..49895df 100644 --- a/go.sum +++ b/go.sum @@ -3,14 +3,23 @@ github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= github.com/lestrrat-go/backoff/v2 v2.0.3 h1:2ABaTa5ifB1L90aoRMjaPa97p0WzzVe93Vggv8oZftw= github.com/lestrrat-go/backoff/v2 v2.0.3/go.mod h1:mU93bMXuG27/Y5erI5E9weqavpTX5qiVFZI4uXAX0xk= +github.com/lestrrat-go/backoff/v2 v2.0.8 h1:oNb5E5isby2kiro9AgdHLv5N5tint1AnDVVf2E2un5A= +github.com/lestrrat-go/backoff/v2 v2.0.8/go.mod h1:rHP/q/r9aT27n24JQLa7JhSQZCKBBOiM/uP402WwN8Y= github.com/lestrrat-go/bufferpool v0.0.0-20180220091733-e7784e1b3e37 h1:px5km9KhQGUKiPWIVZ++FErEMTd06XEuMi2OswGMrqI= github.com/lestrrat-go/bufferpool v0.0.0-20180220091733-e7784e1b3e37/go.mod h1:vs3QXw2t0jsgjLEG7JZt0uE1jcSkxnQr+5bhQ80UJHE= +github.com/lestrrat-go/bufferpool v0.0.0-20210118235918-2deb6a84c94c h1:Txi8K4WxH1BefzJqBrWIRH0JiiZlR4lUUsP+SPOqRXc= +github.com/lestrrat-go/bufferpool v0.0.0-20210118235918-2deb6a84c94c/go.mod h1:KwmXn5FpkX4is3nM+CKsCnFSqdY7ujdNpE2L/KztlDM= github.com/lestrrat-go/msgpack v1.0.0 h1:MWCEeQYr0mNML3sO5AOeIJCXpMGdT+QhznqQWkG8Drc= github.com/lestrrat-go/msgpack v1.0.0/go.mod h1:id/h8SRgREpEIyXa7ZbxuFmF2zh70FvykxZoOk5OHM8= github.com/lestrrat-go/option v0.0.0-20210103042652-6f1ecfceda35 h1:lea8Wt+1ePkVrI2/WD+NgQT5r/XsLAzxeqtyFLcEs10= github.com/lestrrat-go/option v0.0.0-20210103042652-6f1ecfceda35/go.mod h1:5ZHFbivi4xwXxhxY9XHDe2FHo6/Z7WWmtT7T5nBBp3I= +github.com/lestrrat-go/option v1.0.0/go.mod h1:5ZHFbivi4xwXxhxY9XHDe2FHo6/Z7WWmtT7T5nBBp3I= +github.com/lestrrat-go/option v1.0.1 h1:oAzP2fvZGQKWkvHa1/SAcFolBEca1oN+mQ7eooNBEYU= +github.com/lestrrat-go/option v1.0.1/go.mod h1:5ZHFbivi4xwXxhxY9XHDe2FHo6/Z7WWmtT7T5nBBp3I= github.com/lestrrat-go/pdebug v0.0.0-20200204225717-4d6bd78da58d h1:aEZT3f1GGg5RIlHMAy4/4fe4ciOi3SCwYoaURphcB4k= github.com/lestrrat-go/pdebug v0.0.0-20200204225717-4d6bd78da58d/go.mod h1:B06CSso/AWxiPejj+fheUINGeBKeeEZNt8w+EoU7+L8= +github.com/lestrrat-go/pdebug v0.0.0-20210111095411-35b07dbf089b h1:2v0K4PeWeccG1wpznCE71PqO5scFzSj3jZGkQaVYEWg= +github.com/lestrrat-go/pdebug v0.0.0-20210111095411-35b07dbf089b/go.mod h1:RbVbom7RDVgpH5IqUPbhJ425z9iRIRH0tRWp+uxEjCE= github.com/pkg/errors v0.8.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= github.com/pkg/errors v0.9.1 h1:FEBLx1zS214owpjy7qsBeixbURkuhQAwrK5UwLGTwt4= github.com/pkg/errors v0.9.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= @@ -21,6 +30,8 @@ github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UV github.com/stretchr/testify v1.4.0/go.mod h1:j7eGeouHqKxXV5pUuKE4zz7dFj8WfuZ+81PSLYec5m4= github.com/stretchr/testify v1.6.1 h1:hDPOHmpOpP40lSULcqw7IrRb/u7w6RpDC9399XyoNd0= github.com/stretchr/testify v1.6.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= +github.com/stretchr/testify v1.7.0 h1:nwc3DEeHmmLAfoZucVR881uASk0Mfjw8xYJ99tb5CcY= +github.com/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405 h1:yhCVgyC4o1eVCa2tZl7eS0r+SDo693bJlVdllGtEeKM= gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= diff --git a/interface.go b/interface.go index 9114cd3..74eb3b6 100644 --- a/interface.go +++ b/interface.go @@ -21,9 +21,10 @@ type Client interface { Shutdown(context.Context) error } -//nolint:maligned // Buffered is a Client that buffers incoming messages, and sends them // asynchrnously when it can. +// +//nolint:maligned type Buffered struct { closed bool minionCancel func() diff --git a/minion.go b/minion.go index c040084..131bf07 100644 --- a/minion.go +++ b/minion.go @@ -90,6 +90,7 @@ func newMinion(options ...Option) (*minion, error) { var writeQueueSize = 64 var connectOnStart bool + //nolint:forcetypeassert for _, opt := range options { switch opt.Ident() { case identNetwork{}: @@ -306,7 +307,7 @@ func (m *minion) appendMessage(msg *Message) { if pdebug.Enabled { pdebug.Printf("background reader: replying error to client") } - msg.replyCh <- &bufferFullErrInstance + msg.replyCh <- &errBufferFullInstance } return } diff --git a/ping.go b/ping.go index 4361a0e..705020a 100644 --- a/ping.go +++ b/ping.go @@ -16,6 +16,7 @@ func Ping(ctx context.Context, client Client, tag string, record interface{}, op var interval = 5 * time.Minute var replyCh chan error + //nolint:forcetypeassert for _, option := range options { switch option.Ident() { case identPingInterval{}: diff --git a/pool.go b/pool.go index 8d2ac70..0e2f348 100644 --- a/pool.go +++ b/pool.go @@ -11,6 +11,7 @@ func allocMessage() interface{} { } func getMessage() *Message { + //nolint:forcetypeassert return msgpool.Get().(*Message) } diff --git a/unbuffered.go b/unbuffered.go index 8aa2355..be26af0 100644 --- a/unbuffered.go +++ b/unbuffered.go @@ -14,13 +14,13 @@ import ( // buffered client, an unbuffered client handles the Post() method // synchronously, and does not attempt to buffer the payload. // -// * fluent.WithAddress -// * fluent.WithDialTimeout -// * fluent.WithMarshaler -// * fluent.WithMaxConnAttempts -// * fluent.WithNetwork -// * fluent.WithSubSecond -// * fluent.WithTagPrefix +// - fluent.WithAddress +// - fluent.WithDialTimeout +// - fluent.WithMarshaler +// - fluent.WithMaxConnAttempts +// - fluent.WithNetwork +// - fluent.WithSubSecond +// - fluent.WithTagPrefix // // Please see their respective documentation for details. func NewUnbuffered(options ...Option) (client *Unbuffered, err error) { @@ -39,6 +39,7 @@ func NewUnbuffered(options ...Option) (client *Unbuffered, err error) { } var connectOnStart bool + //nolint:forcetypeassert for _, opt := range options { switch opt.Ident() { case identAddress{}: @@ -128,8 +129,7 @@ func (c *Unbuffered) serialize(msg *Message) ([]byte, error) { // If you would like to specify options to `Post()`, you may pass them at the // end of the method. Currently you can use the following: // -// fluent.WithTimestamp: allows you to set arbitrary timestamp values -// +// fluent.WithTimestamp: allows you to set arbitrary timestamp values func (c *Unbuffered) Post(tag string, v interface{}, options ...Option) (err error) { if pdebug.Enabled { g := pdebug.Marker("fluent.Unbuffered.Post").BindError(&err) @@ -137,6 +137,7 @@ func (c *Unbuffered) Post(tag string, v interface{}, options ...Option) (err err } var t time.Time + //nolint:forcetypeassert for _, opt := range options { switch opt.Ident() { case identTimestamp{}: