2012-08-26 06:49:11 +08:00
|
|
|
<?php
|
2016-07-21 23:09:48 +08:00
|
|
|
|
2016-08-09 17:54:42 +08:00
|
|
|
use PrivateBin\Persistence\ServerSalt;
|
2016-07-21 23:09:48 +08:00
|
|
|
|
2016-07-26 14:19:35 +08:00
|
|
|
error_reporting(E_ALL | E_STRICT);
|
2012-08-26 06:49:11 +08:00
|
|
|
|
|
|
|
// change this, if your php files and data is outside of your webservers document root
|
2016-07-26 14:19:35 +08:00
|
|
|
if (!defined('PUBLIC_PATH')) {
|
|
|
|
define('PUBLIC_PATH', '..');
|
|
|
|
}
|
|
|
|
if (!defined('PATH')) {
|
|
|
|
define('PATH', '..' . DIRECTORY_SEPARATOR);
|
|
|
|
}
|
|
|
|
if (!defined('CONF')) {
|
2017-10-04 01:45:47 +08:00
|
|
|
define('CONF', PATH . 'cfg' . DIRECTORY_SEPARATOR . 'conf.php');
|
2016-07-26 14:19:35 +08:00
|
|
|
}
|
|
|
|
if (!is_file(CONF)) {
|
2017-10-04 01:45:47 +08:00
|
|
|
copy(PATH . 'cfg' . DIRECTORY_SEPARATOR . 'conf.sample.php', CONF);
|
2016-07-26 14:19:35 +08:00
|
|
|
}
|
2012-08-26 06:49:11 +08:00
|
|
|
|
2016-07-21 23:09:48 +08:00
|
|
|
require PATH . 'vendor/autoload.php';
|
2016-08-16 17:11:03 +08:00
|
|
|
Helper::updateSubresourceIntegrity();
|
2012-08-26 06:49:11 +08:00
|
|
|
|
2016-08-09 17:54:42 +08:00
|
|
|
class Helper
|
2012-08-26 06:49:11 +08:00
|
|
|
{
|
2015-09-22 04:32:52 +08:00
|
|
|
/**
|
|
|
|
* example ID of a paste
|
|
|
|
*
|
|
|
|
* @var string
|
|
|
|
*/
|
|
|
|
private static $pasteid = '5e9bc25c89fb3bf9';
|
|
|
|
|
|
|
|
/**
|
|
|
|
* example paste
|
|
|
|
*
|
|
|
|
* @var array
|
|
|
|
*/
|
|
|
|
private static $paste = array(
|
2016-10-29 16:24:08 +08:00
|
|
|
'data' => '{"iv":"EN39/wd5Nk8HAiSG2K5AsQ","v":1,"iter":1000,"ks":128,"ts":64,"mode":"ccm","adata":"","cipher":"aes","salt":"QKN1DBXe5PI","ct":"8hA83xDdXjD7K2qfmw5NdA"}',
|
|
|
|
'attachment' => '{"iv":"Pd4pOKWkmDTT9uPwVwd5Ag","v":1,"iter":1000,"ks":128,"ts":64,"mode":"ccm","adata":"","cipher":"aes","salt":"ZIUhFTliVz4","ct":"6nOCU3peNDclDDpFtJEBKA"}',
|
2015-09-26 18:29:27 +08:00
|
|
|
'attachmentname' => '{"iv":"76MkAtOGC4oFogX/aSMxRA","v":1,"iter":1000,"ks":128,"ts":64,"mode":"ccm","adata":"","cipher":"aes","salt":"ZIUhFTliVz4","ct":"b6Ae/U1xJdsX/+lATud4sQ"}',
|
2016-10-29 16:24:08 +08:00
|
|
|
'meta' => array(
|
|
|
|
'formatter' => 'plaintext',
|
|
|
|
'postdate' => 1344803344,
|
2015-09-22 04:32:52 +08:00
|
|
|
'opendiscussion' => true,
|
|
|
|
),
|
|
|
|
);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* example ID of a comment
|
|
|
|
*
|
|
|
|
* @var string
|
|
|
|
*/
|
|
|
|
private static $commentid = '5a52eebf11c4c94b';
|
|
|
|
|
|
|
|
/**
|
|
|
|
* example comment
|
|
|
|
*
|
|
|
|
* @var array
|
|
|
|
*/
|
|
|
|
private static $comment = array(
|
|
|
|
'data' => '{"iv":"Pd4pOKWkmDTT9uPwVwd5Ag","v":1,"iter":1000,"ks":128,"ts":64,"mode":"ccm","adata":"","cipher":"aes","salt":"ZIUhFTliVz4","ct":"6nOCU3peNDclDDpFtJEBKA"}',
|
|
|
|
'meta' => array(
|
|
|
|
'nickname' => '{"iv":"76MkAtOGC4oFogX/aSMxRA","v":1,"iter":1000,"ks":128,"ts":64,"mode":"ccm","adata":"","cipher":"aes","salt":"ZIUhFTliVz4","ct":"b6Ae/U1xJdsX/+lATud4sQ"}',
|
2016-10-29 16:24:08 +08:00
|
|
|
'vizhash' => '',
|
2015-09-22 04:32:52 +08:00
|
|
|
'postdate' => 1344803528,
|
|
|
|
),
|
|
|
|
);
|
|
|
|
|
2016-08-16 17:11:03 +08:00
|
|
|
/**
|
|
|
|
* JS files and their SRI hashes
|
|
|
|
*
|
|
|
|
* @var array
|
|
|
|
*/
|
|
|
|
private static $hashes = array();
|
|
|
|
|
2015-09-22 04:32:52 +08:00
|
|
|
/**
|
|
|
|
* get example paste ID
|
|
|
|
*
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
public static function getPasteId()
|
|
|
|
{
|
|
|
|
return self::$pasteid;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* get example paste
|
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
public static function getPaste($meta = array())
|
2015-09-26 18:29:27 +08:00
|
|
|
{
|
|
|
|
$example = self::getPasteWithAttachment($meta);
|
|
|
|
unset($example['attachment'], $example['attachmentname']);
|
|
|
|
return $example;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* get example paste
|
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
public static function getPasteWithAttachment($meta = array())
|
2015-09-22 04:32:52 +08:00
|
|
|
{
|
2016-10-29 16:24:08 +08:00
|
|
|
$example = self::$paste;
|
2016-08-22 22:20:14 +08:00
|
|
|
$example['meta']['salt'] = ServerSalt::generate();
|
2016-10-29 16:24:08 +08:00
|
|
|
$example['meta'] = array_merge($example['meta'], $meta);
|
2015-09-22 04:32:52 +08:00
|
|
|
return $example;
|
|
|
|
}
|
|
|
|
|
2015-10-18 17:08:28 +08:00
|
|
|
/**
|
|
|
|
* get example paste
|
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
public static function getPasteAsJson($meta = array())
|
|
|
|
{
|
|
|
|
$example = self::getPaste();
|
2016-07-06 17:37:13 +08:00
|
|
|
// the JSON shouldn't contain the salt
|
|
|
|
unset($example['meta']['salt']);
|
2016-07-26 14:19:35 +08:00
|
|
|
if (count($meta)) {
|
2015-10-18 17:08:28 +08:00
|
|
|
$example['meta'] = $meta;
|
2016-07-26 14:19:35 +08:00
|
|
|
}
|
2016-10-29 16:24:08 +08:00
|
|
|
$example['comments'] = array();
|
|
|
|
$example['comment_count'] = 0;
|
2015-10-18 17:08:28 +08:00
|
|
|
$example['comment_offset'] = 0;
|
2016-10-29 16:24:08 +08:00
|
|
|
$example['@context'] = 'js/paste.jsonld';
|
2015-10-18 17:08:28 +08:00
|
|
|
return json_encode($example);
|
|
|
|
}
|
|
|
|
|
2015-09-22 04:32:52 +08:00
|
|
|
/**
|
|
|
|
* get example paste ID
|
|
|
|
*
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
public static function getCommentId()
|
|
|
|
{
|
|
|
|
return self::$commentid;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* get example comment
|
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
public static function getComment($meta = array())
|
|
|
|
{
|
2016-10-29 16:24:08 +08:00
|
|
|
$example = self::$comment;
|
2015-09-22 04:32:52 +08:00
|
|
|
$example['meta'] = array_merge($example['meta'], $meta);
|
|
|
|
return $example;
|
|
|
|
}
|
|
|
|
|
2015-10-03 21:52:37 +08:00
|
|
|
/**
|
|
|
|
* get example comment
|
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
public static function getCommentPost($meta = array())
|
|
|
|
{
|
2016-10-29 16:24:08 +08:00
|
|
|
$example = self::getComment($meta);
|
2015-10-03 21:52:37 +08:00
|
|
|
$example['nickname'] = $example['meta']['nickname'];
|
|
|
|
unset($example['meta']['nickname']);
|
|
|
|
return $example;
|
|
|
|
}
|
|
|
|
|
2015-08-29 16:41:10 +08:00
|
|
|
/**
|
|
|
|
* delete directory and all its contents recursively
|
|
|
|
*
|
|
|
|
* @param string $path
|
|
|
|
* @throws Exception
|
|
|
|
*/
|
2016-08-09 17:54:42 +08:00
|
|
|
public static function rmDir($path)
|
2015-08-28 03:41:21 +08:00
|
|
|
{
|
2017-03-25 06:42:11 +08:00
|
|
|
if (is_dir($path)) {
|
|
|
|
$path .= DIRECTORY_SEPARATOR;
|
|
|
|
$dir = dir($path);
|
|
|
|
while (false !== ($file = $dir->read())) {
|
|
|
|
if ($file != '.' && $file != '..') {
|
|
|
|
if (is_dir($path . $file)) {
|
|
|
|
self::rmDir($path . $file);
|
|
|
|
} elseif (is_file($path . $file)) {
|
|
|
|
if (!unlink($path . $file)) {
|
|
|
|
throw new Exception('Error deleting file "' . $path . $file . '".');
|
|
|
|
}
|
2015-08-28 03:41:21 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2017-03-25 06:42:11 +08:00
|
|
|
$dir->close();
|
|
|
|
if (!rmdir($path)) {
|
|
|
|
throw new Exception('Error deleting directory "' . $path . '".');
|
|
|
|
}
|
2015-08-28 03:41:21 +08:00
|
|
|
}
|
|
|
|
}
|
2015-08-29 16:41:10 +08:00
|
|
|
|
2015-09-23 05:21:31 +08:00
|
|
|
/**
|
|
|
|
* create a backup of the config file
|
|
|
|
*
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public static function confBackup()
|
|
|
|
{
|
2016-07-26 14:19:35 +08:00
|
|
|
if (!is_file(CONF . '.bak') && is_file(CONF)) {
|
2015-09-23 05:21:31 +08:00
|
|
|
rename(CONF, CONF . '.bak');
|
2016-07-26 14:19:35 +08:00
|
|
|
}
|
2015-09-23 05:21:31 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* restor backup of the config file
|
|
|
|
*
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public static function confRestore()
|
|
|
|
{
|
2016-07-26 14:19:35 +08:00
|
|
|
if (is_file(CONF . '.bak')) {
|
2015-09-23 05:21:31 +08:00
|
|
|
rename(CONF . '.bak', CONF);
|
2016-07-26 14:19:35 +08:00
|
|
|
}
|
2015-09-23 05:21:31 +08:00
|
|
|
}
|
|
|
|
|
2015-08-29 16:41:10 +08:00
|
|
|
/**
|
|
|
|
* create ini file
|
|
|
|
*
|
|
|
|
* @param string $pathToFile
|
|
|
|
* @param array $values
|
|
|
|
*/
|
|
|
|
public static function createIniFile($pathToFile, $values)
|
|
|
|
{
|
|
|
|
if (count($values)) {
|
|
|
|
@unlink($pathToFile);
|
|
|
|
$ini = fopen($pathToFile, 'a');
|
|
|
|
foreach ($values as $section => $options) {
|
|
|
|
fwrite($ini, "[$section]" . PHP_EOL);
|
2016-07-26 14:19:35 +08:00
|
|
|
foreach ($options as $option => $setting) {
|
2015-08-29 16:41:10 +08:00
|
|
|
if (is_null($setting)) {
|
|
|
|
continue;
|
|
|
|
} elseif (is_string($setting)) {
|
|
|
|
$setting = '"' . $setting . '"';
|
2015-09-27 09:03:55 +08:00
|
|
|
} elseif (is_array($setting)) {
|
|
|
|
foreach ($setting as $key => $value) {
|
|
|
|
if (is_null($value)) {
|
|
|
|
$value = 'null';
|
|
|
|
} elseif (is_string($value)) {
|
|
|
|
$value = '"' . $value . '"';
|
|
|
|
} else {
|
|
|
|
$value = var_export($value, true);
|
|
|
|
}
|
|
|
|
fwrite($ini, $option . "[$key] = $value" . PHP_EOL);
|
|
|
|
}
|
|
|
|
continue;
|
2015-08-29 16:41:10 +08:00
|
|
|
} else {
|
|
|
|
$setting = var_export($setting, true);
|
|
|
|
}
|
|
|
|
fwrite($ini, "$option = $setting" . PHP_EOL);
|
|
|
|
}
|
|
|
|
fwrite($ini, PHP_EOL);
|
|
|
|
}
|
|
|
|
fclose($ini);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* a var_export that returns arrays without line breaks
|
|
|
|
* by linus@flowingcreativity.net via php.net
|
|
|
|
*
|
|
|
|
* @param mixed $var
|
|
|
|
* @param bool $return
|
|
|
|
* @return void|string
|
|
|
|
*/
|
2016-08-09 17:54:42 +08:00
|
|
|
public static function varExportMin($var, $return = false)
|
2015-08-29 16:41:10 +08:00
|
|
|
{
|
|
|
|
if (is_array($var)) {
|
|
|
|
$toImplode = array();
|
|
|
|
foreach ($var as $key => $value) {
|
2016-08-09 17:54:42 +08:00
|
|
|
$toImplode[] = var_export($key, true) . ' => ' . self::varExportMin($value, true);
|
2015-08-29 16:41:10 +08:00
|
|
|
}
|
|
|
|
$code = 'array(' . implode(', ', $toImplode) . ')';
|
|
|
|
if ($return) {
|
|
|
|
return $code;
|
|
|
|
} else {
|
|
|
|
echo $code;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
return var_export($var, $return);
|
|
|
|
}
|
|
|
|
}
|
2016-08-16 17:11:03 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* update all templates with the latest SRI hashes for all JS files
|
|
|
|
*
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public static function updateSubresourceIntegrity()
|
|
|
|
{
|
|
|
|
$dir = dir(PATH . 'js');
|
|
|
|
while (false !== ($file = $dir->read())) {
|
|
|
|
if (substr($file, -3) === '.js') {
|
|
|
|
self::$hashes[$file] = base64_encode(
|
|
|
|
hash('sha512', file_get_contents(
|
|
|
|
PATH . 'js' . DIRECTORY_SEPARATOR . $file
|
|
|
|
), true)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$dir = dir(PATH . 'tpl');
|
|
|
|
while (false !== ($file = $dir->read())) {
|
|
|
|
if (substr($file, -4) === '.php') {
|
|
|
|
$content = file_get_contents(
|
|
|
|
PATH . 'tpl' . DIRECTORY_SEPARATOR . $file
|
|
|
|
);
|
|
|
|
$content = preg_replace_callback(
|
|
|
|
'#<script type="text/javascript" src="js/([a-z0-9.-]+.js)([^"]*)"( integrity="[^"]+" crossorigin="[^"]+")?></script>#',
|
|
|
|
function ($matches) {
|
|
|
|
if (array_key_exists($matches[1], Helper::$hashes)) {
|
|
|
|
return '<script type="text/javascript" src="js/' .
|
|
|
|
$matches[1] . $matches[2] .
|
|
|
|
'" integrity="sha512-' . Helper::$hashes[$matches[1]] .
|
|
|
|
'" crossorigin="anonymous"></script>';
|
|
|
|
} else {
|
|
|
|
return $matches[0];
|
|
|
|
}
|
|
|
|
},
|
|
|
|
$content
|
|
|
|
);
|
|
|
|
file_put_contents(
|
|
|
|
PATH . 'tpl' . DIRECTORY_SEPARATOR . $file,
|
|
|
|
$content
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2016-07-06 15:01:10 +08:00
|
|
|
}
|