@@ -6,10 +6,14 @@ package rdf
6
6
7
7
import (
8
8
"bytes"
9
+ "cmp"
9
10
"errors"
10
11
"fmt"
11
12
"hash"
13
+ "slices"
12
14
"sort"
15
+
16
+ "gonum.org/v1/gonum/internal/order"
13
17
)
14
18
15
19
// See "Canonical Forms for Isomorphic and Equivalent RDF Graphs: Algorithms
@@ -53,7 +57,7 @@ func lexicalHashes(dst [][]byte, hashes map[string][]byte) {
53
57
dst [i ] = s
54
58
i ++
55
59
}
56
- sort . Sort ( lexical ( dst ) )
60
+ order . BySliceValues ( dst )
57
61
}
58
62
59
63
// IsoCanonicalHashes returns a mapping between the nodes of the RDF graph
@@ -185,7 +189,7 @@ func C14n(dst, src []*Statement, terms map[string]map[string]bool) ([]*Statement
185
189
blanks [i ] = h
186
190
i ++
187
191
}
188
- sort . Strings (blanks )
192
+ slices . Sort (blanks )
189
193
190
194
c14n := make (map [string ]string )
191
195
for i , b := range blanks {
@@ -218,7 +222,7 @@ func C14n(dst, src []*Statement, terms map[string]map[string]bool) ([]*Statement
218
222
n .Object = Term {Value : translate (s .Object .Value , c14n )}
219
223
n .Label = Term {Value : translate (s .Label .Value , c14n )}
220
224
}
221
- sort . Sort ( c14nStatements ( dst ) )
225
+ sortC14nStatements ( dst )
222
226
223
227
return dst , nil
224
228
}
@@ -230,33 +234,20 @@ func translate(term string, mapping map[string]string) string {
230
234
return term
231
235
}
232
236
233
- type c14nStatements []* Statement
237
+ func sortC14nStatements (statements []* Statement ) {
238
+ slices .SortFunc (statements , func (a , b * Statement ) int {
239
+ if n := cmp .Compare (a .Subject .Value , b .Subject .Value ); n != 0 {
240
+ return n
241
+ }
234
242
235
- func (s c14nStatements ) Len () int { return len (s ) }
236
- func (s c14nStatements ) Less (i , j int ) bool {
237
- si := s [i ]
238
- sj := s [j ]
239
- switch {
240
- case si .Subject .Value < sj .Subject .Value :
241
- return true
242
- case si .Subject .Value > sj .Subject .Value :
243
- return false
244
- }
245
- switch { // Always IRI.
246
- case si .Predicate .Value < sj .Predicate .Value :
247
- return true
248
- case si .Predicate .Value > sj .Predicate .Value :
249
- return false
250
- }
251
- switch {
252
- case si .Object .Value < sj .Object .Value :
253
- return true
254
- case si .Object .Value > sj .Object .Value :
255
- return false
256
- }
257
- return si .Label .Value < sj .Label .Value
243
+ // Always IRI.
244
+ if n := cmp .Compare (a .Predicate .Value , b .Predicate .Value ); n != 0 {
245
+ return n
246
+ }
247
+
248
+ return cmp .Compare (a .Object .Value , b .Object .Value )
249
+ })
258
250
}
259
- func (s c14nStatements ) Swap (i , j int ) { s [i ], s [j ] = s [j ], s [i ] }
260
251
261
252
// hashBNodes returns the hashed blank nodes of the graph described by statements
262
253
// using the provided hash function. Hashes are initialised with zero.
@@ -497,20 +488,13 @@ func (b hashBag) add(term string, hash []byte) {
497
488
// state and returns the hash.
498
489
func (b hashBag ) sum (term string ) []byte {
499
490
p := b .hashesFor [term ]
500
- sort . Sort ( lexical ( p ) )
491
+ order . BySliceValues ( p )
501
492
h := hashTuple (b .hash , p ... )
502
493
b .hashesFor [term ] = b .hashesFor [term ][:1 ]
503
494
b .hashesFor [term ][0 ] = h
504
495
return h
505
496
}
506
497
507
- // lexical implements lexical sorting of [][]byte.
508
- type lexical [][]byte
509
-
510
- func (b lexical ) Len () int { return len (b ) }
511
- func (b lexical ) Less (i , j int ) bool { return string (b [i ]) < string (b [j ]) }
512
- func (b lexical ) Swap (i , j int ) { b [i ], b [j ] = b [j ], b [i ] }
513
-
514
498
// hashTuple returns the h hash of the concatenation of t.
515
499
func hashTuple (h hash.Hash , t ... []byte ) []byte {
516
500
h .Reset ()
0 commit comments