cloudfoundry/korifi

View on GitHub
api/handlers/app.go

Summary

Maintainability
B
4 hrs
Test Coverage
A
97%
File `app.go` has 648 lines of code (exceeds 500 allowed). Consider refactoring.
package handlers
 
import (
"context"
"errors"
"fmt"
"net/http"
"net/url"
"strconv"
"time"
 
"code.cloudfoundry.org/korifi/api/authorization"
apierrors "code.cloudfoundry.org/korifi/api/errors"
"code.cloudfoundry.org/korifi/api/payloads"
"code.cloudfoundry.org/korifi/api/presenter"
"code.cloudfoundry.org/korifi/api/repositories"
"code.cloudfoundry.org/korifi/api/routing"
"code.cloudfoundry.org/korifi/api/tools/singleton"
korifiv1alpha1 "code.cloudfoundry.org/korifi/controllers/api/v1alpha1"
 
"github.com/go-logr/logr"
)
 
const (
AppsPath = "/v3/apps"
AppPath = "/v3/apps/{guid}"
AppCurrentDropletRelationshipPath = "/v3/apps/{guid}/relationships/current_droplet"
AppDropletsPath = "/v3/apps/{guid}/droplets"
AppCurrentDropletPath = "/v3/apps/{guid}/droplets/current"
AppProcessesPath = "/v3/apps/{guid}/processes"
AppProcessByTypePath = "/v3/apps/{guid}/processes/{type}"
AppProcessStatsByTypePath = "/v3/apps/{guid}/processes/{type}/stats"
AppProcessScalePath = "/v3/apps/{guid}/processes/{processType}/actions/scale"
AppRoutesPath = "/v3/apps/{guid}/routes"
AppStartPath = "/v3/apps/{guid}/actions/start"
AppStopPath = "/v3/apps/{guid}/actions/stop"
AppRestartPath = "/v3/apps/{guid}/actions/restart"
AppEnvVarsPath = "/v3/apps/{guid}/environment_variables"
AppEnvPath = "/v3/apps/{guid}/env"
AppFeaturePath = "/v3/apps/{guid}/features/{name}"
AppPackagesPath = "/v3/apps/{guid}/packages"
AppSSHEnabledPath = "/v3/apps/{guid}/ssh_enabled"
AppInstanceRestartPath = "/v3/apps/{guid}/processes/{processType}/instances/{instance}"
invalidDropletMsg = "Unable to assign current droplet. Ensure the droplet exists and belongs to this app."
 
AppStartedState = "STARTED"
AppStoppedState = "STOPPED"
AppRevisionKey = "korifi.cloudfoundry.org/app-rev"
)
 
//counterfeiter:generate -o fake -fake-name CFAppRepository . CFAppRepository
type CFAppRepository interface {
GetApp(context.Context, authorization.Info, string) (repositories.AppRecord, error)
ListApps(context.Context, authorization.Info, repositories.ListAppsMessage) ([]repositories.AppRecord, error)
PatchAppEnvVars(context.Context, authorization.Info, repositories.PatchAppEnvVarsMessage) (repositories.AppEnvVarsRecord, error)
CreateApp(context.Context, authorization.Info, repositories.CreateAppMessage) (repositories.AppRecord, error)
SetCurrentDroplet(context.Context, authorization.Info, repositories.SetCurrentDropletMessage) (repositories.CurrentDropletRecord, error)
SetAppDesiredState(context.Context, authorization.Info, repositories.SetAppDesiredStateMessage) (repositories.AppRecord, error)
DeleteApp(context.Context, authorization.Info, repositories.DeleteAppMessage) error
GetAppEnv(context.Context, authorization.Info, string) (repositories.AppEnvRecord, error)
PatchApp(context.Context, authorization.Info, repositories.PatchAppMessage) (repositories.AppRecord, error)
}
 
//counterfeiter:generate -o fake -fake-name PodRepository . PodRepository
type PodRepository interface {
DeletePod(context.Context, authorization.Info, string, repositories.ProcessRecord, string) error
}
 
`App` has 29 methods (exceeds 20 allowed). Consider refactoring.
type App struct {
serverURL url.URL
appRepo CFAppRepository
dropletRepo CFDropletRepository
processRepo CFProcessRepository
routeRepo CFRouteRepository
domainRepo CFDomainRepository
spaceRepo CFSpaceRepository
packageRepo CFPackageRepository
requestValidator RequestValidator
podRepo PodRepository
gaugesCollector GaugesCollector
instancesStateCollector InstancesStateCollector
}
 
