Skip to content
Open
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
21 commits
Select commit Hold shift + click to select a range
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
9 changes: 6 additions & 3 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
@@ -1,3 +1,6 @@

* Supported pool of decoders

## v3.104.5
* Added query client session pool metrics: create_in_progress, in_use, waiters_queue
* Added pool item closing for not-alived item
Expand All @@ -6,7 +9,7 @@
* Fixed bug with session query latency metric collector

## v3.104.3
* Changed argument types in `table.Client.ReadRows` to public types for compatibility with mock-generation
* Changed argument types in `table.Client.ReadRows` to public types for compatibility with mock-generation

## v3.104.2
* Added bindings options into `ydb.ParamsFromMap` for bind wide time types
Expand All @@ -25,10 +28,10 @@
* Supported wide `Date32`, `Datetime64` and `Timestamp64` types

## v3.101.4
* Switched internal type of result `ydb.Driver.Query()` from `*internal/query.Client` to `query.Client` interface
* Switched internal type of result `ydb.Driver.Query()` from `*internal/query.Client` to `query.Client` interface

## v3.101.3
* Added `query.TransactionActor` type alias to `query.TxActor` for compatibility with `table.Client` API's
* Added `query.TransactionActor` type alias to `query.TxActor` for compatibility with `table.Client` API's
* Removed comment `experimental` from `ydb.ParamsBuilder` and `ydb.ParamsFromMap`
* Fixed panic on closing `internal/query/sessionCore.done` channel twice
* Fixed hangup when try to send batch of messages with size more, then grpc limits from topic writer internals
Expand Down
115 changes: 97 additions & 18 deletions internal/topic/topicreadercommon/decoders.go
Original file line number Diff line number Diff line change
Expand Up @@ -5,43 +5,122 @@ import (
"errors"
"fmt"
"io"
"sync"

"github.com/ydb-platform/ydb-go-sdk/v3/internal/grpcwrapper/rawtopic/rawtopiccommon"
"github.com/ydb-platform/ydb-go-sdk/v3/internal/xerrors"
)

type ReadResetter interface {
io.Reader
Reset(r io.Reader) error
}

type decoderPool struct {
pool sync.Pool
}

func (p *decoderPool) Get() ReadResetter {
dec, _ := p.pool.Get().(ReadResetter)

return dec
}

func (p *decoderPool) Put(dec ReadResetter) {
p.pool.Put(dec)
}

func newDecoderPool() *decoderPool {
return &decoderPool{
pool: sync.Pool{},
}
}

type DecoderMap struct {
m map[rawtopiccommon.Codec]PublicCreateDecoderFunc
m map[rawtopiccommon.Codec]PublicCreateDecoderFunc
dp map[rawtopiccommon.Codec]*decoderPool
}

