Home Master Challenge - De1 CTF
Post
Cancel

Master Challenge - De1 CTF

Overview

This was one of the most interesting RSA challenges that I have solve till date. There are numerous points about this challenge that makes it so intriguing and unusual one of them being the hint (I will come to that later :p)
One thing you would have noticed is how it seems that most of variables which have been used in the challenge script are not even acessible to the solver.

You would also notice that there are multiple ciphertexts, modlus, public exponents declared (some which we dont even know the value of).

Lets list them out in the order we’ll be finding them in:

  1. P
  2. Public exponenets e1 and e2
  3. Prime factors of n (mentioned as n12 in the data file):
    1. q1p
    2. q1q
  4. The hint
  5. The flag

All these variables might make this challenge look pretty baffling on the first glance but the key is solving it step by step getting each of the variable one(or two) at a time.

So lets get those variables one at a time………

The Approach

#1. Retrieving p

This one was quite intuitive especially since the ciphertexts and modulus have been give in lists the lengths of which corresponds to 4. This is just a direct implementation of Hastards broadcast attack

1
2
  f=lambda m,e,n,c:pow(m,e,n)==c
  assert(sum(map(f,[p]*4,[4]*4,n,c))==4)

The above code basically checks if p4 modn == c for all 4 values of n and c(taken from the lists)

Moving on to the next unknwon………

#2. Retrieving e1 and e2

Here is the part where we maybe slightly misled by the following lines of code:

1
2
3
4
ee1 = 42
ee2 = 3
assert(pow(e1,ee1,n)==ce1)
assert(pow(e2+tmp,ee2,n)==ce2)

At first glance it looks like we can retrive e2+tmp easily since the exponent ee2=3 is small but after implementation we see that e2+tmp is pretty big and is not susceptible to small exponent attack. By thinking of small exponent attack we are on the right idea but the wrong path since this attack which is not working for ee2 actually works for ee1=42. A direct implementation gets us the value of e1.

Going back to e2; since e2+tmp is really big there is no way we can directly bruteforce tha value of e2 but again we are on the right idea but wrong way of implementation as in, instead of bruteforcing for the value directly we bruteforce to find the actul value of the ciphertext i.e the unmoded value which would be this ce2+k*n11 by iterting for different values of k and matching this with (e2+tmp)3 we canget e2. Following is the implementation of the idea:

1
2
3
4
 for k in range(100000):
	val = root(ce2+k*n11,3)
	if(val[1]):
		e2 = val[0] - tmp

Moving on to stage 3…………

#3 Retrieving q1p and q1q

For this there doesn’t seem to be any way in which we can exploit the implementation to obtain the cipher text with out q1p and q1q and they are the prime factors of the modulus and to obtain them the only way is factorization Since n is too long for normal factorization we use fermats method and we immediately get both the factors.

Thereafter finding hint is just elementry RSA operation. The hint as it turns out to be: orz...you.found.me.but.sorry.no.hint...keep.on.and.enjoy.it! is pretty useless :) , still we got the factors with which we proceed to the last and final stage…………

#4 Retrieving the flag

This is arguably the most difficult value to retrieve amongst all and with good reason. All though it might look like its childs play to obin the flag as we know both the fctors of the modulus p and q1 when we try decrypting we realize the flaw that is GCD(e1,(p-1)*(q1-1)) turns out to be 14 if we try with e2 we get the same result.

Moment I saw this my first reaction was dividing e2 ( or e1) by 14, this gives us a GCD of 1. good! Now we find the private exponent d.

NOTE: we are finding this for e/14

now decrypting with this d what we get is flage1*dmod p*q1

But since this d was the modular inverse of (e1/14) and not e1 what we are left with is:

-> flag14(e1/14)dmod p*q1

-> flag14mod p*q1 as (e2/14)*d==1

Now if take the 14th root of flag we should end up with the pliantext Doing that we get flag and then we convert it to bytes and what we get is…. well its gibberish!! Meaning flag14 is bigger than p*q1

That means thats not the right way of doing ( again we are on the right idea but wrong track :D ) so if flag power 14 is too big how about reducing the power…………

Now lets take only q1 as the modulus. The idea here is that if q1 is less than p*q1 we may be able to retrieve the flag so here’s what we do:

-> c_modq = c_flag mod q1 (c_flag referenced as c1 in challenge script)

this makes some changes which we will see

-> c_flag = flage1mod p*q1

-> c_modq = (flage1mod p*q1)mod q1

-> c_modq = (flage1mod q1)mod p*q1

Now gcd(e2,(q2-1))=2 so again we divide e2/2 and find inverse(e2/2,(q2-1)) and we get d ( for e2/2 )

-> c_modqd = (flag2(e2/2)dmod q2)mod p*q2

Hence we get (flag2mod q1)mod p*q1 since q1<p*q1 we can ommit mod p*q1

-> c_modqd = flag2(e2/2)dmod q1 now assuming flag2<q1 we can get flag by taking root

For all this we haven’t taken into consideration q2 as it is not greater than flag2 hence will not yield the flag when we attempt to take square root

this is the exploit script for the idea:

1
2
3
4
5
6
q1 = q1p
c1_modq = c_flag % q1
GCD1 = gcd(e1,q1-1)
d1 = invert(e1/GCD1,q1-1)
c1_modq = pow(c1_modq,d1,q1)
flag = root(c1_modq,2)[0]

In the end we get decrypted flag as: de1ctf{9b10a98b-71bb-4bdf-a6ff-f319943de21f}

Here is the complete exploit

This post is licensed under CC BY 4.0 by the author.