/*			File Transfer Protocol (FTP) Client
**			for a WorldWideWeb browser
**			===================================
**
**	A cache of one control connection is kept
**
** Note: Port allocation
**
**	It is essential that the port is allocated by the system, rather
**	than chosen in rotation by us (POLL_PORTS), or the following
**	problem occurs.
**
**	It seems that an attempt by the server to connect to a port which has
**	been used recently by a listen on the same socket, or by another
**	socket this or another process causes a hangup of (almost exactly)
**	one minute. Therefore, we have to use a rotating port number.
**	The problem remains that if the application is run twice in quick
**	succession, it will hang for what remains of a minute.
**
** Authors
**	TBL	Tim Berners-lee <timbl@info.cern.ch>
**	DD	Denis DeLaRoca 310 825-4580 <CSP1DWD@mvs.oac.ucla.edu>
** History:
**	 2 May 91	Written TBL, as a part of the WorldWideWeb project.
**	15 Jan 92	Bug fix: close() was used for NETCLOSE for control soc
**	10 Feb 92	Retry if cached connection times out or breaks
**	 8 Dec 92	Bug fix 921208 TBL after DD
**	17 Dec 92	Anon FTP password now just WWWuser@ suggested by DD
**			fails on princeton.edu!
**      30 Jun 95       Re-added cached connection so it works; Added support
**			to display informational messages that the FTP site
**			shows.
**
*/

/* SOCKS mods by:
 * Ying-Da Lee, <ylee@syl.dl.nec.com>
 * NEC Systems Laboratory
 * C&C Software Technology Center
 */
#include "../config.h"
#include <X11/Intrinsic.h>
#include <string.h>
#include <stdlib.h>
#include <time.h>
#include "HTFTP.h"	/* Implemented here */
#include "../libnut/str-tools.h"
#define LINE_LENGTH 1024

#include "HTParse.h"
#include "HTUtils.h"
#include "tcp.h"
#include "HTTCP.h"
#include "HTAnchor.h"
#include "HTFile.h"
#include "HTChunk.h"
#include "HTSort.h"
#include "HText.h"

#ifndef MAXHOSTNAMELEN
#define MAXHOSTNAMELEN 64           /* Arbitrary limit */
#endif

/*For HTAA_LOGIN */
#include "HTAAUtil.h"

#ifndef IPPORT_FTP
#define IPPORT_FTP	21
#endif

#ifdef __STDC__
#include <stdlib.h>
#endif

#ifndef NIL
#define NIL 0
#endif

#ifndef DISABLE_TRACE
extern int www2Trace;
#endif

#define SWP_HACK

/*		Hypertext object building machinery
*/
#include "HTML.h"

#define PUTC(c) (*targetClass.put_character)(target, c)
#define PUTS(s) (*targetClass.put_string)(target, s)
#define START(e) (*targetClass.start_element)(target, e, 0, 0)
#define END(e) (*targetClass.end_element)(target, e)
#define END_TARGET (*targetClass.end_document)(target)
#define FREE_TARGET (*targetClass.free)(target)
struct _HTStructured {
	WWW_CONST HTStructuredClass *	isa;
	/* ... */
};

struct _HTStream {
      WWW_CONST HTStreamClass*	isa;
      /* ... */
};

/* 
** Info for cached connection;  right now we only keep one around for a while
*/  
extern XtAppContext app_context;
extern int ftp_timeout_val;
extern int securityType;
void close_it_up();
extern int noLength;

char *prompt_for_password(char *msg);

/*SWP -- Yukky Cool Kludge*/
int ftpKludge=0;
extern int loading_length;
BOOL usingNLST = 0;

/*SWP -- FTP "redial" using ftpRedial resource for number of times*/
extern int ftpRedial;
extern int ftpRedialSleep;
extern int ftpFilenameLength;
extern int ftpEllipsisLength;
extern int ftpEllipsisMode;

/*SWP -- 9.27.95 -- Directory parsing*/
#define NEW_PARSE
#ifdef NEW_PARSE
int ParseDate(char *szBuffer, char *szFileInfo, char *szMonth, char *szDay, char *szYear, char *szTime);
int ParseFileSizeAndName(char *szBuffer, char *szFileName, char *szSize);
#endif

HText *HT;
int fTimerStarted = 0;
XtIntervalId timer;
  static struct ftpcache {
		int control;
		char p1[256];
		char host[256];
		char username[BUFSIZ];
		char password[BUFSIZ];
	} ftpcache = {
		-1,
		"",
		"",
		"",
		""
	};

#ifdef SOCKS
extern struct in_addr SOCKS_ftpsrv; /* in HTFTP.c */
#endif

/* HTFTPClearCache ()
   Expects: Nothing
   Returns: Nothing
*/
void HTFTPClearCache (void)
{
  ftpcache.password[0] = '\0';
  ftpcache.control = -1;
  ftpcache.p1[0] = 0;
  ftpcache.host[0] = 0;
  ftpcache.username[0] = 0;
}


/*	Module-Wide Variables
**	---------------------
*/
PRIVATE char    response_text[LINE_LENGTH+1];/* Last response from NewsHost */
PRIVATE int control = -1;		/* Current connection */
PRIVATE int	data_soc = -1;		/* Socket for data transfer =invalid */

PRIVATE int     master_socket = -1;	/* Listening socket = invalid	*/
PRIVATE char	port_command[255];	/* Command for setting the port */


#define DATA_BUFFER_SIZE 2048
PRIVATE char data_buffer[DATA_BUFFER_SIZE];		/* Input data buffer */
PRIVATE char * data_read_pointer;
PRIVATE char * data_write_pointer;


/*	Procedure: Read a character from the data connection
**	----------------------------------------------------
*/
PRIVATE int interrupted_in_next_data_char = 0;
PRIVATE char next_data_char NOARGS
{
  int status;
  interrupted_in_next_data_char = 0;
  if (data_read_pointer >= data_write_pointer) 
    {
      status = NETREAD(data_soc, data_buffer, DATA_BUFFER_SIZE);
      if (status == HT_INTERRUPTED)
        interrupted_in_next_data_char = 1;
      if (status <= 0) 
        return (char)-1;
      data_write_pointer = data_buffer + status;
      data_read_pointer = data_buffer;
    }
  return *data_read_pointer++;
}


/*	Execute Command and get Response
**	--------------------------------
**
**	See the state machine illustrated in RFC959, p57. This implements
**	one command/reply sequence.  It also interprets lines which are to
**	be continued, which are marked with a "-" immediately after the
**	status code.
**
**	Continuation then goes on until a line with a matching reply code
**	an a space after it.
**
** On entry,
**	con	points to the connection which is established.
**	cmd	points to a command, or is NIL to just get the response.
**
**	The command is terminated with the CRLF pair.
**
** On exit,
**	returns:  The first digit of the reply type,
**		  or negative for communication failure.
*/
#ifdef __STDC__
PRIVATE int response (char * cmd)
#else
PRIVATE int response (cmd)
    char * cmd;
