mirror of
https://github.com/containers/podman.git
synced 2025-08-06 19:44:14 +08:00
Add service endpoint
add service endpoint for the new API. Also supports the varlink implementation. Signed-off-by: baude <bbaude@redhat.com> Refactor to allow developer more control of API server * Add api.NewServerWithSettings() to create an API server with custom settings * Add api.ListenUnix() to create a UDS net.Listener and setup UDS Signed-off-by: Jhon Honce <jhonce@redhat.com> Signed-off-by: baude <bbaude@redhat.com> More service completion Add podman service command that allows users to run either a RESTful or varlink protocol API service. Addition of docs and RESTful listening. Signed-off-by: baude <bbaude@redhat.com> Signed-off-by: Brent Baude <bbaude@redhat.com>
This commit is contained in:
47
.cirrus.yml
47
.cirrus.yml
@ -640,29 +640,29 @@ verify_test_built_images_task:
|
|||||||
always:
|
always:
|
||||||
<<: *standardlogs
|
<<: *standardlogs
|
||||||
|
|
||||||
upload_snap_task:
|
#upload_snap_task:
|
||||||
only_if: >-
|
# only_if: >-
|
||||||
$CIRRUS_BRANCH != $DEST_BRANCH &&
|
# $CIRRUS_BRANCH != $DEST_BRANCH &&
|
||||||
$CIRRUS_CHANGE_MESSAGE !=~ '.*CI:IMG.*' &&
|
# $CIRRUS_CHANGE_MESSAGE !=~ '.*CI:IMG.*' &&
|
||||||
$CIRRUS_CHANGE_MESSAGE !=~ '.*CI:DOCS.*'
|
# $CIRRUS_CHANGE_MESSAGE !=~ '.*CI:DOCS.*'
|
||||||
|
#
|
||||||
# Only when PR or branch is merged into master
|
# # Only when PR or branch is merged into master
|
||||||
|
#
|
||||||
depends_on:
|
# depends_on:
|
||||||
- "test_building_snap"
|
# - "test_building_snap"
|
||||||
|
#
|
||||||
container:
|
# container:
|
||||||
image: yakshaveinc/snapcraft:core18
|
# image: yakshaveinc/snapcraft:core18
|
||||||
|
#
|
||||||
env:
|
# env:
|
||||||
SNAPCRAFT_LOGIN: ENCRYPTED[d8e82eb31c6372fec07f405f413d57806026b1a9f8400033531ebcd54d6750a5e4a8b1f68e3ec65c98c65e0d9b2a6a75]
|
# SNAPCRAFT_LOGIN: ENCRYPTED[d8e82eb31c6372fec07f405f413d57806026b1a9f8400033531ebcd54d6750a5e4a8b1f68e3ec65c98c65e0d9b2a6a75]
|
||||||
snapcraft_login_file:
|
# snapcraft_login_file:
|
||||||
path: /root/.snapcraft/login.cfg
|
# path: /root/.snapcraft/login.cfg
|
||||||
variable_name: SNAPCRAFT_LOGIN
|
# variable_name: SNAPCRAFT_LOGIN
|
||||||
snapcraft_script:
|
# snapcraft_script:
|
||||||
- 'apt-get -y update'
|
# - 'apt-get -y update'
|
||||||
- 'snapcraft login --with "/root/.snapcraft/login.cfg"'
|
# - 'snapcraft login --with "/root/.snapcraft/login.cfg"'
|
||||||
- 'cd contrib/snapcraft && snapcraft && snapcraft push *.snap --release edge'
|
# - 'cd contrib/snapcraft && snapcraft && snapcraft push *.snap --release edge'
|
||||||
|
|
||||||
|
|
||||||
docs_task:
|
docs_task:
|
||||||
@ -705,7 +705,6 @@ success_task:
|
|||||||
- "special_testing_endpoint"
|
- "special_testing_endpoint"
|
||||||
- "test_build_cache_images"
|
- "test_build_cache_images"
|
||||||
- "test_building_snap"
|
- "test_building_snap"
|
||||||
- "upload_snap"
|
|
||||||
- "verify_test_built_images"
|
- "verify_test_built_images"
|
||||||
- "docs"
|
- "docs"
|
||||||
|
|
||||||
|
10
Makefile
10
Makefile
@ -199,14 +199,6 @@ bin/podman.cross.%: .gopathok
|
|||||||
GOARCH="$${TARGET##*.}" \
|
GOARCH="$${TARGET##*.}" \
|
||||||
$(GO_BUILD) -gcflags '$(GCFLAGS)' -asmflags '$(ASMFLAGS)' -ldflags '$(LDFLAGS_PODMAN)' -tags '$(BUILDTAGS_CROSS)' -o "$@" $(PROJECT)/cmd/podman
|
$(GO_BUILD) -gcflags '$(GCFLAGS)' -asmflags '$(ASMFLAGS)' -ldflags '$(LDFLAGS_PODMAN)' -tags '$(BUILDTAGS_CROSS)' -o "$@" $(PROJECT)/cmd/podman
|
||||||
|
|
||||||
.PHONY: service
|
|
||||||
service: .gopathok
|
|
||||||
$(GO_BUILD) $(BUILDFLAGS) -gcflags '$(GCFLAGS)' -asmflags '$(ASMFLAGS)' -ldflags '$(LDFLAGS_PODMAN)' -tags "$(BUILDTAGS)" -o bin/$@ $(PROJECT)/cmd/service
|
|
||||||
|
|
||||||
.PHONY:
|
|
||||||
run-service:
|
|
||||||
systemd-socket-activate -l 8080 ./bin/service
|
|
||||||
|
|
||||||
.PHONY: run-docker-py-tests
|
.PHONY: run-docker-py-tests
|
||||||
run-docker-py-tests:
|
run-docker-py-tests:
|
||||||
$(eval testLogs=$(shell mktemp))
|
$(eval testLogs=$(shell mktemp))
|
||||||
@ -328,7 +320,7 @@ system.test-binary: .install.ginkgo
|
|||||||
vagrant-check:
|
vagrant-check:
|
||||||
BOX=$(BOX) sh ./vagrant.sh
|
BOX=$(BOX) sh ./vagrant.sh
|
||||||
|
|
||||||
binaries: varlink_generate podman podman-remote service ## Build podman
|
binaries: varlink_generate podman podman-remote ## Build podman
|
||||||
|
|
||||||
install.catatonit:
|
install.catatonit:
|
||||||
./hack/install_catatonit.sh
|
./hack/install_catatonit.sh
|
||||||
|
@ -599,6 +599,12 @@ type VarlinkValues struct {
|
|||||||
Timeout int64
|
Timeout int64
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type ServiceValues struct {
|
||||||
|
PodmanCommand
|
||||||
|
Varlink bool
|
||||||
|
Timeout int64
|
||||||
|
}
|
||||||
|
|
||||||
type SetTrustValues struct {
|
type SetTrustValues struct {
|
||||||
PodmanCommand
|
PodmanCommand
|
||||||
PolicyPath string
|
PolicyPath string
|
||||||
|
@ -26,6 +26,9 @@ func getMainCommands() []*cobra.Command {
|
|||||||
if len(_varlinkCommand.Use) > 0 {
|
if len(_varlinkCommand.Use) > 0 {
|
||||||
rootCommands = append(rootCommands, _varlinkCommand)
|
rootCommands = append(rootCommands, _varlinkCommand)
|
||||||
}
|
}
|
||||||
|
if len(_serviceCommand.Use) > 0 {
|
||||||
|
rootCommands = append(rootCommands, _serviceCommand)
|
||||||
|
}
|
||||||
return rootCommands
|
return rootCommands
|
||||||
}
|
}
|
||||||
|
|
||||||
|
154
cmd/podman/service.go
Normal file
154
cmd/podman/service.go
Normal file
@ -0,0 +1,154 @@
|
|||||||
|
// +build varlink,!remoteclient
|
||||||
|
|
||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"net"
|
||||||
|
"os"
|
||||||
|
"path/filepath"
|
||||||
|
"strings"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/containers/libpod/cmd/podman/cliconfig"
|
||||||
|
"github.com/containers/libpod/cmd/podman/libpodruntime"
|
||||||
|
iopodman "github.com/containers/libpod/cmd/podman/varlink"
|
||||||
|
"github.com/containers/libpod/libpod"
|
||||||
|
"github.com/containers/libpod/pkg/adapter"
|
||||||
|
api "github.com/containers/libpod/pkg/api/server"
|
||||||
|
"github.com/containers/libpod/pkg/rootless"
|
||||||
|
"github.com/containers/libpod/pkg/util"
|
||||||
|
"github.com/containers/libpod/pkg/varlinkapi"
|
||||||
|
"github.com/containers/libpod/version"
|
||||||
|
"github.com/pkg/errors"
|
||||||
|
"github.com/sirupsen/logrus"
|
||||||
|
"github.com/spf13/cobra"
|
||||||
|
"github.com/varlink/go/varlink"
|
||||||
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
serviceCommand cliconfig.ServiceValues
|
||||||
|
serviceDescription = `Run an API service
|
||||||
|
|
||||||
|
Enable a listening service for API access to Podman commands.
|
||||||
|
`
|
||||||
|
|
||||||
|
_serviceCommand = &cobra.Command{
|
||||||
|
Use: "service [flags] [URI]",
|
||||||
|
Short: "Run API service",
|
||||||
|
Long: serviceDescription,
|
||||||
|
RunE: func(cmd *cobra.Command, args []string) error {
|
||||||
|
serviceCommand.InputArgs = args
|
||||||
|
serviceCommand.GlobalFlags = MainGlobalOpts
|
||||||
|
return serviceCmd(&serviceCommand)
|
||||||
|
},
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
serviceCommand.Command = _serviceCommand
|
||||||
|
serviceCommand.SetHelpTemplate(HelpTemplate())
|
||||||
|
serviceCommand.SetUsageTemplate(UsageTemplate())
|
||||||
|
flags := serviceCommand.Flags()
|
||||||
|
flags.Int64VarP(&serviceCommand.Timeout, "timeout", "t", 1000, "Time until the service session expires in milliseconds. Use 0 to disable the timeout")
|
||||||
|
flags.BoolVar(&serviceCommand.Varlink, "varlink", false, "Use legacy varlink service instead of REST")
|
||||||
|
}
|
||||||
|
|
||||||
|
func serviceCmd(c *cliconfig.ServiceValues) error {
|
||||||
|
// For V2, default to the REST socket
|
||||||
|
apiURI := adapter.DefaultAPIAddress
|
||||||
|
if c.Varlink {
|
||||||
|
apiURI = adapter.DefaultVarlinkAddress
|
||||||
|
}
|
||||||
|
|
||||||
|
if rootless.IsRootless() {
|
||||||
|
xdg, err := util.GetRuntimeDir()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
socketName := "podman.sock"
|
||||||
|
if c.Varlink {
|
||||||
|
socketName = "io.podman"
|
||||||
|
}
|
||||||
|
socketDir := filepath.Join(xdg, "podman", socketName)
|
||||||
|
if _, err := os.Stat(filepath.Dir(socketDir)); err != nil {
|
||||||
|
if os.IsNotExist(err) {
|
||||||
|
if err := os.Mkdir(filepath.Dir(socketDir), 0755); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
apiURI = fmt.Sprintf("unix:%s", socketDir)
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(c.InputArgs) > 0 {
|
||||||
|
apiURI = c.InputArgs[0]
|
||||||
|
}
|
||||||
|
|
||||||
|
logrus.Infof("using API endpoint: %s", apiURI)
|
||||||
|
|
||||||
|
// Create a single runtime api consumption
|
||||||
|
runtime, err := libpodruntime.GetRuntimeDisableFDs(getContext(), &c.PodmanCommand)
|
||||||
|
if err != nil {
|
||||||
|
return errors.Wrapf(err, "error creating libpod runtime")
|
||||||
|
}
|
||||||
|
defer runtime.DeferredShutdown(false)
|
||||||
|
|
||||||
|
timeout := time.Duration(c.Timeout) * time.Millisecond
|
||||||
|
if c.Varlink {
|
||||||
|
return runVarlink(runtime, apiURI, timeout, c)
|
||||||
|
}
|
||||||
|
return runREST(runtime, apiURI, timeout)
|
||||||
|
}
|
||||||
|
|
||||||
|
func runREST(r *libpod.Runtime, uri string, timeout time.Duration) error {
|
||||||
|
logrus.Warn("This function is EXPERIMENTAL")
|
||||||
|
fmt.Println("This function is EXPERIMENTAL.")
|
||||||
|
fields := strings.Split(uri, ":")
|
||||||
|
if len(fields) == 1 {
|
||||||
|
return errors.Errorf("%s is an invalid socket destination", uri)
|
||||||
|
}
|
||||||
|
address := strings.Join(fields[1:], ":")
|
||||||
|
l, err := net.Listen(fields[0], address)
|
||||||
|
if err != nil {
|
||||||
|
return errors.Wrapf(err, "unable to create socket %s", uri)
|
||||||
|
}
|
||||||
|
server, err := api.NewServerWithSettings(r, timeout, &l)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return server.Serve()
|
||||||
|
}
|
||||||
|
|
||||||
|
func runVarlink(r *libpod.Runtime, uri string, timeout time.Duration, c *cliconfig.ServiceValues) error {
|
||||||
|
var varlinkInterfaces = []*iopodman.VarlinkInterface{varlinkapi.New(&c.PodmanCommand, r)}
|
||||||
|
service, err := varlink.NewService(
|
||||||
|
"Atomic",
|
||||||
|
"podman",
|
||||||
|
version.Version,
|
||||||
|
"https://github.com/containers/libpod",
|
||||||
|
)
|
||||||
|
if err != nil {
|
||||||
|
return errors.Wrapf(err, "unable to create new varlink service")
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, i := range varlinkInterfaces {
|
||||||
|
if err := service.RegisterInterface(i); err != nil {
|
||||||
|
return errors.Errorf("unable to register varlink interface %v", i)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Run the varlink server at the given address
|
||||||
|
if err = service.Listen(uri, timeout); err != nil {
|
||||||
|
switch err.(type) {
|
||||||
|
case varlink.ServiceTimeoutError:
|
||||||
|
logrus.Infof("varlink service expired (use --timeout to increase session time beyond %d ms, 0 means never timeout)", timeout.String())
|
||||||
|
return nil
|
||||||
|
default:
|
||||||
|
return errors.Wrapf(err, "unable to start varlink service")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
11
cmd/podman/service_dummy.go
Normal file
11
cmd/podman/service_dummy.go
Normal file
@ -0,0 +1,11 @@
|
|||||||
|
// +build !varlink
|
||||||
|
|
||||||
|
package main
|
||||||
|
|
||||||
|
import "github.com/spf13/cobra"
|
||||||
|
|
||||||
|
var (
|
||||||
|
_serviceCommand = &cobra.Command{
|
||||||
|
Use: "",
|
||||||
|
}
|
||||||
|
)
|
@ -51,7 +51,7 @@ func init() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func varlinkCmd(c *cliconfig.VarlinkValues) error {
|
func varlinkCmd(c *cliconfig.VarlinkValues) error {
|
||||||
varlinkURI := adapter.DefaultAddress
|
varlinkURI := adapter.DefaultVarlinkAddress
|
||||||
if rootless.IsRootless() {
|
if rootless.IsRootless() {
|
||||||
xdg, err := util.GetRuntimeDir()
|
xdg, err := util.GetRuntimeDir()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -1,55 +0,0 @@
|
|||||||
package main
|
|
||||||
|
|
||||||
import (
|
|
||||||
"context"
|
|
||||||
"fmt"
|
|
||||||
"os"
|
|
||||||
|
|
||||||
"github.com/containers/libpod/cmd/podman/cliconfig"
|
|
||||||
"github.com/containers/libpod/cmd/podman/libpodruntime"
|
|
||||||
api "github.com/containers/libpod/pkg/api/server"
|
|
||||||
"github.com/containers/storage/pkg/reexec"
|
|
||||||
log "github.com/sirupsen/logrus"
|
|
||||||
"github.com/spf13/cobra"
|
|
||||||
)
|
|
||||||
|
|
||||||
func initConfig() {
|
|
||||||
// we can do more stuff in here.
|
|
||||||
}
|
|
||||||
|
|
||||||
func main() {
|
|
||||||
if reexec.Init() {
|
|
||||||
// We were invoked with a different argv[0] indicating that we
|
|
||||||
// had a specific job to do as a subprocess, and it's done.
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
cobra.OnInitialize(initConfig)
|
|
||||||
log.SetLevel(log.DebugLevel)
|
|
||||||
|
|
||||||
config := cliconfig.PodmanCommand{
|
|
||||||
Command: &cobra.Command{},
|
|
||||||
InputArgs: []string{},
|
|
||||||
GlobalFlags: cliconfig.MainFlags{},
|
|
||||||
Remote: false,
|
|
||||||
}
|
|
||||||
// Create a single runtime for http
|
|
||||||
runtime, err := libpodruntime.GetRuntimeDisableFDs(context.Background(), &config)
|
|
||||||
if err != nil {
|
|
||||||
fmt.Printf("error creating libpod runtime: %s", err.Error())
|
|
||||||
os.Exit(1)
|
|
||||||
}
|
|
||||||
defer runtime.DeferredShutdown(false)
|
|
||||||
|
|
||||||
server, err := api.NewServer(runtime)
|
|
||||||
if err != nil {
|
|
||||||
fmt.Println(err.Error())
|
|
||||||
os.Exit(1)
|
|
||||||
}
|
|
||||||
|
|
||||||
err = server.Serve()
|
|
||||||
if err != nil {
|
|
||||||
fmt.Println(err.Error())
|
|
||||||
os.Exit(1)
|
|
||||||
}
|
|
||||||
}
|
|
@ -73,6 +73,7 @@
|
|||||||
| [podman-rmi(1)](/docs/source/markdown/podman-rmi.1.md) | Removes one or more images |
|
| [podman-rmi(1)](/docs/source/markdown/podman-rmi.1.md) | Removes one or more images |
|
||||||
| [podman-run(1)](/docs/source/markdown/podman-run.1.md) | Run a command in a container |
|
| [podman-run(1)](/docs/source/markdown/podman-run.1.md) | Run a command in a container |
|
||||||
| [podman-save(1)](/docs/source/markdown/podman-save.1.md) | Saves an image to an archive |
|
| [podman-save(1)](/docs/source/markdown/podman-save.1.md) | Saves an image to an archive |
|
||||||
|
| [podman-service(1)](/docs/source/markdown/podman-service.1.md) | Run an API listening service |
|
||||||
| [podman-search(1)](/docs/source/markdown/podman-search.1.md) | Search a registry for an image |
|
| [podman-search(1)](/docs/source/markdown/podman-search.1.md) | Search a registry for an image |
|
||||||
| [podman-start(1)](/docs/source/markdown/podman-start.1.md) | Starts one or more containers |
|
| [podman-start(1)](/docs/source/markdown/podman-start.1.md) | Starts one or more containers |
|
||||||
| [podman-stats(1)](/docs/source/markdown/podman-stats.1.md) | Display a live stream of one or more containers' resource usage statistics |
|
| [podman-stats(1)](/docs/source/markdown/podman-stats.1.md) | Display a live stream of one or more containers' resource usage statistics |
|
||||||
|
@ -1755,6 +1755,19 @@ _podman_search() {
|
|||||||
_complete_ "$options_with_args" "$boolean_options"
|
_complete_ "$options_with_args" "$boolean_options"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
_podman_service() {
|
||||||
|
local options_with_args="
|
||||||
|
-t
|
||||||
|
--timeout
|
||||||
|
"
|
||||||
|
local boolean_options="
|
||||||
|
--help
|
||||||
|
-h
|
||||||
|
--varlink
|
||||||
|
"
|
||||||
|
_complete_ "$options_with_args" "$boolean_options"
|
||||||
|
}
|
||||||
|
|
||||||
_podman_unmount() {
|
_podman_unmount() {
|
||||||
_podman_umount $@
|
_podman_umount $@
|
||||||
}
|
}
|
||||||
|
47
docs/source/markdown/podman-service.1.md
Normal file
47
docs/source/markdown/podman-service.1.md
Normal file
@ -0,0 +1,47 @@
|
|||||||
|
% podman-service(1)
|
||||||
|
|
||||||
|
## NAME
|
||||||
|
podman\-service - Run an API service
|
||||||
|
|
||||||
|
## SYNOPSIS
|
||||||
|
**podman service** [*options*]
|
||||||
|
|
||||||
|
## DESCRIPTION
|
||||||
|
The **podman service** command creates a listening service that will answer API calls for Podman. You may
|
||||||
|
optionally provide an endpoint for the API in URI form. For example, *unix://tmp/foobar.sock* or *tcp:localhost:8080*.
|
||||||
|
If no endpoint is provided, defaults will be used. The default endpoint for a rootfull
|
||||||
|
service is *unix:/run/podman/podman.sock* and rootless is *unix:/$XDG_RUNTIME_DIR/podman/podman.sock* (for
|
||||||
|
example *unix:/run/user/1000/podman/podman.sock*)
|
||||||
|
|
||||||
|
## OPTIONS
|
||||||
|
|
||||||
|
**--timeout**, **-t**
|
||||||
|
|
||||||
|
The time until the session expires in _milliseconds_. The default is 1
|
||||||
|
second. A value of `0` means no timeout and the session will not expire.
|
||||||
|
|
||||||
|
**--varlink**
|
||||||
|
|
||||||
|
Use the varlink protocol instead of the REST-based protocol. This option will be deprecated in the future.
|
||||||
|
|
||||||
|
**--help**, **-h**
|
||||||
|
|
||||||
|
Print usage statement.
|
||||||
|
|
||||||
|
## EXAMPLES
|
||||||
|
|
||||||
|
Run an API listening for 5 seconds using the default socket.
|
||||||
|
```
|
||||||
|
podman service --timeout 5000
|
||||||
|
```
|
||||||
|
|
||||||
|
Run the podman varlink service with an alternate URI and accept the default timeout.
|
||||||
|
```
|
||||||
|
$ podman service --varlink unix:/tmp/io.podman
|
||||||
|
```
|
||||||
|
|
||||||
|
## SEE ALSO
|
||||||
|
podman(1), podman-varlink(1)
|
||||||
|
|
||||||
|
## HISTORY
|
||||||
|
January 2020, Originally compiled by Brent Baude<bbaude@redhat.com>
|
@ -191,6 +191,7 @@ the exit codes follow the `chroot` standard, see below:
|
|||||||
| [podman-rmi(1)](podman-rmi.1.md) | Removes one or more locally stored images. |
|
| [podman-rmi(1)](podman-rmi.1.md) | Removes one or more locally stored images. |
|
||||||
| [podman-run(1)](podman-run.1.md) | Run a command in a new container. |
|
| [podman-run(1)](podman-run.1.md) | Run a command in a new container. |
|
||||||
| [podman-save(1)](podman-save.1.md) | Save an image to a container archive. |
|
| [podman-save(1)](podman-save.1.md) | Save an image to a container archive. |
|
||||||
|
| [podman-service(1)](podman-service.1.md) | Run an API service |
|
||||||
| [podman-search(1)](podman-search.1.md) | Search a registry for an image. |
|
| [podman-search(1)](podman-search.1.md) | Search a registry for an image. |
|
||||||
| [podman-start(1)](podman-start.1.md) | Start one or more containers. |
|
| [podman-start(1)](podman-start.1.md) | Start one or more containers. |
|
||||||
| [podman-stats(1)](podman-stats.1.md) | Display a live stream of one or more container's resource usage statistics. |
|
| [podman-stats(1)](podman-stats.1.md) | Display a live stream of one or more container's resource usage statistics. |
|
||||||
|
@ -57,7 +57,7 @@ func (r RemoteRuntime) RemoteEndpoint() (remoteEndpoint *Endpoint, err error) {
|
|||||||
// last resort is to make a socket connection with the default varlink address for root user
|
// last resort is to make a socket connection with the default varlink address for root user
|
||||||
} else {
|
} else {
|
||||||
logrus.Debug("creating a varlink address based default root address")
|
logrus.Debug("creating a varlink address based default root address")
|
||||||
remoteEndpoint, err = newSocketConnection(DefaultAddress)
|
remoteEndpoint, err = newSocketConnection(DefaultVarlinkAddress)
|
||||||
}
|
}
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
@ -1,7 +1,10 @@
|
|||||||
package adapter
|
package adapter
|
||||||
|
|
||||||
// DefaultAddress is the default address of the varlink socket
|
// DefaultAPIAddress is the default address of the REST socket
|
||||||
const DefaultAddress = "unix:/run/podman/io.podman"
|
const DefaultAPIAddress = "unix:/run/podman/podman.sock"
|
||||||
|
|
||||||
|
// DefaultVarlinkAddress is the default address of the varlink socket
|
||||||
|
const DefaultVarlinkAddress = "unix:/run/podman/io.podman"
|
||||||
|
|
||||||
// EndpointType declares the type of server connection
|
// EndpointType declares the type of server connection
|
||||||
type EndpointType int
|
type EndpointType int
|
||||||
|
31
pkg/api/server/listener_api.go
Normal file
31
pkg/api/server/listener_api.go
Normal file
@ -0,0 +1,31 @@
|
|||||||
|
package server
|
||||||
|
|
||||||
|
import (
|
||||||
|
"net"
|
||||||
|
"os"
|
||||||
|
"path/filepath"
|
||||||
|
|
||||||
|
"github.com/pkg/errors"
|
||||||
|
)
|
||||||
|
|
||||||
|
// ListenUnix follows stdlib net.Listen() API, providing a unix listener for given path
|
||||||
|
// ListenUnix will delete and create files/directories as needed
|
||||||
|
func ListenUnix(network string, path string) (net.Listener, error) {
|
||||||
|
// setup custom listener for API server
|
||||||
|
err := os.MkdirAll(filepath.Dir(path), 0770)
|
||||||
|
if err != nil {
|
||||||
|
return nil, errors.Wrapf(err, "api.ListenUnix() failed to create %s", filepath.Dir(path))
|
||||||
|
}
|
||||||
|
os.Remove(path)
|
||||||
|
|
||||||
|
listener, err := net.Listen(network, path)
|
||||||
|
if err != nil {
|
||||||
|
return nil, errors.Wrapf(err, "api.ListenUnix() failed to create net.Listen(%s, %s)", network, path)
|
||||||
|
}
|
||||||
|
|
||||||
|
_, err = os.Stat(path)
|
||||||
|
if err != nil {
|
||||||
|
return nil, errors.Wrapf(err, "net.Listen(%s, %s) failed to report the failure to create socket", network, path)
|
||||||
|
}
|
||||||
|
return listener, nil
|
||||||
|
}
|
@ -54,9 +54,9 @@ import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
type APIServer struct {
|
type APIServer struct {
|
||||||
http.Server // Where the HTTP work happens
|
http.Server // The HTTP work happens here
|
||||||
*schema.Decoder // Decoder for Query parameters to structs
|
*schema.Decoder // Decoder for Query parameters to structs
|
||||||
context.Context // Context for graceful server shutdown
|
context.Context // Context to carry objects to handlers
|
||||||
*libpod.Runtime // Where the real work happens
|
*libpod.Runtime // Where the real work happens
|
||||||
net.Listener // mux for routing HTTP API calls to libpod routines
|
net.Listener // mux for routing HTTP API calls to libpod routines
|
||||||
context.CancelFunc // Stop APIServer
|
context.CancelFunc // Stop APIServer
|
||||||
@ -64,14 +64,37 @@ type APIServer struct {
|
|||||||
time.Duration // Duration of client access sliding window
|
time.Duration // Duration of client access sliding window
|
||||||
}
|
}
|
||||||
|
|
||||||
// NewServer will create and configure a new API HTTP server
|
// Number of seconds to wait for next request, if exceeded shutdown server
|
||||||
|
const (
|
||||||
|
DefaultServiceDuration = 300 * time.Second
|
||||||
|
UnlimitedServiceDuration = 0 * time.Second
|
||||||
|
)
|
||||||
|
|
||||||
|
// NewServer will create and configure a new API server with all defaults
|
||||||
func NewServer(runtime *libpod.Runtime) (*APIServer, error) {
|
func NewServer(runtime *libpod.Runtime) (*APIServer, error) {
|
||||||
listeners, err := activation.Listeners()
|
return newServer(runtime, DefaultServiceDuration, nil)
|
||||||
if err != nil {
|
}
|
||||||
return nil, errors.Wrap(err, "Cannot retrieve file descriptors from systemd")
|
|
||||||
}
|
// NewServerWithSettings will create and configure a new API server using provided settings
|
||||||
if len(listeners) != 1 {
|
func NewServerWithSettings(runtime *libpod.Runtime, duration time.Duration, listener *net.Listener) (*APIServer, error) {
|
||||||
return nil, errors.Errorf("Wrong number of file descriptors from systemd for socket activation (%d != 1)", len(listeners))
|
return newServer(runtime, duration, listener)
|
||||||
|
}
|
||||||
|
|
||||||
|
func newServer(runtime *libpod.Runtime, duration time.Duration, listener *net.Listener) (*APIServer, error) {
|
||||||
|
// If listener not provided try socket activation protocol
|
||||||
|
if listener == nil {
|
||||||
|
if _, found := os.LookupEnv("LISTEN_FDS"); !found {
|
||||||
|
return nil, errors.Errorf("Cannot create Server, no listener provided and socket activation protocol is not active.")
|
||||||
|
}
|
||||||
|
|
||||||
|
listeners, err := activation.Listeners()
|
||||||
|
if err != nil {
|
||||||
|
return nil, errors.Wrap(err, "Cannot retrieve file descriptors from systemd")
|
||||||
|
}
|
||||||
|
if len(listeners) != 1 {
|
||||||
|
return nil, errors.Errorf("Wrong number of file descriptors for socket activation protocol (%d != 1)", len(listeners))
|
||||||
|
}
|
||||||
|
listener = &listeners[0]
|
||||||
}
|
}
|
||||||
|
|
||||||
router := mux.NewRouter()
|
router := mux.NewRouter()
|
||||||
@ -86,9 +109,9 @@ func NewServer(runtime *libpod.Runtime) (*APIServer, error) {
|
|||||||
Decoder: schema.NewDecoder(),
|
Decoder: schema.NewDecoder(),
|
||||||
Context: nil,
|
Context: nil,
|
||||||
Runtime: runtime,
|
Runtime: runtime,
|
||||||
Listener: listeners[0],
|
Listener: *listener,
|
||||||
CancelFunc: nil,
|
CancelFunc: nil,
|
||||||
Duration: 300 * time.Second,
|
Duration: duration,
|
||||||
}
|
}
|
||||||
server.Timer = time.AfterFunc(server.Duration, func() {
|
server.Timer = time.AfterFunc(server.Duration, func() {
|
||||||
if err := server.Shutdown(); err != nil {
|
if err := server.Shutdown(); err != nil {
|
||||||
@ -182,6 +205,11 @@ func (s *APIServer) Serve() error {
|
|||||||
|
|
||||||
// Shutdown is a clean shutdown waiting on existing clients
|
// Shutdown is a clean shutdown waiting on existing clients
|
||||||
func (s *APIServer) Shutdown() error {
|
func (s *APIServer) Shutdown() error {
|
||||||
|
// Duration == 0 flags no auto-shutdown of server
|
||||||
|
if s.Duration == 0 {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
// We're still in the sliding service window
|
// We're still in the sliding service window
|
||||||
if s.Timer.Stop() {
|
if s.Timer.Stop() {
|
||||||
s.Timer.Reset(s.Duration)
|
s.Timer.Reset(s.Duration)
|
||||||
|
@ -126,11 +126,11 @@ func (c Connection) ContainerExists(nameOrID string) (bool, error) {
|
|||||||
return false, nil
|
return false, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c Connection) StopContainer(nameOrID string, timeout int) error {
|
func (c Connection) StopContainer(nameOrID string, timeout *int) error {
|
||||||
// TODO we might need to distinguish whether a timeout is desired; a zero, the int
|
|
||||||
// zero value is valid; what do folks want to do?
|
|
||||||
params := make(map[string]string)
|
params := make(map[string]string)
|
||||||
params["t"] = strconv.Itoa(timeout)
|
if timeout != nil {
|
||||||
|
params["t"] = strconv.Itoa(*timeout)
|
||||||
|
}
|
||||||
response, err := c.newRequest(http.MethodPost, fmt.Sprintf("/containers/%s/stop", nameOrID), nil, params)
|
response, err := c.newRequest(http.MethodPost, fmt.Sprintf("/containers/%s/stop", nameOrID), nil, params)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
|
Reference in New Issue
Block a user