aos_util.c 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523
  1. #include "aos_util.h"
  2. #include "aos_log.h"
  3. static const char *g_s_wday[] = {
  4. "Sun", "Mon", "Tue", "Wed", "Thu", "Fri", "Sat"
  5. };
  6. static const char *g_s_mon[] = {
  7. "Jan", "Feb", "Mar", "Apr", "May", "Jun", "Jul", "Aug", "Sep", "Oct", "Nov", "Dec"
  8. };
  9. static const char g_s_gmt_format[] = "%s, %.2d %s %.4d %.2d:%.2d:%.2d GMT";
  10. int aos_parse_xml_body(aos_list_t *bc, mxml_node_t **root)
  11. {
  12. aos_buf_t *b;
  13. size_t len;
  14. *root = NULL;
  15. len = (size_t)aos_buf_list_len(bc);
  16. {
  17. int nsize = 0;
  18. char *buffer = (char*)malloc(sizeof(char)*(len+1));
  19. memset(buffer, 0, len + 1);
  20. aos_list_for_each_entry(aos_buf_t, b, bc, node) {
  21. memcpy(buffer + nsize, (char *)b->pos, aos_buf_size(b));
  22. nsize += aos_buf_size(b);
  23. }
  24. *root = mxmlLoadString(NULL, buffer, MXML_OPAQUE_CALLBACK);
  25. free(buffer);
  26. if (NULL == *root) {
  27. return AOSE_INTERNAL_ERROR;
  28. }
  29. }
  30. return AOSE_OK;
  31. }
  32. int aos_convert_to_gmt_time(char* date, const char* format, apr_time_exp_t *tm)
  33. {
  34. int size = apr_snprintf(date, AOS_MAX_GMT_TIME_LEN, format,
  35. g_s_wday[tm->tm_wday], tm->tm_mday, g_s_mon[tm->tm_mon], 1900 + tm->tm_year, tm->tm_hour, tm->tm_min, tm->tm_sec);
  36. if (size >= 0 && size < AOS_MAX_GMT_TIME_LEN) {
  37. return AOSE_OK;
  38. } else {
  39. return AOSE_INTERNAL_ERROR;
  40. }
  41. }
  42. int aos_get_gmt_str_time(char datestr[AOS_MAX_GMT_TIME_LEN])
  43. {
  44. int s;
  45. apr_time_t now;
  46. char buf[128];
  47. apr_time_exp_t result;
  48. now = apr_time_now();
  49. if ((s = apr_time_exp_gmt(&result, now)) != APR_SUCCESS) {
  50. aos_error_log("apr_time_exp_gmt fialure, code:%d %s.", s, apr_strerror(s, buf, sizeof(buf)));
  51. return AOSE_INTERNAL_ERROR;
  52. }
  53. if ((s = aos_convert_to_gmt_time(datestr, g_s_gmt_format, &result))
  54. != AOSE_OK) {
  55. aos_error_log("aos_convert_to_GMT failure, code:%d.", s);
  56. }
  57. return s;
  58. }
  59. int aos_url_encode(char *dest, const char *src, int maxSrcSize)
  60. {
  61. static const char *hex = "0123456789ABCDEF";
  62. int len = 0;
  63. unsigned char c;
  64. while (*src) {
  65. if (++len > maxSrcSize) {
  66. *dest = 0;
  67. return AOSE_INVALID_ARGUMENT;
  68. }
  69. c = *src;
  70. if (isalnum(c) || (c == '-') || (c == '_') || (c == '.') || (c == '~')) {
  71. *dest++ = c;
  72. } else if (*src == ' ') {
  73. *dest++ = '%';
  74. *dest++ = '2';
  75. *dest++ = '0';
  76. } else {
  77. *dest++ = '%';
  78. *dest++ = hex[c >> 4];
  79. *dest++ = hex[c & 15];
  80. }
  81. src++;
  82. }
  83. *dest = 0;
  84. return AOSE_OK;
  85. }
  86. int aos_query_params_to_string(aos_pool_t *p, aos_table_t *query_params, aos_string_t *querystr)
  87. {
  88. int rs;
  89. int pos;
  90. int len;
  91. char sep = '?';
  92. char ebuf[AOS_MAX_QUERY_ARG_LEN*3+1];
  93. char abuf[AOS_MAX_QUERY_ARG_LEN*6+128];
  94. int max_len;
  95. const aos_array_header_t *tarr;
  96. const aos_table_entry_t *telts;
  97. aos_buf_t *querybuf;
  98. if (apr_is_empty_table(query_params)) {
  99. return AOSE_OK;
  100. }
  101. max_len = sizeof(abuf)-1;
  102. querybuf = aos_create_buf(p, 256);
  103. aos_str_null(querystr);
  104. tarr = aos_table_elts(query_params);
  105. telts = (aos_table_entry_t*)tarr->elts;
  106. for (pos = 0; pos < tarr->nelts; ++pos) {
  107. if ((rs = aos_url_encode(ebuf, telts[pos].key, AOS_MAX_QUERY_ARG_LEN)) != AOSE_OK) {
  108. aos_error_log("query params args too big, key:%s.", telts[pos].key);
  109. return AOSE_INVALID_ARGUMENT;
  110. }
  111. len = apr_snprintf(abuf, max_len, "%c%s", sep, ebuf);
  112. if (telts[pos].val != NULL && *telts[pos].val != '\0') {
  113. if ((rs = aos_url_encode(ebuf, telts[pos].val, AOS_MAX_QUERY_ARG_LEN)) != AOSE_OK) {
  114. aos_error_log("query params args too big, value:%s.", telts[pos].val);
  115. return AOSE_INVALID_ARGUMENT;
  116. }
  117. len += apr_snprintf(abuf+len, max_len-len, "=%s", ebuf);
  118. if (len >= AOS_MAX_QUERY_ARG_LEN) {
  119. aos_error_log("query params args too big, %s.", abuf);
  120. return AOSE_INVALID_ARGUMENT;
  121. }
  122. }
  123. aos_buf_append_string(p, querybuf, abuf, len);
  124. sep = '&';
  125. }
  126. // result
  127. querystr->data = (char *)querybuf->pos;
  128. querystr->len = aos_buf_size(querybuf);
  129. return AOSE_OK;
  130. }
  131. void aos_gnome_sort(const char **headers, int size)
  132. {
  133. const char *tmp;
  134. int i = 0, last_highest = 0;
  135. while (i < size) {
  136. if ((i == 0) || apr_strnatcasecmp(headers[i-1], headers[i]) < 0) {
  137. i = ++last_highest;
  138. } else {
  139. tmp = headers[i];
  140. headers[i] = headers[i - 1];
  141. headers[--i] = tmp;
  142. }
  143. }
  144. }
  145. const char* aos_http_method_to_string(http_method_e method)
  146. {
  147. switch (method) {
  148. case HTTP_GET:
  149. return "GET";
  150. case HTTP_HEAD:
  151. return "HEAD";
  152. case HTTP_PUT:
  153. return "PUT";
  154. case HTTP_POST:
  155. return "POST";
  156. case HTTP_DELETE:
  157. return "DELETE";
  158. default:
  159. return "UNKNOWN";
  160. }
  161. }
  162. int aos_base64_encode(const unsigned char *in, int inLen, char *out)
  163. {
  164. static const char *ENC =
  165. "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/";
  166. char *original_out = out;
  167. while (inLen) {
  168. // first 6 bits of char 1
  169. *out++ = ENC[*in >> 2];
  170. if (!--inLen) {
  171. // last 2 bits of char 1, 4 bits of 0
  172. *out++ = ENC[(*in & 0x3) << 4];
  173. *out++ = '=';
  174. *out++ = '=';
  175. break;
  176. }
  177. // last 2 bits of char 1, first 4 bits of char 2
  178. *out++ = ENC[((*in & 0x3) << 4) | (*(in + 1) >> 4)];
  179. in++;
  180. if (!--inLen) {
  181. // last 4 bits of char 2, 2 bits of 0
  182. *out++ = ENC[(*in & 0xF) << 2];
  183. *out++ = '=';
  184. break;
  185. }
  186. // last 4 bits of char 2, first 2 bits of char 3
  187. *out++ = ENC[((*in & 0xF) << 2) | (*(in + 1) >> 6)];
  188. in++;
  189. // last 6 bits of char 3
  190. *out++ = ENC[*in & 0x3F];
  191. in++, inLen--;
  192. }
  193. return (out - original_out);
  194. }
  195. // HMAC-SHA-1:
  196. //
  197. // K - is key padded with zeros to 512 bits
  198. // m - is message
  199. // OPAD - 0x5c5c5c...
  200. // IPAD - 0x363636...
  201. //
  202. // HMAC(K,m) = SHA1((K ^ OPAD) . SHA1((K ^ IPAD) . m))
  203. void HMAC_SHA1(unsigned char hmac[20], const unsigned char *key, int key_len,
  204. const unsigned char *message, int message_len)
  205. {
  206. unsigned char kopad[64], kipad[64];
  207. int i;
  208. unsigned char digest[APR_SHA1_DIGESTSIZE];
  209. apr_sha1_ctx_t context;
  210. if (key_len > 64) {
  211. key_len = 64;
  212. }
  213. for (i = 0; i < key_len; i++) {
  214. kopad[i] = key[i] ^ 0x5c;
  215. kipad[i] = key[i] ^ 0x36;
  216. }
  217. for ( ; i < 64; i++) {
  218. kopad[i] = 0 ^ 0x5c;
  219. kipad[i] = 0 ^ 0x36;
  220. }
  221. apr_sha1_init(&context);
  222. apr_sha1_update(&context, (const char *)kipad, 64);
  223. apr_sha1_update(&context, (const char *)message, (unsigned int)message_len);
  224. apr_sha1_final(digest, &context);
  225. apr_sha1_init(&context);
  226. apr_sha1_update(&context, (const char *)kopad, 64);
  227. apr_sha1_update(&context, (const char *)digest, 20);
  228. apr_sha1_final(hmac, &context);
  229. }
  230. unsigned char* aos_md5(aos_pool_t* pool, const char *in, apr_size_t in_len) {
  231. unsigned char* out;
  232. apr_md5_ctx_t context;
  233. //APR_MD5_DIGESTSIZE: The MD5 digest size, value is 16
  234. out = aos_palloc(pool, APR_MD5_DIGESTSIZE + 1);
  235. if (!out) {
  236. return NULL;
  237. }
  238. if (0 != apr_md5_init(&context)) {
  239. return NULL;
  240. }
  241. if (0 != apr_md5_update(&context, in, in_len)) {
  242. return NULL;
  243. }
  244. if (0 != apr_md5_final(out, &context)) {
  245. return NULL;
  246. }
  247. out[APR_MD5_DIGESTSIZE] = '\0';
  248. return out;
  249. };
  250. int aos_url_decode(const char *in, char *out)
  251. {
  252. static const char tbl[256] = {
  253. -1,-1,-1,-1,-1,-1,-1,-1, -1,-1,-1,-1,-1,-1,-1,-1,
  254. -1,-1,-1,-1,-1,-1,-1,-1, -1,-1,-1,-1,-1,-1,-1,-1,
  255. -1,-1,-1,-1,-1,-1,-1,-1, -1,-1,-1,-1,-1,-1,-1,-1,
  256. 0, 1, 2, 3, 4, 5, 6, 7, 8, 9,-1,-1,-1,-1,-1,-1,
  257. -1,10,11,12,13,14,15,-1, -1,-1,-1,-1,-1,-1,-1,-1,
  258. -1,-1,-1,-1,-1,-1,-1,-1, -1,-1,-1,-1,-1,-1,-1,-1,
  259. -1,10,11,12,13,14,15,-1, -1,-1,-1,-1,-1,-1,-1,-1,
  260. -1,-1,-1,-1,-1,-1,-1,-1, -1,-1,-1,-1,-1,-1,-1,-1,
  261. -1,-1,-1,-1,-1,-1,-1,-1, -1,-1,-1,-1,-1,-1,-1,-1,
  262. -1,-1,-1,-1,-1,-1,-1,-1, -1,-1,-1,-1,-1,-1,-1,-1,
  263. -1,-1,-1,-1,-1,-1,-1,-1, -1,-1,-1,-1,-1,-1,-1,-1,
  264. -1,-1,-1,-1,-1,-1,-1,-1, -1,-1,-1,-1,-1,-1,-1,-1,
  265. -1,-1,-1,-1,-1,-1,-1,-1, -1,-1,-1,-1,-1,-1,-1,-1,
  266. -1,-1,-1,-1,-1,-1,-1,-1, -1,-1,-1,-1,-1,-1,-1,-1,
  267. -1,-1,-1,-1,-1,-1,-1,-1, -1,-1,-1,-1,-1,-1,-1,-1,
  268. -1,-1,-1,-1,-1,-1,-1,-1, -1,-1,-1,-1,-1,-1,-1,-1
  269. };
  270. char c, v1, v2;
  271. if(in != NULL) {
  272. while((c=*in++) != '\0') {
  273. if(c == '%') {
  274. if(!(v1=*in++) || (v1=tbl[(unsigned char)v1])<0 ||
  275. !(v2=*in++) || (v2=tbl[(unsigned char)v2])<0) {
  276. *out = '\0';
  277. return -1;
  278. }
  279. c = (v1<<4)|v2;
  280. } else if (c == '+') {
  281. c = ' ';
  282. }
  283. *out++ = c;
  284. }
  285. }
  286. *out = '\0';
  287. return 0;
  288. }
  289. /*
  290. * Convert a string to a long long integer.
  291. *
  292. * Ignores `locale' stuff. Assumes that the upper and lower case
  293. * alphabets and digits are each contiguous.
  294. */
  295. long long aos_strtoll(const char *nptr, char **endptr, int base)
  296. {
  297. const char *s;
  298. /* LONGLONG */
  299. long long int acc, cutoff;
  300. int c;
  301. int neg, any, cutlim;
  302. /* endptr may be NULL */
  303. #ifdef __GNUC__
  304. /* This outrageous construct just to shut up a GCC warning. */
  305. (void) &acc; (void) &cutoff;
  306. #endif
  307. /*
  308. * Skip white space and pick up leading +/- sign if any.
  309. * If base is 0, allow 0x for hex and 0 for octal, else
  310. * assume decimal; if base is already 16, allow 0x.
  311. */
  312. s = nptr;
  313. do {
  314. c = (unsigned char) *s++;
  315. } while (isspace(c));
  316. if (c == '-') {
  317. neg = 1;
  318. c = *s++;
  319. } else {
  320. neg = 0;
  321. if (c == '+')
  322. c = *s++;
  323. }
  324. if ((base == 0 || base == 16) &&
  325. c == '0' && (*s == 'x' || *s == 'X')) {
  326. c = s[1];
  327. s += 2;
  328. base = 16;
  329. }
  330. if (base == 0)
  331. base = c == '0' ? 8 : 10;
  332. /*
  333. * Compute the cutoff value between legal numbers and illegal
  334. * numbers. That is the largest legal value, divided by the
  335. * base. An input number that is greater than this value, if
  336. * followed by a legal input character, is too big. One that
  337. * is equal to this value may be valid or not; the limit
  338. * between valid and invalid numbers is then based on the last
  339. * digit. For instance, if the range for long longs is
  340. * [-9223372036854775808..9223372036854775807] and the input base
  341. * is 10, cutoff will be set to 922337203685477580 and cutlim to
  342. * either 7 (neg==0) or 8 (neg==1), meaning that if we have
  343. * accumulated a value > 922337203685477580, or equal but the
  344. * next digit is > 7 (or 8), the number is too big, and we will
  345. * return a range error.
  346. *
  347. * Set any if any `digits' consumed; make it negative to indicate
  348. * overflow.
  349. */
  350. cutoff = neg ? LLONG_MIN : LLONG_MAX;
  351. cutlim = (int)(cutoff % base);
  352. cutoff /= base;
  353. if (neg) {
  354. if (cutlim > 0) {
  355. cutlim -= base;
  356. cutoff += 1;
  357. }
  358. cutlim = -cutlim;
  359. }
  360. for (acc = 0, any = 0;; c = (unsigned char) *s++) {
  361. if (isdigit(c))
  362. c -= '0';
  363. else if (isalpha(c))
  364. c -= isupper(c) ? 'A' - 10 : 'a' - 10;
  365. else
  366. break;
  367. if (c >= base)
  368. break;
  369. if (any < 0)
  370. continue;
  371. if (neg) {
  372. if (acc < cutoff || (acc == cutoff && c > cutlim)) {
  373. any = -1;
  374. acc = LLONG_MIN;
  375. errno = ERANGE;
  376. } else {
  377. any = 1;
  378. acc *= base;
  379. acc -= c;
  380. }
  381. } else {
  382. if (acc > cutoff || (acc == cutoff && c > cutlim)) {
  383. any = -1;
  384. acc = LLONG_MAX;
  385. errno = ERANGE;
  386. } else {
  387. any = 1;
  388. acc *= base;
  389. acc += c;
  390. }
  391. }
  392. }
  393. if (endptr != 0)
  394. /* LINTED interface specification */
  395. *endptr = (char *)(any ? s - 1 : nptr);
  396. return (acc);
  397. }
  398. int64_t aos_atoi64(const char *nptr)
  399. {
  400. return aos_strtoull(nptr, NULL, 10);
  401. }
  402. unsigned long long aos_strtoull(const char *nptr, char **endptr, int base)
  403. {
  404. const char *s;
  405. unsigned long long acc, cutoff;
  406. int c;
  407. int neg, any, cutlim;
  408. /*
  409. * See strtoq for comments as to the logic used.
  410. */
  411. s = nptr;
  412. do {
  413. c = (unsigned char) *s++;
  414. } while (isspace(c));
  415. if (c == '-') {
  416. neg = 1;
  417. c = *s++;
  418. } else {
  419. neg = 0;
  420. if (c == '+')
  421. c = *s++;
  422. }
  423. if ((base == 0 || base == 16) &&
  424. c == '0' && (*s == 'x' || *s == 'X')) {
  425. c = s[1];
  426. s += 2;
  427. base = 16;
  428. }
  429. if (base == 0)
  430. base = c == '0' ? 8 : 10;
  431. cutoff = ULLONG_MAX / (unsigned long long)base;
  432. cutlim = ULLONG_MAX % (unsigned long long)base;
  433. for (acc = 0, any = 0;; c = (unsigned char) *s++) {
  434. if (isdigit(c))
  435. c -= '0';
  436. else if (isalpha(c))
  437. c -= isupper(c) ? 'A' - 10 : 'a' - 10;
  438. else
  439. break;
  440. if (c >= base)
  441. break;
  442. if (any < 0)
  443. continue;
  444. if (acc > cutoff || (acc == cutoff && c > cutlim)) {
  445. any = -1;
  446. acc = ULLONG_MAX;
  447. errno = ERANGE;
  448. } else {
  449. any = 1;
  450. acc *= (unsigned long long)base;
  451. acc += c;
  452. }
  453. }
  454. if (neg && any > 0)
  455. #ifdef WIN32
  456. #pragma warning(disable : 4146)
  457. #endif
  458. acc = -acc;
  459. #ifdef WIN32
  460. #pragma warning(default : 4146)
  461. #endif
  462. if (endptr != 0)
  463. *endptr = (char *) (any ? s - 1 : nptr);
  464. return (acc);
  465. }
  466. uint64_t aos_atoui64(const char *nptr)
  467. {
  468. return aos_strtoull(nptr, NULL, 10);
  469. }