2
0

altsvc.c 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706
  1. /***************************************************************************
  2. * _ _ ____ _
  3. * Project ___| | | | _ \| |
  4. * / __| | | | |_) | |
  5. * | (__| |_| | _ <| |___
  6. * \___|\___/|_| \_\_____|
  7. *
  8. * Copyright (C) Daniel Stenberg, <daniel@haxx.se>, et al.
  9. *
  10. * This software is licensed as described in the file COPYING, which
  11. * you should have received as part of this distribution. The terms
  12. * are also available at https://curl.se/docs/copyright.html.
  13. *
  14. * You may opt to use, copy, modify, merge, publish, distribute and/or sell
  15. * copies of the Software, and permit persons to whom the Software is
  16. * furnished to do so, under the terms of the COPYING file.
  17. *
  18. * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
  19. * KIND, either express or implied.
  20. *
  21. * SPDX-License-Identifier: curl
  22. *
  23. ***************************************************************************/
  24. /*
  25. * The Alt-Svc: header is defined in RFC 7838:
  26. * https://datatracker.ietf.org/doc/html/rfc7838
  27. */
  28. #include "curl_setup.h"
  29. #if !defined(CURL_DISABLE_HTTP) && !defined(CURL_DISABLE_ALTSVC)
  30. #include <curl/curl.h>
  31. #include "urldata.h"
  32. #include "altsvc.h"
  33. #include "curl_get_line.h"
  34. #include "strcase.h"
  35. #include "parsedate.h"
  36. #include "sendf.h"
  37. #include "warnless.h"
  38. #include "fopen.h"
  39. #include "rename.h"
  40. #include "strdup.h"
  41. #include "inet_pton.h"
  42. /* The last 3 #include files should be in this order */
  43. #include "curl_printf.h"
  44. #include "curl_memory.h"
  45. #include "memdebug.h"
  46. #define MAX_ALTSVC_LINE 4095
  47. #define MAX_ALTSVC_DATELENSTR "64"
  48. #define MAX_ALTSVC_DATELEN 64
  49. #define MAX_ALTSVC_HOSTLENSTR "512"
  50. #define MAX_ALTSVC_HOSTLEN 512
  51. #define MAX_ALTSVC_ALPNLENSTR "10"
  52. #define MAX_ALTSVC_ALPNLEN 10
  53. #define H3VERSION "h3"
  54. static enum alpnid alpn2alpnid(char *name)
  55. {
  56. if(strcasecompare(name, "h1"))
  57. return ALPN_h1;
  58. if(strcasecompare(name, "h2"))
  59. return ALPN_h2;
  60. if(strcasecompare(name, H3VERSION))
  61. return ALPN_h3;
  62. if(strcasecompare(name, "http/1.1"))
  63. return ALPN_h1;
  64. return ALPN_none; /* unknown, probably rubbish input */
  65. }
  66. /* Given the ALPN ID, return the name */
  67. const char *Curl_alpnid2str(enum alpnid id)
  68. {
  69. switch(id) {
  70. case ALPN_h1:
  71. return "h1";
  72. case ALPN_h2:
  73. return "h2";
  74. case ALPN_h3:
  75. return H3VERSION;
  76. default:
  77. return ""; /* bad */
  78. }
  79. }
  80. static void altsvc_free(struct altsvc *as)
  81. {
  82. free(as->src.host);
  83. free(as->dst.host);
  84. free(as);
  85. }
  86. static struct altsvc *altsvc_createid(const char *srchost,
  87. const char *dsthost,
  88. size_t dlen, /* dsthost length */
  89. enum alpnid srcalpnid,
  90. enum alpnid dstalpnid,
  91. unsigned int srcport,
  92. unsigned int dstport)
  93. {
  94. struct altsvc *as = calloc(1, sizeof(struct altsvc));
  95. size_t hlen;
  96. if(!as)
  97. return NULL;
  98. hlen = strlen(srchost);
  99. DEBUGASSERT(hlen);
  100. DEBUGASSERT(dlen);
  101. if(!hlen || !dlen) {
  102. /* bad input */
  103. free(as);
  104. return NULL;
  105. }
  106. if((hlen > 2) && srchost[0] == '[') {
  107. /* IPv6 address, strip off brackets */
  108. srchost++;
  109. hlen -= 2;
  110. }
  111. else if(srchost[hlen - 1] == '.')
  112. /* strip off trailing dot */
  113. hlen--;
  114. if((dlen > 2) && dsthost[0] == '[') {
  115. /* IPv6 address, strip off brackets */
  116. dsthost++;
  117. dlen -= 2;
  118. }
  119. as->src.host = Curl_memdup0(srchost, hlen);
  120. if(!as->src.host)
  121. goto error;
  122. as->dst.host = Curl_memdup0(dsthost, dlen);
  123. if(!as->dst.host)
  124. goto error;
  125. as->src.alpnid = srcalpnid;
  126. as->dst.alpnid = dstalpnid;
  127. as->src.port = curlx_ultous(srcport);
  128. as->dst.port = curlx_ultous(dstport);
  129. return as;
  130. error:
  131. altsvc_free(as);
  132. return NULL;
  133. }
  134. static struct altsvc *altsvc_create(char *srchost,
  135. char *dsthost,
  136. char *srcalpn,
  137. char *dstalpn,
  138. unsigned int srcport,
  139. unsigned int dstport)
  140. {
  141. enum alpnid dstalpnid = alpn2alpnid(dstalpn);
  142. enum alpnid srcalpnid = alpn2alpnid(srcalpn);
  143. if(!srcalpnid || !dstalpnid)
  144. return NULL;
  145. return altsvc_createid(srchost, dsthost, strlen(dsthost),
  146. srcalpnid, dstalpnid,
  147. srcport, dstport);
  148. }
  149. /* only returns SERIOUS errors */
  150. static CURLcode altsvc_add(struct altsvcinfo *asi, char *line)
  151. {
  152. /* Example line:
  153. h2 example.com 443 h3 shiny.example.com 8443 "20191231 10:00:00" 1
  154. */
  155. char srchost[MAX_ALTSVC_HOSTLEN + 1];
  156. char dsthost[MAX_ALTSVC_HOSTLEN + 1];
  157. char srcalpn[MAX_ALTSVC_ALPNLEN + 1];
  158. char dstalpn[MAX_ALTSVC_ALPNLEN + 1];
  159. char date[MAX_ALTSVC_DATELEN + 1];
  160. unsigned int srcport;
  161. unsigned int dstport;
  162. unsigned int prio;
  163. unsigned int persist;
  164. int rc;
  165. rc = sscanf(line,
  166. "%" MAX_ALTSVC_ALPNLENSTR "s %" MAX_ALTSVC_HOSTLENSTR "s %u "
  167. "%" MAX_ALTSVC_ALPNLENSTR "s %" MAX_ALTSVC_HOSTLENSTR "s %u "
  168. "\"%" MAX_ALTSVC_DATELENSTR "[^\"]\" %u %u",
  169. srcalpn, srchost, &srcport,
  170. dstalpn, dsthost, &dstport,
  171. date, &persist, &prio);
  172. if(9 == rc) {
  173. struct altsvc *as;
  174. time_t expires = Curl_getdate_capped(date);
  175. as = altsvc_create(srchost, dsthost, srcalpn, dstalpn, srcport, dstport);
  176. if(as) {
  177. as->expires = expires;
  178. as->prio = prio;
  179. as->persist = persist ? 1 : 0;
  180. Curl_llist_append(&asi->list, as, &as->node);
  181. }
  182. }
  183. return CURLE_OK;
  184. }
  185. /*
  186. * Load alt-svc entries from the given file. The text based line-oriented file
  187. * format is documented here: https://curl.se/docs/alt-svc.html
  188. *
  189. * This function only returns error on major problems that prevent alt-svc
  190. * handling to work completely. It will ignore individual syntactical errors
  191. * etc.
  192. */
  193. static CURLcode altsvc_load(struct altsvcinfo *asi, const char *file)
  194. {
  195. CURLcode result = CURLE_OK;
  196. FILE *fp;
  197. /* we need a private copy of the filename so that the altsvc cache file
  198. name survives an easy handle reset */
  199. free(asi->filename);
  200. asi->filename = strdup(file);
  201. if(!asi->filename)
  202. return CURLE_OUT_OF_MEMORY;
  203. fp = fopen(file, FOPEN_READTEXT);
  204. if(fp) {
  205. struct dynbuf buf;
  206. Curl_dyn_init(&buf, MAX_ALTSVC_LINE);
  207. while(Curl_get_line(&buf, fp)) {
  208. char *lineptr = Curl_dyn_ptr(&buf);
  209. while(*lineptr && ISBLANK(*lineptr))
  210. lineptr++;
  211. if(*lineptr == '#')
  212. /* skip commented lines */
  213. continue;
  214. altsvc_add(asi, lineptr);
  215. }
  216. Curl_dyn_free(&buf); /* free the line buffer */
  217. fclose(fp);
  218. }
  219. return result;
  220. }
  221. /*
  222. * Write this single altsvc entry to a single output line
  223. */
  224. static CURLcode altsvc_out(struct altsvc *as, FILE *fp)
  225. {
  226. struct tm stamp;
  227. const char *dst6_pre = "";
  228. const char *dst6_post = "";
  229. const char *src6_pre = "";
  230. const char *src6_post = "";
  231. CURLcode result = Curl_gmtime(as->expires, &stamp);
  232. if(result)
  233. return result;
  234. #ifdef USE_IPV6
  235. else {
  236. char ipv6_unused[16];
  237. if(1 == Curl_inet_pton(AF_INET6, as->dst.host, ipv6_unused)) {
  238. dst6_pre = "[";
  239. dst6_post = "]";
  240. }
  241. if(1 == Curl_inet_pton(AF_INET6, as->src.host, ipv6_unused)) {
  242. src6_pre = "[";
  243. src6_post = "]";
  244. }
  245. }
  246. #endif
  247. fprintf(fp,
  248. "%s %s%s%s %u "
  249. "%s %s%s%s %u "
  250. "\"%d%02d%02d "
  251. "%02d:%02d:%02d\" "
  252. "%u %u\n",
  253. Curl_alpnid2str(as->src.alpnid),
  254. src6_pre, as->src.host, src6_post,
  255. as->src.port,
  256. Curl_alpnid2str(as->dst.alpnid),
  257. dst6_pre, as->dst.host, dst6_post,
  258. as->dst.port,
  259. stamp.tm_year + 1900, stamp.tm_mon + 1, stamp.tm_mday,
  260. stamp.tm_hour, stamp.tm_min, stamp.tm_sec,
  261. as->persist, as->prio);
  262. return CURLE_OK;
  263. }
  264. /* ---- library-wide functions below ---- */
  265. /*
  266. * Curl_altsvc_init() creates a new altsvc cache.
  267. * It returns the new instance or NULL if something goes wrong.
  268. */
  269. struct altsvcinfo *Curl_altsvc_init(void)
  270. {
  271. struct altsvcinfo *asi = calloc(1, sizeof(struct altsvcinfo));
  272. if(!asi)
  273. return NULL;
  274. Curl_llist_init(&asi->list, NULL);
  275. /* set default behavior */
  276. asi->flags = CURLALTSVC_H1
  277. #ifdef USE_HTTP2
  278. | CURLALTSVC_H2
  279. #endif
  280. #ifdef USE_HTTP3
  281. | CURLALTSVC_H3
  282. #endif
  283. ;
  284. return asi;
  285. }
  286. /*
  287. * Curl_altsvc_load() loads alt-svc from file.
  288. */
  289. CURLcode Curl_altsvc_load(struct altsvcinfo *asi, const char *file)
  290. {
  291. DEBUGASSERT(asi);
  292. return altsvc_load(asi, file);
  293. }
  294. /*
  295. * Curl_altsvc_ctrl() passes on the external bitmask.
  296. */
  297. CURLcode Curl_altsvc_ctrl(struct altsvcinfo *asi, const long ctrl)
  298. {
  299. DEBUGASSERT(asi);
  300. asi->flags = ctrl;
  301. return CURLE_OK;
  302. }
  303. /*
  304. * Curl_altsvc_cleanup() frees an altsvc cache instance and all associated
  305. * resources.
  306. */
  307. void Curl_altsvc_cleanup(struct altsvcinfo **altsvcp)
  308. {
  309. if(*altsvcp) {
  310. struct Curl_llist_node *e;
  311. struct Curl_llist_node *n;
  312. struct altsvcinfo *altsvc = *altsvcp;
  313. for(e = Curl_llist_head(&altsvc->list); e; e = n) {
  314. struct altsvc *as = Curl_node_elem(e);
  315. n = Curl_node_next(e);
  316. altsvc_free(as);
  317. }
  318. free(altsvc->filename);
  319. free(altsvc);
  320. *altsvcp = NULL; /* clear the pointer */
  321. }
  322. }
  323. /*
  324. * Curl_altsvc_save() writes the altsvc cache to a file.
  325. */
  326. CURLcode Curl_altsvc_save(struct Curl_easy *data,
  327. struct altsvcinfo *altsvc, const char *file)
  328. {
  329. CURLcode result = CURLE_OK;
  330. FILE *out;
  331. char *tempstore = NULL;
  332. if(!altsvc)
  333. /* no cache activated */
  334. return CURLE_OK;
  335. /* if not new name is given, use the one we stored from the load */
  336. if(!file && altsvc->filename)
  337. file = altsvc->filename;
  338. if((altsvc->flags & CURLALTSVC_READONLYFILE) || !file || !file[0])
  339. /* marked as read-only, no file or zero length filename */
  340. return CURLE_OK;
  341. result = Curl_fopen(data, file, &out, &tempstore);
  342. if(!result) {
  343. struct Curl_llist_node *e;
  344. struct Curl_llist_node *n;
  345. fputs("# Your alt-svc cache. https://curl.se/docs/alt-svc.html\n"
  346. "# This file was generated by libcurl! Edit at your own risk.\n",
  347. out);
  348. for(e = Curl_llist_head(&altsvc->list); e; e = n) {
  349. struct altsvc *as = Curl_node_elem(e);
  350. n = Curl_node_next(e);
  351. result = altsvc_out(as, out);
  352. if(result)
  353. break;
  354. }
  355. fclose(out);
  356. if(!result && tempstore && Curl_rename(tempstore, file))
  357. result = CURLE_WRITE_ERROR;
  358. if(result && tempstore)
  359. unlink(tempstore);
  360. }
  361. free(tempstore);
  362. return result;
  363. }
  364. static CURLcode getalnum(const char **ptr, char *alpnbuf, size_t buflen)
  365. {
  366. size_t len;
  367. const char *protop;
  368. const char *p = *ptr;
  369. while(*p && ISBLANK(*p))
  370. p++;
  371. protop = p;
  372. while(*p && !ISBLANK(*p) && (*p != ';') && (*p != '='))
  373. p++;
  374. len = p - protop;
  375. *ptr = p;
  376. if(!len || (len >= buflen))
  377. return CURLE_BAD_FUNCTION_ARGUMENT;
  378. memcpy(alpnbuf, protop, len);
  379. alpnbuf[len] = 0;
  380. return CURLE_OK;
  381. }
  382. /* hostcompare() returns true if 'host' matches 'check'. The first host
  383. * argument may have a trailing dot present that will be ignored.
  384. */
  385. static bool hostcompare(const char *host, const char *check)
  386. {
  387. size_t hlen = strlen(host);
  388. size_t clen = strlen(check);
  389. if(hlen && (host[hlen - 1] == '.'))
  390. hlen--;
  391. if(hlen != clen)
  392. /* they cannot match if they have different lengths */
  393. return FALSE;
  394. return strncasecompare(host, check, hlen);
  395. }
  396. /* altsvc_flush() removes all alternatives for this source origin from the
  397. list */
  398. static void altsvc_flush(struct altsvcinfo *asi, enum alpnid srcalpnid,
  399. const char *srchost, unsigned short srcport)
  400. {
  401. struct Curl_llist_node *e;
  402. struct Curl_llist_node *n;
  403. for(e = Curl_llist_head(&asi->list); e; e = n) {
  404. struct altsvc *as = Curl_node_elem(e);
  405. n = Curl_node_next(e);
  406. if((srcalpnid == as->src.alpnid) &&
  407. (srcport == as->src.port) &&
  408. hostcompare(srchost, as->src.host)) {
  409. Curl_node_remove(e);
  410. altsvc_free(as);
  411. }
  412. }
  413. }
  414. #ifdef DEBUGBUILD
  415. /* to play well with debug builds, we can *set* a fixed time this will
  416. return */
  417. static time_t altsvc_debugtime(void *unused)
  418. {
  419. char *timestr = getenv("CURL_TIME");
  420. (void)unused;
  421. if(timestr) {
  422. long val = strtol(timestr, NULL, 10);
  423. return (time_t)val;
  424. }
  425. return time(NULL);
  426. }
  427. #undef time
  428. #define time(x) altsvc_debugtime(x)
  429. #endif
  430. #define ISNEWLINE(x) (((x) == '\n') || (x) == '\r')
  431. /*
  432. * Curl_altsvc_parse() takes an incoming alt-svc response header and stores
  433. * the data correctly in the cache.
  434. *
  435. * 'value' points to the header *value*. That is contents to the right of the
  436. * header name.
  437. *
  438. * Currently this function rejects invalid data without returning an error.
  439. * Invalid hostname, port number will result in the specific alternative
  440. * being rejected. Unknown protocols are skipped.
  441. */
  442. CURLcode Curl_altsvc_parse(struct Curl_easy *data,
  443. struct altsvcinfo *asi, const char *value,
  444. enum alpnid srcalpnid, const char *srchost,
  445. unsigned short srcport)
  446. {
  447. const char *p = value;
  448. char alpnbuf[MAX_ALTSVC_ALPNLEN] = "";
  449. struct altsvc *as;
  450. unsigned short dstport = srcport; /* the same by default */
  451. CURLcode result = getalnum(&p, alpnbuf, sizeof(alpnbuf));
  452. size_t entries = 0;
  453. #ifdef CURL_DISABLE_VERBOSE_STRINGS
  454. (void)data;
  455. #endif
  456. if(result) {
  457. infof(data, "Excessive alt-svc header, ignoring.");
  458. return CURLE_OK;
  459. }
  460. DEBUGASSERT(asi);
  461. /* "clear" is a magic keyword */
  462. if(strcasecompare(alpnbuf, "clear")) {
  463. /* Flush cached alternatives for this source origin */
  464. altsvc_flush(asi, srcalpnid, srchost, srcport);
  465. return CURLE_OK;
  466. }
  467. do {
  468. if(*p == '=') {
  469. /* [protocol]="[host][:port]" */
  470. enum alpnid dstalpnid = alpn2alpnid(alpnbuf); /* the same by default */
  471. p++;
  472. if(*p == '\"') {
  473. const char *dsthost = "";
  474. size_t dstlen = 0; /* destination hostname length */
  475. const char *value_ptr;
  476. char option[32];
  477. unsigned long num;
  478. char *end_ptr;
  479. bool quoted = FALSE;
  480. time_t maxage = 24 * 3600; /* default is 24 hours */
  481. bool persist = FALSE;
  482. bool valid = TRUE;
  483. p++;
  484. if(*p != ':') {
  485. /* hostname starts here */
  486. const char *hostp = p;
  487. if(*p == '[') {
  488. /* pass all valid IPv6 letters - does not handle zone id */
  489. dstlen = strspn(++p, "0123456789abcdefABCDEF:.");
  490. if(p[dstlen] != ']')
  491. /* invalid host syntax, bail out */
  492. break;
  493. /* we store the IPv6 numerical address *with* brackets */
  494. dstlen += 2;
  495. p = &p[dstlen-1];
  496. }
  497. else {
  498. while(*p && (ISALNUM(*p) || (*p == '.') || (*p == '-')))
  499. p++;
  500. dstlen = p - hostp;
  501. }
  502. if(!dstlen || (dstlen >= MAX_ALTSVC_HOSTLEN)) {
  503. infof(data, "Excessive alt-svc hostname, ignoring.");
  504. valid = FALSE;
  505. }
  506. else {
  507. dsthost = hostp;
  508. }
  509. }
  510. else {
  511. /* no destination name, use source host */
  512. dsthost = srchost;
  513. dstlen = strlen(srchost);
  514. }
  515. if(*p == ':') {
  516. unsigned long port = 0;
  517. p++;
  518. if(ISDIGIT(*p))
  519. /* a port number */
  520. port = strtoul(p, &end_ptr, 10);
  521. else
  522. end_ptr = (char *)p; /* not left uninitialized */
  523. if(!port || port > USHRT_MAX || end_ptr == p || *end_ptr != '\"') {
  524. infof(data, "Unknown alt-svc port number, ignoring.");
  525. valid = FALSE;
  526. }
  527. else {
  528. dstport = curlx_ultous(port);
  529. p = end_ptr;
  530. }
  531. }
  532. if(*p++ != '\"')
  533. break;
  534. /* Handle the optional 'ma' and 'persist' flags. Unknown flags
  535. are skipped. */
  536. for(;;) {
  537. while(ISBLANK(*p))
  538. p++;
  539. if(*p != ';')
  540. break;
  541. p++; /* pass the semicolon */
  542. if(!*p || ISNEWLINE(*p))
  543. break;
  544. result = getalnum(&p, option, sizeof(option));
  545. if(result) {
  546. /* skip option if name is too long */
  547. option[0] = '\0';
  548. }
  549. while(*p && ISBLANK(*p))
  550. p++;
  551. if(*p != '=')
  552. return CURLE_OK;
  553. p++;
  554. while(*p && ISBLANK(*p))
  555. p++;
  556. if(!*p)
  557. return CURLE_OK;
  558. if(*p == '\"') {
  559. /* quoted value */
  560. p++;
  561. quoted = TRUE;
  562. }
  563. value_ptr = p;
  564. if(quoted) {
  565. while(*p && *p != '\"')
  566. p++;
  567. if(!*p++)
  568. return CURLE_OK;
  569. }
  570. else {
  571. while(*p && !ISBLANK(*p) && *p!= ';' && *p != ',')
  572. p++;
  573. }
  574. num = strtoul(value_ptr, &end_ptr, 10);
  575. if((end_ptr != value_ptr) && (num < ULONG_MAX)) {
  576. if(strcasecompare("ma", option))
  577. maxage = (time_t)num;
  578. else if(strcasecompare("persist", option) && (num == 1))
  579. persist = TRUE;
  580. }
  581. }
  582. if(dstalpnid && valid) {
  583. if(!entries++)
  584. /* Flush cached alternatives for this source origin, if any - when
  585. this is the first entry of the line. */
  586. altsvc_flush(asi, srcalpnid, srchost, srcport);
  587. as = altsvc_createid(srchost, dsthost, dstlen,
  588. srcalpnid, dstalpnid,
  589. srcport, dstport);
  590. if(as) {
  591. /* The expires time also needs to take the Age: value (if any) into
  592. account. [See RFC 7838 section 3.1] */
  593. as->expires = maxage + time(NULL);
  594. as->persist = persist;
  595. Curl_llist_append(&asi->list, as, &as->node);
  596. infof(data, "Added alt-svc: %s:%d over %s", dsthost, dstport,
  597. Curl_alpnid2str(dstalpnid));
  598. }
  599. }
  600. }
  601. else
  602. break;
  603. /* after the double quote there can be a comma if there is another
  604. string or a semicolon if no more */
  605. if(*p == ',') {
  606. /* comma means another alternative is presented */
  607. p++;
  608. result = getalnum(&p, alpnbuf, sizeof(alpnbuf));
  609. if(result)
  610. break;
  611. }
  612. }
  613. else
  614. break;
  615. } while(*p && (*p != ';') && (*p != '\n') && (*p != '\r'));
  616. return CURLE_OK;
  617. }
  618. /*
  619. * Return TRUE on a match
  620. */
  621. bool Curl_altsvc_lookup(struct altsvcinfo *asi,
  622. enum alpnid srcalpnid, const char *srchost,
  623. int srcport,
  624. struct altsvc **dstentry,
  625. const int versions) /* one or more bits */
  626. {
  627. struct Curl_llist_node *e;
  628. struct Curl_llist_node *n;
  629. time_t now = time(NULL);
  630. DEBUGASSERT(asi);
  631. DEBUGASSERT(srchost);
  632. DEBUGASSERT(dstentry);
  633. for(e = Curl_llist_head(&asi->list); e; e = n) {
  634. struct altsvc *as = Curl_node_elem(e);
  635. n = Curl_node_next(e);
  636. if(as->expires < now) {
  637. /* an expired entry, remove */
  638. Curl_node_remove(e);
  639. altsvc_free(as);
  640. continue;
  641. }
  642. if((as->src.alpnid == srcalpnid) &&
  643. hostcompare(srchost, as->src.host) &&
  644. (as->src.port == srcport) &&
  645. (versions & (int)as->dst.alpnid)) {
  646. /* match */
  647. *dstentry = as;
  648. return TRUE;
  649. }
  650. }
  651. return FALSE;
  652. }
  653. #endif /* !CURL_DISABLE_HTTP && !CURL_DISABLE_ALTSVC */