Skip to content

Commit

Permalink
support nerdctl+containerd runtime (dapr#1232)
Browse files Browse the repository at this point in the history
Signed-off-by: tianya <[email protected]>
  • Loading branch information
tianyax committed Feb 25, 2023
1 parent a6f8c76 commit d19f69b
Show file tree
Hide file tree
Showing 5 changed files with 76 additions and 33 deletions.
8 changes: 4 additions & 4 deletions cmd/init.go
Original file line number Diff line number Diff line change
Expand Up @@ -48,7 +48,7 @@ var (
var InitCmd = &cobra.Command{
Use: "init",
Short: "Install Dapr on supported hosting platforms. Supported platforms: Kubernetes and self-hosted",
PreRun: func(cmd *cobra.Command, args []string) {
PreRun: func(cmd *cobra.Command, _ []string) {
viper.BindPFlag("network", cmd.Flags().Lookup("network"))
viper.BindPFlag("image-registry", cmd.Flags().Lookup("image-registry"))
},
Expand Down Expand Up @@ -87,7 +87,7 @@ dapr init --runtime-path <path-to-install-directory>
# See more at: https://docs.dapr.io/getting-started/
`,
Run: func(cmd *cobra.Command, args []string) {
Run: func(*cobra.Command, []string) {
print.PendingStatusEvent(os.Stdout, "Making the jump to hyperspace...")
imageRegistryFlag := strings.TrimSpace(viper.GetString("image-registry"))

Expand Down Expand Up @@ -148,7 +148,7 @@ dapr init --runtime-path <path-to-install-directory>
}

if !utils.IsValidContainerRuntime(containerRuntime) {
print.FailureStatusEvent(os.Stdout, "Invalid container runtime. Supported values are docker and podman.")
print.FailureStatusEvent(os.Stdout, "Invalid container runtime. Supported values are docker and podman and containerd.")
os.Exit(1)
}
err := standalone.Init(runtimeVersion, dashboardVersion, dockerNetwork, slimMode, imageRegistryURI, fromDir, containerRuntime, imageVariant, daprRuntimePath)
Expand Down Expand Up @@ -194,7 +194,7 @@ func init() {
InitCmd.Flags().BoolP("help", "h", false, "Print this help message")
InitCmd.Flags().StringArrayVar(&values, "set", []string{}, "set values on the command line (can specify multiple or separate values with commas: key1=val1,key2=val2)")
InitCmd.Flags().String("image-registry", "", "Custom/private docker image repository URL")
InitCmd.Flags().StringVarP(&containerRuntime, "container-runtime", "", "docker", "The container runtime to use. Supported values are docker (default) and podman")
InitCmd.Flags().StringVarP(&containerRuntime, "container-runtime", "", "docker", "The container runtime to use. Supported values are docker (default) and podman and containerd")

RootCmd.AddCommand(InitCmd)
}
8 changes: 4 additions & 4 deletions cmd/uninstall.go
Original file line number Diff line number Diff line change
Expand Up @@ -52,11 +52,11 @@ dapr uninstall -k
# This will remove the .dapr directory present in the path <path-to-install-directory>
dapr uninstall --runtime-path <path-to-install-directory>
`,
PreRun: func(cmd *cobra.Command, args []string) {
PreRun: func(cmd *cobra.Command, _ []string) {
viper.BindPFlag("network", cmd.Flags().Lookup("network"))
viper.BindPFlag("install-path", cmd.Flags().Lookup("install-path"))
},
Run: func(cmd *cobra.Command, args []string) {
Run: func(*cobra.Command, []string) {
var err error

if uninstallKubernetes {
Expand All @@ -69,7 +69,7 @@ dapr uninstall --runtime-path <path-to-install-directory>
err = kubernetes.Uninstall(uninstallNamespace, uninstallAll, timeout)
} else {
if !utils.IsValidContainerRuntime(uninstallContainerRuntime) {
print.FailureStatusEvent(os.Stdout, "Invalid container runtime. Supported values are docker and podman.")
print.FailureStatusEvent(os.Stdout, "Invalid container runtime. Supported values are docker and podman and containerd.")
os.Exit(1)
}
print.InfoStatusEvent(os.Stdout, "Removing Dapr from your machine...")
Expand All @@ -92,6 +92,6 @@ func init() {
UninstallCmd.Flags().String("network", "", "The Docker network from which to remove the Dapr runtime")
UninstallCmd.Flags().StringVarP(&uninstallNamespace, "namespace", "n", "dapr-system", "The Kubernetes namespace to uninstall Dapr from")
UninstallCmd.Flags().BoolP("help", "h", false, "Print this help message")
UninstallCmd.Flags().StringVarP(&uninstallContainerRuntime, "container-runtime", "", "docker", "The container runtime to use. Supported values are docker (default) and podman")
UninstallCmd.Flags().StringVarP(&uninstallContainerRuntime, "container-runtime", "", "docker", "The container runtime to use. Supported values are docker (default) and podman and containerd")
RootCmd.AddCommand(UninstallCmd)
}
37 changes: 22 additions & 15 deletions pkg/standalone/standalone.go
Original file line number Diff line number Diff line change
Expand Up @@ -161,10 +161,9 @@ func Init(runtimeVersion, dashboardVersion string, dockerNetwork string, slimMod
fromDir = strings.TrimSpace(fromDir)
setAirGapInit(fromDir)
if !slimMode {
// If --slim installation is not requested, check if docker is installed.
conatinerRuntimeAvailable := utils.IsDockerInstalled() || utils.IsPodmanInstalled()
if !conatinerRuntimeAvailable {
return errors.New("could not connect to Docker. Docker may not be installed or running")
// If --slim installation is not requested, check if docker or podman or containerd is installed.
if !utils.ContainerRuntimeAvailable() {
return errors.New("could not connect to conatiner. Docker or podman or containerd may not be installed or running")
}

// Initialize default registry only if any of --slim or --image-registry or --from-dir are not given.
Expand Down Expand Up @@ -363,10 +362,8 @@ func runZipkin(wg *sync.WaitGroup, errorChan chan<- error, info initInfo) {
)

if info.dockerNetwork != "" {
args = append(
args,
"--network", info.dockerNetwork,
"--network-alias", DaprZipkinContainerName)
networks := withContainerNetwork(info.containerRuntime, info.dockerNetwork, DaprZipkinContainerName)
args = append(args, networks...)
} else {
args = append(
args,
Expand Down Expand Up @@ -430,10 +427,8 @@ func runRedis(wg *sync.WaitGroup, errorChan chan<- error, info initInfo) {
)

if info.dockerNetwork != "" {
args = append(
args,
"--network", info.dockerNetwork,
"--network-alias", DaprRedisContainerName)
networks := withContainerNetwork(info.containerRuntime, info.dockerNetwork, DaprRedisContainerName)
args = append(args, networks...)
} else {
args = append(
args,
Expand Down Expand Up @@ -510,9 +505,8 @@ func runPlacementService(wg *sync.WaitGroup, errorChan chan<- error, info initIn
}

if info.dockerNetwork != "" {
args = append(args,
"--network", info.dockerNetwork,
"--network-alias", DaprPlacementContainerName)
networks := withContainerNetwork(info.containerRuntime, info.dockerNetwork, DaprPlacementContainerName)
args = append(args, networks...)
} else {
osPort := 50005
if runtime.GOOS == daprWindowsOS {
Expand Down Expand Up @@ -1011,6 +1005,19 @@ func createDefaultConfiguration(zipkinHost, filePath string) error {
return err
}

// withContainerNetwork connect a container to a network.
// Network alias is now only parsed by docker,
// `podman` is only compatible with docker commands
// and does not really implement this feature.
// `containerd` does not support network alias.
func withContainerNetwork(containerCmd, network, containerName string) []string {
networks := []string{"--network", network}
if utils.GetContainerRuntimeCmd(containerCmd) == string(utils.DOCKER) {
networks = append(networks, "--network-alias", containerName)
}
return networks
}

func checkAndOverWriteFile(filePath string, b []byte) error {
_, err := os.Stat(filePath)
if os.IsNotExist(err) {
Expand Down
8 changes: 3 additions & 5 deletions pkg/standalone/uninstall.go
Original file line number Diff line number Diff line change
Expand Up @@ -90,11 +90,9 @@ func Uninstall(uninstallAll bool, dockerNetwork string, containerRuntime string,
print.WarningStatusEvent(os.Stdout, "WARNING: could not delete dapr bin dir: %s", daprBinDir)
}

containerRuntime = strings.TrimSpace(containerRuntime)
runtimeCmd := utils.GetContainerRuntimeCmd(containerRuntime)
conatinerRuntimeAvailable := false
conatinerRuntimeAvailable = utils.IsDockerInstalled() || utils.IsPodmanInstalled()
if conatinerRuntimeAvailable {
if utils.ContainerRuntimeAvailable() {
containerRuntime = strings.TrimSpace(containerRuntime)
runtimeCmd := utils.GetContainerRuntimeCmd(containerRuntime)
containerErrs = removeContainers(uninstallPlacementContainer, uninstallAll, dockerNetwork, runtimeCmd)
}

Expand Down
48 changes: 43 additions & 5 deletions utils/utils.go
Original file line number Diff line number Diff line change
Expand Up @@ -24,7 +24,9 @@ import (
"os"
"os/exec"
"path/filepath"
"runtime"
"strings"
"sync"
"time"

"github.com/dapr/cli/pkg/print"
Expand All @@ -38,25 +40,44 @@ import (
type ContainerRuntime string

const (
DOCKER ContainerRuntime = "docker"
PODMAN ContainerRuntime = "podman"
DOCKER ContainerRuntime = "docker"
PODMAN ContainerRuntime = "podman"
CONTAINERD ContainerRuntime = "containerd"

MACOS = runtime.GOOS == "darwin"

marinerImageVariantName = "mariner"

socketFormat = "%s/dapr-%s-%s.socket"
)

var (
cacheOnce sync.Once
supportMacos bool
)

var supportRuntime = func(containerRuntime string) bool {
cacheOnce.Do(func() {
supportMacos = containerRuntime == string(CONTAINERD) && !MACOS
})
return supportMacos
}

// IsValidContainerRuntime checks if the input is a valid container runtime.
// Valid container runtimes are docker and podman.
// Valid container runtimes are docker and podman and containerd.
func IsValidContainerRuntime(containerRuntime string) bool {
containerRuntime = strings.TrimSpace(containerRuntime)
return containerRuntime == string(DOCKER) || containerRuntime == string(PODMAN)
return containerRuntime == string(DOCKER) || containerRuntime == string(PODMAN) || supportRuntime(containerRuntime)
}

// GetContainerRuntimeCmd returns a valid container runtime to be used by CLI operations.
// If the input is a valid container runtime, it is returned as is.
// If the input is a valid container runtime, it is returned client tool.
// Otherwise the default container runtime, docker, is returned.
func GetContainerRuntimeCmd(containerRuntime string) string {
// containerd runtime use nerdctl tool.
if supportRuntime(containerRuntime) {
return "nerdctl"
}
if IsValidContainerRuntime(containerRuntime) {
return strings.TrimSpace(containerRuntime)
}
Expand Down Expand Up @@ -188,6 +209,23 @@ func IsPodmanInstalled() bool {
return true
}

// IsContainerdInstalled checks whether nerdctl and containerd is installed/running.
func IsContainerdInstalled() bool {
if MACOS {
print.FailureStatusEvent(os.Stderr, "containerd is not supported on macos")
return false
}
if _, err := RunCmdAndWait("nerdctl", "info"); err != nil {
print.FailureStatusEvent(os.Stderr, err.Error())
return false
}
return true
}

func ContainerRuntimeAvailable() bool {
return IsDockerInstalled() || IsPodmanInstalled() || IsContainerdInstalled()
}

// IsDaprListeningOnPort checks if Dapr is litening to a given port.
func IsDaprListeningOnPort(port int, timeout time.Duration) error {
start := time.Now()
Expand Down

0 comments on commit d19f69b

Please sign in to comment.