点击下方名片,设为星标

回复“1024”获取2TB学习资源!

如何在容器中进行抓包?一篇文章搞懂其原理!

大家好,我是民工哥,今天就来同大家一起来学习一下,在容器中进行抓包的命令原理与操作。

简介

nsenter 命令是一个可以在指定进程的命令空间下运行指定程序的命令。它位于 util-linux 包中。

用途

一个最典型的用途就是进入容器的网络命令空间。相当多的容器为了轻量级,是不包含较为基础的命令的,比如说 ip address,ping,telnet,ss,tcpdump 等等命令,这就给调试容器网络带来相当大的困扰:只能通过 docker inspect ContainerID 命令获取到容器 IP,以及无法测试和其他网络的连通性。这时就可以使用 nsenter 命令仅进入该容器的网络命名空间,使用宿主机的命令调试容器网络。

此外,nsenter 也可以进入mntutsipcpiduser命令空间,以及指定根目录和工作目录。

使用

首先看下 nsenter 命令的语法:

nsenter [options] [program [arguments]]

参数说明

-t, --target pid #指定被进入命名空间的目标进程的pid
-m, --mount[=file] #进入mount命令空间。如果指定了file,则进入file的命令空间
-u, --uts[=file] #进入uts命令空间。如果指定了file,则进入file的命令空间
-i, --ipc[=file] #进入ipc命令空间。如果指定了file,则进入file的命令空间
-n, --net[=file] #进入net命令空间。如果指定了file,则进入file的命令空间
-p, --pid[=file] #进入pid命令空间。如果指定了file,则进入file的命令空间
-U, --user[=file] #进入user命令空间。如果指定了file,则进入file的命令空间
-G, --setgid gid  #设置运行程序的gid
-S, --setuid uid  #设置运行程序的uid
-r, --root[=directory] #设置根目录
-w, --wd[=directory]  #设置工作目录

如果没有给出program,则默认执行$SHELL。

示例

运行一个 nginx 容器,查看该容器的 pid:

[root@staight ~]# docker inspect -f {{.State.Pid}} nginx
5645

然后,使用 nsenter 命令进入该容器的网络命令空间:

[root@staight ~]# nsenter -n -t5645
[root@staight ~]# ip addr
1: lo: <LOOPBACK,UP,LOWER_UP> mtu 65536 qdisc noqueue state UNKNOWN group default qlen 1000link/loopback 00:00:00:00:00:00 brd 00:00:00:00:00:00inet 127.0.0.1/8 scope host lovalid_lft forever preferred_lft forever
18: eth0@if19: <BROADCAST,MULTICAST,UP,LOWER_UP> mtu 1500 qdisc noqueue state UP group default link/ether 02:42:ac:11:00:02 brd ff:ff:ff:ff:ff:ff link-netnsid 0inet 172.17.0.2/16 brd 172.17.255.255 scope global eth0valid_lft forever preferred_lft forever

进入成功~

在 Kubernetes 中,在得到容器 pid 之前还需获取容器的 ID,可以使用如下命令获取:

[root@node1 test]# kubectl get pod test -oyaml|grep containerID- containerID: docker://cf0873782d587dbca6aa32f49605229da3748600a9926e85b36916141597ec85

或者更为精确地获取 containerID :

[root@node1 test]# kubectl get pod test -o template --template='{{range .status.containerStatuses}}{{.containerID}}{{end}}'
docker://cf0873782d587dbca6aa32f49605229da3748600a9926e85b36916141597ec85

原理

namespace

namespace 是 Linux 中一些进程的属性的作用域,使用命名空间,可以隔离不同的进程。

Linux在不断的添加命名空间,目前有:

mount:#挂载命名空间,使进程有一个独立的挂载文件系统,始于Linux 2.4.19
ipc:#ipc命名空间,使进程有一个独立的ipc,包括消息队列,共享内存和信号量,始于Linux 2.6.19
uts:#uts命名空间,使进程有一个独立的hostname和domainname,始于Linux 2.6.19
net:#network命令空间,使进程有一个独立的网络栈,始于Linux 2.6.24
pid:#pid命名空间,使进程有一个独立的pid空间,始于Linux 2.6.24
user:#user命名空间,是进程有一个独立的user空间,始于Linux 2.6.23,结束于Linux 3.8
cgroup:#cgroup命名空间,使进程有一个独立的cgroup控制组,始于Linux 4.6

Linux 的每个进程都具有命名空间,可以在 /proc/PID/ns 目录中看到命名空间的文件描述符。

