mirror of
				https://github.com/f4exb/sdrangel.git
				synced 2025-10-22 16:40:21 -04:00 
			
		
		
		
	
		
			
	
	
		
			246 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
		
		
			
		
	
	
			246 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
|  | ///////////////////////////////////////////////////////////////////////////////////
 | ||
|  | // Copyright (C) 2021 Jon Beniston, M7RCE                                        //
 | ||
|  | //                                                                               //
 | ||
|  | // This program is free software; you can redistribute it and/or modify          //
 | ||
|  | // it under the terms of the GNU General Public License as published by          //
 | ||
|  | // the Free Software Foundation as version 3 of the License, or                  //
 | ||
|  | // (at your option) any later version.                                           //
 | ||
|  | //                                                                               //
 | ||
|  | // This program is distributed in the hope that it will be useful,               //
 | ||
|  | // but WITHOUT ANY WARRANTY; without even the implied warranty of                //
 | ||
|  | // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the                  //
 | ||
|  | // GNU General Public License V3 for more details.                               //
 | ||
|  | //                                                                               //
 | ||
|  | // You should have received a copy of the GNU General Public License             //
 | ||
|  | // along with this program. If not, see <http://www.gnu.org/licenses/>.          //
 | ||
|  | ///////////////////////////////////////////////////////////////////////////////////
 | ||
|  | 
 | ||
|  | #include "aviationweather.h"
 | ||
|  | 
 | ||
|  | #include <QDebug>
 | ||
|  | #include <QUrl>
 | ||
|  | #include <QUrlQuery>
 | ||
|  | #include <QNetworkReply>
 | ||
|  | #include <QJsonDocument>
 | ||
|  | #include <QJsonObject>
 | ||
|  | 
 | ||
|  | AviationWeather::AviationWeather() | ||
|  | { | ||
|  |     connect(&m_timer, &QTimer::timeout, this, &AviationWeather::update); | ||
|  | } | ||
|  | 
 | ||
|  | AviationWeather* AviationWeather::create(const QString& apiKey, const QString& service) | ||
|  | { | ||
|  |     if (service == "checkwxapi.com") | ||
|  |     { | ||
|  |         if (!apiKey.isEmpty()) | ||
|  |         { | ||
|  |             return new CheckWXAPI(apiKey); | ||
|  |         } | ||
|  |         else | ||
|  |         { | ||
|  |             qDebug() << "AviationWeather::create: An API key is required for: " << service; | ||
|  |             return nullptr; | ||
|  |         } | ||
|  |     } | ||
|  |     else | ||
|  |     { | ||
|  |         qDebug() << "AviationWeather::create: Unsupported service: " << service; | ||
|  |         return nullptr; | ||
|  |     } | ||
|  | } | ||
|  | 
 | ||
|  | void AviationWeather::getWeatherPeriodically(const QString &icao, int periodInMins) | ||
|  | { | ||
|  |     m_icao = icao; | ||
|  |     m_timer.setInterval(periodInMins*60*1000); | ||
|  |     m_timer.start(); | ||
|  |     update(); | ||
|  | } | ||
|  | 
 | ||
|  | void AviationWeather::update() | ||
|  | { | ||
|  |     getWeather(m_icao); | ||
|  | } | ||
|  | 
 | ||
|  | CheckWXAPI::CheckWXAPI(const QString& apiKey) : | ||
|  |     m_apiKey(apiKey) | ||
|  | { | ||
|  |     m_networkManager = new QNetworkAccessManager(); | ||
|  |     QObject::connect( | ||
|  |         m_networkManager, | ||
|  |         &QNetworkAccessManager::finished, | ||
|  |         this, | ||
|  |         &CheckWXAPI::handleReply | ||
|  |     ); | ||
|  | } | ||
|  | 
 | ||
|  | CheckWXAPI::~CheckWXAPI() | ||
|  | { | ||
|  |     QObject::disconnect( | ||
|  |         m_networkManager, | ||
|  |         &QNetworkAccessManager::finished, | ||
|  |         this, | ||
|  |         &CheckWXAPI::handleReply | ||
|  |     ); | ||
|  |     delete m_networkManager; | ||
|  | } | ||
|  | 
 | ||
|  | void CheckWXAPI::getWeather(const QString &icao) | ||
|  | { | ||
|  |     QUrl url(QString("https://api.checkwx.com/metar/%1/decoded").arg(icao)); | ||
|  |     QNetworkRequest req(url); | ||
|  |     req.setRawHeader(QByteArray("X-API-Key"), m_apiKey.toUtf8()); | ||
|  | 
 | ||
|  |     m_networkManager->get(req); | ||
|  | } | ||
|  | 
 | ||
