chsrc/xy.h

541 lines
12 KiB
C
Raw Normal View History

2023-09-09 16:57:06 +08:00
/** ------------------------------------------------------------
* File : xy.h
* License : MIT
* Authors : Aoran Zeng <ccmywish@qq.com>
* Created on : <2023-08-28>
2023-09-16 22:10:54 +08:00
* Last modified : <2023-09-16>
2023-09-09 16:57:06 +08:00
*
* xy:
*
2023-09-16 22:10:54 +08:00
* y = f(x)
2023-09-09 16:57:06 +08:00
*
* Corss-Platform C utilities in Ruby flavor
*
2023-09-16 22:10:54 +08:00
* MIT LICENSE.txt
2023-09-09 16:57:06 +08:00
* ------------------------------------------------------------*/
2023-08-28 22:21:14 +08:00
2023-08-30 15:34:56 +08:00
#ifndef XY_H
#define XY_H
2023-09-03 14:56:49 +08:00
#include <stdio.h>
#include <stdarg.h>
2023-08-28 22:21:14 +08:00
#include <string.h>
#include <stdlib.h>
#include <stdbool.h>
2023-08-30 15:34:56 +08:00
#include <assert.h>
#include <stddef.h>
2023-09-10 14:50:41 +08:00
#include <unistd.h> // For access()
2023-08-30 15:34:56 +08:00
// #define NDEBUG
2023-08-28 22:21:14 +08:00
2023-08-29 21:58:51 +08:00
#ifdef _WIN32
2023-08-30 17:09:59 +08:00
static bool xy_on_windows = true;
static bool xy_on_linux = false;
static bool xy_on_macos = false;
2023-09-06 19:45:37 +08:00
static bool xy_on_bsd = false;
2023-08-30 17:09:59 +08:00
2023-09-03 12:06:42 +08:00
static char* xy_os_devnull = "nul";
2023-08-29 21:58:51 +08:00
#include <windows.h>
#define xy_useutf8() SetConsoleOutputCP(65001)
2023-08-30 17:09:59 +08:00
2023-09-10 14:50:41 +08:00
#elif defined(__linux__) || defined(__linux)
2023-08-30 17:09:59 +08:00
static bool xy_on_windows = false;
static bool xy_on_linux = true;
static bool xy_on_macos = false;
2023-09-06 19:45:37 +08:00
static bool xy_on_bsd = false;
2023-08-30 17:09:59 +08:00
static char* xy_os_devnull = "/dev/null";
2023-09-03 12:06:42 +08:00
#define xy_useutf8()
2023-09-13 22:41:16 +08:00
#elif defined(__APPLE__)
static bool xy_on_windows = false;
static bool xy_on_linux = false;
static bool xy_on_macos = true;
2023-09-06 19:45:37 +08:00
static bool xy_on_bsd = false;
2023-09-02 21:36:54 +08:00
2023-09-13 22:41:16 +08:00
static char* xy_os_devnull = "/dev/null";
2023-09-03 12:06:42 +08:00
2023-08-29 21:58:51 +08:00
#define xy_useutf8()
#elif defined(__OpenBSD__) || defined(__NetBSD__) || defined(__FreeBSD__)
static bool xy_on_windows = false;
static bool xy_on_linux = false;
static bool xy_on_macos = false;
2023-09-06 19:45:37 +08:00
static bool xy_on_bsd = true;
2023-09-10 14:50:41 +08:00
static char* xy_os_devnull = "/dev/null";
2023-09-03 14:56:49 +08:00
#define xy_useutf8()
2023-08-29 21:58:51 +08:00
#endif
void putf(double n) {printf("%f\n", n);}
void puti(long long n) {printf("%lld\n", n);}
void putb(bool n) {if(n) puts("true"); else puts("false");}
2023-08-30 17:09:59 +08:00
2023-08-31 21:48:05 +08:00
#define xy_arylen(x) (sizeof(x) / sizeof(x[0]))
2023-08-28 22:21:14 +08:00
2023-08-30 09:50:38 +08:00
static inline void*
xy_malloc0 (size_t size)
{
void* ptr = malloc(size);
memset(ptr, 0, size);
return ptr;
}
2023-09-05 15:36:26 +08:00
#define XY_LOG_INFO 00001
#define XY_LOG_SUCCESS 00001<<1
#define XY_LOG_WARN 00001<<2
#define XY_LOG_ERROR 00001<<3
#define xy_success(str) xy_log_ (XY_LOG_SUCCESS, str)
#define xy_info(str) xy_log_ (XY_LOG_INFO, str)
#define xy_warn(str) xy_log_ (XY_LOG_WARN, str)
#define xy_error(str) xy_log_ (XY_LOG_ERROR, str)
static void
2023-09-05 15:36:26 +08:00
xy_log_ (int level, const char* str)
{
char* color_fmt_str = NULL;
bool to_stderr = false;
2023-09-05 15:36:26 +08:00
if (level & XY_LOG_INFO) {
color_fmt_str = "\033[34m%s\033[0m"; // 蓝色
}
2023-09-05 15:36:26 +08:00
else if (level & XY_LOG_SUCCESS) {
color_fmt_str = "\033[32m%s\033[0m"; // 绿色
}
2023-09-05 15:36:26 +08:00
else if (level & XY_LOG_WARN) {
color_fmt_str = "\033[33m%s\033[0m\n"; // 黄色
to_stderr = true;
}
2023-09-05 15:36:26 +08:00
else if (level & XY_LOG_ERROR) {
color_fmt_str = "\033[31m%s\033[0m\n"; // 红色
to_stderr = true;
}
else {
//xy_assert ("CAN'T REACH!");
}
// -2 把中间%s减掉
size_t len = strlen(color_fmt_str) -2;
char* buf = malloc(strlen(str) + len + 1);
sprintf (buf, color_fmt_str, str);
if (to_stderr) {
fprintf(stderr, "%s",buf);
} else {
puts(buf);
}
free(buf);
}
2023-08-30 20:06:28 +08:00
/**
2023-09-01 22:23:03 +08:00
* str中所有的src字符替换成dest,
*
2023-08-30 20:06:28 +08:00
*/
static char*
xy_strch (const char* str, char src,const char* dest)
{
size_t str_len = strlen(str);
size_t dest_len = strlen(dest);
size_t size = str_len*dest_len;
char* ret = (char*)malloc(size);
int i=0;
int j=0;
while(i<str_len) {
if(str[i]==src) {
int k=0;
while(k<dest_len){
ret[j++] = dest[k++];
}
i++;
}
else {
ret[j++] = str[i++];
}
}
ret[j] = 0;
return ret;
}
static char*
xy_2strjoin (const char* str1, const char* str2)
{
size_t len = strlen(str1);
size_t size = len + strlen(str2) + 1;
char* ret = malloc(size);
strcpy(ret, str1);
strcpy(ret+len, str2);
return ret;
}
static char*
xy_strjoin (unsigned int count, ...)
{
size_t al_fixed = 128;
char* ret = calloc(1, al_fixed);
// 已分配次数
int al_times = 1;
// 当前已分配量
size_t al_cur = al_fixed;
const char* str = NULL;
// 需要分配的量
size_t al_need = 0;
// 用于 strcpy() 到 ret 的哪个位置
char* cur = ret + 0;
va_list args;
va_start(args, count);
for(int i=0; i<count; i++)
{
// 是否需要重新分配
bool need_realloc = false;
str = va_arg(args, const char*);
al_need += strlen(str);
while (al_need > al_cur) {
al_times += 1; al_cur = al_times * al_fixed;
need_realloc = true;
}
// printf("al_times %d, al_need %zd, al_cur %zd\n", al_times, al_need, al_cur);
if (need_realloc) {
ptrdiff_t diff = cur - ret;
ret = realloc(ret, al_cur);
cur = ret + diff;
}
if (NULL==ret) {
xy_error ("xy: No availble memory!"); return NULL;
}
strcpy(cur, str);
// puts(ret);
cur += strlen(str);
}
va_end(args);
*cur = '\0';
return ret;
}
2023-08-31 16:54:06 +08:00
2023-09-10 14:50:41 +08:00
static char*
2023-09-04 15:10:18 +08:00
xy_strdup(const char* str)
{
size_t len = strlen(str);
char* new = xy_malloc0(len+1);
strcpy(new, str);
return new;
2023-09-04 15:10:18 +08:00
}
2023-09-05 15:15:55 +08:00
#define XY_STR_BOLD 1
#define XY_STR_FAINT 2
#define XY_STR_ITALIC 3
#define XY_STR_UNDERLINE 4
#define XY_STR_BLINK 5
#define XY_STR_CROSS 9
#define xy_str_to_bold(str) xy_str_to_terminal_style_(XY_STR_BOLD, str)
#define xy_str_to_faint(str) xy_str_to_terminal_style_(XY_STR_FAINT, str)
#define xy_str_to_italic(str) xy_str_to_terminal_style_(XY_STR_ITALIC, str)
#define xy_str_to_underline(str) xy_str_to_terminal_style_(XY_STR_UNDERLINE,str)
#define xy_str_to_blink(str) xy_str_to_terminal_style_(XY_STR_BLINK, str)
#define xy_str_to_cross(str) xy_str_to_terminal_style_(XY_STR_CROSS, str)
#define XY_STR_RED 31
#define XY_STR_GREEN 32
#define XY_STR_YELLOW 33
#define XY_STR_BLUE 34
#define XY_STR_MAGENTA 35
#define XY_STR_CYAN 36
#define xy_str_to_red(str) xy_str_to_terminal_style_(XY_STR_RED, str)
#define xy_str_to_green(str) xy_str_to_terminal_style_(XY_STR_GREEN, str)
#define xy_str_to_yellow(str) xy_str_to_terminal_style_(XY_STR_YELLOW, str)
#define xy_str_to_blue(str) xy_str_to_terminal_style_(XY_STR_BLUE, str)
#define xy_str_to_magenta(str) xy_str_to_terminal_style_(XY_STR_MAGENTA,str)
#define xy_str_to_purple xy_str_to_magenta
#define xy_str_to_cyan(str) xy_str_to_terminal_style_(XY_STR_CYAN, str)
2023-09-10 14:50:41 +08:00
static char*
2023-09-05 15:15:55 +08:00
xy_str_to_terminal_style_(int style, const char* str)
{
char* color_fmt_str = NULL;
if (XY_STR_RED==style)
{
color_fmt_str = "\e[31m%s\e[0m"; // 红色
}
else if (XY_STR_GREEN==style)
{
color_fmt_str = "\e[32m%s\e[0m"; // 绿色
}
else if (XY_STR_YELLOW==style)
{
color_fmt_str = "\e[33m%s\e[0m"; // 黄色
}
else if (XY_STR_BLUE==style)
{
color_fmt_str = "\e[34m%s\e[0m"; // 蓝色
}
else if (XY_STR_MAGENTA==style)
{
color_fmt_str = "\e[35m%s\e[0m"; // 蓝色
}
else if (XY_STR_CYAN==style)
{
color_fmt_str = "\e[36m%s\e[0m"; // 蓝色
}
else if (XY_STR_BOLD==style)
{
color_fmt_str = "\e[1m%s\e[0m";
}
else if (XY_STR_FAINT==style)
{
color_fmt_str = "\e[2m%s\e[0m";
}
else if (XY_STR_ITALIC==style)
{
color_fmt_str = "\e[3m%s\e[0m";
}
else if (XY_STR_UNDERLINE==style)
{
color_fmt_str = "\e[4m%s\e[0m";
}
else if (XY_STR_BLINK==style)
{
color_fmt_str = "\e[5m%s\e[0m";
}
else if (XY_STR_CROSS==style)
{
color_fmt_str = "\e[9m%s\e[0m";
}
// -2 把中间%s减掉
size_t len = strlen(color_fmt_str) -2;
char* buf = malloc(strlen(str) + len + 1);
sprintf (buf, color_fmt_str, str);
return buf;
}
2023-09-10 14:50:41 +08:00
static bool
2023-08-31 16:54:06 +08:00
xy_streql(const char* str1, const char* str2) {
return strcmp(str1, str2) == 0 ? true : false;
}
2023-09-02 21:36:54 +08:00
2023-09-10 14:50:41 +08:00
static char*
2023-09-02 21:36:54 +08:00
xy_str_to_quietcmd (const char* cmd)
{
char* ret = NULL;
#ifdef _WIN32
ret = xy_2strjoin (cmd, " >nul 2>nul ");
2023-09-02 21:36:54 +08:00
#else
ret = xy_2strjoin (cmd, " 1>/dev/null 2>&1 ");
2023-09-02 21:36:54 +08:00
#endif
2023-09-04 15:10:18 +08:00
return ret;
2023-09-02 21:36:54 +08:00
}
2023-09-04 21:15:16 +08:00
2023-09-10 14:50:41 +08:00
static bool
2023-09-04 21:15:16 +08:00
xy_str_end_with (const char* str, const char* suffix)
{
size_t len1 = strlen(str);
size_t len2 = strlen(suffix);
if (0==len2) return true; // 空字符串直接返回
if (len1 < len2) return false;
const char* cur1 = str + len1 - 1;
const char* cur2 = suffix + len2 - 1;
for (int i=0; i<len2; i++)
{
if (*cur1 != *cur2) return false;
cur1--; cur2--;
}
return true;
}
2023-09-10 14:50:41 +08:00
static bool
xy_str_start_with (const char* str, const char* prefix)
{
size_t len1 = strlen(str);
size_t len2 = strlen(prefix);
if (0==len2) return true; // 空字符串直接返回
if (len1 < len2) return false;
const char* cur1 = str;
const char* cur2 = prefix;
for (int i=0; i<len2; i++)
{
if (*cur1 != *cur2) return false;
cur1++; cur2++;
}
return true;
}
2023-09-10 14:50:41 +08:00
static char*
xy_str_delete_prefix (const char* str, const char* prefix)
{
char* new = xy_strdup(str);
bool yes = xy_str_start_with(str, prefix);
if (!yes) return new;
size_t len = strlen(prefix);
char* cur = new + len;
return cur;
}
2023-09-04 21:21:36 +08:00
2023-09-10 14:50:41 +08:00
static char*
2023-09-04 21:21:36 +08:00
xy_str_delete_suffix (const char* str, const char* suffix)
{
char* new = xy_strdup(str);
bool yes = xy_str_end_with(str, suffix);
if (!yes) return new;
size_t len1 = strlen(str);
size_t len2 = strlen(suffix);
char* cur = new + len1 - len2;
*cur = '\0';
return new;
}
2023-09-05 11:09:35 +08:00
2023-09-10 14:50:41 +08:00
static char*
2023-09-05 11:09:35 +08:00
xy_str_strip (const char* str)
{
const char* lf = "\n";
const char* crlf = "\r\n";
char* new = xy_strdup(str);
while (xy_str_start_with(new, lf)) {
new = xy_str_delete_prefix(new, lf);
}
while (xy_str_start_with(new, crlf)) {
new = xy_str_delete_prefix(new, crlf);
}
while (xy_str_end_with(new, lf)) {
new = xy_str_delete_suffix(new, lf);
}
while (xy_str_end_with(new, crlf)) {
new = xy_str_delete_suffix(new, crlf);
}
return new;
}
2023-09-10 13:15:34 +08:00
/**
* cmd后拿到cmd的执行结果 free掉这段内存
*
*/
2023-09-10 14:50:41 +08:00
static char *
2023-09-05 20:48:38 +08:00
xy_getcmd(const char * cmd, bool (*func)(const char*))
{
const int BUFSIZE = 1024;
FILE *stream;
char* buf = (char*)malloc(sizeof(char)*BUFSIZE);
// 执行命令,并将输出保存到 stream 指针指向的文件中。
stream = popen(cmd, "r");
if (stream == NULL) {
2023-09-06 15:59:53 +08:00
printf("命令执行失败。\n");
return NULL;
}
// 从 stream 指针指向的文件中读取数据。
char *ret;
do {
2023-09-06 15:59:53 +08:00
if(fgets(buf, sizeof(buf), stream)==NULL)
{
break;
}
if(func==NULL)
{
ret = buf;
}
else
{
if(func(buf))
{
ret = buf;
break;
}
}
}while(1);
// 关闭 stream 指针。
pclose(stream);
2023-09-05 20:48:38 +08:00
return ret;
}
2023-09-10 13:31:48 +08:00
#define xy_os_home _xy_os_home()
2023-09-10 14:50:41 +08:00
static char*
2023-09-10 13:31:48 +08:00
_xy_os_home ()
{
char* home = NULL;
if (xy_on_windows)
home = getenv("USERPROFILE");
else
home = getenv("HOME");
return home;
}
#define xy_win_powershell_profile _xy_win_powershell_profile()
#define xy_win_powershellv5_profile _xy_win_powershellv5_profile()
2023-09-10 14:50:41 +08:00
static char*
2023-09-10 13:31:48 +08:00
_xy_win_powershell_profile ()
{
return xy_2strjoin(xy_os_home, "\\Documents\\PowerShell\\Microsoft.PowerShell_profile.ps1");
}
char*
_xy_win_powershellv5_profile()
{
return xy_2strjoin(xy_os_home, "\\Documents\\WindowsPowerShell\\Microsoft.PowerShell_profile.ps1");
}
2023-09-10 13:15:34 +08:00
/**
* @note Windows上`path` access()
*/
2023-09-10 14:50:41 +08:00
static bool
2023-09-10 13:15:34 +08:00
xy_file_exist(char* path)
{
char* newpath = path;
if (xy_on_windows)
{
if (xy_str_start_with(path, "~")) {
2023-09-10 13:31:48 +08:00
newpath = xy_2strjoin(xy_os_home, path+1);
2023-09-10 13:15:34 +08:00
}
}
return access(newpath, 0) ? false : true;
}
2023-09-05 20:48:38 +08:00
2023-08-30 15:34:56 +08:00
#endif