| Index: third_party/gsutil/third_party/pyasn1/doc/pyasn1-tutorial.html
|
| diff --git a/third_party/gsutil/third_party/pyasn1/doc/pyasn1-tutorial.html b/third_party/gsutil/third_party/pyasn1/doc/pyasn1-tutorial.html
|
| new file mode 100644
|
| index 0000000000000000000000000000000000000000..2eb82f1e936d568406566584863c22f57fae7696
|
| --- /dev/null
|
| +++ b/third_party/gsutil/third_party/pyasn1/doc/pyasn1-tutorial.html
|
| @@ -0,0 +1,2405 @@
|
| +<html>
|
| +<title>
|
| +PyASN1 programmer's manual
|
| +</title>
|
| +<head>
|
| +</head>
|
| +<body>
|
| +<center>
|
| +<table width=60%>
|
| +<tr>
|
| +<td>
|
| +
|
| +<h3>
|
| +PyASN1 programmer's manual
|
| +</h3>
|
| +
|
| +<p align=right>
|
| +<i>written by <a href=mailto:ilya@glas.net>Ilya Etingof</a>, 2011-2012</i>
|
| +</p>
|
| +
|
| +<p>
|
| +Free and open-source pyasn1 library makes it easier for programmers and
|
| +network engineers to develop, debug and experiment with ASN.1-based protocols
|
| +using Python programming language as a tool.
|
| +</p>
|
| +
|
| +<p>
|
| +Abstract Syntax Notation One
|
| +(<a href=http://en.wikipedia.org/wiki/Abstract_Syntax_Notation_1x>ASN.1</a>)
|
| +is a set of
|
| +<a href=http://www.itu.int/ITU-T/studygroups/com17/languages/X.680-X.693-0207w.zip>
|
| +ITU standards</a> concered with provisioning instrumentation for developing
|
| +data exchange protocols in a robust, clear and interoperabable way for
|
| +various IT systems and applications. Most of the efforts are targeting the
|
| +following areas:
|
| +<ul>
|
| +<li>Data structures: the standard introduces a collection of basic data types
|
| +(similar to integers, bits, strings, arrays and records in a programming
|
| +language) that can be used for defining complex, possibly nested data
|
| +structures representing domain-specific data units.
|
| +<li>Serialization protocols: domain-specific data units expressed in ASN.1
|
| +types could be converted into a series of octets for storage or transmission
|
| +over the wire and then recovered back into their structured form on the
|
| +receiving end. This process is immune to various hardware and software
|
| +related dependencies.
|
| +<li>Data description language: could be used to describe particular set of
|
| +domain-specific data structures and their relationships. Such a description
|
| +could be passed to an ASN.1 compiler for automated generation of program
|
| +code that represents ASN.1 data structures in language-native environment
|
| +and handles data serialization issues.
|
| +</ul>
|
| +</p>
|
| +
|
| +<p>
|
| +This tutorial and algorithms, implemented by pyasn1 library, are
|
| +largely based on the information read in the book
|
| +<a href="http://www.oss.com/asn1/dubuisson.html">
|
| +ASN.1 - Communication between heterogeneous systems</a>
|
| +by Olivier Dubuisson. Another relevant resource is
|
| +<a href=ftp://ftp.rsasecurity.com/pub/pkcs/ascii/layman.asc>
|
| +A Layman's Guide to a Subset of ASN.1, BER, and DER</a> by Burton S. Kaliski.
|
| +It's advised to refer to these books for more in-depth knowledge on the
|
| +subject of ASN.1.
|
| +</p>
|
| +
|
| +<p>
|
| +As of this writing, pyasn1 library implements most of standard ASN.1 data
|
| +structures in a rather detailed and feature-rich manner. Another highly
|
| +important capability of the library is its data serialization facilities.
|
| +The last component of the standard - ASN.1 compiler is planned for
|
| +implementation in the future.
|
| +</p>
|
| +
|
| +</p>
|
| +The pyasn1 library was designed to follow the pre-1995 ASN.1 specification
|
| +(also known as X.208). Later, post 1995, revision (X.680) introduced
|
| +significant changes most of which have not yet been supported by pyasn1.
|
| +</p>
|
| +
|
| +<h3>
|
| +Table of contents
|
| +</h3>
|
| +
|
| +<p>
|
| +<ul>
|
| +<li><a href="#1">1. Data model for ASN.1 types</a>
|
| +<li><a href="#1.1">1.1 Scalar types</a>
|
| +<li><a href="#1.1.1">1.1.1 Boolean type</a>
|
| +<li><a href="#1.1.2">1.1.2 Null type</a>
|
| +<li><a href="#1.1.3">1.1.3 Integer type</a>
|
| +<li><a href="#1.1.4">1.1.4 Enumerated type</a>
|
| +<li><a href="#1.1.5">1.1.5 Real type</a>
|
| +<li><a href="#1.1.6">1.1.6 Bit string type</a>
|
| +<li><a href="#1.1.7">1.1.7 OctetString type</a>
|
| +<li><a href="#1.1.8">1.1.8 ObjectIdentifier type</a>
|
| +<li><a href="#1.1.9">1.1.9 Character string types</a>
|
| +<li><a href="#1.1.10">1.1.10 Useful types</a>
|
| +<li><a href="#1.2">1.2 Tagging</a>
|
| +<li><a href="#1.3">1.3 Constructed types</a>
|
| +<li><a href="#1.3.1">1.3.1 Sequence and Set types</a>
|
| +<li><a href="#1.3.2">1.3.2 SequenceOf and SetOf types</a>
|
| +<li><a href="#1.3.3">1.3.3 Choice type</a>
|
| +<li><a href="#1.3.4">1.3.4 Any type</a>
|
| +<li><a href="#1.4">1.4 Subtype constraints</a>
|
| +<li><a href="#1.4.1">1.4.1 Single value constraint</a>
|
| +<li><a href="#1.4.2">1.4.2 Value range constraint</a>
|
| +<li><a href="#1.4.3">1.4.3 Size constraint</a>
|
| +<li><a href="#1.4.4">1.4.4 Alphabet constraint</a>
|
| +<li><a href="#1.4.5">1.4.5 Constraint combinations</a>
|
| +<li><a href="#1.5">1.5 Types relationships</a>
|
| +<li><a href="#2">2. Codecs</a>
|
| +<li><a href="#2.1">2.1 Encoders</a>
|
| +<li><a href="#2.2">2.2 Decoders</a>
|
| +<li><a href="#2.2.1">2.2.1 Decoding untagged types</a>
|
| +<li><a href="#2.2.2">2.2.2 Ignoring unknown types</a>
|
| +<li><a href="#3">3. Feedback and getting help</a>
|
| +</ul>
|
| +
|
| +
|
| +<a name="1"></a>
|
| +<h3>
|
| +1. Data model for ASN.1 types
|
| +</h3>
|
| +
|
| +<p>
|
| +All ASN.1 types could be categorized into two groups: scalar (also called
|
| +simple or primitive) and constructed. The first group is populated by
|
| +well-known types like Integer or String. Members of constructed group
|
| +hold other types (simple or constructed) as their inner components, thus
|
| +they are semantically close to a programming language records or lists.
|
| +</p>
|
| +
|
| +<p>
|
| +In pyasn1, all ASN.1 types and values are implemented as Python objects.
|
| +The same pyasn1 object can represent either ASN.1 type and/or value
|
| +depending of the presense of value initializer on object instantiation.
|
| +We will further refer to these as <i>pyasn1 type object</i> versus <i>pyasn1
|
| +value object</i>.
|
| +</p>
|
| +
|
| +<p>
|
| +Primitive ASN.1 types are implemented as immutable scalar objects. There values
|
| +could be used just like corresponding native Python values (integers,
|
| +strings/bytes etc) and freely mixed with them in expressions.
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +>>> from pyasn1.type import univ
|
| +>>> asn1IntegerValue = univ.Integer(12)
|
| +>>> asn1IntegerValue - 2
|
| +10
|
| +>>> univ.OctetString('abc') == 'abc'
|
| +True # Python 2
|
| +>>> univ.OctetString(b'abc') == b'abc'
|
| +True # Python 3
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +It would be an error to perform an operation on a pyasn1 type object
|
| +as it holds no value to deal with:
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +>>> from pyasn1.type import univ
|
| +>>> asn1IntegerType = univ.Integer()
|
| +>>> asn1IntegerType - 2
|
| +...
|
| +pyasn1.error.PyAsn1Error: No value for __coerce__()
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<a name="1.1"></a>
|
| +<h4>
|
| +1.1 Scalar types
|
| +</h4>
|
| +
|
| +<p>
|
| +In the sub-sections that follow we will explain pyasn1 mapping to those
|
| +primitive ASN.1 types. Both, ASN.1 notation and corresponding pyasn1
|
| +syntax will be given in each case.
|
| +</p>
|
| +
|
| +<a name="1.1.1"></a>
|
| +<h4>
|
| +1.1.1 Boolean type
|
| +</h4>
|
| +
|
| +<p>
|
| +This is the simplest type those values could be either True or False.
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +;; type specification
|
| +FunFactorPresent ::= BOOLEAN
|
| +
|
| +;; values declaration and assignment
|
| +pythonFunFactor FunFactorPresent ::= TRUE
|
| +cobolFunFactor FunFactorPresent :: FALSE
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +And here's pyasn1 version of it:
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +>>> from pyasn1.type import univ
|
| +>>> class FunFactorPresent(univ.Boolean): pass
|
| +...
|
| +>>> pythonFunFactor = FunFactorPresent(True)
|
| +>>> cobolFunFactor = FunFactorPresent(False)
|
| +>>> pythonFunFactor
|
| +FunFactorPresent('True(1)')
|
| +>>> cobolFunFactor
|
| +FunFactorPresent('False(0)')
|
| +>>> pythonFunFactor == cobolFunFactor
|
| +False
|
| +>>>
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<a name="1.1.2"></a>
|
| +<h4>
|
| +1.1.2 Null type
|
| +</h4>
|
| +
|
| +<p>
|
| +The NULL type is sometimes used to express the absense of any information.
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +;; type specification
|
| +Vote ::= CHOICE {
|
| + agreed BOOLEAN,
|
| + skip NULL
|
| +}
|
| +</td></tr></table>
|
| +
|
| +;; value declaration and assignment
|
| +myVote Vote ::= skip:NULL
|
| +</pre>
|
| +
|
| +<p>
|
| +We will explain the CHOICE type later in this paper, meanwhile the NULL
|
| +type:
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +>>> from pyasn1.type import univ
|
| +>>> skip = univ.Null()
|
| +>>> skip
|
| +Null('')
|
| +>>>
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<a name="1.1.3"></a>
|
| +<h4>
|
| +1.1.3 Integer type
|
| +</h4>
|
| +
|
| +<p>
|
| +ASN.1 defines the values of Integer type as negative or positive of whatever
|
| +length. This definition plays nicely with Python as the latter places no
|
| +limit on Integers. However, some ASN.1 implementations may impose certain
|
| +limits of integer value ranges. Keep that in mind when designing new
|
| +data structures.
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +;; values specification
|
| +age-of-universe INTEGER ::= 13750000000
|
| +mean-martian-surface-temperature INTEGER ::= -63
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +A rather strigntforward mapping into pyasn1:
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +>>> from pyasn1.type import univ
|
| +>>> ageOfUniverse = univ.Integer(13750000000)
|
| +>>> ageOfUniverse
|
| +Integer(13750000000)
|
| +>>>
|
| +>>> meanMartianSurfaceTemperature = univ.Integer(-63)
|
| +>>> meanMartianSurfaceTemperature
|
| +Integer(-63)
|
| +>>>
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +ASN.1 allows to assign human-friendly names to particular values of
|
| +an INTEGER type.
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +Temperature ::= INTEGER {
|
| + freezing(0),
|
| + boiling(100)
|
| +}
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +The Temperature type expressed in pyasn1:
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +>>> from pyasn1.type import univ, namedval
|
| +>>> class Temperature(univ.Integer):
|
| +... namedValues = namedval.NamedValues(('freezing', 0), ('boiling', 100))
|
| +...
|
| +>>> t = Temperature(0)
|
| +>>> t
|
| +Temperature('freezing(0)')
|
| +>>> t + 1
|
| +Temperature(1)
|
| +>>> t + 100
|
| +Temperature('boiling(100)')
|
| +>>> t = Temperature('boiling')
|
| +>>> t
|
| +Temperature('boiling(100)')
|
| +>>> Temperature('boiling') / 2
|
| +Temperature(50)
|
| +>>> -1 < Temperature('freezing')
|
| +True
|
| +>>> 47 > Temperature('boiling')
|
| +False
|
| +>>>
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +These values labels have no effect on Integer type operations, any value
|
| +still could be assigned to a type (information on value constraints will
|
| +follow further in this paper).
|
| +</p>
|
| +
|
| +<a name="1.1.4"></a>
|
| +<h4>
|
| +1.1.4 Enumerated type
|
| +</h4>
|
| +
|
| +<p>
|
| +ASN.1 Enumerated type differs from an Integer type in a number of ways.
|
| +Most important is that its instance can only hold a value that belongs
|
| +to a set of values specified on type declaration.
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +error-status ::= ENUMERATED {
|
| + no-error(0),
|
| + authentication-error(10),
|
| + authorization-error(20),
|
| + general-failure(51)
|
| +}
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +When constructing Enumerated type we will use two pyasn1 features: values
|
| +labels (as mentioned above) and value constraint (will be described in
|
| +more details later on).
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +>>> from pyasn1.type import univ, namedval, constraint
|
| +>>> class ErrorStatus(univ.Enumerated):
|
| +... namedValues = namedval.NamedValues(
|
| +... ('no-error', 0),
|
| +... ('authentication-error', 10),
|
| +... ('authorization-error', 20),
|
| +... ('general-failure', 51)
|
| +... )
|
| +... subtypeSpec = univ.Enumerated.subtypeSpec + \
|
| +... constraint.SingleValueConstraint(0, 10, 20, 51)
|
| +...
|
| +>>> errorStatus = univ.ErrorStatus('no-error')
|
| +>>> errorStatus
|
| +ErrorStatus('no-error(0)')
|
| +>>> errorStatus == univ.ErrorStatus('general-failure')
|
| +False
|
| +>>> univ.ErrorStatus('non-existing-state')
|
| +Traceback (most recent call last):
|
| +...
|
| +pyasn1.error.PyAsn1Error: Can't coerce non-existing-state into integer
|
| +>>>
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +Particular integer values associated with Enumerated value states
|
| +have no meaning. They should not be used as such or in any kind of
|
| +math operation. Those integer values are only used by codecs to
|
| +transfer state from one entity to another.
|
| +</p>
|
| +
|
| +<a name="1.1.5"></a>
|
| +<h4>
|
| +1.1.5 Real type
|
| +</h4>
|
| +
|
| +<p>
|
| +Values of the Real type are a three-component tuple of mantissa, base and
|
| +exponent. All three are integers.
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +pi ::= REAL { mantissa 314159, base 10, exponent -5 }
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +Corresponding pyasn1 objects can be initialized with either a three-component
|
| +tuple or a Python float. Infinite values could be expressed in a way,
|
| +compatible with Python float type.
|
| +
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +>>> from pyasn1.type import univ
|
| +>>> pi = univ.Real((314159, 10, -5))
|
| +>>> pi
|
| +Real((314159, 10,-5))
|
| +>>> float(pi)
|
| +3.14159
|
| +>>> pi == univ.Real(3.14159)
|
| +True
|
| +>>> univ.Real('inf')
|
| +Real('inf')
|
| +>>> univ.Real('-inf') == float('-inf')
|
| +True
|
| +>>>
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +If a Real object is initialized from a Python float or yielded by a math
|
| +operation, the base is set to decimal 10 (what affects encoding).
|
| +</p>
|
| +
|
| +<a name="1.1.6"></a>
|
| +<h4>
|
| +1.1.6 Bit string type
|
| +</h4>
|
| +
|
| +<p>
|
| +ASN.1 BIT STRING type holds opaque binary data of an arbitrarily length.
|
| +A BIT STRING value could be initialized by either a binary (base 2) or
|
| +hex (base 16) value.
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +public-key BIT STRING ::= '1010111011110001010110101101101
|
| + 1011000101010000010110101100010
|
| + 0110101010000111101010111111110'B
|
| +
|
| +signature BIT STRING ::= 'AF01330CD932093392100B39FF00DE0'H
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +The pyasn1 BitString objects can initialize from native ASN.1 notation
|
| +(base 2 or base 16 strings) or from a Python tuple of binary components.
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +>>> from pyasn1.type import univ
|
| +>>> publicKey = univ.BitString(
|
| +... "'1010111011110001010110101101101"
|
| +... "1011000101010000010110101100010"
|
| +... "0110101010000111101010111111110'B"
|
| +)
|
| +>>> publicKey
|
| +BitString("'10101110111100010101101011011011011000101010000010110101100010\
|
| +0110101010000111101010111111110'B")
|
| +>>> signature = univ.BitString(
|
| +... "'AF01330CD932093392100B39FF00DE0'H"
|
| +... )
|
| +>>> signature
|
| +BitString("'101011110000000100110011000011001101100100110010000010010011001\
|
| +1100100100001000000001011001110011111111100000000110111100000'B")
|
| +>>> fingerprint = univ.BitString(
|
| +... (1, 0, 1, 1 ,0, 1, 1, 1, 0, 1, 0, 1)
|
| +... )
|
| +>>> fingerprint
|
| +BitString("'101101110101'B")
|
| +>>>
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +Another BIT STRING initialization method supported by ASN.1 notation
|
| +is to specify only 1-th bits along with their human-friendly label
|
| +and bit offset relative to the beginning of the bit string. With this
|
| +method, all not explicitly mentioned bits are doomed to be zeros.
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +bit-mask BIT STRING ::= {
|
| + read-flag(0),
|
| + write-flag(2),
|
| + run-flag(4)
|
| +}
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +To express this in pyasn1, we will employ the named values feature (as with
|
| +Enumeration type).
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +>>> from pyasn1.type import univ, namedval
|
| +>>> class BitMask(univ.BitString):
|
| +... namedValues = namedval.NamedValues(
|
| +... ('read-flag', 0),
|
| +... ('write-flag', 2),
|
| +... ('run-flag', 4)
|
| +... )
|
| +>>> bitMask = BitMask('read-flag,run-flag')
|
| +>>> bitMask
|
| +BitMask("'10001'B")
|
| +>>> tuple(bitMask)
|
| +(1, 0, 0, 0, 1)
|
| +>>> bitMask[4]
|
| +1
|
| +>>>
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +The BitString objects mimic the properties of Python tuple type in part
|
| +of immutable sequence object protocol support.
|
| +</p>
|
| +
|
| +<a name="1.1.7"></a>
|
| +<h4>
|
| +1.1.7 OctetString type
|
| +</h4>
|
| +
|
| +<p>
|
| +The OCTET STRING type is a confusing subject. According to ASN.1
|
| +specification, this type is similar to BIT STRING, the major difference
|
| +is that the former operates in 8-bit chunks of data. What is important
|
| +to note, is that OCTET STRING was NOT designed to handle text strings - the
|
| +standard provides many other types specialized for text content. For that
|
| +reason, ASN.1 forbids to initialize OCTET STRING values with "quoted text
|
| +strings", only binary or hex initializers, similar to BIT STRING ones,
|
| +are allowed.
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +thumbnail OCTET STRING ::= '1000010111101110101111000000111011'B
|
| +thumbnail OCTET STRING ::= 'FA9823C43E43510DE3422'H
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +However, ASN.1 users (e.g. protocols designers) seem to ignore the original
|
| +purpose of the OCTET STRING type - they used it for handling all kinds of
|
| +data, including text strings.
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +welcome-message OCTET STRING ::= "Welcome to ASN.1 wilderness!"
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +In pyasn1, we have taken a liberal approach and allowed both BIT STRING
|
| +style and quoted text initializers for the OctetString objects. To avoid
|
| +possible collisions, quoted text is the default initialization syntax.
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +>>> from pyasn1.type import univ
|
| +>>> thumbnail = univ.OctetString(
|
| +... binValue='1000010111101110101111000000111011'
|
| +... )
|
| +>>> thumbnail
|
| +OctetString(hexValue='85eebcec0')
|
| +>>> thumbnail = univ.OctetString(
|
| +... hexValue='FA9823C43E43510DE3422'
|
| +... )
|
| +>>> thumbnail
|
| +OctetString(hexValue='fa9823c43e4351de34220')
|
| +>>>
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +Most frequent usage of the OctetString class is to instantiate it with
|
| +a text string.
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +>>> from pyasn1.type import univ
|
| +>>> welcomeMessage = univ.OctetString('Welcome to ASN.1 wilderness!')
|
| +>>> welcomeMessage
|
| +OctetString(b'Welcome to ASN.1 wilderness!')
|
| +>>> print('%s' % welcomeMessage)
|
| +Welcome to ASN.1 wilderness!
|
| +>>> welcomeMessage[11:16]
|
| +OctetString(b'ASN.1')
|
| +>>>
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +OctetString objects support the immutable sequence object protocol.
|
| +In other words, they behave like Python 3 bytes (or Python 2 strings).
|
| +</p>
|
| +
|
| +<p>
|
| +When running pyasn1 on Python 3, it's better to use the bytes objects for
|
| +OctetString instantiation, as it's more reliable and efficient.
|
| +</p>
|
| +
|
| +<p>
|
| +Additionally, OctetString's can also be instantiated with a sequence of
|
| +8-bit integers (ASCII codes).
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +>>> univ.OctetString((77, 101, 101, 103, 111))
|
| +OctetString(b'Meego')
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +It is sometimes convenient to express OctetString instances as 8-bit
|
| +characters (Python 3 bytes or Python 2 strings) or 8-bit integers.
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +>>> octetString = univ.OctetString('ABCDEF')
|
| +>>> octetString.asNumbers()
|
| +(65, 66, 67, 68, 69, 70)
|
| +>>> octetString.asOctets()
|
| +b'ABCDEF'
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<a name="1.1.8"></a>
|
| +<h4>
|
| +1.1.8 ObjectIdentifier type
|
| +</h4>
|
| +
|
| +<p>
|
| +Values of the OBJECT IDENTIFIER type are sequences of integers that could
|
| +be used to identify virtually anything in the world. Various ASN.1-based
|
| +protocols employ OBJECT IDENTIFIERs for their own identification needs.
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +internet-id OBJECT IDENTIFIER ::= {
|
| + iso(1) identified-organization(3) dod(6) internet(1)
|
| +}
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +One of the natural ways to map OBJECT IDENTIFIER type into a Python
|
| +one is to use Python tuples of integers. So this approach is taken by
|
| +pyasn1.
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +>>> from pyasn1.type import univ
|
| +>>> internetId = univ.ObjectIdentifier((1, 3, 6, 1))
|
| +>>> internetId
|
| +ObjectIdentifier('1.3.6.1')
|
| +>>> internetId[2]
|
| +6
|
| +>>> internetId[1:3]
|
| +ObjectIdentifier('3.6')
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +A more human-friendly "dotted" notation is also supported.
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +>>> from pyasn1.type import univ
|
| +>>> univ.ObjectIdentifier('1.3.6.1')
|
| +ObjectIdentifier('1.3.6.1')
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +Symbolic names of the arcs of object identifier, sometimes present in
|
| +ASN.1 specifications, are not preserved and used in pyasn1 objects.
|
| +</p>
|
| +
|
| +<p>
|
| +The ObjectIdentifier objects mimic the properties of Python tuple type in
|
| +part of immutable sequence object protocol support.
|
| +</p>
|
| +
|
| +<a name="1.1.9"></a>
|
| +<h4>
|
| +1.1.9 Character string types
|
| +</h4>
|
| +
|
| +<p>
|
| +ASN.1 standard introduces a diverse set of text-specific types. All of them
|
| +were designed to handle various types of characters. Some of these types seem
|
| +be obsolete nowdays, as their target technologies are gone. Another issue
|
| +to be aware of is that raw OCTET STRING type is sometimes used in practice
|
| +by ASN.1 users instead of specialized character string types, despite
|
| +explicit prohibition imposed by ASN.1 specification.
|
| +</p>
|
| +
|
| +<p>
|
| +The two types are specific to ASN.1 are NumericString and PrintableString.
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +welcome-message ::= PrintableString {
|
| + "Welcome to ASN.1 text types"
|
| +}
|
| +
|
| +dial-pad-numbers ::= NumericString {
|
| + "0", "1", "2", "3", "4", "5", "6", "7", "8", "9"
|
| +}
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +Their pyasn1 implementations are:
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +>>> from pyasn1.type import char
|
| +>>> '%s' % char.PrintableString("Welcome to ASN.1 text types")
|
| +'Welcome to ASN.1 text types'
|
| +>>> dialPadNumbers = char.NumericString(
|
| + "0" "1" "2" "3" "4" "5" "6" "7" "8" "9"
|
| +)
|
| +>>> dialPadNumbers
|
| +NumericString(b'0123456789')
|
| +>>>
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +The following types came to ASN.1 from ISO standards on character sets.
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +>>> from pyasn1.type import char
|
| +>>> char.VisibleString("abc")
|
| +VisibleString(b'abc')
|
| +>>> char.IA5String('abc')
|
| +IA5String(b'abc')
|
| +>>> char.TeletexString('abc')
|
| +TeletexString(b'abc')
|
| +>>> char.VideotexString('abc')
|
| +VideotexString(b'abc')
|
| +>>> char.GraphicString('abc')
|
| +GraphicString(b'abc')
|
| +>>> char.GeneralString('abc')
|
| +GeneralString(b'abc')
|
| +>>>
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +The last three types are relatively recent addition to the family of
|
| +character string types: UniversalString, BMPString, UTF8String.
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +>>> from pyasn1.type import char
|
| +>>> char.UniversalString("abc")
|
| +UniversalString(b'abc')
|
| +>>> char.BMPString('abc')
|
| +BMPString(b'abc')
|
| +>>> char.UTF8String('abc')
|
| +UTF8String(b'abc')
|
| +>>> utf8String = char.UTF8String('У попа была собака')
|
| +>>> utf8String
|
| +UTF8String(b'\xd0\xa3 \xd0\xbf\xd0\xbe\xd0\xbf\xd0\xb0 \xd0\xb1\xd1\x8b\xd0\xbb\xd0\xb0 \
|
| +\xd1\x81\xd0\xbe\xd0\xb1\xd0\xb0\xd0\xba\xd0\xb0')
|
| +>>> print(utf8String)
|
| +У попа была собака
|
| +>>>
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +In pyasn1, all character type objects behave like Python strings. None of
|
| +them is currently constrained in terms of valid alphabet so it's up to
|
| +the data source to keep an eye on data validation for these types.
|
| +</p>
|
| +
|
| +<a name="1.1.10"></a>
|
| +<h4>
|
| +1.1.10 Useful types
|
| +</h4>
|
| +
|
| +<p>
|
| +There are three so-called useful types defined in the standard:
|
| +ObjectDescriptor, GeneralizedTime, UTCTime. They all are subtypes
|
| +of GraphicString or VisibleString types therefore useful types are
|
| +character string types.
|
| +</p>
|
| +
|
| +<p>
|
| +It's advised by the ASN.1 standard to have an instance of ObjectDescriptor
|
| +type holding a human-readable description of corresponding instance of
|
| +OBJECT IDENTIFIER type. There are no formal linkage between these instances
|
| +and provision for ObjectDescriptor uniqueness in the standard.
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +>>> from pyasn1.type import useful
|
| +>>> descrBER = useful.ObjectDescriptor(
|
| + "Basic encoding of a single ASN.1 type"
|
| +)
|
| +>>>
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +GeneralizedTime and UTCTime types are designed to hold a human-readable
|
| +timestamp in a universal and unambiguous form. The former provides
|
| +more flexibility in notation while the latter is more strict but has
|
| +Y2K issues.
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +;; Mar 8 2010 12:00:00 MSK
|
| +moscow-time GeneralizedTime ::= "20110308120000.0"
|
| +;; Mar 8 2010 12:00:00 UTC
|
| +utc-time GeneralizedTime ::= "201103081200Z"
|
| +;; Mar 8 1999 12:00:00 UTC
|
| +utc-time UTCTime ::= "9803081200Z"
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +>>> from pyasn1.type import useful
|
| +>>> moscowTime = useful.GeneralizedTime("20110308120000.0")
|
| +>>> utcTime = useful.UTCTime("9803081200Z")
|
| +>>>
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +Despite their intended use, these types possess no special, time-related,
|
| +handling in pyasn1. They are just printable strings.
|
| +</p>
|
| +
|
| +<a name="1.2"></a>
|
| +<h4>
|
| +1.2 Tagging
|
| +</h4>
|
| +
|
| +<p>
|
| +In order to continue with the Constructed ASN.1 types, we will first have
|
| +to introduce the concept of tagging (and its pyasn1 implementation), as
|
| +some of the Constructed types rely upon the tagging feature.
|
| +</p>
|
| +
|
| +<p>
|
| +When a value is coming into an ASN.1-based system (received from a network
|
| +or read from some storage), the receiving entity has to determine the
|
| +type of the value to interpret and verify it accordingly.
|
| +</p>
|
| +
|
| +<p>
|
| +Historically, the first data serialization protocol introduced in
|
| +ASN.1 was BER (Basic Encoding Rules). According to BER, any serialized
|
| +value is packed into a triplet of (Type, Length, Value) where Type is a
|
| +code that identifies the value (which is called <i>tag</i> in ASN.1),
|
| +length is the number of bytes occupied by the value in its serialized form
|
| +and value is ASN.1 value in a form suitable for serial transmission or storage.
|
| +</p>
|
| +
|
| +<p>
|
| +For that reason almost every ASN.1 type has a tag (which is actually a
|
| +BER type) associated with it by default.
|
| +</p>
|
| +
|
| +<p>
|
| +An ASN.1 tag could be viewed as a tuple of three numbers:
|
| +(Class, Format, Number). While Number identifies a tag, Class component
|
| +is used to create scopes for Numbers. Four scopes are currently defined:
|
| +UNIVERSAL, context-specific, APPLICATION and PRIVATE. The Format component
|
| +is actually a one-bit flag - zero for tags associated with scalar types,
|
| +and one for constructed types (will be discussed later on).
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +MyIntegerType ::= [12] INTEGER
|
| +MyOctetString ::= [APPLICATION 0] OCTET STRING
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +In pyasn1, tags are implemented as immutable, tuple-like objects:
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +>>> from pyasn1.type import tag
|
| +>>> myTag = tag.Tag(tag.tagClassContext, tag.tagFormatSimple, 10)
|
| +>>> myTag
|
| +Tag(tagClass=128, tagFormat=0, tagId=10)
|
| +>>> tuple(myTag)
|
| +(128, 0, 10)
|
| +>>> myTag[2]
|
| +10
|
| +>>> myTag == tag.Tag(tag.tagClassApplication, tag.tagFormatSimple, 10)
|
| +False
|
| +>>>
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +Default tag, associated with any ASN.1 type, could be extended or replaced
|
| +to make new type distinguishable from its ancestor. The standard provides
|
| +two modes of tag mangling - IMPLICIT and EXPLICIT.
|
| +</p>
|
| +
|
| +<p>
|
| +EXPLICIT mode works by appending new tag to the existing ones thus creating
|
| +an ordered set of tags. This set will be considered as a whole for type
|
| +identification and encoding purposes. Important property of EXPLICIT tagging
|
| +mode is that it preserves base type information in encoding what makes it
|
| +possible to completely recover type information from encoding.
|
| +</p>
|
| +
|
| +<p>
|
| +When tagging in IMPLICIT mode, the outermost existing tag is dropped and
|
| +replaced with a new one.
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +MyIntegerType ::= [12] IMPLICIT INTEGER
|
| +MyOctetString ::= [APPLICATION 0] EXPLICIT OCTET STRING
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +To model both modes of tagging, a specialized container TagSet object (holding
|
| +zero, one or more Tag objects) is used in pyasn1.
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +>>> from pyasn1.type import tag
|
| +>>> tagSet = tag.TagSet(
|
| +... tag.Tag(tag.tagClassContext, tag.tagFormatSimple, 10), # base tag
|
| +... tag.Tag(tag.tagClassContext, tag.tagFormatSimple, 10) # effective tag
|
| +... )
|
| +>>> tagSet
|
| +TagSet(Tag(tagClass=128, tagFormat=0, tagId=10))
|
| +>>> tagSet.getBaseTag()
|
| +Tag(tagClass=128, tagFormat=0, tagId=10)
|
| +>>> tagSet = tagSet.tagExplicitly(
|
| +... tag.Tag(tag.tagClassContext, tag.tagFormatSimple, 20)
|
| +... )
|
| +>>> tagSet
|
| +TagSet(Tag(tagClass=128, tagFormat=0, tagId=10),
|
| + Tag(tagClass=128, tagFormat=32, tagId=20))
|
| +>>> tagSet = tagSet.tagExplicitly(
|
| +... tag.Tag(tag.tagClassContext, tag.tagFormatSimple, 30)
|
| +... )
|
| +>>> tagSet
|
| +TagSet(Tag(tagClass=128, tagFormat=0, tagId=10),
|
| + Tag(tagClass=128, tagFormat=32, tagId=20),
|
| + Tag(tagClass=128, tagFormat=32, tagId=30))
|
| +>>> tagSet = tagSet.tagImplicitly(
|
| +... tag.Tag(tag.tagClassContext, tag.tagFormatSimple, 40)
|
| +... )
|
| +>>> tagSet
|
| +TagSet(Tag(tagClass=128, tagFormat=0, tagId=10),
|
| + Tag(tagClass=128, tagFormat=32, tagId=20),
|
| + Tag(tagClass=128, tagFormat=32, tagId=40))
|
| +>>>
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +As a side note: the "base tag" concept (accessible through the getBaseTag()
|
| +method) is specific to pyasn1 -- the base tag is used to identify the original
|
| +ASN.1 type of an object in question. Base tag is never occurs in encoding
|
| +and is mostly used internally by pyasn1 for choosing type-specific data
|
| +processing algorithms. The "effective tag" is the one that always appears in
|
| +encoding and is used on tagSets comparation.
|
| +</p>
|
| +
|
| +<p>
|
| +Any two TagSet objects could be compared to see if one is a derivative
|
| +of the other. Figuring this out is also useful in cases when a type-specific
|
| +data processing algorithms are to be chosen.
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +>>> from pyasn1.type import tag
|
| +>>> tagSet1 = tag.TagSet(
|
| +... tag.Tag(tag.tagClassContext, tag.tagFormatSimple, 10) # base tag
|
| +... tag.Tag(tag.tagClassContext, tag.tagFormatSimple, 10) # effective tag
|
| +... )
|
| +>>> tagSet2 = tagSet1.tagExplicitly(
|
| +... tag.Tag(tag.tagClassContext, tag.tagFormatSimple, 20)
|
| +... )
|
| +>>> tagSet1.isSuperTagSetOf(tagSet2)
|
| +True
|
| +>>> tagSet2.isSuperTagSetOf(tagSet1)
|
| +False
|
| +>>>
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +We will complete this discussion on tagging with a real-world example. The
|
| +following ASN.1 tagged type:
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +MyIntegerType ::= [12] EXPLICIT INTEGER
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +could be expressed in pyasn1 like this:
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +>>> from pyasn1.type import univ, tag
|
| +>>> class MyIntegerType(univ.Integer):
|
| +... tagSet = univ.Integer.tagSet.tagExplicitly(
|
| +... tag.Tag(tag.tagClassContext, tag.tagFormatSimple, 12)
|
| +... )
|
| +>>> myInteger = MyIntegerType(12345)
|
| +>>> myInteger.getTagSet()
|
| +TagSet(Tag(tagClass=0, tagFormat=0, tagId=2),
|
| + Tag(tagClass=128, tagFormat=32, tagId=12))
|
| +>>>
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +Referring to the above code, the tagSet class attribute is a property of any
|
| +pyasn1 type object that assigns default tagSet to a pyasn1 value object. This
|
| +default tagSet specification can be ignored and effectively replaced by some
|
| +other tagSet value passed on object instantiation.
|
| +</p>
|
| +
|
| +<p>
|
| +It's important to understand that the tag set property of pyasn1 type/value
|
| +object can never be modifed in place. In other words, a pyasn1 type/value
|
| +object can never change its tags. The only way is to create a new pyasn1
|
| +type/value object and associate different tag set with it.
|
| +</p>
|
| +
|
| +
|
| +<a name="1.3"></a>
|
| +<h4>
|
| +1.3 Constructed types
|
| +</h4>
|
| +
|
| +<p>
|
| +Besides scalar types, ASN.1 specifies so-called constructed ones - these
|
| +are capable of holding one or more values of other types, both scalar
|
| +and constructed.
|
| +</p>
|
| +
|
| +<p>
|
| +In pyasn1 implementation, constructed ASN.1 types behave like
|
| +Python sequences, and also support additional component addressing methods,
|
| +specific to particular constructed type.
|
| +</p>
|
| +
|
| +<a name="1.3.1"></a>
|
| +<h4>
|
| +1.3.1 Sequence and Set types
|
| +</h4>
|
| +
|
| +<p>
|
| +The Sequence and Set types have many similar properties:
|
| +</p>
|
| +<ul>
|
| +<li>they can hold any number of inner components of different types
|
| +<li>every component has a human-friendly identifier
|
| +<li>any component can have a default value
|
| +<li>some components can be absent.
|
| +</ul>
|
| +
|
| +<p>
|
| +However, Sequence type guarantees the ordering of Sequence value components
|
| +to match their declaration order. By contrast, components of the
|
| +Set type can be ordered to best suite application's needs.
|
| +<p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +Record ::= SEQUENCE {
|
| + id INTEGER,
|
| + room [0] INTEGER OPTIONAL,
|
| + house [1] INTEGER DEFAULT 0
|
| +}
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +Up to this moment, the only method we used for creating new pyasn1 types
|
| +is Python sub-classing. With this method, a new, named Python class is created
|
| +what mimics type derivation in ASN.1 grammar. However, ASN.1 also allows for
|
| +defining anonymous subtypes (room and house components in the example above).
|
| +To support anonymous subtyping in pyasn1, a cloning operation on an existing
|
| +pyasn1 type object can be invoked what creates a new instance of original
|
| +object with possibly modified properties.
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +>>> from pyasn1.type import univ, namedtype, tag
|
| +>>> class Record(univ.Sequence):
|
| +... componentType = namedtype.NamedTypes(
|
| +... namedtype.NamedType('id', univ.Integer()),
|
| +... namedtype.OptionalNamedType(
|
| +... 'room',
|
| +... univ.Integer().subtype(implicitTag=tag.Tag(tag.tagClassContext, tag.tagFormatSimple, 0))
|
| +... ),
|
| +... namedtype.DefaultedNamedType(
|
| +... 'house',
|
| +... univ.Integer(0).subtype(implicitTag=tag.Tag(tag.tagClassContext, tag.tagFormatSimple, 1))
|
| +... )
|
| +... )
|
| +>>>
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +All pyasn1 constructed type classes have a class attribute <b>componentType</b>
|
| +that represent default type specification. Its value is a NamedTypes object.
|
| +</p>
|
| +
|
| +<p>
|
| +The NamedTypes class instance holds a sequence of NameType, OptionalNamedType
|
| +or DefaultedNamedType objects which, in turn, refer to pyasn1 type objects that
|
| +represent inner SEQUENCE components specification.
|
| +</p>
|
| +
|
| +<p>
|
| +Finally, invocation of a subtype() method of pyasn1 type objects in the code
|
| +above returns an implicitly tagged copy of original object.
|
| +</p>
|
| +
|
| +<p>
|
| +Once a SEQUENCE or SET type is decleared with pyasn1, it can be instantiated
|
| +and initialized (continuing the above code):
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +>>> record = Record()
|
| +>>> record.setComponentByName('id', 123)
|
| +>>> print(record.prettyPrint())
|
| +Record:
|
| + id=123
|
| +>>>
|
| +>>> record.setComponentByPosition(1, 321)
|
| +>>> print(record.prettyPrint())
|
| +Record:
|
| + id=123
|
| + room=321
|
| +>>>
|
| +>>> record.setDefaultComponents()
|
| +>>> print(record.prettyPrint())
|
| +Record:
|
| + id=123
|
| + room=321
|
| + house=0
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +Inner components of pyasn1 Sequence/Set objects could be accessed using the
|
| +following methods:
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +>>> record.getComponentByName('id')
|
| +Integer(123)
|
| +>>> record.getComponentByPosition(1)
|
| +Integer(321)
|
| +>>> record[2]
|
| +Integer(0)
|
| +>>> for idx in range(len(record)):
|
| +... print(record.getNameByPosition(idx), record.getComponentByPosition(idx))
|
| +id 123
|
| +room 321
|
| +house 0
|
| +>>>
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +The Set type share all the properties of Sequence type, and additionally
|
| +support by-tag component addressing (as all Set components have distinct
|
| +types).
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +>>> from pyasn1.type import univ, namedtype, tag
|
| +>>> class Gamer(univ.Set):
|
| +... componentType = namedtype.NamedTypes(
|
| +... namedtype.NamedType('score', univ.Integer()),
|
| +... namedtype.NamedType('player', univ.OctetString()),
|
| +... namedtype.NamedType('id', univ.ObjectIdentifier())
|
| +... )
|
| +>>> gamer = Gamer()
|
| +>>> gamer.setComponentByType(univ.Integer().getTagSet(), 121343)
|
| +>>> gamer.setComponentByType(univ.OctetString().getTagSet(), 'Pascal')
|
| +>>> gamer.setComponentByType(univ.ObjectIdentifier().getTagSet(), (1,3,7,2))
|
| +>>> print(gamer.prettyPrint())
|
| +Gamer:
|
| + score=121343
|
| + player=b'Pascal'
|
| + id=1.3.7.2
|
| +>>>
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<a name="1.3.2"></a>
|
| +<h4>
|
| +1.3.2 SequenceOf and SetOf types
|
| +</h4>
|
| +
|
| +<p>
|
| +Both, SequenceOf and SetOf types resemble an unlimited size list of components.
|
| +All the components must be of the same type.
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +Progression ::= SEQUENCE OF INTEGER
|
| +
|
| +arithmeticProgression Progression ::= { 1, 3, 5, 7 }
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +SequenceOf and SetOf types are expressed by the very similar pyasn1 type
|
| +objects. Their components can only be addressed by position and they
|
| +both have a property of automatic resize.
|
| +</p>
|
| +
|
| +<p>
|
| +To specify inner component type, the <b>componentType</b> class attribute
|
| +should refer to another pyasn1 type object.
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +>>> from pyasn1.type import univ
|
| +>>> class Progression(univ.SequenceOf):
|
| +... componentType = univ.Integer()
|
| +>>> arithmeticProgression = Progression()
|
| +>>> arithmeticProgression.setComponentByPosition(1, 111)
|
| +>>> print(arithmeticProgression.prettyPrint())
|
| +Progression:
|
| +-empty- 111
|
| +>>> arithmeticProgression.setComponentByPosition(0, 100)
|
| +>>> print(arithmeticProgression.prettyPrint())
|
| +Progression:
|
| +100 111
|
| +>>>
|
| +>>> for idx in range(len(arithmeticProgression)):
|
| +... arithmeticProgression.getComponentByPosition(idx)
|
| +Integer(100)
|
| +Integer(111)
|
| +>>>
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +Any scalar or constructed pyasn1 type object can serve as an inner component.
|
| +Missing components are prohibited in SequenceOf/SetOf value objects.
|
| +</p>
|
| +
|
| +<a name="1.3.3"></a>
|
| +<h4>
|
| +1.3.3 Choice type
|
| +</h4>
|
| +
|
| +<p>
|
| +Values of ASN.1 CHOICE type can contain only a single value of a type from a
|
| +list of possible alternatives. Alternatives must be ASN.1 types with
|
| +distinct tags for the whole structure to remain unambiguous. Unlike most
|
| +other types, CHOICE is an untagged one, e.g. it has no base tag of its own.
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +CodeOrMessage ::= CHOICE {
|
| + code INTEGER,
|
| + message OCTET STRING
|
| +}
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +In pyasn1 implementation, Choice object behaves like Set but accepts only
|
| +a single inner component at a time. It also offers a few additional methods
|
| +specific to its behaviour.
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +>>> from pyasn1.type import univ, namedtype
|
| +>>> class CodeOrMessage(univ.Choice):
|
| +... componentType = namedtype.NamedTypes(
|
| +... namedtype.NamedType('code', univ.Integer()),
|
| +... namedtype.NamedType('message', univ.OctetString())
|
| +... )
|
| +>>>
|
| +>>> codeOrMessage = CodeOrMessage()
|
| +>>> print(codeOrMessage.prettyPrint())
|
| +CodeOrMessage:
|
| +>>> codeOrMessage.setComponentByName('code', 123)
|
| +>>> print(codeOrMessage.prettyPrint())
|
| +CodeOrMessage:
|
| + code=123
|
| +>>> codeOrMessage.setComponentByName('message', 'my string value')
|
| +>>> print(codeOrMessage.prettyPrint())
|
| +CodeOrMessage:
|
| + message=b'my string value'
|
| +>>>
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +Since there could be only a single inner component value in the pyasn1 Choice
|
| +value object, either of the following methods could be used for fetching it
|
| +(continuing previous code):
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +>>> codeOrMessage.getName()
|
| +'message'
|
| +>>> codeOrMessage.getComponent()
|
| +OctetString(b'my string value')
|
| +>>>
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<a name="1.3.4"></a>
|
| +<h4>
|
| +1.3.4 Any type
|
| +</h4>
|
| +
|
| +<p>
|
| +The ASN.1 ANY type is a kind of wildcard or placeholder that matches
|
| +any other type without knowing it in advance. Like CHOICE type, ANY
|
| +has no base tag.
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +Error ::= SEQUENCE {
|
| + code INTEGER,
|
| + parameter ANY DEFINED BY code
|
| +}
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +The ANY type is frequently used in specifications, where exact type is not
|
| +yet agreed upon between communicating parties or the number of possible
|
| +alternatives of a type is infinite.
|
| +Sometimes an auxiliary selector is kept around to help parties indicate
|
| +the kind of ANY payload in effect ("code" in the example above).
|
| +</p>
|
| +
|
| +<p>
|
| +Values of the ANY type contain serialized ASN.1 value(s) in form of
|
| +an octet string. Therefore pyasn1 Any value object share the properties of
|
| +pyasn1 OctetString object.
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +>>> from pyasn1.type import univ
|
| +>>> someValue = univ.Any(b'\x02\x01\x01')
|
| +>>> someValue
|
| +Any(b'\x02\x01\x01')
|
| +>>> str(someValue)
|
| +'\x02\x01\x01'
|
| +>>> bytes(someValue)
|
| +b'\x02\x01\x01'
|
| +>>>
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +Receiving application is supposed to explicitly deserialize the content of Any
|
| +value object, possibly using auxiliary selector for figuring out its ASN.1
|
| +type to pick appropriate decoder.
|
| +</p>
|
| +
|
| +<p>
|
| +There will be some more talk and code snippets covering Any type in the codecs
|
| +chapters that follow.
|
| +</p>
|
| +
|
| +<a name="1.4"></a>
|
| +<h4>
|
| +1.4 Subtype constraints
|
| +</h4>
|
| +
|
| +<p>
|
| +Most ASN.1 types can correspond to an infinite set of values. To adapt to
|
| +particular application's data model and needs, ASN.1 provides a mechanism
|
| +for limiting the infinite set to values, that make sense in particular case.
|
| +</p>
|
| +
|
| +<p>
|
| +Imposing value constraints on an ASN.1 type can also be seen as creating
|
| +a subtype from its base type.
|
| +</p>
|
| +
|
| +<p>
|
| +In pyasn1, constraints take shape of immutable objects capable
|
| +of evaluating given value against constraint-specific requirements.
|
| +Constraint object is a property of pyasn1 type. Like TagSet property,
|
| +associated with every pyasn1 type, constraints can never be modified
|
| +in place. The only way to modify pyasn1 type constraint is to associate
|
| +new constraint object to a new pyasn1 type object.
|
| +</p>
|
| +
|
| +<p>
|
| +A handful of different flavors of <i>constraints</i> are defined in ASN.1.
|
| +We will discuss them one by one in the following chapters and also explain
|
| +how to combine and apply them to types.
|
| +</p>
|
| +
|
| +<a name="1.4.1"></a>
|
| +<h4>
|
| +1.4.1 Single value constraint
|
| +</h4>
|
| +
|
| +<p>
|
| +This kind of constraint allows for limiting type to a finite, specified set
|
| +of values.
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +DialButton ::= OCTET STRING (
|
| + "0" | "1" | "2" | "3" | "4" | "5" | "6" | "7" | "8" | "9"
|
| +)
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +Its pyasn1 implementation would look like:
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +>>> from pyasn1.type import constraint
|
| +>>> c = constraint.SingleValueConstraint(
|
| + '0','1','2','3','4','5','6','7','8','9'
|
| +)
|
| +>>> c
|
| +SingleValueConstraint(0, 1, 2, 3, 4, 5, 6, 7, 8, 9)
|
| +>>> c('0')
|
| +>>> c('A')
|
| +Traceback (most recent call last):
|
| +...
|
| +pyasn1.type.error.ValueConstraintError:
|
| + SingleValueConstraint(0, 1, 2, 3, 4, 5, 6, 7, 8, 9) failed at: A
|
| +>>>
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +As can be seen in the snippet above, if a value violates the constraint, an
|
| +exception will be thrown. A constrainted pyasn1 type object holds a
|
| +reference to a constraint object (or their combination, as will be explained
|
| +later) and calls it for value verification.
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +>>> from pyasn1.type import univ, constraint
|
| +>>> class DialButton(univ.OctetString):
|
| +... subtypeSpec = constraint.SingleValueConstraint(
|
| +... '0','1','2','3','4','5','6','7','8','9'
|
| +... )
|
| +>>> DialButton('0')
|
| +DialButton(b'0')
|
| +>>> DialButton('A')
|
| +Traceback (most recent call last):
|
| +...
|
| +pyasn1.type.error.ValueConstraintError:
|
| + SingleValueConstraint(0, 1, 2, 3, 4, 5, 6, 7, 8, 9) failed at: A
|
| +>>>
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +Constrained pyasn1 value object can never hold a violating value.
|
| +</p>
|
| +
|
| +<a name="1.4.2"></a>
|
| +<h4>
|
| +1.4.2 Value range constraint
|
| +</h4>
|
| +
|
| +<p>
|
| +A pair of values, compliant to a type to be constrained, denote low and upper
|
| +bounds of allowed range of values of a type.
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +Teenagers ::= INTEGER (13..19)
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +And in pyasn1 terms:
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +>>> from pyasn1.type import univ, constraint
|
| +>>> class Teenagers(univ.Integer):
|
| +... subtypeSpec = constraint.ValueRangeConstraint(13, 19)
|
| +>>> Teenagers(14)
|
| +Teenagers(14)
|
| +>>> Teenagers(20)
|
| +Traceback (most recent call last):
|
| +...
|
| +pyasn1.type.error.ValueConstraintError:
|
| + ValueRangeConstraint(13, 19) failed at: 20
|
| +>>>
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +Value range constraint usually applies numeric types.
|
| +</p>
|
| +
|
| +<a name="1.4.3"></a>
|
| +<h4>
|
| +1.4.3 Size constraint
|
| +</h4>
|
| +
|
| +<p>
|
| +It is sometimes convenient to set or limit the allowed size of a data item
|
| +to be sent from one application to another to manage bandwidth and memory
|
| +consumption issues. Size constraint specifies the lower and upper bounds
|
| +of the size of a valid value.
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +TwoBits ::= BIT STRING (SIZE (2))
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +Express the same grammar in pyasn1:
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +>>> from pyasn1.type import univ, constraint
|
| +>>> class TwoBits(univ.BitString):
|
| +... subtypeSpec = constraint.ValueSizeConstraint(2, 2)
|
| +>>> TwoBits((1,1))
|
| +TwoBits("'11'B")
|
| +>>> TwoBits((1,1,0))
|
| +Traceback (most recent call last):
|
| +...
|
| +pyasn1.type.error.ValueConstraintError:
|
| + ValueSizeConstraint(2, 2) failed at: (1, 1, 0)
|
| +>>>
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +Size constraint can be applied to potentially massive values - bit or octet
|
| +strings, SEQUENCE OF/SET OF values.
|
| +</p>
|
| +
|
| +<a name="1.4.4"></a>
|
| +<h4>
|
| +1.4.4 Alphabet constraint
|
| +</h4>
|
| +
|
| +<p>
|
| +The permitted alphabet constraint is similar to Single value constraint
|
| +but constraint applies to individual characters of a value.
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +MorseCode ::= PrintableString (FROM ("."|"-"|" "))
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +And in pyasn1:
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +>>> from pyasn1.type import char, constraint
|
| +>>> class MorseCode(char.PrintableString):
|
| +... subtypeSpec = constraint.PermittedAlphabetConstraint(".", "-", " ")
|
| +>>> MorseCode("...---...")
|
| +MorseCode('...---...')
|
| +>>> MorseCode("?")
|
| +Traceback (most recent call last):
|
| +...
|
| +pyasn1.type.error.ValueConstraintError:
|
| + PermittedAlphabetConstraint(".", "-", " ") failed at: "?"
|
| +>>>
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +Current implementation does not handle ranges of characters in constraint
|
| +(FROM "A".."Z" syntax), one has to list the whole set in a range.
|
| +</p>
|
| +
|
| +<a name="1.4.5"></a>
|
| +<h4>
|
| +1.4.5 Constraint combinations
|
| +</h4>
|
| +
|
| +<p>
|
| +Up to this moment, we used a single constraint per ASN.1 type. The standard,
|
| +however, allows for combining multiple individual constraints into
|
| +intersections, unions and exclusions.
|
| +</p>
|
| +
|
| +<p>
|
| +In pyasn1 data model, all of these methods of constraint combinations are
|
| +implemented as constraint-like objects holding individual constraint (or
|
| +combination) objects. Like terminal constraint objects, combination objects
|
| +are capable to perform value verification at its set of enclosed constraints
|
| +according to the logic of particular combination.
|
| +</p>
|
| +
|
| +<p>
|
| +Constraints intersection verification succeeds only if a value is
|
| +compliant to each constraint in a set. To begin with, the following
|
| +specification will constitute a valid telephone number:
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +PhoneNumber ::= NumericString (FROM ("0".."9")) (SIZE 11)
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +Constraint intersection object serves the logic above:
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +>>> from pyasn1.type import char, constraint
|
| +>>> class PhoneNumber(char.NumericString):
|
| +... subtypeSpec = constraint.ConstraintsIntersection(
|
| +... constraint.PermittedAlphabetConstraint('0','1','2','3','4','5','6','7','8','9'),
|
| +... constraint.ValueSizeConstraint(11, 11)
|
| +... )
|
| +>>> PhoneNumber('79039343212')
|
| +PhoneNumber('79039343212')
|
| +>>> PhoneNumber('?9039343212')
|
| +Traceback (most recent call last):
|
| +...
|
| +pyasn1.type.error.ValueConstraintError:
|
| + ConstraintsIntersection(
|
| + PermittedAlphabetConstraint('0','1','2','3','4','5','6','7','8','9'),
|
| + ValueSizeConstraint(11, 11)) failed at:
|
| + PermittedAlphabetConstraint('0','1','2','3','4','5','6','7','8','9') failed at: "?039343212"
|
| +>>> PhoneNumber('9343212')
|
| +Traceback (most recent call last):
|
| +...
|
| +pyasn1.type.error.ValueConstraintError:
|
| + ConstraintsIntersection(
|
| + PermittedAlphabetConstraint('0','1','2','3','4','5','6','7','8','9'),
|
| + ValueSizeConstraint(11, 11)) failed at:
|
| + ValueSizeConstraint(10, 10) failed at: "9343212"
|
| +>>>
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +Union of constraints works by making sure that a value is compliant
|
| +to any of the constraint in a set. For instance:
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +CapitalOrSmall ::= IA5String (FROM ('A','B','C') | FROM ('a','b','c'))
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +It's important to note, that a value must fully comply to any single
|
| +constraint in a set. In the specification above, a value of all small or
|
| +all capital letters is compliant, but a mix of small&capitals is not.
|
| +Here's its pyasn1 analogue:
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +>>> from pyasn1.type import char, constraint
|
| +>>> class CapitalOrSmall(char.IA5String):
|
| +... subtypeSpec = constraint.ConstraintsUnion(
|
| +... constraint.PermittedAlphabetConstraint('A','B','C'),
|
| +... constraint.PermittedAlphabetConstraint('a','b','c')
|
| +... )
|
| +>>> CapitalOrSmall('ABBA')
|
| +CapitalOrSmall('ABBA')
|
| +>>> CapitalOrSmall('abba')
|
| +CapitalOrSmall('abba')
|
| +>>> CapitalOrSmall('Abba')
|
| +Traceback (most recent call last):
|
| +...
|
| +pyasn1.type.error.ValueConstraintError:
|
| + ConstraintsUnion(PermittedAlphabetConstraint('A', 'B', 'C'),
|
| + PermittedAlphabetConstraint('a', 'b', 'c')) failed at: failed for "Abba"
|
| +>>>
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +Finally, the exclusion constraint simply negates the logic of value
|
| +verification at a constraint. In the following example, any integer value
|
| +is allowed in a type but not zero.
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +NoZero ::= INTEGER (ALL EXCEPT 0)
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +In pyasn1 the above definition would read:
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +>>> from pyasn1.type import univ, constraint
|
| +>>> class NoZero(univ.Integer):
|
| +... subtypeSpec = constraint.ConstraintsExclusion(
|
| +... constraint.SingleValueConstraint(0)
|
| +... )
|
| +>>> NoZero(1)
|
| +NoZero(1)
|
| +>>> NoZero(0)
|
| +Traceback (most recent call last):
|
| +...
|
| +pyasn1.type.error.ValueConstraintError:
|
| + ConstraintsExclusion(SingleValueConstraint(0)) failed at: 0
|
| +>>>
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +The depth of such a constraints tree, built with constraint combination objects
|
| +at its nodes, has not explicit limit. Value verification is performed in a
|
| +recursive manner till a definite solution is found.
|
| +</p>
|
| +
|
| +<a name="1.5"></a>
|
| +<h4>
|
| +1.5 Types relationships
|
| +</h4>
|
| +
|
| +<p>
|
| +In the course of data processing in an application, it is sometimes
|
| +convenient to figure out the type relationships between pyasn1 type or
|
| +value objects. Formally, two things influence pyasn1 types relationship:
|
| +<i>tag set</i> and <i>subtype constraints</i>. One pyasn1 type is considered
|
| +to be a derivative of another if their TagSet and Constraint objects are
|
| +a derivation of one another.
|
| +</p>
|
| +
|
| +<p>
|
| +The following example illustrates the concept (we use the same tagset but
|
| +different constraints for simplicity):
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +>>> from pyasn1.type import univ, constraint
|
| +>>> i1 = univ.Integer(subtypeSpec=constraint.ValueRangeConstraint(3,8))
|
| +>>> i2 = univ.Integer(subtypeSpec=constraint.ConstraintsIntersection(
|
| +... constraint.ValueRangeConstraint(3,8),
|
| +... constraint.ValueRangeConstraint(4,7)
|
| +... ) )
|
| +>>> i1.isSameTypeWith(i2)
|
| +False
|
| +>>> i1.isSuperTypeOf(i2)
|
| +True
|
| +>>> i1.isSuperTypeOf(i1)
|
| +True
|
| +>>> i2.isSuperTypeOf(i1)
|
| +False
|
| +>>>
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +As can be seen in the above code snippet, there are two methods of any pyasn1
|
| +type/value object that test types for their relationship:
|
| +<b>isSameTypeWith</b>() and <b>isSuperTypeOf</b>(). The former is
|
| +self-descriptive while the latter yields true if the argument appears
|
| +to be a pyasn1 object which has tagset and constraints derived from those
|
| +of the object being called.
|
| +</p>
|
| +
|
| +<a name="2"></a>
|
| +<h3>
|
| +2. Codecs
|
| +</h3>
|
| +
|
| +<p>
|
| +In ASN.1 context,
|
| +<a href=http://en.wikipedia.org/wiki/Codec>codec</a>
|
| +is a program that transforms between concrete data structures and a stream
|
| +of octets, suitable for transmission over the wire. This serialized form of
|
| +data is sometimes called <i>substrate</i> or <i>essence</i>.
|
| +</p>
|
| +
|
| +<p>
|
| +In pyasn1 implementation, substrate takes shape of Python 3 bytes or
|
| +Python 2 string objects.
|
| +</p>
|
| +
|
| +<p>
|
| +One of the properties of a codec is its ability to cope with incomplete
|
| +data and/or substrate what implies codec to be stateful. In other words,
|
| +when decoder runs out of substrate and data item being recovered is still
|
| +incomplete, stateful codec would suspend and complete data item recovery
|
| +whenever the rest of substrate becomes available. Similarly, stateful encoder
|
| +would encode data items in multiple steps waiting for source data to
|
| +arrive. Codec restartability is especially important when application deals
|
| +with large volumes of data and/or runs on low RAM. For an interesting
|
| +discussion on codecs options and design choices, refer to
|
| +<a href=http://directory.apache.org/subprojects/asn1/>Apache ASN.1 project</a>
|
| +.
|
| +</p>
|
| +
|
| +<p>
|
| +As of this writing, codecs implemented in pyasn1 are all stateless, mostly
|
| +to keep the code simple.
|
| +</p>
|
| +
|
| +<p>
|
| +The pyasn1 package currently supports
|
| +<a href=http://en.wikipedia.org/wiki/Basic_encoding_rules>BER</a> codec and
|
| +its variations --
|
| +<a href=http://en.wikipedia.org/wiki/Canonical_encoding_rules>CER</a> and
|
| +<a href=http://en.wikipedia.org/wiki/Distinguished_encoding_rules>DER</a>.
|
| +More ASN.1 codecs are planned for implementation in the future.
|
| +</p>
|
| +
|
| +<a name="2.1"></a>
|
| +<h4>
|
| +2.1 Encoders
|
| +</h4>
|
| +
|
| +<p>
|
| +Encoder is used for transforming pyasn1 value objects into substrate. Only
|
| +pyasn1 value objects could be serialized, attempts to process pyasn1 type
|
| +objects will cause encoder failure.
|
| +</p>
|
| +
|
| +<p>
|
| +The following code will create a pyasn1 Integer object and serialize it with
|
| +BER encoder:
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +>>> from pyasn1.type import univ
|
| +>>> from pyasn1.codec.ber import encoder
|
| +>>> encoder.encode(univ.Integer(123456))
|
| +b'\x02\x03\x01\xe2@'
|
| +>>>
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +BER standard also defines a so-called <i>indefinite length</i> encoding form
|
| +which makes large data items processing more memory efficient. It is mostly
|
| +useful when encoder does not have the whole value all at once and the
|
| +length of the value can not be determined at the beginning of encoding.
|
| +</p>
|
| +
|
| +<p>
|
| +<i>Constructed encoding</i> is another feature of BER closely related to the
|
| +indefinite length form. In essence, a large scalar value (such as ASN.1
|
| +character BitString type) could be chopped into smaller chunks by encoder
|
| +and transmitted incrementally to limit memory consumption. Unlike indefinite
|
| +length case, the length of the whole value must be known in advance when
|
| +using constructed, definite length encoding form.
|
| +</p>
|
| +
|
| +<p>
|
| +Since pyasn1 codecs are not restartable, pyasn1 encoder may only encode data
|
| +item all at once. However, even in this case, generating indefinite length
|
| +encoding may help a low-memory receiver, running a restartable decoder,
|
| +to process a large data item.
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +>>> from pyasn1.type import univ
|
| +>>> from pyasn1.codec.ber import encoder
|
| +>>> encoder.encode(
|
| +... univ.OctetString('The quick brown fox jumps over the lazy dog'),
|
| +... defMode=False,
|
| +... maxChunkSize=8
|
| +... )
|
| +b'$\x80\x04\x08The quic\x04\x08k brown \x04\x08fox jump\x04\x08s over \
|
| +t\x04\x08he lazy \x04\x03dog\x00\x00'
|
| +>>>
|
| +>>> encoder.encode(
|
| +... univ.OctetString('The quick brown fox jumps over the lazy dog'),
|
| +... maxChunkSize=8
|
| +... )
|
| +b'$7\x04\x08The quic\x04\x08k brown \x04\x08fox jump\x04\x08s over \
|
| +t\x04\x08he lazy \x04\x03dog'
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +The <b>defMode</b> encoder parameter disables definite length encoding mode,
|
| +while the optional <b>maxChunkSize</b> parameter specifies desired
|
| +substrate chunk size that influences memory requirements at the decoder's end.
|
| +</p>
|
| +
|
| +<p>
|
| +To use CER or DER encoders one needs to explicitly import and call them - the
|
| +APIs are all compatible.
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +>>> from pyasn1.type import univ
|
| +>>> from pyasn1.codec.ber import encoder as ber_encoder
|
| +>>> from pyasn1.codec.cer import encoder as cer_encoder
|
| +>>> from pyasn1.codec.der import encoder as der_encoder
|
| +>>> ber_encoder.encode(univ.Boolean(True))
|
| +b'\x01\x01\x01'
|
| +>>> cer_encoder.encode(univ.Boolean(True))
|
| +b'\x01\x01\xff'
|
| +>>> der_encoder.encode(univ.Boolean(True))
|
| +b'\x01\x01\xff'
|
| +>>>
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<a name="2.2"></a>
|
| +<h4>
|
| +2.2 Decoders
|
| +</h4>
|
| +
|
| +<p>
|
| +In the process of decoding, pyasn1 value objects are created and linked to
|
| +each other, based on the information containted in the substrate. Thus,
|
| +the original pyasn1 value object(s) are recovered.
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +>>> from pyasn1.type import univ
|
| +>>> from pyasn1.codec.ber import encoder, decoder
|
| +>>> substrate = encoder.encode(univ.Boolean(True))
|
| +>>> decoder.decode(substrate)
|
| +(Boolean('True(1)'), b'')
|
| +>>>
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +Commenting on the code snippet above, pyasn1 decoder accepts substrate
|
| +as an argument and returns a tuple of pyasn1 value object (possibly
|
| +a top-level one in case of constructed object) and unprocessed part
|
| +of input substrate.
|
| +</p>
|
| +
|
| +<p>
|
| +All pyasn1 decoders can handle both definite and indefinite length
|
| +encoding modes automatically, explicit switching into one mode
|
| +to another is not required.
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +>>> from pyasn1.type import univ
|
| +>>> from pyasn1.codec.ber import encoder, decoder
|
| +>>> substrate = encoder.encode(
|
| +... univ.OctetString('The quick brown fox jumps over the lazy dog'),
|
| +... defMode=False,
|
| +... maxChunkSize=8
|
| +... )
|
| +>>> decoder.decode(substrate)
|
| +(OctetString(b'The quick brown fox jumps over the lazy dog'), b'')
|
| +>>>
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +Speaking of BER/CER/DER encoding, in many situations substrate may not contain
|
| +all necessary information needed for complete and accurate ASN.1 values
|
| +recovery. The most obvious cases include implicitly tagged ASN.1 types
|
| +and constrained types.
|
| +</p>
|
| +
|
| +<p>
|
| +As discussed earlier in this handbook, when an ASN.1 type is implicitly
|
| +tagged, previous outermost tag is lost and never appears in substrate.
|
| +If it is the base tag that gets lost, decoder is unable to pick type-specific
|
| +value decoder at its table of built-in types, and therefore recover
|
| +the value part, based only on the information contained in substrate. The
|
| +approach taken by pyasn1 decoder is to use a prototype pyasn1 type object (or
|
| +a set of them) to <i>guide</i> the decoding process by matching [possibly
|
| +incomplete] tags recovered from substrate with those found in prototype pyasn1
|
| +type objects (also called pyasn1 specification object further in this paper).
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +>>> from pyasn1.codec.ber import decoder
|
| +>>> decoder.decode(b'\x02\x01\x0c', asn1Spec=univ.Integer())
|
| +Integer(12), b''
|
| +>>>
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +Decoder would neither modify pyasn1 specification object nor use
|
| +its current values (if it's a pyasn1 value object), but rather use it as
|
| +a hint for choosing proper decoder and as a pattern for creating new objects:
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +>>> from pyasn1.type import univ, tag
|
| +>>> from pyasn1.codec.ber import encoder, decoder
|
| +>>> i = univ.Integer(12345).subtype(
|
| +... implicitTag=tag.Tag(tag.tagClassContext, tag.tagFormatSimple, 40)
|
| +... )
|
| +>>> substrate = encoder.encode(i)
|
| +>>> substrate
|
| +b'\x9f(\x0209'
|
| +>>> decoder.decode(substrate)
|
| +Traceback (most recent call last):
|
| +...
|
| +pyasn1.error.PyAsn1Error:
|
| + TagSet(Tag(tagClass=128, tagFormat=0, tagId=40)) not in asn1Spec
|
| +>>> decoder.decode(substrate, asn1Spec=i)
|
| +(Integer(12345), b'')
|
| +>>>
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +Notice in the example above, that an attempt to run decoder without passing
|
| +pyasn1 specification object fails because recovered tag does not belong
|
| +to any of the built-in types.
|
| +</p>
|
| +
|
| +<p>
|
| +Another important feature of guided decoder operation is the use of
|
| +values constraints possibly present in pyasn1 specification object.
|
| +To explain this, we will decode a random integer object into generic Integer
|
| +and the constrained one.
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +>>> from pyasn1.type import univ, constraint
|
| +>>> from pyasn1.codec.ber import encoder, decoder
|
| +>>> class DialDigit(univ.Integer):
|
| +... subtypeSpec = constraint.ValueRangeConstraint(0,9)
|
| +>>> substrate = encoder.encode(univ.Integer(13))
|
| +>>> decoder.decode(substrate)
|
| +(Integer(13), b'')
|
| +>>> decoder.decode(substrate, asn1Spec=DialDigit())
|
| +Traceback (most recent call last):
|
| +...
|
| +pyasn1.type.error.ValueConstraintError:
|
| + ValueRangeConstraint(0, 9) failed at: 13
|
| +>>>
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +Similarily to encoders, to use CER or DER decoders application has to
|
| +explicitly import and call them - all APIs are compatible.
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +>>> from pyasn1.type import univ
|
| +>>> from pyasn1.codec.ber import encoder as ber_encoder
|
| +>>> substrate = ber_encoder.encode(univ.OctetString('http://pyasn1.sf.net'))
|
| +>>>
|
| +>>> from pyasn1.codec.ber import decoder as ber_decoder
|
| +>>> from pyasn1.codec.cer import decoder as cer_decoder
|
| +>>> from pyasn1.codec.der import decoder as der_decoder
|
| +>>>
|
| +>>> ber_decoder.decode(substrate)
|
| +(OctetString(b'http://pyasn1.sf.net'), b'')
|
| +>>> cer_decoder.decode(substrate)
|
| +(OctetString(b'http://pyasn1.sf.net'), b'')
|
| +>>> der_decoder.decode(substrate)
|
| +(OctetString(b'http://pyasn1.sf.net'), b'')
|
| +>>>
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<a name="2.2.1"></a>
|
| +<h4>
|
| +2.2.1 Decoding untagged types
|
| +</h4>
|
| +
|
| +<p>
|
| +It has already been mentioned, that ASN.1 has two "special case" types:
|
| +CHOICE and ANY. They are different from other types in part of
|
| +tagging - unless these two are additionally tagged, neither of them will
|
| +have their own tag. Therefore these types become invisible in substrate
|
| +and can not be recovered without passing pyasn1 specification object to
|
| +decoder.
|
| +</p>
|
| +
|
| +<p>
|
| +To explain the issue, we will first prepare a Choice object to deal with:
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +>>> from pyasn1.type import univ, namedtype
|
| +>>> class CodeOrMessage(univ.Choice):
|
| +... componentType = namedtype.NamedTypes(
|
| +... namedtype.NamedType('code', univ.Integer()),
|
| +... namedtype.NamedType('message', univ.OctetString())
|
| +... )
|
| +>>>
|
| +>>> codeOrMessage = CodeOrMessage()
|
| +>>> codeOrMessage.setComponentByName('message', 'my string value')
|
| +>>> print(codeOrMessage.prettyPrint())
|
| +CodeOrMessage:
|
| + message=b'my string value'
|
| +>>>
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +Let's now encode this Choice object and then decode its substrate
|
| +with and without pyasn1 specification object:
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +>>> from pyasn1.codec.ber import encoder, decoder
|
| +>>> substrate = encoder.encode(codeOrMessage)
|
| +>>> substrate
|
| +b'\x04\x0fmy string value'
|
| +>>> encoder.encode(univ.OctetString('my string value'))
|
| +b'\x04\x0fmy string value'
|
| +>>>
|
| +>>> decoder.decode(substrate)
|
| +(OctetString(b'my string value'), b'')
|
| +>>> codeOrMessage, substrate = decoder.decode(substrate, asn1Spec=CodeOrMessage())
|
| +>>> print(codeOrMessage.prettyPrint())
|
| +CodeOrMessage:
|
| + message=b'my string value'
|
| +>>>
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +First thing to notice in the listing above is that the substrate produced
|
| +for our Choice value object is equivalent to the substrate for an OctetString
|
| +object initialized to the same value. In other words, any information about
|
| +the Choice component is absent in encoding.
|
| +</p>
|
| +
|
| +<p>
|
| +Sure enough, that kind of substrate will decode into an OctetString object,
|
| +unless original Choice type object is passed to decoder to guide the decoding
|
| +process.
|
| +</p>
|
| +
|
| +<p>
|
| +Similarily untagged ANY type behaves differently on decoding phase - when
|
| +decoder bumps into an Any object in pyasn1 specification, it stops decoding
|
| +and puts all the substrate into a new Any value object in form of an octet
|
| +string. Concerned application could then re-run decoder with an additional,
|
| +more exact pyasn1 specification object to recover the contents of Any
|
| +object.
|
| +</p>
|
| +
|
| +<p>
|
| +As it was mentioned elsewhere in this paper, Any type allows for incomplete
|
| +or changing ASN.1 specification to be handled gracefully by decoder and
|
| +applications.
|
| +</p>
|
| +
|
| +<p>
|
| +To illustrate the working of Any type, we'll have to make the stage
|
| +by encoding a pyasn1 object and then putting its substrate into an any
|
| +object.
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +>>> from pyasn1.type import univ
|
| +>>> from pyasn1.codec.ber import encoder, decoder
|
| +>>> innerSubstrate = encoder.encode(univ.Integer(1234))
|
| +>>> innerSubstrate
|
| +b'\x02\x02\x04\xd2'
|
| +>>> any = univ.Any(innerSubstrate)
|
| +>>> any
|
| +Any(b'\x02\x02\x04\xd2')
|
| +>>> substrate = encoder.encode(any)
|
| +>>> substrate
|
| +b'\x02\x02\x04\xd2'
|
| +>>>
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +As with Choice type encoding, there is no traces of Any type in substrate.
|
| +Obviously, the substrate we are dealing with, will decode into the inner
|
| +[Integer] component, unless pyasn1 specification is given to guide the
|
| +decoder. Continuing previous code:
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +>>> from pyasn1.type import univ
|
| +>>> from pyasn1.codec.ber import encoder, decoder
|
| +
|
| +>>> decoder.decode(substrate)
|
| +(Integer(1234), b'')
|
| +>>> any, substrate = decoder.decode(substrate, asn1Spec=univ.Any())
|
| +>>> any
|
| +Any(b'\x02\x02\x04\xd2')
|
| +>>> decoder.decode(str(any))
|
| +(Integer(1234), b'')
|
| +>>>
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +Both CHOICE and ANY types are widely used in practice. Reader is welcome to
|
| +take a look at
|
| +<a href=http://www.cs.auckland.ac.nz/~pgut001/pubs/x509guide.txt>
|
| +ASN.1 specifications of X.509 applications</a> for more information.
|
| +</p>
|
| +
|
| +<a name="2.2.2"></a>
|
| +<h4>
|
| +2.2.2 Ignoring unknown types
|
| +</h4>
|
| +
|
| +<p>
|
| +When dealing with a loosely specified ASN.1 structure, the receiving
|
| +end may not be aware of some types present in the substrate. It may be
|
| +convenient then to turn decoder into a recovery mode. Whilst there, decoder
|
| +will not bail out when hit an unknown tag but rather treat it as an Any
|
| +type.
|
| +</p>
|
| +
|
| +<table bgcolor="lightgray" border=0 width=100%><TR><TD>
|
| +<pre>
|
| +>>> from pyasn1.type import univ, tag
|
| +>>> from pyasn1.codec.ber import encoder, decoder
|
| +>>> taggedInt = univ.Integer(12345).subtype(
|
| +... implicitTag=tag.Tag(tag.tagClassContext, tag.tagFormatSimple, 40)
|
| +... )
|
| +>>> substrate = encoder.encode(taggedInt)
|
| +>>> decoder.decode(substrate)
|
| +Traceback (most recent call last):
|
| +...
|
| +pyasn1.error.PyAsn1Error: TagSet(Tag(tagClass=128, tagFormat=0, tagId=40)) not in asn1Spec
|
| +>>>
|
| +>>> decoder.decode.defaultErrorState = decoder.stDumpRawValue
|
| +>>> decoder.decode(substrate)
|
| +(Any(b'\x9f(\x0209'), '')
|
| +>>>
|
| +</pre>
|
| +</td></tr></table>
|
| +
|
| +<p>
|
| +It's also possible to configure a custom decoder, to handle unknown tags
|
| +found in substrate. This can be done by means of <b>defaultRawDecoder</b>
|
| +attribute holding a reference to type decoder object. Refer to the source
|
| +for API details.
|
| +</p>
|
| +
|
| +<a name="3"></a>
|
| +<h3>
|
| +3. Feedback and getting help
|
| +</h3>
|
| +
|
| +<p>
|
| +Although pyasn1 software is almost a decade old and used in many production
|
| +environments, it still may have bugs and non-implemented pieces. Anyone
|
| +who happens to run into such defect is welcome to complain to
|
| +<a href=mailto:pyasn1-users@lists.sourceforge.net>pyasn1 mailing list</a>
|
| +or better yet fix the issue and send
|
| +<a href=mailto:ilya@glas.net>me</a> the patch.
|
| +</p>
|
| +
|
| +<p>
|
| +Typically, pyasn1 is used for building arbitrary protocol support into
|
| +various applications. This involves manual translation of ASN.1 data
|
| +structures into their pyasn1 implementations. To save time and effort,
|
| +data structures for some of the popular protocols are pre-programmed
|
| +and kept for further re-use in form of the
|
| +<a href=http://sourceforge.net/projects/pyasn1/files/pyasn1-modules/>
|
| +pyasn1-modules package</a>. For instance, many structures for PKI (X.509,
|
| +PKCS#*, CRMF, OCSP), LDAP and SNMP are present.
|
| +Applications authors are advised to import and use relevant modules
|
| +from that package whenever needed protocol structures are already
|
| +there. New protocol modules contributions are welcome.
|
| +</p>
|
| +
|
| +<p>
|
| +And finally, the latest pyasn1 package revision is available for free
|
| +download from
|
| +<a href=http://sourceforge.net/projects/pyasn1/>project home</a> and
|
| +also from the
|
| +<a href=http://pypi.python.org/pypi>Python package repository</a>.
|
| +</p>
|
| +
|
| +<hr>
|
| +
|
| +</td>
|
| +</tr>
|
| +</table>
|
| +</center>
|
| +</body>
|
| +</html>
|
|
|