diff --git a/.github/workflows/test.yml b/.github/workflows/test.yml index 9215438a97..67984ffad1 100644 --- a/.github/workflows/test.yml +++ b/.github/workflows/test.yml @@ -63,10 +63,10 @@ jobs: doc-validator: runs-on: "ubuntu-latest" container: - image: "grafana/doc-validator:v4.1.1" + image: "grafana/doc-validator:v5.1.0" steps: - name: "Checkout code" - uses: "actions/checkout@v3" + uses: "actions/checkout@v4" - name: "Run doc-validator tool" run: > doc-validator diff --git a/.gitignore b/.gitignore index e04fd0a185..6ca9a4b5b1 100644 --- a/.gitignore +++ b/.gitignore @@ -15,6 +15,7 @@ pyroscope-sync/ data/ data-shared/ data-compactor/ +data-metastore/ .DS_Store **/dist diff --git a/.golangci.yml b/.golangci.yml index 7c43a84bbf..d050c0ae42 100644 --- a/.golangci.yml +++ b/.golangci.yml @@ -81,6 +81,8 @@ issues: exclude-dirs: - win_eventlog$ - pkg/og + - pkg/experiment + # which files to skip: they will be analyzed, but issues from them # won't be reported. Default value is empty list, but there is # no need to include all autogenerated files, we confidently recognize diff --git a/api/compactor/v1/compactor.proto b/api/compactor/v1/compactor.proto new file mode 100644 index 0000000000..513b0e7393 --- /dev/null +++ b/api/compactor/v1/compactor.proto @@ -0,0 +1,95 @@ +syntax = "proto3"; + +package compactor.v1; + +import "metastore/v1/metastore.proto"; + +service CompactionPlanner { + // Used to both retrieve jobs and update the jobs status at the same time. + rpc PollCompactionJobs(PollCompactionJobsRequest) returns (PollCompactionJobsResponse) {} + // Used for admin purposes only. + rpc GetCompactionJobs(GetCompactionRequest) returns (GetCompactionResponse) {} +} + +message PollCompactionJobsRequest { + // A batch of status updates for in-progress jobs from a worker. + repeated CompactionJobStatus job_status_updates = 1; + // How many new jobs a worker can be assigned to. + uint32 job_capacity = 2; +} + +message PollCompactionJobsResponse { + repeated CompactionJob compaction_jobs = 1; +} + +message GetCompactionRequest {} + +message GetCompactionResponse { + // A list of all compaction jobs + repeated CompactionJob compaction_jobs = 1; +} + +// One compaction job may result in multiple output blocks. +message CompactionJob { + // Unique name of the job. + string name = 1; + CompactionOptions options = 2; + // List of the input blocks. + repeated metastore.v1.BlockMeta blocks = 3; + CompactionJobStatus status = 4; + // Fencing token. + uint64 raft_log_index = 5; + // Shard the blocks belong to. + uint32 shard = 6; + // Optional, empty for compaction level 0. + string tenant_id = 7; + uint32 compaction_level = 8; +} + +message CompactionOptions { + // Compaction planner should instruct the compactor + // worker how to compact the blocks: + // - Limits and tenant overrides. + // - Feature flags. + + // How often the compaction worker should update + // the job status. If overdue, the job ownership + // is revoked. + uint64 status_update_interval_seconds = 1; +} + +message CompactionJobStatus { + string job_name = 1; + // Status update allows the planner to keep + // track of the job ownership and compaction + // progress: + // - If the job status is other than IN_PROGRESS, + // the ownership of the job is revoked. + // - FAILURE must only be sent if the failure is + // persistent and the compaction can't be accomplished. + // - completed_job must be empty if the status is + // other than SUCCESS, and vice-versa. + // - UNSPECIFIED must be sent if the worker rejects + // or cancels the compaction job. + // + // Partial results/status is not allowed. + CompactionStatus status = 2; + CompletedJob completed_job = 3; + // Fencing token. + uint64 raft_log_index = 4; + // Shard the blocks belong to. + uint32 shard = 5; + // Optional, empty for compaction level 0. + string tenant_id = 6; +} + +enum CompactionStatus { + COMPACTION_STATUS_UNSPECIFIED = 0; + COMPACTION_STATUS_IN_PROGRESS = 1; + COMPACTION_STATUS_SUCCESS = 2; + COMPACTION_STATUS_FAILURE = 3; +} + +message CompletedJob { + repeated metastore.v1.BlockMeta blocks = 1; +} diff --git a/api/gen/proto/go/compactor/v1/compactor.pb.go b/api/gen/proto/go/compactor/v1/compactor.pb.go new file mode 100644 index 0000000000..a51f77a522 --- /dev/null +++ b/api/gen/proto/go/compactor/v1/compactor.pb.go @@ -0,0 +1,860 @@ +// Code generated by protoc-gen-go. DO NOT EDIT. +// versions: +// protoc-gen-go v1.34.2 +// protoc (unknown) +// source: compactor/v1/compactor.proto + +package compactorv1 + +import ( + v1 "github.com/grafana/pyroscope/api/gen/proto/go/metastore/v1" + protoreflect "google.golang.org/protobuf/reflect/protoreflect" + protoimpl "google.golang.org/protobuf/runtime/protoimpl" + reflect "reflect" + sync "sync" +) + +const ( + // Verify that this generated code is sufficiently up-to-date. + _ = protoimpl.EnforceVersion(20 - protoimpl.MinVersion) + // Verify that runtime/protoimpl is sufficiently up-to-date. + _ = protoimpl.EnforceVersion(protoimpl.MaxVersion - 20) +) + +type CompactionStatus int32 + +const ( + CompactionStatus_COMPACTION_STATUS_UNSPECIFIED CompactionStatus = 0 + CompactionStatus_COMPACTION_STATUS_IN_PROGRESS CompactionStatus = 1 + CompactionStatus_COMPACTION_STATUS_SUCCESS CompactionStatus = 2 + CompactionStatus_COMPACTION_STATUS_FAILURE CompactionStatus = 3 +) + +// Enum value maps for CompactionStatus. +var ( + CompactionStatus_name = map[int32]string{ + 0: "COMPACTION_STATUS_UNSPECIFIED", + 1: "COMPACTION_STATUS_IN_PROGRESS", + 2: "COMPACTION_STATUS_SUCCESS", + 3: "COMPACTION_STATUS_FAILURE", + } + CompactionStatus_value = map[string]int32{ + "COMPACTION_STATUS_UNSPECIFIED": 0, + "COMPACTION_STATUS_IN_PROGRESS": 1, + "COMPACTION_STATUS_SUCCESS": 2, + "COMPACTION_STATUS_FAILURE": 3, + } +) + +func (x CompactionStatus) Enum() *CompactionStatus { + p := new(CompactionStatus) + *p = x + return p +} + +func (x CompactionStatus) String() string { + return protoimpl.X.EnumStringOf(x.Descriptor(), protoreflect.EnumNumber(x)) +} + +func (CompactionStatus) Descriptor() protoreflect.EnumDescriptor { + return file_compactor_v1_compactor_proto_enumTypes[0].Descriptor() +} + +func (CompactionStatus) Type() protoreflect.EnumType { + return &file_compactor_v1_compactor_proto_enumTypes[0] +} + +func (x CompactionStatus) Number() protoreflect.EnumNumber { + return protoreflect.EnumNumber(x) +} + +// Deprecated: Use CompactionStatus.Descriptor instead. +func (CompactionStatus) EnumDescriptor() ([]byte, []int) { + return file_compactor_v1_compactor_proto_rawDescGZIP(), []int{0} +} + +type PollCompactionJobsRequest struct { + state protoimpl.MessageState + sizeCache protoimpl.SizeCache + unknownFields protoimpl.UnknownFields + + // A batch of status updates for in-progress jobs from a worker. + JobStatusUpdates []*CompactionJobStatus `protobuf:"bytes,1,rep,name=job_status_updates,json=jobStatusUpdates,proto3" json:"job_status_updates,omitempty"` + // How many new jobs a worker can be assigned to. + JobCapacity uint32 `protobuf:"varint,2,opt,name=job_capacity,json=jobCapacity,proto3" json:"job_capacity,omitempty"` +} + +func (x *PollCompactionJobsRequest) Reset() { + *x = PollCompactionJobsRequest{} + if protoimpl.UnsafeEnabled { + mi := &file_compactor_v1_compactor_proto_msgTypes[0] + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + ms.StoreMessageInfo(mi) + } +} + +func (x *PollCompactionJobsRequest) String() string { + return protoimpl.X.MessageStringOf(x) +} + +func (*PollCompactionJobsRequest) ProtoMessage() {} + +func (x *PollCompactionJobsRequest) ProtoReflect() protoreflect.Message { + mi := &file_compactor_v1_compactor_proto_msgTypes[0] + if protoimpl.UnsafeEnabled && x != nil { + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + if ms.LoadMessageInfo() == nil { + ms.StoreMessageInfo(mi) + } + return ms + } + return mi.MessageOf(x) +} + +// Deprecated: Use PollCompactionJobsRequest.ProtoReflect.Descriptor instead. +func (*PollCompactionJobsRequest) Descriptor() ([]byte, []int) { + return file_compactor_v1_compactor_proto_rawDescGZIP(), []int{0} +} + +func (x *PollCompactionJobsRequest) GetJobStatusUpdates() []*CompactionJobStatus { + if x != nil { + return x.JobStatusUpdates + } + return nil +} + +func (x *PollCompactionJobsRequest) GetJobCapacity() uint32 { + if x != nil { + return x.JobCapacity + } + return 0 +} + +type PollCompactionJobsResponse struct { + state protoimpl.MessageState + sizeCache protoimpl.SizeCache + unknownFields protoimpl.UnknownFields + + CompactionJobs []*CompactionJob `protobuf:"bytes,1,rep,name=compaction_jobs,json=compactionJobs,proto3" json:"compaction_jobs,omitempty"` +} + +func (x *PollCompactionJobsResponse) Reset() { + *x = PollCompactionJobsResponse{} + if protoimpl.UnsafeEnabled { + mi := &file_compactor_v1_compactor_proto_msgTypes[1] + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + ms.StoreMessageInfo(mi) + } +} + +func (x *PollCompactionJobsResponse) String() string { + return protoimpl.X.MessageStringOf(x) +} + +func (*PollCompactionJobsResponse) ProtoMessage() {} + +func (x *PollCompactionJobsResponse) ProtoReflect() protoreflect.Message { + mi := &file_compactor_v1_compactor_proto_msgTypes[1] + if protoimpl.UnsafeEnabled && x != nil { + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + if ms.LoadMessageInfo() == nil { + ms.StoreMessageInfo(mi) + } + return ms + } + return mi.MessageOf(x) +} + +// Deprecated: Use PollCompactionJobsResponse.ProtoReflect.Descriptor instead. +func (*PollCompactionJobsResponse) Descriptor() ([]byte, []int) { + return file_compactor_v1_compactor_proto_rawDescGZIP(), []int{1} +} + +func (x *PollCompactionJobsResponse) GetCompactionJobs() []*CompactionJob { + if x != nil { + return x.CompactionJobs + } + return nil +} + +type GetCompactionRequest struct { + state protoimpl.MessageState + sizeCache protoimpl.SizeCache + unknownFields protoimpl.UnknownFields +} + +func (x *GetCompactionRequest) Reset() { + *x = GetCompactionRequest{} + if protoimpl.UnsafeEnabled { + mi := &file_compactor_v1_compactor_proto_msgTypes[2] + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + ms.StoreMessageInfo(mi) + } +} + +func (x *GetCompactionRequest) String() string { + return protoimpl.X.MessageStringOf(x) +} + +func (*GetCompactionRequest) ProtoMessage() {} + +func (x *GetCompactionRequest) ProtoReflect() protoreflect.Message { + mi := &file_compactor_v1_compactor_proto_msgTypes[2] + if protoimpl.UnsafeEnabled && x != nil { + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + if ms.LoadMessageInfo() == nil { + ms.StoreMessageInfo(mi) + } + return ms + } + return mi.MessageOf(x) +} + +// Deprecated: Use GetCompactionRequest.ProtoReflect.Descriptor instead. +func (*GetCompactionRequest) Descriptor() ([]byte, []int) { + return file_compactor_v1_compactor_proto_rawDescGZIP(), []int{2} +} + +type GetCompactionResponse struct { + state protoimpl.MessageState + sizeCache protoimpl.SizeCache + unknownFields protoimpl.UnknownFields + + // A list of all compaction jobs + CompactionJobs []*CompactionJob `protobuf:"bytes,1,rep,name=compaction_jobs,json=compactionJobs,proto3" json:"compaction_jobs,omitempty"` +} + +func (x *GetCompactionResponse) Reset() { + *x = GetCompactionResponse{} + if protoimpl.UnsafeEnabled { + mi := &file_compactor_v1_compactor_proto_msgTypes[3] + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + ms.StoreMessageInfo(mi) + } +} + +func (x *GetCompactionResponse) String() string { + return protoimpl.X.MessageStringOf(x) +} + +func (*GetCompactionResponse) ProtoMessage() {} + +func (x *GetCompactionResponse) ProtoReflect() protoreflect.Message { + mi := &file_compactor_v1_compactor_proto_msgTypes[3] + if protoimpl.UnsafeEnabled && x != nil { + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + if ms.LoadMessageInfo() == nil { + ms.StoreMessageInfo(mi) + } + return ms + } + return mi.MessageOf(x) +} + +// Deprecated: Use GetCompactionResponse.ProtoReflect.Descriptor instead. +func (*GetCompactionResponse) Descriptor() ([]byte, []int) { + return file_compactor_v1_compactor_proto_rawDescGZIP(), []int{3} +} + +func (x *GetCompactionResponse) GetCompactionJobs() []*CompactionJob { + if x != nil { + return x.CompactionJobs + } + return nil +} + +// One compaction job may result in multiple output blocks. +type CompactionJob struct { + state protoimpl.MessageState + sizeCache protoimpl.SizeCache + unknownFields protoimpl.UnknownFields + + // Unique name of the job. + Name string `protobuf:"bytes,1,opt,name=name,proto3" json:"name,omitempty"` + Options *CompactionOptions `protobuf:"bytes,2,opt,name=options,proto3" json:"options,omitempty"` + // List of the input blocks. + Blocks []*v1.BlockMeta `protobuf:"bytes,3,rep,name=blocks,proto3" json:"blocks,omitempty"` + Status *CompactionJobStatus `protobuf:"bytes,4,opt,name=status,proto3" json:"status,omitempty"` + // Fencing token. + RaftLogIndex uint64 `protobuf:"varint,5,opt,name=raft_log_index,json=raftLogIndex,proto3" json:"raft_log_index,omitempty"` + // Shard the blocks belong to. + Shard uint32 `protobuf:"varint,6,opt,name=shard,proto3" json:"shard,omitempty"` + // Optional, empty for compaction level 0. + TenantId string `protobuf:"bytes,7,opt,name=tenant_id,json=tenantId,proto3" json:"tenant_id,omitempty"` + CompactionLevel uint32 `protobuf:"varint,8,opt,name=compaction_level,json=compactionLevel,proto3" json:"compaction_level,omitempty"` +} + +func (x *CompactionJob) Reset() { + *x = CompactionJob{} + if protoimpl.UnsafeEnabled { + mi := &file_compactor_v1_compactor_proto_msgTypes[4] + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + ms.StoreMessageInfo(mi) + } +} + +func (x *CompactionJob) String() string { + return protoimpl.X.MessageStringOf(x) +} + +func (*CompactionJob) ProtoMessage() {} + +func (x *CompactionJob) ProtoReflect() protoreflect.Message { + mi := &file_compactor_v1_compactor_proto_msgTypes[4] + if protoimpl.UnsafeEnabled && x != nil { + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + if ms.LoadMessageInfo() == nil { + ms.StoreMessageInfo(mi) + } + return ms + } + return mi.MessageOf(x) +} + +// Deprecated: Use CompactionJob.ProtoReflect.Descriptor instead. +func (*CompactionJob) Descriptor() ([]byte, []int) { + return file_compactor_v1_compactor_proto_rawDescGZIP(), []int{4} +} + +func (x *CompactionJob) GetName() string { + if x != nil { + return x.Name + } + return "" +} + +func (x *CompactionJob) GetOptions() *CompactionOptions { + if x != nil { + return x.Options + } + return nil +} + +func (x *CompactionJob) GetBlocks() []*v1.BlockMeta { + if x != nil { + return x.Blocks + } + return nil +} + +func (x *CompactionJob) GetStatus() *CompactionJobStatus { + if x != nil { + return x.Status + } + return nil +} + +func (x *CompactionJob) GetRaftLogIndex() uint64 { + if x != nil { + return x.RaftLogIndex + } + return 0 +} + +func (x *CompactionJob) GetShard() uint32 { + if x != nil { + return x.Shard + } + return 0 +} + +func (x *CompactionJob) GetTenantId() string { + if x != nil { + return x.TenantId + } + return "" +} + +func (x *CompactionJob) GetCompactionLevel() uint32 { + if x != nil { + return x.CompactionLevel + } + return 0 +} + +type CompactionOptions struct { + state protoimpl.MessageState + sizeCache protoimpl.SizeCache + unknownFields protoimpl.UnknownFields + + // How often the compaction worker should update + // the job status. If overdue, the job ownership + // is revoked. + StatusUpdateIntervalSeconds uint64 `protobuf:"varint,1,opt,name=status_update_interval_seconds,json=statusUpdateIntervalSeconds,proto3" json:"status_update_interval_seconds,omitempty"` +} + +func (x *CompactionOptions) Reset() { + *x = CompactionOptions{} + if protoimpl.UnsafeEnabled { + mi := &file_compactor_v1_compactor_proto_msgTypes[5] + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + ms.StoreMessageInfo(mi) + } +} + +func (x *CompactionOptions) String() string { + return protoimpl.X.MessageStringOf(x) +} + +func (*CompactionOptions) ProtoMessage() {} + +func (x *CompactionOptions) ProtoReflect() protoreflect.Message { + mi := &file_compactor_v1_compactor_proto_msgTypes[5] + if protoimpl.UnsafeEnabled && x != nil { + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + if ms.LoadMessageInfo() == nil { + ms.StoreMessageInfo(mi) + } + return ms + } + return mi.MessageOf(x) +} + +// Deprecated: Use CompactionOptions.ProtoReflect.Descriptor instead. +func (*CompactionOptions) Descriptor() ([]byte, []int) { + return file_compactor_v1_compactor_proto_rawDescGZIP(), []int{5} +} + +func (x *CompactionOptions) GetStatusUpdateIntervalSeconds() uint64 { + if x != nil { + return x.StatusUpdateIntervalSeconds + } + return 0 +} + +type CompactionJobStatus struct { + state protoimpl.MessageState + sizeCache protoimpl.SizeCache + unknownFields protoimpl.UnknownFields + + JobName string `protobuf:"bytes,1,opt,name=job_name,json=jobName,proto3" json:"job_name,omitempty"` + // Status update allows the planner to keep + // track of the job ownership and compaction + // progress: + // - If the job status is other than IN_PROGRESS, + // the ownership of the job is revoked. + // - FAILURE must only be sent if the failure is + // persistent and the compaction can't be accomplished. + // - completed_job must be empty if the status is + // other than SUCCESS, and vice-versa. + // - UNSPECIFIED must be sent if the worker rejects + // or cancels the compaction job. + // + // Partial results/status is not allowed. + Status CompactionStatus `protobuf:"varint,2,opt,name=status,proto3,enum=compactor.v1.CompactionStatus" json:"status,omitempty"` + CompletedJob *CompletedJob `protobuf:"bytes,3,opt,name=completed_job,json=completedJob,proto3" json:"completed_job,omitempty"` + // Fencing token. + RaftLogIndex uint64 `protobuf:"varint,4,opt,name=raft_log_index,json=raftLogIndex,proto3" json:"raft_log_index,omitempty"` + // Shard the blocks belong to. + Shard uint32 `protobuf:"varint,5,opt,name=shard,proto3" json:"shard,omitempty"` + // Optional, empty for compaction level 0. + TenantId string `protobuf:"bytes,6,opt,name=tenant_id,json=tenantId,proto3" json:"tenant_id,omitempty"` +} + +func (x *CompactionJobStatus) Reset() { + *x = CompactionJobStatus{} + if protoimpl.UnsafeEnabled { + mi := &file_compactor_v1_compactor_proto_msgTypes[6] + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + ms.StoreMessageInfo(mi) + } +} + +func (x *CompactionJobStatus) String() string { + return protoimpl.X.MessageStringOf(x) +} + +func (*CompactionJobStatus) ProtoMessage() {} + +func (x *CompactionJobStatus) ProtoReflect() protoreflect.Message { + mi := &file_compactor_v1_compactor_proto_msgTypes[6] + if protoimpl.UnsafeEnabled && x != nil { + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + if ms.LoadMessageInfo() == nil { + ms.StoreMessageInfo(mi) + } + return ms + } + return mi.MessageOf(x) +} + +// Deprecated: Use CompactionJobStatus.ProtoReflect.Descriptor instead. +func (*CompactionJobStatus) Descriptor() ([]byte, []int) { + return file_compactor_v1_compactor_proto_rawDescGZIP(), []int{6} +} + +func (x *CompactionJobStatus) GetJobName() string { + if x != nil { + return x.JobName + } + return "" +} + +func (x *CompactionJobStatus) GetStatus() CompactionStatus { + if x != nil { + return x.Status + } + return CompactionStatus_COMPACTION_STATUS_UNSPECIFIED +} + +func (x *CompactionJobStatus) GetCompletedJob() *CompletedJob { + if x != nil { + return x.CompletedJob + } + return nil +} + +func (x *CompactionJobStatus) GetRaftLogIndex() uint64 { + if x != nil { + return x.RaftLogIndex + } + return 0 +} + +func (x *CompactionJobStatus) GetShard() uint32 { + if x != nil { + return x.Shard + } + return 0 +} + +func (x *CompactionJobStatus) GetTenantId() string { + if x != nil { + return x.TenantId + } + return "" +} + +type CompletedJob struct { + state protoimpl.MessageState + sizeCache protoimpl.SizeCache + unknownFields protoimpl.UnknownFields + + Blocks []*v1.BlockMeta `protobuf:"bytes,1,rep,name=blocks,proto3" json:"blocks,omitempty"` +} + +func (x *CompletedJob) Reset() { + *x = CompletedJob{} + if protoimpl.UnsafeEnabled { + mi := &file_compactor_v1_compactor_proto_msgTypes[7] + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + ms.StoreMessageInfo(mi) + } +} + +func (x *CompletedJob) String() string { + return protoimpl.X.MessageStringOf(x) +} + +func (*CompletedJob) ProtoMessage() {} + +func (x *CompletedJob) ProtoReflect() protoreflect.Message { + mi := &file_compactor_v1_compactor_proto_msgTypes[7] + if protoimpl.UnsafeEnabled && x != nil { + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + if ms.LoadMessageInfo() == nil { + ms.StoreMessageInfo(mi) + } + return ms + } + return mi.MessageOf(x) +} + +// Deprecated: Use CompletedJob.ProtoReflect.Descriptor instead. +func (*CompletedJob) Descriptor() ([]byte, []int) { + return file_compactor_v1_compactor_proto_rawDescGZIP(), []int{7} +} + +func (x *CompletedJob) GetBlocks() []*v1.BlockMeta { + if x != nil { + return x.Blocks + } + return nil +} + +var File_compactor_v1_compactor_proto protoreflect.FileDescriptor + +var file_compactor_v1_compactor_proto_rawDesc = []byte{ + 0x0a, 0x1c, 0x63, 0x6f, 0x6d, 0x70, 0x61, 0x63, 0x74, 0x6f, 0x72, 0x2f, 0x76, 0x31, 0x2f, 0x63, + 0x6f, 0x6d, 0x70, 0x61, 0x63, 0x74, 0x6f, 0x72, 0x2e, 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x12, 0x0c, + 0x63, 0x6f, 0x6d, 0x70, 0x61, 0x63, 0x74, 0x6f, 0x72, 0x2e, 0x76, 0x31, 0x1a, 0x1c, 0x6d, 0x65, + 0x74, 0x61, 0x73, 0x74, 0x6f, 0x72, 0x65, 0x2f, 0x76, 0x31, 0x2f, 0x6d, 0x65, 0x74, 0x61, 0x73, + 0x74, 0x6f, 0x72, 0x65, 0x2e, 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x22, 0x8f, 0x01, 0x0a, 0x19, 0x50, + 0x6f, 0x6c, 0x6c, 0x43, 0x6f, 0x6d, 0x70, 0x61, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x4a, 0x6f, 0x62, + 0x73, 0x52, 0x65, 0x71, 0x75, 0x65, 0x73, 0x74, 0x12, 0x4f, 0x0a, 0x12, 0x6a, 0x6f, 0x62, 0x5f, + 0x73, 0x74, 0x61, 0x74, 0x75, 0x73, 0x5f, 0x75, 0x70, 0x64, 0x61, 0x74, 0x65, 0x73, 0x18, 0x01, + 0x20, 0x03, 0x28, 0x0b, 0x32, 0x21, 0x2e, 0x63, 0x6f, 0x6d, 0x70, 0x61, 0x63, 0x74, 0x6f, 0x72, + 0x2e, 0x76, 0x31, 0x2e, 0x43, 0x6f, 0x6d, 0x70, 0x61, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x4a, 0x6f, + 0x62, 0x53, 0x74, 0x61, 0x74, 0x75, 0x73, 0x52, 0x10, 0x6a, 0x6f, 0x62, 0x53, 0x74, 0x61, 0x74, + 0x75, 0x73, 0x55, 0x70, 0x64, 0x61, 0x74, 0x65, 0x73, 0x12, 0x21, 0x0a, 0x0c, 0x6a, 0x6f, 0x62, + 0x5f, 0x63, 0x61, 0x70, 0x61, 0x63, 0x69, 0x74, 0x79, 0x18, 0x02, 0x20, 0x01, 0x28, 0x0d, 0x52, + 0x0b, 0x6a, 0x6f, 0x62, 0x43, 0x61, 0x70, 0x61, 0x63, 0x69, 0x74, 0x79, 0x22, 0x62, 0x0a, 0x1a, + 0x50, 0x6f, 0x6c, 0x6c, 0x43, 0x6f, 0x6d, 0x70, 0x61, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x4a, 0x6f, + 0x62, 0x73, 0x52, 0x65, 0x73, 0x70, 0x6f, 0x6e, 0x73, 0x65, 0x12, 0x44, 0x0a, 0x0f, 0x63, 0x6f, + 0x6d, 0x70, 0x61, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x5f, 0x6a, 0x6f, 0x62, 0x73, 0x18, 0x01, 0x20, + 0x03, 0x28, 0x0b, 0x32, 0x1b, 0x2e, 0x63, 0x6f, 0x6d, 0x70, 0x61, 0x63, 0x74, 0x6f, 0x72, 0x2e, + 0x76, 0x31, 0x2e, 0x43, 0x6f, 0x6d, 0x70, 0x61, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x4a, 0x6f, 0x62, + 0x52, 0x0e, 0x63, 0x6f, 0x6d, 0x70, 0x61, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x4a, 0x6f, 0x62, 0x73, + 0x22, 0x16, 0x0a, 0x14, 0x47, 0x65, 0x74, 0x43, 0x6f, 0x6d, 0x70, 0x61, 0x63, 0x74, 0x69, 0x6f, + 0x6e, 0x52, 0x65, 0x71, 0x75, 0x65, 0x73, 0x74, 0x22, 0x5d, 0x0a, 0x15, 0x47, 0x65, 0x74, 0x43, + 0x6f, 0x6d, 0x70, 0x61, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x52, 0x65, 0x73, 0x70, 0x6f, 0x6e, 0x73, + 0x65, 0x12, 0x44, 0x0a, 0x0f, 0x63, 0x6f, 0x6d, 0x70, 0x61, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x5f, + 0x6a, 0x6f, 0x62, 0x73, 0x18, 0x01, 0x20, 0x03, 0x28, 0x0b, 0x32, 0x1b, 0x2e, 0x63, 0x6f, 0x6d, + 0x70, 0x61, 0x63, 0x74, 0x6f, 0x72, 0x2e, 0x76, 0x31, 0x2e, 0x43, 0x6f, 0x6d, 0x70, 0x61, 0x63, + 0x74, 0x69, 0x6f, 0x6e, 0x4a, 0x6f, 0x62, 0x52, 0x0e, 0x63, 0x6f, 0x6d, 0x70, 0x61, 0x63, 0x74, + 0x69, 0x6f, 0x6e, 0x4a, 0x6f, 0x62, 0x73, 0x22, 0xce, 0x02, 0x0a, 0x0d, 0x43, 0x6f, 0x6d, 0x70, + 0x61, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x4a, 0x6f, 0x62, 0x12, 0x12, 0x0a, 0x04, 0x6e, 0x61, 0x6d, + 0x65, 0x18, 0x01, 0x20, 0x01, 0x28, 0x09, 0x52, 0x04, 0x6e, 0x61, 0x6d, 0x65, 0x12, 0x39, 0x0a, + 0x07, 0x6f, 0x70, 0x74, 0x69, 0x6f, 0x6e, 0x73, 0x18, 0x02, 0x20, 0x01, 0x28, 0x0b, 0x32, 0x1f, + 0x2e, 0x63, 0x6f, 0x6d, 0x70, 0x61, 0x63, 0x74, 0x6f, 0x72, 0x2e, 0x76, 0x31, 0x2e, 0x43, 0x6f, + 0x6d, 0x70, 0x61, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x4f, 0x70, 0x74, 0x69, 0x6f, 0x6e, 0x73, 0x52, + 0x07, 0x6f, 0x70, 0x74, 0x69, 0x6f, 0x6e, 0x73, 0x12, 0x2f, 0x0a, 0x06, 0x62, 0x6c, 0x6f, 0x63, + 0x6b, 0x73, 0x18, 0x03, 0x20, 0x03, 0x28, 0x0b, 0x32, 0x17, 0x2e, 0x6d, 0x65, 0x74, 0x61, 0x73, + 0x74, 0x6f, 0x72, 0x65, 0x2e, 0x76, 0x31, 0x2e, 0x42, 0x6c, 0x6f, 0x63, 0x6b, 0x4d, 0x65, 0x74, + 0x61, 0x52, 0x06, 0x62, 0x6c, 0x6f, 0x63, 0x6b, 0x73, 0x12, 0x39, 0x0a, 0x06, 0x73, 0x74, 0x61, + 0x74, 0x75, 0x73, 0x18, 0x04, 0x20, 0x01, 0x28, 0x0b, 0x32, 0x21, 0x2e, 0x63, 0x6f, 0x6d, 0x70, + 0x61, 0x63, 0x74, 0x6f, 0x72, 0x2e, 0x76, 0x31, 0x2e, 0x43, 0x6f, 0x6d, 0x70, 0x61, 0x63, 0x74, + 0x69, 0x6f, 0x6e, 0x4a, 0x6f, 0x62, 0x53, 0x74, 0x61, 0x74, 0x75, 0x73, 0x52, 0x06, 0x73, 0x74, + 0x61, 0x74, 0x75, 0x73, 0x12, 0x24, 0x0a, 0x0e, 0x72, 0x61, 0x66, 0x74, 0x5f, 0x6c, 0x6f, 0x67, + 0x5f, 0x69, 0x6e, 0x64, 0x65, 0x78, 0x18, 0x05, 0x20, 0x01, 0x28, 0x04, 0x52, 0x0c, 0x72, 0x61, + 0x66, 0x74, 0x4c, 0x6f, 0x67, 0x49, 0x6e, 0x64, 0x65, 0x78, 0x12, 0x14, 0x0a, 0x05, 0x73, 0x68, + 0x61, 0x72, 0x64, 0x18, 0x06, 0x20, 0x01, 0x28, 0x0d, 0x52, 0x05, 0x73, 0x68, 0x61, 0x72, 0x64, + 0x12, 0x1b, 0x0a, 0x09, 0x74, 0x65, 0x6e, 0x61, 0x6e, 0x74, 0x5f, 0x69, 0x64, 0x18, 0x07, 0x20, + 0x01, 0x28, 0x09, 0x52, 0x08, 0x74, 0x65, 0x6e, 0x61, 0x6e, 0x74, 0x49, 0x64, 0x12, 0x29, 0x0a, + 0x10, 0x63, 0x6f, 0x6d, 0x70, 0x61, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x5f, 0x6c, 0x65, 0x76, 0x65, + 0x6c, 0x18, 0x08, 0x20, 0x01, 0x28, 0x0d, 0x52, 0x0f, 0x63, 0x6f, 0x6d, 0x70, 0x61, 0x63, 0x74, + 0x69, 0x6f, 0x6e, 0x4c, 0x65, 0x76, 0x65, 0x6c, 0x22, 0x58, 0x0a, 0x11, 0x43, 0x6f, 0x6d, 0x70, + 0x61, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x4f, 0x70, 0x74, 0x69, 0x6f, 0x6e, 0x73, 0x12, 0x43, 0x0a, + 0x1e, 0x73, 0x74, 0x61, 0x74, 0x75, 0x73, 0x5f, 0x75, 0x70, 0x64, 0x61, 0x74, 0x65, 0x5f, 0x69, + 0x6e, 0x74, 0x65, 0x72, 0x76, 0x61, 0x6c, 0x5f, 0x73, 0x65, 0x63, 0x6f, 0x6e, 0x64, 0x73, 0x18, + 0x01, 0x20, 0x01, 0x28, 0x04, 0x52, 0x1b, 0x73, 0x74, 0x61, 0x74, 0x75, 0x73, 0x55, 0x70, 0x64, + 0x61, 0x74, 0x65, 0x49, 0x6e, 0x74, 0x65, 0x72, 0x76, 0x61, 0x6c, 0x53, 0x65, 0x63, 0x6f, 0x6e, + 0x64, 0x73, 0x22, 0x82, 0x02, 0x0a, 0x13, 0x43, 0x6f, 0x6d, 0x70, 0x61, 0x63, 0x74, 0x69, 0x6f, + 0x6e, 0x4a, 0x6f, 0x62, 0x53, 0x74, 0x61, 0x74, 0x75, 0x73, 0x12, 0x19, 0x0a, 0x08, 0x6a, 0x6f, + 0x62, 0x5f, 0x6e, 0x61, 0x6d, 0x65, 0x18, 0x01, 0x20, 0x01, 0x28, 0x09, 0x52, 0x07, 0x6a, 0x6f, + 0x62, 0x4e, 0x61, 0x6d, 0x65, 0x12, 0x36, 0x0a, 0x06, 0x73, 0x74, 0x61, 0x74, 0x75, 0x73, 0x18, + 0x02, 0x20, 0x01, 0x28, 0x0e, 0x32, 0x1e, 0x2e, 0x63, 0x6f, 0x6d, 0x70, 0x61, 0x63, 0x74, 0x6f, + 0x72, 0x2e, 0x76, 0x31, 0x2e, 0x43, 0x6f, 0x6d, 0x70, 0x61, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x53, + 0x74, 0x61, 0x74, 0x75, 0x73, 0x52, 0x06, 0x73, 0x74, 0x61, 0x74, 0x75, 0x73, 0x12, 0x3f, 0x0a, + 0x0d, 0x63, 0x6f, 0x6d, 0x70, 0x6c, 0x65, 0x74, 0x65, 0x64, 0x5f, 0x6a, 0x6f, 0x62, 0x18, 0x03, + 0x20, 0x01, 0x28, 0x0b, 0x32, 0x1a, 0x2e, 0x63, 0x6f, 0x6d, 0x70, 0x61, 0x63, 0x74, 0x6f, 0x72, + 0x2e, 0x76, 0x31, 0x2e, 0x43, 0x6f, 0x6d, 0x70, 0x6c, 0x65, 0x74, 0x65, 0x64, 0x4a, 0x6f, 0x62, + 0x52, 0x0c, 0x63, 0x6f, 0x6d, 0x70, 0x6c, 0x65, 0x74, 0x65, 0x64, 0x4a, 0x6f, 0x62, 0x12, 0x24, + 0x0a, 0x0e, 0x72, 0x61, 0x66, 0x74, 0x5f, 0x6c, 0x6f, 0x67, 0x5f, 0x69, 0x6e, 0x64, 0x65, 0x78, + 0x18, 0x04, 0x20, 0x01, 0x28, 0x04, 0x52, 0x0c, 0x72, 0x61, 0x66, 0x74, 0x4c, 0x6f, 0x67, 0x49, + 0x6e, 0x64, 0x65, 0x78, 0x12, 0x14, 0x0a, 0x05, 0x73, 0x68, 0x61, 0x72, 0x64, 0x18, 0x05, 0x20, + 0x01, 0x28, 0x0d, 0x52, 0x05, 0x73, 0x68, 0x61, 0x72, 0x64, 0x12, 0x1b, 0x0a, 0x09, 0x74, 0x65, + 0x6e, 0x61, 0x6e, 0x74, 0x5f, 0x69, 0x64, 0x18, 0x06, 0x20, 0x01, 0x28, 0x09, 0x52, 0x08, 0x74, + 0x65, 0x6e, 0x61, 0x6e, 0x74, 0x49, 0x64, 0x22, 0x3f, 0x0a, 0x0c, 0x43, 0x6f, 0x6d, 0x70, 0x6c, + 0x65, 0x74, 0x65, 0x64, 0x4a, 0x6f, 0x62, 0x12, 0x2f, 0x0a, 0x06, 0x62, 0x6c, 0x6f, 0x63, 0x6b, + 0x73, 0x18, 0x01, 0x20, 0x03, 0x28, 0x0b, 0x32, 0x17, 0x2e, 0x6d, 0x65, 0x74, 0x61, 0x73, 0x74, + 0x6f, 0x72, 0x65, 0x2e, 0x76, 0x31, 0x2e, 0x42, 0x6c, 0x6f, 0x63, 0x6b, 0x4d, 0x65, 0x74, 0x61, + 0x52, 0x06, 0x62, 0x6c, 0x6f, 0x63, 0x6b, 0x73, 0x2a, 0x96, 0x01, 0x0a, 0x10, 0x43, 0x6f, 0x6d, + 0x70, 0x61, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x53, 0x74, 0x61, 0x74, 0x75, 0x73, 0x12, 0x21, 0x0a, + 0x1d, 0x43, 0x4f, 0x4d, 0x50, 0x41, 0x43, 0x54, 0x49, 0x4f, 0x4e, 0x5f, 0x53, 0x54, 0x41, 0x54, + 0x55, 0x53, 0x5f, 0x55, 0x4e, 0x53, 0x50, 0x45, 0x43, 0x49, 0x46, 0x49, 0x45, 0x44, 0x10, 0x00, + 0x12, 0x21, 0x0a, 0x1d, 0x43, 0x4f, 0x4d, 0x50, 0x41, 0x43, 0x54, 0x49, 0x4f, 0x4e, 0x5f, 0x53, + 0x54, 0x41, 0x54, 0x55, 0x53, 0x5f, 0x49, 0x4e, 0x5f, 0x50, 0x52, 0x4f, 0x47, 0x52, 0x45, 0x53, + 0x53, 0x10, 0x01, 0x12, 0x1d, 0x0a, 0x19, 0x43, 0x4f, 0x4d, 0x50, 0x41, 0x43, 0x54, 0x49, 0x4f, + 0x4e, 0x5f, 0x53, 0x54, 0x41, 0x54, 0x55, 0x53, 0x5f, 0x53, 0x55, 0x43, 0x43, 0x45, 0x53, 0x53, + 0x10, 0x02, 0x12, 0x1d, 0x0a, 0x19, 0x43, 0x4f, 0x4d, 0x50, 0x41, 0x43, 0x54, 0x49, 0x4f, 0x4e, + 0x5f, 0x53, 0x54, 0x41, 0x54, 0x55, 0x53, 0x5f, 0x46, 0x41, 0x49, 0x4c, 0x55, 0x52, 0x45, 0x10, + 0x03, 0x32, 0xde, 0x01, 0x0a, 0x11, 0x43, 0x6f, 0x6d, 0x70, 0x61, 0x63, 0x74, 0x69, 0x6f, 0x6e, + 0x50, 0x6c, 0x61, 0x6e, 0x6e, 0x65, 0x72, 0x12, 0x69, 0x0a, 0x12, 0x50, 0x6f, 0x6c, 0x6c, 0x43, + 0x6f, 0x6d, 0x70, 0x61, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x4a, 0x6f, 0x62, 0x73, 0x12, 0x27, 0x2e, + 0x63, 0x6f, 0x6d, 0x70, 0x61, 0x63, 0x74, 0x6f, 0x72, 0x2e, 0x76, 0x31, 0x2e, 0x50, 0x6f, 0x6c, + 0x6c, 0x43, 0x6f, 0x6d, 0x70, 0x61, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x4a, 0x6f, 0x62, 0x73, 0x52, + 0x65, 0x71, 0x75, 0x65, 0x73, 0x74, 0x1a, 0x28, 0x2e, 0x63, 0x6f, 0x6d, 0x70, 0x61, 0x63, 0x74, + 0x6f, 0x72, 0x2e, 0x76, 0x31, 0x2e, 0x50, 0x6f, 0x6c, 0x6c, 0x43, 0x6f, 0x6d, 0x70, 0x61, 0x63, + 0x74, 0x69, 0x6f, 0x6e, 0x4a, 0x6f, 0x62, 0x73, 0x52, 0x65, 0x73, 0x70, 0x6f, 0x6e, 0x73, 0x65, + 0x22, 0x00, 0x12, 0x5e, 0x0a, 0x11, 0x47, 0x65, 0x74, 0x43, 0x6f, 0x6d, 0x70, 0x61, 0x63, 0x74, + 0x69, 0x6f, 0x6e, 0x4a, 0x6f, 0x62, 0x73, 0x12, 0x22, 0x2e, 0x63, 0x6f, 0x6d, 0x70, 0x61, 0x63, + 0x74, 0x6f, 0x72, 0x2e, 0x76, 0x31, 0x2e, 0x47, 0x65, 0x74, 0x43, 0x6f, 0x6d, 0x70, 0x61, 0x63, + 0x74, 0x69, 0x6f, 0x6e, 0x52, 0x65, 0x71, 0x75, 0x65, 0x73, 0x74, 0x1a, 0x23, 0x2e, 0x63, 0x6f, + 0x6d, 0x70, 0x61, 0x63, 0x74, 0x6f, 0x72, 0x2e, 0x76, 0x31, 0x2e, 0x47, 0x65, 0x74, 0x43, 0x6f, + 0x6d, 0x70, 0x61, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x52, 0x65, 0x73, 0x70, 0x6f, 0x6e, 0x73, 0x65, + 0x22, 0x00, 0x42, 0xbb, 0x01, 0x0a, 0x10, 0x63, 0x6f, 0x6d, 0x2e, 0x63, 0x6f, 0x6d, 0x70, 0x61, + 0x63, 0x74, 0x6f, 0x72, 0x2e, 0x76, 0x31, 0x42, 0x0e, 0x43, 0x6f, 0x6d, 0x70, 0x61, 0x63, 0x74, + 0x6f, 0x72, 0x50, 0x72, 0x6f, 0x74, 0x6f, 0x50, 0x01, 0x5a, 0x46, 0x67, 0x69, 0x74, 0x68, 0x75, + 0x62, 0x2e, 0x63, 0x6f, 0x6d, 0x2f, 0x67, 0x72, 0x61, 0x66, 0x61, 0x6e, 0x61, 0x2f, 0x70, 0x79, + 0x72, 0x6f, 0x73, 0x63, 0x6f, 0x70, 0x65, 0x2f, 0x61, 0x70, 0x69, 0x2f, 0x67, 0x65, 0x6e, 0x2f, + 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x2f, 0x67, 0x6f, 0x2f, 0x63, 0x6f, 0x6d, 0x70, 0x61, 0x63, 0x74, + 0x6f, 0x72, 0x2f, 0x76, 0x31, 0x3b, 0x63, 0x6f, 0x6d, 0x70, 0x61, 0x63, 0x74, 0x6f, 0x72, 0x76, + 0x31, 0xa2, 0x02, 0x03, 0x43, 0x58, 0x58, 0xaa, 0x02, 0x0c, 0x43, 0x6f, 0x6d, 0x70, 0x61, 0x63, + 0x74, 0x6f, 0x72, 0x2e, 0x56, 0x31, 0xca, 0x02, 0x0c, 0x43, 0x6f, 0x6d, 0x70, 0x61, 0x63, 0x74, + 0x6f, 0x72, 0x5c, 0x56, 0x31, 0xe2, 0x02, 0x18, 0x43, 0x6f, 0x6d, 0x70, 0x61, 0x63, 0x74, 0x6f, + 0x72, 0x5c, 0x56, 0x31, 0x5c, 0x47, 0x50, 0x42, 0x4d, 0x65, 0x74, 0x61, 0x64, 0x61, 0x74, 0x61, + 0xea, 0x02, 0x0d, 0x43, 0x6f, 0x6d, 0x70, 0x61, 0x63, 0x74, 0x6f, 0x72, 0x3a, 0x3a, 0x56, 0x31, + 0x62, 0x06, 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x33, +} + +var ( + file_compactor_v1_compactor_proto_rawDescOnce sync.Once + file_compactor_v1_compactor_proto_rawDescData = file_compactor_v1_compactor_proto_rawDesc +) + +func file_compactor_v1_compactor_proto_rawDescGZIP() []byte { + file_compactor_v1_compactor_proto_rawDescOnce.Do(func() { + file_compactor_v1_compactor_proto_rawDescData = protoimpl.X.CompressGZIP(file_compactor_v1_compactor_proto_rawDescData) + }) + return file_compactor_v1_compactor_proto_rawDescData +} + +var file_compactor_v1_compactor_proto_enumTypes = make([]protoimpl.EnumInfo, 1) +var file_compactor_v1_compactor_proto_msgTypes = make([]protoimpl.MessageInfo, 8) +var file_compactor_v1_compactor_proto_goTypes = []any{ + (CompactionStatus)(0), // 0: compactor.v1.CompactionStatus + (*PollCompactionJobsRequest)(nil), // 1: compactor.v1.PollCompactionJobsRequest + (*PollCompactionJobsResponse)(nil), // 2: compactor.v1.PollCompactionJobsResponse + (*GetCompactionRequest)(nil), // 3: compactor.v1.GetCompactionRequest + (*GetCompactionResponse)(nil), // 4: compactor.v1.GetCompactionResponse + (*CompactionJob)(nil), // 5: compactor.v1.CompactionJob + (*CompactionOptions)(nil), // 6: compactor.v1.CompactionOptions + (*CompactionJobStatus)(nil), // 7: compactor.v1.CompactionJobStatus + (*CompletedJob)(nil), // 8: compactor.v1.CompletedJob + (*v1.BlockMeta)(nil), // 9: metastore.v1.BlockMeta +} +var file_compactor_v1_compactor_proto_depIdxs = []int32{ + 7, // 0: compactor.v1.PollCompactionJobsRequest.job_status_updates:type_name -> compactor.v1.CompactionJobStatus + 5, // 1: compactor.v1.PollCompactionJobsResponse.compaction_jobs:type_name -> compactor.v1.CompactionJob + 5, // 2: compactor.v1.GetCompactionResponse.compaction_jobs:type_name -> compactor.v1.CompactionJob + 6, // 3: compactor.v1.CompactionJob.options:type_name -> compactor.v1.CompactionOptions + 9, // 4: compactor.v1.CompactionJob.blocks:type_name -> metastore.v1.BlockMeta + 7, // 5: compactor.v1.CompactionJob.status:type_name -> compactor.v1.CompactionJobStatus + 0, // 6: compactor.v1.CompactionJobStatus.status:type_name -> compactor.v1.CompactionStatus + 8, // 7: compactor.v1.CompactionJobStatus.completed_job:type_name -> compactor.v1.CompletedJob + 9, // 8: compactor.v1.CompletedJob.blocks:type_name -> metastore.v1.BlockMeta + 1, // 9: compactor.v1.CompactionPlanner.PollCompactionJobs:input_type -> compactor.v1.PollCompactionJobsRequest + 3, // 10: compactor.v1.CompactionPlanner.GetCompactionJobs:input_type -> compactor.v1.GetCompactionRequest + 2, // 11: compactor.v1.CompactionPlanner.PollCompactionJobs:output_type -> compactor.v1.PollCompactionJobsResponse + 4, // 12: compactor.v1.CompactionPlanner.GetCompactionJobs:output_type -> compactor.v1.GetCompactionResponse + 11, // [11:13] is the sub-list for method output_type + 9, // [9:11] is the sub-list for method input_type + 9, // [9:9] is the sub-list for extension type_name + 9, // [9:9] is the sub-list for extension extendee + 0, // [0:9] is the sub-list for field type_name +} + +func init() { file_compactor_v1_compactor_proto_init() } +func file_compactor_v1_compactor_proto_init() { + if File_compactor_v1_compactor_proto != nil { + return + } + if !protoimpl.UnsafeEnabled { + file_compactor_v1_compactor_proto_msgTypes[0].Exporter = func(v any, i int) any { + switch v := v.(*PollCompactionJobsRequest); i { + case 0: + return &v.state + case 1: + return &v.sizeCache + case 2: + return &v.unknownFields + default: + return nil + } + } + file_compactor_v1_compactor_proto_msgTypes[1].Exporter = func(v any, i int) any { + switch v := v.(*PollCompactionJobsResponse); i { + case 0: + return &v.state + case 1: + return &v.sizeCache + case 2: + return &v.unknownFields + default: + return nil + } + } + file_compactor_v1_compactor_proto_msgTypes[2].Exporter = func(v any, i int) any { + switch v := v.(*GetCompactionRequest); i { + case 0: + return &v.state + case 1: + return &v.sizeCache + case 2: + return &v.unknownFields + default: + return nil + } + } + file_compactor_v1_compactor_proto_msgTypes[3].Exporter = func(v any, i int) any { + switch v := v.(*GetCompactionResponse); i { + case 0: + return &v.state + case 1: + return &v.sizeCache + case 2: + return &v.unknownFields + default: + return nil + } + } + file_compactor_v1_compactor_proto_msgTypes[4].Exporter = func(v any, i int) any { + switch v := v.(*CompactionJob); i { + case 0: + return &v.state + case 1: + return &v.sizeCache + case 2: + return &v.unknownFields + default: + return nil + } + } + file_compactor_v1_compactor_proto_msgTypes[5].Exporter = func(v any, i int) any { + switch v := v.(*CompactionOptions); i { + case 0: + return &v.state + case 1: + return &v.sizeCache + case 2: + return &v.unknownFields + default: + return nil + } + } + file_compactor_v1_compactor_proto_msgTypes[6].Exporter = func(v any, i int) any { + switch v := v.(*CompactionJobStatus); i { + case 0: + return &v.state + case 1: + return &v.sizeCache + case 2: + return &v.unknownFields + default: + return nil + } + } + file_compactor_v1_compactor_proto_msgTypes[7].Exporter = func(v any, i int) any { + switch v := v.(*CompletedJob); i { + case 0: + return &v.state + case 1: + return &v.sizeCache + case 2: + return &v.unknownFields + default: + return nil + } + } + } + type x struct{} + out := protoimpl.TypeBuilder{ + File: protoimpl.DescBuilder{ + GoPackagePath: reflect.TypeOf(x{}).PkgPath(), + RawDescriptor: file_compactor_v1_compactor_proto_rawDesc, + NumEnums: 1, + NumMessages: 8, + NumExtensions: 0, + NumServices: 1, + }, + GoTypes: file_compactor_v1_compactor_proto_goTypes, + DependencyIndexes: file_compactor_v1_compactor_proto_depIdxs, + EnumInfos: file_compactor_v1_compactor_proto_enumTypes, + MessageInfos: file_compactor_v1_compactor_proto_msgTypes, + }.Build() + File_compactor_v1_compactor_proto = out.File + file_compactor_v1_compactor_proto_rawDesc = nil + file_compactor_v1_compactor_proto_goTypes = nil + file_compactor_v1_compactor_proto_depIdxs = nil +} diff --git a/api/gen/proto/go/compactor/v1/compactor_vtproto.pb.go b/api/gen/proto/go/compactor/v1/compactor_vtproto.pb.go new file mode 100644 index 0000000000..6577c16531 --- /dev/null +++ b/api/gen/proto/go/compactor/v1/compactor_vtproto.pb.go @@ -0,0 +1,2212 @@ +// Code generated by protoc-gen-go-vtproto. DO NOT EDIT. +// protoc-gen-go-vtproto version: v0.6.0 +// source: compactor/v1/compactor.proto + +package compactorv1 + +import ( + context "context" + fmt "fmt" + v1 "github.com/grafana/pyroscope/api/gen/proto/go/metastore/v1" + protohelpers "github.com/planetscale/vtprotobuf/protohelpers" + grpc "google.golang.org/grpc" + codes "google.golang.org/grpc/codes" + status "google.golang.org/grpc/status" + proto "google.golang.org/protobuf/proto" + protoimpl "google.golang.org/protobuf/runtime/protoimpl" + io "io" +) + +const ( + // Verify that this generated code is sufficiently up-to-date. + _ = protoimpl.EnforceVersion(20 - protoimpl.MinVersion) + // Verify that runtime/protoimpl is sufficiently up-to-date. + _ = protoimpl.EnforceVersion(protoimpl.MaxVersion - 20) +) + +func (m *PollCompactionJobsRequest) CloneVT() *PollCompactionJobsRequest { + if m == nil { + return (*PollCompactionJobsRequest)(nil) + } + r := new(PollCompactionJobsRequest) + r.JobCapacity = m.JobCapacity + if rhs := m.JobStatusUpdates; rhs != nil { + tmpContainer := make([]*CompactionJobStatus, len(rhs)) + for k, v := range rhs { + tmpContainer[k] = v.CloneVT() + } + r.JobStatusUpdates = tmpContainer + } + if len(m.unknownFields) > 0 { + r.unknownFields = make([]byte, len(m.unknownFields)) + copy(r.unknownFields, m.unknownFields) + } + return r +} + +func (m *PollCompactionJobsRequest) CloneMessageVT() proto.Message { + return m.CloneVT() +} + +func (m *PollCompactionJobsResponse) CloneVT() *PollCompactionJobsResponse { + if m == nil { + return (*PollCompactionJobsResponse)(nil) + } + r := new(PollCompactionJobsResponse) + if rhs := m.CompactionJobs; rhs != nil { + tmpContainer := make([]*CompactionJob, len(rhs)) + for k, v := range rhs { + tmpContainer[k] = v.CloneVT() + } + r.CompactionJobs = tmpContainer + } + if len(m.unknownFields) > 0 { + r.unknownFields = make([]byte, len(m.unknownFields)) + copy(r.unknownFields, m.unknownFields) + } + return r +} + +func (m *PollCompactionJobsResponse) CloneMessageVT() proto.Message { + return m.CloneVT() +} + +func (m *GetCompactionRequest) CloneVT() *GetCompactionRequest { + if m == nil { + return (*GetCompactionRequest)(nil) + } + r := new(GetCompactionRequest) + if len(m.unknownFields) > 0 { + r.unknownFields = make([]byte, len(m.unknownFields)) + copy(r.unknownFields, m.unknownFields) + } + return r +} + +func (m *GetCompactionRequest) CloneMessageVT() proto.Message { + return m.CloneVT() +} + +func (m *GetCompactionResponse) CloneVT() *GetCompactionResponse { + if m == nil { + return (*GetCompactionResponse)(nil) + } + r := new(GetCompactionResponse) + if rhs := m.CompactionJobs; rhs != nil { + tmpContainer := make([]*CompactionJob, len(rhs)) + for k, v := range rhs { + tmpContainer[k] = v.CloneVT() + } + r.CompactionJobs = tmpContainer + } + if len(m.unknownFields) > 0 { + r.unknownFields = make([]byte, len(m.unknownFields)) + copy(r.unknownFields, m.unknownFields) + } + return r +} + +func (m *GetCompactionResponse) CloneMessageVT() proto.Message { + return m.CloneVT() +} + +func (m *CompactionJob) CloneVT() *CompactionJob { + if m == nil { + return (*CompactionJob)(nil) + } + r := new(CompactionJob) + r.Name = m.Name + r.Options = m.Options.CloneVT() + r.Status = m.Status.CloneVT() + r.RaftLogIndex = m.RaftLogIndex + r.Shard = m.Shard + r.TenantId = m.TenantId + r.CompactionLevel = m.CompactionLevel + if rhs := m.Blocks; rhs != nil { + tmpContainer := make([]*v1.BlockMeta, len(rhs)) + for k, v := range rhs { + if vtpb, ok := interface{}(v).(interface{ CloneVT() *v1.BlockMeta }); ok { + tmpContainer[k] = vtpb.CloneVT() + } else { + tmpContainer[k] = proto.Clone(v).(*v1.BlockMeta) + } + } + r.Blocks = tmpContainer + } + if len(m.unknownFields) > 0 { + r.unknownFields = make([]byte, len(m.unknownFields)) + copy(r.unknownFields, m.unknownFields) + } + return r +} + +func (m *CompactionJob) CloneMessageVT() proto.Message { + return m.CloneVT() +} + +func (m *CompactionOptions) CloneVT() *CompactionOptions { + if m == nil { + return (*CompactionOptions)(nil) + } + r := new(CompactionOptions) + r.StatusUpdateIntervalSeconds = m.StatusUpdateIntervalSeconds + if len(m.unknownFields) > 0 { + r.unknownFields = make([]byte, len(m.unknownFields)) + copy(r.unknownFields, m.unknownFields) + } + return r +} + +func (m *CompactionOptions) CloneMessageVT() proto.Message { + return m.CloneVT() +} + +func (m *CompactionJobStatus) CloneVT() *CompactionJobStatus { + if m == nil { + return (*CompactionJobStatus)(nil) + } + r := new(CompactionJobStatus) + r.JobName = m.JobName + r.Status = m.Status + r.CompletedJob = m.CompletedJob.CloneVT() + r.RaftLogIndex = m.RaftLogIndex + r.Shard = m.Shard + r.TenantId = m.TenantId + if len(m.unknownFields) > 0 { + r.unknownFields = make([]byte, len(m.unknownFields)) + copy(r.unknownFields, m.unknownFields) + } + return r +} + +func (m *CompactionJobStatus) CloneMessageVT() proto.Message { + return m.CloneVT() +} + +func (m *CompletedJob) CloneVT() *CompletedJob { + if m == nil { + return (*CompletedJob)(nil) + } + r := new(CompletedJob) + if rhs := m.Blocks; rhs != nil { + tmpContainer := make([]*v1.BlockMeta, len(rhs)) + for k, v := range rhs { + if vtpb, ok := interface{}(v).(interface{ CloneVT() *v1.BlockMeta }); ok { + tmpContainer[k] = vtpb.CloneVT() + } else { + tmpContainer[k] = proto.Clone(v).(*v1.BlockMeta) + } + } + r.Blocks = tmpContainer + } + if len(m.unknownFields) > 0 { + r.unknownFields = make([]byte, len(m.unknownFields)) + copy(r.unknownFields, m.unknownFields) + } + return r +} + +func (m *CompletedJob) CloneMessageVT() proto.Message { + return m.CloneVT() +} + +func (this *PollCompactionJobsRequest) EqualVT(that *PollCompactionJobsRequest) bool { + if this == that { + return true + } else if this == nil || that == nil { + return false + } + if len(this.JobStatusUpdates) != len(that.JobStatusUpdates) { + return false + } + for i, vx := range this.JobStatusUpdates { + vy := that.JobStatusUpdates[i] + if p, q := vx, vy; p != q { + if p == nil { + p = &CompactionJobStatus{} + } + if q == nil { + q = &CompactionJobStatus{} + } + if !p.EqualVT(q) { + return false + } + } + } + if this.JobCapacity != that.JobCapacity { + return false + } + return string(this.unknownFields) == string(that.unknownFields) +} + +func (this *PollCompactionJobsRequest) EqualMessageVT(thatMsg proto.Message) bool { + that, ok := thatMsg.(*PollCompactionJobsRequest) + if !ok { + return false + } + return this.EqualVT(that) +} +func (this *PollCompactionJobsResponse) EqualVT(that *PollCompactionJobsResponse) bool { + if this == that { + return true + } else if this == nil || that == nil { + return false + } + if len(this.CompactionJobs) != len(that.CompactionJobs) { + return false + } + for i, vx := range this.CompactionJobs { + vy := that.CompactionJobs[i] + if p, q := vx, vy; p != q { + if p == nil { + p = &CompactionJob{} + } + if q == nil { + q = &CompactionJob{} + } + if !p.EqualVT(q) { + return false + } + } + } + return string(this.unknownFields) == string(that.unknownFields) +} + +func (this *PollCompactionJobsResponse) EqualMessageVT(thatMsg proto.Message) bool { + that, ok := thatMsg.(*PollCompactionJobsResponse) + if !ok { + return false + } + return this.EqualVT(that) +} +func (this *GetCompactionRequest) EqualVT(that *GetCompactionRequest) bool { + if this == that { + return true + } else if this == nil || that == nil { + return false + } + return string(this.unknownFields) == string(that.unknownFields) +} + +func (this *GetCompactionRequest) EqualMessageVT(thatMsg proto.Message) bool { + that, ok := thatMsg.(*GetCompactionRequest) + if !ok { + return false + } + return this.EqualVT(that) +} +func (this *GetCompactionResponse) EqualVT(that *GetCompactionResponse) bool { + if this == that { + return true + } else if this == nil || that == nil { + return false + } + if len(this.CompactionJobs) != len(that.CompactionJobs) { + return false + } + for i, vx := range this.CompactionJobs { + vy := that.CompactionJobs[i] + if p, q := vx, vy; p != q { + if p == nil { + p = &CompactionJob{} + } + if q == nil { + q = &CompactionJob{} + } + if !p.EqualVT(q) { + return false + } + } + } + return string(this.unknownFields) == string(that.unknownFields) +} + +func (this *GetCompactionResponse) EqualMessageVT(thatMsg proto.Message) bool { + that, ok := thatMsg.(*GetCompactionResponse) + if !ok { + return false + } + return this.EqualVT(that) +} +func (this *CompactionJob) EqualVT(that *CompactionJob) bool { + if this == that { + return true + } else if this == nil || that == nil { + return false + } + if this.Name != that.Name { + return false + } + if !this.Options.EqualVT(that.Options) { + return false + } + if len(this.Blocks) != len(that.Blocks) { + return false + } + for i, vx := range this.Blocks { + vy := that.Blocks[i] + if p, q := vx, vy; p != q { + if p == nil { + p = &v1.BlockMeta{} + } + if q == nil { + q = &v1.BlockMeta{} + } + if equal, ok := interface{}(p).(interface{ EqualVT(*v1.BlockMeta) bool }); ok { + if !equal.EqualVT(q) { + return false + } + } else if !proto.Equal(p, q) { + return false + } + } + } + if !this.Status.EqualVT(that.Status) { + return false + } + if this.RaftLogIndex != that.RaftLogIndex { + return false + } + if this.Shard != that.Shard { + return false + } + if this.TenantId != that.TenantId { + return false + } + if this.CompactionLevel != that.CompactionLevel { + return false + } + return string(this.unknownFields) == string(that.unknownFields) +} + +func (this *CompactionJob) EqualMessageVT(thatMsg proto.Message) bool { + that, ok := thatMsg.(*CompactionJob) + if !ok { + return false + } + return this.EqualVT(that) +} +func (this *CompactionOptions) EqualVT(that *CompactionOptions) bool { + if this == that { + return true + } else if this == nil || that == nil { + return false + } + if this.StatusUpdateIntervalSeconds != that.StatusUpdateIntervalSeconds { + return false + } + return string(this.unknownFields) == string(that.unknownFields) +} + +func (this *CompactionOptions) EqualMessageVT(thatMsg proto.Message) bool { + that, ok := thatMsg.(*CompactionOptions) + if !ok { + return false + } + return this.EqualVT(that) +} +func (this *CompactionJobStatus) EqualVT(that *CompactionJobStatus) bool { + if this == that { + return true + } else if this == nil || that == nil { + return false + } + if this.JobName != that.JobName { + return false + } + if this.Status != that.Status { + return false + } + if !this.CompletedJob.EqualVT(that.CompletedJob) { + return false + } + if this.RaftLogIndex != that.RaftLogIndex { + return false + } + if this.Shard != that.Shard { + return false + } + if this.TenantId != that.TenantId { + return false + } + return string(this.unknownFields) == string(that.unknownFields) +} + +func (this *CompactionJobStatus) EqualMessageVT(thatMsg proto.Message) bool { + that, ok := thatMsg.(*CompactionJobStatus) + if !ok { + return false + } + return this.EqualVT(that) +} +func (this *CompletedJob) EqualVT(that *CompletedJob) bool { + if this == that { + return true + } else if this == nil || that == nil { + return false + } + if len(this.Blocks) != len(that.Blocks) { + return false + } + for i, vx := range this.Blocks { + vy := that.Blocks[i] + if p, q := vx, vy; p != q { + if p == nil { + p = &v1.BlockMeta{} + } + if q == nil { + q = &v1.BlockMeta{} + } + if equal, ok := interface{}(p).(interface{ EqualVT(*v1.BlockMeta) bool }); ok { + if !equal.EqualVT(q) { + return false + } + } else if !proto.Equal(p, q) { + return false + } + } + } + return string(this.unknownFields) == string(that.unknownFields) +} + +func (this *CompletedJob) EqualMessageVT(thatMsg proto.Message) bool { + that, ok := thatMsg.(*CompletedJob) + if !ok { + return false + } + return this.EqualVT(that) +} + +// This is a compile-time assertion to ensure that this generated file +// is compatible with the grpc package it is being compiled against. +// Requires gRPC-Go v1.32.0 or later. +const _ = grpc.SupportPackageIsVersion7 + +// CompactionPlannerClient is the client API for CompactionPlanner service. +// +// For semantics around ctx use and closing/ending streaming RPCs, please refer to https://pkg.go.dev/google.golang.org/grpc/?tab=doc#ClientConn.NewStream. +type CompactionPlannerClient interface { + // Used to both retrieve jobs and update the jobs status at the same time. + PollCompactionJobs(ctx context.Context, in *PollCompactionJobsRequest, opts ...grpc.CallOption) (*PollCompactionJobsResponse, error) + // Used for admin purposes only. + GetCompactionJobs(ctx context.Context, in *GetCompactionRequest, opts ...grpc.CallOption) (*GetCompactionResponse, error) +} + +type compactionPlannerClient struct { + cc grpc.ClientConnInterface +} + +func NewCompactionPlannerClient(cc grpc.ClientConnInterface) CompactionPlannerClient { + return &compactionPlannerClient{cc} +} + +func (c *compactionPlannerClient) PollCompactionJobs(ctx context.Context, in *PollCompactionJobsRequest, opts ...grpc.CallOption) (*PollCompactionJobsResponse, error) { + out := new(PollCompactionJobsResponse) + err := c.cc.Invoke(ctx, "/compactor.v1.CompactionPlanner/PollCompactionJobs", in, out, opts...) + if err != nil { + return nil, err + } + return out, nil +} + +func (c *compactionPlannerClient) GetCompactionJobs(ctx context.Context, in *GetCompactionRequest, opts ...grpc.CallOption) (*GetCompactionResponse, error) { + out := new(GetCompactionResponse) + err := c.cc.Invoke(ctx, "/compactor.v1.CompactionPlanner/GetCompactionJobs", in, out, opts...) + if err != nil { + return nil, err + } + return out, nil +} + +// CompactionPlannerServer is the server API for CompactionPlanner service. +// All implementations must embed UnimplementedCompactionPlannerServer +// for forward compatibility +type CompactionPlannerServer interface { + // Used to both retrieve jobs and update the jobs status at the same time. + PollCompactionJobs(context.Context, *PollCompactionJobsRequest) (*PollCompactionJobsResponse, error) + // Used for admin purposes only. + GetCompactionJobs(context.Context, *GetCompactionRequest) (*GetCompactionResponse, error) + mustEmbedUnimplementedCompactionPlannerServer() +} + +// UnimplementedCompactionPlannerServer must be embedded to have forward compatible implementations. +type UnimplementedCompactionPlannerServer struct { +} + +func (UnimplementedCompactionPlannerServer) PollCompactionJobs(context.Context, *PollCompactionJobsRequest) (*PollCompactionJobsResponse, error) { + return nil, status.Errorf(codes.Unimplemented, "method PollCompactionJobs not implemented") +} +func (UnimplementedCompactionPlannerServer) GetCompactionJobs(context.Context, *GetCompactionRequest) (*GetCompactionResponse, error) { + return nil, status.Errorf(codes.Unimplemented, "method GetCompactionJobs not implemented") +} +func (UnimplementedCompactionPlannerServer) mustEmbedUnimplementedCompactionPlannerServer() {} + +// UnsafeCompactionPlannerServer may be embedded to opt out of forward compatibility for this service. +// Use of this interface is not recommended, as added methods to CompactionPlannerServer will +// result in compilation errors. +type UnsafeCompactionPlannerServer interface { + mustEmbedUnimplementedCompactionPlannerServer() +} + +func RegisterCompactionPlannerServer(s grpc.ServiceRegistrar, srv CompactionPlannerServer) { + s.RegisterService(&CompactionPlanner_ServiceDesc, srv) +} + +func _CompactionPlanner_PollCompactionJobs_Handler(srv interface{}, ctx context.Context, dec func(interface{}) error, interceptor grpc.UnaryServerInterceptor) (interface{}, error) { + in := new(PollCompactionJobsRequest) + if err := dec(in); err != nil { + return nil, err + } + if interceptor == nil { + return srv.(CompactionPlannerServer).PollCompactionJobs(ctx, in) + } + info := &grpc.UnaryServerInfo{ + Server: srv, + FullMethod: "/compactor.v1.CompactionPlanner/PollCompactionJobs", + } + handler := func(ctx context.Context, req interface{}) (interface{}, error) { + return srv.(CompactionPlannerServer).PollCompactionJobs(ctx, req.(*PollCompactionJobsRequest)) + } + return interceptor(ctx, in, info, handler) +} + +func _CompactionPlanner_GetCompactionJobs_Handler(srv interface{}, ctx context.Context, dec func(interface{}) error, interceptor grpc.UnaryServerInterceptor) (interface{}, error) { + in := new(GetCompactionRequest) + if err := dec(in); err != nil { + return nil, err + } + if interceptor == nil { + return srv.(CompactionPlannerServer).GetCompactionJobs(ctx, in) + } + info := &grpc.UnaryServerInfo{ + Server: srv, + FullMethod: "/compactor.v1.CompactionPlanner/GetCompactionJobs", + } + handler := func(ctx context.Context, req interface{}) (interface{}, error) { + return srv.(CompactionPlannerServer).GetCompactionJobs(ctx, req.(*GetCompactionRequest)) + } + return interceptor(ctx, in, info, handler) +} + +// CompactionPlanner_ServiceDesc is the grpc.ServiceDesc for CompactionPlanner service. +// It's only intended for direct use with grpc.RegisterService, +// and not to be introspected or modified (even as a copy) +var CompactionPlanner_ServiceDesc = grpc.ServiceDesc{ + ServiceName: "compactor.v1.CompactionPlanner", + HandlerType: (*CompactionPlannerServer)(nil), + Methods: []grpc.MethodDesc{ + { + MethodName: "PollCompactionJobs", + Handler: _CompactionPlanner_PollCompactionJobs_Handler, + }, + { + MethodName: "GetCompactionJobs", + Handler: _CompactionPlanner_GetCompactionJobs_Handler, + }, + }, + Streams: []grpc.StreamDesc{}, + Metadata: "compactor/v1/compactor.proto", +} + +func (m *PollCompactionJobsRequest) MarshalVT() (dAtA []byte, err error) { + if m == nil { + return nil, nil + } + size := m.SizeVT() + dAtA = make([]byte, size) + n, err := m.MarshalToSizedBufferVT(dAtA[:size]) + if err != nil { + return nil, err + } + return dAtA[:n], nil +} + +func (m *PollCompactionJobsRequest) MarshalToVT(dAtA []byte) (int, error) { + size := m.SizeVT() + return m.MarshalToSizedBufferVT(dAtA[:size]) +} + +func (m *PollCompactionJobsRequest) MarshalToSizedBufferVT(dAtA []byte) (int, error) { + if m == nil { + return 0, nil + } + i := len(dAtA) + _ = i + var l int + _ = l + if m.unknownFields != nil { + i -= len(m.unknownFields) + copy(dAtA[i:], m.unknownFields) + } + if m.JobCapacity != 0 { + i = protohelpers.EncodeVarint(dAtA, i, uint64(m.JobCapacity)) + i-- + dAtA[i] = 0x10 + } + if len(m.JobStatusUpdates) > 0 { + for iNdEx := len(m.JobStatusUpdates) - 1; iNdEx >= 0; iNdEx-- { + size, err := m.JobStatusUpdates[iNdEx].MarshalToSizedBufferVT(dAtA[:i]) + if err != nil { + return 0, err + } + i -= size + i = protohelpers.EncodeVarint(dAtA, i, uint64(size)) + i-- + dAtA[i] = 0xa + } + } + return len(dAtA) - i, nil +} + +func (m *PollCompactionJobsResponse) MarshalVT() (dAtA []byte, err error) { + if m == nil { + return nil, nil + } + size := m.SizeVT() + dAtA = make([]byte, size) + n, err := m.MarshalToSizedBufferVT(dAtA[:size]) + if err != nil { + return nil, err + } + return dAtA[:n], nil +} + +func (m *PollCompactionJobsResponse) MarshalToVT(dAtA []byte) (int, error) { + size := m.SizeVT() + return m.MarshalToSizedBufferVT(dAtA[:size]) +} + +func (m *PollCompactionJobsResponse) MarshalToSizedBufferVT(dAtA []byte) (int, error) { + if m == nil { + return 0, nil + } + i := len(dAtA) + _ = i + var l int + _ = l + if m.unknownFields != nil { + i -= len(m.unknownFields) + copy(dAtA[i:], m.unknownFields) + } + if len(m.CompactionJobs) > 0 { + for iNdEx := len(m.CompactionJobs) - 1; iNdEx >= 0; iNdEx-- { + size, err := m.CompactionJobs[iNdEx].MarshalToSizedBufferVT(dAtA[:i]) + if err != nil { + return 0, err + } + i -= size + i = protohelpers.EncodeVarint(dAtA, i, uint64(size)) + i-- + dAtA[i] = 0xa + } + } + return len(dAtA) - i, nil +} + +func (m *GetCompactionRequest) MarshalVT() (dAtA []byte, err error) { + if m == nil { + return nil, nil + } + size := m.SizeVT() + dAtA = make([]byte, size) + n, err := m.MarshalToSizedBufferVT(dAtA[:size]) + if err != nil { + return nil, err + } + return dAtA[:n], nil +} + +func (m *GetCompactionRequest) MarshalToVT(dAtA []byte) (int, error) { + size := m.SizeVT() + return m.MarshalToSizedBufferVT(dAtA[:size]) +} + +func (m *GetCompactionRequest) MarshalToSizedBufferVT(dAtA []byte) (int, error) { + if m == nil { + return 0, nil + } + i := len(dAtA) + _ = i + var l int + _ = l + if m.unknownFields != nil { + i -= len(m.unknownFields) + copy(dAtA[i:], m.unknownFields) + } + return len(dAtA) - i, nil +} + +func (m *GetCompactionResponse) MarshalVT() (dAtA []byte, err error) { + if m == nil { + return nil, nil + } + size := m.SizeVT() + dAtA = make([]byte, size) + n, err := m.MarshalToSizedBufferVT(dAtA[:size]) + if err != nil { + return nil, err + } + return dAtA[:n], nil +} + +func (m *GetCompactionResponse) MarshalToVT(dAtA []byte) (int, error) { + size := m.SizeVT() + return m.MarshalToSizedBufferVT(dAtA[:size]) +} + +func (m *GetCompactionResponse) MarshalToSizedBufferVT(dAtA []byte) (int, error) { + if m == nil { + return 0, nil + } + i := len(dAtA) + _ = i + var l int + _ = l + if m.unknownFields != nil { + i -= len(m.unknownFields) + copy(dAtA[i:], m.unknownFields) + } + if len(m.CompactionJobs) > 0 { + for iNdEx := len(m.CompactionJobs) - 1; iNdEx >= 0; iNdEx-- { + size, err := m.CompactionJobs[iNdEx].MarshalToSizedBufferVT(dAtA[:i]) + if err != nil { + return 0, err + } + i -= size + i = protohelpers.EncodeVarint(dAtA, i, uint64(size)) + i-- + dAtA[i] = 0xa + } + } + return len(dAtA) - i, nil +} + +func (m *CompactionJob) MarshalVT() (dAtA []byte, err error) { + if m == nil { + return nil, nil + } + size := m.SizeVT() + dAtA = make([]byte, size) + n, err := m.MarshalToSizedBufferVT(dAtA[:size]) + if err != nil { + return nil, err + } + return dAtA[:n], nil +} + +func (m *CompactionJob) MarshalToVT(dAtA []byte) (int, error) { + size := m.SizeVT() + return m.MarshalToSizedBufferVT(dAtA[:size]) +} + +func (m *CompactionJob) MarshalToSizedBufferVT(dAtA []byte) (int, error) { + if m == nil { + return 0, nil + } + i := len(dAtA) + _ = i + var l int + _ = l + if m.unknownFields != nil { + i -= len(m.unknownFields) + copy(dAtA[i:], m.unknownFields) + } + if m.CompactionLevel != 0 { + i = protohelpers.EncodeVarint(dAtA, i, uint64(m.CompactionLevel)) + i-- + dAtA[i] = 0x40 + } + if len(m.TenantId) > 0 { + i -= len(m.TenantId) + copy(dAtA[i:], m.TenantId) + i = protohelpers.EncodeVarint(dAtA, i, uint64(len(m.TenantId))) + i-- + dAtA[i] = 0x3a + } + if m.Shard != 0 { + i = protohelpers.EncodeVarint(dAtA, i, uint64(m.Shard)) + i-- + dAtA[i] = 0x30 + } + if m.RaftLogIndex != 0 { + i = protohelpers.EncodeVarint(dAtA, i, uint64(m.RaftLogIndex)) + i-- + dAtA[i] = 0x28 + } + if m.Status != nil { + size, err := m.Status.MarshalToSizedBufferVT(dAtA[:i]) + if err != nil { + return 0, err + } + i -= size + i = protohelpers.EncodeVarint(dAtA, i, uint64(size)) + i-- + dAtA[i] = 0x22 + } + if len(m.Blocks) > 0 { + for iNdEx := len(m.Blocks) - 1; iNdEx >= 0; iNdEx-- { + if vtmsg, ok := interface{}(m.Blocks[iNdEx]).(interface { + MarshalToSizedBufferVT([]byte) (int, error) + }); ok { + size, err := vtmsg.MarshalToSizedBufferVT(dAtA[:i]) + if err != nil { + return 0, err + } + i -= size + i = protohelpers.EncodeVarint(dAtA, i, uint64(size)) + } else { + encoded, err := proto.Marshal(m.Blocks[iNdEx]) + if err != nil { + return 0, err + } + i -= len(encoded) + copy(dAtA[i:], encoded) + i = protohelpers.EncodeVarint(dAtA, i, uint64(len(encoded))) + } + i-- + dAtA[i] = 0x1a + } + } + if m.Options != nil { + size, err := m.Options.MarshalToSizedBufferVT(dAtA[:i]) + if err != nil { + return 0, err + } + i -= size + i = protohelpers.EncodeVarint(dAtA, i, uint64(size)) + i-- + dAtA[i] = 0x12 + } + if len(m.Name) > 0 { + i -= len(m.Name) + copy(dAtA[i:], m.Name) + i = protohelpers.EncodeVarint(dAtA, i, uint64(len(m.Name))) + i-- + dAtA[i] = 0xa + } + return len(dAtA) - i, nil +} + +func (m *CompactionOptions) MarshalVT() (dAtA []byte, err error) { + if m == nil { + return nil, nil + } + size := m.SizeVT() + dAtA = make([]byte, size) + n, err := m.MarshalToSizedBufferVT(dAtA[:size]) + if err != nil { + return nil, err + } + return dAtA[:n], nil +} + +func (m *CompactionOptions) MarshalToVT(dAtA []byte) (int, error) { + size := m.SizeVT() + return m.MarshalToSizedBufferVT(dAtA[:size]) +} + +func (m *CompactionOptions) MarshalToSizedBufferVT(dAtA []byte) (int, error) { + if m == nil { + return 0, nil + } + i := len(dAtA) + _ = i + var l int + _ = l + if m.unknownFields != nil { + i -= len(m.unknownFields) + copy(dAtA[i:], m.unknownFields) + } + if m.StatusUpdateIntervalSeconds != 0 { + i = protohelpers.EncodeVarint(dAtA, i, uint64(m.StatusUpdateIntervalSeconds)) + i-- + dAtA[i] = 0x8 + } + return len(dAtA) - i, nil +} + +func (m *CompactionJobStatus) MarshalVT() (dAtA []byte, err error) { + if m == nil { + return nil, nil + } + size := m.SizeVT() + dAtA = make([]byte, size) + n, err := m.MarshalToSizedBufferVT(dAtA[:size]) + if err != nil { + return nil, err + } + return dAtA[:n], nil +} + +func (m *CompactionJobStatus) MarshalToVT(dAtA []byte) (int, error) { + size := m.SizeVT() + return m.MarshalToSizedBufferVT(dAtA[:size]) +} + +func (m *CompactionJobStatus) MarshalToSizedBufferVT(dAtA []byte) (int, error) { + if m == nil { + return 0, nil + } + i := len(dAtA) + _ = i + var l int + _ = l + if m.unknownFields != nil { + i -= len(m.unknownFields) + copy(dAtA[i:], m.unknownFields) + } + if len(m.TenantId) > 0 { + i -= len(m.TenantId) + copy(dAtA[i:], m.TenantId) + i = protohelpers.EncodeVarint(dAtA, i, uint64(len(m.TenantId))) + i-- + dAtA[i] = 0x32 + } + if m.Shard != 0 { + i = protohelpers.EncodeVarint(dAtA, i, uint64(m.Shard)) + i-- + dAtA[i] = 0x28 + } + if m.RaftLogIndex != 0 { + i = protohelpers.EncodeVarint(dAtA, i, uint64(m.RaftLogIndex)) + i-- + dAtA[i] = 0x20 + } + if m.CompletedJob != nil { + size, err := m.CompletedJob.MarshalToSizedBufferVT(dAtA[:i]) + if err != nil { + return 0, err + } + i -= size + i = protohelpers.EncodeVarint(dAtA, i, uint64(size)) + i-- + dAtA[i] = 0x1a + } + if m.Status != 0 { + i = protohelpers.EncodeVarint(dAtA, i, uint64(m.Status)) + i-- + dAtA[i] = 0x10 + } + if len(m.JobName) > 0 { + i -= len(m.JobName) + copy(dAtA[i:], m.JobName) + i = protohelpers.EncodeVarint(dAtA, i, uint64(len(m.JobName))) + i-- + dAtA[i] = 0xa + } + return len(dAtA) - i, nil +} + +func (m *CompletedJob) MarshalVT() (dAtA []byte, err error) { + if m == nil { + return nil, nil + } + size := m.SizeVT() + dAtA = make([]byte, size) + n, err := m.MarshalToSizedBufferVT(dAtA[:size]) + if err != nil { + return nil, err + } + return dAtA[:n], nil +} + +func (m *CompletedJob) MarshalToVT(dAtA []byte) (int, error) { + size := m.SizeVT() + return m.MarshalToSizedBufferVT(dAtA[:size]) +} + +func (m *CompletedJob) MarshalToSizedBufferVT(dAtA []byte) (int, error) { + if m == nil { + return 0, nil + } + i := len(dAtA) + _ = i + var l int + _ = l + if m.unknownFields != nil { + i -= len(m.unknownFields) + copy(dAtA[i:], m.unknownFields) + } + if len(m.Blocks) > 0 { + for iNdEx := len(m.Blocks) - 1; iNdEx >= 0; iNdEx-- { + if vtmsg, ok := interface{}(m.Blocks[iNdEx]).(interface { + MarshalToSizedBufferVT([]byte) (int, error) + }); ok { + size, err := vtmsg.MarshalToSizedBufferVT(dAtA[:i]) + if err != nil { + return 0, err + } + i -= size + i = protohelpers.EncodeVarint(dAtA, i, uint64(size)) + } else { + encoded, err := proto.Marshal(m.Blocks[iNdEx]) + if err != nil { + return 0, err + } + i -= len(encoded) + copy(dAtA[i:], encoded) + i = protohelpers.EncodeVarint(dAtA, i, uint64(len(encoded))) + } + i-- + dAtA[i] = 0xa + } + } + return len(dAtA) - i, nil +} + +func (m *PollCompactionJobsRequest) SizeVT() (n int) { + if m == nil { + return 0 + } + var l int + _ = l + if len(m.JobStatusUpdates) > 0 { + for _, e := range m.JobStatusUpdates { + l = e.SizeVT() + n += 1 + l + protohelpers.SizeOfVarint(uint64(l)) + } + } + if m.JobCapacity != 0 { + n += 1 + protohelpers.SizeOfVarint(uint64(m.JobCapacity)) + } + n += len(m.unknownFields) + return n +} + +func (m *PollCompactionJobsResponse) SizeVT() (n int) { + if m == nil { + return 0 + } + var l int + _ = l + if len(m.CompactionJobs) > 0 { + for _, e := range m.CompactionJobs { + l = e.SizeVT() + n += 1 + l + protohelpers.SizeOfVarint(uint64(l)) + } + } + n += len(m.unknownFields) + return n +} + +func (m *GetCompactionRequest) SizeVT() (n int) { + if m == nil { + return 0 + } + var l int + _ = l + n += len(m.unknownFields) + return n +} + +func (m *GetCompactionResponse) SizeVT() (n int) { + if m == nil { + return 0 + } + var l int + _ = l + if len(m.CompactionJobs) > 0 { + for _, e := range m.CompactionJobs { + l = e.SizeVT() + n += 1 + l + protohelpers.SizeOfVarint(uint64(l)) + } + } + n += len(m.unknownFields) + return n +} + +func (m *CompactionJob) SizeVT() (n int) { + if m == nil { + return 0 + } + var l int + _ = l + l = len(m.Name) + if l > 0 { + n += 1 + l + protohelpers.SizeOfVarint(uint64(l)) + } + if m.Options != nil { + l = m.Options.SizeVT() + n += 1 + l + protohelpers.SizeOfVarint(uint64(l)) + } + if len(m.Blocks) > 0 { + for _, e := range m.Blocks { + if size, ok := interface{}(e).(interface { + SizeVT() int + }); ok { + l = size.SizeVT() + } else { + l = proto.Size(e) + } + n += 1 + l + protohelpers.SizeOfVarint(uint64(l)) + } + } + if m.Status != nil { + l = m.Status.SizeVT() + n += 1 + l + protohelpers.SizeOfVarint(uint64(l)) + } + if m.RaftLogIndex != 0 { + n += 1 + protohelpers.SizeOfVarint(uint64(m.RaftLogIndex)) + } + if m.Shard != 0 { + n += 1 + protohelpers.SizeOfVarint(uint64(m.Shard)) + } + l = len(m.TenantId) + if l > 0 { + n += 1 + l + protohelpers.SizeOfVarint(uint64(l)) + } + if m.CompactionLevel != 0 { + n += 1 + protohelpers.SizeOfVarint(uint64(m.CompactionLevel)) + } + n += len(m.unknownFields) + return n +} + +func (m *CompactionOptions) SizeVT() (n int) { + if m == nil { + return 0 + } + var l int + _ = l + if m.StatusUpdateIntervalSeconds != 0 { + n += 1 + protohelpers.SizeOfVarint(uint64(m.StatusUpdateIntervalSeconds)) + } + n += len(m.unknownFields) + return n +} + +func (m *CompactionJobStatus) SizeVT() (n int) { + if m == nil { + return 0 + } + var l int + _ = l + l = len(m.JobName) + if l > 0 { + n += 1 + l + protohelpers.SizeOfVarint(uint64(l)) + } + if m.Status != 0 { + n += 1 + protohelpers.SizeOfVarint(uint64(m.Status)) + } + if m.CompletedJob != nil { + l = m.CompletedJob.SizeVT() + n += 1 + l + protohelpers.SizeOfVarint(uint64(l)) + } + if m.RaftLogIndex != 0 { + n += 1 + protohelpers.SizeOfVarint(uint64(m.RaftLogIndex)) + } + if m.Shard != 0 { + n += 1 + protohelpers.SizeOfVarint(uint64(m.Shard)) + } + l = len(m.TenantId) + if l > 0 { + n += 1 + l + protohelpers.SizeOfVarint(uint64(l)) + } + n += len(m.unknownFields) + return n +} + +func (m *CompletedJob) SizeVT() (n int) { + if m == nil { + return 0 + } + var l int + _ = l + if len(m.Blocks) > 0 { + for _, e := range m.Blocks { + if size, ok := interface{}(e).(interface { + SizeVT() int + }); ok { + l = size.SizeVT() + } else { + l = proto.Size(e) + } + n += 1 + l + protohelpers.SizeOfVarint(uint64(l)) + } + } + n += len(m.unknownFields) + return n +} + +func (m *PollCompactionJobsRequest) UnmarshalVT(dAtA []byte) error { + l := len(dAtA) + iNdEx := 0 + for iNdEx < l { + preIndex := iNdEx + var wire uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + wire |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + fieldNum := int32(wire >> 3) + wireType := int(wire & 0x7) + if wireType == 4 { + return fmt.Errorf("proto: PollCompactionJobsRequest: wiretype end group for non-group") + } + if fieldNum <= 0 { + return fmt.Errorf("proto: PollCompactionJobsRequest: illegal tag %d (wire type %d)", fieldNum, wire) + } + switch fieldNum { + case 1: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field JobStatusUpdates", wireType) + } + var msglen int + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + msglen |= int(b&0x7F) << shift + if b < 0x80 { + break + } + } + if msglen < 0 { + return protohelpers.ErrInvalidLength + } + postIndex := iNdEx + msglen + if postIndex < 0 { + return protohelpers.ErrInvalidLength + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + m.JobStatusUpdates = append(m.JobStatusUpdates, &CompactionJobStatus{}) + if err := m.JobStatusUpdates[len(m.JobStatusUpdates)-1].UnmarshalVT(dAtA[iNdEx:postIndex]); err != nil { + return err + } + iNdEx = postIndex + case 2: + if wireType != 0 { + return fmt.Errorf("proto: wrong wireType = %d for field JobCapacity", wireType) + } + m.JobCapacity = 0 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + m.JobCapacity |= uint32(b&0x7F) << shift + if b < 0x80 { + break + } + } + default: + iNdEx = preIndex + skippy, err := protohelpers.Skip(dAtA[iNdEx:]) + if err != nil { + return err + } + if (skippy < 0) || (iNdEx+skippy) < 0 { + return protohelpers.ErrInvalidLength + } + if (iNdEx + skippy) > l { + return io.ErrUnexpectedEOF + } + m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...) + iNdEx += skippy + } + } + + if iNdEx > l { + return io.ErrUnexpectedEOF + } + return nil +} +func (m *PollCompactionJobsResponse) UnmarshalVT(dAtA []byte) error { + l := len(dAtA) + iNdEx := 0 + for iNdEx < l { + preIndex := iNdEx + var wire uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + wire |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + fieldNum := int32(wire >> 3) + wireType := int(wire & 0x7) + if wireType == 4 { + return fmt.Errorf("proto: PollCompactionJobsResponse: wiretype end group for non-group") + } + if fieldNum <= 0 { + return fmt.Errorf("proto: PollCompactionJobsResponse: illegal tag %d (wire type %d)", fieldNum, wire) + } + switch fieldNum { + case 1: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field CompactionJobs", wireType) + } + var msglen int + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + msglen |= int(b&0x7F) << shift + if b < 0x80 { + break + } + } + if msglen < 0 { + return protohelpers.ErrInvalidLength + } + postIndex := iNdEx + msglen + if postIndex < 0 { + return protohelpers.ErrInvalidLength + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + m.CompactionJobs = append(m.CompactionJobs, &CompactionJob{}) + if err := m.CompactionJobs[len(m.CompactionJobs)-1].UnmarshalVT(dAtA[iNdEx:postIndex]); err != nil { + return err + } + iNdEx = postIndex + default: + iNdEx = preIndex + skippy, err := protohelpers.Skip(dAtA[iNdEx:]) + if err != nil { + return err + } + if (skippy < 0) || (iNdEx+skippy) < 0 { + return protohelpers.ErrInvalidLength + } + if (iNdEx + skippy) > l { + return io.ErrUnexpectedEOF + } + m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...) + iNdEx += skippy + } + } + + if iNdEx > l { + return io.ErrUnexpectedEOF + } + return nil +} +func (m *GetCompactionRequest) UnmarshalVT(dAtA []byte) error { + l := len(dAtA) + iNdEx := 0 + for iNdEx < l { + preIndex := iNdEx + var wire uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + wire |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + fieldNum := int32(wire >> 3) + wireType := int(wire & 0x7) + if wireType == 4 { + return fmt.Errorf("proto: GetCompactionRequest: wiretype end group for non-group") + } + if fieldNum <= 0 { + return fmt.Errorf("proto: GetCompactionRequest: illegal tag %d (wire type %d)", fieldNum, wire) + } + switch fieldNum { + default: + iNdEx = preIndex + skippy, err := protohelpers.Skip(dAtA[iNdEx:]) + if err != nil { + return err + } + if (skippy < 0) || (iNdEx+skippy) < 0 { + return protohelpers.ErrInvalidLength + } + if (iNdEx + skippy) > l { + return io.ErrUnexpectedEOF + } + m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...) + iNdEx += skippy + } + } + + if iNdEx > l { + return io.ErrUnexpectedEOF + } + return nil +} +func (m *GetCompactionResponse) UnmarshalVT(dAtA []byte) error { + l := len(dAtA) + iNdEx := 0 + for iNdEx < l { + preIndex := iNdEx + var wire uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + wire |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + fieldNum := int32(wire >> 3) + wireType := int(wire & 0x7) + if wireType == 4 { + return fmt.Errorf("proto: GetCompactionResponse: wiretype end group for non-group") + } + if fieldNum <= 0 { + return fmt.Errorf("proto: GetCompactionResponse: illegal tag %d (wire type %d)", fieldNum, wire) + } + switch fieldNum { + case 1: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field CompactionJobs", wireType) + } + var msglen int + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + msglen |= int(b&0x7F) << shift + if b < 0x80 { + break + } + } + if msglen < 0 { + return protohelpers.ErrInvalidLength + } + postIndex := iNdEx + msglen + if postIndex < 0 { + return protohelpers.ErrInvalidLength + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + m.CompactionJobs = append(m.CompactionJobs, &CompactionJob{}) + if err := m.CompactionJobs[len(m.CompactionJobs)-1].UnmarshalVT(dAtA[iNdEx:postIndex]); err != nil { + return err + } + iNdEx = postIndex + default: + iNdEx = preIndex + skippy, err := protohelpers.Skip(dAtA[iNdEx:]) + if err != nil { + return err + } + if (skippy < 0) || (iNdEx+skippy) < 0 { + return protohelpers.ErrInvalidLength + } + if (iNdEx + skippy) > l { + return io.ErrUnexpectedEOF + } + m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...) + iNdEx += skippy + } + } + + if iNdEx > l { + return io.ErrUnexpectedEOF + } + return nil +} +func (m *CompactionJob) UnmarshalVT(dAtA []byte) error { + l := len(dAtA) + iNdEx := 0 + for iNdEx < l { + preIndex := iNdEx + var wire uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + wire |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + fieldNum := int32(wire >> 3) + wireType := int(wire & 0x7) + if wireType == 4 { + return fmt.Errorf("proto: CompactionJob: wiretype end group for non-group") + } + if fieldNum <= 0 { + return fmt.Errorf("proto: CompactionJob: illegal tag %d (wire type %d)", fieldNum, wire) + } + switch fieldNum { + case 1: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field Name", wireType) + } + var stringLen uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + stringLen |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + intStringLen := int(stringLen) + if intStringLen < 0 { + return protohelpers.ErrInvalidLength + } + postIndex := iNdEx + intStringLen + if postIndex < 0 { + return protohelpers.ErrInvalidLength + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + m.Name = string(dAtA[iNdEx:postIndex]) + iNdEx = postIndex + case 2: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field Options", wireType) + } + var msglen int + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + msglen |= int(b&0x7F) << shift + if b < 0x80 { + break + } + } + if msglen < 0 { + return protohelpers.ErrInvalidLength + } + postIndex := iNdEx + msglen + if postIndex < 0 { + return protohelpers.ErrInvalidLength + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + if m.Options == nil { + m.Options = &CompactionOptions{} + } + if err := m.Options.UnmarshalVT(dAtA[iNdEx:postIndex]); err != nil { + return err + } + iNdEx = postIndex + case 3: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field Blocks", wireType) + } + var msglen int + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + msglen |= int(b&0x7F) << shift + if b < 0x80 { + break + } + } + if msglen < 0 { + return protohelpers.ErrInvalidLength + } + postIndex := iNdEx + msglen + if postIndex < 0 { + return protohelpers.ErrInvalidLength + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + m.Blocks = append(m.Blocks, &v1.BlockMeta{}) + if unmarshal, ok := interface{}(m.Blocks[len(m.Blocks)-1]).(interface { + UnmarshalVT([]byte) error + }); ok { + if err := unmarshal.UnmarshalVT(dAtA[iNdEx:postIndex]); err != nil { + return err + } + } else { + if err := proto.Unmarshal(dAtA[iNdEx:postIndex], m.Blocks[len(m.Blocks)-1]); err != nil { + return err + } + } + iNdEx = postIndex + case 4: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field Status", wireType) + } + var msglen int + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + msglen |= int(b&0x7F) << shift + if b < 0x80 { + break + } + } + if msglen < 0 { + return protohelpers.ErrInvalidLength + } + postIndex := iNdEx + msglen + if postIndex < 0 { + return protohelpers.ErrInvalidLength + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + if m.Status == nil { + m.Status = &CompactionJobStatus{} + } + if err := m.Status.UnmarshalVT(dAtA[iNdEx:postIndex]); err != nil { + return err + } + iNdEx = postIndex + case 5: + if wireType != 0 { + return fmt.Errorf("proto: wrong wireType = %d for field RaftLogIndex", wireType) + } + m.RaftLogIndex = 0 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + m.RaftLogIndex |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + case 6: + if wireType != 0 { + return fmt.Errorf("proto: wrong wireType = %d for field Shard", wireType) + } + m.Shard = 0 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + m.Shard |= uint32(b&0x7F) << shift + if b < 0x80 { + break + } + } + case 7: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field TenantId", wireType) + } + var stringLen uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + stringLen |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + intStringLen := int(stringLen) + if intStringLen < 0 { + return protohelpers.ErrInvalidLength + } + postIndex := iNdEx + intStringLen + if postIndex < 0 { + return protohelpers.ErrInvalidLength + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + m.TenantId = string(dAtA[iNdEx:postIndex]) + iNdEx = postIndex + case 8: + if wireType != 0 { + return fmt.Errorf("proto: wrong wireType = %d for field CompactionLevel", wireType) + } + m.CompactionLevel = 0 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + m.CompactionLevel |= uint32(b&0x7F) << shift + if b < 0x80 { + break + } + } + default: + iNdEx = preIndex + skippy, err := protohelpers.Skip(dAtA[iNdEx:]) + if err != nil { + return err + } + if (skippy < 0) || (iNdEx+skippy) < 0 { + return protohelpers.ErrInvalidLength + } + if (iNdEx + skippy) > l { + return io.ErrUnexpectedEOF + } + m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...) + iNdEx += skippy + } + } + + if iNdEx > l { + return io.ErrUnexpectedEOF + } + return nil +} +func (m *CompactionOptions) UnmarshalVT(dAtA []byte) error { + l := len(dAtA) + iNdEx := 0 + for iNdEx < l { + preIndex := iNdEx + var wire uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + wire |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + fieldNum := int32(wire >> 3) + wireType := int(wire & 0x7) + if wireType == 4 { + return fmt.Errorf("proto: CompactionOptions: wiretype end group for non-group") + } + if fieldNum <= 0 { + return fmt.Errorf("proto: CompactionOptions: illegal tag %d (wire type %d)", fieldNum, wire) + } + switch fieldNum { + case 1: + if wireType != 0 { + return fmt.Errorf("proto: wrong wireType = %d for field StatusUpdateIntervalSeconds", wireType) + } + m.StatusUpdateIntervalSeconds = 0 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + m.StatusUpdateIntervalSeconds |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + default: + iNdEx = preIndex + skippy, err := protohelpers.Skip(dAtA[iNdEx:]) + if err != nil { + return err + } + if (skippy < 0) || (iNdEx+skippy) < 0 { + return protohelpers.ErrInvalidLength + } + if (iNdEx + skippy) > l { + return io.ErrUnexpectedEOF + } + m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...) + iNdEx += skippy + } + } + + if iNdEx > l { + return io.ErrUnexpectedEOF + } + return nil +} +func (m *CompactionJobStatus) UnmarshalVT(dAtA []byte) error { + l := len(dAtA) + iNdEx := 0 + for iNdEx < l { + preIndex := iNdEx + var wire uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + wire |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + fieldNum := int32(wire >> 3) + wireType := int(wire & 0x7) + if wireType == 4 { + return fmt.Errorf("proto: CompactionJobStatus: wiretype end group for non-group") + } + if fieldNum <= 0 { + return fmt.Errorf("proto: CompactionJobStatus: illegal tag %d (wire type %d)", fieldNum, wire) + } + switch fieldNum { + case 1: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field JobName", wireType) + } + var stringLen uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + stringLen |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + intStringLen := int(stringLen) + if intStringLen < 0 { + return protohelpers.ErrInvalidLength + } + postIndex := iNdEx + intStringLen + if postIndex < 0 { + return protohelpers.ErrInvalidLength + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + m.JobName = string(dAtA[iNdEx:postIndex]) + iNdEx = postIndex + case 2: + if wireType != 0 { + return fmt.Errorf("proto: wrong wireType = %d for field Status", wireType) + } + m.Status = 0 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + m.Status |= CompactionStatus(b&0x7F) << shift + if b < 0x80 { + break + } + } + case 3: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field CompletedJob", wireType) + } + var msglen int + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + msglen |= int(b&0x7F) << shift + if b < 0x80 { + break + } + } + if msglen < 0 { + return protohelpers.ErrInvalidLength + } + postIndex := iNdEx + msglen + if postIndex < 0 { + return protohelpers.ErrInvalidLength + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + if m.CompletedJob == nil { + m.CompletedJob = &CompletedJob{} + } + if err := m.CompletedJob.UnmarshalVT(dAtA[iNdEx:postIndex]); err != nil { + return err + } + iNdEx = postIndex + case 4: + if wireType != 0 { + return fmt.Errorf("proto: wrong wireType = %d for field RaftLogIndex", wireType) + } + m.RaftLogIndex = 0 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + m.RaftLogIndex |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + case 5: + if wireType != 0 { + return fmt.Errorf("proto: wrong wireType = %d for field Shard", wireType) + } + m.Shard = 0 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + m.Shard |= uint32(b&0x7F) << shift + if b < 0x80 { + break + } + } + case 6: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field TenantId", wireType) + } + var stringLen uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + stringLen |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + intStringLen := int(stringLen) + if intStringLen < 0 { + return protohelpers.ErrInvalidLength + } + postIndex := iNdEx + intStringLen + if postIndex < 0 { + return protohelpers.ErrInvalidLength + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + m.TenantId = string(dAtA[iNdEx:postIndex]) + iNdEx = postIndex + default: + iNdEx = preIndex + skippy, err := protohelpers.Skip(dAtA[iNdEx:]) + if err != nil { + return err + } + if (skippy < 0) || (iNdEx+skippy) < 0 { + return protohelpers.ErrInvalidLength + } + if (iNdEx + skippy) > l { + return io.ErrUnexpectedEOF + } + m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...) + iNdEx += skippy + } + } + + if iNdEx > l { + return io.ErrUnexpectedEOF + } + return nil +} +func (m *CompletedJob) UnmarshalVT(dAtA []byte) error { + l := len(dAtA) + iNdEx := 0 + for iNdEx < l { + preIndex := iNdEx + var wire uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + wire |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + fieldNum := int32(wire >> 3) + wireType := int(wire & 0x7) + if wireType == 4 { + return fmt.Errorf("proto: CompletedJob: wiretype end group for non-group") + } + if fieldNum <= 0 { + return fmt.Errorf("proto: CompletedJob: illegal tag %d (wire type %d)", fieldNum, wire) + } + switch fieldNum { + case 1: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field Blocks", wireType) + } + var msglen int + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + msglen |= int(b&0x7F) << shift + if b < 0x80 { + break + } + } + if msglen < 0 { + return protohelpers.ErrInvalidLength + } + postIndex := iNdEx + msglen + if postIndex < 0 { + return protohelpers.ErrInvalidLength + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + m.Blocks = append(m.Blocks, &v1.BlockMeta{}) + if unmarshal, ok := interface{}(m.Blocks[len(m.Blocks)-1]).(interface { + UnmarshalVT([]byte) error + }); ok { + if err := unmarshal.UnmarshalVT(dAtA[iNdEx:postIndex]); err != nil { + return err + } + } else { + if err := proto.Unmarshal(dAtA[iNdEx:postIndex], m.Blocks[len(m.Blocks)-1]); err != nil { + return err + } + } + iNdEx = postIndex + default: + iNdEx = preIndex + skippy, err := protohelpers.Skip(dAtA[iNdEx:]) + if err != nil { + return err + } + if (skippy < 0) || (iNdEx+skippy) < 0 { + return protohelpers.ErrInvalidLength + } + if (iNdEx + skippy) > l { + return io.ErrUnexpectedEOF + } + m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...) + iNdEx += skippy + } + } + + if iNdEx > l { + return io.ErrUnexpectedEOF + } + return nil +} diff --git a/api/gen/proto/go/compactor/v1/compactorv1connect/compactor.connect.go b/api/gen/proto/go/compactor/v1/compactorv1connect/compactor.connect.go new file mode 100644 index 0000000000..fb405e9d27 --- /dev/null +++ b/api/gen/proto/go/compactor/v1/compactorv1connect/compactor.connect.go @@ -0,0 +1,147 @@ +// Code generated by protoc-gen-connect-go. DO NOT EDIT. +// +// Source: compactor/v1/compactor.proto + +package compactorv1connect + +import ( + connect "connectrpc.com/connect" + context "context" + errors "errors" + v1 "github.com/grafana/pyroscope/api/gen/proto/go/compactor/v1" + http "net/http" + strings "strings" +) + +// This is a compile-time assertion to ensure that this generated file and the connect package are +// compatible. If you get a compiler error that this constant is not defined, this code was +// generated with a version of connect newer than the one compiled into your binary. You can fix the +// problem by either regenerating this code with an older version of connect or updating the connect +// version compiled into your binary. +const _ = connect.IsAtLeastVersion1_13_0 + +const ( + // CompactionPlannerName is the fully-qualified name of the CompactionPlanner service. + CompactionPlannerName = "compactor.v1.CompactionPlanner" +) + +// These constants are the fully-qualified names of the RPCs defined in this package. They're +// exposed at runtime as Spec.Procedure and as the final two segments of the HTTP route. +// +// Note that these are different from the fully-qualified method names used by +// google.golang.org/protobuf/reflect/protoreflect. To convert from these constants to +// reflection-formatted method names, remove the leading slash and convert the remaining slash to a +// period. +const ( + // CompactionPlannerPollCompactionJobsProcedure is the fully-qualified name of the + // CompactionPlanner's PollCompactionJobs RPC. + CompactionPlannerPollCompactionJobsProcedure = "/compactor.v1.CompactionPlanner/PollCompactionJobs" + // CompactionPlannerGetCompactionJobsProcedure is the fully-qualified name of the + // CompactionPlanner's GetCompactionJobs RPC. + CompactionPlannerGetCompactionJobsProcedure = "/compactor.v1.CompactionPlanner/GetCompactionJobs" +) + +// These variables are the protoreflect.Descriptor objects for the RPCs defined in this package. +var ( + compactionPlannerServiceDescriptor = v1.File_compactor_v1_compactor_proto.Services().ByName("CompactionPlanner") + compactionPlannerPollCompactionJobsMethodDescriptor = compactionPlannerServiceDescriptor.Methods().ByName("PollCompactionJobs") + compactionPlannerGetCompactionJobsMethodDescriptor = compactionPlannerServiceDescriptor.Methods().ByName("GetCompactionJobs") +) + +// CompactionPlannerClient is a client for the compactor.v1.CompactionPlanner service. +type CompactionPlannerClient interface { + // Used to both retrieve jobs and update the jobs status at the same time. + PollCompactionJobs(context.Context, *connect.Request[v1.PollCompactionJobsRequest]) (*connect.Response[v1.PollCompactionJobsResponse], error) + // Used for admin purposes only. + GetCompactionJobs(context.Context, *connect.Request[v1.GetCompactionRequest]) (*connect.Response[v1.GetCompactionResponse], error) +} + +// NewCompactionPlannerClient constructs a client for the compactor.v1.CompactionPlanner service. By +// default, it uses the Connect protocol with the binary Protobuf Codec, asks for gzipped responses, +// and sends uncompressed requests. To use the gRPC or gRPC-Web protocols, supply the +// connect.WithGRPC() or connect.WithGRPCWeb() options. +// +// The URL supplied here should be the base URL for the Connect or gRPC server (for example, +// http://api.acme.com or https://acme.com/grpc). +func NewCompactionPlannerClient(httpClient connect.HTTPClient, baseURL string, opts ...connect.ClientOption) CompactionPlannerClient { + baseURL = strings.TrimRight(baseURL, "/") + return &compactionPlannerClient{ + pollCompactionJobs: connect.NewClient[v1.PollCompactionJobsRequest, v1.PollCompactionJobsResponse]( + httpClient, + baseURL+CompactionPlannerPollCompactionJobsProcedure, + connect.WithSchema(compactionPlannerPollCompactionJobsMethodDescriptor), + connect.WithClientOptions(opts...), + ), + getCompactionJobs: connect.NewClient[v1.GetCompactionRequest, v1.GetCompactionResponse]( + httpClient, + baseURL+CompactionPlannerGetCompactionJobsProcedure, + connect.WithSchema(compactionPlannerGetCompactionJobsMethodDescriptor), + connect.WithClientOptions(opts...), + ), + } +} + +// compactionPlannerClient implements CompactionPlannerClient. +type compactionPlannerClient struct { + pollCompactionJobs *connect.Client[v1.PollCompactionJobsRequest, v1.PollCompactionJobsResponse] + getCompactionJobs *connect.Client[v1.GetCompactionRequest, v1.GetCompactionResponse] +} + +// PollCompactionJobs calls compactor.v1.CompactionPlanner.PollCompactionJobs. +func (c *compactionPlannerClient) PollCompactionJobs(ctx context.Context, req *connect.Request[v1.PollCompactionJobsRequest]) (*connect.Response[v1.PollCompactionJobsResponse], error) { + return c.pollCompactionJobs.CallUnary(ctx, req) +} + +// GetCompactionJobs calls compactor.v1.CompactionPlanner.GetCompactionJobs. +func (c *compactionPlannerClient) GetCompactionJobs(ctx context.Context, req *connect.Request[v1.GetCompactionRequest]) (*connect.Response[v1.GetCompactionResponse], error) { + return c.getCompactionJobs.CallUnary(ctx, req) +} + +// CompactionPlannerHandler is an implementation of the compactor.v1.CompactionPlanner service. +type CompactionPlannerHandler interface { + // Used to both retrieve jobs and update the jobs status at the same time. + PollCompactionJobs(context.Context, *connect.Request[v1.PollCompactionJobsRequest]) (*connect.Response[v1.PollCompactionJobsResponse], error) + // Used for admin purposes only. + GetCompactionJobs(context.Context, *connect.Request[v1.GetCompactionRequest]) (*connect.Response[v1.GetCompactionResponse], error) +} + +// NewCompactionPlannerHandler builds an HTTP handler from the service implementation. It returns +// the path on which to mount the handler and the handler itself. +// +// By default, handlers support the Connect, gRPC, and gRPC-Web protocols with the binary Protobuf +// and JSON codecs. They also support gzip compression. +func NewCompactionPlannerHandler(svc CompactionPlannerHandler, opts ...connect.HandlerOption) (string, http.Handler) { + compactionPlannerPollCompactionJobsHandler := connect.NewUnaryHandler( + CompactionPlannerPollCompactionJobsProcedure, + svc.PollCompactionJobs, + connect.WithSchema(compactionPlannerPollCompactionJobsMethodDescriptor), + connect.WithHandlerOptions(opts...), + ) + compactionPlannerGetCompactionJobsHandler := connect.NewUnaryHandler( + CompactionPlannerGetCompactionJobsProcedure, + svc.GetCompactionJobs, + connect.WithSchema(compactionPlannerGetCompactionJobsMethodDescriptor), + connect.WithHandlerOptions(opts...), + ) + return "/compactor.v1.CompactionPlanner/", http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { + switch r.URL.Path { + case CompactionPlannerPollCompactionJobsProcedure: + compactionPlannerPollCompactionJobsHandler.ServeHTTP(w, r) + case CompactionPlannerGetCompactionJobsProcedure: + compactionPlannerGetCompactionJobsHandler.ServeHTTP(w, r) + default: + http.NotFound(w, r) + } + }) +} + +// UnimplementedCompactionPlannerHandler returns CodeUnimplemented from all methods. +type UnimplementedCompactionPlannerHandler struct{} + +func (UnimplementedCompactionPlannerHandler) PollCompactionJobs(context.Context, *connect.Request[v1.PollCompactionJobsRequest]) (*connect.Response[v1.PollCompactionJobsResponse], error) { + return nil, connect.NewError(connect.CodeUnimplemented, errors.New("compactor.v1.CompactionPlanner.PollCompactionJobs is not implemented")) +} + +func (UnimplementedCompactionPlannerHandler) GetCompactionJobs(context.Context, *connect.Request[v1.GetCompactionRequest]) (*connect.Response[v1.GetCompactionResponse], error) { + return nil, connect.NewError(connect.CodeUnimplemented, errors.New("compactor.v1.CompactionPlanner.GetCompactionJobs is not implemented")) +} diff --git a/api/gen/proto/go/compactor/v1/compactorv1connect/compactor.connect.mux.go b/api/gen/proto/go/compactor/v1/compactorv1connect/compactor.connect.mux.go new file mode 100644 index 0000000000..cd93b1a6f3 --- /dev/null +++ b/api/gen/proto/go/compactor/v1/compactorv1connect/compactor.connect.mux.go @@ -0,0 +1,32 @@ +// Code generated by protoc-gen-connect-go-mux. DO NOT EDIT. +// +// Source: compactor/v1/compactor.proto + +package compactorv1connect + +import ( + connect "connectrpc.com/connect" + mux "github.com/gorilla/mux" +) + +// This is a compile-time assertion to ensure that this generated file and the connect package are +// compatible. If you get a compiler error that this constant is not defined, this code was +// generated with a version of connect newer than the one compiled into your binary. You can fix the +// problem by either regenerating this code with an older version of connect or updating the connect +// version compiled into your binary. +const _ = connect.IsAtLeastVersion0_1_0 + +// RegisterCompactionPlannerHandler register an HTTP handler to a mux.Router from the service +// implementation. +func RegisterCompactionPlannerHandler(mux *mux.Router, svc CompactionPlannerHandler, opts ...connect.HandlerOption) { + mux.Handle("/compactor.v1.CompactionPlanner/PollCompactionJobs", connect.NewUnaryHandler( + "/compactor.v1.CompactionPlanner/PollCompactionJobs", + svc.PollCompactionJobs, + opts..., + )) + mux.Handle("/compactor.v1.CompactionPlanner/GetCompactionJobs", connect.NewUnaryHandler( + "/compactor.v1.CompactionPlanner/GetCompactionJobs", + svc.GetCompactionJobs, + opts..., + )) +} diff --git a/api/gen/proto/go/metastore/v1/metastore.pb.go b/api/gen/proto/go/metastore/v1/metastore.pb.go new file mode 100644 index 0000000000..0f2e7710d7 --- /dev/null +++ b/api/gen/proto/go/metastore/v1/metastore.pb.go @@ -0,0 +1,808 @@ +// Code generated by protoc-gen-go. DO NOT EDIT. +// versions: +// protoc-gen-go v1.34.2 +// protoc (unknown) +// source: metastore/v1/metastore.proto + +package metastorev1 + +import ( + v1 "github.com/grafana/pyroscope/api/gen/proto/go/types/v1" + protoreflect "google.golang.org/protobuf/reflect/protoreflect" + protoimpl "google.golang.org/protobuf/runtime/protoimpl" + reflect "reflect" + sync "sync" +) + +const ( + // Verify that this generated code is sufficiently up-to-date. + _ = protoimpl.EnforceVersion(20 - protoimpl.MinVersion) + // Verify that runtime/protoimpl is sufficiently up-to-date. + _ = protoimpl.EnforceVersion(protoimpl.MaxVersion - 20) +) + +type AddBlockRequest struct { + state protoimpl.MessageState + sizeCache protoimpl.SizeCache + unknownFields protoimpl.UnknownFields + + Block *BlockMeta `protobuf:"bytes,1,opt,name=block,proto3" json:"block,omitempty"` +} + +func (x *AddBlockRequest) Reset() { + *x = AddBlockRequest{} + if protoimpl.UnsafeEnabled { + mi := &file_metastore_v1_metastore_proto_msgTypes[0] + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + ms.StoreMessageInfo(mi) + } +} + +func (x *AddBlockRequest) String() string { + return protoimpl.X.MessageStringOf(x) +} + +func (*AddBlockRequest) ProtoMessage() {} + +func (x *AddBlockRequest) ProtoReflect() protoreflect.Message { + mi := &file_metastore_v1_metastore_proto_msgTypes[0] + if protoimpl.UnsafeEnabled && x != nil { + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + if ms.LoadMessageInfo() == nil { + ms.StoreMessageInfo(mi) + } + return ms + } + return mi.MessageOf(x) +} + +// Deprecated: Use AddBlockRequest.ProtoReflect.Descriptor instead. +func (*AddBlockRequest) Descriptor() ([]byte, []int) { + return file_metastore_v1_metastore_proto_rawDescGZIP(), []int{0} +} + +func (x *AddBlockRequest) GetBlock() *BlockMeta { + if x != nil { + return x.Block + } + return nil +} + +type AddBlockResponse struct { + state protoimpl.MessageState + sizeCache protoimpl.SizeCache + unknownFields protoimpl.UnknownFields +} + +func (x *AddBlockResponse) Reset() { + *x = AddBlockResponse{} + if protoimpl.UnsafeEnabled { + mi := &file_metastore_v1_metastore_proto_msgTypes[1] + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + ms.StoreMessageInfo(mi) + } +} + +func (x *AddBlockResponse) String() string { + return protoimpl.X.MessageStringOf(x) +} + +func (*AddBlockResponse) ProtoMessage() {} + +func (x *AddBlockResponse) ProtoReflect() protoreflect.Message { + mi := &file_metastore_v1_metastore_proto_msgTypes[1] + if protoimpl.UnsafeEnabled && x != nil { + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + if ms.LoadMessageInfo() == nil { + ms.StoreMessageInfo(mi) + } + return ms + } + return mi.MessageOf(x) +} + +// Deprecated: Use AddBlockResponse.ProtoReflect.Descriptor instead. +func (*AddBlockResponse) Descriptor() ([]byte, []int) { + return file_metastore_v1_metastore_proto_rawDescGZIP(), []int{1} +} + +type BlockMeta struct { + state protoimpl.MessageState + sizeCache protoimpl.SizeCache + unknownFields protoimpl.UnknownFields + + FormatVersion uint64 `protobuf:"varint,1,opt,name=format_version,json=formatVersion,proto3" json:"format_version,omitempty"` + Id string `protobuf:"bytes,2,opt,name=id,proto3" json:"id,omitempty"` + MinTime int64 `protobuf:"varint,3,opt,name=min_time,json=minTime,proto3" json:"min_time,omitempty"` + MaxTime int64 `protobuf:"varint,4,opt,name=max_time,json=maxTime,proto3" json:"max_time,omitempty"` + Shard uint32 `protobuf:"varint,5,opt,name=shard,proto3" json:"shard,omitempty"` + CompactionLevel uint32 `protobuf:"varint,6,opt,name=compaction_level,json=compactionLevel,proto3" json:"compaction_level,omitempty"` + // Optional. Empty if compaction level is 0. + TenantId string `protobuf:"bytes,7,opt,name=tenant_id,json=tenantId,proto3" json:"tenant_id,omitempty"` + Datasets []*Dataset `protobuf:"bytes,8,rep,name=datasets,proto3" json:"datasets,omitempty"` + Size uint64 `protobuf:"varint,9,opt,name=size,proto3" json:"size,omitempty"` +} + +func (x *BlockMeta) Reset() { + *x = BlockMeta{} + if protoimpl.UnsafeEnabled { + mi := &file_metastore_v1_metastore_proto_msgTypes[2] + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + ms.StoreMessageInfo(mi) + } +} + +func (x *BlockMeta) String() string { + return protoimpl.X.MessageStringOf(x) +} + +func (*BlockMeta) ProtoMessage() {} + +func (x *BlockMeta) ProtoReflect() protoreflect.Message { + mi := &file_metastore_v1_metastore_proto_msgTypes[2] + if protoimpl.UnsafeEnabled && x != nil { + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + if ms.LoadMessageInfo() == nil { + ms.StoreMessageInfo(mi) + } + return ms + } + return mi.MessageOf(x) +} + +// Deprecated: Use BlockMeta.ProtoReflect.Descriptor instead. +func (*BlockMeta) Descriptor() ([]byte, []int) { + return file_metastore_v1_metastore_proto_rawDescGZIP(), []int{2} +} + +func (x *BlockMeta) GetFormatVersion() uint64 { + if x != nil { + return x.FormatVersion + } + return 0 +} + +func (x *BlockMeta) GetId() string { + if x != nil { + return x.Id + } + return "" +} + +func (x *BlockMeta) GetMinTime() int64 { + if x != nil { + return x.MinTime + } + return 0 +} + +func (x *BlockMeta) GetMaxTime() int64 { + if x != nil { + return x.MaxTime + } + return 0 +} + +func (x *BlockMeta) GetShard() uint32 { + if x != nil { + return x.Shard + } + return 0 +} + +func (x *BlockMeta) GetCompactionLevel() uint32 { + if x != nil { + return x.CompactionLevel + } + return 0 +} + +func (x *BlockMeta) GetTenantId() string { + if x != nil { + return x.TenantId + } + return "" +} + +func (x *BlockMeta) GetDatasets() []*Dataset { + if x != nil { + return x.Datasets + } + return nil +} + +func (x *BlockMeta) GetSize() uint64 { + if x != nil { + return x.Size + } + return 0 +} + +type Dataset struct { + state protoimpl.MessageState + sizeCache protoimpl.SizeCache + unknownFields protoimpl.UnknownFields + + Labels []*v1.Labels `protobuf:"bytes,8,rep,name=labels,proto3" json:"labels,omitempty"` + TenantId string `protobuf:"bytes,1,opt,name=tenant_id,json=tenantId,proto3" json:"tenant_id,omitempty"` + Name string `protobuf:"bytes,2,opt,name=name,proto3" json:"name,omitempty"` + MinTime int64 `protobuf:"varint,3,opt,name=min_time,json=minTime,proto3" json:"min_time,omitempty"` + MaxTime int64 `protobuf:"varint,4,opt,name=max_time,json=maxTime,proto3" json:"max_time,omitempty"` + // Table of contents lists data sections within the tenant + // service region. The offsets are absolute. + // + // The interpretation of the table of contents is specific + // to the metadata format version. By default, the sections are: + // - 0: profiles.parquet + // - 1: index.tsdb + // - 2: symbols.symdb + TableOfContents []uint64 `protobuf:"varint,5,rep,packed,name=table_of_contents,json=tableOfContents,proto3" json:"table_of_contents,omitempty"` + // Size of the section in bytes. + Size uint64 `protobuf:"varint,6,opt,name=size,proto3" json:"size,omitempty"` + // TODO: delete + // Profile types present in the tenant service data. + ProfileTypes []string `protobuf:"bytes,7,rep,name=profile_types,json=profileTypes,proto3" json:"profile_types,omitempty"` +} + +func (x *Dataset) Reset() { + *x = Dataset{} + if protoimpl.UnsafeEnabled { + mi := &file_metastore_v1_metastore_proto_msgTypes[3] + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + ms.StoreMessageInfo(mi) + } +} + +func (x *Dataset) String() string { + return protoimpl.X.MessageStringOf(x) +} + +func (*Dataset) ProtoMessage() {} + +func (x *Dataset) ProtoReflect() protoreflect.Message { + mi := &file_metastore_v1_metastore_proto_msgTypes[3] + if protoimpl.UnsafeEnabled && x != nil { + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + if ms.LoadMessageInfo() == nil { + ms.StoreMessageInfo(mi) + } + return ms + } + return mi.MessageOf(x) +} + +// Deprecated: Use Dataset.ProtoReflect.Descriptor instead. +func (*Dataset) Descriptor() ([]byte, []int) { + return file_metastore_v1_metastore_proto_rawDescGZIP(), []int{3} +} + +func (x *Dataset) GetLabels() []*v1.Labels { + if x != nil { + return x.Labels + } + return nil +} + +func (x *Dataset) GetTenantId() string { + if x != nil { + return x.TenantId + } + return "" +} + +func (x *Dataset) GetName() string { + if x != nil { + return x.Name + } + return "" +} + +func (x *Dataset) GetMinTime() int64 { + if x != nil { + return x.MinTime + } + return 0 +} + +func (x *Dataset) GetMaxTime() int64 { + if x != nil { + return x.MaxTime + } + return 0 +} + +func (x *Dataset) GetTableOfContents() []uint64 { + if x != nil { + return x.TableOfContents + } + return nil +} + +func (x *Dataset) GetSize() uint64 { + if x != nil { + return x.Size + } + return 0 +} + +func (x *Dataset) GetProfileTypes() []string { + if x != nil { + return x.ProfileTypes + } + return nil +} + +type QueryMetadataRequest struct { + state protoimpl.MessageState + sizeCache protoimpl.SizeCache + unknownFields protoimpl.UnknownFields + + TenantId []string `protobuf:"bytes,1,rep,name=tenant_id,json=tenantId,proto3" json:"tenant_id,omitempty"` + StartTime int64 `protobuf:"varint,2,opt,name=start_time,json=startTime,proto3" json:"start_time,omitempty"` + EndTime int64 `protobuf:"varint,3,opt,name=end_time,json=endTime,proto3" json:"end_time,omitempty"` + Query string `protobuf:"bytes,4,opt,name=query,proto3" json:"query,omitempty"` +} + +func (x *QueryMetadataRequest) Reset() { + *x = QueryMetadataRequest{} + if protoimpl.UnsafeEnabled { + mi := &file_metastore_v1_metastore_proto_msgTypes[4] + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + ms.StoreMessageInfo(mi) + } +} + +func (x *QueryMetadataRequest) String() string { + return protoimpl.X.MessageStringOf(x) +} + +func (*QueryMetadataRequest) ProtoMessage() {} + +func (x *QueryMetadataRequest) ProtoReflect() protoreflect.Message { + mi := &file_metastore_v1_metastore_proto_msgTypes[4] + if protoimpl.UnsafeEnabled && x != nil { + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + if ms.LoadMessageInfo() == nil { + ms.StoreMessageInfo(mi) + } + return ms + } + return mi.MessageOf(x) +} + +// Deprecated: Use QueryMetadataRequest.ProtoReflect.Descriptor instead. +func (*QueryMetadataRequest) Descriptor() ([]byte, []int) { + return file_metastore_v1_metastore_proto_rawDescGZIP(), []int{4} +} + +func (x *QueryMetadataRequest) GetTenantId() []string { + if x != nil { + return x.TenantId + } + return nil +} + +func (x *QueryMetadataRequest) GetStartTime() int64 { + if x != nil { + return x.StartTime + } + return 0 +} + +func (x *QueryMetadataRequest) GetEndTime() int64 { + if x != nil { + return x.EndTime + } + return 0 +} + +func (x *QueryMetadataRequest) GetQuery() string { + if x != nil { + return x.Query + } + return "" +} + +type QueryMetadataResponse struct { + state protoimpl.MessageState + sizeCache protoimpl.SizeCache + unknownFields protoimpl.UnknownFields + + Blocks []*BlockMeta `protobuf:"bytes,1,rep,name=blocks,proto3" json:"blocks,omitempty"` +} + +func (x *QueryMetadataResponse) Reset() { + *x = QueryMetadataResponse{} + if protoimpl.UnsafeEnabled { + mi := &file_metastore_v1_metastore_proto_msgTypes[5] + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + ms.StoreMessageInfo(mi) + } +} + +func (x *QueryMetadataResponse) String() string { + return protoimpl.X.MessageStringOf(x) +} + +func (*QueryMetadataResponse) ProtoMessage() {} + +func (x *QueryMetadataResponse) ProtoReflect() protoreflect.Message { + mi := &file_metastore_v1_metastore_proto_msgTypes[5] + if protoimpl.UnsafeEnabled && x != nil { + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + if ms.LoadMessageInfo() == nil { + ms.StoreMessageInfo(mi) + } + return ms + } + return mi.MessageOf(x) +} + +// Deprecated: Use QueryMetadataResponse.ProtoReflect.Descriptor instead. +func (*QueryMetadataResponse) Descriptor() ([]byte, []int) { + return file_metastore_v1_metastore_proto_rawDescGZIP(), []int{5} +} + +func (x *QueryMetadataResponse) GetBlocks() []*BlockMeta { + if x != nil { + return x.Blocks + } + return nil +} + +type ReadIndexRequest struct { + state protoimpl.MessageState + sizeCache protoimpl.SizeCache + unknownFields protoimpl.UnknownFields + + DebugRequestId string `protobuf:"bytes,1,opt,name=debug_request_id,json=debugRequestId,proto3" json:"debug_request_id,omitempty"` // for debug logging, // todo delete +} + +func (x *ReadIndexRequest) Reset() { + *x = ReadIndexRequest{} + if protoimpl.UnsafeEnabled { + mi := &file_metastore_v1_metastore_proto_msgTypes[6] + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + ms.StoreMessageInfo(mi) + } +} + +func (x *ReadIndexRequest) String() string { + return protoimpl.X.MessageStringOf(x) +} + +func (*ReadIndexRequest) ProtoMessage() {} + +func (x *ReadIndexRequest) ProtoReflect() protoreflect.Message { + mi := &file_metastore_v1_metastore_proto_msgTypes[6] + if protoimpl.UnsafeEnabled && x != nil { + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + if ms.LoadMessageInfo() == nil { + ms.StoreMessageInfo(mi) + } + return ms + } + return mi.MessageOf(x) +} + +// Deprecated: Use ReadIndexRequest.ProtoReflect.Descriptor instead. +func (*ReadIndexRequest) Descriptor() ([]byte, []int) { + return file_metastore_v1_metastore_proto_rawDescGZIP(), []int{6} +} + +func (x *ReadIndexRequest) GetDebugRequestId() string { + if x != nil { + return x.DebugRequestId + } + return "" +} + +type ReadIndexResponse struct { + state protoimpl.MessageState + sizeCache protoimpl.SizeCache + unknownFields protoimpl.UnknownFields + + ReadIndex uint64 `protobuf:"varint,1,opt,name=read_index,json=readIndex,proto3" json:"read_index,omitempty"` +} + +func (x *ReadIndexResponse) Reset() { + *x = ReadIndexResponse{} + if protoimpl.UnsafeEnabled { + mi := &file_metastore_v1_metastore_proto_msgTypes[7] + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + ms.StoreMessageInfo(mi) + } +} + +func (x *ReadIndexResponse) String() string { + return protoimpl.X.MessageStringOf(x) +} + +func (*ReadIndexResponse) ProtoMessage() {} + +func (x *ReadIndexResponse) ProtoReflect() protoreflect.Message { + mi := &file_metastore_v1_metastore_proto_msgTypes[7] + if protoimpl.UnsafeEnabled && x != nil { + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + if ms.LoadMessageInfo() == nil { + ms.StoreMessageInfo(mi) + } + return ms + } + return mi.MessageOf(x) +} + +// Deprecated: Use ReadIndexResponse.ProtoReflect.Descriptor instead. +func (*ReadIndexResponse) Descriptor() ([]byte, []int) { + return file_metastore_v1_metastore_proto_rawDescGZIP(), []int{7} +} + +func (x *ReadIndexResponse) GetReadIndex() uint64 { + if x != nil { + return x.ReadIndex + } + return 0 +} + +var File_metastore_v1_metastore_proto protoreflect.FileDescriptor + +var file_metastore_v1_metastore_proto_rawDesc = []byte{ + 0x0a, 0x1c, 0x6d, 0x65, 0x74, 0x61, 0x73, 0x74, 0x6f, 0x72, 0x65, 0x2f, 0x76, 0x31, 0x2f, 0x6d, + 0x65, 0x74, 0x61, 0x73, 0x74, 0x6f, 0x72, 0x65, 0x2e, 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x12, 0x0c, + 0x6d, 0x65, 0x74, 0x61, 0x73, 0x74, 0x6f, 0x72, 0x65, 0x2e, 0x76, 0x31, 0x1a, 0x14, 0x74, 0x79, + 0x70, 0x65, 0x73, 0x2f, 0x76, 0x31, 0x2f, 0x74, 0x79, 0x70, 0x65, 0x73, 0x2e, 0x70, 0x72, 0x6f, + 0x74, 0x6f, 0x22, 0x40, 0x0a, 0x0f, 0x41, 0x64, 0x64, 0x42, 0x6c, 0x6f, 0x63, 0x6b, 0x52, 0x65, + 0x71, 0x75, 0x65, 0x73, 0x74, 0x12, 0x2d, 0x0a, 0x05, 0x62, 0x6c, 0x6f, 0x63, 0x6b, 0x18, 0x01, + 0x20, 0x01, 0x28, 0x0b, 0x32, 0x17, 0x2e, 0x6d, 0x65, 0x74, 0x61, 0x73, 0x74, 0x6f, 0x72, 0x65, + 0x2e, 0x76, 0x31, 0x2e, 0x42, 0x6c, 0x6f, 0x63, 0x6b, 0x4d, 0x65, 0x74, 0x61, 0x52, 0x05, 0x62, + 0x6c, 0x6f, 0x63, 0x6b, 0x22, 0x12, 0x0a, 0x10, 0x41, 0x64, 0x64, 0x42, 0x6c, 0x6f, 0x63, 0x6b, + 0x52, 0x65, 0x73, 0x70, 0x6f, 0x6e, 0x73, 0x65, 0x22, 0x9d, 0x02, 0x0a, 0x09, 0x42, 0x6c, 0x6f, + 0x63, 0x6b, 0x4d, 0x65, 0x74, 0x61, 0x12, 0x25, 0x0a, 0x0e, 0x66, 0x6f, 0x72, 0x6d, 0x61, 0x74, + 0x5f, 0x76, 0x65, 0x72, 0x73, 0x69, 0x6f, 0x6e, 0x18, 0x01, 0x20, 0x01, 0x28, 0x04, 0x52, 0x0d, + 0x66, 0x6f, 0x72, 0x6d, 0x61, 0x74, 0x56, 0x65, 0x72, 0x73, 0x69, 0x6f, 0x6e, 0x12, 0x0e, 0x0a, + 0x02, 0x69, 0x64, 0x18, 0x02, 0x20, 0x01, 0x28, 0x09, 0x52, 0x02, 0x69, 0x64, 0x12, 0x19, 0x0a, + 0x08, 0x6d, 0x69, 0x6e, 0x5f, 0x74, 0x69, 0x6d, 0x65, 0x18, 0x03, 0x20, 0x01, 0x28, 0x03, 0x52, + 0x07, 0x6d, 0x69, 0x6e, 0x54, 0x69, 0x6d, 0x65, 0x12, 0x19, 0x0a, 0x08, 0x6d, 0x61, 0x78, 0x5f, + 0x74, 0x69, 0x6d, 0x65, 0x18, 0x04, 0x20, 0x01, 0x28, 0x03, 0x52, 0x07, 0x6d, 0x61, 0x78, 0x54, + 0x69, 0x6d, 0x65, 0x12, 0x14, 0x0a, 0x05, 0x73, 0x68, 0x61, 0x72, 0x64, 0x18, 0x05, 0x20, 0x01, + 0x28, 0x0d, 0x52, 0x05, 0x73, 0x68, 0x61, 0x72, 0x64, 0x12, 0x29, 0x0a, 0x10, 0x63, 0x6f, 0x6d, + 0x70, 0x61, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x5f, 0x6c, 0x65, 0x76, 0x65, 0x6c, 0x18, 0x06, 0x20, + 0x01, 0x28, 0x0d, 0x52, 0x0f, 0x63, 0x6f, 0x6d, 0x70, 0x61, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x4c, + 0x65, 0x76, 0x65, 0x6c, 0x12, 0x1b, 0x0a, 0x09, 0x74, 0x65, 0x6e, 0x61, 0x6e, 0x74, 0x5f, 0x69, + 0x64, 0x18, 0x07, 0x20, 0x01, 0x28, 0x09, 0x52, 0x08, 0x74, 0x65, 0x6e, 0x61, 0x6e, 0x74, 0x49, + 0x64, 0x12, 0x31, 0x0a, 0x08, 0x64, 0x61, 0x74, 0x61, 0x73, 0x65, 0x74, 0x73, 0x18, 0x08, 0x20, + 0x03, 0x28, 0x0b, 0x32, 0x15, 0x2e, 0x6d, 0x65, 0x74, 0x61, 0x73, 0x74, 0x6f, 0x72, 0x65, 0x2e, + 0x76, 0x31, 0x2e, 0x44, 0x61, 0x74, 0x61, 0x73, 0x65, 0x74, 0x52, 0x08, 0x64, 0x61, 0x74, 0x61, + 0x73, 0x65, 0x74, 0x73, 0x12, 0x12, 0x0a, 0x04, 0x73, 0x69, 0x7a, 0x65, 0x18, 0x09, 0x20, 0x01, + 0x28, 0x04, 0x52, 0x04, 0x73, 0x69, 0x7a, 0x65, 0x22, 0xff, 0x01, 0x0a, 0x07, 0x44, 0x61, 0x74, + 0x61, 0x73, 0x65, 0x74, 0x12, 0x28, 0x0a, 0x06, 0x6c, 0x61, 0x62, 0x65, 0x6c, 0x73, 0x18, 0x08, + 0x20, 0x03, 0x28, 0x0b, 0x32, 0x10, 0x2e, 0x74, 0x79, 0x70, 0x65, 0x73, 0x2e, 0x76, 0x31, 0x2e, + 0x4c, 0x61, 0x62, 0x65, 0x6c, 0x73, 0x52, 0x06, 0x6c, 0x61, 0x62, 0x65, 0x6c, 0x73, 0x12, 0x1b, + 0x0a, 0x09, 0x74, 0x65, 0x6e, 0x61, 0x6e, 0x74, 0x5f, 0x69, 0x64, 0x18, 0x01, 0x20, 0x01, 0x28, + 0x09, 0x52, 0x08, 0x74, 0x65, 0x6e, 0x61, 0x6e, 0x74, 0x49, 0x64, 0x12, 0x12, 0x0a, 0x04, 0x6e, + 0x61, 0x6d, 0x65, 0x18, 0x02, 0x20, 0x01, 0x28, 0x09, 0x52, 0x04, 0x6e, 0x61, 0x6d, 0x65, 0x12, + 0x19, 0x0a, 0x08, 0x6d, 0x69, 0x6e, 0x5f, 0x74, 0x69, 0x6d, 0x65, 0x18, 0x03, 0x20, 0x01, 0x28, + 0x03, 0x52, 0x07, 0x6d, 0x69, 0x6e, 0x54, 0x69, 0x6d, 0x65, 0x12, 0x19, 0x0a, 0x08, 0x6d, 0x61, + 0x78, 0x5f, 0x74, 0x69, 0x6d, 0x65, 0x18, 0x04, 0x20, 0x01, 0x28, 0x03, 0x52, 0x07, 0x6d, 0x61, + 0x78, 0x54, 0x69, 0x6d, 0x65, 0x12, 0x2a, 0x0a, 0x11, 0x74, 0x61, 0x62, 0x6c, 0x65, 0x5f, 0x6f, + 0x66, 0x5f, 0x63, 0x6f, 0x6e, 0x74, 0x65, 0x6e, 0x74, 0x73, 0x18, 0x05, 0x20, 0x03, 0x28, 0x04, + 0x52, 0x0f, 0x74, 0x61, 0x62, 0x6c, 0x65, 0x4f, 0x66, 0x43, 0x6f, 0x6e, 0x74, 0x65, 0x6e, 0x74, + 0x73, 0x12, 0x12, 0x0a, 0x04, 0x73, 0x69, 0x7a, 0x65, 0x18, 0x06, 0x20, 0x01, 0x28, 0x04, 0x52, + 0x04, 0x73, 0x69, 0x7a, 0x65, 0x12, 0x23, 0x0a, 0x0d, 0x70, 0x72, 0x6f, 0x66, 0x69, 0x6c, 0x65, + 0x5f, 0x74, 0x79, 0x70, 0x65, 0x73, 0x18, 0x07, 0x20, 0x03, 0x28, 0x09, 0x52, 0x0c, 0x70, 0x72, + 0x6f, 0x66, 0x69, 0x6c, 0x65, 0x54, 0x79, 0x70, 0x65, 0x73, 0x22, 0x83, 0x01, 0x0a, 0x14, 0x51, + 0x75, 0x65, 0x72, 0x79, 0x4d, 0x65, 0x74, 0x61, 0x64, 0x61, 0x74, 0x61, 0x52, 0x65, 0x71, 0x75, + 0x65, 0x73, 0x74, 0x12, 0x1b, 0x0a, 0x09, 0x74, 0x65, 0x6e, 0x61, 0x6e, 0x74, 0x5f, 0x69, 0x64, + 0x18, 0x01, 0x20, 0x03, 0x28, 0x09, 0x52, 0x08, 0x74, 0x65, 0x6e, 0x61, 0x6e, 0x74, 0x49, 0x64, + 0x12, 0x1d, 0x0a, 0x0a, 0x73, 0x74, 0x61, 0x72, 0x74, 0x5f, 0x74, 0x69, 0x6d, 0x65, 0x18, 0x02, + 0x20, 0x01, 0x28, 0x03, 0x52, 0x09, 0x73, 0x74, 0x61, 0x72, 0x74, 0x54, 0x69, 0x6d, 0x65, 0x12, + 0x19, 0x0a, 0x08, 0x65, 0x6e, 0x64, 0x5f, 0x74, 0x69, 0x6d, 0x65, 0x18, 0x03, 0x20, 0x01, 0x28, + 0x03, 0x52, 0x07, 0x65, 0x6e, 0x64, 0x54, 0x69, 0x6d, 0x65, 0x12, 0x14, 0x0a, 0x05, 0x71, 0x75, + 0x65, 0x72, 0x79, 0x18, 0x04, 0x20, 0x01, 0x28, 0x09, 0x52, 0x05, 0x71, 0x75, 0x65, 0x72, 0x79, + 0x22, 0x48, 0x0a, 0x15, 0x51, 0x75, 0x65, 0x72, 0x79, 0x4d, 0x65, 0x74, 0x61, 0x64, 0x61, 0x74, + 0x61, 0x52, 0x65, 0x73, 0x70, 0x6f, 0x6e, 0x73, 0x65, 0x12, 0x2f, 0x0a, 0x06, 0x62, 0x6c, 0x6f, + 0x63, 0x6b, 0x73, 0x18, 0x01, 0x20, 0x03, 0x28, 0x0b, 0x32, 0x17, 0x2e, 0x6d, 0x65, 0x74, 0x61, + 0x73, 0x74, 0x6f, 0x72, 0x65, 0x2e, 0x76, 0x31, 0x2e, 0x42, 0x6c, 0x6f, 0x63, 0x6b, 0x4d, 0x65, + 0x74, 0x61, 0x52, 0x06, 0x62, 0x6c, 0x6f, 0x63, 0x6b, 0x73, 0x22, 0x3c, 0x0a, 0x10, 0x52, 0x65, + 0x61, 0x64, 0x49, 0x6e, 0x64, 0x65, 0x78, 0x52, 0x65, 0x71, 0x75, 0x65, 0x73, 0x74, 0x12, 0x28, + 0x0a, 0x10, 0x64, 0x65, 0x62, 0x75, 0x67, 0x5f, 0x72, 0x65, 0x71, 0x75, 0x65, 0x73, 0x74, 0x5f, + 0x69, 0x64, 0x18, 0x01, 0x20, 0x01, 0x28, 0x09, 0x52, 0x0e, 0x64, 0x65, 0x62, 0x75, 0x67, 0x52, + 0x65, 0x71, 0x75, 0x65, 0x73, 0x74, 0x49, 0x64, 0x22, 0x32, 0x0a, 0x11, 0x52, 0x65, 0x61, 0x64, + 0x49, 0x6e, 0x64, 0x65, 0x78, 0x52, 0x65, 0x73, 0x70, 0x6f, 0x6e, 0x73, 0x65, 0x12, 0x1d, 0x0a, + 0x0a, 0x72, 0x65, 0x61, 0x64, 0x5f, 0x69, 0x6e, 0x64, 0x65, 0x78, 0x18, 0x01, 0x20, 0x01, 0x28, + 0x04, 0x52, 0x09, 0x72, 0x65, 0x61, 0x64, 0x49, 0x6e, 0x64, 0x65, 0x78, 0x32, 0x8b, 0x02, 0x0a, + 0x10, 0x4d, 0x65, 0x74, 0x61, 0x73, 0x74, 0x6f, 0x72, 0x65, 0x53, 0x65, 0x72, 0x76, 0x69, 0x63, + 0x65, 0x12, 0x4b, 0x0a, 0x08, 0x41, 0x64, 0x64, 0x42, 0x6c, 0x6f, 0x63, 0x6b, 0x12, 0x1d, 0x2e, + 0x6d, 0x65, 0x74, 0x61, 0x73, 0x74, 0x6f, 0x72, 0x65, 0x2e, 0x76, 0x31, 0x2e, 0x41, 0x64, 0x64, + 0x42, 0x6c, 0x6f, 0x63, 0x6b, 0x52, 0x65, 0x71, 0x75, 0x65, 0x73, 0x74, 0x1a, 0x1e, 0x2e, 0x6d, + 0x65, 0x74, 0x61, 0x73, 0x74, 0x6f, 0x72, 0x65, 0x2e, 0x76, 0x31, 0x2e, 0x41, 0x64, 0x64, 0x42, + 0x6c, 0x6f, 0x63, 0x6b, 0x52, 0x65, 0x73, 0x70, 0x6f, 0x6e, 0x73, 0x65, 0x22, 0x00, 0x12, 0x5a, + 0x0a, 0x0d, 0x51, 0x75, 0x65, 0x72, 0x79, 0x4d, 0x65, 0x74, 0x61, 0x64, 0x61, 0x74, 0x61, 0x12, + 0x22, 0x2e, 0x6d, 0x65, 0x74, 0x61, 0x73, 0x74, 0x6f, 0x72, 0x65, 0x2e, 0x76, 0x31, 0x2e, 0x51, + 0x75, 0x65, 0x72, 0x79, 0x4d, 0x65, 0x74, 0x61, 0x64, 0x61, 0x74, 0x61, 0x52, 0x65, 0x71, 0x75, + 0x65, 0x73, 0x74, 0x1a, 0x23, 0x2e, 0x6d, 0x65, 0x74, 0x61, 0x73, 0x74, 0x6f, 0x72, 0x65, 0x2e, + 0x76, 0x31, 0x2e, 0x51, 0x75, 0x65, 0x72, 0x79, 0x4d, 0x65, 0x74, 0x61, 0x64, 0x61, 0x74, 0x61, + 0x52, 0x65, 0x73, 0x70, 0x6f, 0x6e, 0x73, 0x65, 0x22, 0x00, 0x12, 0x4e, 0x0a, 0x09, 0x52, 0x65, + 0x61, 0x64, 0x49, 0x6e, 0x64, 0x65, 0x78, 0x12, 0x1e, 0x2e, 0x6d, 0x65, 0x74, 0x61, 0x73, 0x74, + 0x6f, 0x72, 0x65, 0x2e, 0x76, 0x31, 0x2e, 0x52, 0x65, 0x61, 0x64, 0x49, 0x6e, 0x64, 0x65, 0x78, + 0x52, 0x65, 0x71, 0x75, 0x65, 0x73, 0x74, 0x1a, 0x1f, 0x2e, 0x6d, 0x65, 0x74, 0x61, 0x73, 0x74, + 0x6f, 0x72, 0x65, 0x2e, 0x76, 0x31, 0x2e, 0x52, 0x65, 0x61, 0x64, 0x49, 0x6e, 0x64, 0x65, 0x78, + 0x52, 0x65, 0x73, 0x70, 0x6f, 0x6e, 0x73, 0x65, 0x22, 0x00, 0x42, 0xbb, 0x01, 0x0a, 0x10, 0x63, + 0x6f, 0x6d, 0x2e, 0x6d, 0x65, 0x74, 0x61, 0x73, 0x74, 0x6f, 0x72, 0x65, 0x2e, 0x76, 0x31, 0x42, + 0x0e, 0x4d, 0x65, 0x74, 0x61, 0x73, 0x74, 0x6f, 0x72, 0x65, 0x50, 0x72, 0x6f, 0x74, 0x6f, 0x50, + 0x01, 0x5a, 0x46, 0x67, 0x69, 0x74, 0x68, 0x75, 0x62, 0x2e, 0x63, 0x6f, 0x6d, 0x2f, 0x67, 0x72, + 0x61, 0x66, 0x61, 0x6e, 0x61, 0x2f, 0x70, 0x79, 0x72, 0x6f, 0x73, 0x63, 0x6f, 0x70, 0x65, 0x2f, + 0x61, 0x70, 0x69, 0x2f, 0x67, 0x65, 0x6e, 0x2f, 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x2f, 0x67, 0x6f, + 0x2f, 0x6d, 0x65, 0x74, 0x61, 0x73, 0x74, 0x6f, 0x72, 0x65, 0x2f, 0x76, 0x31, 0x3b, 0x6d, 0x65, + 0x74, 0x61, 0x73, 0x74, 0x6f, 0x72, 0x65, 0x76, 0x31, 0xa2, 0x02, 0x03, 0x4d, 0x58, 0x58, 0xaa, + 0x02, 0x0c, 0x4d, 0x65, 0x74, 0x61, 0x73, 0x74, 0x6f, 0x72, 0x65, 0x2e, 0x56, 0x31, 0xca, 0x02, + 0x0c, 0x4d, 0x65, 0x74, 0x61, 0x73, 0x74, 0x6f, 0x72, 0x65, 0x5c, 0x56, 0x31, 0xe2, 0x02, 0x18, + 0x4d, 0x65, 0x74, 0x61, 0x73, 0x74, 0x6f, 0x72, 0x65, 0x5c, 0x56, 0x31, 0x5c, 0x47, 0x50, 0x42, + 0x4d, 0x65, 0x74, 0x61, 0x64, 0x61, 0x74, 0x61, 0xea, 0x02, 0x0d, 0x4d, 0x65, 0x74, 0x61, 0x73, + 0x74, 0x6f, 0x72, 0x65, 0x3a, 0x3a, 0x56, 0x31, 0x62, 0x06, 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x33, +} + +var ( + file_metastore_v1_metastore_proto_rawDescOnce sync.Once + file_metastore_v1_metastore_proto_rawDescData = file_metastore_v1_metastore_proto_rawDesc +) + +func file_metastore_v1_metastore_proto_rawDescGZIP() []byte { + file_metastore_v1_metastore_proto_rawDescOnce.Do(func() { + file_metastore_v1_metastore_proto_rawDescData = protoimpl.X.CompressGZIP(file_metastore_v1_metastore_proto_rawDescData) + }) + return file_metastore_v1_metastore_proto_rawDescData +} + +var file_metastore_v1_metastore_proto_msgTypes = make([]protoimpl.MessageInfo, 8) +var file_metastore_v1_metastore_proto_goTypes = []any{ + (*AddBlockRequest)(nil), // 0: metastore.v1.AddBlockRequest + (*AddBlockResponse)(nil), // 1: metastore.v1.AddBlockResponse + (*BlockMeta)(nil), // 2: metastore.v1.BlockMeta + (*Dataset)(nil), // 3: metastore.v1.Dataset + (*QueryMetadataRequest)(nil), // 4: metastore.v1.QueryMetadataRequest + (*QueryMetadataResponse)(nil), // 5: metastore.v1.QueryMetadataResponse + (*ReadIndexRequest)(nil), // 6: metastore.v1.ReadIndexRequest + (*ReadIndexResponse)(nil), // 7: metastore.v1.ReadIndexResponse + (*v1.Labels)(nil), // 8: types.v1.Labels +} +var file_metastore_v1_metastore_proto_depIdxs = []int32{ + 2, // 0: metastore.v1.AddBlockRequest.block:type_name -> metastore.v1.BlockMeta + 3, // 1: metastore.v1.BlockMeta.datasets:type_name -> metastore.v1.Dataset + 8, // 2: metastore.v1.Dataset.labels:type_name -> types.v1.Labels + 2, // 3: metastore.v1.QueryMetadataResponse.blocks:type_name -> metastore.v1.BlockMeta + 0, // 4: metastore.v1.MetastoreService.AddBlock:input_type -> metastore.v1.AddBlockRequest + 4, // 5: metastore.v1.MetastoreService.QueryMetadata:input_type -> metastore.v1.QueryMetadataRequest + 6, // 6: metastore.v1.MetastoreService.ReadIndex:input_type -> metastore.v1.ReadIndexRequest + 1, // 7: metastore.v1.MetastoreService.AddBlock:output_type -> metastore.v1.AddBlockResponse + 5, // 8: metastore.v1.MetastoreService.QueryMetadata:output_type -> metastore.v1.QueryMetadataResponse + 7, // 9: metastore.v1.MetastoreService.ReadIndex:output_type -> metastore.v1.ReadIndexResponse + 7, // [7:10] is the sub-list for method output_type + 4, // [4:7] is the sub-list for method input_type + 4, // [4:4] is the sub-list for extension type_name + 4, // [4:4] is the sub-list for extension extendee + 0, // [0:4] is the sub-list for field type_name +} + +func init() { file_metastore_v1_metastore_proto_init() } +func file_metastore_v1_metastore_proto_init() { + if File_metastore_v1_metastore_proto != nil { + return + } + if !protoimpl.UnsafeEnabled { + file_metastore_v1_metastore_proto_msgTypes[0].Exporter = func(v any, i int) any { + switch v := v.(*AddBlockRequest); i { + case 0: + return &v.state + case 1: + return &v.sizeCache + case 2: + return &v.unknownFields + default: + return nil + } + } + file_metastore_v1_metastore_proto_msgTypes[1].Exporter = func(v any, i int) any { + switch v := v.(*AddBlockResponse); i { + case 0: + return &v.state + case 1: + return &v.sizeCache + case 2: + return &v.unknownFields + default: + return nil + } + } + file_metastore_v1_metastore_proto_msgTypes[2].Exporter = func(v any, i int) any { + switch v := v.(*BlockMeta); i { + case 0: + return &v.state + case 1: + return &v.sizeCache + case 2: + return &v.unknownFields + default: + return nil + } + } + file_metastore_v1_metastore_proto_msgTypes[3].Exporter = func(v any, i int) any { + switch v := v.(*Dataset); i { + case 0: + return &v.state + case 1: + return &v.sizeCache + case 2: + return &v.unknownFields + default: + return nil + } + } + file_metastore_v1_metastore_proto_msgTypes[4].Exporter = func(v any, i int) any { + switch v := v.(*QueryMetadataRequest); i { + case 0: + return &v.state + case 1: + return &v.sizeCache + case 2: + return &v.unknownFields + default: + return nil + } + } + file_metastore_v1_metastore_proto_msgTypes[5].Exporter = func(v any, i int) any { + switch v := v.(*QueryMetadataResponse); i { + case 0: + return &v.state + case 1: + return &v.sizeCache + case 2: + return &v.unknownFields + default: + return nil + } + } + file_metastore_v1_metastore_proto_msgTypes[6].Exporter = func(v any, i int) any { + switch v := v.(*ReadIndexRequest); i { + case 0: + return &v.state + case 1: + return &v.sizeCache + case 2: + return &v.unknownFields + default: + return nil + } + } + file_metastore_v1_metastore_proto_msgTypes[7].Exporter = func(v any, i int) any { + switch v := v.(*ReadIndexResponse); i { + case 0: + return &v.state + case 1: + return &v.sizeCache + case 2: + return &v.unknownFields + default: + return nil + } + } + } + type x struct{} + out := protoimpl.TypeBuilder{ + File: protoimpl.DescBuilder{ + GoPackagePath: reflect.TypeOf(x{}).PkgPath(), + RawDescriptor: file_metastore_v1_metastore_proto_rawDesc, + NumEnums: 0, + NumMessages: 8, + NumExtensions: 0, + NumServices: 1, + }, + GoTypes: file_metastore_v1_metastore_proto_goTypes, + DependencyIndexes: file_metastore_v1_metastore_proto_depIdxs, + MessageInfos: file_metastore_v1_metastore_proto_msgTypes, + }.Build() + File_metastore_v1_metastore_proto = out.File + file_metastore_v1_metastore_proto_rawDesc = nil + file_metastore_v1_metastore_proto_goTypes = nil + file_metastore_v1_metastore_proto_depIdxs = nil +} diff --git a/api/gen/proto/go/metastore/v1/metastore_vtproto.pb.go b/api/gen/proto/go/metastore/v1/metastore_vtproto.pb.go new file mode 100644 index 0000000000..fca443b263 --- /dev/null +++ b/api/gen/proto/go/metastore/v1/metastore_vtproto.pb.go @@ -0,0 +1,2407 @@ +// Code generated by protoc-gen-go-vtproto. DO NOT EDIT. +// protoc-gen-go-vtproto version: v0.6.0 +// source: metastore/v1/metastore.proto + +package metastorev1 + +import ( + context "context" + fmt "fmt" + v1 "github.com/grafana/pyroscope/api/gen/proto/go/types/v1" + protohelpers "github.com/planetscale/vtprotobuf/protohelpers" + grpc "google.golang.org/grpc" + codes "google.golang.org/grpc/codes" + status "google.golang.org/grpc/status" + proto "google.golang.org/protobuf/proto" + protoimpl "google.golang.org/protobuf/runtime/protoimpl" + io "io" +) + +const ( + // Verify that this generated code is sufficiently up-to-date. + _ = protoimpl.EnforceVersion(20 - protoimpl.MinVersion) + // Verify that runtime/protoimpl is sufficiently up-to-date. + _ = protoimpl.EnforceVersion(protoimpl.MaxVersion - 20) +) + +func (m *AddBlockRequest) CloneVT() *AddBlockRequest { + if m == nil { + return (*AddBlockRequest)(nil) + } + r := new(AddBlockRequest) + r.Block = m.Block.CloneVT() + if len(m.unknownFields) > 0 { + r.unknownFields = make([]byte, len(m.unknownFields)) + copy(r.unknownFields, m.unknownFields) + } + return r +} + +func (m *AddBlockRequest) CloneMessageVT() proto.Message { + return m.CloneVT() +} + +func (m *AddBlockResponse) CloneVT() *AddBlockResponse { + if m == nil { + return (*AddBlockResponse)(nil) + } + r := new(AddBlockResponse) + if len(m.unknownFields) > 0 { + r.unknownFields = make([]byte, len(m.unknownFields)) + copy(r.unknownFields, m.unknownFields) + } + return r +} + +func (m *AddBlockResponse) CloneMessageVT() proto.Message { + return m.CloneVT() +} + +func (m *BlockMeta) CloneVT() *BlockMeta { + if m == nil { + return (*BlockMeta)(nil) + } + r := new(BlockMeta) + r.FormatVersion = m.FormatVersion + r.Id = m.Id + r.MinTime = m.MinTime + r.MaxTime = m.MaxTime + r.Shard = m.Shard + r.CompactionLevel = m.CompactionLevel + r.TenantId = m.TenantId + r.Size = m.Size + if rhs := m.Datasets; rhs != nil { + tmpContainer := make([]*Dataset, len(rhs)) + for k, v := range rhs { + tmpContainer[k] = v.CloneVT() + } + r.Datasets = tmpContainer + } + if len(m.unknownFields) > 0 { + r.unknownFields = make([]byte, len(m.unknownFields)) + copy(r.unknownFields, m.unknownFields) + } + return r +} + +func (m *BlockMeta) CloneMessageVT() proto.Message { + return m.CloneVT() +} + +func (m *Dataset) CloneVT() *Dataset { + if m == nil { + return (*Dataset)(nil) + } + r := new(Dataset) + r.TenantId = m.TenantId + r.Name = m.Name + r.MinTime = m.MinTime + r.MaxTime = m.MaxTime + r.Size = m.Size + if rhs := m.Labels; rhs != nil { + tmpContainer := make([]*v1.Labels, len(rhs)) + for k, v := range rhs { + if vtpb, ok := interface{}(v).(interface{ CloneVT() *v1.Labels }); ok { + tmpContainer[k] = vtpb.CloneVT() + } else { + tmpContainer[k] = proto.Clone(v).(*v1.Labels) + } + } + r.Labels = tmpContainer + } + if rhs := m.TableOfContents; rhs != nil { + tmpContainer := make([]uint64, len(rhs)) + copy(tmpContainer, rhs) + r.TableOfContents = tmpContainer + } + if rhs := m.ProfileTypes; rhs != nil { + tmpContainer := make([]string, len(rhs)) + copy(tmpContainer, rhs) + r.ProfileTypes = tmpContainer + } + if len(m.unknownFields) > 0 { + r.unknownFields = make([]byte, len(m.unknownFields)) + copy(r.unknownFields, m.unknownFields) + } + return r +} + +func (m *Dataset) CloneMessageVT() proto.Message { + return m.CloneVT() +} + +func (m *QueryMetadataRequest) CloneVT() *QueryMetadataRequest { + if m == nil { + return (*QueryMetadataRequest)(nil) + } + r := new(QueryMetadataRequest) + r.StartTime = m.StartTime + r.EndTime = m.EndTime + r.Query = m.Query + if rhs := m.TenantId; rhs != nil { + tmpContainer := make([]string, len(rhs)) + copy(tmpContainer, rhs) + r.TenantId = tmpContainer + } + if len(m.unknownFields) > 0 { + r.unknownFields = make([]byte, len(m.unknownFields)) + copy(r.unknownFields, m.unknownFields) + } + return r +} + +func (m *QueryMetadataRequest) CloneMessageVT() proto.Message { + return m.CloneVT() +} + +func (m *QueryMetadataResponse) CloneVT() *QueryMetadataResponse { + if m == nil { + return (*QueryMetadataResponse)(nil) + } + r := new(QueryMetadataResponse) + if rhs := m.Blocks; rhs != nil { + tmpContainer := make([]*BlockMeta, len(rhs)) + for k, v := range rhs { + tmpContainer[k] = v.CloneVT() + } + r.Blocks = tmpContainer + } + if len(m.unknownFields) > 0 { + r.unknownFields = make([]byte, len(m.unknownFields)) + copy(r.unknownFields, m.unknownFields) + } + return r +} + +func (m *QueryMetadataResponse) CloneMessageVT() proto.Message { + return m.CloneVT() +} + +func (m *ReadIndexRequest) CloneVT() *ReadIndexRequest { + if m == nil { + return (*ReadIndexRequest)(nil) + } + r := new(ReadIndexRequest) + r.DebugRequestId = m.DebugRequestId + if len(m.unknownFields) > 0 { + r.unknownFields = make([]byte, len(m.unknownFields)) + copy(r.unknownFields, m.unknownFields) + } + return r +} + +func (m *ReadIndexRequest) CloneMessageVT() proto.Message { + return m.CloneVT() +} + +func (m *ReadIndexResponse) CloneVT() *ReadIndexResponse { + if m == nil { + return (*ReadIndexResponse)(nil) + } + r := new(ReadIndexResponse) + r.ReadIndex = m.ReadIndex + if len(m.unknownFields) > 0 { + r.unknownFields = make([]byte, len(m.unknownFields)) + copy(r.unknownFields, m.unknownFields) + } + return r +} + +func (m *ReadIndexResponse) CloneMessageVT() proto.Message { + return m.CloneVT() +} + +func (this *AddBlockRequest) EqualVT(that *AddBlockRequest) bool { + if this == that { + return true + } else if this == nil || that == nil { + return false + } + if !this.Block.EqualVT(that.Block) { + return false + } + return string(this.unknownFields) == string(that.unknownFields) +} + +func (this *AddBlockRequest) EqualMessageVT(thatMsg proto.Message) bool { + that, ok := thatMsg.(*AddBlockRequest) + if !ok { + return false + } + return this.EqualVT(that) +} +func (this *AddBlockResponse) EqualVT(that *AddBlockResponse) bool { + if this == that { + return true + } else if this == nil || that == nil { + return false + } + return string(this.unknownFields) == string(that.unknownFields) +} + +func (this *AddBlockResponse) EqualMessageVT(thatMsg proto.Message) bool { + that, ok := thatMsg.(*AddBlockResponse) + if !ok { + return false + } + return this.EqualVT(that) +} +func (this *BlockMeta) EqualVT(that *BlockMeta) bool { + if this == that { + return true + } else if this == nil || that == nil { + return false + } + if this.FormatVersion != that.FormatVersion { + return false + } + if this.Id != that.Id { + return false + } + if this.MinTime != that.MinTime { + return false + } + if this.MaxTime != that.MaxTime { + return false + } + if this.Shard != that.Shard { + return false + } + if this.CompactionLevel != that.CompactionLevel { + return false + } + if this.TenantId != that.TenantId { + return false + } + if len(this.Datasets) != len(that.Datasets) { + return false + } + for i, vx := range this.Datasets { + vy := that.Datasets[i] + if p, q := vx, vy; p != q { + if p == nil { + p = &Dataset{} + } + if q == nil { + q = &Dataset{} + } + if !p.EqualVT(q) { + return false + } + } + } + if this.Size != that.Size { + return false + } + return string(this.unknownFields) == string(that.unknownFields) +} + +func (this *BlockMeta) EqualMessageVT(thatMsg proto.Message) bool { + that, ok := thatMsg.(*BlockMeta) + if !ok { + return false + } + return this.EqualVT(that) +} +func (this *Dataset) EqualVT(that *Dataset) bool { + if this == that { + return true + } else if this == nil || that == nil { + return false + } + if this.TenantId != that.TenantId { + return false + } + if this.Name != that.Name { + return false + } + if this.MinTime != that.MinTime { + return false + } + if this.MaxTime != that.MaxTime { + return false + } + if len(this.TableOfContents) != len(that.TableOfContents) { + return false + } + for i, vx := range this.TableOfContents { + vy := that.TableOfContents[i] + if vx != vy { + return false + } + } + if this.Size != that.Size { + return false + } + if len(this.ProfileTypes) != len(that.ProfileTypes) { + return false + } + for i, vx := range this.ProfileTypes { + vy := that.ProfileTypes[i] + if vx != vy { + return false + } + } + if len(this.Labels) != len(that.Labels) { + return false + } + for i, vx := range this.Labels { + vy := that.Labels[i] + if p, q := vx, vy; p != q { + if p == nil { + p = &v1.Labels{} + } + if q == nil { + q = &v1.Labels{} + } + if equal, ok := interface{}(p).(interface{ EqualVT(*v1.Labels) bool }); ok { + if !equal.EqualVT(q) { + return false + } + } else if !proto.Equal(p, q) { + return false + } + } + } + return string(this.unknownFields) == string(that.unknownFields) +} + +func (this *Dataset) EqualMessageVT(thatMsg proto.Message) bool { + that, ok := thatMsg.(*Dataset) + if !ok { + return false + } + return this.EqualVT(that) +} +func (this *QueryMetadataRequest) EqualVT(that *QueryMetadataRequest) bool { + if this == that { + return true + } else if this == nil || that == nil { + return false + } + if len(this.TenantId) != len(that.TenantId) { + return false + } + for i, vx := range this.TenantId { + vy := that.TenantId[i] + if vx != vy { + return false + } + } + if this.StartTime != that.StartTime { + return false + } + if this.EndTime != that.EndTime { + return false + } + if this.Query != that.Query { + return false + } + return string(this.unknownFields) == string(that.unknownFields) +} + +func (this *QueryMetadataRequest) EqualMessageVT(thatMsg proto.Message) bool { + that, ok := thatMsg.(*QueryMetadataRequest) + if !ok { + return false + } + return this.EqualVT(that) +} +func (this *QueryMetadataResponse) EqualVT(that *QueryMetadataResponse) bool { + if this == that { + return true + } else if this == nil || that == nil { + return false + } + if len(this.Blocks) != len(that.Blocks) { + return false + } + for i, vx := range this.Blocks { + vy := that.Blocks[i] + if p, q := vx, vy; p != q { + if p == nil { + p = &BlockMeta{} + } + if q == nil { + q = &BlockMeta{} + } + if !p.EqualVT(q) { + return false + } + } + } + return string(this.unknownFields) == string(that.unknownFields) +} + +func (this *QueryMetadataResponse) EqualMessageVT(thatMsg proto.Message) bool { + that, ok := thatMsg.(*QueryMetadataResponse) + if !ok { + return false + } + return this.EqualVT(that) +} +func (this *ReadIndexRequest) EqualVT(that *ReadIndexRequest) bool { + if this == that { + return true + } else if this == nil || that == nil { + return false + } + if this.DebugRequestId != that.DebugRequestId { + return false + } + return string(this.unknownFields) == string(that.unknownFields) +} + +func (this *ReadIndexRequest) EqualMessageVT(thatMsg proto.Message) bool { + that, ok := thatMsg.(*ReadIndexRequest) + if !ok { + return false + } + return this.EqualVT(that) +} +func (this *ReadIndexResponse) EqualVT(that *ReadIndexResponse) bool { + if this == that { + return true + } else if this == nil || that == nil { + return false + } + if this.ReadIndex != that.ReadIndex { + return false + } + return string(this.unknownFields) == string(that.unknownFields) +} + +func (this *ReadIndexResponse) EqualMessageVT(thatMsg proto.Message) bool { + that, ok := thatMsg.(*ReadIndexResponse) + if !ok { + return false + } + return this.EqualVT(that) +} + +// This is a compile-time assertion to ensure that this generated file +// is compatible with the grpc package it is being compiled against. +// Requires gRPC-Go v1.32.0 or later. +const _ = grpc.SupportPackageIsVersion7 + +// MetastoreServiceClient is the client API for MetastoreService service. +// +// For semantics around ctx use and closing/ending streaming RPCs, please refer to https://pkg.go.dev/google.golang.org/grpc/?tab=doc#ClientConn.NewStream. +type MetastoreServiceClient interface { + AddBlock(ctx context.Context, in *AddBlockRequest, opts ...grpc.CallOption) (*AddBlockResponse, error) + QueryMetadata(ctx context.Context, in *QueryMetadataRequest, opts ...grpc.CallOption) (*QueryMetadataResponse, error) + ReadIndex(ctx context.Context, in *ReadIndexRequest, opts ...grpc.CallOption) (*ReadIndexResponse, error) +} + +type metastoreServiceClient struct { + cc grpc.ClientConnInterface +} + +func NewMetastoreServiceClient(cc grpc.ClientConnInterface) MetastoreServiceClient { + return &metastoreServiceClient{cc} +} + +func (c *metastoreServiceClient) AddBlock(ctx context.Context, in *AddBlockRequest, opts ...grpc.CallOption) (*AddBlockResponse, error) { + out := new(AddBlockResponse) + err := c.cc.Invoke(ctx, "/metastore.v1.MetastoreService/AddBlock", in, out, opts...) + if err != nil { + return nil, err + } + return out, nil +} + +func (c *metastoreServiceClient) QueryMetadata(ctx context.Context, in *QueryMetadataRequest, opts ...grpc.CallOption) (*QueryMetadataResponse, error) { + out := new(QueryMetadataResponse) + err := c.cc.Invoke(ctx, "/metastore.v1.MetastoreService/QueryMetadata", in, out, opts...) + if err != nil { + return nil, err + } + return out, nil +} + +func (c *metastoreServiceClient) ReadIndex(ctx context.Context, in *ReadIndexRequest, opts ...grpc.CallOption) (*ReadIndexResponse, error) { + out := new(ReadIndexResponse) + err := c.cc.Invoke(ctx, "/metastore.v1.MetastoreService/ReadIndex", in, out, opts...) + if err != nil { + return nil, err + } + return out, nil +} + +// MetastoreServiceServer is the server API for MetastoreService service. +// All implementations must embed UnimplementedMetastoreServiceServer +// for forward compatibility +type MetastoreServiceServer interface { + AddBlock(context.Context, *AddBlockRequest) (*AddBlockResponse, error) + QueryMetadata(context.Context, *QueryMetadataRequest) (*QueryMetadataResponse, error) + ReadIndex(context.Context, *ReadIndexRequest) (*ReadIndexResponse, error) + mustEmbedUnimplementedMetastoreServiceServer() +} + +// UnimplementedMetastoreServiceServer must be embedded to have forward compatible implementations. +type UnimplementedMetastoreServiceServer struct { +} + +func (UnimplementedMetastoreServiceServer) AddBlock(context.Context, *AddBlockRequest) (*AddBlockResponse, error) { + return nil, status.Errorf(codes.Unimplemented, "method AddBlock not implemented") +} +func (UnimplementedMetastoreServiceServer) QueryMetadata(context.Context, *QueryMetadataRequest) (*QueryMetadataResponse, error) { + return nil, status.Errorf(codes.Unimplemented, "method QueryMetadata not implemented") +} +func (UnimplementedMetastoreServiceServer) ReadIndex(context.Context, *ReadIndexRequest) (*ReadIndexResponse, error) { + return nil, status.Errorf(codes.Unimplemented, "method ReadIndex not implemented") +} +func (UnimplementedMetastoreServiceServer) mustEmbedUnimplementedMetastoreServiceServer() {} + +// UnsafeMetastoreServiceServer may be embedded to opt out of forward compatibility for this service. +// Use of this interface is not recommended, as added methods to MetastoreServiceServer will +// result in compilation errors. +type UnsafeMetastoreServiceServer interface { + mustEmbedUnimplementedMetastoreServiceServer() +} + +func RegisterMetastoreServiceServer(s grpc.ServiceRegistrar, srv MetastoreServiceServer) { + s.RegisterService(&MetastoreService_ServiceDesc, srv) +} + +func _MetastoreService_AddBlock_Handler(srv interface{}, ctx context.Context, dec func(interface{}) error, interceptor grpc.UnaryServerInterceptor) (interface{}, error) { + in := new(AddBlockRequest) + if err := dec(in); err != nil { + return nil, err + } + if interceptor == nil { + return srv.(MetastoreServiceServer).AddBlock(ctx, in) + } + info := &grpc.UnaryServerInfo{ + Server: srv, + FullMethod: "/metastore.v1.MetastoreService/AddBlock", + } + handler := func(ctx context.Context, req interface{}) (interface{}, error) { + return srv.(MetastoreServiceServer).AddBlock(ctx, req.(*AddBlockRequest)) + } + return interceptor(ctx, in, info, handler) +} + +func _MetastoreService_QueryMetadata_Handler(srv interface{}, ctx context.Context, dec func(interface{}) error, interceptor grpc.UnaryServerInterceptor) (interface{}, error) { + in := new(QueryMetadataRequest) + if err := dec(in); err != nil { + return nil, err + } + if interceptor == nil { + return srv.(MetastoreServiceServer).QueryMetadata(ctx, in) + } + info := &grpc.UnaryServerInfo{ + Server: srv, + FullMethod: "/metastore.v1.MetastoreService/QueryMetadata", + } + handler := func(ctx context.Context, req interface{}) (interface{}, error) { + return srv.(MetastoreServiceServer).QueryMetadata(ctx, req.(*QueryMetadataRequest)) + } + return interceptor(ctx, in, info, handler) +} + +func _MetastoreService_ReadIndex_Handler(srv interface{}, ctx context.Context, dec func(interface{}) error, interceptor grpc.UnaryServerInterceptor) (interface{}, error) { + in := new(ReadIndexRequest) + if err := dec(in); err != nil { + return nil, err + } + if interceptor == nil { + return srv.(MetastoreServiceServer).ReadIndex(ctx, in) + } + info := &grpc.UnaryServerInfo{ + Server: srv, + FullMethod: "/metastore.v1.MetastoreService/ReadIndex", + } + handler := func(ctx context.Context, req interface{}) (interface{}, error) { + return srv.(MetastoreServiceServer).ReadIndex(ctx, req.(*ReadIndexRequest)) + } + return interceptor(ctx, in, info, handler) +} + +// MetastoreService_ServiceDesc is the grpc.ServiceDesc for MetastoreService service. +// It's only intended for direct use with grpc.RegisterService, +// and not to be introspected or modified (even as a copy) +var MetastoreService_ServiceDesc = grpc.ServiceDesc{ + ServiceName: "metastore.v1.MetastoreService", + HandlerType: (*MetastoreServiceServer)(nil), + Methods: []grpc.MethodDesc{ + { + MethodName: "AddBlock", + Handler: _MetastoreService_AddBlock_Handler, + }, + { + MethodName: "QueryMetadata", + Handler: _MetastoreService_QueryMetadata_Handler, + }, + { + MethodName: "ReadIndex", + Handler: _MetastoreService_ReadIndex_Handler, + }, + }, + Streams: []grpc.StreamDesc{}, + Metadata: "metastore/v1/metastore.proto", +} + +func (m *AddBlockRequest) MarshalVT() (dAtA []byte, err error) { + if m == nil { + return nil, nil + } + size := m.SizeVT() + dAtA = make([]byte, size) + n, err := m.MarshalToSizedBufferVT(dAtA[:size]) + if err != nil { + return nil, err + } + return dAtA[:n], nil +} + +func (m *AddBlockRequest) MarshalToVT(dAtA []byte) (int, error) { + size := m.SizeVT() + return m.MarshalToSizedBufferVT(dAtA[:size]) +} + +func (m *AddBlockRequest) MarshalToSizedBufferVT(dAtA []byte) (int, error) { + if m == nil { + return 0, nil + } + i := len(dAtA) + _ = i + var l int + _ = l + if m.unknownFields != nil { + i -= len(m.unknownFields) + copy(dAtA[i:], m.unknownFields) + } + if m.Block != nil { + size, err := m.Block.MarshalToSizedBufferVT(dAtA[:i]) + if err != nil { + return 0, err + } + i -= size + i = protohelpers.EncodeVarint(dAtA, i, uint64(size)) + i-- + dAtA[i] = 0xa + } + return len(dAtA) - i, nil +} + +func (m *AddBlockResponse) MarshalVT() (dAtA []byte, err error) { + if m == nil { + return nil, nil + } + size := m.SizeVT() + dAtA = make([]byte, size) + n, err := m.MarshalToSizedBufferVT(dAtA[:size]) + if err != nil { + return nil, err + } + return dAtA[:n], nil +} + +func (m *AddBlockResponse) MarshalToVT(dAtA []byte) (int, error) { + size := m.SizeVT() + return m.MarshalToSizedBufferVT(dAtA[:size]) +} + +func (m *AddBlockResponse) MarshalToSizedBufferVT(dAtA []byte) (int, error) { + if m == nil { + return 0, nil + } + i := len(dAtA) + _ = i + var l int + _ = l + if m.unknownFields != nil { + i -= len(m.unknownFields) + copy(dAtA[i:], m.unknownFields) + } + return len(dAtA) - i, nil +} + +func (m *BlockMeta) MarshalVT() (dAtA []byte, err error) { + if m == nil { + return nil, nil + } + size := m.SizeVT() + dAtA = make([]byte, size) + n, err := m.MarshalToSizedBufferVT(dAtA[:size]) + if err != nil { + return nil, err + } + return dAtA[:n], nil +} + +func (m *BlockMeta) MarshalToVT(dAtA []byte) (int, error) { + size := m.SizeVT() + return m.MarshalToSizedBufferVT(dAtA[:size]) +} + +func (m *BlockMeta) MarshalToSizedBufferVT(dAtA []byte) (int, error) { + if m == nil { + return 0, nil + } + i := len(dAtA) + _ = i + var l int + _ = l + if m.unknownFields != nil { + i -= len(m.unknownFields) + copy(dAtA[i:], m.unknownFields) + } + if m.Size != 0 { + i = protohelpers.EncodeVarint(dAtA, i, uint64(m.Size)) + i-- + dAtA[i] = 0x48 + } + if len(m.Datasets) > 0 { + for iNdEx := len(m.Datasets) - 1; iNdEx >= 0; iNdEx-- { + size, err := m.Datasets[iNdEx].MarshalToSizedBufferVT(dAtA[:i]) + if err != nil { + return 0, err + } + i -= size + i = protohelpers.EncodeVarint(dAtA, i, uint64(size)) + i-- + dAtA[i] = 0x42 + } + } + if len(m.TenantId) > 0 { + i -= len(m.TenantId) + copy(dAtA[i:], m.TenantId) + i = protohelpers.EncodeVarint(dAtA, i, uint64(len(m.TenantId))) + i-- + dAtA[i] = 0x3a + } + if m.CompactionLevel != 0 { + i = protohelpers.EncodeVarint(dAtA, i, uint64(m.CompactionLevel)) + i-- + dAtA[i] = 0x30 + } + if m.Shard != 0 { + i = protohelpers.EncodeVarint(dAtA, i, uint64(m.Shard)) + i-- + dAtA[i] = 0x28 + } + if m.MaxTime != 0 { + i = protohelpers.EncodeVarint(dAtA, i, uint64(m.MaxTime)) + i-- + dAtA[i] = 0x20 + } + if m.MinTime != 0 { + i = protohelpers.EncodeVarint(dAtA, i, uint64(m.MinTime)) + i-- + dAtA[i] = 0x18 + } + if len(m.Id) > 0 { + i -= len(m.Id) + copy(dAtA[i:], m.Id) + i = protohelpers.EncodeVarint(dAtA, i, uint64(len(m.Id))) + i-- + dAtA[i] = 0x12 + } + if m.FormatVersion != 0 { + i = protohelpers.EncodeVarint(dAtA, i, uint64(m.FormatVersion)) + i-- + dAtA[i] = 0x8 + } + return len(dAtA) - i, nil +} + +func (m *Dataset) MarshalVT() (dAtA []byte, err error) { + if m == nil { + return nil, nil + } + size := m.SizeVT() + dAtA = make([]byte, size) + n, err := m.MarshalToSizedBufferVT(dAtA[:size]) + if err != nil { + return nil, err + } + return dAtA[:n], nil +} + +func (m *Dataset) MarshalToVT(dAtA []byte) (int, error) { + size := m.SizeVT() + return m.MarshalToSizedBufferVT(dAtA[:size]) +} + +func (m *Dataset) MarshalToSizedBufferVT(dAtA []byte) (int, error) { + if m == nil { + return 0, nil + } + i := len(dAtA) + _ = i + var l int + _ = l + if m.unknownFields != nil { + i -= len(m.unknownFields) + copy(dAtA[i:], m.unknownFields) + } + if len(m.Labels) > 0 { + for iNdEx := len(m.Labels) - 1; iNdEx >= 0; iNdEx-- { + if vtmsg, ok := interface{}(m.Labels[iNdEx]).(interface { + MarshalToSizedBufferVT([]byte) (int, error) + }); ok { + size, err := vtmsg.MarshalToSizedBufferVT(dAtA[:i]) + if err != nil { + return 0, err + } + i -= size + i = protohelpers.EncodeVarint(dAtA, i, uint64(size)) + } else { + encoded, err := proto.Marshal(m.Labels[iNdEx]) + if err != nil { + return 0, err + } + i -= len(encoded) + copy(dAtA[i:], encoded) + i = protohelpers.EncodeVarint(dAtA, i, uint64(len(encoded))) + } + i-- + dAtA[i] = 0x42 + } + } + if len(m.ProfileTypes) > 0 { + for iNdEx := len(m.ProfileTypes) - 1; iNdEx >= 0; iNdEx-- { + i -= len(m.ProfileTypes[iNdEx]) + copy(dAtA[i:], m.ProfileTypes[iNdEx]) + i = protohelpers.EncodeVarint(dAtA, i, uint64(len(m.ProfileTypes[iNdEx]))) + i-- + dAtA[i] = 0x3a + } + } + if m.Size != 0 { + i = protohelpers.EncodeVarint(dAtA, i, uint64(m.Size)) + i-- + dAtA[i] = 0x30 + } + if len(m.TableOfContents) > 0 { + var pksize2 int + for _, num := range m.TableOfContents { + pksize2 += protohelpers.SizeOfVarint(uint64(num)) + } + i -= pksize2 + j1 := i + for _, num := range m.TableOfContents { + for num >= 1<<7 { + dAtA[j1] = uint8(uint64(num)&0x7f | 0x80) + num >>= 7 + j1++ + } + dAtA[j1] = uint8(num) + j1++ + } + i = protohelpers.EncodeVarint(dAtA, i, uint64(pksize2)) + i-- + dAtA[i] = 0x2a + } + if m.MaxTime != 0 { + i = protohelpers.EncodeVarint(dAtA, i, uint64(m.MaxTime)) + i-- + dAtA[i] = 0x20 + } + if m.MinTime != 0 { + i = protohelpers.EncodeVarint(dAtA, i, uint64(m.MinTime)) + i-- + dAtA[i] = 0x18 + } + if len(m.Name) > 0 { + i -= len(m.Name) + copy(dAtA[i:], m.Name) + i = protohelpers.EncodeVarint(dAtA, i, uint64(len(m.Name))) + i-- + dAtA[i] = 0x12 + } + if len(m.TenantId) > 0 { + i -= len(m.TenantId) + copy(dAtA[i:], m.TenantId) + i = protohelpers.EncodeVarint(dAtA, i, uint64(len(m.TenantId))) + i-- + dAtA[i] = 0xa + } + return len(dAtA) - i, nil +} + +func (m *QueryMetadataRequest) MarshalVT() (dAtA []byte, err error) { + if m == nil { + return nil, nil + } + size := m.SizeVT() + dAtA = make([]byte, size) + n, err := m.MarshalToSizedBufferVT(dAtA[:size]) + if err != nil { + return nil, err + } + return dAtA[:n], nil +} + +func (m *QueryMetadataRequest) MarshalToVT(dAtA []byte) (int, error) { + size := m.SizeVT() + return m.MarshalToSizedBufferVT(dAtA[:size]) +} + +func (m *QueryMetadataRequest) MarshalToSizedBufferVT(dAtA []byte) (int, error) { + if m == nil { + return 0, nil + } + i := len(dAtA) + _ = i + var l int + _ = l + if m.unknownFields != nil { + i -= len(m.unknownFields) + copy(dAtA[i:], m.unknownFields) + } + if len(m.Query) > 0 { + i -= len(m.Query) + copy(dAtA[i:], m.Query) + i = protohelpers.EncodeVarint(dAtA, i, uint64(len(m.Query))) + i-- + dAtA[i] = 0x22 + } + if m.EndTime != 0 { + i = protohelpers.EncodeVarint(dAtA, i, uint64(m.EndTime)) + i-- + dAtA[i] = 0x18 + } + if m.StartTime != 0 { + i = protohelpers.EncodeVarint(dAtA, i, uint64(m.StartTime)) + i-- + dAtA[i] = 0x10 + } + if len(m.TenantId) > 0 { + for iNdEx := len(m.TenantId) - 1; iNdEx >= 0; iNdEx-- { + i -= len(m.TenantId[iNdEx]) + copy(dAtA[i:], m.TenantId[iNdEx]) + i = protohelpers.EncodeVarint(dAtA, i, uint64(len(m.TenantId[iNdEx]))) + i-- + dAtA[i] = 0xa + } + } + return len(dAtA) - i, nil +} + +func (m *QueryMetadataResponse) MarshalVT() (dAtA []byte, err error) { + if m == nil { + return nil, nil + } + size := m.SizeVT() + dAtA = make([]byte, size) + n, err := m.MarshalToSizedBufferVT(dAtA[:size]) + if err != nil { + return nil, err + } + return dAtA[:n], nil +} + +func (m *QueryMetadataResponse) MarshalToVT(dAtA []byte) (int, error) { + size := m.SizeVT() + return m.MarshalToSizedBufferVT(dAtA[:size]) +} + +func (m *QueryMetadataResponse) MarshalToSizedBufferVT(dAtA []byte) (int, error) { + if m == nil { + return 0, nil + } + i := len(dAtA) + _ = i + var l int + _ = l + if m.unknownFields != nil { + i -= len(m.unknownFields) + copy(dAtA[i:], m.unknownFields) + } + if len(m.Blocks) > 0 { + for iNdEx := len(m.Blocks) - 1; iNdEx >= 0; iNdEx-- { + size, err := m.Blocks[iNdEx].MarshalToSizedBufferVT(dAtA[:i]) + if err != nil { + return 0, err + } + i -= size + i = protohelpers.EncodeVarint(dAtA, i, uint64(size)) + i-- + dAtA[i] = 0xa + } + } + return len(dAtA) - i, nil +} + +func (m *ReadIndexRequest) MarshalVT() (dAtA []byte, err error) { + if m == nil { + return nil, nil + } + size := m.SizeVT() + dAtA = make([]byte, size) + n, err := m.MarshalToSizedBufferVT(dAtA[:size]) + if err != nil { + return nil, err + } + return dAtA[:n], nil +} + +func (m *ReadIndexRequest) MarshalToVT(dAtA []byte) (int, error) { + size := m.SizeVT() + return m.MarshalToSizedBufferVT(dAtA[:size]) +} + +func (m *ReadIndexRequest) MarshalToSizedBufferVT(dAtA []byte) (int, error) { + if m == nil { + return 0, nil + } + i := len(dAtA) + _ = i + var l int + _ = l + if m.unknownFields != nil { + i -= len(m.unknownFields) + copy(dAtA[i:], m.unknownFields) + } + if len(m.DebugRequestId) > 0 { + i -= len(m.DebugRequestId) + copy(dAtA[i:], m.DebugRequestId) + i = protohelpers.EncodeVarint(dAtA, i, uint64(len(m.DebugRequestId))) + i-- + dAtA[i] = 0xa + } + return len(dAtA) - i, nil +} + +func (m *ReadIndexResponse) MarshalVT() (dAtA []byte, err error) { + if m == nil { + return nil, nil + } + size := m.SizeVT() + dAtA = make([]byte, size) + n, err := m.MarshalToSizedBufferVT(dAtA[:size]) + if err != nil { + return nil, err + } + return dAtA[:n], nil +} + +func (m *ReadIndexResponse) MarshalToVT(dAtA []byte) (int, error) { + size := m.SizeVT() + return m.MarshalToSizedBufferVT(dAtA[:size]) +} + +func (m *ReadIndexResponse) MarshalToSizedBufferVT(dAtA []byte) (int, error) { + if m == nil { + return 0, nil + } + i := len(dAtA) + _ = i + var l int + _ = l + if m.unknownFields != nil { + i -= len(m.unknownFields) + copy(dAtA[i:], m.unknownFields) + } + if m.ReadIndex != 0 { + i = protohelpers.EncodeVarint(dAtA, i, uint64(m.ReadIndex)) + i-- + dAtA[i] = 0x8 + } + return len(dAtA) - i, nil +} + +func (m *AddBlockRequest) SizeVT() (n int) { + if m == nil { + return 0 + } + var l int + _ = l + if m.Block != nil { + l = m.Block.SizeVT() + n += 1 + l + protohelpers.SizeOfVarint(uint64(l)) + } + n += len(m.unknownFields) + return n +} + +func (m *AddBlockResponse) SizeVT() (n int) { + if m == nil { + return 0 + } + var l int + _ = l + n += len(m.unknownFields) + return n +} + +func (m *BlockMeta) SizeVT() (n int) { + if m == nil { + return 0 + } + var l int + _ = l + if m.FormatVersion != 0 { + n += 1 + protohelpers.SizeOfVarint(uint64(m.FormatVersion)) + } + l = len(m.Id) + if l > 0 { + n += 1 + l + protohelpers.SizeOfVarint(uint64(l)) + } + if m.MinTime != 0 { + n += 1 + protohelpers.SizeOfVarint(uint64(m.MinTime)) + } + if m.MaxTime != 0 { + n += 1 + protohelpers.SizeOfVarint(uint64(m.MaxTime)) + } + if m.Shard != 0 { + n += 1 + protohelpers.SizeOfVarint(uint64(m.Shard)) + } + if m.CompactionLevel != 0 { + n += 1 + protohelpers.SizeOfVarint(uint64(m.CompactionLevel)) + } + l = len(m.TenantId) + if l > 0 { + n += 1 + l + protohelpers.SizeOfVarint(uint64(l)) + } + if len(m.Datasets) > 0 { + for _, e := range m.Datasets { + l = e.SizeVT() + n += 1 + l + protohelpers.SizeOfVarint(uint64(l)) + } + } + if m.Size != 0 { + n += 1 + protohelpers.SizeOfVarint(uint64(m.Size)) + } + n += len(m.unknownFields) + return n +} + +func (m *Dataset) SizeVT() (n int) { + if m == nil { + return 0 + } + var l int + _ = l + l = len(m.TenantId) + if l > 0 { + n += 1 + l + protohelpers.SizeOfVarint(uint64(l)) + } + l = len(m.Name) + if l > 0 { + n += 1 + l + protohelpers.SizeOfVarint(uint64(l)) + } + if m.MinTime != 0 { + n += 1 + protohelpers.SizeOfVarint(uint64(m.MinTime)) + } + if m.MaxTime != 0 { + n += 1 + protohelpers.SizeOfVarint(uint64(m.MaxTime)) + } + if len(m.TableOfContents) > 0 { + l = 0 + for _, e := range m.TableOfContents { + l += protohelpers.SizeOfVarint(uint64(e)) + } + n += 1 + protohelpers.SizeOfVarint(uint64(l)) + l + } + if m.Size != 0 { + n += 1 + protohelpers.SizeOfVarint(uint64(m.Size)) + } + if len(m.ProfileTypes) > 0 { + for _, s := range m.ProfileTypes { + l = len(s) + n += 1 + l + protohelpers.SizeOfVarint(uint64(l)) + } + } + if len(m.Labels) > 0 { + for _, e := range m.Labels { + if size, ok := interface{}(e).(interface { + SizeVT() int + }); ok { + l = size.SizeVT() + } else { + l = proto.Size(e) + } + n += 1 + l + protohelpers.SizeOfVarint(uint64(l)) + } + } + n += len(m.unknownFields) + return n +} + +func (m *QueryMetadataRequest) SizeVT() (n int) { + if m == nil { + return 0 + } + var l int + _ = l + if len(m.TenantId) > 0 { + for _, s := range m.TenantId { + l = len(s) + n += 1 + l + protohelpers.SizeOfVarint(uint64(l)) + } + } + if m.StartTime != 0 { + n += 1 + protohelpers.SizeOfVarint(uint64(m.StartTime)) + } + if m.EndTime != 0 { + n += 1 + protohelpers.SizeOfVarint(uint64(m.EndTime)) + } + l = len(m.Query) + if l > 0 { + n += 1 + l + protohelpers.SizeOfVarint(uint64(l)) + } + n += len(m.unknownFields) + return n +} + +func (m *QueryMetadataResponse) SizeVT() (n int) { + if m == nil { + return 0 + } + var l int + _ = l + if len(m.Blocks) > 0 { + for _, e := range m.Blocks { + l = e.SizeVT() + n += 1 + l + protohelpers.SizeOfVarint(uint64(l)) + } + } + n += len(m.unknownFields) + return n +} + +func (m *ReadIndexRequest) SizeVT() (n int) { + if m == nil { + return 0 + } + var l int + _ = l + l = len(m.DebugRequestId) + if l > 0 { + n += 1 + l + protohelpers.SizeOfVarint(uint64(l)) + } + n += len(m.unknownFields) + return n +} + +func (m *ReadIndexResponse) SizeVT() (n int) { + if m == nil { + return 0 + } + var l int + _ = l + if m.ReadIndex != 0 { + n += 1 + protohelpers.SizeOfVarint(uint64(m.ReadIndex)) + } + n += len(m.unknownFields) + return n +} + +func (m *AddBlockRequest) UnmarshalVT(dAtA []byte) error { + l := len(dAtA) + iNdEx := 0 + for iNdEx < l { + preIndex := iNdEx + var wire uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + wire |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + fieldNum := int32(wire >> 3) + wireType := int(wire & 0x7) + if wireType == 4 { + return fmt.Errorf("proto: AddBlockRequest: wiretype end group for non-group") + } + if fieldNum <= 0 { + return fmt.Errorf("proto: AddBlockRequest: illegal tag %d (wire type %d)", fieldNum, wire) + } + switch fieldNum { + case 1: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field Block", wireType) + } + var msglen int + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + msglen |= int(b&0x7F) << shift + if b < 0x80 { + break + } + } + if msglen < 0 { + return protohelpers.ErrInvalidLength + } + postIndex := iNdEx + msglen + if postIndex < 0 { + return protohelpers.ErrInvalidLength + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + if m.Block == nil { + m.Block = &BlockMeta{} + } + if err := m.Block.UnmarshalVT(dAtA[iNdEx:postIndex]); err != nil { + return err + } + iNdEx = postIndex + default: + iNdEx = preIndex + skippy, err := protohelpers.Skip(dAtA[iNdEx:]) + if err != nil { + return err + } + if (skippy < 0) || (iNdEx+skippy) < 0 { + return protohelpers.ErrInvalidLength + } + if (iNdEx + skippy) > l { + return io.ErrUnexpectedEOF + } + m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...) + iNdEx += skippy + } + } + + if iNdEx > l { + return io.ErrUnexpectedEOF + } + return nil +} +func (m *AddBlockResponse) UnmarshalVT(dAtA []byte) error { + l := len(dAtA) + iNdEx := 0 + for iNdEx < l { + preIndex := iNdEx + var wire uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + wire |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + fieldNum := int32(wire >> 3) + wireType := int(wire & 0x7) + if wireType == 4 { + return fmt.Errorf("proto: AddBlockResponse: wiretype end group for non-group") + } + if fieldNum <= 0 { + return fmt.Errorf("proto: AddBlockResponse: illegal tag %d (wire type %d)", fieldNum, wire) + } + switch fieldNum { + default: + iNdEx = preIndex + skippy, err := protohelpers.Skip(dAtA[iNdEx:]) + if err != nil { + return err + } + if (skippy < 0) || (iNdEx+skippy) < 0 { + return protohelpers.ErrInvalidLength + } + if (iNdEx + skippy) > l { + return io.ErrUnexpectedEOF + } + m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...) + iNdEx += skippy + } + } + + if iNdEx > l { + return io.ErrUnexpectedEOF + } + return nil +} +func (m *BlockMeta) UnmarshalVT(dAtA []byte) error { + l := len(dAtA) + iNdEx := 0 + for iNdEx < l { + preIndex := iNdEx + var wire uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + wire |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + fieldNum := int32(wire >> 3) + wireType := int(wire & 0x7) + if wireType == 4 { + return fmt.Errorf("proto: BlockMeta: wiretype end group for non-group") + } + if fieldNum <= 0 { + return fmt.Errorf("proto: BlockMeta: illegal tag %d (wire type %d)", fieldNum, wire) + } + switch fieldNum { + case 1: + if wireType != 0 { + return fmt.Errorf("proto: wrong wireType = %d for field FormatVersion", wireType) + } + m.FormatVersion = 0 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + m.FormatVersion |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + case 2: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field Id", wireType) + } + var stringLen uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + stringLen |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + intStringLen := int(stringLen) + if intStringLen < 0 { + return protohelpers.ErrInvalidLength + } + postIndex := iNdEx + intStringLen + if postIndex < 0 { + return protohelpers.ErrInvalidLength + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + m.Id = string(dAtA[iNdEx:postIndex]) + iNdEx = postIndex + case 3: + if wireType != 0 { + return fmt.Errorf("proto: wrong wireType = %d for field MinTime", wireType) + } + m.MinTime = 0 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + m.MinTime |= int64(b&0x7F) << shift + if b < 0x80 { + break + } + } + case 4: + if wireType != 0 { + return fmt.Errorf("proto: wrong wireType = %d for field MaxTime", wireType) + } + m.MaxTime = 0 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + m.MaxTime |= int64(b&0x7F) << shift + if b < 0x80 { + break + } + } + case 5: + if wireType != 0 { + return fmt.Errorf("proto: wrong wireType = %d for field Shard", wireType) + } + m.Shard = 0 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + m.Shard |= uint32(b&0x7F) << shift + if b < 0x80 { + break + } + } + case 6: + if wireType != 0 { + return fmt.Errorf("proto: wrong wireType = %d for field CompactionLevel", wireType) + } + m.CompactionLevel = 0 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + m.CompactionLevel |= uint32(b&0x7F) << shift + if b < 0x80 { + break + } + } + case 7: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field TenantId", wireType) + } + var stringLen uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + stringLen |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + intStringLen := int(stringLen) + if intStringLen < 0 { + return protohelpers.ErrInvalidLength + } + postIndex := iNdEx + intStringLen + if postIndex < 0 { + return protohelpers.ErrInvalidLength + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + m.TenantId = string(dAtA[iNdEx:postIndex]) + iNdEx = postIndex + case 8: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field Datasets", wireType) + } + var msglen int + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + msglen |= int(b&0x7F) << shift + if b < 0x80 { + break + } + } + if msglen < 0 { + return protohelpers.ErrInvalidLength + } + postIndex := iNdEx + msglen + if postIndex < 0 { + return protohelpers.ErrInvalidLength + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + m.Datasets = append(m.Datasets, &Dataset{}) + if err := m.Datasets[len(m.Datasets)-1].UnmarshalVT(dAtA[iNdEx:postIndex]); err != nil { + return err + } + iNdEx = postIndex + case 9: + if wireType != 0 { + return fmt.Errorf("proto: wrong wireType = %d for field Size", wireType) + } + m.Size = 0 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + m.Size |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + default: + iNdEx = preIndex + skippy, err := protohelpers.Skip(dAtA[iNdEx:]) + if err != nil { + return err + } + if (skippy < 0) || (iNdEx+skippy) < 0 { + return protohelpers.ErrInvalidLength + } + if (iNdEx + skippy) > l { + return io.ErrUnexpectedEOF + } + m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...) + iNdEx += skippy + } + } + + if iNdEx > l { + return io.ErrUnexpectedEOF + } + return nil +} +func (m *Dataset) UnmarshalVT(dAtA []byte) error { + l := len(dAtA) + iNdEx := 0 + for iNdEx < l { + preIndex := iNdEx + var wire uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + wire |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + fieldNum := int32(wire >> 3) + wireType := int(wire & 0x7) + if wireType == 4 { + return fmt.Errorf("proto: Dataset: wiretype end group for non-group") + } + if fieldNum <= 0 { + return fmt.Errorf("proto: Dataset: illegal tag %d (wire type %d)", fieldNum, wire) + } + switch fieldNum { + case 1: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field TenantId", wireType) + } + var stringLen uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + stringLen |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + intStringLen := int(stringLen) + if intStringLen < 0 { + return protohelpers.ErrInvalidLength + } + postIndex := iNdEx + intStringLen + if postIndex < 0 { + return protohelpers.ErrInvalidLength + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + m.TenantId = string(dAtA[iNdEx:postIndex]) + iNdEx = postIndex + case 2: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field Name", wireType) + } + var stringLen uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + stringLen |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + intStringLen := int(stringLen) + if intStringLen < 0 { + return protohelpers.ErrInvalidLength + } + postIndex := iNdEx + intStringLen + if postIndex < 0 { + return protohelpers.ErrInvalidLength + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + m.Name = string(dAtA[iNdEx:postIndex]) + iNdEx = postIndex + case 3: + if wireType != 0 { + return fmt.Errorf("proto: wrong wireType = %d for field MinTime", wireType) + } + m.MinTime = 0 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + m.MinTime |= int64(b&0x7F) << shift + if b < 0x80 { + break + } + } + case 4: + if wireType != 0 { + return fmt.Errorf("proto: wrong wireType = %d for field MaxTime", wireType) + } + m.MaxTime = 0 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + m.MaxTime |= int64(b&0x7F) << shift + if b < 0x80 { + break + } + } + case 5: + if wireType == 0 { + var v uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + v |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + m.TableOfContents = append(m.TableOfContents, v) + } else if wireType == 2 { + var packedLen int + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + packedLen |= int(b&0x7F) << shift + if b < 0x80 { + break + } + } + if packedLen < 0 { + return protohelpers.ErrInvalidLength + } + postIndex := iNdEx + packedLen + if postIndex < 0 { + return protohelpers.ErrInvalidLength + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + var elementCount int + var count int + for _, integer := range dAtA[iNdEx:postIndex] { + if integer < 128 { + count++ + } + } + elementCount = count + if elementCount != 0 && len(m.TableOfContents) == 0 { + m.TableOfContents = make([]uint64, 0, elementCount) + } + for iNdEx < postIndex { + var v uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + v |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + m.TableOfContents = append(m.TableOfContents, v) + } + } else { + return fmt.Errorf("proto: wrong wireType = %d for field TableOfContents", wireType) + } + case 6: + if wireType != 0 { + return fmt.Errorf("proto: wrong wireType = %d for field Size", wireType) + } + m.Size = 0 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + m.Size |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + case 7: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field ProfileTypes", wireType) + } + var stringLen uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + stringLen |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + intStringLen := int(stringLen) + if intStringLen < 0 { + return protohelpers.ErrInvalidLength + } + postIndex := iNdEx + intStringLen + if postIndex < 0 { + return protohelpers.ErrInvalidLength + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + m.ProfileTypes = append(m.ProfileTypes, string(dAtA[iNdEx:postIndex])) + iNdEx = postIndex + case 8: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field Labels", wireType) + } + var msglen int + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + msglen |= int(b&0x7F) << shift + if b < 0x80 { + break + } + } + if msglen < 0 { + return protohelpers.ErrInvalidLength + } + postIndex := iNdEx + msglen + if postIndex < 0 { + return protohelpers.ErrInvalidLength + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + m.Labels = append(m.Labels, &v1.Labels{}) + if unmarshal, ok := interface{}(m.Labels[len(m.Labels)-1]).(interface { + UnmarshalVT([]byte) error + }); ok { + if err := unmarshal.UnmarshalVT(dAtA[iNdEx:postIndex]); err != nil { + return err + } + } else { + if err := proto.Unmarshal(dAtA[iNdEx:postIndex], m.Labels[len(m.Labels)-1]); err != nil { + return err + } + } + iNdEx = postIndex + default: + iNdEx = preIndex + skippy, err := protohelpers.Skip(dAtA[iNdEx:]) + if err != nil { + return err + } + if (skippy < 0) || (iNdEx+skippy) < 0 { + return protohelpers.ErrInvalidLength + } + if (iNdEx + skippy) > l { + return io.ErrUnexpectedEOF + } + m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...) + iNdEx += skippy + } + } + + if iNdEx > l { + return io.ErrUnexpectedEOF + } + return nil +} +func (m *QueryMetadataRequest) UnmarshalVT(dAtA []byte) error { + l := len(dAtA) + iNdEx := 0 + for iNdEx < l { + preIndex := iNdEx + var wire uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + wire |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + fieldNum := int32(wire >> 3) + wireType := int(wire & 0x7) + if wireType == 4 { + return fmt.Errorf("proto: QueryMetadataRequest: wiretype end group for non-group") + } + if fieldNum <= 0 { + return fmt.Errorf("proto: QueryMetadataRequest: illegal tag %d (wire type %d)", fieldNum, wire) + } + switch fieldNum { + case 1: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field TenantId", wireType) + } + var stringLen uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + stringLen |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + intStringLen := int(stringLen) + if intStringLen < 0 { + return protohelpers.ErrInvalidLength + } + postIndex := iNdEx + intStringLen + if postIndex < 0 { + return protohelpers.ErrInvalidLength + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + m.TenantId = append(m.TenantId, string(dAtA[iNdEx:postIndex])) + iNdEx = postIndex + case 2: + if wireType != 0 { + return fmt.Errorf("proto: wrong wireType = %d for field StartTime", wireType) + } + m.StartTime = 0 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + m.StartTime |= int64(b&0x7F) << shift + if b < 0x80 { + break + } + } + case 3: + if wireType != 0 { + return fmt.Errorf("proto: wrong wireType = %d for field EndTime", wireType) + } + m.EndTime = 0 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + m.EndTime |= int64(b&0x7F) << shift + if b < 0x80 { + break + } + } + case 4: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field Query", wireType) + } + var stringLen uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + stringLen |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + intStringLen := int(stringLen) + if intStringLen < 0 { + return protohelpers.ErrInvalidLength + } + postIndex := iNdEx + intStringLen + if postIndex < 0 { + return protohelpers.ErrInvalidLength + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + m.Query = string(dAtA[iNdEx:postIndex]) + iNdEx = postIndex + default: + iNdEx = preIndex + skippy, err := protohelpers.Skip(dAtA[iNdEx:]) + if err != nil { + return err + } + if (skippy < 0) || (iNdEx+skippy) < 0 { + return protohelpers.ErrInvalidLength + } + if (iNdEx + skippy) > l { + return io.ErrUnexpectedEOF + } + m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...) + iNdEx += skippy + } + } + + if iNdEx > l { + return io.ErrUnexpectedEOF + } + return nil +} +func (m *QueryMetadataResponse) UnmarshalVT(dAtA []byte) error { + l := len(dAtA) + iNdEx := 0 + for iNdEx < l { + preIndex := iNdEx + var wire uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + wire |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + fieldNum := int32(wire >> 3) + wireType := int(wire & 0x7) + if wireType == 4 { + return fmt.Errorf("proto: QueryMetadataResponse: wiretype end group for non-group") + } + if fieldNum <= 0 { + return fmt.Errorf("proto: QueryMetadataResponse: illegal tag %d (wire type %d)", fieldNum, wire) + } + switch fieldNum { + case 1: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field Blocks", wireType) + } + var msglen int + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + msglen |= int(b&0x7F) << shift + if b < 0x80 { + break + } + } + if msglen < 0 { + return protohelpers.ErrInvalidLength + } + postIndex := iNdEx + msglen + if postIndex < 0 { + return protohelpers.ErrInvalidLength + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + m.Blocks = append(m.Blocks, &BlockMeta{}) + if err := m.Blocks[len(m.Blocks)-1].UnmarshalVT(dAtA[iNdEx:postIndex]); err != nil { + return err + } + iNdEx = postIndex + default: + iNdEx = preIndex + skippy, err := protohelpers.Skip(dAtA[iNdEx:]) + if err != nil { + return err + } + if (skippy < 0) || (iNdEx+skippy) < 0 { + return protohelpers.ErrInvalidLength + } + if (iNdEx + skippy) > l { + return io.ErrUnexpectedEOF + } + m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...) + iNdEx += skippy + } + } + + if iNdEx > l { + return io.ErrUnexpectedEOF + } + return nil +} +func (m *ReadIndexRequest) UnmarshalVT(dAtA []byte) error { + l := len(dAtA) + iNdEx := 0 + for iNdEx < l { + preIndex := iNdEx + var wire uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + wire |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + fieldNum := int32(wire >> 3) + wireType := int(wire & 0x7) + if wireType == 4 { + return fmt.Errorf("proto: ReadIndexRequest: wiretype end group for non-group") + } + if fieldNum <= 0 { + return fmt.Errorf("proto: ReadIndexRequest: illegal tag %d (wire type %d)", fieldNum, wire) + } + switch fieldNum { + case 1: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field DebugRequestId", wireType) + } + var stringLen uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + stringLen |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + intStringLen := int(stringLen) + if intStringLen < 0 { + return protohelpers.ErrInvalidLength + } + postIndex := iNdEx + intStringLen + if postIndex < 0 { + return protohelpers.ErrInvalidLength + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + m.DebugRequestId = string(dAtA[iNdEx:postIndex]) + iNdEx = postIndex + default: + iNdEx = preIndex + skippy, err := protohelpers.Skip(dAtA[iNdEx:]) + if err != nil { + return err + } + if (skippy < 0) || (iNdEx+skippy) < 0 { + return protohelpers.ErrInvalidLength + } + if (iNdEx + skippy) > l { + return io.ErrUnexpectedEOF + } + m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...) + iNdEx += skippy + } + } + + if iNdEx > l { + return io.ErrUnexpectedEOF + } + return nil +} +func (m *ReadIndexResponse) UnmarshalVT(dAtA []byte) error { + l := len(dAtA) + iNdEx := 0 + for iNdEx < l { + preIndex := iNdEx + var wire uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + wire |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + fieldNum := int32(wire >> 3) + wireType := int(wire & 0x7) + if wireType == 4 { + return fmt.Errorf("proto: ReadIndexResponse: wiretype end group for non-group") + } + if fieldNum <= 0 { + return fmt.Errorf("proto: ReadIndexResponse: illegal tag %d (wire type %d)", fieldNum, wire) + } + switch fieldNum { + case 1: + if wireType != 0 { + return fmt.Errorf("proto: wrong wireType = %d for field ReadIndex", wireType) + } + m.ReadIndex = 0 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + m.ReadIndex |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + default: + iNdEx = preIndex + skippy, err := protohelpers.Skip(dAtA[iNdEx:]) + if err != nil { + return err + } + if (skippy < 0) || (iNdEx+skippy) < 0 { + return protohelpers.ErrInvalidLength + } + if (iNdEx + skippy) > l { + return io.ErrUnexpectedEOF + } + m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...) + iNdEx += skippy + } + } + + if iNdEx > l { + return io.ErrUnexpectedEOF + } + return nil +} diff --git a/api/gen/proto/go/metastore/v1/metastorev1connect/metastore.connect.go b/api/gen/proto/go/metastore/v1/metastorev1connect/metastore.connect.go new file mode 100644 index 0000000000..9fd81df27d --- /dev/null +++ b/api/gen/proto/go/metastore/v1/metastorev1connect/metastore.connect.go @@ -0,0 +1,173 @@ +// Code generated by protoc-gen-connect-go. DO NOT EDIT. +// +// Source: metastore/v1/metastore.proto + +package metastorev1connect + +import ( + connect "connectrpc.com/connect" + context "context" + errors "errors" + v1 "github.com/grafana/pyroscope/api/gen/proto/go/metastore/v1" + http "net/http" + strings "strings" +) + +// This is a compile-time assertion to ensure that this generated file and the connect package are +// compatible. If you get a compiler error that this constant is not defined, this code was +// generated with a version of connect newer than the one compiled into your binary. You can fix the +// problem by either regenerating this code with an older version of connect or updating the connect +// version compiled into your binary. +const _ = connect.IsAtLeastVersion1_13_0 + +const ( + // MetastoreServiceName is the fully-qualified name of the MetastoreService service. + MetastoreServiceName = "metastore.v1.MetastoreService" +) + +// These constants are the fully-qualified names of the RPCs defined in this package. They're +// exposed at runtime as Spec.Procedure and as the final two segments of the HTTP route. +// +// Note that these are different from the fully-qualified method names used by +// google.golang.org/protobuf/reflect/protoreflect. To convert from these constants to +// reflection-formatted method names, remove the leading slash and convert the remaining slash to a +// period. +const ( + // MetastoreServiceAddBlockProcedure is the fully-qualified name of the MetastoreService's AddBlock + // RPC. + MetastoreServiceAddBlockProcedure = "/metastore.v1.MetastoreService/AddBlock" + // MetastoreServiceQueryMetadataProcedure is the fully-qualified name of the MetastoreService's + // QueryMetadata RPC. + MetastoreServiceQueryMetadataProcedure = "/metastore.v1.MetastoreService/QueryMetadata" + // MetastoreServiceReadIndexProcedure is the fully-qualified name of the MetastoreService's + // ReadIndex RPC. + MetastoreServiceReadIndexProcedure = "/metastore.v1.MetastoreService/ReadIndex" +) + +// These variables are the protoreflect.Descriptor objects for the RPCs defined in this package. +var ( + metastoreServiceServiceDescriptor = v1.File_metastore_v1_metastore_proto.Services().ByName("MetastoreService") + metastoreServiceAddBlockMethodDescriptor = metastoreServiceServiceDescriptor.Methods().ByName("AddBlock") + metastoreServiceQueryMetadataMethodDescriptor = metastoreServiceServiceDescriptor.Methods().ByName("QueryMetadata") + metastoreServiceReadIndexMethodDescriptor = metastoreServiceServiceDescriptor.Methods().ByName("ReadIndex") +) + +// MetastoreServiceClient is a client for the metastore.v1.MetastoreService service. +type MetastoreServiceClient interface { + AddBlock(context.Context, *connect.Request[v1.AddBlockRequest]) (*connect.Response[v1.AddBlockResponse], error) + QueryMetadata(context.Context, *connect.Request[v1.QueryMetadataRequest]) (*connect.Response[v1.QueryMetadataResponse], error) + ReadIndex(context.Context, *connect.Request[v1.ReadIndexRequest]) (*connect.Response[v1.ReadIndexResponse], error) +} + +// NewMetastoreServiceClient constructs a client for the metastore.v1.MetastoreService service. By +// default, it uses the Connect protocol with the binary Protobuf Codec, asks for gzipped responses, +// and sends uncompressed requests. To use the gRPC or gRPC-Web protocols, supply the +// connect.WithGRPC() or connect.WithGRPCWeb() options. +// +// The URL supplied here should be the base URL for the Connect or gRPC server (for example, +// http://api.acme.com or https://acme.com/grpc). +func NewMetastoreServiceClient(httpClient connect.HTTPClient, baseURL string, opts ...connect.ClientOption) MetastoreServiceClient { + baseURL = strings.TrimRight(baseURL, "/") + return &metastoreServiceClient{ + addBlock: connect.NewClient[v1.AddBlockRequest, v1.AddBlockResponse]( + httpClient, + baseURL+MetastoreServiceAddBlockProcedure, + connect.WithSchema(metastoreServiceAddBlockMethodDescriptor), + connect.WithClientOptions(opts...), + ), + queryMetadata: connect.NewClient[v1.QueryMetadataRequest, v1.QueryMetadataResponse]( + httpClient, + baseURL+MetastoreServiceQueryMetadataProcedure, + connect.WithSchema(metastoreServiceQueryMetadataMethodDescriptor), + connect.WithClientOptions(opts...), + ), + readIndex: connect.NewClient[v1.ReadIndexRequest, v1.ReadIndexResponse]( + httpClient, + baseURL+MetastoreServiceReadIndexProcedure, + connect.WithSchema(metastoreServiceReadIndexMethodDescriptor), + connect.WithClientOptions(opts...), + ), + } +} + +// metastoreServiceClient implements MetastoreServiceClient. +type metastoreServiceClient struct { + addBlock *connect.Client[v1.AddBlockRequest, v1.AddBlockResponse] + queryMetadata *connect.Client[v1.QueryMetadataRequest, v1.QueryMetadataResponse] + readIndex *connect.Client[v1.ReadIndexRequest, v1.ReadIndexResponse] +} + +// AddBlock calls metastore.v1.MetastoreService.AddBlock. +func (c *metastoreServiceClient) AddBlock(ctx context.Context, req *connect.Request[v1.AddBlockRequest]) (*connect.Response[v1.AddBlockResponse], error) { + return c.addBlock.CallUnary(ctx, req) +} + +// QueryMetadata calls metastore.v1.MetastoreService.QueryMetadata. +func (c *metastoreServiceClient) QueryMetadata(ctx context.Context, req *connect.Request[v1.QueryMetadataRequest]) (*connect.Response[v1.QueryMetadataResponse], error) { + return c.queryMetadata.CallUnary(ctx, req) +} + +// ReadIndex calls metastore.v1.MetastoreService.ReadIndex. +func (c *metastoreServiceClient) ReadIndex(ctx context.Context, req *connect.Request[v1.ReadIndexRequest]) (*connect.Response[v1.ReadIndexResponse], error) { + return c.readIndex.CallUnary(ctx, req) +} + +// MetastoreServiceHandler is an implementation of the metastore.v1.MetastoreService service. +type MetastoreServiceHandler interface { + AddBlock(context.Context, *connect.Request[v1.AddBlockRequest]) (*connect.Response[v1.AddBlockResponse], error) + QueryMetadata(context.Context, *connect.Request[v1.QueryMetadataRequest]) (*connect.Response[v1.QueryMetadataResponse], error) + ReadIndex(context.Context, *connect.Request[v1.ReadIndexRequest]) (*connect.Response[v1.ReadIndexResponse], error) +} + +// NewMetastoreServiceHandler builds an HTTP handler from the service implementation. It returns the +// path on which to mount the handler and the handler itself. +// +// By default, handlers support the Connect, gRPC, and gRPC-Web protocols with the binary Protobuf +// and JSON codecs. They also support gzip compression. +func NewMetastoreServiceHandler(svc MetastoreServiceHandler, opts ...connect.HandlerOption) (string, http.Handler) { + metastoreServiceAddBlockHandler := connect.NewUnaryHandler( + MetastoreServiceAddBlockProcedure, + svc.AddBlock, + connect.WithSchema(metastoreServiceAddBlockMethodDescriptor), + connect.WithHandlerOptions(opts...), + ) + metastoreServiceQueryMetadataHandler := connect.NewUnaryHandler( + MetastoreServiceQueryMetadataProcedure, + svc.QueryMetadata, + connect.WithSchema(metastoreServiceQueryMetadataMethodDescriptor), + connect.WithHandlerOptions(opts...), + ) + metastoreServiceReadIndexHandler := connect.NewUnaryHandler( + MetastoreServiceReadIndexProcedure, + svc.ReadIndex, + connect.WithSchema(metastoreServiceReadIndexMethodDescriptor), + connect.WithHandlerOptions(opts...), + ) + return "/metastore.v1.MetastoreService/", http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { + switch r.URL.Path { + case MetastoreServiceAddBlockProcedure: + metastoreServiceAddBlockHandler.ServeHTTP(w, r) + case MetastoreServiceQueryMetadataProcedure: + metastoreServiceQueryMetadataHandler.ServeHTTP(w, r) + case MetastoreServiceReadIndexProcedure: + metastoreServiceReadIndexHandler.ServeHTTP(w, r) + default: + http.NotFound(w, r) + } + }) +} + +// UnimplementedMetastoreServiceHandler returns CodeUnimplemented from all methods. +type UnimplementedMetastoreServiceHandler struct{} + +func (UnimplementedMetastoreServiceHandler) AddBlock(context.Context, *connect.Request[v1.AddBlockRequest]) (*connect.Response[v1.AddBlockResponse], error) { + return nil, connect.NewError(connect.CodeUnimplemented, errors.New("metastore.v1.MetastoreService.AddBlock is not implemented")) +} + +func (UnimplementedMetastoreServiceHandler) QueryMetadata(context.Context, *connect.Request[v1.QueryMetadataRequest]) (*connect.Response[v1.QueryMetadataResponse], error) { + return nil, connect.NewError(connect.CodeUnimplemented, errors.New("metastore.v1.MetastoreService.QueryMetadata is not implemented")) +} + +func (UnimplementedMetastoreServiceHandler) ReadIndex(context.Context, *connect.Request[v1.ReadIndexRequest]) (*connect.Response[v1.ReadIndexResponse], error) { + return nil, connect.NewError(connect.CodeUnimplemented, errors.New("metastore.v1.MetastoreService.ReadIndex is not implemented")) +} diff --git a/api/gen/proto/go/metastore/v1/metastorev1connect/metastore.connect.mux.go b/api/gen/proto/go/metastore/v1/metastorev1connect/metastore.connect.mux.go new file mode 100644 index 0000000000..bde9d0a43b --- /dev/null +++ b/api/gen/proto/go/metastore/v1/metastorev1connect/metastore.connect.mux.go @@ -0,0 +1,37 @@ +// Code generated by protoc-gen-connect-go-mux. DO NOT EDIT. +// +// Source: metastore/v1/metastore.proto + +package metastorev1connect + +import ( + connect "connectrpc.com/connect" + mux "github.com/gorilla/mux" +) + +// This is a compile-time assertion to ensure that this generated file and the connect package are +// compatible. If you get a compiler error that this constant is not defined, this code was +// generated with a version of connect newer than the one compiled into your binary. You can fix the +// problem by either regenerating this code with an older version of connect or updating the connect +// version compiled into your binary. +const _ = connect.IsAtLeastVersion0_1_0 + +// RegisterMetastoreServiceHandler register an HTTP handler to a mux.Router from the service +// implementation. +func RegisterMetastoreServiceHandler(mux *mux.Router, svc MetastoreServiceHandler, opts ...connect.HandlerOption) { + mux.Handle("/metastore.v1.MetastoreService/AddBlock", connect.NewUnaryHandler( + "/metastore.v1.MetastoreService/AddBlock", + svc.AddBlock, + opts..., + )) + mux.Handle("/metastore.v1.MetastoreService/QueryMetadata", connect.NewUnaryHandler( + "/metastore.v1.MetastoreService/QueryMetadata", + svc.QueryMetadata, + opts..., + )) + mux.Handle("/metastore.v1.MetastoreService/ReadIndex", connect.NewUnaryHandler( + "/metastore.v1.MetastoreService/ReadIndex", + svc.ReadIndex, + opts..., + )) +} diff --git a/api/gen/proto/go/querybackend/v1/querybackend.pb.go b/api/gen/proto/go/querybackend/v1/querybackend.pb.go new file mode 100644 index 0000000000..fdb37dbfc7 --- /dev/null +++ b/api/gen/proto/go/querybackend/v1/querybackend.pb.go @@ -0,0 +1,1627 @@ +// Code generated by protoc-gen-go. DO NOT EDIT. +// versions: +// protoc-gen-go v1.34.2 +// protoc (unknown) +// source: querybackend/v1/querybackend.proto + +package querybackendv1 + +import ( + _ "github.com/grafana/pyroscope/api/gen/proto/go/google/v1" + v1 "github.com/grafana/pyroscope/api/gen/proto/go/metastore/v1" + v11 "github.com/grafana/pyroscope/api/gen/proto/go/types/v1" + protoreflect "google.golang.org/protobuf/reflect/protoreflect" + protoimpl "google.golang.org/protobuf/runtime/protoimpl" + reflect "reflect" + sync "sync" +) + +const ( + // Verify that this generated code is sufficiently up-to-date. + _ = protoimpl.EnforceVersion(20 - protoimpl.MinVersion) + // Verify that runtime/protoimpl is sufficiently up-to-date. + _ = protoimpl.EnforceVersion(protoimpl.MaxVersion - 20) +) + +type QueryType int32 + +const ( + QueryType_QUERY_UNSPECIFIED QueryType = 0 + QueryType_QUERY_LABEL_NAMES QueryType = 1 + QueryType_QUERY_LABEL_VALUES QueryType = 2 + QueryType_QUERY_SERIES_LABELS QueryType = 3 + QueryType_QUERY_TIME_SERIES QueryType = 4 + QueryType_QUERY_TREE QueryType = 5 +) + +// Enum value maps for QueryType. +var ( + QueryType_name = map[int32]string{ + 0: "QUERY_UNSPECIFIED", + 1: "QUERY_LABEL_NAMES", + 2: "QUERY_LABEL_VALUES", + 3: "QUERY_SERIES_LABELS", + 4: "QUERY_TIME_SERIES", + 5: "QUERY_TREE", + } + QueryType_value = map[string]int32{ + "QUERY_UNSPECIFIED": 0, + "QUERY_LABEL_NAMES": 1, + "QUERY_LABEL_VALUES": 2, + "QUERY_SERIES_LABELS": 3, + "QUERY_TIME_SERIES": 4, + "QUERY_TREE": 5, + } +) + +func (x QueryType) Enum() *QueryType { + p := new(QueryType) + *p = x + return p +} + +func (x QueryType) String() string { + return protoimpl.X.EnumStringOf(x.Descriptor(), protoreflect.EnumNumber(x)) +} + +func (QueryType) Descriptor() protoreflect.EnumDescriptor { + return file_querybackend_v1_querybackend_proto_enumTypes[0].Descriptor() +} + +func (QueryType) Type() protoreflect.EnumType { + return &file_querybackend_v1_querybackend_proto_enumTypes[0] +} + +func (x QueryType) Number() protoreflect.EnumNumber { + return protoreflect.EnumNumber(x) +} + +// Deprecated: Use QueryType.Descriptor instead. +func (QueryType) EnumDescriptor() ([]byte, []int) { + return file_querybackend_v1_querybackend_proto_rawDescGZIP(), []int{0} +} + +type ReportType int32 + +const ( + ReportType_REPORT_UNSPECIFIED ReportType = 0 + ReportType_REPORT_LABEL_NAMES ReportType = 1 + ReportType_REPORT_LABEL_VALUES ReportType = 2 + ReportType_REPORT_SERIES_LABELS ReportType = 3 + ReportType_REPORT_TIME_SERIES ReportType = 4 + ReportType_REPORT_TREE ReportType = 5 +) + +// Enum value maps for ReportType. +var ( + ReportType_name = map[int32]string{ + 0: "REPORT_UNSPECIFIED", + 1: "REPORT_LABEL_NAMES", + 2: "REPORT_LABEL_VALUES", + 3: "REPORT_SERIES_LABELS", + 4: "REPORT_TIME_SERIES", + 5: "REPORT_TREE", + } + ReportType_value = map[string]int32{ + "REPORT_UNSPECIFIED": 0, + "REPORT_LABEL_NAMES": 1, + "REPORT_LABEL_VALUES": 2, + "REPORT_SERIES_LABELS": 3, + "REPORT_TIME_SERIES": 4, + "REPORT_TREE": 5, + } +) + +func (x ReportType) Enum() *ReportType { + p := new(ReportType) + *p = x + return p +} + +func (x ReportType) String() string { + return protoimpl.X.EnumStringOf(x.Descriptor(), protoreflect.EnumNumber(x)) +} + +func (ReportType) Descriptor() protoreflect.EnumDescriptor { + return file_querybackend_v1_querybackend_proto_enumTypes[1].Descriptor() +} + +func (ReportType) Type() protoreflect.EnumType { + return &file_querybackend_v1_querybackend_proto_enumTypes[1] +} + +func (x ReportType) Number() protoreflect.EnumNumber { + return protoreflect.EnumNumber(x) +} + +// Deprecated: Use ReportType.Descriptor instead. +func (ReportType) EnumDescriptor() ([]byte, []int) { + return file_querybackend_v1_querybackend_proto_rawDescGZIP(), []int{1} +} + +type InvokeOptions struct { + state protoimpl.MessageState + sizeCache protoimpl.SizeCache + unknownFields protoimpl.UnknownFields +} + +func (x *InvokeOptions) Reset() { + *x = InvokeOptions{} + if protoimpl.UnsafeEnabled { + mi := &file_querybackend_v1_querybackend_proto_msgTypes[0] + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + ms.StoreMessageInfo(mi) + } +} + +func (x *InvokeOptions) String() string { + return protoimpl.X.MessageStringOf(x) +} + +func (*InvokeOptions) ProtoMessage() {} + +func (x *InvokeOptions) ProtoReflect() protoreflect.Message { + mi := &file_querybackend_v1_querybackend_proto_msgTypes[0] + if protoimpl.UnsafeEnabled && x != nil { + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + if ms.LoadMessageInfo() == nil { + ms.StoreMessageInfo(mi) + } + return ms + } + return mi.MessageOf(x) +} + +// Deprecated: Use InvokeOptions.ProtoReflect.Descriptor instead. +func (*InvokeOptions) Descriptor() ([]byte, []int) { + return file_querybackend_v1_querybackend_proto_rawDescGZIP(), []int{0} +} + +type InvokeRequest struct { + state protoimpl.MessageState + sizeCache protoimpl.SizeCache + unknownFields protoimpl.UnknownFields + + Tenant []string `protobuf:"bytes,1,rep,name=tenant,proto3" json:"tenant,omitempty"` + StartTime int64 `protobuf:"varint,2,opt,name=start_time,json=startTime,proto3" json:"start_time,omitempty"` + EndTime int64 `protobuf:"varint,3,opt,name=end_time,json=endTime,proto3" json:"end_time,omitempty"` + LabelSelector string `protobuf:"bytes,4,opt,name=label_selector,json=labelSelector,proto3" json:"label_selector,omitempty"` + Query []*Query `protobuf:"bytes,5,rep,name=query,proto3" json:"query,omitempty"` + QueryPlan *QueryPlan `protobuf:"bytes,6,opt,name=query_plan,json=queryPlan,proto3" json:"query_plan,omitempty"` + Options *InvokeOptions `protobuf:"bytes,7,opt,name=options,proto3" json:"options,omitempty"` +} + +func (x *InvokeRequest) Reset() { + *x = InvokeRequest{} + if protoimpl.UnsafeEnabled { + mi := &file_querybackend_v1_querybackend_proto_msgTypes[1] + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + ms.StoreMessageInfo(mi) + } +} + +func (x *InvokeRequest) String() string { + return protoimpl.X.MessageStringOf(x) +} + +func (*InvokeRequest) ProtoMessage() {} + +func (x *InvokeRequest) ProtoReflect() protoreflect.Message { + mi := &file_querybackend_v1_querybackend_proto_msgTypes[1] + if protoimpl.UnsafeEnabled && x != nil { + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + if ms.LoadMessageInfo() == nil { + ms.StoreMessageInfo(mi) + } + return ms + } + return mi.MessageOf(x) +} + +// Deprecated: Use InvokeRequest.ProtoReflect.Descriptor instead. +func (*InvokeRequest) Descriptor() ([]byte, []int) { + return file_querybackend_v1_querybackend_proto_rawDescGZIP(), []int{1} +} + +func (x *InvokeRequest) GetTenant() []string { + if x != nil { + return x.Tenant + } + return nil +} + +func (x *InvokeRequest) GetStartTime() int64 { + if x != nil { + return x.StartTime + } + return 0 +} + +func (x *InvokeRequest) GetEndTime() int64 { + if x != nil { + return x.EndTime + } + return 0 +} + +func (x *InvokeRequest) GetLabelSelector() string { + if x != nil { + return x.LabelSelector + } + return "" +} + +func (x *InvokeRequest) GetQuery() []*Query { + if x != nil { + return x.Query + } + return nil +} + +func (x *InvokeRequest) GetQueryPlan() *QueryPlan { + if x != nil { + return x.QueryPlan + } + return nil +} + +func (x *InvokeRequest) GetOptions() *InvokeOptions { + if x != nil { + return x.Options + } + return nil +} + +// Query plan is represented by a DAG, where each node +// might be either "merge" or "read" (leaves). Each node +// references a range: merge nodes refer to other nodes, +// while read nodes refer to the blocks. +type QueryPlan struct { + state protoimpl.MessageState + sizeCache protoimpl.SizeCache + unknownFields protoimpl.UnknownFields + + // Each node is encoded with 3 values: + // - node type: 0 - read, 1 - merge; + // - range offset; + // - range length. + Graph []uint32 `protobuf:"varint,1,rep,packed,name=graph,proto3" json:"graph,omitempty"` + // The blocks matching the query. + Blocks []*v1.BlockMeta `protobuf:"bytes,2,rep,name=blocks,proto3" json:"blocks,omitempty"` +} + +func (x *QueryPlan) Reset() { + *x = QueryPlan{} + if protoimpl.UnsafeEnabled { + mi := &file_querybackend_v1_querybackend_proto_msgTypes[2] + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + ms.StoreMessageInfo(mi) + } +} + +func (x *QueryPlan) String() string { + return protoimpl.X.MessageStringOf(x) +} + +func (*QueryPlan) ProtoMessage() {} + +func (x *QueryPlan) ProtoReflect() protoreflect.Message { + mi := &file_querybackend_v1_querybackend_proto_msgTypes[2] + if protoimpl.UnsafeEnabled && x != nil { + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + if ms.LoadMessageInfo() == nil { + ms.StoreMessageInfo(mi) + } + return ms + } + return mi.MessageOf(x) +} + +// Deprecated: Use QueryPlan.ProtoReflect.Descriptor instead. +func (*QueryPlan) Descriptor() ([]byte, []int) { + return file_querybackend_v1_querybackend_proto_rawDescGZIP(), []int{2} +} + +func (x *QueryPlan) GetGraph() []uint32 { + if x != nil { + return x.Graph + } + return nil +} + +func (x *QueryPlan) GetBlocks() []*v1.BlockMeta { + if x != nil { + return x.Blocks + } + return nil +} + +type Query struct { + state protoimpl.MessageState + sizeCache protoimpl.SizeCache + unknownFields protoimpl.UnknownFields + + QueryType QueryType `protobuf:"varint,1,opt,name=query_type,json=queryType,proto3,enum=querybackend.v1.QueryType" json:"query_type,omitempty"` + // Exactly one of the following fields should be set, + // depending on the query type. + LabelNames *LabelNamesQuery `protobuf:"bytes,2,opt,name=label_names,json=labelNames,proto3" json:"label_names,omitempty"` + LabelValues *LabelValuesQuery `protobuf:"bytes,3,opt,name=label_values,json=labelValues,proto3" json:"label_values,omitempty"` + SeriesLabels *SeriesLabelsQuery `protobuf:"bytes,4,opt,name=series_labels,json=seriesLabels,proto3" json:"series_labels,omitempty"` + TimeSeries *TimeSeriesQuery `protobuf:"bytes,5,opt,name=time_series,json=timeSeries,proto3" json:"time_series,omitempty"` + Tree *TreeQuery `protobuf:"bytes,6,opt,name=tree,proto3" json:"tree,omitempty"` +} + +func (x *Query) Reset() { + *x = Query{} + if protoimpl.UnsafeEnabled { + mi := &file_querybackend_v1_querybackend_proto_msgTypes[3] + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + ms.StoreMessageInfo(mi) + } +} + +func (x *Query) String() string { + return protoimpl.X.MessageStringOf(x) +} + +func (*Query) ProtoMessage() {} + +func (x *Query) ProtoReflect() protoreflect.Message { + mi := &file_querybackend_v1_querybackend_proto_msgTypes[3] + if protoimpl.UnsafeEnabled && x != nil { + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + if ms.LoadMessageInfo() == nil { + ms.StoreMessageInfo(mi) + } + return ms + } + return mi.MessageOf(x) +} + +// Deprecated: Use Query.ProtoReflect.Descriptor instead. +func (*Query) Descriptor() ([]byte, []int) { + return file_querybackend_v1_querybackend_proto_rawDescGZIP(), []int{3} +} + +func (x *Query) GetQueryType() QueryType { + if x != nil { + return x.QueryType + } + return QueryType_QUERY_UNSPECIFIED +} + +func (x *Query) GetLabelNames() *LabelNamesQuery { + if x != nil { + return x.LabelNames + } + return nil +} + +func (x *Query) GetLabelValues() *LabelValuesQuery { + if x != nil { + return x.LabelValues + } + return nil +} + +func (x *Query) GetSeriesLabels() *SeriesLabelsQuery { + if x != nil { + return x.SeriesLabels + } + return nil +} + +func (x *Query) GetTimeSeries() *TimeSeriesQuery { + if x != nil { + return x.TimeSeries + } + return nil +} + +func (x *Query) GetTree() *TreeQuery { + if x != nil { + return x.Tree + } + return nil +} + +type InvokeResponse struct { + state protoimpl.MessageState + sizeCache protoimpl.SizeCache + unknownFields protoimpl.UnknownFields + + Reports []*Report `protobuf:"bytes,1,rep,name=reports,proto3" json:"reports,omitempty"` + Diagnostics *Diagnostics `protobuf:"bytes,2,opt,name=diagnostics,proto3" json:"diagnostics,omitempty"` +} + +func (x *InvokeResponse) Reset() { + *x = InvokeResponse{} + if protoimpl.UnsafeEnabled { + mi := &file_querybackend_v1_querybackend_proto_msgTypes[4] + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + ms.StoreMessageInfo(mi) + } +} + +func (x *InvokeResponse) String() string { + return protoimpl.X.MessageStringOf(x) +} + +func (*InvokeResponse) ProtoMessage() {} + +func (x *InvokeResponse) ProtoReflect() protoreflect.Message { + mi := &file_querybackend_v1_querybackend_proto_msgTypes[4] + if protoimpl.UnsafeEnabled && x != nil { + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + if ms.LoadMessageInfo() == nil { + ms.StoreMessageInfo(mi) + } + return ms + } + return mi.MessageOf(x) +} + +// Deprecated: Use InvokeResponse.ProtoReflect.Descriptor instead. +func (*InvokeResponse) Descriptor() ([]byte, []int) { + return file_querybackend_v1_querybackend_proto_rawDescGZIP(), []int{4} +} + +func (x *InvokeResponse) GetReports() []*Report { + if x != nil { + return x.Reports + } + return nil +} + +func (x *InvokeResponse) GetDiagnostics() *Diagnostics { + if x != nil { + return x.Diagnostics + } + return nil +} + +// Diagnostic messages, events, statistics, analytics, etc. +type Diagnostics struct { + state protoimpl.MessageState + sizeCache protoimpl.SizeCache + unknownFields protoimpl.UnknownFields +} + +func (x *Diagnostics) Reset() { + *x = Diagnostics{} + if protoimpl.UnsafeEnabled { + mi := &file_querybackend_v1_querybackend_proto_msgTypes[5] + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + ms.StoreMessageInfo(mi) + } +} + +func (x *Diagnostics) String() string { + return protoimpl.X.MessageStringOf(x) +} + +func (*Diagnostics) ProtoMessage() {} + +func (x *Diagnostics) ProtoReflect() protoreflect.Message { + mi := &file_querybackend_v1_querybackend_proto_msgTypes[5] + if protoimpl.UnsafeEnabled && x != nil { + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + if ms.LoadMessageInfo() == nil { + ms.StoreMessageInfo(mi) + } + return ms + } + return mi.MessageOf(x) +} + +// Deprecated: Use Diagnostics.ProtoReflect.Descriptor instead. +func (*Diagnostics) Descriptor() ([]byte, []int) { + return file_querybackend_v1_querybackend_proto_rawDescGZIP(), []int{5} +} + +type Report struct { + state protoimpl.MessageState + sizeCache protoimpl.SizeCache + unknownFields protoimpl.UnknownFields + + ReportType ReportType `protobuf:"varint,1,opt,name=report_type,json=reportType,proto3,enum=querybackend.v1.ReportType" json:"report_type,omitempty"` + // Exactly one of the following fields should be set, + // depending on the report type. + LabelNames *LabelNamesReport `protobuf:"bytes,2,opt,name=label_names,json=labelNames,proto3" json:"label_names,omitempty"` + LabelValues *LabelValuesReport `protobuf:"bytes,3,opt,name=label_values,json=labelValues,proto3" json:"label_values,omitempty"` + SeriesLabels *SeriesLabelsReport `protobuf:"bytes,4,opt,name=series_labels,json=seriesLabels,proto3" json:"series_labels,omitempty"` + TimeSeries *TimeSeriesReport `protobuf:"bytes,5,opt,name=time_series,json=timeSeries,proto3" json:"time_series,omitempty"` + Tree *TreeReport `protobuf:"bytes,6,opt,name=tree,proto3" json:"tree,omitempty"` +} + +func (x *Report) Reset() { + *x = Report{} + if protoimpl.UnsafeEnabled { + mi := &file_querybackend_v1_querybackend_proto_msgTypes[6] + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + ms.StoreMessageInfo(mi) + } +} + +func (x *Report) String() string { + return protoimpl.X.MessageStringOf(x) +} + +func (*Report) ProtoMessage() {} + +func (x *Report) ProtoReflect() protoreflect.Message { + mi := &file_querybackend_v1_querybackend_proto_msgTypes[6] + if protoimpl.UnsafeEnabled && x != nil { + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + if ms.LoadMessageInfo() == nil { + ms.StoreMessageInfo(mi) + } + return ms + } + return mi.MessageOf(x) +} + +// Deprecated: Use Report.ProtoReflect.Descriptor instead. +func (*Report) Descriptor() ([]byte, []int) { + return file_querybackend_v1_querybackend_proto_rawDescGZIP(), []int{6} +} + +func (x *Report) GetReportType() ReportType { + if x != nil { + return x.ReportType + } + return ReportType_REPORT_UNSPECIFIED +} + +func (x *Report) GetLabelNames() *LabelNamesReport { + if x != nil { + return x.LabelNames + } + return nil +} + +func (x *Report) GetLabelValues() *LabelValuesReport { + if x != nil { + return x.LabelValues + } + return nil +} + +func (x *Report) GetSeriesLabels() *SeriesLabelsReport { + if x != nil { + return x.SeriesLabels + } + return nil +} + +func (x *Report) GetTimeSeries() *TimeSeriesReport { + if x != nil { + return x.TimeSeries + } + return nil +} + +func (x *Report) GetTree() *TreeReport { + if x != nil { + return x.Tree + } + return nil +} + +type LabelNamesQuery struct { + state protoimpl.MessageState + sizeCache protoimpl.SizeCache + unknownFields protoimpl.UnknownFields +} + +func (x *LabelNamesQuery) Reset() { + *x = LabelNamesQuery{} + if protoimpl.UnsafeEnabled { + mi := &file_querybackend_v1_querybackend_proto_msgTypes[7] + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + ms.StoreMessageInfo(mi) + } +} + +func (x *LabelNamesQuery) String() string { + return protoimpl.X.MessageStringOf(x) +} + +func (*LabelNamesQuery) ProtoMessage() {} + +func (x *LabelNamesQuery) ProtoReflect() protoreflect.Message { + mi := &file_querybackend_v1_querybackend_proto_msgTypes[7] + if protoimpl.UnsafeEnabled && x != nil { + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + if ms.LoadMessageInfo() == nil { + ms.StoreMessageInfo(mi) + } + return ms + } + return mi.MessageOf(x) +} + +// Deprecated: Use LabelNamesQuery.ProtoReflect.Descriptor instead. +func (*LabelNamesQuery) Descriptor() ([]byte, []int) { + return file_querybackend_v1_querybackend_proto_rawDescGZIP(), []int{7} +} + +type LabelNamesReport struct { + state protoimpl.MessageState + sizeCache protoimpl.SizeCache + unknownFields protoimpl.UnknownFields + + Query *LabelNamesQuery `protobuf:"bytes,1,opt,name=query,proto3" json:"query,omitempty"` + LabelNames []string `protobuf:"bytes,2,rep,name=label_names,json=labelNames,proto3" json:"label_names,omitempty"` +} + +func (x *LabelNamesReport) Reset() { + *x = LabelNamesReport{} + if protoimpl.UnsafeEnabled { + mi := &file_querybackend_v1_querybackend_proto_msgTypes[8] + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + ms.StoreMessageInfo(mi) + } +} + +func (x *LabelNamesReport) String() string { + return protoimpl.X.MessageStringOf(x) +} + +func (*LabelNamesReport) ProtoMessage() {} + +func (x *LabelNamesReport) ProtoReflect() protoreflect.Message { + mi := &file_querybackend_v1_querybackend_proto_msgTypes[8] + if protoimpl.UnsafeEnabled && x != nil { + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + if ms.LoadMessageInfo() == nil { + ms.StoreMessageInfo(mi) + } + return ms + } + return mi.MessageOf(x) +} + +// Deprecated: Use LabelNamesReport.ProtoReflect.Descriptor instead. +func (*LabelNamesReport) Descriptor() ([]byte, []int) { + return file_querybackend_v1_querybackend_proto_rawDescGZIP(), []int{8} +} + +func (x *LabelNamesReport) GetQuery() *LabelNamesQuery { + if x != nil { + return x.Query + } + return nil +} + +func (x *LabelNamesReport) GetLabelNames() []string { + if x != nil { + return x.LabelNames + } + return nil +} + +type LabelValuesQuery struct { + state protoimpl.MessageState + sizeCache protoimpl.SizeCache + unknownFields protoimpl.UnknownFields + + LabelName string `protobuf:"bytes,1,opt,name=label_name,json=labelName,proto3" json:"label_name,omitempty"` +} + +func (x *LabelValuesQuery) Reset() { + *x = LabelValuesQuery{} + if protoimpl.UnsafeEnabled { + mi := &file_querybackend_v1_querybackend_proto_msgTypes[9] + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + ms.StoreMessageInfo(mi) + } +} + +func (x *LabelValuesQuery) String() string { + return protoimpl.X.MessageStringOf(x) +} + +func (*LabelValuesQuery) ProtoMessage() {} + +func (x *LabelValuesQuery) ProtoReflect() protoreflect.Message { + mi := &file_querybackend_v1_querybackend_proto_msgTypes[9] + if protoimpl.UnsafeEnabled && x != nil { + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + if ms.LoadMessageInfo() == nil { + ms.StoreMessageInfo(mi) + } + return ms + } + return mi.MessageOf(x) +} + +// Deprecated: Use LabelValuesQuery.ProtoReflect.Descriptor instead. +func (*LabelValuesQuery) Descriptor() ([]byte, []int) { + return file_querybackend_v1_querybackend_proto_rawDescGZIP(), []int{9} +} + +func (x *LabelValuesQuery) GetLabelName() string { + if x != nil { + return x.LabelName + } + return "" +} + +type LabelValuesReport struct { + state protoimpl.MessageState + sizeCache protoimpl.SizeCache + unknownFields protoimpl.UnknownFields + + Query *LabelValuesQuery `protobuf:"bytes,1,opt,name=query,proto3" json:"query,omitempty"` + LabelValues []string `protobuf:"bytes,2,rep,name=label_values,json=labelValues,proto3" json:"label_values,omitempty"` +} + +func (x *LabelValuesReport) Reset() { + *x = LabelValuesReport{} + if protoimpl.UnsafeEnabled { + mi := &file_querybackend_v1_querybackend_proto_msgTypes[10] + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + ms.StoreMessageInfo(mi) + } +} + +func (x *LabelValuesReport) String() string { + return protoimpl.X.MessageStringOf(x) +} + +func (*LabelValuesReport) ProtoMessage() {} + +func (x *LabelValuesReport) ProtoReflect() protoreflect.Message { + mi := &file_querybackend_v1_querybackend_proto_msgTypes[10] + if protoimpl.UnsafeEnabled && x != nil { + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + if ms.LoadMessageInfo() == nil { + ms.StoreMessageInfo(mi) + } + return ms + } + return mi.MessageOf(x) +} + +// Deprecated: Use LabelValuesReport.ProtoReflect.Descriptor instead. +func (*LabelValuesReport) Descriptor() ([]byte, []int) { + return file_querybackend_v1_querybackend_proto_rawDescGZIP(), []int{10} +} + +func (x *LabelValuesReport) GetQuery() *LabelValuesQuery { + if x != nil { + return x.Query + } + return nil +} + +func (x *LabelValuesReport) GetLabelValues() []string { + if x != nil { + return x.LabelValues + } + return nil +} + +type SeriesLabelsQuery struct { + state protoimpl.MessageState + sizeCache protoimpl.SizeCache + unknownFields protoimpl.UnknownFields + + LabelNames []string `protobuf:"bytes,1,rep,name=label_names,json=labelNames,proto3" json:"label_names,omitempty"` +} + +func (x *SeriesLabelsQuery) Reset() { + *x = SeriesLabelsQuery{} + if protoimpl.UnsafeEnabled { + mi := &file_querybackend_v1_querybackend_proto_msgTypes[11] + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + ms.StoreMessageInfo(mi) + } +} + +func (x *SeriesLabelsQuery) String() string { + return protoimpl.X.MessageStringOf(x) +} + +func (*SeriesLabelsQuery) ProtoMessage() {} + +func (x *SeriesLabelsQuery) ProtoReflect() protoreflect.Message { + mi := &file_querybackend_v1_querybackend_proto_msgTypes[11] + if protoimpl.UnsafeEnabled && x != nil { + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + if ms.LoadMessageInfo() == nil { + ms.StoreMessageInfo(mi) + } + return ms + } + return mi.MessageOf(x) +} + +// Deprecated: Use SeriesLabelsQuery.ProtoReflect.Descriptor instead. +func (*SeriesLabelsQuery) Descriptor() ([]byte, []int) { + return file_querybackend_v1_querybackend_proto_rawDescGZIP(), []int{11} +} + +func (x *SeriesLabelsQuery) GetLabelNames() []string { + if x != nil { + return x.LabelNames + } + return nil +} + +type SeriesLabelsReport struct { + state protoimpl.MessageState + sizeCache protoimpl.SizeCache + unknownFields protoimpl.UnknownFields + + Query *SeriesLabelsQuery `protobuf:"bytes,1,opt,name=query,proto3" json:"query,omitempty"` + SeriesLabels []*v11.Labels `protobuf:"bytes,2,rep,name=series_labels,json=seriesLabels,proto3" json:"series_labels,omitempty"` +} + +func (x *SeriesLabelsReport) Reset() { + *x = SeriesLabelsReport{} + if protoimpl.UnsafeEnabled { + mi := &file_querybackend_v1_querybackend_proto_msgTypes[12] + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + ms.StoreMessageInfo(mi) + } +} + +func (x *SeriesLabelsReport) String() string { + return protoimpl.X.MessageStringOf(x) +} + +func (*SeriesLabelsReport) ProtoMessage() {} + +func (x *SeriesLabelsReport) ProtoReflect() protoreflect.Message { + mi := &file_querybackend_v1_querybackend_proto_msgTypes[12] + if protoimpl.UnsafeEnabled && x != nil { + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + if ms.LoadMessageInfo() == nil { + ms.StoreMessageInfo(mi) + } + return ms + } + return mi.MessageOf(x) +} + +// Deprecated: Use SeriesLabelsReport.ProtoReflect.Descriptor instead. +func (*SeriesLabelsReport) Descriptor() ([]byte, []int) { + return file_querybackend_v1_querybackend_proto_rawDescGZIP(), []int{12} +} + +func (x *SeriesLabelsReport) GetQuery() *SeriesLabelsQuery { + if x != nil { + return x.Query + } + return nil +} + +func (x *SeriesLabelsReport) GetSeriesLabels() []*v11.Labels { + if x != nil { + return x.SeriesLabels + } + return nil +} + +type TimeSeriesQuery struct { + state protoimpl.MessageState + sizeCache protoimpl.SizeCache + unknownFields protoimpl.UnknownFields + + Step float64 `protobuf:"fixed64,1,opt,name=step,proto3" json:"step,omitempty"` + GroupBy []string `protobuf:"bytes,2,rep,name=group_by,json=groupBy,proto3" json:"group_by,omitempty"` + Aggregation *v11.TimeSeriesAggregationType `protobuf:"varint,3,opt,name=aggregation,proto3,enum=types.v1.TimeSeriesAggregationType,oneof" json:"aggregation,omitempty"` +} + +func (x *TimeSeriesQuery) Reset() { + *x = TimeSeriesQuery{} + if protoimpl.UnsafeEnabled { + mi := &file_querybackend_v1_querybackend_proto_msgTypes[13] + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + ms.StoreMessageInfo(mi) + } +} + +func (x *TimeSeriesQuery) String() string { + return protoimpl.X.MessageStringOf(x) +} + +func (*TimeSeriesQuery) ProtoMessage() {} + +func (x *TimeSeriesQuery) ProtoReflect() protoreflect.Message { + mi := &file_querybackend_v1_querybackend_proto_msgTypes[13] + if protoimpl.UnsafeEnabled && x != nil { + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + if ms.LoadMessageInfo() == nil { + ms.StoreMessageInfo(mi) + } + return ms + } + return mi.MessageOf(x) +} + +// Deprecated: Use TimeSeriesQuery.ProtoReflect.Descriptor instead. +func (*TimeSeriesQuery) Descriptor() ([]byte, []int) { + return file_querybackend_v1_querybackend_proto_rawDescGZIP(), []int{13} +} + +func (x *TimeSeriesQuery) GetStep() float64 { + if x != nil { + return x.Step + } + return 0 +} + +func (x *TimeSeriesQuery) GetGroupBy() []string { + if x != nil { + return x.GroupBy + } + return nil +} + +func (x *TimeSeriesQuery) GetAggregation() v11.TimeSeriesAggregationType { + if x != nil && x.Aggregation != nil { + return *x.Aggregation + } + return v11.TimeSeriesAggregationType(0) +} + +type TimeSeriesReport struct { + state protoimpl.MessageState + sizeCache protoimpl.SizeCache + unknownFields protoimpl.UnknownFields + + Query *TimeSeriesQuery `protobuf:"bytes,1,opt,name=query,proto3" json:"query,omitempty"` + TimeSeries []*v11.Series `protobuf:"bytes,2,rep,name=time_series,json=timeSeries,proto3" json:"time_series,omitempty"` +} + +func (x *TimeSeriesReport) Reset() { + *x = TimeSeriesReport{} + if protoimpl.UnsafeEnabled { + mi := &file_querybackend_v1_querybackend_proto_msgTypes[14] + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + ms.StoreMessageInfo(mi) + } +} + +func (x *TimeSeriesReport) String() string { + return protoimpl.X.MessageStringOf(x) +} + +func (*TimeSeriesReport) ProtoMessage() {} + +func (x *TimeSeriesReport) ProtoReflect() protoreflect.Message { + mi := &file_querybackend_v1_querybackend_proto_msgTypes[14] + if protoimpl.UnsafeEnabled && x != nil { + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + if ms.LoadMessageInfo() == nil { + ms.StoreMessageInfo(mi) + } + return ms + } + return mi.MessageOf(x) +} + +// Deprecated: Use TimeSeriesReport.ProtoReflect.Descriptor instead. +func (*TimeSeriesReport) Descriptor() ([]byte, []int) { + return file_querybackend_v1_querybackend_proto_rawDescGZIP(), []int{14} +} + +func (x *TimeSeriesReport) GetQuery() *TimeSeriesQuery { + if x != nil { + return x.Query + } + return nil +} + +func (x *TimeSeriesReport) GetTimeSeries() []*v11.Series { + if x != nil { + return x.TimeSeries + } + return nil +} + +type TreeQuery struct { + state protoimpl.MessageState + sizeCache protoimpl.SizeCache + unknownFields protoimpl.UnknownFields + + MaxNodes int64 `protobuf:"varint,1,opt,name=max_nodes,json=maxNodes,proto3" json:"max_nodes,omitempty"` +} + +func (x *TreeQuery) Reset() { + *x = TreeQuery{} + if protoimpl.UnsafeEnabled { + mi := &file_querybackend_v1_querybackend_proto_msgTypes[15] + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + ms.StoreMessageInfo(mi) + } +} + +func (x *TreeQuery) String() string { + return protoimpl.X.MessageStringOf(x) +} + +func (*TreeQuery) ProtoMessage() {} + +func (x *TreeQuery) ProtoReflect() protoreflect.Message { + mi := &file_querybackend_v1_querybackend_proto_msgTypes[15] + if protoimpl.UnsafeEnabled && x != nil { + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + if ms.LoadMessageInfo() == nil { + ms.StoreMessageInfo(mi) + } + return ms + } + return mi.MessageOf(x) +} + +// Deprecated: Use TreeQuery.ProtoReflect.Descriptor instead. +func (*TreeQuery) Descriptor() ([]byte, []int) { + return file_querybackend_v1_querybackend_proto_rawDescGZIP(), []int{15} +} + +func (x *TreeQuery) GetMaxNodes() int64 { + if x != nil { + return x.MaxNodes + } + return 0 +} + +type TreeReport struct { + state protoimpl.MessageState + sizeCache protoimpl.SizeCache + unknownFields protoimpl.UnknownFields + + Query *TreeQuery `protobuf:"bytes,1,opt,name=query,proto3" json:"query,omitempty"` + Tree []byte `protobuf:"bytes,2,opt,name=tree,proto3" json:"tree,omitempty"` +} + +func (x *TreeReport) Reset() { + *x = TreeReport{} + if protoimpl.UnsafeEnabled { + mi := &file_querybackend_v1_querybackend_proto_msgTypes[16] + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + ms.StoreMessageInfo(mi) + } +} + +func (x *TreeReport) String() string { + return protoimpl.X.MessageStringOf(x) +} + +func (*TreeReport) ProtoMessage() {} + +func (x *TreeReport) ProtoReflect() protoreflect.Message { + mi := &file_querybackend_v1_querybackend_proto_msgTypes[16] + if protoimpl.UnsafeEnabled && x != nil { + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + if ms.LoadMessageInfo() == nil { + ms.StoreMessageInfo(mi) + } + return ms + } + return mi.MessageOf(x) +} + +// Deprecated: Use TreeReport.ProtoReflect.Descriptor instead. +func (*TreeReport) Descriptor() ([]byte, []int) { + return file_querybackend_v1_querybackend_proto_rawDescGZIP(), []int{16} +} + +func (x *TreeReport) GetQuery() *TreeQuery { + if x != nil { + return x.Query + } + return nil +} + +func (x *TreeReport) GetTree() []byte { + if x != nil { + return x.Tree + } + return nil +} + +var File_querybackend_v1_querybackend_proto protoreflect.FileDescriptor + +var file_querybackend_v1_querybackend_proto_rawDesc = []byte{ + 0x0a, 0x22, 0x71, 0x75, 0x65, 0x72, 0x79, 0x62, 0x61, 0x63, 0x6b, 0x65, 0x6e, 0x64, 0x2f, 0x76, + 0x31, 0x2f, 0x71, 0x75, 0x65, 0x72, 0x79, 0x62, 0x61, 0x63, 0x6b, 0x65, 0x6e, 0x64, 0x2e, 0x70, + 0x72, 0x6f, 0x74, 0x6f, 0x12, 0x0f, 0x71, 0x75, 0x65, 0x72, 0x79, 0x62, 0x61, 0x63, 0x6b, 0x65, + 0x6e, 0x64, 0x2e, 0x76, 0x31, 0x1a, 0x17, 0x67, 0x6f, 0x6f, 0x67, 0x6c, 0x65, 0x2f, 0x76, 0x31, + 0x2f, 0x70, 0x72, 0x6f, 0x66, 0x69, 0x6c, 0x65, 0x2e, 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x1a, 0x1c, + 0x6d, 0x65, 0x74, 0x61, 0x73, 0x74, 0x6f, 0x72, 0x65, 0x2f, 0x76, 0x31, 0x2f, 0x6d, 0x65, 0x74, + 0x61, 0x73, 0x74, 0x6f, 0x72, 0x65, 0x2e, 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x1a, 0x14, 0x74, 0x79, + 0x70, 0x65, 0x73, 0x2f, 0x76, 0x31, 0x2f, 0x74, 0x79, 0x70, 0x65, 0x73, 0x2e, 0x70, 0x72, 0x6f, + 0x74, 0x6f, 0x22, 0x0f, 0x0a, 0x0d, 0x49, 0x6e, 0x76, 0x6f, 0x6b, 0x65, 0x4f, 0x70, 0x74, 0x69, + 0x6f, 0x6e, 0x73, 0x22, 0xab, 0x02, 0x0a, 0x0d, 0x49, 0x6e, 0x76, 0x6f, 0x6b, 0x65, 0x52, 0x65, + 0x71, 0x75, 0x65, 0x73, 0x74, 0x12, 0x16, 0x0a, 0x06, 0x74, 0x65, 0x6e, 0x61, 0x6e, 0x74, 0x18, + 0x01, 0x20, 0x03, 0x28, 0x09, 0x52, 0x06, 0x74, 0x65, 0x6e, 0x61, 0x6e, 0x74, 0x12, 0x1d, 0x0a, + 0x0a, 0x73, 0x74, 0x61, 0x72, 0x74, 0x5f, 0x74, 0x69, 0x6d, 0x65, 0x18, 0x02, 0x20, 0x01, 0x28, + 0x03, 0x52, 0x09, 0x73, 0x74, 0x61, 0x72, 0x74, 0x54, 0x69, 0x6d, 0x65, 0x12, 0x19, 0x0a, 0x08, + 0x65, 0x6e, 0x64, 0x5f, 0x74, 0x69, 0x6d, 0x65, 0x18, 0x03, 0x20, 0x01, 0x28, 0x03, 0x52, 0x07, + 0x65, 0x6e, 0x64, 0x54, 0x69, 0x6d, 0x65, 0x12, 0x25, 0x0a, 0x0e, 0x6c, 0x61, 0x62, 0x65, 0x6c, + 0x5f, 0x73, 0x65, 0x6c, 0x65, 0x63, 0x74, 0x6f, 0x72, 0x18, 0x04, 0x20, 0x01, 0x28, 0x09, 0x52, + 0x0d, 0x6c, 0x61, 0x62, 0x65, 0x6c, 0x53, 0x65, 0x6c, 0x65, 0x63, 0x74, 0x6f, 0x72, 0x12, 0x2c, + 0x0a, 0x05, 0x71, 0x75, 0x65, 0x72, 0x79, 0x18, 0x05, 0x20, 0x03, 0x28, 0x0b, 0x32, 0x16, 0x2e, + 0x71, 0x75, 0x65, 0x72, 0x79, 0x62, 0x61, 0x63, 0x6b, 0x65, 0x6e, 0x64, 0x2e, 0x76, 0x31, 0x2e, + 0x51, 0x75, 0x65, 0x72, 0x79, 0x52, 0x05, 0x71, 0x75, 0x65, 0x72, 0x79, 0x12, 0x39, 0x0a, 0x0a, + 0x71, 0x75, 0x65, 0x72, 0x79, 0x5f, 0x70, 0x6c, 0x61, 0x6e, 0x18, 0x06, 0x20, 0x01, 0x28, 0x0b, + 0x32, 0x1a, 0x2e, 0x71, 0x75, 0x65, 0x72, 0x79, 0x62, 0x61, 0x63, 0x6b, 0x65, 0x6e, 0x64, 0x2e, + 0x76, 0x31, 0x2e, 0x51, 0x75, 0x65, 0x72, 0x79, 0x50, 0x6c, 0x61, 0x6e, 0x52, 0x09, 0x71, 0x75, + 0x65, 0x72, 0x79, 0x50, 0x6c, 0x61, 0x6e, 0x12, 0x38, 0x0a, 0x07, 0x6f, 0x70, 0x74, 0x69, 0x6f, + 0x6e, 0x73, 0x18, 0x07, 0x20, 0x01, 0x28, 0x0b, 0x32, 0x1e, 0x2e, 0x71, 0x75, 0x65, 0x72, 0x79, + 0x62, 0x61, 0x63, 0x6b, 0x65, 0x6e, 0x64, 0x2e, 0x76, 0x31, 0x2e, 0x49, 0x6e, 0x76, 0x6f, 0x6b, + 0x65, 0x4f, 0x70, 0x74, 0x69, 0x6f, 0x6e, 0x73, 0x52, 0x07, 0x6f, 0x70, 0x74, 0x69, 0x6f, 0x6e, + 0x73, 0x22, 0x52, 0x0a, 0x09, 0x51, 0x75, 0x65, 0x72, 0x79, 0x50, 0x6c, 0x61, 0x6e, 0x12, 0x14, + 0x0a, 0x05, 0x67, 0x72, 0x61, 0x70, 0x68, 0x18, 0x01, 0x20, 0x03, 0x28, 0x0d, 0x52, 0x05, 0x67, + 0x72, 0x61, 0x70, 0x68, 0x12, 0x2f, 0x0a, 0x06, 0x62, 0x6c, 0x6f, 0x63, 0x6b, 0x73, 0x18, 0x02, + 0x20, 0x03, 0x28, 0x0b, 0x32, 0x17, 0x2e, 0x6d, 0x65, 0x74, 0x61, 0x73, 0x74, 0x6f, 0x72, 0x65, + 0x2e, 0x76, 0x31, 0x2e, 0x42, 0x6c, 0x6f, 0x63, 0x6b, 0x4d, 0x65, 0x74, 0x61, 0x52, 0x06, 0x62, + 0x6c, 0x6f, 0x63, 0x6b, 0x73, 0x22, 0x87, 0x03, 0x0a, 0x05, 0x51, 0x75, 0x65, 0x72, 0x79, 0x12, + 0x39, 0x0a, 0x0a, 0x71, 0x75, 0x65, 0x72, 0x79, 0x5f, 0x74, 0x79, 0x70, 0x65, 0x18, 0x01, 0x20, + 0x01, 0x28, 0x0e, 0x32, 0x1a, 0x2e, 0x71, 0x75, 0x65, 0x72, 0x79, 0x62, 0x61, 0x63, 0x6b, 0x65, + 0x6e, 0x64, 0x2e, 0x76, 0x31, 0x2e, 0x51, 0x75, 0x65, 0x72, 0x79, 0x54, 0x79, 0x70, 0x65, 0x52, + 0x09, 0x71, 0x75, 0x65, 0x72, 0x79, 0x54, 0x79, 0x70, 0x65, 0x12, 0x41, 0x0a, 0x0b, 0x6c, 0x61, + 0x62, 0x65, 0x6c, 0x5f, 0x6e, 0x61, 0x6d, 0x65, 0x73, 0x18, 0x02, 0x20, 0x01, 0x28, 0x0b, 0x32, + 0x20, 0x2e, 0x71, 0x75, 0x65, 0x72, 0x79, 0x62, 0x61, 0x63, 0x6b, 0x65, 0x6e, 0x64, 0x2e, 0x76, + 0x31, 0x2e, 0x4c, 0x61, 0x62, 0x65, 0x6c, 0x4e, 0x61, 0x6d, 0x65, 0x73, 0x51, 0x75, 0x65, 0x72, + 0x79, 0x52, 0x0a, 0x6c, 0x61, 0x62, 0x65, 0x6c, 0x4e, 0x61, 0x6d, 0x65, 0x73, 0x12, 0x44, 0x0a, + 0x0c, 0x6c, 0x61, 0x62, 0x65, 0x6c, 0x5f, 0x76, 0x61, 0x6c, 0x75, 0x65, 0x73, 0x18, 0x03, 0x20, + 0x01, 0x28, 0x0b, 0x32, 0x21, 0x2e, 0x71, 0x75, 0x65, 0x72, 0x79, 0x62, 0x61, 0x63, 0x6b, 0x65, + 0x6e, 0x64, 0x2e, 0x76, 0x31, 0x2e, 0x4c, 0x61, 0x62, 0x65, 0x6c, 0x56, 0x61, 0x6c, 0x75, 0x65, + 0x73, 0x51, 0x75, 0x65, 0x72, 0x79, 0x52, 0x0b, 0x6c, 0x61, 0x62, 0x65, 0x6c, 0x56, 0x61, 0x6c, + 0x75, 0x65, 0x73, 0x12, 0x47, 0x0a, 0x0d, 0x73, 0x65, 0x72, 0x69, 0x65, 0x73, 0x5f, 0x6c, 0x61, + 0x62, 0x65, 0x6c, 0x73, 0x18, 0x04, 0x20, 0x01, 0x28, 0x0b, 0x32, 0x22, 0x2e, 0x71, 0x75, 0x65, + 0x72, 0x79, 0x62, 0x61, 0x63, 0x6b, 0x65, 0x6e, 0x64, 0x2e, 0x76, 0x31, 0x2e, 0x53, 0x65, 0x72, + 0x69, 0x65, 0x73, 0x4c, 0x61, 0x62, 0x65, 0x6c, 0x73, 0x51, 0x75, 0x65, 0x72, 0x79, 0x52, 0x0c, + 0x73, 0x65, 0x72, 0x69, 0x65, 0x73, 0x4c, 0x61, 0x62, 0x65, 0x6c, 0x73, 0x12, 0x41, 0x0a, 0x0b, + 0x74, 0x69, 0x6d, 0x65, 0x5f, 0x73, 0x65, 0x72, 0x69, 0x65, 0x73, 0x18, 0x05, 0x20, 0x01, 0x28, + 0x0b, 0x32, 0x20, 0x2e, 0x71, 0x75, 0x65, 0x72, 0x79, 0x62, 0x61, 0x63, 0x6b, 0x65, 0x6e, 0x64, + 0x2e, 0x76, 0x31, 0x2e, 0x54, 0x69, 0x6d, 0x65, 0x53, 0x65, 0x72, 0x69, 0x65, 0x73, 0x51, 0x75, + 0x65, 0x72, 0x79, 0x52, 0x0a, 0x74, 0x69, 0x6d, 0x65, 0x53, 0x65, 0x72, 0x69, 0x65, 0x73, 0x12, + 0x2e, 0x0a, 0x04, 0x74, 0x72, 0x65, 0x65, 0x18, 0x06, 0x20, 0x01, 0x28, 0x0b, 0x32, 0x1a, 0x2e, + 0x71, 0x75, 0x65, 0x72, 0x79, 0x62, 0x61, 0x63, 0x6b, 0x65, 0x6e, 0x64, 0x2e, 0x76, 0x31, 0x2e, + 0x54, 0x72, 0x65, 0x65, 0x51, 0x75, 0x65, 0x72, 0x79, 0x52, 0x04, 0x74, 0x72, 0x65, 0x65, 0x22, + 0x83, 0x01, 0x0a, 0x0e, 0x49, 0x6e, 0x76, 0x6f, 0x6b, 0x65, 0x52, 0x65, 0x73, 0x70, 0x6f, 0x6e, + 0x73, 0x65, 0x12, 0x31, 0x0a, 0x07, 0x72, 0x65, 0x70, 0x6f, 0x72, 0x74, 0x73, 0x18, 0x01, 0x20, + 0x03, 0x28, 0x0b, 0x32, 0x17, 0x2e, 0x71, 0x75, 0x65, 0x72, 0x79, 0x62, 0x61, 0x63, 0x6b, 0x65, + 0x6e, 0x64, 0x2e, 0x76, 0x31, 0x2e, 0x52, 0x65, 0x70, 0x6f, 0x72, 0x74, 0x52, 0x07, 0x72, 0x65, + 0x70, 0x6f, 0x72, 0x74, 0x73, 0x12, 0x3e, 0x0a, 0x0b, 0x64, 0x69, 0x61, 0x67, 0x6e, 0x6f, 0x73, + 0x74, 0x69, 0x63, 0x73, 0x18, 0x02, 0x20, 0x01, 0x28, 0x0b, 0x32, 0x1c, 0x2e, 0x71, 0x75, 0x65, + 0x72, 0x79, 0x62, 0x61, 0x63, 0x6b, 0x65, 0x6e, 0x64, 0x2e, 0x76, 0x31, 0x2e, 0x44, 0x69, 0x61, + 0x67, 0x6e, 0x6f, 0x73, 0x74, 0x69, 0x63, 0x73, 0x52, 0x0b, 0x64, 0x69, 0x61, 0x67, 0x6e, 0x6f, + 0x73, 0x74, 0x69, 0x63, 0x73, 0x22, 0x0d, 0x0a, 0x0b, 0x44, 0x69, 0x61, 0x67, 0x6e, 0x6f, 0x73, + 0x74, 0x69, 0x63, 0x73, 0x22, 0x90, 0x03, 0x0a, 0x06, 0x52, 0x65, 0x70, 0x6f, 0x72, 0x74, 0x12, + 0x3c, 0x0a, 0x0b, 0x72, 0x65, 0x70, 0x6f, 0x72, 0x74, 0x5f, 0x74, 0x79, 0x70, 0x65, 0x18, 0x01, + 0x20, 0x01, 0x28, 0x0e, 0x32, 0x1b, 0x2e, 0x71, 0x75, 0x65, 0x72, 0x79, 0x62, 0x61, 0x63, 0x6b, + 0x65, 0x6e, 0x64, 0x2e, 0x76, 0x31, 0x2e, 0x52, 0x65, 0x70, 0x6f, 0x72, 0x74, 0x54, 0x79, 0x70, + 0x65, 0x52, 0x0a, 0x72, 0x65, 0x70, 0x6f, 0x72, 0x74, 0x54, 0x79, 0x70, 0x65, 0x12, 0x42, 0x0a, + 0x0b, 0x6c, 0x61, 0x62, 0x65, 0x6c, 0x5f, 0x6e, 0x61, 0x6d, 0x65, 0x73, 0x18, 0x02, 0x20, 0x01, + 0x28, 0x0b, 0x32, 0x21, 0x2e, 0x71, 0x75, 0x65, 0x72, 0x79, 0x62, 0x61, 0x63, 0x6b, 0x65, 0x6e, + 0x64, 0x2e, 0x76, 0x31, 0x2e, 0x4c, 0x61, 0x62, 0x65, 0x6c, 0x4e, 0x61, 0x6d, 0x65, 0x73, 0x52, + 0x65, 0x70, 0x6f, 0x72, 0x74, 0x52, 0x0a, 0x6c, 0x61, 0x62, 0x65, 0x6c, 0x4e, 0x61, 0x6d, 0x65, + 0x73, 0x12, 0x45, 0x0a, 0x0c, 0x6c, 0x61, 0x62, 0x65, 0x6c, 0x5f, 0x76, 0x61, 0x6c, 0x75, 0x65, + 0x73, 0x18, 0x03, 0x20, 0x01, 0x28, 0x0b, 0x32, 0x22, 0x2e, 0x71, 0x75, 0x65, 0x72, 0x79, 0x62, + 0x61, 0x63, 0x6b, 0x65, 0x6e, 0x64, 0x2e, 0x76, 0x31, 0x2e, 0x4c, 0x61, 0x62, 0x65, 0x6c, 0x56, + 0x61, 0x6c, 0x75, 0x65, 0x73, 0x52, 0x65, 0x70, 0x6f, 0x72, 0x74, 0x52, 0x0b, 0x6c, 0x61, 0x62, + 0x65, 0x6c, 0x56, 0x61, 0x6c, 0x75, 0x65, 0x73, 0x12, 0x48, 0x0a, 0x0d, 0x73, 0x65, 0x72, 0x69, + 0x65, 0x73, 0x5f, 0x6c, 0x61, 0x62, 0x65, 0x6c, 0x73, 0x18, 0x04, 0x20, 0x01, 0x28, 0x0b, 0x32, + 0x23, 0x2e, 0x71, 0x75, 0x65, 0x72, 0x79, 0x62, 0x61, 0x63, 0x6b, 0x65, 0x6e, 0x64, 0x2e, 0x76, + 0x31, 0x2e, 0x53, 0x65, 0x72, 0x69, 0x65, 0x73, 0x4c, 0x61, 0x62, 0x65, 0x6c, 0x73, 0x52, 0x65, + 0x70, 0x6f, 0x72, 0x74, 0x52, 0x0c, 0x73, 0x65, 0x72, 0x69, 0x65, 0x73, 0x4c, 0x61, 0x62, 0x65, + 0x6c, 0x73, 0x12, 0x42, 0x0a, 0x0b, 0x74, 0x69, 0x6d, 0x65, 0x5f, 0x73, 0x65, 0x72, 0x69, 0x65, + 0x73, 0x18, 0x05, 0x20, 0x01, 0x28, 0x0b, 0x32, 0x21, 0x2e, 0x71, 0x75, 0x65, 0x72, 0x79, 0x62, + 0x61, 0x63, 0x6b, 0x65, 0x6e, 0x64, 0x2e, 0x76, 0x31, 0x2e, 0x54, 0x69, 0x6d, 0x65, 0x53, 0x65, + 0x72, 0x69, 0x65, 0x73, 0x52, 0x65, 0x70, 0x6f, 0x72, 0x74, 0x52, 0x0a, 0x74, 0x69, 0x6d, 0x65, + 0x53, 0x65, 0x72, 0x69, 0x65, 0x73, 0x12, 0x2f, 0x0a, 0x04, 0x74, 0x72, 0x65, 0x65, 0x18, 0x06, + 0x20, 0x01, 0x28, 0x0b, 0x32, 0x1b, 0x2e, 0x71, 0x75, 0x65, 0x72, 0x79, 0x62, 0x61, 0x63, 0x6b, + 0x65, 0x6e, 0x64, 0x2e, 0x76, 0x31, 0x2e, 0x54, 0x72, 0x65, 0x65, 0x52, 0x65, 0x70, 0x6f, 0x72, + 0x74, 0x52, 0x04, 0x74, 0x72, 0x65, 0x65, 0x22, 0x11, 0x0a, 0x0f, 0x4c, 0x61, 0x62, 0x65, 0x6c, + 0x4e, 0x61, 0x6d, 0x65, 0x73, 0x51, 0x75, 0x65, 0x72, 0x79, 0x22, 0x6b, 0x0a, 0x10, 0x4c, 0x61, + 0x62, 0x65, 0x6c, 0x4e, 0x61, 0x6d, 0x65, 0x73, 0x52, 0x65, 0x70, 0x6f, 0x72, 0x74, 0x12, 0x36, + 0x0a, 0x05, 0x71, 0x75, 0x65, 0x72, 0x79, 0x18, 0x01, 0x20, 0x01, 0x28, 0x0b, 0x32, 0x20, 0x2e, + 0x71, 0x75, 0x65, 0x72, 0x79, 0x62, 0x61, 0x63, 0x6b, 0x65, 0x6e, 0x64, 0x2e, 0x76, 0x31, 0x2e, + 0x4c, 0x61, 0x62, 0x65, 0x6c, 0x4e, 0x61, 0x6d, 0x65, 0x73, 0x51, 0x75, 0x65, 0x72, 0x79, 0x52, + 0x05, 0x71, 0x75, 0x65, 0x72, 0x79, 0x12, 0x1f, 0x0a, 0x0b, 0x6c, 0x61, 0x62, 0x65, 0x6c, 0x5f, + 0x6e, 0x61, 0x6d, 0x65, 0x73, 0x18, 0x02, 0x20, 0x03, 0x28, 0x09, 0x52, 0x0a, 0x6c, 0x61, 0x62, + 0x65, 0x6c, 0x4e, 0x61, 0x6d, 0x65, 0x73, 0x22, 0x31, 0x0a, 0x10, 0x4c, 0x61, 0x62, 0x65, 0x6c, + 0x56, 0x61, 0x6c, 0x75, 0x65, 0x73, 0x51, 0x75, 0x65, 0x72, 0x79, 0x12, 0x1d, 0x0a, 0x0a, 0x6c, + 0x61, 0x62, 0x65, 0x6c, 0x5f, 0x6e, 0x61, 0x6d, 0x65, 0x18, 0x01, 0x20, 0x01, 0x28, 0x09, 0x52, + 0x09, 0x6c, 0x61, 0x62, 0x65, 0x6c, 0x4e, 0x61, 0x6d, 0x65, 0x22, 0x6f, 0x0a, 0x11, 0x4c, 0x61, + 0x62, 0x65, 0x6c, 0x56, 0x61, 0x6c, 0x75, 0x65, 0x73, 0x52, 0x65, 0x70, 0x6f, 0x72, 0x74, 0x12, + 0x37, 0x0a, 0x05, 0x71, 0x75, 0x65, 0x72, 0x79, 0x18, 0x01, 0x20, 0x01, 0x28, 0x0b, 0x32, 0x21, + 0x2e, 0x71, 0x75, 0x65, 0x72, 0x79, 0x62, 0x61, 0x63, 0x6b, 0x65, 0x6e, 0x64, 0x2e, 0x76, 0x31, + 0x2e, 0x4c, 0x61, 0x62, 0x65, 0x6c, 0x56, 0x61, 0x6c, 0x75, 0x65, 0x73, 0x51, 0x75, 0x65, 0x72, + 0x79, 0x52, 0x05, 0x71, 0x75, 0x65, 0x72, 0x79, 0x12, 0x21, 0x0a, 0x0c, 0x6c, 0x61, 0x62, 0x65, + 0x6c, 0x5f, 0x76, 0x61, 0x6c, 0x75, 0x65, 0x73, 0x18, 0x02, 0x20, 0x03, 0x28, 0x09, 0x52, 0x0b, + 0x6c, 0x61, 0x62, 0x65, 0x6c, 0x56, 0x61, 0x6c, 0x75, 0x65, 0x73, 0x22, 0x34, 0x0a, 0x11, 0x53, + 0x65, 0x72, 0x69, 0x65, 0x73, 0x4c, 0x61, 0x62, 0x65, 0x6c, 0x73, 0x51, 0x75, 0x65, 0x72, 0x79, + 0x12, 0x1f, 0x0a, 0x0b, 0x6c, 0x61, 0x62, 0x65, 0x6c, 0x5f, 0x6e, 0x61, 0x6d, 0x65, 0x73, 0x18, + 0x01, 0x20, 0x03, 0x28, 0x09, 0x52, 0x0a, 0x6c, 0x61, 0x62, 0x65, 0x6c, 0x4e, 0x61, 0x6d, 0x65, + 0x73, 0x22, 0x85, 0x01, 0x0a, 0x12, 0x53, 0x65, 0x72, 0x69, 0x65, 0x73, 0x4c, 0x61, 0x62, 0x65, + 0x6c, 0x73, 0x52, 0x65, 0x70, 0x6f, 0x72, 0x74, 0x12, 0x38, 0x0a, 0x05, 0x71, 0x75, 0x65, 0x72, + 0x79, 0x18, 0x01, 0x20, 0x01, 0x28, 0x0b, 0x32, 0x22, 0x2e, 0x71, 0x75, 0x65, 0x72, 0x79, 0x62, + 0x61, 0x63, 0x6b, 0x65, 0x6e, 0x64, 0x2e, 0x76, 0x31, 0x2e, 0x53, 0x65, 0x72, 0x69, 0x65, 0x73, + 0x4c, 0x61, 0x62, 0x65, 0x6c, 0x73, 0x51, 0x75, 0x65, 0x72, 0x79, 0x52, 0x05, 0x71, 0x75, 0x65, + 0x72, 0x79, 0x12, 0x35, 0x0a, 0x0d, 0x73, 0x65, 0x72, 0x69, 0x65, 0x73, 0x5f, 0x6c, 0x61, 0x62, + 0x65, 0x6c, 0x73, 0x18, 0x02, 0x20, 0x03, 0x28, 0x0b, 0x32, 0x10, 0x2e, 0x74, 0x79, 0x70, 0x65, + 0x73, 0x2e, 0x76, 0x31, 0x2e, 0x4c, 0x61, 0x62, 0x65, 0x6c, 0x73, 0x52, 0x0c, 0x73, 0x65, 0x72, + 0x69, 0x65, 0x73, 0x4c, 0x61, 0x62, 0x65, 0x6c, 0x73, 0x22, 0x9c, 0x01, 0x0a, 0x0f, 0x54, 0x69, + 0x6d, 0x65, 0x53, 0x65, 0x72, 0x69, 0x65, 0x73, 0x51, 0x75, 0x65, 0x72, 0x79, 0x12, 0x12, 0x0a, + 0x04, 0x73, 0x74, 0x65, 0x70, 0x18, 0x01, 0x20, 0x01, 0x28, 0x01, 0x52, 0x04, 0x73, 0x74, 0x65, + 0x70, 0x12, 0x19, 0x0a, 0x08, 0x67, 0x72, 0x6f, 0x75, 0x70, 0x5f, 0x62, 0x79, 0x18, 0x02, 0x20, + 0x03, 0x28, 0x09, 0x52, 0x07, 0x67, 0x72, 0x6f, 0x75, 0x70, 0x42, 0x79, 0x12, 0x4a, 0x0a, 0x0b, + 0x61, 0x67, 0x67, 0x72, 0x65, 0x67, 0x61, 0x74, 0x69, 0x6f, 0x6e, 0x18, 0x03, 0x20, 0x01, 0x28, + 0x0e, 0x32, 0x23, 0x2e, 0x74, 0x79, 0x70, 0x65, 0x73, 0x2e, 0x76, 0x31, 0x2e, 0x54, 0x69, 0x6d, + 0x65, 0x53, 0x65, 0x72, 0x69, 0x65, 0x73, 0x41, 0x67, 0x67, 0x72, 0x65, 0x67, 0x61, 0x74, 0x69, + 0x6f, 0x6e, 0x54, 0x79, 0x70, 0x65, 0x48, 0x00, 0x52, 0x0b, 0x61, 0x67, 0x67, 0x72, 0x65, 0x67, + 0x61, 0x74, 0x69, 0x6f, 0x6e, 0x88, 0x01, 0x01, 0x42, 0x0e, 0x0a, 0x0c, 0x5f, 0x61, 0x67, 0x67, + 0x72, 0x65, 0x67, 0x61, 0x74, 0x69, 0x6f, 0x6e, 0x22, 0x7d, 0x0a, 0x10, 0x54, 0x69, 0x6d, 0x65, + 0x53, 0x65, 0x72, 0x69, 0x65, 0x73, 0x52, 0x65, 0x70, 0x6f, 0x72, 0x74, 0x12, 0x36, 0x0a, 0x05, + 0x71, 0x75, 0x65, 0x72, 0x79, 0x18, 0x01, 0x20, 0x01, 0x28, 0x0b, 0x32, 0x20, 0x2e, 0x71, 0x75, + 0x65, 0x72, 0x79, 0x62, 0x61, 0x63, 0x6b, 0x65, 0x6e, 0x64, 0x2e, 0x76, 0x31, 0x2e, 0x54, 0x69, + 0x6d, 0x65, 0x53, 0x65, 0x72, 0x69, 0x65, 0x73, 0x51, 0x75, 0x65, 0x72, 0x79, 0x52, 0x05, 0x71, + 0x75, 0x65, 0x72, 0x79, 0x12, 0x31, 0x0a, 0x0b, 0x74, 0x69, 0x6d, 0x65, 0x5f, 0x73, 0x65, 0x72, + 0x69, 0x65, 0x73, 0x18, 0x02, 0x20, 0x03, 0x28, 0x0b, 0x32, 0x10, 0x2e, 0x74, 0x79, 0x70, 0x65, + 0x73, 0x2e, 0x76, 0x31, 0x2e, 0x53, 0x65, 0x72, 0x69, 0x65, 0x73, 0x52, 0x0a, 0x74, 0x69, 0x6d, + 0x65, 0x53, 0x65, 0x72, 0x69, 0x65, 0x73, 0x22, 0x28, 0x0a, 0x09, 0x54, 0x72, 0x65, 0x65, 0x51, + 0x75, 0x65, 0x72, 0x79, 0x12, 0x1b, 0x0a, 0x09, 0x6d, 0x61, 0x78, 0x5f, 0x6e, 0x6f, 0x64, 0x65, + 0x73, 0x18, 0x01, 0x20, 0x01, 0x28, 0x03, 0x52, 0x08, 0x6d, 0x61, 0x78, 0x4e, 0x6f, 0x64, 0x65, + 0x73, 0x22, 0x52, 0x0a, 0x0a, 0x54, 0x72, 0x65, 0x65, 0x52, 0x65, 0x70, 0x6f, 0x72, 0x74, 0x12, + 0x30, 0x0a, 0x05, 0x71, 0x75, 0x65, 0x72, 0x79, 0x18, 0x01, 0x20, 0x01, 0x28, 0x0b, 0x32, 0x1a, + 0x2e, 0x71, 0x75, 0x65, 0x72, 0x79, 0x62, 0x61, 0x63, 0x6b, 0x65, 0x6e, 0x64, 0x2e, 0x76, 0x31, + 0x2e, 0x54, 0x72, 0x65, 0x65, 0x51, 0x75, 0x65, 0x72, 0x79, 0x52, 0x05, 0x71, 0x75, 0x65, 0x72, + 0x79, 0x12, 0x12, 0x0a, 0x04, 0x74, 0x72, 0x65, 0x65, 0x18, 0x02, 0x20, 0x01, 0x28, 0x0c, 0x52, + 0x04, 0x74, 0x72, 0x65, 0x65, 0x2a, 0x91, 0x01, 0x0a, 0x09, 0x51, 0x75, 0x65, 0x72, 0x79, 0x54, + 0x79, 0x70, 0x65, 0x12, 0x15, 0x0a, 0x11, 0x51, 0x55, 0x45, 0x52, 0x59, 0x5f, 0x55, 0x4e, 0x53, + 0x50, 0x45, 0x43, 0x49, 0x46, 0x49, 0x45, 0x44, 0x10, 0x00, 0x12, 0x15, 0x0a, 0x11, 0x51, 0x55, + 0x45, 0x52, 0x59, 0x5f, 0x4c, 0x41, 0x42, 0x45, 0x4c, 0x5f, 0x4e, 0x41, 0x4d, 0x45, 0x53, 0x10, + 0x01, 0x12, 0x16, 0x0a, 0x12, 0x51, 0x55, 0x45, 0x52, 0x59, 0x5f, 0x4c, 0x41, 0x42, 0x45, 0x4c, + 0x5f, 0x56, 0x41, 0x4c, 0x55, 0x45, 0x53, 0x10, 0x02, 0x12, 0x17, 0x0a, 0x13, 0x51, 0x55, 0x45, + 0x52, 0x59, 0x5f, 0x53, 0x45, 0x52, 0x49, 0x45, 0x53, 0x5f, 0x4c, 0x41, 0x42, 0x45, 0x4c, 0x53, + 0x10, 0x03, 0x12, 0x15, 0x0a, 0x11, 0x51, 0x55, 0x45, 0x52, 0x59, 0x5f, 0x54, 0x49, 0x4d, 0x45, + 0x5f, 0x53, 0x45, 0x52, 0x49, 0x45, 0x53, 0x10, 0x04, 0x12, 0x0e, 0x0a, 0x0a, 0x51, 0x55, 0x45, + 0x52, 0x59, 0x5f, 0x54, 0x52, 0x45, 0x45, 0x10, 0x05, 0x2a, 0x98, 0x01, 0x0a, 0x0a, 0x52, 0x65, + 0x70, 0x6f, 0x72, 0x74, 0x54, 0x79, 0x70, 0x65, 0x12, 0x16, 0x0a, 0x12, 0x52, 0x45, 0x50, 0x4f, + 0x52, 0x54, 0x5f, 0x55, 0x4e, 0x53, 0x50, 0x45, 0x43, 0x49, 0x46, 0x49, 0x45, 0x44, 0x10, 0x00, + 0x12, 0x16, 0x0a, 0x12, 0x52, 0x45, 0x50, 0x4f, 0x52, 0x54, 0x5f, 0x4c, 0x41, 0x42, 0x45, 0x4c, + 0x5f, 0x4e, 0x41, 0x4d, 0x45, 0x53, 0x10, 0x01, 0x12, 0x17, 0x0a, 0x13, 0x52, 0x45, 0x50, 0x4f, + 0x52, 0x54, 0x5f, 0x4c, 0x41, 0x42, 0x45, 0x4c, 0x5f, 0x56, 0x41, 0x4c, 0x55, 0x45, 0x53, 0x10, + 0x02, 0x12, 0x18, 0x0a, 0x14, 0x52, 0x45, 0x50, 0x4f, 0x52, 0x54, 0x5f, 0x53, 0x45, 0x52, 0x49, + 0x45, 0x53, 0x5f, 0x4c, 0x41, 0x42, 0x45, 0x4c, 0x53, 0x10, 0x03, 0x12, 0x16, 0x0a, 0x12, 0x52, + 0x45, 0x50, 0x4f, 0x52, 0x54, 0x5f, 0x54, 0x49, 0x4d, 0x45, 0x5f, 0x53, 0x45, 0x52, 0x49, 0x45, + 0x53, 0x10, 0x04, 0x12, 0x0f, 0x0a, 0x0b, 0x52, 0x45, 0x50, 0x4f, 0x52, 0x54, 0x5f, 0x54, 0x52, + 0x45, 0x45, 0x10, 0x05, 0x32, 0x62, 0x0a, 0x13, 0x51, 0x75, 0x65, 0x72, 0x79, 0x42, 0x61, 0x63, + 0x6b, 0x65, 0x6e, 0x64, 0x53, 0x65, 0x72, 0x76, 0x69, 0x63, 0x65, 0x12, 0x4b, 0x0a, 0x06, 0x49, + 0x6e, 0x76, 0x6f, 0x6b, 0x65, 0x12, 0x1e, 0x2e, 0x71, 0x75, 0x65, 0x72, 0x79, 0x62, 0x61, 0x63, + 0x6b, 0x65, 0x6e, 0x64, 0x2e, 0x76, 0x31, 0x2e, 0x49, 0x6e, 0x76, 0x6f, 0x6b, 0x65, 0x52, 0x65, + 0x71, 0x75, 0x65, 0x73, 0x74, 0x1a, 0x1f, 0x2e, 0x71, 0x75, 0x65, 0x72, 0x79, 0x62, 0x61, 0x63, + 0x6b, 0x65, 0x6e, 0x64, 0x2e, 0x76, 0x31, 0x2e, 0x49, 0x6e, 0x76, 0x6f, 0x6b, 0x65, 0x52, 0x65, + 0x73, 0x70, 0x6f, 0x6e, 0x73, 0x65, 0x22, 0x00, 0x42, 0xd3, 0x01, 0x0a, 0x13, 0x63, 0x6f, 0x6d, + 0x2e, 0x71, 0x75, 0x65, 0x72, 0x79, 0x62, 0x61, 0x63, 0x6b, 0x65, 0x6e, 0x64, 0x2e, 0x76, 0x31, + 0x42, 0x11, 0x51, 0x75, 0x65, 0x72, 0x79, 0x62, 0x61, 0x63, 0x6b, 0x65, 0x6e, 0x64, 0x50, 0x72, + 0x6f, 0x74, 0x6f, 0x50, 0x01, 0x5a, 0x4c, 0x67, 0x69, 0x74, 0x68, 0x75, 0x62, 0x2e, 0x63, 0x6f, + 0x6d, 0x2f, 0x67, 0x72, 0x61, 0x66, 0x61, 0x6e, 0x61, 0x2f, 0x70, 0x79, 0x72, 0x6f, 0x73, 0x63, + 0x6f, 0x70, 0x65, 0x2f, 0x61, 0x70, 0x69, 0x2f, 0x67, 0x65, 0x6e, 0x2f, 0x70, 0x72, 0x6f, 0x74, + 0x6f, 0x2f, 0x67, 0x6f, 0x2f, 0x71, 0x75, 0x65, 0x72, 0x79, 0x62, 0x61, 0x63, 0x6b, 0x65, 0x6e, + 0x64, 0x2f, 0x76, 0x31, 0x3b, 0x71, 0x75, 0x65, 0x72, 0x79, 0x62, 0x61, 0x63, 0x6b, 0x65, 0x6e, + 0x64, 0x76, 0x31, 0xa2, 0x02, 0x03, 0x51, 0x58, 0x58, 0xaa, 0x02, 0x0f, 0x51, 0x75, 0x65, 0x72, + 0x79, 0x62, 0x61, 0x63, 0x6b, 0x65, 0x6e, 0x64, 0x2e, 0x56, 0x31, 0xca, 0x02, 0x0f, 0x51, 0x75, + 0x65, 0x72, 0x79, 0x62, 0x61, 0x63, 0x6b, 0x65, 0x6e, 0x64, 0x5c, 0x56, 0x31, 0xe2, 0x02, 0x1b, + 0x51, 0x75, 0x65, 0x72, 0x79, 0x62, 0x61, 0x63, 0x6b, 0x65, 0x6e, 0x64, 0x5c, 0x56, 0x31, 0x5c, + 0x47, 0x50, 0x42, 0x4d, 0x65, 0x74, 0x61, 0x64, 0x61, 0x74, 0x61, 0xea, 0x02, 0x10, 0x51, 0x75, + 0x65, 0x72, 0x79, 0x62, 0x61, 0x63, 0x6b, 0x65, 0x6e, 0x64, 0x3a, 0x3a, 0x56, 0x31, 0x62, 0x06, + 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x33, +} + +var ( + file_querybackend_v1_querybackend_proto_rawDescOnce sync.Once + file_querybackend_v1_querybackend_proto_rawDescData = file_querybackend_v1_querybackend_proto_rawDesc +) + +func file_querybackend_v1_querybackend_proto_rawDescGZIP() []byte { + file_querybackend_v1_querybackend_proto_rawDescOnce.Do(func() { + file_querybackend_v1_querybackend_proto_rawDescData = protoimpl.X.CompressGZIP(file_querybackend_v1_querybackend_proto_rawDescData) + }) + return file_querybackend_v1_querybackend_proto_rawDescData +} + +var file_querybackend_v1_querybackend_proto_enumTypes = make([]protoimpl.EnumInfo, 2) +var file_querybackend_v1_querybackend_proto_msgTypes = make([]protoimpl.MessageInfo, 17) +var file_querybackend_v1_querybackend_proto_goTypes = []any{ + (QueryType)(0), // 0: querybackend.v1.QueryType + (ReportType)(0), // 1: querybackend.v1.ReportType + (*InvokeOptions)(nil), // 2: querybackend.v1.InvokeOptions + (*InvokeRequest)(nil), // 3: querybackend.v1.InvokeRequest + (*QueryPlan)(nil), // 4: querybackend.v1.QueryPlan + (*Query)(nil), // 5: querybackend.v1.Query + (*InvokeResponse)(nil), // 6: querybackend.v1.InvokeResponse + (*Diagnostics)(nil), // 7: querybackend.v1.Diagnostics + (*Report)(nil), // 8: querybackend.v1.Report + (*LabelNamesQuery)(nil), // 9: querybackend.v1.LabelNamesQuery + (*LabelNamesReport)(nil), // 10: querybackend.v1.LabelNamesReport + (*LabelValuesQuery)(nil), // 11: querybackend.v1.LabelValuesQuery + (*LabelValuesReport)(nil), // 12: querybackend.v1.LabelValuesReport + (*SeriesLabelsQuery)(nil), // 13: querybackend.v1.SeriesLabelsQuery + (*SeriesLabelsReport)(nil), // 14: querybackend.v1.SeriesLabelsReport + (*TimeSeriesQuery)(nil), // 15: querybackend.v1.TimeSeriesQuery + (*TimeSeriesReport)(nil), // 16: querybackend.v1.TimeSeriesReport + (*TreeQuery)(nil), // 17: querybackend.v1.TreeQuery + (*TreeReport)(nil), // 18: querybackend.v1.TreeReport + (*v1.BlockMeta)(nil), // 19: metastore.v1.BlockMeta + (*v11.Labels)(nil), // 20: types.v1.Labels + (v11.TimeSeriesAggregationType)(0), // 21: types.v1.TimeSeriesAggregationType + (*v11.Series)(nil), // 22: types.v1.Series +} +var file_querybackend_v1_querybackend_proto_depIdxs = []int32{ + 5, // 0: querybackend.v1.InvokeRequest.query:type_name -> querybackend.v1.Query + 4, // 1: querybackend.v1.InvokeRequest.query_plan:type_name -> querybackend.v1.QueryPlan + 2, // 2: querybackend.v1.InvokeRequest.options:type_name -> querybackend.v1.InvokeOptions + 19, // 3: querybackend.v1.QueryPlan.blocks:type_name -> metastore.v1.BlockMeta + 0, // 4: querybackend.v1.Query.query_type:type_name -> querybackend.v1.QueryType + 9, // 5: querybackend.v1.Query.label_names:type_name -> querybackend.v1.LabelNamesQuery + 11, // 6: querybackend.v1.Query.label_values:type_name -> querybackend.v1.LabelValuesQuery + 13, // 7: querybackend.v1.Query.series_labels:type_name -> querybackend.v1.SeriesLabelsQuery + 15, // 8: querybackend.v1.Query.time_series:type_name -> querybackend.v1.TimeSeriesQuery + 17, // 9: querybackend.v1.Query.tree:type_name -> querybackend.v1.TreeQuery + 8, // 10: querybackend.v1.InvokeResponse.reports:type_name -> querybackend.v1.Report + 7, // 11: querybackend.v1.InvokeResponse.diagnostics:type_name -> querybackend.v1.Diagnostics + 1, // 12: querybackend.v1.Report.report_type:type_name -> querybackend.v1.ReportType + 10, // 13: querybackend.v1.Report.label_names:type_name -> querybackend.v1.LabelNamesReport + 12, // 14: querybackend.v1.Report.label_values:type_name -> querybackend.v1.LabelValuesReport + 14, // 15: querybackend.v1.Report.series_labels:type_name -> querybackend.v1.SeriesLabelsReport + 16, // 16: querybackend.v1.Report.time_series:type_name -> querybackend.v1.TimeSeriesReport + 18, // 17: querybackend.v1.Report.tree:type_name -> querybackend.v1.TreeReport + 9, // 18: querybackend.v1.LabelNamesReport.query:type_name -> querybackend.v1.LabelNamesQuery + 11, // 19: querybackend.v1.LabelValuesReport.query:type_name -> querybackend.v1.LabelValuesQuery + 13, // 20: querybackend.v1.SeriesLabelsReport.query:type_name -> querybackend.v1.SeriesLabelsQuery + 20, // 21: querybackend.v1.SeriesLabelsReport.series_labels:type_name -> types.v1.Labels + 21, // 22: querybackend.v1.TimeSeriesQuery.aggregation:type_name -> types.v1.TimeSeriesAggregationType + 15, // 23: querybackend.v1.TimeSeriesReport.query:type_name -> querybackend.v1.TimeSeriesQuery + 22, // 24: querybackend.v1.TimeSeriesReport.time_series:type_name -> types.v1.Series + 17, // 25: querybackend.v1.TreeReport.query:type_name -> querybackend.v1.TreeQuery + 3, // 26: querybackend.v1.QueryBackendService.Invoke:input_type -> querybackend.v1.InvokeRequest + 6, // 27: querybackend.v1.QueryBackendService.Invoke:output_type -> querybackend.v1.InvokeResponse + 27, // [27:28] is the sub-list for method output_type + 26, // [26:27] is the sub-list for method input_type + 26, // [26:26] is the sub-list for extension type_name + 26, // [26:26] is the sub-list for extension extendee + 0, // [0:26] is the sub-list for field type_name +} + +func init() { file_querybackend_v1_querybackend_proto_init() } +func file_querybackend_v1_querybackend_proto_init() { + if File_querybackend_v1_querybackend_proto != nil { + return + } + if !protoimpl.UnsafeEnabled { + file_querybackend_v1_querybackend_proto_msgTypes[0].Exporter = func(v any, i int) any { + switch v := v.(*InvokeOptions); i { + case 0: + return &v.state + case 1: + return &v.sizeCache + case 2: + return &v.unknownFields + default: + return nil + } + } + file_querybackend_v1_querybackend_proto_msgTypes[1].Exporter = func(v any, i int) any { + switch v := v.(*InvokeRequest); i { + case 0: + return &v.state + case 1: + return &v.sizeCache + case 2: + return &v.unknownFields + default: + return nil + } + } + file_querybackend_v1_querybackend_proto_msgTypes[2].Exporter = func(v any, i int) any { + switch v := v.(*QueryPlan); i { + case 0: + return &v.state + case 1: + return &v.sizeCache + case 2: + return &v.unknownFields + default: + return nil + } + } + file_querybackend_v1_querybackend_proto_msgTypes[3].Exporter = func(v any, i int) any { + switch v := v.(*Query); i { + case 0: + return &v.state + case 1: + return &v.sizeCache + case 2: + return &v.unknownFields + default: + return nil + } + } + file_querybackend_v1_querybackend_proto_msgTypes[4].Exporter = func(v any, i int) any { + switch v := v.(*InvokeResponse); i { + case 0: + return &v.state + case 1: + return &v.sizeCache + case 2: + return &v.unknownFields + default: + return nil + } + } + file_querybackend_v1_querybackend_proto_msgTypes[5].Exporter = func(v any, i int) any { + switch v := v.(*Diagnostics); i { + case 0: + return &v.state + case 1: + return &v.sizeCache + case 2: + return &v.unknownFields + default: + return nil + } + } + file_querybackend_v1_querybackend_proto_msgTypes[6].Exporter = func(v any, i int) any { + switch v := v.(*Report); i { + case 0: + return &v.state + case 1: + return &v.sizeCache + case 2: + return &v.unknownFields + default: + return nil + } + } + file_querybackend_v1_querybackend_proto_msgTypes[7].Exporter = func(v any, i int) any { + switch v := v.(*LabelNamesQuery); i { + case 0: + return &v.state + case 1: + return &v.sizeCache + case 2: + return &v.unknownFields + default: + return nil + } + } + file_querybackend_v1_querybackend_proto_msgTypes[8].Exporter = func(v any, i int) any { + switch v := v.(*LabelNamesReport); i { + case 0: + return &v.state + case 1: + return &v.sizeCache + case 2: + return &v.unknownFields + default: + return nil + } + } + file_querybackend_v1_querybackend_proto_msgTypes[9].Exporter = func(v any, i int) any { + switch v := v.(*LabelValuesQuery); i { + case 0: + return &v.state + case 1: + return &v.sizeCache + case 2: + return &v.unknownFields + default: + return nil + } + } + file_querybackend_v1_querybackend_proto_msgTypes[10].Exporter = func(v any, i int) any { + switch v := v.(*LabelValuesReport); i { + case 0: + return &v.state + case 1: + return &v.sizeCache + case 2: + return &v.unknownFields + default: + return nil + } + } + file_querybackend_v1_querybackend_proto_msgTypes[11].Exporter = func(v any, i int) any { + switch v := v.(*SeriesLabelsQuery); i { + case 0: + return &v.state + case 1: + return &v.sizeCache + case 2: + return &v.unknownFields + default: + return nil + } + } + file_querybackend_v1_querybackend_proto_msgTypes[12].Exporter = func(v any, i int) any { + switch v := v.(*SeriesLabelsReport); i { + case 0: + return &v.state + case 1: + return &v.sizeCache + case 2: + return &v.unknownFields + default: + return nil + } + } + file_querybackend_v1_querybackend_proto_msgTypes[13].Exporter = func(v any, i int) any { + switch v := v.(*TimeSeriesQuery); i { + case 0: + return &v.state + case 1: + return &v.sizeCache + case 2: + return &v.unknownFields + default: + return nil + } + } + file_querybackend_v1_querybackend_proto_msgTypes[14].Exporter = func(v any, i int) any { + switch v := v.(*TimeSeriesReport); i { + case 0: + return &v.state + case 1: + return &v.sizeCache + case 2: + return &v.unknownFields + default: + return nil + } + } + file_querybackend_v1_querybackend_proto_msgTypes[15].Exporter = func(v any, i int) any { + switch v := v.(*TreeQuery); i { + case 0: + return &v.state + case 1: + return &v.sizeCache + case 2: + return &v.unknownFields + default: + return nil + } + } + file_querybackend_v1_querybackend_proto_msgTypes[16].Exporter = func(v any, i int) any { + switch v := v.(*TreeReport); i { + case 0: + return &v.state + case 1: + return &v.sizeCache + case 2: + return &v.unknownFields + default: + return nil + } + } + } + file_querybackend_v1_querybackend_proto_msgTypes[13].OneofWrappers = []any{} + type x struct{} + out := protoimpl.TypeBuilder{ + File: protoimpl.DescBuilder{ + GoPackagePath: reflect.TypeOf(x{}).PkgPath(), + RawDescriptor: file_querybackend_v1_querybackend_proto_rawDesc, + NumEnums: 2, + NumMessages: 17, + NumExtensions: 0, + NumServices: 1, + }, + GoTypes: file_querybackend_v1_querybackend_proto_goTypes, + DependencyIndexes: file_querybackend_v1_querybackend_proto_depIdxs, + EnumInfos: file_querybackend_v1_querybackend_proto_enumTypes, + MessageInfos: file_querybackend_v1_querybackend_proto_msgTypes, + }.Build() + File_querybackend_v1_querybackend_proto = out.File + file_querybackend_v1_querybackend_proto_rawDesc = nil + file_querybackend_v1_querybackend_proto_goTypes = nil + file_querybackend_v1_querybackend_proto_depIdxs = nil +} diff --git a/api/gen/proto/go/querybackend/v1/querybackend_vtproto.pb.go b/api/gen/proto/go/querybackend/v1/querybackend_vtproto.pb.go new file mode 100644 index 0000000000..1e5ac09fe1 --- /dev/null +++ b/api/gen/proto/go/querybackend/v1/querybackend_vtproto.pb.go @@ -0,0 +1,4500 @@ +// Code generated by protoc-gen-go-vtproto. DO NOT EDIT. +// protoc-gen-go-vtproto version: v0.6.0 +// source: querybackend/v1/querybackend.proto + +package querybackendv1 + +import ( + context "context" + binary "encoding/binary" + fmt "fmt" + v1 "github.com/grafana/pyroscope/api/gen/proto/go/metastore/v1" + v11 "github.com/grafana/pyroscope/api/gen/proto/go/types/v1" + protohelpers "github.com/planetscale/vtprotobuf/protohelpers" + grpc "google.golang.org/grpc" + codes "google.golang.org/grpc/codes" + status "google.golang.org/grpc/status" + proto "google.golang.org/protobuf/proto" + protoimpl "google.golang.org/protobuf/runtime/protoimpl" + io "io" + math "math" +) + +const ( + // Verify that this generated code is sufficiently up-to-date. + _ = protoimpl.EnforceVersion(20 - protoimpl.MinVersion) + // Verify that runtime/protoimpl is sufficiently up-to-date. + _ = protoimpl.EnforceVersion(protoimpl.MaxVersion - 20) +) + +func (m *InvokeOptions) CloneVT() *InvokeOptions { + if m == nil { + return (*InvokeOptions)(nil) + } + r := new(InvokeOptions) + if len(m.unknownFields) > 0 { + r.unknownFields = make([]byte, len(m.unknownFields)) + copy(r.unknownFields, m.unknownFields) + } + return r +} + +func (m *InvokeOptions) CloneMessageVT() proto.Message { + return m.CloneVT() +} + +func (m *InvokeRequest) CloneVT() *InvokeRequest { + if m == nil { + return (*InvokeRequest)(nil) + } + r := new(InvokeRequest) + r.StartTime = m.StartTime + r.EndTime = m.EndTime + r.LabelSelector = m.LabelSelector + r.QueryPlan = m.QueryPlan.CloneVT() + r.Options = m.Options.CloneVT() + if rhs := m.Tenant; rhs != nil { + tmpContainer := make([]string, len(rhs)) + copy(tmpContainer, rhs) + r.Tenant = tmpContainer + } + if rhs := m.Query; rhs != nil { + tmpContainer := make([]*Query, len(rhs)) + for k, v := range rhs { + tmpContainer[k] = v.CloneVT() + } + r.Query = tmpContainer + } + if len(m.unknownFields) > 0 { + r.unknownFields = make([]byte, len(m.unknownFields)) + copy(r.unknownFields, m.unknownFields) + } + return r +} + +func (m *InvokeRequest) CloneMessageVT() proto.Message { + return m.CloneVT() +} + +func (m *QueryPlan) CloneVT() *QueryPlan { + if m == nil { + return (*QueryPlan)(nil) + } + r := new(QueryPlan) + if rhs := m.Graph; rhs != nil { + tmpContainer := make([]uint32, len(rhs)) + copy(tmpContainer, rhs) + r.Graph = tmpContainer + } + if rhs := m.Blocks; rhs != nil { + tmpContainer := make([]*v1.BlockMeta, len(rhs)) + for k, v := range rhs { + if vtpb, ok := interface{}(v).(interface{ CloneVT() *v1.BlockMeta }); ok { + tmpContainer[k] = vtpb.CloneVT() + } else { + tmpContainer[k] = proto.Clone(v).(*v1.BlockMeta) + } + } + r.Blocks = tmpContainer + } + if len(m.unknownFields) > 0 { + r.unknownFields = make([]byte, len(m.unknownFields)) + copy(r.unknownFields, m.unknownFields) + } + return r +} + +func (m *QueryPlan) CloneMessageVT() proto.Message { + return m.CloneVT() +} + +func (m *Query) CloneVT() *Query { + if m == nil { + return (*Query)(nil) + } + r := new(Query) + r.QueryType = m.QueryType + r.LabelNames = m.LabelNames.CloneVT() + r.LabelValues = m.LabelValues.CloneVT() + r.SeriesLabels = m.SeriesLabels.CloneVT() + r.TimeSeries = m.TimeSeries.CloneVT() + r.Tree = m.Tree.CloneVT() + if len(m.unknownFields) > 0 { + r.unknownFields = make([]byte, len(m.unknownFields)) + copy(r.unknownFields, m.unknownFields) + } + return r +} + +func (m *Query) CloneMessageVT() proto.Message { + return m.CloneVT() +} + +func (m *InvokeResponse) CloneVT() *InvokeResponse { + if m == nil { + return (*InvokeResponse)(nil) + } + r := new(InvokeResponse) + r.Diagnostics = m.Diagnostics.CloneVT() + if rhs := m.Reports; rhs != nil { + tmpContainer := make([]*Report, len(rhs)) + for k, v := range rhs { + tmpContainer[k] = v.CloneVT() + } + r.Reports = tmpContainer + } + if len(m.unknownFields) > 0 { + r.unknownFields = make([]byte, len(m.unknownFields)) + copy(r.unknownFields, m.unknownFields) + } + return r +} + +func (m *InvokeResponse) CloneMessageVT() proto.Message { + return m.CloneVT() +} + +func (m *Diagnostics) CloneVT() *Diagnostics { + if m == nil { + return (*Diagnostics)(nil) + } + r := new(Diagnostics) + if len(m.unknownFields) > 0 { + r.unknownFields = make([]byte, len(m.unknownFields)) + copy(r.unknownFields, m.unknownFields) + } + return r +} + +func (m *Diagnostics) CloneMessageVT() proto.Message { + return m.CloneVT() +} + +func (m *Report) CloneVT() *Report { + if m == nil { + return (*Report)(nil) + } + r := new(Report) + r.ReportType = m.ReportType + r.LabelNames = m.LabelNames.CloneVT() + r.LabelValues = m.LabelValues.CloneVT() + r.SeriesLabels = m.SeriesLabels.CloneVT() + r.TimeSeries = m.TimeSeries.CloneVT() + r.Tree = m.Tree.CloneVT() + if len(m.unknownFields) > 0 { + r.unknownFields = make([]byte, len(m.unknownFields)) + copy(r.unknownFields, m.unknownFields) + } + return r +} + +func (m *Report) CloneMessageVT() proto.Message { + return m.CloneVT() +} + +func (m *LabelNamesQuery) CloneVT() *LabelNamesQuery { + if m == nil { + return (*LabelNamesQuery)(nil) + } + r := new(LabelNamesQuery) + if len(m.unknownFields) > 0 { + r.unknownFields = make([]byte, len(m.unknownFields)) + copy(r.unknownFields, m.unknownFields) + } + return r +} + +func (m *LabelNamesQuery) CloneMessageVT() proto.Message { + return m.CloneVT() +} + +func (m *LabelNamesReport) CloneVT() *LabelNamesReport { + if m == nil { + return (*LabelNamesReport)(nil) + } + r := new(LabelNamesReport) + r.Query = m.Query.CloneVT() + if rhs := m.LabelNames; rhs != nil { + tmpContainer := make([]string, len(rhs)) + copy(tmpContainer, rhs) + r.LabelNames = tmpContainer + } + if len(m.unknownFields) > 0 { + r.unknownFields = make([]byte, len(m.unknownFields)) + copy(r.unknownFields, m.unknownFields) + } + return r +} + +func (m *LabelNamesReport) CloneMessageVT() proto.Message { + return m.CloneVT() +} + +func (m *LabelValuesQuery) CloneVT() *LabelValuesQuery { + if m == nil { + return (*LabelValuesQuery)(nil) + } + r := new(LabelValuesQuery) + r.LabelName = m.LabelName + if len(m.unknownFields) > 0 { + r.unknownFields = make([]byte, len(m.unknownFields)) + copy(r.unknownFields, m.unknownFields) + } + return r +} + +func (m *LabelValuesQuery) CloneMessageVT() proto.Message { + return m.CloneVT() +} + +func (m *LabelValuesReport) CloneVT() *LabelValuesReport { + if m == nil { + return (*LabelValuesReport)(nil) + } + r := new(LabelValuesReport) + r.Query = m.Query.CloneVT() + if rhs := m.LabelValues; rhs != nil { + tmpContainer := make([]string, len(rhs)) + copy(tmpContainer, rhs) + r.LabelValues = tmpContainer + } + if len(m.unknownFields) > 0 { + r.unknownFields = make([]byte, len(m.unknownFields)) + copy(r.unknownFields, m.unknownFields) + } + return r +} + +func (m *LabelValuesReport) CloneMessageVT() proto.Message { + return m.CloneVT() +} + +func (m *SeriesLabelsQuery) CloneVT() *SeriesLabelsQuery { + if m == nil { + return (*SeriesLabelsQuery)(nil) + } + r := new(SeriesLabelsQuery) + if rhs := m.LabelNames; rhs != nil { + tmpContainer := make([]string, len(rhs)) + copy(tmpContainer, rhs) + r.LabelNames = tmpContainer + } + if len(m.unknownFields) > 0 { + r.unknownFields = make([]byte, len(m.unknownFields)) + copy(r.unknownFields, m.unknownFields) + } + return r +} + +func (m *SeriesLabelsQuery) CloneMessageVT() proto.Message { + return m.CloneVT() +} + +func (m *SeriesLabelsReport) CloneVT() *SeriesLabelsReport { + if m == nil { + return (*SeriesLabelsReport)(nil) + } + r := new(SeriesLabelsReport) + r.Query = m.Query.CloneVT() + if rhs := m.SeriesLabels; rhs != nil { + tmpContainer := make([]*v11.Labels, len(rhs)) + for k, v := range rhs { + if vtpb, ok := interface{}(v).(interface{ CloneVT() *v11.Labels }); ok { + tmpContainer[k] = vtpb.CloneVT() + } else { + tmpContainer[k] = proto.Clone(v).(*v11.Labels) + } + } + r.SeriesLabels = tmpContainer + } + if len(m.unknownFields) > 0 { + r.unknownFields = make([]byte, len(m.unknownFields)) + copy(r.unknownFields, m.unknownFields) + } + return r +} + +func (m *SeriesLabelsReport) CloneMessageVT() proto.Message { + return m.CloneVT() +} + +func (m *TimeSeriesQuery) CloneVT() *TimeSeriesQuery { + if m == nil { + return (*TimeSeriesQuery)(nil) + } + r := new(TimeSeriesQuery) + r.Step = m.Step + if rhs := m.GroupBy; rhs != nil { + tmpContainer := make([]string, len(rhs)) + copy(tmpContainer, rhs) + r.GroupBy = tmpContainer + } + if rhs := m.Aggregation; rhs != nil { + tmpVal := *rhs + r.Aggregation = &tmpVal + } + if len(m.unknownFields) > 0 { + r.unknownFields = make([]byte, len(m.unknownFields)) + copy(r.unknownFields, m.unknownFields) + } + return r +} + +func (m *TimeSeriesQuery) CloneMessageVT() proto.Message { + return m.CloneVT() +} + +func (m *TimeSeriesReport) CloneVT() *TimeSeriesReport { + if m == nil { + return (*TimeSeriesReport)(nil) + } + r := new(TimeSeriesReport) + r.Query = m.Query.CloneVT() + if rhs := m.TimeSeries; rhs != nil { + tmpContainer := make([]*v11.Series, len(rhs)) + for k, v := range rhs { + if vtpb, ok := interface{}(v).(interface{ CloneVT() *v11.Series }); ok { + tmpContainer[k] = vtpb.CloneVT() + } else { + tmpContainer[k] = proto.Clone(v).(*v11.Series) + } + } + r.TimeSeries = tmpContainer + } + if len(m.unknownFields) > 0 { + r.unknownFields = make([]byte, len(m.unknownFields)) + copy(r.unknownFields, m.unknownFields) + } + return r +} + +func (m *TimeSeriesReport) CloneMessageVT() proto.Message { + return m.CloneVT() +} + +func (m *TreeQuery) CloneVT() *TreeQuery { + if m == nil { + return (*TreeQuery)(nil) + } + r := new(TreeQuery) + r.MaxNodes = m.MaxNodes + if len(m.unknownFields) > 0 { + r.unknownFields = make([]byte, len(m.unknownFields)) + copy(r.unknownFields, m.unknownFields) + } + return r +} + +func (m *TreeQuery) CloneMessageVT() proto.Message { + return m.CloneVT() +} + +func (m *TreeReport) CloneVT() *TreeReport { + if m == nil { + return (*TreeReport)(nil) + } + r := new(TreeReport) + r.Query = m.Query.CloneVT() + if rhs := m.Tree; rhs != nil { + tmpBytes := make([]byte, len(rhs)) + copy(tmpBytes, rhs) + r.Tree = tmpBytes + } + if len(m.unknownFields) > 0 { + r.unknownFields = make([]byte, len(m.unknownFields)) + copy(r.unknownFields, m.unknownFields) + } + return r +} + +func (m *TreeReport) CloneMessageVT() proto.Message { + return m.CloneVT() +} + +func (this *InvokeOptions) EqualVT(that *InvokeOptions) bool { + if this == that { + return true + } else if this == nil || that == nil { + return false + } + return string(this.unknownFields) == string(that.unknownFields) +} + +func (this *InvokeOptions) EqualMessageVT(thatMsg proto.Message) bool { + that, ok := thatMsg.(*InvokeOptions) + if !ok { + return false + } + return this.EqualVT(that) +} +func (this *InvokeRequest) EqualVT(that *InvokeRequest) bool { + if this == that { + return true + } else if this == nil || that == nil { + return false + } + if len(this.Tenant) != len(that.Tenant) { + return false + } + for i, vx := range this.Tenant { + vy := that.Tenant[i] + if vx != vy { + return false + } + } + if this.StartTime != that.StartTime { + return false + } + if this.EndTime != that.EndTime { + return false + } + if this.LabelSelector != that.LabelSelector { + return false + } + if len(this.Query) != len(that.Query) { + return false + } + for i, vx := range this.Query { + vy := that.Query[i] + if p, q := vx, vy; p != q { + if p == nil { + p = &Query{} + } + if q == nil { + q = &Query{} + } + if !p.EqualVT(q) { + return false + } + } + } + if !this.QueryPlan.EqualVT(that.QueryPlan) { + return false + } + if !this.Options.EqualVT(that.Options) { + return false + } + return string(this.unknownFields) == string(that.unknownFields) +} + +func (this *InvokeRequest) EqualMessageVT(thatMsg proto.Message) bool { + that, ok := thatMsg.(*InvokeRequest) + if !ok { + return false + } + return this.EqualVT(that) +} +func (this *QueryPlan) EqualVT(that *QueryPlan) bool { + if this == that { + return true + } else if this == nil || that == nil { + return false + } + if len(this.Graph) != len(that.Graph) { + return false + } + for i, vx := range this.Graph { + vy := that.Graph[i] + if vx != vy { + return false + } + } + if len(this.Blocks) != len(that.Blocks) { + return false + } + for i, vx := range this.Blocks { + vy := that.Blocks[i] + if p, q := vx, vy; p != q { + if p == nil { + p = &v1.BlockMeta{} + } + if q == nil { + q = &v1.BlockMeta{} + } + if equal, ok := interface{}(p).(interface{ EqualVT(*v1.BlockMeta) bool }); ok { + if !equal.EqualVT(q) { + return false + } + } else if !proto.Equal(p, q) { + return false + } + } + } + return string(this.unknownFields) == string(that.unknownFields) +} + +func (this *QueryPlan) EqualMessageVT(thatMsg proto.Message) bool { + that, ok := thatMsg.(*QueryPlan) + if !ok { + return false + } + return this.EqualVT(that) +} +func (this *Query) EqualVT(that *Query) bool { + if this == that { + return true + } else if this == nil || that == nil { + return false + } + if this.QueryType != that.QueryType { + return false + } + if !this.LabelNames.EqualVT(that.LabelNames) { + return false + } + if !this.LabelValues.EqualVT(that.LabelValues) { + return false + } + if !this.SeriesLabels.EqualVT(that.SeriesLabels) { + return false + } + if !this.TimeSeries.EqualVT(that.TimeSeries) { + return false + } + if !this.Tree.EqualVT(that.Tree) { + return false + } + return string(this.unknownFields) == string(that.unknownFields) +} + +func (this *Query) EqualMessageVT(thatMsg proto.Message) bool { + that, ok := thatMsg.(*Query) + if !ok { + return false + } + return this.EqualVT(that) +} +func (this *InvokeResponse) EqualVT(that *InvokeResponse) bool { + if this == that { + return true + } else if this == nil || that == nil { + return false + } + if len(this.Reports) != len(that.Reports) { + return false + } + for i, vx := range this.Reports { + vy := that.Reports[i] + if p, q := vx, vy; p != q { + if p == nil { + p = &Report{} + } + if q == nil { + q = &Report{} + } + if !p.EqualVT(q) { + return false + } + } + } + if !this.Diagnostics.EqualVT(that.Diagnostics) { + return false + } + return string(this.unknownFields) == string(that.unknownFields) +} + +func (this *InvokeResponse) EqualMessageVT(thatMsg proto.Message) bool { + that, ok := thatMsg.(*InvokeResponse) + if !ok { + return false + } + return this.EqualVT(that) +} +func (this *Diagnostics) EqualVT(that *Diagnostics) bool { + if this == that { + return true + } else if this == nil || that == nil { + return false + } + return string(this.unknownFields) == string(that.unknownFields) +} + +func (this *Diagnostics) EqualMessageVT(thatMsg proto.Message) bool { + that, ok := thatMsg.(*Diagnostics) + if !ok { + return false + } + return this.EqualVT(that) +} +func (this *Report) EqualVT(that *Report) bool { + if this == that { + return true + } else if this == nil || that == nil { + return false + } + if this.ReportType != that.ReportType { + return false + } + if !this.LabelNames.EqualVT(that.LabelNames) { + return false + } + if !this.LabelValues.EqualVT(that.LabelValues) { + return false + } + if !this.SeriesLabels.EqualVT(that.SeriesLabels) { + return false + } + if !this.TimeSeries.EqualVT(that.TimeSeries) { + return false + } + if !this.Tree.EqualVT(that.Tree) { + return false + } + return string(this.unknownFields) == string(that.unknownFields) +} + +func (this *Report) EqualMessageVT(thatMsg proto.Message) bool { + that, ok := thatMsg.(*Report) + if !ok { + return false + } + return this.EqualVT(that) +} +func (this *LabelNamesQuery) EqualVT(that *LabelNamesQuery) bool { + if this == that { + return true + } else if this == nil || that == nil { + return false + } + return string(this.unknownFields) == string(that.unknownFields) +} + +func (this *LabelNamesQuery) EqualMessageVT(thatMsg proto.Message) bool { + that, ok := thatMsg.(*LabelNamesQuery) + if !ok { + return false + } + return this.EqualVT(that) +} +func (this *LabelNamesReport) EqualVT(that *LabelNamesReport) bool { + if this == that { + return true + } else if this == nil || that == nil { + return false + } + if !this.Query.EqualVT(that.Query) { + return false + } + if len(this.LabelNames) != len(that.LabelNames) { + return false + } + for i, vx := range this.LabelNames { + vy := that.LabelNames[i] + if vx != vy { + return false + } + } + return string(this.unknownFields) == string(that.unknownFields) +} + +func (this *LabelNamesReport) EqualMessageVT(thatMsg proto.Message) bool { + that, ok := thatMsg.(*LabelNamesReport) + if !ok { + return false + } + return this.EqualVT(that) +} +func (this *LabelValuesQuery) EqualVT(that *LabelValuesQuery) bool { + if this == that { + return true + } else if this == nil || that == nil { + return false + } + if this.LabelName != that.LabelName { + return false + } + return string(this.unknownFields) == string(that.unknownFields) +} + +func (this *LabelValuesQuery) EqualMessageVT(thatMsg proto.Message) bool { + that, ok := thatMsg.(*LabelValuesQuery) + if !ok { + return false + } + return this.EqualVT(that) +} +func (this *LabelValuesReport) EqualVT(that *LabelValuesReport) bool { + if this == that { + return true + } else if this == nil || that == nil { + return false + } + if !this.Query.EqualVT(that.Query) { + return false + } + if len(this.LabelValues) != len(that.LabelValues) { + return false + } + for i, vx := range this.LabelValues { + vy := that.LabelValues[i] + if vx != vy { + return false + } + } + return string(this.unknownFields) == string(that.unknownFields) +} + +func (this *LabelValuesReport) EqualMessageVT(thatMsg proto.Message) bool { + that, ok := thatMsg.(*LabelValuesReport) + if !ok { + return false + } + return this.EqualVT(that) +} +func (this *SeriesLabelsQuery) EqualVT(that *SeriesLabelsQuery) bool { + if this == that { + return true + } else if this == nil || that == nil { + return false + } + if len(this.LabelNames) != len(that.LabelNames) { + return false + } + for i, vx := range this.LabelNames { + vy := that.LabelNames[i] + if vx != vy { + return false + } + } + return string(this.unknownFields) == string(that.unknownFields) +} + +func (this *SeriesLabelsQuery) EqualMessageVT(thatMsg proto.Message) bool { + that, ok := thatMsg.(*SeriesLabelsQuery) + if !ok { + return false + } + return this.EqualVT(that) +} +func (this *SeriesLabelsReport) EqualVT(that *SeriesLabelsReport) bool { + if this == that { + return true + } else if this == nil || that == nil { + return false + } + if !this.Query.EqualVT(that.Query) { + return false + } + if len(this.SeriesLabels) != len(that.SeriesLabels) { + return false + } + for i, vx := range this.SeriesLabels { + vy := that.SeriesLabels[i] + if p, q := vx, vy; p != q { + if p == nil { + p = &v11.Labels{} + } + if q == nil { + q = &v11.Labels{} + } + if equal, ok := interface{}(p).(interface{ EqualVT(*v11.Labels) bool }); ok { + if !equal.EqualVT(q) { + return false + } + } else if !proto.Equal(p, q) { + return false + } + } + } + return string(this.unknownFields) == string(that.unknownFields) +} + +func (this *SeriesLabelsReport) EqualMessageVT(thatMsg proto.Message) bool { + that, ok := thatMsg.(*SeriesLabelsReport) + if !ok { + return false + } + return this.EqualVT(that) +} +func (this *TimeSeriesQuery) EqualVT(that *TimeSeriesQuery) bool { + if this == that { + return true + } else if this == nil || that == nil { + return false + } + if this.Step != that.Step { + return false + } + if len(this.GroupBy) != len(that.GroupBy) { + return false + } + for i, vx := range this.GroupBy { + vy := that.GroupBy[i] + if vx != vy { + return false + } + } + if p, q := this.Aggregation, that.Aggregation; (p == nil && q != nil) || (p != nil && (q == nil || *p != *q)) { + return false + } + return string(this.unknownFields) == string(that.unknownFields) +} + +func (this *TimeSeriesQuery) EqualMessageVT(thatMsg proto.Message) bool { + that, ok := thatMsg.(*TimeSeriesQuery) + if !ok { + return false + } + return this.EqualVT(that) +} +func (this *TimeSeriesReport) EqualVT(that *TimeSeriesReport) bool { + if this == that { + return true + } else if this == nil || that == nil { + return false + } + if !this.Query.EqualVT(that.Query) { + return false + } + if len(this.TimeSeries) != len(that.TimeSeries) { + return false + } + for i, vx := range this.TimeSeries { + vy := that.TimeSeries[i] + if p, q := vx, vy; p != q { + if p == nil { + p = &v11.Series{} + } + if q == nil { + q = &v11.Series{} + } + if equal, ok := interface{}(p).(interface{ EqualVT(*v11.Series) bool }); ok { + if !equal.EqualVT(q) { + return false + } + } else if !proto.Equal(p, q) { + return false + } + } + } + return string(this.unknownFields) == string(that.unknownFields) +} + +func (this *TimeSeriesReport) EqualMessageVT(thatMsg proto.Message) bool { + that, ok := thatMsg.(*TimeSeriesReport) + if !ok { + return false + } + return this.EqualVT(that) +} +func (this *TreeQuery) EqualVT(that *TreeQuery) bool { + if this == that { + return true + } else if this == nil || that == nil { + return false + } + if this.MaxNodes != that.MaxNodes { + return false + } + return string(this.unknownFields) == string(that.unknownFields) +} + +func (this *TreeQuery) EqualMessageVT(thatMsg proto.Message) bool { + that, ok := thatMsg.(*TreeQuery) + if !ok { + return false + } + return this.EqualVT(that) +} +func (this *TreeReport) EqualVT(that *TreeReport) bool { + if this == that { + return true + } else if this == nil || that == nil { + return false + } + if !this.Query.EqualVT(that.Query) { + return false + } + if string(this.Tree) != string(that.Tree) { + return false + } + return string(this.unknownFields) == string(that.unknownFields) +} + +func (this *TreeReport) EqualMessageVT(thatMsg proto.Message) bool { + that, ok := thatMsg.(*TreeReport) + if !ok { + return false + } + return this.EqualVT(that) +} + +// This is a compile-time assertion to ensure that this generated file +// is compatible with the grpc package it is being compiled against. +// Requires gRPC-Go v1.32.0 or later. +const _ = grpc.SupportPackageIsVersion7 + +// QueryBackendServiceClient is the client API for QueryBackendService service. +// +// For semantics around ctx use and closing/ending streaming RPCs, please refer to https://pkg.go.dev/google.golang.org/grpc/?tab=doc#ClientConn.NewStream. +type QueryBackendServiceClient interface { + Invoke(ctx context.Context, in *InvokeRequest, opts ...grpc.CallOption) (*InvokeResponse, error) +} + +type queryBackendServiceClient struct { + cc grpc.ClientConnInterface +} + +func NewQueryBackendServiceClient(cc grpc.ClientConnInterface) QueryBackendServiceClient { + return &queryBackendServiceClient{cc} +} + +func (c *queryBackendServiceClient) Invoke(ctx context.Context, in *InvokeRequest, opts ...grpc.CallOption) (*InvokeResponse, error) { + out := new(InvokeResponse) + err := c.cc.Invoke(ctx, "/querybackend.v1.QueryBackendService/Invoke", in, out, opts...) + if err != nil { + return nil, err + } + return out, nil +} + +// QueryBackendServiceServer is the server API for QueryBackendService service. +// All implementations must embed UnimplementedQueryBackendServiceServer +// for forward compatibility +type QueryBackendServiceServer interface { + Invoke(context.Context, *InvokeRequest) (*InvokeResponse, error) + mustEmbedUnimplementedQueryBackendServiceServer() +} + +// UnimplementedQueryBackendServiceServer must be embedded to have forward compatible implementations. +type UnimplementedQueryBackendServiceServer struct { +} + +func (UnimplementedQueryBackendServiceServer) Invoke(context.Context, *InvokeRequest) (*InvokeResponse, error) { + return nil, status.Errorf(codes.Unimplemented, "method Invoke not implemented") +} +func (UnimplementedQueryBackendServiceServer) mustEmbedUnimplementedQueryBackendServiceServer() {} + +// UnsafeQueryBackendServiceServer may be embedded to opt out of forward compatibility for this service. +// Use of this interface is not recommended, as added methods to QueryBackendServiceServer will +// result in compilation errors. +type UnsafeQueryBackendServiceServer interface { + mustEmbedUnimplementedQueryBackendServiceServer() +} + +func RegisterQueryBackendServiceServer(s grpc.ServiceRegistrar, srv QueryBackendServiceServer) { + s.RegisterService(&QueryBackendService_ServiceDesc, srv) +} + +func _QueryBackendService_Invoke_Handler(srv interface{}, ctx context.Context, dec func(interface{}) error, interceptor grpc.UnaryServerInterceptor) (interface{}, error) { + in := new(InvokeRequest) + if err := dec(in); err != nil { + return nil, err + } + if interceptor == nil { + return srv.(QueryBackendServiceServer).Invoke(ctx, in) + } + info := &grpc.UnaryServerInfo{ + Server: srv, + FullMethod: "/querybackend.v1.QueryBackendService/Invoke", + } + handler := func(ctx context.Context, req interface{}) (interface{}, error) { + return srv.(QueryBackendServiceServer).Invoke(ctx, req.(*InvokeRequest)) + } + return interceptor(ctx, in, info, handler) +} + +// QueryBackendService_ServiceDesc is the grpc.ServiceDesc for QueryBackendService service. +// It's only intended for direct use with grpc.RegisterService, +// and not to be introspected or modified (even as a copy) +var QueryBackendService_ServiceDesc = grpc.ServiceDesc{ + ServiceName: "querybackend.v1.QueryBackendService", + HandlerType: (*QueryBackendServiceServer)(nil), + Methods: []grpc.MethodDesc{ + { + MethodName: "Invoke", + Handler: _QueryBackendService_Invoke_Handler, + }, + }, + Streams: []grpc.StreamDesc{}, + Metadata: "querybackend/v1/querybackend.proto", +} + +func (m *InvokeOptions) MarshalVT() (dAtA []byte, err error) { + if m == nil { + return nil, nil + } + size := m.SizeVT() + dAtA = make([]byte, size) + n, err := m.MarshalToSizedBufferVT(dAtA[:size]) + if err != nil { + return nil, err + } + return dAtA[:n], nil +} + +func (m *InvokeOptions) MarshalToVT(dAtA []byte) (int, error) { + size := m.SizeVT() + return m.MarshalToSizedBufferVT(dAtA[:size]) +} + +func (m *InvokeOptions) MarshalToSizedBufferVT(dAtA []byte) (int, error) { + if m == nil { + return 0, nil + } + i := len(dAtA) + _ = i + var l int + _ = l + if m.unknownFields != nil { + i -= len(m.unknownFields) + copy(dAtA[i:], m.unknownFields) + } + return len(dAtA) - i, nil +} + +func (m *InvokeRequest) MarshalVT() (dAtA []byte, err error) { + if m == nil { + return nil, nil + } + size := m.SizeVT() + dAtA = make([]byte, size) + n, err := m.MarshalToSizedBufferVT(dAtA[:size]) + if err != nil { + return nil, err + } + return dAtA[:n], nil +} + +func (m *InvokeRequest) MarshalToVT(dAtA []byte) (int, error) { + size := m.SizeVT() + return m.MarshalToSizedBufferVT(dAtA[:size]) +} + +func (m *InvokeRequest) MarshalToSizedBufferVT(dAtA []byte) (int, error) { + if m == nil { + return 0, nil + } + i := len(dAtA) + _ = i + var l int + _ = l + if m.unknownFields != nil { + i -= len(m.unknownFields) + copy(dAtA[i:], m.unknownFields) + } + if m.Options != nil { + size, err := m.Options.MarshalToSizedBufferVT(dAtA[:i]) + if err != nil { + return 0, err + } + i -= size + i = protohelpers.EncodeVarint(dAtA, i, uint64(size)) + i-- + dAtA[i] = 0x3a + } + if m.QueryPlan != nil { + size, err := m.QueryPlan.MarshalToSizedBufferVT(dAtA[:i]) + if err != nil { + return 0, err + } + i -= size + i = protohelpers.EncodeVarint(dAtA, i, uint64(size)) + i-- + dAtA[i] = 0x32 + } + if len(m.Query) > 0 { + for iNdEx := len(m.Query) - 1; iNdEx >= 0; iNdEx-- { + size, err := m.Query[iNdEx].MarshalToSizedBufferVT(dAtA[:i]) + if err != nil { + return 0, err + } + i -= size + i = protohelpers.EncodeVarint(dAtA, i, uint64(size)) + i-- + dAtA[i] = 0x2a + } + } + if len(m.LabelSelector) > 0 { + i -= len(m.LabelSelector) + copy(dAtA[i:], m.LabelSelector) + i = protohelpers.EncodeVarint(dAtA, i, uint64(len(m.LabelSelector))) + i-- + dAtA[i] = 0x22 + } + if m.EndTime != 0 { + i = protohelpers.EncodeVarint(dAtA, i, uint64(m.EndTime)) + i-- + dAtA[i] = 0x18 + } + if m.StartTime != 0 { + i = protohelpers.EncodeVarint(dAtA, i, uint64(m.StartTime)) + i-- + dAtA[i] = 0x10 + } + if len(m.Tenant) > 0 { + for iNdEx := len(m.Tenant) - 1; iNdEx >= 0; iNdEx-- { + i -= len(m.Tenant[iNdEx]) + copy(dAtA[i:], m.Tenant[iNdEx]) + i = protohelpers.EncodeVarint(dAtA, i, uint64(len(m.Tenant[iNdEx]))) + i-- + dAtA[i] = 0xa + } + } + return len(dAtA) - i, nil +} + +func (m *QueryPlan) MarshalVT() (dAtA []byte, err error) { + if m == nil { + return nil, nil + } + size := m.SizeVT() + dAtA = make([]byte, size) + n, err := m.MarshalToSizedBufferVT(dAtA[:size]) + if err != nil { + return nil, err + } + return dAtA[:n], nil +} + +func (m *QueryPlan) MarshalToVT(dAtA []byte) (int, error) { + size := m.SizeVT() + return m.MarshalToSizedBufferVT(dAtA[:size]) +} + +func (m *QueryPlan) MarshalToSizedBufferVT(dAtA []byte) (int, error) { + if m == nil { + return 0, nil + } + i := len(dAtA) + _ = i + var l int + _ = l + if m.unknownFields != nil { + i -= len(m.unknownFields) + copy(dAtA[i:], m.unknownFields) + } + if len(m.Blocks) > 0 { + for iNdEx := len(m.Blocks) - 1; iNdEx >= 0; iNdEx-- { + if vtmsg, ok := interface{}(m.Blocks[iNdEx]).(interface { + MarshalToSizedBufferVT([]byte) (int, error) + }); ok { + size, err := vtmsg.MarshalToSizedBufferVT(dAtA[:i]) + if err != nil { + return 0, err + } + i -= size + i = protohelpers.EncodeVarint(dAtA, i, uint64(size)) + } else { + encoded, err := proto.Marshal(m.Blocks[iNdEx]) + if err != nil { + return 0, err + } + i -= len(encoded) + copy(dAtA[i:], encoded) + i = protohelpers.EncodeVarint(dAtA, i, uint64(len(encoded))) + } + i-- + dAtA[i] = 0x12 + } + } + if len(m.Graph) > 0 { + var pksize2 int + for _, num := range m.Graph { + pksize2 += protohelpers.SizeOfVarint(uint64(num)) + } + i -= pksize2 + j1 := i + for _, num := range m.Graph { + for num >= 1<<7 { + dAtA[j1] = uint8(uint64(num)&0x7f | 0x80) + num >>= 7 + j1++ + } + dAtA[j1] = uint8(num) + j1++ + } + i = protohelpers.EncodeVarint(dAtA, i, uint64(pksize2)) + i-- + dAtA[i] = 0xa + } + return len(dAtA) - i, nil +} + +func (m *Query) MarshalVT() (dAtA []byte, err error) { + if m == nil { + return nil, nil + } + size := m.SizeVT() + dAtA = make([]byte, size) + n, err := m.MarshalToSizedBufferVT(dAtA[:size]) + if err != nil { + return nil, err + } + return dAtA[:n], nil +} + +func (m *Query) MarshalToVT(dAtA []byte) (int, error) { + size := m.SizeVT() + return m.MarshalToSizedBufferVT(dAtA[:size]) +} + +func (m *Query) MarshalToSizedBufferVT(dAtA []byte) (int, error) { + if m == nil { + return 0, nil + } + i := len(dAtA) + _ = i + var l int + _ = l + if m.unknownFields != nil { + i -= len(m.unknownFields) + copy(dAtA[i:], m.unknownFields) + } + if m.Tree != nil { + size, err := m.Tree.MarshalToSizedBufferVT(dAtA[:i]) + if err != nil { + return 0, err + } + i -= size + i = protohelpers.EncodeVarint(dAtA, i, uint64(size)) + i-- + dAtA[i] = 0x32 + } + if m.TimeSeries != nil { + size, err := m.TimeSeries.MarshalToSizedBufferVT(dAtA[:i]) + if err != nil { + return 0, err + } + i -= size + i = protohelpers.EncodeVarint(dAtA, i, uint64(size)) + i-- + dAtA[i] = 0x2a + } + if m.SeriesLabels != nil { + size, err := m.SeriesLabels.MarshalToSizedBufferVT(dAtA[:i]) + if err != nil { + return 0, err + } + i -= size + i = protohelpers.EncodeVarint(dAtA, i, uint64(size)) + i-- + dAtA[i] = 0x22 + } + if m.LabelValues != nil { + size, err := m.LabelValues.MarshalToSizedBufferVT(dAtA[:i]) + if err != nil { + return 0, err + } + i -= size + i = protohelpers.EncodeVarint(dAtA, i, uint64(size)) + i-- + dAtA[i] = 0x1a + } + if m.LabelNames != nil { + size, err := m.LabelNames.MarshalToSizedBufferVT(dAtA[:i]) + if err != nil { + return 0, err + } + i -= size + i = protohelpers.EncodeVarint(dAtA, i, uint64(size)) + i-- + dAtA[i] = 0x12 + } + if m.QueryType != 0 { + i = protohelpers.EncodeVarint(dAtA, i, uint64(m.QueryType)) + i-- + dAtA[i] = 0x8 + } + return len(dAtA) - i, nil +} + +func (m *InvokeResponse) MarshalVT() (dAtA []byte, err error) { + if m == nil { + return nil, nil + } + size := m.SizeVT() + dAtA = make([]byte, size) + n, err := m.MarshalToSizedBufferVT(dAtA[:size]) + if err != nil { + return nil, err + } + return dAtA[:n], nil +} + +func (m *InvokeResponse) MarshalToVT(dAtA []byte) (int, error) { + size := m.SizeVT() + return m.MarshalToSizedBufferVT(dAtA[:size]) +} + +func (m *InvokeResponse) MarshalToSizedBufferVT(dAtA []byte) (int, error) { + if m == nil { + return 0, nil + } + i := len(dAtA) + _ = i + var l int + _ = l + if m.unknownFields != nil { + i -= len(m.unknownFields) + copy(dAtA[i:], m.unknownFields) + } + if m.Diagnostics != nil { + size, err := m.Diagnostics.MarshalToSizedBufferVT(dAtA[:i]) + if err != nil { + return 0, err + } + i -= size + i = protohelpers.EncodeVarint(dAtA, i, uint64(size)) + i-- + dAtA[i] = 0x12 + } + if len(m.Reports) > 0 { + for iNdEx := len(m.Reports) - 1; iNdEx >= 0; iNdEx-- { + size, err := m.Reports[iNdEx].MarshalToSizedBufferVT(dAtA[:i]) + if err != nil { + return 0, err + } + i -= size + i = protohelpers.EncodeVarint(dAtA, i, uint64(size)) + i-- + dAtA[i] = 0xa + } + } + return len(dAtA) - i, nil +} + +func (m *Diagnostics) MarshalVT() (dAtA []byte, err error) { + if m == nil { + return nil, nil + } + size := m.SizeVT() + dAtA = make([]byte, size) + n, err := m.MarshalToSizedBufferVT(dAtA[:size]) + if err != nil { + return nil, err + } + return dAtA[:n], nil +} + +func (m *Diagnostics) MarshalToVT(dAtA []byte) (int, error) { + size := m.SizeVT() + return m.MarshalToSizedBufferVT(dAtA[:size]) +} + +func (m *Diagnostics) MarshalToSizedBufferVT(dAtA []byte) (int, error) { + if m == nil { + return 0, nil + } + i := len(dAtA) + _ = i + var l int + _ = l + if m.unknownFields != nil { + i -= len(m.unknownFields) + copy(dAtA[i:], m.unknownFields) + } + return len(dAtA) - i, nil +} + +func (m *Report) MarshalVT() (dAtA []byte, err error) { + if m == nil { + return nil, nil + } + size := m.SizeVT() + dAtA = make([]byte, size) + n, err := m.MarshalToSizedBufferVT(dAtA[:size]) + if err != nil { + return nil, err + } + return dAtA[:n], nil +} + +func (m *Report) MarshalToVT(dAtA []byte) (int, error) { + size := m.SizeVT() + return m.MarshalToSizedBufferVT(dAtA[:size]) +} + +func (m *Report) MarshalToSizedBufferVT(dAtA []byte) (int, error) { + if m == nil { + return 0, nil + } + i := len(dAtA) + _ = i + var l int + _ = l + if m.unknownFields != nil { + i -= len(m.unknownFields) + copy(dAtA[i:], m.unknownFields) + } + if m.Tree != nil { + size, err := m.Tree.MarshalToSizedBufferVT(dAtA[:i]) + if err != nil { + return 0, err + } + i -= size + i = protohelpers.EncodeVarint(dAtA, i, uint64(size)) + i-- + dAtA[i] = 0x32 + } + if m.TimeSeries != nil { + size, err := m.TimeSeries.MarshalToSizedBufferVT(dAtA[:i]) + if err != nil { + return 0, err + } + i -= size + i = protohelpers.EncodeVarint(dAtA, i, uint64(size)) + i-- + dAtA[i] = 0x2a + } + if m.SeriesLabels != nil { + size, err := m.SeriesLabels.MarshalToSizedBufferVT(dAtA[:i]) + if err != nil { + return 0, err + } + i -= size + i = protohelpers.EncodeVarint(dAtA, i, uint64(size)) + i-- + dAtA[i] = 0x22 + } + if m.LabelValues != nil { + size, err := m.LabelValues.MarshalToSizedBufferVT(dAtA[:i]) + if err != nil { + return 0, err + } + i -= size + i = protohelpers.EncodeVarint(dAtA, i, uint64(size)) + i-- + dAtA[i] = 0x1a + } + if m.LabelNames != nil { + size, err := m.LabelNames.MarshalToSizedBufferVT(dAtA[:i]) + if err != nil { + return 0, err + } + i -= size + i = protohelpers.EncodeVarint(dAtA, i, uint64(size)) + i-- + dAtA[i] = 0x12 + } + if m.ReportType != 0 { + i = protohelpers.EncodeVarint(dAtA, i, uint64(m.ReportType)) + i-- + dAtA[i] = 0x8 + } + return len(dAtA) - i, nil +} + +func (m *LabelNamesQuery) MarshalVT() (dAtA []byte, err error) { + if m == nil { + return nil, nil + } + size := m.SizeVT() + dAtA = make([]byte, size) + n, err := m.MarshalToSizedBufferVT(dAtA[:size]) + if err != nil { + return nil, err + } + return dAtA[:n], nil +} + +func (m *LabelNamesQuery) MarshalToVT(dAtA []byte) (int, error) { + size := m.SizeVT() + return m.MarshalToSizedBufferVT(dAtA[:size]) +} + +func (m *LabelNamesQuery) MarshalToSizedBufferVT(dAtA []byte) (int, error) { + if m == nil { + return 0, nil + } + i := len(dAtA) + _ = i + var l int + _ = l + if m.unknownFields != nil { + i -= len(m.unknownFields) + copy(dAtA[i:], m.unknownFields) + } + return len(dAtA) - i, nil +} + +func (m *LabelNamesReport) MarshalVT() (dAtA []byte, err error) { + if m == nil { + return nil, nil + } + size := m.SizeVT() + dAtA = make([]byte, size) + n, err := m.MarshalToSizedBufferVT(dAtA[:size]) + if err != nil { + return nil, err + } + return dAtA[:n], nil +} + +func (m *LabelNamesReport) MarshalToVT(dAtA []byte) (int, error) { + size := m.SizeVT() + return m.MarshalToSizedBufferVT(dAtA[:size]) +} + +func (m *LabelNamesReport) MarshalToSizedBufferVT(dAtA []byte) (int, error) { + if m == nil { + return 0, nil + } + i := len(dAtA) + _ = i + var l int + _ = l + if m.unknownFields != nil { + i -= len(m.unknownFields) + copy(dAtA[i:], m.unknownFields) + } + if len(m.LabelNames) > 0 { + for iNdEx := len(m.LabelNames) - 1; iNdEx >= 0; iNdEx-- { + i -= len(m.LabelNames[iNdEx]) + copy(dAtA[i:], m.LabelNames[iNdEx]) + i = protohelpers.EncodeVarint(dAtA, i, uint64(len(m.LabelNames[iNdEx]))) + i-- + dAtA[i] = 0x12 + } + } + if m.Query != nil { + size, err := m.Query.MarshalToSizedBufferVT(dAtA[:i]) + if err != nil { + return 0, err + } + i -= size + i = protohelpers.EncodeVarint(dAtA, i, uint64(size)) + i-- + dAtA[i] = 0xa + } + return len(dAtA) - i, nil +} + +func (m *LabelValuesQuery) MarshalVT() (dAtA []byte, err error) { + if m == nil { + return nil, nil + } + size := m.SizeVT() + dAtA = make([]byte, size) + n, err := m.MarshalToSizedBufferVT(dAtA[:size]) + if err != nil { + return nil, err + } + return dAtA[:n], nil +} + +func (m *LabelValuesQuery) MarshalToVT(dAtA []byte) (int, error) { + size := m.SizeVT() + return m.MarshalToSizedBufferVT(dAtA[:size]) +} + +func (m *LabelValuesQuery) MarshalToSizedBufferVT(dAtA []byte) (int, error) { + if m == nil { + return 0, nil + } + i := len(dAtA) + _ = i + var l int + _ = l + if m.unknownFields != nil { + i -= len(m.unknownFields) + copy(dAtA[i:], m.unknownFields) + } + if len(m.LabelName) > 0 { + i -= len(m.LabelName) + copy(dAtA[i:], m.LabelName) + i = protohelpers.EncodeVarint(dAtA, i, uint64(len(m.LabelName))) + i-- + dAtA[i] = 0xa + } + return len(dAtA) - i, nil +} + +func (m *LabelValuesReport) MarshalVT() (dAtA []byte, err error) { + if m == nil { + return nil, nil + } + size := m.SizeVT() + dAtA = make([]byte, size) + n, err := m.MarshalToSizedBufferVT(dAtA[:size]) + if err != nil { + return nil, err + } + return dAtA[:n], nil +} + +func (m *LabelValuesReport) MarshalToVT(dAtA []byte) (int, error) { + size := m.SizeVT() + return m.MarshalToSizedBufferVT(dAtA[:size]) +} + +func (m *LabelValuesReport) MarshalToSizedBufferVT(dAtA []byte) (int, error) { + if m == nil { + return 0, nil + } + i := len(dAtA) + _ = i + var l int + _ = l + if m.unknownFields != nil { + i -= len(m.unknownFields) + copy(dAtA[i:], m.unknownFields) + } + if len(m.LabelValues) > 0 { + for iNdEx := len(m.LabelValues) - 1; iNdEx >= 0; iNdEx-- { + i -= len(m.LabelValues[iNdEx]) + copy(dAtA[i:], m.LabelValues[iNdEx]) + i = protohelpers.EncodeVarint(dAtA, i, uint64(len(m.LabelValues[iNdEx]))) + i-- + dAtA[i] = 0x12 + } + } + if m.Query != nil { + size, err := m.Query.MarshalToSizedBufferVT(dAtA[:i]) + if err != nil { + return 0, err + } + i -= size + i = protohelpers.EncodeVarint(dAtA, i, uint64(size)) + i-- + dAtA[i] = 0xa + } + return len(dAtA) - i, nil +} + +func (m *SeriesLabelsQuery) MarshalVT() (dAtA []byte, err error) { + if m == nil { + return nil, nil + } + size := m.SizeVT() + dAtA = make([]byte, size) + n, err := m.MarshalToSizedBufferVT(dAtA[:size]) + if err != nil { + return nil, err + } + return dAtA[:n], nil +} + +func (m *SeriesLabelsQuery) MarshalToVT(dAtA []byte) (int, error) { + size := m.SizeVT() + return m.MarshalToSizedBufferVT(dAtA[:size]) +} + +func (m *SeriesLabelsQuery) MarshalToSizedBufferVT(dAtA []byte) (int, error) { + if m == nil { + return 0, nil + } + i := len(dAtA) + _ = i + var l int + _ = l + if m.unknownFields != nil { + i -= len(m.unknownFields) + copy(dAtA[i:], m.unknownFields) + } + if len(m.LabelNames) > 0 { + for iNdEx := len(m.LabelNames) - 1; iNdEx >= 0; iNdEx-- { + i -= len(m.LabelNames[iNdEx]) + copy(dAtA[i:], m.LabelNames[iNdEx]) + i = protohelpers.EncodeVarint(dAtA, i, uint64(len(m.LabelNames[iNdEx]))) + i-- + dAtA[i] = 0xa + } + } + return len(dAtA) - i, nil +} + +func (m *SeriesLabelsReport) MarshalVT() (dAtA []byte, err error) { + if m == nil { + return nil, nil + } + size := m.SizeVT() + dAtA = make([]byte, size) + n, err := m.MarshalToSizedBufferVT(dAtA[:size]) + if err != nil { + return nil, err + } + return dAtA[:n], nil +} + +func (m *SeriesLabelsReport) MarshalToVT(dAtA []byte) (int, error) { + size := m.SizeVT() + return m.MarshalToSizedBufferVT(dAtA[:size]) +} + +func (m *SeriesLabelsReport) MarshalToSizedBufferVT(dAtA []byte) (int, error) { + if m == nil { + return 0, nil + } + i := len(dAtA) + _ = i + var l int + _ = l + if m.unknownFields != nil { + i -= len(m.unknownFields) + copy(dAtA[i:], m.unknownFields) + } + if len(m.SeriesLabels) > 0 { + for iNdEx := len(m.SeriesLabels) - 1; iNdEx >= 0; iNdEx-- { + if vtmsg, ok := interface{}(m.SeriesLabels[iNdEx]).(interface { + MarshalToSizedBufferVT([]byte) (int, error) + }); ok { + size, err := vtmsg.MarshalToSizedBufferVT(dAtA[:i]) + if err != nil { + return 0, err + } + i -= size + i = protohelpers.EncodeVarint(dAtA, i, uint64(size)) + } else { + encoded, err := proto.Marshal(m.SeriesLabels[iNdEx]) + if err != nil { + return 0, err + } + i -= len(encoded) + copy(dAtA[i:], encoded) + i = protohelpers.EncodeVarint(dAtA, i, uint64(len(encoded))) + } + i-- + dAtA[i] = 0x12 + } + } + if m.Query != nil { + size, err := m.Query.MarshalToSizedBufferVT(dAtA[:i]) + if err != nil { + return 0, err + } + i -= size + i = protohelpers.EncodeVarint(dAtA, i, uint64(size)) + i-- + dAtA[i] = 0xa + } + return len(dAtA) - i, nil +} + +func (m *TimeSeriesQuery) MarshalVT() (dAtA []byte, err error) { + if m == nil { + return nil, nil + } + size := m.SizeVT() + dAtA = make([]byte, size) + n, err := m.MarshalToSizedBufferVT(dAtA[:size]) + if err != nil { + return nil, err + } + return dAtA[:n], nil +} + +func (m *TimeSeriesQuery) MarshalToVT(dAtA []byte) (int, error) { + size := m.SizeVT() + return m.MarshalToSizedBufferVT(dAtA[:size]) +} + +func (m *TimeSeriesQuery) MarshalToSizedBufferVT(dAtA []byte) (int, error) { + if m == nil { + return 0, nil + } + i := len(dAtA) + _ = i + var l int + _ = l + if m.unknownFields != nil { + i -= len(m.unknownFields) + copy(dAtA[i:], m.unknownFields) + } + if m.Aggregation != nil { + i = protohelpers.EncodeVarint(dAtA, i, uint64(*m.Aggregation)) + i-- + dAtA[i] = 0x18 + } + if len(m.GroupBy) > 0 { + for iNdEx := len(m.GroupBy) - 1; iNdEx >= 0; iNdEx-- { + i -= len(m.GroupBy[iNdEx]) + copy(dAtA[i:], m.GroupBy[iNdEx]) + i = protohelpers.EncodeVarint(dAtA, i, uint64(len(m.GroupBy[iNdEx]))) + i-- + dAtA[i] = 0x12 + } + } + if m.Step != 0 { + i -= 8 + binary.LittleEndian.PutUint64(dAtA[i:], uint64(math.Float64bits(float64(m.Step)))) + i-- + dAtA[i] = 0x9 + } + return len(dAtA) - i, nil +} + +func (m *TimeSeriesReport) MarshalVT() (dAtA []byte, err error) { + if m == nil { + return nil, nil + } + size := m.SizeVT() + dAtA = make([]byte, size) + n, err := m.MarshalToSizedBufferVT(dAtA[:size]) + if err != nil { + return nil, err + } + return dAtA[:n], nil +} + +func (m *TimeSeriesReport) MarshalToVT(dAtA []byte) (int, error) { + size := m.SizeVT() + return m.MarshalToSizedBufferVT(dAtA[:size]) +} + +func (m *TimeSeriesReport) MarshalToSizedBufferVT(dAtA []byte) (int, error) { + if m == nil { + return 0, nil + } + i := len(dAtA) + _ = i + var l int + _ = l + if m.unknownFields != nil { + i -= len(m.unknownFields) + copy(dAtA[i:], m.unknownFields) + } + if len(m.TimeSeries) > 0 { + for iNdEx := len(m.TimeSeries) - 1; iNdEx >= 0; iNdEx-- { + if vtmsg, ok := interface{}(m.TimeSeries[iNdEx]).(interface { + MarshalToSizedBufferVT([]byte) (int, error) + }); ok { + size, err := vtmsg.MarshalToSizedBufferVT(dAtA[:i]) + if err != nil { + return 0, err + } + i -= size + i = protohelpers.EncodeVarint(dAtA, i, uint64(size)) + } else { + encoded, err := proto.Marshal(m.TimeSeries[iNdEx]) + if err != nil { + return 0, err + } + i -= len(encoded) + copy(dAtA[i:], encoded) + i = protohelpers.EncodeVarint(dAtA, i, uint64(len(encoded))) + } + i-- + dAtA[i] = 0x12 + } + } + if m.Query != nil { + size, err := m.Query.MarshalToSizedBufferVT(dAtA[:i]) + if err != nil { + return 0, err + } + i -= size + i = protohelpers.EncodeVarint(dAtA, i, uint64(size)) + i-- + dAtA[i] = 0xa + } + return len(dAtA) - i, nil +} + +func (m *TreeQuery) MarshalVT() (dAtA []byte, err error) { + if m == nil { + return nil, nil + } + size := m.SizeVT() + dAtA = make([]byte, size) + n, err := m.MarshalToSizedBufferVT(dAtA[:size]) + if err != nil { + return nil, err + } + return dAtA[:n], nil +} + +func (m *TreeQuery) MarshalToVT(dAtA []byte) (int, error) { + size := m.SizeVT() + return m.MarshalToSizedBufferVT(dAtA[:size]) +} + +func (m *TreeQuery) MarshalToSizedBufferVT(dAtA []byte) (int, error) { + if m == nil { + return 0, nil + } + i := len(dAtA) + _ = i + var l int + _ = l + if m.unknownFields != nil { + i -= len(m.unknownFields) + copy(dAtA[i:], m.unknownFields) + } + if m.MaxNodes != 0 { + i = protohelpers.EncodeVarint(dAtA, i, uint64(m.MaxNodes)) + i-- + dAtA[i] = 0x8 + } + return len(dAtA) - i, nil +} + +func (m *TreeReport) MarshalVT() (dAtA []byte, err error) { + if m == nil { + return nil, nil + } + size := m.SizeVT() + dAtA = make([]byte, size) + n, err := m.MarshalToSizedBufferVT(dAtA[:size]) + if err != nil { + return nil, err + } + return dAtA[:n], nil +} + +func (m *TreeReport) MarshalToVT(dAtA []byte) (int, error) { + size := m.SizeVT() + return m.MarshalToSizedBufferVT(dAtA[:size]) +} + +func (m *TreeReport) MarshalToSizedBufferVT(dAtA []byte) (int, error) { + if m == nil { + return 0, nil + } + i := len(dAtA) + _ = i + var l int + _ = l + if m.unknownFields != nil { + i -= len(m.unknownFields) + copy(dAtA[i:], m.unknownFields) + } + if len(m.Tree) > 0 { + i -= len(m.Tree) + copy(dAtA[i:], m.Tree) + i = protohelpers.EncodeVarint(dAtA, i, uint64(len(m.Tree))) + i-- + dAtA[i] = 0x12 + } + if m.Query != nil { + size, err := m.Query.MarshalToSizedBufferVT(dAtA[:i]) + if err != nil { + return 0, err + } + i -= size + i = protohelpers.EncodeVarint(dAtA, i, uint64(size)) + i-- + dAtA[i] = 0xa + } + return len(dAtA) - i, nil +} + +func (m *InvokeOptions) SizeVT() (n int) { + if m == nil { + return 0 + } + var l int + _ = l + n += len(m.unknownFields) + return n +} + +func (m *InvokeRequest) SizeVT() (n int) { + if m == nil { + return 0 + } + var l int + _ = l + if len(m.Tenant) > 0 { + for _, s := range m.Tenant { + l = len(s) + n += 1 + l + protohelpers.SizeOfVarint(uint64(l)) + } + } + if m.StartTime != 0 { + n += 1 + protohelpers.SizeOfVarint(uint64(m.StartTime)) + } + if m.EndTime != 0 { + n += 1 + protohelpers.SizeOfVarint(uint64(m.EndTime)) + } + l = len(m.LabelSelector) + if l > 0 { + n += 1 + l + protohelpers.SizeOfVarint(uint64(l)) + } + if len(m.Query) > 0 { + for _, e := range m.Query { + l = e.SizeVT() + n += 1 + l + protohelpers.SizeOfVarint(uint64(l)) + } + } + if m.QueryPlan != nil { + l = m.QueryPlan.SizeVT() + n += 1 + l + protohelpers.SizeOfVarint(uint64(l)) + } + if m.Options != nil { + l = m.Options.SizeVT() + n += 1 + l + protohelpers.SizeOfVarint(uint64(l)) + } + n += len(m.unknownFields) + return n +} + +func (m *QueryPlan) SizeVT() (n int) { + if m == nil { + return 0 + } + var l int + _ = l + if len(m.Graph) > 0 { + l = 0 + for _, e := range m.Graph { + l += protohelpers.SizeOfVarint(uint64(e)) + } + n += 1 + protohelpers.SizeOfVarint(uint64(l)) + l + } + if len(m.Blocks) > 0 { + for _, e := range m.Blocks { + if size, ok := interface{}(e).(interface { + SizeVT() int + }); ok { + l = size.SizeVT() + } else { + l = proto.Size(e) + } + n += 1 + l + protohelpers.SizeOfVarint(uint64(l)) + } + } + n += len(m.unknownFields) + return n +} + +func (m *Query) SizeVT() (n int) { + if m == nil { + return 0 + } + var l int + _ = l + if m.QueryType != 0 { + n += 1 + protohelpers.SizeOfVarint(uint64(m.QueryType)) + } + if m.LabelNames != nil { + l = m.LabelNames.SizeVT() + n += 1 + l + protohelpers.SizeOfVarint(uint64(l)) + } + if m.LabelValues != nil { + l = m.LabelValues.SizeVT() + n += 1 + l + protohelpers.SizeOfVarint(uint64(l)) + } + if m.SeriesLabels != nil { + l = m.SeriesLabels.SizeVT() + n += 1 + l + protohelpers.SizeOfVarint(uint64(l)) + } + if m.TimeSeries != nil { + l = m.TimeSeries.SizeVT() + n += 1 + l + protohelpers.SizeOfVarint(uint64(l)) + } + if m.Tree != nil { + l = m.Tree.SizeVT() + n += 1 + l + protohelpers.SizeOfVarint(uint64(l)) + } + n += len(m.unknownFields) + return n +} + +func (m *InvokeResponse) SizeVT() (n int) { + if m == nil { + return 0 + } + var l int + _ = l + if len(m.Reports) > 0 { + for _, e := range m.Reports { + l = e.SizeVT() + n += 1 + l + protohelpers.SizeOfVarint(uint64(l)) + } + } + if m.Diagnostics != nil { + l = m.Diagnostics.SizeVT() + n += 1 + l + protohelpers.SizeOfVarint(uint64(l)) + } + n += len(m.unknownFields) + return n +} + +func (m *Diagnostics) SizeVT() (n int) { + if m == nil { + return 0 + } + var l int + _ = l + n += len(m.unknownFields) + return n +} + +func (m *Report) SizeVT() (n int) { + if m == nil { + return 0 + } + var l int + _ = l + if m.ReportType != 0 { + n += 1 + protohelpers.SizeOfVarint(uint64(m.ReportType)) + } + if m.LabelNames != nil { + l = m.LabelNames.SizeVT() + n += 1 + l + protohelpers.SizeOfVarint(uint64(l)) + } + if m.LabelValues != nil { + l = m.LabelValues.SizeVT() + n += 1 + l + protohelpers.SizeOfVarint(uint64(l)) + } + if m.SeriesLabels != nil { + l = m.SeriesLabels.SizeVT() + n += 1 + l + protohelpers.SizeOfVarint(uint64(l)) + } + if m.TimeSeries != nil { + l = m.TimeSeries.SizeVT() + n += 1 + l + protohelpers.SizeOfVarint(uint64(l)) + } + if m.Tree != nil { + l = m.Tree.SizeVT() + n += 1 + l + protohelpers.SizeOfVarint(uint64(l)) + } + n += len(m.unknownFields) + return n +} + +func (m *LabelNamesQuery) SizeVT() (n int) { + if m == nil { + return 0 + } + var l int + _ = l + n += len(m.unknownFields) + return n +} + +func (m *LabelNamesReport) SizeVT() (n int) { + if m == nil { + return 0 + } + var l int + _ = l + if m.Query != nil { + l = m.Query.SizeVT() + n += 1 + l + protohelpers.SizeOfVarint(uint64(l)) + } + if len(m.LabelNames) > 0 { + for _, s := range m.LabelNames { + l = len(s) + n += 1 + l + protohelpers.SizeOfVarint(uint64(l)) + } + } + n += len(m.unknownFields) + return n +} + +func (m *LabelValuesQuery) SizeVT() (n int) { + if m == nil { + return 0 + } + var l int + _ = l + l = len(m.LabelName) + if l > 0 { + n += 1 + l + protohelpers.SizeOfVarint(uint64(l)) + } + n += len(m.unknownFields) + return n +} + +func (m *LabelValuesReport) SizeVT() (n int) { + if m == nil { + return 0 + } + var l int + _ = l + if m.Query != nil { + l = m.Query.SizeVT() + n += 1 + l + protohelpers.SizeOfVarint(uint64(l)) + } + if len(m.LabelValues) > 0 { + for _, s := range m.LabelValues { + l = len(s) + n += 1 + l + protohelpers.SizeOfVarint(uint64(l)) + } + } + n += len(m.unknownFields) + return n +} + +func (m *SeriesLabelsQuery) SizeVT() (n int) { + if m == nil { + return 0 + } + var l int + _ = l + if len(m.LabelNames) > 0 { + for _, s := range m.LabelNames { + l = len(s) + n += 1 + l + protohelpers.SizeOfVarint(uint64(l)) + } + } + n += len(m.unknownFields) + return n +} + +func (m *SeriesLabelsReport) SizeVT() (n int) { + if m == nil { + return 0 + } + var l int + _ = l + if m.Query != nil { + l = m.Query.SizeVT() + n += 1 + l + protohelpers.SizeOfVarint(uint64(l)) + } + if len(m.SeriesLabels) > 0 { + for _, e := range m.SeriesLabels { + if size, ok := interface{}(e).(interface { + SizeVT() int + }); ok { + l = size.SizeVT() + } else { + l = proto.Size(e) + } + n += 1 + l + protohelpers.SizeOfVarint(uint64(l)) + } + } + n += len(m.unknownFields) + return n +} + +func (m *TimeSeriesQuery) SizeVT() (n int) { + if m == nil { + return 0 + } + var l int + _ = l + if m.Step != 0 { + n += 9 + } + if len(m.GroupBy) > 0 { + for _, s := range m.GroupBy { + l = len(s) + n += 1 + l + protohelpers.SizeOfVarint(uint64(l)) + } + } + if m.Aggregation != nil { + n += 1 + protohelpers.SizeOfVarint(uint64(*m.Aggregation)) + } + n += len(m.unknownFields) + return n +} + +func (m *TimeSeriesReport) SizeVT() (n int) { + if m == nil { + return 0 + } + var l int + _ = l + if m.Query != nil { + l = m.Query.SizeVT() + n += 1 + l + protohelpers.SizeOfVarint(uint64(l)) + } + if len(m.TimeSeries) > 0 { + for _, e := range m.TimeSeries { + if size, ok := interface{}(e).(interface { + SizeVT() int + }); ok { + l = size.SizeVT() + } else { + l = proto.Size(e) + } + n += 1 + l + protohelpers.SizeOfVarint(uint64(l)) + } + } + n += len(m.unknownFields) + return n +} + +func (m *TreeQuery) SizeVT() (n int) { + if m == nil { + return 0 + } + var l int + _ = l + if m.MaxNodes != 0 { + n += 1 + protohelpers.SizeOfVarint(uint64(m.MaxNodes)) + } + n += len(m.unknownFields) + return n +} + +func (m *TreeReport) SizeVT() (n int) { + if m == nil { + return 0 + } + var l int + _ = l + if m.Query != nil { + l = m.Query.SizeVT() + n += 1 + l + protohelpers.SizeOfVarint(uint64(l)) + } + l = len(m.Tree) + if l > 0 { + n += 1 + l + protohelpers.SizeOfVarint(uint64(l)) + } + n += len(m.unknownFields) + return n +} + +func (m *InvokeOptions) UnmarshalVT(dAtA []byte) error { + l := len(dAtA) + iNdEx := 0 + for iNdEx < l { + preIndex := iNdEx + var wire uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + wire |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + fieldNum := int32(wire >> 3) + wireType := int(wire & 0x7) + if wireType == 4 { + return fmt.Errorf("proto: InvokeOptions: wiretype end group for non-group") + } + if fieldNum <= 0 { + return fmt.Errorf("proto: InvokeOptions: illegal tag %d (wire type %d)", fieldNum, wire) + } + switch fieldNum { + default: + iNdEx = preIndex + skippy, err := protohelpers.Skip(dAtA[iNdEx:]) + if err != nil { + return err + } + if (skippy < 0) || (iNdEx+skippy) < 0 { + return protohelpers.ErrInvalidLength + } + if (iNdEx + skippy) > l { + return io.ErrUnexpectedEOF + } + m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...) + iNdEx += skippy + } + } + + if iNdEx > l { + return io.ErrUnexpectedEOF + } + return nil +} +func (m *InvokeRequest) UnmarshalVT(dAtA []byte) error { + l := len(dAtA) + iNdEx := 0 + for iNdEx < l { + preIndex := iNdEx + var wire uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + wire |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + fieldNum := int32(wire >> 3) + wireType := int(wire & 0x7) + if wireType == 4 { + return fmt.Errorf("proto: InvokeRequest: wiretype end group for non-group") + } + if fieldNum <= 0 { + return fmt.Errorf("proto: InvokeRequest: illegal tag %d (wire type %d)", fieldNum, wire) + } + switch fieldNum { + case 1: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field Tenant", wireType) + } + var stringLen uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + stringLen |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + intStringLen := int(stringLen) + if intStringLen < 0 { + return protohelpers.ErrInvalidLength + } + postIndex := iNdEx + intStringLen + if postIndex < 0 { + return protohelpers.ErrInvalidLength + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + m.Tenant = append(m.Tenant, string(dAtA[iNdEx:postIndex])) + iNdEx = postIndex + case 2: + if wireType != 0 { + return fmt.Errorf("proto: wrong wireType = %d for field StartTime", wireType) + } + m.StartTime = 0 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + m.StartTime |= int64(b&0x7F) << shift + if b < 0x80 { + break + } + } + case 3: + if wireType != 0 { + return fmt.Errorf("proto: wrong wireType = %d for field EndTime", wireType) + } + m.EndTime = 0 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + m.EndTime |= int64(b&0x7F) << shift + if b < 0x80 { + break + } + } + case 4: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field LabelSelector", wireType) + } + var stringLen uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + stringLen |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + intStringLen := int(stringLen) + if intStringLen < 0 { + return protohelpers.ErrInvalidLength + } + postIndex := iNdEx + intStringLen + if postIndex < 0 { + return protohelpers.ErrInvalidLength + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + m.LabelSelector = string(dAtA[iNdEx:postIndex]) + iNdEx = postIndex + case 5: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field Query", wireType) + } + var msglen int + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + msglen |= int(b&0x7F) << shift + if b < 0x80 { + break + } + } + if msglen < 0 { + return protohelpers.ErrInvalidLength + } + postIndex := iNdEx + msglen + if postIndex < 0 { + return protohelpers.ErrInvalidLength + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + m.Query = append(m.Query, &Query{}) + if err := m.Query[len(m.Query)-1].UnmarshalVT(dAtA[iNdEx:postIndex]); err != nil { + return err + } + iNdEx = postIndex + case 6: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field QueryPlan", wireType) + } + var msglen int + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + msglen |= int(b&0x7F) << shift + if b < 0x80 { + break + } + } + if msglen < 0 { + return protohelpers.ErrInvalidLength + } + postIndex := iNdEx + msglen + if postIndex < 0 { + return protohelpers.ErrInvalidLength + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + if m.QueryPlan == nil { + m.QueryPlan = &QueryPlan{} + } + if err := m.QueryPlan.UnmarshalVT(dAtA[iNdEx:postIndex]); err != nil { + return err + } + iNdEx = postIndex + case 7: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field Options", wireType) + } + var msglen int + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + msglen |= int(b&0x7F) << shift + if b < 0x80 { + break + } + } + if msglen < 0 { + return protohelpers.ErrInvalidLength + } + postIndex := iNdEx + msglen + if postIndex < 0 { + return protohelpers.ErrInvalidLength + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + if m.Options == nil { + m.Options = &InvokeOptions{} + } + if err := m.Options.UnmarshalVT(dAtA[iNdEx:postIndex]); err != nil { + return err + } + iNdEx = postIndex + default: + iNdEx = preIndex + skippy, err := protohelpers.Skip(dAtA[iNdEx:]) + if err != nil { + return err + } + if (skippy < 0) || (iNdEx+skippy) < 0 { + return protohelpers.ErrInvalidLength + } + if (iNdEx + skippy) > l { + return io.ErrUnexpectedEOF + } + m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...) + iNdEx += skippy + } + } + + if iNdEx > l { + return io.ErrUnexpectedEOF + } + return nil +} +func (m *QueryPlan) UnmarshalVT(dAtA []byte) error { + l := len(dAtA) + iNdEx := 0 + for iNdEx < l { + preIndex := iNdEx + var wire uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + wire |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + fieldNum := int32(wire >> 3) + wireType := int(wire & 0x7) + if wireType == 4 { + return fmt.Errorf("proto: QueryPlan: wiretype end group for non-group") + } + if fieldNum <= 0 { + return fmt.Errorf("proto: QueryPlan: illegal tag %d (wire type %d)", fieldNum, wire) + } + switch fieldNum { + case 1: + if wireType == 0 { + var v uint32 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + v |= uint32(b&0x7F) << shift + if b < 0x80 { + break + } + } + m.Graph = append(m.Graph, v) + } else if wireType == 2 { + var packedLen int + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + packedLen |= int(b&0x7F) << shift + if b < 0x80 { + break + } + } + if packedLen < 0 { + return protohelpers.ErrInvalidLength + } + postIndex := iNdEx + packedLen + if postIndex < 0 { + return protohelpers.ErrInvalidLength + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + var elementCount int + var count int + for _, integer := range dAtA[iNdEx:postIndex] { + if integer < 128 { + count++ + } + } + elementCount = count + if elementCount != 0 && len(m.Graph) == 0 { + m.Graph = make([]uint32, 0, elementCount) + } + for iNdEx < postIndex { + var v uint32 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + v |= uint32(b&0x7F) << shift + if b < 0x80 { + break + } + } + m.Graph = append(m.Graph, v) + } + } else { + return fmt.Errorf("proto: wrong wireType = %d for field Graph", wireType) + } + case 2: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field Blocks", wireType) + } + var msglen int + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + msglen |= int(b&0x7F) << shift + if b < 0x80 { + break + } + } + if msglen < 0 { + return protohelpers.ErrInvalidLength + } + postIndex := iNdEx + msglen + if postIndex < 0 { + return protohelpers.ErrInvalidLength + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + m.Blocks = append(m.Blocks, &v1.BlockMeta{}) + if unmarshal, ok := interface{}(m.Blocks[len(m.Blocks)-1]).(interface { + UnmarshalVT([]byte) error + }); ok { + if err := unmarshal.UnmarshalVT(dAtA[iNdEx:postIndex]); err != nil { + return err + } + } else { + if err := proto.Unmarshal(dAtA[iNdEx:postIndex], m.Blocks[len(m.Blocks)-1]); err != nil { + return err + } + } + iNdEx = postIndex + default: + iNdEx = preIndex + skippy, err := protohelpers.Skip(dAtA[iNdEx:]) + if err != nil { + return err + } + if (skippy < 0) || (iNdEx+skippy) < 0 { + return protohelpers.ErrInvalidLength + } + if (iNdEx + skippy) > l { + return io.ErrUnexpectedEOF + } + m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...) + iNdEx += skippy + } + } + + if iNdEx > l { + return io.ErrUnexpectedEOF + } + return nil +} +func (m *Query) UnmarshalVT(dAtA []byte) error { + l := len(dAtA) + iNdEx := 0 + for iNdEx < l { + preIndex := iNdEx + var wire uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + wire |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + fieldNum := int32(wire >> 3) + wireType := int(wire & 0x7) + if wireType == 4 { + return fmt.Errorf("proto: Query: wiretype end group for non-group") + } + if fieldNum <= 0 { + return fmt.Errorf("proto: Query: illegal tag %d (wire type %d)", fieldNum, wire) + } + switch fieldNum { + case 1: + if wireType != 0 { + return fmt.Errorf("proto: wrong wireType = %d for field QueryType", wireType) + } + m.QueryType = 0 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + m.QueryType |= QueryType(b&0x7F) << shift + if b < 0x80 { + break + } + } + case 2: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field LabelNames", wireType) + } + var msglen int + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + msglen |= int(b&0x7F) << shift + if b < 0x80 { + break + } + } + if msglen < 0 { + return protohelpers.ErrInvalidLength + } + postIndex := iNdEx + msglen + if postIndex < 0 { + return protohelpers.ErrInvalidLength + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + if m.LabelNames == nil { + m.LabelNames = &LabelNamesQuery{} + } + if err := m.LabelNames.UnmarshalVT(dAtA[iNdEx:postIndex]); err != nil { + return err + } + iNdEx = postIndex + case 3: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field LabelValues", wireType) + } + var msglen int + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + msglen |= int(b&0x7F) << shift + if b < 0x80 { + break + } + } + if msglen < 0 { + return protohelpers.ErrInvalidLength + } + postIndex := iNdEx + msglen + if postIndex < 0 { + return protohelpers.ErrInvalidLength + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + if m.LabelValues == nil { + m.LabelValues = &LabelValuesQuery{} + } + if err := m.LabelValues.UnmarshalVT(dAtA[iNdEx:postIndex]); err != nil { + return err + } + iNdEx = postIndex + case 4: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field SeriesLabels", wireType) + } + var msglen int + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + msglen |= int(b&0x7F) << shift + if b < 0x80 { + break + } + } + if msglen < 0 { + return protohelpers.ErrInvalidLength + } + postIndex := iNdEx + msglen + if postIndex < 0 { + return protohelpers.ErrInvalidLength + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + if m.SeriesLabels == nil { + m.SeriesLabels = &SeriesLabelsQuery{} + } + if err := m.SeriesLabels.UnmarshalVT(dAtA[iNdEx:postIndex]); err != nil { + return err + } + iNdEx = postIndex + case 5: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field TimeSeries", wireType) + } + var msglen int + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + msglen |= int(b&0x7F) << shift + if b < 0x80 { + break + } + } + if msglen < 0 { + return protohelpers.ErrInvalidLength + } + postIndex := iNdEx + msglen + if postIndex < 0 { + return protohelpers.ErrInvalidLength + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + if m.TimeSeries == nil { + m.TimeSeries = &TimeSeriesQuery{} + } + if err := m.TimeSeries.UnmarshalVT(dAtA[iNdEx:postIndex]); err != nil { + return err + } + iNdEx = postIndex + case 6: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field Tree", wireType) + } + var msglen int + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + msglen |= int(b&0x7F) << shift + if b < 0x80 { + break + } + } + if msglen < 0 { + return protohelpers.ErrInvalidLength + } + postIndex := iNdEx + msglen + if postIndex < 0 { + return protohelpers.ErrInvalidLength + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + if m.Tree == nil { + m.Tree = &TreeQuery{} + } + if err := m.Tree.UnmarshalVT(dAtA[iNdEx:postIndex]); err != nil { + return err + } + iNdEx = postIndex + default: + iNdEx = preIndex + skippy, err := protohelpers.Skip(dAtA[iNdEx:]) + if err != nil { + return err + } + if (skippy < 0) || (iNdEx+skippy) < 0 { + return protohelpers.ErrInvalidLength + } + if (iNdEx + skippy) > l { + return io.ErrUnexpectedEOF + } + m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...) + iNdEx += skippy + } + } + + if iNdEx > l { + return io.ErrUnexpectedEOF + } + return nil +} +func (m *InvokeResponse) UnmarshalVT(dAtA []byte) error { + l := len(dAtA) + iNdEx := 0 + for iNdEx < l { + preIndex := iNdEx + var wire uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + wire |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + fieldNum := int32(wire >> 3) + wireType := int(wire & 0x7) + if wireType == 4 { + return fmt.Errorf("proto: InvokeResponse: wiretype end group for non-group") + } + if fieldNum <= 0 { + return fmt.Errorf("proto: InvokeResponse: illegal tag %d (wire type %d)", fieldNum, wire) + } + switch fieldNum { + case 1: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field Reports", wireType) + } + var msglen int + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + msglen |= int(b&0x7F) << shift + if b < 0x80 { + break + } + } + if msglen < 0 { + return protohelpers.ErrInvalidLength + } + postIndex := iNdEx + msglen + if postIndex < 0 { + return protohelpers.ErrInvalidLength + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + m.Reports = append(m.Reports, &Report{}) + if err := m.Reports[len(m.Reports)-1].UnmarshalVT(dAtA[iNdEx:postIndex]); err != nil { + return err + } + iNdEx = postIndex + case 2: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field Diagnostics", wireType) + } + var msglen int + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + msglen |= int(b&0x7F) << shift + if b < 0x80 { + break + } + } + if msglen < 0 { + return protohelpers.ErrInvalidLength + } + postIndex := iNdEx + msglen + if postIndex < 0 { + return protohelpers.ErrInvalidLength + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + if m.Diagnostics == nil { + m.Diagnostics = &Diagnostics{} + } + if err := m.Diagnostics.UnmarshalVT(dAtA[iNdEx:postIndex]); err != nil { + return err + } + iNdEx = postIndex + default: + iNdEx = preIndex + skippy, err := protohelpers.Skip(dAtA[iNdEx:]) + if err != nil { + return err + } + if (skippy < 0) || (iNdEx+skippy) < 0 { + return protohelpers.ErrInvalidLength + } + if (iNdEx + skippy) > l { + return io.ErrUnexpectedEOF + } + m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...) + iNdEx += skippy + } + } + + if iNdEx > l { + return io.ErrUnexpectedEOF + } + return nil +} +func (m *Diagnostics) UnmarshalVT(dAtA []byte) error { + l := len(dAtA) + iNdEx := 0 + for iNdEx < l { + preIndex := iNdEx + var wire uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + wire |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + fieldNum := int32(wire >> 3) + wireType := int(wire & 0x7) + if wireType == 4 { + return fmt.Errorf("proto: Diagnostics: wiretype end group for non-group") + } + if fieldNum <= 0 { + return fmt.Errorf("proto: Diagnostics: illegal tag %d (wire type %d)", fieldNum, wire) + } + switch fieldNum { + default: + iNdEx = preIndex + skippy, err := protohelpers.Skip(dAtA[iNdEx:]) + if err != nil { + return err + } + if (skippy < 0) || (iNdEx+skippy) < 0 { + return protohelpers.ErrInvalidLength + } + if (iNdEx + skippy) > l { + return io.ErrUnexpectedEOF + } + m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...) + iNdEx += skippy + } + } + + if iNdEx > l { + return io.ErrUnexpectedEOF + } + return nil +} +func (m *Report) UnmarshalVT(dAtA []byte) error { + l := len(dAtA) + iNdEx := 0 + for iNdEx < l { + preIndex := iNdEx + var wire uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + wire |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + fieldNum := int32(wire >> 3) + wireType := int(wire & 0x7) + if wireType == 4 { + return fmt.Errorf("proto: Report: wiretype end group for non-group") + } + if fieldNum <= 0 { + return fmt.Errorf("proto: Report: illegal tag %d (wire type %d)", fieldNum, wire) + } + switch fieldNum { + case 1: + if wireType != 0 { + return fmt.Errorf("proto: wrong wireType = %d for field ReportType", wireType) + } + m.ReportType = 0 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + m.ReportType |= ReportType(b&0x7F) << shift + if b < 0x80 { + break + } + } + case 2: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field LabelNames", wireType) + } + var msglen int + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + msglen |= int(b&0x7F) << shift + if b < 0x80 { + break + } + } + if msglen < 0 { + return protohelpers.ErrInvalidLength + } + postIndex := iNdEx + msglen + if postIndex < 0 { + return protohelpers.ErrInvalidLength + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + if m.LabelNames == nil { + m.LabelNames = &LabelNamesReport{} + } + if err := m.LabelNames.UnmarshalVT(dAtA[iNdEx:postIndex]); err != nil { + return err + } + iNdEx = postIndex + case 3: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field LabelValues", wireType) + } + var msglen int + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + msglen |= int(b&0x7F) << shift + if b < 0x80 { + break + } + } + if msglen < 0 { + return protohelpers.ErrInvalidLength + } + postIndex := iNdEx + msglen + if postIndex < 0 { + return protohelpers.ErrInvalidLength + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + if m.LabelValues == nil { + m.LabelValues = &LabelValuesReport{} + } + if err := m.LabelValues.UnmarshalVT(dAtA[iNdEx:postIndex]); err != nil { + return err + } + iNdEx = postIndex + case 4: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field SeriesLabels", wireType) + } + var msglen int + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + msglen |= int(b&0x7F) << shift + if b < 0x80 { + break + } + } + if msglen < 0 { + return protohelpers.ErrInvalidLength + } + postIndex := iNdEx + msglen + if postIndex < 0 { + return protohelpers.ErrInvalidLength + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + if m.SeriesLabels == nil { + m.SeriesLabels = &SeriesLabelsReport{} + } + if err := m.SeriesLabels.UnmarshalVT(dAtA[iNdEx:postIndex]); err != nil { + return err + } + iNdEx = postIndex + case 5: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field TimeSeries", wireType) + } + var msglen int + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + msglen |= int(b&0x7F) << shift + if b < 0x80 { + break + } + } + if msglen < 0 { + return protohelpers.ErrInvalidLength + } + postIndex := iNdEx + msglen + if postIndex < 0 { + return protohelpers.ErrInvalidLength + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + if m.TimeSeries == nil { + m.TimeSeries = &TimeSeriesReport{} + } + if err := m.TimeSeries.UnmarshalVT(dAtA[iNdEx:postIndex]); err != nil { + return err + } + iNdEx = postIndex + case 6: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field Tree", wireType) + } + var msglen int + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + msglen |= int(b&0x7F) << shift + if b < 0x80 { + break + } + } + if msglen < 0 { + return protohelpers.ErrInvalidLength + } + postIndex := iNdEx + msglen + if postIndex < 0 { + return protohelpers.ErrInvalidLength + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + if m.Tree == nil { + m.Tree = &TreeReport{} + } + if err := m.Tree.UnmarshalVT(dAtA[iNdEx:postIndex]); err != nil { + return err + } + iNdEx = postIndex + default: + iNdEx = preIndex + skippy, err := protohelpers.Skip(dAtA[iNdEx:]) + if err != nil { + return err + } + if (skippy < 0) || (iNdEx+skippy) < 0 { + return protohelpers.ErrInvalidLength + } + if (iNdEx + skippy) > l { + return io.ErrUnexpectedEOF + } + m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...) + iNdEx += skippy + } + } + + if iNdEx > l { + return io.ErrUnexpectedEOF + } + return nil +} +func (m *LabelNamesQuery) UnmarshalVT(dAtA []byte) error { + l := len(dAtA) + iNdEx := 0 + for iNdEx < l { + preIndex := iNdEx + var wire uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + wire |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + fieldNum := int32(wire >> 3) + wireType := int(wire & 0x7) + if wireType == 4 { + return fmt.Errorf("proto: LabelNamesQuery: wiretype end group for non-group") + } + if fieldNum <= 0 { + return fmt.Errorf("proto: LabelNamesQuery: illegal tag %d (wire type %d)", fieldNum, wire) + } + switch fieldNum { + default: + iNdEx = preIndex + skippy, err := protohelpers.Skip(dAtA[iNdEx:]) + if err != nil { + return err + } + if (skippy < 0) || (iNdEx+skippy) < 0 { + return protohelpers.ErrInvalidLength + } + if (iNdEx + skippy) > l { + return io.ErrUnexpectedEOF + } + m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...) + iNdEx += skippy + } + } + + if iNdEx > l { + return io.ErrUnexpectedEOF + } + return nil +} +func (m *LabelNamesReport) UnmarshalVT(dAtA []byte) error { + l := len(dAtA) + iNdEx := 0 + for iNdEx < l { + preIndex := iNdEx + var wire uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + wire |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + fieldNum := int32(wire >> 3) + wireType := int(wire & 0x7) + if wireType == 4 { + return fmt.Errorf("proto: LabelNamesReport: wiretype end group for non-group") + } + if fieldNum <= 0 { + return fmt.Errorf("proto: LabelNamesReport: illegal tag %d (wire type %d)", fieldNum, wire) + } + switch fieldNum { + case 1: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field Query", wireType) + } + var msglen int + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + msglen |= int(b&0x7F) << shift + if b < 0x80 { + break + } + } + if msglen < 0 { + return protohelpers.ErrInvalidLength + } + postIndex := iNdEx + msglen + if postIndex < 0 { + return protohelpers.ErrInvalidLength + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + if m.Query == nil { + m.Query = &LabelNamesQuery{} + } + if err := m.Query.UnmarshalVT(dAtA[iNdEx:postIndex]); err != nil { + return err + } + iNdEx = postIndex + case 2: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field LabelNames", wireType) + } + var stringLen uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + stringLen |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + intStringLen := int(stringLen) + if intStringLen < 0 { + return protohelpers.ErrInvalidLength + } + postIndex := iNdEx + intStringLen + if postIndex < 0 { + return protohelpers.ErrInvalidLength + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + m.LabelNames = append(m.LabelNames, string(dAtA[iNdEx:postIndex])) + iNdEx = postIndex + default: + iNdEx = preIndex + skippy, err := protohelpers.Skip(dAtA[iNdEx:]) + if err != nil { + return err + } + if (skippy < 0) || (iNdEx+skippy) < 0 { + return protohelpers.ErrInvalidLength + } + if (iNdEx + skippy) > l { + return io.ErrUnexpectedEOF + } + m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...) + iNdEx += skippy + } + } + + if iNdEx > l { + return io.ErrUnexpectedEOF + } + return nil +} +func (m *LabelValuesQuery) UnmarshalVT(dAtA []byte) error { + l := len(dAtA) + iNdEx := 0 + for iNdEx < l { + preIndex := iNdEx + var wire uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + wire |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + fieldNum := int32(wire >> 3) + wireType := int(wire & 0x7) + if wireType == 4 { + return fmt.Errorf("proto: LabelValuesQuery: wiretype end group for non-group") + } + if fieldNum <= 0 { + return fmt.Errorf("proto: LabelValuesQuery: illegal tag %d (wire type %d)", fieldNum, wire) + } + switch fieldNum { + case 1: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field LabelName", wireType) + } + var stringLen uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + stringLen |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + intStringLen := int(stringLen) + if intStringLen < 0 { + return protohelpers.ErrInvalidLength + } + postIndex := iNdEx + intStringLen + if postIndex < 0 { + return protohelpers.ErrInvalidLength + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + m.LabelName = string(dAtA[iNdEx:postIndex]) + iNdEx = postIndex + default: + iNdEx = preIndex + skippy, err := protohelpers.Skip(dAtA[iNdEx:]) + if err != nil { + return err + } + if (skippy < 0) || (iNdEx+skippy) < 0 { + return protohelpers.ErrInvalidLength + } + if (iNdEx + skippy) > l { + return io.ErrUnexpectedEOF + } + m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...) + iNdEx += skippy + } + } + + if iNdEx > l { + return io.ErrUnexpectedEOF + } + return nil +} +func (m *LabelValuesReport) UnmarshalVT(dAtA []byte) error { + l := len(dAtA) + iNdEx := 0 + for iNdEx < l { + preIndex := iNdEx + var wire uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + wire |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + fieldNum := int32(wire >> 3) + wireType := int(wire & 0x7) + if wireType == 4 { + return fmt.Errorf("proto: LabelValuesReport: wiretype end group for non-group") + } + if fieldNum <= 0 { + return fmt.Errorf("proto: LabelValuesReport: illegal tag %d (wire type %d)", fieldNum, wire) + } + switch fieldNum { + case 1: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field Query", wireType) + } + var msglen int + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + msglen |= int(b&0x7F) << shift + if b < 0x80 { + break + } + } + if msglen < 0 { + return protohelpers.ErrInvalidLength + } + postIndex := iNdEx + msglen + if postIndex < 0 { + return protohelpers.ErrInvalidLength + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + if m.Query == nil { + m.Query = &LabelValuesQuery{} + } + if err := m.Query.UnmarshalVT(dAtA[iNdEx:postIndex]); err != nil { + return err + } + iNdEx = postIndex + case 2: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field LabelValues", wireType) + } + var stringLen uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + stringLen |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + intStringLen := int(stringLen) + if intStringLen < 0 { + return protohelpers.ErrInvalidLength + } + postIndex := iNdEx + intStringLen + if postIndex < 0 { + return protohelpers.ErrInvalidLength + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + m.LabelValues = append(m.LabelValues, string(dAtA[iNdEx:postIndex])) + iNdEx = postIndex + default: + iNdEx = preIndex + skippy, err := protohelpers.Skip(dAtA[iNdEx:]) + if err != nil { + return err + } + if (skippy < 0) || (iNdEx+skippy) < 0 { + return protohelpers.ErrInvalidLength + } + if (iNdEx + skippy) > l { + return io.ErrUnexpectedEOF + } + m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...) + iNdEx += skippy + } + } + + if iNdEx > l { + return io.ErrUnexpectedEOF + } + return nil +} +func (m *SeriesLabelsQuery) UnmarshalVT(dAtA []byte) error { + l := len(dAtA) + iNdEx := 0 + for iNdEx < l { + preIndex := iNdEx + var wire uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + wire |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + fieldNum := int32(wire >> 3) + wireType := int(wire & 0x7) + if wireType == 4 { + return fmt.Errorf("proto: SeriesLabelsQuery: wiretype end group for non-group") + } + if fieldNum <= 0 { + return fmt.Errorf("proto: SeriesLabelsQuery: illegal tag %d (wire type %d)", fieldNum, wire) + } + switch fieldNum { + case 1: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field LabelNames", wireType) + } + var stringLen uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + stringLen |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + intStringLen := int(stringLen) + if intStringLen < 0 { + return protohelpers.ErrInvalidLength + } + postIndex := iNdEx + intStringLen + if postIndex < 0 { + return protohelpers.ErrInvalidLength + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + m.LabelNames = append(m.LabelNames, string(dAtA[iNdEx:postIndex])) + iNdEx = postIndex + default: + iNdEx = preIndex + skippy, err := protohelpers.Skip(dAtA[iNdEx:]) + if err != nil { + return err + } + if (skippy < 0) || (iNdEx+skippy) < 0 { + return protohelpers.ErrInvalidLength + } + if (iNdEx + skippy) > l { + return io.ErrUnexpectedEOF + } + m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...) + iNdEx += skippy + } + } + + if iNdEx > l { + return io.ErrUnexpectedEOF + } + return nil +} +func (m *SeriesLabelsReport) UnmarshalVT(dAtA []byte) error { + l := len(dAtA) + iNdEx := 0 + for iNdEx < l { + preIndex := iNdEx + var wire uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + wire |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + fieldNum := int32(wire >> 3) + wireType := int(wire & 0x7) + if wireType == 4 { + return fmt.Errorf("proto: SeriesLabelsReport: wiretype end group for non-group") + } + if fieldNum <= 0 { + return fmt.Errorf("proto: SeriesLabelsReport: illegal tag %d (wire type %d)", fieldNum, wire) + } + switch fieldNum { + case 1: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field Query", wireType) + } + var msglen int + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + msglen |= int(b&0x7F) << shift + if b < 0x80 { + break + } + } + if msglen < 0 { + return protohelpers.ErrInvalidLength + } + postIndex := iNdEx + msglen + if postIndex < 0 { + return protohelpers.ErrInvalidLength + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + if m.Query == nil { + m.Query = &SeriesLabelsQuery{} + } + if err := m.Query.UnmarshalVT(dAtA[iNdEx:postIndex]); err != nil { + return err + } + iNdEx = postIndex + case 2: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field SeriesLabels", wireType) + } + var msglen int + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + msglen |= int(b&0x7F) << shift + if b < 0x80 { + break + } + } + if msglen < 0 { + return protohelpers.ErrInvalidLength + } + postIndex := iNdEx + msglen + if postIndex < 0 { + return protohelpers.ErrInvalidLength + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + m.SeriesLabels = append(m.SeriesLabels, &v11.Labels{}) + if unmarshal, ok := interface{}(m.SeriesLabels[len(m.SeriesLabels)-1]).(interface { + UnmarshalVT([]byte) error + }); ok { + if err := unmarshal.UnmarshalVT(dAtA[iNdEx:postIndex]); err != nil { + return err + } + } else { + if err := proto.Unmarshal(dAtA[iNdEx:postIndex], m.SeriesLabels[len(m.SeriesLabels)-1]); err != nil { + return err + } + } + iNdEx = postIndex + default: + iNdEx = preIndex + skippy, err := protohelpers.Skip(dAtA[iNdEx:]) + if err != nil { + return err + } + if (skippy < 0) || (iNdEx+skippy) < 0 { + return protohelpers.ErrInvalidLength + } + if (iNdEx + skippy) > l { + return io.ErrUnexpectedEOF + } + m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...) + iNdEx += skippy + } + } + + if iNdEx > l { + return io.ErrUnexpectedEOF + } + return nil +} +func (m *TimeSeriesQuery) UnmarshalVT(dAtA []byte) error { + l := len(dAtA) + iNdEx := 0 + for iNdEx < l { + preIndex := iNdEx + var wire uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + wire |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + fieldNum := int32(wire >> 3) + wireType := int(wire & 0x7) + if wireType == 4 { + return fmt.Errorf("proto: TimeSeriesQuery: wiretype end group for non-group") + } + if fieldNum <= 0 { + return fmt.Errorf("proto: TimeSeriesQuery: illegal tag %d (wire type %d)", fieldNum, wire) + } + switch fieldNum { + case 1: + if wireType != 1 { + return fmt.Errorf("proto: wrong wireType = %d for field Step", wireType) + } + var v uint64 + if (iNdEx + 8) > l { + return io.ErrUnexpectedEOF + } + v = uint64(binary.LittleEndian.Uint64(dAtA[iNdEx:])) + iNdEx += 8 + m.Step = float64(math.Float64frombits(v)) + case 2: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field GroupBy", wireType) + } + var stringLen uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + stringLen |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + intStringLen := int(stringLen) + if intStringLen < 0 { + return protohelpers.ErrInvalidLength + } + postIndex := iNdEx + intStringLen + if postIndex < 0 { + return protohelpers.ErrInvalidLength + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + m.GroupBy = append(m.GroupBy, string(dAtA[iNdEx:postIndex])) + iNdEx = postIndex + case 3: + if wireType != 0 { + return fmt.Errorf("proto: wrong wireType = %d for field Aggregation", wireType) + } + var v v11.TimeSeriesAggregationType + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + v |= v11.TimeSeriesAggregationType(b&0x7F) << shift + if b < 0x80 { + break + } + } + m.Aggregation = &v + default: + iNdEx = preIndex + skippy, err := protohelpers.Skip(dAtA[iNdEx:]) + if err != nil { + return err + } + if (skippy < 0) || (iNdEx+skippy) < 0 { + return protohelpers.ErrInvalidLength + } + if (iNdEx + skippy) > l { + return io.ErrUnexpectedEOF + } + m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...) + iNdEx += skippy + } + } + + if iNdEx > l { + return io.ErrUnexpectedEOF + } + return nil +} +func (m *TimeSeriesReport) UnmarshalVT(dAtA []byte) error { + l := len(dAtA) + iNdEx := 0 + for iNdEx < l { + preIndex := iNdEx + var wire uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + wire |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + fieldNum := int32(wire >> 3) + wireType := int(wire & 0x7) + if wireType == 4 { + return fmt.Errorf("proto: TimeSeriesReport: wiretype end group for non-group") + } + if fieldNum <= 0 { + return fmt.Errorf("proto: TimeSeriesReport: illegal tag %d (wire type %d)", fieldNum, wire) + } + switch fieldNum { + case 1: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field Query", wireType) + } + var msglen int + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + msglen |= int(b&0x7F) << shift + if b < 0x80 { + break + } + } + if msglen < 0 { + return protohelpers.ErrInvalidLength + } + postIndex := iNdEx + msglen + if postIndex < 0 { + return protohelpers.ErrInvalidLength + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + if m.Query == nil { + m.Query = &TimeSeriesQuery{} + } + if err := m.Query.UnmarshalVT(dAtA[iNdEx:postIndex]); err != nil { + return err + } + iNdEx = postIndex + case 2: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field TimeSeries", wireType) + } + var msglen int + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + msglen |= int(b&0x7F) << shift + if b < 0x80 { + break + } + } + if msglen < 0 { + return protohelpers.ErrInvalidLength + } + postIndex := iNdEx + msglen + if postIndex < 0 { + return protohelpers.ErrInvalidLength + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + m.TimeSeries = append(m.TimeSeries, &v11.Series{}) + if unmarshal, ok := interface{}(m.TimeSeries[len(m.TimeSeries)-1]).(interface { + UnmarshalVT([]byte) error + }); ok { + if err := unmarshal.UnmarshalVT(dAtA[iNdEx:postIndex]); err != nil { + return err + } + } else { + if err := proto.Unmarshal(dAtA[iNdEx:postIndex], m.TimeSeries[len(m.TimeSeries)-1]); err != nil { + return err + } + } + iNdEx = postIndex + default: + iNdEx = preIndex + skippy, err := protohelpers.Skip(dAtA[iNdEx:]) + if err != nil { + return err + } + if (skippy < 0) || (iNdEx+skippy) < 0 { + return protohelpers.ErrInvalidLength + } + if (iNdEx + skippy) > l { + return io.ErrUnexpectedEOF + } + m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...) + iNdEx += skippy + } + } + + if iNdEx > l { + return io.ErrUnexpectedEOF + } + return nil +} +func (m *TreeQuery) UnmarshalVT(dAtA []byte) error { + l := len(dAtA) + iNdEx := 0 + for iNdEx < l { + preIndex := iNdEx + var wire uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + wire |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + fieldNum := int32(wire >> 3) + wireType := int(wire & 0x7) + if wireType == 4 { + return fmt.Errorf("proto: TreeQuery: wiretype end group for non-group") + } + if fieldNum <= 0 { + return fmt.Errorf("proto: TreeQuery: illegal tag %d (wire type %d)", fieldNum, wire) + } + switch fieldNum { + case 1: + if wireType != 0 { + return fmt.Errorf("proto: wrong wireType = %d for field MaxNodes", wireType) + } + m.MaxNodes = 0 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + m.MaxNodes |= int64(b&0x7F) << shift + if b < 0x80 { + break + } + } + default: + iNdEx = preIndex + skippy, err := protohelpers.Skip(dAtA[iNdEx:]) + if err != nil { + return err + } + if (skippy < 0) || (iNdEx+skippy) < 0 { + return protohelpers.ErrInvalidLength + } + if (iNdEx + skippy) > l { + return io.ErrUnexpectedEOF + } + m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...) + iNdEx += skippy + } + } + + if iNdEx > l { + return io.ErrUnexpectedEOF + } + return nil +} +func (m *TreeReport) UnmarshalVT(dAtA []byte) error { + l := len(dAtA) + iNdEx := 0 + for iNdEx < l { + preIndex := iNdEx + var wire uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + wire |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + fieldNum := int32(wire >> 3) + wireType := int(wire & 0x7) + if wireType == 4 { + return fmt.Errorf("proto: TreeReport: wiretype end group for non-group") + } + if fieldNum <= 0 { + return fmt.Errorf("proto: TreeReport: illegal tag %d (wire type %d)", fieldNum, wire) + } + switch fieldNum { + case 1: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field Query", wireType) + } + var msglen int + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + msglen |= int(b&0x7F) << shift + if b < 0x80 { + break + } + } + if msglen < 0 { + return protohelpers.ErrInvalidLength + } + postIndex := iNdEx + msglen + if postIndex < 0 { + return protohelpers.ErrInvalidLength + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + if m.Query == nil { + m.Query = &TreeQuery{} + } + if err := m.Query.UnmarshalVT(dAtA[iNdEx:postIndex]); err != nil { + return err + } + iNdEx = postIndex + case 2: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field Tree", wireType) + } + var byteLen int + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + byteLen |= int(b&0x7F) << shift + if b < 0x80 { + break + } + } + if byteLen < 0 { + return protohelpers.ErrInvalidLength + } + postIndex := iNdEx + byteLen + if postIndex < 0 { + return protohelpers.ErrInvalidLength + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + m.Tree = append(m.Tree[:0], dAtA[iNdEx:postIndex]...) + if m.Tree == nil { + m.Tree = []byte{} + } + iNdEx = postIndex + default: + iNdEx = preIndex + skippy, err := protohelpers.Skip(dAtA[iNdEx:]) + if err != nil { + return err + } + if (skippy < 0) || (iNdEx+skippy) < 0 { + return protohelpers.ErrInvalidLength + } + if (iNdEx + skippy) > l { + return io.ErrUnexpectedEOF + } + m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...) + iNdEx += skippy + } + } + + if iNdEx > l { + return io.ErrUnexpectedEOF + } + return nil +} diff --git a/api/gen/proto/go/querybackend/v1/querybackendv1connect/querybackend.connect.go b/api/gen/proto/go/querybackend/v1/querybackendv1connect/querybackend.connect.go new file mode 100644 index 0000000000..637f1d3f58 --- /dev/null +++ b/api/gen/proto/go/querybackend/v1/querybackendv1connect/querybackend.connect.go @@ -0,0 +1,114 @@ +// Code generated by protoc-gen-connect-go. DO NOT EDIT. +// +// Source: querybackend/v1/querybackend.proto + +package querybackendv1connect + +import ( + connect "connectrpc.com/connect" + context "context" + errors "errors" + v1 "github.com/grafana/pyroscope/api/gen/proto/go/querybackend/v1" + http "net/http" + strings "strings" +) + +// This is a compile-time assertion to ensure that this generated file and the connect package are +// compatible. If you get a compiler error that this constant is not defined, this code was +// generated with a version of connect newer than the one compiled into your binary. You can fix the +// problem by either regenerating this code with an older version of connect or updating the connect +// version compiled into your binary. +const _ = connect.IsAtLeastVersion1_13_0 + +const ( + // QueryBackendServiceName is the fully-qualified name of the QueryBackendService service. + QueryBackendServiceName = "querybackend.v1.QueryBackendService" +) + +// These constants are the fully-qualified names of the RPCs defined in this package. They're +// exposed at runtime as Spec.Procedure and as the final two segments of the HTTP route. +// +// Note that these are different from the fully-qualified method names used by +// google.golang.org/protobuf/reflect/protoreflect. To convert from these constants to +// reflection-formatted method names, remove the leading slash and convert the remaining slash to a +// period. +const ( + // QueryBackendServiceInvokeProcedure is the fully-qualified name of the QueryBackendService's + // Invoke RPC. + QueryBackendServiceInvokeProcedure = "/querybackend.v1.QueryBackendService/Invoke" +) + +// These variables are the protoreflect.Descriptor objects for the RPCs defined in this package. +var ( + queryBackendServiceServiceDescriptor = v1.File_querybackend_v1_querybackend_proto.Services().ByName("QueryBackendService") + queryBackendServiceInvokeMethodDescriptor = queryBackendServiceServiceDescriptor.Methods().ByName("Invoke") +) + +// QueryBackendServiceClient is a client for the querybackend.v1.QueryBackendService service. +type QueryBackendServiceClient interface { + Invoke(context.Context, *connect.Request[v1.InvokeRequest]) (*connect.Response[v1.InvokeResponse], error) +} + +// NewQueryBackendServiceClient constructs a client for the querybackend.v1.QueryBackendService +// service. By default, it uses the Connect protocol with the binary Protobuf Codec, asks for +// gzipped responses, and sends uncompressed requests. To use the gRPC or gRPC-Web protocols, supply +// the connect.WithGRPC() or connect.WithGRPCWeb() options. +// +// The URL supplied here should be the base URL for the Connect or gRPC server (for example, +// http://api.acme.com or https://acme.com/grpc). +func NewQueryBackendServiceClient(httpClient connect.HTTPClient, baseURL string, opts ...connect.ClientOption) QueryBackendServiceClient { + baseURL = strings.TrimRight(baseURL, "/") + return &queryBackendServiceClient{ + invoke: connect.NewClient[v1.InvokeRequest, v1.InvokeResponse]( + httpClient, + baseURL+QueryBackendServiceInvokeProcedure, + connect.WithSchema(queryBackendServiceInvokeMethodDescriptor), + connect.WithClientOptions(opts...), + ), + } +} + +// queryBackendServiceClient implements QueryBackendServiceClient. +type queryBackendServiceClient struct { + invoke *connect.Client[v1.InvokeRequest, v1.InvokeResponse] +} + +// Invoke calls querybackend.v1.QueryBackendService.Invoke. +func (c *queryBackendServiceClient) Invoke(ctx context.Context, req *connect.Request[v1.InvokeRequest]) (*connect.Response[v1.InvokeResponse], error) { + return c.invoke.CallUnary(ctx, req) +} + +// QueryBackendServiceHandler is an implementation of the querybackend.v1.QueryBackendService +// service. +type QueryBackendServiceHandler interface { + Invoke(context.Context, *connect.Request[v1.InvokeRequest]) (*connect.Response[v1.InvokeResponse], error) +} + +// NewQueryBackendServiceHandler builds an HTTP handler from the service implementation. It returns +// the path on which to mount the handler and the handler itself. +// +// By default, handlers support the Connect, gRPC, and gRPC-Web protocols with the binary Protobuf +// and JSON codecs. They also support gzip compression. +func NewQueryBackendServiceHandler(svc QueryBackendServiceHandler, opts ...connect.HandlerOption) (string, http.Handler) { + queryBackendServiceInvokeHandler := connect.NewUnaryHandler( + QueryBackendServiceInvokeProcedure, + svc.Invoke, + connect.WithSchema(queryBackendServiceInvokeMethodDescriptor), + connect.WithHandlerOptions(opts...), + ) + return "/querybackend.v1.QueryBackendService/", http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { + switch r.URL.Path { + case QueryBackendServiceInvokeProcedure: + queryBackendServiceInvokeHandler.ServeHTTP(w, r) + default: + http.NotFound(w, r) + } + }) +} + +// UnimplementedQueryBackendServiceHandler returns CodeUnimplemented from all methods. +type UnimplementedQueryBackendServiceHandler struct{} + +func (UnimplementedQueryBackendServiceHandler) Invoke(context.Context, *connect.Request[v1.InvokeRequest]) (*connect.Response[v1.InvokeResponse], error) { + return nil, connect.NewError(connect.CodeUnimplemented, errors.New("querybackend.v1.QueryBackendService.Invoke is not implemented")) +} diff --git a/api/gen/proto/go/querybackend/v1/querybackendv1connect/querybackend.connect.mux.go b/api/gen/proto/go/querybackend/v1/querybackendv1connect/querybackend.connect.mux.go new file mode 100644 index 0000000000..2a58ad96eb --- /dev/null +++ b/api/gen/proto/go/querybackend/v1/querybackendv1connect/querybackend.connect.mux.go @@ -0,0 +1,27 @@ +// Code generated by protoc-gen-connect-go-mux. DO NOT EDIT. +// +// Source: querybackend/v1/querybackend.proto + +package querybackendv1connect + +import ( + connect "connectrpc.com/connect" + mux "github.com/gorilla/mux" +) + +// This is a compile-time assertion to ensure that this generated file and the connect package are +// compatible. If you get a compiler error that this constant is not defined, this code was +// generated with a version of connect newer than the one compiled into your binary. You can fix the +// problem by either regenerating this code with an older version of connect or updating the connect +// version compiled into your binary. +const _ = connect.IsAtLeastVersion0_1_0 + +// RegisterQueryBackendServiceHandler register an HTTP handler to a mux.Router from the service +// implementation. +func RegisterQueryBackendServiceHandler(mux *mux.Router, svc QueryBackendServiceHandler, opts ...connect.HandlerOption) { + mux.Handle("/querybackend.v1.QueryBackendService/Invoke", connect.NewUnaryHandler( + "/querybackend.v1.QueryBackendService/Invoke", + svc.Invoke, + opts..., + )) +} diff --git a/api/gen/proto/go/segmentwriter/v1/push.pb.go b/api/gen/proto/go/segmentwriter/v1/push.pb.go new file mode 100644 index 0000000000..0739e90c41 --- /dev/null +++ b/api/gen/proto/go/segmentwriter/v1/push.pb.go @@ -0,0 +1,389 @@ +// Code generated by protoc-gen-go. DO NOT EDIT. +// versions: +// protoc-gen-go v1.34.2 +// protoc (unknown) +// source: segmentwriter/v1/push.proto + +package segmentwriterv1 + +import ( + v1 "github.com/grafana/pyroscope/api/gen/proto/go/types/v1" + protoreflect "google.golang.org/protobuf/reflect/protoreflect" + protoimpl "google.golang.org/protobuf/runtime/protoimpl" + reflect "reflect" + sync "sync" +) + +const ( + // Verify that this generated code is sufficiently up-to-date. + _ = protoimpl.EnforceVersion(20 - protoimpl.MinVersion) + // Verify that runtime/protoimpl is sufficiently up-to-date. + _ = protoimpl.EnforceVersion(protoimpl.MaxVersion - 20) +) + +type PushResponse struct { + state protoimpl.MessageState + sizeCache protoimpl.SizeCache + unknownFields protoimpl.UnknownFields +} + +func (x *PushResponse) Reset() { + *x = PushResponse{} + if protoimpl.UnsafeEnabled { + mi := &file_segmentwriter_v1_push_proto_msgTypes[0] + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + ms.StoreMessageInfo(mi) + } +} + +func (x *PushResponse) String() string { + return protoimpl.X.MessageStringOf(x) +} + +func (*PushResponse) ProtoMessage() {} + +func (x *PushResponse) ProtoReflect() protoreflect.Message { + mi := &file_segmentwriter_v1_push_proto_msgTypes[0] + if protoimpl.UnsafeEnabled && x != nil { + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + if ms.LoadMessageInfo() == nil { + ms.StoreMessageInfo(mi) + } + return ms + } + return mi.MessageOf(x) +} + +// Deprecated: Use PushResponse.ProtoReflect.Descriptor instead. +func (*PushResponse) Descriptor() ([]byte, []int) { + return file_segmentwriter_v1_push_proto_rawDescGZIP(), []int{0} +} + +// WriteRawRequest writes a pprof profile +type PushRequest struct { + state protoimpl.MessageState + sizeCache protoimpl.SizeCache + unknownFields protoimpl.UnknownFields + + // series is a set raw pprof profiles and accompanying labels + Series *RawProfileSeries `protobuf:"bytes,1,opt,name=series,proto3" json:"series,omitempty"` +} + +func (x *PushRequest) Reset() { + *x = PushRequest{} + if protoimpl.UnsafeEnabled { + mi := &file_segmentwriter_v1_push_proto_msgTypes[1] + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + ms.StoreMessageInfo(mi) + } +} + +func (x *PushRequest) String() string { + return protoimpl.X.MessageStringOf(x) +} + +func (*PushRequest) ProtoMessage() {} + +func (x *PushRequest) ProtoReflect() protoreflect.Message { + mi := &file_segmentwriter_v1_push_proto_msgTypes[1] + if protoimpl.UnsafeEnabled && x != nil { + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + if ms.LoadMessageInfo() == nil { + ms.StoreMessageInfo(mi) + } + return ms + } + return mi.MessageOf(x) +} + +// Deprecated: Use PushRequest.ProtoReflect.Descriptor instead. +func (*PushRequest) Descriptor() ([]byte, []int) { + return file_segmentwriter_v1_push_proto_rawDescGZIP(), []int{1} +} + +func (x *PushRequest) GetSeries() *RawProfileSeries { + if x != nil { + return x.Series + } + return nil +} + +// RawProfileSeries represents the pprof profile and its associated labels +type RawProfileSeries struct { + state protoimpl.MessageState + sizeCache protoimpl.SizeCache + unknownFields protoimpl.UnknownFields + + // LabelPair is the key value pairs to identify the corresponding profile + Labels []*v1.LabelPair `protobuf:"bytes,1,rep,name=labels,proto3" json:"labels,omitempty"` + // samples are the set of profile bytes + Sample *RawSample `protobuf:"bytes,2,opt,name=sample,proto3" json:"sample,omitempty"` + Shard uint32 `protobuf:"varint,3,opt,name=shard,proto3" json:"shard,omitempty"` +} + +func (x *RawProfileSeries) Reset() { + *x = RawProfileSeries{} + if protoimpl.UnsafeEnabled { + mi := &file_segmentwriter_v1_push_proto_msgTypes[2] + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + ms.StoreMessageInfo(mi) + } +} + +func (x *RawProfileSeries) String() string { + return protoimpl.X.MessageStringOf(x) +} + +func (*RawProfileSeries) ProtoMessage() {} + +func (x *RawProfileSeries) ProtoReflect() protoreflect.Message { + mi := &file_segmentwriter_v1_push_proto_msgTypes[2] + if protoimpl.UnsafeEnabled && x != nil { + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + if ms.LoadMessageInfo() == nil { + ms.StoreMessageInfo(mi) + } + return ms + } + return mi.MessageOf(x) +} + +// Deprecated: Use RawProfileSeries.ProtoReflect.Descriptor instead. +func (*RawProfileSeries) Descriptor() ([]byte, []int) { + return file_segmentwriter_v1_push_proto_rawDescGZIP(), []int{2} +} + +func (x *RawProfileSeries) GetLabels() []*v1.LabelPair { + if x != nil { + return x.Labels + } + return nil +} + +func (x *RawProfileSeries) GetSample() *RawSample { + if x != nil { + return x.Sample + } + return nil +} + +func (x *RawProfileSeries) GetShard() uint32 { + if x != nil { + return x.Shard + } + return 0 +} + +// RawSample is the set of bytes that correspond to a pprof profile +type RawSample struct { + state protoimpl.MessageState + sizeCache protoimpl.SizeCache + unknownFields protoimpl.UnknownFields + + // raw_profile is the set of bytes of the pprof profile + RawProfile []byte `protobuf:"bytes,1,opt,name=raw_profile,json=rawProfile,proto3" json:"raw_profile,omitempty"` + // unique ID of the profile + ID string `protobuf:"bytes,2,opt,name=ID,proto3" json:"ID,omitempty"` +} + +func (x *RawSample) Reset() { + *x = RawSample{} + if protoimpl.UnsafeEnabled { + mi := &file_segmentwriter_v1_push_proto_msgTypes[3] + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + ms.StoreMessageInfo(mi) + } +} + +func (x *RawSample) String() string { + return protoimpl.X.MessageStringOf(x) +} + +func (*RawSample) ProtoMessage() {} + +func (x *RawSample) ProtoReflect() protoreflect.Message { + mi := &file_segmentwriter_v1_push_proto_msgTypes[3] + if protoimpl.UnsafeEnabled && x != nil { + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + if ms.LoadMessageInfo() == nil { + ms.StoreMessageInfo(mi) + } + return ms + } + return mi.MessageOf(x) +} + +// Deprecated: Use RawSample.ProtoReflect.Descriptor instead. +func (*RawSample) Descriptor() ([]byte, []int) { + return file_segmentwriter_v1_push_proto_rawDescGZIP(), []int{3} +} + +func (x *RawSample) GetRawProfile() []byte { + if x != nil { + return x.RawProfile + } + return nil +} + +func (x *RawSample) GetID() string { + if x != nil { + return x.ID + } + return "" +} + +var File_segmentwriter_v1_push_proto protoreflect.FileDescriptor + +var file_segmentwriter_v1_push_proto_rawDesc = []byte{ + 0x0a, 0x1b, 0x73, 0x65, 0x67, 0x6d, 0x65, 0x6e, 0x74, 0x77, 0x72, 0x69, 0x74, 0x65, 0x72, 0x2f, + 0x76, 0x31, 0x2f, 0x70, 0x75, 0x73, 0x68, 0x2e, 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x12, 0x10, 0x73, + 0x65, 0x67, 0x6d, 0x65, 0x6e, 0x74, 0x77, 0x72, 0x69, 0x74, 0x65, 0x72, 0x2e, 0x76, 0x31, 0x1a, + 0x14, 0x74, 0x79, 0x70, 0x65, 0x73, 0x2f, 0x76, 0x31, 0x2f, 0x74, 0x79, 0x70, 0x65, 0x73, 0x2e, + 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x22, 0x0e, 0x0a, 0x0c, 0x50, 0x75, 0x73, 0x68, 0x52, 0x65, 0x73, + 0x70, 0x6f, 0x6e, 0x73, 0x65, 0x22, 0x49, 0x0a, 0x0b, 0x50, 0x75, 0x73, 0x68, 0x52, 0x65, 0x71, + 0x75, 0x65, 0x73, 0x74, 0x12, 0x3a, 0x0a, 0x06, 0x73, 0x65, 0x72, 0x69, 0x65, 0x73, 0x18, 0x01, + 0x20, 0x01, 0x28, 0x0b, 0x32, 0x22, 0x2e, 0x73, 0x65, 0x67, 0x6d, 0x65, 0x6e, 0x74, 0x77, 0x72, + 0x69, 0x74, 0x65, 0x72, 0x2e, 0x76, 0x31, 0x2e, 0x52, 0x61, 0x77, 0x50, 0x72, 0x6f, 0x66, 0x69, + 0x6c, 0x65, 0x53, 0x65, 0x72, 0x69, 0x65, 0x73, 0x52, 0x06, 0x73, 0x65, 0x72, 0x69, 0x65, 0x73, + 0x22, 0x8a, 0x01, 0x0a, 0x10, 0x52, 0x61, 0x77, 0x50, 0x72, 0x6f, 0x66, 0x69, 0x6c, 0x65, 0x53, + 0x65, 0x72, 0x69, 0x65, 0x73, 0x12, 0x2b, 0x0a, 0x06, 0x6c, 0x61, 0x62, 0x65, 0x6c, 0x73, 0x18, + 0x01, 0x20, 0x03, 0x28, 0x0b, 0x32, 0x13, 0x2e, 0x74, 0x79, 0x70, 0x65, 0x73, 0x2e, 0x76, 0x31, + 0x2e, 0x4c, 0x61, 0x62, 0x65, 0x6c, 0x50, 0x61, 0x69, 0x72, 0x52, 0x06, 0x6c, 0x61, 0x62, 0x65, + 0x6c, 0x73, 0x12, 0x33, 0x0a, 0x06, 0x73, 0x61, 0x6d, 0x70, 0x6c, 0x65, 0x18, 0x02, 0x20, 0x01, + 0x28, 0x0b, 0x32, 0x1b, 0x2e, 0x73, 0x65, 0x67, 0x6d, 0x65, 0x6e, 0x74, 0x77, 0x72, 0x69, 0x74, + 0x65, 0x72, 0x2e, 0x76, 0x31, 0x2e, 0x52, 0x61, 0x77, 0x53, 0x61, 0x6d, 0x70, 0x6c, 0x65, 0x52, + 0x06, 0x73, 0x61, 0x6d, 0x70, 0x6c, 0x65, 0x12, 0x14, 0x0a, 0x05, 0x73, 0x68, 0x61, 0x72, 0x64, + 0x18, 0x03, 0x20, 0x01, 0x28, 0x0d, 0x52, 0x05, 0x73, 0x68, 0x61, 0x72, 0x64, 0x22, 0x3c, 0x0a, + 0x09, 0x52, 0x61, 0x77, 0x53, 0x61, 0x6d, 0x70, 0x6c, 0x65, 0x12, 0x1f, 0x0a, 0x0b, 0x72, 0x61, + 0x77, 0x5f, 0x70, 0x72, 0x6f, 0x66, 0x69, 0x6c, 0x65, 0x18, 0x01, 0x20, 0x01, 0x28, 0x0c, 0x52, + 0x0a, 0x72, 0x61, 0x77, 0x50, 0x72, 0x6f, 0x66, 0x69, 0x6c, 0x65, 0x12, 0x0e, 0x0a, 0x02, 0x49, + 0x44, 0x18, 0x02, 0x20, 0x01, 0x28, 0x09, 0x52, 0x02, 0x49, 0x44, 0x32, 0x5f, 0x0a, 0x14, 0x53, + 0x65, 0x67, 0x6d, 0x65, 0x6e, 0x74, 0x57, 0x72, 0x69, 0x74, 0x65, 0x72, 0x53, 0x65, 0x72, 0x76, + 0x69, 0x63, 0x65, 0x12, 0x47, 0x0a, 0x04, 0x50, 0x75, 0x73, 0x68, 0x12, 0x1d, 0x2e, 0x73, 0x65, + 0x67, 0x6d, 0x65, 0x6e, 0x74, 0x77, 0x72, 0x69, 0x74, 0x65, 0x72, 0x2e, 0x76, 0x31, 0x2e, 0x50, + 0x75, 0x73, 0x68, 0x52, 0x65, 0x71, 0x75, 0x65, 0x73, 0x74, 0x1a, 0x1e, 0x2e, 0x73, 0x65, 0x67, + 0x6d, 0x65, 0x6e, 0x74, 0x77, 0x72, 0x69, 0x74, 0x65, 0x72, 0x2e, 0x76, 0x31, 0x2e, 0x50, 0x75, + 0x73, 0x68, 0x52, 0x65, 0x73, 0x70, 0x6f, 0x6e, 0x73, 0x65, 0x22, 0x00, 0x42, 0xd2, 0x01, 0x0a, + 0x14, 0x63, 0x6f, 0x6d, 0x2e, 0x73, 0x65, 0x67, 0x6d, 0x65, 0x6e, 0x74, 0x77, 0x72, 0x69, 0x74, + 0x65, 0x72, 0x2e, 0x76, 0x31, 0x42, 0x09, 0x50, 0x75, 0x73, 0x68, 0x50, 0x72, 0x6f, 0x74, 0x6f, + 0x50, 0x01, 0x5a, 0x4e, 0x67, 0x69, 0x74, 0x68, 0x75, 0x62, 0x2e, 0x63, 0x6f, 0x6d, 0x2f, 0x67, + 0x72, 0x61, 0x66, 0x61, 0x6e, 0x61, 0x2f, 0x70, 0x79, 0x72, 0x6f, 0x73, 0x63, 0x6f, 0x70, 0x65, + 0x2f, 0x61, 0x70, 0x69, 0x2f, 0x67, 0x65, 0x6e, 0x2f, 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x2f, 0x67, + 0x6f, 0x2f, 0x73, 0x65, 0x67, 0x6d, 0x65, 0x6e, 0x74, 0x77, 0x72, 0x69, 0x74, 0x65, 0x72, 0x2f, + 0x76, 0x31, 0x3b, 0x73, 0x65, 0x67, 0x6d, 0x65, 0x6e, 0x74, 0x77, 0x72, 0x69, 0x74, 0x65, 0x72, + 0x76, 0x31, 0xa2, 0x02, 0x03, 0x53, 0x58, 0x58, 0xaa, 0x02, 0x10, 0x53, 0x65, 0x67, 0x6d, 0x65, + 0x6e, 0x74, 0x77, 0x72, 0x69, 0x74, 0x65, 0x72, 0x2e, 0x56, 0x31, 0xca, 0x02, 0x10, 0x53, 0x65, + 0x67, 0x6d, 0x65, 0x6e, 0x74, 0x77, 0x72, 0x69, 0x74, 0x65, 0x72, 0x5c, 0x56, 0x31, 0xe2, 0x02, + 0x1c, 0x53, 0x65, 0x67, 0x6d, 0x65, 0x6e, 0x74, 0x77, 0x72, 0x69, 0x74, 0x65, 0x72, 0x5c, 0x56, + 0x31, 0x5c, 0x47, 0x50, 0x42, 0x4d, 0x65, 0x74, 0x61, 0x64, 0x61, 0x74, 0x61, 0xea, 0x02, 0x11, + 0x53, 0x65, 0x67, 0x6d, 0x65, 0x6e, 0x74, 0x77, 0x72, 0x69, 0x74, 0x65, 0x72, 0x3a, 0x3a, 0x56, + 0x31, 0x62, 0x06, 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x33, +} + +var ( + file_segmentwriter_v1_push_proto_rawDescOnce sync.Once + file_segmentwriter_v1_push_proto_rawDescData = file_segmentwriter_v1_push_proto_rawDesc +) + +func file_segmentwriter_v1_push_proto_rawDescGZIP() []byte { + file_segmentwriter_v1_push_proto_rawDescOnce.Do(func() { + file_segmentwriter_v1_push_proto_rawDescData = protoimpl.X.CompressGZIP(file_segmentwriter_v1_push_proto_rawDescData) + }) + return file_segmentwriter_v1_push_proto_rawDescData +} + +var file_segmentwriter_v1_push_proto_msgTypes = make([]protoimpl.MessageInfo, 4) +var file_segmentwriter_v1_push_proto_goTypes = []any{ + (*PushResponse)(nil), // 0: segmentwriter.v1.PushResponse + (*PushRequest)(nil), // 1: segmentwriter.v1.PushRequest + (*RawProfileSeries)(nil), // 2: segmentwriter.v1.RawProfileSeries + (*RawSample)(nil), // 3: segmentwriter.v1.RawSample + (*v1.LabelPair)(nil), // 4: types.v1.LabelPair +} +var file_segmentwriter_v1_push_proto_depIdxs = []int32{ + 2, // 0: segmentwriter.v1.PushRequest.series:type_name -> segmentwriter.v1.RawProfileSeries + 4, // 1: segmentwriter.v1.RawProfileSeries.labels:type_name -> types.v1.LabelPair + 3, // 2: segmentwriter.v1.RawProfileSeries.sample:type_name -> segmentwriter.v1.RawSample + 1, // 3: segmentwriter.v1.SegmentWriterService.Push:input_type -> segmentwriter.v1.PushRequest + 0, // 4: segmentwriter.v1.SegmentWriterService.Push:output_type -> segmentwriter.v1.PushResponse + 4, // [4:5] is the sub-list for method output_type + 3, // [3:4] is the sub-list for method input_type + 3, // [3:3] is the sub-list for extension type_name + 3, // [3:3] is the sub-list for extension extendee + 0, // [0:3] is the sub-list for field type_name +} + +func init() { file_segmentwriter_v1_push_proto_init() } +func file_segmentwriter_v1_push_proto_init() { + if File_segmentwriter_v1_push_proto != nil { + return + } + if !protoimpl.UnsafeEnabled { + file_segmentwriter_v1_push_proto_msgTypes[0].Exporter = func(v any, i int) any { + switch v := v.(*PushResponse); i { + case 0: + return &v.state + case 1: + return &v.sizeCache + case 2: + return &v.unknownFields + default: + return nil + } + } + file_segmentwriter_v1_push_proto_msgTypes[1].Exporter = func(v any, i int) any { + switch v := v.(*PushRequest); i { + case 0: + return &v.state + case 1: + return &v.sizeCache + case 2: + return &v.unknownFields + default: + return nil + } + } + file_segmentwriter_v1_push_proto_msgTypes[2].Exporter = func(v any, i int) any { + switch v := v.(*RawProfileSeries); i { + case 0: + return &v.state + case 1: + return &v.sizeCache + case 2: + return &v.unknownFields + default: + return nil + } + } + file_segmentwriter_v1_push_proto_msgTypes[3].Exporter = func(v any, i int) any { + switch v := v.(*RawSample); i { + case 0: + return &v.state + case 1: + return &v.sizeCache + case 2: + return &v.unknownFields + default: + return nil + } + } + } + type x struct{} + out := protoimpl.TypeBuilder{ + File: protoimpl.DescBuilder{ + GoPackagePath: reflect.TypeOf(x{}).PkgPath(), + RawDescriptor: file_segmentwriter_v1_push_proto_rawDesc, + NumEnums: 0, + NumMessages: 4, + NumExtensions: 0, + NumServices: 1, + }, + GoTypes: file_segmentwriter_v1_push_proto_goTypes, + DependencyIndexes: file_segmentwriter_v1_push_proto_depIdxs, + MessageInfos: file_segmentwriter_v1_push_proto_msgTypes, + }.Build() + File_segmentwriter_v1_push_proto = out.File + file_segmentwriter_v1_push_proto_rawDesc = nil + file_segmentwriter_v1_push_proto_goTypes = nil + file_segmentwriter_v1_push_proto_depIdxs = nil +} diff --git a/api/gen/proto/go/segmentwriter/v1/push_vtproto.pb.go b/api/gen/proto/go/segmentwriter/v1/push_vtproto.pb.go new file mode 100644 index 0000000000..c877160d7f --- /dev/null +++ b/api/gen/proto/go/segmentwriter/v1/push_vtproto.pb.go @@ -0,0 +1,971 @@ +// Code generated by protoc-gen-go-vtproto. DO NOT EDIT. +// protoc-gen-go-vtproto version: v0.6.0 +// source: segmentwriter/v1/push.proto + +package segmentwriterv1 + +import ( + context "context" + fmt "fmt" + v1 "github.com/grafana/pyroscope/api/gen/proto/go/types/v1" + protohelpers "github.com/planetscale/vtprotobuf/protohelpers" + grpc "google.golang.org/grpc" + codes "google.golang.org/grpc/codes" + status "google.golang.org/grpc/status" + proto "google.golang.org/protobuf/proto" + protoimpl "google.golang.org/protobuf/runtime/protoimpl" + io "io" +) + +const ( + // Verify that this generated code is sufficiently up-to-date. + _ = protoimpl.EnforceVersion(20 - protoimpl.MinVersion) + // Verify that runtime/protoimpl is sufficiently up-to-date. + _ = protoimpl.EnforceVersion(protoimpl.MaxVersion - 20) +) + +func (m *PushResponse) CloneVT() *PushResponse { + if m == nil { + return (*PushResponse)(nil) + } + r := new(PushResponse) + if len(m.unknownFields) > 0 { + r.unknownFields = make([]byte, len(m.unknownFields)) + copy(r.unknownFields, m.unknownFields) + } + return r +} + +func (m *PushResponse) CloneMessageVT() proto.Message { + return m.CloneVT() +} + +func (m *PushRequest) CloneVT() *PushRequest { + if m == nil { + return (*PushRequest)(nil) + } + r := new(PushRequest) + r.Series = m.Series.CloneVT() + if len(m.unknownFields) > 0 { + r.unknownFields = make([]byte, len(m.unknownFields)) + copy(r.unknownFields, m.unknownFields) + } + return r +} + +func (m *PushRequest) CloneMessageVT() proto.Message { + return m.CloneVT() +} + +func (m *RawProfileSeries) CloneVT() *RawProfileSeries { + if m == nil { + return (*RawProfileSeries)(nil) + } + r := new(RawProfileSeries) + r.Sample = m.Sample.CloneVT() + r.Shard = m.Shard + if rhs := m.Labels; rhs != nil { + tmpContainer := make([]*v1.LabelPair, len(rhs)) + for k, v := range rhs { + if vtpb, ok := interface{}(v).(interface{ CloneVT() *v1.LabelPair }); ok { + tmpContainer[k] = vtpb.CloneVT() + } else { + tmpContainer[k] = proto.Clone(v).(*v1.LabelPair) + } + } + r.Labels = tmpContainer + } + if len(m.unknownFields) > 0 { + r.unknownFields = make([]byte, len(m.unknownFields)) + copy(r.unknownFields, m.unknownFields) + } + return r +} + +func (m *RawProfileSeries) CloneMessageVT() proto.Message { + return m.CloneVT() +} + +func (m *RawSample) CloneVT() *RawSample { + if m == nil { + return (*RawSample)(nil) + } + r := new(RawSample) + r.ID = m.ID + if rhs := m.RawProfile; rhs != nil { + tmpBytes := make([]byte, len(rhs)) + copy(tmpBytes, rhs) + r.RawProfile = tmpBytes + } + if len(m.unknownFields) > 0 { + r.unknownFields = make([]byte, len(m.unknownFields)) + copy(r.unknownFields, m.unknownFields) + } + return r +} + +func (m *RawSample) CloneMessageVT() proto.Message { + return m.CloneVT() +} + +func (this *PushResponse) EqualVT(that *PushResponse) bool { + if this == that { + return true + } else if this == nil || that == nil { + return false + } + return string(this.unknownFields) == string(that.unknownFields) +} + +func (this *PushResponse) EqualMessageVT(thatMsg proto.Message) bool { + that, ok := thatMsg.(*PushResponse) + if !ok { + return false + } + return this.EqualVT(that) +} +func (this *PushRequest) EqualVT(that *PushRequest) bool { + if this == that { + return true + } else if this == nil || that == nil { + return false + } + if !this.Series.EqualVT(that.Series) { + return false + } + return string(this.unknownFields) == string(that.unknownFields) +} + +func (this *PushRequest) EqualMessageVT(thatMsg proto.Message) bool { + that, ok := thatMsg.(*PushRequest) + if !ok { + return false + } + return this.EqualVT(that) +} +func (this *RawProfileSeries) EqualVT(that *RawProfileSeries) bool { + if this == that { + return true + } else if this == nil || that == nil { + return false + } + if len(this.Labels) != len(that.Labels) { + return false + } + for i, vx := range this.Labels { + vy := that.Labels[i] + if p, q := vx, vy; p != q { + if p == nil { + p = &v1.LabelPair{} + } + if q == nil { + q = &v1.LabelPair{} + } + if equal, ok := interface{}(p).(interface{ EqualVT(*v1.LabelPair) bool }); ok { + if !equal.EqualVT(q) { + return false + } + } else if !proto.Equal(p, q) { + return false + } + } + } + if !this.Sample.EqualVT(that.Sample) { + return false + } + if this.Shard != that.Shard { + return false + } + return string(this.unknownFields) == string(that.unknownFields) +} + +func (this *RawProfileSeries) EqualMessageVT(thatMsg proto.Message) bool { + that, ok := thatMsg.(*RawProfileSeries) + if !ok { + return false + } + return this.EqualVT(that) +} +func (this *RawSample) EqualVT(that *RawSample) bool { + if this == that { + return true + } else if this == nil || that == nil { + return false + } + if string(this.RawProfile) != string(that.RawProfile) { + return false + } + if this.ID != that.ID { + return false + } + return string(this.unknownFields) == string(that.unknownFields) +} + +func (this *RawSample) EqualMessageVT(thatMsg proto.Message) bool { + that, ok := thatMsg.(*RawSample) + if !ok { + return false + } + return this.EqualVT(that) +} + +// This is a compile-time assertion to ensure that this generated file +// is compatible with the grpc package it is being compiled against. +// Requires gRPC-Go v1.32.0 or later. +const _ = grpc.SupportPackageIsVersion7 + +// SegmentWriterServiceClient is the client API for SegmentWriterService service. +// +// For semantics around ctx use and closing/ending streaming RPCs, please refer to https://pkg.go.dev/google.golang.org/grpc/?tab=doc#ClientConn.NewStream. +type SegmentWriterServiceClient interface { + Push(ctx context.Context, in *PushRequest, opts ...grpc.CallOption) (*PushResponse, error) +} + +type segmentWriterServiceClient struct { + cc grpc.ClientConnInterface +} + +func NewSegmentWriterServiceClient(cc grpc.ClientConnInterface) SegmentWriterServiceClient { + return &segmentWriterServiceClient{cc} +} + +func (c *segmentWriterServiceClient) Push(ctx context.Context, in *PushRequest, opts ...grpc.CallOption) (*PushResponse, error) { + out := new(PushResponse) + err := c.cc.Invoke(ctx, "/segmentwriter.v1.SegmentWriterService/Push", in, out, opts...) + if err != nil { + return nil, err + } + return out, nil +} + +// SegmentWriterServiceServer is the server API for SegmentWriterService service. +// All implementations must embed UnimplementedSegmentWriterServiceServer +// for forward compatibility +type SegmentWriterServiceServer interface { + Push(context.Context, *PushRequest) (*PushResponse, error) + mustEmbedUnimplementedSegmentWriterServiceServer() +} + +// UnimplementedSegmentWriterServiceServer must be embedded to have forward compatible implementations. +type UnimplementedSegmentWriterServiceServer struct { +} + +func (UnimplementedSegmentWriterServiceServer) Push(context.Context, *PushRequest) (*PushResponse, error) { + return nil, status.Errorf(codes.Unimplemented, "method Push not implemented") +} +func (UnimplementedSegmentWriterServiceServer) mustEmbedUnimplementedSegmentWriterServiceServer() {} + +// UnsafeSegmentWriterServiceServer may be embedded to opt out of forward compatibility for this service. +// Use of this interface is not recommended, as added methods to SegmentWriterServiceServer will +// result in compilation errors. +type UnsafeSegmentWriterServiceServer interface { + mustEmbedUnimplementedSegmentWriterServiceServer() +} + +func RegisterSegmentWriterServiceServer(s grpc.ServiceRegistrar, srv SegmentWriterServiceServer) { + s.RegisterService(&SegmentWriterService_ServiceDesc, srv) +} + +func _SegmentWriterService_Push_Handler(srv interface{}, ctx context.Context, dec func(interface{}) error, interceptor grpc.UnaryServerInterceptor) (interface{}, error) { + in := new(PushRequest) + if err := dec(in); err != nil { + return nil, err + } + if interceptor == nil { + return srv.(SegmentWriterServiceServer).Push(ctx, in) + } + info := &grpc.UnaryServerInfo{ + Server: srv, + FullMethod: "/segmentwriter.v1.SegmentWriterService/Push", + } + handler := func(ctx context.Context, req interface{}) (interface{}, error) { + return srv.(SegmentWriterServiceServer).Push(ctx, req.(*PushRequest)) + } + return interceptor(ctx, in, info, handler) +} + +// SegmentWriterService_ServiceDesc is the grpc.ServiceDesc for SegmentWriterService service. +// It's only intended for direct use with grpc.RegisterService, +// and not to be introspected or modified (even as a copy) +var SegmentWriterService_ServiceDesc = grpc.ServiceDesc{ + ServiceName: "segmentwriter.v1.SegmentWriterService", + HandlerType: (*SegmentWriterServiceServer)(nil), + Methods: []grpc.MethodDesc{ + { + MethodName: "Push", + Handler: _SegmentWriterService_Push_Handler, + }, + }, + Streams: []grpc.StreamDesc{}, + Metadata: "segmentwriter/v1/push.proto", +} + +func (m *PushResponse) MarshalVT() (dAtA []byte, err error) { + if m == nil { + return nil, nil + } + size := m.SizeVT() + dAtA = make([]byte, size) + n, err := m.MarshalToSizedBufferVT(dAtA[:size]) + if err != nil { + return nil, err + } + return dAtA[:n], nil +} + +func (m *PushResponse) MarshalToVT(dAtA []byte) (int, error) { + size := m.SizeVT() + return m.MarshalToSizedBufferVT(dAtA[:size]) +} + +func (m *PushResponse) MarshalToSizedBufferVT(dAtA []byte) (int, error) { + if m == nil { + return 0, nil + } + i := len(dAtA) + _ = i + var l int + _ = l + if m.unknownFields != nil { + i -= len(m.unknownFields) + copy(dAtA[i:], m.unknownFields) + } + return len(dAtA) - i, nil +} + +func (m *PushRequest) MarshalVT() (dAtA []byte, err error) { + if m == nil { + return nil, nil + } + size := m.SizeVT() + dAtA = make([]byte, size) + n, err := m.MarshalToSizedBufferVT(dAtA[:size]) + if err != nil { + return nil, err + } + return dAtA[:n], nil +} + +func (m *PushRequest) MarshalToVT(dAtA []byte) (int, error) { + size := m.SizeVT() + return m.MarshalToSizedBufferVT(dAtA[:size]) +} + +func (m *PushRequest) MarshalToSizedBufferVT(dAtA []byte) (int, error) { + if m == nil { + return 0, nil + } + i := len(dAtA) + _ = i + var l int + _ = l + if m.unknownFields != nil { + i -= len(m.unknownFields) + copy(dAtA[i:], m.unknownFields) + } + if m.Series != nil { + size, err := m.Series.MarshalToSizedBufferVT(dAtA[:i]) + if err != nil { + return 0, err + } + i -= size + i = protohelpers.EncodeVarint(dAtA, i, uint64(size)) + i-- + dAtA[i] = 0xa + } + return len(dAtA) - i, nil +} + +func (m *RawProfileSeries) MarshalVT() (dAtA []byte, err error) { + if m == nil { + return nil, nil + } + size := m.SizeVT() + dAtA = make([]byte, size) + n, err := m.MarshalToSizedBufferVT(dAtA[:size]) + if err != nil { + return nil, err + } + return dAtA[:n], nil +} + +func (m *RawProfileSeries) MarshalToVT(dAtA []byte) (int, error) { + size := m.SizeVT() + return m.MarshalToSizedBufferVT(dAtA[:size]) +} + +func (m *RawProfileSeries) MarshalToSizedBufferVT(dAtA []byte) (int, error) { + if m == nil { + return 0, nil + } + i := len(dAtA) + _ = i + var l int + _ = l + if m.unknownFields != nil { + i -= len(m.unknownFields) + copy(dAtA[i:], m.unknownFields) + } + if m.Shard != 0 { + i = protohelpers.EncodeVarint(dAtA, i, uint64(m.Shard)) + i-- + dAtA[i] = 0x18 + } + if m.Sample != nil { + size, err := m.Sample.MarshalToSizedBufferVT(dAtA[:i]) + if err != nil { + return 0, err + } + i -= size + i = protohelpers.EncodeVarint(dAtA, i, uint64(size)) + i-- + dAtA[i] = 0x12 + } + if len(m.Labels) > 0 { + for iNdEx := len(m.Labels) - 1; iNdEx >= 0; iNdEx-- { + if vtmsg, ok := interface{}(m.Labels[iNdEx]).(interface { + MarshalToSizedBufferVT([]byte) (int, error) + }); ok { + size, err := vtmsg.MarshalToSizedBufferVT(dAtA[:i]) + if err != nil { + return 0, err + } + i -= size + i = protohelpers.EncodeVarint(dAtA, i, uint64(size)) + } else { + encoded, err := proto.Marshal(m.Labels[iNdEx]) + if err != nil { + return 0, err + } + i -= len(encoded) + copy(dAtA[i:], encoded) + i = protohelpers.EncodeVarint(dAtA, i, uint64(len(encoded))) + } + i-- + dAtA[i] = 0xa + } + } + return len(dAtA) - i, nil +} + +func (m *RawSample) MarshalVT() (dAtA []byte, err error) { + if m == nil { + return nil, nil + } + size := m.SizeVT() + dAtA = make([]byte, size) + n, err := m.MarshalToSizedBufferVT(dAtA[:size]) + if err != nil { + return nil, err + } + return dAtA[:n], nil +} + +func (m *RawSample) MarshalToVT(dAtA []byte) (int, error) { + size := m.SizeVT() + return m.MarshalToSizedBufferVT(dAtA[:size]) +} + +func (m *RawSample) MarshalToSizedBufferVT(dAtA []byte) (int, error) { + if m == nil { + return 0, nil + } + i := len(dAtA) + _ = i + var l int + _ = l + if m.unknownFields != nil { + i -= len(m.unknownFields) + copy(dAtA[i:], m.unknownFields) + } + if len(m.ID) > 0 { + i -= len(m.ID) + copy(dAtA[i:], m.ID) + i = protohelpers.EncodeVarint(dAtA, i, uint64(len(m.ID))) + i-- + dAtA[i] = 0x12 + } + if len(m.RawProfile) > 0 { + i -= len(m.RawProfile) + copy(dAtA[i:], m.RawProfile) + i = protohelpers.EncodeVarint(dAtA, i, uint64(len(m.RawProfile))) + i-- + dAtA[i] = 0xa + } + return len(dAtA) - i, nil +} + +func (m *PushResponse) SizeVT() (n int) { + if m == nil { + return 0 + } + var l int + _ = l + n += len(m.unknownFields) + return n +} + +func (m *PushRequest) SizeVT() (n int) { + if m == nil { + return 0 + } + var l int + _ = l + if m.Series != nil { + l = m.Series.SizeVT() + n += 1 + l + protohelpers.SizeOfVarint(uint64(l)) + } + n += len(m.unknownFields) + return n +} + +func (m *RawProfileSeries) SizeVT() (n int) { + if m == nil { + return 0 + } + var l int + _ = l + if len(m.Labels) > 0 { + for _, e := range m.Labels { + if size, ok := interface{}(e).(interface { + SizeVT() int + }); ok { + l = size.SizeVT() + } else { + l = proto.Size(e) + } + n += 1 + l + protohelpers.SizeOfVarint(uint64(l)) + } + } + if m.Sample != nil { + l = m.Sample.SizeVT() + n += 1 + l + protohelpers.SizeOfVarint(uint64(l)) + } + if m.Shard != 0 { + n += 1 + protohelpers.SizeOfVarint(uint64(m.Shard)) + } + n += len(m.unknownFields) + return n +} + +func (m *RawSample) SizeVT() (n int) { + if m == nil { + return 0 + } + var l int + _ = l + l = len(m.RawProfile) + if l > 0 { + n += 1 + l + protohelpers.SizeOfVarint(uint64(l)) + } + l = len(m.ID) + if l > 0 { + n += 1 + l + protohelpers.SizeOfVarint(uint64(l)) + } + n += len(m.unknownFields) + return n +} + +func (m *PushResponse) UnmarshalVT(dAtA []byte) error { + l := len(dAtA) + iNdEx := 0 + for iNdEx < l { + preIndex := iNdEx + var wire uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + wire |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + fieldNum := int32(wire >> 3) + wireType := int(wire & 0x7) + if wireType == 4 { + return fmt.Errorf("proto: PushResponse: wiretype end group for non-group") + } + if fieldNum <= 0 { + return fmt.Errorf("proto: PushResponse: illegal tag %d (wire type %d)", fieldNum, wire) + } + switch fieldNum { + default: + iNdEx = preIndex + skippy, err := protohelpers.Skip(dAtA[iNdEx:]) + if err != nil { + return err + } + if (skippy < 0) || (iNdEx+skippy) < 0 { + return protohelpers.ErrInvalidLength + } + if (iNdEx + skippy) > l { + return io.ErrUnexpectedEOF + } + m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...) + iNdEx += skippy + } + } + + if iNdEx > l { + return io.ErrUnexpectedEOF + } + return nil +} +func (m *PushRequest) UnmarshalVT(dAtA []byte) error { + l := len(dAtA) + iNdEx := 0 + for iNdEx < l { + preIndex := iNdEx + var wire uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + wire |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + fieldNum := int32(wire >> 3) + wireType := int(wire & 0x7) + if wireType == 4 { + return fmt.Errorf("proto: PushRequest: wiretype end group for non-group") + } + if fieldNum <= 0 { + return fmt.Errorf("proto: PushRequest: illegal tag %d (wire type %d)", fieldNum, wire) + } + switch fieldNum { + case 1: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field Series", wireType) + } + var msglen int + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + msglen |= int(b&0x7F) << shift + if b < 0x80 { + break + } + } + if msglen < 0 { + return protohelpers.ErrInvalidLength + } + postIndex := iNdEx + msglen + if postIndex < 0 { + return protohelpers.ErrInvalidLength + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + if m.Series == nil { + m.Series = &RawProfileSeries{} + } + if err := m.Series.UnmarshalVT(dAtA[iNdEx:postIndex]); err != nil { + return err + } + iNdEx = postIndex + default: + iNdEx = preIndex + skippy, err := protohelpers.Skip(dAtA[iNdEx:]) + if err != nil { + return err + } + if (skippy < 0) || (iNdEx+skippy) < 0 { + return protohelpers.ErrInvalidLength + } + if (iNdEx + skippy) > l { + return io.ErrUnexpectedEOF + } + m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...) + iNdEx += skippy + } + } + + if iNdEx > l { + return io.ErrUnexpectedEOF + } + return nil +} +func (m *RawProfileSeries) UnmarshalVT(dAtA []byte) error { + l := len(dAtA) + iNdEx := 0 + for iNdEx < l { + preIndex := iNdEx + var wire uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + wire |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + fieldNum := int32(wire >> 3) + wireType := int(wire & 0x7) + if wireType == 4 { + return fmt.Errorf("proto: RawProfileSeries: wiretype end group for non-group") + } + if fieldNum <= 0 { + return fmt.Errorf("proto: RawProfileSeries: illegal tag %d (wire type %d)", fieldNum, wire) + } + switch fieldNum { + case 1: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field Labels", wireType) + } + var msglen int + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + msglen |= int(b&0x7F) << shift + if b < 0x80 { + break + } + } + if msglen < 0 { + return protohelpers.ErrInvalidLength + } + postIndex := iNdEx + msglen + if postIndex < 0 { + return protohelpers.ErrInvalidLength + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + m.Labels = append(m.Labels, &v1.LabelPair{}) + if unmarshal, ok := interface{}(m.Labels[len(m.Labels)-1]).(interface { + UnmarshalVT([]byte) error + }); ok { + if err := unmarshal.UnmarshalVT(dAtA[iNdEx:postIndex]); err != nil { + return err + } + } else { + if err := proto.Unmarshal(dAtA[iNdEx:postIndex], m.Labels[len(m.Labels)-1]); err != nil { + return err + } + } + iNdEx = postIndex + case 2: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field Sample", wireType) + } + var msglen int + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + msglen |= int(b&0x7F) << shift + if b < 0x80 { + break + } + } + if msglen < 0 { + return protohelpers.ErrInvalidLength + } + postIndex := iNdEx + msglen + if postIndex < 0 { + return protohelpers.ErrInvalidLength + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + if m.Sample == nil { + m.Sample = &RawSample{} + } + if err := m.Sample.UnmarshalVT(dAtA[iNdEx:postIndex]); err != nil { + return err + } + iNdEx = postIndex + case 3: + if wireType != 0 { + return fmt.Errorf("proto: wrong wireType = %d for field Shard", wireType) + } + m.Shard = 0 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + m.Shard |= uint32(b&0x7F) << shift + if b < 0x80 { + break + } + } + default: + iNdEx = preIndex + skippy, err := protohelpers.Skip(dAtA[iNdEx:]) + if err != nil { + return err + } + if (skippy < 0) || (iNdEx+skippy) < 0 { + return protohelpers.ErrInvalidLength + } + if (iNdEx + skippy) > l { + return io.ErrUnexpectedEOF + } + m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...) + iNdEx += skippy + } + } + + if iNdEx > l { + return io.ErrUnexpectedEOF + } + return nil +} +func (m *RawSample) UnmarshalVT(dAtA []byte) error { + l := len(dAtA) + iNdEx := 0 + for iNdEx < l { + preIndex := iNdEx + var wire uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + wire |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + fieldNum := int32(wire >> 3) + wireType := int(wire & 0x7) + if wireType == 4 { + return fmt.Errorf("proto: RawSample: wiretype end group for non-group") + } + if fieldNum <= 0 { + return fmt.Errorf("proto: RawSample: illegal tag %d (wire type %d)", fieldNum, wire) + } + switch fieldNum { + case 1: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field RawProfile", wireType) + } + var byteLen int + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + byteLen |= int(b&0x7F) << shift + if b < 0x80 { + break + } + } + if byteLen < 0 { + return protohelpers.ErrInvalidLength + } + postIndex := iNdEx + byteLen + if postIndex < 0 { + return protohelpers.ErrInvalidLength + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + m.RawProfile = append(m.RawProfile[:0], dAtA[iNdEx:postIndex]...) + if m.RawProfile == nil { + m.RawProfile = []byte{} + } + iNdEx = postIndex + case 2: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field ID", wireType) + } + var stringLen uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + stringLen |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + intStringLen := int(stringLen) + if intStringLen < 0 { + return protohelpers.ErrInvalidLength + } + postIndex := iNdEx + intStringLen + if postIndex < 0 { + return protohelpers.ErrInvalidLength + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + m.ID = string(dAtA[iNdEx:postIndex]) + iNdEx = postIndex + default: + iNdEx = preIndex + skippy, err := protohelpers.Skip(dAtA[iNdEx:]) + if err != nil { + return err + } + if (skippy < 0) || (iNdEx+skippy) < 0 { + return protohelpers.ErrInvalidLength + } + if (iNdEx + skippy) > l { + return io.ErrUnexpectedEOF + } + m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...) + iNdEx += skippy + } + } + + if iNdEx > l { + return io.ErrUnexpectedEOF + } + return nil +} diff --git a/api/gen/proto/go/segmentwriter/v1/segmentwriterv1connect/push.connect.go b/api/gen/proto/go/segmentwriter/v1/segmentwriterv1connect/push.connect.go new file mode 100644 index 0000000000..24b00a76c0 --- /dev/null +++ b/api/gen/proto/go/segmentwriter/v1/segmentwriterv1connect/push.connect.go @@ -0,0 +1,114 @@ +// Code generated by protoc-gen-connect-go. DO NOT EDIT. +// +// Source: segmentwriter/v1/push.proto + +package segmentwriterv1connect + +import ( + connect "connectrpc.com/connect" + context "context" + errors "errors" + v1 "github.com/grafana/pyroscope/api/gen/proto/go/segmentwriter/v1" + http "net/http" + strings "strings" +) + +// This is a compile-time assertion to ensure that this generated file and the connect package are +// compatible. If you get a compiler error that this constant is not defined, this code was +// generated with a version of connect newer than the one compiled into your binary. You can fix the +// problem by either regenerating this code with an older version of connect or updating the connect +// version compiled into your binary. +const _ = connect.IsAtLeastVersion1_13_0 + +const ( + // SegmentWriterServiceName is the fully-qualified name of the SegmentWriterService service. + SegmentWriterServiceName = "segmentwriter.v1.SegmentWriterService" +) + +// These constants are the fully-qualified names of the RPCs defined in this package. They're +// exposed at runtime as Spec.Procedure and as the final two segments of the HTTP route. +// +// Note that these are different from the fully-qualified method names used by +// google.golang.org/protobuf/reflect/protoreflect. To convert from these constants to +// reflection-formatted method names, remove the leading slash and convert the remaining slash to a +// period. +const ( + // SegmentWriterServicePushProcedure is the fully-qualified name of the SegmentWriterService's Push + // RPC. + SegmentWriterServicePushProcedure = "/segmentwriter.v1.SegmentWriterService/Push" +) + +// These variables are the protoreflect.Descriptor objects for the RPCs defined in this package. +var ( + segmentWriterServiceServiceDescriptor = v1.File_segmentwriter_v1_push_proto.Services().ByName("SegmentWriterService") + segmentWriterServicePushMethodDescriptor = segmentWriterServiceServiceDescriptor.Methods().ByName("Push") +) + +// SegmentWriterServiceClient is a client for the segmentwriter.v1.SegmentWriterService service. +type SegmentWriterServiceClient interface { + Push(context.Context, *connect.Request[v1.PushRequest]) (*connect.Response[v1.PushResponse], error) +} + +// NewSegmentWriterServiceClient constructs a client for the segmentwriter.v1.SegmentWriterService +// service. By default, it uses the Connect protocol with the binary Protobuf Codec, asks for +// gzipped responses, and sends uncompressed requests. To use the gRPC or gRPC-Web protocols, supply +// the connect.WithGRPC() or connect.WithGRPCWeb() options. +// +// The URL supplied here should be the base URL for the Connect or gRPC server (for example, +// http://api.acme.com or https://acme.com/grpc). +func NewSegmentWriterServiceClient(httpClient connect.HTTPClient, baseURL string, opts ...connect.ClientOption) SegmentWriterServiceClient { + baseURL = strings.TrimRight(baseURL, "/") + return &segmentWriterServiceClient{ + push: connect.NewClient[v1.PushRequest, v1.PushResponse]( + httpClient, + baseURL+SegmentWriterServicePushProcedure, + connect.WithSchema(segmentWriterServicePushMethodDescriptor), + connect.WithClientOptions(opts...), + ), + } +} + +// segmentWriterServiceClient implements SegmentWriterServiceClient. +type segmentWriterServiceClient struct { + push *connect.Client[v1.PushRequest, v1.PushResponse] +} + +// Push calls segmentwriter.v1.SegmentWriterService.Push. +func (c *segmentWriterServiceClient) Push(ctx context.Context, req *connect.Request[v1.PushRequest]) (*connect.Response[v1.PushResponse], error) { + return c.push.CallUnary(ctx, req) +} + +// SegmentWriterServiceHandler is an implementation of the segmentwriter.v1.SegmentWriterService +// service. +type SegmentWriterServiceHandler interface { + Push(context.Context, *connect.Request[v1.PushRequest]) (*connect.Response[v1.PushResponse], error) +} + +// NewSegmentWriterServiceHandler builds an HTTP handler from the service implementation. It returns +// the path on which to mount the handler and the handler itself. +// +// By default, handlers support the Connect, gRPC, and gRPC-Web protocols with the binary Protobuf +// and JSON codecs. They also support gzip compression. +func NewSegmentWriterServiceHandler(svc SegmentWriterServiceHandler, opts ...connect.HandlerOption) (string, http.Handler) { + segmentWriterServicePushHandler := connect.NewUnaryHandler( + SegmentWriterServicePushProcedure, + svc.Push, + connect.WithSchema(segmentWriterServicePushMethodDescriptor), + connect.WithHandlerOptions(opts...), + ) + return "/segmentwriter.v1.SegmentWriterService/", http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { + switch r.URL.Path { + case SegmentWriterServicePushProcedure: + segmentWriterServicePushHandler.ServeHTTP(w, r) + default: + http.NotFound(w, r) + } + }) +} + +// UnimplementedSegmentWriterServiceHandler returns CodeUnimplemented from all methods. +type UnimplementedSegmentWriterServiceHandler struct{} + +func (UnimplementedSegmentWriterServiceHandler) Push(context.Context, *connect.Request[v1.PushRequest]) (*connect.Response[v1.PushResponse], error) { + return nil, connect.NewError(connect.CodeUnimplemented, errors.New("segmentwriter.v1.SegmentWriterService.Push is not implemented")) +} diff --git a/api/gen/proto/go/segmentwriter/v1/segmentwriterv1connect/push.connect.mux.go b/api/gen/proto/go/segmentwriter/v1/segmentwriterv1connect/push.connect.mux.go new file mode 100644 index 0000000000..a335402be4 --- /dev/null +++ b/api/gen/proto/go/segmentwriter/v1/segmentwriterv1connect/push.connect.mux.go @@ -0,0 +1,27 @@ +// Code generated by protoc-gen-connect-go-mux. DO NOT EDIT. +// +// Source: segmentwriter/v1/push.proto + +package segmentwriterv1connect + +import ( + connect "connectrpc.com/connect" + mux "github.com/gorilla/mux" +) + +// This is a compile-time assertion to ensure that this generated file and the connect package are +// compatible. If you get a compiler error that this constant is not defined, this code was +// generated with a version of connect newer than the one compiled into your binary. You can fix the +// problem by either regenerating this code with an older version of connect or updating the connect +// version compiled into your binary. +const _ = connect.IsAtLeastVersion0_1_0 + +// RegisterSegmentWriterServiceHandler register an HTTP handler to a mux.Router from the service +// implementation. +func RegisterSegmentWriterServiceHandler(mux *mux.Router, svc SegmentWriterServiceHandler, opts ...connect.HandlerOption) { + mux.Handle("/segmentwriter.v1.SegmentWriterService/Push", connect.NewUnaryHandler( + "/segmentwriter.v1.SegmentWriterService/Push", + svc.Push, + opts..., + )) +} diff --git a/api/metastore/v1/metastore.proto b/api/metastore/v1/metastore.proto new file mode 100644 index 0000000000..4574016a37 --- /dev/null +++ b/api/metastore/v1/metastore.proto @@ -0,0 +1,74 @@ +syntax = "proto3"; + +package metastore.v1; + +import "types/v1/types.proto"; + +service MetastoreService { + rpc AddBlock(AddBlockRequest) returns (AddBlockResponse) {} + rpc QueryMetadata(QueryMetadataRequest) returns (QueryMetadataResponse) {} + rpc ReadIndex(ReadIndexRequest) returns (ReadIndexResponse) {} +} + +message AddBlockRequest { + BlockMeta block = 1; +} + +message AddBlockResponse {} + +message BlockMeta { + uint64 format_version = 1; + string id = 2; + int64 min_time = 3; + int64 max_time = 4; + uint32 shard = 5; + uint32 compaction_level = 6; + // Optional. Empty if compaction level is 0. + string tenant_id = 7; + repeated Dataset datasets = 8; + uint64 size = 9; +} + +message Dataset { + repeated types.v1.Labels labels = 8; + + string tenant_id = 1; + string name = 2; + int64 min_time = 3; + int64 max_time = 4; + + // Table of contents lists data sections within the tenant + // service region. The offsets are absolute. + // + // The interpretation of the table of contents is specific + // to the metadata format version. By default, the sections are: + // - 0: profiles.parquet + // - 1: index.tsdb + // - 2: symbols.symdb + repeated uint64 table_of_contents = 5; + // Size of the section in bytes. + uint64 size = 6; + + // TODO: delete + // Profile types present in the tenant service data. + repeated string profile_types = 7; +} + +message QueryMetadataRequest { + repeated string tenant_id = 1; + int64 start_time = 2; + int64 end_time = 3; + string query = 4; +} + +message QueryMetadataResponse { + repeated BlockMeta blocks = 1; +} + +message ReadIndexRequest { + string debug_request_id = 1; // for debug logging, // todo delete +} + +message ReadIndexResponse { + uint64 read_index = 1; +} diff --git a/api/openapiv2/gen/phlare.swagger.json b/api/openapiv2/gen/phlare.swagger.json index 1557a38618..be50cd6e5d 100644 --- a/api/openapiv2/gen/phlare.swagger.json +++ b/api/openapiv2/gen/phlare.swagger.json @@ -8,6 +8,12 @@ { "name": "AdHocProfileService" }, + { + "name": "MetastoreService" + }, + { + "name": "CompactionPlanner" + }, { "name": "PusherService" }, @@ -17,6 +23,12 @@ { "name": "QuerierService" }, + { + "name": "QueryBackendService" + }, + { + "name": "SegmentWriterService" + }, { "name": "SettingsService" }, @@ -362,6 +374,46 @@ "additionalProperties": {}, "description": "`Any` contains an arbitrary serialized protocol buffer message along with a\nURL that describes the type of the serialized message.\n\nProtobuf library provides support to pack/unpack Any values in the form\nof utility functions or additional generated methods of the Any type.\n\nExample 1: Pack and unpack a message in C++.\n\n Foo foo = ...;\n Any any;\n any.PackFrom(foo);\n ...\n if (any.UnpackTo(\u0026foo)) {\n ...\n }\n\nExample 2: Pack and unpack a message in Java.\n\n Foo foo = ...;\n Any any = Any.pack(foo);\n ...\n if (any.is(Foo.class)) {\n foo = any.unpack(Foo.class);\n }\n // or ...\n if (any.isSameTypeAs(Foo.getDefaultInstance())) {\n foo = any.unpack(Foo.getDefaultInstance());\n }\n\n Example 3: Pack and unpack a message in Python.\n\n foo = Foo(...)\n any = Any()\n any.Pack(foo)\n ...\n if any.Is(Foo.DESCRIPTOR):\n any.Unpack(foo)\n ...\n\n Example 4: Pack and unpack a message in Go\n\n foo := \u0026pb.Foo{...}\n any, err := anypb.New(foo)\n if err != nil {\n ...\n }\n ...\n foo := \u0026pb.Foo{}\n if err := any.UnmarshalTo(foo); err != nil {\n ...\n }\n\nThe pack methods provided by protobuf library will by default use\n'type.googleapis.com/full.type.name' as the type URL and the unpack\nmethods only use the fully qualified type name after the last '/'\nin the type URL, for example \"foo.bar.com/x/y.z\" will yield type\nname \"y.z\".\n\nJSON\n====\nThe JSON representation of an `Any` value uses the regular\nrepresentation of the deserialized, embedded message, with an\nadditional field `@type` which contains the type URL. Example:\n\n package google.profile;\n message Person {\n string first_name = 1;\n string last_name = 2;\n }\n\n {\n \"@type\": \"type.googleapis.com/google.profile.Person\",\n \"firstName\": \u003cstring\u003e,\n \"lastName\": \u003cstring\u003e\n }\n\nIf the embedded message type is well-known and has a custom JSON\nrepresentation, that representation will be embedded adding a field\n`value` which holds the custom JSON in addition to the `@type`\nfield. Example (for message [google.protobuf.Duration][]):\n\n {\n \"@type\": \"type.googleapis.com/google.protobuf.Duration\",\n \"value\": \"1.212s\"\n }" }, + "pushv1PushResponse": { + "type": "object" + }, + "pushv1RawProfileSeries": { + "type": "object", + "properties": { + "labels": { + "type": "array", + "items": { + "type": "object", + "$ref": "#/definitions/v1LabelPair" + }, + "title": "LabelPair is the key value pairs to identify the corresponding profile" + }, + "samples": { + "type": "array", + "items": { + "type": "object", + "$ref": "#/definitions/pushv1RawSample" + }, + "title": "samples are the set of profile bytes" + } + }, + "title": "RawProfileSeries represents the pprof profile and its associated labels" + }, + "pushv1RawSample": { + "type": "object", + "properties": { + "rawProfile": { + "type": "string", + "format": "byte", + "title": "raw_profile is the set of bytes of the pprof profile" + }, + "ID": { + "type": "string", + "title": "unique ID of the profile" + } + }, + "title": "RawSample is the set of bytes that correspond to a pprof profile" + }, "querierv1ProfileTypesResponse": { "type": "object", "properties": { @@ -386,6 +438,46 @@ } } }, + "segmentwriterv1PushResponse": { + "type": "object" + }, + "segmentwriterv1RawProfileSeries": { + "type": "object", + "properties": { + "labels": { + "type": "array", + "items": { + "type": "object", + "$ref": "#/definitions/v1LabelPair" + }, + "title": "LabelPair is the key value pairs to identify the corresponding profile" + }, + "sample": { + "$ref": "#/definitions/segmentwriterv1RawSample", + "title": "samples are the set of profile bytes" + }, + "shard": { + "type": "integer", + "format": "int64" + } + }, + "title": "RawProfileSeries represents the pprof profile and its associated labels" + }, + "segmentwriterv1RawSample": { + "type": "object", + "properties": { + "rawProfile": { + "type": "string", + "format": "byte", + "title": "raw_profile is the set of bytes of the pprof profile" + }, + "ID": { + "type": "string", + "title": "unique ID of the profile" + } + }, + "title": "RawSample is the set of bytes that correspond to a pprof profile" + }, "typesv1Location": { "type": "object", "properties": { @@ -451,6 +543,9 @@ } } }, + "v1AddBlockResponse": { + "type": "object" + }, "v1AnalyzeQueryResponse": { "type": "object", "properties": { @@ -531,6 +626,49 @@ } } }, + "v1BlockMeta": { + "type": "object", + "properties": { + "formatVersion": { + "type": "string", + "format": "uint64" + }, + "id": { + "type": "string" + }, + "minTime": { + "type": "string", + "format": "int64" + }, + "maxTime": { + "type": "string", + "format": "int64" + }, + "shard": { + "type": "integer", + "format": "int64" + }, + "compactionLevel": { + "type": "integer", + "format": "int64" + }, + "tenantId": { + "type": "string", + "description": "Optional. Empty if compaction level is 0." + }, + "datasets": { + "type": "array", + "items": { + "type": "object", + "$ref": "#/definitions/v1Dataset" + } + }, + "size": { + "type": "string", + "format": "uint64" + } + } + }, "v1BlockMetadataResponse": { "type": "object", "properties": { @@ -586,6 +724,160 @@ } } }, + "v1CompactionJob": { + "type": "object", + "properties": { + "name": { + "type": "string", + "description": "Unique name of the job." + }, + "options": { + "$ref": "#/definitions/v1CompactionOptions" + }, + "blocks": { + "type": "array", + "items": { + "type": "object", + "$ref": "#/definitions/v1BlockMeta" + }, + "description": "List of the input blocks." + }, + "status": { + "$ref": "#/definitions/v1CompactionJobStatus" + }, + "raftLogIndex": { + "type": "string", + "format": "uint64", + "description": "Fencing token." + }, + "shard": { + "type": "integer", + "format": "int64", + "description": "Shard the blocks belong to." + }, + "tenantId": { + "type": "string", + "description": "Optional, empty for compaction level 0." + }, + "compactionLevel": { + "type": "integer", + "format": "int64" + } + }, + "description": "One compaction job may result in multiple output blocks." + }, + "v1CompactionJobStatus": { + "type": "object", + "properties": { + "jobName": { + "type": "string" + }, + "status": { + "$ref": "#/definitions/v1CompactionStatus", + "description": "Status update allows the planner to keep\ntrack of the job ownership and compaction\nprogress:\n- If the job status is other than IN_PROGRESS,\n the ownership of the job is revoked.\n- FAILURE must only be sent if the failure is\n persistent and the compaction can't be accomplished.\n- completed_job must be empty if the status is\n other than SUCCESS, and vice-versa.\n- UNSPECIFIED must be sent if the worker rejects\n or cancels the compaction job.\n\nPartial results/status is not allowed." + }, + "completedJob": { + "$ref": "#/definitions/v1CompletedJob" + }, + "raftLogIndex": { + "type": "string", + "format": "uint64", + "description": "Fencing token." + }, + "shard": { + "type": "integer", + "format": "int64", + "description": "Shard the blocks belong to." + }, + "tenantId": { + "type": "string", + "description": "Optional, empty for compaction level 0." + } + } + }, + "v1CompactionOptions": { + "type": "object", + "properties": { + "statusUpdateIntervalSeconds": { + "type": "string", + "format": "uint64", + "description": "How often the compaction worker should update\nthe job status. If overdue, the job ownership\nis revoked." + } + }, + "description": "Compaction planner should instruct the compactor\n worker how to compact the blocks:\n - Limits and tenant overrides.\n - Feature flags." + }, + "v1CompactionStatus": { + "type": "string", + "enum": [ + "COMPACTION_STATUS_UNSPECIFIED", + "COMPACTION_STATUS_IN_PROGRESS", + "COMPACTION_STATUS_SUCCESS", + "COMPACTION_STATUS_FAILURE" + ], + "default": "COMPACTION_STATUS_UNSPECIFIED" + }, + "v1CompletedJob": { + "type": "object", + "properties": { + "blocks": { + "type": "array", + "items": { + "type": "object", + "$ref": "#/definitions/v1BlockMeta" + } + } + } + }, + "v1Dataset": { + "type": "object", + "properties": { + "labels": { + "type": "array", + "items": { + "type": "object", + "$ref": "#/definitions/v1Labels" + } + }, + "tenantId": { + "type": "string" + }, + "name": { + "type": "string" + }, + "minTime": { + "type": "string", + "format": "int64" + }, + "maxTime": { + "type": "string", + "format": "int64" + }, + "tableOfContents": { + "type": "array", + "items": { + "type": "string", + "format": "uint64" + }, + "description": "Table of contents lists data sections within the tenant\nservice region. The offsets are absolute.\n\nThe interpretation of the table of contents is specific\nto the metadata format version. By default, the sections are:\n - 0: profiles.parquet\n - 1: index.tsdb\n - 2: symbols.symdb" + }, + "size": { + "type": "string", + "format": "uint64", + "description": "Size of the section in bytes." + }, + "profileTypes": { + "type": "array", + "items": { + "type": "string" + }, + "description": "TODO: delete\nProfile types present in the tenant service data." + } + } + }, + "v1Diagnostics": { + "type": "object", + "description": "Diagnostic messages, events, statistics, analytics, etc." + }, "v1DiffResponse": { "type": "object", "properties": { @@ -758,6 +1050,19 @@ } } }, + "v1GetCompactionResponse": { + "type": "object", + "properties": { + "compactionJobs": { + "type": "array", + "items": { + "type": "object", + "$ref": "#/definitions/v1CompactionJob" + }, + "title": "A list of all compaction jobs" + } + } + }, "v1GetFileResponse": { "type": "object", "properties": { @@ -849,6 +1154,42 @@ }, "title": "Hints are used to propagate information about querying" }, + "v1InvokeOptions": { + "type": "object", + "description": "Query workers might not have access to the tenant\n overrides, therefore all the necessary options should\n be listed in the request explicitly." + }, + "v1InvokeResponse": { + "type": "object", + "properties": { + "reports": { + "type": "array", + "items": { + "type": "object", + "$ref": "#/definitions/v1Report" + } + }, + "diagnostics": { + "$ref": "#/definitions/v1Diagnostics" + } + } + }, + "v1LabelNamesQuery": { + "type": "object" + }, + "v1LabelNamesReport": { + "type": "object", + "properties": { + "query": { + "$ref": "#/definitions/v1LabelNamesQuery" + }, + "labelNames": { + "type": "array", + "items": { + "type": "string" + } + } + } + }, "v1LabelNamesResponse": { "type": "object", "properties": { @@ -871,6 +1212,28 @@ } } }, + "v1LabelValuesQuery": { + "type": "object", + "properties": { + "labelName": { + "type": "string" + } + } + }, + "v1LabelValuesReport": { + "type": "object", + "properties": { + "query": { + "$ref": "#/definitions/v1LabelValuesQuery" + }, + "labelValues": { + "type": "array", + "items": { + "type": "string" + } + } + } + }, "v1LabelValuesResponse": { "type": "object", "properties": { @@ -1077,6 +1440,18 @@ } } }, + "v1PollCompactionJobsResponse": { + "type": "object", + "properties": { + "compactionJobs": { + "type": "array", + "items": { + "type": "object", + "$ref": "#/definitions/v1CompactionJob" + } + } + } + }, "v1ProfileFormat": { "type": "string", "enum": [ @@ -1136,8 +1511,30 @@ } } }, - "v1PushResponse": { - "type": "object" + "v1Query": { + "type": "object", + "properties": { + "queryType": { + "$ref": "#/definitions/v1QueryType" + }, + "labelNames": { + "$ref": "#/definitions/v1LabelNamesQuery", + "description": "Exactly one of the following fields should be set,\ndepending on the query type." + }, + "labelValues": { + "$ref": "#/definitions/v1LabelValuesQuery" + }, + "seriesLabels": { + "$ref": "#/definitions/v1SeriesLabelsQuery" + }, + "timeSeries": { + "$ref": "#/definitions/v1TimeSeriesQuery" + }, + "tree": { + "$ref": "#/definitions/v1TreeQuery", + "description": "pprof\n function_details\n call_graph\n top_table\n ..." + } + } }, "v1QueryImpact": { "type": "object", @@ -1155,6 +1552,40 @@ } } }, + "v1QueryMetadataResponse": { + "type": "object", + "properties": { + "blocks": { + "type": "array", + "items": { + "type": "object", + "$ref": "#/definitions/v1BlockMeta" + } + } + } + }, + "v1QueryPlan": { + "type": "object", + "properties": { + "graph": { + "type": "array", + "items": { + "type": "integer", + "format": "int64" + }, + "description": "Each node is encoded with 3 values:\n - node type: 0 - read, 1 - merge;\n - range offset;\n - range length." + }, + "blocks": { + "type": "array", + "items": { + "type": "object", + "$ref": "#/definitions/v1BlockMeta" + }, + "description": "The blocks matching the query.\n\nTODO: Use opaque byte array to avoid unnecessary\n proto encoding/decoding overhead in transient nodes.\n Graph nodes reference ranges, thus each range could\n be a blob of bytes:\n repeated bytes block_ranges = 2;" + } + }, + "description": "Query plan is represented by a DAG, where each node\nmight be either \"merge\" or \"read\" (leaves). Each node\nreferences a range: merge nodes refer to other nodes,\nwhile read nodes refer to the blocks." + }, "v1QueryScope": { "type": "object", "properties": { @@ -1197,42 +1628,62 @@ } } }, - "v1RawProfileSeries": { + "v1QueryType": { + "type": "string", + "enum": [ + "QUERY_UNSPECIFIED", + "QUERY_LABEL_NAMES", + "QUERY_LABEL_VALUES", + "QUERY_SERIES_LABELS", + "QUERY_TIME_SERIES", + "QUERY_TREE" + ], + "default": "QUERY_UNSPECIFIED" + }, + "v1ReadIndexResponse": { "type": "object", "properties": { - "labels": { - "type": "array", - "items": { - "type": "object", - "$ref": "#/definitions/v1LabelPair" - }, - "title": "LabelPair is the key value pairs to identify the corresponding profile" - }, - "samples": { - "type": "array", - "items": { - "type": "object", - "$ref": "#/definitions/v1RawSample" - }, - "title": "samples are the set of profile bytes" + "readIndex": { + "type": "string", + "format": "uint64" } - }, - "title": "RawProfileSeries represents the pprof profile and its associated labels" + } }, - "v1RawSample": { + "v1Report": { "type": "object", "properties": { - "rawProfile": { - "type": "string", - "format": "byte", - "title": "raw_profile is the set of bytes of the pprof profile" + "reportType": { + "$ref": "#/definitions/v1ReportType" }, - "ID": { - "type": "string", - "title": "unique ID of the profile" + "labelNames": { + "$ref": "#/definitions/v1LabelNamesReport", + "description": "Exactly one of the following fields should be set,\ndepending on the report type." + }, + "labelValues": { + "$ref": "#/definitions/v1LabelValuesReport" + }, + "seriesLabels": { + "$ref": "#/definitions/v1SeriesLabelsReport" + }, + "timeSeries": { + "$ref": "#/definitions/v1TimeSeriesReport" + }, + "tree": { + "$ref": "#/definitions/v1TreeReport" } - }, - "title": "RawSample is the set of bytes that correspond to a pprof profile" + } + }, + "v1ReportType": { + "type": "string", + "enum": [ + "REPORT_UNSPECIFIED", + "REPORT_LABEL_NAMES", + "REPORT_LABEL_VALUES", + "REPORT_SERIES_LABELS", + "REPORT_TIME_SERIES", + "REPORT_TREE" + ], + "default": "REPORT_UNSPECIFIED" }, "v1Sample": { "type": "object", @@ -1412,6 +1863,32 @@ } } }, + "v1SeriesLabelsQuery": { + "type": "object", + "properties": { + "labelNames": { + "type": "array", + "items": { + "type": "string" + } + } + } + }, + "v1SeriesLabelsReport": { + "type": "object", + "properties": { + "query": { + "$ref": "#/definitions/v1SeriesLabelsQuery" + }, + "seriesLabels": { + "type": "array", + "items": { + "type": "object", + "$ref": "#/definitions/v1Labels" + } + } + } + }, "v1SeriesProfile": { "type": "object", "properties": { @@ -1501,6 +1978,60 @@ ], "default": "TIME_SERIES_AGGREGATION_TYPE_SUM" }, + "v1TimeSeriesQuery": { + "type": "object", + "properties": { + "step": { + "type": "number", + "format": "double" + }, + "groupBy": { + "type": "array", + "items": { + "type": "string" + } + }, + "aggregation": { + "$ref": "#/definitions/v1TimeSeriesAggregationType" + } + } + }, + "v1TimeSeriesReport": { + "type": "object", + "properties": { + "query": { + "$ref": "#/definitions/v1TimeSeriesQuery" + }, + "timeSeries": { + "type": "array", + "items": { + "type": "object", + "$ref": "#/definitions/v1Series" + } + } + } + }, + "v1TreeQuery": { + "type": "object", + "properties": { + "maxNodes": { + "type": "string", + "format": "int64" + } + } + }, + "v1TreeReport": { + "type": "object", + "properties": { + "query": { + "$ref": "#/definitions/v1TreeQuery" + }, + "tree": { + "type": "string", + "format": "byte" + } + } + }, "v1ValueType": { "type": "object", "properties": { diff --git a/api/querybackend/v1/querybackend.proto b/api/querybackend/v1/querybackend.proto new file mode 100644 index 0000000000..c368f7a633 --- /dev/null +++ b/api/querybackend/v1/querybackend.proto @@ -0,0 +1,145 @@ +syntax = "proto3"; + +package querybackend.v1; + +import "google/v1/profile.proto"; +import "metastore/v1/metastore.proto"; +import "types/v1/types.proto"; + +service QueryBackendService { + rpc Invoke(InvokeRequest) returns (InvokeResponse) {} +} + +message InvokeOptions { + // Query workers might not have access to the tenant + // overrides, therefore all the necessary options should + // be listed in the request explicitly. +} + +message InvokeRequest { + repeated string tenant = 1; + int64 start_time = 2; + int64 end_time = 3; + string label_selector = 4; + repeated Query query = 5; + QueryPlan query_plan = 6; + InvokeOptions options = 7; +} + +// Query plan is represented by a DAG, where each node +// might be either "merge" or "read" (leaves). Each node +// references a range: merge nodes refer to other nodes, +// while read nodes refer to the blocks. +message QueryPlan { + // Each node is encoded with 3 values: + // - node type: 0 - read, 1 - merge; + // - range offset; + // - range length. + repeated uint32 graph = 1; + + // The blocks matching the query. + repeated metastore.v1.BlockMeta blocks = 2; + // TODO: Use opaque byte array to avoid unnecessary + // proto encoding/decoding overhead in transient nodes. + // Graph nodes reference ranges, thus each range could + // be a blob of bytes: + // repeated bytes block_ranges = 2; +} + +message Query { + QueryType query_type = 1; + // Exactly one of the following fields should be set, + // depending on the query type. + LabelNamesQuery label_names = 2; + LabelValuesQuery label_values = 3; + SeriesLabelsQuery series_labels = 4; + TimeSeriesQuery time_series = 5; + TreeQuery tree = 6; + // pprof + // function_details + // call_graph + // top_table + // ... +} + +enum QueryType { + QUERY_UNSPECIFIED = 0; + QUERY_LABEL_NAMES = 1; + QUERY_LABEL_VALUES = 2; + QUERY_SERIES_LABELS = 3; + QUERY_TIME_SERIES = 4; + QUERY_TREE = 5; +} + +message InvokeResponse { + repeated Report reports = 1; + Diagnostics diagnostics = 2; +} + +// Diagnostic messages, events, statistics, analytics, etc. +message Diagnostics {} + +message Report { + ReportType report_type = 1; + // Exactly one of the following fields should be set, + // depending on the report type. + LabelNamesReport label_names = 2; + LabelValuesReport label_values = 3; + SeriesLabelsReport series_labels = 4; + TimeSeriesReport time_series = 5; + TreeReport tree = 6; +} + +enum ReportType { + REPORT_UNSPECIFIED = 0; + REPORT_LABEL_NAMES = 1; + REPORT_LABEL_VALUES = 2; + REPORT_SERIES_LABELS = 3; + REPORT_TIME_SERIES = 4; + REPORT_TREE = 5; +} + +message LabelNamesQuery {} + +message LabelNamesReport { + LabelNamesQuery query = 1; + repeated string label_names = 2; +} + +message LabelValuesQuery { + string label_name = 1; +} + +message LabelValuesReport { + LabelValuesQuery query = 1; + repeated string label_values = 2; +} + +message SeriesLabelsQuery { + repeated string label_names = 1; +} + +message SeriesLabelsReport { + SeriesLabelsQuery query = 1; + repeated types.v1.Labels series_labels = 2; +} + +message TimeSeriesQuery { + double step = 1; + repeated string group_by = 2; + optional types.v1.TimeSeriesAggregationType aggregation = 3; +} + +message TimeSeriesReport { + TimeSeriesQuery query = 1; + repeated types.v1.Series time_series = 2; +} + +message TreeQuery { + int64 max_nodes = 1; +} + +message TreeReport { + TreeQuery query = 1; + bytes tree = 2; +} diff --git a/api/segmentwriter/v1/push.proto b/api/segmentwriter/v1/push.proto new file mode 100644 index 0000000000..8a13966f8c --- /dev/null +++ b/api/segmentwriter/v1/push.proto @@ -0,0 +1,36 @@ +syntax = "proto3"; + +package segmentwriter.v1; + +import "types/v1/types.proto"; + +service SegmentWriterService { + rpc Push(PushRequest) returns (PushResponse) {} +} + +message PushResponse {} + +// WriteRawRequest writes a pprof profile +message PushRequest { + // series is a set raw pprof profiles and accompanying labels + RawProfileSeries series = 1; +} + +// RawProfileSeries represents the pprof profile and its associated labels +message RawProfileSeries { + // LabelPair is the key value pairs to identify the corresponding profile + repeated types.v1.LabelPair labels = 1; + + // samples are the set of profile bytes + RawSample sample = 2; + + uint32 shard = 3; +} + +// RawSample is the set of bytes that correspond to a pprof profile +message RawSample { + // raw_profile is the set of bytes of the pprof profile + bytes raw_profile = 1; + // unique ID of the profile + string ID = 2; +} diff --git a/docs/sources/configure-client/grafana-agent/ebpf/setup-docker.md b/docs/sources/configure-client/grafana-agent/ebpf/setup-docker.md index 500d77d7be..02ac9f7cd2 100644 --- a/docs/sources/configure-client/grafana-agent/ebpf/setup-docker.md +++ b/docs/sources/configure-client/grafana-agent/ebpf/setup-docker.md @@ -107,6 +107,6 @@ docker run \ To verify that the profiles are received by the Pyroscope server, go to the Pyroscope UI or [Grafana Pyroscope datasource][pyroscope-ds]. Then select a profile type and a service from the dropdown menu. -[pyroscope-ds]: /docs/grafana/latest/datasources/grafana-pyroscope/ +[pyroscope-ds]: /docs/grafana//datasources/pyroscope/ [config-reference]: ../configuration/ [gcloud]: /products/cloud/ diff --git a/docs/sources/configure-client/grafana-agent/ebpf/setup-kubernetes.md b/docs/sources/configure-client/grafana-agent/ebpf/setup-kubernetes.md index 377a040e17..a418acc983 100644 --- a/docs/sources/configure-client/grafana-agent/ebpf/setup-kubernetes.md +++ b/docs/sources/configure-client/grafana-agent/ebpf/setup-kubernetes.md @@ -127,5 +127,5 @@ To verify that the profiles are received by the Pyroscope server, go to the Pyro [gcloud]: /products/cloud/ [helm]: https://helm.sh/docs/intro/install/ [kubectl]: https://kubernetes.io/docs/tasks/tools/install-kubectl/ -[pyroscope-ds]: /docs/grafana/latest/datasources/grafana-pyroscope/ +[pyroscope-ds]: /docs/grafana//datasources/pyroscope/ [config-reference]: ../configuration/ diff --git a/docs/sources/configure-client/grafana-agent/ebpf/setup-linux.md b/docs/sources/configure-client/grafana-agent/ebpf/setup-linux.md index 7201ca653d..97fbc513e3 100644 --- a/docs/sources/configure-client/grafana-agent/ebpf/setup-linux.md +++ b/docs/sources/configure-client/grafana-agent/ebpf/setup-linux.md @@ -136,7 +136,7 @@ Make sure you're running the agent with root privileges which are required for t To verify that the profiles are received by the Pyroscope server, go to the Pyroscope UI or [Grafana Pyroscope datasource][pyroscope-ds]. Select a profile type and a service from the drop-down menu. [agent-install]: /docs/agent/latest/flow/setup/install/linux/ -[pyroscope-ds]: /docs/grafana/latest/datasources/grafana-pyroscope/ +[pyroscope-ds]: /docs/grafana//datasources/pyroscope/ [config-reference]: ../configuration/ [gcloud]: /products/cloud/ [discovery.process](/docs/agent/next/flow/reference/components/discovery.process/) diff --git a/docs/sources/configure-client/grafana-agent/go_pull.md b/docs/sources/configure-client/grafana-agent/go_pull.md index 40790a51ed..776b67d4b1 100644 --- a/docs/sources/configure-client/grafana-agent/go_pull.md +++ b/docs/sources/configure-client/grafana-agent/go_pull.md @@ -240,12 +240,12 @@ router.PathPrefix("/debug/pprof").Handler(http.DefaultServeMux) ### Grafana Alloy -- [Grafana Alloy](https://grafana.com/docs/alloy/latest/) -- [pyroscope.scrape](https://grafana.com/docs/alloy/latest/reference/components/pyroscope/pyroscope.scrape/) -- [pyroscope.write](https://grafana.com/docs/alloy/latest/reference/components/pyroscope/pyroscope.write/) -- [discovery.kubernetes](https://grafana.com/docs/alloy/latest/reference/components/discovery/discovery.kubernetes/) -- [discovery.docker](/docs/alloy/latest/flow/reference/components/discovery/discovery.docker/) -- [discovery.relabel](/docs/alloy/latest/flow/reference/components/discovery/discovery.relabel/) +- [Grafana Alloy](https://grafana.com/docs/alloy//) +- [pyroscope.scrape](https://grafana.com/docs/alloy//reference/components/pyroscope/pyroscope.scrape/) +- [pyroscope.write](https://grafana.com/docs/alloy//reference/components/pyroscope/pyroscope.write/) +- [discovery.kubernetes](https://grafana.com/docs/alloy//reference/components/discovery/discovery.kubernetes/) +- [discovery.docker](/docs/alloy//reference/components/discovery/discovery.docker/) +- [discovery.relabel](/docs/alloy//reference/components/discovery/discovery.relabel/) ### Grafana Agent diff --git a/docs/sources/configure-client/grafana-agent/java/_index.md b/docs/sources/configure-client/grafana-agent/java/_index.md index 7ea5c5f336..f1d5241a13 100644 --- a/docs/sources/configure-client/grafana-agent/java/_index.md +++ b/docs/sources/configure-client/grafana-agent/java/_index.md @@ -291,12 +291,12 @@ For more information: ### Grafana Alloy -- [Grafana Alloy](https://grafana.com/docs/alloy/latest/) -- [pyroscope.scrape](https://grafana.com/docs/alloy/latest/reference/components/pyroscope/pyroscope.scrape/) -- [pyroscope.write](https://grafana.com/docs/alloy/latest/reference/components/pyroscope/pyroscope.write/) -- [discovery.kubernetes](https://grafana.com/docs/alloy/latest/reference/components/discovery/discovery.kubernetes/) -- [discovery.docker](/docs/alloy/latest/flow/reference/components/discovery/discovery.docker/) -- [discovery.relabel](/docs/alloy/latest/flow/reference/components/discovery/discovery.relabel/) +- [Grafana Alloy](https://grafana.com/docs/alloy//) +- [pyroscope.scrape](https://grafana.com/docs/alloy//reference/components/pyroscope/pyroscope.scrape/) +- [pyroscope.write](https://grafana.com/docs/alloy//reference/components/pyroscope/pyroscope.write/) +- [discovery.kubernetes](https://grafana.com/docs/alloy//reference/components/discovery/discovery.kubernetes/) +- [discovery.docker](/docs/alloy//reference/components/discovery/discovery.docker/) +- [discovery.relabel](/docs/alloy//reference/components/discovery/discovery.relabel/) ### Grafana Agent diff --git a/docs/sources/configure-client/language-sdks/dotnet.md b/docs/sources/configure-client/language-sdks/dotnet.md index dc626da665..71962698d8 100644 --- a/docs/sources/configure-client/language-sdks/dotnet.md +++ b/docs/sources/configure-client/language-sdks/dotnet.md @@ -24,6 +24,7 @@ The .NET Profiler supports the following profiling types: * Allocations * Lock contention * Exceptions +* Live heap (requires .NET 7+) ### Compatibility @@ -44,13 +45,13 @@ The Pyroscope server can be a local server for development or a remote server fo 1. Obtain `Pyroscope.Profiler.Native.so` and `Pyroscope.Linux.ApiWrapper.x64.so` from the [latest tarball](https://github.com/pyroscope-io/pyroscope-dotnet/releases/): ```bash -curl -s -L https://github.com/grafana/pyroscope-dotnet/releases/download/v0.8.14-pyroscope/pyroscope.0.8.14-glibc-x86_64.tar.gz | tar xvz -C . +curl -s -L https://github.com/grafana/pyroscope-dotnet/releases/download/v0.8.19-pyroscope/pyroscope.0.8.19-glibc-x86_64.tar.gz | tar xvz -C . ``` Or copy them from the [latest docker image](https://hub.docker.com/r/pyroscope/pyroscope-dotnet/tags). We have `glibc` and `musl` versions: ```dockerfile -COPY --from=pyroscope/pyroscope-dotnet:0.8.14-glibc /Pyroscope.Profiler.Native.so ./Pyroscope.Profiler.Native.so -COPY --from=pyroscope/pyroscope-dotnet:0.8.14-glibc /Pyroscope.Linux.ApiWrapper.x64.so ./Pyroscope.Linux.ApiWrapper.x64.so +COPY --from=pyroscope/pyroscope-dotnet:0.8.19-glibc /Pyroscope.Profiler.Native.so ./Pyroscope.Profiler.Native.so +COPY --from=pyroscope/pyroscope-dotnet:0.8.19-glibc /Pyroscope.Linux.ApiWrapper.x64.so ./Pyroscope.Linux.ApiWrapper.x64.so ```` 2. Set the following required environment variables to enable profiler @@ -171,6 +172,7 @@ Here is a simple [example](https://github.com/grafana/pyroscope/blob/main/exampl | PYROSCOPE_PROFILING_EXCEPTION_ENABLED | Boolean | If set to true, enables the Exceptions profiling. Defaults to false. | | PYROSCOPE_PROFILING_ALLOCATION_ENABLED | Boolean | If set to true, enables the Allocations profiling. Defaults to false. | | PYROSCOPE_PROFILING_LOCK_ENABLED | Boolean | If set to true, enables the Lock Contention profiling. Defaults to false. | +| PYROSCOPE_PROFILING_HEAP_ENABLED | Boolean | If set to true, enables the Live heap profiling. Requires .NET 7+. Defaults to false. | | PYROSCOPE_BASIC_AUTH_USER | String | For HTTP Basic Authentication, use this to send profiles to authenticated server, for example Grafana Cloud | | PYROSCOPE_BASIC_AUTH_PASSWORD | String | For HTTP Basic Authentication, use this to send profiles to authenticated server, for example Grafana Cloud | | PYROSCOPE_TENANT_ID | String | Only needed if using multi-tenancy in Pyroscope. | diff --git a/docs/sources/configure-client/language-sdks/java.md b/docs/sources/configure-client/language-sdks/java.md index a708215bfc..43c478b52a 100644 --- a/docs/sources/configure-client/language-sdks/java.md +++ b/docs/sources/configure-client/language-sdks/java.md @@ -51,12 +51,12 @@ First, add the Pyroscope dependency: io.pyroscope agent - 0.13.1 + 0.14.0 ``` ```gradle -implementation("io.pyroscope:agent:0.13.1") +implementation("io.pyroscope:agent:0.14.0") ``` {{< /code >}} diff --git a/docs/sources/view-and-analyze-profile-data/_index.md b/docs/sources/view-and-analyze-profile-data/_index.md index 6c78a8c2c8..352d4805e8 100644 --- a/docs/sources/view-and-analyze-profile-data/_index.md +++ b/docs/sources/view-and-analyze-profile-data/_index.md @@ -40,4 +40,4 @@ For more information on using profiles in Grafana, refer to [Pyroscope and profi The Pyroscope app plugin works for Grafana Cloud. -For more information on configuring these data sources, refer to the Pyroscope data source documentation in [Grafana Cloud](/docs/grafana-cloud/connect-externally-hosted/data-sources/pyroscope/) and [Grafana](/docs/grafana/latest/datasources/grafana-pyroscope/). +For more information on configuring these data sources, refer to the Pyroscope data source documentation in [Grafana Cloud](/docs/grafana-cloud/connect-externally-hosted/data-sources/pyroscope/) and [Grafana](/docs/grafana//datasources/pyroscope/). diff --git a/docs/sources/view-and-analyze-profile-data/profiling-types/_index.md b/docs/sources/view-and-analyze-profile-data/profiling-types/_index.md index 97e36b016e..c8b84f2a83 100644 --- a/docs/sources/view-and-analyze-profile-data/profiling-types/_index.md +++ b/docs/sources/view-and-analyze-profile-data/profiling-types/_index.md @@ -27,23 +27,23 @@ For information on auto-instrumentation and supported language SDKs, refer to [C Various languages support different profiling types. Pyroscope supports the following profiling types: -| Profile Type | Go | Java | .NET | Ruby | Python | Rust | Node.js | eBPF (Go) | eBPF (Python) | -|--------------------|-------|-------|-------|-------|--------|-------|---------|-----------|--------------| -| CPU | Yes | Yes | Yes | Yes | Yes | Yes | Yes | Yes | Yes | -| Alloc Objects | Yes | Yes | Yes | | | | | | | -| Alloc Space | Yes | Yes | Yes | | | | | | | -| Inuse Objects | Yes | | | | | | | | | -| Inuse Space | Yes | | | | | | | | | -| Goroutines | Yes | | | | | | | | | -| Mutex Count | Yes | | Yes | | | | | | | -| Mutex Duration | Yes | | Yes | | | | | | | -| Block Count | Yes | | | | | | | | | -| Block Duration | Yes | | | | | | | | | -| Lock Count | | Yes | Yes | | | | | | | -| Lock Duration | | Yes | Yes | | | | | | | -| Exceptions | | | Yes | | | | | | | -| Wall | | | Yes | | | | | | | -| Heap | | | | | | | Yes | | | +| Profile Type | Go | Java | .NET | Ruby | Python | Rust | Node.js | eBPF (Go) | eBPF (Python)| +|--------------------|-------|-------|------------|-------|--------|-------|---------|-----------|--------------| +| CPU | Yes | Yes | Yes | Yes | Yes | Yes | Yes | Yes | Yes | +| Alloc Objects | Yes | Yes | Yes | | | | | | | +| Alloc Space | Yes | Yes | Yes | | | | | | | +| Inuse Objects | Yes | | Yes (7.0+) | | | | | | | +| Inuse Space | Yes | | Yes (7.0+) | | | | | | | +| Goroutines | Yes | | | | | | | | | +| Mutex Count | Yes | | Yes | | | | | | | +| Mutex Duration | Yes | | Yes | | | | | | | +| Block Count | Yes | | | | | | | | | +| Block Duration | Yes | | | | | | | | | +| Lock Count | | Yes | Yes | | | | | | | +| Lock Duration | | Yes | Yes | | | | | | | +| Exceptions | | | Yes | | | | | | | +| Wall | | | Yes | | | | | | | +| Heap | | | | | | | Yes | | | ## CPU profiling diff --git a/examples/language-sdk-instrumentation/dotnet/fast-slow/Dockerfile b/examples/language-sdk-instrumentation/dotnet/fast-slow/Dockerfile index 659b978978..3babe9d89d 100644 --- a/examples/language-sdk-instrumentation/dotnet/fast-slow/Dockerfile +++ b/examples/language-sdk-instrumentation/dotnet/fast-slow/Dockerfile @@ -2,8 +2,8 @@ FROM mcr.microsoft.com/dotnet/sdk:6.0 WORKDIR /dotnet -COPY --from=pyroscope/pyroscope-dotnet:0.8.14-glibc /Pyroscope.Profiler.Native.so ./Pyroscope.Profiler.Native.so -COPY --from=pyroscope/pyroscope-dotnet:0.8.14-glibc /Pyroscope.Linux.ApiWrapper.x64.so ./Pyroscope.Linux.ApiWrapper.x64.so +COPY --from=pyroscope/pyroscope-dotnet:0.8.19-glibc /Pyroscope.Profiler.Native.so ./Pyroscope.Profiler.Native.so +COPY --from=pyroscope/pyroscope-dotnet:0.8.19-glibc /Pyroscope.Linux.ApiWrapper.x64.so ./Pyroscope.Linux.ApiWrapper.x64.so ADD example . diff --git a/examples/language-sdk-instrumentation/dotnet/fast-slow/musl.Dockerfile b/examples/language-sdk-instrumentation/dotnet/fast-slow/musl.Dockerfile index 6ae9fe8a5a..38bdebacbb 100644 --- a/examples/language-sdk-instrumentation/dotnet/fast-slow/musl.Dockerfile +++ b/examples/language-sdk-instrumentation/dotnet/fast-slow/musl.Dockerfile @@ -2,8 +2,8 @@ FROM mcr.microsoft.com/dotnet/sdk:6.0-alpine WORKDIR /dotnet -COPY --from=pyroscope/pyroscope-dotnet:0.8.14-musl /Pyroscope.Profiler.Native.so ./Pyroscope.Profiler.Native.so -COPY --from=pyroscope/pyroscope-dotnet:0.8.14-musl /Pyroscope.Linux.ApiWrapper.x64.so ./Pyroscope.Linux.ApiWrapper.x64.so +COPY --from=pyroscope/pyroscope-dotnet:0.8.19-musl /Pyroscope.Profiler.Native.so ./Pyroscope.Profiler.Native.so +COPY --from=pyroscope/pyroscope-dotnet:0.8.19-musl /Pyroscope.Linux.ApiWrapper.x64.so ./Pyroscope.Linux.ApiWrapper.x64.so ADD example . diff --git a/examples/language-sdk-instrumentation/dotnet/rideshare/Dockerfile b/examples/language-sdk-instrumentation/dotnet/rideshare/Dockerfile index e7b8f0a9a1..0295d3e416 100644 --- a/examples/language-sdk-instrumentation/dotnet/rideshare/Dockerfile +++ b/examples/language-sdk-instrumentation/dotnet/rideshare/Dockerfile @@ -1,14 +1,33 @@ -FROM mcr.microsoft.com/dotnet/sdk:6.0 +ARG SDK_VERSION=8.0 +# The build images takes an SDK image of the buildplatform, so the platform the build is running on. +FROM --platform=$BUILDPLATFORM mcr.microsoft.com/dotnet/sdk:$SDK_VERSION AS build + +ARG TARGETPLATFORM +ARG BUILDPLATFORM +ARG SDK_VERSION WORKDIR /dotnet -COPY --from=pyroscope/pyroscope-dotnet:0.8.14-glibc /Pyroscope.Profiler.Native.so ./Pyroscope.Profiler.Native.so -COPY --from=pyroscope/pyroscope-dotnet:0.8.14-glibc /Pyroscope.Linux.ApiWrapper.x64.so ./Pyroscope.Linux.ApiWrapper.x64.so +ADD example . + +# Set the target framework to SDK_VERSION +RUN sed -i -E 's|.*|net'$SDK_VERSION'|' Example.csproj +# We hardcode linux-x64 here, as the profiler doesn't support any other platform +RUN dotnet publish -o . --framework net$SDK_VERSION --runtime linux-x64 --no-self-contained -ADD example . +# This fetches the SDK +FROM --platform=linux/amd64 pyroscope/pyroscope-dotnet:0.8.20-glibc AS sdk + +# Runtime only image of the targetplatfrom, so the platform the image will be running on. +FROM --platform=linux/amd64 mcr.microsoft.com/dotnet/aspnet:$SDK_VERSION + +WORKDIR /dotnet + +COPY --from=sdk /Pyroscope.Profiler.Native.so ./Pyroscope.Profiler.Native.so +COPY --from=sdk /Pyroscope.Linux.ApiWrapper.x64.so ./Pyroscope.Linux.ApiWrapper.x64.so +COPY --from=build /dotnet/ ./ -RUN dotnet publish -o . -r $(dotnet --info | grep RID | cut -b 6- | tr -d ' ') ENV CORECLR_ENABLE_PROFILING=1 ENV CORECLR_PROFILER={BD1A650D-AC5D-4896-B64F-D6FA25D6B26A} @@ -22,6 +41,7 @@ ENV PYROSCOPE_PROFILING_ENABLED=1 ENV PYROSCOPE_PROFILING_ALLOCATION_ENABLED=true ENV PYROSCOPE_PROFILING_CONTENTION_ENABLED=true ENV PYROSCOPE_PROFILING_EXCEPTION_ENABLED=true +ENV PYROSCOPE_PROFILING_HEAP_ENABLED=true ENV RIDESHARE_LISTEN_PORT=5000 diff --git a/examples/language-sdk-instrumentation/dotnet/rideshare/docker-compose.yml b/examples/language-sdk-instrumentation/dotnet/rideshare/docker-compose.yml index 2a08555941..61d07cc424 100644 --- a/examples/language-sdk-instrumentation/dotnet/rideshare/docker-compose.yml +++ b/examples/language-sdk-instrumentation/dotnet/rideshare/docker-compose.yml @@ -5,7 +5,6 @@ services: ports: - 4040:4040 us-east: - platform: linux/amd64 ports: - 5000 environment: @@ -16,7 +15,6 @@ services: build: context: . eu-north: - platform: linux/amd64 ports: - 5000 environment: @@ -26,8 +24,9 @@ services: - RIDESHARE_LISTEN_PORT=5000 build: context: . + args: + SDK_VERSION: "6.0" ap-south: - platform: linux/amd64 ports: - 5000 environment: @@ -38,7 +37,6 @@ services: build: context: . ap-south-alpine: - platform: linux/amd64 ports: - 5000 environment: diff --git a/examples/language-sdk-instrumentation/dotnet/rideshare/example/Example.csproj b/examples/language-sdk-instrumentation/dotnet/rideshare/example/Example.csproj index 95e3a743d9..4f41aee696 100644 --- a/examples/language-sdk-instrumentation/dotnet/rideshare/example/Example.csproj +++ b/examples/language-sdk-instrumentation/dotnet/rideshare/example/Example.csproj @@ -1,6 +1,6 @@ - net6.0 + net8.0 example Exe example diff --git a/examples/language-sdk-instrumentation/dotnet/rideshare/musl.Dockerfile b/examples/language-sdk-instrumentation/dotnet/rideshare/musl.Dockerfile index 15e29b6f04..e94e70ee07 100644 --- a/examples/language-sdk-instrumentation/dotnet/rideshare/musl.Dockerfile +++ b/examples/language-sdk-instrumentation/dotnet/rideshare/musl.Dockerfile @@ -1,14 +1,33 @@ -FROM mcr.microsoft.com/dotnet/sdk:6.0-alpine +ARG SDK_VERSION=8.0 +# The build images takes an SDK image of the buildplatform, so the platform the build is running on. +FROM --platform=$BUILDPLATFORM mcr.microsoft.com/dotnet/sdk:$SDK_VERSION-alpine AS build + +ARG TARGETPLATFORM +ARG BUILDPLATFORM +ARG SDK_VERSION WORKDIR /dotnet -COPY --from=pyroscope/pyroscope-dotnet:0.8.14-musl /Pyroscope.Profiler.Native.so ./Pyroscope.Profiler.Native.so -COPY --from=pyroscope/pyroscope-dotnet:0.8.14-musl /Pyroscope.Linux.ApiWrapper.x64.so ./Pyroscope.Linux.ApiWrapper.x64.so +ADD example . +# Set the target framework to SDK_VERSION +RUN sed -i -E 's|.*|net'$SDK_VERSION'|' Example.csproj -ADD example . +# We hardcode linux-x64 here, as the profiler doesn't support any other platform +RUN dotnet publish -o . --framework net$SDK_VERSION --runtime linux-musl-x64 --no-self-contained + +# This fetches the SDK +FROM --platform=linux/amd64 pyroscope/pyroscope-dotnet:0.8.20-musl AS sdk + +# Runtime only image of the targetplatfrom, so the platform the image will be running on. +FROM --platform=linux/amd64 mcr.microsoft.com/dotnet/aspnet:$SDK_VERSION-alpine + +WORKDIR /dotnet + +COPY --from=sdk /Pyroscope.Profiler.Native.so ./Pyroscope.Profiler.Native.so +COPY --from=sdk /Pyroscope.Linux.ApiWrapper.x64.so ./Pyroscope.Linux.ApiWrapper.x64.so +COPY --from=build /dotnet/ ./ -RUN dotnet publish -o . -r $(dotnet --info | grep RID | cut -b 6- | tr -d ' ') ENV CORECLR_ENABLE_PROFILING=1 ENV CORECLR_PROFILER={BD1A650D-AC5D-4896-B64F-D6FA25D6B26A} @@ -22,6 +41,8 @@ ENV PYROSCOPE_PROFILING_ENABLED=1 ENV PYROSCOPE_PROFILING_ALLOCATION_ENABLED=true ENV PYROSCOPE_PROFILING_CONTENTION_ENABLED=true ENV PYROSCOPE_PROFILING_EXCEPTION_ENABLED=true +ENV PYROSCOPE_PROFILING_HEAP_ENABLED=true ENV RIDESHARE_LISTEN_PORT=5000 + CMD sh -c "ASPNETCORE_URLS=http://*:${RIDESHARE_LISTEN_PORT} exec dotnet /dotnet/example.dll" diff --git a/examples/language-sdk-instrumentation/dotnet/web-new/Dockerfile b/examples/language-sdk-instrumentation/dotnet/web-new/Dockerfile index d3e1774821..67b10e040d 100644 --- a/examples/language-sdk-instrumentation/dotnet/web-new/Dockerfile +++ b/examples/language-sdk-instrumentation/dotnet/web-new/Dockerfile @@ -3,8 +3,8 @@ FROM --platform=linux/amd64 mcr.microsoft.com/dotnet/sdk:6.0 WORKDIR /dotnet -COPY --from=pyroscope/pyroscope-dotnet:0.8.14-glibc /Pyroscope.Profiler.Native.so ./Pyroscope.Profiler.Native.so -COPY --from=pyroscope/pyroscope-dotnet:0.8.14-glibc /Pyroscope.Linux.ApiWrapper.x64.so ./Pyroscope.Linux.ApiWrapper.x64.so +COPY --from=pyroscope/pyroscope-dotnet:0.8.19-glibc /Pyroscope.Profiler.Native.so ./Pyroscope.Profiler.Native.so +COPY --from=pyroscope/pyroscope-dotnet:0.8.19-glibc /Pyroscope.Linux.ApiWrapper.x64.so ./Pyroscope.Linux.ApiWrapper.x64.so ADD example . diff --git a/examples/language-sdk-instrumentation/golang-push/rideshare/go.mod b/examples/language-sdk-instrumentation/golang-push/rideshare/go.mod index 2e28f44778..ed890136bd 100644 --- a/examples/language-sdk-instrumentation/golang-push/rideshare/go.mod +++ b/examples/language-sdk-instrumentation/golang-push/rideshare/go.mod @@ -22,9 +22,9 @@ require ( github.com/go-logr/logr v1.4.1 // indirect github.com/go-logr/stdr v1.2.2 // indirect github.com/golang/protobuf v1.5.3 // indirect - github.com/grafana/pyroscope-go/godeltaprof v0.1.7 // indirect + github.com/grafana/pyroscope-go/godeltaprof v0.1.8 // indirect github.com/grpc-ecosystem/grpc-gateway/v2 v2.18.1 // indirect - github.com/klauspost/compress v1.17.4 // indirect + github.com/klauspost/compress v1.17.8 // indirect github.com/pmezard/go-difflib v1.0.1-0.20181226105442-5d4384ee4fb2 // indirect go.opentelemetry.io/otel/metric v1.22.0 // indirect go.opentelemetry.io/proto/otlp v1.0.0 // indirect diff --git a/examples/language-sdk-instrumentation/golang-push/rideshare/go.sum b/examples/language-sdk-instrumentation/golang-push/rideshare/go.sum index 8d7a9b76d5..e7f6c92c13 100644 --- a/examples/language-sdk-instrumentation/golang-push/rideshare/go.sum +++ b/examples/language-sdk-instrumentation/golang-push/rideshare/go.sum @@ -1390,8 +1390,8 @@ github.com/grafana/otel-profiling-go v0.5.1/go.mod h1:ftN/t5A/4gQI19/8MoWurBEtC6 github.com/grafana/pyroscope-go v1.1.1 h1:PQoUU9oWtO3ve/fgIiklYuGilvsm8qaGhlY4Vw6MAcQ= github.com/grafana/pyroscope-go v1.1.1/go.mod h1:Mw26jU7jsL/KStNSGGuuVYdUq7Qghem5P8aXYXSXG88= github.com/grafana/pyroscope-go/godeltaprof v0.1.6/go.mod h1:Tk376Nbldo4Cha9RgiU7ik8WKFkNpfds98aUzS8omLE= -github.com/grafana/pyroscope-go/godeltaprof v0.1.7 h1:C11j63y7gymiW8VugJ9ZW0pWfxTZugdSJyC48olk5KY= -github.com/grafana/pyroscope-go/godeltaprof v0.1.7/go.mod h1:Tk376Nbldo4Cha9RgiU7ik8WKFkNpfds98aUzS8omLE= +github.com/grafana/pyroscope-go/godeltaprof v0.1.8 h1:iwOtYXeeVSAeYefJNaxDytgjKtUuKQbJqgAIjlnicKg= +github.com/grafana/pyroscope-go/godeltaprof v0.1.8/go.mod h1:2+l7K7twW49Ct4wFluZD3tZ6e0SjanjcUUBPVD/UuGU= github.com/grpc-ecosystem/grpc-gateway v1.16.0 h1:gmcG1KaJ57LophUzW0Hy8NmPhnMZb4M0+kPpLofRdBo= github.com/grpc-ecosystem/grpc-gateway v1.16.0/go.mod h1:BDjrQk3hbvj6Nolgz8mAMFbcEtjT1g+wF4CSlocrBnw= github.com/grpc-ecosystem/grpc-gateway/v2 v2.7.0/go.mod h1:hgWBS7lorOAVIJEQMi4ZsPv9hVvWI6+ch50m39Pf2Ks= @@ -1414,8 +1414,8 @@ github.com/kisielk/gotool v1.0.0/go.mod h1:XhKaO+MFFWcvkIS/tQcRk01m1F5IRFswLeQ+o github.com/klauspost/asmfmt v1.3.2/go.mod h1:AG8TuvYojzulgDAMCnYn50l/5QV3Bs/tp6j0HLHbNSE= github.com/klauspost/compress v1.15.9/go.mod h1:PhcZ0MbTNciWF3rruxRgKxI5NkcHHrHUDtV4Yw2GlzU= github.com/klauspost/compress v1.17.3/go.mod h1:/dCuZOvVtNoHsyb+cuJD3itjs3NbnF6KH9zAO4BDxPM= -github.com/klauspost/compress v1.17.4 h1:Ej5ixsIri7BrIjBkRZLTo6ghwrEtHFk7ijlczPW4fZ4= -github.com/klauspost/compress v1.17.4/go.mod h1:/dCuZOvVtNoHsyb+cuJD3itjs3NbnF6KH9zAO4BDxPM= +github.com/klauspost/compress v1.17.8 h1:YcnTYrq7MikUT7k0Yb5eceMmALQPYBW/Xltxn0NAMnU= +github.com/klauspost/compress v1.17.8/go.mod h1:Di0epgTjJY877eYKx5yC51cX2A2Vl2ibi7bDH9ttBbw= github.com/klauspost/cpuid/v2 v2.0.9/go.mod h1:FInQzS24/EEf25PyTYn52gqo7WaD8xa0213Md/qVLRg= github.com/kr/fs v0.1.0/go.mod h1:FFnZGqtBN9Gxj7eW1uZ42v5BccTP0vu6NEaFoC2HwRg= github.com/kr/pretty v0.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORNo= diff --git a/examples/language-sdk-instrumentation/golang-push/simple/go.mod b/examples/language-sdk-instrumentation/golang-push/simple/go.mod index d99f8bb778..32fec30699 100644 --- a/examples/language-sdk-instrumentation/golang-push/simple/go.mod +++ b/examples/language-sdk-instrumentation/golang-push/simple/go.mod @@ -5,6 +5,6 @@ go 1.17 require github.com/grafana/pyroscope-go v1.1.1 require ( - github.com/grafana/pyroscope-go/godeltaprof v0.1.7 // indirect - github.com/klauspost/compress v1.17.3 // indirect + github.com/grafana/pyroscope-go/godeltaprof v0.1.8 // indirect + github.com/klauspost/compress v1.17.8 // indirect ) diff --git a/examples/language-sdk-instrumentation/golang-push/simple/go.sum b/examples/language-sdk-instrumentation/golang-push/simple/go.sum index 210c61b491..613f4a39af 100644 --- a/examples/language-sdk-instrumentation/golang-push/simple/go.sum +++ b/examples/language-sdk-instrumentation/golang-push/simple/go.sum @@ -1,7 +1,8 @@ github.com/grafana/pyroscope-go v1.1.1 h1:PQoUU9oWtO3ve/fgIiklYuGilvsm8qaGhlY4Vw6MAcQ= github.com/grafana/pyroscope-go v1.1.1/go.mod h1:Mw26jU7jsL/KStNSGGuuVYdUq7Qghem5P8aXYXSXG88= github.com/grafana/pyroscope-go/godeltaprof v0.1.6/go.mod h1:Tk376Nbldo4Cha9RgiU7ik8WKFkNpfds98aUzS8omLE= -github.com/grafana/pyroscope-go/godeltaprof v0.1.7 h1:C11j63y7gymiW8VugJ9ZW0pWfxTZugdSJyC48olk5KY= -github.com/grafana/pyroscope-go/godeltaprof v0.1.7/go.mod h1:Tk376Nbldo4Cha9RgiU7ik8WKFkNpfds98aUzS8omLE= -github.com/klauspost/compress v1.17.3 h1:qkRjuerhUU1EmXLYGkSH6EZL+vPSxIrYjLNAK4slzwA= +github.com/grafana/pyroscope-go/godeltaprof v0.1.8 h1:iwOtYXeeVSAeYefJNaxDytgjKtUuKQbJqgAIjlnicKg= +github.com/grafana/pyroscope-go/godeltaprof v0.1.8/go.mod h1:2+l7K7twW49Ct4wFluZD3tZ6e0SjanjcUUBPVD/UuGU= github.com/klauspost/compress v1.17.3/go.mod h1:/dCuZOvVtNoHsyb+cuJD3itjs3NbnF6KH9zAO4BDxPM= +github.com/klauspost/compress v1.17.8 h1:YcnTYrq7MikUT7k0Yb5eceMmALQPYBW/Xltxn0NAMnU= +github.com/klauspost/compress v1.17.8/go.mod h1:Di0epgTjJY877eYKx5yC51cX2A2Vl2ibi7bDH9ttBbw= diff --git a/examples/language-sdk-instrumentation/java/fib/Dockerfile b/examples/language-sdk-instrumentation/java/fib/Dockerfile index b919b0c2eb..c9db4cb127 100644 --- a/examples/language-sdk-instrumentation/java/fib/Dockerfile +++ b/examples/language-sdk-instrumentation/java/fib/Dockerfile @@ -4,7 +4,7 @@ WORKDIR /opt/app RUN apt-get update && apt-get install ca-certificates -y && update-ca-certificates && apt-get install -y git -ADD https://github.com/grafana/pyroscope-java/releases/download/v0.13.1/pyroscope.jar /opt/app/pyroscope.jar +ADD https://github.com/grafana/pyroscope-java/releases/download/v0.14.0/pyroscope.jar /opt/app/pyroscope.jar COPY Main.java ./ diff --git a/examples/language-sdk-instrumentation/java/rideshare/Dockerfile b/examples/language-sdk-instrumentation/java/rideshare/Dockerfile index 79a7dcae2c..8088e822ca 100644 --- a/examples/language-sdk-instrumentation/java/rideshare/Dockerfile +++ b/examples/language-sdk-instrumentation/java/rideshare/Dockerfile @@ -37,6 +37,6 @@ COPY --from=builder /opt/app/build/libs/rideshare-1.0-SNAPSHOT.jar /opt/app/buil WORKDIR /opt/app -ADD https://github.com/grafana/pyroscope-java/releases/download/v0.13.1/pyroscope.jar /opt/app/pyroscope.jar +ADD https://github.com/grafana/pyroscope-java/releases/download/v0.14.0/pyroscope.jar /opt/app/pyroscope.jar CMD sh -c "exec java -Dserver.port=${RIDESHARE_LISTEN_PORT} -javaagent:pyroscope.jar -jar ./build/libs/rideshare-1.0-SNAPSHOT.jar" diff --git a/examples/language-sdk-instrumentation/java/rideshare/build.gradle.kts b/examples/language-sdk-instrumentation/java/rideshare/build.gradle.kts index c482284773..89717bef37 100644 --- a/examples/language-sdk-instrumentation/java/rideshare/build.gradle.kts +++ b/examples/language-sdk-instrumentation/java/rideshare/build.gradle.kts @@ -12,7 +12,7 @@ repositories { } dependencies { - implementation("io.pyroscope:agent:0.13.1") + implementation("io.pyroscope:agent:0.14.0") implementation("org.springframework.boot:spring-boot-starter-web") testImplementation("org.junit.jupiter:junit-jupiter-api:5.8.2") testRuntimeOnly("org.junit.jupiter:junit-jupiter-engine:5.8.2") diff --git a/examples/language-sdk-instrumentation/java/simple/Dockerfile b/examples/language-sdk-instrumentation/java/simple/Dockerfile index e8d3960b70..f764a0ab42 100644 --- a/examples/language-sdk-instrumentation/java/simple/Dockerfile +++ b/examples/language-sdk-instrumentation/java/simple/Dockerfile @@ -2,7 +2,7 @@ FROM openjdk:11.0.11-jdk WORKDIR /opt/app -ADD https://github.com/grafana/pyroscope-java/releases/download/v0.13.1/pyroscope.jar /opt/app/pyroscope.jar +ADD https://github.com/grafana/pyroscope-java/releases/download/v0.14.0/pyroscope.jar /opt/app/pyroscope.jar COPY Main.java ./Main.java RUN javac Main.java diff --git a/examples/language-sdk-instrumentation/python/rideshare/django/app/requirements.txt b/examples/language-sdk-instrumentation/python/rideshare/django/app/requirements.txt index 4d956638c8..cbd3138c0c 100644 --- a/examples/language-sdk-instrumentation/python/rideshare/django/app/requirements.txt +++ b/examples/language-sdk-instrumentation/python/rideshare/django/app/requirements.txt @@ -2,4 +2,4 @@ Django==3.2.25 djangorestframework==3.12.4 gunicorn==20.1.0 psycopg2-binary==2.9.1 -pyroscope-io==0.8.6 +pyroscope-io==0.8.7 diff --git a/examples/language-sdk-instrumentation/python/rideshare/fastapi/Dockerfile b/examples/language-sdk-instrumentation/python/rideshare/fastapi/Dockerfile index 55c0e27984..3c5e2d0ac2 100644 --- a/examples/language-sdk-instrumentation/python/rideshare/fastapi/Dockerfile +++ b/examples/language-sdk-instrumentation/python/rideshare/fastapi/Dockerfile @@ -1,6 +1,6 @@ FROM python:3.9 -RUN pip3 install fastapi pyroscope-io==0.8.6 uvicorn[standard] +RUN pip3 install fastapi pyroscope-io==0.8.7 uvicorn[standard] ENV FLASK_ENV=development ENV PYTHONUNBUFFERED=1 diff --git a/examples/language-sdk-instrumentation/python/simple/requirements.txt b/examples/language-sdk-instrumentation/python/simple/requirements.txt index 2b889cb104..ad747d4bd2 100644 --- a/examples/language-sdk-instrumentation/python/simple/requirements.txt +++ b/examples/language-sdk-instrumentation/python/simple/requirements.txt @@ -1 +1 @@ -pyroscope-io==0.8.6 +pyroscope-io==0.8.7 diff --git a/examples/language-sdk-instrumentation/ruby/rideshare/Gemfile b/examples/language-sdk-instrumentation/ruby/rideshare/Gemfile index 7a56c065ed..cb1b69c54e 100644 --- a/examples/language-sdk-instrumentation/ruby/rideshare/Gemfile +++ b/examples/language-sdk-instrumentation/ruby/rideshare/Gemfile @@ -5,7 +5,7 @@ source "https://rubygems.org" git_source(:github) { |repo_name| "https://github.com/#{repo_name}" } # gem "rails" -gem 'pyroscope', '= 0.5.11' +gem 'pyroscope', '= 0.5.12' gem "sinatra", "~> 2.1" gem "thin", "~> 1.8" diff --git a/examples/language-sdk-instrumentation/ruby/rideshare/Gemfile.lock b/examples/language-sdk-instrumentation/ruby/rideshare/Gemfile.lock index c9a62c43d6..fec93dc0e4 100644 --- a/examples/language-sdk-instrumentation/ruby/rideshare/Gemfile.lock +++ b/examples/language-sdk-instrumentation/ruby/rideshare/Gemfile.lock @@ -3,14 +3,16 @@ GEM specs: daemons (1.4.1) eventmachine (1.2.7) - ffi (1.16.3) + ffi (1.17.0-aarch64-linux-gnu) + ffi (1.17.0-arm64-darwin) + ffi (1.17.0-x86_64-linux-gnu) mustermann (2.0.2) ruby2_keywords (~> 0.0.1) - pyroscope (0.5.11-aarch64-linux) + pyroscope (0.5.12-aarch64-linux) ffi - pyroscope (0.5.11-arm64-darwin) + pyroscope (0.5.12-arm64-darwin) ffi - pyroscope (0.5.11-x86_64-linux) + pyroscope (0.5.12-x86_64-linux) ffi rack (2.2.8.1) rack-protection (2.2.4) @@ -33,7 +35,7 @@ PLATFORMS x86_64-linux DEPENDENCIES - pyroscope (= 0.5.11) + pyroscope (= 0.5.12) sinatra (~> 2.1) thin (~> 1.8) diff --git a/examples/language-sdk-instrumentation/ruby/rideshare_rails/Gemfile b/examples/language-sdk-instrumentation/ruby/rideshare_rails/Gemfile index df050cd491..484dcba5c2 100644 --- a/examples/language-sdk-instrumentation/ruby/rideshare_rails/Gemfile +++ b/examples/language-sdk-instrumentation/ruby/rideshare_rails/Gemfile @@ -15,7 +15,7 @@ gem "sqlite3", "~> 1.4" # Use the Puma web server [https://github.com/puma/puma] gem "puma", "~> 5.6" -gem 'pyroscope', '= 0.5.11' +gem 'pyroscope', '= 0.5.12' gem "pyroscope-otel", "~> 0.1.1" gem 'opentelemetry-sdk', "~> 1.2.0" diff --git a/examples/language-sdk-instrumentation/ruby/rideshare_rails/Gemfile.lock b/examples/language-sdk-instrumentation/ruby/rideshare_rails/Gemfile.lock index f32bf752e7..da63ba210d 100644 --- a/examples/language-sdk-instrumentation/ruby/rideshare_rails/Gemfile.lock +++ b/examples/language-sdk-instrumentation/ruby/rideshare_rails/Gemfile.lock @@ -74,7 +74,11 @@ GEM irb (>= 1.5.0) reline (>= 0.3.1) erubi (1.12.0) - ffi (1.16.3) + ffi (1.17.0) + ffi (1.17.0-aarch64-linux-gnu) + ffi (1.17.0-arm64-darwin) + ffi (1.17.0-x86_64-darwin) + ffi (1.17.0-x86_64-linux-gnu) globalid (1.2.1) activesupport (>= 6.1) i18n (1.14.1) @@ -172,15 +176,15 @@ GEM stringio puma (5.6.8) nio4r (~> 2.0) - pyroscope (0.5.11) + pyroscope (0.5.12) ffi - pyroscope (0.5.11-aarch64-linux) + pyroscope (0.5.12-aarch64-linux) ffi - pyroscope (0.5.11-arm64-darwin) + pyroscope (0.5.12-arm64-darwin) ffi - pyroscope (0.5.11-x86_64-darwin) + pyroscope (0.5.12-x86_64-darwin) ffi - pyroscope (0.5.11-x86_64-linux) + pyroscope (0.5.12-x86_64-linux) ffi pyroscope-otel (0.1.1) opentelemetry-api (~> 1.1.0) @@ -260,7 +264,7 @@ DEPENDENCIES opentelemetry-instrumentation-rails opentelemetry-sdk (~> 1.2.0) puma (~> 5.6) - pyroscope (= 0.5.11) + pyroscope (= 0.5.12) pyroscope-otel (~> 0.1.1) rails (~> 7.0.8) sprockets-rails diff --git a/examples/language-sdk-instrumentation/ruby/simple/Gemfile b/examples/language-sdk-instrumentation/ruby/simple/Gemfile index 4266da763e..d64da8dce1 100644 --- a/examples/language-sdk-instrumentation/ruby/simple/Gemfile +++ b/examples/language-sdk-instrumentation/ruby/simple/Gemfile @@ -1,3 +1,3 @@ source 'https://rubygems.org' -gem 'pyroscope', '= 0.5.11' +gem 'pyroscope', '= 0.5.12' diff --git a/examples/language-sdk-instrumentation/ruby/simple/Gemfile.lock b/examples/language-sdk-instrumentation/ruby/simple/Gemfile.lock index 690fcafcbb..06c80a5d56 100644 --- a/examples/language-sdk-instrumentation/ruby/simple/Gemfile.lock +++ b/examples/language-sdk-instrumentation/ruby/simple/Gemfile.lock @@ -1,12 +1,14 @@ GEM remote: https://rubygems.org/ specs: - ffi (1.16.3) - pyroscope (0.5.11-aarch64-linux) + ffi (1.17.0-aarch64-linux-gnu) + ffi (1.17.0-arm64-darwin) + ffi (1.17.0-x86_64-linux-gnu) + pyroscope (0.5.12-aarch64-linux) ffi - pyroscope (0.5.11-arm64-darwin) + pyroscope (0.5.12-arm64-darwin) ffi - pyroscope (0.5.11-x86_64-linux) + pyroscope (0.5.12-x86_64-linux) ffi PLATFORMS @@ -15,7 +17,7 @@ PLATFORMS x86_64-linux DEPENDENCIES - pyroscope (= 0.5.11) + pyroscope (= 0.5.12) BUNDLED WITH 2.4.10 diff --git a/examples/tracing/tempo/docker-compose.yml b/examples/tracing/tempo/docker-compose.yml index 811624db3e..76f1d52e80 100644 --- a/examples/tracing/tempo/docker-compose.yml +++ b/examples/tracing/tempo/docker-compose.yml @@ -43,7 +43,6 @@ services: dockerfile: Dockerfile.otel-instrumentation rideshare-dotnet-eu-west: - platform: linux/amd64 ports: - 5000 hostname: rideshare-dotnet-eu-west diff --git a/go.mod b/go.mod index b58ea62763..c25c484efb 100644 --- a/go.mod +++ b/go.mod @@ -32,11 +32,14 @@ require ( github.com/grafana/dskit v0.0.0-20231221015914-de83901bf4d6 github.com/grafana/jfr-parser/pprof v0.0.0-20240228024232-8abcb81c304c github.com/grafana/pyroscope-go v1.0.3 - github.com/grafana/pyroscope-go/godeltaprof v0.1.7 + github.com/grafana/pyroscope-go/godeltaprof v0.1.8 github.com/grafana/pyroscope/api v0.4.0 github.com/grafana/regexp v0.0.0-20221123153739-15dc172cd2db github.com/grpc-ecosystem/grpc-gateway/v2 v2.19.0 github.com/hashicorp/golang-lru/v2 v2.0.7 + github.com/hashicorp/raft v1.7.0 + github.com/hashicorp/raft-wal v0.4.1 + github.com/iancoleman/strcase v0.3.0 github.com/json-iterator/go v1.1.12 github.com/k0kubun/pp/v3 v3.2.0 github.com/klauspost/compress v1.17.9 @@ -67,6 +70,7 @@ require ( github.com/uber/jaeger-client-go v2.30.0+incompatible github.com/valyala/bytebufferpool v1.0.0 github.com/xlab/treeprint v1.2.0 + go.etcd.io/bbolt v1.3.10 go.opentelemetry.io/proto/otlp v1.1.0 go.uber.org/atomic v1.11.0 go.uber.org/goleak v1.3.0 @@ -118,16 +122,21 @@ require ( github.com/aws/aws-sdk-go-v2/service/sts v1.19.2 // indirect github.com/aws/smithy-go v1.13.5 // indirect github.com/bboreham/go-loser v0.0.0-20230920113527-fcc2c21820a3 // indirect + github.com/benbjohnson/immutable v0.4.0 // indirect github.com/beorn7/perks v1.0.1 // indirect github.com/chainguard-dev/git-urls v1.0.2 // indirect github.com/clbanning/mxj v1.8.4 // indirect + github.com/coreos/etcd v3.3.27+incompatible // indirect github.com/coreos/go-semver v0.3.0 // indirect + github.com/coreos/go-systemd v0.0.0-20191104093116-d3cd4ed1dbcf // indirect github.com/coreos/go-systemd/v22 v22.5.0 // indirect + github.com/coreos/pkg v0.0.0-20220810130054-c7d1c02cb6cf // indirect github.com/davecgh/go-spew v1.1.2-0.20180830191138-d8f796af33cc // indirect github.com/dolthub/maphash v0.1.0 // indirect github.com/edsrzf/mmap-go v1.1.0 // indirect github.com/efficientgo/core v1.0.0-rc.2 // indirect github.com/efficientgo/e2e v0.14.1-0.20230710114240-c316eb95ae5b // indirect + github.com/emicklei/go-restful/v3 v3.11.0 // indirect github.com/facette/natsort v0.0.0-20181210072756-2cd4dd1e2dcb // indirect github.com/fsnotify/fsnotify v1.7.0 // indirect github.com/go-logfmt/logfmt v0.6.0 // indirect @@ -149,7 +158,9 @@ require ( github.com/golang/protobuf v1.5.4 // indirect github.com/golang/snappy v0.0.4 // indirect github.com/google/btree v1.1.2 // indirect + github.com/google/gnostic-models v0.6.8 // indirect github.com/google/go-querystring v1.1.0 // indirect + github.com/google/gofuzz v1.2.0 // indirect github.com/google/s2a-go v0.1.7 // indirect github.com/googleapis/enterprise-certificate-proxy v0.3.2 // indirect github.com/googleapis/gax-go/v2 v2.12.3 // indirect @@ -157,9 +168,10 @@ require ( github.com/hashicorp/consul/api v1.28.2 // indirect github.com/hashicorp/errwrap v1.1.0 // indirect github.com/hashicorp/go-cleanhttp v0.5.2 // indirect - github.com/hashicorp/go-hclog v1.5.0 // indirect + github.com/hashicorp/go-hclog v1.6.2 // indirect github.com/hashicorp/go-immutable-radix v1.3.1 // indirect github.com/hashicorp/go-msgpack v1.1.5 // indirect + github.com/hashicorp/go-msgpack/v2 v2.1.1 // indirect github.com/hashicorp/go-multierror v1.1.1 // indirect github.com/hashicorp/go-rootcerts v1.0.2 // indirect github.com/hashicorp/go-sockaddr v1.0.6 // indirect @@ -167,6 +179,7 @@ require ( github.com/hashicorp/golang-lru v0.6.0 // indirect github.com/hashicorp/memberlist v0.5.0 // indirect github.com/hashicorp/serf v0.10.1 // indirect + github.com/imdario/mergo v0.3.16 // indirect github.com/jmespath/go-jmespath v0.4.0 // indirect github.com/josharian/intern v1.0.0 // indirect github.com/jpillora/backoff v1.0.0 // indirect @@ -201,6 +214,7 @@ require ( github.com/segmentio/encoding v0.3.6 // indirect github.com/sercand/kuberesolver/v5 v5.1.1 // indirect github.com/soheilhy/cmux v0.1.5 // indirect + github.com/spf13/pflag v1.0.5 // indirect github.com/stretchr/objx v0.5.2 // indirect github.com/tencentyun/cos-go-sdk-v5 v0.7.40 // indirect github.com/uber/jaeger-lib v2.4.1+incompatible // indirect @@ -225,12 +239,17 @@ require ( google.golang.org/api v0.172.0 // indirect google.golang.org/genproto v0.0.0-20240213162025-012b6fc9bca9 // indirect google.golang.org/genproto/googleapis/rpc v0.0.0-20240318140521-94a12d6c2237 // indirect + gopkg.in/inf.v0 v0.9.1 // indirect gopkg.in/ini.v1 v1.67.0 // indirect gopkg.in/yaml.v2 v2.4.0 // indirect + k8s.io/api v0.29.2 // indirect k8s.io/apimachinery v0.29.2 // indirect k8s.io/client-go v0.29.2 // indirect k8s.io/klog/v2 v2.120.1 // indirect + k8s.io/kube-openapi v0.0.0-20231010175941-2dd684a91f00 // indirect k8s.io/utils v0.0.0-20230726121419-3b25d923346b // indirect + sigs.k8s.io/json v0.0.0-20221116044647-bc3834ca7abd // indirect + sigs.k8s.io/structured-merge-diff/v4 v4.4.1 // indirect ) replace ( diff --git a/go.sum b/go.sum index 8ed60b7dd1..3cffe035b9 100644 --- a/go.sum +++ b/go.sum @@ -129,6 +129,8 @@ github.com/bboreham/go-loser v0.0.0-20230920113527-fcc2c21820a3 h1:6df1vn4bBlDDo github.com/bboreham/go-loser v0.0.0-20230920113527-fcc2c21820a3/go.mod h1:CIWtjkly68+yqLPbvwwR/fjNJA/idrtULjZWh2v1ys0= github.com/benbjohnson/clock v1.3.5 h1:VvXlSJBzZpA/zum6Sj74hxwYI2DIxRWuNIoXAzHZz5o= github.com/benbjohnson/clock v1.3.5/go.mod h1:J11/hYXuz8f4ySSvYwY0FKfm+ezbsZBKZxNJlLklBHA= +github.com/benbjohnson/immutable v0.4.0 h1:CTqXbEerYso8YzVPxmWxh2gnoRQbbB9X1quUC8+vGZA= +github.com/benbjohnson/immutable v0.4.0/go.mod h1:iAr8OjJGLnLmVUr9MZ/rz4PWUy6Ouc2JLYuMArmvAJM= github.com/beorn7/perks v0.0.0-20180321164747-3a771d992973/go.mod h1:Dwedo/Wpr24TaqPxmxbtue+5NUziq4I4S80YR8gNf3Q= github.com/beorn7/perks v1.0.0/go.mod h1:KWe93zE9D1o94FZ5RNwFwVgaQK1VOXiVxmqh+CedLV8= github.com/beorn7/perks v1.0.1 h1:VlbKKnNfV8bJzeqoa4cOKqO6bYr3WgKZxO8Z16+hsOM= @@ -159,10 +161,16 @@ github.com/colega/go-yaml-yaml v0.0.0-20220720105220-255a8d16d094 h1:FpZSn61BWXb github.com/colega/go-yaml-yaml v0.0.0-20220720105220-255a8d16d094/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= github.com/colega/zeropool v0.0.0-20230505084239-6fb4a4f75381 h1:d5EKgQfRQvO97jnISfR89AiCCCJMwMFoSxUiU0OGCRU= github.com/colega/zeropool v0.0.0-20230505084239-6fb4a4f75381/go.mod h1:OU76gHeRo8xrzGJU3F3I1CqX1ekM8dfJw0+wPeMwnp0= +github.com/coreos/etcd v3.3.27+incompatible h1:QIudLb9KeBsE5zyYxd1mjzRSkzLg9Wf9QlRwFgd6oTA= +github.com/coreos/etcd v3.3.27+incompatible/go.mod h1:uF7uidLiAD3TWHmW31ZFd/JWoc32PjwdhPthX9715RE= github.com/coreos/go-semver v0.3.0 h1:wkHLiw0WNATZnSG7epLsujiMCgPAc9xhjJ4tgnAxmfM= github.com/coreos/go-semver v0.3.0/go.mod h1:nnelYz7RCh+5ahJtPPxZlU+153eP4D4r3EedlOD2RNk= +github.com/coreos/go-systemd v0.0.0-20191104093116-d3cd4ed1dbcf h1:iW4rZ826su+pqaw19uhpSCzhj44qo35pNgKFGqzDKkU= +github.com/coreos/go-systemd v0.0.0-20191104093116-d3cd4ed1dbcf/go.mod h1:F5haX7vjVVG0kc13fIWeqUViNPyEJxv/OmvnBo0Yme4= github.com/coreos/go-systemd/v22 v22.5.0 h1:RrqgGjYQKalulkV8NGVIfkXQf6YYmOyiJKk8iXXhfZs= github.com/coreos/go-systemd/v22 v22.5.0/go.mod h1:Y58oyj3AT4RCenI/lSvhwexgC+NSVTIJ3seZv2GcEnc= +github.com/coreos/pkg v0.0.0-20220810130054-c7d1c02cb6cf h1:GOPo6vn/vTN+3IwZBvXX0y5doJfSC7My0cdzelyOCsQ= +github.com/coreos/pkg v0.0.0-20220810130054-c7d1c02cb6cf/go.mod h1:E3G3o1h8I7cfcXa63jLwjI0eiQQMgzzUDFVpN/nH/eA= github.com/creack/pty v1.1.9/go.mod h1:oKZEueFk5CKHvIhNR5MUki03XCEU+Q6VDXinZuGJ33E= github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= @@ -206,6 +214,8 @@ github.com/envoyproxy/go-control-plane v0.12.0/go.mod h1:ZBTaoJ23lqITozF0M6G4/Ir github.com/envoyproxy/protoc-gen-validate v0.1.0/go.mod h1:iSmxcyjqTsJpI2R4NaDN7+kN2VEUnK/pcBlmesArF7c= github.com/envoyproxy/protoc-gen-validate v1.0.4 h1:gVPz/FMfvh57HdSJQyvBtF00j8JU4zdyUgIUNhlgg0A= github.com/envoyproxy/protoc-gen-validate v1.0.4/go.mod h1:qys6tmnRsYrQqIhm2bvKZH4Blx/1gTIZ2UKVY1M+Yew= +github.com/evanphx/json-patch v5.6.0+incompatible h1:jBYDEEiFBPxA0v50tFdvOzQQTCvpL6mnFh5mB2/l16U= +github.com/evanphx/json-patch v5.6.0+incompatible/go.mod h1:50XU6AFN0ol/bzJsmQLiYLvXMP4fmwYFNcr97nuDLSk= github.com/facette/natsort v0.0.0-20181210072756-2cd4dd1e2dcb h1:IT4JYU7k4ikYg1SCxNI1/Tieq/NFvh6dzLdgi7eu0tM= github.com/facette/natsort v0.0.0-20181210072756-2cd4dd1e2dcb/go.mod h1:bH6Xx7IW64qjjJq8M2u4dxNaBiDfKK+z/3eGDpXEQhc= github.com/fatih/color v1.7.0/go.mod h1:Zm6kSWBoL9eyXnKyktHP6abPY2pDugNf5KwzbycvMj4= @@ -330,6 +340,7 @@ github.com/google/go-cmp v0.5.3/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/ github.com/google/go-cmp v0.5.4/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= github.com/google/go-cmp v0.5.5/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= github.com/google/go-cmp v0.5.8/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY= +github.com/google/go-cmp v0.5.9/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY= github.com/google/go-cmp v0.6.0 h1:ofyhxvXcZhMsU5ulbFiLKl/XBFqE1GSq7atu8tAmTRI= github.com/google/go-cmp v0.6.0/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY= github.com/google/go-github/v58 v58.0.1-0.20240111193443-e9f52699f5e5 h1:Cm3eMs9Qj7fqDQOascVTJg37N0T7Vb2foS//WopCpWw= @@ -386,8 +397,8 @@ github.com/grafana/memberlist v0.3.1-0.20220708130638-bd88e10a3d91 h1:/NipyHnOmv github.com/grafana/memberlist v0.3.1-0.20220708130638-bd88e10a3d91/go.mod h1:MS2lj3INKhZjWNqd3N0m3J+Jxf3DAOnAH9VT3Sh9MUE= github.com/grafana/pyroscope-go v1.0.3 h1:8WWmItzLfg4m8G+j//ElSjMeMr88Y6Lvblar6qeTyKk= github.com/grafana/pyroscope-go v1.0.3/go.mod h1:0d7ftwSMBV/Awm7CCiYmHQEG8Y44Ma3YSjt+nWcWztY= -github.com/grafana/pyroscope-go/godeltaprof v0.1.7 h1:C11j63y7gymiW8VugJ9ZW0pWfxTZugdSJyC48olk5KY= -github.com/grafana/pyroscope-go/godeltaprof v0.1.7/go.mod h1:Tk376Nbldo4Cha9RgiU7ik8WKFkNpfds98aUzS8omLE= +github.com/grafana/pyroscope-go/godeltaprof v0.1.8 h1:iwOtYXeeVSAeYefJNaxDytgjKtUuKQbJqgAIjlnicKg= +github.com/grafana/pyroscope-go/godeltaprof v0.1.8/go.mod h1:2+l7K7twW49Ct4wFluZD3tZ6e0SjanjcUUBPVD/UuGU= github.com/grafana/regexp v0.0.0-20221123153739-15dc172cd2db h1:7aN5cccjIqCLTzedH7MZzRZt5/lsAHch6Z3L2ZGn5FA= github.com/grafana/regexp v0.0.0-20221123153739-15dc172cd2db/go.mod h1:M5qHK+eWfAv8VR/265dIuEpL3fNfeC21tXXp9itM24A= github.com/grpc-ecosystem/grpc-gateway/v2 v2.19.0 h1:Wqo399gCIufwto+VfwCSvsnfGpF/w5E9CNxSwbpD6No= @@ -405,14 +416,16 @@ github.com/hashicorp/errwrap v1.1.0/go.mod h1:YH+1FKiLXxHSkmPseP+kNlulaMuP3n2brv github.com/hashicorp/go-cleanhttp v0.5.0/go.mod h1:JpRdi6/HCYpAwUzNwuwqhbovhLtngrth3wmdIIUrZ80= github.com/hashicorp/go-cleanhttp v0.5.2 h1:035FKYIWjmULyFRBKPs8TBQoi0x6d9G4xc9neXJWAZQ= github.com/hashicorp/go-cleanhttp v0.5.2/go.mod h1:kO/YDlP8L1346E6Sodw+PrpBSV4/SoxCXGY6BqNFT48= -github.com/hashicorp/go-hclog v1.5.0 h1:bI2ocEMgcVlz55Oj1xZNBsVi900c7II+fWDyV9o+13c= -github.com/hashicorp/go-hclog v1.5.0/go.mod h1:W4Qnvbt70Wk/zYJryRzDRU/4r0kIg0PVHBcfoyhpF5M= +github.com/hashicorp/go-hclog v1.6.2 h1:NOtoftovWkDheyUM/8JW3QMiXyxJK3uHRK7wV04nD2I= +github.com/hashicorp/go-hclog v1.6.2/go.mod h1:W4Qnvbt70Wk/zYJryRzDRU/4r0kIg0PVHBcfoyhpF5M= github.com/hashicorp/go-immutable-radix v1.0.0/go.mod h1:0y9vanUI8NX6FsYoO3zeMjhV/C5i9g4Q3DwcSNZ4P60= github.com/hashicorp/go-immutable-radix v1.3.1 h1:DKHmCUm2hRBK510BaiZlwvpD40f8bJFeZnpfm2KLowc= github.com/hashicorp/go-immutable-radix v1.3.1/go.mod h1:0y9vanUI8NX6FsYoO3zeMjhV/C5i9g4Q3DwcSNZ4P60= github.com/hashicorp/go-msgpack v0.5.3/go.mod h1:ahLV/dePpqEmjfWmKiqvPkv/twdG7iPBM1vqhUKIvfM= github.com/hashicorp/go-msgpack v1.1.5 h1:9byZdVjKTe5mce63pRVNP1L7UAmdHOTEMGehn6KvJWs= github.com/hashicorp/go-msgpack v1.1.5/go.mod h1:gWVc3sv/wbDmR3rQsj1CAktEZzoz1YNK9NfGLXJ69/4= +github.com/hashicorp/go-msgpack/v2 v2.1.1 h1:xQEY9yB2wnHitoSzk/B9UjXWRQ67QKu5AOm8aFp8N3I= +github.com/hashicorp/go-msgpack/v2 v2.1.1/go.mod h1:upybraOAblm4S7rx0+jeNy+CWWhzywQsSRV5033mMu4= github.com/hashicorp/go-multierror v1.0.0/go.mod h1:dHtQlpGsu+cZNNAkkCN/P3hoUDHhCYQXV3UM06sGGrk= github.com/hashicorp/go-multierror v1.1.0/go.mod h1:spPvp8C1qA32ftKqdAHm4hHTbPw+vmowP0z+KUhOZdA= github.com/hashicorp/go-multierror v1.1.1 h1:H5DkEtf6CXdFp0N0Em5UCwQpXMWke8IA0+lD48awMYo= @@ -442,6 +455,10 @@ github.com/hashicorp/logutils v1.0.0/go.mod h1:QIAnNjmIWmVIIkWDTG1z5v++HQmx9WQRO github.com/hashicorp/mdns v1.0.4/go.mod h1:mtBihi+LeNXGtG8L9dX59gAEa12BDtBQSp4v/YAJqrc= github.com/hashicorp/nomad/api v0.0.0-20240306004928-3e7191ccb702 h1:fI1LXuBaS1d9z1kmb++Og6YD8uMRwadXorCwE+xgOFA= github.com/hashicorp/nomad/api v0.0.0-20240306004928-3e7191ccb702/go.mod h1:z71gkJdrkAt/Rl6C7Q79VE7AwJ5lUF+M+fzFTyIHYB0= +github.com/hashicorp/raft v1.7.0 h1:4u24Qn6lQ6uwziM++UgsyiT64Q8GyRn43CV41qPiz1o= +github.com/hashicorp/raft v1.7.0/go.mod h1:N1sKh6Vn47mrWvEArQgILTyng8GoDRNYlgKyK7PMjs0= +github.com/hashicorp/raft-wal v0.4.1 h1:aU8XZ6x8R9BAIB/83Z1dTDtXvDVmv9YVYeXxd/1QBSA= +github.com/hashicorp/raft-wal v0.4.1/go.mod h1:A6vP5o8hGOs1LHfC1Okh9xPwWDcmb6Vvuz/QyqUXlOE= github.com/hashicorp/serf v0.10.1 h1:Z1H2J60yRKvfDYAOZLd2MU0ND4AH/WDz7xYHDWQsIPY= github.com/hashicorp/serf v0.10.1/go.mod h1:yL2t6BqATOLGc5HF7qbFkTfXoPIY0WZdWHfEvMqbG+4= github.com/hetznercloud/hcloud-go/v2 v2.6.0 h1:RJOA2hHZ7rD1pScA4O1NF6qhkHyUdbbxjHgFNot8928= @@ -450,6 +467,8 @@ github.com/hexops/gotextdiff v1.0.3 h1:gitA9+qJrrTCsiCl7+kh75nPqQt1cx4ZkudSTLoUq github.com/hexops/gotextdiff v1.0.3/go.mod h1:pSWU5MAI3yDq+fZBTazCSJysOMbxWL1BSow5/V2vxeg= github.com/huaweicloud/huaweicloud-sdk-go-obs v3.23.3+incompatible h1:tKTaPHNVwikS3I1rdyf1INNvgJXWSf/+TzqsiGbrgnQ= github.com/huaweicloud/huaweicloud-sdk-go-obs v3.23.3+incompatible/go.mod h1:l7VUhRbTKCzdOacdT4oWCwATKyvZqUOlOqr0Ous3k4s= +github.com/iancoleman/strcase v0.3.0 h1:nTXanmYxhfFAMjZL34Ov6gkzEsSJZ5DbhxWjvSASxEI= +github.com/iancoleman/strcase v0.3.0/go.mod h1:iwCmte+B7n89clKwxIoIXy/HfoL7AsD47ZCWhYzw7ho= github.com/ianlancetaylor/demangle v0.0.0-20181102032728-5e5cf60278f6/go.mod h1:aSSvb/t6k1mPoxDqO4vJh6VOCGPwU4O0C2/Eqndh1Sc= github.com/ianlancetaylor/demangle v0.0.0-20210905161508-09a460cdf81d/go.mod h1:aYm2/VgdVmcIU8iMfdMvDMsRAQjcfZSKFby6HOFvi/w= github.com/imdario/mergo v0.3.16 h1:wwQJbIsHYGMUyLSPrEq1CT16AhnhNJQ51+4fdHUnCl4= @@ -480,7 +499,6 @@ github.com/k0kubun/pp/v3 v3.2.0 h1:h33hNTZ9nVFNP3u2Fsgz8JXiF5JINoZfFq4SvKJwNcs= github.com/k0kubun/pp/v3 v3.2.0/go.mod h1:ODtJQbQcIRfAD3N+theGCV1m/CBxweERz2dapdz1EwA= github.com/kisielk/errcheck v1.5.0/go.mod h1:pFxgyoBC7bSaBwPgfKdkLd5X25qrDl4LWUI2bnpBCr8= github.com/kisielk/gotool v1.0.0/go.mod h1:XhKaO+MFFWcvkIS/tQcRk01m1F5IRFswLeQ+oQHNcck= -github.com/klauspost/compress v1.17.3/go.mod h1:/dCuZOvVtNoHsyb+cuJD3itjs3NbnF6KH9zAO4BDxPM= github.com/klauspost/compress v1.17.9 h1:6KIumPrER1LHsvBVuDa0r5xaG0Es51mhhB9BQB2qeMA= github.com/klauspost/compress v1.17.9/go.mod h1:Di0epgTjJY877eYKx5yC51cX2A2Vl2ibi7bDH9ttBbw= github.com/klauspost/cpuid/v2 v2.0.1/go.mod h1:FInQzS24/EEf25PyTYn52gqo7WaD8xa0213Md/qVLRg= @@ -714,6 +732,8 @@ github.com/yuin/goldmark v1.1.27/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9de github.com/yuin/goldmark v1.1.32/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74= github.com/yuin/goldmark v1.2.1/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74= github.com/yuin/goldmark v1.4.13/go.mod h1:6yULJ656Px+3vBD8DxQVa3kxgyrAnzto9xy5taEt/CY= +go.etcd.io/bbolt v1.3.10 h1:+BqfJTcCzTItrop8mq/lbzL8wSGtj94UO/3U31shqG0= +go.etcd.io/bbolt v1.3.10/go.mod h1:bK3UQLPJZly7IlNmV7uVHJDxfe5aK9Ll93e/74Y9oEQ= go.etcd.io/etcd/api/v3 v3.5.7 h1:sbcmosSVesNrWOJ58ZQFitHMdncusIifYcrBfwrlJSY= go.etcd.io/etcd/api/v3 v3.5.7/go.mod h1:9qew1gCdDDLu+VwmeG+iFpL+QlpHTo7iubavdVDgCAA= go.etcd.io/etcd/client/pkg/v3 v3.5.7 h1:y3kf5Gbp4e4q7egZdn5T7W9TSHUvkClN6u+Rq9mEOmg= diff --git a/go.work.sum b/go.work.sum index fca2569ad2..10144c38ff 100644 --- a/go.work.sum +++ b/go.work.sum @@ -524,6 +524,8 @@ github.com/Bose/minisentinel v0.0.0-20200130220412-917c5a9223bb/go.mod h1:WsAABb github.com/BurntSushi/toml v0.3.1 h1:WXkYYl6Yr3qBf1K79EBnL4mak0OimBfB0XUf9Vl28OQ= github.com/BurntSushi/xgb v0.0.0-20160522181843-27f122750802 h1:1BDTz0u9nC3//pOCMdNH+CiXJVYJh5UQNCOBG7jbELc= github.com/DataDog/datadog-go v3.2.0+incompatible h1:qSG2N4FghB1He/r2mFrWKCaL7dXCilEuNEeAn20fdD4= +github.com/DataDog/zstd v1.5.2 h1:vUG4lAyuPCXO0TLbXvPv7EB7cNK1QV/luu55UHLrrn8= +github.com/DataDog/zstd v1.5.2/go.mod h1:g4AWEaM3yOg3HYfnJ3YIawPnVdXJh9QME85blwSAmyw= github.com/KimMachineGun/automemlimit v0.5.0 h1:BeOe+BbJc8L5chL3OwzVYjVzyvPALdd5wxVVOWuUZmQ= github.com/KimMachineGun/automemlimit v0.5.0/go.mod h1:di3GCKiu9Y+1fs92erCbUvKzPkNyViN3mA0vti/ykEQ= github.com/Knetic/govaluate v3.0.1-0.20171022003610-9aa49832a739+incompatible h1:1G1pk05UrOh0NlF1oeaaix1x8XzrfjIDK47TY0Zehcw= @@ -542,6 +544,8 @@ github.com/OneOfOne/xxhash v1.2.6/go.mod h1:eZbhyaAYD41SGSSsnmcpxVoRiQ/MPUTjUdII github.com/PuerkitoBio/purell v1.1.1 h1:WEQqlqaGbrPkxLJWfBwQmfEAE1Z7ONdDLqrN38tNFfI= github.com/PuerkitoBio/urlesc v0.0.0-20170810143723-de5bf2ad4578 h1:d+Bc7a5rLufV/sSk/8dngufqelfh6jnri85riMAaF/M= github.com/QcloudApi/qcloud_sign_golang v0.0.0-20141224014652-e4130a326409 h1:DTQ/38ao/CfXsrK0cSAL+h4R/u0VVvfWLZEOlLwEROI= +github.com/Sereal/Sereal/Go/sereal v0.0.0-20231009093132-b9187f1a92c6 h1:5kUcJJAKWWI82Xnp/CaU0eu5hLlHkmm9acjowSkwCd0= +github.com/Sereal/Sereal/Go/sereal v0.0.0-20231009093132-b9187f1a92c6/go.mod h1:JwrycNnC8+sZPDyzM3MQ86LvaGzSpfxg885KOOwFRW4= github.com/VividCortex/gohistogram v1.0.0 h1:6+hBz+qvs0JOrrNhhmR7lFxo5sINxBCGXrdtl/UvroE= github.com/VividCortex/gohistogram v1.0.0/go.mod h1:Pf5mBqqDxYaXu3hDrrU+w6nw50o/4+TcAqDqk/vUH7g= github.com/afex/hystrix-go v0.0.0-20180502004556-fa1af6a1f4f5 h1:rFw4nCn9iMW+Vajsk51NtYIcwSTkXr+JGrMd36kTDJw= @@ -576,11 +580,17 @@ github.com/aws/aws-sdk-go-v2/service/cloudwatch v1.8.1 h1:w/fPGB0t5rWwA43mux4e9o github.com/aws/aws-sdk-go-v2/service/cloudwatch v1.8.1/go.mod h1:CM+19rL1+4dFWnOQKwDc7H1KwXTz+h61oUSHyhV0b3o= github.com/baiyubin/aliyun-sts-go-sdk v0.0.0-20180326062324-cfa1a18b161f h1:ZNv7On9kyUzm7fvRZumSyy/IUiSC7AzL0I1jKKtwooA= github.com/baiyubin/aliyun-sts-go-sdk v0.0.0-20180326062324-cfa1a18b161f/go.mod h1:AuiFmCCPBSrqvVMvuqFuk0qogytodnVFVSN5CeJB8Gc= +github.com/benmathews/bench v0.0.0-20210120214102-f7c75b9ef6e7 h1:nYTgFk9sOL3rmNew6rR2anUWWCzmSYPMJiSmowV8Yls= +github.com/benmathews/bench v0.0.0-20210120214102-f7c75b9ef6e7/go.mod h1:peX7BEhSFSvvnxdido50pUMhlFi24dVgtTU1oZkHTUU= +github.com/benmathews/hdrhistogram-writer v0.0.0-20210120211942-3cb1c7c33f95 h1:tAEzz8rP6JRzrARM5HecEuhY23qL2CSGRTFcNzwjOWI= +github.com/benmathews/hdrhistogram-writer v0.0.0-20210120211942-3cb1c7c33f95/go.mod h1:2MBckC8FahPaeLz58Qe6ZyVKm8UU1gHkINEv9Sw7pnI= github.com/bgentry/speakeasy v0.1.0 h1:ByYyxL9InA1OWqxJqqp2A5pYHUrCiAL6K3J+LKSsQkY= github.com/bitly/go-simplejson v0.5.0 h1:6IH+V8/tVMab511d5bn4M7EwGXZf9Hj6i2xSwkNEM+Y= github.com/bitly/go-simplejson v0.5.0/go.mod h1:cXHtHw4XUPsvGaxgjIAn8PhEWG9NfngEKAMDJEczWVA= github.com/bmizerany/assert v0.0.0-20160611221934-b7ed37b82869 h1:DDGfHa7BWjL4YnC6+E63dPcxHo2sUxDIu8g3QgEJdRY= github.com/bmizerany/assert v0.0.0-20160611221934-b7ed37b82869/go.mod h1:Ekp36dRnpXw/yCqJaO+ZrUyxD+3VXMFFr56k5XYrpB4= +github.com/boltdb/bolt v1.3.1 h1:JQmyP4ZBrce+ZQu0dY660FMfatumYDLun9hBCUVIkF4= +github.com/boltdb/bolt v1.3.1/go.mod h1:clJnj/oiGkjum5o1McbSZDSLxVThjynRyGBgiAx27Ps= github.com/bsm/redislock v0.9.1 h1:uTTZU82xg2PjI8X5T9PGcX/5k1FX3Id7bqkwy1As6c0= github.com/bsm/redislock v0.9.1/go.mod h1:ToFoB1xQbOJYG7e2ZBiPXotlhImqWgEa4+u/lLQ1nSc= github.com/casbin/casbin/v2 v2.37.0 h1:/poEwPSovi4bTOcP752/CsTQiRz2xycyVKFG7GUhbDw= @@ -634,6 +644,8 @@ github.com/creack/pty v1.1.11 h1:07n33Z8lZxZ2qwegKbObQohDhXDQxiMMz1NOUGYlesw= github.com/creack/pty v1.1.11/go.mod h1:oKZEueFk5CKHvIhNR5MUki03XCEU+Q6VDXinZuGJ33E= github.com/cristalhq/hedgedhttp v0.9.1 h1:g68L9cf8uUyQKQJwciD0A1Vgbsz+QgCjuB1I8FAsCDs= github.com/cristalhq/hedgedhttp v0.9.1/go.mod h1:XkqWU6qVMutbhW68NnzjWrGtH8NUx1UfYqGYtHVKIsI= +github.com/davecgh/go-xdr v0.0.0-20161123171359-e6a2ba005892 h1:qg9VbHo1TlL0KDM0vYvBG9EY0X0Yku5WYIPoFWt8f6o= +github.com/davecgh/go-xdr v0.0.0-20161123171359-e6a2ba005892/go.mod h1:CTDl0pzVzE5DEzZhPfvhY/9sPFMQIxaJ9VAMs9AagrE= github.com/dgryski/go-rendezvous v0.0.0-20200823014737-9f7001d12a5f h1:lO4WD4F/rVNCu3HqELle0jiPLLBs70cWOduZpkS1E78= github.com/dgryski/go-rendezvous v0.0.0-20200823014737-9f7001d12a5f/go.mod h1:cuUVRXasLTGF7a8hSLbxyZXjz+1KgoB3wDUb6vlszIc= github.com/digitalocean/godo v1.104.1/go.mod h1:VAI/L5YDzMuPRU01lEEUSQ/sp5Z//1HnnFv/RBTEdbg= @@ -690,6 +702,8 @@ github.com/go-openapi/swag v0.22.5/go.mod h1:Gl91UqO+btAM0plGGxHqJcQZ1ZTy6jbmrid github.com/go-openapi/swag v0.22.6/go.mod h1:Gl91UqO+btAM0plGGxHqJcQZ1ZTy6jbmridBTsDy8A0= github.com/go-playground/assert/v2 v2.0.1 h1:MsBgLAaY856+nPRTKrp3/OZK38U/wa0CcBYNjji3q3A= github.com/go-playground/assert/v2 v2.0.1/go.mod h1:VDjEfimB/XKnb+ZQfWdccd7VUvScMdVu0Titje2rxJ4= +github.com/go-playground/assert/v2 v2.2.0 h1:JvknZsQTYeFEAhQwI4qEt9cyV5ONwRHC+lYKSsYSR8s= +github.com/go-playground/assert/v2 v2.2.0/go.mod h1:VDjEfimB/XKnb+ZQfWdccd7VUvScMdVu0Titje2rxJ4= github.com/go-playground/locales v0.13.0 h1:HyWk6mgj5qFqCT5fjGBuRArbVDfE4hi8+e8ceBS/t7Q= github.com/go-playground/locales v0.13.0/go.mod h1:taPMhCMXrRLJO55olJkUXHZBHCxTMfnGwq/HNwmWNS8= github.com/go-playground/universal-translator v0.17.0 h1:icxd5fm+REJzpZx7ZfpaD876Lmtgy7VtROAbHHXk8no= @@ -749,7 +763,6 @@ github.com/google/flatbuffers v2.0.8+incompatible h1:ivUb1cGomAB101ZM1T0nOiWz9pS github.com/google/flatbuffers v2.0.8+incompatible/go.mod h1:1AeVuKshWv4vARoZatz6mlQ0JxURH0Kv5+zNeJKJCa8= github.com/google/go-cmp v0.5.6/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= github.com/google/go-cmp v0.5.7/go.mod h1:n+brtR0CgQNWTVd5ZUFpTBC8YFBDLK/h/bpaJ8/DtOE= -github.com/google/go-cmp v0.5.9/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY= github.com/google/go-pkcs11 v0.2.1-0.20230907215043-c6f79328ddf9 h1:OF1IPgv+F4NmqmJ98KTjdN97Vs1JxDPB3vbmYzV2dpk= github.com/google/go-pkcs11 v0.2.1-0.20230907215043-c6f79328ddf9/go.mod h1:6eQoGcuNJpa7jnd5pMGdkSaQpNDYvPlXWMcjXXThLlY= github.com/google/martian/v3 v3.1.0/go.mod h1:y5Zk1BBys9G+gd6Jrk0W3cC1+ELVxBWuIGO+w/tUAp0= @@ -795,7 +808,6 @@ github.com/hashicorp/go-cleanhttp v0.5.1/go.mod h1:JpRdi6/HCYpAwUzNwuwqhbovhLtng github.com/hashicorp/go-hclog v0.9.2/go.mod h1:5CU+agLiy3J7N7QjHK5d05KxGsuXiQLrjA0H7acj2lQ= github.com/hashicorp/go-hclog v0.12.0/go.mod h1:whpDNt7SSdeAju8AWKIWsul05p54N/39EeqMAyrmvFQ= github.com/hashicorp/go-hclog v1.2.0/go.mod h1:whpDNt7SSdeAju8AWKIWsul05p54N/39EeqMAyrmvFQ= -github.com/hashicorp/go-msgpack v0.5.5/go.mod h1:ahLV/dePpqEmjfWmKiqvPkv/twdG7iPBM1vqhUKIvfM= github.com/hashicorp/go-syslog v1.0.0 h1:KaodqZuhUoZereWVIYmpUgZysurB1kBLX2j0MwMrUAE= github.com/hashicorp/go-version v1.2.1/go.mod h1:fltr4n8CU8Ke44wwGCBoEymUuxUHl09ZGVZPK5anwXA= github.com/hashicorp/golang-lru v0.5.4/go.mod h1:iADmTwqILo4mZ8BN3D2Q6+9jd8WM5uGBxy+E8yxSoD4= @@ -804,6 +816,10 @@ github.com/hashicorp/hcl v1.0.0/go.mod h1:E5yfLk+7swimpb2L/Alb/PJmXilQ/rhwaUYs4T github.com/hashicorp/logutils v1.0.0 h1:dLEQVugN8vlakKOUE3ihGLTZJRB4j+M2cdTm/ORI65Y= github.com/hashicorp/mdns v1.0.4 h1:sY0CMhFmjIPDMlTB+HfymFHCaYLhgifZ0QhjaYKD/UQ= github.com/hashicorp/nomad/api v0.0.0-20230721134942-515895c7690c/go.mod h1:O23qLAZuCx4htdY9zBaO4cJPXgleSFEdq6D/sezGgYE= +github.com/hashicorp/raft-boltdb v0.0.0-20220329195025-15018e9b97e0 h1:CO8dBMLH6dvE1jTn/30ZZw3iuPsNfajshWoJTnVc5cc= +github.com/hashicorp/raft-boltdb v0.0.0-20220329195025-15018e9b97e0/go.mod h1:nTakvJ4XYq45UXtn0DbwR4aU9ZdjlnIenpbs6Cd+FM0= +github.com/hashicorp/raft-boltdb/v2 v2.2.2 h1:rlkPtOllgIcKLxVT4nutqlTH2NRFn+tO1wwZk/4Dxqw= +github.com/hashicorp/raft-boltdb/v2 v2.2.2/go.mod h1:N8YgaZgNJLpZC+h+by7vDu5rzsRgONThTEeUS3zWbfY= github.com/hashicorp/serf v0.9.6/go.mod h1:TXZNMjZQijwlDvp+r0b63xZ45H7JmCmgg4gpTwn9UV4= github.com/hashicorp/serf v0.9.7/go.mod h1:TXZNMjZQijwlDvp+r0b63xZ45H7JmCmgg4gpTwn9UV4= github.com/hetznercloud/hcloud-go/v2 v2.4.0/go.mod h1:l7fA5xsncFBzQTyw29/dw5Yr88yEGKKdc6BHf24ONS0= @@ -814,8 +830,6 @@ github.com/hudl/fargo v1.4.0 h1:ZDDILMbB37UlAVLlWcJ2Iz1XuahZZTDZfdCKeclfq2s= github.com/hudl/fargo v1.4.0/go.mod h1:9Ai6uvFy5fQNq6VPKtg+Ceq1+eTY4nKUlR2JElEOcDo= github.com/iancoleman/strcase v0.2.0 h1:05I4QRnGpI0m37iZQRuskXh+w77mr6Z41lwQzuHLwW0= github.com/iancoleman/strcase v0.2.0/go.mod h1:iwCmte+B7n89clKwxIoIXy/HfoL7AsD47ZCWhYzw7ho= -github.com/iancoleman/strcase v0.3.0 h1:nTXanmYxhfFAMjZL34Ov6gkzEsSJZ5DbhxWjvSASxEI= -github.com/iancoleman/strcase v0.3.0/go.mod h1:iwCmte+B7n89clKwxIoIXy/HfoL7AsD47ZCWhYzw7ho= github.com/ianlancetaylor/demangle v0.0.0-20200824232613-28f6c0f3b639/go.mod h1:aSSvb/t6k1mPoxDqO4vJh6VOCGPwU4O0C2/Eqndh1Sc= github.com/imdario/mergo v0.3.11/go.mod h1:jmQim1M+e3UYxmgPu/WyfjB3N3VflVyUjjjwH0dnCYA= github.com/inconshreveable/mousetrap v1.0.0 h1:Z8tu5sraLXCXIcARxBp/8cbvlwVa7Z1NHg9XEKhtSvM= @@ -837,6 +851,7 @@ github.com/kisielk/gotool v1.0.0 h1:AV2c/EiW3KqPNT9ZKl07ehoAGi4C5/01Cfbblndcapg= github.com/klauspost/asmfmt v1.3.2 h1:4Ri7ox3EwapiOjCki+hw14RyKk201CN4rzyCJRFLpK4= github.com/klauspost/asmfmt v1.3.2/go.mod h1:AG8TuvYojzulgDAMCnYn50l/5QV3Bs/tp6j0HLHbNSE= github.com/klauspost/compress v1.17.1/go.mod h1:ntbaceVETuRiXiv4DpjP66DpAtAGkEQskQzEyD//IeE= +github.com/knz/go-libedit v1.10.1 h1:0pHpWtx9vcvC0xGZqEQlQdfSQs7WRlAjuPvk3fOZDCo= github.com/konsorten/go-windows-terminal-sequences v1.0.3 h1:CE8S1cTafDpPvMhIxNJKvHsGVBgn1xWYf1NbHQhywc8= github.com/kr/fs v0.1.0 h1:Jskdu9ieNAYnjxsi0LbQp1ulIKZV1LAFgK1tWhpZgl8= github.com/kr/fs v0.1.0/go.mod h1:FFnZGqtBN9Gxj7eW1uZ42v5BccTP0vu6NEaFoC2HwRg= @@ -951,6 +966,8 @@ github.com/performancecopilot/speed/v4 v4.0.0 h1:VxEDCmdkfbQYDlcr/GC9YoN9PQ6p8ul github.com/performancecopilot/speed/v4 v4.0.0/go.mod h1:qxrSyuDGrTOWfV+uKRFhfxw6h/4HXRGUiZiufxo49BM= github.com/peterbourgon/diskv v2.0.1+incompatible h1:UBdAOUP5p4RWqPBg048CAvpKN+vxiaj6gdUUzhl4XmI= github.com/peterbourgon/diskv v2.0.1+incompatible/go.mod h1:uqqh8zWWbv1HBMNONnaR/tNboyR3/BZd58JJSHlUSCU= +github.com/philhofer/fwd v1.1.2 h1:bnDivRJ1EWPjUIRXV5KfORO897HTbpFAQddBdE8t7Gw= +github.com/philhofer/fwd v1.1.2/go.mod h1:qkPdfjR2SIEbspLqpe1tO4n5yICnr2DY7mqEx2tUTP0= github.com/pkg/browser v0.0.0-20210911075715-681adbf594b8/go.mod h1:HKlIX3XHQyzLZPlr7++PzdhaXEj94dEiJgZDTsxEqUI= github.com/pkg/diff v0.0.0-20210226163009-20ebb0f2a09e h1:aoZm08cpOy4WuID//EZDgcC4zIxODThtZNPirFr42+A= github.com/pkg/diff v0.0.0-20210226163009-20ebb0f2a09e/go.mod h1:pJLUxLENpZxwdsKMEsNbx1VGcRFpLqf3715MtcvvzbA= @@ -959,6 +976,8 @@ github.com/pkg/sftp v1.13.1/go.mod h1:3HaPG6Dq1ILlpPZRO0HVMrsydcdLt6HRDccSgb87qR github.com/pkg/sftp v1.13.6 h1:JFZT4XbOU7l77xGSpOdW+pwIMqP044IyjXX6FGyEKFo= github.com/pkg/sftp v1.13.6/go.mod h1:tz1ryNURKu77RL+GuCzmoJYxQczL3wLNNpPWagdg4Qk= github.com/posener/complete v1.2.3 h1:NP0eAhjcjImqslEwo/1hq7gpajME0fTLTezBKDqfXqo= +github.com/pquerna/ffjson v0.0.0-20190930134022-aa0246cd15f7 h1:xoIK0ctDddBMnc74udxJYBqlo9Ylnsp1waqjLsnef20= +github.com/pquerna/ffjson v0.0.0-20190930134022-aa0246cd15f7/go.mod h1:YARuvh7BUWHNhzDq2OM5tzR2RiCcN2D7sapiKyCel/M= github.com/prashantv/gostub v1.1.0 h1:BTyx3RfQjRHnUWaGF9oQos79AlQ5k8WNktv7VGvVH4g= github.com/prashantv/gostub v1.1.0/go.mod h1:A5zLQHz7ieHGG7is6LLXLz7I8+3LZzsrV0P1IAHhP5U= github.com/prometheus/client_golang v1.11.1/go.mod h1:Z6t4BnS23TR94PD6BsDNk8yVqroYurpAkEiz0P2BEV0= @@ -996,6 +1015,8 @@ github.com/satori/go.uuid v1.2.1-0.20181028125025-b2ce2384e17b h1:gQZ0qzfKHQIybL github.com/satori/go.uuid v1.2.1-0.20181028125025-b2ce2384e17b/go.mod h1:dA0hQrYB0VpLJoorglMZABFdXlWrHn1NEOzdhQKdks0= github.com/scaleway/scaleway-sdk-go v1.0.0-beta.21/go.mod h1:fCa7OJZ/9DRTnOKmxvT6pn+LPWUptQAmHF/SBJUGEcg= github.com/segmentio/asm v1.1.3 h1:WM03sfUOENvvKexOLp+pCqgb/WDjsi7EK8gIsICtzhc= +github.com/segmentio/fasthash v1.0.3 h1:EI9+KE1EwvMLBWwjpRDc+fEM+prwxDYbslddQGtrmhM= +github.com/segmentio/fasthash v1.0.3/go.mod h1:waKX8l2N8yckOgmSsXJi7x1ZfdKZ4x7KRMzBtS3oedY= github.com/shoenig/test v0.6.6 h1:Oe8TPH9wAbv++YPNDKJWUnI8Q4PPWCx3UbOfH+FxiMU= github.com/shoenig/test v0.6.6/go.mod h1:byHiCGXqrVaflBLAMq/srcZIHynQPQgeyvkvXnjqq0k= github.com/shopspring/decimal v1.2.0 h1:abSATXmQEYyShuxI4/vyW3tV1MrKAJzCZ/0zLUXYbsQ= @@ -1029,7 +1050,6 @@ github.com/streadway/handy v0.0.0-20200128134331-0f66f006fb2e/go.mod h1:qNTQ5P5J github.com/stretchr/testify v1.5.1/go.mod h1:5W2xD1RspED5o8YsWQXVCued0rvSQ+mT+I5cxcmMvtA= github.com/stretchr/testify v1.7.5/go.mod h1:yNjHg4UonilssWZ8iaSj1OCr/vHnekPRkoO+kdMU+MU= github.com/stretchr/testify v1.8.3/go.mod h1:sz/lmYIOXD/1dqDmKjjqLyZ2RngseejIcXlSw2iwfAo= -github.com/stretchr/testify v1.8.4/go.mod h1:sz/lmYIOXD/1dqDmKjjqLyZ2RngseejIcXlSw2iwfAo= github.com/subosito/gotenv v1.4.1/go.mod h1:ayKnFf/c6rvx/2iiLrJUk1e6plDbT3edrFNGqEflhK0= github.com/subosito/gotenv v1.4.2 h1:X1TuBLAMDFbaTAChgCBLu3DU3UPyELpnF2jjJ2cz/S8= github.com/subosito/gotenv v1.4.2/go.mod h1:ayKnFf/c6rvx/2iiLrJUk1e6plDbT3edrFNGqEflhK0= @@ -1038,6 +1058,8 @@ github.com/tencentcloud/tencentcloud-sdk-go/tencentcloud/kms v1.0.194 h1:YB6qJyC github.com/thanos-io/objstore v0.0.0-20230727115635-d0c43443ecda h1:DtxaU/a7QRPiUhwtPrZFlS81y+9Mgny4KoLq65cu04U= github.com/thanos-io/objstore v0.0.0-20230727115635-d0c43443ecda/go.mod h1:IS7Z25+0KaknyU2P5PTP/5hwY6Yr/FzbInF88Yd5auU= github.com/tidwall/pretty v1.0.0 h1:HsD+QiTn7sK6flMKIvNmpqz1qrpP3Ps6jOKIKMooyg4= +github.com/tinylib/msgp v1.1.8 h1:FCXC1xanKO4I8plpHGH2P7koL/RzZs12l/+r7vakfm0= +github.com/tinylib/msgp v1.1.8/go.mod h1:qkpG+2ldGg4xRFmx+jfTvZPxfGFhi64BcnL9vkCm/Tw= github.com/tv42/httpunix v0.0.0-20150427012821-b75d8614f926 h1:G3dpKMzFDjgEh2q1Z7zUUtKa8ViPtH+ocF0bE0g00O8= github.com/vmihailenco/msgpack/v5 v5.3.5 h1:5gO0H1iULLWGhs2H5tbAHIZTV8/cYafcFOr9znI5mJU= github.com/vmihailenco/msgpack/v5 v5.3.5/go.mod h1:7xyJ9e+0+9SaZT0Wt1RGleJXzli6Q/V5KbhBonMG9jc= @@ -1074,6 +1096,8 @@ go.etcd.io/etcd/client/v2 v2.305.0/go.mod h1:h9puh54ZTgAKtEbut2oe9P4L/oqKCVB6xsX go.etcd.io/etcd/client/v2 v2.305.4 h1:Dcx3/MYyfKcPNLpR4VVQUP5KgYrBeJtktBwEKkw08Ao= go.etcd.io/etcd/client/v2 v2.305.4/go.mod h1:Ud+VUwIi9/uQHOMA+4ekToJ12lTxlv0zB/+DHwTGEbU= go.etcd.io/etcd/client/v3 v3.5.4/go.mod h1:ZaRkVgBZC+L+dLCjTcF1hRXpgZXQPOvnA/Ak/gq3kiY= +go.etcd.io/gofail v0.1.0 h1:XItAMIhOojXFQMgrxjnd2EIIHun/d5qL0Pf7FzVTkFg= +go.etcd.io/gofail v0.1.0/go.mod h1:VZBCXYGZhHAinaBiiqYvuDynvahNsAyLFwB3kEHKz1M= go.mongodb.org/mongo-driver v1.13.1/go.mod h1:wcDf1JBCXy2mOW0bWHwO/IOYqdca1MPCwDtFu/Z9+eo= go.opencensus.io v0.22.5/go.mod h1:5pWMHQbX5EPX2/62yrJeAkowc+lfs/XD7Uxpq3pI6kk= go.opencensus.io v0.23.0/go.mod h1:XItmlyltB5F7CS4xOC1DcqMoFqwtC6OG2xF7mCv7P7E= @@ -1219,6 +1243,8 @@ golang.org/x/tools v0.17.0/go.mod h1:xsh6VxdV005rRVaS6SSAf9oiAqljS7UZUacMZ8Bnsps golang.org/x/xerrors v0.0.0-20220411194840-2f41105eb62f/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= golang.org/x/xerrors v0.0.0-20220517211312-f3a8303e98df/go.mod h1:K8+ghG5WaK9qNqU5K3HdILfMLy1f3aNYFI/wnl100a8= gonum.org/v1/gonum v0.8.2 h1:CCXrcPKiGGotvnN6jfUsKk4rRqm7q09/YbKb5xCEvtM= +gonum.org/v1/gonum v0.12.0 h1:xKuo6hzt+gMav00meVPUlXwSdoEJP46BR+wdxQEFK2o= +gonum.org/v1/gonum v0.12.0/go.mod h1:73TDxJfAAHeA8Mk9mf8NlIppyhQNo5GLTcYeqgo2lvY= gonum.org/v1/netlib v0.0.0-20190313105609-8cb42192e0e0 h1:OE9mWmgKkjJyEmDAAtGMPjXu+YNeGvK9VTSHY6+Qihc= gonum.org/v1/plot v0.0.0-20190515093506-e2840ee46a6b h1:Qh4dB5D/WpoUUp3lSod7qgoyEHbDGPUWjIbnqdqqe1k= google.golang.org/api v0.35.0/go.mod h1:/XrVsuzM0rZmrsbjJutiuftIzeuTQcEeaYcSk/mQ1dg= @@ -1367,6 +1393,8 @@ google.golang.org/protobuf v1.33.0/go.mod h1:c6P6GXX6sHbq/GpV6MGZEdwhWPcYBgnhAHh gopkg.in/errgo.v2 v2.1.0 h1:0vLT13EuvQ0hNvakwLuFZ/jYrLp5F3kcWHXdRggjCE8= gopkg.in/gcfg.v1 v1.2.3 h1:m8OOJ4ccYHnx2f4gQwpno8nAX5OGOh7RLaaz0pj3Ogs= gopkg.in/gcfg.v1 v1.2.3/go.mod h1:yesOnuUOFQAhST5vPY4nbZsb/huCgGGXlipJsBn0b3o= +gopkg.in/mgo.v2 v2.0.0-20190816093944-a6b53ec6cb22 h1:VpOs+IwYnYBaFnrNAeB8UUWtL3vEUnzSCL1nVjPhqrw= +gopkg.in/mgo.v2 v2.0.0-20190816093944-a6b53ec6cb22/go.mod h1:yeKp02qBN3iKW1OzL3MGk2IdtZzaj7SFntXj72NppTA= gopkg.in/natefinch/lumberjack.v2 v2.2.1 h1:bBRl1b0OH9s/DuPhuXpNl+VtCaJXFZ5/uEFST95x9zc= gopkg.in/natefinch/lumberjack.v2 v2.2.1/go.mod h1:YD8tP3GAjkrDg1eZH7EGmyESg/lsYskCTPBJVb9jqSc= gopkg.in/square/go-jose.v2 v2.6.0 h1:NGk74WTnPKBNUhNzQX7PYcTLUjoq7mzKk2OKbvwk2iI= @@ -1377,6 +1405,8 @@ gopkg.in/telebot.v3 v3.2.1 h1:3I4LohaAyJBiivGmkfB+CiVu7QFOWkuZ4+KHgO/G3rs= gopkg.in/telebot.v3 v3.2.1/go.mod h1:GJKwwWqp9nSkIVN51eRKU78aB5f5OnQuWdwiIZfPbko= gopkg.in/tomb.v1 v1.0.0-20141024135613-dd632973f1e7 h1:uRGJdciOHaEIrze2W8Q3AKkepLTh2hOroT7a+7czfdQ= gopkg.in/tomb.v1 v1.0.0-20141024135613-dd632973f1e7/go.mod h1:dt/ZhP58zS4L8KSrWDmTeBkI65Dw0HsyUHuEVlX15mw= +gopkg.in/vmihailenco/msgpack.v2 v2.9.2 h1:gjPqo9orRVlSAH/065qw3MsFCDpH7fa1KpiizXyllY4= +gopkg.in/vmihailenco/msgpack.v2 v2.9.2/go.mod h1:/3Dn1Npt9+MYyLpYYXjInO/5jvMLamn+AEGwNEOatn8= gopkg.in/warnings.v0 v0.1.2 h1:wFXVbFY8DY5/xOe1ECiWdKCzZlxgshcYVNkBHstARME= gopkg.in/warnings.v0 v0.1.2/go.mod h1:jksf8JmL6Qr/oQM2OXTHunEvvTAsrWBLb6OOjuVWRNI= gopkg.in/yaml.v2 v2.2.3/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= @@ -1391,11 +1421,14 @@ k8s.io/gengo v0.0.0-20201113003025-83324d819ded h1:JApXBKYyB7l9xx+DK7/+mFjC7A9Bt k8s.io/gengo v0.0.0-20201113003025-83324d819ded/go.mod h1:FiNAH4ZV3gBg2Kwh89tzAEV2be7d5xI0vBa/VySYy3E= k8s.io/gengo v0.0.0-20210813121822-485abfe95c7c h1:GohjlNKauSai7gN4wsJkeZ3WAJx4Sh+oT/b5IYn5suA= k8s.io/gengo v0.0.0-20210813121822-485abfe95c7c/go.mod h1:FiNAH4ZV3gBg2Kwh89tzAEV2be7d5xI0vBa/VySYy3E= +k8s.io/gengo v0.0.0-20230829151522-9cce18d56c01 h1:pWEwq4Asjm4vjW7vcsmijwBhOr1/shsbSYiWXmNGlks= +k8s.io/gengo v0.0.0-20230829151522-9cce18d56c01/go.mod h1:FiNAH4ZV3gBg2Kwh89tzAEV2be7d5xI0vBa/VySYy3E= k8s.io/klog v1.0.0 h1:Pt+yjF5aB1xDSVbau4VsWe+dQNzA0qv1LlXdC2dF6Q8= k8s.io/klog v1.0.0/go.mod h1:4Bi6QPql/J/LkTDqv7R/cd3hPo4k2DG6Ptcz060Ez5I= k8s.io/klog/v2 v2.80.1/go.mod h1:y1WjHnz7Dj687irZUWR/WLkLc5N1YHtjLdmgWjndZn0= k8s.io/klog/v2 v2.100.1/go.mod h1:y1WjHnz7Dj687irZUWR/WLkLc5N1YHtjLdmgWjndZn0= k8s.io/kube-openapi v0.0.0-20230717233707-2695361300d9/go.mod h1:wZK2AVp1uHCp4VamDVgBP2COHZjqD1T68Rf0CM3YjSM= +nullprogram.com/x/optparse v1.0.0 h1:xGFgVi5ZaWOnYdac2foDT3vg0ZZC9ErXFV57mr4OHrI= rsc.io/binaryregexp v0.2.0 h1:HfqmD5MEmC0zvwBuF187nq9mdnXjXsSivRiXN7SmRkE= rsc.io/pdf v0.1.1 h1:k1MczvYDUvJBe93bYd7wrZLLUEcLZAuF824/I4e5Xr4= rsc.io/quote/v3 v3.1.0 h1:9JKUTTIUgS6kzR9mK1YuGKv6Nl+DijDNIc0ghT58FaY= diff --git a/operations/pyroscope/helm/pyroscope/rendered/micro-services-hpa.yaml b/operations/pyroscope/helm/pyroscope/rendered/micro-services-hpa.yaml index 8a706e88ef..396e644416 100644 --- a/operations/pyroscope/helm/pyroscope/rendered/micro-services-hpa.yaml +++ b/operations/pyroscope/helm/pyroscope/rendered/micro-services-hpa.yaml @@ -2134,6 +2134,13 @@ spec: - "-runtime-config.file=/etc/pyroscope/overrides/overrides.yaml" - "-log.level=debug" - "-store-gateway.sharding-ring.replication-factor=3" + env: + - name: POD_NAME + valueFrom: + fieldRef: + fieldPath: metadata.name + - name: NAMESPACE_FQDN + value: "default.svc.cluster.local." ports: - name: http2 containerPort: 4040 @@ -2227,6 +2234,13 @@ spec: - "-runtime-config.file=/etc/pyroscope/overrides/overrides.yaml" - "-log.level=debug" - "-store-gateway.sharding-ring.replication-factor=3" + env: + - name: POD_NAME + valueFrom: + fieldRef: + fieldPath: metadata.name + - name: NAMESPACE_FQDN + value: "default.svc.cluster.local." ports: - name: http2 containerPort: 4040 @@ -2320,6 +2334,13 @@ spec: - "-runtime-config.file=/etc/pyroscope/overrides/overrides.yaml" - "-log.level=debug" - "-store-gateway.sharding-ring.replication-factor=3" + env: + - name: POD_NAME + valueFrom: + fieldRef: + fieldPath: metadata.name + - name: NAMESPACE_FQDN + value: "default.svc.cluster.local." ports: - name: http2 containerPort: 4040 @@ -2413,6 +2434,13 @@ spec: - "-runtime-config.file=/etc/pyroscope/overrides/overrides.yaml" - "-log.level=debug" - "-store-gateway.sharding-ring.replication-factor=3" + env: + - name: POD_NAME + valueFrom: + fieldRef: + fieldPath: metadata.name + - name: NAMESPACE_FQDN + value: "default.svc.cluster.local." ports: - name: http2 containerPort: 4040 @@ -2807,6 +2835,13 @@ spec: - "-runtime-config.file=/etc/pyroscope/overrides/overrides.yaml" - "-log.level=debug" - "-store-gateway.sharding-ring.replication-factor=3" + env: + - name: POD_NAME + valueFrom: + fieldRef: + fieldPath: metadata.name + - name: NAMESPACE_FQDN + value: "default.svc.cluster.local." ports: - name: http2 containerPort: 4040 @@ -2908,6 +2943,13 @@ spec: - "-runtime-config.file=/etc/pyroscope/overrides/overrides.yaml" - "-log.level=debug" - "-store-gateway.sharding-ring.replication-factor=3" + env: + - name: POD_NAME + valueFrom: + fieldRef: + fieldPath: metadata.name + - name: NAMESPACE_FQDN + value: "default.svc.cluster.local." ports: - name: http2 containerPort: 4040 @@ -3005,6 +3047,13 @@ spec: - "-runtime-config.file=/etc/pyroscope/overrides/overrides.yaml" - "-log.level=debug" - "-store-gateway.sharding-ring.replication-factor=3" + env: + - name: POD_NAME + valueFrom: + fieldRef: + fieldPath: metadata.name + - name: NAMESPACE_FQDN + value: "default.svc.cluster.local." ports: - name: http2 containerPort: 4040 diff --git a/operations/pyroscope/helm/pyroscope/rendered/micro-services.yaml b/operations/pyroscope/helm/pyroscope/rendered/micro-services.yaml index 1a58cd3160..d9442c5cf5 100644 --- a/operations/pyroscope/helm/pyroscope/rendered/micro-services.yaml +++ b/operations/pyroscope/helm/pyroscope/rendered/micro-services.yaml @@ -2279,6 +2279,13 @@ spec: - "-runtime-config.file=/etc/pyroscope/overrides/overrides.yaml" - "-log.level=debug" - "-store-gateway.sharding-ring.replication-factor=3" + env: + - name: POD_NAME + valueFrom: + fieldRef: + fieldPath: metadata.name + - name: NAMESPACE_FQDN + value: "default.svc.cluster.local." ports: - name: http2 containerPort: 4040 @@ -2373,6 +2380,13 @@ spec: - "-runtime-config.file=/etc/pyroscope/overrides/overrides.yaml" - "-log.level=debug" - "-store-gateway.sharding-ring.replication-factor=3" + env: + - name: POD_NAME + valueFrom: + fieldRef: + fieldPath: metadata.name + - name: NAMESPACE_FQDN + value: "default.svc.cluster.local." ports: - name: http2 containerPort: 4040 @@ -2467,6 +2481,13 @@ spec: - "-runtime-config.file=/etc/pyroscope/overrides/overrides.yaml" - "-log.level=debug" - "-store-gateway.sharding-ring.replication-factor=3" + env: + - name: POD_NAME + valueFrom: + fieldRef: + fieldPath: metadata.name + - name: NAMESPACE_FQDN + value: "default.svc.cluster.local." ports: - name: http2 containerPort: 4040 @@ -2561,6 +2582,13 @@ spec: - "-runtime-config.file=/etc/pyroscope/overrides/overrides.yaml" - "-log.level=debug" - "-store-gateway.sharding-ring.replication-factor=3" + env: + - name: POD_NAME + valueFrom: + fieldRef: + fieldPath: metadata.name + - name: NAMESPACE_FQDN + value: "default.svc.cluster.local." ports: - name: http2 containerPort: 4040 @@ -2655,6 +2683,13 @@ spec: - "-runtime-config.file=/etc/pyroscope/overrides/overrides.yaml" - "-log.level=debug" - "-store-gateway.sharding-ring.replication-factor=3" + env: + - name: POD_NAME + valueFrom: + fieldRef: + fieldPath: metadata.name + - name: NAMESPACE_FQDN + value: "default.svc.cluster.local." ports: - name: http2 containerPort: 4040 @@ -2749,6 +2784,13 @@ spec: - "-runtime-config.file=/etc/pyroscope/overrides/overrides.yaml" - "-log.level=debug" - "-store-gateway.sharding-ring.replication-factor=3" + env: + - name: POD_NAME + valueFrom: + fieldRef: + fieldPath: metadata.name + - name: NAMESPACE_FQDN + value: "default.svc.cluster.local." ports: - name: http2 containerPort: 4040 @@ -3028,6 +3070,13 @@ spec: - "-runtime-config.file=/etc/pyroscope/overrides/overrides.yaml" - "-log.level=debug" - "-store-gateway.sharding-ring.replication-factor=3" + env: + - name: POD_NAME + valueFrom: + fieldRef: + fieldPath: metadata.name + - name: NAMESPACE_FQDN + value: "default.svc.cluster.local." ports: - name: http2 containerPort: 4040 @@ -3129,6 +3178,13 @@ spec: - "-runtime-config.file=/etc/pyroscope/overrides/overrides.yaml" - "-log.level=debug" - "-store-gateway.sharding-ring.replication-factor=3" + env: + - name: POD_NAME + valueFrom: + fieldRef: + fieldPath: metadata.name + - name: NAMESPACE_FQDN + value: "default.svc.cluster.local." ports: - name: http2 containerPort: 4040 @@ -3226,6 +3282,13 @@ spec: - "-runtime-config.file=/etc/pyroscope/overrides/overrides.yaml" - "-log.level=debug" - "-store-gateway.sharding-ring.replication-factor=3" + env: + - name: POD_NAME + valueFrom: + fieldRef: + fieldPath: metadata.name + - name: NAMESPACE_FQDN + value: "default.svc.cluster.local." ports: - name: http2 containerPort: 4040 diff --git a/operations/pyroscope/helm/pyroscope/rendered/single-binary.yaml b/operations/pyroscope/helm/pyroscope/rendered/single-binary.yaml index c85ac1b5d4..86ca35e594 100644 --- a/operations/pyroscope/helm/pyroscope/rendered/single-binary.yaml +++ b/operations/pyroscope/helm/pyroscope/rendered/single-binary.yaml @@ -1376,6 +1376,13 @@ spec: - "-config.file=/etc/pyroscope/config.yaml" - "-runtime-config.file=/etc/pyroscope/overrides/overrides.yaml" - "-log.level=debug" + env: + - name: POD_NAME + valueFrom: + fieldRef: + fieldPath: metadata.name + - name: NAMESPACE_FQDN + value: "default.svc.cluster.local." ports: - name: http2 containerPort: 4040 diff --git a/operations/pyroscope/helm/pyroscope/templates/deployments-statefulsets.yaml b/operations/pyroscope/helm/pyroscope/templates/deployments-statefulsets.yaml index b6162ab30a..026e9f2b53 100644 --- a/operations/pyroscope/helm/pyroscope/templates/deployments-statefulsets.yaml +++ b/operations/pyroscope/helm/pyroscope/templates/deployments-statefulsets.yaml @@ -70,8 +70,14 @@ spec: {{- range $key, $value := $extraArgs }} - "-{{ $key }}={{ $value }}" {{- end }} - {{- with $values.extraEnvVars }} env: + - name: POD_NAME + valueFrom: + fieldRef: + fieldPath: metadata.name + - name: NAMESPACE_FQDN + value: "{{ .Release.Namespace }}.svc{{ .Values.pyroscope.cluster_domain }}" + {{- with $values.extraEnvVars }} {{- range $key, $value := . }} - name: {{ $key }} {{- if kindIs "map" $value }} diff --git a/operations/pyroscope/helm/pyroscope/templates/services.yaml b/operations/pyroscope/helm/pyroscope/templates/services.yaml index d20d2a5aae..193c7419af 100644 --- a/operations/pyroscope/helm/pyroscope/templates/services.yaml +++ b/operations/pyroscope/helm/pyroscope/templates/services.yaml @@ -22,6 +22,9 @@ spec: targetPort: {{ $values.service.port_name }} protocol: TCP name: {{ $values.service.port_name }} + {{- with $values.service.extraPorts }} + {{- toYaml . | nindent 4 }} + {{- end }} selector: {{- include "pyroscope.selectorLabels" . | nindent 4 }} app.kubernetes.io/component: {{ $component | quote }} @@ -45,6 +48,12 @@ spec: targetPort: {{ $values.service.port_name }} protocol: TCP name: {{ $values.service.port_name }} + {{- with $values.service.extraPorts }} + {{- toYaml . | nindent 4 }} + {{- end }} + {{- if $values.service.publishNotReadyAddresses }} + publishNotReadyAddresses: true + {{- end }} selector: {{- include "pyroscope.selectorLabels" . | nindent 4 }} app.kubernetes.io/component: {{ $component | quote }} diff --git a/pkg/api/api.go b/pkg/api/api.go index cdaef6470b..778fe31abc 100644 --- a/pkg/api/api.go +++ b/pkg/api/api.go @@ -12,6 +12,8 @@ import ( "net/http" "strings" + segmentwriter "github.com/grafana/pyroscope/pkg/experiment/ingester" + "connectrpc.com/connect" "github.com/felixge/fgprof" "github.com/go-kit/log" @@ -22,12 +24,18 @@ import ( "github.com/grafana/dskit/server" grpcgw "github.com/grpc-ecosystem/grpc-gateway/v2/runtime" + compactorv1 "github.com/grafana/pyroscope/api/gen/proto/go/compactor/v1" + metastorev1 "github.com/grafana/pyroscope/api/gen/proto/go/metastore/v1" + querybackendv1 "github.com/grafana/pyroscope/api/gen/proto/go/querybackend/v1" + "github.com/grafana/pyroscope/pkg/experiment/metastore" + "github.com/grafana/pyroscope/pkg/experiment/querybackend" "github.com/grafana/pyroscope/public" "github.com/grafana/pyroscope/api/gen/proto/go/adhocprofiles/v1/adhocprofilesv1connect" "github.com/grafana/pyroscope/api/gen/proto/go/ingester/v1/ingesterv1connect" "github.com/grafana/pyroscope/api/gen/proto/go/push/v1/pushv1connect" "github.com/grafana/pyroscope/api/gen/proto/go/querier/v1/querierv1connect" + "github.com/grafana/pyroscope/api/gen/proto/go/segmentwriter/v1/segmentwriterv1connect" "github.com/grafana/pyroscope/api/gen/proto/go/settings/v1/settingsv1connect" statusv1 "github.com/grafana/pyroscope/api/gen/proto/go/status/v1" "github.com/grafana/pyroscope/api/gen/proto/go/storegateway/v1/storegatewayv1connect" @@ -268,6 +276,10 @@ func (a *API) RegisterIngester(svc *ingester.Ingester) { ingesterv1connect.RegisterIngesterServiceHandler(a.server.HTTP, svc, a.connectOptionsAuthRecovery()...) } +func (a *API) RegisterSegmentWriter(svc *segmentwriter.SegmentWriterService) { + segmentwriterv1connect.RegisterSegmentWriterServiceHandler(a.server.HTTP, svc, a.connectOptionsAuthRecovery()...) +} + func (a *API) RegisterStoreGateway(svc *storegateway.StoreGateway) { storegatewayv1connect.RegisterStoreGatewayServiceHandler(a.server.HTTP, svc, a.connectOptionsAuthRecovery()...) @@ -323,6 +335,15 @@ func (a *API) RegisterAdHocProfiles(ahp *adhocprofiles.AdHocProfiles) { adhocprofilesv1connect.RegisterAdHocProfileServiceHandler(a.server.HTTP, ahp, a.connectOptionsAuthRecovery()...) } +func (a *API) RegisterMetastore(svc *metastore.Metastore) { + metastorev1.RegisterMetastoreServiceServer(a.server.GRPC, svc) + compactorv1.RegisterCompactionPlannerServer(a.server.GRPC, svc) +} + +func (a *API) RegisterQueryBackend(svc *querybackend.QueryBackend) { + querybackendv1.RegisterQueryBackendServiceServer(a.server.GRPC, svc) +} + func (a *API) connectOptionsRecovery() []connect.HandlerOption { return append(connectapi.DefaultHandlerOptions(), a.recoveryMiddleware) } diff --git a/pkg/experiment/compactor/compaction_worker.go b/pkg/experiment/compactor/compaction_worker.go new file mode 100644 index 0000000000..f046615ac3 --- /dev/null +++ b/pkg/experiment/compactor/compaction_worker.go @@ -0,0 +1,297 @@ +package compactor + +import ( + "context" + "flag" + "fmt" + "os" + "path/filepath" + "runtime/debug" + "sync" + "time" + + "github.com/go-kit/log" + "github.com/go-kit/log/level" + "github.com/grafana/dskit/services" + "github.com/opentracing/opentracing-go" + "github.com/pkg/errors" + "github.com/prometheus/client_golang/prometheus" + + compactorv1 "github.com/grafana/pyroscope/api/gen/proto/go/compactor/v1" + metastorev1 "github.com/grafana/pyroscope/api/gen/proto/go/metastore/v1" + "github.com/grafana/pyroscope/pkg/experiment/metastore/client" + "github.com/grafana/pyroscope/pkg/experiment/querybackend/block" + "github.com/grafana/pyroscope/pkg/objstore" +) + +type Worker struct { + *services.BasicService + + config Config + logger log.Logger + metastoreClient *metastoreclient.Client + storage objstore.Bucket + metrics *compactionWorkerMetrics + + jobMutex sync.RWMutex + pendingJobs map[string]*compactorv1.CompactionJob + activeJobs map[string]*compactorv1.CompactionJob + completedJobs map[string]*compactorv1.CompactionJobStatus + + queue chan *compactorv1.CompactionJob +} + +type Config struct { + JobCapacity int `yaml:"job_capacity"` + SmallObjectSize int `yaml:"small_object_size_bytes"` + TempDir string `yaml:"temp_dir"` +} + +func (cfg *Config) RegisterFlags(f *flag.FlagSet) { + const prefix = "compaction-worker." + tempdir := filepath.Join(os.TempDir(), "pyroscope-compactor") + f.IntVar(&cfg.JobCapacity, prefix+"job-capacity", 3, "How many concurrent jobs will a worker run at most.") + f.IntVar(&cfg.SmallObjectSize, prefix+"small-object-size-bytes", 8<<20, "Size of the object that can be loaded in memory.") + f.StringVar(&cfg.TempDir, prefix+"temp-dir", tempdir, "Temporary directory for compaction jobs.") +} + +func New(config Config, logger log.Logger, metastoreClient *metastoreclient.Client, storage objstore.Bucket, reg prometheus.Registerer) (*Worker, error) { + w := &Worker{ + config: config, + logger: logger, + metastoreClient: metastoreClient, + storage: storage, + pendingJobs: make(map[string]*compactorv1.CompactionJob), + activeJobs: make(map[string]*compactorv1.CompactionJob), + completedJobs: make(map[string]*compactorv1.CompactionJobStatus), + metrics: newMetrics(reg), + queue: make(chan *compactorv1.CompactionJob, 2*config.JobCapacity), + } + w.BasicService = services.NewBasicService(w.starting, w.running, w.stopping) + return w, nil +} + +func (w *Worker) starting(ctx context.Context) (err error) { + return nil +} + +func (w *Worker) running(ctx context.Context) error { + ticker := time.NewTicker(5 * time.Second) + defer ticker.Stop() + go func() { + for { + select { + case <-ctx.Done(): + return + + case job := <-w.queue: + w.jobMutex.Lock() + delete(w.pendingJobs, job.Name) + w.activeJobs[job.Name] = job + w.jobMutex.Unlock() + + _ = level.Info(w.logger).Log("msg", "starting compaction job", "job", job.Name) + status := w.startJob(ctx, job) + _ = level.Info(w.logger).Log("msg", "compaction job finished", "job", job.Name) + + w.jobMutex.Lock() + delete(w.activeJobs, job.Name) + w.completedJobs[job.Name] = status + w.jobMutex.Unlock() + } + } + }() + + for { + select { + case <-ticker.C: + w.poll(ctx) + + case <-ctx.Done(): + return nil + } + } +} + +func (w *Worker) poll(ctx context.Context) { + w.jobMutex.Lock() + level.Debug(w.logger).Log( + "msg", "polling for compaction jobs and status updates", + "active_jobs", len(w.activeJobs), + "pending_jobs", len(w.pendingJobs), + "pending_updates", len(w.completedJobs)) + + pendingStatusUpdates := make([]*compactorv1.CompactionJobStatus, 0, len(w.completedJobs)) + for _, update := range w.completedJobs { + level.Debug(w.logger).Log("msg", "completed job update", "job", update.JobName, "status", update.Status) + pendingStatusUpdates = append(pendingStatusUpdates, update) + } + for _, activeJob := range w.activeJobs { + level.Debug(w.logger).Log("msg", "in progress job update", "job", activeJob.Name) + update := activeJob.Status.CloneVT() + update.Status = compactorv1.CompactionStatus_COMPACTION_STATUS_IN_PROGRESS + pendingStatusUpdates = append(pendingStatusUpdates, update) + } + for _, pendingJob := range w.pendingJobs { + level.Debug(w.logger).Log("msg", "pending job update", "job", pendingJob.Name) + update := pendingJob.Status.CloneVT() + update.Status = compactorv1.CompactionStatus_COMPACTION_STATUS_IN_PROGRESS + pendingStatusUpdates = append(pendingStatusUpdates, update) + } + + jobCapacity := w.config.JobCapacity - len(w.activeJobs) - len(w.pendingJobs) + if jobCapacity < 0 { + jobCapacity = 0 + } + w.jobMutex.Unlock() + + if len(pendingStatusUpdates) > 0 || jobCapacity > 0 { + jobsResponse, err := w.metastoreClient.PollCompactionJobs(ctx, &compactorv1.PollCompactionJobsRequest{ + JobStatusUpdates: pendingStatusUpdates, + JobCapacity: uint32(jobCapacity), + }) + + if err != nil { + level.Error(w.logger).Log("msg", "failed to poll compaction jobs", "err", err) + return + } + + level.Debug(w.logger).Log("msg", "poll response received", "compaction_jobs", len(jobsResponse.CompactionJobs)) + + pendingJobs := make([]*compactorv1.CompactionJob, 0, len(jobsResponse.CompactionJobs)) + for _, job := range jobsResponse.CompactionJobs { + pendingJobs = append(pendingJobs, job.CloneVT()) + } + + w.jobMutex.Lock() + for _, update := range pendingStatusUpdates { + delete(w.completedJobs, update.JobName) + } + for _, job := range pendingJobs { + w.pendingJobs[job.Name] = job + } + w.jobMutex.Unlock() + + for _, job := range pendingJobs { + select { + case w.queue <- job: + default: + level.Warn(w.logger).Log("msg", "dropping job", "job_name", job.Name) + w.jobMutex.Lock() + delete(w.pendingJobs, job.Name) + w.jobMutex.Unlock() + } + } + } +} + +func (w *Worker) stopping(err error) error { + // TODO aleks: handle shutdown + return nil +} + +func (w *Worker) startJob(ctx context.Context, job *compactorv1.CompactionJob) *compactorv1.CompactionJobStatus { + jobStartTime := time.Now() + labels := []string{job.TenantId, fmt.Sprint(job.Shard), fmt.Sprint(job.CompactionLevel)} + statusName := "unknown" + defer func() { + elapsed := time.Since(jobStartTime) + jobStatusLabel := append(labels, statusName) + w.metrics.jobDuration.WithLabelValues(jobStatusLabel...).Observe(elapsed.Seconds()) + w.metrics.jobsCompleted.WithLabelValues(jobStatusLabel...).Inc() + w.metrics.jobsInProgress.WithLabelValues(labels...).Dec() + }() + w.metrics.jobsInProgress.WithLabelValues(labels...).Inc() + + sp, ctx := opentracing.StartSpanFromContext(ctx, "StartCompactionJob", + opentracing.Tag{Key: "Job", Value: job.String()}, + opentracing.Tag{Key: "Tenant", Value: job.TenantId}, + opentracing.Tag{Key: "Shard", Value: job.Shard}, + opentracing.Tag{Key: "CompactionLevel", Value: job.CompactionLevel}, + opentracing.Tag{Key: "BlockCount", Value: len(job.Blocks)}, + ) + defer sp.Finish() + + _ = level.Info(w.logger).Log( + "msg", "compacting blocks for job", + "job", job.Name, + "blocks", len(job.Blocks)) + + tempdir := filepath.Join(w.config.TempDir, job.Name) + sourcedir := filepath.Join(tempdir, "source") + // TODO(kolesnikovae): Return the actual error once we + // can handle compaction failures in metastore. + compacted, err := pretendEverythingIsOK(func() ([]*metastorev1.BlockMeta, error) { + return block.Compact(ctx, job.Blocks, w.storage, + block.WithCompactionTempDir(tempdir), + block.WithCompactionObjectOptions( + block.WithObjectMaxSizeLoadInMemory(w.config.SmallObjectSize), + block.WithObjectDownload(sourcedir), + ), + ) + }) + + logger := log.With(w.logger, + "job_name", job.Name, + "job_shard", job.Shard, + "job_tenant", job.TenantId, + "job_compaction_level", job.CompactionLevel, + ) + + switch { + case err == nil: + _ = level.Info(logger).Log( + "msg", "successful compaction for job", + "input_blocks", len(job.Blocks), + "output_blocks", len(compacted)) + + for _, c := range compacted { + _ = level.Info(logger).Log( + "msg", "new compacted block", + "block_id", c.Id, + "block_tenant", c.TenantId, + "block_shard", c.Shard, + "block_size", c.Size, + "block_compaction_level", c.CompactionLevel, + "block_min_time", c.MinTime, + "block_max_time", c.MinTime, + "datasets", len(c.Datasets)) + } + + job.Status.Status = compactorv1.CompactionStatus_COMPACTION_STATUS_SUCCESS + job.Status.CompletedJob = &compactorv1.CompletedJob{Blocks: compacted} + statusName = "success" + + case errors.Is(err, context.Canceled): + _ = level.Warn(logger).Log("msg", "job cancelled", "job", job.Name) + job.Status.Status = compactorv1.CompactionStatus_COMPACTION_STATUS_UNSPECIFIED + statusName = "cancelled" + + default: + _ = level.Error(logger).Log("msg", "failed to compact blocks", "err", err, "job", job.Name) + job.Status.Status = compactorv1.CompactionStatus_COMPACTION_STATUS_FAILURE + statusName = "failure" + } + + return job.Status +} + +func pretendEverythingIsOK(fn func() ([]*metastorev1.BlockMeta, error)) (m []*metastorev1.BlockMeta, err error) { + defer func() { + if r := recover(); r != nil { + fmt.Println("ignoring compaction panic:", r) + fmt.Println(string(debug.Stack())) + m = nil + } + if err != nil { + if errors.Is(err, context.Canceled) { + // We can handle this. + return + } + fmt.Println("ignoring compaction error:", err) + m = nil + } + err = nil + }() + return fn() +} diff --git a/pkg/experiment/compactor/compaction_worker_metrics.go b/pkg/experiment/compactor/compaction_worker_metrics.go new file mode 100644 index 0000000000..122bfc3401 --- /dev/null +++ b/pkg/experiment/compactor/compaction_worker_metrics.go @@ -0,0 +1,36 @@ +package compactor + +import "github.com/prometheus/client_golang/prometheus" + +type compactionWorkerMetrics struct { + jobsCompleted *prometheus.CounterVec + jobsInProgress *prometheus.GaugeVec + jobDuration *prometheus.HistogramVec +} + +func newMetrics(r prometheus.Registerer) *compactionWorkerMetrics { + m := &compactionWorkerMetrics{} + + m.jobsCompleted = prometheus.NewCounterVec(prometheus.CounterOpts{ + Name: "pyroscope_compaction_jobs_completed_count", + Help: "Total number of compactions that were executed.", + }, []string{"tenant", "shard", "level", "outcome"}) + m.jobsInProgress = prometheus.NewGaugeVec(prometheus.GaugeOpts{ + Name: "pyroscope_compaction_jobs_current", + Help: "The number of active compaction jobs per level", + }, []string{"tenant", "shard", "level"}) + m.jobDuration = prometheus.NewHistogramVec(prometheus.HistogramOpts{ + Name: "pyroscope_compaction_jobs_duration_seconds", + Help: "Duration of compaction job runs", + Buckets: prometheus.ExponentialBuckets(1, 2, 14), + }, []string{"tenant", "shard", "level", "outcome"}) + + if r != nil { + r.MustRegister( + m.jobsCompleted, + m.jobsInProgress, + m.jobDuration, + ) + } + return m +} diff --git a/pkg/experiment/distributor/distributor_series.go b/pkg/experiment/distributor/distributor_series.go new file mode 100644 index 0000000000..935cdcc014 --- /dev/null +++ b/pkg/experiment/distributor/distributor_series.go @@ -0,0 +1,176 @@ +package distributor + +import ( + "fmt" + "hash/fnv" + + "github.com/grafana/dskit/ring" + + v1 "github.com/grafana/pyroscope/api/gen/proto/go/types/v1" + distributormodel "github.com/grafana/pyroscope/pkg/distributor/model" + phlaremodel "github.com/grafana/pyroscope/pkg/model" +) + +// FIXME(kolesnikovae): +// 1. Essentially, we do not need dskit ring. Instead, it would be better to access +// the memberlist/serf directly and build the distribution from there (generating +// tokens as we want). Or, alternatively, we could implement BasicLifecyclerDelegate +// interface. +// 2. Ensure we have access to all ingester instances, regardless of their state. +// The ring exposes only healthy instances, which is not what we want, and this +// will lead to vast shard relocations and will deteriorate data locality if +// instances leave and join the ring frequently. +// Currently, the heartbeat timeout is set to 1m by default, which should prevent +// us from severe problems, but it's still a problem. +// 3. Health checks are useless. It's better to sorry than to ask for permission: +// client should mark failed/slow instances and not rely on the ring to do so. +// 4. Implement stream statistics (see seriesPlacement interface). This could be done +// using Count-Min Sketch, or Count-Min-Log Sketch, or HyperLogLog(+(+)). +// 5. Push API should be streaming: use of batching is not efficient. + +type seriesDistributor struct { + tenantServices map[tenantServiceKey]*tenantServicePlacement + seriesPlacement seriesPlacement + distribution *distribution +} + +type seriesPlacement interface { + tenantServiceSize(tenantServiceKey, []shard) int + tenantServiceSeriesShard(*distributormodel.ProfileSeries, []shard) int +} + +type defaultSeriesPlacement struct{} + +func (defaultSeriesPlacement) tenantServiceSize(k tenantServiceKey, shards []shard) int { return 2 } + +func (defaultSeriesPlacement) tenantServiceSeriesShard(s *distributormodel.ProfileSeries, shards []shard) int { + k := fnv64(phlaremodel.LabelPairsString(s.Labels)) + return int(k % uint64(len(shards))) +} + +func newSeriesDistributor(r ring.ReadRing) (d *seriesDistributor, err error) { + d = &seriesDistributor{seriesPlacement: defaultSeriesPlacement{}} + if d.distribution, err = getDistribution(r, maxDistributionAge); err != nil { + return nil, fmt.Errorf("series distributor: %w", err) + } + d.tenantServices = make(map[tenantServiceKey]*tenantServicePlacement) + return d, nil +} + +func (d *seriesDistributor) buildRequests(tenantID string, series []*distributormodel.ProfileSeries) []*ingestionRequest { + var size int + for _, s := range series { + d.append(tenantID, s) + size++ + } + trackers := make([]*ingestionRequest, 0, size) + for _, p := range d.tenantServices { + for _, s := range p.series { + t := &ingestionRequest{profile: s} + t.shard, t.instances = p.pickShard(s) + trackers = append(trackers, t) + } + } + // Do not retain the series, but do keep shards: + // profileTracker references ring instances. + d.tenantServices = nil + return trackers +} + +func (d *seriesDistributor) append(tenant string, s *distributormodel.ProfileSeries) { + k := newTenantServiceKey(tenant, s.Labels) + p, ok := d.tenantServices[k] + if !ok { + p = d.newTenantServicePlacement(k) + d.tenantServices[k] = p + } + p.series = append(p.series, s) +} + +// Although a request may contain multiple series +// that belong to different services, the tenant is +// always the same (as of now). +type tenantServiceKey struct { + tenant string + service string +} + +func newTenantServiceKey(tenant string, seriesLabels []*v1.LabelPair) tenantServiceKey { + service := phlaremodel.Labels(seriesLabels).Get(phlaremodel.LabelNameServiceName) + return tenantServiceKey{ + tenant: tenant, + service: service, + } +} + +func (k tenantServiceKey) hash() uint64 { return fnv64(k.tenant, k.service) } + +const minShardsPerTenantService = 3 + +func (d *seriesDistributor) newTenantServicePlacement(key tenantServiceKey) *tenantServicePlacement { + size := d.seriesPlacement.tenantServiceSize(key, d.distribution.shards) + if size <= 0 { + size = len(d.distribution.shards) + } + return &tenantServicePlacement{ + seriesDistributor: d, + tenantServiceKey: key, + series: make([]*distributormodel.ProfileSeries, 0, 16), + // scope is a slice of shards that belong to the service. + // It might be larger than the actual number of shards allowed for use. + // In case of a delivery failure, at least minShardsPerTenantService + // options of the shard placement (instances) are available: the series + // will be sent to another shard location (ingester), but will still be + // associated with the shard. + scope: d.distribution.serviceShards(max(size, minShardsPerTenantService), key.hash()), + size: size, + } +} + +type tenantServicePlacement struct { + *seriesDistributor + tenantServiceKey + series []*distributormodel.ProfileSeries + scope []shard + size int +} + +// Pick the exact shard for the key from N options +// and find instances where the shard may be placed. +func (p *tenantServicePlacement) pickShard(s *distributormodel.ProfileSeries) (uint32, []*ring.InstanceDesc) { + // Limit the scope for selection to the actual number + // of shards, allowed for the tenant service. + i := p.seriesPlacement.tenantServiceSeriesShard(s, p.scope[:p.size]) + x := p.scope[i] + instances := make([]*ring.InstanceDesc, len(p.scope)) + for j, o := range p.scope { + instances[j] = &p.distribution.desc[o.instance] + } + instances[0], instances[i] = instances[i], instances[0] + return x.id, instances +} + +type ingestionRequest struct { + profile *distributormodel.ProfileSeries + // Note that the instances reference shared objects, and must not be modified. + instances []*ring.InstanceDesc + shard uint32 +} + +func (p *ingestionRequest) next() (instance *ring.InstanceDesc, ok bool) { + for len(p.instances) > 0 { + instance, p.instances = p.instances[0], p.instances[1:] + if instance.State == ring.ACTIVE { + return instance, true + } + } + return nil, false +} + +func fnv64(keys ...string) uint64 { + h := fnv.New64a() + for _, k := range keys { + _, _ = h.Write([]byte(k)) + } + return h.Sum64() +} diff --git a/pkg/experiment/distributor/distributor_sharding.go b/pkg/experiment/distributor/distributor_sharding.go new file mode 100644 index 0000000000..cdb54232c3 --- /dev/null +++ b/pkg/experiment/distributor/distributor_sharding.go @@ -0,0 +1,129 @@ +package distributor + +import ( + "fmt" + "math/rand" + "slices" + "strings" + "sync" + "time" + + "github.com/grafana/dskit/ring" +) + +var ( + distributionCache sync.RWMutex + cachedDistribution *distribution +) + +const maxDistributionAge = time.Second * 5 + +func getDistribution(r ring.ReadRing, maxAge time.Duration) (*distribution, error) { + distributionCache.RLock() + d := cachedDistribution + if d != nil && !d.isExpired(maxAge) { + distributionCache.RUnlock() + return d, nil + } + distributionCache.RUnlock() + distributionCache.Lock() + defer distributionCache.Unlock() + if d != nil && !d.isExpired(maxAge) { + return d, nil + } + var shards = 64 + var instances = 128 + if d != nil { + shards = len(d.shards) + instances = len(d.desc) + } + n := newDistribution(shards, instances) + if err := n.readRing(r); err != nil { + return nil, fmt.Errorf("failed to read ring: %w", err) + } + cachedDistribution = n + return n, nil +} + +type distribution struct { + timestamp time.Time + shards []shard + desc []ring.InstanceDesc +} + +type shard struct { + id uint32 // 0 shard ID is used as a sentinel (zero value is invalid). + instance uint32 // references the instance in shards.desc. +} + +func newDistribution(shards, instances int) *distribution { + return &distribution{ + shards: make([]shard, 0, shards), + desc: make([]ring.InstanceDesc, 0, instances), + timestamp: time.Now(), + } +} + +func (d *distribution) isExpired(maxAge time.Duration) bool { + return time.Now().Add(-maxAge).After(d.timestamp) +} + +func (d *distribution) readRing(r ring.ReadRing) error { + all, err := r.GetAllHealthy(ring.Write) + if err != nil { + return err + } + if len(all.Instances) == 0 { + return ring.ErrEmptyRing + } + d.desc = all.Instances + // Jump hashing needs order. + slices.SortFunc(d.desc, func(a, b ring.InstanceDesc) int { + return strings.Compare(a.Id, b.Id) + }) + i := uint32(0) + for j := range all.Instances { + for range all.Instances[j].Tokens { + i++ + d.shards = append(d.shards, shard{ + id: i, + instance: uint32(j), + }) + } + } + return nil +} + +// The constant determines which keys are generated for the +// jump hashing function. A generated value is added to the +// tenant service key hash to produce the next jump hashing +// key. The seed is fixed to ensure deterministic behaviour +// across instances. The value is a random generated with a +// crypto/rand.Read, and decoded as a little-endian uint64. +const serviceShardsRandSeed = 4349676827832284783 + +func (d *distribution) serviceShards(n int, service uint64) []shard { + // TODO(kolesnikovae): Precompute the jump hash keys (e.g., 1K should be enough). + rnd := rand.New(rand.NewSource(serviceShardsRandSeed)) + m := len(d.shards) + if m < n { + n = m + } + s := make([]shard, 0, n) + for i := 0; i < n; i++ { + j := jump(service&^rnd.Uint64(), m) + s = append(s, d.shards[j]) + } + return s +} + +// https://arxiv.org/pdf/1406.2294 +func jump(key uint64, buckets int) int { + var b, j = -1, 0 + for j < buckets { + b = j + key = key*2862933555777941757 + 1 + j = int(float64(b+1) * (float64(int64(1)<<31) / float64((key>>33)+1))) + } + return b +} diff --git a/pkg/experiment/distributor/singlereplica/singlereplica.go b/pkg/experiment/distributor/singlereplica/singlereplica.go new file mode 100644 index 0000000000..4ea150a30d --- /dev/null +++ b/pkg/experiment/distributor/singlereplica/singlereplica.go @@ -0,0 +1,25 @@ +package singlereplica + +import ( + "time" + + "github.com/grafana/dskit/ring" +) + +// The replication strategy that returns all the instances, regardless +// of their health and placement to allow the caller to decide which +// instances to use on its own. + +type replicationStrategy struct{} + +func (replicationStrategy) Filter( + instances []ring.InstanceDesc, + _ ring.Operation, + _ int, + _ time.Duration, + _ bool, +) ([]ring.InstanceDesc, int, error) { + return instances, 0, nil +} + +func NewReplicationStrategy() ring.ReplicationStrategy { return replicationStrategy{} } diff --git a/pkg/experiment/ingester/loki/index/buf.go b/pkg/experiment/ingester/loki/index/buf.go new file mode 100644 index 0000000000..f9d8f23391 --- /dev/null +++ b/pkg/experiment/ingester/loki/index/buf.go @@ -0,0 +1,114 @@ +package index + +import ( + "bytes" + "fmt" + "io" + "sync" + + "github.com/pkg/errors" +) + +type BufferWriter struct { + buf *bytes.Buffer + pos uint64 +} + +var pool = sync.Pool{ + New: func() interface{} { + return NewBufferWriter() + }, +} + +func GetBufferWriterFromPool() *BufferWriter { + res := pool.Get().(*BufferWriter) + res.Reset() + return res +} + +func PutBufferWriterToPool(fw *BufferWriter) { + fw.Reset() + pool.Put(fw) +} + +// NewBufferWriter returns a new BufferWriter. +// todo: pooling memory +func NewBufferWriter() *BufferWriter { + return &BufferWriter{ + buf: bytes.NewBuffer(make([]byte, 0, 0x2000)), + pos: 0, + } +} + +func (fw *BufferWriter) Pos() uint64 { + return fw.pos +} + +func (fw *BufferWriter) Write(bufs ...[]byte) error { + for _, buf := range bufs { + n, err := fw.buf.Write(buf) + if err != nil { + return err + } + fw.pos += uint64(n) + } + return nil +} + +func (fw *BufferWriter) Flush() error { + return nil +} + +func (fw *BufferWriter) WriteAt(buf []byte, pos uint64) error { + if pos > fw.pos { + return fmt.Errorf("position out of range") + } + if pos+uint64(len(buf)) > fw.pos { + return fmt.Errorf("write exceeds buffer size") + } + copy(fw.buf.Bytes()[pos:], buf) + return nil +} + +func (fw *BufferWriter) Read(buf []byte) (int, error) { + return fw.buf.Read(buf) +} + +func (fw *BufferWriter) ReadFrom(r io.Reader) (int64, error) { + n, err := fw.buf.ReadFrom(r) + if err != nil { + return n, err + } + fw.pos += uint64(n) + return n, err +} + +func (fw *BufferWriter) AddPadding(size int) error { + p := fw.pos % uint64(size) + if p == 0 { + return nil + } + p = uint64(size) - p + + if err := fw.Write(make([]byte, p)); err != nil { + return errors.Wrap(err, "add padding") + } + return nil +} + +func (fw *BufferWriter) Buffer() ([]byte, io.Closer, error) { + return fw.buf.Bytes(), io.NopCloser(nil), nil +} + +func (fw *BufferWriter) Close() error { + return nil +} + +func (fw *BufferWriter) Reset() { + fw.pos = 0 + fw.buf.Reset() +} + +func (fw *BufferWriter) Remove() error { + return nil +} diff --git a/pkg/experiment/ingester/loki/index/cmp.go b/pkg/experiment/ingester/loki/index/cmp.go new file mode 100644 index 0000000000..db87443a9d --- /dev/null +++ b/pkg/experiment/ingester/loki/index/cmp.go @@ -0,0 +1,76 @@ +package index + +import ( + "bytes" + "context" + "fmt" + "os" + + "github.com/prometheus/common/model" + "github.com/prometheus/prometheus/storage" + + phlaremodel "github.com/grafana/pyroscope/pkg/model" + "github.com/grafana/pyroscope/pkg/phlaredb/tsdb/index" +) + +type IIndexWriter interface { + AddSymbol(symbol string) error + AddSeries(ref storage.SeriesRef, lbs phlaremodel.Labels, fp model.Fingerprint, chunks ...index.ChunkMeta) error + Close() error +} + +func NewCompareIndexWriter(ctx context.Context, filepath string, fileWriter IIndexWriter) (*CompareIndexWriter, error) { + mem, err := NewWriter(ctx, SegmentsIndexWriterBufSize) + if err != nil { + return nil, fmt.Errorf("error creating memory index writer: %w", err) + } + return &CompareIndexWriter{ + filepath: filepath, + file: fileWriter, + mem: mem, + }, nil +} + +// This is a quick hack to test correctness of the new writer. +type CompareIndexWriter struct { + filepath string + file IIndexWriter + mem *Writer +} + +func (c *CompareIndexWriter) AddSymbol(symbol string) error { + ferr := c.file.AddSymbol(symbol) + merr := c.mem.AddSymbol(symbol) + if ferr != nil || merr != nil { + return fmt.Errorf("[CompareIndexWriter] error adding symbol: %v %v", ferr, merr) + } + return nil +} + +func (c *CompareIndexWriter) AddSeries(ref storage.SeriesRef, lbs phlaremodel.Labels, fp model.Fingerprint, chunks ...index.ChunkMeta) error { + ferr := c.file.AddSeries(ref, lbs, fp, chunks...) + merr := c.mem.AddSeries(ref, lbs, fp, chunks...) + if ferr != nil || merr != nil { + return fmt.Errorf("[CompareIndexWriter] error adding series: %v %v", ferr, merr) + } + return nil +} + +func (c *CompareIndexWriter) Close() error { + ferr := c.file.Close() + merr := c.mem.Close() + if ferr != nil || merr != nil { + return fmt.Errorf("[CompareIndexWriter] error closing index writer: %v %v", ferr, merr) + } + fileIndex, ferr := os.ReadFile(c.filepath) + if ferr != nil { + fmt.Printf("[CompareIndexWriter] error reading index file: %v\n", ferr) + return fmt.Errorf("[CompareIndexWriter] error reading index file: %v", ferr) + } + memIndex := c.mem.f.buf.Bytes() + if !bytes.Equal(fileIndex, memIndex) { + fmt.Printf("[CompareIndexWriter] index files do not match\n") + return fmt.Errorf("[CompareIndexWriter] index files do not match") + } + return nil +} diff --git a/pkg/experiment/ingester/loki/index/index.go b/pkg/experiment/ingester/loki/index/index.go new file mode 100644 index 0000000000..9833d07216 --- /dev/null +++ b/pkg/experiment/ingester/loki/index/index.go @@ -0,0 +1,1999 @@ +// Copyright 2017 The Prometheus 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. + +// A tsdb index writer, that does not use files and mmap +// To be for tiny segments in v2 POC branch +// Inspired by loki https://raw.githubusercontent.com/grafana/loki/main/pkg/storage/wal/index/index.go +// But actually copied from pyroscope and modified accordingly + +package index + +import ( + "bytes" + "context" + "encoding/binary" + "fmt" + "hash" + "hash/crc32" + "io" + "math" + "os" + "sort" + "unsafe" + + "github.com/grafana/pyroscope/pkg/phlaredb/tsdb/index" + + "github.com/pkg/errors" + "github.com/prometheus/common/model" + "github.com/prometheus/prometheus/model/labels" + "github.com/prometheus/prometheus/storage" + tsdb_enc "github.com/prometheus/prometheus/tsdb/encoding" + + typesv1 "github.com/grafana/pyroscope/api/gen/proto/go/types/v1" + phlaremodel "github.com/grafana/pyroscope/pkg/model" + "github.com/grafana/pyroscope/pkg/phlaredb/block" + "github.com/grafana/pyroscope/pkg/phlaredb/tsdb/encoding" +) + +const ( + // MagicIndex 4 bytes at the head of an index file. + MagicIndex = 0xBAAAD700 + // HeaderLen represents number of bytes reserved of index for header. + HeaderLen = 5 + + // FormatV1 represents 1 version of index. + FormatV1 = 1 + // FormatV2 represents 2 version of index. + FormatV2 = 2 + + IndexFilename = "index" + + // store every 1024 series' fingerprints in the fingerprint offsets table + fingerprintInterval = 1 << 10 + + SegmentsIndexWriterBufSize = 2 * 0x1000 // small for segments + BlocksIndexWriterBufSize = 1 << 22 // large for blocks +) + +type indexWriterStage uint8 + +const ( + idxStageNone indexWriterStage = iota + idxStageSymbols + idxStageSeries + idxStageDone +) + +func (s indexWriterStage) String() string { + switch s { + case idxStageNone: + return "none" + case idxStageSymbols: + return "symbols" + case idxStageSeries: + return "series" + case idxStageDone: + return "done" + } + return "" +} + +// The table gets initialized with sync.Once but may still cause a race +// with any other use of the crc32 package anywhere. Thus we initialize it +// before. +var castagnoliTable *crc32.Table + +func init() { + castagnoliTable = crc32.MakeTable(crc32.Castagnoli) +} + +// newCRC32 initializes a CRC32 hash with a preconfigured polynomial, so the +// polynomial may be easily changed in one location at a later time, if necessary. +func newCRC32() hash.Hash32 { + return crc32.New(castagnoliTable) +} + +type symbolCacheEntry struct { + index uint32 + lastValue string + lastValueIndex uint32 +} + +// Writer implements the IndexWriter interface for the standard +// serialization format. +type Writer struct { + ctx context.Context + + f *BufferWriter + + // Temporary file for postings. + fP *BufferWriter + // Temporary file for posting offsets table. + fPO *BufferWriter + cntPO uint64 + + toc TOC + stage indexWriterStage + postingsStart uint64 // Due to padding, can differ from TOC entry. + + // Reusable memory. + buf1 encoding.Encbuf + buf2 encoding.Encbuf + + numSymbols int + symbols *Symbols + symbolFile io.Closer + lastSymbol string + symbolCache map[string]symbolCacheEntry + + labelIndexes []labelIndexHashEntry // Label index offsets. + labelNames map[string]uint64 // Label names, and their usage. + // Keeps track of the fingerprint/offset for every n series + fingerprintOffsets index.FingerprintOffsets + + // Hold last series to validate that clients insert new series in order. + lastSeries phlaremodel.Labels + lastSeriesHash uint64 + lastRef storage.SeriesRef + + crc32 hash.Hash + + Version int +} + +// TOC represents index Table Of Content that states where each section of index starts. +type TOC struct { + Symbols uint64 + Series uint64 + LabelIndices uint64 + LabelIndicesTable uint64 + Postings uint64 + PostingsTable uint64 + FingerprintOffsets uint64 + Metadata Metadata +} + +// Metadata is TSDB-level metadata +type Metadata struct { + From, Through int64 + Checksum uint32 +} + +func (m *Metadata) EnsureBounds(from, through int64) { + if m.From == 0 || from < m.From { + m.From = from + } + + if m.Through == 0 || through > m.Through { + m.Through = through + } +} + +// NewTOCFromByteSlice return parsed TOC from given index byte slice. +func NewTOCFromByteSlice(bs ByteSlice) (*TOC, error) { + if bs.Len() < indexTOCLen { + return nil, tsdb_enc.ErrInvalidSize + } + b := bs.Range(bs.Len()-indexTOCLen, bs.Len()) + + expCRC := binary.BigEndian.Uint32(b[len(b)-4:]) + d := encoding.DecWrap(tsdb_enc.Decbuf{B: b[:len(b)-4]}) + if d.Crc32(castagnoliTable) != expCRC { + return nil, errors.Wrap(tsdb_enc.ErrInvalidChecksum, "read TOC") + } + + if err := d.Err(); err != nil { + return nil, err + } + + return &TOC{ + Symbols: d.Be64(), + Series: d.Be64(), + LabelIndices: d.Be64(), + LabelIndicesTable: d.Be64(), + Postings: d.Be64(), + PostingsTable: d.Be64(), + FingerprintOffsets: d.Be64(), + Metadata: Metadata{ + From: d.Be64int64(), + Through: d.Be64int64(), + Checksum: expCRC, + }, + }, nil +} + +// NewWriter returns a new Writer to the given filename. It serializes data in format version 2. +func NewWriter(ctx context.Context, bufferSize int) (*Writer, error) { + iw := &Writer{ + ctx: ctx, + f: GetBufferWriterFromPool(), + fP: GetBufferWriterFromPool(), + fPO: GetBufferWriterFromPool(), + stage: idxStageNone, + + // Reusable memory. + buf1: encoding.EncWrap(tsdb_enc.Encbuf{B: make([]byte, 0, bufferSize)}), + buf2: encoding.EncWrap(tsdb_enc.Encbuf{B: make([]byte, 0, bufferSize)}), + + symbolCache: make(map[string]symbolCacheEntry, 1<<8), + labelNames: make(map[string]uint64, 1<<8), + crc32: newCRC32(), + } + if err := iw.writeMeta(); err != nil { + return nil, err + } + return iw, nil +} + +func (w *Writer) write(bufs ...[]byte) error { + return w.f.Write(bufs...) +} + +func (w *Writer) writeAt(buf []byte, pos uint64) error { + return w.f.WriteAt(buf, pos) +} + +func (w *Writer) addPadding(size int) error { + return w.f.AddPadding(size) +} + +// ensureStage handles transitions between write stages and ensures that IndexWriter +// methods are called in an order valid for the implementation. +func (w *Writer) ensureStage(s indexWriterStage) error { + select { + case <-w.ctx.Done(): + return w.ctx.Err() + default: + } + + if w.stage == s { + return nil + } + if w.stage < s-1 { + // A stage has been skipped. + if err := w.ensureStage(s - 1); err != nil { + return err + } + } + if w.stage > s { + return errors.Errorf("invalid stage %q, currently at %q", s, w.stage) + } + + // Mark start of sections in table of contents. + switch s { + case idxStageSymbols: + w.toc.Symbols = w.f.pos + if err := w.startSymbols(); err != nil { + return err + } + case idxStageSeries: + if err := w.finishSymbols(); err != nil { + return err + } + w.toc.Series = w.f.pos + + case idxStageDone: + w.toc.LabelIndices = w.f.pos + // LabelIndices generation depends on the posting offset + // table produced at this stage. + if err := w.writePostingsToTmpFiles(); err != nil { + return err + } + if err := w.writeLabelIndices(); err != nil { + return err + } + + w.toc.Postings = w.f.pos + if err := w.writePostings(); err != nil { + return err + } + + w.toc.LabelIndicesTable = w.f.pos + if err := w.writeLabelIndexesOffsetTable(); err != nil { + return err + } + + w.toc.PostingsTable = w.f.pos + if err := w.writePostingsOffsetTable(); err != nil { + return err + } + + w.toc.FingerprintOffsets = w.f.pos + if err := w.writeFingerprintOffsetsTable(); err != nil { + return err + } + + if err := w.writeTOC(); err != nil { + return err + } + } + + w.stage = s + return nil +} + +func (w *Writer) writeMeta() error { + w.buf1.Reset() + w.buf1.PutBE32(MagicIndex) + w.buf1.PutByte(FormatV2) + + return w.write(w.buf1.Get()) +} + +// AddSeries adds the series one at a time along with its chunks. +// Requires a specific fingerprint to be passed in the case where the "desired" +// fingerprint differs from what labels.Hash() produces. For example, +// multitenant TSDBs embed a tenant label, but the actual series has no such +// label and so the derived fingerprint differs. +func (w *Writer) AddSeries(ref storage.SeriesRef, lset phlaremodel.Labels, fp model.Fingerprint, chunks ...index.ChunkMeta) error { + if err := w.ensureStage(idxStageSeries); err != nil { + return err + } + + // Put the supplied fingerprint instead of the calculated hash. + // This allows us to have a synthetic label (__loki_tenant__) in + // the pre-compacted TSDBs which map to fingerprints (and chunks) + // without this label in storage. + labelHash := uint64(fp) + + if ref < w.lastRef && len(w.lastSeries) != 0 { + return errors.Errorf("series with reference greater than %d already added", ref) + } + // We add padding to 16 bytes to increase the addressable space we get through 4 byte + // series references. + if err := w.addPadding(16); err != nil { + return errors.Errorf("failed to write padding bytes: %v", err) + } + + if w.f.pos%16 != 0 { + return errors.Errorf("series write not 16-byte aligned at %d", w.f.pos) + } + + w.buf2.Reset() + w.buf2.PutBE64(labelHash) + w.buf2.PutUvarint(len(lset)) + + for _, l := range lset { + var err error + cacheEntry, ok := w.symbolCache[l.Name] + nameIndex := cacheEntry.index + if !ok { + nameIndex, err = w.symbols.ReverseLookup(l.Name) + if err != nil { + return errors.Errorf("symbol entry for %q does not exist, %v", l.Name, err) + } + } + w.labelNames[l.Name]++ + w.buf2.PutUvarint32(nameIndex) + + valueIndex := cacheEntry.lastValueIndex + if !ok || cacheEntry.lastValue != l.Value { + valueIndex, err = w.symbols.ReverseLookup(l.Value) + if err != nil { + return errors.Errorf("symbol entry for %q does not exist, %v", l.Value, err) + } + w.symbolCache[l.Name] = symbolCacheEntry{ + index: nameIndex, + lastValue: l.Value, + lastValueIndex: valueIndex, + } + } + w.buf2.PutUvarint32(valueIndex) + } + + w.buf2.PutUvarint(len(chunks)) + + if len(chunks) > 0 { + c := chunks[0] + w.toc.Metadata.EnsureBounds(c.MinTime, c.MaxTime) + + w.buf2.PutVarint64(c.MinTime) + w.buf2.PutUvarint64(uint64(c.MaxTime - c.MinTime)) + w.buf2.PutUvarint32(c.KB) + w.buf2.PutUvarint32(c.SeriesIndex) + w.buf2.PutBE32(c.Checksum) + t0 := c.MaxTime + + for _, c := range chunks[1:] { + w.toc.Metadata.EnsureBounds(c.MinTime, c.MaxTime) + // Encode the diff against previous chunk as varint + // instead of uvarint because chunks may overlap + w.buf2.PutVarint64(c.MinTime - t0) + w.buf2.PutUvarint64(uint64(c.MaxTime - c.MinTime)) + w.buf2.PutUvarint32(c.KB) + w.buf2.PutUvarint32(c.SeriesIndex) + t0 = c.MaxTime + + w.buf2.PutBE32(c.Checksum) + } + } + + w.buf1.Reset() + w.buf1.PutUvarint(w.buf2.Len()) + + w.buf2.PutHash(w.crc32) + + w.lastSeries = append(w.lastSeries[:0], lset...) + w.lastSeriesHash = labelHash + w.lastRef = ref + + if ref%fingerprintInterval == 0 { + sRef := w.f.pos / 16 + w.fingerprintOffsets = append(w.fingerprintOffsets, [2]uint64{sRef, labelHash}) + } + + if err := w.write(w.buf1.Get(), w.buf2.Get()); err != nil { + return errors.Wrap(err, "write series data") + } + + return nil +} + +func (w *Writer) startSymbols() error { + // We are at w.toc.Symbols. + // Leave 4 bytes of space for the length, and another 4 for the number of symbols + // which will both be calculated later. + return w.write([]byte("alenblen")) +} + +func (w *Writer) AddSymbol(sym string) error { + if err := w.ensureStage(idxStageSymbols); err != nil { + return err + } + if w.numSymbols != 0 && sym <= w.lastSymbol { + return errors.Errorf("symbol %q out-of-order", sym) + } + w.lastSymbol = sym + w.numSymbols++ + w.buf1.Reset() + w.buf1.PutUvarintStr(sym) + return w.write(w.buf1.Get()) +} + +func (w *Writer) finishSymbols() error { + symbolTableSize := w.f.pos - w.toc.Symbols - 4 + // The symbol table's part is 4 bytes. So the total symbol table size must be less than or equal to 2^32-1 + if symbolTableSize > math.MaxUint32 { + return errors.Errorf("symbol table size exceeds 4 bytes: %d", symbolTableSize) + } + + // Write out the length and symbol count. + w.buf1.Reset() + w.buf1.PutBE32int(int(symbolTableSize)) + w.buf1.PutBE32int(w.numSymbols) + if err := w.writeAt(w.buf1.Get(), w.toc.Symbols); err != nil { + return err + } + + hashPos := w.f.pos + // Leave space for the hash. We can only calculate it + // now that the number of symbols is known, so mmap and do it from there. + if err := w.write([]byte("hash")); err != nil { + return err + } + if err := w.f.Flush(); err != nil { + return err + } + + //sf, err := fileutil.OpenMmapFile(w.f.name) + buf, sf, err := w.f.Buffer() + if err != nil { + return err + } + w.symbolFile = sf + hash := crc32.Checksum(buf[w.toc.Symbols+4:hashPos], castagnoliTable) + w.buf1.Reset() + w.buf1.PutBE32(hash) + if err := w.writeAt(w.buf1.Get(), hashPos); err != nil { + return err + } + + // Load in the symbol table efficiently for the rest of the index writing. + w.symbols, err = NewSymbols(RealByteSlice(buf), FormatV2, int(w.toc.Symbols)) + if err != nil { + return errors.Wrap(err, "read symbols") + } + return nil +} + +func (w *Writer) writeLabelIndices() error { + if err := w.fPO.Flush(); err != nil { + return err + } + + // Find all the label values in the tmp posting offset table. + //f, err := fileutil.OpenMmapFile(w.fPO.name) + buf, closer, err := w.fPO.Buffer() + if err != nil { + return err + } + defer closer.Close() + + d := encoding.DecWrap(tsdb_enc.NewDecbufRaw(RealByteSlice(buf), int(w.fPO.pos))) + cnt := w.cntPO + current := []byte{} + values := []uint32{} + for d.Err() == nil && cnt > 0 { + cnt-- + d.Uvarint() // Keycount. + name := d.UvarintBytes() // Label name. + value := yoloString(d.UvarintBytes()) // Label value. + d.Uvarint64() // Offset. + if len(name) == 0 { + continue // All index is ignored. + } + + if !bytes.Equal(name, current) && len(values) > 0 { + // We've reached a new label name. + if err := w.writeLabelIndex(string(current), values); err != nil { + return err + } + values = values[:0] + } + current = name + sid, err := w.symbols.ReverseLookup(value) + if err != nil { + return err + } + values = append(values, sid) + } + if d.Err() != nil { + return d.Err() + } + + // Handle the last label. + if len(values) > 0 { + if err := w.writeLabelIndex(string(current), values); err != nil { + return err + } + } + return nil +} + +func (w *Writer) writeLabelIndex(name string, values []uint32) error { + // Align beginning to 4 bytes for more efficient index list scans. + if err := w.addPadding(4); err != nil { + return err + } + + w.labelIndexes = append(w.labelIndexes, labelIndexHashEntry{ + keys: []string{name}, + offset: w.f.pos, + }) + + startPos := w.f.pos + // Leave 4 bytes of space for the length, which will be calculated later. + if err := w.write([]byte("alen")); err != nil { + return err + } + w.crc32.Reset() + + w.buf1.Reset() + w.buf1.PutBE32int(1) // Number of names. + w.buf1.PutBE32int(len(values)) + w.buf1.WriteToHash(w.crc32) + if err := w.write(w.buf1.Get()); err != nil { + return err + } + + for _, v := range values { + w.buf1.Reset() + w.buf1.PutBE32(v) + w.buf1.WriteToHash(w.crc32) + if err := w.write(w.buf1.Get()); err != nil { + return err + } + } + + // Write out the length. + w.buf1.Reset() + l := w.f.pos - startPos - 4 + if l > math.MaxUint32 { + return errors.Errorf("label index size exceeds 4 bytes: %d", l) + } + w.buf1.PutBE32int(int(l)) + if err := w.writeAt(w.buf1.Get(), startPos); err != nil { + return err + } + + w.buf1.Reset() + w.buf1.PutHashSum(w.crc32) + return w.write(w.buf1.Get()) +} + +// writeLabelIndexesOffsetTable writes the label indices offset table. +func (w *Writer) writeLabelIndexesOffsetTable() error { + startPos := w.f.pos + // Leave 4 bytes of space for the length, which will be calculated later. + if err := w.write([]byte("alen")); err != nil { + return err + } + w.crc32.Reset() + + w.buf1.Reset() + w.buf1.PutBE32int(len(w.labelIndexes)) + w.buf1.WriteToHash(w.crc32) + if err := w.write(w.buf1.Get()); err != nil { + return err + } + + for _, e := range w.labelIndexes { + w.buf1.Reset() + w.buf1.PutUvarint(len(e.keys)) + for _, k := range e.keys { + w.buf1.PutUvarintStr(k) + } + w.buf1.PutUvarint64(e.offset) + w.buf1.WriteToHash(w.crc32) + if err := w.write(w.buf1.Get()); err != nil { + return err + } + } + // Write out the length. + w.buf1.Reset() + l := w.f.pos - startPos - 4 + if l > math.MaxUint32 { + return errors.Errorf("label indexes offset table size exceeds 4 bytes: %d", l) + } + w.buf1.PutBE32int(int(l)) + if err := w.writeAt(w.buf1.Get(), startPos); err != nil { + return err + } + + w.buf1.Reset() + w.buf1.PutHashSum(w.crc32) + return w.write(w.buf1.Get()) +} + +// writePostingsOffsetTable writes the postings offset table. +func (w *Writer) writePostingsOffsetTable() error { + // Ensure everything is in the temporary file. + if err := w.fPO.Flush(); err != nil { + return err + } + + startPos := w.f.pos + // Leave 4 bytes of space for the length, which will be calculated later. + if err := w.write([]byte("alen")); err != nil { + return err + } + + // Copy over the tmp posting offset table, however we need to + // adjust the offsets. + adjustment := w.postingsStart + + w.buf1.Reset() + w.crc32.Reset() + w.buf1.PutBE32int(int(w.cntPO)) // Count. + w.buf1.WriteToHash(w.crc32) + if err := w.write(w.buf1.Get()); err != nil { + return err + } + + //f, err := fileutil.OpenMmapFile(w.fPO.name) + buf, closer, err := w.fPO.Buffer() + if err != nil { + return err + } + defer func() { + if closer != nil { + closer.Close() + } + }() + d := encoding.DecWrap(tsdb_enc.NewDecbufRaw(RealByteSlice(buf), int(w.fPO.pos))) + cnt := w.cntPO + for d.Err() == nil && cnt > 0 { + w.buf1.Reset() + w.buf1.PutUvarint(d.Uvarint()) // Keycount. + w.buf1.PutUvarintStr(yoloString(d.UvarintBytes())) // Label name. + w.buf1.PutUvarintStr(yoloString(d.UvarintBytes())) // Label value. + w.buf1.PutUvarint64(d.Uvarint64() + adjustment) // Offset. + w.buf1.WriteToHash(w.crc32) + if err := w.write(w.buf1.Get()); err != nil { + return err + } + cnt-- + } + if d.Err() != nil { + return d.Err() + } + + // Cleanup temporary file. + //if err := f.Close(); err != nil { + // return err + //} + //f = nil + if err := w.fPO.Close(); err != nil { + return err + } + if err := w.fPO.Remove(); err != nil { + return err + } + //w.fPO = nil + + // Write out the length. + w.buf1.Reset() + l := w.f.pos - startPos - 4 + if l > math.MaxUint32 { + return errors.Errorf("postings offset table size exceeds 4 bytes: %d", l) + } + w.buf1.PutBE32int(int(l)) + if err := w.writeAt(w.buf1.Get(), startPos); err != nil { + return err + } + + // Finally write the hash. + w.buf1.Reset() + w.buf1.PutHashSum(w.crc32) + return w.write(w.buf1.Get()) +} + +func (w *Writer) writeFingerprintOffsetsTable() error { + w.buf1.Reset() + w.buf2.Reset() + + w.buf1.PutBE32int(len(w.fingerprintOffsets)) // Count. + // build offsets + for _, x := range w.fingerprintOffsets { + w.buf1.PutBE64(x[0]) // series offset + w.buf1.PutBE64(x[1]) // hash + } + + // write length + ln := w.buf1.Len() + // TODO(owen-d): can remove the uint32 cast in the future + // Had to uint32 wrap these for arm32 builds, which we'll remove in the future. + if uint32(ln) > uint32(math.MaxUint32) { + return errors.Errorf("fingerprint offset size exceeds 4 bytes: %d", ln) + } + + w.buf2.PutBE32int(ln) + if err := w.write(w.buf2.Get()); err != nil { + return err + } + + // write offsets+checksum + w.buf1.PutHash(w.crc32) + if err := w.write(w.buf1.Get()); err != nil { + return errors.Wrap(err, "failure writing fingerprint offsets") + } + return nil +} + +const indexTOCLen = 8*9 + crc32.Size + +func (w *Writer) writeTOC() error { + w.buf1.Reset() + + w.buf1.PutBE64(w.toc.Symbols) + w.buf1.PutBE64(w.toc.Series) + w.buf1.PutBE64(w.toc.LabelIndices) + w.buf1.PutBE64(w.toc.LabelIndicesTable) + w.buf1.PutBE64(w.toc.Postings) + w.buf1.PutBE64(w.toc.PostingsTable) + w.buf1.PutBE64(w.toc.FingerprintOffsets) + + // metadata + w.buf1.PutBE64int64(w.toc.Metadata.From) + w.buf1.PutBE64int64(w.toc.Metadata.Through) + + w.buf1.PutHash(w.crc32) + + return w.write(w.buf1.Get()) +} + +func (w *Writer) writePostingsToTmpFiles() error { + names := make([]string, 0, len(w.labelNames)) + for n := range w.labelNames { + names = append(names, n) + } + sort.Strings(names) + + if err := w.f.Flush(); err != nil { + return err + } + //f, err := fileutil.OpenMmapFile(w.f.name) + buf, closer, err := w.f.Buffer() + if err != nil { + return err + } + defer closer.Close() + + // Write out the special all posting. + offsets := []uint32{} + d := encoding.DecWrap(tsdb_enc.NewDecbufRaw(RealByteSlice(buf), int(w.toc.LabelIndices))) + d.Skip(int(w.toc.Series)) + for d.Len() > 0 { + d.ConsumePadding() + startPos := w.toc.LabelIndices - uint64(d.Len()) + if startPos%16 != 0 { + return errors.Errorf("series not 16-byte aligned at %d", startPos) + } + offsets = append(offsets, uint32(startPos/16)) + // Skip to next series. + x := d.Uvarint() + d.Skip(x + crc32.Size) + if err := d.Err(); err != nil { + return err + } + } + if err := w.writePosting("", "", offsets); err != nil { + return err + } + maxPostings := uint64(len(offsets)) // No label name can have more postings than this. + + for len(names) > 0 { + batchNames := []string{} + var c uint64 + // Try to bunch up label names into one loop, but avoid + // using more memory than a single label name can. + for len(names) > 0 { + if w.labelNames[names[0]]+c > maxPostings { + if c > 0 { + break + } + return fmt.Errorf("corruption detected when writing postings to index: label %q has %d uses, but maxPostings is %d", names[0], w.labelNames[names[0]], maxPostings) + } + batchNames = append(batchNames, names[0]) + c += w.labelNames[names[0]] + names = names[1:] + } + + nameSymbols := map[uint32]string{} + for _, name := range batchNames { + sid, err := w.symbols.ReverseLookup(name) + if err != nil { + return err + } + nameSymbols[sid] = name + } + // Label name -> label value -> positions. + postings := map[uint32]map[uint32][]uint32{} + + d := encoding.DecWrap(tsdb_enc.NewDecbufRaw(RealByteSlice(buf), int(w.toc.LabelIndices))) + d.Skip(int(w.toc.Series)) + for d.Len() > 0 { + d.ConsumePadding() + startPos := w.toc.LabelIndices - uint64(d.Len()) + l := d.Uvarint() // Length of this series in bytes. + startLen := d.Len() + + _ = d.Be64() // skip fingerprint + // See if label names we want are in the series. + numLabels := d.Uvarint() + for i := 0; i < numLabels; i++ { + lno := uint32(d.Uvarint()) + lvo := uint32(d.Uvarint()) + + if _, ok := nameSymbols[lno]; ok { + if _, ok := postings[lno]; !ok { + postings[lno] = map[uint32][]uint32{} + } + postings[lno][lvo] = append(postings[lno][lvo], uint32(startPos/16)) + } + } + // Skip to next series. + d.Skip(l - (startLen - d.Len()) + crc32.Size) + if err := d.Err(); err != nil { + return err + } + } + + for _, name := range batchNames { + // Write out postings for this label name. + sid, err := w.symbols.ReverseLookup(name) + if err != nil { + return err + } + values := make([]uint32, 0, len(postings[sid])) + for v := range postings[sid] { + values = append(values, v) + } + // Symbol numbers are in order, so the strings will also be in order. + sort.Sort(uint32slice(values)) + for _, v := range values { + value, err := w.symbols.Lookup(v) + if err != nil { + return err + } + if err := w.writePosting(name, value, postings[sid][v]); err != nil { + return err + } + } + } + select { + case <-w.ctx.Done(): + return w.ctx.Err() + default: + } + } + return nil +} + +func (w *Writer) writePosting(name, value string, offs []uint32) error { + // Align beginning to 4 bytes for more efficient postings list scans. + if err := w.fP.AddPadding(4); err != nil { + return err + } + + // Write out postings offset table to temporary file as we go. + w.buf1.Reset() + w.buf1.PutUvarint(2) + w.buf1.PutUvarintStr(name) + w.buf1.PutUvarintStr(value) + w.buf1.PutUvarint64(w.fP.pos) // This is relative to the postings tmp file, not the final index file. + if err := w.fPO.Write(w.buf1.Get()); err != nil { + return err + } + w.cntPO++ + + w.buf1.Reset() + w.buf1.PutBE32int(len(offs)) + + for _, off := range offs { + if off > (1<<32)-1 { + return errors.Errorf("series offset %d exceeds 4 bytes", off) + } + w.buf1.PutBE32(off) + } + + w.buf2.Reset() + l := w.buf1.Len() + // We convert to uint to make code compile on 32-bit systems, as math.MaxUint32 doesn't fit into int there. + if uint(l) > math.MaxUint32 { + return errors.Errorf("posting size exceeds 4 bytes: %d", l) + } + w.buf2.PutBE32int(l) + w.buf1.PutHash(w.crc32) + return w.fP.Write(w.buf2.Get(), w.buf1.Get()) +} + +func (w *Writer) writePostings() error { + // There's padding in the tmp file, make sure it actually works. + if err := w.f.AddPadding(4); err != nil { + return err + } + w.postingsStart = w.f.pos + + // Copy temporary file into main index. + if err := w.fP.Flush(); err != nil { + return err + } + //if _, err := w.fP.f.Seek(0, 0); err != nil { + // return err + //} + // Don't need to calculate a checksum, so can copy directly. + //n, err := io.CopyBuffer(w.f.fbuf, w.fP.f, make([]byte, 1<<20)) + //buf := make([]byte, cap(w.buf1.B)) + //buf := w.buf1.B[:cap(w.buf1.B)] + //n, err := io.CopyBuffer(w.f.fbuf, w.fP.f, buf) + //if err != nil { + // return err + //} + n, err := w.f.ReadFrom(w.fP) + if err != nil { + return err + } + if uint64(n) != w.fP.pos { + return errors.Errorf("wrote %d bytes to posting temporary file, but only read back %d", w.fP.pos, n) + } + //w.f.pos += uint64(n) + + if err := w.fP.Close(); err != nil { + return err + } + if err := w.fP.Remove(); err != nil { + return err + } + //w.fP = nil + return nil +} + +type uint32slice []uint32 + +func (s uint32slice) Len() int { return len(s) } +func (s uint32slice) Swap(i, j int) { s[i], s[j] = s[j], s[i] } +func (s uint32slice) Less(i, j int) bool { return s[i] < s[j] } + +type labelIndexHashEntry struct { + keys []string + offset uint64 +} + +func (w *Writer) Close() error { + // Even if this fails, we need to close all the files. + ensureErr := w.ensureStage(idxStageDone) + + if w.symbolFile != nil { + if err := w.symbolFile.Close(); err != nil { + return err + } + } + if w.fP != nil { + if err := w.fP.Close(); err != nil { + return err + } + } + if w.fPO != nil { + if err := w.fPO.Close(); err != nil { + return err + } + } + if err := w.f.Close(); err != nil { + return err + } + // w.f is kept around a bit longer and returned to pool by users + PutBufferWriterToPool(w.fP) + PutBufferWriterToPool(w.fPO) + w.fP = nil + w.fPO = nil + + return ensureErr +} + +// StringIter iterates over a sorted list of strings. +type StringIter interface { + // Next advances the iterator and returns true if another value was found. + Next() bool + + // At returns the value at the current iterator position. + At() string + + // Err returns the last error of the iterator. + Err() error +} + +type Reader struct { + b ByteSlice + toc *TOC + + // Close that releases the underlying resources of the byte slice. + c io.Closer + + // Map of LabelName to a list of some LabelValues's position in the offset table. + // The first and last values for each name are always present. + postings map[string][]postingOffset + // For the v1 format, labelname -> labelvalue -> offset. + postingsV1 map[string]map[string]uint64 + + symbols *Symbols + nameSymbols map[uint32]string // Cache of the label name symbol lookups, + // as there are not many and they are half of all lookups. + + fingerprintOffsets index.FingerprintOffsets + + dec *Decoder + + version int +} + +type postingOffset struct { + value string + off int +} + +// ByteSlice abstracts a byte slice. +type ByteSlice interface { + Len() int + Range(start, end int) []byte +} + +type RealByteSlice []byte + +func (b RealByteSlice) Len() int { + return len(b) +} + +func (b RealByteSlice) Range(start, end int) []byte { + return b[start:end] +} + +func (b RealByteSlice) Sub(start, end int) ByteSlice { + return b[start:end] +} + +// NewReader returns a new index reader on the given byte slice. It automatically +// handles different format versions. +func NewReader(b ByteSlice) (*Reader, error) { + return newReader(b, io.NopCloser(nil)) +} + +type nopCloser struct{} + +func (nopCloser) Close() error { return nil } + +// NewFileReader returns a new index reader against the given index file. +func NewFileReader(path string) (*Reader, error) { + b, err := os.ReadFile(path) + if err != nil { + return nil, err + } + r, err := newReader(RealByteSlice(b), nopCloser{}) + if err != nil { + return r, err + } + + return r, nil +} + +func newReader(b ByteSlice, c io.Closer) (*Reader, error) { + r := &Reader{ + b: b, + c: c, + postings: map[string][]postingOffset{}, + } + + // Verify header. + if r.b.Len() < HeaderLen { + return nil, errors.Wrap(tsdb_enc.ErrInvalidSize, "index header") + } + if m := binary.BigEndian.Uint32(r.b.Range(0, 4)); m != MagicIndex { + return nil, errors.Errorf("invalid magic number %x", m) + } + r.version = int(r.b.Range(4, 5)[0]) + + if r.version != FormatV1 && r.version != FormatV2 { + return nil, errors.Errorf("unknown index file version %d", r.version) + } + + var err error + r.toc, err = NewTOCFromByteSlice(b) + if err != nil { + return nil, errors.Wrap(err, "read TOC") + } + + r.symbols, err = NewSymbols(r.b, r.version, int(r.toc.Symbols)) + if err != nil { + return nil, errors.Wrap(err, "read symbols") + } + + if r.version == FormatV1 { + // Earlier V1 formats don't have a sorted postings offset table, so + // load the whole offset table into memory. + r.postingsV1 = map[string]map[string]uint64{} + if err := ReadOffsetTable(r.b, r.toc.PostingsTable, func(key []string, off uint64, _ int) error { + if len(key) != 2 { + return errors.Errorf("unexpected key length for posting table %d", len(key)) + } + if _, ok := r.postingsV1[key[0]]; !ok { + r.postingsV1[key[0]] = map[string]uint64{} + r.postings[key[0]] = nil // Used to get a list of labelnames in places. + } + r.postingsV1[key[0]][key[1]] = off + return nil + }); err != nil { + return nil, errors.Wrap(err, "read postings table") + } + } else { + var lastKey []string + lastOff := 0 + valueCount := 0 + // For the postings offset table we keep every label name but only every nth + // label value (plus the first and last one), to save memory. + if err := ReadOffsetTable(r.b, r.toc.PostingsTable, func(key []string, _ uint64, off int) error { + if len(key) != 2 { + return errors.Errorf("unexpected key length for posting table %d", len(key)) + } + if _, ok := r.postings[key[0]]; !ok { + // Next label name. + r.postings[key[0]] = []postingOffset{} + if lastKey != nil { + // Always include last value for each label name. + r.postings[lastKey[0]] = append(r.postings[lastKey[0]], postingOffset{value: lastKey[1], off: lastOff}) + } + lastKey = nil + valueCount = 0 + } + if valueCount%symbolFactor == 0 { + r.postings[key[0]] = append(r.postings[key[0]], postingOffset{value: key[1], off: off}) + lastKey = nil + } else { + lastKey = key + lastOff = off + } + valueCount++ + return nil + }); err != nil { + return nil, errors.Wrap(err, "read postings table") + } + if lastKey != nil { + r.postings[lastKey[0]] = append(r.postings[lastKey[0]], postingOffset{value: lastKey[1], off: lastOff}) + } + // Trim any extra space in the slices. + for k, v := range r.postings { + l := make([]postingOffset, len(v)) + copy(l, v) + r.postings[k] = l + } + } + + r.nameSymbols = make(map[uint32]string, len(r.postings)) + for k := range r.postings { + if k == "" { + continue + } + off, err := r.symbols.ReverseLookup(k) + if err != nil { + return nil, errors.Wrap(err, "reverse symbol lookup") + } + r.nameSymbols[off] = k + } + + r.fingerprintOffsets, err = readFingerprintOffsetsTable(r.b, r.toc.FingerprintOffsets) + if err != nil { + return nil, errors.Wrap(err, "loading fingerprint offsets") + } + + r.dec = &Decoder{LookupSymbol: r.lookupSymbol} + + return r, nil +} + +// Version returns the file format version of the underlying index. +func (r *Reader) Version() int { + return r.version +} + +// FileInfo returns some general stats about the underlying file +func (r *Reader) FileInfo() block.File { + k, v := index.AllPostingsKey() + postings, err := r.Postings(k, nil, v) + if err != nil { + panic(err) + } + var numSeries uint64 + for postings.Next() { + numSeries++ + } + return block.File{ + RelPath: block.IndexFilename, + SizeBytes: uint64(r.Size()), + TSDB: &block.TSDBFile{ + NumSeries: numSeries, + }, + } +} + +// Range marks a byte range. +type Range struct { + Start, End int64 +} + +// PostingsRanges returns a new map of byte range in the underlying index file +// for all postings lists. +func (r *Reader) PostingsRanges() (map[labels.Label]Range, error) { + m := map[labels.Label]Range{} + if err := ReadOffsetTable(r.b, r.toc.PostingsTable, func(key []string, off uint64, _ int) error { + if len(key) != 2 { + return errors.Errorf("unexpected key length for posting table %d", len(key)) + } + d := encoding.DecWrap(tsdb_enc.NewDecbufAt(r.b, int(off), castagnoliTable)) + if d.Err() != nil { + return d.Err() + } + m[labels.Label{Name: key[0], Value: key[1]}] = Range{ + Start: int64(off) + 4, + End: int64(off) + 4 + int64(d.Len()), + } + return nil + }); err != nil { + return nil, errors.Wrap(err, "read postings table") + } + return m, nil +} + +type Symbols struct { + bs ByteSlice + version int + off int + + offsets []int + seen int +} + +const symbolFactor = 32 + +// NewSymbols returns a Symbols object for symbol lookups. +func NewSymbols(bs ByteSlice, version, off int) (*Symbols, error) { + s := &Symbols{ + bs: bs, + version: version, + off: off, + } + d := encoding.DecWrap(tsdb_enc.NewDecbufAt(bs, off, castagnoliTable)) + var ( + origLen = d.Len() + cnt = d.Be32int() + basePos = off + 4 + ) + s.offsets = make([]int, 0, 1+cnt/symbolFactor) + for d.Err() == nil && s.seen < cnt { + if s.seen%symbolFactor == 0 { + s.offsets = append(s.offsets, basePos+origLen-d.Len()) + } + d.UvarintBytes() // The symbol. + s.seen++ + } + if d.Err() != nil { + return nil, d.Err() + } + return s, nil +} + +func (s Symbols) Lookup(o uint32) (string, error) { + d := encoding.DecWrap(tsdb_enc.Decbuf{ + B: s.bs.Range(0, s.bs.Len()), + }) + + if s.version == FormatV2 { + if int(o) >= s.seen { + return "", errors.Errorf("unknown symbol offset %d", o) + } + d.Skip(s.offsets[int(o/symbolFactor)]) + // Walk until we find the one we want. + for i := o - (o / symbolFactor * symbolFactor); i > 0; i-- { + d.UvarintBytes() + } + } else { + d.Skip(int(o)) + } + sym := d.UvarintStr() + if d.Err() != nil { + return "", d.Err() + } + return sym, nil +} + +func (s Symbols) ReverseLookup(sym string) (uint32, error) { + if len(s.offsets) == 0 { + return 0, errors.Errorf("unknown symbol %q - no symbols", sym) + } + i := sort.Search(len(s.offsets), func(i int) bool { + // Any decoding errors here will be lost, however + // we already read through all of this at startup. + d := encoding.DecWrap(tsdb_enc.Decbuf{ + B: s.bs.Range(0, s.bs.Len()), + }) + d.Skip(s.offsets[i]) + return yoloString(d.UvarintBytes()) > sym + }) + d := encoding.DecWrap(tsdb_enc.Decbuf{ + B: s.bs.Range(0, s.bs.Len()), + }) + if i > 0 { + i-- + } + d.Skip(s.offsets[i]) + res := i * symbolFactor + var lastLen int + var lastSymbol string + for d.Err() == nil && res <= s.seen { + lastLen = d.Len() + lastSymbol = yoloString(d.UvarintBytes()) + if lastSymbol >= sym { + break + } + res++ + } + if d.Err() != nil { + return 0, d.Err() + } + if lastSymbol != sym { + return 0, errors.Errorf("unknown symbol %q", sym) + } + if s.version == FormatV2 { + return uint32(res), nil + } + return uint32(s.bs.Len() - lastLen), nil +} + +func (s Symbols) Size() int { + return len(s.offsets) * 8 +} + +func (s Symbols) Iter() StringIter { + d := encoding.DecWrap(tsdb_enc.NewDecbufAt(s.bs, s.off, castagnoliTable)) + cnt := d.Be32int() + return &symbolsIter{ + d: d, + cnt: cnt, + } +} + +// symbolsIter implements StringIter. +type symbolsIter struct { + d encoding.Decbuf + cnt int + cur string + err error +} + +func (s *symbolsIter) Next() bool { + if s.cnt == 0 || s.err != nil { + return false + } + s.cur = yoloString(s.d.UvarintBytes()) + s.cnt-- + if s.d.Err() != nil { + s.err = s.d.Err() + return false + } + return true +} + +func (s symbolsIter) At() string { return s.cur } +func (s symbolsIter) Err() error { return s.err } + +// ReadOffsetTable reads an offset table and at the given position calls f for each +// found entry. If f returns an error it stops decoding and returns the received error. +func ReadOffsetTable(bs ByteSlice, off uint64, f func([]string, uint64, int) error) error { + d := encoding.DecWrap(tsdb_enc.NewDecbufAt(bs, int(off), castagnoliTable)) + startLen := d.Len() + cnt := d.Be32() + + for d.Err() == nil && d.Len() > 0 && cnt > 0 { + offsetPos := startLen - d.Len() + keyCount := d.Uvarint() + // The Postings offset table takes only 2 keys per entry (name and value of label), + // and the LabelIndices offset table takes only 1 key per entry (a label name). + // Hence setting the size to max of both, i.e. 2. + keys := make([]string, 0, 2) + + for i := 0; i < keyCount; i++ { + keys = append(keys, d.UvarintStr()) + } + o := d.Uvarint64() + if d.Err() != nil { + break + } + if err := f(keys, o, offsetPos); err != nil { + return err + } + cnt-- + } + return d.Err() +} + +func readFingerprintOffsetsTable(bs ByteSlice, off uint64) (index.FingerprintOffsets, error) { + d := encoding.DecWrap(tsdb_enc.NewDecbufAt(bs, int(off), castagnoliTable)) + cnt := d.Be32() + res := make(index.FingerprintOffsets, 0, int(cnt)) + + for d.Err() == nil && d.Len() > 0 && cnt > 0 { + res = append(res, [2]uint64{d.Be64(), d.Be64()}) + cnt-- + } + + return res, d.Err() +} + +// Close the reader and its underlying resources. +func (r *Reader) Close() error { + return r.c.Close() +} + +func (r *Reader) lookupSymbol(o uint32) (string, error) { + if s, ok := r.nameSymbols[o]; ok { + return s, nil + } + return r.symbols.Lookup(o) +} + +func (r *Reader) Bounds() (int64, int64) { + return r.toc.Metadata.From, r.toc.Metadata.Through +} + +func (r *Reader) Checksum() uint32 { + return r.toc.Metadata.Checksum +} + +// Symbols returns an iterator over the symbols that exist within the index. +func (r *Reader) Symbols() StringIter { + return r.symbols.Iter() +} + +// SymbolTableSize returns the symbol table size in bytes. +func (r *Reader) SymbolTableSize() uint64 { + return uint64(r.symbols.Size()) +} + +// SortedLabelValues returns value tuples that exist for the given label name. +// It is not safe to use the return value beyond the lifetime of the byte slice +// passed into the Reader. +func (r *Reader) SortedLabelValues(name string, matchers ...*labels.Matcher) ([]string, error) { + values, err := r.LabelValues(name, matchers...) + if err == nil && r.version == FormatV1 { + sort.Strings(values) + } + return values, err +} + +// LabelValues returns value tuples that exist for the given label name. +// It is not safe to use the return value beyond the lifetime of the byte slice +// passed into the Reader. +// TODO(replay): Support filtering by matchers +func (r *Reader) LabelValues(name string, matchers ...*labels.Matcher) ([]string, error) { + if len(matchers) > 0 { + return nil, errors.Errorf("matchers parameter is not implemented: %+v", matchers) + } + + if r.version == FormatV1 { + e, ok := r.postingsV1[name] + if !ok { + return nil, nil + } + values := make([]string, 0, len(e)) + for k := range e { + values = append(values, k) + } + return values, nil + + } + e, ok := r.postings[name] + if !ok { + return nil, nil + } + if len(e) == 0 { + return nil, nil + } + values := make([]string, 0, len(e)*symbolFactor) + + d := encoding.DecWrap(tsdb_enc.NewDecbufAt(r.b, int(r.toc.PostingsTable), nil)) + d.Skip(e[0].off) + lastVal := e[len(e)-1].value + + skip := 0 + for d.Err() == nil { + if skip == 0 { + // These are always the same number of bytes, + // and it's faster to skip than parse. + skip = d.Len() + d.Uvarint() // Keycount. + d.UvarintBytes() // Label name. + skip -= d.Len() + } else { + d.Skip(skip) + } + s := yoloString(d.UvarintBytes()) // Label value. + values = append(values, s) + if s == lastVal { + break + } + d.Uvarint64() // Offset. + } + if d.Err() != nil { + return nil, errors.Wrap(d.Err(), "get postings offset entry") + } + return values, nil +} + +// LabelNamesFor returns all the label names for the series referred to by IDs. +// The names returned are sorted. +func (r *Reader) LabelNamesFor(ids ...storage.SeriesRef) ([]string, error) { + // Gather offsetsMap the name offsetsMap in the symbol table first + offsetsMap := make(map[uint32]struct{}) + for _, id := range ids { + offset := id + // In version 2 series IDs are no longer exact references but series are 16-byte padded + // and the ID is the multiple of 16 of the actual position. + if r.version == FormatV2 { + offset = id * 16 + } + + d := encoding.DecWrap(tsdb_enc.NewDecbufUvarintAt(r.b, int(offset), castagnoliTable)) + buf := d.Get() + if d.Err() != nil { + return nil, errors.Wrap(d.Err(), "get buffer for series") + } + + offsets, err := r.dec.LabelNamesOffsetsFor(buf) + if err != nil { + return nil, errors.Wrap(err, "get label name offsets") + } + for _, off := range offsets { + offsetsMap[off] = struct{}{} + } + } + + // Lookup the unique symbols. + names := make([]string, 0, len(offsetsMap)) + for off := range offsetsMap { + name, err := r.lookupSymbol(off) + if err != nil { + return nil, errors.Wrap(err, "lookup symbol in LabelNamesFor") + } + names = append(names, name) + } + + sort.Strings(names) + + return names, nil +} + +// LabelValueFor returns label value for the given label name in the series referred to by ID. +func (r *Reader) LabelValueFor(id storage.SeriesRef, label string) (string, error) { + offset := id + // In version 2 series IDs are no longer exact references but series are 16-byte padded + // and the ID is the multiple of 16 of the actual position. + if r.version == FormatV2 { + offset = id * 16 + } + d := encoding.DecWrap(tsdb_enc.NewDecbufUvarintAt(r.b, int(offset), castagnoliTable)) + buf := d.Get() + if d.Err() != nil { + return "", errors.Wrap(d.Err(), "label values for") + } + + value, err := r.dec.LabelValueFor(buf, label) + if err != nil { + return "", storage.ErrNotFound + } + + if value == "" { + return "", storage.ErrNotFound + } + + return value, nil +} + +// Series reads the series with the given ID and writes its labels and chunks into lbls and chks. +func (r *Reader) Series(id storage.SeriesRef, lbls *phlaremodel.Labels, chks *[]index.ChunkMeta) (uint64, error) { + offset := id + // In version 2 series IDs are no longer exact references but series are 16-byte padded + // and the ID is the multiple of 16 of the actual position. + if r.version == FormatV2 { + offset = id * 16 + } + d := encoding.DecWrap(tsdb_enc.NewDecbufUvarintAt(r.b, int(offset), castagnoliTable)) + if d.Err() != nil { + return 0, d.Err() + } + + fprint, err := r.dec.Series(d.Get(), lbls, chks, false) + if err != nil { + return 0, errors.Wrap(err, "read series") + } + return fprint, nil +} + +// SeriesBy is like Series but allows to group labels by name. This avoid looking up all label symbols for requested series. +func (r *Reader) SeriesBy(id storage.SeriesRef, lbls *phlaremodel.Labels, chks *[]index.ChunkMeta, by ...string) (uint64, error) { + offset := id + // In version 2 series IDs are no longer exact references but series are 16-byte padded + // and the ID is the multiple of 16 of the actual position. + if r.version == FormatV2 { + offset = id * 16 + } + d := encoding.DecWrap(tsdb_enc.NewDecbufUvarintAt(r.b, int(offset), castagnoliTable)) + if d.Err() != nil { + return 0, d.Err() + } + + fprint, err := r.dec.Series(d.Get(), lbls, chks, true, by...) + if err != nil { + return 0, errors.Wrap(err, "read series") + } + return fprint, nil +} + +func (r *Reader) Postings(name string, shard *index.ShardAnnotation, values ...string) (index.Postings, error) { + if r.version == FormatV1 { + e, ok := r.postingsV1[name] + if !ok { + return index.EmptyPostings(), nil + } + res := make([]index.Postings, 0, len(values)) + for _, v := range values { + postingsOff, ok := e[v] + if !ok { + continue + } + // Read from the postings table. + d := encoding.DecWrap(tsdb_enc.NewDecbufAt(r.b, int(postingsOff), castagnoliTable)) + _, p, err := r.dec.Postings(d.Get()) + if err != nil { + return nil, errors.Wrap(err, "decode postings") + } + res = append(res, p) + } + return index.Merge(res...), nil + } + + e, ok := r.postings[name] + if !ok { + return index.EmptyPostings(), nil + } + + if len(values) == 0 { + return index.EmptyPostings(), nil + } + + res := make([]index.Postings, 0, len(values)) + skip := 0 + valueIndex := 0 + for valueIndex < len(values) && values[valueIndex] < e[0].value { + // Discard values before the start. + valueIndex++ + } + for valueIndex < len(values) { + value := values[valueIndex] + + i := sort.Search(len(e), func(i int) bool { return e[i].value >= value }) + if i == len(e) { + // We're past the end. + break + } + if i > 0 && e[i].value != value { + // Need to look from previous entry. + i-- + } + // Don't Crc32 the entire postings offset table, this is very slow + // so hope any issues were caught at startup. + d := encoding.DecWrap(tsdb_enc.NewDecbufAt(r.b, int(r.toc.PostingsTable), nil)) + d.Skip(e[i].off) + + // Iterate on the offset table. + var postingsOff uint64 // The offset into the postings table. + for d.Err() == nil { + if skip == 0 { + // These are always the same number of bytes, + // and it's faster to skip than parse. + skip = d.Len() + d.Uvarint() // Keycount. + d.UvarintBytes() // Label name. + skip -= d.Len() + } else { + d.Skip(skip) + } + v := d.UvarintBytes() // Label value. + postingsOff = d.Uvarint64() // Offset. + for string(v) >= value { + if string(v) == value { + // Read from the postings table. + d2 := encoding.DecWrap(tsdb_enc.NewDecbufAt(r.b, int(postingsOff), castagnoliTable)) + _, p, err := r.dec.Postings(d2.Get()) + if err != nil { + return nil, errors.Wrap(err, "decode postings") + } + res = append(res, p) + } + valueIndex++ + if valueIndex == len(values) { + break + } + value = values[valueIndex] + } + if i+1 == len(e) || value >= e[i+1].value || valueIndex == len(values) { + // Need to go to a later postings offset entry, if there is one. + break + } + } + if d.Err() != nil { + return nil, errors.Wrap(d.Err(), "get postings offset entry") + } + } + + merged := index.Merge(res...) + if shard != nil { + return index.NewShardedPostings(merged, *shard, r.fingerprintOffsets), nil + } + + return merged, nil +} + +// Size returns the size of an index file. +func (r *Reader) Size() int64 { + return int64(r.b.Len()) +} + +// LabelNames returns all the unique label names present in the index. +// TODO(twilkie) implement support for matchers +func (r *Reader) LabelNames(matchers ...*labels.Matcher) ([]string, error) { + if len(matchers) > 0 { + return nil, errors.Errorf("matchers parameter is not implemented: %+v", matchers) + } + + labelNames := make([]string, 0, len(r.postings)) + allPostingsKeyName, _ := index.AllPostingsKey() + for name := range r.postings { + //if name == index.allPostingsKey.Name { + if name == allPostingsKeyName { + // This is not from any metric. + continue + } + labelNames = append(labelNames, name) + } + sort.Strings(labelNames) + return labelNames, nil +} + +// Decoder provides decoding methods for the v1 and v2 index file format. +// +// It currently does not contain decoding methods for all entry types but can be extended +// by them if there's demand. +type Decoder struct { + LookupSymbol func(uint32) (string, error) +} + +// Postings returns a postings list for b and its number of elements. +func (dec *Decoder) Postings(b []byte) (int, index.Postings, error) { + d := encoding.DecWrap(tsdb_enc.Decbuf{B: b}) + n := d.Be32int() + l := d.Get() + if d.Err() != nil { + return 0, nil, d.Err() + } + if len(l) != 4*n { + return 0, nil, fmt.Errorf("unexpected postings length, should be %d bytes for %d postings, got %d bytes", 4*n, n, len(l)) + } + return n, index.NewBigEndianPostings(l), nil +} + +// LabelNamesOffsetsFor decodes the offsets of the name symbols for a given series. +// They are returned in the same order they're stored, which should be sorted lexicographically. +func (dec *Decoder) LabelNamesOffsetsFor(b []byte) ([]uint32, error) { + d := encoding.DecWrap(tsdb_enc.Decbuf{B: b}) + _ = d.Be64() // skip fingerprint + k := d.Uvarint() + + offsets := make([]uint32, k) + for i := 0; i < k; i++ { + offsets[i] = uint32(d.Uvarint()) + _ = d.Uvarint() // skip the label value + + if d.Err() != nil { + return nil, errors.Wrap(d.Err(), "read series label offsets") + } + } + + return offsets, d.Err() +} + +// LabelValueFor decodes a label for a given series. +func (dec *Decoder) LabelValueFor(b []byte, label string) (string, error) { + d := encoding.DecWrap(tsdb_enc.Decbuf{B: b}) + _ = d.Be64() // skip fingerprint + k := d.Uvarint() + + for i := 0; i < k; i++ { + lno := uint32(d.Uvarint()) + lvo := uint32(d.Uvarint()) + + if d.Err() != nil { + return "", errors.Wrap(d.Err(), "read series label offsets") + } + + ln, err := dec.LookupSymbol(lno) + if err != nil { + return "", errors.Wrap(err, "lookup label name") + } + + if ln == label { + lv, err := dec.LookupSymbol(lvo) + if err != nil { + return "", errors.Wrap(err, "lookup label value") + } + + return lv, nil + } + } + + return "", d.Err() +} + +// Series decodes a series entry from the given byte slice into lset and chks. +func (dec *Decoder) Series(b []byte, lbls *phlaremodel.Labels, chks *[]index.ChunkMeta, group bool, by ...string) (uint64, error) { + if lbls != nil { + *lbls = (*lbls)[:0] + } + *chks = (*chks)[:0] + + d := encoding.DecWrap(tsdb_enc.Decbuf{B: b}) + + fprint := d.Be64() + k := d.Uvarint() + + for i := 0; i < k; i++ { + lno := uint32(d.Uvarint()) + lvo := uint32(d.Uvarint()) + + if d.Err() != nil { + return 0, errors.Wrap(d.Err(), "read series label offsets") + } + if lbls == nil { + continue + } + if group && len(by) == 0 { + // If we're grouping by all labels, we don't need to decode them. + continue + } + ln, err := dec.LookupSymbol(lno) + if err != nil { + return 0, errors.Wrap(err, "lookup label name") + } + if group { + var found bool + for _, b := range by { + if b == ln { + found = true + break + } + } + if !found { + continue + } + } + lv, err := dec.LookupSymbol(lvo) + if err != nil { + return 0, errors.Wrap(err, "lookup label value") + } + + *lbls = append(*lbls, &typesv1.LabelPair{Name: ln, Value: lv}) + } + + // Read the chunks meta data. + k = d.Uvarint() + + if k == 0 { + return 0, d.Err() + } + + t0 := d.Varint64() + maxt := int64(d.Uvarint64()) + t0 + kb := uint32(d.Uvarint()) + entries := uint32(d.Uvarint64()) + checksum := d.Be32() + + *chks = append(*chks, index.ChunkMeta{ + Checksum: checksum, + MinTime: t0, + MaxTime: maxt, + KB: kb, + SeriesIndex: entries, + }) + t0 = maxt + + for i := 1; i < k; i++ { + // Decode the diff against previous chunk as varint + // instead of uvarint because chunks may overlap + mint := d.Varint64() + t0 + maxt := int64(d.Uvarint64()) + mint + kb := uint32(d.Uvarint()) + entries := uint32(d.Uvarint64()) + checksum := d.Be32() + t0 = maxt + + if d.Err() != nil { + return 0, errors.Wrapf(d.Err(), "read meta for chunk %d", i) + } + + *chks = append(*chks, index.ChunkMeta{ + Checksum: checksum, + MinTime: mint, + MaxTime: maxt, + KB: kb, + SeriesIndex: entries, + }) + } + return fprint, d.Err() +} + +func yoloString(b []byte) string { + return *((*string)(unsafe.Pointer(&b))) +} + +func (w *Writer) ReleaseIndexBuffer() *BufferWriter { + res := w.f + w.f = nil + return res +} diff --git a/pkg/experiment/ingester/loki/index/index_test.go b/pkg/experiment/ingester/loki/index/index_test.go new file mode 100644 index 0000000000..be250e1064 --- /dev/null +++ b/pkg/experiment/ingester/loki/index/index_test.go @@ -0,0 +1,597 @@ +// Copyright 2017 The Prometheus 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. + +package index + +import ( + "context" + "fmt" + "hash/crc32" + "math/rand" + "os" + "path/filepath" + "sort" + "testing" + + "github.com/grafana/pyroscope/pkg/phlaredb/tsdb/index" + + "github.com/pkg/errors" + "github.com/stretchr/testify/require" + "go.uber.org/goleak" + + "github.com/prometheus/common/model" + "github.com/prometheus/prometheus/model/labels" + "github.com/prometheus/prometheus/storage" + "github.com/prometheus/prometheus/tsdb/encoding" + "github.com/prometheus/prometheus/util/testutil" + + typesv1 "github.com/grafana/pyroscope/api/gen/proto/go/types/v1" + "github.com/grafana/pyroscope/pkg/iter" + phlaremodel "github.com/grafana/pyroscope/pkg/model" +) + +func TestMain(m *testing.M) { + goleak.VerifyTestMain(m, + goleak.IgnoreTopFunction("github.com/golang/glog.(*fileSink).flushDaemon"), + goleak.IgnoreTopFunction("github.com/dgraph-io/ristretto.(*defaultPolicy).processItems"), + goleak.IgnoreTopFunction("github.com/dgraph-io/ristretto.(*Cache).processItems"), + ) +} + +type series struct { + l phlaremodel.Labels + chunks []index.ChunkMeta +} + +type mockIndex struct { + series map[storage.SeriesRef]series + // we're forced to use a anonymous struct here because we can't use typesv1.LabelPair as it's not comparable. + postings map[struct{ Name, Value string }][]storage.SeriesRef + symbols map[string]struct{} +} + +func newMockIndex() mockIndex { + allPostingsKeyName, allPostingsKeyValue := index.AllPostingsKey() + ix := mockIndex{ + series: make(map[storage.SeriesRef]series), + postings: make(map[struct{ Name, Value string }][]storage.SeriesRef), + symbols: make(map[string]struct{}), + } + ix.postings[struct { + Name string + Value string + }{allPostingsKeyName, allPostingsKeyValue}] = []storage.SeriesRef{} + return ix +} + +func (m mockIndex) Symbols() (map[string]struct{}, error) { + return m.symbols, nil +} + +func (m mockIndex) AddSeries(ref storage.SeriesRef, l phlaremodel.Labels, chunks ...index.ChunkMeta) error { + allPostingsKeyName, allPostingsKeyValue := index.AllPostingsKey() + + if _, ok := m.series[ref]; ok { + return errors.Errorf("series with reference %d already added", ref) + } + for _, lbl := range l { + m.symbols[lbl.Name] = struct{}{} + m.symbols[lbl.Value] = struct{}{} + if _, ok := m.postings[struct { + Name string + Value string + }{lbl.Name, lbl.Value}]; !ok { + m.postings[struct { + Name string + Value string + }{lbl.Name, lbl.Value}] = []storage.SeriesRef{} + } + m.postings[struct { + Name string + Value string + }{lbl.Name, lbl.Value}] = append(m.postings[struct { + Name string + Value string + }{lbl.Name, lbl.Value}], ref) + } + m.postings[struct { + Name string + Value string + }{allPostingsKeyName, allPostingsKeyValue}] = append(m.postings[struct { + Name string + Value string + }{allPostingsKeyName, allPostingsKeyValue}], ref) + + s := series{l: l} + // Actual chunk data is not stored in the index. + s.chunks = append(s.chunks, chunks...) + m.series[ref] = s + + return nil +} + +func (m mockIndex) Close() error { + return nil +} + +func (m mockIndex) LabelValues(name string) ([]string, error) { + values := []string{} + for l := range m.postings { + if l.Name == name { + values = append(values, l.Value) + } + } + return values, nil +} + +func (m mockIndex) Postings(name string, values ...string) (index.Postings, error) { + p := []index.Postings{} + for _, value := range values { + p = append(p, iter.NewSliceSeekIterator(m.postings[struct { + Name string + Value string + }{Name: name, Value: value}])) + } + return index.Merge(p...), nil +} + +func (m mockIndex) Series(ref storage.SeriesRef, lset *phlaremodel.Labels, chks *[]index.ChunkMeta) error { + s, ok := m.series[ref] + if !ok { + return errors.New("not found") + } + *lset = append((*lset)[:0], s.l...) + *chks = append((*chks)[:0], s.chunks...) + + return nil +} + +func TestIndexRW_Create_Open(t *testing.T) { + + // An empty index must still result in a readable file. + iw, err := NewWriter(context.Background(), BlocksIndexWriterBufSize) + require.NoError(t, err) + require.NoError(t, iw.Close()) + + bytes := iw.ReleaseIndexBuffer().buf.Bytes() + ir, err := NewReader(RealByteSlice(bytes)) + require.NoError(t, err) + require.NoError(t, ir.Close()) + + // Modify magic header must cause open to fail. + //f, err := os.OpenFile(fn, os.O_WRONLY, 0o666) + //require.NoError(t, err) + //err = iw.f.WriteAt([]byte{0, 0}, 0) + bytes[0] = 0 + require.NoError(t, err) + //f.Close() + + //_, err = NewFileReader(dir) + //require.Error(t, err) +} + +func TestIndexRW_Postings(t *testing.T) { + + iw, err := NewWriter(context.Background(), BlocksIndexWriterBufSize) + require.NoError(t, err) + + series := []phlaremodel.Labels{ + phlaremodel.LabelsFromStrings("a", "1", "b", "1"), + phlaremodel.LabelsFromStrings("a", "1", "b", "2"), + phlaremodel.LabelsFromStrings("a", "1", "b", "3"), + phlaremodel.LabelsFromStrings("a", "1", "b", "4"), + } + + require.NoError(t, iw.AddSymbol("1")) + require.NoError(t, iw.AddSymbol("2")) + require.NoError(t, iw.AddSymbol("3")) + require.NoError(t, iw.AddSymbol("4")) + require.NoError(t, iw.AddSymbol("a")) + require.NoError(t, iw.AddSymbol("b")) + + // Postings lists are only written if a series with the respective + // reference was added before. + require.NoError(t, iw.AddSeries(1, series[0], model.Fingerprint(series[0].Hash()))) + require.NoError(t, iw.AddSeries(2, series[1], model.Fingerprint(series[1].Hash()))) + require.NoError(t, iw.AddSeries(3, series[2], model.Fingerprint(series[2].Hash()))) + require.NoError(t, iw.AddSeries(4, series[3], model.Fingerprint(series[3].Hash()))) + + require.NoError(t, iw.Close()) + + ir, err := NewReader(RealByteSlice(iw.ReleaseIndexBuffer().buf.Bytes())) + require.NoError(t, err) + + p, err := ir.Postings("a", nil, "1") + require.NoError(t, err) + + var l phlaremodel.Labels + var c []index.ChunkMeta + + for i := 0; p.Next(); i++ { + _, err := ir.Series(p.At(), &l, &c) + + require.NoError(t, err) + require.Equal(t, 0, len(c)) + require.Equal(t, series[i], l) + } + require.NoError(t, p.Err()) + + // The label indices are no longer used, so test them by hand here. + labelIndices := map[string][]string{} + require.NoError(t, ReadOffsetTable(ir.b, ir.toc.LabelIndicesTable, func(key []string, off uint64, _ int) error { + if len(key) != 1 { + return errors.Errorf("unexpected key length for label indices table %d", len(key)) + } + + d := encoding.NewDecbufAt(ir.b, int(off), castagnoliTable) + vals := []string{} + nc := d.Be32int() + if nc != 1 { + return errors.Errorf("unexpected number of label indices table names %d", nc) + } + for i := d.Be32(); i > 0; i-- { + v, err := ir.lookupSymbol(d.Be32()) + if err != nil { + return err + } + vals = append(vals, v) + } + labelIndices[key[0]] = vals + return d.Err() + })) + require.Equal(t, map[string][]string{ + "a": {"1"}, + "b": {"1", "2", "3", "4"}, + }, labelIndices) + + require.NoError(t, ir.Close()) +} + +func TestPostingsMany(t *testing.T) { + + iw, err := NewWriter(context.Background(), BlocksIndexWriterBufSize) + require.NoError(t, err) + + // Create a label in the index which has 999 values. + symbols := map[string]struct{}{} + series := []phlaremodel.Labels{} + for i := 1; i < 1000; i++ { + v := fmt.Sprintf("%03d", i) + series = append(series, phlaremodel.LabelsFromStrings("i", v, "foo", "bar")) + symbols[v] = struct{}{} + } + symbols["i"] = struct{}{} + symbols["foo"] = struct{}{} + symbols["bar"] = struct{}{} + syms := []string{} + for s := range symbols { + syms = append(syms, s) + } + sort.Strings(syms) + for _, s := range syms { + require.NoError(t, iw.AddSymbol(s)) + } + + sort.Slice(series, func(i, j int) bool { + return series[i].Hash() < series[j].Hash() + }) + + for i, s := range series { + require.NoError(t, iw.AddSeries(storage.SeriesRef(i), s, model.Fingerprint(s.Hash()))) + } + require.NoError(t, iw.Close()) + + ir, err := NewReader(RealByteSlice(iw.ReleaseIndexBuffer().buf.Bytes())) + require.NoError(t, err) + defer func() { require.NoError(t, ir.Close()) }() + + cases := []struct { + in []string + }{ + // Simple cases, everything is present. + {in: []string{"002"}}, + {in: []string{"031", "032", "033"}}, + {in: []string{"032", "033"}}, + {in: []string{"127", "128"}}, + {in: []string{"127", "128", "129"}}, + {in: []string{"127", "129"}}, + {in: []string{"128", "129"}}, + {in: []string{"998", "999"}}, + {in: []string{"999"}}, + // Before actual values. + {in: []string{"000"}}, + {in: []string{"000", "001"}}, + {in: []string{"000", "002"}}, + // After actual values. + {in: []string{"999a"}}, + {in: []string{"999", "999a"}}, + {in: []string{"998", "999", "999a"}}, + // In the middle of actual values. + {in: []string{"126a", "127", "128"}}, + {in: []string{"127", "127a", "128"}}, + {in: []string{"127", "127a", "128", "128a", "129"}}, + {in: []string{"127", "128a", "129"}}, + {in: []string{"128", "128a", "129"}}, + {in: []string{"128", "129", "129a"}}, + {in: []string{"126a", "126b", "127", "127a", "127b", "128", "128a", "128b", "129", "129a", "129b"}}, + } + + for _, c := range cases { + it, err := ir.Postings("i", nil, c.in...) + require.NoError(t, err) + + got := []string{} + var lbls phlaremodel.Labels + var metas []index.ChunkMeta + for it.Next() { + _, err := ir.Series(it.At(), &lbls, &metas) + require.NoError(t, err) + got = append(got, lbls.Get("i")) + } + require.NoError(t, it.Err()) + exp := []string{} + for _, e := range c.in { + if _, ok := symbols[e]; ok && e != "l" { + exp = append(exp, e) + } + } + + // sort expected values by label hash instead of lexicographically by labelset + sort.Slice(exp, func(i, j int) bool { + return labels.FromStrings("i", exp[i], "foo", "bar").Hash() < labels.FromStrings("i", exp[j], "foo", "bar").Hash() + }) + + require.Equal(t, exp, got, fmt.Sprintf("input: %v", c.in)) + } +} + +func TestPersistence_index_e2e(t *testing.T) { + lbls, err := labels.ReadLabels("../../../../phlaredb/tsdb/testdata/20kseries.json", 20000) + require.NoError(t, err) + + flbls := make([]phlaremodel.Labels, len(lbls)) + for i, ls := range lbls { + flbls[i] = make(phlaremodel.Labels, 0, len(ls)) + for _, l := range ls { + flbls[i] = append(flbls[i], &typesv1.LabelPair{Name: l.Name, Value: l.Value}) + } + } + + // Sort labels as the index writer expects series in sorted order by fingerprint. + sort.Slice(flbls, func(i, j int) bool { + return flbls[i].Hash() < flbls[j].Hash() + }) + + symbols := map[string]struct{}{} + for _, lset := range lbls { + for _, l := range lset { + symbols[l.Name] = struct{}{} + symbols[l.Value] = struct{}{} + } + } + + var input index.IndexWriterSeriesSlice + + // Generate ChunkMetas for every label set. + for i, lset := range flbls { + var metas []index.ChunkMeta + + for j := 0; j <= (i % 20); j++ { + metas = append(metas, index.ChunkMeta{ + MinTime: int64(j * 10000), + MaxTime: int64((j + 1) * 10000), + Checksum: rand.Uint32(), + }) + } + input = append(input, &index.IndexWriterSeries{ + Labels: lset, + Chunks: metas, + }) + } + + iw, err := NewWriter(context.Background(), BlocksIndexWriterBufSize) + require.NoError(t, err) + + syms := []string{} + for s := range symbols { + syms = append(syms, s) + } + sort.Strings(syms) + for _, s := range syms { + require.NoError(t, iw.AddSymbol(s)) + } + + // Population procedure as done by compaction. + var ( + postings = index.NewMemPostings() + values = map[string]map[string]struct{}{} + ) + + mi := newMockIndex() + + for i, s := range input { + err = iw.AddSeries(storage.SeriesRef(i), s.Labels, model.Fingerprint(s.Labels.Hash()), s.Chunks...) + require.NoError(t, err) + require.NoError(t, mi.AddSeries(storage.SeriesRef(i), s.Labels, s.Chunks...)) + + for _, l := range s.Labels { + valset, ok := values[l.Name] + if !ok { + valset = map[string]struct{}{} + values[l.Name] = valset + } + valset[l.Value] = struct{}{} + } + postings.Add(storage.SeriesRef(i), s.Labels) + } + + err = iw.Close() + require.NoError(t, err) + + ir, err := NewReader(RealByteSlice(iw.ReleaseIndexBuffer().buf.Bytes())) + require.NoError(t, err) + + for p := range mi.postings { + gotp, err := ir.Postings(p.Name, nil, p.Value) + require.NoError(t, err) + + expp, err := mi.Postings(p.Name, p.Value) + require.NoError(t, err) + + var lset, explset phlaremodel.Labels + var chks, expchks []index.ChunkMeta + + for gotp.Next() { + require.True(t, expp.Next()) + + ref := gotp.At() + + _, err := ir.Series(ref, &lset, &chks) + require.NoError(t, err) + + err = mi.Series(expp.At(), &explset, &expchks) + require.NoError(t, err) + require.Equal(t, explset, lset) + require.Equal(t, expchks, chks) + } + require.False(t, expp.Next(), "Expected no more postings for %q=%q", p.Name, p.Value) + require.NoError(t, gotp.Err()) + } + + labelPairs := map[string][]string{} + for l := range mi.postings { + labelPairs[l.Name] = append(labelPairs[l.Name], l.Value) + } + for k, v := range labelPairs { + sort.Strings(v) + + res, err := ir.SortedLabelValues(k) + require.NoError(t, err) + + require.Equal(t, len(v), len(res)) + for i := 0; i < len(v); i++ { + require.Equal(t, v[i], res[i]) + } + } + + gotSymbols := []string{} + it := ir.Symbols() + for it.Next() { + gotSymbols = append(gotSymbols, it.At()) + } + require.NoError(t, it.Err()) + expSymbols := []string{} + for s := range mi.symbols { + expSymbols = append(expSymbols, s) + } + sort.Strings(expSymbols) + require.Equal(t, expSymbols, gotSymbols) + + require.NoError(t, ir.Close()) +} + +func TestDecbufUvarintWithInvalidBuffer(t *testing.T) { + b := RealByteSlice([]byte{0x81, 0x81, 0x81, 0x81, 0x81, 0x81}) + + db := encoding.NewDecbufUvarintAt(b, 0, castagnoliTable) + require.Error(t, db.Err()) +} + +func TestReaderWithInvalidBuffer(t *testing.T) { + b := RealByteSlice([]byte{0x81, 0x81, 0x81, 0x81, 0x81, 0x81}) + + _, err := NewReader(b) + require.Error(t, err) +} + +// TestNewFileReaderErrorNoOpenFiles ensures that in case of an error no file remains open. +func TestNewFileReaderErrorNoOpenFiles(t *testing.T) { + dir := testutil.NewTemporaryDirectory("block", t) + + idxName := filepath.Join(dir.Path(), "index") + err := os.WriteFile(idxName, []byte("corrupted contents"), 0o666) + require.NoError(t, err) + + _, err = NewFileReader(idxName) + require.Error(t, err) + + // dir.Close will fail on Win if idxName fd is not closed on error path. + dir.Close() +} + +func TestSymbols(t *testing.T) { + buf := encoding.Encbuf{} + + // Add prefix to the buffer to simulate symbols as part of larger buffer. + buf.PutUvarintStr("something") + + symbolsStart := buf.Len() + buf.PutBE32int(204) // Length of symbols table. + buf.PutBE32int(100) // Number of symbols. + for i := 0; i < 100; i++ { + // i represents index in unicode characters table. + buf.PutUvarintStr(string(rune(i))) // Symbol. + } + checksum := crc32.Checksum(buf.Get()[symbolsStart+4:], castagnoliTable) + buf.PutBE32(checksum) // Check sum at the end. + + s, err := NewSymbols(RealByteSlice(buf.Get()), FormatV2, symbolsStart) + require.NoError(t, err) + + // We store only 4 offsets to symbols. + require.Equal(t, 32, s.Size()) + + for i := 99; i >= 0; i-- { + s, err := s.Lookup(uint32(i)) + require.NoError(t, err) + require.Equal(t, string(rune(i)), s) + } + _, err = s.Lookup(100) + require.Error(t, err) + + for i := 99; i >= 0; i-- { + r, err := s.ReverseLookup(string(rune(i))) + require.NoError(t, err) + require.Equal(t, uint32(i), r) + } + _, err = s.ReverseLookup(string(rune(100))) + require.Error(t, err) + + iter := s.Iter() + i := 0 + for iter.Next() { + require.Equal(t, string(rune(i)), iter.At()) + i++ + } + require.NoError(t, iter.Err()) +} + +func TestDecoder_Postings_WrongInput(t *testing.T) { + _, _, err := (&Decoder{}).Postings([]byte("the cake is a lie")) + require.Error(t, err) +} + +func TestWriter_ShouldReturnErrorOnSeriesWithDuplicatedLabelNames(t *testing.T) { + w, err := NewWriter(context.Background(), BlocksIndexWriterBufSize) + require.NoError(t, err) + + require.NoError(t, w.AddSymbol("__name__")) + require.NoError(t, w.AddSymbol("metric_1")) + require.NoError(t, w.AddSymbol("metric_2")) + + require.NoError(t, w.AddSeries(0, phlaremodel.LabelsFromStrings("__name__", "metric_1", "__name__", "metric_2"), 0)) + + err = w.Close() + require.Error(t, err) + require.ErrorContains(t, err, "corruption detected when writing postings to index") +} diff --git a/pkg/experiment/ingester/segment.go b/pkg/experiment/ingester/segment.go new file mode 100644 index 0000000000..877b06773b --- /dev/null +++ b/pkg/experiment/ingester/segment.go @@ -0,0 +1,562 @@ +package ingester + +import ( + "bytes" + "context" + "crypto/rand" + "encoding/json" + "fmt" + "os" + "path" + "path/filepath" + "runtime/pprof" + "slices" + "strings" + "sync" + "time" + + "github.com/go-kit/log" + "github.com/go-kit/log/level" + "github.com/google/uuid" + "github.com/oklog/ulid" + "github.com/prometheus/common/model" + "github.com/thanos-io/objstore" + + profilev1 "github.com/grafana/pyroscope/api/gen/proto/go/google/v1" + metastorev1 "github.com/grafana/pyroscope/api/gen/proto/go/metastore/v1" + typesv1 "github.com/grafana/pyroscope/api/gen/proto/go/types/v1" + phlaremodel "github.com/grafana/pyroscope/pkg/model" + "github.com/grafana/pyroscope/pkg/phlaredb" + "github.com/grafana/pyroscope/pkg/phlaredb/block" + "github.com/grafana/pyroscope/pkg/phlaredb/symdb" + "github.com/grafana/pyroscope/pkg/tenant" + "github.com/grafana/pyroscope/pkg/util/math" +) + +const pathSegments = "segments" +const pathDLQ = "dlq" +const pathAnon = tenant.DefaultTenantID +const pathBlock = "block.bin" + +type shardKey uint32 + +type segmentsWriter struct { + segmentDuration time.Duration + phlarectx context.Context + l log.Logger + shards map[shardKey]*shard + shardsLock sync.RWMutex + cfg phlaredb.Config + bucket objstore.Bucket + metastoreClient metastorev1.MetastoreServiceClient + //wg sync.WaitGroup + cancel context.CancelFunc + metrics *segmentMetrics + cancelCtx context.Context +} + +type shard struct { + sw *segmentsWriter + current *segment + currentLock sync.RWMutex + wg sync.WaitGroup + l log.Logger + concatBuf []byte +} + +func (sh *shard) ingest(fn func(head segmentIngest) error) (segmentWaitFlushed, error) { + sh.currentLock.RLock() + s := sh.current + s.inFlightProfiles.Add(1) + sh.currentLock.RUnlock() + defer s.inFlightProfiles.Done() + return s, fn(s) +} + +func (sh *shard) loop(ctx context.Context) { + ticker := time.NewTicker(sh.sw.segmentDuration) + defer ticker.Stop() + for { + select { + case <-ticker.C: + sh.flushSegment(context.Background()) + case <-ctx.Done(): + sh.flushSegment(context.Background()) + return + } + } +} + +func (sh *shard) flushSegment(ctx context.Context) { + sh.currentLock.Lock() + s := sh.current + sh.current = sh.sw.newSegment(sh, s.shard, sh.l) + sh.currentLock.Unlock() + + go func() { // not blocking next ticks in case metastore/s3 latency is high + t1 := time.Now() + s.inFlightProfiles.Wait() + s.debuginfo.waitInflight = time.Since(t1) + + err := s.flush(ctx) + if err != nil { + _ = level.Error(sh.sw.l).Log("msg", "failed to flush segment", "err", err) + } + if s.debuginfo.movedHeads > 0 { + _ = level.Debug(s.l).Log("msg", + "writing segment block done", + "heads-count", len(s.heads), + "heads-moved-count", s.debuginfo.movedHeads, + "inflight-duration", s.debuginfo.waitInflight, + "flush-heads-duration", s.debuginfo.flushHeadsDuration, + "flush-block-duration", s.debuginfo.flushBlockDuration, + "store-meta-duration", s.debuginfo.storeMetaDuration, + "total-duration", time.Since(t1)) + } + }() +} + +func newSegmentWriter(phlarectx context.Context, l log.Logger, metrics *segmentMetrics, cfg phlaredb.Config, bucket objstore.Bucket, segmentDuration time.Duration, metastoreClient metastorev1.MetastoreServiceClient) *segmentsWriter { + ctx, cancelFunc := context.WithCancel(context.Background()) + sw := &segmentsWriter{ + metrics: metrics, + segmentDuration: segmentDuration, + phlarectx: phlarectx, + l: l, + bucket: bucket, + cfg: cfg, + shards: make(map[shardKey]*shard), + metastoreClient: metastoreClient, + cancel: cancelFunc, + cancelCtx: ctx, + } + + return sw +} + +func (sw *segmentsWriter) ingest(shard shardKey, fn func(head segmentIngest) error) (await segmentWaitFlushed, err error) { + sw.shardsLock.RLock() + s, ok := sw.shards[shard] + sw.shardsLock.RUnlock() + if ok { + return s.ingest(fn) + } + + sw.shardsLock.Lock() + s, ok = sw.shards[shard] + if ok { + sw.shardsLock.Unlock() + return s.ingest(fn) + } + + s = sw.newShard(shard) + sw.shards[shard] = s + sw.shardsLock.Unlock() + return s.ingest(fn) +} + +func (sw *segmentsWriter) Stop() error { + sw.l.Log("msg", "stopping segments writer") + sw.cancel() + sw.shardsLock.Lock() + defer sw.shardsLock.Unlock() + for _, s := range sw.shards { + s.wg.Wait() + } + sw.l.Log("msg", "segments writer stopped") + + return nil +} + +func (sw *segmentsWriter) newShard(sk shardKey) *shard { + sl := log.With(sw.l, "shard", fmt.Sprintf("%d", sk)) + sh := &shard{ + sw: sw, + l: sl, + concatBuf: make([]byte, 4*0x1000), + } + sh.current = sw.newSegment(sh, sk, sl) + sh.wg.Add(1) + go func() { + defer sh.wg.Done() + sh.loop(sw.cancelCtx) + }() + return sh +} +func (sw *segmentsWriter) newSegment(sh *shard, sk shardKey, sl log.Logger) *segment { + id := ulid.MustNew(ulid.Timestamp(time.Now()), rand.Reader) + sshard := fmt.Sprintf("%d", sk) + dataPath := path.Join(sw.cfg.DataPath, pathSegments, sshard, pathAnon, id.String()) + s := &segment{ + l: log.With(sl, "segment-id", id.String()), + ulid: id, + heads: make(map[serviceKey]serviceHead), + sw: sw, + sh: sh, + shard: sk, + sshard: sshard, + dataPath: dataPath, + doneChan: make(chan error, 1), + } + return s +} + +func (s *segment) flush(ctx context.Context) (err error) { + t1 := time.Now() + var heads []serviceHead + + defer func() { + s.cleanup() + if err != nil { + s.doneChan <- err + } + close(s.doneChan) + s.sw.metrics.flushSegmentDuration.WithLabelValues(s.sshard).Observe(time.Since(t1).Seconds()) + + }() + pprof.Do(ctx, pprof.Labels("segment_op", "flush_heads"), func(ctx context.Context) { + heads = s.flushHeads(ctx) + }) + s.debuginfo.movedHeads = len(heads) + if len(heads) == 0 { + return nil + } + + blockPath, blockMeta, err := s.flushBlock(heads) + if err != nil { + return fmt.Errorf("failed to flush block %s: %w", s.ulid.String(), err) + } + err = s.sw.uploadBlock(blockPath, s) + if err != nil { + return fmt.Errorf("failed to upload block %s: %w", s.ulid.String(), err) + } + err = s.sw.storeMeta(ctx, blockMeta, s) + if err != nil { + level.Error(s.l).Log("msg", "failed to store meta", "err", err) + errDLQErr := s.sw.storeMetaDLQ(ctx, blockMeta, s) + if errDLQErr == nil { + return nil + } + return fmt.Errorf("failed to store meta %s: %w %w", s.ulid.String(), err, errDLQErr) + } + return nil +} + +func (s *segment) flushBlock(heads []serviceHead) (string, *metastorev1.BlockMeta, error) { + t1 := time.Now() + meta := &metastorev1.BlockMeta{ + FormatVersion: 1, + Id: s.ulid.String(), + MinTime: 0, + MaxTime: 0, + Shard: uint32(s.shard), + CompactionLevel: 0, + TenantId: "", + Datasets: make([]*metastorev1.Dataset, 0, len(heads)), + Size: 0, + } + + blockPath := path.Join(s.dataPath, pathBlock) + blockFile, err := os.OpenFile(blockPath, os.O_CREATE|os.O_WRONLY|os.O_EXCL, 0644) + if err != nil { + return "", nil, err + } + defer blockFile.Close() + + w := withWriterOffset(blockFile) + + for i, e := range heads { + svc, err := concatSegmentHead(s.sh, e, w) + if err != nil { + _ = level.Error(s.l).Log("msg", "failed to concat segment head", "err", err) + continue + } + if i == 0 { + meta.MinTime = svc.MinTime + meta.MaxTime = svc.MaxTime + } else { + meta.MinTime = math.Min(meta.MinTime, svc.MinTime) + meta.MaxTime = math.Max(meta.MaxTime, svc.MaxTime) + } + s.sw.metrics.headSizeBytes.WithLabelValues(s.sshard, e.key.tenant).Observe(float64(svc.Size)) + meta.Datasets = append(meta.Datasets, svc) + } + + meta.Size = uint64(w.offset) + s.debuginfo.flushBlockDuration = time.Since(t1) + return blockPath, meta, nil +} + +func concatSegmentHead(sh *shard, e serviceHead, w *writerOffset) (*metastorev1.Dataset, error) { + tenantServiceOffset := w.offset + b := e.head.Meta() + ptypes := e.head.MustProfileTypeNames() + + profiles, index, symbols := getFilesForSegment(e.head, b) + + offsets := make([]uint64, 3) + var err error + offsets[0], err = concatFile(w, e.head, profiles, sh.concatBuf) + if err != nil { + return nil, err + } + offsets[1], err = concatFile(w, e.head, index, sh.concatBuf) + if err != nil { + return nil, err + } + offsets[2], err = concatFile(w, e.head, symbols, sh.concatBuf) + if err != nil { + return nil, err + } + + tenantServiceSize := w.offset - tenantServiceOffset + + svc := &metastorev1.Dataset{ + TenantId: e.key.tenant, + Name: e.key.service, + MinTime: int64(b.MinTime), + MaxTime: int64(b.MaxTime), + Size: uint64(tenantServiceSize), + // - 0: profiles.parquet + // - 1: index.tsdb + // - 2: symbols.symdb + TableOfContents: offsets, + ProfileTypes: ptypes, + } + return svc, nil +} + +func (s *segment) flushHeads(ctx context.Context) (moved []serviceHead) { + t1 := time.Now() + defer func() { + s.sw.metrics.flushHeadsDuration.WithLabelValues(s.sshard).Observe(time.Since(t1).Seconds()) + s.debuginfo.flushHeadsDuration = time.Since(t1) + }() + wg := sync.WaitGroup{} + mutex := new(sync.Mutex) + for _, e := range s.heads { + wg.Add(1) + e := e + go func() { + defer wg.Done() + eMoved, err := s.flushHead(ctx, e) + if err != nil { + level.Error(s.l).Log("msg", "failed to flush head", "err", err) + } + if eMoved { + mutex.Lock() + moved = append(moved, e) + mutex.Unlock() + } + }() + } + wg.Wait() + + slices.SortFunc(moved, func(i, j serviceHead) int { + c := strings.Compare(i.key.tenant, j.key.tenant) + if c != 0 { + return c + } + return strings.Compare(i.key.service, j.key.service) + }) + return moved +} + +func (s *segment) flushHead(ctx context.Context, e serviceHead) (moved bool, err error) { + th := time.Now() + if err := e.head.Flush(ctx); err != nil { + s.sw.metrics.flushServiceHeadDuration.WithLabelValues(s.sshard, e.key.tenant).Observe(time.Since(th).Seconds()) + s.sw.metrics.flushServiceHeadError.WithLabelValues(s.sshard, e.key.tenant).Inc() + return false, fmt.Errorf("failed to flush head %v: %w", e.head.BlockID(), err) + } + s.sw.metrics.flushServiceHeadDuration.WithLabelValues(s.sshard, e.key.tenant).Observe(time.Since(th).Seconds()) + stats, _ := json.Marshal(e.head.GetMetaStats()) + level.Debug(s.l).Log( + "msg", "flushed head", + "head", e.head.BlockID(), + "stats", string(stats), + "head-flush-duration", time.Since(th).String(), + ) + if err := e.head.Move(); err != nil { + if e.head.GetMetaStats().NumSamples == 0 { + _ = level.Debug(s.l).Log("msg", "skipping empty head", "head", e.head.BlockID()) + return false, nil + } + s.sw.metrics.flushServiceHeadError.WithLabelValues(s.sshard, e.key.tenant).Inc() + return false, fmt.Errorf("failed to move head %v: %w", e.head.BlockID(), err) + } + profiles, index, symbols := getFilesForSegment(e.head, e.head.Meta()) + if profiles == nil || index == nil || symbols == nil { + s.sw.metrics.flushServiceHeadError.WithLabelValues(s.sshard, e.key.tenant).Inc() + return false, fmt.Errorf("failed to find files %v %v %v", profiles, index, symbols) + } + return true, nil +} + +type serviceKey struct { + tenant string + service string +} +type serviceHead struct { + key serviceKey + head *phlaredb.Head +} + +type segment struct { + ulid ulid.ULID + shard shardKey + sshard string + inFlightProfiles sync.WaitGroup + heads map[serviceKey]serviceHead + headsLock sync.RWMutex + sw *segmentsWriter + dataPath string + doneChan chan error + l log.Logger + + debuginfo struct { + movedHeads int + waitInflight time.Duration + flushHeadsDuration time.Duration + flushBlockDuration time.Duration + storeMetaDuration time.Duration + } + sh *shard +} + +type segmentIngest interface { + ingest(ctx context.Context, tenantID string, p *profilev1.Profile, id uuid.UUID, labels []*typesv1.LabelPair) error +} + +type segmentWaitFlushed interface { + waitFlushed(ctx context.Context) error +} + +func (s *segment) waitFlushed(ctx context.Context) error { + select { + case <-ctx.Done(): + return fmt.Errorf("waitFlushed: %s %w", s.ulid.String(), ctx.Err()) + case err := <-s.doneChan: + return err + } +} + +func (s *segment) ingest(ctx context.Context, tenantID string, p *profilev1.Profile, id uuid.UUID, labels []*typesv1.LabelPair) error { + var err error + k := serviceKey{ + tenant: tenantID, + service: phlaremodel.Labels(labels).Get(phlaremodel.LabelNameServiceName), + } + s.sw.metrics.segmentIngestBytes.WithLabelValues(s.sshard, tenantID).Observe(float64(p.SizeVT())) + h, err := s.headForIngest(k) + if err != nil { + return err + } + return h.Ingest(ctx, p, id, labels...) +} + +func (s *segment) headForIngest(k serviceKey) (*phlaredb.Head, error) { + var err error + + s.headsLock.RLock() + h, ok := s.heads[k] + s.headsLock.RUnlock() + if ok { + return h.head, nil + } + + s.headsLock.Lock() + defer s.headsLock.Unlock() + h, ok = s.heads[k] + if ok { + return h.head, nil + } + + cfg := s.sw.cfg + cfg.DataPath = path.Join(s.dataPath) + cfg.SymDBFormat = symdb.FormatV3 + + nh, err := phlaredb.NewHead(s.sw.phlarectx, cfg, noopLimiter{}) + if err != nil { + return nil, err + } + + s.heads[k] = serviceHead{ + key: k, + head: nh, + } + + return nh, nil +} + +func (s *segment) cleanup() { + if err := os.RemoveAll(s.dataPath); err != nil { + _ = level.Error(s.l).Log("msg", "failed to cleanup segment", "err", err, "f", s.dataPath) + } +} + +func (sw *segmentsWriter) uploadBlock(blockPath string, s *segment) error { + t1 := time.Now() + + dst, err := filepath.Rel(sw.cfg.DataPath, blockPath) + if err != nil { + return err + } + if err := objstore.UploadFile(sw.phlarectx, sw.l, sw.bucket, blockPath, dst); err != nil { + return err + } + st, _ := os.Stat(blockPath) + if st != nil { + sw.metrics.segmentBlockSizeBytes.WithLabelValues(s.sshard).Observe(float64(st.Size())) + } + sw.metrics.blockUploadDuration.WithLabelValues(s.sshard).Observe(time.Since(t1).Seconds()) + sw.l.Log("msg", "uploaded block", "path", dst, "time-took", time.Since(t1)) + + return nil +} + +func (sw *segmentsWriter) storeMeta(ctx context.Context, meta *metastorev1.BlockMeta, s *segment) error { + t1 := time.Now() + + _, err := sw.metastoreClient.AddBlock(ctx, &metastorev1.AddBlockRequest{ + Block: meta, + }) + if err != nil { + sw.metrics.storeMetaErrors.WithLabelValues(s.sshard).Inc() + return err + } + sw.metrics.storeMetaDuration.WithLabelValues(s.sshard).Observe(time.Since(t1).Seconds()) + s.debuginfo.storeMetaDuration = time.Since(t1) + return nil +} + +func (sw *segmentsWriter) storeMetaDLQ(ctx context.Context, meta *metastorev1.BlockMeta, s *segment) error { + metaBlob, err := meta.MarshalVT() + if err != nil { + sw.metrics.storeMetaDLQ.WithLabelValues(s.sshard, "err").Inc() + return err + } + + fullPath := path.Join(pathDLQ, s.sshard, pathAnon, s.ulid.String(), "meta.pb") + if err = sw.bucket.Upload(ctx, + fullPath, + bytes.NewReader(metaBlob)); err != nil { + sw.metrics.storeMetaDLQ.WithLabelValues(s.sshard, "err").Inc() + return err + } + sw.metrics.storeMetaDLQ.WithLabelValues(s.sshard, "OK").Inc() + return nil +} + +func getFilesForSegment(_ *phlaredb.Head, b *block.Meta) (profiles *block.File, index *block.File, symbols *block.File) { + profiles = b.FileByRelPath("profiles.parquet") + index = b.FileByRelPath("index.tsdb") + symbols = b.FileByRelPath("symbols.symdb") + return +} + +type noopLimiter struct{} + +func (noopLimiter) AllowProfile(model.Fingerprint, phlaremodel.Labels, int64) error { return nil } + +func (noopLimiter) Stop() {} diff --git a/pkg/experiment/ingester/segment_metrics.go b/pkg/experiment/ingester/segment_metrics.go new file mode 100644 index 0000000000..255efe6769 --- /dev/null +++ b/pkg/experiment/ingester/segment_metrics.go @@ -0,0 +1,123 @@ +package ingester + +import ( + "github.com/prometheus/client_golang/prometheus" +) + +type segmentMetrics struct { + segmentIngestBytes *prometheus.HistogramVec + segmentBlockSizeBytes *prometheus.HistogramVec + headSizeBytes *prometheus.HistogramVec + storeMetaDuration *prometheus.HistogramVec + segmentFlushWaitDuration *prometheus.HistogramVec + segmentFlushTimeouts *prometheus.CounterVec + storeMetaErrors *prometheus.CounterVec + storeMetaDLQ *prometheus.CounterVec + blockUploadDuration *prometheus.HistogramVec + flushSegmentDuration *prometheus.HistogramVec + flushHeadsDuration *prometheus.HistogramVec + flushServiceHeadDuration *prometheus.HistogramVec + flushServiceHeadError *prometheus.CounterVec +} + +var ( + networkTimingBuckets = prometheus.ExponentialBucketsRange(0.005, 4, 20) + dataTimingBuckets = prometheus.ExponentialBucketsRange(0.001, 1, 20) + segmentFlushWaitBuckets = []float64{.1, .2, .3, .4, .5, .6, .7, .8, .9, 1, 1.1, 1.2, 1.3, 1.4, 1.5, 1.6, 1.7, 1.8, 1.9, 2} +) + +func newSegmentMetrics(reg prometheus.Registerer) *segmentMetrics { + + m := &segmentMetrics{ + segmentIngestBytes: prometheus.NewHistogramVec( + prometheus.HistogramOpts{ + Namespace: "pyroscope", + Name: "segment_ingest_bytes", + Buckets: prometheus.ExponentialBucketsRange(10*1024, 15*1024*1024, 20), + }, + []string{"shard", "tenant"}), + segmentBlockSizeBytes: prometheus.NewHistogramVec( + prometheus.HistogramOpts{ + Namespace: "pyroscope", + Name: "segment_block_size_bytes", + Buckets: prometheus.ExponentialBucketsRange(100*1024, 100*1024*1024, 20), + }, + []string{"shard"}), + storeMetaDuration: prometheus.NewHistogramVec(prometheus.HistogramOpts{ + Namespace: "pyroscope", + Name: "segment_store_meta_duration_seconds", + Buckets: networkTimingBuckets, + }, []string{"shard"}), + blockUploadDuration: prometheus.NewHistogramVec(prometheus.HistogramOpts{ + Namespace: "pyroscope", + Name: "segment_block_upload_duration_seconds", + Buckets: networkTimingBuckets, + }, []string{"shard"}), + + storeMetaErrors: prometheus.NewCounterVec( + prometheus.CounterOpts{ + Namespace: "pyroscope", + Name: "segment_store_meta_errors", + }, []string{"shard"}), + storeMetaDLQ: prometheus.NewCounterVec( + prometheus.CounterOpts{ + Namespace: "pyroscope", + Name: "segment_store_meta_dlq", + }, []string{"shard", "status"}), + + segmentFlushWaitDuration: prometheus.NewHistogramVec(prometheus.HistogramOpts{ + Namespace: "pyroscope", + Name: "segment_ingester_wait_duration_seconds", + Buckets: segmentFlushWaitBuckets, + }, []string{"tenant"}), + segmentFlushTimeouts: prometheus.NewCounterVec( + prometheus.CounterOpts{ + Namespace: "pyroscope", + Name: "segment_ingester_wait_timeouts", + }, []string{"tenant"}), + flushHeadsDuration: prometheus.NewHistogramVec(prometheus.HistogramOpts{ + Namespace: "pyroscope", + Name: "segment_flush_heads_duration_seconds", + Buckets: dataTimingBuckets, + }, []string{"shard"}), + flushServiceHeadDuration: prometheus.NewHistogramVec(prometheus.HistogramOpts{ + Namespace: "pyroscope", + Name: "segment_flush_service_head_duration_seconds", + Buckets: dataTimingBuckets, + }, []string{"shard", "tenant"}), + flushSegmentDuration: prometheus.NewHistogramVec(prometheus.HistogramOpts{ + Namespace: "pyroscope", + Name: "segment_flush_segment_duration_seconds", + Buckets: networkTimingBuckets, + }, []string{"shard"}), + + flushServiceHeadError: prometheus.NewCounterVec( + prometheus.CounterOpts{ + Namespace: "pyroscope", + Name: "segment_flush_service_head_errors", + }, []string{"shard", "tenant"}), + headSizeBytes: prometheus.NewHistogramVec( + prometheus.HistogramOpts{ + Namespace: "pyroscope", + Name: "segment_head_size_bytes", + Buckets: prometheus.ExponentialBucketsRange(10*1024, 100*1024*1024, 30), + }, []string{"shard", "tenant"}), + } + + if reg != nil { + reg.MustRegister(m.segmentIngestBytes) + reg.MustRegister(m.segmentBlockSizeBytes) + reg.MustRegister(m.storeMetaDuration) + reg.MustRegister(m.segmentFlushWaitDuration) + reg.MustRegister(m.segmentFlushTimeouts) + reg.MustRegister(m.storeMetaErrors) + reg.MustRegister(m.storeMetaDLQ) + reg.MustRegister(m.blockUploadDuration) + reg.MustRegister(m.flushHeadsDuration) + reg.MustRegister(m.flushServiceHeadDuration) + reg.MustRegister(m.flushServiceHeadError) + reg.MustRegister(m.flushSegmentDuration) + reg.MustRegister(m.headSizeBytes) + } + return m +} diff --git a/pkg/experiment/ingester/segment_test.go b/pkg/experiment/ingester/segment_test.go new file mode 100644 index 0000000000..f9fa91c5c0 --- /dev/null +++ b/pkg/experiment/ingester/segment_test.go @@ -0,0 +1,272 @@ +package ingester + +import ( + "bytes" + "context" + "crypto/rand" + "encoding/json" + "fmt" + "io" + "os" + "path/filepath" + "slices" + "strings" + "testing" + "time" + + "github.com/oklog/ulid" + "github.com/prometheus/client_golang/prometheus" + model2 "github.com/prometheus/common/model" + "github.com/prometheus/prometheus/util/testutil" + "github.com/stretchr/testify/assert" + "github.com/stretchr/testify/require" + "google.golang.org/grpc" + + ingestv1 "github.com/grafana/pyroscope/api/gen/proto/go/ingester/v1" + metastorev1 "github.com/grafana/pyroscope/api/gen/proto/go/metastore/v1" + typesv1 "github.com/grafana/pyroscope/api/gen/proto/go/types/v1" + "github.com/grafana/pyroscope/pkg/model" + "github.com/grafana/pyroscope/pkg/objstore/providers/filesystem" + "github.com/grafana/pyroscope/pkg/objstore/providers/memory" + phlarecontext "github.com/grafana/pyroscope/pkg/phlare/context" + "github.com/grafana/pyroscope/pkg/phlaredb" + "github.com/grafana/pyroscope/pkg/phlaredb/block" + pprofth "github.com/grafana/pyroscope/pkg/pprof/testhelper" +) + +type metastoreClient struct { + AddBlock_ func(ctx context.Context, in *metastorev1.AddBlockRequest, opts ...grpc.CallOption) (*metastorev1.AddBlockResponse, error) +} + +func (m *metastoreClient) AddBlock(ctx context.Context, in *metastorev1.AddBlockRequest, opts ...grpc.CallOption) (*metastorev1.AddBlockResponse, error) { + return m.AddBlock_(ctx, in, opts...) +} + +func (m *metastoreClient) QueryMetadata(ctx context.Context, in *metastorev1.QueryMetadataRequest, opts ...grpc.CallOption) (*metastorev1.QueryMetadataResponse, error) { + panic("implement me") +} + +func (m *metastoreClient) ReadIndex(ctx context.Context, in *metastorev1.ReadIndexRequest, opts ...grpc.CallOption) (*metastorev1.ReadIndexResponse, error) { + panic("implement me") +} + +const testSVCName = "svc239" +const testTenant = "tenant42" +const testShard = shardKey(239) + +func testProfile() *pprofth.ProfileBuilder { + return pprofth.NewProfileBuilder(time.Now().UnixNano()). + CPUProfile(). + WithLabels(model.LabelNameServiceName, testSVCName). + ForStacktraceString("foo", "bar"). + AddSamples(1) +} + +var cpuProfileType = &typesv1.ProfileType{ + ID: "process_cpu:cpu:nanoseconds:cpu:nanoseconds", + Name: "process_cpu", + SampleType: "cpu", + SampleUnit: "nanoseconds", + PeriodType: "cpu", + PeriodUnit: "nanoseconds", +} + +func TestSegmentIngest(t *testing.T) { + sw := newTestSegmentWriter(t) + defer sw.Stop() + blocks := make(chan *metastorev1.BlockMeta, 1) + sw.client.AddBlock_ = func(ctx context.Context, in *metastorev1.AddBlockRequest, opts ...grpc.CallOption) (*metastorev1.AddBlockResponse, error) { + blocks <- in.Block + close(blocks) + return &metastorev1.AddBlockResponse{}, nil + } + awaiter, err := sw.ingest(testShard, func(head segmentIngest) error { + p := testProfile() + err := head.ingest(context.Background(), testTenant, p.Profile, p.UUID, p.Labels) + assert.NoError(t, err) + return nil + }) + assert.NoError(t, err) + + err = awaiter.waitFlushed(context.Background()) + assert.NoError(t, err) + + meta := <-blocks + assert.Len(t, meta.Datasets, 1) + assert.Equal(t, uint32(testShard), meta.Shard) + assert.Equal(t, testTenant, meta.Datasets[0].TenantId) + assert.Equal(t, testSVCName, meta.Datasets[0].Name) + + blockQuerier := sw.createBlockFromMeta(meta, meta.Datasets[0]) + + q := blockQuerier.Queriers() + err = q.Open(context.Background()) + assert.NoError(t, err) + + res, err := q[0].SelectMergeByStacktraces(context.Background(), &ingestv1.SelectProfilesRequest{ + LabelSelector: fmt.Sprintf("{%s=\"%s\"}", model.LabelNameServiceName, testSVCName), + Type: cpuProfileType, + Start: 0, + End: time.Now().UnixMilli(), + }, 100) + assert.NoError(t, err) + collapsed := bytes.NewBuffer(nil) + res.WriteCollapsed(collapsed) + assert.Equal(t, string(collapsed.String()), "bar;foo 1\n") +} + +func TestSegmentIngestDLQ(t *testing.T) { + sw := newTestSegmentWriter(t) + defer sw.Stop() + sw.client.AddBlock_ = func(ctx context.Context, in *metastorev1.AddBlockRequest, opts ...grpc.CallOption) (*metastorev1.AddBlockResponse, error) { + return nil, fmt.Errorf("metastore unavailable") + } + awaiter, err := sw.ingest(testShard, func(head segmentIngest) error { + p := testProfile() + err := head.ingest(context.Background(), testTenant, p.Profile, p.UUID, p.Labels) + assert.NoError(t, err) + return nil + }) + assert.NoError(t, err) + + err = awaiter.waitFlushed(context.Background()) + assert.NoError(t, err) + + metas := getMetadataDLQ(sw) + assert.Len(t, metas, 1) + meta := metas[0] + + assert.Len(t, meta.Datasets, 1) + assert.Equal(t, uint32(testShard), meta.Shard) + assert.Equal(t, testTenant, meta.Datasets[0].TenantId) + assert.Equal(t, testSVCName, meta.Datasets[0].Name) + + blockQuerier := sw.createBlockFromMeta(meta, meta.Datasets[0]) + + q := blockQuerier.Queriers() + err = q.Open(context.Background()) + assert.NoError(t, err) + + res, err := q[0].SelectMergeByStacktraces(context.Background(), &ingestv1.SelectProfilesRequest{ + LabelSelector: fmt.Sprintf("{%s=\"%s\"}", model.LabelNameServiceName, testSVCName), + Type: cpuProfileType, + Start: 0, + End: time.Now().UnixMilli(), + }, 100) + assert.NoError(t, err) + collapsed := bytes.NewBuffer(nil) + res.WriteCollapsed(collapsed) + assert.Equal(t, string(collapsed.String()), "bar;foo 1\n") +} + +func getMetadataDLQ(sw sw) []*metastorev1.BlockMeta { + objects := sw.bucket.Objects() + dlqFiles := []string{} + for s := range objects { + if strings.HasPrefix(s, pathDLQ) { + dlqFiles = append(dlqFiles, s) + } + } + slices.Sort(dlqFiles) + var metas []*metastorev1.BlockMeta + for _, s := range dlqFiles { + var meta = new(metastorev1.BlockMeta) + err := meta.UnmarshalVT(objects[s]) + assert.NoError(sw.t, err) + metas = append(metas, meta) + } + return metas +} + +type sw struct { + *segmentsWriter + bucket *memory.InMemBucket + client *metastoreClient + phlarectx context.Context + t *testing.T +} + +func newTestSegmentWriter(t *testing.T) sw { + l := testutil.NewLogger(t) + phlarectx := phlarecontext.WithLogger(context.Background(), l) + reg := prometheus.NewRegistry() + phlarectx = phlarecontext.WithRegistry(phlarectx, reg) + cfg := phlaredb.Config{ + DataPath: t.TempDir(), + } + bucket := memory.NewInMemBucket() + client := new(metastoreClient) + res := newSegmentWriter(phlarectx, + l, + newSegmentMetrics(reg), + cfg, + bucket, + 1*time.Second, + client, + ) + return sw{ + t: t, + phlarectx: phlarectx, + segmentsWriter: res, + bucket: bucket, + client: client, + } +} + +func (sw *sw) createBlockFromMeta(meta *metastorev1.BlockMeta, ts *metastorev1.Dataset) *phlaredb.BlockQuerier { + dir := sw.t.TempDir() + blockid, err := ulid.New(uint64(meta.MaxTime), rand.Reader) + require.NoError(sw.t, err) + blockDir := filepath.Join(dir, blockid.String()) + sw.t.Logf("block dir: %s", blockDir) + os.Mkdir(blockDir, 0755) + blockBucket, err := filesystem.NewBucket(dir) + blobReader, err := sw.bucket.Get(context.Background(), fmt.Sprintf("%s/%d/%s/%s/%s", pathSegments, testShard, pathAnon, meta.Id, pathBlock)) + require.NoError(sw.t, err) + blob, err := io.ReadAll(blobReader) + require.NoError(sw.t, err) + + //block.MetaFilename + profiles := blob[ts.TableOfContents[0]:ts.TableOfContents[1]] + tsdb := blob[ts.TableOfContents[1]:ts.TableOfContents[2]] + symbols := blob[ts.TableOfContents[2] : ts.TableOfContents[0]+ts.Size] + + err = os.WriteFile(filepath.Join(blockDir, "profiles.parquet"), profiles, 0644) + assert.NoError(sw.t, err) + err = os.WriteFile(filepath.Join(blockDir, "index.tsdb"), tsdb, 0644) + assert.NoError(sw.t, err) + err = os.WriteFile(filepath.Join(blockDir, "symbols.symdb"), symbols, 0644) + assert.NoError(sw.t, err) + + blockMeta := block.Meta{ + ULID: blockid, + MinTime: model2.TimeFromUnix(meta.MinTime), + MaxTime: model2.TimeFromUnix(meta.MaxTime), + Files: []block.File{ + { + RelPath: "profiles.parquet", + SizeBytes: uint64(len(profiles)), + }, + { + RelPath: "index.tsdb", + SizeBytes: uint64(len(tsdb)), + }, + { + RelPath: "symbols.symdb", + SizeBytes: uint64(len(symbols)), + }, + }, + Version: block.MetaVersion3, + } + blockMetaJson, err := json.Marshal(&blockMeta) + assert.NoError(sw.t, err) + err = os.WriteFile(filepath.Join(blockDir, block.MetaFilename), blockMetaJson, 0644) + assert.NoError(sw.t, err) + + blockQuerier := phlaredb.NewBlockQuerier(sw.phlarectx, blockBucket) + + err = blockQuerier.Sync(context.Background()) + require.NoError(sw.t, err) + + return blockQuerier +} diff --git a/pkg/experiment/ingester/service.go b/pkg/experiment/ingester/service.go new file mode 100644 index 0000000000..98680b0efe --- /dev/null +++ b/pkg/experiment/ingester/service.go @@ -0,0 +1,218 @@ +package ingester + +import ( + "context" + "flag" + "fmt" + segmentWriterV1 "github.com/grafana/pyroscope/api/gen/proto/go/segmentwriter/v1" + "time" + + "github.com/google/uuid" + + profilev1 "github.com/grafana/pyroscope/api/gen/proto/go/google/v1" + metastoreclient "github.com/grafana/pyroscope/pkg/experiment/metastore/client" + "github.com/grafana/pyroscope/pkg/pprof" + "github.com/grafana/pyroscope/pkg/tenant" + "github.com/grafana/pyroscope/pkg/validation" + + "connectrpc.com/connect" + "github.com/go-kit/log" + "github.com/go-kit/log/level" + "github.com/grafana/dskit/multierror" + "github.com/grafana/dskit/ring" + "github.com/grafana/dskit/services" + "github.com/pkg/errors" + "github.com/prometheus/client_golang/prometheus" + + phlareobj "github.com/grafana/pyroscope/pkg/objstore" + phlarecontext "github.com/grafana/pyroscope/pkg/phlare/context" + "github.com/grafana/pyroscope/pkg/phlaredb" + "github.com/grafana/pyroscope/pkg/util" +) + +type Config struct { + LifecyclerConfig ring.LifecyclerConfig `yaml:"lifecycler,omitempty"` + SegmentDuration time.Duration `yaml:"segmentDuration,omitempty"` + Async bool `yaml:"async,omitempty"` //todo make it pertenant +} + +// RegisterFlags registers the flags. +func (cfg *Config) RegisterFlags(f *flag.FlagSet) { + const prefix = "segment-writer." + cfg.LifecyclerConfig.RegisterFlagsWithPrefix(prefix, f, util.Logger) + f.DurationVar(&cfg.SegmentDuration, prefix+"segment.duration", 500*time.Millisecond, "Timeout when flushing segments to bucket.") + f.BoolVar(&cfg.Async, prefix+"async", false, "Enable async mode for segment writer.") +} + +func (cfg *Config) Validate() error { + return nil +} + +type SegmentWriterService struct { + services.Service + + cfg Config + dbConfig phlaredb.Config + logger log.Logger + phlarectx context.Context + + lifecycler *ring.Lifecycler + subservices *services.Manager + subservicesWatcher *services.FailureWatcher + + storageBucket phlareobj.Bucket + + //limiters *limiters + segmentWriter *segmentsWriter + + //limits Limits + reg prometheus.Registerer +} + +type ingesterFlusherCompat struct { + *SegmentWriterService +} + +func (i *ingesterFlusherCompat) Flush() { + err := i.SegmentWriterService.Flush() + if err != nil { + level.Error(i.SegmentWriterService.logger).Log("msg", "flush failed", "err", err) + } +} + +func New(phlarectx context.Context, cfg Config, dbConfig phlaredb.Config, storageBucket phlareobj.Bucket, metastoreClient *metastoreclient.Client) (*SegmentWriterService, error) { + reg := phlarecontext.Registry(phlarectx) + reg = prometheus.WrapRegistererWith(prometheus.Labels{"component": "segment-writer"}, reg) + phlarectx = phlarecontext.WithRegistry(phlarectx, reg) + log := phlarecontext.Logger(phlarectx) + phlarectx = phlaredb.ContextWithHeadMetrics(phlarectx, reg, "pyroscope_segment_writer") + + i := &SegmentWriterService{ + cfg: cfg, + phlarectx: phlarectx, + logger: log, + reg: reg, + dbConfig: dbConfig, + storageBucket: storageBucket, + } + + var ( + err error + ) + + i.lifecycler, err = ring.NewLifecycler( + cfg.LifecyclerConfig, + &ingesterFlusherCompat{i}, + "segment-writer-ring-name", + "segment-writer-ring-key", + true, + i.logger, prometheus.WrapRegistererWithPrefix("pyroscope_segment_writer_", i.reg)) + if err != nil { + return nil, err + } + + i.subservices, err = services.NewManager(i.lifecycler) + if err != nil { + return nil, errors.Wrap(err, "services manager") + } + if storageBucket == nil { + return nil, errors.New("storage bucket is required for segment writer") + } + if metastoreClient == nil { + return nil, errors.New("metastore client is required for segment writer") + } + metrics := newSegmentMetrics(i.reg) + + i.segmentWriter = newSegmentWriter(i.phlarectx, i.logger, metrics, i.dbConfig, storageBucket, cfg.SegmentDuration, metastoreClient) + i.subservicesWatcher = services.NewFailureWatcher() + i.subservicesWatcher.WatchManager(i.subservices) + i.Service = services.NewBasicService(i.starting, i.running, i.stopping) + return i, nil +} + +func (i *SegmentWriterService) starting(ctx context.Context) error { + return services.StartManagerAndAwaitHealthy(ctx, i.subservices) +} + +func (i *SegmentWriterService) running(ctx context.Context) error { + select { + case <-ctx.Done(): + return nil + case err := <-i.subservicesWatcher.Chan(): // handle lifecycler errors + return fmt.Errorf("lifecycler failed: %w", err) + } +} + +func (i *SegmentWriterService) stopping(_ error) error { + errs := multierror.New() + errs.Add(services.StopManagerAndAwaitStopped(context.Background(), i.subservices)) + errs.Add(i.segmentWriter.Stop()) + return errs.Err() +} + +func (i *SegmentWriterService) Push(ctx context.Context, req *connect.Request[segmentWriterV1.PushRequest]) (*connect.Response[segmentWriterV1.PushResponse], error) { + tenantID, err := tenant.ExtractTenantIDFromContext(ctx) + if err != nil { + return nil, connect.NewError(connect.CodeInvalidArgument, err) + } + series := req.Msg.Series + var shard = shardKey(series.Shard) + wait, err := i.segmentWriter.ingest(shard, func(segment segmentIngest) error { + return i.ingestToSegment(ctx, segment, series, tenantID) + }) + if err != nil { + return nil, err + } + if i.cfg.Async { + return connect.NewResponse(&segmentWriterV1.PushResponse{}), nil + } + t1 := time.Now() + if err = wait.waitFlushed(ctx); err != nil { + i.segmentWriter.metrics.segmentFlushTimeouts.WithLabelValues(tenantID).Inc() + i.segmentWriter.metrics.segmentFlushWaitDuration.WithLabelValues(tenantID).Observe(time.Since(t1).Seconds()) + if errors.Is(err, context.DeadlineExceeded) { + level.Error(i.logger).Log("msg", "flush timeout", "err", err) + } else { + level.Error(i.logger).Log("msg", "flush err", "err", err) + } + return nil, err + } + i.segmentWriter.metrics.segmentFlushWaitDuration.WithLabelValues(tenantID).Observe(time.Since(t1).Seconds()) + return connect.NewResponse(&segmentWriterV1.PushResponse{}), nil +} + +func (i *SegmentWriterService) ingestToSegment(ctx context.Context, segment segmentIngest, series *segmentWriterV1.RawProfileSeries, tenantID string) error { + sample := series.Sample + id, err := uuid.Parse(sample.ID) + if err != nil { + return err + } + return pprof.FromBytes(sample.RawProfile, func(p *profilev1.Profile, size int) error { + if err = segment.ingest(ctx, tenantID, p, id, series.Labels); err != nil { + reason := validation.ReasonOf(err) + if reason != validation.Unknown { + validation.DiscardedProfiles.WithLabelValues(string(reason), tenantID).Add(float64(1)) + validation.DiscardedBytes.WithLabelValues(string(reason), tenantID).Add(float64(size)) + } + } + return nil + }) +} + +func (i *SegmentWriterService) Flush() error { + return i.segmentWriter.Stop() +} + +func (i *SegmentWriterService) TransferOut(ctx context.Context) error { + return ring.ErrTransferDisabled +} + +// CheckReady is used to indicate to k8s when the ingesters are ready for +// the addition removal of another ingester. Returns 204 when the ingester is +// ready, 500 otherwise. +func (i *SegmentWriterService) CheckReady(ctx context.Context) error { + if s := i.State(); s != services.Running && s != services.Stopping { + return fmt.Errorf("ingester not ready: %v", s) + } + return i.lifecycler.CheckReady(ctx) +} diff --git a/pkg/experiment/ingester/writer_offset.go b/pkg/experiment/ingester/writer_offset.go new file mode 100644 index 0000000000..b97f34e52c --- /dev/null +++ b/pkg/experiment/ingester/writer_offset.go @@ -0,0 +1,48 @@ +package ingester + +import ( + "io" + "os" + + "github.com/grafana/pyroscope/pkg/phlaredb" + "github.com/grafana/pyroscope/pkg/phlaredb/block" +) + +type writerOffset struct { + io.Writer + offset int64 + //err error +} + +func withWriterOffset(w io.Writer) *writerOffset { + return &writerOffset{Writer: w} +} + +//func (w *writerOffset) write(p []byte) { +// if w.err == nil { +// n, err := w.Writer.Write(p) +// w.offset += int64(n) +// w.err = err +// } +//} + +func (w *writerOffset) Write(p []byte) (n int, err error) { + n, err = w.Writer.Write(p) + w.offset += int64(n) + return n, err +} + +func concatFile(w *writerOffset, h *phlaredb.Head, f *block.File, buf []byte) (uint64, error) { + o := w.offset + fp := h.LocalPathFor(f.RelPath) + file, err := os.Open(fp) + if err != nil { + return 0, err + } + defer file.Close() + _, err = io.CopyBuffer(w, file, buf) + if err != nil { + return 0, err + } + return uint64(o), nil +} diff --git a/pkg/experiment/metastore/client/client.go b/pkg/experiment/metastore/client/client.go new file mode 100644 index 0000000000..8dd198b6ac --- /dev/null +++ b/pkg/experiment/metastore/client/client.go @@ -0,0 +1,85 @@ +package metastoreclient + +import ( + "context" + + "github.com/go-kit/log" + + "github.com/grafana/dskit/grpcclient" + "github.com/grafana/dskit/services" + "github.com/opentracing-contrib/go-grpc" + "github.com/opentracing/opentracing-go" + "google.golang.org/grpc" + + compactorv1 "github.com/grafana/pyroscope/api/gen/proto/go/compactor/v1" + metastorev1 "github.com/grafana/pyroscope/api/gen/proto/go/metastore/v1" +) + +type Client struct { + metastorev1.MetastoreServiceClient + compactorv1.CompactionPlannerClient + service services.Service + conn *grpc.ClientConn +} + +func New(address string, grpcClientConfig grpcclient.Config, logger log.Logger) (*Client, error) { + conn, err := dial(address, grpcClientConfig, logger) + if err != nil { + return nil, err + } + var c Client + c.MetastoreServiceClient = metastorev1.NewMetastoreServiceClient(conn) + c.CompactionPlannerClient = compactorv1.NewCompactionPlannerClient(conn) + c.service = services.NewIdleService(c.starting, c.stopping) + c.conn = conn + return &c, nil +} + +func (c *Client) Service() services.Service { return c.service } +func (c *Client) starting(context.Context) error { return nil } +func (c *Client) stopping(error) error { return c.conn.Close() } + +func dial(address string, grpcClientConfig grpcclient.Config, _ log.Logger) (*grpc.ClientConn, error) { + if err := grpcClientConfig.Validate(); err != nil { + return nil, err + } + options, err := grpcClientConfig.DialOption(nil, nil) + if err != nil { + return nil, err + } + // TODO: https://github.com/grpc/grpc-proto/blob/master/grpc/service_config/service_config.proto + options = append(options, + grpc.WithDefaultServiceConfig(grpcServiceConfig), + grpc.WithUnaryInterceptor(otgrpc.OpenTracingClientInterceptor(opentracing.GlobalTracer())), + ) + // TODO: Implement k8s grpc resolver. + // Note that this may require additional permissions. + // Consider: https://github.com/sercand/kuberesolver + // if os.Getenv("KUBERNETES_SERVICE_HOST") != "" { + // builder, err := NewGrpcResolverBuilder(logger, address) + // if err != nil { + // return nil, fmt.Errorf("failed to create grpc resolver builder: %w", err) + // } + // options = append(options, grpc.WithResolvers(builder)) + // return grpc.Dial(builder.resolverAddrStub(), options...) + // } + return grpc.Dial(address, options...) +} + +const grpcServiceConfig = `{ + "healthCheckConfig": { + "serviceName": "metastore.v1.MetastoreService.RaftLeader" + }, + "loadBalancingPolicy":"round_robin", + "methodConfig": [{ + "name": [{"service": "metastore.v1.MetastoreService"}], + "waitForReady": true, + "retryPolicy": { + "MaxAttempts": 16, + "InitialBackoff": ".01s", + "MaxBackoff": ".01s", + "BackoffMultiplier": 1.0, + "RetryableStatusCodes": [ "UNAVAILABLE" ] + } + }] +}` diff --git a/pkg/experiment/metastore/client/grpc_endpointslice_resolver.go b/pkg/experiment/metastore/client/grpc_endpointslice_resolver.go new file mode 100644 index 0000000000..b029f03555 --- /dev/null +++ b/pkg/experiment/metastore/client/grpc_endpointslice_resolver.go @@ -0,0 +1,202 @@ +package metastoreclient + +import ( + "context" + "fmt" + "regexp" + "strings" + + "github.com/go-kit/log" + "github.com/prometheus/prometheus/discovery" + promk8s "github.com/prometheus/prometheus/discovery/kubernetes" + "github.com/prometheus/prometheus/discovery/targetgroup" + "google.golang.org/grpc/resolver" +) + +const GrpcEndpointSLiceResovlerSchema = "metastore-endpointslice" + +type EndpointSliceResolverBuilder struct { + l log.Logger + name string + namespace string + port string +} + +func NewGrpcResolverBuilder(l log.Logger, address string) (*EndpointSliceResolverBuilder, error) { + g := &EndpointSliceResolverBuilder{l: log.With(l, "component", "metastore-grpc-resolver-builder")} + name, namespace, port, err := getEndpointSliceTargetFromDnsTarget(address) + if err != nil { + return nil, fmt.Errorf("failed to parse target: %w", err) + } + g.name = name + g.namespace = namespace + g.port = port + g.l.Log("msg", "created new grpc resolver builder", "name", name, "namespace", namespace, "port", port) + + return g, nil +} + +func (g *EndpointSliceResolverBuilder) Build(target resolver.Target, cc resolver.ClientConn, opts resolver.BuildOptions) (resolver.Resolver, error) { + g.l.Log("msg", "building resolver", "target", target, "opts", fmt.Sprintf("%+v", opts)) + rr := &GrpcResolver{w: nil, l: log.With(g.l, "component", "metastore-grpc-resolver", "target", target)} + newWatcher, err := NewEndpointSliceWatcher(g.l, g.name, g.namespace, func(ips []string) { + addresses := make([]resolver.Address, 0, len(ips)) + for _, ip := range ips { + addresses = append(addresses, resolver.Address{Addr: ip + ":" + g.port}) + } + err := cc.UpdateState(resolver.State{Addresses: addresses}) + if err != nil { + rr.l.Log("msg", "failed to update state", "err", err) + } else { + rr.l.Log("msg", "updated state", "addresses", fmt.Sprintf("%+v", addresses)) + } + }) + if err != nil { + return nil, fmt.Errorf("failed to create watcher: %w", err) + } + rr.w = newWatcher + return rr, nil +} + +func (g *EndpointSliceResolverBuilder) Scheme() string { + return GrpcEndpointSLiceResovlerSchema +} + +func (g *EndpointSliceResolverBuilder) resolverAddrStub() string { + return fmt.Sprintf("%s://stub:239", GrpcEndpointSLiceResovlerSchema) +} + +type GrpcResolver struct { + w *EndpointSliceWatcher + l log.Logger +} + +func (g *GrpcResolver) ResolveNow(o resolver.ResolveNowOptions) { + //g.l.Log("msg", "resolve now", "opts", o) +} + +func (g *GrpcResolver) Close() { + g.l.Log("msg", "close") + g.w.Close() +} + +func getEndpointSliceTargetFromDnsTarget(src string) (string, string, string, error) { + + re := regexp.MustCompile("dns:///_grpc._tcp\\.([\\S^.]+)\\.([\\S^.]+)(\\.svc\\.cluster\\.local\\.):([0-9]+)") + all := re.FindSubmatch([]byte(src)) + if len(all) == 0 { + return "", "", "", fmt.Errorf("failed to parse target") + } + name := string(all[1]) + namespace := string(all[2]) + port := string(all[4]) + return name, namespace, port, nil +} + +type EndpointSliceWatcher struct { + l log.Logger + d discovery.Discoverer + ctx context.Context + cancel context.CancelFunc + name string + namespace string + cb func(ips []string) +} + +func (w *EndpointSliceWatcher) watch(up chan []*targetgroup.Group) { + isNeededSlice := func(group *targetgroup.Group) bool { //todo proper selection + //if strings.Contains(group.Source, "endpointslice/"+"pyroscope"+"/"+"pyroscope-micro-services-metastore-headless") { + // return true + //} + //if strings.Contains(group.Source, "endpointslice/"+"profiles-dev-003"+"/"+"pyroscope-metastore-headless") { + // return true + //} + substr := "endpointslice/" + w.namespace + "/" + w.name + w.l.Log("msg", "checking group", "source", group.Source, "substr", substr) + if strings.Contains(group.Source, substr) { + return true + } + return false + } + w.l.Log("msg", "starting watch") + for { + select { + case <-w.ctx.Done(): + w.l.Log("msg", "context done, stopping watch") + return + case groups := <-up: + ipset := make(map[string]string) + for _, group := range groups { + if !isNeededSlice(group) { + w.l.Log("msg", "skipping group", "source", group.Source) + continue + } + w.l.Log("msg", "processing group", "source", group.Source) + for _, target := range group.Targets { + ip := target["__meta_kubernetes_pod_ip"] + ready := target["__meta_kubernetes_pod_ready"] + phase := target["__meta_kubernetes_pod_phase"] + podname := target["__meta_kubernetes_pod_name"] + w.l.Log("msg", "received new target", "tt", fmt.Sprintf(">>%s %s %s<<", ip, phase, ready)) + ipset[string(ip)] = string(podname) + } + } + if len(ipset) == 0 { + continue + } + if w.cb != nil { + ipss := make([]string, 0, len(ipset)) + for k := range ipset { + ipss = append(ipss, k) + } + w.cb(ipss) + } + w.l.Log("msg", "received new target groups", "ips", fmt.Sprintf("%+v", ipset)) + } + } +} + +func (w *EndpointSliceWatcher) Close() error { + w.cancel() + return nil +} + +func NewEndpointSliceWatcher(l log.Logger, name, namespace string, cb func(ips []string)) (*EndpointSliceWatcher, error) { + l = log.With(l, "component", "metastore-watcher") + sdc := &promk8s.SDConfig{ + Role: promk8s.RoleEndpointSlice, + Selectors: []promk8s.SelectorConfig{ + { + Role: promk8s.RoleEndpointSlice, + Label: "app.kubernetes.io/component=metastore", + }, + }, + } + refreshMetrics := discovery.NewRefreshMetrics(nil) + m := sdc.NewDiscovererMetrics(nil, refreshMetrics) + d, err := sdc.NewDiscoverer(discovery.DiscovererOptions{ + Logger: log.With(l, "component", "metastore-watcher-discovery"), + Metrics: m, + HTTPClientOptions: nil, + }) + if err != nil { + l.Log("msg", "failed to create discoverer", "err", err) + return nil, err + } + ctx, cacnel := context.WithCancel(context.Background()) + up := make(chan []*targetgroup.Group) + + l.Log("msg", "starting watcher") + w := &EndpointSliceWatcher{ + d: d, + cancel: cacnel, + ctx: ctx, + l: l, + cb: cb, + name: name, + namespace: namespace, + } + go d.Run(ctx, up) + go w.watch(up) + return w, nil +} diff --git a/pkg/experiment/metastore/compactionpb/compaction.pb.go b/pkg/experiment/metastore/compactionpb/compaction.pb.go new file mode 100644 index 0000000000..0cd0129d38 --- /dev/null +++ b/pkg/experiment/metastore/compactionpb/compaction.pb.go @@ -0,0 +1,403 @@ +// Code generated by protoc-gen-go. DO NOT EDIT. +// versions: +// protoc-gen-go v1.34.2 +// protoc (unknown) +// source: experiment/metastore/compactionpb/compaction.proto + +package compactionpb + +import ( + protoreflect "google.golang.org/protobuf/reflect/protoreflect" + protoimpl "google.golang.org/protobuf/runtime/protoimpl" + reflect "reflect" + sync "sync" +) + +const ( + // Verify that this generated code is sufficiently up-to-date. + _ = protoimpl.EnforceVersion(20 - protoimpl.MinVersion) + // Verify that runtime/protoimpl is sufficiently up-to-date. + _ = protoimpl.EnforceVersion(protoimpl.MaxVersion - 20) +) + +type CompactionStatus int32 + +const ( + CompactionStatus_COMPACTION_STATUS_UNSPECIFIED CompactionStatus = 0 + CompactionStatus_COMPACTION_STATUS_IN_PROGRESS CompactionStatus = 1 + CompactionStatus_COMPACTION_STATUS_SUCCESS CompactionStatus = 2 + CompactionStatus_COMPACTION_STATUS_FAILURE CompactionStatus = 3 +) + +// Enum value maps for CompactionStatus. +var ( + CompactionStatus_name = map[int32]string{ + 0: "COMPACTION_STATUS_UNSPECIFIED", + 1: "COMPACTION_STATUS_IN_PROGRESS", + 2: "COMPACTION_STATUS_SUCCESS", + 3: "COMPACTION_STATUS_FAILURE", + } + CompactionStatus_value = map[string]int32{ + "COMPACTION_STATUS_UNSPECIFIED": 0, + "COMPACTION_STATUS_IN_PROGRESS": 1, + "COMPACTION_STATUS_SUCCESS": 2, + "COMPACTION_STATUS_FAILURE": 3, + } +) + +func (x CompactionStatus) Enum() *CompactionStatus { + p := new(CompactionStatus) + *p = x + return p +} + +func (x CompactionStatus) String() string { + return protoimpl.X.EnumStringOf(x.Descriptor(), protoreflect.EnumNumber(x)) +} + +func (CompactionStatus) Descriptor() protoreflect.EnumDescriptor { + return file_experiment_metastore_compactionpb_compaction_proto_enumTypes[0].Descriptor() +} + +func (CompactionStatus) Type() protoreflect.EnumType { + return &file_experiment_metastore_compactionpb_compaction_proto_enumTypes[0] +} + +func (x CompactionStatus) Number() protoreflect.EnumNumber { + return protoreflect.EnumNumber(x) +} + +// Deprecated: Use CompactionStatus.Descriptor instead. +func (CompactionStatus) EnumDescriptor() ([]byte, []int) { + return file_experiment_metastore_compactionpb_compaction_proto_rawDescGZIP(), []int{0} +} + +type CompactionJob struct { + state protoimpl.MessageState + sizeCache protoimpl.SizeCache + unknownFields protoimpl.UnknownFields + + // Unique name of the job. + Name string `protobuf:"bytes,1,opt,name=name,proto3" json:"name,omitempty"` + // List of the input blocks. + Blocks []string `protobuf:"bytes,2,rep,name=blocks,proto3" json:"blocks,omitempty"` + // Compaction level (all blocks are the same) + CompactionLevel uint32 `protobuf:"varint,3,opt,name=compaction_level,json=compactionLevel,proto3" json:"compaction_level,omitempty"` + // The index of the raft command that changed the status of the job. + // Used as a fencing token in conjunction with the lease_expires_at + // field to manage ownership of the compaction job. Any access to the + // job must be guarded by the check: current_index >= raft_log_index. + // If the check fails, the access should be denied. + // + // The index is updated every time the job is assigned to a worker. + RaftLogIndex uint64 `protobuf:"varint,4,opt,name=raft_log_index,json=raftLogIndex,proto3" json:"raft_log_index,omitempty"` + // Shard the blocks belong to. + Shard uint32 `protobuf:"varint,5,opt,name=shard,proto3" json:"shard,omitempty"` + // Optional, empty for compaction level 0. + TenantId string `protobuf:"bytes,6,opt,name=tenant_id,json=tenantId,proto3" json:"tenant_id,omitempty"` + Status CompactionStatus `protobuf:"varint,7,opt,name=status,proto3,enum=compaction.CompactionStatus" json:"status,omitempty"` + // The time the compaction job lease expires. If a lease is expired, the + // job is considered abandoned and can be picked up by another worker. + // The expiration check should be done by comparing the timestamp of + // the raft log entry (command that accesses the job) with the value of + // this field. + // + // The lease is extended every time the owner reports a status update. + LeaseExpiresAt int64 `protobuf:"varint,8,opt,name=lease_expires_at,json=leaseExpiresAt,proto3" json:"lease_expires_at,omitempty"` +} + +func (x *CompactionJob) Reset() { + *x = CompactionJob{} + if protoimpl.UnsafeEnabled { + mi := &file_experiment_metastore_compactionpb_compaction_proto_msgTypes[0] + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + ms.StoreMessageInfo(mi) + } +} + +func (x *CompactionJob) String() string { + return protoimpl.X.MessageStringOf(x) +} + +func (*CompactionJob) ProtoMessage() {} + +func (x *CompactionJob) ProtoReflect() protoreflect.Message { + mi := &file_experiment_metastore_compactionpb_compaction_proto_msgTypes[0] + if protoimpl.UnsafeEnabled && x != nil { + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + if ms.LoadMessageInfo() == nil { + ms.StoreMessageInfo(mi) + } + return ms + } + return mi.MessageOf(x) +} + +// Deprecated: Use CompactionJob.ProtoReflect.Descriptor instead. +func (*CompactionJob) Descriptor() ([]byte, []int) { + return file_experiment_metastore_compactionpb_compaction_proto_rawDescGZIP(), []int{0} +} + +func (x *CompactionJob) GetName() string { + if x != nil { + return x.Name + } + return "" +} + +func (x *CompactionJob) GetBlocks() []string { + if x != nil { + return x.Blocks + } + return nil +} + +func (x *CompactionJob) GetCompactionLevel() uint32 { + if x != nil { + return x.CompactionLevel + } + return 0 +} + +func (x *CompactionJob) GetRaftLogIndex() uint64 { + if x != nil { + return x.RaftLogIndex + } + return 0 +} + +func (x *CompactionJob) GetShard() uint32 { + if x != nil { + return x.Shard + } + return 0 +} + +func (x *CompactionJob) GetTenantId() string { + if x != nil { + return x.TenantId + } + return "" +} + +func (x *CompactionJob) GetStatus() CompactionStatus { + if x != nil { + return x.Status + } + return CompactionStatus_COMPACTION_STATUS_UNSPECIFIED +} + +func (x *CompactionJob) GetLeaseExpiresAt() int64 { + if x != nil { + return x.LeaseExpiresAt + } + return 0 +} + +type JobPreQueue struct { + state protoimpl.MessageState + sizeCache protoimpl.SizeCache + unknownFields protoimpl.UnknownFields + + CompactionLevel uint32 `protobuf:"varint,1,opt,name=compaction_level,json=compactionLevel,proto3" json:"compaction_level,omitempty"` + Shard uint32 `protobuf:"varint,2,opt,name=shard,proto3" json:"shard,omitempty"` + Tenant string `protobuf:"bytes,3,opt,name=tenant,proto3" json:"tenant,omitempty"` + Blocks []string `protobuf:"bytes,4,rep,name=blocks,proto3" json:"blocks,omitempty"` +} + +func (x *JobPreQueue) Reset() { + *x = JobPreQueue{} + if protoimpl.UnsafeEnabled { + mi := &file_experiment_metastore_compactionpb_compaction_proto_msgTypes[1] + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + ms.StoreMessageInfo(mi) + } +} + +func (x *JobPreQueue) String() string { + return protoimpl.X.MessageStringOf(x) +} + +func (*JobPreQueue) ProtoMessage() {} + +func (x *JobPreQueue) ProtoReflect() protoreflect.Message { + mi := &file_experiment_metastore_compactionpb_compaction_proto_msgTypes[1] + if protoimpl.UnsafeEnabled && x != nil { + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + if ms.LoadMessageInfo() == nil { + ms.StoreMessageInfo(mi) + } + return ms + } + return mi.MessageOf(x) +} + +// Deprecated: Use JobPreQueue.ProtoReflect.Descriptor instead. +func (*JobPreQueue) Descriptor() ([]byte, []int) { + return file_experiment_metastore_compactionpb_compaction_proto_rawDescGZIP(), []int{1} +} + +func (x *JobPreQueue) GetCompactionLevel() uint32 { + if x != nil { + return x.CompactionLevel + } + return 0 +} + +func (x *JobPreQueue) GetShard() uint32 { + if x != nil { + return x.Shard + } + return 0 +} + +func (x *JobPreQueue) GetTenant() string { + if x != nil { + return x.Tenant + } + return "" +} + +func (x *JobPreQueue) GetBlocks() []string { + if x != nil { + return x.Blocks + } + return nil +} + +var File_experiment_metastore_compactionpb_compaction_proto protoreflect.FileDescriptor + +var file_experiment_metastore_compactionpb_compaction_proto_rawDesc = []byte{ + 0x0a, 0x32, 0x65, 0x78, 0x70, 0x65, 0x72, 0x69, 0x6d, 0x65, 0x6e, 0x74, 0x2f, 0x6d, 0x65, 0x74, + 0x61, 0x73, 0x74, 0x6f, 0x72, 0x65, 0x2f, 0x63, 0x6f, 0x6d, 0x70, 0x61, 0x63, 0x74, 0x69, 0x6f, + 0x6e, 0x70, 0x62, 0x2f, 0x63, 0x6f, 0x6d, 0x70, 0x61, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x2e, 0x70, + 0x72, 0x6f, 0x74, 0x6f, 0x12, 0x0a, 0x63, 0x6f, 0x6d, 0x70, 0x61, 0x63, 0x74, 0x69, 0x6f, 0x6e, + 0x22, 0x9f, 0x02, 0x0a, 0x0d, 0x43, 0x6f, 0x6d, 0x70, 0x61, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x4a, + 0x6f, 0x62, 0x12, 0x12, 0x0a, 0x04, 0x6e, 0x61, 0x6d, 0x65, 0x18, 0x01, 0x20, 0x01, 0x28, 0x09, + 0x52, 0x04, 0x6e, 0x61, 0x6d, 0x65, 0x12, 0x16, 0x0a, 0x06, 0x62, 0x6c, 0x6f, 0x63, 0x6b, 0x73, + 0x18, 0x02, 0x20, 0x03, 0x28, 0x09, 0x52, 0x06, 0x62, 0x6c, 0x6f, 0x63, 0x6b, 0x73, 0x12, 0x29, + 0x0a, 0x10, 0x63, 0x6f, 0x6d, 0x70, 0x61, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x5f, 0x6c, 0x65, 0x76, + 0x65, 0x6c, 0x18, 0x03, 0x20, 0x01, 0x28, 0x0d, 0x52, 0x0f, 0x63, 0x6f, 0x6d, 0x70, 0x61, 0x63, + 0x74, 0x69, 0x6f, 0x6e, 0x4c, 0x65, 0x76, 0x65, 0x6c, 0x12, 0x24, 0x0a, 0x0e, 0x72, 0x61, 0x66, + 0x74, 0x5f, 0x6c, 0x6f, 0x67, 0x5f, 0x69, 0x6e, 0x64, 0x65, 0x78, 0x18, 0x04, 0x20, 0x01, 0x28, + 0x04, 0x52, 0x0c, 0x72, 0x61, 0x66, 0x74, 0x4c, 0x6f, 0x67, 0x49, 0x6e, 0x64, 0x65, 0x78, 0x12, + 0x14, 0x0a, 0x05, 0x73, 0x68, 0x61, 0x72, 0x64, 0x18, 0x05, 0x20, 0x01, 0x28, 0x0d, 0x52, 0x05, + 0x73, 0x68, 0x61, 0x72, 0x64, 0x12, 0x1b, 0x0a, 0x09, 0x74, 0x65, 0x6e, 0x61, 0x6e, 0x74, 0x5f, + 0x69, 0x64, 0x18, 0x06, 0x20, 0x01, 0x28, 0x09, 0x52, 0x08, 0x74, 0x65, 0x6e, 0x61, 0x6e, 0x74, + 0x49, 0x64, 0x12, 0x34, 0x0a, 0x06, 0x73, 0x74, 0x61, 0x74, 0x75, 0x73, 0x18, 0x07, 0x20, 0x01, + 0x28, 0x0e, 0x32, 0x1c, 0x2e, 0x63, 0x6f, 0x6d, 0x70, 0x61, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x2e, + 0x43, 0x6f, 0x6d, 0x70, 0x61, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x53, 0x74, 0x61, 0x74, 0x75, 0x73, + 0x52, 0x06, 0x73, 0x74, 0x61, 0x74, 0x75, 0x73, 0x12, 0x28, 0x0a, 0x10, 0x6c, 0x65, 0x61, 0x73, + 0x65, 0x5f, 0x65, 0x78, 0x70, 0x69, 0x72, 0x65, 0x73, 0x5f, 0x61, 0x74, 0x18, 0x08, 0x20, 0x01, + 0x28, 0x03, 0x52, 0x0e, 0x6c, 0x65, 0x61, 0x73, 0x65, 0x45, 0x78, 0x70, 0x69, 0x72, 0x65, 0x73, + 0x41, 0x74, 0x22, 0x7e, 0x0a, 0x0b, 0x4a, 0x6f, 0x62, 0x50, 0x72, 0x65, 0x51, 0x75, 0x65, 0x75, + 0x65, 0x12, 0x29, 0x0a, 0x10, 0x63, 0x6f, 0x6d, 0x70, 0x61, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x5f, + 0x6c, 0x65, 0x76, 0x65, 0x6c, 0x18, 0x01, 0x20, 0x01, 0x28, 0x0d, 0x52, 0x0f, 0x63, 0x6f, 0x6d, + 0x70, 0x61, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x4c, 0x65, 0x76, 0x65, 0x6c, 0x12, 0x14, 0x0a, 0x05, + 0x73, 0x68, 0x61, 0x72, 0x64, 0x18, 0x02, 0x20, 0x01, 0x28, 0x0d, 0x52, 0x05, 0x73, 0x68, 0x61, + 0x72, 0x64, 0x12, 0x16, 0x0a, 0x06, 0x74, 0x65, 0x6e, 0x61, 0x6e, 0x74, 0x18, 0x03, 0x20, 0x01, + 0x28, 0x09, 0x52, 0x06, 0x74, 0x65, 0x6e, 0x61, 0x6e, 0x74, 0x12, 0x16, 0x0a, 0x06, 0x62, 0x6c, + 0x6f, 0x63, 0x6b, 0x73, 0x18, 0x04, 0x20, 0x03, 0x28, 0x09, 0x52, 0x06, 0x62, 0x6c, 0x6f, 0x63, + 0x6b, 0x73, 0x2a, 0x96, 0x01, 0x0a, 0x10, 0x43, 0x6f, 0x6d, 0x70, 0x61, 0x63, 0x74, 0x69, 0x6f, + 0x6e, 0x53, 0x74, 0x61, 0x74, 0x75, 0x73, 0x12, 0x21, 0x0a, 0x1d, 0x43, 0x4f, 0x4d, 0x50, 0x41, + 0x43, 0x54, 0x49, 0x4f, 0x4e, 0x5f, 0x53, 0x54, 0x41, 0x54, 0x55, 0x53, 0x5f, 0x55, 0x4e, 0x53, + 0x50, 0x45, 0x43, 0x49, 0x46, 0x49, 0x45, 0x44, 0x10, 0x00, 0x12, 0x21, 0x0a, 0x1d, 0x43, 0x4f, + 0x4d, 0x50, 0x41, 0x43, 0x54, 0x49, 0x4f, 0x4e, 0x5f, 0x53, 0x54, 0x41, 0x54, 0x55, 0x53, 0x5f, + 0x49, 0x4e, 0x5f, 0x50, 0x52, 0x4f, 0x47, 0x52, 0x45, 0x53, 0x53, 0x10, 0x01, 0x12, 0x1d, 0x0a, + 0x19, 0x43, 0x4f, 0x4d, 0x50, 0x41, 0x43, 0x54, 0x49, 0x4f, 0x4e, 0x5f, 0x53, 0x54, 0x41, 0x54, + 0x55, 0x53, 0x5f, 0x53, 0x55, 0x43, 0x43, 0x45, 0x53, 0x53, 0x10, 0x02, 0x12, 0x1d, 0x0a, 0x19, + 0x43, 0x4f, 0x4d, 0x50, 0x41, 0x43, 0x54, 0x49, 0x4f, 0x4e, 0x5f, 0x53, 0x54, 0x41, 0x54, 0x55, + 0x53, 0x5f, 0x46, 0x41, 0x49, 0x4c, 0x55, 0x52, 0x45, 0x10, 0x03, 0x42, 0xad, 0x01, 0x0a, 0x0e, + 0x63, 0x6f, 0x6d, 0x2e, 0x63, 0x6f, 0x6d, 0x70, 0x61, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x42, 0x0f, + 0x43, 0x6f, 0x6d, 0x70, 0x61, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x50, 0x72, 0x6f, 0x74, 0x6f, 0x50, + 0x01, 0x5a, 0x42, 0x67, 0x69, 0x74, 0x68, 0x75, 0x62, 0x2e, 0x63, 0x6f, 0x6d, 0x2f, 0x67, 0x72, + 0x61, 0x66, 0x61, 0x6e, 0x61, 0x2f, 0x70, 0x79, 0x72, 0x6f, 0x73, 0x63, 0x6f, 0x70, 0x65, 0x2f, + 0x70, 0x6b, 0x67, 0x2f, 0x65, 0x78, 0x70, 0x65, 0x72, 0x69, 0x6d, 0x65, 0x6e, 0x74, 0x2f, 0x6d, + 0x65, 0x74, 0x61, 0x73, 0x74, 0x6f, 0x72, 0x65, 0x2f, 0x63, 0x6f, 0x6d, 0x70, 0x61, 0x63, 0x74, + 0x69, 0x6f, 0x6e, 0x70, 0x62, 0xa2, 0x02, 0x03, 0x43, 0x58, 0x58, 0xaa, 0x02, 0x0a, 0x43, 0x6f, + 0x6d, 0x70, 0x61, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0xca, 0x02, 0x0a, 0x43, 0x6f, 0x6d, 0x70, 0x61, + 0x63, 0x74, 0x69, 0x6f, 0x6e, 0xe2, 0x02, 0x16, 0x43, 0x6f, 0x6d, 0x70, 0x61, 0x63, 0x74, 0x69, + 0x6f, 0x6e, 0x5c, 0x47, 0x50, 0x42, 0x4d, 0x65, 0x74, 0x61, 0x64, 0x61, 0x74, 0x61, 0xea, 0x02, + 0x0a, 0x43, 0x6f, 0x6d, 0x70, 0x61, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x62, 0x06, 0x70, 0x72, 0x6f, + 0x74, 0x6f, 0x33, +} + +var ( + file_experiment_metastore_compactionpb_compaction_proto_rawDescOnce sync.Once + file_experiment_metastore_compactionpb_compaction_proto_rawDescData = file_experiment_metastore_compactionpb_compaction_proto_rawDesc +) + +func file_experiment_metastore_compactionpb_compaction_proto_rawDescGZIP() []byte { + file_experiment_metastore_compactionpb_compaction_proto_rawDescOnce.Do(func() { + file_experiment_metastore_compactionpb_compaction_proto_rawDescData = protoimpl.X.CompressGZIP(file_experiment_metastore_compactionpb_compaction_proto_rawDescData) + }) + return file_experiment_metastore_compactionpb_compaction_proto_rawDescData +} + +var file_experiment_metastore_compactionpb_compaction_proto_enumTypes = make([]protoimpl.EnumInfo, 1) +var file_experiment_metastore_compactionpb_compaction_proto_msgTypes = make([]protoimpl.MessageInfo, 2) +var file_experiment_metastore_compactionpb_compaction_proto_goTypes = []any{ + (CompactionStatus)(0), // 0: compaction.CompactionStatus + (*CompactionJob)(nil), // 1: compaction.CompactionJob + (*JobPreQueue)(nil), // 2: compaction.JobPreQueue +} +var file_experiment_metastore_compactionpb_compaction_proto_depIdxs = []int32{ + 0, // 0: compaction.CompactionJob.status:type_name -> compaction.CompactionStatus + 1, // [1:1] is the sub-list for method output_type + 1, // [1:1] is the sub-list for method input_type + 1, // [1:1] is the sub-list for extension type_name + 1, // [1:1] is the sub-list for extension extendee + 0, // [0:1] is the sub-list for field type_name +} + +func init() { file_experiment_metastore_compactionpb_compaction_proto_init() } +func file_experiment_metastore_compactionpb_compaction_proto_init() { + if File_experiment_metastore_compactionpb_compaction_proto != nil { + return + } + if !protoimpl.UnsafeEnabled { + file_experiment_metastore_compactionpb_compaction_proto_msgTypes[0].Exporter = func(v any, i int) any { + switch v := v.(*CompactionJob); i { + case 0: + return &v.state + case 1: + return &v.sizeCache + case 2: + return &v.unknownFields + default: + return nil + } + } + file_experiment_metastore_compactionpb_compaction_proto_msgTypes[1].Exporter = func(v any, i int) any { + switch v := v.(*JobPreQueue); i { + case 0: + return &v.state + case 1: + return &v.sizeCache + case 2: + return &v.unknownFields + default: + return nil + } + } + } + type x struct{} + out := protoimpl.TypeBuilder{ + File: protoimpl.DescBuilder{ + GoPackagePath: reflect.TypeOf(x{}).PkgPath(), + RawDescriptor: file_experiment_metastore_compactionpb_compaction_proto_rawDesc, + NumEnums: 1, + NumMessages: 2, + NumExtensions: 0, + NumServices: 0, + }, + GoTypes: file_experiment_metastore_compactionpb_compaction_proto_goTypes, + DependencyIndexes: file_experiment_metastore_compactionpb_compaction_proto_depIdxs, + EnumInfos: file_experiment_metastore_compactionpb_compaction_proto_enumTypes, + MessageInfos: file_experiment_metastore_compactionpb_compaction_proto_msgTypes, + }.Build() + File_experiment_metastore_compactionpb_compaction_proto = out.File + file_experiment_metastore_compactionpb_compaction_proto_rawDesc = nil + file_experiment_metastore_compactionpb_compaction_proto_goTypes = nil + file_experiment_metastore_compactionpb_compaction_proto_depIdxs = nil +} diff --git a/pkg/experiment/metastore/compactionpb/compaction.proto b/pkg/experiment/metastore/compactionpb/compaction.proto new file mode 100644 index 0000000000..3884b27c36 --- /dev/null +++ b/pkg/experiment/metastore/compactionpb/compaction.proto @@ -0,0 +1,47 @@ +syntax = "proto3"; + +package compaction; + +message CompactionJob { + // Unique name of the job. + string name = 1; + // List of the input blocks. + repeated string blocks = 2; + // Compaction level (all blocks are the same) + uint32 compaction_level = 3; + // The index of the raft command that changed the status of the job. + // Used as a fencing token in conjunction with the lease_expires_at + // field to manage ownership of the compaction job. Any access to the + // job must be guarded by the check: current_index >= raft_log_index. + // If the check fails, the access should be denied. + // + // The index is updated every time the job is assigned to a worker. + uint64 raft_log_index = 4; + // Shard the blocks belong to. + uint32 shard = 5; + // Optional, empty for compaction level 0. + string tenant_id = 6; + CompactionStatus status = 7; + // The time the compaction job lease expires. If a lease is expired, the + // job is considered abandoned and can be picked up by another worker. + // The expiration check should be done by comparing the timestamp of + // the raft log entry (command that accesses the job) with the value of + // this field. + // + // The lease is extended every time the owner reports a status update. + int64 lease_expires_at = 8; +} + +enum CompactionStatus { + COMPACTION_STATUS_UNSPECIFIED = 0; + COMPACTION_STATUS_IN_PROGRESS = 1; + COMPACTION_STATUS_SUCCESS = 2; + COMPACTION_STATUS_FAILURE = 3; +} + +message JobPreQueue { + uint32 compaction_level = 1; + uint32 shard = 2; + string tenant = 3; + repeated string blocks = 4; +} diff --git a/pkg/experiment/metastore/compactionpb/compaction_vtproto.pb.go b/pkg/experiment/metastore/compactionpb/compaction_vtproto.pb.go new file mode 100644 index 0000000000..c339e2c9d0 --- /dev/null +++ b/pkg/experiment/metastore/compactionpb/compaction_vtproto.pb.go @@ -0,0 +1,620 @@ +// Code generated by protoc-gen-go-vtproto. DO NOT EDIT. +// protoc-gen-go-vtproto version: v0.6.0 +// source: experiment/metastore/compactionpb/compaction.proto + +package compactionpb + +import ( + fmt "fmt" + protohelpers "github.com/planetscale/vtprotobuf/protohelpers" + protoimpl "google.golang.org/protobuf/runtime/protoimpl" + io "io" +) + +const ( + // Verify that this generated code is sufficiently up-to-date. + _ = protoimpl.EnforceVersion(20 - protoimpl.MinVersion) + // Verify that runtime/protoimpl is sufficiently up-to-date. + _ = protoimpl.EnforceVersion(protoimpl.MaxVersion - 20) +) + +func (m *CompactionJob) MarshalVT() (dAtA []byte, err error) { + if m == nil { + return nil, nil + } + size := m.SizeVT() + dAtA = make([]byte, size) + n, err := m.MarshalToSizedBufferVT(dAtA[:size]) + if err != nil { + return nil, err + } + return dAtA[:n], nil +} + +func (m *CompactionJob) MarshalToVT(dAtA []byte) (int, error) { + size := m.SizeVT() + return m.MarshalToSizedBufferVT(dAtA[:size]) +} + +func (m *CompactionJob) MarshalToSizedBufferVT(dAtA []byte) (int, error) { + if m == nil { + return 0, nil + } + i := len(dAtA) + _ = i + var l int + _ = l + if m.unknownFields != nil { + i -= len(m.unknownFields) + copy(dAtA[i:], m.unknownFields) + } + if m.LeaseExpiresAt != 0 { + i = protohelpers.EncodeVarint(dAtA, i, uint64(m.LeaseExpiresAt)) + i-- + dAtA[i] = 0x40 + } + if m.Status != 0 { + i = protohelpers.EncodeVarint(dAtA, i, uint64(m.Status)) + i-- + dAtA[i] = 0x38 + } + if len(m.TenantId) > 0 { + i -= len(m.TenantId) + copy(dAtA[i:], m.TenantId) + i = protohelpers.EncodeVarint(dAtA, i, uint64(len(m.TenantId))) + i-- + dAtA[i] = 0x32 + } + if m.Shard != 0 { + i = protohelpers.EncodeVarint(dAtA, i, uint64(m.Shard)) + i-- + dAtA[i] = 0x28 + } + if m.RaftLogIndex != 0 { + i = protohelpers.EncodeVarint(dAtA, i, uint64(m.RaftLogIndex)) + i-- + dAtA[i] = 0x20 + } + if m.CompactionLevel != 0 { + i = protohelpers.EncodeVarint(dAtA, i, uint64(m.CompactionLevel)) + i-- + dAtA[i] = 0x18 + } + if len(m.Blocks) > 0 { + for iNdEx := len(m.Blocks) - 1; iNdEx >= 0; iNdEx-- { + i -= len(m.Blocks[iNdEx]) + copy(dAtA[i:], m.Blocks[iNdEx]) + i = protohelpers.EncodeVarint(dAtA, i, uint64(len(m.Blocks[iNdEx]))) + i-- + dAtA[i] = 0x12 + } + } + if len(m.Name) > 0 { + i -= len(m.Name) + copy(dAtA[i:], m.Name) + i = protohelpers.EncodeVarint(dAtA, i, uint64(len(m.Name))) + i-- + dAtA[i] = 0xa + } + return len(dAtA) - i, nil +} + +func (m *JobPreQueue) MarshalVT() (dAtA []byte, err error) { + if m == nil { + return nil, nil + } + size := m.SizeVT() + dAtA = make([]byte, size) + n, err := m.MarshalToSizedBufferVT(dAtA[:size]) + if err != nil { + return nil, err + } + return dAtA[:n], nil +} + +func (m *JobPreQueue) MarshalToVT(dAtA []byte) (int, error) { + size := m.SizeVT() + return m.MarshalToSizedBufferVT(dAtA[:size]) +} + +func (m *JobPreQueue) MarshalToSizedBufferVT(dAtA []byte) (int, error) { + if m == nil { + return 0, nil + } + i := len(dAtA) + _ = i + var l int + _ = l + if m.unknownFields != nil { + i -= len(m.unknownFields) + copy(dAtA[i:], m.unknownFields) + } + if len(m.Blocks) > 0 { + for iNdEx := len(m.Blocks) - 1; iNdEx >= 0; iNdEx-- { + i -= len(m.Blocks[iNdEx]) + copy(dAtA[i:], m.Blocks[iNdEx]) + i = protohelpers.EncodeVarint(dAtA, i, uint64(len(m.Blocks[iNdEx]))) + i-- + dAtA[i] = 0x22 + } + } + if len(m.Tenant) > 0 { + i -= len(m.Tenant) + copy(dAtA[i:], m.Tenant) + i = protohelpers.EncodeVarint(dAtA, i, uint64(len(m.Tenant))) + i-- + dAtA[i] = 0x1a + } + if m.Shard != 0 { + i = protohelpers.EncodeVarint(dAtA, i, uint64(m.Shard)) + i-- + dAtA[i] = 0x10 + } + if m.CompactionLevel != 0 { + i = protohelpers.EncodeVarint(dAtA, i, uint64(m.CompactionLevel)) + i-- + dAtA[i] = 0x8 + } + return len(dAtA) - i, nil +} + +func (m *CompactionJob) SizeVT() (n int) { + if m == nil { + return 0 + } + var l int + _ = l + l = len(m.Name) + if l > 0 { + n += 1 + l + protohelpers.SizeOfVarint(uint64(l)) + } + if len(m.Blocks) > 0 { + for _, s := range m.Blocks { + l = len(s) + n += 1 + l + protohelpers.SizeOfVarint(uint64(l)) + } + } + if m.CompactionLevel != 0 { + n += 1 + protohelpers.SizeOfVarint(uint64(m.CompactionLevel)) + } + if m.RaftLogIndex != 0 { + n += 1 + protohelpers.SizeOfVarint(uint64(m.RaftLogIndex)) + } + if m.Shard != 0 { + n += 1 + protohelpers.SizeOfVarint(uint64(m.Shard)) + } + l = len(m.TenantId) + if l > 0 { + n += 1 + l + protohelpers.SizeOfVarint(uint64(l)) + } + if m.Status != 0 { + n += 1 + protohelpers.SizeOfVarint(uint64(m.Status)) + } + if m.LeaseExpiresAt != 0 { + n += 1 + protohelpers.SizeOfVarint(uint64(m.LeaseExpiresAt)) + } + n += len(m.unknownFields) + return n +} + +func (m *JobPreQueue) SizeVT() (n int) { + if m == nil { + return 0 + } + var l int + _ = l + if m.CompactionLevel != 0 { + n += 1 + protohelpers.SizeOfVarint(uint64(m.CompactionLevel)) + } + if m.Shard != 0 { + n += 1 + protohelpers.SizeOfVarint(uint64(m.Shard)) + } + l = len(m.Tenant) + if l > 0 { + n += 1 + l + protohelpers.SizeOfVarint(uint64(l)) + } + if len(m.Blocks) > 0 { + for _, s := range m.Blocks { + l = len(s) + n += 1 + l + protohelpers.SizeOfVarint(uint64(l)) + } + } + n += len(m.unknownFields) + return n +} + +func (m *CompactionJob) UnmarshalVT(dAtA []byte) error { + l := len(dAtA) + iNdEx := 0 + for iNdEx < l { + preIndex := iNdEx + var wire uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + wire |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + fieldNum := int32(wire >> 3) + wireType := int(wire & 0x7) + if wireType == 4 { + return fmt.Errorf("proto: CompactionJob: wiretype end group for non-group") + } + if fieldNum <= 0 { + return fmt.Errorf("proto: CompactionJob: illegal tag %d (wire type %d)", fieldNum, wire) + } + switch fieldNum { + case 1: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field Name", wireType) + } + var stringLen uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + stringLen |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + intStringLen := int(stringLen) + if intStringLen < 0 { + return protohelpers.ErrInvalidLength + } + postIndex := iNdEx + intStringLen + if postIndex < 0 { + return protohelpers.ErrInvalidLength + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + m.Name = string(dAtA[iNdEx:postIndex]) + iNdEx = postIndex + case 2: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field Blocks", wireType) + } + var stringLen uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + stringLen |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + intStringLen := int(stringLen) + if intStringLen < 0 { + return protohelpers.ErrInvalidLength + } + postIndex := iNdEx + intStringLen + if postIndex < 0 { + return protohelpers.ErrInvalidLength + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + m.Blocks = append(m.Blocks, string(dAtA[iNdEx:postIndex])) + iNdEx = postIndex + case 3: + if wireType != 0 { + return fmt.Errorf("proto: wrong wireType = %d for field CompactionLevel", wireType) + } + m.CompactionLevel = 0 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + m.CompactionLevel |= uint32(b&0x7F) << shift + if b < 0x80 { + break + } + } + case 4: + if wireType != 0 { + return fmt.Errorf("proto: wrong wireType = %d for field RaftLogIndex", wireType) + } + m.RaftLogIndex = 0 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + m.RaftLogIndex |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + case 5: + if wireType != 0 { + return fmt.Errorf("proto: wrong wireType = %d for field Shard", wireType) + } + m.Shard = 0 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + m.Shard |= uint32(b&0x7F) << shift + if b < 0x80 { + break + } + } + case 6: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field TenantId", wireType) + } + var stringLen uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + stringLen |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + intStringLen := int(stringLen) + if intStringLen < 0 { + return protohelpers.ErrInvalidLength + } + postIndex := iNdEx + intStringLen + if postIndex < 0 { + return protohelpers.ErrInvalidLength + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + m.TenantId = string(dAtA[iNdEx:postIndex]) + iNdEx = postIndex + case 7: + if wireType != 0 { + return fmt.Errorf("proto: wrong wireType = %d for field Status", wireType) + } + m.Status = 0 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + m.Status |= CompactionStatus(b&0x7F) << shift + if b < 0x80 { + break + } + } + case 8: + if wireType != 0 { + return fmt.Errorf("proto: wrong wireType = %d for field LeaseExpiresAt", wireType) + } + m.LeaseExpiresAt = 0 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + m.LeaseExpiresAt |= int64(b&0x7F) << shift + if b < 0x80 { + break + } + } + default: + iNdEx = preIndex + skippy, err := protohelpers.Skip(dAtA[iNdEx:]) + if err != nil { + return err + } + if (skippy < 0) || (iNdEx+skippy) < 0 { + return protohelpers.ErrInvalidLength + } + if (iNdEx + skippy) > l { + return io.ErrUnexpectedEOF + } + m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...) + iNdEx += skippy + } + } + + if iNdEx > l { + return io.ErrUnexpectedEOF + } + return nil +} +func (m *JobPreQueue) UnmarshalVT(dAtA []byte) error { + l := len(dAtA) + iNdEx := 0 + for iNdEx < l { + preIndex := iNdEx + var wire uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + wire |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + fieldNum := int32(wire >> 3) + wireType := int(wire & 0x7) + if wireType == 4 { + return fmt.Errorf("proto: JobPreQueue: wiretype end group for non-group") + } + if fieldNum <= 0 { + return fmt.Errorf("proto: JobPreQueue: illegal tag %d (wire type %d)", fieldNum, wire) + } + switch fieldNum { + case 1: + if wireType != 0 { + return fmt.Errorf("proto: wrong wireType = %d for field CompactionLevel", wireType) + } + m.CompactionLevel = 0 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + m.CompactionLevel |= uint32(b&0x7F) << shift + if b < 0x80 { + break + } + } + case 2: + if wireType != 0 { + return fmt.Errorf("proto: wrong wireType = %d for field Shard", wireType) + } + m.Shard = 0 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + m.Shard |= uint32(b&0x7F) << shift + if b < 0x80 { + break + } + } + case 3: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field Tenant", wireType) + } + var stringLen uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + stringLen |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + intStringLen := int(stringLen) + if intStringLen < 0 { + return protohelpers.ErrInvalidLength + } + postIndex := iNdEx + intStringLen + if postIndex < 0 { + return protohelpers.ErrInvalidLength + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + m.Tenant = string(dAtA[iNdEx:postIndex]) + iNdEx = postIndex + case 4: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field Blocks", wireType) + } + var stringLen uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + stringLen |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + intStringLen := int(stringLen) + if intStringLen < 0 { + return protohelpers.ErrInvalidLength + } + postIndex := iNdEx + intStringLen + if postIndex < 0 { + return protohelpers.ErrInvalidLength + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + m.Blocks = append(m.Blocks, string(dAtA[iNdEx:postIndex])) + iNdEx = postIndex + default: + iNdEx = preIndex + skippy, err := protohelpers.Skip(dAtA[iNdEx:]) + if err != nil { + return err + } + if (skippy < 0) || (iNdEx+skippy) < 0 { + return protohelpers.ErrInvalidLength + } + if (iNdEx + skippy) > l { + return io.ErrUnexpectedEOF + } + m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...) + iNdEx += skippy + } + } + + if iNdEx > l { + return io.ErrUnexpectedEOF + } + return nil +} diff --git a/pkg/experiment/metastore/metastore.go b/pkg/experiment/metastore/metastore.go new file mode 100644 index 0000000000..063983a3c2 --- /dev/null +++ b/pkg/experiment/metastore/metastore.go @@ -0,0 +1,283 @@ +package metastore + +import ( + "context" + "flag" + "fmt" + "net" + "os" + "path/filepath" + "strings" + "sync" + "time" + + "github.com/go-kit/log" + "github.com/go-kit/log/level" + "github.com/grafana/dskit/flagext" + "github.com/grafana/dskit/grpcclient" + "github.com/grafana/dskit/services" + "github.com/hashicorp/raft" + raftwal "github.com/hashicorp/raft-wal" + "github.com/pkg/errors" + "github.com/prometheus/client_golang/prometheus" + + compactorv1 "github.com/grafana/pyroscope/api/gen/proto/go/compactor/v1" + metastorev1 "github.com/grafana/pyroscope/api/gen/proto/go/metastore/v1" + "github.com/grafana/pyroscope/pkg/experiment/metastore/client" + "github.com/grafana/pyroscope/pkg/experiment/metastore/raftleader" + "github.com/grafana/pyroscope/pkg/util/health" +) + +const ( + snapshotsRetain = 3 + walCacheEntries = 512 + transportConnPoolSize = 10 + transportTimeout = 10 * time.Second + + raftTrailingLogs = 18 << 10 + raftSnapshotInterval = 180 * time.Second + raftSnapshotThreshold = 8 << 10 + + metastoreRaftLeaderHealthServiceName = "metastore.v1.MetastoreService.RaftLeader" +) + +type Config struct { + Address string `yaml:"address"` + GRPCClientConfig grpcclient.Config `yaml:"grpc_client_config" doc:"description=Configures the gRPC client used to communicate with the metastore."` + DataDir string `yaml:"data_dir"` + Raft RaftConfig `yaml:"raft"` +} + +type RaftConfig struct { + Dir string `yaml:"dir"` + + BootstrapPeers []string `yaml:"bootstrap_peers"` + BootstrapExpectPeers int `yaml:"bootstrap_expect_peers"` + + ServerID string `yaml:"server_id"` + BindAddress string `yaml:"bind_address"` + AdvertiseAddress string `yaml:"advertise_address"` + + ApplyTimeout time.Duration `yaml:"apply_timeout" doc:"hidden"` +} + +func (cfg *Config) RegisterFlags(f *flag.FlagSet) { + const prefix = "metastore." + f.StringVar(&cfg.Address, prefix+"address", "localhost:9095", "") + cfg.GRPCClientConfig.RegisterFlagsWithPrefix(prefix+"grpc-client-config", f) + f.StringVar(&cfg.DataDir, prefix+"data-dir", "./data-metastore/data", "") + cfg.Raft.RegisterFlagsWithPrefix(prefix+"raft.", f) +} + +func (cfg *RaftConfig) RegisterFlagsWithPrefix(prefix string, f *flag.FlagSet) { + f.StringVar(&cfg.Dir, prefix+"dir", "./data-metastore/raft", "") + f.Var((*flagext.StringSlice)(&cfg.BootstrapPeers), prefix+"bootstrap-peers", "") + f.IntVar(&cfg.BootstrapExpectPeers, prefix+"bootstrap-expect-peers", 1, "Expected number of peers including the local node.") + f.StringVar(&cfg.BindAddress, prefix+"bind-address", "localhost:9099", "") + f.StringVar(&cfg.ServerID, prefix+"server-id", "localhost:9099", "") + f.StringVar(&cfg.AdvertiseAddress, prefix+"advertise-address", "localhost:9099", "") + f.DurationVar(&cfg.ApplyTimeout, prefix+"apply-timeout", 5*time.Second, "") +} + +type Metastore struct { + service services.Service + metastorev1.MetastoreServiceServer + compactorv1.CompactionPlannerServer + + config Config + logger log.Logger + reg prometheus.Registerer + limits Limits + + // In-memory state. + state *metastoreState + + // Persistent state. + db *boltdb + + // Raft module. + wal *raftwal.WAL + snapshots *raft.FileSnapshotStore + transport *raft.NetworkTransport + raft *raft.Raft + leaderhealth *raftleader.HealthObserver + + logStore raft.LogStore + stableStore raft.StableStore + snapshotStore raft.SnapshotStore + + walDir string + + done chan struct{} + wg sync.WaitGroup + metrics *metastoreMetrics + client *metastoreclient.Client + readySince time.Time +} + +type Limits interface{} + +func New(config Config, limits Limits, logger log.Logger, reg prometheus.Registerer, hs health.Service, client *metastoreclient.Client) (*Metastore, error) { + metrics := newMetastoreMetrics(reg) + m := &Metastore{ + config: config, + logger: logger, + reg: reg, + limits: limits, + db: newDB(config, logger, metrics), + done: make(chan struct{}), + metrics: metrics, + client: client, + } + m.leaderhealth = raftleader.NewRaftLeaderHealthObserver(hs, logger, raftleader.NewMetrics(reg)) + m.state = newMetastoreState(logger, m.db, m.reg) + m.service = services.NewBasicService(m.starting, m.running, m.stopping) + return m, nil +} + +func (m *Metastore) Service() services.Service { return m.service } + +func (m *Metastore) Shutdown() error { + m.shutdownRaft() + m.db.shutdown() + return nil +} + +func (m *Metastore) starting(ctx context.Context) error { + if err := m.db.open(false); err != nil { + return fmt.Errorf("failed to initialize database: %w", err) + } + if err := m.initRaft(); err != nil { + return fmt.Errorf("failed to initialize raft: %w", err) + } + m.wg.Add(1) + go m.cleanupLoop() + return nil +} + +func (m *Metastore) stopping(_ error) error { + close(m.done) + m.wg.Wait() + return m.Shutdown() +} + +func (m *Metastore) running(ctx context.Context) error { + <-ctx.Done() + return nil +} + +func (m *Metastore) initRaft() (err error) { + defer func() { + if err != nil { + // If the initialization fails, initialized components + // should be de-initialized gracefully. + m.shutdownRaft() + } + }() + + hasState, err := m.openRaftStore() + if err != nil { + return err + } + + addr, err := net.ResolveTCPAddr("tcp", m.config.Raft.AdvertiseAddress) + if err != nil { + return err + } + m.transport, err = raft.NewTCPTransport(m.config.Raft.BindAddress, addr, transportConnPoolSize, transportTimeout, os.Stderr) + if err != nil { + return err + } + + config := raft.DefaultConfig() + // TODO: Wrap gokit + // config.Logger + config.LogLevel = "debug" + config.TrailingLogs = raftTrailingLogs + config.SnapshotThreshold = raftSnapshotThreshold + config.SnapshotInterval = raftSnapshotInterval + config.LocalID = raft.ServerID(m.config.Raft.ServerID) + + fsm := newFSM(m.logger, m.db, m.state) + m.raft, err = raft.NewRaft(config, fsm, m.logStore, m.stableStore, m.snapshotStore, m.transport) + if err != nil { + return fmt.Errorf("starting raft node: %w", err) + } + + if !hasState { + _ = level.Warn(m.logger).Log("msg", "no existing state found, trying to bootstrap cluster") + if err = m.bootstrap(); err != nil { + return fmt.Errorf("failed to bootstrap cluster: %w", err) + } + } + + m.leaderhealth.Register(m.raft, metastoreRaftLeaderHealthServiceName) + return nil +} + +func (m *Metastore) openRaftStore() (hasState bool, err error) { + if err = m.createRaftDirs(); err != nil { + return false, err + } + m.wal, err = raftwal.Open(m.walDir) + if err != nil { + return false, fmt.Errorf("failed to open WAL: %w", err) + } + m.snapshots, err = raft.NewFileSnapshotStore(m.config.Raft.Dir, snapshotsRetain, os.Stderr) + if err != nil { + return false, fmt.Errorf("failed to open shapshot store: %w", err) + } + m.logStore = m.wal + m.logStore, _ = raft.NewLogCache(walCacheEntries, m.logStore) + m.stableStore = m.wal + m.snapshotStore = m.snapshots + if hasState, err = raft.HasExistingState(m.logStore, m.stableStore, m.snapshotStore); err != nil { + return hasState, fmt.Errorf("failed to check for existing state: %w", err) + } + return hasState, nil +} + +func (m *Metastore) createRaftDirs() (err error) { + m.walDir = filepath.Join(m.config.Raft.Dir, "wal") + if err = os.MkdirAll(m.walDir, 0755); err != nil { + return fmt.Errorf("WAL dir: %w", err) + } + if err = os.MkdirAll(m.config.Raft.Dir, 0755); err != nil { + return fmt.Errorf("snapshot directory: %w", err) + } + return nil +} + +func (m *Metastore) shutdownRaft() { + if m.raft != nil { + // If raft has been initialized, try to transfer leadership. + // Only after this we remove the leader health observer and + // shutdown the raft. + // There is a chance that client will still be trying to connect + // to this instance, therefore retrying is still required. + if err := m.raft.LeadershipTransfer().Error(); err != nil { + switch { + case errors.Is(err, raft.ErrNotLeader): + // Not a leader, nothing to do. + case strings.Contains(err.Error(), "cannot find peer"): + // It's likely that there's just one node in the cluster. + default: + _ = level.Error(m.logger).Log("msg", "failed to transfer leadership", "err", err) + } + } + m.leaderhealth.Deregister(m.raft, metastoreRaftLeaderHealthServiceName) + if err := m.raft.Shutdown().Error(); err != nil { + _ = level.Error(m.logger).Log("msg", "failed to shutdown raft", "err", err) + } + } + if m.transport != nil { + if err := m.transport.Close(); err != nil { + _ = level.Error(m.logger).Log("msg", "failed to close transport", "err", err) + } + } + if m.wal != nil { + if err := m.wal.Close(); err != nil { + _ = level.Error(m.logger).Log("msg", "failed to close WAL", "err", err) + } + } +} diff --git a/pkg/experiment/metastore/metastore_boltdb.go b/pkg/experiment/metastore/metastore_boltdb.go new file mode 100644 index 0000000000..4bd3d9c022 --- /dev/null +++ b/pkg/experiment/metastore/metastore_boltdb.go @@ -0,0 +1,288 @@ +package metastore + +import ( + "context" + "encoding/binary" + "fmt" + "io" + "os" + "path/filepath" + "runtime/pprof" + "time" + + "github.com/go-kit/log" + "github.com/go-kit/log/level" + "github.com/hashicorp/raft" + "go.etcd.io/bbolt" +) + +const ( + boltDBFileName = "metastore.boltdb" + boltDBSnapshotName = "metastore_snapshot.boltdb" + boltDBInitialMmapSize = 1 << 30 +) + +type boltdb struct { + logger log.Logger + boltdb *bbolt.DB + config Config + path string + metrics *metastoreMetrics +} + +type snapshot struct { + logger log.Logger + tx *bbolt.Tx + metrics *metastoreMetrics +} + +func newDB(config Config, logger log.Logger, metrics *metastoreMetrics) *boltdb { + return &boltdb{ + logger: logger, + config: config, + metrics: metrics, + } +} + +func (db *boltdb) open(readOnly bool) (err error) { + defer func() { + if err != nil { + // If the initialization fails, initialized components + // should be de-initialized gracefully. + db.shutdown() + } + }() + + if err = os.MkdirAll(db.config.DataDir, 0755); err != nil { + return fmt.Errorf("db dir: %w", err) + } + + if db.path == "" { + db.path = filepath.Join(db.config.DataDir, boltDBFileName) + } + + opts := *bbolt.DefaultOptions + opts.ReadOnly = readOnly + opts.NoSync = true + opts.InitialMmapSize = boltDBInitialMmapSize + if db.boltdb, err = bbolt.Open(db.path, 0644, &opts); err != nil { + return fmt.Errorf("failed to open db: %w", err) + } + + if !readOnly { + err = db.boltdb.Update(func(tx *bbolt.Tx) error { + _, err := tx.CreateBucketIfNotExists(blockMetadataBucketNameBytes) + if err != nil { + return err + } + _, err = tx.CreateBucketIfNotExists(compactionJobBucketNameBytes) + return err + }) + if err != nil { + return fmt.Errorf("failed to create bucket: %w", err) + } + } + + return nil +} + +func (db *boltdb) shutdown() { + if db.boltdb != nil { + if err := db.boltdb.Close(); err != nil { + _ = level.Error(db.logger).Log("msg", "failed to close database", "err", err) + } + } +} + +func (db *boltdb) restore(snapshot io.Reader) error { + t1 := time.Now() + defer func() { + db.metrics.boltDBRestoreSnapshotDuration.Observe(time.Since(t1).Seconds()) + }() + // Snapshot is a full copy of the database, therefore we copy + // it on disk and use it instead of the current database. + path, err := db.copySnapshot(snapshot) + if err == nil { + // First check the snapshot. + restored := *db + restored.path = path + err = restored.open(true) + // Also check applied index. + restored.shutdown() + } + if err != nil { + return fmt.Errorf("failed to restore snapshot: %w", err) + } + // Note that we do not keep the previous database: in case if the + // snapshot is corrupted, we should try another one. + return db.openSnapshot(path) +} + +func (db *boltdb) copySnapshot(snapshot io.Reader) (path string, err error) { + path = filepath.Join(db.config.DataDir, boltDBSnapshotName) + snapFile, err := os.Create(path) + if err != nil { + return "", err + } + _, err = io.Copy(snapFile, snapshot) + if syncErr := syncFD(snapFile); err == nil { + err = syncErr + } + return path, err +} + +func (db *boltdb) openSnapshot(path string) (err error) { + db.shutdown() + if err = os.Rename(path, db.path); err != nil { + return err + } + if err = syncPath(db.path); err != nil { + return err + } + return db.open(false) +} + +func syncPath(path string) (err error) { + d, err := os.Open(path) + if err != nil { + return err + } + return syncFD(d) +} + +func syncFD(f *os.File) (err error) { + err = f.Sync() + if closeErr := f.Close(); err == nil { + return closeErr + } + return err +} + +func (db *boltdb) createSnapshot() (*snapshot, error) { + s := snapshot{logger: db.logger, metrics: db.metrics} + tx, err := db.boltdb.Begin(false) + if err != nil { + return nil, fmt.Errorf("failed to open a transaction for snapshot: %w", err) + } + s.tx = tx + return &s, nil +} + +func (s *snapshot) Persist(sink raft.SnapshotSink) (err error) { + pprof.Do(context.Background(), pprof.Labels("metastore_op", "persist"), func(ctx context.Context) { + err = s.persist(sink) + }) + return err +} + +func (s *snapshot) persist(sink raft.SnapshotSink) error { + var err error + t1 := time.Now() + _ = s.logger.Log("msg", "persisting snapshot", "sink_id", sink.ID()) + defer func() { + s.metrics.boltDBPersistSnapshotDuration.Observe(time.Since(t1).Seconds()) + s.logger.Log("msg", "persisted snapshot", "sink_id", sink.ID(), "err", err, "duration", time.Since(t1)) + if err != nil { + _ = s.logger.Log("msg", "failed to persist snapshot", "err", err) + if err = sink.Cancel(); err != nil { + _ = s.logger.Log("msg", "failed to cancel snapshot sink", "err", err) + return + } + } + if err = sink.Close(); err != nil { + _ = s.logger.Log("msg", "failed to close sink", "err", err) + } + }() + _ = level.Info(s.logger).Log("msg", "persisting snapshot") + if _, err = s.tx.WriteTo(sink); err != nil { + _ = level.Error(s.logger).Log("msg", "failed to write snapshot", "err", err) + return err + } + return nil +} + +func (s *snapshot) Release() { + if s.tx != nil { + // This is an in-memory rollback, no error expected. + _ = s.tx.Rollback() + } +} + +func getOrCreateSubBucket(parent *bbolt.Bucket, name []byte) (*bbolt.Bucket, error) { + bucket := parent.Bucket(name) + if bucket == nil { + return parent.CreateBucket(name) + } + return bucket, nil +} + +const blockMetadataBucketName = "block_metadata" +const compactionJobBucketName = "compaction_job" + +var blockMetadataBucketNameBytes = []byte(blockMetadataBucketName) +var compactionJobBucketNameBytes = []byte(compactionJobBucketName) + +func getBlockMetadataBucket(tx *bbolt.Tx) (*bbolt.Bucket, error) { + mdb := tx.Bucket(blockMetadataBucketNameBytes) + if mdb == nil { + return nil, bbolt.ErrBucketNotFound + } + return mdb, nil +} + +func updateBlockMetadataBucket(tx *bbolt.Tx, name []byte, fn func(*bbolt.Bucket) error) error { + mdb, err := getBlockMetadataBucket(tx) + if err != nil { + return err + } + bucket, err := getOrCreateSubBucket(mdb, name) + if err != nil { + return err + } + return fn(bucket) +} + +// Bucket |Key +// [4:shard]|[block_id] +func keyForBlockMeta(shard uint32, tenant string, id string) (bucket, key []byte) { + k := make([]byte, 4+len(tenant)) + binary.BigEndian.PutUint32(k, shard) + copy(k[4:], tenant) + return k, []byte(id) +} + +func parseBucketName(b []byte) (shard uint32, tenant string, ok bool) { + if len(b) >= 4 { + return binary.BigEndian.Uint32(b), string(b[4:]), true + } + return 0, "", false +} + +func updateCompactionJobBucket(tx *bbolt.Tx, name []byte, fn func(*bbolt.Bucket) error) error { + cdb, err := getCompactionJobBucket(tx) + if err != nil { + return err + } + bucket, err := getOrCreateSubBucket(cdb, name) + if err != nil { + return err + } + return fn(bucket) +} + +// Bucket |Key +// [4:shard]|[job_name] +func keyForCompactionJob(shard uint32, tenant string, jobName string) (bucket, key []byte) { + bucket = make([]byte, 4+len(tenant)) + binary.BigEndian.PutUint32(bucket, shard) + copy(bucket[4:], tenant) + return bucket, []byte(jobName) +} + +func getCompactionJobBucket(tx *bbolt.Tx) (*bbolt.Bucket, error) { + cdb := tx.Bucket(compactionJobBucketNameBytes) + if cdb == nil { + return nil, bbolt.ErrBucketNotFound + } + return cdb, nil +} diff --git a/pkg/experiment/metastore/metastore_bootstrap.go b/pkg/experiment/metastore/metastore_bootstrap.go new file mode 100644 index 0000000000..00db8fe508 --- /dev/null +++ b/pkg/experiment/metastore/metastore_bootstrap.go @@ -0,0 +1,129 @@ +package metastore + +import ( + "context" + "errors" + "fmt" + "net" + "slices" + "strings" + "time" + + "github.com/go-kit/log" + "github.com/go-kit/log/level" + "github.com/grafana/dskit/dns" + "github.com/hashicorp/raft" +) + +func (m *Metastore) bootstrap() error { + peers, err := m.bootstrapPeers() + if err != nil { + return fmt.Errorf("failed to resolve peers: %w", err) + } + logger := log.With(m.logger, + "server_id", m.config.Raft.ServerID, + "advertise_address", m.config.Raft.AdvertiseAddress, + "peers", fmt.Sprint(peers)) + lastPeer := peers[len(peers)-1] + if raft.ServerAddress(m.config.Raft.AdvertiseAddress) != lastPeer.Address { + _ = level.Info(logger).Log("msg", "not the bootstrap node, skipping") + return nil + } + _ = level.Info(logger).Log("msg", "bootstrapping raft") + bootstrap := m.raft.BootstrapCluster(raft.Configuration{Servers: peers}) + if bootstrapErr := bootstrap.Error(); bootstrapErr != nil { + if !errors.Is(bootstrapErr, raft.ErrCantBootstrap) { + return fmt.Errorf("failed to bootstrap raft: %w", bootstrapErr) + } + } + return nil +} + +func (m *Metastore) bootstrapPeers() ([]raft.Server, error) { + // The peer list always includes the local node. + peers := make([]raft.Server, 0, len(m.config.Raft.BootstrapPeers)+1) + peers = append(peers, raft.Server{ + Suffrage: raft.Voter, + ID: raft.ServerID(m.config.Raft.ServerID), + Address: raft.ServerAddress(m.config.Raft.AdvertiseAddress), + }) + // Note that raft requires stable node IDs, therefore we're using + // the node FQDN:port for both purposes: as the identifier and as the + // address. This requires a DNS SRV record lookup without further + // resolution of A records (dnssrvnoa+). + // + // Alternatively, peers may be specified explicitly in the + // "{addr}" format, where the node is the optional node + // identifier. + var resolve []string + for _, peer := range m.config.Raft.BootstrapPeers { + if strings.Contains(peer, "+") { + resolve = append(resolve, peer) + } else { + peers = append(peers, parsePeer(peer)) + } + } + if len(resolve) > 0 { + ctx, cancel := context.WithTimeout(context.Background(), 10*time.Second) + defer cancel() + prov := dns.NewProvider(m.logger, m.reg, dns.MiekgdnsResolverType) + if err := prov.Resolve(ctx, resolve); err != nil { + return nil, fmt.Errorf("failed to resolve bootstrap peers: %w", err) + } + resolvedPeers := prov.Addresses() + if len(resolvedPeers) == 0 { + // The local node is the only one in the cluster, but peers + // were supposed to be present. Stop here to avoid bootstrapping + // a single-node cluster. + return nil, fmt.Errorf("bootstrap peers can't be resolved") + } + for _, peer := range resolvedPeers { + peers = append(peers, raft.Server{ + Suffrage: raft.Voter, + ID: raft.ServerID(peer), + Address: raft.ServerAddress(peer), + }) + } + } + // Finally, we sort and deduplicate the peers: the first one + // is to boostrap the cluster. If there are nodes with distinct + // IDs but the same address, bootstrapping will fail. + slices.SortFunc(peers, func(a, b raft.Server) int { + return strings.Compare(string(a.ID), string(b.ID)) + }) + peers = slices.CompactFunc(peers, func(a, b raft.Server) bool { + return a.ID == b.ID + }) + if len(peers) != m.config.Raft.BootstrapExpectPeers { + return nil, fmt.Errorf("expected number of bootstrap peers not reached: got %d, expected %d", + len(peers), m.config.Raft.BootstrapExpectPeers) + } + return peers, nil +} + +func parsePeer(raw string) raft.Server { + // The string may be "{addr}" or "{addr}/{node_id}". + parts := strings.SplitN(raw, "/", 2) + var addr string + var node string + if len(parts) == 2 { + addr = parts[0] + node = parts[1] + } else { + addr = raw + } + host, _, err := net.SplitHostPort(addr) + if err != nil { + // No port specified. + host = addr + } + if node == "" { + // No node_id specified. + node = host + } + return raft.Server{ + Suffrage: raft.Voter, + ID: raft.ServerID(node), + Address: raft.ServerAddress(addr), + } +} diff --git a/pkg/experiment/metastore/metastore_compaction_planner.go b/pkg/experiment/metastore/metastore_compaction_planner.go new file mode 100644 index 0000000000..5ea280f19a --- /dev/null +++ b/pkg/experiment/metastore/metastore_compaction_planner.go @@ -0,0 +1,258 @@ +package metastore + +import ( + "context" + "fmt" + "sync" + "time" + + "github.com/cespare/xxhash/v2" + "github.com/go-kit/log/level" + "github.com/prometheus/client_golang/prometheus" + "go.etcd.io/bbolt" + + compactorv1 "github.com/grafana/pyroscope/api/gen/proto/go/compactor/v1" + metastorev1 "github.com/grafana/pyroscope/api/gen/proto/go/metastore/v1" + "github.com/grafana/pyroscope/pkg/experiment/metastore/compactionpb" +) + +const ( + jobPollInterval = 5 * time.Second + jobLeaseDuration = 3 * jobPollInterval +) + +var ( + // TODO aleks: for illustration purposes, to be moved externally + globalCompactionStrategy = compactionStrategy{ + levels: map[uint32]compactionLevelStrategy{ + 0: {maxBlocks: 20}, + }, + defaultStrategy: compactionLevelStrategy{ + maxBlocks: 10, + }, + maxCompactionLevel: 3, + // 0: 0.5 + // 1: 10s + // 2: 100s + // 3: 1000s // 16m40s + } +) + +type compactionStrategy struct { + levels map[uint32]compactionLevelStrategy + defaultStrategy compactionLevelStrategy + maxCompactionLevel uint32 +} + +type compactionLevelStrategy struct { + maxBlocks int + maxTotalSizeBytes uint64 +} + +func getStrategyForLevel(compactionLevel uint32) compactionLevelStrategy { + strategy, ok := globalCompactionStrategy.levels[compactionLevel] + if !ok { + strategy = globalCompactionStrategy.defaultStrategy + } + return strategy +} + +func (s compactionLevelStrategy) shouldCreateJob(blocks []string) bool { + // NB: Total block size does not reflect the actual size of the data + // to be read for compaction (at once) or queried. A better heuristic + // would be max tenant service size. + return len(blocks) >= s.maxBlocks +} + +type jobPreQueue struct { + mu sync.Mutex + blocksByLevel map[uint32][]string +} + +func (m *Metastore) GetCompactionJobs(_ context.Context, req *compactorv1.GetCompactionRequest) (*compactorv1.GetCompactionResponse, error) { + return nil, nil +} + +func (m *metastoreState) tryCreateJob(block *metastorev1.BlockMeta, raftLogIndex uint64) *compactionpb.CompactionJob { + key := tenantShard{ + tenant: block.TenantId, + shard: block.Shard, + } + preQueue := m.getOrCreatePreQueue(key) + preQueue.mu.Lock() + defer preQueue.mu.Unlock() + + if block.CompactionLevel >= globalCompactionStrategy.maxCompactionLevel { + level.Info(m.logger).Log("msg", "skipping block at max compaction level", "block", block.Id, "compaction_level", block.CompactionLevel) + return nil + } + + queuedBlocks := append(preQueue.blocksByLevel[block.CompactionLevel], block.Id) + + level.Debug(m.logger).Log( + "msg", "adding block for compaction", + "block", block.Id, + "shard", block.Shard, + "tenant", block.TenantId, + "compaction_level", block.CompactionLevel, + "size", block.Size, + "queue_size", len(queuedBlocks), + "raft_log_index", raftLogIndex) + + strategy := getStrategyForLevel(block.CompactionLevel) + + var job *compactionpb.CompactionJob + if strategy.shouldCreateJob(queuedBlocks) { + blockIds := make([]string, 0, len(queuedBlocks)) + for _, b := range queuedBlocks { + blockIds = append(blockIds, b) + } + job = &compactionpb.CompactionJob{ + Name: fmt.Sprintf("L%d-S%d-%d", block.CompactionLevel, block.Shard, calculateHash(queuedBlocks)), + Blocks: blockIds, + Status: compactionpb.CompactionStatus_COMPACTION_STATUS_UNSPECIFIED, + Shard: block.Shard, + TenantId: block.TenantId, + CompactionLevel: block.CompactionLevel, + } + level.Info(m.logger).Log( + "msg", "created compaction job", + "job", job.Name, + "blocks", len(queuedBlocks), + "shard", block.Shard, + "tenant", block.TenantId, + "compaction_level", block.CompactionLevel) + } + return job +} + +func (m *metastoreState) addCompactionJob(job *compactionpb.CompactionJob) { + level.Debug(m.logger).Log("msg", "adding compaction job to priority queue", "job", job.Name) + if ok := m.compactionJobQueue.enqueue(job); !ok { + level.Warn(m.logger).Log("msg", "a compaction job with this name already exists", "job", job.Name) + return + } + + // reset the pre-queue for this level + key := tenantShard{ + tenant: job.TenantId, + shard: job.Shard, + } + preQueue := m.getOrCreatePreQueue(key) + preQueue.mu.Lock() + defer preQueue.mu.Unlock() + preQueue.blocksByLevel[job.CompactionLevel] = preQueue.blocksByLevel[job.CompactionLevel][:0] +} + +func (m *metastoreState) addBlockToCompactionJobQueue(block *metastorev1.BlockMeta) { + key := tenantShard{ + tenant: block.TenantId, + shard: block.Shard, + } + preQueue := m.getOrCreatePreQueue(key) + preQueue.mu.Lock() + defer preQueue.mu.Unlock() + + preQueue.blocksByLevel[block.CompactionLevel] = append(preQueue.blocksByLevel[block.CompactionLevel], block.Id) +} + +func calculateHash(blocks []string) uint64 { + b := make([]byte, 0, 1024) + for _, blk := range blocks { + b = append(b, blk...) + } + return xxhash.Sum64(b) +} + +type compactionMetrics struct { + addedBlocks *prometheus.CounterVec + deletedBlocks *prometheus.CounterVec + addedJobs *prometheus.CounterVec + assignedJobs *prometheus.CounterVec + completedJobs *prometheus.CounterVec +} + +func newCompactionMetrics(reg prometheus.Registerer) *compactionMetrics { + m := &compactionMetrics{ + addedBlocks: prometheus.NewCounterVec(prometheus.CounterOpts{ + Namespace: "pyroscope", + Name: "metastore_compaction_added_blocks_count", + Help: "The number of blocks added for compaction", + }, []string{"shard", "tenant", "level"}), + deletedBlocks: prometheus.NewCounterVec(prometheus.CounterOpts{ + Namespace: "pyroscope", + Name: "metastore_compaction_deleted_blocks_count", + Help: "The number of blocks deleted as a result of compaction", + }, []string{"shard", "tenant", "level"}), + addedJobs: prometheus.NewCounterVec(prometheus.CounterOpts{ + Namespace: "pyroscope", + Name: "metastore_compaction_added_jobs_count", + Help: "The number of created compaction jobs", + }, []string{"shard", "tenant", "level"}), + assignedJobs: prometheus.NewCounterVec(prometheus.CounterOpts{ + Namespace: "pyroscope", + Name: "metastore_compaction_assigned_jobs_count", + Help: "The number of assigned compaction jobs", + }, []string{"shard", "tenant", "level"}), + completedJobs: prometheus.NewCounterVec(prometheus.CounterOpts{ + Namespace: "pyroscope", + Name: "metastore_compaction_completed_jobs_count", + Help: "The number of completed compaction jobs", + }, []string{"shard", "tenant", "level"}), + } + if reg != nil { + reg.MustRegister( + m.addedBlocks, + m.deletedBlocks, + m.addedJobs, + m.assignedJobs, + m.completedJobs, + ) + } + return m +} + +func (m *metastoreState) consumeBlock(block *metastorev1.BlockMeta, tx *bbolt.Tx, raftLogIndex uint64) (err error, jobToAdd *compactionpb.CompactionJob, blockForQueue *metastorev1.BlockMeta) { + // create and store an optional compaction job + if job := m.tryCreateJob(block, raftLogIndex); job != nil { + level.Debug(m.logger).Log("msg", "persisting compaction job", "job", job.Name) + jobBucketName, jobKey := keyForCompactionJob(block.Shard, block.TenantId, job.Name) + err := updateCompactionJobBucket(tx, jobBucketName, func(bucket *bbolt.Bucket) error { + data, _ := job.MarshalVT() + return bucket.Put(jobKey, data) + }) + if err != nil { + return err, nil, nil + } + err = m.persistJobPreQueue(block.Shard, block.TenantId, block.CompactionLevel, []string{}, tx) + jobToAdd = job + } else { + key := tenantShard{ + tenant: block.TenantId, + shard: block.Shard, + } + queue := m.getOrCreatePreQueue(key).blocksByLevel[block.CompactionLevel] + queue = append(queue, block.Id) + err := m.persistJobPreQueue(block.Shard, block.TenantId, block.CompactionLevel, queue, tx) + if err != nil { + return err, nil, nil + } + blockForQueue = block + } + return err, jobToAdd, blockForQueue +} + +func (m *metastoreState) persistJobPreQueue(shard uint32, tenant string, compactionLevel uint32, queue []string, tx *bbolt.Tx) error { + jobBucketName, _ := keyForCompactionJob(shard, tenant, "") + preQueue := &compactionpb.JobPreQueue{ + CompactionLevel: compactionLevel, + Shard: shard, + Tenant: tenant, + Blocks: queue, + } + key := []byte(fmt.Sprintf("job-pre-queue-%d", compactionLevel)) + return updateCompactionJobBucket(tx, jobBucketName, func(bucket *bbolt.Bucket) error { + data, _ := preQueue.MarshalVT() + return bucket.Put(key, data) + }) +} diff --git a/pkg/experiment/metastore/metastore_compaction_queue.go b/pkg/experiment/metastore/metastore_compaction_queue.go new file mode 100644 index 0000000000..ceb0b5ce21 --- /dev/null +++ b/pkg/experiment/metastore/metastore_compaction_queue.go @@ -0,0 +1,210 @@ +package metastore + +import ( + "container/heap" + "slices" + "strings" + "sync" + + "github.com/grafana/pyroscope/pkg/experiment/metastore/compactionpb" +) + +// A priority queue for compaction jobs. Jobs are prioritized by the compaction +// level, and the deadline time. +// +// The queue is supposed to be used by the compaction planner to schedule jobs. +// +// Compaction workers own jobs while they are in progress. Ownership handling is +// implemented using lease deadlines and fencing tokens: +// https://martin.kleppmann.com/2016/02/08/how-to-do-distributed-locking.html + +type jobQueue struct { + mu sync.Mutex + jobs map[string]*jobQueueEntry + pq priorityQueue + + lease int64 +} + +// newJobQueue creates a new job queue with the given lease duration. +// +// Typically, callers should update jobs at the interval not exceeding +// the half of the lease duration. +func newJobQueue(lease int64) *jobQueue { + pq := make(priorityQueue, 0) + heap.Init(&pq) + return &jobQueue{ + jobs: make(map[string]*jobQueueEntry), + pq: pq, + lease: lease, + } +} + +type jobQueueEntry struct { + // The index of the job in the heap. + index int + // The original proto message. + *compactionpb.CompactionJob +} + +func (c *jobQueueEntry) less(x *jobQueueEntry) bool { + if c.Status != x.Status { + // Peek jobs in the "initial" (unspecified) state first. + return c.Status < x.Status + } + if c.LeaseExpiresAt != x.LeaseExpiresAt { + // Jobs with earlier deadlines should be at the top. + return c.LeaseExpiresAt < x.LeaseExpiresAt + } + // Compact lower level jobs first. + if c.CompactionLevel != x.CompactionLevel { + // Jobs with earlier deadlines should be at the top. + return c.CompactionLevel < x.CompactionLevel + } + return c.Name < x.Name +} + +func (q *jobQueue) dequeue(now int64, raftLogIndex uint64) *compactionpb.CompactionJob { + q.mu.Lock() + defer q.mu.Unlock() + for q.pq.Len() > 0 { + job := q.pq[0] + if job.Status == compactionpb.CompactionStatus_COMPACTION_STATUS_IN_PROGRESS && + now <= job.LeaseExpiresAt { + // If the top job is in progress and not expired, stop checking further + return nil + } + // Actually remove it from the heap, update and push it back. + heap.Pop(&q.pq) + job.LeaseExpiresAt = q.getNewDeadline(now) + job.Status = compactionpb.CompactionStatus_COMPACTION_STATUS_IN_PROGRESS + // If job.status is "in progress", the ownership of the job is being revoked. + job.RaftLogIndex = raftLogIndex + heap.Push(&q.pq, job) + return job.CompactionJob + } + return nil +} + +func (q *jobQueue) update(name string, now int64, raftLogIndex uint64) bool { + q.mu.Lock() + defer q.mu.Unlock() + if job, exists := q.jobs[name]; exists { + if job.RaftLogIndex > raftLogIndex { + return false + } + job.LeaseExpiresAt = q.getNewDeadline(now) + job.Status = compactionpb.CompactionStatus_COMPACTION_STATUS_IN_PROGRESS + // De-prioritize the job, as the deadline has been postponed. + heap.Fix(&q.pq, job.index) + return true + } + return false +} + +func (q *jobQueue) getNewDeadline(now int64) int64 { + return now + q.lease +} + +func (q *jobQueue) isOwner(name string, raftLogIndex uint64) bool { + q.mu.Lock() + defer q.mu.Unlock() + if job, exists := q.jobs[name]; exists { + if job.RaftLogIndex > raftLogIndex { + return false + } + } + return true +} + +func (q *jobQueue) evict(name string, raftLogIndex uint64) bool { + q.mu.Lock() + defer q.mu.Unlock() + if job, exists := q.jobs[name]; exists { + if job.RaftLogIndex > raftLogIndex { + return false + } + delete(q.jobs, name) + heap.Remove(&q.pq, job.index) + } + return true +} + +func (q *jobQueue) enqueue(job *compactionpb.CompactionJob) bool { + q.mu.Lock() + defer q.mu.Unlock() + if _, exists := q.jobs[job.Name]; exists { + return false + } + j := &jobQueueEntry{CompactionJob: job} + q.jobs[job.Name] = j + heap.Push(&q.pq, j) + return true +} + +func (q *jobQueue) putJob(job *compactionpb.CompactionJob) { + q.jobs[job.Name] = &jobQueueEntry{CompactionJob: job} +} + +func (q *jobQueue) rebuild() { + q.pq = slices.Grow(q.pq[0:], len(q.jobs)) + for _, job := range q.jobs { + q.pq = append(q.pq, job) + } + heap.Init(&q.pq) +} + +func (q *jobQueue) stats() (int, string, string, string, string) { + q.mu.Lock() + defer q.mu.Unlock() + + newJobs := make([]string, 0) + inProgressJobs := make([]string, 0) + completedJobs := make([]string, 0) + failedJobs := make([]string, 0) + for _, job := range q.jobs { + switch job.Status { + case compactionpb.CompactionStatus_COMPACTION_STATUS_UNSPECIFIED: + newJobs = append(newJobs, job.Name) + case compactionpb.CompactionStatus_COMPACTION_STATUS_IN_PROGRESS: + inProgressJobs = append(inProgressJobs, job.Name) + case compactionpb.CompactionStatus_COMPACTION_STATUS_SUCCESS: + completedJobs = append(completedJobs, job.Name) + case compactionpb.CompactionStatus_COMPACTION_STATUS_FAILURE: + failedJobs = append(failedJobs, job.Name) + } + } + return len(q.jobs), strings.Join(newJobs, ", "), strings.Join(inProgressJobs, ", "), strings.Join(completedJobs, ", "), strings.Join(failedJobs, ", ") +} + +// TODO(kolesnikovae): container/heap is not very efficient, +// consider implementing own heap, specific to the case. + +type priorityQueue []*jobQueueEntry + +func (pq priorityQueue) Len() int { return len(pq) } + +func (pq priorityQueue) Less(i, j int) bool { return pq[i].less(pq[j]) } + +func (pq priorityQueue) Swap(i, j int) { + pq[i], pq[j] = pq[j], pq[i] + pq[i].index = i + pq[j].index = j +} + +func (pq *priorityQueue) Push(x interface{}) { + n := len(*pq) + job := x.(*jobQueueEntry) + job.index = n + *pq = append(*pq, job) +} + +func (pq *priorityQueue) Pop() interface{} { + old := *pq + n := len(old) + job := old[n-1] + old[n-1] = nil + job.index = -1 + *pq = old[0 : n-1] + return job +} diff --git a/pkg/experiment/metastore/metastore_compaction_queue_test.go b/pkg/experiment/metastore/metastore_compaction_queue_test.go new file mode 100644 index 0000000000..09f1af46dd --- /dev/null +++ b/pkg/experiment/metastore/metastore_compaction_queue_test.go @@ -0,0 +1,71 @@ +package metastore + +import ( + "testing" + + "github.com/stretchr/testify/assert" + "github.com/stretchr/testify/require" + + "github.com/grafana/pyroscope/pkg/experiment/metastore/compactionpb" +) + +func Test_compactionJobQueue(t *testing.T) { + var now int64 // Timestamp of the raft command. + lease := int64(10) // Job lease duration. + q := newJobQueue(lease) + + assert.True(t, q.enqueue(&compactionpb.CompactionJob{ + Name: "job1", + RaftLogIndex: 1, + CompactionLevel: 0, + })) + assert.True(t, q.enqueue(&compactionpb.CompactionJob{ + Name: "job2", + RaftLogIndex: 2, + CompactionLevel: 1, + })) + assert.True(t, q.enqueue(&compactionpb.CompactionJob{ + Name: "job3", + RaftLogIndex: 3, + CompactionLevel: 0, + })) + + // Token here is the raft command index. + assertJob(t, q.dequeue(now, 4), "job1", 4) // L0 + assertJob(t, q.dequeue(now, 5), "job3", 5) // L0 + assertJob(t, q.dequeue(now, 6), "job2", 6) // L1 + require.Nil(t, q.dequeue(now, 7)) // No jobs left. + require.Nil(t, q.dequeue(now, 8)) // No jobs left. + + // Time has passed. Updating the jobs: all but job1. + now += lease / 2 + assert.True(t, q.update("job3", now, 9)) // Postpone the deadline. + assert.True(t, q.update("job2", now, 10)) // Postpone the deadline. + require.Nil(t, q.dequeue(now, 11)) // No jobs left. + + // Time has passed: the initial lease has expired. + now += lease/2 + 1 + assertJob(t, q.dequeue(now, 12), "job1", 12) // Seizing ownership of expired job. + require.Nil(t, q.dequeue(now, 13)) // No jobs available yet. + + // Owner of the job1 awakes and tries to update the job. + assert.False(t, q.update("job1", now, 4)) // Postpone the deadline; stale owner is rejected. + assert.True(t, q.update("job1", now, 12)) // Postpone the deadline; new owner succeeds. + + assert.False(t, q.evict("job1", 4)) // Evicting the job; stale owner is rejected. + assert.True(t, q.evict("job1", 12)) // Postpone the deadline; new owner succeeds. + + // Jobs are evicted in the end, regardless of the status. + // We ignore expired lease, as long as nobody else has taken the job. + assert.True(t, q.evict("job2", 10)) + assert.True(t, q.evict("job3", 9)) + + // No jobs left. + require.Nil(t, q.dequeue(now, 14)) +} + +func assertJob(t *testing.T, j *compactionpb.CompactionJob, name string, commitIndex uint64) { + require.NotNil(t, j) + assert.Equal(t, name, j.Name) + assert.Equal(t, commitIndex, j.RaftLogIndex) +} diff --git a/pkg/experiment/metastore/metastore_fsm.go b/pkg/experiment/metastore/metastore_fsm.go new file mode 100644 index 0000000000..5d7c2caba5 --- /dev/null +++ b/pkg/experiment/metastore/metastore_fsm.go @@ -0,0 +1,218 @@ +package metastore + +import ( + "fmt" + "io" + "reflect" + "time" + + "github.com/go-kit/log" + "github.com/go-kit/log/level" + "github.com/hashicorp/raft" + "google.golang.org/protobuf/proto" + + compactorv1 "github.com/grafana/pyroscope/api/gen/proto/go/compactor/v1" + metastorev1 "github.com/grafana/pyroscope/api/gen/proto/go/metastore/v1" + "github.com/grafana/pyroscope/pkg/experiment/metastore/raftlogpb" + "github.com/grafana/pyroscope/pkg/util" +) + +// The map is used to determine the type of the given command, +// when the request is converted to a Raft log entry. +var commandTypeMap = map[reflect.Type]raftlogpb.CommandType{ + reflect.TypeOf(new(metastorev1.AddBlockRequest)): raftlogpb.CommandType_COMMAND_TYPE_ADD_BLOCK, + reflect.TypeOf(new(raftlogpb.TruncateCommand)): raftlogpb.CommandType_COMMAND_TYPE_TRUNCATE, + reflect.TypeOf(new(compactorv1.PollCompactionJobsRequest)): raftlogpb.CommandType_COMMAND_TYPE_POLL_COMPACTION_JOBS_STATUS, +} + +// The map is used to determine the handler for the given command, +// read from the Raft log entry. +var commandHandlers = map[raftlogpb.CommandType]commandHandler{ + raftlogpb.CommandType_COMMAND_TYPE_ADD_BLOCK: func(fsm *FSM, cmd *raft.Log, raw []byte) fsmResponse { + return handleCommand(raw, cmd, fsm.state.applyAddBlock) + }, + raftlogpb.CommandType_COMMAND_TYPE_TRUNCATE: func(fsm *FSM, cmd *raft.Log, raw []byte) fsmResponse { + return handleCommand(raw, cmd, fsm.state.applyTruncate) + }, + raftlogpb.CommandType_COMMAND_TYPE_POLL_COMPACTION_JOBS_STATUS: func(fsm *FSM, cmd *raft.Log, raw []byte) fsmResponse { + return handleCommand(raw, cmd, fsm.state.applyPollCompactionJobs) + }, +} + +// TODO: Add registration functions. + +type FSM struct { + logger log.Logger + state *metastoreState + db *boltdb +} + +type fsmResponse struct { + msg proto.Message + err error +} + +type fsmError struct { + log *raft.Log + err error +} + +func errResponse(l *raft.Log, err error) fsmResponse { + return fsmResponse{err: &fsmError{log: l, err: err}} +} + +func (e *fsmError) Error() string { + if e.err == nil { + return "" + } + if e.log == nil { + return e.err.Error() + } + return fmt.Sprintf("term: %d; index: %d; appended_at: %v; error: %v", + e.log.Index, e.log.Term, e.log.AppendedAt, e.err) +} + +type commandHandler func(*FSM, *raft.Log, []byte) fsmResponse + +// TODO(kolesnikovae): replace commandCall with interface: +// type command[Req, Resp proto.Message] interface { +// apply(Req) (Resp, error) +// } + +type commandCall[Req, Resp proto.Message] func(*raft.Log, Req) (Resp, error) + +func newFSM(logger log.Logger, db *boltdb, state *metastoreState) *FSM { + return &FSM{ + logger: logger, + db: db, + state: state, + } +} + +// TODO(kolesnikovae): Implement BatchingFSM. + +func (fsm *FSM) Apply(l *raft.Log) interface{} { + switch l.Type { + case raft.LogNoop: + case raft.LogBarrier: + case raft.LogConfiguration: + case raft.LogCommand: + return fsm.applyCommand(l) + default: + _ = level.Warn(fsm.logger).Log("msg", "unexpected log entry, ignoring", "type", l.Type.String()) + } + return nil +} + +// applyCommand receives raw command from the raft log (FSM.Apply), +// and calls the corresponding handler on the _local_ FSM, based on +// the command type. +func (fsm *FSM) applyCommand(l *raft.Log) interface{} { + t1 := time.Now() + defer func() { + fsm.db.metrics.fsmApplyCommandHandlerDuration.Observe(time.Since(t1).Seconds()) + }() + var e raftlogpb.RaftLogEntry + if err := proto.Unmarshal(l.Data, &e); err != nil { + return errResponse(l, err) + } + if handler, ok := commandHandlers[e.Type]; ok { + return handler(fsm, l, e.Payload) + } + return errResponse(l, fmt.Errorf("unknown command type: %v", e.Type.String())) +} + +// handleCommand receives payload of the command from the raft log (FSM.Apply), +// and the function that processes the command. Returned response is wrapped in +// fsmResponse and is available to the FSM.Apply caller. +func handleCommand[Req, Resp proto.Message](raw []byte, cmd *raft.Log, call commandCall[Req, Resp]) fsmResponse { + var resp fsmResponse + defer func() { + if r := recover(); r != nil { + resp.err = util.PanicError(r) + } + }() + req := newProto[Req]() + if resp.err = proto.Unmarshal(raw, req); resp.err != nil { + return resp + } + resp.msg, resp.err = call(cmd, req) + return resp +} + +func newProto[T proto.Message]() T { + var msg T + msgType := reflect.TypeOf(msg).Elem() + return reflect.New(msgType).Interface().(T) +} + +func (fsm *FSM) Snapshot() (raft.FSMSnapshot, error) { + // Snapshot should only capture a pointer to the state, and any + // expensive IO should happen as part of FSMSnapshot.Persist. + return fsm.db.createSnapshot() +} + +func (fsm *FSM) Restore(snapshot io.ReadCloser) error { + t1 := time.Now() + _ = level.Info(fsm.logger).Log("msg", "restoring snapshot") + defer func() { + _ = snapshot.Close() + fsm.db.metrics.fsmRestoreSnapshotDuration.Observe(time.Since(t1).Seconds()) + }() + if err := fsm.db.restore(snapshot); err != nil { + return fmt.Errorf("failed to restore from snapshot: %w", err) + } + if err := fsm.state.restore(fsm.db); err != nil { + return fmt.Errorf("failed to restore state: %w", err) + } + return nil +} + +// applyCommand issues the command to the raft log based on the request type, +// and returns the response of FSM.Apply. +func applyCommand[Req, Resp proto.Message]( + log *raft.Raft, + req Req, + timeout time.Duration, +) ( + future raft.ApplyFuture, + resp Resp, + err error, +) { + defer func() { + if r := recover(); r != nil { + err = util.PanicError(r) + } + }() + raw, err := marshallRequest(req) + if err != nil { + return nil, resp, err + } + future = log.Apply(raw, timeout) + if err = future.Error(); err != nil { + return nil, resp, err + } + fsmResp := future.Response().(fsmResponse) + if fsmResp.msg != nil { + resp, _ = fsmResp.msg.(Resp) + } + return future, resp, fsmResp.err +} + +func marshallRequest[Req proto.Message](req Req) ([]byte, error) { + cmdType, ok := commandTypeMap[reflect.TypeOf(req)] + if !ok { + return nil, fmt.Errorf("unknown command type: %T", req) + } + var err error + entry := raftlogpb.RaftLogEntry{Type: cmdType} + entry.Payload, err = proto.Marshal(req) + if err != nil { + return nil, err + } + raw, err := proto.Marshal(&entry) + if err != nil { + return nil, err + } + return raw, nil +} diff --git a/pkg/experiment/metastore/metastore_hack.go b/pkg/experiment/metastore/metastore_hack.go new file mode 100644 index 0000000000..8ad66d3efb --- /dev/null +++ b/pkg/experiment/metastore/metastore_hack.go @@ -0,0 +1,100 @@ +package metastore + +import ( + "sync" + "time" + + "github.com/go-kit/log" + "github.com/go-kit/log/level" + "github.com/hashicorp/raft" + "github.com/oklog/ulid" + "google.golang.org/protobuf/types/known/anypb" + + "github.com/grafana/pyroscope/pkg/experiment/metastore/raftlogpb" +) + +// FIXME(kolesnikovae): +// Remove once compaction is implemented. +// Or use index instead of the timestamp. + +func (m *Metastore) cleanupLoop() { + t := time.NewTicker(10 * time.Minute) + defer func() { + t.Stop() + m.wg.Done() + }() + for { + select { + case <-m.done: + return + case <-t.C: + if m.raft.State() != raft.Leader { + continue + } + timestamp := uint64(time.Now().Add(-12 * time.Hour).UnixMilli()) + req := &raftlogpb.TruncateCommand{Timestamp: timestamp} + _, _, err := applyCommand[*raftlogpb.TruncateCommand, *anypb.Any](m.raft, req, m.config.Raft.ApplyTimeout) + if err != nil { + _ = level.Error(m.logger).Log("msg", "failed to apply truncate command", "err", err) + } + } + } +} + +func (m *metastoreState) applyTruncate(_ *raft.Log, request *raftlogpb.TruncateCommand) (*anypb.Any, error) { + m.shardsMutex.Lock() + var g sync.WaitGroup + g.Add(len(m.shards)) + for shardID, shard := range m.shards { + go truncateSegmentsBefore(m.db, m.logger, &g, shardID, shard, request.Timestamp) + } + m.shardsMutex.Unlock() + g.Wait() + return &anypb.Any{}, nil +} + +func truncateSegmentsBefore( + db *boltdb, + log log.Logger, + wg *sync.WaitGroup, + shardID uint32, + shard *metastoreShard, + t uint64, +) { + defer wg.Done() + var c int + tx, err := db.boltdb.Begin(true) + if err != nil { + _ = level.Error(log).Log("msg", "failed to start transaction", "err", err) + return + } + defer func() { + if err = tx.Commit(); err != nil { + _ = level.Error(log).Log("msg", "failed to commit transaction", "err", err) + return + } + _ = level.Info(log).Log("msg", "stale segments truncated", "segments", c) + }() + + bucket, err := getBlockMetadataBucket(tx) + if err != nil { + _ = level.Error(log).Log("msg", "failed to get metadata bucket", "err", err) + return + } + shardBucket, _ := keyForBlockMeta(shardID, "", "") + bucket = bucket.Bucket(shardBucket) + + shard.segmentsMutex.Lock() + defer shard.segmentsMutex.Unlock() + + for k, segment := range shard.segments { + if ulid.MustParse(segment.Id).Time() < t { + if err = bucket.Delete([]byte(segment.Id)); err != nil { + _ = level.Error(log).Log("msg", "failed to delete stale segments", "err", err) + return + } + delete(shard.segments, k) + c++ + } + } +} diff --git a/pkg/experiment/metastore/metastore_metrics.go b/pkg/experiment/metastore/metastore_metrics.go new file mode 100644 index 0000000000..e0f404a1b0 --- /dev/null +++ b/pkg/experiment/metastore/metastore_metrics.go @@ -0,0 +1,54 @@ +package metastore + +import ( + "github.com/grafana/dskit/instrument" + "github.com/prometheus/client_golang/prometheus" +) + +type metastoreMetrics struct { + boltDBPersistSnapshotDuration prometheus.Histogram + boltDBRestoreSnapshotDuration prometheus.Histogram + fsmRestoreSnapshotDuration prometheus.Histogram + fsmApplyCommandHandlerDuration prometheus.Histogram + raftAddBlockDuration prometheus.Histogram +} + +func newMetastoreMetrics(reg prometheus.Registerer) *metastoreMetrics { + var dataTimingBuckets = prometheus.ExponentialBucketsRange(0.01, 20, 48) + m := &metastoreMetrics{ + boltDBPersistSnapshotDuration: prometheus.NewHistogram(prometheus.HistogramOpts{ + Namespace: "pyroscope", + Name: "metastore_boltdb_persist_snapshot_duration_seconds", + //Buckets: dataTimingBuckets, + Buckets: instrument.DefBuckets, + }), + boltDBRestoreSnapshotDuration: prometheus.NewHistogram(prometheus.HistogramOpts{ + Namespace: "pyroscope", + Name: "metastore_boltdb_restore_snapshot_duration_seconds", + Buckets: dataTimingBuckets, + }), + fsmRestoreSnapshotDuration: prometheus.NewHistogram(prometheus.HistogramOpts{ + Namespace: "pyroscope", + Name: "metastore_fsm_restore_snapshot_duration_seconds", + Buckets: dataTimingBuckets, + }), + fsmApplyCommandHandlerDuration: prometheus.NewHistogram(prometheus.HistogramOpts{ + Namespace: "pyroscope", + Name: "metastore_fsm_apply_command_handler_duration_seconds", + Buckets: dataTimingBuckets, + }), + raftAddBlockDuration: prometheus.NewHistogram(prometheus.HistogramOpts{ + Namespace: "pyroscope", + Name: "metastore_raft_add_block_duration_seconds", + Buckets: dataTimingBuckets, + }), + } + if reg != nil { + reg.MustRegister(m.boltDBPersistSnapshotDuration) + reg.MustRegister(m.boltDBRestoreSnapshotDuration) + reg.MustRegister(m.fsmRestoreSnapshotDuration) + reg.MustRegister(m.fsmApplyCommandHandlerDuration) + reg.MustRegister(m.raftAddBlockDuration) + } + return m +} diff --git a/pkg/experiment/metastore/metastore_readindex.go b/pkg/experiment/metastore/metastore_readindex.go new file mode 100644 index 0000000000..d031eadf56 --- /dev/null +++ b/pkg/experiment/metastore/metastore_readindex.go @@ -0,0 +1,132 @@ +package metastore + +import ( + "context" + "fmt" + "time" + + "github.com/go-kit/log" + "github.com/google/uuid" + + metastorev1 "github.com/grafana/pyroscope/api/gen/proto/go/metastore/v1" +) + +var tcheckFreq = 10 * time.Millisecond + +func (m *Metastore) ReadIndex(ctx context.Context, req *metastorev1.ReadIndexRequest) (*metastorev1.ReadIndexResponse, error) { + //todo + //If the leader has not yet marked an entry from its current term committed, it waits until it + //has done so. The Leader Completeness Property guarantees that a leader has all committed + //entries, but at the start of its term, it may not know which those are. To find out, it needs to + //commit an entry from its term. Raft handles this by having each leader commit a blank no-op + //entry into the log at the start of its term. As soon as this no-op entry is committed, the leader’s + //commit index will be at least as large as any other servers’ during its term. + t := time.Now() + readIndex := m.raft.CommitIndex() + raftLogger := func() log.Logger { + return log.With(m.logger, "component", "raft_debug", + "request_id", req.DebugRequestId, + "op", "ReadIndex", + "read_index", readIndex, + "applied_index", m.raft.AppliedIndex(), + "commit_index", m.raft.CommitIndex(), + "last_index", m.raft.LastIndex(), + "duration", time.Since(t), + ) + } + + raftLogger().Log("msg", "verify_leader") + if err := m.raft.VerifyLeader().Error(); err != nil { + return new(metastorev1.ReadIndexResponse), err + } + + tcheck := time.NewTicker(tcheckFreq) + defer tcheck.Stop() + timeout := time.NewTimer(5 * time.Second) + defer timeout.Stop() + + for { + select { + case <-tcheck.C: + appliedIndex := m.raft.AppliedIndex() + raftLogger().Log("msg", "tick") + if appliedIndex >= readIndex { + raftLogger().Log("msg", "caught up") + return &metastorev1.ReadIndexResponse{ReadIndex: readIndex}, nil + } + continue + case <-timeout.C: + raftLogger().Log("err", "timeout") + return new(metastorev1.ReadIndexResponse), fmt.Errorf("timeout") + case <-ctx.Done(): + raftLogger().Log("err", "context canceled") + return new(metastorev1.ReadIndexResponse), fmt.Errorf("canceled %w", ctx.Err()) + } + } +} + +func (m *Metastore) CheckReady(ctx context.Context) (err error) { + const ( + ready = "ready" + notReady = "not_ready" + status = "status" + ) + debugRequestId := uuid.Must(uuid.NewRandom()).String() //todo delete + readIndex := uint64(0) + t := time.Now() + raftLogger := func() log.Logger { + return log.With(m.logger, "component", "raft_debug", + "request_id", debugRequestId, + "op", "CheckReady", + "read_index", readIndex, + "applied_index", m.raft.AppliedIndex(), + "commit_index", m.raft.CommitIndex(), + "last_index", m.raft.LastIndex(), + "duration", time.Since(t), + ) + } + raftLogger().Log("msg", "check") + req := new(metastorev1.ReadIndexRequest) + req.DebugRequestId = debugRequestId + res, err := m.client.ReadIndex(ctx, req) + if err != nil { + err = fmt.Errorf("failed to get read index: %w", err) + raftLogger().Log(status, notReady, "err", err) + return err + } + readIndex = res.ReadIndex + + tcheck := time.NewTicker(tcheckFreq) + defer tcheck.Stop() + timeout := time.NewTimer(5 * time.Second) + defer timeout.Stop() + + for { + select { + case <-tcheck.C: + commitIndex := m.raft.CommitIndex() + raftLogger().Log("msg", "tick") + if commitIndex >= res.ReadIndex { + if m.readySince.IsZero() { + m.readySince = time.Now() + } + minReadyTime := 30 * time.Second + if time.Since(m.readySince) < minReadyTime { + err := fmt.Errorf("waiting for %v after being ready", minReadyTime) + raftLogger().Log(status, notReady, "err", err) + return err + } + + raftLogger().Log(status, ready) + return nil + } + continue + case <-timeout.C: + raftLogger().Log(status, notReady, "err", "timeout") + return fmt.Errorf("metastore ready check timeout") + case <-ctx.Done(): + raftLogger().Log(status, notReady, "err", "context canceled") + return fmt.Errorf("metastore check context canceled %w", ctx.Err()) + } + } +} diff --git a/pkg/experiment/metastore/metastore_state.go b/pkg/experiment/metastore/metastore_state.go new file mode 100644 index 0000000000..3994f01f56 --- /dev/null +++ b/pkg/experiment/metastore/metastore_state.go @@ -0,0 +1,228 @@ +package metastore + +import ( + "errors" + "fmt" + "strings" + "sync" + + "github.com/go-kit/log" + "github.com/go-kit/log/level" + "github.com/prometheus/client_golang/prometheus" + "go.etcd.io/bbolt" + + metastorev1 "github.com/grafana/pyroscope/api/gen/proto/go/metastore/v1" + "github.com/grafana/pyroscope/pkg/experiment/metastore/compactionpb" +) + +const ( + compactionBucketJobPreQueuePrefix = "job-pre-queue" +) + +type tenantShard struct { + tenant string + shard uint32 +} + +type metastoreState struct { + logger log.Logger + compactionMetrics *compactionMetrics + + shardsMutex sync.Mutex + shards map[uint32]*metastoreShard + + compactionPlansMutex sync.Mutex + preCompactionQueues map[tenantShard]*jobPreQueue + compactionJobQueue *jobQueue + + db *boltdb +} + +type metastoreShard struct { + segmentsMutex sync.Mutex + segments map[string]*metastorev1.BlockMeta +} + +func newMetastoreState(logger log.Logger, db *boltdb, reg prometheus.Registerer) *metastoreState { + return &metastoreState{ + logger: logger, + shards: make(map[uint32]*metastoreShard), + db: db, + preCompactionQueues: make(map[tenantShard]*jobPreQueue), + compactionJobQueue: newJobQueue(jobLeaseDuration.Nanoseconds()), + compactionMetrics: newCompactionMetrics(reg), + } +} + +func (m *metastoreState) reset(db *boltdb) { + m.shardsMutex.Lock() + clear(m.shards) + clear(m.preCompactionQueues) + m.compactionJobQueue = newJobQueue(jobLeaseDuration.Nanoseconds()) + m.db = db + m.shardsMutex.Unlock() +} + +func (m *metastoreState) getOrCreateShard(shardID uint32) *metastoreShard { + m.shardsMutex.Lock() + defer m.shardsMutex.Unlock() + if shard, ok := m.shards[shardID]; ok { + return shard + } + shard := newMetastoreShard() + m.shards[shardID] = shard + return shard +} + +func (m *metastoreState) restore(db *boltdb) error { + m.reset(db) + return db.boltdb.View(func(tx *bbolt.Tx) error { + if err := m.restoreBlockMetadata(tx); err != nil { + return fmt.Errorf("failed to restore metadata entries: %w", err) + } + return m.restoreCompactionPlan(tx) + }) +} + +func (m *metastoreState) restoreBlockMetadata(tx *bbolt.Tx) error { + mdb, err := getBlockMetadataBucket(tx) + switch { + case err == nil: + case errors.Is(err, bbolt.ErrBucketNotFound): + return nil + default: + return err + } + // List shards in the block_metadata bucket: + // block_metadata/[{shard_id}]/[block_id] + // TODO(kolesnikovae): Load concurrently. + return mdb.ForEachBucket(func(name []byte) error { + shardID, _, ok := parseBucketName(name) + if !ok { + _ = level.Error(m.logger).Log("msg", "malformed bucket name", "name", string(name)) + return nil + } + shard := m.getOrCreateShard(shardID) + return shard.loadSegments(mdb.Bucket(name)) + }) +} + +func (m *metastoreState) restoreCompactionPlan(tx *bbolt.Tx) error { + cdb, err := getCompactionJobBucket(tx) + switch { + case err == nil: + case errors.Is(err, bbolt.ErrBucketNotFound): + return nil + default: + return err + } + return cdb.ForEachBucket(func(name []byte) error { + shard, tenant, ok := parseBucketName(name) + if !ok { + _ = level.Error(m.logger).Log("msg", "malformed bucket name", "name", string(name)) + return nil + } + key := tenantShard{ + tenant: tenant, + shard: shard, + } + preQueue := m.getOrCreatePreQueue(key) + + return m.loadCompactionPlan(cdb.Bucket(name), preQueue) + }) + +} + +func (m *metastoreState) getOrCreatePreQueue(key tenantShard) *jobPreQueue { + m.compactionPlansMutex.Lock() + defer m.compactionPlansMutex.Unlock() + + if preQueue, ok := m.preCompactionQueues[key]; ok { + return preQueue + } + plan := &jobPreQueue{ + blocksByLevel: make(map[uint32][]string), + } + m.preCompactionQueues[key] = plan + return plan +} + +func (m *metastoreState) findJob(name string) *compactionpb.CompactionJob { + m.compactionJobQueue.mu.Lock() + defer m.compactionJobQueue.mu.Unlock() + if jobEntry, exists := m.compactionJobQueue.jobs[name]; exists { + return jobEntry.CompactionJob + } + return nil +} + +func newMetastoreShard() *metastoreShard { + return &metastoreShard{ + segments: make(map[string]*metastorev1.BlockMeta), + } +} + +func (s *metastoreShard) putSegment(segment *metastorev1.BlockMeta) { + s.segmentsMutex.Lock() + s.segments[segment.Id] = segment + s.segmentsMutex.Unlock() +} + +func (s *metastoreShard) deleteSegment(segment *metastorev1.BlockMeta) { + s.segmentsMutex.Lock() + delete(s.segments, segment.Id) + s.segmentsMutex.Unlock() +} + +func (s *metastoreShard) loadSegments(b *bbolt.Bucket) error { + s.segmentsMutex.Lock() + defer s.segmentsMutex.Unlock() + c := b.Cursor() + for k, v := c.First(); k != nil; k, v = c.Next() { + var md metastorev1.BlockMeta + if err := md.UnmarshalVT(v); err != nil { + return fmt.Errorf("failed to block %q: %w", string(k), err) + } + s.segments[md.Id] = &md + } + return nil +} + +func (m *metastoreState) loadCompactionPlan(b *bbolt.Bucket, preQueue *jobPreQueue) error { + preQueue.mu.Lock() + defer preQueue.mu.Unlock() + + c := b.Cursor() + for k, v := c.First(); k != nil; k, v = c.Next() { + if strings.HasPrefix(string(k), compactionBucketJobPreQueuePrefix) { + var storedPreQueue compactionpb.JobPreQueue + if err := storedPreQueue.UnmarshalVT(v); err != nil { + return fmt.Errorf("failed to load job pre queue %q: %w", string(k), err) + } + preQueue.blocksByLevel[storedPreQueue.CompactionLevel] = storedPreQueue.Blocks + level.Debug(m.logger).Log( + "msg", "restored pre queue", + "shard", storedPreQueue.Shard, + "compaction_level", storedPreQueue.CompactionLevel, + "block_count", len(storedPreQueue.Blocks), + "blocks", storedPreQueue.Blocks) + } else { + var job compactionpb.CompactionJob + if err := job.UnmarshalVT(v); err != nil { + return fmt.Errorf("failed to unmarshal job %q: %w", string(k), err) + } + m.compactionJobQueue.enqueue(&job) + level.Debug(m.logger).Log( + "msg", "restored job into queue", + "shard", job.Shard, + "tenant", job.TenantId, + "compaction_level", job.CompactionLevel, + "job_status", job.Status.String(), + "raft_log_index", job.RaftLogIndex, + "lease_expires_at", job.LeaseExpiresAt, + "block_count", len(job.Blocks), + "blocks", job.Blocks) + } + } + return nil +} diff --git a/pkg/experiment/metastore/metastore_state_add_block.go b/pkg/experiment/metastore/metastore_state_add_block.go new file mode 100644 index 0000000000..9a4052ee51 --- /dev/null +++ b/pkg/experiment/metastore/metastore_state_add_block.go @@ -0,0 +1,92 @@ +package metastore + +import ( + "context" + "errors" + "fmt" + "time" + + "google.golang.org/grpc/codes" + "google.golang.org/grpc/status" + + "github.com/go-kit/log/level" + + metastorev1 "github.com/grafana/pyroscope/api/gen/proto/go/metastore/v1" + "github.com/grafana/pyroscope/pkg/experiment/metastore/compactionpb" + + "github.com/hashicorp/raft" + "go.etcd.io/bbolt" +) + +func (m *Metastore) AddBlock(_ context.Context, req *metastorev1.AddBlockRequest) (*metastorev1.AddBlockResponse, error) { + _ = level.Info(m.logger).Log( + "msg", "adding block", + "block_id", req.Block.Id, + "shard", req.Block.Shard, + "raft_commit_index", m.raft.CommitIndex(), + "raft_last_index", m.raft.LastIndex(), + "raft_applied_index", m.raft.AppliedIndex()) + t1 := time.Now() + defer func() { + m.metrics.raftAddBlockDuration.Observe(time.Since(t1).Seconds()) + level.Debug(m.logger).Log("msg", "add block duration", "block_id", req.Block.Id, "shard", req.Block.Shard, "duration", time.Since(t1)) + }() + _, resp, err := applyCommand[*metastorev1.AddBlockRequest, *metastorev1.AddBlockResponse](m.raft, req, m.config.Raft.ApplyTimeout) + if err != nil { + _ = level.Error(m.logger).Log("msg", "failed to apply add block", "block_id", req.Block.Id, "shard", req.Block.Shard, "err", err) + if m.shouldRetryAddBlock(err) { + return resp, status.Error(codes.Unavailable, err.Error()) + } + } + return resp, err +} + +func (m *Metastore) shouldRetryAddBlock(err error) bool { + return errors.Is(err, raft.ErrLeadershipLost) || + errors.Is(err, raft.ErrNotLeader) || + errors.Is(err, raft.ErrLeadershipTransferInProgress) || + errors.Is(err, raft.ErrRaftShutdown) +} + +func (m *metastoreState) applyAddBlock(log *raft.Log, request *metastorev1.AddBlockRequest) (*metastorev1.AddBlockResponse, error) { + name, key := keyForBlockMeta(request.Block.Shard, "", request.Block.Id) + value, err := request.Block.MarshalVT() + if err != nil { + return nil, err + } + + var jobToAdd *compactionpb.CompactionJob + var blockToAddToQueue *metastorev1.BlockMeta + + err = m.db.boltdb.Update(func(tx *bbolt.Tx) error { + err := updateBlockMetadataBucket(tx, name, func(bucket *bbolt.Bucket) error { + return bucket.Put(key, value) + }) + if err != nil { + return err + } + err, jobToAdd, blockToAddToQueue = m.consumeBlock(request.Block, tx, log.Index) + return nil + }) + if err != nil { + _ = level.Error(m.logger).Log( + "msg", "failed to add block", + "block", request.Block.Id, + "err", err, + ) + return nil, err + } + m.getOrCreateShard(request.Block.Shard).putSegment(request.Block) + if jobToAdd != nil { + m.addCompactionJob(jobToAdd) + m.compactionMetrics.addedBlocks.WithLabelValues( + fmt.Sprint(jobToAdd.Shard), jobToAdd.TenantId, fmt.Sprint(jobToAdd.CompactionLevel)).Inc() + m.compactionMetrics.addedJobs.WithLabelValues( + fmt.Sprint(jobToAdd.Shard), jobToAdd.TenantId, fmt.Sprint(jobToAdd.CompactionLevel)).Inc() + } else if blockToAddToQueue != nil { + m.addBlockToCompactionJobQueue(blockToAddToQueue) + m.compactionMetrics.addedBlocks.WithLabelValues( + fmt.Sprint(blockToAddToQueue.Shard), blockToAddToQueue.TenantId, fmt.Sprint(blockToAddToQueue.CompactionLevel)).Inc() + } + return &metastorev1.AddBlockResponse{}, nil +} diff --git a/pkg/experiment/metastore/metastore_state_poll_compaction_jobs.go b/pkg/experiment/metastore/metastore_state_poll_compaction_jobs.go new file mode 100644 index 0000000000..b0802bc2a5 --- /dev/null +++ b/pkg/experiment/metastore/metastore_state_poll_compaction_jobs.go @@ -0,0 +1,342 @@ +package metastore + +import ( + "context" + "fmt" + "math" + + "github.com/go-kit/log/level" + "github.com/hashicorp/raft" + "github.com/pkg/errors" + "go.etcd.io/bbolt" + + compactorv1 "github.com/grafana/pyroscope/api/gen/proto/go/compactor/v1" + metastorev1 "github.com/grafana/pyroscope/api/gen/proto/go/metastore/v1" + "github.com/grafana/pyroscope/pkg/experiment/metastore/compactionpb" +) + +func (m *Metastore) PollCompactionJobs(_ context.Context, req *compactorv1.PollCompactionJobsRequest) (*compactorv1.PollCompactionJobsResponse, error) { + level.Debug(m.logger).Log( + "msg", "received poll compaction jobs request", + "num_updates", len(req.JobStatusUpdates), + "job_capacity", req.JobCapacity, + "raft_commit_index", m.raft.CommitIndex(), + "raft_last_index", m.raft.LastIndex(), + "raft_applied_index", m.raft.AppliedIndex()) + _, resp, err := applyCommand[*compactorv1.PollCompactionJobsRequest, *compactorv1.PollCompactionJobsResponse](m.raft, req, m.config.Raft.ApplyTimeout) + return resp, err +} + +type jobResult struct { + newBlocks []*metastorev1.BlockMeta + deletedBlocks []*metastorev1.BlockMeta + newJobs []*compactionpb.CompactionJob + newQueuedBlocks []*metastorev1.BlockMeta + deletedJobs []*compactionpb.CompactionJob + + newJobAssignments []*compactionpb.CompactionJob +} + +func (m *metastoreState) applyPollCompactionJobs(raft *raft.Log, request *compactorv1.PollCompactionJobsRequest) (resp *compactorv1.PollCompactionJobsResponse, err error) { + resp = &compactorv1.PollCompactionJobsResponse{} + level.Debug(m.logger).Log( + "msg", "applying poll compaction jobs", + "num_updates", len(request.JobStatusUpdates), + "job_capacity", request.JobCapacity, + "raft_log_index", raft.Index) + + jResult := &jobResult{ + newBlocks: make([]*metastorev1.BlockMeta, 0), + deletedBlocks: make([]*metastorev1.BlockMeta, 0), + newJobs: make([]*compactionpb.CompactionJob, 0), + newQueuedBlocks: make([]*metastorev1.BlockMeta, 0), + deletedJobs: make([]*compactionpb.CompactionJob, 0), + newJobAssignments: make([]*compactionpb.CompactionJob, 0), + } + + err = m.db.boltdb.Update(func(tx *bbolt.Tx) error { + for _, statusUpdate := range request.JobStatusUpdates { + job := m.findJob(statusUpdate.JobName) + if job == nil { + level.Error(m.logger).Log("msg", "error processing update for compaction job, job not found", "job", statusUpdate.JobName, "err", err) + continue + } + + level.Debug(m.logger).Log("msg", "processing status update for compaction job", "job", statusUpdate.JobName, "status", statusUpdate.Status) + name, _ := keyForCompactionJob(statusUpdate.Shard, statusUpdate.TenantId, statusUpdate.JobName) + err := updateCompactionJobBucket(tx, name, func(bucket *bbolt.Bucket) error { + switch statusUpdate.Status { // TODO: handle other cases + case compactorv1.CompactionStatus_COMPACTION_STATUS_SUCCESS: + err := m.processCompletedJob(tx, job, statusUpdate, jResult, raft.Index) + if err != nil { + level.Error(m.logger).Log("msg", "failed to update completed job", "job", job.Name, "err", err) + return errors.Wrap(err, "failed to update completed job") + } + case compactorv1.CompactionStatus_COMPACTION_STATUS_IN_PROGRESS: + if m.compactionJobQueue.isOwner(statusUpdate.JobName, statusUpdate.RaftLogIndex) { + err := m.persistJobDeadline(tx, job, m.compactionJobQueue.getNewDeadline(raft.AppendedAt.UnixNano())) + if err != nil { + return errors.Wrap(err, "failed to update compaction job deadline") + } + m.compactionJobQueue.update(statusUpdate.JobName, raft.AppendedAt.UnixNano(), statusUpdate.RaftLogIndex) + } else { + level.Warn(m.logger).Log("msg", "compaction job status update rejected", "job", job.Name, "raft_log_index", statusUpdate.RaftLogIndex) + return errors.New("compaction job status update rejected") + } + } + return nil + }) + if err != nil { + level.Error(m.logger).Log("msg", "error processing update for compaction job", "job", job.Name, "err", err) + continue + } + } + + if request.JobCapacity > 0 { + jResult.newJobAssignments, err = m.assignNewJobs(tx, int(request.JobCapacity), raft.Index, raft.AppendedAt.UnixNano()) + if err != nil { + return err + } + } + + return nil + }) + if err != nil { + return nil, err + } + + // now update the state + for _, b := range jResult.newBlocks { + m.getOrCreateShard(b.Shard).putSegment(b) + m.compactionMetrics.addedBlocks.WithLabelValues(fmt.Sprint(b.Shard), b.TenantId, fmt.Sprint(b.CompactionLevel)).Inc() + } + + for _, b := range jResult.deletedBlocks { + m.getOrCreateShard(b.Shard).deleteSegment(b) + m.compactionMetrics.deletedBlocks.WithLabelValues(fmt.Sprint(b.Shard), b.TenantId, fmt.Sprint(b.CompactionLevel)).Inc() + } + + for _, j := range jResult.newJobs { + m.addCompactionJob(j) + m.compactionMetrics.addedJobs.WithLabelValues(fmt.Sprint(j.Shard), j.TenantId, fmt.Sprint(j.CompactionLevel)).Inc() + } + + for _, b := range jResult.newQueuedBlocks { + m.addBlockToCompactionJobQueue(b) + // already counted above + } + + for _, j := range jResult.deletedJobs { + m.compactionJobQueue.evict(j.Name, j.RaftLogIndex) + m.compactionMetrics.completedJobs.WithLabelValues(fmt.Sprint(j.Shard), j.TenantId, fmt.Sprint(j.CompactionLevel)).Inc() + } + + resp.CompactionJobs, err = m.convertJobs(jResult.newJobAssignments) + for _, j := range resp.CompactionJobs { + m.compactionMetrics.assignedJobs.WithLabelValues(fmt.Sprint(j.Shard), j.TenantId, fmt.Sprint(j.CompactionLevel)).Inc() + } + + return resp, err +} + +func (m *metastoreState) convertJobs(jobs []*compactionpb.CompactionJob) ([]*compactorv1.CompactionJob, error) { + res := make([]*compactorv1.CompactionJob, 0, len(jobs)) + for _, job := range jobs { + // populate block metadata (workers rely on it) + blocks := make([]*metastorev1.BlockMeta, 0, len(job.Blocks)) + for _, bId := range job.Blocks { + b := m.findBlock(job.Shard, bId) + if b == nil { + level.Error(m.logger).Log( + "msg", "failed to populate job details, block not found", + "block", bId, + "shard", job.Shard, + "job", job.Name) + continue + } + blocks = append(blocks, b) + } + if len(blocks) == 0 { + evicted := m.compactionJobQueue.evict(job.Name, math.MaxInt64) + level.Warn(m.logger).Log("msg", "skipping assigned compaction job since it has no valid blocks", "job", job.Name, "evicted", evicted) + continue + } + + res = append(res, &compactorv1.CompactionJob{ + Name: job.Name, + Blocks: blocks, + Status: &compactorv1.CompactionJobStatus{ + JobName: job.Name, + Status: compactorv1.CompactionStatus(job.Status), + RaftLogIndex: job.RaftLogIndex, + Shard: job.Shard, + TenantId: job.TenantId, + }, + CompactionLevel: job.CompactionLevel, + RaftLogIndex: job.RaftLogIndex, + Shard: job.Shard, + TenantId: job.TenantId, + }) + } + return res, nil +} + +func (m *metastoreState) processCompletedJob( + tx *bbolt.Tx, + job *compactionpb.CompactionJob, + update *compactorv1.CompactionJobStatus, + jResult *jobResult, + raftLogIndex uint64, +) error { + ownsJob := m.compactionJobQueue.isOwner(job.Name, update.RaftLogIndex) + if !ownsJob { + return errors.New(fmt.Sprintf("deadline exceeded for job with id %s", job.Name)) + } + jBucket, jKey := keyForCompactionJob(job.Shard, job.TenantId, job.Name) + err := updateCompactionJobBucket(tx, jBucket, func(bucket *bbolt.Bucket) error { + return bucket.Delete(jKey) + }) + if err != nil { + return err + } + jResult.deletedJobs = append(jResult.deletedJobs, job) + for _, b := range update.CompletedJob.Blocks { + bName, bKey := keyForBlockMeta(b.Shard, b.TenantId, b.Id) + err = updateBlockMetadataBucket(tx, bName, func(bucket *bbolt.Bucket) error { + bValue, _ := b.MarshalVT() + return bucket.Put(bKey, bValue) + }) + if err != nil { + _ = level.Error(m.logger).Log( + "msg", "failed to add block", + "block", b.Id, + "err", err, + ) + return err + } + jResult.newBlocks = append(jResult.newBlocks, b) + + // create and store an optional compaction job + err, jobToAdd, blockForQueue := m.consumeBlock(b, tx, raftLogIndex) + if err != nil { + return err + } + if jobToAdd != nil { + jResult.newJobs = append(jResult.newJobs, jobToAdd) + } else if blockForQueue != nil { + jResult.newQueuedBlocks = append(jResult.newQueuedBlocks, blockForQueue) + } + } + + // delete source blocks + bName, _ := keyForBlockMeta(job.Shard, job.TenantId, "") + err = updateBlockMetadataBucket(tx, bName, func(bucket *bbolt.Bucket) error { + for _, bId := range job.Blocks { + level.Debug(m.logger).Log("msg", "deleting block from storage", "block", bId, "compaction_job", job.Name) + b := m.findBlock(job.Shard, bId) + if b == nil { + level.Error(m.logger).Log("msg", "failed to delete block from storage, block not found", "block", bId, "shard", job.Shard) + return errors.Wrapf(err, "failed to find compaction job source block %s for deletion", bId) + } + + _, bKey := keyForBlockMeta(b.Shard, b.TenantId, b.Id) + err := bucket.Delete(bKey) + if err != nil { + return errors.Wrapf(err, "failed to delete compaction job source block %s", b.Id) + } + jResult.deletedBlocks = append(jResult.deletedBlocks, b) + } + return nil + }) + if err != nil { + return err + } + job.RaftLogIndex = update.RaftLogIndex + return nil +} + +func (m *metastoreState) findBlock(shard uint32, blockId string) *metastorev1.BlockMeta { + segmentShard := m.getOrCreateShard(shard) + segmentShard.segmentsMutex.Lock() + defer segmentShard.segmentsMutex.Unlock() + + return segmentShard.segments[blockId] +} + +func (m *metastoreState) persistAssignedJob(tx *bbolt.Tx, job *compactionpb.CompactionJob) error { + return m.persistJob(tx, job, func(storedJob *compactionpb.CompactionJob) { + storedJob.Status = job.Status + storedJob.LeaseExpiresAt = job.LeaseExpiresAt + storedJob.RaftLogIndex = job.RaftLogIndex + }) +} + +func (m *metastoreState) persistJobDeadline(tx *bbolt.Tx, job *compactionpb.CompactionJob, leaseExpiresAt int64) error { + return m.persistJob(tx, job, func(storedJob *compactionpb.CompactionJob) { + storedJob.LeaseExpiresAt = leaseExpiresAt + }) +} + +func (m *metastoreState) persistJob(tx *bbolt.Tx, job *compactionpb.CompactionJob, fn func(compactionJob *compactionpb.CompactionJob)) error { + jobBucketName, jobKey := keyForCompactionJob(job.Shard, job.TenantId, job.Name) + err := updateCompactionJobBucket(tx, jobBucketName, func(bucket *bbolt.Bucket) error { + storedJobData := bucket.Get(jobKey) + if storedJobData == nil { + return errors.New("compaction job not found in storage") + } + var storedJob compactionpb.CompactionJob + err := storedJob.UnmarshalVT(storedJobData) + if err != nil { + return errors.Wrap(err, "failed to unmarshal compaction job data") + } + fn(&storedJob) + jobData, _ := storedJob.MarshalVT() + return bucket.Put(jobKey, jobData) + }) + return err +} + +func (m *metastoreState) assignNewJobs(tx *bbolt.Tx, jobCapacity int, raftLogIndex uint64, now int64) ([]*compactionpb.CompactionJob, error) { + jobsToAssign := m.findJobsToAssign(jobCapacity, raftLogIndex, now) + level.Debug(m.logger).Log("msg", "compaction jobs to assign", "jobs", len(jobsToAssign), "raft_log_index", raftLogIndex, "capacity", jobCapacity) + + for _, job := range jobsToAssign { + // mark job "in progress" + err := m.persistAssignedJob(tx, job) + if err != nil { + level.Error(m.logger).Log("msg", "failed to update job status", "job", job.Name, "err", err) + // return the job back to the queue + m.compactionJobQueue.enqueue(job) + return nil, errors.Wrap(err, "failed to update job status") + } + } + + return jobsToAssign, nil +} + +func (m *metastoreState) findJobsToAssign(jobCapacity int, raftLogIndex uint64, now int64) []*compactionpb.CompactionJob { + jobsToAssign := make([]*compactionpb.CompactionJob, 0, jobCapacity) + jobCount, newJobs, inProgressJobs, completedJobs, failedJobs := m.compactionJobQueue.stats() + level.Debug(m.logger).Log( + "msg", "looking for jobs to assign", + "job_capacity", jobCapacity, + "raft_log_index", raftLogIndex, + "job_queue_size", jobCount, + "new_jobs_in_queue", newJobs, + "in_progress_jobs_in_queue", inProgressJobs, + "completed_jobs_in_queue", completedJobs, + "failed_jobs_in_queue", failedJobs, + ) + + var j *compactionpb.CompactionJob + for len(jobsToAssign) < jobCapacity { + j = m.compactionJobQueue.dequeue(now, raftLogIndex) + if j == nil { + break + } + level.Debug(m.logger).Log("msg", "assigning job to raftLogIndex", "job", j, "raft_log_index", raftLogIndex) + jobsToAssign = append(jobsToAssign, j) + } + + return jobsToAssign +} diff --git a/pkg/experiment/metastore/metastore_state_query_metadata.go b/pkg/experiment/metastore/metastore_state_query_metadata.go new file mode 100644 index 0000000000..64d9d88d7c --- /dev/null +++ b/pkg/experiment/metastore/metastore_state_query_metadata.go @@ -0,0 +1,148 @@ +package metastore + +import ( + "context" + "fmt" + "slices" + "strings" + "sync" + + "github.com/prometheus/prometheus/model/labels" + "github.com/prometheus/prometheus/promql/parser" + "golang.org/x/sync/errgroup" + "google.golang.org/grpc/codes" + "google.golang.org/grpc/status" + + metastorev1 "github.com/grafana/pyroscope/api/gen/proto/go/metastore/v1" + "github.com/grafana/pyroscope/pkg/model" +) + +func (m *Metastore) QueryMetadata( + ctx context.Context, + request *metastorev1.QueryMetadataRequest, +) (*metastorev1.QueryMetadataResponse, error) { + // TODO(kolesnikovae): ReadIndex + return m.state.listBlocksForQuery(ctx, request) +} + +type metadataQuery struct { + startTime int64 + endTime int64 + tenants map[string]struct{} + serviceMatcher *labels.Matcher +} + +func newMetadataQuery(request *metastorev1.QueryMetadataRequest) (*metadataQuery, error) { + if len(request.TenantId) == 0 { + return nil, fmt.Errorf("tenant_id is required") + } + q := &metadataQuery{ + startTime: request.StartTime, + endTime: request.EndTime, + tenants: make(map[string]struct{}, len(request.TenantId)), + } + for _, tenant := range request.TenantId { + q.tenants[tenant] = struct{}{} + } + selectors, err := parser.ParseMetricSelector(request.Query) + if err != nil { + return nil, fmt.Errorf("failed to parse label selectors: %w", err) + } + for _, m := range selectors { + if m.Name == model.LabelNameServiceName { + q.serviceMatcher = m + break + } + } + // We could also validate that the service has the profile type + // queried, but that's not really necessary: querying an irrelevant + // profile type is rather a rare/invalid case. + return q, nil +} + +func (q *metadataQuery) matchBlock(b *metastorev1.BlockMeta) bool { + return inRange(b.MinTime, b.MaxTime, q.startTime, q.endTime) +} + +func (q *metadataQuery) matchService(s *metastorev1.Dataset) bool { + _, ok := q.tenants[s.TenantId] + if !ok { + return false + } + if !inRange(s.MinTime, s.MaxTime, q.startTime, q.endTime) { + return false + } + if q.serviceMatcher != nil { + return q.serviceMatcher.Matches(s.Name) + } + return true +} + +func inRange(blockStart, blockEnd, queryStart, queryEnd int64) bool { + return blockStart <= queryEnd && blockEnd >= queryStart +} + +func (s *metastoreShard) listBlocksForQuery(q *metadataQuery) map[string]*metastorev1.BlockMeta { + s.segmentsMutex.Lock() + defer s.segmentsMutex.Unlock() + md := make(map[string]*metastorev1.BlockMeta, 32) + for _, segment := range s.segments { + if !q.matchBlock(segment) { + continue + } + var block *metastorev1.BlockMeta + for _, svc := range segment.Datasets { + if q.matchService(svc) { + if block == nil { + block = cloneBlockForQuery(segment) + md[segment.Id] = block + } + block.Datasets = append(block.Datasets, svc) + } + } + } + return md +} + +func cloneBlockForQuery(b *metastorev1.BlockMeta) *metastorev1.BlockMeta { + datasets := b.Datasets + b.Datasets = nil + c := b.CloneVT() + b.Datasets = datasets + c.Datasets = make([]*metastorev1.Dataset, 0, len(b.Datasets)) + return c +} + +func (m *metastoreState) listBlocksForQuery( + ctx context.Context, + request *metastorev1.QueryMetadataRequest, +) (*metastorev1.QueryMetadataResponse, error) { + q, err := newMetadataQuery(request) + if err != nil { + return nil, status.Error(codes.InvalidArgument, err.Error()) + } + var respMutex sync.Mutex + var resp metastorev1.QueryMetadataResponse + g, ctx := errgroup.WithContext(ctx) + m.shardsMutex.Lock() + for _, s := range m.shards { + s := s + g.Go(func() error { + blocks := s.listBlocksForQuery(q) + respMutex.Lock() + for _, b := range blocks { + resp.Blocks = append(resp.Blocks, b) + } + respMutex.Unlock() + return nil + }) + } + m.shardsMutex.Unlock() + if err = g.Wait(); err != nil { + return nil, err + } + slices.SortFunc(resp.Blocks, func(a, b *metastorev1.BlockMeta) int { + return strings.Compare(a.Id, b.Id) + }) + return &resp, nil +} diff --git a/pkg/experiment/metastore/raftleader/raftleader.go b/pkg/experiment/metastore/raftleader/raftleader.go new file mode 100644 index 0000000000..c00a4f251f --- /dev/null +++ b/pkg/experiment/metastore/raftleader/raftleader.go @@ -0,0 +1,133 @@ +package raftleader + +import ( + "sync" + + "github.com/prometheus/client_golang/prometheus" + + "github.com/go-kit/log" + "github.com/go-kit/log/level" + "github.com/hashicorp/raft" + "google.golang.org/grpc/health/grpc_health_v1" + + "github.com/grafana/pyroscope/pkg/util/health" +) + +type HealthObserver struct { + server health.Service + logger log.Logger + mu sync.Mutex + registered map[serviceKey]*raftService + metrics *Metrics +} +type Metrics struct { + status prometheus.Gauge +} + +func NewMetrics(reg prometheus.Registerer) *Metrics { + m := &Metrics{ + status: prometheus.NewGauge(prometheus.GaugeOpts{ + Namespace: "pyroscope", + Name: "metastore_raft_status", + }), + } + if reg != nil { + reg.MustRegister(m.status) + } + return m +} + +func NewRaftLeaderHealthObserver(hs health.Service, logger log.Logger, m *Metrics) *HealthObserver { + return &HealthObserver{ + server: hs, + logger: logger, + metrics: m, + registered: make(map[serviceKey]*raftService), + } +} + +func (hs *HealthObserver) Register(r *raft.Raft, service string) { + hs.mu.Lock() + defer hs.mu.Unlock() + k := serviceKey{raft: r, service: service} + if _, ok := hs.registered[k]; ok { + return + } + svc := &raftService{ + server: hs.server, + hs: hs, + logger: log.With(hs.logger, "service", service), + service: service, + raft: r, + c: make(chan raft.Observation, 1), + stop: make(chan struct{}), + done: make(chan struct{}), + } + _ = level.Debug(svc.logger).Log("msg", "registering health check") + svc.updateStatus() + go svc.run() + svc.observer = raft.NewObserver(svc.c, true, func(o *raft.Observation) bool { + _, ok := o.Data.(raft.LeaderObservation) + return ok + }) + r.RegisterObserver(svc.observer) + hs.registered[k] = svc +} + +func (hs *HealthObserver) Deregister(r *raft.Raft, service string) { + hs.mu.Lock() + k := serviceKey{raft: r, service: service} + svc, ok := hs.registered[k] + delete(hs.registered, k) + hs.mu.Unlock() + if ok { + close(svc.stop) + <-svc.done + } +} + +type serviceKey struct { + raft *raft.Raft + service string +} + +type raftService struct { + server health.Service + hs *HealthObserver + logger log.Logger + service string + raft *raft.Raft + observer *raft.Observer + c chan raft.Observation + stop chan struct{} + done chan struct{} +} + +func (svc *raftService) run() { + defer func() { + close(svc.done) + }() + for { + select { + case <-svc.c: + svc.updateStatus() + case <-svc.stop: + _ = level.Debug(svc.logger).Log("msg", "deregistering health check") + // We explicitly remove the service from serving when we stop observing it. + svc.server.SetServingStatus(svc.service, grpc_health_v1.HealthCheckResponse_NOT_SERVING) + svc.raft.DeregisterObserver(svc.observer) + return + } + } +} + +func (svc *raftService) updateStatus() { + status := grpc_health_v1.HealthCheckResponse_NOT_SERVING + if svc.raft.State() == raft.Leader { + status = grpc_health_v1.HealthCheckResponse_SERVING + } + svc.hs.metrics.status.Set(float64(svc.raft.State())) + + _ = level.Info(svc.logger).Log("msg", "updating health status", "status", status) + svc.server.SetServingStatus(svc.service, status) +} diff --git a/pkg/experiment/metastore/raftlogpb/raflog.pb.go b/pkg/experiment/metastore/raftlogpb/raflog.pb.go new file mode 100644 index 0000000000..19014faa75 --- /dev/null +++ b/pkg/experiment/metastore/raftlogpb/raflog.pb.go @@ -0,0 +1,292 @@ +// Code generated by protoc-gen-go. DO NOT EDIT. +// versions: +// protoc-gen-go v1.34.2 +// protoc (unknown) +// source: experiment/metastore/raftlogpb/raflog.proto + +package raftlogpb + +import ( + protoreflect "google.golang.org/protobuf/reflect/protoreflect" + protoimpl "google.golang.org/protobuf/runtime/protoimpl" + reflect "reflect" + sync "sync" +) + +const ( + // Verify that this generated code is sufficiently up-to-date. + _ = protoimpl.EnforceVersion(20 - protoimpl.MinVersion) + // Verify that runtime/protoimpl is sufficiently up-to-date. + _ = protoimpl.EnforceVersion(protoimpl.MaxVersion - 20) +) + +type CommandType int32 + +const ( + CommandType_COMMAND_TYPE_UNKNOWN CommandType = 0 + CommandType_COMMAND_TYPE_ADD_BLOCK CommandType = 1 + CommandType_COMMAND_TYPE_POLL_COMPACTION_JOBS_STATUS CommandType = 2 + // This is a temporary solution. + CommandType_COMMAND_TYPE_TRUNCATE CommandType = 4196 +) + +// Enum value maps for CommandType. +var ( + CommandType_name = map[int32]string{ + 0: "COMMAND_TYPE_UNKNOWN", + 1: "COMMAND_TYPE_ADD_BLOCK", + 2: "COMMAND_TYPE_POLL_COMPACTION_JOBS_STATUS", + 4196: "COMMAND_TYPE_TRUNCATE", + } + CommandType_value = map[string]int32{ + "COMMAND_TYPE_UNKNOWN": 0, + "COMMAND_TYPE_ADD_BLOCK": 1, + "COMMAND_TYPE_POLL_COMPACTION_JOBS_STATUS": 2, + "COMMAND_TYPE_TRUNCATE": 4196, + } +) + +func (x CommandType) Enum() *CommandType { + p := new(CommandType) + *p = x + return p +} + +func (x CommandType) String() string { + return protoimpl.X.EnumStringOf(x.Descriptor(), protoreflect.EnumNumber(x)) +} + +func (CommandType) Descriptor() protoreflect.EnumDescriptor { + return file_experiment_metastore_raftlogpb_raflog_proto_enumTypes[0].Descriptor() +} + +func (CommandType) Type() protoreflect.EnumType { + return &file_experiment_metastore_raftlogpb_raflog_proto_enumTypes[0] +} + +func (x CommandType) Number() protoreflect.EnumNumber { + return protoreflect.EnumNumber(x) +} + +// Deprecated: Use CommandType.Descriptor instead. +func (CommandType) EnumDescriptor() ([]byte, []int) { + return file_experiment_metastore_raftlogpb_raflog_proto_rawDescGZIP(), []int{0} +} + +type RaftLogEntry struct { + state protoimpl.MessageState + sizeCache protoimpl.SizeCache + unknownFields protoimpl.UnknownFields + + Type CommandType `protobuf:"varint,1,opt,name=type,proto3,enum=raft_log.CommandType" json:"type,omitempty"` + Payload []byte `protobuf:"bytes,2,opt,name=payload,proto3" json:"payload,omitempty"` +} + +func (x *RaftLogEntry) Reset() { + *x = RaftLogEntry{} + if protoimpl.UnsafeEnabled { + mi := &file_experiment_metastore_raftlogpb_raflog_proto_msgTypes[0] + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + ms.StoreMessageInfo(mi) + } +} + +func (x *RaftLogEntry) String() string { + return protoimpl.X.MessageStringOf(x) +} + +func (*RaftLogEntry) ProtoMessage() {} + +func (x *RaftLogEntry) ProtoReflect() protoreflect.Message { + mi := &file_experiment_metastore_raftlogpb_raflog_proto_msgTypes[0] + if protoimpl.UnsafeEnabled && x != nil { + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + if ms.LoadMessageInfo() == nil { + ms.StoreMessageInfo(mi) + } + return ms + } + return mi.MessageOf(x) +} + +// Deprecated: Use RaftLogEntry.ProtoReflect.Descriptor instead. +func (*RaftLogEntry) Descriptor() ([]byte, []int) { + return file_experiment_metastore_raftlogpb_raflog_proto_rawDescGZIP(), []int{0} +} + +func (x *RaftLogEntry) GetType() CommandType { + if x != nil { + return x.Type + } + return CommandType_COMMAND_TYPE_UNKNOWN +} + +func (x *RaftLogEntry) GetPayload() []byte { + if x != nil { + return x.Payload + } + return nil +} + +type TruncateCommand struct { + state protoimpl.MessageState + sizeCache protoimpl.SizeCache + unknownFields protoimpl.UnknownFields + + Timestamp uint64 `protobuf:"varint,1,opt,name=timestamp,proto3" json:"timestamp,omitempty"` +} + +func (x *TruncateCommand) Reset() { + *x = TruncateCommand{} + if protoimpl.UnsafeEnabled { + mi := &file_experiment_metastore_raftlogpb_raflog_proto_msgTypes[1] + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + ms.StoreMessageInfo(mi) + } +} + +func (x *TruncateCommand) String() string { + return protoimpl.X.MessageStringOf(x) +} + +func (*TruncateCommand) ProtoMessage() {} + +func (x *TruncateCommand) ProtoReflect() protoreflect.Message { + mi := &file_experiment_metastore_raftlogpb_raflog_proto_msgTypes[1] + if protoimpl.UnsafeEnabled && x != nil { + ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) + if ms.LoadMessageInfo() == nil { + ms.StoreMessageInfo(mi) + } + return ms + } + return mi.MessageOf(x) +} + +// Deprecated: Use TruncateCommand.ProtoReflect.Descriptor instead. +func (*TruncateCommand) Descriptor() ([]byte, []int) { + return file_experiment_metastore_raftlogpb_raflog_proto_rawDescGZIP(), []int{1} +} + +func (x *TruncateCommand) GetTimestamp() uint64 { + if x != nil { + return x.Timestamp + } + return 0 +} + +var File_experiment_metastore_raftlogpb_raflog_proto protoreflect.FileDescriptor + +var file_experiment_metastore_raftlogpb_raflog_proto_rawDesc = []byte{ + 0x0a, 0x2b, 0x65, 0x78, 0x70, 0x65, 0x72, 0x69, 0x6d, 0x65, 0x6e, 0x74, 0x2f, 0x6d, 0x65, 0x74, + 0x61, 0x73, 0x74, 0x6f, 0x72, 0x65, 0x2f, 0x72, 0x61, 0x66, 0x74, 0x6c, 0x6f, 0x67, 0x70, 0x62, + 0x2f, 0x72, 0x61, 0x66, 0x6c, 0x6f, 0x67, 0x2e, 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x12, 0x08, 0x72, + 0x61, 0x66, 0x74, 0x5f, 0x6c, 0x6f, 0x67, 0x22, 0x53, 0x0a, 0x0c, 0x52, 0x61, 0x66, 0x74, 0x4c, + 0x6f, 0x67, 0x45, 0x6e, 0x74, 0x72, 0x79, 0x12, 0x29, 0x0a, 0x04, 0x74, 0x79, 0x70, 0x65, 0x18, + 0x01, 0x20, 0x01, 0x28, 0x0e, 0x32, 0x15, 0x2e, 0x72, 0x61, 0x66, 0x74, 0x5f, 0x6c, 0x6f, 0x67, + 0x2e, 0x43, 0x6f, 0x6d, 0x6d, 0x61, 0x6e, 0x64, 0x54, 0x79, 0x70, 0x65, 0x52, 0x04, 0x74, 0x79, + 0x70, 0x65, 0x12, 0x18, 0x0a, 0x07, 0x70, 0x61, 0x79, 0x6c, 0x6f, 0x61, 0x64, 0x18, 0x02, 0x20, + 0x01, 0x28, 0x0c, 0x52, 0x07, 0x70, 0x61, 0x79, 0x6c, 0x6f, 0x61, 0x64, 0x22, 0x2f, 0x0a, 0x0f, + 0x54, 0x72, 0x75, 0x6e, 0x63, 0x61, 0x74, 0x65, 0x43, 0x6f, 0x6d, 0x6d, 0x61, 0x6e, 0x64, 0x12, + 0x1c, 0x0a, 0x09, 0x74, 0x69, 0x6d, 0x65, 0x73, 0x74, 0x61, 0x6d, 0x70, 0x18, 0x01, 0x20, 0x01, + 0x28, 0x04, 0x52, 0x09, 0x74, 0x69, 0x6d, 0x65, 0x73, 0x74, 0x61, 0x6d, 0x70, 0x2a, 0x8d, 0x01, + 0x0a, 0x0b, 0x43, 0x6f, 0x6d, 0x6d, 0x61, 0x6e, 0x64, 0x54, 0x79, 0x70, 0x65, 0x12, 0x18, 0x0a, + 0x14, 0x43, 0x4f, 0x4d, 0x4d, 0x41, 0x4e, 0x44, 0x5f, 0x54, 0x59, 0x50, 0x45, 0x5f, 0x55, 0x4e, + 0x4b, 0x4e, 0x4f, 0x57, 0x4e, 0x10, 0x00, 0x12, 0x1a, 0x0a, 0x16, 0x43, 0x4f, 0x4d, 0x4d, 0x41, + 0x4e, 0x44, 0x5f, 0x54, 0x59, 0x50, 0x45, 0x5f, 0x41, 0x44, 0x44, 0x5f, 0x42, 0x4c, 0x4f, 0x43, + 0x4b, 0x10, 0x01, 0x12, 0x2c, 0x0a, 0x28, 0x43, 0x4f, 0x4d, 0x4d, 0x41, 0x4e, 0x44, 0x5f, 0x54, + 0x59, 0x50, 0x45, 0x5f, 0x50, 0x4f, 0x4c, 0x4c, 0x5f, 0x43, 0x4f, 0x4d, 0x50, 0x41, 0x43, 0x54, + 0x49, 0x4f, 0x4e, 0x5f, 0x4a, 0x4f, 0x42, 0x53, 0x5f, 0x53, 0x54, 0x41, 0x54, 0x55, 0x53, 0x10, + 0x02, 0x12, 0x1a, 0x0a, 0x15, 0x43, 0x4f, 0x4d, 0x4d, 0x41, 0x4e, 0x44, 0x5f, 0x54, 0x59, 0x50, + 0x45, 0x5f, 0x54, 0x52, 0x55, 0x4e, 0x43, 0x41, 0x54, 0x45, 0x10, 0xe4, 0x20, 0x42, 0x98, 0x01, + 0x0a, 0x0c, 0x63, 0x6f, 0x6d, 0x2e, 0x72, 0x61, 0x66, 0x74, 0x5f, 0x6c, 0x6f, 0x67, 0x42, 0x0b, + 0x52, 0x61, 0x66, 0x6c, 0x6f, 0x67, 0x50, 0x72, 0x6f, 0x74, 0x6f, 0x50, 0x01, 0x5a, 0x3f, 0x67, + 0x69, 0x74, 0x68, 0x75, 0x62, 0x2e, 0x63, 0x6f, 0x6d, 0x2f, 0x67, 0x72, 0x61, 0x66, 0x61, 0x6e, + 0x61, 0x2f, 0x70, 0x79, 0x72, 0x6f, 0x73, 0x63, 0x6f, 0x70, 0x65, 0x2f, 0x70, 0x6b, 0x67, 0x2f, + 0x65, 0x78, 0x70, 0x65, 0x72, 0x69, 0x6d, 0x65, 0x6e, 0x74, 0x2f, 0x6d, 0x65, 0x74, 0x61, 0x73, + 0x74, 0x6f, 0x72, 0x65, 0x2f, 0x72, 0x61, 0x66, 0x74, 0x6c, 0x6f, 0x67, 0x70, 0x62, 0xa2, 0x02, + 0x03, 0x52, 0x58, 0x58, 0xaa, 0x02, 0x07, 0x52, 0x61, 0x66, 0x74, 0x4c, 0x6f, 0x67, 0xca, 0x02, + 0x07, 0x52, 0x61, 0x66, 0x74, 0x4c, 0x6f, 0x67, 0xe2, 0x02, 0x13, 0x52, 0x61, 0x66, 0x74, 0x4c, + 0x6f, 0x67, 0x5c, 0x47, 0x50, 0x42, 0x4d, 0x65, 0x74, 0x61, 0x64, 0x61, 0x74, 0x61, 0xea, 0x02, + 0x07, 0x52, 0x61, 0x66, 0x74, 0x4c, 0x6f, 0x67, 0x62, 0x06, 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x33, +} + +var ( + file_experiment_metastore_raftlogpb_raflog_proto_rawDescOnce sync.Once + file_experiment_metastore_raftlogpb_raflog_proto_rawDescData = file_experiment_metastore_raftlogpb_raflog_proto_rawDesc +) + +func file_experiment_metastore_raftlogpb_raflog_proto_rawDescGZIP() []byte { + file_experiment_metastore_raftlogpb_raflog_proto_rawDescOnce.Do(func() { + file_experiment_metastore_raftlogpb_raflog_proto_rawDescData = protoimpl.X.CompressGZIP(file_experiment_metastore_raftlogpb_raflog_proto_rawDescData) + }) + return file_experiment_metastore_raftlogpb_raflog_proto_rawDescData +} + +var file_experiment_metastore_raftlogpb_raflog_proto_enumTypes = make([]protoimpl.EnumInfo, 1) +var file_experiment_metastore_raftlogpb_raflog_proto_msgTypes = make([]protoimpl.MessageInfo, 2) +var file_experiment_metastore_raftlogpb_raflog_proto_goTypes = []any{ + (CommandType)(0), // 0: raft_log.CommandType + (*RaftLogEntry)(nil), // 1: raft_log.RaftLogEntry + (*TruncateCommand)(nil), // 2: raft_log.TruncateCommand +} +var file_experiment_metastore_raftlogpb_raflog_proto_depIdxs = []int32{ + 0, // 0: raft_log.RaftLogEntry.type:type_name -> raft_log.CommandType + 1, // [1:1] is the sub-list for method output_type + 1, // [1:1] is the sub-list for method input_type + 1, // [1:1] is the sub-list for extension type_name + 1, // [1:1] is the sub-list for extension extendee + 0, // [0:1] is the sub-list for field type_name +} + +func init() { file_experiment_metastore_raftlogpb_raflog_proto_init() } +func file_experiment_metastore_raftlogpb_raflog_proto_init() { + if File_experiment_metastore_raftlogpb_raflog_proto != nil { + return + } + if !protoimpl.UnsafeEnabled { + file_experiment_metastore_raftlogpb_raflog_proto_msgTypes[0].Exporter = func(v any, i int) any { + switch v := v.(*RaftLogEntry); i { + case 0: + return &v.state + case 1: + return &v.sizeCache + case 2: + return &v.unknownFields + default: + return nil + } + } + file_experiment_metastore_raftlogpb_raflog_proto_msgTypes[1].Exporter = func(v any, i int) any { + switch v := v.(*TruncateCommand); i { + case 0: + return &v.state + case 1: + return &v.sizeCache + case 2: + return &v.unknownFields + default: + return nil + } + } + } + type x struct{} + out := protoimpl.TypeBuilder{ + File: protoimpl.DescBuilder{ + GoPackagePath: reflect.TypeOf(x{}).PkgPath(), + RawDescriptor: file_experiment_metastore_raftlogpb_raflog_proto_rawDesc, + NumEnums: 1, + NumMessages: 2, + NumExtensions: 0, + NumServices: 0, + }, + GoTypes: file_experiment_metastore_raftlogpb_raflog_proto_goTypes, + DependencyIndexes: file_experiment_metastore_raftlogpb_raflog_proto_depIdxs, + EnumInfos: file_experiment_metastore_raftlogpb_raflog_proto_enumTypes, + MessageInfos: file_experiment_metastore_raftlogpb_raflog_proto_msgTypes, + }.Build() + File_experiment_metastore_raftlogpb_raflog_proto = out.File + file_experiment_metastore_raftlogpb_raflog_proto_rawDesc = nil + file_experiment_metastore_raftlogpb_raflog_proto_goTypes = nil + file_experiment_metastore_raftlogpb_raflog_proto_depIdxs = nil +} diff --git a/pkg/experiment/metastore/raftlogpb/raflog.proto b/pkg/experiment/metastore/raftlogpb/raflog.proto new file mode 100644 index 0000000000..fa41df2fd9 --- /dev/null +++ b/pkg/experiment/metastore/raftlogpb/raflog.proto @@ -0,0 +1,21 @@ +syntax = "proto3"; + +package raft_log; + +message RaftLogEntry { + CommandType type = 1; + bytes payload = 2; +} + +enum CommandType { + COMMAND_TYPE_UNKNOWN = 0; + COMMAND_TYPE_ADD_BLOCK = 1; + COMMAND_TYPE_POLL_COMPACTION_JOBS_STATUS = 2; + + // This is a temporary solution. + COMMAND_TYPE_TRUNCATE = 4196; +} + +message TruncateCommand { + uint64 timestamp = 1; +} diff --git a/pkg/experiment/metastore/raftlogpb/raflog_vtproto.pb.go b/pkg/experiment/metastore/raftlogpb/raflog_vtproto.pb.go new file mode 100644 index 0000000000..283c760666 --- /dev/null +++ b/pkg/experiment/metastore/raftlogpb/raflog_vtproto.pb.go @@ -0,0 +1,307 @@ +// Code generated by protoc-gen-go-vtproto. DO NOT EDIT. +// protoc-gen-go-vtproto version: v0.6.0 +// source: experiment/metastore/raftlogpb/raflog.proto + +package raftlogpb + +import ( + fmt "fmt" + protohelpers "github.com/planetscale/vtprotobuf/protohelpers" + protoimpl "google.golang.org/protobuf/runtime/protoimpl" + io "io" +) + +const ( + // Verify that this generated code is sufficiently up-to-date. + _ = protoimpl.EnforceVersion(20 - protoimpl.MinVersion) + // Verify that runtime/protoimpl is sufficiently up-to-date. + _ = protoimpl.EnforceVersion(protoimpl.MaxVersion - 20) +) + +func (m *RaftLogEntry) MarshalVT() (dAtA []byte, err error) { + if m == nil { + return nil, nil + } + size := m.SizeVT() + dAtA = make([]byte, size) + n, err := m.MarshalToSizedBufferVT(dAtA[:size]) + if err != nil { + return nil, err + } + return dAtA[:n], nil +} + +func (m *RaftLogEntry) MarshalToVT(dAtA []byte) (int, error) { + size := m.SizeVT() + return m.MarshalToSizedBufferVT(dAtA[:size]) +} + +func (m *RaftLogEntry) MarshalToSizedBufferVT(dAtA []byte) (int, error) { + if m == nil { + return 0, nil + } + i := len(dAtA) + _ = i + var l int + _ = l + if m.unknownFields != nil { + i -= len(m.unknownFields) + copy(dAtA[i:], m.unknownFields) + } + if len(m.Payload) > 0 { + i -= len(m.Payload) + copy(dAtA[i:], m.Payload) + i = protohelpers.EncodeVarint(dAtA, i, uint64(len(m.Payload))) + i-- + dAtA[i] = 0x12 + } + if m.Type != 0 { + i = protohelpers.EncodeVarint(dAtA, i, uint64(m.Type)) + i-- + dAtA[i] = 0x8 + } + return len(dAtA) - i, nil +} + +func (m *TruncateCommand) MarshalVT() (dAtA []byte, err error) { + if m == nil { + return nil, nil + } + size := m.SizeVT() + dAtA = make([]byte, size) + n, err := m.MarshalToSizedBufferVT(dAtA[:size]) + if err != nil { + return nil, err + } + return dAtA[:n], nil +} + +func (m *TruncateCommand) MarshalToVT(dAtA []byte) (int, error) { + size := m.SizeVT() + return m.MarshalToSizedBufferVT(dAtA[:size]) +} + +func (m *TruncateCommand) MarshalToSizedBufferVT(dAtA []byte) (int, error) { + if m == nil { + return 0, nil + } + i := len(dAtA) + _ = i + var l int + _ = l + if m.unknownFields != nil { + i -= len(m.unknownFields) + copy(dAtA[i:], m.unknownFields) + } + if m.Timestamp != 0 { + i = protohelpers.EncodeVarint(dAtA, i, uint64(m.Timestamp)) + i-- + dAtA[i] = 0x8 + } + return len(dAtA) - i, nil +} + +func (m *RaftLogEntry) SizeVT() (n int) { + if m == nil { + return 0 + } + var l int + _ = l + if m.Type != 0 { + n += 1 + protohelpers.SizeOfVarint(uint64(m.Type)) + } + l = len(m.Payload) + if l > 0 { + n += 1 + l + protohelpers.SizeOfVarint(uint64(l)) + } + n += len(m.unknownFields) + return n +} + +func (m *TruncateCommand) SizeVT() (n int) { + if m == nil { + return 0 + } + var l int + _ = l + if m.Timestamp != 0 { + n += 1 + protohelpers.SizeOfVarint(uint64(m.Timestamp)) + } + n += len(m.unknownFields) + return n +} + +func (m *RaftLogEntry) UnmarshalVT(dAtA []byte) error { + l := len(dAtA) + iNdEx := 0 + for iNdEx < l { + preIndex := iNdEx + var wire uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + wire |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + fieldNum := int32(wire >> 3) + wireType := int(wire & 0x7) + if wireType == 4 { + return fmt.Errorf("proto: RaftLogEntry: wiretype end group for non-group") + } + if fieldNum <= 0 { + return fmt.Errorf("proto: RaftLogEntry: illegal tag %d (wire type %d)", fieldNum, wire) + } + switch fieldNum { + case 1: + if wireType != 0 { + return fmt.Errorf("proto: wrong wireType = %d for field Type", wireType) + } + m.Type = 0 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + m.Type |= CommandType(b&0x7F) << shift + if b < 0x80 { + break + } + } + case 2: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field Payload", wireType) + } + var byteLen int + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + byteLen |= int(b&0x7F) << shift + if b < 0x80 { + break + } + } + if byteLen < 0 { + return protohelpers.ErrInvalidLength + } + postIndex := iNdEx + byteLen + if postIndex < 0 { + return protohelpers.ErrInvalidLength + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + m.Payload = append(m.Payload[:0], dAtA[iNdEx:postIndex]...) + if m.Payload == nil { + m.Payload = []byte{} + } + iNdEx = postIndex + default: + iNdEx = preIndex + skippy, err := protohelpers.Skip(dAtA[iNdEx:]) + if err != nil { + return err + } + if (skippy < 0) || (iNdEx+skippy) < 0 { + return protohelpers.ErrInvalidLength + } + if (iNdEx + skippy) > l { + return io.ErrUnexpectedEOF + } + m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...) + iNdEx += skippy + } + } + + if iNdEx > l { + return io.ErrUnexpectedEOF + } + return nil +} +func (m *TruncateCommand) UnmarshalVT(dAtA []byte) error { + l := len(dAtA) + iNdEx := 0 + for iNdEx < l { + preIndex := iNdEx + var wire uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + wire |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + fieldNum := int32(wire >> 3) + wireType := int(wire & 0x7) + if wireType == 4 { + return fmt.Errorf("proto: TruncateCommand: wiretype end group for non-group") + } + if fieldNum <= 0 { + return fmt.Errorf("proto: TruncateCommand: illegal tag %d (wire type %d)", fieldNum, wire) + } + switch fieldNum { + case 1: + if wireType != 0 { + return fmt.Errorf("proto: wrong wireType = %d for field Timestamp", wireType) + } + m.Timestamp = 0 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return protohelpers.ErrIntOverflow + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + m.Timestamp |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + default: + iNdEx = preIndex + skippy, err := protohelpers.Skip(dAtA[iNdEx:]) + if err != nil { + return err + } + if (skippy < 0) || (iNdEx+skippy) < 0 { + return protohelpers.ErrInvalidLength + } + if (iNdEx + skippy) > l { + return io.ErrUnexpectedEOF + } + m.unknownFields = append(m.unknownFields, dAtA[iNdEx:iNdEx+skippy]...) + iNdEx += skippy + } + } + + if iNdEx > l { + return io.ErrUnexpectedEOF + } + return nil +} diff --git a/pkg/experiment/querybackend/backend.go b/pkg/experiment/querybackend/backend.go new file mode 100644 index 0000000000..40ec1dd132 --- /dev/null +++ b/pkg/experiment/querybackend/backend.go @@ -0,0 +1,146 @@ +package querybackend + +import ( + "context" + "flag" + "fmt" + + "github.com/go-kit/log" + "github.com/grafana/dskit/grpcclient" + "github.com/grafana/dskit/services" + "github.com/opentracing/opentracing-go" + "github.com/prometheus/client_golang/prometheus" + "go.uber.org/atomic" + "golang.org/x/sync/errgroup" + "google.golang.org/grpc/codes" + "google.golang.org/grpc/status" + + metastorev1 "github.com/grafana/pyroscope/api/gen/proto/go/metastore/v1" + querybackendv1 "github.com/grafana/pyroscope/api/gen/proto/go/querybackend/v1" + "github.com/grafana/pyroscope/pkg/experiment/querybackend/queryplan" + "github.com/grafana/pyroscope/pkg/iter" + "github.com/grafana/pyroscope/pkg/util" +) + +const defaultConcurrencyLimit = 25 + +type Config struct { + Address string `yaml:"address"` + GRPCClientConfig grpcclient.Config `yaml:"grpc_client_config" doc:"description=Configures the gRPC client used to communicate between the query-frontends and the query-schedulers."` +} + +func (cfg *Config) RegisterFlags(f *flag.FlagSet) { + f.StringVar(&cfg.Address, "query-backend.address", "localhost:9095", "") + cfg.GRPCClientConfig.RegisterFlagsWithPrefix("query-backend.grpc-client-config", f) +} + +func (cfg *Config) Validate() error { + if cfg.Address == "" { + return fmt.Errorf("query-backend.address is required") + } + return cfg.GRPCClientConfig.Validate() +} + +type QueryHandler interface { + Invoke(context.Context, *querybackendv1.InvokeRequest) (*querybackendv1.InvokeResponse, error) +} + +type QueryBackend struct { + service services.Service + querybackendv1.QueryBackendServiceServer + + config Config + logger log.Logger + reg prometheus.Registerer + + backendClient QueryHandler + blockReader QueryHandler + + concurrency uint32 + running atomic.Uint32 +} + +func New( + config Config, + logger log.Logger, + reg prometheus.Registerer, + backendClient QueryHandler, + blockReader QueryHandler, +) (*QueryBackend, error) { + q := QueryBackend{ + config: config, + logger: logger, + reg: reg, + backendClient: backendClient, + blockReader: blockReader, + + concurrency: defaultConcurrencyLimit, + } + q.service = services.NewIdleService(q.starting, q.stopping) + return &q, nil +} + +func (q *QueryBackend) Service() services.Service { return q.service } +func (q *QueryBackend) starting(context.Context) error { return nil } +func (q *QueryBackend) stopping(error) error { return nil } + +func (q *QueryBackend) Invoke( + ctx context.Context, + req *querybackendv1.InvokeRequest, +) (*querybackendv1.InvokeResponse, error) { + span, ctx := opentracing.StartSpanFromContext(ctx, "QueryBackend.Invoke") + defer span.Finish() + + p := queryplan.Open(req.QueryPlan) + switch r := p.Root(); r.Type { + case queryplan.NodeMerge: + return q.merge(ctx, req, r.Children()) + case queryplan.NodeRead: + return q.withThrottling(func() (*querybackendv1.InvokeResponse, error) { + return q.read(ctx, req, r.Blocks()) + }) + default: + panic("query plan: unknown node type") + } +} + +func (q *QueryBackend) merge( + ctx context.Context, + request *querybackendv1.InvokeRequest, + children iter.Iterator[*queryplan.Node], +) (*querybackendv1.InvokeResponse, error) { + request.QueryPlan = nil + m := newAggregator(request) + g, ctx := errgroup.WithContext(ctx) + for children.Next() { + req := request.CloneVT() + req.QueryPlan = children.At().Plan().Proto() + g.Go(util.RecoverPanic(func() error { + // TODO: Speculative retry. + return m.aggregateResponse(q.backendClient.Invoke(ctx, req)) + })) + } + if err := g.Wait(); err != nil { + return nil, err + } + return m.response() +} + +func (q *QueryBackend) read( + ctx context.Context, + request *querybackendv1.InvokeRequest, + blocks iter.Iterator[*metastorev1.BlockMeta], +) (*querybackendv1.InvokeResponse, error) { + request.QueryPlan = &querybackendv1.QueryPlan{ + Blocks: iter.MustSlice(blocks), + } + return q.blockReader.Invoke(ctx, request) +} + +func (q *QueryBackend) withThrottling(fn func() (*querybackendv1.InvokeResponse, error)) (*querybackendv1.InvokeResponse, error) { + if q.running.Inc() > q.concurrency { + return nil, status.Error(codes.ResourceExhausted, "all minions are busy, please try later") + } + defer q.running.Dec() + return fn() +} diff --git a/pkg/experiment/querybackend/block/compaction.go b/pkg/experiment/querybackend/block/compaction.go new file mode 100644 index 0000000000..582215e2d1 --- /dev/null +++ b/pkg/experiment/querybackend/block/compaction.go @@ -0,0 +1,606 @@ +package block + +import ( + "context" + "crypto/rand" + "fmt" + "os" + "path/filepath" + "slices" + "sort" + "strings" + "sync" + "time" + + "github.com/grafana/dskit/multierror" + "github.com/oklog/ulid" + "github.com/parquet-go/parquet-go" + "github.com/prometheus/common/model" + "github.com/prometheus/prometheus/storage" + "golang.org/x/sync/errgroup" + + metastorev1 "github.com/grafana/pyroscope/api/gen/proto/go/metastore/v1" + phlaremodel "github.com/grafana/pyroscope/pkg/model" + "github.com/grafana/pyroscope/pkg/objstore" + "github.com/grafana/pyroscope/pkg/phlaredb/block" + schemav1 "github.com/grafana/pyroscope/pkg/phlaredb/schemas/v1" + "github.com/grafana/pyroscope/pkg/phlaredb/symdb" + "github.com/grafana/pyroscope/pkg/phlaredb/tsdb/index" + "github.com/grafana/pyroscope/pkg/util" +) + +var ( + ErrNoBlocksToMerge = fmt.Errorf("no blocks to merge") + ErrShardMergeMismatch = fmt.Errorf("only blocks from the same shard can be merged") +) + +type CompactionOption func(*compactionConfig) + +func WithCompactionObjectOptions(options ...ObjectOption) CompactionOption { + return func(p *compactionConfig) { + p.objectOptions = append(p.objectOptions, options...) + } +} + +func WithCompactionTempDir(tempdir string) CompactionOption { + return func(p *compactionConfig) { + p.tempdir = tempdir + } +} + +func WithCompactionDestination(storage objstore.Bucket) CompactionOption { + return func(p *compactionConfig) { + p.destination = storage + } +} + +type compactionConfig struct { + objectOptions []ObjectOption + tempdir string + source objstore.BucketReader + destination objstore.Bucket +} + +func Compact( + ctx context.Context, + blocks []*metastorev1.BlockMeta, + storage objstore.Bucket, + options ...CompactionOption, +) (m []*metastorev1.BlockMeta, err error) { + c := &compactionConfig{ + tempdir: os.TempDir(), + source: storage, + destination: storage, + } + for _, option := range options { + option(c) + } + + objects := ObjectsFromMetas(storage, blocks, c.objectOptions...) + plan, err := PlanCompaction(objects) + if err != nil { + return nil, err + } + + if err = objects.Open(ctx); err != nil { + return nil, err + } + defer func() { + err = multierror.New(err, objects.Close()).Err() + }() + + compacted := make([]*metastorev1.BlockMeta, 0, len(plan)) + for _, p := range plan { + md, compactionErr := p.Compact(ctx, c.destination, c.tempdir) + if compactionErr != nil { + return nil, compactionErr + } + compacted = append(compacted, md) + } + + return compacted, nil +} + +func PlanCompaction(objects Objects) ([]*CompactionPlan, error) { + if len(objects) == 0 { + // Even if there's just a single object, we still need to rewrite it. + return nil, ErrNoBlocksToMerge + } + + r := objects[0] + var c uint32 + for _, obj := range objects { + if r.meta.Shard != obj.meta.Shard { + return nil, ErrShardMergeMismatch + } + c = max(c, obj.meta.CompactionLevel) + } + c++ + + m := make(map[string]*CompactionPlan) + for _, obj := range objects { + for _, s := range obj.meta.Datasets { + tm, ok := m[s.TenantId] + if !ok { + tm = newBlockCompaction(s.TenantId, r.meta.Shard, c) + m[s.TenantId] = tm + } + sm := tm.addDataset(s) + // Bind objects to datasets. + sm.append(NewDataset(s, obj)) + } + } + + ordered := make([]*CompactionPlan, 0, len(m)) + for _, tm := range m { + ordered = append(ordered, tm) + slices.SortFunc(tm.datasets, func(a, b *datasetCompaction) int { + return strings.Compare(a.meta.Name, b.meta.Name) + }) + } + slices.SortFunc(ordered, func(a, b *CompactionPlan) int { + return strings.Compare(a.tenantID, b.tenantID) + }) + + return ordered, nil +} + +type CompactionPlan struct { + tenantID string + datasetMap map[string]*datasetCompaction + datasets []*datasetCompaction + meta *metastorev1.BlockMeta +} + +func newBlockCompaction(tenantID string, shard uint32, compactionLevel uint32) *CompactionPlan { + return &CompactionPlan{ + tenantID: tenantID, + datasetMap: make(map[string]*datasetCompaction), + meta: &metastorev1.BlockMeta{ + FormatVersion: 1, + // TODO(kolesnikovae): Make it deterministic? + Id: ulid.MustNew(uint64(time.Now().UnixMilli()), rand.Reader).String(), + TenantId: tenantID, + Shard: shard, + CompactionLevel: compactionLevel, + Datasets: nil, + MinTime: 0, + MaxTime: 0, + Size: 0, + }, + } +} + +func (b *CompactionPlan) Estimate() { + // TODO(kolesnikovae): Implement. +} + +func (b *CompactionPlan) Compact(ctx context.Context, dst objstore.Bucket, tmpdir string) (m *metastorev1.BlockMeta, err error) { + w := NewBlockWriter(dst, ObjectPath(b.meta), tmpdir) + defer func() { + err = multierror.New(err, w.Close()).Err() + }() + // Datasets are compacted in a strict order. + for _, s := range b.datasets { + s.estimate() + // TODO(kolesnikovae): Wait until the required resources are available? + if err = s.compact(ctx, w); err != nil { + return nil, fmt.Errorf("compacting block: %w", err) + } + b.meta.Datasets = append(b.meta.Datasets, s.meta) + } + if err = w.Flush(ctx); err != nil { + return nil, fmt.Errorf("flushing block writer: %w", err) + } + b.meta.Size = w.Offset() + return b.meta, nil +} + +func (b *CompactionPlan) addDataset(s *metastorev1.Dataset) *datasetCompaction { + sm, ok := b.datasetMap[s.Name] + if !ok { + sm = newDatasetCompaction(s.TenantId, s.Name) + b.datasetMap[s.Name] = sm + b.datasets = append(b.datasets, sm) + } + if b.meta.MinTime == 0 || s.MinTime < b.meta.MinTime { + b.meta.MinTime = s.MinTime + } + if s.MaxTime > b.meta.MaxTime { + b.meta.MaxTime = s.MaxTime + } + return sm +} + +type compactionEstimates struct { + inMemorySizeInputSymbols int64 + inMemorySizeInputIndex int64 + inMemorySizeInputProfiles int64 + + inMemorySizeOutputSymbols int64 + inMemorySizeOutputIndex int64 + inMemorySizeOutputProfiles int64 + + outputSizeIndex int64 + outputSizeSymbols int64 + outputSizeProfiles int64 +} + +func (m *compactionEstimates) inMemorySizeTotal() int64 { + return m.inMemorySizeInputSymbols + + m.inMemorySizeInputIndex + + m.inMemorySizeInputProfiles + + m.inMemorySizeOutputSymbols + + m.inMemorySizeOutputIndex + + m.inMemorySizeOutputProfiles +} + +type datasetCompaction struct { + meta *metastorev1.Dataset + ptypes map[string]struct{} + path string // Set at open. + + datasets []*Dataset + + indexRewriter *indexRewriter + symbolsRewriter *symbolsRewriter + profilesWriter *profilesWriter + + estimates compactionEstimates + samples uint64 + series uint64 + profiles uint64 + + flushOnce sync.Once +} + +func newDatasetCompaction(tenantID, name string) *datasetCompaction { + return &datasetCompaction{ + ptypes: make(map[string]struct{}, 10), + meta: &metastorev1.Dataset{ + TenantId: tenantID, + Name: name, + // Updated at append. + MinTime: 0, + MaxTime: 0, + // Updated at writeTo. + TableOfContents: nil, + Size: 0, + ProfileTypes: nil, + }, + } +} + +func (m *datasetCompaction) append(s *Dataset) { + m.datasets = append(m.datasets, s) + if m.meta.MinTime == 0 || s.meta.MinTime < m.meta.MinTime { + m.meta.MinTime = s.meta.MinTime + } + if s.meta.MaxTime > m.meta.MaxTime { + m.meta.MaxTime = s.meta.MaxTime + } + for _, pt := range s.meta.ProfileTypes { + m.ptypes[pt] = struct{}{} + } +} + +func (m *datasetCompaction) compact(ctx context.Context, w *Writer) (err error) { + if err = m.open(ctx, w.Dir()); err != nil { + return fmt.Errorf("failed to open sections for compaction: %w", err) + } + defer func() { + err = multierror.New(err, m.cleanup()).Err() + }() + if err = m.mergeAndClose(ctx); err != nil { + return fmt.Errorf("failed to merge profiles: %w", err) + } + if err = m.writeTo(w); err != nil { + return fmt.Errorf("failed to write sections: %w", err) + } + return nil +} + +// TODO(kolesnikovae): +// - Add statistics to the block meta. +// - Measure. Ideally, we should track statistics. +func (m *datasetCompaction) estimate() { + columns := len(schemav1.ProfilesSchema.Columns()) + // Datasets are to be opened concurrently. + for _, s := range m.datasets { + s1 := s.sectionSize(SectionSymbols) + // It's likely that both symbols and tsdb sections will + // be heavily deduplicated, so the actual output size will + // be smaller than we estimate – to be deduced later. + m.estimates.outputSizeSymbols += s1 + // Both the symbols and the tsdb are loaded into memory entirely. + // It's multiplied here according to experiments. + // https://gist.github.com/kolesnikovae/6f7bdc0b8a14174a8e63485300144b4a + m.estimates.inMemorySizeInputSymbols += s1 * 3 // Pessimistic estimate. + + s2 := s.sectionSize(SectionTSDB) + m.estimates.outputSizeIndex += s2 + // TSDB index is loaded into memory entirely, but is not decoded. + m.estimates.inMemorySizeInputIndex += int64(nextPowerOfTwo(uint32(s2))) + + s3 := s.sectionSize(SectionProfiles) + m.estimates.outputSizeProfiles += s3 + // All columns are to be opened. + // Assuming async read mode – 2 buffers per column: + m.estimates.inMemorySizeInputProfiles += int64(2 * columns * estimateReadBufferSize(s3)) + } + const symbolsDuplicationRatio = 0.5 // Two blocks are likely to have a half of symbols in common. + m.estimates.outputSizeSymbols = int64(float64(m.estimates.outputSizeSymbols) * symbolsDuplicationRatio) + // Duplication of series and profiles is ignored. + + // Output block memory footprint. + m.estimates.inMemorySizeOutputIndex = m.estimates.outputSizeIndex * 8 // A guess. We keep all labels in memory. + m.estimates.inMemorySizeOutputSymbols += m.estimates.outputSizeProfiles * 4 // Mind the lookup table of rewriter. + // This is the most difficult part to estimate. + // Parquet keeps ALL RG pages in memory. We have a limit of 10K rows per RG, + // therefore it's very likely, that the whole table will be loaded into memory, + // plus overhead of memory fragmentation. It's likely impossible to have a + // reasonable estimate here. + const rowSizeGuess = 2 << 10 + // Worst case should be appx ~32MB. If a doubled estimated output size is less than that, use it. + columnBuffers := int64(nextPowerOfTwo(maxRowsPerRowGroup * rowSizeGuess)) + if s := 2 * m.estimates.outputSizeProfiles; s < columnBuffers { + columnBuffers = s + } + pageBuffers := int64(columns * estimatePageBufferSize(m.estimates.outputSizeProfiles)) + m.estimates.inMemorySizeOutputProfiles += columnBuffers + pageBuffers +} + +func (m *datasetCompaction) open(ctx context.Context, path string) (err error) { + m.path = path + defer func() { + if err != nil { + err = multierror.New(err, m.cleanup()).Err() + } + }() + + if err = os.MkdirAll(m.path, 0o777); err != nil { + return err + } + + m.profilesWriter, err = newProfileWriter(m.path, m.estimates.outputSizeProfiles) + if err != nil { + return err + } + + m.indexRewriter = newIndexRewriter(m.path) + m.symbolsRewriter = newSymbolsRewriter(m.path) + + g, ctx := errgroup.WithContext(ctx) + for _, s := range m.datasets { + s := s + g.Go(util.RecoverPanic(func() error { + if openErr := s.Open(ctx, allSections...); openErr != nil { + return fmt.Errorf("opening tenant dataset (block %s): %w", s.obj.path, openErr) + } + return nil + })) + } + if err = g.Wait(); err != nil { + merr := multierror.New(err) + for _, s := range m.datasets { + merr.Add(s.Close()) + } + return merr.Err() + } + + return nil +} + +func (m *datasetCompaction) mergeAndClose(ctx context.Context) (err error) { + defer func() { + err = multierror.New(err, m.close()).Err() + }() + return m.merge(ctx) +} + +func (m *datasetCompaction) merge(ctx context.Context) (err error) { + rows, err := NewMergeRowProfileIterator(m.datasets) + if err != nil { + return err + } + defer func() { + err = multierror.New(err, rows.Close()).Err() + }() + var i int + for rows.Next() { + if i++; i%1000 == 0 { + if err = ctx.Err(); err != nil { + return err + } + } + if err = m.writeRow(rows.At()); err != nil { + return err + } + } + return rows.Err() +} + +func (m *datasetCompaction) writeRow(r ProfileEntry) (err error) { + if err = m.indexRewriter.rewriteRow(r); err != nil { + return err + } + if err = m.symbolsRewriter.rewriteRow(r); err != nil { + return err + } + return m.profilesWriter.writeRow(r) +} + +func (m *datasetCompaction) close() (err error) { + m.flushOnce.Do(func() { + merr := multierror.New() + merr.Add(m.symbolsRewriter.Flush()) + merr.Add(m.indexRewriter.Flush()) + merr.Add(m.profilesWriter.Close()) + m.samples = m.symbolsRewriter.samples + m.series = m.indexRewriter.NumSeries() + m.profiles = m.profilesWriter.profiles + m.symbolsRewriter = nil + m.indexRewriter = nil + m.profilesWriter = nil + // Note that m.datasets are closed by merge + // iterator as they reach the end of the profile + // table. We do it here again just in case. + // TODO(kolesnikovae): Double check error handling. + m.datasets = nil + err = merr.Err() + }) + return err +} + +func (m *datasetCompaction) writeTo(w *Writer) (err error) { + off := w.Offset() + m.meta.TableOfContents, err = w.ReadFromFiles( + FileNameProfilesParquet, + block.IndexFilename, + symdb.DefaultFileName, + ) + if err != nil { + return err + } + m.meta.Size = w.Offset() - off + m.meta.ProfileTypes = make([]string, 0, len(m.ptypes)) + for pt := range m.ptypes { + m.meta.ProfileTypes = append(m.meta.ProfileTypes, pt) + } + sort.Strings(m.meta.ProfileTypes) + return nil +} + +func (m *datasetCompaction) cleanup() error { + return os.RemoveAll(m.path) +} + +func newIndexRewriter(path string) *indexRewriter { + return &indexRewriter{ + symbols: make(map[string]struct{}), + path: path, + } +} + +type indexRewriter struct { + series []seriesLabels + symbols map[string]struct{} + chunks []index.ChunkMeta // one chunk per series + previousFp model.Fingerprint + + path string +} + +type seriesLabels struct { + labels phlaremodel.Labels + fingerprint model.Fingerprint +} + +func (rw *indexRewriter) rewriteRow(e ProfileEntry) error { + if rw.previousFp != e.Fingerprint || len(rw.series) == 0 { + series := e.Labels.Clone() + for _, l := range series { + rw.symbols[l.Name] = struct{}{} + rw.symbols[l.Value] = struct{}{} + } + rw.series = append(rw.series, seriesLabels{ + labels: series, + fingerprint: e.Fingerprint, + }) + rw.chunks = append(rw.chunks, index.ChunkMeta{ + MinTime: e.Timestamp, + MaxTime: e.Timestamp, + SeriesIndex: uint32(len(rw.series) - 1), + }) + rw.previousFp = e.Fingerprint + } + rw.chunks[len(rw.chunks)-1].MaxTime = e.Timestamp + e.Row.SetSeriesIndex(rw.chunks[len(rw.chunks)-1].SeriesIndex) + return nil +} + +func (rw *indexRewriter) NumSeries() uint64 { return uint64(len(rw.series)) } + +func (rw *indexRewriter) Flush() error { + w, err := index.NewWriterSize(context.Background(), + filepath.Join(rw.path, block.IndexFilename), + // There is no particular reason to use a buffer (bufio.Writer) + // larger than the default one when writing on disk + 4<<10) + if err != nil { + return err + } + + // Sort symbols + symbols := make([]string, 0, len(rw.symbols)) + for s := range rw.symbols { + symbols = append(symbols, s) + } + sort.Strings(symbols) + + // Add symbols + for _, symbol := range symbols { + if err = w.AddSymbol(symbol); err != nil { + return err + } + } + + // Add Series + for i, series := range rw.series { + if err = w.AddSeries(storage.SeriesRef(i), series.labels, series.fingerprint, rw.chunks[i]); err != nil { + return err + } + } + + return w.Close() +} + +type symbolsRewriter struct { + w *symdb.SymDB + rw map[*Dataset]*symdb.Rewriter + samples uint64 + + stacktraces []uint32 +} + +func newSymbolsRewriter(path string) *symbolsRewriter { + return &symbolsRewriter{ + rw: make(map[*Dataset]*symdb.Rewriter), + w: symdb.NewSymDB(symdb.DefaultConfig(). + WithVersion(symdb.FormatV3). + WithDirectory(path)), + } +} + +func (s *symbolsRewriter) rewriteRow(e ProfileEntry) (err error) { + rw := s.rewriterFor(e.Dataset) + e.Row.ForStacktraceIDsValues(func(values []parquet.Value) { + s.loadStacktraceIDs(values) + if err = rw.Rewrite(e.Row.StacktracePartitionID(), s.stacktraces); err != nil { + return + } + s.samples += uint64(len(values)) + for i, v := range values { + values[i] = parquet.Int64Value(int64(s.stacktraces[i])).Level(v.RepetitionLevel(), v.DefinitionLevel(), v.Column()) + } + }) + return err +} + +func (s *symbolsRewriter) rewriterFor(x *Dataset) *symdb.Rewriter { + rw, ok := s.rw[x] + if !ok { + rw = symdb.NewRewriter(s.w, x.Symbols()) + s.rw[x] = rw + } + return rw +} + +func (s *symbolsRewriter) loadStacktraceIDs(values []parquet.Value) { + s.stacktraces = slices.Grow(s.stacktraces[0:], len(values))[:len(values)] + for i := range values { + s.stacktraces[i] = values[i].Uint32() + } +} + +func (s *symbolsRewriter) Flush() error { return s.w.Flush() } diff --git a/pkg/experiment/querybackend/block/compaction_test.go b/pkg/experiment/querybackend/block/compaction_test.go new file mode 100644 index 0000000000..37de05f0d5 --- /dev/null +++ b/pkg/experiment/querybackend/block/compaction_test.go @@ -0,0 +1,38 @@ +package block + +import ( + "context" + "os" + "path/filepath" + "testing" + + "github.com/stretchr/testify/require" + "google.golang.org/protobuf/encoding/protojson" + + compactorv1 "github.com/grafana/pyroscope/api/gen/proto/go/compactor/v1" + "github.com/grafana/pyroscope/pkg/objstore/testutil" +) + +func Test_CompactBlocks(t *testing.T) { + ctx := context.Background() + bucket, _ := testutil.NewFilesystemBucket(t, ctx, "testdata") + + var blockMetas compactorv1.CompletedJob // same contract, can break in the future + blockMetasData, err := os.ReadFile("testdata/block-metas.json") + require.NoError(t, err) + err = protojson.Unmarshal(blockMetasData, &blockMetas) + require.NoError(t, err) + + dst, tempdir := testutil.NewFilesystemBucket(t, ctx, t.TempDir()) + compactedBlocks, err := Compact(ctx, blockMetas.Blocks, bucket, + WithCompactionDestination(dst), + WithCompactionTempDir(tempdir), + WithCompactionObjectOptions( + WithObjectDownload(filepath.Join(tempdir, "source")), + WithObjectMaxSizeLoadInMemory(0)), // Force download. + ) + + require.NoError(t, err) + require.Len(t, compactedBlocks, 1) + // TODO: Assertions. +} diff --git a/pkg/experiment/querybackend/block/constants.go b/pkg/experiment/querybackend/block/constants.go new file mode 100644 index 0000000000..92b390ce37 --- /dev/null +++ b/pkg/experiment/querybackend/block/constants.go @@ -0,0 +1,82 @@ +package block + +import ( + "github.com/grafana/pyroscope/pkg/tenant" +) + +const ( + DirPathSegment = "segments/" + DirPathBlock = "blocks/" + DirNameAnonTenant = tenant.DefaultTenantID + + FileNameProfilesParquet = "profiles.parquet" + FileNameDataObject = "block.bin" +) + +const ( + defaultObjectSizeLoadInMemory = 1 << 20 + defaultTenantDatasetSizeLoadInMemory = 1 << 20 + + maxRowsPerRowGroup = 10 << 10 + symbolsPrefetchSize = 32 << 10 + compactionCopyBufferSize = 32 << 10 +) + +func estimateReadBufferSize(s int64) int { + const minSize = 64 << 10 + const maxSize = 1 << 20 + // Parquet has global buffer map, where buffer size is key, + // so we want a low cardinality here. + e := nextPowerOfTwo(uint32(s / 10)) + if e < minSize { + return minSize + } + return int(min(e, maxSize)) +} + +// This is a verbatim copy of estimateReadBufferSize. +// It's kept for the sake of clarity and to avoid confusion. +func estimatePageBufferSize(s int64) int { + const minSize = 64 << 10 + const maxSize = 1 << 20 + e := nextPowerOfTwo(uint32(s / 10)) + if e < minSize { + return minSize + } + return int(min(e, maxSize)) +} + +func estimateFooterSize(size int64) int64 { + var s int64 + // as long as we don't keep the exact footer sizes in the meta estimate it + if size > 0 { + s = size / 10000 + } + // set a minimum footer size of 32KiB + if s < 32<<10 { + s = 32 << 10 + } + // set a maximum footer size of 512KiB + if s > 512<<10 { + s = 512 << 10 + } + // now check clamp it to the actual size of the whole object + if s > size { + s = size + } + return s +} + +func nextPowerOfTwo(n uint32) uint32 { + if n == 0 { + return 1 + } + n-- + n |= n >> 1 + n |= n >> 2 + n |= n >> 4 + n |= n >> 8 + n |= n >> 16 + n++ + return n +} diff --git a/pkg/experiment/querybackend/block/object.go b/pkg/experiment/querybackend/block/object.go new file mode 100644 index 0000000000..738c7ea2da --- /dev/null +++ b/pkg/experiment/querybackend/block/object.go @@ -0,0 +1,243 @@ +package block + +import ( + "context" + "fmt" + "path/filepath" + "strconv" + "strings" + + "github.com/grafana/dskit/multierror" + "golang.org/x/sync/errgroup" + + metastorev1 "github.com/grafana/pyroscope/api/gen/proto/go/metastore/v1" + "github.com/grafana/pyroscope/pkg/objstore" + "github.com/grafana/pyroscope/pkg/util" + "github.com/grafana/pyroscope/pkg/util/bufferpool" + "github.com/grafana/pyroscope/pkg/util/refctr" +) + +// TODO Next: +// - Buffer pool. +// - In-memory threshold option. +// - Store the object size in metadata. +// - Separate storages for segments and compacted blocks. +// - Local cache? Useful for all-in-one deployments. +// - Distributed cache. + +type Section uint32 + +const ( + // Table of contents sections. + _ Section = iota + SectionProfiles + SectionTSDB + SectionSymbols +) + +var allSections = []Section{ + SectionProfiles, + SectionTSDB, + SectionSymbols, +} + +var ( + // Version-specific. + sectionNames = [...][]string{1: {"invalid", "profiles", "tsdb", "symbols"}} + sectionIndices = [...][]int{1: {-1, 0, 1, 2}} +) + +func (sc Section) open(ctx context.Context, s *Dataset) (err error) { + switch sc { + case SectionTSDB: + return openTSDB(ctx, s) + case SectionSymbols: + return openSymbols(ctx, s) + case SectionProfiles: + return openProfileTable(ctx, s) + default: + panic(fmt.Sprintf("bug: unknown section: %d", sc)) + } +} + +// Object represents a block or a segment in the object storage. +type Object struct { + path string + meta *metastorev1.BlockMeta + storage objstore.BucketReader + local *objstore.ReadOnlyFile + + refs refctr.Counter + buf *bufferpool.Buffer + err error + + memSize int + downloadDir string +} + +type ObjectOption func(*Object) + +func WithObjectPath(path string) ObjectOption { + return func(obj *Object) { + obj.path = path + } +} + +func WithObjectMaxSizeLoadInMemory(size int) ObjectOption { + return func(obj *Object) { + obj.memSize = size + } +} + +func WithObjectDownload(dir string) ObjectOption { + return func(obj *Object) { + obj.downloadDir = dir + } +} + +func NewObject(storage objstore.Bucket, meta *metastorev1.BlockMeta, opts ...ObjectOption) *Object { + o := &Object{ + storage: storage, + meta: meta, + path: ObjectPath(meta), + memSize: defaultObjectSizeLoadInMemory, + } + for _, opt := range opts { + opt(o) + } + return o +} + +func ObjectPath(md *metastorev1.BlockMeta) string { + topLevel := DirPathBlock + tenantDirName := md.TenantId + if md.CompactionLevel == 0 { + topLevel = DirPathSegment + tenantDirName = DirNameAnonTenant + } + var b strings.Builder + b.WriteString(topLevel) + b.WriteString(strconv.Itoa(int(md.Shard))) + b.WriteByte('/') + b.WriteString(tenantDirName) + b.WriteByte('/') + b.WriteString(md.Id) + b.WriteByte('/') + b.WriteString(FileNameDataObject) + return b.String() +} + +// Open opens the object, loading the data into memory if it's small enough. +// +// Open may be called multiple times concurrently, but the +// object is only initialized once. While it is possible to open +// the object repeatedly after close, the caller must pass the +// failure reason to the "CloseWithError" call, preventing further +// use, if applicable. +func (obj *Object) Open(ctx context.Context) error { + return obj.refs.IncErr(func() error { + return obj.open(ctx) + }) +} + +func (obj *Object) open(ctx context.Context) (err error) { + if obj.err != nil { + // In case if the object has been already closed with an error, + // and then released, return the error immediately. + return obj.err + } + if len(obj.meta.Datasets) == 0 { + return nil + } + // Estimate the size of the sections to process, and load the + // data into memory, if it's small enough. + if obj.meta.Size > uint64(obj.memSize) { + // Otherwise, download the object to the local directory, + // if it's specified, and use the local file. + if obj.downloadDir != "" { + return obj.Download(ctx) + } + // The object will be read from the storage directly. + return nil + } + obj.buf = bufferpool.GetBuffer(int(obj.meta.Size)) + defer func() { + if err != nil { + _ = obj.closeErr(err) + } + }() + if err = objstore.ReadRange(ctx, obj.buf, obj.path, obj.storage, 0, int64(obj.meta.Size)); err != nil { + return fmt.Errorf("loading object into memory %s: %w", obj.path, err) + } + return nil +} + +func (obj *Object) Close() error { + return obj.CloseWithError(nil) +} + +// CloseWithError closes the object, releasing all the acquired resources, +// once the last reference is released. If the provided error is not nil, +// the object will be marked as failed, preventing any further use. +func (obj *Object) CloseWithError(err error) (closeErr error) { + obj.refs.Dec(func() { + closeErr = obj.closeErr(err) + }) + return closeErr +} + +func (obj *Object) closeErr(err error) (closeErr error) { + obj.err = err + if obj.buf != nil { + bufferpool.Put(obj.buf) + obj.buf = nil + } + if obj.local != nil { + closeErr = obj.local.Close() + obj.local = nil + } + return closeErr +} + +func (obj *Object) Meta() *metastorev1.BlockMeta { return obj.meta } + +func (obj *Object) Download(ctx context.Context) error { + dir := filepath.Join(obj.downloadDir, obj.meta.Id) + local, err := objstore.Download(ctx, obj.path, obj.storage, dir) + if err != nil { + return err + } + obj.storage = local + obj.local = local + return nil +} + +// ObjectsFromMetas binds block metas to corresponding objects in the storage. +func ObjectsFromMetas(storage objstore.Bucket, blocks []*metastorev1.BlockMeta, options ...ObjectOption) Objects { + objects := make([]*Object, len(blocks)) + for i, m := range blocks { + objects[i] = NewObject(storage, m, options...) + } + return objects +} + +type Objects []*Object + +func (s Objects) Open(ctx context.Context) error { + g, ctx := errgroup.WithContext(ctx) + for i := range s { + i := i + g.Go(util.RecoverPanic(func() error { + return s[i].Open(ctx) + })) + } + return g.Wait() +} + +func (s Objects) Close() error { + var m multierror.MultiError + for i := range s { + m.Add(s[i].Close()) + } + return m.Err() +} diff --git a/pkg/experiment/querybackend/block/section_profiles.go b/pkg/experiment/querybackend/block/section_profiles.go new file mode 100644 index 0000000000..031bcbf78b --- /dev/null +++ b/pkg/experiment/querybackend/block/section_profiles.go @@ -0,0 +1,389 @@ +package block + +import ( + "context" + "encoding/binary" + "fmt" + "io" + "math" + "os" + "path/filepath" + + "github.com/parquet-go/parquet-go" + "github.com/pkg/errors" + "github.com/prometheus/common/model" + + "github.com/grafana/pyroscope/pkg/iter" + phlaremodel "github.com/grafana/pyroscope/pkg/model" + "github.com/grafana/pyroscope/pkg/objstore" + phlareparquet "github.com/grafana/pyroscope/pkg/parquet" + "github.com/grafana/pyroscope/pkg/phlaredb" + "github.com/grafana/pyroscope/pkg/phlaredb/query" + schemav1 "github.com/grafana/pyroscope/pkg/phlaredb/schemas/v1" + "github.com/grafana/pyroscope/pkg/phlaredb/tsdb/index" + "github.com/grafana/pyroscope/pkg/util/bufferpool" + "github.com/grafana/pyroscope/pkg/util/build" + "github.com/grafana/pyroscope/pkg/util/loser" +) + +func openProfileTable(_ context.Context, s *Dataset) (err error) { + offset := s.sectionOffset(SectionProfiles) + size := s.sectionSize(SectionProfiles) + if buf := s.inMemoryBuffer(); buf != nil { + offset -= int64(s.offset()) + s.profiles, err = openParquetFile( + s.inMemoryBucket(buf), s.obj.path, offset, size, + 0, // Do not prefetch the footer. + parquet.SkipBloomFilters(true), + parquet.FileReadMode(parquet.ReadModeSync), + parquet.ReadBufferSize(4<<10)) + } else { + s.profiles, err = openParquetFile( + s.obj.storage, s.obj.path, offset, size, + estimateFooterSize(size), + parquet.SkipBloomFilters(true), + parquet.FileReadMode(parquet.ReadModeAsync), + parquet.ReadBufferSize(estimateReadBufferSize(size))) + } + if err != nil { + return fmt.Errorf("opening profile parquet table: %w", err) + } + return nil +} + +type ParquetFile struct { + *parquet.File + + reader objstore.ReaderAtCloser + cancel context.CancelFunc + + storage objstore.BucketReader + path string + off int64 + size int64 +} + +func openParquetFile( + storage objstore.BucketReader, + path string, + offset, size, footerSize int64, + options ...parquet.FileOption, +) (p *ParquetFile, err error) { + // The context is used for GetRange calls and should not + // be canceled until the parquet file is closed. + ctx, cancel := context.WithCancel(context.Background()) + defer func() { + if err != nil { + cancel() + } + }() + + p = &ParquetFile{ + cancel: cancel, + storage: storage, + path: path, + off: offset, + size: size, + } + + r, err := storage.ReaderAt(ctx, path) + if err != nil { + return nil, fmt.Errorf("creating object reader: %w", err) + } + + var ra io.ReaderAt + ra = io.NewSectionReader(r, offset, size) + if footerSize > 0 { + buf := bufferpool.GetBuffer(int(footerSize)) + defer func() { + // Footer is not used after the file was opened. + bufferpool.Put(buf) + }() + if err = p.fetchFooter(ctx, buf, footerSize); err != nil { + return nil, err + } + rf := newReaderWithFooter(ra, buf.B, size) + defer rf.free() + ra = rf + } + + f, err := parquet.OpenFile(ra, size, options...) + if err != nil { + return nil, err + } + + p.reader = r + p.File = f + return p, nil +} + +func (f *ParquetFile) RowReader() *parquet.Reader { + return parquet.NewReader(f.File, schemav1.ProfilesSchema) +} + +func (f *ParquetFile) fetchFooter(ctx context.Context, buf *bufferpool.Buffer, estimatedSize int64) error { + // Fetch the footer of estimated size at the estimated offset. + estimatedOffset := f.off + f.size - estimatedSize + if err := objstore.ReadRange(ctx, buf, f.path, f.storage, estimatedOffset, estimatedSize); err != nil { + return err + } + // Footer size is an uint32 located at size-8. + sb := buf.B[len(buf.B)-8 : len(buf.B)-4] + s := int64(binary.LittleEndian.Uint32(sb)) + s += 8 // Include the footer size itself and the magic signature. + if estimatedSize >= s { + // The footer has been fetched. + return nil + } + // Fetch footer to buf for sure. + return objstore.ReadRange(ctx, buf, f.path, f.storage, f.off+f.size-s, s) +} + +func (f *ParquetFile) Close() error { + if f.cancel != nil { + f.cancel() + } + if f.reader != nil { + return f.reader.Close() + } + return nil +} + +func (f *ParquetFile) Column(ctx context.Context, columnName string, predicate query.Predicate) query.Iterator { + idx, _ := query.GetColumnIndexByPath(f.Root(), columnName) + if idx == -1 { + return query.NewErrIterator(fmt.Errorf("column '%s' not found in parquet table", columnName)) + } + return query.NewSyncIterator(ctx, f.RowGroups(), idx, columnName, 1<<10, predicate, columnName) +} + +type profilesWriter struct { + *parquet.GenericWriter[*schemav1.Profile] + file *os.File + buf []parquet.Row + profiles uint64 +} + +func newProfileWriter(dst string, sizeTotal int64) (*profilesWriter, error) { + f, err := os.Create(filepath.Join(dst, FileNameProfilesParquet)) + if err != nil { + return nil, err + } + return &profilesWriter{ + file: f, + buf: make([]parquet.Row, 1), + GenericWriter: parquet.NewGenericWriter[*schemav1.Profile](f, + parquet.CreatedBy("github.com/grafana/pyroscope/", build.Version, build.Revision), + parquet.PageBufferSize(estimatePageBufferSize(sizeTotal)), + // Note that parquet keeps ALL RG pages in memory (ColumnPageBuffers). + parquet.MaxRowsPerRowGroup(maxRowsPerRowGroup), + schemav1.ProfilesSchema, + // parquet.ColumnPageBuffers(), + ), + }, nil +} + +func (p *profilesWriter) writeRow(e ProfileEntry) error { + p.buf[0] = parquet.Row(e.Row) + _, err := p.GenericWriter.WriteRows(p.buf) + p.profiles++ + return err +} + +func (p *profilesWriter) Close() error { + err := p.GenericWriter.Close() + if err != nil { + return err + } + return p.file.Close() +} + +type readerWithFooter struct { + reader io.ReaderAt + footer []byte + offset int64 + size int64 +} + +func newReaderWithFooter(r io.ReaderAt, footer []byte, size int64) *readerWithFooter { + footerSize := int64(len(footer)) + footerOffset := size - footerSize + return &readerWithFooter{ + reader: r, + footer: footer, + offset: footerOffset, + size: footerSize, + } +} + +func (f *readerWithFooter) hitsHeaderMagic(off, length int64) bool { + return off == 0 && length == 4 +} + +func (f *readerWithFooter) hitsFooter(off, length int64) bool { + return length <= f.size && off >= f.offset && off+length <= f.offset+f.size +} + +var parquetMagic = []byte("PAR1") + +func (f *readerWithFooter) free() { + f.footer = nil + f.size = -1 +} + +func (f *readerWithFooter) ReadAt(p []byte, off int64) (n int, err error) { + if f.hitsHeaderMagic(off, int64(len(p))) { + copy(p, parquetMagic) + return len(p), nil + } + if f.hitsFooter(off, int64(len(p))) { + copy(p, f.footer[off-f.offset:]) + return len(p), nil + } + return f.reader.ReadAt(p, off) +} + +type ProfileEntry struct { + Dataset *Dataset + + Timestamp int64 + Fingerprint model.Fingerprint + Labels phlaremodel.Labels + Row schemav1.ProfileRow +} + +func NewMergeRowProfileIterator(src []*Dataset) (iter.Iterator[ProfileEntry], error) { + its := make([]iter.Iterator[ProfileEntry], len(src)) + for i, s := range src { + it, err := NewProfileRowIterator(s) + if err != nil { + return nil, err + } + its[i] = it + } + if len(its) == 1 { + return its[0], nil + } + return &DedupeProfileRowIterator{ + Iterator: iter.NewTreeIterator(loser.New( + its, + ProfileEntry{ + Timestamp: math.MaxInt64, + }, + func(it iter.Iterator[ProfileEntry]) ProfileEntry { return it.At() }, + func(r1, r2 ProfileEntry) bool { + // first handle max profileRow if it's either r1 or r2 + if r1.Timestamp == math.MaxInt64 { + return false + } + if r2.Timestamp == math.MaxInt64 { + return true + } + // then handle normal profileRows + if cmp := phlaremodel.CompareLabelPairs(r1.Labels, r2.Labels); cmp != 0 { + return cmp < 0 + } + return r1.Timestamp < r2.Timestamp + }, + func(it iter.Iterator[ProfileEntry]) { _ = it.Close() }, + )), + }, nil +} + +type DedupeProfileRowIterator struct { + iter.Iterator[ProfileEntry] + + prevFP model.Fingerprint + prevTimeNanos int64 +} + +func (it *DedupeProfileRowIterator) Next() bool { + for { + if !it.Iterator.Next() { + return false + } + currentProfile := it.Iterator.At() + if it.prevFP == currentProfile.Fingerprint && it.prevTimeNanos == currentProfile.Timestamp { + // skip duplicate profile + continue + } + it.prevFP = currentProfile.Fingerprint + it.prevTimeNanos = currentProfile.Timestamp + return true + } +} + +type profileRowIterator struct { + reader *Dataset + index phlaredb.IndexReader + profiles iter.Iterator[parquet.Row] + allPostings index.Postings + err error + + currentRow ProfileEntry + currentSeriesIdx uint32 + chunks []index.ChunkMeta +} + +func NewProfileRowIterator(s *Dataset) (iter.Iterator[ProfileEntry], error) { + k, v := index.AllPostingsKey() + tsdb := s.Index() + allPostings, err := tsdb.Postings(k, nil, v) + if err != nil { + return nil, err + } + return &profileRowIterator{ + reader: s, + index: tsdb, + profiles: phlareparquet.NewBufferedRowReaderIterator(s.ProfileRowReader(), 4), + allPostings: allPostings, + currentSeriesIdx: math.MaxUint32, + chunks: make([]index.ChunkMeta, 1), + }, nil +} + +func (p *profileRowIterator) At() ProfileEntry { + return p.currentRow +} + +func (p *profileRowIterator) Next() bool { + if !p.profiles.Next() { + return false + } + p.currentRow.Dataset = p.reader + p.currentRow.Row = schemav1.ProfileRow(p.profiles.At()) + seriesIndex := p.currentRow.Row.SeriesIndex() + p.currentRow.Timestamp = p.currentRow.Row.TimeNanos() + // do we have a new series? + if seriesIndex == p.currentSeriesIdx { + return true + } + p.currentSeriesIdx = seriesIndex + if !p.allPostings.Next() { + if err := p.allPostings.Err(); err != nil { + p.err = err + return false + } + p.err = errors.New("unexpected end of postings") + return false + } + + fp, err := p.index.Series(p.allPostings.At(), &p.currentRow.Labels, &p.chunks) + if err != nil { + p.err = err + return false + } + p.currentRow.Fingerprint = model.Fingerprint(fp) + return true +} + +func (p *profileRowIterator) Err() error { + if p.err != nil { + return p.err + } + return p.profiles.Err() +} + +func (p *profileRowIterator) Close() error { + return p.reader.Close() +} diff --git a/pkg/experiment/querybackend/block/section_symbols.go b/pkg/experiment/querybackend/block/section_symbols.go new file mode 100644 index 0000000000..e7626c88bb --- /dev/null +++ b/pkg/experiment/querybackend/block/section_symbols.go @@ -0,0 +1,24 @@ +package block + +import ( + "context" + "fmt" + + "github.com/grafana/pyroscope/pkg/phlaredb/symdb" +) + +func openSymbols(ctx context.Context, s *Dataset) (err error) { + offset := s.sectionOffset(SectionSymbols) + size := s.sectionSize(SectionSymbols) + if buf := s.inMemoryBuffer(); buf != nil { + offset -= int64(s.offset()) + s.symbols, err = symdb.OpenObject(ctx, s.inMemoryBucket(buf), s.obj.path, offset, size) + } else { + s.symbols, err = symdb.OpenObject(ctx, s.obj.storage, s.obj.path, offset, size, + symdb.WithPrefetchSize(symbolsPrefetchSize)) + } + if err != nil { + return fmt.Errorf("opening symbols: %w", err) + } + return nil +} diff --git a/pkg/experiment/querybackend/block/section_tsdb.go b/pkg/experiment/querybackend/block/section_tsdb.go new file mode 100644 index 0000000000..ac93674014 --- /dev/null +++ b/pkg/experiment/querybackend/block/section_tsdb.go @@ -0,0 +1,49 @@ +package block + +import ( + "context" + "fmt" + + "github.com/grafana/pyroscope/pkg/objstore" + "github.com/grafana/pyroscope/pkg/phlaredb/tsdb/index" + "github.com/grafana/pyroscope/pkg/util/bufferpool" +) + +func openTSDB(ctx context.Context, s *Dataset) (err error) { + offset := s.sectionOffset(SectionTSDB) + size := s.sectionSize(SectionTSDB) + s.tsdb = new(tsdbBuffer) + defer func() { + if err != nil { + _ = s.tsdb.Close() + } + }() + if buf := s.inMemoryBuffer(); buf != nil { + offset -= int64(s.offset()) + s.tsdb.index, err = index.NewReader(index.RealByteSlice(buf[offset : offset+size])) + } else { + s.tsdb.buf = bufferpool.GetBuffer(int(size)) + if err = objstore.ReadRange(ctx, s.tsdb.buf, s.obj.path, s.obj.storage, offset, size); err == nil { + s.tsdb.index, err = index.NewReader(index.RealByteSlice(s.tsdb.buf.B)) + } + } + if err != nil { + return fmt.Errorf("opening tsdb: %w", err) + } + return nil +} + +type tsdbBuffer struct { + index *index.Reader + buf *bufferpool.Buffer +} + +func (b *tsdbBuffer) Close() (err error) { + if b.buf != nil { + bufferpool.Put(b.buf) + } + if b.index != nil { + err = b.index.Close() + } + return err +} diff --git a/pkg/experiment/querybackend/block/tenant_service.go b/pkg/experiment/querybackend/block/tenant_service.go new file mode 100644 index 0000000000..47bfc470a1 --- /dev/null +++ b/pkg/experiment/querybackend/block/tenant_service.go @@ -0,0 +1,209 @@ +package block + +import ( + "context" + "fmt" + + "github.com/grafana/dskit/multierror" + "github.com/parquet-go/parquet-go" + "golang.org/x/sync/errgroup" + + metastorev1 "github.com/grafana/pyroscope/api/gen/proto/go/metastore/v1" + "github.com/grafana/pyroscope/pkg/objstore" + "github.com/grafana/pyroscope/pkg/objstore/providers/memory" + "github.com/grafana/pyroscope/pkg/phlaredb" + "github.com/grafana/pyroscope/pkg/phlaredb/symdb" + "github.com/grafana/pyroscope/pkg/util" + "github.com/grafana/pyroscope/pkg/util/bufferpool" + "github.com/grafana/pyroscope/pkg/util/refctr" +) + +type Dataset struct { + meta *metastorev1.Dataset + obj *Object + + refs refctr.Counter + buf *bufferpool.Buffer + err error + + tsdb *tsdbBuffer + symbols *symdb.Reader + profiles *ParquetFile + + memSize int +} + +func NewDataset(meta *metastorev1.Dataset, obj *Object) *Dataset { + return &Dataset{ + meta: meta, + obj: obj, + memSize: defaultTenantDatasetSizeLoadInMemory, + } +} + +type DatasetOption func(*Dataset) + +func WithDatasetMaxSizeLoadInMemory(size int) DatasetOption { + return func(s *Dataset) { + s.memSize = size + } +} + +// Open opens the dataset, initializing the sections specified. +// +// Open may be called multiple times concurrently, but the dataset +// is only initialized once. While it is possible to open the dataset +// repeatedly after close, the caller must pass the failure reason to +// the CloseWithError call, preventing further use, if applicable. +func (s *Dataset) Open(ctx context.Context, sections ...Section) (err error) { + return s.refs.IncErr(func() error { + return s.open(ctx, sections...) + }) +} + +func (s *Dataset) open(ctx context.Context, sections ...Section) (err error) { + if s.err != nil { + // The tenant dataset has been already closed with an error. + return s.err + } + if err = s.obj.Open(ctx); err != nil { + return fmt.Errorf("failed to open object: %w", err) + } + defer func() { + // Close the object here because the tenant dataset won't be + // closed if it fails to open. + if err != nil { + _ = s.closeErr(err) + } + }() + if s.obj.buf == nil && s.meta.Size < uint64(s.memSize) { + s.buf = bufferpool.GetBuffer(int(s.meta.Size)) + off, size := int64(s.offset()), int64(s.meta.Size) + if err = objstore.ReadRange(ctx, s.buf, s.obj.path, s.obj.storage, off, size); err != nil { + return fmt.Errorf("loading sections into memory: %w", err) + } + } + g, ctx := errgroup.WithContext(ctx) + for _, sc := range sections { + sc := sc + g.Go(util.RecoverPanic(func() error { + if openErr := sc.open(ctx, s); openErr != nil { + return fmt.Errorf("openning section %v: %w", s.sectionName(sc), openErr) + } + return nil + })) + } + return g.Wait() +} + +func (s *Dataset) Close() error { return s.CloseWithError(nil) } + +// CloseWithError closes the tenant dataset and disposes all the resources +// associated with it. +// +// Any further attempts to open the dataset will return the provided error. +func (s *Dataset) CloseWithError(err error) (closeErr error) { + s.refs.Dec(func() { + closeErr = s.closeErr(err) + }) + return closeErr +} + +func (s *Dataset) closeErr(err error) error { + s.err = err + if s.buf != nil { + bufferpool.Put(s.buf) + s.buf = nil + } + var merr multierror.MultiError + if s.tsdb != nil { + merr.Add(s.tsdb.Close()) + } + if s.symbols != nil { + merr.Add(s.symbols.Close()) + } + if s.profiles != nil { + merr.Add(s.profiles.Close()) + } + if s.obj != nil { + merr.Add(s.obj.CloseWithError(err)) + } + return merr.Err() +} + +func (s *Dataset) Meta() *metastorev1.Dataset { return s.meta } + +func (s *Dataset) Profiles() *ParquetFile { return s.profiles } + +func (s *Dataset) ProfileRowReader() parquet.RowReader { return s.profiles.RowReader() } + +func (s *Dataset) Symbols() symdb.SymbolsReader { return s.symbols } + +func (s *Dataset) Index() phlaredb.IndexReader { return s.tsdb.index } + +// Offset of the tenant dataset section within the object. +func (s *Dataset) offset() uint64 { return s.meta.TableOfContents[0] } + +func (s *Dataset) sectionIndex(sc Section) int { + var n []int + switch s.obj.meta.FormatVersion { + default: + n = sectionIndices[1] + } + if int(sc) >= len(n) { + panic(fmt.Sprintf("bug: invalid section index: %d (total: %d)", sc, len(n))) + } + return n[sc] +} + +func (s *Dataset) sectionName(sc Section) string { + var n []string + switch s.obj.meta.FormatVersion { + default: + n = sectionNames[1] + } + if int(sc) >= len(n) { + panic(fmt.Sprintf("bug: invalid section index: %d (total: %d)", sc, len(n))) + } + return n[sc] +} + +func (s *Dataset) sectionOffset(sc Section) int64 { + return int64(s.meta.TableOfContents[s.sectionIndex(sc)]) +} + +func (s *Dataset) sectionSize(sc Section) int64 { + idx := s.sectionIndex(sc) + off := s.meta.TableOfContents[idx] + var next uint64 + if idx == len(s.meta.TableOfContents)-1 { + next = s.offset() + s.meta.Size + } else { + next = s.meta.TableOfContents[idx+1] + } + return int64(next - off) +} + +func (s *Dataset) inMemoryBuffer() []byte { + if s.obj.buf != nil { + // If the entire object is loaded into memory, + // return the tenant dataset sub-slice. + lo := s.offset() + hi := lo + s.meta.Size + buf := s.obj.buf.B + return buf[lo:hi] + } + if s.buf != nil { + // Otherwise, if the tenant dataset is loaded into memory + // individually, return the buffer. + return s.buf.B + } + // Otherwise, the tenant dataset is not loaded into memory. + return nil +} + +func (s *Dataset) inMemoryBucket(buf []byte) objstore.Bucket { + bucket := memory.NewInMemBucket() + bucket.Set(s.obj.path, buf) + return objstore.NewBucket(bucket) +} diff --git a/pkg/experiment/querybackend/block/testdata/.gitignore b/pkg/experiment/querybackend/block/testdata/.gitignore new file mode 100644 index 0000000000..7ad764b60f --- /dev/null +++ b/pkg/experiment/querybackend/block/testdata/.gitignore @@ -0,0 +1 @@ +blocks/ diff --git a/pkg/experiment/querybackend/block/testdata/block-metas.json b/pkg/experiment/querybackend/block/testdata/block-metas.json new file mode 100644 index 0000000000..9762eaf057 --- /dev/null +++ b/pkg/experiment/querybackend/block/testdata/block-metas.json @@ -0,0 +1,216 @@ +{ + "blocks": [ + { + "id": "01J2VJR31PT3X4NDJC4Q2BHWQ1", + "minTime": "1721060035611", + "maxTime": "1721060035611", + "shard": 1, + "datasets": [ + { + "tenantId": "anonymous", + "name": "pyroscope-test/query-frontend", + "minTime": "1721060035611", + "maxTime": "1721060035611", + "tableOfContents": [ + "0", + "4549", + "7747" + ], + "size": "19471" + } + ] + }, + { + "id": "01J2VJQPYDC160REPAD2VN88XN", + "minTime": "1721060023235", + "maxTime": "1721060023235", + "shard": 1, + "datasets": [ + { + "tenantId": "anonymous", + "name": "pyroscope-test/ingester", + "minTime": "1721060023235", + "maxTime": "1721060023235", + "tableOfContents": [ + "0", + "3794", + "6281" + ], + "size": "22242" + } + ] + }, + { + "id": "01J2VJQRGBK8YFWVV8K1MPRRWM", + "minTime": "1721060010831", + "maxTime": "1721060010831", + "shard": 1, + "datasets": [ + { + "tenantId": "anonymous", + "name": "pyroscope-test/alloy", + "minTime": "1721060010831", + "maxTime": "1721060010831", + "tableOfContents": [ + "0", + "3949", + "6565" + ], + "size": "17664" + } + ] + }, + { + "id": "01J2VJQRTMSCY4VDYBP5N4N5JK", + "minTime": "1721060025159", + "maxTime": "1721060025159", + "shard": 1, + "datasets": [ + { + "tenantId": "anonymous", + "name": "pyroscope-test/query-frontend", + "minTime": "1721060025159", + "maxTime": "1721060025159", + "tableOfContents": [ + "0", + "3834", + "6029" + ], + "size": "21765" + } + ] + }, + { + "id": "01J2VJQTJ3PGF7KB39ARR1BX3Y", + "minTime": "1721060026913", + "maxTime": "1721060026913", + "shard": 1, + "datasets": [ + { + "tenantId": "anonymous", + "name": "pyroscope-test/ingester", + "minTime": "1721060026913", + "maxTime": "1721060026913", + "tableOfContents": [ + "0", + "4808", + "8321" + ], + "size": "28169" + } + ] + }, + { + "id": "01J2VJQV544TF571FDSK2H692P", + "minTime": "1721060013534", + "maxTime": "1721060013534", + "shard": 1, + "datasets": [ + { + "tenantId": "anonymous", + "name": "pyroscope-test/query-frontend", + "minTime": "1721060013534", + "maxTime": "1721060013534", + "tableOfContents": [ + "0", + "4201", + "6780" + ], + "size": "15785" + } + ] + }, + { + "id": "01J2VJQX8DYHSEBK7BAQSCJBMG", + "minTime": "1721060015603", + "maxTime": "1721060015603", + "shard": 1, + "datasets": [ + { + "tenantId": "anonymous", + "name": "pyroscope-test/ingester", + "minTime": "1721060015603", + "maxTime": "1721060015603", + "tableOfContents": [ + "0", + "4543", + "7406" + ], + "size": "27431" + } + ] + }, + { + "id": "01J2VJQYQVZTPZMMJKE7F2XC47", + "minTime": "1721060031203", + "maxTime": "1721060031203", + "shard": 1, + "datasets": [ + { + "tenantId": "anonymous", + "name": "pyroscope-test/ingester", + "minTime": "1721060031203", + "maxTime": "1721060031203", + "tableOfContents": [ + "0", + "5086", + "8600" + ], + "size": "36655" + } + ] + }, + { + "id": "01J2VJQZPARDJQ779S1JMV0XQA", + "minTime": "1721060032190", + "maxTime": "1721060032190", + "shard": 1, + "datasets": [ + { + "tenantId": "anonymous", + "name": "pyroscope-test/ingester", + "minTime": "1721060032190", + "maxTime": "1721060032190", + "tableOfContents": [ + "0", + "3825", + "6312" + ], + "size": "24273" + } + ] + }, + { + "id": "01J2VJR0R3NQS23SDADNA6XHCM", + "minTime": "1721060033248", + "maxTime": "1721060033802", + "shard": 1, + "datasets": [ + { + "tenantId": "anonymous", + "name": "pyroscope-test/alloy", + "minTime": "1721060033248", + "maxTime": "1721060033248", + "tableOfContents": [ + "0", + "4547", + "7152" + ], + "size": "27397" + }, + { + "tenantId": "anonymous", + "name": "pyroscope-test/ingester", + "minTime": "1721060033680", + "maxTime": "1721060033802", + "tableOfContents": [ + "27397", + "32839", + "37073" + ], + "size": "50561" + } + ] + } + ] +} diff --git a/pkg/experiment/querybackend/block/testdata/segments/1/anonymous/01J2VJQPYDC160REPAD2VN88XN/block.bin b/pkg/experiment/querybackend/block/testdata/segments/1/anonymous/01J2VJQPYDC160REPAD2VN88XN/block.bin new file mode 100644 index 0000000000..871f44e7b8 Binary files /dev/null and b/pkg/experiment/querybackend/block/testdata/segments/1/anonymous/01J2VJQPYDC160REPAD2VN88XN/block.bin differ diff --git a/pkg/experiment/querybackend/block/testdata/segments/1/anonymous/01J2VJQRGBK8YFWVV8K1MPRRWM/block.bin b/pkg/experiment/querybackend/block/testdata/segments/1/anonymous/01J2VJQRGBK8YFWVV8K1MPRRWM/block.bin new file mode 100644 index 0000000000..200595ac24 Binary files /dev/null and b/pkg/experiment/querybackend/block/testdata/segments/1/anonymous/01J2VJQRGBK8YFWVV8K1MPRRWM/block.bin differ diff --git a/pkg/experiment/querybackend/block/testdata/segments/1/anonymous/01J2VJQRTMSCY4VDYBP5N4N5JK/block.bin b/pkg/experiment/querybackend/block/testdata/segments/1/anonymous/01J2VJQRTMSCY4VDYBP5N4N5JK/block.bin new file mode 100644 index 0000000000..429f6118eb Binary files /dev/null and b/pkg/experiment/querybackend/block/testdata/segments/1/anonymous/01J2VJQRTMSCY4VDYBP5N4N5JK/block.bin differ diff --git a/pkg/experiment/querybackend/block/testdata/segments/1/anonymous/01J2VJQTJ3PGF7KB39ARR1BX3Y/block.bin b/pkg/experiment/querybackend/block/testdata/segments/1/anonymous/01J2VJQTJ3PGF7KB39ARR1BX3Y/block.bin new file mode 100644 index 0000000000..99cabde890 Binary files /dev/null and b/pkg/experiment/querybackend/block/testdata/segments/1/anonymous/01J2VJQTJ3PGF7KB39ARR1BX3Y/block.bin differ diff --git a/pkg/experiment/querybackend/block/testdata/segments/1/anonymous/01J2VJQV544TF571FDSK2H692P/block.bin b/pkg/experiment/querybackend/block/testdata/segments/1/anonymous/01J2VJQV544TF571FDSK2H692P/block.bin new file mode 100644 index 0000000000..697a9fcd38 Binary files /dev/null and b/pkg/experiment/querybackend/block/testdata/segments/1/anonymous/01J2VJQV544TF571FDSK2H692P/block.bin differ diff --git a/pkg/experiment/querybackend/block/testdata/segments/1/anonymous/01J2VJQX8DYHSEBK7BAQSCJBMG/block.bin b/pkg/experiment/querybackend/block/testdata/segments/1/anonymous/01J2VJQX8DYHSEBK7BAQSCJBMG/block.bin new file mode 100644 index 0000000000..b5dec6d947 Binary files /dev/null and b/pkg/experiment/querybackend/block/testdata/segments/1/anonymous/01J2VJQX8DYHSEBK7BAQSCJBMG/block.bin differ diff --git a/pkg/experiment/querybackend/block/testdata/segments/1/anonymous/01J2VJQYQVZTPZMMJKE7F2XC47/block.bin b/pkg/experiment/querybackend/block/testdata/segments/1/anonymous/01J2VJQYQVZTPZMMJKE7F2XC47/block.bin new file mode 100644 index 0000000000..d6ee393216 Binary files /dev/null and b/pkg/experiment/querybackend/block/testdata/segments/1/anonymous/01J2VJQYQVZTPZMMJKE7F2XC47/block.bin differ diff --git a/pkg/experiment/querybackend/block/testdata/segments/1/anonymous/01J2VJQZPARDJQ779S1JMV0XQA/block.bin b/pkg/experiment/querybackend/block/testdata/segments/1/anonymous/01J2VJQZPARDJQ779S1JMV0XQA/block.bin new file mode 100644 index 0000000000..d10d1da8d3 Binary files /dev/null and b/pkg/experiment/querybackend/block/testdata/segments/1/anonymous/01J2VJQZPARDJQ779S1JMV0XQA/block.bin differ diff --git a/pkg/experiment/querybackend/block/testdata/segments/1/anonymous/01J2VJR0R3NQS23SDADNA6XHCM/block.bin b/pkg/experiment/querybackend/block/testdata/segments/1/anonymous/01J2VJR0R3NQS23SDADNA6XHCM/block.bin new file mode 100644 index 0000000000..2a53f1519c Binary files /dev/null and b/pkg/experiment/querybackend/block/testdata/segments/1/anonymous/01J2VJR0R3NQS23SDADNA6XHCM/block.bin differ diff --git a/pkg/experiment/querybackend/block/testdata/segments/1/anonymous/01J2VJR31PT3X4NDJC4Q2BHWQ1/block.bin b/pkg/experiment/querybackend/block/testdata/segments/1/anonymous/01J2VJR31PT3X4NDJC4Q2BHWQ1/block.bin new file mode 100644 index 0000000000..f4ba460cc4 Binary files /dev/null and b/pkg/experiment/querybackend/block/testdata/segments/1/anonymous/01J2VJR31PT3X4NDJC4Q2BHWQ1/block.bin differ diff --git a/pkg/experiment/querybackend/block/writer.go b/pkg/experiment/querybackend/block/writer.go new file mode 100644 index 0000000000..ecf29a0168 --- /dev/null +++ b/pkg/experiment/querybackend/block/writer.go @@ -0,0 +1,107 @@ +package block + +import ( + "context" + "io" + "os" + "path/filepath" + "strconv" + + "github.com/grafana/pyroscope/pkg/objstore" + "github.com/grafana/pyroscope/pkg/util/bufferpool" +) + +// TODO(kolesnikovae): +// - Avoid staging files where possible. +// - If stage files are required, at least avoid +// recreating them for each tenant dataset. +// - objstore.Bucket should provide object writer. + +type Writer struct { + storage objstore.Bucket + path string + local string + off uint64 + w *os.File + + tmp string + n int + cur string + + buf *bufferpool.Buffer +} + +func NewBlockWriter(storage objstore.Bucket, path string, tmp string) *Writer { + b := &Writer{ + storage: storage, + path: path, + tmp: tmp, + local: filepath.Join(tmp, FileNameDataObject), + buf: bufferpool.GetBuffer(compactionCopyBufferSize), + } + return b +} + +// Dir returns path to the new temp directory. +func (b *Writer) Dir() string { + b.n++ + b.cur = filepath.Join(b.tmp, strconv.Itoa(b.n)) + return b.cur +} + +// ReadFromFiles located in the directory Dir. +func (b *Writer) ReadFromFiles(files ...string) (toc []uint64, err error) { + toc = make([]uint64, len(files)) + for i := range files { + toc[i] = b.off + if err = b.ReadFromFile(files[i]); err != nil { + break + } + } + return toc, err +} + +// ReadFromFile located in the directory Dir. +func (b *Writer) ReadFromFile(file string) (err error) { + if b.w == nil { + if b.w, err = os.Create(b.local); err != nil { + return err + } + } + f, err := os.Open(filepath.Join(b.cur, file)) + if err != nil { + return err + } + defer func() { + _ = f.Close() + }() + b.buf.B = b.buf.B[:cap(b.buf.B)] + n, err := io.CopyBuffer(b.w, f, b.buf.B) + b.off += uint64(n) + return err +} + +func (b *Writer) Offset() uint64 { return b.off } + +func (b *Writer) Flush(ctx context.Context) error { + if err := b.w.Close(); err != nil { + return err + } + b.w = nil + f, err := os.Open(b.local) + if err != nil { + return err + } + defer func() { + _ = f.Close() + }() + return b.storage.Upload(ctx, b.path, f) +} + +func (b *Writer) Close() error { + bufferpool.Put(b.buf) + if b.w != nil { + return b.w.Close() + } + return nil +} diff --git a/pkg/experiment/querybackend/block_reader.go b/pkg/experiment/querybackend/block_reader.go new file mode 100644 index 0000000000..29ef4c3c1d --- /dev/null +++ b/pkg/experiment/querybackend/block_reader.go @@ -0,0 +1,125 @@ +package querybackend + +import ( + "context" + "fmt" + + "github.com/go-kit/log" + "github.com/opentracing/opentracing-go" + "github.com/prometheus/common/model" + "github.com/prometheus/prometheus/model/labels" + "github.com/prometheus/prometheus/promql/parser" + "golang.org/x/sync/errgroup" + "google.golang.org/grpc/codes" + "google.golang.org/grpc/status" + + querybackendv1 "github.com/grafana/pyroscope/api/gen/proto/go/querybackend/v1" + "github.com/grafana/pyroscope/pkg/experiment/querybackend/block" + "github.com/grafana/pyroscope/pkg/objstore" + "github.com/grafana/pyroscope/pkg/util" +) + +// Block reader reads objects from the object storage. Each block is currently +// represented by a single object. +// +// An object consists of data sets – regions within the block that include some +// tenant data. Each such dataset consists of 3 sections: profile table, TSDB, +// and symbol database. +// +// A single Invoke request typically spans multiple blocks (objects). +// Querying an object involves processing multiple datasets in parallel. +// Multiple parallel queries can be executed on the same tenant datasets. +// +// Thus, queries share the same "execution context": the object and a tenant +// dataset. +// +// object-a dataset-a query-a +// query-b +// dataset-b query-a +// query-b +// object-b dataset-a query-a +// query-b +// dataset-b query-a +// query-b +// + +type BlockReader struct { + log log.Logger + storage objstore.Bucket + + // TODO: + // - Use a worker pool instead of the errgroup. + // - Reusable query context. + // - Query pipelining: currently, queries share the same context, + // and reuse resources, but the data is processed independently. + // Instead, they should share the processing pipeline, if possible. +} + +func NewBlockReader(logger log.Logger, storage objstore.Bucket) *BlockReader { + return &BlockReader{ + log: logger, + storage: storage, + } +} + +func (b *BlockReader) Invoke( + ctx context.Context, + req *querybackendv1.InvokeRequest, +) (*querybackendv1.InvokeResponse, error) { + span, ctx := opentracing.StartSpanFromContext(ctx, "BlockReader.Invoke") + defer span.Finish() + vr, err := validateRequest(req) + if err != nil { + return nil, status.Errorf(codes.InvalidArgument, "request validation failed: %v", err) + } + g, ctx := errgroup.WithContext(ctx) + m := newAggregator(req) + for _, md := range req.QueryPlan.Blocks { + obj := block.NewObject(b.storage, md) + for _, meta := range md.Datasets { + c := newQueryContext(ctx, b.log, meta, vr, obj) + for _, query := range req.Query { + q := query + g.Go(util.RecoverPanic(func() error { + r, err := executeQuery(c, q) + if err != nil { + return err + } + return m.aggregateReport(r) + })) + } + } + } + if err = g.Wait(); err != nil { + return nil, err + } + return m.response() +} + +type request struct { + src *querybackendv1.InvokeRequest + matchers []*labels.Matcher + startTime int64 // Unix nano. + endTime int64 // Unix nano. +} + +func validateRequest(req *querybackendv1.InvokeRequest) (*request, error) { + if len(req.Query) == 0 { + return nil, fmt.Errorf("no queries provided") + } + if req.QueryPlan == nil || len(req.QueryPlan.Blocks) == 0 { + return nil, fmt.Errorf("no blocks planned") + } + matchers, err := parser.ParseMetricSelector(req.LabelSelector) + if err != nil { + return nil, fmt.Errorf("label selection is invalid: %w", err) + } + // TODO: Validate the rest, just in case. + r := request{ + src: req, + matchers: matchers, + startTime: model.Time(req.StartTime).UnixNano(), + endTime: model.Time(req.EndTime).UnixNano(), + } + return &r, nil +} diff --git a/pkg/experiment/querybackend/client/client.go b/pkg/experiment/querybackend/client/client.go new file mode 100644 index 0000000000..3ac72eee5a --- /dev/null +++ b/pkg/experiment/querybackend/client/client.go @@ -0,0 +1,73 @@ +package querybackendclient + +import ( + "context" + + "github.com/grafana/dskit/grpcclient" + "github.com/grafana/dskit/services" + "github.com/opentracing-contrib/go-grpc" + "github.com/opentracing/opentracing-go" + "google.golang.org/grpc" + + querybackendv1 "github.com/grafana/pyroscope/api/gen/proto/go/querybackend/v1" +) + +type Client struct { + service services.Service + grpcClient querybackendv1.QueryBackendServiceClient +} + +func New(address string, grpcClientConfig grpcclient.Config) (*Client, error) { + conn, err := dial(address, grpcClientConfig) + if err != nil { + return nil, err + } + var c Client + c.grpcClient = querybackendv1.NewQueryBackendServiceClient(conn) + c.service = services.NewIdleService(c.starting, c.stopping) + return &c, nil +} + +func dial(address string, grpcClientConfig grpcclient.Config) (*grpc.ClientConn, error) { + if err := grpcClientConfig.Validate(); err != nil { + return nil, err + } + grpcClientConfig.BackoffOnRatelimits = false + grpcClientConfig.ConnectTimeout = 0 + options, err := grpcClientConfig.DialOption(nil, nil) + if err != nil { + return nil, err + } + // TODO: https://github.com/grpc/grpc-proto/blob/master/grpc/service_config/service_config.proto + options = append(options, + grpc.WithUnaryInterceptor(otgrpc.OpenTracingClientInterceptor(opentracing.GlobalTracer())), + grpc.WithDefaultServiceConfig(grpcServiceConfig), + ) + return grpc.Dial(address, options...) +} + +func (b *Client) Service() services.Service { return b.service } +func (b *Client) starting(context.Context) error { return nil } +func (b *Client) stopping(error) error { return nil } + +func (b *Client) Invoke(ctx context.Context, req *querybackendv1.InvokeRequest) (*querybackendv1.InvokeResponse, error) { + return b.grpcClient.Invoke(ctx, req) +} + +const grpcServiceConfig = `{ + "loadBalancingPolicy":"round_robin", + "methodConfig": [{ + "name": [{"service": ""}], + "waitForReady": true, + "retryPolicy": { + "MaxAttempts": 500, + "InitialBackoff": ".01s", + "MaxBackoff": ".5s", + "BackoffMultiplier": 1.0, + "RetryableStatusCodes": [ + "UNAVAILABLE", + "RESOURCE_EXHAUSTED" + ] + } + }] +}` diff --git a/pkg/experiment/querybackend/query.go b/pkg/experiment/querybackend/query.go new file mode 100644 index 0000000000..3a56b8f064 --- /dev/null +++ b/pkg/experiment/querybackend/query.go @@ -0,0 +1,141 @@ +package querybackend + +import ( + "context" + "fmt" + "sync" + + "github.com/go-kit/log" + "github.com/iancoleman/strcase" + "github.com/opentracing/opentracing-go" + + metastorev1 "github.com/grafana/pyroscope/api/gen/proto/go/metastore/v1" + querybackendv1 "github.com/grafana/pyroscope/api/gen/proto/go/querybackend/v1" + "github.com/grafana/pyroscope/pkg/experiment/querybackend/block" +) + +// TODO(kolesnikovae): We have a procedural definition of our queries, +// thus we have handlers. Instead, in order to enable pipelining and +// reduce the boilerplate, we should define query execution plans. + +var ( + handlerMutex = new(sync.RWMutex) + queryHandlers = map[querybackendv1.QueryType]queryHandler{} +) + +type queryHandler func(*queryContext, *querybackendv1.Query) (*querybackendv1.Report, error) + +func registerQueryHandler(t querybackendv1.QueryType, h queryHandler) { + handlerMutex.Lock() + defer handlerMutex.Unlock() + if _, ok := queryHandlers[t]; ok { + panic(fmt.Sprintf("%s: handler already registered", t)) + } + queryHandlers[t] = h +} + +func getQueryHandler(t querybackendv1.QueryType) (queryHandler, error) { + handlerMutex.RLock() + defer handlerMutex.RUnlock() + handler, ok := queryHandlers[t] + if !ok { + return nil, fmt.Errorf("unknown query type %s", t) + } + return handler, nil +} + +var ( + depMutex = new(sync.RWMutex) + queryDependencies = map[querybackendv1.QueryType][]block.Section{} +) + +func registerQueryDependencies(t querybackendv1.QueryType, deps ...block.Section) { + depMutex.Lock() + defer depMutex.Unlock() + if _, ok := queryDependencies[t]; ok { + panic(fmt.Sprintf("%s: dependencies already registered", t)) + } + queryDependencies[t] = deps +} + +func registerQueryType( + qt querybackendv1.QueryType, + rt querybackendv1.ReportType, + q queryHandler, + a aggregatorProvider, + deps ...block.Section, +) { + registerQueryReportType(qt, rt) + registerQueryHandler(qt, q) + registerQueryDependencies(qt, deps...) + registerAggregator(rt, a) +} + +type queryContext struct { + ctx context.Context + log log.Logger + meta *metastorev1.Dataset + req *request + obj *block.Object + ds *block.Dataset + err error +} + +func newQueryContext( + ctx context.Context, + logger log.Logger, + meta *metastorev1.Dataset, + req *request, + obj *block.Object, +) *queryContext { + return &queryContext{ + ctx: ctx, + log: logger, + req: req, + meta: meta, + obj: obj, + ds: block.NewDataset(meta, obj), + } +} + +func executeQuery(q *queryContext, query *querybackendv1.Query) (r *querybackendv1.Report, err error) { + var span opentracing.Span + span, q.ctx = opentracing.StartSpanFromContext(q.ctx, "executeQuery."+strcase.ToCamel(query.QueryType.String())) + defer span.Finish() + handle, err := getQueryHandler(query.QueryType) + if err != nil { + return nil, err + } + if err = q.open(); err != nil { + return nil, fmt.Errorf("failed to initialize query context: %w", err) + } + defer func() { + _ = q.close(err) + }() + if r, err = handle(q, query); r != nil { + r.ReportType = QueryReportType(query.QueryType) + } + return r, err +} + +func (q *queryContext) open() error { + return q.ds.Open(q.ctx, q.sections()...) +} + +func (q *queryContext) close(err error) error { + return q.ds.CloseWithError(err) +} + +func (q *queryContext) sections() []block.Section { + sections := make(map[block.Section]struct{}, 3) + for _, qt := range q.req.src.Query { + for _, s := range queryDependencies[qt.QueryType] { + sections[s] = struct{}{} + } + } + unique := make([]block.Section, 0, len(sections)) + for s := range sections { + unique = append(unique, s) + } + return unique +} diff --git a/pkg/experiment/querybackend/query_label_names.go b/pkg/experiment/querybackend/query_label_names.go new file mode 100644 index 0000000000..995c21ef13 --- /dev/null +++ b/pkg/experiment/querybackend/query_label_names.go @@ -0,0 +1,100 @@ +package querybackend + +import ( + "sort" + "sync" + + "github.com/prometheus/prometheus/model/labels" + + querybackendv1 "github.com/grafana/pyroscope/api/gen/proto/go/querybackend/v1" + "github.com/grafana/pyroscope/pkg/experiment/querybackend/block" + "github.com/grafana/pyroscope/pkg/model" + "github.com/grafana/pyroscope/pkg/phlaredb" +) + +func init() { + registerQueryType( + querybackendv1.QueryType_QUERY_LABEL_NAMES, + querybackendv1.ReportType_REPORT_LABEL_NAMES, + queryLabelNames, + newLabelNameAggregator, + []block.Section{block.SectionTSDB}..., + ) +} + +func queryLabelNames(q *queryContext, query *querybackendv1.Query) (*querybackendv1.Report, error) { + var names []string + var err error + if len(q.req.matchers) == 0 { + names, err = q.ds.Index().LabelNames() + } else { + names, err = labelNamesForMatchers(q.ds.Index(), q.req.matchers) + } + if err != nil { + return nil, err + } + resp := &querybackendv1.Report{ + LabelNames: &querybackendv1.LabelNamesReport{ + Query: query.LabelNames.CloneVT(), + LabelNames: names, + }, + } + return resp, nil +} + +func labelNamesForMatchers(reader phlaredb.IndexReader, matchers []*labels.Matcher) ([]string, error) { + postings, err := phlaredb.PostingsForMatchers(reader, nil, matchers...) + if err != nil { + return nil, err + } + l := make(map[string]struct{}) + for postings.Next() { + var n []string + if n, err = reader.LabelNamesFor(postings.At()); err != nil { + return nil, err + } + for _, name := range n { + l[name] = struct{}{} + } + } + if err = postings.Err(); err != nil { + return nil, err + } + names := make([]string, len(l)) + var i int + for name := range l { + names[i] = name + i++ + } + sort.Strings(names) + return names, nil +} + +type labelNameAggregator struct { + init sync.Once + query *querybackendv1.LabelNamesQuery + names *model.LabelMerger +} + +func newLabelNameAggregator(*querybackendv1.InvokeRequest) aggregator { + return new(labelNameAggregator) +} + +func (m *labelNameAggregator) aggregate(report *querybackendv1.Report) error { + r := report.LabelNames + m.init.Do(func() { + m.query = r.Query.CloneVT() + m.names = model.NewLabelMerger() + }) + m.names.MergeLabelNames(r.LabelNames) + return nil +} + +func (m *labelNameAggregator) build() *querybackendv1.Report { + return &querybackendv1.Report{ + LabelNames: &querybackendv1.LabelNamesReport{ + Query: m.query, + LabelNames: m.names.LabelNames(), + }, + } +} diff --git a/pkg/experiment/querybackend/query_label_values.go b/pkg/experiment/querybackend/query_label_values.go new file mode 100644 index 0000000000..5f273092d7 --- /dev/null +++ b/pkg/experiment/querybackend/query_label_values.go @@ -0,0 +1,103 @@ +package querybackend + +import ( + "errors" + "sort" + "sync" + + "github.com/prometheus/prometheus/model/labels" + "github.com/prometheus/prometheus/storage" + + querybackendv1 "github.com/grafana/pyroscope/api/gen/proto/go/querybackend/v1" + "github.com/grafana/pyroscope/pkg/experiment/querybackend/block" + "github.com/grafana/pyroscope/pkg/model" + "github.com/grafana/pyroscope/pkg/phlaredb" +) + +func init() { + registerQueryType( + querybackendv1.QueryType_QUERY_LABEL_VALUES, + querybackendv1.ReportType_REPORT_LABEL_VALUES, + queryLabelValues, + newLabelValueAggregator, + []block.Section{block.SectionTSDB}..., + ) +} + +func queryLabelValues(q *queryContext, query *querybackendv1.Query) (*querybackendv1.Report, error) { + var values []string + var err error + if len(q.req.matchers) == 0 { + values, err = q.ds.Index().LabelValues(query.LabelValues.LabelName) + } else { + values, err = labelValuesForMatchers(q.ds.Index(), query.LabelValues.LabelName, q.req.matchers) + } + if err != nil { + return nil, err + } + resp := &querybackendv1.Report{ + LabelValues: &querybackendv1.LabelValuesReport{ + Query: query.LabelValues.CloneVT(), + LabelValues: values, + }, + } + return resp, nil +} + +func labelValuesForMatchers(reader phlaredb.IndexReader, name string, matchers []*labels.Matcher) ([]string, error) { + postings, err := phlaredb.PostingsForMatchers(reader, nil, matchers...) + if err != nil { + return nil, err + } + l := make(map[string]struct{}) + for postings.Next() { + var v string + if v, err = reader.LabelValueFor(postings.At(), name); err != nil { + if errors.Is(err, storage.ErrNotFound) { + continue + } + return nil, err + } + l[v] = struct{}{} + } + if err = postings.Err(); err != nil { + return nil, err + } + values := make([]string, len(l)) + var i int + for v := range l { + values[i] = v + i++ + } + sort.Strings(values) + return values, nil +} + +type labelValueAggregator struct { + init sync.Once + query *querybackendv1.LabelValuesQuery + values *model.LabelMerger +} + +func newLabelValueAggregator(*querybackendv1.InvokeRequest) aggregator { + return new(labelValueAggregator) +} + +func (m *labelValueAggregator) aggregate(report *querybackendv1.Report) error { + r := report.LabelValues + m.init.Do(func() { + m.query = r.Query.CloneVT() + m.values = model.NewLabelMerger() + }) + m.values.MergeLabelValues(r.LabelValues) + return nil +} + +func (m *labelValueAggregator) build() *querybackendv1.Report { + return &querybackendv1.Report{ + LabelValues: &querybackendv1.LabelValuesReport{ + Query: m.query, + LabelValues: m.values.LabelValues(), + }, + } +} diff --git a/pkg/experiment/querybackend/query_profile_entry.go b/pkg/experiment/querybackend/query_profile_entry.go new file mode 100644 index 0000000000..f6b9ef578b --- /dev/null +++ b/pkg/experiment/querybackend/query_profile_entry.go @@ -0,0 +1,90 @@ +package querybackend + +import ( + "github.com/parquet-go/parquet-go" + "github.com/prometheus/common/model" + "github.com/prometheus/prometheus/model/labels" + + "github.com/grafana/pyroscope/pkg/iter" + phlaremodel "github.com/grafana/pyroscope/pkg/model" + "github.com/grafana/pyroscope/pkg/phlaredb" + parquetquery "github.com/grafana/pyroscope/pkg/phlaredb/query" + schemav1 "github.com/grafana/pyroscope/pkg/phlaredb/schemas/v1" + "github.com/grafana/pyroscope/pkg/phlaredb/tsdb/index" +) + +func profileEntryIterator(q *queryContext, groupBy ...string) (iter.Iterator[ProfileEntry], error) { + series, err := getSeriesLabels(q.ds.Index(), q.req.matchers, groupBy...) + if err != nil { + return nil, err + } + results := parquetquery.NewBinaryJoinIterator(0, + q.ds.Profiles().Column(q.ctx, "SeriesIndex", parquetquery.NewMapPredicate(series)), + q.ds.Profiles().Column(q.ctx, "TimeNanos", parquetquery.NewIntBetweenPredicate(q.req.startTime, q.req.endTime)), + ) + results = parquetquery.NewBinaryJoinIterator(0, results, + q.ds.Profiles().Column(q.ctx, "StacktracePartition", nil), + ) + + buf := make([][]parquet.Value, 3) + entries := iter.NewAsyncBatchIterator[*parquetquery.IteratorResult, ProfileEntry]( + results, 128, + func(r *parquetquery.IteratorResult) ProfileEntry { + buf = r.Columns(buf, + schemav1.SeriesIndexColumnName, + schemav1.TimeNanosColumnName, + schemav1.StacktracePartitionColumnName) + x := series[buf[0][0].Uint32()] + return ProfileEntry{ + RowNum: r.RowNumber[0], + Timestamp: model.TimeFromUnixNano(buf[1][0].Int64()), + Fingerprint: x.fingerprint, + Labels: x.labels, + Partition: buf[2][0].Uint64(), + } + }, + func([]ProfileEntry) {}, + ) + return entries, nil +} + +type ProfileEntry struct { + RowNum int64 + Timestamp model.Time + Fingerprint model.Fingerprint + Labels phlaremodel.Labels + Partition uint64 +} + +func (e ProfileEntry) RowNumber() int64 { return e.RowNum } + +type seriesLabels struct { + fingerprint model.Fingerprint + labels phlaremodel.Labels +} + +func getSeriesLabels(reader phlaredb.IndexReader, matchers []*labels.Matcher, by ...string) (map[uint32]seriesLabels, error) { + postings, err := getPostings(reader, matchers...) + if err != nil { + return nil, err + } + chunks := make([]index.ChunkMeta, 1) + series := make(map[uint32]seriesLabels) + l := make(phlaremodel.Labels, 0, 6) + for postings.Next() { + fp, err := reader.SeriesBy(postings.At(), &l, &chunks, by...) + if err != nil { + return nil, err + } + _, ok := series[chunks[0].SeriesIndex] + if ok { + continue + } + series[chunks[0].SeriesIndex] = seriesLabels{ + fingerprint: model.Fingerprint(fp), + labels: l.Clone(), + } + } + + return series, postings.Err() +} diff --git a/pkg/experiment/querybackend/query_series_labels.go b/pkg/experiment/querybackend/query_series_labels.go new file mode 100644 index 0000000000..1fa1e2a3a4 --- /dev/null +++ b/pkg/experiment/querybackend/query_series_labels.go @@ -0,0 +1,94 @@ +package querybackend + +import ( + "sync" + + "github.com/prometheus/prometheus/model/labels" + + querybackendv1 "github.com/grafana/pyroscope/api/gen/proto/go/querybackend/v1" + typesv1 "github.com/grafana/pyroscope/api/gen/proto/go/types/v1" + "github.com/grafana/pyroscope/pkg/experiment/querybackend/block" + "github.com/grafana/pyroscope/pkg/model" + "github.com/grafana/pyroscope/pkg/phlaredb" + "github.com/grafana/pyroscope/pkg/phlaredb/tsdb/index" +) + +func init() { + registerQueryType( + querybackendv1.QueryType_QUERY_SERIES_LABELS, + querybackendv1.ReportType_REPORT_SERIES_LABELS, + querySeriesLabels, + newSeriesLabelsAggregator, + []block.Section{block.SectionTSDB}..., + ) +} + +func querySeriesLabels(q *queryContext, query *querybackendv1.Query) (*querybackendv1.Report, error) { + postings, err := getPostings(q.ds.Index(), q.req.matchers...) + if err != nil { + return nil, err + } + var tmp model.Labels + var c []index.ChunkMeta + l := make(map[uint64]model.Labels) + for postings.Next() { + fp, _ := q.ds.Index().SeriesBy(postings.At(), &tmp, &c, query.SeriesLabels.LabelNames...) + if _, ok := l[fp]; ok { + continue + } + l[fp] = tmp.Clone() + } + if err = postings.Err(); err != nil { + return nil, err + } + series := make([]*typesv1.Labels, len(l)) + var i int + for _, s := range l { + series[i] = &typesv1.Labels{Labels: s} + i++ + } + resp := &querybackendv1.Report{ + SeriesLabels: &querybackendv1.SeriesLabelsReport{ + Query: query.SeriesLabels.CloneVT(), + SeriesLabels: series, + }, + } + return resp, nil +} + +func getPostings(reader phlaredb.IndexReader, matchers ...*labels.Matcher) (index.Postings, error) { + if len(matchers) == 0 { + k, v := index.AllPostingsKey() + return reader.Postings(k, nil, v) + } + return phlaredb.PostingsForMatchers(reader, nil, matchers...) +} + +type seriesLabelsAggregator struct { + init sync.Once + query *querybackendv1.SeriesLabelsQuery + series *model.LabelMerger +} + +func newSeriesLabelsAggregator(*querybackendv1.InvokeRequest) aggregator { + return new(seriesLabelsAggregator) +} + +func (a *seriesLabelsAggregator) aggregate(report *querybackendv1.Report) error { + r := report.SeriesLabels + a.init.Do(func() { + a.query = r.Query.CloneVT() + a.series = model.NewLabelMerger() + }) + a.series.MergeSeries(r.SeriesLabels) + return nil +} + +func (a *seriesLabelsAggregator) build() *querybackendv1.Report { + return &querybackendv1.Report{ + SeriesLabels: &querybackendv1.SeriesLabelsReport{ + Query: a.query, + SeriesLabels: a.series.SeriesLabels(), + }, + } +} diff --git a/pkg/experiment/querybackend/query_time_series.go b/pkg/experiment/querybackend/query_time_series.go new file mode 100644 index 0000000000..7a09ee6855 --- /dev/null +++ b/pkg/experiment/querybackend/query_time_series.go @@ -0,0 +1,114 @@ +package querybackend + +import ( + "strings" + "sync" + "time" + + "github.com/grafana/dskit/runutil" + + querybackendv1 "github.com/grafana/pyroscope/api/gen/proto/go/querybackend/v1" + typesv1 "github.com/grafana/pyroscope/api/gen/proto/go/types/v1" + "github.com/grafana/pyroscope/pkg/experiment/querybackend/block" + phlaremodel "github.com/grafana/pyroscope/pkg/model" + parquetquery "github.com/grafana/pyroscope/pkg/phlaredb/query" + schemav1 "github.com/grafana/pyroscope/pkg/phlaredb/schemas/v1" +) + +func init() { + registerQueryType( + querybackendv1.QueryType_QUERY_TIME_SERIES, + querybackendv1.ReportType_REPORT_TIME_SERIES, + queryTimeSeries, + newTimeSeriesAggregator, + []block.Section{ + block.SectionTSDB, + block.SectionProfiles, + }..., + ) +} + +func queryTimeSeries(q *queryContext, query *querybackendv1.Query) (r *querybackendv1.Report, err error) { + entries, err := profileEntryIterator(q, query.TimeSeries.GroupBy...) + if err != nil { + return nil, err + } + defer runutil.CloseWithErrCapture(&err, entries, "failed to close profile entry iterator") + + column, err := schemav1.ResolveColumnByPath(q.ds.Profiles().Schema(), strings.Split("TotalValue", ".")) + if err != nil { + return nil, err + } + + rows := parquetquery.NewRepeatedRowIterator(q.ctx, entries, q.ds.Profiles().RowGroups(), column.ColumnIndex) + defer runutil.CloseWithErrCapture(&err, rows, "failed to close column iterator") + + builder := phlaremodel.NewTimeSeriesBuilder(query.TimeSeries.GroupBy...) + for rows.Next() { + row := rows.At() + builder.Add( + row.Row.Fingerprint, + row.Row.Labels, + int64(row.Row.Timestamp), + float64(row.Values[0][0].Int64()), + ) + } + if err = rows.Err(); err != nil { + return nil, err + } + + resp := &querybackendv1.Report{ + TimeSeries: &querybackendv1.TimeSeriesReport{ + Query: query.TimeSeries.CloneVT(), + TimeSeries: builder.Build(), + }, + } + + return resp, nil +} + +type timeSeriesAggregator struct { + init sync.Once + startTime int64 + endTime int64 + query *querybackendv1.TimeSeriesQuery + series *phlaremodel.TimeSeriesMerger +} + +func newTimeSeriesAggregator(req *querybackendv1.InvokeRequest) aggregator { + return &timeSeriesAggregator{ + startTime: req.StartTime, + endTime: req.EndTime, + } +} + +func (a *timeSeriesAggregator) aggregate(report *querybackendv1.Report) error { + r := report.TimeSeries + a.init.Do(func() { + a.series = phlaremodel.NewTimeSeriesMerger(true) + a.query = r.Query.CloneVT() + }) + a.series.MergeTimeSeries(r.TimeSeries) + return nil +} + +func (a *timeSeriesAggregator) build() *querybackendv1.Report { + // TODO(kolesnikovae): Average aggregation should be implemented in + // the way that it can be distributed (count + sum), and should be done + // at "aggregate" call. + sum := typesv1.TimeSeriesAggregationType_TIME_SERIES_AGGREGATION_TYPE_SUM + stepMilli := time.Duration(a.query.GetStep() * float64(time.Second)).Milliseconds() + seriesIterator := phlaremodel.NewTimeSeriesMergeIterator(a.series.TimeSeries()) + return &querybackendv1.Report{ + TimeSeries: &querybackendv1.TimeSeriesReport{ + Query: a.query, + TimeSeries: phlaremodel.RangeSeries( + seriesIterator, + a.startTime, + a.endTime, + stepMilli, + &sum, + ), + }, + } +} diff --git a/pkg/experiment/querybackend/query_tree.go b/pkg/experiment/querybackend/query_tree.go new file mode 100644 index 0000000000..b440244342 --- /dev/null +++ b/pkg/experiment/querybackend/query_tree.go @@ -0,0 +1,95 @@ +package querybackend + +import ( + "sync" + + "github.com/grafana/dskit/runutil" + + querybackendv1 "github.com/grafana/pyroscope/api/gen/proto/go/querybackend/v1" + "github.com/grafana/pyroscope/pkg/experiment/querybackend/block" + "github.com/grafana/pyroscope/pkg/model" + parquetquery "github.com/grafana/pyroscope/pkg/phlaredb/query" + v1 "github.com/grafana/pyroscope/pkg/phlaredb/schemas/v1" + "github.com/grafana/pyroscope/pkg/phlaredb/symdb" +) + +func init() { + registerQueryType( + querybackendv1.QueryType_QUERY_TREE, + querybackendv1.ReportType_REPORT_TREE, + queryTree, + newTreeAggregator, + []block.Section{ + block.SectionTSDB, + block.SectionProfiles, + block.SectionSymbols, + }..., + ) +} + +func queryTree(q *queryContext, query *querybackendv1.Query) (*querybackendv1.Report, error) { + entries, err := profileEntryIterator(q) + if err != nil { + return nil, err + } + defer runutil.CloseWithErrCapture(&err, entries, "failed to close profile entry iterator") + + var columns v1.SampleColumns + if err = columns.Resolve(q.ds.Profiles().Schema()); err != nil { + return nil, err + } + + profiles := parquetquery.NewRepeatedRowIterator(q.ctx, entries, q.ds.Profiles().RowGroups(), + columns.StacktraceID.ColumnIndex, + columns.Value.ColumnIndex) + defer runutil.CloseWithErrCapture(&err, profiles, "failed to close profile stream") + + resolver := symdb.NewResolver(q.ctx, q.ds.Symbols()) + defer resolver.Release() + for profiles.Next() { + p := profiles.At() + resolver.AddSamplesFromParquetRow(p.Row.Partition, p.Values[0], p.Values[1]) + } + if err = profiles.Err(); err != nil { + return nil, err + } + + tree, err := resolver.Tree() + if err != nil { + return nil, err + } + + resp := &querybackendv1.Report{ + Tree: &querybackendv1.TreeReport{ + Query: query.Tree.CloneVT(), + Tree: tree.Bytes(query.Tree.GetMaxNodes()), + }, + } + return resp, nil +} + +type treeAggregator struct { + init sync.Once + query *querybackendv1.TreeQuery + tree *model.TreeMerger +} + +func newTreeAggregator(*querybackendv1.InvokeRequest) aggregator { return new(treeAggregator) } + +func (a *treeAggregator) aggregate(report *querybackendv1.Report) error { + r := report.Tree + a.init.Do(func() { + a.tree = model.NewTreeMerger() + a.query = r.Query.CloneVT() + }) + return a.tree.MergeTreeBytes(r.Tree) +} + +func (a *treeAggregator) build() *querybackendv1.Report { + return &querybackendv1.Report{ + Tree: &querybackendv1.TreeReport{ + Query: a.query, + Tree: a.tree.Tree().Bytes(a.query.GetMaxNodes()), + }, + } +} diff --git a/pkg/experiment/querybackend/queryplan/query_plan.go b/pkg/experiment/querybackend/queryplan/query_plan.go new file mode 100644 index 0000000000..bc33f3016e --- /dev/null +++ b/pkg/experiment/querybackend/queryplan/query_plan.go @@ -0,0 +1,349 @@ +package queryplan + +import ( + "fmt" + "io" + "math" + "slices" + "strings" + "unsafe" + + metastorev1 "github.com/grafana/pyroscope/api/gen/proto/go/metastore/v1" + querybackendv1 "github.com/grafana/pyroscope/api/gen/proto/go/querybackend/v1" + "github.com/grafana/pyroscope/pkg/iter" +) + +// QueryPlan represents a physical query plan structured as a DAG. +// Each node in the graph can either be a "merge" or a "read" operation (leaves). +// Merge nodes reference other nodes, while read nodes reference data blocks. +type QueryPlan struct { + nodes []node + blocks []*metastorev1.BlockMeta +} + +type Node struct { + Type NodeType + + p *QueryPlan + n node +} + +type NodeType uint32 + +const ( + _ NodeType = iota + NodeRead + NodeMerge +) + +var typeNames = [...]string{"invalid", "read", "merge"} + +func (t NodeType) String() string { + if int(t) >= len(typeNames) { + return typeNames[0] + } + return typeNames[t] +} + +type node struct { + typ NodeType + // Node of merge type refers to nodes. + // Node of read type refers to blocks. + off uint32 + len uint32 +} + +func Open(p *querybackendv1.QueryPlan) *QueryPlan { + if len(p.Blocks) == 0 { + return new(QueryPlan) + } + qp := QueryPlan{blocks: p.Blocks} + if len(p.Graph) != 0 || len(p.Graph)%3 == 0 { + qp.nodes = unsafe.Slice((*node)(unsafe.Pointer(unsafe.SliceData(p.Graph))), len(p.Graph)/3) + } + return &qp +} + +// Build creates a query plan from the list of block metadata. +// +// NOTE(kolesnikovae): At this point it only groups blocks into uniform ranges, +// and builds a DAG of reads and merges. In practice, however, we may want to +// implement more sophisticated strategies. For example, it would be beneficial +// to group blocks based on the tenant services to ensure that a single read +// covers exactly one service, and does not have to deal with stack trace +// cardinality issues. Another example is grouping by shards to minimize the +// number of unique series (assuming the shards are still built based on the +// series labels) a reader or merger should handle. In general, the strategy +// should depend on the query type. +func Build( + blocks []*metastorev1.BlockMeta, + maxReads, maxMerges int64, +) *QueryPlan { + if len(blocks) == 0 { + return new(QueryPlan) + } + // First, we create leaves: the entire range of blocks + // is split into smaller uniform ranges, which will be + // fetched by workers. + s := int(math.Max(float64(maxReads), float64(maxMerges))) + ranges := uniformSplit(make([][2]uint32, s), len(blocks), maxReads) + nodes := make([]node, len(ranges)) + for i, b := range ranges { + nodes[i] = node{ + typ: NodeRead, + // Block range. + off: b[0], + len: b[1], + } + } + if len(nodes) < 2 { + return &QueryPlan{ + blocks: blocks, + nodes: nodes, + } + } + // Next we create merge nodes. + var off int + for { + // Split should not be applied to the same (sub-)range + // twice, therefore we keep track of the offset within + // the nodes slice. + length := len(nodes) + ranges = uniformSplit(ranges, len(nodes)-off, maxMerges) + for _, n := range ranges { + // Range offset does not account for the offset within + // the nodes slice, therefore we add it here. + nodes = append(nodes, node{ + typ: NodeMerge, + off: n[0] + uint32(off), + len: n[1], + }) + } + if len(ranges) == 1 { + // The root node has been added. + break + } else if len(ranges) == 0 { + // Create a virtual root, that will be a parent of all the + // top level nodes. We find the offset of child nodes based + // on the last node: its children is the last range of nodes + // that have a parent. + n := nodes[len(nodes)-1] + o := n.off + n.len + l := uint32(len(nodes)) - o + nodes = append(nodes, node{ + typ: NodeMerge, + off: o, + len: l, + }) + break + } + off += length + } + return &QueryPlan{ + blocks: blocks, + nodes: nodes, + } +} + +// Root returns the root node of the query plan. +func (p *QueryPlan) Root() *Node { + if len(p.nodes) == 0 { + return &Node{Type: NodeRead, p: p} + } + n := Node{p: p} + n.n = p.nodes[len(p.nodes)-1] + n.Type = n.n.typ + return &n +} + +// Plan returns the query plan scoped to the node. +// The plan references the parent plan blocks. +func (n *Node) Plan() *QueryPlan { + // BFS traversal. Our goal is to preserve the order of + // nodes as in the original plan, and only shift offsets. + nodes := make([]node, 0, 32) + stack := make([]node, 0, 32) + stack = append(stack, n.n) + var x node + for len(stack) > 0 { + x, stack = stack[0], stack[1:] + if x.typ == NodeMerge { + // Add child nodes in the reverse order, to honour + // the order of the original plan, compensating the + // stack LIFO at this level. We do it after append + // to stack to not modify the original plan. + s := len(stack) + stack = append(stack, n.p.nodes[x.off:x.off+x.len]...) + slices.Reverse(stack[s:]) + } + nodes = append(nodes, x) + } + if len(nodes) == 0 { + return new(QueryPlan) + } + // Swap merge and read nodes as their order is changed + // during the traversal. The order of nodes within the + // same type is revered as well (order within the level + // is fixed at traversal). + var p NodeType // Previous node type. + var s int + for i, c := range nodes { + if p != 0 && c.typ != p { + s = i + break + } + p = c.typ + } + slices.Reverse(nodes[:s]) // Merge nodes. + slices.Reverse(nodes[s:]) // Read nodes. + tmp := make([]node, s) + copy(tmp, nodes[:s]) + copy(nodes, nodes[s:]) + copy(nodes[len(nodes)-s:], tmp) + if nodes[0].typ != NodeRead { + panic("bug: first node must be a read node") + } + + // Offset correction by node type: as the plan is a subtree, + // the offsets of the child nodes are shifted. + o := [3]int{0, -1, -1} // Table of offsets by type. + bs := nodes[0].off // Offset of the first referenced block. + for i, c := range nodes { + // Correct the offset. + if o[c.typ] < 0 { + // This is the first node of the type: we remember + // the offset and reset it to zero, as children of + // the first node are _always_ placed at the very + // beginning (for both read and merge nodes). + nodes[i].off = 0 + o[c.typ] = int(c.len) + } else { + nodes[i].off = uint32(o[c.typ]) + o[c.typ] += int(c.len) + } + } + + // Update references to the blocks. + be := bs + uint32(o[NodeRead]) + blocks := n.p.blocks[bs:be] + + return &QueryPlan{ + nodes: nodes, + blocks: blocks, + } +} + +func (p *QueryPlan) Proto() *querybackendv1.QueryPlan { + return &querybackendv1.QueryPlan{ + Graph: unsafe.Slice((*uint32)(unsafe.Pointer(unsafe.SliceData(p.nodes))), len(p.nodes)*3), + Blocks: p.blocks, + } +} + +func (p *QueryPlan) String() string { + var b strings.Builder + printPlan(&b, "", p, false) + return b.String() +} + +func (n *Node) Children() iter.Iterator[*Node] { + if n.n.typ != NodeMerge { + return iter.NewEmptyIterator[*Node]() + } + return &nodeIterator{n: n} +} + +func (n *Node) Blocks() iter.Iterator[*metastorev1.BlockMeta] { + if n.n.typ != NodeRead { + return iter.NewEmptyIterator[*metastorev1.BlockMeta]() + } + return &blockIterator{n: n} +} + +type nodeIterator struct { + n *Node + i int +} + +func (i *nodeIterator) Err() error { return nil } +func (i *nodeIterator) Close() error { return nil } + +func (i *nodeIterator) Next() bool { + if i.i >= int(i.n.n.len) { + return false + } + i.i++ + return true +} + +func (i *nodeIterator) At() *Node { + n := i.n.p.nodes[int(i.n.n.off)+i.i-1] + return &Node{ + Type: i.n.Type, + p: i.n.p, + n: n, + } +} + +type blockIterator struct { + n *Node + i int +} + +func (i *blockIterator) Err() error { return nil } +func (i *blockIterator) Close() error { return nil } + +func (i *blockIterator) Next() bool { + if i.i >= int(i.n.n.len) { + return false + } + i.i++ + return true +} + +func (i *blockIterator) At() *metastorev1.BlockMeta { + return i.n.p.blocks[int(i.n.n.off)+i.i-1] +} + +// uniformSplit splits a slice of length s into +// uniform ranges not exceeding the size max. +func uniformSplit(ret [][2]uint32, s int, max int64) [][2]uint32 { + ret = ret[:0] + n := math.Ceil(float64(s) / float64(max)) // Find number of parts. + o := int(math.Ceil(float64(s) / n)) // Find optimal part size. + for i := 0; i < s; i += o { + r := i + o + if r > s { + r = s + } + ret = append(ret, [2]uint32{uint32(i), uint32(r - i)}) + } + return ret +} + +func printPlan(w io.Writer, pad string, p *QueryPlan, debug bool) { + r := p.Root() + if debug { + _, _ = fmt.Fprintf(w, pad+"%s {children: %d, nodes: %d, blocks: %d}\n", + r.Type, r.n.len, len(r.p.nodes), len(r.p.blocks)) + } else { + _, _ = fmt.Fprintf(w, pad+"%s (%d)\n", r.Type, r.n.len) + } + + switch r.Type { + case NodeMerge: + c := r.Children() + for c.Next() { + printPlan(w, pad+"\t", c.At().Plan(), debug) + } + + case NodeRead: + b := r.Blocks() + for b.Next() { + _, _ = fmt.Fprintf(w, pad+"\t"+"%+v\n", b.At()) + } + + default: + panic("unknown type") + } +} diff --git a/pkg/experiment/querybackend/queryplan/query_plan_test.go b/pkg/experiment/querybackend/queryplan/query_plan_test.go new file mode 100644 index 0000000000..488e2f7ce6 --- /dev/null +++ b/pkg/experiment/querybackend/queryplan/query_plan_test.go @@ -0,0 +1,128 @@ +package queryplan + +import ( + "bytes" + "fmt" + "os" + "testing" + + "github.com/stretchr/testify/assert" + "github.com/stretchr/testify/require" + + metastorev1 "github.com/grafana/pyroscope/api/gen/proto/go/metastore/v1" + querybackendv1 "github.com/grafana/pyroscope/api/gen/proto/go/querybackend/v1" + "github.com/grafana/pyroscope/pkg/iter" +) + +func Test_Plan(t *testing.T) { + blocks := []*metastorev1.BlockMeta{ + {Id: "1"}, {Id: "2"}, + {Id: "3"}, {Id: "4"}, + {Id: "5"}, {Id: "6"}, + {Id: "7"}, {Id: "8"}, + {Id: "9"}, {Id: "10"}, + {Id: "11"}, {Id: "12"}, + {Id: "13"}, {Id: "14"}, + {Id: "15"}, {Id: "16"}, + {Id: "17"}, {Id: "18"}, + {Id: "19"}, {Id: "20"}, + {Id: "21"}, {Id: "22"}, + {Id: "23"}, {Id: "24"}, + {Id: "25"}, + } + + p := Build(blocks, 2, 3) + var buf bytes.Buffer + printPlan(&buf, "", p, true) + // Ensure that the plan has not been modified + // during traversal performed by printPlan. + assert.Equal(t, Build(blocks, 2, 3), p) + + expected, err := os.ReadFile("testdata/plan.txt") + require.NoError(t, err) + assert.Equal(t, string(expected), buf.String()) + + // Root node (sub-)plan must be identical to the original plan. + buf.Reset() + printPlan(&buf, "", p.Root().Plan(), true) + assert.Equal(t, string(expected), buf.String()) +} + +func Test_Plan_propagation(t *testing.T) { + blocks := []*metastorev1.BlockMeta{ + {Id: "01J2JY1K5J4T2WNDV05CHVFCA9"}, + {Id: "01J2JY21VVYYV4PMDGK4TVMZ6H"}, + {Id: "01J2JY2GF83EF0QMW94T19MXHN"}, + {Id: "01J2JY45S90MWF6ZER08BFGPGP"}, + {Id: "01J2JY5JR0C9V64EP32RPH61E7"}, + {Id: "01J2JY61BG7QBPNK54EY8N0T6K"}, + {Id: "01J2JZ0A7MPZMR0R745HAZD1S9"}, + {Id: "01J2JZ0RY9WCA01S322EG201R8"}, + } + + var buf bytes.Buffer + expected := `merge +read [id:"01J2JY1K5J4T2WNDV05CHVFCA9" id:"01J2JY21VVYYV4PMDGK4TVMZ6H"] +read [id:"01J2JY2GF83EF0QMW94T19MXHN" id:"01J2JY45S90MWF6ZER08BFGPGP"] +read [id:"01J2JY5JR0C9V64EP32RPH61E7" id:"01J2JY61BG7QBPNK54EY8N0T6K"] +read [id:"01J2JZ0A7MPZMR0R745HAZD1S9" id:"01J2JZ0RY9WCA01S322EG201R8"] +` + + p := Build(blocks, 2, 5).Proto() + n := []*querybackendv1.QueryPlan{p} + var x *QueryPlan + for len(n) > 0 { + x, n = Open(n[0]), n[1:] + + switch r := x.Root(); r.Type { + case NodeMerge: + _, _ = fmt.Fprintln(&buf, "merge") + c := r.Children() + for c.Next() { + n = append(n, c.At().Plan().Proto()) + } + + case NodeRead: + _, _ = fmt.Fprintln(&buf, "read", iter.MustSlice(r.Blocks())) + + default: + panic("query plan: unknown node type") + } + } + + require.Equal(t, expected, buf.String()) +} + +func Test_Plan_skip_top_merge(t *testing.T) { + blocks := []*metastorev1.BlockMeta{ + {Id: "01J2JY1K5J4T2WNDV05CHVFCA9"}, + {Id: "01J2JY21VVYYV4PMDGK4TVMZ6H"}, + } + + var buf bytes.Buffer + expected := `[id:"01J2JY1K5J4T2WNDV05CHVFCA9" id:"01J2JY21VVYYV4PMDGK4TVMZ6H"]` + + p := Build(blocks, 2, 5).Proto() + n := []*querybackendv1.QueryPlan{p} + var x *QueryPlan + for len(n) > 0 { + x, n = Open(n[0]), n[1:] + + switch r := x.Root(); r.Type { + case NodeMerge: + _, _ = fmt.Fprintln(&buf, "merge") + c := r.Children() + for c.Next() { + n = append(n, c.At().Plan().Proto()) + } + + case NodeRead: + _, _ = fmt.Fprint(&buf, iter.MustSlice(r.Blocks())) + + default: + panic("query plan: unknown node type") + } + } + + require.Equal(t, expected, buf.String()) +} diff --git a/pkg/experiment/querybackend/queryplan/testdata/plan.txt b/pkg/experiment/querybackend/queryplan/testdata/plan.txt new file mode 100644 index 0000000000..84b82bf71e --- /dev/null +++ b/pkg/experiment/querybackend/queryplan/testdata/plan.txt @@ -0,0 +1,46 @@ +merge {children: 2, nodes: 21, blocks: 25} + merge {children: 3, nodes: 13, blocks: 18} + merge {children: 3, nodes: 4, blocks: 6} + read {children: 2, nodes: 1, blocks: 2} + id:"1" + id:"2" + read {children: 2, nodes: 1, blocks: 2} + id:"3" + id:"4" + read {children: 2, nodes: 1, blocks: 2} + id:"5" + id:"6" + merge {children: 3, nodes: 4, blocks: 6} + read {children: 2, nodes: 1, blocks: 2} + id:"7" + id:"8" + read {children: 2, nodes: 1, blocks: 2} + id:"9" + id:"10" + read {children: 2, nodes: 1, blocks: 2} + id:"11" + id:"12" + merge {children: 3, nodes: 4, blocks: 6} + read {children: 2, nodes: 1, blocks: 2} + id:"13" + id:"14" + read {children: 2, nodes: 1, blocks: 2} + id:"15" + id:"16" + read {children: 2, nodes: 1, blocks: 2} + id:"17" + id:"18" + merge {children: 2, nodes: 7, blocks: 7} + merge {children: 3, nodes: 4, blocks: 6} + read {children: 2, nodes: 1, blocks: 2} + id:"19" + id:"20" + read {children: 2, nodes: 1, blocks: 2} + id:"21" + id:"22" + read {children: 2, nodes: 1, blocks: 2} + id:"23" + id:"24" + merge {children: 1, nodes: 2, blocks: 1} + read {children: 1, nodes: 1, blocks: 1} + id:"25" diff --git a/pkg/experiment/querybackend/report_aggregator.go b/pkg/experiment/querybackend/report_aggregator.go new file mode 100644 index 0000000000..44be8d83f9 --- /dev/null +++ b/pkg/experiment/querybackend/report_aggregator.go @@ -0,0 +1,155 @@ +package querybackend + +import ( + "fmt" + "sync" + + querybackendv1 "github.com/grafana/pyroscope/api/gen/proto/go/querybackend/v1" +) + +var ( + aggregatorMutex = new(sync.RWMutex) + aggregators = map[querybackendv1.ReportType]aggregatorProvider{} + queryReportType = map[querybackendv1.QueryType]querybackendv1.ReportType{} +) + +type aggregatorProvider func(*querybackendv1.InvokeRequest) aggregator + +type aggregator interface { + // The method is called concurrently. + aggregate(*querybackendv1.Report) error + // build the aggregation result. It's guaranteed that aggregate() + // was called at least once before report() is called. + build() *querybackendv1.Report +} + +func registerAggregator(t querybackendv1.ReportType, ap aggregatorProvider) { + aggregatorMutex.Lock() + defer aggregatorMutex.Unlock() + _, ok := aggregators[t] + if ok { + panic(fmt.Sprintf("%s: aggregator already registered", t)) + } + aggregators[t] = ap +} + +func getAggregator(r *querybackendv1.InvokeRequest, x *querybackendv1.Report) (aggregator, error) { + aggregatorMutex.RLock() + defer aggregatorMutex.RUnlock() + a, ok := aggregators[x.ReportType] + if !ok { + return nil, fmt.Errorf("unknown build type %s", x.ReportType) + } + return a(r), nil +} + +func registerQueryReportType(q querybackendv1.QueryType, r querybackendv1.ReportType) { + aggregatorMutex.Lock() + defer aggregatorMutex.Unlock() + v, ok := queryReportType[q] + if ok { + panic(fmt.Sprintf("%s: handler already registered (%s)", q, v)) + } + queryReportType[q] = r +} + +func QueryReportType(q querybackendv1.QueryType) querybackendv1.ReportType { + aggregatorMutex.RLock() + defer aggregatorMutex.RUnlock() + r, ok := queryReportType[q] + if !ok { + panic(fmt.Sprintf("unknown build type %s", q)) + } + return r +} + +type reportAggregator struct { + request *querybackendv1.InvokeRequest + sm sync.Mutex + staged map[querybackendv1.ReportType]*querybackendv1.Report + aggregators map[querybackendv1.ReportType]aggregator +} + +func newAggregator(request *querybackendv1.InvokeRequest) *reportAggregator { + return &reportAggregator{ + request: request, + staged: make(map[querybackendv1.ReportType]*querybackendv1.Report), + aggregators: make(map[querybackendv1.ReportType]aggregator), + } +} + +func (ra *reportAggregator) aggregateResponse(resp *querybackendv1.InvokeResponse, err error) error { + if err != nil { + return err + } + for _, r := range resp.Reports { + if err = ra.aggregateReport(r); err != nil { + return err + } + } + return nil +} + +func (ra *reportAggregator) aggregateReport(r *querybackendv1.Report) (err error) { + if r == nil { + return nil + } + ra.sm.Lock() + v, found := ra.staged[r.ReportType] + if !found { + // We delay aggregation until we have at least two + // reports of the same type. Otherwise, we just store + // the report and will return it as is, if it is the + // only one. + ra.staged[r.ReportType] = r + ra.sm.Unlock() + return nil + } + // Found a staged report of the same type. + if v != nil { + // It should be aggregated and removed from the table. + err = ra.aggregateReportNoCheck(v) + ra.staged[r.ReportType] = nil + } + ra.sm.Unlock() + if err != nil { + return err + } + return ra.aggregateReportNoCheck(r) +} + +func (ra *reportAggregator) aggregateReportNoCheck(report *querybackendv1.Report) (err error) { + a, ok := ra.aggregators[report.ReportType] + if !ok { + a, err = getAggregator(ra.request, report) + if err != nil { + return err + } + ra.aggregators[report.ReportType] = a + } + return a.aggregate(report) +} + +func (ra *reportAggregator) aggregateStaged() error { + for _, r := range ra.staged { + if r != nil { + if err := ra.aggregateReportNoCheck(r); err != nil { + return err + } + } + } + return nil +} + +func (ra *reportAggregator) response() (*querybackendv1.InvokeResponse, error) { + if err := ra.aggregateStaged(); err != nil { + return nil, err + } + reports := make([]*querybackendv1.Report, 0, len(ra.staged)) + for t, a := range ra.aggregators { + r := a.build() + r.ReportType = t + reports = append(reports, r) + } + return &querybackendv1.InvokeResponse{Reports: reports}, nil +} diff --git a/pkg/experiment/queryfrontend/frontend_meta.go b/pkg/experiment/queryfrontend/frontend_meta.go new file mode 100644 index 0000000000..8917aa430f --- /dev/null +++ b/pkg/experiment/queryfrontend/frontend_meta.go @@ -0,0 +1,188 @@ +package queryfrontend + +import ( + "context" + "fmt" + "math/rand" + "slices" + "strings" + + "github.com/go-kit/log" + "github.com/go-kit/log/level" + "github.com/prometheus/prometheus/model/labels" + "github.com/prometheus/prometheus/promql/parser" + + metastorev1 "github.com/grafana/pyroscope/api/gen/proto/go/metastore/v1" + querybackendv1 "github.com/grafana/pyroscope/api/gen/proto/go/querybackend/v1" + metastoreclient "github.com/grafana/pyroscope/pkg/experiment/metastore/client" + "github.com/grafana/pyroscope/pkg/experiment/querybackend" + querybackendclient "github.com/grafana/pyroscope/pkg/experiment/querybackend/client" + "github.com/grafana/pyroscope/pkg/experiment/querybackend/queryplan" + phlaremodel "github.com/grafana/pyroscope/pkg/model" +) + +func ListMetadata( + ctx context.Context, + client *metastoreclient.Client, + logger log.Logger, + tenants []string, + startTime, endTime int64, + query string, +) ([]*metastorev1.BlockMeta, error) { + _ = level.Info(logger).Log("msg", "listing metadata", + "tenants", strings.Join(tenants, ","), + "start", startTime, + "end", endTime, + "query", query, + ) + resp, err := client.QueryMetadata(ctx, &metastorev1.QueryMetadataRequest{ + TenantId: tenants, + StartTime: startTime, + EndTime: endTime, + Query: query, + }) + if err != nil { + // TODO: Not sure if we want to pass it through + return nil, err + } + // TODO: Metrics + printStats(logger, resp.Blocks) + return resp.Blocks, nil +} + +func printStats(logger log.Logger, blocks []*metastorev1.BlockMeta) { + type blockMetaStats struct { + level uint32 + minTime int64 + maxTime int64 + size uint64 + count int + } + m := make(map[uint32]*blockMetaStats) + for _, b := range blocks { + s, ok := m[b.CompactionLevel] + if !ok { + s = &blockMetaStats{level: b.CompactionLevel} + m[b.CompactionLevel] = s + } + for _, x := range b.Datasets { + s.size += x.Size + } + s.count++ + } + sorted := make([]*blockMetaStats, 0, len(m)) + for _, s := range m { + sorted = append(sorted, s) + } + slices.SortFunc(sorted, func(a, b *blockMetaStats) int { + return int(a.level - b.level) + }) + fields := make([]interface{}, 0, 4+len(sorted)*2) + fields = append(fields, "msg", "block metadata list", "blocks_total", fmt.Sprint(len(blocks))) + for _, s := range sorted { + fields = append(fields, + fmt.Sprintf("l%d_blocks", s.level), fmt.Sprint(s.count), + fmt.Sprintf("l%d_size", s.level), fmt.Sprint(s.size), + ) + } + _ = level.Info(logger).Log(fields...) +} + +var xrand = rand.New(rand.NewSource(4349676827832284783)) + +func Query( + ctx context.Context, + startTime, endTime int64, + tenants []string, + labelSelector string, + q *querybackendv1.Query, + mc *metastoreclient.Client, + qc *querybackendclient.Client, + logger log.Logger, +) (*querybackendv1.Report, error) { + blocks, err := ListMetadata(ctx, mc, logger, tenants, startTime, endTime, labelSelector) + if err != nil { + return nil, err + } + if len(blocks) == 0 { + return nil, nil + } + // Randomize the order of blocks to avoid hotspots. + xrand.Shuffle(len(blocks), func(i, j int) { + blocks[i], blocks[j] = blocks[j], blocks[i] + }) + // TODO: Params. + p := queryplan.Build(blocks, 2, 10) + resp, err := qc.Invoke(ctx, &querybackendv1.InvokeRequest{ + Tenant: tenants, + StartTime: startTime, + EndTime: endTime, + LabelSelector: labelSelector, + Options: &querybackendv1.InvokeOptions{}, + QueryPlan: p.Proto(), + Query: []*querybackendv1.Query{q}, + }) + if err != nil { + return nil, err + } + return findReport(querybackend.QueryReportType(q.QueryType), resp.Reports), nil +} + +func BuildLabelSelectorFromMatchers(matchers []string) (string, error) { + parsed, err := parseMatchers(matchers) + if err != nil { + return "", fmt.Errorf("parsing label selector: %w", err) + } + return matchersToLabelSelector(parsed), nil +} + +func BuildLabelSelectorWithProfileType(labelSelector, profileTypeID string) (string, error) { + matchers, err := parser.ParseMetricSelector(labelSelector) + if err != nil { + return "", fmt.Errorf("parsing label selector %q: %w", labelSelector, err) + } + profileType, err := phlaremodel.ParseProfileTypeSelector(profileTypeID) + if err != nil { + return "", fmt.Errorf("parsing profile type ID %q: %w", profileTypeID, err) + } + matchers = append(matchers, phlaremodel.SelectorFromProfileType(profileType)) + return matchersToLabelSelector(matchers), nil +} + +func parseMatchers(matchers []string) ([]*labels.Matcher, error) { + parsed := make([]*labels.Matcher, 0, len(matchers)) + for _, m := range matchers { + s, err := parser.ParseMetricSelector(m) + if err != nil { + return nil, fmt.Errorf("failed to parse label selector %q: %w", s, err) + } + parsed = append(parsed, s...) + } + return parsed, nil +} + +func matchersToLabelSelector(matchers []*labels.Matcher) string { + var q strings.Builder + q.WriteByte('{') + for i, m := range matchers { + if i > 0 { + q.WriteByte(',') + } + q.WriteString(m.Name) + q.WriteString(m.Type.String()) + q.WriteByte('"') + q.WriteString(m.Value) + q.WriteByte('"') + } + q.WriteByte('}') + return q.String() +} + +func findReport(r querybackendv1.ReportType, reports []*querybackendv1.Report) *querybackendv1.Report { + for _, x := range reports { + if x.ReportType == r { + return x + } + } + return nil +} diff --git a/pkg/experiment/queryfrontend/frontend_profile_types.go b/pkg/experiment/queryfrontend/frontend_profile_types.go new file mode 100644 index 0000000000..da4b760fab --- /dev/null +++ b/pkg/experiment/queryfrontend/frontend_profile_types.go @@ -0,0 +1,155 @@ +package queryfrontend + +import ( + "context" + "slices" + "sort" + + "connectrpc.com/connect" + "github.com/go-kit/log" + + querierv1 "github.com/grafana/pyroscope/api/gen/proto/go/querier/v1" + typesv1 "github.com/grafana/pyroscope/api/gen/proto/go/types/v1" + metastoreclient "github.com/grafana/pyroscope/pkg/experiment/metastore/client" + phlaremodel "github.com/grafana/pyroscope/pkg/model" +) + +var profileTypeLabels2 = []string{ + "__profile_type__", + "service_name", +} + +var profileTypeLabels5 = []string{ + "__name__", + "__profile_type__", + "__type__", + "pyroscope_app", + "service_name", +} + +func IsProfileTypeQuery(labels, matchers []string) bool { + if len(matchers) > 0 { + return false + } + var s []string + switch len(labels) { + case 2: + s = profileTypeLabels2 + case 5: + s = profileTypeLabels5 + default: + return false + } + sort.Strings(labels) + return slices.Compare(s, labels) == 0 +} + +func ListProfileTypesFromMetadataAsSeriesLabels( + ctx context.Context, + client *metastoreclient.Client, + logger log.Logger, + tenants []string, + startTime, endTime int64, + labels []string, + +) (*connect.Response[querierv1.SeriesResponse], error) { + resp, err := listProfileTypesFromMetadata(ctx, client, logger, tenants, startTime, endTime) + if err != nil { + return nil, err + } + return connect.NewResponse(&querierv1.SeriesResponse{ + LabelsSet: resp.buildSeriesLabels(labels), + }), nil +} + +func listProfileTypesFromMetadata( + ctx context.Context, + client *metastoreclient.Client, + logger log.Logger, + tenants []string, + startTime, endTime int64, +) (*ptypes, error) { + metas, err := ListMetadata(ctx, client, logger, tenants, startTime, endTime, "{}") + if err != nil { + return nil, err + } + p := newProfileTypesResponseBuilder(len(metas) * 8) + for _, m := range metas { + for _, s := range m.Datasets { + p.addServiceProfileTypes(s.Name, s.ProfileTypes...) + } + } + return p, nil +} + +type ptypes struct { + services map[string]map[string]struct{} +} + +func newProfileTypesResponseBuilder(size int) *ptypes { + return &ptypes{ + services: make(map[string]map[string]struct{}, size), + } +} + +func (p *ptypes) addServiceProfileTypes(s string, types ...string) { + sp, ok := p.services[s] + if !ok { + sp = make(map[string]struct{}, len(types)) + p.services[s] = sp + } + for _, t := range types { + sp[t] = struct{}{} + } +} + +func (p *ptypes) buildSeriesLabels(names []string) (labels []*typesv1.Labels) { + switch len(names) { + case 2: + labels = p.buildSeriesLabels2() + case 5: + labels = p.buildSeriesLabels5() + default: + panic("bug: invalid request: expected 2 or 5 label names") + } + slices.SortFunc(labels, func(a, b *typesv1.Labels) int { + return phlaremodel.CompareLabelPairs(a.Labels, b.Labels) + }) + return labels +} + +func (p *ptypes) buildSeriesLabels2() []*typesv1.Labels { + labels := make([]*typesv1.Labels, 0, len(p.services)*4) + for n, types := range p.services { + for t := range types { + labels = append(labels, &typesv1.Labels{ + Labels: []*typesv1.LabelPair{ + {Name: "__profile_type__", Value: t}, + {Name: "service_name", Value: n}, + }, + }) + } + } + return labels +} + +func (p *ptypes) buildSeriesLabels5() []*typesv1.Labels { + labels := make([]*typesv1.Labels, 0, len(p.services)*4) + for n, types := range p.services { + for t := range types { + pt, err := phlaremodel.ParseProfileTypeSelector(t) + if err != nil { + panic(err) + } + labels = append(labels, &typesv1.Labels{ + Labels: []*typesv1.LabelPair{ + {Name: "__profile_type__", Value: t}, + {Name: "service_name", Value: n}, + {Name: "__name__", Value: pt.Name}, + {Name: "__type__", Value: pt.SampleType}, + }, + }) + } + } + return labels +} diff --git a/pkg/frontend/frontend_select_series.go b/pkg/frontend/frontend_select_time_series.go similarity index 96% rename from pkg/frontend/frontend_select_series.go rename to pkg/frontend/frontend_select_time_series.go index a7cf37ac78..0c1b4ea9b3 100644 --- a/pkg/frontend/frontend_select_series.go +++ b/pkg/frontend/frontend_select_time_series.go @@ -51,7 +51,7 @@ func (f *Frontend) SelectSeries(ctx context.Context, g.SetLimit(maxConcurrent) } - m := phlaremodel.NewSeriesMerger(false) + m := phlaremodel.NewTimeSeriesMerger(false) interval := validationutil.MaxDurationOrZeroPerTenant(tenantIDs, f.limits.QuerySplitDuration) intervals := NewTimeIntervalIterator(time.UnixMilli(c.Msg.Start), time.UnixMilli(c.Msg.End), interval, WithAlignment(time.Second*time.Duration(c.Msg.Step))) @@ -75,7 +75,7 @@ func (f *Frontend) SelectSeries(ctx context.Context, if err != nil { return err } - m.MergeSeries(resp.Msg.Series) + m.MergeTimeSeries(resp.Msg.Series) return nil }) } @@ -84,5 +84,5 @@ func (f *Frontend) SelectSeries(ctx context.Context, return nil, err } - return connect.NewResponse(&querierv1.SelectSeriesResponse{Series: m.Series()}), nil + return connect.NewResponse(&querierv1.SelectSeriesResponse{Series: m.TimeSeries()}), nil } diff --git a/pkg/frontend/frontend_series.go b/pkg/frontend/frontend_series_labels.go similarity index 100% rename from pkg/frontend/frontend_series.go rename to pkg/frontend/frontend_series_labels.go diff --git a/pkg/iter/iter.go b/pkg/iter/iter.go index 12a183426f..1460289246 100644 --- a/pkg/iter/iter.go +++ b/pkg/iter/iter.go @@ -163,6 +163,14 @@ func Slice[T any](it Iterator[T]) ([]T, error) { return result, it.Err() } +func MustSlice[T any](it Iterator[T]) []T { + s, err := Slice(it) + if err != nil { + panic(err) + } + return s +} + // CloneN returns N copy of the iterator. // The returned iterators are independent of the original iterator. // The original might be exhausted and should be discarded. diff --git a/pkg/model/labels_merger.go b/pkg/model/labels_merger.go new file mode 100644 index 0000000000..ebfb66cfa6 --- /dev/null +++ b/pkg/model/labels_merger.go @@ -0,0 +1,101 @@ +package model + +import ( + "slices" + "sort" + "sync" + + typesv1 "github.com/grafana/pyroscope/api/gen/proto/go/types/v1" +) + +type LabelMerger struct { + mu sync.Mutex + names map[string]struct{} + values map[string]struct{} + series map[uint64]*typesv1.Labels +} + +func NewLabelMerger() *LabelMerger { + return &LabelMerger{ + names: make(map[string]struct{}), + values: make(map[string]struct{}), + series: make(map[uint64]*typesv1.Labels), + } +} + +func (m *LabelMerger) MergeLabelNames(names []string) { + m.mu.Lock() + defer m.mu.Unlock() + for _, n := range names { + m.names[n] = struct{}{} + } +} + +func (m *LabelMerger) MergeLabelValues(values []string) { + m.mu.Lock() + defer m.mu.Unlock() + for _, v := range values { + m.values[v] = struct{}{} + } +} + +func (m *LabelMerger) HasNames() bool { + return len(m.names) > 0 +} + +func (m *LabelMerger) LabelNames() []string { + m.mu.Lock() + defer m.mu.Unlock() + s := make([]string, len(m.names)) + var i int + for n := range m.names { + s[i] = n + i++ + } + sort.Strings(s) + return s +} + +func (m *LabelMerger) HasValues() bool { + return len(m.values) > 0 +} + +func (m *LabelMerger) LabelValues() []string { + m.mu.Lock() + defer m.mu.Unlock() + s := make([]string, len(m.values)) + var i int + for v := range m.values { + s[i] = v + i++ + } + sort.Strings(s) + return s +} + +func (m *LabelMerger) MergeSeries(series []*typesv1.Labels) { + m.mu.Lock() + defer m.mu.Unlock() + for _, s := range series { + m.series[Labels(s.Labels).Hash()] = s + } +} + +func (m *LabelMerger) SeriesLabels() []*typesv1.Labels { + m.mu.Lock() + defer m.mu.Unlock() + s := make([]*typesv1.Labels, len(m.series)) + var i int + for _, v := range m.series { + s[i] = v + i++ + } + slices.SortFunc(s, func(a, b *typesv1.Labels) int { + return CompareLabelPairs(a.Labels, b.Labels) + }) + return s +} + +func (m *LabelMerger) HasSeries() bool { + return len(m.series) > 0 +} diff --git a/pkg/iter/profiles.go b/pkg/model/profiles.go similarity index 77% rename from pkg/iter/profiles.go rename to pkg/model/profiles.go index b4d00ea01b..5ea1fe6b65 100644 --- a/pkg/iter/profiles.go +++ b/pkg/model/profiles.go @@ -1,10 +1,10 @@ -package iter +package model import ( "github.com/grafana/dskit/multierror" "github.com/prometheus/common/model" - phlaremodel "github.com/grafana/pyroscope/pkg/model" + "github.com/grafana/pyroscope/pkg/iter" "github.com/grafana/pyroscope/pkg/util/loser" ) @@ -13,20 +13,20 @@ type Timestamp interface { } type Profile interface { - Labels() phlaremodel.Labels + Labels() Labels Timestamp } func lessProfile(p1, p2 Profile) bool { if p1.Timestamp() == p2.Timestamp() { // todo we could compare SeriesRef here - return phlaremodel.CompareLabelPairs(p1.Labels(), p2.Labels()) < 0 + return CompareLabelPairs(p1.Labels(), p2.Labels()) < 0 } return p1.Timestamp() < p2.Timestamp() } type MergeIterator[P Profile] struct { - tree *loser.Tree[P, Iterator[P]] + tree *loser.Tree[P, iter.Iterator[P]] closeErrs multierror.MultiError current P deduplicate bool @@ -35,34 +35,34 @@ type MergeIterator[P Profile] struct { // NewMergeIterator returns an iterator that k-way merges the given iterators. // The given iterators must be sorted by timestamp and labels themselves. // Optionally, the iterator can deduplicate profiles with the same timestamp and labels. -func NewMergeIterator[P Profile](max P, deduplicate bool, iters ...Iterator[P]) Iterator[P] { +func NewMergeIterator[P Profile](max P, deduplicate bool, iters ...iter.Iterator[P]) iter.Iterator[P] { if len(iters) == 0 { - return NewEmptyIterator[P]() + return iter.NewEmptyIterator[P]() } if len(iters) == 1 { // No need to merge a single iterator. // We should never allow a single iterator to be passed in because return iters[0] } - iter := &MergeIterator[P]{ + m := &MergeIterator[P]{ deduplicate: deduplicate, current: max, } - iter.tree = loser.New( + m.tree = loser.New( iters, max, - func(s Iterator[P]) P { + func(s iter.Iterator[P]) P { return s.At() }, func(p1, p2 P) bool { return lessProfile(p1, p2) }, - func(s Iterator[P]) { + func(s iter.Iterator[P]) { if err := s.Close(); err != nil { - iter.closeErrs.Add(err) + m.closeErrs.Add(err) } }) - return iter + return m } func (i *MergeIterator[P]) Next() bool { @@ -74,7 +74,7 @@ func (i *MergeIterator[P]) Next() bool { return true } if next.At().Timestamp() != i.current.Timestamp() || - phlaremodel.CompareLabelPairs(next.At().Labels(), i.current.Labels()) != 0 { + CompareLabelPairs(next.At().Labels(), i.current.Labels()) != 0 { i.current = next.At() return true } @@ -97,11 +97,11 @@ func (i *MergeIterator[P]) Close() error { } type TimeRangedIterator[T Timestamp] struct { - Iterator[T] + iter.Iterator[T] min, max model.Time } -func NewTimeRangedIterator[T Timestamp](it Iterator[T], min, max model.Time) Iterator[T] { +func NewTimeRangedIterator[T Timestamp](it iter.Iterator[T], min, max model.Time) iter.Iterator[T] { return &TimeRangedIterator[T]{ Iterator: it, min: min, diff --git a/pkg/iter/profiles_test.go b/pkg/model/profiles_test.go similarity index 86% rename from pkg/iter/profiles_test.go rename to pkg/model/profiles_test.go index 22701efa27..185b81bca1 100644 --- a/pkg/iter/profiles_test.go +++ b/pkg/model/profiles_test.go @@ -1,4 +1,4 @@ -package iter +package model import ( "math" @@ -8,21 +8,21 @@ import ( "github.com/stretchr/testify/require" "go.uber.org/goleak" - phlaremodel "github.com/grafana/pyroscope/pkg/model" + "github.com/grafana/pyroscope/pkg/iter" ) var ( - aLabels = phlaremodel.LabelsFromStrings("foo", "a") - bLabels = phlaremodel.LabelsFromStrings("foo", "b") - cLabels = phlaremodel.LabelsFromStrings("foo", "c") + aLabels = LabelsFromStrings("foo", "a") + bLabels = LabelsFromStrings("foo", "b") + cLabels = LabelsFromStrings("foo", "c") ) type profile struct { - labels phlaremodel.Labels + labels Labels timestamp model.Time } -func (p profile) Labels() phlaremodel.Labels { +func (p profile) Labels() Labels { return p.labels } @@ -120,9 +120,9 @@ func TestMergeIterator(t *testing.T) { } { tt := tt t.Run(tt.name, func(t *testing.T) { - iters := make([]Iterator[profile], len(tt.input)) + iters := make([]iter.Iterator[profile], len(tt.input)) for i, input := range tt.input { - iters[i] = NewSliceIterator(input) + iters[i] = iter.NewSliceIterator(input) } it := NewMergeIterator( profile{timestamp: math.MaxInt64}, @@ -162,9 +162,9 @@ func Test_BufferedIterator(t *testing.T) { }, } { t.Run(tc.name, func(t *testing.T) { - actual, err := Slice( - NewBufferedIterator( - NewSliceIterator(tc.in), tc.size), + actual, err := iter.Slice( + iter.NewBufferedIterator( + iter.NewSliceIterator(tc.in), tc.size), ) require.NoError(t, err) require.Equal(t, tc.in, actual) @@ -175,8 +175,8 @@ func Test_BufferedIterator(t *testing.T) { func Test_BufferedIteratorClose(t *testing.T) { defer goleak.VerifyNone(t, goleak.IgnoreCurrent()) - it := NewBufferedIterator( - NewSliceIterator(generatesProfiles(t, 100)), 10) + it := iter.NewBufferedIterator( + iter.NewSliceIterator(generatesProfiles(t, 100)), 10) require.NoError(t, it.Close()) } diff --git a/pkg/model/time_series.go b/pkg/model/time_series.go new file mode 100644 index 0000000000..b8d7182c5b --- /dev/null +++ b/pkg/model/time_series.go @@ -0,0 +1,200 @@ +package model + +import ( + "math" + "sort" + + "github.com/prometheus/common/model" + "github.com/samber/lo" + + typesv1 "github.com/grafana/pyroscope/api/gen/proto/go/types/v1" + "github.com/grafana/pyroscope/pkg/iter" +) + +type TimeSeriesValue struct { + Ts int64 + Lbs []*typesv1.LabelPair + LabelsHash uint64 + Value float64 +} + +func (p TimeSeriesValue) Labels() Labels { return p.Lbs } +func (p TimeSeriesValue) Timestamp() model.Time { return model.Time(p.Ts) } + +type TimeSeriesIterator struct { + point []*typesv1.Point + curr TimeSeriesValue +} + +func NewSeriesIterator(lbs []*typesv1.LabelPair, points []*typesv1.Point) *TimeSeriesIterator { + return &TimeSeriesIterator{ + point: points, + + curr: TimeSeriesValue{ + Lbs: lbs, + LabelsHash: Labels(lbs).Hash(), + }, + } +} + +func (s *TimeSeriesIterator) Next() bool { + if len(s.point) == 0 { + return false + } + p := s.point[0] + s.point = s.point[1:] + s.curr.Ts = p.Timestamp + s.curr.Value = p.Value + return true +} + +func (s *TimeSeriesIterator) At() TimeSeriesValue { return s.curr } +func (s *TimeSeriesIterator) Err() error { return nil } +func (s *TimeSeriesIterator) Close() error { return nil } + +func NewTimeSeriesMergeIterator(series []*typesv1.Series) iter.Iterator[TimeSeriesValue] { + iters := make([]iter.Iterator[TimeSeriesValue], 0, len(series)) + for _, s := range series { + iters = append(iters, NewSeriesIterator(s.Labels, s.Points)) + } + return NewMergeIterator(TimeSeriesValue{Ts: math.MaxInt64}, false, iters...) +} + +type TimeSeriesAggregator interface { + Add(ts int64, value float64) + GetAndReset() *typesv1.Point + IsEmpty() bool + GetTimestamp() int64 +} + +func NewTimeSeriesAggregator(aggregation *typesv1.TimeSeriesAggregationType) TimeSeriesAggregator { + if aggregation == nil { + return &sumTimeSeriesAggregator{ts: -1} + } + if *aggregation == typesv1.TimeSeriesAggregationType_TIME_SERIES_AGGREGATION_TYPE_AVERAGE { + return &avgTimeSeriesAggregator{ts: -1} + } + return &sumTimeSeriesAggregator{ts: -1} +} + +type sumTimeSeriesAggregator struct { + ts int64 + sum float64 +} + +func (a *sumTimeSeriesAggregator) Add(ts int64, value float64) { + a.ts = ts + a.sum += value +} + +func (a *sumTimeSeriesAggregator) GetAndReset() *typesv1.Point { + tsCopy := a.ts + sumCopy := a.sum + a.ts = -1 + a.sum = 0 + return &typesv1.Point{ + Timestamp: tsCopy, + Value: sumCopy, + } +} + +func (a *sumTimeSeriesAggregator) IsEmpty() bool { return a.ts == -1 } +func (a *sumTimeSeriesAggregator) GetTimestamp() int64 { return a.ts } + +type avgTimeSeriesAggregator struct { + ts int64 + sum float64 + count int64 +} + +func (a *avgTimeSeriesAggregator) Add(ts int64, value float64) { + a.ts = ts + a.sum += value + a.count++ +} + +func (a *avgTimeSeriesAggregator) GetAndReset() *typesv1.Point { + avg := a.sum / float64(a.count) + tsCopy := a.ts + a.ts = -1 + a.sum = 0 + a.count = 0 + return &typesv1.Point{ + Timestamp: tsCopy, + Value: avg, + } +} + +func (a *avgTimeSeriesAggregator) IsEmpty() bool { return a.ts == -1 } +func (a *avgTimeSeriesAggregator) GetTimestamp() int64 { return a.ts } + +// RangeSeries aggregates profiles into series. +// Series contains points spaced by step from start to end. +// Profiles from the same step are aggregated into one point. +func RangeSeries(it iter.Iterator[TimeSeriesValue], start, end, step int64, aggregation *typesv1.TimeSeriesAggregationType) []*typesv1.Series { + defer it.Close() + seriesMap := make(map[uint64]*typesv1.Series) + aggregators := make(map[uint64]TimeSeriesAggregator) + + if !it.Next() { + return nil + } + + // advance from the start to the end, adding each step results to the map. +Outer: + for currentStep := start; currentStep <= end; currentStep += step { + for { + aggregator, ok := aggregators[it.At().LabelsHash] + if !ok { + aggregator = NewTimeSeriesAggregator(aggregation) + aggregators[it.At().LabelsHash] = aggregator + } + if it.At().Ts > currentStep { + if !aggregator.IsEmpty() { + series := seriesMap[it.At().LabelsHash] + series.Points = append(series.Points, aggregator.GetAndReset()) + } + break // no more profiles for the currentStep + } + // find or create series + series, ok := seriesMap[it.At().LabelsHash] + if !ok { + seriesMap[it.At().LabelsHash] = &typesv1.Series{ + Labels: it.At().Lbs, + Points: []*typesv1.Point{}, + } + aggregator.Add(currentStep, it.At().Value) + if !it.Next() { + break Outer + } + continue + } + // Aggregate point if it is in the current step. + if aggregator.GetTimestamp() == currentStep { + aggregator.Add(currentStep, it.At().Value) + if !it.Next() { + break Outer + } + continue + } + // Next step is missing + if !aggregator.IsEmpty() { + series.Points = append(series.Points, aggregator.GetAndReset()) + } + aggregator.Add(currentStep, it.At().Value) + if !it.Next() { + break Outer + } + } + } + for lblHash, aggregator := range aggregators { + if !aggregator.IsEmpty() { + seriesMap[lblHash].Points = append(seriesMap[lblHash].Points, aggregator.GetAndReset()) + } + } + series := lo.Values(seriesMap) + sort.Slice(series, func(i, j int) bool { + return CompareLabelPairs(series[i].Labels, series[j].Labels) < 0 + }) + return series +} diff --git a/pkg/model/time_series_builder.go b/pkg/model/time_series_builder.go new file mode 100644 index 0000000000..9d8b070fae --- /dev/null +++ b/pkg/model/time_series_builder.go @@ -0,0 +1,77 @@ +package model + +import ( + "sort" + + "github.com/prometheus/common/model" + "github.com/samber/lo" + + typesv1 "github.com/grafana/pyroscope/api/gen/proto/go/types/v1" +) + +type TimeSeriesBuilder struct { + labelsByFingerprint map[model.Fingerprint]string + labelBuf []byte + by []string + + series seriesByLabels +} + +func NewTimeSeriesBuilder(by ...string) *TimeSeriesBuilder { + var b TimeSeriesBuilder + b.Init(by...) + return &b +} + +func (s *TimeSeriesBuilder) Init(by ...string) { + s.labelsByFingerprint = map[model.Fingerprint]string{} + s.series = make(seriesByLabels) + s.labelBuf = make([]byte, 0, 1024) + s.by = by +} + +func (s *TimeSeriesBuilder) Add(fp model.Fingerprint, lbs Labels, ts int64, value float64) { + labelsByString, ok := s.labelsByFingerprint[fp] + if !ok { + s.labelBuf = lbs.BytesWithLabels(s.labelBuf, s.by...) + labelsByString = string(s.labelBuf) + s.labelsByFingerprint[fp] = labelsByString + if _, ok := s.series[labelsByString]; !ok { + s.series[labelsByString] = &typesv1.Series{ + Labels: lbs.WithLabels(s.by...), + Points: []*typesv1.Point{ + { + Timestamp: ts, + Value: value, + }, + }, + } + return + } + } + series := s.series[labelsByString] + series.Points = append(series.Points, &typesv1.Point{ + Timestamp: ts, + Value: value, + }) +} + +func (s *TimeSeriesBuilder) Build() []*typesv1.Series { + return s.series.normalize() +} + +type seriesByLabels map[string]*typesv1.Series + +func (m seriesByLabels) normalize() []*typesv1.Series { + result := lo.Values(m) + sort.Slice(result, func(i, j int) bool { + return CompareLabelPairs(result[i].Labels, result[j].Labels) < 0 + }) + // we have to sort the points in each series because labels reduction may have changed the order + for _, s := range result { + sort.Slice(s.Points, func(i, j int) bool { + return s.Points[i].Timestamp < s.Points[j].Timestamp + }) + } + return result +} diff --git a/pkg/model/series.go b/pkg/model/time_series_merger.go similarity index 71% rename from pkg/model/series.go rename to pkg/model/time_series_merger.go index 5de5a8f2dd..dee074eecf 100644 --- a/pkg/model/series.go +++ b/pkg/model/time_series_merger.go @@ -8,34 +8,34 @@ import ( ) func MergeSeries(aggregation *typesv1.TimeSeriesAggregationType, series ...[]*typesv1.Series) []*typesv1.Series { - var m *SeriesMerger + var m *TimeSeriesMerger if aggregation == nil || *aggregation == typesv1.TimeSeriesAggregationType_TIME_SERIES_AGGREGATION_TYPE_SUM { - m = NewSeriesMerger(true) + m = NewTimeSeriesMerger(true) } else { - m = NewSeriesMerger(false) + m = NewTimeSeriesMerger(false) } for _, s := range series { - m.MergeSeries(s) + m.MergeTimeSeries(s) } - return m.Series() + return m.TimeSeries() } -type SeriesMerger struct { +type TimeSeriesMerger struct { mu sync.Mutex series map[uint64]*typesv1.Series sum bool } -// NewSeriesMerger creates a new series merger. If sum is set, samples +// NewTimeSeriesMerger creates a new series merger. If sum is set, samples // with matching timestamps are summed, otherwise duplicates are retained. -func NewSeriesMerger(sum bool) *SeriesMerger { - return &SeriesMerger{ +func NewTimeSeriesMerger(sum bool) *TimeSeriesMerger { + return &TimeSeriesMerger{ series: make(map[uint64]*typesv1.Series), sum: sum, } } -func (m *SeriesMerger) MergeSeries(s []*typesv1.Series) { +func (m *TimeSeriesMerger) MergeTimeSeries(s []*typesv1.Series) { m.mu.Lock() defer m.mu.Unlock() for _, x := range s { @@ -49,7 +49,11 @@ func (m *SeriesMerger) MergeSeries(s []*typesv1.Series) { } } -func (m *SeriesMerger) Series() []*typesv1.Series { +func (m *TimeSeriesMerger) IsEmpty() bool { + return len(m.series) == 0 +} + +func (m *TimeSeriesMerger) TimeSeries() []*typesv1.Series { if len(m.series) == 0 { return nil } @@ -66,7 +70,7 @@ func (m *SeriesMerger) Series() []*typesv1.Series { return r } -func (m *SeriesMerger) mergePoints(points []*typesv1.Point) int { +func (m *TimeSeriesMerger) mergePoints(points []*typesv1.Point) int { l := len(points) if l < 2 { return l diff --git a/pkg/model/series_test.go b/pkg/model/time_series_merger_test.go similarity index 100% rename from pkg/model/series_test.go rename to pkg/model/time_series_merger_test.go diff --git a/pkg/model/time_series_test.go b/pkg/model/time_series_test.go new file mode 100644 index 0000000000..6088b45166 --- /dev/null +++ b/pkg/model/time_series_test.go @@ -0,0 +1,154 @@ +package model + +import ( + "testing" + + typesv1 "github.com/grafana/pyroscope/api/gen/proto/go/types/v1" + "github.com/grafana/pyroscope/pkg/iter" + "github.com/grafana/pyroscope/pkg/testhelper" +) + +func Test_RangeSeriesSum(t *testing.T) { + seriesA := NewLabelsBuilder(nil).Set("foo", "bar").Labels() + seriesB := NewLabelsBuilder(nil).Set("foo", "buzz").Labels() + for _, tc := range []struct { + name string + in []TimeSeriesValue + out []*typesv1.Series + }{ + { + name: "single series", + in: []TimeSeriesValue{ + {Ts: 1, Value: 1}, + {Ts: 1, Value: 1}, + {Ts: 2, Value: 2}, + {Ts: 3, Value: 3}, + {Ts: 4, Value: 4}, + {Ts: 5, Value: 5}, + }, + out: []*typesv1.Series{ + { + Points: []*typesv1.Point{ + {Timestamp: 1, Value: 2}, + {Timestamp: 2, Value: 2}, + {Timestamp: 3, Value: 3}, + {Timestamp: 4, Value: 4}, + {Timestamp: 5, Value: 5}, + }, + }, + }, + }, + { + name: "multiple series", + in: []TimeSeriesValue{ + {Ts: 1, Value: 1, Lbs: seriesA, LabelsHash: seriesA.Hash()}, + {Ts: 1, Value: 1, Lbs: seriesB, LabelsHash: seriesB.Hash()}, + {Ts: 2, Value: 1, Lbs: seriesA, LabelsHash: seriesA.Hash()}, + {Ts: 3, Value: 1, Lbs: seriesB, LabelsHash: seriesB.Hash()}, + {Ts: 3, Value: 1, Lbs: seriesB, LabelsHash: seriesB.Hash()}, + {Ts: 4, Value: 4, Lbs: seriesB, LabelsHash: seriesB.Hash()}, + {Ts: 4, Value: 4, Lbs: seriesB, LabelsHash: seriesB.Hash()}, + {Ts: 4, Value: 4, Lbs: seriesA, LabelsHash: seriesA.Hash()}, + {Ts: 5, Value: 5, Lbs: seriesA, LabelsHash: seriesA.Hash()}, + }, + out: []*typesv1.Series{ + { + Labels: seriesA, + Points: []*typesv1.Point{ + {Timestamp: 1, Value: 1}, + {Timestamp: 2, Value: 1}, + {Timestamp: 4, Value: 4}, + {Timestamp: 5, Value: 5}, + }, + }, + { + Labels: seriesB, + Points: []*typesv1.Point{ + {Timestamp: 1, Value: 1}, + {Timestamp: 3, Value: 2}, + {Timestamp: 4, Value: 8}, + }, + }, + }, + }, + } { + t.Run(tc.name, func(t *testing.T) { + in := iter.NewSliceIterator(tc.in) + out := RangeSeries(in, 1, 5, 1, nil) + testhelper.EqualProto(t, tc.out, out) + }) + } +} + +func Test_RangeSeriesAvg(t *testing.T) { + seriesA := NewLabelsBuilder(nil).Set("foo", "bar").Labels() + seriesB := NewLabelsBuilder(nil).Set("foo", "buzz").Labels() + for _, tc := range []struct { + name string + in []TimeSeriesValue + out []*typesv1.Series + }{ + { + name: "single series", + in: []TimeSeriesValue{ + {Ts: 1, Value: 1}, + {Ts: 1, Value: 2}, + {Ts: 2, Value: 2}, + {Ts: 2, Value: 3}, + {Ts: 3, Value: 4}, + {Ts: 4, Value: 5}, + }, + out: []*typesv1.Series{ + { + Points: []*typesv1.Point{ + {Timestamp: 1, Value: 1.5}, // avg of 1 and 2 + {Timestamp: 2, Value: 2.5}, // avg of 2 and 3 + {Timestamp: 3, Value: 4}, + {Timestamp: 4, Value: 5}, + }, + }, + }, + }, + { + name: "multiple series", + in: []TimeSeriesValue{ + {Ts: 1, Value: 1, Lbs: seriesA, LabelsHash: seriesA.Hash()}, + {Ts: 1, Value: 1, Lbs: seriesB, LabelsHash: seriesB.Hash()}, + {Ts: 2, Value: 1, Lbs: seriesA, LabelsHash: seriesA.Hash()}, + {Ts: 2, Value: 2, Lbs: seriesA, LabelsHash: seriesA.Hash()}, + {Ts: 3, Value: 1, Lbs: seriesB, LabelsHash: seriesB.Hash()}, + {Ts: 3, Value: 2, Lbs: seriesB, LabelsHash: seriesB.Hash()}, + {Ts: 4, Value: 4, Lbs: seriesB, LabelsHash: seriesB.Hash()}, + {Ts: 4, Value: 6, Lbs: seriesB, LabelsHash: seriesB.Hash()}, + {Ts: 4, Value: 4, Lbs: seriesA, LabelsHash: seriesA.Hash()}, + {Ts: 5, Value: 5, Lbs: seriesA, LabelsHash: seriesA.Hash()}, + }, + out: []*typesv1.Series{ + { + Labels: seriesA, + Points: []*typesv1.Point{ + {Timestamp: 1, Value: 1}, + {Timestamp: 2, Value: 1.5}, // avg of 1 and 2 + {Timestamp: 4, Value: 4}, + {Timestamp: 5, Value: 5}, + }, + }, + { + Labels: seriesB, + Points: []*typesv1.Point{ + {Timestamp: 1, Value: 1}, + {Timestamp: 3, Value: 1.5}, // avg of 1 and 2 + {Timestamp: 4, Value: 5}, // avg of 4 and 6 + }, + }, + }, + }, + } { + t.Run(tc.name, func(t *testing.T) { + in := iter.NewSliceIterator(tc.in) + aggregation := typesv1.TimeSeriesAggregationType_TIME_SERIES_AGGREGATION_TYPE_AVERAGE + out := RangeSeries(in, 1, 5, 1, &aggregation) + testhelper.EqualProto(t, tc.out, out) + }) + } +} diff --git a/pkg/model/tree.go b/pkg/model/tree.go index 613fd47a27..46c5201d8f 100644 --- a/pkg/model/tree.go +++ b/pkg/model/tree.go @@ -6,7 +6,6 @@ import ( "io" "sort" "strings" - "sync" dvarint "github.com/dennwc/varint" "github.com/xlab/treeprint" @@ -375,6 +374,17 @@ var errMalformedTreeBytes = fmt.Errorf("malformed tree bytes") const estimateBytesPerNode = 16 // Chosen empirically. +func MustUnmarshalTree(b []byte) *Tree { + if len(b) == 0 { + return new(Tree) + } + t, err := UnmarshalTree(b) + if err != nil { + panic(err) + } + return t +} + func UnmarshalTree(b []byte) (*Tree, error) { t := new(Tree) if len(b) < 2 { @@ -432,41 +442,3 @@ func UnmarshalTree(b []byte) (*Tree, error) { return t, nil } - -type TreeMerger struct { - mu sync.Mutex - t *Tree -} - -func NewTreeMerger() *TreeMerger { - return new(TreeMerger) -} - -func (m *TreeMerger) MergeTree(t *Tree) { - m.mu.Lock() - defer m.mu.Unlock() - if m.t != nil { - m.t.Merge(t) - } else { - m.t = t - } -} - -func (m *TreeMerger) MergeTreeBytes(b []byte) error { - // TODO(kolesnikovae): Ideally, we should not have - // the intermediate tree t but update m.t reading - // raw bytes b directly. - t, err := UnmarshalTree(b) - if err != nil { - return err - } - m.MergeTree(t) - return nil -} - -func (m *TreeMerger) Tree() *Tree { - if m.t == nil { - return new(Tree) - } - return m.t -} diff --git a/pkg/model/tree_merger.go b/pkg/model/tree_merger.go new file mode 100644 index 0000000000..fa75d891b5 --- /dev/null +++ b/pkg/model/tree_merger.go @@ -0,0 +1,47 @@ +package model + +import ( + "sync" +) + +type TreeMerger struct { + mu sync.Mutex + t *Tree +} + +func NewTreeMerger() *TreeMerger { + return new(TreeMerger) +} + +func (m *TreeMerger) MergeTree(t *Tree) { + m.mu.Lock() + defer m.mu.Unlock() + if m.t != nil { + m.t.Merge(t) + } else { + m.t = t + } +} + +func (m *TreeMerger) MergeTreeBytes(b []byte) error { + // TODO(kolesnikovae): Ideally, we should not have + // the intermediate tree t but update m.t reading + // raw bytes b directly. + t, err := UnmarshalTree(b) + if err != nil { + return err + } + m.MergeTree(t) + return nil +} + +func (m *TreeMerger) Tree() *Tree { + if m.t == nil { + return new(Tree) + } + return m.t +} + +func (m *TreeMerger) IsEmpty() bool { + return m.t == nil +} diff --git a/pkg/objstore/providers/memory/bucket_client.go b/pkg/objstore/providers/memory/bucket_client.go new file mode 100644 index 0000000000..5b20aefa70 --- /dev/null +++ b/pkg/objstore/providers/memory/bucket_client.go @@ -0,0 +1,230 @@ +// Copyright (c) The Thanos Authors. +// Licensed under the Apache License 2.0. + +package memory + +import ( + "bytes" + "context" + "errors" + "io" + "sort" + "strings" + "sync" + "time" + + "github.com/thanos-io/objstore" +) + +var errNotFound = errors.New("inmem: object not found") + +// InMemBucket implements the objstore.Bucket interfaces against local memory. +// Methods from Bucket interface are thread-safe. Objects are assumed to be immutable. +type InMemBucket struct { + mtx sync.RWMutex + objects map[string][]byte + attrs map[string]objstore.ObjectAttributes +} + +// NewInMemBucket returns a new in memory Bucket. +// NOTE: Returned bucket is just a naive in memory bucket implementation. For test use cases only. +func NewInMemBucket() *InMemBucket { + return &InMemBucket{ + objects: map[string][]byte{}, + attrs: map[string]objstore.ObjectAttributes{}, + } +} + +// Objects returns a copy of the internally stored objects. +// NOTE: For assert purposes. +func (b *InMemBucket) Objects() map[string][]byte { + b.mtx.RLock() + defer b.mtx.RUnlock() + + objs := make(map[string][]byte) + for k, v := range b.objects { + objs[k] = v + } + + return objs +} + +// Iter calls f for each entry in the given directory. The argument to f is the full +// object name including the prefix of the inspected directory. +func (b *InMemBucket) Iter(_ context.Context, dir string, f func(string) error, options ...objstore.IterOption) error { + unique := map[string]struct{}{} + params := objstore.ApplyIterOptions(options...) + + var dirPartsCount int + dirParts := strings.SplitAfter(dir, objstore.DirDelim) + for _, p := range dirParts { + if p == "" { + continue + } + dirPartsCount++ + } + + b.mtx.RLock() + for filename := range b.objects { + if !strings.HasPrefix(filename, dir) || dir == filename { + continue + } + + if params.Recursive { + // Any object matching the prefix should be included. + unique[filename] = struct{}{} + continue + } + + parts := strings.SplitAfter(filename, objstore.DirDelim) + unique[strings.Join(parts[:dirPartsCount+1], "")] = struct{}{} + } + b.mtx.RUnlock() + + var keys []string + for n := range unique { + keys = append(keys, n) + } + sort.Slice(keys, func(i, j int) bool { + if strings.HasSuffix(keys[i], objstore.DirDelim) && strings.HasSuffix(keys[j], objstore.DirDelim) { + return strings.Compare(keys[i], keys[j]) < 0 + } + if strings.HasSuffix(keys[i], objstore.DirDelim) { + return false + } + if strings.HasSuffix(keys[j], objstore.DirDelim) { + return true + } + + return strings.Compare(keys[i], keys[j]) < 0 + }) + + for _, k := range keys { + if err := f(k); err != nil { + return err + } + } + return nil +} + +// Get returns a reader for the given object name. +func (b *InMemBucket) Get(_ context.Context, name string) (io.ReadCloser, error) { + if name == "" { + return nil, errors.New("inmem: object name is empty") + } + + b.mtx.RLock() + file, ok := b.objects[name] + b.mtx.RUnlock() + if !ok { + return nil, errNotFound + } + + return io.NopCloser(bytes.NewReader(file)), nil +} + +// GetRange returns a new range reader for the given object name and range. +func (b *InMemBucket) GetRange(_ context.Context, name string, off, length int64) (io.ReadCloser, error) { + if name == "" { + return nil, errors.New("inmem: object name is empty") + } + + b.mtx.RLock() + file, ok := b.objects[name] + b.mtx.RUnlock() + if !ok { + return nil, errNotFound + } + + if int64(len(file)) < off { + return io.NopCloser(bytes.NewReader(nil)), nil + } + + if length == 0 { + return io.NopCloser(bytes.NewReader(nil)), nil + } + if length == -1 { + return io.NopCloser(bytes.NewReader(file[off:])), nil + } + + if int64(len(file)) <= off+length { + // Just return maximum of what we have. + length = int64(len(file)) - off + } + + return io.NopCloser(bytes.NewReader(file[off : off+length])), nil +} + +// Exists checks if the given directory exists in memory. +func (b *InMemBucket) Exists(_ context.Context, name string) (bool, error) { + b.mtx.RLock() + defer b.mtx.RUnlock() + _, ok := b.objects[name] + return ok, nil +} + +// Attributes returns information about the specified object. +func (b *InMemBucket) Attributes(_ context.Context, name string) (objstore.ObjectAttributes, error) { + b.mtx.RLock() + attrs, ok := b.attrs[name] + b.mtx.RUnlock() + if !ok { + return objstore.ObjectAttributes{}, errNotFound + } + return attrs, nil +} + +// Upload writes the file specified in src to into the memory. +func (b *InMemBucket) Upload(_ context.Context, name string, r io.Reader) error { + b.mtx.Lock() + defer b.mtx.Unlock() + body, err := io.ReadAll(r) + if err != nil { + return err + } + b.objects[name] = body + b.attrs[name] = objstore.ObjectAttributes{ + Size: int64(len(body)), + LastModified: time.Now(), + } + return nil +} + +// Delete removes all data prefixed with the dir. +func (b *InMemBucket) Delete(_ context.Context, name string) error { + b.mtx.Lock() + defer b.mtx.Unlock() + if _, ok := b.objects[name]; !ok { + return errNotFound + } + delete(b.objects, name) + delete(b.attrs, name) + return nil +} + +// IsObjNotFoundErr returns true if error means that object is not found. Relevant to Get operations. +func (b *InMemBucket) IsObjNotFoundErr(err error) bool { + return errors.Is(err, errNotFound) +} + +// IsAccessDeniedErr returns true if access to object is denied. +func (b *InMemBucket) IsAccessDeniedErr(err error) bool { + return false +} + +func (b *InMemBucket) Close() error { return nil } + +// Name returns the bucket name. +func (b *InMemBucket) Name() string { + return "inmem" +} + +func (b *InMemBucket) Set(name string, data []byte) { + b.mtx.Lock() + defer b.mtx.Unlock() + b.objects[name] = data + b.attrs[name] = objstore.ObjectAttributes{ + Size: int64(len(data)), + LastModified: time.Now(), + } +} diff --git a/pkg/objstore/read_only_file.go b/pkg/objstore/read_only_file.go new file mode 100644 index 0000000000..350010262a --- /dev/null +++ b/pkg/objstore/read_only_file.go @@ -0,0 +1,179 @@ +package objstore + +import ( + "context" + "fmt" + "io" + "os" + "path/filepath" + "sync" + "time" + + "github.com/grafana/dskit/multierror" + "github.com/thanos-io/objstore" + + "github.com/grafana/pyroscope/pkg/util/bufferpool" +) + +type ReadOnlyFile struct { + size int64 + name string + path string + mu sync.Mutex + readers []*fileReader +} + +func Download(ctx context.Context, name string, src BucketReader, dir string) (*ReadOnlyFile, error) { + f, err := download(ctx, name, src, dir) + if err != nil { + return nil, fmt.Errorf("downloading %s: %w", name, err) + } + return f, nil +} + +func download(ctx context.Context, name string, src BucketReader, dir string) (f *ReadOnlyFile, err error) { + r, err := src.Get(ctx, name) + if err != nil { + return nil, err + } + f = &ReadOnlyFile{ + size: 0, + name: name, + path: filepath.Join(dir, filepath.Base(name)), + } + defer func() { + if err != nil { + _ = f.Close() + _ = r.Close() + } + }() + if err = os.MkdirAll(dir, 0755); err != nil { + return nil, err + } + dst, err := os.Create(f.path) + if err != nil { + return nil, err + } + buf := bufferpool.GetBuffer(32 << 10) + defer bufferpool.Put(buf) + buf.B = buf.B[:cap(buf.B)] + n, err := io.CopyBuffer(dst, r, buf.B) + if err != nil { + return nil, err + } + f.size = n + return f, nil +} + +func (f *ReadOnlyFile) Close() error { + var m multierror.MultiError + for _, r := range f.readers { + m.Add(r.Close()) + } + m.Add(os.RemoveAll(f.path)) + f.readers = f.readers[:0] + return m.Err() +} + +func (f *ReadOnlyFile) Iter(context.Context, string, func(string) error, ...objstore.IterOption) error { + return nil +} + +func (f *ReadOnlyFile) Exists(_ context.Context, name string) (bool, error) { + return name == f.name, nil +} + +func (f *ReadOnlyFile) IsObjNotFoundErr(err error) bool { return os.IsNotExist(err) } + +func (f *ReadOnlyFile) IsAccessDeniedErr(err error) bool { return os.IsPermission(err) } + +func (f *ReadOnlyFile) Attributes(_ context.Context, name string) (attrs objstore.ObjectAttributes, err error) { + if name != f.name { + return attrs, os.ErrNotExist + } + return objstore.ObjectAttributes{ + Size: f.size, + LastModified: time.Unix(0, 0), // We don't care. + }, nil +} + +func (f *ReadOnlyFile) ReaderAt(_ context.Context, name string) (ReaderAtCloser, error) { + return f.borrowOrCreateReader(name) +} + +func (f *ReadOnlyFile) Get(_ context.Context, name string) (io.ReadCloser, error) { + r, err := f.borrowOrCreateReader(name) + if err != nil { + return nil, err + } + if _, err = r.Seek(0, io.SeekStart); err != nil { + _ = r.Close() + return nil, err + } + return r, nil +} + +func (f *ReadOnlyFile) GetRange(_ context.Context, name string, off, length int64) (io.ReadCloser, error) { + if off < 0 || length < 0 { + return nil, fmt.Errorf("%w: invalid offset", os.ErrInvalid) + } + r, err := f.borrowOrCreateReader(name) + if err != nil { + return nil, err + } + if _, err = r.Seek(off, io.SeekStart); err != nil { + _ = r.Close() + return nil, err + } + r.reader = io.LimitReader(r.reader, length) + return r, nil +} + +func (f *ReadOnlyFile) borrowOrCreateReader(name string) (*fileReader, error) { + if name != f.name { + return nil, os.ErrNotExist + } + f.mu.Lock() + defer f.mu.Unlock() + if len(f.readers) > 0 { + ff := f.readers[len(f.readers)-1] + f.readers = f.readers[:len(f.readers)-1] + ff.reader = ff.File + return ff, nil + } + return f.openReader() +} + +func (f *ReadOnlyFile) returnReader(r *fileReader) { + f.mu.Lock() + defer f.mu.Unlock() + f.readers = append(f.readers, r) +} + +func (f *ReadOnlyFile) openReader() (*fileReader, error) { + ff, err := os.Open(f.path) + if err != nil { + return nil, err + } + return &fileReader{ + parent: f, + File: ff, + reader: ff, + }, nil +} + +type fileReader struct { + parent *ReadOnlyFile + reader io.Reader + *os.File +} + +func (r *fileReader) Close() error { + r.reader = nil + r.parent.returnReader(r) + return nil +} + +func (r *fileReader) Read(p []byte) (int, error) { + return r.reader.Read(p) +} diff --git a/pkg/objstore/reader.go b/pkg/objstore/reader.go index 30cb96a6e1..5fa13d79c2 100644 --- a/pkg/objstore/reader.go +++ b/pkg/objstore/reader.go @@ -2,6 +2,7 @@ package objstore import ( "context" + "fmt" "io" "github.com/thanos-io/objstore" @@ -91,3 +92,47 @@ func (b *ReaderAt) ReadAt(p []byte, off int64) (int, error) { func (b *ReaderAt) Close() error { return nil } + +func ReadRange(ctx context.Context, reader io.ReaderFrom, name string, storage objstore.BucketReader, off, size int64) error { + if size == 0 { + attrs, err := storage.Attributes(ctx, name) + if err != nil { + return err + } + size = attrs.Size + } + if size == 0 { + return nil + } + rc, err := storage.GetRange(ctx, name, off, size) + if err != nil { + return err + } + defer func() { + _ = rc.Close() + }() + n, err := reader.ReadFrom(io.LimitReader(rc, size)) + if err != nil { + return err + } + if n != size { + return fmt.Errorf("read %d bytes, expected %d", n, size) + } + return nil +} + +type BucketReaderWithOffset struct { + BucketReader + offset int64 +} + +func NewBucketReaderWithOffset(r BucketReader, offset int64) *BucketReaderWithOffset { + return &BucketReaderWithOffset{ + BucketReader: r, + offset: offset, + } +} + +func (r *BucketReaderWithOffset) GetRange(ctx context.Context, name string, off, length int64) (io.ReadCloser, error) { + return r.BucketReader.GetRange(ctx, name, r.offset+off, length) +} diff --git a/pkg/og/flameql/flameql.go b/pkg/og/flameql/flameql.go index 94eb15c1af..bca663025e 100644 --- a/pkg/og/flameql/flameql.go +++ b/pkg/og/flameql/flameql.go @@ -97,7 +97,7 @@ func ValidateAppName(n string) error { } func IsTagKeyRuneAllowed(r rune) bool { - return (r >= 'a' && r <= 'z') || (r >= 'A' && r <= 'Z') || (r >= '0' && r <= '9') || r == '_' + return (r >= 'a' && r <= 'z') || (r >= 'A' && r <= 'Z') || (r >= '0' && r <= '9') || r == '_' || r == '.' } func IsAppNameRuneAllowed(r rune) bool { diff --git a/pkg/og/flameql/flameql_test.go b/pkg/og/flameql/flameql_test.go index 15219eebae..f6baea64e1 100644 --- a/pkg/og/flameql/flameql_test.go +++ b/pkg/og/flameql/flameql_test.go @@ -43,6 +43,7 @@ var _ = Describe("ValidateTagKey", func() { testCases := []testCase{ {"foo_BAR_12_baz_qux", nil}, + {"service.namespace", nil}, {ReservedTagKeyName, ErrTagKeyReserved}, {"", ErrTagKeyIsRequired}, diff --git a/pkg/phlare/modules.go b/pkg/phlare/modules.go index 38aa02fce7..dea217d25d 100644 --- a/pkg/phlare/modules.go +++ b/pkg/phlare/modules.go @@ -80,9 +80,15 @@ const ( TenantSettings string = "tenant-settings" AdHocProfiles string = "ad-hoc-profiles" - // QueryFrontendTripperware string = "query-frontend-tripperware" - // IndexGateway string = "index-gateway" - // IndexGatewayRing string = "index-gateway-ring" + // Experimental modules + + Metastore string = "metastore" + MetastoreClient string = "metastore-client" + SegmentWriter string = "segment-writer" + QueryBackend string = "query-backend" + QueryBackendClient string = "query-backend-client" + CompactionWorker string = "compaction-worker" + HealthService string = "health-service" ) var objectStoreTypeStats = usagestats.NewString("store_object_type") @@ -333,6 +339,7 @@ func (f *Phlare) initMemberlistKV() (services.Service, error) { f.Cfg.Distributor.DistributorRing.KVStore.MemberlistKV = f.MemberlistKV.GetMemberlistKV f.Cfg.Ingester.LifecyclerConfig.RingConfig.KVStore.MemberlistKV = f.MemberlistKV.GetMemberlistKV + f.Cfg.SegmentWriter.LifecyclerConfig.RingConfig.KVStore.MemberlistKV = f.MemberlistKV.GetMemberlistKV f.Cfg.QueryScheduler.ServiceDiscovery.SchedulerRing.KVStore.MemberlistKV = f.MemberlistKV.GetMemberlistKV f.Cfg.OverridesExporter.Ring.Ring.KVStore.MemberlistKV = f.MemberlistKV.GetMemberlistKV f.Cfg.StoreGateway.ShardingRing.Ring.KVStore.MemberlistKV = f.MemberlistKV.GetMemberlistKV diff --git a/pkg/phlare/modules_experimental.go b/pkg/phlare/modules_experimental.go new file mode 100644 index 0000000000..d54500e5d5 --- /dev/null +++ b/pkg/phlare/modules_experimental.go @@ -0,0 +1,90 @@ +package phlare + +import ( + "fmt" + + "github.com/go-kit/log" + "github.com/grafana/dskit/services" + "google.golang.org/grpc/health/grpc_health_v1" + + compactionworker "github.com/grafana/pyroscope/pkg/experiment/compactor" + segmentwriter "github.com/grafana/pyroscope/pkg/experiment/ingester" + "github.com/grafana/pyroscope/pkg/experiment/metastore" + metastoreclient "github.com/grafana/pyroscope/pkg/experiment/metastore/client" + "github.com/grafana/pyroscope/pkg/experiment/querybackend" + querybackendclient "github.com/grafana/pyroscope/pkg/experiment/querybackend/client" + "github.com/grafana/pyroscope/pkg/util/health" +) + +func (f *Phlare) initSegmentWriter() (services.Service, error) { + ingester, err := segmentwriter.New( + f.context(), + f.Cfg.SegmentWriter, + f.Cfg.PhlareDB, + f.storageBucket, f.metastoreClient) + if err != nil { + return nil, fmt.Errorf("failed to create segment writer: %w", err) + } + f.segmentWriter = ingester + f.API.RegisterSegmentWriter(ingester) + return f.segmentWriter, nil +} + +func (f *Phlare) initCompactionWorker() (svc services.Service, err error) { + logger := log.With(f.logger, "component", "compaction-worker") + f.compactionWorker, err = compactionworker.New(f.Cfg.CompactionWorker, logger, f.metastoreClient, f.storageBucket, f.reg) + if err != nil { + return nil, err + } + return f.compactionWorker, nil +} + +func (f *Phlare) initMetastore() (services.Service, error) { + logger := log.With(f.logger, "component", "metastore") + m, err := metastore.New(f.Cfg.Metastore, f.TenantLimits, logger, f.reg, f.healthService, f.metastoreClient) + if err != nil { + return nil, err + } + f.API.RegisterMetastore(m) + f.metastore = m + return m.Service(), nil +} + +func (f *Phlare) initMetastoreClient() (services.Service, error) { + mc, err := metastoreclient.New(f.Cfg.Metastore.Address, f.Cfg.Metastore.GRPCClientConfig, f.logger) + if err != nil { + return nil, err + } + f.metastoreClient = mc + return mc.Service(), nil +} + +func (f *Phlare) initQueryBackend() (services.Service, error) { + br := querybackend.NewBlockReader(f.logger, f.storageBucket) + logger := log.With(f.logger, "component", "query-backend") + b, err := querybackend.New(f.Cfg.QueryBackend, logger, f.reg, f.queryBackendClient, br) + if err != nil { + return nil, err + } + f.API.RegisterQueryBackend(b) + return b.Service(), nil +} + +func (f *Phlare) initQueryBackendClient() (services.Service, error) { + c, err := querybackendclient.New( + f.Cfg.QueryBackend.Address, + f.Cfg.QueryBackend.GRPCClientConfig, + ) + if err != nil { + return nil, err + } + f.queryBackendClient = c + return c.Service(), nil +} + +func (f *Phlare) initHealthService() (services.Service, error) { + healthService := health.NewGRPCHealthService() + grpc_health_v1.RegisterHealthServer(f.Server.GRPC, healthService) + f.healthService = healthService + return healthService, nil +} diff --git a/pkg/phlare/phlare.go b/pkg/phlare/phlare.go index df60bbd64b..e92ac5ebb2 100644 --- a/pkg/phlare/phlare.go +++ b/pkg/phlare/phlare.go @@ -42,6 +42,12 @@ import ( "github.com/grafana/pyroscope/pkg/cfg" "github.com/grafana/pyroscope/pkg/compactor" "github.com/grafana/pyroscope/pkg/distributor" + compactionworker "github.com/grafana/pyroscope/pkg/experiment/compactor" + segmentwriter "github.com/grafana/pyroscope/pkg/experiment/ingester" + "github.com/grafana/pyroscope/pkg/experiment/metastore" + metastoreclient "github.com/grafana/pyroscope/pkg/experiment/metastore/client" + "github.com/grafana/pyroscope/pkg/experiment/querybackend" + querybackendclient "github.com/grafana/pyroscope/pkg/experiment/querybackend/client" "github.com/grafana/pyroscope/pkg/frontend" "github.com/grafana/pyroscope/pkg/ingester" phlareobj "github.com/grafana/pyroscope/pkg/objstore" @@ -59,6 +65,7 @@ import ( "github.com/grafana/pyroscope/pkg/usagestats" "github.com/grafana/pyroscope/pkg/util" "github.com/grafana/pyroscope/pkg/util/cli" + "github.com/grafana/pyroscope/pkg/util/health" "github.com/grafana/pyroscope/pkg/validation" "github.com/grafana/pyroscope/pkg/validation/exporter" ) @@ -91,6 +98,16 @@ type Config struct { ConfigFile string `yaml:"-"` ConfigExpandEnv bool `yaml:"-"` + + // Experimental modules. + // TODO(kolesnikovae): + // - Generalized experimental features? + // - Better naming. + v2Experiment bool + SegmentWriter segmentwriter.Config `yaml:"segment_writer" doc:"hidden"` + Metastore metastore.Config `yaml:"metastore" doc:"hidden"` + QueryBackend querybackend.Config `yaml:"query_backend" doc:"hidden"` + CompactionWorker compactionworker.Config `yaml:"compaction_worker" doc:"hidden"` } func newDefaultConfig() *Config { @@ -149,6 +166,14 @@ func (c *Config) RegisterFlagsWithContext(ctx context.Context, f *flag.FlagSet) c.LimitsConfig.RegisterFlags(f) c.Compactor.RegisterFlags(f, log.NewLogfmtLogger(os.Stderr)) c.API.RegisterFlags(f) + + c.v2Experiment = os.Getenv("PYROSCOPE_V2_EXPERIMENT") != "" + if c.v2Experiment { + c.Metastore.RegisterFlags(f) + c.SegmentWriter.RegisterFlags(f) + c.QueryBackend.RegisterFlags(f) + c.CompactionWorker.RegisterFlags(f) + } } // registerServerFlagsWithChangedDefaultValues registers *Config.Server flags, but overrides some defaults set by the weaveworks package. @@ -191,6 +216,7 @@ func (c *Config) Validate() error { func (c *Config) ApplyDynamicConfig() cfg.Source { c.Ingester.LifecyclerConfig.RingConfig.KVStore.Store = "memberlist" + c.SegmentWriter.LifecyclerConfig.RingConfig.KVStore.Store = "memberlist" c.Distributor.DistributorRing.KVStore.Store = c.Ingester.LifecyclerConfig.RingConfig.KVStore.Store c.OverridesExporter.Ring.Ring.KVStore.Store = c.Ingester.LifecyclerConfig.RingConfig.KVStore.Store c.Frontend.QuerySchedulerDiscovery.SchedulerRing.KVStore.Store = c.Ingester.LifecyclerConfig.RingConfig.KVStore.Store @@ -242,6 +268,17 @@ type Phlare struct { auth connect.Option ingester *ingester.Ingester frontend *frontend.Frontend + + // Experimental modules. + //nolint:unused + segmentWriter *segmentwriter.SegmentWriterService + metastore *metastore.Metastore + metastoreClient *metastoreclient.Client + //nolint:unused + queryBackend *querybackend.QueryBackend + queryBackendClient *querybackendclient.Client + compactionWorker *compactionworker.Worker + healthService health.Service } func New(cfg Config) (*Phlare, error) { @@ -250,9 +287,10 @@ func New(cfg Config) (*Phlare, error) { usagestats.Edition("oss") phlare := &Phlare{ - Cfg: cfg, - logger: logger, - reg: prometheus.DefaultRegisterer, + Cfg: cfg, + logger: logger, + reg: prometheus.DefaultRegisterer, + healthService: health.NoOpService, } if err := cfg.Validate(); err != nil { return nil, err @@ -311,7 +349,7 @@ func (f *Phlare) setupModuleManager() error { // Add dependencies deps := map[string][]string{ - All: {Ingester, Distributor, QueryScheduler, QueryFrontend, Querier, StoreGateway, Admin, TenantSettings, Compactor, AdHocProfiles}, + All: {Ingester, Distributor, QueryFrontend, QueryScheduler, Querier, StoreGateway, Compactor, Admin, TenantSettings, AdHocProfiles}, Server: {GRPCGateway}, API: {Server}, @@ -334,6 +372,32 @@ func (f *Phlare) setupModuleManager() error { AdHocProfiles: {API, Overrides, Storage}, } + // Experimental modules. + if f.Cfg.v2Experiment { + experimentalModules := map[string][]string{ + SegmentWriter: {Overrides, API, MemberlistKV, Storage, UsageReport, MetastoreClient}, + Metastore: {Overrides, API, HealthService, MetastoreClient}, + CompactionWorker: {Overrides, API, Storage, Overrides, MetastoreClient}, + QueryBackend: {Overrides, API, Storage, Overrides, QueryBackendClient}, + HealthService: {Overrides, API}, + } + for k, v := range experimentalModules { + deps[k] = v + } + + // TODO(kolesnikovae): Inject new distributor dependencies, if any. + deps[All] = append(deps[All], SegmentWriter, Metastore, CompactionWorker, QueryBackend, HealthService) + deps[QueryFrontend] = append(deps[QueryFrontend], MetastoreClient, QueryBackendClient) + + mm.RegisterModule(SegmentWriter, f.initSegmentWriter) + mm.RegisterModule(Metastore, f.initMetastore) + mm.RegisterModule(CompactionWorker, f.initCompactionWorker) + mm.RegisterModule(QueryBackend, f.initQueryBackend) + mm.RegisterModule(MetastoreClient, f.initMetastoreClient, modules.UserInvisibleModule) + mm.RegisterModule(QueryBackendClient, f.initQueryBackendClient, modules.UserInvisibleModule) + mm.RegisterModule(HealthService, f.initHealthService, modules.UserInvisibleModule) + } + for mod, targets := range deps { if err := mm.AddDependency(mod, targets...); err != nil { return err @@ -510,6 +574,12 @@ func (f *Phlare) readyHandler(sm *services.Manager) http.HandlerFunc { return } } + if f.segmentWriter != nil { + if err := f.segmentWriter.CheckReady(r.Context()); err != nil { + http.Error(w, "Segment Writer not ready: "+err.Error(), http.StatusServiceUnavailable) + return + } + } if f.frontend != nil { if err := f.frontend.CheckReady(r.Context()); err != nil { diff --git a/pkg/phlaredb/block_querier.go b/pkg/phlaredb/block_querier.go index e5d45b5e35..1628dcc9ad 100644 --- a/pkg/phlaredb/block_querier.go +++ b/pkg/phlaredb/block_querier.go @@ -633,7 +633,7 @@ func (queriers Queriers) SelectMatchingProfiles(ctx context.Context, params *ing if err != nil { return nil, err } - return iter.NewMergeIterator(maxBlockProfile, true, iters...), nil + return phlaremodel.NewMergeIterator(maxBlockProfile, true, iters...), nil } func (queriers Queriers) LabelValues(ctx context.Context, req *connect.Request[typesv1.LabelValuesRequest]) (*connect.Response[typesv1.LabelValuesResponse], error) { @@ -1611,7 +1611,7 @@ func (b *singleBlockQuerier) SelectMatchingProfiles(ctx context.Context, params iters = append(iters, iter.NewSliceIterator(currentSeriesSlice)) } - return iter.NewMergeIterator(maxBlockProfile, false, iters...), nil + return phlaremodel.NewMergeIterator(maxBlockProfile, false, iters...), nil } func (b *singleBlockQuerier) SelectMergeByLabels( diff --git a/pkg/phlaredb/filter_profiles_bidi.go b/pkg/phlaredb/filter_profiles_bidi.go index 7cdf6dc131..64e185dc32 100644 --- a/pkg/phlaredb/filter_profiles_bidi.go +++ b/pkg/phlaredb/filter_profiles_bidi.go @@ -12,6 +12,7 @@ import ( ingestv1 "github.com/grafana/pyroscope/api/gen/proto/go/ingester/v1" "github.com/grafana/pyroscope/pkg/iter" + phlaremodel "github.com/grafana/pyroscope/pkg/model" ) type BidiServerMerge[Res any, Req any] interface { @@ -76,7 +77,7 @@ func filterProfiles[B BidiServerMerge[Res, Req], Res filterResponse, Req filterR querierIndex: i, } } - if err := iter.ReadBatch(ctx, iter.NewMergeIterator(ProfileWithIndex{ + if err := iter.ReadBatch(ctx, phlaremodel.NewMergeIterator(ProfileWithIndex{ Profile: maxBlockProfile, Index: 0, }, true, its...), batchProfileSize, func(ctx context.Context, batch []ProfileWithIndex) error { diff --git a/pkg/phlaredb/head.go b/pkg/phlaredb/head.go index e6efcbf7a0..24239e7489 100644 --- a/pkg/phlaredb/head.go +++ b/pkg/phlaredb/head.go @@ -327,6 +327,15 @@ func (h *Head) LabelNames(ctx context.Context, req *connect.Request[typesv1.Labe }), nil } +func (h *Head) MustProfileTypeNames() []string { + ptypes, err := h.profiles.index.ix.LabelValues(phlaremodel.LabelNameProfileType, nil) + if err != nil { + panic(err) + } + sort.Strings(ptypes) + return ptypes +} + // ProfileTypes returns the possible profile types. func (h *Head) ProfileTypes(ctx context.Context, req *connect.Request[ingestv1.ProfileTypesRequest]) (*connect.Response[ingestv1.ProfileTypesResponse], error) { values, err := h.profiles.index.ix.LabelValues(phlaremodel.LabelNameProfileType, nil) @@ -657,3 +666,11 @@ func (h *Head) GetMetaStats() block.MetaStats { defer h.metaLock.RUnlock() return h.meta.GetStats() } + +func (h *Head) Meta() *block.Meta { + return h.meta +} + +func (h *Head) LocalPathFor(relPath string) string { + return filepath.Join(h.localPath, relPath) +} diff --git a/pkg/phlaredb/head_queriers.go b/pkg/phlaredb/head_queriers.go index ac5233e45a..a39c9d6624 100644 --- a/pkg/phlaredb/head_queriers.go +++ b/pkg/phlaredb/head_queriers.go @@ -395,12 +395,12 @@ func (q *headInMemoryQuerier) SelectMatchingProfiles(ctx context.Context, params NewSeriesIterator( profileSeries.lbs, profileSeries.fp, - iter.NewTimeRangedIterator(iter.NewSliceIterator(profiles), start, end), + phlaremodel.NewTimeRangedIterator(iter.NewSliceIterator(profiles), start, end), ), ) } - return iter.NewMergeIterator(maxBlockProfile, false, iters...), nil + return phlaremodel.NewMergeIterator(maxBlockProfile, false, iters...), nil } func (q *headInMemoryQuerier) SelectMergeByStacktraces(ctx context.Context, params *ingestv1.SelectProfilesRequest, maxNodes int64) (*phlaremodel.Tree, error) { @@ -598,16 +598,14 @@ func (q *headInMemoryQuerier) MergeByLabels( sp, _ := opentracing.StartSpanFromContext(ctx, "MergeByLabels - HeadInMemory") defer sp.Finish() - seriesBuilder := seriesBuilder{} - seriesBuilder.init(by...) - + seriesBuilder := phlaremodel.NewTimeSeriesBuilder(by...) if len(sts.GetCallSite()) == 0 { for rows.Next() { p, ok := rows.At().(ProfileWithLabels) if !ok { return nil, errors.New("expected ProfileWithLabels") } - seriesBuilder.add(p.Fingerprint(), p.Labels(), int64(p.Timestamp()), float64(p.Total())) + seriesBuilder.Add(p.Fingerprint(), p.Labels(), int64(p.Timestamp()), float64(p.Total())) } } else { r := symdb.NewResolver(ctx, q.head.symdb, @@ -622,7 +620,7 @@ func (q *headInMemoryQuerier) MergeByLabels( if err := r.CallSiteValues(&v, p.StacktracePartition(), p.Samples()); err != nil { return nil, err } - seriesBuilder.add(p.Fingerprint(), p.Labels(), int64(p.Timestamp()), float64(v.Total)) + seriesBuilder.Add(p.Fingerprint(), p.Labels(), int64(p.Timestamp()), float64(v.Total)) } } @@ -630,7 +628,7 @@ func (q *headInMemoryQuerier) MergeByLabels( return nil, err } - return seriesBuilder.build(), nil + return seriesBuilder.Build(), nil } func (q *headInMemoryQuerier) SelectMergeByLabels( @@ -654,8 +652,7 @@ func (q *headInMemoryQuerier) SelectMergeByLabels( start = model.Time(params.Start) end = model.Time(params.End) ) - seriesBuilder := seriesBuilder{} - seriesBuilder.init(by...) + seriesBuilder := phlaremodel.NewTimeSeriesBuilder(by...) index.mutex.RLock() defer index.mutex.RUnlock() @@ -673,7 +670,7 @@ func (q *headInMemoryQuerier) SelectMergeByLabels( if p.Timestamp() > end { break } - seriesBuilder.add(fp, profileSeries.lbs, int64(p.Timestamp()), float64(p.Total())) + seriesBuilder.Add(fp, profileSeries.lbs, int64(p.Timestamp()), float64(p.Total())) } } } else { @@ -696,11 +693,11 @@ func (q *headInMemoryQuerier) SelectMergeByLabels( if err = r.CallSiteValues(&v, p.StacktracePartition, p.Samples); err != nil { return nil, err } - seriesBuilder.add(fp, profileSeries.lbs, int64(p.Timestamp()), float64(v.Total)) + seriesBuilder.Add(fp, profileSeries.lbs, int64(p.Timestamp()), float64(v.Total)) } } } - return seriesBuilder.build(), nil + return seriesBuilder.Build(), nil } func (q *headInMemoryQuerier) Series(ctx context.Context, params *ingestv1.SeriesRequest) ([]*typesv1.Labels, error) { diff --git a/pkg/phlaredb/metrics.go b/pkg/phlaredb/metrics.go index 0f9d686c3c..6e584e9a11 100644 --- a/pkg/phlaredb/metrics.go +++ b/pkg/phlaredb/metrics.go @@ -45,109 +45,113 @@ type headMetrics struct { } func newHeadMetrics(reg prometheus.Registerer) *headMetrics { + return newHeadMetricsWithPrefix(reg, "pyroscope") +} + +func newHeadMetricsWithPrefix(reg prometheus.Registerer, prefix string) *headMetrics { m := &headMetrics{ seriesCreated: prometheus.NewCounterVec(prometheus.CounterOpts{ - Name: "pyroscope_tsdb_head_series_created_total", + Name: prefix + "_tsdb_head_series_created_total", Help: "Total number of series created in the head", }, []string{"profile_name"}), rowsWritten: prometheus.NewCounterVec( prometheus.CounterOpts{ - Name: "pyroscope_rows_written", + Name: prefix + "_rows_written", Help: "Number of rows written to a parquet table.", }, []string{"type"}), profilesCreated: prometheus.NewCounterVec(prometheus.CounterOpts{ - Name: "pyroscope_head_profiles_created_total", + Name: prefix + "_head_profiles_created_total", Help: "Total number of profiles created in the head", }, []string{"profile_name"}), sampleValuesIngested: prometheus.NewCounterVec( prometheus.CounterOpts{ - Name: "pyroscope_head_ingested_sample_values_total", + Name: prefix + "_head_ingested_sample_values_total", Help: "Number of sample values ingested into the head per profile type.", }, []string{"profile_name"}), sampleValuesReceived: prometheus.NewCounterVec( prometheus.CounterOpts{ - Name: "pyroscope_head_received_sample_values_total", + Name: prefix + "_head_received_sample_values_total", Help: "Number of sample values received into the head per profile type.", }, []string{"profile_name"}), sizeBytes: prometheus.NewGaugeVec( prometheus.GaugeOpts{ - Name: "pyroscope_head_size_bytes", + Name: prefix + "_head_size_bytes", Help: "Size of a particular in memory store within the head phlaredb block.", }, []string{"type"}), series: prometheus.NewGauge(prometheus.GaugeOpts{ - Name: "pyroscope_tsdb_head_series", + Name: prefix + "_tsdb_head_series", Help: "Total number of series in the head block.", }), profiles: prometheus.NewGauge(prometheus.GaugeOpts{ - Name: "pyroscope_head_profiles", + Name: prefix + "_head_profiles", Help: "Total number of profiles in the head block.", }), flushedFileSizeBytes: prometheus.NewHistogramVec(prometheus.HistogramOpts{ - Name: "pyroscope_head_flushed_table_size_bytes", + Name: prefix + "_head_flushed_table_size_bytes", Help: "Size of a flushed table in bytes.", // [2MB, 4MB, 8MB, 16MB, 32MB, 64MB, 128MB, 256MB, 512MB, 1GB, 2GB] Buckets: prometheus.ExponentialBuckets(2*1024*1024, 2, 11), }, []string{"name"}), flushedBlockSizeBytes: prometheus.NewHistogram(prometheus.HistogramOpts{ - Name: "pyroscope_head_flushed_block_size_bytes", + Name: prefix + "_head_flushed_block_size_bytes", Help: "Size of a flushed block in bytes.", // [50MB, 75MB, 112.5MB, 168.75MB, 253.125MB, 379.688MB, 569.532MB, 854.298MB, 1.281MB, 1.922MB, 2.883MB] Buckets: prometheus.ExponentialBuckets(50*1024*1024, 1.5, 11), }), flushedBlockDurationSeconds: prometheus.NewHistogram(prometheus.HistogramOpts{ - Name: "pyroscope_head_flushed_block_duration_seconds", + Name: prefix + "_head_flushed_block_duration_seconds", Help: "Time to flushed a block in seconds.", // [5s, 7.5s, 11.25s, 16.875s, 25.3125s, 37.96875s, 56.953125s, 85.4296875s, 128.14453125s, 192.216796875s] Buckets: prometheus.ExponentialBuckets(5, 1.5, 10), }), flushedBlockSeries: prometheus.NewHistogram(prometheus.HistogramOpts{ - Name: "pyroscope_head_flushed_block_series", + Name: prefix + "_head_flushed_block_series", Help: "Number of series in a flushed block.", // [1k, 3k, 5k, 7k, 9k, 11k, 13k, 15k, 17k, 19k] Buckets: prometheus.LinearBuckets(1000, 2000, 10), }), flushedBlockSamples: prometheus.NewHistogram(prometheus.HistogramOpts{ - Name: "pyroscope_head_flushed_block_samples", + Name: prefix + "_head_flushed_block_samples", Help: "Number of samples in a flushed block.", // [200k, 400k, 800k, 1.6M, 3.2M, 6.4M, 12.8M, 25.6M, 51.2M, 102.4M, 204.8M, 409.6M, 819.2M, 1.6384G, 3.2768G] Buckets: prometheus.ExponentialBuckets(200_000, 2, 15), }), flusehdBlockProfiles: prometheus.NewHistogram(prometheus.HistogramOpts{ - Name: "pyroscope_head_flushed_block_profiles", + Name: prefix + "_head_flushed_block_profiles", Help: "Number of profiles in a flushed block.", // [20k, 40k, 80k, 160k, 320k, 640k, 1.28M, 2.56M, 5.12M, 10.24M] Buckets: prometheus.ExponentialBuckets(20_000, 2, 10), }), blockDurationSeconds: prometheus.NewHistogram(prometheus.HistogramOpts{ - Name: "pyroscope_head_block_duration_seconds", + Name: prefix + "_head_block_duration_seconds", Help: "Duration of a block in seconds (the range it covers).", // [20m, 40m, 1h, 1h20, 1h40, 2h, 2h20, 2h40, 3h, 3h20, 3h40, 4h, 4h20, 4h40, 5h, 5h20, 5h40, 6h, 6h20, 6h40, 7h, 7h20, 7h40, 8h] Buckets: prometheus.LinearBuckets(1200, 1200, 24), }), flushedBlocks: prometheus.NewCounterVec(prometheus.CounterOpts{ - Name: "pyroscope_head_flushed_blocks_total", + Name: prefix + "_head_flushed_blocks_total", Help: "Total number of blocks flushed.", }, []string{"status"}), flushedBlocksReasons: prometheus.NewCounterVec(prometheus.CounterOpts{ - Name: "pyroscope_head_flushed_reason_total", + Name: prefix + "_head_flushed_reason_total", Help: "Total count of reasons why block has been flushed.", }, []string{"reason"}), writtenProfileSegments: prometheus.NewCounterVec(prometheus.CounterOpts{ - Name: "pyroscope_head_written_profile_segments_total", + Name: prefix + "_head_written_profile_segments_total", Help: "Total number and status of profile row groups segments written.", }, []string{"status"}), writtenProfileSegmentsBytes: prometheus.NewHistogram(prometheus.HistogramOpts{ - Name: "pyroscope_head_written_profile_segments_size_bytes", + Name: prefix + "_head_written_profile_segments_size_bytes", Help: "Size of a flushed table in bytes.", // [512KB, 1MB, 2MB, 4MB, 8MB, 16MB, 32MB, 64MB, 128MB, 256MB, 512MB] Buckets: prometheus.ExponentialBuckets(512*1024, 2, 11), }), samples: prometheus.NewGauge(prometheus.GaugeOpts{ - Name: "pyroscope_head_samples", + Name: prefix + "_head_samples", Help: "Number of samples in the head.", }), } @@ -181,6 +185,10 @@ func (m *headMetrics) register(reg prometheus.Registerer) { m.writtenProfileSegmentsBytes = util.RegisterOrGet(reg, m.writtenProfileSegmentsBytes) } +func ContextWithHeadMetrics(ctx context.Context, reg prometheus.Registerer, prefix string) context.Context { + return contextWithHeadMetrics(ctx, newHeadMetricsWithPrefix(reg, prefix)) +} + func contextWithHeadMetrics(ctx context.Context, m *headMetrics) context.Context { return context.WithValue(ctx, headMetricsContextKey, m) } diff --git a/pkg/phlaredb/querier.go b/pkg/phlaredb/querier.go index f8a286950a..c2bcb121fa 100644 --- a/pkg/phlaredb/querier.go +++ b/pkg/phlaredb/querier.go @@ -39,6 +39,7 @@ type IndexReader interface { // by the reference. // Returns storage.ErrNotFound if the ref does not resolve to a known series. Series(ref storage.SeriesRef, lset *phlaremodel.Labels, chks *[]index.ChunkMeta) (uint64, error) + SeriesBy(ref storage.SeriesRef, lset *phlaremodel.Labels, chks *[]index.ChunkMeta, by ...string) (uint64, error) // LabelNames returns all the unique label names present in the index in sorted order. LabelNames(matchers ...*labels.Matcher) ([]string, error) diff --git a/pkg/phlaredb/sample_merge.go b/pkg/phlaredb/sample_merge.go index 9d21e935d1..b06ceb4b11 100644 --- a/pkg/phlaredb/sample_merge.go +++ b/pkg/phlaredb/sample_merge.go @@ -2,14 +2,11 @@ package phlaredb import ( "context" - "sort" "strings" "github.com/grafana/dskit/runutil" "github.com/opentracing/opentracing-go" "github.com/parquet-go/parquet-go" - "github.com/prometheus/common/model" - "github.com/samber/lo" profilev1 "github.com/grafana/pyroscope/api/gen/proto/go/google/v1" typesv1 "github.com/grafana/pyroscope/api/gen/proto/go/types/v1" @@ -161,67 +158,6 @@ func mergeBySpans[T interface{ StacktracePartition() uint64 }](ctx context.Conte return profiles.Err() } -type seriesByLabels map[string]*typesv1.Series - -func (m seriesByLabels) normalize() []*typesv1.Series { - result := lo.Values(m) - sort.Slice(result, func(i, j int) bool { - return phlaremodel.CompareLabelPairs(result[i].Labels, result[j].Labels) < 0 - }) - // we have to sort the points in each series because labels reduction may have changed the order - for _, s := range result { - sort.Slice(s.Points, func(i, j int) bool { - return s.Points[i].Timestamp < s.Points[j].Timestamp - }) - } - return result -} - -type seriesBuilder struct { - labelsByFingerprint map[model.Fingerprint]string - labelBuf []byte - by []string - - series seriesByLabels -} - -func (s *seriesBuilder) init(by ...string) { - s.labelsByFingerprint = map[model.Fingerprint]string{} - s.series = make(seriesByLabels) - s.labelBuf = make([]byte, 0, 1024) - s.by = by -} - -func (s *seriesBuilder) add(fp model.Fingerprint, lbs phlaremodel.Labels, ts int64, value float64) { - labelsByString, ok := s.labelsByFingerprint[fp] - if !ok { - s.labelBuf = lbs.BytesWithLabels(s.labelBuf, s.by...) - labelsByString = string(s.labelBuf) - s.labelsByFingerprint[fp] = labelsByString - if _, ok := s.series[labelsByString]; !ok { - s.series[labelsByString] = &typesv1.Series{ - Labels: lbs.WithLabels(s.by...), - Points: []*typesv1.Point{ - { - Timestamp: ts, - Value: value, - }, - }, - } - return - } - } - series := s.series[labelsByString] - series.Points = append(series.Points, &typesv1.Point{ - Timestamp: ts, - Value: value, - }) -} - -func (s *seriesBuilder) build() []*typesv1.Series { - return s.series.normalize() -} - func mergeByLabels[T Profile]( ctx context.Context, profileSource Source, @@ -236,8 +172,7 @@ func mergeByLabels[T Profile]( profiles := query.NewRepeatedRowIterator(ctx, rows, profileSource.RowGroups(), column.ColumnIndex) defer runutil.CloseWithErrCapture(&err, profiles, "failed to close profile stream") - seriesBuilder := seriesBuilder{} - seriesBuilder.init(by...) + seriesBuilder := phlaremodel.NewTimeSeriesBuilder(by...) for profiles.Next() { values := profiles.At() @@ -246,10 +181,10 @@ func mergeByLabels[T Profile]( for _, e := range values.Values { total += e[0].Int64() } - seriesBuilder.add(p.Fingerprint(), p.Labels(), int64(p.Timestamp()), float64(total)) + seriesBuilder.Add(p.Fingerprint(), p.Labels(), int64(p.Timestamp()), float64(total)) } - return seriesBuilder.build(), profiles.Err() + return seriesBuilder.Build(), profiles.Err() } func mergeByLabelsWithStackTraceSelector[T Profile]( @@ -268,8 +203,8 @@ func mergeByLabelsWithStackTraceSelector[T Profile]( columns.Value.ColumnIndex, ) - seriesBuilder := seriesBuilder{} - seriesBuilder.init(by...) + seriesBuilder := phlaremodel.TimeSeriesBuilder{} + seriesBuilder.Init(by...) defer runutil.CloseWithErrCapture(&err, profiles, "failed to close profile stream") var v symdb.CallSiteValues @@ -279,8 +214,8 @@ func mergeByLabelsWithStackTraceSelector[T Profile]( if err = r.CallSiteValuesParquet(&v, h.StacktracePartition(), row.Values[0], row.Values[1]); err != nil { return nil, err } - seriesBuilder.add(h.Fingerprint(), h.Labels(), int64(h.Timestamp()), float64(v.Total)) + seriesBuilder.Add(h.Fingerprint(), h.Labels(), int64(h.Timestamp()), float64(v.Total)) } - return seriesBuilder.build(), profiles.Err() + return seriesBuilder.Build(), profiles.Err() } diff --git a/pkg/phlaredb/symdb/block_reader.go b/pkg/phlaredb/symdb/block_reader.go index 8561b71c07..861a063f82 100644 --- a/pkg/phlaredb/symdb/block_reader.go +++ b/pkg/phlaredb/symdb/block_reader.go @@ -20,6 +20,7 @@ import ( "github.com/grafana/pyroscope/pkg/objstore" "github.com/grafana/pyroscope/pkg/phlaredb/block" schemav1 "github.com/grafana/pyroscope/pkg/phlaredb/schemas/v1" + "github.com/grafana/pyroscope/pkg/util/bufferpool" "github.com/grafana/pyroscope/pkg/util/refctr" ) @@ -36,6 +37,83 @@ type Reader struct { meta *block.Meta files map[string]block.File parquetFiles *parquetFiles + + prefetchSize uint64 +} + +type Option func(*Reader) + +func WithPrefetchSize(size uint64) Option { + return func(r *Reader) { + r.prefetchSize = size + } +} + +func OpenObject(ctx context.Context, b objstore.BucketReader, name string, offset, size int64, options ...Option) (*Reader, error) { + f := block.File{ + RelPath: name, + SizeBytes: uint64(size), + } + r := &Reader{ + bucket: objstore.NewBucketReaderWithOffset(b, offset), + file: f, + } + for _, opt := range options { + opt(r) + } + + var err error + if r.prefetchSize > 0 { + err = r.openIndexWithPrefetch(ctx) + } else { + err = r.openIndex(ctx) + } + if err != nil { + return nil, fmt.Errorf("opening index section: %w", err) + } + + if err = r.buildPartitions(); err != nil { + return nil, err + } + + return r, nil +} + +func (r *Reader) openIndexWithPrefetch(ctx context.Context) (err error) { + prefetchSize := r.prefetchSize + if prefetchSize > r.file.SizeBytes { + prefetchSize = r.file.SizeBytes + } + n, err := r.prefetchIndex(ctx, prefetchSize) + if err == nil && n != 0 { + _, err = r.prefetchIndex(ctx, prefetchSize) + } + return err +} + +func (r *Reader) prefetchIndex(ctx context.Context, size uint64) (n uint64, err error) { + if size < uint64(FooterSize) { + size = uint64(FooterSize) + } + prefetchOffset := r.file.SizeBytes - size + buf := bufferpool.GetBuffer(int(size)) + defer bufferpool.Put(buf) + if err = objstore.ReadRange(ctx, buf, r.file.RelPath, r.bucket, int64(prefetchOffset), int64(size)); err != nil { + return 0, fmt.Errorf("fetching index: %w", err) + } + footerOffset := size - uint64(FooterSize) + if err = r.footer.UnmarshalBinary(buf.B[footerOffset:]); err != nil { + return 0, fmt.Errorf("unmarshaling footer: %w", err) + } + if prefetchOffset > (r.footer.IndexOffset) { + return r.file.SizeBytes - r.footer.IndexOffset, nil + } + // prefetch offset is less that or equal to the index offset. + indexOffset := r.footer.IndexOffset - prefetchOffset + if r.index, err = OpenIndex(buf.B[indexOffset:footerOffset]); err != nil { + return 0, fmt.Errorf("opening index: %w", err) + } + return 0, nil } func Open(ctx context.Context, b objstore.BucketReader, m *block.Meta) (*Reader, error) { @@ -113,13 +191,6 @@ func (r *Reader) partitionReader(h *PartitionHeader) (*partition, error) { // openIndex locates footer and loads the index section from // the file into the memory. -// -// NOTE(kolesnikovae): Pre-fetch: we could speculatively fetch -// the footer and the index section into a larger buffer rather -// than retrieving them synchronously. -// -// NOTE(kolesnikovae): It is possible to skip the footer, if it -// was cached, and the index section offset and size are known. func (r *Reader) openIndex(ctx context.Context) error { if r.file.SizeBytes == 0 { attrs, err := r.bucket.Attributes(ctx, r.file.RelPath) diff --git a/pkg/phlaredb/tsdb/index/index.go b/pkg/phlaredb/tsdb/index/index.go index 31081d97ed..d2eb1205eb 100644 --- a/pkg/phlaredb/tsdb/index/index.go +++ b/pkg/phlaredb/tsdb/index/index.go @@ -208,6 +208,10 @@ func NewTOCFromByteSlice(bs ByteSlice) (*TOC, error) { // NewWriter returns a new Writer to the given filename. It serializes data in format version 2. func NewWriter(ctx context.Context, fn string) (*Writer, error) { + return NewWriterSize(ctx, fn, 4<<20) +} + +func NewWriterSize(ctx context.Context, fn string, bufferSize int) (*Writer, error) { dir := filepath.Dir(fn) df, err := fileutil.OpenDir(dir) @@ -221,17 +225,17 @@ func NewWriter(ctx context.Context, fn string) (*Writer, error) { } // Main index file we are building. - f, err := NewFileWriter(fn) + f, err := NewFileWriter(fn, bufferSize) if err != nil { return nil, err } // Temporary file for postings. - fP, err := NewFileWriter(fn + "_tmp_p") + fP, err := NewFileWriter(fn+"_tmp_p", bufferSize) if err != nil { return nil, err } // Temporary file for posting offset table. - fPO, err := NewFileWriter(fn + "_tmp_po") + fPO, err := NewFileWriter(fn+"_tmp_po", bufferSize) if err != nil { return nil, err } @@ -247,8 +251,8 @@ func NewWriter(ctx context.Context, fn string) (*Writer, error) { stage: idxStageNone, // Reusable memory. - buf1: encoding.EncWrap(tsdb_enc.Encbuf{B: make([]byte, 0, 1<<22)}), - buf2: encoding.EncWrap(tsdb_enc.Encbuf{B: make([]byte, 0, 1<<22)}), + buf1: encoding.EncWrap(tsdb_enc.Encbuf{B: make([]byte, 0, bufferSize)}), + buf2: encoding.EncWrap(tsdb_enc.Encbuf{B: make([]byte, 0, bufferSize)}), symbolCache: make(map[string]symbolCacheEntry, 1<<8), labelNames: make(map[string]uint64, 1<<8), @@ -279,14 +283,14 @@ type FileWriter struct { name string } -func NewFileWriter(name string) (*FileWriter, error) { +func NewFileWriter(name string, bufferSize int) (*FileWriter, error) { f, err := os.OpenFile(name, os.O_CREATE|os.O_RDWR, 0o666) if err != nil { return nil, err } return &FileWriter{ f: f, - fbuf: bufio.NewWriterSize(f, 1<<22), + fbuf: bufio.NewWriterSize(f, bufferSize), pos: 0, name: name, }, nil @@ -1076,7 +1080,8 @@ func (w *Writer) writePostings() error { return err } // Don't need to calculate a checksum, so can copy directly. - n, err := io.CopyBuffer(w.f.fbuf, w.fP.f, make([]byte, 1<<20)) + buf := w.buf1.B[:cap(w.buf1.B)] + n, err := io.CopyBuffer(w.f.fbuf, w.fP.f, buf) if err != nil { return err } diff --git a/pkg/phlaredb/tsdb/index/index_test.go b/pkg/phlaredb/tsdb/index/index_test.go index 8f355fc6bc..2044b9443a 100644 --- a/pkg/phlaredb/tsdb/index/index_test.go +++ b/pkg/phlaredb/tsdb/index/index_test.go @@ -384,7 +384,7 @@ func TestPersistence_index_e2e(t *testing.T) { } } - var input indexWriterSeriesSlice + var input IndexWriterSeriesSlice // Generate ChunkMetas for every label set. for i, lset := range flbls { @@ -397,9 +397,9 @@ func TestPersistence_index_e2e(t *testing.T) { Checksum: rand.Uint32(), }) } - input = append(input, &indexWriterSeries{ - labels: lset, - chunks: metas, + input = append(input, &IndexWriterSeries{ + Labels: lset, + Chunks: metas, }) } @@ -424,11 +424,11 @@ func TestPersistence_index_e2e(t *testing.T) { mi := newMockIndex() for i, s := range input { - err = iw.AddSeries(storage.SeriesRef(i), s.labels, model.Fingerprint(s.labels.Hash()), s.chunks...) + err = iw.AddSeries(storage.SeriesRef(i), s.Labels, model.Fingerprint(s.Labels.Hash()), s.Chunks...) require.NoError(t, err) - require.NoError(t, mi.AddSeries(storage.SeriesRef(i), s.labels, s.chunks...)) + require.NoError(t, mi.AddSeries(storage.SeriesRef(i), s.Labels, s.Chunks...)) - for _, l := range s.labels { + for _, l := range s.Labels { valset, ok := values[l.Name] if !ok { valset = map[string]struct{}{} @@ -436,7 +436,7 @@ func TestPersistence_index_e2e(t *testing.T) { } valset[l.Value] = struct{}{} } - postings.Add(storage.SeriesRef(i), s.labels) + postings.Add(storage.SeriesRef(i), s.Labels) } err = iw.Close() diff --git a/pkg/phlaredb/tsdb/index/postings.go b/pkg/phlaredb/tsdb/index/postings.go index a63bfb7b7d..9af139deba 100644 --- a/pkg/phlaredb/tsdb/index/postings.go +++ b/pkg/phlaredb/tsdb/index/postings.go @@ -730,6 +730,10 @@ type bigEndianPostings struct { cur uint32 } +func NewBigEndianPostings(list []byte) Postings { + return newBigEndianPostings(list) +} + func newBigEndianPostings(list []byte) *bigEndianPostings { return &bigEndianPostings{list: list} } diff --git a/pkg/phlaredb/tsdb/index/test_utils.go b/pkg/phlaredb/tsdb/index/test_utils.go index 9c66ad34d2..d7b78d2437 100644 --- a/pkg/phlaredb/tsdb/index/test_utils.go +++ b/pkg/phlaredb/tsdb/index/test_utils.go @@ -4,16 +4,16 @@ import ( phlaremodel "github.com/grafana/pyroscope/pkg/model" ) -type indexWriterSeries struct { - labels phlaremodel.Labels - chunks []ChunkMeta // series file offset of chunks +type IndexWriterSeries struct { + Labels phlaremodel.Labels + Chunks []ChunkMeta // series file offset of chunks } -type indexWriterSeriesSlice []*indexWriterSeries +type IndexWriterSeriesSlice []*IndexWriterSeries -func (s indexWriterSeriesSlice) Len() int { return len(s) } -func (s indexWriterSeriesSlice) Swap(i, j int) { s[i], s[j] = s[j], s[i] } +func (s IndexWriterSeriesSlice) Len() int { return len(s) } +func (s IndexWriterSeriesSlice) Swap(i, j int) { s[i], s[j] = s[j], s[i] } -func (s indexWriterSeriesSlice) Less(i, j int) bool { - return phlaremodel.CompareLabelPairs(s[i].labels, s[j].labels) < 0 +func (s IndexWriterSeriesSlice) Less(i, j int) bool { + return phlaremodel.CompareLabelPairs(s[i].Labels, s[j].Labels) < 0 } diff --git a/pkg/querier/querier.go b/pkg/querier/querier.go index 78977b9b2f..6f9bb2ddf6 100644 --- a/pkg/querier/querier.go +++ b/pkg/querier/querier.go @@ -34,7 +34,6 @@ import ( "github.com/grafana/pyroscope/api/gen/proto/go/vcs/v1/vcsv1connect" connectapi "github.com/grafana/pyroscope/pkg/api/connect" "github.com/grafana/pyroscope/pkg/clientpool" - "github.com/grafana/pyroscope/pkg/iter" phlaremodel "github.com/grafana/pyroscope/pkg/model" phlareobj "github.com/grafana/pyroscope/pkg/objstore" "github.com/grafana/pyroscope/pkg/phlaredb/bucketindex" @@ -975,7 +974,7 @@ func (q *Querier) SelectSeries(ctx context.Context, req *connect.Request[querier return nil, connect.NewError(connect.CodeInternal, err) } - result := rangeSeries(it, req.Msg.Start, req.Msg.End, stepMs, req.Msg.Aggregation) + result := phlaremodel.RangeSeries(it, req.Msg.Start, req.Msg.End, stepMs, req.Msg.Aggregation) if it.Err() != nil { return nil, connect.NewError(connect.CodeInternal, it.Err()) } @@ -1040,77 +1039,6 @@ func (q *Querier) selectSeries(ctx context.Context, req *connect.Request[querier return responses, nil } -// rangeSeries aggregates profiles into series. -// Series contains points spaced by step from start to end. -// Profiles from the same step are aggregated into one point. -func rangeSeries(it iter.Iterator[ProfileValue], start, end, step int64, aggregation *typesv1.TimeSeriesAggregationType) []*typesv1.Series { - defer it.Close() - seriesMap := make(map[uint64]*typesv1.Series) - aggregators := make(map[uint64]TimeSeriesAggregator) - - if !it.Next() { - return nil - } - - // advance from the start to the end, adding each step results to the map. -Outer: - for currentStep := start; currentStep <= end; currentStep += step { - for { - aggregator, ok := aggregators[it.At().LabelsHash] - if !ok { - aggregator = NewTimeSeriesAggregator(aggregation) - aggregators[it.At().LabelsHash] = aggregator - } - if it.At().Ts > currentStep { - if !aggregator.IsEmpty() { - series := seriesMap[it.At().LabelsHash] - series.Points = append(series.Points, aggregator.GetAndReset()) - } - break // no more profiles for the currentStep - } - // find or create series - series, ok := seriesMap[it.At().LabelsHash] - if !ok { - seriesMap[it.At().LabelsHash] = &typesv1.Series{ - Labels: it.At().Lbs, - Points: []*typesv1.Point{}, - } - aggregator.Add(currentStep, it.At().Value) - if !it.Next() { - break Outer - } - continue - } - // Aggregate point if it is in the current step. - if aggregator.GetTimestamp() == currentStep { - aggregator.Add(currentStep, it.At().Value) - if !it.Next() { - break Outer - } - continue - } - // Next step is missing - if !aggregator.IsEmpty() { - series.Points = append(series.Points, aggregator.GetAndReset()) - } - aggregator.Add(currentStep, it.At().Value) - if !it.Next() { - break Outer - } - } - } - for lblHash, aggregator := range aggregators { - if !aggregator.IsEmpty() { - seriesMap[lblHash].Points = append(seriesMap[lblHash].Points, aggregator.GetAndReset()) - } - } - series := lo.Values(seriesMap) - sort.Slice(series, func(i, j int) bool { - return phlaremodel.CompareLabelPairs(series[i].Labels, series[j].Labels) < 0 - }) - return series -} - func uniqueSortedStrings(responses []ResponseFromReplica[[]string]) []string { total := 0 for _, r := range responses { @@ -1186,87 +1114,3 @@ func (q *Querier) selectSpanProfile(ctx context.Context, req *querierv1.SelectMe storegatewayTree.Merge(ingesterTree) return storegatewayTree, nil } - -type TimeSeriesAggregator interface { - Add(ts int64, value float64) - GetAndReset() *typesv1.Point - IsEmpty() bool - GetTimestamp() int64 -} - -func NewTimeSeriesAggregator(aggregation *typesv1.TimeSeriesAggregationType) TimeSeriesAggregator { - if aggregation == nil { - return &sumTimeSeriesAggregator{ - ts: -1, - } - } - if *aggregation == typesv1.TimeSeriesAggregationType_TIME_SERIES_AGGREGATION_TYPE_AVERAGE { - return &avgTimeSeriesAggregator{ - ts: -1, - } - } - return &sumTimeSeriesAggregator{ - ts: -1, - } -} - -type sumTimeSeriesAggregator struct { - ts int64 - sum float64 -} - -func (a *sumTimeSeriesAggregator) Add(ts int64, value float64) { - a.ts = ts - a.sum += value -} - -func (a *sumTimeSeriesAggregator) GetAndReset() *typesv1.Point { - tsCopy := a.ts - sumCopy := a.sum - a.ts = -1 - a.sum = 0 - return &typesv1.Point{ - Timestamp: tsCopy, - Value: sumCopy, - } -} - -func (a *sumTimeSeriesAggregator) IsEmpty() bool { - return a.ts == -1 -} - -func (a *sumTimeSeriesAggregator) GetTimestamp() int64 { - return a.ts -} - -type avgTimeSeriesAggregator struct { - ts int64 - sum float64 - count int64 -} - -func (a *avgTimeSeriesAggregator) Add(ts int64, value float64) { - a.ts = ts - a.sum += value - a.count++ -} - -func (a *avgTimeSeriesAggregator) GetAndReset() *typesv1.Point { - avg := a.sum / float64(a.count) - tsCopy := a.ts - a.ts = -1 - a.sum = 0 - a.count = 0 - return &typesv1.Point{ - Timestamp: tsCopy, - Value: avg, - } -} - -func (a *avgTimeSeriesAggregator) IsEmpty() bool { - return a.ts == -1 -} - -func (a *avgTimeSeriesAggregator) GetTimestamp() int64 { - return a.ts -} diff --git a/pkg/querier/querier_test.go b/pkg/querier/querier_test.go index 0183acbc8f..0e5e593e7f 100644 --- a/pkg/querier/querier_test.go +++ b/pkg/querier/querier_test.go @@ -29,7 +29,6 @@ import ( querierv1 "github.com/grafana/pyroscope/api/gen/proto/go/querier/v1" typesv1 "github.com/grafana/pyroscope/api/gen/proto/go/types/v1" "github.com/grafana/pyroscope/pkg/clientpool" - "github.com/grafana/pyroscope/pkg/iter" phlaremodel "github.com/grafana/pyroscope/pkg/model" objstoreclient "github.com/grafana/pyroscope/pkg/objstore/client" "github.com/grafana/pyroscope/pkg/objstore/providers/filesystem" @@ -1005,147 +1004,6 @@ func (f *fakeQuerierIngester) MergeProfilesPprof(ctx context.Context) clientpool return res } -func Test_RangeSeriesSum(t *testing.T) { - for _, tc := range []struct { - name string - in []ProfileValue - out []*typesv1.Series - }{ - { - name: "single series", - in: []ProfileValue{ - {Ts: 1, Value: 1}, - {Ts: 1, Value: 1}, - {Ts: 2, Value: 2}, - {Ts: 3, Value: 3}, - {Ts: 4, Value: 4}, - {Ts: 5, Value: 5}, - }, - out: []*typesv1.Series{ - { - Points: []*typesv1.Point{ - {Timestamp: 1, Value: 2}, - {Timestamp: 2, Value: 2}, - {Timestamp: 3, Value: 3}, - {Timestamp: 4, Value: 4}, - {Timestamp: 5, Value: 5}, - }, - }, - }, - }, - { - name: "multiple series", - in: []ProfileValue{ - {Ts: 1, Value: 1, Lbs: foobarlabels, LabelsHash: foobarlabels.Hash()}, - {Ts: 1, Value: 1, Lbs: foobuzzlabels, LabelsHash: foobuzzlabels.Hash()}, - {Ts: 2, Value: 1, Lbs: foobarlabels, LabelsHash: foobarlabels.Hash()}, - {Ts: 3, Value: 1, Lbs: foobuzzlabels, LabelsHash: foobuzzlabels.Hash()}, - {Ts: 3, Value: 1, Lbs: foobuzzlabels, LabelsHash: foobuzzlabels.Hash()}, - {Ts: 4, Value: 4, Lbs: foobuzzlabels, LabelsHash: foobuzzlabels.Hash()}, - {Ts: 4, Value: 4, Lbs: foobuzzlabels, LabelsHash: foobuzzlabels.Hash()}, - {Ts: 4, Value: 4, Lbs: foobarlabels, LabelsHash: foobarlabels.Hash()}, - {Ts: 5, Value: 5, Lbs: foobarlabels, LabelsHash: foobarlabels.Hash()}, - }, - out: []*typesv1.Series{ - { - Labels: foobarlabels, - Points: []*typesv1.Point{ - {Timestamp: 1, Value: 1}, - {Timestamp: 2, Value: 1}, - {Timestamp: 4, Value: 4}, - {Timestamp: 5, Value: 5}, - }, - }, - { - Labels: foobuzzlabels, - Points: []*typesv1.Point{ - {Timestamp: 1, Value: 1}, - {Timestamp: 3, Value: 2}, - {Timestamp: 4, Value: 8}, - }, - }, - }, - }, - } { - t.Run(tc.name, func(t *testing.T) { - in := iter.NewSliceIterator(tc.in) - out := rangeSeries(in, 1, 5, 1, nil) - testhelper.EqualProto(t, tc.out, out) - }) - } -} - -func Test_RangeSeriesAvg(t *testing.T) { - for _, tc := range []struct { - name string - in []ProfileValue - out []*typesv1.Series - }{ - { - name: "single series", - in: []ProfileValue{ - {Ts: 1, Value: 1}, - {Ts: 1, Value: 2}, - {Ts: 2, Value: 2}, - {Ts: 2, Value: 3}, - {Ts: 3, Value: 4}, - {Ts: 4, Value: 5}, - }, - out: []*typesv1.Series{ - { - Points: []*typesv1.Point{ - {Timestamp: 1, Value: 1.5}, // avg of 1 and 2 - {Timestamp: 2, Value: 2.5}, // avg of 2 and 3 - {Timestamp: 3, Value: 4}, - {Timestamp: 4, Value: 5}, - }, - }, - }, - }, - { - name: "multiple series", - in: []ProfileValue{ - {Ts: 1, Value: 1, Lbs: foobarlabels, LabelsHash: foobarlabels.Hash()}, - {Ts: 1, Value: 1, Lbs: foobuzzlabels, LabelsHash: foobuzzlabels.Hash()}, - {Ts: 2, Value: 1, Lbs: foobarlabels, LabelsHash: foobarlabels.Hash()}, - {Ts: 2, Value: 2, Lbs: foobarlabels, LabelsHash: foobarlabels.Hash()}, - {Ts: 3, Value: 1, Lbs: foobuzzlabels, LabelsHash: foobuzzlabels.Hash()}, - {Ts: 3, Value: 2, Lbs: foobuzzlabels, LabelsHash: foobuzzlabels.Hash()}, - {Ts: 4, Value: 4, Lbs: foobuzzlabels, LabelsHash: foobuzzlabels.Hash()}, - {Ts: 4, Value: 6, Lbs: foobuzzlabels, LabelsHash: foobuzzlabels.Hash()}, - {Ts: 4, Value: 4, Lbs: foobarlabels, LabelsHash: foobarlabels.Hash()}, - {Ts: 5, Value: 5, Lbs: foobarlabels, LabelsHash: foobarlabels.Hash()}, - }, - out: []*typesv1.Series{ - { - Labels: foobarlabels, - Points: []*typesv1.Point{ - {Timestamp: 1, Value: 1}, - {Timestamp: 2, Value: 1.5}, // avg of 1 and 2 - {Timestamp: 4, Value: 4}, - {Timestamp: 5, Value: 5}, - }, - }, - { - Labels: foobuzzlabels, - Points: []*typesv1.Point{ - {Timestamp: 1, Value: 1}, - {Timestamp: 3, Value: 1.5}, // avg of 1 and 2 - {Timestamp: 4, Value: 5}, // avg of 4 and 6 - }, - }, - }, - }, - } { - t.Run(tc.name, func(t *testing.T) { - in := iter.NewSliceIterator(tc.in) - aggregation := typesv1.TimeSeriesAggregationType_TIME_SERIES_AGGREGATION_TYPE_AVERAGE - out := rangeSeries(in, 1, 5, 1, &aggregation) - testhelper.EqualProto(t, tc.out, out) - }) - } -} - func Test_splitQueryToStores(t *testing.T) { for _, tc := range []struct { name string diff --git a/pkg/querier/select_merge.go b/pkg/querier/select_merge.go index e4bb2a48c6..4460342638 100644 --- a/pkg/querier/select_merge.go +++ b/pkg/querier/select_merge.go @@ -10,7 +10,6 @@ import ( "github.com/grafana/dskit/multierror" "github.com/opentracing/opentracing-go" otlog "github.com/opentracing/opentracing-go/log" - "github.com/prometheus/common/model" "github.com/samber/lo" "golang.org/x/sync/errgroup" @@ -463,23 +462,8 @@ func selectMergePprofProfile(ctx context.Context, ty *typesv1.ProfileType, respo return p, nil } -type ProfileValue struct { - Ts int64 - Lbs []*typesv1.LabelPair - LabelsHash uint64 - Value float64 -} - -func (p ProfileValue) Labels() phlaremodel.Labels { - return p.Lbs -} - -func (p ProfileValue) Timestamp() model.Time { - return model.Time(p.Ts) -} - // selectMergeSeries selects the profile from each ingester by deduping them and request merges of total values. -func selectMergeSeries(ctx context.Context, aggregation *typesv1.TimeSeriesAggregationType, responses []ResponseFromReplica[clientpool.BidiClientMergeProfilesLabels]) (iter.Iterator[ProfileValue], error) { +func selectMergeSeries(ctx context.Context, aggregation *typesv1.TimeSeriesAggregationType, responses []ResponseFromReplica[clientpool.BidiClientMergeProfilesLabels]) (iter.Iterator[phlaremodel.TimeSeriesValue], error) { mergeResults := make([]MergeResult[[]*typesv1.Series], len(responses)) iters := make([]MergeIterator, len(responses)) var wg sync.WaitGroup @@ -524,12 +508,12 @@ func selectMergeSeries(ctx context.Context, aggregation *typesv1.TimeSeriesAggre } var series = phlaremodel.MergeSeries(aggregation, results...) - seriesIters := make([]iter.Iterator[ProfileValue], 0, len(series)) + seriesIters := make([]iter.Iterator[phlaremodel.TimeSeriesValue], 0, len(series)) for _, s := range series { s := s - seriesIters = append(seriesIters, newSeriesIterator(s.Labels, s.Points)) + seriesIters = append(seriesIters, phlaremodel.NewSeriesIterator(s.Labels, s.Points)) } - return iter.NewMergeIterator(ProfileValue{Ts: math.MaxInt64}, false, seriesIters...), nil + return phlaremodel.NewMergeIterator(phlaremodel.TimeSeriesValue{Ts: math.MaxInt64}, false, seriesIters...), nil } // selectMergeSpanProfile selects the profile from each ingester by deduping them and @@ -581,43 +565,3 @@ func selectMergeSpanProfile(ctx context.Context, responses []ResponseFromReplica span.LogFields(otlog.String("msg", "building tree")) return m.Tree(), nil } - -type seriesIterator struct { - point []*typesv1.Point - - curr ProfileValue -} - -func newSeriesIterator(lbs []*typesv1.LabelPair, points []*typesv1.Point) *seriesIterator { - return &seriesIterator{ - point: points, - - curr: ProfileValue{ - Lbs: lbs, - LabelsHash: phlaremodel.Labels(lbs).Hash(), - }, - } -} - -func (s *seriesIterator) Next() bool { - if len(s.point) == 0 { - return false - } - p := s.point[0] - s.point = s.point[1:] - s.curr.Ts = p.Timestamp - s.curr.Value = p.Value - return true -} - -func (s *seriesIterator) At() ProfileValue { - return s.curr -} - -func (s *seriesIterator) Err() error { - return nil -} - -func (s *seriesIterator) Close() error { - return nil -} diff --git a/pkg/querier/select_merge_test.go b/pkg/querier/select_merge_test.go index 6431a4ceac..12ab1fd6a2 100644 --- a/pkg/querier/select_merge_test.go +++ b/pkg/querier/select_merge_test.go @@ -17,10 +17,7 @@ import ( "github.com/grafana/pyroscope/pkg/testhelper" ) -var ( - foobarlabels = phlaremodel.Labels([]*typesv1.LabelPair{{Name: "foo", Value: "bar"}}) - foobuzzlabels = phlaremodel.Labels([]*typesv1.LabelPair{{Name: "foo", Value: "buzz"}}) -) +var foobarlabels = phlaremodel.Labels([]*typesv1.LabelPair{{Name: "foo", Value: "bar"}}) func TestSelectMergeStacktraces(t *testing.T) { resp1 := newFakeBidiClientStacktraces([]*ingestv1.ProfileSets{ @@ -202,7 +199,7 @@ func TestSelectMergeByLabels(t *testing.T) { }) values, err := iter.Slice(res) require.NoError(t, err) - require.Equal(t, []ProfileValue{ + require.Equal(t, []phlaremodel.TimeSeriesValue{ {Ts: 1, Value: 1.0, Lbs: foobarlabels, LabelsHash: foobarlabels.Hash()}, {Ts: 2, Value: 2.0, Lbs: foobarlabels, LabelsHash: foobarlabels.Hash()}, {Ts: 3, Value: 3.0, Lbs: foobarlabels, LabelsHash: foobarlabels.Hash()}, diff --git a/pkg/util/bufferpool/pool.go b/pkg/util/bufferpool/pool.go new file mode 100644 index 0000000000..b4fa477460 --- /dev/null +++ b/pkg/util/bufferpool/pool.go @@ -0,0 +1,98 @@ +package bufferpool + +import ( + "bytes" + "io" + "sync" +) + +// Sized *bytes.Buffer pools: from 2^9 (512b) to 2^30 (1GB). +var pools [maxPool]sync.Pool + +type Buffer struct { + B []byte + p int64 +} + +const ( + minBits = 9 + maxPool = 22 +) + +// GetBuffer returns a buffer from the pool, or creates a new one. +// The returned buffer has at least the requested capacity. +func GetBuffer(size int) *Buffer { + i := poolIndex(size) + if i < 0 { + return &Buffer{B: make([]byte, 0, size)} + } + x := pools[i].Get() + if x != nil { + return x.(*Buffer) + } + c := 2 << (minBits + i - 1) + c += bytes.MinRead + return &Buffer{ + B: make([]byte, 0, c), + p: i, + } +} + +// Put places the buffer into the pool. +func Put(b *Buffer) { + if b == nil { + return + } + if p := returnPool(cap(b.B), b.p); p > 0 { + b.B = b.B[:0] + pools[p].Put(b) + } +} + +func returnPool(c int, p int64) int64 { + // Empty buffers are ignored. + if c == 0 { + return -1 + } + i := poolIndex(c) + if p == 0 { + // The buffer does not belong to any pool, or it's + // of the smallest size. We pick the pool based on + // its current capacity. + return i + } + d := i - p + if d < 0 { + // This buffer was likely obtained outside the pool. + // For example, an empty one, or with pre-allocated + // byte slice. + return i + } + if d > 1 { + // Relocate the buffer, if it's capacity has been + // grown by more than a power of two. + return i + } + // Otherwise, keep the buffer in the current pool. + return p +} + +func poolIndex(n int) (i int64) { + n-- + n >>= minBits + for n > 0 { + n >>= 1 + i++ + } + if i >= maxPool { + return -1 + } + return i +} + +func (b *Buffer) ReadFrom(r io.Reader) (int64, error) { + buf := bytes.NewBuffer(b.B) + n, err := buf.ReadFrom(r) + b.B = buf.Bytes() + return n, err +} diff --git a/pkg/util/bufferpool/pool_test.go b/pkg/util/bufferpool/pool_test.go new file mode 100644 index 0000000000..5097915208 --- /dev/null +++ b/pkg/util/bufferpool/pool_test.go @@ -0,0 +1,22 @@ +package bufferpool + +import ( + "testing" + + "github.com/stretchr/testify/assert" +) + +func Test_returnPool(t *testing.T) { + assert.EqualValues(t, 0, returnPool(512, 0)) // Buffers can be added to the pool. + assert.EqualValues(t, 1, returnPool(513, 0)) + assert.EqualValues(t, 1, returnPool(1<<10, 0)) + assert.EqualValues(t, -1, returnPool(0, 0)) // Empty buffers are ignored. + assert.EqualValues(t, -1, returnPool(0, 10)) // + assert.EqualValues(t, 5, returnPool(1<<14, 0)) // New buffers are added to the appropriate pool. + assert.EqualValues(t, 5, returnPool(1<<14, 3)) // Buffer of a capacity exceeding the next power of two are relocated. + assert.EqualValues(t, 4, returnPool(1<<14, 4)) // Buffer of a capacity not exceeding the next power of two are retained. + assert.EqualValues(t, 5, returnPool(1<<14, 5)) // Buffer of the nominal capacity. + assert.EqualValues(t, 5, returnPool(1<<14, 6)) // Buffer of a smaller capacity must be relocated. + assert.EqualValues(t, 21, returnPool(1<<30, 13)) + assert.EqualValues(t, -1, returnPool(1<<30+1, 13)) // No pools for buffers larger than 4MB. +} diff --git a/pkg/util/health/health.go b/pkg/util/health/health.go new file mode 100644 index 0000000000..675596e1bf --- /dev/null +++ b/pkg/util/health/health.go @@ -0,0 +1,33 @@ +package health + +import ( + "github.com/grafana/dskit/services" + "google.golang.org/grpc/health" + "google.golang.org/grpc/health/grpc_health_v1" +) + +type Service interface { + SetServingStatus(string, grpc_health_v1.HealthCheckResponse_ServingStatus) +} + +type noopService struct{} + +var NoOpService = noopService{} + +func (noopService) SetServingStatus(string, grpc_health_v1.HealthCheckResponse_ServingStatus) {} + +func NewGRPCHealthService() *GRPCHealthService { + s := health.NewServer() + return &GRPCHealthService{ + Server: s, + Service: services.NewIdleService(nil, func(error) error { + s.Shutdown() + return nil + }), + } +} + +type GRPCHealthService struct { + services.Service + *health.Server +} diff --git a/pkg/util/http.go b/pkg/util/http.go index 43920d4798..3149f5df42 100644 --- a/pkg/util/http.go +++ b/pkg/util/http.go @@ -13,12 +13,13 @@ import ( "strings" "time" + "github.com/grafana/dskit/instrument" + "github.com/dustin/go-humanize" "github.com/felixge/httpsnoop" "github.com/go-kit/log" "github.com/go-kit/log/level" "github.com/gorilla/mux" - "github.com/grafana/dskit/instrument" dslog "github.com/grafana/dskit/log" "github.com/grafana/dskit/middleware" "github.com/grafana/dskit/multierror" diff --git a/pkg/util/interceptor.go b/pkg/util/interceptor.go index c4437a1d93..e65f386e3c 100644 --- a/pkg/util/interceptor.go +++ b/pkg/util/interceptor.go @@ -68,7 +68,6 @@ func NewLogInterceptor(logger log.Logger) connect.UnaryInterceptorFunc { "route", req.Spec().Procedure, "tenant", tenantID, "traceID", traceID, - "parameters", req.Any(), "duration", time.Since(begin), ) }() diff --git a/pkg/util/recovery.go b/pkg/util/recovery.go index bcd7905e3b..7884663829 100644 --- a/pkg/util/recovery.go +++ b/pkg/util/recovery.go @@ -29,7 +29,7 @@ var ( return http.HandlerFunc(func(w http.ResponseWriter, req *http.Request) { defer func() { if p := recover(); p != nil { - httputil.Error(w, httpgrpc.Errorf(http.StatusInternalServerError, "error while processing request: %v", panicError(p))) + httputil.Error(w, httpgrpc.Errorf(http.StatusInternalServerError, "error while processing request: %v", PanicError(p))) } }() next.ServeHTTP(w, req) @@ -39,7 +39,7 @@ var ( RecoveryInterceptor recoveryInterceptor ) -func panicError(p interface{}) error { +func PanicError(p interface{}) error { stack := make([]byte, maxStacksize) stack = stack[:runtime.Stack(stack, true)] // keep a multiline stack @@ -53,7 +53,7 @@ func RecoverPanic(f func() error) func() error { return func() (err error) { defer func() { if p := recover(); p != nil { - err = panicError(p) + err = PanicError(p) } }() return f() @@ -66,7 +66,7 @@ func (recoveryInterceptor) WrapUnary(next connect.UnaryFunc) connect.UnaryFunc { return func(ctx context.Context, req connect.AnyRequest) (resp connect.AnyResponse, err error) { defer func() { if p := recover(); p != nil { - err = connect.NewError(connect.CodeInternal, panicError(p)) + err = connect.NewError(connect.CodeInternal, PanicError(p)) } }() return next(ctx, req) @@ -77,7 +77,7 @@ func (recoveryInterceptor) WrapStreamingHandler(next connect.StreamingHandlerFun return func(ctx context.Context, conn connect.StreamingHandlerConn) (err error) { defer func() { if p := recover(); p != nil { - err = connect.NewError(connect.CodeInternal, panicError(p)) + err = connect.NewError(connect.CodeInternal, PanicError(p)) } }() return next(ctx, conn) diff --git a/pkg/util/refctr/refctr.go b/pkg/util/refctr/refctr.go index 7bfcba35ee..df2c541232 100644 --- a/pkg/util/refctr/refctr.go +++ b/pkg/util/refctr/refctr.go @@ -3,8 +3,9 @@ package refctr import "sync" type Counter struct { - m sync.Mutex - c int + m sync.Mutex + c int + err error } // Inc increments the counter and calls the init function, @@ -30,15 +31,43 @@ func (r *Counter) Inc(init func() error) (err error) { return init() } +// IncErr is identical to Inc, with the only difference that if the +// function fails, the error is returned on any further IncErr call, +// preventing from calling the faulty initialization function again. +func (r *Counter) IncErr(init func() error) (err error) { + r.m.Lock() + if r.err != nil { + err = r.err + r.m.Unlock() + return err + } + defer func() { + // If initialization fails, we need to make sure + // the next call makes another attempt. + if err != nil { + r.err = err + r.c-- + } + r.m.Unlock() + }() + if r.c++; r.c > 1 { + return nil + } + // Mutex is acquired during the call in order to serialize + // access to the resources, so that the consequent callers + // only have access to them after initialization finishes. + return init() +} + // Dec decrements the counter and calls the release function, // if this is the last reference. func (r *Counter) Dec(release func()) { r.m.Lock() + defer r.m.Unlock() if r.c < 0 { panic("bug: negative reference counter") } if r.c--; r.c < 1 { release() } - r.m.Unlock() } diff --git a/tools/dev/experiment/local_install.sh b/tools/dev/experiment/local_install.sh new file mode 100755 index 0000000000..8a6bfe8f51 --- /dev/null +++ b/tools/dev/experiment/local_install.sh @@ -0,0 +1,38 @@ +#!/usr/bin/env bash + +set -x +set -e + +if [ -z "$DOCKER_REPO" ]; then + echo "Specify the docker repository using the DOCKER_REPO variable." + exit 1 +fi + +IMAGE_NAME=$DOCKER_REPO/pyroscope +PYROSCOPE_TEST_NAMESPACE=pyroscope-test +HELM_CHART=./operations/pyroscope/helm/pyroscope +SCRIPT_DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")" && pwd)" +VALUES_FILE="$SCRIPT_DIR/values-micro-services-experiment.yaml" + +# build and push image + +GOOS=linux GOARCH=amd64 IMAGE_PREFIX="$DOCKER_REPO/" make docker-image/pyroscope/push + +IMAGE_TAG=$(docker image ls | grep "$IMAGE_NAME" | head -n 1 | awk '{print $2}') +if [ -z "$IMAGE_TAG" ]; then + echo "Error: can't find image tag for $IMAGE_NAME." + exit 1 +fi + +echo "using image: $IMAGE_NAME:$IMAGE_TAG" + +# deploy + +helm -n "$PYROSCOPE_TEST_NAMESPACE" upgrade --install \ + --create-namespace pyroscope \ + --values "$VALUES_FILE" \ + --set pyroscope.image.repository="$IMAGE_NAME" \ + --set pyroscope.image.tag="$IMAGE_TAG" \ + "$HELM_CHART" + +kubectl --namespace "$PYROSCOPE_TEST_NAMESPACE" port-forward svc/pyroscope-query-frontend 4040:4040 diff --git a/tools/dev/experiment/local_uninstall.sh b/tools/dev/experiment/local_uninstall.sh new file mode 100755 index 0000000000..60d88b776a --- /dev/null +++ b/tools/dev/experiment/local_uninstall.sh @@ -0,0 +1,8 @@ +#!/usr/bin/env bash + +set -e +set -o pipefail + +PYROSCOPE_TEST_NAMESPACE=pyroscope-test + +helm -n "$PYROSCOPE_TEST_NAMESPACE" uninstall pyroscope diff --git a/tools/dev/experiment/values-micro-services-experiment.yaml b/tools/dev/experiment/values-micro-services-experiment.yaml new file mode 100644 index 0000000000..f7782d25fc --- /dev/null +++ b/tools/dev/experiment/values-micro-services-experiment.yaml @@ -0,0 +1,71 @@ +pyroscope: + extraEnvVars: + PYROSCOPE_V2_EXPERIMENT: 1 + extraArgs: + query-backend.address: "dns:///_grpc._tcp.pyroscope-query-worker-headless.$(NAMESPACE_FQDN):9095" + metastore.address: "dns:///_grpc._tcp.pyroscope-metastore-headless.$(NAMESPACE_FQDN):9095" + metastore.raft.bind-address: ":9099" + metastore.raft.server-id: "$(POD_NAME).pyroscope-metastore-headless.$(NAMESPACE_FQDN)" + metastore.raft.advertise-address: "$(POD_NAME).pyroscope-metastore-headless.$(NAMESPACE_FQDN):9099" + metastore.raft.bootstrap-peers: "dnssrvnoa+_raft._tcp.pyroscope-metastore-headless.$(NAMESPACE_FQDN):9099" + metastore.raft.bootstrap-expect-peers: "3" + + components: + distributor: + kind: Deployment + replicaCount: 2 + + segment-writer: + kind: StatefulSet + replicaCount: 3 + persistence: + enabled: false + + query-frontend: + kind: Deployment + replicaCount: 2 + + query-backend: + kind: Deployment + replicaCount: 3 + service: + extraPorts: + - name: grpc + port: 9095 + protocol: TCP + targetPort: 9095 + + metastore: + kind: StatefulSet + replicaCount: 3 + service: + publishNotReadyAddresses: true + extraPorts: + - name: grpc + port: 9095 + protocol: TCP + targetPort: 9095 + - name: raft + port: 9099 + protocol: TCP + targetPort: 9099 + persistence: + enabled: true + size: 10Gi + extraVolumeMounts: + - name: data + mountPath: /data-metastore + subPath: metastore + + compaction-worker: + kind: StatefulSet + replicaCount: 3 + persistence: + enabled: false + extraVolumeMounts: + - name: data + mountPath: /data-compaction-worker + subPath: compaction-worker + +minio: + enabled: true diff --git a/yarn.lock b/yarn.lock index 0a3d1fc59f..bf32e4dc19 100644 --- a/yarn.lock +++ b/yarn.lock @@ -9704,9 +9704,9 @@ electron-to-chromium@^1.4.431: integrity sha512-zZM48Lmy2FKWgqyvsX9XK+J6FfP7aCDUFLmgooLJzA7v1agCs/sxSoBpTIwDLhmbhpx9yJIxj2INig/ncjJRqg== elliptic@^6.5.3, elliptic@^6.5.4: - version "6.5.4" - resolved "https://registry.yarnpkg.com/elliptic/-/elliptic-6.5.4.tgz#da37cebd31e79a1367e941b592ed1fbebd58abbb" - integrity sha512-iLhC6ULemrljPZb+QutR5TQGB+pdW6KGD5RSegS+8sorOZT+rdQFbsQFJgvN3eRqNALqJer4oQ16YvJHlU8hzQ== + version "6.5.7" + resolved "https://registry.yarnpkg.com/elliptic/-/elliptic-6.5.7.tgz#8ec4da2cb2939926a1b9a73619d768207e647c8b" + integrity sha512-ESVCtTwiA+XhY3wyh24QqRGBoP3rEdDUl3EDUUo9tft074fi19IrdpH7hLCMMP3CIj7jb3W96rn8lt/BqIlt5Q== dependencies: bn.js "^4.11.9" brorand "^1.1.0"