-
Notifications
You must be signed in to change notification settings - Fork 16
/
dap.go
170 lines (141 loc) · 4.41 KB
/
dap.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
157
158
159
160
161
162
163
164
165
166
167
168
169
170
package main
import (
"context"
"fmt"
"log"
"net/http"
"github.com/nci/gsky/metrics"
"github.com/nci/gsky/utils"
)
func serveDap(ctx context.Context, conf *utils.Config, r *http.Request, w http.ResponseWriter, query map[string][]string, metricsCollector *metrics.MetricsCollector) {
ceStr := query["dap4.ce"][0]
ce, err := utils.ParseDap4ConstraintExpr(ceStr)
if err != nil {
logDapError(err)
metricsCollector.Info.HTTPStatus = 400
http.Error(w, fmt.Sprintf("Failed to parse dap4.ce: %v", err), 400)
return
}
if *verbose {
utils.DumpDap4CE(ce)
}
wcsParams, err := dapToWcs(ce, conf)
if err != nil {
logDapError(err)
metricsCollector.Info.HTTPStatus = 400
http.Error(w, fmt.Sprintf("Failed to parse dap4.ce: %v", err), 400)
return
}
serveWCS(ctx, *wcsParams, conf, r, w, query, metricsCollector)
}
func dapToWcs(ce *utils.DapConstraints, conf *utils.Config) (*utils.WCSParams, error) {
defaultBbox := []float64{-180, -90, 180, 90}
defaultGeoSize := []int{-1, -1}
wcsParams := &utils.WCSParams{BBox: defaultBbox, Coverages: []string{ce.Dataset}, NoReprojection: true, AxisMapping: 1}
idx, err := utils.GetCoverageIndex(*wcsParams, conf)
if err != nil {
return wcsParams, fmt.Errorf("dataset not found: %v", ce.Dataset)
}
layer := &conf.Layers[idx]
if utils.CheckDisableServices(layer, "dap4") {
return wcsParams, fmt.Errorf("dap4 is disabled for this dataset: %v", ce.Dataset)
}
if len(layer.DefaultGeoBbox) == 4 {
wcsParams.BBox = layer.DefaultGeoBbox
}
if len(layer.DefaultGeoSize) == 2 {
defaultGeoSize = layer.DefaultGeoSize
}
wcsParams.Service = new(string)
wcsParams.Request = new(string)
wcsParams.CRS = new(string)
wcsParams.Version = new(string)
wcsParams.Format = new(string)
wcsParams.Width = new(int)
wcsParams.Height = new(int)
*wcsParams.Service = "WCS"
*wcsParams.Request = "GetCoverage"
*wcsParams.CRS = "EPSG:4326"
*wcsParams.Version = "1.0.0"
*wcsParams.Format = "dap4"
*wcsParams.Width = defaultGeoSize[1]
*wcsParams.Height = defaultGeoSize[0]
var varExpr []string
for _, vp := range ce.VarParams {
if vp.IsAxis {
if vp.Name == "x" {
if len(vp.IdxSelectors) > 0 {
return wcsParams, fmt.Errorf("index-based selection is not supported for axis: %s", vp.Name)
}
isOutRange := *vp.ValStart < defaultBbox[0] || *vp.ValStart > defaultBbox[2]
if !isOutRange {
wcsParams.BBox[0] = *vp.ValStart
}
if vp.ValEnd == nil {
continue
}
isOutRange = *vp.ValEnd < defaultBbox[0] || *vp.ValEnd > defaultBbox[2]
if !isOutRange {
wcsParams.BBox[2] = *vp.ValEnd
}
continue
}
if vp.Name == "y" {
if len(vp.IdxSelectors) > 0 {
return wcsParams, fmt.Errorf("index-based selection is not supported for axis: %s", vp.Name)
}
isOutRange := *vp.ValStart < defaultBbox[1] || *vp.ValStart > defaultBbox[3]
if !isOutRange {
wcsParams.BBox[1] = *vp.ValStart
}
if vp.ValEnd == nil {
continue
}
isOutRange = *vp.ValEnd < defaultBbox[1] || *vp.ValEnd > defaultBbox[3]
if !isOutRange {
wcsParams.BBox[3] = *vp.ValEnd
}
continue
}
var axisParam *utils.AxisParam
if len(vp.IdxSelectors) > 0 {
axisParam = &utils.AxisParam{Name: vp.Name}
axisParam.IdxSelectors = make([]*utils.AxisIdxSelector, len(vp.IdxSelectors))
for i, sel := range vp.IdxSelectors {
axisParam.IdxSelectors[i] = &utils.AxisIdxSelector{Start: sel.Start, End: sel.End, Step: sel.Step, IsRange: sel.IsRange, IsAll: sel.IsAll}
}
} else {
axisParam = &utils.AxisParam{Name: vp.Name, Start: vp.ValStart, End: vp.ValEnd}
}
wcsParams.Axes = append(wcsParams.Axes, axisParam)
} else {
varExpr = append(varExpr, vp.Name)
}
}
if len(varExpr) == 0 {
var specialVars = map[string]struct{}{"x": struct{}{}, "y": struct{}{}}
foundOthers := false
for _, axis := range wcsParams.Axes {
if _, found := specialVars[axis.Name]; !found {
foundOthers = true
break
}
}
if !foundOthers {
return wcsParams, fmt.Errorf("querying special variables (i.e. x, y) is not supported")
}
varExpr = append(varExpr, utils.EmptyTileNS)
if len(ce.VarParams) > 0 {
wcsParams.AxisMapping = 0
}
}
bandExpr, err := utils.ParseBandExpressions(varExpr)
if err != nil {
return wcsParams, fmt.Errorf("parsing error in main variable expressions: %v", err)
}
wcsParams.BandExpr = bandExpr
return wcsParams, nil
}
func logDapError(err error) {
log.Printf("DAP: error: %s", err)
}