Annotation of embedaddon/curl/src/tool_metalink.c, revision 1.1.1.1

1.1       misho       1: /***************************************************************************
                      2:  *                                  _   _ ____  _
                      3:  *  Project                     ___| | | |  _ \| |
                      4:  *                             / __| | | | |_) | |
                      5:  *                            | (__| |_| |  _ <| |___
                      6:  *                             \___|\___/|_| \_\_____|
                      7:  *
                      8:  * Copyright (C) 1998 - 2020, 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.haxx.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:  ***************************************************************************/
                     22: #include "tool_setup.h"
                     23: 
                     24: #ifdef USE_METALINK
                     25: 
                     26: #include <sys/stat.h>
                     27: #include <stdlib.h>
                     28: 
                     29: #ifdef HAVE_FCNTL_H
                     30: #  include <fcntl.h>
                     31: #endif
                     32: 
                     33: #undef HAVE_NSS_CONTEXT
                     34: 
                     35: #ifdef USE_OPENSSL
                     36: #  include <openssl/md5.h>
                     37: #  include <openssl/sha.h>
                     38: #elif defined(USE_GNUTLS_NETTLE)
                     39: #  include <nettle/md5.h>
                     40: #  include <nettle/sha.h>
                     41: #  define MD5_CTX    struct md5_ctx
                     42: #  define SHA_CTX    struct sha1_ctx
                     43: #  define SHA256_CTX struct sha256_ctx
                     44: #elif defined(USE_GNUTLS)
                     45: #  include <gcrypt.h>
                     46: #  define MD5_CTX    gcry_md_hd_t
                     47: #  define SHA_CTX    gcry_md_hd_t
                     48: #  define SHA256_CTX gcry_md_hd_t
                     49: #elif defined(USE_NSS)
                     50: #  include <nss.h>
                     51: #  include <pk11pub.h>
                     52: #  define MD5_CTX    void *
                     53: #  define SHA_CTX    void *
                     54: #  define SHA256_CTX void *
                     55: #  define HAVE_NSS_CONTEXT
                     56:    static NSSInitContext *nss_context;
                     57: #elif (defined(__MAC_OS_X_VERSION_MAX_ALLOWED) && \
                     58:               (__MAC_OS_X_VERSION_MAX_ALLOWED >= 1040)) || \
                     59:       (defined(__IPHONE_OS_VERSION_MAX_ALLOWED) && \
                     60:               (__IPHONE_OS_VERSION_MAX_ALLOWED >= 20000))
                     61: /* For Apple operating systems: CommonCrypto has the functions we need.
                     62:    The library's headers are even backward-compatible with OpenSSL's
                     63:    headers as long as we define COMMON_DIGEST_FOR_OPENSSL first.
                     64: 
                     65:    These functions are available on Tiger and later, as well as iOS 2.0
                     66:    and later. If you're building for an older cat, well, sorry. */
                     67: #  define COMMON_DIGEST_FOR_OPENSSL
                     68: #  include <CommonCrypto/CommonDigest.h>
                     69: #elif defined(USE_WIN32_CRYPTO)
                     70: /* For Windows: If no other crypto library is provided, we fallback
                     71:    to the hash functions provided within the Microsoft Windows CryptoAPI */
                     72: #  include <wincrypt.h>
                     73: /* Custom structure in order to store the required provider and hash handle */
                     74: struct win32_crypto_hash {
                     75:   HCRYPTPROV hCryptProv;
                     76:   HCRYPTHASH hHash;
                     77: };
                     78: /* Custom Microsoft AES Cryptographic Provider defines required for MinGW */
                     79: #  ifndef ALG_SID_SHA_256
                     80: #    define ALG_SID_SHA_256  12
                     81: #  endif
                     82: #  ifndef CALG_SHA_256
                     83: #    define CALG_SHA_256 (ALG_CLASS_HASH | ALG_TYPE_ANY | ALG_SID_SHA_256)
                     84: #  endif
                     85: #  define MD5_CTX    struct win32_crypto_hash
                     86: #  define SHA_CTX    struct win32_crypto_hash
                     87: #  define SHA256_CTX struct win32_crypto_hash
                     88: #else
                     89: #  error "Can't compile METALINK support without a crypto library."
                     90: #endif
                     91: 
                     92: #define ENABLE_CURLX_PRINTF
                     93: /* use our own printf() functions */
                     94: #include "curlx.h"
                     95: 
                     96: #include "tool_getparam.h"
                     97: #include "tool_paramhlp.h"
                     98: #include "tool_cfgable.h"
                     99: #include "tool_metalink.h"
                    100: #include "tool_operate.h"
                    101: #include "tool_msgs.h"
                    102: 
                    103: #include "memdebug.h" /* keep this as LAST include */
                    104: 
                    105: /* Copied from tool_getparam.c */
                    106: #define GetStr(str,val) do { \
                    107:   if(*(str)) { \
                    108:     free(*(str)); \
                    109:     *(str) = NULL; \
                    110:   } \
                    111:   if((val)) \
                    112:     *(str) = strdup((val)); \
                    113:   if(!(val)) \
                    114:     return PARAM_NO_MEM; \
                    115: } while(0)
                    116: 
                    117: #if defined(USE_OPENSSL)
                    118: /* Functions are already defined */
                    119: #elif defined(USE_GNUTLS_NETTLE)
                    120: 
                    121: static int MD5_Init(MD5_CTX *ctx)
                    122: {
                    123:   md5_init(ctx);
                    124:   return 1;
                    125: }
                    126: 
                    127: static void MD5_Update(MD5_CTX *ctx,
                    128:                        const unsigned char *input,
                    129:                        unsigned int inputLen)
                    130: {
                    131:   md5_update(ctx, inputLen, input);
                    132: }
                    133: 
                    134: static void MD5_Final(unsigned char digest[16], MD5_CTX *ctx)
                    135: {
                    136:   md5_digest(ctx, 16, digest);
                    137: }
                    138: 
                    139: static int SHA1_Init(SHA_CTX *ctx)
                    140: {
                    141:   sha1_init(ctx);
                    142:   return 1;
                    143: }
                    144: 
                    145: static void SHA1_Update(SHA_CTX *ctx,
                    146:                         const unsigned char *input,
                    147:                         unsigned int inputLen)
                    148: {
                    149:   sha1_update(ctx, inputLen, input);
                    150: }
                    151: 
                    152: static void SHA1_Final(unsigned char digest[20], SHA_CTX *ctx)
                    153: {
                    154:   sha1_digest(ctx, 20, digest);
                    155: }
                    156: 
                    157: static int SHA256_Init(SHA256_CTX *ctx)
                    158: {
                    159:   sha256_init(ctx);
                    160:   return 1;
                    161: }
                    162: 
                    163: static void SHA256_Update(SHA256_CTX *ctx,
                    164:                           const unsigned char *input,
                    165:                           unsigned int inputLen)
                    166: {
                    167:   sha256_update(ctx, inputLen, input);
                    168: }
                    169: 
                    170: static void SHA256_Final(unsigned char digest[32], SHA256_CTX *ctx)
                    171: {
                    172:   sha256_digest(ctx, 32, digest);
                    173: }
                    174: 
                    175: #elif defined(USE_GNUTLS)
                    176: 
                    177: static int MD5_Init(MD5_CTX *ctx)
                    178: {
                    179:   gcry_md_open(ctx, GCRY_MD_MD5, 0);
                    180:   return 1;
                    181: }
                    182: 
                    183: static void MD5_Update(MD5_CTX *ctx,
                    184:                        const unsigned char *input,
                    185:                        unsigned int inputLen)
                    186: {
                    187:   gcry_md_write(*ctx, input, inputLen);
                    188: }
                    189: 
                    190: static void MD5_Final(unsigned char digest[16], MD5_CTX *ctx)
                    191: {
                    192:   memcpy(digest, gcry_md_read(*ctx, 0), 16);
                    193:   gcry_md_close(*ctx);
                    194: }
                    195: 
                    196: static int SHA1_Init(SHA_CTX *ctx)
                    197: {
                    198:   gcry_md_open(ctx, GCRY_MD_SHA1, 0);
                    199:   return 1;
                    200: }
                    201: 
                    202: static void SHA1_Update(SHA_CTX *ctx,
                    203:                         const unsigned char *input,
                    204:                         unsigned int inputLen)
                    205: {
                    206:   gcry_md_write(*ctx, input, inputLen);
                    207: }
                    208: 
                    209: static void SHA1_Final(unsigned char digest[20], SHA_CTX *ctx)
                    210: {
                    211:   memcpy(digest, gcry_md_read(*ctx, 0), 20);
                    212:   gcry_md_close(*ctx);
                    213: }
                    214: 
                    215: static int SHA256_Init(SHA256_CTX *ctx)
                    216: {
                    217:   gcry_md_open(ctx, GCRY_MD_SHA256, 0);
                    218:   return 1;
                    219: }
                    220: 
                    221: static void SHA256_Update(SHA256_CTX *ctx,
                    222:                           const unsigned char *input,
                    223:                           unsigned int inputLen)
                    224: {
                    225:   gcry_md_write(*ctx, input, inputLen);
                    226: }
                    227: 
                    228: static void SHA256_Final(unsigned char digest[32], SHA256_CTX *ctx)
                    229: {
                    230:   memcpy(digest, gcry_md_read(*ctx, 0), 32);
                    231:   gcry_md_close(*ctx);
                    232: }
                    233: 
                    234: #elif defined(USE_NSS)
                    235: 
                    236: static int nss_hash_init(void **pctx, SECOidTag hash_alg)
                    237: {
                    238:   PK11Context *ctx;
                    239: 
                    240:   /* we have to initialize NSS if not initialized already */
                    241:   if(!NSS_IsInitialized() && !nss_context) {
                    242:     static NSSInitParameters params;
                    243:     params.length = sizeof(params);
                    244:     nss_context = NSS_InitContext("", "", "", "", &params, NSS_INIT_READONLY
                    245:         | NSS_INIT_NOCERTDB   | NSS_INIT_NOMODDB       | NSS_INIT_FORCEOPEN
                    246:         | NSS_INIT_NOROOTINIT | NSS_INIT_OPTIMIZESPACE | NSS_INIT_PK11RELOAD);
                    247:   }
                    248: 
                    249:   ctx = PK11_CreateDigestContext(hash_alg);
                    250:   if(!ctx)
                    251:     return /* failure */ 0;
                    252: 
                    253:   if(PK11_DigestBegin(ctx) != SECSuccess) {
                    254:     PK11_DestroyContext(ctx, PR_TRUE);
                    255:     return /* failure */ 0;
                    256:   }
                    257: 
                    258:   *pctx = ctx;
                    259:   return /* success */ 1;
                    260: }
                    261: 
                    262: static void nss_hash_final(void **pctx, unsigned char *out, unsigned int len)
                    263: {
                    264:   PK11Context *ctx = *pctx;
                    265:   unsigned int outlen;
                    266:   PK11_DigestFinal(ctx, out, &outlen, len);
                    267:   PK11_DestroyContext(ctx, PR_TRUE);
                    268: }
                    269: 
                    270: static int MD5_Init(MD5_CTX *pctx)
                    271: {
                    272:   return nss_hash_init(pctx, SEC_OID_MD5);
                    273: }
                    274: 
                    275: static void MD5_Update(MD5_CTX *pctx,
                    276:                        const unsigned char *input,
                    277:                        unsigned int input_len)
                    278: {
                    279:   PK11_DigestOp(*pctx, input, input_len);
                    280: }
                    281: 
                    282: static void MD5_Final(unsigned char digest[16], MD5_CTX *pctx)
                    283: {
                    284:   nss_hash_final(pctx, digest, 16);
                    285: }
                    286: 
                    287: static int SHA1_Init(SHA_CTX *pctx)
                    288: {
                    289:   return nss_hash_init(pctx, SEC_OID_SHA1);
                    290: }
                    291: 
                    292: static void SHA1_Update(SHA_CTX *pctx,
                    293:                         const unsigned char *input,
                    294:                         unsigned int input_len)
                    295: {
                    296:   PK11_DigestOp(*pctx, input, input_len);
                    297: }
                    298: 
                    299: static void SHA1_Final(unsigned char digest[20], SHA_CTX *pctx)
                    300: {
                    301:   nss_hash_final(pctx, digest, 20);
                    302: }
                    303: 
                    304: static int SHA256_Init(SHA256_CTX *pctx)
                    305: {
                    306:   return nss_hash_init(pctx, SEC_OID_SHA256);
                    307: }
                    308: 
                    309: static void SHA256_Update(SHA256_CTX *pctx,
                    310:                           const unsigned char *input,
                    311:                           unsigned int input_len)
                    312: {
                    313:   PK11_DigestOp(*pctx, input, input_len);
                    314: }
                    315: 
                    316: static void SHA256_Final(unsigned char digest[32], SHA256_CTX *pctx)
                    317: {
                    318:   nss_hash_final(pctx, digest, 32);
                    319: }
                    320: 
                    321: #elif defined(USE_WIN32_CRYPTO)
                    322: 
                    323: static void win32_crypto_final(struct win32_crypto_hash *ctx,
                    324:                                unsigned char *digest,
                    325:                                unsigned int digestLen)
                    326: {
                    327:   unsigned long length;
                    328:   CryptGetHashParam(ctx->hHash, HP_HASHVAL, NULL, &length, 0);
                    329:   if(length == digestLen)
                    330:     CryptGetHashParam(ctx->hHash, HP_HASHVAL, digest, &length, 0);
                    331:   if(ctx->hHash)
                    332:     CryptDestroyHash(ctx->hHash);
                    333:   if(ctx->hCryptProv)
                    334:     CryptReleaseContext(ctx->hCryptProv, 0);
                    335: }
                    336: 
                    337: static int MD5_Init(MD5_CTX *ctx)
                    338: {
                    339:   if(CryptAcquireContext(&ctx->hCryptProv, NULL, NULL, PROV_RSA_FULL,
                    340:                          CRYPT_VERIFYCONTEXT | CRYPT_SILENT)) {
                    341:     CryptCreateHash(ctx->hCryptProv, CALG_MD5, 0, 0, &ctx->hHash);
                    342:   }
                    343:   return 1;
                    344: }
                    345: 
                    346: static void MD5_Update(MD5_CTX *ctx,
                    347:                        const unsigned char *input,
                    348:                        unsigned int inputLen)
                    349: {
                    350:   CryptHashData(ctx->hHash, (unsigned char *)input, inputLen, 0);
                    351: }
                    352: 
                    353: static void MD5_Final(unsigned char digest[16], MD5_CTX *ctx)
                    354: {
                    355:   win32_crypto_final(ctx, digest, 16);
                    356: }
                    357: 
                    358: static int SHA1_Init(SHA_CTX *ctx)
                    359: {
                    360:   if(CryptAcquireContext(&ctx->hCryptProv, NULL, NULL, PROV_RSA_FULL,
                    361:                          CRYPT_VERIFYCONTEXT | CRYPT_SILENT)) {
                    362:     CryptCreateHash(ctx->hCryptProv, CALG_SHA1, 0, 0, &ctx->hHash);
                    363:   }
                    364:   return 1;
                    365: }
                    366: 
                    367: static void SHA1_Update(SHA_CTX *ctx,
                    368:                         const unsigned char *input,
                    369:                         unsigned int inputLen)
                    370: {
                    371:   CryptHashData(ctx->hHash, (unsigned char *)input, inputLen, 0);
                    372: }
                    373: 
                    374: static void SHA1_Final(unsigned char digest[20], SHA_CTX *ctx)
                    375: {
                    376:   win32_crypto_final(ctx, digest, 20);
                    377: }
                    378: 
                    379: static int SHA256_Init(SHA256_CTX *ctx)
                    380: {
                    381:   if(CryptAcquireContext(&ctx->hCryptProv, NULL, NULL, PROV_RSA_AES,
                    382:                          CRYPT_VERIFYCONTEXT | CRYPT_SILENT)) {
                    383:     CryptCreateHash(ctx->hCryptProv, CALG_SHA_256, 0, 0, &ctx->hHash);
                    384:   }
                    385:   return 1;
                    386: }
                    387: 
                    388: static void SHA256_Update(SHA256_CTX *ctx,
                    389:                           const unsigned char *input,
                    390:                           unsigned int inputLen)
                    391: {
                    392:   CryptHashData(ctx->hHash, (unsigned char *)input, inputLen, 0);
                    393: }
                    394: 
                    395: static void SHA256_Final(unsigned char digest[32], SHA256_CTX *ctx)
                    396: {
                    397:   win32_crypto_final(ctx, digest, 32);
                    398: }
                    399: 
                    400: #endif /* CRYPTO LIBS */
                    401: 
                    402: const digest_params MD5_DIGEST_PARAMS[] = {
                    403:   {
                    404:     CURLX_FUNCTION_CAST(digest_init_func, MD5_Init),
                    405:     CURLX_FUNCTION_CAST(digest_update_func, MD5_Update),
                    406:     CURLX_FUNCTION_CAST(digest_final_func, MD5_Final),
                    407:     sizeof(MD5_CTX),
                    408:     16
                    409:   }
                    410: };
                    411: 
                    412: const digest_params SHA1_DIGEST_PARAMS[] = {
                    413:   {
                    414:     CURLX_FUNCTION_CAST(digest_init_func, SHA1_Init),
                    415:     CURLX_FUNCTION_CAST(digest_update_func, SHA1_Update),
                    416:     CURLX_FUNCTION_CAST(digest_final_func, SHA1_Final),
                    417:     sizeof(SHA_CTX),
                    418:     20
                    419:   }
                    420: };
                    421: 
                    422: const digest_params SHA256_DIGEST_PARAMS[] = {
                    423:   {
                    424:     CURLX_FUNCTION_CAST(digest_init_func, SHA256_Init),
                    425:     CURLX_FUNCTION_CAST(digest_update_func, SHA256_Update),
                    426:     CURLX_FUNCTION_CAST(digest_final_func, SHA256_Final),
                    427:     sizeof(SHA256_CTX),
                    428:     32
                    429:   }
                    430: };
                    431: 
                    432: static const metalink_digest_def SHA256_DIGEST_DEF[] = {
                    433:   {"sha-256", SHA256_DIGEST_PARAMS}
                    434: };
                    435: 
                    436: static const metalink_digest_def SHA1_DIGEST_DEF[] = {
                    437:   {"sha-1", SHA1_DIGEST_PARAMS}
                    438: };
                    439: 
                    440: static const metalink_digest_def MD5_DIGEST_DEF[] = {
                    441:   {"md5", MD5_DIGEST_PARAMS}
                    442: };
                    443: 
                    444: /*
                    445:  * The alias of supported hash functions in the order by preference
                    446:  * (basically stronger hash comes first). We included "sha-256" and
                    447:  * "sha256". The former is the name defined in the IANA registry named
                    448:  * "Hash Function Textual Names". The latter is widely (and
                    449:  * historically) used in Metalink version 3.
                    450:  */
                    451: static const metalink_digest_alias digest_aliases[] = {
                    452:   {"sha-256", SHA256_DIGEST_DEF},
                    453:   {"sha256", SHA256_DIGEST_DEF},
                    454:   {"sha-1", SHA1_DIGEST_DEF},
                    455:   {"sha1", SHA1_DIGEST_DEF},
                    456:   {"md5", MD5_DIGEST_DEF},
                    457:   {NULL, NULL}
                    458: };
                    459: 
                    460: static digest_context *digest_init(const digest_params *dparams)
                    461: {
                    462:   digest_context *ctxt;
                    463: 
                    464:   /* Create digest context */
                    465:   ctxt = malloc(sizeof(*ctxt));
                    466: 
                    467:   if(!ctxt)
                    468:     return ctxt;
                    469: 
                    470:   ctxt->digest_hashctx = malloc(dparams->digest_ctxtsize);
                    471: 
                    472:   if(!ctxt->digest_hashctx) {
                    473:     free(ctxt);
                    474:     return NULL;
                    475:   }
                    476: 
                    477:   ctxt->digest_hash = dparams;
                    478: 
                    479:   if(dparams->digest_init(ctxt->digest_hashctx) != 1) {
                    480:     free(ctxt->digest_hashctx);
                    481:     free(ctxt);
                    482:     return NULL;
                    483:   }
                    484: 
                    485:   return ctxt;
                    486: }
                    487: 
                    488: static int digest_update(digest_context *context,
                    489:                          const unsigned char *data,
                    490:                          unsigned int len)
                    491: {
                    492:   (*context->digest_hash->digest_update)(context->digest_hashctx, data, len);
                    493: 
                    494:   return 0;
                    495: }
                    496: 
                    497: static int digest_final(digest_context *context, unsigned char *result)
                    498: {
                    499:   if(result)
                    500:     (*context->digest_hash->digest_final)(result, context->digest_hashctx);
                    501: 
                    502:   free(context->digest_hashctx);
                    503:   free(context);
                    504: 
                    505:   return 0;
                    506: }
                    507: 
                    508: static unsigned char hex_to_uint(const char *s)
                    509: {
                    510:   char buf[3];
                    511:   unsigned long val;
                    512:   buf[0] = s[0];
                    513:   buf[1] = s[1];
                    514:   buf[2] = 0;
                    515:   val = strtoul(buf, NULL, 16);
                    516:   return (unsigned char)(val&0xff);
                    517: }
                    518: 
                    519: /*
                    520:  * Check checksum of file denoted by filename. The expected hash value
                    521:  * is given in hex_hash which is hex-encoded string.
                    522:  *
                    523:  * This function returns 1 if it succeeds or one of the following
                    524:  * integers:
                    525:  *
                    526:  * 0:
                    527:  *   Checksum didn't match.
                    528:  * -1:
                    529:  *   Could not open file; or could not read data from file.
                    530:  * -2:
                    531:  *   Hash algorithm not available.
                    532:  */
                    533: static int check_hash(const char *filename,
                    534:                       const metalink_digest_def *digest_def,
                    535:                       const unsigned char *digest, FILE *error)
                    536: {
                    537:   unsigned char *result;
                    538:   digest_context *dctx;
                    539:   int check_ok, flags, fd;
                    540: 
                    541:   flags = O_RDONLY;
                    542: #ifdef O_BINARY
                    543:   /* O_BINARY is required in order to avoid binary EOF in text mode */
                    544:   flags |= O_BINARY;
                    545: #endif
                    546: 
                    547:   fd = open(filename, flags);
                    548:   if(fd == -1) {
                    549:     fprintf(error, "Metalink: validating (%s) [%s] FAILED (%s)\n", filename,
                    550:             digest_def->hash_name, strerror(errno));
                    551:     return -1;
                    552:   }
                    553: 
                    554:   dctx = digest_init(digest_def->dparams);
                    555:   if(!dctx) {
                    556:     fprintf(error, "Metalink: validating (%s) [%s] FAILED (%s)\n", filename,
                    557:             digest_def->hash_name, "failed to initialize hash algorithm");
                    558:     close(fd);
                    559:     return -2;
                    560:   }
                    561: 
                    562:   result = malloc(digest_def->dparams->digest_resultlen);
                    563:   if(!result) {
                    564:     close(fd);
                    565:     digest_final(dctx, NULL);
                    566:     return -1;
                    567:   }
                    568:   while(1) {
                    569:     unsigned char buf[4096];
                    570:     ssize_t len = read(fd, buf, sizeof(buf));
                    571:     if(len == 0) {
                    572:       break;
                    573:     }
                    574:     else if(len == -1) {
                    575:       fprintf(error, "Metalink: validating (%s) [%s] FAILED (%s)\n", filename,
                    576:               digest_def->hash_name, strerror(errno));
                    577:       digest_final(dctx, result);
                    578:       close(fd);
                    579:       return -1;
                    580:     }
                    581:     digest_update(dctx, buf, (unsigned int)len);
                    582:   }
                    583:   digest_final(dctx, result);
                    584:   check_ok = memcmp(result, digest,
                    585:                     digest_def->dparams->digest_resultlen) == 0;
                    586:   /* sha*sum style verdict output */
                    587:   if(check_ok)
                    588:     fprintf(error, "Metalink: validating (%s) [%s] OK\n", filename,
                    589:             digest_def->hash_name);
                    590:   else
                    591:     fprintf(error, "Metalink: validating (%s) [%s] FAILED (digest mismatch)\n",
                    592:             filename, digest_def->hash_name);
                    593: 
                    594:   free(result);
                    595:   close(fd);
                    596:   return check_ok;
                    597: }
                    598: 
                    599: int metalink_check_hash(struct GlobalConfig *config,
                    600:                         metalinkfile *mlfile,
                    601:                         const char *filename)
                    602: {
                    603:   int rv;
                    604:   fprintf(config->errors, "Metalink: validating (%s)...\n", filename);
                    605:   if(mlfile->checksum == NULL) {
                    606:     fprintf(config->errors,
                    607:             "Metalink: validating (%s) FAILED (digest missing)\n", filename);
                    608:     return -2;
                    609:   }
                    610:   rv = check_hash(filename, mlfile->checksum->digest_def,
                    611:                   mlfile->checksum->digest, config->errors);
                    612:   return rv;
                    613: }
                    614: 
                    615: static metalink_checksum *
                    616: checksum_from_hex_digest(const metalink_digest_def *digest_def,
                    617:                          const char *hex_digest)
                    618: {
                    619:   metalink_checksum *chksum;
                    620:   unsigned char *digest;
                    621:   size_t i;
                    622:   size_t len = strlen(hex_digest);
                    623:   digest = malloc(len/2);
                    624:   if(!digest)
                    625:     return 0;
                    626: 
                    627:   for(i = 0; i < len; i += 2) {
                    628:     digest[i/2] = hex_to_uint(hex_digest + i);
                    629:   }
                    630:   chksum = malloc(sizeof(metalink_checksum));
                    631:   if(chksum) {
                    632:     chksum->digest_def = digest_def;
                    633:     chksum->digest = digest;
                    634:   }
                    635:   else
                    636:     free(digest);
                    637:   return chksum;
                    638: }
                    639: 
                    640: static metalink_resource *new_metalink_resource(const char *url)
                    641: {
                    642:   metalink_resource *res;
                    643:   res = malloc(sizeof(metalink_resource));
                    644:   if(res) {
                    645:     res->next = NULL;
                    646:     res->url = strdup(url);
                    647:     if(!res->url) {
                    648:       free(res);
                    649:       return NULL;
                    650:     }
                    651:   }
                    652:   return res;
                    653: }
                    654: 
                    655: /* Returns nonzero if hex_digest is properly formatted; that is each
                    656:    letter is in [0-9A-Za-z] and the length of the string equals to the
                    657:    result length of digest * 2. */
                    658: static int check_hex_digest(const char *hex_digest,
                    659:                             const metalink_digest_def *digest_def)
                    660: {
                    661:   size_t i;
                    662:   for(i = 0; hex_digest[i]; ++i) {
                    663:     char c = hex_digest[i];
                    664:     if(!(('0' <= c && c <= '9') || ('a' <= c && c <= 'z') ||
                    665:          ('A' <= c && c <= 'Z'))) {
                    666:       return 0;
                    667:     }
                    668:   }
                    669:   return digest_def->dparams->digest_resultlen * 2 == i;
                    670: }
                    671: 
                    672: static metalinkfile *new_metalinkfile(metalink_file_t *fileinfo)
                    673: {
                    674:   metalinkfile *f;
                    675:   f = (metalinkfile*)malloc(sizeof(metalinkfile));
                    676:   if(!f)
                    677:     return NULL;
                    678: 
                    679:   f->next = NULL;
                    680:   f->filename = strdup(fileinfo->name);
                    681:   if(!f->filename) {
                    682:     free(f);
                    683:     return NULL;
                    684:   }
                    685:   f->checksum = NULL;
                    686:   f->resource = NULL;
                    687:   if(fileinfo->checksums) {
                    688:     const metalink_digest_alias *digest_alias;
                    689:     for(digest_alias = digest_aliases; digest_alias->alias_name;
                    690:         ++digest_alias) {
                    691:       metalink_checksum_t **p;
                    692:       for(p = fileinfo->checksums; *p; ++p) {
                    693:         if(curl_strequal(digest_alias->alias_name, (*p)->type) &&
                    694:            check_hex_digest((*p)->hash, digest_alias->digest_def)) {
                    695:           f->checksum =
                    696:             checksum_from_hex_digest(digest_alias->digest_def,
                    697:                                      (*p)->hash);
                    698:           break;
                    699:         }
                    700:       }
                    701:       if(f->checksum) {
                    702:         break;
                    703:       }
                    704:     }
                    705:   }
                    706:   if(fileinfo->resources) {
                    707:     metalink_resource_t **p;
                    708:     metalink_resource root, *tail;
                    709:     root.next = NULL;
                    710:     tail = &root;
                    711:     for(p = fileinfo->resources; *p; ++p) {
                    712:       metalink_resource *res;
                    713:       /* Filter by type if it is non-NULL. In Metalink v3, type
                    714:          includes the type of the resource. In curl, we are only
                    715:          interested in HTTP, HTTPS and FTP. In addition to them,
                    716:          Metalink v3 file may contain bittorrent type URL, which
                    717:          points to the BitTorrent metainfo file. We ignore it here.
                    718:          In Metalink v4, type was deprecated and all
                    719:          fileinfo->resources point to the target file. BitTorrent
                    720:          metainfo file URL may be appeared in fileinfo->metaurls.
                    721:       */
                    722:       if((*p)->type == NULL ||
                    723:          curl_strequal((*p)->type, "http") ||
                    724:          curl_strequal((*p)->type, "https") ||
                    725:          curl_strequal((*p)->type, "ftp") ||
                    726:          curl_strequal((*p)->type, "ftps")) {
                    727:         res = new_metalink_resource((*p)->url);
                    728:         if(res) {
                    729:           tail->next = res;
                    730:           tail = res;
                    731:         }
                    732:         else {
                    733:           tail = root.next;
                    734: 
                    735:           /* clean up the linked list */
                    736:           while(tail) {
                    737:             res = tail->next;
                    738:             free(tail->url);
                    739:             free(tail);
                    740:             tail = res;
                    741:           }
                    742:           free(f->filename);
                    743:           free(f);
                    744:           return NULL;
                    745:         }
                    746:       }
                    747:     }
                    748:     f->resource = root.next;
                    749:   }
                    750:   return f;
                    751: }
                    752: 
                    753: int parse_metalink(struct OperationConfig *config, struct OutStruct *outs,
                    754:                    const char *metalink_url)
                    755: {
                    756:   metalink_error_t r;
                    757:   metalink_t* metalink;
                    758:   metalink_file_t **files;
                    759:   bool warnings = FALSE;
                    760: 
                    761:   /* metlaink_parse_final deletes outs->metalink_parser */
                    762:   r = metalink_parse_final(outs->metalink_parser, NULL, 0, &metalink);
                    763:   outs->metalink_parser = NULL;
                    764:   if(r != 0) {
                    765:     return -1;
                    766:   }
                    767:   if(metalink->files == NULL) {
                    768:     fprintf(config->global->errors, "Metalink: parsing (%s) WARNING "
                    769:             "(missing or invalid file name)\n",
                    770:             metalink_url);
                    771:     metalink_delete(metalink);
                    772:     return -1;
                    773:   }
                    774:   for(files = metalink->files; *files; ++files) {
                    775:     struct getout *url;
                    776:     /* Skip an entry which has no resource. */
                    777:     if(!(*files)->resources) {
                    778:       fprintf(config->global->errors, "Metalink: parsing (%s) WARNING "
                    779:               "(missing or invalid resource)\n",
                    780:               metalink_url);
                    781:       continue;
                    782:     }
                    783:     if(config->url_get ||
                    784:        ((config->url_get = config->url_list) != NULL)) {
                    785:       /* there's a node here, if it already is filled-in continue to
                    786:          find an "empty" node */
                    787:       while(config->url_get && (config->url_get->flags & GETOUT_URL))
                    788:         config->url_get = config->url_get->next;
                    789:     }
                    790: 
                    791:     /* now there might or might not be an available node to fill in! */
                    792: 
                    793:     if(config->url_get)
                    794:       /* existing node */
                    795:       url = config->url_get;
                    796:     else
                    797:       /* there was no free node, create one! */
                    798:       url = new_getout(config);
                    799: 
                    800:     if(url) {
                    801:       metalinkfile *mlfile = new_metalinkfile(*files);
                    802:       if(!mlfile)
                    803:         break;
                    804: 
                    805:       if(!mlfile->checksum) {
                    806:         warnings = TRUE;
                    807:         fprintf(config->global->errors,
                    808:                 "Metalink: parsing (%s) WARNING (digest missing)\n",
                    809:                 metalink_url);
                    810:       }
                    811:       /* Set name as url */
                    812:       GetStr(&url->url, mlfile->filename);
                    813: 
                    814:       /* set flag metalink here */
                    815:       url->flags |= GETOUT_URL | GETOUT_METALINK;
                    816: 
                    817:       if(config->metalinkfile_list) {
                    818:         config->metalinkfile_last->next = mlfile;
                    819:         config->metalinkfile_last = mlfile;
                    820:       }
                    821:       else {
                    822:         config->metalinkfile_list = config->metalinkfile_last = mlfile;
                    823:       }
                    824:     }
                    825:   }
                    826:   metalink_delete(metalink);
                    827:   return (warnings) ? -2 : 0;
                    828: }
                    829: 
                    830: size_t metalink_write_cb(void *buffer, size_t sz, size_t nmemb,
                    831:                          void *userdata)
                    832: {
                    833:   struct per_transfer *per = userdata;
                    834:   struct OutStruct *outs = &per->outs;
                    835:   struct OperationConfig *config = per->config;
                    836:   int rv;
                    837: 
                    838:   /*
                    839:    * Once that libcurl has called back tool_write_cb() the returned value
                    840:    * is checked against the amount that was intended to be written, if
                    841:    * it does not match then it fails with CURLE_WRITE_ERROR. So at this
                    842:    * point returning a value different from sz*nmemb indicates failure.
                    843:    */
                    844:   const size_t failure = (sz && nmemb) ? 0 : 1;
                    845: 
                    846:   if(!config)
                    847:     return failure;
                    848: 
                    849:   rv = metalink_parse_update(outs->metalink_parser, buffer, sz * nmemb);
                    850:   if(rv == 0)
                    851:     return sz * nmemb;
                    852:   else {
                    853:     fprintf(config->global->errors, "Metalink: parsing FAILED\n");
                    854:     return failure;
                    855:   }
                    856: }
                    857: 
                    858: /*
                    859:  * Returns nonzero if content_type includes mediatype.
                    860:  */
                    861: static int check_content_type(const char *content_type, const char *media_type)
                    862: {
                    863:   const char *ptr = content_type;
                    864:   size_t media_type_len = strlen(media_type);
                    865:   for(; *ptr && (*ptr == ' ' || *ptr == '\t'); ++ptr);
                    866:   if(!*ptr) {
                    867:     return 0;
                    868:   }
                    869:   return curl_strnequal(ptr, media_type, media_type_len) &&
                    870:     (*(ptr + media_type_len) == '\0' || *(ptr + media_type_len) == ' ' ||
                    871:      *(ptr + media_type_len) == '\t' || *(ptr + media_type_len) == ';');
                    872: }
                    873: 
                    874: int check_metalink_content_type(const char *content_type)
                    875: {
                    876:   return check_content_type(content_type, "application/metalink+xml");
                    877: }
                    878: 
                    879: int count_next_metalink_resource(metalinkfile *mlfile)
                    880: {
                    881:   int count = 0;
                    882:   metalink_resource *res;
                    883:   for(res = mlfile->resource; res; res = res->next, ++count);
                    884:   return count;
                    885: }
                    886: 
                    887: static void delete_metalink_checksum(metalink_checksum *chksum)
                    888: {
                    889:   if(chksum == NULL) {
                    890:     return;
                    891:   }
                    892:   Curl_safefree(chksum->digest);
                    893:   Curl_safefree(chksum);
                    894: }
                    895: 
                    896: static void delete_metalink_resource(metalink_resource *res)
                    897: {
                    898:   if(res == NULL) {
                    899:     return;
                    900:   }
                    901:   Curl_safefree(res->url);
                    902:   Curl_safefree(res);
                    903: }
                    904: 
                    905: void delete_metalinkfile(metalinkfile *mlfile)
                    906: {
                    907:   metalink_resource *res;
                    908:   if(mlfile == NULL) {
                    909:     return;
                    910:   }
                    911:   Curl_safefree(mlfile->filename);
                    912:   delete_metalink_checksum(mlfile->checksum);
                    913:   for(res = mlfile->resource; res;) {
                    914:     metalink_resource *next;
                    915:     next = res->next;
                    916:     delete_metalink_resource(res);
                    917:     res = next;
                    918:   }
                    919:   Curl_safefree(mlfile);
                    920: }
                    921: 
                    922: void clean_metalink(struct OperationConfig *config)
                    923: {
                    924:   if(config) {
                    925:     while(config->metalinkfile_list) {
                    926:       metalinkfile *mlfile = config->metalinkfile_list;
                    927:       config->metalinkfile_list = config->metalinkfile_list->next;
                    928:       delete_metalinkfile(mlfile);
                    929:     }
                    930:     config->metalinkfile_last = 0;
                    931:   }
                    932: }
                    933: 
                    934: void metalink_cleanup(void)
                    935: {
                    936: #ifdef HAVE_NSS_CONTEXT
                    937:   if(nss_context) {
                    938:     NSS_ShutdownContext(nss_context);
                    939:     nss_context = NULL;
                    940:   }
                    941: #endif
                    942: }
                    943: 
                    944: #endif /* USE_METALINK */

FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>