2021-10-25 11:44:37 +00:00
|
|
|
// Licensed to the LF AI & Data foundation under one
|
|
|
|
// or more contributor license agreements. See the NOTICE file
|
|
|
|
// distributed with this work for additional information
|
|
|
|
// regarding copyright ownership. The ASF licenses this file
|
|
|
|
// to you under the Apache License, Version 2.0 (the
|
|
|
|
// "License"); you may not use this file except in compliance
|
2021-04-19 03:35:38 +00:00
|
|
|
// with the License. You may obtain a copy of the License at
|
|
|
|
//
|
2021-10-25 11:44:37 +00:00
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
2021-04-19 03:35:38 +00:00
|
|
|
//
|
2021-10-25 11:44:37 +00:00
|
|
|
// 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.
|
2021-06-08 11:25:37 +00:00
|
|
|
|
2021-06-22 02:42:07 +00:00
|
|
|
package datacoord
|
2021-01-15 09:09:41 +00:00
|
|
|
|
|
|
|
import (
|
2021-03-23 08:57:59 +00:00
|
|
|
"context"
|
2021-01-15 09:09:41 +00:00
|
|
|
"fmt"
|
|
|
|
"sync"
|
|
|
|
"time"
|
|
|
|
|
2021-04-22 06:45:57 +00:00
|
|
|
"github.com/milvus-io/milvus/internal/log"
|
|
|
|
"github.com/milvus-io/milvus/internal/proto/commonpb"
|
2021-12-23 02:29:08 +00:00
|
|
|
"github.com/milvus-io/milvus/internal/proto/datapb"
|
2021-04-22 06:45:57 +00:00
|
|
|
"github.com/milvus-io/milvus/internal/util/trace"
|
|
|
|
"github.com/milvus-io/milvus/internal/util/tsoutil"
|
2021-12-23 02:29:08 +00:00
|
|
|
"go.uber.org/zap"
|
2021-01-15 09:09:41 +00:00
|
|
|
)
|
|
|
|
|
2021-09-07 05:59:58 +00:00
|
|
|
var (
|
2021-09-16 11:55:49 +00:00
|
|
|
// allocPool pool of Allocation, to reduce allocation of Allocation
|
2021-09-07 05:59:58 +00:00
|
|
|
allocPool = sync.Pool{
|
|
|
|
New: func() interface{} {
|
|
|
|
return &Allocation{}
|
|
|
|
},
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
2021-12-17 10:49:09 +00:00
|
|
|
// getAllocation unifies way to retrieve allocation struct
|
2021-09-07 05:59:58 +00:00
|
|
|
func getAllocation(numOfRows int64) *Allocation {
|
|
|
|
v := allocPool.Get()
|
|
|
|
a, ok := v.(*Allocation)
|
|
|
|
if !ok {
|
|
|
|
a = &Allocation{}
|
|
|
|
}
|
|
|
|
if a == nil {
|
|
|
|
return &Allocation{
|
|
|
|
NumOfRows: numOfRows,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
a.NumOfRows = numOfRows
|
|
|
|
a.ExpireTime = 0
|
|
|
|
a.SegmentID = 0
|
|
|
|
return a
|
|
|
|
}
|
|
|
|
|
2021-12-17 10:49:09 +00:00
|
|
|
// putAllocation puts an allocation for recycling
|
2021-09-07 05:59:58 +00:00
|
|
|
func putAllocation(a *Allocation) {
|
|
|
|
allocPool.Put(a)
|
|
|
|
}
|
|
|
|
|
2021-12-17 10:49:09 +00:00
|
|
|
// Manager manages segment related operations.
|
2021-06-03 11:06:33 +00:00
|
|
|
type Manager interface {
|
2021-10-08 11:05:25 +00:00
|
|
|
// AllocSegment allocates rows and record the allocation.
|
2021-07-23 13:58:33 +00:00
|
|
|
AllocSegment(ctx context.Context, collectionID, partitionID UniqueID, channelName string, requestRows int64) ([]*Allocation, error)
|
2021-10-08 11:05:25 +00:00
|
|
|
// DropSegment drops the segment from manager.
|
2021-03-23 08:57:59 +00:00
|
|
|
DropSegment(ctx context.Context, segmentID UniqueID)
|
2022-04-25 03:07:47 +00:00
|
|
|
// SealAllSegments seals all segments of collection with collectionID and return sealed segments.
|
|
|
|
// If segIDs is not empty, also seals segments in segIDs.
|
|
|
|
SealAllSegments(ctx context.Context, collectionID UniqueID, segIDs []UniqueID) ([]UniqueID, error)
|
2021-10-08 11:05:25 +00:00
|
|
|
// GetFlushableSegments returns flushable segment ids
|
2021-05-25 07:35:37 +00:00
|
|
|
GetFlushableSegments(ctx context.Context, channel string, ts Timestamp) ([]UniqueID, error)
|
2021-10-08 11:05:25 +00:00
|
|
|
// ExpireAllocations notifies segment status to expire old allocations
|
2021-06-24 06:20:10 +00:00
|
|
|
ExpireAllocations(channel string, ts Timestamp) error
|
2021-11-11 16:22:42 +00:00
|
|
|
// DropSegmentsOfChannel drops all segments in a channel
|
|
|
|
DropSegmentsOfChannel(ctx context.Context, channel string)
|
2021-01-15 09:09:41 +00:00
|
|
|
}
|
|
|
|
|
2021-09-14 09:27:55 +00:00
|
|
|
// Allocation records the allocation info
|
2021-07-12 09:24:25 +00:00
|
|
|
type Allocation struct {
|
2021-07-23 13:58:33 +00:00
|
|
|
SegmentID UniqueID
|
|
|
|
NumOfRows int64
|
|
|
|
ExpireTime Timestamp
|
2021-06-03 11:06:33 +00:00
|
|
|
}
|
|
|
|
|
2021-10-02 06:45:56 +00:00
|
|
|
// make sure SegmentManager implements Manager
|
|
|
|
var _ Manager = (*SegmentManager)(nil)
|
|
|
|
|
2021-06-24 06:20:10 +00:00
|
|
|
// SegmentManager handles segment related logic
|
2021-06-03 11:06:33 +00:00
|
|
|
type SegmentManager struct {
|
2021-07-12 09:24:25 +00:00
|
|
|
meta *meta
|
|
|
|
mu sync.RWMutex
|
|
|
|
allocator allocator
|
|
|
|
helper allocHelper
|
|
|
|
segments []UniqueID
|
|
|
|
estimatePolicy calUpperLimitPolicy
|
2021-07-23 13:58:33 +00:00
|
|
|
allocPolicy AllocatePolicy
|
2021-06-08 11:25:37 +00:00
|
|
|
segmentSealPolicies []segmentSealPolicy
|
|
|
|
channelSealPolicies []channelSealPolicy
|
|
|
|
flushPolicy flushPolicy
|
2021-04-12 08:35:51 +00:00
|
|
|
}
|
|
|
|
|
2021-05-21 10:30:41 +00:00
|
|
|
type allocHelper struct {
|
|
|
|
afterCreateSegment func(segment *datapb.SegmentInfo) error
|
|
|
|
}
|
2021-05-28 01:55:21 +00:00
|
|
|
|
2021-12-17 14:34:42 +00:00
|
|
|
// allocOption allocation option applies to `SegmentManager`
|
2021-06-24 06:20:10 +00:00
|
|
|
type allocOption interface {
|
|
|
|
apply(manager *SegmentManager)
|
|
|
|
}
|
|
|
|
|
|
|
|
// allocFunc function shortcut for allocOption
|
|
|
|
type allocFunc func(manager *SegmentManager)
|
|
|
|
|
|
|
|
// implement allocOption
|
|
|
|
func (f allocFunc) apply(manager *SegmentManager) {
|
|
|
|
f(manager)
|
2021-03-05 08:52:45 +00:00
|
|
|
}
|
2021-01-15 09:09:41 +00:00
|
|
|
|
2021-06-24 06:20:10 +00:00
|
|
|
// get allocOption with allocHelper setting
|
2021-05-21 10:30:41 +00:00
|
|
|
func withAllocHelper(helper allocHelper) allocOption {
|
2021-06-24 06:20:10 +00:00
|
|
|
return allocFunc(func(manager *SegmentManager) { manager.helper = helper })
|
2021-04-12 08:35:51 +00:00
|
|
|
}
|
2021-05-21 10:30:41 +00:00
|
|
|
|
2021-06-24 06:20:10 +00:00
|
|
|
// get default allocHelper, which does nothing
|
2021-05-21 10:30:41 +00:00
|
|
|
func defaultAllocHelper() allocHelper {
|
|
|
|
return allocHelper{
|
|
|
|
afterCreateSegment: func(segment *datapb.SegmentInfo) error { return nil },
|
2021-01-15 09:09:41 +00:00
|
|
|
}
|
2021-05-21 10:30:41 +00:00
|
|
|
}
|
|
|
|
|
2021-06-24 06:20:10 +00:00
|
|
|
// get allocOption with estimatePolicy
|
2021-05-21 10:30:41 +00:00
|
|
|
func withCalUpperLimitPolicy(policy calUpperLimitPolicy) allocOption {
|
2021-06-24 06:20:10 +00:00
|
|
|
return allocFunc(func(manager *SegmentManager) { manager.estimatePolicy = policy })
|
2021-01-15 09:09:41 +00:00
|
|
|
}
|
|
|
|
|
2021-06-24 06:20:10 +00:00
|
|
|
// get allocOption with allocPolicy
|
2021-07-23 13:58:33 +00:00
|
|
|
func withAllocPolicy(policy AllocatePolicy) allocOption {
|
2021-06-24 06:20:10 +00:00
|
|
|
return allocFunc(func(manager *SegmentManager) { manager.allocPolicy = policy })
|
2021-05-21 10:30:41 +00:00
|
|
|
}
|
|
|
|
|
2021-06-24 06:20:10 +00:00
|
|
|
// get allocOption with segmentSealPolicies
|
2021-06-08 11:25:37 +00:00
|
|
|
func withSegmentSealPolices(policies ...segmentSealPolicy) allocOption {
|
2021-06-24 06:20:10 +00:00
|
|
|
return allocFunc(func(manager *SegmentManager) {
|
|
|
|
// do override instead of append, to override default options
|
|
|
|
manager.segmentSealPolicies = policies
|
|
|
|
})
|
2021-06-08 11:25:37 +00:00
|
|
|
}
|
|
|
|
|
2021-06-24 06:20:10 +00:00
|
|
|
// get allocOption with channelSealPolicies
|
2021-06-08 11:25:37 +00:00
|
|
|
func withChannelSealPolices(policies ...channelSealPolicy) allocOption {
|
2021-06-24 06:20:10 +00:00
|
|
|
return allocFunc(func(manager *SegmentManager) {
|
|
|
|
// do override instead of append, to override default options
|
|
|
|
manager.channelSealPolicies = policies
|
|
|
|
})
|
2021-05-21 10:30:41 +00:00
|
|
|
}
|
|
|
|
|
2021-06-24 06:20:10 +00:00
|
|
|
// get allocOption with flushPolicy
|
2021-05-21 10:30:41 +00:00
|
|
|
func withFlushPolicy(policy flushPolicy) allocOption {
|
2021-06-24 06:20:10 +00:00
|
|
|
return allocFunc(func(manager *SegmentManager) { manager.flushPolicy = policy })
|
2021-05-21 10:30:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func defaultCalUpperLimitPolicy() calUpperLimitPolicy {
|
2021-07-12 09:24:25 +00:00
|
|
|
return calBySchemaPolicy
|
2021-05-21 10:30:41 +00:00
|
|
|
}
|
|
|
|
|
2021-12-17 14:34:42 +00:00
|
|
|
func defaultAllocatePolicy() AllocatePolicy {
|
2021-07-23 13:58:33 +00:00
|
|
|
return AllocatePolicyV1
|
2021-05-21 10:30:41 +00:00
|
|
|
}
|
|
|
|
|
2021-08-20 07:42:12 +00:00
|
|
|
func defaultSegmentSealPolicy() []segmentSealPolicy {
|
|
|
|
return []segmentSealPolicy{
|
2022-05-06 15:41:51 +00:00
|
|
|
sealByLifetimePolicy(Params.DataCoordCfg.SegmentMaxLifetime),
|
2021-12-23 10:39:11 +00:00
|
|
|
getSegmentCapacityPolicy(Params.DataCoordCfg.SegmentSealProportion),
|
2021-08-20 07:42:12 +00:00
|
|
|
}
|
2021-06-08 11:25:37 +00:00
|
|
|
}
|
|
|
|
|
2021-05-21 10:30:41 +00:00
|
|
|
func defaultFlushPolicy() flushPolicy {
|
2021-07-12 09:24:25 +00:00
|
|
|
return flushPolicyV1
|
2021-05-21 10:30:41 +00:00
|
|
|
}
|
|
|
|
|
2022-02-22 05:15:51 +00:00
|
|
|
// newSegmentManager should be the only way to retrieve SegmentManager.
|
2021-06-03 11:06:33 +00:00
|
|
|
func newSegmentManager(meta *meta, allocator allocator, opts ...allocOption) *SegmentManager {
|
|
|
|
manager := &SegmentManager{
|
2021-07-12 09:24:25 +00:00
|
|
|
meta: meta,
|
|
|
|
allocator: allocator,
|
|
|
|
helper: defaultAllocHelper(),
|
|
|
|
segments: make([]UniqueID, 0),
|
2021-06-08 11:25:37 +00:00
|
|
|
estimatePolicy: defaultCalUpperLimitPolicy(),
|
2021-12-17 14:34:42 +00:00
|
|
|
allocPolicy: defaultAllocatePolicy(),
|
2021-08-20 07:42:12 +00:00
|
|
|
segmentSealPolicies: defaultSegmentSealPolicy(), // default only segment size policy
|
|
|
|
channelSealPolicies: []channelSealPolicy{}, // no default channel seal policy
|
2021-06-08 11:25:37 +00:00
|
|
|
flushPolicy: defaultFlushPolicy(),
|
2021-05-21 10:30:41 +00:00
|
|
|
}
|
|
|
|
for _, opt := range opts {
|
2021-06-03 11:06:33 +00:00
|
|
|
opt.apply(manager)
|
2021-05-21 10:30:41 +00:00
|
|
|
}
|
2021-06-03 11:06:33 +00:00
|
|
|
manager.loadSegmentsFromMeta()
|
|
|
|
return manager
|
2021-01-15 09:09:41 +00:00
|
|
|
}
|
|
|
|
|
2021-06-24 06:20:10 +00:00
|
|
|
// loadSegmentsFromMeta generate corresponding segment status for each segment from meta
|
2021-06-03 11:06:33 +00:00
|
|
|
func (s *SegmentManager) loadSegmentsFromMeta() {
|
|
|
|
segments := s.meta.GetUnFlushedSegments()
|
2021-07-12 09:24:25 +00:00
|
|
|
segmentsID := make([]UniqueID, 0, len(segments))
|
|
|
|
for _, segment := range segments {
|
|
|
|
segmentsID = append(segmentsID, segment.GetID())
|
2021-06-03 11:06:33 +00:00
|
|
|
}
|
2021-07-12 09:24:25 +00:00
|
|
|
s.segments = segmentsID
|
2021-06-03 11:06:33 +00:00
|
|
|
}
|
2021-06-24 06:20:10 +00:00
|
|
|
|
|
|
|
// AllocSegment allocate segment per request collcation, partication, channel and rows
|
2021-06-03 11:06:33 +00:00
|
|
|
func (s *SegmentManager) AllocSegment(ctx context.Context, collectionID UniqueID,
|
2021-07-23 13:58:33 +00:00
|
|
|
partitionID UniqueID, channelName string, requestRows int64) ([]*Allocation, error) {
|
2021-03-23 08:57:59 +00:00
|
|
|
sp, _ := trace.StartSpanFromContext(ctx)
|
|
|
|
defer sp.Finish()
|
2021-04-25 01:51:57 +00:00
|
|
|
s.mu.Lock()
|
|
|
|
defer s.mu.Unlock()
|
2021-01-15 09:09:41 +00:00
|
|
|
|
2021-07-23 13:58:33 +00:00
|
|
|
// filter segments
|
|
|
|
segments := make([]*SegmentInfo, 0)
|
2021-07-12 09:24:25 +00:00
|
|
|
for _, segmentID := range s.segments {
|
2021-07-23 13:58:33 +00:00
|
|
|
segment := s.meta.GetSegment(segmentID)
|
2021-07-12 09:24:25 +00:00
|
|
|
if segment == nil {
|
2021-07-23 13:58:33 +00:00
|
|
|
log.Warn("Failed to get seginfo from meta", zap.Int64("id", segmentID))
|
2021-06-03 11:06:33 +00:00
|
|
|
continue
|
|
|
|
}
|
2021-12-01 14:39:37 +00:00
|
|
|
if !satisfy(segment, collectionID, partitionID, channelName) || !isGrowing(segment) {
|
2021-06-24 06:20:10 +00:00
|
|
|
continue
|
|
|
|
}
|
2021-07-23 13:58:33 +00:00
|
|
|
segments = append(segments, segment)
|
2021-04-12 08:35:51 +00:00
|
|
|
}
|
|
|
|
|
2022-02-22 05:15:51 +00:00
|
|
|
// Apply allocation policy.
|
2021-07-23 13:58:33 +00:00
|
|
|
maxCountPerSegment, err := s.estimateMaxNumOfRows(collectionID)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
2021-01-15 09:09:41 +00:00
|
|
|
}
|
2021-07-23 13:58:33 +00:00
|
|
|
newSegmentAllocations, existedSegmentAllocations := s.allocPolicy(segments,
|
|
|
|
requestRows, int64(maxCountPerSegment))
|
2021-01-15 09:09:41 +00:00
|
|
|
|
2021-07-23 13:58:33 +00:00
|
|
|
// create new segments and add allocations
|
2021-08-23 09:59:51 +00:00
|
|
|
expireTs, err := s.genExpireTs(ctx)
|
2021-01-15 09:09:41 +00:00
|
|
|
if err != nil {
|
2021-07-12 09:24:25 +00:00
|
|
|
return nil, err
|
2021-01-15 09:09:41 +00:00
|
|
|
}
|
2021-07-23 13:58:33 +00:00
|
|
|
for _, allocation := range newSegmentAllocations {
|
|
|
|
segment, err := s.openNewSegment(ctx, collectionID, partitionID, channelName)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
allocation.ExpireTime = expireTs
|
|
|
|
allocation.SegmentID = segment.GetID()
|
|
|
|
if err := s.meta.AddAllocation(segment.GetID(), allocation); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, allocation := range existedSegmentAllocations {
|
|
|
|
allocation.ExpireTime = expireTs
|
|
|
|
if err := s.meta.AddAllocation(allocation.SegmentID, allocation); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
2021-06-24 06:20:10 +00:00
|
|
|
|
2021-07-23 13:58:33 +00:00
|
|
|
allocations := append(newSegmentAllocations, existedSegmentAllocations...)
|
|
|
|
return allocations, nil
|
2021-01-15 09:09:41 +00:00
|
|
|
}
|
|
|
|
|
2021-12-01 14:39:37 +00:00
|
|
|
func satisfy(segment *SegmentInfo, collectionID, partitionID UniqueID, channel string) bool {
|
|
|
|
return segment.GetCollectionID() == collectionID && segment.GetPartitionID() == partitionID &&
|
|
|
|
segment.GetInsertChannel() == channel
|
|
|
|
}
|
|
|
|
|
|
|
|
func isGrowing(segment *SegmentInfo) bool {
|
|
|
|
return segment.GetState() == commonpb.SegmentState_Growing
|
|
|
|
}
|
|
|
|
|
2021-08-23 09:59:51 +00:00
|
|
|
func (s *SegmentManager) genExpireTs(ctx context.Context) (Timestamp, error) {
|
|
|
|
ts, err := s.allocator.allocTimestamp(ctx)
|
2021-05-21 10:30:41 +00:00
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
|
|
|
}
|
|
|
|
physicalTs, logicalTs := tsoutil.ParseTS(ts)
|
2021-12-23 10:39:11 +00:00
|
|
|
expirePhysicalTs := physicalTs.Add(time.Duration(Params.DataCoordCfg.SegAssignmentExpiration) * time.Millisecond)
|
2021-05-21 10:30:41 +00:00
|
|
|
expireTs := tsoutil.ComposeTS(expirePhysicalTs.UnixNano()/int64(time.Millisecond), int64(logicalTs))
|
|
|
|
return expireTs, nil
|
|
|
|
}
|
|
|
|
|
2021-07-12 09:24:25 +00:00
|
|
|
func (s *SegmentManager) openNewSegment(ctx context.Context, collectionID UniqueID, partitionID UniqueID, channelName string) (*SegmentInfo, error) {
|
2021-04-12 08:35:51 +00:00
|
|
|
sp, _ := trace.StartSpanFromContext(ctx)
|
|
|
|
defer sp.Finish()
|
2021-08-23 09:59:51 +00:00
|
|
|
id, err := s.allocator.allocID(ctx)
|
2021-04-12 08:35:51 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2021-06-22 10:24:08 +00:00
|
|
|
maxNumOfRows, err := s.estimateMaxNumOfRows(collectionID)
|
2021-04-12 08:35:51 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2021-06-04 03:45:45 +00:00
|
|
|
|
2021-05-21 10:30:41 +00:00
|
|
|
segmentInfo := &datapb.SegmentInfo{
|
2021-05-31 10:47:32 +00:00
|
|
|
ID: id,
|
|
|
|
CollectionID: collectionID,
|
|
|
|
PartitionID: partitionID,
|
|
|
|
InsertChannel: channelName,
|
2021-06-04 03:45:45 +00:00
|
|
|
NumOfRows: 0,
|
2021-05-31 10:47:32 +00:00
|
|
|
State: commonpb.SegmentState_Growing,
|
2021-06-22 10:24:08 +00:00
|
|
|
MaxRowNum: int64(maxNumOfRows),
|
2021-05-21 10:30:41 +00:00
|
|
|
LastExpireTime: 0,
|
2021-04-12 08:35:51 +00:00
|
|
|
}
|
2021-07-12 09:24:25 +00:00
|
|
|
segment := NewSegmentInfo(segmentInfo)
|
|
|
|
if err := s.meta.AddSegment(segment); err != nil {
|
2021-05-21 10:30:41 +00:00
|
|
|
return nil, err
|
2021-04-12 08:35:51 +00:00
|
|
|
}
|
2021-07-12 09:24:25 +00:00
|
|
|
s.segments = append(s.segments, id)
|
2022-03-02 07:35:55 +00:00
|
|
|
log.Info("datacoord: estimateTotalRows: ",
|
2021-05-21 10:30:41 +00:00
|
|
|
zap.Int64("CollectionID", segmentInfo.CollectionID),
|
|
|
|
zap.Int64("SegmentID", segmentInfo.ID),
|
2021-06-22 10:24:08 +00:00
|
|
|
zap.Int("Rows", maxNumOfRows),
|
|
|
|
zap.String("Channel", segmentInfo.InsertChannel))
|
2021-05-21 10:30:41 +00:00
|
|
|
|
2021-09-27 14:46:12 +00:00
|
|
|
return segment, s.helper.afterCreateSegment(segmentInfo)
|
2021-04-12 08:35:51 +00:00
|
|
|
}
|
|
|
|
|
2021-06-22 10:24:08 +00:00
|
|
|
func (s *SegmentManager) estimateMaxNumOfRows(collectionID UniqueID) (int, error) {
|
2021-07-07 06:02:01 +00:00
|
|
|
collMeta := s.meta.GetCollection(collectionID)
|
|
|
|
if collMeta == nil {
|
2021-12-07 06:36:27 +00:00
|
|
|
return -1, fmt.Errorf("failed to get collection %d", collectionID)
|
2021-01-15 09:09:41 +00:00
|
|
|
}
|
2021-07-12 09:24:25 +00:00
|
|
|
return s.estimatePolicy(collMeta.Schema)
|
2021-01-15 09:09:41 +00:00
|
|
|
}
|
|
|
|
|
2021-10-08 11:05:25 +00:00
|
|
|
// DropSegment drop the segment from manager.
|
2021-06-03 11:06:33 +00:00
|
|
|
func (s *SegmentManager) DropSegment(ctx context.Context, segmentID UniqueID) {
|
2021-03-23 08:57:59 +00:00
|
|
|
sp, _ := trace.StartSpanFromContext(ctx)
|
|
|
|
defer sp.Finish()
|
2021-04-25 01:51:57 +00:00
|
|
|
s.mu.Lock()
|
|
|
|
defer s.mu.Unlock()
|
2021-07-12 09:24:25 +00:00
|
|
|
for i, id := range s.segments {
|
|
|
|
if id == segmentID {
|
|
|
|
s.segments = append(s.segments[:i], s.segments[i+1:]...)
|
|
|
|
break
|
2021-06-24 06:20:10 +00:00
|
|
|
}
|
|
|
|
}
|
2021-07-12 09:24:25 +00:00
|
|
|
segment := s.meta.GetSegment(segmentID)
|
|
|
|
if segment == nil {
|
2021-10-21 06:16:37 +00:00
|
|
|
log.Warn("Failed to get segment", zap.Int64("id", segmentID))
|
2021-10-08 11:05:25 +00:00
|
|
|
return
|
2021-07-12 09:24:25 +00:00
|
|
|
}
|
|
|
|
s.meta.SetAllocations(segmentID, []*Allocation{})
|
|
|
|
for _, allocation := range segment.allocations {
|
2021-09-07 05:59:58 +00:00
|
|
|
putAllocation(allocation)
|
2021-07-12 09:24:25 +00:00
|
|
|
}
|
2021-01-15 09:09:41 +00:00
|
|
|
}
|
|
|
|
|
2022-04-25 03:07:47 +00:00
|
|
|
// SealAllSegments seals all segments of collection with collectionID and return sealed segments
|
|
|
|
func (s *SegmentManager) SealAllSegments(ctx context.Context, collectionID UniqueID, segIDs []UniqueID) ([]UniqueID, error) {
|
2021-04-23 02:15:30 +00:00
|
|
|
sp, _ := trace.StartSpanFromContext(ctx)
|
|
|
|
defer sp.Finish()
|
2021-04-25 01:51:57 +00:00
|
|
|
s.mu.Lock()
|
|
|
|
defer s.mu.Unlock()
|
2021-12-19 12:00:42 +00:00
|
|
|
var ret []UniqueID
|
2022-04-25 03:07:47 +00:00
|
|
|
segCandidates := s.segments
|
|
|
|
if len(segIDs) != 0 {
|
|
|
|
segCandidates = segIDs
|
|
|
|
}
|
|
|
|
for _, id := range segCandidates {
|
2021-07-12 09:24:25 +00:00
|
|
|
info := s.meta.GetSegment(id)
|
2021-07-07 06:02:01 +00:00
|
|
|
if info == nil {
|
2022-04-25 03:07:47 +00:00
|
|
|
log.Warn("failed to get seg info from meta", zap.Int64("segment ID", id))
|
2021-06-23 08:56:11 +00:00
|
|
|
continue
|
|
|
|
}
|
2021-06-24 06:20:10 +00:00
|
|
|
if info.CollectionID != collectionID {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if info.State == commonpb.SegmentState_Sealed {
|
2021-07-12 09:24:25 +00:00
|
|
|
ret = append(ret, id)
|
2021-06-08 11:25:37 +00:00
|
|
|
continue
|
2021-06-03 11:06:33 +00:00
|
|
|
}
|
2021-07-12 09:24:25 +00:00
|
|
|
if err := s.meta.SetState(id, commonpb.SegmentState_Sealed); err != nil {
|
2021-06-23 08:56:11 +00:00
|
|
|
return nil, err
|
2021-06-08 11:25:37 +00:00
|
|
|
}
|
2021-07-12 09:24:25 +00:00
|
|
|
ret = append(ret, id)
|
2021-06-03 11:06:33 +00:00
|
|
|
}
|
2021-06-23 08:56:11 +00:00
|
|
|
return ret, nil
|
2021-04-25 01:51:57 +00:00
|
|
|
}
|
|
|
|
|
2021-09-16 11:55:49 +00:00
|
|
|
// GetFlushableSegments get segment ids with Sealed State and flushable (meets flushPolicy)
|
2021-10-15 03:46:33 +00:00
|
|
|
func (s *SegmentManager) GetFlushableSegments(ctx context.Context, channel string, t Timestamp) ([]UniqueID, error) {
|
2021-04-25 01:51:57 +00:00
|
|
|
s.mu.Lock()
|
|
|
|
defer s.mu.Unlock()
|
2021-03-23 08:57:59 +00:00
|
|
|
sp, _ := trace.StartSpanFromContext(ctx)
|
|
|
|
defer sp.Finish()
|
2021-07-27 06:34:48 +00:00
|
|
|
if err := s.tryToSealSegment(t, channel); err != nil {
|
2021-05-21 10:30:41 +00:00
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2021-07-12 09:24:25 +00:00
|
|
|
ret := make([]UniqueID, 0, len(s.segments))
|
|
|
|
for _, id := range s.segments {
|
|
|
|
info := s.meta.GetSegment(id)
|
2021-07-27 06:34:48 +00:00
|
|
|
if info == nil || info.InsertChannel != channel {
|
2021-05-25 07:35:37 +00:00
|
|
|
continue
|
|
|
|
}
|
2021-07-12 09:24:25 +00:00
|
|
|
if s.flushPolicy(info, t) {
|
|
|
|
ret = append(ret, id)
|
2021-05-21 10:30:41 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret, nil
|
|
|
|
}
|
|
|
|
|
2021-06-24 06:20:10 +00:00
|
|
|
// ExpireAllocations notify segment status to expire old allocations
|
|
|
|
func (s *SegmentManager) ExpireAllocations(channel string, ts Timestamp) error {
|
|
|
|
s.mu.Lock()
|
|
|
|
defer s.mu.Unlock()
|
2021-07-12 09:24:25 +00:00
|
|
|
for _, id := range s.segments {
|
|
|
|
segment := s.meta.GetSegment(id)
|
2021-07-27 06:34:48 +00:00
|
|
|
if segment == nil || segment.InsertChannel != channel {
|
2021-06-24 06:20:10 +00:00
|
|
|
continue
|
|
|
|
}
|
2021-08-12 02:48:08 +00:00
|
|
|
allocations := make([]*Allocation, 0, len(segment.allocations))
|
2021-07-12 09:24:25 +00:00
|
|
|
for i := 0; i < len(segment.allocations); i++ {
|
2021-07-23 13:58:33 +00:00
|
|
|
if segment.allocations[i].ExpireTime <= ts {
|
2021-07-12 09:24:25 +00:00
|
|
|
a := segment.allocations[i]
|
2021-09-07 05:59:58 +00:00
|
|
|
putAllocation(a)
|
2021-08-12 02:48:08 +00:00
|
|
|
} else {
|
|
|
|
allocations = append(allocations, segment.allocations[i])
|
2021-06-24 06:20:10 +00:00
|
|
|
}
|
|
|
|
}
|
2021-08-12 02:48:08 +00:00
|
|
|
s.meta.SetAllocations(segment.GetID(), allocations)
|
2021-06-24 06:20:10 +00:00
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2021-06-08 11:25:37 +00:00
|
|
|
// tryToSealSegment applies segment & channel seal policies
|
2021-07-27 06:34:48 +00:00
|
|
|
func (s *SegmentManager) tryToSealSegment(ts Timestamp, channel string) error {
|
2021-07-12 09:24:25 +00:00
|
|
|
channelInfo := make(map[string][]*SegmentInfo)
|
|
|
|
for _, id := range s.segments {
|
|
|
|
info := s.meta.GetSegment(id)
|
2021-07-27 06:34:48 +00:00
|
|
|
if info == nil || info.InsertChannel != channel {
|
2021-06-24 06:20:10 +00:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
channelInfo[info.InsertChannel] = append(channelInfo[info.InsertChannel], info)
|
|
|
|
if info.State == commonpb.SegmentState_Sealed {
|
2021-05-21 10:30:41 +00:00
|
|
|
continue
|
|
|
|
}
|
2021-06-08 11:25:37 +00:00
|
|
|
// change shouldSeal to segment seal policy logic
|
|
|
|
for _, policy := range s.segmentSealPolicies {
|
2021-07-12 09:24:25 +00:00
|
|
|
if policy(info, ts) {
|
|
|
|
if err := s.meta.SetState(id, commonpb.SegmentState_Sealed); err != nil {
|
2021-06-08 11:25:37 +00:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
break
|
|
|
|
}
|
2021-05-21 10:30:41 +00:00
|
|
|
}
|
2021-06-08 11:25:37 +00:00
|
|
|
}
|
2021-06-24 06:20:10 +00:00
|
|
|
for channel, segmentInfos := range channelInfo {
|
2021-06-08 11:25:37 +00:00
|
|
|
for _, policy := range s.channelSealPolicies {
|
2021-06-24 06:20:10 +00:00
|
|
|
vs := policy(channel, segmentInfos, ts)
|
|
|
|
for _, info := range vs {
|
|
|
|
if info.State == commonpb.SegmentState_Sealed {
|
2021-06-08 11:25:37 +00:00
|
|
|
continue
|
|
|
|
}
|
2021-07-07 06:02:01 +00:00
|
|
|
if err := s.meta.SetState(info.GetID(), commonpb.SegmentState_Sealed); err != nil {
|
2021-06-08 11:25:37 +00:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
2021-01-15 09:09:41 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
2021-11-11 16:22:42 +00:00
|
|
|
|
|
|
|
// DropSegmentsOfChannel drops all segments in a channel
|
|
|
|
func (s *SegmentManager) DropSegmentsOfChannel(ctx context.Context, channel string) {
|
|
|
|
s.mu.Lock()
|
|
|
|
defer s.mu.Unlock()
|
|
|
|
|
|
|
|
validSegments := make([]int64, 0, len(s.segments))
|
|
|
|
for _, sid := range s.segments {
|
|
|
|
segment := s.meta.GetSegment(sid)
|
2021-11-12 09:31:11 +00:00
|
|
|
if segment == nil {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if segment.GetInsertChannel() != channel {
|
2021-11-11 16:22:42 +00:00
|
|
|
validSegments = append(validSegments, sid)
|
2021-11-12 09:31:11 +00:00
|
|
|
continue
|
2021-11-11 16:22:42 +00:00
|
|
|
}
|
|
|
|
s.meta.SetAllocations(sid, nil)
|
|
|
|
for _, allocation := range segment.allocations {
|
|
|
|
putAllocation(allocation)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
s.segments = validSegments
|
|
|
|
}
|