mirror of
https://github.com/Luzifer/share.git
synced 2025-01-06 18:56:03 +00:00
154 lines
4.4 KiB
Go
154 lines
4.4 KiB
Go
package main
|
|
|
|
import (
|
|
"bytes"
|
|
"embed"
|
|
"fmt"
|
|
"io"
|
|
"os"
|
|
"strings"
|
|
|
|
"github.com/pkg/errors"
|
|
"github.com/sirupsen/logrus"
|
|
|
|
"github.com/Luzifer/rconfig/v2"
|
|
)
|
|
|
|
var (
|
|
cfg = struct {
|
|
BaseURL string `flag:"base-url" default:"" description:"URL to prepend before filename"`
|
|
Bootstrap bool `flag:"bootstrap" default:"false" description:"Upload frontend files into bucket"`
|
|
Bucket string `flag:"bucket" default:"" description:"S3 bucket to upload files to" validate:"nonzero"`
|
|
ContentType string `flag:"content-type,c" default:"" description:"Force content-type to be set to this value"`
|
|
Endpoint string `flag:"endpoint" default:"" description:"Override AWS S3 endpoint (i.e. to use MinIO)"`
|
|
FileTemplate string `flag:"file-template" vardefault:"file_template" description:"Full name template of the uploaded file"`
|
|
Listen string `flag:"listen" default:"" description:"Enable HTTP server if set to IP/Port (e.g. ':3000')"`
|
|
LogLevel string `flag:"log-level" default:"info" description:"Log level (debug, info, warn, error, fatal)"`
|
|
Progress bool `flag:"progress" default:"false" description:"Show progress bar while uploading"`
|
|
VersionAndExit bool `flag:"version" default:"false" description:"Prints current version and exits"`
|
|
}{}
|
|
|
|
//go:embed frontend/*
|
|
frontend embed.FS
|
|
|
|
version = "dev"
|
|
)
|
|
|
|
func initApp() (err error) {
|
|
rconfig.AutoEnv(true)
|
|
rconfig.SetVariableDefaults(map[string]string{
|
|
"file_template": `file/{{ printf "%.6s" .Hash }}/{{ .SafeFileName }}`,
|
|
})
|
|
|
|
if err = rconfig.ParseAndValidate(&cfg); err != nil {
|
|
return fmt.Errorf("parsing CLI options: %w", err)
|
|
}
|
|
|
|
l, err := logrus.ParseLevel(cfg.LogLevel)
|
|
if err != nil {
|
|
return fmt.Errorf("parsing log-level: %w", err)
|
|
}
|
|
logrus.SetLevel(l)
|
|
|
|
return nil
|
|
}
|
|
|
|
func main() {
|
|
var err error
|
|
if err = initApp(); err != nil {
|
|
logrus.WithError(err).Fatal("initializing app")
|
|
}
|
|
|
|
if cfg.VersionAndExit {
|
|
fmt.Printf("share %s\n", version) //nolint:forbidigo // Fine for version info
|
|
os.Exit(0)
|
|
}
|
|
|
|
switch {
|
|
case cfg.Bootstrap:
|
|
if err := doBootstrap(); err != nil {
|
|
logrus.WithError(err).Fatal("bootstrapping resources")
|
|
}
|
|
logrus.Info("Bucket bootstrap finished: Frontend uploaded")
|
|
|
|
case cfg.Listen != "":
|
|
logrus.WithFields(logrus.Fields{
|
|
"addr": cfg.Listen,
|
|
"version": version,
|
|
}).Info("share HTTP server started")
|
|
if err := doListen(); err != nil {
|
|
logrus.WithError(err).Fatal("running HTTP server")
|
|
}
|
|
|
|
default:
|
|
if err := doCLIUpload(); err != nil {
|
|
logrus.WithError(err).Fatal("uploading file")
|
|
}
|
|
}
|
|
}
|
|
|
|
func doCLIUpload() error {
|
|
if len(rconfig.Args()) == 1 {
|
|
return errors.New("missing argument: file to upload")
|
|
}
|
|
|
|
if cfg.BaseURL == "" {
|
|
logrus.Warn("No BaseURL configured, output will be no complete URL")
|
|
}
|
|
|
|
var inFile io.ReadSeeker
|
|
|
|
inFileName := rconfig.Args()[1]
|
|
|
|
if inFileName == "-" {
|
|
if cfg.ContentType == "" {
|
|
// If we don't have an explicitly set content-type assume stdin contains text
|
|
inFileName = "stdin"
|
|
cfg.ContentType = "text/plain"
|
|
} else if ext, err := mimeResolver.ExtensionsByType(cfg.ContentType); err == nil {
|
|
inFileName = strings.Join([]string{"stdin", ext}, "")
|
|
}
|
|
|
|
// Stdin is not seekable, so we need to buffer it
|
|
buf := new(bytes.Buffer)
|
|
if _, err := io.Copy(buf, os.Stdin); err != nil {
|
|
logrus.WithError(err).Fatal("reading stdin")
|
|
}
|
|
|
|
inFile = bytes.NewReader(buf.Bytes())
|
|
} else {
|
|
inFileHandle, err := os.Open(inFileName) //#nosec:G304 // Inentional read of arbitrary file
|
|
if err != nil {
|
|
return errors.Wrap(err, "opening source file")
|
|
}
|
|
defer inFileHandle.Close() //nolint:errcheck // Irrelevant, file is closed by process exit
|
|
inFile = inFileHandle
|
|
}
|
|
|
|
url, err := executeUpload(inFileName, inFile, true, cfg.ContentType, false)
|
|
if err != nil {
|
|
return errors.Wrap(err, "uploading file")
|
|
}
|
|
|
|
fmt.Println(url) //nolint:forbidigo // Intended as programmatic payload
|
|
return nil
|
|
}
|
|
|
|
func doBootstrap() error {
|
|
files, err := frontend.ReadDir("frontend")
|
|
if err != nil {
|
|
return fmt.Errorf("listing embedded files: %w", err)
|
|
}
|
|
|
|
for _, asset := range files {
|
|
content, err := frontend.ReadFile(strings.Join([]string{"frontend", asset.Name()}, "/"))
|
|
if err != nil {
|
|
return errors.Wrap(err, "reading baked asset")
|
|
}
|
|
|
|
if _, err := executeUpload(asset.Name(), bytes.NewReader(content), false, "", true); err != nil {
|
|
return errors.Wrapf(err, "uploading bootstrap asset %q", asset)
|
|
}
|
|
}
|
|
return nil
|
|
}
|