Annotation of embedaddon/php/ext/ftp/tests/server.inc, revision 1.1

1.1     ! misho       1: <?php
        !             2: 
        !             3: $socket = null;
        !             4: $errno = 0;
        !             5: $context = stream_context_create(array('ssl' => array('local_cert' => dirname(__FILE__).'/cert.pem', 'passphrase' => 'pass')));
        !             6: 
        !             7: for ($i=0; $i<10 && !$socket; ++$i) {
        !             8:        $port = rand(50000, 65535);
        !             9:        
        !            10:        $socket = stream_socket_server("tcp://127.0.0.1:$port", $errno, $errstr, STREAM_SERVER_BIND|STREAM_SERVER_LISTEN, $context);
        !            11: }
        !            12: //set anther random port that is not the same as $port
        !            13: do{
        !            14:        $pasv_port = rand(50000, 65535);
        !            15: }while($pasv_port == $port);
        !            16: 
        !            17: if (!$socket) {
        !            18:        die("could not start/bind the ftp server\n");
        !            19: }
        !            20: 
        !            21: 
        !            22: 
        !            23: 
        !            24: $pid = pcntl_fork();
        !            25: 
        !            26: 
        !            27: 
        !            28: function pasv_listen($action){
        !            29:        global $pasv_port, $tmp_file;
        !            30:        $tmp_file = 'nm2.php';
        !            31:        $pid = pcntl_fork();
        !            32:        if($pid === 0){
        !            33:                $soc  = stream_socket_server("tcp://127.0.0.1:$pasv_port");
        !            34:                $fs = stream_socket_accept($soc, 3);
        !            35:                switch ($action) {
        !            36:                        case 'fget':
        !            37:                        case 'get':
        !            38:                        //listen for 3 seconds 3 seconds
        !            39:                        fputs($fs, "I am passive.\r\n");
        !            40:                        break;
        !            41:                        case 'put':
        !            42:                        file_put_contents($tmp_file,  stream_get_contents($fs));
        !            43:                        break;
        !            44:                        case 'list':
        !            45:                        fputs($fs, "drwxr-x---   3 owner  group      4096 Jul 12 12:16 .\r\n");
        !            46:                        fputs($fs, "drwxr-x---   3 owner  group      4096 Jul 12 12:16 ..\r\n");
        !            47:                        fputs($fs, "drwxr-x---   3 owner  group      4096 Jul 12 12:16 public_ftp\r\n");
        !            48:                        break;
        !            49:                        case 'list_null':
        !            50:                        fputs($fs, "\r\n");
        !            51:                        break;
        !            52:                }
        !            53:                fclose($fs);
        !            54:                exit;
        !            55:        }
        !            56: }
        !            57: 
        !            58: 
        !            59: 
        !            60: if ($pid) {
        !            61: 
        !            62:        function dump_and_exit($buf)
        !            63:        {
        !            64:                var_dump($buf);
        !            65:                fclose($GLOBALS['s']);
        !            66:                exit;
        !            67:        }
        !            68: 
        !            69:        function anonymous()
        !            70:        {
        !            71:                return $GLOBALS['user'] === 'anonymous';
        !            72:        }
        !            73: 
        !            74:        /* quick&dirty realpath() like function */
        !            75:        function change_dir($dir)
        !            76:        {
        !            77:                global $cwd;
        !            78: 
        !            79:                if ($dir[0] == '/') {
        !            80:                        $cwd = $dir;
        !            81:                        return;
        !            82:                }
        !            83: 
        !            84:                $cwd = "$cwd/$dir";
        !            85: 
        !            86:                do {
        !            87:                        $old = $cwd;
        !            88:                        $cwd = preg_replace('@/?[^/]+/\.\.@', '', $cwd);
        !            89:                } while ($old != $cwd);
        !            90: 
        !            91:                $cwd = strtr($cwd, array('//' => '/'));
        !            92:                if (!$cwd) $cwd = '/';
        !            93:        }
        !            94: 
        !            95:        $s = stream_socket_accept($socket);
        !            96: 
        !            97:        if (!$s) die("Error accepting a new connection\n");
        !            98: 
        !            99:        fputs($s, "220----- PHP FTP server 0.3 -----\r\n220 Service ready\r\n");
        !           100:        $buf = fread($s, 2048);
        !           101: 
        !           102: 
        !           103:        function user_auth($buf) {
        !           104:                global $user, $s, $ssl, $bug37799;
        !           105: 
        !           106:                if (!empty($ssl)) {
        !           107:                        if ($buf !== "AUTH TLS\r\n") {
        !           108:                                fputs($s, "500 Syntax error, command unrecognized.\r\n");
        !           109:                                dump_and_exit($buf);
        !           110:                        }
        !           111: 
        !           112:                        if (empty($bug37799)) {
        !           113:                                fputs($s, "234 auth type accepted\r\n");
        !           114:                        } else {
        !           115:                                fputs($s, "666 dummy\r\n");
        !           116:                                fputs($s, "666 bogus msg\r\n");
        !           117:                                exit;
        !           118:                        }
        !           119: 
        !           120:                        if (!stream_socket_enable_crypto($s, true, STREAM_CRYPTO_METHOD_SSLv23_SERVER)) {
        !           121:                                die("SSLv23 handshake failed.\n");
        !           122:                        }
        !           123: 
        !           124:                        if (!preg_match('/^PBSZ \d+\r\n$/', $buf = fread($s, 2048))) {
        !           125:                                fputs($s, "501 bogus data\r\n");
        !           126:                                dump_and_exit($buf);
        !           127:                        }
        !           128: 
        !           129:                        fputs($s, "200 OK\r\n");
        !           130:                        $buf = fread($s, 2048);
        !           131: 
        !           132:                        if ($buf !== "PROT P\r\n") {
        !           133:                                fputs($s, "504 Wrong protection.\r\n");
        !           134:                                dump_and_exit($buf);
        !           135:                        }
        !           136: 
        !           137:                        fputs($s, "200 OK\r\n");
        !           138: 
        !           139:                        $buf = fread($s, 2048);
        !           140:                }
        !           141: 
        !           142:                if (!preg_match('/^USER (\w+)\r\n$/', $buf, $m)) {
        !           143:                        fputs($s, "500 Syntax error, command unrecognized.\r\n");
        !           144:                        dump_and_exit($buf);
        !           145:                }
        !           146:                $user = $m[1];
        !           147:                if ($user !== 'user' && $user !== 'anonymous') {
        !           148:                        fputs($s, "530 Not logged in.\r\n");
        !           149:                        fclose($s);
        !           150:                        exit;
        !           151:                }
        !           152: 
        !           153:                if (anonymous()) {
        !           154:                        fputs($s, "230 Anonymous user logged in\r\n");
        !           155: 
        !           156:                } else {
        !           157:                        fputs($s, "331 User name ok, need password\r\n");
        !           158: 
        !           159:                        if (!preg_match('/^PASS (\w+)\r\n$/', $buf = fread($s, 100), $m)) {
        !           160:                                fputs($s, "500 Syntax error, command unrecognized.\r\n");
        !           161:                                dump_and_exit($buf);
        !           162:                        }
        !           163: 
        !           164:                        $pass = $m[1];
        !           165:                        if ($pass === 'pass') {
        !           166:                                fputs($s, "230 User logged in\r\n");
        !           167:                        } else {
        !           168:                                fputs($s, "530 Not logged in.\r\n");
        !           169:                                fclose($s);
        !           170:                                exit;
        !           171:                        }
        !           172:                }
        !           173:        }
        !           174: 
        !           175:        user_auth($buf);
        !           176: 
        !           177:        $cwd = '/';
        !           178:        $num_bogus_cmds = 0;
        !           179: 
        !           180:        while($buf = fread($s, 4098)) {
        !           181:                if (!empty($bogus)) {
        !           182:                        fputs($s, "502 Command not implemented (".$num_bogus_cmds++.").\r\n");
        !           183: 
        !           184:                } else if ($buf === "HELP\r\n") {
        !           185:                        fputs($s, "214-There is help available for the following commands:\r\n");
        !           186:                        fputs($s, " USER\r\n");
        !           187:                        fputs($s, " HELP\r\n");
        !           188:                        fputs($s, "214 end of list\r\n");
        !           189: 
        !           190:                } elseif ($buf === "HELP HELP\r\n") {
        !           191:                        fputs($s, "214 Syntax: HELP [<SP> <string>] <CRLF>\r\n");
        !           192: 
        !           193:                } elseif ($buf === "PWD\r\n") {
        !           194:                        fputs($s, "257 \"$cwd\" is current directory.\r\n");
        !           195: 
        !           196:                } elseif ($buf === "CDUP\r\n") {
        !           197:                        change_dir('..');
        !           198:                        fputs($s, "250 CDUP command successful.\r\n");
        !           199: 
        !           200:                } elseif ($buf === "SYST\r\n") {
        !           201:                        if (isset($bug27809)) {
        !           202:                                fputs($s, "215   OS/400 is the remote operating system. The TCP/IP version is \"V5R2M0\"\r\n");
        !           203:                        } else {
        !           204:                                fputs($s, "215 UNIX Type: L8.\r\n");
        !           205:                        }
        !           206: 
        !           207:                } elseif ($buf === "TYPE A\r\n") {
        !           208:                        $ascii = true;
        !           209:                        fputs($s, "200 OK\r\n");
        !           210: 
        !           211:                } elseif ($buf === "TYPE I\r\n") {
        !           212:                        $ascii = false;
        !           213:                        fputs($s, "200 OK\r\n");
        !           214: 
        !           215:                } elseif ($buf === "QUIT\r\n") {
        !           216:                        break;
        !           217: 
        !           218:                } elseif (preg_match("~^PORT (\d+),(\d+),(\d+),(\d+),(\d+),(\d+)\r\n$~", $buf, $m)) {
        !           219:                        $host = "$m[1].$m[2].$m[3].$m[4]";
        !           220:                        $port = ((int)$m[5] << 8) + (int)$m[6];
        !           221:                        fputs($s, "200 OK.\r\n");
        !           222: 
        !           223:                } elseif (preg_match("~^STOR ([\w/.-]+)\r\n$~", $buf, $m)) {
        !           224:                        fputs($s, "150 File status okay; about to open data connection\r\n");
        !           225: 
        !           226:                        if(empty($pasv))
        !           227:                        {
        !           228:                                if (!$fs = stream_socket_client("tcp://$host:$port")) {
        !           229:                                        fputs($s, "425 Can't open data connection\r\n");
        !           230:                                        continue;
        !           231:                                }
        !           232: 
        !           233:                                $data = stream_get_contents($fs);
        !           234:                                $orig = file_get_contents(dirname(__FILE__).'/'.$m[1]);
        !           235: 
        !           236: 
        !           237:                                if (isset($ascii) && !$ascii && $orig === $data) {
        !           238:                                        fputs($s, "226 Closing data Connection.\r\n");
        !           239: 
        !           240:                                } elseif ((!empty($ascii) || isset($bug39583)) && $data === strtr($orig, array("\r\n" => "\n", "\r" => "\n", "\n" => "\r\n"))) {
        !           241:                                        fputs($s, "226 Closing data Connection.\r\n");
        !           242: 
        !           243:                                } else {
        !           244:                                        var_dump($data);
        !           245:                                        var_dump($orig);
        !           246:                                        fputs($s, "552 Requested file action aborted.\r\n");
        !           247:                                }
        !           248:                                fclose($fs);
        !           249:                        }else{
        !           250:                                $data = file_get_contents('nm2.php');
        !           251:                                $orig = file_get_contents(dirname(__FILE__).'/'.$m[1]);
        !           252:                                if ( $orig === $data) {
        !           253:                                        fputs($s, "226 Closing data Connection.\r\n");
        !           254: 
        !           255:                                } else {
        !           256:                                        var_dump($data);
        !           257:                                        var_dump($orig);
        !           258:                                        fputs($s, "552 Requested file action aborted.\r\n");
        !           259:                                }
        !           260:                        }
        !           261: 
        !           262:                } elseif (preg_match("~^CWD ([A-Za-z./]+)\r\n$~", $buf, $m)) {
        !           263:                        change_dir($m[1]);
        !           264:                        fputs($s, "250 CWD command successful.\r\n");
        !           265: 
        !           266:                } elseif (preg_match("~^NLST(?: ([A-Za-z./]+))?\r\n$~", $buf, $m)) {
        !           267: 
        !           268:                        if (isset($m[1]) && $m[1] === 'bogusdir') {
        !           269:                                fputs($s, "250 $m[1]: No such file or directory\r\n");
        !           270:                                continue;
        !           271:                        }
        !           272: 
        !           273:                        // there are some servers that don't open the ftp-data socket if there's nothing to send
        !           274:                        if (isset($bug39458) && isset($m[1]) && $m[1] === 'emptydir') {
        !           275:                                fputs($s, "226 Transfer complete.\r\n");
        !           276:                                continue;
        !           277:                        }
        !           278: 
        !           279:                        fputs($s, "150 File status okay; about to open data connection\r\n");
        !           280: 
        !           281:                        if (!$fs = stream_socket_client("tcp://$host:$port")) {
        !           282:                                fputs($s, "425 Can't open data connection\r\n");
        !           283:                                continue;
        !           284:                        }
        !           285: 
        !           286:                        if (empty($m[1]) || $m[1] !== 'emptydir') {
        !           287:                                fputs($fs, "file1\r\nfile1\r\nfile\nb0rk\r\n");
        !           288:                        }
        !           289: 
        !           290:                        fputs($s, "226 Closing data Connection.\r\n");
        !           291:                        fclose($fs);
        !           292: 
        !           293:                } elseif (preg_match("~^MKD ([A-Za-z./]+)\r\n$~", $buf, $m)) {
        !           294:                        if (isset($bug7216)) {
        !           295:                                fputs($s, "257 OK.\r\n");
        !           296:                        } else {
        !           297:                                fputs($s, "257 \"/path/to/ftproot$cwd$m[1]\" created.\r\n");
        !           298:                        }
        !           299: 
        !           300:                } elseif (preg_match('/^USER /', $buf)) {
        !           301:                        user_auth($buf);
        !           302: 
        !           303:                } elseif (preg_match('/^MDTM ([\w\h]+)/', $buf, $matches)) {
        !           304:                        switch ($matches [1]){
        !           305:                                case "A":
        !           306:                                fputs($s, "213 19980615100045.014\r\n");
        !           307:                                break;
        !           308:                                case "B":
        !           309:                                fputs($s, "213 19980615100045.014\r\n");
        !           310:                                break;
        !           311:                                case "C":
        !           312:                                fputs($s, "213 19980705132316\r\n");
        !           313:                                break;
        !           314:                                case "19990929043300 File6":
        !           315:                                fputs($s, "213 19991005213102\r\n");
        !           316:                                break;
        !           317:                                default :
        !           318:                                fputs($s, "550 No file named \"{$matches [1]}\"\r\n");
        !           319:                                break;
        !           320:                        }
        !           321:                }elseif (preg_match('/^RETR ([\w\h]+)/', $buf, $matches)) {
        !           322:                        if(!empty($pasv)){
        !           323:                                ;
        !           324:                        }
        !           325:                        else if (!$fs = stream_socket_client("tcp://$host:$port")) {
        !           326:                                fputs($s, "425 Can't open data connection\r\n");
        !           327:                                continue;
        !           328:                        }
        !           329: 
        !           330:                        switch($matches[1]){
        !           331: 
        !           332:                                case "pasv":
        !           333:                                        fputs($s, "150 File status okay; about to open data connection.\r\n");
        !           334:                                        //the data connection is handled in another forked process
        !           335:                                        // called from outside this while loop
        !           336:                                        fputs($s, "226 Closing data Connection.\r\n");
        !           337:                                        break;
        !           338:                                case "a story":
        !           339:                                        fputs($s, "150 File status okay; about to open data connection.\r\n");
        !           340:                                        fputs($fs, "For sale: baby shoes, never worn.\r\n");
        !           341:                                        fputs($s, "226 Closing data Connection.\r\n");
        !           342:                                        break;
        !           343:                                case "binary data":
        !           344:                                        fputs($s, "150 File status okay; about to open data connection.\r\n");
        !           345:                                        $transfer_type = $ascii? 'ASCII' : 'BINARY' ;
        !           346:                                        fputs($fs, $transfer_type."Foo\0Bar\r\n");
        !           347:                                        fputs($s, "226 Closing data Connection.\r\n");
        !           348:                                        break;
        !           349:                                case "fget":
        !           350:                                        fputs($s, "150 File status okay; about to open data connection.\r\n");
        !           351:                                        $transfer_type = $ascii? 'ASCII' : 'BINARY' ;
        !           352:                                        fputs($fs, $transfer_type."FooBar\r\n");
        !           353:                                        fputs($s, "226 Closing data Connection.\r\n");
        !           354:                                        break;
        !           355:                                case "fgetresume":
        !           356:                                        fputs($s, "150 File status okay; about to open data connection.\r\n");
        !           357:                                        $transfer_type = $ascii? 'ASCII' : 'BINARY' ;
        !           358:                                        fputs($fs, "Bar\r\n");
        !           359:                                        fputs($s, "226 Closing data Connection.\r\n");
        !           360:                                        break;
        !           361:                                default:
        !           362:                                        fputs($s, "550 {$matches[1]}: No such file or directory \r\n");
        !           363:                                        break;
        !           364:                        }
        !           365:                        if(isset($fs))
        !           366:                                fclose($fs);
        !           367: 
        !           368: 
        !           369:                }elseif (preg_match('/^PASV/', $buf, $matches)) {
        !           370:                        $port = $pasv_port;
        !           371:                        $p2 = $port % ((int) 1 << 8);
        !           372:                        $p1 = ($port-$p2)/((int) 1 << 8);
        !           373:                        $host = "127.0.0.1";
        !           374:                        fputs($s, "227 Entering Passive Mode. (127,0,0,1,{$p1},{$p2})\r\n");
        !           375: 
        !           376: 
        !           377:                } elseif (preg_match('/^SITE EXEC/', $buf, $matches)) {
        !           378:                        fputs($s, "200 OK\r\n");
        !           379: 
        !           380:                } elseif (preg_match('/^RMD/', $buf, $matches)) {
        !           381:                        fputs($s, "250 OK\r\n");
        !           382: 
        !           383:                } elseif (preg_match('/^SITE CHMOD/', $buf, $matches)) {
        !           384:                        fputs($s, "200 OK\r\n");
        !           385: 
        !           386:                } elseif (preg_match('/^ALLO (\d+)/', $buf, $matches)) {
        !           387:                        fputs($s, "200 " . $matches[1] . " bytes allocated\r\n");
        !           388: 
        !           389:                }elseif (preg_match('/^LIST www\//', $buf, $matches)) {
        !           390:                        fputs($s, "150 Opening ASCII mode data connection for file list\r\n");
        !           391:                        fputs($s, "226 Transfer complete\r\n");
        !           392: 
        !           393:                }elseif (preg_match('/^LIST no_exists\//', $buf, $matches)) {
        !           394:                        fputs($s, "425 Error establishing connection\r\n");
        !           395: 
        !           396:                }elseif (preg_match('/^REST \d+/', $buf, $matches)) {
        !           397:                        fputs($s, "350 OK\r\n");
        !           398:                }
        !           399: 
        !           400:                else {
        !           401:                        fputs($s, "500 Syntax error, command unrecognized.\r\n");
        !           402:                        dump_and_exit($buf);
        !           403:                }
        !           404:        }
        !           405:        fclose($s);
        !           406:        exit;
        !           407: }
        !           408: 
        !           409: fclose($socket);
        !           410: ?>

FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>