hbase.c 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519
  1. #include "hbase.h"
  2. #ifdef OS_DARWIN
  3. #include <mach-o/dyld.h> // for _NSGetExecutablePath
  4. #endif
  5. #include "hatomic.h"
  6. #ifndef RAND_MAX
  7. #define RAND_MAX 2147483647
  8. #endif
  9. static hatomic_t s_alloc_cnt = HATOMIC_VAR_INIT(0);
  10. static hatomic_t s_free_cnt = HATOMIC_VAR_INIT(0);
  11. long hv_alloc_cnt() {
  12. return s_alloc_cnt;
  13. }
  14. long hv_free_cnt() {
  15. return s_free_cnt;
  16. }
  17. void* hv_malloc(size_t size) {
  18. hatomic_inc(&s_alloc_cnt);
  19. void* ptr = malloc(size);
  20. if (!ptr) {
  21. fprintf(stderr, "malloc failed!\n");
  22. exit(-1);
  23. }
  24. return ptr;
  25. }
  26. void* hv_realloc(void* oldptr, size_t newsize, size_t oldsize) {
  27. hatomic_inc(&s_alloc_cnt);
  28. if (oldptr) hatomic_inc(&s_free_cnt);
  29. void* ptr = realloc(oldptr, newsize);
  30. if (!ptr) {
  31. fprintf(stderr, "realloc failed!\n");
  32. exit(-1);
  33. }
  34. if (newsize > oldsize) {
  35. memset((char*)ptr + oldsize, 0, newsize - oldsize);
  36. }
  37. return ptr;
  38. }
  39. void* hv_calloc(size_t nmemb, size_t size) {
  40. hatomic_inc(&s_alloc_cnt);
  41. void* ptr = calloc(nmemb, size);
  42. if (!ptr) {
  43. fprintf(stderr, "calloc failed!\n");
  44. exit(-1);
  45. }
  46. return ptr;
  47. }
  48. void* hv_zalloc(size_t size) {
  49. hatomic_inc(&s_alloc_cnt);
  50. void* ptr = malloc(size);
  51. if (!ptr) {
  52. fprintf(stderr, "malloc failed!\n");
  53. exit(-1);
  54. }
  55. memset(ptr, 0, size);
  56. return ptr;
  57. }
  58. void hv_free(void* ptr) {
  59. if (ptr) {
  60. free(ptr);
  61. ptr = NULL;
  62. hatomic_inc(&s_free_cnt);
  63. }
  64. }
  65. char* hv_strupper(char* str) {
  66. char* p = str;
  67. while (*p != '\0') {
  68. if (*p >= 'a' && *p <= 'z') {
  69. *p &= ~0x20;
  70. }
  71. ++p;
  72. }
  73. return str;
  74. }
  75. char* hv_strlower(char* str) {
  76. char* p = str;
  77. while (*p != '\0') {
  78. if (*p >= 'A' && *p <= 'Z') {
  79. *p |= 0x20;
  80. }
  81. ++p;
  82. }
  83. return str;
  84. }
  85. char* hv_strreverse(char* str) {
  86. if (str == NULL) return NULL;
  87. char* b = str;
  88. char* e = str;
  89. while(*e) {++e;}
  90. --e;
  91. char tmp;
  92. while (e > b) {
  93. tmp = *e;
  94. *e = *b;
  95. *b = tmp;
  96. --e;
  97. ++b;
  98. }
  99. return str;
  100. }
  101. // n = sizeof(dest_buf)
  102. char* hv_strncpy(char* dest, const char* src, size_t n) {
  103. assert(dest != NULL && src != NULL);
  104. char* ret = dest;
  105. while (*src != '\0' && --n > 0) {
  106. *dest++ = *src++;
  107. }
  108. *dest = '\0';
  109. return ret;
  110. }
  111. // n = sizeof(dest_buf)
  112. char* hv_strncat(char* dest, const char* src, size_t n) {
  113. assert(dest != NULL && src != NULL);
  114. char* ret = dest;
  115. while (*dest) {++dest;--n;}
  116. while (*src != '\0' && --n > 0) {
  117. *dest++ = *src++;
  118. }
  119. *dest = '\0';
  120. return ret;
  121. }
  122. bool hv_strstartswith(const char* str, const char* start) {
  123. assert(str != NULL && start != NULL);
  124. while (*str && *start && *str == *start) {
  125. ++str;
  126. ++start;
  127. }
  128. return *start == '\0';
  129. }
  130. bool hv_strendswith(const char* str, const char* end) {
  131. assert(str != NULL && end != NULL);
  132. int len1 = 0;
  133. int len2 = 0;
  134. while (*str) {++str; ++len1;}
  135. while (*end) {++end; ++len2;}
  136. if (len1 < len2) return false;
  137. while (len2-- > 0) {
  138. --str;
  139. --end;
  140. if (*str != *end) {
  141. return false;
  142. }
  143. }
  144. return true;
  145. }
  146. bool hv_strcontains(const char* str, const char* sub) {
  147. assert(str != NULL && sub != NULL);
  148. return strstr(str, sub) != NULL;
  149. }
  150. bool hv_wildcard_match(const char* str, const char* pattern) {
  151. assert(str != NULL && pattern != NULL);
  152. bool match = false;
  153. while (*str && *pattern) {
  154. if (*pattern == '*') {
  155. match = hv_strendswith(str, pattern + 1);
  156. break;
  157. } else if (*str != *pattern) {
  158. match = false;
  159. break;
  160. } else {
  161. ++str;
  162. ++pattern;
  163. }
  164. }
  165. return match ? match : (*str == '\0' && *pattern == '\0');
  166. }
  167. char* hv_strnchr(const char* s, char c, size_t n) {
  168. assert(s != NULL);
  169. const char* p = s;
  170. while (*p != '\0' && n-- > 0) {
  171. if (*p == c) return (char*)p;
  172. ++p;
  173. }
  174. return NULL;
  175. }
  176. char* hv_strrchr_dir(const char* filepath) {
  177. char* p = (char*)filepath;
  178. while (*p) ++p;
  179. while (--p >= filepath) {
  180. #ifdef OS_WIN
  181. if (*p == '/' || *p == '\\')
  182. #else
  183. if (*p == '/')
  184. #endif
  185. return p;
  186. }
  187. return NULL;
  188. }
  189. const char* hv_basename(const char* filepath) {
  190. const char* pos = hv_strrchr_dir(filepath);
  191. return pos ? pos+1 : filepath;
  192. }
  193. const char* hv_suffixname(const char* filename) {
  194. const char* pos = hv_strrchr_dot(filename);
  195. return pos ? pos+1 : "";
  196. }
  197. int hv_mkdir_p(const char* dir) {
  198. if (access(dir, 0) == 0) {
  199. return EEXIST;
  200. }
  201. char tmp[MAX_PATH] = {0};
  202. hv_strncpy(tmp, dir, sizeof(tmp));
  203. char* p = tmp;
  204. char delim = '/';
  205. while (*p) {
  206. #ifdef OS_WIN
  207. if (*p == '/' || *p == '\\') {
  208. delim = *p;
  209. #else
  210. if (*p == '/') {
  211. #endif
  212. *p = '\0';
  213. hv_mkdir(tmp);
  214. *p = delim;
  215. }
  216. ++p;
  217. }
  218. if (hv_mkdir(tmp) != 0) {
  219. return EPERM;
  220. }
  221. return 0;
  222. }
  223. int hv_rmdir_p(const char* dir) {
  224. if (access(dir, 0) != 0) {
  225. return ENOENT;
  226. }
  227. if (rmdir(dir) != 0) {
  228. return EPERM;
  229. }
  230. char tmp[MAX_PATH] = {0};
  231. hv_strncpy(tmp, dir, sizeof(tmp));
  232. char* p = tmp;
  233. while (*p) ++p;
  234. while (--p >= tmp) {
  235. #ifdef OS_WIN
  236. if (*p == '/' || *p == '\\') {
  237. #else
  238. if (*p == '/') {
  239. #endif
  240. *p = '\0';
  241. if (rmdir(tmp) != 0) {
  242. return 0;
  243. }
  244. }
  245. }
  246. return 0;
  247. }
  248. bool hv_exists(const char* path) {
  249. return access(path, 0) == 0;
  250. }
  251. bool hv_isdir(const char* path) {
  252. if (access(path, 0) != 0) return false;
  253. struct stat st;
  254. memset(&st, 0, sizeof(st));
  255. stat(path, &st);
  256. return S_ISDIR(st.st_mode);
  257. }
  258. bool hv_isfile(const char* path) {
  259. if (access(path, 0) != 0) return false;
  260. struct stat st;
  261. memset(&st, 0, sizeof(st));
  262. stat(path, &st);
  263. return S_ISREG(st.st_mode);
  264. }
  265. bool hv_islink(const char* path) {
  266. #ifdef OS_WIN
  267. return hv_isdir(path) && (GetFileAttributes(path) & FILE_ATTRIBUTE_REPARSE_POINT);
  268. #else
  269. if (access(path, 0) != 0) return false;
  270. struct stat st;
  271. memset(&st, 0, sizeof(st));
  272. lstat(path, &st);
  273. return S_ISLNK(st.st_mode);
  274. #endif
  275. }
  276. size_t hv_filesize(const char* filepath) {
  277. struct stat st;
  278. memset(&st, 0, sizeof(st));
  279. stat(filepath, &st);
  280. return st.st_size;
  281. }
  282. char* get_executable_path(char* buf, int size) {
  283. #ifdef OS_WIN
  284. GetModuleFileName(NULL, buf, size);
  285. #elif defined(OS_LINUX)
  286. if (readlink("/proc/self/exe", buf, size) == -1) {
  287. return NULL;
  288. }
  289. #elif defined(OS_DARWIN)
  290. _NSGetExecutablePath(buf, (uint32_t*)&size);
  291. #endif
  292. return buf;
  293. }
  294. char* get_executable_dir(char* buf, int size) {
  295. char filepath[MAX_PATH] = {0};
  296. get_executable_path(filepath, sizeof(filepath));
  297. char* pos = hv_strrchr_dir(filepath);
  298. if (pos) {
  299. *pos = '\0';
  300. strncpy(buf, filepath, size);
  301. }
  302. return buf;
  303. }
  304. char* get_executable_file(char* buf, int size) {
  305. char filepath[MAX_PATH] = {0};
  306. get_executable_path(filepath, sizeof(filepath));
  307. char* pos = hv_strrchr_dir(filepath);
  308. if (pos) {
  309. strncpy(buf, pos+1, size);
  310. }
  311. return buf;
  312. }
  313. char* get_run_dir(char* buf, int size) {
  314. return getcwd(buf, size);
  315. }
  316. int hv_rand(int min, int max) {
  317. static int s_seed = 0;
  318. assert(max > min);
  319. if (s_seed == 0) {
  320. s_seed = time(NULL);
  321. srand(s_seed);
  322. }
  323. int _rand = rand();
  324. _rand = min + (int) ((double) ((double) (max) - (min) + 1.0) * ((_rand) / ((RAND_MAX) + 1.0)));
  325. return _rand;
  326. }
  327. char* hv_random_string(char *buf, int len) {
  328. static char s_characters[] = {
  329. 'A', 'B', 'C', 'D', 'E', 'F', 'G', 'H', 'I', 'J', 'K', 'L', 'M', 'N', 'O', 'P', 'Q', 'R', 'S', 'T', 'U',
  330. 'V', 'W', 'X', 'Y', 'Z', 'a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'i', 'j', 'k', 'l', 'm', 'n', 'o', 'p',
  331. 'q', 'r', 's', 't', 'u', 'v', 'w', 'x', 'y', 'z', '0', '1', '2', '3', '4', '5', '6', '7', '8', '9',
  332. };
  333. int i = 0;
  334. for (; i < len; i++) {
  335. buf[i] = s_characters[hv_rand(0, sizeof(s_characters) - 1)];
  336. }
  337. buf[i] = '\0';
  338. return buf;
  339. }
  340. bool hv_getboolean(const char* str) {
  341. if (str == NULL) return false;
  342. int len = strlen(str);
  343. if (len == 0) return false;
  344. switch (len) {
  345. case 1: return *str == '1' || *str == 'y' || *str == 'Y';
  346. case 2: return stricmp(str, "on") == 0;
  347. case 3: return stricmp(str, "yes") == 0;
  348. case 4: return stricmp(str, "true") == 0;
  349. case 6: return stricmp(str, "enable") == 0;
  350. default: return false;
  351. }
  352. }
  353. size_t hv_parse_size(const char* str) {
  354. size_t size = 0, n = 0;
  355. const char* p = str;
  356. char c;
  357. while ((c = *p) != '\0') {
  358. if (c >= '0' && c <= '9') {
  359. n = n * 10 + c - '0';
  360. } else {
  361. switch (c) {
  362. case 'K': case 'k': n <<= 10; break;
  363. case 'M': case 'm': n <<= 20; break;
  364. case 'G': case 'g': n <<= 30; break;
  365. case 'T': case 't': n <<= 40; break;
  366. default: break;
  367. }
  368. size += n;
  369. n = 0;
  370. }
  371. ++p;
  372. }
  373. return size + n;
  374. }
  375. time_t hv_parse_time(const char* str) {
  376. time_t time = 0, n = 0;
  377. const char* p = str;
  378. char c;
  379. while ((c = *p) != '\0') {
  380. if (c >= '0' && c <= '9') {
  381. n = n * 10 + c - '0';
  382. } else {
  383. switch (c) {
  384. case 's': break;
  385. case 'm': n *= 60; break;
  386. case 'h': n *= 60 * 60; break;
  387. case 'd': n *= 24 * 60 * 60; break;
  388. case 'w': n *= 7 * 24 * 60 * 60; break;
  389. default: break;
  390. }
  391. time += n;
  392. n = 0;
  393. }
  394. ++p;
  395. }
  396. return time + n;
  397. }
  398. int hv_parse_url(hurl_t* stURL, const char* strURL) {
  399. if (stURL == NULL || strURL == NULL) return -1;
  400. memset(stURL, 0, sizeof(hurl_t));
  401. const char* begin = strURL;
  402. const char* end = strURL;
  403. while (*end != '\0') ++end;
  404. if (end - begin > 65535) return -2;
  405. // scheme://
  406. const char* sp = strURL;
  407. const char* ep = strstr(sp, "://");
  408. if (ep) {
  409. // stURL->fields[HV_URL_SCHEME].off = sp - begin;
  410. stURL->fields[HV_URL_SCHEME].len = ep - sp;
  411. sp = ep + 3;
  412. }
  413. // user:pswd@host:port
  414. ep = strchr(sp, '/');
  415. if (ep == NULL) ep = end;
  416. const char* user = sp;
  417. const char* host = sp;
  418. const char* pos = hv_strnchr(sp, '@', ep - sp);
  419. if (pos) {
  420. // user:pswd
  421. const char* pswd = hv_strnchr(user, ':', pos - user);
  422. if (pswd) {
  423. stURL->fields[HV_URL_PASSWORD].off = pswd + 1 - begin;
  424. stURL->fields[HV_URL_PASSWORD].len = pos - pswd - 1;
  425. } else {
  426. pswd = pos;
  427. }
  428. stURL->fields[HV_URL_USERNAME].off = user - begin;
  429. stURL->fields[HV_URL_USERNAME].len = pswd - user;
  430. // @
  431. host = pos + 1;
  432. }
  433. // port
  434. const char* port = hv_strnchr(host, ':', ep - host);
  435. if (port) {
  436. stURL->fields[HV_URL_PORT].off = port + 1 - begin;
  437. stURL->fields[HV_URL_PORT].len = ep - port - 1;
  438. // atoi
  439. for (unsigned short i = 1; i <= stURL->fields[HV_URL_PORT].len; ++i) {
  440. stURL->port = stURL->port * 10 + (port[i] - '0');
  441. }
  442. } else {
  443. port = ep;
  444. // set default port
  445. stURL->port = 80;
  446. if (stURL->fields[HV_URL_SCHEME].len > 0) {
  447. if (strncmp(strURL, "https://", 8) == 0) {
  448. stURL->port = 443;
  449. }
  450. }
  451. }
  452. // host
  453. stURL->fields[HV_URL_HOST].off = host - begin;
  454. stURL->fields[HV_URL_HOST].len = port - host;
  455. if (ep == end) return 0;
  456. // /path
  457. sp = ep;
  458. ep = strchr(sp, '?');
  459. if (ep == NULL) ep = end;
  460. stURL->fields[HV_URL_PATH].off = sp - begin;
  461. stURL->fields[HV_URL_PATH].len = ep - sp;
  462. if (ep == end) return 0;
  463. // ?query
  464. sp = ep + 1;
  465. ep = strchr(sp, '#');
  466. if (ep == NULL) ep = end;
  467. stURL->fields[HV_URL_QUERY].off = sp - begin;
  468. stURL->fields[HV_URL_QUERY].len = ep - sp;
  469. if (ep == end) return 0;
  470. // #fragment
  471. sp = ep + 1;
  472. ep = end;
  473. stURL->fields[HV_URL_FRAGMENT].off = sp - begin;
  474. stURL->fields[HV_URL_FRAGMENT].len = ep - sp;
  475. return 0;
  476. }