feat: initial version

This commit is contained in:
2021-11-25 18:27:08 +01:00
commit 4c6483d971
14 changed files with 1064 additions and 0 deletions
+96
View File
@@ -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
}
+69
View File
@@ -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)
})
}
}