func NewApp(
Function `NewApp` has 12 arguments (exceeds 5 allowed). Consider refactoring.
serverURL url.URL,
appRepo CFAppRepository,
dropletRepo CFDropletRepository,
processRepo CFProcessRepository,
routeRepo CFRouteRepository,
domainRepo CFDomainRepository,
spaceRepo CFSpaceRepository,
packageRepo CFPackageRepository,
requestValidator RequestValidator,
podRepo PodRepository,
gaugesCollector GaugesCollector,
instancesStateCollector InstancesStateCollector,
) *App {
return &App{
serverURL: serverURL,
appRepo: appRepo,
dropletRepo: dropletRepo,
processRepo: processRepo,
routeRepo: routeRepo,
domainRepo: domainRepo,
spaceRepo: spaceRepo,
packageRepo: packageRepo,
requestValidator: requestValidator,
podRepo: podRepo,
gaugesCollector: gaugesCollector,
instancesStateCollector: instancesStateCollector,
}
}
 
func (h *App) get(r *http.Request) (*routing.Response, error) {
appGUID := routing.URLParam(r, "guid")
authInfo, _ := authorization.InfoFromContext(r.Context())
logger := logr.FromContextOrDiscard(r.Context()).WithName("handlers.app.get")
 
app, err := h.appRepo.GetApp(r.Context(), authInfo, appGUID)
if err != nil {
return nil, apierrors.LogAndReturn(logger, apierrors.ForbiddenAsNotFound(err), "Failed to fetch app from Kubernetes", "GUID", appGUID)
}
return routing.NewResponse(http.StatusOK).WithBody(presenter.ForApp(app, h.serverURL)), nil
}
 
//nolint:dupl
func (h *App) create(r *http.Request) (*routing.Response, error) {
authInfo, _ := authorization.InfoFromContext(r.Context())
logger := logr.FromContextOrDiscard(r.Context()).WithName("handlers.app.create")
var payload payloads.AppCreate
if err := h.requestValidator.DecodeAndValidateJSONPayload(r, &payload); err != nil {
return nil, apierrors.LogAndReturn(logger, err, "failed to decode json payload")
}
 
spaceGUID := payload.Relationships.Space.Data.GUID
_, err := h.spaceRepo.GetSpace(r.Context(), authInfo, spaceGUID)
if err != nil {
return nil, apierrors.LogAndReturn(
logger,
apierrors.AsUnprocessableEntity(err, "Invalid space. Ensure that the space exists and you have access to it.", apierrors.NotFoundError{}, apierrors.ForbiddenError{}),
"spaceGUID", spaceGUID,
)
}
 
appRecord, err := h.appRepo.CreateApp(r.Context(), authInfo, payload.ToAppCreateMessage())
if err != nil {
return nil, apierrors.LogAndReturn(logger, err, "Failed to create app", "App Name", payload.Name)
}
 
err = h.processRepo.CreateProcess(r.Context(), authInfo, repositories.CreateProcessMessage{
AppGUID: appRecord.GUID,
SpaceGUID: spaceGUID,
Type: korifiv1alpha1.ProcessTypeWeb,
})
if err != nil {
return nil, apierrors.LogAndReturn(logger, err, "Failed to create web process", "App Name", payload.Name)
}
 
return routing.NewResponse(http.StatusCreated).WithBody(presenter.ForApp(appRecord, h.serverURL)), nil
}
 
func (h *App) list(r *http.Request) (*routing.Response, error) { //nolint:dupl
authInfo, _ := authorization.InfoFromContext(r.Context())
logger := logr.FromContextOrDiscard(r.Context()).WithName("handlers.app.list")
 
payload := new(payloads.AppList)
err := h.requestValidator.DecodeAndValidateURLValues(r, payload)
if err != nil {
return nil, apierrors.LogAndReturn(logger, err, "Unable to decode request query parameters")
}
 
appList, err := h.appRepo.ListApps(r.Context(), authInfo, payload.ToMessage())
if err != nil {
return nil, apierrors.LogAndReturn(logger, err, "Failed to fetch app(s) from Kubernetes")
}
 
return routing.NewResponse(http.StatusOK).WithBody(presenter.ForList(presenter.ForApp, appList, h.serverURL, *r.URL)), nil
}
 
