| 
									
										
										
										
											2013-06-15 23:25:49 +02:00
										 |  |  | # | 
					
						
							| 
									
										
										
										
											2014-06-23 22:20:10 +02:00
										 |  |  | # Util/Padding.py :  Functions to manage padding | 
					
						
							| 
									
										
										
										
											2013-06-15 23:25:49 +02:00
										 |  |  | # | 
					
						
							|  |  |  | # =================================================================== | 
					
						
							|  |  |  | # | 
					
						
							| 
									
										
										
										
											2014-06-23 22:20:10 +02:00
										 |  |  | # Copyright (c) 2014, Legrandin <helderijs@gmail.com> | 
					
						
							|  |  |  | # All rights reserved. | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # Redistribution and use in source and binary forms, with or without | 
					
						
							|  |  |  | # modification, are permitted provided that the following conditions | 
					
						
							|  |  |  | # are met: | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # 1. Redistributions of source code must retain the above copyright | 
					
						
							|  |  |  | #    notice, this list of conditions and the following disclaimer. | 
					
						
							|  |  |  | # 2. Redistributions in binary form must reproduce the above copyright | 
					
						
							|  |  |  | #    notice, this list of conditions and the following disclaimer in | 
					
						
							|  |  |  | #    the documentation and/or other materials provided with the | 
					
						
							|  |  |  | #    distribution. | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS | 
					
						
							|  |  |  | # "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT | 
					
						
							|  |  |  | # LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS | 
					
						
							|  |  |  | # FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE | 
					
						
							|  |  |  | # COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, | 
					
						
							|  |  |  | # INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, | 
					
						
							|  |  |  | # BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; | 
					
						
							|  |  |  | # LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER | 
					
						
							|  |  |  | # CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT | 
					
						
							|  |  |  | # LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN | 
					
						
							|  |  |  | # ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE | 
					
						
							|  |  |  | # POSSIBILITY OF SUCH DAMAGE. | 
					
						
							| 
									
										
										
										
											2013-06-15 23:25:49 +02:00
										 |  |  | # =================================================================== | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | """ Functions to manage padding
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | This module provides minimal support for adding and removing standard padding | 
					
						
							|  |  |  | from data. | 
					
						
							|  |  |  | """
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-04 22:04:34 +01:00
										 |  |  | __all__ = [ 'ValueError', 'pad', 'unpad' ] | 
					
						
							| 
									
										
										
										
											2013-06-15 23:25:49 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | from Crypto.Util.py3compat import * | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def pad(data_to_pad, block_size, style='pkcs7'): | 
					
						
							|  |  |  |     """Apply standard padding.
 | 
					
						
							| 
									
										
										
										
											2014-03-04 22:04:34 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-15 23:25:49 +02:00
										 |  |  |     :Parameters: | 
					
						
							|  |  |  |       data_to_pad : byte string | 
					
						
							|  |  |  |         The data that needs to be padded. | 
					
						
							|  |  |  |       block_size : integer | 
					
						
							|  |  |  |         The block boundary to use for padding. The output length is guaranteed | 
					
						
							|  |  |  |         to be a multiple of ``block_size``. | 
					
						
							|  |  |  |       style : string | 
					
						
							|  |  |  |         Padding algorithm. It can be *'pkcs7'* (default), *'iso7816'* or *'x923'*. | 
					
						
							|  |  |  |     :Return: | 
					
						
							|  |  |  |       The original data with the appropriate padding added at the end. | 
					
						
							|  |  |  |     """
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     padding_len = block_size-len(data_to_pad)%block_size | 
					
						
							|  |  |  |     if style == 'pkcs7': | 
					
						
							|  |  |  |         padding = bchr(padding_len)*padding_len | 
					
						
							|  |  |  |     elif style == 'x923': | 
					
						
							|  |  |  |         padding = bchr(0)*(padding_len-1) + bchr(padding_len) | 
					
						
							|  |  |  |     elif style == 'iso7816': | 
					
						
							|  |  |  |         padding = bchr(128) + bchr(0)*(padding_len-1) | 
					
						
							|  |  |  |     else: | 
					
						
							|  |  |  |         raise ValueError("Unknown padding style") | 
					
						
							|  |  |  |     return data_to_pad + padding | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def unpad(padded_data, block_size, style='pkcs7'): | 
					
						
							|  |  |  |     """Remove standard padding.
 | 
					
						
							| 
									
										
										
										
											2014-03-04 22:04:34 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-15 23:25:49 +02:00
										 |  |  |     :Parameters: | 
					
						
							|  |  |  |       padded_data : byte string | 
					
						
							|  |  |  |         A piece of data with padding that needs to be stripped. | 
					
						
							|  |  |  |       block_size : integer | 
					
						
							|  |  |  |         The block boundary to use for padding. The input length | 
					
						
							|  |  |  |         must be a multiple of ``block_size``. | 
					
						
							|  |  |  |       style : string | 
					
						
							|  |  |  |         Padding algorithm. It can be *'pkcs7'* (default), *'iso7816'* or *'x923'*. | 
					
						
							|  |  |  |     :Return: | 
					
						
							|  |  |  |         Data without padding. | 
					
						
							| 
									
										
										
										
											2014-03-04 22:04:34 +01:00
										 |  |  |     :Raises ValueError: | 
					
						
							| 
									
										
										
										
											2013-06-15 23:25:49 +02:00
										 |  |  |         if the padding is incorrect. | 
					
						
							|  |  |  |     """
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     pdata_len = len(padded_data) | 
					
						
							|  |  |  |     if pdata_len % block_size: | 
					
						
							| 
									
										
										
										
											2014-03-04 22:04:34 +01:00
										 |  |  |         raise ValueError("Input data is not padded") | 
					
						
							| 
									
										
										
										
											2013-06-15 23:25:49 +02:00
										 |  |  |     if style in ('pkcs7', 'x923'): | 
					
						
							|  |  |  |         padding_len = bord(padded_data[-1]) | 
					
						
							|  |  |  |         if padding_len<1 or padding_len>min(block_size, pdata_len): | 
					
						
							| 
									
										
										
										
											2014-03-04 22:04:34 +01:00
										 |  |  |             raise ValueError("Padding is incorrect.") | 
					
						
							| 
									
										
										
										
											2013-06-15 23:25:49 +02:00
										 |  |  |         if style == 'pkcs7': | 
					
						
							|  |  |  |             if padded_data[-padding_len:]!=bchr(padding_len)*padding_len: | 
					
						
							| 
									
										
										
										
											2014-03-04 22:04:34 +01:00
										 |  |  |                 raise ValueError("PKCS#7 padding is incorrect.") | 
					
						
							| 
									
										
										
										
											2013-06-15 23:25:49 +02:00
										 |  |  |         else: | 
					
						
							|  |  |  |             if padded_data[-padding_len:-1]!=bchr(0)*(padding_len-1): | 
					
						
							| 
									
										
										
										
											2014-03-04 22:04:34 +01:00
										 |  |  |                 raise ValueError("ANSI X.923 padding is incorrect.") | 
					
						
							| 
									
										
										
										
											2013-06-15 23:25:49 +02:00
										 |  |  |     elif style == 'iso7816': | 
					
						
							|  |  |  |         padding_len = pdata_len - padded_data.rfind(bchr(128)) | 
					
						
							|  |  |  |         if padding_len<1 or padding_len>min(block_size, pdata_len): | 
					
						
							| 
									
										
										
										
											2014-03-04 22:04:34 +01:00
										 |  |  |             raise ValueError("Padding is incorrect.") | 
					
						
							| 
									
										
										
										
											2013-06-15 23:25:49 +02:00
										 |  |  |         if padding_len>1 and padded_data[1-padding_len:]!=bchr(0)*(padding_len-1): | 
					
						
							| 
									
										
										
										
											2014-03-04 22:04:34 +01:00
										 |  |  |             raise ValueError("ISO 7816-4 padding is incorrect.") | 
					
						
							| 
									
										
										
										
											2013-06-15 23:25:49 +02:00
										 |  |  |     else: | 
					
						
							|  |  |  |         raise ValueError("Unknown padding style") | 
					
						
							|  |  |  |     return padded_data[:-padding_len] | 
					
						
							|  |  |  | 
 |