Space treatment, a bit more memory safe handling, exclude option, typo fixes
All checks were successful
MLC Release workflow / Archlinux (release) Successful in 1m35s

This commit is contained in:
Blue 2025-03-30 22:07:20 +03:00
parent ceab08a26d
commit bf88e05a13
Signed by: blue
GPG Key ID: 9B203B252A63EE38
10 changed files with 111 additions and 73 deletions

View File

@ -1,7 +1,10 @@
# Changelog # Changelog
## MLC 1.3.4 (UNRELEASED) ## MLC 1.3.4 (March 30, 2025)
- Build fixes - Build fixes
- Source and Destination paths now can contain spaces
- Regex to copy non music files can have spaces
- A feature to define a regex to exclude paths from rendering
## MLC 1.3.3 (October 13, 2023) ## MLC 1.3.3 (October 13, 2023)
- Regex to specify non-music files to copy - Regex to specify non-music files to copy
@ -28,7 +31,7 @@
- New logging system - New logging system
- Artist, Album artist, Album and Title are now utf16 encoded, should fix broken titles - Artist, Album artist, Album and Title are now utf16 encoded, should fix broken titles
- BPM tag is now rounded, as it supposed to be by spec - BPM tag is now rounded, as it supposed to be by spec
- Lyrics is not set now for USLT tag for unsychronized lyrics is not supported in LAME - Lyrics is not set now for USLT tag for unsynchronized lyrics is not supported in LAME
- Date bug fix, it was MMDD instead of standard DDMM - Date bug fix, it was MMDD instead of standard DDMM
## MLC 1.0.1 (July 21, 2023) ## MLC 1.0.1 (July 21, 2023)

View File

@ -13,13 +13,13 @@ This is a program to compile your local lossless music library to lossy formats
### Building ### Building
```sh ```shell
$ git clone https://git.macaw.me/blue/mlc git clone https://git.macaw.me/blue/mlc
$ cd mlc cd mlc
$ mkdir build mkdir build
$ cd build cd build
$ cmake .. cmake ..
$ cmake --build . cmake --build .
``` ```
### Usage ### Usage
@ -27,19 +27,19 @@ $ cmake --build .
Just to compile lossless library to lossy you can use this command Just to compile lossless library to lossy you can use this command
assuming you are in the same directory you have just built `MLC`: assuming you are in the same directory you have just built `MLC`:
``` ```shell
./mlc path/to/lossless/library path/to/store/lossy/library ./mlc path/to/lossless/library path/to/store/lossy/library
``` ```
There are more ways to use `MLC`, please refer to help for more options: There are more ways to use `MLC`, please refer to help for more options:
```sh ```shell
$ ./mlc help ./mlc help
# or # or
$ ./mlc --help ./mlc --help
# #
# or # or
$ ./mlc -h ./mlc -h
``` ```
`MLC` has a way to configure conversion process, it will generate global for you user config on the first launch. `MLC` has a way to configure conversion process, it will generate global for you user config on the first launch.
@ -49,14 +49,16 @@ You can also make local configs for each directory you launch mlc from.
To output the default config run the following, assuming you are in the same directory you have just built `MLC`: To output the default config run the following, assuming you are in the same directory you have just built `MLC`:
```sh ```shell
$ ./mlc config ./mlc config #to print
#or
./mlc config > config.conf #to save to config.conf in current directory
``` ```
To use non default config run the following, assuming you are in the same directory you have just built `MLC`: To use non default config run the following, assuming you are in the same directory you have just built `MLC`:
```sh ```shell
$ ./mlc path/to/lossless/library path/to/store/lossy/library -c path/to/config/file ./mlc path/to/lossless/library path/to/store/lossy/library -c path/to/config/file
``` ```
### About ### About

View File

