/**************************************************************************** ** ** Copyright (C) 2021 The Qt Company Ltd. ** Contact: https://www.qt.io/licensing/ ** ** This file is part of the plugins of the Qt Toolkit. ** ** $QT_BEGIN_LICENSE:LGPL$ ** Commercial License Usage ** Licensees holding valid commercial Qt licenses may use this file in ** accordance with the commercial license agreement provided with the ** Software or, alternatively, in accordance with the terms contained in ** a written agreement between you and The Qt Company. For licensing terms ** and conditions see https://www.qt.io/terms-conditions. For further ** information use the contact form at https://www.qt.io/contact-us. ** ** GNU Lesser General Public License Usage ** Alternatively, this file may be used under the terms of the GNU Lesser ** General Public License version 3 as published by the Free Software ** Foundation and appearing in the file LICENSE.LGPL3 included in the ** packaging of this file. Please review the following information to ** ensure the GNU Lesser General Public License version 3 requirements ** will be met: https://www.gnu.org/licenses/lgpl-3.0.html. ** ** GNU General Public License Usage ** Alternatively, this file may be used under the terms of the GNU ** General Public License version 2.0 or (at your option) the GNU General ** Public license version 3 or any later version approved by the KDE Free ** Qt Foundation. The licenses are as published by the Free Software ** Foundation and appearing in the file LICENSE.GPL2 and LICENSE.GPL3 ** included in the packaging of this file. Please review the following ** information to ensure the GNU General Public License requirements will ** be met: https://www.gnu.org/licenses/gpl-2.0.html and ** https://www.gnu.org/licenses/gpl-3.0.html. ** ** $QT_END_LICENSE$ ** ****************************************************************************/ #include "qnetworkmanagerservice.h" #include #include #include #include #include #include #include #include #include #include #define DBUS_PROPERTIES_INTERFACE "org.freedesktop.DBus.Properties" #define NM_DBUS_SERVICE "org.freedesktop.NetworkManager" #define NM_DBUS_PATH "/org/freedesktop/NetworkManager" #define NM_DBUS_INTERFACE NM_DBUS_SERVICE #define NM_CONNECTION_DBUS_INTERFACE NM_DBUS_SERVICE ".Connection.Active" #define NM_DEVICE_DBUS_INTERFACE NM_DBUS_SERVICE ".Device" QT_BEGIN_NAMESPACE QNetworkManagerInterfaceBase::QNetworkManagerInterfaceBase(QObject *parent) : QDBusAbstractInterface(QLatin1String(NM_DBUS_SERVICE), QLatin1String(NM_DBUS_PATH), NM_DBUS_INTERFACE, QDBusConnection::systemBus(), parent) { } bool QNetworkManagerInterfaceBase::networkManagerAvailable() { return QNetworkManagerInterfaceBase().isValid(); } QNetworkManagerInterface::QNetworkManagerInterface(QObject *parent) : QNetworkManagerInterfaceBase(parent) { if (!isValid()) return; PropertiesDBusInterface managerPropertiesInterface( QLatin1String(NM_DBUS_SERVICE), QLatin1String(NM_DBUS_PATH), DBUS_PROPERTIES_INTERFACE, QDBusConnection::systemBus()); QList argumentList; argumentList << QLatin1String(NM_DBUS_INTERFACE); QDBusPendingReply propsReply = managerPropertiesInterface.callWithArgumentList( QDBus::Block, QLatin1String("GetAll"), argumentList); if (!propsReply.isError()) { propertyMap = propsReply.value(); } else { qWarning() << "propsReply" << propsReply.error().message(); } QDBusConnection::systemBus().connect( QLatin1String(NM_DBUS_SERVICE), QLatin1String(NM_DBUS_PATH), QLatin1String(DBUS_PROPERTIES_INTERFACE), QLatin1String("PropertiesChanged"), this, SLOT(setProperties(QString, QMap, QList))); } QNetworkManagerInterface::~QNetworkManagerInterface() { QDBusConnection::systemBus().disconnect( QLatin1String(NM_DBUS_SERVICE), QLatin1String(NM_DBUS_PATH), QLatin1String(DBUS_PROPERTIES_INTERFACE), QLatin1String("PropertiesChanged"), this, SLOT(setProperties(QString, QMap, QList))); } QNetworkManagerInterface::NMState QNetworkManagerInterface::state() const { if (propertyMap.contains("State")) return static_cast(propertyMap.value("State").toUInt()); return QNetworkManagerInterface::NM_STATE_UNKNOWN; } QNetworkManagerInterface::NMConnectivityState QNetworkManagerInterface::connectivityState() const { if (propertyMap.contains("Connectivity")) return static_cast(propertyMap.value("Connectivity").toUInt()); return QNetworkManagerInterface::NM_CONNECTIVITY_UNKNOWN; } static QDBusInterface getPrimaryDevice(const QDBusObjectPath &devicePath) { const QDBusInterface connection(NM_DBUS_SERVICE, devicePath.path(), NM_CONNECTION_DBUS_INTERFACE, QDBusConnection::systemBus()); if (!connection.isValid()) return QDBusInterface({}, {}); const auto devicePaths = connection.property("Devices").value>(); if (devicePaths.isEmpty()) return QDBusInterface({}, {}); const QDBusObjectPath primaryDevicePath = devicePaths.front(); return QDBusInterface(NM_DBUS_SERVICE, primaryDevicePath.path(), NM_DEVICE_DBUS_INTERFACE, QDBusConnection::systemBus()); } std::optional QNetworkManagerInterface::primaryConnectionDevicePath() const { auto it = propertyMap.constFind(u"PrimaryConnection"_qs); if (it != propertyMap.cend()) return it->value(); return std::nullopt; } auto QNetworkManagerInterface::deviceType() const -> NMDeviceType { if (const auto path = primaryConnectionDevicePath()) return extractDeviceType(*path); return NM_DEVICE_TYPE_UNKNOWN; } auto QNetworkManagerInterface::meteredState() const -> NMMetered { if (const auto path = primaryConnectionDevicePath()) return extractDeviceMetered(*path); return NM_METERED_UNKNOWN; } auto QNetworkManagerInterface::extractDeviceType(const QDBusObjectPath &devicePath) const -> NMDeviceType { QDBusInterface primaryDevice = getPrimaryDevice(devicePath); if (!primaryDevice.isValid()) return NM_DEVICE_TYPE_UNKNOWN; const QVariant deviceType = primaryDevice.property("DeviceType"); if (!deviceType.isValid()) return NM_DEVICE_TYPE_UNKNOWN; return static_cast(deviceType.toUInt()); } auto QNetworkManagerInterface::extractDeviceMetered(const QDBusObjectPath &devicePath) const -> NMMetered { QDBusInterface primaryDevice = getPrimaryDevice(devicePath); if (!primaryDevice.isValid()) return NM_METERED_UNKNOWN; const QVariant metered = primaryDevice.property("Metered"); if (!metered.isValid()) return NM_METERED_UNKNOWN; return static_cast(metered.toUInt()); } void QNetworkManagerInterface::setProperties(const QString &interfaceName, const QMap &map, const QStringList &invalidatedProperties) { Q_UNUSED(interfaceName); Q_UNUSED(invalidatedProperties); for (auto i = map.cbegin(), end = map.cend(); i != end; ++i) { bool valueChanged = true; auto it = propertyMap.lowerBound(i.key()); if (it != propertyMap.end() && it.key() == i.key()) { valueChanged = (it.value() != i.value()); *it = *i; } else { propertyMap.insert(it, i.key(), i.value()); } if (valueChanged) { if (i.key() == QLatin1String("State")) { quint32 state = i.value().toUInt(); Q_EMIT stateChanged(static_cast(state)); } else if (i.key() == QLatin1String("Connectivity")) { quint32 state = i.value().toUInt(); Q_EMIT connectivityChanged(static_cast(state)); } else if (i.key() == QLatin1String("PrimaryConnection")) { const QDBusObjectPath devicePath = i->value(); Q_EMIT deviceTypeChanged(extractDeviceType(devicePath)); Q_EMIT meteredChanged(extractDeviceMetered(devicePath)); } else if (i.key() == QLatin1String("Metered")) { Q_EMIT meteredChanged(static_cast(i->toUInt())); } } } } QT_END_NAMESPACE