| 
									
										
										
										
											2013-11-21 15:46:49 +01:00
										 |  |  | # Copyright (C) 2003-2013 Python Software Foundation | 
					
						
							| 
									
										
										
										
											2019-05-15 22:14:38 +02:00
										 |  |  | import copy | 
					
						
							|  |  |  | import operator | 
					
						
							|  |  |  | import pickle | 
					
						
							| 
									
										
										
										
											2020-11-02 23:01:40 +02:00
										 |  |  | import struct | 
					
						
							| 
									
										
										
										
											2003-02-25 12:58:58 +00:00
										 |  |  | import unittest | 
					
						
							|  |  |  | import plistlib | 
					
						
							|  |  |  | import os | 
					
						
							| 
									
										
										
										
											2021-07-17 14:11:04 +04:00
										 |  |  | import sys | 
					
						
							|  |  |  | import json | 
					
						
							| 
									
										
										
										
											2004-10-26 11:02:08 +00:00
										 |  |  | import datetime | 
					
						
							| 
									
										
										
										
											2013-11-21 15:46:49 +01:00
										 |  |  | import codecs | 
					
						
							| 
									
										
										
										
											2021-07-17 14:11:04 +04:00
										 |  |  | import subprocess | 
					
						
							| 
									
										
										
										
											2013-11-21 15:46:49 +01:00
										 |  |  | import binascii | 
					
						
							|  |  |  | import collections | 
					
						
							| 
									
										
										
										
											2024-01-02 02:51:24 +08:00
										 |  |  | import time | 
					
						
							|  |  |  | import zoneinfo | 
					
						
							| 
									
										
										
										
											2008-05-20 21:35:26 +00:00
										 |  |  | from test import support | 
					
						
							| 
									
										
										
										
											2020-07-09 21:25:10 +08:00
										 |  |  | from test.support import os_helper | 
					
						
							| 
									
										
										
										
											2013-11-21 15:46:49 +01:00
										 |  |  | from io import BytesIO | 
					
						
							| 
									
										
										
										
											2003-02-25 12:58:58 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-15 22:14:38 +02:00
										 |  |  | from plistlib import UID | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-21 15:46:49 +01:00
										 |  |  | ALL_FORMATS=(plistlib.FMT_XML, plistlib.FMT_BINARY) | 
					
						
							| 
									
										
										
										
											2004-10-26 11:02:08 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-21 15:46:49 +01:00
										 |  |  | # The testdata is generated using Mac/Tools/plistlib_generate_testdata.py | 
					
						
							|  |  |  | # (which using PyObjC to control the Cocoa classes for generating plists) | 
					
						
							|  |  |  | TESTDATA={ | 
					
						
							|  |  |  |     plistlib.FMT_XML: binascii.a2b_base64(b'''
 | 
					
						
							|  |  |  |         PD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0iVVRGLTgiPz4KPCFET0NU | 
					
						
							|  |  |  |         WVBFIHBsaXN0IFBVQkxJQyAiLS8vQXBwbGUvL0RURCBQTElTVCAxLjAvL0VO | 
					
						
							|  |  |  |         IiAiaHR0cDovL3d3dy5hcHBsZS5jb20vRFREcy9Qcm9wZXJ0eUxpc3QtMS4w | 
					
						
							| 
									
										
										
										
											2014-01-15 11:32:35 +01:00
										 |  |  |         LmR0ZCI+CjxwbGlzdCB2ZXJzaW9uPSIxLjAiPgo8ZGljdD4KCTxrZXk+YUJp | 
					
						
							|  |  |  |         Z0ludDwva2V5PgoJPGludGVnZXI+OTIyMzM3MjAzNjg1NDc3NTc2NDwvaW50 | 
					
						
							| 
									
										
										
										
											2014-02-06 11:19:18 +01:00
										 |  |  |         ZWdlcj4KCTxrZXk+YUJpZ0ludDI8L2tleT4KCTxpbnRlZ2VyPjkyMjMzNzIw | 
					
						
							|  |  |  |         MzY4NTQ3NzU4NTI8L2ludGVnZXI+Cgk8a2V5PmFEYXRlPC9rZXk+Cgk8ZGF0 | 
					
						
							|  |  |  |         ZT4yMDA0LTEwLTI2VDEwOjMzOjMzWjwvZGF0ZT4KCTxrZXk+YURpY3Q8L2tl | 
					
						
							|  |  |  |         eT4KCTxkaWN0PgoJCTxrZXk+YUZhbHNlVmFsdWU8L2tleT4KCQk8ZmFsc2Uv | 
					
						
							|  |  |  |         PgoJCTxrZXk+YVRydWVWYWx1ZTwva2V5PgoJCTx0cnVlLz4KCQk8a2V5PmFV | 
					
						
							|  |  |  |         bmljb2RlVmFsdWU8L2tleT4KCQk8c3RyaW5nPk3DpHNzaWcsIE1hw588L3N0 | 
					
						
							|  |  |  |         cmluZz4KCQk8a2V5PmFub3RoZXJTdHJpbmc8L2tleT4KCQk8c3RyaW5nPiZs | 
					
						
							|  |  |  |         dDtoZWxsbyAmYW1wOyAnaGknIHRoZXJlISZndDs8L3N0cmluZz4KCQk8a2V5 | 
					
						
							|  |  |  |         PmRlZXBlckRpY3Q8L2tleT4KCQk8ZGljdD4KCQkJPGtleT5hPC9rZXk+CgkJ | 
					
						
							|  |  |  |         CTxpbnRlZ2VyPjE3PC9pbnRlZ2VyPgoJCQk8a2V5PmI8L2tleT4KCQkJPHJl | 
					
						
							|  |  |  |         YWw+MzIuNTwvcmVhbD4KCQkJPGtleT5jPC9rZXk+CgkJCTxhcnJheT4KCQkJ | 
					
						
							|  |  |  |         CTxpbnRlZ2VyPjE8L2ludGVnZXI+CgkJCQk8aW50ZWdlcj4yPC9pbnRlZ2Vy | 
					
						
							|  |  |  |         PgoJCQkJPHN0cmluZz50ZXh0PC9zdHJpbmc+CgkJCTwvYXJyYXk+CgkJPC9k | 
					
						
							|  |  |  |         aWN0PgoJPC9kaWN0PgoJPGtleT5hRmxvYXQ8L2tleT4KCTxyZWFsPjAuNTwv | 
					
						
							|  |  |  |         cmVhbD4KCTxrZXk+YUxpc3Q8L2tleT4KCTxhcnJheT4KCQk8c3RyaW5nPkE8 | 
					
						
							|  |  |  |         L3N0cmluZz4KCQk8c3RyaW5nPkI8L3N0cmluZz4KCQk8aW50ZWdlcj4xMjwv | 
					
						
							|  |  |  |         aW50ZWdlcj4KCQk8cmVhbD4zMi41PC9yZWFsPgoJCTxhcnJheT4KCQkJPGlu | 
					
						
							|  |  |  |         dGVnZXI+MTwvaW50ZWdlcj4KCQkJPGludGVnZXI+MjwvaW50ZWdlcj4KCQkJ | 
					
						
							|  |  |  |         PGludGVnZXI+MzwvaW50ZWdlcj4KCQk8L2FycmF5PgoJPC9hcnJheT4KCTxr | 
					
						
							|  |  |  |         ZXk+YU5lZ2F0aXZlQmlnSW50PC9rZXk+Cgk8aW50ZWdlcj4tODAwMDAwMDAw | 
					
						
							|  |  |  |         MDA8L2ludGVnZXI+Cgk8a2V5PmFOZWdhdGl2ZUludDwva2V5PgoJPGludGVn | 
					
						
							|  |  |  |         ZXI+LTU8L2ludGVnZXI+Cgk8a2V5PmFTdHJpbmc8L2tleT4KCTxzdHJpbmc+ | 
					
						
							|  |  |  |         RG9vZGFoPC9zdHJpbmc+Cgk8a2V5PmFuRW1wdHlEaWN0PC9rZXk+Cgk8ZGlj | 
					
						
							|  |  |  |         dC8+Cgk8a2V5PmFuRW1wdHlMaXN0PC9rZXk+Cgk8YXJyYXkvPgoJPGtleT5h | 
					
						
							|  |  |  |         bkludDwva2V5PgoJPGludGVnZXI+NzI4PC9pbnRlZ2VyPgoJPGtleT5uZXN0 | 
					
						
							|  |  |  |         ZWREYXRhPC9rZXk+Cgk8YXJyYXk+CgkJPGRhdGE+CgkJUEd4dmRITWdiMlln | 
					
						
							| 
									
										
										
										
											2014-01-15 11:32:35 +01:00
										 |  |  |         WW1sdVlYSjVJR2QxYm1zK0FBRUNBenhzYjNSeklHOW1JR0pwYm1GeWVTQm5k | 
					
						
							| 
									
										
										
										
											2014-02-06 11:19:18 +01:00
										 |  |  |         VzVyCgkJUGdBQkFnTThiRzkwY3lCdlppQmlhVzVoY25rZ1ozVnVhejRBQVFJ | 
					
						
							|  |  |  |         RFBHeHZkSE1nYjJZZ1ltbHVZWEo1CgkJSUdkMWJtcytBQUVDQXp4c2IzUnpJ | 
					
						
							|  |  |  |         RzltSUdKcGJtRnllU0JuZFc1clBnQUJBZ004Ykc5MGN5QnZaaUJpCgkJYVc1 | 
					
						
							|  |  |  |         aGNua2daM1Z1YXo0QUFRSURQR3h2ZEhNZ2IyWWdZbWx1WVhKNUlHZDFibXMr | 
					
						
							|  |  |  |         QUFFQ0F6eHNiM1J6CgkJSUc5bUlHSnBibUZ5ZVNCbmRXNXJQZ0FCQWdNOGJH | 
					
						
							|  |  |  |         OTBjeUJ2WmlCaWFXNWhjbmtnWjNWdWF6NEFBUUlECgkJUEd4dmRITWdiMlln | 
					
						
							|  |  |  |         WW1sdVlYSjVJR2QxYm1zK0FBRUNBdz09CgkJPC9kYXRhPgoJPC9hcnJheT4K | 
					
						
							|  |  |  |         CTxrZXk+c29tZURhdGE8L2tleT4KCTxkYXRhPgoJUEdKcGJtRnllU0JuZFc1 | 
					
						
							|  |  |  |         clBnPT0KCTwvZGF0YT4KCTxrZXk+c29tZU1vcmVEYXRhPC9rZXk+Cgk8ZGF0 | 
					
						
							|  |  |  |         YT4KCVBHeHZkSE1nYjJZZ1ltbHVZWEo1SUdkMWJtcytBQUVDQXp4c2IzUnpJ | 
					
						
							|  |  |  |         RzltSUdKcGJtRnllU0JuZFc1clBnQUJBZ004CgliRzkwY3lCdlppQmlhVzVo | 
					
						
							|  |  |  |         Y25rZ1ozVnVhejRBQVFJRFBHeHZkSE1nYjJZZ1ltbHVZWEo1SUdkMWJtcytB | 
					
						
							|  |  |  |         QUVDQXp4cwoJYjNSeklHOW1JR0pwYm1GeWVTQm5kVzVyUGdBQkFnTThiRzkw | 
					
						
							|  |  |  |         Y3lCdlppQmlhVzVoY25rZ1ozVnVhejRBQVFJRFBHeHYKCWRITWdiMllnWW1s | 
					
						
							|  |  |  |         dVlYSjVJR2QxYm1zK0FBRUNBenhzYjNSeklHOW1JR0pwYm1GeWVTQm5kVzVy | 
					
						
							|  |  |  |         UGdBQkFnTThiRzkwCgljeUJ2WmlCaWFXNWhjbmtnWjNWdWF6NEFBUUlEUEd4 | 
					
						
							|  |  |  |         dmRITWdiMllnWW1sdVlYSjVJR2QxYm1zK0FBRUNBdz09Cgk8L2RhdGE+Cgk8 | 
					
						
							|  |  |  |         a2V5PsOFYmVucmFhPC9rZXk+Cgk8c3RyaW5nPlRoYXQgd2FzIGEgdW5pY29k | 
					
						
							|  |  |  |         ZSBrZXkuPC9zdHJpbmc+CjwvZGljdD4KPC9wbGlzdD4K'''),
 | 
					
						
							| 
									
										
										
										
											2013-11-21 15:46:49 +01:00
										 |  |  |     plistlib.FMT_BINARY: binascii.a2b_base64(b'''
 | 
					
						
							| 
									
										
										
										
											2014-02-06 11:19:18 +01:00
										 |  |  |         YnBsaXN0MDDfEBABAgMEBQYHCAkKCwwNDg8QERITFCgpLzAxMjM0NTc2OFdh | 
					
						
							|  |  |  |         QmlnSW50WGFCaWdJbnQyVWFEYXRlVWFEaWN0VmFGbG9hdFVhTGlzdF8QD2FO | 
					
						
							|  |  |  |         ZWdhdGl2ZUJpZ0ludFxhTmVnYXRpdmVJbnRXYVN0cmluZ1thbkVtcHR5RGlj | 
					
						
							|  |  |  |         dFthbkVtcHR5TGlzdFVhbkludFpuZXN0ZWREYXRhWHNvbWVEYXRhXHNvbWVN | 
					
						
							|  |  |  |         b3JlRGF0YWcAxQBiAGUAbgByAGEAYRN/////////1BQAAAAAAAAAAIAAAAAA | 
					
						
							|  |  |  |         AAAsM0GcuX30AAAA1RUWFxgZGhscHR5bYUZhbHNlVmFsdWVaYVRydWVWYWx1 | 
					
						
							|  |  |  |         ZV1hVW5pY29kZVZhbHVlXWFub3RoZXJTdHJpbmdaZGVlcGVyRGljdAgJawBN | 
					
						
							|  |  |  |         AOQAcwBzAGkAZwAsACAATQBhAN9fEBU8aGVsbG8gJiAnaGknIHRoZXJlIT7T | 
					
						
							|  |  |  |         HyAhIiMkUWFRYlFjEBEjQEBAAAAAAACjJSYnEAEQAlR0ZXh0Iz/gAAAAAAAA | 
					
						
							|  |  |  |         pSorLCMtUUFRQhAMoyUmLhADE////+1foOAAE//////////7VkRvb2RhaNCg | 
					
						
							|  |  |  |         EQLYoTZPEPo8bG90cyBvZiBiaW5hcnkgZ3Vuaz4AAQIDPGxvdHMgb2YgYmlu | 
					
						
							|  |  |  |         YXJ5IGd1bms+AAECAzxsb3RzIG9mIGJpbmFyeSBndW5rPgABAgM8bG90cyBv | 
					
						
							|  |  |  |         ZiBiaW5hcnkgZ3Vuaz4AAQIDPGxvdHMgb2YgYmluYXJ5IGd1bms+AAECAzxs | 
					
						
							|  |  |  |         b3RzIG9mIGJpbmFyeSBndW5rPgABAgM8bG90cyBvZiBiaW5hcnkgZ3Vuaz4A | 
					
						
							|  |  |  |         AQIDPGxvdHMgb2YgYmluYXJ5IGd1bms+AAECAzxsb3RzIG9mIGJpbmFyeSBn | 
					
						
							|  |  |  |         dW5rPgABAgM8bG90cyBvZiBiaW5hcnkgZ3Vuaz4AAQIDTTxiaW5hcnkgZ3Vu | 
					
						
							|  |  |  |         az5fEBdUaGF0IHdhcyBhIHVuaWNvZGUga2V5LgAIACsAMwA8AEIASABPAFUA | 
					
						
							|  |  |  |         ZwB0AHwAiACUAJoApQCuALsAygDTAOQA7QD4AQQBDwEdASsBNgE3ATgBTwFn | 
					
						
							|  |  |  |         AW4BcAFyAXQBdgF/AYMBhQGHAYwBlQGbAZ0BnwGhAaUBpwGwAbkBwAHBAcIB | 
					
						
							|  |  |  |         xQHHAsQC0gAAAAAAAAIBAAAAAAAAADkAAAAAAAAAAAAAAAAAAALs'''),
 | 
					
						
							| 
									
										
										
										
											2019-05-15 22:14:38 +02:00
										 |  |  |     'KEYED_ARCHIVE': binascii.a2b_base64(b'''
 | 
					
						
							|  |  |  |         YnBsaXN0MDDUAQIDBAUGHB1YJHZlcnNpb25YJG9iamVjdHNZJGFyY2hpdmVy | 
					
						
							|  |  |  |         VCR0b3ASAAGGoKMHCA9VJG51bGzTCQoLDA0OVnB5dHlwZVYkY2xhc3NZTlMu | 
					
						
							|  |  |  |         c3RyaW5nEAGAAl8QE0tleUFyY2hpdmUgVUlEIFRlc3TTEBESExQZWiRjbGFz | 
					
						
							|  |  |  |         c25hbWVYJGNsYXNzZXNbJGNsYXNzaGludHNfEBdPQ19CdWlsdGluUHl0aG9u | 
					
						
							|  |  |  |         VW5pY29kZaQVFhcYXxAXT0NfQnVpbHRpblB5dGhvblVuaWNvZGVfEBBPQ19Q | 
					
						
							|  |  |  |         eXRob25Vbmljb2RlWE5TU3RyaW5nWE5TT2JqZWN0ohobXxAPT0NfUHl0aG9u | 
					
						
							|  |  |  |         U3RyaW5nWE5TU3RyaW5nXxAPTlNLZXllZEFyY2hpdmVy0R4fVHJvb3SAAQAI | 
					
						
							|  |  |  |         ABEAGgAjAC0AMgA3ADsAQQBIAE8AVgBgAGIAZAB6AIEAjACVAKEAuwDAANoA | 
					
						
							|  |  |  |         7QD2AP8BAgEUAR0BLwEyATcAAAAAAAACAQAAAAAAAAAgAAAAAAAAAAAAAAAA | 
					
						
							|  |  |  |         AAABOQ=='''),
 | 
					
						
							| 
									
										
										
										
											2013-11-21 15:46:49 +01:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2004-10-26 11:02:08 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-19 20:13:49 +02:00
										 |  |  | XML_PLIST_WITH_ENTITY=b'''\
 | 
					
						
							|  |  |  | <?xml version="1.0" encoding="UTF-8"?> | 
					
						
							|  |  |  | <!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd" [ | 
					
						
							|  |  |  |    <!ENTITY entity "replacement text"> | 
					
						
							|  |  |  |   ]> | 
					
						
							|  |  |  | <plist version="1.0"> | 
					
						
							|  |  |  |   <dict> | 
					
						
							|  |  |  |     <key>A</key> | 
					
						
							|  |  |  |     <string>&entity;</string> | 
					
						
							|  |  |  |   </dict> | 
					
						
							|  |  |  | </plist> | 
					
						
							|  |  |  | '''
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-02 23:01:40 +02:00
										 |  |  | INVALID_BINARY_PLISTS = [ | 
					
						
							|  |  |  |     ('too short data', | 
					
						
							|  |  |  |         b'' | 
					
						
							|  |  |  |     ), | 
					
						
							|  |  |  |     ('too large offset_table_offset and offset_size = 1', | 
					
						
							|  |  |  |         b'\x00\x08' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x01\x01' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x01' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x00' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x2a' | 
					
						
							|  |  |  |     ), | 
					
						
							|  |  |  |     ('too large offset_table_offset and nonstandard offset_size', | 
					
						
							|  |  |  |         b'\x00\x00\x00\x08' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x03\x01' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x01' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x00' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x2c' | 
					
						
							|  |  |  |     ), | 
					
						
							|  |  |  |     ('integer overflow in offset_table_offset', | 
					
						
							|  |  |  |         b'\x00\x08' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x01\x01' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x01' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x00' | 
					
						
							|  |  |  |         b'\xff\xff\xff\xff\xff\xff\xff\xff' | 
					
						
							|  |  |  |     ), | 
					
						
							|  |  |  |     ('too large top_object', | 
					
						
							|  |  |  |         b'\x00\x08' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x01\x01' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x01' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x01' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x09' | 
					
						
							|  |  |  |     ), | 
					
						
							|  |  |  |     ('integer overflow in top_object', | 
					
						
							|  |  |  |         b'\x00\x08' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x01\x01' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x01' | 
					
						
							|  |  |  |         b'\xff\xff\xff\xff\xff\xff\xff\xff' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x09' | 
					
						
							|  |  |  |     ), | 
					
						
							|  |  |  |     ('too large num_objects and offset_size = 1', | 
					
						
							|  |  |  |         b'\x00\x08' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x01\x01' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\xff' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x00' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x09' | 
					
						
							|  |  |  |     ), | 
					
						
							|  |  |  |     ('too large num_objects and nonstandard offset_size', | 
					
						
							|  |  |  |         b'\x00\x00\x00\x08' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x03\x01' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\xff' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x00' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x09' | 
					
						
							|  |  |  |     ), | 
					
						
							|  |  |  |     ('extremally large num_objects (32 bit)', | 
					
						
							|  |  |  |         b'\x00\x08' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x01\x01' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x7f\xff\xff\xff' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x00' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x09' | 
					
						
							|  |  |  |     ), | 
					
						
							|  |  |  |     ('extremally large num_objects (64 bit)', | 
					
						
							|  |  |  |         b'\x00\x08' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x01\x01' | 
					
						
							|  |  |  |         b'\x00\x00\x00\xff\xff\xff\xff\xff' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x00' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x09' | 
					
						
							|  |  |  |     ), | 
					
						
							|  |  |  |     ('integer overflow in num_objects', | 
					
						
							|  |  |  |         b'\x00\x08' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x01\x01' | 
					
						
							|  |  |  |         b'\xff\xff\xff\xff\xff\xff\xff\xff' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x00' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x09' | 
					
						
							|  |  |  |     ), | 
					
						
							|  |  |  |     ('offset_size = 0', | 
					
						
							|  |  |  |         b'\x00\x08' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x01' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x01' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x00' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x09' | 
					
						
							|  |  |  |     ), | 
					
						
							|  |  |  |     ('ref_size = 0', | 
					
						
							|  |  |  |         b'\xa1\x01\x00\x08\x0a' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x01\x00' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x02' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x00' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x0b' | 
					
						
							|  |  |  |     ), | 
					
						
							|  |  |  |     ('too large offset', | 
					
						
							|  |  |  |         b'\x00\x2a' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x01\x01' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x01' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x00' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x09' | 
					
						
							|  |  |  |     ), | 
					
						
							|  |  |  |     ('integer overflow in offset', | 
					
						
							|  |  |  |         b'\x00\xff\xff\xff\xff\xff\xff\xff\xff' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x08\x01' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x01' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x00' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x09' | 
					
						
							|  |  |  |     ), | 
					
						
							|  |  |  |     ('too large array size', | 
					
						
							|  |  |  |         b'\xaf\x00\x01\xff\x00\x08\x0c' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x01\x01' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x02' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x00' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x0d' | 
					
						
							|  |  |  |     ), | 
					
						
							|  |  |  |     ('extremally large array size (32-bit)', | 
					
						
							|  |  |  |         b'\xaf\x02\x7f\xff\xff\xff\x01\x00\x08\x0f' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x01\x01' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x02' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x00' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x10' | 
					
						
							|  |  |  |     ), | 
					
						
							|  |  |  |     ('extremally large array size (64-bit)', | 
					
						
							|  |  |  |         b'\xaf\x03\x00\x00\x00\xff\xff\xff\xff\xff\x01\x00\x08\x13' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x01\x01' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x02' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x00' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x14' | 
					
						
							|  |  |  |     ), | 
					
						
							|  |  |  |     ('integer overflow in array size', | 
					
						
							|  |  |  |         b'\xaf\x03\xff\xff\xff\xff\xff\xff\xff\xff\x01\x00\x08\x13' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x01\x01' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x02' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x00' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x14' | 
					
						
							|  |  |  |     ), | 
					
						
							|  |  |  |     ('too large reference index', | 
					
						
							|  |  |  |         b'\xa1\x02\x00\x08\x0a' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x01\x01' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x02' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x00' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x0b' | 
					
						
							|  |  |  |     ), | 
					
						
							|  |  |  |     ('integer overflow in reference index', | 
					
						
							|  |  |  |         b'\xa1\xff\xff\xff\xff\xff\xff\xff\xff\x00\x08\x11' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x01\x08' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x02' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x00' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x12' | 
					
						
							|  |  |  |     ), | 
					
						
							|  |  |  |     ('too large bytes size', | 
					
						
							|  |  |  |         b'\x4f\x00\x23\x41\x08' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x01\x01' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x01' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x00' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x0c' | 
					
						
							|  |  |  |     ), | 
					
						
							|  |  |  |     ('extremally large bytes size (32-bit)', | 
					
						
							|  |  |  |         b'\x4f\x02\x7f\xff\xff\xff\x41\x08' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x01\x01' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x01' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x00' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x0f' | 
					
						
							|  |  |  |     ), | 
					
						
							|  |  |  |     ('extremally large bytes size (64-bit)', | 
					
						
							|  |  |  |         b'\x4f\x03\x00\x00\x00\xff\xff\xff\xff\xff\x41\x08' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x01\x01' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x01' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x00' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x13' | 
					
						
							|  |  |  |     ), | 
					
						
							|  |  |  |     ('integer overflow in bytes size', | 
					
						
							|  |  |  |         b'\x4f\x03\xff\xff\xff\xff\xff\xff\xff\xff\x41\x08' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x01\x01' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x01' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x00' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x13' | 
					
						
							|  |  |  |     ), | 
					
						
							|  |  |  |     ('too large ASCII size', | 
					
						
							|  |  |  |         b'\x5f\x00\x23\x41\x08' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x01\x01' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x01' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x00' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x0c' | 
					
						
							|  |  |  |     ), | 
					
						
							|  |  |  |     ('extremally large ASCII size (32-bit)', | 
					
						
							|  |  |  |         b'\x5f\x02\x7f\xff\xff\xff\x41\x08' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x01\x01' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x01' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x00' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x0f' | 
					
						
							|  |  |  |     ), | 
					
						
							|  |  |  |     ('extremally large ASCII size (64-bit)', | 
					
						
							|  |  |  |         b'\x5f\x03\x00\x00\x00\xff\xff\xff\xff\xff\x41\x08' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x01\x01' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x01' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x00' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x13' | 
					
						
							|  |  |  |     ), | 
					
						
							|  |  |  |     ('integer overflow in ASCII size', | 
					
						
							|  |  |  |         b'\x5f\x03\xff\xff\xff\xff\xff\xff\xff\xff\x41\x08' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x01\x01' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x01' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x00' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x13' | 
					
						
							|  |  |  |     ), | 
					
						
							|  |  |  |     ('invalid ASCII', | 
					
						
							|  |  |  |         b'\x51\xff\x08' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x01\x01' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x01' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x00' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x0a' | 
					
						
							|  |  |  |     ), | 
					
						
							|  |  |  |     ('too large UTF-16 size', | 
					
						
							|  |  |  |         b'\x6f\x00\x13\x20\xac\x00\x08' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x01\x01' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x01' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x00' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x0e' | 
					
						
							|  |  |  |     ), | 
					
						
							|  |  |  |     ('extremally large UTF-16 size (32-bit)', | 
					
						
							|  |  |  |         b'\x6f\x02\x4f\xff\xff\xff\x20\xac\x00\x08' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x01\x01' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x01' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x00' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x11' | 
					
						
							|  |  |  |     ), | 
					
						
							|  |  |  |     ('extremally large UTF-16 size (64-bit)', | 
					
						
							|  |  |  |         b'\x6f\x03\x00\x00\x00\xff\xff\xff\xff\xff\x20\xac\x00\x08' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x01\x01' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x01' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x00' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x15' | 
					
						
							|  |  |  |     ), | 
					
						
							|  |  |  |     ('integer overflow in UTF-16 size', | 
					
						
							|  |  |  |         b'\x6f\x03\xff\xff\xff\xff\xff\xff\xff\xff\x20\xac\x00\x08' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x01\x01' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x01' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x00' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x15' | 
					
						
							|  |  |  |     ), | 
					
						
							|  |  |  |     ('invalid UTF-16', | 
					
						
							|  |  |  |         b'\x61\xd8\x00\x08' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x01\x01' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x01' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x00' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x0b' | 
					
						
							|  |  |  |     ), | 
					
						
							|  |  |  |     ('non-hashable key', | 
					
						
							|  |  |  |         b'\xd1\x01\x01\xa0\x08\x0b' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x01\x01' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x02' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x00' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x0c' | 
					
						
							|  |  |  |     ), | 
					
						
							|  |  |  |     ('too large datetime (datetime overflow)', | 
					
						
							|  |  |  |         b'\x33\x42\x50\x00\x00\x00\x00\x00\x00\x08' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x01\x01' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x01' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x00' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x11' | 
					
						
							|  |  |  |     ), | 
					
						
							|  |  |  |     ('too large datetime (timedelta overflow)', | 
					
						
							|  |  |  |         b'\x33\x42\xe0\x00\x00\x00\x00\x00\x00\x08' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x01\x01' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x01' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x00' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x11' | 
					
						
							|  |  |  |     ), | 
					
						
							|  |  |  |     ('invalid datetime (Infinity)', | 
					
						
							|  |  |  |         b'\x33\x7f\xf0\x00\x00\x00\x00\x00\x00\x08' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x01\x01' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x01' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x00' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x11' | 
					
						
							|  |  |  |     ), | 
					
						
							|  |  |  |     ('invalid datetime (NaN)', | 
					
						
							|  |  |  |         b'\x33\x7f\xf8\x00\x00\x00\x00\x00\x00\x08' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x01\x01' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x01' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x00' | 
					
						
							|  |  |  |         b'\x00\x00\x00\x00\x00\x00\x00\x11' | 
					
						
							|  |  |  |     ), | 
					
						
							|  |  |  | ] | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-10-26 11:02:08 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-02-25 12:58:58 +00:00
										 |  |  | class TestPlistlib(unittest.TestCase): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def tearDown(self): | 
					
						
							|  |  |  |         try: | 
					
						
							| 
									
										
										
										
											2020-07-09 21:25:10 +08:00
										 |  |  |             os.unlink(os_helper.TESTFN) | 
					
						
							| 
									
										
										
										
											2003-02-25 12:58:58 +00:00
										 |  |  |         except: | 
					
						
							|  |  |  |             pass | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-21 15:46:49 +01:00
										 |  |  |     def _create(self, fmt=None): | 
					
						
							| 
									
										
										
										
											2004-10-26 11:02:08 +00:00
										 |  |  |         pl = dict( | 
					
						
							| 
									
										
										
										
											2003-02-25 12:58:58 +00:00
										 |  |  |             aString="Doodah", | 
					
						
							| 
									
										
										
										
											2004-10-26 11:02:08 +00:00
										 |  |  |             aList=["A", "B", 12, 32.5, [1, 2, 3]], | 
					
						
							|  |  |  |             aFloat = 0.5, | 
					
						
							| 
									
										
										
										
											2003-02-25 12:58:58 +00:00
										 |  |  |             anInt = 728, | 
					
						
							| 
									
										
										
										
											2014-01-15 11:32:35 +01:00
										 |  |  |             aBigInt = 2 ** 63 - 44, | 
					
						
							| 
									
										
										
										
											2014-02-06 11:19:18 +01:00
										 |  |  |             aBigInt2 = 2 ** 63 + 44, | 
					
						
							| 
									
										
										
										
											2014-01-15 11:32:35 +01:00
										 |  |  |             aNegativeInt = -5, | 
					
						
							|  |  |  |             aNegativeBigInt = -80000000000, | 
					
						
							| 
									
										
										
										
											2004-10-26 11:02:08 +00:00
										 |  |  |             aDict=dict( | 
					
						
							|  |  |  |                 anotherString="<hello & 'hi' there!>", | 
					
						
							| 
									
										
										
										
											2007-05-02 19:09:54 +00:00
										 |  |  |                 aUnicodeValue='M\xe4ssig, Ma\xdf', | 
					
						
							| 
									
										
										
										
											2003-02-25 12:58:58 +00:00
										 |  |  |                 aTrueValue=True, | 
					
						
							|  |  |  |                 aFalseValue=False, | 
					
						
							| 
									
										
										
										
											2004-10-26 11:02:08 +00:00
										 |  |  |                 deeperDict=dict(a=17, b=32.5, c=[1, 2, "text"]), | 
					
						
							| 
									
										
										
										
											2003-02-25 12:58:58 +00:00
										 |  |  |             ), | 
					
						
							| 
									
										
										
										
											2013-11-21 15:46:49 +01:00
										 |  |  |             someData = b"<binary gunk>", | 
					
						
							|  |  |  |             someMoreData = b"<lots of binary gunk>\0\1\2\3" * 10, | 
					
						
							|  |  |  |             nestedData = [b"<lots of binary gunk>\0\1\2\3" * 10], | 
					
						
							| 
									
										
										
										
											2004-10-26 11:02:08 +00:00
										 |  |  |             aDate = datetime.datetime(2004, 10, 26, 10, 33, 33), | 
					
						
							| 
									
										
										
										
											2012-05-29 12:04:54 +02:00
										 |  |  |             anEmptyDict = dict(), | 
					
						
							|  |  |  |             anEmptyList = list() | 
					
						
							| 
									
										
										
										
											2003-02-25 12:58:58 +00:00
										 |  |  |         ) | 
					
						
							| 
									
										
										
										
											2007-05-02 19:09:54 +00:00
										 |  |  |         pl['\xc5benraa'] = "That was a unicode key." | 
					
						
							| 
									
										
										
										
											2003-02-25 12:58:58 +00:00
										 |  |  |         return pl | 
					
						
							| 
									
										
										
										
											2003-03-07 17:30:48 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-02-25 12:58:58 +00:00
										 |  |  |     def test_create(self): | 
					
						
							|  |  |  |         pl = self._create() | 
					
						
							|  |  |  |         self.assertEqual(pl["aString"], "Doodah") | 
					
						
							|  |  |  |         self.assertEqual(pl["aDict"]["aFalseValue"], False) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_io(self): | 
					
						
							|  |  |  |         pl = self._create() | 
					
						
							| 
									
										
										
										
											2020-07-09 21:25:10 +08:00
										 |  |  |         with open(os_helper.TESTFN, 'wb') as fp: | 
					
						
							| 
									
										
										
										
											2013-11-21 15:46:49 +01:00
										 |  |  |             plistlib.dump(pl, fp) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-09 21:25:10 +08:00
										 |  |  |         with open(os_helper.TESTFN, 'rb') as fp: | 
					
						
							| 
									
										
										
										
											2013-11-21 15:46:49 +01:00
										 |  |  |             pl2 = plistlib.load(fp) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-03-07 17:30:48 +00:00
										 |  |  |         self.assertEqual(dict(pl), dict(pl2)) | 
					
						
							| 
									
										
										
										
											2003-02-25 12:58:58 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-21 15:46:49 +01:00
										 |  |  |         self.assertRaises(AttributeError, plistlib.dump, pl, 'filename') | 
					
						
							|  |  |  |         self.assertRaises(AttributeError, plistlib.load, 'filename') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-15 11:32:35 +01:00
										 |  |  |     def test_invalid_type(self): | 
					
						
							|  |  |  |         pl = [ object() ] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for fmt in ALL_FORMATS: | 
					
						
							|  |  |  |             with self.subTest(fmt=fmt): | 
					
						
							|  |  |  |                 self.assertRaises(TypeError, plistlib.dumps, pl, fmt=fmt) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-15 22:14:38 +02:00
										 |  |  |     def test_invalid_uid(self): | 
					
						
							|  |  |  |         with self.assertRaises(TypeError): | 
					
						
							|  |  |  |             UID("not an int") | 
					
						
							|  |  |  |         with self.assertRaises(ValueError): | 
					
						
							|  |  |  |             UID(2 ** 64) | 
					
						
							|  |  |  |         with self.assertRaises(ValueError): | 
					
						
							|  |  |  |             UID(-19) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-15 11:32:35 +01:00
										 |  |  |     def test_int(self): | 
					
						
							|  |  |  |         for pl in [0, 2**8-1, 2**8, 2**16-1, 2**16, 2**32-1, 2**32, | 
					
						
							| 
									
										
										
										
											2014-02-06 11:19:18 +01:00
										 |  |  |                    2**63-1, 2**64-1, 1, -2**63]: | 
					
						
							| 
									
										
										
										
											2014-01-15 11:32:35 +01:00
										 |  |  |             for fmt in ALL_FORMATS: | 
					
						
							|  |  |  |                 with self.subTest(pl=pl, fmt=fmt): | 
					
						
							|  |  |  |                     data = plistlib.dumps(pl, fmt=fmt) | 
					
						
							|  |  |  |                     pl2 = plistlib.loads(data) | 
					
						
							|  |  |  |                     self.assertIsInstance(pl2, int) | 
					
						
							|  |  |  |                     self.assertEqual(pl, pl2) | 
					
						
							|  |  |  |                     data2 = plistlib.dumps(pl2, fmt=fmt) | 
					
						
							|  |  |  |                     self.assertEqual(data, data2) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for fmt in ALL_FORMATS: | 
					
						
							|  |  |  |             for pl in (2 ** 64 + 1, 2 ** 127-1, -2**64, -2 ** 127): | 
					
						
							|  |  |  |                 with self.subTest(pl=pl, fmt=fmt): | 
					
						
							|  |  |  |                     self.assertRaises(OverflowError, plistlib.dumps, | 
					
						
							|  |  |  |                                       pl, fmt=fmt) | 
					
						
							| 
									
										
										
										
											2013-11-21 15:46:49 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-30 23:26:11 +02:00
										 |  |  |     def test_bytearray(self): | 
					
						
							|  |  |  |         for pl in (b'<binary gunk>', b"<lots of binary gunk>\0\1\2\3" * 10): | 
					
						
							|  |  |  |             for fmt in ALL_FORMATS: | 
					
						
							|  |  |  |                 with self.subTest(pl=pl, fmt=fmt): | 
					
						
							|  |  |  |                     data = plistlib.dumps(bytearray(pl), fmt=fmt) | 
					
						
							|  |  |  |                     pl2 = plistlib.loads(data) | 
					
						
							|  |  |  |                     self.assertIsInstance(pl2, bytes) | 
					
						
							|  |  |  |                     self.assertEqual(pl2, pl) | 
					
						
							|  |  |  |                     data2 = plistlib.dumps(pl2, fmt=fmt) | 
					
						
							|  |  |  |                     self.assertEqual(data, data2) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-08-07 14:26:40 +00:00
										 |  |  |     def test_bytes(self): | 
					
						
							| 
									
										
										
										
											2004-10-26 11:02:08 +00:00
										 |  |  |         pl = self._create() | 
					
						
							| 
									
										
										
										
											2013-11-21 15:46:49 +01:00
										 |  |  |         data = plistlib.dumps(pl) | 
					
						
							|  |  |  |         pl2 = plistlib.loads(data) | 
					
						
							| 
									
										
										
										
											2004-10-26 11:02:08 +00:00
										 |  |  |         self.assertEqual(dict(pl), dict(pl2)) | 
					
						
							| 
									
										
										
										
											2013-11-21 15:46:49 +01:00
										 |  |  |         data2 = plistlib.dumps(pl2) | 
					
						
							| 
									
										
										
										
											2004-10-26 11:02:08 +00:00
										 |  |  |         self.assertEqual(data, data2) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-23 13:47:22 +02:00
										 |  |  |     def test_indentation_array(self): | 
					
						
							| 
									
										
										
										
											2013-11-21 15:46:49 +01:00
										 |  |  |         data = [[[[[[[[{'test': b'aaaaaa'}]]]]]]]] | 
					
						
							|  |  |  |         self.assertEqual(plistlib.loads(plistlib.dumps(data)), data) | 
					
						
							| 
									
										
										
										
											2013-04-23 13:47:22 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def test_indentation_dict(self): | 
					
						
							| 
									
										
										
										
											2013-11-21 15:46:49 +01:00
										 |  |  |         data = {'1': {'2': {'3': {'4': {'5': {'6': {'7': {'8': {'9': b'aaaaaa'}}}}}}}}} | 
					
						
							|  |  |  |         self.assertEqual(plistlib.loads(plistlib.dumps(data)), data) | 
					
						
							| 
									
										
										
										
											2013-04-23 13:47:22 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def test_indentation_dict_mix(self): | 
					
						
							| 
									
										
										
										
											2013-11-21 15:46:49 +01:00
										 |  |  |         data = {'1': {'2': [{'3': [[[[[{'test': b'aaaaaa'}]]]]]}]}} | 
					
						
							|  |  |  |         self.assertEqual(plistlib.loads(plistlib.dumps(data)), data) | 
					
						
							| 
									
										
										
										
											2013-04-23 13:47:22 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-15 22:14:38 +02:00
										 |  |  |     def test_uid(self): | 
					
						
							|  |  |  |         data = UID(1) | 
					
						
							|  |  |  |         self.assertEqual(plistlib.loads(plistlib.dumps(data, fmt=plistlib.FMT_BINARY)), data) | 
					
						
							|  |  |  |         dict_data = { | 
					
						
							|  |  |  |             'uid0': UID(0), | 
					
						
							|  |  |  |             'uid2': UID(2), | 
					
						
							|  |  |  |             'uid8': UID(2 ** 8), | 
					
						
							|  |  |  |             'uid16': UID(2 ** 16), | 
					
						
							|  |  |  |             'uid32': UID(2 ** 32), | 
					
						
							|  |  |  |             'uid63': UID(2 ** 63) | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         self.assertEqual(plistlib.loads(plistlib.dumps(dict_data, fmt=plistlib.FMT_BINARY)), dict_data) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_uid_data(self): | 
					
						
							|  |  |  |         uid = UID(1) | 
					
						
							|  |  |  |         self.assertEqual(uid.data, 1) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_uid_eq(self): | 
					
						
							|  |  |  |         self.assertEqual(UID(1), UID(1)) | 
					
						
							|  |  |  |         self.assertNotEqual(UID(1), UID(2)) | 
					
						
							|  |  |  |         self.assertNotEqual(UID(1), "not uid") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_uid_hash(self): | 
					
						
							|  |  |  |         self.assertEqual(hash(UID(1)), hash(UID(1))) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_uid_repr(self): | 
					
						
							|  |  |  |         self.assertEqual(repr(UID(1)), "UID(1)") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_uid_index(self): | 
					
						
							|  |  |  |         self.assertEqual(operator.index(UID(1)), 1) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_uid_pickle(self): | 
					
						
							|  |  |  |         for proto in range(pickle.HIGHEST_PROTOCOL + 1): | 
					
						
							|  |  |  |             self.assertEqual(pickle.loads(pickle.dumps(UID(19), protocol=proto)), UID(19)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_uid_copy(self): | 
					
						
							|  |  |  |         self.assertEqual(copy.copy(UID(1)), UID(1)) | 
					
						
							|  |  |  |         self.assertEqual(copy.deepcopy(UID(1)), UID(1)) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-10-26 11:02:08 +00:00
										 |  |  |     def test_appleformatting(self): | 
					
						
							| 
									
										
										
										
											2019-09-05 03:11:35 -05:00
										 |  |  |         for fmt in ALL_FORMATS: | 
					
						
							|  |  |  |             with self.subTest(fmt=fmt): | 
					
						
							|  |  |  |                 pl = plistlib.loads(TESTDATA[fmt]) | 
					
						
							|  |  |  |                 data = plistlib.dumps(pl, fmt=fmt) | 
					
						
							|  |  |  |                 self.assertEqual(data, TESTDATA[fmt], | 
					
						
							|  |  |  |                     "generated data was not identical to Apple's output") | 
					
						
							| 
									
										
										
										
											2013-11-21 15:46:49 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-10-26 11:02:08 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def test_appleformattingfromliteral(self): | 
					
						
							| 
									
										
										
										
											2013-11-21 15:46:49 +01:00
										 |  |  |         self.maxDiff = None | 
					
						
							|  |  |  |         for fmt in ALL_FORMATS: | 
					
						
							|  |  |  |             with self.subTest(fmt=fmt): | 
					
						
							|  |  |  |                 pl = self._create(fmt=fmt) | 
					
						
							| 
									
										
										
										
											2014-07-23 18:49:31 +03:00
										 |  |  |                 pl2 = plistlib.loads(TESTDATA[fmt], fmt=fmt) | 
					
						
							|  |  |  |                 self.assertEqual(dict(pl), dict(pl2), | 
					
						
							|  |  |  |                     "generated data was not identical to Apple's output") | 
					
						
							| 
									
										
										
										
											2013-11-21 15:46:49 +01:00
										 |  |  |                 pl2 = plistlib.loads(TESTDATA[fmt]) | 
					
						
							|  |  |  |                 self.assertEqual(dict(pl), dict(pl2), | 
					
						
							|  |  |  |                     "generated data was not identical to Apple's output") | 
					
						
							| 
									
										
										
										
											2004-10-26 11:02:08 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-08-07 14:26:40 +00:00
										 |  |  |     def test_bytesio(self): | 
					
						
							| 
									
										
										
										
											2013-11-21 15:46:49 +01:00
										 |  |  |         for fmt in ALL_FORMATS: | 
					
						
							|  |  |  |             with self.subTest(fmt=fmt): | 
					
						
							|  |  |  |                 b = BytesIO() | 
					
						
							|  |  |  |                 pl = self._create(fmt=fmt) | 
					
						
							|  |  |  |                 plistlib.dump(pl, b, fmt=fmt) | 
					
						
							| 
									
										
										
										
											2014-07-23 18:49:31 +03:00
										 |  |  |                 pl2 = plistlib.load(BytesIO(b.getvalue()), fmt=fmt) | 
					
						
							|  |  |  |                 self.assertEqual(dict(pl), dict(pl2)) | 
					
						
							| 
									
										
										
										
											2013-11-21 15:46:49 +01:00
										 |  |  |                 pl2 = plistlib.load(BytesIO(b.getvalue())) | 
					
						
							|  |  |  |                 self.assertEqual(dict(pl), dict(pl2)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_keysort_bytesio(self): | 
					
						
							|  |  |  |         pl = collections.OrderedDict() | 
					
						
							|  |  |  |         pl['b'] = 1 | 
					
						
							|  |  |  |         pl['a'] = 2 | 
					
						
							|  |  |  |         pl['c'] = 3 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for fmt in ALL_FORMATS: | 
					
						
							|  |  |  |             for sort_keys in (False, True): | 
					
						
							|  |  |  |                 with self.subTest(fmt=fmt, sort_keys=sort_keys): | 
					
						
							|  |  |  |                     b = BytesIO() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                     plistlib.dump(pl, b, fmt=fmt, sort_keys=sort_keys) | 
					
						
							|  |  |  |                     pl2 = plistlib.load(BytesIO(b.getvalue()), | 
					
						
							|  |  |  |                         dict_type=collections.OrderedDict) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                     self.assertEqual(dict(pl), dict(pl2)) | 
					
						
							|  |  |  |                     if sort_keys: | 
					
						
							|  |  |  |                         self.assertEqual(list(pl2.keys()), ['a', 'b', 'c']) | 
					
						
							|  |  |  |                     else: | 
					
						
							|  |  |  |                         self.assertEqual(list(pl2.keys()), ['b', 'a', 'c']) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_keysort(self): | 
					
						
							|  |  |  |         pl = collections.OrderedDict() | 
					
						
							|  |  |  |         pl['b'] = 1 | 
					
						
							|  |  |  |         pl['a'] = 2 | 
					
						
							|  |  |  |         pl['c'] = 3 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for fmt in ALL_FORMATS: | 
					
						
							|  |  |  |             for sort_keys in (False, True): | 
					
						
							|  |  |  |                 with self.subTest(fmt=fmt, sort_keys=sort_keys): | 
					
						
							|  |  |  |                     data = plistlib.dumps(pl, fmt=fmt, sort_keys=sort_keys) | 
					
						
							|  |  |  |                     pl2 = plistlib.loads(data, dict_type=collections.OrderedDict) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                     self.assertEqual(dict(pl), dict(pl2)) | 
					
						
							|  |  |  |                     if sort_keys: | 
					
						
							|  |  |  |                         self.assertEqual(list(pl2.keys()), ['a', 'b', 'c']) | 
					
						
							|  |  |  |                     else: | 
					
						
							|  |  |  |                         self.assertEqual(list(pl2.keys()), ['b', 'a', 'c']) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_keys_no_string(self): | 
					
						
							|  |  |  |         pl = { 42: 'aNumber' } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for fmt in ALL_FORMATS: | 
					
						
							|  |  |  |             with self.subTest(fmt=fmt): | 
					
						
							|  |  |  |                 self.assertRaises(TypeError, plistlib.dumps, pl, fmt=fmt) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 b = BytesIO() | 
					
						
							|  |  |  |                 self.assertRaises(TypeError, plistlib.dump, pl, b, fmt=fmt) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_skipkeys(self): | 
					
						
							|  |  |  |         pl = { | 
					
						
							|  |  |  |             42: 'aNumber', | 
					
						
							|  |  |  |             'snake': 'aWord', | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for fmt in ALL_FORMATS: | 
					
						
							|  |  |  |             with self.subTest(fmt=fmt): | 
					
						
							|  |  |  |                 data = plistlib.dumps( | 
					
						
							|  |  |  |                     pl, fmt=fmt, skipkeys=True, sort_keys=False) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 pl2 = plistlib.loads(data) | 
					
						
							|  |  |  |                 self.assertEqual(pl2, {'snake': 'aWord'}) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 fp = BytesIO() | 
					
						
							|  |  |  |                 plistlib.dump( | 
					
						
							|  |  |  |                     pl, fp, fmt=fmt, skipkeys=True, sort_keys=False) | 
					
						
							|  |  |  |                 data = fp.getvalue() | 
					
						
							|  |  |  |                 pl2 = plistlib.loads(fp.getvalue()) | 
					
						
							|  |  |  |                 self.assertEqual(pl2, {'snake': 'aWord'}) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_tuple_members(self): | 
					
						
							|  |  |  |         pl = { | 
					
						
							|  |  |  |             'first': (1, 2), | 
					
						
							|  |  |  |             'second': (1, 2), | 
					
						
							|  |  |  |             'third': (3, 4), | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for fmt in ALL_FORMATS: | 
					
						
							|  |  |  |             with self.subTest(fmt=fmt): | 
					
						
							|  |  |  |                 data = plistlib.dumps(pl, fmt=fmt) | 
					
						
							|  |  |  |                 pl2 = plistlib.loads(data) | 
					
						
							|  |  |  |                 self.assertEqual(pl2, { | 
					
						
							|  |  |  |                     'first': [1, 2], | 
					
						
							|  |  |  |                     'second': [1, 2], | 
					
						
							|  |  |  |                     'third': [3, 4], | 
					
						
							|  |  |  |                 }) | 
					
						
							| 
									
										
										
										
											2017-12-12 19:03:08 +02:00
										 |  |  |                 if fmt != plistlib.FMT_BINARY: | 
					
						
							|  |  |  |                     self.assertIsNot(pl2['first'], pl2['second']) | 
					
						
							| 
									
										
										
										
											2013-11-21 15:46:49 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def test_list_members(self): | 
					
						
							|  |  |  |         pl = { | 
					
						
							|  |  |  |             'first': [1, 2], | 
					
						
							|  |  |  |             'second': [1, 2], | 
					
						
							|  |  |  |             'third': [3, 4], | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for fmt in ALL_FORMATS: | 
					
						
							|  |  |  |             with self.subTest(fmt=fmt): | 
					
						
							|  |  |  |                 data = plistlib.dumps(pl, fmt=fmt) | 
					
						
							|  |  |  |                 pl2 = plistlib.loads(data) | 
					
						
							|  |  |  |                 self.assertEqual(pl2, { | 
					
						
							|  |  |  |                     'first': [1, 2], | 
					
						
							|  |  |  |                     'second': [1, 2], | 
					
						
							|  |  |  |                     'third': [3, 4], | 
					
						
							|  |  |  |                 }) | 
					
						
							|  |  |  |                 self.assertIsNot(pl2['first'], pl2['second']) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_dict_members(self): | 
					
						
							|  |  |  |         pl = { | 
					
						
							|  |  |  |             'first': {'a': 1}, | 
					
						
							|  |  |  |             'second': {'a': 1}, | 
					
						
							|  |  |  |             'third': {'b': 2 }, | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for fmt in ALL_FORMATS: | 
					
						
							|  |  |  |             with self.subTest(fmt=fmt): | 
					
						
							|  |  |  |                 data = plistlib.dumps(pl, fmt=fmt) | 
					
						
							|  |  |  |                 pl2 = plistlib.loads(data) | 
					
						
							|  |  |  |                 self.assertEqual(pl2, { | 
					
						
							|  |  |  |                     'first': {'a': 1}, | 
					
						
							|  |  |  |                     'second': {'a': 1}, | 
					
						
							|  |  |  |                     'third': {'b': 2 }, | 
					
						
							|  |  |  |                 }) | 
					
						
							|  |  |  |                 self.assertIsNot(pl2['first'], pl2['second']) | 
					
						
							| 
									
										
										
										
											2003-07-01 20:22:30 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-11-12 08:34:32 +00:00
										 |  |  |     def test_controlcharacters(self): | 
					
						
							| 
									
										
										
										
											2004-11-12 09:36:12 +00:00
										 |  |  |         for i in range(128): | 
					
						
							|  |  |  |             c = chr(i) | 
					
						
							|  |  |  |             testString = "string containing %s" % c | 
					
						
							|  |  |  |             if i >= 32 or c in "\r\n\t": | 
					
						
							|  |  |  |                 # \r, \n and \t are the only legal control chars in XML | 
					
						
							| 
									
										
										
										
											2017-10-31 14:05:53 +02:00
										 |  |  |                 data = plistlib.dumps(testString, fmt=plistlib.FMT_XML) | 
					
						
							|  |  |  |                 if c != "\r": | 
					
						
							|  |  |  |                     self.assertEqual(plistlib.loads(data), testString) | 
					
						
							| 
									
										
										
										
											2004-11-12 09:36:12 +00:00
										 |  |  |             else: | 
					
						
							| 
									
										
										
										
											2017-10-31 14:05:53 +02:00
										 |  |  |                 with self.assertRaises(ValueError): | 
					
						
							|  |  |  |                     plistlib.dumps(testString, fmt=plistlib.FMT_XML) | 
					
						
							|  |  |  |             plistlib.dumps(testString, fmt=plistlib.FMT_BINARY) | 
					
						
							| 
									
										
										
										
											2004-11-12 08:34:32 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-04 20:04:30 +03:00
										 |  |  |     def test_non_bmp_characters(self): | 
					
						
							|  |  |  |         pl = {'python': '\U0001f40d'} | 
					
						
							|  |  |  |         for fmt in ALL_FORMATS: | 
					
						
							|  |  |  |             with self.subTest(fmt=fmt): | 
					
						
							|  |  |  |                 data = plistlib.dumps(pl, fmt=fmt) | 
					
						
							|  |  |  |                 self.assertEqual(plistlib.loads(data), pl) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-31 14:05:53 +02:00
										 |  |  |     def test_lone_surrogates(self): | 
					
						
							|  |  |  |         for fmt in ALL_FORMATS: | 
					
						
							|  |  |  |             with self.subTest(fmt=fmt): | 
					
						
							|  |  |  |                 with self.assertRaises(UnicodeEncodeError): | 
					
						
							|  |  |  |                     plistlib.dumps('\ud8ff', fmt=fmt) | 
					
						
							|  |  |  |                 with self.assertRaises(UnicodeEncodeError): | 
					
						
							|  |  |  |                     plistlib.dumps('\udcff', fmt=fmt) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-11-12 08:34:32 +00:00
										 |  |  |     def test_nondictroot(self): | 
					
						
							| 
									
										
										
										
											2013-11-21 15:46:49 +01:00
										 |  |  |         for fmt in ALL_FORMATS: | 
					
						
							|  |  |  |             with self.subTest(fmt=fmt): | 
					
						
							|  |  |  |                 test1 = "abc" | 
					
						
							|  |  |  |                 test2 = [1, 2, 3, "abc"] | 
					
						
							|  |  |  |                 result1 = plistlib.loads(plistlib.dumps(test1, fmt=fmt)) | 
					
						
							|  |  |  |                 result2 = plistlib.loads(plistlib.dumps(test2, fmt=fmt)) | 
					
						
							|  |  |  |                 self.assertEqual(test1, result1) | 
					
						
							|  |  |  |                 self.assertEqual(test2, result2) | 
					
						
							| 
									
										
										
										
											2003-02-25 12:58:58 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-05-28 02:19:19 -07:00
										 |  |  |     def test_invalidarray(self): | 
					
						
							|  |  |  |         for i in ["<key>key inside an array</key>", | 
					
						
							|  |  |  |                   "<key>key inside an array2</key><real>3</real>", | 
					
						
							|  |  |  |                   "<true/><key>key inside an array3</key>"]: | 
					
						
							| 
									
										
										
										
											2013-11-21 15:46:49 +01:00
										 |  |  |             self.assertRaises(ValueError, plistlib.loads, | 
					
						
							| 
									
										
										
										
											2011-05-28 02:19:19 -07:00
										 |  |  |                               ("<plist><array>%s</array></plist>"%i).encode()) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_invaliddict(self): | 
					
						
							|  |  |  |         for i in ["<key><true/>k</key><string>compound key</string>", | 
					
						
							|  |  |  |                   "<key>single key</key>", | 
					
						
							|  |  |  |                   "<string>missing key</string>", | 
					
						
							|  |  |  |                   "<key>k1</key><string>v1</string><real>5.3</real>" | 
					
						
							|  |  |  |                   "<key>k1</key><key>k2</key><string>double key</string>"]: | 
					
						
							| 
									
										
										
										
											2013-11-21 15:46:49 +01:00
										 |  |  |             self.assertRaises(ValueError, plistlib.loads, | 
					
						
							| 
									
										
										
										
											2011-05-28 02:19:19 -07:00
										 |  |  |                               ("<plist><dict>%s</dict></plist>"%i).encode()) | 
					
						
							| 
									
										
										
										
											2013-11-21 15:46:49 +01:00
										 |  |  |             self.assertRaises(ValueError, plistlib.loads, | 
					
						
							| 
									
										
										
										
											2011-05-28 02:19:19 -07:00
										 |  |  |                               ("<plist><array><dict>%s</dict></array></plist>"%i).encode()) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_invalidinteger(self): | 
					
						
							| 
									
										
										
										
											2013-11-21 15:46:49 +01:00
										 |  |  |         self.assertRaises(ValueError, plistlib.loads, | 
					
						
							| 
									
										
										
										
											2011-05-28 02:19:19 -07:00
										 |  |  |                           b"<plist><integer>not integer</integer></plist>") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_invalidreal(self): | 
					
						
							| 
									
										
										
										
											2013-11-21 15:46:49 +01:00
										 |  |  |         self.assertRaises(ValueError, plistlib.loads, | 
					
						
							| 
									
										
										
										
											2011-05-28 02:19:19 -07:00
										 |  |  |                           b"<plist><integer>not real</integer></plist>") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-20 09:26:33 +02:00
										 |  |  |     def test_integer_notations(self): | 
					
						
							|  |  |  |         pl = b"<plist><integer>456</integer></plist>" | 
					
						
							|  |  |  |         value = plistlib.loads(pl) | 
					
						
							|  |  |  |         self.assertEqual(value, 456) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         pl = b"<plist><integer>0xa</integer></plist>" | 
					
						
							|  |  |  |         value = plistlib.loads(pl) | 
					
						
							|  |  |  |         self.assertEqual(value, 10) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         pl = b"<plist><integer>0123</integer></plist>" | 
					
						
							|  |  |  |         value = plistlib.loads(pl) | 
					
						
							|  |  |  |         self.assertEqual(value, 123) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-21 15:46:49 +01:00
										 |  |  |     def test_xml_encodings(self): | 
					
						
							|  |  |  |         base = TESTDATA[plistlib.FMT_XML] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for xml_encoding, encoding, bom in [ | 
					
						
							|  |  |  |                     (b'utf-8', 'utf-8', codecs.BOM_UTF8), | 
					
						
							|  |  |  |                     (b'utf-16', 'utf-16-le', codecs.BOM_UTF16_LE), | 
					
						
							|  |  |  |                     (b'utf-16', 'utf-16-be', codecs.BOM_UTF16_BE), | 
					
						
							|  |  |  |                     # Expat does not support UTF-32 | 
					
						
							|  |  |  |                     #(b'utf-32', 'utf-32-le', codecs.BOM_UTF32_LE), | 
					
						
							|  |  |  |                     #(b'utf-32', 'utf-32-be', codecs.BOM_UTF32_BE), | 
					
						
							|  |  |  |                 ]: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             pl = self._create(fmt=plistlib.FMT_XML) | 
					
						
							|  |  |  |             with self.subTest(encoding=encoding): | 
					
						
							|  |  |  |                 data = base.replace(b'UTF-8', xml_encoding) | 
					
						
							|  |  |  |                 data = bom + data.decode('utf-8').encode(encoding) | 
					
						
							|  |  |  |                 pl2 = plistlib.loads(data) | 
					
						
							|  |  |  |                 self.assertEqual(dict(pl), dict(pl2)) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-01 11:45:01 -05:00
										 |  |  |     def test_dump_invalid_format(self): | 
					
						
							|  |  |  |         with self.assertRaises(ValueError): | 
					
						
							|  |  |  |             plistlib.dumps({}, fmt="blah") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_load_invalid_file(self): | 
					
						
							|  |  |  |         with self.assertRaises(plistlib.InvalidFileException): | 
					
						
							|  |  |  |             plistlib.loads(b"these are not plist file contents") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_modified_uid_negative(self): | 
					
						
							|  |  |  |         neg_uid = UID(1) | 
					
						
							|  |  |  |         neg_uid.data = -1  # dodge the negative check in the constructor | 
					
						
							|  |  |  |         with self.assertRaises(ValueError): | 
					
						
							|  |  |  |             plistlib.dumps(neg_uid, fmt=plistlib.FMT_BINARY) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_modified_uid_huge(self): | 
					
						
							|  |  |  |         huge_uid = UID(1) | 
					
						
							|  |  |  |         huge_uid.data = 2 ** 64  # dodge the size check in the constructor | 
					
						
							|  |  |  |         with self.assertRaises(OverflowError): | 
					
						
							|  |  |  |             plistlib.dumps(huge_uid, fmt=plistlib.FMT_BINARY) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-19 20:13:49 +02:00
										 |  |  |     def test_xml_plist_with_entity_decl(self): | 
					
						
							|  |  |  |         with self.assertRaisesRegex(plistlib.InvalidFileException, | 
					
						
							|  |  |  |                                     "XML entity declarations are not supported"): | 
					
						
							|  |  |  |             plistlib.loads(XML_PLIST_WITH_ENTITY, fmt=plistlib.FMT_XML) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-02 02:51:24 +08:00
										 |  |  |     def test_load_aware_datetime(self): | 
					
						
							|  |  |  |         dt = plistlib.loads(b"<plist><date>2023-12-10T08:03:30Z</date></plist>", | 
					
						
							|  |  |  |                             aware_datetime=True) | 
					
						
							|  |  |  |         self.assertEqual(dt.tzinfo, datetime.UTC) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @unittest.skipUnless("America/Los_Angeles" in zoneinfo.available_timezones(), | 
					
						
							|  |  |  |                          "Can't find timezone datebase") | 
					
						
							|  |  |  |     def test_dump_aware_datetime(self): | 
					
						
							|  |  |  |         dt = datetime.datetime(2345, 6, 7, 8, 9, 10, | 
					
						
							|  |  |  |                                tzinfo=zoneinfo.ZoneInfo("America/Los_Angeles")) | 
					
						
							|  |  |  |         for fmt in ALL_FORMATS: | 
					
						
							|  |  |  |             s = plistlib.dumps(dt, fmt=fmt, aware_datetime=True) | 
					
						
							|  |  |  |             loaded_dt = plistlib.loads(s, fmt=fmt, aware_datetime=True) | 
					
						
							|  |  |  |             self.assertEqual(loaded_dt.tzinfo, datetime.UTC) | 
					
						
							|  |  |  |             self.assertEqual(loaded_dt, dt) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_dump_utc_aware_datetime(self): | 
					
						
							|  |  |  |         dt = datetime.datetime(2345, 6, 7, 8, 9, 10, tzinfo=datetime.UTC) | 
					
						
							|  |  |  |         for fmt in ALL_FORMATS: | 
					
						
							|  |  |  |             s = plistlib.dumps(dt, fmt=fmt, aware_datetime=True) | 
					
						
							|  |  |  |             loaded_dt = plistlib.loads(s, fmt=fmt, aware_datetime=True) | 
					
						
							|  |  |  |             self.assertEqual(loaded_dt.tzinfo, datetime.UTC) | 
					
						
							|  |  |  |             self.assertEqual(loaded_dt, dt) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @unittest.skipUnless("America/Los_Angeles" in zoneinfo.available_timezones(), | 
					
						
							|  |  |  |                          "Can't find timezone datebase") | 
					
						
							|  |  |  |     def test_dump_aware_datetime_without_aware_datetime_option(self): | 
					
						
							|  |  |  |         dt = datetime.datetime(2345, 6, 7, 8, | 
					
						
							|  |  |  |                                tzinfo=zoneinfo.ZoneInfo("America/Los_Angeles")) | 
					
						
							|  |  |  |         s = plistlib.dumps(dt, fmt=plistlib.FMT_XML, aware_datetime=False) | 
					
						
							|  |  |  |         self.assertIn(b"2345-06-07T08:00:00Z", s) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_dump_utc_aware_datetime_without_aware_datetime_option(self): | 
					
						
							|  |  |  |         dt = datetime.datetime(2345, 6, 7, 8, tzinfo=datetime.UTC) | 
					
						
							|  |  |  |         s = plistlib.dumps(dt, fmt=plistlib.FMT_XML, aware_datetime=False) | 
					
						
							|  |  |  |         self.assertIn(b"2345-06-07T08:00:00Z", s) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_dump_naive_datetime_with_aware_datetime_option(self): | 
					
						
							|  |  |  |         # Save a naive datetime with aware_datetime set to true.  This will lead | 
					
						
							|  |  |  |         # to having different time as compared to the current machine's | 
					
						
							|  |  |  |         # timezone, which is UTC. | 
					
						
							| 
									
										
										
										
											2024-01-01 21:31:43 +01:00
										 |  |  |         dt = datetime.datetime(2003, 6, 7, 8, tzinfo=None) | 
					
						
							| 
									
										
										
										
											2024-01-02 02:51:24 +08:00
										 |  |  |         for fmt in ALL_FORMATS: | 
					
						
							|  |  |  |             s = plistlib.dumps(dt, fmt=fmt, aware_datetime=True) | 
					
						
							|  |  |  |             parsed = plistlib.loads(s, aware_datetime=False) | 
					
						
							| 
									
										
										
										
											2024-01-02 21:45:36 +02:00
										 |  |  |             expected = dt.astimezone(datetime.UTC).replace(tzinfo=None) | 
					
						
							|  |  |  |             self.assertEqual(parsed, expected) | 
					
						
							| 
									
										
										
										
											2024-01-02 02:51:24 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-30 23:26:11 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | class TestBinaryPlistlib(unittest.TestCase): | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-02 23:01:40 +02:00
										 |  |  |     @staticmethod | 
					
						
							|  |  |  |     def decode(*objects, offset_size=1, ref_size=1): | 
					
						
							|  |  |  |         data = [b'bplist00'] | 
					
						
							|  |  |  |         offset = 8 | 
					
						
							|  |  |  |         offsets = [] | 
					
						
							|  |  |  |         for x in objects: | 
					
						
							|  |  |  |             offsets.append(offset.to_bytes(offset_size, 'big')) | 
					
						
							|  |  |  |             data.append(x) | 
					
						
							|  |  |  |             offset += len(x) | 
					
						
							|  |  |  |         tail = struct.pack('>6xBBQQQ', offset_size, ref_size, | 
					
						
							|  |  |  |                            len(objects), 0, offset) | 
					
						
							|  |  |  |         data.extend(offsets) | 
					
						
							|  |  |  |         data.append(tail) | 
					
						
							|  |  |  |         return plistlib.loads(b''.join(data), fmt=plistlib.FMT_BINARY) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-23 16:13:33 +03:00
										 |  |  |     def test_nonstandard_refs_size(self): | 
					
						
							|  |  |  |         # Issue #21538: Refs and offsets are 24-bit integers | 
					
						
							|  |  |  |         data = (b'bplist00' | 
					
						
							|  |  |  |                 b'\xd1\x00\x00\x01\x00\x00\x02QaQb' | 
					
						
							|  |  |  |                 b'\x00\x00\x08\x00\x00\x0f\x00\x00\x11' | 
					
						
							|  |  |  |                 b'\x00\x00\x00\x00\x00\x00' | 
					
						
							|  |  |  |                 b'\x03\x03' | 
					
						
							|  |  |  |                 b'\x00\x00\x00\x00\x00\x00\x00\x03' | 
					
						
							|  |  |  |                 b'\x00\x00\x00\x00\x00\x00\x00\x00' | 
					
						
							|  |  |  |                 b'\x00\x00\x00\x00\x00\x00\x00\x13') | 
					
						
							|  |  |  |         self.assertEqual(plistlib.loads(data), {'a': 'b'}) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-30 23:26:11 +02:00
										 |  |  |     def test_dump_duplicates(self): | 
					
						
							|  |  |  |         # Test effectiveness of saving duplicated objects | 
					
						
							| 
									
										
										
										
											2020-11-02 23:01:40 +02:00
										 |  |  |         for x in (None, False, True, 12345, 123.45, 'abcde', 'абвгд', b'abcde', | 
					
						
							| 
									
										
										
										
											2017-11-30 23:26:11 +02:00
										 |  |  |                   datetime.datetime(2004, 10, 26, 10, 33, 33), | 
					
						
							| 
									
										
										
										
											2019-09-05 03:11:35 -05:00
										 |  |  |                   bytearray(b'abcde'), [12, 345], (12, 345), {'12': 345}): | 
					
						
							| 
									
										
										
										
											2017-11-30 23:26:11 +02:00
										 |  |  |             with self.subTest(x=x): | 
					
						
							|  |  |  |                 data = plistlib.dumps([x]*1000, fmt=plistlib.FMT_BINARY) | 
					
						
							|  |  |  |                 self.assertLess(len(data), 1100, repr(data)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_identity(self): | 
					
						
							|  |  |  |         for x in (None, False, True, 12345, 123.45, 'abcde', b'abcde', | 
					
						
							|  |  |  |                   datetime.datetime(2004, 10, 26, 10, 33, 33), | 
					
						
							| 
									
										
										
										
											2019-09-05 03:11:35 -05:00
										 |  |  |                   bytearray(b'abcde'), [12, 345], (12, 345), {'12': 345}): | 
					
						
							| 
									
										
										
										
											2017-11-30 23:26:11 +02:00
										 |  |  |             with self.subTest(x=x): | 
					
						
							|  |  |  |                 data = plistlib.dumps([x]*2, fmt=plistlib.FMT_BINARY) | 
					
						
							|  |  |  |                 a, b = plistlib.loads(data) | 
					
						
							|  |  |  |                 if isinstance(x, tuple): | 
					
						
							|  |  |  |                     x = list(x) | 
					
						
							|  |  |  |                 self.assertEqual(a, x) | 
					
						
							|  |  |  |                 self.assertEqual(b, x) | 
					
						
							|  |  |  |                 self.assertIs(a, b) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_cycles(self): | 
					
						
							|  |  |  |         # recursive list | 
					
						
							|  |  |  |         a = [] | 
					
						
							|  |  |  |         a.append(a) | 
					
						
							|  |  |  |         b = plistlib.loads(plistlib.dumps(a, fmt=plistlib.FMT_BINARY)) | 
					
						
							|  |  |  |         self.assertIs(b[0], b) | 
					
						
							|  |  |  |         # recursive tuple | 
					
						
							|  |  |  |         a = ([],) | 
					
						
							|  |  |  |         a[0].append(a) | 
					
						
							|  |  |  |         b = plistlib.loads(plistlib.dumps(a, fmt=plistlib.FMT_BINARY)) | 
					
						
							|  |  |  |         self.assertIs(b[0][0], b) | 
					
						
							|  |  |  |         # recursive dict | 
					
						
							|  |  |  |         a = {} | 
					
						
							|  |  |  |         a['x'] = a | 
					
						
							|  |  |  |         b = plistlib.loads(plistlib.dumps(a, fmt=plistlib.FMT_BINARY)) | 
					
						
							|  |  |  |         self.assertIs(b['x'], b) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-02 23:01:40 +02:00
										 |  |  |     def test_deep_nesting(self): | 
					
						
							|  |  |  |         for N in [300, 100000]: | 
					
						
							|  |  |  |             chunks = [b'\xa1' + (i + 1).to_bytes(4, 'big') for i in range(N)] | 
					
						
							|  |  |  |             try: | 
					
						
							|  |  |  |                 result = self.decode(*chunks, b'\x54seed', offset_size=4, ref_size=4) | 
					
						
							|  |  |  |             except RecursionError: | 
					
						
							|  |  |  |                 pass | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 for i in range(N): | 
					
						
							|  |  |  |                     self.assertIsInstance(result, list) | 
					
						
							|  |  |  |                     self.assertEqual(len(result), 1) | 
					
						
							|  |  |  |                     result = result[0] | 
					
						
							|  |  |  |                 self.assertEqual(result, 'seed') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-08 15:00:02 +03:00
										 |  |  |     def test_large_timestamp(self): | 
					
						
							|  |  |  |         # Issue #26709: 32-bit timestamp out of range | 
					
						
							|  |  |  |         for ts in -2**31-1, 2**31: | 
					
						
							|  |  |  |             with self.subTest(ts=ts): | 
					
						
							| 
									
										
										
										
											2023-04-27 11:32:30 -06:00
										 |  |  |                 d = (datetime.datetime(1970, 1, 1, 0, 0) + | 
					
						
							| 
									
										
										
										
											2016-04-08 15:00:02 +03:00
										 |  |  |                      datetime.timedelta(seconds=ts)) | 
					
						
							|  |  |  |                 data = plistlib.dumps(d, fmt=plistlib.FMT_BINARY) | 
					
						
							|  |  |  |                 self.assertEqual(plistlib.loads(data), d) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-02 23:01:40 +02:00
										 |  |  |     def test_load_singletons(self): | 
					
						
							|  |  |  |         self.assertIs(self.decode(b'\x00'), None) | 
					
						
							|  |  |  |         self.assertIs(self.decode(b'\x08'), False) | 
					
						
							|  |  |  |         self.assertIs(self.decode(b'\x09'), True) | 
					
						
							|  |  |  |         self.assertEqual(self.decode(b'\x0f'), b'') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_load_int(self): | 
					
						
							|  |  |  |         self.assertEqual(self.decode(b'\x10\x00'), 0) | 
					
						
							|  |  |  |         self.assertEqual(self.decode(b'\x10\xfe'), 0xfe) | 
					
						
							|  |  |  |         self.assertEqual(self.decode(b'\x11\xfe\xdc'), 0xfedc) | 
					
						
							|  |  |  |         self.assertEqual(self.decode(b'\x12\xfe\xdc\xba\x98'), 0xfedcba98) | 
					
						
							|  |  |  |         self.assertEqual(self.decode(b'\x13\x01\x23\x45\x67\x89\xab\xcd\xef'), | 
					
						
							|  |  |  |                          0x0123456789abcdef) | 
					
						
							|  |  |  |         self.assertEqual(self.decode(b'\x13\xfe\xdc\xba\x98\x76\x54\x32\x10'), | 
					
						
							|  |  |  |                          -0x123456789abcdf0) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_unsupported(self): | 
					
						
							|  |  |  |         unsupported = [*range(1, 8), *range(10, 15), | 
					
						
							|  |  |  |                        0x20, 0x21, *range(0x24, 0x33), *range(0x34, 0x40)] | 
					
						
							|  |  |  |         for i in [0x70, 0x90, 0xb0, 0xc0, 0xe0, 0xf0]: | 
					
						
							|  |  |  |             unsupported.extend(i + j for j in range(16)) | 
					
						
							|  |  |  |         for token in unsupported: | 
					
						
							|  |  |  |             with self.subTest(f'token {token:02x}'): | 
					
						
							|  |  |  |                 with self.assertRaises(plistlib.InvalidFileException): | 
					
						
							|  |  |  |                     self.decode(bytes([token]) + b'\x00'*16) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-31 14:05:53 +02:00
										 |  |  |     def test_invalid_binary(self): | 
					
						
							| 
									
										
										
										
											2020-11-02 23:01:40 +02:00
										 |  |  |         for name, data in INVALID_BINARY_PLISTS: | 
					
						
							|  |  |  |             with self.subTest(name): | 
					
						
							|  |  |  |                 with self.assertRaises(plistlib.InvalidFileException): | 
					
						
							|  |  |  |                     plistlib.loads(b'bplist00' + data, fmt=plistlib.FMT_BINARY) | 
					
						
							| 
									
										
										
										
											2017-10-31 14:05:53 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-02 02:51:24 +08:00
										 |  |  |     def test_load_aware_datetime(self): | 
					
						
							|  |  |  |         data = (b'bplist003B\x04>\xd0d\x00\x00\x00\x08\x00\x00\x00\x00\x00\x00' | 
					
						
							|  |  |  |                 b'\x01\x01\x00\x00\x00\x00\x00\x00\x00\x01\x00\x00\x00\x00\x00' | 
					
						
							|  |  |  |                 b'\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x11') | 
					
						
							|  |  |  |         self.assertEqual(plistlib.loads(data, aware_datetime=True), | 
					
						
							|  |  |  |                          datetime.datetime(2345, 6, 7, 8, tzinfo=datetime.UTC)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @unittest.skipUnless("America/Los_Angeles" in zoneinfo.available_timezones(), | 
					
						
							|  |  |  |                          "Can't find timezone datebase") | 
					
						
							|  |  |  |     def test_dump_aware_datetime_without_aware_datetime_option(self): | 
					
						
							|  |  |  |         dt = datetime.datetime(2345, 6, 7, 8, | 
					
						
							|  |  |  |                                tzinfo=zoneinfo.ZoneInfo("America/Los_Angeles")) | 
					
						
							|  |  |  |         msg = "can't subtract offset-naive and offset-aware datetimes" | 
					
						
							|  |  |  |         with self.assertRaisesRegex(TypeError, msg): | 
					
						
							|  |  |  |             plistlib.dumps(dt, fmt=plistlib.FMT_BINARY, aware_datetime=False) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_dump_utc_aware_datetime_without_aware_datetime_option(self): | 
					
						
							|  |  |  |         dt = datetime.datetime(2345, 6, 7, 8, tzinfo=datetime.UTC) | 
					
						
							|  |  |  |         msg = "can't subtract offset-naive and offset-aware datetimes" | 
					
						
							|  |  |  |         with self.assertRaisesRegex(TypeError, msg): | 
					
						
							|  |  |  |             plistlib.dumps(dt, fmt=plistlib.FMT_BINARY, aware_datetime=False) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-21 15:46:49 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-15 22:14:38 +02:00
										 |  |  | class TestKeyedArchive(unittest.TestCase): | 
					
						
							|  |  |  |     def test_keyed_archive_data(self): | 
					
						
							|  |  |  |         # This is the structure of a NSKeyedArchive packed plist | 
					
						
							|  |  |  |         data = { | 
					
						
							|  |  |  |             '$version': 100000, | 
					
						
							|  |  |  |             '$objects': [ | 
					
						
							|  |  |  |                 '$null', { | 
					
						
							|  |  |  |                     'pytype': 1, | 
					
						
							|  |  |  |                     '$class': UID(2), | 
					
						
							|  |  |  |                     'NS.string': 'KeyArchive UID Test' | 
					
						
							|  |  |  |                 }, | 
					
						
							|  |  |  |                 { | 
					
						
							|  |  |  |                     '$classname': 'OC_BuiltinPythonUnicode', | 
					
						
							|  |  |  |                     '$classes': [ | 
					
						
							|  |  |  |                         'OC_BuiltinPythonUnicode', | 
					
						
							|  |  |  |                         'OC_PythonUnicode', | 
					
						
							|  |  |  |                         'NSString', | 
					
						
							|  |  |  |                         'NSObject' | 
					
						
							|  |  |  |                     ], | 
					
						
							|  |  |  |                     '$classhints': [ | 
					
						
							|  |  |  |                         'OC_PythonString', 'NSString' | 
					
						
							|  |  |  |                     ] | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             ], | 
					
						
							|  |  |  |             '$archiver': 'NSKeyedArchiver', | 
					
						
							|  |  |  |             '$top': { | 
					
						
							|  |  |  |                 'root': UID(1) | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         self.assertEqual(plistlib.loads(TESTDATA["KEYED_ARCHIVE"]), data) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-06 02:00:50 +00:00
										 |  |  | class MiscTestCase(unittest.TestCase): | 
					
						
							|  |  |  |     def test__all__(self): | 
					
						
							| 
									
										
										
										
											2020-08-17 07:20:40 +02:00
										 |  |  |         not_exported = {"PlistFormat", "PLISTHEADER"} | 
					
						
							|  |  |  |         support.check__all__(self, plistlib, not_exported=not_exported) | 
					
						
							| 
									
										
										
										
											2016-06-06 02:00:50 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-17 14:11:04 +04:00
										 |  |  | @unittest.skipUnless(sys.platform == "darwin", "plutil utility is for Mac os") | 
					
						
							|  |  |  | class TestPlutil(unittest.TestCase): | 
					
						
							|  |  |  |     file_name = "plutil_test.plist" | 
					
						
							|  |  |  |     properties = { | 
					
						
							|  |  |  |             "fname" : "H", | 
					
						
							|  |  |  |             "lname":"A", | 
					
						
							|  |  |  |             "marks" : {"a":100, "b":0x10} | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     exptected_properties = { | 
					
						
							|  |  |  |         "fname" : "H", | 
					
						
							|  |  |  |         "lname": "A", | 
					
						
							|  |  |  |         "marks" : {"a":100, "b":16} | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     pl = { | 
					
						
							|  |  |  |             "HexType" : 0x0100000c, | 
					
						
							|  |  |  |             "IntType" : 0o123 | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @classmethod | 
					
						
							|  |  |  |     def setUpClass(cls) -> None: | 
					
						
							|  |  |  |         ## Generate plist file with plistlib and parse with plutil | 
					
						
							|  |  |  |         with open(cls.file_name,'wb') as f: | 
					
						
							|  |  |  |             plistlib.dump(cls.properties, f, fmt=plistlib.FMT_BINARY) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @classmethod | 
					
						
							|  |  |  |     def tearDownClass(cls) -> None: | 
					
						
							|  |  |  |         os.remove(cls.file_name) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def get_lint_status(self): | 
					
						
							|  |  |  |         return subprocess.run(['plutil', "-lint", self.file_name], capture_output=True, text=True).stdout | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def convert_to_json(self): | 
					
						
							|  |  |  |         """Convert binary file to json using plutil
 | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         subprocess.run(['plutil', "-convert", 'json', self.file_name]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def convert_to_bin(self): | 
					
						
							|  |  |  |         """Convert file to binary using plutil
 | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         subprocess.run(['plutil', "-convert", 'binary1', self.file_name]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def write_pl(self): | 
					
						
							|  |  |  |         """Write Hex properties to file using writePlist
 | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         with open(self.file_name, 'wb') as f: | 
					
						
							|  |  |  |             plistlib.dump(self.pl, f, fmt=plistlib.FMT_BINARY) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_lint_status(self): | 
					
						
							|  |  |  |         # check lint status of file using plutil | 
					
						
							|  |  |  |         self.assertEqual(f"{self.file_name}: OK\n", self.get_lint_status()) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def check_content(self): | 
					
						
							|  |  |  |         # check file content with plutil converting binary to json | 
					
						
							|  |  |  |         self.convert_to_json() | 
					
						
							|  |  |  |         with open(self.file_name) as f: | 
					
						
							|  |  |  |             ff = json.loads(f.read()) | 
					
						
							|  |  |  |             self.assertEqual(ff, self.exptected_properties) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def check_plistlib_parse(self): | 
					
						
							|  |  |  |         # Generate plist files with plutil and parse with plistlib | 
					
						
							|  |  |  |         self.convert_to_bin() | 
					
						
							|  |  |  |         with open(self.file_name, 'rb') as f: | 
					
						
							|  |  |  |             self.assertEqual(plistlib.load(f), self.exptected_properties) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_octal_and_hex(self): | 
					
						
							|  |  |  |         self.write_pl() | 
					
						
							|  |  |  |         self.convert_to_json() | 
					
						
							|  |  |  |         with open(self.file_name, 'r') as f: | 
					
						
							|  |  |  |             p = json.loads(f.read()) | 
					
						
							|  |  |  |             self.assertEqual(p.get("HexType"), 16777228) | 
					
						
							|  |  |  |             self.assertEqual(p.get("IntType"), 83) | 
					
						
							| 
									
										
										
										
											2016-06-06 02:00:50 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-02 02:51:24 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-02-25 12:58:58 +00:00
										 |  |  | if __name__ == '__main__': | 
					
						
							| 
									
										
										
										
											2019-11-01 11:45:01 -05:00
										 |  |  |     unittest.main() |