func (h *App) setCurrentDroplet(r *http.Request) (*routing.Response, error) {
authInfo, _ := authorization.InfoFromContext(r.Context())
logger := logr.FromContextOrDiscard(r.Context()).WithName("handlers.app.set-current-droplet")
appGUID := routing.URLParam(r, "guid")
 
var payload payloads.AppSetCurrentDroplet
if err := h.requestValidator.DecodeAndValidateJSONPayload(r, &payload); err != nil {
return nil, apierrors.LogAndReturn(logger, err, "failed to decode json payload")
}
 
app, err := h.appRepo.GetApp(r.Context(), authInfo, appGUID)
if err != nil {
return nil, apierrors.LogAndReturn(logger, apierrors.ForbiddenAsNotFound(err), "Failed to fetch app from Kubernetes", "AppGUID", appGUID)
}
 
dropletGUID := payload.Data.GUID
droplet, err := h.dropletRepo.GetDroplet(r.Context(), authInfo, dropletGUID)
if err != nil {
return nil, apierrors.LogAndReturn(
logger,
apierrors.AsUnprocessableEntity(err, invalidDropletMsg, apierrors.ForbiddenError{}, apierrors.NotFoundError{}),
"Error fetching droplet",
)
}
 
if droplet.AppGUID != appGUID {
return nil, apierrors.LogAndReturn(
logger,
apierrors.NewUnprocessableEntityError(fmt.Errorf("droplet %s does not belong to app %s", droplet.GUID, appGUID), invalidDropletMsg),
invalidDropletMsg,
)
}
 
currentDroplet, err := h.appRepo.SetCurrentDroplet(r.Context(), authInfo, repositories.SetCurrentDropletMessage{
AppGUID: appGUID,
DropletGUID: dropletGUID,
SpaceGUID: app.SpaceGUID,
})
if err != nil {
return nil, apierrors.LogAndReturn(logger, err, "Error setting current droplet")
}
 
return routing.NewResponse(http.StatusOK).WithBody(presenter.ForCurrentDroplet(currentDroplet, h.serverURL)), nil
}
 
func (h *App) listDroplets(r *http.Request) (*routing.Response, error) {
authInfo, _ := authorization.InfoFromContext(r.Context())
logger := logr.FromContextOrDiscard(r.Context()).WithName("handlers.app.get-droplets")
appGUID := routing.URLParam(r, "guid")
 
app, err := h.appRepo.GetApp(r.Context(), authInfo, appGUID)
if err != nil {
return nil, apierrors.LogAndReturn(logger, apierrors.ForbiddenAsNotFound(err), "Failed to fetch app from Kubernetes", "AppGUID", appGUID)
}
 
droplets, err := h.dropletRepo.ListDroplets(r.Context(), authInfo, repositories.ListDropletsMessage{AppGUIDs: []string{appGUID}})
if err != nil {
return nil, apierrors.LogAndReturn(logger, apierrors.ForbiddenAsNotFound(err), "Failed to fetch droplet from Kubernetes", "dropletGUID", app.DropletGUID)
}
 
return routing.NewResponse(http.StatusOK).WithBody(presenter.ForList(presenter.ForDroplet, droplets, h.serverURL, *r.URL)), nil
}
 
func (h *App) getCurrentDroplet(r *http.Request) (*routing.Response, error) {
authInfo, _ := authorization.InfoFromContext(r.Context())
logger := logr.FromContextOrDiscard(r.Context()).WithName("handlers.app.get-current-droplet")
appGUID := routing.URLParam(r, "guid")
 
app, err := h.appRepo.GetApp(r.Context(), authInfo, appGUID)
if err != nil {
return nil, apierrors.LogAndReturn(logger, apierrors.ForbiddenAsNotFound(err), "Failed to fetch app from Kubernetes", "AppGUID", appGUID)
}
 
if app.DropletGUID == "" {
return nil, apierrors.LogAndReturn(
logger,
apierrors.DropletForbiddenAsNotFound(apierrors.NewNotFoundError(nil, repositories.DropletResourceType)),
"App does not have a current droplet assigned",
"appGUID", app.GUID,
)
}
 
droplet, err := h.dropletRepo.GetDroplet(r.Context(), authInfo, app.DropletGUID)
if err != nil {
return nil, apierrors.LogAndReturn(logger, apierrors.DropletForbiddenAsNotFound(err), "Failed to fetch droplet from Kubernetes", "dropletGUID", app.DropletGUID)
}
 
return routing.NewResponse(http.StatusOK).WithBody(presenter.ForDroplet(droplet, h.serverURL)), nil
}
 
func (h *App) start(r *http.Request) (*routing.Response, error) {
authInfo, _ := authorization.InfoFromContext(r.Context())
logger := logr.FromContextOrDiscard(r.Context()).WithName("handlers.app.start")
appGUID := routing.URLParam(r, "guid")
 
app, err := h.appRepo.GetApp(r.Context(), authInfo, appGUID)
if err != nil {
return nil, apierrors.LogAndReturn(logger, apierrors.ForbiddenAsNotFound(err), "Failed to fetch app from Kubernetes", "AppGUID", appGUID)
}
 
app, err = h.startApp(r.Context(), authInfo, app)
if err != nil {
return nil, apierrors.LogAndReturn(logger, err, "Failed to start app", "AppGUID", appGUID)
}
 
return routing.NewResponse(http.StatusOK).WithBody(presenter.ForApp(app, h.serverURL)), nil
}
 
