| @ -0,0 +1,474 @@ | |||
| <?php | |||
| error_reporting(E_ALL); | |||
| ini_set("display_errors", 1); | |||
| function is__writable($path) { | |||
| if ($path[strlen($path) - 1] == '/') // recursively return a temporary file path | |||
| return is__writable($path . uniqid(mt_rand()) . '.tmp'); | |||
| else if (is_dir($path)) | |||
| return is__writable($path . '/' . uniqid(mt_rand()) . '.tmp'); | |||
| // check tmp file for read/write capabilities | |||
| $rm = file_exists($path); | |||
| $f = @fopen($path, 'a'); | |||
| if ($f === false) | |||
| return false; | |||
| fclose($f); | |||
| if (!$rm) | |||
| unlink($path); | |||
| return true; | |||
| } | |||
| function from($source, $name) { | |||
| if (is_array($name)) { | |||
| $data = array(); | |||
| foreach ($name as $k) | |||
| $data[$k] = isset($source[$k]) ? $source[$k] : null; | |||
| return $data; | |||
| } | |||
| return isset($source[$name]) ? $source[$name] : null; | |||
| } | |||
| function htaccess(){ return <<<EOT | |||
| # Don't show directory listings for URLs which map to a directory. | |||
| Options -Indexes | |||
| # Make HTMLy handle any 404 errors. | |||
| ErrorDocument 404 /installer.php | |||
| # Set the default handler. | |||
| DirectoryIndex installer.php | |||
| # Requires mod_expires to be enabled. | |||
| # Various rewrite rules. | |||
| <IfModule mod_rewrite.c> | |||
| RewriteEngine on | |||
| # Uncomment the following to redirect all visitors to the www version | |||
| # RewriteCond %{HTTP_HOST} !^www\. [NC] | |||
| # RewriteRule ^ http://www.%{HTTP_HOST}%{REQUEST_URI} [L,R=301] | |||
| # Uncomment the following to redirect all visitors to non www version | |||
| # RewriteCond %{HTTP_HOST} ^www\.(.+)$ [NC] | |||
| # RewriteRule ^ http://%1%{REQUEST_URI} [L,R=301] | |||
| # If your site is running in a VirtualDocumentRoot at http://example.com/, | |||
| # uncomment the following line: | |||
| # RewriteBase / | |||
| # Pass all requests not referring directly to files in the filesystem to index.php. | |||
| RewriteCond %{REQUEST_FILENAME} !-f | |||
| RewriteCond %{REQUEST_FILENAME} !-d | |||
| RewriteRule ^ installer.php [L] | |||
| </IfModule> | |||
| EOT; | |||
| } | |||
| class Message { | |||
| protected $errors = array(); | |||
| public function error($message) { | |||
| $this->errors[] = $message; | |||
| } | |||
| protected $warnings = array(); | |||
| public function warning($message) { | |||
| $this->warnings[] = $message; | |||
| } | |||
| public function run() { | |||
| $string = ""; | |||
| if (!empty($this->errors)) { | |||
| foreach ($this->errors as $error) { | |||
| $string .= '<p class="error">' . $error . "</p>"; | |||
| } | |||
| } | |||
| if (!empty($this->warnings)) { | |||
| foreach ($this->warnings as $warning) { | |||
| $string .= '<p class="warning">' . $warning . "</p>"; | |||
| } | |||
| } | |||
| return $string; | |||
| } | |||
| } | |||
| class Settings | |||
| { | |||
| protected $user = ""; | |||
| protected $userPassword = ""; | |||
| protected $siteUrl = ""; | |||
| protected $overwriteEmptyForm = array( | |||
| "social.twitter" => "", | |||
| "social.facebook" => "", | |||
| ); | |||
| protected function extractUser() | |||
| { | |||
| $this->user = (string)$_REQUEST["user_name"]; | |||
| unset($_REQUEST["user_name"]); | |||
| $this->userPassword = (string)$_REQUEST["user_password"]; | |||
| unset($_REQUEST["user_password"]); | |||
| } | |||
| protected function convertRequestToConfig() | |||
| { | |||
| $array = array(); | |||
| foreach ($_REQUEST as $name => $value) { | |||
| if (!is_string($value) || empty($value)) | |||
| continue; | |||
| $name = str_replace("_", ".", $name); | |||
| $array[$name] = $value; | |||
| } | |||
| foreach ($this->overwriteEmptyForm as $name => $value) { | |||
| if (!isset($array[$name])) { | |||
| $array[$name] = $value; | |||
| } | |||
| } | |||
| return $array; | |||
| } | |||
| protected function generateSiteUrl() | |||
| { | |||
| $dir = trim(dirname(substr($_SERVER["SCRIPT_FILENAME"], strlen($_SERVER["DOCUMENT_ROOT"]))), '/'); | |||
| if ($dir == '.' || $dir == '..') { | |||
| $dir = ''; | |||
| } | |||
| $port = ''; | |||
| if ($_SERVER["SERVER_PORT"] != "80") { | |||
| $port = ':' . $_SERVER["SERVER_PORT"]; | |||
| } | |||
| $scheme = isset($_SERVER['HTTPS']) && $_SERVER['HTTPS'] != 'off' ? 'https' : 'http'; | |||
| if ($dir === '') { | |||
| $this->siteUrl = $scheme . '://' . trim($_SERVER['SERVER_NAME'], "/") . $port . "/"; | |||
| return; | |||
| } | |||
| $this->siteUrl = $scheme . '://' . trim($_SERVER['SERVER_NAME'], "/") . $port . "/" . $dir . '/'; | |||
| } | |||
| protected function overwriteINI($data, $string) | |||
| { | |||
| foreach ($data as $word => $value) { | |||
| $string = preg_replace("/^" . $word . " = .+$/m", $word . ' = "' . $value . '"', $string); | |||
| } | |||
| return $string; | |||
| } | |||
| protected function saveConfigs() | |||
| { | |||
| $this->extractUser(); | |||
| //save config.ini | |||
| $config = array( | |||
| "site.url" => $this->siteUrl, | |||
| "timezone" => $this->getTimeZone(), | |||
| ); | |||
| $config += $this->convertRequestToConfig(); | |||
| $configFile = file_get_contents("config/config.ini.example"); | |||
| $configFile = $this->overwriteINI($config, $configFile); | |||
| file_put_contents("config/config.ini", $configFile); | |||
| //save users/[Username].ini | |||
| $userFile = file_get_contents("config/users/username.ini.example"); | |||
| $parsed = parse_ini_string($userFile); | |||
| if (isset($parsed['encryption'])) { | |||
| $userFile = $this->overwriteINI(array( | |||
| 'encryption' => 'sha512', | |||
| 'password' => hash('sha512', $this->userPassword), | |||
| 'role' => 'admin', | |||
| ), $userFile); | |||
| } else { | |||
| $userFile = $this->overwriteINI(array( | |||
| "password" => $this->userPassword, | |||
| 'role' => 'admin', | |||
| ), $userFile); | |||
| } | |||
| file_put_contents("config/users/" . $this->user . ".ini", $userFile); | |||
| } | |||
| protected function testTheEnvironment() | |||
| { | |||
| $message = new Message; | |||
| if (!defined('PHP_VERSION_ID') || PHP_VERSION_ID < 50300) { | |||
| $message->error('HTMLy requires at least PHP 5.3 to run.'); | |||
| } | |||
| if (!in_array('https', stream_get_wrappers())) { | |||
| $message->error('Installer needs the https wrapper, please install openssl.'); | |||
| } | |||
| if (function_exists('apache_get_modules') && !in_array('mod_rewrite', apache_get_modules())) { | |||
| $message->warning('mod_rewrite must be enabled if you use Apache.'); | |||
| } | |||
| if (!is__writable("./")) { | |||
| $message->error('no permission to write in the Directory.'); | |||
| } | |||
| return $message->run(); | |||
| } | |||
| public function __construct() | |||
| { | |||
| $message = $this->testTheEnvironment(); | |||
| if (is__writable("./") && !file_exists(".htaccess")) { | |||
| file_put_contents(".htaccess", htaccess()); | |||
| } | |||
| $this->generateSiteUrl(); | |||
| if (!empty($message)) { | |||
| echo $message; | |||
| } elseif ($this->runForm()) { | |||
| unlink(__FILE__); | |||
| header("Location:" . $this->siteUrl . "add/content?type=post"); | |||
| exit(); | |||
| } | |||
| } | |||
| protected function getTimeZone() | |||
| { | |||
| static $ip; | |||
| if (empty($ip)) { | |||
| $ip = @file_get_contents("http://ipecho.net/plain"); | |||
| if (!is_string($ip)) { | |||
| $ip = $_SERVER['REMOTE_ADDR']; | |||
| } | |||
| } | |||
| $json = @json_decode(@file_get_contents("http://ip-api.com/json/" . $ip), true); | |||
| if (isset($json['timezone'])) | |||
| return $json['timezone']; | |||
| return 'Europe/Berlin'; | |||
| } | |||
| protected function runForm() | |||
| { | |||
| if (from($_REQUEST, 'user_name') && from($_REQUEST, 'user_password')) { | |||
| $this->saveConfigs(); | |||
| $_SESSION[$this->siteUrl]["user"] = $this->user; | |||
| return true; | |||
| } else { | |||
| unset($_SESSION[$this->siteUrl]["user"]); | |||
| return false; | |||
| } | |||
| } | |||
| } | |||
| if(from($_SERVER,'QUERY_STRING') == "rewriteRule.html") | |||
| { | |||
| echo "YES!"; | |||
| die(); | |||
| } | |||
| session_start(); | |||
| new Settings; | |||
| ?> | |||
| <!DOCTYPE html> | |||
| <html> | |||
| <head> | |||
| <style> | |||
| * { | |||
| margin: 0; | |||
| padding: 0; | |||
| } | |||
| *, *:before, *:after { | |||
| -moz-box-sizing: border-box; -webkit-box-sizing: border-box; box-sizing: border-box; | |||
| } | |||
| .responsive { | |||
| min-width: 512px; | |||
| width: 100%; | |||
| max-width: 730px; | |||
| } | |||
| section, footer, header, aside, nav { | |||
| display: block; | |||
| } | |||
| .error:before { | |||
| content: "Error: "; | |||
| color: red; | |||
| font-weight: bold; | |||
| font-size: 120%; | |||
| } | |||
| .warning:before { | |||
| content: "Warning: "; | |||
| color: gold; | |||
| font-weight: bold; | |||
| font-size: 120%; | |||
| } | |||
| body { | |||
| font-size: 17px; | |||
| line-height: 1.6em; | |||
| font-family: Georgia, sans-serif; | |||
| background: #F7F7F7; | |||
| color: #444444; | |||
| } | |||
| #cover { | |||
| padding: 0 0 20px 0; | |||
| float: left; | |||
| width: 100%; | |||
| } | |||
| #header-wrapper { | |||
| float: left; | |||
| width: 100%; | |||
| position: relative; | |||
| } | |||
| #header { | |||
| position: relative; | |||
| padding: 0 15px; | |||
| margin: 0 auto; | |||
| } | |||
| #branding { | |||
| text-align: left; | |||
| position: relative; | |||
| width: 100%; | |||
| float: left; | |||
| margin: 1em 0; | |||
| text-shadow: 0 1px 0 #ffffff; | |||
| } | |||
| #branding h1{ | |||
| font-size: 36px; | |||
| font-family: Georgia,sans-serif; | |||
| margin: 0; | |||
| } | |||
| #branding h1 a{ | |||
| color: inherit; | |||
| text-decoration: inherit; | |||
| } | |||
| #branding h1 a:hover{ | |||
| color: black; | |||
| } | |||
| #branding p { | |||
| margin: 0; | |||
| font-style: italic; | |||
| margin-top: 5px; | |||
| } | |||
| #main-wrapper { | |||
| float: left; | |||
| width: 100%; | |||
| background: #ffffff; | |||
| position: relative; | |||
| border-top: 1px solid #DFDFDF; | |||
| border-bottom: 1px solid #DFDFDF; | |||
| } | |||
| #main { | |||
| position: relative; | |||
| padding: 0; | |||
| margin: 0 auto; | |||
| background: #ffffff; | |||
| overflow: hidden; | |||
| padding: 30px 15px; | |||
| } | |||
| label{ | |||
| width: 100%; | |||
| max-width: 180px; | |||
| float:left; | |||
| } | |||
| input:not([type=submit]), select{ | |||
| float:left; | |||
| width: 100%; | |||
| max-width: 520px; | |||
| font-size: 80%; | |||
| } | |||
| input{ | |||
| padding: 2px; | |||
| } | |||
| input[type=submit]{ | |||
| margin-top: 10px; | |||
| padding: 5px; | |||
| width: 100%; | |||
| } | |||
| span.required { | |||
| color: red; | |||
| } | |||
| </style> | |||
| <link rel="icon" href="" type="image/ico" /> | |||
| <title>HTMLy Installer</title> | |||
| </head> | |||
| <body> | |||
| <div id="cover"> | |||
| <div id="header-wrapper"> | |||
| <header id="header" class="responsive"> | |||
| <div id="branding"> | |||
| <h1> | |||
| HTMLy | |||
| </h1> | |||
| <div id="blog-tagline"> | |||
| <p>the HTMLy Installer Tool</p> | |||
| </div> | |||
| </div> | |||
| </header> | |||
| </div> | |||
| </div> | |||
| <div id="main-wrapper"> | |||
| <div id="main" class="responsive"> | |||
| <form method="POST"> | |||
| <label for="user_name">Username:<span class="required">*</span></label> | |||
| <input name="user_name" value="" placeholder="Your User Name" required> | |||
| <br/> | |||
| <label for="user_password">Password:<span class="required">*</span></label> | |||
| <input name="user_password" value="" type="password" placeholder="Password" required> | |||
| <br/> | |||
| <br/> | |||
| <label for="blog_title">Blog Title:</label> | |||
| <input name="blog_title" value="" placeholder="HTMLy"> | |||
| <br/> | |||
| <label for="blog_tagline">Blog Tagline:</label> | |||
| <input name="blog_tagline" value="" placeholder="Just another HTMLy blog"> | |||
| <br/> | |||
| <label for="blog_description">Blog Description:</label> | |||
| <input name="blog_description" value="" placeholder="Proudly powered by HTMLy, a databaseless blogging platform."> | |||
| <br/> | |||
| <label for="blog_copyright">Blog Copyright:</label> | |||
| <input name="blog_copyright" value="" placeholder="(c) Your name."> | |||
| <br/> | |||
| <br/> | |||
| <label for="social_twitter">Twitter Link:</label> | |||
| <input name="social_twitter" type="url" value="" placeholder="https://twitter.com/gohtmly"> | |||
| <br/> | |||
| <label for="social_facebook">Facebook Link:</label> | |||
| <input name="social_facebook" type="url" value="" placeholder="https://www.facebook.com/gohtmly"> | |||
| <br/> | |||
| <label for="comment_system">Comment System:</label> | |||
| <select name="comment_system" onchange="checkCommentSystemSelection();" id="comment.system"> | |||
| <option value="disable">disable</option> | |||
| <option value="facebook">facebook</option> | |||
| <option value="disqus">disqus</option> | |||
| </select> | |||
| <div id="facebook" style="display:none"> | |||
| <br/> | |||
| <label for="fb_appid">Facebook AppId:</label> | |||
| <input name="fb_appid" value="" placeholder="facebook AppId"> | |||
| </div> | |||
| <div id="disqus" style="display:none"> | |||
| <br/> | |||
| <label for="disqus_shortname">Disqus Shortname:</label> | |||
| <input name="disqus_shortname" value="" placeholder="disqus shortname"> | |||
| </div> | |||
| <br/><input type="submit" value="Install via Tool"> | |||
| </form> | |||
| <br><br> | |||
| <div><small><em>Based on HTMLy installer (https://github.com/Kanti/htmly-installer) by <a href="https://github.com/Kanti" target="_blank">Matthias Vogel</a></em></small></div> | |||
| </div> | |||
| </div> | |||
| <script> | |||
| function checkCommentSystemSelection(){ | |||
| a = document.getElementById("comment.system"); | |||
| if(a.value == "facebook") | |||
| document.getElementById("facebook").setAttribute("style","display:inline"); | |||
| else | |||
| document.getElementById("facebook").setAttribute("style","display:none"); | |||
| if(a.value == "disqus") | |||
| document.getElementById("disqus").setAttribute("style","display:inline"); | |||
| else | |||
| document.getElementById("disqus").setAttribute("style","display:none"); | |||
| return a.value; | |||
| } | |||
| </script> | |||
| </body> | |||
| </html> | |||