neowofetch/uwufetch.c

1458 lines
46 KiB
C
Raw Normal View History

/*
* UwUfetch is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <https://www.gnu.org/licenses/>.
*/
2021-09-10 00:51:37 +02:00
#define _GNU_SOURCE // for strcasestr
2021-09-07 06:59:34 +02:00
2021-11-06 00:34:53 +01:00
#ifdef __APPLE__
#include <TargetConditionals.h> // for checking iOS
2021-11-06 00:34:53 +01:00
#endif
#include <dirent.h>
#include <getopt.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
2021-07-24 12:27:38 +02:00
#if defined(__APPLE__) || defined(__FREEBSD__)
#include <sys/sysctl.h>
#include <time.h>
2021-11-01 16:48:54 +01:00
#else // defined(__APPLE__) || defined(__FREEBSD__)
#ifdef __FREEBSD__
#else // defined(__FREEBSD__) || defined(_WIN32)
#ifndef _WIN32
#include <sys/sysinfo.h>
#else // _WIN32
#include <sysinfoapi.h>
#endif // _WIN32
#endif // defined(__FREEBSD__) || defined(_WIN32)
#endif // defined(__APPLE__) || defined(__FREEBSD__)
#ifndef _WIN32
#include <sys/ioctl.h>
#include <sys/utsname.h>
#else // _WIN32
#include <windows.h>
2021-11-01 16:48:54 +01:00
CONSOLE_SCREEN_BUFFER_INFO csbi;
#endif // _WIN32
// COLORS
#define NORMAL "\x1b[0m"
#define BOLD "\x1b[1m"
#define BLACK "\x1b[30m"
#define RED "\x1b[31m"
#define GREEN "\x1b[32m"
2021-11-01 16:48:54 +01:00
#define SPRING_GREEN "\x1b[38;5;120m"
#define YELLOW "\x1b[33m"
#define BLUE "\x1b[34m"
#define MAGENTA "\x1b[0;35m"
#define CYAN "\x1b[36m"
#define WHITE "\x1b[37m"
#define PINK "\x1b[38;5;201m"
#define LPINK "\x1b[38;5;213m"
2021-11-01 16:48:54 +01:00
#ifdef _WIN32
#define BLOCK_CHAR "\xdb"
#else // _WIN32
#define BLOCK_CHAR "\u2587"
#endif // _WIN32
2021-04-15 22:59:43 +02:00
#ifdef __APPLE__
// buffers where data fetched from sysctl are stored
// CPU
#define CPUBUFFERLEN 128
2021-04-15 22:59:43 +02:00
2021-04-16 10:48:39 +02:00
char cpu_buffer[CPUBUFFERLEN];
size_t cpu_buffer_len = CPUBUFFERLEN;
2021-04-15 22:59:43 +02:00
2021-04-16 10:48:39 +02:00
// Installed RAM
int64_t mem_buffer = 0;
2021-04-16 10:48:39 +02:00
size_t mem_buffer_len = sizeof(mem_buffer);
2021-04-15 22:59:43 +02:00
2021-04-16 10:48:39 +02:00
// uptime
struct timeval time_buffer;
size_t time_buffer_len = sizeof(time_buffer);
2021-11-01 16:48:54 +01:00
#endif // __APPLE__
struct package_manager {
char command_string[128]; // command to get number of packages installed
char pkgman_name[16]; // name of the package manager
};
2021-05-03 03:46:55 +02:00
// initialise the variables to store data, gpu array can hold up to 8 gpus
// int target_width = 0, screen_width = 0, screen_height = 0, ram_total,
// ram_used = 0, pkgs = 0; long uptime = 0;
// all flags available
struct configuration {
int ascii_image_flag, // when (0) ascii is printed, when (1) image is
// printed
show_user_info, show_os, show_host, show_kernel, show_cpu, show_gpu,
show_ram, show_resolution, show_shell, show_pkgs, show_uptime,
show_colors;
};
2021-03-18 21:54:29 +01:00
char* terminal_cursor_move = "\033[18C";
struct info {
char user[128], host[256], shell[64], host_model[256], host_model_version[256], kernel[256],
version_name[64], cpu_model[256], gpu_model[64][256], pkgman_name[64],
image_name[128], *config_directory, *cache_content;
int target_width, screen_width, screen_height, ram_total, ram_used, pkgs;
long uptime;
#ifndef _WIN32
struct utsname sys_var;
#endif // _WIN32
#ifndef __APPLE__
#ifdef __linux__
struct sysinfo sys;
#else // __linux__
#ifdef _WIN32
struct _SYSTEM_INFO sys;
#endif // _WIN32
#endif // __linux__
#endif // __APPLE__
#ifndef _WIN32
struct winsize win;
#else // _WIN32
int ws_col, ws_rows;
#endif // _WIN32
};
2021-11-01 16:48:54 +01:00
// functions definitions, to use them in main()
struct configuration parse_config(struct info* user_info);
#ifdef _WIN32
int pkgman(struct info* user_info, struct configuration* config_flags);
#else // _WIN32
int pkgman(struct info* user_info);
#endif // _WIN32
void print_info(struct configuration* config_flags, struct info* user_info);
void write_cache(struct info* user_info);
int read_cache(struct info* user_info);
void print_cache(struct configuration* config_flags, struct info* user_info);
#ifdef _WIN32
struct info get_info(struct configuration* config_flags);
#else // _WIN32
struct info get_info();
#endif // _WIN32
void list(char* arg);
void replace(char* original, char* search, char* replacer);
void replace_ignorecase(char* original, char* search, char* replacer);
void print_ascii(struct info* user_info);
void print_image(struct info* user_info);
void usage(char* arg);
void uwu_kernel(char* kernel);
void uwu_hw(char* hwname);
void uwu_name(struct configuration* config_flags, struct info* user_info);
void truncate_name(char* name, int target_width);
void remove_brackets(char* str);
int main(int argc, char* argv[]) {
char* cache_env = getenv("UWUFETCH_CACHE_ENABLED");
struct configuration config_flags;
struct info user_info = {0};
if (cache_env != NULL) {
int cache_enabled = 0;
char buffer[128];
2021-10-14 22:48:08 +02:00
sscanf(cache_env, "%4[TRUEtrue1]", buffer);
cache_enabled =
(strcmp(buffer, "true") == 0 || strcmp(buffer, "TRUE") == 0 ||
strcmp(buffer, "1") == 0);
if (cache_enabled) {
2021-10-12 13:26:18 +02:00
// if no cache file found write to it
if (!read_cache(&user_info)) {
user_info = get_info(&config_flags);
write_cache(&user_info);
}
config_flags = parse_config(&user_info);
print_cache(&config_flags, &user_info);
2021-10-12 13:26:18 +02:00
return 0;
}
}
#ifdef _WIN32
2021-11-01 16:48:54 +01:00
// packages disabled by default because chocolatey is slow
config_flags.show_pkgs = 0;
2021-11-01 16:48:54 +01:00
#endif
int opt = 0;
static struct option long_options[] = {
{"ascii", no_argument, NULL, 'a'},
{"config", required_argument, NULL, 'c'},
// {"cache", no_argument, NULL, 'C'},
{"distro", required_argument, NULL, 'd'},
{"write-cache", no_argument, NULL, 'w'},
{"help", no_argument, NULL, 'h'},
{"image", optional_argument, NULL, 'i'},
{"list", no_argument, NULL, 'l'},
{NULL, 0, NULL, 0}};
user_info = get_info(&config_flags);
config_flags = parse_config(&user_info);
while ((opt = getopt_long(argc, argv, "ac:d:hi::lw", long_options, NULL)) !=
-1) {
switch (opt) {
case 'a':
config_flags.ascii_image_flag = 0;
break;
case 'c':
user_info.config_directory = optarg;
break;
case 'd':
if (optarg) sprintf(user_info.version_name, "%s", optarg);
break;
case 'h':
usage(argv[0]);
return 0;
case 'i':
config_flags.ascii_image_flag = 1;
if (!optarg && argv[optind] != NULL && argv[optind][0] != '-')
sprintf(user_info.image_name, "%s", argv[optind++]);
else if (optarg)
sprintf(user_info.image_name, "%s", optarg);
break;
case 'l':
list(argv[0]);
return 0;
case 'w':
write_cache(&user_info);
print_cache(&config_flags, &user_info);
return 0;
default:
break;
}
}
if ((argc == 1 && config_flags.ascii_image_flag == 0) ||
(argc > 1 && config_flags.ascii_image_flag == 0)) {
2021-07-22 18:39:34 +02:00
printf("\n"); // print a new line
printf("\033[1A"); // go up one line if possible
print_ascii(&user_info);
} else if (config_flags.ascii_image_flag == 1)
print_image(&user_info);
2021-09-07 06:59:34 +02:00
print_info(&config_flags, &user_info);
}
struct configuration parse_config(struct info* user_info) {
char line[256];
struct configuration config_flags = {
0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1}; // enabling all flags by default
FILE* config = NULL;
if (user_info->config_directory == NULL) {
if (getenv("HOME") != NULL) {
char homedir[512];
sprintf(homedir, "%s/.config/uwufetch/config", getenv("HOME"));
2021-11-01 16:55:33 +01:00
config = fopen(homedir, "r");
}
} else
config = fopen(user_info->config_directory, "r");
if (config == NULL) return config_flags;
while (fgets(line, sizeof(line), config)) {
char buffer[128] = {0};
sscanf(line, "distro=%s", user_info->version_name);
if (sscanf(line, "ascii=%[truefalse]", buffer))
config_flags.ascii_image_flag = !strcmp(buffer, "false");
if (sscanf(line, "image=\"%[^\"]\"", user_info->image_name)) {
if (user_info->image_name[0] ==
'~') { // image name with ~ does not work
memmove(&user_info->image_name[0], &user_info->image_name[1],
strlen(user_info->image_name));
char temp[128] = "/home/";
strcat(temp, user_info->user);
strcat(temp, user_info->image_name);
sprintf(user_info->image_name, "%s", temp);
}
config_flags.ascii_image_flag = 1;
}
if (sscanf(line, "user=%[truefalse]", buffer))
config_flags.show_user_info = !strcmp(buffer, "true");
if (sscanf(line, "os=%[truefalse]", buffer))
config_flags.show_os = strcmp(buffer, "false");
if (sscanf(line, "host=%[truefalse]", buffer))
config_flags.show_host = strcmp(buffer, "false");
if (sscanf(line, "kernel=%[truefalse]", buffer))
config_flags.show_kernel = strcmp(buffer, "false");
if (sscanf(line, "cpu=%[truefalse]", buffer))
config_flags.show_cpu = strcmp(buffer, "false");
if (sscanf(line, "gpu=%[truefalse]", buffer))
config_flags.show_gpu = strcmp(buffer, "false");
if (sscanf(line, "ram=%[truefalse]", buffer))
config_flags.show_ram = strcmp(buffer, "false");
if (sscanf(line, "resolution=%[truefalse]", buffer))
config_flags.show_resolution = strcmp(buffer, "false");
if (sscanf(line, "shell=%[truefalse]", buffer))
config_flags.show_shell = strcmp(buffer, "false");
if (sscanf(line, "pkgs=%[truefalse]", buffer))
config_flags.show_pkgs = strcmp(buffer, "false");
if (sscanf(line, "uptime=%[truefalse]", buffer))
config_flags.show_uptime = strcmp(buffer, "false");
if (sscanf(line, "colors=%[truefalse]", buffer))
config_flags.show_colors = strcmp(buffer, "false");
}
fclose(config);
return config_flags;
2021-04-14 00:21:29 +02:00
}
#ifdef _WIN32
int pkgman(struct info* user_info, struct configuration* config_flags)
#else // _WIN32
int pkgman(struct info* user_info)
#endif
{ // this is just a function that returns the total of installed packages
int total = 0;
#ifndef __APPLE__
// this function is not used on mac os because it causes lots of
// problems
#ifndef _WIN32
2021-09-10 00:51:37 +02:00
struct package_manager pkgmans[] = {
{"apt list --installed 2> /dev/null | wc -l", "(apt)"},
{"apk info 2> /dev/null | wc -l", "(apk)"},
{"dnf list installed 2> /dev/null | wc -l", "(dnf)"},
{"qlist -I 2> /dev/null | wc -l", "(emerge)"},
2021-04-17 15:25:06 +02:00
{"flatpak list 2> /dev/null | wc -l", "(flatpak)"},
2021-07-30 23:54:45 +02:00
{"snap list 2> /dev/null | wc -l", "(snap)"},
{"guix package --list-installed 2> /dev/null | wc -l", "(guix)"},
{"nix-store -q --requisites /run/current-system/sw 2> /dev/null | wc "
"-l",
"(nix)"},
{"pacman -Qq 2> /dev/null | wc -l", "(pacman)"},
2021-07-23 18:00:18 +02:00
{"pkg info 2>/dev/null | wc -l", "(pkg)"},
2021-04-16 10:48:39 +02:00
{"port installed 2> /dev/null | tail -n +2 | wc -l", "(port)"},
{"brew list 2> /dev/null | wc -l", "(brew)"},
{"rpm -qa --last 2> /dev/null | wc -l", "(rpm)"},
{"xbps-query -l 2> /dev/null | wc -l", "(xbps)"},
{"zypper -q se --installed-only 2> /dev/null | wc -l", "(zypper)"}};
const unsigned long pkgman_count = sizeof(pkgmans) / sizeof(pkgmans[0]);
// to format the pkgman_name string properly
int comma_separator = 0;
for (long unsigned int i = 0; i < pkgman_count;
i++) { // long unsigned int instead of int because of -Wsign-compare
struct package_manager* current = &pkgmans[i];
FILE* fp = popen(current->command_string, "r");
unsigned int pkg_count;
if (fscanf(fp, "%u", &pkg_count) == 3) continue;
fclose(fp);
total += pkg_count;
if (pkg_count > 0) {
if (comma_separator) strcat(user_info->pkgman_name, ", ");
comma_separator++;
char spkg_count[16];
sprintf(spkg_count, "%u", pkg_count);
strcat(user_info->pkgman_name, spkg_count);
strcat(user_info->pkgman_name, " ");
strcat(
user_info->pkgman_name,
current->pkgman_name); // this is the line that breaks mac os,
// but something strange happens before
}
}
#else // _WIN32
if (config_flags->show_pkgs) {
FILE* fp = popen("choco list -l --no-color 2> nul", "r");
2021-11-01 16:48:54 +01:00
unsigned int pkg_count;
char buffer[7562] = {0};
while (fgets(buffer, sizeof(buffer), fp)) {
2021-11-01 16:48:54 +01:00
sscanf(buffer, "%u packages installed.", &pkg_count);
}
if (fp) pclose(fp);
2021-11-01 16:48:54 +01:00
total = pkg_count;
char spkg_count[16];
sprintf(spkg_count, "%u", pkg_count);
strcat(user_info->pkgman_name, spkg_count);
strcat(user_info->pkgman_name, " ");
strcat(user_info->pkgman_name, "(chocolatey)");
2021-11-01 16:48:54 +01:00
}
#endif // _WIN32
2021-11-01 16:48:54 +01:00
#endif
return total;
}
2021-04-15 22:59:43 +02:00
#ifdef __APPLE__
int uptime_apple() {
2021-04-16 10:48:39 +02:00
int mib[2] = {CTL_KERN, KERN_BOOTTIME};
sysctl(mib, 2, &time_buffer, &time_buffer_len, NULL, 0);
2021-04-15 22:59:43 +02:00
2021-04-16 10:48:39 +02:00
time_t bsec = time_buffer.tv_sec;
time_t csec = time(NULL);
2021-04-15 22:59:43 +02:00
2021-04-16 10:48:39 +02:00
return difftime(csec, bsec);
2021-04-15 22:59:43 +02:00
}
#endif
2021-07-24 12:27:38 +02:00
#ifdef __FREEBSD__
int uptime_freebsd() { // this code is from coreutils uptime:
// https://github.com/coreutils/coreutils/blob/master/src/uptime.c
int boot_time = 0;
2021-09-10 00:51:37 +02:00
static int request[2] = {CTL_KERN, KERN_BOOTTIME};
2021-07-24 12:27:38 +02:00
struct timeval result;
size_t result_len = sizeof result;
2021-09-10 00:51:37 +02:00
if (sysctl(request, 2, &result, &result_len, NULL, 0) >= 0)
2021-07-24 12:27:38 +02:00
boot_time = result.tv_sec;
2021-09-10 00:51:37 +02:00
int time_now = time(NULL);
2021-07-24 12:27:38 +02:00
return time_now - boot_time;
}
#endif
void print_info(struct configuration* config_flags, struct info* user_info) {
#ifdef _WIN32
#define responsively_printf(buf, format, ...) \
{ \
sprintf(buf, format, __VA_ARGS__); \
printf("%.*s\n", user_info->ws_col - 1, buf); \
}
#else // _WIN32
#define responsively_printf(buf, format, ...) \
{ \
sprintf(buf, format, __VA_ARGS__); \
printf("%.*s\n", user_info->win.ws_col - 1, buf); \
}
#endif // _WIN32
2021-11-01 16:48:54 +01:00
char print_buf[1024]; // for responsively print
// print collected info - from host to cpu info
printf("\033[9A"); // to align info text
if (config_flags->show_user_info)
responsively_printf(print_buf, "%s%s%s%s@%s", terminal_cursor_move,
NORMAL, BOLD, user_info->user, user_info->host);
uwu_name(config_flags, user_info);
if (config_flags->show_os)
responsively_printf(print_buf, "%s%s%sOWOS %s%s",
terminal_cursor_move, NORMAL, BOLD, NORMAL,
user_info->version_name);
if (config_flags->show_host)
2021-12-18 23:24:58 +01:00
responsively_printf(print_buf, "%s%s%sMOWODEL %s%s",
terminal_cursor_move, NORMAL, BOLD, NORMAL,
user_info->host_model);
if (config_flags->show_kernel)
responsively_printf(print_buf, "%s%s%sKEWNEL %s%s",
terminal_cursor_move, NORMAL, BOLD, NORMAL,
user_info->kernel);
if (config_flags->show_cpu)
responsively_printf(print_buf, "%s%s%sCPUWU %s%s",
terminal_cursor_move, NORMAL, BOLD, NORMAL,
user_info->cpu_model);
// print the gpus
if (config_flags->show_gpu)
for (int i = 0; user_info->gpu_model[i][0]; i++)
responsively_printf(print_buf, "%s%s%sGPUWU %s%s",
terminal_cursor_move, NORMAL, BOLD, NORMAL,
user_info->gpu_model[i]);
2021-03-18 21:54:29 +01:00
// print ram to uptime and colors
if (config_flags->show_ram)
responsively_printf(print_buf, "%s%s%sWAM %s%i MiB/%i MiB",
terminal_cursor_move, NORMAL, BOLD, NORMAL,
(user_info->ram_used), user_info->ram_total);
if (config_flags->show_resolution)
if (user_info->screen_width != 0 || user_info->screen_height != 0)
responsively_printf(print_buf, "%s%s%sRESOWUTION%s %dx%d",
terminal_cursor_move, NORMAL, BOLD, NORMAL,
user_info->screen_width,
user_info->screen_height);
if (config_flags->show_shell)
responsively_printf(print_buf, "%s%s%sSHEWW %s%s",
terminal_cursor_move, NORMAL, BOLD, NORMAL,
user_info->shell);
#if defined(__APPLE__) && !defined(__IPHONE__) // some time ago __IPHONE__ was defined as TARGET_OS_IPHONE, but it was defined also in m1 macs, so i changed it
if (config_flags->show_pkgs)
system(
"ls $(brew --cellar) | wc -l | awk -F' ' '{print \" \x1b[34mw "
" w \x1b[0m\x1b[1mPKGS\x1b[0m \"$1 \" (brew)\"}'");
#else
if (config_flags->show_pkgs)
responsively_printf(print_buf, "%s%s%sPKGS %s%s%d: %s",
terminal_cursor_move, NORMAL, BOLD, NORMAL, NORMAL,
user_info->pkgs, user_info->pkgman_name);
#endif
if (config_flags->show_uptime) {
if (user_info->uptime == 0) {
#ifdef __APPLE__
user_info->uptime = uptime_apple();
#else
#ifdef __FREEBSD__
user_info->uptime = uptime_freebsd();
#else
#ifdef _WIN32
user_info->uptime = GetTickCount() / 1000;
#else // _WIN32
user_info->uptime = user_info->sys.uptime;
#endif // _WIN32
#endif
#endif
}
switch (user_info->uptime) {
2021-07-22 18:39:34 +02:00
case 0 ... 3599:
responsively_printf(print_buf, "%s%s%sUWUPTIME %s%lim",
terminal_cursor_move, NORMAL, BOLD, NORMAL,
user_info->uptime / 60 % 60);
2021-07-22 18:39:34 +02:00
break;
case 3600 ... 86399:
responsively_printf(print_buf, "%s%s%sUWUPTIME %s%lih, %lim",
terminal_cursor_move, NORMAL, BOLD, NORMAL,
user_info->uptime / 3600,
user_info->uptime / 60 % 60);
2021-07-22 18:39:34 +02:00
break;
default:
responsively_printf(
print_buf, "%s%s%sUWUPTIME %s%lid, %lih, %lim",
terminal_cursor_move, NORMAL, BOLD, NORMAL,
user_info->uptime / 86400, user_info->uptime / 3600 % 24,
user_info->uptime / 60 % 60);
}
}
if (config_flags->show_colors)
2021-11-01 16:48:54 +01:00
printf("%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s\n",
terminal_cursor_move, BOLD, BLACK, BLOCK_CHAR, BLOCK_CHAR, RED,
BLOCK_CHAR, BLOCK_CHAR, GREEN, BLOCK_CHAR, BLOCK_CHAR, YELLOW,
BLOCK_CHAR, BLOCK_CHAR, BLUE, BLOCK_CHAR, BLOCK_CHAR, MAGENTA,
BLOCK_CHAR, BLOCK_CHAR, CYAN, BLOCK_CHAR, BLOCK_CHAR, WHITE,
BLOCK_CHAR, BLOCK_CHAR, NORMAL);
}
void write_cache(struct info* user_info) {
char cache_file[512];
sprintf(cache_file, "%s/.cache/uwufetch.cache", getenv("HOME"));
FILE* cache_fp = fopen(cache_file, "w");
if (cache_fp == NULL) return;
// writing all info to the cache file
#ifdef __APPLE__
user_info->uptime = uptime_apple();
#else
#ifdef __FREEBSD__
uptime = uptime_freebsd();
#else
#ifndef _WIN32
user_info->uptime = user_info->sys.uptime;
#endif // _WIN32
#endif
#endif
fprintf(
cache_fp,
"user=%s\nhost=%s\nversion_name=%s\nhost_model=%s\nkernel=%s\ncpu=%"
"s\nscreen_width=%d\nscreen_height=%d\nshell=%s\npkgs=%d\npkgman_name=%"
"s\n",
user_info->user, user_info->host, user_info->version_name,
user_info->host_model, user_info->kernel, user_info->cpu_model,
user_info->screen_width, user_info->screen_height, user_info->shell,
user_info->pkgs, user_info->pkgman_name);
for (int i = 0; user_info->gpu_model[i][0]; i++)
fprintf(cache_fp, "gpu=%s\n", user_info->gpu_model[i]);
#ifdef __APPLE__
/* char brew_command[2048];
sprintf(brew_command, "ls $(brew --cellar) | wc -l | awk -F' ' '{print \"
\x1b[34mw w \x1b[0m\x1b[1mPKGS\x1b[0m \"$1 \" (brew)\"}'
> %s", cache_file); system(brew_command); */
#endif
fclose(cache_fp);
return;
}
// return whether the cache file is found
int read_cache(struct info* user_info) {
char cache_file[512];
sprintf(cache_file, "%s/.cache/uwufetch.cache", getenv("HOME"));
FILE* cache_fp = fopen(cache_file, "r");
if (cache_fp == NULL) return 0;
char line[256];
2021-10-12 13:26:18 +02:00
int gpun = 0;
while (fgets(line, sizeof(line), cache_fp)) {
sscanf(line, "user=%99[^\n]", user_info->user);
sscanf(line, "host=%99[^\n]", user_info->host);
sscanf(line, "version_name=%99[^\n]", user_info->version_name);
sscanf(line, "host_model=%99[^\n]", user_info->host_model);
sscanf(line, "kernel=%99[^\n]", user_info->kernel);
sscanf(line, "cpu=%99[^\n]", user_info->cpu_model);
if (sscanf(line, "gpu=%99[^\n]", user_info->gpu_model[gpun]) != 0)
2021-10-12 13:26:18 +02:00
gpun++;
sscanf(line, "screen_width=%i", &user_info->screen_width);
sscanf(line, "screen_height=%i", &user_info->screen_height);
sscanf(line, "shell=%99[^\n]", user_info->shell);
sscanf(line, "pkgs=%i", &user_info->pkgs);
sscanf(line, "pkgman_name=%99[^\n]", user_info->pkgman_name);
}
fclose(cache_fp);
return 1;
}
void print_cache(struct configuration* config_flags, struct info* user_info) {
#ifndef __APPLE__
#ifndef _WIN32
sysinfo(&user_info->sys); // to get uptime
#endif // _WIN32
FILE* meminfo;
#ifdef __FREEBSD__
2021-11-01 16:55:33 +01:00
meminfo = popen("LANG=EN_us freecolor -om 2> /dev/null", "r");
#else // __FREEBSD__
2021-11-01 16:55:33 +01:00
meminfo = popen("LANG=EN_us free -m 2> /dev/null", "r");
#endif // __FREEBSD__
char line[256];
while (fgets(line, sizeof(line), meminfo))
// free command prints like this: "Mem:" total used free shared
// buff/cache available
sscanf(line, "Mem: %d %d", &user_info->ram_total, &user_info->ram_used);
fclose(meminfo);
#elif defined(_WIN32)
2021-10-14 22:48:08 +02:00
// wmic OS get FreePhysicalMemory
FILE* mem_used_fp;
2021-11-01 16:55:33 +01:00
mem_used_fp = popen("wmic OS GET FreePhysicalMemory", "r");
char mem_used_ch[2137];
2021-11-01 16:48:54 +01:00
printf("\n\n\n\\\n");
while (fgets(mem_used_ch, sizeof(mem_used_ch), mem_used_fp) != NULL) {
2021-11-01 16:48:54 +01:00
printf("%s\n", mem_used_ch);
}
pclose(mem_used_fp);
int mem_used = atoi(mem_used_ch);
ram_used = mem_used / 1024;
2021-11-01 16:48:54 +01:00
#else // __APPLE__
2021-10-14 22:48:08 +02:00
// Used ram
FILE *mem_wired_fp, *mem_active_fp, *mem_compressed_fp;
mem_wired_fp =
popen("vm_stat | awk '/wired/ { printf $4 }' | cut -d '.' -f 1", "r");
mem_active_fp =
popen("vm_stat | awk '/active/ { printf $3 }' | cut -d '.' -f 1", "r");
mem_compressed_fp = popen(
"vm_stat | awk '/occupied/ { printf $5 }' | cut -d '.' -f 1", "r");
char mem_wired_ch[2137], mem_active_ch[2137], mem_compressed_ch[2137];
while (fgets(mem_wired_ch, sizeof(mem_wired_ch), mem_wired_fp) != NULL) {
while (fgets(mem_active_ch, sizeof(mem_active_ch), mem_active_fp) !=
NULL) {
while (fgets(mem_compressed_ch, sizeof(mem_compressed_ch),
mem_compressed_fp) != NULL) {
}
}
}
pclose(mem_wired_fp);
pclose(mem_active_fp);
pclose(mem_compressed_fp);
int mem_wired = atoi(mem_wired_ch);
int mem_active = atoi(mem_active_ch);
int mem_compressed = atoi(mem_compressed_ch);
2021-10-14 22:48:08 +02:00
// Total ram
sysctlbyname("hw.memsize", &mem_buffer, &mem_buffer_len, NULL, 0);
user_info->ram_used = ((mem_wired + mem_active + mem_compressed) * 4 / 1024);
2021-11-01 16:48:54 +01:00
#endif // __APPLE__
print_ascii(user_info);
print_info(config_flags, user_info);
return;
}
#ifdef _WIN32
struct info get_info(struct configuration* config_flags)
#else // _WIN32
struct info get_info()
#endif // _WIN32
{ // get all necessary info
struct info user_info = {0};
char line[256]; // var to scan file lines
2021-11-01 16:48:54 +01:00
// terminal width used to truncate long names
#ifndef _WIN32
ioctl(STDOUT_FILENO, TIOCGWINSZ, &user_info.win);
user_info.target_width = user_info.win.ws_col - 30;
#else // _WIN32
2021-11-01 16:48:54 +01:00
GetConsoleScreenBufferInfo(GetStdHandle(STD_OUTPUT_HANDLE), &csbi);
user_info.ws_col = csbi.srWindow.Right - csbi.srWindow.Left - 29;
user_info.ws_rows = csbi.srWindow.Bottom - csbi.srWindow.Top + 1;
#endif // _WIN32
// os version, cpu and board info
FILE* os_release = fopen("/etc/os-release", "r");
2021-07-22 18:39:34 +02:00
#ifndef __FREEBSD__
FILE* cpuinfo = fopen("/proc/cpuinfo", "r");
2021-07-22 18:39:34 +02:00
#else
FILE* cpuinfo = popen("sysctl -a | egrep -i 'hw.model'", "r");
2021-07-22 18:39:34 +02:00
#endif
FILE* host_model_info =
fopen("/sys/devices/virtual/dmi/id/board_name",
2021-12-18 23:24:58 +01:00
"r"); // try to get board name
if (!host_model_info)
host_model_info = fopen("/sys/devices/virtual/dmi/id/product_name",
"r"); // if couldn't then try another
if (!host_model_info) // if failed
host_model_info = fopen("/etc/hostname", "r"); // etc.
if (host_model_info) { // if succeeded to open one of the file
fgets(line, 256, host_model_info);
line[strlen(line) - 1] = '\0';
sprintf(user_info.host_model, "%s", line);
fclose(host_model_info);
}
#ifdef _WIN32
2021-11-01 16:48:54 +01:00
host_model_info = popen("wmic computersystem get model", "r");
while (fgets(line, sizeof(line), host_model_info)) {
2021-11-01 16:48:54 +01:00
if (strstr(line, "Model") != 0)
continue;
else {
sprintf(user_info.host_model, "%s", line);
user_info.host_model[strlen(user_info.host_model) - 2] = '\0';
2021-11-01 16:48:54 +01:00
break;
}
}
2021-11-04 15:33:04 +01:00
#elif defined(__FREEBSD__) || defined(__APPLE__)
#if defined(__FREEBSD__)
#define HOSTCTL "hw.hv_vendor"
#elif defined(__APPLE__)
#define HOSTCTL "hw.model"
#endif
host_model_info = popen("sysctl -a " HOSTCTL, "r");
2021-07-24 15:37:16 +02:00
while (fgets(line, sizeof(line), host_model_info))
2021-12-06 11:52:08 +01:00
if (sscanf(line, HOSTCTL ": %[^\n]", user_info.host_model)) break;
#endif // _WIN32
FILE* host_model_version =
fopen("/sys/devices/virtual/dmi/id/product_version", "r");
2021-05-03 03:46:55 +02:00
if (os_release) { // get normal vars
2021-11-01 16:48:54 +01:00
while (fgets(line, sizeof(line), os_release))
if (sscanf(line, "\nID=\"%s\"", user_info.version_name) ||
sscanf(line, "\nID=%s", user_info.version_name))
2021-11-01 16:48:54 +01:00
break;
2021-09-10 01:37:20 +02:00
// trying to detect amogos because in its os-release file ID value is
// just "debian"
if (strcmp(user_info.version_name, "debian") == 0 ||
strcmp(user_info.version_name, "raspbian") ==
0) // will be removed when amogos will have an os-release file
// with ID=amogos
2021-11-01 16:48:54 +01:00
{
DIR* amogos_plymouth = opendir("/usr/share/plymouth/themes/amogos");
if (amogos_plymouth) {
2021-11-01 16:48:54 +01:00
closedir(amogos_plymouth);
sprintf(user_info.version_name, "amogos");
2021-09-10 01:37:20 +02:00
}
2021-11-01 16:48:54 +01:00
}
if (host_model_version) {
while (fgets(line, sizeof(line), host_model_version))
if (sscanf(line, "%[^\n]", user_info.host_model_version)) break;
if (host_model_version) {
2021-11-01 16:48:54 +01:00
char version[32];
while (fgets(line, sizeof(line), host_model_version)) {
if (sscanf(line, "%[^\n]", version)) {
strcat(user_info.host_model_version, " ");
strcat(user_info.host_model_version, version);
2021-11-01 16:48:54 +01:00
break;
2021-09-10 00:51:37 +02:00
}
}
2021-06-13 13:57:28 +02:00
}
2021-05-03 03:46:55 +02:00
}
while (fgets(line, sizeof(line), cpuinfo)) {
2021-07-22 18:39:34 +02:00
#ifdef __FREEBSD__
if (sscanf(line, "hw.model: %[^\n]", user_info.cpu_model))
2021-07-22 18:39:34 +02:00
#else
if (sscanf(line, "model name : %[^\n]", user_info.cpu_model))
break;
2021-11-01 16:48:54 +01:00
#endif // __FREEBSD__
}
char* tmp_user = getenv("USER");
2021-10-03 23:48:10 +02:00
if (tmp_user == NULL)
sprintf(user_info.user, "%s", "");
2021-10-03 23:48:10 +02:00
else
sprintf(user_info.user, "%s", tmp_user);
2021-11-01 16:48:54 +01:00
fclose(os_release);
} else { // try for android vars, next for Apple var, or unknown system
DIR* system_app = opendir("/system/app/");
DIR* system_priv_app = opendir("/system/priv-app/");
DIR* library = opendir("/Library/");
if (system_app && system_priv_app) { // android
closedir(system_app);
closedir(system_priv_app);
sprintf(user_info.version_name, "android");
// android vars
FILE* whoami = popen("whoami", "r");
if (fscanf(whoami, "%s", user_info.user) == 3)
sprintf(user_info.user, "unknown");
fclose(whoami);
2021-11-01 16:55:33 +01:00
host_model_info = popen("getprop ro.product.model", "r");
while (fgets(line, sizeof(line), host_model_info))
if (sscanf(line, "%[^\n]", user_info.host_model)) break;
2021-07-22 18:39:34 +02:00
#ifndef __FREEBSD__
while (fgets(line, sizeof(line), cpuinfo))
if (sscanf(line, "Hardware : %[^\n]",
user_info.cpu_model))
break;
2021-07-22 18:39:34 +02:00
#endif
} else if (library) // Apple
2021-04-16 10:48:39 +02:00
{
closedir(library);
#ifdef __APPLE__
sysctlbyname("machdep.cpu.brand_string", &cpu_buffer,
&cpu_buffer_len, NULL, 0);
2021-04-15 22:59:43 +02:00
#ifndef __IPHONE__
sprintf(user_info.version_name, "macos");
#else
sprintf(user_info.version_name, "ios");
#endif
sprintf(user_info.cpu_model, "%s", cpu_buffer);
2021-04-16 10:48:39 +02:00
#endif
} else
sprintf(user_info.version_name, "unknown");
}
2021-07-22 18:39:34 +02:00
#ifndef __FREEBSD__
fclose(cpuinfo);
2021-07-22 18:39:34 +02:00
#endif
#ifndef _WIN32
gethostname(user_info.host, 256);
// #endif // _WIN32
char* tmp_shell = getenv("SHELL");
if (tmp_shell == NULL)
sprintf(user_info.shell, "%s", "");
else
sprintf(user_info.shell, "%s", tmp_shell);
if (strlen(user_info.shell) > 16)
memmove(&user_info.shell, &user_info.shell[27],
strlen(user_info.shell)); // android shell was too long, this
// works only for termux
2021-11-01 16:48:54 +01:00
#else
cpuinfo = popen("wmic cpu get caption", "r");
while (fgets(line, sizeof(line), cpuinfo)) {
2021-11-01 16:48:54 +01:00
if (strstr(line, "Caption") != 0)
continue;
else {
sprintf(user_info.cpu_model, "%s", line);
user_info.cpu_model[strlen(user_info.cpu_model) - 2] = '\0';
2021-11-01 16:48:54 +01:00
break;
}
}
FILE* user_host_fp = popen("wmic computersystem get username", "r");
while (fgets(line, sizeof(line), user_host_fp)) {
2021-11-01 16:48:54 +01:00
if (strstr(line, "UserName") != 0)
continue;
else {
sscanf(line, "%[^\\]%s", user_info.host, user_info.user);
memmove(user_info.user, user_info.user + 1,
sizeof(user_info.user) - 1);
2021-11-01 16:48:54 +01:00
break;
}
}
FILE* shell_fp = popen("powershell $PSVersionTable", "r");
sprintf(user_info.shell, "PowerShell ");
2021-11-01 16:48:54 +01:00
char tmp_shell[64];
while (fgets(line, sizeof(line), shell_fp))
if (sscanf(line, "PSVersion %s", tmp_shell) != 0)
break;
strcat(user_info.shell, tmp_shell);
#endif // _WIN32
// truncate CPU name
truncate_name(user_info.cpu_model, user_info.target_width);
2021-11-01 16:48:54 +01:00
// system resources
#ifndef _WIN32
uname(&user_info.sys_var);
#endif // _WIN32
2021-04-16 14:10:33 +02:00
#ifndef __APPLE__
#ifndef __FREEBSD__
#ifndef _WIN32
sysinfo(&user_info.sys); // somehow this function has to be called again in
// print_info()
#else // _WIN32
GetSystemInfo(&user_info.sys);
#endif // _WIN32
#endif
2021-04-16 14:10:33 +02:00
#endif
#ifndef _WIN32
truncate_name(user_info.sys_var.release, user_info.target_width);
sprintf(user_info.kernel, "%s %s %s", user_info.sys_var.sysname,
user_info.sys_var.release, user_info.sys_var.machine);
truncate_name(user_info.kernel, user_info.target_width);
#else // _WIN32
sprintf(user_info.version_name, "windows");
FILE* kernel_fp = popen("wmic computersystem get systemtype", "r");
while (fgets(line, sizeof(line), kernel_fp)) {
2021-11-01 16:48:54 +01:00
if (strstr(line, "SystemType") != 0)
continue;
else {
sprintf(user_info.kernel, "%s", line);
user_info.kernel[strlen(user_info.kernel) - 2] = '\0';
2021-11-01 16:48:54 +01:00
break;
}
}
if (kernel_fp != NULL) pclose(kernel_fp);
#endif // _WIN32
2021-11-01 16:48:54 +01:00
// ram
2021-04-15 22:59:43 +02:00
#ifndef __APPLE__
#ifdef _WIN32
FILE* mem_used_fp = popen("wmic os get freevirtualmemory", "r");
FILE* mem_total_fp = popen("wmic os get totalvirtualmemorysize", "r");
2021-11-01 16:48:54 +01:00
char mem_used_ch[2137] = {0}, mem_total_ch[2137] = {0};
while (fgets(mem_total_ch, sizeof(mem_total_ch), mem_total_fp) != NULL) {
2021-11-01 16:48:54 +01:00
if (strstr(mem_total_ch, "TotalVirtualMemorySize") != 0)
continue;
else if (strstr(mem_total_ch, " ") == 0)
continue;
else
user_info.ram_total = atoi(mem_total_ch) / 1024;
2021-11-01 16:48:54 +01:00
}
while (fgets(mem_used_ch, sizeof(mem_used_ch), mem_used_fp) != NULL) {
2021-11-01 16:48:54 +01:00
if (strstr(mem_used_ch, "FreeVirtualMemory") != 0)
continue;
else if (strstr(mem_used_ch, " ") == 0)
continue;
else
user_info.ram_used =
user_info.ram_total - (atoi(mem_used_ch) / 1024);
2021-11-01 16:48:54 +01:00
}
pclose(mem_used_fp);
pclose(mem_total_fp);
#else
FILE* meminfo;
#ifdef __FREEBSD__
2021-11-01 16:55:33 +01:00
meminfo = popen("LANG=EN_us freecolor -om 2> /dev/null", "r");
#else
2021-11-01 16:55:33 +01:00
meminfo = popen("LANG=EN_us free -m 2> /dev/null", "r");
#endif
while (fgets(line, sizeof(line), meminfo))
// free command prints like this: "Mem:" total used free shared
// buff/cache available
sscanf(line, "Mem: %d %d", &user_info.ram_total, &user_info.ram_used);
fclose(meminfo);
#endif
2021-04-15 22:59:43 +02:00
#else
// Used
2021-04-16 10:48:39 +02:00
FILE *mem_wired_fp, *mem_active_fp, *mem_compressed_fp;
mem_wired_fp =
popen("vm_stat | awk '/wired/ { printf $4 }' | cut -d '.' -f 1", "r");
mem_active_fp =
popen("vm_stat | awk '/active/ { printf $3 }' | cut -d '.' -f 1", "r");
mem_compressed_fp = popen(
"vm_stat | awk '/occupied/ { printf $5 }' | cut -d '.' -f 1", "r");
2021-04-15 22:59:43 +02:00
char mem_wired_ch[2137], mem_active_ch[2137], mem_compressed_ch[2137];
while (fgets(mem_wired_ch, sizeof(mem_wired_ch), mem_wired_fp) != NULL) {
while (fgets(mem_active_ch, sizeof(mem_active_ch), mem_active_fp) !=
NULL) {
while (fgets(mem_compressed_ch, sizeof(mem_compressed_ch),
mem_compressed_fp) != NULL) {
2021-04-16 10:48:39 +02:00
}
2021-04-15 22:59:43 +02:00
}
}
2021-04-16 10:48:39 +02:00
pclose(mem_wired_fp);
pclose(mem_active_fp);
pclose(mem_compressed_fp);
2021-04-15 22:59:43 +02:00
int mem_wired = atoi(mem_wired_ch);
int mem_active = atoi(mem_active_ch);
2021-04-15 22:59:43 +02:00
int mem_compressed = atoi(mem_compressed_ch);
// Total
sysctlbyname("hw.memsize", &mem_buffer, &mem_buffer_len, NULL, 0);
2021-12-06 11:52:08 +01:00
user_info.ram_used = ((mem_wired + mem_active + mem_compressed) * 4 / 1024);
user_info.ram_total = mem_buffer / 1024 / 1024;
2021-04-15 22:59:43 +02:00
#endif
/* ---------- gpu ---------- */
int gpun = 0; // number of the gpu that the program is searching for to put
// in the array
#ifndef _WIN32
setenv("LANG", "en_US", 1); // force language to english
#endif // _WIN32
FILE* gpu;
#ifndef _WIN32
2021-11-01 16:55:33 +01:00
gpu = popen("lshw -class display 2> /dev/null", "r");
2021-07-23 11:06:46 +02:00
// add all gpus to the array gpu_model
while (fgets(line, sizeof(line), gpu))
if (sscanf(line, " product: %[^\n]", user_info.gpu_model[gpun]))
gpun++;
#endif // _WIN32
if (strlen(user_info.gpu_model[0]) < 2) {
// get gpus with lspci command
if (strcmp(user_info.version_name, "android") != 0) {
2021-04-16 10:48:39 +02:00
#ifndef __APPLE__
#ifdef _WIN32
gpu = popen("wmic PATH Win32_VideoController GET Name", "r");
#else
gpu = popen("lspci -mm 2> /dev/null | grep \"VGA\" | awk -F '\"' "
"'{print $4 $5 $6}'",
"r");
#endif
#else
gpu = popen("system_profiler SPDisplaysDataType | awk -F ': ' "
"'/Chipset Model: /{ print $2 }'",
"r");
#endif
} else
2021-11-01 16:55:33 +01:00
gpu = popen("getprop ro.hardware.vulkan 2> /dev/null", "r");
}
// get all the gpus
while (fgets(line, sizeof(line), gpu)) {
if (strstr(line, "Name"))
continue;
else if (strlen(line) == 2)
continue;
// ^^^ for windows
else if (sscanf(line, "%[^\n]", user_info.gpu_model[gpun]))
gpun++;
}
fclose(gpu);
// truncate GPU name and remove square brackets
for (int i = 0; i < gpun; i++) {
remove_brackets(user_info.gpu_model[i]);
truncate_name(user_info.gpu_model[i], user_info.target_width);
}
// Resolution
#ifndef _WIN32
FILE* resolution =
popen("xwininfo -root 2> /dev/null | grep -E 'Width|Height'", "r");
while (fgets(line, sizeof(line), resolution)) {
sscanf(line, " Width: %d", &user_info.screen_width);
sscanf(line, " Height: %d", &user_info.screen_height);
2021-04-15 23:52:12 +02:00
}
#endif // _WIN32
2021-04-15 23:52:12 +02:00
if (strcmp(user_info.version_name, "windows"))
2021-11-01 19:44:02 +01:00
terminal_cursor_move = "\033[21C";
// package count
#ifdef _WIN32
user_info.pkgs = pkgman(&user_info, config_flags);
#else // _WIN32
user_info.pkgs = pkgman(&user_info);
#endif // _WIN32
uwu_kernel(user_info.kernel);
for (int i = 0; user_info.gpu_model[i][0]; i++)
uwu_hw(user_info.gpu_model[i]);
uwu_hw(user_info.cpu_model);
uwu_hw(user_info.host_model);
return user_info;
}
void list(char* arg) { // prints distribution list
// distributions are listed by distribution branch
// to make the output easier to understand by the user.
printf("%s -d <options>\n"
" Available distributions:\n"
" %sArch linux %sbased:\n"
" %sarch, arcolinux, %sartix, endeavouros %smanjaro, "
"manjaro-arm, %sxerolinux\n\n"
" %sDebian/%sUbuntu %sbased:\n"
" %samogos, debian, %slinuxmint, neon %spop, %sraspbian "
"%subuntu\n\n"
2021-05-03 01:04:18 +02:00
" %sBSD %sbased:\n"
2021-11-06 00:34:53 +01:00
" %sfreebsd, %sopenbsd, %sm%sa%sc%so%ss, %sios\n\n"
" %sOther/spare distributions:\n"
" %salpine, %sfedora, %sgentoo, %sslackware, %ssolus, %svoid, "
"opensuse-leap, android, %sgnu, guix, %swindows, %sunknown\n\n",
arg, BLUE, NORMAL, BLUE, MAGENTA, GREEN, BLUE, // Arch based colors
RED, YELLOW, NORMAL, RED, GREEN, BLUE, RED,
YELLOW, // Debian based colors
RED, NORMAL, RED, YELLOW, GREEN, YELLOW, RED, PINK, BLUE,
WHITE, // BSD/Apple colors
NORMAL, BLUE, BLUE, PINK, MAGENTA, WHITE, GREEN, YELLOW, BLUE,
WHITE); // Other/spare distributions colors
}
2021-07-23 00:32:09 +02:00
/*
This replaces all terms in a string with another term.
replace("Hello World!", "World", "everyone")
This returns "Hello everyone!".
*/
void replace(char* original, char* search, char* replacer) {
char* ch;
char buffer[1024];
while ((ch = strstr(original, search))) {
2021-10-03 23:54:59 +02:00
ch = strstr(original, search);
2021-10-03 23:48:10 +02:00
strncpy(buffer, original, ch - original);
buffer[ch - original] = 0;
sprintf(buffer + (ch - original), "%s%s", replacer,
ch + strlen(search));
2021-10-03 23:48:10 +02:00
original[0] = 0;
strcpy(original, buffer);
}
2021-07-23 00:32:09 +02:00
}
void replace_ignorecase(char* original, char* search, char* replacer) {
char* ch;
2021-10-03 23:48:10 +02:00
char buffer[1024];
#ifndef _WIN32
2021-10-03 23:54:59 +02:00
while ((ch = strcasestr(original, search)))
2021-11-01 16:48:54 +01:00
#else
while ((ch = strstr(original, search)))
#endif // _WIN32
2021-10-03 23:48:10 +02:00
{
strncpy(buffer, original, ch - original);
buffer[ch - original] = 0;
sprintf(buffer + (ch - original), "%s%s", replacer,
ch + strlen(search));
2021-10-03 23:48:10 +02:00
original[0] = 0;
strcpy(original, buffer);
}
2021-09-07 06:59:34 +02:00
}
void print_ascii(
struct info* user_info) { // prints logo (as ascii art) of the given system.
// distributions listed alphabetically.
2021-07-23 00:32:09 +02:00
printf("\n");
FILE* file;
2021-07-23 00:32:09 +02:00
char ascii_file[1024];
// First tries to get ascii art file from local directory. Good when
// modifying these files.
sprintf(ascii_file, "./res/ascii/%s.txt", user_info->version_name);
2021-11-01 16:55:33 +01:00
file = fopen(ascii_file, "r");
2021-07-23 00:32:09 +02:00
// Now tries to get file from normal directory
if (!file) {
if (strcmp(user_info->version_name, "android") == 0) {
sprintf(ascii_file,
"/data/data/com.termux/files/usr/lib/uwufetch/ascii/%s.txt",
user_info->version_name);
} else {
sprintf(ascii_file, "/usr/lib/uwufetch/ascii/%s.txt",
user_info->version_name);
2021-07-23 00:32:09 +02:00
}
2021-11-01 16:55:33 +01:00
file = fopen(ascii_file, "r");
if (!file) {
2021-07-23 00:32:09 +02:00
// Prevent infinite loops
if (strcmp(user_info->version_name, "unknown") == 0) {
2021-07-23 00:32:09 +02:00
printf("No\nunknown\nascii\nfile\n\n\n\n");
return;
}
sprintf(user_info->version_name, "unknown");
return print_ascii(user_info);
2021-07-23 00:32:09 +02:00
}
}
2021-07-23 01:33:04 +02:00
char line[256];
while (fgets(line, 256, file)) {
2021-07-23 00:32:09 +02:00
replace(line, "{NORMAL}", NORMAL);
replace(line, "{BOLD}", BOLD);
replace(line, "{BLACK}", BLACK);
replace(line, "{RED}", RED);
replace(line, "{GREEN}", GREEN);
2021-11-01 16:48:54 +01:00
replace(line, "{SPRING_GREEN}", SPRING_GREEN);
2021-07-23 00:32:09 +02:00
replace(line, "{YELLOW}", YELLOW);
replace(line, "{BLUE}", BLUE);
replace(line, "{MAGENTA}", MAGENTA);
replace(line, "{CYAN}", CYAN);
replace(line, "{WHITE}", WHITE);
replace(line, "{PINK}", PINK);
replace(line, "{LPINK}", LPINK);
2021-11-01 16:55:33 +01:00
// For manjaro and amogos and windows
#ifdef _WIN32
2021-11-01 16:48:54 +01:00
replace(line, "{BLOCK}", "\xdc");
replace(line, "{BLOCK_VERTICAL}", "\xdb");
#else // _WIN32
2021-11-01 16:55:33 +01:00
replace(line, "{BLOCK}", "\u2584");
replace(line, "{BLOCK_VERTICAL}", "\u2587");
#endif // _WIN32
2021-07-23 01:33:04 +02:00
replace(line, "{BACKGROUND_GREEN}", "\e[0;42m");
2021-09-10 00:51:37 +02:00
replace(line, "{BACKGROUND_RED}", "\e[0;41m");
replace(line, "{BACKGROUND_WHITE}", "\e[0;47m");
printf("%s", line);
2021-07-23 00:32:09 +02:00
}
// Always set color to NORMAL, so there's no need to do this in every ascii
// file.
2021-07-23 00:32:09 +02:00
printf(NORMAL);
fclose(file);
}
void print_image(
struct info* user_info) { // prints logo (as an image) of the given system.
// distributions listed alphabetically.
#ifndef __IPHONE__
char command[256];
if (strlen(user_info->image_name) > 1)
sprintf(command, "viu -t -w 18 -h 8 %s 2> /dev/null",
user_info->image_name);
else {
if (strcmp(user_info->version_name, "android") == 0)
sprintf(command,
"viu -t -w 18 -h 8 "
"/data/data/com.termux/files/usr/lib/uwufetch/%s.png 2> "
"/dev/null",
user_info->version_name);
else
sprintf(command,
"viu -t -w 18 -h 8 /usr/lib/uwufetch/%s.png 2> /dev/null",
user_info->version_name);
}
printf("\n");
if (system(command) !=
0) { // if viu is not installed or the image is missing
printf("\033[0E\033[3C%s\n"
" There was an\n"
" error: viu\n"
" is not installed\n"
" or the image\n"
2021-09-20 11:44:46 +02:00
" is not found\n"
" Read IMAGES.md\n"
" for more info.\n\n",
RED);
}
2021-11-06 00:34:53 +01:00
#else
// unfortunately, the iOS stdlib does not have
// system();
// because it reports that it is not available under iOS during compilation
2021-11-06 00:34:53 +01:00
printf("\033[0E\033[3C%s\n"
" There was an\n"
" error: images\n"
" are currently\n"
" disabled on iOS.\n\n",
RED);
2021-11-06 00:34:53 +01:00
#endif
}
void usage(char* arg) {
printf("Usage: %s <args>\n"
" -a, --ascii prints logo as ascii text (default)\n"
" -c --config use custom config path\n"
" -d, --distro lets you choose the logo to print\n"
" -h, --help prints this help page\n"
#ifndef __IPHONE__
" -i, --image prints logo as image and use a custom "
"image if provided\n"
" %sworks in most terminals\n"
2021-11-06 00:34:53 +01:00
#else
" -i, --image prints logo as image and use a custom "
"image if provided\n"
2021-11-06 00:34:53 +01:00
" %sdisabled under iOS\n"
#endif
" read README.md for more info%s\n"
" -l, --list lists all supported distributions\n"
" -w, --write-cache writes to the cache file "
"(~/.cache/uwufetch.cache)\n"
" using the cache set $UWUFETCH_CACHE_ENABLED to TRUE, true "
"or 1\n",
2021-11-06 00:34:53 +01:00
arg,
#ifndef __IPHONE__
2021-11-06 00:34:53 +01:00
BLUE,
#else
RED,
#endif
NORMAL);
}
#ifdef _WIN32
// windows sucks and hasn't a strstep, so I copied one from
// https://stackoverflow.com/questions/8512958/is-there-a-windows-variant-of-strsep-function
char* strsep(char** stringp, const char* delim) {
char* start = *stringp;
char* p;
2021-11-01 16:48:54 +01:00
p = (start != NULL) ? strpbrk(start, delim) : NULL;
if (p == NULL)
*stringp = NULL;
else {
*p = '\0';
2021-11-01 16:48:54 +01:00
*stringp = p + 1;
}
return start;
}
#endif
void uwu_kernel(char* kernel) {
2021-07-22 18:39:34 +02:00
#define KERNEL_TO_UWU(str, original, uwufied) \
if (strcmp(str, original) == 0) sprintf(str, "%s", uwufied)
2021-07-04 19:38:10 +02:00
char* temp_kernel = kernel;
char* token;
2021-07-04 19:38:10 +02:00
char splitted[16][128] = {};
int count = 0;
while ((token = strsep(&temp_kernel, " "))) {
2021-07-04 19:38:10 +02:00
strcpy(splitted[count], token);
count++;
}
strcpy(kernel, "");
for (int i = 0; i < 16; i++) {
2021-07-04 19:54:28 +02:00
// kernel name
2021-07-04 19:38:10 +02:00
KERNEL_TO_UWU(splitted[i], "Linux", "Linuwu");
2021-07-23 22:15:31 +02:00
else KERNEL_TO_UWU(splitted[i], "linux", "linuwu");
else KERNEL_TO_UWU(splitted[i], "alpine", "Nyalpine");
2021-09-10 00:51:37 +02:00
else KERNEL_TO_UWU(splitted[i], "amogos", "AmogOwOS");
2021-07-23 22:15:31 +02:00
else KERNEL_TO_UWU(splitted[i], "arch", "Nyarch Linuwu");
else KERNEL_TO_UWU(splitted[i], "artix", "Nyartix Linuwu");
else KERNEL_TO_UWU(splitted[i], "debian", "Debinyan");
else KERNEL_TO_UWU(splitted[i], "endeavouros", "endeavOwO");
else KERNEL_TO_UWU(splitted[i], "EndeavourOS", "endeavOwO");
else KERNEL_TO_UWU(splitted[i], "fedora", "Fedowa");
else KERNEL_TO_UWU(splitted[i], "gentoo", "GentOwO");
else KERNEL_TO_UWU(splitted[i], "gnu", "gnUwU");
else KERNEL_TO_UWU(splitted[i], "guix", "gnUwU gUwUix");
else KERNEL_TO_UWU(splitted[i], "linuxmint", "LinUWU Miwint");
else KERNEL_TO_UWU(splitted[i], "manjaro", "Myanjawo");
else KERNEL_TO_UWU(splitted[i], "manjaro-arm", "Myanjawo AWM");
else KERNEL_TO_UWU(splitted[i], "neon", "KDE NeOwOn");
else KERNEL_TO_UWU(splitted[i], "nixos", "nixOwOs");
else KERNEL_TO_UWU(splitted[i], "opensuse-leap", "OwOpenSUSE Leap");
else KERNEL_TO_UWU(splitted[i], "opensuse-tumbleweed",
"OwOpenSUSE Tumbleweed");
2021-07-23 22:15:31 +02:00
else KERNEL_TO_UWU(splitted[i], "pop", "PopOwOS");
else KERNEL_TO_UWU(splitted[i], "raspbian", "RaspNyan");
else KERNEL_TO_UWU(splitted[i], "slackware", "Swackwawe");
else KERNEL_TO_UWU(splitted[i], "solus", "sOwOlus");
else KERNEL_TO_UWU(splitted[i], "ubuntu", "Uwuntu");
else KERNEL_TO_UWU(splitted[i], "void", "OwOid");
2021-09-29 21:47:47 +02:00
else KERNEL_TO_UWU(splitted[i], "xerolinux", "xuwulinux");
else KERNEL_TO_UWU(
splitted[i], "android",
"Nyandroid"); // android at the end because it could be not
// considered as an actual distribution of gnu/linux
2021-07-04 19:54:28 +02:00
// BSD
2021-07-23 22:15:31 +02:00
else KERNEL_TO_UWU(splitted[i], "freebsd", "FweeBSD");
else KERNEL_TO_UWU(splitted[i], "openbsd", "OwOpenBSD");
2021-11-06 00:34:53 +01:00
//// Apple family
2021-07-23 22:15:31 +02:00
else KERNEL_TO_UWU(splitted[i], "macos", "macOwOS");
2021-11-06 00:34:53 +01:00
else KERNEL_TO_UWU(splitted[i], "ios", "iOwOS");
2021-07-04 19:54:28 +02:00
// Windows
2021-07-23 22:15:31 +02:00
else KERNEL_TO_UWU(splitted[i], "windows", "WinyandOwOws");
2021-07-04 19:54:28 +02:00
if (i != 0) strcat(kernel, " ");
2021-07-04 19:38:10 +02:00
strcat(kernel, splitted[i]);
}
2021-07-22 18:39:34 +02:00
#undef KERNEL_TO_UWU
2021-07-04 19:38:10 +02:00
}
void uwu_hw(char* hwname) {
2021-09-10 00:51:37 +02:00
#define HW_TO_UWU(original, uwuified) \
replace_ignorecase(hwname, original, uwuified);
2021-09-10 00:51:37 +02:00
HW_TO_UWU("lenovo", "LenOwO")
HW_TO_UWU("cpu", "CC\bPUwU"); // for some reasons this caused a segfault,
// using a \b char fixes it
2021-10-04 14:14:58 +02:00
HW_TO_UWU("gpu", "GG\bPUwU")
2021-09-10 00:51:37 +02:00
HW_TO_UWU("graphics", "Gwaphics")
HW_TO_UWU("corporation", "COwOpowation")
HW_TO_UWU("nvidia", "NyaVIDIA")
HW_TO_UWU("mobile", "Mwobile")
HW_TO_UWU("intel", "Inteww")
HW_TO_UWU("radeon", "Radenyan")
HW_TO_UWU("geforce", "GeFOwOce")
HW_TO_UWU("raspberry", "Nyasberry")
HW_TO_UWU("broadcom", "Bwoadcom")
HW_TO_UWU("motorola", "MotOwOwa")
HW_TO_UWU("proliant", "ProLinyant")
HW_TO_UWU("poweredge", "POwOwEdge")
2021-10-04 14:14:58 +02:00
HW_TO_UWU("apple", "Nyaa\bpple")
2021-09-10 00:51:37 +02:00
HW_TO_UWU("electronic", "ElectrOwOnic")
2021-09-07 06:59:34 +02:00
#undef HW_TO_UWU
}
void uwu_name(struct configuration* config_flags,
struct info* user_info) { // uwufies distro name
#define STRING_TO_UWU(original, uwufied) \
if (strcmp(user_info->version_name, original) == 0) \
sprintf(user_info->version_name, "%s", uwufied)
// linux
STRING_TO_UWU("alpine", "Nyalpine");
2021-09-10 00:51:37 +02:00
else STRING_TO_UWU("amogos", "AmogOwOS");
else STRING_TO_UWU("arch", "Nyarch Linuwu");
2021-08-29 13:14:01 +02:00
else STRING_TO_UWU("arcolinux", "ArcOwO Linuwu");
else STRING_TO_UWU("artix", "Nyartix Linuwu");
else STRING_TO_UWU("debian", "Debinyan");
else STRING_TO_UWU("endeavouros", "endeavOwO");
2021-05-07 00:21:05 +02:00
else STRING_TO_UWU("EndeavourOS", "endeavOwO");
else STRING_TO_UWU("fedora", "Fedowa");
else STRING_TO_UWU("gentoo", "GentOwO");
else STRING_TO_UWU("gnu", "gnUwU");
else STRING_TO_UWU("guix", "gnUwU gUwUix");
else STRING_TO_UWU("linuxmint", "LinUWU Miwint");
else STRING_TO_UWU("manjaro", "Myanjawo");
2021-07-22 21:05:36 +02:00
else STRING_TO_UWU("manjaro-arm", "Myanjawo AWM");
else STRING_TO_UWU("neon", "KDE NeOwOn");
else STRING_TO_UWU("nixos", "nixOwOs");
2021-07-22 21:05:36 +02:00
else STRING_TO_UWU("opensuse-leap", "OwOpenSUSE Leap");
else STRING_TO_UWU("opensuse-tumbleweed", "OwOpenSUSE Tumbleweed");
else STRING_TO_UWU("pop", "PopOwOS");
else STRING_TO_UWU("raspbian", "RaspNyan");
else STRING_TO_UWU("slackware", "Swackwawe");
else STRING_TO_UWU("solus", "sOwOlus");
else STRING_TO_UWU("ubuntu", "Uwuntu");
2021-07-22 21:05:36 +02:00
else STRING_TO_UWU("void", "OwOid");
2021-09-29 21:47:47 +02:00
else STRING_TO_UWU("xerolinux", "xuwulinux");
else STRING_TO_UWU(
"android",
"Nyandroid"); // android at the end because it could be not considered
// as an actual distribution of gnu/linux
// BSD
else STRING_TO_UWU("freebsd", "FweeBSD");
else STRING_TO_UWU("openbsd", "OwOpenBSD");
2021-11-06 00:34:53 +01:00
//// Apple family
2021-04-15 22:59:43 +02:00
else STRING_TO_UWU("macos", "macOwOS");
2021-11-06 00:34:53 +01:00
else STRING_TO_UWU("ios", "iOwOS");
2021-04-15 22:59:43 +02:00
2021-05-03 03:46:55 +02:00
// Windows
else STRING_TO_UWU("windows", "WinyandOwOws");
else {
sprintf(user_info->version_name, "%s", "unknown");
if (config_flags->ascii_image_flag == 1) {
print_image(user_info);
printf("\n");
}
}
#undef STRING_TO_UWU
}
void truncate_name(char* name, int target_width) {
2021-07-22 18:39:34 +02:00
char arr[target_width];
for (int i = 0; i < target_width; i++) arr[i] = name[i];
2021-07-22 18:39:34 +02:00
name = arr;
}
// remove square brackets (for gpu names)
void remove_brackets(char* str) {
int i = 0, j;
while (i < (int)strlen(str)) {
if (str[i] == '[' || str[i] == ']') {
for (j = i; j < (int)strlen(str); j++) str[j] = str[j + 1];
} else
i++;
}
2021-07-22 15:31:34 +02:00
}