Files

266 lines
5.2 KiB
Go
Raw Permalink Normal View History

2024-06-22 09:05:06 -07:00
package main
2024-06-22 09:39:26 -07:00
import (
2024-06-22 21:51:00 -07:00
"context"
2024-06-22 09:39:26 -07:00
"encoding/csv"
"flag"
"fmt"
"io"
"log/slog"
"os"
"slices"
"strings"
2024-06-22 21:51:00 -07:00
"github.com/flamingcow66/airtable"
2024-06-22 09:39:26 -07:00
)
2024-06-22 11:12:27 -07:00
type Directory struct {
Students map[string]*Student
Parents map[string]*Parent
}
2024-06-22 09:39:26 -07:00
type Person struct {
2024-06-22 11:12:27 -07:00
Email string
2024-06-22 12:18:08 -07:00
Name string
2024-06-22 09:39:26 -07:00
}
type Student struct {
Person
2024-06-22 11:12:27 -07:00
Class string
Grade string
Parents []*Parent
}
type Parent struct {
Person
2024-06-22 09:39:26 -07:00
}
2024-06-22 09:05:06 -07:00
func main() {
2024-06-22 09:39:26 -07:00
directoryPath := flag.String("directory", "", "path to directory CSV file")
flag.Parse()
l := slog.Default()
if *directoryPath == "" {
fatal(l, "please pass --directory")
}
2024-06-22 11:12:27 -07:00
dir, err := loadDirectory(l, *directoryPath)
2024-06-22 09:39:26 -07:00
if err != nil {
fatal(l, "failed to load directory", "error", err)
}
2024-06-22 11:12:27 -07:00
l.Info("loaded directory", "directory", dir)
2024-06-22 12:18:08 -07:00
2024-06-22 21:51:00 -07:00
at, err := airtable.NewFromEnv()
2024-06-22 12:18:08 -07:00
if err != nil {
fatal(l, "failed to create airtable client", "error", err)
}
2024-06-22 21:51:00 -07:00
dirBase, err := at.GetBaseByName(context.Background(), "Directory")
2024-06-22 12:18:08 -07:00
if err != nil {
fatal(l, "failed to find Directory base in Airtable", "error", err)
}
2024-06-22 21:51:00 -07:00
parentsTable, err := dirBase.GetTableByName(context.Background(), "Parents")
2024-06-22 12:18:08 -07:00
if err != nil {
fatal(l, "failed to get Parents table", "error", err)
}
2024-06-23 16:32:39 -07:00
parents, err := parentsTable.ReplaceRecords(context.Background(), dir.GetParentRecords(), []string{"Email"})
2024-06-22 12:18:08 -07:00
if err != nil {
2024-06-23 16:32:39 -07:00
fatal(l, "failed to upsert Parents", "error", err)
2024-06-22 12:18:08 -07:00
}
2024-06-23 16:32:39 -07:00
l.Info("upserted parents", "parents", parents)
2024-06-22 09:39:26 -07:00
}
2024-06-22 11:12:27 -07:00
func loadDirectory(l *slog.Logger, path string) (*Directory, error) {
d := &Directory{
Students: map[string]*Student{},
Parents: map[string]*Parent{},
}
2024-06-22 09:39:26 -07:00
fh, err := os.Open(path)
if err != nil {
return nil, err
}
r := csv.NewReader(fh)
headers, err := r.Read()
if err != nil {
return nil, err
}
l.Info("headers", "headers", headers)
iStudentFirstName := slices.Index(headers, "First Name")
if iStudentFirstName == -1 {
return nil, fmt.Errorf("'First Name' field missing")
}
iStudentLastName := slices.Index(headers, "Last Name")
if iStudentLastName == -1 {
return nil, fmt.Errorf("'Last Name' field missing")
}
iStudentClass := slices.Index(headers, "Class")
if iStudentClass == -1 {
return nil, fmt.Errorf("'Class' field missing")
}
iStudentGrade := slices.Index(headers, "Grade")
if iStudentClass == -1 {
return nil, fmt.Errorf("'Grade' field missing")
}
2024-06-22 11:12:27 -07:00
iParent1FirstName := slices.Index(headers, "Parent 1\nFirst")
if iParent1FirstName == -1 {
return nil, fmt.Errorf("'Parent 1\\nFirst' field missing")
}
iParent1LastName := slices.Index(headers, "Parent 1\nLast")
if iParent1LastName == -1 {
return nil, fmt.Errorf("'Parent 1\\nLast' field missing")
}
iParent1Email := slices.Index(headers, "Parent 1\nEmail")
if iParent1Email == -1 {
return nil, fmt.Errorf("'Parent 1\\nEmail' field missing")
}
iParent2FirstName := slices.Index(headers, "Parent 2\nFirst")
if iParent2FirstName == -1 {
return nil, fmt.Errorf("'Parent 2\\nFirst' field missing")
}
iParent2LastName := slices.Index(headers, "Parent 2\nLast")
if iParent2LastName == -1 {
return nil, fmt.Errorf("'Parent 2\\nLast' field missing")
}
iParent2Email := slices.Index(headers, "Parent 2 Email")
if iParent2Email == -1 {
return nil, fmt.Errorf("'Parent 2 Email' field missing")
}
2024-06-22 09:39:26 -07:00
for {
row, err := r.Read()
if err == io.EOF {
break
} else if err != nil {
return nil, err
}
class := row[iStudentClass]
if strings.Contains(class, "/") {
class = ""
}
2024-06-22 11:12:27 -07:00
parents := []*Parent{}
parent1 := d.AddParent(
2024-06-22 12:18:08 -07:00
strings.ToLower(row[iParent1Email]),
2024-06-22 11:12:27 -07:00
fmt.Sprintf("%s %s", row[iParent1FirstName], row[iParent1LastName]),
)
parents = append(parents, parent1)
if row[iParent2Email] != "" {
parent2 := d.AddParent(
2024-06-22 12:18:08 -07:00
strings.ToLower(row[iParent2Email]),
2024-06-22 11:12:27 -07:00
fmt.Sprintf("%s %s", row[iParent2FirstName], row[iParent2LastName]),
)
parents = append(parents, parent2)
}
2024-06-22 09:39:26 -07:00
studentEmail := fmt.Sprintf(
"%s.%s@heliosschool.org",
strings.ToLower(row[iStudentFirstName]),
strings.ToLower(row[iStudentLastName]),
)
2024-06-22 11:12:27 -07:00
d.AddStudent(
studentEmail,
fmt.Sprintf("%s %s", row[iStudentFirstName], row[iStudentLastName]),
class,
row[iStudentGrade],
parents,
)
2024-06-22 09:39:26 -07:00
}
2024-06-22 11:12:27 -07:00
return d, nil
2024-06-22 09:39:26 -07:00
}
func fatal(l *slog.Logger, msg string, args ...any) {
l.Error(msg, args...)
os.Exit(1)
2024-06-22 09:05:06 -07:00
}
2024-06-22 11:12:27 -07:00
func (d *Directory) AddParent(email, name string) *Parent {
p := d.Parents[email]
if p != nil {
return p
}
p = &Parent{
Person: Person{
Email: email,
2024-06-22 12:18:08 -07:00
Name: name,
2024-06-22 11:12:27 -07:00
},
}
d.Parents[p.Email] = p
return p
}
func (d *Directory) AddStudent(email, name, class, grade string, parents []*Parent) *Student {
s := &Student{
Person: Person{
Email: email,
2024-06-22 12:18:08 -07:00
Name: name,
2024-06-22 11:12:27 -07:00
},
Class: class,
Grade: grade,
Parents: parents,
}
d.Students[email] = s
return s
}
2024-06-23 16:32:39 -07:00
func (d *Directory) GetParentRecords() []*airtable.Record {
records := []*airtable.Record{}
2024-06-22 12:18:08 -07:00
for _, parent := range d.Parents {
2024-06-23 16:32:39 -07:00
records = append(records, &airtable.Record{
2024-06-22 12:18:08 -07:00
Fields: map[string]any{
"Email": parent.Email,
"Name": parent.Name,
},
})
}
return records
}
2024-06-22 11:12:27 -07:00
func (p Person) String() string {
return fmt.Sprintf(
"%s <%s>",
p.Name,
p.Email,
)
}
func (s Student) String() string {
return s.Person.String()
}
func (p Parent) String() string {
return p.Person.String()
}