func (h *App) startApp(ctx context.Context, authInfo authorization.Info, app repositories.AppRecord) (repositories.AppRecord, error) {
if app.DropletGUID == "" {
return repositories.AppRecord{}, apierrors.NewUnprocessableEntityError(errors.New("app droplet not set"), "Assign a droplet before starting this app.")
}
 
app, err := h.appRepo.SetAppDesiredState(ctx, authInfo, repositories.SetAppDesiredStateMessage{
AppGUID: app.GUID,
SpaceGUID: app.SpaceGUID,
DesiredState: AppStartedState,
})
if err != nil {
return repositories.AppRecord{}, fmt.Errorf("failed to update app desired state: %w", err)
}
 
return app, nil
}
 
func (h *App) stop(r *http.Request) (*routing.Response, error) {
authInfo, _ := authorization.InfoFromContext(r.Context())
logger := logr.FromContextOrDiscard(r.Context()).WithName("handlers.app.stop")
appGUID := routing.URLParam(r, "guid")
 
app, err := h.appRepo.GetApp(r.Context(), authInfo, appGUID)
if err != nil {
return nil, apierrors.LogAndReturn(logger, apierrors.ForbiddenAsNotFound(err), "Failed to fetch app from Kubernetes", "AppGUID", appGUID)
}
 
app, err = h.stopApp(r.Context(), authInfo, app)
if err != nil {
return nil, apierrors.LogAndReturn(logger, err, "Failed to stop app", "AppGUID", appGUID)
}
 
return routing.NewResponse(http.StatusOK).WithBody(presenter.ForApp(app, h.serverURL)), nil
}
 
func (h *App) stopApp(ctx context.Context, authInfo authorization.Info, app repositories.AppRecord) (repositories.AppRecord, error) {
app, err := h.appRepo.SetAppDesiredState(ctx, authInfo, repositories.SetAppDesiredStateMessage{
AppGUID: app.GUID,
SpaceGUID: app.SpaceGUID,
DesiredState: AppStoppedState,
})
if err != nil {
return repositories.AppRecord{}, fmt.Errorf("failed to update app desired state: %w", err)
}
 
return app, nil
}
 
func (h *App) getProcesses(r *http.Request) (*routing.Response, error) {
authInfo, _ := authorization.InfoFromContext(r.Context())
logger := logr.FromContextOrDiscard(r.Context()).WithName("handlers.app.get-processes")
appGUID := routing.URLParam(r, "guid")
 
app, err := h.appRepo.GetApp(r.Context(), authInfo, appGUID)
if err != nil {
return nil, apierrors.LogAndReturn(logger, apierrors.ForbiddenAsNotFound(err), "Failed to fetch app from Kubernetes", "AppGUID", appGUID)
}
 
fetchProcessesForAppMessage := repositories.ListProcessesMessage{
AppGUIDs: []string{appGUID},
SpaceGUID: app.SpaceGUID,
}
 
processList, err := h.processRepo.ListProcesses(r.Context(), authInfo, fetchProcessesForAppMessage)
if err != nil {
return nil, apierrors.LogAndReturn(logger, err, "Failed to fetch app Process(es) from Kubernetes")
}
 
return routing.NewResponse(http.StatusOK).WithBody(presenter.ForProcessList(processList, h.serverURL, *r.URL)), nil
}
 
func (h *App) getRoutes(r *http.Request) (*routing.Response, error) {
authInfo, _ := authorization.InfoFromContext(r.Context())
logger := logr.FromContextOrDiscard(r.Context()).WithName("handlers.app.get-routes")
appGUID := routing.URLParam(r, "guid")
 
app, err := h.appRepo.GetApp(r.Context(), authInfo, appGUID)
if err != nil {
return nil, apierrors.LogAndReturn(logger, apierrors.ForbiddenAsNotFound(err), "Failed to fetch app from Kubernetes", "AppGUID", appGUID)
}
 
routes, err := h.lookupAppRouteAndDomainList(r.Context(), authInfo, app.GUID, app.SpaceGUID)
if err != nil {
return nil, apierrors.LogAndReturn(logger, err, "Failed to fetch route or domains from Kubernetes")
}
 
return routing.NewResponse(http.StatusOK).WithBody(presenter.ForList(presenter.ForRoute, routes, h.serverURL, *r.URL)), nil
}
 
