mirror of
				https://github.com/f4exb/sdrangel.git
				synced 2025-10-26 02:20:26 -04:00 
			
		
		
		
	
		
			
	
	
		
			420 lines
		
	
	
		
			15 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
		
		
			
		
	
	
			420 lines
		
	
	
		
			15 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
|  | ///////////////////////////////////////////////////////////////////////////////////
 | ||
|  | // Copyright (C) 2020 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 <QFile>
 | ||
|  | #include <QFileInfo>
 | ||
|  | #include <QByteArray>
 | ||
|  | #include <QList>
 | ||
|  | #include <QDebug>
 | ||
|  | #include <QLocale>
 | ||
|  | #include <QStandardPaths>
 | ||
|  | 
 | ||
|  | #include "util/ourairportsdb.h"
 | ||
|  | 
 | ||
|  | QMutex OurAirportsDB::m_mutex; | ||
|  | QSharedPointer<QHash<int, AirportInformation *>> OurAirportsDB::m_airportsById; | ||
|  | QSharedPointer<QHash<QString, AirportInformation *>> OurAirportsDB::m_airportsByIdent; | ||
|  | QDateTime OurAirportsDB::m_modifiedDateTime; | ||
|  | 
 | ||
|  | AirportInformation::~AirportInformation() | ||
|  | { | ||
|  |     qDeleteAll(m_frequencies); | ||
|  | } | ||
|  | 
 | ||
|  | QString AirportInformation::getImageName() const | ||
|  | { | ||
|  |     switch (m_type) | ||
|  |     { | ||
|  |     case AirportType::Large: | ||
|  |         return "airport_large.png"; | ||
|  |     case AirportType::Medium: | ||
|  |         return "airport_medium.png"; | ||
|  |     case AirportType::Heliport: | ||
|  |         return "heliport.png"; | ||
|  |     default: | ||
|  |         return "airport_small.png"; | ||
|  |     } | ||
|  | } | ||
|  | 
 | ||
|  | OurAirportsDB::OurAirportsDB(QObject *parent) : | ||
|  |     QObject(parent) | ||
|  | { | ||
|  |     connect(&m_dlm, &HttpDownloadManager::downloadComplete, this, &OurAirportsDB::downloadFinished); | ||
|  | } | ||
|  | 
 | ||
|  | OurAirportsDB::~OurAirportsDB() | ||
|  | { | ||
|  |     disconnect(&m_dlm, &HttpDownloadManager::downloadComplete, this, &OurAirportsDB::downloadFinished); | ||
|  | } | ||
|  | 
 | ||
|  | void OurAirportsDB::downloadAirportInformation() | ||
|  | { | ||
|  |     // Download airport database
 | ||
|  |     QString urlString = AIRPORTS_URL; | ||
|  |     QUrl dbURL(urlString); | ||
|  |     qDebug() << "OurAirportsDB::downloadAirportInformation: Downloading " << urlString; | ||
|  |     emit downloadingURL(urlString); | ||
|  |     QNetworkReply *reply = m_dlm.download(dbURL, OurAirportsDB::getAirportDBFilename()); | ||
|  |     connect(reply, &QNetworkReply::downloadProgress, this, [this](qint64 bytesRead, qint64 totalBytes) { | ||
|  |         emit downloadProgress(bytesRead, totalBytes); | ||
|  |     }); | ||
|  | } | ||
|  | 
 | ||
