Hentai-Downloader/source/config/downloaderconfig.d

222 lines
6.1 KiB
D

module config.downloaderconfig;
/++
+ This struct represents the config of this
+ programm and is used by the other classes
+ to configure stuff
+/
struct Config
{
/++
+ This variable holds the folder/path into which
+ mangas are downloaded by default
+/
string standard_download_folder;
/++
+ This variable determins if mangas should be downloaded
+ again even if they already are downloaded
+/
bool redownload_mangas_regardless;
/++
+ This variable determins if the downloader should output debug infos
+/
bool enable_debug_output;
}
/++
+ This class handles reading and creating the config files
+/
class DownloaderConfig
{
static private:
import std.stdio : writeln;
import std.json : JSONValue, parseJSON;
import std.file : exists, readText, copy, mkdirRecurse;
import std.stdio : toFile;
import std.string: strip;
import std.array : replace;
/++
+ This specifies the default path to the config
+/
string default_config_path = "/.config/hentai_downloader/config.json";
/++
+ This specifies the default path to the template file config file
+/
immutable string default_config_template_path = "./default_config_template.json";
/++
+ This holds the text of the `default_config_template.json` file in case it
+ doesn't exist in the default path anymore
+/
immutable string default_config_template_text =
`
{
"standard_download_folder" : "~/Downloads/Lewds/",
"redownload_mangas_regardless" : false,
"enable_debug_output" : false
}
`;
/++
+ This function checks the config in `config_path` exists
+/
bool configExists(string config_path)
{
return exists(config_path);
}
/++
+ This function creates a new default config at `config_path`
+/
void createNewConfig(string config_path)
{
writeln("[*] Creating new config file in ", config_path);
// The folder that holds the config file
immutable string config_dir = "/home/" ~ getUsername() ~ "/.config/hentai_downloader";
/* writeln("Config folder:", config_dir); */
// if the folder ~/.config/hentai_downloader doesnt exist
if(!exists(config_dir))
mkdirRecurse(config_dir); // Create it
/* writeln("Config file: ", config_path); */
// Write to template file into the config folder
if(!exists(config_path))
{
/* writeln("Saving default config to file: ", config_path); */
default_config_template_text.toFile(config_path);
}
}
/++
+ Turns a relative path e.g. "~/.config" into an absolute path "/home/user/.config"
+/
string makeRelativePathAbsolute(string path)
{
return path.replace("~", "/home/"~getUsername());
}
/++
+ This function checks if the folder specified in the config file exists
+ and if not creates it
+/
void checkStandardFolder(Config config)
{
// FIXME: nested folders break me
// Replace the relative path
string absolutePath = makeRelativePathAbsolute(config.standard_download_folder);
if(!exists(absolutePath))
{
// If the `standard_download_folder` doesnt exist, create it!
absolutePath.mkdirRecurse();
}
}
/++
+ Gets the username of the current user
+/
string getUsername()
{
import std.process : executeShell;
import core.stdc.stdlib : exit, EXIT_FAILURE;
// Execute whoami to get the current username
auto whoami = executeShell(`whoami`);
if(whoami.status != 0)
{
// If whoami fails exit the program
// FIXME: raise an exception or something
writeln("[!] Failed to get the current username");
exit(EXIT_FAILURE);
}
// Return the stripped username
return whoami.output.strip();
}
/++
+ Get the users .config path
+/
string getUserConfigPath()
{
// Return the combined path to the .config folder of the user
return "/home/" ~ getUsername() ~ default_config_path;
}
/++
+ This function trys to parse the config file
+ given in `config_path`, and return a Config struct
+/
Config parseConfig(string config_path)
{
import std.conv : to;
// Read the config file
string config_text = to!string(readText(config_path));
// Parse the config as JSON
auto config_json = parseJSON(config_text);
// Create a new config
Config _config;
// Assign the values that were parsed from the config file
_config.standard_download_folder = config_json["standard_download_folder"].str();
_config.redownload_mangas_regardless = config_json["redownload_mangas_regardless"].boolean();
_config.enable_debug_output = config_json["enable_debug_output"].boolean();
// Adjust the foler path
_config.standard_download_folder = makeRelativePathAbsolute(_config.standard_download_folder);
return _config;
}
/++
+ Loads and checks the given config
+/
Config loadMyConfig(string config_path)
{
// Get the cofig path for this user
config_path = getUserConfigPath();
// Check if the given config exists
if(!configExists(config_path))
createNewConfig(config_path); // If it doesn't, create it!
// Parse the config
Config _config = parseConfig(config_path);
// Check the download folder
checkStandardFolder(_config);
return _config;
}
public:
/++
+ This loads and parses the default config at `default_config_path`
+ into a `Config` struct
+/
static Config loadConfig()
{
return loadMyConfig(this.default_config_path);
}
/++
+ This loads and parses a custom config from `custom_config_path`
+ into a `Config` struct
+/
static Config loadConfig(string custom_config_path)
{
return loadMyConfig(custom_config_path);
}
}