[root@staight ns]# pwd
/proc/1/ns
[root@staight ns]# ll
total 0
lrwxrwxrwx 1 root root 0 Sep 23 19:53 ipc -> ipc:[4026531839]
lrwxrwxrwx 1 root root 0 Sep 23 19:53 mnt -> mnt:[4026531840]
lrwxrwxrwx 1 root root 0 Sep 23 19:53 net -> net:[4026531956]
lrwxrwxrwx 1 root root 0 Sep 23 19:53 pid -> pid:[4026531836]
lrwxrwxrwx 1 root root 0 Sep 23 19:53 user -> user:[4026531837]
lrwxrwxrwx 1 root root 0 Sep 23 19:53 uts -> uts:[4026531838]

clone

clone 是 Linux 的系统调用函数,用于创建一个新的进程。

clone 和 fork 比较类似,但更为精细化,比如说使用 clone 创建出的子进程可以共享父进程的虚拟地址空间,文件描述符表,信号处理表等等。不过这里要强调的是,clone 函数还能为新进程指定命名空间。

clone的语法
#define _GNU_SOURCE
#include <sched.h>int clone(int (*fn)(void *), void *child_stack,int flags, void *arg, .../* pid_t *ptid, void *newtls, pid_t *ctid */ );

其中 flags 即可指定命名空间,包括:

CLONE_NEWCGROUP:cgroupCLONE_NEWIPC:ipcCLONE_NEWNET:netCLONE_NEWNS:mountCLONE_NEWPID:pidCLONE_NEWUSER:userCLONE_NEWUTS:uts
使用示例:
pid = clone(childFunc, stackTop, CLONE_NEWUTS | SIGCHLD, argv[1]);

setns

clone 用于创建新的命令空间,而 setns 则用来让当前线程(单线程即进程)加入一个命名空间。

语法
#define _GNU_SOURCE             /* See feature_test_macros(7) */
#include <sched.h>int setns(int fd, int nstype);
  • fd 参数是一个指向一个命名空间的文件描述符,位于/proc/PID/ns/目录。

  • nstype指定了允许进入的命名空间,一般可设置为0,表示允许进入所有命名空间。

因此,往往该函数的用法为:

  • 调用setns函数:指定该线程的命名空间。

  • 调用execvp函数:执行指定路径的程序,创建子进程并替换父进程。

这样,就可以指定命名空间运行新的程序了。

代码示例:

#define _GNU_SOURCE
#include <fcntl.h>
#include <sched.h>
#include <unistd.h>
#include <stdlib.h>
#include <stdio.h>#define errExit(msg)    do { perror(msg); exit(EXIT_FAILURE); \} while (0)int
main(int argc, char *argv[])
{int fd;if (argc < 3) {fprintf(stderr, "%s /proc/PID/ns/FILE cmd args...\n", argv[0]);exit(EXIT_FAILURE);}fd = open(argv[1], O_RDONLY); /* Get file descriptor for namespace */if (fd == -1)errExit("open");if (setns(fd, 0) == -1)       /* Join that namespace */errExit("setns");execvp(argv[2], &argv[2]);    /* Execute a command in namespace */errExit("execvp");
}
使用示例:
./ns_exec /proc/3550/ns/uts /bin/bash

nsenter

那么,最后就是 nsenter 了,nsenter 相当于在setns的示例程序之上做了一层封装,使我们无需指定命名空间的文件描述符,而是指定进程号即可。

指定进程号PID以及需要进入的命名空间后,nsenter会帮我们找到对应的命名空间文件描述符/proc/PID/ns/FD,然后使用该命名空间运行新的程序。

来源:https://staight.github.io/2019/09/23/nsenter

读者专属技术群

构建高质量的技术交流社群,欢迎从事后端开发、运维技术进群(备注岗位,已在技术交流群的请勿重复添加),主要以技术交流、内推、行业探讨为主。广告人士勿入,切勿轻信私聊,防止被骗。

扫码加我好友,拉你进群

推荐阅读 点击标题可跳转

又一款轻量级监控利器!开源了

去了一家不到20人的IT公司后,真的是大开眼界

目前工资最高的外包公司汇总(2023最新版)!

卸载 Postman!事实证明,它更牛逼。。。

果真有这样的公司,够横!直接干上了热搜

C 盘真的一下子就不红了!微软官方工具就能解决

面试官:如何修改正运行的Docker容器端口映射?

PS:因为公众号平台更改了推送规则,如果不想错过内容,记得读完点一下在看,加个星标,这样每次新文章推送才会第一时间出现在你的订阅列表里。点在看支持我们吧!