|  | void OurAirportsDB::downloadFinished(const QString& filename, bool success) | ||
|  | { | ||
|  |     if (!success) | ||
|  |     { | ||
|  |         qWarning() << "OurAirportsDB::downloadFinished: Failed to download: " << filename; | ||
|  |         emit downloadError(QString("Failed to download: %1").arg(filename)); | ||
|  |     } | ||
|  |     else if (filename == OurAirportsDB::getAirportDBFilename()) | ||
|  |     { | ||
|  |         // Now download airport frequencies
 | ||
|  |         QString urlString = AIRPORT_FREQUENCIES_URL; | ||
|  |         QUrl dbURL(urlString); | ||
|  |         qDebug() << "OurAirportsDB::downloadFinished: Downloading " << urlString; | ||
|  |         emit downloadingURL(urlString); | ||
|  |         QNetworkReply *reply = m_dlm.download(dbURL, OurAirportsDB::getAirportFrequenciesDBFilename()); | ||
|  |         connect(reply, &QNetworkReply::downloadProgress, this, [this](qint64 bytesRead, qint64 totalBytes) { | ||
|  |             emit downloadProgress(bytesRead, totalBytes); | ||
|  |         }); | ||
|  |     } | ||
|  |     else if (filename == OurAirportsDB::getAirportFrequenciesDBFilename()) | ||
|  |     { | ||
|  |         emit downloadAirportInformationFinished(); | ||
|  |     } | ||
|  |     else | ||
|  |     { | ||
|  |         qDebug() << "OurAirportsDB::downloadFinished: Unexpected filename: " << filename; | ||
|  |         emit downloadError(QString("Unexpected filename: %1").arg(filename)); | ||
|  |     } | ||
|  | } | ||
|  | 
 | ||
|  | QSharedPointer<const QHash<int, AirportInformation *>> OurAirportsDB::getAirportsById() | ||
|  | { | ||
|  |     QMutexLocker locker(&m_mutex); | ||
|  | 
 | ||
|  |     readDB(); | ||
|  |     return m_airportsById; | ||
|  | } | ||
|  | 
 | ||
|  | QSharedPointer<const QHash<QString, AirportInformation *>> OurAirportsDB::getAirportsByIdent() | ||
|  | { | ||
|  |     QMutexLocker locker(&m_mutex); | ||
|  | 
 | ||
|  |     readDB(); | ||
|  |     return m_airportsByIdent; | ||
|  | } | ||
|  | 
 | ||
|  | void OurAirportsDB::readDB() | ||
|  | { | ||
|  |     QFileInfo airportDBFileInfo(getAirportDBFilename()); | ||
|  |     QDateTime airportDBModifiedDateTime = airportDBFileInfo.lastModified(); | ||
|  | 
 | ||
|  |     if (!m_airportsById || (airportDBModifiedDateTime > m_modifiedDateTime)) | ||
|  |     { | ||
|  |         // Using shared pointer, so old object, if it exists, will be deleted, when no longer user
 | ||
|  |         m_airportsById = QSharedPointer<QHash<int, AirportInformation *>>(OurAirportsDB::readAirportsDB(getAirportDBFilename())); | ||
|  |         if (m_airportsById != nullptr) | ||
|  |         { | ||
|  |             OurAirportsDB::readFrequenciesDB(getAirportFrequenciesDBFilename(), m_airportsById.get()); | ||
|  |             m_airportsByIdent = QSharedPointer<QHash<QString, AirportInformation *>>(identHash(m_airportsById.get())); | ||
|  |         } | ||
|  | 
 | ||
|  |         m_modifiedDateTime = airportDBModifiedDateTime; | ||
|  |     } | ||
|  | } | ||
|  | 
 | ||
|  | QString OurAirportsDB::getDataDir() | ||
|  | { | ||
|  |     // Get directory to store app data in
 | ||
|  |     QStringList locations = QStandardPaths::standardLocations(QStandardPaths::AppDataLocation); | ||
|  |     // First dir is writable
 | ||
|  |     return locations[0]; | ||
|  | } | ||
|  | 
 | ||
|  | QString OurAirportsDB::getAirportDBFilename() | ||
|  | { | ||
|  |     return getDataDir() + "/airportDatabase.csv"; | ||
|  | } | ||
|  | 
 | ||
|  | QString OurAirportsDB::getAirportFrequenciesDBFilename() | ||
|  | { | ||
|  |     return getDataDir() + "/airportFrequenciesDatabase.csv"; | ||
|  | } | ||
|  | 
 | ||
|  | QString OurAirportsDB::trimQuotes(const QString s) | ||
|  | { | ||
|  |     if (s.startsWith('\"') && s.endsWith('\"')) { | ||
|  |         return s.mid(1, s.size() - 2); | ||
|  |     } else { | ||
|  |         return s; | ||
|  |     } | ||
|  | } | ||
|  | 
 | ||