func NewDecoderMap() DecoderMap {
return DecoderMap{
m: map[rawtopiccommon.Codec]PublicCreateDecoderFunc{
rawtopiccommon.CodecRaw: func(input io.Reader) (io.Reader, error) {
return input, nil
},
rawtopiccommon.CodecGzip: func(input io.Reader) (io.Reader, error) {
return gzip.NewReader(input)
},
},
dm := DecoderMap{
m: make(map[rawtopiccommon.Codec]PublicCreateDecoderFunc),
dp: make(map[rawtopiccommon.Codec]*decoderPool),
}

dm.AddDecoder(rawtopiccommon.CodecRaw, func(input io.Reader) (ReadResetter, error) {
return &nopResetter{Reader: input}, nil
})

dm.AddDecoder(rawtopiccommon.CodecGzip, func(input io.Reader) (ReadResetter, error) {
gz, err := gzip.NewReader(input)
if err != nil {
return nil, err
}

return gz, nil
})

return dm
}

func (m *DecoderMap) AddDecoder(codec rawtopiccommon.Codec, createFunc PublicCreateDecoderFunc) {
m.m[codec] = createFunc
type pooledDecoder struct {
ReadResetter
pool *decoderPool
}

func (p *pooledDecoder) Close() error {
if closer, ok := p.ReadResetter.(io.Closer); ok {
closer.Close()
}
p.pool.Put(p.ReadResetter)

return nil
}

func (m *DecoderMap) Decode(codec rawtopiccommon.Codec, input io.Reader) (io.Reader, error) {
if f := m.m[codec]; f != nil {
return f(input)
createFunc, ok := m.m[codec]
if !ok {
return nil, xerrors.WithStackTrace(xerrors.Wrap(
fmt.Errorf("ydb: failed decompress message with codec %v: %w", codec, ErrPublicUnexpectedCodec),
))
}

pool := m.dp[codec]
decoder := pool.Get()
if decoder == nil {
var err error
decoder, err = createFunc(input)
if err != nil {
return nil, err
}
} else {
if err := decoder.Reset(input); err != nil {
return nil, err
}
}

return nil, xerrors.WithStackTrace(xerrors.Wrap(
fmt.Errorf("ydb: failed decompress message with codec %v: %w", codec, ErrPublicUnexpectedCodec),
))
return &pooledDecoder{
ReadResetter: decoder,
pool: pool,
}, nil
}

func (m *DecoderMap) AddDecoder(codec rawtopiccommon.Codec, createFunc func(io.Reader) (ReadResetter, error)) {
m.m[codec] = createFunc
m.dp[codec] = newDecoderPool()
}

type nopResetter struct {
io.Reader
}

func (n *nopResetter) Reset(r io.Reader) error {
n.Reader = r

return nil
}

type PublicCreateDecoderFunc func(input io.Reader) (io.Reader, error)
type PublicCreateDecoderFunc func(input io.Reader) (ReadResetter, error)

// ErrPublicUnexpectedCodec return when try to read message content with unknown codec
var ErrPublicUnexpectedCodec = xerrors.Wrap(errors.New("ydb: unexpected codec"))
168 changes: 168 additions & 0 deletions internal/topic/topicreadercommon/decoders_test.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,168 @@
package topicreadercommon

import (
"bytes"
"compress/gzip"
"errors"
"io"
"testing"

"github.com/stretchr/testify/require"

"github.com/ydb-platform/ydb-go-sdk/v3/internal/grpcwrapper/rawtopic/rawtopiccommon"
)

func TestDecoderMap(t *testing.T) {
decoderMap := NewDecoderMap()

t.Run("DecodeRaw", func(t *testing.T) {
data := []byte("test data")
reader := bytes.NewReader(data)

decodedReader, err := decoderMap.Decode(rawtopiccommon.CodecRaw, reader)
require.NoError(t, err)

result, err := io.ReadAll(decodedReader)
require.NoError(t, err)
require.Equal(t, data, result)
})

t.Run("DecodeGzip", func(t *testing.T) {
data := []byte("test data")
var buf bytes.Buffer
gzipWriter := gzip.NewWriter(&buf)
_, err := gzipWriter.Write(data)
require.NoError(t, err)
require.NoError(t, gzipWriter.Close())

decodedReader, err := decoderMap.Decode(rawtopiccommon.CodecGzip, &buf)
require.NoError(t, err)

result, err := io.ReadAll(decodedReader)
require.NoError(t, err)
require.Equal(t, data, result)
})

t.Run("DecodeUnknownCodec", func(t *testing.T) {
_, err := decoderMap.Decode(rawtopiccommon.Codec(999), bytes.NewReader([]byte{}))
require.Error(t, err)
require.True(t, errors.Is(err, ErrPublicUnexpectedCodec))
})

t.Run("DecodeCustomCodec", func(t *testing.T) {
dm := NewDecoderMap()
customCodec := rawtopiccommon.Codec(1001)
dm.AddDecoder(customCodec, func(input io.Reader) (ReadResetter, error) {
return gzip.NewReader(input)
})
require.Len(t, dm.dp, 3)

data := []byte("custom test data")
var buf bytes.Buffer
gzipWriter := gzip.NewWriter(&buf)
_, err := gzipWriter.Write(data)
require.NoError(t, err)
require.NoError(t, gzipWriter.Close())

decodedReader, err := dm.Decode(customCodec, &buf)
require.NoError(t, err)
defer decodedReader.(io.Closer).Close()

result, err := io.ReadAll(decodedReader)
require.NoError(t, err)
require.Equal(t, string(data), string(result))

data2 := []byte("second test data")
var buf2 bytes.Buffer
gzipWriter2 := gzip.NewWriter(&buf2)
_, err = gzipWriter2.Write(data2)
require.NoError(t, err)
require.NoError(t, gzipWriter2.Close())

decodedReader2, err := dm.Decode(customCodec, &buf2)
require.NoError(t, err)
defer decodedReader2.(io.Closer).Close()

result2, err := io.ReadAll(decodedReader2)
require.NoError(t, err)
require.Equal(t, string(data2), string(result2))
})

t.Run("DecodeCustomCodec", func(t *testing.T) {
dm := NewDecoderMap()
customCodec := rawtopiccommon.Codec(1001)
dm.AddDecoder(customCodec, func(input io.Reader) (ReadResetter, error) {
return gzip.NewReader(input)
})
require.Len(t, dm.dp, 3)

data := []byte("custom test data")
var buf bytes.Buffer
gzipWriter := gzip.NewWriter(&buf)
_, err := gzipWriter.Write(data)
require.NoError(t, err)
require.NoError(t, gzipWriter.Close())

decodedReader, err := dm.Decode(customCodec, &buf)
require.NoError(t, err)
result, err := io.ReadAll(decodedReader)
require.NoError(t, err)
require.Equal(t, string(data), string(result))
require.NoError(t, decodedReader.(io.Closer).Close())

data2 := []byte("second test data")
var buf2 bytes.Buffer
gzipWriter2 := gzip.NewWriter(&buf2)
_, err = gzipWriter2.Write(data2)
require.NoError(t, err)
require.NoError(t, gzipWriter2.Close())

decodedReader2, err := dm.Decode(customCodec, &buf2)
require.NoError(t, err)
result2, err := io.ReadAll(decodedReader2)
require.NoError(t, err)
require.Equal(t, string(data2), string(result2))
require.NoError(t, decodedReader2.(io.Closer).Close())
})

t.Run("PoolReuse", func(t *testing.T) {
dm := NewDecoderMap()
customCodec := rawtopiccommon.Codec(1002)

dm.AddDecoder(customCodec, func(input io.Reader) (ReadResetter, error) {
return gzip.NewReader(input)
})

data1 := []byte("hello")
var buf1 bytes.Buffer
gzipWriter1 := gzip.NewWriter(&buf1)
_, err := gzipWriter1.Write(data1)
require.NoError(t, err)
require.NoError(t, gzipWriter1.Close())

reader1, err := dm.Decode(customCodec, &buf1)
require.NoError(t, err, "first decoding should succeed")

result1, err := io.ReadAll(reader1)
require.NoError(t, err, "reading first message should succeed")
require.Equal(t, string(data1), string(result1), "data should match")

require.NoError(t, reader1.(io.Closer).Close(), "closing first reader should succeed")

data2 := []byte("world")
var buf2 bytes.Buffer
gzipWriter2 := gzip.NewWriter(&buf2)
_, err = gzipWriter2.Write(data2)
require.NoError(t, err)
require.NoError(t, gzipWriter2.Close())

reader2, err := dm.Decode(customCodec, &buf2)
require.NoError(t, err, "second decoding should succeed")

result2, err := io.ReadAll(reader2)
require.NoError(t, err, "reading second message should succeed")
require.Equal(t, string(data2), string(result2), "data of second message should match")

require.NoError(t, reader2.(io.Closer).Close(), "closing second reader should succeed")
})
}
16 changes: 13 additions & 3 deletions internal/topic/topicreadercommon/message.go
Original file line number Diff line number Diff line change
Expand Up @@ -58,10 +58,11 @@ func (m *PublicMessage) UnmarshalTo(dst PublicMessageContentUnmarshaler) error {
if m.dataConsumed {
return xerrors.WithStackTrace(errMessageWasReadEarly)
}

m.dataConsumed = true
err := callbackOnReaderContent(globalReadMessagePool, m, m.UncompressedSize, dst)
m.data.Close()

return callbackOnReaderContent(globalReadMessagePool, m, m.UncompressedSize, dst)
return err
}

// Read implements io.Reader
Expand All @@ -73,7 +74,16 @@ func (m *PublicMessage) UnmarshalTo(dst PublicMessageContentUnmarshaler) error {
func (m *PublicMessage) Read(p []byte) (n int, err error) {
m.dataConsumed = true

return m.data.Read(p)
n, err = m.data.Read(p)
if err != nil {
m.data.Close()
}

return n, err
}

func (m *PublicMessage) Close() error {
return m.data.Close()
}

// PublicMessageContentUnmarshaler is interface for unmarshal message content
Expand Down
Loading
Loading