mirror of
https://github.com/moby/moby.git
synced 2025-10-14 01:48:05 +03:00
The Container.State struct holds the container's state, and most of its fields are expected to change dynamically. Some o these state-changes are explicit, for example, setting the container to be "stopped". Other state changes can be more explicit, for example due to the containers' process exiting or being "OOM" killed by the kernel. The distinction between explicit ("desired") state changes and "state" ("actual state") is sometimes vague; for some properties, we clearly separated them, for example if a user requested the container to be stopped or restarted, we store state in the Container object itself; HasBeenManuallyStopped bool // used for unless-stopped restart policy HasBeenManuallyRestarted bool `json:"-"` // used to distinguish restart caused by restart policy from the manual one Other properties are more ambiguous. such as "HasBeenStartedBefore" and "RestartCount", which are stored on the Container (and persisted to disk), but may be more related to "actual" state, and likely should not be persisted; RestartCount int HasBeenStartedBefore bool Given that (per the above) concurrency must be taken into account, most changes to the `container.State` struct should be protected; here's where things get blurry. While the `State` type provides various accessor methods, only some of them take concurrency into account; for example, [State.IsRunning] and [State.GetPID] acquire a lock, whereas [State.ExitCodeValue] does not. Even the (commonly used) [State.StateString] has no locking at all. The way to handle this is error-prone; [container.State] contains a mutex, and it's exported. Given that its embedded in the [container.Container] struct, it's also exposed as an exported mutex for the container. The assumption here is that by "merging" the two, the caller to acquire a lock when either the container _or_ its state must be mutated. However, because some methods on `container.State` handle their own locking, consumers must be deeply familiar with the internals; if both changes to the `Container` AND `Container.State` must be made. This gets amplified more as some (exported!) methods, such as [container.SetRunning] mutate multiple fields, but don't acquire a lock (so expect the caller to hold one), but their (also exported) counterpart (e.g. [State.IsRunning]) do. It should be clear from the above, that this needs some architectural changes; a clearer separation between "desired" and "actual" state (opening the potential to update the container's config without manually touching its `State`), possibly a method to obtain a read-only copy of the current state (for those querying state), and reviewing which fields belong where (and should be persisted to disk, or only remain in memory). This PR preserves the status quo; it makes no structural changes, other than exposing where we access the container's state. Where previously the State fields and methods were referred to as "part of the container" (e.g. `ctr.IsRunning()` or `ctr.Running`), we now explicitly reference the embedded `State` (`ctr.State.IsRunning`, `ctr.State.Running`). The exception (for now) is the mutex, which is still referenced through the embedded struct (`ctr.Lock()` instead of `ctr.State.Lock()`), as this is (mostly) by design to protect the container, and what's in it (including its `State`). [State.IsRunning]:c4afa77157/daemon/container/state.go (L205-L209)
[State.GetPID]:c4afa77157/daemon/container/state.go (L211-L216)
[State.ExitCodeValue]:c4afa77157/daemon/container/state.go (L218-L228)
[State.StateString]:c4afa77157/daemon/container/state.go (L102-L131)
[container.State]:c4afa77157/daemon/container/state.go (L15-L23)
[container.Container]:c4afa77157/daemon/container/container.go (L67-L75)
[container.SetRunning]:c4afa77157/daemon/container/state.go (L230-L277)
Signed-off-by: Sebastiaan van Stijn <github@gone.nl>
32 lines
714 B
Go
32 lines
714 B
Go
package daemon
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"time"
|
|
|
|
"github.com/moby/go-archive"
|
|
"github.com/moby/moby/v2/daemon/internal/metrics"
|
|
)
|
|
|
|
// ContainerChanges returns a list of container fs changes
|
|
func (daemon *Daemon) ContainerChanges(ctx context.Context, name string) ([]archive.Change, error) {
|
|
start := time.Now()
|
|
|
|
container, err := daemon.GetContainer(name)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if isWindows && container.State.IsRunning() {
|
|
return nil, errors.New("Windows does not support diff of a running container")
|
|
}
|
|
|
|
c, err := daemon.imageService.Changes(ctx, container)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
metrics.ContainerActions.WithValues("changes").UpdateSince(start)
|
|
return c, nil
|
|
}
|