#endif
{
  int result;				/* Three-digit decimal code */
#ifdef OLD
  int continuation_response = -1;
#endif
  int status;
  char	continuation;
  int multiline_response = 0;

  int messageStarted = 0;

char *ptr;
char bytestr[256],*byteptr;
int bytes;
  
  if (!control || control == -1) 
    {
#ifndef DISABLE_TRACE
      if(www2Trace) 
        fprintf(stderr, "FTP: No control connection set up!!\n");
#endif
      return -99;
    }
  
  if (cmd) 
    {
#ifndef DISABLE_TRACE
      if (www2Trace) 
        fprintf(stderr, "  Tx: %s", cmd);
#endif
      
      status = NETWRITE(control, cmd, (int)strlen(cmd));
      if (status<0) 
        {
#ifndef DISABLE_TRACE
          if (www2Trace) fprintf(stderr, 
                             "FTP: Error %d sending command: closing socket %d\n",
                             status, control);
#endif
          CLOSE_CONTROL(control);
          control = -1;
          return status;
	}
    }
  
  /* Patch to be generally compatible with RFC 959 servers  -spok@cs.cmu.edu  */
  /* Multiline responses start with a number and a hyphen;
     end with same number and a space.  When it ends, the number must
     be flush left. */
  do 
    {
      char *p = response_text;
      /* If nonzero, it's set to initial code of multiline response */
      for (;;)
        {
          int foo;
          /* This is set to 0 at the start of HTGetCharacter. */
          extern int interrupted_in_htgetcharacter;
          
          foo = (*p++ = HTGetCharacter ());
          if (interrupted_in_htgetcharacter)
            {
#ifndef DISABLE_TRACE
              if (www2Trace)
                fprintf (stderr, "FTP: Interrupted in HTGetCharacter, apparently.\n");
#endif
              CLOSE_CONTROL (control);
              control = -1;
              return HT_INTERRUPTED;
            }
          
          if (foo == LF || 
       /* if (((*p++=NEXT_CHAR) == '\n') || */
              (p == &response_text[LINE_LENGTH])) 
            {
              *p++=0;                 /* Terminate the string */
#ifndef DISABLE_TRACE
              if (www2Trace) 
                fprintf(stderr, "    Rx: %s", response_text);
#endif
	      if (!strncmp(response_text,"150",3)) {
                  if((ptr=strrchr(response_text,'(')) && *ptr){
                      bytes = atoi((ptr+1));
		  } else {   
                      bytes=0;
                  }
		if (bytes==0) {
			loading_length=(-1);
		}
		else {
			noLength=0;
			loading_length=bytes;
		}
	      }
	      else {
		noLength=0;
		HTMeter(100,NULL);
		noLength=1;
	      }

              sscanf(response_text, "%d%c", &result, &continuation);

		if ((response_text[0] == '2') || (response_text[0] == '5')) {
				
			if (continuation == '-') {
				char *p;

				if (messageStarted == 0) {
					HText_appendText(HT, "<PRE>\n");
        				HTProgress ("Receiving directory message");
					messageStarted = 1;
				}

				p = strchr(response_text, '-');
				p++;
				if (p != NULL)
					HText_appendText (HT, p);
			}
		}

				
              if (continuation == '-' && !multiline_response) 
                {
                  multiline_response = result;
                }
              else if (multiline_response && continuation == ' ' &&
                       multiline_response == result &&
                       isdigit(response_text[0])) 
                {
                  /* End of response (number must be flush on left) */
                  multiline_response = 0;
                }
              break;
            } /* if end of line */
          
          if (*(p-1) < 0) 
            {
#ifndef DISABLE_TRACE
              if (www2Trace) 
                fprintf(stderr, "Error on rx: closing socket %d\n",
                        control);
#endif
#ifdef SWP_HACK
	      loading_length=(-1);
#endif
              strcpy (response_text, "000 *** TCP read error on response\n");
              CLOSE_CONTROL(control);
              control = -1;
              return -1;	/* End of file on response */
            }
        } /* Loop over characters */
    } 
  while (multiline_response);

  if (messageStarted)
	  HText_appendText(HT, "</PRE><HR>\n");

  
#ifdef OLD
  do 
    {
      char *p = response_text;
      for(;;) 
        {  
          int foo;
          /* This is set to 0 at the start of HTGetCharacter. */
          extern int interrupted_in_htgetcharacter;

          foo = (*p++ = HTGetCharacter ());
          if (interrupted_in_htgetcharacter)
            {
#ifndef DISABLE_TRACE
              if (www2Trace)
                fprintf (stderr, "FTP: Interrupted in HTGetCharacter, apparently.\n");
#endif
              CLOSE_CONTROL (control);
              control = -1;
              return HT_INTERRUPTED;
            }

          if (foo == LF || 
              p == &response_text[LINE_LENGTH]) 
            {
              char continuation;
              int rv;

              *p++=0;			/* Terminate the string */
#ifndef DISABLE_TRACE
              if (www2Trace) 
                fprintf(stderr, "    Rx: %s", response_text);
#endif
              /* Clear out result ahead of time to see if we couldn't
                 read a real value. */
              result = -1;
              rv = sscanf(response_text, "%d%c", &result, &continuation);
              /* Try just continuing if we couldn't pull out
                 a value for result and the response_text starts with
                 whitespace. */
              if (rv < 2 && result == -1 && 
                  (*response_text == ' ' || *response_text == '\t'))
                {
                  /* Dunno what to do here -- the code isn't really
                     set up to deal with continuation lines starting
                     with whitespace.  Testcase is
                     reports.adm.cs.cmu.edu. */
                }
              else if (continuation_response == -1) 
                {
                  if (continuation == '-')  /* start continuation */
                    continuation_response = result;
                } 
              else 
                { 	/* continuing */
                  if (continuation_response == result && continuation == ' ')
                    continuation_response = -1;	/* ended */
                }	
              break;	    
            } /* if end of line */
          
          if (*(p-1) == EOF) 
            {
#ifndef DISABLE_TRACE
              if (www2Trace) 
                fprintf(stderr, "Error on rx: closing socket %d\n",
                        control);
#endif
              strcpy (response_text, "000 *** TCP read error on response\n");
              CLOSE_CONTROL(control);
              control = -1;
              return -1;	/* End of file on response */
            }
        } /* Loop over characters */
        
    } 
  while (continuation_response != -1);
#endif
  
  if (result == 421) 
    {
#ifndef DISABLE_TRACE
      if(www2Trace) 
        fprintf(stderr, "FTP: They close so we close socket %d\n",
                control);
#endif
#ifdef SWP_HACK
      loading_length=(-1);
#endif
      CLOSE_CONTROL(control);
      return -1;
    }

  if (result==550) {
	HTProgress(response_text);
  }

  return result/100;
}


/*	Get a valid connection to the host
**	----------------------------------
**
** On entry,
**	arg	points to the name of the host in a hypertext address
** On exit,
**	returns	<0 if error
**		socket number if success
**
**	This routine takes care of managing timed-out connections, and
**	limiting the number of connections in use at any one time.
**
**	It ensures that all connections are logged in if they exist.
**	It ensures they have the port number transferred.
*/

