feat: initial version
This commit is contained in:
@@ -0,0 +1,96 @@
|
||||
package main
|
||||
|
||||
import (
|
||||
"context"
|
||||
"errors"
|
||||
"fmt"
|
||||
"net/http"
|
||||
"os"
|
||||
"os/signal"
|
||||
"sync"
|
||||
"syscall"
|
||||
"time"
|
||||
|
||||
"github.com/alecthomas/kong"
|
||||
"github.com/apex/log"
|
||||
"github.com/apex/log/handlers/json"
|
||||
|
||||
"gitlab.com/unboundsoftware/s3uploader/server"
|
||||
"gitlab.com/unboundsoftware/s3uploader/storage"
|
||||
)
|
||||
|
||||
var CLI struct {
|
||||
Port int `name:"port" env:"PORT" help:"Port which the service listens to" default:"80"`
|
||||
Bucket string `name:"bucket" env:"BUCKET" help:"The AWS S3 bucket where the uploaded objects should be stored" required:"true"`
|
||||
ReturnURL string `name:"return-url" env:"RETURN_URL" help:"Base-url to be prepended to all returned locations" required:"true"`
|
||||
}
|
||||
|
||||
func main() {
|
||||
_ = kong.Parse(&CLI)
|
||||
log.SetHandler(json.New(os.Stdout))
|
||||
logger := log.WithField("service", "s3uploader")
|
||||
|
||||
if err := start(logger); err != nil {
|
||||
logger.WithError(err).Error("process error")
|
||||
}
|
||||
}
|
||||
|
||||
func start(logger log.Interface) error {
|
||||
rootCtx, rootCancel := context.WithCancel(context.Background())
|
||||
defer rootCancel()
|
||||
|
||||
s3, err := storage.New(CLI.Bucket)
|
||||
if err != nil {
|
||||
return fmt.Errorf("storage failed: %w", err)
|
||||
}
|
||||
srv := server.New(s3, CLI.ReturnURL, logger)
|
||||
httpSrvAddr := fmt.Sprintf(":%d", CLI.Port)
|
||||
httpSrv := &http.Server{Addr: httpSrvAddr, Handler: srv}
|
||||
|
||||
wg := sync.WaitGroup{}
|
||||
|
||||
sigint := make(chan os.Signal, 1)
|
||||
signal.Notify(sigint, os.Interrupt, syscall.SIGTERM)
|
||||
|
||||
wg.Add(1)
|
||||
go func() {
|
||||
defer wg.Done()
|
||||
sig := <-sigint
|
||||
if sig != nil {
|
||||
// In case our shutdown logic is broken/incomplete we reset signal
|
||||
// handlers so next signal goes to go itself. Go is more aggressive when
|
||||
// shutting down goroutines
|
||||
signal.Reset(os.Interrupt, syscall.SIGTERM)
|
||||
logger.Info("Got shutdown signal..")
|
||||
rootCancel()
|
||||
}
|
||||
}()
|
||||
|
||||
wg.Add(1)
|
||||
go func() {
|
||||
defer wg.Done()
|
||||
<-rootCtx.Done()
|
||||
|
||||
close(sigint)
|
||||
|
||||
ctx, cancel := context.WithTimeout(context.Background(), 15*time.Second)
|
||||
defer cancel()
|
||||
err := httpSrv.Shutdown(ctx)
|
||||
logger.WithError(err).Info("Shutdown of HTTP server complete")
|
||||
}()
|
||||
|
||||
wg.Add(1)
|
||||
go func() {
|
||||
defer wg.Done()
|
||||
logger.Info(fmt.Sprintf("Serving HTTP API on %s", httpSrvAddr))
|
||||
err := httpSrv.ListenAndServe()
|
||||
if err != nil && !errors.Is(err, http.ErrServerClosed) {
|
||||
logger.WithError(err).Error("HTTP server failed")
|
||||
rootCancel()
|
||||
}
|
||||
}()
|
||||
|
||||
wg.Wait()
|
||||
|
||||
return nil
|
||||
}
|
||||
@@ -0,0 +1,69 @@
|
||||
package main
|
||||
|
||||
import (
|
||||
"os"
|
||||
"syscall"
|
||||
"testing"
|
||||
"time"
|
||||
|
||||
mocks "gitlab.com/unboundsoftware/apex-mocks"
|
||||
)
|
||||
|
||||
func TestMainFunc_Success(t *testing.T) {
|
||||
os.Args = []string{"s3uploader", "--port", "7777", "--bucket", "test-bucket-somewhere", "--return-url", "https://example.org"}
|
||||
go func() {
|
||||
time.Sleep(time.Second)
|
||||
_ = syscall.Kill(syscall.Getpid(), syscall.SIGTERM)
|
||||
}()
|
||||
main()
|
||||
}
|
||||
|
||||
func TestMainFunc_Invalid_AWS_Config(t *testing.T) {
|
||||
os.Args = []string{"s3uploader", "--port", "7777", "--bucket", "test-bucket-somewhere", "--return-url", "https://example.org"}
|
||||
_ = os.Setenv("AWS_STS_REGIONAL_ENDPOINTS", "unknown_value")
|
||||
defer func() {
|
||||
_ = os.Unsetenv("AWS_STS_REGIONAL_ENDPOINTS")
|
||||
}()
|
||||
main()
|
||||
}
|
||||
|
||||
func Test_start(t *testing.T) {
|
||||
type args struct {
|
||||
port int
|
||||
bucket string
|
||||
url string
|
||||
}
|
||||
tests := []struct {
|
||||
name string
|
||||
args args
|
||||
wantErr bool
|
||||
wantLogged []string
|
||||
}{
|
||||
{
|
||||
name: "invalid port",
|
||||
args: args{
|
||||
port: 77777,
|
||||
bucket: "some-bucket",
|
||||
url: "https://example.org",
|
||||
},
|
||||
wantErr: false,
|
||||
wantLogged: []string{
|
||||
"info: Serving HTTP API on :77777",
|
||||
"error: HTTP server failed",
|
||||
"info: Shutdown of HTTP server complete",
|
||||
},
|
||||
},
|
||||
}
|
||||
for _, tt := range tests {
|
||||
t.Run(tt.name, func(t *testing.T) {
|
||||
logger := mocks.New()
|
||||
CLI.Port = tt.args.port
|
||||
CLI.Bucket = tt.args.bucket
|
||||
CLI.ReturnURL = tt.args.url
|
||||
if err := start(logger.Logger); (err != nil) != tt.wantErr {
|
||||
t.Errorf("start() error = %v, wantErr %v", err, tt.wantErr)
|
||||
}
|
||||
logger.Check(t, tt.wantLogged)
|
||||
})
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user