#include "fastfetch.h"
#include "common/commandoption.h"
#include "common/init.h"
#include "common/io/io.h"
#include "common/jsonconfig.h"
#include "common/printing.h"
#include "detection/version/version.h"
#include "logo/logo.h"
#include "util/stringUtils.h"
#include "util/mallocHelper.h"
#include "fastfetch_datatext.h"
#include <stdlib.h>
#include <ctype.h>
#include <string.h>
#ifdef WIN32
#include "util/windows/getline.h"
#endif
static void printCommandFormatHelpJson(void)
{
yyjson_mut_doc* doc = yyjson_mut_doc_new(NULL);
yyjson_mut_val* root = yyjson_mut_obj(doc);
yyjson_mut_doc_set_root(doc, root);
for (uint32_t i = 0; i <= 'Z' - 'A'; ++i)
{
for (FFModuleBaseInfo** modules = ffModuleInfos[i]; *modules; ++modules)
{
FFModuleBaseInfo* baseInfo = *modules;
if (!baseInfo->formatArgs.count) continue;
FF_STRBUF_AUTO_DESTROY type = ffStrbufCreateS(baseInfo->name);
ffStrbufLowerCase(&type);
ffStrbufAppendS(&type, "Format");
yyjson_mut_val* obj = yyjson_mut_obj(doc);
if (yyjson_mut_obj_add(root, yyjson_mut_strbuf(doc, &type), obj))
{
FF_STRBUF_AUTO_DESTROY content = ffStrbufCreateF("Output format of the module `%s`. See Wiki for formatting syntax\n", baseInfo->name);
for (unsigned i = 0; i < baseInfo->formatArgs.count; i++)
{
const FFModuleFormatArg* arg = &baseInfo->formatArgs.args[i];
ffStrbufAppendF(&content, " %u. {%s}: %s\n", i + 1, arg->name, arg->desc);
}
ffStrbufTrimRight(&content, '\n');
yyjson_mut_obj_add_strbuf(doc, obj, "description", &content);
yyjson_mut_obj_add_str(doc, obj, "type", "string");
}
}
}
yyjson_mut_write_fp(stdout, doc, YYJSON_WRITE_PRETTY, NULL, NULL);
putchar('\n');
yyjson_mut_doc_free(doc);
}
static void printCommandFormatHelp(const char* command)
{
FF_STRBUF_AUTO_DESTROY type = ffStrbufCreateNS((uint32_t) (strlen(command) - strlen("-format")), command);
ffStrbufLowerCase(&type);
for (FFModuleBaseInfo** modules = ffModuleInfos[toupper(command[0]) - 'A']; *modules; ++modules)
{
FFModuleBaseInfo* baseInfo = *modules;
if (ffStrbufIgnCaseEqualS(&type, baseInfo->name))
{
if (baseInfo->formatArgs.count > 0)
{
printf("--%s-format:\n", type.chars);
printf("Sets the format string for %s output.\n", baseInfo->name);
puts("To see how a format string is constructed, take a look at \"fastfetch --help format\".");
puts("The following values are passed:");
for (unsigned i = 0; i < baseInfo->formatArgs.count; i++)
{
const FFModuleFormatArg* arg = &baseInfo->formatArgs.args[i];
printf("%16s {%u}: %s\n", arg->name, i + 1, arg->desc);
}
}
else
fprintf(stderr, "Error: Module '%s' doesn't support output formatting\n", baseInfo->name);
return;
}
}
fprintf(stderr, "Error: Module '%s' is not supported\n", type.chars);
}
static void printFullHelp()
{
fputs("Fastfetch is a neofetch-like tool for fetching system information and displaying them in a pretty way\n\n", stdout);
if (!instance.config.display.pipe)
fputs("\e[1;4mUsage:\e[m \e[1mfastfetch\e[m \e[3m<?options>\e[m\n\n", stdout);
else
fputs("Usage: fastfetch <?options>\n\n", stdout);
yyjson_doc* doc = yyjson_read(FASTFETCH_DATATEXT_JSON_HELP, strlen(FASTFETCH_DATATEXT_JSON_HELP), YYJSON_READ_NOFLAG);
assert(doc);
yyjson_val *groupKey, *flagArr;
size_t groupIdx, groupMax;
yyjson_obj_foreach(yyjson_doc_get_root(doc), groupIdx, groupMax, groupKey, flagArr)
{
if (!instance.config.display.pipe)
fputs("\e[1;4m", stdout);
printf("%s options:", yyjson_get_str(groupKey));
if (!instance.config.display.pipe)
fputs("\e[m", stdout);
putchar('\n');
yyjson_val* flagObj;
size_t flagIdx, flagMax;
yyjson_arr_foreach(flagArr, flagIdx, flagMax, flagObj)
{
yyjson_val* shortKey = yyjson_obj_get(flagObj, "short");
if (shortKey)
{
fputs(" ", stdout);
if (!instance.config.display.pipe)
fputs("\e[1m", stdout);
printf("-%s", yyjson_get_str(shortKey));
if (!instance.config.display.pipe)
fputs("\e[m", stdout);
fputs(", ", stdout);
}
else
{
fputs(" ", stdout);
}
yyjson_val* longKey = yyjson_obj_get(flagObj, "long");
assert(longKey);
if (!instance.config.display.pipe)
fputs("\e[1m", stdout);
printf("--%s", yyjson_get_str(longKey));
if (!instance.config.display.pipe)
fputs("\e[m", stdout);
yyjson_val* argObj = yyjson_obj_get(flagObj, "arg");
if (argObj)
{
yyjson_val* typeKey = yyjson_obj_get(argObj, "type");
assert(typeKey);
yyjson_val* optionalKey = yyjson_obj_get(argObj, "optional");
bool optional = optionalKey && yyjson_get_bool(optionalKey);
putchar(' ');
if (!instance.config.display.pipe)
fputs("\e[3m", stdout);
printf("<%s%s>", optional ? "?" : "", yyjson_get_str(typeKey));
if (!instance.config.display.pipe)
fputs("\e[m", stdout);
}
yyjson_val* descKey = yyjson_obj_get(flagObj, "desc");
assert(descKey);
if (yyjson_is_arr(descKey))
{
if (instance.config.display.pipe)
putchar(':');
yyjson_val* descStr;
size_t descIdx, descMax;
yyjson_arr_foreach(descKey, descIdx, descMax, descStr)
{
if (!instance.config.display.pipe)
printf("\e[46G%s\n", yyjson_get_str(descStr));
else
printf(" %s", yyjson_get_str(descStr));
}
if (instance.config.display.pipe)
putchar('\n');
}
else
{
if (!instance.config.display.pipe)
fputs("\e[46G", stdout);
else
fputs(": ", stdout);
puts(yyjson_get_str(descKey));
}
}
putchar('\n');
}
yyjson_doc_free(doc);
puts("\n\
Parsing is not case sensitive. E.g. `--print-logos` is equal to `--Print-Logos`\n\
If a value starts with a ?, it is optional. An optional boolean value defaults to true if not specified.\n\
More detailed help messages for each options can be printed with `-h <option_without_dash_prefix>`\n\
All options can be made permanent with command `fastfetch <options> --gen-config`");
}
static bool printSpecificCommandHelp(const char* command)
{
yyjson_doc* doc = yyjson_read(FASTFETCH_DATATEXT_JSON_HELP, strlen(FASTFETCH_DATATEXT_JSON_HELP), YYJSON_READ_NOFLAG);
assert(doc);
yyjson_val *groupKey, *flagArr;
size_t groupIdx, groupMax;
yyjson_obj_foreach(yyjson_doc_get_root(doc), groupIdx, groupMax, groupKey, flagArr)
{
yyjson_val* flagObj;
size_t flagIdx, flagMax;
yyjson_arr_foreach(flagArr, flagIdx, flagMax, flagObj)
{
yyjson_val* pseudo = yyjson_obj_get(flagObj, "pseudo");
if (pseudo && yyjson_get_bool(pseudo))
continue;
yyjson_val* longKey = yyjson_obj_get(flagObj, "long");
assert(longKey);
if (ffStrEqualsIgnCase(command, yyjson_get_str(longKey)))
{
puts(yyjson_get_str(yyjson_obj_get(flagObj, "desc")));
printf("%10s: ", "Usage");
yyjson_val* shortKey = yyjson_obj_get(flagObj, "short");
if (shortKey)
{
if (!instance.config.display.pipe)
fputs("\e[1m", stdout);
printf("-%s", yyjson_get_str(shortKey));
if (!instance.config.display.pipe)
fputs("\e[m", stdout);
fputs(", ", stdout);
}
if (!instance.config.display.pipe)
fputs("\e[1m", stdout);
printf("--%s", yyjson_get_str(longKey));
if (!instance.config.display.pipe)
fputs("\e[m", stdout);
yyjson_val* argObj = yyjson_obj_get(flagObj, "arg");
if (argObj)
{
yyjson_val* typeKey = yyjson_obj_get(argObj, "type");
assert(typeKey);
yyjson_val* optionalKey = yyjson_obj_get(argObj, "optional");
bool optional = optionalKey && yyjson_get_bool(optionalKey);
putchar(' ');
if (!instance.config.display.pipe)
fputs("\e[3m", stdout);
printf("<%s%s>", optional ? "?" : "", yyjson_get_str(typeKey));
if (!instance.config.display.pipe)
fputs("\e[m", stdout);
putchar('\n');
yyjson_val* defaultKey = yyjson_obj_get(argObj, "default");
if (defaultKey)
{
if (ffStrEqualsIgnCase(yyjson_get_str(typeKey), "structure"))
printf("%10s: %s\n", "Default", FASTFETCH_DATATEXT_STRUCTURE);
else if (yyjson_is_bool(defaultKey))
printf("%10s: %s\n", "Default", yyjson_get_bool(defaultKey) ? "true" : "false");
else if (yyjson_is_num(defaultKey))
printf("%10s: %d\n", "Default", yyjson_get_int(defaultKey));
else if (yyjson_is_str(defaultKey))
printf("%10s: %s\n", "Default", yyjson_get_str(defaultKey));
else
printf("%10s: Unknown\n", "Default");
}
yyjson_val* enumKey = yyjson_obj_get(argObj, "enum");
if (enumKey)
{
printf("%10s:\n", "Options");
yyjson_val *optKey, *optVal;
size_t optIdx, optMax;
yyjson_obj_foreach(enumKey, optIdx, optMax, optKey, optVal)
printf("%12s: %s\n", yyjson_get_str(optKey), yyjson_get_str(optVal));
}
}
else
putchar('\n');
yyjson_val* remarkKey = yyjson_obj_get(flagObj, "remark");
if (remarkKey)
{
if (yyjson_is_str(remarkKey))
printf("%10s: %s\n", "Remark", yyjson_get_str(remarkKey));
else if (yyjson_is_arr(remarkKey) && yyjson_arr_size(remarkKey) > 0)
{
yyjson_val* remarkStr;
size_t remarkIdx, remarkMax;
yyjson_arr_foreach(remarkKey, remarkIdx, remarkMax, remarkStr)
{
if (remarkIdx == 0)
printf("%10s: %s\n", "Remark", yyjson_get_str(remarkStr));
else
printf(" %s\n", yyjson_get_str(remarkStr));
}
}
}
yyjson_doc_free(doc);
return true;
}
}
}
yyjson_doc_free(doc);
return false;
}
static void printCommandHelp(const char* command)
{
if(command == NULL)
printFullHelp();
else if(ffStrEqualsIgnCase(command, "format-json"))
printCommandFormatHelpJson();
else if(ffCharIsEnglishAlphabet(command[0]) && ffStrEndsWithIgnCase(command, "-format")) printCommandFormatHelp(command);
else if(!printSpecificCommandHelp(command))
fprintf(stderr, "Error: No specific help for command '%s' provided\n", command);
}
static void listAvailablePresets(bool pretty)
{
FF_LIST_FOR_EACH(FFstrbuf, path, instance.state.platform.dataDirs)
{
ffStrbufAppendS(path, "fastfetch/presets/");
ffListFilesRecursively(path->chars, pretty);
}
if (instance.state.platform.exePath.length)
{
FF_STRBUF_AUTO_DESTROY absolutePath = ffStrbufCreateCopy(&instance.state.platform.exePath);
ffStrbufSubstrBeforeLastC(&absolutePath, '/');
ffStrbufAppendS(&absolutePath, "/presets/");
ffListFilesRecursively(absolutePath.chars, pretty);
}
}
static void listAvailableLogos(void)
{
FF_LIST_FOR_EACH(FFstrbuf, path, instance.state.platform.dataDirs)
{
ffStrbufAppendS(path, "fastfetch/logos/");
ffListFilesRecursively(path->chars, true);
}
}
static void listConfigPaths(void)
{
FF_LIST_FOR_EACH(FFstrbuf, folder, instance.state.platform.configDirs)
{
bool exists = false;
uint32_t length = folder->length + (uint32_t) strlen("fastfetch") + 1 ;
ffStrbufAppendS(folder, "fastfetch/config.jsonc");
exists = ffPathExists(folder->chars, FF_PATHTYPE_FILE);
ffStrbufSubstrBefore(folder, length);
printf("%s%s\n", folder->chars, exists ? " (*)" : "");
}
}
static void listDataPaths(void)
{
FF_LIST_FOR_EACH(FFstrbuf, folder, instance.state.platform.dataDirs)
{
ffStrbufAppendS(folder, "fastfetch/");
puts(folder->chars);
}
}
static void listModules(bool pretty)
{
unsigned count = 0;
for (int i = 0; i <= 'Z' - 'A'; ++i)
{
for (FFModuleBaseInfo** modules = ffModuleInfos[i]; *modules; ++modules)
{
++count;
if (pretty)
printf("%d)%s%-14s: %s\n", count, count > 9 ? " " : " ", (*modules)->name, (*modules)->description);
else
printf("%s:%s\n", (*modules)->name, (*modules)->description);
}
}
}
static bool parseJsoncFile(const char* path, bool strictJson)
{
assert(!instance.state.configDoc);
{
yyjson_read_err error;
instance.state.configDoc = yyjson_read_file(path, strictJson ? 0 : YYJSON_READ_ALLOW_COMMENTS | YYJSON_READ_ALLOW_TRAILING_COMMAS, NULL, &error);
if (!instance.state.configDoc)
{
if (error.code != YYJSON_READ_ERROR_FILE_OPEN)
{
size_t row = 0, col = error.pos;
FF_STRBUF_AUTO_DESTROY content = ffStrbufCreate();
if (ffAppendFileBuffer(path, &content))
yyjson_locate_pos(content.chars, content.length, error.pos, &row, &col, NULL);
fprintf(stderr, "Error: failed to parse JSON config file `%s` at (%zu, %zu): %s\n", path, row, col, error.msg);
exit(477);
}
return false;
}
}
{
const char* error = NULL;
yyjson_val* const root = yyjson_doc_get_root(instance.state.configDoc);
if (!yyjson_is_obj(root))
error = "Invalid JSON config format. Root value must be an object";
if (
error ||
(error = ffOptionsParseLogoJsonConfig(&instance.config.logo, root)) ||
(error = ffOptionsParseGeneralJsonConfig(&instance.config.general, root)) ||
(error = ffOptionsParseDisplayJsonConfig(&instance.config.display, root)) ||
false
) {
fprintf(stderr, "JsonConfig Error: %s\n", error);
exit(477);
}
}
return true;
}
static void generateConfigFile(bool force, const char* filePath)
{
if (!filePath)
{
ffStrbufSet(&instance.state.genConfigPath, FF_LIST_GET(FFstrbuf, instance.state.platform.configDirs, 0));
ffStrbufAppendS(&instance.state.genConfigPath, "fastfetch/config.jsonc");
}
else
{
ffStrbufSetS(&instance.state.genConfigPath, filePath);
}
if (!force && ffPathExists(instance.state.genConfigPath.chars, FF_PATHTYPE_ANY))
{
fprintf(stderr, "Error: file `%s` exists. Use `--gen-config-force` to overwrite\n", instance.state.genConfigPath.chars);
exit(477);
}
}
static void optionParseConfigFile(FFdata* data, const char* key, const char* value)
{
if (data->configLoaded)
{
fprintf(stderr, "Error: only one config file can be loaded\n");
exit(413);
}
data->configLoaded = true;
if(value == NULL)
{
fprintf(stderr, "Error: usage: %s <config>\n", key);
exit(413);
}
if (value[0] == '\0' || ffStrEqualsIgnCase(value, "none"))
return;
FF_STRBUF_AUTO_DESTROY absolutePath = ffStrbufCreateS(value);
bool strictJson = ffStrbufEndsWithIgnCaseS(&absolutePath, ".json");
bool needExtension = !strictJson && !ffStrbufEndsWithIgnCaseS(&absolutePath, ".jsonc");
if (needExtension)
ffStrbufAppendS(&absolutePath, ".jsonc");
if (parseJsoncFile(absolutePath.chars, strictJson)) return;
FF_LIST_FOR_EACH(FFstrbuf, path, instance.state.platform.dataDirs)
{
ffStrbufSet(&absolutePath, path);
ffStrbufAppendS(&absolutePath, "fastfetch/presets/");
ffStrbufAppendS(&absolutePath, value);
if (needExtension)
ffStrbufAppendS(&absolutePath, ".jsonc");
if (parseJsoncFile(absolutePath.chars, strictJson)) return;
}
if (instance.state.platform.exePath.length)
{
uint32_t lastSlash = ffStrbufLastIndexC(&instance.state.platform.exePath, '/') + 1;
assert(lastSlash < instance.state.platform.exePath.length);
ffStrbufSetNS(&absolutePath, lastSlash, instance.state.platform.exePath.chars);
ffStrbufAppendS(&absolutePath, value);
if (needExtension)
ffStrbufAppendS(&absolutePath, ".jsonc");
if (parseJsoncFile(absolutePath.chars, strictJson)) return;
ffStrbufSubstrBefore(&absolutePath, lastSlash);
ffStrbufAppendS(&absolutePath, "presets/");
ffStrbufAppendS(&absolutePath, value);
if (needExtension)
ffStrbufAppendS(&absolutePath, ".jsonc");
if (parseJsoncFile(absolutePath.chars, strictJson)) return;
}
fprintf(stderr, "Error: couldn't find config: %s\n", value);
exit(414);
}
static void printVersion()
{
FFVersionResult* result = &ffVersionResult;
printf("%s %s%s%s (%s)\n", result->projectName, result->version, result->versionTweak, result->debugMode ? "-debug" : "", result->architecture);
}
static void parseCommand(FFdata* data, char* key, char* value)
{
if(ffStrEqualsIgnCase(key, "-h") || ffStrEqualsIgnCase(key, "--help"))
{
printCommandHelp(value);
exit(0);
}
if(ffStrEqualsIgnCase(key, "--help-raw"))
{
puts(FASTFETCH_DATATEXT_JSON_HELP);
exit(0);
}
else if(ffStrEqualsIgnCase(key, "-v") || ffStrEqualsIgnCase(key, "--version"))
{
printVersion();
exit(0);
}
else if(ffStrEqualsIgnCase(key, "--version-raw"))
{
puts(FASTFETCH_PROJECT_VERSION);
exit(0);
}
else if(ffStrStartsWithIgnCase(key, "--print-"))
{
const char* subkey = key + strlen("--print-");
if(ffStrEndsWithIgnCase(subkey, "structure"))
puts(FASTFETCH_DATATEXT_STRUCTURE);
else if(ffStrEqualsIgnCase(subkey, "logos"))
ffLogoBuiltinPrint();
else
{
fprintf(stderr, "Error: unsupported print option: %s\n", key);
exit(415);
}
exit(0);
}
else if(ffStrStartsWithIgnCase(key, "--list-"))
{
const char* subkey = key + strlen("--list-");
if(ffStrEqualsIgnCase(subkey, "modules"))
listModules(!value || !ffStrEqualsIgnCase(value, "autocompletion"));
else if(ffStrEqualsIgnCase(subkey, "presets"))
listAvailablePresets(!value || !ffStrEqualsIgnCase(value, "autocompletion"));
else if(ffStrEqualsIgnCase(subkey, "config-paths"))
listConfigPaths();
else if(ffStrEqualsIgnCase(subkey, "data-paths"))
listDataPaths();
else if(ffStrEqualsIgnCase(subkey, "features"))
ffListFeatures();
else if(ffStrEqualsIgnCase(subkey, "logos"))
{
if (value)
{
if (ffStrEqualsIgnCase(value, "autocompletion"))
ffLogoBuiltinListAutocompletion();
else if (ffStrEqualsIgnCase(value, "builtin"))
ffLogoBuiltinList();
else if (ffStrEqualsIgnCase(value, "custom"))
listAvailableLogos();
else
{
fprintf(stderr, "Error: unsupported logo type: %s\n", value);
exit(415);
}
}
else
{
puts("Builtin logos:");
ffLogoBuiltinList();
puts("\nCustom logos:");
listAvailableLogos();
}
}
else
{
fprintf(stderr, "Error: unsupported list option: %s\n", key);
exit(415);
}
exit(0);
}
else if(ffStrEqualsIgnCase(key, "--gen-config"))
generateConfigFile(false, value);
else if(ffStrEqualsIgnCase(key, "--gen-config-force"))
generateConfigFile(true, value);
else if(ffStrEqualsIgnCase(key, "-c") || ffStrEqualsIgnCase(key, "--load-config") || ffStrEqualsIgnCase(key, "--config"))
optionParseConfigFile(data, key, value);
else if(ffStrEqualsIgnCase(key, "--format"))
{
switch (ffOptionParseEnum(key, value, (FFKeyValuePair[]) {
{ "default", 0},
{ "json", 1 },
{},
}))
{
case 0:
if (instance.state.resultDoc)
{
yyjson_mut_doc_free(instance.state.resultDoc);
instance.state.resultDoc = NULL;
}
break;
case 1:
if (!instance.state.resultDoc)
{
instance.state.resultDoc = yyjson_mut_doc_new(NULL);
yyjson_mut_doc_set_root(instance.state.resultDoc, yyjson_mut_arr(instance.state.resultDoc));
}
break;
}
}
else
return;
key[0] = '\0';
if (value) value[0] = '\0';
}
static void parseOption(FFdata* data, const char* key, const char* value)
{
if(ffStrEqualsIgnCase(key, "-s") || ffStrEqualsIgnCase(key, "--structure"))
ffOptionParseString(key, value, &data->structure);
else if(
ffOptionsParseGeneralCommandLine(&instance.config.general, key, value) ||
ffOptionsParseLogoCommandLine(&instance.config.logo, key, value) ||
ffOptionsParseDisplayCommandLine(&instance.config.display, key, value) ||
ffParseModuleOptions(key, value)
) {}
else
{
fprintf(stderr, "Error: unknown option: %s\n", key);
exit(400);
}
}
static void parseConfigFiles(void)
{
if (__builtin_expect(instance.state.genConfigPath.length == 0, true))
{
FF_LIST_FOR_EACH(FFstrbuf, dir, instance.state.platform.configDirs)
{
uint32_t dirLength = dir->length;
ffStrbufAppendS(dir, "fastfetch/config.jsonc");
bool success = parseJsoncFile(dir->chars, false);
ffStrbufSubstrBefore(dir, dirLength);
if (success) return;
}
}
}
static void parseArguments(FFdata* data, int argc, char** argv, void (*parser)(FFdata* data, char* key, char* value))
{
for(int i = 1; i < argc; i++)
{
const char* key = argv[i];
if(*key == '\0')
continue;
if(*key != '-')
{
fprintf(stderr, "Error: invalid option: %s. An option must start with `-`\n", key);
exit(400);
}
if(i == argc - 1 || (
argv[i + 1][0] == '-' &&
argv[i + 1][1] != '\0' && !ffStrEqualsIgnCase(argv[i], "--separator-string") )) {
parser(data, argv[i], NULL);
}
else
{
parser(data, argv[i], argv[i + 1]);
++i;
}
}
}
static void run(FFdata* data)
{
const bool useJsonConfig = data->structure.length == 0 && instance.state.configDoc;
if (useJsonConfig)
ffPrintJsonConfig(true , instance.state.resultDoc);
else
ffPrepareCommandOption(data);
ffStart();
#if defined(_WIN32)
if (!instance.config.display.noBuffer) fflush(stdout);
#endif
if (useJsonConfig)
ffPrintJsonConfig(false, instance.state.resultDoc);
else
ffPrintCommandOption(data, instance.state.resultDoc);
if (instance.state.resultDoc)
yyjson_mut_write_fp(stdout, instance.state.resultDoc, YYJSON_WRITE_INF_AND_NAN_AS_NULL | YYJSON_WRITE_PRETTY_TWO_SPACES | YYJSON_WRITE_NEWLINE_AT_END, NULL, NULL);
else
ffFinish();
}
static void writeConfigFile(FFdata* data, const FFstrbuf* filename)
{
yyjson_mut_doc* doc = yyjson_mut_doc_new(NULL);
yyjson_mut_val* root = yyjson_mut_obj(doc);
yyjson_mut_doc_set_root(doc, root);
yyjson_mut_obj_add_str(doc, root, "$schema", "https://github.com/fastfetch-cli/fastfetch/raw/dev/doc/json_schema.json");
ffOptionsGenerateLogoJsonConfig(&instance.config.logo, doc);
ffOptionsGenerateDisplayJsonConfig(&instance.config.display, doc);
ffOptionsGenerateGeneralJsonConfig(&instance.config.general, doc);
ffMigrateCommandOptionToJsonc(data, doc);
if (ffStrbufEqualS(filename, "-"))
yyjson_mut_write_fp(stdout, doc, YYJSON_WRITE_INF_AND_NAN_AS_NULL | YYJSON_WRITE_PRETTY_TWO_SPACES | YYJSON_WRITE_NEWLINE_AT_END, NULL, NULL);
else
{
size_t len;
FF_AUTO_FREE const char* str = yyjson_mut_write(doc, YYJSON_WRITE_INF_AND_NAN_AS_NULL | YYJSON_WRITE_PRETTY_TWO_SPACES | YYJSON_WRITE_NEWLINE_AT_END, &len);
if (!str)
{
printf("Error: failed to generate config file\n");
exit(1);
}
if (ffWriteFileData(filename->chars, len, str))
printf("The generated config file has been written in `%s`\n", filename->chars);
else
{
printf("Error: failed to write file in `%s`\n", filename->chars);
exit(1);
}
}
yyjson_mut_doc_free(doc);
}
int main(int argc, char** argv)
{
ffInitInstance();
atexit(ffDestroyInstance);
FFdata data = {
.structure = ffStrbufCreate(),
.configLoaded = false,
};
parseArguments(&data, argc, argv, parseCommand);
if(!data.configLoaded && !getenv("NO_CONFIG"))
parseConfigFiles();
parseArguments(&data, argc, argv, (void*) parseOption);
if (__builtin_expect(instance.state.genConfigPath.length == 0, true))
run(&data);
else
writeConfigFile(&data, &instance.state.genConfigPath);
ffStrbufDestroy(&data.structure);
}