ftpd.c
上传用户:xclgwcj
上传日期:2007-04-30
资源大小:6k
文件大小:10k
源码类别:

Ftp服务器

开发平台:

DOS

  1. /*
  2. FTP server example for DSock, copyright (C) 2002 by DM&P.
  3. This is a simple FTP server for demo. You can improve it to support
  4. multi-user and support more commands.
  5. */
  6. #include "..dsock.h"
  7. #include <stdio.h>
  8. #include <dos.h>
  9. #include <string.h>
  10. #include <dir.h>
  11. #include <conio.h>
  12. #include <time.h>
  13. #define FTP_PORT      21
  14. #define FTP_TIME_OUT  30
  15. BOOL  FtpServer(SOCKET s);
  16. int main()
  17. {
  18.   DWORD   dwIp;
  19.   char    szBuf[1024];
  20.   SOCKET  s;
  21.   char    *sz = "-\|/";
  22.   uint    nCnt = 0;
  23.   printf("nDM&P DSock FTP Server Examplen");
  24.   /* Open DSock library */
  25.   if(DSock_Open() == FALSE)
  26.   {
  27.     printf("Unable to initialize socket libraryn");
  28.     return 1;
  29.   }
  30.   /* Load network setup from config file */
  31.   DSock_LoadConfigFile("dsock.cfg");
  32.   /* Show server IP */
  33.   dwIp = DSock_GetHostIp();
  34.   inet_ntoa(szBuf, dwIp);
  35.   printf("FTP Server IP : %sn", szBuf);
  36.   /* Ceate socket for FTP server */
  37.   s = SocketCreate(TCP_SOCKET);
  38.   if(s == INVALID_SOCKET)
  39.   {
  40.     printf("SocketCreate() errorn");
  41.     DSock_Close();
  42.     return 1;
  43.   }
  44.   if(SocketBind(s, 0L, FTP_PORT) == FALSE)
  45.   {
  46.     printf("SocketBind() errorn");
  47.     SocketDestory(s);
  48.     DSock_Close();
  49.     return 1;
  50.   }
  51.   if(SocketListen(s) == FALSE)
  52.   {
  53.     printf("SocketListen() errorn");
  54.     SocketDestory(s);
  55.     DSock_Close();
  56.     return 1;
  57.   }
  58.   /*
  59.   We change FtpServer() to non-blocking function for programmer to check FTP
  60.   connection. You can add HTTP or TELNET server checking routine into checking
  61.   loop to support multiple services.
  62.   */
  63.   while(TRUE)
  64.   {
  65.     printf("%cr", sz[nCnt++ / 500 % 4]);
  66.     if(kbhit())         /* Press any key to break */
  67.       break;
  68.     FtpServer(s);
  69.   }
  70.   /* Destory FTP server socket */
  71.   SocketDestory(s);
  72.   /* Close DSock library */
  73.   DSock_Close();
  74.   return 0;
  75. }
  76. BOOL FtpServer(SOCKET s)
  77. {
  78.   char          szBuf[1024];
  79.   static BOOL     bConnected = FALSE;
  80.   /* Because FtpServer() will immediately return when it process FTP command,
  81.   we should save IP and port in PORT command. */
  82.   static DWORD    dwIp;
  83.   static WORD     wPort;
  84.   static clock_t  clk;  /* For time out checking */
  85.   char          *p, *p2;
  86.   /* Listen when socket is not connected */
  87.   if(bConnected == FALSE)
  88.   {
  89.     if(SocketAccept(s, &dwIp) == FALSE)
  90.       return TRUE;
  91.     else
  92.     {
  93.       SocketPutString(s, "220 DSock FTP Server, service ready.rn");
  94.       inet_ntoa(szBuf, dwIp);
  95.       printf("FTP: Accepting connection from %sn", szBuf);
  96.       bConnected = TRUE;
  97.       clk = clock();
  98.     }
  99.   }
  100.   /* Socket already connected but ... */
  101.   if(SocketIsConnected(s) == FALSE)
  102.   {
  103.     bConnected = FALSE;
  104.     printf("FTP: Remote disconected.n");
  105.     SocketClose(s);
  106.     SocketBind(s, 0L, FTP_PORT);
  107.     SocketListen(s);
  108.     return TRUE;
  109.   }
  110.   /* Is idle time out ? */
  111.   if((clock() - clk) >= 18.2 * FTP_TIME_OUT)
  112.   {
  113.     bConnected = FALSE;
  114.     printf("FTP: User idled %d seconds, disconnect.n", FTP_TIME_OUT);
  115.     SocketClose(s);
  116.     SocketBind(s, 0L, FTP_PORT);
  117.     SocketListen(s);
  118.     return TRUE;
  119.   }
  120.   /* Is any data available ? */
  121.   if(SocketDataReady(s) == FALSE)
  122.     return TRUE;
  123.   /* Reset timer when user send command */
  124.   clk = clock();
  125.   SocketGetString(s, szBuf, 1024);
  126.   /* Debug information */
  127.   printf("FTP: %sn", szBuf);
  128.   p = szBuf;
  129.   while(*p != ' ' && *p != NULL)
  130.     p++;
  131.   if(p)
  132.   {
  133.     *p = NULL;
  134.     p2 = p + 1;
  135.   }
  136.   /* p2 point to the second parameter */
  137.   /*** Process FTP commands ***/
  138.   /* You should save user name for password checking. We accept all user
  139.   name here. */
  140.   if(stricmp("user", szBuf) == 0)
  141.   {
  142.     SocketPutString(s, "331 Password required for %s.rn", p2);
  143.   }
  144.   /* You can load user names and passwords from password file */
  145.   else if(stricmp("pass", szBuf) == 0)
  146.   {
  147.     if(stricmp("dmp", p2) != 0) /* default password for all user */
  148.     {
  149.       SocketPutString(s, "530 Login incorrect.rn");
  150.       SocketClose(s);
  151.       SocketBind(s, 0L, FTP_PORT);
  152.       SocketListen(s);
  153.       bConnected = FALSE;
  154.     }
  155.     SocketPutString(s, "230 Logged in okay.rn");
  156.   }
  157. #define MakeLong(high, low) 
  158.     ((long)(((WORD) (low)) | (((DWORD) ((WORD) (high))) << 16)))
  159. #define MakeWord(high, low) 
  160.     ((WORD) (((BYTE) (low)) | (((WORD) ((BYTE) (high))) << 8)))
  161.   /* Port command for data connection. */
  162.   else if(stricmp("port", szBuf) == 0)
  163.   {
  164.     char  c[6];
  165.     int   i[2];
  166.     sscanf(p2, "%d,%d,%d,%d,%d,%d", &c[0], &c[1], &c[2], &c[3], &c[4], &c[5]);
  167.     i[0] = MakeWord(c[0], c[1]);
  168.     i[1] = MakeWord(c[2], c[3]);
  169.     wPort = MakeWord(c[4], c[5]);
  170.     dwIp = MakeLong(i[0], i[1]);
  171.     SocketPutString(s, "200 PORT command successful.rn");
  172.   }
  173.   /* We use binary mode for all commands */
  174.   else if(stricmp("type", szBuf) == 0)
  175.   {
  176.     SocketPutString(s, "200 Type set to %s.rn", p2);
  177.   }
  178.   /* Client uplaod file */
  179.   else if(stricmp("stor", szBuf) == 0)
  180.   {
  181.     SOCKET  s2;
  182.     FILE    *fp = fopen(p2, "wb");
  183.     SocketPutString(s, "150 Opening BINARY mode data connection.rn");
  184.     s2 = SocketCreate(TCP_SOCKET);
  185.     SocketConnect(s2, dwIp, wPort);
  186.     if(fp == NULL)
  187.     {
  188.       SocketPutString(s, "550 Unable to create file.rn");
  189.     }
  190.     else
  191.     {
  192.       long  lSize = 0;
  193.       int   nLen;
  194.       while(TRUE)
  195.       {
  196.         nLen = SocketRecv(s2, (uchar *)szBuf, 1024);
  197.         lSize += nLen;
  198.         printf("FTP: %lu bytes receivedr", lSize);
  199.         fwrite(szBuf, nLen, 1, fp);
  200.         if(nLen < 1024)
  201.           break;
  202.       }
  203.       printf("n");
  204.       fclose(fp);
  205.       SocketPutString(s, "226 Transfer complete.rn");
  206.     }
  207.     SocketClose(s2);
  208.     SocketDestory(s2);
  209.   }
  210.   /* Client download file */
  211.   else if(stricmp("retr", szBuf) == 0)
  212.   {
  213.     SOCKET  s2;
  214.     FILE    *fp = fopen(p2, "rb");
  215.     SocketPutString(s, "150 Opening BINARY mode data connection.rn");
  216.     s2 = SocketCreate(TCP_SOCKET);
  217.     SocketConnect(s2, dwIp, wPort);
  218.     if(fp == NULL)
  219.     {
  220.       SocketPutString(s, "550 No such file or directory.rn");
  221.     }
  222.     else
  223.     {
  224.       long  lSize = 0;
  225.       int   nLen;
  226.       while(TRUE)
  227.       {
  228.         nLen = fread(szBuf, 1, 1024, fp);
  229.         if(SocketSend(s2, (uchar *)szBuf, nLen) == 0)
  230.         {
  231.           printf("Unable to send data.");
  232.           break;
  233.         }
  234.         /* error handle is ignored */
  235.         lSize += nLen;
  236.         printf("FTP: %lu bytes sentr", lSize);
  237.         if(nLen < 1024)
  238.           break;
  239.       }
  240.       printf("n");
  241.       fclose(fp);
  242.       SocketPutString(s, "226 Transfer complete.rn");
  243.     }
  244.     SocketClose(s2);
  245.     SocketDestory(s2);
  246.   }
  247.   /* Change work directory */
  248.   else if(stricmp("cwd", szBuf) == 0)
  249.   {
  250.     if(chdir(p2) != 0)
  251.       SocketPutString(s, "550 No such file or directory.rn");
  252.     else
  253.       SocketPutString(s, "250 CWD command sucessful.rn");
  254.   }
  255.   /* Make directory */
  256.   else if(stricmp("mkd", szBuf) == 0 || stricmp("xmkd", szBuf) == 0)
  257.   {
  258.     if(mkdir(p2) != 0)
  259.       SocketPutString(s, "550 Path or file function not found.rn");
  260.     else
  261.       SocketPutString(s, "257 New directory created.rn");
  262.   }
  263.   /* Remove direcory */
  264.   else if(stricmp("rmd", szBuf) == 0 || stricmp("xrmd", szBuf) == 0)
  265.   {
  266.     if(rmdir(p2) != 0)
  267.       SocketPutString(s, "550 Path or file not found.rn");
  268.     else
  269.       SocketPutString(s, "257 Command successful.rn");
  270.   }
  271.   /* Delete file */
  272.   else if(stricmp("dele", szBuf) == 0)
  273.   {
  274.     if(remove(p2) != 0)
  275.       SocketPutString(s, "550 No such file or directory.rn");
  276.     else
  277.       SocketPutString(s, "257 Command successful.rn");
  278.   }
  279.   else if(stricmp("cdup", szBuf) == 0)
  280.   {
  281.     if(chdir("..") != 0)
  282.       SocketPutString(s, "550 No such file or directory.rn");
  283.     else
  284.       SocketPutString(s, "250 CDUP command sucessful.rn");
  285.   }
  286.   /* Print work directory */
  287.   else if(stricmp("pwd", szBuf) == 0 || stricmp("xpwd", szBuf) == 0)
  288.   {
  289.     char  szBuf2[256];
  290.     getcwd(szBuf2, 256);
  291.     p = szBuf2;
  292.     while(*p)
  293.     {
  294.       if(*p == '\')
  295.         *p = '/';
  296.       p++;
  297.     }
  298.     p = szBuf2;
  299.     while(*p != ':')
  300.       p++;
  301.     sprintf(szBuf, "257 "%s" is current directory.rn", p + 1);
  302.     SocketPutString(s, szBuf);
  303.   }
  304.   /* Directory list */
  305.   else if(stricmp("list", szBuf) == 0 || stricmp("nlst", szBuf) == 0)
  306.   {
  307.     SOCKET        s2;
  308.     struct ffblk  ffblk;
  309.     int          nCode;
  310.     char          *szAttr = "-rwxr--r--";
  311.     static char   *szMonth[] =
  312.     {
  313.       "Jan",
  314.       "Feb",
  315.       "Mar",
  316.       "Apr",
  317.       "May",
  318.       "Jun",
  319.       "Jul",
  320.       "Aug",
  321.       "Sep",
  322.       "Oct",
  323.       "Nov",
  324.       "Dec",
  325.     };
  326.     SocketPutString(s,
  327.                         "150 Opening ASCII mode data connection for directory listing.rn");
  328.     s2 = SocketCreate(TCP_SOCKET);
  329.     SocketConnect(s2, dwIp, wPort);
  330.     nCode = findfirst("*.*", &ffblk, 0xff);
  331.     while(!nCode)
  332.     {
  333.       szAttr[0] = ffblk.ff_attrib & FA_DIREC ? 'd' : '-';
  334.       SocketPutString(s2, "%10s   1 501      501    %10lu %3s %2d  %4d %srn",
  335.                             szAttr, ffblk.ff_fsize,
  336.                             szMonth[((ffblk.ff_fdate & 0x01e0) >> 5) - 1],
  337.                             ffblk.ff_fdate & 0x001f, 1980 + (ffblk.ff_fdate >> 9),
  338.                             ffblk.ff_name);
  339.       nCode = findnext(&ffblk);
  340.     }
  341.     SocketClose(s2);
  342.     SocketDestory(s2);
  343.     SocketPutString(s, "226 Transfer complete.rn");
  344.   }
  345.   else if(stricmp("quit", szBuf) == 0)
  346.   {
  347.     SocketPutString(s, "221 Bye!rn");
  348.     SocketClose(s);
  349.     SocketBind(s, 0L, FTP_PORT);
  350.     SocketListen(s);
  351.     bConnected = FALSE;
  352.   }
  353.   else
  354.   {
  355.     SocketPutString(s, "500 Command not understood.rn");
  356.   }
  357.   return TRUE;
  358. }