Tijdens de zomer van 1940 eiste Duitsland toegang tot de Zweedse telefoonkabels om gecodeerde berichten te versturen vanuit het bezette Noorwegen naar het thuisland. Zweden kon niet anders dan toegeven, maar begon meteen de lijnen af te luisteren. Al snel ontdekten ze dat er een nieuw cryptografisch systeem gebruikt werd. De T52 Geheimschreiber (of kortweg G-schreiber) — een machine met meer dan 8 biljard verschillende instellingen — verzond geheime informatie die maar niet gekraakt leek te kunnen worden.

T52
De Siemens & Halske T52 — ook gekend als de Geheimfernschreiber of Schlüsselfernschreibmaschine (SFM) — was een codeermachine die Duitsland tijdens de Tweede Wereldoorlog gebruikte. De machine en zijn gecodeerde berichten kregen van de Britse geheime dienst de codenaam Sturgeon.

De Zweedse inlichtingendiensten gaven wiskundige Arne Beurling opdracht om de code te kraken op basis van slechts een handvol gecodeerde berichten en zonder enige voorkennis over het mechanisme dat gebruikt was bij het coderen. Na twee weken puzzelen met enkel potlood en papier, slaagde Beurling erin de berichten te ontcijferen. Hij kon ook beschrijven hoe ze een complementaire machine konden bouwen om berichten te decoderen. Dankzij deze doorbraak konden Zweedse ambtenaren berichten onderscheppen waarin sprake was van een nakende invasie van de Sovjet-Unie. Helaas sloeg het personeel van Stalin hun waarschuwingen in de wind.

Het boek The Codebreakers1 van Bengt Beckman brengt relaas van het hele verhaal. In zijn voorwoord voor het boek schrijft Peter Jones:

Tot op heden weet niemand welke redenering Beurling maakte tijdens die twee weken dat hij zijn zinnen zette op de G-schreiber. In 1976 werd hij door een afdeling van het Zweedse leger ondervraagd, en raakte daarbij zeer geïrriteerd toen ze hem onder druk zetten om zijn werkmethode uit de doeken te doen. Uiteindelijk antwoordde hij: "Een goochelaar legt toch ook niet uit hoe zijn trucs werken." Het lijkt erop dat de enige aanwijzing die Beurling ooit gaf, verscholen zat in de cryptische (of wat had je gedacht) opmerking dat drieën en vijven belangrijk waren.

Opgave

Uiteindelijk bleek de procedure die de T52 Geheimschreiber gebruikte helemaal niet zo ingewikkeld. Er wordt gewerkt met een alfabet van $$m$$ symbolen die allemaal verschillend zijn en een vaste volgorde hebben. De posities van de symbolen in het alfabet worden genummerd van 0 tot en met $$m - 1$$.

