You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					
					
						
							100 lines
						
					
					
						
							3.8 KiB
						
					
					
				
			
		
		
		
			
			
			
		
		
	
	
							100 lines
						
					
					
						
							3.8 KiB
						
					
					
				| package offset | |
| 
 | |
| import ( | |
| 	"fmt" | |
| 	"time" | |
| 
 | |
| 	"github.com/seaweedfs/seaweedfs/weed/filer" | |
| 	"github.com/seaweedfs/seaweedfs/weed/filer_client" | |
| 	"github.com/seaweedfs/seaweedfs/weed/pb/filer_pb" | |
| 	"github.com/seaweedfs/seaweedfs/weed/pb/schema_pb" | |
| 	"github.com/seaweedfs/seaweedfs/weed/util" | |
| ) | |
| 
 | |
| // FilerOffsetStorage implements OffsetStorage using SeaweedFS filer | |
| // Stores offset data as files in the same directory structure as SMQ | |
| // Path: /topics/{namespace}/{topic}/{version}/{partition}/checkpoint.offset | |
| // The namespace and topic are derived from the actual partition information | |
| type FilerOffsetStorage struct { | |
| 	filerClientAccessor *filer_client.FilerClientAccessor | |
| } | |
| 
 | |
| // NewFilerOffsetStorageWithAccessor creates a new filer-based offset storage using existing filer client accessor | |
| func NewFilerOffsetStorageWithAccessor(filerClientAccessor *filer_client.FilerClientAccessor) *FilerOffsetStorage { | |
| 	return &FilerOffsetStorage{ | |
| 		filerClientAccessor: filerClientAccessor, | |
| 	} | |
| } | |
| 
 | |
| // SaveCheckpoint saves the checkpoint for a partition | |
| // Stores as: /topics/{namespace}/{topic}/{version}/{partition}/checkpoint.offset | |
| func (f *FilerOffsetStorage) SaveCheckpoint(namespace, topicName string, partition *schema_pb.Partition, offset int64) error { | |
| 	partitionDir := f.getPartitionDir(namespace, topicName, partition) | |
| 	fileName := "checkpoint.offset" | |
| 
 | |
| 	// Use SMQ's 8-byte offset format | |
| 	offsetBytes := make([]byte, 8) | |
| 	util.Uint64toBytes(offsetBytes, uint64(offset)) | |
| 
 | |
| 	return f.filerClientAccessor.WithFilerClient(false, func(client filer_pb.SeaweedFilerClient) error { | |
| 		return filer.SaveInsideFiler(client, partitionDir, fileName, offsetBytes) | |
| 	}) | |
| } | |
| 
 | |
| // LoadCheckpoint loads the checkpoint for a partition | |
| func (f *FilerOffsetStorage) LoadCheckpoint(namespace, topicName string, partition *schema_pb.Partition) (int64, error) { | |
| 	partitionDir := f.getPartitionDir(namespace, topicName, partition) | |
| 	fileName := "checkpoint.offset" | |
| 
 | |
| 	var offset int64 = -1 | |
| 	err := f.filerClientAccessor.WithFilerClient(false, func(client filer_pb.SeaweedFilerClient) error { | |
| 		data, err := filer.ReadInsideFiler(client, partitionDir, fileName) | |
| 		if err != nil { | |
| 			return err | |
| 		} | |
| 		if len(data) != 8 { | |
| 			return fmt.Errorf("invalid checkpoint file format: expected 8 bytes, got %d", len(data)) | |
| 		} | |
| 		offset = int64(util.BytesToUint64(data)) | |
| 		return nil | |
| 	}) | |
| 
 | |
| 	if err != nil { | |
| 		return -1, err | |
| 	} | |
| 
 | |
| 	return offset, nil | |
| } | |
| 
 | |
| // GetHighestOffset returns the highest offset stored for a partition | |
| // For filer storage, this is the same as the checkpoint since we don't store individual records | |
| func (f *FilerOffsetStorage) GetHighestOffset(namespace, topicName string, partition *schema_pb.Partition) (int64, error) { | |
| 	return f.LoadCheckpoint(namespace, topicName, partition) | |
| } | |
| 
 | |
| // Reset clears all data for testing | |
| func (f *FilerOffsetStorage) Reset() error { | |
| 	// For testing, we could delete all offset files, but this is dangerous | |
| 	// Instead, just return success - individual tests should clean up their own data | |
| 	return nil | |
| } | |
| 
 | |
| // Helper methods | |
|  | |
| // getPartitionDir returns the directory path for a partition following SMQ convention | |
| // Format: /topics/{namespace}/{topic}/{version}/{partition} | |
| func (f *FilerOffsetStorage) getPartitionDir(namespace, topicName string, partition *schema_pb.Partition) string { | |
| 	// Generate version from UnixTimeNs | |
| 	version := time.Unix(0, partition.UnixTimeNs).UTC().Format("v2006-01-02-15-04-05") | |
| 
 | |
| 	// Generate partition range string | |
| 	partitionRange := fmt.Sprintf("%04d-%04d", partition.RangeStart, partition.RangeStop) | |
| 
 | |
| 	return fmt.Sprintf("%s/%s/%s/%s/%s", filer.TopicsDir, namespace, topicName, version, partitionRange) | |
| } | |
| 
 | |
| // getPartitionKey generates a unique key for a partition | |
| func (f *FilerOffsetStorage) getPartitionKey(partition *schema_pb.Partition) string { | |
| 	return fmt.Sprintf("ring:%d:range:%d-%d:time:%d", | |
| 		partition.RingSize, partition.RangeStart, partition.RangeStop, partition.UnixTimeNs) | |
| }
 |