From b12c069769e856f43c6f509d6e3528031795440e Mon Sep 17 00:00:00 2001
From: Thomas Steur <thomas.steur@gmail.com>
Date: Thu, 14 Jan 2016 03:45:01 +0000
Subject: [PATCH] add possibility to view config values, including description,
 default value and whether it is an adjusted config value in the ui

---
 composer.json                                 |   2 +-
 composer.lock                                 |  17 +-
 core/Twig.php                                 |  24 ++
 plugins/Diagnostics/ConfigReader.php          | 157 ++++++++++++
 plugins/Diagnostics/Controller.php            |  72 ++++++
 plugins/Diagnostics/Diagnostics.php           |  15 ++
 plugins/Diagnostics/Menu.php                  |  28 +++
 .../Test/Integration/ConfigReaderTest.php     | 238 ++++++++++++++++++
 plugins/Diagnostics/lang/en.json              |   8 +
 plugins/Diagnostics/plugin.json               |   5 +-
 .../Diagnostics/stylesheets/configfile.less   |  22 ++
 plugins/Diagnostics/templates/configfile.twig |  55 ++++
 tests/UI/specs/UIIntegration_spec.js          |   6 +
 tests/resources/Config/global.ini.php         |   4 +
 14 files changed, 642 insertions(+), 11 deletions(-)
 create mode 100644 plugins/Diagnostics/ConfigReader.php
 create mode 100644 plugins/Diagnostics/Controller.php
 create mode 100644 plugins/Diagnostics/Menu.php
 create mode 100644 plugins/Diagnostics/Test/Integration/ConfigReaderTest.php
 create mode 100644 plugins/Diagnostics/lang/en.json
 create mode 100644 plugins/Diagnostics/stylesheets/configfile.less
 create mode 100644 plugins/Diagnostics/templates/configfile.twig

diff --git a/composer.json b/composer.json
index e7cc14f778..3c6521b827 100644
--- a/composer.json
+++ b/composer.json
@@ -46,7 +46,7 @@
         "piwik/decompress": "~1.0",
         "piwik/network": "~0.1.0",
         "piwik/cache": "~0.2.5",
-        "piwik/ini": "^1.0.3",
+        "piwik/ini": "^1.0.6",
         "php-di/php-di": "5.0.0-beta1",
         "psr/log": "~1.0",
         "monolog/monolog": "~1.11",
diff --git a/composer.lock b/composer.lock
index 263c85b4ab..fc23cdfa2d 100644
--- a/composer.lock
+++ b/composer.lock
@@ -4,8 +4,8 @@
         "Read more about it at https://getcomposer.org/doc/01-basic-usage.md#composer-lock-the-lock-file",
         "This file is @generated automatically"
     ],
-    "hash": "09112ef01f28686b387148c407503c7c",
-    "content-hash": "ff9b83524f413ac80daad8eb47099042",
+    "hash": "de61be52972a0fe8fe751306c271f4b8",
+    "content-hash": "68130b067cdceef8346b47d858b763a3",
     "packages": [
         {
             "name": "container-interop/container-interop",
@@ -300,7 +300,6 @@
                 "phpdoc",
                 "reflection"
             ],
-            "abandoned": "php-di/phpdoc-reader",
             "time": "2014-08-21 08:20:45"
         },
         {
@@ -875,16 +874,16 @@
         },
         {
             "name": "piwik/ini",
-            "version": "1.0.4",
+            "version": "1.0.6",
             "source": {
                 "type": "git",
                 "url": "https://github.com/piwik/component-ini.git",
-                "reference": "9269255fd187e5bda2e5778041c8d143eb615b0a"
+                "reference": "bd2711ba4d5e20e4ca09b6829dc2831576b59dc3"
             },
             "dist": {
                 "type": "zip",
-                "url": "https://api.github.com/repos/piwik/component-ini/zipball/9269255fd187e5bda2e5778041c8d143eb615b0a",
-                "reference": "9269255fd187e5bda2e5778041c8d143eb615b0a",
+                "url": "https://api.github.com/repos/piwik/component-ini/zipball/bd2711ba4d5e20e4ca09b6829dc2831576b59dc3",
+                "reference": "bd2711ba4d5e20e4ca09b6829dc2831576b59dc3",
                 "shasum": ""
             },
             "require": {
@@ -904,7 +903,7 @@
             "license": [
                 "LGPL-3.0"
             ],
-            "time": "2015-04-21 04:59:09"
+            "time": "2016-01-14 21:13:33"
         },
         {
             "name": "piwik/network",
@@ -1551,7 +1550,7 @@
                 "performance",
                 "profiling"
             ],
-            "time": "2015-02-26 14:37:51"
+            "time": "2014-08-28 17:34:52"
         },
         {
             "name": "guzzle/guzzle",
diff --git a/core/Twig.php b/core/Twig.php
index 06706d7b44..5968b965eb 100755
--- a/core/Twig.php
+++ b/core/Twig.php
@@ -104,6 +104,8 @@ class Twig
         $this->twig->addTokenParser(new RenderTokenParser());
 
         $this->addTest_false();
+        $this->addTest_true();
+        $this->addTest_emptyString();
     }
 
     private function addTest_false()
