Yiming Jing | 6ea24ae | 2021-07-22 13:50:52 -0700 | [diff] [blame] | 1 | <!-- cargo-sync-readme start --> |
| 2 | |
| 3 | [](./LICENSE-MIT) |
| 4 | [](./LICENSE-APACHE) |
| 5 | [](https://docs.rs/x509-parser) |
| 6 | [](https://crates.io/crates/x509-parser) |
| 7 | [](https://crates.io/crates/x509-parser) |
| 8 | [](https://github.com/rusticata/x509-parser/actions) |
Joel Galenson | 5e6c68c | 2021-09-23 14:06:09 -0700 | [diff] [blame] | 9 | [](#rust-version-requirements) |
Yiming Jing | 6ea24ae | 2021-07-22 13:50:52 -0700 | [diff] [blame] | 10 | |
| 11 | # X.509 Parser |
| 12 | |
| 13 | A X.509 v3 ([RFC5280]) parser, implemented with the [nom](https://github.com/Geal/nom) |
| 14 | parser combinator framework. |
| 15 | |
| 16 | It is written in pure Rust, fast, and makes extensive use of zero-copy. A lot of care is taken |
| 17 | to ensure security and safety of this crate, including design (recursion limit, defensive |
| 18 | programming), tests, and fuzzing. It also aims to be panic-free. |
| 19 | |
| 20 | The code is available on [Github](https://github.com/rusticata/x509-parser) |
| 21 | and is part of the [Rusticata](https://github.com/rusticata) project. |
| 22 | |
| 23 | Certificates are usually encoded in two main formats: PEM (usually the most common format) or |
| 24 | DER. A PEM-encoded certificate is a container, storing a DER object. See the |
| 25 | [`pem`](https://docs.rs/x509-parser/latest/x509_parser/pem/index.html) module for more documentation. |
| 26 | |
| 27 | To decode a DER-encoded certificate, the main parsing method is |
Joel Galenson | 1d90639 | 2021-08-11 15:58:47 +0000 | [diff] [blame] | 28 | [`X509Certificate::from_der`] ( |
| 29 | part of the [`FromDer`](https://docs.rs/x509-parser/latest/x509_parser/traits/trait.FromDer.html) trait |
| 30 | ), which builds a |
| 31 | [`X509Certificate`](https://docs.rs/x509-parser/latest/x509_parser/certificate/struct.X509Certificate.html) object. |
| 32 | |
| 33 | An alternative method is to use [`X509CertificateParser`](https://docs.rs/x509-parser/latest/x509_parser/certificate/struct.X509CertificateParser.html), |
| 34 | which allows specifying parsing options (for example, not automatically parsing option contents). |
Yiming Jing | 6ea24ae | 2021-07-22 13:50:52 -0700 | [diff] [blame] | 35 | |
| 36 | The returned objects for parsers follow the definitions of the RFC. This means that accessing |
| 37 | fields is done by accessing struct members recursively. Some helper functions are provided, for |
Joel Galenson | 1d90639 | 2021-08-11 15:58:47 +0000 | [diff] [blame] | 38 | example [`X509Certificate::issuer()`](https://docs.rs/x509-parser/latest/x509_parser/certificate/struct.X509Certificate.html#method.issuer) returns the |
Yiming Jing | 6ea24ae | 2021-07-22 13:50:52 -0700 | [diff] [blame] | 39 | same as accessing `<object>.tbs_certificate.issuer`. |
| 40 | |
| 41 | For PEM-encoded certificates, use the [`pem`](https://docs.rs/x509-parser/latest/x509_parser/pem/index.html) module. |
| 42 | |
| 43 | # Examples |
| 44 | |
| 45 | Parsing a certificate in DER format: |
| 46 | |
| 47 | ```rust |
| 48 | use x509_parser::prelude::*; |
| 49 | |
| 50 | static IGCA_DER: &[u8] = include_bytes!("../assets/IGC_A.der"); |
| 51 | |
Joel Galenson | 1d90639 | 2021-08-11 15:58:47 +0000 | [diff] [blame] | 52 | let res = X509Certificate::from_der(IGCA_DER); |
Yiming Jing | 6ea24ae | 2021-07-22 13:50:52 -0700 | [diff] [blame] | 53 | match res { |
| 54 | Ok((rem, cert)) => { |
| 55 | assert!(rem.is_empty()); |
| 56 | // |
Joel Galenson | 1d90639 | 2021-08-11 15:58:47 +0000 | [diff] [blame] | 57 | assert_eq!(cert.version(), X509Version::V3); |
Yiming Jing | 6ea24ae | 2021-07-22 13:50:52 -0700 | [diff] [blame] | 58 | }, |
| 59 | _ => panic!("x509 parsing failed: {:?}", res), |
| 60 | } |
| 61 | ``` |
| 62 | |
| 63 | To parse a CRL and print information about revoked certificates: |
| 64 | |
| 65 | ```rust |
| 66 | # |
| 67 | # |
Joel Galenson | 1d90639 | 2021-08-11 15:58:47 +0000 | [diff] [blame] | 68 | let res = CertificateRevocationList::from_der(DER); |
Yiming Jing | 6ea24ae | 2021-07-22 13:50:52 -0700 | [diff] [blame] | 69 | match res { |
| 70 | Ok((_rem, crl)) => { |
| 71 | for revoked in crl.iter_revoked_certificates() { |
| 72 | println!("Revoked certificate serial: {}", revoked.raw_serial_as_string()); |
| 73 | println!(" Reason: {}", revoked.reason_code().unwrap_or_default().1); |
| 74 | } |
| 75 | }, |
| 76 | _ => panic!("CRL parsing failed: {:?}", res), |
| 77 | } |
| 78 | ``` |
| 79 | |
| 80 | See also `examples/print-cert.rs`. |
| 81 | |
| 82 | # Features |
| 83 | |
| 84 | - The `verify` feature adds support for (cryptographic) signature verification, based on `ring`. |
| 85 | It adds the |
Joel Galenson | 1d90639 | 2021-08-11 15:58:47 +0000 | [diff] [blame] | 86 | [`X509Certificate::verify_signature()`](https://docs.rs/x509-parser/latest/x509_parser/certificate/struct.X509Certificate.html#method.verify_signature) |
Yiming Jing | 6ea24ae | 2021-07-22 13:50:52 -0700 | [diff] [blame] | 87 | to `X509Certificate`. |
| 88 | |
| 89 | ```rust |
| 90 | /// Cryptographic signature verification: returns true if certificate was signed by issuer |
| 91 | #[cfg(feature = "verify")] |
| 92 | pub fn check_signature(cert: &X509Certificate<'_>, issuer: &X509Certificate<'_>) -> bool { |
Joel Galenson | 1d90639 | 2021-08-11 15:58:47 +0000 | [diff] [blame] | 93 | let issuer_public_key = issuer.public_key(); |
Yiming Jing | 6ea24ae | 2021-07-22 13:50:52 -0700 | [diff] [blame] | 94 | cert |
| 95 | .verify_signature(Some(issuer_public_key)) |
| 96 | .is_ok() |
| 97 | } |
| 98 | ``` |
| 99 | |
Joel Galenson | 1d90639 | 2021-08-11 15:58:47 +0000 | [diff] [blame] | 100 | - The `validate` features add methods to run more validation functions on the certificate structure |
| 101 | and values using the [`Validate`](https://docs.rs/x509-parser/latest/x509_parser/validate/trait.Validate.html) trait. |
| 102 | It does not validate any cryptographic parameter (see `verify` above). |
| 103 | |
Yiming Jing | 6ea24ae | 2021-07-22 13:50:52 -0700 | [diff] [blame] | 104 | ## Rust version requirements |
| 105 | |
Joel Galenson | 5e6c68c | 2021-09-23 14:06:09 -0700 | [diff] [blame] | 106 | `x509-parser` requires **Rustc version 1.46 or greater**, based on nom 7 |
Yiming Jing | 6ea24ae | 2021-07-22 13:50:52 -0700 | [diff] [blame] | 107 | dependencies and for proc-macro attributes support. |
| 108 | |
| 109 | [RFC5280]: https://tools.ietf.org/html/rfc5280 |
| 110 | <!-- cargo-sync-readme end --> |
| 111 | |
| 112 | ## Changes |
| 113 | |
| 114 | See [CHANGELOG.md](CHANGELOG.md) |
| 115 | |
| 116 | # License |
| 117 | |
| 118 | Licensed under either of |
| 119 | |
| 120 | * Apache License, Version 2.0 |
| 121 | ([LICENSE-APACHE](LICENSE-APACHE) or http://www.apache.org/licenses/LICENSE-2.0) |
| 122 | * MIT license |
| 123 | ([LICENSE-MIT](LICENSE-MIT) or http://opensource.org/licenses/MIT) |
| 124 | |
| 125 | at your option. |
| 126 | |
| 127 | ## Contribution |
| 128 | |
| 129 | Unless you explicitly state otherwise, any contribution intentionally submitted |
| 130 | for inclusion in the work by you, as defined in the Apache-2.0 license, shall be |
| 131 | dual licensed as above, without any additional terms or conditions. |