| 
									
										
										
										
											2022-10-28 17:44:34 +02:00
										 |  |  | package restore | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							|  |  |  | 	"sync" | 
					
						
							|  |  |  | 	"time" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	"github.com/restic/restic/internal/ui/progress" | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-31 13:43:57 +02:00
										 |  |  | type State struct { | 
					
						
							|  |  |  | 	FilesFinished   uint64 | 
					
						
							|  |  |  | 	FilesTotal      uint64 | 
					
						
							|  |  |  | 	AllBytesWritten uint64 | 
					
						
							|  |  |  | 	AllBytesTotal   uint64 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-28 17:44:34 +02:00
										 |  |  | type Progress struct { | 
					
						
							|  |  |  | 	updater progress.Updater | 
					
						
							|  |  |  | 	m       sync.Mutex | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	progressInfoMap map[string]progressInfoEntry | 
					
						
							| 
									
										
										
										
											2024-05-31 13:43:57 +02:00
										 |  |  | 	s               State | 
					
						
							| 
									
										
										
										
											2022-10-28 17:44:34 +02:00
										 |  |  | 	started         time.Time | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	printer ProgressPrinter | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | type progressInfoEntry struct { | 
					
						
							|  |  |  | 	bytesWritten uint64 | 
					
						
							|  |  |  | 	bytesTotal   uint64 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-01 11:19:09 +02:00
										 |  |  | type term interface { | 
					
						
							|  |  |  | 	Print(line string) | 
					
						
							|  |  |  | 	SetStatus(lines []string) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-28 17:44:34 +02:00
										 |  |  | type ProgressPrinter interface { | 
					
						
							| 
									
										
										
										
											2024-05-31 13:43:57 +02:00
										 |  |  | 	Update(progress State, duration time.Duration) | 
					
						
							|  |  |  | 	Finish(progress State, duration time.Duration) | 
					
						
							| 
									
										
										
										
											2022-10-28 17:44:34 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func NewProgress(printer ProgressPrinter, interval time.Duration) *Progress { | 
					
						
							|  |  |  | 	p := &Progress{ | 
					
						
							|  |  |  | 		progressInfoMap: make(map[string]progressInfoEntry), | 
					
						
							|  |  |  | 		started:         time.Now(), | 
					
						
							|  |  |  | 		printer:         printer, | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	p.updater = *progress.NewUpdater(interval, p.update) | 
					
						
							|  |  |  | 	return p | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (p *Progress) update(runtime time.Duration, final bool) { | 
					
						
							|  |  |  | 	p.m.Lock() | 
					
						
							|  |  |  | 	defer p.m.Unlock() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if !final { | 
					
						
							| 
									
										
										
										
											2024-05-31 13:43:57 +02:00
										 |  |  | 		p.printer.Update(p.s, runtime) | 
					
						
							| 
									
										
										
										
											2022-10-28 17:44:34 +02:00
										 |  |  | 	} else { | 
					
						
							| 
									
										
										
										
											2024-05-31 13:43:57 +02:00
										 |  |  | 		p.printer.Finish(p.s, runtime) | 
					
						
							| 
									
										
										
										
											2022-10-28 17:44:34 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // AddFile starts tracking a new file with the given size | 
					
						
							|  |  |  | func (p *Progress) AddFile(size uint64) { | 
					
						
							| 
									
										
										
										
											2024-05-31 11:07:53 +02:00
										 |  |  | 	if p == nil { | 
					
						
							|  |  |  | 		return | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-28 17:44:34 +02:00
										 |  |  | 	p.m.Lock() | 
					
						
							|  |  |  | 	defer p.m.Unlock() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-31 13:43:57 +02:00
										 |  |  | 	p.s.FilesTotal++ | 
					
						
							|  |  |  | 	p.s.AllBytesTotal += size | 
					
						
							| 
									
										
										
										
											2022-10-28 17:44:34 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // AddProgress accumulates the number of bytes written for a file | 
					
						
							|  |  |  | func (p *Progress) AddProgress(name string, bytesWrittenPortion uint64, bytesTotal uint64) { | 
					
						
							| 
									
										
										
										
											2024-05-31 11:07:53 +02:00
										 |  |  | 	if p == nil { | 
					
						
							|  |  |  | 		return | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-28 17:44:34 +02:00
										 |  |  | 	p.m.Lock() | 
					
						
							|  |  |  | 	defer p.m.Unlock() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	entry, exists := p.progressInfoMap[name] | 
					
						
							|  |  |  | 	if !exists { | 
					
						
							|  |  |  | 		entry.bytesTotal = bytesTotal | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	entry.bytesWritten += bytesWrittenPortion | 
					
						
							|  |  |  | 	p.progressInfoMap[name] = entry | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-31 13:43:57 +02:00
										 |  |  | 	p.s.AllBytesWritten += bytesWrittenPortion | 
					
						
							| 
									
										
										
										
											2022-10-28 17:44:34 +02:00
										 |  |  | 	if entry.bytesWritten == entry.bytesTotal { | 
					
						
							|  |  |  | 		delete(p.progressInfoMap, name) | 
					
						
							| 
									
										
										
										
											2024-05-31 13:43:57 +02:00
										 |  |  | 		p.s.FilesFinished++ | 
					
						
							| 
									
										
										
										
											2022-10-28 17:44:34 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (p *Progress) Finish() { | 
					
						
							|  |  |  | 	p.updater.Done() | 
					
						
							|  |  |  | } |