forked from kellydunn/golang-geo
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathpolygon_test.go
156 lines (127 loc) · 3.93 KB
/
polygon_test.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
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
package geo
import (
"encoding/json"
"os"
"testing"
)
// Ensures that the library can detect if a point is in a polygon.
// Uses Brunei and the capital of Brunei as a set of test points.
func TestPointInPolygon(t *testing.T) {
brunei, err := polygonFromFile("test/data/brunei.json")
if err != nil {
t.Error("brunei json file failed to parse: ", err)
}
point := Point{lng: 114.9480600, lat: 4.9402900}
if !brunei.Contains(&point) {
t.Error("Expected the capital of Brunei to be in Brunei, but it wasn't.")
}
}
// Ensures that the polygon logic can correctly identify if a polygon does not contain a point.
// Uses Brunei, Seattle, and a point directly outside of Brunei limits as test points.
func TestPointNotInPolygon(t *testing.T) {
brunei, err := polygonFromFile("test/data/brunei.json")
if err != nil {
t.Error("brunei json file failed to parse: ", err)
}
// Seattle, WA should not be inside of Brunei
point := NewPoint(47.45, 122.30)
if brunei.Contains(point) {
t.Error("Seattle, WA [47.45, 122.30] should not be inside of Brunei")
}
// A point just outside of the successful bounds in Brunei
// Should not be contained in the Polygon
precision := NewPoint(114.659596, 4.007636)
if brunei.Contains(precision) {
t.Error("A point just outside of Brunei should not be contained in the Polygon")
}
}
// Ensures that a point can be contained in a complex polygon (e.g. a donut)
// This particular Polygon has a hole in it.
func TestPointInPolygonWithHole(t *testing.T) {
nsw, err := polygonFromFile("test/data/nsw.json")
if err != nil {
t.Error("nsw json file failed to parse: ", err)
}
act, err := polygonFromFile("test/data/act.json")
if err != nil {
t.Error("act json file failed to parse: ", err)
}
// Look at two contours
canberra := Point{lng: 149.128684300000030000, lat: -35.2819998}
isnsw := nsw.Contains(&canberra)
isact := act.Contains(&canberra)
if !isnsw && !isact {
t.Error("Canberra should be in NSW and also in the sub-contour ACT state")
}
// Using NSW as a multi-contour polygon
nswmulti := &Polygon{}
for _, p := range nsw.Points() {
nswmulti.Add(p)
}
for _, p := range act.Points() {
nswmulti.Add(p)
}
isnsw = nswmulti.Contains(&canberra)
if isnsw {
t.Error("Canberra should not be in NSW as it falls in the donut contour of the ACT")
}
sydney := Point{lng: 151.209, lat: -33.866}
if !nswmulti.Contains(&sydney) {
t.Error("Sydney should be in NSW")
}
losangeles := Point{lng: 118.28333, lat: 34.01667}
isnsw = nswmulti.Contains(&losangeles)
if isnsw {
t.Error("Los Angeles should not be in NSW")
}
}
// Ensures that jumping over the equator and the greenwich meridian
// Doesn't give us any false positives or false negatives
func TestEquatorGreenwichContains(t *testing.T) {
point1 := NewPoint(0.0, 0.0)
point2 := NewPoint(0.1, 0.1)
point3 := NewPoint(0.1, -0.1)
point4 := NewPoint(-0.1, -0.1)
point5 := NewPoint(-0.1, 0.1)
polygon, err := polygonFromFile("test/data/equator_greenwich.json")
if err != nil {
t.Errorf("error parsing polygon", err)
}
if !polygon.Contains(point1) {
t.Errorf("Should contain middle point of earth")
}
if !polygon.Contains(point2) {
t.Errorf("Should contain point %v", point2)
}
if !polygon.Contains(point3) {
t.Errorf("Should contain point %v", point3)
}
if !polygon.Contains(point4) {
t.Errorf("Should contain point %v", point4)
}
if !polygon.Contains(point5) {
t.Errorf("Should contain point %v", point5)
}
}
// A test struct used to encapsulate and
// Unmarshal JSON into.
type testPoints struct {
Points []*Point
}
// Opens a JSON file and unmarshals the data into a Polygon
func polygonFromFile(filename string) (*Polygon, error) {
p := &Polygon{}
file, err := os.Open(filename)
if err != nil {
return nil, err
}
points := new(testPoints)
jsonParser := json.NewDecoder(file)
if err = jsonParser.Decode(&points); err != nil {
return nil, err
}
for _, point := range points.Points {
p.Add(point)
}
return p, nil
}