12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187 |
- /* Parse HyperText Document Address HTParse.c
- * ================================
- */
- #include <HTUtils.h>
- #include <HTParse.h>
- #include <LYUtils.h>
- #include <LYLeaks.h>
- #include <LYStrings.h>
- #include <LYCharUtils.h>
- #ifdef HAVE_ALLOCA_H
- #include <alloca.h>
- #else
- #ifdef __MINGW32__
- #include <malloc.h>
- #endif /* __MINGW32__ */
- #endif
- #define HEX_ESCAPE '%'
- struct struct_parts {
- char *access;
- char *host;
- char *absolute;
- char *relative;
- char *search; /* treated normally as part of path */
- char *anchor;
- };
- #if 0 /* for debugging */
- static void show_parts(const char *name, struct struct_parts *parts, int line)
- {
- if (TRACE) {
- CTRACE((tfp, "struct_parts(%s) %s@%d\n", name, __FILE__, line));
- CTRACE((tfp, " access '%s'\n", NONNULL(parts->access)));
- CTRACE((tfp, " host '%s'\n", NONNULL(parts->host)));
- CTRACE((tfp, " absolute '%s'\n", NONNULL(parts->absolute)));
- CTRACE((tfp, " relative '%s'\n", NONNULL(parts->relative)));
- CTRACE((tfp, " search '%s'\n", NONNULL(parts->search)));
- CTRACE((tfp, " anchor '%s'\n", NONNULL(parts->anchor)));
- }
- }
- #define SHOW_PARTS(name) show_parts(#name, &name, __LINE__)
- #else
- #define SHOW_PARTS(name) /* nothing */
- #endif
- /* Strip white space off a string. HTStrip()
- * -------------------------------
- *
- * On exit,
- * Return value points to first non-white character, or to 0 if none.
- * All trailing white space is OVERWRITTEN with zero.
- */
- char *HTStrip(char *s)
- {
- #define SPACE(c) ((c == ' ') || (c == '\t') || (c == '\n'))
- char *p = s;
- for (p = s; *p; p++) ; /* Find end of string */
- for (p--; p >= s; p--) {
- if (SPACE(*p))
- *p = '\0'; /* Zap trailing blanks */
- else
- break;
- }
- while (SPACE(*s))
- s++; /* Strip leading blanks */
- return s;
- }
- /* Scan a filename for its constituents. scan()
- * -------------------------------------
- *
- * On entry,
- * name points to a document name which may be incomplete.
- * On exit,
- * absolute or relative may be nonzero (but not both).
- * host, anchor and access may be nonzero if they were specified.
- * Any which are nonzero point to zero terminated strings.
- */
- static void scan(char *name,
- struct struct_parts *parts)
- {
- char *after_access;
- char *p;
- parts->access = NULL;
- parts->host = NULL;
- parts->absolute = NULL;
- parts->relative = NULL;
- parts->search = NULL; /* normally not used - kw */
- parts->anchor = NULL;
- /*
- * Scan left-to-right for a scheme (access).
- */
- after_access = name;
- for (p = name; *p; p++) {
- if (*p == ':') {
- *p = '\0';
- parts->access = name; /* Access name has been specified */
- after_access = (p + 1);
- break;
- }
- if (*p == '/' || *p == '#' || *p == ';' || *p == '?')
- break;
- }
- /*
- * Scan left-to-right for a fragment (anchor).
- */
- for (p = after_access; *p; p++) {
- if (*p == '#') {
- parts->anchor = (p + 1);
- *p = '\0'; /* terminate the rest */
- break; /* leave things after first # alone - kw */
- }
- }
- /*
- * Scan left-to-right for a host or absolute path.
- */
- p = after_access;
- if (*p == '/') {
- if (p[1] == '/') {
- parts->host = (p + 2); /* host has been specified */
- *p = '\0'; /* Terminate access */
- p = strchr(parts->host, '/'); /* look for end of host name if any */
- if (p != NULL) {
- *p = '\0'; /* Terminate host */
- parts->absolute = (p + 1); /* Root has been found */
- } else {
- p = strchr(parts->host, '?');
- if (p != NULL) {
- *p = '\0'; /* Terminate host */
- parts->search = (p + 1);
- }
- }
- } else {
- parts->absolute = (p + 1); /* Root found but no host */
- }
- } else {
- parts->relative = (*after_access) ?
- after_access : NULL; /* NULL for "" */
- }
- /*
- * Check schemes that commonly have unescaped hashes.
- */
- if (parts->access && parts->anchor &&
- /* optimize */ strchr("lnsdLNSD", *parts->access) != NULL) {
- if ((!parts->host && strcasecomp(parts->access, "lynxcgi")) ||
- !strcasecomp(parts->access, "nntp") ||
- !strcasecomp(parts->access, "snews") ||
- !strcasecomp(parts->access, "news") ||
- !strcasecomp(parts->access, "data")) {
- /*
- * Access specified but no host and not a lynxcgi URL, so the
- * anchor may not really be one, e.g., news:j462#36487@foo.bar, or
- * it's an nntp or snews URL, or news URL with a host. Restore the
- * '#' in the address.
- */
- /* but only if we have found a path component of which this will
- * become part. - kw */
- if (parts->relative || parts->absolute) {
- *(parts->anchor - 1) = '#';
- parts->anchor = NULL;
- }
- }
- }
- } /*scan */
- #if defined(HAVE_ALLOCA) && !defined(LY_FIND_LEAKS)
- #define LYalloca(x) alloca(x)
- #define LYalloca_free(x) {}
- #else
- #define LYalloca(x) malloc(x)
- #define LYalloca_free(x) free(x)
- #endif
- static char *strchr_or_end(char *string, int ch)
- {
- char *result = strchr(string, ch);
- if (result == 0) {
- result = string + strlen(string);
- }
- return result;
- }
- /* Parse a Name relative to another name. HTParse()
- * --------------------------------------
- *
- * This returns those parts of a name which are given (and requested)
- * substituting bits from the related name where necessary.
- *
- * On entry,
- * aName A filename given
- * relatedName A name relative to which aName is to be parsed
- * wanted A mask for the bits which are wanted.
- *
- * On exit,
- * returns A pointer to a malloc'd string which MUST BE FREED
- */
- char *HTParse(const char *aName,
- const char *relatedName,
- int wanted)
- {
- char *result = NULL;
- char *tail = NULL; /* a pointer to the end of the 'result' string */
- char *return_value = NULL;
- int len, len1, len2;
- char *name = NULL;
- char *rel = NULL;
- char *p, *q;
- char *acc_method;
- struct struct_parts given, related;
- CTRACE((tfp, "HTParse: aName:`%s'\n", aName));
- CTRACE((tfp, " relatedName:`%s'\n", relatedName));
- if (wanted & (PARSE_STRICTPATH | PARSE_QUERY)) { /* if detail wanted... */
- if ((wanted & (PARSE_STRICTPATH | PARSE_QUERY))
- == (PARSE_STRICTPATH | PARSE_QUERY)) /* if strictpath AND query */
- wanted |= PARSE_PATH; /* then treat as if PARSE_PATH wanted */
- if (wanted & PARSE_PATH) /* if PARSE_PATH wanted */
- wanted &= ~(PARSE_STRICTPATH | PARSE_QUERY); /* ignore details */
- }
- /* *INDENT-OFF* */
- CTRACE((tfp, " want:%s%s%s%s%s%s%s\n",
- wanted & PARSE_PUNCTUATION ? " punc" : "",
- wanted & PARSE_ANCHOR ? " anchor" : "",
- wanted & PARSE_PATH ? " path" : "",
- wanted & PARSE_HOST ? " host" : "",
- wanted & PARSE_ACCESS ? " access" : "",
- wanted & PARSE_STRICTPATH ? " PATH" : "",
- wanted & PARSE_QUERY ? " QUERY" : ""));
- /* *INDENT-ON* */
- /*
- * Allocate the temporary string. Optimized.
- */
- len1 = strlen(aName) + 1;
- len2 = strlen(relatedName) + 1;
- len = len1 + len2 + 8; /* Lots of space: more than enough */
- result = tail = (char *) LYalloca(len * 2 + len1 + len2);
- if (result == NULL) {
- outofmem(__FILE__, "HTParse");
- }
- *result = '\0';
- name = result + len;
- rel = name + len1;
- /*
- * Make working copy of the input string to cut up.
- */
- memcpy(name, aName, len1);
- /*
- * Cut up the string into URL fields.
- */
- scan(name, &given);
- SHOW_PARTS(given);
- /*
- * Now related string.
- */
- if ((given.access && given.host && given.absolute) || !*relatedName) {
- /*
- * Inherit nothing!
- */
- related.access = NULL;
- related.host = NULL;
- related.absolute = NULL;
- related.relative = NULL;
- related.search = NULL;
- related.anchor = NULL;
- } else {
- memcpy(rel, relatedName, len2);
- scan(rel, &related);
- }
- SHOW_PARTS(related);
- /*
- * Handle the scheme (access) field.
- */
- if (given.access && given.host && !given.relative && !given.absolute) {
- if (!strcmp(given.access, "http") ||
- !strcmp(given.access, "https") ||
- !strcmp(given.access, "ftp"))
- /*
- * Assume root.
- */
- given.absolute = "";
- }
- acc_method = given.access ? given.access : related.access;
- if (wanted & PARSE_ACCESS) {
- if (acc_method) {
- strcpy(tail, acc_method);
- tail += strlen(tail);
- if (wanted & PARSE_PUNCTUATION) {
- *tail++ = ':';
- *tail = '\0';
- }
- }
- }
- /*
- * If different schemes, inherit nothing.
- *
- * We'll try complying with RFC 1808 and the Fielding draft, and inherit
- * nothing if both schemes are given, rather than only when they differ,
- * except for file URLs - FM
- *
- * After trying it for a while, it's still premature, IHMO, to go along
- * with it, so this is back to inheriting for identical schemes whether or
- * not they are "file". If you want to try it again yourself, uncomment
- * the strcasecomp() below. - FM
- */
- if ((given.access && related.access) &&
- ( /* strcasecomp(given.access, "file") || */
- strcmp(given.access, related.access))) {
- related.host = NULL;
- related.absolute = NULL;
- related.relative = NULL;
- related.search = NULL;
- related.anchor = NULL;
- }
- /*
- * Handle the host field.
- */
- if (wanted & PARSE_HOST) {
- if (given.host || related.host) {
- if (wanted & PARSE_PUNCTUATION) {
- *tail++ = '/';
- *tail++ = '/';
- }
- strcpy(tail, given.host ? given.host : related.host);
- #define CLEAN_URLS
- #ifdef CLEAN_URLS
- /*
- * Ignore default port numbers, and trailing dots on FQDNs, which
- * will only cause identical addresses to look different. (related
- * is already a clean url).
- */
- {
- char *p2, *h;
- if ((p2 = strchr(result, '@')) != NULL)
- tail = (p2 + 1);
- p2 = strchr(tail, ':');
- if (p2 != NULL && !isdigit(UCH(p2[1])))
- /*
- * Colon not followed by a port number.
- */
- *p2 = '\0';
- if (p2 != NULL && *p2 != '\0' && acc_method != NULL) {
- /*
- * Port specified.
- */
- #define ACC_METHOD(a,b) (!strcmp(acc_method, a) && !strcmp(p2, b))
- if (ACC_METHOD("http", ":80") ||
- ACC_METHOD("https", ":443") ||
- ACC_METHOD("gopher", ":70") ||
- ACC_METHOD("ftp", ":21") ||
- ACC_METHOD("wais", ":210") ||
- ACC_METHOD("nntp", ":119") ||
- ACC_METHOD("news", ":119") ||
- ACC_METHOD("newspost", ":119") ||
- ACC_METHOD("newsreply", ":119") ||
- ACC_METHOD("snews", ":563") ||
- ACC_METHOD("snewspost", ":563") ||
- ACC_METHOD("snewsreply", ":563") ||
- ACC_METHOD("finger", ":79") ||
- ACC_METHOD("telnet", ":23") ||
- ACC_METHOD("tn3270", ":23") ||
- ACC_METHOD("rlogin", ":513") ||
- ACC_METHOD("cso", ":105"))
- *p2 = '\0'; /* It is the default: ignore it */
- }
- if (p2 == NULL) {
- int len3 = strlen(tail);
- if (len3 > 0) {
- h = tail + len3 - 1; /* last char of hostname */
- if (*h == '.')
- *h = '\0'; /* chop final . */
- }
- } else if (p2 != result) {
- h = p2;
- h--; /* End of hostname */
- if (*h == '.') {
- /*
- * Slide p2 over h.
- */
- while (*p2 != '\0')
- *h++ = *p2++;
- *h = '\0'; /* terminate */
- }
- }
- }
- #endif /* CLEAN_URLS */
- }
- }
- /*
- * Trim any blanks from the result so far - there's no excuse for blanks
- * in a hostname. Also update the tail here.
- */
- tail = LYRemoveBlanks(result);
- /*
- * If host in given or related was ended directly with a '?' (no slash),
- * fake the search part into absolute. This is the only case search is
- * returned from scan. A host must have been present. this restores the
- * '?' at which the host part had been truncated in scan, we have to do
- * this after host part handling is done. - kw
- */
- if (given.search && *(given.search - 1) == '\0') {
- given.absolute = given.search - 1;
- given.absolute[0] = '?';
- } else if (related.search && !related.absolute &&
- *(related.search - 1) == '\0') {
- related.absolute = related.search - 1;
- related.absolute[0] = '?';
- }
- /*
- * If different hosts, inherit no path.
- */
- if (given.host && related.host)
- if (strcmp(given.host, related.host) != 0) {
- related.absolute = NULL;
- related.relative = NULL;
- related.anchor = NULL;
- }
- /*
- * Handle the path.
- */
- if (wanted & (PARSE_PATH | PARSE_STRICTPATH | PARSE_QUERY)) {
- int want_detail = (wanted & (PARSE_STRICTPATH | PARSE_QUERY));
- if (acc_method && !given.absolute && given.relative) {
- /*
- * Treat all given nntp or snews paths, or given paths for news
- * URLs with a host, as absolute.
- */
- switch (*acc_method) {
- case 'N':
- case 'n':
- if (!strcasecomp(acc_method, "nntp") ||
- (!strcasecomp(acc_method, "news") &&
- !strncasecomp(result, "news://", 7))) {
- given.absolute = given.relative;
- given.relative = NULL;
- }
- break;
- case 'S':
- case 's':
- if (!strcasecomp(acc_method, "snews")) {
- given.absolute = given.relative;
- given.relative = NULL;
- }
- break;
- }
- }
- if (given.absolute) { /* All is given */
- if (wanted & PARSE_PUNCTUATION)
- *tail++ = '/';
- strcpy(tail, given.absolute);
- CTRACE((tfp, "HTParse: (ABS)\n"));
- } else if (related.absolute) { /* Adopt path not name */
- char *base = tail;
- *tail++ = '/';
- strcpy(tail, related.absolute);
- if (given.relative) {
- /* RFC 1808 part 4 step 5 (if URL path is empty) */
- /* a) if given has params, add/replace that */
- if (given.relative[0] == ';') {
- strcpy(strchr_or_end(tail, ';'), given.relative);
- }
- /* b) if given has query, add/replace that */
- else if (given.relative[0] == '?') {
- strcpy(strchr_or_end(tail, '?'), given.relative);
- }
- /* otherwise fall through to RFC 1808 part 4 step 6 */
- else {
- p = strchr(tail, '?'); /* Search part? */
- if (p == NULL)
- p = (tail + strlen(tail) - 1);
- for (; *p != '/'; p--) ; /* last / */
- p[1] = '\0'; /* Remove filename */
- strcat(p, given.relative); /* Add given one */
- }
- HTSimplify(base);
- if (*base == '\0')
- strcpy(base, "/");
- }
- CTRACE((tfp, "HTParse: (Related-ABS)\n"));
- } else if (given.relative) {
- strcpy(tail, given.relative); /* what we've got */
- CTRACE((tfp, "HTParse: (REL)\n"));
- } else if (related.relative) {
- strcpy(tail, related.relative);
- CTRACE((tfp, "HTParse: (Related-REL)\n"));
- } else { /* No inheritance */
- if (!isLYNXCGI(aName) &&
- !isLYNXEXEC(aName) &&
- !isLYNXPROG(aName)) {
- *tail++ = '/';
- *tail = '\0';
- }
- if (!strcmp(result, "news:/"))
- result[5] = '*';
- CTRACE((tfp, "HTParse: (No inheritance)\n"));
- }
- if (want_detail) {
- p = strchr(tail, '?'); /* Search part? */
- if (p) {
- if (PARSE_STRICTPATH) {
- *p = '\0';
- } else {
- if (!(wanted & PARSE_PUNCTUATION))
- p++;
- do {
- *tail++ = *p;
- } while (*p++);
- }
- } else {
- if (wanted & PARSE_QUERY)
- *tail = '\0';
- }
- }
- }
- /*
- * Handle the fragment (anchor). Never inherit.
- */
- if (wanted & PARSE_ANCHOR) {
- if (given.anchor && *given.anchor) {
- tail += strlen(tail);
- if (wanted & PARSE_PUNCTUATION)
- *tail++ = '#';
- strcpy(tail, given.anchor);
- }
- }
- /*
- * If there are any blanks remaining in the string, escape them as needed.
- * See the discussion in LYLegitimizeHREF() for example.
- */
- if ((p = strchr(result, ' ')) != 0) {
- switch (is_url(result)) {
- case UNKNOWN_URL_TYPE:
- CTRACE((tfp, "HTParse: ignore:`%s'\n", result));
- break;
- case LYNXEXEC_URL_TYPE:
- case LYNXPROG_URL_TYPE:
- case LYNXCGI_URL_TYPE:
- case LYNXPRINT_URL_TYPE:
- case LYNXHIST_URL_TYPE:
- case LYNXDOWNLOAD_URL_TYPE:
- case LYNXKEYMAP_URL_TYPE:
- case LYNXIMGMAP_URL_TYPE:
- case LYNXCOOKIE_URL_TYPE:
- case LYNXDIRED_URL_TYPE:
- case LYNXOPTIONS_URL_TYPE:
- case LYNXCFG_URL_TYPE:
- case LYNXCOMPILE_OPTS_URL_TYPE:
- case LYNXMESSAGES_URL_TYPE:
- CTRACE((tfp, "HTParse: spaces:`%s'\n", result));
- break;
- case NOT_A_URL_TYPE:
- default:
- CTRACE((tfp, "HTParse: encode:`%s'\n", result));
- do {
- q = p + strlen(p) + 2;
- while (q != p + 1) {
- q[0] = q[-2];
- --q;
- }
- p[0] = '%';
- p[1] = '2';
- p[2] = '0';
- } while ((p = strchr(result, ' ')) != 0);
- break;
- }
- }
- CTRACE((tfp, "HTParse: result:`%s'\n", result));
- StrAllocCopy(return_value, result);
- LYalloca_free(result);
- /* FIXME: could be optimized using HTParse() internals */
- if (*relatedName &&
- ((wanted & PARSE_ALL_WITHOUT_ANCHOR) == PARSE_ALL_WITHOUT_ANCHOR)) {
- /*
- * Check whether to fill in localhost. - FM
- */
- LYFillLocalFileURL(&return_value, relatedName);
- CTRACE((tfp, "pass LYFillLocalFile:`%s'\n", return_value));
- }
- return return_value; /* exactly the right length */
- }
- /* HTParseAnchor(), fast HTParse() specialization
- * ----------------------------------------------
- *
- * On exit,
- * returns A pointer within input string (probably to its end '\0')
- */
- const char *HTParseAnchor(const char *aName)
- {
- const char *p = aName;
- for (; *p && *p != '#'; p++) ;
- if (*p == '#') {
- /* the safe way based on HTParse() -
- * keeping in mind scan() peculiarities on schemes:
- */
- struct struct_parts given;
- char *name = (char *) LYalloca((p - aName) + strlen(p) + 1);
- if (name == NULL) {
- outofmem(__FILE__, "HTParseAnchor");
- }
- strcpy(name, aName);
- scan(name, &given);
- LYalloca_free(name);
- p++; /*next to '#' */
- if (given.anchor == NULL) {
- for (; *p; p++) /*scroll to end '\0' */
- ;
- }
- }
- return p;
- }
- /* Simplify a filename. HTSimplify()
- * --------------------
- *
- * A unix-style file is allowed to contain the sequence xxx/../ which may
- * be replaced by "" , and the sequence "/./" which may be replaced by "/".
- * Simplification helps us recognize duplicate filenames.
- *
- * Thus, /etc/junk/../fred becomes /etc/fred
- * /etc/junk/./fred becomes /etc/junk/fred
- *
- * but we should NOT change
- * http://fred.xxx.edu/../..
- *
- * or ../../albert.html
- */
- void HTSimplify(char *filename)
- {
- char *p;
- char *q, *q1;
- if (filename == NULL)
- return;
- if (!(filename[0] && filename[1]) ||
- filename[0] == '?' || filename[1] == '?' || filename[2] == '?')
- return;
- if (strchr(filename, '/') != NULL) {
- for (p = (filename + 2); *p; p++) {
- if (*p == '?') {
- /*
- * We're still treating a ?searchpart as part of the path in
- * HTParse() and scan(), but if we encounter a '?' here, assume
- * it's the delimiter and break. We also could check for a
- * parameter delimiter (';') here, but the current Fielding
- * draft (wisely or ill-advisedly :) says that it should be
- * ignored and collapsing be allowed in it's value). The only
- * defined parameter at present is ;type=[A, I, or D] for ftp
- * URLs, so if there's a "/..", "/../", "/./", or terminal '.'
- * following the ';', it must be due to the ';' being an
- * unescaped path character and not actually a parameter
- * delimiter. - FM
- */
- break;
- }
- if (*p == '/') {
- if ((p[1] == '.') && (p[2] == '.') &&
- (p[3] == '/' || p[3] == '?' || p[3] == '\0')) {
- /*
- * Handle "../", "..?" or "..".
- */
- for (q = (p - 1); (q >= filename) && (*q != '/'); q--)
- /*
- * Back up to previous slash or beginning of string.
- */
- ;
- if ((q[0] == '/') &&
- (strncmp(q, "/../", 4) &&
- strncmp(q, "/..?", 4)) &&
- !((q - 1) > filename && q[-1] == '/')) {
- /*
- * Not at beginning of string or in a host field, so
- * remove the "/xxx/..".
- */
- q1 = (p + 3);
- p = q;
- while (*q1 != '\0')
- *p++ = *q1++;
- *p = '\0'; /* terminate */
- /*
- * Start again with previous slash.
- */
- p = (q - 1);
- }
- } else if (p[1] == '.' && p[2] == '/') {
- /*
- * Handle "./" by removing both characters.
- */
- q = p;
- q1 = (p + 2);
- while (*q1 != '\0')
- *q++ = *q1++;
- *q = '\0'; /* terminate */
- p--;
- } else if (p[1] == '.' && p[2] == '?') {
- /*
- * Handle ".?" by removing the dot.
- */
- q = (p + 1);
- q1 = (p + 2);
- while (*q1 != '\0')
- *q++ = *q1++;
- *q = '\0'; /* terminate */
- p--;
- } else if (p[1] == '.' && p[2] == '\0') {
- /*
- * Handle terminal "." by removing the character.
- */
- p[1] = '\0';
- }
- }
- }
- if (p >= filename + 2 && *p == '?' && *(p - 1) == '.') {
- if (*(p - 2) == '/') {
- /*
- * Handle "/.?" by removing the dot.
- */
- q = p - 1;
- q1 = p;
- while (*q1 != '\0')
- *q++ = *q1++;
- *q = '\0';
- } else if (*(p - 2) == '.' &&
- p >= filename + 4 && *(p - 3) == '/' &&
- (*(p - 4) != '/' ||
- (p > filename + 4 && *(p - 5) != ':'))) {
- /*
- * Handle "xxx/..?"
- */
- for (q = (p - 4); (q > filename) && (*q != '/'); q--)
- /*
- * Back up to previous slash or beginning of string.
- */
- ;
- if (*q == '/') {
- if (q > filename && *(q - 1) == '/' &&
- !(q > filename + 1 && *(q - 1) != ':'))
- return;
- q++;
- }
- if (strncmp(q, "../", 3) && strncmp(q, "./", 2)) {
- /*
- * Not after "//" at beginning of string or after "://",
- * and xxx is not ".." or ".", so remove the "xxx/..".
- */
- q1 = p;
- p = q;
- while (*q1 != '\0')
- *p++ = *q1++;
- *p = '\0'; /* terminate */
- }
- }
- }
- }
- }
- /* Make Relative Name. HTRelative()
- * -------------------
- *
- * This function creates and returns a string which gives an expression of
- * one address as related to another. Where there is no relation, an absolute
- * address is returned.
- *
- * On entry,
- * Both names must be absolute, fully qualified names of nodes
- * (no anchor bits)
- *
- * On exit,
- * The return result points to a newly allocated name which, if
- * parsed by HTParse relative to relatedName, will yield aName.
- * The caller is responsible for freeing the resulting name later.
- *
- */
- char *HTRelative(const char *aName,
- const char *relatedName)
- {
- char *result = NULL;
- const char *p = aName;
- const char *q = relatedName;
- const char *after_access = NULL;
- const char *path = NULL;
- const char *last_slash = NULL;
- int slashes = 0;
- for (; *p; p++, q++) { /* Find extent of match */
- if (*p != *q)
- break;
- if (*p == ':')
- after_access = p + 1;
- if (*p == '/') {
- last_slash = p;
- slashes++;
- if (slashes == 3)
- path = p;
- }
- }
- /* q, p point to the first non-matching character or zero */
- if (!after_access) { /* Different access */
- StrAllocCopy(result, aName);
- } else if (slashes < 3) { /* Different nodes */
- StrAllocCopy(result, after_access);
- } else if (slashes == 3) { /* Same node, different path */
- StrAllocCopy(result, path);
- } else { /* Some path in common */
- int levels = 0;
- for (; *q && (*q != '#'); q++)
- if (*q == '/')
- levels++;
- result = typecallocn(char, 3 * levels + strlen(last_slash) + 1);
- if (result == NULL)
- outofmem(__FILE__, "HTRelative");
- result[0] = '\0';
- for (; levels; levels--)
- strcat(result, "../");
- strcat(result, last_slash + 1);
- }
- CTRACE((tfp,
- "HTparse: `%s' expressed relative to\n `%s' is\n `%s'.\n",
- aName, relatedName, result));
- return result;
- }
- /* Escape undesirable characters using % HTEscape()
- * -------------------------------------
- *
- * This function takes a pointer to a string in which
- * some characters may be unacceptable unescaped.
- * It returns a string which has these characters
- * represented by a '%' character followed by two hex digits.
- *
- * Unlike HTUnEscape(), this routine returns a calloc'd string.
- */
- /* *INDENT-OFF* */
- static const unsigned char isAcceptable[96] =
- /* Bit 0 xalpha -- see HTFile.h
- * Bit 1 xpalpha -- as xalpha but with plus.
- * Bit 2 ... path -- as xpalphas but with /
- */
- /* 0 1 2 3 4 5 6 7 8 9 A B C D E F */
- { 0,0,0,0,0,0,0,0,0,0,7,6,0,7,7,4, /* 2x !"#$%&'()*+,-./ */
- 7,7,7,7,7,7,7,7,7,7,0,0,0,0,0,0, /* 3x 0123456789:;<=>? */
- 7,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7, /* 4x @ABCDEFGHIJKLMNO */
- 7,7,7,7,7,7,7,7,7,7,7,0,0,0,0,7, /* 5X PQRSTUVWXYZ[\]^_ */
- 0,7,7,7,7,7,7,7,7,7,7,7,7,7,7,7, /* 6x `abcdefghijklmno */
- 7,7,7,7,7,7,7,7,7,7,7,0,0,0,0,0 }; /* 7X pqrstuvwxyz{|}~ DEL */
- /* *INDENT-ON* */
- static const char *hex = "0123456789ABCDEF";
- #define ACCEPTABLE(a) ( a>=32 && a<128 && ((isAcceptable[a-32]) & mask))
- char *HTEscape(const char *str,
- unsigned char mask)
- {
- const char *p;
- char *q;
- char *result;
- int unacceptable = 0;
- for (p = str; *p; p++)
- if (!ACCEPTABLE(UCH(TOASCII(*p))))
- unacceptable++;
- result = typecallocn(char, p - str + unacceptable + unacceptable + 1);
- if (result == NULL)
- outofmem(__FILE__, "HTEscape");
- for (q = result, p = str; *p; p++) {
- unsigned char a = TOASCII(*p);
- if (!ACCEPTABLE(a)) {
- *q++ = HEX_ESCAPE; /* Means hex coming */
- *q++ = hex[a >> 4];
- *q++ = hex[a & 15];
- } else
- *q++ = *p;
- }
- *q++ = '\0'; /* Terminate */
- return result;
- }
- /* Escape unsafe characters using % HTEscapeUnsafe()
- * --------------------------------
- *
- * This function takes a pointer to a string in which
- * some characters may be that may be unsafe are unescaped.
- * It returns a string which has these characters
- * represented by a '%' character followed by two hex digits.
- *
- * Unlike HTUnEscape(), this routine returns a malloc'd string.
- */
- #define UNSAFE(ch) (((ch) <= 32) || ((ch) >= 127))
- char *HTEscapeUnsafe(const char *str)
- {
- const char *p;
- char *q;
- char *result;
- int unacceptable = 0;
- for (p = str; *p; p++)
- if (UNSAFE(UCH(TOASCII(*p))))
- unacceptable++;
- result = typecallocn(char, p - str + unacceptable + unacceptable + 1);
- if (result == NULL)
- outofmem(__FILE__, "HTEscapeUnsafe");
- for (q = result, p = str; *p; p++) {
- unsigned char a = TOASCII(*p);
- if (UNSAFE(a)) {
- *q++ = HEX_ESCAPE; /* Means hex coming */
- *q++ = hex[a >> 4];
- *q++ = hex[a & 15];
- } else
- *q++ = *p;
- }
- *q++ = '\0'; /* Terminate */
- return result;
- }
- /* Escape undesirable characters using % but space to +. HTEscapeSP()
- * -----------------------------------------------------
- *
- * This function takes a pointer to a string in which
- * some characters may be unacceptable unescaped.
- * It returns a string which has these characters
- * represented by a '%' character followed by two hex digits,
- * except that spaces are converted to '+' instead of %2B.
- *
- * Unlike HTUnEscape(), this routine returns a calloced string.
- */
- char *HTEscapeSP(const char *str,
- unsigned char mask)
- {
- const char *p;
- char *q;
- char *result;
- int unacceptable = 0;
- for (p = str; *p; p++)
- if (!(*p == ' ' || ACCEPTABLE(UCH(TOASCII(*p)))))
- unacceptable++;
- result = typecallocn(char, p - str + unacceptable + unacceptable + 1);
- if (result == NULL)
- outofmem(__FILE__, "HTEscape");
- for (q = result, p = str; *p; p++) {
- unsigned char a = TOASCII(*p);
- if (a == 32) {
- *q++ = '+';
- } else if (!ACCEPTABLE(a)) {
- *q++ = HEX_ESCAPE; /* Means hex coming */
- *q++ = hex[a >> 4];
- *q++ = hex[a & 15];
- } else {
- *q++ = *p;
- }
- }
- *q++ = '\0'; /* Terminate */
- return result;
- }
- /* Decode %xx escaped characters. HTUnEscape()
- * ------------------------------
- *
- * This function takes a pointer to a string in which some
- * characters may have been encoded in %xy form, where xy is
- * the ASCII hex code for character 16x+y.
- * The string is converted in place, as it will never grow.
- */
- static char from_hex(char c)
- {
- return (char) (c >= '0' && c <= '9' ? c - '0'
- : c >= 'A' && c <= 'F' ? c - 'A' + 10
- : c - 'a' + 10); /* accept small letters just in case */
- }
- char *HTUnEscape(char *str)
- {
- char *p = str;
- char *q = str;
- if (!(p && *p))
- return str;
- while (*p != '\0') {
- if (*p == HEX_ESCAPE &&
- /*
- * Tests shouldn't be needed, but better safe than sorry.
- */
- p[1] && p[2] &&
- isxdigit(UCH(p[1])) &&
- isxdigit(UCH(p[2]))) {
- p++;
- if (*p)
- *q = (char) (from_hex(*p++) * 16);
- if (*p) {
- /*
- * Careful! FROMASCII() may evaluate its arg more than once!
- */
- /* S/390 -- gil -- 0221 */
- *q = (char) (*q + from_hex(*p++));
- }
- *q = FROMASCII(*q);
- q++;
- } else {
- *q++ = *p++;
- }
- }
- *q++ = '\0';
- return str;
- } /* HTUnEscape */
- /* Decode some %xx escaped characters. HTUnEscapeSome()
- * ----------------------------------- Klaus Weide
- * (kweide@tezcat.com)
- * This function takes a pointer to a string in which some
- * characters may have been encoded in %xy form, where xy is
- * the ASCII hex code for character 16x+y, and a pointer to
- * a second string containing one or more characters which
- * should be unescaped if escaped in the first string.
- * The first string is converted in place, as it will never grow.
- */
- char *HTUnEscapeSome(char *str,
- const char *do_trans)
- {
- char *p = str;
- char *q = str;
- char testcode;
- if (p == NULL || *p == '\0' || do_trans == NULL || *do_trans == '\0')
- return str;
- while (*p != '\0') {
- if (*p == HEX_ESCAPE &&
- p[1] && p[2] && /* tests shouldn't be needed, but.. */
- isxdigit(UCH(p[1])) &&
- isxdigit(UCH(p[2])) &&
- (testcode = (char) FROMASCII(from_hex(p[1]) * 16 +
- from_hex(p[2]))) && /* %00 no good */
- strchr(do_trans, testcode)) { /* it's one of the ones we want */
- *q++ = testcode;
- p += 3;
- } else {
- *q++ = *p++;
- }
- }
- *q++ = '\0';
- return str;
- } /* HTUnEscapeSome */
- /* *INDENT-OFF* */
- static const unsigned char crfc[96] =
- /* Bit 0 xalpha -- need "quoting"
- * Bit 1 xpalpha -- need \escape if quoted
- */
- /* 0 1 2 3 4 5 6 7 8 9 A B C D E F */
- { 1,0,3,0,0,0,0,0,1,1,0,0,1,0,1,0, /* 2x !"#$%&'()*+,-./ */
- 0,0,0,0,0,0,0,0,0,0,1,1,1,0,1,0, /* 3x 0123456789:;<=>? */
- 1,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, /* 4x @ABCDEFGHIJKLMNO */
- 0,0,0,0,0,0,0,0,0,0,0,1,2,1,0,0, /* 5X PQRSTUVWXYZ[\]^_ */
- 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, /* 6x `abcdefghijklmno */
- 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,3 }; /* 7X pqrstuvwxyz{|}~ DEL */
- /* *INDENT-ON* */
- #define ASCII_TAB '\011'
- #define ASCII_LF '\012'
- #define ASCII_CR '\015'
- #define ASCII_SPC '\040'
- #define ASCII_BAK '\134'
- /*
- * Turn a string which is not a RFC 822 token into a quoted-string. - KW
- * The "quoted" parameter tells whether we need the beginning/ending quote
- * marks. If not, the caller will provide them -TD
- */
- void HTMake822Word(char **str,
- int quoted)
- {
- const char *p;
- char *q;
- char *result;
- unsigned char a;
- int added = 0;
- if (isEmpty(*str)) {
- StrAllocCopy(*str, quoted ? "\"\"" : "");
- return;
- }
- for (p = *str; *p; p++) {
- a = TOASCII(*p); /* S/390 -- gil -- 0240 */
- if (a < 32 || a >= 128 ||
- ((crfc[a - 32]) & 1)) {
- if (!added)
- added = 2;
- if (a >= 160 || a == '\t')
- continue;
- if (a == '\r' || a == '\n')
- added += 2;
- else if ((a & 127) < 32 || ((crfc[a - 32]) & 2))
- added++;
- }
- }
- if (!added)
- return;
- result = typecallocn(char, p - (*str) + added + 1);
- if (result == NULL)
- outofmem(__FILE__, "HTMake822Word");
- q = result;
- if (quoted)
- *q++ = '"';
- /*
- * Having converted the character to ASCII, we can't use symbolic
- * escape codes, since they're in the host character set, which
- * is not necessarily ASCII. Thus we use octal escape codes instead.
- * -- gil (Paul Gilmartin) <pg@sweng.stortek.com>
- */
- /* S/390 -- gil -- 0268 */
- for (p = *str; *p; p++) {
- a = TOASCII(*p);
- if ((a != ASCII_TAB) &&
- ((a & 127) < ASCII_SPC ||
- (a < 128 && ((crfc[a - 32]) & 2))))
- *q++ = ASCII_BAK;
- *q++ = *p;
- if (a == ASCII_LF ||
- (a == ASCII_CR && (TOASCII(*(p + 1)) != ASCII_LF)))
- *q++ = ' ';
- }
- if (quoted)
- *q++ = '"';
- *q++ = '\0'; /* Terminate */
- FREE(*str);
- *str = result;
- }
|