File: sort_values.go

package info (click to toggle)
golang-github-maxatome-go-testdeep 1.14.0-1
  • links: PTS, VCS
  • area: main
  • in suites: forky, sid, trixie
  • size: 2,416 kB
  • sloc: perl: 1,012; yacc: 130; makefile: 2
file content (67 lines) | stat: -rw-r--r-- 1,872 bytes parent folder | download | duplicates (2)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
// Copyright (c) 2019, Maxime Soulé
// All rights reserved.
//
// This source code is licensed under the BSD-style license found in the
// LICENSE file in the root directory of this source tree.

package tdutil

import (
	"reflect"
	"sort"

	"github.com/maxatome/go-testdeep/internal/visited"
)

// SortableValues is used to allow the sorting of a [][reflect.Value]
// slice. It is used with the standard sort package:
//
//	vals := []reflect.Value{a, b, c, d}
//	sort.Sort(SortableValues(vals))
//	// vals contents now sorted
//
// Replace [sort.Sort] by [sort.Stable] for a stable sort. See [sort]
// documentation.
//
// Sorting rules are as follows:
//   - nil is always lower
//   - different types are sorted by their name
//   - false is lesser than true
//   - float and int numbers are sorted by their value
//   - complex numbers are sorted by their real, then by their imaginary parts
//   - strings are sorted by their value
//   - map: shorter length is lesser, then sorted by address
//   - functions, channels and unsafe pointer are sorted by their address
//   - struct: comparison is spread to each field
//   - pointer: comparison is spread to the pointed value
//   - arrays: comparison is spread to each item
//   - slice: comparison is spread to each item, then shorter length is lesser
//   - interface: comparison is spread to the value
//
// Cyclic references are correctly handled.
func SortableValues(s []reflect.Value) sort.Interface {
	r := &rValues{
		Slice: s,
	}
	if len(s) > 1 {
		r.Visited = visited.NewVisited()
	}
	return r
}

type rValues struct {
	Visited visited.Visited
	Slice   []reflect.Value
}

func (v *rValues) Len() int {
	return len(v.Slice)
}

func (v *rValues) Less(i, j int) bool {
	return cmp(v.Visited, v.Slice[i], v.Slice[j]) < 0
}

func (v *rValues) Swap(i, j int) {
	v.Slice[i], v.Slice[j] = v.Slice[j], v.Slice[i]
}