Source code of Leftypol imageboard
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

2323 lines
66 KiB

13 years ago
<?php
12 years ago
/*
* Copyright (c) 2010-2014 Tinyboard Development Group
12 years ago
*/
12 years ago
if (realpath($_SERVER['SCRIPT_FILENAME']) == str_replace('\\', '/', __FILE__)) {
12 years ago
// You cannot request this file directly.
exit;
}
define('TINYBOARD', null);
$microtime_start = microtime(true);
12 years ago
require_once 'inc/display.php';
require_once 'inc/template.php';
require_once 'inc/database.php';
require_once 'inc/events.php';
require_once 'inc/api.php';
require_once 'inc/bans.php';
12 years ago
require_once 'inc/lib/gettext/gettext.inc';
// the user is not currently logged in as a moderator
$mod = false;
register_shutdown_function('fatal_error_handler');
mb_internal_encoding('UTF-8');
loadConfig();
function init_locale($locale, $error='error') {
if (_setlocale(LC_ALL, $locale) === false) {
$error('The specified locale (' . $locale . ') does not exist on your platform!');
}
if (extension_loaded('gettext')) {
bindtextdomain('tinyboard', './inc/locale');
bind_textdomain_codeset('tinyboard', 'UTF-8');
textdomain('tinyboard');
} else {
_bindtextdomain('tinyboard', './inc/locale');
_bind_textdomain_codeset('tinyboard', 'UTF-8');
_textdomain('tinyboard');
}
}
$current_locale = 'en';
12 years ago
function loadConfig() {
global $board, $config, $__ip, $debug, $__version, $microtime_start, $current_locale;
12 years ago
$error = function_exists('error') ? 'error' : 'basic_error_function_because_the_other_isnt_loaded_yet';
12 years ago
reset_events();
12 years ago
if (!isset($_SERVER['REMOTE_ADDR']))
12 years ago
$_SERVER['REMOTE_ADDR'] = '0.0.0.0';
$arrays = array(
'db',
'api',
'cache',
'cookies',
'error',
'dir',
'mod',
'spam',
'filters',
'wordfilters',
'custom_capcode',
'custom_tripcode',
'dnsbl',
'dnsbl_exceptions',
'remote',
'allowed_ext',
'allowed_ext_files',
'file_icons',
'footer',
'stylesheets',
'additional_javascript',
'markup',
'custom_pages',
'dashboard_links'
);
12 years ago
$config = array();
12 years ago
foreach ($arrays as $key) {
12 years ago
$config[$key] = array();
}
12 years ago
if (!file_exists('inc/instance-config.php'))
12 years ago
$error('Tinyboard is not configured! Create inc/instance-config.php.');
// Initialize locale as early as possible
$config['locale'] = 'en';
$configstr = file_get_contents('inc/instance-config.php');
if (isset($board['dir']) && file_exists($board['dir'] . '/config.php')) {
$configstr .= file_get_contents($board['dir'] . '/config.php');
}
$matches = array();
preg_match_all('/[^\/*#]\$config\s*\[\s*[\'"]locale[\'"]\s*\]\s*=\s*([\'"])(.*?)\1/', $configstr, $matches);
if ($matches && isset ($matches[2]) && $matches[2]) {
$matches = $matches[2];
$config['locale'] = $matches[count($matches)-1];
}
if ($config['locale'] != $current_locale) {
$current_locale = $config['locale'];
init_locale($config['locale'], $error);
}
require 'inc/config.php';
12 years ago
require 'inc/instance-config.php';
12 years ago
if (isset($board['dir']) && file_exists($board['dir'] . '/config.php')) {
12 years ago
require $board['dir'] . '/config.php';
}
if ($config['locale'] != $current_locale) {
$current_locale = $config['locale'];
init_locale($config['locale'], $error);
}
12 years ago
if (!isset($__version))
12 years ago
$__version = file_exists('.installed') ? trim(file_get_contents('.installed')) : false;
$config['version'] = $__version;
12 years ago
date_default_timezone_set($config['timezone']);
if (!isset($config['global_message']))
$config['global_message'] = false;
12 years ago
if (!isset($config['post_url']))
12 years ago
$config['post_url'] = $config['root'] . $config['file_post'];
12 years ago
if (!isset($config['referer_match']))
if (isset($_SERVER['HTTP_HOST'])) {
12 years ago
$config['referer_match'] = '/^' .
10 years ago
(preg_match('@^https?://@', $config['root']) ? '' :
12 years ago
'https?:\/\/' . $_SERVER['HTTP_HOST']) .
preg_quote($config['root'], '/') .
'(' .
str_replace('%s', $config['board_regex'], preg_quote($config['board_path'], '/')) .
12 years ago
'(' .
preg_quote($config['file_index'], '/') . '|' .
str_replace('%d', '\d+', preg_quote($config['file_page'])) .
')?' .
'|' .
str_replace('%s', $config['board_regex'], preg_quote($config['board_path'], '/')) .
12 years ago
preg_quote($config['dir']['res'], '/') .
'(' .
str_replace('%d', '\d+', preg_quote($config['file_page'], '/')) . '|' .
str_replace('%d', '\d+', preg_quote($config['file_page50'], '/')) .
')' .
12 years ago
'|' .
preg_quote($config['file_mod'], '/') . '\?\/.+' .
')([#?](.+)?)?$/ui';
} else {
12 years ago
// CLI mode
$config['referer_match'] = '//';
}
12 years ago
if (!isset($config['cookies']['path']))
12 years ago
$config['cookies']['path'] = &$config['root'];
12 years ago
if (!isset($config['dir']['static']))
12 years ago
$config['dir']['static'] = $config['root'] . 'static/';
if (!isset($config['image_blank']))
$config['image_blank'] = $config['dir']['static'] . 'blank.gif';
12 years ago
if (!isset($config['image_sticky']))
12 years ago
$config['image_sticky'] = $config['dir']['static'] . 'sticky.gif';
12 years ago
if (!isset($config['image_locked']))
12 years ago
$config['image_locked'] = $config['dir']['static'] . 'locked.gif';
12 years ago
if (!isset($config['image_bumplocked']))
12 years ago
$config['image_bumplocked'] = $config['dir']['static'] . 'sage.gif';
12 years ago
if (!isset($config['image_deleted']))
12 years ago
$config['image_deleted'] = $config['dir']['static'] . 'deleted.png';
12 years ago
if (!isset($config['uri_thumb']))
12 years ago
$config['uri_thumb'] = $config['root'] . $board['dir'] . $config['dir']['thumb'];
12 years ago
elseif (isset($board['dir']))
12 years ago
$config['uri_thumb'] = sprintf($config['uri_thumb'], $board['dir']);
12 years ago
if (!isset($config['uri_img']))
12 years ago
$config['uri_img'] = $config['root'] . $board['dir'] . $config['dir']['img'];
12 years ago
elseif (isset($board['dir']))
12 years ago
$config['uri_img'] = sprintf($config['uri_img'], $board['dir']);
12 years ago
if (!isset($config['uri_stylesheets']))
12 years ago
$config['uri_stylesheets'] = $config['root'] . 'stylesheets/';
12 years ago
if (!isset($config['url_stylesheet']))
12 years ago
$config['url_stylesheet'] = $config['uri_stylesheets'] . 'style.css';
12 years ago
if (!isset($config['url_javascript']))
12 years ago
$config['url_javascript'] = $config['root'] . $config['file_script'];
12 years ago
if (!isset($config['additional_javascript_url']))
12 years ago
$config['additional_javascript_url'] = $config['root'];
if (!isset($config['uri_flags']))
$config['uri_flags'] = $config['root'] . 'static/flags/%s.png';
if (!isset($config['user_flag']))
$config['user_flag'] = false;
if (!isset($config['user_flags']))
$config['user_flags'] = array();
12 years ago
if ($config['root_file']) {
12 years ago
chdir($config['root_file']);
12 years ago
}
12 years ago
12 years ago
if ($config['verbose_errors']) {
set_error_handler('verbose_error_handler');
12 years ago
error_reporting(E_ALL);
ini_set('display_errors', true);
ini_set('html_errors', false);
} else {
ini_set('display_errors', false);
12 years ago
}
12 years ago
// Keep the original address to properly comply with other board configurations
12 years ago
if (!isset($__ip))
12 years ago
$__ip = $_SERVER['REMOTE_ADDR'];
12 years ago
// ::ffff:0.0.0.0
12 years ago
if (preg_match('/^\:\:(ffff\:)?(\d+\.\d+\.\d+\.\d+)$/', $__ip, $m))
12 years ago
$_SERVER['REMOTE_ADDR'] = $m[2];
12 years ago
if ($config['syslog'])
12 years ago
openlog('tinyboard', LOG_ODELAY, LOG_SYSLOG); // open a connection to sysem logger
12 years ago
if ($config['recaptcha'])
12 years ago
require_once 'inc/lib/recaptcha/recaptchalib.php';
12 years ago
if ($config['cache']['enabled'])
12 years ago
require_once 'inc/cache.php';
if (in_array('webm', $config['allowed_ext_files'])) {
require_once 'inc/lib/webm/posthandler.php';
event_handler('post', 'postHandler');
}
event('load-config');
if ($config['debug']) {
if (!isset($debug)) {
$debug = array(
'sql' => array(),
'exec' => array(),
'purge' => array(),
'cached' => array(),
'write' => array(),
'time' => array(
'db_queries' => 0,
'exec' => 0,
),
'start' => $microtime_start,
'start_debug' => microtime(true)
);
$debug['start'] = $microtime_start;
}
}
12 years ago
}
function basic_error_function_because_the_other_isnt_loaded_yet($message, $priority = true) {
global $config;
12 years ago
if ($config['syslog'] && $priority !== false) {
12 years ago
// Use LOG_NOTICE instead of LOG_ERR or LOG_WARNING because most error message are not significant.
_syslog($priority !== true ? $priority : LOG_NOTICE, $message);
}
12 years ago
// Yes, this is horrible.
die('<!DOCTYPE html><html><head><title>Error</title>' .
'<style type="text/css">' .
'body{text-align:center;font-family:arial, helvetica, sans-serif;font-size:10pt;}' .
'p{padding:0;margin:20px 0;}' .
'p.c{font-size:11px;}' .
'</style></head>' .
'<body><h2>Error</h2>' . $message . '<hr/>' .
'<p class="c">This alternative error page is being displayed because the other couldn\'t be found or hasn\'t loaded yet.</p></body></html>');
}
function fatal_error_handler() {
12 years ago
if ($error = error_get_last()) {
if ($error['type'] == E_ERROR) {
if (function_exists('error')) {
12 years ago
error('Caught fatal error: ' . $error['message'] . ' in <strong>' . $error['file'] . '</strong> on line ' . $error['line'], LOG_ERR);
} else {
basic_error_function_because_the_other_isnt_loaded_yet('Caught fatal error: ' . $error['message'] . ' in ' . $error['file'] . ' on line ' . $error['line'], LOG_ERR);
}
13 years ago
}
}
12 years ago
}
12 years ago
12 years ago
function _syslog($priority, $message) {
12 years ago
if (isset($_SERVER['REMOTE_ADDR'], $_SERVER['REQUEST_METHOD'], $_SERVER['REQUEST_URI'])) {
12 years ago
// CGI
syslog($priority, $message . ' - client: ' . $_SERVER['REMOTE_ADDR'] . ', request: "' . $_SERVER['REQUEST_METHOD'] . ' ' . $_SERVER['REQUEST_URI'] . '"');
} else {
syslog($priority, $message);
}
}
12 years ago
function verbose_error_handler($errno, $errstr, $errfile, $errline) {
if (error_reporting() == 0)
return false; // Looks like this warning was suppressed by the @ operator.
error(utf8tohtml($errstr), true, array(
'file' => $errfile . ':' . $errline,
'errno' => $errno,
'error' => $errstr,
'backtrace' => array_slice(debug_backtrace(), 1)
));
}
function define_groups() {
global $config;
foreach ($config['mod']['groups'] as $group_value => $group_name)
defined($group_name) or define($group_name, $group_value, true);
ksort($config['mod']['groups']);
}
function create_antibot($board, $thread = null) {
require_once dirname(__FILE__) . '/anti-bot.php';
return _create_antibot($board, $thread);
}
12 years ago
function rebuildThemes($action, $boardname = false) {
global $config, $board;
// Save the global variables
$_config = $config;
$_board = $board;
12 years ago
// List themes
$query = query("SELECT `theme` FROM ``theme_settings`` WHERE `name` IS NULL AND `value` IS NULL") or error(db_error());
11 years ago
10 years ago
while ($theme = $query->fetch(PDO::FETCH_ASSOC)) {
rebuildTheme($theme['theme'], $action, $boardname);
12 years ago
}
// Restore them
$config = $_config;
$board = $_board;
12 years ago
}
function loadThemeConfig($_theme) {
global $config;
12 years ago
if (!file_exists($config['dir']['themes'] . '/' . $_theme . '/info.php'))
12 years ago
return false;
// Load theme information into $theme
include $config['dir']['themes'] . '/' . $_theme . '/info.php';
return $theme;
}
function rebuildTheme($theme, $action, $board = false) {
12 years ago
global $config, $_theme;
$_theme = $theme;
$theme = loadThemeConfig($_theme);
12 years ago
if (file_exists($config['dir']['themes'] . '/' . $_theme . '/theme.php')) {
12 years ago
require_once $config['dir']['themes'] . '/' . $_theme . '/theme.php';
$theme['build_function']($action, themeSettings($_theme), $board);
12 years ago
}
}
function themeSettings($theme) {
$query = prepare("SELECT `name`, `value` FROM ``theme_settings`` WHERE `theme` = :theme AND `name` IS NOT NULL");
12 years ago
$query->bindValue(':theme', $theme);
$query->execute() or error(db_error($query));
$settings = array();
10 years ago
while ($s = $query->fetch(PDO::FETCH_ASSOC)) {
12 years ago
$settings[$s['name']] = $s['value'];
}
return $settings;
}
function sprintf3($str, $vars, $delim = '%') {
$replaces = array();
12 years ago
foreach ($vars as $k => $v) {
12 years ago
$replaces[$delim . $k . $delim] = $v;
}
return str_replace(array_keys($replaces),
array_values($replaces), $str);
12 years ago
}
function mb_substr_replace($string, $replacement, $start, $length) {
return mb_substr($string, 0, $start) . $replacement . mb_substr($string, $start + $length);
}
12 years ago
function setupBoard($array) {
global $board, $config;
12 years ago
$board = array(
'uri' => $array['uri'],
'title' => $array['title'],
'subtitle' => $array['subtitle']
);
// older versions
$board['name'] = &$board['title'];
12 years ago
$board['dir'] = sprintf($config['board_path'], $board['uri']);
$board['url'] = sprintf($config['board_abbreviation'], $board['uri']);
12 years ago
loadConfig();
12 years ago
if (!file_exists($board['dir']))
@mkdir($board['dir'], 0777) or error("Couldn't create " . $board['dir'] . ". Check permissions.", true);
12 years ago
if (!file_exists($board['dir'] . $config['dir']['img']))
@mkdir($board['dir'] . $config['dir']['img'], 0777)
or error("Couldn't create " . $board['dir'] . $config['dir']['img'] . ". Check permissions.", true);
12 years ago
if (!file_exists($board['dir'] . $config['dir']['thumb']))
@mkdir($board['dir'] . $config['dir']['thumb'], 0777)
or error("Couldn't create " . $board['dir'] . $config['dir']['img'] . ". Check permissions.", true);
12 years ago
if (!file_exists($board['dir'] . $config['dir']['res']))
@mkdir($board['dir'] . $config['dir']['res'], 0777)
or error("Couldn't create " . $board['dir'] . $config['dir']['img'] . ". Check permissions.", true);
12 years ago
}
function openBoard($uri) {
global $config, $build_pages;
if ($config['try_smarter'])
$build_pages = array();
$board = getBoardInfo($uri);
if ($board) {
setupBoard($board);
return true;
}
return false;
}
function getBoardInfo($uri) {
12 years ago
global $config;
11 years ago
12 years ago
if ($config['cache']['enabled'] && ($board = cache::get('board_' . $uri))) {
return $board;
}
$query = prepare("SELECT * FROM ``boards`` WHERE `uri` = :uri LIMIT 1");
12 years ago
$query->bindValue(':uri', $uri);
$query->execute() or error(db_error($query));
10 years ago
if ($board = $query->fetch(PDO::FETCH_ASSOC)) {
12 years ago
if ($config['cache']['enabled'])
12 years ago
cache::set('board_' . $uri, $board);
return $board;
11 years ago
}
return false;
12 years ago
}
12 years ago
12 years ago
function boardTitle($uri) {
$board = getBoardInfo($uri);
if ($board)
12 years ago
return $board['title'];
11 years ago
return false;
12 years ago
}
function purge($uri) {
global $config, $debug;
// Fix for Unicode
$uri = rawurlencode($uri);
$noescape = "/!~*()+:";
$noescape = preg_split('//', $noescape);
$noescape_url = array_map("rawurlencode", $noescape);
$uri = str_replace($noescape_url, $noescape, $uri);
if (preg_match($config['referer_match'], $config['root']) && isset($_SERVER['REQUEST_URI'])) {
12 years ago
$uri = (str_replace('\\', '/', dirname($_SERVER['REQUEST_URI'])) == '/' ? '/' : str_replace('\\', '/', dirname($_SERVER['REQUEST_URI'])) . '/') . $uri;
} else {
$uri = $config['root'] . $uri;
}
12 years ago
if ($config['debug']) {
12 years ago
$debug['purge'][] = $uri;
}
12 years ago
foreach ($config['purge'] as &$purge) {
12 years ago
$host = &$purge[0];
$port = &$purge[1];
$http_host = isset($purge[2]) ? $purge[2] : (isset($_SERVER['HTTP_HOST']) ? $_SERVER['HTTP_HOST'] : 'localhost');
12 years ago
$request = "PURGE {$uri} HTTP/1.1\r\nHost: {$http_host}\r\nUser-Agent: Tinyboard\r\nConnection: Close\r\n\r\n";
12 years ago
if ($fp = fsockopen($host, $port, $errno, $errstr, $config['purge_timeout'])) {
12 years ago
fwrite($fp, $request);
fclose($fp);
} else {
12 years ago
// Cannot connect?
error('Could not PURGE for ' . $host);
}
}
12 years ago
}
function file_write($path, $data, $simple = false, $skip_purge = false) {
global $config, $debug;
12 years ago
if (preg_match('/^remote:\/\/(.+)\:(.+)$/', $path, $m)) {
if (isset($config['remote'][$m[1]])) {
12 years ago
require_once 'inc/remote.php';
12 years ago
$remote = new Remote($config['remote'][$m[1]]);
$remote->write($data, $m[2]);
return;
} else {
error('Invalid remote server: ' . $m[1]);
}
}
12 years ago
if (!$fp = fopen($path, $simple ? 'w' : 'c'))
12 years ago
error('Unable to open file for writing: ' . $path);
12 years ago
// File locking
12 years ago
if (!$simple && !flock($fp, LOCK_EX)) {
12 years ago
error('Unable to lock file: ' . $path);
}
12 years ago
// Truncate file
12 years ago
if (!$simple && !ftruncate($fp, 0))
12 years ago
error('Unable to truncate file: ' . $path);
12 years ago
// Write data
if (($bytes = fwrite($fp, $data)) === false)
12 years ago
error('Unable to write to file: ' . $path);
12 years ago
// Unlock
12 years ago
if (!$simple)
12 years ago
flock($fp, LOCK_UN);
12 years ago
// Close
12 years ago
if (!fclose($fp))
12 years ago
error('Unable to close file: ' . $path);
/**
* Create gzipped file.
*
* When writing into a file foo.bar and the size is larger or equal to 1
* KiB, this also produces the gzipped version foo.bar.gz
*
* This is useful with nginx with gzip_static on.
*/
if ($config['gzip_static']) {
$gzpath = "$path.gz";
if ($bytes & ~0x3ff) { // if ($bytes >= 1024)
if (file_put_contents($gzpath, gzencode($data), $simple ? 0 : LOCK_EX) === false)
error("Unable to write to file: $gzpath");
if (!touch($gzpath, filemtime($path), fileatime($path)))
error("Unable to touch file: $gzpath");
}
else {
@unlink($gzpath);
}
}
if (!$skip_purge && isset($config['purge'])) {
12 years ago
// Purge cache
12 years ago
if (basename($path) == $config['file_index']) {
12 years ago
// Index file (/index.html); purge "/" as well
$uri = dirname($path);
// root
12 years ago
if ($uri == '.')
12 years ago
$uri = '';
else
$uri .= '/';
purge($uri);
13 years ago
}
12 years ago
purge($path);
13 years ago
}
if ($config['debug']) {
$debug['write'][] = $path . ': ' . $bytes . ' bytes';
}
12 years ago
event('write', $path);
}
function file_unlink($path) {
global $config, $debug;
12 years ago
if ($config['debug']) {
if (!isset($debug['unlink']))
12 years ago
$debug['unlink'] = array();
$debug['unlink'][] = $path;
}
12 years ago
$ret = @unlink($path);
12 years ago
if (isset($config['purge']) && $path[0] != '/' && isset($_SERVER['HTTP_HOST'])) {
12 years ago
// Purge cache
12 years ago
if (basename($path) == $config['file_index']) {
12 years ago
// Index file (/index.html); purge "/" as well
$uri = dirname($path);
// root
12 years ago
if ($uri == '.')
12 years ago
$uri = '';
else
$uri .= '/';
purge($uri);
13 years ago
}
12 years ago
purge($path);
13 years ago
}
12 years ago
event('unlink', $path);
12 years ago
return $ret;
}
function hasPermission($action = null, $board = null, $_mod = null) {
global $config;
12 years ago
if (isset($_mod))
12 years ago
$mod = &$_mod;
else
global $mod;
12 years ago
if (!is_array($mod))
12 years ago
return false;
12 years ago
if (isset($action) && $mod['type'] < $action)
12 years ago
return false;
12 years ago
if (!isset($board) || $config['mod']['skip_per_board'])
12 years ago
return true;
12 years ago
if (!isset($mod['boards']))
12 years ago
return false;
12 years ago
if (!in_array('*', $mod['boards']) && !in_array($board, $mod['boards']))
12 years ago
return false;
12 years ago
return true;
}
function listBoards($just_uri = false) {
12 years ago
global $config;
$just_uri ? $cache_name = 'all_boards_uri' : $cache_name = 'all_boards';
if ($config['cache']['enabled'] && ($boards = cache::get($cache_name)))
return $boards;
if (!$just_uri) {
$query = query("SELECT * FROM ``boards`` ORDER BY `uri`") or error(db_error());
$boards = $query->fetchAll();
} else {
$boards = array();
$query = query("SELECT `uri` FROM ``boards``") or error(db_error());
while ($board = $query->fetchColumn()) {
$boards[] = $board;
}
}
12 years ago
if ($config['cache']['enabled'])
cache::set($cache_name, $boards);
12 years ago
return $boards;
}
function until($timestamp) {
$difference = $timestamp - time();
12 years ago
if ($difference < 60) {
return $difference . ' ' . ngettext('second', 'seconds', $difference);
12 years ago
} elseif ($difference < 60*60) {
return ($num = round($difference/(60))) . ' ' . ngettext('minute', 'minutes', $num);
12 years ago
} elseif ($difference < 60*60*24) {
return ($num = round($difference/(60*60))) . ' ' . ngettext('hour', 'hours', $num);
12 years ago
} elseif ($difference < 60*60*24*7) {
return ($num = round($difference/(60*60*24))) . ' ' . ngettext('day', 'days', $num);
12 years ago
} elseif ($difference < 60*60*24*365) {
return ($num = round($difference/(60*60*24*7))) . ' ' . ngettext('week', 'weeks', $num);
12 years ago
}
11 years ago
return ($num = round($difference/(60*60*24*365))) . ' ' . ngettext('year', 'years', $num);
12 years ago