| Size: 38689 Comment:  | Size: 66454 Comment: py3 print | 
| Deletions are marked like this. | Additions are marked like this. | 
| Line 5: | Line 5: | 
| This page was first created at Sage Days 103, 7-9 August 2019 by Sarah Arpin, Catalina Camacho-Navarro, Holly Paige Chaos, Amy Feaver, Eva Goedhart, Rebecca Lauren Miller, Alexis Newton, and Nandita Sahajpal. Text edited by Holly Paige Chaos, Amy Feaver, Eva Goedhart, and Alexis Newton. This project was led by Amy Feaver. | This page was first created at Sage Days 103, 7-9 August 2019 by Sarah Arpin, Catalina Camacho-Navarro, Holly Paige Chaos, Amy Feaver, Eva Goedhart, Sara Lapan, Rebecca Lauren Miller, Alexis Newton, and Nandita Sahajpal. Text edited by Holly Paige Chaos, Amy Feaver, Eva Goedhart, and Alexis Newton. This project was led by Amy Feaver and Eva Goedhart. | 
| Line 29: | Line 29: | 
| print "Put your message inside the provided quotes (with no additional quotes or apostrophes!), and select your desired shift: " @interact def shift_cipher(message = input_box(default='"secrets"', width = 50), shift=slider(0,25,1,3)): | pretty_print(html("<h>Put your message inside the provided quotes (with no additional quotes or apostrophes!), and select your desired shift:<h>")) @interact def shift_cipher(message = input_box(default='"secrets"', label="Message:"), shift=slider(0,25,1,3, label="Shift by:")): | 
| Line 36: | Line 36: | 
| print "This is your encrypted text shifted by ",shift,":" print C | print("This is your encrypted text shifted by",shift,":") print(C) | 
| Line 50: | Line 50: | 
| print "Enter the encrypted text in quotes, and enter a guess for the shift amount:" @interact def shift_decrypt(text = input_box('"KL"'), shift_by = input_box(0)): | pretty_print(html("<h>Enter the encrypted text in quotes, and enter a guess for the shift amount:<h>")) @interact def shift_decrypt(text = input_box(default='"KL"',label="Message:"), shift_by = input_box(default = 0, label="Shift by:")): | 
| Line 56: | Line 56: | 
| print "If the shift was by", shift_by,", then the original message was:" print decrypt | print("If the shift was by", shift_by,", then the original message was:") print(decrypt) | 
| Line 59: | Line 59: | 
| print "These are the possibilities for the plaintext:" print decrypt | print("\nThese are the possibilities for the plaintext:") print(decrypt) | 
| Line 62: | Line 62: | 
| print "These are the possibilities ranked by likelihood with the chi-squared function:" print decrypt | print("\nThese are the possibilities ranked by likelihood with the chi-squared function:") print(decrypt) | 
| Line 80: | Line 80: | 
| print "Put your message in between the provided quotes (with no additional quotes or apostrophes!), and select your desired a and b: " @interact def affine_cipher(message = input_box(default='"secrets"', width = 50), a=[1,3,5,7,9,11,15,17,19,21,23], b =[0..25]): | pretty_print(html("<h>Put your message in between the provided quotes (with no additional quotes or apostrophes!), and select your desired a and b:<h>")) @interact def affine_cipher(message = input_box(default='"secrets"', label="Message:"), a=[1,3,5,7,9,11,15,17,19,21,23], b =[0..25]): | 
| Line 87: | Line 87: | 
| print "This is your encrypted text:" print C | print("This is your encrypted text:") print(C) | 
| Line 100: | Line 100: | 
| print "Enter the encrypted text in quotes, and enter a guess for the a and b:" @interact def shift_decrypt(text = input_box('"XNSILPCVA"'), a=[1,3,5,7,9,11,15,17,19,21,23,25], b =[0..25]): | pretty_print(html("<h>Enter the encrypted text in quotes, and enter a guess for the a and b:<h>")) @interact def shift_decrypt(text = input_box('"XNSILPCVA"', label="Message:"), a=[1,3,5,7,9,11,15,17,19,21,23,25], b =[0..25]): | 
| Line 106: | Line 106: | 
| print "If the a =", a, "and the b =",b, ", then the original message was:" print decrypt | print("If the a =", a, "and the b =",b, ", then the original message was:") print(decrypt) | 
| Line 109: | Line 109: | 
| print "\nThese are the possibilities for the plaintext:" print decrypt | print("\nThese are the possibilities for the plaintext:") print(decrypt) | 
| Line 112: | Line 112: | 
| print "\nThese are the top 10 possibilities ranked by likelihood with the chi-squared function:" print decrypt | print("\nThese are the top 10 possibilities ranked by likelihood with the chi-squared function:") print(decrypt) | 
| Line 123: | Line 123: | 
| pretty_print(html('<h1> Substitution Cipher')) pretty_print(html("<h>Select your letter substitutions and enter your message inside the quotes.<h>")) from string import ascii_uppercase left_over_letters=[0] +[let for let in ascii_uppercase] @interact def _(A =selector(left_over_letters, default=0)): if A!=0: left_over_letters.remove(A) # print left_over_letters @interact def _(B =selector(left_over_letters, default=0)): if B!=0: left_over_letters.remove(B) # print left_over_letters @interact def _(C =selector(left_over_letters, default=0)): if C!=0: left_over_letters.remove(C) @interact def _(D =selector(left_over_letters, default=0)): if D!=0: left_over_letters.remove(D) @interact def _(E =selector(left_over_letters, default=0)): if E!=0: left_over_letters.remove(E) @interact def _(F =selector(left_over_letters, default=0)): if F!=0: left_over_letters.remove(F) @interact def _(G =selector(left_over_letters, default=0)): if G!=0: left_over_letters.remove(G) @interact def _(H =selector(left_over_letters, default=0)): if H!=0: left_over_letters.remove(H) @interact def _(I =selector(left_over_letters, default=0)): if I!=0: left_over_letters.remove(I) @interact def _(J =selector(left_over_letters, default=0)): if J!=0: left_over_letters.remove(J) @interact def _(K =selector(left_over_letters, default=0)): if K!=0: left_over_letters.remove(K) @interact def _(L =selector(left_over_letters, default=0)): if L!=0: left_over_letters.remove(L) @interact def _(M =selector(left_over_letters, default=0)): if M!=0: left_over_letters.remove(M) @interact def _(N =selector(left_over_letters, default=0)): if N!=0: left_over_letters.remove(N) @interact def _(O =selector(left_over_letters, default=0)): if O!=0: left_over_letters.remove(O) @interact def _(P =selector(left_over_letters, default=0)): if P!=0: left_over_letters.remove(P) @interact def _(Q =selector(left_over_letters, default=0)): if Q!=0: left_over_letters.remove(Q) @interact def _(R =selector(left_over_letters, default=0)): if R!=0: left_over_letters.remove(R) @interact def _(S =selector(left_over_letters, default=0)): if S!=0: left_over_letters.remove(S) @interact def _(T =selector(left_over_letters, default=0)): if T!=0: left_over_letters.remove(T) @interact def _(U =selector(left_over_letters, default=0)): if U!=0: left_over_letters.remove(U) @interact def _(V =selector(left_over_letters, default=0)): if V!=0: left_over_letters.remove(V) @interact def _(W =selector(left_over_letters, default=0)): if W!=0: left_over_letters.remove(W) @interact def _(X =selector(left_over_letters, default=0)): if X!=0: left_over_letters.remove(X) @interact def _(Y =selector(left_over_letters, default=0)): if Y!=0: left_over_letters.remove(Y) @interact def _(Z =selector(left_over_letters, default=0)): if Z!=0: left_over_letters.remove(Z) @interact def _(text=input_box(default='"MESSAGE"',label="Message:")): new_ordering=[A,B,C,D,E,F,G,H,I,J,K,L,M,N,O,P,Q,R,S,T,U,V,W,X,Y,Z]; new_key=[ord(new_ordering[i])-65 for i in range(26)] alphabet=AlphabeticStrings() Es=SubstitutionCryptosystem(alphabet) Key = alphabet(new_key) e = Es(Key) TEXT0=text TEXT=alphabet.encoding(TEXT0) print "Ciphertext:", e(TEXT) | |
| Line 135: | Line 281: | 
| Use this interact to encrypt a message using the playfair cipher. | |
| Line 151: | Line 299: | 
| if (ch<>"J") & (pf.find(ch)==-1): # ensures no letter is repeated | if (ch!="J") & (pf.find(ch)==-1): # ensures no letter is repeated | 
| Line 168: | Line 316: | 
| if (i0<>i1) & (j0<>j1): | if (i0!=i1) & (j0!=j1): | 
| Line 170: | Line 318: | 
| if (i0==i1) & (j0<>j1): | if (i0==i1) & (j0!=j1): | 
| Line 172: | Line 320: | 
| if (i0<>i1) & (j0==j1): | if (i0!=i1) & (j0==j1): | 
| Line 211: | Line 359: | 
| print('Enter your message and the key to construct you polybius square. Warning: both the message and the key must be in quotes.') @interact def _(Message=input_box(default="'message'"),Key=input_box(default="'key'"),showmatrix=checkbox(True, label='Show polybius square')): | pretty_print(html("<h>Enter your message and the key to construct you polybius square. Warning: both the message and the key must be in quotes.<h>")) @interact def _(Message=input_box(default='"message"', label="Message:"),Key=input_box(default='"key"', label="Key:"),showmatrix=checkbox(True, label='Show polybius square:')): | 
| Line 244: | Line 392: | 
| if (ch<>"J") & (pf.find(ch)==-1): # ensures no letter is repeated | if (ch!="J") & (pf.find(ch)==-1): # ensures no letter is repeated | 
| Line 261: | Line 409: | 
| if (i0<>i1) & (j0<>j1):## if di[0] and di[1] are not in the same column or row, switch to corners in the same row | if (i0!=i1) & (j0!=j1):## if di[0] and di[1] are not in the same column or row, switch to corners in the same row | 
| Line 263: | Line 411: | 
| if (i0==i1) & (j0<>j1):## if di[0] and di[1] are in the same row, then switch left | if (i0==i1) & (j0!=j1):## if di[0] and di[1] are in the same row, then switch left | 
| Line 265: | Line 413: | 
| if (i0<>i1) & (j0==j1):## if di[0] and di[1] are in the same column, then switch up | if (i0!=i1) & (j0==j1):## if di[0] and di[1] are in the same column, then switch up | 
| Line 280: | Line 428: | 
| if len(pl1)%2<>0: | if len(pl1)%2: | 
| Line 283: | Line 431: | 
| if pl2<>pl: | if pl2!=pl: | 
| Line 286: | Line 434: | 
| if len(pl2)<>len(pl): | if len(pl2)!=len(pl): | 
| Line 319: | Line 467: | 
| if (ch=='X') & (pl.find(ch)<>0): | if (ch=='X') & (pl.find(ch)!=0): | 
| Line 325: | Line 473: | 
| print 'Enter your ciphertext and a guess for the key to construct you polybius square.' print 'Warning: both the message and the key must be in quotes.' @interact def _(Ciphertext=input_box(default="'LYXAXGDA'"),Key=input_box(default="'key'", label='Guess key'),showmatrix=checkbox(True, label='Show polybius square')): print 'These are some of the possibilities for the plaintext:' print playfair_decrypt_options(playfair_decrypt(Ciphertext,Key)) | pretty_print(html("<h>Enter your ciphertext and a guess for the key to construct you polybius square. Warning: both the message and the key must be in quotes.<h>")) @interact def _(Ciphertext=input_box(default='"LYXAXGDA"', label="Message:"),Key=input_box(default='"key"', label='Guess key:'),showmatrix=checkbox(True, label='Show polybius square:')): print('These are some of the possibilities for the plaintext:') print(playfair_decrypt_options(playfair_decrypt(Ciphertext,Key))) | 
| Line 333: | Line 480: | 
| print '----------------------' | print('----------------------') | 
| Line 352: | Line 499: | 
| print "This interact prints a bar graph showing the distribution of letters in the input text. Warning: the smaller the input text the less accurate the distribution will be. Letters that do not occur will not appear in the graph." | pretty_print(html("<h>This interact prints a bar graph showing the distribution of letters in the input text. Warning: the smaller the input text the less accurate the distribution will be. Letters that do not occur will not appear in the graph.<h>")) | 
| Line 356: | Line 503: | 
| def frequencyAnalysis(text = input_box('"Nyllappppunz tf uhtl pz Dlllnilya Klbjl aol Aybl Lhaly vm aol Avbwll. Olhy fl, olhy fl! Kll kll kll. H olhk vm aolzl spnly jbiz jhyyfpun aol aypwwf avthohdrz hyl jvtpun mv aoll. Ahrl zolsalyz pu aol avtiz. Ahttf yhu av aol vaoly avduzwlvwsl huk hhykchyrz. Doha pz oly LAH? Oly LAH wslhzl! Avps, iypun fvby mvvk jbwz huk vps huk il zdpma. Aol dvtlu huk aol jopsk Vjjvapvu JPPP zovbsk wpjr ihtivv ha Hapapzvapun. Zll.Uhuuh Db Zohjho z puuly uvvksl jbwz: uva ubbaaf zlzhtl uvapvuz."', width = 150)): | def frequencyAnalysis(text = input_box('"Nyllappppunz tf uhtl pz Dlllnilya Klbjl aol Aybl Lhaly vm aol Avbwll. Olhy fl, olhy fl! Kll kll kll. H olhk vm aolzl spnly jbiz jhyyfpun aol aypwwf avthohdrz hyl jvtpun mv aoll. Ahrl zolsalyz pu aol avtiz. Ahttf yhu av aol vaoly avduzwlvwsl huk hhykchyrz. Doha pz oly LAH? Oly LAH wslhzl! Avps, iypun fvby mvvk jbwz huk vps huk il zdpma. Aol dvtlu huk aol jopsk Vjjvapvu JPPP zovbsk wpjr ihtivv ha Hapapzvapun. Zll.Uhuuh Db Zohjho z puuly uvvksl jbwz: uva ubbaaf zlzhtl uvapvuz."',label = "Message:",width=150)): | 
| Line 388: | Line 535: | 
| print "Warning: the shorter the input text is, the less accurate the distribution will be." | pretty_print(html("<h>Warning: the shorter the input text is, the less accurate the distribution will be.<h>")) | 
| Line 391: | Line 538: | 
| def frequencyAnalysis(text = input_box('"Nyllappppunz tf uhtl pz Dlllnilya Klbjl aol Aybl Lhaly vm aol Avbwll. Olhy fl, olhy fl! Kll kll kll. H olhk vm aolzl spnly jbiz jhyyfpun aol aypwwf avthohdrz hyl jvtpun mv aoll. Ahrl zolsalyz pu aol avtiz. Ahttf yhu av aol vaoly avduzwlvwsl huk hhykchyrz. Doha pz oly LAH? Oly LAH wslhzl! Avps, iypun fvby mvvk jbwz huk vps huk il zdpma. Aol dvtlu huk aol jopsk Vjjvapvu JPPP zovbsk wpjr ihtivv ha Hapapzvapun. Zll.Uhuuh Db Zohjho z puuly uvvksl jbwz: uva ubbaaf zlzhtl uvapvuz."', width = 150)): | def frequencyAnalysis(text = input_box('"Nyllappppunz tf uhtl pz Dlllnilya Klbjl aol Aybl Lhaly vm aol Avbwll. Olhy fl, olhy fl! Kll kll kll. H olhk vm aolzl spnly jbiz jhyyfpun aol aypwwf avthohdrz hyl jvtpun mv aoll. Ahrl zolsalyz pu aol avtiz. Ahttf yhu av aol vaoly avduzwlvwsl huk hhykchyrz. Doha pz oly LAH? Oly LAH wslhzl! Avps, iypun fvby mvvk jbwz huk vps huk il zdpma. Aol dvtlu huk aol jopsk Vjjvapvu JPPP zovbsk wpjr ihtivv ha Hapapzvapun. Zll.Uhuuh Db Zohjho z puuly uvvksl jbwz: uva ubbaaf zlzhtl uvapvuz."', label = "Message:",width = 150)): | 
| Line 423: | Line 570: | 
| print "Put your message and codeword inside the quotes: " | pretty_print(html("<h>Put your message and codeword inside the quotes:<h>")) | 
| Line 425: | Line 572: | 
| def vigenere_cipher(message = input_box(default ="'secrets hi'", width = 50), code_word = input_box(default="'cat'", width = 50)): | def vigenere_cipher(message = input_box(default ='"secrets hi"',label="Message:"), code_word = input_box(default='"cat"',label="Key:")): | 
| Line 444: | Line 591: | 
| print "Put your encrypted message and codeword inside the quotes: " | pretty_print(html("<h>Put your encrypted message and codeword inside the quotes:<h>")) | 
| Line 446: | Line 593: | 
| def vigenere_cipher(message = input_box(default ="'UEVTEMUHB'", width = 50), code_word = input_box(default="'cat'", width = 50)): | def vigenere_cipher(message = input_box(default ='"UEVTEMUHB"',label = "Message:"), code_word = input_box(default='"cat"', label = "Key:")): | 
| Line 471: | Line 618: | 
| print "Enter your message to be encrypted via one-time pad in the Plain Text box below:" @interact def one_time_pad(plain_text = input_box("'message'",label="Plain Text:")): | pretty_print(html("<h>Enter your message to be encrypted via one-time pad in the message box below:<h>")) @interact def one_time_pad(plain_text = input_box('"message"',label="Message:")): | 
| Line 503: | Line 650: | 
| Use this interact to encrypt a message with the Hill cipher. Be sure to use an invertible matrix so that your message can be decrypted! | Use this interact to encrypt a message with the Hill cipher. If your message is not a multiple of n, then it will be padded with z's. Be sure to use an invertible matrix so that your message can be decrypted! | 
| Line 508: | Line 655: | 
| print "Please select the size of your key:" | pretty_print(html("<h>Please select the size of your key:<h>")) | 
| Line 514: | Line 661: | 
| def two_matrix(message=input_box(default='"Alexis smells"'), a=input_box(default=1), b=input_box(default=3), c=input_box(default=3), d=input_box(default=4)): | def two_matrix(message=input_box(default='"Alexis smells"',label = "Message:"), a=input_box(default=1), b=input_box(default=3), c=input_box(default=3), d=input_box(default=4)): | 
| Line 526: | Line 673: | 
| message=E.encoding(message) | newmessage = "" for char in message: if char.isalpha(): newmessage+=char.lower() if len(newmessage) % 2 == 1: newmessage+="z" message=E.encoding(newmessage) | 
| Line 530: | Line 683: | 
| print "Please input your message (in quotes) and the numbers in your square matrix key:" | pretty_print(html("<h>Please input your message (in quotes) and the numbers in your square matrix key:<h>")) | 
| Line 532: | Line 685: | 
| def three_matrix(message=input_box(default='"Alexis smells"'), a=input_box(default=1), b=input_box(default=2), c=input_box(default=3), d=input_box(default=5), e=input_box(default=2), f=input_box(default=6), g=input_box(default=7), h=input_box(default=9), i=input_box(default=9)): | def three_matrix(message=input_box(default='"Alexis smells"',label = "Message:"), a=input_box(default=1), b=input_box(default=2), c=input_box(default=3), d=input_box(default=5), e=input_box(default=2), f=input_box(default=6), g=input_box(default=7), h=input_box(default=9), i=input_box(default=9)): | 
| Line 544: | Line 697: | 
| message=E.encoding(message) | newmessage = "" for char in message: if char.isalpha(): newmessage+=char.lower() if len(newmessage) % 3 == 2: newmessage+="z" elif len(newmessage) % 3 == 1: newmessage+="zz" message=E.encoding(newmessage) | 
| Line 548: | Line 709: | 
| print "Please input your message (in quotes) and the numbers in your square matrix key:" | pretty_print(html("<h>Please input your message (in quotes) and the numbers in your square matrix key:<h>")) | 
| Line 550: | Line 711: | 
| def four_matrix(message=input_box(default='"Alexis smells"'), a=input_box(default=17), b=input_box(default=8), c=input_box(default=7), d=input_box(default=10), e=input_box(default=0), f=input_box(default=17), g=input_box(default=5), h=input_box(default=8), i=input_box(default=18), j=input_box(default=12), k=input_box(default=6), l=input_box(default=17), m=input_box(default=0), n=input_box(default=15), o=input_box(default=0), p=input_box(default=17)): | def four_matrix(message=input_box(default='"Alexis smells"',label="Message:"), a=input_box(default=17), b=input_box(default=8), c=input_box(default=7), d=input_box(default=10), e=input_box(default=0), f=input_box(default=17), g=input_box(default=5), h=input_box(default=8), i=input_box(default=18), j=input_box(default=12), k=input_box(default=6), l=input_box(default=17), m=input_box(default=0), n=input_box(default=15), o=input_box(default=0), p=input_box(default=17)): | 
| Line 562: | Line 723: | 
| message=E.encoding(message) | newmessage = "" for char in message: if char.isalpha(): newmessage+=char.lower() if len(newmessage) % 4 == 3: newmessage+="z" elif len(newmessage) % 4 == 2: newmessage+="zz" elif len(newmessage) % 4 == 1: newmessage+="zzz" message=E.encoding(newmessage) | 
| Line 579: | Line 750: | 
| print "Please select the size of your key:" | pretty_print(html("<h>Please select the size of your key:<h>")) | 
| Line 585: | Line 756: | 
| def two_decrypt(coded_text=input_box(default='"HSVAKSCYLENB"'), a=input_box(default=1), b=input_box(default=3), c=input_box(default=3), d=input_box(default=4)): | def two_decrypt(coded_text=input_box(default='"HSVAKSCYLENB"',label="Message:"), a=input_box(default=1), b=input_box(default=3), c=input_box(default=3), d=input_box(default=4)): | 
| Line 613: | Line 784: | 
| print "Please input your encrypted message and your key:" | pretty_print(html("<h>Please input your encrypted message and your key:<h>")) | 
| Line 615: | Line 786: | 
| def three_decrypt(coded_text=input_box(default='"FGYHQTCSGKYB"'), a=input_box(default=1), b=input_box(default=2), c=input_box(default=3), d=input_box(default=5), e=input_box(default=2), f=input_box(default=6), g=input_box(default=7), h=input_box(default=9), i=input_box(default=9)): | def three_decrypt(coded_text=input_box(default='"FGYHQTCSGKYB"',label = "Message:"), a=input_box(default=1), b=input_box(default=2), c=input_box(default=3), d=input_box(default=5), e=input_box(default=2), f=input_box(default=6), g=input_box(default=7), h=input_box(default=9), i=input_box(default=9)): | 
| Line 643: | Line 814: | 
| print "Please input your encrypted message (In quotes) and your key:" | pretty_print(html("<h>Please input your encrypted message (In quotes) and your key:<h>")) | 
| Line 645: | Line 816: | 
| def four_decrypt(coded_text=input_box(default='"UIBBSMUGGXTX"'), a=input_box(default=17), b=input_box(default=8), c=input_box(default=7), d=input_box(default=10), e=input_box(default=0), f=input_box(default=17), g=input_box(default=5), h=input_box(default=8), i=input_box(default=18), j=input_box(default=12), k=input_box(default=6), l=input_box(default=17), m=input_box(default=0), n=input_box(default=15), o=input_box(default=0), p=input_box(default=17)): | def four_decrypt(coded_text=input_box(default='UIBBSMUGGXTX',type=str,label="Message:"), a=input_box(default=17), b=input_box(default=8), c=input_box(default=7), d=input_box(default=10), e=input_box(default=0), f=input_box(default=17), g=input_box(default=5), h=input_box(default=8), i=input_box(default=18), j=input_box(default=12), k=input_box(default=6), l=input_box(default=17), m=input_box(default=0), n=input_box(default=15), o=input_box(default=0), p=input_box(default=17)): | 
| Line 677: | Line 848: | 
| This section gives visual representations of the modular arithmetic necessary for RSA, Diffie-Hellman, and El Gamal. | |
| Line 684: | Line 855: | 
| Given a positive integer n, this prints the multiplication mod n. Each entry in this table corresponds to the product of the row number by the column number, modulo n. | |
| Line 686: | Line 859: | 
| print "This tool creates a multiplication table modulo 𝑛." | pretty_print(html("<h1>Multiplication Table modulo n</h1>")) pretty_print(html("<h>This tool creates a multiplication table modulo 𝑛.<h>")) | 
| Line 694: | Line 868: | 
| === Modular Exponentiation === by Rebecca Lauren Miller, Kate Stange Given a modulus n and a nonnegative exponent a, this displays a graph where each integer between 0 and n-1 is mapped to its a-th power, mod n. {{{#!sagecell #Last edited 8/9/19 at 2:46pm pretty_print(html("<h1>Arrow Diagram modulo n</h1>")) pretty_print(html("<h>Input your modulus, 𝑛, and an integer, 𝑎. The output will be an arrow diagram picture of 𝑥↦𝑎𝑥 on the ring ℤ/𝑛ℤ, i.e. the elements modulo 𝑛.<h>")) @interact def modular_multiplication_graph(n = input_box(default = 7, width = 25), a = input_box(default = 2, width = 25)): R = IntegerModRing(n) left=[' '+str(r)+' ' for r in R] right=[' '+str(r)+' ' for r in R] pre_pos=graphs.CompleteBipartiteGraph(len(left),len(right)).get_pos() G = DiGraph() pos={} for (i,v) in enumerate(left+right): G.add_vertex(v) pos[v]=pre_pos[i] for l in range(n): G.add_edge(left[l],right[lift(R(a*l))]) show(G.plot(pos=pos)) }}} === Discrete Log Problem === by Sara Lapan The discrete logarithm, log(x) with base a, is an integer b such that a^b^ = x. Computing logarithms is computationally difficult, and there are no efficient algorithms known for the worst case scenarios. However, the discrete exponentiation is comparatively simple (for instance, it can be done efficiently using squaring). This asymmetry in complexity has been exploited in constructing cryptographic systems. Typically, it is much easier to solve for x in x = a^b^ (mod m) when a, b, and m are known, than it is to solve for b when x, a, and m are known. ==== Solving for x ==== Interact to find x when a, b, and m are known: {{{#!sagecell pretty_print(html("<h1>Solving for x</h1>")) pretty_print(html("<h>This will evaluate x=a^b (mod m). Choose your base (a), exponent (b), and modulus (m). These should all be positive integers.<h>")) @interact def DLP_solve(a=input_box(default=5),b=input_box(default=25),m=input_box(default=47)): if (not a in ZZ) or (not b in ZZ) or (not m in ZZ) or (a<=0) or (b<=0) or (m<=0): print "*********** ERROR: a,b,m should all be positive integers. ***********" else: a=Integer(a) b=Integer(b) m=Integer(m) print('This is the evaluation process using squares:') print('') C=b.str(base=2) L=len(C) S=[] T=[] ans=str(a) ans_num=a for i in range(L): pow=L-i-1 S+=[(2^pow)] print "Step",i+1,":",str(a)+"^"+str(2^i),"=",ans,"=",ans_num,"mod",m ans=str(ans_num)+"^"+str(2) ans_num= (ans_num)^2%m if C[pow]=="1": T+=[2^i] else: T expansion = "" STR="" STR_eval="" STR_eval_num=1 while len(T)>1: expansion += str(T[-1])+"+" STR += "("+str(a)+"^"+str(T[-1])+")" STR_eval += "("+str(a^(T[-1])%47)+")" STR_eval_num = STR_eval_num*(a^(T[-1])%47) T.remove(T[-1]) expansion+=str(T[0]) STR += "("+str(a)+"^"+str(T[0])+")" STR_eval += "("+str(a^(T[0])%47)+")" STR_eval_num = STR_eval_num*(a^(T[0])%47) STR_eval_num = STR_eval_num%47 print "Step",L+1,":",str(a)+"^"+str(b),"=",STR,"=",STR_eval,"=",STR_eval_num,"mod",m print " Since, as a sum of powers of 2,",str(b)+" is "+expansion+"." print "CONCLUSION: "+str(STR_eval_num)+" = "+str(a)+"^"+str(b)+" mod",m,". It takes",L+1,"steps to calculate x with this method." }}} ==== Solving for b ==== Interact to find b when a, x, and m are known: {{{#!sagecell pretty_print(html("<h1>Solving for b</h1>")) pretty_print(html("<h>This will solve for the exponent, b, in x=a^b (mod m) assuming an integer solution exists. Choose your base (a), modulus (m), and solution (x). These should all be positive integers.<h>")) @interact def DLP_break(a=input_box(default=5),x=input_box(default=22),m=input_box(default=47)): if (not a in ZZ) or (not x in ZZ) or (not m in ZZ) or (a<=0) or (x<0) or (m<=0): print("*********** ERROR: a,m,x should all be integers with a,m>0. ***********") print() elif x==1: print("b=0") else: a=Integer(a) x=Integer(x) m=Integer(m) ind=0 for i in [1..m]: temp = a^i %m if temp==x: ind=1 print("This process took",i+1,"steps to determine that b="+str(i),"by evaluating",str(a)+"^i for i=0,...,"+str(i)+".") print() print("NOTE: Without restricting the size of b, there is not a unique solution for b. However, the solution above is the smallest possible solution.") else: temp if ind==0: print("*********** ERROR: This process took",m,"steps to determine that there is no integer solution for b.***********") }}} | |
| Line 696: | Line 993: | 
| Named for the authors Rivest, Shamir, Aldeman, RSA uses exponentiation and modular arithmetic to encrypt and decrypt messages between two parties. Each of those parties has their own secret and public key. To see how it works, following along while Alice and Babette share a message. -EG | Named for the authors Rivest, Shamir, and Aldeman, RSA uses exponentiation and modular arithmetic to encrypt and decrypt messages between two parties. Each of those parties has their own secret and public key. To see how it works, following along while Alice and Babette share a message. | 
| Line 701: | Line 998: | 
| Babette sent Alice an encrypted message. You , as Alice, will provide information so that you can read Babette's message. | Babette sent Alice an encrypted message. You, as Alice, will provide information so that you can read Babette's message. | 
| Line 705: | Line 1002: | 
| pretty_print(html("<h1>RSA, From Alice's Perspective</h1>")) | |
| Line 708: | Line 1006: | 
| print " The size of the primes depends on the size of Babette's message. Her message requires p,q > 10. A longer and stronger encrypted message requires larger primes." | print "   The size of the primes depends on the size of Babette's message. Her message requires p,q > 10. A longer and stronger encrypted" print " message requires larger primes." | 
| Line 711: | Line 1010: | 
| print " If e is not relativley prime to φ(pq), then we can not decrypt the message." | print " If e is not relatively prime to φ(pq), then we can not decrypt the message." | 
| Line 717: | Line 1016: | 
| #print "************************************************************************************************" #print "WARNINGS: p and q should be different primes, both larger than 10." #print "e should be relatively prime to φ(pq). To check this, see the factorization of φ(pq) below." #print "************************************************************************************************" #print "" | |
| Line 734: | Line 1028: | 
| print "*********** e must be replatively prime to φ(pq) - see factorization below. ***********" | print "*********** e must be relatively prime to φ(pq) - see factorization below. ***********" | 
| Line 785: | Line 1079: | 
| #Last edited 8/8/19 at 12:30pm print "Hi, Babette! Let's send a message to Alice using RSA." p = next_prime(100) q = next_prime(p) phi = (p-1)*(q-1) e = 13 N = p*q R = IntegerModRing(phi) d = (e^(R(e).multiplicative_order()-1)).mod(phi) print "Alice's public key is: N =",N,", e =",e,"." message = raw_input("Type a message for Alice:") ascii_secret = [] for char in message: ascii_secret.append(ord(char)) print "We turn these characters into ascii:" print ascii_secret | #Last edited 8/9/19 2:40pm pretty_print(html("<h1>RSA, From Babette's Perspective</h1>")) print "Hi, Babette! Let's send a message to Alice using her PUBLIC key (N, e) with RSA." | 
| Line 802: | Line 1083: | 
| print "Then we encode them by raising each ascii number to the e-th power modulo N." encrypted_ascii = [] for ascii in ascii_secret: encrypted_ascii.append(power_mod(ascii,e,N)) print encrypted_ascii print "" @interact def rsa(): print "Alice receives our secret and uses her private key to decrypt the message." | print "1. Input Babette's secret message for Alice below." print " Make sure that there are no apostrophes or extra quotation marks in your message." @interact def rsa(message = input_box(default = 'Secrets for Alice', type=str,label="Message:")): p = next_prime(100) q = next_prime(p) phi = (p-1)*(q-1) e = 13 N = p*q R = IntegerModRing(phi) d = (e^(R(e).multiplicative_order()-1)).mod(phi) ascii_secret = [] for char in message: ascii_secret.append(ord(char)) print "2. Using ASCII, we take the characters in our message and convert each of them into integers." print "" print " ",ascii_secret print "" print "Alice's PUBLIC key is given to be (N, e) = (",N,",",e,")." print "" print "4. We encode the list of numbers by raising each integer to the e-th power modulo N. Recall that e is called the encryption key. This is what get's sent to Alice:" encrypted_ascii = [] for ascii in ascii_secret: encrypted_ascii.append(power_mod(ascii,e,N)) print "" print " ",encrypted_ascii print "" print "5. To decrypt, Alice raises each integer to the d-th power modulo N, where d is Alice's PRIVATE decryption key." | 
| Line 814: | Line 1114: | 
| print decrypted_ascii | print "" print " ", decrypted_ascii | 
| Line 819: | Line 1120: | 
| print "Going from ascii to letters, she figures out your message was: " print decrypted_secret | print "6. Going from the integers to letters using ASCII, we find that Babette's message was " print "" print " ",decrypted_secret | 
| Line 830: | Line 1132: | 
| }}} == Diffe-Hellman Key Exchange == | #Last edited 8/9/19 at 3:52pm print "Hi, Alice! Let's send a message to Babette with your digital signature so that Babette knows that it is really Alice." print "" print "1. Make Alice's PRIVATE key: Input two distinct primes, p and q, that are each greater than 10, and an integer, e, that is relatively prime to the the Euler φ-function of the product pq." @interact def rsa(message_to_babette = input_box(default = 'Hi',type=str,label="message:"),p_a = input_box(default = 503,label = "p: "), q_a = input_box(default = 499,label = "q: "),e_a = input_box(default = 5,label = "e:")): p_a = ZZ(p_a) q_a = ZZ(q_a) e_a = ZZ(e_a) p_b = 1123 q_b = 4999 e_b = 5 if p_a < 10: print "*********** Make p larger. ***********" return " " if q_a < 10: print "*********** Make q larger. ***********" return " " if not p_a.is_prime(): print "*********** p needs to be prime. ***********" return " " if not q_a.is_prime(): print "*********** q needs to be prime. ***********" return " " phi_a = (p_a-1)*(q_a-1) phi_b = (p_b-1)*(q_b-1) if not gcd(e_a,phi_a) == 1: print "*********** e must be replatively prime to φ(pq) - see factorization below. ***********" print "" print "φ(pq) = ",phi_a.factor() return " " print "" print "φ(pq) = ",phi_a.factor() print "" N_a = p_a*q_a N_b = p_b*q_b if N_b < N_a: print "Choose primes for p or q so that their product",N_a ,"is smaller than ",N_b,"." print " This is not needed for general digital signatures, but is necessary for this program to decrypt the message correctly." return " " R = IntegerModRing(phi_a) d_a = (e_a^(R(e_a).multiplicative_order()-1)).mod(phi_a) S = IntegerModRing(phi_b) d_b = (e_b^(S(e_b).multiplicative_order()-1)).mod(phi_b) print "2. Alice's PRIVATE key is (p,q,d) =(",p_a,",",q_a,",",d_a,"), where the decryption key d is the inverse of e modulo φ(N)." print "" print " Alice's PUBLIC key is (N,e) =(",N_a,",",e_a,")." print "" print "We are given Babette's PUBLIC key of (N_b,e_b) = (",N_b,",",e_b,")." print "" ascii_secret = [] for char in message_to_babette: ascii_secret.append(ord(char)) encrypted_ascii = [] for ascii in ascii_secret: ascii = ZZ(ascii) signed = (ascii^d_a).mod(N_a) encrypted_ascii.append((signed^e_b).mod(N_b)) decrypted_ascii = [] for ascii in encrypted_ascii: ascii = ZZ(ascii) unencrypt = (ascii^d_b).mod(N_b) unsign = (unencrypt^e_a).mod(N_a) decrypted_ascii.append(unsign) print "3. Use ASCII to convert the plaintext message to integers." print "" print " ",ascii print "" print "4. Sign the message using Alice's PRIVATE key by raising each integer in the list to the d-th power modulo N." print "" print " ",signed print "" print "5. Finally, to encrypt the signed message, use Babette's PUBLIC key by raising every integer to the e_b-th power modulo N_b." print "" print " ",encrypted_ascii print "" print "6. To decrypt the signed encrypted message, Babette will use Alice's PUBLIC key (",N_a,",",e_a,") AND Babette's PRIVATE key (",p_b,",",q_b,",", d_b,"), as given here by the program." print "" print " ",decrypted_ascii print "" decrypted_secret = "" for ascii in decrypted_ascii: decrypted_secret += chr(ascii) print "7. Using the ASCII code to convert the intgers back to letters, we find out the signed secret message was from Alice and read " print " ",decrypted_secret }}} | 
Sage Interactions - Cryptography
This page was first created at Sage Days 103, 7-9 August 2019 by Sarah Arpin, Catalina Camacho-Navarro, Holly Paige Chaos, Amy Feaver, Eva Goedhart, Sara Lapan, Rebecca Lauren Miller, Alexis Newton, and Nandita Sahajpal. Text edited by Holly Paige Chaos, Amy Feaver, Eva Goedhart, and Alexis Newton. This project was led by Amy Feaver and Eva Goedhart.
We acknowledge Katherine Stange, who allowed us to use code from her cryptography course as a starting point for many of these interacts. Dr. Stange's original code and course page can be found at http://crypto.katestange.net/
If you have cryptography-related interactions that you are interested in adding to this page, please do so. You can also contact Amy Feaver at [email protected]
goto interact main page
Contents
Shift Cipher
The shift cipher is a classical cryptosystem that takes plaintext and shifts it through the alphabet by a given number of letters. For example, a shift of 2 would replace all A's with C's, all B's with D's, etc. When the end of the alphabet is reached, the letters are shifted cyclically back to the beginning. Thus, a shift of 2 would replace Y's with A's and Z's with B's.
Shift Cipher Encryption
by Sarah Arpin, Alexis Newton
You can use this interact to encrypt a message with a shift cipher.
Shift Cipher Decryption
by Sarah Arpin, Alexis Newton
If you know that your message was encrypted using a shift cipher, you can use the known shift value to decrypt. If this is not known, brute force can be used to get 26 possible decrypted messages. The chi-squared function ranks the brute force results by likelihood according to letter frequency.
Affine Cipher
An affine cipher combines the idea of a shift cipher with a multiplicative cipher. In this particular example, we map consecutive letters of the alphabet to consecutive numbers, starting with A=0 (you can also do this cipher differently, and starting with A=1). The user selects two values, a and b. The value a is the multiplier and must be relatively prime to 26 in order to guarantee that each letter is encoded uniquely. The value b is the addend. Each letter's value is multiplied by a, and the product is added to b. This is then replaced with a new letter, corresponding to the result modulo 26.
Affine Cipher Encryption
by Sarah Arpin, Alexis Newton
You can use this interact to encrypt a message with the affine cipher. Notice that the only choices for a can be numbers that are relatively prime to 26. This cipher will encipher a letter m of your message as a*m + b.
Affine Cipher Decryption
by Sarah Arpin, Alexis Newton
If you know that your message was encrypted using an affine cipher, you can use the known a and b values to decrypt. If these are not known, brute force can be used to get a list of possible decrypted messages. The chi-squared function ranks these results by likelihood according to letter frequency.
Substitution Cipher
by Catalina Camacho-Navarro
A substitution cipher encrypts messages by assigning each letter of the alphabet to another letter. For instance, if A is assigned to F, then all A's in the original message will be substituted with F's in the encrypted message. Brute force or frequency analysis can be used to decrypt a message encrypted with a substitution cipher.
Playfair Cipher
by Catalina Camacho-Navarro
Based on code from Alasdair McAndrew at trac.sagemath.org/ticket/8559.
A playfair cipher is a special type of substitution cipher in which the plaintext is broken up into two-letter digraphs with some restrictions. Those digraphs are encrypted using a Polybius square, (i.e. a 5x5 grid in which each letter of the alphabet is its own entry with the exception of ij which are placed together). The positions of the letters in the digraph determine how the digraph is encrypted.
Playfair Encryption
Use this interact to encrypt a message using the playfair cipher.
Playfair Decryption
Frequency Analysis Tools
Frequency analysis is a technique for breaking a substitution cipher that utilizes the frequencies of letters appearing in the English language. For example, E is the most common letter in the English language, so if a piece of encrypted text had many instances of the letter Q, you would guess that Q had been substituted in for E. The next two interacts create a couple of basic tools that could be useful in cracking a substitution cipher.
Letter Frequency Counter
by Rebecca Lauren Miller, Katherine Stange
This tool looks at the percentage of appearances of each letter in the input text and plots these percentages. The encrypted input text is a bit strange, but was constructed by Amy Feaver to give a short block of text that matched the frequencies of letters in the English language relatively well, to make this message easier to decrypt.
Frequency Analysis Decryption Guesser
by Rebecca Lauren Miller, Katherine Stange
This interact prints a suggested translation of the input text by matching frequencies of letters in the input to frequencies of letters in the English language. With the output you will see that some letters were substituted incorrectly, and others were not. Usually frequency analysis requires additional work and some trial and error to discover the original message, especially if the input text is not long enough.
Vigenère Cipher
A Vigenère cipher is an example of a polyalphabetic cipher. Using a secret codeword as the key, the Vigenère encrypts each letter of a message by shifting it according to the corresponding letter in the key. For example, we will use the key "CAT" to encrypt our default text "secrets hi". To do this the message is first broken up into three-letter chunks, because the key is three letters long, to be "SEC RET SHI". Next each letter of the chunk is shifted by the value of the corresponding letter in the key. The standard shifts are A=0, B=1, C=2, etc. So in our example, S shifts by C=2 letters to U, E shifts by A=0 letters and remains at E, and C shifts by T=19 letters to V. Thus "SECRETSHI" becomes UEVTEMUHB when encrypted. To decrypt the message, simply use the keyword to undo the encryption process. Cryptography by Simon Rubinstein-Salzedo was used as reference for this interact.
Vigenère Cipher Encryption
by Holly Paige Chaos, Rebecca Lauren Miller, Katherine Stange
Use this interact to encrypt a message using the Vigenère Cipher.
Vigenère Cipher Decryption
by Holly Paige Chaos, Rebecca Lauren Miller, Katherine Stange
If you used the Vigenère Cipher to encrypt a message, you can use this interact to decrypt by inputting your key and encrypted text.
One-Time Pad
by Sarah Arpin, Alexis Newton
One-time pad is an encryption method that cannot be cracked. It requires a single-use shared key (known as a one-time pad) the length of the message or longer. In this method, every letter is first converted to numbers using the standard A=0, B=1, C=2, etc. Then each character in the message is multiplied modulo 26 by the number in the corresponding position in the key. This is then converted back to letters to produce the encrypted text.
Hill Cipher
The Hill cipher requires some basic knowledge of Linear Algebra. In this encryption method, an invertible n x n matrix of integers modulo 26 is used as the key. The message is first converted to numbers and spit into chunks size n. These chunks are then converted to n x 1 vectors and multiplied by the key modulo 26 to produce 1 x n vectors. The integers from these vectors are converted back letters to produce the encrypted text.
Hill Cipher Encryption
by Holly Paige Chaos, Alexis Newton
Use this interact to encrypt a message with the Hill cipher. If your message is not a multiple of n, then it will be padded with z's. Be sure to use an invertible matrix so that your message can be decrypted!
Hill Cipher Decryption
by Holly Paige Chaos, Alexis Newton
Use this interact to decrypt messages encrypted by the Hill cipher. Remember that this only works if the message was encrypted using an invertible matrix as the key!
Modular Arithmetic (Preliminaries for RSA, Diffie-Hellman, El Gamal)
This section gives visual representations of the modular arithmetic necessary for RSA, Diffie-Hellman, and El Gamal.
Modular Arithmetic Multiplication Table
by Rebecca Lauren Miller, Kate Stange
Given a positive integer n, this prints the multiplication mod n. Each entry in this table corresponds to the product of the row number by the column number, modulo n.
Modular Exponentiation
by Rebecca Lauren Miller, Kate Stange
Given a modulus n and a nonnegative exponent a, this displays a graph where each integer between 0 and n-1 is mapped to its a-th power, mod n.
Discrete Log Problem
by Sara Lapan
The discrete logarithm, log(x) with base a, is an integer b such that ab = x. Computing logarithms is computationally difficult, and there are no efficient algorithms known for the worst case scenarios. However, the discrete exponentiation is comparatively simple (for instance, it can be done efficiently using squaring). This asymmetry in complexity has been exploited in constructing cryptographic systems. Typically, it is much easier to solve for x in x = ab (mod m) when a, b, and m are known, than it is to solve for b when x, a, and m are known.
Solving for x
Interact to find x when a, b, and m are known:
Solving for b
Interact to find b when a, x, and m are known:
RSA
Named for the authors Rivest, Shamir, and Aldeman, RSA uses exponentiation and modular arithmetic to encrypt and decrypt messages between two parties. Each of those parties has their own secret and public key. To see how it works, following along while Alice and Babette share a message.
RSA, From Alice's Perspective
by Sarah Arpin, Eva Goedhart
Babette sent Alice an encrypted message. You, as Alice, will provide information so that you can read Babette's message.
RSA, From Babette's Perspective
by Sarah Arpin, Eva Goedhart
RSA With Digital Signatures
by Sarah Arpin, Eva Goedhart
