Annotation of libwww/Library/src/HTProxy.c, revision 2.20.2.1
2.1       frystyk     1: /*                                                                  HTProxy.c
                      2: **     GATEWAY AND PROXY MANAGER
                      3: **
                      4: **     (c) COPYRIGHT MIT 1995.
                      5: **     Please first read the full copyright statement in the file COPYRIGH.
2.20.2.1! kahan       6: **     @(#) $Id: HTProxy.c,v 2.20 1999/06/03 18:16:09 frystyk Exp $
2.1       frystyk     7: **
                      8: **     Replaces the old env variables for gateways and proxies. However for
                      9: **     backward compatibility there is a function that reads the env variables
                     10: **     at start up. Note that there is a difference between a proxy and a
                     11: **     gateway!
                     12: **
                     13: ** Authors
                     14: **     HF      Henrik Frystyk, frystyk@w3.org
                     15: ** History
                     16: **       4 Jun 95 Written on a rainy day
                     17: */
                     18: 
                     19: #if !defined(HT_DIRECT_WAIS) && !defined(HT_DEFAULT_WAIS_GATEWAY)
                     20: #define HT_DEFAULT_WAIS_GATEWAY "http://www.w3.org:8001/"
                     21: #endif
                     22: 
                     23: /* Library include files */
2.15      frystyk    24: #include "wwwsys.h"
2.11      frystyk    25: #include "WWWUtil.h"
                     26: #include "WWWCore.h"
                     27: #include "WWWHTTP.h"
2.12      frystyk    28: #include "WWWApp.h"
2.1       frystyk    29: #include "HTProxy.h"                                    /* Implemented here */
                     30: 
                     31: /* Variables and typedefs local to this module */
                     32: 
                     33: typedef struct _HTProxy {
                     34:     char *     access;
                     35:     char *     url;                              /* URL of Gateway or Proxy */
2.17      frystyk    36: #ifdef HT_POSIX_REGEX
                     37:     regex_t *  regex;                            /* Compiled regex */
                     38: #endif
2.1       frystyk    39: } HTProxy;
                     40: 
                     41: typedef struct _HTHostlist {
                     42:     char *     access;
2.17      frystyk    43:     char *     host;                             /* Host or domain name */
2.1       frystyk    44:     unsigned   port;
2.17      frystyk    45: #ifdef HT_POSIX_REGEX
                     46:     regex_t *  regex;                            /* Compiled regex */
                     47: #endif
2.1       frystyk    48: } HTHostList;
                     49: 
                     50: PRIVATE HTList * proxies = NULL;                   /* List of proxy servers */
                     51: PRIVATE HTList * gateways = NULL;                       /* List of gateways */
                     52: PRIVATE HTList * noproxy = NULL;   /* Don't proxy on these hosts and domains */
2.20.2.1! kahan      53: PRIVATE int      noproxy_is_onlyproxy = 0; /* Interpret the noproxy list as an onlyproxy one */
2.1       frystyk    54: 
                     55: #if 0
                     56: PRIVATE HTList * onlyproxy = NULL;  /* Proxy only on these hosts and domains */
                     57: #endif
                     58: 
                     59: /* ------------------------------------------------------------------------- */
                     60: 
2.17      frystyk    61: #ifdef HT_POSIX_REGEX
                     62: PRIVATE char * get_regex_error (int errcode, regex_t * compiled)
                     63: {
                     64:     size_t length = regerror (errcode, compiled, NULL, 0);
                     65:     char * str = NULL;
                     66:     if ((str = (char *) HT_MALLOC(length+1)) == NULL)
                     67:        HT_OUTOFMEM("get_regex_error");
                     68:     (void) regerror (errcode, compiled, str, length);
                     69:     return str;
                     70: }
                     71: 
                     72: PRIVATE regex_t * get_regex_t (const char * regex_str, int cflags)
                     73: {
                     74:     regex_t * regex = NULL;
                     75:     if (regex_str && *regex_str) {
                     76:        int status;
                     77:        if ((regex = (regex_t *) HT_CALLOC(1, sizeof(regex_t))) == NULL)
                     78:            HT_OUTOFMEM("get_regex_t");
                     79:        if ((status = regcomp(regex, regex_str, cflags))) {
                     80:            char * err_msg = get_regex_error(status, regex);
2.18      frystyk    81:            HTTRACE(PROT_TRACE, "HTProxy..... Regular expression error: %s\n" _ err_msg);
2.17      frystyk    82:            HT_FREE(err_msg);
                     83:            HT_FREE(regex);
                     84:        }
                     85:     }
                     86:     return regex;
                     87: }
                     88: #endif
                     89: 
