2010-06-02 88 views
0

我有一个服务器端的基于C的CGI代码:CGI缓冲问题

cgiFormFileSize("UPDATEFILE", &size); //UPDATEFILE = file being uploaded 
cgiFormFileName("UPDATEFILE", file_name, 1024); 
cgiFormFileContentType("UPDATEFILE", mime_type, 1024); 
buffer = malloc(sizeof(char) * size); 

if (cgiFormFileOpen("UPDATEFILE", &file) != cgiFormSuccess) { 
    exit(1); 
} 
output = fopen("/tmp/cgi.tar.gz", "w+"); 

inc = size/(1024*100); 
fptr = fopen("progress_bar.txt", "w+");  
while (cgiFormFileRead(file, b, sizeof(b), &got_count) == cgiFormSuccess) 
{ 
    fwrite(b,sizeof(char),got_count,output); 
    i++; 
    if(i == inc && j<=100) 
    { 
    fprintf(fptr,"%d", j); 
    fflush(fptr); 
    i = 0; 
    j++; // j is the progress bar increment value 
    } 
} 
fclose(fptr); 
cgiFormFileClose(file); 
retval = system("mkdir /tmp/update-tmp;\ 
       cd /tmp/update-tmp;\ 
       tar -xzf ../cgi.tar.gz;\ 
       bash -c /tmp/update-tmp/update.sh"); 

然而,这并不工作方式如上面看到的。不是将1,2,... 100逐一打印到progress_bar.txt(由fptr引用),而是在ONE GO处打印,看来它缓冲并写入文件。 fflush()也没有工作。

任何线索/建议将非常感激。

+1

你怎么知道不是一个接一个写的? – Freddy 2010-06-02 21:53:56

+0

你怎么知道它一气呵成呢?在循环内部没有任何人为延迟,这将几乎立即完成。 – 2010-06-02 21:54:01

+0

我没有理解你想要做什么,以及预期的结果是什么。请提供更多信息。 – Amirshk 2010-06-02 22:02:12

回答

0

首先,在循环之前打开文件并在循环结束后关闭。太多的IO。

问题是在这里w+ - 这会截断你的文件。使用a+。 (fopen帮助)

+0

我也尝试过。在循环之前打开文件并在循环之后关闭它,它也不起作用。我不想追加数据,我想每次都更新数据,这就是为什么我使用w + – Punit 2010-06-02 21:56:39

+0

那么为什么你使用fseek? – Amirshk 2010-06-02 22:00:15

+0

要每次打印文件开始处的值,但在这种情况下,这不是必需的,因为每次都会打开和关闭文件。 – Punit 2010-06-02 22:01:19

0

一个接一个写它,它只是它做得太快,以至于你很难看到文件的值不是99。

如果您将sleep(1)放在循环中,这很容易证明,因此它足够慢,以便您能够捕捉它。

+0

它这样做(我尝试睡觉(1);),但不是在所需的点。一旦CGI完成复制数据并写入文本文件;之后它打印到文本文件。 – Punit 2010-06-02 22:17:54

+0

是什么让你认为'cgiFormFileRead()'被缓慢调用,就像文件上传一样?上传的文件看起来更有可能被完全缓冲,然后提供给您的CGI。 – caf 2010-06-02 23:24:00

+0

你是对的,这种事情正在发生。然而,我不明白“cgiFormFileRead()被缓慢调用,因为文件上传”你有什么想法我现在应该尝试 – Punit 2010-06-03 20:01:47