368 lines
8.2 KiB
Go
Raw Normal View History

package main
2020-03-20 05:49:12 +10:30
//go:generate go-bindata -pkg assets -o assets/static.go -prefix data/ data
import (
2017-02-22 21:13:07 +10:30
"bytes"
"encoding/json"
"fmt"
"io"
"io/ioutil"
"log"
"mime/multipart"
"net/http"
"os"
"path/filepath"
"strings"
"time"
"image"
_ "image/gif"
_ "image/jpeg"
_ "image/png"
"github.com/fogleman/gg"
2017-02-22 21:13:07 +10:30
"github.com/pborman/getopt"
// "github.com/skratchdot/open-golang/open"
"golang.org/x/image/font/inconsolata"
"github.com/tardisx/discord-auto-upload/config"
2021-01-31 18:48:48 +10:30
"github.com/tardisx/discord-auto-upload/web"
)
2021-01-31 18:48:48 +10:30
var lastCheck = time.Now()
2017-02-22 21:13:07 +10:30
var newLastCheck = time.Now()
func main() {
2017-02-22 21:13:07 +10:30
parseOptions()
// log.Print("Opening web browser")
// open.Start("http://localhost:9090")
web.StartWebServer()
2017-07-26 22:40:21 +09:30
checkUpdates()
sendLogToWeb(fmt.Sprintf("Waiting for images to appear in %s", config.Config.Path))
2017-02-22 21:13:07 +10:30
// wander the path, forever
for {
2021-02-07 11:42:13 +10:30
if checkPath(config.Config.Path) {
err := filepath.Walk(config.Config.Path,
func(path string, f os.FileInfo, err error) error { return checkFile(path, f, err) })
if err != nil {
log.Fatal("could not watch path", err)
}
lastCheck = newLastCheck
2017-02-22 21:13:07 +10:30
}
sendLogToWeb(fmt.Sprintf("sleeping for %ds before next check of %s", config.Config.Watch, config.Config.Path))
time.Sleep(time.Duration(config.Config.Watch) * time.Second)
2017-07-27 12:18:02 +09:30
}
}
2021-02-07 11:42:13 +10:30
func checkPath(path string) bool {
2017-02-22 21:13:07 +10:30
src, err := os.Stat(path)
if err != nil {
log.Printf("Problem with path '%s': %s", path, err)
2021-02-07 11:42:13 +10:30
return false
2017-02-22 21:13:07 +10:30
}
if !src.IsDir() {
log.Printf("Problem with path '%s': is not a directory", path)
2021-02-07 11:42:13 +10:30
return false
2017-02-22 21:13:07 +10:30
}
2021-02-07 11:42:13 +10:30
return true
}
2017-02-22 21:13:07 +10:30
func checkUpdates() {
type GithubRelease struct {
HTMLURL string
TagName string
Name string
Body string
}
sendLogToWeb("checking for new version")
2017-02-22 21:13:07 +10:30
client := &http.Client{Timeout: time.Second * 5}
resp, err := client.Get("https://api.github.com/repos/tardisx/discord-auto-upload/releases/latest")
if err != nil {
sendLogToWeb(fmt.Sprintf("WARNING: Update check failed: %v", err))
return
2017-02-22 21:13:07 +10:30
}
defer resp.Body.Close()
body, err := ioutil.ReadAll(resp.Body)
if err != nil {
log.Fatal("could not check read update response")
}
var latest GithubRelease
err = json.Unmarshal(body, &latest)
if err != nil {
log.Fatal("could not parse JSON: ", err)
}
if config.CurrentVersion < latest.TagName {
fmt.Printf("You are currently on version %s, but version %s is available\n", config.CurrentVersion, latest.TagName)
2017-02-22 21:13:07 +10:30
fmt.Println("----------- Release Info -----------")
fmt.Println(latest.Body)
fmt.Println("------------------------------------")
fmt.Println("Upgrade at https://github.com/tardisx/discord-auto-upload/releases/latest")
sendLogToWeb(fmt.Sprintf("New version available: %s - download at https://github.com/tardisx/discord-auto-upload/releases/latest"))
2017-02-22 21:13:07 +10:30
}
}
func parseOptions() {
2017-02-22 21:13:07 +10:30
// Declare the flags to be used
helpFlag := getopt.BoolLong("help", 'h', "help")
versionFlag := getopt.BoolLong("version", 'v', "show version")
getopt.SetParameters("")
2017-02-22 21:13:07 +10:30
getopt.Parse()
2017-02-22 21:13:07 +10:30
if *helpFlag {
getopt.PrintUsage(os.Stderr)
os.Exit(0)
}
2017-02-21 12:28:26 +10:30
2017-02-22 21:13:07 +10:30
if *versionFlag {
fmt.Println("dau - https://github.com/tardisx/discord-auto-upload")
fmt.Printf("Version: %s\n", config.CurrentVersion)
2017-02-22 21:13:07 +10:30
os.Exit(0)
}
2021-02-07 11:42:13 +10:30
// grab the config
config.LoadOrInit()
}
func checkFile(path string, f os.FileInfo, err error) error {
2017-02-22 21:13:07 +10:30
if f.ModTime().After(lastCheck) && f.Mode().IsRegular() {
if fileEligible(path) {
2017-02-22 21:13:07 +10:30
// process file
processFile(path)
2017-02-22 21:13:07 +10:30
}
2017-02-22 21:13:07 +10:30
if newLastCheck.Before(f.ModTime()) {
newLastCheck = f.ModTime()
}
}
2017-02-22 21:13:07 +10:30
return nil
}
func sendLogToWeb(entry string) {
web.LogInput <- web.LogEntry{
Timestamp: time.Now(),
Entry: entry,
}
}
func fileEligible(file string) bool {
if config.Config.Exclude != "" && strings.Contains(file, config.Config.Exclude) {
return false
}
2017-02-22 21:13:07 +10:30
extension := strings.ToLower(filepath.Ext(file))
if extension == ".png" || extension == ".jpg" || extension == ".gif" {
return true
}
2017-02-22 21:13:07 +10:30
return false
}
func processFile(file string) {
if !config.Config.NoWatermark {
sendLogToWeb("Copying to temp location and watermarking ")
file = mungeFile(file)
}
2021-02-07 11:42:13 +10:30
if config.Config.WebHookURL == "" {
sendLogToWeb("WebHookURL is not configured - cannot upload!")
2021-02-07 11:42:13 +10:30
return
}
sendLogToWeb("Uploading")
2017-02-22 21:13:07 +10:30
extraParams := map[string]string{}
if config.Config.Username != "" {
log.Print("Overriding username with " + config.Config.Username)
extraParams["username"] = config.Config.Username
2017-02-22 21:13:07 +10:30
}
type DiscordAPIResponseAttachment struct {
URL string
ProxyURL string
Size int
Width int
Height int
Filename string
}
type DiscordAPIResponse struct {
Attachments []DiscordAPIResponseAttachment
ID int64 `json:",string"`
}
2017-02-28 22:07:57 +10:30
var retriesRemaining = 5
for retriesRemaining > 0 {
2020-03-26 11:40:35 +10:30
request, err := newfileUploadRequest(config.Config.WebHookURL, extraParams, "file", file)
2017-02-22 21:13:07 +10:30
if err != nil {
2017-02-28 22:07:57 +10:30
log.Fatal(err)
2017-02-22 21:13:07 +10:30
}
2017-02-28 22:07:57 +10:30
start := time.Now()
client := &http.Client{Timeout: time.Second * 30}
resp, err := client.Do(request)
2017-02-22 21:13:07 +10:30
if err != nil {
2017-02-28 22:07:57 +10:30
log.Print("Error performing request:", err)
retriesRemaining--
sleepForRetries(retriesRemaining)
continue
} else {
if resp.StatusCode != 200 {
log.Print("Bad response from server:", resp.StatusCode)
2020-03-26 11:40:35 +10:30
if b, err := ioutil.ReadAll(resp.Body); err == nil {
log.Print("Body:", string(b))
2021-01-31 18:48:48 +10:30
}
2017-02-28 22:07:57 +10:30
retriesRemaining--
sleepForRetries(retriesRemaining)
continue
}
resBody, err := ioutil.ReadAll(resp.Body)
if err != nil {
log.Print("could not deal with body: ", err)
retriesRemaining--
sleepForRetries(retriesRemaining)
continue
}
resp.Body.Close()
var res DiscordAPIResponse
err = json.Unmarshal(resBody, &res)
if err != nil {
log.Print("could not parse JSON: ", err)
fmt.Println("Response was:", string(resBody[:]))
retriesRemaining--
sleepForRetries(retriesRemaining)
continue
}
if len(res.Attachments) < 1 {
log.Print("bad response - no attachments?")
retriesRemaining--
sleepForRetries(retriesRemaining)
continue
}
var a = res.Attachments[0]
elapsed := time.Since(start)
rate := float64(a.Size) / elapsed.Seconds() / 1024.0
log.Printf("Uploaded to %s %dx%d", a.URL, a.Width, a.Height)
log.Printf("id: %d, %d bytes transferred in %.2f seconds (%.2f KiB/s)", res.ID, a.Size, elapsed.Seconds(), rate)
break
2017-02-22 21:13:07 +10:30
}
2017-02-28 22:07:57 +10:30
}
2017-02-22 21:13:07 +10:30
if !config.Config.NoWatermark {
sendLogToWeb(fmt.Sprintf("Removing temporary file: %s", file))
2017-02-28 22:07:57 +10:30
os.Remove(file)
}
2017-02-28 22:07:57 +10:30
if retriesRemaining == 0 {
log.Fatal("Failed to upload, even after retries")
2017-02-22 21:13:07 +10:30
}
2017-02-28 22:07:57 +10:30
}
2017-02-28 22:07:57 +10:30
func sleepForRetries(retry int) {
if retry == 0 {
return
}
retryTime := (6-retry)*(6-retry) + 6
sendLogToWeb(fmt.Sprintf("Will retry in %d seconds (%d remaining attempts)", retryTime, retry))
time.Sleep(time.Duration(retryTime) * time.Second)
}
func newfileUploadRequest(uri string, params map[string]string, paramName, path string) (*http.Request, error) {
2017-02-22 21:13:07 +10:30
file, err := os.Open(path)
if err != nil {
return nil, err
}
defer file.Close()
body := &bytes.Buffer{}
writer := multipart.NewWriter(body)
part, err := writer.CreateFormFile(paramName, filepath.Base(path))
if err != nil {
return nil, err
}
_, err = io.Copy(part, file)
if err != nil {
log.Fatal("Could not copy: ", err)
}
for key, val := range params {
_ = writer.WriteField(key, val)
}
err = writer.Close()
if err != nil {
return nil, err
}
req, err := http.NewRequest("POST", uri, body)
req.Header.Set("Content-Type", writer.FormDataContentType())
return req, err
}
func mungeFile(path string) string {
reader, err := os.Open(path)
if err != nil {
log.Fatal(err)
}
defer reader.Close()
im, _, err := image.Decode(reader)
if err != nil {
log.Fatal(err)
}
bounds := im.Bounds()
// var S float64 = float64(bounds.Max.X)
dc := gg.NewContext(bounds.Max.X, bounds.Max.Y)
dc.Clear()
dc.SetRGB(0, 0, 0)
dc.SetFontFace(inconsolata.Regular8x16)
dc.DrawImage(im, 0, 0)
dc.DrawRoundedRectangle(0, float64(bounds.Max.Y-18.0), 320, float64(bounds.Max.Y), 0)
dc.SetRGB(0, 0, 0)
dc.Fill()
dc.SetRGB(1, 1, 1)
dc.DrawString("github.com/tardisx/discord-auto-upload", 5.0, float64(bounds.Max.Y)-5.0)
tempfile, err := ioutil.TempFile("", "dau")
if err != nil {
log.Fatal(err)
}
tempfile.Close()
os.Remove(tempfile.Name())
actualName := tempfile.Name() + ".png"
dc.SavePNG(actualName)
return actualName
}