2.4       frystyk    90: /*
2.1       frystyk    91: **     Existing entries are replaced with new ones
                     92: */
2.17      frystyk    93: PRIVATE BOOL add_object (HTList * list, const char * access, const char * url,
                     94:                         BOOL regex, int regex_flags)
2.1       frystyk    95: {
                     96:     HTProxy *me;
                     97:     if (!list || !access || !url || !*url)
                     98:        return NO;
2.7       frystyk    99:     if ((me = (HTProxy *) HT_CALLOC(1, sizeof(HTProxy))) == NULL)
                    100:        HT_OUTOFMEM("add_object");
2.1       frystyk   101:     StrAllocCopy(me->access, access);                      /* Access method */
2.17      frystyk   102: 
                    103: #ifdef HT_POSIX_REGEX
                    104:     /* 
                    105:     **  If we support regular expressions then compile one up for
                    106:     **  this regular expression. Otherwise use is as a normal
                    107:     **  access scheme.
                    108:     */
                    109:     if (regex) {
                    110:        me->regex = get_regex_t(access,
                    111:                                regex_flags < 0 ?
                    112:                                W3C_DEFAULT_REGEX_FLAGS : regex_flags);
                    113:     } else
                    114: #endif
2.1       frystyk   115:     {
                    116:        char *ptr = me->access;
                    117:        while ((*ptr = TOLOWER(*ptr))) ptr++;
                    118:     }
2.17      frystyk   119: 
2.1       frystyk   120:     me->url = HTParse(url, "", PARSE_ACCESS+PARSE_HOST+PARSE_PUNCTUATION);
                    121:     if (*(me->url+strlen(me->url)-1) != '/')
                    122:        StrAllocCat(me->url, "/");
                    123:     me->url = HTSimplify(&me->url);
                    124: 
                    125:     /* See if we already have this one */
                    126:     {
                    127:        HTList *cur = list;
                    128:        HTProxy *pres;
                    129:        while ((pres = (HTProxy *) HTList_nextObject(cur)) != NULL) {
                    130:            if (!strcmp(pres->access, me->access))
                    131:                break;                                 /* We already have it */
                    132:        }
                    133:        if (pres) {
2.18      frystyk   134:            HTTRACE(PROT_TRACE, "HTProxy..... replacing for `%s\' access %s\n" _ 
                    135:                        me->url _ me->access);
2.7       frystyk   136:            HT_FREE(pres->access);
                    137:            HT_FREE(pres->url);
2.17      frystyk   138: #ifdef HT_POSIX_REGEX
                    139:            if (pres->regex) regfree(pres->regex);
                    140: #endif
2.1       frystyk   141:            HTList_removeObject(list, (void *) pres);
2.7       frystyk   142:            HT_FREE(pres);
2.1       frystyk   143:        }
2.18      frystyk   144:        HTTRACE(PROT_TRACE, "HTProxy..... adding for `%s\' access %s\n" _ 
                    145:                    me->url _ me->access);
2.1       frystyk   146:        HTList_addObject(list, (void *) me);
                    147:     }
                    148:     return YES;
                    149: }
                    150: 
2.4       frystyk   151: PRIVATE BOOL remove_allObjects (HTList * list)
2.1       frystyk   152: {
                    153:     if (list) {
                    154:        HTList *cur = list;
                    155:        HTProxy *pres;
                    156:        while ((pres = (HTProxy *) HTList_nextObject(cur)) != NULL) {
2.7       frystyk   157:            HT_FREE(pres->access);
                    158:            HT_FREE(pres->url);
2.17      frystyk   159: #ifdef HT_POSIX_REGEX
                    160:            if (pres->regex) regfree(pres->regex);
                    161: #endif
2.7       frystyk   162:            HT_FREE(pres);
2.1       frystyk   163:        }
                    164:        return YES;
                    165:     }
                    166:     return NO;
                    167: }
                    168: 
