xref: /plugin/statistics/vendor/mustangostang/spyc/Spyc.php (revision d5ef99ddb7dfb0cfae33e9257bd1d788f682c50f)
1*d5ef99ddSAndreas Gohr<?php
2*d5ef99ddSAndreas Gohr/**
3*d5ef99ddSAndreas Gohr   * Spyc -- A Simple PHP YAML Class
4*d5ef99ddSAndreas Gohr   * @version 0.6.2
5*d5ef99ddSAndreas Gohr   * @author Vlad Andersen <vlad.andersen@gmail.com>
6*d5ef99ddSAndreas Gohr   * @author Chris Wanstrath <chris@ozmm.org>
7*d5ef99ddSAndreas Gohr   * @link https://github.com/mustangostang/spyc/
8*d5ef99ddSAndreas Gohr   * @copyright Copyright 2005-2006 Chris Wanstrath, 2006-2011 Vlad Andersen
9*d5ef99ddSAndreas Gohr   * @license http://www.opensource.org/licenses/mit-license.php MIT License
10*d5ef99ddSAndreas Gohr   * @package Spyc
11*d5ef99ddSAndreas Gohr   */
12*d5ef99ddSAndreas Gohr
13*d5ef99ddSAndreas Gohrif (!function_exists('spyc_load')) {
14*d5ef99ddSAndreas Gohr  /**
15*d5ef99ddSAndreas Gohr   * Parses YAML to array.
16*d5ef99ddSAndreas Gohr   * @param string $string YAML string.
17*d5ef99ddSAndreas Gohr   * @return array
18*d5ef99ddSAndreas Gohr   */
19*d5ef99ddSAndreas Gohr  function spyc_load ($string) {
20*d5ef99ddSAndreas Gohr    return Spyc::YAMLLoadString($string);
21*d5ef99ddSAndreas Gohr  }
22*d5ef99ddSAndreas Gohr}
23*d5ef99ddSAndreas Gohr
24*d5ef99ddSAndreas Gohrif (!function_exists('spyc_load_file')) {
25*d5ef99ddSAndreas Gohr  /**
26*d5ef99ddSAndreas Gohr   * Parses YAML to array.
27*d5ef99ddSAndreas Gohr   * @param string $file Path to YAML file.
28*d5ef99ddSAndreas Gohr   * @return array
29*d5ef99ddSAndreas Gohr   */
30*d5ef99ddSAndreas Gohr  function spyc_load_file ($file) {
31*d5ef99ddSAndreas Gohr    return Spyc::YAMLLoad($file);
32*d5ef99ddSAndreas Gohr  }
33*d5ef99ddSAndreas Gohr}
34*d5ef99ddSAndreas Gohr
35*d5ef99ddSAndreas Gohrif (!function_exists('spyc_dump')) {
36*d5ef99ddSAndreas Gohr  /**
37*d5ef99ddSAndreas Gohr   * Dumps array to YAML.
38*d5ef99ddSAndreas Gohr   * @param array $data Array.
39*d5ef99ddSAndreas Gohr   * @return string
40*d5ef99ddSAndreas Gohr   */
41*d5ef99ddSAndreas Gohr  function spyc_dump ($data) {
42*d5ef99ddSAndreas Gohr    return Spyc::YAMLDump($data, false, false, true);
43*d5ef99ddSAndreas Gohr  }
44*d5ef99ddSAndreas Gohr}
45*d5ef99ddSAndreas Gohr
46*d5ef99ddSAndreas Gohrif (!class_exists('Spyc')) {
47*d5ef99ddSAndreas Gohr
48*d5ef99ddSAndreas Gohr/**
49*d5ef99ddSAndreas Gohr   * The Simple PHP YAML Class.
50*d5ef99ddSAndreas Gohr   *
51*d5ef99ddSAndreas Gohr   * This class can be used to read a YAML file and convert its contents
52*d5ef99ddSAndreas Gohr   * into a PHP array.  It currently supports a very limited subsection of
53*d5ef99ddSAndreas Gohr   * the YAML spec.
54*d5ef99ddSAndreas Gohr   *
55*d5ef99ddSAndreas Gohr   * Usage:
56*d5ef99ddSAndreas Gohr   * <code>
57*d5ef99ddSAndreas Gohr   *   $Spyc  = new Spyc;
58*d5ef99ddSAndreas Gohr   *   $array = $Spyc->load($file);
59*d5ef99ddSAndreas Gohr   * </code>
60*d5ef99ddSAndreas Gohr   * or:
61*d5ef99ddSAndreas Gohr   * <code>
62*d5ef99ddSAndreas Gohr   *   $array = Spyc::YAMLLoad($file);
63*d5ef99ddSAndreas Gohr   * </code>
64*d5ef99ddSAndreas Gohr   * or:
65*d5ef99ddSAndreas Gohr   * <code>
66*d5ef99ddSAndreas Gohr   *   $array = spyc_load_file($file);
67*d5ef99ddSAndreas Gohr   * </code>
68*d5ef99ddSAndreas Gohr   * @package Spyc
69*d5ef99ddSAndreas Gohr   */
70*d5ef99ddSAndreas Gohrclass Spyc {
71*d5ef99ddSAndreas Gohr
72*d5ef99ddSAndreas Gohr  // SETTINGS
73*d5ef99ddSAndreas Gohr
74*d5ef99ddSAndreas Gohr  const REMPTY = "\0\0\0\0\0";
75*d5ef99ddSAndreas Gohr
76*d5ef99ddSAndreas Gohr  /**
77*d5ef99ddSAndreas Gohr   * Setting this to true will force YAMLDump to enclose any string value in
78*d5ef99ddSAndreas Gohr   * quotes.  False by default.
79*d5ef99ddSAndreas Gohr   *
80*d5ef99ddSAndreas Gohr   * @var bool
81*d5ef99ddSAndreas Gohr   */
82*d5ef99ddSAndreas Gohr  public $setting_dump_force_quotes = false;
83*d5ef99ddSAndreas Gohr
84*d5ef99ddSAndreas Gohr  /**
85*d5ef99ddSAndreas Gohr   * Setting this to true will forse YAMLLoad to use syck_load function when
86*d5ef99ddSAndreas Gohr   * possible. False by default.
87*d5ef99ddSAndreas Gohr   * @var bool
88*d5ef99ddSAndreas Gohr   */
89*d5ef99ddSAndreas Gohr  public $setting_use_syck_is_possible = false;
90*d5ef99ddSAndreas Gohr
91*d5ef99ddSAndreas Gohr  /**
92*d5ef99ddSAndreas Gohr   * Setting this to true will forse YAMLLoad to use syck_load function when
93*d5ef99ddSAndreas Gohr   * possible. False by default.
94*d5ef99ddSAndreas Gohr   * @var bool
95*d5ef99ddSAndreas Gohr   */
96*d5ef99ddSAndreas Gohr  public $setting_empty_hash_as_object = false;
97*d5ef99ddSAndreas Gohr
98*d5ef99ddSAndreas Gohr
99*d5ef99ddSAndreas Gohr  /**#@+
100*d5ef99ddSAndreas Gohr  * @access private
101*d5ef99ddSAndreas Gohr  * @var mixed
102*d5ef99ddSAndreas Gohr  */
103*d5ef99ddSAndreas Gohr  private $_dumpIndent;
104*d5ef99ddSAndreas Gohr  private $_dumpWordWrap;
105*d5ef99ddSAndreas Gohr  private $_containsGroupAnchor = false;
106*d5ef99ddSAndreas Gohr  private $_containsGroupAlias = false;
107*d5ef99ddSAndreas Gohr  private $path;
108*d5ef99ddSAndreas Gohr  private $result;
109*d5ef99ddSAndreas Gohr  private $LiteralPlaceHolder = '___YAML_Literal_Block___';
110*d5ef99ddSAndreas Gohr  private $SavedGroups = array();
111*d5ef99ddSAndreas Gohr  private $indent;
112*d5ef99ddSAndreas Gohr  /**
113*d5ef99ddSAndreas Gohr   * Path modifier that should be applied after adding current element.
114*d5ef99ddSAndreas Gohr   * @var array
115*d5ef99ddSAndreas Gohr   */
116*d5ef99ddSAndreas Gohr  private $delayedPath = array();
117*d5ef99ddSAndreas Gohr
118*d5ef99ddSAndreas Gohr  /**#@+
119*d5ef99ddSAndreas Gohr  * @access public
120*d5ef99ddSAndreas Gohr  * @var mixed
121*d5ef99ddSAndreas Gohr  */
122*d5ef99ddSAndreas Gohr  public $_nodeId;
123*d5ef99ddSAndreas Gohr
124*d5ef99ddSAndreas Gohr/**
125*d5ef99ddSAndreas Gohr * Load a valid YAML string to Spyc.
126*d5ef99ddSAndreas Gohr * @param string $input
127*d5ef99ddSAndreas Gohr * @return array
128*d5ef99ddSAndreas Gohr */
129*d5ef99ddSAndreas Gohr  public function load ($input) {
130*d5ef99ddSAndreas Gohr    return $this->_loadString($input);
131*d5ef99ddSAndreas Gohr  }
132*d5ef99ddSAndreas Gohr
133*d5ef99ddSAndreas Gohr /**
134*d5ef99ddSAndreas Gohr * Load a valid YAML file to Spyc.
135*d5ef99ddSAndreas Gohr * @param string $file
136*d5ef99ddSAndreas Gohr * @return array
137*d5ef99ddSAndreas Gohr */
138*d5ef99ddSAndreas Gohr  public function loadFile ($file) {
139*d5ef99ddSAndreas Gohr    return $this->_load($file);
140*d5ef99ddSAndreas Gohr  }
141*d5ef99ddSAndreas Gohr
142*d5ef99ddSAndreas Gohr  /**
143*d5ef99ddSAndreas Gohr     * Load YAML into a PHP array statically
144*d5ef99ddSAndreas Gohr     *
145*d5ef99ddSAndreas Gohr     * The load method, when supplied with a YAML stream (string or file),
146*d5ef99ddSAndreas Gohr     * will do its best to convert YAML in a file into a PHP array.  Pretty
147*d5ef99ddSAndreas Gohr     * simple.
148*d5ef99ddSAndreas Gohr     *  Usage:
149*d5ef99ddSAndreas Gohr     *  <code>
150*d5ef99ddSAndreas Gohr     *   $array = Spyc::YAMLLoad('lucky.yaml');
151*d5ef99ddSAndreas Gohr     *   print_r($array);
152*d5ef99ddSAndreas Gohr     *  </code>
153*d5ef99ddSAndreas Gohr     * @access public
154*d5ef99ddSAndreas Gohr     * @return array
155*d5ef99ddSAndreas Gohr     * @param string $input Path of YAML file or string containing YAML
156*d5ef99ddSAndreas Gohr     * @param array set options
157*d5ef99ddSAndreas Gohr     */
158*d5ef99ddSAndreas Gohr  public static function YAMLLoad($input, $options = []) {
159*d5ef99ddSAndreas Gohr    $Spyc = new Spyc;
160*d5ef99ddSAndreas Gohr    foreach ($options as $key => $value) {
161*d5ef99ddSAndreas Gohr        if (property_exists($Spyc, $key)) {
162*d5ef99ddSAndreas Gohr            $Spyc->$key = $value;
163*d5ef99ddSAndreas Gohr        }
164*d5ef99ddSAndreas Gohr    }
165*d5ef99ddSAndreas Gohr    return $Spyc->_load($input);
166*d5ef99ddSAndreas Gohr  }
167*d5ef99ddSAndreas Gohr
168*d5ef99ddSAndreas Gohr  /**
169*d5ef99ddSAndreas Gohr     * Load a string of YAML into a PHP array statically
170*d5ef99ddSAndreas Gohr     *
171*d5ef99ddSAndreas Gohr     * The load method, when supplied with a YAML string, will do its best
172*d5ef99ddSAndreas Gohr     * to convert YAML in a string into a PHP array.  Pretty simple.
173*d5ef99ddSAndreas Gohr     *
174*d5ef99ddSAndreas Gohr     * Note: use this function if you don't want files from the file system
175*d5ef99ddSAndreas Gohr     * loaded and processed as YAML.  This is of interest to people concerned
176*d5ef99ddSAndreas Gohr     * about security whose input is from a string.
177*d5ef99ddSAndreas Gohr     *
178*d5ef99ddSAndreas Gohr     *  Usage:
179*d5ef99ddSAndreas Gohr     *  <code>
180*d5ef99ddSAndreas Gohr     *   $array = Spyc::YAMLLoadString("---\n0: hello world\n");
181*d5ef99ddSAndreas Gohr     *   print_r($array);
182*d5ef99ddSAndreas Gohr     *  </code>
183*d5ef99ddSAndreas Gohr     * @access public
184*d5ef99ddSAndreas Gohr     * @return array
185*d5ef99ddSAndreas Gohr     * @param string $input String containing YAML
186*d5ef99ddSAndreas Gohr     * @param array set options
187*d5ef99ddSAndreas Gohr     */
188*d5ef99ddSAndreas Gohr  public static function YAMLLoadString($input, $options = []) {
189*d5ef99ddSAndreas Gohr    $Spyc = new Spyc;
190*d5ef99ddSAndreas Gohr    foreach ($options as $key => $value) {
191*d5ef99ddSAndreas Gohr        if (property_exists($Spyc, $key)) {
192*d5ef99ddSAndreas Gohr            $Spyc->$key = $value;
193*d5ef99ddSAndreas Gohr        }
194*d5ef99ddSAndreas Gohr    }
195*d5ef99ddSAndreas Gohr    return $Spyc->_loadString($input);
196*d5ef99ddSAndreas Gohr  }
197*d5ef99ddSAndreas Gohr
198*d5ef99ddSAndreas Gohr  /**
199*d5ef99ddSAndreas Gohr     * Dump YAML from PHP array statically
200*d5ef99ddSAndreas Gohr     *
201*d5ef99ddSAndreas Gohr     * The dump method, when supplied with an array, will do its best
202*d5ef99ddSAndreas Gohr     * to convert the array into friendly YAML.  Pretty simple.  Feel free to
203*d5ef99ddSAndreas Gohr     * save the returned string as nothing.yaml and pass it around.
204*d5ef99ddSAndreas Gohr     *
205*d5ef99ddSAndreas Gohr     * Oh, and you can decide how big the indent is and what the wordwrap
206*d5ef99ddSAndreas Gohr     * for folding is.  Pretty cool -- just pass in 'false' for either if
207*d5ef99ddSAndreas Gohr     * you want to use the default.
208*d5ef99ddSAndreas Gohr     *
209*d5ef99ddSAndreas Gohr     * Indent's default is 2 spaces, wordwrap's default is 40 characters.  And
210*d5ef99ddSAndreas Gohr     * you can turn off wordwrap by passing in 0.
211*d5ef99ddSAndreas Gohr     *
212*d5ef99ddSAndreas Gohr     * @access public
213*d5ef99ddSAndreas Gohr     * @return string
214*d5ef99ddSAndreas Gohr     * @param array|\stdClass $array PHP array
215*d5ef99ddSAndreas Gohr     * @param int $indent Pass in false to use the default, which is 2
216*d5ef99ddSAndreas Gohr     * @param int $wordwrap Pass in 0 for no wordwrap, false for default (40)
217*d5ef99ddSAndreas Gohr     * @param bool $no_opening_dashes Do not start YAML file with "---\n"
218*d5ef99ddSAndreas Gohr     */
219*d5ef99ddSAndreas Gohr  public static function YAMLDump($array, $indent = false, $wordwrap = false, $no_opening_dashes = false) {
220*d5ef99ddSAndreas Gohr    $spyc = new Spyc;
221*d5ef99ddSAndreas Gohr    return $spyc->dump($array, $indent, $wordwrap, $no_opening_dashes);
222*d5ef99ddSAndreas Gohr  }
223*d5ef99ddSAndreas Gohr
224*d5ef99ddSAndreas Gohr
225*d5ef99ddSAndreas Gohr  /**
226*d5ef99ddSAndreas Gohr     * Dump PHP array to YAML
227*d5ef99ddSAndreas Gohr     *
228*d5ef99ddSAndreas Gohr     * The dump method, when supplied with an array, will do its best
229*d5ef99ddSAndreas Gohr     * to convert the array into friendly YAML.  Pretty simple.  Feel free to
230*d5ef99ddSAndreas Gohr     * save the returned string as tasteful.yaml and pass it around.
231*d5ef99ddSAndreas Gohr     *
232*d5ef99ddSAndreas Gohr     * Oh, and you can decide how big the indent is and what the wordwrap
233*d5ef99ddSAndreas Gohr     * for folding is.  Pretty cool -- just pass in 'false' for either if
234*d5ef99ddSAndreas Gohr     * you want to use the default.
235*d5ef99ddSAndreas Gohr     *
236*d5ef99ddSAndreas Gohr     * Indent's default is 2 spaces, wordwrap's default is 40 characters.  And
237*d5ef99ddSAndreas Gohr     * you can turn off wordwrap by passing in 0.
238*d5ef99ddSAndreas Gohr     *
239*d5ef99ddSAndreas Gohr     * @access public
240*d5ef99ddSAndreas Gohr     * @return string
241*d5ef99ddSAndreas Gohr     * @param array $array PHP array
242*d5ef99ddSAndreas Gohr     * @param int $indent Pass in false to use the default, which is 2
243*d5ef99ddSAndreas Gohr     * @param int $wordwrap Pass in 0 for no wordwrap, false for default (40)
244*d5ef99ddSAndreas Gohr     */
245*d5ef99ddSAndreas Gohr  public function dump($array,$indent = false,$wordwrap = false, $no_opening_dashes = false) {
246*d5ef99ddSAndreas Gohr    // Dumps to some very clean YAML.  We'll have to add some more features
247*d5ef99ddSAndreas Gohr    // and options soon.  And better support for folding.
248*d5ef99ddSAndreas Gohr
249*d5ef99ddSAndreas Gohr    // New features and options.
250*d5ef99ddSAndreas Gohr    if ($indent === false or !is_numeric($indent)) {
251*d5ef99ddSAndreas Gohr      $this->_dumpIndent = 2;
252*d5ef99ddSAndreas Gohr    } else {
253*d5ef99ddSAndreas Gohr      $this->_dumpIndent = $indent;
254*d5ef99ddSAndreas Gohr    }
255*d5ef99ddSAndreas Gohr
256*d5ef99ddSAndreas Gohr    if ($wordwrap === false or !is_numeric($wordwrap)) {
257*d5ef99ddSAndreas Gohr      $this->_dumpWordWrap = 40;
258*d5ef99ddSAndreas Gohr    } else {
259*d5ef99ddSAndreas Gohr      $this->_dumpWordWrap = $wordwrap;
260*d5ef99ddSAndreas Gohr    }
261*d5ef99ddSAndreas Gohr
262*d5ef99ddSAndreas Gohr    // New YAML document
263*d5ef99ddSAndreas Gohr    $string = "";
264*d5ef99ddSAndreas Gohr    if (!$no_opening_dashes) $string = "---\n";
265*d5ef99ddSAndreas Gohr
266*d5ef99ddSAndreas Gohr    // Start at the base of the array and move through it.
267*d5ef99ddSAndreas Gohr    if ($array) {
268*d5ef99ddSAndreas Gohr      $array = (array)$array;
269*d5ef99ddSAndreas Gohr      $previous_key = -1;
270*d5ef99ddSAndreas Gohr      foreach ($array as $key => $value) {
271*d5ef99ddSAndreas Gohr        if (!isset($first_key)) $first_key = $key;
272*d5ef99ddSAndreas Gohr        $string .= $this->_yamlize($key,$value,0,$previous_key, $first_key, $array);
273*d5ef99ddSAndreas Gohr        $previous_key = $key;
274*d5ef99ddSAndreas Gohr      }
275*d5ef99ddSAndreas Gohr    }
276*d5ef99ddSAndreas Gohr    return $string;
277*d5ef99ddSAndreas Gohr  }
278*d5ef99ddSAndreas Gohr
279*d5ef99ddSAndreas Gohr  /**
280*d5ef99ddSAndreas Gohr     * Attempts to convert a key / value array item to YAML
281*d5ef99ddSAndreas Gohr     * @access private
282*d5ef99ddSAndreas Gohr     * @return string
283*d5ef99ddSAndreas Gohr     * @param $key The name of the key
284*d5ef99ddSAndreas Gohr     * @param $value The value of the item
285*d5ef99ddSAndreas Gohr     * @param $indent The indent of the current node
286*d5ef99ddSAndreas Gohr     */
287*d5ef99ddSAndreas Gohr  private function _yamlize($key,$value,$indent, $previous_key = -1, $first_key = 0, $source_array = null) {
288*d5ef99ddSAndreas Gohr    if(is_object($value)) $value = (array)$value;
289*d5ef99ddSAndreas Gohr    if (is_array($value)) {
290*d5ef99ddSAndreas Gohr      if (empty ($value))
291*d5ef99ddSAndreas Gohr        return $this->_dumpNode($key, array(), $indent, $previous_key, $first_key, $source_array);
292*d5ef99ddSAndreas Gohr      // It has children.  What to do?
293*d5ef99ddSAndreas Gohr      // Make it the right kind of item
294*d5ef99ddSAndreas Gohr      $string = $this->_dumpNode($key, self::REMPTY, $indent, $previous_key, $first_key, $source_array);
295*d5ef99ddSAndreas Gohr      // Add the indent
296*d5ef99ddSAndreas Gohr      $indent += $this->_dumpIndent;
297*d5ef99ddSAndreas Gohr      // Yamlize the array
298*d5ef99ddSAndreas Gohr      $string .= $this->_yamlizeArray($value,$indent);
299*d5ef99ddSAndreas Gohr    } elseif (!is_array($value)) {
300*d5ef99ddSAndreas Gohr      // It doesn't have children.  Yip.
301*d5ef99ddSAndreas Gohr      $string = $this->_dumpNode($key, $value, $indent, $previous_key, $first_key, $source_array);
302*d5ef99ddSAndreas Gohr    }
303*d5ef99ddSAndreas Gohr    return $string;
304*d5ef99ddSAndreas Gohr  }
305*d5ef99ddSAndreas Gohr
306*d5ef99ddSAndreas Gohr  /**
307*d5ef99ddSAndreas Gohr     * Attempts to convert an array to YAML
308*d5ef99ddSAndreas Gohr     * @access private
309*d5ef99ddSAndreas Gohr     * @return string
310*d5ef99ddSAndreas Gohr     * @param $array The array you want to convert
311*d5ef99ddSAndreas Gohr     * @param $indent The indent of the current level
312*d5ef99ddSAndreas Gohr     */
313*d5ef99ddSAndreas Gohr  private function _yamlizeArray($array,$indent) {
314*d5ef99ddSAndreas Gohr    if (is_array($array)) {
315*d5ef99ddSAndreas Gohr      $string = '';
316*d5ef99ddSAndreas Gohr      $previous_key = -1;
317*d5ef99ddSAndreas Gohr      foreach ($array as $key => $value) {
318*d5ef99ddSAndreas Gohr        if (!isset($first_key)) $first_key = $key;
319*d5ef99ddSAndreas Gohr        $string .= $this->_yamlize($key, $value, $indent, $previous_key, $first_key, $array);
320*d5ef99ddSAndreas Gohr        $previous_key = $key;
321*d5ef99ddSAndreas Gohr      }
322*d5ef99ddSAndreas Gohr      return $string;
323*d5ef99ddSAndreas Gohr    } else {
324*d5ef99ddSAndreas Gohr      return false;
325*d5ef99ddSAndreas Gohr    }
326*d5ef99ddSAndreas Gohr  }
327*d5ef99ddSAndreas Gohr
328*d5ef99ddSAndreas Gohr  /**
329*d5ef99ddSAndreas Gohr     * Returns YAML from a key and a value
330*d5ef99ddSAndreas Gohr     * @access private
331*d5ef99ddSAndreas Gohr     * @return string
332*d5ef99ddSAndreas Gohr     * @param $key The name of the key
333*d5ef99ddSAndreas Gohr     * @param $value The value of the item
334*d5ef99ddSAndreas Gohr     * @param $indent The indent of the current node
335*d5ef99ddSAndreas Gohr     */
336*d5ef99ddSAndreas Gohr  private function _dumpNode($key, $value, $indent, $previous_key = -1, $first_key = 0, $source_array = null) {
337*d5ef99ddSAndreas Gohr    // do some folding here, for blocks
338*d5ef99ddSAndreas Gohr    if (is_string ($value) && ((strpos($value,"\n") !== false || strpos($value,": ") !== false || strpos($value,"- ") !== false ||
339*d5ef99ddSAndreas Gohr      strpos($value,"*") !== false || strpos($value,"#") !== false || strpos($value,"<") !== false || strpos($value,">") !== false || strpos ($value, '%') !== false || strpos ($value, '  ') !== false ||
340*d5ef99ddSAndreas Gohr      strpos($value,"[") !== false || strpos($value,"]") !== false || strpos($value,"{") !== false || strpos($value,"}") !== false) || strpos($value,"&") !== false || strpos($value, "'") !== false || strpos($value, "!") === 0 ||
341*d5ef99ddSAndreas Gohr      substr ($value, -1, 1) == ':')
342*d5ef99ddSAndreas Gohr    ) {
343*d5ef99ddSAndreas Gohr      $value = $this->_doLiteralBlock($value,$indent);
344*d5ef99ddSAndreas Gohr    } else {
345*d5ef99ddSAndreas Gohr      $value  = $this->_doFolding($value,$indent);
346*d5ef99ddSAndreas Gohr    }
347*d5ef99ddSAndreas Gohr
348*d5ef99ddSAndreas Gohr    if ($value === array()) $value = '[ ]';
349*d5ef99ddSAndreas Gohr    if ($value === "") $value = '""';
350*d5ef99ddSAndreas Gohr    if (self::isTranslationWord($value)) {
351*d5ef99ddSAndreas Gohr      $value = $this->_doLiteralBlock($value, $indent);
352*d5ef99ddSAndreas Gohr    }
353*d5ef99ddSAndreas Gohr    if (trim ($value) != $value)
354*d5ef99ddSAndreas Gohr       $value = $this->_doLiteralBlock($value,$indent);
355*d5ef99ddSAndreas Gohr
356*d5ef99ddSAndreas Gohr    if (is_bool($value)) {
357*d5ef99ddSAndreas Gohr       $value = $value ? "true" : "false";
358*d5ef99ddSAndreas Gohr    }
359*d5ef99ddSAndreas Gohr
360*d5ef99ddSAndreas Gohr    if ($value === null) $value = 'null';
361*d5ef99ddSAndreas Gohr    if ($value === "'" . self::REMPTY . "'") $value = null;
362*d5ef99ddSAndreas Gohr
363*d5ef99ddSAndreas Gohr    $spaces = str_repeat(' ',$indent);
364*d5ef99ddSAndreas Gohr
365*d5ef99ddSAndreas Gohr    //if (is_int($key) && $key - 1 == $previous_key && $first_key===0) {
366*d5ef99ddSAndreas Gohr    if (is_array ($source_array) && array_keys($source_array) === range(0, count($source_array) - 1)) {
367*d5ef99ddSAndreas Gohr      // It's a sequence
368*d5ef99ddSAndreas Gohr      $string = $spaces.'- '.$value."\n";
369*d5ef99ddSAndreas Gohr    } else {
370*d5ef99ddSAndreas Gohr      // if ($first_key===0)  throw new Exception('Keys are all screwy.  The first one was zero, now it\'s "'. $key .'"');
371*d5ef99ddSAndreas Gohr      // It's mapped
372*d5ef99ddSAndreas Gohr      if (strpos($key, ":") !== false || strpos($key, "#") !== false) { $key = '"' . $key . '"'; }
373*d5ef99ddSAndreas Gohr      $string = rtrim ($spaces.$key.': '.$value)."\n";
374*d5ef99ddSAndreas Gohr    }
375*d5ef99ddSAndreas Gohr    return $string;
376*d5ef99ddSAndreas Gohr  }
377*d5ef99ddSAndreas Gohr
378*d5ef99ddSAndreas Gohr  /**
379*d5ef99ddSAndreas Gohr     * Creates a literal block for dumping
380*d5ef99ddSAndreas Gohr     * @access private
381*d5ef99ddSAndreas Gohr     * @return string
382*d5ef99ddSAndreas Gohr     * @param $value
383*d5ef99ddSAndreas Gohr     * @param $indent int The value of the indent
384*d5ef99ddSAndreas Gohr     */
385*d5ef99ddSAndreas Gohr  private function _doLiteralBlock($value,$indent) {
386*d5ef99ddSAndreas Gohr    if ($value === "\n") return '\n';
387*d5ef99ddSAndreas Gohr    if (strpos($value, "\n") === false && strpos($value, "'") === false) {
388*d5ef99ddSAndreas Gohr      return sprintf ("'%s'", $value);
389*d5ef99ddSAndreas Gohr    }
390*d5ef99ddSAndreas Gohr    if (strpos($value, "\n") === false && strpos($value, '"') === false) {
391*d5ef99ddSAndreas Gohr      return sprintf ('"%s"', $value);
392*d5ef99ddSAndreas Gohr    }
393*d5ef99ddSAndreas Gohr    $exploded = explode("\n",$value);
394*d5ef99ddSAndreas Gohr    $newValue = '|';
395*d5ef99ddSAndreas Gohr    if (isset($exploded[0]) && ($exploded[0] == "|" || $exploded[0] == "|-" || $exploded[0] == ">")) {
396*d5ef99ddSAndreas Gohr        $newValue = $exploded[0];
397*d5ef99ddSAndreas Gohr        unset($exploded[0]);
398*d5ef99ddSAndreas Gohr    }
399*d5ef99ddSAndreas Gohr    $indent += $this->_dumpIndent;
400*d5ef99ddSAndreas Gohr    $spaces   = str_repeat(' ',$indent);
401*d5ef99ddSAndreas Gohr    foreach ($exploded as $line) {
402*d5ef99ddSAndreas Gohr      $line = trim($line);
403*d5ef99ddSAndreas Gohr      if (strpos($line, '"') === 0 && strrpos($line, '"') == (strlen($line)-1) || strpos($line, "'") === 0 && strrpos($line, "'") == (strlen($line)-1)) {
404*d5ef99ddSAndreas Gohr        $line = substr($line, 1, -1);
405*d5ef99ddSAndreas Gohr      }
406*d5ef99ddSAndreas Gohr      $newValue .= "\n" . $spaces . ($line);
407*d5ef99ddSAndreas Gohr    }
408*d5ef99ddSAndreas Gohr    return $newValue;
409*d5ef99ddSAndreas Gohr  }
410*d5ef99ddSAndreas Gohr
411*d5ef99ddSAndreas Gohr  /**
412*d5ef99ddSAndreas Gohr     * Folds a string of text, if necessary
413*d5ef99ddSAndreas Gohr     * @access private
414*d5ef99ddSAndreas Gohr     * @return string
415*d5ef99ddSAndreas Gohr     * @param $value The string you wish to fold
416*d5ef99ddSAndreas Gohr     */
417*d5ef99ddSAndreas Gohr  private function _doFolding($value,$indent) {
418*d5ef99ddSAndreas Gohr    // Don't do anything if wordwrap is set to 0
419*d5ef99ddSAndreas Gohr
420*d5ef99ddSAndreas Gohr    if ($this->_dumpWordWrap !== 0 && is_string ($value) && strlen($value) > $this->_dumpWordWrap) {
421*d5ef99ddSAndreas Gohr      $indent += $this->_dumpIndent;
422*d5ef99ddSAndreas Gohr      $indent = str_repeat(' ',$indent);
423*d5ef99ddSAndreas Gohr      $wrapped = wordwrap($value,$this->_dumpWordWrap,"\n$indent");
424*d5ef99ddSAndreas Gohr      $value   = ">\n".$indent.$wrapped;
425*d5ef99ddSAndreas Gohr    } else {
426*d5ef99ddSAndreas Gohr      if ($this->setting_dump_force_quotes && is_string ($value) && $value !== self::REMPTY)
427*d5ef99ddSAndreas Gohr        $value = '"' . $value . '"';
428*d5ef99ddSAndreas Gohr      if (is_numeric($value) && is_string($value))
429*d5ef99ddSAndreas Gohr        $value = '"' . $value . '"';
430*d5ef99ddSAndreas Gohr    }
431*d5ef99ddSAndreas Gohr
432*d5ef99ddSAndreas Gohr
433*d5ef99ddSAndreas Gohr    return $value;
434*d5ef99ddSAndreas Gohr  }
435*d5ef99ddSAndreas Gohr
436*d5ef99ddSAndreas Gohr  private function isTrueWord($value) {
437*d5ef99ddSAndreas Gohr    $words = self::getTranslations(array('true', 'on', 'yes', 'y'));
438*d5ef99ddSAndreas Gohr    return in_array($value, $words, true);
439*d5ef99ddSAndreas Gohr  }
440*d5ef99ddSAndreas Gohr
441*d5ef99ddSAndreas Gohr  private function isFalseWord($value) {
442*d5ef99ddSAndreas Gohr    $words = self::getTranslations(array('false', 'off', 'no', 'n'));
443*d5ef99ddSAndreas Gohr    return in_array($value, $words, true);
444*d5ef99ddSAndreas Gohr  }
445*d5ef99ddSAndreas Gohr
446*d5ef99ddSAndreas Gohr  private function isNullWord($value) {
447*d5ef99ddSAndreas Gohr    $words = self::getTranslations(array('null', '~'));
448*d5ef99ddSAndreas Gohr    return in_array($value, $words, true);
449*d5ef99ddSAndreas Gohr  }
450*d5ef99ddSAndreas Gohr
451*d5ef99ddSAndreas Gohr  private function isTranslationWord($value) {
452*d5ef99ddSAndreas Gohr    return (
453*d5ef99ddSAndreas Gohr      self::isTrueWord($value)  ||
454*d5ef99ddSAndreas Gohr      self::isFalseWord($value) ||
455*d5ef99ddSAndreas Gohr      self::isNullWord($value)
456*d5ef99ddSAndreas Gohr    );
457*d5ef99ddSAndreas Gohr  }
458*d5ef99ddSAndreas Gohr
459*d5ef99ddSAndreas Gohr  /**
460*d5ef99ddSAndreas Gohr     * Coerce a string into a native type
461*d5ef99ddSAndreas Gohr     * Reference: http://yaml.org/type/bool.html
462*d5ef99ddSAndreas Gohr     * TODO: Use only words from the YAML spec.
463*d5ef99ddSAndreas Gohr     * @access private
464*d5ef99ddSAndreas Gohr     * @param $value The value to coerce
465*d5ef99ddSAndreas Gohr     */
466*d5ef99ddSAndreas Gohr  private function coerceValue(&$value) {
467*d5ef99ddSAndreas Gohr    if (self::isTrueWord($value)) {
468*d5ef99ddSAndreas Gohr      $value = true;
469*d5ef99ddSAndreas Gohr    } else if (self::isFalseWord($value)) {
470*d5ef99ddSAndreas Gohr      $value = false;
471*d5ef99ddSAndreas Gohr    } else if (self::isNullWord($value)) {
472*d5ef99ddSAndreas Gohr      $value = null;
473*d5ef99ddSAndreas Gohr    }
474*d5ef99ddSAndreas Gohr  }
475*d5ef99ddSAndreas Gohr
476*d5ef99ddSAndreas Gohr  /**
477*d5ef99ddSAndreas Gohr     * Given a set of words, perform the appropriate translations on them to
478*d5ef99ddSAndreas Gohr     * match the YAML 1.1 specification for type coercing.
479*d5ef99ddSAndreas Gohr     * @param $words The words to translate
480*d5ef99ddSAndreas Gohr     * @access private
481*d5ef99ddSAndreas Gohr     */
482*d5ef99ddSAndreas Gohr  private static function getTranslations(array $words) {
483*d5ef99ddSAndreas Gohr    $result = array();
484*d5ef99ddSAndreas Gohr    foreach ($words as $i) {
485*d5ef99ddSAndreas Gohr      $result = array_merge($result, array(ucfirst($i), strtoupper($i), strtolower($i)));
486*d5ef99ddSAndreas Gohr    }
487*d5ef99ddSAndreas Gohr    return $result;
488*d5ef99ddSAndreas Gohr  }
489*d5ef99ddSAndreas Gohr
490*d5ef99ddSAndreas Gohr// LOADING FUNCTIONS
491*d5ef99ddSAndreas Gohr
492*d5ef99ddSAndreas Gohr  private function _load($input) {
493*d5ef99ddSAndreas Gohr    $Source = $this->loadFromSource($input);
494*d5ef99ddSAndreas Gohr    return $this->loadWithSource($Source);
495*d5ef99ddSAndreas Gohr  }
496*d5ef99ddSAndreas Gohr
497*d5ef99ddSAndreas Gohr  private function _loadString($input) {
498*d5ef99ddSAndreas Gohr    $Source = $this->loadFromString($input);
499*d5ef99ddSAndreas Gohr    return $this->loadWithSource($Source);
500*d5ef99ddSAndreas Gohr  }
501*d5ef99ddSAndreas Gohr
502*d5ef99ddSAndreas Gohr  private function loadWithSource($Source) {
503*d5ef99ddSAndreas Gohr    if (empty ($Source)) return array();
504*d5ef99ddSAndreas Gohr    if ($this->setting_use_syck_is_possible && function_exists ('syck_load')) {
505*d5ef99ddSAndreas Gohr      $array = syck_load (implode ("\n", $Source));
506*d5ef99ddSAndreas Gohr      return is_array($array) ? $array : array();
507*d5ef99ddSAndreas Gohr    }
508*d5ef99ddSAndreas Gohr
509*d5ef99ddSAndreas Gohr    $this->path = array();
510*d5ef99ddSAndreas Gohr    $this->result = array();
511*d5ef99ddSAndreas Gohr
512*d5ef99ddSAndreas Gohr    $cnt = count($Source);
513*d5ef99ddSAndreas Gohr    for ($i = 0; $i < $cnt; $i++) {
514*d5ef99ddSAndreas Gohr      $line = $Source[$i];
515*d5ef99ddSAndreas Gohr
516*d5ef99ddSAndreas Gohr      $this->indent = strlen($line) - strlen(ltrim($line));
517*d5ef99ddSAndreas Gohr      $tempPath = $this->getParentPathByIndent($this->indent);
518*d5ef99ddSAndreas Gohr      $line = self::stripIndent($line, $this->indent);
519*d5ef99ddSAndreas Gohr      if (self::isComment($line)) continue;
520*d5ef99ddSAndreas Gohr      if (self::isEmpty($line)) continue;
521*d5ef99ddSAndreas Gohr      $this->path = $tempPath;
522*d5ef99ddSAndreas Gohr
523*d5ef99ddSAndreas Gohr      $literalBlockStyle = self::startsLiteralBlock($line);
524*d5ef99ddSAndreas Gohr      if ($literalBlockStyle) {
525*d5ef99ddSAndreas Gohr        $line = rtrim ($line, $literalBlockStyle . " \n");
526*d5ef99ddSAndreas Gohr        $literalBlock = '';
527*d5ef99ddSAndreas Gohr        $line .= ' '.$this->LiteralPlaceHolder;
528*d5ef99ddSAndreas Gohr        $literal_block_indent = strlen($Source[$i+1]) - strlen(ltrim($Source[$i+1]));
529*d5ef99ddSAndreas Gohr        while (++$i < $cnt && $this->literalBlockContinues($Source[$i], $this->indent)) {
530*d5ef99ddSAndreas Gohr          $literalBlock = $this->addLiteralLine($literalBlock, $Source[$i], $literalBlockStyle, $literal_block_indent);
531*d5ef99ddSAndreas Gohr        }
532*d5ef99ddSAndreas Gohr        $i--;
533*d5ef99ddSAndreas Gohr      }
534*d5ef99ddSAndreas Gohr
535*d5ef99ddSAndreas Gohr      // Strip out comments
536*d5ef99ddSAndreas Gohr      if (strpos ($line, '#')) {
537*d5ef99ddSAndreas Gohr          $line = preg_replace('/\s*#([^"\']+)$/','',$line);
538*d5ef99ddSAndreas Gohr      }
539*d5ef99ddSAndreas Gohr
540*d5ef99ddSAndreas Gohr      while (++$i < $cnt && self::greedilyNeedNextLine($line)) {
541*d5ef99ddSAndreas Gohr        $line = rtrim ($line, " \n\t\r") . ' ' . ltrim ($Source[$i], " \t");
542*d5ef99ddSAndreas Gohr      }
543*d5ef99ddSAndreas Gohr      $i--;
544*d5ef99ddSAndreas Gohr
545*d5ef99ddSAndreas Gohr      $lineArray = $this->_parseLine($line);
546*d5ef99ddSAndreas Gohr
547*d5ef99ddSAndreas Gohr      if ($literalBlockStyle)
548*d5ef99ddSAndreas Gohr        $lineArray = $this->revertLiteralPlaceHolder ($lineArray, $literalBlock);
549*d5ef99ddSAndreas Gohr
550*d5ef99ddSAndreas Gohr      $this->addArray($lineArray, $this->indent);
551*d5ef99ddSAndreas Gohr
552*d5ef99ddSAndreas Gohr      foreach ($this->delayedPath as $indent => $delayedPath)
553*d5ef99ddSAndreas Gohr        $this->path[$indent] = $delayedPath;
554*d5ef99ddSAndreas Gohr
555*d5ef99ddSAndreas Gohr      $this->delayedPath = array();
556*d5ef99ddSAndreas Gohr
557*d5ef99ddSAndreas Gohr    }
558*d5ef99ddSAndreas Gohr    return $this->result;
559*d5ef99ddSAndreas Gohr  }
560*d5ef99ddSAndreas Gohr
561*d5ef99ddSAndreas Gohr  private function loadFromSource ($input) {
562*d5ef99ddSAndreas Gohr    if (!empty($input) && strpos($input, "\n") === false && file_exists($input))
563*d5ef99ddSAndreas Gohr      $input = file_get_contents($input);
564*d5ef99ddSAndreas Gohr
565*d5ef99ddSAndreas Gohr    return $this->loadFromString($input);
566*d5ef99ddSAndreas Gohr  }
567*d5ef99ddSAndreas Gohr
568*d5ef99ddSAndreas Gohr  private function loadFromString ($input) {
569*d5ef99ddSAndreas Gohr    $lines = explode("\n",$input);
570*d5ef99ddSAndreas Gohr    foreach ($lines as $k => $_) {
571*d5ef99ddSAndreas Gohr      $lines[$k] = rtrim ($_, "\r");
572*d5ef99ddSAndreas Gohr    }
573*d5ef99ddSAndreas Gohr    return $lines;
574*d5ef99ddSAndreas Gohr  }
575*d5ef99ddSAndreas Gohr
576*d5ef99ddSAndreas Gohr  /**
577*d5ef99ddSAndreas Gohr     * Parses YAML code and returns an array for a node
578*d5ef99ddSAndreas Gohr     * @access private
579*d5ef99ddSAndreas Gohr     * @return array
580*d5ef99ddSAndreas Gohr     * @param string $line A line from the YAML file
581*d5ef99ddSAndreas Gohr     */
582*d5ef99ddSAndreas Gohr  private function _parseLine($line) {
583*d5ef99ddSAndreas Gohr    if (!$line) return array();
584*d5ef99ddSAndreas Gohr    $line = trim($line);
585*d5ef99ddSAndreas Gohr    if (!$line) return array();
586*d5ef99ddSAndreas Gohr
587*d5ef99ddSAndreas Gohr    $array = array();
588*d5ef99ddSAndreas Gohr
589*d5ef99ddSAndreas Gohr    $group = $this->nodeContainsGroup($line);
590*d5ef99ddSAndreas Gohr    if ($group) {
591*d5ef99ddSAndreas Gohr      $this->addGroup($line, $group);
592*d5ef99ddSAndreas Gohr      $line = $this->stripGroup ($line, $group);
593*d5ef99ddSAndreas Gohr    }
594*d5ef99ddSAndreas Gohr
595*d5ef99ddSAndreas Gohr    if ($this->startsMappedSequence($line)) {
596*d5ef99ddSAndreas Gohr      return $this->returnMappedSequence($line);
597*d5ef99ddSAndreas Gohr    }
598*d5ef99ddSAndreas Gohr
599*d5ef99ddSAndreas Gohr    if ($this->startsMappedValue($line)) {
600*d5ef99ddSAndreas Gohr      return $this->returnMappedValue($line);
601*d5ef99ddSAndreas Gohr    }
602*d5ef99ddSAndreas Gohr
603*d5ef99ddSAndreas Gohr    if ($this->isArrayElement($line))
604*d5ef99ddSAndreas Gohr      return $this->returnArrayElement($line);
605*d5ef99ddSAndreas Gohr
606*d5ef99ddSAndreas Gohr    if ($this->isPlainArray($line))
607*d5ef99ddSAndreas Gohr     return $this->returnPlainArray($line);
608*d5ef99ddSAndreas Gohr
609*d5ef99ddSAndreas Gohr    return $this->returnKeyValuePair($line);
610*d5ef99ddSAndreas Gohr
611*d5ef99ddSAndreas Gohr  }
612*d5ef99ddSAndreas Gohr
613*d5ef99ddSAndreas Gohr  /**
614*d5ef99ddSAndreas Gohr     * Finds the type of the passed value, returns the value as the new type.
615*d5ef99ddSAndreas Gohr     * @access private
616*d5ef99ddSAndreas Gohr     * @param string $value
617*d5ef99ddSAndreas Gohr     * @return mixed
618*d5ef99ddSAndreas Gohr     */
619*d5ef99ddSAndreas Gohr  private function _toType($value) {
620*d5ef99ddSAndreas Gohr    if ($value === '') return "";
621*d5ef99ddSAndreas Gohr
622*d5ef99ddSAndreas Gohr    if ($this->setting_empty_hash_as_object && $value === '{}') {
623*d5ef99ddSAndreas Gohr      return new stdClass();
624*d5ef99ddSAndreas Gohr    }
625*d5ef99ddSAndreas Gohr
626*d5ef99ddSAndreas Gohr    $first_character = $value[0];
627*d5ef99ddSAndreas Gohr    $last_character = substr($value, -1, 1);
628*d5ef99ddSAndreas Gohr
629*d5ef99ddSAndreas Gohr    $is_quoted = false;
630*d5ef99ddSAndreas Gohr    do {
631*d5ef99ddSAndreas Gohr      if (!$value) break;
632*d5ef99ddSAndreas Gohr      if ($first_character != '"' && $first_character != "'") break;
633*d5ef99ddSAndreas Gohr      if ($last_character != '"' && $last_character != "'") break;
634*d5ef99ddSAndreas Gohr      $is_quoted = true;
635*d5ef99ddSAndreas Gohr    } while (0);
636*d5ef99ddSAndreas Gohr
637*d5ef99ddSAndreas Gohr    if ($is_quoted) {
638*d5ef99ddSAndreas Gohr      $value = str_replace('\n', "\n", $value);
639*d5ef99ddSAndreas Gohr      if ($first_character == "'")
640*d5ef99ddSAndreas Gohr        return strtr(substr ($value, 1, -1), array ('\'\'' => '\'', '\\\''=> '\''));
641*d5ef99ddSAndreas Gohr      return strtr(substr ($value, 1, -1), array ('\\"' => '"', '\\\''=> '\''));
642*d5ef99ddSAndreas Gohr    }
643*d5ef99ddSAndreas Gohr
644*d5ef99ddSAndreas Gohr    if (strpos($value, ' #') !== false && !$is_quoted)
645*d5ef99ddSAndreas Gohr      $value = preg_replace('/\s+#(.+)$/','',$value);
646*d5ef99ddSAndreas Gohr
647*d5ef99ddSAndreas Gohr    if ($first_character == '[' && $last_character == ']') {
648*d5ef99ddSAndreas Gohr      // Take out strings sequences and mappings
649*d5ef99ddSAndreas Gohr      $innerValue = trim(substr ($value, 1, -1));
650*d5ef99ddSAndreas Gohr      if ($innerValue === '') return array();
651*d5ef99ddSAndreas Gohr      $explode = $this->_inlineEscape($innerValue);
652*d5ef99ddSAndreas Gohr      // Propagate value array
653*d5ef99ddSAndreas Gohr      $value  = array();
654*d5ef99ddSAndreas Gohr      foreach ($explode as $v) {
655*d5ef99ddSAndreas Gohr        $value[] = $this->_toType($v);
656*d5ef99ddSAndreas Gohr      }
657*d5ef99ddSAndreas Gohr      return $value;
658*d5ef99ddSAndreas Gohr    }
659*d5ef99ddSAndreas Gohr
660*d5ef99ddSAndreas Gohr    if (strpos($value,': ')!==false && $first_character != '{') {
661*d5ef99ddSAndreas Gohr      $array = explode(': ',$value);
662*d5ef99ddSAndreas Gohr      $key   = trim($array[0]);
663*d5ef99ddSAndreas Gohr      array_shift($array);
664*d5ef99ddSAndreas Gohr      $value = trim(implode(': ',$array));
665*d5ef99ddSAndreas Gohr      $value = $this->_toType($value);
666*d5ef99ddSAndreas Gohr      return array($key => $value);
667*d5ef99ddSAndreas Gohr    }
668*d5ef99ddSAndreas Gohr
669*d5ef99ddSAndreas Gohr    if ($first_character == '{' && $last_character == '}') {
670*d5ef99ddSAndreas Gohr      $innerValue = trim(substr ($value, 1, -1));
671*d5ef99ddSAndreas Gohr      if ($innerValue === '') return array();
672*d5ef99ddSAndreas Gohr      // Inline Mapping
673*d5ef99ddSAndreas Gohr      // Take out strings sequences and mappings
674*d5ef99ddSAndreas Gohr      $explode = $this->_inlineEscape($innerValue);
675*d5ef99ddSAndreas Gohr      // Propagate value array
676*d5ef99ddSAndreas Gohr      $array = array();
677*d5ef99ddSAndreas Gohr      foreach ($explode as $v) {
678*d5ef99ddSAndreas Gohr        $SubArr = $this->_toType($v);
679*d5ef99ddSAndreas Gohr        if (empty($SubArr)) continue;
680*d5ef99ddSAndreas Gohr        if (is_array ($SubArr)) {
681*d5ef99ddSAndreas Gohr          $array[key($SubArr)] = $SubArr[key($SubArr)]; continue;
682*d5ef99ddSAndreas Gohr        }
683*d5ef99ddSAndreas Gohr        $array[] = $SubArr;
684*d5ef99ddSAndreas Gohr      }
685*d5ef99ddSAndreas Gohr      return $array;
686*d5ef99ddSAndreas Gohr    }
687*d5ef99ddSAndreas Gohr
688*d5ef99ddSAndreas Gohr    if ($value == 'null' || $value == 'NULL' || $value == 'Null' || $value == '' || $value == '~') {
689*d5ef99ddSAndreas Gohr      return null;
690*d5ef99ddSAndreas Gohr    }
691*d5ef99ddSAndreas Gohr
692*d5ef99ddSAndreas Gohr    if ( is_numeric($value) && preg_match ('/^(-|)[1-9]+[0-9]*$/', $value) ){
693*d5ef99ddSAndreas Gohr      $intvalue = (int)$value;
694*d5ef99ddSAndreas Gohr      if ($intvalue != PHP_INT_MAX && $intvalue != ~PHP_INT_MAX)
695*d5ef99ddSAndreas Gohr        $value = $intvalue;
696*d5ef99ddSAndreas Gohr      return $value;
697*d5ef99ddSAndreas Gohr    }
698*d5ef99ddSAndreas Gohr
699*d5ef99ddSAndreas Gohr    if ( is_string($value) && preg_match('/^0[xX][0-9a-fA-F]+$/', $value)) {
700*d5ef99ddSAndreas Gohr      // Hexadecimal value.
701*d5ef99ddSAndreas Gohr      return hexdec($value);
702*d5ef99ddSAndreas Gohr    }
703*d5ef99ddSAndreas Gohr
704*d5ef99ddSAndreas Gohr    $this->coerceValue($value);
705*d5ef99ddSAndreas Gohr
706*d5ef99ddSAndreas Gohr    if (is_numeric($value)) {
707*d5ef99ddSAndreas Gohr      if ($value === '0') return 0;
708*d5ef99ddSAndreas Gohr      if (rtrim ($value, 0) === $value)
709*d5ef99ddSAndreas Gohr        $value = (float)$value;
710*d5ef99ddSAndreas Gohr      return $value;
711*d5ef99ddSAndreas Gohr    }
712*d5ef99ddSAndreas Gohr
713*d5ef99ddSAndreas Gohr    return $value;
714*d5ef99ddSAndreas Gohr  }
715*d5ef99ddSAndreas Gohr
716*d5ef99ddSAndreas Gohr  /**
717*d5ef99ddSAndreas Gohr     * Used in inlines to check for more inlines or quoted strings
718*d5ef99ddSAndreas Gohr     * @access private
719*d5ef99ddSAndreas Gohr     * @return array
720*d5ef99ddSAndreas Gohr     */
721*d5ef99ddSAndreas Gohr  private function _inlineEscape($inline) {
722*d5ef99ddSAndreas Gohr    // There's gotta be a cleaner way to do this...
723*d5ef99ddSAndreas Gohr    // While pure sequences seem to be nesting just fine,
724*d5ef99ddSAndreas Gohr    // pure mappings and mappings with sequences inside can't go very
725*d5ef99ddSAndreas Gohr    // deep.  This needs to be fixed.
726*d5ef99ddSAndreas Gohr
727*d5ef99ddSAndreas Gohr    $seqs = array();
728*d5ef99ddSAndreas Gohr    $maps = array();
729*d5ef99ddSAndreas Gohr    $saved_strings = array();
730*d5ef99ddSAndreas Gohr    $saved_empties = array();
731*d5ef99ddSAndreas Gohr
732*d5ef99ddSAndreas Gohr    // Check for empty strings
733*d5ef99ddSAndreas Gohr    $regex = '/("")|(\'\')/';
734*d5ef99ddSAndreas Gohr    if (preg_match_all($regex,$inline,$strings)) {
735*d5ef99ddSAndreas Gohr      $saved_empties = $strings[0];
736*d5ef99ddSAndreas Gohr      $inline  = preg_replace($regex,'YAMLEmpty',$inline);
737*d5ef99ddSAndreas Gohr    }
738*d5ef99ddSAndreas Gohr    unset($regex);
739*d5ef99ddSAndreas Gohr
740*d5ef99ddSAndreas Gohr    // Check for strings
741*d5ef99ddSAndreas Gohr    $regex = '/(?:(")|(?:\'))((?(1)[^"]+|[^\']+))(?(1)"|\')/';
742*d5ef99ddSAndreas Gohr    if (preg_match_all($regex,$inline,$strings)) {
743*d5ef99ddSAndreas Gohr      $saved_strings = $strings[0];
744*d5ef99ddSAndreas Gohr      $inline  = preg_replace($regex,'YAMLString',$inline);
745*d5ef99ddSAndreas Gohr    }
746*d5ef99ddSAndreas Gohr    unset($regex);
747*d5ef99ddSAndreas Gohr
748*d5ef99ddSAndreas Gohr    // echo $inline;
749*d5ef99ddSAndreas Gohr
750*d5ef99ddSAndreas Gohr    $i = 0;
751*d5ef99ddSAndreas Gohr    do {
752*d5ef99ddSAndreas Gohr
753*d5ef99ddSAndreas Gohr    // Check for sequences
754*d5ef99ddSAndreas Gohr    while (preg_match('/\[([^{}\[\]]+)\]/U',$inline,$matchseqs)) {
755*d5ef99ddSAndreas Gohr      $seqs[] = $matchseqs[0];
756*d5ef99ddSAndreas Gohr      $inline = preg_replace('/\[([^{}\[\]]+)\]/U', ('YAMLSeq' . (count($seqs) - 1) . 's'), $inline, 1);
757*d5ef99ddSAndreas Gohr    }
758*d5ef99ddSAndreas Gohr
759*d5ef99ddSAndreas Gohr    // Check for mappings
760*d5ef99ddSAndreas Gohr    while (preg_match('/{([^\[\]{}]+)}/U',$inline,$matchmaps)) {
761*d5ef99ddSAndreas Gohr      $maps[] = $matchmaps[0];
762*d5ef99ddSAndreas Gohr      $inline = preg_replace('/{([^\[\]{}]+)}/U', ('YAMLMap' . (count($maps) - 1) . 's'), $inline, 1);
763*d5ef99ddSAndreas Gohr    }
764*d5ef99ddSAndreas Gohr
765*d5ef99ddSAndreas Gohr    if ($i++ >= 10) break;
766*d5ef99ddSAndreas Gohr
767*d5ef99ddSAndreas Gohr    } while (strpos ($inline, '[') !== false || strpos ($inline, '{') !== false);
768*d5ef99ddSAndreas Gohr
769*d5ef99ddSAndreas Gohr    $explode = explode(',',$inline);
770*d5ef99ddSAndreas Gohr    $explode = array_map('trim', $explode);
771*d5ef99ddSAndreas Gohr    $stringi = 0; $i = 0;
772*d5ef99ddSAndreas Gohr
773*d5ef99ddSAndreas Gohr    while (1) {
774*d5ef99ddSAndreas Gohr
775*d5ef99ddSAndreas Gohr    // Re-add the sequences
776*d5ef99ddSAndreas Gohr    if (!empty($seqs)) {
777*d5ef99ddSAndreas Gohr      foreach ($explode as $key => $value) {
778*d5ef99ddSAndreas Gohr        if (strpos($value,'YAMLSeq') !== false) {
779*d5ef99ddSAndreas Gohr          foreach ($seqs as $seqk => $seq) {
780*d5ef99ddSAndreas Gohr            $explode[$key] = str_replace(('YAMLSeq'.$seqk.'s'),$seq,$value);
781*d5ef99ddSAndreas Gohr            $value = $explode[$key];
782*d5ef99ddSAndreas Gohr          }
783*d5ef99ddSAndreas Gohr        }
784*d5ef99ddSAndreas Gohr      }
785*d5ef99ddSAndreas Gohr    }
786*d5ef99ddSAndreas Gohr
787*d5ef99ddSAndreas Gohr    // Re-add the mappings
788*d5ef99ddSAndreas Gohr    if (!empty($maps)) {
789*d5ef99ddSAndreas Gohr      foreach ($explode as $key => $value) {
790*d5ef99ddSAndreas Gohr        if (strpos($value,'YAMLMap') !== false) {
791*d5ef99ddSAndreas Gohr          foreach ($maps as $mapk => $map) {
792*d5ef99ddSAndreas Gohr            $explode[$key] = str_replace(('YAMLMap'.$mapk.'s'), $map, $value);
793*d5ef99ddSAndreas Gohr            $value = $explode[$key];
794*d5ef99ddSAndreas Gohr          }
795*d5ef99ddSAndreas Gohr        }
796*d5ef99ddSAndreas Gohr      }
797*d5ef99ddSAndreas Gohr    }
798*d5ef99ddSAndreas Gohr
799*d5ef99ddSAndreas Gohr
800*d5ef99ddSAndreas Gohr    // Re-add the strings
801*d5ef99ddSAndreas Gohr    if (!empty($saved_strings)) {
802*d5ef99ddSAndreas Gohr      foreach ($explode as $key => $value) {
803*d5ef99ddSAndreas Gohr        while (strpos($value,'YAMLString') !== false) {
804*d5ef99ddSAndreas Gohr          $explode[$key] = preg_replace('/YAMLString/',$saved_strings[$stringi],$value, 1);
805*d5ef99ddSAndreas Gohr          unset($saved_strings[$stringi]);
806*d5ef99ddSAndreas Gohr          ++$stringi;
807*d5ef99ddSAndreas Gohr          $value = $explode[$key];
808*d5ef99ddSAndreas Gohr        }
809*d5ef99ddSAndreas Gohr      }
810*d5ef99ddSAndreas Gohr    }
811*d5ef99ddSAndreas Gohr
812*d5ef99ddSAndreas Gohr
813*d5ef99ddSAndreas Gohr    // Re-add the empties
814*d5ef99ddSAndreas Gohr    if (!empty($saved_empties)) {
815*d5ef99ddSAndreas Gohr      foreach ($explode as $key => $value) {
816*d5ef99ddSAndreas Gohr        while (strpos($value,'YAMLEmpty') !== false) {
817*d5ef99ddSAndreas Gohr          $explode[$key] = preg_replace('/YAMLEmpty/', '', $value, 1);
818*d5ef99ddSAndreas Gohr          $value = $explode[$key];
819*d5ef99ddSAndreas Gohr        }
820*d5ef99ddSAndreas Gohr      }
821*d5ef99ddSAndreas Gohr    }
822*d5ef99ddSAndreas Gohr
823*d5ef99ddSAndreas Gohr    $finished = true;
824*d5ef99ddSAndreas Gohr    foreach ($explode as $key => $value) {
825*d5ef99ddSAndreas Gohr      if (strpos($value,'YAMLSeq') !== false) {
826*d5ef99ddSAndreas Gohr        $finished = false; break;
827*d5ef99ddSAndreas Gohr      }
828*d5ef99ddSAndreas Gohr      if (strpos($value,'YAMLMap') !== false) {
829*d5ef99ddSAndreas Gohr        $finished = false; break;
830*d5ef99ddSAndreas Gohr      }
831*d5ef99ddSAndreas Gohr      if (strpos($value,'YAMLString') !== false) {
832*d5ef99ddSAndreas Gohr        $finished = false; break;
833*d5ef99ddSAndreas Gohr      }
834*d5ef99ddSAndreas Gohr      if (strpos($value,'YAMLEmpty') !== false) {
835*d5ef99ddSAndreas Gohr        $finished = false; break;
836*d5ef99ddSAndreas Gohr      }
837*d5ef99ddSAndreas Gohr    }
838*d5ef99ddSAndreas Gohr    if ($finished) break;
839*d5ef99ddSAndreas Gohr
840*d5ef99ddSAndreas Gohr    $i++;
841*d5ef99ddSAndreas Gohr    if ($i > 10)
842*d5ef99ddSAndreas Gohr      break; // Prevent infinite loops.
843*d5ef99ddSAndreas Gohr    }
844*d5ef99ddSAndreas Gohr
845*d5ef99ddSAndreas Gohr
846*d5ef99ddSAndreas Gohr    return $explode;
847*d5ef99ddSAndreas Gohr  }
848*d5ef99ddSAndreas Gohr
849*d5ef99ddSAndreas Gohr  private function literalBlockContinues ($line, $lineIndent) {
850*d5ef99ddSAndreas Gohr    if (!trim($line)) return true;
851*d5ef99ddSAndreas Gohr    if (strlen($line) - strlen(ltrim($line)) > $lineIndent) return true;
852*d5ef99ddSAndreas Gohr    return false;
853*d5ef99ddSAndreas Gohr  }
854*d5ef99ddSAndreas Gohr
855*d5ef99ddSAndreas Gohr  private function referenceContentsByAlias ($alias) {
856*d5ef99ddSAndreas Gohr    do {
857*d5ef99ddSAndreas Gohr      if (!isset($this->SavedGroups[$alias])) { echo "Bad group name: $alias."; break; }
858*d5ef99ddSAndreas Gohr      $groupPath = $this->SavedGroups[$alias];
859*d5ef99ddSAndreas Gohr      $value = $this->result;
860*d5ef99ddSAndreas Gohr      foreach ($groupPath as $k) {
861*d5ef99ddSAndreas Gohr        $value = $value[$k];
862*d5ef99ddSAndreas Gohr      }
863*d5ef99ddSAndreas Gohr    } while (false);
864*d5ef99ddSAndreas Gohr    return $value;
865*d5ef99ddSAndreas Gohr  }
866*d5ef99ddSAndreas Gohr
867*d5ef99ddSAndreas Gohr  private function addArrayInline ($array, $indent) {
868*d5ef99ddSAndreas Gohr      $CommonGroupPath = $this->path;
869*d5ef99ddSAndreas Gohr      if (empty ($array)) return false;
870*d5ef99ddSAndreas Gohr
871*d5ef99ddSAndreas Gohr      foreach ($array as $k => $_) {
872*d5ef99ddSAndreas Gohr        $this->addArray(array($k => $_), $indent);
873*d5ef99ddSAndreas Gohr        $this->path = $CommonGroupPath;
874*d5ef99ddSAndreas Gohr      }
875*d5ef99ddSAndreas Gohr      return true;
876*d5ef99ddSAndreas Gohr  }
877*d5ef99ddSAndreas Gohr
878*d5ef99ddSAndreas Gohr  private function addArray ($incoming_data, $incoming_indent) {
879*d5ef99ddSAndreas Gohr
880*d5ef99ddSAndreas Gohr   // print_r ($incoming_data);
881*d5ef99ddSAndreas Gohr
882*d5ef99ddSAndreas Gohr    if (count ($incoming_data) > 1)
883*d5ef99ddSAndreas Gohr      return $this->addArrayInline ($incoming_data, $incoming_indent);
884*d5ef99ddSAndreas Gohr
885*d5ef99ddSAndreas Gohr    $key = key ($incoming_data);
886*d5ef99ddSAndreas Gohr    $value = isset($incoming_data[$key]) ? $incoming_data[$key] : null;
887*d5ef99ddSAndreas Gohr    if ($key === '__!YAMLZero') $key = '0';
888*d5ef99ddSAndreas Gohr
889*d5ef99ddSAndreas Gohr    if ($incoming_indent == 0 && !$this->_containsGroupAlias && !$this->_containsGroupAnchor) { // Shortcut for root-level values.
890*d5ef99ddSAndreas Gohr      if ($key || $key === '' || $key === '0') {
891*d5ef99ddSAndreas Gohr        $this->result[$key] = $value;
892*d5ef99ddSAndreas Gohr      } else {
893*d5ef99ddSAndreas Gohr        $this->result[] = $value; end ($this->result); $key = key ($this->result);
894*d5ef99ddSAndreas Gohr      }
895*d5ef99ddSAndreas Gohr      $this->path[$incoming_indent] = $key;
896*d5ef99ddSAndreas Gohr      return;
897*d5ef99ddSAndreas Gohr    }
898*d5ef99ddSAndreas Gohr
899*d5ef99ddSAndreas Gohr
900*d5ef99ddSAndreas Gohr
901*d5ef99ddSAndreas Gohr    $history = array();
902*d5ef99ddSAndreas Gohr    // Unfolding inner array tree.
903*d5ef99ddSAndreas Gohr    $history[] = $_arr = $this->result;
904*d5ef99ddSAndreas Gohr    foreach ($this->path as $k) {
905*d5ef99ddSAndreas Gohr      $history[] = $_arr = $_arr[$k];
906*d5ef99ddSAndreas Gohr    }
907*d5ef99ddSAndreas Gohr
908*d5ef99ddSAndreas Gohr    if ($this->_containsGroupAlias) {
909*d5ef99ddSAndreas Gohr      $value = $this->referenceContentsByAlias($this->_containsGroupAlias);
910*d5ef99ddSAndreas Gohr      $this->_containsGroupAlias = false;
911*d5ef99ddSAndreas Gohr    }
912*d5ef99ddSAndreas Gohr
913*d5ef99ddSAndreas Gohr
914*d5ef99ddSAndreas Gohr    // Adding string or numeric key to the innermost level or $this->arr.
915*d5ef99ddSAndreas Gohr    if (is_string($key) && $key == '<<') {
916*d5ef99ddSAndreas Gohr      if (!is_array ($_arr)) { $_arr = array (); }
917*d5ef99ddSAndreas Gohr
918*d5ef99ddSAndreas Gohr      $_arr = array_merge ($_arr, $value);
919*d5ef99ddSAndreas Gohr    } else if ($key || $key === '' || $key === '0') {
920*d5ef99ddSAndreas Gohr      if (!is_array ($_arr))
921*d5ef99ddSAndreas Gohr        $_arr = array ($key=>$value);
922*d5ef99ddSAndreas Gohr      else
923*d5ef99ddSAndreas Gohr        $_arr[$key] = $value;
924*d5ef99ddSAndreas Gohr    } else {
925*d5ef99ddSAndreas Gohr      if (!is_array ($_arr)) { $_arr = array ($value); $key = 0; }
926*d5ef99ddSAndreas Gohr      else { $_arr[] = $value; end ($_arr); $key = key ($_arr); }
927*d5ef99ddSAndreas Gohr    }
928*d5ef99ddSAndreas Gohr
929*d5ef99ddSAndreas Gohr    $reverse_path = array_reverse($this->path);
930*d5ef99ddSAndreas Gohr    $reverse_history = array_reverse ($history);
931*d5ef99ddSAndreas Gohr    $reverse_history[0] = $_arr;
932*d5ef99ddSAndreas Gohr    $cnt = count($reverse_history) - 1;
933*d5ef99ddSAndreas Gohr    for ($i = 0; $i < $cnt; $i++) {
934*d5ef99ddSAndreas Gohr      $reverse_history[$i+1][$reverse_path[$i]] = $reverse_history[$i];
935*d5ef99ddSAndreas Gohr    }
936*d5ef99ddSAndreas Gohr    $this->result = $reverse_history[$cnt];
937*d5ef99ddSAndreas Gohr
938*d5ef99ddSAndreas Gohr    $this->path[$incoming_indent] = $key;
939*d5ef99ddSAndreas Gohr
940*d5ef99ddSAndreas Gohr    if ($this->_containsGroupAnchor) {
941*d5ef99ddSAndreas Gohr      $this->SavedGroups[$this->_containsGroupAnchor] = $this->path;
942*d5ef99ddSAndreas Gohr      if (is_array ($value)) {
943*d5ef99ddSAndreas Gohr        $k = key ($value);
944*d5ef99ddSAndreas Gohr        if (!is_int ($k)) {
945*d5ef99ddSAndreas Gohr          $this->SavedGroups[$this->_containsGroupAnchor][$incoming_indent + 2] = $k;
946*d5ef99ddSAndreas Gohr        }
947*d5ef99ddSAndreas Gohr      }
948*d5ef99ddSAndreas Gohr      $this->_containsGroupAnchor = false;
949*d5ef99ddSAndreas Gohr    }
950*d5ef99ddSAndreas Gohr
951*d5ef99ddSAndreas Gohr  }
952*d5ef99ddSAndreas Gohr
953*d5ef99ddSAndreas Gohr  private static function startsLiteralBlock ($line) {
954*d5ef99ddSAndreas Gohr    $lastChar = substr (trim($line), -1);
955*d5ef99ddSAndreas Gohr    if ($lastChar != '>' && $lastChar != '|') return false;
956*d5ef99ddSAndreas Gohr    if ($lastChar == '|') return $lastChar;
957*d5ef99ddSAndreas Gohr    // HTML tags should not be counted as literal blocks.
958*d5ef99ddSAndreas Gohr    if (preg_match ('#<.*?>$#', $line)) return false;
959*d5ef99ddSAndreas Gohr    return $lastChar;
960*d5ef99ddSAndreas Gohr  }
961*d5ef99ddSAndreas Gohr
962*d5ef99ddSAndreas Gohr  private static function greedilyNeedNextLine($line) {
963*d5ef99ddSAndreas Gohr    $line = trim ($line);
964*d5ef99ddSAndreas Gohr    if (!strlen($line)) return false;
965*d5ef99ddSAndreas Gohr    if (substr ($line, -1, 1) == ']') return false;
966*d5ef99ddSAndreas Gohr    if ($line[0] == '[') return true;
967*d5ef99ddSAndreas Gohr    if (preg_match ('#^[^:]+?:\s*\[#', $line)) return true;
968*d5ef99ddSAndreas Gohr    return false;
969*d5ef99ddSAndreas Gohr  }
970*d5ef99ddSAndreas Gohr
971*d5ef99ddSAndreas Gohr  private function addLiteralLine ($literalBlock, $line, $literalBlockStyle, $indent = -1) {
972*d5ef99ddSAndreas Gohr    $line = self::stripIndent($line, $indent);
973*d5ef99ddSAndreas Gohr    if ($literalBlockStyle !== '|') {
974*d5ef99ddSAndreas Gohr        $line = self::stripIndent($line);
975*d5ef99ddSAndreas Gohr    }
976*d5ef99ddSAndreas Gohr    $line = rtrim ($line, "\r\n\t ") . "\n";
977*d5ef99ddSAndreas Gohr    if ($literalBlockStyle == '|') {
978*d5ef99ddSAndreas Gohr      return $literalBlock . $line;
979*d5ef99ddSAndreas Gohr    }
980*d5ef99ddSAndreas Gohr    if (strlen($line) == 0)
981*d5ef99ddSAndreas Gohr      return rtrim($literalBlock, ' ') . "\n";
982*d5ef99ddSAndreas Gohr    if ($line == "\n" && $literalBlockStyle == '>') {
983*d5ef99ddSAndreas Gohr      return rtrim ($literalBlock, " \t") . "\n";
984*d5ef99ddSAndreas Gohr    }
985*d5ef99ddSAndreas Gohr    if ($line != "\n")
986*d5ef99ddSAndreas Gohr      $line = trim ($line, "\r\n ") . " ";
987*d5ef99ddSAndreas Gohr    return $literalBlock . $line;
988*d5ef99ddSAndreas Gohr  }
989*d5ef99ddSAndreas Gohr
990*d5ef99ddSAndreas Gohr   function revertLiteralPlaceHolder ($lineArray, $literalBlock) {
991*d5ef99ddSAndreas Gohr     foreach ($lineArray as $k => $_) {
992*d5ef99ddSAndreas Gohr      if (is_array($_))
993*d5ef99ddSAndreas Gohr        $lineArray[$k] = $this->revertLiteralPlaceHolder ($_, $literalBlock);
994*d5ef99ddSAndreas Gohr      else if (substr($_, -1 * strlen ($this->LiteralPlaceHolder)) == $this->LiteralPlaceHolder)
995*d5ef99ddSAndreas Gohr	       $lineArray[$k] = rtrim ($literalBlock, " \r\n");
996*d5ef99ddSAndreas Gohr     }
997*d5ef99ddSAndreas Gohr     return $lineArray;
998*d5ef99ddSAndreas Gohr   }
999*d5ef99ddSAndreas Gohr
1000*d5ef99ddSAndreas Gohr  private static function stripIndent ($line, $indent = -1) {
1001*d5ef99ddSAndreas Gohr    if ($indent == -1) $indent = strlen($line) - strlen(ltrim($line));
1002*d5ef99ddSAndreas Gohr    return substr ($line, $indent);
1003*d5ef99ddSAndreas Gohr  }
1004*d5ef99ddSAndreas Gohr
1005*d5ef99ddSAndreas Gohr  private function getParentPathByIndent ($indent) {
1006*d5ef99ddSAndreas Gohr    if ($indent == 0) return array();
1007*d5ef99ddSAndreas Gohr    $linePath = $this->path;
1008*d5ef99ddSAndreas Gohr    do {
1009*d5ef99ddSAndreas Gohr      end($linePath); $lastIndentInParentPath = key($linePath);
1010*d5ef99ddSAndreas Gohr      if ($indent <= $lastIndentInParentPath) array_pop ($linePath);
1011*d5ef99ddSAndreas Gohr    } while ($indent <= $lastIndentInParentPath);
1012*d5ef99ddSAndreas Gohr    return $linePath;
1013*d5ef99ddSAndreas Gohr  }
1014*d5ef99ddSAndreas Gohr
1015*d5ef99ddSAndreas Gohr
1016*d5ef99ddSAndreas Gohr  private function clearBiggerPathValues ($indent) {
1017*d5ef99ddSAndreas Gohr
1018*d5ef99ddSAndreas Gohr
1019*d5ef99ddSAndreas Gohr    if ($indent == 0) $this->path = array();
1020*d5ef99ddSAndreas Gohr    if (empty ($this->path)) return true;
1021*d5ef99ddSAndreas Gohr
1022*d5ef99ddSAndreas Gohr    foreach ($this->path as $k => $_) {
1023*d5ef99ddSAndreas Gohr      if ($k > $indent) unset ($this->path[$k]);
1024*d5ef99ddSAndreas Gohr    }
1025*d5ef99ddSAndreas Gohr
1026*d5ef99ddSAndreas Gohr    return true;
1027*d5ef99ddSAndreas Gohr  }
1028*d5ef99ddSAndreas Gohr
1029*d5ef99ddSAndreas Gohr
1030*d5ef99ddSAndreas Gohr  private static function isComment ($line) {
1031*d5ef99ddSAndreas Gohr    if (!$line) return false;
1032*d5ef99ddSAndreas Gohr    if ($line[0] == '#') return true;
1033*d5ef99ddSAndreas Gohr    if (trim($line, " \r\n\t") == '---') return true;
1034*d5ef99ddSAndreas Gohr    return false;
1035*d5ef99ddSAndreas Gohr  }
1036*d5ef99ddSAndreas Gohr
1037*d5ef99ddSAndreas Gohr  private static function isEmpty ($line) {
1038*d5ef99ddSAndreas Gohr    return (trim ($line) === '');
1039*d5ef99ddSAndreas Gohr  }
1040*d5ef99ddSAndreas Gohr
1041*d5ef99ddSAndreas Gohr
1042*d5ef99ddSAndreas Gohr  private function isArrayElement ($line) {
1043*d5ef99ddSAndreas Gohr    if (!$line || !is_scalar($line)) return false;
1044*d5ef99ddSAndreas Gohr    if (substr($line, 0, 2) != '- ') return false;
1045*d5ef99ddSAndreas Gohr    if (strlen ($line) > 3)
1046*d5ef99ddSAndreas Gohr      if (substr($line,0,3) == '---') return false;
1047*d5ef99ddSAndreas Gohr
1048*d5ef99ddSAndreas Gohr    return true;
1049*d5ef99ddSAndreas Gohr  }
1050*d5ef99ddSAndreas Gohr
1051*d5ef99ddSAndreas Gohr  private function isHashElement ($line) {
1052*d5ef99ddSAndreas Gohr    return strpos($line, ':');
1053*d5ef99ddSAndreas Gohr  }
1054*d5ef99ddSAndreas Gohr
1055*d5ef99ddSAndreas Gohr  private function isLiteral ($line) {
1056*d5ef99ddSAndreas Gohr    if ($this->isArrayElement($line)) return false;
1057*d5ef99ddSAndreas Gohr    if ($this->isHashElement($line)) return false;
1058*d5ef99ddSAndreas Gohr    return true;
1059*d5ef99ddSAndreas Gohr  }
1060*d5ef99ddSAndreas Gohr
1061*d5ef99ddSAndreas Gohr
1062*d5ef99ddSAndreas Gohr  private static function unquote ($value) {
1063*d5ef99ddSAndreas Gohr    if (!$value) return $value;
1064*d5ef99ddSAndreas Gohr    if (!is_string($value)) return $value;
1065*d5ef99ddSAndreas Gohr    if ($value[0] == '\'') return trim ($value, '\'');
1066*d5ef99ddSAndreas Gohr    if ($value[0] == '"') return trim ($value, '"');
1067*d5ef99ddSAndreas Gohr    return $value;
1068*d5ef99ddSAndreas Gohr  }
1069*d5ef99ddSAndreas Gohr
1070*d5ef99ddSAndreas Gohr  private function startsMappedSequence ($line) {
1071*d5ef99ddSAndreas Gohr    return (substr($line, 0, 2) == '- ' && substr ($line, -1, 1) == ':');
1072*d5ef99ddSAndreas Gohr  }
1073*d5ef99ddSAndreas Gohr
1074*d5ef99ddSAndreas Gohr  private function returnMappedSequence ($line) {
1075*d5ef99ddSAndreas Gohr    $array = array();
1076*d5ef99ddSAndreas Gohr    $key         = self::unquote(trim(substr($line,1,-1)));
1077*d5ef99ddSAndreas Gohr    $array[$key] = array();
1078*d5ef99ddSAndreas Gohr    $this->delayedPath = array(strpos ($line, $key) + $this->indent => $key);
1079*d5ef99ddSAndreas Gohr    return array($array);
1080*d5ef99ddSAndreas Gohr  }
1081*d5ef99ddSAndreas Gohr
1082*d5ef99ddSAndreas Gohr  private function checkKeysInValue($value) {
1083*d5ef99ddSAndreas Gohr    if (strchr('[{"\'', $value[0]) === false) {
1084*d5ef99ddSAndreas Gohr      if (strchr($value, ': ') !== false) {
1085*d5ef99ddSAndreas Gohr          throw new Exception('Too many keys: '.$value);
1086*d5ef99ddSAndreas Gohr      }
1087*d5ef99ddSAndreas Gohr    }
1088*d5ef99ddSAndreas Gohr  }
1089*d5ef99ddSAndreas Gohr
1090*d5ef99ddSAndreas Gohr  private function returnMappedValue ($line) {
1091*d5ef99ddSAndreas Gohr    $this->checkKeysInValue($line);
1092*d5ef99ddSAndreas Gohr    $array = array();
1093*d5ef99ddSAndreas Gohr    $key         = self::unquote (trim(substr($line,0,-1)));
1094*d5ef99ddSAndreas Gohr    $array[$key] = '';
1095*d5ef99ddSAndreas Gohr    return $array;
1096*d5ef99ddSAndreas Gohr  }
1097*d5ef99ddSAndreas Gohr
1098*d5ef99ddSAndreas Gohr  private function startsMappedValue ($line) {
1099*d5ef99ddSAndreas Gohr    return (substr ($line, -1, 1) == ':');
1100*d5ef99ddSAndreas Gohr  }
1101*d5ef99ddSAndreas Gohr
1102*d5ef99ddSAndreas Gohr  private function isPlainArray ($line) {
1103*d5ef99ddSAndreas Gohr    return ($line[0] == '[' && substr ($line, -1, 1) == ']');
1104*d5ef99ddSAndreas Gohr  }
1105*d5ef99ddSAndreas Gohr
1106*d5ef99ddSAndreas Gohr  private function returnPlainArray ($line) {
1107*d5ef99ddSAndreas Gohr    return $this->_toType($line);
1108*d5ef99ddSAndreas Gohr  }
1109*d5ef99ddSAndreas Gohr
1110*d5ef99ddSAndreas Gohr  private function returnKeyValuePair ($line) {
1111*d5ef99ddSAndreas Gohr    $array = array();
1112*d5ef99ddSAndreas Gohr    $key = '';
1113*d5ef99ddSAndreas Gohr    if (strpos ($line, ': ')) {
1114*d5ef99ddSAndreas Gohr      // It's a key/value pair most likely
1115*d5ef99ddSAndreas Gohr      // If the key is in double quotes pull it out
1116*d5ef99ddSAndreas Gohr      if (($line[0] == '"' || $line[0] == "'") && preg_match('/^(["\'](.*)["\'](\s)*:)/',$line,$matches)) {
1117*d5ef99ddSAndreas Gohr        $value = trim(str_replace($matches[1],'',$line));
1118*d5ef99ddSAndreas Gohr        $key   = $matches[2];
1119*d5ef99ddSAndreas Gohr      } else {
1120*d5ef99ddSAndreas Gohr        // Do some guesswork as to the key and the value
1121*d5ef99ddSAndreas Gohr        $explode = explode(': ', $line);
1122*d5ef99ddSAndreas Gohr        $key     = trim(array_shift($explode));
1123*d5ef99ddSAndreas Gohr        $value   = trim(implode(': ', $explode));
1124*d5ef99ddSAndreas Gohr        $this->checkKeysInValue($value);
1125*d5ef99ddSAndreas Gohr      }
1126*d5ef99ddSAndreas Gohr      // Set the type of the value.  Int, string, etc
1127*d5ef99ddSAndreas Gohr      $value = $this->_toType($value);
1128*d5ef99ddSAndreas Gohr
1129*d5ef99ddSAndreas Gohr      if ($key === '0') $key = '__!YAMLZero';
1130*d5ef99ddSAndreas Gohr      $array[$key] = $value;
1131*d5ef99ddSAndreas Gohr    } else {
1132*d5ef99ddSAndreas Gohr      $array = array ($line);
1133*d5ef99ddSAndreas Gohr    }
1134*d5ef99ddSAndreas Gohr    return $array;
1135*d5ef99ddSAndreas Gohr
1136*d5ef99ddSAndreas Gohr  }
1137*d5ef99ddSAndreas Gohr
1138*d5ef99ddSAndreas Gohr
1139*d5ef99ddSAndreas Gohr  private function returnArrayElement ($line) {
1140*d5ef99ddSAndreas Gohr     if (strlen($line) <= 1) return array(array()); // Weird %)
1141*d5ef99ddSAndreas Gohr     $array = array();
1142*d5ef99ddSAndreas Gohr     $value   = trim(substr($line,1));
1143*d5ef99ddSAndreas Gohr     $value   = $this->_toType($value);
1144*d5ef99ddSAndreas Gohr     if ($this->isArrayElement($value)) {
1145*d5ef99ddSAndreas Gohr       $value = $this->returnArrayElement($value);
1146*d5ef99ddSAndreas Gohr     }
1147*d5ef99ddSAndreas Gohr     $array[] = $value;
1148*d5ef99ddSAndreas Gohr     return $array;
1149*d5ef99ddSAndreas Gohr  }
1150*d5ef99ddSAndreas Gohr
1151*d5ef99ddSAndreas Gohr
1152*d5ef99ddSAndreas Gohr  private function nodeContainsGroup ($line) {
1153*d5ef99ddSAndreas Gohr    $symbolsForReference = 'A-z0-9_\-';
1154*d5ef99ddSAndreas Gohr    if (strpos($line, '&') === false && strpos($line, '*') === false) return false; // Please die fast ;-)
1155*d5ef99ddSAndreas Gohr    if ($line[0] == '&' && preg_match('/^(&['.$symbolsForReference.']+)/', $line, $matches)) return $matches[1];
1156*d5ef99ddSAndreas Gohr    if ($line[0] == '*' && preg_match('/^(\*['.$symbolsForReference.']+)/', $line, $matches)) return $matches[1];
1157*d5ef99ddSAndreas Gohr    if (preg_match('/(&['.$symbolsForReference.']+)$/', $line, $matches)) return $matches[1];
1158*d5ef99ddSAndreas Gohr    if (preg_match('/(\*['.$symbolsForReference.']+$)/', $line, $matches)) return $matches[1];
1159*d5ef99ddSAndreas Gohr    if (preg_match ('#^\s*<<\s*:\s*(\*[^\s]+).*$#', $line, $matches)) return $matches[1];
1160*d5ef99ddSAndreas Gohr    return false;
1161*d5ef99ddSAndreas Gohr
1162*d5ef99ddSAndreas Gohr  }
1163*d5ef99ddSAndreas Gohr
1164*d5ef99ddSAndreas Gohr  private function addGroup ($line, $group) {
1165*d5ef99ddSAndreas Gohr    if ($group[0] == '&') $this->_containsGroupAnchor = substr ($group, 1);
1166*d5ef99ddSAndreas Gohr    if ($group[0] == '*') $this->_containsGroupAlias = substr ($group, 1);
1167*d5ef99ddSAndreas Gohr    //print_r ($this->path);
1168*d5ef99ddSAndreas Gohr  }
1169*d5ef99ddSAndreas Gohr
1170*d5ef99ddSAndreas Gohr  private function stripGroup ($line, $group) {
1171*d5ef99ddSAndreas Gohr    $line = trim(str_replace($group, '', $line));
1172*d5ef99ddSAndreas Gohr    return $line;
1173*d5ef99ddSAndreas Gohr  }
1174*d5ef99ddSAndreas Gohr}
1175*d5ef99ddSAndreas Gohr}
1176*d5ef99ddSAndreas Gohr
1177*d5ef99ddSAndreas Gohr// Enable use of Spyc from command line
1178*d5ef99ddSAndreas Gohr// The syntax is the following: php Spyc.php spyc.yaml
1179*d5ef99ddSAndreas Gohr
1180*d5ef99ddSAndreas Gohrdo {
1181*d5ef99ddSAndreas Gohr  if (PHP_SAPI != 'cli') break;
1182*d5ef99ddSAndreas Gohr  if (empty ($_SERVER['argc']) || $_SERVER['argc'] < 2) break;
1183*d5ef99ddSAndreas Gohr  if (empty ($_SERVER['PHP_SELF']) || FALSE === strpos ($_SERVER['PHP_SELF'], 'Spyc.php') ) break;
1184*d5ef99ddSAndreas Gohr  $file = $argv[1];
1185*d5ef99ddSAndreas Gohr  echo json_encode (spyc_load_file ($file));
1186*d5ef99ddSAndreas Gohr} while (0);
1187