Skip to main content

Convert NSKeyedArchiver plist into a deserialized human readable plist

Project description

NSKeyedArchive plist deserializer

Deserializes NSKeyedArchiver created plists, which are frequent in macOS/iOS. These are serialized versions of plists (or data classes) and are meant for machine reading. The deserialized version is human readable for analysts and investigators who need to review the data.

The library recursively deserializes the entire plist and returns a dictionary/list object representing the entire plist. Certain NSKeyedArchiver plists contain circular references, which results in infinite looping. The code detects and breaks these loops wherever found to return useable data.

New: From version 1.4.0, you can pass any plist, even if not an NSKA. If the option full_recurse_convert_nska=True is used, it will recurse any plist, and find and convert all nested NSKA from any NS.data field found. Also new in 1.5.0 is the ability to get a dictionary returned as the top level plist object (instead of list)

Requirements: Python 3.6+ (3.8 or higher recommended)

Due to improvements in the built-in plistlib library in Python 3.8, it is recommended to use 3.8 or above. For 3.7 or lower, it should work fine for most plists, some might fail to save correctly. If you don't care about saving the deserialized plist (using the built-in library functions), then this should make no difference.

Installation (via pip/pip3)

pip3 install nska_deserialize

Usage

Use the functions deserialize_plist or deserialize_plist_from_string to convert NSKeyedArchives (NSKA).

By default full_recurse_convert_nska=False maintaining old behaviour which is to throw an exception if the archive is not NSKA, and no recursive processing for nested NSKA data blobs. If set to True, then it will process any plist even if not an NSKA (at root level).

By default format=list maintains old behaviour providing a list as the top level of the plist. If format=dict is used, this switches to a dictionary.

From a file
import nska_deserialize as nd

input_path = '/Users/yogesh/Desktop/sample.sfl2'

with open(input_path, 'rb') as f:
    try:
        deserialized_plist = nd.deserialize_plist(f, full_recurse_convert_nska=True, format=dict)
        print(deserialized_plist)
    except (nd.DeserializeError, 
            nd.biplist.NotBinaryPlistException, 
            nd.biplist.InvalidPlistException,
            nd.plistlib.InvalidFileException,
            nd.ccl_bplist.BplistError, 
            ValueError, 
            TypeError, OSError, OverflowError) as ex:
        # These are all possible errors from libraries imported

        print('Had exception: ' + str(ex))
        deserialized_plist = None

    if deserialized_plist:
        output_path_plist = input_path + '_deserialized.plist'
        output_path_json  = input_path + '_deserialized.json'

        nd.write_plist_to_json_file(deserialized_plist, output_path_json)
        nd.write_plist_to_file(deserialized_plist, output_path_plist)
From a String
import nska_deserialize as nd

plist_in_string = b"{notional plist as string that might have come from a database}"

try:
    deserialized_plist = nd.deserialize_plist_from_string(plist_in_string, full_recurse_convert_nska=True, format=dict)
    print(deserialized_plist)
except (nd.DeserializeError, 
        nd.biplist.NotBinaryPlistException, 
        nd.biplist.InvalidPlistException,
        nd.plistlib.InvalidFileException,
        nd.ccl_bplist.BplistError, 
        ValueError, 
        TypeError, OSError, OverflowError) as ex:
    # These are all possible errors from libraries imported

    print('Had exception: ' + str(ex))
    deserialized_plist = None

if deserialized_plist:
    output_path_plist = input_path + '_deserialized.plist'
    output_path_json  = input_path + '_deserialized.json'

    nd.write_plist_to_json_file(deserialized_plist, output_path_json)
    nd.write_plist_to_file(deserialized_plist, output_path_plist)

Change log

v1.5.1
Minor bug fix - Empty NSKeyedArchive will not raise an exception if it is valid.

v1.5.0
Minor bug fix - If root level element was NULL, this would create an invalid plist as None/NULL values are not allowed in plists. This now converts it to a blank string.
New option format=dict will return a dictionary at the top level of a plist instead of the default list format (Thanks @cvandeplas - Christophe Vandeplas).

v1.4.0
New boolean option full_recurse_convert_nska for full recursion of plist and conversion of all nested NSKeyedArchive data.

v1.3.3
Fixes an issue with CF$UID conversion, this was not being applied to all plists resulting in empty output for certain plists.
Python 3.12 compatible and tested.

v1.3.2
Adds NSUUID type to ccl_bplist, which should remove at least some exceptions related to unhashable type: 'NsKeyedArchiverDictionary'.

v1.3.1
Python 3.9 compatible (earlier versions of library may have problems with XML plists on python 3.9).

v1.2
Support for macOS Big Sur plists, some have hexadecimal integers in XML, which caused problems with underlying plist parsers.

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

nska_deserialize-1.5.1.tar.gz (14.2 kB view details)

Uploaded Source

Built Distribution

nska_deserialize-1.5.1-py3-none-any.whl (13.3 kB view details)

Uploaded Python 3

File details

Details for the file nska_deserialize-1.5.1.tar.gz.

File metadata

  • Download URL: nska_deserialize-1.5.1.tar.gz
  • Upload date:
  • Size: 14.2 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/5.1.1 CPython/3.12.3

File hashes

Hashes for nska_deserialize-1.5.1.tar.gz
Algorithm Hash digest
SHA256 a8040fdce19c54673c08bfdba9a3bc9eb1ec519b06b9575f227440ba78ec0809
MD5 afc97afda1a83354a32c627d7bbb1565
BLAKE2b-256 bce431b373daac149996eac5cca3967f9b8f0a9506fa59a0fd4f360037009b3b

See more details on using hashes here.

File details

Details for the file nska_deserialize-1.5.1-py3-none-any.whl.

File metadata

File hashes

Hashes for nska_deserialize-1.5.1-py3-none-any.whl
Algorithm Hash digest
SHA256 ef8bf253f2f8cf5d73e56ac5315e090cbc8614ef3c769ff2a86ff341a5214977
MD5 82014b2117468baf4200dc93c510de5d
BLAKE2b-256 4899dda67703b590584d0a303cf64875befedc0ff51b994d11e016eed9f9af5a

See more details on using hashes here.

Supported by

AWS AWS Cloud computing and Security Sponsor Datadog Datadog Monitoring Fastly Fastly CDN Google Google Download Analytics Microsoft Microsoft PSF Sponsor Pingdom Pingdom Monitoring Sentry Sentry Error logging StatusPage StatusPage Status page