2.4       frystyk   169: /*     Add an entry to a list of host names
                    170: **     ------------------------------------
                    171: **     Existing entries are replaced with new ones
                    172: */
2.9       frystyk   173: PRIVATE BOOL add_hostname (HTList * list, const char * host,
2.17      frystyk   174:                           const char * access, unsigned port,
                    175:                           BOOL regex, int regex_flags)
2.4       frystyk   176: {
                    177:     HTHostList *me;
                    178:     if (!list || !host || !*host)
                    179:        return NO;
2.7       frystyk   180:     if ((me = (HTHostList *) HT_CALLOC(1, sizeof(HTHostList))) == NULL)
                    181:         HT_OUTOFMEM("add_hostname");
2.17      frystyk   182: #ifdef HT_POSIX_REGEX
                    183:     if (regex)
                    184:        me->regex = get_regex_t(host,
                    185:                                regex_flags < 0 ?
                    186:                                W3C_DEFAULT_REGEX_FLAGS : regex_flags);
                    187: #endif
                    188: 
2.4       frystyk   189:     if (access) {
                    190:        char *ptr;
                    191:        StrAllocCopy(me->access, access);                   /* Access method */
                    192:        ptr = me->access;
                    193:        while ((*ptr = TOLOWER(*ptr))) ptr++;
                    194:     }
                    195:     StrAllocCopy(me->host, host);                              /* Host name */
                    196:     {
                    197:        char *ptr = me->host;
                    198:        while ((*ptr = TOLOWER(*ptr))) ptr++;
                    199:     }
                    200:     me->port = port;                                         /* Port number */
2.18      frystyk   201:     HTTRACE(PROT_TRACE, "HTHostList.. adding `%s\' to list\n" _ me->host);
2.4       frystyk   202:     HTList_addObject(list, (void *) me);
                    203:     return YES;
                    204: }
2.1       frystyk   205: 
2.4       frystyk   206: PRIVATE BOOL remove_AllHostnames (HTList * list)
                    207: {
                    208:     if (list) {
                    209:        HTList *cur = list;
                    210:        HTHostList *pres;
                    211:        while ((pres = (HTHostList *) HTList_nextObject(cur)) != NULL) {
2.7       frystyk   212:            HT_FREE(pres->access);
                    213:            HT_FREE(pres->host);
2.17      frystyk   214: #ifdef HT_POSIX_REGEX
                    215:            if (pres->regex) regfree(pres->regex);
                    216: #endif
2.7       frystyk   217:            HT_FREE(pres);
2.4       frystyk   218:        }
                    219:        return YES;
                    220:     }
                    221:     return NO;
                    222: }
                    223: 
                    224: /*     HTProxy_add
                    225: **     -----------
2.1       frystyk   226: **     Registers a proxy as the server to contact for a specific
                    227: **     access method. `proxy' should be a fully valid name, like
                    228: **     "http://proxy.w3.org:8001" but domain name is not required.
                    229: **     If an entry exists for this access then delete it and use the 
                    230: **     ne one. Returns YES if OK, else NO
                    231: */