func (h *App) scaleProcess(r *http.Request) (*routing.Response, error) {
authInfo, _ := authorization.InfoFromContext(r.Context())
appGUID := routing.URLParam(r, "guid")
processType := routing.URLParam(r, "processType")
logger := logr.FromContextOrDiscard(r.Context()).
WithName("handlers.app.scale-process").
WithValues("appGUID", appGUID, "processType", processType)
 
var payload payloads.ProcessScale
if err := h.requestValidator.DecodeAndValidateJSONPayload(r, &payload); err != nil {
return nil, apierrors.LogAndReturn(logger, err, "failed to decode json payload")
}
 
app, err := h.appRepo.GetApp(r.Context(), authInfo, appGUID)
if err != nil {
return nil, apierrors.LogAndReturn(logger, apierrors.ForbiddenAsNotFound(err), "falied to get app")
}
 
appProcesses, err := h.processRepo.ListProcesses(r.Context(), authInfo, repositories.ListProcessesMessage{
AppGUIDs: []string{app.GUID},
SpaceGUID: app.SpaceGUID,
})
if err != nil {
return nil, apierrors.LogAndReturn(logger, err, "failed to list processes for app")
}
 
process, hasProcessType := findProcessType(appProcesses, processType)
if !hasProcessType {
return nil, apierrors.LogAndReturn(logger,
apierrors.NewNotFoundError(nil, repositories.ProcessResourceType),
"app does not have required process type",
)
}
 
scaledProcessRecord, err := h.processRepo.ScaleProcess(r.Context(), authInfo, repositories.ScaleProcessMessage{
GUID: process.GUID,
SpaceGUID: app.SpaceGUID,
ProcessScaleValues: payload.ToRecord(),
})
if err != nil {
return nil, apierrors.LogAndReturn(logger, err, "Failed due to error from Kubernetes", "appGUID", appGUID)
}
 
return routing.NewResponse(http.StatusOK).WithBody(presenter.ForProcess(scaledProcessRecord, h.serverURL)), nil
}
 
func findProcessType(processes []repositories.ProcessRecord, processType string) (repositories.ProcessRecord, bool) {
for _, proc := range processes {
if proc.Type == processType {
return proc, true
}
}
 
return repositories.ProcessRecord{}, false
}
 
func (h *App) restart(r *http.Request) (*routing.Response, error) {
authInfo, _ := authorization.InfoFromContext(r.Context())
logger := logr.FromContextOrDiscard(r.Context()).WithName("handlers.app.restart")
appGUID := routing.URLParam(r, "guid")
 
app, err := h.appRepo.GetApp(r.Context(), authInfo, appGUID)
if err != nil {
return nil, apierrors.LogAndReturn(logger, apierrors.ForbiddenAsNotFound(err), "Failed to fetch app from Kubernetes", "AppGUID", appGUID)
}
 
app, err = h.stopApp(r.Context(), authInfo, app)
if err != nil {
return nil, apierrors.LogAndReturn(logger, err, "Failed to stop app", "AppGUID", appGUID)
}
 
app, err = h.startApp(r.Context(), authInfo, app)
if err != nil {
return nil, apierrors.LogAndReturn(logger, err, "Failed to start app", "AppGUID", appGUID)
}
 
return routing.NewResponse(http.StatusOK).WithBody(presenter.ForApp(app, h.serverURL)), nil
}
 
func (h *App) delete(r *http.Request) (*routing.Response, error) {
authInfo, _ := authorization.InfoFromContext(r.Context())
logger := logr.FromContextOrDiscard(r.Context()).WithName("handlers.app.delete")
appGUID := routing.URLParam(r, "guid")
 
app, err := h.appRepo.GetApp(r.Context(), authInfo, appGUID)
if err != nil {
return nil, apierrors.LogAndReturn(logger, apierrors.ForbiddenAsNotFound(err), "Failed to fetch app from Kubernetes", "AppGUID", appGUID)
}
 
err = h.appRepo.DeleteApp(r.Context(), authInfo, repositories.DeleteAppMessage{
AppGUID: appGUID,
SpaceGUID: app.SpaceGUID,
})
if err != nil {
return nil, apierrors.LogAndReturn(logger, err, "Failed to delete app", "AppGUID", appGUID)
}
 
return routing.NewResponse(http.StatusAccepted).WithHeader("Location", presenter.JobURLForRedirects(appGUID, presenter.AppDeleteOperation, h.serverURL)), nil
}
 
func (h *App) lookupAppRouteAndDomainList(ctx context.Context, authInfo authorization.Info, appGUID, spaceGUID string) ([]repositories.RouteRecord, error) {
routeRecords, err := h.routeRepo.ListRoutesForApp(ctx, authInfo, appGUID, spaceGUID)
if err != nil {
return []repositories.RouteRecord{}, err
}
 
return getDomainsForRoutes(ctx, h.domainRepo, authInfo, routeRecords)
}
 
