PHP को चंकित प्रतिक्रिया उत्पन्न करने के लिए कैसे करें




http chunked-encoding (6)

16.02.2013 के रूप में कार्य नमूना

<?php

function dump_chunk($chunk) {
    //echo sprintf("%x\r\n", strlen($chunk));
    echo sprintf("\r\n");
    echo $chunk;
    echo "\r\n";
}

ob_start();
?>
<!DOCTYPE html>
<html>
    <head>
        <meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
        <title>Title</title>
    </head>
    <body>
        <?php
        ob_end_flush();
        flush();
        ob_flush();
        for ($i = 0; $i < 5; $i++) {
            sleep(1);
            dump_chunk('Sending data chunk ' . ($i + 1) . ' of 1000 <br />');
            flush();
            ob_flush();
        }
        sleep(1); // needed for last animation
        ?>
    </body>
</html>

मैं इस समस्या के लिए googled लेकिन इसके लिए कोई जवाब नहीं है।

मैं चाहता हूं कि मेरी PHP स्क्रिप्ट को खंडित ( http://en.wikipedia.org/wiki/Chunked_transfer_encoding ) में HTTP प्रतिक्रिया उत्पन्न करें। यह कैसे करना है?

अद्यतन: मैंने इसे समझ लिया। मुझे ट्रांसफर-एन्कोडिंग हेडर निर्दिष्ट करना होगा और इसे फ्लश करना होगा।

header("Transfer-encoding: chunked");
flush(); 

फ्लश आवश्यक है। अन्यथा, सामग्री-लंबाई शीर्षलेख उत्पन्न किया जाएगा।

और, मुझे अपने आप को टुकड़े बनाना है। एक सहायक समारोह के साथ, यह मुश्किल नहीं है।

function dump_chunk($chunk)
{
    echo sprintf("%x\r\n", strlen($chunk));
    echo $chunk;
    echo "\r\n";
}

आउटपुट बफर ब्राउजर को तब तक नहीं भेजा जाएगा जब तक कि यह पूरा न हो जाए। डिफ़ॉल्ट आकार 4096 बाइट है। इसलिए आपको या तो बफर आकार बदलने या अपने भाग को पैड करने की आवश्यकता है। साथ ही, पेज प्रदर्शित होने से पहले ब्राउजर का अपना न्यूनतम बफर हो सकता है। ध्यान दें कि http://en.wikipedia.org/wiki/Chunked_transfer_encoding अनुसार खंडित एन्कोडिंग के बारे में http://en.wikipedia.org/wiki/Chunked_transfer_encoding , 0\r\n\r\n खंड भेजना प्रतिक्रिया को समाप्त करता है।

यदि आप आउटपुट बफरिंग आकार सेटिंग को बदलना चाहते हैं, तो मैंने पढ़ा है कि आप ini_set('output_buffering', $value) उपयोग नहीं कर सकते हैं। इसके बजाए, अपनी php.ini फ़ाइल में निम्नलिखित जोड़कर आउटपुट बफरिंग सेटिंग बदलें।

php_value output_buffering 1024

यहां आपके हिस्सों को पैड करने का एक उदाहरण दिया गया है

header("Transfer-Encoding: chunked");
header("Content-Encoding: none");

// Send chunk to browser
function send_chunk($chunk)
{
    // The chunk must fill the output buffer or php won't send it
    $chunk = str_pad($chunk, 4096);

    printf("%x\r\n%s\r\n", strlen($chunk), $chunk);
    flush();
}

// Send your content in chunks
for($i=0; $i<10; $i++)
{
    send_chunk("This is Chunk #$i.<br>\r\n");
    usleep(500000);
}

// note that if you send an empty chunk
// the browser won't display additional output
echo "0\r\n\r\n";
flush();

यहां एक छोटा संस्करण है जो आउटपुट को समाप्त करने वाले 0\r\n\r\n\ करता है:

$output = "hello world";
header("Transfer-Encoding: chunked");
header('Content-Encoding: none');
printf("%x\r\n%s\r\n", strlen($output), $output);
ob_flush();
print("0\r\n\r\n");
ob_flush();
flush();
sleep(10);
print('POST PROCESSING');

इसे काम करने के लिए मुझे दो अतिरिक्त चीजें करने की ज़रूरत थी। यदि कोई ऑब्जेक्ट अभी तक शुरू नहीं हुआ है तो ob_start पर कॉल करें। और एक लंबी खाली स्ट्रिंग भी गूंजें। ब्राउजर को इसके पहले खंड में भेजे जाने वाले कुछ निश्चित सामग्री की आवश्यकता होती है।

header('Content-Encoding', 'chunked');
header('Transfer-Encoding', 'chunked');
header('Content-Type', 'text/html');
header('Connection', 'keep-alive');

if (ob_get_level() == 0) ob_start();
echo str_pad('',4096)."\n";
ob_flush();
flush();

एक आसान तरीका (अपना खुद का डेटा हिस्सा बनाये बिना) फ्लश () का उपयोग करना है:

<?php 
$fp = fopen('data.bin', "rb");
flush();
fpassthru($fp);
?>

यह मेरे लिए निम्न HTTP अनुरोध और प्रतिक्रिया में दिखाए गए अनुसार काम करता है।

निवेदन:

00000000  47 45 54 20 2f 67 65 74  2d 63 68 75 6e 6b 2e 70 GET /get -chunk.p
00000010  68 70 20 48 54 54 50 2f  31 2e 31 0d 0a 55 73 65 hp HTTP/ 1.1..Use
00000020  72 2d 41 67 65 6e 74 3a  20 63 75 72 6c 2f 37 2e r-Agent:  curl/7.
00000030  32 31 2e 37 20 28 69 36  38 36 2d 70 63 2d 6c 69 21.7 (i6 86-pc-li
00000040  6e 75 78 2d 67 6e 75 29  20 6c 69 62 63 75 72 6c nux-gnu)  libcurl
00000050  2f 37 2e 32 31 2e 37 20  4f 70 65 6e 53 53 4c 2f /7.21.7  OpenSSL/
00000060  31 2e 30 2e 30 64 20 7a  6c 69 62 2f 31 2e 32 2e 1.0.0d z lib/1.2.
00000070  35 20 6c 69 62 73 73 68  32 2f 31 2e 32 2e 37 0d 5 libssh 2/1.2.7.
00000080  0a 48 6f 73 74 3a 20 67  61 69 61 0d 0a 41 63 63 .Host: g aia..Acc
00000090  65 70 74 3a 20 2a 2f 2a  0d 0a 0d 0a             ept: */* ....

उत्तर:

00000000  48 54 54 50 2f 31 2e 31  20 32 30 30 20 4f 4b 0d HTTP/1.1  200 OK.
00000010  0a 44 61 74 65 3a 20 54  68 75 2c 20 32 32 20 53 .Date: T hu, 22 S
00000020  65 70 20 32 30 31 31 20  32 33 3a 35 33 3a 30 32 ep 2011  23:53:02
00000030  20 47 4d 54 0d 0a 53 65  72 76 65 72 3a 20 41 70  GMT..Se rver: Ap
00000040  61 63 68 65 2f 32 2e 32  2e 38 20 28 46 65 64 6f ache/2.2 .8 (Fedo
00000050  72 61 29 0d 0a 58 2d 50  6f 77 65 72 65 64 2d 42 ra)..X-P owered-B
00000060  79 3a 20 50 48 50 2f 35  2e 32 2e 36 0d 0a 54 72 y: PHP/5 .2.6..Tr
00000070  61 6e 73 66 65 72 2d 45  6e 63 6f 64 69 6e 67 3a ansfer-E ncoding:
00000080  20 63 68 75 6e 6b 65 64  0d 0a 43 6f 6e 74 65 6e  chunked ..Conten
00000090  74 2d 54 79 70 65 3a 20  74 65 78 74 2f 68 74 6d t-Type:  text/htm
000000A0  6c 3b 20 63 68 61 72 73  65 74 3d 55 54 46 2d 38 l; chars et=UTF-8
000000B0  0d 0a 0d 0a                                      ....
000000B4  61 0d 0a 54 45 53 54 20  44 41 54 41 0a 0d 0a 30 a..TEST  DATA...0
000000C4  0d 0a 0d 0a                                      ....

यदि आप कोई सामग्री-लंबाई शीर्षलेख निर्दिष्ट नहीं करते हैं, और एक फ्लश होता है तो एक PHP प्रतिक्रिया हमेशा छिड़काई जाएगी। (जो x बाइट्स के बाद स्वचालित रूप से हो जाएगा, बस इतना नहीं पता कि कितना)।

यह देखभाल करने के लिए एक अजीब चीज है। क्या यह किसी तरह का अकादमिक / सीखने का अभ्यास है या क्या कोई असली दुनिया की समस्या है जिसे आप हल करने की कोशिश कर रहे हैं?


यह थोड़ा सा गेज प्राप्त हुआ है ... यदि आपको बड़े assed भाग, (0x1000 octets या तो) पर ध्यान नहीं दिया जाता है, तो हाँ, PHP उन्हें बना देगा।

<?php

while (true) {
    # output data
    flush()
    usleep(pow(2,18));
}
?>

PHP क्रमांकित वर्ग, आदि उत्पन्न करेगा

यदि आप छोटे छोटे हिस्से भेजना चाहते हैं, जैसा कि आप एक AJAX क्लाइंट के साथ कर सकते हैं ... ठीक है, मैंने PHP.NET पर कुछ शोध के साथ ओपीएस प्रश्न को जोड़ दिया है, और ऐसा लगता है कि वह एक अच्छी चीज पर था ।

$ echo -en "प्राप्त / खंडित / HTTP / 1.1 \ r \ n होस्ट: ec \ r \ n \ r \ n" | एनसी लोकलहोस्ट 80

HTTP/1.1 200 OK
Date: Wed, 23 May 2012 13:03:01 GMT
Server: Apache/2.2.9 (Debian) PHP/5.3.5-1 with Suhosin-Patch mod_ssl/2.2.9 OpenSSL/0.9.8o
X-Powered-By: PHP/5.3.5-1
Transfer-encoding: chunked
Content-Type: text/html

14
Teachers have class.
50
We secure our friends not by accepting favors but by doing them.
            -- Thucydides
48
Vulcans never bluff.
            -- Spock, "The Doomsday Machine", stardate 4202.1
31
All kings is mostly rapscallions.
            -- Mark Twain
41
Reappraisal, n.:
    An abrupt change of mind after being found out.
49
He who knows, does not speak.  He who speaks, does not know.
            -- Lao Tsu

अंततः यह स्वयं (गलत) खंड गणना के बाहर निकल जाएगा या नहीं, यह देखा जाना बाकी है ... लेकिन मैंने इसका कोई संकेत नहीं देखा।

<?php
header("Transfer-encoding: chunked");
@apache_setenv('no-gzip', 1);
@ini_set('zlib.output_compression', 0);
@ini_set('implicit_flush', 1);
for ($i = 0; $i < ob_get_level(); $i++)  ob_end_flush();
ob_implicit_flush(1); flush();

function dump_chunk($chunk)
{
  printf("%x\r\n%s\r\n", strlen($chunk), $chunk);
  flush();
}

for (;;) {
  $output = array();
  exec("/usr/games/fortune", $output);
  dump_chunk(implode("\n", $output));
  usleep(pow(2,18));
}
?>




chunked-encoding