2.9       frystyk   232: PUBLIC BOOL HTProxy_add (const char * access, const char * proxy)
2.1       frystyk   233: {
2.11      frystyk   234:     /*
                    235:     **  If this is the first time here then also add a before filter to handle
2.12      frystyk   236:     **  proxy authentication and the normal AA after filter as well.
                    237:     **  These filters will be removed if we remove all proxies again.
2.11      frystyk   238:     */
                    239:     if (!proxies) {
                    240:        proxies = HTList_new();
2.13      frystyk   241:        HTNet_addBefore(HTAA_proxyBeforeFilter, NULL, NULL,
                    242:                        HT_FILTER_MIDDLE);
                    243:        HTNet_addAfter(HTAuthFilter, NULL, NULL,
                    244:                       HT_NO_PROXY_ACCESS, HT_FILTER_MIDDLE);
2.14      frystyk   245:        HTNet_addAfter(HTAuthFilter, NULL, NULL,
                    246:                       HT_PROXY_REAUTH, HT_FILTER_MIDDLE);
2.11      frystyk   247:     }
2.17      frystyk   248:     return add_object(proxies, access, proxy, NO, -1);
                    249: }
                    250: 
                    251: /*     HTProxy_addRegex
                    252: **     ----------------
                    253: **     Registers a proxy as the server to contact for any URL matching the
                    254: **     regular expression. `proxy' should be a fully valid name, like
                    255: **     "http://proxy.w3.org:8001".
                    256: **     If an entry exists for this access then delete it and use the 
                    257: **     new one. Returns YES if OK, else NO
                    258: */
                    259: PUBLIC BOOL HTProxy_addRegex (const char * regex,
                    260:                              const char * proxy,
                    261:                              int regex_flags)
                    262: {
                    263:     /*
                    264:     **  If this is the first time here then also add a before filter to handle
                    265:     **  proxy authentication and the normal AA after filter as well.
                    266:     **  These filters will be removed if we remove all proxies again.
                    267:     */
                    268:     if (!proxies) {
                    269:        proxies = HTList_new();
                    270:        HTNet_addBefore(HTAA_proxyBeforeFilter, NULL, NULL,
                    271:                        HT_FILTER_MIDDLE);
                    272:        HTNet_addAfter(HTAuthFilter, NULL, NULL,
                    273:                       HT_NO_PROXY_ACCESS, HT_FILTER_MIDDLE);
                    274:        HTNet_addAfter(HTAuthFilter, NULL, NULL,
                    275:                       HT_PROXY_REAUTH, HT_FILTER_MIDDLE);
                    276:     }
                    277: #ifdef HT_POSIX_REGEX
                    278:     return add_object(proxies, regex, proxy, YES, regex_flags);
                    279: #else
                    280:     return add_object(proxies, regex, proxy, NO, -1);
                    281: #endif
2.1       frystyk   282: }
                    283: 
2.4       frystyk   284: /*
2.1       frystyk   285: **     Removes all registered proxies
                    286: */
2.4       frystyk   287: PUBLIC BOOL HTProxy_deleteAll (void)
2.1       frystyk   288: {
2.4       frystyk   289:     if (remove_allObjects(proxies)) {
2.1       frystyk   290:        HTList_delete(proxies);
2.11      frystyk   291: 
                    292:        /*
                    293:        ** If we have no more proxies then there is no reason for checking
2.12      frystyk   294:        ** proxy authentication. We therefore unregister the filters for
                    295:        ** handling proxy authentication
2.11      frystyk   296:        */
2.13      frystyk   297:        HTNet_deleteBefore(HTAA_proxyBeforeFilter);
2.19      frystyk   298:         HTNet_deleteAfterStatus(HT_NO_PROXY_ACCESS);
                    299:         HTNet_deleteAfterStatus(HT_PROXY_REAUTH);
2.11      frystyk   300: 
2.1       frystyk   301:        proxies = NULL;
                    302:        return YES;
                    303:     }
                    304:     return NO;
                    305: }
                    306: 
2.4       frystyk   307: /*     HTGateway_add
                    308: **     -------------
2.1       frystyk   309: **     Registers a gateway as the server to contact for a specific
                    310: **     access method. `gateway' should be a fully valid name, like
                    311: **     "http://gateway.w3.org:8001" but domain name is not required.
                    312: **     If an entry exists for this access then delete it and use the 
                    313: **     ne one. Returns YES if OK, else NO
                    314: */
2.9       frystyk   315: PUBLIC BOOL HTGateway_add (const char * access, const char * gate)
2.1       frystyk   316: {
                    317:     if (!gateways)
                    318:        gateways = HTList_new();
2.17      frystyk   319:     return add_object(gateways, access, gate, NO, -1);
2.1       frystyk   320: }
                    321: 
2.4       frystyk   322: /*
2.1       frystyk   323: **     Removes all registered gateways
                    324: */
2.4       frystyk   325: PUBLIC BOOL HTGateway_deleteAll (void)
2.1       frystyk   326: {
2.4       frystyk   327:     if (remove_allObjects(gateways)) {
2.1       frystyk   328:        HTList_delete(gateways);
                    329:        gateways = NULL;
                    330:        return YES;
                    331:     }
                    332:     return NO;
                    333: }
                    334: 