func getDomainsForRoutes(ctx context.Context, domainRepo CFDomainRepository, authInfo authorization.Info, routeRecords []repositories.RouteRecord) ([]repositories.RouteRecord, error) {
domainGUIDToDomainRecord := make(map[string]repositories.DomainRecord)
for i, routeRecord := range routeRecords {
currentDomainGUID := routeRecord.Domain.GUID
domainRecord, has := domainGUIDToDomainRecord[currentDomainGUID]
if !has {
var err error
domainRecord, err = domainRepo.GetDomain(ctx, authInfo, currentDomainGUID)
if err != nil {
// err = errors.New("resource not found for route's specified domain ref")
return []repositories.RouteRecord{}, err
}
domainGUIDToDomainRecord[currentDomainGUID] = domainRecord
}
routeRecords[i].Domain = domainRecord
}
 
return routeRecords, nil
}
 
func (h *App) getEnvVars(r *http.Request) (*routing.Response, error) {
authInfo, _ := authorization.InfoFromContext(r.Context())
logger := logr.FromContextOrDiscard(r.Context()).WithName("handlers.app.get-env-vars")
appGUID := routing.URLParam(r, "guid")
 
appEnvRecord, err := h.appRepo.GetAppEnv(r.Context(), authInfo, appGUID)
if err != nil {
return nil, apierrors.LogAndReturn(logger, err, "Failed to fetch app environment variables", "AppGUID", appGUID)
}
 
appEnvVarsRecord := repositories.AppEnvVarsRecord{
AppGUID: appEnvRecord.AppGUID,
EnvironmentVariables: appEnvRecord.EnvironmentVariables,
}
 
return routing.NewResponse(http.StatusOK).WithBody(presenter.ForAppEnvVars(appEnvVarsRecord, h.serverURL)), nil
}
 
func (h *App) updateEnvVars(r *http.Request) (*routing.Response, error) {
authInfo, _ := authorization.InfoFromContext(r.Context())
logger := logr.FromContextOrDiscard(r.Context()).WithName("handlers.app.update-env-vars")
appGUID := routing.URLParam(r, "guid")
 
var payload payloads.AppPatchEnvVars
if err := h.requestValidator.DecodeAndValidateJSONPayload(r, &payload); err != nil {
return nil, apierrors.LogAndReturn(logger, err, "failed to decode payload")
}
 
app, err := h.appRepo.GetApp(r.Context(), authInfo, appGUID)
if err != nil {
return nil, apierrors.LogAndReturn(logger, apierrors.ForbiddenAsNotFound(err), "Failed to fetch app from Kubernetes", "AppGUID", appGUID)
}
 
envVarsRecord, err := h.appRepo.PatchAppEnvVars(r.Context(), authInfo, payload.ToMessage(appGUID, app.SpaceGUID))
if err != nil {
return nil, apierrors.LogAndReturn(logger, err, "Error updating app environment variables")
}
 
return routing.NewResponse(http.StatusOK).WithBody(presenter.ForAppEnvVars(envVarsRecord, h.serverURL)), nil
}
 
func (h *App) getEnvironment(r *http.Request) (*routing.Response, error) {
authInfo, _ := authorization.InfoFromContext(r.Context())
logger := logr.FromContextOrDiscard(r.Context()).WithName("handlers.app.get-environment")
appGUID := routing.URLParam(r, "guid")
 
appEnvRecord, err := h.appRepo.GetAppEnv(r.Context(), authInfo, appGUID)
if err != nil {
return nil, apierrors.LogAndReturn(logger, err, "Failed to fetch app environment variables", "AppGUID", appGUID)
}
 
return routing.NewResponse(http.StatusOK).WithBody(presenter.ForAppEnv(appEnvRecord)), nil
}
 
func (h *App) getProcess(r *http.Request) (*routing.Response, error) {
authInfo, _ := authorization.InfoFromContext(r.Context())
logger := logr.FromContextOrDiscard(r.Context()).WithName("handlers.app.get-process")
appGUID := routing.URLParam(r, "guid")
processType := routing.URLParam(r, "type")
 
app, err := h.appRepo.GetApp(r.Context(), authInfo, appGUID)
if err != nil {
return nil, apierrors.LogAndReturn(logger, apierrors.ForbiddenAsNotFound(err), "Failed to fetch app from Kubernetes", "AppGUID", appGUID)
}
 
process, err := h.getSingleProcess(r.Context(), authInfo, repositories.ListProcessesMessage{
AppGUIDs: []string{appGUID},
ProcessTypes: []string{processType},
SpaceGUID: app.SpaceGUID,
})
if err != nil {
return nil, apierrors.LogAndReturn(logger, err, "Failed to get process", "AppGUID", appGUID)
}
 
return routing.NewResponse(http.StatusOK).WithBody(presenter.ForProcess(process, h.serverURL)), nil
}
 
func (h *App) getSingleProcess(ctx context.Context, authInfo authorization.Info, listMessage repositories.ListProcessesMessage) (repositories.ProcessRecord, error) {
processes, err := h.processRepo.ListProcesses(ctx, authInfo, listMessage)
if err != nil {
return repositories.ProcessRecord{}, err
}
 
return singleton.Get(processes)
}
 
