package entity import ( "testing" "github.com/photoprism/photoprism/internal/form" "github.com/stretchr/testify/assert" ) func TestMarker_TableName(t *testing.T) { m := &Marker{} assert.Contains(t, m.TableName(), "markers") } func TestNewMarker(t *testing.T) { m := NewMarker(1000000, "lt9k3pw1wowuy3c3", SrcImage, MarkerLabel, 0.308333, 0.206944, 0.355556, 0.355556) assert.IsType(t, &Marker{}, m) assert.Equal(t, uint(1000000), m.FileID) assert.Equal(t, "lt9k3pw1wowuy3c3", m.SubjectUID) assert.Equal(t, SrcImage, m.MarkerSrc) assert.Equal(t, MarkerLabel, m.MarkerType) } func TestMarker_SaveForm(t *testing.T) { t.Run("fa-ge add new name to marker then rename marker", func(t *testing.T) { m := MarkerFixtures.Get("fa-gr-1") m2 := MarkerFixtures.Get("fa-gr-2") m3 := MarkerFixtures.Get("fa-gr-3") assert.Empty(t, m.SubjectUID) assert.Empty(t, m2.SubjectUID) assert.Empty(t, m3.SubjectUID) m.MarkerInvalid = true m.Score = 50 //set new name f := form.Marker{SubjectSrc: SrcManual, MarkerName: "Jane Doe", MarkerInvalid: false} err := m.SaveForm(f) if err != nil { t.Fatal(err) } assert.NotEmpty(t, m.SubjectUID) assert.Equal(t, "Jane Doe", m.GetSubject().SubjectName) assert.Equal(t, "Jane Doe", FindMarker(9).GetSubject().SubjectName) assert.Equal(t, "Jane Doe", FindMarker(10).GetSubject().SubjectName) //rename f3 := form.Marker{SubjectSrc: SrcManual, MarkerName: "Franzilein", MarkerInvalid: false} err3 := FindMarker(9).SaveForm(f3) if err3 != nil { t.Fatal(err3) } assert.Equal(t, "Franzilein", FindMarker(8).GetSubject().SubjectName) assert.Equal(t, "Franzilein", FindMarker(9).GetSubject().SubjectName) assert.Equal(t, "Franzilein", FindMarker(10).GetSubject().SubjectName) }) } func TestUpdateOrCreateMarker(t *testing.T) { t.Run("success", func(t *testing.T) { m := NewMarker(1000000, "lt9k3pw1wowuy3c3", SrcImage, MarkerLabel, 0.308333, 0.206944, 0.355556, 0.355556) assert.IsType(t, &Marker{}, m) assert.Equal(t, uint(1000000), m.FileID) assert.Equal(t, "lt9k3pw1wowuy3c3", m.SubjectUID) assert.Equal(t, SrcImage, m.MarkerSrc) assert.Equal(t, MarkerLabel, m.MarkerType) m, err := UpdateOrCreateMarker(m) if err != nil { t.Fatal(err) } if m == nil { t.Fatal("result should not be nil") } if m.ID <= 0 { t.Errorf("ID should be > 0") } }) } func TestMarker_Updates(t *testing.T) { t.Run("success", func(t *testing.T) { m := NewMarker(1000000, "lt9k3pw1wowuy3c4", SrcImage, MarkerLabel, 0.308333, 0.206944, 0.355556, 0.355556) m, err := UpdateOrCreateMarker(m) if err != nil { t.Fatal(err) } assert.Equal(t, SrcImage, m.MarkerSrc) assert.Equal(t, MarkerLabel, m.MarkerType) if err := m.Updates(Marker{MarkerSrc: SrcMeta}); err != nil { t.Fatal(err) } assert.Equal(t, SrcMeta, m.MarkerSrc) assert.Equal(t, MarkerLabel, m.MarkerType) if m.ID <= 0 { t.Errorf("ID should be > 0") } }) } func TestMarker_Update(t *testing.T) { t.Run("success", func(t *testing.T) { m := NewMarker(1000000, "lt9k3pw1wowuy3c4", SrcImage, MarkerLabel, 0.308333, 0.206944, 0.355556, 0.355556) m, err := UpdateOrCreateMarker(m) if err != nil { t.Fatal(err) } assert.Equal(t, MarkerLabel, m.MarkerType) if err := m.Update("MarkerSrc", SrcMeta); err != nil { t.Fatal(err) } assert.Equal(t, SrcMeta, m.MarkerSrc) assert.Equal(t, MarkerLabel, m.MarkerType) if m.ID <= 0 { t.Errorf("ID should be > 0") } }) } func TestMarker_Save(t *testing.T) { t.Run("success", func(t *testing.T) { m := NewMarker(1000000, "lt9k3pw1wowuy3c4", SrcImage, MarkerLabel, 0.308333, 0.206944, 0.355556, 0.355556) m, err := UpdateOrCreateMarker(m) if err != nil { t.Fatal(err) } assert.Equal(t, MarkerLabel, m.MarkerType) m.MarkerSrc = SrcMeta assert.Equal(t, SrcMeta, m.MarkerSrc) initialDate := m.UpdatedAt if err := m.Save(); err != nil { t.Fatal(err) } afterDate := m.UpdatedAt assert.Equal(t, SrcMeta, m.MarkerSrc) assert.True(t, afterDate.After(initialDate)) if m.ID <= 0 { t.Errorf("ID should be > 0") } p := PhotoFixtures.Get("19800101_000002_D640C559") assert.Empty(t, p.Files) p.PreloadFiles(true) assert.NotEmpty(t, p.Files) t.Logf("FILES: %#v", p.Files) }) t.Run("invalid position", func(t *testing.T) { m := Marker{X: 0, Y: 0} err := m.Save() assert.Equal(t, "marker: invalid position", err.Error()) }) } func TestMarker_ClearSubject(t *testing.T) { t.Run("1000003-2", func(t *testing.T) { m := MarkerFixtures.Get("1000003-2") assert.NotEmpty(t, m.MarkerName) err := m.ClearSubject(SrcAuto) if err != nil { t.Fatal(err) } assert.Empty(t, m.MarkerName) }) t.Run("actress-a-1", func(t *testing.T) { m := MarkerFixtures.Get("actress-a-2") // id 12 m2 := MarkerFixtures.Get("actress-a-1") // id 11 m3 := MarkerFixtures.Get("actress-a-3") // id 13 m4 := MarkerFixtures.Get("actress-a-4") // id 14 assert.Equal(t, "jqy1y111h1njaaac", m.SubjectUID) assert.Equal(t, "jqy1y111h1njaaac", m2.SubjectUID) assert.Equal(t, "jqy1y111h1njaaac", m3.SubjectUID) assert.Equal(t, "jqy1y111h1njaaac", m4.SubjectUID) assert.Equal(t, "GMH5NISEEULNJL6RATITOA3TMZXMTMCI", m.GetFace().ID) assert.Equal(t, "GMH5NISEEULNJL6RATITOA3TMZXMTMCI", m2.GetFace().ID) assert.Equal(t, "GMH5NISEEULNJL6RATITOA3TMZXMTMCI", m3.GetFace().ID) assert.Equal(t, "GMH5NISEEULNJL6RATITOA3TMZXMTMCI", m4.GetFace().ID) assert.Equal(t, int(0), FindMarker(12).GetFace().Collisions) //remove similar face err := m.ClearSubject(SrcAuto) if err != nil { t.Fatal(err) } t.Log(FindMarker(11).FaceID) t.Log(FindMarker(12).FaceID) t.Log(FindMarker(13).FaceID) t.Log(FindMarker(14).FaceID) assert.Empty(t, m.SubjectUID) assert.Equal(t, "jqy1y111h1njaaac", FindMarker(11).SubjectUID) assert.Equal(t, "jqy1y111h1njaaac", FindMarker(13).SubjectUID) assert.Equal(t, "jqy1y111h1njaaac", FindMarker(14).SubjectUID) assert.Empty(t, m.FaceID) assert.Equal(t, "GMH5NISEEULNJL6RATITOA3TMZXMTMCI", FindMarker(11).GetFace().ID) assert.Equal(t, "GMH5NISEEULNJL6RATITOA3TMZXMTMCI", FindMarker(13).GetFace().ID) assert.Equal(t, "GMH5NISEEULNJL6RATITOA3TMZXMTMCI", FindMarker(14).GetFace().ID) assert.Equal(t, int(0), FindMarker(11).GetFace().Collisions) f := form.Marker{SubjectSrc: SrcManual, MarkerName: "Actress B", MarkerInvalid: false} err2 := m.SaveForm(f) if err2 != nil { t.Fatal(err2) } assert.Equal(t, "Actress B", m.GetSubject().SubjectName) assert.Equal(t, "jqy1y111h1njaaac", FindMarker(11).SubjectUID) assert.Equal(t, "jqy1y111h1njaaac", FindMarker(13).SubjectUID) assert.NotEmpty(t, m.FaceID) assert.Equal(t, "GMH5NISEEULNJL6RATITOA3TMZXMTMCI", FindMarker(11).GetFace().ID) assert.Equal(t, "GMH5NISEEULNJL6RATITOA3TMZXMTMCI", FindMarker(13).GetFace().ID) }) } func TestMarker_ClearFace(t *testing.T) { t.Run("1000003-2", func(t *testing.T) { m := MarkerFixtures.Get("1000003-2") assert.NotEmpty(t, m.FaceID) updated, err := m.ClearFace() if err != nil { t.Fatal(err) } assert.True(t, updated) assert.Empty(t, m.FaceID) }) t.Run("empty face id", func(t *testing.T) { m := Marker{FaceID: ""} updated, err := m.ClearFace() if err != nil { t.Fatal(err) } assert.False(t, updated) assert.Empty(t, m.FaceID) }) t.Run("subject src manual", func(t *testing.T) { m := Marker{FaceID: "123ab"} assert.NotEmpty(t, m.FaceID) assert.Empty(t, m.MatchedAt) updated, err := m.ClearFace() if err != nil { t.Fatal(err) } assert.True(t, updated) assert.Empty(t, m.FaceID) assert.NotEmpty(t, m.MatchedAt) }) } func TestMarker_SyncSubject(t *testing.T) { t.Run("no face marker", func(t *testing.T) { m := Marker{MarkerType: "test", Subject: nil} assert.Nil(t, m.SyncSubject(false)) }) t.Run("subject is nil", func(t *testing.T) { m := Marker{MarkerType: MarkerFace, Subject: nil} assert.Nil(t, m.SyncSubject(false)) }) } func TestMarker_Create(t *testing.T) { t.Run("invalid position", func(t *testing.T) { m := Marker{X: 0, Y: 0} err := m.Create() assert.Equal(t, "marker: invalid position", err.Error()) }) } func TestMarker_Embeddings(t *testing.T) { t.Run("success", func(t *testing.T) { m := MarkerFixtures.Get("1000003-4") assert.Equal(t, 0.013083286379677253, m.Embeddings()[0][0]) }) t.Run("empty embedding", func(t *testing.T) { m := Marker{} m.EmbeddingsJSON = []byte("") assert.Empty(t, m.Embeddings()) }) t.Run("invalid embedding json", func(t *testing.T) { m := Marker{} m.EmbeddingsJSON = []byte("[false]") assert.Empty(t, m.Embeddings()[0]) }) } func TestMarker_HasFace(t *testing.T) { t.Run("true", func(t *testing.T) { m := MarkerFixtures.Get("1000003-6") assert.True(t, m.HasFace(nil, -1)) assert.True(t, m.HasFace(FaceFixtures.Pointer("joe-biden"), -1)) }) t.Run("false", func(t *testing.T) { m := MarkerFixtures.Get("1000003-6") assert.False(t, m.HasFace(FaceFixtures.Pointer("joe-biden"), 0.1)) }) t.Run("face id empty", func(t *testing.T) { m := Marker{FaceID: ""} assert.False(t, m.HasFace(FaceFixtures.Pointer("joe-biden"), 0.1)) }) t.Run("face dist < 0", func(t *testing.T) { m := Marker{FaceID: "123", FaceDist: -1} assert.False(t, m.HasFace(FaceFixtures.Pointer("joe-biden"), 0.1)) }) t.Run("face id = f.ID", func(t *testing.T) { m := Marker{FaceID: "VF7ANLDET2BKZNT4VQWJMMC6HBEFDOG6"} assert.True(t, m.HasFace(FaceFixtures.Pointer("joe-biden"), 0.1)) }) } func TestMarker_GetSubject(t *testing.T) { t.Run("return subject", func(t *testing.T) { m := Marker{Subject: &Subject{SubjectName: "Test Subject"}} assert.Equal(t, "Test Subject", m.GetSubject().SubjectName) }) t.Run("uid empty, marker name not empty", func(t *testing.T) { m := Marker{SubjectUID: "", MarkerName: "Hans Mayer"} assert.Equal(t, "Hans Mayer", m.GetSubject().SubjectName) }) } func TestMarker_GetFace(t *testing.T) { t.Run("return face", func(t *testing.T) { m := Marker{Face: &Face{ID: "1234"}} assert.Equal(t, "1234", m.GetFace().ID) }) t.Run("find face with ID", func(t *testing.T) { m := Marker{FaceID: "VF7ANLDET2BKZNT4VQWJMMC6HBEFDOG6"} assert.Equal(t, "jqy3y652h8njw0sx", m.GetFace().SubjectUID) }) t.Run("low quality marker", func(t *testing.T) { m := Marker{FaceID: "", SubjectSrc: SrcManual, Size: 130} assert.Nil(t, m.GetFace()) }) t.Run("create face", func(t *testing.T) { m := Marker{FaceID: "", SubjectSrc: SrcManual, Size: 160, Score: 40} assert.NotEmpty(t, m.GetFace().ID) }) } func TestFindMarker(t *testing.T) { t.Run("nil", func(t *testing.T) { assert.Nil(t, FindMarker(0000)) }) }