439 lines
10 KiB
Go
439 lines
10 KiB
Go
|
package nixstore
|
||
|
|
||
|
import (
|
||
|
"log"
|
||
|
"sync"
|
||
|
"time"
|
||
|
)
|
||
|
|
||
|
const (
|
||
|
progressLogRateLimit = 1 * time.Second
|
||
|
)
|
||
|
|
||
|
type ActivityType uint64
|
||
|
|
||
|
const (
|
||
|
ActUnknown ActivityType = 0
|
||
|
ActCopyPath ActivityType = 99 + iota
|
||
|
ActDownload
|
||
|
ActRealise
|
||
|
ActCopyPaths
|
||
|
ActBuilds
|
||
|
ActBuild
|
||
|
ActOptimiseStore
|
||
|
ActVerifyPaths
|
||
|
ActSubstitute
|
||
|
ActQueryPathInfo
|
||
|
ActPostBuildHook
|
||
|
)
|
||
|
|
||
|
func (at ActivityType) String() string {
|
||
|
return map[ActivityType]string{
|
||
|
ActUnknown: "unknown",
|
||
|
ActCopyPath: "copying path",
|
||
|
ActDownload: "downloading path",
|
||
|
ActRealise: "realising path",
|
||
|
ActCopyPaths: "copying paths",
|
||
|
ActBuilds: "running builds",
|
||
|
ActBuild: "building",
|
||
|
ActOptimiseStore: "optimising store",
|
||
|
ActVerifyPaths: "verifying paths",
|
||
|
ActSubstitute: "substituting path", // paths?
|
||
|
ActQueryPathInfo: "querying path info",
|
||
|
ActPostBuildHook: "running post-build hook",
|
||
|
}[at]
|
||
|
}
|
||
|
|
||
|
type ResultType uint64
|
||
|
|
||
|
const (
|
||
|
ResFileLinked ResultType = 100 + iota
|
||
|
ResBuildLogLine
|
||
|
ResUntrustedPath
|
||
|
ResCorruptedPath
|
||
|
ResSetPhase
|
||
|
ResProgress
|
||
|
ResSetExpected
|
||
|
ResPostBuildLogLine
|
||
|
)
|
||
|
|
||
|
func (rt ResultType) String() string {
|
||
|
return map[ResultType]string{
|
||
|
ResFileLinked: "linked file", // [size: int, blocks: int]
|
||
|
ResBuildLogLine: "log line from build", // [log line: str]
|
||
|
ResUntrustedPath: "untrusted path", // [path: str]
|
||
|
ResCorruptedPath: "corrupted path", // [path: str]
|
||
|
ResSetPhase: "entering phase", // [phase: str]
|
||
|
ResProgress: "progress update", // [done: int, expected: int, running: int, failed: int]
|
||
|
ResSetExpected: "update expectation", // [activityType: int, expected: int]
|
||
|
ResPostBuildLogLine: "log line from post-build", // [log line: str]
|
||
|
}[rt]
|
||
|
}
|
||
|
|
||
|
type ActivityTracker struct {
|
||
|
actions []*ActivityLogger
|
||
|
mu sync.Mutex
|
||
|
}
|
||
|
|
||
|
func NewActivityTracker() *ActivityTracker {
|
||
|
return &ActivityTracker{}
|
||
|
}
|
||
|
|
||
|
type ActivityLogger struct {
|
||
|
// Static for lifetime of logger.
|
||
|
action string
|
||
|
at *ActivityTracker
|
||
|
|
||
|
// Guarded by mu.
|
||
|
activities map[uint64]Activity
|
||
|
logs []ActivityMetaLog
|
||
|
mu sync.Mutex
|
||
|
}
|
||
|
|
||
|
func (al *ActivityLogger) AddLog(message string) {
|
||
|
if al == nil {
|
||
|
log.Println(message)
|
||
|
return
|
||
|
}
|
||
|
|
||
|
al.mu.Lock()
|
||
|
al.logs = append(al.logs, ActivityMetaLog{
|
||
|
Timestamp: time.Now(),
|
||
|
LogLine: message,
|
||
|
})
|
||
|
al.mu.Unlock()
|
||
|
}
|
||
|
|
||
|
func (al *ActivityLogger) AddError(errorCode uint64, message string) {
|
||
|
if al == nil {
|
||
|
log.Printf("error %d: %v", errorCode, message)
|
||
|
return
|
||
|
}
|
||
|
|
||
|
al.mu.Lock()
|
||
|
al.logs = append(al.logs, ActivityMetaLog{
|
||
|
Timestamp: time.Now(),
|
||
|
ErrorCode: errorCode,
|
||
|
ErrorMessage: message,
|
||
|
})
|
||
|
al.mu.Unlock()
|
||
|
}
|
||
|
|
||
|
type Activity interface {
|
||
|
ActivityType() ActivityType
|
||
|
Meta() *ActivityMeta
|
||
|
}
|
||
|
|
||
|
type ActivityMetaLog struct {
|
||
|
Timestamp time.Time
|
||
|
|
||
|
// One of:
|
||
|
// Line
|
||
|
LogLine string
|
||
|
|
||
|
// Error
|
||
|
ErrorCode uint64
|
||
|
ErrorMessage string
|
||
|
|
||
|
// Started
|
||
|
ActivityStarted Activity
|
||
|
|
||
|
// Finished
|
||
|
ActivityFinished Activity
|
||
|
}
|
||
|
|
||
|
type LogLineType int
|
||
|
|
||
|
const (
|
||
|
LogLineBuild LogLineType = iota
|
||
|
LogLinePostBuild
|
||
|
)
|
||
|
|
||
|
type ActivityLog struct {
|
||
|
Timestamp time.Time
|
||
|
|
||
|
// One of:
|
||
|
LogLineType LogLineType
|
||
|
LogLine string
|
||
|
|
||
|
NewPhase string
|
||
|
|
||
|
Progress []any
|
||
|
|
||
|
Expected []any
|
||
|
}
|
||
|
|
||
|
type ActivityMeta struct {
|
||
|
ActivityID uint64
|
||
|
Level uint64
|
||
|
String string
|
||
|
Fields []any
|
||
|
ParentActivityID uint64
|
||
|
|
||
|
mu sync.Mutex
|
||
|
logs []*ActivityLog
|
||
|
phase string
|
||
|
progress []any
|
||
|
firstProgressLog *ActivityLog
|
||
|
lastProgressLogCreated time.Time
|
||
|
lastProgressLog *ActivityLog
|
||
|
expected []any
|
||
|
}
|
||
|
|
||
|
func (am *ActivityMeta) RecordResult(result ResultType, fields []any) {
|
||
|
if am == nil {
|
||
|
return
|
||
|
}
|
||
|
now := time.Now()
|
||
|
am.mu.Lock()
|
||
|
switch result {
|
||
|
case ResBuildLogLine, ResPostBuildLogLine:
|
||
|
var llt LogLineType
|
||
|
if result == ResBuildLogLine {
|
||
|
llt = LogLineBuild
|
||
|
} else {
|
||
|
llt = LogLinePostBuild
|
||
|
}
|
||
|
am.logs = append(am.logs, &ActivityLog{
|
||
|
Timestamp: now,
|
||
|
LogLineType: llt,
|
||
|
LogLine: fields[0].(string),
|
||
|
})
|
||
|
case ResSetPhase:
|
||
|
phase := fields[0].(string)
|
||
|
am.logs = append(am.logs, &ActivityLog{
|
||
|
Timestamp: now,
|
||
|
NewPhase: phase,
|
||
|
})
|
||
|
am.phase = phase
|
||
|
case ResProgress:
|
||
|
am.progress = fields
|
||
|
|
||
|
if am.firstProgressLog == nil {
|
||
|
// If we've never logged progress before, log it.
|
||
|
log := &ActivityLog{
|
||
|
Timestamp: now,
|
||
|
Progress: fields,
|
||
|
}
|
||
|
am.logs = append(am.logs, log)
|
||
|
am.firstProgressLog = log
|
||
|
} else if am.lastProgressLog == nil {
|
||
|
// If we don't have a tail log entry, log it.
|
||
|
log := &ActivityLog{
|
||
|
Timestamp: now,
|
||
|
Progress: fields,
|
||
|
}
|
||
|
am.logs = append(am.logs, log)
|
||
|
am.lastProgressLog = log
|
||
|
am.lastProgressLogCreated = now
|
||
|
} else {
|
||
|
am.lastProgressLog.Timestamp = now
|
||
|
am.lastProgressLog.Progress = fields
|
||
|
if time.Since(am.lastProgressLogCreated) > progressLogRateLimit {
|
||
|
// If our last progress log was more than rate limit ago, bump the log entry forwards and mark that we should create a new one.
|
||
|
am.lastProgressLog = nil
|
||
|
}
|
||
|
}
|
||
|
case ResSetExpected:
|
||
|
am.expected = fields
|
||
|
am.logs = append(am.logs, &ActivityLog{
|
||
|
Timestamp: now,
|
||
|
Expected: fields,
|
||
|
})
|
||
|
}
|
||
|
am.mu.Unlock()
|
||
|
}
|
||
|
|
||
|
var (
|
||
|
activityRegistry = map[ActivityType]func(ActivityMeta) Activity{}
|
||
|
)
|
||
|
|
||
|
func registerActivityType(a Activity, factory func(ActivityMeta) Activity) {
|
||
|
activityRegistry[a.ActivityType()] = factory
|
||
|
}
|
||
|
|
||
|
type CopyPathActivity struct{ ActivityMeta }
|
||
|
|
||
|
func (*CopyPathActivity) ActivityType() ActivityType { return ActCopyPath }
|
||
|
func (a *CopyPathActivity) Meta() *ActivityMeta { return &a.ActivityMeta }
|
||
|
|
||
|
func init() {
|
||
|
registerActivityType(&CopyPathActivity{}, func(am ActivityMeta) Activity { return &CopyPathActivity{am} })
|
||
|
}
|
||
|
|
||
|
type DownloadActivity struct{ ActivityMeta }
|
||
|
|
||
|
func (*DownloadActivity) ActivityType() ActivityType { return ActDownload }
|
||
|
func (a *DownloadActivity) Meta() *ActivityMeta { return &a.ActivityMeta }
|
||
|
|
||
|
func init() {
|
||
|
registerActivityType(&DownloadActivity{}, func(am ActivityMeta) Activity { return &DownloadActivity{am} })
|
||
|
}
|
||
|
|
||
|
type RealiseActivity struct{ ActivityMeta }
|
||
|
|
||
|
func (*RealiseActivity) ActivityType() ActivityType { return ActRealise }
|
||
|
func (a *RealiseActivity) Meta() *ActivityMeta { return &a.ActivityMeta }
|
||
|
|
||
|
func init() {
|
||
|
registerActivityType(&RealiseActivity{}, func(am ActivityMeta) Activity { return &RealiseActivity{am} })
|
||
|
}
|
||
|
|
||
|
type BuildsActivity struct{ ActivityMeta }
|
||
|
|
||
|
func (*BuildsActivity) ActivityType() ActivityType { return ActBuilds }
|
||
|
func (a *BuildsActivity) Meta() *ActivityMeta { return &a.ActivityMeta }
|
||
|
|
||
|
func init() {
|
||
|
registerActivityType(&BuildsActivity{}, func(am ActivityMeta) Activity { return &BuildsActivity{am} })
|
||
|
}
|
||
|
|
||
|
type BuildActivity struct{ ActivityMeta }
|
||
|
|
||
|
func (*BuildActivity) ActivityType() ActivityType { return ActBuild }
|
||
|
func (a *BuildActivity) Meta() *ActivityMeta { return &a.ActivityMeta }
|
||
|
|
||
|
func init() {
|
||
|
registerActivityType(&BuildActivity{}, func(am ActivityMeta) Activity { return &BuildActivity{am} })
|
||
|
}
|
||
|
|
||
|
type OptimiseStoreActivity struct{ ActivityMeta }
|
||
|
|
||
|
func (*OptimiseStoreActivity) ActivityType() ActivityType { return ActOptimiseStore }
|
||
|
func (a *OptimiseStoreActivity) Meta() *ActivityMeta { return &a.ActivityMeta }
|
||
|
|
||
|
func init() {
|
||
|
registerActivityType(&OptimiseStoreActivity{}, func(am ActivityMeta) Activity { return &OptimiseStoreActivity{am} })
|
||
|
}
|
||
|
|
||
|
type VerifyPathsActivity struct{ ActivityMeta }
|
||
|
|
||
|
func (*VerifyPathsActivity) ActivityType() ActivityType { return ActVerifyPaths }
|
||
|
func (a *VerifyPathsActivity) Meta() *ActivityMeta { return &a.ActivityMeta }
|
||
|
|
||
|
func init() {
|
||
|
registerActivityType(&VerifyPathsActivity{}, func(am ActivityMeta) Activity { return &VerifyPathsActivity{am} })
|
||
|
}
|
||
|
|
||
|
type SubstituteActivity struct{ ActivityMeta }
|
||
|
|
||
|
func (*SubstituteActivity) ActivityType() ActivityType { return ActSubstitute }
|
||
|
func (a *SubstituteActivity) Meta() *ActivityMeta { return &a.ActivityMeta }
|
||
|
|
||
|
func init() {
|
||
|
registerActivityType(&SubstituteActivity{}, func(am ActivityMeta) Activity { return &SubstituteActivity{am} })
|
||
|
}
|
||
|
|
||
|
type QueryPathInfoActivity struct{ ActivityMeta }
|
||
|
|
||
|
func (*QueryPathInfoActivity) ActivityType() ActivityType { return ActQueryPathInfo }
|
||
|
func (a *QueryPathInfoActivity) Meta() *ActivityMeta { return &a.ActivityMeta }
|
||
|
|
||
|
func init() {
|
||
|
registerActivityType(&QueryPathInfoActivity{}, func(am ActivityMeta) Activity { return &QueryPathInfoActivity{am} })
|
||
|
}
|
||
|
|
||
|
type PostBuildHookActivity struct{ ActivityMeta }
|
||
|
|
||
|
func (*PostBuildHookActivity) ActivityType() ActivityType { return ActPostBuildHook }
|
||
|
func (a *PostBuildHookActivity) Meta() *ActivityMeta { return &a.ActivityMeta }
|
||
|
|
||
|
func init() {
|
||
|
registerActivityType(&PostBuildHookActivity{}, func(am ActivityMeta) Activity { return &PostBuildHookActivity{am} })
|
||
|
}
|
||
|
|
||
|
func (at *ActivityTracker) StartAction(actionName string) *ActivityLogger {
|
||
|
if at == nil {
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
al := &ActivityLogger{
|
||
|
action: actionName,
|
||
|
at: at,
|
||
|
activities: make(map[uint64]Activity),
|
||
|
}
|
||
|
at.mu.Lock()
|
||
|
at.actions = append(at.actions, al)
|
||
|
at.mu.Unlock()
|
||
|
return al
|
||
|
}
|
||
|
|
||
|
func (at *ActivityTracker) EndAction(al *ActivityLogger) {
|
||
|
if at == nil {
|
||
|
return
|
||
|
}
|
||
|
|
||
|
at.mu.Lock()
|
||
|
newActions := make([]*ActivityLogger, 0, len(at.actions)-1)
|
||
|
for _, oal := range at.actions {
|
||
|
if oal != al {
|
||
|
newActions = append(newActions, oal)
|
||
|
}
|
||
|
}
|
||
|
at.actions = newActions
|
||
|
at.mu.Unlock()
|
||
|
}
|
||
|
|
||
|
func (al *ActivityLogger) Close() {
|
||
|
if al == nil {
|
||
|
return
|
||
|
}
|
||
|
|
||
|
al.at.EndAction(al)
|
||
|
}
|
||
|
|
||
|
func (al *ActivityLogger) StartActivity(at ActivityType, am ActivityMeta) Activity {
|
||
|
if al == nil {
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
af, ok := activityRegistry[at]
|
||
|
if !ok {
|
||
|
return nil
|
||
|
}
|
||
|
a := af(am)
|
||
|
|
||
|
al.mu.Lock()
|
||
|
al.logs = append(al.logs, ActivityMetaLog{
|
||
|
Timestamp: time.Now(),
|
||
|
ActivityStarted: a,
|
||
|
})
|
||
|
al.activities[am.ActivityID] = a
|
||
|
al.mu.Unlock()
|
||
|
|
||
|
return a
|
||
|
}
|
||
|
|
||
|
func (al *ActivityLogger) Activity(activityID uint64) Activity {
|
||
|
if al == nil {
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
var a Activity
|
||
|
al.mu.Lock()
|
||
|
a = al.activities[activityID]
|
||
|
al.mu.Unlock()
|
||
|
return a
|
||
|
}
|
||
|
|
||
|
func (al *ActivityLogger) ActivityResult(a Activity, resultType ResultType, data []any) {
|
||
|
if a == nil || al == nil {
|
||
|
return
|
||
|
}
|
||
|
|
||
|
a.Meta().RecordResult(resultType, data)
|
||
|
}
|
||
|
|
||
|
func (al *ActivityLogger) EndActivity(a Activity) {
|
||
|
if al == nil || a == nil {
|
||
|
return
|
||
|
}
|
||
|
|
||
|
al.mu.Lock()
|
||
|
al.logs = append(al.logs, ActivityMetaLog{
|
||
|
Timestamp: time.Now(),
|
||
|
ActivityFinished: a,
|
||
|
})
|
||
|
al.mu.Unlock()
|
||
|
}
|