package recognize import ( "strings" "testing" ) func intp(n int) *int { return &n } func inputWith(paths ...string) Input { files := make([]File, len(paths)) for i, p := range paths { files[i] = File{Path: p, Size: 1 << 20} } return Input{Files: files} } func TestValidateSchema_OK(t *testing.T) { in := inputWith("a.mkv", "b.mkv") p := Plan{ Type: MediaSeries, Title: "Show", Files: []PlanFile{ {Src: "a.mkv", Role: RoleEpisode, Season: intp(1), Episode: intp(1)}, {Src: "b.mkv", Role: RoleEpisode, Season: intp(1), Episode: intp(2)}, }, } if err := validateSchema(&p, in); err != nil { t.Fatalf("validateSchema: %v", err) } } func TestValidateSchema_Errors(t *testing.T) { in := inputWith("a.mkv") tests := []struct { name string p Plan want string }{ {"empty type", Plan{Title: "x", Files: []PlanFile{{Src: "a.mkv", Role: RoleMain}}}, "type пустое"}, {"bad type", Plan{Type: "show", Title: "x", Files: []PlanFile{{Src: "a.mkv", Role: RoleMain}}}, "неизвестный type"}, {"empty title", Plan{Type: MediaMovie, Files: []PlanFile{{Src: "a.mkv", Role: RoleMain}}}, "title пустое"}, {"no files", Plan{Type: MediaMovie, Title: "x"}, "files пуст"}, {"bad role", Plan{Type: MediaMovie, Title: "x", Files: []PlanFile{{Src: "a.mkv", Role: "boss"}}}, "неизвестная role"}, {"empty src", Plan{Type: MediaMovie, Title: "x", Files: []PlanFile{{Src: "", Role: RoleMain}}}, "пустым src"}, {"unknown src", Plan{Type: MediaMovie, Title: "x", Files: []PlanFile{{Src: "z.mkv", Role: RoleMain}}}, "не найден"}, {"episode no num", Plan{Type: MediaSeries, Title: "x", Files: []PlanFile{{Src: "a.mkv", Role: RoleEpisode, Season: intp(1)}}}, "без номера episode"}, } for _, tt := range tests { t.Run(tt.name, func(t *testing.T) { err := validateSchema(&tt.p, in) if err == nil || !strings.Contains(err.Error(), tt.want) { t.Errorf("err = %v, want contains %q", err, tt.want) } }) } } func TestParsePlan_FencedJSON(t *testing.T) { in := inputWith("film.mkv") raw := "Вот результат:\n```json\n{\"type\":\"movie\",\"title\":\"Film\"," + "\"files\":[{\"src\":\"film.mkv\",\"role\":\"main\"}]}\n```" p, err := parsePlan(raw, in) if err != nil { t.Fatalf("parsePlan: %v", err) } if p.Title != "Film" || p.Type != MediaMovie { t.Errorf("plan = %+v", p) } } func TestParsePlan_UnknownFieldTolerated(t *testing.T) { in := inputWith("film.mkv") raw := `{"type":"movie","title":"Film","extra_field":123, "files":[{"src":"film.mkv","role":"main"}]}` if _, err := parsePlan(raw, in); err != nil { t.Fatalf("unknown field should be tolerated: %v", err) } } func TestStructuralWarnings_Movie(t *testing.T) { twoMains := Plan{Type: MediaMovie, Files: []PlanFile{ {Role: RoleMain}, {Role: RoleMain}, }} if w := structuralWarnings(twoMains); len(w) != 1 || !strings.Contains(w[0], "ожидался ровно 1") { t.Errorf("warnings = %v", w) } noMain := Plan{Type: MediaMovie, Files: []PlanFile{{Role: RoleSample}}} if w := structuralWarnings(noMain); len(w) != 1 { t.Errorf("want 1 warning for 0 mains, got %v", w) } clean := Plan{Type: MediaMovie, Files: []PlanFile{{Role: RoleMain}, {Role: RoleSample}}} if w := structuralWarnings(clean); len(w) != 0 { t.Errorf("clean movie should have no warnings, got %v", w) } } func TestSeriesWarnings_GapAndDup(t *testing.T) { files := []PlanFile{ {Role: RoleEpisode, Season: intp(1), Episode: intp(1)}, {Role: RoleEpisode, Season: intp(1), Episode: intp(1)}, // дубль {Role: RoleEpisode, Season: intp(1), Episode: intp(4)}, // пропуск 2,3 } w := seriesWarnings(files) var dup, gap bool for _, s := range w { if strings.Contains(s, "дубль") { dup = true } if strings.Contains(s, "пропуск") { gap = true } } if !dup || !gap { t.Errorf("want dup and gap warnings, got %v", w) } } func TestSeriesWarnings_Clean(t *testing.T) { files := []PlanFile{ {Role: RoleEpisode, Season: intp(1), Episode: intp(1)}, {Role: RoleEpisode, Season: intp(1), Episode: intp(2)}, {Role: RoleEpisode, Season: intp(2), Episode: intp(1)}, } if w := seriesWarnings(files); len(w) != 0 { t.Errorf("clean series should have no warnings, got %v", w) } } func TestConsistencyWarnings(t *testing.T) { yearMismatch := consistencyWarnings( Plan{Type: MediaMovie, Year: 2001}, PreParse{Year: 1999}, ) if len(yearMismatch) != 1 || !strings.Contains(yearMismatch[0], "год расходится") { t.Errorf("warnings = %v", yearMismatch) } typeMismatch := consistencyWarnings( Plan{Type: MediaMovie}, PreParse{Season: 2}, ) if len(typeMismatch) != 1 || !strings.Contains(typeMismatch[0], "тип расходится") { t.Errorf("warnings = %v", typeMismatch) } agree := consistencyWarnings( Plan{Type: MediaSeries, Year: 2006}, PreParse{Year: 2006, Season: 2}, ) if len(agree) != 0 { t.Errorf("agreeing signals should not warn, got %v", agree) } } func TestDecide_MetadataDisabled(t *testing.T) { p := Plan{Type: MediaMovie, Title: "X", Confidence: 0.99, Files: []PlanFile{{Role: RoleMain}}} d := decide(p, PreParse{}, nil, false, 0.85) if d.Auto { t.Error("без метабаз авто недопустимо") } if len(d.Reasons) == 0 || !strings.Contains(d.Reasons[0], "метабазы отключены") { t.Errorf("first reason should be DB match, got %v", d.Reasons) } } func TestDecide_NoMatch(t *testing.T) { p := Plan{Type: MediaMovie, Title: "X", Confidence: 0.99, Files: []PlanFile{{Role: RoleMain}}} d := decide(p, PreParse{}, nil, true, 0.85) if d.Auto || !strings.Contains(d.Reasons[0], "не найдено в базе") { t.Errorf("reasons = %v", d.Reasons) } } func TestDecide_AutoMovie(t *testing.T) { p := Plan{Type: MediaMovie, Title: "The Matrix", Year: 1999, Confidence: 0.95, Files: []PlanFile{{Role: RoleMain}, {Role: RoleSample}}} match := &Match{Provider: "tmdb", ProviderID: "603", Title: "The Matrix", Year: 1999} d := decide(p, PreParse{Year: 1999}, match, true, 0.85) if !d.Auto { t.Errorf("clean movie with match must be auto, reasons: %v", d.Reasons) } } func TestDecide_LowConfidenceBlocksAuto(t *testing.T) { p := Plan{Type: MediaMovie, Title: "X", Year: 2000, Confidence: 0.5, Files: []PlanFile{{Role: RoleMain}}} match := &Match{Provider: "tmdb", ProviderID: "1", Title: "X", Year: 2000} d := decide(p, PreParse{}, match, true, 0.85) if d.Auto || !hasReason(d.Reasons, "уверенность") { t.Errorf("low confidence must block auto, reasons: %v", d.Reasons) } } func TestDecide_AutoSeriesEpisodeCount(t *testing.T) { s := 2 mk := func(e int) PlanFile { ep := e; return PlanFile{Role: RoleEpisode, Season: &s, Episode: &ep} } p := Plan{Type: MediaSeries, Title: "Fargo", Year: 2014, Confidence: 0.9, Files: []PlanFile{mk(1), mk(2), mk(3)}} match := &Match{Provider: "tmdb", ProviderID: "1", Title: "Fargo", Year: 2014, SeasonEpisodeCounts: map[int]int{2: 3}} if d := decide(p, PreParse{Year: 2014}, match, true, 0.85); !d.Auto { t.Errorf("full season must be auto, reasons: %v", d.Reasons) } // Неполный пак (в базе 10) — авто блокируется. match.SeasonEpisodeCounts = map[int]int{2: 10} if d := decide(p, PreParse{Year: 2014}, match, true, 0.85); d.Auto || !hasReason(d.Reasons, "распознано серий 3, в базе 10") { t.Errorf("partial pack must block auto, reasons: %v", d.Reasons) } // Нет данных о числе серий — авто блокируется. match.SeasonEpisodeCounts = nil if d := decide(p, PreParse{Year: 2014}, match, true, 0.85); d.Auto || !hasReason(d.Reasons, "нет данных о числе серий") { t.Errorf("missing counts must block auto, reasons: %v", d.Reasons) } } func TestPreParse(t *testing.T) { pre := preParse("The.Matrix.1999.1080p.BluRay.x264") if pre.Year != 1999 { t.Errorf("year = %d, want 1999", pre.Year) } if !strings.Contains(strings.ToLower(pre.Title), "matrix") { t.Errorf("title = %q", pre.Title) } series := preParse("Some.Show.S02E05.720p") if series.Season != 2 || series.Episode != 5 { t.Errorf("season/episode = %d/%d, want 2/5", series.Season, series.Episode) } }