1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
|
use openssl::hash::MessageDigest;
use openssl::nid::Nid;
use openssl::pkey::{Id, PKey, Private};
use openssl::stack::Stack;
use openssl::x509::extension::SubjectAlternativeName;
use openssl::x509::{X509Name, X509Req};
use std::fs::{read, write};
use std::net::IpAddr;
use std::str::FromStr;
use crate::path;
#[allow(clippy::too_many_arguments)]
pub fn generate_req(
common_name: &Option<String>,
country: &Option<String>,
state: &Option<String>,
locality: &Option<String>,
organization: &Option<String>,
organizational_unit: &Option<String>,
subject_alternative_names: &Option<String>,
pkey: &PKey<Private>,
) -> X509Req {
let mut x509req_builder = X509Req::builder().unwrap();
x509req_builder.set_pubkey(pkey).unwrap();
x509req_builder.set_version(0).unwrap();
let mut x509_name_builder = X509Name::builder().unwrap();
if let Some(cn) = common_name {
x509_name_builder
.append_entry_by_nid(Nid::COMMONNAME, cn)
.unwrap();
}
if let Some(c) = country {
x509_name_builder
.append_entry_by_nid(Nid::COUNTRYNAME, c)
.unwrap();
}
if let Some(s) = state {
x509_name_builder
.append_entry_by_nid(Nid::STATEORPROVINCENAME, s)
.unwrap();
}
if let Some(l) = locality {
x509_name_builder
.append_entry_by_nid(Nid::LOCALITYNAME, l)
.unwrap();
}
if let Some(o) = organization {
x509_name_builder
.append_entry_by_nid(Nid::ORGANIZATIONNAME, o)
.unwrap();
}
if let Some(ou) = organizational_unit {
x509_name_builder
.append_entry_by_nid(Nid::ORGANIZATIONALUNITNAME, ou)
.unwrap();
}
let x509_name = x509_name_builder.build();
x509req_builder.set_subject_name(&x509_name).unwrap();
let mut subject_alt_name = SubjectAlternativeName::new();
if let Some(san) = subject_alternative_names {
let alt_names = san.split(',');
for name in alt_names {
if IpAddr::from_str(name).is_ok() {
subject_alt_name.ip(name);
} else {
subject_alt_name.dns(name);
}
}
}
if let Some(cn) = common_name {
if IpAddr::from_str(cn).is_ok() {
subject_alt_name.ip(cn);
} else {
subject_alt_name.dns(cn);
}
}
let subject_alt_name = subject_alt_name
.build(&x509req_builder.x509v3_context(None))
.unwrap();
let mut stack = Stack::new().unwrap();
stack.push(subject_alt_name).unwrap();
x509req_builder.add_extensions(&stack).unwrap();
let digest_algorithm = match pkey.id() {
Id::RSA => MessageDigest::sha256(),
Id::EC => MessageDigest::sha384(),
_ => MessageDigest::sha256(),
};
x509req_builder.sign(pkey, digest_algorithm).unwrap();
x509req_builder.build()
}
pub fn save_req(path: &str, req: &X509Req) {
println!("{}", path);
path::ensure_dir(path);
write(path, req.to_pem().unwrap()).unwrap();
}
pub fn read_req(path: &str) -> X509Req {
X509Req::from_pem(&read(path).unwrap()).unwrap()
}
|