diff options
Diffstat (limited to 'src/LYShowInfo.c')
-rw-r--r-- | src/LYShowInfo.c | 499 |
1 files changed, 499 insertions, 0 deletions
diff --git a/src/LYShowInfo.c b/src/LYShowInfo.c new file mode 100644 index 0000000..0338716 --- /dev/null +++ b/src/LYShowInfo.c @@ -0,0 +1,499 @@ +/* $LynxId: LYShowInfo.c,v 1.83 2018/12/27 22:35:44 Chuck.Martin Exp $ */ +#include <HTUtils.h> +#include <HTFile.h> +#include <HTParse.h> +#include <HTAlert.h> +#include <HTTP.h> +#include <LYCurses.h> +#include <LYUtils.h> +#include <LYStructs.h> +#include <LYGlobalDefs.h> +#include <LYShowInfo.h> +#include <LYCharUtils.h> +#include <GridText.h> +#include <LYReadCFG.h> +#include <LYCharSets.h> +#include <LYStrings.h> + +#include <LYLeaks.h> + +#ifdef DIRED_SUPPORT +#include <HTAAProt.h> +#include <time.h> +#include <LYLocal.h> +#endif /* DIRED_SUPPORT */ + +#define BEGIN_DL(text) fprintf(fp0, "<h2>%s</h2>\n<dl compact>", LYEntifyTitle(&buffer, text)) +#define END_DL() fprintf(fp0, "\n</dl>\n") + +#define ADD_SS(label,value) dt_String(fp0, label, value, 0) +#define ADD_WW(label,value) dt_String(fp0, label, value, 1) +#define ADD_NN(label,value,units) dt_Number(fp0, label, (long) value, units) + +static int label_columns; + +/* + * LYNX_VERSION and LYNX_DATE are automatically generated by PRCS, the tool + * which we use to archive versions of Lynx. We use a convention for naming + * the successive versions: + * {release}{status}{patch} + * where + * {release} is the release that we are working on, e.g., 2.8.4 + * {status} is one of "dev", "pre" or "rel", and + * {patch} is a number assigned by PRCS. + */ +BOOL LYVersionIsRelease(void) +{ + return (BOOL) (strstr(LYNX_VERSION, "rel") != 0); +} + +const char *LYVersionStatus(void) +{ + if (LYVersionIsRelease()) + return REL_VERSION; + else if (strstr(LYNX_VERSION, "pre") != 0) + return PRE_VERSION; + return DEV_VERSION; +} + +const char *LYVersionDate(void) +{ + static char temp[LYNX_DATE_LEN + 1]; + + LYStrNCpy(temp, &LYNX_DATE[LYNX_DATE_OFF], LYNX_DATE_LEN); + return temp; +} + +static void dt_String(FILE *fp, + const char *label, + const char *value, + int allow_wide) +{ + int have; + int need; + char *the_label = NULL; + char *the_value = NULL; + + StrAllocCopy(the_label, label); + StrAllocCopy(the_value, value); + + have = (int) strlen(the_label); + need = LYstrExtent(the_label, have, label_columns); + + LYEntify(&the_label, TRUE); + LYEntify(&the_value, TRUE); + + fprintf(fp, "<dt>"); + while (need++ < label_columns) + fprintf(fp, " "); + if (LYwideLines && allow_wide) + fprintf(fp, "<em>%s</em> <pre>%s</pre>\n", the_label, the_value); + else + fprintf(fp, "<em>%s</em> %s\n", the_label, the_value); + + FREE(the_label); + FREE(the_value); +} + +static void dt_Number(FILE *fp0, + const char *label, + long number, + const char *units) +{ + char *value = NULL; + char *buffer = NULL; + + HTSprintf(&value, "%ld %s", number, LYEntifyTitle(&buffer, units)); + ADD_SS(label, value); + FREE(value); + FREE(buffer); +} + +static void dt_URL(FILE *fp0, const char *address) +{ + if (address == NULL) + address = ""; + ADD_WW(gettext("URL:"), address); + + /* + * If the display handles UTF-8, and if the address uses %xy formatted + * characters, show the decoded URL on the next line. + */ + if (LYCharSet_UC[current_char_set].enc == UCT_ENC_UTF8) { + char *working = NULL; + + StrAllocCopy(working, address); + if (strcmp(HTUnEscape(working), address)) { + fprintf(fp0, "\n<br>%s\n", working); + } + free(working); + } +} + +/* + * LYShowInfo prints a page of info about the current file and the link that + * the cursor is on. + */ +int LYShowInfo(DocInfo *doc, + DocInfo *newdoc, + char *owner_address) +{ + static char tempfile[LY_MAXPATH] = "\0"; + + int url_type; + FILE *fp0; + char *Title = NULL; + const char *cp; + char *temp = NULL; + char *buffer = NULL; + + BOOLEAN LYInfoAdvanced = (BOOL) (user_mode == ADVANCED_MODE); + +#ifdef DIRED_SUPPORT + struct stat dir_info; + const char *name; +#endif /* DIRED_SUPPORT */ + + if (LYReuseTempfiles) { + fp0 = LYOpenTempRewrite(tempfile, HTML_SUFFIX, "w"); + } else { + (void) LYRemoveTemp(tempfile); + fp0 = LYOpenTemp(tempfile, HTML_SUFFIX, "w"); + } + if (fp0 == NULL) { + HTAlert(CANNOT_OPEN_TEMP); + return (-1); + } + + /* + * Point the address pointer at this Url + */ + LYLocalFileToURL(&newdoc->address, tempfile); + + if (nlinks > 0 && links[doc->link].lname != NULL && + (url_type = is_url(links[doc->link].lname)) != 0 && + (url_type == LYNXEXEC_URL_TYPE || + url_type == LYNXPROG_URL_TYPE)) { + char *last_slash = strrchr(links[doc->link].lname, '/'); + int next_to_last = (int) strlen(links[doc->link].lname) - 1; + + if ((last_slash - links[doc->link].lname) == next_to_last) { + links[doc->link].lname[next_to_last] = '\0'; + } + } + + label_columns = 9; + + WriteInternalTitle(fp0, SHOWINFO_TITLE); + + fprintf(fp0, "<h1>%s %s (%s) (<a href=\"%s\">%s</a>)", + LYNX_NAME, LYNX_VERSION, + LYVersionDate(), + (LYVersionIsRelease()? LYNX_WWW_HOME : LYNX_WWW_DIST), + LYVersionStatus()); + + fprintf(fp0, "</h1>\n"); /* don't forget to close <h1> */ + +#ifdef DIRED_SUPPORT + if (lynx_edit_mode && nlinks > 0) { + + BEGIN_DL(gettext("Directory that you are currently viewing")); + + temp = HTfullURL_toFile(doc->address); + ADD_SS(gettext("Name:"), temp); + FREE(temp); + + dt_URL(fp0, doc->address); + + END_DL(); + + temp = HTfullURL_toFile(links[doc->link].lname); + + if (lstat(temp, &dir_info) == -1) { + CTRACE((tfp, "lstat(%s) failed, errno=%d\n", temp, errno)); + HTAlert(CURRENT_LINK_STATUS_FAILED); + } else { + char modes[80]; + + label_columns = 16; + if (S_ISDIR(dir_info.st_mode)) { + BEGIN_DL(gettext("Directory that you have currently selected")); + } else if (S_ISREG(dir_info.st_mode)) { + BEGIN_DL(gettext("File that you have currently selected")); +#ifdef S_IFLNK + } else if (S_ISLNK(dir_info.st_mode)) { + BEGIN_DL(gettext("Symbolic link that you have currently selected")); +#endif + } else { + BEGIN_DL(gettext("Item that you have currently selected")); + } + ADD_SS(gettext("Full name:"), temp); +#ifdef S_IFLNK + if (S_ISLNK(dir_info.st_mode)) { + char buf[MAX_LINE]; + int buf_size; + size_t limit = sizeof(buf) - 1; + + if ((buf_size = (int) readlink(temp, buf, limit)) != -1) { + if (buf_size > (int) limit) + buf_size = (int) limit; + buf[buf_size] = '\0'; + } else { + sprintf(buf, "%.*s", (int) limit, + gettext("Unable to follow link")); + } + ADD_SS(gettext("Points to file:"), buf); + } +#endif + name = HTAA_UidToName((int) dir_info.st_uid); + if (*name) + ADD_SS(gettext("Name of owner:"), name); + name = HTAA_GidToName((int) dir_info.st_gid); + if (*name) + ADD_SS(gettext("Group name:"), name); + if (S_ISREG(dir_info.st_mode)) { + ADD_NN(gettext("File size:"), + (long) dir_info.st_size, + gettext("(bytes)")); + } + /* + * Include date and time information. + */ + ADD_SS(gettext("Creation date:"), + ctime(&dir_info.st_ctime)); + + ADD_SS(gettext("Last modified:"), + ctime(&dir_info.st_mtime)); + + ADD_SS(gettext("Last accessed:"), + ctime(&dir_info.st_atime)); + + END_DL(); + + label_columns = 9; + BEGIN_DL(gettext("Access Permissions")); + modes[0] = '\0'; + modes[1] = '\0'; /* In case there are no permissions */ + modes[2] = '\0'; + if ((dir_info.st_mode & S_IRUSR)) + strcat(modes, ", read"); + if ((dir_info.st_mode & S_IWUSR)) + strcat(modes, ", write"); + if ((dir_info.st_mode & S_IXUSR)) { + if (S_ISDIR(dir_info.st_mode)) + strcat(modes, ", search"); + else { + strcat(modes, ", execute"); + if ((dir_info.st_mode & S_ISUID)) + strcat(modes, ", setuid"); + } + } + ADD_SS(gettext("Owner:"), &modes[2]); + + modes[0] = '\0'; + modes[1] = '\0'; /* In case there are no permissions */ + modes[2] = '\0'; + if ((dir_info.st_mode & S_IRGRP)) + strcat(modes, ", read"); + if ((dir_info.st_mode & S_IWGRP)) + strcat(modes, ", write"); + if ((dir_info.st_mode & S_IXGRP)) { + if (S_ISDIR(dir_info.st_mode)) + strcat(modes, ", search"); + else { + strcat(modes, ", execute"); + if ((dir_info.st_mode & S_ISGID)) + strcat(modes, ", setgid"); + } + } + ADD_SS(gettext("Group:"), &modes[2]); + + modes[0] = '\0'; + modes[1] = '\0'; /* In case there are no permissions */ + modes[2] = '\0'; + if ((dir_info.st_mode & S_IROTH)) + strcat(modes, ", read"); + if ((dir_info.st_mode & S_IWOTH)) + strcat(modes, ", write"); + if ((dir_info.st_mode & S_IXOTH)) { + if (S_ISDIR(dir_info.st_mode)) + strcat(modes, ", search"); + else { + strcat(modes, ", execute"); +#ifdef S_ISVTX + if ((dir_info.st_mode & S_ISVTX)) + strcat(modes, ", sticky"); +#endif + } + } + ADD_SS(gettext("World:"), &modes[2]); + END_DL(); + } + FREE(temp); + } else { +#endif /* DIRED_SUPPORT */ + + BEGIN_DL(gettext("File that you are currently viewing")); + + LYformTitle(&Title, doc->title); + HTSprintf(&temp, "%s%s", + LYEntifyTitle(&buffer, Title), + ((doc->isHEAD && + !strstr(Title, " (HEAD)") && + !strstr(Title, " - HEAD")) ? " (HEAD)" : "")); + ADD_SS(gettext("Linkname:"), temp); + FREE(temp); + + dt_URL(fp0, doc->address); + + if (HTLoadedDocumentCharset()) { + ADD_SS(gettext("Charset:"), + HTLoadedDocumentCharset()); + } else { + LYUCcharset *p_in = HTAnchor_getUCInfoStage(HTMainAnchor, + UCT_STAGE_PARSER); + + if (!p_in || isEmpty(p_in->MIMEname) || + HTAnchor_getUCLYhndl(HTMainAnchor, UCT_STAGE_PARSER) < 0) { + p_in = HTAnchor_getUCInfoStage(HTMainAnchor, UCT_STAGE_MIME); + } + if (p_in && non_empty(p_in->MIMEname) && + HTAnchor_getUCLYhndl(HTMainAnchor, UCT_STAGE_MIME) >= 0) { + HTSprintf(&temp, "%s %s", + LYEntifyTitle(&buffer, p_in->MIMEname), + gettext("(assumed)")); + ADD_SS(gettext("Charset:"), p_in->MIMEname); + FREE(temp); + } + } + + if ((cp = HText_getServer()) != NULL && *cp != '\0') + ADD_SS(gettext("Server:"), cp); + + if ((cp = HText_getDate()) != NULL && *cp != '\0') + ADD_SS(gettext("Date:"), cp); + + if ((cp = HText_getLastModified()) != NULL && *cp != '\0') + ADD_SS(gettext("Last Mod:"), cp); + + if (LYInfoAdvanced) { + if (HTMainAnchor && HTMainAnchor->expires) { + ADD_SS(gettext("Expires:"), HTMainAnchor->expires); + } + if (HTMainAnchor && HTMainAnchor->cache_control) { + ADD_SS(gettext("Cache-Control:"), HTMainAnchor->cache_control); + } + if (HTMainAnchor && HTMainAnchor->content_length > 0) { + ADD_NN(gettext("Content-Length:"), + HTMainAnchor->content_length, + gettext("bytes")); + } else { + ADD_NN(gettext("Length:"), + HText_getNumOfBytes(), + gettext("bytes")); + } + if (HTMainAnchor && HTMainAnchor->content_language) { + ADD_SS(gettext("Language:"), HTMainAnchor->content_language); + } + } + + if (doc->post_data) { + fprintf(fp0, "<dt><em>%s</em> <xmp>%.*s</xmp>\n", + LYEntifyTitle(&buffer, gettext("Post Data:")), + BStrLen(doc->post_data), + BStrData(doc->post_data)); + ADD_SS(gettext("Post Content Type:"), doc->post_content_type); + } + + ADD_SS(gettext("Owner(s):"), + (owner_address + ? owner_address + : NO_NOTHING)); + + ADD_NN(gettext("size:"), + HText_getNumOfLines(), + gettext("lines")); + + StrAllocCopy(temp, + ((lynx_mode == FORMS_LYNX_MODE) + ? gettext("forms mode") + : (HTisDocumentSource() + ? gettext("source") + : gettext("normal")))); + if (doc->safe) + StrAllocCat(temp, gettext(", safe")); + if (doc->internal_link) + StrAllocCat(temp, gettext(", via internal link")); + + if (LYInfoAdvanced) { + if (HText_hasNoCacheSet(HTMainText)) + StrAllocCat(temp, gettext(", no-cache")); + if (HTAnchor_isISMAPScript((HTAnchor *) HTMainAnchor)) + StrAllocCat(temp, gettext(", ISMAP script")); + if (doc->bookmark) + StrAllocCat(temp, gettext(", bookmark file")); + } + + ADD_SS(gettext("mode:"), temp); + FREE(temp); + + END_DL(); + + if (nlinks > 0) { + BEGIN_DL(gettext("Link that you currently have selected")); + ADD_SS(gettext("Linkname:"), + LYGetHiliteStr(doc->link, 0)); + if (lynx_mode == FORMS_LYNX_MODE && + links[doc->link].type == WWW_FORM_LINK_TYPE) { + if (links[doc->link].l_form->submit_method) { + int method = links[doc->link].l_form->submit_method; + char *enctype = links[doc->link].l_form->submit_enctype; + + ADD_SS(gettext("Method:"), + ((method == URL_POST_METHOD) ? "POST" : + ((method == URL_MAIL_METHOD) ? "(email)" : + "GET"))); + ADD_SS(gettext("Enctype:"), + (non_empty(enctype) + ? enctype + : "application/x-www-form-urlencoded")); + } + if (links[doc->link].l_form->submit_action) { + ADD_SS(gettext("Action:"), + links[doc->link].l_form->submit_action); + } + if (!(links[doc->link].l_form->submit_method && + links[doc->link].l_form->submit_action)) { + fprintf(fp0, "<dt> %s\n", + LYEntifyTitle(&buffer, gettext("(Form field)"))); + } + } else { + dt_URL(fp0, NonNull(links[doc->link].lname)); + } + END_DL(); + + } else { + fprintf(fp0, "<h2>%s</h2>", + LYEntifyTitle(&buffer, + gettext("No Links on the current page"))); + } + + if ((cp = HText_getHttpHeaders()) != 0) { + fprintf(fp0, "<h2>%s</h2>", + LYEntifyTitle(&buffer, gettext("Server Headers:"))); + fprintf(fp0, "<pre>%s</pre>", + LYEntifyTitle(&buffer, cp)); + } +#ifdef DIRED_SUPPORT + } +#endif /* DIRED_SUPPORT */ + EndInternalPage(fp0); + + LYrefresh(); + + LYCloseTemp(tempfile); + FREE(Title); + FREE(buffer); + + return (0); +} |