224 lines
6.9 KiB
Go
224 lines
6.9 KiB
Go
package platform
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"strings"
|
|
"time"
|
|
|
|
"github.com/Gleipnir-Technology/bob"
|
|
"github.com/Gleipnir-Technology/bob/dialect/psql"
|
|
"github.com/Gleipnir-Technology/bob/dialect/psql/sm"
|
|
"github.com/Gleipnir-Technology/nidus-sync/db"
|
|
"github.com/Gleipnir-Technology/nidus-sync/db/models"
|
|
"github.com/Gleipnir-Technology/nidus-sync/db/sql"
|
|
"github.com/google/uuid"
|
|
"github.com/uber/h3-go/v4"
|
|
)
|
|
|
|
type Inspection struct {
|
|
Action string
|
|
Date *time.Time
|
|
Notes string
|
|
Location string
|
|
LocationID uuid.UUID
|
|
}
|
|
|
|
func BreedingSourcesByCell(ctx context.Context, org Organization, c h3.Cell) ([]BreedingSourceSummary, error) {
|
|
boundary, err := c.Boundary()
|
|
if err != nil {
|
|
return nil, fmt.Errorf("Failed to get cell boundary: %w", err)
|
|
}
|
|
geom_query := gisStatement(boundary)
|
|
rows, err := org.model.Pointlocations(
|
|
sm.Where(
|
|
psql.F("ST_Within", "geospatial", geom_query),
|
|
),
|
|
sm.OrderBy("lasttreatdate"),
|
|
).All(ctx, db.PGInstance.BobDB)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("Failed to query rows: %w", err)
|
|
}
|
|
return toBreedingSourceSummary(rows), nil
|
|
}
|
|
func SourceByGlobalID(ctx context.Context, org Organization, id uuid.UUID) (*BreedingSourceDetail, error) {
|
|
row, err := org.model.Pointlocations(
|
|
models.SelectWhere.FieldseekerPointlocations.Globalid.EQ(id),
|
|
).One(ctx, db.PGInstance.BobDB)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("Failed to get point location: %w", err)
|
|
}
|
|
return toBreedingSource(row)
|
|
}
|
|
|
|
func TrapsBySource(ctx context.Context, org Organization, sourceID uuid.UUID) ([]TrapNearby, error) {
|
|
locations, err := sql.TrapLocationBySourceID(org.ID, sourceID).All(ctx, db.PGInstance.BobDB)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("Failed to query rows: %w", err)
|
|
}
|
|
|
|
location_ids := make([]uuid.UUID, 0)
|
|
var args []bob.Expression
|
|
for _, location := range locations {
|
|
location_ids = append(location_ids, location.TrapLocationGlobalid)
|
|
args = append(args, psql.Arg(location.TrapLocationGlobalid))
|
|
}
|
|
trap_data, err := sql.TrapDataByLocationIDRecent(org.ID, location_ids).All(ctx, db.PGInstance.BobDB)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("Failed to query trap data: %w", err)
|
|
}
|
|
|
|
counts, err := sql.TrapCountByLocationID(org.ID, location_ids).All(ctx, db.PGInstance.BobDB)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("Failed to query trap counts: %w", err)
|
|
}
|
|
|
|
traps, err := toTemplateTrapsNearby(locations, trap_data, counts)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("Failed to convert trap data: %w", err)
|
|
}
|
|
return traps, nil
|
|
}
|
|
|
|
func TreatmentsBySource(ctx context.Context, org Organization, sourceID uuid.UUID) ([]Treatment, error) {
|
|
rows, err := org.model.Treatments(
|
|
sm.Where(
|
|
models.FieldseekerTreatments.Columns.Pointlocid.EQ(psql.Arg(sourceID)),
|
|
),
|
|
sm.OrderBy("enddatetime").Desc(),
|
|
).All(ctx, db.PGInstance.BobDB)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("Failed to query rows: %w", err)
|
|
}
|
|
return toTreatment(rows)
|
|
}
|
|
|
|
func TrapByGlobalId(ctx context.Context, org Organization, id uuid.UUID) (*Trap, error) {
|
|
trap_location, err := org.model.Traplocations(
|
|
sm.Where(models.FieldseekerTraplocations.Columns.Globalid.EQ(psql.Arg(id))),
|
|
).One(ctx, db.PGInstance.BobDB)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("Failed to get trap location: %w", err)
|
|
}
|
|
|
|
trap_data, err := sql.TrapDataByLocationIDRecent(org.ID, []uuid.UUID{id}).All(ctx, db.PGInstance.BobDB)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("Failed to query trap data: %w", err)
|
|
}
|
|
|
|
counts, err := sql.TrapCountByLocationID(org.ID, []uuid.UUID{id}).All(ctx, db.PGInstance.BobDB)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("Failed to query trap counts: %w", err)
|
|
}
|
|
result, err := toTrap(trap_location, trap_data, counts)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("to trap: %w", err)
|
|
}
|
|
return &result, err
|
|
}
|
|
|
|
func TrapsByCell(ctx context.Context, org Organization, c h3.Cell) (results []TrapSummary, err error) {
|
|
boundary, err := c.Boundary()
|
|
if err != nil {
|
|
return results, fmt.Errorf("Failed to get cell boundary: %w", err)
|
|
}
|
|
geom_query := gisStatement(boundary)
|
|
rows, err := org.model.Traplocations(
|
|
sm.Where(
|
|
psql.F("ST_Within", "geospatial", geom_query),
|
|
),
|
|
sm.OrderBy("objectid"),
|
|
).All(ctx, db.PGInstance.BobDB)
|
|
if err != nil {
|
|
return results, fmt.Errorf("Failed to query rows: %w", err)
|
|
}
|
|
return toTemplateTrapSummary(rows)
|
|
}
|
|
|
|
func TreatmentsByCell(ctx context.Context, org Organization, c h3.Cell) ([]Treatment, error) {
|
|
var results []Treatment
|
|
boundary, err := c.Boundary()
|
|
if err != nil {
|
|
return results, fmt.Errorf("Failed to get cell boundary: %w", err)
|
|
}
|
|
geom_query := gisStatement(boundary)
|
|
rows, err := org.model.Treatments(
|
|
sm.Where(
|
|
psql.F("ST_Within", "geospatial", geom_query),
|
|
),
|
|
sm.OrderBy("pointlocid"),
|
|
sm.OrderBy("enddatetime"),
|
|
).All(ctx, db.PGInstance.BobDB)
|
|
if err != nil {
|
|
return results, fmt.Errorf("Failed to query rows: %w", err)
|
|
}
|
|
return toTreatment(rows)
|
|
}
|
|
func InspectionsByCell(ctx context.Context, org Organization, c h3.Cell) ([]Inspection, error) {
|
|
var results []Inspection
|
|
|
|
boundary, err := c.Boundary()
|
|
if err != nil {
|
|
return results, fmt.Errorf("Failed to get cell boundary: %w", err)
|
|
}
|
|
geom_query := gisStatement(boundary)
|
|
rows, err := org.model.Mosquitoinspections(
|
|
sm.Where(
|
|
psql.F("ST_Within", "geospatial", geom_query),
|
|
),
|
|
sm.OrderBy("pointlocid"),
|
|
sm.OrderBy("enddatetime"),
|
|
).All(ctx, db.PGInstance.BobDB)
|
|
if err != nil {
|
|
return results, fmt.Errorf("Failed to query rows: %w", err)
|
|
}
|
|
return toTemplateInspection(rows)
|
|
}
|
|
func InspectionsBySource(ctx context.Context, org Organization, sourceID uuid.UUID) ([]Inspection, error) {
|
|
var results []Inspection
|
|
|
|
rows, err := org.model.Mosquitoinspections(
|
|
sm.Where(
|
|
models.FieldseekerMosquitoinspections.Columns.Pointlocid.EQ(psql.Arg(sourceID)),
|
|
),
|
|
sm.OrderBy("enddatetime").Desc(),
|
|
).All(ctx, db.PGInstance.BobDB)
|
|
if err != nil {
|
|
return results, fmt.Errorf("Failed to query rows: %w", err)
|
|
}
|
|
return toTemplateInspection(rows)
|
|
}
|
|
func toBreedingSourceSummary(points []*models.FieldseekerPointlocation) []BreedingSourceSummary {
|
|
results := make([]BreedingSourceSummary, len(points))
|
|
for i, r := range points {
|
|
var last_inspected *time.Time
|
|
if !r.Lastinspectdate.IsNull() {
|
|
l := r.Lastinspectdate.MustGet()
|
|
last_inspected = &l
|
|
}
|
|
var last_treat_date *time.Time
|
|
if !r.Lasttreatdate.IsNull() {
|
|
l := r.Lasttreatdate.MustGet()
|
|
last_treat_date = &l
|
|
}
|
|
results[i] = BreedingSourceSummary{
|
|
ID: r.Globalid,
|
|
LastInspected: last_inspected,
|
|
LastTreated: last_treat_date,
|
|
Type: r.Habitat.GetOr("none"),
|
|
}
|
|
}
|
|
return results
|
|
}
|
|
func gisStatement(cb h3.CellBoundary) string {
|
|
var content strings.Builder
|
|
for i, p := range cb {
|
|
if i != 0 {
|
|
content.WriteString(", ")
|
|
}
|
|
content.WriteString(fmt.Sprintf("%f %f", p.Lng, p.Lat))
|
|
}
|
|
// Repeat the first coordinate to close the polygon
|
|
content.WriteString(fmt.Sprintf(", %f %f", cb[0].Lng, cb[0].Lat))
|
|
return fmt.Sprintf("ST_GeomFromText('POLYGON((%s))', 3857)", content.String())
|
|
}
|