|  | void CheckWXAPI::handleReply(QNetworkReply* reply) | ||
|  | { | ||
|  |     if (reply) | ||
|  |     { | ||
|  |         if (!reply->error()) | ||
|  |         { | ||
|  |             QJsonDocument document = QJsonDocument::fromJson(reply->readAll()); | ||
|  |             if (document.isObject()) | ||
|  |             { | ||
|  |                 QJsonObject obj = document.object(); | ||
|  |                 if (obj.contains(QStringLiteral("data"))) | ||
|  |                 { | ||
|  |                     QJsonValue val = obj.value(QStringLiteral("data")); | ||
|  |                     if (val.isArray()) | ||
|  |                     { | ||
|  |                         QJsonArray array = val.toArray(); | ||
|  |                         for (auto mainObjRef : array) | ||
|  |                         { | ||
|  |                             QJsonObject mainObj = mainObjRef.toObject(); | ||
|  |                             METAR metar; | ||
|  | 
 | ||
|  |                             if (mainObj.contains(QStringLiteral("icao"))) { | ||
|  |                                metar.m_icao = mainObj.value(QStringLiteral("icao")).toString(); | ||
|  |                             } | ||
|  | 
 | ||
|  |                             if (mainObj.contains(QStringLiteral("raw_text"))) { | ||
|  |                                metar.m_text = mainObj.value(QStringLiteral("raw_text")).toString(); | ||
|  |                             } | ||
|  | 
 | ||
|  |                             if (mainObj.contains(QStringLiteral("observed"))) { | ||
|  |                                metar.m_dateTime = QDateTime::fromString(mainObj.value(QStringLiteral("observed")).toString(), Qt::ISODate); | ||
|  |                             } | ||
|  | 
 | ||
|  |                             if (mainObj.contains(QStringLiteral("wind"))) { | ||
|  |                                 QJsonObject windObj = mainObj.value(QStringLiteral("wind")).toObject(); | ||
|  |                                 if (windObj.contains(QStringLiteral("degrees"))) { | ||
|  |                                     metar.m_windDirection = windObj.value(QStringLiteral("degrees")).toDouble(); | ||
|  |                                 } | ||
|  |                                 if (windObj.contains(QStringLiteral("speed_kts"))) { | ||
|  |                                     metar.m_windSpeed = windObj.value(QStringLiteral("speed_kts")).toDouble(); | ||
|  |                                 } | ||
|  |                                 if (windObj.contains(QStringLiteral("wind.gust_kts"))) { | ||
|  |                                     metar.m_windGusts = windObj.value(QStringLiteral("wind.gust_kts")).toDouble(); | ||
|  |                                 } | ||
|  |                             } | ||
|  |                             if (mainObj.contains(QStringLiteral("visibility"))) { | ||
|  |                                 QJsonObject visibiltyObj = mainObj.value(QStringLiteral("visibility")).toObject(); | ||
|  |                                 if (visibiltyObj.contains(QStringLiteral("meters"))) { | ||
|  |                                     metar.m_visibility = visibiltyObj.value(QStringLiteral("meters")).toString(); | ||
|  |                                 } | ||
|  |                             } | ||
|  | 
 | ||
|  |                             if (mainObj.contains(QStringLiteral("conditions"))) { | ||
|  |                                 QJsonArray conditions = mainObj.value(QStringLiteral("conditions")).toArray(); | ||
|  |                                 for (auto condition : conditions) { | ||
|  |                                     QJsonObject conditionObj = condition.toObject(); | ||
|  |                                     if (conditionObj.contains(QStringLiteral("text"))) { | ||
|  |                                         metar.m_conditions.append(conditionObj.value(QStringLiteral("text")).toString()); | ||
|  |                                     } | ||
|  |                                 } | ||
|  |                             } | ||
|  | 
 | ||
|  |                             if (mainObj.contains(QStringLiteral("ceiling"))) { | ||
|  |                                 QJsonObject ceilingObj = mainObj.value(QStringLiteral("ceiling")).toObject(); | ||
|  |                                 if (ceilingObj.contains(QStringLiteral("feet"))) { | ||
|  |                                     metar.m_ceiling = ceilingObj.value(QStringLiteral("feet")).toDouble(); | ||
|  |                                 } | ||
|  |                             } | ||
|  | 
 | ||
|  |                             if (mainObj.contains(QStringLiteral("clouds"))) { | ||
|  |                                 QJsonArray clouds = mainObj.value(QStringLiteral("clouds")).toArray(); | ||
|  |                                 for (auto cloud : clouds) { | ||
|  |                                     QJsonObject cloudObj = cloud.toObject(); | ||
|  |                                     // "Clear skies" doesn't have an altitude
 | ||
|  |                                     if (cloudObj.contains(QStringLiteral("text")) && cloudObj.contains(QStringLiteral("feet"))) { | ||
|  |                                         metar.m_clouds.append(QString("%1 %2 ft").arg(cloudObj.value(QStringLiteral("text")).toString()) | ||
|  |                                                                                  .arg(cloudObj.value(QStringLiteral("feet")).toDouble())); | ||
|  |                                     } else if (cloudObj.contains(QStringLiteral("text"))) { | ||
|  |                                         metar.m_clouds.append(cloudObj.value(QStringLiteral("text")).toString()); | ||
|  |                                     } | ||
|  |                                 } | ||
|  |                             } | ||
|  | 
 | ||
|  |                             if (mainObj.contains(QStringLiteral("temperature"))) { | ||
|  |                                 QJsonObject tempObj = mainObj.value(QStringLiteral("temperature")).toObject(); | ||
|  |                                 if (tempObj.contains(QStringLiteral("celsius"))) { | ||
|  |                                     metar.m_temperature = tempObj.value(QStringLiteral("celsius")).toDouble(); | ||
|  |                                 } | ||
|  |                             } | ||
|  | 
 | ||
|  |                             if (mainObj.contains(QStringLiteral("dewpoint"))) { | ||
|  |                                 QJsonObject dewpointObj = mainObj.value(QStringLiteral("dewpoint")).toObject(); | ||
|  |                                 if (dewpointObj.contains(QStringLiteral("celsius"))) { | ||
|  |                                     metar.m_dewpoint = dewpointObj.value(QStringLiteral("celsius")).toDouble(); | ||
|  |                                 } | ||
|  |                             } | ||
|  | 
 | ||
|  |                             if (mainObj.contains(QStringLiteral("barometer"))) { | ||
|  |                                 QJsonObject pressureObj = mainObj.value(QStringLiteral("barometer")).toObject(); | ||
|  |                                 if (pressureObj.contains(QStringLiteral("hpa"))) { | ||
|  |                                     metar.m_pressure = pressureObj.value(QStringLiteral("hpa")).toDouble(); | ||
|  |                                 } | ||
|  |                             } | ||
|  | 
 | ||
|  |                             if (mainObj.contains(QStringLiteral("humidity"))) { | ||
|  |                                 QJsonObject humidityObj = mainObj.value(QStringLiteral("humidity")).toObject(); | ||
|  |                                 if (humidityObj.contains(QStringLiteral("percent"))) { | ||
|  |                                     metar.m_humidity = humidityObj.value(QStringLiteral("percent")).toDouble(); | ||
|  |                                 } | ||
|  |                             } | ||
|  | 
 | ||
|  |                             if (mainObj.contains(QStringLiteral("flight_category"))) { | ||
|  |                                metar.m_flightCateogory = mainObj.value(QStringLiteral("flight_category")).toString(); | ||
|  |                             } | ||
|  | 
 | ||
|  |                             if (!metar.m_icao.isEmpty()) { | ||
|  |                                 emit weatherUpdated(metar); | ||
|  |                             } else { | ||
|  |                                 qDebug() << "CheckWXAPI::handleReply: object doesn't contain icao: " << mainObj; | ||
|  |                             } | ||
|  |                         } | ||
|  |                     } | ||
|  |                     else | ||
|  |                     { | ||
|  |                         qDebug() << "CheckWXAPI::handleReply: data isn't an array: " << obj; | ||
|  |                     } | ||
|  |                 } | ||
|  |                 else | ||
|  |                 { | ||
|  |                     qDebug() << "CheckWXAPI::handleReply: Object doesn't contain data: " << obj; | ||
|  |                 } | ||
|  |             } | ||
|  |             else | ||
|  |             { | ||
|  |                 qDebug() << "CheckWXAPI::handleReply: Document is not an object: " << document; | ||
|  |             } | ||
|  |         } | ||
|  |         else | ||
|  |         { | ||
|  |             qDebug() << "CheckWXAPI::handleReply: error: " << reply->error(); | ||
|  |         } | ||
|  |         reply->deleteLater(); | ||
|  |     } | ||
|  |     else | ||
|  |     { | ||
|  |         qDebug() << "CheckWXAPI::handleReply: reply is null"; | ||
|  |     } | ||
|  | } |