identity_test.go 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480
  1. package identity
  2. import (
  3. "io"
  4. "io/ioutil"
  5. "os"
  6. "path"
  7. "reflect"
  8. "strings"
  9. "testing"
  10. "unsafe"
  11. "github.com/stretchr/testify/assert"
  12. "gopkg.in/src-d/go-git.v4/plumbing"
  13. "gopkg.in/src-d/go-git.v4/plumbing/object"
  14. "gopkg.in/src-d/go-git.v4/plumbing/storer"
  15. "gopkg.in/src-d/hercules.v10/internal/core"
  16. "gopkg.in/src-d/hercules.v10/internal/test"
  17. )
  18. func fixtureIdentityDetector() *Detector {
  19. peopleDict := map[string]int{}
  20. peopleDict["vadim@sourced.tech"] = 0
  21. peopleDict["gmarkhor@gmail.com"] = 0
  22. reversePeopleDict := make([]string, 1)
  23. reversePeopleDict[0] = "Vadim"
  24. id := Detector{
  25. PeopleDict: peopleDict,
  26. ReversedPeopleDict: reversePeopleDict,
  27. }
  28. id.Initialize(test.Repository)
  29. return &id
  30. }
  31. func TestIdentityDetectorMeta(t *testing.T) {
  32. id := fixtureIdentityDetector()
  33. assert.Equal(t, id.Name(), "IdentityDetector")
  34. assert.Equal(t, len(id.Requires()), 0)
  35. assert.Equal(t, len(id.Provides()), 1)
  36. assert.Equal(t, id.Provides()[0], DependencyAuthor)
  37. opts := id.ListConfigurationOptions()
  38. assert.Len(t, opts, 2)
  39. assert.Equal(t, opts[0].Name, ConfigIdentityDetectorPeopleDictPath)
  40. assert.Equal(t, opts[1].Name, ConfigIdentityDetectorExactSignatures)
  41. logger := core.NewLogger()
  42. assert.NoError(t, id.Configure(map[string]interface{}{
  43. core.ConfigLogger: logger,
  44. }))
  45. assert.Equal(t, logger, id.l)
  46. }
  47. func TestIdentityDetectorConfigure(t *testing.T) {
  48. id := fixtureIdentityDetector()
  49. facts := map[string]interface{}{}
  50. m1 := map[string]int{"one": 0}
  51. m2 := []string{"one"}
  52. facts[FactIdentityDetectorPeopleDict] = m1
  53. facts[FactIdentityDetectorReversedPeopleDict] = m2
  54. assert.Nil(t, id.Configure(facts))
  55. assert.Equal(t, m1, facts[FactIdentityDetectorPeopleDict])
  56. assert.Equal(t, m2, facts[FactIdentityDetectorReversedPeopleDict])
  57. assert.Equal(t, id.PeopleDict, facts[FactIdentityDetectorPeopleDict])
  58. assert.Equal(t, id.ReversedPeopleDict, facts[FactIdentityDetectorReversedPeopleDict])
  59. id = fixtureIdentityDetector()
  60. tmpf, err := ioutil.TempFile("", "hercules-test-")
  61. assert.Nil(t, err)
  62. defer os.Remove(tmpf.Name())
  63. _, err = tmpf.WriteString(`Egor|egor@sourced.tech
  64. Vadim|vadim@sourced.tech`)
  65. assert.Nil(t, err)
  66. assert.Nil(t, tmpf.Close())
  67. delete(facts, FactIdentityDetectorPeopleDict)
  68. delete(facts, FactIdentityDetectorReversedPeopleDict)
  69. facts[ConfigIdentityDetectorPeopleDictPath] = tmpf.Name()
  70. assert.Nil(t, id.Configure(facts))
  71. assert.Len(t, id.PeopleDict, 2)
  72. assert.Len(t, id.ReversedPeopleDict, 1)
  73. assert.Equal(t, id.ReversedPeopleDict[0], "Vadim")
  74. delete(facts, FactIdentityDetectorPeopleDict)
  75. delete(facts, FactIdentityDetectorReversedPeopleDict)
  76. id = fixtureIdentityDetector()
  77. id.PeopleDict = nil
  78. assert.Nil(t, id.Configure(facts))
  79. assert.Equal(t, id.PeopleDict, facts[FactIdentityDetectorPeopleDict])
  80. assert.Equal(t, id.ReversedPeopleDict, facts[FactIdentityDetectorReversedPeopleDict])
  81. assert.Len(t, id.PeopleDict, 4)
  82. assert.Len(t, id.ReversedPeopleDict, 3)
  83. assert.Equal(t, id.ReversedPeopleDict[0], "Egor")
  84. assert.Equal(t, facts[FactIdentityDetectorPeopleCount], 2)
  85. delete(facts, FactIdentityDetectorPeopleDict)
  86. delete(facts, FactIdentityDetectorReversedPeopleDict)
  87. id = fixtureIdentityDetector()
  88. id.ReversedPeopleDict = nil
  89. assert.Nil(t, id.Configure(facts))
  90. assert.Equal(t, id.PeopleDict, facts[FactIdentityDetectorPeopleDict])
  91. assert.Equal(t, id.ReversedPeopleDict, facts[FactIdentityDetectorReversedPeopleDict])
  92. assert.Len(t, id.PeopleDict, 4)
  93. assert.Len(t, id.ReversedPeopleDict, 3)
  94. assert.Equal(t, id.ReversedPeopleDict[0], "Egor")
  95. assert.Equal(t, facts[FactIdentityDetectorPeopleCount], 2)
  96. delete(facts, FactIdentityDetectorPeopleDict)
  97. delete(facts, FactIdentityDetectorReversedPeopleDict)
  98. delete(facts, ConfigIdentityDetectorPeopleDictPath)
  99. commits := make([]*object.Commit, 0)
  100. iter, err := test.Repository.CommitObjects()
  101. commit, err := iter.Next()
  102. for ; err != io.EOF; commit, err = iter.Next() {
  103. if err != nil {
  104. panic(err)
  105. }
  106. commits = append(commits, commit)
  107. }
  108. facts[core.ConfigPipelineCommits] = commits
  109. id = fixtureIdentityDetector()
  110. id.PeopleDict = nil
  111. id.ReversedPeopleDict = nil
  112. assert.Nil(t, id.Configure(facts))
  113. assert.Equal(t, id.PeopleDict, facts[FactIdentityDetectorPeopleDict])
  114. assert.Equal(t, id.ReversedPeopleDict, facts[FactIdentityDetectorReversedPeopleDict])
  115. assert.True(t, len(id.PeopleDict) >= 3)
  116. assert.True(t, len(id.ReversedPeopleDict) >= 4)
  117. }
  118. func TestIdentityDetectorRegistration(t *testing.T) {
  119. summoned := core.Registry.Summon((&Detector{}).Name())
  120. assert.Len(t, summoned, 1)
  121. assert.Equal(t, summoned[0].Name(), "IdentityDetector")
  122. summoned = core.Registry.Summon((&Detector{}).Provides()[0])
  123. assert.Len(t, summoned, 1)
  124. assert.Equal(t, summoned[0].Name(), "IdentityDetector")
  125. }
  126. func TestIdentityDetectorConfigureEmpty(t *testing.T) {
  127. id := Detector{}
  128. assert.Panics(t, func() { id.Configure(map[string]interface{}{}) })
  129. }
  130. func TestIdentityDetectorConsume(t *testing.T) {
  131. commit, _ := test.Repository.CommitObject(plumbing.NewHash(
  132. "5c0e755dd85ac74584d9988cc361eccf02ce1a48"))
  133. deps := map[string]interface{}{}
  134. deps[core.DependencyCommit] = commit
  135. res, err := fixtureIdentityDetector().Consume(deps)
  136. assert.Nil(t, err)
  137. assert.Equal(t, res[DependencyAuthor].(int), 0)
  138. commit, _ = test.Repository.CommitObject(plumbing.NewHash(
  139. "8a03b5620b1caa72ec9cb847ea88332621e2950a"))
  140. deps[core.DependencyCommit] = commit
  141. res, err = fixtureIdentityDetector().Consume(deps)
  142. assert.Nil(t, err)
  143. assert.Equal(t, res[DependencyAuthor].(int), AuthorMissing)
  144. }
  145. func TestIdentityDetectorConsumeExact(t *testing.T) {
  146. commit, _ := test.Repository.CommitObject(plumbing.NewHash(
  147. "5c0e755dd85ac74584d9988cc361eccf02ce1a48"))
  148. deps := map[string]interface{}{}
  149. deps[core.DependencyCommit] = commit
  150. id := fixtureIdentityDetector()
  151. id.ExactSignatures = true
  152. id.PeopleDict = map[string]int{
  153. "vadim markovtsev <gmarkhor@gmail.com>": 0,
  154. "vadim markovtsev <vadim@sourced.tech>": 1,
  155. }
  156. res, err := id.Consume(deps)
  157. assert.Nil(t, err)
  158. assert.Equal(t, res[DependencyAuthor].(int), 1)
  159. commit, _ = test.Repository.CommitObject(plumbing.NewHash(
  160. "8a03b5620b1caa72ec9cb847ea88332621e2950a"))
  161. deps[core.DependencyCommit] = commit
  162. res, err = id.Consume(deps)
  163. assert.Nil(t, err)
  164. assert.Equal(t, res[DependencyAuthor].(int), AuthorMissing)
  165. }
  166. func TestIdentityDetectorLoadPeopleDict(t *testing.T) {
  167. id := fixtureIdentityDetector()
  168. err := id.LoadPeopleDict(path.Join("..", "..", "test_data", "identities"))
  169. assert.Nil(t, err)
  170. assert.Equal(t, len(id.PeopleDict), 7)
  171. assert.Contains(t, id.PeopleDict, "linus torvalds")
  172. assert.Contains(t, id.PeopleDict, "torvalds@linux-foundation.org")
  173. assert.Contains(t, id.PeopleDict, "vadim markovtsev")
  174. assert.Contains(t, id.PeopleDict, "vadim@sourced.tech")
  175. assert.Contains(t, id.PeopleDict, "another@one.com")
  176. assert.Contains(t, id.PeopleDict, "máximo cuadros")
  177. assert.Contains(t, id.PeopleDict, "maximo@sourced.tech")
  178. assert.Equal(t, len(id.ReversedPeopleDict), 4)
  179. assert.Equal(t, id.ReversedPeopleDict[0], "Linus Torvalds")
  180. assert.Equal(t, id.ReversedPeopleDict[1], "Vadim Markovtsev")
  181. assert.Equal(t, id.ReversedPeopleDict[2], "Máximo Cuadros")
  182. assert.Equal(t, id.ReversedPeopleDict[3], AuthorMissingName)
  183. }
  184. func TestIdentityDetectorLoadPeopleDictWrongPath(t *testing.T) {
  185. id := fixtureIdentityDetector()
  186. err := id.LoadPeopleDict(path.Join("identities"))
  187. assert.NotNil(t, err)
  188. }
  189. func TestIdentityDetectorGeneratePeopleDict(t *testing.T) {
  190. id := fixtureIdentityDetector()
  191. commits := make([]*object.Commit, 0)
  192. iter, err := test.Repository.CommitObjects()
  193. commit, err := iter.Next()
  194. for ; err != io.EOF; commit, err = iter.Next() {
  195. if err != nil {
  196. panic(err)
  197. }
  198. commits = append(commits, commit)
  199. }
  200. {
  201. i := 0
  202. for ; commits[i].Author.Name != "Vadim Markovtsev"; i++ {
  203. }
  204. if i > 0 {
  205. commit := commits[0]
  206. commits[0] = commits[i]
  207. commits[i] = commit
  208. }
  209. i = 1
  210. for ; commits[i].Author.Name != "Alexander Bezzubov"; i++ {
  211. }
  212. if i > 0 {
  213. commit := commits[1]
  214. commits[1] = commits[i]
  215. commits[i] = commit
  216. }
  217. i = 2
  218. for ; commits[i].Author.Name != "Máximo Cuadros"; i++ {
  219. }
  220. if i > 0 {
  221. commit := commits[2]
  222. commits[2] = commits[i]
  223. commits[i] = commit
  224. }
  225. }
  226. id.GeneratePeopleDict(commits)
  227. assert.True(t, len(id.PeopleDict) >= 7)
  228. assert.True(t, len(id.ReversedPeopleDict) >= 3)
  229. assert.Equal(t, id.PeopleDict["vadim markovtsev"], 0)
  230. assert.Equal(t, id.PeopleDict["vadim@sourced.tech"], 0)
  231. assert.Equal(t, id.PeopleDict["gmarkhor@gmail.com"], 0)
  232. assert.Equal(t, id.PeopleDict["alexander bezzubov"], 1)
  233. assert.Equal(t, id.PeopleDict["bzz@apache.org"], 1)
  234. assert.Equal(t, id.PeopleDict["máximo cuadros"], 2)
  235. assert.Equal(t, id.PeopleDict["mcuadros@gmail.com"], 2)
  236. assert.Equal(t, id.ReversedPeopleDict[0], "vadim markovtsev|gmarkhor@gmail.com|vadim@athenian.co|vadim@sourced.tech")
  237. assert.Equal(t, id.ReversedPeopleDict[1], "alexander bezzubov|bzz@apache.org")
  238. assert.Equal(t, id.ReversedPeopleDict[2], "máximo cuadros|mcuadros@gmail.com")
  239. assert.NotEqual(t, id.ReversedPeopleDict[len(id.ReversedPeopleDict)-1], AuthorMissingName)
  240. }
  241. func TestIdentityDetectorGeneratePeopleDictExact(t *testing.T) {
  242. id := fixtureIdentityDetector()
  243. id.ExactSignatures = true
  244. commits := make([]*object.Commit, 0)
  245. iter, err := test.Repository.CommitObjects()
  246. commit, err := iter.Next()
  247. for ; err != io.EOF; commit, err = iter.Next() {
  248. if err != nil {
  249. panic(err)
  250. }
  251. commits = append(commits, commit)
  252. }
  253. id.GeneratePeopleDict(commits)
  254. ass := assert.New(t)
  255. ass.Equal(len(id.PeopleDict), len(id.ReversedPeopleDict))
  256. ass.True(len(id.ReversedPeopleDict) >= 24)
  257. ass.Contains(id.PeopleDict, "vadim markovtsev <vadim@sourced.tech>")
  258. ass.Contains(id.PeopleDict, "vadim markovtsev <vadim@athenian.co>")
  259. ass.NotEqual(id.ReversedPeopleDict[len(id.ReversedPeopleDict)-1], AuthorMissingName)
  260. }
  261. func TestIdentityDetectorLoadPeopleDictInvalidPath(t *testing.T) {
  262. id := fixtureIdentityDetector()
  263. ipath := "/xxxyyyzzzInvalidPath!hehe"
  264. err := id.LoadPeopleDict(ipath)
  265. assert.NotNil(t, err)
  266. assert.Equal(t, err.(*os.PathError).Path, ipath)
  267. }
  268. type fakeBlobEncodedObject struct {
  269. Contents string
  270. }
  271. func (obj fakeBlobEncodedObject) Hash() plumbing.Hash {
  272. return plumbing.NewHash("ffffffffffffffffffffffffffffffffffffffff")
  273. }
  274. func (obj fakeBlobEncodedObject) Type() plumbing.ObjectType {
  275. return plumbing.BlobObject
  276. }
  277. func (obj fakeBlobEncodedObject) SetType(plumbing.ObjectType) {}
  278. func (obj fakeBlobEncodedObject) Size() int64 {
  279. return int64(len(obj.Contents))
  280. }
  281. func (obj fakeBlobEncodedObject) SetSize(int64) {}
  282. func (obj fakeBlobEncodedObject) Reader() (io.ReadCloser, error) {
  283. return ioutil.NopCloser(strings.NewReader(obj.Contents)), nil
  284. }
  285. func (obj fakeBlobEncodedObject) Writer() (io.WriteCloser, error) {
  286. return nil, nil
  287. }
  288. type fakeTreeEncodedObject struct {
  289. Name string
  290. }
  291. func (obj fakeTreeEncodedObject) Hash() plumbing.Hash {
  292. return plumbing.NewHash("ffffffffffffffffffffffffffffffffffffffff")
  293. }
  294. func (obj fakeTreeEncodedObject) Type() plumbing.ObjectType {
  295. return plumbing.TreeObject
  296. }
  297. func (obj fakeTreeEncodedObject) SetType(plumbing.ObjectType) {}
  298. func (obj fakeTreeEncodedObject) Size() int64 {
  299. return 1
  300. }
  301. func (obj fakeTreeEncodedObject) SetSize(int64) {}
  302. func (obj fakeTreeEncodedObject) Reader() (io.ReadCloser, error) {
  303. return ioutil.NopCloser(strings.NewReader(
  304. "100644 " + obj.Name + "\x00ffffffffffffffffffffffffffffffffffffffff")), nil
  305. }
  306. func (obj fakeTreeEncodedObject) Writer() (io.WriteCloser, error) {
  307. return nil, nil
  308. }
  309. type fakeEncodedObjectStorer struct {
  310. Name string
  311. Contents string
  312. }
  313. func (strr fakeEncodedObjectStorer) NewEncodedObject() plumbing.EncodedObject {
  314. return nil
  315. }
  316. func (strr fakeEncodedObjectStorer) HasEncodedObject(plumbing.Hash) error {
  317. return nil
  318. }
  319. func (strr fakeEncodedObjectStorer) SetEncodedObject(plumbing.EncodedObject) (plumbing.Hash, error) {
  320. return plumbing.NewHash("0000000000000000000000000000000000000000"), nil
  321. }
  322. func (strr fakeEncodedObjectStorer) EncodedObject(objType plumbing.ObjectType, hash plumbing.Hash) (plumbing.EncodedObject, error) {
  323. if objType == plumbing.TreeObject {
  324. return fakeTreeEncodedObject{Name: strr.Name}, nil
  325. } else if objType == plumbing.BlobObject {
  326. return fakeBlobEncodedObject{Contents: strr.Contents}, nil
  327. }
  328. return nil, nil
  329. }
  330. func (strr fakeEncodedObjectStorer) IterEncodedObjects(plumbing.ObjectType) (storer.EncodedObjectIter, error) {
  331. return nil, nil
  332. }
  333. func (strr fakeEncodedObjectStorer) EncodedObjectSize(plumbing.Hash) (int64, error) {
  334. return 0, nil
  335. }
  336. func getFakeCommitWithFile(name string, contents string) *object.Commit {
  337. c := object.Commit{
  338. Hash: plumbing.NewHash("ffffffffffffffffffffffffffffffffffffffff"),
  339. Author: object.Signature{
  340. Name: "Vadim Markovtsev",
  341. Email: "vadim@sourced.tech",
  342. },
  343. Committer: object.Signature{
  344. Name: "Vadim Markovtsev",
  345. Email: "vadim@sourced.tech",
  346. },
  347. Message: "Virtual file " + name,
  348. TreeHash: plumbing.NewHash("ffffffffffffffffffffffffffffffffffffffff"),
  349. }
  350. voc := reflect.ValueOf(&c)
  351. voc = voc.Elem()
  352. f := voc.FieldByName("s")
  353. ptr := unsafe.Pointer(f.UnsafeAddr())
  354. strr := fakeEncodedObjectStorer{Name: name, Contents: contents}
  355. *(*storer.EncodedObjectStorer)(ptr) = strr
  356. return &c
  357. }
  358. func TestIdentityDetectorGeneratePeopleDictMailmap(t *testing.T) {
  359. id := fixtureIdentityDetector()
  360. commits := make([]*object.Commit, 0)
  361. iter, err := test.Repository.CommitObjects()
  362. commit, err := iter.Next()
  363. for ; err != io.EOF; commit, err = iter.Next() {
  364. if err != nil {
  365. panic(err)
  366. }
  367. commits = append(commits, commit)
  368. }
  369. fake := getFakeCommitWithFile(
  370. ".mailmap",
  371. "Strange Guy <vadim@sourced.tech>\nVadim Markovtsev <vadim@sourced.tech> Strange Guy <vadim@sourced.tech>")
  372. commits = append(commits, fake)
  373. id.GeneratePeopleDict(commits)
  374. assert.Contains(t, id.ReversedPeopleDict,
  375. "strange guy|vadim markovtsev|gmarkhor@gmail.com|vadim@athenian.co|vadim@sourced.tech")
  376. }
  377. func TestIdentityDetectorMergeReversedDictsLiteral(t *testing.T) {
  378. pa1 := [...]string{"one|one@one", "two|aaa@two"}
  379. pa2 := [...]string{"two|aaa@two", "three|one@one"}
  380. people, merged := MergeReversedDictsLiteral(pa1[:], pa2[:])
  381. assert.Len(t, people, 3)
  382. assert.Len(t, merged, 3)
  383. assert.Equal(t, people["one|one@one"], MergedIndex{0, 0, -1})
  384. assert.Equal(t, people["two|aaa@two"], MergedIndex{1, 1, 0})
  385. assert.Equal(t, people["three|one@one"], MergedIndex{2, -1, 1})
  386. assert.Equal(t, merged, []string{"one|one@one", "two|aaa@two", "three|one@one"})
  387. pa1 = [...]string{"two|aaa@two", "one|one@one"}
  388. people, merged = MergeReversedDictsLiteral(pa1[:], pa2[:])
  389. assert.Len(t, people, 3)
  390. assert.Len(t, merged, 3)
  391. assert.Equal(t, people["one|one@one"], MergedIndex{1, 1, -1})
  392. assert.Equal(t, people["two|aaa@two"], MergedIndex{0, 0, 0})
  393. assert.Equal(t, people["three|one@one"], MergedIndex{2, -1, 1})
  394. assert.Equal(t, merged, []string{"two|aaa@two", "one|one@one", "three|one@one"})
  395. }
  396. func TestIdentityDetectorMergeReversedDictsIdentities(t *testing.T) {
  397. pa1 := [...]string{"one|one@one", "two|aaa@two"}
  398. pa2 := [...]string{"two|aaa@two", "three|one@one"}
  399. people, merged := MergeReversedDictsIdentities(pa1[:], pa2[:])
  400. assert.Len(t, people, 3)
  401. assert.Len(t, merged, 2)
  402. assert.Equal(t, people["one|one@one"], MergedIndex{0, 0, -1})
  403. assert.Equal(t, people["two|aaa@two"], MergedIndex{1, 1, 0})
  404. assert.Equal(t, people["three|one@one"], MergedIndex{0, -1, 1})
  405. assert.Equal(t, merged, []string{"one|three|one@one", "two|aaa@two"})
  406. }
  407. func TestIdentityDetectorMergeReversedDictsIdentitiesStrikeBack(t *testing.T) {
  408. pa1 := [...]string{"one|one@one", "two|aaa@two", "three|three@three"}
  409. pa2 := [...]string{"two|aaa@two", "three|one@one"}
  410. people, merged := MergeReversedDictsIdentities(pa1[:], pa2[:])
  411. assert.Len(t, people, 4)
  412. assert.Len(t, merged, 2)
  413. assert.Equal(t, people["one|one@one"], MergedIndex{0, 0, -1})
  414. assert.Equal(t, people["two|aaa@two"], MergedIndex{1, 1, 0})
  415. assert.Equal(t, people["three|one@one"], MergedIndex{0, -1, 1})
  416. assert.Equal(t, people["three|three@three"], MergedIndex{0, 2, -1})
  417. assert.Equal(t, merged, []string{"one|three|one@one|three@three", "two|aaa@two"})
  418. pa1 = [...]string{"one|one@one", "two|aaa@two", "three|aaa@two"}
  419. people, merged = MergeReversedDictsIdentities(pa1[:], pa2[:])
  420. assert.Len(t, people, 4)
  421. assert.Len(t, merged, 1)
  422. assert.Equal(t, people["one|one@one"], MergedIndex{0, 0, -1})
  423. assert.Equal(t, people["two|aaa@two"], MergedIndex{0, 1, 0})
  424. assert.Equal(t, people["three|one@one"], MergedIndex{0, -1, 1})
  425. assert.Equal(t, people["three|aaa@two"], MergedIndex{0, 2, -1})
  426. assert.Equal(t, merged, []string{"one|three|two|aaa@two|one@one"})
  427. }
  428. func TestIdentityDetectorFork(t *testing.T) {
  429. id1 := fixtureIdentityDetector()
  430. clones := id1.Fork(1)
  431. assert.Len(t, clones, 1)
  432. id2 := clones[0].(*Detector)
  433. assert.True(t, id1 == id2)
  434. id1.Merge([]core.PipelineItem{id2})
  435. }