Сертификаты Java и SSL - PullRequest
       2

Сертификаты Java и SSL

10 голосов
/ 28 августа 2011

Я пытаюсь установить соединение с моим PHP-скриптом на Java с использованием уровня защищенных сокетов (HTTPS), но я обнаружил, что для обеспечения максимальной безопасности / достоверности я должен импортировать сертификат SSL, который использует мой веб-сайтв мое приложение ... Что-то, что я не знаю, как это сделать.

Если это помогает, мой сертификат SSL не является самоподписанным, скорее предоставляется StartSSL И я использую Eclipse IDE.

Может ли кто-нибудь указать мне правильное направление?т.е. какие файлы мне нужны, куда я должен их импортировать и какой код мне нужен в Java и т. д.

Ответы [ 5 ]

27 голосов
/ 28 августа 2011

Я обнаружил, что для обеспечения максимальной безопасности / действительности я должен импортировать сертификат SSL, который мой веб-сайт использует в моем приложении

Вы частично правы, когда делаете это заявление. Вам не нужно импортировать ваш сертификат SSL. Достаточно импортировать сертификат StartSSL CA.

Кроме того, нет такой вещи, как импорт сертификата в приложение Java. Поддержка SSL в Java основана на концепции хранилищ ключей и доверенных сертификатов, а не на каком-то сертификате, упакованном в вашем приложении. Если вы публикуете свое приложение для загрузки и выполнения конечными пользователями, вам не нужно публиковать свой сертификат или, в этом случае, свой закрытый ключ в своем приложении. Закрытый ключ и связанный сертификат будут храниться в хранилище ключей, к которому имеют доступ только вы.

Конечные пользователи вашего приложения будут полагаться на поддержку SSL в среде выполнения Java, что позволит приложению устанавливать SSL-соединения с сайтами после проверки сертификата сервера. Среда выполнения Java поставляется с набором сертификатов CA по умолчанию в хранилище доверенных сертификатов, и единственным условием успешного установления SSL-соединений является выдача SSL-сертификата сервера одним из CA в хранилище доверенных сертификатов. Сертификаты StartSSL отсутствуют в доверенном хранилище среды выполнения Java , по крайней мере, начиная с версии 6, и поэтому:

  • Вы можете указать конечным пользователям выполнить операцию импорта сертификата StartSSL CA в хранилище доверенных сертификатов Java. Ссылки, которые могут помочь, включают эту ветку форума StartSSL (для импорта сертификатов CA в хранилище доверенных сертификатов необходимы только первые 4 шага), проект GitHub и это сообщение в блоге ; отказ от ответственности - я не пытался использовать ни один из них, и вы должны использовать его на свой страх и риск.
  • Или вы можете инициализировать свое приложение с собственным складом доверенных сертификатов с помощью флагов запуска -Djavax.net.ssl.trustStore=<path_to_truststore> -Djavax.net.ssl.trustStorePassword=<truststore_password> JVM или выполнить следующий код перед инициализацией соединений SSL:

    System.setProperty("javax.net.ssl.trustStore","<path_to_truststore>");
    System.setProperty("javax.net.ssl.trustStorePassword","<truststore_password>");
    

    Это жизнеспособный подход, только если ваше приложение является приложением Java SE, которое не является апплетом (или приложением с аналогичными ограничениями в отношении указания хранилища доверенных сертификатов).


Также помогло бы прочитать документацию keytool .

2 голосов
/ 25 января 2018

Очевидно, что инженеры почтовой пушки по какой-то причине не хотят давать нам четкие инструкции о том, как решить эту проблему. Это то, что я сделал

Мы запускаем tomcat8 и подключаемся через веб-службы jersey к API mailgun. Я следовал инструкциям этого пользователя, и он работал нормально. Надеюсь, это кому-нибудь поможет.

1/22 мы обновили наши SSL-сертификаты, поскольку инфраструктура PKI Symantec стала ненадежной. Некоторые старые версии Java не имеют CA «DigiCert Global Root G2».

Есть несколько вариантов:

Импортируйте CA "DigiCert Global Root G2" в ваш файл "cacerts". Обновите свой JRE до 8u91 (или выше), который включает этот корень. Чтобы импортировать «DigiCert Global Root G2» Вы можете загрузить рут с https://www.digicert.com/digicert-root-certificates.htm. Убедитесь, что вы загружаете правильный корневой сертификат.

После загрузки сертификата его необходимо импортировать с помощью команды, подобной следующей:

keytool -import -trustcacerts -keystore / path / to / cacerts -storepass changeit -noprompt -alias digicert-global-root-g2 -file /path/to/digicert.crt Вам нужно будет указать путь к вашему хранилищу ключей Java и местоположение загруженного корневого сертификата.


