1
0
Fork 0
mirror of https://github.com/Luzifer/share.git synced 2024-10-18 05:14:23 +00:00
share/main.go

149 lines
4.1 KiB
Go
Raw Normal View History

2017-12-02 14:25:54 +00:00
package main
import (
"bytes"
"embed"
_ "embed"
2017-12-02 14:25:54 +00:00
"fmt"
"io"
2017-12-02 14:25:54 +00:00
"os"
"strings"
2017-12-02 14:25:54 +00:00
"github.com/pkg/errors"
2017-12-02 14:25:54 +00:00
log "github.com/sirupsen/logrus"
"github.com/Luzifer/rconfig/v2"
2017-12-02 14:25:54 +00:00
)
var (
cfg = struct {
BaseURL string `flag:"base-url" default:"" description:"URL to prepend before filename"`
BasePath string `flag:"base-path" default:"" description:"DEPRECATED: Path to upload the file to"`
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"`
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"`
2017-12-02 14:25:54 +00:00
VersionAndExit bool `flag:"version" default:"false" description:"Prints current version and exits"`
}{}
//go:embed frontend/*
frontend embed.FS
2017-12-02 14:25:54 +00:00
version = "dev"
)
func initApp() {
rconfig.AutoEnv(true)
rconfig.SetVariableDefaults(map[string]string{
"file_template": `file/{{ printf "%.6s" .Hash }}/{{ .SafeFileName }}`,
})
2017-12-02 14:25:54 +00:00
if err := rconfig.ParseAndValidate(&cfg); err != nil {
log.Fatalf("Unable to parse commandline options: %s", err)
}
if cfg.VersionAndExit {
fmt.Printf("share %s\n", version)
os.Exit(0)
}
if l, err := log.ParseLevel(cfg.LogLevel); err != nil {
log.WithError(err).Fatal("Unable to parse log level")
} else {
log.SetLevel(l)
}
if cfg.BasePath != "" {
cfg.FileTemplate = strings.Join([]string{strings.TrimRight(cfg.BasePath, "/"), `{{ .SafeFileName }}`}, "/")
log.WithField("file-template", cfg.FileTemplate).Warn("Using deprecated base-path parameter! Using update file-template...")
}
2017-12-02 14:25:54 +00:00
}
func main() {
initApp()
switch {
case cfg.Bootstrap:
if err := doBootstrap(); err != nil {
log.WithError(err).Fatal("Bootstrap failed")
}
log.Info("Bucket bootstrap finished: Frontend uploaded")
2017-12-02 14:25:54 +00:00
case cfg.Listen != "":
if err := doListen(); err != nil {
log.WithError(err).Fatal("HTTP server ended unclean")
}
2017-12-02 15:34:21 +00:00
default:
if err := doCLIUpload(); err != nil {
log.WithError(err).Fatal("Upload failed")
}
}
}
func doCLIUpload() error {
if len(rconfig.Args()) == 1 {
return errors.New("missing argument: file to upload")
2017-12-02 14:25:54 +00:00
}
if cfg.BaseURL == "" {
log.Warn("No BaseURL configured, output will be no complete URL")
2017-12-02 14:25:54 +00:00
}
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 {
log.WithError(err).Fatal("Could not read stdin")
}
inFile = bytes.NewReader(buf.Bytes())
} else {
inFileHandle, err := os.Open(inFileName)
if err != nil {
return errors.Wrap(err, "opening source file")
}
defer inFileHandle.Close()
inFile = inFileHandle
2017-12-02 14:25:54 +00:00
}
url, err := executeUpload(inFileName, inFile, true, cfg.ContentType, false)
2017-12-02 14:25:54 +00:00
if err != nil {
return errors.Wrap(err, "uploading file")
2017-12-02 14:25:54 +00:00
}
fmt.Println(url)
return nil
2017-12-02 14:25:54 +00:00
}
func doBootstrap() error {
for _, asset := range []string{"index.html", "app.js", "bundle.css", "bundle.js"} {
content, err := frontend.ReadFile(strings.Join([]string{"frontend", asset}, "/"))
if err != nil {
return errors.Wrap(err, "reading baked asset")
}
if _, err := executeUpload(asset, bytes.NewReader(content), false, "", true); err != nil {
return errors.Wrapf(err, "uploading bootstrap asset %q", asset)
}
2017-12-02 14:25:54 +00:00
}
return nil
2017-12-02 14:25:54 +00:00
}