with '#' will be ignored, and an empty message aborts the commit. Author: Ethan Alicea <64653625+Tech-Gamer@users.noreply.github.com> On branch develop Your branch is up to date with 'origin/develop'. Changes to be committed: modified: .github/workflows/push.yaml modified: .github/workflows/release.yaml modified: CHANGELOG.md modified: Dockerfile modified: Makefile modified: README.md modified: cmd/configure.go modified: cmd/diagnostics.go modified: cmd/root.go modified: config/config.go modified: environment/allocations.go modified: environment/docker.go modified: environment/docker/api.go modified: environment/docker/container.go modified: environment/docker/environment.go modified: environment/docker/power.go modified: environment/docker/stats.go modified: environment/environment.go modified: environment/settings.go modified: events/events.go modified: go.mod modified: internal/cron/activity_cron.go modified: internal/cron/cron.go modified: internal/cron/sftp_cron.go modified: internal/database/database.go modified: internal/progress/progress.go modified: internal/progress/progress_test.go modified: loggers/cli/cli.go new file: oryxBuildBinary modified: parser/parser.go modified: remote/http.go modified: remote/servers.go modified: remote/types.go modified: router/downloader/downloader.go modified: router/middleware.go modified: router/middleware/middleware.go modified: router/middleware/request_error.go modified: router/router.go modified: router/router_download.go modified: router/router_server.go modified: router/router_server_backup.go modified: router/router_server_files.go modified: router/router_server_transfer.go modified: router/router_server_ws.go modified: router/router_system.go modified: router/router_transfer.go modified: router/tokens/parser.go modified: router/websocket/listeners.go modified: router/websocket/websocket.go modified: server/activity.go modified: server/backup.go modified: server/backup/backup.go modified: server/backup/backup_local.go modified: server/backup/backup_s3.go modified: server/configuration.go modified: server/console.go modified: server/crash.go modified: server/events.go modified: server/filesystem/archive.go modified: server/filesystem/filesystem.go modified: server/filesystem/filesystem_test.go modified: server/install.go modified: server/installer/installer.go modified: server/listeners.go modified: server/manager.go modified: server/mounts.go modified: server/power.go modified: server/power_test.go modified: server/resources.go modified: server/server.go modified: server/transfer/archive.go modified: server/transfer/source.go modified: server/transfer/transfer.go modified: server/update.go modified: sftp/event.go modified: sftp/handler.go modified: sftp/server.go modified: wings.go
221 lines
6.4 KiB
Go
221 lines
6.4 KiB
Go
package docker
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"io"
|
|
"sync"
|
|
|
|
"emperror.dev/errors"
|
|
"github.com/apex/log"
|
|
"github.com/docker/docker/api/types"
|
|
"github.com/docker/docker/client"
|
|
|
|
"github.com/Tech-Gamer/nwy-wings/environment"
|
|
"github.com/Tech-Gamer/nwy-wings/events"
|
|
"github.com/Tech-Gamer/nwy-wings/remote"
|
|
"github.com/Tech-Gamer/nwy-wings/system"
|
|
)
|
|
|
|
type Metadata struct {
|
|
Image string
|
|
Stop remote.ProcessStopConfiguration
|
|
}
|
|
|
|
// Ensure that the Docker environment is always implementing all the methods
|
|
// from the base environment interface.
|
|
var _ environment.ProcessEnvironment = (*Environment)(nil)
|
|
|
|
type Environment struct {
|
|
mu sync.RWMutex
|
|
|
|
// The public identifier for this environment. In this case it is the Docker container
|
|
// name that will be used for all instances created under it.
|
|
Id string
|
|
|
|
// The environment configuration.
|
|
Configuration *environment.Configuration
|
|
|
|
meta *Metadata
|
|
|
|
// The Docker client being used for this instance.
|
|
client *client.Client
|
|
|
|
// Controls the hijacked response stream which exists only when we're attached to
|
|
// the running container instance.
|
|
stream *types.HijackedResponse
|
|
|
|
// Holds the stats stream used by the polling commands so that we can easily close it out.
|
|
stats io.ReadCloser
|
|
|
|
emitter *events.Bus
|
|
|
|
logCallbackMx sync.Mutex
|
|
logCallback func([]byte)
|
|
|
|
// Tracks the environment state.
|
|
st *system.AtomicString
|
|
}
|
|
|
|
// New creates a new base Docker environment. The ID passed through will be the
|
|
// ID that is used to reference the container from here on out. This should be
|
|
// unique per-server (we use the UUID by default). The container does not need
|
|
// to exist at this point.
|
|
func New(id string, m *Metadata, c *environment.Configuration) (*Environment, error) {
|
|
cli, err := environment.Docker()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
e := &Environment{
|
|
Id: id,
|
|
Configuration: c,
|
|
meta: m,
|
|
client: cli,
|
|
st: system.NewAtomicString(environment.ProcessOfflineState),
|
|
emitter: events.NewBus(),
|
|
}
|
|
|
|
return e, nil
|
|
}
|
|
|
|
func (e *Environment) log() *log.Entry {
|
|
return log.WithField("environment", e.Type()).WithField("container_id", e.Id)
|
|
}
|
|
|
|
func (e *Environment) Type() string {
|
|
return "docker"
|
|
}
|
|
|
|
// SetStream sets the current stream value from the Docker client. If a nil
|
|
// value is provided we assume that the stream is no longer operational and the
|
|
// instance is effectively offline.
|
|
func (e *Environment) SetStream(s *types.HijackedResponse) {
|
|
e.mu.Lock()
|
|
e.stream = s
|
|
e.mu.Unlock()
|
|
}
|
|
|
|
// IsAttached determines if this process is currently attached to the
|
|
// container instance by checking if the stream is nil or not.
|
|
func (e *Environment) IsAttached() bool {
|
|
e.mu.RLock()
|
|
defer e.mu.RUnlock()
|
|
return e.stream != nil
|
|
}
|
|
|
|
// Events returns an event bus for the environment.
|
|
func (e *Environment) Events() *events.Bus {
|
|
return e.emitter
|
|
}
|
|
|
|
// Exists determines if the container exists in this environment. The ID passed
|
|
// through should be the server UUID since containers are created utilizing the
|
|
// server UUID as the name and docker will work fine when using the container
|
|
// name as the lookup parameter in addition to the longer ID auto-assigned when
|
|
// the container is created.
|
|
func (e *Environment) Exists() (bool, error) {
|
|
_, err := e.ContainerInspect(context.Background())
|
|
if err != nil {
|
|
// If this error is because the container instance wasn't found via Docker we
|
|
// can safely ignore the error and just return false.
|
|
if client.IsErrNotFound(err) {
|
|
return false, nil
|
|
}
|
|
return false, err
|
|
}
|
|
return true, nil
|
|
}
|
|
|
|
// IsRunning determines if the server's docker container is currently running.
|
|
// If there is no container present, an error will be raised (since this
|
|
// shouldn't be a case that ever happens under correctly developed
|
|
// circumstances).
|
|
//
|
|
// You can confirm if the instance wasn't found by using client.IsErrNotFound
|
|
// from the Docker API.
|
|
//
|
|
// @see docker/client/errors.go
|
|
func (e *Environment) IsRunning(ctx context.Context) (bool, error) {
|
|
c, err := e.ContainerInspect(ctx)
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
return c.State.Running, nil
|
|
}
|
|
|
|
// ExitState returns the container exit state, the exit code and whether or not
|
|
// the container was killed by the OOM killer.
|
|
func (e *Environment) ExitState() (uint32, bool, error) {
|
|
c, err := e.ContainerInspect(context.Background())
|
|
if err != nil {
|
|
// I'm not entirely sure how this can happen to be honest. I tried deleting a
|
|
// container _while_ a server was running and wings gracefully saw the crash and
|
|
// created a new container for it.
|
|
//
|
|
// However, someone reported an error in Discord about this scenario happening,
|
|
// so I guess this should prevent it? They didn't tell me how they caused it though
|
|
// so that's a mystery that will have to go unsolved.
|
|
//
|
|
// @see https://github.com/pterodactyl/panel/issues/2003
|
|
if client.IsErrNotFound(err) {
|
|
return 1, false, nil
|
|
}
|
|
return 0, false, errors.WrapIf(err, "environment/docker: failed to inspect container")
|
|
}
|
|
return uint32(c.State.ExitCode), c.State.OOMKilled, nil
|
|
}
|
|
|
|
// Config returns the environment configuration allowing a process to make
|
|
// modifications of the environment on the fly.
|
|
func (e *Environment) Config() *environment.Configuration {
|
|
e.mu.RLock()
|
|
defer e.mu.RUnlock()
|
|
|
|
return e.Configuration
|
|
}
|
|
|
|
// SetStopConfiguration sets the stop configuration for the environment.
|
|
func (e *Environment) SetStopConfiguration(c remote.ProcessStopConfiguration) {
|
|
e.mu.Lock()
|
|
e.meta.Stop = c
|
|
e.mu.Unlock()
|
|
}
|
|
|
|
func (e *Environment) SetImage(i string) {
|
|
e.mu.Lock()
|
|
defer e.mu.Unlock()
|
|
|
|
e.meta.Image = i
|
|
}
|
|
|
|
func (e *Environment) State() string {
|
|
return e.st.Load()
|
|
}
|
|
|
|
// SetState sets the state of the environment. This emits an event that server's
|
|
// can hook into to take their own actions and track their own state based on
|
|
// the environment.
|
|
func (e *Environment) SetState(state string) {
|
|
if state != environment.ProcessOfflineState &&
|
|
state != environment.ProcessStartingState &&
|
|
state != environment.ProcessRunningState &&
|
|
state != environment.ProcessStoppingState {
|
|
panic(errors.New(fmt.Sprintf("invalid server state received: %s", state)))
|
|
}
|
|
|
|
// Emit the event to any listeners that are currently registered.
|
|
if e.State() != state {
|
|
// If the state changed make sure we update the internal tracking to note that.
|
|
e.st.Store(state)
|
|
e.Events().Publish(environment.StateChangeEvent, state)
|
|
}
|
|
}
|
|
|
|
func (e *Environment) SetLogCallback(f func([]byte)) {
|
|
e.logCallbackMx.Lock()
|
|
defer e.logCallbackMx.Unlock()
|
|
|
|
e.logCallback = f
|
|
}
|