|  | // Read OurAirport's airport CSV file
 | ||
|  | // See comments for readOSNDB
 | ||
|  | QHash<int, AirportInformation *> *OurAirportsDB::readAirportsDB(const QString &filename) | ||
|  | { | ||
|  |     int cnt = 0; | ||
|  |     QHash<int, AirportInformation *> *airportInfo = nullptr; | ||
|  | 
 | ||
|  |     // Column numbers used for the data as of 2020/10/28
 | ||
|  |     int idCol = 0; | ||
|  |     int identCol = 1; | ||
|  |     int typeCol = 2; | ||
|  |     int nameCol = 3; | ||
|  |     int latitudeCol = 4; | ||
|  |     int longitudeCol = 5; | ||
|  |     int elevationCol = 6; | ||
|  | 
 | ||
|  |     qDebug() << "OurAirportsDB::readAirportsDB: " << filename; | ||
|  | 
 | ||
|  |     FILE *file; | ||
|  |     QByteArray utfFilename = filename.toUtf8(); | ||
|  |     QLocale cLocale(QLocale::C); | ||
|  |     if ((file = fopen(utfFilename.constData(), "r")) != NULL) | ||
|  |     { | ||
|  |         char row[2048]; | ||
|  | 
 | ||
|  |         if (fgets(row, sizeof(row), file)) | ||
|  |         { | ||
|  |             airportInfo = new QHash<int, AirportInformation *>(); | ||
|  |             airportInfo->reserve(70000); | ||
|  | 
 | ||
|  |             // Read header
 | ||
|  |             int idx = 0; | ||
|  |             char *p = strtok(row, ","); | ||
|  |             while (p != NULL) | ||
|  |             { | ||
|  |                 if (!strcmp(p, "id")) | ||
|  |                     idCol = idx; | ||
|  |                 else if (!strcmp(p, "ident")) | ||
|  |                     identCol = idx; | ||
|  |                 else if (!strcmp(p, "type")) | ||
|  |                     typeCol = idx; | ||
|  |                 else if (!strcmp(p, "name")) | ||
|  |                     nameCol = idx; | ||
|  |                 else if (!strcmp(p, "latitude_deg")) | ||
|  |                     latitudeCol = idx; | ||
|  |                 else if (!strcmp(p, "longitude_deg")) | ||
|  |                     longitudeCol = idx; | ||
|  |                 else if (!strcmp(p, "elevation_ft")) | ||
|  |                     elevationCol = idx; | ||
|  |                 p = strtok(NULL, ","); | ||
|  |                 idx++; | ||
|  |             } | ||
|  |             // Read data
 | ||
|  |             while (fgets(row, sizeof(row), file)) | ||
|  |             { | ||
|  |                 int id = 0; | ||
|  |                 char *idString = NULL; | ||
|  |                 char *ident = NULL; | ||
|  |                 size_t identLen = 0; | ||
|  |                 char *type = NULL; | ||
|  |                 size_t typeLen = 0; | ||
|  |                 char *name = NULL; | ||
|  |                 size_t nameLen = 0; | ||
|  |                 float latitude = 0.0f; | ||
|  |                 char *latitudeString = NULL; | ||
|  |                 size_t latitudeLen = 0; | ||
|  |                 float longitude = 0.0f; | ||
|  |                 char *longitudeString = NULL; | ||
|  |                 size_t longitudeLen = 0; | ||
|  |                 float elevation = 0.0f; | ||
|  |                 char *elevationString = NULL; | ||
|  |                 size_t elevationLen = 0; | ||
|  | 
 | ||
|  |                 p = strtok(row, ","); | ||
|  |                 idx = 0; | ||
|  |                 while (p != NULL) | ||
|  |                 { | ||
|  |                     // Read strings, stripping quotes
 | ||
|  |                     if (idx == idCol) | ||
|  |                     { | ||
|  |                         idString = p; | ||
|  |                         idString[strlen(idString)] = '\0'; | ||
|  |                         id = strtol(idString, NULL, 10); | ||
|  |                     } | ||
|  |                     else if (idx == identCol) | ||
|  |                     { | ||
|  |                         ident = p+1; | ||
|  |                         identLen = strlen(ident)-1; | ||
|  |                         ident[identLen] = '\0'; | ||
|  |                     } | ||
|  |                     else if (idx == typeCol) | ||
|  |                     { | ||
|  |                         type = p+1; | ||
|  |                         typeLen = strlen(type)-1; | ||
|  |                         type[typeLen] = '\0'; | ||
|  |                     } | ||
|  |                     else if (idx == nameCol) | ||
|  |                     { | ||
|  |                         name = p+1; | ||
|  |                         nameLen = strlen(name)-1; | ||
|  |                         name[nameLen] = '\0'; | ||
|  |                     } | ||
|  |                     else if (idx == latitudeCol) | ||
|  |                     { | ||
|  |                         latitudeString = p; | ||
|  |                         latitudeLen = strlen(latitudeString)-1; | ||
|  |                         latitudeString[latitudeLen] = '\0'; | ||
|  |                         latitude = cLocale.toFloat(latitudeString); | ||
|  |                     } | ||
|  |                     else if (idx == longitudeCol) | ||
|  |                     { | ||
|  |                         longitudeString = p; | ||
|  |                         longitudeLen = strlen(longitudeString)-1; | ||
|  |                         longitudeString[longitudeLen] = '\0'; | ||
|  |                         longitude = cLocale.toFloat(longitudeString); | ||
|  |                     } | ||
|  |                     else if (idx == elevationCol) | ||
|  |                     { | ||
|  |                         elevationString = p; | ||
|  |                         elevationLen = strlen(elevationString)-1; | ||
|  |                         elevationString[elevationLen] = '\0'; | ||
|  |                         elevation = cLocale.toFloat(elevationString); | ||
|  |                     } | ||
|  |                     p = strtok(NULL, ","); | ||
|  |                     idx++; | ||
|  |                 } | ||
|  | 
 | ||
|  |                 // Only create the entry if we have some interesting data
 | ||
|  |                 if (((latitude != 0.0f) || (longitude != 0.0f)) && (type && strcmp(type, "closed"))) | ||
|  |                 { | ||
|  |                     AirportInformation *airport = new AirportInformation(); | ||
|  |                     airport->m_id = id; | ||
|  |                     airport->m_ident = QString(ident); | ||
|  |                     if (!strcmp(type, "small_airport")) { | ||
|  |                         airport->m_type = AirportInformation::AirportType::Small; | ||
|  |                     } else if (!strcmp(type, "medium_airport")) { | ||
|  |                         airport->m_type = AirportInformation::AirportType::Medium; | ||
|  |                     } else if (!strcmp(type, "large_airport")) { | ||
|  |                         airport->m_type = AirportInformation::AirportType::Large; | ||
|  |                     } else if (!strcmp(type, "heliport")) { | ||
|  |                         airport->m_type = AirportInformation::AirportType::Heliport; | ||
|  |                     } | ||
|  |                     airport->m_name = QString(name); | ||
|  |                     airport->m_latitude = latitude; | ||
|  |                     airport->m_longitude = longitude; | ||
|  |                     airport->m_elevation = elevation; | ||
|  |                     airportInfo->insert(id, airport); | ||
|  |                     cnt++; | ||
|  |                 } | ||
|  |             } | ||
|  |         } | ||
|  |         fclose(file); | ||
|  |     } | ||
|  |     else | ||
|  |         qDebug() << "OurAirportsDB::readAirportsDB: Failed to open " << filename; | ||
|  | 
 | ||
|  |     qDebug() << "OurAirportsDB::readAirportsDB: Read " << cnt << " airports"; | ||
|  | 
 | ||
|  |     return airportInfo; | ||
|  | } | ||
|  | 
 | ||