func (h *App) getProcessStats(r *http.Request) (*routing.Response, error) {
appGUID := routing.URLParam(r, "guid")
processType := routing.URLParam(r, "type")
 
authInfo, _ := authorization.InfoFromContext(r.Context())
logger := logr.FromContextOrDiscard(r.Context()).WithName("handlers.app.get-process-stats").WithValues("appGUID", appGUID, "processType", processType)
 
app, err := h.appRepo.GetApp(r.Context(), authInfo, appGUID)
if err != nil {
return nil, apierrors.LogAndReturn(logger, apierrors.ForbiddenAsNotFound(err), "failed to fetch app from Kubernetes")
}
 
process, err := h.getSingleProcess(r.Context(), authInfo, repositories.ListProcessesMessage{
AppGUIDs: []string{appGUID},
ProcessTypes: []string{processType},
SpaceGUID: app.SpaceGUID,
})
if err != nil {
return nil, apierrors.LogAndReturn(logger, err, "failed to get process")
}
 
gauges, err := h.gaugesCollector.CollectProcessGauges(r.Context(), appGUID, process.GUID)
if err != nil {
return nil, apierrors.LogAndReturn(logger, err, "failed to get process gauges from log cache", "processGUID", process.GUID)
}
 
instancesState, err := h.instancesStateCollector.CollectProcessInstancesStates(r.Context(), process.GUID)
if err != nil {
return nil, apierrors.LogAndReturn(logger, apierrors.ForbiddenAsNotFound(err), "failed to get process instances state", "processGUID", process.GUID)
}
 
return routing.NewResponse(http.StatusOK).WithBody(presenter.ForProcessStats(gauges, instancesState, time.Now())), nil
}
 
func (h *App) getPackages(r *http.Request) (*routing.Response, error) {
authInfo, _ := authorization.InfoFromContext(r.Context())
logger := logr.FromContextOrDiscard(r.Context()).WithName("handlers.app.get-packages")
appGUID := routing.URLParam(r, "guid")
 
_, err := h.appRepo.GetApp(r.Context(), authInfo, appGUID)
if err != nil {
return nil, apierrors.LogAndReturn(logger, apierrors.ForbiddenAsNotFound(err), "Failed to fetch app from Kubernetes", "AppGUID", appGUID)
}
 
fetchPackagesForAppMessage := repositories.ListPackagesMessage{
AppGUIDs: []string{appGUID},
States: []string{},
}
 
packageList, err := h.packageRepo.ListPackages(r.Context(), authInfo, fetchPackagesForAppMessage)
if err != nil {
return nil, apierrors.LogAndReturn(logger, err, "Failed to fetch app Package(s) from Kubernetes")
}
 
return routing.NewResponse(http.StatusOK).WithBody(presenter.ForList(presenter.ForPackage, packageList, h.serverURL, *r.URL)), nil
}
 
//nolint:dupl
func (h *App) update(r *http.Request) (*routing.Response, error) {
authInfo, _ := authorization.InfoFromContext(r.Context())
logger := logr.FromContextOrDiscard(r.Context()).WithName("handlers.app.update")
appGUID := routing.URLParam(r, "guid")
 
app, err := h.appRepo.GetApp(r.Context(), authInfo, appGUID)
if err != nil {
return nil, apierrors.LogAndReturn(logger, apierrors.ForbiddenAsNotFound(err), "Failed to fetch app from Kubernetes", "AppGUID", appGUID)
}
 
var payload payloads.AppPatch
if err = h.requestValidator.DecodeAndValidateJSONPayload(r, &payload); err != nil {
return nil, apierrors.LogAndReturn(logger, err, "failed to decode payload")
}
 
app, err = h.appRepo.PatchApp(r.Context(), authInfo, payload.ToMessage(appGUID, app.SpaceGUID))
if err != nil {
return nil, apierrors.LogAndReturn(logger, err, "Failed to patch app", "AppGUID", appGUID)
}
return routing.NewResponse(http.StatusOK).WithBody(presenter.ForApp(app, h.serverURL)), nil
}
 
func (h *App) getSSHEnabled(r *http.Request) (*routing.Response, error) {
return routing.NewResponse(http.StatusOK).WithBody(presenter.AppSSHEnabled{
Enabled: false,
Reason: "Disabled globally",
}), nil
}
 
func (h *App) getAppFeature(r *http.Request) (*routing.Response, error) {
featureName := routing.URLParam(r, "name")
switch featureName {
case "ssh":
return routing.NewResponse(http.StatusOK).WithBody(map[string]any{
"name": "ssh",
"description": "Enable SSHing into the app.",
"enabled": false,
}), nil
case "revisions":
return routing.NewResponse(http.StatusOK).WithBody(map[string]any{
"name": "revisions",
"description": "Enable versioning of an application",
"enabled": false,
}), nil
default:
return nil, apierrors.NewNotFoundError(nil, "Feature")
}
}
 
