The repository has moved to https://github.com/pycrate-org/pycrate and is now archived.
Pycrate is a french word for qualifying bad wine (when it's close to vinegar !). The present software library has nothing to do with wine (except it is developed in France), it is simply a Python library for manipulating various digital formats in an easy way, with a funny name. Most of the format supported are related in one way or another to cellular network's signalling. It is the glorious successor of libmich, which was started back in 2009, served well and retired in 2017.
It provides basically a runtime for encoding and decoding data structures, including CSN.1 and ASN.1. Additionally, it features a 3G and LTE mobile core network.
The whole library is licensed under LGPL v2.1 and is compatible with more recent version of the LGPL: all licensed files have an header making it self-explanatory. For more details, please report to the license.txt file.
Pycrate has a growing wiki. Use it as much as possible before opening an issue. Feel free also to propose some additional content.
The library is designed to work with Python 3 (3.5 and greater), from the official Python implementation CPython, and is systematically tested both on Linux, MacOS and Windows. It should also support alternative Python engines such as pypy, nuitka or Cython ; this is however not regularly tested. It should also support any other operating systems which has a decent Python 3 support.
Python2 support is entirely abandonned end of 2022, and hence release 0.5.5 is the last one with full support of Python 2.7.
Currently none. Only the Python builtins and few internal modules of Python (e.g. os, system, re, struct, datetime) are required for most of the features. The json internal module is required for supporting the JSON API. If you want to run pycrate in Python2 (which is bad !), you will however need to install the enum34 package.
The pycrate_ether/SCTP module can optionally use the external crc32c module from ICRAR.
The pycrate_mobile/TS24301_EMM and pycrate_mobile/TS24501_FGMM modules use CryptoMobile as optional dependency to encrypt and decrypt LTE and 5G NAS messages.
The pycrate_corenet part requires also pysctp and CryptoMobile to run.
The pycrate_diameter/parse_iana_diameter_xml.py file uses lxml to translate xml files from IANA to Python dictionnaries ; this is however not required for standard runtime.
The pycrate_osmo/SEDebugMux.py module relies on the crcmod to compute custom CRC in the frame format.
An installation script is available. As soon as you have cloned or downloaded the repository, you can use it to install the library within your Python package directory:
python setup.py install
Run it as superuser for a system-wide install, or as-is for a user home-directory level install. You can also run develop instead of install if you want a developer-friendly installation.
It is also possible to test the library before installing it (this will create two local directories ./test_asn/ and ./pycrate.egg-info/ that you can just delete afterwards):
python -m unittest test.test_pycrate
Or to build the library without installing it in the system:
python setup.py build
It is also possible to recompile all ASN.1 modules, this will take few minutes (but if I did not do any mistake, all ASN.1 modules provided in ./pycrate_asn1dir/ should have been compiled with the latest version of the compiler):
python -m pycrate_asn1c.asnproc
More generally, installation is not required, and simply having all pycrate_* subdirectories into the PYTHONPATH enables to use the entire library.
Alternatively, you can install the library with the pip
command:
pip install pycrate
The install package is available on pypi. It contains the library from the last tagged release on github (which may be months old).
This library is free software, and you are free to use it (or not to use it). In case you encounter a problem with it, first read this README completely and then check the Wiki ; moreover many classes, methods and functions are documented with docstrings, and finally you can have a look at the source code (it won't bite you).
If after all those steps, you still have a question or you think you found a bug, please open an issue (see below). Specific support requires time and may not be always possible. In case you require such support, please consider also contributing in one way or another (see below, too).
In case you are using this library in any of your project and you find it useful, do not hesitate to send me an email. It is always a pleasure to know where code provided on the Internet can end up... (I am personally aware of some dark places where a pycrate's fork lies).
When filling an issue, please provide precise and contextual information about your case and the error you potentially encounter:
This is the bare minimum if you want to get help. And when you consider your issue has been addressed, please close it: "A good issue is a closed one !" as would have said my great grandmother.
If you are willing to extend the library, do not hesitate to contact me by email or preferably through the github service (ideally, open a pull request). For important changes, please elaborate about your need and provide some justification. Any patch or submission is always very welcome!
In case you do not want to deep dive in the code, you can still contribute in many ways:
Getting contributions is extremely important to encourage the continuous development of the library, and to confirm the choice made to open-source it.
Pycrate is actually more a software suite than a single library. It is composed of several subdirectories, each providing specific services.
The core of the library.
Some of the most useful features are provided by the pack_val() functions from the utils module and the Charpy class from the charpy module. They help to deal easily with packing and unpacking bytes and integers (signed / unsigned, little / big endian) in an aligned and unaligned way. All lengths of fields are provided in bits, hence facilitating the handling of unaligned structures.
The modules provided here implement Ethernet and IP-oriented protocols and formats.
The modules here implement various multimedia formats.
Most of the classes here implement a complete recipe to parse all of those format in a single shot, by using their from_char() method.
All the modules here serve the sole purpose of compiling ASN.1 specifications. The most important ones are:
This compiler support most of the ASN.1 language features, including parameterization and class objects and sets (especially useful when working with table constraints). It has however few restrictions, the biggest being the need for the left part of the ASN.1 assignment ::= being on a single line. Also, old-school ASN.1 macros are not supported ; hence, the compiler cannot parse SNMP MIBs.
This subdirectory contains several ASN.1 specifications that are supported and precompiled for pycrate. Very few specifications have been changed in order to work with pycrate :
This subdirectory contains the ASN.1 runtime, that is loaded and used by the ASN.1 specifications compiled with the compiler in pycrate_asn1c. It supports the PER encoding rules (aligned and not, canonical also), and the BER, CER, DER and JER encoding rules.
This subdirectory contains a CSN.1 to Python translater in the file trans.py, and a CSN.1 runtime in the file csnobj.py, in order to encode and decode CSN.1 structures translated to Python objects.
This subdirectory contains CSN.1 structures extracted from 3GPP specifications (in the .csn files), and translated into Python objects. The following specifications have been used: TS 44.018, TS 44.060 and TS 24.008.
This subdirectory implements most of the 3GPP NAS protocol formats:
This subdirectory contains the following modules:
This subdirectory contains the following modules:
This subdirectory implements a signaling server that supports IuCS and IuPS over Iuh interfaces (including HNBAP and RUA/RANAP) for interfacing with 3G femtocells, and S1 interfaces (including S1AP) for interfacing with LTE eNodeBs. It handles many procedures to drive femtocells, eNodeBs and mobile terminals connecting through them. In terms of services, it mostly support short messages and data connectivity. It does not handle call services, neither active mobility procedures (handovers).
It can be easily (common, running a mobile core network is not that easy) configured and used thanks to the corenet project, also open-source.
This subdirectory contains the main module for the GMR-1 Radio Resources protocol: TS101376_04_08, and the submodule for specific Information Elements: TS101376_04_08_IE. This is a partial implementation of the ETSI specification TS 101 376-04-08 for the GEO-Mobile Radio Interface Specifications (Release 3).
This subdirectory contains all the CSN.1 definitions for the GMR-1 Radio Resources protocol, extracted from the 3 ETSI specifications for GEO-Mobile Radio release 3: TS 101 376-04-08, TS 101 376-04-12 and 101 376-04-13.
Most of the modules have doc strings. I try also to write readable sources and to comment them as much as possible for understanding them easily (and to allow also myself to understand my own code years after...). A wiki is provided and extended from time to time, to bring examples and methods on how to use the different modules (any contribution on this would be very welcome, too). Finally, the code provided in the test/ subdirectory is also representative on how to use the different modules.
Basically, a pycrate's object exposes the following methods:
set_val()
/ get_val()
/ get_val_d()
, which sets value into and gets value from the objectfrom_bytes()
/ to_bytes()
, which converts a buffer into values according to the internal structure of the object, and backfrom_json()
/ to_json()
, for working with JSON-encoded valueshex()
/ bin()
, for getting hexadecimal and binary representation of the serialized obect's valuerepr()
/ show()
, for providing nice python's internal representation, and printable representation of the object's valueWhen a Python module from pycrate_asn1dir/ is loaded, it creates Python classes corresponding to ASN.1 modules (all dash characters are converted to underscore). Each ASN.1 object has a corresponding Python instance, exposing the following methods:
from_asn1()
/ to_asn1()
, which converts ASN.1 textual value to Python value and backfrom_aper()
/ to_aper()
, which converts aligned PER encoded value to Python value and backfrom_uper()
/ to_uper()
, which converts unaligned PERfrom_ber()
/ to_ber()
, which converts BERfrom_cer()
/ to_cer()
, which converts CERfrom_der()
/ to_der()
, which converts DERfrom_jer()
/ to_jer()
, which converts JERfrom_oer()
/ to_oer()
, which converts OERfrom_coer()
/ to_coer()
, which converts canonical OERset_val()
/ get_val()
to set and get Python's values into the ASN.1 objectget_proto()
to return to internal structure of the ASN.1 objectAll the methods useful for working with ASN.1 objects at runtime can be found in the file pycrate_asn1rt/asnobj.py.
Four different tools are provided (yet):
It is possible to test the pycrate_showmedia.py tool with media test files provided in ./test/res/, or any other supported media file.
$ ./tools/pycrate_showmedia.py --help
usage: pycrate_showmedia.py [-h] [-bl BL] [-wt] input
print the internal structure of the input media file,supported formats are:
BMP, GIF, JPEG, MP3, MPEG4, PNG, TIFF
positional arguments:
input input media file
optional arguments:
-h, --help show this help message and exit
-bl BL maximum length for buffer representation
-wt show also absent / transparent fields
$ ./tools/pycrate_showmedia.py ./test/res/xkcd_wireless_signal.png
### PNG ###
<sig [PNG signature] : '\x89PNG\r\n\x1a\n'>
### PNGBody ###
### PNGChunk ###
<len : 13>
<type : 'IHDR'>
### IHDR ###
<width : 238>
<height : 415>
<depth [bit depth] : 8>
<color [color type] : 0 (Greyscale)>
<comp [compression method] : 0 (inflate/deflate with sliding window)>
<filter [filter method] : 0 (no interlace)>
<interlace [interlace method] : 0 (no interlace)>
<crc : 0x7d8cb12e>
### PNGChunk ###
<len : 9>
<type : 'pHYs'>
<data :
00 00 0c 4e 00 00 0c 4e 01 | '\x00\x00\x0cN\x00\x00\x0cN\x01'>
<crc : 0x7f778c23>
### PNGChunk ###
<len : 792>
<type : 'iCCP'>
<data :
50 68 6f 74 6f 73 68 6f 70 20 49 43 43 20 70 72 | 'Photoshop ICC pr'
6f 66 69 6c 65 00 00 78 da 63 60 60 9e e0 e8 e2 | 'ofile\x00\x00x\xdac``\x9e\xe0\xe8\xe2'
[...]
32 fd fc ea eb 82 ef e1 3f 05 7e 9d fa d3 fa cf | '2\xfd\xfc\xea\xeb\x82\xef\xe1?\x05~\x9d\xfa\xd3\xfa\xcf'
f1 ff 7f 00 0d 00 0f 34 | '\xf1\xff\x7f\x00\r\x00\x0f4'>
<crc : 0xfa96f15d>
### PNGChunk ###
<len : 32>
<type : 'cHRM'>
<data :
00 00 6e 27 00 00 73 af 00 00 df f2 00 00 83 30 | "\x00\x00n'\x00\x00s\xaf\x00\x00\xdf\xf2\x00\x00\x830"
00 00 77 43 00 00 c8 0a 00 00 34 95 00 00 2e dc | '\x00\x00wC\x00\x00\xc8\n\x00\x004\x95\x00\x00.\xdc'>
<crc : 0x20bf171a>
### PNGChunk ###
<len : 21130>
<type : 'IDAT'>
<data :
78 da ed bd 79 50 8d fd 1f ff ff bc ce 39 73 4e | 'x\xda\xed\xbdyP\x8d\xfd\x1f\xff\xff\xbc\xce9sN'
db b4 37 95 32 b4 19 94 06 2d 7e 11 26 b2 fc 10 | '\xdb\xb47\x952\xb4\x19\x94\x06-~\x11&\xb2\xfc\x10'
[...]
91 a3 d8 5b fc e1 cb 51 fd ab fb c9 cc ec ee 21 | '\x91\xa3\xd8[\xfc\xe1\xcbQ\xfd\xab\xfb\xc9\xcc\xec\xee!'
7d 70 6e f3 18 ce c1 c1 6d 8c 81 44 32 cf 51 ba | '}pn\xf3\x18\xce\xc1\xc1m\x8c\x81D2\xcfQ\xba'
...>
<crc : 0xa9fbdd38>
### PNGChunk ###
<len : 0>
<type : 'IEND'>
<data : >
<crc : 0xae426082>
It is possible to test the pycrate_asn1compile.py tool with some test ASN.1 specification from ./test/res/, or any other valid ASN.1 specification of your choice.
$ ./tools/pycrate_asn1compile.py --help
usage: pycrate_asn1compile.py [-h] [-s SPEC] [-i INPUT [INPUT ...]] [-o OUTPUT] [-g GENERATOR_PATH] [-j] [-fautotags] [-fextimpl] [-fverifwarn]
compile ASN.1 input file(s) for the pycrate ASN.1 runtime
optional arguments:
-h, --help show this help message and exit
-s SPEC provide a specification shortname, instead of ASN.1 input file(s)
-i INPUT [INPUT ...] ASN.1 input file(s) or directory
-o OUTPUT compiled output Python (and json) source file(s)
-g GENERATOR_PATH, --generator GENERATOR_PATH
provide an alternative python generator file path
-j output a json file with information on ASN.1 objects dependency
-fautotags force AUTOMATIC TAGS for all ASN.1 modules
-fextimpl force EXTENSIBILITY IMPLIED for all ASN.1 modules
-fverifwarn force warning instead of raising during the verification stage
$ ./tools/pycrate_asn1compile.py -i ./test/res/Hardcore.asn -o Hardcore
[proc] [./test/res/Hardcore.asn] module HardcoreSyntax (oid: []): 116 ASN.1 assignments found
--- compilation cycle ---
--- compilation cycle ---
--- compilation cycle ---
--- verifications ---
[proc] ASN.1 modules processed: ['HardcoreSyntax']
[proc] ASN.1 objects compiled: 75 types, 3 sets, 37 values
[proc] done
After compiling a module, it is possible to load it in Python and use it for encoding / decoding any objects defined in it.
Python 3.8.5 (default, Jul 28 2020, 12:59:40)
[GCC 9.3.0] on linux
Type "help", "copyright", "credits" or "license" for more information.
>>> from Hardcore import HardcoreSyntax
>>> HardcoreSyntax # this is the only ASN.1 module provided in Hardcore.asn
<class 'Hardcore.HardcoreSyntax'>
>>> Final = HardcoreSyntax.Final # this is the Final object defined at line 115
>>> Final
<Final (SEQUENCE)>
>>> Final.get_proto() # warning: this can return very laaaaaaarge definitions
('SEQUENCE', {
w1: ('SEQUENCE', {
r10: ('SEQUENCE', {
low: 'INTEGER',
high: 'INTEGER',
bool: 'BOOLEAN',
null (OPT): 'NULL'
}),
r90: ('SEQUENCE', {
low: 'INTEGER',
high: 'INTEGER',
bool: 'BOOLEAN',
null (OPT): 'NULL'
})
}),
w2: ('SEQUENCE', {
r10: ('SEQUENCE', {
low: 'INTEGER',
high: 'INTEGER',
bool: 'BOOLEAN',
null (OPT): 'NULL'
}),
r90: ('SEQUENCE', {
low: 'INTEGER',
high: 'INTEGER',
bool: 'BOOLEAN',
null (OPT): 'NULL'
})
}),
bool: 'BOOLEAN'
})
>>> V = {
... 'w1':{'r10':{'low':5, 'high':50, 'bool':False}, 'r90':{'low':50, 'high':95, 'bool':False, 'null':0}},
... 'w2':{'r10':{'low':1, 'high':10, 'bool':False}, 'r90':{'low':90, 'high':100, 'bool':True}},
... 'bool': True}
>>> Final.set_val(V)
>>> print(Final.to_asn1()) # .to_asn1() returns a printable ASN.1 representation of the value
{
w1 {
r10 {
low 5,
high 50,
bool FALSE
},
r90 {
low 50,
high 95,
bool FALSE,
null NULL
}
},
w2 {
r10 {
low 1,
high 10,
bool FALSE
},
r90 {
low 90,
high 100,
bool TRUE
}
},
bool TRUE
}
>>> Final.to_aper() # aligned PER
b'*\x85\x92\x80@\x01\x00\x08\x02\xd5`'
>>> Final.to_uper() # unaligned PER
b'*\x85\x92\x80@@\x02\x00\xb5X'
>>> Final.to_ber()
b'05\xa0\x18\xa0\t\x80\x01\x05\x81\x012\x82\x01\x00\xa1\x0b\x80\x012\x81\x01_\x82\x01\x00\x83\x00\xa1\x16\xa0\t\x80\x01\x01\x81\x01\n\x82\x01\x00\xa1\t\x80\x01Z\x81\x01d\x82\x01\xff\x82\x01\xff'
>>> Final.to_cer()
b'0\x80\xa0\x80\xa0\x80\x80\x01\x05\x81\x012\x82\x01\x00\x00\x00\xa1\x80\x80\x012\x81\x01_\x82\x01\x00\x83\x00\x00\x00\x00\x00\xa1\x80\xa0\x80\x80\x01\x01\x81\x01\n\x82\x01\x00\x00\x00\xa1\x80\x80\x01Z\x81\x01d\x82\x01\xff\x00\x00\x00\x00\x82\x01\xff\x00\x00'
>>> Final.to_der()
b'05\xa0\x18\xa0\t\x80\x01\x05\x81\x012\x82\x01\x00\xa1\x0b\x80\x012\x81\x01_\x82\x01\x00\x83\x00\xa1\x16\xa0\t\x80\x01\x01\x81\x01\n\x82\x01\x00\xa1\t\x80\x01Z\x81\x01d\x82\x01\xff\x82\x01\xff'
>>> Final.from_ber( Final.to_ber() )
>>> Final() == V # or Final._val == V
True
For more information about the API exposed for each ASN.1 object, you can check the docstrings of all ASN.1 objects, and also read the source file pycrate_asn1rt/asnobj.py. Do not forget to have a look at the Wiki, too!