mirror of
				https://github.com/Icinga/icingaweb2.git
				synced 2025-10-26 17:04:15 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			311 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			311 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| <?php
 | |
| /* Icinga Web 2 | (c) 2015 Icinga Development Team | GPLv2+ */
 | |
| 
 | |
| namespace Icinga\File\Ini;
 | |
| 
 | |
| use ErrorException;
 | |
| use Icinga\File\Ini\Dom\Section;
 | |
| use Icinga\File\Ini\Dom\Comment;
 | |
| use Icinga\File\Ini\Dom\Document;
 | |
| use Icinga\File\Ini\Dom\Directive;
 | |
| use Icinga\Application\Logger;
 | |
| use Icinga\Exception\ConfigurationError;
 | |
| use Icinga\Exception\NotReadableError;
 | |
| use Icinga\Application\Config;
 | |
| 
 | |
| class IniParser
 | |
| {
 | |
|     const LINE_START = 0;
 | |
|     const SECTION = 1;
 | |
|     const ESCAPE = 2;
 | |
|     const DIRECTIVE_KEY = 4;
 | |
|     const DIRECTIVE_VALUE_START = 5;
 | |
|     const DIRECTIVE_VALUE = 6;
 | |
|     const DIRECTIVE_VALUE_QUOTED = 7;
 | |
|     const COMMENT = 8;
 | |
|     const COMMENT_END = 9;
 | |
|     const LINE_END = 10;
 | |
| 
 | |
|     /**
 | |
|      * Cancel the parsing with an error
 | |
|      *
 | |
|      * @param $message  The error description
 | |
|      * @param $line     The line in which the error occured
 | |
|      *
 | |
|      * @throws ConfigurationError
 | |
|      */
 | |
|     private static function throwParseError($message, $line)
 | |
|     {
 | |
|         throw new ConfigurationError(sprintf('Ini parser error: %s. (l. %d)', $message, $line));
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Read the ini file contained in a string and return a mutable DOM that can be used
 | |
|      * to change the content of an INI file.
 | |
|      *
 | |
|      * @param $str                  A string containing the whole ini file
 | |
|      *
 | |
|      * @return Document             The mutable DOM object.
 | |
|      * @throws ConfigurationError   In case the file is not parseable
 | |
|      */
 | |
|     public static function parseIni($str)
 | |
|     {
 | |
|         $doc = new Document();
 | |
|         $sec = null;
 | |
|         $dir = null;
 | |
|         $coms = array();
 | |
|         $state = self::LINE_START;
 | |
|         $escaping = null;
 | |
|         $token = '';
 | |
|         $line = 0;
 | |
| 
 | |
|         for ($i = 0; $i < strlen($str); $i++) {
 | |
|             $s = $str[$i];
 | |
|             switch ($state) {
 | |
|                 case self::LINE_START:
 | |
|                     if (ctype_space($s)) {
 | |
|                         continue 2;
 | |
|                     }
 | |
|                     switch ($s) {
 | |
|                         case '[':
 | |
|                             $state = self::SECTION;
 | |
|                             break;
 | |
|                         case ';':
 | |
|                             $state = self::COMMENT;
 | |
|                             break;
 | |
|                         default:
 | |
|                             $state = self::DIRECTIVE_KEY;
 | |
|                             $token = $s;
 | |
|                             break;
 | |
|                     }
 | |
|                     break;
 | |
| 
 | |
|                 case self::ESCAPE:
 | |
|                     $token .= $s;
 | |
|                     $state = $escaping;
 | |
|                     $escaping = null;
 | |
|                     break;
 | |
| 
 | |
|                 case self::SECTION:
 | |
|                     if ($s === "\n") {
 | |
|                         self::throwParseError('Unterminated SECTION', $line);
 | |
|                     } elseif ($s === '\\') {
 | |
|                         $state = self::ESCAPE;
 | |
|                         $escaping = self::SECTION;
 | |
|                     } elseif ($s !== ']') {
 | |
|                         $token .= $s;
 | |
|                     } else {
 | |
|                         $sec = new Section($token);
 | |
|                         $sec->setCommentsPre($coms);
 | |
|                         $doc->addSection($sec);
 | |
|                         $dir = null;
 | |
|                         $coms = array();
 | |
| 
 | |
|                         $state = self::LINE_END;
 | |
|                         $token = '';
 | |
|                     }
 | |
|                     break;
 | |
| 
 | |
|                 case self::DIRECTIVE_KEY:
 | |
|                     if ($s !== '=') {
 | |
|                         $token .= $s;
 | |
|                     } else {
 | |
|                         $dir = new Directive($token);
 | |
|                         $dir->setCommentsPre($coms);
 | |
|                         if (isset($sec)) {
 | |
|                             $sec->addDirective($dir);
 | |
|                         } else {
 | |
|                             Logger::warning(sprintf(
 | |
|                                 'Ini parser warning: section-less directive "%s" ignored. (l. %d)',
 | |
|                                 $token,
 | |
|                                 $line
 | |
|                             ));
 | |
|                         }
 | |
| 
 | |
|                         $coms = array();
 | |
|                         $state = self::DIRECTIVE_VALUE_START;
 | |
|                         $token = '';
 | |
|                     }
 | |
|                     break;
 | |
| 
 | |
|                 case self::DIRECTIVE_VALUE_START:
 | |
|                     if (ctype_space($s)) {
 | |
|                         continue 2;
 | |
|                     } elseif ($s === '"') {
 | |
|                         $state = self::DIRECTIVE_VALUE_QUOTED;
 | |
|                     } else {
 | |
|                         $state = self::DIRECTIVE_VALUE;
 | |
|                         $token = $s;
 | |
|                     }
 | |
|                     break;
 | |
| 
 | |
|                 case self::DIRECTIVE_VALUE:
 | |
|                     /*
 | |
|                         Escaping non-quoted values is not supported by php_parse_ini, it might
 | |
|                         be reasonable to include in case we are switching completely our own
 | |
|                         parser implementation
 | |
|                     */
 | |
|                     if ($s === "\n" || $s === ";") {
 | |
|                         $dir->setValue($token);
 | |
|                         $token = '';
 | |
| 
 | |
|                         if ($s === "\n") {
 | |
|                             $state = self::LINE_START;
 | |
|                             $line ++;
 | |
|                         } elseif ($s === ';') {
 | |
|                             $state = self::COMMENT;
 | |
|                         }
 | |
|                     } else {
 | |
|                         $token .= $s;
 | |
|                     }
 | |
|                     break;
 | |
| 
 | |
|                 case self::DIRECTIVE_VALUE_QUOTED:
 | |
|                     if ($s === '\\') {
 | |
|                         $state = self::ESCAPE;
 | |
|                         $escaping = self::DIRECTIVE_VALUE_QUOTED;
 | |
|                     } elseif ($s !== '"') {
 | |
|                         $token .= $s;
 | |
|                     } else {
 | |
|                         $dir->setValue($token);
 | |
|                         $token = '';
 | |
|                         $state = self::LINE_END;
 | |
|                     }
 | |
|                     break;
 | |
| 
 | |
|                 case self::COMMENT:
 | |
|                 case self::COMMENT_END:
 | |
|                     if ($s !== "\n") {
 | |
|                         $token .= $s;
 | |
|                     } else {
 | |
|                         $com = new Comment();
 | |
|                         $com->setContent($token);
 | |
|                         $token = '';
 | |
| 
 | |
|                         // Comments at the line end belong to the current line's directive or section. Comments
 | |
|                         // on empty lines belong to the next directive that shows up.
 | |
|                         if ($state === self::COMMENT_END) {
 | |
|                             if (isset($dir)) {
 | |
|                                 $dir->setCommentPost($com);
 | |
|                             } else {
 | |
|                                 $sec->setCommentPost($com);
 | |
|                             }
 | |
|                         } else {
 | |
|                             $coms[] = $com;
 | |
|                         }
 | |
|                         $state = self::LINE_START;
 | |
|                         $line ++;
 | |
|                     }
 | |
|                     break;
 | |
| 
 | |
|                 case self::LINE_END:
 | |
|                     if ($s === "\n") {
 | |
|                         $state = self::LINE_START;
 | |
|                         $line ++;
 | |
|                     } elseif ($s === ';') {
 | |
|                         $state = self::COMMENT_END;
 | |
|                     }
 | |
|                     break;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         // process the last token
 | |
|         switch ($state) {
 | |
|             case self::COMMENT:
 | |
|             case self::COMMENT_END:
 | |
|                 $com = new Comment();
 | |
|                 $com->setContent($token);
 | |
|                 if ($state === self::COMMENT_END) {
 | |
|                     if (isset($dir)) {
 | |
|                         $dir->setCommentPost($com);
 | |
|                     } else {
 | |
|                         $sec->setCommentPost($com);
 | |
|                     }
 | |
|                 } else {
 | |
|                     $coms[] = $com;
 | |
|                 }
 | |
|                 break;
 | |
| 
 | |
|             case self::DIRECTIVE_VALUE:
 | |
|                 $dir->setValue($token);
 | |
|                 $sec->addDirective($dir);
 | |
|                 break;
 | |
| 
 | |
|             case self::ESCAPE:
 | |
|             case self::DIRECTIVE_VALUE_QUOTED:
 | |
|             case self::DIRECTIVE_KEY:
 | |
|             case self::SECTION:
 | |
|                 self::throwParseError('File ended in unterminated state ' . $state, $line);
 | |
|         }
 | |
|         if (! empty($coms)) {
 | |
|             $doc->setCommentsDangling($coms);
 | |
|         }
 | |
|         return $doc;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Read the ini file and parse it with ::parseIni()
 | |
|      *
 | |
|      * @param   string  $file       The ini file to read
 | |
|      *
 | |
|      * @return  Config
 | |
|      * @throws  NotReadableError    When the file cannot be read
 | |
|      */
 | |
|     public static function parseIniFile($file)
 | |
|     {
 | |
|         if (($path = realpath($file)) === false) {
 | |
|             throw new NotReadableError('Couldn\'t compute the absolute path of `%s\'', $file);
 | |
|         }
 | |
| 
 | |
|         if (($content = file_get_contents($path)) === false) {
 | |
|             throw new NotReadableError('Couldn\'t read the file `%s\'', $path);
 | |
|         }
 | |
| 
 | |
|         try {
 | |
|             $configArray = parse_ini_string($content, true, INI_SCANNER_RAW);
 | |
|         } catch (ErrorException $e) {
 | |
|             throw new ConfigurationError('Couldn\'t parse the INI file `%s\'', $path, $e);
 | |
|         }
 | |
| 
 | |
|         $unescaped = array();
 | |
|         foreach ($configArray as $section => $options) {
 | |
|             $unescaped[self::unescapeSectionName($section)] = array_map([__CLASS__, 'unescapeOptionValue'], $options);
 | |
|         }
 | |
| 
 | |
|         return Config::fromArray($unescaped)->setConfigFile($file);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Unescape significant characters in the given section name
 | |
|      *
 | |
|      * @param   string  $str
 | |
|      *
 | |
|      * @return  string
 | |
|      */
 | |
|     protected static function unescapeSectionName($str)
 | |
|     {
 | |
|         $str = str_replace('\"', '"', $str);
 | |
|         $str = str_replace('\;', ';', $str);
 | |
| 
 | |
|         return str_replace('\\\\', '\\', $str);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Unescape significant characters in the given option value
 | |
|      *
 | |
|      * @param   string  $str
 | |
|      *
 | |
|      * @return  string
 | |
|      */
 | |
|     protected static function unescapeOptionValue($str)
 | |
|     {
 | |
|         $str = str_replace('\n', "\n", $str);
 | |
|         $str = str_replace('\r', "\r", $str);
 | |
|         $str = str_replace('\"', '"', $str);
 | |
|         $str = str_replace('\\\\', '\\', $str);
 | |
| 
 | |
|         // This replacement is a work-around for PHP bug #76965. Fixed with versions 7.1.24, 7.2.12 and 7.3.0.
 | |
|         return preg_replace('~^([\'"])(.*?)\1\s+$~', '$2', $str);
 | |
|     }
 | |
| }
 |