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 "-config", srctop_file("test", "test.cnf"), @req_new );
37 my $val = "subjectAltName=DNS:example.com";
38 my $val2 = " " . $val;
41 ok( run(app([@addext_args, "-addext", $val])));
42 ok(!run(app([@addext_args, "-addext", $val, "-addext", $val])));
43 ok(!run(app([@addext_args, "-addext", $val, "-addext", $val2])));
44 ok(!run(app([@addext_args, "-addext", $val, "-addext", $val3])));
45 ok(!run(app([@addext_args, "-addext", $val2, "-addext", $val3])));
47 subtest "generating alt certificate requests with RSA" => sub {
51 skip "RSA is not supported by this OpenSSL build", 2
54 ok(run(app(["openssl", "req",
55 "-config", srctop_file("test", "test.cnf"),
57 "-new", "-out", "testreq-rsa.pem", "-utf8",
58 "-key", srctop_file("test", "testrsa.pem")])),
59 "Generating request");
61 ok(run(app(["openssl", "req",
62 "-config", srctop_file("test", "test.cnf"),
63 "-verify", "-in", "testreq-rsa.pem", "-noout"])),
64 "Verifying signature on request");
66 ok(run(app(["openssl", "req",
67 "-config", srctop_file("test", "test.cnf"),
69 "-verify", "-in", "testreq-rsa.pem", "-noout"])),
70 "Verifying signature on request");
75 subtest "generating certificate requests with RSA" => sub {
79 skip "RSA is not supported by this OpenSSL build", 2
82 ok(!run(app(["openssl", "req",
83 "-config", srctop_file("test", "test.cnf"),
84 "-new", "-out", "testreq-rsa.pem", "-utf8",
85 "-key", srctop_file("test", "testrsa.pem"),
87 "Checking that mismatching keyform fails");
89 ok(run(app(["openssl", "req",
90 "-config", srctop_file("test", "test.cnf"),
91 "-new", "-out", "testreq-rsa.pem", "-utf8",
92 "-key", srctop_file("test", "testrsa.pem"),
94 "Generating request");
96 ok(run(app(["openssl", "req",
97 "-config", srctop_file("test", "test.cnf"),
98 "-verify", "-in", "testreq-rsa.pem", "-noout"])),
99 "Verifying signature on request");
101 ok(run(app(["openssl", "req",
102 "-config", srctop_file("test", "test.cnf"),
103 "-new", "-out", "testreq_withattrs_pem.pem", "-utf8",
104 "-key", srctop_file("test", "testrsa_withattrs.pem")])),
105 "Generating request from a key with extra attributes - PEM");
107 ok(run(app(["openssl", "req",
108 "-config", srctop_file("test", "test.cnf"),
109 "-verify", "-in", "testreq_withattrs_pem.pem", "-noout"])),
110 "Verifying signature on request from a key with extra attributes - PEM");
112 ok(run(app(["openssl", "req",
113 "-config", srctop_file("test", "test.cnf"),
114 "-new", "-out", "testreq_withattrs_der.pem", "-utf8",
115 "-key", srctop_file("test", "testrsa_withattrs.der"),
116 "-keyform", "DER"])),
117 "Generating request from a key with extra attributes - PEM");
119 ok(run(app(["openssl", "req",
120 "-config", srctop_file("test", "test.cnf"),
121 "-verify", "-in", "testreq_withattrs_der.pem", "-noout"])),
122 "Verifying signature on request from a key with extra attributes - PEM");
126 subtest "generating certificate requests with RSA-PSS" => sub {
130 skip "RSA is not supported by this OpenSSL build", 2
133 ok(run(app(["openssl", "req",
134 "-config", srctop_file("test", "test.cnf"),
135 "-new", "-out", "testreq-rsapss.pem", "-utf8",
136 "-key", srctop_file("test", "testrsapss.pem")])),
137 "Generating request");
138 ok(run(app(["openssl", "req",
139 "-config", srctop_file("test", "test.cnf"),
140 "-verify", "-in", "testreq-rsapss.pem", "-noout"])),
141 "Verifying signature on request");
143 ok(run(app(["openssl", "req",
144 "-config", srctop_file("test", "test.cnf"),
145 "-new", "-out", "testreq-rsapss2.pem", "-utf8",
146 "-sigopt", "rsa_padding_mode:pss",
147 "-sigopt", "rsa_pss_saltlen:-1",
148 "-key", srctop_file("test", "testrsapss.pem")])),
149 "Generating request");
150 ok(run(app(["openssl", "req",
151 "-config", srctop_file("test", "test.cnf"),
152 "-verify", "-in", "testreq-rsapss2.pem", "-noout"])),
153 "Verifying signature on request");
155 ok(run(app(["openssl", "req",
156 "-config", srctop_file("test", "test.cnf"),
157 "-new", "-out", "testreq-rsapssmand.pem", "-utf8",
158 "-sigopt", "rsa_padding_mode:pss",
159 "-key", srctop_file("test", "testrsapssmandatory.pem")])),
160 "Generating request");
161 ok(run(app(["openssl", "req",
162 "-config", srctop_file("test", "test.cnf"),
163 "-verify", "-in", "testreq-rsapssmand.pem", "-noout"])),
164 "Verifying signature on request");
166 ok(run(app(["openssl", "req",
167 "-config", srctop_file("test", "test.cnf"),
168 "-new", "-out", "testreq-rsapssmand2.pem", "-utf8",
169 "-sigopt", "rsa_pss_saltlen:100",
170 "-key", srctop_file("test", "testrsapssmandatory.pem")])),
171 "Generating request");
172 ok(run(app(["openssl", "req",
173 "-config", srctop_file("test", "test.cnf"),
174 "-verify", "-in", "testreq-rsapssmand2.pem", "-noout"])),
175 "Verifying signature on request");
177 ok(!run(app(["openssl", "req",
178 "-config", srctop_file("test", "test.cnf"),
179 "-new", "-out", "testreq-rsapss3.pem", "-utf8",
180 "-sigopt", "rsa_padding_mode:pkcs1",
181 "-key", srctop_file("test", "testrsapss.pem")])),
182 "Generating request with expected failure");
184 ok(!run(app(["openssl", "req",
185 "-config", srctop_file("test", "test.cnf"),
186 "-new", "-out", "testreq-rsapss3.pem", "-utf8",
187 "-sigopt", "rsa_pss_saltlen:-4",
188 "-key", srctop_file("test", "testrsapss.pem")])),
189 "Generating request with expected failure");
191 ok(!run(app(["openssl", "req",
192 "-config", srctop_file("test", "test.cnf"),
193 "-new", "-out", "testreq-rsapssmand3.pem", "-utf8",
194 "-sigopt", "rsa_pss_saltlen:10",
195 "-key", srctop_file("test", "testrsapssmandatory.pem")])),
196 "Generating request with expected failure");
198 ok(!run(app(["openssl", "req",
199 "-config", srctop_file("test", "test.cnf"),
200 "-new", "-out", "testreq-rsapssmand3.pem", "-utf8",
202 "-key", srctop_file("test", "testrsapssmandatory.pem")])),
203 "Generating request with expected failure");
207 subtest "generating certificate requests with DSA" => sub {
211 skip "DSA is not supported by this OpenSSL build", 2
214 ok(run(app(["openssl", "req",
215 "-config", srctop_file("test", "test.cnf"),
216 "-new", "-out", "testreq-dsa.pem", "-utf8",
217 "-key", srctop_file("test", "testdsa.pem")])),
218 "Generating request");
220 ok(run(app(["openssl", "req",
221 "-config", srctop_file("test", "test.cnf"),
222 "-verify", "-in", "testreq-dsa.pem", "-noout"])),
223 "Verifying signature on request");
227 subtest "generating certificate requests with ECDSA" => sub {
231 skip "ECDSA is not supported by this OpenSSL build", 2
234 ok(run(app(["openssl", "req",
235 "-config", srctop_file("test", "test.cnf"),
236 "-new", "-out", "testreq-ec.pem", "-utf8",
237 "-key", srctop_file("test", "testec-p256.pem")])),
238 "Generating request");
240 ok(run(app(["openssl", "req",
241 "-config", srctop_file("test", "test.cnf"),
242 "-verify", "-in", "testreq-ec.pem", "-noout"])),
243 "Verifying signature on request");
247 subtest "generating certificate requests with Ed25519" => sub {
251 skip "Ed25519 is not supported by this OpenSSL build", 2
254 ok(run(app(["openssl", "req",
255 "-config", srctop_file("test", "test.cnf"),
256 "-new", "-out", "testreq-ed25519.pem", "-utf8",
257 "-key", srctop_file("test", "tested25519.pem")])),
258 "Generating request");
260 ok(run(app(["openssl", "req",
261 "-config", srctop_file("test", "test.cnf"),
262 "-verify", "-in", "testreq-ed25519.pem", "-noout"])),
263 "Verifying signature on request");
267 subtest "generating certificate requests with Ed448" => sub {
271 skip "Ed448 is not supported by this OpenSSL build", 2
274 ok(run(app(["openssl", "req",
275 "-config", srctop_file("test", "test.cnf"),
276 "-new", "-out", "testreq-ed448.pem", "-utf8",
277 "-key", srctop_file("test", "tested448.pem")])),
278 "Generating request");
280 ok(run(app(["openssl", "req",
281 "-config", srctop_file("test", "test.cnf"),
282 "-verify", "-in", "testreq-ed448.pem", "-noout"])),
283 "Verifying signature on request");
287 subtest "generating certificate requests" => sub {
290 ok(run(app(["openssl", "req", "-config", srctop_file("test", "test.cnf"),
291 @req_new, "-out", "testreq.pem"])),
292 "Generating request");
294 ok(run(app(["openssl", "req", "-config", srctop_file("test", "test.cnf"),
295 "-verify", "-in", "testreq.pem", "-noout"])),
296 "Verifying signature on request");
299 subtest "generating SM2 certificate requests" => sub {
303 skip "SM2 is not supported by this OpenSSL build", 4
305 ok(run(app(["openssl", "req",
306 "-config", srctop_file("test", "test.cnf"),
307 "-new", "-key", srctop_file(@certs, "sm2.key"),
308 "-sigopt", "distid:1234567812345678",
309 "-out", "testreq-sm2.pem", "-sm3"])),
310 "Generating SM2 certificate request");
312 ok(run(app(["openssl", "req",
313 "-config", srctop_file("test", "test.cnf"),
314 "-verify", "-in", "testreq-sm2.pem", "-noout",
315 "-vfyopt", "distid:1234567812345678", "-sm3"])),
316 "Verifying signature on SM2 certificate request");
318 ok(run(app(["openssl", "req",
319 "-config", srctop_file("test", "test.cnf"),
320 "-new", "-key", srctop_file(@certs, "sm2.key"),
321 "-sigopt", "hexdistid:DEADBEEF",
322 "-out", "testreq-sm2.pem", "-sm3"])),
323 "Generating SM2 certificate request with hex id");
325 ok(run(app(["openssl", "req",
326 "-config", srctop_file("test", "test.cnf"),
327 "-verify", "-in", "testreq-sm2.pem", "-noout",
328 "-vfyopt", "hexdistid:DEADBEEF", "-sm3"])),
329 "Verifying signature on SM2 certificate request");
333 my @openssl_args = ("req", "-config", srctop_file("apps", "openssl.cnf"));
335 run_conversion('req conversions',
337 run_conversion('req conversions -- testreq2',
338 srctop_file("test", "testreq2.pem"));
344 subtest $title => sub {
345 run(app(["openssl", @openssl_args,
346 "-in", $reqfile, "-inform", "p",
348 stderr => "req-check.err", stdout => undef));
349 open DATA, "req-check.err";
351 plan skip_all => "skipping req conversion test for $reqfile"
352 if grep /Unknown Public Key/, map { s/\R//; } <DATA>;
354 tconversion( -type => 'req', -in => $reqfile,
355 -args => [ @openssl_args ] );
358 unlink "req-check.err";
364 # Test both generation and verification of certs w.r.t. RFC 5280 requirements
366 my $ca_cert; # will be set below
369 my $ss = $cert =~ m/self-signed/;
370 my $is_ca = $cert =~ m/CA/;
371 my $cn = $is_ca ? "CA" : "EE";
372 my $ca_key = srctop_file(@certs, "ca-key.pem");
373 my $key = $is_ca ? $ca_key : srctop_file(@certs, "ee-key.pem");
374 my @cmd = ("openssl", "req", "-config", "\"\"", "-x509",
375 "-key", $key, "-subj", "/CN=$cn", @_, "-out", $cert);
376 push(@cmd, ("-CA", $ca_cert, "-CAkey", $ca_key)) unless $ss;
377 ok(run(app([@cmd])), "generate $cert");
381 my $expect = shift @_;
382 cert_contains($cert, "Subject Key Identifier", $expect);
386 my $expect = shift @_;
387 cert_contains($cert, "Authority Key Identifier", $expect);
391 my $expect = shift @_;
392 cert_contains($cert, "Key Usage", $expect);
396 my $expect = shift @_;
397 my $trusted = shift @_;
398 $trusted = $cert unless $trusted;
399 ok(run(app(["openssl", "verify", "-x509_strict", "-trusted", $trusted,
400 "-partial_chain", $cert])) == $expect,
401 "strict verify allow $cert");
404 my @v3_ca = ("-addext", "basicConstraints = critical,CA:true",
405 "-addext", "keyUsage = keyCertSign");
406 my $SKID_AKID = "subjectKeyIdentifier,authorityKeyIdentifier";
407 my $cert = "self-signed_v1_CA_no_KIDs.pem";
408 generate_cert($cert);
409 cert_ext_has_n_different_lines($cert, 0, $SKID_AKID); # no SKID and no AKID
410 #TODO strict_verify($cert, 1); # self-signed v1 root cert should be accepted as CA
412 $ca_cert = "self-signed_v3_CA_default_SKID.pem";
413 generate_cert($ca_cert, @v3_ca);
414 has_SKID($ca_cert, 1);
415 has_AKID($ca_cert, 0);
416 strict_verify($ca_cert, 1);
418 $cert = "self-signed_v3_CA_no_SKID.pem";
419 generate_cert($cert, @v3_ca, "-addext", "subjectKeyIdentifier = none");
420 cert_ext_has_n_different_lines($cert, 0, $SKID_AKID); # no SKID and no AKID
421 #TODO strict_verify($cert, 0);
423 $cert = "self-signed_v3_CA_both_KIDs.pem";
424 generate_cert($cert, @v3_ca, "-addext", "subjectKeyIdentifier = hash",
425 "-addext", "authorityKeyIdentifier = keyid");
426 cert_ext_has_n_different_lines($cert, 3, $SKID_AKID); # SKID == AKID
427 strict_verify($cert, 1);
429 $cert = "self-signed_v3_EE_wrong_keyUsage.pem";
430 generate_cert($cert, "-addext", "keyUsage = keyCertSign");
431 #TODO strict_verify($cert, 1); # should be accepted because RFC 5280 does not apply
433 $cert = "v3_EE_default_KIDs.pem";
434 generate_cert($cert, "-addext", "keyUsage = dataEncipherment");
435 cert_ext_has_n_different_lines($cert, 4, $SKID_AKID); # SKID != AKID
436 strict_verify($cert, 1, $ca_cert);
438 $cert = "v3_EE_no_AKID.pem";
439 generate_cert($cert, "-addext", "authorityKeyIdentifier = none");
442 strict_verify($cert, 0, $ca_cert);
444 $cert = "self-issued_v3_EE_default_KIDs.pem";
445 generate_cert($cert, "-addext", "keyUsage = dataEncipherment",
446 "-in", srctop_file(@certs, "x509-check.csr"));
447 cert_ext_has_n_different_lines($cert, 4, $SKID_AKID); # SKID != AKID
448 strict_verify($cert, 1);
450 my $cert = "self-signed_CA_no_keyUsage.pem";
451 generate_cert($cert, "-in", srctop_file(@certs, "ext-check.csr"));
452 has_keyUsage($cert, 0);
453 my $cert = "self-signed_CA_with_keyUsages.pem";
454 generate_cert($cert, "-in", srctop_file(@certs, "ext-check.csr"),
455 "-copy_extensions", "copy");
456 has_keyUsage($cert, 1);