nidus-sync/platform/compliance.go
Eli Ribble fe2041f22b
Add an evidence field to compliance reports
This allows us to show a page with information about what the district
is concerned about when asking the user to fill a report.
2026-04-21 21:35:40 +00:00

135 lines
4.3 KiB
Go

package platform
import (
"context"
"fmt"
"strconv"
"time"
"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/platform/background"
"github.com/Gleipnir-Technology/nidus-sync/platform/event"
"github.com/Gleipnir-Technology/nidus-sync/platform/types"
"github.com/aarondl/opt/omit"
"github.com/aarondl/opt/omitnull"
//"github.com/rs/zerolog/log"
)
func ComplianceRequestMailerCreate(ctx context.Context, user User, site_id int32) (int32, error) {
txn, err := db.PGInstance.BobDB.BeginTx(ctx, nil)
if err != nil {
return 0, fmt.Errorf("start txn: %w", err)
}
defer txn.Rollback(ctx)
site, err := models.FindSite(ctx, txn, site_id)
if err != nil {
return 0, fmt.Errorf("find site: %w", err)
}
if site.OrganizationID != user.Organization.ID {
return 0, fmt.Errorf("permission denied")
}
features, err := models.Features.Query(
models.SelectWhere.Features.SiteID.EQ(site.ID),
).All(ctx, txn)
if err != nil {
return 0, fmt.Errorf("find features: %w", err)
}
feature_ids := make([]int32, len(features))
for i, f := range features {
feature_ids[i] = f.ID
}
feature_pools, err := models.FeaturePools.Query(
sm.Where(
models.FeaturePools.Columns.FeatureID.EQ(psql.Any(feature_ids)),
),
).All(ctx, txn)
if err != nil {
return 0, fmt.Errorf("find feature pools: %w", err)
}
if len(feature_pools) != 1 {
return 0, fmt.Errorf("wrong number of pools: %d", len(feature_pools))
}
feature_pool := feature_pools[0]
var feature *models.Feature
for _, f := range features {
if f.ID == feature_pool.FeatureID {
feature = f
}
}
if feature == nil {
return 0, fmt.Errorf("match feature %d", feature_pool.FeatureID)
}
location := types.Location{
Latitude: feature.LocationLatitude.GetOr(0),
Longitude: feature.LocationLongitude.GetOr(0),
}
signal, err := SignalCreateFromPool(ctx, txn, user, site.ID, feature_pool.FeatureID, location)
if err != nil {
return 0, fmt.Errorf("create signal from ppol: %w", err)
}
lead_id, err := leadCreate(ctx, txn, user, *signal, site.ID, &location)
if err != nil {
return 0, fmt.Errorf("create lead from ppol: %w", err)
}
public_id, err := GenerateReportID()
if err != nil {
return 0, fmt.Errorf("create public id: %w", err)
}
setter := models.ComplianceReportRequestSetter{
Created: omit.From(time.Now()),
Creator: omit.From(int32(user.ID)),
// ID
PublicID: omit.From(public_id),
LeadID: omitnull.From(*lead_id),
}
req, err := models.ComplianceReportRequests.Insert(&setter).One(ctx, txn)
if err != nil {
return 0, fmt.Errorf("create compliance report request: %w", err)
}
err = background.NewComplianceMailer(ctx, txn, req.ID)
if err != nil {
return 0, fmt.Errorf("create background compliance mailer job: %w", err)
}
event.Updated(event.TypeSite, user.Organization.ID, strconv.Itoa(int(site.ID)))
txn.Commit(ctx)
return req.ID, nil
}
func ComplianceReportRequestByLeadID(ctx context.Context, lead_ids []int32) (map[int32][]*types.ComplianceReportRequest, error) {
rows, err := models.ComplianceReportRequests.Query(
sm.Where(models.ComplianceReportRequests.Columns.LeadID.EQ(psql.Any(lead_ids))),
).All(ctx, db.PGInstance.BobDB)
if err != nil {
return nil, fmt.Errorf("query reports: %w", err)
}
results := make(map[int32][]*types.ComplianceReportRequest, len(lead_ids))
for _, lead_id := range lead_ids {
results[lead_id] = make([]*types.ComplianceReportRequest, 0)
}
for _, row := range rows {
lead_id := row.LeadID.MustGet()
crrs, ok := results[lead_id]
if !ok {
return nil, fmt.Errorf("impossible")
}
crrs = append(crrs, types.ComplianceReportRequestFromModel(row))
results[lead_id] = crrs
}
return results, nil
}
func ComplianceReportRequestFromPublicID(ctx context.Context, public_id string) (*types.ComplianceReportRequest, error) {
row, err := models.ComplianceReportRequests.Query(
sm.Where(models.ComplianceReportRequests.Columns.PublicID.EQ(psql.Arg(public_id))),
).One(ctx, db.PGInstance.BobDB)
if err != nil {
if err.Error() == "sql: no rows in result set" {
return nil, nil
}
return nil, fmt.Errorf("query CRR: %w", err)
}
return types.ComplianceReportRequestFromModel(row), nil
}