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.
197 lines
6.0 KiB
197 lines
6.0 KiB
package plugin
|
|
|
|
import (
|
|
"context"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/seaweedfs/seaweedfs/weed/pb/plugin_pb"
|
|
)
|
|
|
|
func TestRunDetectionIncludesLatestSuccessfulRun(t *testing.T) {
|
|
pluginSvc, err := New(Options{})
|
|
if err != nil {
|
|
t.Fatalf("New plugin error: %v", err)
|
|
}
|
|
defer pluginSvc.Shutdown()
|
|
|
|
jobType := "vacuum"
|
|
pluginSvc.registry.UpsertFromHello(&plugin_pb.WorkerHello{
|
|
WorkerId: "worker-a",
|
|
Capabilities: []*plugin_pb.JobTypeCapability{
|
|
{JobType: jobType, CanDetect: true, MaxDetectionConcurrency: 1},
|
|
},
|
|
})
|
|
session := &streamSession{workerID: "worker-a", outgoing: make(chan *plugin_pb.AdminToWorkerMessage, 1)}
|
|
pluginSvc.putSession(session)
|
|
|
|
oldSuccess := time.Date(2026, 1, 1, 0, 0, 0, 0, time.UTC)
|
|
latestSuccess := time.Date(2026, 2, 1, 0, 0, 0, 0, time.UTC)
|
|
if err := pluginSvc.store.AppendRunRecord(jobType, &JobRunRecord{Outcome: RunOutcomeSuccess, CompletedAt: timeToPtr(oldSuccess)}); err != nil {
|
|
t.Fatalf("AppendRunRecord old success: %v", err)
|
|
}
|
|
if err := pluginSvc.store.AppendRunRecord(jobType, &JobRunRecord{Outcome: RunOutcomeError, CompletedAt: timeToPtr(latestSuccess.Add(2 * time.Hour))}); err != nil {
|
|
t.Fatalf("AppendRunRecord error run: %v", err)
|
|
}
|
|
if err := pluginSvc.store.AppendRunRecord(jobType, &JobRunRecord{Outcome: RunOutcomeSuccess, CompletedAt: timeToPtr(latestSuccess)}); err != nil {
|
|
t.Fatalf("AppendRunRecord latest success: %v", err)
|
|
}
|
|
|
|
resultCh := make(chan error, 1)
|
|
go func() {
|
|
_, runErr := pluginSvc.RunDetection(context.Background(), jobType, &plugin_pb.ClusterContext{}, 10)
|
|
resultCh <- runErr
|
|
}()
|
|
|
|
message := <-session.outgoing
|
|
detectRequest := message.GetRunDetectionRequest()
|
|
if detectRequest == nil {
|
|
t.Fatalf("expected run detection request message")
|
|
}
|
|
if detectRequest.LastSuccessfulRun == nil {
|
|
t.Fatalf("expected last_successful_run to be set")
|
|
}
|
|
if got := detectRequest.LastSuccessfulRun.AsTime().UTC(); !got.Equal(latestSuccess) {
|
|
t.Fatalf("unexpected last_successful_run, got=%s want=%s", got, latestSuccess)
|
|
}
|
|
|
|
pluginSvc.handleDetectionComplete("worker-a", &plugin_pb.DetectionComplete{
|
|
RequestId: message.RequestId,
|
|
JobType: jobType,
|
|
Success: true,
|
|
})
|
|
|
|
if runErr := <-resultCh; runErr != nil {
|
|
t.Fatalf("RunDetection error: %v", runErr)
|
|
}
|
|
}
|
|
|
|
func TestRunDetectionOmitsLastSuccessfulRunWhenNoSuccessHistory(t *testing.T) {
|
|
pluginSvc, err := New(Options{})
|
|
if err != nil {
|
|
t.Fatalf("New plugin error: %v", err)
|
|
}
|
|
defer pluginSvc.Shutdown()
|
|
|
|
jobType := "vacuum"
|
|
pluginSvc.registry.UpsertFromHello(&plugin_pb.WorkerHello{
|
|
WorkerId: "worker-a",
|
|
Capabilities: []*plugin_pb.JobTypeCapability{
|
|
{JobType: jobType, CanDetect: true, MaxDetectionConcurrency: 1},
|
|
},
|
|
})
|
|
session := &streamSession{workerID: "worker-a", outgoing: make(chan *plugin_pb.AdminToWorkerMessage, 1)}
|
|
pluginSvc.putSession(session)
|
|
|
|
if err := pluginSvc.store.AppendRunRecord(jobType, &JobRunRecord{
|
|
Outcome: RunOutcomeError,
|
|
CompletedAt: timeToPtr(time.Date(2026, 2, 10, 0, 0, 0, 0, time.UTC)),
|
|
}); err != nil {
|
|
t.Fatalf("AppendRunRecord error run: %v", err)
|
|
}
|
|
|
|
resultCh := make(chan error, 1)
|
|
go func() {
|
|
_, runErr := pluginSvc.RunDetection(context.Background(), jobType, &plugin_pb.ClusterContext{}, 10)
|
|
resultCh <- runErr
|
|
}()
|
|
|
|
message := <-session.outgoing
|
|
detectRequest := message.GetRunDetectionRequest()
|
|
if detectRequest == nil {
|
|
t.Fatalf("expected run detection request message")
|
|
}
|
|
if detectRequest.LastSuccessfulRun != nil {
|
|
t.Fatalf("expected last_successful_run to be nil when no success history")
|
|
}
|
|
|
|
pluginSvc.handleDetectionComplete("worker-a", &plugin_pb.DetectionComplete{
|
|
RequestId: message.RequestId,
|
|
JobType: jobType,
|
|
Success: true,
|
|
})
|
|
|
|
if runErr := <-resultCh; runErr != nil {
|
|
t.Fatalf("RunDetection error: %v", runErr)
|
|
}
|
|
}
|
|
|
|
func TestRunDetectionWithReportCapturesDetectionActivities(t *testing.T) {
|
|
pluginSvc, err := New(Options{})
|
|
if err != nil {
|
|
t.Fatalf("New plugin error: %v", err)
|
|
}
|
|
defer pluginSvc.Shutdown()
|
|
|
|
jobType := "vacuum"
|
|
pluginSvc.registry.UpsertFromHello(&plugin_pb.WorkerHello{
|
|
WorkerId: "worker-a",
|
|
Capabilities: []*plugin_pb.JobTypeCapability{
|
|
{JobType: jobType, CanDetect: true, MaxDetectionConcurrency: 1},
|
|
},
|
|
})
|
|
session := &streamSession{workerID: "worker-a", outgoing: make(chan *plugin_pb.AdminToWorkerMessage, 1)}
|
|
pluginSvc.putSession(session)
|
|
|
|
reportCh := make(chan *DetectionReport, 1)
|
|
errCh := make(chan error, 1)
|
|
go func() {
|
|
report, runErr := pluginSvc.RunDetectionWithReport(context.Background(), jobType, &plugin_pb.ClusterContext{}, 10)
|
|
reportCh <- report
|
|
errCh <- runErr
|
|
}()
|
|
|
|
message := <-session.outgoing
|
|
requestID := message.GetRequestId()
|
|
if requestID == "" {
|
|
t.Fatalf("expected request id in detection request")
|
|
}
|
|
|
|
pluginSvc.handleDetectionProposals("worker-a", &plugin_pb.DetectionProposals{
|
|
RequestId: requestID,
|
|
JobType: jobType,
|
|
Proposals: []*plugin_pb.JobProposal{
|
|
{
|
|
ProposalId: "proposal-1",
|
|
JobType: jobType,
|
|
Summary: "vacuum proposal",
|
|
Detail: "based on garbage ratio",
|
|
},
|
|
},
|
|
})
|
|
pluginSvc.handleDetectionComplete("worker-a", &plugin_pb.DetectionComplete{
|
|
RequestId: requestID,
|
|
JobType: jobType,
|
|
Success: true,
|
|
TotalProposals: 1,
|
|
})
|
|
|
|
report := <-reportCh
|
|
if report == nil {
|
|
t.Fatalf("expected detection report")
|
|
}
|
|
if report.RequestID == "" {
|
|
t.Fatalf("expected detection report request id")
|
|
}
|
|
if report.WorkerID != "worker-a" {
|
|
t.Fatalf("expected worker-a, got %q", report.WorkerID)
|
|
}
|
|
if len(report.Proposals) != 1 {
|
|
t.Fatalf("expected one proposal in report, got %d", len(report.Proposals))
|
|
}
|
|
if runErr := <-errCh; runErr != nil {
|
|
t.Fatalf("RunDetectionWithReport error: %v", runErr)
|
|
}
|
|
|
|
activities := pluginSvc.ListActivities(jobType, 0)
|
|
stages := map[string]bool{}
|
|
for _, activity := range activities {
|
|
if activity.RequestID != report.RequestID {
|
|
continue
|
|
}
|
|
stages[activity.Stage] = true
|
|
}
|
|
if !stages["requested"] || !stages["proposal"] || !stages["completed"] {
|
|
t.Fatalf("expected requested/proposal/completed activities, got stages=%v", stages)
|
|
}
|
|
}
|