第二届SWCTF部分WP2

本文涉及的产品
密钥管理服务KMS,1000个密钥,100个凭据,1个月
简介: 第二届SWCTF部分WP

还是用Python3来运行脚本


在开头看到 flag{<CAP>g<CAP>00d_jOb_cUre_boy!}


但是去提交时却发现是错的


这里简单说一下:


在计算机界CAP其实是指的是一种理论,包括一致性、可用性、分区容错性。


当然我们其实不用知道这些什么理论,我们只需要知道这里<CAP>是什么意思


这里cap的作用是标志


比如


html语言标签hl用于标记


<hl>test<hl> 则表示test是一级标题,用hl把 test 标记为一级标题


那么<CAP>g<CAP>呢?


看一下你键盘上的CAP键,它是切换大小写的


所以这里表示把 g 标记为大写

当我们继续查看后面的output时,发现真正的flag,和我们前面说的标记后的结果一样

故正确的flag为

flag{G00d_jOb_cUre_boy!}

2、crypto

(1)Crypto1

这道题题目给了一个ip  nc 43.143.14.222 12345

在kali中直接连接,给了e、n、c,求m

然后去查了一些相关东西,发现是RSA算法,RSA算法常用于非对称加密

最开始我搞错了类型,用的是已知 p ,q,e  求 d

我直接将n和c去替换p和q的值,把d当做输出的m

#coding=utf-8
import gmpy2
p = 186048964179157130117126822461827656144
q = 205819884276150158370426802196552944211
e = 65537
d = gmpy2.invert(e,(p-1)*(q-1))  # gmpy2.invert(e,φ(N))
print (d)

当然这样跑出来的结果对这道题来说肯定是不对的

然后我还尝试把p和q的数据调换位置但是还是不行,因为类型都搞错了

这种加密有很多种类型,通常是知道其中的几个去求另外的几个

比如 已知e,dp, n, c,求m,m表示明文,c表示密文(但是这里并没有给dp)


后面又找到一个 已知(e,n,c),求m。(低解密指数攻击)

import gmpy2
import binascii
import RSAwienerHacker
e = 284100478693161642327695712452505468891794410301906465434604643365855064101922252698327584524956955373553355814138784402605517536436009073372339264422522610010012877243630454889127160056358637599704871937659443985644871453345576728414422489075791739731547285138648307770775155312545928721094602949588237119345
n = 468459887279781789188886188573017406548524570309663876064881031936564733341508945283407498306248145591559137207097347130203582813352382018491852922849186827279111555223982032271701972642438224730082216672110316142528108239708171781850491578433309964093293907697072741538649347894863899103340030347858867705231
c = 350429162418561525458539070186062788413426454598897326594935655762503536409897624028778814302849485850451243934994919418665502401195173255808119461832488053305530748068788500746791135053620550583421369214031040191188956888321397450005528879987036183922578645840167009612661903399312419253694928377398939392827
d = RSAwienerHacker.hack_RSA(e,n)
m = gmpy2.powmod(c,d,n)
print(binascii.unhexlify(hex(m)[2:]))

但是我在装RSAwienerHacker时遇到了问题

于是只能寻找其他类型,后面又发现了一个

已知p、q、e、密文c,求明文m,这里我们可以把这个n拆分成p和q

使用网站对n进行质因数分解

这里要注意,由于我们每次连接拿到的数据都不一样,即n不一定能分解成p和q,如下图

所以,如果拿到的n不能进行分解就多换几次,换到一个能分解的数据

通过脚本传入c、e、p、q的值

import gmpy2 
import binascii
c = 202081350231076713865040280238183184825
e = 65537
p = 13326601478294332001
q = 15920747257748120237
# 计算私钥 d
phi = (p-1)*(q-1)
d = gmpy2.invert(e, phi)
# 解密 m
m = gmpy2.powmod(c,d,p*q)
print(binascii.unhexlify(hex(m)[2:]))

这里的脚本输出的是hex十六进制

我们直接 print(m)就可以得到十进制的m值了

提交m的值

拿到flag: snert{This_just_a_easy_RSA}

(2)Crypto2

这道题也是给了一个ip,我们直接访问

得到一串base64字符串

MjZlMjFlOGZkOGVkZGExZWVkZGExZWVkZGExZWVkZWUxZWJmZWM4NGVlNGRiMjBhMWU=

根据提示 /code查看加密代码

这道题最重要的是看懂代码


首先介绍一些知识


① 导入模块或者函数


import 模块:导入一个模块;


相当于导入的是一个文件夹,是个相对路径。


from…import:导入了一个模块中的一个函数;


相当于导入的是一个文件夹中的文件,是个绝对路径。


from…import *:把一个模块中所有函数都导入进来,相当于导入的是一个文件夹中所有文件,所有函数都是绝对路径。


② 定义一个由自己想要功能的函数