PRIVATE int get_connection ARGS1 (char *,arg)
{
  int status, con;
  
  static char host[BUFSIZ];
  static char username[BUFSIZ];
  static char password[BUFSIZ];
  char dummy[MAXHOSTNAMELEN+32];  

int redial=0;

  if (!arg) 
    return -1;		/* Bad if no name sepcified	*/
  if (!*arg) 
    return -1;		/* Bad if name had zero length	*/
  
#ifndef DISABLE_TRACE
  if (www2Trace) 
    fprintf(stderr, "FTP: Looking for %s\n", arg);
#endif

  {
    char *p1 = HTParse(arg, "", PARSE_HOST);
    char *p2 = strrchr(p1, '@');        /* user? */
    char * pw;
    char * un;

    /* Save the actual host */
    {
    char *tmpptr;

	strcpy(host,p1);
	tmpptr=strchr(host,'/');
	if (tmpptr) {
		*tmpptr='\0';
	}
    }

    if (p2) {
        un = p1;
        *p2=0;                            /* terminate */
        p1 = p2+1;                        /* point to host */
        pw = strchr(un, ':');
        if (pw) 
          {
            *pw++ = 0;
/*
            password = pw;
*/
          }
	if (strcmp(un,username)) { /*new username*/
		strcpy(username,un);
		if (pw) {
			strcpy(password,pw);
		}
		else {
			*password='\0';
		}
	}
	else { /*same username*/
		if (!*(ftpcache.host) || (*(ftpcache.host) && strcmp(host,ftpcache.host))) { /*new host*/
			*password='\0';
		}
	}
    }
    /*no username*/
    else { 
	if (strcmp(username,"anonymous")) { /*last one was not anon*/
		*username='\0';
		*password='\0';
	}
    }

    /* copy hostname into dummy URL, since username:password@ 
       might have been part of original */ 
    sprintf(dummy, "ftp://%s", p1);

#ifndef DISABLE_TRACE
    if(www2Trace)
    	fprintf (stderr, "FTP: set dummy to %s\n", dummy);
#endif

    /*Is the cache connection still good?*/
    if (ftpcache.control!=(-1) && *(ftpcache.host) && !strcmp(ftpcache.host,host) && *username) {
	/*Did we use a username before?*/
	if (!*username) {
		return(ftpcache.control);
	}

	/*Is the username and password used the same?*/
	if (*username && !strcmp(ftpcache.username,username) &&
	    *password && !strcmp(ftpcache.password,password)) {
		/* For security Icon */
		if (*username && strcmp(username,"anonymous") && strcmp(username,"ftp")) {
			/*not anon login...assuming a real login*/
			securityType=HTAA_LOGIN;
		}
		else {
			securityType=HTAA_NONE;
		}
		return(ftpcache.control);
	}
	/*Something has changed...reopen connection*/
	else {
		close(ftpcache.control);
	}
    }

    /*Connection is not good. Reopen*/
    strcpy(ftpcache.p1,p1);
    strcpy(ftpcache.host,host);
    if (*username) {
	strcpy(ftpcache.username,username);
    }
    else {
	ftpcache.username[0]='\0';
    }
    if (*password) {
	strcpy(ftpcache.password,password);
    }
    else {
	ftpcache.password[0]='\0';
    }

/*
    if ((ftpcache.p1[0] != NULL) && (strcmp(ftpcache.p1,p1) == 0) && (ftpcache.control != -1)) {
    	return ftpcache.control;
    } else {
	close(ftpcache.control);
	strcpy(ftpcache.p1,p1);
    }
*/

    if (!*username) {
      free(p1);
    }
  }

  /*default the redial values if out of range*/
  if (ftpRedial<0) {
	ftpRedial=0;
  }
  if (ftpRedialSleep<1) {
	ftpRedialSleep=1;
  }

redialFTP:

  con = -1;

#ifndef DISABLE_TRACE
  if (www2Trace)
	fprintf(stderr,"dummy = %s\n",dummy);
#endif

  status = HTDoConnect (dummy, "FTP", IPPORT_FTP, &con);
  
  if (status < 0)
    {
#ifndef DISABLE_TRACE
      if (www2Trace)
        {
          if (status == HT_INTERRUPTED)
            fprintf (stderr,
                     "FTP: Interrupted on connect\n");
          else
            fprintf(stderr, 
                    "FTP: Unable to connect to remote host for `%s'.\n",
                    arg);
        }
#endif
      if (status == HT_INTERRUPTED)
        HTProgress ("Connection interrupted.");
      if (con != -1)
        {
          CLOSE_CONTROL(con);
          con = -1;
        }
/*
      if (username) 
        free(username);
*/
      HTProgress ("Unable to connect to remote host.");
      return status;			/* Bad return */
    }
  
#ifndef DISABLE_TRACE
  if (www2Trace) 
    fprintf(stderr, "FTP connected, assigning control socket %d\n", con);
#endif
  control = con;			/* Current control connection */

  ftpcache.control = control;

  /* Initialise buffering for contron connection */
  HTInitInput (con);
  

  /* Now we log in; Look up username, prompt for pw. */
  {
    int status = response (NIL);	/* Get greeting */

    if (status == HT_INTERRUPTED)
      {
#ifndef DISABLE_TRACE
        if (www2Trace)
          fprintf (stderr,
                   "FTP: Interrupted at beginning of login.\n");
#endif
#ifdef SWP_HACK
	loading_length=(-1);
#endif
        HTProgress ("Connection interrupted.");
        CLOSE_CONTROL(control);
        control = -1;
        return HT_INTERRUPTED;
      }
    if (status == 2) 
      {		/* Send username */
        char * command;
        if (*username) 
          {
            command = (char*)malloc(10+strlen(username)+2+1);
            sprintf(command, "USER %s%c%c", username, CR, LF);
	    if (!*password) {

		char *pw=NULL;

		pw=prompt_for_password("Please Enter Your FTP Password:");
		if (pw && *pw) {
			strcpy(password,pw);
			strcpy(ftpcache.password,password);
			free(pw);
		}
		else {
			*password='\0';
			*(ftpcache.password)='\0';
        		HTProgress ("Connection aborted.");
        		CLOSE_CONTROL(control);
        		control = -1;
        		return HT_INTERRUPTED;
		}
	    }
          } 
        else 
          {
            command = (char*)malloc(25);
            sprintf(command, "USER anonymous%c%c", CR, LF);
	    strcpy(username,"anonymous");
	    strcpy(ftpcache.username,username);
          }
        status = response (command);
        free(command);
        if (status == HT_INTERRUPTED)
          {
#ifndef DISABLE_TRACE
            if (www2Trace)
              fprintf (stderr,
                       "FTP: Interrupted while sending username.\n");
#endif
#ifdef SWP_HACK
	    loading_length=(-1);
#endif
            HTProgress ("Connection interrupted.");
            CLOSE_CONTROL(control);
            control = -1;
            return HT_INTERRUPTED;
          }
      }
    if (status == 3) 
      {		/* Send password */
        char * command;
        if (*password) 
          {
            command = (char*)malloc(10+strlen(password)+2+1);
            sprintf(command, "PASS %s%c%c", password, CR, LF);
          } 
        else 
          {
            char * user = getenv("USER");
            extern char *machine_with_domain;
            char *host = machine_with_domain;
            if (!user) 
              user = "WWWuser";
            /* If not fully qualified, suppress it as ftp.uu.net
               prefers a blank to a bad name */
            if (!strchr(host, '.')) host = "";

            command = (char*)malloc(20+strlen(host)+2+1);
            sprintf(command,
		    "PASS %s@%s%c%c", user ? user : "WWWuser",
		    host, CR, LF); /*@@*/
	    sprintf(password,"%s@%s",(user?user:"WWWuser"),host);
	    strcpy(ftpcache.password,password);
          }
        status = response (command);
        free(command);
        if (status == HT_INTERRUPTED)
          {
#ifndef DISABLE_TRACE
            if (www2Trace)
              fprintf (stderr,
                       "FTP: Interrupted while sending password.\n");
#endif
#ifdef SWP_HACK
	    loading_length=(-1);
#endif
            HTProgress ("Connection interrupted.");
            CLOSE_CONTROL(control);
            control = -1;
            return HT_INTERRUPTED;
          }
      }
    
    if (status == 3) 
      {
        char temp[80];
/*
    	if (username) 
		free(username);
*/
        sprintf (temp, "ACCT noaccount%c%c", CR, LF);
        status = response (temp);
        if (status == HT_INTERRUPTED)
          {
#ifndef DISABLE_TRACE
            if (www2Trace)
              fprintf (stderr,
                       "FTP: Interrupted while sending ACCT.\n");
#endif
#ifdef SWP_HACK
	    loading_length=(-1);
#endif
            HTProgress ("Connection interrupted.");
            CLOSE_CONTROL(control);
            control = -1;
            return HT_INTERRUPTED;
          }
      }
    if (status != 2) 
      {
	if (status==HT_INTERRUPTED) {
#ifndef DISABLE_TRACE
		if (www2Trace)
			fprintf (stderr,
				 "FTP: Interrupted in redial attempt.\n");
#endif
#ifdef SWP_HACK
		loading_length=(-1);
#endif
		HTProgress ("Connection interrupted.");
		CLOSE_CONTROL(control);
		control = -1;
		return HT_INTERRUPTED;
	}

	if (*username && strcmp(username,"anonymous")) {
		HText_appendText(HT, "<H2>FTP login using username \"");
		HText_appendText(HT, username);
		HText_appendText(HT, "\" failed.</H2><BR>");
		if (*password) {
			HText_appendText(HT, "If you have a login ");
			HText_appendText(HT, "on this machine please check ");
			HText_appendText(HT, "to make sure the password you ");
			HText_appendText(HT, "are specifying is correct.");
		} else {
			HText_appendText(HT, "This is probably because you ");
			HText_appendText(HT, "didn't specify a password ");
			HText_appendText(HT, "along with your username.<BR>");
			HText_appendText(HT, "To do this you have to specify ");
			HText_appendText(HT, "the FTP line like this:<BR>");
			HText_appendText(HT, "<P>");
			HText_appendText(HT, "ftp://username:password@ftp_site/");
			HText_appendText(HT, "<P>");
			HText_appendText(HT, "<strong>OR</strong>");
			HText_appendText(HT, "<P>");
			HText_appendText(HT, "You can now just specify a username ");
			HText_appendText(HT, "and you will be prompted for your ");
			HText_appendText(HT, "password.");
			HText_appendText(HT, "<P>");
			HText_appendText(HT, "e.g. ftp://username@ftp_site/");
		}
	} else {

	char buf[BUFSIZ];

		if (redial<ftpRedial) {
			/*close down current connection*/
			ftpcache.control = -1;
			CLOSE_CONTROL(control);
			control = -1;

			/*tell them in the progress string*/
			sprintf(buf,"Login failed. Redial Attempt %d/%d. Sleeping %d seconds.",redial,ftpRedial,ftpRedialSleep);
			HTProgress(buf);

			/*if we're tracing, explain it all*/
#ifndef DISABLE_TRACE
			if (www2Trace) 
				fprintf(stderr, "FTP: Login fail: %s", response_text);
#endif
/*
			sleep(ftpRedialSleep);
*/

/* Commented out until we get a new "sleep" routine...SWP

			if (my_sleep(ftpRedialSleep,1)) {
				if (www2Trace)
#ifndef DISABLE_TRACE
					fprintf (stderr,
						 "FTP: Interrupted in sleep during redial attempt.\n");
#endif
#ifdef SWP_HACK
				loading_length=(-1);
#endif
				HTProgress ("Connection interrupted.");
				CLOSE_CONTROL(control);
				control = -1;
				return HT_INTERRUPTED;
			}

*/

			/*index redial and try again*/
			redial++;
			goto redialFTP;
		}

		/*Printout message and stop retrying*/
		sprintf(buf,"<H2>Anonymous FTP login failed.<br><br>There were %d redial attempts made.</h2>",redial);
		HText_appendText(HT, buf);

	}
        HTProgress("Login failed");
#ifndef DISABLE_TRACE
        if (www2Trace) 
          fprintf(stderr, "FTP: Login fail: %s", response_text);
#endif

	HText_appendText(HT,"\n\n<hr><p>Reason for Failure:<br><br><plaintext>\n");
	HText_appendText(HT,response_text);

#ifdef SWP_HACK
	loading_length=(-1);
#endif

	ftpcache.control = -1;

        CLOSE_CONTROL(control);
        control = -1;
        return -1;		/* Bad return */
      }
#ifndef DISABLE_TRACE
    if (www2Trace) 
      fprintf(stderr, "FTP: Logged in.\n");
#endif

    /* For security Icon */
    if (*username && strcmp(username,"anonymous") && strcmp(username,"ftp")) {
	/*not anon login...assuming a real login*/
	securityType=HTAA_LOGIN;
    }
    else {
	securityType=HTAA_NONE;
    }
  }
  
  return con;			/* Good return */
} /* Scope of con */


/*	Close Master (listening) socket
**	-------------------------------
**
**
*/
#ifdef __STDC__
PRIVATE void close_master_socket(void)
#else
PRIVATE void close_master_socket()
#endif
{
#ifndef DISABLE_TRACE
  if (www2Trace) 
    fprintf(stderr, "FTP: Closing master socket %d\n", master_socket);
#endif
  NETCLOSE(master_socket);
  master_socket = -1;

  return;
}