De sleutel die gebruikt wordt voor het coderen en decoderen bestaat verder uit twee natuurlijke getallen $$a$$ en $$b$$. Elk symbool op positie $$x$$ in het alfabet wordt gecodeerd als het symbool op positie $$C(x)$$ in het alfabet, waarbij \[C(x) = (ax + b)\ \textrm{mod}\ m \qquad (0 \leq x < m)\] en de operator $$\textrm{mod}$$ geeft de rest na gehele deling. Elk symbool op positie $$x$$ in het alfabet wordt gedecodeerd als het symbool op positie $$D(x)$$ in het alfabet, waarbij \[D(x) = a'(x - b)\ \textrm{mod}\ m \qquad (0 \leq x < m)\] en $$a'$$ staat voor de multiplicatieve inverse van $$a$$ modulo $$m$$. Met andere woorden, $$a'$$ is het natuurlijk getal uit het interval $$[0, m[$$ waarvoor geldt dat \[1 = aa'\ \textrm{mod}\ m\] De multiplicatieve inverse van $$a$$ bestaat enkel als $$a$$ en $$m$$ copriem zijn. Dat is het geval als de grootste gemene deler van $$a$$ en $$m$$ gelijk is aan 1. In dat geval bestaat er ook maar één getal in het interval $$[0, m[$$ dat voldoet aan de voorwaarde van de multiplicatieve inverse.

Toen de Duitsers hoogte begonnen te krijgen van het feit dat de gecodeerde berichten van de T52 makkelijk te kraken waren, probeerden ze die nog complexer te maken. Ze deden dat door een bericht niet één maar twee keer te coderen. Een eerste keer door een T52 met sleutels $$a_1$$ en $$b_1$$ en een tweede keer door een T52 met sleutels $$a_2$$ en $$b_2$$. Dit kan enkel als de twee machines hetzelfde alfabet gebruiken. Uit de vaststelling dat \[C_2(C_1(x)) = (a_2((a_1x + b_1)\ \textrm{mod}\ m) + b_2)\ \textrm{mod}\ m = (a_1a_2x + (a_2b_1 + b_2))\ \textrm{mod}\ m\] volgt echter dat dit dezelfde codering is als deze die gemaakt wordt door één enkele T52 met sleutels $$a_1a_2$$ en $$(a_2b_1 + b_2)$$.

Definieer een klasse T52 waarmee we machines kunnen voorstellen die berichten vercijferen volgens de Geheimschreiber codering. Bij het aanmaken van een nieuwe machine (T52) moeten drie argumenten doorgegeven worden: twee natuurlijke getallen $$a$$ en $$b$$ (Number) en een alfabet (String) met $$m$$ karakters. Als $$a$$ en $$m$$ niet copriem zijn, dan moet een Error opgeworpen worden met de boodschap a en m zijn niet copriem, waarbij de cursieve fragmenten moeten ingevuld worden met de waarden van $$a$$ en $$m$$. Als niet alle karakters van het alfabet verschillend zijn, dan moet een Error opgeworpen worden met de boodschap alfabet bevat herhaalde symbolen.

Grootste gemene deler

Je kunt deze JavaScript implementatie gebruiken om de grootste gemene deler (ggd) van twee natuurlijke getallen $$a$$ en $$b$$ te berekenen:

const ggd = (a, b) => b === 0 ? Math.abs(a) : ggd(b, a % b);

Op een machine $$m$$ (T52) moet je minstens de volgende methoden kunnen aanroepen:

Definieer ook nog een functie combineer waaraan twee machines $$m_1$$ en $$m_2$$ (T52) moeten doorgegeven worden. Als machines $$m_1$$ en $$m_2$$ niet hetzelfde alfabet gebruiken, dan moet de functie een Error opwerpen met de boodschap alfabetten zijn verschillend. Anders moet de functie een nieuwe machine (T52) teruggeven die werkt alsof we eerst berichten zouden coderen met machine $$m_1$$ en het resultaat daarna nog eens zouden coderen met machine $$m_2$$.

Voorbeeld

> const machine1 = new T52(3, 5, "ABCDEFGHIJKLMNOPQRSTUVWXYZ")

> machine1.codeerSymbool("G")
"X"
> machine1.codeerSymbool("S")
"H"
> machine1.codeerSymbool("-")
"-"

> machine1.decodeerSymbool("X")
"G"
> machine1.decodeerSymbool("H")
"S"
> machine1.decodeerSymbool("-")
"-"

> machine1.codeer("G-SCHREIBER")
"X-HLAERDIRE"
> machine1.decodeer("X-HLAERDIRE")
"G-SCHREIBER"

> const machine2 = new T52(17, 11, "ABCDEFGHIJKLMNOPQRSTUVWXYZ")
> machine2.codeer("X-HLAERDIRE")
"M-AQLBOKROB"
> machine2.decodeer("M-AQLBOKROB")
"X-HLAERDIRE"

> const machine12 = combineer(machine1, machine2)
> machine12.codeer("G-SCHREIBER")
"M-AQLBOKROB"
> machine12.decodeer("M-AQLBOKROB")
"G-SCHREIBER"

> new T52(4, 5, "ABCDEFGHIJKLMNOPQRSTUVWXYZ")
Error: 4 en 26 zijn niet copriem

> new T52(3, 5, "ABCDEFGHIJKLMMLKJIHGFEDCBA")
Error: alfabet bevat herhaalde symbolen

> combineer(machine1, new T52(17, 11, "abcdefghijklmnopqrstuvwxyz"))
Error: alfabetten zijn verschillend