/* Copyright © 2023 Thomas von Dein This program is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program. If not, see . */ package lib import ( "encoding/json" "errors" "fmt" "github.com/imroc/req/v3" "github.com/tlinden/up/upctl/cfg" "os" "path/filepath" "time" ) type Response struct { Code int `json:"code"` Success bool `json:"success"` Message string `json:"message"` } type Request struct { R *req.Request Url string } func Setup(c *cfg.Config, path string) *Request { client := req.C() if c.Debug { client.DevMode() } client.SetUserAgent("upctl-" + cfg.VERSION) R := client.R() if c.Retries > 0 { // Enable retry and set the maximum retry count. R.SetRetryCount(c.Retries). // Set the retry sleep interval with a commonly used // algorithm: capped exponential backoff with jitter // (https://aws.amazon.com/blogs/architecture/exponential-backoff-and-jitter/). SetRetryBackoffInterval(1*time.Second, 5*time.Second). AddRetryHook(func(resp *req.Response, err error) { req := resp.Request.RawRequest if c.Debug { fmt.Println("Retrying endpoint request:", req.Method, req.URL) } }) } if len(c.Apikey) > 0 { client.SetCommonBearerAuthToken(c.Apikey) } return &Request{Url: c.Endpoint + "/file/", R: R} } func GatherFiles(rq *Request, args []string) error { for _, file := range args { info, err := os.Stat(file) if os.IsNotExist(err) { return err } if info.IsDir() { err := filepath.Walk(file, func(path string, info os.FileInfo, err error) error { if err != nil { return err } if !info.IsDir() { rq.R.SetFile("upload[]", path) } return nil }) if err != nil { return err } } else { rq.R.SetFile("upload[]", file) } } return nil } func Upload(c *cfg.Config, args []string) error { // setup url, req.Request, timeout handling etc rq := Setup(c, "/file/") // collect files to upload from @argv if err := GatherFiles(rq, args); err != nil { return err } // actual post w/ settings resp, err := rq.R. SetFormData(map[string]string{ "expire": c.Expire, }). SetUploadCallbackWithInterval(func(info req.UploadInfo) { fmt.Printf("\r%q uploaded %.2f%%", info.FileName, float64(info.UploadedSize)/float64(info.FileSize)*100.0) fmt.Println() }, 10*time.Millisecond). Post(rq.Url) fmt.Println("") if err != nil { return err } // we expect a json response r := Response{} json.Unmarshal([]byte(resp.String()), &r) if c.Debug { trace := resp.TraceInfo() // Use `resp.Request.TraceInfo()` to avoid unnecessary struct copy in production. fmt.Println(trace.Blame()) // Print out exactly where the http request is slowing down. fmt.Println("----------") fmt.Println(trace) } if !r.Success { return errors.New(r.Message) } fmt.Println(r.Message) return nil }