func (h *App) restartInstance(r *http.Request) (*routing.Response, error) {
authInfo, _ := authorization.InfoFromContext(r.Context())
logger := logr.FromContextOrDiscard(r.Context()).WithName("handlers.app.restart-instance")
appGUID := routing.URLParam(r, "guid")
instanceID := routing.URLParam(r, "instance")
processType := routing.URLParam(r, "processType")
 
app, err := h.appRepo.GetApp(r.Context(), authInfo, appGUID)
if err != nil {
return nil, apierrors.LogAndReturn(logger, apierrors.NewNotFoundError(nil, repositories.AppResourceType), "Failed to fetch app from Kubernetes", "AppGUID", appGUID)
}
appProcesses, err := h.processRepo.ListProcesses(r.Context(), authInfo, repositories.ListProcessesMessage{
AppGUIDs: []string{appGUID},
SpaceGUID: app.SpaceGUID,
})
if err != nil {
return nil, apierrors.LogAndReturn(logger, err, "failed to list processes for app")
}
 
process, hasProcessType := findProcessType(appProcesses, processType)
if !hasProcessType {
return nil, apierrors.LogAndReturn(logger,
apierrors.NewNotFoundError(nil, repositories.ProcessResourceType),
"app does not have required process type",
)
}
instance, err := strconv.Atoi(instanceID)
if err != nil {
return nil, apierrors.LogAndReturn(
logger,
apierrors.AsUnprocessableEntity(err, "Invalid Instance ID. Instance ID is not a valid Integer.", apierrors.NotFoundError{}, apierrors.ForbiddenError{}),
"InstanceID", instanceID,
)
}
if int(process.DesiredInstances) <= instance {
return nil, apierrors.LogAndReturn(logger,
apierrors.NewNotFoundError(nil, fmt.Sprintf("Instance %d of process %s", instance, processType)), "Instance not found", "AppGUID", appGUID, "InstanceID", instanceID, "Process", process)
}
err = h.podRepo.DeletePod(r.Context(), authInfo, app.Revision, process, instanceID)
if err != nil {
return nil, apierrors.LogAndReturn(logger, apierrors.ForbiddenAsNotFound(err), "Failed to restart instance", "AppGUID", appGUID, "InstanceID", instanceID, "Process", process)
}
 
return routing.NewResponse(http.StatusNoContent), nil
}
 
func (h *App) UnauthenticatedRoutes() []routing.Route {
return nil
}
 
func (h *App) AuthenticatedRoutes() []routing.Route {
return []routing.Route{
{Method: "GET", Pattern: AppPath, Handler: h.get},
{Method: "GET", Pattern: AppsPath, Handler: h.list},
{Method: "POST", Pattern: AppsPath, Handler: h.create},
{Method: "PATCH", Pattern: AppCurrentDropletRelationshipPath, Handler: h.setCurrentDroplet},
{Method: "GET", Pattern: AppDropletsPath, Handler: h.listDroplets},
{Method: "GET", Pattern: AppCurrentDropletPath, Handler: h.getCurrentDroplet},
{Method: "POST", Pattern: AppStartPath, Handler: h.start},
{Method: "POST", Pattern: AppStopPath, Handler: h.stop},
{Method: "POST", Pattern: AppRestartPath, Handler: h.restart},
{Method: "POST", Pattern: AppProcessScalePath, Handler: h.scaleProcess},
{Method: "GET", Pattern: AppProcessesPath, Handler: h.getProcesses},
{Method: "GET", Pattern: AppProcessByTypePath, Handler: h.getProcess},
{Method: "GET", Pattern: AppProcessStatsByTypePath, Handler: h.getProcessStats},
{Method: "GET", Pattern: AppRoutesPath, Handler: h.getRoutes},
{Method: "DELETE", Pattern: AppPath, Handler: h.delete},
{Method: "GET", Pattern: AppEnvVarsPath, Handler: h.getEnvVars},
{Method: "PATCH", Pattern: AppEnvVarsPath, Handler: h.updateEnvVars},
{Method: "GET", Pattern: AppEnvPath, Handler: h.getEnvironment},
{Method: "GET", Pattern: AppPackagesPath, Handler: h.getPackages},
{Method: "GET", Pattern: AppFeaturePath, Handler: h.getAppFeature},
{Method: "PATCH", Pattern: AppPath, Handler: h.update},
{Method: "GET", Pattern: AppSSHEnabledPath, Handler: h.getSSHEnabled},
{Method: "DELETE", Pattern: AppInstanceRestartPath, Handler: h.restartInstance},
}
}