自己动手写Docker系列 -- 5.4实现进入容器的namespace,exec命令

本文介绍如何实现进入Docker容器的namespace,通过Cgo调用setns系统调用来实现在Go程序中执行exec命令。文章详细解释了Cgo代码的编写和exec命令的工作原理,包括如何借助/proc/self/exe来确保C代码在exec时运行,从而成功进入命名空间。

简介

在上篇中我们实现了将容器后台运行,本篇中我们将实现docker的ps命令,查看当前正在运行中的容器列表

源码说明

同时放到了Gitee和Github上,都可进行获取

本章节对应的版本标签是:5.4,防止后面代码过多,不好查看,可切换到标签版本进行查看

代码实现

这一部分实现起来就有点麻烦了,其中的一个nsenter始终不能运行正常,折腾了好一阵子发现,需要导出包才行,相关的会在代码中详细的说明

首先我们是需要使用setns去再次进入到我们容器的namespace中:

setns是一个系统调用,可以根据提供的PID再次进入到指定的Namespace 中。它需要先打开/proc/[pid]/ns/文件夹下对应的文件,然后使当前进程进入到指定的Namespace 中

但是一个具有多线程的进程是无法使用setns调用进入到对应的命名空间的,而Go启动一个程序就会进入多线程状态,所以无法简单使用命令调用去实现这个功能,需要借助C来实现

Cgo是一个很炫酷的功能,允许Go程序去调用C的函数与标准库。你只需要以一种特殊的方式在Go的源代码里写出需要调用的C的代码,Cgo就可以把你的C源码文件和Go文件整合成一个包

Cgo代码实现

新建一个文件夹 nsenter,新建文件:nsenter.go

编写Cgo的进入命名空间的代码,如下:

package nsenter

/*
#define _GNU_SOURCE
#include <unistd.h>
#include <errno.h>
#include <sched.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <fcntl.h>

// 构造函数:这里作用是在被引用的时候,这段代码就会执行
__attribute__((constructor)) static void enter_namespace(void) {
	char *mydocker_pid;
    // 从环境变量中获取需要进入的PID
    // 如果没有PID,直接退出,不执行后面的处理逻辑
	mydocker_pid = getenv("mydocker_pid");
	if (mydocker_pid) {
		fprintf(stdout, "got mydocker_pid=%s\n", mydocker_pid);
	} else {
		fprintf(stdout, "missing mydocker_pid env skip nsenter");
		return;
	}
	char *mydocker_cmd;
    // 从环境变量中获取需要执行的命令,没有命令,直接退出
	mydocker_cmd = getenv("mydocker_cmd");
	if (mydocker_cmd) {
		fprintf(stdout, "got mydocker_cmd=%s\n", mydocker_cmd);
	} else {
		fprintf(stdout, "missing mydocker_cmd env skip nsenter");
		return;
	}
	int i;
	char nspath[1024];
	char *namespaces[] = { "ipc", "uts", "net", "pid", "mnt" };

	for (i=0; i<5; i++) {
		sprintf(nspath, "/proc/%s/ns/%s", mydocker_pid, namespaces[i]);
		int fd = open(nspath, O_RDONLY);
        / 调用setns进入对应的namespace
		if (setns(fd, 0) == -1) {
			fprintf(stderr, "setns on %s namespace failed: %s\n", namespaces[i], strerror(errno));
		} else {
			fprintf(stdout, "setns on %s namespace succeeded\n", namespaces[i]);
		}
		close(fd);
	}
    // 进入后执行指定的命令
	int res = system(mydocker_cmd);
	exit(0);
	return;
}
*/
import "C"

如上所示,这样就把进入命名空间的Cgo文件写好了,具体使用在后面会详细说明

Exec命令实现

我们在main中增加exec命令:

func main() {
   
   
	......
	app.Commands = []cli.Command{
   
   
		command.InitCommand,
		command.RunCommand,
		command.CommitCommand,
		command.ListCommand,
		command.LogCommand,
		command.ExecCommand,
	}
    ......
}

在main_command.go文件,增加Exec指令解析

var ExecCommand = cli.Command{
   
   
	Name:  "exec",
	Usage: "exec a command into container",
	Action: func(context *cli.Context) 
<think>好的,用户想要了解如何使用docker exec -it进入容器。首先,我需要回忆一下Docker的基本命令,特别是关于容器操作的。记得docker exec是用来在运行中的容器里执行命令的,而-it参数通常是结合使用的,用于交互式终端。 接下来,我得确认用户可能的需求层次。他们可能已经知道一些Docker基础,但具体到exec-it参数的使用可能不太清楚。需要解释清楚这两个参数的作用,以及为什么需要一起使用。比如,-i是保持STDIN打开,-t是分配伪终端,这样用户才能像在本地终端一样与容器交互。 然后,要举一些例子,这样用户更容易理解。比如进入一个运行中的Ubuntu容器,执行bash。这时候命令应该是docker exec -it container_name bash。但要注意,容器里必须有bash,否则可能需要用sh或者其他shell。 还要提醒用户常见的错误,比如容器没有在运行,这时候exec会失败。或者容器里没有所需的shell,导致无法进入。这时候可能需要先确认容器状态,或者安装相应的shell。 另外,用户可能会混淆docker attach和docker exec的区别,可能需要简单对比一下。attach是连接到正在运行的进程,而exec是在容器内启动新进程,这样退出时不会影响原来容器的运行。 最后,总结一下步骤,确保用户能按照步骤操作,同时避免常见错误。可能还需要提到如何查看容器名称或ID,使用docker ps命令来确认容器是否在运行。</think>Docker中`docker exec -it`进入容器的分步说明: 1. **基础命令结构** ```shell $$ docker exec -it <容器名称或ID> <命令> $$ ``` 示例:`docker exec -it my_nginx bash` 2. **参数解析** - `-i` (--interactive):保持标准输入流(STDIN)打开 - `-t` (--tty):分配伪终端(pseudo-TTY) - 组合使用`-it`可实现交互式终端操作 3. **典型使用场景** - 进入正在运行的容器执行命令 ```shell $ docker exec -it redis-server redis-cli $ ``` - 调试容器环境 ```shell $ docker exec -it web_app sh $ ``` 4. **常见问题处理** - **错误提示**:`Error response from daemon: Container is not running` - 解决方法:先用`docker start <容器名>`启动容器 - **无法找到shell**:`OCI runtime exec failed: exec failed:... no such file or directory` - 确认容器镜像是否包含`/bin/bash`或`/bin/sh` - 改用通用shell:`docker exec -it my_container sh` 5. **退出容器** - 输入`exit`或按`Ctrl+D`退出终端会话 - 注意:退出不会停止容器运行 6. **与`docker attach`的区别** - `exec`新建进程,`attach`连接主进程 - `exec`退出不会影响容器运行,`attach`退出可能导致容器停止 7. **实用技巧** - 查看运行中容器列表: ```shell $ docker ps $ ``` - 进入Windows容器: ```shell $ docker exec -it winserver cmd $ ``` 关键点:必须确保容器处于运行状态(`Up`状态),且目标命令容器内存在。对于Alpine等精简镜像,建议使用`sh`代替`bash`。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值