Fence Plus decryption is a short string of processing, given the number of rows row, based on the length of the string to calculate the column, a square.
The process of encryption is to arrange the clear text from top to bottom in columns, then scramble the rows according to the key, and finally merge from left to right in line order to form the ciphertext.
Decryption process: The above process is reversed, each line according to the order of the key to the original square of the order, and restore the original square from the ciphertext, and finally in the order of columns from top to bottom from left to right decryption.
The implementation is as follows: All implementations are encapsulated into a class railfence, the number of columns and keys can be specified at initialization, the default number of columns is 2, no key. The initialization functions are as follows:
def __init__ (self, row = 2, mask = None): if row < 2: raise ValueError (U ' not acceptable row number or mask Valu E ') Self . Row = row if mask! = None and not isinstance (mask, types. StringType, types. Unicodetype)): Raise ValueError (U ' not acceptable mask value ') Self . Mask = Mask self . Length = 0 Self . Column = 0
Encryption process, you can choose whether to remove whitespace characters. The first is the type check, the calculation of the number of columns, the core is the calculation of the parameters to get gird This two-dimensional list representation of the square, is also the core of the fence encryption. The specific implementation is as follows:
def encrypt (self, src, nowhitespace = False): If is not isinstance (SRC, (types). StringType, types. Unicodetype)): Raise TypeError (U ' encryption src text is not string ') if nowhitespace:self. Nowhitespace = ' For i in Src:if I in string.whitespace:continue self. Nowhitespace + = i else:self. Nowhitespace = src self. Length = Len (self. Nowhitespace) self. Column = Int (Math.ceil (self). Length/self. Row)) Try:self.__check () except Exception, Msg:print msg #get Mask Order Self.__getorder () Grid = [[] for I in range (self. Row)] for the C in range (self. Column): EndIndex = (c + 1) * self. Row if EndIndex > self. Length:endindex = self. Length r = self. Nowhitespace[c * Self. Row:endindex] for I,j in Enumerate (R): if self. Mask! = None and Len (Self. Order) > 0:grid[self. Order[i]].append (j) Else:grid[i].append (j) return ". Join ([". Join (L) for L in G RID])
The main method is to iterate through the number of columns, each time a string that takes a number of columns from the plaintext is saved in the Traverse R, where the end subscript of the last column is processed beyond the string length. The column string is then sequentially added to the corresponding positions of the columns in the square grid.
The decryption process is a bit more complicated, because the order of the key pairs is scrambled, the order of the scrambled rows needs to be resumed, the previous phalanx is obtained, and then the concatenated string is followed in the order of the columns to get the decrypted string. The specific implementation is as follows:
Def decrypt (self, DST): If not isinstance (DST, (types). StringType, types. Unicodetype)): Raise TypeError (U ' decryption DST text is not a string ') self. Length = Len (DST) self. Column = Int (Math.ceil (self). Length/self. Row)) Try:self.__check () except Exception, Msg:print msg #get Mask Order Self.__getorder () Grid = [[] for I in range (self. ROW)] space = self. Row * Self. Column-self. Length ns = self. Row-space Preve = 0 for I in range (self. ROW): if self. Mask! = none:s = Preve O = 0 for x, y in enumerate (self. Order): if i = = Y:o = x break if O < n S:e = s + self. Column Else:e = s + (self. Column-1) R = dst[s:e] Preve = e Grid[o] = list (r) Else: StartIndex = 0 EndIndex = 0 if i < self. Row-space:startindex = i * self. Column EndIndex = StartIndex + self. Column Else:startindex = ns * Self. Column + (I-NS) * (self. Column-1) EndIndex = StartIndex + (self. Column-1) R = dst[startindex:endindex] grid[i] = List (r) res = ' for C in R Ange (self. Column): For I in range (self. Row): line = grid[i [] If len (line) = = C:res + = ' Else: Res + = Line[c] return res
Actual operation
The test code is as follows, encrypted with 4 lines and the key is BCAF:
RF = Railfence (4, ' BCAF ') e = Rf.encrypt (' The anwser is wctf{c01umnar},if u are a big new,u can help us think more ques Tion,tks. ') Print "Encrypt:", e print "Decrypt:", Rf.decrypt (E)
Results such as:
Description: The decryption process given here is known to encrypt the number of columns, if unknown, only need to traverse the number of columns, the decryption function can be repeatedly called.
Full code is here: https://github.com/OshynSong/web/blob/master/RailFence.py
Fence Plus decrypt Python implementation (key encryption supported)