为什么CF FTP传输速度比标准FTP慢数倍?

为什么CF FTP传输速度比标准FTP慢数倍?,ftp,coldfusion-9,windows-server,filezilla,cfftp,Ftp,Coldfusion 9,Windows Server,Filezilla,Cfftp,我有一个ColdFusion应用程序,用于在开发服务器和生产服务器之间传输文件。实际发送文件的代码如下所示: ftp = new Ftp(); ftp.setUsername(username); ftp.setPassword(password); ftp.setServer(server); ftp.setTimeout(1800); ftp.setConnection('dev'); ftp.open(); ftp.putFile(transferMode="binary",localFi

我有一个ColdFusion应用程序,用于在开发服务器和生产服务器之间传输文件。实际发送文件的代码如下所示:

ftp = new Ftp();
ftp.setUsername(username);
ftp.setPassword(password);
ftp.setServer(server);
ftp.setTimeout(1800);
ftp.setConnection('dev');
ftp.open();
ftp.putFile(transferMode="binary",localFile=localpath,remoteFile=remotepath);
ftp.close(connection='dev');
当使用上述代码发送文件时,我的速度限制在100KB/s以下(通过接收服务器上的FileZilla进行监控)。如果我使用Windows命令行FTP工具发送完全相同的文件,我的速度将超过1000KB/s

我创建了一个全新的文件,除了上面的代码之外什么都没有,这对传输速度没有影响,所以我知道它与原始应用程序中的周围代码没有任何关系

那么,是什么导致了这些极低的速度呢

编辑:正在完成所有测试,将文件从生产服务器传输到开发服务器。我还尝试使用
标记而不是cfscript,我得到了相同的结果

Edit#2:我最终使用了
cfexecute
,代码如下:

ftp = new Ftp();
ftp.setUsername(username);
ftp.setPassword(password);
ftp.setServer(server);
ftp.setTimeout(1800);
ftp.setConnection('dev');
ftp.open();
ftp.putFile(transferMode="binary",localFile=localpath,remoteFile=remotepath);
ftp.close(connection='dev');
从我的FTP脚本:

public function sendFiles(required string localpath, required string remotepath) {
    this.writeFtpInstructions(localpath);
    exe = "C:\Windows\system32\ftp.exe";
    params = "-s:" & request.localapproot & "/" & "upload.txt";
    outputfile = request.localapproot & '/ftp.log';
    timeout = 120;
    Request.cliExec(exe,params,outputfile,timeout);
}

public function writeFtpInstructions(required string localpath) {
    instructions = request.localapproot & "/" & "upload.txt";
    crlf = chr(13) & chr(10);
    data = "";
    data &= "open " & this.server & crlf;
    data &= this.username & crlf;
    data &= this.password & crlf;
    data &= "cd " & request.remoteapproot & crlf;
    data &= "put " & localpath & crlf;
    data &= "quit";
    FileWrite(instructions, data);
}
cliExec()
函数(创建包装器所必需的,因为在cfscript中没有与
cfexecute
等效的函数):



我一直在寻找,但我不知道为什么速度较慢。但是,从理论上讲,您应该能够通过windows命令行使用cfexecute来实现这一点。您甚至可以创建一个批处理文件并在一次调用中完成。

根据我在CF9上使用cfftp的经验,不可能传输更大的文件。如果您在FTP服务器端查看活动传输,您会注意到传输以最高速度开始,但随着传输的数据越来越多,传输速度不断下降。在转移了100MB之后,减少的幅度开始变得非常剧烈,直到它们最终达到一位数的爬网。最终传输超时并失败。我试图处理一个330MB的文件&发现使用cfftp传输是不可能的。cfexecute不是我使用标准windows ftp.exe的选项,因为它似乎不支持SFTP

我最终通过coldfusion找到了一个外部java类,并决定使用JSch(http://www.jcraft.com/jsch/). 具有讽刺意味的是,CF9似乎对CFFTP使用了这个类的一个变体(jsch-0.1.41m.jar),但是使用这个最新下载的版本(jsch-0.1.45.jar)的结果却大不相同

以下是我为概念验证而编写的代码:

<cfscript>
    stAppPrefs = {
        stISOFtp = {
             server = 'sftp.server.com',
             port = '22',
             username = 'youser',
             password = 'pa$$w0rd'
        }
    };

    /* Side-Load JSch Java Class (http://www.jcraft.com/jsch/) */
    try {
        // Load Class Using Mark Mandel's JavaLoader (http://www.compoundtheory.com/?action=javaloader.index)
        /* 
        Add Mark's LoaderClass To The ColdFusion Class Path Under CF Admin:
        Java and JVM : ColdFusion Class Path : C:\inetpub\wwwroot\javaloader\lib\classloader-20100119110136.jar
        Then Restart The Coldfusion Application Service
        */
        loader = CreateObject("component", "javaloader.JavaLoader").init([expandPath("jsch-0.1.45.jar")]);
        // Initiate Instance
        jsch = loader.create("com.jcraft.jsch.JSch").init();
    }
    catch(any excpt) {          
        WriteOutput("Error loading ""jsch-0.1.45.jar"" java class: " & excpt.Message & "<br>");
        abort;
    }

    /* SFTP Session & Channel */
    try {
        // Create SFTP Session
        session = jsch.getSession(stAppPrefs.stISOFtp.username, stAppPrefs.stISOFtp.server, stAppPrefs.stISOFtp.port);
        // Turn Off & Use Username/Password
        session.setConfig("StrictHostKeyChecking", "no");
        session.setPassword(stAppPrefs.stISOFtp.password);
        session.connect();
        // Create Channel To Transfer File(s) On
        channel = session.openChannel("sftp");
        channel.connect();
    }
    catch(any excpt) { 
        WriteOutput("Error connecting to FTP server: " & excpt.Message & "<br>");
        abort;
    } 

    // Returns Array Of Java Objects, One For Each Zip Compressed File Listed In Root DIR
    // WriteDump(channel.ls('*.zip'));      
    // Get First Zip File Listed For Transfer From SFTP To CF Server
    serverFile = channel.ls('*.zip')[1].getFilename();

    /* Debug */
    startTime = Now();
    WriteOutput("Transfer Started: " & TimeFormat(startTime, 'hh:mm:ss') & "<br>");
    /* // Debug */

    /* Transfer File From SFTP Server To CF Server */
    try {
        // Create File On Server To Write Byte Stream To
        transferFile = CreateObject("java", "java.io.File").init(expandPath(serverFile));
        channel.get(serverFile, CreateObject("java", "java.io.FileOutputStream").init(transferFile));
        // Close The File Output Stream
        transferFile = '';
    }
    catch(any excpt) { 
        WriteOutput("Error transfering file """ & expandPath(serverFile) & """: " & excpt.Message & "<br>");
        abort;
    }

    /* Debug */
    finishTime = Now();
    WriteOutput("Transfer Finished: " & TimeFormat(finishTime, 'hh:mm:ss') & "<br>");
    expiredTime = (finishTime - startTime);
    WriteOutput("Duration: " & TimeFormat(expiredTime, 'HH:MM:SS') & "<br>");
    WriteOutput("File Size: " & NumberFormat(Evaluate(GetFileInfo(ExpandPath(serverFile)).size / 1024), '_,___._') & " KB<br>");
    WriteOutput("Transfer Rate: " & NumberFormat(Evaluate(Evaluate(GetFileInfo(ExpandPath(serverFile)).size / 1024) / Evaluate(((TimeFormat(expiredTime, 'H') * 60 * 60) + (TimeFormat(expiredTime, 'M') * 60) + TimeFormat(expiredTime, 'S')))), '_,___._') & " KB/Sec <br>");
    /* // Debug */

    channel.disconnect();
    session.disconnect();
    </cfscript>

实现的传输速度与我使用FielZILFTP客户端和手动下载的速度一致。我发现这种方法是解决cfftp不足的可行方法。

您的测试使用的是与您的CF脚本运行在同一台服务器上的windows命令行ftp工具吗?是,我所有的测试都从生产服务器转移到开发服务器。我只想说非常感谢你发布这个问题和你的解决方案。通过同样的步骤,我节省了很多时间。:)我也是这么想的。在我接受答案之前,我会让这个问题再拖一段时间,但如果我最终实现了
cfexecute
,我会给你评分。:)听起来很公平。希望你能找到一个更好的解决方案。看来这个解决方案行不通。我将继续执行
cfexecute
。谢谢只是为了跟进-我将我的
cfftp
调用替换为
cfexecute
,现在传输速度一直是1.3MB/s,而不是低于100KB/s。这样一个奇怪的问题…你是用批处理脚本还是几个后续的cfexecute调用来解决的?我很想看看你是怎么做的。
Transfer Started: 09:37:57
Transfer Finished: 09:42:01
Duration: 00:04:04
File Size: 331,770.8 KB
Transfer Rate: 1,359.7 KB/Sec