2 # Copyright 2015-2021 The OpenSSL Project Authors. All Rights Reserved.
4 # Licensed under the Apache License 2.0 (the "License"). You may not use
5 # this file except in compliance with the License. You can obtain a copy
6 # in the file LICENSE in the source distribution or at
7 # https://www.openssl.org/source/license.html
13 use OpenSSL::Test::Utils;
14 use OpenSSL::Test qw/:DEFAULT srctop_file/;
20 require_ok(srctop_file('test', 'recipes', 'tconversion.pl'));
22 my @certs = qw(test certs);
24 # What type of key to generate?
26 if (disabled("rsa")) {
27 @req_new = ("-newkey", "dsa:".srctop_file("apps", "dsa512.pem"));
30 note("There should be a 2 sequences of .'s and some +'s.");
31 note("There should not be more that at most 80 per line");
34 # Check for duplicate -addext parameters, and one "working" case.
35 my @addext_args = ( "openssl", "req", "-new", "-out", "testreq.pem",
36 "-key", srctop_file("test", "certs", "ee-key.pem"),
37 "-config", srctop_file("test", "test.cnf"), @req_new );
38 my $val = "subjectAltName=DNS:example.com";
39 my $val2 = " " . $val;
42 ok( run(app([@addext_args, "-addext", $val])));
43 ok(!run(app([@addext_args, "-addext", $val, "-addext", $val])));
44 ok(!run(app([@addext_args, "-addext", $val, "-addext", $val2])));
45 ok(!run(app([@addext_args, "-addext", $val, "-addext", $val3])));
46 ok(!run(app([@addext_args, "-addext", $val2, "-addext", $val3])));
48 subtest "generating alt certificate requests with RSA" => sub {
52 skip "RSA is not supported by this OpenSSL build", 2
55 ok(run(app(["openssl", "req",
56 "-config", srctop_file("test", "test.cnf"),
58 "-new", "-out", "testreq-rsa.pem", "-utf8",
59 "-key", srctop_file("test", "testrsa.pem")])),
60 "Generating request");
62 ok(run(app(["openssl", "req",
63 "-config", srctop_file("test", "test.cnf"),
64 "-verify", "-in", "testreq-rsa.pem", "-noout"])),
65 "Verifying signature on request");
67 ok(run(app(["openssl", "req",
68 "-config", srctop_file("test", "test.cnf"),
70 "-verify", "-in", "testreq-rsa.pem", "-noout"])),
71 "Verifying signature on request");
76 subtest "generating certificate requests with RSA" => sub {
80 skip "RSA is not supported by this OpenSSL build", 2
83 ok(!run(app(["openssl", "req",
84 "-config", srctop_file("test", "test.cnf"),
85 "-new", "-out", "testreq-rsa.pem", "-utf8",
86 "-key", srctop_file("test", "testrsa.pem"),
88 "Checking that mismatching keyform fails");
90 ok(run(app(["openssl", "req",
91 "-config", srctop_file("test", "test.cnf"),
92 "-new", "-out", "testreq-rsa.pem", "-utf8",
93 "-key", srctop_file("test", "testrsa.pem"),
95 "Generating request");
97 ok(run(app(["openssl", "req",
98 "-config", srctop_file("test", "test.cnf"),
99 "-verify", "-in", "testreq-rsa.pem", "-noout"])),
100 "Verifying signature on request");
102 ok(run(app(["openssl", "req",
103 "-config", srctop_file("test", "test.cnf"),
104 "-new", "-out", "testreq_withattrs_pem.pem", "-utf8",
105 "-key", srctop_file("test", "testrsa_withattrs.pem")])),
106 "Generating request from a key with extra attributes - PEM");
108 ok(run(app(["openssl", "req",
109 "-config", srctop_file("test", "test.cnf"),
110 "-verify", "-in", "testreq_withattrs_pem.pem", "-noout"])),
111 "Verifying signature on request from a key with extra attributes - PEM");
113 ok(run(app(["openssl", "req",
114 "-config", srctop_file("test", "test.cnf"),
115 "-new", "-out", "testreq_withattrs_der.pem", "-utf8",
116 "-key", srctop_file("test", "testrsa_withattrs.der"),
117 "-keyform", "DER"])),
118 "Generating request from a key with extra attributes - PEM");
120 ok(run(app(["openssl", "req",
121 "-config", srctop_file("test", "test.cnf"),
122 "-verify", "-in", "testreq_withattrs_der.pem", "-noout"])),
123 "Verifying signature on request from a key with extra attributes - PEM");
127 subtest "generating certificate requests with RSA-PSS" => sub {
131 skip "RSA is not supported by this OpenSSL build", 2
134 ok(run(app(["openssl", "req",
135 "-config", srctop_file("test", "test.cnf"),
136 "-new", "-out", "testreq-rsapss.pem", "-utf8",
137 "-key", srctop_file("test", "testrsapss.pem")])),
138 "Generating request");
139 ok(run(app(["openssl", "req",
140 "-config", srctop_file("test", "test.cnf"),
141 "-verify", "-in", "testreq-rsapss.pem", "-noout"])),
142 "Verifying signature on request");
144 ok(run(app(["openssl", "req",
145 "-config", srctop_file("test", "test.cnf"),
146 "-new", "-out", "testreq-rsapss2.pem", "-utf8",
147 "-sigopt", "rsa_padding_mode:pss",
148 "-sigopt", "rsa_pss_saltlen:-1",
149 "-key", srctop_file("test", "testrsapss.pem")])),
150 "Generating request");
151 ok(run(app(["openssl", "req",
152 "-config", srctop_file("test", "test.cnf"),
153 "-verify", "-in", "testreq-rsapss2.pem", "-noout"])),
154 "Verifying signature on request");
156 ok(run(app(["openssl", "req",
157 "-config", srctop_file("test", "test.cnf"),
158 "-new", "-out", "testreq-rsapssmand.pem", "-utf8",
159 "-sigopt", "rsa_padding_mode:pss",
160 "-key", srctop_file("test", "testrsapssmandatory.pem")])),
161 "Generating request");
162 ok(run(app(["openssl", "req",
163 "-config", srctop_file("test", "test.cnf"),
164 "-verify", "-in", "testreq-rsapssmand.pem", "-noout"])),
165 "Verifying signature on request");
167 ok(run(app(["openssl", "req",
168 "-config", srctop_file("test", "test.cnf"),
169 "-new", "-out", "testreq-rsapssmand2.pem", "-utf8",
170 "-sigopt", "rsa_pss_saltlen:100",
171 "-key", srctop_file("test", "testrsapssmandatory.pem")])),
172 "Generating request");
173 ok(run(app(["openssl", "req",
174 "-config", srctop_file("test", "test.cnf"),
175 "-verify", "-in", "testreq-rsapssmand2.pem", "-noout"])),
176 "Verifying signature on request");
178 ok(!run(app(["openssl", "req",
179 "-config", srctop_file("test", "test.cnf"),
180 "-new", "-out", "testreq-rsapss3.pem", "-utf8",
181 "-sigopt", "rsa_padding_mode:pkcs1",
182 "-key", srctop_file("test", "testrsapss.pem")])),
183 "Generating request with expected failure");
185 ok(!run(app(["openssl", "req",
186 "-config", srctop_file("test", "test.cnf"),
187 "-new", "-out", "testreq-rsapss3.pem", "-utf8",
188 "-sigopt", "rsa_pss_saltlen:-4",
189 "-key", srctop_file("test", "testrsapss.pem")])),
190 "Generating request with expected failure");
192 ok(!run(app(["openssl", "req",
193 "-config", srctop_file("test", "test.cnf"),
194 "-new", "-out", "testreq-rsapssmand3.pem", "-utf8",
195 "-sigopt", "rsa_pss_saltlen:10",
196 "-key", srctop_file("test", "testrsapssmandatory.pem")])),
197 "Generating request with expected failure");
199 ok(!run(app(["openssl", "req",
200 "-config", srctop_file("test", "test.cnf"),
201 "-new", "-out", "testreq-rsapssmand3.pem", "-utf8",
203 "-key", srctop_file("test", "testrsapssmandatory.pem")])),
204 "Generating request with expected failure");
208 subtest "generating certificate requests with DSA" => sub {
212 skip "DSA is not supported by this OpenSSL build", 2
215 ok(run(app(["openssl", "req",
216 "-config", srctop_file("test", "test.cnf"),
217 "-new", "-out", "testreq-dsa.pem", "-utf8",
218 "-key", srctop_file("test", "testdsa.pem")])),
219 "Generating request");
221 ok(run(app(["openssl", "req",
222 "-config", srctop_file("test", "test.cnf"),
223 "-verify", "-in", "testreq-dsa.pem", "-noout"])),
224 "Verifying signature on request");
228 subtest "generating certificate requests with ECDSA" => sub {
232 skip "ECDSA is not supported by this OpenSSL build", 2
235 ok(run(app(["openssl", "req",
236 "-config", srctop_file("test", "test.cnf"),
237 "-new", "-out", "testreq-ec.pem", "-utf8",
238 "-key", srctop_file("test", "testec-p256.pem")])),
239 "Generating request");
241 ok(run(app(["openssl", "req",
242 "-config", srctop_file("test", "test.cnf"),
243 "-verify", "-in", "testreq-ec.pem", "-noout"])),
244 "Verifying signature on request");
248 subtest "generating certificate requests with Ed25519" => sub {
252 skip "Ed25519 is not supported by this OpenSSL build", 2
255 ok(run(app(["openssl", "req",
256 "-config", srctop_file("test", "test.cnf"),
257 "-new", "-out", "testreq-ed25519.pem", "-utf8",
258 "-key", srctop_file("test", "tested25519.pem")])),
259 "Generating request");
261 ok(run(app(["openssl", "req",
262 "-config", srctop_file("test", "test.cnf"),
263 "-verify", "-in", "testreq-ed25519.pem", "-noout"])),
264 "Verifying signature on request");
268 subtest "generating certificate requests with Ed448" => sub {
272 skip "Ed448 is not supported by this OpenSSL build", 2
275 ok(run(app(["openssl", "req",
276 "-config", srctop_file("test", "test.cnf"),
277 "-new", "-out", "testreq-ed448.pem", "-utf8",
278 "-key", srctop_file("test", "tested448.pem")])),
279 "Generating request");
281 ok(run(app(["openssl", "req",
282 "-config", srctop_file("test", "test.cnf"),
283 "-verify", "-in", "testreq-ed448.pem", "-noout"])),
284 "Verifying signature on request");
288 subtest "generating certificate requests" => sub {
291 ok(run(app(["openssl", "req", "-config", srctop_file("test", "test.cnf"),
292 "-key", srctop_file("test", "certs", "ee-key.pem"),
293 @req_new, "-out", "testreq.pem"])),
294 "Generating request");
296 ok(run(app(["openssl", "req", "-config", srctop_file("test", "test.cnf"),
297 "-verify", "-in", "testreq.pem", "-noout"])),
298 "Verifying signature on request");
301 subtest "generating SM2 certificate requests" => sub {
305 skip "SM2 is not supported by this OpenSSL build", 4
307 ok(run(app(["openssl", "req",
308 "-config", srctop_file("test", "test.cnf"),
309 "-new", "-key", srctop_file(@certs, "sm2.key"),
310 "-sigopt", "distid:1234567812345678",
311 "-out", "testreq-sm2.pem", "-sm3"])),
312 "Generating SM2 certificate request");
314 ok(run(app(["openssl", "req",
315 "-config", srctop_file("test", "test.cnf"),
316 "-verify", "-in", "testreq-sm2.pem", "-noout",
317 "-vfyopt", "distid:1234567812345678", "-sm3"])),
318 "Verifying signature on SM2 certificate request");
320 ok(run(app(["openssl", "req",
321 "-config", srctop_file("test", "test.cnf"),
322 "-new", "-key", srctop_file(@certs, "sm2.key"),
323 "-sigopt", "hexdistid:DEADBEEF",
324 "-out", "testreq-sm2.pem", "-sm3"])),
325 "Generating SM2 certificate request with hex id");
327 ok(run(app(["openssl", "req",
328 "-config", srctop_file("test", "test.cnf"),
329 "-verify", "-in", "testreq-sm2.pem", "-noout",
330 "-vfyopt", "hexdistid:DEADBEEF", "-sm3"])),
331 "Verifying signature on SM2 certificate request");
335 my @openssl_args = ("req", "-config", srctop_file("apps", "openssl.cnf"));
337 run_conversion('req conversions',
339 run_conversion('req conversions -- testreq2',
340 srctop_file("test", "testreq2.pem"));
346 subtest $title => sub {
347 run(app(["openssl", @openssl_args,
348 "-in", $reqfile, "-inform", "p",
350 stderr => "req-check.err", stdout => undef));
351 open DATA, "req-check.err";
353 plan skip_all => "skipping req conversion test for $reqfile"
354 if grep /Unknown Public Key/, map { s/\R//; } <DATA>;
356 tconversion( -type => 'req', -in => $reqfile,
357 -args => [ @openssl_args ] );
360 unlink "req-check.err";
366 # Test both generation and verification of certs w.r.t. RFC 5280 requirements
368 my $ca_cert; # will be set below
371 my $ss = $cert =~ m/self-signed/;
372 my $is_ca = $cert =~ m/CA/;
373 my $cn = $is_ca ? "CA" : "EE";
374 my $ca_key = srctop_file(@certs, "ca-key.pem");
375 my $key = $is_ca ? $ca_key : srctop_file(@certs, "ee-key.pem");
376 my @cmd = ("openssl", "req", "-config", "", "-x509",
377 "-key", $key, "-subj", "/CN=$cn", @_, "-out", $cert);
378 push(@cmd, ("-CA", $ca_cert, "-CAkey", $ca_key)) unless $ss;
379 ok(run(app([@cmd])), "generate $cert");
383 my $expect = shift @_;
384 cert_contains($cert, "Subject Key Identifier", $expect);
388 my $expect = shift @_;
389 cert_contains($cert, "Authority Key Identifier", $expect);
393 my $expect = shift @_;
394 cert_contains($cert, "Key Usage", $expect);
398 my $expect = shift @_;
399 my $trusted = shift @_;
400 $trusted = $cert unless $trusted;
401 ok(run(app(["openssl", "verify", "-x509_strict", "-trusted", $trusted,
402 "-partial_chain", $cert])) == $expect,
403 "strict verify allow $cert");
406 my @v3_ca = ("-addext", "basicConstraints = critical,CA:true",
407 "-addext", "keyUsage = keyCertSign");
408 my $SKID_AKID = "subjectKeyIdentifier,authorityKeyIdentifier";
409 my $cert = "self-signed_v1_CA_no_KIDs.pem";
410 generate_cert($cert);
411 cert_ext_has_n_different_lines($cert, 0, $SKID_AKID); # no SKID and no AKID
412 #TODO strict_verify($cert, 1); # self-signed v1 root cert should be accepted as CA
414 $ca_cert = "self-signed_v3_CA_default_SKID.pem";
415 generate_cert($ca_cert, @v3_ca);
416 has_SKID($ca_cert, 1);
417 has_AKID($ca_cert, 0);
418 strict_verify($ca_cert, 1);
420 $cert = "self-signed_v3_CA_no_SKID.pem";
421 generate_cert($cert, @v3_ca, "-addext", "subjectKeyIdentifier = none");
422 cert_ext_has_n_different_lines($cert, 0, $SKID_AKID); # no SKID and no AKID
423 #TODO strict_verify($cert, 0);
425 $cert = "self-signed_v3_CA_both_KIDs.pem";
426 generate_cert($cert, @v3_ca, "-addext", "subjectKeyIdentifier = hash",
427 "-addext", "authorityKeyIdentifier = keyid");
428 cert_ext_has_n_different_lines($cert, 3, $SKID_AKID); # SKID == AKID
429 strict_verify($cert, 1);
431 $cert = "self-signed_v3_EE_wrong_keyUsage.pem";
432 generate_cert($cert, "-addext", "keyUsage = keyCertSign");
433 #TODO strict_verify($cert, 1); # should be accepted because RFC 5280 does not apply
435 $cert = "v3_EE_default_KIDs.pem";
436 generate_cert($cert, "-addext", "keyUsage = dataEncipherment");
437 cert_ext_has_n_different_lines($cert, 4, $SKID_AKID); # SKID != AKID
438 strict_verify($cert, 1, $ca_cert);
440 $cert = "v3_EE_no_AKID.pem";
441 generate_cert($cert, "-addext", "authorityKeyIdentifier = none");
444 strict_verify($cert, 0, $ca_cert);
446 $cert = "self-issued_v3_EE_default_KIDs.pem";
447 generate_cert($cert, "-addext", "keyUsage = dataEncipherment",
448 "-in", srctop_file(@certs, "x509-check.csr"));
449 cert_ext_has_n_different_lines($cert, 4, $SKID_AKID); # SKID != AKID
450 strict_verify($cert, 1);
452 my $cert = "self-signed_CA_no_keyUsage.pem";
453 generate_cert($cert, "-in", srctop_file(@certs, "ext-check.csr"));
454 has_keyUsage($cert, 0);
455 my $cert = "self-signed_CA_with_keyUsages.pem";
456 generate_cert($cert, "-in", srctop_file(@certs, "ext-check.csr"),
457 "-copy_extensions", "copy");
458 has_keyUsage($cert, 1);