2.4       frystyk   335: /*     HTNoProxy_add
                    336: **     -------------
2.1       frystyk   337: **     Registers a host name or a domain as a place where no proxy should
                    338: **     be contacted - for example a very fast link. If `port' is '0' then
                    339: **     it applies to all ports and if `access' is NULL then it applies to
                    340: **     to all access methods.
                    341: **
                    342: **     Examples:       w3.org
                    343: **                     www.close.com
                    344: */
2.9       frystyk   345: PUBLIC BOOL HTNoProxy_add (const char * host, const char * access,
2.4       frystyk   346:                           unsigned port)
2.1       frystyk   347: {
                    348:     if (!noproxy)
                    349:        noproxy = HTList_new();    
2.17      frystyk   350:     return add_hostname(noproxy, host, access, port, NO, -1);
                    351: }
                    352: 
                    353: /*     HTNoProxy_addRegex
                    354: **     ------------------
                    355: **     Registers a regular expression where URIs matching this expression
                    356: **      should go directly and not via a proxy.
                    357: **
                    358: */
                    359: PUBLIC BOOL HTNoProxy_addRegex (const char * regex, int regex_flags)
                    360: {
                    361:     if (!noproxy)
                    362:        noproxy = HTList_new();    
                    363: #ifdef HT_POSIX_REGEX
                    364:     return add_hostname(noproxy, regex, NULL, 0, YES, regex_flags);
                    365: #else
                    366:     return add_hostname(noproxy, regex, NULL, 0, NO, -1);
                    367: #endif
2.1       frystyk   368: }
                    369: 
2.4       frystyk   370: /*     HTNoProxy_deleteAll
                    371: **     -------------------
2.1       frystyk   372: **     Removes all registered no_proxy directives
                    373: */
2.4       frystyk   374: PUBLIC BOOL HTNoProxy_deleteAll (void)
2.1       frystyk   375: {
2.4       frystyk   376:     if (remove_AllHostnames(noproxy)) {
2.1       frystyk   377:        HTList_delete(noproxy);
                    378:        noproxy = NULL;
                    379:        return YES;
                    380:     }
                    381:     return NO;
                    382: }
                    383: 
2.20.2.1! kahan     384: 
        !           385: /*     HTNProxy_noProxyIsOnlyProxy
        !           386: **     `----------------------------
        !           387: **     Returns the state of the noproxy_is_onlyproxy flag
        !           388: */
        !           389: PUBLIC int HTProxy_NoProxyIsOnlyProxy (void)
        !           390: {
        !           391:   return noproxy_is_onlyproxy;
        !           392: }
        !           393: 
        !           394: /*     HTNProxy_setNoProxyisOnlyProxy
        !           395: **     --------------------------
        !           396: **     Sets the state of the noproxy_is_onlyproxy flag
        !           397: */
        !           398: PUBLIC void HTProxy_setNoProxyIsOnlyProxy (int value)
        !           399: {
        !           400:   noproxy_is_onlyproxy = value;
        !           401: }
        !           402: 
2.4       frystyk   403: /*     HTProxy_find
                    404: **     ------------
2.1       frystyk   405: **     This function evaluates the lists of registered proxies and if
                    406: **     one is found for the actual access method and it is not registered
                    407: **     in the `noproxy' list, then a URL containing the host to be contacted
                    408: **     is returned to the caller. This string must be freed be the caller.
                    409: **
                    410: **     Returns: proxy  If OK (must be freed by caller)
                    411: **              NULL   If no proxy is found or error
                    412: */
