Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Pkcs7 pr4 #3

Open
wants to merge 5 commits into
base: mbedtls-2.16
Choose a base branch
from
Open

Pkcs7 pr4 #3

wants to merge 5 commits into from

Conversation

naynajain
Copy link
Owner

Hi,

I have the second iteration for PKCS7 PR ready to be pushed to upstream community.
I would like to know if any feedbacks. This is still based on 2.16 version and I will update it to "development" branch before sending upstream.

Few notes:

  1. This version supports single signer. However, even for single signer you can have multiple certificates as certificate chain. But it seems it is not easy to differentiate whether the list of certificates are certificate chain or just different certificates. So, I restricted the whole support to single root certificate, and not even certificate chain for the single signer support. Moreover, we do not need also this support, so it is probably ok to leave it like this as of now rather than spending too much time on figuring out how to support it.

  2. As per CRL support, it seems signedData has one more purpose of passing around certificates and CRLs packed as PKCS7 signed data. I couldn't find openssl command line way to specify CRL file while using PKCS7 for signature. It might take it automatically from CA, but not very sure. However, I did find openssl crl2pkcs7 command as specified in the following link. From the following two links, I came to conclusion that CRL works more in combination with certificates to be passed around rather than actually part of PKCS7 signature usecase.
    https://www.mkssoftware.com/docs/man1/openssl_crl2pkcs7.1.asp
    And if you see the GnuTLS link (https://www.gnutls.org/manual/html_node/Cryptographic-Message-Syntax-_002f-PKCS7.html) , it specifically says this - In certain cases the same format is also used to transport lists of certificates and CRLs.
    We do not have such use case for CRL and even from revocation perspective we make use of dbx and not the CRLs atleast as of now. So, again I have strictly made it to not consider CRL. If CRL is added, it will fail while parsing it as signer info.

  3. I did realize that I removed the test for CRL but didn't remove it from Makefile. I will fix that before sending upstream.

  4. I thought too much about error codes. After switching multiple times between narrowing it further to give exact parse fail or to keep it only a high level fail, I finally chose to keep it high level until I understand how people are having issues in interpreting it. I think it will improve over time.

Please let me know if any feedbacks by Tuesday EOD.. so that we can try to send second iteration sometime by end of this week.

Thanks & Regards,
- Nayna

naynajain and others added 5 commits October 25, 2020 01:41
PKCS7 signing format is used by OpenPOWER Key Management, which is
using mbedtls as its crypto library.

This patch adds the limited support of pkcs7 parser and verification
to the mbedtls. The limitations are:

* Only signed data is supported.
* CRLs are not currently handled.
* Single signer is supported.

Signed-off-by: Daniel Axtens <[email protected]>
Signed-off-by: Nayna Jain <[email protected]>
OpenSSL provides APIs to generate only the signted data
format PKCS7 i.e. without content type OID. This patch
adds support to parse the data correctly even if formatted
only as signed data.

Signed-off-by: Nayna Jain <[email protected]>
This patch updates the generate_errors.pl to handle
PKCS7 code as well.

Signed-off-by: Nayna Jain <[email protected]>
This patch adds the updates generated by running generate_features.pl
for pkcs7.

Signed-off-by: Nayna Jain <[email protected]>
The patch enables pkcs7 where File I/O may not be available.

Reported-by: Eric Richter<[email protected]>
Signed-off-by: Daniel Axtens <[email protected]>
Copy link

@daxtens daxtens left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Hi Nayna!

I've looked at everything except test_suite_pkcs7.function. It looks good! 👍 I have left some minor comments but I'm happy with the overall design and function.

Looking back at https://ibm.ent.box.com/notes/724259293796 - did we conclude on the zero signers case? I think at the moment we will reject a zero signerInfo case, but I think we could support it in future without a big API change: a user could distinguish between a present signerInfo and an absent signerInfo by looking at the version field of signerInfo. It would be 1 if there's a signerInfo and 0 if there isn't. That's a bit inelegant but it should work, so I guess that's probably OK.

Similarly, with zero certificates - I think looking at the code, we will successfully parse a PKCS#7 message with 1 or 0 certificates. If a user wanted to determine if there was a certificate parsed, they could check to see if signed_data->certs.version != 0.

Finally, I'm not sure I fully understood what the mbedTLS team meant when they asked for 'negative tests' in your last PR - do you think you've covered that?

Kind regards,
Daniel

#if ( ( defined(MBEDTLS_PKCS7_C) ) && ( !defined(MBEDTLS_ASN1_PARSE_C) ) && \
( !defined(MBEDTLS_OID_C) ) && ( !defined(MBEDTLD_PK_PARSE_C) ) && \
( !defined(MBEDTLS_X509_CRT_PARSE_C) ) && ( !defined(MBEDTLS_BIGNUM_C) ) )
#error "MBEDTLS_PKCS7_C is defined, but not all prerequisites"
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think this will only error if you have none of the prerequisites. Should it be:


#if ( defined(MBEDTLS_PKCS7_C) ) && ( ( !defined(MBEDTLS_ASN1_PARSE_C) ) || \
    ( !defined(MBEDTLS_OID_C) ) || ( !defined(MBEDTLD_PK_PARSE_C) ) || \
    ( !defined(MBEDTLS_X509_CRT_PARSE_C) ) || ( !defined(MBEDTLS_BIGNUM_C) ) )

?

Copy link
Owner Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thanks for catching it. Fixed.

MBEDTLS_PKCS7_DATA,
MBEDTLS_PKCS7_SIGNED_DATA,
MBEDTLS_PKCS7_ENVELOPED_DATA,
MBEDTLS_PKCS7_SIGNED_AND_ENVELOPED_DAYA,
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

DATA, not DAYA :)

Copy link
Owner Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Super catch. Thanks :-)

int version;
mbedtls_x509_buf serial;
mbedtls_x509_name issuer;
mbedtls_x509_buf issuer_raw;
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Minor: Do we need to store issuer_raw if we also have issuer?

Copy link
Owner Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Hmm probably no.. I actually added it very initially.. I will recheck it and remove if not needed.

Copy link
Owner Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think I kept it for the reason it is in x509_crt.h
mbedtls_x509_buf issuer_raw; /**< The raw issuer data (DER). Used for quick comparison. */

I think since signer and issuer are supposed to be unique, and might be used for checks, I thought to keep it for reason of quick comparision.

With that I am not removing it.

Thanks & Regards,
- Nayna

{
int version;
mbedtls_pkcs7_buf digest_alg_identifiers;
struct mbedtls_pkcs7_data content;
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

You just established a typedef for this: should you use the typedef instead of the full struct name?

struct mbedtls_pkcs7_data content;
mbedtls_x509_crt certs;
mbedtls_x509_crl crl;
struct mbedtls_pkcs7_signer_info signers;
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Likewise.

@@ -1048,6 +1048,99 @@ cert_md5.crt: cert_md5.csr
$(MBEDTLS_CERT_WRITE) request_file=$< serial=6 issuer_crt=$(test_ca_crt) issuer_key=$(test_ca_key_file_rsa) issuer_pwd=$(test_ca_pwd_rsa) not_before=20190210144406 not_after=20290210144406 md=MD5 version=3 output_file=$@
all_final += cert_md5.crt

#PKCS7 test data
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

pkcs7-rsa-sha256-1.crt:
$(OPENSSL) req -x509 -subj="/C=NL/O=PKCS7/CN=PKCS7 Cert 1" -sha256 -nodes -days 365 -newkey rsa:2048 -keyout pkcs7-rsa-sha256-1.key -out pkcs7-rsa-sha256-1.crt
cat pkcs7-rsa-sha256-1.crt > pkcs7-rsa-sha256-1.pem
cat pkcs7-rsa-sha256-1.key >> pkcs7-rsa-sha256-1.pem
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

you can write cat pkcs7-rsa-sha256-1.crt pkcs7-rsa-sha256-1.key > pkcs7-rsa-sha256-1.pem as cat will automatically concatenate its arguments :)

#PKCS7 test data
pkcs7_test_cert_1 = pkcs7-rsa-sha256-1.crt
pkcs7_test_cert_2 = pkcs7-rsa-sha256-2.crt
pkcs7_test_file = pkcs7_data.txt
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I can't find any other reference to these variables - am I missing something?

