QWebView / Qt WebKit не открывает некоторые страницы SSL; перенаправления не допускаются? - PullRequest
12 голосов
/ 03 декабря 2011

Чистая установка Qt SDK 1.1.4 в Windows 7 с Visual C ++ 2008 SP1; Я использую Qt Creator. Почему этот код не загружает некоторые веб-страницы?

#include <QtGui/QApplication>
#include <QtWebKit/QWebView>

int main(int argc, char *argv[])
{
    QApplication a(argc, argv);
    QWebView b;
    b.load(QUrl("https://gmail.com")); // doesn't work
    //b.load(QUrl("https://accounts.google.com")); // works
    //b.load(QUrl("https://google.com")); // doesn't work
    //b.load(QUrl("https://www.google.com")); // works
    b.show();

    return a.exec();
}

Почему некоторые URL не работают, а другие работают?

Я думаю, что google.com / www.google.com особенно показателен; google.com обычно перенаправляет на www.google.com. И gmail.com перенаправляет на account.google.com. Разве WebKit не позволяет перенаправлять защищенные страницы? Если да, то как это исправить?

Кстати, Qt SDK 1.1.4, похоже, включает OpenSSL; Я заметил его присутствие в C: \ QtSDK \ Desktop \ Qt \ 4.7.4 \ msvc2008 \ bin \ ssleay32.dll. Также обратите внимание, что некоторые страницы, кажется, работают, но не другие.

РЕДАКТИРОВАТЬ: Еще два URL:

b.load(QUrl("https://support.motionview3d.com/help/_media/images/directory.png")); // doesn't work
b.load(QUrl("https://mail.google.com")); // works

Опять же, оба они отлично работают в других веб-браузерах.

Ответы [ 2 ]

18 голосов
/ 03 декабря 2011

Возможно, вы получаете ошибки SSL, которые вы можете обработать в слоте. Хотя это и не лучшее окончательное решение, вы можете использовать слот, чтобы игнорировать все ошибки SSL. Я сделал это подклассами QWebView:

qwebview.h:

#ifndef WEBVIEW_H
#define WEBVIEW_H

#include <QWebView>

class WebView : public QWebView
{
    Q_OBJECT

    public:
        WebView(QWidget *parent = 0);
    private slots:
        void handleSslErrors(QNetworkReply* reply, const QList<QSslError> &errors);
};

#endif // WEBVIEW_H

qwebview.cpp:

#include "webview.h"
#include <QNetworkReply>
#include <QtDebug>
#include <QSslError>

WebView::WebView(QWidget *parent) :
    QWebView(parent)
{
    load(QUrl("https://gmail.com"));

    connect(page()->networkAccessManager(),
            SIGNAL(sslErrors(QNetworkReply*, const QList<QSslError> & )),
            this,
            SLOT(handleSslErrors(QNetworkReply*, const QList<QSslError> & )));  
}

void WebView::handleSslErrors(QNetworkReply* reply, const QList<QSslError> &errors)
{
    qDebug() << "handleSslErrors: ";
    foreach (QSslError e, errors)
    {
        qDebug() << "ssl error: " << e;
    }

    reply->ignoreSslErrors();
}

main.cpp "

#include <QApplication>
#include "WebView.h"

int main(int argc, char *argv[])
{
    QApplication a(argc, argv);
    WebView w;
    w.show();
    return a.exec();
}

Выполнение этого должно привести к выводу отладки следующим образом:

handleSslErrors:  
ssl error:  "The host name did not match any of the valid hosts for this certificate" 
ssl error:  "No error" 
ssl error:  "No error" 
...

В вашей окончательной программе вы, конечно, захотите правильно обрабатывать ошибки SSL:)

0 голосов
/ 28 октября 2015

Я обычно использую решение "Арнольда Спенса", но иногда это не сработает.

в этом случае просто измените конфигурацию Ssl по умолчанию следующим образом:

QSslConfiguration sslconf = QSslConfiguration::defaultConfiguration();
QList<QSslCertificate> cert_list = sslconf.caCertificates();
QList<QSslCertificate> cert_new = QSslCertificate::fromData("CaCertificates");
cert_list += cert_new;

sslconf.setCaCertificates(cert_list);
sslconf.setProtocol(QSsl::AnyProtocol);
QSslConfiguration::setDefaultConfiguration(sslconf);

Здесь мы изменили конфигурацию для всего приложения.

Рекомендую также обработать сигнал sslErrors ..

...