ShellShock 攻击实验

1  ShellShock 简介

     Shellshock,又称Bashdoor,是在Unix中广泛使用的Bash shell中的一个安全漏洞,首次于2014年9月24日公开。许多互联网守护进程,如网页服务器,使用bash来处理某些命令,从而允许攻击者在易受攻击的Bash版本上执行任意代码。这可使攻击者在未授权的情况下访问计算机系统。

2  实验目的

在本实验中,学生需要亲手重现攻击来理解该漏洞,并回答一些问题。

3  实验前的准备工作

3.1  环境搭建

      以root权限安装4.1版bash(4.2版本以上的漏洞已经被堵上了) bash4.1 原来的下载地址是 http://ftp/gnu.org/gnu/bash/bash-4.1.tar.gz ,为了加快速度,我们这里使用下面的下载地址 http://labfile.oss.aliyuncs.com/bash-4.1.tar.gz

下载:

安装:

链接:

到这里就安装完了,接下来检测是否存在shellshock漏洞。

输出vulnerable的话,说明bash有漏洞。

上面的这个命令分为了两部分:

后面是 bash -c "echo this is a test" ,功能用bash执行后面的字符串中的命令,也就是把引号中的内容当成一个bash脚本来执行。

前面 env x='() { :; }; echo vulnerable' 是定义了带有环境变量 x 的临时环境,放在最前面的语法含义是,在这个环境中执行后面的脚本。这一整句是一条设置环境系统的语句,按理来说,它的功能只是执行系统环境变量的功能。这条语句中的 “ echo vulnerable ” 是一条打印语句,理论上不应该出现在环境变量设置里面,就算出现了也不应该执行的。表示打印 “ vulnerable ” 这个字符的意思。理论上,这句命令是不应该被执行的,如果被执行了,就意味着有漏洞。入侵者可以把”echo XXXXXX“ 这句命令,换成其他的shell命令,从而做一些不被允许的事情。

env 一般还可以省略。也就等价于:

export x='() { :; }; echo vulnerable'
bash -c "echo this is a test"

:冒号:的含义是pass,也就是什么都不做。

Bash把 x 定义中的内容当成了Shell命令来执行了。

现在一切就绪,进入下一步吧。

1.2  预备知识

了解bash自定义函数,只需要函数名就能够调用该函数。

这个时候的Bash的环境变量:KEY = foo      VALUE = () { echo bar; }

来看看ShellShock漏洞的真身:

为什么调用bash的时候输出Hello World了呢?瞧瞧他内部的情况:KEY = foo        VALUE = () { :; }; echo Hello World

这里同理,我们函数定义后面的内容意外的被执行了。所以,问题就在于Bash对于环境变量中的函数定义的转换是有漏洞的:  Bash可以把foo 的函数体 export到环境变量里面,就是把函数的代码当成了环境变量存下来的。启动Bash的时候,它不但会导入这个函数,而且也会把函数定义后面的命令执行。

