-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathradialrange.go
91 lines (77 loc) · 2.62 KB
/
radialrange.go
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
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
package zrange
import "github.com/mmcloughlin/geohash"
// RadialRange uses a radius in kilometers, a latitude, and a longitude to return
// a slice of one or more ranges of keys that can be used to efficiently perform
// Geohash-based spatial queries.
//
// This method uses an algorithm that was derived from the "Search" section of this page:
// https://web.archive.org/web/20180526044934/https://github.com/yinqiwen/ardb/wiki/Spatial-Index#search
//
// RadialRange expands upon the ideas referenced above, by:
//
// • Sorting key ranges
//
// • Combining overlapping key ranges
//
// • Handling overflows resulting from bitshifting, such as when querying for: (-90, -180)
//
func (params RadialRangeParams) RadialRange() HashRanges {
return params.
SetDefaults().
FindNeighborsWithRadius().
SortMinAsc().
CombineRanges()
}
// RadialRangeParams defaults to expecting 64-bit Geohash-encoded keys.
type RadialRangeParams struct {
BitsOfPrecision uint
Radius,
Latitude,
Longitude float64
}
// SetDefaults sets the default values for the RadialRangeParams type.
func (params RadialRangeParams) SetDefaults() RadialRangeParams {
if params.BitsOfPrecision == 0 {
params.BitsOfPrecision = 64
}
return params
}
func (params RadialRangeParams) radiusToBits() uint {
const initialSignificantBits = 2
for i := len(radiusToBits) - 1; i > 0; i-- {
if params.Radius < radiusToBits[i] {
return uint(2*i + initialSignificantBits)
}
}
return uint(initialSignificantBits)
}
// FindNeighborsWithRadius uses the radius and coordinates to find neighboring
// hash ranges. SetDefaults should be called before use. RadialRange should be
// used instead of calling this method directly, unless more customized behavior
// is desired.
func (params RadialRangeParams) FindNeighborsWithRadius() HashRanges {
rangeBits := params.radiusToBits()
queryPoint := geohash.EncodeIntWithPrecision(
params.Latitude,
params.Longitude,
rangeBits,
)
neighborList := neighbors(geohash.NeighborsIntWithPrecision(queryPoint, rangeBits))
neighborList = append(neighborList, queryPoint)
rangeBitsDiff := params.BitsOfPrecision - rangeBits
return neighborList.shiftIntoRanges(rangeBitsDiff)
}
// WithinRadius determines whether a Geohash is within the specified radius.
// Its potential benefits are dependent on the data model in use.
func (params RadialRangeParams) WithinRadius(geohashID uint64) bool {
params = params.SetDefaults()
latitude, longitude := geohash.DecodeIntWithPrecision(
geohashID,
params.BitsOfPrecision,
)
distanceKm := Haversine(
params.Latitude, params.Longitude,
latitude, longitude,
)
return distanceKm < params.Radius
}