59 lines
1.7 KiB
Go
59 lines
1.7 KiB
Go
// Package raster converts rasters to Cloud-Optimized GeoTIFFs and reads their
|
|
// footprints using the GDAL command-line tools (gdal_translate, gdalinfo),
|
|
// which must be installed in the worker environment.
|
|
package raster
|
|
|
|
import (
|
|
"context"
|
|
"encoding/json"
|
|
"fmt"
|
|
"os/exec"
|
|
"strings"
|
|
)
|
|
|
|
// GDALConverter shells out to GDAL.
|
|
type GDALConverter struct {
|
|
compression string
|
|
}
|
|
|
|
// NewGDALConverter returns a converter using DEFLATE compression.
|
|
func NewGDALConverter() *GDALConverter {
|
|
return &GDALConverter{compression: "DEFLATE"}
|
|
}
|
|
|
|
// ToCOG converts the source raster to a Cloud-Optimized GeoTIFF at dst. The COG
|
|
// driver builds internal tiling and overviews.
|
|
func (c *GDALConverter) ToCOG(ctx context.Context, src, dst string) error {
|
|
cmd := exec.CommandContext(ctx, "gdal_translate",
|
|
"-of", "COG",
|
|
"-co", "COMPRESS="+c.compression,
|
|
src, dst,
|
|
)
|
|
var stderr strings.Builder
|
|
cmd.Stderr = &stderr
|
|
if err := cmd.Run(); err != nil {
|
|
return fmt.Errorf("gdal_translate: %w: %s", err, strings.TrimSpace(stderr.String()))
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// Footprint returns the raster's footprint as a GeoJSON polygon in EPSG:4326, or
|
|
// nil if the raster has no spatial reference.
|
|
func (c *GDALConverter) Footprint(ctx context.Context, src string) ([]byte, error) {
|
|
out, err := exec.CommandContext(ctx, "gdalinfo", "-json", src).Output()
|
|
if err != nil {
|
|
return nil, fmt.Errorf("gdalinfo: %w", err)
|
|
}
|
|
|
|
var info struct {
|
|
Wgs84Extent json.RawMessage `json:"wgs84Extent"`
|
|
}
|
|
if err := json.Unmarshal(out, &info); err != nil {
|
|
return nil, fmt.Errorf("parse gdalinfo: %w", err)
|
|
}
|
|
if len(info.Wgs84Extent) == 0 || string(info.Wgs84Extent) == "null" {
|
|
return nil, nil
|
|
}
|
|
return info.Wgs84Extent, nil
|
|
}
|