@ -6,11 +6,12 @@ namespace fs = std::filesystem;
static const std::string flac(".flac"); static const std::string flac(".flac");
Collection::Collection(const std::filesystem::path& path, TaskManager* tm) : Collection::Collection(const std::filesystem::path& path, const std::shared_ptr<TaskManager>& tm, const std::shared_ptr<Settings>& st):
path(path), path(path),
countMusical(0), countMusical(0),
counted(false), counted(false),
taskManager(tm) taskManager(tm),
settings(st)
{} {}
Collection::~Collection() Collection::~Collection()
@ -33,13 +34,16 @@ uint32_t Collection::countMusicFiles() const {
++countMusical; ++countMusical;
} else if (fs::is_directory(path)) { } else if (fs::is_directory(path)) {
for (const fs::directory_entry& entry : fs::directory_iterator(path)) { for (const fs::directory_entry& entry : fs::directory_iterator(path)) {
if (settings->isExcluded(entry.path()))
continue;
switch (entry.status().type()) { switch (entry.status().type()) {
case fs::file_type::regular: case fs::file_type::regular:
if (isMusic(entry.path())) if (isMusic(entry.path()))
++countMusical; ++countMusical;
break; break;
case fs::file_type::directory: { case fs::file_type::directory: {
Collection collection(entry.path()); Collection collection(entry.path(), taskManager, settings);
countMusical += collection.countMusicFiles(); countMusical += collection.countMusicFiles();
} break; } break;
default: default:
@ -53,25 +57,24 @@ uint32_t Collection::countMusicFiles() const {
} }
void Collection::convert(const std::string& outPath) { void Collection::convert(const std::string& outPath) {
if (taskManager == nullptr)
throw 6;
fs::path out = fs::absolute(outPath); fs::path out = fs::absolute(outPath);
fs::create_directories(out); fs::create_directories(out);
out = fs::canonical(outPath); out = fs::canonical(outPath);
for (const fs::directory_entry& entry : fs::directory_iterator(path)) { for (const fs::directory_entry& entry : fs::directory_iterator(path)) {
fs::path sourcePath = entry.path();
if (settings->isExcluded(sourcePath))
continue;
switch (entry.status().type()) { switch (entry.status().type()) {
case fs::file_type::regular: { case fs::file_type::regular: {
fs::path sourcePath = entry.path();
if (isMusic(sourcePath)) if (isMusic(sourcePath))
taskManager->queueConvert(sourcePath, out / sourcePath.stem()); taskManager->queueConvert(sourcePath, out / sourcePath.stem());
else else
taskManager->queueCopy(sourcePath, out / sourcePath.filename()); taskManager->queueCopy(sourcePath, out / sourcePath.filename());
} break; } break;
case fs::file_type::directory: { case fs::file_type::directory: {
fs::path sourcePath = entry.path(); Collection collection(sourcePath, taskManager, settings);
Collection collection(sourcePath, taskManager);
fs::path::iterator itr = sourcePath.end(); fs::path::iterator itr = sourcePath.end();
--itr; --itr;
collection.convert(std::string(out / *itr)); collection.convert(std::string(out / *itr));

View File

@ -3,14 +3,16 @@
#include <string> #include <string>
#include <iostream> #include <iostream>
#include <filesystem> #include <filesystem>
#include <memory>
#include "settings.h"
#include "flactomp3.h" #include "flactomp3.h"
class TaskManager; class TaskManager;
class Collection { class Collection {
public: public:
Collection(const std::filesystem::path& path, TaskManager* tm = nullptr); Collection(const std::filesystem::path& path, const std::shared_ptr<TaskManager>& tm, const std::shared_ptr<Settings>& st);
~Collection(); ~Collection();
void list() const; void list() const;
@ -24,6 +26,7 @@ private:
std::filesystem::path path; std::filesystem::path path;
mutable uint32_t countMusical; mutable uint32_t countMusical;
mutable bool counted; mutable bool counted;
TaskManager* taskManager; std::shared_ptr<TaskManager> taskManager;
std::shared_ptr<Settings> settings;
}; };

View File

@ -92,7 +92,15 @@
# Variable bitrate # Variable bitrate
# Switches on or off variable bitrate # Switches on or off variable bitrate
# VBR files are usually smaller, but not supped to be worse # VBR files are usually smaller, but not supposed to be worse
# in terms of quality. VBR files might be a bit more tricky for the player # in terms of quality. VBR files might be a bit more tricky for the player
# Allowed values are: [true, false] # Allowed values are: [true, false]
#vbr true #vbr true
# Exclude
# MLC renders any music file it finds in source directory
# UNLESS its path matches the following regex
# Allowed value is the regex without any additional syntax,
# for example: exclude [Ss]hamefull?\s[Ss]ong[Ss]
# If you don't want to exclude anything leave this option blank
#exclude

View File

@ -57,16 +57,16 @@ int main(int argc, char **argv) {
} }
logger->setSeverity(settings->getLogLevel()); logger->setSeverity(settings->getLogLevel());
TaskManager taskManager(settings, logger); std::shared_ptr<TaskManager> taskManager = std::make_shared<TaskManager>(settings, logger);
taskManager.start(); taskManager->start();
std::chrono::time_point start = std::chrono::system_clock::now(); std::chrono::time_point start = std::chrono::system_clock::now();
Collection collection(input, &taskManager); Collection collection(input, taskManager, settings);
collection.convert(output); collection.convert(output);
taskManager.wait(); taskManager->wait();
std::cout << std::endl; std::cout << std::endl;
taskManager.stop(); taskManager->stop();
std::chrono::time_point end = std::chrono::system_clock::now(); std::chrono::time_point end = std::chrono::system_clock::now();
std::chrono::duration<double> seconds = end - start; std::chrono::duration<double> seconds = end - start;

View File

@ -17,6 +17,7 @@ enum class Option {
destination, destination,
parallel, parallel,
filesToCopy, filesToCopy,
exclude,
encodingQuality, encodingQuality,
outputQuality, outputQuality,
vbr, vbr,
@ -42,6 +43,7 @@ constexpr std::array<std::string_view, static_cast<int>(Option::_optionsSize)> o
"destination", "destination",
"parallel", "parallel",
"filesToCopy", "filesToCopy",
"exclude",
"encodingQuality", "encodingQuality",
"outputQuality", "outputQuality",
"vbr" "vbr"
@ -264,69 +266,74 @@ void Settings::readConfigLine(const std::string& line) {
if (option == Option::_optionsSize) if (option == Option::_optionsSize)
return; return;
std::string value;
std::getline(stream >> std::ws, value);
strip(value);
if (value.empty())
return;
switch (option) { switch (option) {
case Option::level: { case Option::level: {
std::string lv; std::string lv;
if (!logLevel.has_value() && stream >> lv) { if (!logLevel.has_value() && std::istringstream(value) >> lv) {
Logger::Severity level = Logger::stringToSeverity(lv); Logger::Severity level = Logger::stringToSeverity(lv);
if (level < Logger::Severity::_severitySize) if (level < Logger::Severity::_severitySize)
logLevel = level; logLevel = level;
} }
} break; } break;
case Option::type: { case Option::type: {
std::string lv; std::string tp;
if (!outputType.has_value() && stream >> lv) { if (!outputType.has_value() && std::istringstream(value) >> tp) {
Type type = stringToType(lv); Type type = stringToType(tp);
if (type < _typesSize) if (type < _typesSize)
outputType = type; outputType = type;
} }
} break; } break;
case Option::source: { case Option::source: {
std::string path; if (!input.has_value()) {
if (stream >> path) { input = value;
if (!input.has_value()) { } else if (!output.has_value()) {
input = path; output = input;
} else if (!output.has_value()) { input = value;
output = input;
input = path;
}
} }
} break; } break;
case Option::destination: { case Option::destination: {
std::string path; if (!output.has_value())
if (!output.has_value() && stream >> path) output = value;
output = path;
} break; } break;
case Option::parallel: { case Option::parallel: {
unsigned int count; unsigned int count;
if (!threads.has_value() && stream >> count) if (!threads.has_value() && std::istringstream(value) >> count)
threads = count; threads = count;
} break; } break;
case Option::filesToCopy: { case Option::filesToCopy: {
std::string regex; if (!nonMusic.has_value()) {
if (!nonMusic.has_value() && stream >> regex) { if (value == "all")
if (regex == "all") value = "";
regex = ""; else if (value == "none")
else if (regex == "none") value = "a^";
regex = "a^";
nonMusic = regex; nonMusic = value;
} }
} break; } break;
case Option::exclude: {
if (!excluded.has_value())
excluded = value;
} break;
case Option::outputQuality: { case Option::outputQuality: {
unsigned int value; unsigned int quality;
if (!outputQuality.has_value() && stream >> value) if (!outputQuality.has_value() && std::istringstream(value) >> quality)
outputQuality = std::clamp(value, minQuality, maxQuality); outputQuality = std::clamp(quality, minQuality, maxQuality);
} break; } break;
case Option::encodingQuality: { case Option::encodingQuality: {
unsigned int value; unsigned int quality;
if (!encodingQuality.has_value() && stream >> value) if (!encodingQuality.has_value() && std::istringstream(value) >> quality)
encodingQuality = std::clamp(value, minQuality, maxQuality); encodingQuality = std::clamp(quality, minQuality, maxQuality);
} break; } break;
case Option::vbr: { case Option::vbr: {
bool value; bool vb;
if (!vbr.has_value() && stream >> std::boolalpha >> value) if (!vbr.has_value() && std::istringstream(value) >> std::boolalpha >> vb)
vbr = value; vbr = vb;
} break; } break;
default: default:
break; break;
@ -365,9 +372,16 @@ std::string Settings::resolvePath(const std::string& line) {
} }
bool Settings::matchNonMusic(const std::string& fileName) const { bool Settings::matchNonMusic(const std::string& fileName) const {
if (nonMusic.has_value()) if (!nonMusic.has_value())
return std::regex_search(fileName, nonMusic.value());
else
return true; return true;
return std::regex_search(fileName, nonMusic.value());
}
bool Settings::isExcluded(const std::string& path) const {
if (!excluded.has_value())
return false;
return std::regex_search(path, excluded.value());
} }

View File

@ -40,6 +40,7 @@ public:
Action getAction() const; Action getAction() const;
unsigned int getThreads() const; unsigned int getThreads() const;
bool matchNonMusic(const std::string& fileName) const; bool matchNonMusic(const std::string& fileName) const;
bool isExcluded(const std::string& path) const;
unsigned char getEncodingQuality() const; unsigned char getEncodingQuality() const;
unsigned char getOutputQuality() const; unsigned char getOutputQuality() const;
bool getVBR() const; bool getVBR() const;
@ -69,6 +70,7 @@ private:
std::optional<std::string> configPath; std::optional<std::string> configPath;
std::optional<unsigned int> threads; std::optional<unsigned int> threads;
std::optional<std::regex> nonMusic; std::optional<std::regex> nonMusic;
std::optional<std::regex> excluded;
std::optional<unsigned char> encodingQuality; std::optional<unsigned char> encodingQuality;
std::optional<unsigned char> outputQuality; std::optional<unsigned char> outputQuality;
std::optional<bool> vbr; std::optional<bool> vbr;

View File

@ -22,6 +22,9 @@ TaskManager::~TaskManager() {
} }
void TaskManager::queueConvert(const std::filesystem::path& source, const std::filesystem::path& destination) { void TaskManager::queueConvert(const std::filesystem::path& source, const std::filesystem::path& destination) {
if (settings->isExcluded(source))
return;
std::unique_lock<std::mutex> lock(queueMutex); std::unique_lock<std::mutex> lock(queueMutex);
jobs.emplace(Job::convert, source, destination); jobs.emplace(Job::convert, source, destination);
@ -83,7 +86,7 @@ void TaskManager::loop() {
lock.lock(); lock.lock();
++completeTasks; ++completeTasks;
printResilt(job, result); printResult(job, result);
--busyThreads; --busyThreads;
lock.unlock(); lock.unlock();
waitConditional.notify_all(); waitConditional.notify_all();
@ -138,7 +141,7 @@ TaskManager::JobResult TaskManager::execute(Job& job) {
}}; }};
} }
void TaskManager::printResilt(const TaskManager::Job& job, const TaskManager::JobResult& result) { void TaskManager::printResult(const TaskManager::Job& job, const TaskManager::JobResult& result) {
std::string msg; std::string msg;
switch (job.type) { switch (job.type) {
case Job::copy: case Job::copy:

View File

@ -36,7 +36,7 @@ public:
private: private:
void loop(); void loop();
JobResult execute(Job& job); JobResult execute(Job& job);
void printResilt(const Job& job, const JobResult& result); void printResult(const Job& job, const JobResult& result);
static JobResult mp3Job(const Job& job, const std::shared_ptr<Settings>& settings); static JobResult mp3Job(const Job& job, const std::shared_ptr<Settings>& settings);
static JobResult copyJob(const Job& job, const std::shared_ptr<Settings>& settings); static JobResult copyJob(const Job& job, const std::shared_ptr<Settings>& settings);