2.9       frystyk   413: PUBLIC char * HTProxy_find (const char * url)
2.1       frystyk   414: {
                    415:     char * access;
                    416:     char * proxy = NULL;
2.20.2.1! kahan     417:     int  no_proxy_found = 0;
        !           418: 
2.1       frystyk   419:     if (!url || !proxies)
                    420:        return NULL;
                    421:     access = HTParse(url, "", PARSE_ACCESS);
                    422: 
                    423:     /* First check if the host (if any) is registered in the noproxy list */
                    424:     if (noproxy) {
                    425:        char *host = HTParse(url, "", PARSE_HOST);
                    426:        char *ptr;
                    427:        unsigned port=0;
                    428:        if ((ptr = strchr(host, ':')) != NULL) {
                    429:            *ptr++ = '\0';                                  /* Chop off port */
                    430:            if (*ptr) port = (unsigned) atoi(ptr);
                    431:        }
                    432:        if (*host) {                               /* If we have a host name */
                    433:            HTList *cur = noproxy;
                    434:            HTHostList *pres;
                    435:            while ((pres = (HTHostList *) HTList_nextObject(cur)) != NULL) {
2.17      frystyk   436: #ifdef HT_POSIX_REGEX
                    437:                if (pres->regex) {
                    438:                    BOOL match = regexec(pres->regex, url, 0, NULL, 0) ? NO : YES;
                    439:                    if (match) {
2.18      frystyk   440:                        HTTRACE(PROT_TRACE, "GetProxy.... No proxy directive found: `%s\'\n" _ pres->host);
2.20.2.1! kahan     441:                        no_proxy_found = 1;
        !           442:                        break;
2.17      frystyk   443:                    }
                    444:                } else
                    445: #endif
2.1       frystyk   446:                if (!pres->access ||
                    447:                    (pres->access && !strcmp(pres->access, access))) {
2.20      frystyk   448:                    if ((pres->port == 0) || (pres->port == port)) {
2.1       frystyk   449:                        char *np = pres->host+strlen(pres->host);
                    450:                        char *hp = host+strlen(host);
                    451:                        while (np>=pres->host && hp>=host && (*np--==*hp--));
                    452:                        if (np==pres->host-1 && (hp==host-1 || *hp=='.')) {
2.18      frystyk   453:                            HTTRACE(PROT_TRACE, "GetProxy.... No proxy directive found: `%s\'\n" _ pres->host);
2.20.2.1! kahan     454:                            no_proxy_found = 1;
        !           455:                            break;
2.1       frystyk   456:                        }
                    457:                    }
                    458:                }
                    459:            }
                    460:        }
2.7       frystyk   461:        HT_FREE(host);
2.1       frystyk   462:     }
                    463: 
2.20.2.1! kahan     464:     if ((no_proxy_found && !noproxy_is_onlyproxy) 
        !           465:        || (!no_proxy_found && noproxy_is_onlyproxy)) {
        !           466:       HT_FREE(access);
        !           467:       return NULL;
        !           468:     }
        !           469:     
2.1       frystyk   470:     /* Now check if we have a proxy registered for this access method */
                    471:     {
                    472:        HTList *cur = proxies;
                    473:        HTProxy *pres;
                    474:        while ((pres = (HTProxy *) HTList_nextObject(cur)) != NULL) {
2.17      frystyk   475: #ifdef HT_POSIX_REGEX
                    476:            if (pres->regex) {
                    477:                BOOL match = regexec(pres->regex, url, 0, NULL, 0) ? NO : YES;
                    478:                if (match) {
                    479:                    StrAllocCopy(proxy, pres->url);
2.18      frystyk   480:                    HTTRACE(PROT_TRACE, "GetProxy.... Found: `%s\'\n" _ pres->url);
2.17      frystyk   481:                    break;
                    482:                }
                    483:            } else
                    484: #endif
2.1       frystyk   485:            if (!strcmp(pres->access, access)) {
                    486:                StrAllocCopy(proxy, pres->url);
2.18      frystyk   487:                HTTRACE(PROT_TRACE, "GetProxy.... Found: `%s\'\n" _ pres->url);
2.1       frystyk   488:                break;
                    489:            }
                    490:        }
                    491:     }
2.7       frystyk   492:     HT_FREE(access);
2.1       frystyk   493:     return proxy;
2.9       frystyk   494: }
2.1       frystyk   495: 
                    496: 
2.4       frystyk   497: /*     HTGateway_find
                    498: **     --------------
2.1       frystyk   499: **     This function evaluates the lists of registered gateways and if
                    500: **     one is found for the actual access method then it is returned
                    501: **
                    502: **     Returns: gateway If OK (must be freed by caller)
                    503: **              NULL    If no gateway is found or error
                    504: */
