1
0
Fork 0
golang-github-blevesearch-b.../geo/parse_test.go
Daniel Baumann 982828099e
Adding upstream version 2.5.1.
Signed-off-by: Daniel Baumann <daniel@debian.org>
2025-05-19 00:20:02 +02:00

478 lines
11 KiB
Go

// Copyright (c) 2017 Couchbase, Inc.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package geo
import (
"encoding/json"
"reflect"
"testing"
)
func TestExtractGeoPoint(t *testing.T) {
tests := []struct {
in interface{}
lon float64
lat float64
success bool
}{
// values are ints
{
in: map[string]interface{}{
"lat": 5,
"lon": 5,
},
lon: 5,
lat: 5,
success: true,
},
// values are uints
{
in: map[string]interface{}{
"lat": uint(5),
"lon": uint(5),
},
lon: 5,
lat: 5,
success: true,
},
// values float64 as with parsed JSON
{
in: map[string]interface{}{
"lat": 5.0,
"lon": 5.0,
},
lon: 5,
lat: 5,
success: true,
},
// values are bool (not supported)
{
in: map[string]interface{}{
"lat": true,
"lon": false,
},
lon: 0,
lat: 0,
success: false,
},
// using lng variant of lon
{
in: map[string]interface{}{
"lat": 5.0,
"lng": 5.0,
},
lon: 5,
lat: 5,
success: true,
},
// using struct
{
in: struct {
Lon float64
Lat float64
}{
Lon: 3.0,
Lat: 7.5,
},
lon: 3.0,
lat: 7.5,
success: true,
},
// struct with lng alternate
{
in: struct {
Lng float64
Lat float64
}{
Lng: 3.0,
Lat: 7.5,
},
lon: 3.0,
lat: 7.5,
success: true,
},
// test going throug interface
{
in: &s11{
lon: 4.0,
lat: 6.9,
},
lon: 4.0,
lat: 6.9,
success: true,
},
// test going throug interface with lng variant
{
in: &s12{
lng: 4.0,
lat: 6.9,
},
lon: 4.0,
lat: 6.9,
success: true,
},
// try GeoJSON slice
{
in: []interface{}{3.4, 5.9},
lon: 3.4,
lat: 5.9,
success: true,
},
// try GeoJSON slice too long
{
in: []interface{}{3.4, 5.9, 9.4},
lon: 0,
lat: 0,
success: false,
},
// slice of floats
{
in: []float64{3.4, 5.9},
lon: 3.4,
lat: 5.9,
success: true,
},
// values are nil (not supported)
{
in: map[string]interface{}{
"lat": nil,
"lon": nil,
},
lon: 0,
lat: 0,
success: false,
},
// input is nil
{
in: nil,
lon: 0,
lat: 0,
success: false,
},
}
for _, test := range tests {
lon, lat, success := ExtractGeoPoint(test.in)
if success != test.success {
t.Errorf("expected extract geo point %t, got %t for %v", test.success, success, test.in)
}
if lon != test.lon {
t.Errorf("expected lon %f, got %f for %v", test.lon, lon, test.in)
}
if lat != test.lat {
t.Errorf("expected lat %f, got %f for %v", test.lat, lat, test.in)
}
}
}
type s11 struct {
lon float64
lat float64
}
func (s *s11) Lon() float64 {
return s.lon
}
func (s *s11) Lat() float64 {
return s.lat
}
type s12 struct {
lng float64
lat float64
}
func (s *s12) Lng() float64 {
return s.lng
}
func (s *s12) Lat() float64 {
return s.lat
}
func TestExtractGeoShape(t *testing.T) {
tests := []struct {
in interface{}
resTyp string
coordinates [][][][]float64
center []float64
success bool
}{
// valid point slice
{
in: map[string]interface{}{
"coordinates": []interface{}{3.4, 5.9},
"type": "Point",
},
resTyp: "point",
coordinates: [][][][]float64{{{{3.4, 5.9}}}},
success: true,
},
// invalid point slice
{
in: map[string]interface{}{
"coordinates": []interface{}{3.4},
"type": "point"},
resTyp: "point",
coordinates: nil,
success: false,
},
// valid multipoint slice containing single point
{
in: map[string]interface{}{
"coordinates": [][]interface{}{{3.4, 5.9}},
"type": "multipoint"},
resTyp: "multipoint",
coordinates: [][][][]float64{{{{3.4, 5.9}}}},
success: true,
},
// valid multipoint slice
{
in: map[string]interface{}{
"coordinates": [][]interface{}{{3.4, 5.9}, {6.7, 9.8}},
"type": "multipoint"},
resTyp: "multipoint",
coordinates: [][][][]float64{{{{3.4, 5.9}, {6.7, 9.8}}}},
success: true,
},
// valid multipoint slice containing one invalid entry
{
in: map[string]interface{}{
"coordinates": [][]interface{}{{3.4, 5.9}, {6.7}},
"type": "multipoint"},
resTyp: "multipoint",
coordinates: [][][][]float64{{{{3.4, 5.9}}}},
success: true,
},
// invalid multipoint slice
{
in: map[string]interface{}{
"coordinates": [][]interface{}{{3.4}},
"type": "multipoint"},
resTyp: "multipoint",
coordinates: nil,
success: false,
},
// valid linestring slice
{
in: map[string]interface{}{
"coordinates": [][]interface{}{{3.4, 4.4}, {8.4, 9.4}},
"type": "linestring"},
resTyp: "linestring",
coordinates: [][][][]float64{{{{3.4, 4.4}, {8.4, 9.4}}}},
success: true,
},
// valid linestring slice
{
in: map[string]interface{}{
"coordinates": [][]interface{}{{3.4, 4.4}, {8.4, 9.4}, {10.1, 12.3}},
"type": "linestring"},
resTyp: "linestring",
coordinates: [][][][]float64{{{{3.4, 4.4}, {8.4, 9.4}, {10.1, 12.3}}}},
success: true,
},
// invalid linestring slice with single entry
{
in: map[string]interface{}{
"coordinates": [][]interface{}{{3.4, 4.4}},
"type": "linestring"},
resTyp: "linestring",
coordinates: nil,
success: false,
},
// invalid linestring slice with wrong paranthesis
{
in: map[string]interface{}{
"coordinates": [][][]interface{}{{{3.4, 4.4}, {8.4, 9.4}}},
"type": "linestring"},
resTyp: "linestring",
coordinates: nil,
success: false,
},
// valid envelope
{
in: map[string]interface{}{
"coordinates": [][]interface{}{{3.4, 4.4}, {8.4, 9.4}},
"type": "envelope"},
resTyp: "envelope",
coordinates: [][][][]float64{{{{3.4, 4.4}, {8.4, 9.4}}}},
success: true,
},
// invalid envelope
{
in: map[string]interface{}{
"coordinates": [][]interface{}{{3.4, 4.4}},
"type": "envelope"},
resTyp: "envelope",
coordinates: nil,
success: false,
},
// invalid envelope
{
in: map[string]interface{}{
"coordinates": [][][]interface{}{{{3.4, 4.4}, {8.4, 9.4}}},
"type": "envelope"},
resTyp: "envelope",
coordinates: nil,
success: false,
},
// invalid envelope with >2 vertices
{
in: map[string]interface{}{
"coordinates": [][]interface{}{{3.4, 4.4}, {5.6, 6.4}, {7.4, 7.4}},
"type": "envelope"},
resTyp: "envelope",
coordinates: nil,
success: false,
},
// valid circle
{
in: map[string]interface{}{
"coordinates": []interface{}{4.4, 5.0},
"radius": "200m",
"type": "circle"},
resTyp: "circle",
center: []float64{4.4, 5.0},
success: true,
},
// invalid circle
{
in: map[string]interface{}{
"coordinates": []interface{}{4.4},
"radius": "200m",
"type": "circle"},
resTyp: "circle",
success: false,
},
}
for _, test := range tests {
res, success := extractGeoShape(test.in)
if success != test.success {
t.Errorf("expected extract geo point: %t, got: %t for: %v", test.success, success, test.in)
}
if success && res.Type != test.resTyp {
t.Errorf("expected shape type: %v, got: %v for input: %v", test.resTyp, res.Type, test.in)
}
if success && !reflect.DeepEqual(test.coordinates, res.Coordinates) {
t.Errorf("expected result %+v, got %+v for %v", test.coordinates, res.Coordinates, test.in)
}
if success && test.center != nil && !reflect.DeepEqual(test.center, res.Center) {
t.Errorf("expected center %+v, got %+v for %v", test.center, res.Center, test.in)
}
}
}
func TestExtractGeoShapeCoordinates(t *testing.T) {
tests := []struct {
x []byte
typ string
expectOK bool
}{
{
x: []byte(`[
[
[77.58894681930542,12.976498523818783],
[77.58677959442139,12.974533005048169],
[77.58894681930542,12.976498523818783]
]
]`),
typ: PolygonType,
expectOK: true,
},
{ // Invalid construct, but handled
x: []byte(`[
[
{"lon":77.58894681930542,"lat":12.976498523818783},
{"lon":77.58677959442139,"lat":12.974533005048169},
{"lon":77.58894681930542,"lat":12.976498523818783}
]
]`),
typ: PolygonType,
expectOK: false,
},
{ // Invalid construct causes panic (within extract3DCoordinates), fix MB-65807
x: []byte(`{
"coordinates": [
[77.58894681930542,12.976498523818783],
[77.58677959442139,12.974533005048169],
[77.58894681930542,12.976498523818783]
]
}`),
typ: PolygonType,
expectOK: false,
},
{
x: []byte(`[
[
[
[-0.163421630859375,51.531600743186644],
[-0.15277862548828125,51.52455221546295],
[-0.15895843505859375,51.53693981046689],
[-0.163421630859375,51.531600743186644]
]
],
[
[
[-0.1902008056640625,51.5091698216777],
[-0.1599884033203125,51.51322956905176],
[-0.1902008056640625,51.5091698216777]
]
]
]`),
typ: MultiPolygonType,
expectOK: true,
},
{ // Invalid construct causes panic (within extract3DCoordinates), fix MB-65807
x: []byte(`[
{
"coordinates": [
[-0.163421630859375,51.531600743186644],
[-0.15277862548828125,51.52455221546295],
[-0.15895843505859375,51.53693981046689],
[-0.163421630859375,51.531600743186644]
]
},
{
"coordinates": [
[-0.1902008056640625,51.5091698216777],
[-0.1599884033203125,51.51322956905176],
[-0.1902008056640625,51.5091698216777]
]
}
]`),
typ: MultiPolygonType,
expectOK: false,
},
}
for i := range tests {
var x interface{}
if err := json.Unmarshal(tests[i].x, &x); err != nil {
t.Fatalf("[%d] JSON err: %v", i+1, err)
}
res, ok := ExtractGeoShapeCoordinates(x, tests[i].typ)
if ok != tests[i].expectOK {
t.Errorf("[%d] expected ok %t, got %t", i+1, tests[i].expectOK, ok)
}
if ok && res.Type != tests[i].typ {
t.Errorf("[%d] expected type %s, got %s", i+1, tests[i].typ, res.Type)
}
}
}