add some app infrastructure services

This commit is contained in:
2026-06-07 15:49:43 -05:00
parent 1eb62ed186
commit 30b06e077c
10 changed files with 328 additions and 11 deletions

View File

@@ -0,0 +1,80 @@
#include "ConfigService.hpp"
#include <iostream>
#include <exception>
ConfigService::ConfigService(const std::string& filePath) {
if(!loadFromFile(filePath)) {
std::cout << "Error loading file " << filePath << std::endl;
}
}
bool ConfigService::loadFromFile(const std::string& filePath) {
try {
config_.clear();
config_.readFile(filePath.c_str());
lastError_.clear();
return true;
} catch (const libconfig::FileIOException&) {
lastError_ = "Unable to read config file: " + filePath;
} catch (const libconfig::ParseException& error) {
lastError_ = std::string("Parse error in ") + error.getFile() + ":" +
std::to_string(error.getLine()) + " - " + error.getError();
} catch (const std::exception& error) {
lastError_ = error.what();
}
return false;
}
bool ConfigService::getLoggerConfig(const std::string& id, LoggerConfig* loggerConfig) const {
try {
const libconfig::Setting& loggers = config_.lookup("Loggers");
for (int index = 0; index < loggers.getLength(); ++index) {
const libconfig::Setting& loggerSetting = loggers[index];
std::string loggerId;
if (!loggerSetting.lookupValue("Id", loggerId) || loggerId != id) {
continue;
}
return parseLoggerConfig(loggerSetting, loggerConfig);
}
} catch (const libconfig::SettingException& ex) {
std::cout << "libconfig setting exception: " << ex.what() << std::endl;
return false;
}
return false;
}
const std::string& ConfigService::lastError() const {
return lastError_;
}
bool ConfigService::parseLoggerConfig(const libconfig::Setting& loggerSetting, LoggerConfig* loggerConfig) const {
if (!loggerSetting.lookupValue("Id", loggerConfig->id)) {
return false;
}
try {
const libconfig::Setting& flags = loggerSetting.lookup("FlagsEnabled");
for (int index = 0; index < flags.getLength(); ++index) {
loggerConfig->flagsEnabled.push_back(static_cast<const char*>(flags[index]));
}
} catch (const libconfig::SettingException&) {
loggerConfig->flagsEnabled.clear();
}
loggerSetting.lookupValue("ShowTime", loggerConfig->showTime);
loggerSetting.lookupValue("ShowSourceTrace", loggerConfig->showSourceTrace);
loggerSetting.lookupValue("CoutEnabled", loggerConfig->coutEnabled);
loggerSetting.lookupValue("FileEnabled", loggerConfig->fileEnabled);
loggerSetting.lookupValue("FilePath", loggerConfig->filePath);
return true;
}

View File

@@ -1,6 +1,47 @@
#pragma once
#include <libconfig.h++>
#include <optional>
#include <string>
#include <vector>
struct LoggerConfig {
std::string id;
std::vector<std::string> flagsEnabled;
bool showTime = false;
bool showSourceTrace = false;
bool coutEnabled = false;
bool fileEnabled = false;
std::string filePath;
};
struct AudioConfig {
uint32_t sampleRate;
uint32_t channels;
uint32_t stereoMode;
uint32_t bufferSize;
float pitchStandard;
int32_t midiHome;
int32_t notesPerOctave;
};
class ConfigService {
public:
ConfigService(const std::string& filePath);
~ConfigService() = default;
bool loadFromFile(const std::string& filePath);
bool getLoggerConfig(const std::string& id, LoggerConfig* loggerConfig) const;
const std::string& lastError() const;
private:
bool parseLoggerConfig(const libconfig::Setting& loggerSetting, LoggerConfig* loggerConfig) const;
libconfig::Config config_;
std::string lastError_;
};

View File

