Files
contentserver/cmd/socket.go

121 lines
3.1 KiB
Go

package cmd
import (
"context"
"fmt"
"net"
"github.com/foomo/contentserver/pkg/handler"
"github.com/foomo/contentserver/pkg/repo"
"github.com/foomo/keel/log"
keelhttp "github.com/foomo/keel/net/http"
"github.com/spf13/cobra"
"github.com/spf13/viper"
"go.uber.org/zap"
)
func NewSocketCommand() *cobra.Command {
v := viper.New()
cmd := &cobra.Command{
Use: "socket <url>",
Short: "Start socket server",
Args: cobra.ExactArgs(1),
ValidArgsFunction: func(cmd *cobra.Command, args []string, toComplete string) ([]string, cobra.ShellCompDirective) {
var comps []string
if len(args) == 0 {
comps = cobra.AppendActiveHelp(comps, "You must specify the URL for the repository you are adding")
} else {
comps = cobra.AppendActiveHelp(comps, "This command does not take any more arguments")
}
return comps, cobra.ShellCompDirectiveNoFileComp
},
RunE: func(cmd *cobra.Command, args []string) error {
l := log.Logger()
// Create storage based on configuration
storage, err := createStorage(cmd.Context(), v, l)
if err != nil {
return fmt.Errorf("failed to create storage: %w", err)
}
history, err := repo.NewHistory(l,
repo.HistoryWithStorage(storage),
repo.HistoryWithHistoryDir(historyDirFlag(v)),
repo.HistoryWithHistoryLimit(historyLimitFlag(v)),
)
if err != nil {
return fmt.Errorf("failed to create history: %w", err)
}
defer func() {
if closeErr := history.Close(); closeErr != nil {
l.Error("failed to close history storage", zap.Error(closeErr))
}
}()
r := repo.New(l,
args[0],
history,
repo.WithHTTPClient(
keelhttp.NewHTTPClient(
keelhttp.HTTPClientWithTimeout(repositoryTimeoutFlag(v)),
keelhttp.HTTPClientWithTelemetry(),
),
),
repo.WithPoll(pollFlag(v)),
repo.WithPollInterval(pollIntevalFlag(v)),
)
// create socket server
handle := handler.NewSocket(l, r)
// listen on socket
var lc net.ListenConfig
ln, err := lc.Listen(cmd.Context(), "tcp", addressFlag(v))
if err != nil {
return err
}
// start repo
up := make(chan bool, 1)
r.OnLoaded(func() {
up <- true
})
go r.Start(context.Background()) //nolint:errcheck
<-up
l.Info("started listening", zap.String("address", addressFlag(v)))
for {
// this blocks until connection or error
conn, err := ln.Accept()
if err != nil {
l.Error("runSocketServer: could not accept connection", zap.Error(err))
continue
}
// a goroutine handles conn so that the loop can accept other connections
go func() {
l.Debug("accepted connection", zap.String("source", conn.RemoteAddr().String()))
handle.Serve(conn)
if err := conn.Close(); err != nil {
l.Warn("failed to close connection", zap.Error(err))
}
}()
}
},
}
flags := cmd.Flags()
addAddressFlag(flags, v)
addPollFlag(flags, v)
addPollIntervalFlag(flags, v)
addHistoryDirFlag(flags, v)
addHistoryLimitFlag(flags, v)
addStorageTypeFlag(flags, v)
addStorageBlobBucketFlag(flags, v)
addStorageBlobPrefixFlag(flags, v)
addRepositoryTimeoutFlag(flags, v)
return cmd
}