495 lines
9.8 KiB
Go
495 lines
9.8 KiB
Go
//
|
|
// 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 (
|
|
"flag"
|
|
"fmt"
|
|
"html/template"
|
|
golog "log"
|
|
"log/syslog"
|
|
notrand "math/rand"
|
|
"os"
|
|
"strconv"
|
|
"strings"
|
|
"time"
|
|
|
|
"humungus.tedunangst.com/r/webs/httpsig"
|
|
"humungus.tedunangst.com/r/webs/log"
|
|
)
|
|
|
|
var softwareVersion = "develop"
|
|
|
|
func init() {
|
|
notrand.Seed(time.Now().Unix())
|
|
}
|
|
|
|
type WhatAbout struct {
|
|
ID int64
|
|
Name string
|
|
Display string
|
|
About string
|
|
HTAbout template.HTML
|
|
Onts []string
|
|
Key string
|
|
URL string
|
|
Options UserOptions
|
|
SecKey httpsig.PrivateKey
|
|
}
|
|
|
|
type UserOptions struct {
|
|
SkinnyCSS bool `json:",omitempty"`
|
|
OmitImages bool `json:",omitempty"`
|
|
MentionAll bool `json:",omitempty"`
|
|
InlineQuotes bool `json:",omitempty"`
|
|
Avatar string `json:",omitempty"`
|
|
Banner string `json:",omitempty"`
|
|
MapLink string `json:",omitempty"`
|
|
Reaction string `json:",omitempty"`
|
|
MeCount int64
|
|
ChatCount int64
|
|
}
|
|
|
|
type KeyInfo struct {
|
|
keyname string
|
|
seckey httpsig.PrivateKey
|
|
}
|
|
|
|
const serverUID int64 = -2
|
|
const readyLuserOne int64 = 1
|
|
|
|
type Honk struct {
|
|
ID int64
|
|
UserID int64
|
|
Username string
|
|
What string
|
|
Honker string
|
|
Handle string
|
|
Handles string
|
|
Oonker string
|
|
Oondle string
|
|
XID string
|
|
RID string
|
|
Date time.Time
|
|
URL string
|
|
Noise string
|
|
Precis string
|
|
Format string
|
|
Convoy string
|
|
Audience []string
|
|
Public bool
|
|
Whofore int64
|
|
Replies []*Honk
|
|
Flags int64
|
|
HTPrecis template.HTML
|
|
HTML template.HTML
|
|
Style string
|
|
Open string
|
|
Donks []*Donk
|
|
Onts []string
|
|
Place *Place
|
|
Time *Time
|
|
Mentions []Mention
|
|
Badonks []Badonk
|
|
}
|
|
|
|
type Badonk struct {
|
|
Who string
|
|
What string
|
|
}
|
|
|
|
type Chonk struct {
|
|
ID int64
|
|
UserID int64
|
|
XID string
|
|
Who string
|
|
Target string
|
|
Date time.Time
|
|
Noise string
|
|
Format string
|
|
Donks []*Donk
|
|
Handle string
|
|
HTML template.HTML
|
|
}
|
|
|
|
type Chatter struct {
|
|
Target string
|
|
Chonks []*Chonk
|
|
}
|
|
|
|
type Mention struct {
|
|
Who string
|
|
Where string
|
|
}
|
|
|
|
func (mention *Mention) IsPresent(noise string) bool {
|
|
nick := strings.TrimLeft(mention.Who, "@")
|
|
idx := strings.IndexByte(nick, '@')
|
|
if idx != -1 {
|
|
nick = nick[:idx]
|
|
}
|
|
return strings.Contains(noise, ">@"+nick) || strings.Contains(noise, "@<span>"+nick)
|
|
}
|
|
|
|
func OntIsPresent(ont, noise string) bool {
|
|
ont = strings.ToLower(ont[1:])
|
|
idx := strings.IndexByte(noise, '#')
|
|
for idx >= 0 {
|
|
if strings.HasPrefix(noise[idx:], "#<span>") {
|
|
idx += 6
|
|
}
|
|
idx += 1
|
|
if idx+len(ont) >= len(noise) {
|
|
return false
|
|
}
|
|
test := noise[idx : idx+len(ont)]
|
|
test = strings.ToLower(test)
|
|
if test == ont {
|
|
return true
|
|
}
|
|
newidx := strings.IndexByte(noise[idx:], '#')
|
|
if newidx == -1 {
|
|
return false
|
|
}
|
|
idx += newidx
|
|
}
|
|
return false
|
|
}
|
|
|
|
type OldRevision struct {
|
|
Precis string
|
|
Noise string
|
|
}
|
|
|
|
const (
|
|
flagIsAcked = 1
|
|
flagIsBonked = 2
|
|
flagIsSaved = 4
|
|
flagIsUntagged = 8
|
|
flagIsReacted = 16
|
|
)
|
|
|
|
func (honk *Honk) IsAcked() bool {
|
|
return honk.Flags&flagIsAcked != 0
|
|
}
|
|
|
|
func (honk *Honk) IsBonked() bool {
|
|
return honk.Flags&flagIsBonked != 0
|
|
}
|
|
|
|
func (honk *Honk) IsSaved() bool {
|
|
return honk.Flags&flagIsSaved != 0
|
|
}
|
|
|
|
func (honk *Honk) IsUntagged() bool {
|
|
return honk.Flags&flagIsUntagged != 0
|
|
}
|
|
|
|
func (honk *Honk) IsReacted() bool {
|
|
return honk.Flags&flagIsReacted != 0
|
|
}
|
|
|
|
type Donk struct {
|
|
FileID int64
|
|
XID string
|
|
Name string
|
|
Desc string
|
|
URL string
|
|
Media string
|
|
Local bool
|
|
External bool
|
|
}
|
|
|
|
type Place struct {
|
|
Name string
|
|
Latitude float64
|
|
Longitude float64
|
|
Url string
|
|
}
|
|
|
|
type Duration int64
|
|
|
|
func (d Duration) String() string {
|
|
s := time.Duration(d).String()
|
|
if strings.HasSuffix(s, "m0s") {
|
|
s = s[:len(s)-2]
|
|
}
|
|
if strings.HasSuffix(s, "h0m") {
|
|
s = s[:len(s)-2]
|
|
}
|
|
return s
|
|
}
|
|
|
|
func parseDuration(s string) time.Duration {
|
|
didx := strings.IndexByte(s, 'd')
|
|
if didx != -1 {
|
|
days, _ := strconv.ParseInt(s[:didx], 10, 0)
|
|
dur, _ := time.ParseDuration(s[didx:])
|
|
return dur + 24*time.Hour*time.Duration(days)
|
|
}
|
|
dur, _ := time.ParseDuration(s)
|
|
return dur
|
|
}
|
|
|
|
type Time struct {
|
|
StartTime time.Time
|
|
EndTime time.Time
|
|
Duration Duration
|
|
}
|
|
|
|
type Honker struct {
|
|
ID int64
|
|
UserID int64
|
|
Name string
|
|
XID string
|
|
Handle string
|
|
Flavor string
|
|
Combos []string
|
|
Meta HonkerMeta
|
|
}
|
|
|
|
type HonkerMeta struct {
|
|
Notes string
|
|
}
|
|
|
|
type SomeThing struct {
|
|
What int
|
|
XID string
|
|
Owner string
|
|
Name string
|
|
}
|
|
|
|
const (
|
|
SomeNothing int = iota
|
|
SomeActor
|
|
SomeCollection
|
|
)
|
|
|
|
var serverName string
|
|
var serverPrefix string
|
|
var masqName string
|
|
var dataDir = "."
|
|
var viewDir = "."
|
|
var iconName = "icon.png"
|
|
var serverMsg template.HTML
|
|
var aboutMsg template.HTML
|
|
var loginMsg template.HTML
|
|
|
|
func ElaborateUnitTests() {
|
|
}
|
|
|
|
func unplugserver(hostname string) {
|
|
db := opendatabase()
|
|
xid := fmt.Sprintf("%%https://%s/%%", hostname)
|
|
db.Exec("delete from honkers where xid like ? and flavor = 'dub'", xid)
|
|
db.Exec("delete from doovers where rcpt like ?", xid)
|
|
}
|
|
|
|
func reexecArgs(cmd string) []string {
|
|
args := []string{"-datadir", dataDir}
|
|
args = append(args, log.Args()...)
|
|
args = append(args, cmd)
|
|
return args
|
|
}
|
|
|
|
var elog, ilog, dlog *golog.Logger
|
|
|
|
func main() {
|
|
flag.StringVar(&dataDir, "datadir", dataDir, "data directory")
|
|
flag.StringVar(&viewDir, "viewdir", viewDir, "view directory")
|
|
flag.Parse()
|
|
|
|
log.Init(log.Options{Progname: "honk", Facility: syslog.LOG_UUCP})
|
|
elog = log.E
|
|
ilog = log.I
|
|
dlog = log.D
|
|
|
|
args := flag.Args()
|
|
cmd := "run"
|
|
if len(args) > 0 {
|
|
cmd = args[0]
|
|
}
|
|
switch cmd {
|
|
case "init":
|
|
initdb()
|
|
case "upgrade":
|
|
upgradedb()
|
|
case "version":
|
|
fmt.Println(softwareVersion)
|
|
os.Exit(0)
|
|
}
|
|
db := opendatabase()
|
|
dbversion := 0
|
|
getconfig("dbversion", &dbversion)
|
|
if dbversion != myVersion {
|
|
elog.Fatal("incorrect database version. run upgrade.")
|
|
}
|
|
getconfig("servermsg", &serverMsg)
|
|
getconfig("aboutmsg", &aboutMsg)
|
|
getconfig("loginmsg", &loginMsg)
|
|
getconfig("servername", &serverName)
|
|
getconfig("masqname", &masqName)
|
|
if masqName == "" {
|
|
masqName = serverName
|
|
}
|
|
serverPrefix = fmt.Sprintf("https://%s/", serverName)
|
|
getconfig("usersep", &userSep)
|
|
getconfig("honksep", &honkSep)
|
|
getconfig("devel", &develMode)
|
|
getconfig("fasttimeout", &fastTimeout)
|
|
getconfig("slowtimeout", &slowTimeout)
|
|
getconfig("signgets", &signGets)
|
|
prepareStatements(db)
|
|
switch cmd {
|
|
case "admin":
|
|
adminscreen()
|
|
case "import":
|
|
if len(args) != 4 {
|
|
elog.Fatal("import username mastodon|twitter srcdir")
|
|
}
|
|
importMain(args[1], args[2], args[3])
|
|
case "devel":
|
|
if len(args) != 2 {
|
|
elog.Fatal("need an argument: devel (on|off)")
|
|
}
|
|
switch args[1] {
|
|
case "on":
|
|
setconfig("devel", 1)
|
|
case "off":
|
|
setconfig("devel", 0)
|
|
default:
|
|
elog.Fatal("argument must be on or off")
|
|
}
|
|
case "setconfig":
|
|
if len(args) != 3 {
|
|
elog.Fatal("need an argument: setconfig key val")
|
|
}
|
|
var val interface{}
|
|
var err error
|
|
if val, err = strconv.Atoi(args[2]); err != nil {
|
|
val = args[2]
|
|
}
|
|
setconfig(args[1], val)
|
|
case "adduser":
|
|
adduser()
|
|
case "deluser":
|
|
if len(args) < 2 {
|
|
fmt.Printf("usage: honk deluser username\n")
|
|
return
|
|
}
|
|
deluser(args[1])
|
|
case "chpass":
|
|
if len(args) < 2 {
|
|
fmt.Printf("usage: honk chpass username\n")
|
|
return
|
|
}
|
|
chpass(args[1])
|
|
case "follow":
|
|
if len(args) < 3 {
|
|
fmt.Printf("usage: honk follow username url\n")
|
|
return
|
|
}
|
|
user, err := butwhatabout(args[1])
|
|
if err != nil {
|
|
fmt.Printf("user not found\n")
|
|
return
|
|
}
|
|
var meta HonkerMeta
|
|
mj, _ := jsonify(&meta)
|
|
honkerid, err := savehonker(user, args[2], "", "presub", "", mj)
|
|
if err != nil {
|
|
fmt.Printf("had some trouble with that: %s\n", err)
|
|
return
|
|
}
|
|
followyou(user, honkerid, true)
|
|
case "unfollow":
|
|
if len(args) < 3 {
|
|
fmt.Printf("usage: honk unfollow username url\n")
|
|
return
|
|
}
|
|
user, err := butwhatabout(args[1])
|
|
if err != nil {
|
|
fmt.Printf("user not found\n")
|
|
return
|
|
}
|
|
row := db.QueryRow("select honkerid from honkers where xid = ? and userid = ? and flavor in ('sub')", args[2], user.ID)
|
|
var honkerid int64
|
|
err = row.Scan(&honkerid)
|
|
if err != nil {
|
|
fmt.Printf("sorry couldn't find them\n")
|
|
return
|
|
}
|
|
unfollowyou(user, honkerid, true)
|
|
case "sendmsg":
|
|
if len(args) < 4 {
|
|
fmt.Printf("usage: honk send username filename rcpt\n")
|
|
return
|
|
}
|
|
user, err := butwhatabout(args[1])
|
|
if err != nil {
|
|
fmt.Printf("user not found\n")
|
|
return
|
|
}
|
|
data, err := os.ReadFile(args[2])
|
|
if err != nil {
|
|
fmt.Printf("can't read file\n")
|
|
return
|
|
}
|
|
deliverate(user.ID, args[3], data)
|
|
case "cleanup":
|
|
arg := "30"
|
|
if len(args) > 1 {
|
|
arg = args[1]
|
|
}
|
|
cleanupdb(arg)
|
|
case "unplug":
|
|
if len(args) < 2 {
|
|
fmt.Printf("usage: honk unplug servername\n")
|
|
return
|
|
}
|
|
name := args[1]
|
|
unplugserver(name)
|
|
case "backup":
|
|
if len(args) < 2 {
|
|
fmt.Printf("usage: honk backup dirname\n")
|
|
return
|
|
}
|
|
name := args[1]
|
|
svalbard(name)
|
|
case "ping":
|
|
if len(args) < 3 {
|
|
fmt.Printf("usage: honk ping (from username) (to username or url)\n")
|
|
return
|
|
}
|
|
name := args[1]
|
|
targ := args[2]
|
|
user, err := butwhatabout(name)
|
|
if err != nil {
|
|
elog.Printf("unknown user")
|
|
return
|
|
}
|
|
ping(user, targ)
|
|
case "run":
|
|
serve()
|
|
case "backend":
|
|
backendServer()
|
|
case "test":
|
|
ElaborateUnitTests()
|
|
default:
|
|
elog.Fatal("unknown command")
|
|
}
|
|
}
|