@@ -0,0 +1,108 @@
#include <chrono> // Tracking the time when the log function is called
#include <string>
#include <source_location>
#include "LoggerService.hpp"
#include <iostream>
#include <iomanip>
#include <fstream>
#include <filesystem>
namespace fs = std::filesystem;
LoggerService::LoggerService(ConfigService* config, const std::string& loggerId) {
if(!(config->getLoggerConfig(loggerId, &configuration_))) {
std::cout << "Failed to get logger configuration fom config service" << std::endl;
return;
}
standardOutputEnabled_ = configuration_.coutEnabled;
fileOutputEnabled_ = configuration_.fileEnabled;
additionaldetailsEnabled_ = configuration_.showSourceTrace;
timeEnabled_ = configuration_.showTime;
id_ = configuration_.id;
for(std::string& flag : configuration_.flagsEnabled) {
bool found = false;
for(const char* validFlag : LogFlagStrings) {
if(flag == std::string(validFlag)) {
found = true;
for(int i = 0; i < LogFlag::Count; i++) {
if(LogFlagStrings[i] == flag) {
activeFlags_.emplace_back(static_cast<LogFlag>(i));
break;
}
}
}
}
if(!found) {
std::cout << "Log flag '" << flag << "' in configuration file is not a valid log flag" << std::endl;
}
}
const auto now = std::chrono::system_clock::now();
const std::time_t t_c = std::chrono::system_clock::to_time_t(now);
std::string finaltime = std::ctime(&t_c);
finaltime.pop_back(); // removing the newline
std::replace(finaltime.begin(),finaltime.end(), ':' , '-'); // filenames cant have dashes
std::replace(finaltime.begin(),finaltime.end(), ' ' , '_');
fs::create_directories(std::string(BINARY_DIR) + "/" + finaltime);
std::string logPath = std::string(BINARY_DIR) + "/" + finaltime + "/" + configuration_.filePath;
if(fileOutputEnabled_) outfile_.open(logPath);
log("Logger", LogFlag::Info, "Logger initialized.");
}
LoggerService::~LoggerService() {
if(outfile_) outfile_.close();
}
void LoggerService::log(std::string component, LogFlag flag, std::string message, std::source_location Source) {
// check if flag is in the list of active flags
bool culled = true;
for(LogFlag& testFlag : activeFlags_) {
if(flag == testFlag) {
culled = false;
break;
}
}
if(culled) return;
std::string finalmessage = "";
if(timeEnabled_) {
finalmessage = finalmessage + "[" + "Not Implemented" + "] ";
}
std::string componentTrace = "[" + id_ + ": " + component + "] ";
finalmessage += componentTrace; // component is the section of the program (For example Mesh or Engine) that is calling the logger
std::string level = "";
level = LogFlagStrings[flag];
// level.append(7 - level.length(), ' ') pads out the level string with whitespace so every line is aligned the same
// it looked weird though
finalmessage = finalmessage + "[" + level + "] ";
finalmessage = finalmessage + message + " ";
if (additionaldetailsEnabled_) {
finalmessage = finalmessage + "[Function: " + Source.function_name() + "]" + " " + "[Line: " + std::to_string(Source.line()) + "]" + " " + "[File: " + Source.file_name() + "]";
}
if(standardOutputEnabled_) {
std::cout << finalmessage << std::endl;
}
if(fileOutputEnabled_) {
outfile_ << finalmessage << std::endl;
}
return;
}

View File

@@ -0,0 +1,46 @@
#pragma once
#include <vector>
#include <string>
#include <fstream>
#include <source_location>
#include "ConfigService.hpp"
enum LogFlag {
Debug,
Info,
Warning,
Error,
Count
};
static constexpr const char* LogFlagStrings[] = {
"Debug",
"Info",
"Warning",
"Error"
};
class LoggerService {
public:
LoggerService(ConfigService* config, const std::string& loggerId);
~LoggerService();
void log(std::string component, LogFlag flag, std::string message, std::source_location Source = std::source_location::current()); // Using the <source_location>
private:
bool standardOutputEnabled_;
bool fileOutputEnabled_;
bool additionaldetailsEnabled_;
bool timeEnabled_;
std::ofstream outfile_;
std::string id_;
std::vector<LogFlag> activeFlags_;
LoggerConfig configuration_;
};

View File

@@ -6,6 +6,8 @@
#include <QQmlContext>
#include "TimerComponent.hpp"
#include "LoggerService.hpp"
#include "ConfigService.hpp"
#include "synth/AudioEngine.hpp"
int main(int argc, char* argv[]) {
@@ -28,8 +30,12 @@ int main(int argc, char* argv[]) {
return -1;
}
// create app objects
ConfigService config = ConfigService("config/sonobulus.cfg");
LoggerService logger = LoggerService(&config, "Engine");
// audio synthesizer doohickey
AudioEngine audioEngine = AudioEngine();
AudioEngine audioEngine = AudioEngine(&logger);
audioEngine.start();
// execute app

View File

@@ -5,12 +5,14 @@
#include <cmath>
#include <numbers>
AudioEngine::AudioEngine() {
AudioEngine::AudioEngine(LoggerService* logger) : logger_(logger) {
if(audioDevice_.getDeviceCount() < 1) {
std::cout << "No audio devices found" << std::endl;
}
if(logger_ == nullptr) std::cout << "err: logger nullptr" << std::endl;
}
AudioEngine::~AudioEngine() {
@@ -41,7 +43,8 @@ bool AudioEngine::start() {
}
// sanity check
std::cout << "sample rate: " << sampleRate_ << " buffer frames: " << bufferFrames_ << std::endl;
std::string msg = "sample rate: " + std::to_string(sampleRate_) + ", buffer frames: " + std::to_string(bufferFrames_);
logger_->log("AudioEngine", LogFlag::Info, msg);
return true;
}
@@ -58,7 +61,7 @@ bool AudioEngine::stop() {
int32_t AudioEngine::audioCallback(void* outputBuffer, void* inputBuffer, uint32_t nFrames, double, RtAudioStreamStatus status, void* userData) {
// error if the callback is late
if(status) std::cout << "stream underflow" << std::endl;
if(status) static_cast<AudioEngine*>(userData)->logger_->log("AudioEngine", LogFlag::Warning, "Audio buffer underflow");
return static_cast<AudioEngine*>(userData)->process(static_cast<float*>(outputBuffer), static_cast<size_t>(nFrames));

View File

@@ -5,6 +5,9 @@
#include <RtAudio.h>
#include "LoggerService.hpp"
#if defined(_WIN32)
#define AUDIO_API RtAudio::WINDOWS_WASAPI
#else
@@ -15,7 +18,7 @@ class AudioEngine {
public:
AudioEngine();
AudioEngine(LoggerService* logger);
~AudioEngine();
bool start();
@@ -36,4 +39,6 @@ private:
float phase_ = 0.0f;
LoggerService* logger_;
};