Page Menu
Home
c4science
Search
Configure Global Search
Log In
Files
F92768719
LinesOfALargeExecFuture.php
No One
Temporary
Actions
Download File
Edit File
Delete File
View Transforms
Subscribe
Mute Notifications
Award Token
Subscribers
None
File Metadata
Details
File Info
Storage
Attached
Created
Sat, Nov 23, 13:52
Size
3 KB
Mime Type
text/x-php
Expires
Mon, Nov 25, 13:52 (1 d, 20 h)
Engine
blob
Format
Raw Data
Handle
22508800
Attached To
rPHU libphutil
LinesOfALargeExecFuture.php
View Options
<?php
/**
* Read the output stream of an @{class:ExecFuture} one line at a time. This
* abstraction allows you to process large inputs without holding them in
* memory. If you know your inputs fit in memory, it is generally more efficient
* (and certainly simpler) to read the entire input and `explode()` it. For
* more information, see @{class:LinesOfALarge}. See also
* @{class:LinesOfALargeFile} for a similar abstraction that works on files.
*
* $future = new ExecFuture('hg log ...');
* foreach (new LinesOfALargeExecFuture($future) as $line) {
* // ...
* }
*
* If the subprocess exits with an error, a @{class:CommandException} will be
* thrown.
*
* On destruction, this class terminates the subprocess if it has not already
* exited.
*
* @task construct Construction
* @task internals Internals
* @group filesystem
*/
final
class
LinesOfALargeExecFuture
extends
LinesOfALarge
{
private
$future
;
private
$didRewind
;
/* -( Construction )------------------------------------------------------- */
/**
* To construct, pass an @{class:ExecFuture}.
*
* @param ExecFuture Future to wrap.
* @return void
* @task construct
*/
public
function
__construct
(
ExecFuture
$future
)
{
$this
->
future
=
$future
;
}
/* -( Internals )---------------------------------------------------------- */
/**
* On destruction, we terminate the subprocess if it hasn't exited already.
*
* @return void
* @task internals
*/
public
function
__destruct
()
{
if
(!
$this
->
future
->
isReady
())
{
$this
->
future
->
resolveKill
();
}
}
/**
* The PHP foreach() construct calls rewind() once, so we allow the first
* rewind(), without effect. Subsequent rewinds mean misuse.
*
* @return void
* @task internals
*/
protected
function
willRewind
()
{
if
(
$this
->
didRewind
)
{
throw
new
Exception
(
"You can not reiterate over a LinesOfALargeExecFuture object. The "
.
"entire goal of the construct is to avoid keeping output in memory. "
.
"What you are attempting to do is silly and doesn't make any sense."
);
}
$this
->
didRewind
=
true
;
}
/**
* Read more data from the subprocess.
*
* @return string Bytes read from stdout.
* @task internals
*/
protected
function
readMore
()
{
$future
=
$this
->
future
;
while
(
true
)
{
// Read is nonblocking, so we need to sit in this loop waiting for input
// or we'll incorrectly signal EOF to the parent.
list
(
$stdout
)
=
$future
->
read
();
$future
->
discardBuffers
();
if
(
strlen
(
$stdout
))
{
return
$stdout
;
}
// If we didn't read anything, we can exit the loop if the subprocess
// has exited.
if
(
$future
->
isReady
())
{
// Throw if the process exits with a nozero status code. This makes
// error handling simpler, and prevents us from returning part of a line
// if the process terminates mid-output.
$future
->
resolvex
();
// Read and return anything that's left.
list
(
$stdout
)
=
$future
->
read
();
$future
->
discardBuffers
();
return
$stdout
;
}
}
}
}
Event Timeline
Log In to Comment