-
Notifications
You must be signed in to change notification settings - Fork 137
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
(edited by dishmaker) support private tags and tag numbers >30 that a…
…re stored in long form draft: CustomClass generic type feat: add decode_implicit and decode_explicit back fix: cargo test -p der now pass refactor: raname to Explicit struct refactor: TAG, T order docs: CustomClass feat: add ContextSpecificRef back (ContextSpecificExplicitRef) update der-derive to current der implementation feat: add tests for context-specific fix: dependency crates fixed for x509-cert cargo fmt fix: crates depending on context-specific and long TagNumber fix x509-cert
- Loading branch information
Showing
34 changed files
with
1,873 additions
and
829 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,261 @@ | ||
use crate::{ | ||
Class, Decode, DecodeValue, Encode, EncodeValue, Error, Header, Length, Reader, Tag, TagNumber, | ||
Tagged, Writer, | ||
}; | ||
|
||
use super::AnyRef; | ||
|
||
/// `APPLICATION`, `CONTEXT-SPECIFIC` or `PRIVATE` tagged value. | ||
/// | ||
/// `EXPLICIT` encoding - always constructed. | ||
pub struct AnyCustomClassExplicit<T> { | ||
/// Value of the field. Should implement [`Decode`] | ||
pub value: T, | ||
|
||
/// Class of the field. | ||
/// | ||
/// Supported classes: [`Class::Application`], [`Class::ContextSpecific`], [`Class::Private`] | ||
pub class: Class, | ||
|
||
/// Tag number without the leading class bits `0b11000000` | ||
/// and without constructed `0b00100000` flag. | ||
pub tag_number: TagNumber, | ||
} | ||
|
||
/// `APPLICATION`, `CONTEXT-SPECIFIC` or `PRIVATE` tagged value. | ||
/// | ||
/// `IMPLICIT` encoding - constructed bit should match inner value's tag. | ||
pub struct AnyCustomClassImplicit<T> { | ||
/// Value of the field. Should implement [`DecodeValue`] | ||
pub value: T, | ||
|
||
/// Class of the field. | ||
/// | ||
/// Supported classes: [`Class::Application`], [`Class::ContextSpecific`], [`Class::Private`] | ||
pub class: Class, | ||
|
||
/// Tag number without the leading class bits `0b11000000` | ||
/// and without constructed `0b00100000` flag. | ||
pub tag_number: TagNumber, | ||
|
||
/// Constructed flag. Should match value's tag constructed flag. | ||
pub constructed: bool, | ||
} | ||
|
||
impl<'a, T> AnyCustomClassExplicit<T> | ||
where | ||
T: Decode<'a>, | ||
{ | ||
/// Decodes `APPLICATION`, `CONTEXT-SPECIFIC` or `PRIVATE` tagged value. | ||
/// | ||
/// Returns Ok only if both [`Class`] and [`TagNumber`] match the decoded tag. | ||
/// | ||
/// Skips `CONTEXT-SPECIFIC` fields, lower than [`TagNumber`]. | ||
pub fn decode_skipping<R: Reader<'a>>( | ||
class: Class, | ||
tag_number: TagNumber, | ||
reader: &mut R, | ||
) -> Result<Option<Self>, T::Error> { | ||
decode_peeking(reader, class, tag_number, |reader| { | ||
Self::decode_checked(class, tag_number, reader) | ||
}) | ||
} | ||
|
||
/// Decodes `APPLICATION`, `CONTEXT-SPECIFIC` or `PRIVATE` tagged value. | ||
/// | ||
/// Returns Ok only if both [`Class`] and [`TagNumber`] match the decoded tag. | ||
pub fn decode_checked<R: Reader<'a>>( | ||
class: Class, | ||
tag_number: TagNumber, | ||
reader: &mut R, | ||
) -> Result<Self, T::Error> { | ||
let any_explicit = Self::decode(reader)?; | ||
|
||
if any_explicit.class == class && any_explicit.tag_number == tag_number { | ||
Ok(any_explicit) | ||
} else { | ||
let expected = expected_tag_constructed(class, tag_number, true); | ||
Err(any_explicit.tag().unexpected_error(Some(expected)).into()) | ||
} | ||
} | ||
} | ||
|
||
impl<'a, T> AnyCustomClassImplicit<T> | ||
where | ||
T: Tagged + DecodeValue<'a> + 'a, | ||
{ | ||
/// Decodes `APPLICATION`, `CONTEXT-SPECIFIC` or `PRIVATE` tagged value. | ||
/// | ||
/// Returns Ok only if both [`Class`] and [`TagNumber`] match the decoded tag. | ||
/// | ||
/// Skips `CONTEXT-SPECIFIC` fields, lower than [`TagNumber`]. | ||
pub fn decode_skipping<R: Reader<'a>>( | ||
class: Class, | ||
tag_number: TagNumber, | ||
reader: &mut R, | ||
) -> Result<Option<Self>, T::Error> { | ||
decode_peeking::<_, _, T::Error, _>(reader, class, tag_number, |reader| { | ||
Self::decode_checked(class, tag_number, reader) | ||
}) | ||
} | ||
|
||
/// Decodes `APPLICATION`, `CONTEXT-SPECIFIC` or `PRIVATE` tagged value. | ||
/// | ||
/// Returns Ok only if both [`Class`] and [`TagNumber`] match the decoded tag. | ||
pub fn decode_checked<R: Reader<'a>>( | ||
class: Class, | ||
tag_number: TagNumber, | ||
reader: &mut R, | ||
) -> Result<Self, T::Error> { | ||
let any_implicit = Self::decode(reader)?; | ||
if any_implicit.class == class && any_implicit.tag_number == tag_number { | ||
Ok(any_implicit) | ||
} else { | ||
let expected = expected_tag_constructed(class, tag_number, true); | ||
Err(any_implicit.tag().unexpected_error(Some(expected)).into()) | ||
} | ||
} | ||
} | ||
|
||
impl<'a, T> Decode<'a> for AnyCustomClassExplicit<T> | ||
where | ||
T: Decode<'a>, | ||
{ | ||
type Error = T::Error; | ||
|
||
fn decode<R: Reader<'a>>(reader: &mut R) -> Result<Self, Self::Error> { | ||
let header = Header::decode(reader)?; | ||
|
||
if !header.tag.is_constructed() { | ||
return Err(header.tag.non_canonical_error().into()); | ||
} | ||
|
||
Ok(Self { | ||
value: reader.read_nested(header.length, |reader| T::decode(reader))?, | ||
class: header.tag.class(), | ||
tag_number: header.tag.number(), | ||
}) | ||
} | ||
} | ||
|
||
impl<'a, T> Decode<'a> for AnyCustomClassImplicit<T> | ||
where | ||
T: Tagged + DecodeValue<'a> + 'a, | ||
{ | ||
type Error = T::Error; | ||
|
||
fn decode<R: Reader<'a>>(reader: &mut R) -> Result<Self, Self::Error> { | ||
let header = Header::decode(reader)?; | ||
|
||
let value = reader.read_nested(header.length, |reader| T::decode_value(reader, header))?; | ||
|
||
if header.tag.is_constructed() != value.tag().is_constructed() { | ||
return Err(header.tag.non_canonical_error().into()); | ||
} | ||
Ok(Self { | ||
value, | ||
class: header.tag.class(), | ||
tag_number: header.tag.number(), | ||
constructed: header.tag.is_constructed(), | ||
}) | ||
} | ||
} | ||
|
||
impl<T> EncodeValue for AnyCustomClassExplicit<T> | ||
where | ||
T: EncodeValue + Tagged, | ||
{ | ||
fn value_len(&self) -> Result<Length, Error> { | ||
self.value.encoded_len() | ||
} | ||
|
||
fn encode_value(&self, writer: &mut impl Writer) -> Result<(), Error> { | ||
self.value.encode(writer) | ||
} | ||
} | ||
|
||
impl<T> EncodeValue for AnyCustomClassImplicit<T> | ||
where | ||
T: EncodeValue + Tagged, | ||
{ | ||
fn value_len(&self) -> Result<Length, Error> { | ||
self.value.value_len() | ||
} | ||
|
||
fn encode_value(&self, writer: &mut impl Writer) -> Result<(), Error> { | ||
self.value.encode_value(writer) | ||
} | ||
} | ||
|
||
impl<T> Tagged for AnyCustomClassExplicit<T> { | ||
fn tag(&self) -> Tag { | ||
expected_tag_constructed(self.class, self.tag_number, true) | ||
} | ||
} | ||
|
||
impl<T> Tagged for AnyCustomClassImplicit<T> { | ||
fn tag(&self) -> Tag { | ||
expected_tag_constructed(self.class, self.tag_number, self.constructed) | ||
} | ||
} | ||
|
||
/// Attempt to decode a custom class-tagged field with the given | ||
/// helper callback. | ||
fn decode_peeking<'a, F, R: Reader<'a>, E, T>( | ||
reader: &mut R, | ||
expected_class: Class, | ||
expected_number: TagNumber, | ||
f: F, | ||
) -> Result<Option<T>, E> | ||
where | ||
F: FnOnce(&mut R) -> Result<T, E>, | ||
E: From<Error>, | ||
{ | ||
while let Some(tag) = Tag::peek_optional(reader)? { | ||
if is_unskippable_tag(tag, expected_class, expected_number) { | ||
break; | ||
} else if tag.number() == expected_number { | ||
return Some(f(reader)).transpose(); | ||
} else { | ||
AnyRef::decode(reader)?; | ||
} | ||
} | ||
|
||
Ok(None) | ||
} | ||
|
||
/// Returns if this tag is of different class than eg. CONTEXT-SPECIFIC | ||
/// or tag number is higher than expected | ||
fn is_unskippable_tag(tag: Tag, expected_class: Class, expected_number: TagNumber) -> bool { | ||
if expected_class != tag.class() { | ||
return true; | ||
} | ||
match expected_class { | ||
Class::Application => tag.number() > expected_number, | ||
Class::ContextSpecific => tag.number() > expected_number, | ||
Class::Private => tag.number() != expected_number, | ||
Class::Universal => tag.number() != expected_number, | ||
} | ||
} | ||
|
||
pub(crate) const fn expected_tag_constructed( | ||
class: Class, | ||
number: TagNumber, | ||
constructed: bool, | ||
) -> Tag { | ||
match class { | ||
Class::Application => Tag::Application { | ||
constructed, | ||
number, | ||
}, | ||
Class::ContextSpecific => Tag::ContextSpecific { | ||
constructed, | ||
number, | ||
}, | ||
Class::Private => Tag::Private { | ||
constructed, | ||
number, | ||
}, | ||
Class::Universal => Tag::Null, | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,21 @@ | ||
//! Application field. | ||
use crate::tag::CLASS_APPLICATION; | ||
|
||
use super::custom_class::{ | ||
CustomClassExplicit, CustomClassExplicitRef, CustomClassImplicit, CustomClassImplicitRef, | ||
}; | ||
|
||
/// Application class, EXPLICIT | ||
pub type ApplicationExplicit<const TAG: u16, T> = CustomClassExplicit<TAG, T, CLASS_APPLICATION>; | ||
|
||
/// Application class, IMPLICIT | ||
pub type ApplicationImplicit<const TAG: u16, T> = CustomClassImplicit<TAG, T, CLASS_APPLICATION>; | ||
|
||
/// Application class, reference, EXPLICIT | ||
pub type ApplicationExplicitRef<'a, const TAG: u16, T> = | ||
CustomClassExplicitRef<'a, TAG, T, CLASS_APPLICATION>; | ||
|
||
/// Application class, reference, IMPLICIT | ||
pub type ApplicationImplicitRef<'a, const TAG: u16, T> = | ||
CustomClassImplicitRef<'a, TAG, T, CLASS_APPLICATION>; |
Oops, something went wrong.