Skip to content

Commit

Permalink
HANA: Implement NewMVTTileProvider
Browse files Browse the repository at this point in the history
  • Loading branch information
mrylov committed Jan 16, 2023
1 parent 2c15e12 commit 67b9be1
Show file tree
Hide file tree
Showing 6 changed files with 422 additions and 4 deletions.
132 changes: 132 additions & 0 deletions mvtprovider/hana/README.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,132 @@
# HANA MVT Provider

The HANA MVT provider manages querying for tile requests against a HANA database with [Vector Tiles support](https://help.sap.com/docs/HANA_CLOUD_DATABASE/bc9e455fe75541b8a248b4c09b086cf5/8cd683c4bb664fd8a71fc3f19ffa7e42.html) to handle the MVT encoding at the database.

The connection between tegola and HANA is configured in a `tegola.toml` file. An example minimum connection config:


```toml
[[providers]]
name = "test_hana" # provider name is referenced from map layers (required)
type = "mvt_hana" # the type of data provider must be "mvt_hana" for this data provider (required)
uri = "hdb://myuser:mypassword@something.hanacloud.ondemand.com:443?" # HANA connection string (required)
```

### Connection Properties

- `uri` (string): [Required] HANA connection string
- `name` (string): [Required] provider name is referenced from map layers
- `type` (string): [Required] the type of data provider. must be "hana" to use this data provider
- `srid` (int): [Optional] The default SRID for the provider. Defaults to WebMercator (3857) but also supports WGS84 (4326)

#### Connection string properties

**Example**

```
# {protocol}://{user}:{password}@{host}:{port}/{database}?{options}=
hdb://myuser:mypassword@something.hanacloud.ondemand.com:443?TLSInsecureSkipVerify&timeout=3600&max_connections=30
```

**Options**

- `timeout`: [Optional] Driver side connection timeout in seconds.
- `TLSRootCAFile` [Optional] Path,- filename to root certificate(s).
- `TLSServerName` [Optional] ServerName to verify the hostname. By setting TLSServerName=host, the provider will set TLSServerName same as 'host' value in `uri`.
- `TLSInsecureSkipVerify` [Optional] Controls whether a client verifies the server's certificate chain and host name.
- `max_connections`: [Optional] The max connections to maintain in the connection pool. Defaults to 100. 0 means no max.
- `max_connection_idle_time`: [Optional] The maximum time an idle connection is kept alive. Defaults to "30m".
- `max_connection_life_time` [Optional] The maximum time a connection lives before it is terminated and recreated. Defaults to "1h".

## Provider Layers

In addition to the connection configuration above, Provider Layers need to be configured. A Provider Layer tells tegola how to query HANA for a certain layer. When using the HANA MVT Provider the `ST_AsMVTGeom()` MUST be used. An example minimum config using the `sql` config option:

```toml
[[providers.layers]]
name = "landuse"
# MVT data provider can use both table names and SQL statements. Internally, the provider wraps an SQL query by using
# ST_AsMVT and ST_AsMVTGeom functions.
tablename = "gis.zoning_base_3857"
```

### Provider Layers Properties

- `name` (string): [Required] the name of the layer. This is used to reference this layer from map layers.
- `geometry_fieldname` (string): [Optional] the name of the filed which contains the geometry for the feature. Defaults to `geom`.
- `id_fieldname` (string): [Optional] the name of the feature id field. defaults to `gid`.
- `geometry_type` (string): [Optional] the layer geometry type. If not set, the table will be inspected at startup to try and infer the gemetry type. Valid values are: `Point`, `LineString`, `Polygon`, `MultiPoint`, `MultiLineString`, `MultiPolygon`, `GeometryCollection`.
- `srid` (int): [Optional] the SRID of the layer. Supports `3857` (WebMercator) only.
- `sql` (string): [Required] custom SQL to use use. Supports the following tokens:
- `!BBOX!` - [Required] will be replaced with the bounding box of the tile before the query is sent to the database. `!bbox!` and`!BOX!` are supported as well for compatibilitiy with queries from Mapnik and MapServer styles.
- `!X!` - [Optional] will replaced with the "X" value of the requested tile.
- `!Y!` - [Optional] will replaced with the "Y" value of the requested tile.
- `!Z!` - [Optional] will replaced with the "Z" value of the requested tile.
- `!ZOOM!` - [Optional] will be replaced with the "Z" (zoom) value of the requested tile.
- `!SCALE_DENOMINATOR!` - [Optional] scale denominator, assuming 90.7 DPI (i.e. 0.28mm pixel size)
- `!PIXEL_WIDTH!` - [Optional] the pixel width in meters, assuming 256x256 tiles
- `!PIXEL_HEIGHT!` - [Optional] the pixel height in meters, assuming 256x256 tiles
- `!ID_FIELD!` - [Optional] the id field name
- `!GEOM_FIELD!` - [Optional] the geom field name
- `!GEOM_TYPE!` - [Optional] the geom type if defined otherwise ""
- `buffer` (int): [Optional] the buffer distance by which the clipped geometry may exceed the tile's area. Defaults to 256.
- `clip_geometry` (bool): [Optional] the flag to control whether the geometry is clipped to the tile bounds or not. Defaults to `TRUE`.

## Example mvt_hana and map config

```toml
[[providers]]
name = "test_hana"
type = "mvt_hana"
uri = "hdb://myuser:mypassword@something.hanacloud.ondemand.com:443?" # HANA connection string (required)
srid = 3857 # The only supported srid is 3857 (optional)

[[providers.layers]]
name = "landuse"
sql = "SELECT geom, gid FROM gis.landuse WHERE !BBOX!"

[[maps]]
name = "cities"
center = [-90.2,38.6,3.0] # where to center of the map (lon, lat, zoom)

[[maps.layers]]
name = "landuse"
provider_layer = "test_hana.landuse"
min_zoom = 0
max_zoom = 14
```

## Example mvt_hana and map config for SRID 4326

When using a 4326 projection with ST_AsMVT the SQL statement needs to be modified. `ST_AsMVTGeom` is expecting data in 3857 projection so the geometries and the `!BBOX!` token need to be transformed prior to `ST_AsMVTGeom` processing them. For example:

```toml
[[providers]]
name = "test_hana"
type = "mvt_hana"
uri = "hdb://myuser:mypassword@something.hanacloud.ondemand.com:443?" # HANA connection string (required)
srid = 3857 # The only supported srid is 3857 (optional)

[[providers.layers]]
name = "landuse"
sql = "SELECT * FROM (SELECT id, name, geom.ST_Transform(3857) AS geom FROM ne_50m_rivers_lake_centerlines) AS sub WHERE !BBOX!"

[[maps]]
name = "cities"
center = [-90.2,38.6,3.0] # where to center of the map (lon, lat, zoom)

[[maps.layers]]
name = "landuse"
provider_layer = "test_hana.landuse"
min_zoom = 0
max_zoom = 14
```

## Testing

Testing is designed to work against a live SAP HANA database. To see how to set up a database check this [github actions script](https://github.com/go-spatial/tegola/blob/master/.github/worksflows/on_pr_push.yml). To run the HANA tests, the following environment variables need to be set:

```bash
$ export RUN_HANA_TESTS=yes
$ export HANA_CONNECTION_STRING="hdb://myuser:mypassword@something.hanacloud.ondemand.com:443?TLSInsecureSkipVerify"
```
5 changes: 5 additions & 0 deletions mvtprovider/hana/doc.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
// Package hana is a placeholder for the SAP HANA database.
// The hana provider is provided by the provider.Hana provider
// please consult that for information on how to use it.
// please note that the provider type will be mvt_hana
package hana
133 changes: 131 additions & 2 deletions provider/hana/hana.go
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
package hana

import (
"bytes"
"context"
"database/sql"
"fmt"
Expand Down Expand Up @@ -140,7 +141,8 @@ type Provider struct {
collectorsRegistered bool

// Collectors for Query times
queryHistogramSeconds *prometheus.HistogramVec
mvtProviderQueryHistogramSeconds *prometheus.HistogramVec
queryHistogramSeconds *prometheus.HistogramVec
}

func (p *Provider) Collectors(prefix string, cfgFn func(configKey string) map[string]interface{}) ([]observability.Collector, error) {
Expand All @@ -154,6 +156,15 @@ func (p *Provider) Collectors(prefix string, cfgFn func(configKey string) map[st
return nil, err
}

p.mvtProviderQueryHistogramSeconds = prometheus.NewHistogramVec(
prometheus.HistogramOpts{
Name: prefix + "_mvt_provider_sql_query_seconds",
Help: "A histogram of query time for sql for mvt providers",
Buckets: buckets,
},
[]string{"map_name", "z"},
)

p.queryHistogramSeconds = prometheus.NewHistogramVec(
prometheus.HistogramOpts{
Name: prefix + "_provider_sql_query_seconds",
Expand All @@ -164,7 +175,7 @@ func (p *Provider) Collectors(prefix string, cfgFn func(configKey string) map[st
)

p.collectorsRegistered = true
return append(collectors, p.queryHistogramSeconds), nil
return append(collectors, p.mvtProviderQueryHistogramSeconds, p.queryHistogramSeconds), nil
}

const (
Expand Down Expand Up @@ -363,6 +374,9 @@ func CreateProvider(config dict.Dicter, maps []provider.Map, providerType string
}

majorVersion, _ := strconv.Atoi(strings.Split(dbVersion, ".")[0])
if providerType == MVTProviderType && majorVersion < 4 {
return nil, fmt.Errorf("MVT provider is only available in HANA Cloud")
}

srid := -1
if srid, err = config.Int(ConfigKeySRID, &srid); err != nil {
Expand Down Expand Up @@ -538,6 +552,23 @@ func CreateProvider(config dict.Dicter, maps []provider.Map, providerType string
}
}

if providerType == MVTProviderType {
var buffer uint = 256
if buffer, err = layer.Uint(ConfigKeyBuffer, &buffer); err != nil {
return nil, err
}

var clipGeom bool = true
if clipGeom, err = layer.Bool(ConfigKeyClipGeometry, &clipGeom); err != nil {
return nil, err
}

l.sql, err = genMVTSQL(&l, getFieldNames(l.fields), buffer, clipGeom)
if err != nil {
return nil, err
}
}

if debugLayerSQL {
log.Debugf("SQL for Layer(%v):\n%v\n", lName, l.sql)
}
Expand Down Expand Up @@ -800,6 +831,104 @@ func (p Provider) TileFeatures(ctx context.Context, layer string, tile provider.
return rows.Err()
}

func (p Provider) MVTForLayers(ctx context.Context, tile provider.Tile, params provider.Params, layers []provider.Layer) ([]byte, error) {
var mapName string

{
mapNameVal := ctx.Value(observability.ObserveVarMapName)
if mapNameVal != nil {
// if it's not convertible to a string, we will ignore it.
mapName, _ = mapNameVal.(string)
}
}

args := make([]interface{}, 0)
var mvtBytes bytes.Buffer
var totalSeconds float64
totalSeconds = 0.0

for i := range layers {
layer := layers[i]
if debug {
log.Debugf("looking for layer: %v", layer)
}
l, ok := p.Layer(layer.Name)

if !ok {
// Should we error here, or have a flag so that we don't
// spam the user?
log.Warnf("provider layer not found %v", layer.Name)
}
if debugLayerSQL {
log.Debugf("SQL for Layer(%v):\n%v\n", l.Name(), l.sql)
}

sqlQuery, err := replaceTokens(p.dbVersion, l.sql, l.IDFieldName(), l.GeomFieldName(), l.GeomType(), l.SRID(), tile, false)
if err := ctxErr(ctx, err); err != nil {
return nil, err
}

// replace configured query parameters if any
sqlQuery = params.ReplaceParams(sqlQuery, &args)

now := time.Now()

extent, _ := getTileExtent(tile, false)
srid := l.SRID()
rows, err := p.pool.QueryContextWithBBox(ctx, sqlQuery, extent, srid, true)

if err := ctxErr(ctx, err); err != nil {
return []byte{}, err
}

defer rows.Close()

if err := ctxErr(ctx, err); err != nil {
return []byte{}, err
}

lob := &driver.Lob{}
lob.SetWriter(new(bytes.Buffer))

if rows.Next() {
if err := ctx.Err(); err != nil {
return []byte{}, err
}

err = rows.Scan(lob)
if err := ctxErr(ctx, err); err != nil {
return []byte{}, err
}

mvtBytes.Write(lob.Writer().(*bytes.Buffer).Bytes())
} else {
return nil, fmt.Errorf("unable to read the result set for layer (%v)", l.Name())
}

totalSeconds += time.Since(now).Seconds()

if debugExecuteSQL {
log.Debugf("%s:%s: %v", EnvSQLDebugName, EnvSQLDebugExecute, sqlQuery)
if err != nil {
log.Errorf("%s:%s: returned error %v", EnvSQLDebugName, EnvSQLDebugExecute, err)
} else {
log.Debugf("%s:%s: returned %v bytes", EnvSQLDebugName, EnvSQLDebugExecute, lob.Writer().(*bytes.Buffer).Len())
}
}
}

if p.mvtProviderQueryHistogramSeconds != nil {
z, _, _ := tile.ZXY()
lbls := prometheus.Labels{
"z": strconv.FormatUint(uint64(z), 10),
"map_name": mapName,
}
p.mvtProviderQueryHistogramSeconds.With(lbls).Observe(totalSeconds)
}

return mvtBytes.Bytes(), nil
}

// Close will close the Provider's database connectio
func (p *Provider) Close() { p.pool.Close() }

Expand Down
Loading

0 comments on commit 67b9be1

Please sign in to comment.