/*	Open a master socket for listening on
**	-------------------------------------
**
**	When data is transferred, we open a port, and wait for the server to
**	connect with the data.
**
** On entry,
**	master_socket	Must be negative if not set up already.
** On exit,
**	Returns		socket number if good
**			less than zero if error.
**	master_socket	is socket number if good, else negative.
**	port_number	is valid if good.
*/
#ifdef __STDC__
PRIVATE int get_listen_socket(void)
#else
PRIVATE int get_listen_socket()
#endif
{
  struct sockaddr_in soc_address;	/* Binary network address */
  struct sockaddr_in *sin = &soc_address;
  int new_socket;			/* Will be master_socket */
  
  /* Create internet socket */
  new_socket = socket (AF_INET, SOCK_STREAM, IPPROTO_TCP);
  
  if (new_socket < 0)
    return -1;
  
#ifndef DISABLE_TRACE
  if (www2Trace) 
    fprintf(stderr, "FTP: Opened master socket number %d\n", new_socket);
#endif
    
  /* Search for a free port. */
  sin->sin_family = AF_INET;	    /* Family = internet, host order  */
  sin->sin_addr.s_addr = INADDR_ANY; /* Any peer address */
  {
    int status;
    int address_length = sizeof(soc_address);
#ifdef SOCKS
    status = Rgetsockname(control,
#else
    status = getsockname(control,
#endif
                         (struct sockaddr *)&soc_address,
			 &address_length);
    if (status<0) 
      return -1;

#ifndef DISABLE_TRACE
      if(www2Trace) {
	fprintf(stderr, "FTP: This host is %s\n",
		HTInetString(sin));
      }
#endif

    soc_address.sin_port = 0; /* Unspecified: please allocate */
#ifdef SOCKS
    status=Rbind(new_socket,
#else
    status=bind(new_socket,
#endif
              (struct sockaddr*)&soc_address,
                /* Cast to generic sockaddr */
#ifdef SOCKS
              sizeof(soc_address), SOCKS_ftpsrv.s_addr);
#else
              sizeof(soc_address));
#endif
      if (status<0) 
        return -1;
      
      address_length = sizeof(soc_address);
#ifdef SOCKS
    status = Rgetsockname(new_socket,
#else
    status = getsockname(new_socket,
#endif
                           (struct sockaddr*)&soc_address,
                           &address_length);
    if (status<0) 
      return -1;
  }
  
#ifndef DISABLE_TRACE
  if(www2Trace) {
	fprintf(stderr, "FTP: bound to port %d on %s\n",
		(int)ntohs(sin->sin_port),
		HTInetString(sin));
  }
#endif

  if (master_socket >= 0) 
    close_master_socket ();
  
  master_socket = new_socket;
  
  /* Now we must find out who we are to tell the other guy */
  (void)HTHostName(); 	/* Make address valid - doesn't work*/
  sprintf(port_command, "PORT %d,%d,%d,%d,%d,%d%c%c",
          (int)*((unsigned char *)(&sin->sin_addr)+0),
          (int)*((unsigned char *)(&sin->sin_addr)+1),
          (int)*((unsigned char *)(&sin->sin_addr)+2),
          (int)*((unsigned char *)(&sin->sin_addr)+3),
          (int)*((unsigned char *)(&sin->sin_port)+0),
          (int)*((unsigned char *)(&sin->sin_port)+1),
          CR, LF);
  
  /* Inform TCP that we will accept connections */
#ifdef SOCKS
  if (Rlisten (master_socket, 1) < 0) 
#else
  if (listen (master_socket, 1) < 0) 
#endif
    {
      close_master_socket ();
      return -1;
    }

#ifndef DISABLE_TRACE
    if(www2Trace) {
	fprintf(stderr, "FTP: Master socket(), bind() and listen() all OK\n");
    }
#endif

  return master_socket;		/* Good */
} /* get_listen_socket */



/*	Read a directory into an hypertext object from the data socket
**	--------------------------------------------------------------
**
** On entry,
**	anchor		Parent anchor to link the this node to
**	address		Address of the directory
** On exit,
**	returns		HT_LOADED if OK
**			<0 if error.
**
** Author: Charles Henrich (henrich@crh.cl.msu.edu)  October 2, 1993
**
** Completely re-wrote this chunk of code to present FTP directory information
** in a much more useful manner.  Also included the use of icons. -Crh
*/
PRIVATE int read_directory
ARGS4 (
  HTParentAnchor *,		parent,
  WWW_CONST char *,			address,
  HTFormat,			format_out,
  HTStream *,			sink )
{
  HTFormat format;
  HTAtom *pencoding;
  char *filename = HTParse(address, "", PARSE_PATH + PARSE_PUNCTUATION);
  char buffer[BUFSIZ];
  char buf[BUFSIZ];
  char itemtype;
  char itemname[BUFSIZ];
  char itemsize[BUFSIZ];
  char *full_ftp_name, *ptr;
  int count, ret, cmpr, c, rv;
  extern char *HTgeticonname(HTFormat, char *);
char *ellipsis_string=(char *)calloc(1024,sizeof(char));
#ifdef NEW_PARSE
int nTime;
char szDate[256];
int nStringLen;
int nSpaces;
int nOldSpaces;
char szFileInfo[32];
char szMonth[32];
char szDay[16];
char szYear[32];
char szTime[32];
#endif

  HTProgress("Reading FTP directory");

  HText_appendText (HT, "<H1>FTP Directory ");
  HText_appendText (HT, filename);
  HText_appendText (HT, "</H1>\n");
#ifdef NEW_PARSE
  HText_appendText (HT, "<PRE>");
#endif
  HText_appendText (HT, "<DL>\n");
  data_read_pointer = data_write_pointer = data_buffer;

  /* If this isnt the root level, spit out a parent directory entry */

  if(strcmp(filename,"/") != 0)
    {
      HText_appendText(HT,"<DD>");

      HText_appendText(HT,"<A HREF=\"");
      
      strcpy(buffer,filename);
      ptr = strrchr(buffer,'/');
      
      if(ptr != NULL) *ptr='\0';
      
      if(buffer[0] == '\0') 
        HText_appendText(HT,"/");
      else
        HText_appendText(HT, buffer);
      
      HText_appendText(HT,"\"><IMG SRC=\"");
      HText_appendText(HT, HTgeticonname(NULL, "directory"));
      HText_appendText(HT,"\"> Parent Directory</a>");
    }
  
  /* Loop until we hit EOF */
  while(1)
    {
      /* Read in a line of data */
      for(count = 0; count < BUFSIZ; count++)
        {
          c = next_data_char ();
          if (interrupted_in_next_data_char)
            {
#ifndef DISABLE_TRACE
              if (www2Trace)
                fprintf (stderr, "FTP: Picked up interrupted_in_next_data_char\n");
#endif
              return HT_INTERRUPTED;
            }
          
          if (c == '\r')
            {
              c = next_data_char ();
              if (interrupted_in_next_data_char)
                {
#ifndef DISABLE_TRACE
                  if (www2Trace)
                    fprintf 
                      (stderr, "FTP: Picked up interrupted_in_next_data_char\n");
#endif
                  return HT_INTERRUPTED;
                }
              
              if (c != '\n') 
                break;
            }
          
          if (c == '\n' || c == (char)EOF) 
            break;
          
          buffer[count] = c;
	}
      
      if(c == (char)EOF) 
        break;
      
      buffer[count] = 0;
      
      /* Parse the input buffer, extract the item type, and the item size */
      
#if 0
      ret=sscanf(buffer,"%c%*9s%*d %*s %*s %s", &itemtype, itemsize);

      if (ret != 2) 
        continue;
#endif
      /* Retain whole string -- we don't use it at the moment, but we will. */
      full_ftp_name = strdup (buffer);
      /* Read but disregard itemsize -- this effectively guarantees we will know
         what we should display and what we shouldn't -- don't ask. */
      if (usingNLST==1) {
	ret=sscanf(buffer,"%c%*9s %*d %*s %*s %s", &itemtype, itemsize);
	if (ret != 2) {
		free (full_ftp_name);
		continue;
	}
      }
      else if (usingNLST==2) { /*only name*/
	if (!strcmp(buffer,".") || !strcmp(buffer,"..")) {
		free(full_ftp_name);
		continue;
	}

	ptr=strrchr(buffer,'.');
	itemtype='-';
	if (ptr && *ptr) {
		if (!my_strncasecmp(ptr,".dir",4)) {
			*ptr='\0';
			itemtype='d';
		}
	}
      }
      else { /*using LIST command*/
	ret=sscanf(buffer,"%c%*9s %*d %*s %*s %s", &itemtype, itemsize);
	if (ret != 2) {
		free (full_ftp_name);
		continue;
	}
      }

      if (!buffer || !*buffer) {
	continue;
      }

      if (usingNLST==2) { /*only name*/
	strcpy(itemname,buffer);
	nTime=(-1);
      }
      else {
	/* Due to the various time stamp formats, its "safer" to retrieve the        */
	/* filename by taking it from the right side of the string, we do that here. */
	ptr = strrchr(buffer,' ');
      
	if(ptr == NULL) 
		continue;
      
	strcpy(itemname,ptr+1);

	if (!strcmp(itemname,".") || !strcmp(itemname,"..")) {
		free(full_ftp_name);
		continue;
	}

#ifdef NEW_PARSE
	nTime = ParseDate(buffer, szFileInfo, szMonth, szDay, szYear, szTime);

	if (usingNLST==1) {
		ParseFileSizeAndName(buffer, itemname, itemsize);
	}

	if (nTime == 3) {  /* ie a dos or NT server possibly */
		if (!ParseFileSizeAndName(buffer, itemname, itemsize)) {
			itemtype = 'd';
		}
		else {
			itemtype = '-';
		}
	}
#endif
      }

      HText_appendText (HT, "<DD>");
      /* Spit out the anchor refrence, and continue on... */
      
      HText_appendText (HT, "<A HREF=\"");
      /* Assuming it's a relative reference... */
      if (itemname && itemname[0] != '/')
        {
          HText_appendText (HT, filename);
          if (filename[strlen(filename)-1] != '/') 
            HText_appendText (HT, "/");
        }
      HText_appendText (HT, itemname);
      HText_appendText (HT, "\">");
      
      /* There are 3 "types", directory, link and file.  If its a directory we     */
      /* just spit out the name with a directory icon.  If its a link, we go       */
      /* retrieve the proper name (i.e. the input looks like bob -> ../../../bob   */
      /* so we want to hop past the -> and just grab bob.  The link case falls     */
      /* through to the filetype case.  The filetype shows name and filesize, and  */
      /* then attempts to select the correct icon based on file extension.         */
      switch(itemtype) {
        case 'd':
          {
	    if (compact_string(itemname,ellipsis_string,ftpFilenameLength,ftpEllipsisMode,ftpEllipsisLength)) {
		strcpy(itemname,ellipsis_string);
	    }
            sprintf(buffer,"%s",itemname);
            HText_appendText(HT, "<IMG SRC=\"");
            HText_appendText(HT, HTgeticonname(NULL, "directory"));
            HText_appendText(HT, "\"> ");
            break;
          }
          
        case 'l':
          {
            ptr = strrchr(buffer,' ');
            if(ptr != NULL)
              {
                *ptr = '\0';
                ptr = strrchr(buffer,' ');
              }
            
            if(ptr != NULL)
              {
                *ptr = '\0';
                ptr = strrchr(buffer,' ');
              }
            
            if(ptr != NULL) strcpy(itemname,ptr+1);

	    if (compact_string(itemname,ellipsis_string,ftpFilenameLength,ftpEllipsisMode,ftpEllipsisLength)) {
		strcpy(itemname,ellipsis_string);
	    }
          }
          
        case '-':
          {
            
            /* If this is a link type, and the bytes are small, 
               its probably a directory so lets not show the byte
               count */
#if 0            
            if(itemtype == 'l' && atoi(itemsize) < 128) 
              {
                sprintf(buffer,"%s",itemname);
              }
            else
              {
                sprintf(buffer,"%s (%s bytes)",itemname,itemsize);
              }
#endif

#if 0
            if(itemtype == 'l') 
              {
#endif
		if (compact_string(itemname,ellipsis_string,ftpFilenameLength,ftpEllipsisMode,ftpEllipsisLength)) {
			strcpy(itemname,ellipsis_string);
		}
                sprintf(buffer,"%s",itemname);
#if 0
              }
            else
              {
                /* code doesn't work for this, and neither does pre. */
                sprintf(buffer,"<code>%s</code>",full_ftp_name);
              }
#endif
            
            format = HTFileFormat(itemname, &pencoding, WWW_SOURCE, &cmpr);
            
            if (1)
              {
                HText_appendText(HT, "<IMG SRC=\"");

                /* If this is a link, and we can't figure out what
                   kind of file it is by extension, throw up the unknown
                   icon; however, if it isn't a link and we can't figure
                   out what it is, throw up the text icon...
                   
                   Unless it's compressed. */
                if(itemtype == 'l' && cmpr == COMPRESSED_NOT) 
                  {
                    /* If it's unknown, let's call it a menu (since symlinks
                       are most commonly used on FTP servers to point to
                       directories, IMHO... -marc */
                    HText_appendText(HT, HTgeticonname(format, "directory") );
                  }
                else
                  {
                    HText_appendText(HT, HTgeticonname(format, "text")); 
                  }
                
                HText_appendText(HT, "\"> ");
              }
            else
              {
                HText_appendText(HT, "<IMG SRC=\"");
                HText_appendText(HT, HTgeticonname(format, "application")); 
                HText_appendText(HT, "\"> ");
              }

            break;
            }

          default:
            {
              HText_appendText(HT, "<IMG SRC=\"");
              HText_appendText(HT, HTgeticonname(NULL, "unknown")); 
              HText_appendText(HT, "\"> ");
              break;
            }
      }
      
	HText_appendText (HT, buffer);
#ifndef NEW_PARSE
	HText_appendText (HT, "</A>\n");
#endif

#ifdef NEW_PARSE
	HText_appendText (HT, "</A>");

	nStringLen = strlen(buffer);
	nSpaces = ftpFilenameLength - nStringLen;
/*
	if (itemtype != 'd') {
*/

		if (nTime == 1) {
			struct tm *ptr;
			time_t t;

			t=time(0);
			ptr=localtime(&t);
			sprintf(szYear,"%d",1900+ptr->tm_year);
			sprintf(szDate, "%*s%9s %s %s %s %2.2s, %s", nSpaces, " ", itemsize, szFileInfo, szTime, szMonth, szDay, szYear); 
		}
		else if (nTime == 0) {
			sprintf(szDate, "%*s%9s %s %s %s %2.2s, %s", nSpaces, " ", itemsize, szFileInfo, "     ", szMonth, szDay, szYear);
		}
		else {
			/*nSpaces += strlen(itemsize); */
			sprintf(szDate, "%*s  %9.9s  %s %s", nSpaces, " ", itemsize, szMonth, szTime);
		}
/*
	}
	else {
		nOldSpaces = nSpaces;
		nSpaces += 22;
		if (nTime == 1) {
			sprintf(szDate, "%*s  %s %s %2.2s", nSpaces, szFileInfo, szTime, szMonth, szDay); 
		}
		else if (nTime == 0) {
			sprintf(szDate, "%*s  %s %s %2.2s, %s", nSpaces, szFileInfo, "00:00", szMonth, szDay, szYear); 
		}
		else {
			sprintf(szDate, "%*s             %s %s", nOldSpaces, " ", szMonth, szTime);  
		}
	}
*/

	if (usingNLST!=2) {
		HText_appendText (HT, szDate);
	}
	HText_appendText (HT, "\n");
#endif

	free (full_ftp_name);
    }
  
  HText_appendText (HT, "</DL>\n");
#ifdef NEW_PARSE
  HText_appendText (HT, "</PRE>\n");
#endif
  HText_endAppend (HT);

  rv = response (NIL);
  if (rv == HT_INTERRUPTED)
    return rv;
  return rv == 2 ? HT_LOADED : -1;
}



/*	Retrieve File from Server
**	-------------------------
**
** On entry,
**	name		WWW address of a file: document, including hostname
** On exit,
**	returns		Socket number for file if good.
**			<0 if bad.
*/
PUBLIC int HTFTPLoad
ARGS4 (
  char *,			name,
  HTParentAnchor *,		anchor,
  HTFormat,			format_out,
  HTStream *,			sink
)
{
  BOOL isDirectory = NO;
  int status;
  int retry;			/* How many times tried? */
  HTFormat format;
  int compressed = 0;
  int try;

  HTProgress("Retrieval in progress");
	if (fTimerStarted) {
		XtRemoveTimeOut(timer);
		fTimerStarted = 0;
	}
	HT = HText_new ();
	HText_beginAppend (HT);
  
  for (retry = 0; retry < 2; retry++) 
    {
#ifndef DISABLE_TRACE
      if (www2Trace)
        fprintf (stderr, "FTP: TRYING in HTFTPLoad, attempt %d\n", retry);
#endif
      status = get_connection(name);
      if (status < 0) 
        {
          CLOSE_CONTROL (control);
          control = -1;
          /* HT_INTERRUPTED will fall through. */
#ifdef SWP_HACK
	  loading_length=(-1);
#endif
          return status;
        }
      
      status = get_listen_socket();
      if (status < 0)
        {
	  HText_appendText(HT, "FTP terminated because login failed");
	  HTProgress("Login failed");
          CLOSE_CONTROL (control);
          control = -1;
          close_master_socket ();
#ifdef SWP_HACK
	  loading_length=(-1);
#endif
          /* HT_INTERRUPTED would fall through, if we could interrupt
             somehow in the middle of it, which we currently can't. */
          return status;
        }
      
      /* Inform the server of the port number we will listen on */
      {
        status = response (port_command);
        if (status == HT_INTERRUPTED)
          {
#ifndef DISABLE_TRACE
            if (www2Trace)
              fprintf (stderr, "FTP: Interrupted in response (port_command)\n");
#endif
#ifdef SWP_HACK
	    loading_length=(-1);
#endif
            HTProgress ("Connection interrupted.");
            CLOSE_CONTROL (control);
            control = -1;
            close_master_socket ();
            return HT_INTERRUPTED;
          }
        if (status !=2) 
          {		/* Could have timed out */
            if (status < 0) 
              {
                CLOSE_CONTROL (control);
                control = -1;
                close_master_socket ();
                continue;		/* try again - net error*/
              }

            CLOSE_CONTROL (control);
            control = -1;
            close_master_socket ();
#ifdef SWP_HACK
	    loading_length=(-1);
#endif
            return HT_NOT_LOADED;			/* bad reply */
          }
#ifndef DISABLE_TRACE
        if (www2Trace)
          fprintf(stderr, "FTP: Port defined.\n");
#endif
      }
      status = 0;
      break;	/* No more retries */
    } /* for retries */

  if (status < 0)
    {
      close_master_socket ();
      CLOSE_CONTROL (control);
      control = -1;
#ifdef SWP_HACK
      loading_length=(-1);
#endif
      return status;	/* Failed with this code */
    }
  
  /* Ask for the file: */    
  {
    char *filename = HTParse(name, "", PARSE_PATH + PARSE_PUNCTUATION);
    char *fname, *ptr;
    char command[LINE_LENGTH+1];
    HTAtom *encoding;

    if (!(*filename)) 
      StrAllocCopy(filename, "/");
    format = HTFileFormat (filename, &encoding, WWW_PLAINTEXT, &compressed);
    
    sprintf(command, "TYPE %s%c%c", "I", CR, LF);
    status = response (command);
    if (status != 2) 
      {
        if (status == HT_INTERRUPTED)
          HTProgress ("Connection interrupted.");
        close_master_socket ();
        CLOSE_CONTROL (control);
        control = -1;
        free (filename);
#ifdef SWP_HACK
	loading_length=(-1);
#endif
        return (status == HT_INTERRUPTED) ? HT_INTERRUPTED : -1;
      }

    fname=strdup(filename);
    try=(-1);

tryAgain:
    try++;
    sprintf(command, "RETR %s%c%c", fname, CR, LF);
    status = response (command);
    if (status == HT_INTERRUPTED)
      {
#ifndef DISABLE_TRACE
        if (www2Trace)
          fprintf (stderr, "FTP: Interrupted while sending RETR\n");
#endif
        HTProgress ("Connection interrupted.");
        CLOSE_CONTROL (control);
        control = -1;
        close_master_socket ();
        free (filename);
#ifdef SWP_HACK
	loading_length=(-1);
#endif
        return HT_INTERRUPTED;
      }

    if (status != 1) {  /* Failed : try to CWD to it */
        sprintf(command, "CWD %s%c%c", fname, CR, LF);
        status = response (command);
        if (status == HT_INTERRUPTED)
          {
#ifndef DISABLE_TRACE
            if (www2Trace)
              fprintf (stderr, "FTP: Interrupted while sending CWD\n");
#endif
            HTProgress ("Connection interrupted.");
            CLOSE_CONTROL (control);
            control = -1;
            close_master_socket ();
            free (filename);
#ifdef SWP_HACK
	    loading_length=(-1);
#endif
            return HT_INTERRUPTED;
          }

	/* If we failed CWD and not at root and on type 2 NLST, try VMS style*/
	if (status==5 && strcmp(filename,"/") && usingNLST==2) {
		switch(try) {
			case 0:
				/* First, put filename in fname*/
				strcpy(fname,filename+1);
				goto tryAgain;

			case 1:
				/* First, put filename in fname*/
				strcpy(fname,filename+1);
				/* Back up to last character*/
				for (ptr=fname+strlen(fname)-1; ptr && *ptr && isspace(*ptr); ptr--) {
					*ptr='\0';
				}
				/* If the last char is a /, get rid of it*/
				if (fname[strlen(fname)-1]=='/') {
					fname[strlen(fname)-1]='\0';
				}
				goto tryAgain;

			case 2:
				/* First, put filename in fname*/
				strcpy(fname,filename+1);
				/* Back up to last character*/
				for (ptr=fname+strlen(fname)-1; ptr && *ptr && isspace(*ptr); ptr--) {
					*ptr='\0';
				}
				/* If the last char is a /, get rid of it*/
				if (fname[strlen(fname)-1]=='/') {
					fname[strlen(fname)-1]='\0';
				}
				/* Now change all / to .*/
				ptr=fname;
				while (ptr && *ptr) {
					if (*ptr=='/') {
						*ptr='.';
					}
					ptr++;
				}
				goto tryAgain;
			default:
				/*Failure...*/
				goto skipDir;
		}
	}

	/*if status is 2, we successfully did a CWD*/
	/*if status is 5 and we are at the root, assume we are on a vms
		machine and try to print out a directory*/
        if (status == 2 || (status == 5 && !strcmp(filename,"/")))
          {
		/* Successed : let's NLST it */
		isDirectory = YES;
		usingNLST=1;
		sprintf(command, "NLST %s %c%c", NLST_PARAMS, CR, LF);
		status = response (command);
		if (status == HT_INTERRUPTED) {
#ifndef DISABLE_TRACE
			if (www2Trace)
				fprintf (stderr, "FTP: Interrupted while sending LIST\n");
#endif
			HTProgress ("Connection interrupted.");
			CLOSE_CONTROL (control);
			control = -1;
			close_master_socket ();
			free (filename);
#ifdef SWP_HACK
			loading_length=(-1);
#endif
			return HT_INTERRUPTED;
		}

		if (status==5) { /*unrecognized command or failed*/
			isDirectory = YES;
			usingNLST=2;
			sprintf(command, "NLST%c%c", CR, LF);
			status = response (command);
			if (status == HT_INTERRUPTED) {
#ifndef DISABLE_TRACE
				if (www2Trace)
					fprintf (stderr, "FTP: Interrupted while sending NLST\n");
#endif
				HTProgress ("Connection interrupted.");
				CLOSE_CONTROL (control);
				control = -1;
				close_master_socket ();
				free (filename);
#ifdef SWP_HACK
				loading_length=(-1);
#endif
				return HT_INTERRUPTED;
			}
		}

		if (status==5) { /*unrecognized command or failed*/
			isDirectory = YES;
			usingNLST=0;
			sprintf(command, "LIST%c%c", CR, LF);
			status = response (command);
			if (status == HT_INTERRUPTED) {
#ifndef DISABLE_TRACE
				if (www2Trace)
					fprintf (stderr, "FTP: Interrupted while sending LIST\n");
#endif
				HTProgress ("Connection interrupted.");
				CLOSE_CONTROL (control);
				control = -1;
				close_master_socket ();
				free (filename);
#ifdef SWP_HACK
				loading_length=(-1);
#endif
				return HT_INTERRUPTED;
			}
		}
	  }
    }

skipDir:

    free(filename);
    free(fname);
    if (status != 1) 
      {
        CLOSE_CONTROL (control);
        control = -1;
        close_master_socket ();
#ifdef SWP_HACK
	loading_length=(-1);
#endif
        return HT_NOT_LOADED; /* Action not started */
      }
  }
  
  /* Wait for the connection */
  {
    struct sockaddr_in soc_address;

    int	soc_addrlen = sizeof(soc_address);
#ifdef SOCKS
    status = Raccept(master_socket,
#else
    status = accept(master_socket,
#endif
                    (struct sockaddr *)&soc_address,
                    &soc_addrlen);

    if (status < 0)
      {
        CLOSE_CONTROL (control);
        control = -1;
        close_master_socket ();
        /* We can't interrupt out of an accept. */
#ifdef SWP_HACK
	loading_length=(-1);
#endif
        return HT_NOT_LOADED;
      }

#ifndef DISABLE_TRACE
    if(www2Trace) {
	fprintf(stderr, "FTP: Accepted new socket %d\n", status);
    }
#endif

    data_soc = status;
  }

  if (isDirectory) 
    {
      int s = read_directory (anchor, name, format_out, sink);

#ifdef SWP_HACK
      loading_length=(-1);
#endif

      close_master_socket ();
      NETCLOSE (data_soc);
      data_soc = -1;

#ifndef DISABLE_TRACE
      if (www2Trace)
        fprintf (stderr, "FTP: Returning %d after doing read_directory\n", s);
#endif
      /* HT_INTERRUPTED should fall right through. */
      return s;
    } 
  else 
    {
      /* We reproduce ParseSocket below because of socket/child process
         problem. */
      HTStream * stream;
      HTStreamClass targetClass;    
      int rv;
      
      stream = HTStreamStack(format,
                             format_out,
                             compressed,
                             sink, anchor);
      
      if (!stream) 
        {
          char buffer[1024];	/* @@@@@@@@ */
          sprintf(buffer, "Sorry, can't convert from %s to %s.",
                  HTAtom_name(format), HTAtom_name(format_out));
          HTProgress (buffer);
#ifndef DISABLE_TRACE
          if (www2Trace) 
            fprintf(stderr, "FTP: %s\n", buffer);
#endif
#ifdef SWP_HACK
	  loading_length=(-1);
#endif
          return HT_NOT_LOADED;
        }
      
      targetClass = *(stream->isa);	/* Copy pointers to procedures */
      ftpKludge=1;
      rv = HTCopy(data_soc, stream, 0);
      ftpKludge=0;
      loading_length=(-1);
      if (rv == -1)
        {
          rv = HT_INTERRUPTED;
        }
      else
        {
          (*targetClass.end_document)(stream);
          /* Do NOT call *targetClass.free yet -- sockets aren't closed. */
          rv = HT_LOADED;
        }

#ifndef DISABLE_TRACE
      if (www2Trace)
        fprintf (stderr, "FTP: Got back %d from our local equivalent of ParseSocket\n", rv);
#endif

      /* Reset buffering to control connection -- probably
         no longer necessary, since we don't use a connection
         more than once. */
      HTInitInput(control);

#ifndef DISABLE_TRACE
      if (www2Trace)
        fprintf (stderr, "FTP: Closing data socket %d\n", data_soc);
#endif
      NETCLOSE (data_soc);
      data_soc = -1;

      /* Unfortunately, picking up the final reply sometimes causes
         serious problems.  It *probably* isn't safe not to do this,
         as there is the possibility that FTP servers could die if they
         try to send it and we're not listening.

         Testcase for problems (10/30/93): uxc.cso.uiuc.edu,
         AnswerGarden COPYRIGHT in X11R5 contrib clients.
         
         Of course, we may already be triggering hostile actions
         by allowing client-side interrupts as follows... */
      if (rv != HT_INTERRUPTED)
        {
#ifndef DISABLE_TRACE
          if (www2Trace)
            fprintf (stderr, "FTP: Picking up final reply...\n");
#endif

          status = response (NIL);		/* Pick up final reply */
          if (status == HT_INTERRUPTED)
            {
#ifndef DISABLE_TRACE
              if (www2Trace)
                fprintf (stderr, "FTP: Interrupted picking up final reply.\n");
#endif
              HTProgress ("Connection interrupted.");

              CLOSE_CONTROL (control);
              control = -1;
              close_master_socket ();

              (*targetClass.handle_interrupt)(stream);

              return HT_INTERRUPTED;
            }
          if (status != 2)
            {
              CLOSE_CONTROL (control);
              control = -1;
              close_master_socket ();
              return HT_NOT_LOADED;
            }
        }
      
      close_master_socket ();

      if (rv != HT_INTERRUPTED)
        {
          /* WAIT until all sockets have been closed. */
#ifndef DISABLE_TRACE
          if (www2Trace)
            fprintf (stderr, "FTP: Calling free method, finally.\n");
#endif
          (*targetClass.free)(stream);
        }

	timer = XtAppAddTimeOut(app_context, ftp_timeout_val*1000, close_it_up, NULL);
	fTimerStarted = 1;

      return rv == HT_INTERRUPTED ? HT_INTERRUPTED : HT_LOADED;
    }
} /* End HTFTPLoad */


/* HTFTPMkDir  Request that a directory be created on the FTP site.
** Expects:    *name is a pointer to a string that consists of the FTP URL with 
**                   the remote directory name.
** Returns     0 if successful, nonzero on error
*/
PUBLIC int HTFTPMkDir ARGS1 ( char *, name )
{
 char *curpath, *path;
 char command[ LINE_LENGTH+1];
 int status, method = 0;

 HTProgress ("FTP mkdir in progress");
 if(fTimerStarted) {
   XtRemoveTimeOut (timer);
   fTimerStarted = 0;
 }

 /* Open a connection (or get a cached connection) to the FTP site */
 status = get_connection (name);
 if (status < 0) {
   CLOSE_CONTROL (control);
   control = -1;
   return status;
 }

 /* The remote directory name is in the url, so pull it out 
    i.e. ftp://warez.yomama.com/pub/31337&warez_boy
    means to make the directory warez_boy at ftp://warez.yomama.com/pub/31337
 */
 if ((path = strchr(name, '&')) == NULL) { /* No dirname in this URL */
   close_master_socket ();
   CLOSE_CONTROL (control);
   control = -1;
   return -1;		
 }
 *path = '\0';	      /* Make the url normal */
 path++;	      /* Move to the dirname */
 /* *path is the directory name to create */

 curpath =  HTParse (name, "", PARSE_PATH+PARSE_PUNCTUATION); 
 if (!curpath || !(*curpath)) 
   curpath = strdup ("/");
 /* *curpath is the remote directory in which to create *path */

 /* First change to current directory on the server */
 sprintf (command, "CWD %s%c%c", curpath, CR, LF);
 status = response (command);
 if (status != 2) {
   close_master_socket ();
   CLOSE_CONTROL (control);
   control = -1;
   if (status = HT_INTERRUPTED) 
     HTProgress ("Connection interrupted");
   return (status==HT_INTERRUPTED)?-2:-1;
 }

 /* Now create the directory on the server */
 sprintf (command, "MKD %s%c%c", path, CR, LF);
 status = response (command);
 if (status != 2) {
   close_master_socket ();
   CLOSE_CONTROL (control);
   control = -1;
   if (status = HT_INTERRUPTED) 
     HTProgress ("Connection interrupted");
   return (status==HT_INTERRUPTED)?-2:-1;
 }

 /* Clean up */
 close_master_socket ();
 CLOSE_CONTROL (control);
 control = -1;
 HTProgress ("Created remote directory.");
 return 0;
} /* end HTFTPMkDir */


/* HTFTRemove  Request that a file (or directory) be removed from the FTP site
** Expects:    *name is a pointer to a string that consists of the FTP URL with the remote filename
**             included.
** Returns     0 if successful, nonzero on error
*/
PUBLIC int HTFTPRemove ARGS1 ( char *, name )
{
 char *fname, *filename, *path;
 char command[ LINE_LENGTH+1];
 int status, method = 0, didIt = 0;

 HTProgress ("FTP remove in progress");
 if(fTimerStarted) {
   XtRemoveTimeOut (timer);
   fTimerStarted = 0;
 }

 /* Open a connection (or get a cached connection) to the FTP site */
 status = get_connection (name);
 if (status < 0) {
   CLOSE_CONTROL (control);
   control = -1;
   return status;
 }

 /* Pull out the filename (and path) */
 fname = HTParse (name, "", PARSE_PATH+PARSE_PUNCTUATION);
 if(!(*fname)) 
   StrAllocCopy (filename, "/");

 /* Pull out just the filename */
 filename = strrchr (fname, '/');
 filename++;
 if (!(*filename)) {   /* No filename in the URL */
   close_master_socket ();
   CLOSE_CONTROL (control);
   control = -1;
   return -1;
 }
 
 /* *fname is the full path to the file, *filename is just the filename */
 for (method =0; method < 2; method++ ) {
   switch (method) {

     /* First, attempt to CWD to fname, if successful, fname is a directory. 
	So, CDUP to get to the parent and call RMD on filename  */
   case 0:
     sprintf (command, "CWD %s%c%c", fname, CR, LF);
     status = response (command);
     if (status != 2) {
       if (status == 5) {  /* Not a directory */
	 continue;
       }
       close_master_socket ();
       CLOSE_CONTROL (control);
       control = -1;
       if (status == HT_INTERRUPTED) 
	 HTProgress ("Connection interrupted.");
       return (status == HT_INTERRUPTED)?-2:-1;
     }
     /* Must be a directory, move up and RMD it*/
     *(filename-1) = 0;  /* Make fname -> just the path of the parent directory */
     sprintf (command, "CWD %s%c%c", fname, CR, LF);
     status = response (command);
     if (status != 2) {
       close_master_socket ();
       CLOSE_CONTROL (control);
       control = -1;
       if (status == HT_INTERRUPTED) 
	 HTProgress ("Connection interrupted.");
       return (status == HT_INTERRUPTED)?-2:-1;
     }
     sprintf (command, "RMD %s%c%c", filename, CR, LF);
     status = response (command);
     if (status != 2) {
       close_master_socket ();
       CLOSE_CONTROL (control);
       control = -1;
       if (status == HT_INTERRUPTED)
	 HTProgress ("Connection interrupted.");
       return (status == HT_INTERRUPTED)?-2:status;
     }
     didIt = 1;
     break;

     /* If the first attempt failed, CWD to fname and DELE filename */
   case 1:
     *(filename-1) = 0;  /* Make fname -> just the path of the file */
     sprintf (command, "CWD %s%c%c", fname, CR, LF);
     status = response (command);
     if (status != 2) {
       close_master_socket ();
       CLOSE_CONTROL (control);
       control = -1;
       if (status == HT_INTERRUPTED) {
	 HTProgress ("Connection interrupted.");
       }
       return (status == HT_INTERRUPTED)?HT_INTERRUPTED:status;
     }

     sprintf (command, "DELE %s%c%c", filename, CR, LF);
     status = response (command);
     if (status != 2) {
       close_master_socket ();
       CLOSE_CONTROL (control);
       control = -1;
       if (status == HT_INTERRUPTED) {
	 HTProgress ("Connection interrupted.");
       }
       return (status == HT_INTERRUPTED)?-2:status;
     }
     didIt = 1;
     break;
   } /* end of switch (method) */
   if(didIt)
     break;
 } /* end for(method.. */

 /* Clean up */
 close_master_socket ();
 CLOSE_CONTROL (control);
 control = -1;
 return 0;
} /* end HTFTPRemove */


/* HTFTPSend   Send File to the FTP site
** Expects:    *name is a pointer to a string that consists of the FTP URL with the local filename
**             appended to the URL (delimited by an &. i.e. ftp://warez.mama.com/pub&/tmp/bubba.tgz
**             would send /tmp/bubba.tgz to warez.mama.com:/pub
** Returns     0 if successful, nonzero on error
*/
#define OUTBUFSIZE 4096  /* Size of the chunk of the file read */
PUBLIC int HTFTPSend ARGS1 ( char *, name ) {
 int status;
 FILE *f;
 char *fname, *filename, *path;
 char command[ LINE_LENGTH+1], outBuf[ OUTBUFSIZE+1];
 long bLeft, bDone, bTotal, chunk;
 extern int twirl_increment;
 int next_twirl = twirl_increment, intr = 0;
 struct sockaddr_in soc_address;
 int soc_addrlen = sizeof (soc_address);
 struct stat sbuf;
 
        HTProgress ("FTP send in progress.");

	if (fTimerStarted) {
	  XtRemoveTimeOut (timer);
	  fTimerStarted = 0;
	}
	
	/* The local filename is in the url, so pull it out 
		 i.e. ftp://warez.yomama.com/pub/31337&/u/warezboy/Mosaic0.1a.tar.gz
		 means to send /u/warezboy/Mosaic0.1a.tar.gz to warez.yomama.com/pub/31337
	*/
	if ((fname = strchr(name, '&')) == NULL) { /* No local filename in this URL */
	  close_master_socket ();
	  CLOSE_CONTROL (control);
	  control = -1;
	  return -1;		
	}

	*fname = '\0';	      /* Make the url normal */
	fname++;	      /* Move to the filename */

	filename = strrchr (fname, '/');
	filename++;
	if (!(*filename)) {		/* no filename */
	  close_master_socket ();
	  CLOSE_CONTROL (control);
	  control = -1;
	  return -1;		
	}
	
	/* *fname is the full path and filename, *filename is just the filename */	
	/* get size information */
	if( stat (fname, &sbuf) < 0) {
	  CLOSE_CONTROL (control);
	  control = -1;
	  close_master_socket ();
	  return -1;
	}
		
	bTotal = sbuf.st_size;
#ifndef DISABLE_TRACE
	if(www2Trace)
	  fprintf (stderr, "HTFTPSend: Attempting to send %s (%s) (%lu)\n", fname, filename, bTotal);
#endif

	status = get_connection (name);
	if (status<0) {
	  CLOSE_CONTROL (control);
	  control = -1;
	  close_master_socket ();
	  return status;
	}

	status = get_listen_socket ();
	if (status<0) {
	  CLOSE_CONTROL (control);
	  control = -1;
	  close_master_socket ();
	  return status;
	}
	
	status = response (port_command);
	if (status == HT_INTERRUPTED) {
     	  HTProgress ("Connection interrupted.");
	  CLOSE_CONTROL (control);
	  control = -1;
	  close_master_socket ();
	  return -2;
	}

	if (status != 2) {		/* If the port command was not successful */
	  CLOSE_CONTROL (control);
	  control = -1;
	  close_master_socket ();
	  if (status < 0) { /*If we were going to try again, we would do it here.... */
	    return -3;
	  }
	  return -3;
	}
	
	/* Logged in, set up the port, now let's send the sucka */

	/* Set the type to image */
	sprintf (command, "TYPE %s%c%c", "I", CR, LF);
	status = response (command);	
	if (status != 2) {
		close_master_socket ();
		CLOSE_CONTROL (control);
		control = -1;
		if (status == HT_INTERRUPTED) 
			HTProgress ("Connection interrupted.");
		return (status == HT_INTERRUPTED)?-2:-1;	
	}
	
	/* Move to correct directory */
	path = HTParse (name, "", PARSE_PATH+PARSE_PUNCTUATION);
	if (!(*path))
		StrAllocCopy (path, "/");
		
	sprintf (command, "CWD %s%c%c", path, CR, LF);
	status = response (command);

	if (status != 2) {
		close_master_socket ();
		CLOSE_CONTROL (control);
		control = -1;
		if (status == HT_INTERRUPTED) 
			HTProgress ("Connection interrupted.");
		return (status == HT_INTERRUPTED)?-2:-1;	
	}

	/* Send it */
	sprintf (command, "STOR %s%c%c", filename, CR, LF);
	status = response (command);
	if (status == HT_INTERRUPTED) {
		HTProgress ("Connection interrupted.");
		CLOSE_CONTROL (control);
		control = -1;
		close_master_socket ();
		return -2;
	}
	
	if (status != 1) { /* Does not seem to understand the STOR command */
		HTProgress ("FTP host does not understand STOR command.");
		CLOSE_CONTROL (control);
		control = -1;
		close_master_socket ();
		return -3;
	}
	
	/* Ready to send the data now, server is primed and ready... here we go, go go */
	
#ifdef SOCKS
	status = Raccept (master_socket, (struct sockaddr *)&soc_address, &soc_addrlen);
#else
	status = accept (master_socket, (struct sockaddr *)&soc_address, &soc_addrlen);
#endif

	if (status < 0) {
		CLOSE_CONTROL (control);
		control = -1;
		close_master_socket ();
		return -2;
	}

	data_soc = status;
	/* Server has contacted us... send them data */
	/* Send the data! */

	if( (f = fopen( fname, "r")) == NULL) {
	  CLOSE_CONTROL (control);
	  control = -1;
	  close_master_socket ();	  
	  return -1;
	}
		
	HTMeter (0,NULL);
	bDone = 0;
	bLeft = bTotal;
	mo_busy ();
	for (;;) {
	 
	  if (bDone > next_twirl) {
	    intr = HTCheckActiveIcon(1);
    	    next_twirl += twirl_increment;
	  } else {
	    intr = HTCheckActiveIcon(0);
	  }	
	  if (intr) {
	    HTProgress ("Data transfer interrupted");
	    HTMeter (100,NULL);
	    break;
	  }
		
	  if (bLeft < OUTBUFSIZE) { /* Handle last block */
	    if ((chunk = fread (outBuf, 1, bLeft, f)) == 0) 
	      break;	
	    NETWRITE (data_soc, outBuf, chunk);
	    bLeft -= chunk;
	    bDone += chunk;
	  } else if (bLeft <= 0) {  /* Exit */
	    break;		
	  } else {		    /* Handle a block of the data */
	    if ( (chunk = fread (outBuf, 1, OUTBUFSIZE, f)) == 0) 
	      break;
	    NETWRITE (data_soc, outBuf, chunk);
	    bLeft -= chunk;
	    bDone += chunk;
	  }
	  HTMeter ((bDone*100)/bTotal, NULL);
	}	
	
	mo_not_busy ();
	/* Done, now clean up */
	fclose (f);
	HTMeter (100, NULL);

 	CLOSE_CONTROL (control);
	control = -1;
	close_master_socket();

#ifndef DISABLE_TRACE
	if(www2Trace)
	  fprintf (stderr, "HTFTPSend: Closing data socket\n");
#endif
	NETCLOSE (data_soc);
	data_soc = -1;

	if (bLeft != 0)  {
#ifndef DISABLE_TRACE
	  if(www2Trace)
	    fprintf (stderr, "HTFTPSend: Error sending file %lu bytes left\n", bLeft);
#endif
	  return intr?-2:-1;
	}
	
	timer = XtAppAddTimeOut(app_context, ftp_timeout_val*1000, close_it_up, NULL);
	fTimerStarted = 1;
	
#ifndef DISABLE_TRACE
	if(www2Trace)
	  fprintf (stderr, "HTFTPSend: File sent, returning OK\n");
#endif
	return 0;      
} /* End of HTFTPSend */


CLOSE_CONTROL(s)
int s;
{
	NETCLOSE(s);
	ftpcache.control = -1;
}

void
close_it_up()
{
	NETCLOSE(ftpcache.control);
	ftpcache.control = -1;
}

#ifdef NEW_PARSE
/*
 * This code based off of Rick Vestal's FTP parse code for the NCSA Windows
 * Mosaic client.
 *
 * Modified for X by Scott Powers
 * 9.27.95
 */

int ParseFileSizeAndName(char *szBuffer, char *szFileName, char *szSize) {

char *szPtr,*szName,*szEndPtr,*szLength;
static char *tmpbuf=NULL;

	if (!szBuffer) {
		return(0);
	}

	if (!tmpbuf) {
		tmpbuf=(char *)calloc(BUFSIZ,sizeof(char));
	}

	if (usingNLST==1) {
		strcpy(tmpbuf,szBuffer);

		/*filename*/
		szPtr=strrchr(tmpbuf,' ');
		while (szPtr && (*szPtr == ' ')) {
			szPtr--;
		}
		*(szPtr+1)='\0';
		if (szPtr && *szPtr=='>') { /*deal with a link*/
			if (szPtr) {
				szPtr=strrchr(tmpbuf,' ');
				while (szPtr && (*szPtr == ' ')) {
					szPtr--;
				}
				*(szPtr+1)='\0';
			}
			if (szPtr) {
				szPtr=strrchr(tmpbuf,' ');
				while (szPtr && (*szPtr == ' ')) {
					szPtr--;
				}
				*(szPtr+1)='\0';
			}
		}

		if (szPtr) {
			/*year/time*/
			szPtr=strrchr(tmpbuf,' ');
			while (szPtr && (*szPtr == ' ')) {
				szPtr--;
			}
		}
		*(szPtr+1)='\0';

		if (szPtr) {
			/*date*/
			szPtr=strrchr(tmpbuf,' ');
			while (szPtr && (*szPtr == ' ')) {
				szPtr--;
			}
		}
		*(szPtr+1)='\0';

		if (szPtr) {
			/*month*/
			szPtr=strrchr(tmpbuf,' ');
			while (szPtr && (*szPtr == ' ')) {
				szPtr--;
			}
		}
		*(szPtr+1)='\0';

		if (szPtr) {
			/*filesize*/
			szPtr=strrchr(tmpbuf,' ');
		}

		/*beginning of filesize*/
		szPtr++;

/*
		szSize=szBuffer+(szPtr-tmpbuf);
*/
		strcpy(szSize,szPtr);
	}
	else {
		szPtr = strrchr(szBuffer, ' ');
		szName = szPtr + 1;

		if (szPtr) {
			strcpy(szFileName, szName);
		}

		/* go to end of file length */
		while (szPtr && *szPtr == ' ') {
			szPtr--;
		}

		szEndPtr = szPtr+1;
		if (*szPtr != '>') {
			while (szPtr && *szPtr != ' ') {
				szPtr--;
			}
			if (szPtr) {
				szLength = szPtr+1;
				strncpy(szSize, szLength, szEndPtr  - szLength);
				szSize[szEndPtr - szLength] = '\0';
			}
		}
		else {
			return(0);  /* a directory */
		}
	}

	return(1); /* not a directory */
}


int ParseDate(char *szBuffer, char *szFileInfo, char *szMonth, char *szDay, char *szYear, char *szTime) {
	
char *szPtr,*szEndPtr;
int nCount;
char *tmpbuf=(char *)calloc(BUFSIZ,sizeof(char));

	if (!szBuffer) {
		free(tmpbuf);
		return(0);
	}

	if ( (*szBuffer != 'd') && (*szBuffer != 'l') && (*szBuffer != '-')) {
	  	/* Hopefully this is the dos format */

		szPtr = szBuffer;
		strncpy(szMonth, szBuffer, 8);
		szMonth[8] = '\0';

		szPtr = szPtr + 10;
		if (szPtr) {
			strncpy(szTime, szPtr, 7);
			szTime[7] = '\0';
		}

		szPtr = szPtr + 15;
		if (szPtr) {
			if (*szPtr == 'D') {
				*szDay = 'd';
				szDay[1] = '\0';
			}
			else {
				*szDay = 'f';
				szDay[1] = '\0';
			}
		}

		free(tmpbuf);
		return(3); /* ie the info is this dos way */
	}
	else {
		szPtr = NULL;
		nCount = 0;

		/* alright, use this ugly loop to go to each of the month, day, year, whatever parts */
		while (szPtr || ((nCount == 0) && szBuffer)) {
			switch (nCount) {
				case 0:  /* file info */
					strncpy(szFileInfo, szBuffer, 10);
					szFileInfo[10] = '\0';

					if (usingNLST==1) {
						strcpy(tmpbuf,szBuffer);
						/*filename*/
						szPtr=strrchr(tmpbuf,' ');
						while (szPtr && (*szPtr == ' ')) {
							szPtr--;
						}
						*(szPtr+1)='\0';
						if (szPtr && *szPtr=='>') { /*deal with a link*/
							if (szPtr) {
								szPtr=strrchr(tmpbuf,' ');
								while (szPtr && (*szPtr == ' ')) {
									szPtr--;
								}
								*(szPtr+1)='\0';
							}
							if (szPtr) {
								szPtr=strrchr(tmpbuf,' ');
								while (szPtr && (*szPtr == ' ')) {
									szPtr--;
								}
								*(szPtr+1)='\0';
							}
						}

						if (szPtr) {
							/*year/time*/
							szPtr=strrchr(tmpbuf,' ');
							while (szPtr && (*szPtr == ' ')) {
								szPtr--;
							}
						}
						*(szPtr+1)='\0';

						if (szPtr) {
							/*date*/
							szPtr=strrchr(tmpbuf,' ');
							while (szPtr && (*szPtr == ' ')) {
								szPtr--;
							}
						}
						*(szPtr+1)='\0';

						if (szPtr) {
							/*month*/
							szPtr=strrchr(tmpbuf,' ');
						}
						/*beginning of month*/
						szPtr++;

						szPtr=szBuffer+(szPtr-tmpbuf);
					}
					else {
						szPtr = strchr(szBuffer, ' ');
						while (szPtr && (*szPtr == ' ')) {
							szPtr++;
						}

						if (szPtr) {  
							szPtr = strchr(szPtr, ' ');
							while (szPtr && (*szPtr == ' '))  {
								szPtr++;
							}
						}
						if (szPtr) {
							szPtr = strchr(szPtr, ' ');
							while (szPtr && (*szPtr == ' '))  {
								szPtr++;
							}
						}
						if (szPtr) {
							szPtr = strchr(szPtr, ' ');
							while (szPtr && (*szPtr == ' '))  {
								szPtr++;
							}
						}
						if (szPtr) {
							szPtr = strchr(szPtr, ' ');
							while (szPtr && (*szPtr == ' '))  {
								szPtr++;
							}
						}
						/* now we are at the month entry */
					}

					break;

				case 1:
					szEndPtr = strchr(szPtr, ' ');
					if (szEndPtr) {
						strncpy(szMonth, szPtr, szEndPtr - szPtr);
						szMonth[szEndPtr - szPtr] = '\0';
						szPtr = szEndPtr+1;  /* go to next entry (day) */
						while (szPtr && (*szPtr == ' '))  {
								szPtr++;
						}
					}
					else {
						strcpy(szMonth, " ");
					}
					break;	

				case 2:
					szEndPtr = strchr(szPtr, ' ');
					if (szEndPtr) {
						strncpy(szDay, szPtr, szEndPtr - szPtr);
						szDay[szEndPtr - szPtr] = '\0';
						szPtr = szEndPtr+1;  
						while (szPtr && (*szPtr == ' '))  {
								szPtr++;
						}
					}
					else {
						strcpy(szDay, " ");
					}
					break;

				case 3:
					szEndPtr = strchr(szPtr, ' ');
					if (szEndPtr) {
						strncpy(szYear, szPtr, szEndPtr - szPtr);
						szYear[szEndPtr - szPtr] = '\0';
						szPtr = szEndPtr+1;  
					}
					else if (szEndPtr) {
						strcpy(szYear, " ");
					}
					break;

				case 4:
					szPtr = NULL;

			}
			nCount++;

		}

		szPtr = strchr(szYear, ':');
		if (!szPtr) {
			free(tmpbuf);
			return(0);  /* ie the info is month, day, year */
		}

		szPtr -= 2;  /* beginning of time; */

		strncpy(szTime, szPtr, 5);
		szTime[5] = '\0';
	
		free(tmpbuf);
		return(1);  /* ie the info is month, day, time */
	}
}

#endif