delimiter = $delimiter; $this->trimWhitespace = (bool) $trimWhitespace; } /** * fromFile(): defined by Reader interface. * * @see ReaderInterface::fromFile() * @param string $filename * @return array * @throws Exception\RuntimeException if the file cannot be read */ public function fromFile($filename) { if (! is_file($filename) || ! is_readable($filename)) { throw new Exception\RuntimeException(sprintf( "File '%s' doesn't exist or not readable", $filename )); } $this->directory = dirname($filename); $config = $this->parse(file_get_contents($filename)); return $this->process($config); } /** * fromString(): defined by Reader interface. * * @see ReaderInterface::fromString() * @param string $string * @return array * @throws Exception\RuntimeException if an @include key is found */ public function fromString($string) { if (empty($string)) { return []; } $this->directory = null; $config = $this->parse($string); return $this->process($config); } /** * Process the array for @include * * @param array $data * @return array * @throws Exception\RuntimeException if an @include key is found */ protected function process(array $data) { foreach ($data as $key => $value) { if (trim($key) === '@include') { if ($this->directory === null) { throw new Exception\RuntimeException('Cannot process @include statement for a string'); } $reader = clone $this; unset($data[$key]); $data = array_replace_recursive($data, $reader->fromFile($this->directory . '/' . $value)); } } return $data; } /** * Parse Java-style properties string * * @todo Support use of the equals sign "key=value" as key-value delimiter * @todo Ignore whitespace that precedes text past the first line of multiline values * * @param string $string * @return array */ protected function parse($string) { $delimiter = $this->delimiter; $delimLength = strlen($delimiter); $result = []; $lines = explode("\n", $string); $key = ''; $isWaitingOtherLine = false; foreach ($lines as $i => $line) { // Ignore empty lines and commented lines if (empty($line) || (! $isWaitingOtherLine && strpos($line, "#") === 0) || (! $isWaitingOtherLine && strpos($line, "!") === 0) ) { continue; } // Add a new key-value pair or append value to a previous pair if (! $isWaitingOtherLine) { $key = substr($line, 0, strpos($line, $delimiter)); $value = substr($line, strpos($line, $delimiter) + $delimLength, strlen($line)); } else { $value .= $line; } // Check if ends with single '\' (indicating another line is expected) if (strrpos($value, "\\") === strlen($value) - strlen("\\")) { $value = substr($value, 0, -1); $isWaitingOtherLine = true; } else { $isWaitingOtherLine = false; } $key = $this->trimWhitespace ? trim($key) : $key; $value = $this->trimWhitespace && ! $isWaitingOtherLine ? trim($value) : $value; $result[$key] = stripslashes($value); unset($lines[$i]); } return $result; } }