规则:


  • 函数代码块以 def 关键词开头,后接函数标识符名称和圆括号()。
  • 任何传入参数和自变量必须放在圆括号中间。圆括号之间可以用于定义参数。
  • 函数的第一行语句可以选择性地使用文档字符串—用于存放函数说明。
  • 函数内容以冒号起始,并且缩进。
  • return [表达式] 结束函数,选择性地返回一个值给调用方,不带表达式的return相当于返回 None。

eg:


def functionname( parameters ):


"函数_文档字符串"


function_suite


return [expression]


③ hex() 函数


用于将10进制整数转换成16进制,以字符串形式表示。


hex 语法:hex(x)

eg:

>>> hex(255)
'0xff'
>>> hex(-42)
'-0x2a'
>>> hex(12)
'0xc'
>>> type(hex(12))
<class 'str'>  //字符串

④ ord() 函数

ord() 函数是 chr() 函数(对于8位的ASCII字符串)或 unichr() 函数(对于Unicode对象)的配对函数,它以一个字符(长度为1的字符串)作为参数,返回对应的 ASCII 数值,或者 Unicode 数值,如果所给的 Unicode 字符超出了你的 Python 定义范围,则会引发一个 TypeError 的异常。


语法: ord(c)


其中c表示字符,返回对应的 ASCII 数值

eg:

>>> ord('a')
97
>>> ord('b')
98
>>> ord('c')
99

⑤ Python切片(这个真的很重要)

在Python中,切片(slice)是对序列型对象(如list, string, tuple)的一种高级索引方法。普通索引只取出序列中一个下标对应的元素,而切片取出序列中一个范围对应的元素,这里的范围不是狭义上的连续片段。


用法:object[start_index : end_index : step]


前闭后开

参数说明:


start_index:切片的起始位置(包括该位置)


缺省时取0或-1(即step为正数取0,负数取-1)


end_index:切片的结束位置(不包括该位置)


缺省时默认为序列长度(step为正数取正,step负数取负)


step:表示步长。可取正负数,正数表示从左往右,负数表示从右往左。


缺省时默认为1

eg:

>>> a = list(range(10))
>>> a
[0, 1, 2, 3, 4, 5, 6, 7, 8, 9]
>>> a[:5]
[0, 1, 2, 3, 4]
>>> a[5:]
[5, 6, 7, 8, 9]
>>> a[2:8]
[2, 3, 4, 5, 6, 7]
>>> a[::2]
[0, 2, 4, 6, 8]
>>> a[::-1]
[9, 8, 7, 6, 5, 4, 3, 2, 1, 0]

⑥ encode()和decode()函数


以 encoding 指定的编码格式编码和解码字符串。errors参数可以指定不同的错误处理方案。


语法:str.encode(encoding='UTF-8',errors='strict')  //编码


          str.decode(encoding='UTF-8',errors='strict')  //解码

⑦ 点“.”怎么理解


其实可以把点前面的内容整体看成字符串,点后面的内容表示要执行的操作


比如上面的 str.encode()表示对str进行编码操作


了解完这些后我们回到这道题


这道题我最开始做时搞错了顺序,我用给的base64编码在那里顺推,后面发现不对


再次去看题目,我们/code查看的是加密后的代码,所以这道题要逆向推回去


我是纯手工推的


先看最后一句代码


return b64encode(code.encode()).decode()


最后返回的结果是对前面这个整体内容进行了一个decode()解码操作


而我们是逆推,所以需要先对它进行encode()编码,默认是UTF-8

但是我们可以发现无论是编码还是解码,内容都没有变化

逆推到第二步,加密代码是对我下面标出来的这个整体进行了b64encode()编码


所以这里我们需要对它进行base64解码

得到 26e21e8fd8edda1eedda1eedda1eedee1ebfec84ee4db20a1e

下一步,加密代码是对code进行了encode()

所以我们需要对上面得到的字符串进行decode()解码操作

和前面一样,也没有变化

得到的这个东西就是倒数第二句代码print出的code


即 code=26e21e8fd8edda1eedda1eedda1eedee1ebfec84ee4db20a1e


再看上一句代码


code=code[::-1]


根据前面说的Python切片,这里表示是对code取了一个逆序


我们是逆推,所以对code也取一次逆序即可


我直接用的网站转反向文字

当然我们也可以通过Python代码来实现

 得到 e1a02bd4ee48cefbe1eedee1addee1addee1adde8df8e12e62

继续看上一句代码

code+=hex((ord(i)*a+b)%n)[2:]

始终记住我们是逆推,所以这里要从外往里看

最后的操作是对我框出来的这个整体进行了一个切片处理

[2:]  start_index=2表示从原始序列中的第三个元素开始取(因为0对应的是第一个元素)


end_index省略了,则表示取到最后


step也省略了,取默认值为1


切片操作后得到code=e1a02bd4ee48cefbe1eedee1addee1addee1adde8df8e12e62


也就是说,在我们前面得到的code值前面理论上应该是还有两个被切掉了的元素,而且是未知的


