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.
		
		
		
		
		
			
		
			
				
					
					
						
							140 lines
						
					
					
						
							4.1 KiB
						
					
					
				
			
		
		
		
			
			
			
		
		
	
	
							140 lines
						
					
					
						
							4.1 KiB
						
					
					
				| package shell | |
| 
 | |
| import ( | |
| 	"context" | |
| 	"flag" | |
| 	"fmt" | |
| 	"io" | |
| 	"strings" | |
| 
 | |
| 	"github.com/seaweedfs/seaweedfs/weed/mq/topic" | |
| 	"github.com/seaweedfs/seaweedfs/weed/pb/filer_pb" | |
| 	"github.com/seaweedfs/seaweedfs/weed/util" | |
| ) | |
| 
 | |
| func init() { | |
| 	Commands = append(Commands, &commandMqTopicTruncate{}) | |
| } | |
| 
 | |
| type commandMqTopicTruncate struct { | |
| } | |
| 
 | |
| func (c *commandMqTopicTruncate) Name() string { | |
| 	return "mq.topic.truncate" | |
| } | |
| 
 | |
| func (c *commandMqTopicTruncate) Help() string { | |
| 	return `clear all data from a topic while preserving topic structure | |
|  | |
| 	Example: | |
| 		mq.topic.truncate -namespace <namespace> -topic <topic_name> | |
|  | |
| 	This command removes all log files and parquet files from all partitions | |
| 	of the specified topic, while keeping the topic configuration intact. | |
| ` | |
| } | |
| 
 | |
| func (c *commandMqTopicTruncate) HasTag(CommandTag) bool { | |
| 	return false | |
| } | |
| 
 | |
| func (c *commandMqTopicTruncate) Do(args []string, commandEnv *CommandEnv, writer io.Writer) error { | |
| 	// parse parameters | |
| 	mqCommand := flag.NewFlagSet(c.Name(), flag.ContinueOnError) | |
| 	namespace := mqCommand.String("namespace", "", "namespace name") | |
| 	topicName := mqCommand.String("topic", "", "topic name") | |
| 	if err := mqCommand.Parse(args); err != nil { | |
| 		return err | |
| 	} | |
| 
 | |
| 	if *namespace == "" { | |
| 		return fmt.Errorf("namespace is required") | |
| 	} | |
| 	if *topicName == "" { | |
| 		return fmt.Errorf("topic name is required") | |
| 	} | |
| 
 | |
| 	// Verify topic exists by trying to read its configuration | |
| 	t := topic.NewTopic(*namespace, *topicName) | |
| 
 | |
| 	err := commandEnv.WithFilerClient(false, func(client filer_pb.SeaweedFilerClient) error { | |
| 		_, err := t.ReadConfFile(client) | |
| 		if err != nil { | |
| 			return fmt.Errorf("topic %s.%s does not exist or cannot be read: %v", *namespace, *topicName, err) | |
| 		} | |
| 		return nil | |
| 	}) | |
| 	if err != nil { | |
| 		return err | |
| 	} | |
| 
 | |
| 	fmt.Fprintf(writer, "Truncating topic %s.%s...\n", *namespace, *topicName) | |
| 
 | |
| 	// Discover and clear all partitions using centralized logic | |
| 	partitions, err := t.DiscoverPartitions(context.Background(), commandEnv) | |
| 	if err != nil { | |
| 		return fmt.Errorf("failed to discover topic partitions: %v", err) | |
| 	} | |
| 
 | |
| 	if len(partitions) == 0 { | |
| 		fmt.Fprintf(writer, "No partitions found for topic %s.%s\n", *namespace, *topicName) | |
| 		return nil | |
| 	} | |
| 
 | |
| 	fmt.Fprintf(writer, "Found %d partitions, clearing data...\n", len(partitions)) | |
| 
 | |
| 	// Clear data from each partition | |
| 	totalFilesDeleted := 0 | |
| 	for _, partitionPath := range partitions { | |
| 		filesDeleted, err := c.clearPartitionData(commandEnv, partitionPath, writer) | |
| 		if err != nil { | |
| 			fmt.Fprintf(writer, "Warning: failed to clear partition %s: %v\n", partitionPath, err) | |
| 			continue | |
| 		} | |
| 		totalFilesDeleted += filesDeleted | |
| 		fmt.Fprintf(writer, "Cleared partition: %s (%d files)\n", partitionPath, filesDeleted) | |
| 	} | |
| 
 | |
| 	fmt.Fprintf(writer, "Successfully truncated topic %s.%s - deleted %d files from %d partitions\n", | |
| 		*namespace, *topicName, totalFilesDeleted, len(partitions)) | |
| 
 | |
| 	return nil | |
| } | |
| 
 | |
| // clearPartitionData deletes all data files (log files, parquet files) from a partition directory | |
| // Returns the number of files deleted | |
| func (c *commandMqTopicTruncate) clearPartitionData(commandEnv *CommandEnv, partitionPath string, writer io.Writer) (int, error) { | |
| 	filesDeleted := 0 | |
| 
 | |
| 	err := filer_pb.ReadDirAllEntries(context.Background(), commandEnv, util.FullPath(partitionPath), "", func(entry *filer_pb.Entry, isLast bool) error { | |
| 		if entry.IsDirectory { | |
| 			return nil // Skip subdirectories | |
| 		} | |
| 
 | |
| 		fileName := entry.Name | |
| 
 | |
| 		// Preserve configuration files | |
| 		if strings.HasSuffix(fileName, ".conf") || | |
| 			strings.HasSuffix(fileName, ".config") || | |
| 			fileName == "topic.conf" || | |
| 			fileName == "partition.conf" { | |
| 			fmt.Fprintf(writer, "  Preserving config file: %s\n", fileName) | |
| 			return nil | |
| 		} | |
| 
 | |
| 		// Delete all data files (log files, parquet files, offset files, etc.) | |
| 		deleteErr := filer_pb.Remove(context.Background(), commandEnv, partitionPath, fileName, false, true, true, false, nil) | |
| 
 | |
| 		if deleteErr != nil { | |
| 			fmt.Fprintf(writer, "  Warning: failed to delete %s/%s: %v\n", partitionPath, fileName, deleteErr) | |
| 			// Continue with other files rather than failing entirely | |
| 		} else { | |
| 			fmt.Fprintf(writer, "  Deleted: %s\n", fileName) | |
| 			filesDeleted++ | |
| 		} | |
| 
 | |
| 		return nil | |
| 	}) | |
| 
 | |
| 	return filesDeleted, err | |
| }
 |