Skip to content
  • Categories
  • Recent
  • Tags
  • Popular
  • Users
  • Groups
  • Search
  • Get Qt Extensions
  • Unsolved
Collapse
Brand Logo
  1. Home
  2. Qt Development
  3. QML and Qt Quick
  4. Add client certificate to QQuickWebEngineProfile clientCertificateStore in QML
Forum Updated to NodeBB v4.3 + New Features

Add client certificate to QQuickWebEngineProfile clientCertificateStore in QML

Scheduled Pinned Locked Moved Unsolved QML and Qt Quick
14 Posts 3 Posters 1.4k Views 1 Watching
  • Oldest to Newest
  • Newest to Oldest
  • Most Votes
Reply
  • Reply as topic
Log in to reply
This topic has been deleted. Only users with topic management privileges can see it.
  • fcarneyF fcarney

    @notawhiteowl said in Add client certificate to QQuickWebEngineProfile clientCertificateStore in QML:

    clientCertificateStore

    WebEngineProfile != QQuickWebEngineProfile

    N Offline
    N Offline
    notawhiteowl
    wrote on last edited by
    #5

    @fcarney In the qml file, for example, if I console.log(profile) in the qml file it logs QQuickWebEngineProfile(0x55f1064907f0)

    So, I believe that profile here is a qquickwebengineprofile.
    that same profile is being used successfully, elsewhere in the code, to set the user-agent in the WebEngineView, so I believe that QQuickWebEngineProfile profile is in fact the QQuickWebEngineProfile of the WebEngineView.... does this sound correct?
    and according to the documentation link, that is the exact object which contains the public function QWebEngineClientCertificateStore * clientCertificateStore()

    So what am I missing?
    I really appreciate you trying to help me, I'm sorry that my ignorance is the primary roadblock.

    The full QML is at: https://github.com/jellyfin/jellyfin-media-player/blob/master/src/ui/webview.qml
    and I believe it's being instantiated here: https://github.com/jellyfin/jellyfin-media-player/blob/master/src/main.cpp

    If you wouldn't mind taking a glance at it and pointing me in the right direction, I'd really appreciate it. There is a dearth of real-world examples for this case.

    The second question is this: Does QWebEngineClientCertificateSelection even access the in-memory keystore? Or will it only provide certificateOptions from the system store?

    1 Reply Last reply
    0
    • fcarneyF Offline
      fcarneyF Offline
      fcarney
      wrote on last edited by
      #6

      They may be the same object, but it doesn't expose that to QML. The APIs are different between QML and C++. You are looking at C++ docs and wondering why it doesn't work in QML.

      C++ is a perfectly valid school of magic.

      1 Reply Last reply
      0
      • eyllanescE eyllanesc

        @notawhiteowl A possible solution is to create a QObject as a helper:

        class Helper: public QObject{
        public:
            using QObject;
            Q_INVOKABLE void foo(QQuickWebEngineProfile *profile){
                // profile->clientCertificateStore();
            }
        };
        

        So I expose a Helper object as contextProperty:

        Helper helper;
        engine()->rootContext()->setContextProperty("helper", &helper);
        

        And then you use it in QML:

        Component.onCompleted: helper.foo(web.profile);
        
        N Offline
        N Offline
        notawhiteowl
        wrote on last edited by
        #7

        @eyllanesc I sent you an email to see if you have availability coming up. hoping it didnt go to spam.

        eyllanescE 2 Replies Last reply
        0
        • N notawhiteowl

          @eyllanesc I sent you an email to see if you have availability coming up. hoping it didnt go to spam.

          eyllanescE Offline
          eyllanescE Offline
          eyllanesc
          wrote on last edited by
          #8
          This post is deleted!
          1 Reply Last reply
          0
          • N notawhiteowl

            @eyllanesc I sent you an email to see if you have availability coming up. hoping it didnt go to spam.

            eyllanescE Offline
            eyllanescE Offline
            eyllanesc
            wrote on last edited by
            #9

            @notawhiteowl @notawhiteowl No, no problem. Just for this I have placed my signature, I will respond to you in a short time.

            If you have an account in discord, we could talk more comfortably there. My account is eyllanesc#5911.

            1 Reply Last reply
            0
            • fcarneyF Offline
              fcarneyF Offline
              fcarney
              wrote on last edited by
              #10

              Would have QML != C++ been more clear?

              C++ is a perfectly valid school of magic.

              N 1 Reply Last reply
              0
              • fcarneyF fcarney

                Would have QML != C++ been more clear?

                N Offline
                N Offline
                notawhiteowl
                wrote on last edited by notawhiteowl
                #11

                @fcarney i appreciate your comfort in these topics very much. bottom line, do you think it will be possible to implement the in-memory client keystore and utilize it in this case, or would it need to involve a major rewrite?

                1 Reply Last reply
                0
                • fcarneyF Offline
                  fcarneyF Offline
                  fcarney
                  wrote on last edited by
                  #12

                  @notawhiteowl said in Add client certificate to QQuickWebEngineProfile clientCertificateStore in QML:

                  do you think it will be possible to implement the in-memory client keystore and utilize it in this case, or would it need to involve a major rewrite?

                  As @eyllanesc showed you will have to write your own c++ functions to expose that functionality to QML.

                  C++ is a perfectly valid school of magic.

                  N 1 Reply Last reply
                  0
                  • fcarneyF fcarney

                    @notawhiteowl said in Add client certificate to QQuickWebEngineProfile clientCertificateStore in QML:

                    do you think it will be possible to implement the in-memory client keystore and utilize it in this case, or would it need to involve a major rewrite?

                    As @eyllanesc showed you will have to write your own c++ functions to expose that functionality to QML.

                    N Offline
                    N Offline
                    notawhiteowl
                    wrote on last edited by notawhiteowl
                    #13

                    @fcarney Thank you for your input! We have gotten as far as this helper object.

                    #include <QGuiApplication>
                    #include <QQmlApplicationEngine>
                    #include <QQmlContext>
                    #include <QSslCertificate>
                    #include <QtWebEngine>
                    
                    class ClientCertificateHelper: public QObject{
                        Q_OBJECT
                    public:
                        using QObject::QObject;
                        Q_INVOKABLE bool init(QQuickWebEngineProfile *profile){
                            QSslKey key;
                            QSslCertificate certificate;
                            QString pfx_path = "/path/to/client.pfx";
                            QFile file(pfx_path);
                            if(!file.open(QFile::ReadOnly)){
                                return false;
                            }
                            if(QSslCertificate::importPkcs12(&file, &key, &certificate, nullptr, "exportpassword")){
                                profile->clientCertificateStore()->add(certificate, key);
                                QLOG_DEBUG() << "Client certificate successfully loaded "+pfx_path;
                                return true;
                            }
                            QLOG_DEBUG() << "FAILED to load client certificate from "+pfx_path;
                            return false;
                        } 
                    };
                    #include "main.moc"
                    ...
                    ...
                    QObject::connect(engine, &QQmlApplicationEngine::objectCreated, [=](QObject* object, const QUrl& url)
                        {
                          Q_UNUSED(url);
                    
                          if (object == nullptr)
                            throw FatalException(QObject::tr("Failed to parse application engine script."));
                    
                          KonvergoWindow* window = Globals::MainWindow();
                    
                          QObject* webChannel = qvariant_cast<QObject*>(window->property("webChannel"));
                          Q_ASSERT(webChannel);
                          ComponentManager::Get().setWebChannel(qobject_cast<QWebChannel*>(webChannel));
                    
                          QObject::connect(uniqueApp, &UniqueApplication::otherApplicationStarted, window, &KonvergoWindow::otherAppFocus);
                        });
                        
                        engine->rootContext()->setContextProperty("clientCertificateHelper", &clientCertificateHelper);
                        engine->load(QUrl(QStringLiteral("qrc:/ui/webview.qml")));
                    
                        Log::UpdateLogLevel();
                    
                        // run our application
                        int ret = app.exec();
                    

                    and have modified webview.qml thusly:

                    import QtQuick 2.15
                    import QtQuick.Window 2.15
                    import QtWebEngine 1.10
                    ...
                    ...
                    Component.onCompleted:
                        {
                          if (clientCertificateHelper.init(profile))
                          {
                            console.log("Helper returned true")
                          }
                          else
                          {
                            console.log("Helper did not return true")
                          }
                          forceActiveFocus()
                          mainWindow.reloadWebClient.connect(reload)
                        }
                    ...
                    ...
                    onSelectClientCertificate: function(selection)
                         {
                          console.log("CLIENT CERTIFICATE REQUESTED")
                          selection.certificates[0].select();
                         }
                    

                    So, the helper object returns true, the certificate constructor is happy, but selectClientCertificate is never fired.
                    So, I have three particular questions.
                    https://doc.qt.io/qt-5/qwebenginepage.html#selectClientCertificate
                    does not mention firing the signal based on in memory store,
                    but then https://doc.qt.io/Qt-5/qwebengineclientcertificatestore.html suggests that it does.

                    1. What is the reason for this discrepancy?
                    2. Why is selectClientCertificate not firing in my application, when the server is known-working?
                    3. Is my goal of using the in-memory store reasonable in this application?
                    1 Reply Last reply
                    0
                    • N Offline
                      N Offline
                      notawhiteowl
                      wrote on last edited by
                      #14

                      ugh. I guess this feature has just been broken for over a year with misleading documentation. no big deal.
                      thanks for your help everybody.
                      https://bugreports.qt.io/browse/QTBUG-86132

                      1 Reply Last reply
                      0

                      • Login

                      • Login or register to search.
                      • First post
                        Last post
                      0
                      • Categories
                      • Recent
                      • Tags
                      • Popular
                      • Users
                      • Groups
                      • Search
                      • Get Qt Extensions
                      • Unsolved