到这里时我就开始犹豫了,因为存在两个未知的元素,继续逆推出来的结果会不会有问题


但是目前暂时没有其它思路和方向,只能继续逆推


是一个hex()函数,即取十六进制


于是我尝试给它转换回十进制


但是这里又有一个问题,因为code前面本来就还有两个未知的元素存在


这样转换出来的结果多半是会有问题的

继续看里面的代码

前面我们说了ord()函数是取字符的ASCII码值,我们暂且给它看成一个整体


即把我框起来的东西看成一个未知数x,而a、b、n的值都是给了的,直接代入


假设 ord(i) = x


则 (x*13+15)%257 = code (注意这里的code表示的是code整个字符串中的一个结果而已)


因为从前面的代码可以看出,code最开始里面是空的


后面 code+ = XXX ( code+ = XXX 其实就是 code = code+XXX )



这里问题又来了,(x*13+15)%257 = code 最后得到的结果是除以257后的余数


但是257的余数可以是一位数,可以是二位数,也可以是三位数,这里我们该怎么取呢?


前面转十进制我们得到的数据是


1416275277295255838555349926665087454400284814091576735247970


比如我们可以取1,也可以取14,取141,都比257小,都可以作为257的余数,那到底是哪个?


这里的一个整体思路就是取出一个结果,带入上面的等式,去推x的值


再将x的值转换为对应的ASCII字符


而且我们不知道商是多少,但是可以肯定商是整数


也就是说这里现在存在了两个不确定:


一是code前面存在两个被切掉的元素未知


二是取余数也存在多种情况,到底应该怎么取


我去尝试推了几组数据但是都不行,没有找到一个整数的x


后面我去研究它那个十六进制转十进制


发现每次放进去的数据位数不同时也会导致结果不同


即整体放入和单独或者分批次放入转换出来的结果是不一样的


比如e是14

e1是225

e1a则是3610

我们至少应该统一一个标准,因为code的最终结果是由每一次这样的结果拼接起来的

我是以两个十六进制数为一组,单独对每组十六进制数进行十进制转换

比如e1为第一组,转十进制得到225

接着是a0,得到160,后面以此类推

将这些转换出来的数一个个代入余数 ,去推x的值


即(x*13+15)%257 = 商(整数)......余数


因为商是一个未知的整数,于是从0开始依次增大往后取进行尝试


只要推出来的x也是整数即可


将225代入余数,当我们的商取到5时,发现此时的x为整数


即 x = (257*5+225-15)/13 = 115


推出的 x 就是 i 的ASCII值,由前面代码我们知道 i 是在text里面取的

我们将这些数转换成对应的ASCII字符,拼接起来就可以得到text

比如第一个115对应ASCII字符是 s

依次按照上面方法往后面推,当我推出前四个后越来越激动,snert!!!

第五个推出来是 {

多半是对了

于是继续往后推

最终得到flag

snert{Just_so_so_so_Esay}

一定要好好学Python啊,不然只能像我这样纯手工推

目录
相关文章
|
5月前
日更[ACTF新生赛2020]Oruga wp
日更[ACTF新生赛2020]Oruga wp
42 0
|
7月前
|
SQL 安全 数据库
技术心得:实验室每日一题WP
技术心得:实验室每日一题WP
34 0
|
8月前
|
网络安全
2024年---蓝桥杯网络安全赛道部分WP
2024年---蓝桥杯网络安全赛道部分WP
|
SQL 安全 PHP
第二届SWCTF部分WP3
第二届SWCTF部分WP
91 0
|
Python
第二届SWCTF部分WP1
第二届SWCTF部分WP
91 0
|
JSON 安全 Android开发
渔人杯部分wp
渔人杯部分wp
125 0
|
XML 数据格式
bugku-眼见非实-wp
bugku-眼见非实-wp
|
PHP 开发者
【第二届PHP全球开发者大会】惠新宸(鸟哥):PHP7性能之源
5月14-15日的第二届2016PHP全球开发者大会在北京国际会议中心举行。国内最有影响力的PHP技术专家、新浪微博任平台及数据部总架构师兼首席PHP顾问惠新宸(“鸟哥”)做了题为《PHP7性能之源》的演讲,通过与PHP5中的检查机制、JIT编译器、WP等相对比,详解PHP7的性能提升的奥秘。
4876 0
|
Web App开发 移动开发 开发者
微软CEO鲍尔默力推HTML5:称其为平台的粘合剂
据国外媒体报道,微软CEO史蒂夫·鲍尔默(Steve Ballmer)上周参加微软“专业开发者大会”(Professional Developers Conference)时,将Windows Azure、HTML 5和Windows Phone 7作为开发者平台推广。
873 0
|
测试技术 Shell PHP
MOCTF新春欢乐赛部分WP
已经过去有段时间了,比赛在年前,现在才有时间整理一下当时做出的一些题目。 当作是一个复习梳理了 比赛地址:http://happy.moctf.com/ 官方所有WP:https://github.
1252 0