Class: RASN1::Types::Base Abstract
- Inherits:
-
Object
- Object
- RASN1::Types::Base
- Defined in:
- lib/rasn1/types/base.rb,
lib/rasn1/tracer.rb
Overview
This is base class for all ASN.1 types.
Subclasses SHOULD define:
-
an ID constant defining ASN.1 BER/DER identification number,
-
a method #der_to_value converting DER into #value.
-
a private method #value_to_der converting its #value to DER,
Define an optional value
An optional value may be defined using :optional
key from #initialize:
Integer.new(:int, optional: true)
An optional value implies:
-
while parsing, if decoded ID is not optional expected ID, no ASN1Error is raised, and parser tries next field,
-
while encoding, if #value is
nil
, this value is not encoded.
Define a default value
A default value may be defined using :default
key from #initialize:
Integer.new(:int, default: 0)
A default value implies:
-
while parsing, if decoded ID is not expected one, no ASN1Error is raised and parser sets default value to this ID. Then parser tries next field,
-
while encoding, if #value is equal to default value, this value is not encoded.
Define a tagged value
ASN.1 permits to define tagged values. By example:
-- context specific tag
CType ::= [0] EXPLICIT INTEGER
-- application specific tag
AType ::= [APPLICATION 1] EXPLICIT INTEGER
-- private tag
PType ::= [PRIVATE 2] EXPLICIT INTEGER
These types may be defined as:
ctype = RASN1::Types::Integer.new(explicit: 0) # with explicit, default #asn1_class is :context
atype = RASN1::Types::Integer.new(explicit: 1, class: :application)
ptype = RASN1::Types::Integer.new(explicit: 2, class: :private)
Sometimes, an EXPLICIT type should be CONSTRUCTED. To do that, use :constructed
option:
ptype = RASN1::Types::Integer.new(explicit: 2, class: :private, constructed: true)
Implicit tagged values may also be defined:
ctype_implicit = RASN1::Types::Integer.new(implicit: 0)
Direct Known Subclasses
Any, Choice, Constructed, Primitive, Wrapper::ExplicitWrapper
Constant Summary collapse
- CLASSES =
Allowed ASN.1 classes
{ universal: 0x00, application: 0x40, context: 0x80, private: 0xc0 }.freeze
- CLASS_MASK =
Binary mask to get class
0xc0
- MULTI_OCTETS_ID =
0x1f
- INDEFINITE_LENGTH =
Length value for indefinite length
0x80
Instance Attribute Summary collapse
- #asn1_class ⇒ Symbol readonly
-
#default ⇒ Object?
readonly
Default value, if defined.
- #name ⇒ String? readonly
- #options ⇒ Hash[Symbol, Object]
Class Method Summary collapse
-
.constrained? ⇒ Booleran
Say if a type is constrained.
-
.encoded_type ⇒ String
Get ASN.1 type used to encode this one.
-
.parse(der_or_ber, options = {}) ⇒ Object
Parse a DER or BER string.
-
.start_tracing ⇒ Object
Patch #do_parse to add tracing ability.
-
.stop_tracing ⇒ Object
Unpatch #do_parse to remove tracing ability.
-
.type ⇒ String
Get ASN.1 type.
Instance Method Summary collapse
-
#==(other) ⇒ Boolean
Objects are equal if they have same class AND same DER.
-
#can_build? ⇒ Boolean
Say if DER can be built (not default value, not optional without value, has a value).
-
#constructed? ⇒ ::Boolean
true
if this is a constructed type. -
#der_to_value(der, ber: false) ⇒ void
Make value from DER/BER string.
- #do_parse(der, ber: false) ⇒ Array(::Integer, String)
- #do_parse_explicit(data) ⇒ void
- #do_parse_explicit_with_tracing(data) ⇒ Object
-
#do_parse_with_tracing(der, ber:) ⇒ Object
Parse
der
with tracing abillity. -
#explicit? ⇒ ::Boolean?
Say if a tagged type is explicit.
-
#id ⇒ Integer
Get identifier value.
-
#implicit? ⇒ ::Boolean?
Say if a tagged type is implicit.
-
#initialize(options = {}) ⇒ Base
constructor
A new instance of Base.
-
#initialize_copy ⇒ Object
Deep copy @value and @default.
- #inspect(level = 0) ⇒ String
-
#optional? ⇒ ::Boolean
Say if this type is optional.
-
#parse!(der, ber: false) ⇒ Integer
abstract
Parse a DER string.
-
#primitive? ⇒ ::Boolean
true
if this is a primitive type. - #specific_initializer ⇒ Object abstract
-
#tagged? ⇒ ::Boolean
Say if this type is tagged or not.
-
#to_der ⇒ String
abstract
DER-formated string.
- #trace ⇒ String
-
#type ⇒ String
Get ASN.1 type.
-
#value ⇒ Object
Get value or default value.
-
#value=(val) ⇒ Object
Set value.
-
#value? ⇒ Boolean
Say if a value is set.
-
#value_size ⇒ Integer
Give size in octets of encoded value.
- #void_value ⇒ Object abstract
Constructor Details
#initialize(options = {}) ⇒ Base
Returns a new instance of Base.
125 126 127 128 129 130 131 132 |
# File 'lib/rasn1/types/base.rb', line 125 def initialize(={}) @constructed = nil set_value(.delete(:value)) self. = specific_initializer @raw_data = ''.b @raw_length = ''.b end |
Instance Attribute Details
#asn1_class ⇒ Symbol (readonly)
69 70 71 |
# File 'lib/rasn1/types/base.rb', line 69 def asn1_class @asn1_class end |
#default ⇒ Object? (readonly)
Returns default value, if defined.
71 72 73 |
# File 'lib/rasn1/types/base.rb', line 71 def default @default end |
#name ⇒ String? (readonly)
67 68 69 |
# File 'lib/rasn1/types/base.rb', line 67 def name @name end |
#options ⇒ Hash[Symbol, Object]
73 74 75 |
# File 'lib/rasn1/types/base.rb', line 73 def @options end |
Class Method Details
.constrained? ⇒ Booleran
Say if a type is constrained. Always return false
for predefined types
109 110 111 |
# File 'lib/rasn1/types/base.rb', line 109 def self.constrained? false end |
.encoded_type ⇒ String
Get ASN.1 type used to encode this one
91 92 93 |
# File 'lib/rasn1/types/base.rb', line 91 def self.encoded_type type end |
.parse(der_or_ber, options = {}) ⇒ Object
More options are supported. See #initialize.
Parse a DER or BER string
100 101 102 103 104 |
# File 'lib/rasn1/types/base.rb', line 100 def self.parse(der_or_ber, ={}) obj = self.new() obj.parse!(der_or_ber, ber: [:ber]) obj end |
.start_tracing ⇒ Object
Patch #do_parse to add tracing ability
68 69 70 71 72 73 |
# File 'lib/rasn1/tracer.rb', line 68 def start_tracing alias_method :do_parse_without_tracing, :do_parse alias_method :do_parse, :do_parse_with_tracing alias_method :do_parse_explicit_without_tracing, :do_parse_explicit alias_method :do_parse_explicit, :do_parse_explicit_with_tracing end |
.stop_tracing ⇒ Object
Unpatch #do_parse to remove tracing ability
77 78 79 80 |
# File 'lib/rasn1/tracer.rb', line 77 def stop_tracing alias_method :do_parse, :do_parse_without_tracing alias_method :do_parse_explicit, :do_parse_explicit_without_tracing end |
.type ⇒ String
Get ASN.1 type
83 84 85 86 87 |
# File 'lib/rasn1/types/base.rb', line 83 def self.type return @type if defined? @type @type = self.to_s.gsub(/.*::/, '').gsub(/([a-z0-9])([A-Z])/, '\1 \2').upcase end |
Instance Method Details
#==(other) ⇒ Boolean
Objects are equal if they have same class AND same DER
259 260 261 |
# File 'lib/rasn1/types/base.rb', line 259 def ==(other) (other.class == self.class) && (other.to_der == self.to_der) end |
#can_build? ⇒ Boolean
Say if DER can be built (not default value, not optional without value, has a value)
286 287 288 |
# File 'lib/rasn1/types/base.rb', line 286 def can_build? value? && (@default.nil? || (@value != @default)) end |
#constructed? ⇒ ::Boolean
Returns true
if this is a constructed type.
204 205 206 |
# File 'lib/rasn1/types/base.rb', line 204 def constructed? (self.class < Constructed) || !!@constructed end |
#der_to_value(der, ber: false) ⇒ void
This method returns an undefined value.
Make value from DER/BER string
334 335 336 |
# File 'lib/rasn1/types/base.rb', line 334 def der_to_value(der, ber: false) # rubocop:disable Lint/UnusedMethodArgument @value = der end |
#do_parse(der, ber: false) ⇒ Array(::Integer, String)
308 309 310 311 312 313 314 315 316 317 |
# File 'lib/rasn1/types/base.rb', line 308 def do_parse(der, ber: false) return [0, ''] unless check_id(der) id_size = Types.decode_identifier_octets(der).last total_length, data = get_data(der[id_size..], ber) total_length += id_size @no_value = false [total_length, data] end |
#do_parse_explicit(data) ⇒ void
This method returns an undefined value.
323 324 325 326 327 |
# File 'lib/rasn1/types/base.rb', line 323 def do_parse_explicit(data) type = explicit_type type.parse!(data) @value = type.value end |
#do_parse_explicit_with_tracing(data) ⇒ Object
92 93 94 95 96 |
# File 'lib/rasn1/tracer.rb', line 92 def do_parse_explicit_with_tracing(data) RASN1.tracer.tracing_level += 1 do_parse_explicit_without_tracing(data) RASN1.tracer.tracing_level -= 1 end |
#do_parse_with_tracing(der, ber:) ⇒ Object
Parse der
with tracing abillity
86 87 88 89 90 |
# File 'lib/rasn1/tracer.rb', line 86 def do_parse_with_tracing(der, ber:) ret = do_parse_without_tracing(der, ber: ber) RASN1.tracer.trace(self.trace) ret end |
#explicit? ⇒ ::Boolean?
Say if a tagged type is explicit
180 181 182 |
# File 'lib/rasn1/types/base.rb', line 180 def explicit? defined?(@tag) ? @tag == :explicit : nil # rubocop:disable Style/ReturnNilInPredicateMethodDefinition end |
#id ⇒ Integer
Get identifier value
216 217 218 |
# File 'lib/rasn1/types/base.rb', line 216 def id id_value end |
#implicit? ⇒ ::Boolean?
Say if a tagged type is implicit
187 188 189 |
# File 'lib/rasn1/types/base.rb', line 187 def implicit? defined?(@tag) ? @tag == :implicit : nil # rubocop:disable Style/ReturnNilInPredicateMethodDefinition end |
#initialize_copy ⇒ Object
Deep copy @value and @default.
138 139 140 141 142 143 |
# File 'lib/rasn1/types/base.rb', line 138 def initialize_copy(*) super @value = @value.dup @no_value = @no_value.dup @default = @default.dup end |
#inspect(level = 0) ⇒ String
248 249 250 251 252 253 254 |
# File 'lib/rasn1/types/base.rb', line 248 def inspect(level=0) str = common_inspect(level) str << ' ' << inspect_value str << ' OPTIONAL' if optional? str << " DEFAULT #{@default}" unless @default.nil? str end |
#optional? ⇒ ::Boolean
Say if this type is optional
167 168 169 |
# File 'lib/rasn1/types/base.rb', line 167 def optional? @optional end |
#parse!(der, ber: false) ⇒ Integer
This method SHOULD be partly implemented by subclasses to parse data. Subclasses SHOULD respond to #der_to_value
.
Parse a DER string. This method updates object.
227 228 229 230 231 232 233 234 235 236 237 238 |
# File 'lib/rasn1/types/base.rb', line 227 def parse!(der, ber: false) total_length, data = do_parse(der, ber: ber) return 0 if total_length.zero? if explicit? do_parse_explicit(data) else der_to_value(data, ber: ber) end total_length end |
#primitive? ⇒ ::Boolean
Returns true
if this is a primitive type.
199 200 201 |
# File 'lib/rasn1/types/base.rb', line 199 def primitive? (self.class < Primitive) && !@constructed end |
#specific_initializer ⇒ Object
To help subclass initialize itself. Default implementation do nothing.
135 |
# File 'lib/rasn1/types/base.rb', line 135 def specific_initializer; end |
#tagged? ⇒ ::Boolean
Say if this type is tagged or not
173 174 175 |
# File 'lib/rasn1/types/base.rb', line 173 def tagged? !@tag.nil? end |
#to_der ⇒ String
This method SHOULD be partly implemented by subclasses, which SHOULD respond to #value_to_der
.
Returns DER-formated string.
194 195 196 |
# File 'lib/rasn1/types/base.rb', line 194 def to_der build end |
#trace ⇒ String
292 293 294 295 296 297 298 299 300 301 |
# File 'lib/rasn1/types/base.rb', line 292 def trace return trace_real if value? msg = msg_type if default.nil? # rubocop:disable Style/ConditionalAssignment msg << ' NONE' else msg << " DEFAULT VALUE #{default}" end end |
#type ⇒ String
Get ASN.1 type
210 211 212 |
# File 'lib/rasn1/types/base.rb', line 210 def type self.class.type end |
#value ⇒ Object
Get value or default value
146 147 148 149 150 151 152 |
# File 'lib/rasn1/types/base.rb', line 146 def value if value? @value else @default end end |
#value=(val) ⇒ Object
Set value. If val
is nil
, unset value
156 157 158 |
# File 'lib/rasn1/types/base.rb', line 156 def value=(val) set_value(val) end |
#value? ⇒ Boolean
Say if a value is set
279 280 281 |
# File 'lib/rasn1/types/base.rb', line 279 def value? !@no_value end |
#value_size ⇒ Integer
Give size in octets of encoded value
242 243 244 |
# File 'lib/rasn1/types/base.rb', line 242 def value_size value_to_der.size end |
#void_value ⇒ Object
Define ‘void’ value (i.e. ‘value’ when no value was set)
161 162 163 |
# File 'lib/rasn1/types/base.rb', line 161 def void_value '' end |