This task was very strange. The author clearily didn't understand quantum key exchange at all, and in this particular case knowing how this should work was making the task harder to solve...
In the task we get some intercepted quantum transfers sat->ground, ground->sat, sat->ground and a ciphertext in public channel:
269118188444e7af980a245aedce5fb2811b560ccfc5db8e41f102a23f8d595ffde84cb1b3f7af8efd7a919bd2a7e6d3
In general quantum key exchange is quite simple:
- Alice sends set of random bits encoded in one of two bases (let's call them X and Y)
- Bob receives the qbits and randomly selects which base to use to measure value (since he can't know which base Alice used for each of qbits). Statistically he will get about half of the bases right.
- Bob sends to Alice information which bases he used for each of the qbits.
- Alice compares the list from Bob with the bases she actually used and sends to Bob information which bases he got right.
In case Bob used the right base he knows the bit value Alice wanted to send. Those agreed bits are then normally used as some symmetric key for the communication.
We have our 3 transmissions, and they are exactly what described above:
- First one is the whole dump of initial qbits sent by Alice. We have two bases -> straight (
-
or|
) and cross (/
or\
). - Second one is bases Bob used - straight
+
or crossx
. - Last one is the list of bits Bob got right marked as
v
.
So we can simply read values sent by Alice in 1st transmission for which there is v
in last tramissions and we will get the qbits that they got right. We don't really know if \
is 0 or 1 and if |
is 0 or 1, but this is simple enough, we can just test all 4 options.
def get_agreed_bytes(data_sent, bases_measured, bases_correct, v1, v2):
agreed_bits = []
for i in range(len(bases_correct)):
if bases_correct[i] == 'v':
if bases_measured[i] == '+':
if data_sent[i] == '-':
agreed_bits.append(v1)
else:
agreed_bits.append(abs(v1 - 1))
else:
if data_sent[i] == '/':
agreed_bits.append(v2)
else:
agreed_bits.append((abs(v2 - 1)))
return hex(int("".join([str(c) for c in agreed_bits]), 2))[2:-1]
Now it was strange, because we get some bits and we don't know how to use them as encryption key to recover the flag. Fortunately we noticed that byte values seem in printable range and I printed out the agreed bytes and one of the options was:
iv:281e6bfc14a9aad39845f29b30ef1334,key:b340fe5025b06657034822b340ceb9d4,algo:aes_cbc
Which makes little sens in terms of crypto channel key exchange, because Alice can't know which bits Bob will get, but whatever...
So we have all we need -> algo and parameters and we can just decrypt the flag: h4ck1t{tr4nsmi55i0n_0v3r_bb84_l00ks_s3cur3_0k}
Complete solver here
To było dość dziwne zadanie. Autor ewidentnie nie rozumie o co chodzi w kwantowej wymianie klucza i paradoksalnie w tym zadaniu znajomość tego zagadnienia tylko utrudniała rozwiązanie zadania...
W zadaniu dostajemy kilka przechwyconych kwantowych transmisji sat->ground, ground->sat, sat->ground i szyfrogram w publicznym kanale:
269118188444e7af980a245aedce5fb2811b560ccfc5db8e41f102a23f8d595ffde84cb1b3f7af8efd7a919bd2a7e6d3
Generalnie kwantowa wymiana klucza jest dość prosta:
- Alice wysyła do Boba losowe bity kodowane w jednej z dwóch baz (nazwijmy je X i Y)
- Bob odbiera qbity i wybiera losowo w jakiej bazie odczytać wartość (ponieważ nie wie w jakiej bazie wartość została zakodowana przez Alice). Statystycznie powinien odczytać około połowy poprawnie.
- Bob wysyła do Alice informacje których baz użył dla każdego z qbitów.
- Alice porównuje listę od Boba ze swoją listą z bazami których użyła i wysyła do Boba informacje, które bazy wybrał poprawnie.
Jeśli Bob użył dobrej bazy dla danego bitu to zna wartość którą Alice chciała wysłać. Bity, które się zgodziły są zwykle używane do generacji klucza dla kryptografii symetrycznej.
Mamy 3 tranmisje danych i są dokładnie tym co opisane wyżej:
- Pierwsza to zrzut qitów które Alice wysłała. Mamy tam dwie bazy -> prostą (
-
i|
) oraz skośną (/
i\
). - Druga to lista baz których użył Bob - prosta
+
lub skośnax
. - Ostatnia to lista bitów które Bob odczytał dobrze, oznaczonych przez
v
.
Możemy więc po prostu odczytać wartości które Alice wysłała w 1 transmisji, dla których w trzeciej transmisji mamy v
i w ten sposób poznamy listę qbitów które się zgodziły.
Nie wiemy co prawda czy \
to 1 czy 0 oraz czy |
to 1 czy 0, ale możemy przetestować wszystkie 4 opcje.
def get_agreed_bytes(data_sent, bases_measured, bases_correct, v1, v2):
agreed_bits = []
for i in range(len(bases_correct)):
if bases_correct[i] == 'v':
if bases_measured[i] == '+':
if data_sent[i] == '-':
agreed_bits.append(v1)
else:
agreed_bits.append(abs(v1 - 1))
else:
if data_sent[i] == '/':
agreed_bits.append(v2)
else:
agreed_bits.append((abs(v2 - 1)))
return hex(int("".join([str(c) for c in agreed_bits]), 2))[2:-1]
Teraz następuje dość dziwny krok, bo mamy zgodne bity, ale nie wiemy co dalej z nimi zrobić żeby odzyskać flagę. Szczęślwie zauważyliśmy, że wartości bajtów wyglądają na drukowalne ascii więc wypisaliśmy sobie uzyskane możliwości i dostaliśmy dla jednego z nich:
iv:281e6bfc14a9aad39845f29b30ef1334,key:b340fe5025b06657034822b340ceb9d4,algo:aes_cbc
Co zupełnie nie ma sensu z punktu widzenia kwantowej wymiany klucza, bo Alice nie mogła wiedzieć które bity Bob odzyta poprawnie, no ale co tam...
Mamy podany algorytm i prametry więc odszyfrowujemy flagę: h4ck1t{tr4nsmi55i0n_0v3r_bb84_l00ks_s3cur3_0k}
Cały solver tutaj