Так 1. /path/to/digicert.crt - это файл, который вы только что загрузили. 2. / path / to / cacerts - это ваш путь JRE. Я "найду / -name cacerts -print", это поможет вам быстро найти все java cacerts в вашей файловой системе. Для меня это был / usr / lib / jvm / java-7-openjdk-amd64 / jre / lib / security / cacerts

1 голос
/ 15 января 2015

Следующий метод загружает хранилище ключей по умолчанию (cacerts), проверяет, установлен ли сертификат, и устанавливает его, если нет. Это устраняет необходимость вручную запускать команду keystore на любых серверах.

Предполагается, что пароль хранилища ключей по умолчанию (changeit) не изменился, обновите CACERTS_PASSWORD, если нет. Обратите внимание, что метод сохраняет хранилище ключей после добавления сертификата, поэтому после запуска, когда сертификат будет постоянно находиться в хранилище.

import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.net.MalformedURLException;
import java.net.URL;
import java.security.KeyStore;
import java.security.KeyStoreException;
import java.security.NoSuchAlgorithmException;
import java.security.cert.Certificate;
import java.security.cert.CertificateException;
import java.security.cert.CertificateFactory;

/**
 * Add a certificate to the cacerts keystore if it's not already included
 */
public class SslUtil {
    private static final String CACERTS_PATH = "/lib/security/cacerts";

    // NOTE: DO NOT STORE PASSWORDS IN PLAIN TEXT CODE, LOAD AT RUNTIME FROM A SECURE CONFIG
    // DEFAULT CACERTS PASSWORD IS PROVIDED HERE AS A QUICK, NOT-FOR-PRODUCTION WORKING EXAMPLE
    // ALSO, CHANGE THE DEFAULT CACERTS PASSWORD, AS IT IMPLORES YOU TO!
    private static final String CACERTS_PASSWORD = "changeit";

    /**
     * Add a certificate to the cacerts keystore if it's not already included
     * 
     * @param alias The alias for the certificate, if added
     * @param certInputStream The certificate input stream
     * @throws KeyStoreException
     * @throws NoSuchAlgorithmException
     * @throws CertificateException
     * @throws IOException
     */
    public static void ensureSslCertIsInKeystore(String alias, InputStream certInputStream)
            throws KeyStoreException, NoSuchAlgorithmException, CertificateException, IOException{
        //get default cacerts file
        final File cacertsFile = new File(System.getProperty("java.home") + CACERTS_PATH);
        if (!cacertsFile.exists()) {
            throw new FileNotFoundException(cacertsFile.getAbsolutePath());
        }

        //load cacerts keystore
        FileInputStream cacertsIs = new FileInputStream(cacertsFile);
        final KeyStore cacerts = KeyStore.getInstance(KeyStore.getDefaultType());
        cacerts.load(cacertsIs, CACERTS_PASSWORD.toCharArray());
        cacertsIs.close();

        //load certificate from input stream
        final CertificateFactory cf = CertificateFactory.getInstance("X.509");
        final Certificate cert = cf.generateCertificate(certInputStream);
        certInputStream.close();

        //check if cacerts contains the certificate
        if (cacerts.getCertificateAlias(cert) == null) {
            //cacerts doesn't contain the certificate, add it
            cacerts.setCertificateEntry(alias, cert);
            //write the updated cacerts keystore
            FileOutputStream cacertsOs = new FileOutputStream(cacertsFile);
            cacerts.store(cacertsOs, CACERTS_PASSWORD.toCharArray());
            cacertsOs.close();
        }
    }
}

Используйте это так:

SslUtil.ensureSslCertIsInKeystore("startssl", new FileInputStream("/path/to/cert.crt"));
0 голосов
/ 29 августа 2011

Я обнаружил, что для обеспечения максимальной безопасности / действительности я должен импортировать сертификат SSL

Нет, нет. Этот шаг необходим только в том случае, если ваши клиенты еще не доверяют подписавшему сертификат сервера, который возникает, только если сертификат сервера самоподписан или подписан, например внутренним CA.

0 голосов
/ 28 августа 2011

Просмотрите следующую статью: http://stilius.net/java/java_ssl.php Содержит пример кода, который может помочь в случае, если вы пытаетесь получить доступ к своему сценарию из кода.

Обратите внимание, что вы должны использовать системусвойства

javax.net.ssl.keyStore
javax.net.ssl.keyStorePassword

для передачи SSL-сертификата в JVM или импорта его в хранилище ключей JRE с помощью keytool tool

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...