Copy link
Owner Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Hmm.. I did see other commands where they were using the variables, and thought to do so but then looks like I didn't. I will update the commands to use the variable.

pkcs7_data.txt:
echo "Hello" > $@
echo 2 >> pkcs7_data_1.txt
all_final += pkcs7_data.txt
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

all_final only contains pkcs7_data.txt, but you've also created pkcs7_data_1.txt. Is this intentional?

Copy link
Owner Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

yes because that gets created as part of pkcs7_data.txt target. all_final is mainly to add targets.

pkcs7_data_crl.signed:
cat pkcs7-rsa-sha256-1.pem > pkcs7-rsa-crt-crl.pem
cat crl_sha256.pem >> pkcs7-rsa-crt-crl.pem
$(OPENSSL) smime -sign -binary -in pkcs7_data.txt -out $@ -md sha512 -signer pkcs7-rsa-crt-crl.pem -inkey pkcs7-rsa-sha256-1.key -noattr -outform DER -out $@
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

If you depend on crl_sha256.pem you should probably add that to the top line (pkcs7_data_crl.signed: crl_sha256.pem). There are a few dependencies like this in your changes.

@naynajain
Copy link
Owner Author

Hi Nayna!

I've looked at everything except test_suite_pkcs7.function. It looks good! +1 I have left some minor comments but I'm happy with the overall design and function.

Looking back at https://ibm.ent.box.com/notes/724259293796 - did we conclude on the zero signers case? I think at the moment we will reject a zero signerInfo case, but I think we could support it in future without a big API change: a user could distinguish between a present signerInfo and an absent signerInfo by looking at the version field of signerInfo. It would be 1 if there's a signerInfo and 0 if there isn't. That's a bit inelegant but it should work, so I guess that's probably OK.

Thanks Daniel.. We didn't discuss it again, but as I started understanding more.. I felt that
zero signer is a requirement when it used to transport list of certificates and CRLs. Openssl also provides this feature - https://www.mkssoftware.com/docs/man1/openssl_crl2pkcs7.1.asp
I felt atleast as per our requirements I think we are not doing this usecase.. I agree the API should be such that it can be extended easily in future.. then in that case I am hoping that struct will not have to be modified for supporting this atleast.. so as you said.. yes without changing the API

Similarly, with zero certificates - I think looking at the code, we will successfully parse a PKCS#7 message with 1 or 0 certificates. If a user wanted to determine if there was a certificate parsed, they could check to see if signed_data->certs.version != 0.

Finally, I'm not sure I fully understood what the mbedTLS team meant when they asked for 'negative tests' in your last PR - do you think you've covered that?

My understanding from negative tests is like - what if certificate section of ASN is corrupted.. and it fails to parse. so we should have some of those cases. I think I did cover like badcert, badsigner that is by modifing ASN files and ensuring it fails to parse and returns with failure error code. I hope I understood it right.. lets see .. writing test cases had been bit tricky.

Kind regards,
Daniel

Thanks for your review.

Thanks & Regards,
- Nayna

@naynajain
Copy link
Owner Author

Will include the other feedbacks.

Thanks & Regards,
- Nayna

@naynajain
Copy link
Owner Author

I just realized.. why don't we use the count for certificates/crls/signers like below.
typedef struct mbedtls_pkcs7_signed_data
{
int version;
mbedtls_pkcs7_buf digest_alg_identifiers;
struct mbedtls_pkcs7_data content;
int no_of_certs;
mbedtls_x509_crt certs;
int no_of_crls;
mbedtls_x509_crl crl;
int no_of_signers;
mbedtls_pkcs7_signer_info signers;
}
mbedtls_pkcs7_signed_data;

Their internal (static) parsing function can return the count or negative error code. As per count, 0 or positive number is valid value.

I think that would really simplify everything. I have added them to pkcs7 signed data struct. Hmm.. the tests are not broken so I am hoping any of the consumers will not be broken by this change.

I am sending PR as of now with this change.. if you think this idea has issues, we will do it asap once I am back..

@daxtens
Copy link

daxtens commented Nov 11, 2020

Just confirming, I think that tracking the number of certs/crls/signers with another variable makes sense. :)

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

2 participants