| 
									
										
										
										
											2020-04-04 16:45:26 +08:00
										 |  |  | // Copyright 2020 The Gitea Authors. All rights reserved. | 
					
						
							|  |  |  | // Copyright (c) 2015, Wade Simmons | 
					
						
							| 
									
										
										
										
											2022-11-27 13:20:29 -05:00
										 |  |  | // SPDX-License-Identifier: MIT | 
					
						
							| 
									
										
										
										
											2020-04-04 16:45:26 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | // gocovmerge takes the results from multiple `go test -coverprofile` runs and | 
					
						
							|  |  |  | // merges them into one profile | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-24 11:47:09 -05:00
										 |  |  | //go:build ignore | 
					
						
							| 
									
										
										
										
											2020-04-04 16:45:26 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | package main | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							|  |  |  | 	"flag" | 
					
						
							|  |  |  | 	"fmt" | 
					
						
							|  |  |  | 	"io" | 
					
						
							|  |  |  | 	"log" | 
					
						
							|  |  |  | 	"os" | 
					
						
							|  |  |  | 	"sort" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	"golang.org/x/tools/cover" | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-20 18:46:10 +01:00
										 |  |  | func mergeProfiles(p, merge *cover.Profile) { | 
					
						
							| 
									
										
										
										
											2020-04-04 16:45:26 +08:00
										 |  |  | 	if p.Mode != merge.Mode { | 
					
						
							|  |  |  | 		log.Fatalf("cannot merge profiles with different modes") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	// Since the blocks are sorted, we can keep track of where the last block | 
					
						
							|  |  |  | 	// was inserted and only look at the blocks after that as targets for merge | 
					
						
							|  |  |  | 	startIndex := 0 | 
					
						
							|  |  |  | 	for _, b := range merge.Blocks { | 
					
						
							|  |  |  | 		startIndex = mergeProfileBlock(p, b, startIndex) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func mergeProfileBlock(p *cover.Profile, pb cover.ProfileBlock, startIndex int) int { | 
					
						
							|  |  |  | 	sortFunc := func(i int) bool { | 
					
						
							|  |  |  | 		pi := p.Blocks[i+startIndex] | 
					
						
							|  |  |  | 		return pi.StartLine >= pb.StartLine && (pi.StartLine != pb.StartLine || pi.StartCol >= pb.StartCol) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	i := 0 | 
					
						
							|  |  |  | 	if sortFunc(i) != true { | 
					
						
							|  |  |  | 		i = sort.Search(len(p.Blocks)-startIndex, sortFunc) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	i += startIndex | 
					
						
							|  |  |  | 	if i < len(p.Blocks) && p.Blocks[i].StartLine == pb.StartLine && p.Blocks[i].StartCol == pb.StartCol { | 
					
						
							|  |  |  | 		if p.Blocks[i].EndLine != pb.EndLine || p.Blocks[i].EndCol != pb.EndCol { | 
					
						
							|  |  |  | 			log.Fatalf("OVERLAP MERGE: %v %v %v", p.FileName, p.Blocks[i], pb) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		switch p.Mode { | 
					
						
							|  |  |  | 		case "set": | 
					
						
							|  |  |  | 			p.Blocks[i].Count |= pb.Count | 
					
						
							|  |  |  | 		case "count", "atomic": | 
					
						
							|  |  |  | 			p.Blocks[i].Count += pb.Count | 
					
						
							|  |  |  | 		default: | 
					
						
							|  |  |  | 			log.Fatalf("unsupported covermode: '%s'", p.Mode) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		if i > 0 { | 
					
						
							|  |  |  | 			pa := p.Blocks[i-1] | 
					
						
							|  |  |  | 			if pa.EndLine >= pb.EndLine && (pa.EndLine != pb.EndLine || pa.EndCol > pb.EndCol) { | 
					
						
							|  |  |  | 				log.Fatalf("OVERLAP BEFORE: %v %v %v", p.FileName, pa, pb) | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if i < len(p.Blocks)-1 { | 
					
						
							|  |  |  | 			pa := p.Blocks[i+1] | 
					
						
							|  |  |  | 			if pa.StartLine <= pb.StartLine && (pa.StartLine != pb.StartLine || pa.StartCol < pb.StartCol) { | 
					
						
							|  |  |  | 				log.Fatalf("OVERLAP AFTER: %v %v %v", p.FileName, pa, pb) | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		p.Blocks = append(p.Blocks, cover.ProfileBlock{}) | 
					
						
							|  |  |  | 		copy(p.Blocks[i+1:], p.Blocks[i:]) | 
					
						
							|  |  |  | 		p.Blocks[i] = pb | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return i + 1 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func addProfile(profiles []*cover.Profile, p *cover.Profile) []*cover.Profile { | 
					
						
							|  |  |  | 	i := sort.Search(len(profiles), func(i int) bool { return profiles[i].FileName >= p.FileName }) | 
					
						
							|  |  |  | 	if i < len(profiles) && profiles[i].FileName == p.FileName { | 
					
						
							|  |  |  | 		mergeProfiles(profiles[i], p) | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		profiles = append(profiles, nil) | 
					
						
							|  |  |  | 		copy(profiles[i+1:], profiles[i:]) | 
					
						
							|  |  |  | 		profiles[i] = p | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return profiles | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func dumpProfiles(profiles []*cover.Profile, out io.Writer) { | 
					
						
							|  |  |  | 	if len(profiles) == 0 { | 
					
						
							|  |  |  | 		return | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	fmt.Fprintf(out, "mode: %s\n", profiles[0].Mode) | 
					
						
							|  |  |  | 	for _, p := range profiles { | 
					
						
							|  |  |  | 		for _, b := range p.Blocks { | 
					
						
							|  |  |  | 			fmt.Fprintf(out, "%s:%d.%d,%d.%d %d %d\n", p.FileName, b.StartLine, b.StartCol, b.EndLine, b.EndCol, b.NumStmt, b.Count) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func main() { | 
					
						
							|  |  |  | 	flag.Parse() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	var merged []*cover.Profile | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for _, file := range flag.Args() { | 
					
						
							|  |  |  | 		profiles, err := cover.ParseProfiles(file) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							| 
									
										
										
										
											2021-08-26 16:50:04 +08:00
										 |  |  | 			log.Fatalf("failed to parse profile '%s': %v", file, err) | 
					
						
							| 
									
										
										
										
											2020-04-04 16:45:26 +08:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		for _, p := range profiles { | 
					
						
							|  |  |  | 			merged = addProfile(merged, p) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	dumpProfiles(merged, os.Stdout) | 
					
						
							|  |  |  | } |