Files
automana/asanarules/rules.go
2021-09-19 06:49:00 +00:00

474 lines
9.5 KiB
Go

package asanarules
import "bytes"
import "fmt"
import "math/rand"
import "strings"
import "time"
import "cloud.google.com/go/civil"
import "github.com/firestuff/asana-rules/asanaclient"
import "golang.org/x/net/html"
import "golang.org/x/net/html/atom"
type queryMutator func(*asanaclient.WorkspaceClient, *asanaclient.SearchQuery) error
type taskActor func(*asanaclient.WorkspaceClient, *asanaclient.Task) error
type taskFilter func(*asanaclient.WorkspaceClient, *asanaclient.Task) (bool, error)
type workspaceClientGetter func(*asanaclient.Client) (*asanaclient.WorkspaceClient, error)
type periodic struct {
period int
done chan bool
workspaceClientGetter workspaceClientGetter
queryMutators []queryMutator
taskFilters []taskFilter
taskActors []taskActor
}
var periodics = []*periodic{}
func EverySeconds(seconds int) *periodic {
ret := &periodic{
period: seconds,
done: make(chan bool),
}
periodics = append(periodics, ret)
return ret
}
func Loop() {
client := asanaclient.NewClientFromEnv()
for _, periodic := range periodics {
periodic.start(client)
}
for _, periodic := range periodics {
periodic.wait()
}
}
func (p *periodic) InWorkspace(name string) *periodic {
if p.workspaceClientGetter != nil {
panic("Multiple calls to InWorkspace()")
}
p.workspaceClientGetter = func(c *asanaclient.Client) (*asanaclient.WorkspaceClient, error) {
return c.InWorkspace(name)
}
return p
}
// Query mutators
func (p *periodic) InMyTasksSections(names ...string) *periodic {
p.queryMutators = append(p.queryMutators, func(wc *asanaclient.WorkspaceClient, q *asanaclient.SearchQuery) error {
utl, err := wc.GetMyUserTaskList()
if err != nil {
return err
}
secsByName, err := wc.GetSectionsByName(utl)
if err != nil {
return err
}
for _, name := range names {
sec, found := secsByName[name]
if !found {
return fmt.Errorf("Section '%s' not found", name)
}
q.SectionsAny = append(q.SectionsAny, sec)
}
return nil
})
return p
}
func (p *periodic) DueInDays(days int) *periodic {
p.queryMutators = append(p.queryMutators, func(wc *asanaclient.WorkspaceClient, q *asanaclient.SearchQuery) error {
if q.DueOn != nil {
return fmt.Errorf("Multiple clauses set DueOn")
}
d := civil.DateOf(time.Now())
d = d.AddDays(days)
q.DueOn = &d
return nil
})
return p
}
func (p *periodic) DueInAtLeastDays(days int) *periodic {
p.queryMutators = append(p.queryMutators, func(wc *asanaclient.WorkspaceClient, q *asanaclient.SearchQuery) error {
if q.DueAfter != nil {
return fmt.Errorf("Multiple clauses set DueAfter")
}
d := civil.DateOf(time.Now())
d = d.AddDays(days)
q.DueAfter = &d
return nil
})
return p
}
func (p *periodic) DueInAtMostDays(days int) *periodic {
p.queryMutators = append(p.queryMutators, func(wc *asanaclient.WorkspaceClient, q *asanaclient.SearchQuery) error {
if q.DueBefore != nil {
return fmt.Errorf("Multiple clauses set DueBefore")
}
d := civil.DateOf(time.Now())
d = d.AddDays(days)
q.DueBefore = &d
return nil
})
return p
}
func (p *periodic) OnlyIncomplete() *periodic {
p.queryMutators = append(p.queryMutators, func(wc *asanaclient.WorkspaceClient, q *asanaclient.SearchQuery) error {
if q.Completed != nil {
return fmt.Errorf("Multiple clauses set Completed")
}
q.Completed = asanaclient.FALSE
return nil
})
return p
}
func (p *periodic) OnlyComplete() *periodic {
p.queryMutators = append(p.queryMutators, func(wc *asanaclient.WorkspaceClient, q *asanaclient.SearchQuery) error {
if q.Completed != nil {
return fmt.Errorf("Multiple clauses set Completed")
}
q.Completed = asanaclient.TRUE
return nil
})
return p
}
func (p *periodic) WithTagsAnyOf(names ...string) *periodic {
p.queryMutators = append(p.queryMutators, func(wc *asanaclient.WorkspaceClient, q *asanaclient.SearchQuery) error {
if len(q.TagsAny) > 0 {
return fmt.Errorf("Multiple clauses set TagsAny")
}
tagsByName, err := wc.GetTagsByName()
if err != nil {
return err
}
for _, name := range names {
tag, found := tagsByName[name]
if !found {
return fmt.Errorf("Tag '%s' not found", name)
}
q.TagsAny = append(q.TagsAny, tag)
}
return nil
})
return p
}
func (p *periodic) WithoutTagsAnyOf(names ...string) *periodic {
p.queryMutators = append(p.queryMutators, func(wc *asanaclient.WorkspaceClient, q *asanaclient.SearchQuery) error {
if len(q.TagsNot) > 0 {
return fmt.Errorf("Multiple clauses set TagsNot")
}
tagsByName, err := wc.GetTagsByName()
if err != nil {
return err
}
for _, name := range names {
tag, found := tagsByName[name]
if !found {
return fmt.Errorf("Tag '%s' not found", name)
}
q.TagsNot = append(q.TagsNot, tag)
}
return nil
})
return p
}
// Task filters
func (p *periodic) WithUnlinkedURL() *periodic {
p.taskFilters = append(p.taskFilters, func(wc *asanaclient.WorkspaceClient, t *asanaclient.Task) (bool, error) {
return hasUnlinkedURL(t.ParsedHTMLNotes), nil
})
return p
}
func (p *periodic) WithoutDue() *periodic {
// We can't mutate the query because due_on=null is buggy in the Asana API
p.taskFilters = append(p.taskFilters, func(wc *asanaclient.WorkspaceClient, t *asanaclient.Task) (bool, error) {
return t.ParsedDueOn == nil, nil
})
return p
}
// Task actors
func (p *periodic) FixUnlinkedURL() *periodic {
p.taskActors = append(p.taskActors, func(wc *asanaclient.WorkspaceClient, t *asanaclient.Task) error {
fixUnlinkedURL(t.ParsedHTMLNotes)
buf := &bytes.Buffer{}
err := html.Render(buf, t.ParsedHTMLNotes)
if err != nil {
return err
}
notes := buf.String()
update := &asanaclient.Task{
GID: t.GID,
HTMLNotes: strings.TrimSuffix(strings.TrimPrefix(notes, "<html><head></head>"), "</html>"),
}
return wc.UpdateTask(update)
})
return p
}
func (p *periodic) MoveToMyTasksSection(name string) *periodic {
p.taskActors = append(p.taskActors, func(wc *asanaclient.WorkspaceClient, t *asanaclient.Task) error {
utl, err := wc.GetMyUserTaskList()
if err != nil {
return err
}
sec, err := wc.GetSectionByName(utl, name)
if err != nil {
return err
}
return wc.AddTaskToSection(t, sec)
})
return p
}
func (p *periodic) PrintTasks() *periodic {
p.taskActors = append(p.taskActors, func(wc *asanaclient.WorkspaceClient, t *asanaclient.Task) error {
fmt.Printf("%s\n", t)
return nil
})
return p
}
// Infra
func (p *periodic) start(client *asanaclient.Client) {
err := p.validate()
if err != nil {
panic(err)
}
go p.loop(client)
}
func (p *periodic) validate() error {
return nil
}
func (p *periodic) wait() {
<-p.done
}
func (p *periodic) loop(client *asanaclient.Client) {
for {
time.Sleep(time.Duration(rand.Intn(p.period)) * time.Second)
err := p.exec(client)
if err != nil {
fmt.Printf("ERROR: %s\n", err)
// continue
}
}
close(p.done)
}
func (p *periodic) exec(c *asanaclient.Client) error {
wc, err := p.workspaceClientGetter(c)
if err != nil {
return err
}
q := &asanaclient.SearchQuery{}
for _, mut := range p.queryMutators {
err = mut(wc, q)
if err != nil {
return err
}
}
tasks, err := wc.Search(q)
if err != nil {
return err
}
filteredTasks := []*asanaclient.Task{}
for _, task := range tasks {
included := true
for _, filter := range p.taskFilters {
include, err := filter(wc, task)
if err != nil {
return err
}
if !include {
included = false
break
}
}
if included {
filteredTasks = append(filteredTasks, task)
}
}
for _, task := range filteredTasks {
for _, act := range p.taskActors {
err = act(wc, task)
if err != nil {
return err
}
}
}
return nil
}
// Helpers
func fixUnlinkedURL(node *html.Node) {
if node == nil {
return
}
if node.Type == html.ElementNode && node.Data == "a" {
// Don't go down this tree, since it's a link
return
}
if node.Type == html.TextNode {
accum := []string{}
nodes := []*html.Node{}
for _, line := range strings.Split(node.Data, "\n") {
if strings.HasPrefix(line, "http://") || strings.HasPrefix(line, "https://") {
if len(accum) > 0 {
accum = append(accum, "") // Trailing newline
nodes = append(nodes, &html.Node{
Type: html.TextNode,
Data: strings.Join(accum, "\n"),
})
accum = []string{""}
}
nodes = append(nodes, &html.Node{
Type: html.ElementNode,
Data: "a",
DataAtom: atom.A,
Attr: []html.Attribute{
html.Attribute{
Key: "href",
Val: line,
},
},
FirstChild: &html.Node{
Type: html.TextNode,
Data: line,
},
})
} else {
accum = append(accum, line)
}
}
if len(nodes) == 0 {
return
}
nodes = append(nodes, &html.Node{
Type: html.TextNode,
Data: strings.Join(accum, "\n"),
})
for i, iter := range nodes {
if i == len(nodes)-1 {
// Last node
iter.NextSibling = node.NextSibling
} else {
iter.NextSibling = nodes[i+1]
}
}
node.Data = ""
node.NextSibling = nodes[0]
return
}
fixUnlinkedURL(node.FirstChild)
fixUnlinkedURL(node.NextSibling)
}
func hasUnlinkedURL(node *html.Node) bool {
if node == nil {
return false
}
if node.Type == html.ElementNode && node.Data == "a" {
// Don't go down this tree, since it's a link
return false
}
if node.Type == html.TextNode {
for _, line := range strings.Split(node.Data, "\n") {
if strings.HasPrefix(line, "http://") || strings.HasPrefix(line, "https://") {
return true
}
}
}
if hasUnlinkedURL(node.FirstChild) {
return true
}
if hasUnlinkedURL(node.NextSibling) {
return true
}
return false
}