/* * Copyright (C) 2016 by Jonathan Naylor G4KLX * * 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; either version 2 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 for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. */ #include "Reflectors.h" #include "Log.h" #include #include #include #include #include #include CReflectors::CReflectors(const std::string& hostsFile, unsigned int reloadTime) : m_hostsFile(hostsFile), m_reflectors(), m_current(), m_search(), m_timer(1000U, reloadTime * 60U) { if (reloadTime > 0U) m_timer.start(); } CReflectors::~CReflectors() { for (std::vector::iterator it = m_reflectors.begin(); it != m_reflectors.end(); ++it) delete *it; } bool CReflectors::load() { FILE* fp = ::fopen(m_hostsFile.c_str(), "rt"); if (fp == NULL) { LogWarning("Cannot open the YSF Hosts file - %s", m_hostsFile.c_str()); return false; } // Clear out the old reflector list for (std::vector::iterator it = m_reflectors.begin(); it != m_reflectors.end(); ++it) delete *it; char buffer[100U]; while (::fgets(buffer, 100U, fp) != NULL) { if (buffer[0U] == '#') continue; char* p1 = ::strtok(buffer, ";\r\n"); char* p2 = ::strtok(NULL, ";\r\n"); char* p3 = ::strtok(NULL, ";\r\n"); char* p4 = ::strtok(NULL, ";\r\n"); char* p5 = ::strtok(NULL, ";\r\n"); char* p6 = ::strtok(NULL, "\r\n"); if (p1 != NULL && p2 != NULL && p3 != NULL && p4 != NULL && p5 != NULL && p6 != NULL) { std::string host = std::string(p4); in_addr address = CUDPSocket::lookup(host); if (address.s_addr != INADDR_NONE) { CYSFReflector* refl = new CYSFReflector; refl->m_id = std::string(p1); refl->m_name = std::string(p2); refl->m_desc = std::string(p3); refl->m_address = address; refl->m_port = (unsigned int)::atoi(p5); refl->m_count = std::string(p6);; m_reflectors.push_back(refl); } } } ::fclose(fp); size_t size = m_reflectors.size(); if (size == 0U) return false; LogInfo("Loaded %u YSF reflectors", size); return true; } CYSFReflector* CReflectors::find(const std::string& id) { for (std::vector::iterator it = m_reflectors.begin(); it != m_reflectors.end(); ++it) { if (id == (*it)->m_id) return *it; } LogMessage("Trying to find non existent reflector with an id of %s", id.c_str()); return NULL; } static bool refComparison(const CYSFReflector* r1, const CYSFReflector* r2) { assert(r1 != NULL); assert(r2 != NULL); std::string name1 = r1->m_name; std::string name2 = r2->m_name; for (unsigned int i = 0U; i < 16U; i++) { int c = ::toupper(name1.at(i)) - ::toupper(name2.at(i)); if (c != 0) return c < 0; } return false; } std::vector& CReflectors::current() { m_current.clear(); for (std::vector::iterator it = m_reflectors.begin(); it != m_reflectors.end(); ++it) m_current.push_back(*it); std::sort(m_current.begin(), m_current.end(), refComparison); return m_current; } std::vector& CReflectors::search(const std::string& name) { m_search.clear(); std::string trimmed = name; trimmed.erase(std::find_if(trimmed.rbegin(), trimmed.rend(), std::not1(std::ptr_fun(std::isspace))).base(), trimmed.end()); std::transform(trimmed.begin(), trimmed.end(), trimmed.begin(), ::toupper); unsigned int len = trimmed.size(); for (std::vector::iterator it = m_reflectors.begin(); it != m_reflectors.end(); ++it) { std::string reflector = (*it)->m_name; reflector.erase(std::find_if(reflector.rbegin(), reflector.rend(), std::not1(std::ptr_fun(std::isspace))).base(), reflector.end()); std::transform(reflector.begin(), reflector.end(), reflector.begin(), ::toupper); if (trimmed == reflector.substr(0U, len)) m_search.push_back(*it); } std::sort(m_search.begin(), m_search.end(), refComparison); return m_search; } void CReflectors::clock(unsigned int ms) { m_timer.clock(ms); if (m_timer.isRunning() && m_timer.hasExpired()) { load(); m_timer.start(); } }