1

x509 証明書を使用した S/Mime 署名付きメッセージの検証に問題があります。これは私のコードです:

public class verifyMsg {

private static void verify(SMIMESignedParser s) throws Exception {

    Security.addProvider(new BouncyCastleProvider());
    System.out.println("wbilem");
    CertStore certs = s.getCertificatesAndCRLs("Collection", "BC");

    SignerInformationStore signers = s.getSignerInfos();
    Collection c = signers.getSigners();
    Iterator it = c.iterator();

    while (it.hasNext()) {
        File f = new File("signature.crt");
        FileInputStream fis = new FileInputStream(f);
        DataInputStream dis = new DataInputStream(fis);
        byte[] keyBytes = new byte[(int) f.length()];
        dis.readFully(keyBytes);
        dis.close();
        fis.close();
        SignerInformation signer = (SignerInformation) it.next();
        Collection certCollection = certs.getCertificates(signer.getSID());
        Iterator certIt = certCollection.iterator();
        FileInputStream fr = new FileInputStream("signature.crt");
        CertificateFactory cf = CertificateFactory.getInstance("X509");
        X509Certificate cert = (X509Certificate) cf.generateCertificate(fr);


        if (signer.verify(cert, "BC")) { //problem is there...
            System.out.println("signature verified");
        } else {
            System.out.println("signature failed!");
        }

    }
}

public static void main(String[] args) throws Exception {
    Properties props = System.getProperties();
    Session session = Session.getDefaultInstance(props, null);

    try {
        FileInputStream fr = new FileInputStream("signature.crt");
        CertificateFactory cf = CertificateFactory.getInstance("X509");
        X509Certificate c = (X509Certificate) cf.generateCertificate(fr);
        System.out.println("Read in the following certificate:");
        System.out.println("\tCertificate for: " + c.getSubjectDN());
        System.out.println("\tCertificate issued by: " + c.getIssuerDN());
        System.out.println("\tThe certificate is valid from " + c.getNotBefore() + " to " + c.getNotAfter());
        System.out.println("\tCertificate SN# " + c.getSerialNumber());
        System.out.println("\tGenerated with " + c.getSigAlgName());
        System.out.println(c.getPublicKey());
    } catch (Exception e) {
        e.printStackTrace();
    }

    try {
        MimeMessage msg = new MimeMessage(session, new SharedFileInputStream("G:\\MIME.txt"));
        if (msg.isMimeType("multipart/signed")) {

            SMIMESignedParser s = new SMIMESignedParser((MimeMultipart) msg.getContent());
            System.out.println("Status:");
            verify(s);
        } else if (msg.isMimeType("application/pkcs7-mime")) {

            // in this case the content is wrapped in the signature block.
            //
            SMIMESignedParser s = new SMIMESignedParser(msg);
            System.out.println("Status1:");
            verify(s);
        } else {
            System.err.println("Not a signed message!");
        }

    } catch (MessagingException e) {
        // TODO Auto-generated catch block
        e.printStackTrace();
    } catch (IOException e) {
        // TODO Auto-generated catch block
        e.printStackTrace();
    } catch (CMSException e) {
        // TODO Auto-generated catch block
        e.printStackTrace();
    }
}

}

そして、私はこの例外に問題があります:

CMSSignerDigestMismatchException: メッセージ ダイジェスト属性値が計算値と一致しません。私は何が間違っているのか分かりません。私はjdk 1.4.2を使用しています。

4

2 に答える 2

1

問題はメッセージにあることがわかりました。バイト配列を文字列に変換してから、この文字列を入力ストリームに変換しました。今、私は変換せずにinputstreamバイト配列に与え、すべてが大丈夫です:)

于 2011-12-22T14:55:04.130 に答える
0

この問題を解決しました。S/MIME でデータに署名している場合、S/MIME 標準に従って、MIME-Version、Content-Type などを指定する必要があり、ヘッダーはメッセージから分離され、空白行で署名されます。ほとんどのメッセージのタイプが「text/plain」の場合、メッセージにヘッダーを追加しませんが、これは明らかです。ただし、S/MIME を使用してデータに署名している場合は、S/MIME 標準ヘッダーと、miss mach などの問題を探します。ダイジェストで発生します。

MIME バージョンに署名する前に、次のヘッダーをメッセージに追加します。1.0 Content-Type: text/plain;

または、メッセージの前に空白行を 1 行追加します。

上記のアプローチは私にとってはうまくいきます。

于 2013-12-18T03:36:56.407 に答える