面试官:如何在 Docker 容器中抓包?问倒一大片。。。。相关推荐

  1. 如何在Docker容器中挂载主机目录

    本文翻译自:How to mount a host directory in a Docker container I am trying to mount a host directory into ...

  2. 如何在Docker容器中运行GUI程序

    如何在Docker容器中运行GUI程序 各位,今天我们将学习如何在Docker之中运行GUI程序.我们可以轻易地在Docker容器中运行大多数GUI程序且不出错.Docker是一个开源项目,提供了一个 ...

  3. 如何在Docker容器中运行Docker [3种方法]

    在本博客中,我将向您介绍在docker中运行docker所需的三种不同方法. Docker In Docker的用处 dockerIndocker的一个潜在用处是CI管道,在代码成功构建后,您需要在其 ...

  4. sigterm信号_详解如何在 docker 容器中捕获信号

    概述 玩过docker的朋友可能都使用过 docker stop 命令来停止正在运行的容器,有些会使用 docker kill 命令强行关闭容器或者把某个信号传递给容器中的进程.这些操作的本质都是通过 ...

  5. Beats:如何在 Docker 容器中运行 Filebeat

    今天在这篇博客中,我们将学习如何在容器环境中运行 Filebeat. 为了快速了解 Filebeat 是做什么用的: Filebeat用于转发和集中日志数据 它重量轻,小型化,使用的资源更少 它作为代 ...

  6. 面试官 | 如何在 Spring Boot 中进行参数校验?

    作者 | 狂乱的贵公子 来源 | cnblogs.com/cjsblog/p/8946768.html 开发过程中,后台的参数校验是必不可少的,所以经常会看到类似下面这样的代码 这样写并没有什么错,还 ...

  7. 关于fi dd ler 手机抓包 网卡地址地址_面试官:你给我讲讲抓包神器tcpdump的原理...

    点击上方蓝色字关注我们~ 面试官 你说你会网络编程?你说你熟悉网络知识,那你使用过tcpdump吗?能给我讲下tcpdump是什么,或者你有用过tcpdump解决过实际问题吗? 如果你学过网络,甚至搞 ...

  8. 如何在Docker容器中安装RabbitMQ

    1.Docker环境 视频教程:https://www.bilibili.com/video/BV1xv4y1S7kA/ 2.搜索镜像 https://hub.docker.com/网站搜索rabbi ...

  9. 进入docker容器中查看文件夹_如何在docker容器文件夹中提供文件夹权限

    我正在我的Dockerfile中创建一个文件夹,我想给它一个写权限.但是当我尝试这样做时,我得到了许可被拒绝的错误 FROM python:2.7 RUN pip install Flask==0.1 ...

最新文章

  1. linux的文本处理(一)
  2. C++中常用的字符串函数
  3. 835 由于安全层无法对远程计算机进行身份验证_干货 | 看黑客都是这样远程桌面安全设置...
  4. linux inotify-tools 监控文件变化
  5. IBM RAS:高效存储的优化组合
  6. 和 Gmail 先生面对面
  7. 并发编程-08安全发布对象之发布与逸出
  8. 高等数学上-赵立军-北京大学出版社-题解-练习2.4
  9. 构造方法与对象的创建 java 1614780813
  10. mysql utf8转gbk cmd_utf8转成gbk
  11. 数据结构区间问题总结
  12. JavaScript实现继承的方式
  13. GPUImageMovieWriter录制视频问题
  14. div section article区分--20150227
  15. 使用Ajax开发一个用户注册程序
  16. 傅里叶变换中的假频**
  17. 【游戏开发进阶】教你在Windows平台编译tolua runtime的各个平台库(Unity | 热更新 | tolua | 交叉编译)
  18. 跑实验_word2vector词向量实现_基于搜狗新闻预料+维基百科
  19. 线性代数_1、二阶、三阶行列式、排列、逆序
  20. JS-JavaScript_简介及基本使用

热门文章

  1. devops项目经理_DevOps招聘经理应准备回答的20个问题
  2. JS(8)、for循环
  3. windows git and linux git lib operater
  4. 【推荐收藏】【机器学习实战】训练模型(挑战全网最全)
  5. python读取word文档中的换行_python - 读取没有换行符的文件
  6. 03-PDI(Kettle)导入与导出CDC
  7. 网络聊天室————韩顺平版qq(有离线功能)
  8. 对象背后的那只看不见的手——软件世界的有神论
  9. 【MATLAB】全局莫兰指数(含p值和z值)
  10. python机器交易_全解用 Python 建立能源市场算法交易的机器学习框架