aboutsummaryrefslogtreecommitdiffstats
path: root/examples
diff options
context:
space:
mode:
authorFriedemann Kleint <Friedemann.Kleint@qt.io>2021-07-09 11:38:59 +0200
committerCristián Maureira-Fredes <Cristian.Maureira-Fredes@qt.io>2021-07-12 12:25:12 +0200
commit38376cdb3bfeb69d34ca40090994e83600f9710b (patch)
treecda41bea61971172fbbe3c11efeb62f754c87264 /examples
parent081bafdc94d2cc81fa88517598061237cea0329d (diff)
Add QtNetworkAuth
Ported redditclient example. [ChangeLog][PySide6] The QtNetworkAuth module has been added. Task-number: PYSIDE-1570 Change-Id: I8a057870bf5a59cab227c271c412eb5b9ec4a7b8 Reviewed-by: Cristian Maureira-Fredes <cristian.maureira-fredes@qt.io>
Diffstat (limited to 'examples')
-rw-r--r--examples/networkauth/redditclient/doc/redditclient.pngbin0 -> 40526 bytes
-rw-r--r--examples/networkauth/redditclient/doc/redditclient.rst21
-rw-r--r--examples/networkauth/redditclient/main.py65
-rw-r--r--examples/networkauth/redditclient/redditclient.pyproject3
-rw-r--r--examples/networkauth/redditclient/redditmodel.py107
-rw-r--r--examples/networkauth/redditclient/redditwrapper.py111
6 files changed, 307 insertions, 0 deletions
diff --git a/examples/networkauth/redditclient/doc/redditclient.png b/examples/networkauth/redditclient/doc/redditclient.png
new file mode 100644
index 000000000..013450a9e
--- /dev/null
+++ b/examples/networkauth/redditclient/doc/redditclient.png
Binary files differ
diff --git a/examples/networkauth/redditclient/doc/redditclient.rst b/examples/networkauth/redditclient/doc/redditclient.rst
new file mode 100644
index 000000000..48b720abb
--- /dev/null
+++ b/examples/networkauth/redditclient/doc/redditclient.rst
@@ -0,0 +1,21 @@
+Reddit Example
+==============
+
+Demonstrates authenticating with OAuth 2 to access Reddit.
+
+The Reddit example uses OAuth 2, as supported by Qt Network Authorization,
+to sign in to Reddit and display the Reddit posts (in text format) associated
+with the authenticated user.
+
+To use this example, a consumer key from Reddit is needed.
+To register the application visit https://www.reddit.com/prefs/apps/.
+
+.. note::
+ Choose installed app when creating the application.
+
+.. note::
+ Set the redirect URI to http://127.0.0.1:1337/ in Reddit settings.
+
+.. image:: redditclient.png
+ :width: 400
+ :alt: Reddit Example Screenshot
diff --git a/examples/networkauth/redditclient/main.py b/examples/networkauth/redditclient/main.py
new file mode 100644
index 000000000..813c61964
--- /dev/null
+++ b/examples/networkauth/redditclient/main.py
@@ -0,0 +1,65 @@
+#############################################################################
+##
+## Copyright (C) 2021 The Qt Company Ltd.
+## Contact: http://www.qt.io/licensing/
+##
+## This file is part of the Qt for Python examples of the Qt Toolkit.
+##
+## $QT_BEGIN_LICENSE:BSD$
+## You may use this file under the terms of the BSD license as follows:
+##
+## "Redistribution and use in source and binary forms, with or without
+## modification, are permitted provided that the following conditions are
+## met:
+## * Redistributions of source code must retain the above copyright
+## notice, this list of conditions and the following disclaimer.
+## * Redistributions in binary form must reproduce the above copyright
+## notice, this list of conditions and the following disclaimer in
+## the documentation and/or other materials provided with the
+## distribution.
+## * Neither the name of The Qt Company Ltd nor the names of its
+## contributors may be used to endorse or promote products derived
+## from this software without specific prior written permission.
+##
+##
+## THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+## "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+## LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+## A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+## OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+## SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+## LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+## DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+## THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+## (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+## OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE."
+##
+## $QT_END_LICENSE$
+##
+#############################################################################
+
+"""PySide6 port of the networkauth redditclient example from Qt v6.x"""
+
+from argparse import ArgumentParser, RawTextHelpFormatter
+import sys
+
+from PySide6.QtWidgets import QApplication, QListView
+
+from redditmodel import RedditModel
+
+
+if __name__ == '__main__':
+ parser = ArgumentParser(description='Qt Reddit client example',
+ formatter_class=RawTextHelpFormatter)
+ parser.add_argument('--client', '-i', type=str, help='Client id')
+ options = parser.parse_args()
+ if not options.client:
+ print('Specify a client id', file=sys.stderr)
+ sys.exit(-1)
+
+ app = QApplication(sys.argv)
+ view = QListView()
+ model = RedditModel(options.client)
+ view.setModel(model)
+ view.show()
+ sys.exit(app.exec())
diff --git a/examples/networkauth/redditclient/redditclient.pyproject b/examples/networkauth/redditclient/redditclient.pyproject
new file mode 100644
index 000000000..6ac1969cc
--- /dev/null
+++ b/examples/networkauth/redditclient/redditclient.pyproject
@@ -0,0 +1,3 @@
+{
+ "files": ["main.py", "redditmodel.py", "redditwrapper.py"]
+}
diff --git a/examples/networkauth/redditclient/redditmodel.py b/examples/networkauth/redditclient/redditmodel.py
new file mode 100644
index 000000000..546532112
--- /dev/null
+++ b/examples/networkauth/redditclient/redditmodel.py
@@ -0,0 +1,107 @@
+#############################################################################
+##
+## Copyright (C) 2021 The Qt Company Ltd.
+## Contact: http://www.qt.io/licensing/
+##
+## This file is part of the Qt for Python examples of the Qt Toolkit.
+##
+## $QT_BEGIN_LICENSE:BSD$
+## You may use this file under the terms of the BSD license as follows:
+##
+## "Redistribution and use in source and binary forms, with or without
+## modification, are permitted provided that the following conditions are
+## met:
+## * Redistributions of source code must retain the above copyright
+## notice, this list of conditions and the following disclaimer.
+## * Redistributions in binary form must reproduce the above copyright
+## notice, this list of conditions and the following disclaimer in
+## the documentation and/or other materials provided with the
+## distribution.
+## * Neither the name of The Qt Company Ltd nor the names of its
+## contributors may be used to endorse or promote products derived
+## from this software without specific prior written permission.
+##
+##
+## THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+## "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+## LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+## A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+## OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+## SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+## LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+## DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+## THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+## (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+## OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE."
+##
+## $QT_END_LICENSE$
+##
+#############################################################################
+
+import functools
+from PySide6.QtCore import (QAbstractTableModel, QJsonDocument, QModelIndex,
+ Qt, Signal, Slot)
+from PySide6.QtNetwork import QNetworkReply
+
+from redditwrapper import RedditWrapper
+
+
+class RedditModel(QAbstractTableModel):
+
+ error = Signal(str)
+
+ def __init__(self, client_id):
+ super().__init__()
+ self._reddit_wrapper = RedditWrapper(client_id)
+ self._reddit_wrapper.authenticated.connect(self.update)
+ self._live_thread_reply = None
+ self._threads = []
+ self.grant()
+
+ def rowCount(self, parent):
+ return len(self._threads)
+
+ def columnCount(self, parent):
+ return 1 if self._threads else 0
+
+ def data(self, index, role):
+ if not index.isValid():
+ return None
+ if role == Qt.DisplayRole:
+ children_object = self._threads[index.row()]
+ data_object = children_object["data"]
+ return data_object["title"]
+ return None
+
+ def grant(self):
+ self._reddit_wrapper.grant()
+
+ @Slot(QNetworkReply)
+ def reply_finished(self, reply):
+ reply.deleteLater()
+ if reply.error() != QNetworkReply.NoError:
+ error = reply.errorString()
+ print(f"Reddit error: {error}")
+ self.error.emit(error)
+ return
+ json = reply.readAll()
+ document = QJsonDocument.fromJson(json)
+ root_object = document.object()
+ kind = root_object["kind"]
+ assert(kind == "Listing")
+ data_object = root_object["data"]
+ children_array = data_object["children"]
+ if not children_array:
+ return
+
+ self.beginInsertRows(QModelIndex(), len(self._threads),
+ len(children_array) + len(self._threads) - 1)
+ for childValue in children_array:
+ self._threads.append(childValue)
+ self.endInsertRows()
+
+ @Slot()
+ def update(self):
+ reply = self._reddit_wrapper.request_hot_threads()
+ reply.finished.connect(functools.partial(self.reply_finished,
+ reply=reply))
diff --git a/examples/networkauth/redditclient/redditwrapper.py b/examples/networkauth/redditclient/redditwrapper.py
new file mode 100644
index 000000000..e069fdeb0
--- /dev/null
+++ b/examples/networkauth/redditclient/redditwrapper.py
@@ -0,0 +1,111 @@
+#############################################################################
+##
+## Copyright (C) 2021 The Qt Company Ltd.
+## Contact: http://www.qt.io/licensing/
+##
+## This file is part of the Qt for Python examples of the Qt Toolkit.
+##
+## $QT_BEGIN_LICENSE:BSD$
+## You may use this file under the terms of the BSD license as follows:
+##
+## "Redistribution and use in source and binary forms, with or without
+## modification, are permitted provided that the following conditions are
+## met:
+## * Redistributions of source code must retain the above copyright
+## notice, this list of conditions and the following disclaimer.
+## * Redistributions in binary form must reproduce the above copyright
+## notice, this list of conditions and the following disclaimer in
+## the documentation and/or other materials provided with the
+## distribution.
+## * Neither the name of The Qt Company Ltd nor the names of its
+## contributors may be used to endorse or promote products derived
+## from this software without specific prior written permission.
+##
+##
+## THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+## "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+## LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+## A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+## OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+## SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+## LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+## DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+## THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+## (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+## OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE."
+##
+## $QT_END_LICENSE$
+##
+#############################################################################
+
+import functools
+
+from PySide6.QtCore import QJsonDocument, QObject, QUrl, Signal, Slot
+from PySide6.QtGui import QDesktopServices
+from PySide6.QtNetwork import QNetworkReply
+from PySide6.QtNetworkAuth import (QAbstractOAuth,
+ QOAuth2AuthorizationCodeFlow,
+ QOAuthHttpServerReplyHandler)
+
+
+AUTHORIZATION_URL = "https://www.reddit.com/api/v1/authorize"
+ACCESSTOKEN_URL = "https://www.reddit.com/api/v1/access_token"
+
+
+NEW_URL = "https://oauth.reddit.com/new"
+HOT_URL = "https://oauth.reddit.com/hot"
+LIVE_THREADS_URL = "https://oauth.reddit.com/live/XXXX/about.json"
+
+
+class RedditWrapper(QObject):
+
+ authenticated = Signal()
+ subscribed = Signal(QUrl)
+
+ def __init__(self, clientIdentifier, parent=None):
+ super().__init__(parent)
+
+ self._oauth2 = QOAuth2AuthorizationCodeFlow()
+ self._oauth2.statusChanged.connect(self.status_changed)
+ self._oauth2.authorizeWithBrowser.connect(QDesktopServices.openUrl)
+ self._oauth2.setClientIdentifier(clientIdentifier)
+ self._reply_handler = QOAuthHttpServerReplyHandler(1337, self)
+ self._oauth2.setReplyHandler(self._reply_handler)
+ self._oauth2.setAuthorizationUrl(QUrl(AUTHORIZATION_URL))
+ self._oauth2.setAccessTokenUrl(QUrl(ACCESSTOKEN_URL))
+ self._oauth2.setScope("identity read")
+
+ @Slot()
+ def status_changed(self, status):
+ if status == QAbstractOAuth.Status.Granted:
+ self.authenticated.emit()
+
+ def request_hot_threads(self):
+ print("Getting hot threads...")
+ return self._oauth2.get(QUrl(HOT_URL))
+
+ def grant(self):
+ self._oauth2.grant()
+
+ @Slot(QNetworkReply)
+ def reply_finished(self, reply):
+ print('RedditWrapper.reply_finished()', reply.error())
+ reply.deleteLater()
+ if reply.error() != QNetworkReply.NoError:
+ error = reply.errorString()
+ print(f"Reddit error: {error}")
+ return
+
+ json = reply.readAll()
+ document = QJsonDocument.fromJson(json)
+ assert(document.isObject())
+ root_object = document.object()
+ data_object = root_object["data"]
+ websocketUrl = QUrl(data_object["websocket_url"])
+ self.subscribed.emit(websocketUrl)
+
+ def subscribe_to_live_updates(self):
+ print("Susbscribing...")
+ reply = self._oauth2.get(QUrl(LIVE_THREADS_URL))
+ reply.finished.connect(functools.partial(self.reply_finished,
+ reply=reply))