@@ -117,6 +119,28 @@ class Twig
         $this->twig->addTest($test);
     }
 
+    private function addTest_true()
+    {
+        $test = new Twig_SimpleTest(
+            'true',
+            function ($value) {
+                return true === $value;
+            }
+        );
+        $this->twig->addTest($test);
+    }
+
+    private function addTest_emptyString()
+    {
+        $test = new Twig_SimpleTest(
+            'emptyString',
+            function ($value) {
+                return '' === $value;
+            }
+        );
+        $this->twig->addTest($test);
+    }
+
     protected function addFunction_getJavascriptTranslations()
     {
         $getJavascriptTranslations = new Twig_SimpleFunction(
diff --git a/plugins/Diagnostics/ConfigReader.php b/plugins/Diagnostics/ConfigReader.php
new file mode 100644
index 0000000000..ca47af2f69
--- /dev/null
+++ b/plugins/Diagnostics/ConfigReader.php
@@ -0,0 +1,157 @@
+<?php
+/**
+ * Piwik - free/libre analytics platform
+ *
+ * @link http://piwik.org
+ * @license http://www.gnu.org/licenses/gpl-3.0.html GPL v3 or later
+ */
+namespace Piwik\Plugins\Diagnostics;
+
+use Piwik\Development;
+use Piwik\Ini\IniReader;
+use Piwik\Application\Kernel\GlobalSettingsProvider;
+use Piwik\Settings;
+
+/**
+ * A diagnostic report contains all the results of all the diagnostics.
+ */
+class ConfigReader
+{
+    /**
+     * @var GlobalSettingsProvider
+     */
+    private $settings;
+
+    /**
+     * @var IniReader
+     */
+    private $iniReader;
+
+    public function __construct(GlobalSettingsProvider $settings, IniReader $iniReader)
+    {
+        $this->settings = $settings;
+        $this->iniReader = $iniReader;
+    }
+
+    public function getConfigValuesFromFiles()
+    {
+        $ini = $this->settings->getIniFileChain();
+        $descriptions = $this->iniReader->readComments($this->settings->getPathGlobal());
+
+        $copy = array();
+        foreach ($ini->getAll() as $category => $values) {
+            if ($this->shouldSkipCategory($category)) {
+                continue;
+            }
+
+            $local = $this->getFromLocalConfig($category);
+            if (empty($local)) {
+                $local = array();
+            }
+
+            $global = $this->getFromGlobalConfig($category);
+            if (empty($global)) {
+                $global = array();
+            }
+
+            $copy[$category] = array();
+            foreach ($values as $key => $value) {
+
+                $newValue = $value;
+                if (strpos(strtolower($key), 'password') !== false) {
+                    $newValue = str_pad('', strlen($value), '*');
+                }
+
+                $defaultValue = null;
+                if (array_key_exists($key, $global)) {
+                    $defaultValue = $global[$key];
+                }
+
+                $description = '';
+                if (!empty($descriptions[$category][$key])) {
+                    $description = trim($descriptions[$category][$key]);
+                }
+
+                $copy[$category][$key] = array(
+                    'value' => $newValue,
+                    'description' => $description,
+                    'isCustomValue' => array_key_exists($key, $local),
+                    'defaultValue' => $defaultValue,
+                );
+            }
+        }
+
+        return $copy;
+    }
+
+    private function shouldSkipCategory($category)
+    {
+        $category = strtolower($category);
+        if ($category === 'database') {
+            return true;
+        }
+
+        $developmentOnlySections = array('database_tests', 'tests', 'debugtests');
+
+        return !Development::isEnabled() && in_array($category, $developmentOnlySections);
+    }
+
+    public function getFromGlobalConfig($name)
+    {
+        return $this->settings->getIniFileChain()->getFrom($this->settings->getPathGlobal(), $name);
+    }
+
+    public function getFromLocalConfig($name)
+    {
+        return $this->settings->getIniFileChain()->getFrom($this->settings->getPathLocal(), $name);
+    }
+
+    /**
+     * Adds config values that can be used to overwrite a plugin system setting and adds a description + default value
+     * for already existing configured config values that overwrite a plugin system setting.
+     *
+     * @param array $configValues
+     * @param \Piwik\Plugin\Settings[] $pluginSettings
+     * @return array
+     */
+    public function addConfigValuesFromPluginSettings($configValues, $pluginSettings)
+    {
+        foreach ($pluginSettings as $pluginSetting) {
+            $pluginName = $pluginSetting->getPluginName();
+            $configs[$pluginName] = array();
+
+            foreach ($pluginSetting->getSettings() as $setting) {
+                if ($setting instanceof Settings\SystemSetting && $setting->isReadableByCurrentUser()) {
+
+                    $description = '';
+                    if (!empty($setting->description)) {
+                        $description .= $setting->description . ' ';
+                    }
+
+                    if (!empty($setting->inlineHelp)) {
+                        $description .= $setting->inlineHelp;
+                    }
+
+                    if (isset($configValues[$pluginName][$setting->getName()])) {
+                        $configValues[$pluginName][$setting->getName()]['defaultValue'] = $setting->defaultValue;
+                        $configValues[$pluginName][$setting->getName()]['description']  = trim($description);
+                    } else {
+                        $defaultValue = $setting->getValue();
+                        $configValues[$pluginName][$setting->getName()] = array(
+                            'value' => null,
+                            'description' => trim($description),
+                            'isCustomValue' => false,
+                            'defaultValue' => $defaultValue
+                        );
+                    }
+                }
+            }
+
+            if (empty($configValues[$pluginName])) {
+                unset($configValues[$pluginName]);
+            }
+        }
+
+        return $configValues;
+    }
+}
diff --git a/plugins/Diagnostics/Controller.php b/plugins/Diagnostics/Controller.php
new file mode 100644
index 0000000000..be166ccdf9
--- /dev/null
+++ b/plugins/Diagnostics/Controller.php
@@ -0,0 +1,72 @@
+<?php
+/**
+ * Piwik - free/libre analytics platform
+ *
+ * @link http://piwik.org
+ * @license http://www.gnu.org/licenses/gpl-3.0.html GPL v3 or later
+ */
+
+namespace Piwik\Plugins\Diagnostics;
+
+use Piwik\Config;
+use Piwik\Piwik;
+use Piwik\View;
+use Piwik\Settings;
+
+/**
+ * A controller let's you for example create a page that can be added to a menu. For more information read our guide
+ * http://developer.piwik.org/guides/mvc-in-piwik or have a look at the our API references for controller and view:
+ * http://developer.piwik.org/api-reference/Piwik/Plugin/Controller and
+ * http://developer.piwik.org/api-reference/Piwik/View
+ */
+class Controller extends \Piwik\Plugin\ControllerAdmin
+{
+    /**
+     * @var ConfigReader
+     */
+    private $configReader;
+
+    public function __construct(ConfigReader $configReader)
+    {
+        $this->configReader = $configReader;
+        parent::__construct();
+    }
+
+    public function configfile()
+    {
+        Piwik::checkUserHasSuperUserAccess();
+
+        $allSettings = Settings\Manager::getAllPluginSettings();
+
+        $configValues = $this->configReader->getConfigValuesFromFiles();
+        $configValues = $this->configReader->addConfigValuesFromPluginSettings($configValues, $allSettings);
+        $configValues = $this->sortConfigValues($configValues);
+
+        return $this->renderTemplate('configfile', array(
+            'allConfigValues' => $configValues
+        ));
+    }
+
+    private function sortConfigValues($configValues)
+    {
+        // we sort by sections alphabetically
+        uksort($configValues, function ($section1, $section2) {
+            return strcasecmp($section1, $section2);
+        });
+
+        foreach ($configValues as $category => &$settings) {
+            // we sort keys alphabetically but list the ones that are changed first
+            uksort($settings, function ($setting1, $setting2) use ($settings) {
+                if ($settings[$setting1]['isCustomValue'] && !$settings[$setting2]['isCustomValue']) {
+                    return -1;
+                } elseif (!$settings[$setting1]['isCustomValue'] && $settings[$setting2]['isCustomValue']) {
+                    return 1;
+                }
+                return strcasecmp($setting1, $setting2);
+            });
+        }
+
+        return $configValues;
+    }
+
+}
diff --git a/plugins/Diagnostics/Diagnostics.php b/plugins/Diagnostics/Diagnostics.php
index f69d1f45d1..9760be5b05 100644
--- a/plugins/Diagnostics/Diagnostics.php
+++ b/plugins/Diagnostics/Diagnostics.php
@@ -12,4 +12,19 @@ use Piwik\Plugin;
 
 class Diagnostics extends Plugin
 {
+    /**
+     * @see Piwik\Plugin::registerEvents
+     */
+    public function registerEvents()
+    {
+        return array(
+            'AssetManager.getStylesheetFiles' => 'getStylesheetFiles',
+        );
+    }
+
+    public function getStylesheetFiles(&$stylesheets)
+    {
+        $stylesheets[] = "plugins/Diagnostics/stylesheets/configfile.less";
+    }
+
 }
diff --git a/plugins/Diagnostics/Menu.php b/plugins/Diagnostics/Menu.php
new file mode 100644
index 0000000000..52c28699b5
--- /dev/null
+++ b/plugins/Diagnostics/Menu.php
@@ -0,0 +1,28 @@
+<?php
+/**
+ * Piwik - free/libre analytics platform
+ *
+ * @link http://piwik.org
+ * @license http://www.gnu.org/licenses/gpl-3.0.html GPL v3 or later
+ */
+
+namespace Piwik\Plugins\Diagnostics;
+
+use Piwik\Menu\MenuAdmin;
+use Piwik\Piwik;
+
+/**
+ * This class allows you to add, remove or rename menu items.
+ * To configure a menu (such as Admin Menu, Reporting Menu, User Menu...) simply call the corresponding methods as
+ * described in the API-Reference http://developer.piwik.org/api-reference/Piwik/Menu/MenuAbstract
+ */
+class Menu extends \Piwik\Plugin\Menu
+{
+    public function configureAdminMenu(MenuAdmin $menu)
+    {
+        if (Piwik::hasUserSuperUserAccess()) {
+            $menu->addDiagnosticItem('Diagnostics_ConfigFileTitle', $this->urlForAction('configfile'), $orderId = 30);
+        }
+    }
+
+}
diff --git a/plugins/Diagnostics/Test/Integration/ConfigReaderTest.php b/plugins/Diagnostics/Test/Integration/ConfigReaderTest.php
new file mode 100644
index 0000000000..a44da5cd7e
--- /dev/null
+++ b/plugins/Diagnostics/Test/Integration/ConfigReaderTest.php
@@ -0,0 +1,238 @@
+<?php
+/**
+ * Piwik - free/libre analytics platform
+ *
+ * @link http://piwik.org
+ * @license http://www.gnu.org/licenses/gpl-3.0.html GPL v3 or later
+ */
+
+namespace Piwik\Plugins\Diagnostics\Test\Integration\Commands;
+
+use Piwik\Application\Kernel\GlobalSettingsProvider;
+use Piwik\Ini\IniReader;
+use Piwik\Plugins\Diagnostics\ConfigReader;
+use Piwik\Plugins\ExampleSettingsPlugin\Settings;
+use Piwik\Tests\Fixtures\OneVisitorTwoVisits;
+use Piwik\Tests\Framework\TestCase\IntegrationTestCase;
+
+/**
+ * TODO: This could be a unit test if we could inject the ArchiveTableDao in the command
+ * @group Diagnostics
+ * @group Plugins
+ */
+class ConfigReaderTest extends IntegrationTestCase
+{
+    /**
+     * @var ConfigReader
+     */
+    private $configReader;
+
+    public function setUp()
+    {
+        $settings = new GlobalSettingsProvider($this->configPath('global.ini.php'), $this->configPath('config.ini.php'), $this->configPath('common.config.ini.php'));
+        $this->configReader = new ConfigReader($settings, new IniReader());
+    }
+
+    public function test_getConfigValuesFromFiles()
+    {
+        $fileConfig = $this->configReader->getConfigValuesFromFiles();
+
+        $expected = array (
+            'Category' =>
+                array (
+                    'key1' =>
+                        array (
+                            'value' => 'value_overwritten',
+                            'description' => '',
+                            'isCustomValue' => true,
+                            'defaultValue' => 'value1',
+                        ),
+                    'key2' =>
+                        array (
+                            'value' => 'valueCommon',
+                            'description' => '',
+                            'isCustomValue' => false,
+                            'defaultValue' => 'value2',
+                        ),
+                    'key3' =>
+                        array (
+                            'value' => '${@piwik(crash))}',
+                            'description' => '',
+                            'isCustomValue' => false,
+                            'defaultValue' => NULL,
+                        ),
+                ),
+            'CategoryOnlyInGlobalFile' =>
+                array (
+                    'key3' =>
+                        array (
+                            'value' => 'value3',
+                            'description' => 'test comment',
+                            'isCustomValue' => false,
+                            'defaultValue' => 'value3',
+                        ),
+                    'key4' =>
+                        array (
+                            'value' => 'value4',
+                            'description' => 'test comment 4',
+                            'isCustomValue' => false,
+                            'defaultValue' => 'value4',
+                        ),
+                ),
+            'TestArray' =>
+                array (
+                    'installed' =>
+                        array (
+                            'value' =>
+                                array (
+                                    0 => 'plugin"1',
+                                    1 => 'plugin2',
+                                    2 => 'plugin3',
+                                ),
+                            'description' => 'test comment 2
+with multiple lines',
+                            'isCustomValue' => true,
+                            'defaultValue' =>
+                                array (
+                                    0 => 'plugin1',
+                                    1 => 'plugin4',
+                                ),
+                        ),
+                ),
+            'TestArrayOnlyInGlobalFile' =>
+                array (
+                    'my_array' =>
+                        array (
+                            'value' =>
+                                array (
+                                    0 => 'value1',
+                                    1 => 'value2',
+                                ),
+                            'description' => '',
+                            'isCustomValue' => false,
+                            'defaultValue' =>
+                                array (
+                                    0 => 'value1',
+                                    1 => 'value2',
+                                ),
+                        ),
+                ),
+            'GeneralSection' =>
+                array (
+                    'password' =>
+                        array (
+                            'value' => '****',
+                            'description' => '',
+                            'isCustomValue' => true,
+                            'defaultValue' => NULL,
+                        ),
+                    'login' =>
+                        array (
+                            'value' => 'tes"t',
+                            'description' => '',
+                            'isCustomValue' => true,
+                            'defaultValue' => NULL,
+                        ),
+                ),
+            'TestOnlyInCommon' =>
+                array (
+                    'value' =>
+                        array (
+                            'value' => 'commonValue',
+                            'description' => '',
+                            'isCustomValue' => false,
+                            'defaultValue' => NULL,
+                        ),
+                ),
+            'Tracker' =>
+                array (
+                    'commonConfigTracker' =>
+                        array (
+                            'value' => 'commonConfigTrackerValue',
+                            'description' => '',
+                            'isCustomValue' => false,
+                            'defaultValue' => NULL,
+                        ),
+                ),
+        );
+        $this->assertEquals($expected, $fileConfig);
+    }
+
+    public function test_addConfigValuesFromPluginSettings()
+    {
+        $settings = new Settings();
+
+        $configValues = $this->configReader->addConfigValuesFromPluginSettings(array(), array($settings));
+
+        $expected = array (
+            'ExampleSettingsPlugin' =>
+                array (
+                    'metric' =>
+                        array (
+                            'value' => NULL,
+                            'description' => 'Choose the metric that should be displayed in the browser tab',
+                            'isCustomValue' => false,
+                            'defaultValue' => 'nb_visits',
+                        ),
+                    'browsers' =>
+                        array (
+                            'value' => NULL,
+                            'description' => 'The value will be only displayed in the following browsers',
+                            'isCustomValue' => false,
+                            'defaultValue' =>
+                                array (
+                                    0 => 'firefox',
+                                    1 => 'chromium',
+                                    2 => 'safari',
+                                ),
+                        ),
+                    'description' =>
+                        array (
+                            'value' => NULL,
+                            'description' => 'This description will be displayed next to the value',
+                            'isCustomValue' => false,
+                            'defaultValue' => 'This is the value:
+Another line',
+                        ),
+                    'password' =>
+                        array (
+                            'value' => NULL,
+                            'description' => 'Password for the 3rd API where we fetch the value',
+                            'isCustomValue' => false,
+                            'defaultValue' => NULL,
+                        ),
+                ),
+        );
+        $this->assertEquals($expected, $configValues);
+    }
+
+    public function test_addConfigValuesFromPluginSettings_shouldAddDescriptionAndDefaultValueForExistingConfigValues()
+    {
+        $settings = new Settings();
+
+        $existing = array(
+            'ExampleSettingsPlugin' =>
+                array (
+                    'metric' =>
+                        array (
+                            'value' => NULL,
+                            'description' => '',
+                            'isCustomValue' => false,
+                            'defaultValue' => null,
+                        ),
+                    )
+        );
+
+        $configValues = $this->configReader->addConfigValuesFromPluginSettings($existing, array($settings));
+
+        $this->assertSame('Choose the metric that should be displayed in the browser tab', $configValues['ExampleSettingsPlugin']['metric']['description']);
+        $this->assertSame('nb_visits', $configValues['ExampleSettingsPlugin']['metric']['defaultValue']);
+    }
+
+    private function configPath($file)
+    {
+        return PIWIK_INCLUDE_PATH . '/tests/resources/Config/' . $file;
+    }
+}
+
+AnalyzeArchiveTableTest::$fixture = new OneVisitorTwoVisits();
\ No newline at end of file
diff --git a/plugins/Diagnostics/lang/en.json b/plugins/Diagnostics/lang/en.json
new file mode 100644
index 0000000000..3154699723
--- /dev/null
+++ b/plugins/Diagnostics/lang/en.json
@@ -0,0 +1,8 @@
+{
+    "Diagnostics": {
+        "ConfigFileTitle": "Config file",
+        "ConfigFileIntroduction": "Here you can view the Piwik configuration. If you are running Piwik in a load balanced environment the page might be different depending from which server this page is loaded. Rows with a different background color are changed config values that are specified for example in the %s file.",
+        "HideUnchanged": "If you want to see only changed values you can %shide all unchanged values%s.",
+        "Sections": "Sections"
+    }
+}
\ No newline at end of file
diff --git a/plugins/Diagnostics/plugin.json b/plugins/Diagnostics/plugin.json
index da53096d70..da6ccab3fa 100644
--- a/plugins/Diagnostics/plugin.json
+++ b/plugins/Diagnostics/plugin.json
@@ -1,3 +1,6 @@
 {
-  "description": "Performs diagnostics to check that Piwik is installed and runs correctly."
+    "description": "Performs diagnostics to check that Piwik is installed and runs correctly.",
+    "require": {
+        "piwik": ">=2.16.0-b2"
+    }
 }
\ No newline at end of file
diff --git a/plugins/Diagnostics/stylesheets/configfile.less b/plugins/Diagnostics/stylesheets/configfile.less
new file mode 100644
index 0000000000..f399ce3596
--- /dev/null
+++ b/plugins/Diagnostics/stylesheets/configfile.less
@@ -0,0 +1,22 @@
+.diagnostics.configfile {
+  .custom-value {
+    background-color: @theme-color-background-tinyContrast;
+  }
+
+  .defaultValue {
+    font-style: italic;
+  }
+
+  td.name {
+    max-width: 330px;
+    word-wrap: break-word;
+    width: 25%;
+  }
+
+  td.value {
+    word-wrap: break-word;
+    max-width: 400px;
+    width: 25%;
+  }
+
+}
diff --git a/plugins/Diagnostics/templates/configfile.twig b/plugins/Diagnostics/templates/configfile.twig
new file mode 100644
index 0000000000..46113b51da
--- /dev/null
+++ b/plugins/Diagnostics/templates/configfile.twig
@@ -0,0 +1,55 @@
+{% extends 'admin.twig' %}
+
+{% macro humanReadableValue(value) %}
+    {% if value is false %}
+        false
+    {% elseif value is true %}
+        true
+    {% elseif value is null %}
+    {% elseif value is emptyString %}
+        ''
+    {% else %}
+        {{ value|join(', ') }}
+    {% endif %}
+{% endmacro %}
+
+{% block content %}
+    <h2 piwik-enriched-headline>{{ 'Diagnostics_ConfigFileTitle'|translate }}</h2>
+    <p>
+        {{ 'Diagnostics_ConfigFileIntroduction'|translate('<code>"config/config.ini.php"</code>')|raw }}
+        {{ 'Diagnostics_HideUnchanged'|translate('<a ng-click="hideGlobalConfigValues=!hideGlobalConfigValues">', '</a>')|raw }}
+
+        <h3>{{ 'Diagnostics_Sections'|translate }}</h3>
+        {% for category, values in allConfigValues %}
+            <a href="#{{ category|e('html_attr') }}">{{ category }}</a><br />
+        {% endfor %}
+    </p>
+
+    <table class="simple-table diagnostics configfile">
+        <tbody>
+        {% for category, configValues in allConfigValues %}
+            <tr><td colspan="3"><a name="{{ category|e('html_attr') }}"></a><h3>{{ category }}</h3></td></tr>
+
+            {% for key, configEntry in configValues %}
+                <tr {% if configEntry.isCustomValue %}class="custom-value"{% else %}ng-hide="hideGlobalConfigValues"{% endif %}>
+                    <td class="name">{{ key }}{% if configEntry.value is iterable %}[]{% endif %}</td>
+                    <td class="value">
+                        {{ _self.humanReadableValue(configEntry.value) }}
+                    </td>
+                    <td class="description">
+                        {{ configEntry.description }}
+
+                        {% if (configEntry.isCustomValue or configEntry.value is null) and configEntry.defaultValue is not null %}
+                            {% if configEntry.description %}<br />{% endif %}
+
+                            {{ 'General_Default'|translate }}:
+                            <span class="defaultValue">{{ _self.humanReadableValue(configEntry.defaultValue) }}<span>
+                        {% endif %}
+                    </td>
+                </tr>
+            {% endfor %}
+        {% endfor %}
+        </tbody>
+    </table>
+
+{% endblock %}
\ No newline at end of file
diff --git a/tests/UI/specs/UIIntegration_spec.js b/tests/UI/specs/UIIntegration_spec.js
index 65fa37d721..9703f28814 100644
--- a/tests/UI/specs/UIIntegration_spec.js
+++ b/tests/UI/specs/UIIntegration_spec.js
@@ -498,6 +498,12 @@ describe("UIIntegrationTest", function () { // TODO: Rename to Piwik?
         }, done);
     });
 
+    it('should load the config file page correctly', function (done) {
+        expect.screenshot('admin_diagnostics_configfile').to.be.captureSelector('.pageWrap', function (page) {
+            page.load("?" + generalParams + "&module=Diagnostics&action=configfile");
+        }, done);
+    });
+
     it('should load the Settings > Visitor Generator admin page correctly', function (done) {
         expect.screenshot('admin_visitor_generator').to.be.captureSelector('.pageWrap', function (page) {
             page.load("?" + generalParams + "&module=VisitorGenerator&action=index");
diff --git a/tests/resources/Config/global.ini.php b/tests/resources/Config/global.ini.php
index 20271ed43d..87e1437453 100644
--- a/tests/resources/Config/global.ini.php
+++ b/tests/resources/Config/global.ini.php
@@ -3,10 +3,14 @@ key1 = value1
 key2 = value2
 
 [CategoryOnlyInGlobalFile]
+; test comment
 key3 = "value3"
+; test comment 4
 key4 = value4
 
 [TestArray]
+; test comment 2
+; with multiple lines
 installed[] = plugin1
 installed[] = plugin4
 
-- 
GitLab