forked from grafana/loki
-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
chore(dataobj): logs section metadata and encoding/decoding
This PR introduces metadata for a "logs section," which is intended to hold a sequence of log records across one or more streams. The code is a near-identical copy of grafana#15676. Future work is needed to identify if dataset sections can have their encoding, decoding, and dataset implementations deduplicated.
- Loading branch information
Showing
12 changed files
with
1,954 additions
and
13 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,152 @@ | ||
package encoding | ||
|
||
import ( | ||
"context" | ||
"fmt" | ||
|
||
"github.com/grafana/loki/v3/pkg/dataobj/internal/dataset" | ||
"github.com/grafana/loki/v3/pkg/dataobj/internal/metadata/filemd" | ||
"github.com/grafana/loki/v3/pkg/dataobj/internal/metadata/logsmd" | ||
"github.com/grafana/loki/v3/pkg/dataobj/internal/result" | ||
) | ||
|
||
// LogsDataset implements returns a [dataset.Dataset] from a [LogsDecoder] for | ||
// the given section. | ||
func LogsDataset(dec LogsDecoder, sec *filemd.SectionInfo) dataset.Dataset { | ||
return &logsDataset{dec: dec, sec: sec} | ||
} | ||
|
||
type logsDataset struct { | ||
dec LogsDecoder | ||
sec *filemd.SectionInfo | ||
} | ||
|
||
func (ds *logsDataset) ListColumns(ctx context.Context) result.Seq[dataset.Column] { | ||
return result.Iter(func(yield func(dataset.Column) bool) error { | ||
columns, err := ds.dec.Columns(ctx, ds.sec) | ||
if err != nil { | ||
return err | ||
} | ||
|
||
for _, column := range columns { | ||
if !yield(&logsDatasetColumn{dec: ds.dec, desc: column}) { | ||
return nil | ||
} | ||
} | ||
|
||
return err | ||
}) | ||
|
||
} | ||
|
||
func (ds *logsDataset) ListPages(ctx context.Context, columns []dataset.Column) result.Seq[dataset.Pages] { | ||
// TODO(rfratto): Switch to batch retrieval instead of iterating over each column. | ||
return result.Iter(func(yield func(dataset.Pages) bool) error { | ||
for _, column := range columns { | ||
pages, err := result.Collect(column.ListPages(ctx)) | ||
if err != nil { | ||
return err | ||
} else if !yield(pages) { | ||
return nil | ||
} | ||
} | ||
|
||
return nil | ||
}) | ||
} | ||
|
||
func (ds *logsDataset) ReadPages(ctx context.Context, pages []dataset.Page) result.Seq[dataset.PageData] { | ||
// TODO(rfratto): Switch to batch retrieval instead of iterating over each page. | ||
return result.Iter(func(yield func(dataset.PageData) bool) error { | ||
for _, page := range pages { | ||
data, err := page.ReadPage(ctx) | ||
if err != nil { | ||
return err | ||
} else if !yield(data) { | ||
return nil | ||
} | ||
} | ||
|
||
return nil | ||
}) | ||
} | ||
|
||
type logsDatasetColumn struct { | ||
dec LogsDecoder | ||
desc *logsmd.ColumnDesc | ||
|
||
info *dataset.ColumnInfo | ||
} | ||
|
||
func (col *logsDatasetColumn) ColumnInfo() *dataset.ColumnInfo { | ||
if col.info != nil { | ||
return col.info | ||
} | ||
|
||
col.info = &dataset.ColumnInfo{ | ||
Name: col.desc.Info.Name, | ||
Type: col.desc.Info.ValueType, | ||
Compression: col.desc.Info.Compression, | ||
|
||
RowsCount: int(col.desc.Info.RowsCount), | ||
CompressedSize: int(col.desc.Info.CompressedSize), | ||
UncompressedSize: int(col.desc.Info.UncompressedSize), | ||
|
||
Statistics: col.desc.Info.Statistics, | ||
} | ||
return col.info | ||
} | ||
|
||
func (col *logsDatasetColumn) ListPages(ctx context.Context) result.Seq[dataset.Page] { | ||
return result.Iter(func(yield func(dataset.Page) bool) error { | ||
pageSets, err := result.Collect(col.dec.Pages(ctx, []*logsmd.ColumnDesc{col.desc})) | ||
if err != nil { | ||
return err | ||
} else if len(pageSets) != 1 { | ||
return fmt.Errorf("unexpected number of page sets: got=%d want=1", len(pageSets)) | ||
} | ||
|
||
for _, page := range pageSets[0] { | ||
if !yield(&logsDatasetPage{dec: col.dec, desc: page}) { | ||
return nil | ||
} | ||
} | ||
|
||
return nil | ||
}) | ||
} | ||
|
||
type logsDatasetPage struct { | ||
dec LogsDecoder | ||
desc *logsmd.PageDesc | ||
|
||
info *dataset.PageInfo | ||
} | ||
|
||
func (p *logsDatasetPage) PageInfo() *dataset.PageInfo { | ||
if p.info != nil { | ||
return p.info | ||
} | ||
|
||
p.info = &dataset.PageInfo{ | ||
UncompressedSize: int(p.desc.Info.UncompressedSize), | ||
CompressedSize: int(p.desc.Info.CompressedSize), | ||
CRC32: p.desc.Info.Crc32, | ||
RowCount: int(p.desc.Info.RowsCount), | ||
|
||
Encoding: p.desc.Info.Encoding, | ||
Stats: p.desc.Info.Statistics, | ||
} | ||
return p.info | ||
} | ||
|
||
func (p *logsDatasetPage) ReadPage(ctx context.Context) (dataset.PageData, error) { | ||
pages, err := result.Collect(p.dec.ReadPages(ctx, []*logsmd.PageDesc{p.desc})) | ||
if err != nil { | ||
return nil, err | ||
} else if len(pages) != 1 { | ||
return nil, fmt.Errorf("unexpected number of pages: got=%d want=1", len(pages)) | ||
} | ||
|
||
return pages[0], nil | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.