|  | // Create hash table using ICAO identifier as key
 | ||
|  | QHash<QString, AirportInformation *> *OurAirportsDB::identHash(QHash<int, AirportInformation *> *in) | ||
|  | { | ||
|  |     QHash<QString, AirportInformation *> *out = new QHash<QString, AirportInformation *>(); | ||
|  |     QHashIterator<int, AirportInformation *> i(*in); | ||
|  |     while (i.hasNext()) | ||
|  |     { | ||
|  |         i.next(); | ||
|  |         AirportInformation *info = i.value(); | ||
|  |         out->insert(info->m_ident, info); | ||
|  |     } | ||
|  |     return out; | ||
|  | } | ||
|  | 
 | ||
|  | // Read OurAirport's airport frequencies CSV file
 | ||
|  | bool OurAirportsDB::readFrequenciesDB(const QString &filename, QHash<int, AirportInformation *> *airportInfo) | ||
|  | { | ||
|  |     int cnt = 0; | ||
|  | 
 | ||
|  |     // Column numbers used for the data as of 2020/10/28
 | ||
|  |     int airportRefCol = 1; | ||
|  |     int typeCol = 3; | ||
|  |     int descriptionCol = 4; | ||
|  |     int frequencyCol = 5; | ||
|  | 
 | ||
|  |     qDebug() << "OurAirportsDB::readFrequenciesDB: " << filename; | ||
|  | 
 | ||
|  |     QFile file(filename); | ||
|  |     if (file.open(QIODevice::ReadOnly)) | ||
|  |     { | ||
|  |         QList<QByteArray> colNames; | ||
|  |         int idx; | ||
|  | 
 | ||
|  |         // Read header
 | ||
|  |         if (!file.atEnd()) | ||
|  |         { | ||
|  |             QByteArray row = file.readLine().trimmed(); | ||
|  |             colNames = row.split(','); | ||
|  |             // Work out which columns the data is in, based on the headers
 | ||
|  |             idx = colNames.indexOf("airport_ref"); | ||
|  |             if (idx >= 0) | ||
|  |                  airportRefCol = idx; | ||
|  |             idx = colNames.indexOf("type"); | ||
|  |             if (idx >= 0) | ||
|  |                  typeCol = idx; | ||
|  |             idx = colNames.indexOf("descrption"); | ||
|  |             if (idx >= 0) | ||
|  |                  descriptionCol = idx; | ||
|  |             idx = colNames.indexOf("frequency_mhz"); | ||
|  |             if (idx >= 0) | ||
|  |                  frequencyCol = idx; | ||
|  |         } | ||
|  |         // Read data
 | ||
|  |         while (!file.atEnd()) | ||
|  |         { | ||
|  |             QByteArray row = file.readLine(); | ||
|  |             QList<QByteArray> cols = row.split(','); | ||
|  | 
 | ||
|  |             bool ok = false; | ||
|  |             int airportRef = cols[airportRefCol].toInt(&ok, 10); | ||
|  |             if (ok) | ||
|  |             { | ||
|  |                 if (airportInfo->contains(airportRef)) | ||
|  |                 { | ||
|  |                     QString type = trimQuotes(cols[typeCol]); | ||
|  |                     QString description = trimQuotes(cols[descriptionCol]); | ||
|  |                     float frequency = cols[frequencyCol].toFloat(); | ||
|  | 
 | ||
|  |                     AirportInformation::FrequencyInformation *frequencyInfo = new AirportInformation::FrequencyInformation(); | ||
|  |                     frequencyInfo->m_type = type; | ||
|  |                     frequencyInfo->m_description = description; | ||
|  |                     frequencyInfo->m_frequency = frequency; | ||
|  |                     airportInfo->value(airportRef)->m_frequencies.append(frequencyInfo); | ||
|  |                     cnt++; | ||
|  |                 } | ||
|  |             } | ||
|  |         } | ||
|  |         file.close(); | ||
|  |     } | ||
|  |     else | ||
|  |     { | ||
|  |         qDebug() << "Failed to open " << filename << " " << file.errorString(); | ||
|  |         return false; | ||
|  |     } | ||
|  | 
 | ||
|  |     qDebug() << "OurAirportsDB::readFrequenciesDB: - read " << cnt << " airports"; | ||
|  | 
 | ||
|  |     return true; | ||
|  | } | ||
|  | 
 |