honk/deliverator.go

227 lines
4.8 KiB
Go
Raw Normal View History

2019-04-14 21:57:14 +02:00
//
// Copyright (c) 2019 Ted Unangst <tedu@tedunangst.com>
//
// Permission to use, copy, modify, and distribute this software for any
// purpose with or without fee is hereby granted, provided that the above
// copyright notice and this permission notice appear in all copies.
//
// THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
// WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
// MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
// ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
// WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
// ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
// OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
package main
import (
"bytes"
"database/sql"
notrand "math/rand"
"sync"
2019-04-14 21:57:14 +02:00
"time"
2019-10-29 18:38:51 +01:00
"humungus.tedunangst.com/r/webs/gate"
2019-04-14 21:57:14 +02:00
)
type Doover struct {
ID int64
When time.Time
Userid int64
Tries int64
Rcpt string
Msgs [][]byte
}
func sayitagain(doover Doover) {
rcpt := doover.Rcpt
2019-04-14 21:57:14 +02:00
var drift time.Duration
doover.Tries += 1
switch doover.Tries {
2019-04-14 21:57:14 +02:00
case 1:
drift = 5 * time.Minute
case 2:
drift = 1 * time.Hour
case 3:
2019-08-06 02:41:57 +02:00
drift = 4 * time.Hour
2019-04-14 21:57:14 +02:00
case 4:
2019-08-06 02:41:57 +02:00
drift = 12 * time.Hour
case 5:
2019-04-14 21:57:14 +02:00
drift = 24 * time.Hour
default:
2022-02-06 06:42:13 +01:00
ilog.Printf("he's dead jim: %s", rcpt)
2019-04-14 21:57:14 +02:00
return
}
drift += time.Duration(notrand.Int63n(int64(drift / 10)))
2022-04-12 20:35:44 +02:00
when := time.Now().Add(drift)
data := bytes.Join(doover.Msgs, []byte{0})
_, err := stmtAddDoover.Exec(when.UTC().Format(dbtimeformat), doover.Tries, doover.Userid, rcpt, data)
if err != nil {
2022-02-06 06:42:13 +01:00
elog.Printf("error saving doover: %s", err)
}
select {
case pokechan <- 0:
default:
}
2019-04-14 21:57:14 +02:00
}
var dqmtx sync.Mutex
func delinquent(rcpt string, msg []byte) bool {
dqmtx.Lock()
defer dqmtx.Unlock()
row := stmtDeliquentCheck.QueryRow(rcpt)
var dooverid int64
var data []byte
err := row.Scan(&dooverid, data)
if err == sql.ErrNoRows {
return false
}
if err != nil {
elog.Printf("error scanning deliquent check: %s", err)
return true
}
data = append(data, 0)
data = append(data, msg...)
_, err = stmtDeliquentUpdate.Exec(data, dooverid)
if err != nil {
elog.Printf("error updating deliquent: %s", err)
return true
}
return true
}
func deliverate(userid int64, rcpt string, msg []byte) {
if delinquent(rcpt, msg) {
return
}
var d Doover
d.Userid = userid
d.Tries = 0
d.Rcpt = rcpt
d.Msgs = append(d.Msgs, msg)
deliveration(d)
}
2020-05-24 02:05:21 +02:00
var garage = gate.NewLimiter(40)
func deliveration(doover Doover) {
2019-10-29 18:38:51 +01:00
garage.Start()
defer garage.Finish()
var ki *KeyInfo
ok := ziggies.Get(doover.Userid, &ki)
if !ok {
2022-02-06 06:42:13 +01:00
elog.Printf("lost key for delivery")
return
}
2019-04-16 20:40:23 +02:00
var inbox string
rcpt := doover.Rcpt
2019-04-16 20:40:23 +02:00
// already did the box indirection
if rcpt[0] == '%' {
inbox = rcpt[1:]
} else {
var box *Box
ok := boxofboxes.Get(rcpt, &box)
if !ok {
2022-02-06 06:42:13 +01:00
ilog.Printf("failed getting inbox for %s", rcpt)
sayitagain(doover)
2019-04-16 20:40:23 +02:00
return
}
inbox = box.In
2019-04-14 21:57:14 +02:00
}
for i, msg := range doover.Msgs {
if i > 0 {
time.Sleep(2 * time.Second)
}
err := PostMsg(ki.keyname, ki.seckey, inbox, msg)
if err != nil {
ilog.Printf("failed to post json to %s: %s", inbox, err)
doover.Msgs = doover.Msgs[i:]
sayitagain(doover)
return
}
}
}
2019-10-16 08:35:47 +02:00
var pokechan = make(chan int, 1)
2019-09-17 15:45:02 +02:00
func getdoovers() []Doover {
rows, err := stmtGetDoovers.Query()
if err != nil {
2022-02-06 06:42:13 +01:00
elog.Printf("wat?")
2019-09-17 15:45:02 +02:00
time.Sleep(1 * time.Minute)
return nil
}
defer rows.Close()
var doovers []Doover
for rows.Next() {
var d Doover
var dt string
err := rows.Scan(&d.ID, &dt)
if err != nil {
2022-02-06 06:42:13 +01:00
elog.Printf("error scanning dooverid: %s", err)
2019-09-17 15:45:02 +02:00
continue
}
d.When, _ = time.Parse(dbtimeformat, dt)
doovers = append(doovers, d)
}
return doovers
}
func extractdoover(d *Doover) error {
dqmtx.Lock()
defer dqmtx.Unlock()
row := stmtLoadDoover.QueryRow(d.ID)
var data []byte
err := row.Scan(&d.Tries, &d.Userid, &d.Rcpt, &data)
if err != nil {
return err
}
_, err = stmtZapDoover.Exec(d.ID)
if err != nil {
return err
}
d.Msgs = bytes.Split(data, []byte{0})
return nil
}
func redeliverator() {
2022-04-12 20:35:44 +02:00
sleeper := time.NewTimer(5 * time.Second)
for {
select {
case <-pokechan:
if !sleeper.Stop() {
<-sleeper.C
}
time.Sleep(5 * time.Second)
case <-sleeper.C:
}
2019-09-17 15:45:02 +02:00
doovers := getdoovers()
2022-04-12 20:35:44 +02:00
now := time.Now()
nexttime := now.Add(24 * time.Hour)
for _, d := range doovers {
if d.When.Before(now) {
err := extractdoover(&d)
if err != nil {
elog.Printf("error extracting doover: %s", err)
continue
}
ilog.Printf("redeliverating %s try %d", d.Rcpt, d.Tries)
deliveration(d)
} else if d.When.Before(nexttime) {
nexttime = d.When
}
}
2022-04-12 20:35:44 +02:00
now = time.Now()
dur := 5 * time.Second
if now.Before(nexttime) {
dur += nexttime.Sub(now).Round(time.Second)
}
sleeper.Reset(dur)
2019-04-14 21:57:14 +02:00
}
}