2.9       frystyk   505: PUBLIC char * HTGateway_find (const char * url)
2.1       frystyk   506: {
                    507:     char * access;
                    508:     char * gateway = NULL;
                    509:     if (!url || !gateways)
                    510:        return NULL;
                    511:     access = HTParse(url, "", PARSE_ACCESS);
                    512: 
                    513:     /* Check if we have a gateway registered for this access method */
                    514:     {
                    515:        HTList *cur = gateways;
                    516:        HTProxy *pres;
                    517:        while ((pres = (HTProxy *) HTList_nextObject(cur)) != NULL) {
                    518:            if (!strcmp(pres->access, access)) {
                    519:                StrAllocCopy(gateway, pres->url);
2.18      frystyk   520:                HTTRACE(PROT_TRACE, "GetGateway.. Found: `%s\'\n" _ pres->url);
2.1       frystyk   521:                break;
                    522:            }
                    523:        }
                    524:     }
2.7       frystyk   525:     HT_FREE(access);
2.1       frystyk   526:     return gateway;
                    527: }
                    528: 
                    529: 
                    530: /*
                    531: **     This function maintains backwards compatibility with the old 
                    532: **     environment variables and searches for the most common values:
                    533: **     http, ftp, news, wais, and gopher
                    534: */
2.4       frystyk   535: PUBLIC void HTProxy_getEnvVar (void)
2.1       frystyk   536: {
                    537:     char buf[80];
2.9       frystyk   538:     static const char *accesslist[] = {
2.1       frystyk   539:        "http",
                    540:        "ftp",
                    541:        "news",
                    542:        "wais",
                    543:        "gopher",
                    544:        NULL
                    545:     };
2.9       frystyk   546:     const char **access = accesslist;
2.18      frystyk   547:     HTTRACE(PROT_TRACE, "Proxy....... Looking for environment variables\n");
2.1       frystyk   548:     while (*access) {
2.11      frystyk   549:        BOOL found = NO;
2.1       frystyk   550:        char *gateway=NULL;
                    551:        char *proxy=NULL;
                    552: 
2.11      frystyk   553:        /* Search for proxy gateways */
                    554:        if (found == NO) {
                    555:            strcpy(buf, *access);
                    556:            strcat(buf, "_proxy");
                    557:            if ((proxy = (char *) getenv(buf)) && *proxy) {
                    558:                HTProxy_add(*access, proxy);
                    559:                found = YES;
                    560:            }
                    561: 
                    562:            /* Try the same with upper case */
                    563:            if (found == NO) {
                    564:                char * up = buf;
                    565:                while ((*up = TOUPPER(*up))) up++;
                    566:                if ((proxy = (char *) getenv(buf)) && *proxy) {
                    567:                    HTProxy_add(*access, proxy);
                    568:                    found = YES;
                    569:                }
                    570:            }
                    571:        }
                    572: 
                    573:        /* As a last resort, search for gateway servers */
                    574:        if (found == NO) {
                    575:            strcpy(buf, "WWW_");
                    576:            strcat(buf, *access);
                    577:            strcat(buf, "_GATEWAY");
                    578:            if ((gateway = (char *) getenv(buf)) && *gateway) {
                    579:                HTGateway_add(*access, gateway);
                    580:                found = YES;
                    581:            }
                    582:        }
2.1       frystyk   583:        ++access;
                    584:     }
                    585: 
                    586:     /* Search for `noproxy' directive */
                    587:     {
                    588:        char *noproxy = getenv("no_proxy");
                    589:        if (noproxy && *noproxy) {
                    590:            char *str = NULL;
                    591:            char *strptr;
                    592:            char *name;
                    593:            StrAllocCopy(str, noproxy);          /* Get copy we can mutilate */
                    594:            strptr = str;
                    595:            while ((name = HTNextField(&strptr)) != NULL) {
                    596:                char *portstr = strchr(name, ':');
                    597:                unsigned port=0;
                    598:                if (portstr) {
                    599:                    *portstr++ = '\0';
                    600:                    if (*portstr) port = (unsigned) atoi(portstr);
                    601:                }
                    602: 
                    603:                /* Register it for all access methods */
2.4       frystyk   604:                HTNoProxy_add(name, NULL, port);
2.1       frystyk   605:            }
2.7       frystyk   606:            HT_FREE(str);
2.1       frystyk   607:        }
                    608:     }
                    609: }
                    610: 
Webmaster