Files
flux/internal/handlers/server.go

251 lines
6.3 KiB
Go

package handlers
import (
"archive/tar"
"compress/gzip"
"context"
"database/sql"
"encoding/json"
"fmt"
"io"
"net/http"
"os"
"path/filepath"
"strconv"
_ "embed"
"github.com/docker/docker/api/types/image"
"github.com/google/uuid"
"github.com/juls0730/flux/internal/docker"
"github.com/juls0730/flux/internal/services/appManagerService"
proxyManagerService "github.com/juls0730/flux/internal/services/proxy"
"github.com/juls0730/flux/pkg"
"github.com/juls0730/flux/pkg/API"
_ "github.com/mattn/go-sqlite3"
"go.uber.org/zap"
"go.uber.org/zap/zapcore"
)
var (
//go:embed schema.sql
schema string
DefaultConfig = pkg.DaemonConfig{
Builder: "paketobuildpacks/builder-jammy-tiny",
CompressionLevel: 0,
DaemonHost: "0.0.0.0:5647",
ProxyHost: "0.0.0.0:7465",
}
)
type FluxServer struct {
db *sql.DB
docker *docker.DockerClient
proxy *proxyManagerService.ProxyManager
appManager *appManagerService.AppManager
rootDir string
config pkg.DaemonConfig
logger *zap.SugaredLogger
}
func NewServer() *FluxServer {
flux := &FluxServer{}
verbosity, err := strconv.Atoi(os.Getenv("FLUXD_VERBOSITY"))
if err != nil {
verbosity = 0
}
config := zap.NewProductionConfig()
if os.Getenv("DEBUG") == "true" {
config = zap.NewDevelopmentConfig()
verbosity = -1
}
config.Level = zap.NewAtomicLevelAt(zapcore.Level(verbosity))
lameLogger, err := config.Build()
if err != nil {
fmt.Printf("Failed to create logger: %v\n", err)
os.Exit(1)
}
flux.logger = lameLogger.Sugar()
rootDir := os.Getenv("FLUXD_ROOT_DIR")
if rootDir == "" {
rootDir = "/var/fluxd"
}
flux.rootDir = rootDir
configPath := filepath.Join(flux.rootDir, "config.json")
if _, err := os.Stat(configPath); err != nil {
if err := os.MkdirAll(rootDir, 0755); err != nil {
flux.logger.Fatalw("Failed to create fluxd directory", zap.Error(err))
}
configBytes, err := json.Marshal(DefaultConfig)
if err != nil {
flux.logger.Fatalw("Failed to marshal default config", zap.Error(err))
}
fmt.Printf("Config file not found creating default config file at %s\n", configPath)
if err := os.WriteFile(configPath, configBytes, 0644); err != nil {
flux.logger.Fatalw("Failed to write config file", zap.Error(err))
}
}
configFile, err := os.ReadFile(configPath)
if err != nil {
flux.logger.Fatalw("Failed to read config file", zap.Error(err))
}
// apply the config file over the default config, this way if we have missing fields, they will be filled in with
// the default values
flux.config = DefaultConfig
if err := json.Unmarshal(configFile, &flux.config); err != nil {
flux.logger.Fatalw("Failed to parse config file", zap.Error(err))
}
dbPath := filepath.Join(flux.rootDir, "fluxd.db")
flux.db, err = sql.Open("sqlite3", dbPath)
if err != nil {
flux.logger.Fatalw("Failed to open database", zap.Error(err))
}
err = flux.db.Ping()
if err != nil {
flux.logger.Fatalw("Failed to ping database", zap.Error(err))
}
_, err = flux.db.Exec(schema)
if err != nil {
flux.logger.Fatalw("Failed to create database schema", zap.Error(err))
}
flux.docker = docker.NewDocker(nil, flux.logger)
if err != nil {
flux.logger.Fatalw("Failed to create docker client", zap.Error(err))
}
flux.logger.Infof("Pulling builder image %s this may take a while...", flux.config.Builder)
events, err := flux.docker.ImagePull(context.Background(), fmt.Sprintf("%s:latest", flux.config.Builder), image.PullOptions{})
if err != nil {
flux.logger.Fatalw("Failed to pull builder image", zap.Error(err))
}
// blocking until the iamge is pulled
io.Copy(io.Discard, events)
flux.proxy = proxyManagerService.NewProxyManager(flux.logger)
flux.appManager = appManagerService.NewAppManager(flux.db, flux.docker, flux.proxy, flux.logger)
err = flux.appManager.Init()
if err != nil {
flux.logger.Fatalw("Failed to initialize apps", zap.Error(err))
}
return flux
}
func (s *FluxServer) Stop() {
s.logger.Sync()
}
func (s *FluxServer) ListenAndServe() error {
s.logger.Infow("Starting server", zap.String("daemon_host", s.config.DaemonHost), zap.String("proxy_host", s.config.ProxyHost))
go s.proxy.ListenAndServe(s.config.ProxyHost)
return http.ListenAndServe(s.config.DaemonHost, nil)
}
func (s *FluxServer) DaemonInfoHandler(w http.ResponseWriter, r *http.Request) {
w.Header().Set("Content-Type", "application/json")
json.NewEncoder(w).Encode(API.Info{
CompressionLevel: s.config.CompressionLevel,
Version: pkg.Version,
})
}
// This extracts and uploads a tar file to a temporary directory, and returns the path to the directory
func (s *FluxServer) UploadAppCode(code io.Reader, appId uuid.UUID) (string, error) {
var err error
outputPath, err := os.MkdirTemp(os.TempDir(), appId.String())
if err != nil {
s.logger.Errorw("Failed to create project directory", zap.Error(err))
return "", err
}
var gzReader *gzip.Reader
defer func() {
if gzReader != nil {
gzReader.Close()
}
}()
if s.config.CompressionLevel > 0 {
gzReader, err = gzip.NewReader(code)
if err != nil {
s.logger.Infow("Failed to create gzip reader", zap.Error(err))
return "", err
}
}
var tarReader *tar.Reader
if gzReader != nil {
tarReader = tar.NewReader(gzReader)
} else {
tarReader = tar.NewReader(code)
}
s.logger.Infow("Extracting files for project", zap.String("project", outputPath))
for {
header, err := tarReader.Next()
if err == io.EOF {
break
}
if err != nil {
s.logger.Debugw("Failed to read tar header", zap.Error(err))
return "", err
}
// Construct full path
path := filepath.Join(outputPath, header.Name)
// Handle different file types
switch header.Typeflag {
case tar.TypeDir:
if err = os.MkdirAll(path, 0755); err != nil {
s.logger.Debugw("Failed to extract directory", zap.Error(err))
return "", err
}
case tar.TypeReg:
if err = os.MkdirAll(filepath.Dir(path), 0755); err != nil {
s.logger.Debugw("Failed to extract directory", zap.Error(err))
return "", err
}
outFile, err := os.Create(path)
if err != nil {
s.logger.Debugw("Failed to extract file", zap.Error(err))
return "", err
}
defer outFile.Close()
if _, err = io.Copy(outFile, tarReader); err != nil {
s.logger.Debugw("Failed to copy file during extraction", zap.Error(err))
return "", err
}
}
}
return outputPath, nil
}