// SPDX-License-Identifier: GPL-2.0-only /** * Digital Voice Modem - Modem Host Software * GPLv2 Open Source. Use is subject to license terms. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * @package DVM / DFSI peer application * @derivedfrom MMDVMHost (https://github.com/g4klx/MMDVMHost) * @license GPLv2 License (https://opensource.org/licenses/GPL-2.0) * * Copyright (C) 2024 Patrick McDonnell, W3AXL * */ #include "ActivityLog.h" #include "common/network/BaseNetwork.h" #include "common/Log.h" // for CurrentLogFileLevel() and LogGetNetwork() #include #if defined(CATCH2_TEST_COMPILATION) #include #endif #include #include #include #include #include #include // --------------------------------------------------------------------------- // Constants // --------------------------------------------------------------------------- #define EOL "\r\n" const uint32_t ACT_LOG_BUFFER_LEN = 501U; // --------------------------------------------------------------------------- // Global Variables // --------------------------------------------------------------------------- static std::string m_actFilePath; static std::string m_actFileRoot; static FILE* m_actFpLog = nullptr; static struct tm m_actTm; // --------------------------------------------------------------------------- // Global Functions // --------------------------------------------------------------------------- /// /// Helper to open the activity log file, file handle. /// /// True, if log file is opened, otherwise false. static bool ActivityLogOpen() { if (CurrentLogFileLevel() == 0U) return true; time_t now; ::time(&now); struct tm* tm = ::gmtime(&now); if (tm->tm_mday == m_actTm.tm_mday && tm->tm_mon == m_actTm.tm_mon && tm->tm_year == m_actTm.tm_year) { if (m_actFpLog != nullptr) return true; } else { if (m_actFpLog != nullptr) ::fclose(m_actFpLog); } char filename[200U]; ::sprintf(filename, "%s/%s-%04d-%02d-%02d.activity.log", LogGetFilePath().c_str(), LogGetFileRoot().c_str(), tm->tm_year + 1900, tm->tm_mon + 1, tm->tm_mday); m_actFpLog = ::fopen(filename, "a+t"); m_actTm = *tm; return m_actFpLog != nullptr; } /// /// Initializes the activity log. /// /// Full-path to the activity log file. /// Prefix of the activity log file name. bool ActivityLogInitialise(const std::string& filePath, const std::string& fileRoot) { #if defined(CATCH2_TEST_COMPILATION) return true; #endif m_actFilePath = filePath; m_actFileRoot = fileRoot; return ::ActivityLogOpen(); } /// /// Finalizes the activity log. /// void ActivityLogFinalise() { #if defined(CATCH2_TEST_COMPILATION) return; #endif if (m_actFpLog != nullptr) ::fclose(m_actFpLog); } /// /// Writes a new entry to the activity log. /// /// This is a variable argument function. /// Formatted string to write to activity log. void ActivityLog(const char* msg, ...) { #if defined(CATCH2_TEST_COMPILATION) return; #endif assert(msg != nullptr); char buffer[ACT_LOG_BUFFER_LEN]; struct timeval now; ::gettimeofday(&now, NULL); struct tm* tm = ::gmtime(&now.tv_sec); ::sprintf(buffer, "A: %04d-%02d-%02d %02d:%02d:%02d.%03lu ", tm->tm_year + 1900, tm->tm_mon + 1, tm->tm_mday, tm->tm_hour, tm->tm_min, tm->tm_sec, now.tv_usec / 1000U); va_list vl, vl_len; va_start(vl, msg); va_copy(vl_len, vl); size_t len = ::vsnprintf(nullptr, 0U, msg, vl_len); ::vsnprintf(buffer + ::strlen(buffer), len + 1U, msg, vl); va_end(vl_len); va_end(vl); bool ret = ::ActivityLogOpen(); if (!ret) return; if (LogGetNetwork() != nullptr) { network::BaseNetwork* network = (network::BaseNetwork*)LogGetNetwork();; network->writeActLog(buffer); } if (CurrentLogFileLevel() == 0U) return; ::fprintf(m_actFpLog, "%s\n", buffer); ::fflush(m_actFpLog); if (2U >= g_logDisplayLevel && g_logDisplayLevel != 0U) { ::fprintf(stdout, "%s" EOL, buffer); ::fflush(stdout); } }