注意:在环境变量中定义函数的命令时,() 后面和 { 之间的空格是必须的,否则Bash不会认为它是一个函数。

3  攻击Set-UID程序  

本实验中,我们通过攻击Set-UID程序来获得root权限。

我们知道system()函数将调用"/bin/sh -c" 来运行指定的命令, 这也意味着 /bin/bash 会被调用,你能够利用shellshock漏洞来获取权限么? 首先,确保安装了带有漏洞的bash版本,并让 /bin/sh 指向 /bin/bash。

在 /home/shiyanlou 目录下新建一个 shock.c 文件:

按 I 键切换到插入模式,再输入如下内容:

编译这段代码,并设置其为Set-UID程序,保证它的所有者是root。

我们注意到这里使用了 setuid(geteuid()) 来使real uid = effective uid ,这在Set-UID程序中不是普遍现象,但它确实有时会发生。

直接执行shock文件:

可以看到执行的是shock.c里的代码。

以下是加了“ export foo='() { :; }; bash' ”之后,运行shock文件的结果:

可以看到转换成 root 权限模式了,并且shock.c里的 “/bin/ls -l”没有执行。

假如这段代码不具有root权限,看看结果:

执行了是shock.c里的“/bin/ls -l”命令。

这里就能得出,export foo='() { :; }; bash' 这条语句,把 foo 的函数体 export到环境变量里面,就是把函数的代码当成了环境变量存下来的。一旦运行一个具有 root 权限的可执行文件,就会触发shellshock漏洞,如果一个文件不具有root权限,就不能触发shellshock漏洞。

如果 setuid(geteuid()) 语句被去掉了,再试试看攻击,我们还能够拿到权限么?

失败了!这就说明如果 real uid 和 effective uid 相同的话,定义在环境变量中的内容在该程序内有效,那样shellshock漏洞就能够被利用了。但是如果两个 uid 不同的话,环境变量失效,就无法发动攻击了,这可以从 bash的源代码中得到印证(variables.c,在308到369行之间)请指出是哪一行导致了这样的不同,并说明bash这样设计的原因。

这里给出部分代码

/* Initialize the shell variables from the current environment.
If PRIVMODE is nonzero, don't import functions from ENV or
parse $SHELLOPTS. */
void
initialize_shell_variables (env, privmode)
char **env;
int privmode;
{
char *name, *string, *temp_string;
int c, char_index, string_index, string_length;
SHELL_VAR *temp_var; create_variable_tables (); for (string_index = ; string = env[string_index++]; )
{ char_index = ;
name = string;
while ((c = *string++) && c != '=')
;
if (string[-] == '=')
char_index = string - name - ; /* If there are weird things in the environment, like `=xxx' or a
string without an `=', just skip them. */
if (char_index == )
continue; /* ASSERT(name[char_index] == '=') */
name[char_index] = '\0';
/* Now, name = env variable name, string = env variable value, and
char_index == strlen (name) */ temp_var = (SHELL_VAR *)NULL; /* If exported function, define it now. Don't import functions from
the environment in privileged mode. */
if (privmode == && read_but_dont_execute == && STREQN ("() {", string, ))
{
string_length = strlen (string);
temp_string = (char *)xmalloc ( + string_length + char_index); strcpy (temp_string, name);
temp_string[char_index] = ' ';
strcpy (temp_string + char_index + , string); parse_and_execute (temp_string, name, SEVAL_NONINT|SEVAL_NOHIST); /* Ancient backwards compatibility. Old versions of bash exported
functions like name()=() {...} */
if (name[char_index - ] == ')' && name[char_index - ] == '(')
name[char_index - ] = '\0'; if (temp_var = find_function (name))
{
VSETATTR (temp_var, (att_exported|att_imported));
array_needs_making = ;
}
else
report_error (_("error importing function definition for `%s'"), name); /* ( */
if (name[char_index - ] == ')' && name[char_index - ] == '\0')
name[char_index - ] = '('; /* ) */
}

摘出其中关键部分并简化

void initialize_shell_variables(){
// 循环遍历所有环境变量
for (string_index = ; string = env[string_index++]; ) {
/*...*/
/* 如果有export过的函数, 在这里定义 */
/* 无法导入在特权模式下(root下)定义的函数 */
if (privmode == && read_but_dont_execute == &&
STREQN (“() {“, string, )) {
[...]
// 这里是shellshock发生的地方
// 传递函数定义 + 运行额外的指令
parse_and_execute (temp_string, name,
SEVAL_NONINT|SEVAL_NOHIST);
[...]
} }

就是上述那一行判断逻辑导致了两者的不同,primode即私有模式,要求real uid 与 effective uid保持一致。

4  实验总结

漏洞成因调用了bash执行系统命令,通过特定格式的指令设置环境变量,进而在执行bash脚本时会先执行把我们设置的环境变量执行。有了bash不等于有一切,要看是什么权限运行的bash。cgi服务器进程一般不会是root权限。如果是的话,那个网站的网(yun)管(wei)就可以开了又开了。所以拿到的很大几率是一个像"www_data"这样的用户权限拿到www_data可以做什么呢?可以看网站的配置文件,里面可能有数据库密码,然后就可以进数据库了。”

参考链接:https://blog.csdn.net/fish43237/article/details/39609031

http://www.klogk.com/articles/2014-09-28/a-short-introduction-to-ShellShock/

2019-2020-11 20199317 《Linux内核原理与分析》 第十一周作业的更多相关文章

  1. 2019-2020-1 20199329《Linux内核原理与分析》第九周作业

    <Linux内核原理与分析>第九周作业 一.本周内容概述: 阐释linux操作系统的整体构架 理解linux系统的一般执行过程和进程调度的时机 理解linux系统的中断和进程上下文切换 二 ...

  2. 2019-2020-1 20199329《Linux内核原理与分析》第二周作业

    <Linux内核原理与分析>第二周作业 一.上周问题总结: 未能及时整理笔记 Linux还需要多用 markdown格式不熟练 发布博客时间超过规定期限 二.本周学习内容: <庖丁解 ...

  3. 20169212《Linux内核原理与分析》第二周作业

    <Linux内核原理与分析>第二周作业 这一周学习了MOOCLinux内核分析的第一讲,计算机是如何工作的?由于本科对相关知识的不熟悉,所以感觉有的知识理解起来了有一定的难度,不过多查查资 ...

  4. 20169210《Linux内核原理与分析》第二周作业

    <Linux内核原理与分析>第二周作业 本周作业分为两部分:第一部分为观看学习视频并完成实验楼实验一:第二部分为看<Linux内核设计与实现>1.2.18章并安装配置内核. 第 ...

  5. 2018-2019-1 20189221 《Linux内核原理与分析》第九周作业

    2018-2019-1 20189221 <Linux内核原理与分析>第九周作业 实验八 理理解进程调度时机跟踪分析进程调度与进程切换的过程 进程调度 进度调度时机: 1.中断处理过程(包 ...

  6. 2017-2018-1 20179215《Linux内核原理与分析》第二周作业

    20179215<Linux内核原理与分析>第二周作业 这一周主要了解了计算机是如何工作的,包括现在存储程序计算机的工作模型.X86汇编指令包括几种内存地址的寻址方式和push.pop.c ...

  7. 2019-2020-1 20209313《Linux内核原理与分析》第二周作业

    2019-2020-1 20209313<Linux内核原理与分析>第二周作业 零.总结 阐明自己对"计算机是如何工作的"理解. 一.myod 步骤 复习c文件处理内容 ...

  8. 2018-2019-1 20189221《Linux内核原理与分析》第一周作业

    Linux内核原理与分析 - 第一周作业 实验1 Linux系统简介 Linux历史 1991 年 10 月,Linus Torvalds想在自己的电脑上运行UNIX,可是 UNIX 的商业版本非常昂 ...

  9. 《Linux内核原理与分析》第一周作业 20189210

    实验一 Linux系统简介 这一节主要学习了Linux的历史,Linux有关的重要人物以及学习Linux的方法,Linux和Windows的区别.其中学到了LInux中的应用程序大都为开源自由的软件, ...

  10. 2018-2019-1 20189221《Linux内核原理与分析》第二周作业

    读书报告 <庖丁解牛Linux内核分析> 第 1 章 计算工作原理 1.1 存储程序计算机工作模型 1.2 x86-32汇编基础 1.3汇编一个简单的C语言程序并分析其汇编指令执行过程 因 ...

随机推荐

  1. NOIP模拟 39

    考的嘛也不是. 伤心(怎么可能) T1稍想想组合数,然后牢记: 取模题随时取模,包括刚刚读入的数据  T2想到了基环树,然而不会打QAQ.. 非常简洁但非常大神的做法:随便断掉环上的一条边 利用“这条 ...

  2. NOIP模拟 9

    %liu_runda Orz T1 随 矩阵快速幂结合概率期望 但n3无法承受 利用原根的性质,将乘法转化成加法 就变成循环矩阵n^2了 改题时苦b地卡了关:误把1当成原根的1次方,错误地认为矩阵的阶 ...

  3. 搞清楚 Python 的迭代器、可迭代对象、生成器

    很多伙伴对 Python 的迭代器.可迭代对象.生成器这几个概念有点搞不清楚,我来说说我的理解,希望对需要的朋友有所帮助. 1 迭代器协议 迭代器协议是核心,搞懂了这个,上面的几个概念也就很好理解了. ...

  4. Vue组件间通信方式到底有几种

    1. 前言 Vue的一个核心思想就是组件化.所谓组件化,就是把页面拆分成多个组件 (component),每个组件依赖的 CSS.JavaScript.模板.图片等资源放在一起开发和维护.组件是资源独 ...

  5. 机器学习环境搭建安装TensorFlow1.13.1+Anaconda3.5.3+Python3.7.1+Win10

    安装Python3.7.1 此处不再赘述安装过程,作为记录 安装Anaconda3.5.3 Anaconda3-5.3.0-Windows-x86_64.exe 方案1. 可以直接从官网https:/ ...

  6. go中的数据结构字典-map

    1. map的使用 golang中的map是一种数据类型,将键与值绑定到一起,底层是用哈希表实现的,可以快速的通过键找到对应的值. 类型表示:map[keyType][valueType] key一定 ...

  7. WIN7安装Docker Toolbox、制作镜像并发到阿里云

    一.安装Docker Toolbox,并配置国内源加速 WIndows7不支持Hyper-v,所以只能采用Docker Toolbox的方式使用Docker.传送门:http://mirrors.al ...

  8. java本地缓存

    1.为什么要使用缓存 由于服务器.数据库.网络等资源有限,无法支撑越来越多的请求与计算量,所以将一部分数据放在缓存中,以此减小薄弱环节的计算量和请求流程. 网站中缓存的应用场景:        1:可 ...

  9. PHP程序员-常用工具

    三连问 经常有社区的同学问: “我的PHP程序有没有阻塞,我的PHP程序有没有开启协程(对自己写好的代码表示不自信),我的PHP程序有没有问题”.然后贴出了自己的程序,然后进入了愉快的灌水环节,随着时 ...

  10. 技术人如何利用 github+Jekyll ,搭建一个独立免费的技术博客

    上次有人留言说,技术博客是程序员的标配,但据我所知绝大部分技术同学到现在仍然没有自己的技术博客.原因有很多,有的是懒的写,有的是怕写不好,还有的是一直想憋个大招,幻想做到完美再发出来,结果一直胎死腹中 ...