252 lines
6.1 KiB
Go
252 lines
6.1 KiB
Go
package server
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"os"
|
|
"path/filepath"
|
|
"sync"
|
|
|
|
"github.com/juls0730/flux/pkg"
|
|
"go.uber.org/zap"
|
|
)
|
|
|
|
type App struct {
|
|
ID int64 `json:"id,omitempty"`
|
|
Deployment *Deployment `json:"-"`
|
|
Name string `json:"name,omitempty"`
|
|
DeploymentID int64 `json:"deployment_id,omitempty"`
|
|
}
|
|
|
|
func CreateApp(ctx context.Context, imageName string, projectPath string, projectConfig pkg.ProjectConfig) (*App, error) {
|
|
app := &App{
|
|
Name: projectConfig.Name,
|
|
}
|
|
logger.Debugw("Creating deployment", zap.String("name", app.Name))
|
|
|
|
deployment, err := CreateDeployment(projectConfig.Port, projectConfig.Url, Flux.db)
|
|
app.Deployment = deployment
|
|
if err != nil {
|
|
logger.Errorw("Failed to create deployment", zap.Error(err))
|
|
return nil, err
|
|
}
|
|
|
|
container, err := CreateContainer(ctx, imageName, projectPath, projectConfig, true, deployment)
|
|
if err != nil || container == nil {
|
|
return nil, fmt.Errorf("failed to create container: %v", err)
|
|
}
|
|
|
|
if appInsertStmt == nil {
|
|
appInsertStmt, err = Flux.db.Prepare("INSERT INTO apps (name, deployment_id) VALUES ($1, $2) RETURNING id, name, deployment_id")
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to prepare statement: %v", err)
|
|
}
|
|
}
|
|
|
|
// create app in the database
|
|
err = appInsertStmt.QueryRow(projectConfig.Name, deployment.ID).Scan(&app.ID, &app.Name, &app.DeploymentID)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to insert app: %v", err)
|
|
}
|
|
|
|
err = deployment.Start(ctx)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to start deployment: %v", err)
|
|
}
|
|
|
|
Flux.appManager.AddApp(app.Name, app)
|
|
|
|
return app, nil
|
|
}
|
|
|
|
func (app *App) Upgrade(ctx context.Context, projectConfig pkg.ProjectConfig, imageName string, projectPath string) error {
|
|
logger.Debugw("Upgrading deployment", zap.String("name", app.Name))
|
|
|
|
// if deploy is not started, start it
|
|
deploymentStatus, err := app.Deployment.Status(ctx)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to get deployment status: %v", err)
|
|
}
|
|
|
|
if deploymentStatus != "running" {
|
|
err = app.Deployment.Start(ctx)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to start deployment: %v", err)
|
|
}
|
|
}
|
|
|
|
err = app.Deployment.Upgrade(ctx, projectConfig, imageName, projectPath)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to upgrade deployment: %v", err)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (app *App) Remove(ctx context.Context) error {
|
|
Flux.appManager.RemoveApp(app.Name)
|
|
|
|
err := app.Deployment.Remove(ctx)
|
|
if err != nil {
|
|
logger.Errorw("Failed to remove deployment", zap.Error(err))
|
|
return err
|
|
}
|
|
|
|
_, err = Flux.db.Exec("DELETE FROM apps WHERE id = ?", app.ID)
|
|
if err != nil {
|
|
logger.Errorw("Failed to delete app", zap.Error(err))
|
|
return err
|
|
}
|
|
|
|
projectPath := filepath.Join(Flux.rootDir, "apps", app.Name)
|
|
err = os.RemoveAll(projectPath)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to remove project directory: %v", err)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
type AppManager struct {
|
|
sync.Map
|
|
}
|
|
|
|
func (am *AppManager) GetApp(name string) *App {
|
|
app, exists := am.Load(name)
|
|
if !exists {
|
|
return nil
|
|
}
|
|
|
|
return app.(*App)
|
|
}
|
|
|
|
func (am *AppManager) GetAllApps() []*App {
|
|
var apps []*App
|
|
am.Range(func(key, value interface{}) bool {
|
|
if app, ok := value.(*App); ok {
|
|
apps = append(apps, app)
|
|
}
|
|
return true
|
|
})
|
|
return apps
|
|
}
|
|
|
|
func (am *AppManager) RemoveApp(name string) {
|
|
am.Delete(name)
|
|
}
|
|
|
|
func (am *AppManager) AddApp(name string, app *App) {
|
|
if app.Deployment.Containers == nil || app.Deployment.Head == nil || len(app.Deployment.Containers) == 0 {
|
|
panic("nil containers")
|
|
}
|
|
|
|
am.Store(name, app)
|
|
}
|
|
|
|
func (am *AppManager) DeleteApp(name string) error {
|
|
app := am.GetApp(name)
|
|
if app == nil {
|
|
return fmt.Errorf("app not found")
|
|
}
|
|
|
|
err := app.Remove(context.Background())
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
am.Delete(name)
|
|
|
|
return nil
|
|
}
|
|
|
|
func (am *AppManager) Init() {
|
|
logger.Info("Initializing deployments")
|
|
|
|
if Flux.db == nil {
|
|
logger.Panic("DB is nil")
|
|
}
|
|
|
|
rows, err := Flux.db.Query("SELECT id, name, deployment_id FROM apps")
|
|
if err != nil {
|
|
logger.Warnw("Failed to query apps", zap.Error(err))
|
|
return
|
|
}
|
|
defer rows.Close()
|
|
|
|
var apps []App
|
|
for rows.Next() {
|
|
var app App
|
|
if err := rows.Scan(&app.ID, &app.Name, &app.DeploymentID); err != nil {
|
|
logger.Warnw("Failed to scan app", zap.Error(err))
|
|
return
|
|
}
|
|
apps = append(apps, app)
|
|
}
|
|
|
|
for _, app := range apps {
|
|
deployment := &Deployment{}
|
|
var headContainer *Container
|
|
Flux.db.QueryRow("SELECT id, url, port FROM deployments WHERE id = ?", app.DeploymentID).Scan(&deployment.ID, &deployment.URL, &deployment.Port)
|
|
deployment.Containers = make([]*Container, 0)
|
|
|
|
rows, err = Flux.db.Query("SELECT id, container_id, deployment_id, head FROM containers WHERE deployment_id = ?", app.DeploymentID)
|
|
if err != nil {
|
|
logger.Warnw("Failed to query containers", zap.Error(err))
|
|
return
|
|
}
|
|
defer rows.Close()
|
|
|
|
for rows.Next() {
|
|
var container Container
|
|
var containerIDString string
|
|
rows.Scan(&container.ID, &containerIDString, &container.DeploymentID, &container.Head)
|
|
container.Deployment = deployment
|
|
copy(container.ContainerID[:], containerIDString)
|
|
|
|
if container.Head {
|
|
if headContainer != nil {
|
|
logger.Fatal("Several containers are marked as head")
|
|
}
|
|
|
|
headContainer = &container
|
|
}
|
|
|
|
rows, err := Flux.db.Query("SELECT id, volume_id, container_id, mountpoint FROM volumes WHERE container_id = ?", container.ContainerID[:])
|
|
if err != nil {
|
|
logger.Warnw("Failed to query volumes", zap.Error(err))
|
|
return
|
|
}
|
|
defer rows.Close()
|
|
|
|
for rows.Next() {
|
|
var volume Volume
|
|
rows.Scan(&volume.ID, &volume.VolumeID, &volume.ContainerID, &volume.Mountpoint)
|
|
container.Volumes = append(container.Volumes, volume)
|
|
}
|
|
|
|
deployment.Containers = append(deployment.Containers, &container)
|
|
}
|
|
|
|
if headContainer == nil {
|
|
logger.Fatal("head container is nil!")
|
|
}
|
|
|
|
deployment.Head = headContainer
|
|
app.Deployment = deployment
|
|
am.AddApp(app.Name, &app)
|
|
|
|
status, err := deployment.Status(context.Background())
|
|
if err != nil {
|
|
logger.Warnw("Failed to get deployment status", zap.Error(err))
|
|
continue
|
|
}
|
|
|
|
if status != "running" {
|
|
continue
|
|
}
|
|
|
|
deployment.Proxy, _ = deployment.NewDeploymentProxy()
|
|
Flux.proxy.AddDeployment(deployment)
|
|
}
|
|
}
|