2005-03-09 11:07:07 +08:00
|
|
|
<?php
|
|
|
|
|
2005-09-15 22:08:16 +08:00
|
|
|
/** @file splfileobject.inc
|
2005-03-09 11:07:07 +08:00
|
|
|
* @ingroup SPL
|
2005-04-15 05:02:09 +08:00
|
|
|
* @brief class FileObject
|
2005-03-09 11:07:07 +08:00
|
|
|
* @author Marcus Boerger
|
2006-07-15 23:08:41 +08:00
|
|
|
* @date 2003 - 2006
|
2005-03-09 11:07:07 +08:00
|
|
|
*
|
|
|
|
* SPL - Standard PHP Library
|
|
|
|
*/
|
|
|
|
|
|
|
|
/** @ingroup SPL
|
|
|
|
* @brief Object representation for any stream
|
|
|
|
* @author Marcus Boerger
|
2006-07-15 23:08:41 +08:00
|
|
|
* @version 1.1
|
2005-03-09 11:07:07 +08:00
|
|
|
* @since PHP 5.1
|
|
|
|
*/
|
2006-07-15 23:08:41 +08:00
|
|
|
class SplFileObject extends SplFileInfo implements RecursiveIterator, SeekableIterator
|
2005-03-09 11:07:07 +08:00
|
|
|
{
|
2005-09-15 11:33:04 +08:00
|
|
|
/** Flag: wheter to suppress new lines */
|
|
|
|
const DROP_NEW_LINE = 0x00000001;
|
|
|
|
|
2005-03-09 11:07:07 +08:00
|
|
|
private $fp;
|
|
|
|
private $fname;
|
|
|
|
private $line = NULL;
|
|
|
|
private $lnum = 0;
|
|
|
|
private $max_len = 0;
|
|
|
|
private $flags = 0;
|
2006-07-15 23:08:41 +08:00
|
|
|
private $delimiter= ',';
|
|
|
|
private $enclosure= '"';
|
2005-03-09 11:07:07 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Constructs a new file object
|
|
|
|
*
|
|
|
|
* @param $file_name The name of the stream to open
|
2005-04-15 05:02:09 +08:00
|
|
|
* @param $open_mode The file open mode
|
2005-03-09 11:07:07 +08:00
|
|
|
* @param $use_include_path Whether to search in include paths
|
|
|
|
* @param $context A stream context
|
2005-04-15 05:02:09 +08:00
|
|
|
* @throw RuntimeException If file cannot be opened (e.g. insufficient
|
|
|
|
* access rights).
|
2005-03-09 11:07:07 +08:00
|
|
|
*/
|
|
|
|
function __construct($file_name, $open_mode = 'r', $use_include_path = false, $context = NULL)
|
|
|
|
{
|
|
|
|
$this->fp = fopen($file_name, $open_mode, $use_include_path, $context);
|
|
|
|
if (!$this->fp)
|
|
|
|
{
|
|
|
|
throw new RuntimeException("Cannot open file $file_name");
|
|
|
|
}
|
|
|
|
$this->fname = $file_name;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2005-06-17 05:27:51 +08:00
|
|
|
* @return whether the end of the stream is reached
|
2005-03-09 11:07:07 +08:00
|
|
|
*/
|
|
|
|
function eof()
|
|
|
|
{
|
|
|
|
return eof($this->fp);
|
|
|
|
}
|
|
|
|
|
|
|
|
/** increase current line number
|
|
|
|
* @return next line from stream
|
|
|
|
*/
|
|
|
|
function fgets()
|
|
|
|
{
|
2005-05-04 06:18:19 +08:00
|
|
|
$this->freeLine();
|
2005-03-09 11:07:07 +08:00
|
|
|
$this->lnum++;
|
|
|
|
$buf = fgets($this->fp, $this->max_len);
|
|
|
|
|
|
|
|
return $buf;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2005-05-31 04:10:38 +08:00
|
|
|
* @param delimiter character used as field separator
|
|
|
|
* @param enclosure end of
|
|
|
|
* @return array containing read data
|
2005-03-09 11:07:07 +08:00
|
|
|
*/
|
2006-07-15 23:08:41 +08:00
|
|
|
function fgetcsv($delimiter = NULL, $enclosure = NULL)
|
2005-03-09 11:07:07 +08:00
|
|
|
{
|
2005-05-04 06:18:19 +08:00
|
|
|
$this->freeLine();
|
2005-03-09 11:07:07 +08:00
|
|
|
$this->lnum++;
|
2006-07-15 23:08:41 +08:00
|
|
|
switch(fun_num_args())
|
|
|
|
{
|
|
|
|
case 0:
|
|
|
|
$delimiter = $this->delimiter;
|
|
|
|
case 1:
|
|
|
|
$enclosure = $this->enclosure;
|
|
|
|
default:
|
|
|
|
case 2:
|
|
|
|
break;
|
|
|
|
}
|
2005-03-09 11:07:07 +08:00
|
|
|
return fgetcsv($this->fp, $this->max_len, $delimiter, $enclosure);
|
|
|
|
}
|
|
|
|
|
2006-07-15 23:08:41 +08:00
|
|
|
/**
|
|
|
|
* Set the delimiter and enclosure character used in fgetcsv
|
|
|
|
*
|
|
|
|
* @param delimiter new delimiter, defaults to ','
|
|
|
|
* @param enclosure new enclosure, defaults to '"'
|
|
|
|
*/
|
|
|
|
function setCsvControl($delimiter = ';', $enclosure = '"')
|
|
|
|
{
|
|
|
|
$this->delimiter = $delimiter;
|
|
|
|
$this->enclosure = $enclosure;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return array(delimiter, enclosure) as used in fgetcsv
|
|
|
|
*/
|
|
|
|
function getCsvControl($delimiter = ',', $enclosure = '"')
|
|
|
|
{
|
|
|
|
return array($this->delimiter, $this->enclosure);
|
|
|
|
}
|
|
|
|
|
2005-03-09 11:07:07 +08:00
|
|
|
/**
|
2005-06-17 05:27:51 +08:00
|
|
|
* @param operation lock operation (LOCK_SH, LOCK_EX, LOCK_UN, LOCK_NB)
|
2005-03-09 11:07:07 +08:00
|
|
|
* @retval $wouldblock whether the operation would block
|
|
|
|
*/
|
|
|
|
function flock($operation, &$wouldblock)
|
|
|
|
{
|
|
|
|
return flock($this->fp, $operation, $wouldblock);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Flush current data
|
|
|
|
* @return success or failure
|
|
|
|
*/
|
|
|
|
function fflush()
|
|
|
|
{
|
|
|
|
return fflush($this->fp);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return current file position
|
|
|
|
*/
|
|
|
|
function ftell()
|
|
|
|
{
|
|
|
|
return ftell($this->fp);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2005-05-31 04:10:38 +08:00
|
|
|
* @param pos new file position
|
|
|
|
* @param whence seek method (SEEK_SET, SEEK_CUR, SEEK_END)
|
|
|
|
* @return Upon success, returns 0; otherwise, returns -1. Note that
|
|
|
|
* seeking past EOF is not considered an error.
|
2005-03-09 11:07:07 +08:00
|
|
|
*/
|
|
|
|
function fseek($pos, $whence = SEEK_SET)
|
|
|
|
{
|
|
|
|
return fseek($this->fp, $pos, $whence);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return next char from file
|
|
|
|
* @note a new line character does not increase $this->lnum
|
|
|
|
*/
|
|
|
|
function fgetc()
|
|
|
|
{
|
2005-05-04 06:18:19 +08:00
|
|
|
$this->freeLine();
|
|
|
|
$c = fgetc($this->fp);
|
|
|
|
if ($c == '\n') {
|
|
|
|
$this->lnum++;
|
|
|
|
}
|
2005-03-09 11:07:07 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/** Read and return remaining part of stream
|
|
|
|
* @return size of remaining part passed through
|
|
|
|
*/
|
|
|
|
function fpassthru()
|
|
|
|
{
|
|
|
|
return fpassthru($this->fp);
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Get a line from the file and strip HTML tags
|
2005-05-31 04:10:38 +08:00
|
|
|
* @param $allowable_tags tags to keep in the string
|
2005-03-09 11:07:07 +08:00
|
|
|
*/
|
|
|
|
function fgetss($allowable_tags = NULL)
|
|
|
|
{
|
|
|
|
return fgetss($this->fp, $allowable_tags);
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Scan the next line
|
|
|
|
* @param $format string specifying format to parse
|
|
|
|
*/
|
|
|
|
function fscanf($format /* , ... */)
|
|
|
|
{
|
2005-05-04 06:18:19 +08:00
|
|
|
$this->freeLine();
|
2005-03-09 11:07:07 +08:00
|
|
|
$this->lnum++;
|
|
|
|
return fscanf($this->fp, $format /* , ... */);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param $str to write
|
2005-05-31 04:10:38 +08:00
|
|
|
* @param $length maximum line length to write
|
2005-03-09 11:07:07 +08:00
|
|
|
*/
|
|
|
|
function fwrite($str, $length = NULL)
|
|
|
|
{
|
|
|
|
return fwrite($this->fp, $length);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return array of file stat information
|
|
|
|
*/
|
|
|
|
function fstat()
|
|
|
|
{
|
|
|
|
return fstat($this->fp);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2005-05-31 04:10:38 +08:00
|
|
|
* @param $size new size to truncate file to
|
2005-03-09 11:07:07 +08:00
|
|
|
*/
|
|
|
|
function ftruncate($size)
|
|
|
|
{
|
|
|
|
return ftruncate($this->fp, $size);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param $flags new flag set
|
|
|
|
*/
|
|
|
|
function setFlags($flags)
|
|
|
|
{
|
|
|
|
$this->flags = $flags;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return current set of flags
|
|
|
|
*/
|
|
|
|
function getFlags()
|
|
|
|
{
|
|
|
|
return $this->flags;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param $max_len set the maximum line length read
|
|
|
|
*/
|
|
|
|
function setMaxLineLen($max_len)
|
|
|
|
{
|
|
|
|
$this->max_len = $max_len;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2005-06-17 05:27:51 +08:00
|
|
|
* @return current setting for max line
|
2005-03-09 11:07:07 +08:00
|
|
|
*/
|
|
|
|
function getMaxLineLen()
|
|
|
|
{
|
|
|
|
return $this->max_len;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return false
|
|
|
|
*/
|
|
|
|
function hasChildren()
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return false
|
|
|
|
*/
|
|
|
|
function getChildren()
|
|
|
|
{
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Invalidate current line buffer and set line number to 0.
|
|
|
|
*/
|
|
|
|
function rewind()
|
|
|
|
{
|
2005-05-04 06:18:19 +08:00
|
|
|
$this->freeLine();
|
2005-03-09 11:07:07 +08:00
|
|
|
$this->lnum = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return whether more data can be read
|
|
|
|
*/
|
|
|
|
function valid()
|
|
|
|
{
|
|
|
|
return !$this->eof();
|
2005-04-15 05:02:09 +08:00
|
|
|
}
|
2005-03-09 11:07:07 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @note Fill current line buffer if not done yet.
|
|
|
|
* @return line buffer
|
|
|
|
*/
|
|
|
|
function current()
|
|
|
|
{
|
|
|
|
if (is_null($this->line))
|
|
|
|
{
|
|
|
|
$this->line = getCurrentLine();
|
|
|
|
}
|
|
|
|
return $this->line;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return line number
|
2005-05-04 06:18:19 +08:00
|
|
|
* @note fgetc() will increase the line number when reaing a new line char.
|
|
|
|
* This has the effect key() called on a read a new line will already
|
|
|
|
* return the increased line number.
|
2005-05-04 06:28:44 +08:00
|
|
|
* @note Line counting works as long as you only read the file and do not
|
|
|
|
* use fseek().
|
2005-03-09 11:07:07 +08:00
|
|
|
*/
|
|
|
|
function key()
|
|
|
|
{
|
|
|
|
return $this->lnum;
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Invalidate current line buffer.
|
|
|
|
*/
|
|
|
|
function next()
|
|
|
|
{
|
2005-05-04 06:18:19 +08:00
|
|
|
$this->freeLine();
|
2005-03-09 11:07:07 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return next line read from file and increase the line counter
|
|
|
|
*/
|
|
|
|
private function readLine()
|
|
|
|
{
|
2005-05-04 06:18:19 +08:00
|
|
|
if ($this->eof())
|
|
|
|
{
|
|
|
|
$this->freeLine();
|
|
|
|
throw new RuntimeException("Cannot read from file " . $this->fname);
|
|
|
|
}
|
|
|
|
if ($this->line) {
|
|
|
|
$this->lnum++;
|
|
|
|
}
|
|
|
|
$this->freeLine();
|
|
|
|
$this->line = fgets($this->fp, $this->max_len);
|
|
|
|
return $this->line;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Free the current line buffer and increment the line counter
|
|
|
|
*/
|
|
|
|
private function freeLine()
|
|
|
|
{
|
|
|
|
if ($this->line) {
|
|
|
|
$this->line = NULL;
|
|
|
|
}
|
2005-03-09 11:07:07 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2005-05-04 06:18:19 +08:00
|
|
|
* @note If you DO overload this function key() and current() will increment
|
|
|
|
* $this->lnum automatically. If not then function reaLine() will do
|
|
|
|
* that for you.
|
2005-03-09 11:07:07 +08:00
|
|
|
*/
|
|
|
|
function getCurrentLine()
|
|
|
|
{
|
2005-05-04 06:18:19 +08:00
|
|
|
$this->freeLine();
|
|
|
|
if ($this->eof())
|
|
|
|
{
|
|
|
|
throw new RuntimeException("Cannot read from file " . $this->fname);
|
|
|
|
}
|
2005-03-09 11:07:07 +08:00
|
|
|
$this->readLine();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return current line
|
|
|
|
*/
|
|
|
|
function __toString()
|
|
|
|
{
|
|
|
|
return current();
|
|
|
|
}
|
2005-05-04 06:18:19 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @param $line_pos Seek to this line
|
|
|
|
*/
|
|
|
|
function seek($line_pos)
|
|
|
|
{
|
|
|
|
$this->rewind();
|
|
|
|
while($this->lnum < $line_pos && !$this->eof())
|
|
|
|
{
|
|
|
|
$this->getCurrentLine();
|
|
|
|
}
|
|
|
|
}
|
2005-03-09 11:07:07 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
?>
|