We see that c is checked against kn, and they must be the same in order for our password to be correct.
random.seed(997)k = [random.randint(0, 256)for _ inrange(len(x))]a ={ b:do_thing(ord(c), d)for (b, c), d inzip(enumerate(x), k)}b =list(range(len(x)))random.shuffle(b)c = [a[i]for i in b[::-1]]
This part is a little confusing. The first thing to notice is that the RNG is seeded, so the values of k and b are always the same.
Since we know the value that c must be, and the value of b after random.shuffle() is known, we can recover a.
c = knprint("Need c =", c)a = [Nonefor _ inrange(len(b[::-1]))]for i inrange(len(b[::-1])): a[b[::-1][i]]= c[i]print("Need a =", a)
Now, we need to work out what the value of x must be. Notice that every character in x is passed through the do_thing() function, with the corresponding value in k.
a ={ b:do_thing(ord(c), d)for (b, c), d inzip(enumerate(x), k)}
What we need to do is to recover the value of each character in x, knowing the corresponding values in k. To do that, we need to understand the do_thing() function.
defdo_thing(a,b):return ((a <<1) & b) ^ ((a <<1) | b)
So this operation flips every bit in (a << 1), where the corresponding bit in b is 1. This is the same as (a << 1) ^ b.
Hence, to undo this operation and recover the flag, we simply perform the following:
defundo_thing(a,b):return (a ^ b) >>1
Here's the full solver script to obtain the password.
defundo_thing(a,b):return (a ^ b) >>1x ='a'*25random.seed(997)k = [random.randint(0, 256)for _ inrange(len(x))]print("k =", k)a ={ b:do_thing(ord(c), d)for (b, c), d inzip(enumerate(x), k)}b =list(range(len(x)))random.shuffle(b)c = [a[i]for i in b[::-1]]kn = [47, 123, 113, 232, 118, 98, 183, 183, 77, 64, 218, 223, 232, 82, 16, 72, 68, 191, 54, 116, 38, 151, 174, 234, 127]
valid =len(list(filter(lambdas: kn[s[0]] == s[1], enumerate(c))))# i.e. c = knprint("---")c = knprint("Need c =", c)a = [Nonefor _ inrange(len(b[::-1]))]for i inrange(len(b[::-1])): a[b[::-1][i]]= c[i]print("Need a =", a)undo_a ={ b:chr(undo_thing(a[b], d))for (b, c), d inzip(enumerate(x), k)}print(''.join(undo_a[i] for i inrange(25)))