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.
2325 lines
75 KiB
2325 lines
75 KiB
11 years ago
|
<?php
|
||
|
|
||
|
/*
|
||
10 years ago
|
* Copyright (c) 2010-2013 Tinyboard Development Group
|
||
11 years ago
|
*/
|
||
|
|
||
|
if (realpath($_SERVER['SCRIPT_FILENAME']) == str_replace('\\', '/', __FILE__)) {
|
||
|
// You cannot request this file directly.
|
||
|
exit;
|
||
|
}
|
||
|
|
||
11 years ago
|
function mod_page($title, $template, $args, $subtitle = false) {
|
||
11 years ago
|
global $config, $mod;
|
||
|
|
||
|
echo Element('page.html', array(
|
||
|
'config' => $config,
|
||
|
'mod' => $mod,
|
||
11 years ago
|
'hide_dashboard_link' => $template == 'mod/dashboard.html',
|
||
11 years ago
|
'title' => $title,
|
||
11 years ago
|
'subtitle' => $subtitle,
|
||
11 years ago
|
'body' => Element($template,
|
||
|
array_merge(
|
||
|
array('config' => $config, 'mod' => $mod),
|
||
|
$args
|
||
|
)
|
||
|
)
|
||
|
)
|
||
|
);
|
||
|
}
|
||
|
|
||
10 years ago
|
function mod_login($redirect = false) {
|
||
11 years ago
|
global $config;
|
||
|
|
||
11 years ago
|
$args = array();
|
||
|
|
||
|
if (isset($_POST['login'])) {
|
||
|
// Check if inputs are set and not empty
|
||
|
if (!isset($_POST['username'], $_POST['password']) || $_POST['username'] == '' || $_POST['password'] == '') {
|
||
|
$args['error'] = $config['error']['invalid'];
|
||
|
} elseif (!login($_POST['username'], $_POST['password'])) {
|
||
|
if ($config['syslog'])
|
||
|
_syslog(LOG_WARNING, 'Unauthorized login attempt!');
|
||
|
|
||
|
$args['error'] = $config['error']['invalid'];
|
||
|
} else {
|
||
11 years ago
|
modLog('Logged in');
|
||
11 years ago
|
|
||
|
// Login successful
|
||
|
// Set cookies
|
||
|
setCookies();
|
||
|
|
||
10 years ago
|
if ($redirect)
|
||
|
header('Location: ?' . $redirect, true, $config['redirect_http']);
|
||
|
else
|
||
|
header('Location: ?/', true, $config['redirect_http']);
|
||
11 years ago
|
}
|
||
|
}
|
||
|
|
||
|
if (isset($_POST['username']))
|
||
|
$args['username'] = $_POST['username'];
|
||
|
|
||
11 years ago
|
mod_page(_('Login'), 'mod/login.html', $args);
|
||
11 years ago
|
}
|
||
|
|
||
|
function mod_confirm($request) {
|
||
11 years ago
|
mod_page(_('Confirm action'), 'mod/confirm.html', array('request' => $request, 'token' => make_secure_link_token($request)));
|
||
11 years ago
|
}
|
||
|
|
||
11 years ago
|
function mod_logout() {
|
||
10 years ago
|
global $config;
|
||
11 years ago
|
destroyCookies();
|
||
|
|
||
|
header('Location: ?/', true, $config['redirect_http']);
|
||
|
}
|
||
|
|
||
11 years ago
|
function mod_dashboard() {
|
||
11 years ago
|
global $config, $mod;
|
||
11 years ago
|
|
||
11 years ago
|
$args = array();
|
||
|
|
||
|
$args['boards'] = listBoards();
|
||
|
|
||
11 years ago
|
if (hasPermission($config['mod']['noticeboard'])) {
|
||
|
if (!$config['cache']['enabled'] || !$args['noticeboard'] = cache::get('noticeboard_preview')) {
|
||
10 years ago
|
$query = prepare("SELECT ``noticeboard``.*, `username` FROM ``noticeboard`` LEFT JOIN ``mods`` ON ``mods``.`id` = `mod` ORDER BY `id` DESC LIMIT :limit");
|
||
11 years ago
|
$query->bindValue(':limit', $config['mod']['noticeboard_dashboard'], PDO::PARAM_INT);
|
||
|
$query->execute() or error(db_error($query));
|
||
|
$args['noticeboard'] = $query->fetchAll(PDO::FETCH_ASSOC);
|
||
|
|
||
11 years ago
|
if ($config['cache']['enabled'])
|
||
11 years ago
|
cache::set('noticeboard_preview', $args['noticeboard']);
|
||
|
}
|
||
|
}
|
||
|
|
||
10 years ago
|
if (!$config['cache']['enabled'] || ($args['unread_pms'] = cache::get('pm_unreadcount_' . $mod['id'])) == false) {
|
||
10 years ago
|
$query = prepare('SELECT COUNT(*) FROM ``pms`` WHERE `to` = :id AND `unread` = 1');
|
||
11 years ago
|
$query->bindValue(':id', $mod['id']);
|
||
|
$query->execute() or error(db_error($query));
|
||
10 years ago
|
$args['unread_pms'] = $query->fetchColumn();
|
||
11 years ago
|
|
||
|
if ($config['cache']['enabled'])
|
||
|
cache::set('pm_unreadcount_' . $mod['id'], $args['unread_pms']);
|
||
|
}
|
||
11 years ago
|
|
||
10 years ago
|
$query = query('SELECT COUNT(*) FROM ``reports``') or error(db_error($query));
|
||
10 years ago
|
$args['reports'] = $query->fetchColumn();
|
||
11 years ago
|
|
||
11 years ago
|
if ($mod['type'] >= ADMIN && $config['check_updates']) {
|
||
|
if (!$config['version'])
|
||
|
error(_('Could not find current version! (Check .installed)'));
|
||
|
|
||
|
if (isset($_COOKIE['update'])) {
|
||
|
$latest = unserialize($_COOKIE['update']);
|
||
|
} else {
|
||
|
$ctx = stream_context_create(array('http' => array('timeout' => 5)));
|
||
|
if ($code = @file_get_contents('http://tinyboard.org/version.txt', 0, $ctx)) {
|
||
10 years ago
|
$ver = strtok($code, "\n");
|
||
|
|
||
|
if (preg_match('@^// v(\d+)\.(\d+)\.(\d+)\s*[email protected]', $ver, $matches)) {
|
||
|
$latest = array(
|
||
|
'massive' => $matches[1],
|
||
|
'major' => $matches[2],
|
||
|
'minor' => $matches[3]
|
||
11 years ago
|
);
|
||
10 years ago
|
if (preg_match('/v(\d+)\.(\d)\.(\d+)(-dev.+)?$/', $config['version'], $matches)) {
|
||
|
$current = array(
|
||
|
'massive' => (int) $matches[1],
|
||
|
'major' => (int) $matches[2],
|
||
|
'minor' => (int) $matches[3]
|
||
|
);
|
||
|
if (isset($m[4])) {
|
||
|
// Development versions are always ahead in the versioning numbers
|
||
|
$current['minor'] --;
|
||
|
}
|
||
|
// Check if it's newer
|
||
|
if (!( $latest['massive'] > $current['massive'] ||
|
||
|
$latest['major'] > $current['major'] ||
|
||
|
($latest['massive'] == $current['massive'] &&
|
||
|
$latest['major'] == $current['major'] &&
|
||
|
$latest['minor'] > $current['minor']
|
||
|
)))
|
||
|
$latest = false;
|
||
|
} else {
|
||
11 years ago
|
$latest = false;
|
||
10 years ago
|
}
|
||
11 years ago
|
} else {
|
||
10 years ago
|
// Couldn't get latest version
|
||
11 years ago
|
$latest = false;
|
||
|
}
|
||
|
} else {
|
||
|
// Couldn't get latest version
|
||
|
$latest = false;
|
||
|
}
|
||
|
|
||
|
setcookie('update', serialize($latest), time() + $config['check_updates_time'], $config['cookies']['jail'] ? $config['cookies']['path'] : '/', null, false, true);
|
||
|
}
|
||
|
|
||
|
if ($latest)
|
||
|
$args['newer_release'] = $latest;
|
||
|
}
|
||
|
|
||
11 years ago
|
mod_page(_('Dashboard'), 'mod/dashboard.html', $args);
|
||
11 years ago
|
}
|
||
|
|
||
10 years ago
|
function mod_search_redirect() {
|
||
|
global $config;
|
||
|
|
||
|
if (!hasPermission($config['mod']['search']))
|
||
|
error($config['error']['noaccess']);
|
||
|
|
||
10 years ago
|
if (isset($_POST['query'], $_POST['type']) && in_array($_POST['type'], array('posts', 'IP_notes', 'bans', 'log'))) {
|
||
10 years ago
|
$query = $_POST['query'];
|
||
|
$query = urlencode($query);
|
||
|
$query = str_replace('_', '%5F', $query);
|
||
|
$query = str_replace('+', '_', $query);
|
||
|
|
||
10 years ago
|
if ($query === '') {
|
||
|
header('Location: ?/', true, $config['redirect_http']);
|
||
|
return;
|
||
|
}
|
||
|
|
||
10 years ago
|
header('Location: ?/search/' . $_POST['type'] . '/' . $query, true, $config['redirect_http']);
|
||
|
} else {
|
||
|
header('Location: ?/', true, $config['redirect_http']);
|
||
|
}
|
||
|
}
|
||
|
|
||
10 years ago
|
function mod_search($type, $search_query_escaped, $page_no = 1) {
|
||
10 years ago
|
global $pdo, $config;
|
||
|
|
||
|
if (!hasPermission($config['mod']['search']))
|
||
|
error($config['error']['noaccess']);
|
||
|
|
||
|
// Unescape query
|
||
10 years ago
|
$query = str_replace('_', ' ', $search_query_escaped);
|
||
10 years ago
|
$query = urldecode($query);
|
||
|
$search_query = $query;
|
||
|
|
||
|
// Form a series of LIKE clauses for the query.
|
||
|
// This gets a little complicated.
|
||
|
|
||
|
// Escape "escape" character
|
||
|
$query = str_replace('!', '!!', $query);
|
||
|
|
||
|
// Escape SQL wildcard
|
||
|
$query = str_replace('%', '!%', $query);
|
||
|
|
||
|
// Use asterisk as wildcard instead
|
||
|
$query = str_replace('*', '%', $query);
|
||
|
|
||
10 years ago
|
$query = str_replace('`', '!`', $query);
|
||
|
|
||
10 years ago
|
// Array of phrases to match
|
||
|
$match = array();
|
||
|
|
||
|
// Exact phrases ("like this")
|
||
10 years ago
|
if (preg_match_all('/"(.+?)"/', $query, $exact_phrases)) {
|
||
|
$exact_phrases = $exact_phrases[1];
|
||
|
foreach ($exact_phrases as $phrase) {
|
||
10 years ago
|
$query = str_replace("\"{$phrase}\"", '', $query);
|
||
|
$match[] = $pdo->quote($phrase);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Non-exact phrases (ie. plain keywords)
|
||
|
$keywords = explode(' ', $query);
|
||
|
foreach ($keywords as $word) {
|
||
|
if (empty($word))
|
||
|
continue;
|
||
|
$match[] = $pdo->quote($word);
|
||
|
}
|
||
|
|
||
|
// Which `field` to search?
|
||
|
if ($type == 'posts')
|
||
10 years ago
|
$sql_field = array('body_nomarkup', 'filename', 'subject', 'filehash', 'ip', 'name', 'trip');
|
||
10 years ago
|
if ($type == 'IP_notes')
|
||
|
$sql_field = 'body';
|
||
|
if ($type == 'bans')
|
||
|
$sql_field = 'reason';
|
||
10 years ago
|
if ($type == 'log')
|
||
|
$sql_field = 'text';
|
||
10 years ago
|
|
||
|
// Build the "LIKE 'this' AND LIKE 'that'" etc. part of the SQL query
|
||
|
$sql_like = '';
|
||
|
foreach ($match as $phrase) {
|
||
|
if (!empty($sql_like))
|
||
|
$sql_like .= ' AND ';
|
||
|
$phrase = preg_replace('/^\'(.+)\'$/', '\'%$1%\'', $phrase);
|
||
10 years ago
|
if (is_array($sql_field)) {
|
||
|
foreach ($sql_field as $field) {
|
||
|
$sql_like .= '`' . $field . '` LIKE ' . $phrase . ' ESCAPE \'!\' OR';
|
||
|
}
|
||
|
$sql_like = preg_replace('/ OR$/', '', $sql_like);
|
||
|
} else {
|
||
|
$sql_like .= '`' . $sql_field . '` LIKE ' . $phrase . ' ESCAPE \'!\'';
|
||
|
}
|
||
10 years ago
|
}
|
||
|
|
||
10 years ago
|
|
||
|
// Compile SQL query
|
||
|
|
||
10 years ago
|
if ($type == 'posts') {
|
||
10 years ago
|
$query = '';
|
||
|
|
||
|
$boards = listBoards();
|
||
|
if (empty($boards))
|
||
|
error(_('There are no boards to search!'));
|
||
|
|
||
|
foreach ($boards as $board) {
|
||
|
openBoard($board['uri']);
|
||
|
if (!hasPermission($config['mod']['search_posts'], $board['uri']))
|
||
|
continue;
|
||
|
|
||
|
if (!empty($query))
|
||
|
$query .= ' UNION ALL ';
|
||
10 years ago
|
$query .= sprintf("SELECT *, '%s' AS `board` FROM ``posts_%s`` WHERE %s", $board['uri'], $board['uri'], $sql_like);
|
||
10 years ago
|
}
|
||
|
|
||
|
// You weren't allowed to search any boards
|
||
|
if (empty($query))
|
||
|
error($config['error']['noaccess']);
|
||
|
|
||
|
$query .= ' ORDER BY `sticky` DESC, `id` DESC';
|
||
10 years ago
|
}
|
||
|
|
||
|
if ($type == 'IP_notes') {
|
||
10 years ago
|
$query = 'SELECT * FROM ``ip_notes`` LEFT JOIN ``mods`` ON `mod` = ``mods``.`id` WHERE ' . $sql_like . ' ORDER BY `time` DESC';
|
||
10 years ago
|
$sql_table = 'ip_notes';
|
||
10 years ago
|
if (!hasPermission($config['mod']['view_notes']) || !hasPermission($config['mod']['show_ip']))
|
||
|
error($config['error']['noaccess']);
|
||
10 years ago
|
}
|
||
|
|
||
|
if ($type == 'bans') {
|
||
10 years ago
|
$query = 'SELECT ``bans``.*, `username` FROM ``bans`` LEFT JOIN ``mods`` ON `mod` = ``mods``.`id` WHERE ' . $sql_like . ' ORDER BY (`expires` IS NOT NULL AND `expires` < UNIX_TIMESTAMP()), `set` DESC';
|
||
10 years ago
|
$sql_table = 'bans';
|
||
10 years ago
|
if (!hasPermission($config['mod']['view_banlist']))
|
||
|
error($config['error']['noaccess']);
|
||
10 years ago
|
}
|
||
|
|
||
|
if ($type == 'log') {
|
||
10 years ago
|
$query = 'SELECT `username`, `mod`, `ip`, `board`, `time`, `text` FROM ``modlogs`` LEFT JOIN ``mods`` ON `mod` = ``mods``.`id` WHERE ' . $sql_like . ' ORDER BY `time` DESC';
|
||
10 years ago
|
$sql_table = 'modlogs';
|
||
10 years ago
|
if (!hasPermission($config['mod']['modlog']))
|
||
|
error($config['error']['noaccess']);
|
||
10 years ago
|
}
|
||
10 years ago
|
|
||
10 years ago
|
// Execute SQL query (with pages)
|
||
|
$q = query($query . ' LIMIT ' . (($page_no - 1) * $config['mod']['search_page']) . ', ' . $config['mod']['search_page']) or error(db_error());
|
||
|
$results = $q->fetchAll(PDO::FETCH_ASSOC);
|
||
10 years ago
|
|
||
10 years ago
|
// Get total result count
|
||
10 years ago
|
if ($type == 'posts') {
|
||
|
$q = query("SELECT COUNT(*) FROM ($query) AS `tmp_table`") or error(db_error());
|
||
|
$result_count = $q->fetchColumn();
|
||
|
} else {
|
||
|
$q = query('SELECT COUNT(*) FROM `' . $sql_table . '` WHERE ' . $sql_like) or error(db_error());
|
||
|
$result_count = $q->fetchColumn();
|
||
|
}
|
||
10 years ago
|
|
||
|
if ($type == 'bans') {
|
||
10 years ago
|
foreach ($results as &$ban) {
|
||
|
if (filter_var($ban['ip'], FILTER_VALIDATE_IP) !== false)
|
||
|
$ban['real_ip'] = true;
|
||
|
}
|
||
|
}
|
||
|
|
||
10 years ago
|
if ($type == 'posts') {
|
||
|
foreach ($results as &$post) {
|
||
|
$post['snippet'] = pm_snippet($post['body']);
|
||
|
}
|
||
|
}
|
||
|
|
||
10 years ago
|
// $results now contains the search results
|
||
10 years ago
|
|
||
10 years ago
|
mod_page(_('Search results'), 'mod/search_results.html', array(
|
||
|
'search_type' => $type,
|
||
|
'search_query' => $search_query,
|
||
10 years ago
|
'search_query_escaped' => $search_query_escaped,
|
||
|
'result_count' => $result_count,
|
||
10 years ago
|
'results' => $results
|
||
|
));
|
||
|
}
|
||
|
|
||
11 years ago
|
function mod_edit_board($boardName) {
|
||
|
global $board, $config;
|
||
|
|
||
|
if (!openBoard($boardName))
|
||
|
error($config['error']['noboard']);
|
||
|
|
||
|
if (!hasPermission($config['mod']['manageboards'], $board['uri']))
|
||
|
error($config['error']['noaccess']);
|
||
|
|
||
|
if (isset($_POST['title'], $_POST['subtitle'])) {
|
||
|
if (isset($_POST['delete'])) {
|
||
|
if (!hasPermission($config['mod']['manageboards'], $board['uri']))
|
||
|
error($config['error']['deleteboard']);
|
||
|
|
||
10 years ago
|
$query = prepare('DELETE FROM ``boards`` WHERE `uri` = :uri');
|
||
11 years ago
|
$query->bindValue(':uri', $board['uri']);
|
||
|
$query->execute() or error(db_error($query));
|
||
|
|
||
10 years ago
|
if ($config['cache']['enabled']) {
|
||
|
cache::delete('board_' . $board['uri']);
|
||
|
cache::delete('all_boards');
|
||
|
}
|
||
11 years ago
|
|
||
10 years ago
|
modLog('Deleted board: ' . sprintf($config['board_abbreviation'], $board['uri']), false);
|
||
11 years ago
|
|
||
|
// Delete posting table
|
||
10 years ago
|
$query = query(sprintf('DROP TABLE IF EXISTS ``posts_%s``', $board['uri'])) or error(db_error());
|
||
11 years ago
|
|
||
|
// Clear reports
|
||
10 years ago
|
$query = prepare('DELETE FROM ``reports`` WHERE `board` = :id');
|
||
11 years ago
|
$query->bindValue(':id', $board['uri'], PDO::PARAM_INT);
|
||
|
$query->execute() or error(db_error($query));
|
||
|
|
||
|
// Delete from table
|
||
10 years ago
|
$query = prepare('DELETE FROM ``boards`` WHERE `uri` = :uri');
|
||
11 years ago
|
$query->bindValue(':uri', $board['uri'], PDO::PARAM_INT);
|
||
|
$query->execute() or error(db_error($query));
|
||
|
|
||
10 years ago
|
$query = prepare("SELECT `board`, `post` FROM ``cites`` WHERE `target_board` = :board");
|
||
11 years ago
|
$query->bindValue(':board', $board['uri']);
|
||
|
$query->execute() or error(db_error($query));
|
||
|
while ($cite = $query->fetch(PDO::FETCH_ASSOC)) {
|
||
|
if ($board['uri'] != $cite['board']) {
|
||
|
if (!isset($tmp_board))
|
||
|
$tmp_board = $board;
|
||
|
openBoard($cite['board']);
|
||
|
rebuildPost($cite['post']);
|
||
|
}
|
||
|
}
|
||
|
|
||
10 years ago
|
$query = prepare('DELETE FROM ``cites`` WHERE `board` = :board OR `target_board` = :board');
|
||
11 years ago
|
$query->bindValue(':board', $board['uri']);
|
||
|
$query->execute() or error(db_error($query));
|
||
|
|
||
10 years ago
|
$query = prepare('DELETE FROM ``antispam`` WHERE `board` = :board');
|
||
11 years ago
|
$query->bindValue(':board', $board['uri']);
|
||
|
$query->execute() or error(db_error($query));
|
||
10 years ago
|
|
||
|
// Remove board from users/permissions table
|
||
10 years ago
|
$query = query('SELECT `id`,`boards` FROM ``mods``') or error(db_error());
|
||
10 years ago
|
while ($user = $query->fetch(PDO::FETCH_ASSOC)) {
|
||
|
$user_boards = explode(',', $user['boards']);
|
||
|
if (in_array($board['uri'], $user_boards)) {
|
||
|
unset($user_boards[array_search($board['uri'], $user_boards)]);
|
||
10 years ago
|
$_query = prepare('UPDATE ``mods`` SET `boards` = :boards WHERE `id` = :id');
|
||
10 years ago
|
$_query->bindValue(':boards', implode(',', $user_boards));
|
||
|
$_query->bindValue(':id', $user['id']);
|
||
|
$_query->execute() or error(db_error($_query));
|
||
|
}
|
||
|
}
|
||
10 years ago
|
|
||
|
// Delete entire board directory
|
||
|
rrmdir($board['uri'] . '/');
|
||
11 years ago
|
} else {
|
||
10 years ago
|
$query = prepare('UPDATE ``boards`` SET `title` = :title, `subtitle` = :subtitle WHERE `uri` = :uri');
|
||
11 years ago
|
$query->bindValue(':uri', $board['uri']);
|
||
|
$query->bindValue(':title', $_POST['title']);
|
||
|
$query->bindValue(':subtitle', $_POST['subtitle']);
|
||
|
$query->execute() or error(db_error($query));
|
||
10 years ago
|
|
||
|
modLog('Edited board information for ' . sprintf($config['board_abbreviation'], $board['uri']), false);
|
||
11 years ago
|
}
|
||
|
|
||
11 years ago
|
if ($config['cache']['enabled']) {
|
||
|
cache::delete('board_' . $board['uri']);
|
||
|
cache::delete('all_boards');
|
||
|
}
|
||
|
|
||
11 years ago
|
rebuildThemes('boards');
|
||
|
|
||
|
header('Location: ?/', true, $config['redirect_http']);
|
||
|
} else {
|
||
11 years ago
|
mod_page(sprintf('%s: ' . $config['board_abbreviation'], _('Edit board'), $board['uri']), 'mod/board.html', array('board' => $board));
|
||
11 years ago
|
}
|
||
|
}
|
||
|
|
||
|
function mod_new_board() {
|
||
|
global $config, $board;
|
||
|
|
||
|
if (!hasPermission($config['mod']['newboard']))
|
||
|
error($config['error']['noaccess']);
|
||
|
|
||
|
if (isset($_POST['uri'], $_POST['title'], $_POST['subtitle'])) {
|
||
|
if ($_POST['uri'] == '')
|
||
|
error(sprintf($config['error']['required'], 'URI'));
|
||
|
|
||
|
if ($_POST['title'] == '')
|
||
|
error(sprintf($config['error']['required'], 'title'));
|
||
|
|
||
10 years ago
|
if (!preg_match('/^' . $config['board_regex'] . '$/u', $_POST['uri']))
|
||
11 years ago
|
error(sprintf($config['error']['invalidfield'], 'URI'));
|
||
|
|
||
10 years ago
|
$bytes = 0;
|
||
|
$chars = preg_split('//u', $_POST['uri'], -1, PREG_SPLIT_NO_EMPTY);
|
||
|
foreach ($chars as $char) {
|
||
|
$o = 0;
|
||
|
$ord = ordutf8($char, $o);
|
||
|
if ($ord > 0x0080)
|
||
|
$bytes += 5; // @01ff
|
||
|
else
|
||
|
$bytes ++;
|
||
|
}
|
||
|
$bytes + strlen('posts_.frm');
|
||
|
|
||
|
if ($bytes > 255) {
|
||
|
error('Your filesystem cannot handle a board URI of that length (' . $bytes . '/255 bytes)');
|
||
|
exit;
|
||
|
}
|
||
|
|
||
11 years ago
|
if (openBoard($_POST['uri'])) {
|
||
|
error(sprintf($config['error']['boardexists'], $board['url']));
|
||
|
}
|
||
|
|
||
10 years ago
|
$query = prepare('INSERT INTO ``boards`` VALUES (:uri, :title, :subtitle)');
|
||
11 years ago
|
$query->bindValue(':uri', $_POST['uri']);
|
||
|
$query->bindValue(':title', $_POST['title']);
|
||
|
$query->bindValue(':subtitle', $_POST['subtitle']);
|
||
|
$query->execute() or error(db_error($query));
|
||
|
|
||
|
modLog('Created a new board: ' . sprintf($config['board_abbreviation'], $_POST['uri']));
|
||
|
|
||
|
if (!openBoard($_POST['uri']))
|
||
|
error(_("Couldn't open board after creation."));
|
||
|
|
||
10 years ago
|
$query = Element('posts.sql', array('board' => $board['uri']));
|
||
|
|
||
|
if (mysql_version() < 50503)
|
||
|
$query = preg_replace('/(CHARSET=|CHARACTER SET )utf8mb4/', '$1utf8', $query);
|
||
|
|
||
|
query($query) or error(db_error());
|
||
11 years ago
|
|
||
|
if ($config['cache']['enabled'])
|
||
11 years ago
|
cache::delete('all_boards');
|
||
11 years ago
|
|
||
|
// Build the board
|
||
|
buildIndex();
|
||
|
|
||
|
rebuildThemes('boards');
|
||
|
|
||
|
header('Location: ?/' . $board['uri'] . '/' . $config['file_index'], true, $config['redirect_http']);
|
||
|
}
|
||
|
|
||
11 years ago
|
mod_page(_('New board'), 'mod/board.html', array('new' => true));
|
||
11 years ago
|
}
|
||
|
|
||
11 years ago
|
function mod_noticeboard($page_no = 1) {
|
||
|
global $config, $pdo, $mod;
|
||
|
|
||
11 years ago
|
if ($page_no < 1)
|
||
|
error($config['error']['404']);
|
||
|
|
||
11 years ago
|
if (!hasPermission($config['mod']['noticeboard']))
|
||
|
error($config['error']['noaccess']);
|
||
|
|
||
|
if (isset($_POST['subject'], $_POST['body'])) {
|
||
|
if (!hasPermission($config['mod']['noticeboard_post']))
|
||
|
error($config['error']['noaccess']);
|
||
|
|
||
10 years ago
|
$_POST['body'] = escape_markup_modifiers($_POST['body']);
|
||
11 years ago
|
markup($_POST['body']);
|
||
|
|
||
10 years ago
|
$query = prepare('INSERT INTO ``noticeboard`` VALUES (NULL, :mod, :time, :subject, :body)');
|
||
11 years ago
|
$query->bindValue(':mod', $mod['id']);
|
||
|
$query->bindvalue(':time', time());
|
||
|
$query->bindValue(':subject', $_POST['subject']);
|
||
|
$query->bindValue(':body', $_POST['body']);
|
||
|
$query->execute() or error(db_error($query));
|
||
|
|
||
11 years ago
|
if ($config['cache']['enabled'])
|
||
11 years ago
|
cache::delete('noticeboard_preview');
|
||
|
|
||
11 years ago
|
modLog('Posted a noticeboard entry');
|
||
|
|
||
11 years ago
|
header('Location: ?/noticeboard#' . $pdo->lastInsertId(), true, $config['redirect_http']);
|
||
|
}
|
||
|
|
||
10 years ago
|
$query = prepare("SELECT ``noticeboard``.*, `username` FROM ``noticeboard`` LEFT JOIN ``mods`` ON ``mods``.`id` = `mod` ORDER BY `id` DESC LIMIT :offset, :limit");
|
||
11 years ago
|
$query->bindValue(':limit', $config['mod']['noticeboard_page'], PDO::PARAM_INT);
|
||
|
$query->bindValue(':offset', ($page_no - 1) * $config['mod']['noticeboard_page'], PDO::PARAM_INT);
|
||
|
$query->execute() or error(db_error($query));
|
||
|
$noticeboard = $query->fetchAll(PDO::FETCH_ASSOC);
|
||
|
|
||
11 years ago
|
if (empty($noticeboard) && $page_no > 1)
|
||
11 years ago
|
error($config['error']['404']);
|
||
|
|
||
10 years ago
|
$query = prepare("SELECT COUNT(*) FROM ``noticeboard``");
|
||
11 years ago
|
$query->execute() or error(db_error($query));
|
||
10 years ago
|
$count = $query->fetchColumn();
|
||
11 years ago
|
|
||
11 years ago
|
mod_page(_('Noticeboard'), 'mod/noticeboard.html', array('noticeboard' => $noticeboard, 'count' => $count));
|
||
11 years ago
|
}
|
||
|
|
||
11 years ago
|
function mod_noticeboard_delete($id) {
|
||
|
global $config;
|
||
|
|
||
|
if (!hasPermission($config['mod']['noticeboard_delete']))
|
||
|
error($config['error']['noaccess']);
|
||
|
|
||
10 years ago
|
$query = prepare('DELETE FROM ``noticeboard`` WHERE `id` = :id');
|
||
11 years ago
|
$query->bindValue(':id', $id);
|
||
|
$query->execute() or error(db_error($query));
|
||
|
|
||
|
modLog('Deleted a noticeboard entry');
|
||
|
|
||
11 years ago
|
if ($config['cache']['enabled'])
|
||
|
cache::delete('noticeboard_preview');
|
||
|
|
||
11 years ago
|
header('Location: ?/noticeboard', true, $config['redirect_http']);
|
||
|
}
|
||
|
|
||
|
function mod_news($page_no = 1) {
|
||
|
global $config, $pdo, $mod;
|
||
|
|
||
|
if ($page_no < 1)
|
||
|
error($config['error']['404']);
|
||
|
|
||
|
if (isset($_POST['subject'], $_POST['body'])) {
|
||
|
if (!hasPermission($config['mod']['news']))
|
||
|
error($config['error']['noaccess']);
|
||
|
|
||
10 years ago
|
$_POST['body'] = escape_markup_modifiers($_POST['body']);
|
||
11 years ago
|
markup($_POST['body']);
|
||
|
|
||
10 years ago
|
$query = prepare('INSERT INTO ``news`` VALUES (NULL, :name, :time, :subject, :body)');
|
||
11 years ago
|
$query->bindValue(':name', isset($_POST['name']) && hasPermission($config['mod']['news_custom']) ? $_POST['name'] : $mod['username']);
|
||
|
$query->bindvalue(':time', time());
|
||
|
$query->bindValue(':subject', $_POST['subject']);
|
||
|
$query->bindValue(':body', $_POST['body']);
|
||
|
$query->execute() or error(db_error($query));
|
||
|
|
||
|
modLog('Posted a news entry');
|
||
|
|
||
|
rebuildThemes('news');
|
||
|
|
||
|
header('Location: ?/news#' . $pdo->lastInsertId(), true, $config['redirect_http']);
|
||
|
}
|
||
|
|
||
10 years ago
|
$query = prepare("SELECT * FROM ``news`` ORDER BY `id` DESC LIMIT :offset, :limit");
|
||
11 years ago
|
$query->bindValue(':limit', $config['mod']['news_page'], PDO::PARAM_INT);
|
||
|
$query->bindValue(':offset', ($page_no - 1) * $config['mod']['news_page'], PDO::PARAM_INT);
|
||
|
$query->execute() or error(db_error($query));
|
||
|
$news = $query->fetchAll(PDO::FETCH_ASSOC);
|
||
|
|
||
|
if (empty($news) && $page_no > 1)
|
||
|
error($config['error']['404']);
|
||
|
|
||
10 years ago
|
$query = prepare("SELECT COUNT(*) FROM ``news``");
|
||
11 years ago
|
$query->execute() or error(db_error($query));
|
||
10 years ago
|
$count = $query->fetchColumn();
|
||
11 years ago
|
|
||
11 years ago
|
mod_page(_('News'), 'mod/news.html', array('news' => $news, 'count' => $count));
|
||
11 years ago
|
}
|
||
|
|
||
|
function mod_news_delete($id) {
|
||
|
global $config;
|
||
|
|
||
|
if (!hasPermission($config['mod']['news_delete']))
|
||
|
error($config['error']['noaccess']);
|
||
|
|
||
10 years ago
|
$query = prepare('DELETE FROM ``news`` WHERE `id` = :id');
|
||
11 years ago
|
$query->bindValue(':id', $id);
|
||
|
$query->execute() or error(db_error($query));
|
||
|
|
||
|
modLog('Deleted a news entry');
|
||
|
|
||
|
header('Location: ?/news', true, $config['redirect_http']);
|
||
|
}
|
||
|
|
||
11 years ago
|
function mod_log($page_no = 1) {
|
||
|
global $config;
|
||
|
|
||
11 years ago
|
if ($page_no < 1)
|
||
|
error($config['error']['404']);
|
||
|
|
||
11 years ago
|
if (!hasPermission($config['mod']['modlog']))
|
||
|
error($config['error']['noaccess']);
|
||
|
|
||
10 years ago
|
$query = prepare("SELECT `username`, `mod`, `ip`, `board`, `time`, `text` FROM ``modlogs`` LEFT JOIN ``mods`` ON `mod` = ``mods``.`id` ORDER BY `time` DESC LIMIT :offset, :limit");
|
||
11 years ago
|
$query->bindValue(':limit', $config['mod']['modlog_page'], PDO::PARAM_INT);
|
||
|
$query->bindValue(':offset', ($page_no - 1) * $config['mod']['modlog_page'], PDO::PARAM_INT);
|
||
|
$query->execute() or error(db_error($query));
|
||
|
$logs = $query->fetchAll(PDO::FETCH_ASSOC);
|
||
|
|
||
11 years ago
|
if (empty($logs) && $page_no > 1)
|
||
11 years ago
|
error($config['error']['404']);
|
||
|
|
||
10 years ago
|
$query = prepare("SELECT COUNT(*) FROM ``modlogs``");
|
||
11 years ago
|
$query->execute() or error(db_error($query));
|
||
10 years ago
|
$count = $query->fetchColumn();
|
||
11 years ago
|
|
||
11 years ago
|
mod_page(_('Moderation log'), 'mod/log.html', array('logs' => $logs, 'count' => $count));
|
||
11 years ago
|
}
|
||
|
|
||
11 years ago
|
function mod_user_log($username, $page_no = 1) {
|
||
|
global $config;
|
||
|
|
||
|
if ($page_no < 1)
|
||
|
error($config['error']['404']);
|
||
|
|
||
|
if (!hasPermission($config['mod']['modlog']))
|
||
|
error($config['error']['noaccess']);
|
||
|
|
||
10 years ago
|
$query = prepare("SELECT `username`, `mod`, `ip`, `board`, `time`, `text` FROM ``modlogs`` LEFT JOIN ``mods`` ON `mod` = ``mods``.`id` WHERE `username` = :username ORDER BY `time` DESC LIMIT :offset, :limit");
|
||
11 years ago
|
$query->bindValue(':username', $username);
|
||
|
$query->bindValue(':limit', $config['mod']['modlog_page'], PDO::PARAM_INT);
|
||
|
$query->bindValue(':offset', ($page_no - 1) * $config['mod']['modlog_page'], PDO::PARAM_INT);
|
||
|
$query->execute() or error(db_error($query));
|
||
|
$logs = $query->fetchAll(PDO::FETCH_ASSOC);
|
||
|
|
||
|
if (empty($logs) && $page_no > 1)
|
||
|
error($config['error']['404']);
|
||
|
|
||
10 years ago
|
$query = prepare("SELECT COUNT(*) FROM ``modlogs`` LEFT JOIN ``mods`` ON `mod` = ``mods``.`id` WHERE `username` = :username");
|
||
11 years ago
|
$query->bindValue(':username', $username);
|
||
11 years ago
|
$query->execute() or error(db_error($query));
|
||
10 years ago
|
$count = $query->fetchColumn();
|
||
11 years ago
|
|
||
|
mod_page(_('Moderation log'), 'mod/log.html', array('logs' => $logs, 'count' => $count, 'username' => $username));
|
||
|
}
|
||
|
|
||
11 years ago
|
function mod_view_board($boardName, $page_no = 1) {
|
||
|
global $config, $mod;
|
||
|
|
||
|
if (!openBoard($boardName))
|
||
|
error($config['error']['noboard']);
|
||
|
|
||
|
if (!$page = index($page_no, $mod)) {
|
||
|
error($config['error']['404']);
|
||
|
}
|
||
|
|
||
|
$page['pages'] = getPages(true);
|
||
|
$page['pages'][$page_no-1]['selected'] = true;
|
||
|
$page['btn'] = getPageButtons($page['pages'], true);
|
||
|
$page['mod'] = true;
|
||
|
$page['config'] = $config;
|
||
11 years ago
|
|
||
11 years ago
|
echo Element('index.html', $page);
|
||
|
}
|
||
|
|
||
|
function mod_view_thread($boardName, $thread) {
|
||
|
global $config, $mod;
|
||
|
|
||
|
if (!openBoard($boardName))
|
||
|
error($config['error']['noboard']);
|
||
|
|
||
|
$page = buildThread($thread, true, $mod);
|
||
|
echo $page;
|
||
|
}
|
||
|
|
||
11 years ago
|
function mod_ip_remove_note($ip, $id) {
|
||
|
global $config, $mod;
|
||
|
|
||
11 years ago
|
if (!hasPermission($config['mod']['remove_notes']))
|
||
|
error($config['error']['noaccess']);
|
||
|
|
||
11 years ago
|
if (filter_var($ip, FILTER_VALIDATE_IP) === false)
|
||
|
error("Invalid IP address.");
|
||
|
|
||
10 years ago
|
$query = prepare('DELETE FROM ``ip_notes`` WHERE `ip` = :ip AND `id` = :id');
|
||
11 years ago
|
$query->bindValue(':ip', $ip);
|
||
|
$query->bindValue(':id', $id);
|
||
|
$query->execute() or error(db_error($query));
|
||
|
|
||
11 years ago
|
modLog("Removed a note for <a href=\"?/IP/{$ip}\">{$ip}</a>");
|
||
|
|
||
11 years ago
|
header('Location: ?/IP/' . $ip . '#notes', true, $config['redirect_http']);
|
||
11 years ago
|
}
|
||
|
|
||
11 years ago
|
function mod_page_ip($ip) {
|
||
|
global $config, $mod;
|
||
|
|
||
11 years ago
|
if (filter_var($ip, FILTER_VALIDATE_IP) === false)
|
||
11 years ago
|
error("Invalid IP address.");
|
||
|
|
||
11 years ago
|
if (isset($_POST['ban_id'], $_POST['unban'])) {
|
||
11 years ago
|
if (!hasPermission($config['mod']['unban']))
|
||
|
error($config['error']['noaccess']);
|
||
|
|
||
11 years ago
|
require_once 'inc/mod/ban.php';
|
||
|
|
||
|
unban($_POST['ban_id']);
|
||
11 years ago
|
|
||
|
header('Location: ?/IP/' . $ip . '#bans', true, $config['redirect_http']);
|
||
11 years ago
|
return;
|
||
|
}
|
||
|
|
||
11 years ago
|
if (isset($_POST['note'])) {
|
||
11 years ago
|
if (!hasPermission($config['mod']['create_notes']))
|
||
|
error($config['error']['noaccess']);
|
||
11 years ago
|
|
||
10 years ago
|
$_POST['note'] = escape_markup_modifiers($_POST['note']);
|
||
11 years ago
|
markup($_POST['note']);
|
||
10 years ago
|
$query = prepare('INSERT INTO ``ip_notes`` VALUES (NULL, :ip, :mod, :time, :body)');
|
||
11 years ago
|
$query->bindValue(':ip', $ip);
|
||
|
$query->bindValue(':mod', $mod['id']);
|
||
|
$query->bindValue(':time', time());
|
||
|
$query->bindValue(':body', $_POST['note']);
|
||
|
$query->execute() or error(db_error($query));
|
||
|
|
||
11 years ago
|
modLog("Added a note for <a href=\"?/IP/{$ip}\">{$ip}</a>");
|
||
|
|
||
11 years ago
|
header('Location: ?/IP/' . $ip . '#notes', true, $config['redirect_http']);
|
||
11 years ago
|
return;
|
||
|
}
|
||
|
|
||
11 years ago
|
$args = array();
|
||
|
$args['ip'] = $ip;
|
||
|
$args['posts'] = array();
|
||
|
|
||
11 years ago
|
if ($config['mod']['dns_lookup'])
|
||
|
$args['hostname'] = rDNS($ip);
|
||
|
|
||
11 years ago
|
$boards = listBoards();
|
||
|
foreach ($boards as $board) {
|
||
11 years ago
|
openBoard($board['uri']);
|
||
10 years ago
|
if (!hasPermission($config['mod']['show_ip'], $board['uri']))
|
||
|
continue;
|
||
10 years ago
|
$query = prepare(sprintf('SELECT * FROM ``posts_%s`` WHERE `ip` = :ip ORDER BY `sticky` DESC, `id` DESC LIMIT :limit', $board['uri']));
|
||
11 years ago
|
$query->bindValue(':ip', $ip);
|
||
|
$query->bindValue(':limit', $config['mod']['ip_recentposts'], PDO::PARAM_INT);
|
||
|
$query->execute() or error(db_error($query));
|
||
|
|
||
11 years ago
|
while ($post = $query->fetch(PDO::FETCH_ASSOC)) {
|
||
11 years ago
|
if (!$post['thread']) {
|
||
10 years ago
|
$po = new Thread($post, '?/', $mod, false);
|
||
11 years ago
|
} else {
|
||
10 years ago
|
$po = new Post($post, '?/', $mod);
|
||
11 years ago
|
}
|
||
|
|
||
|
if (!isset($args['posts'][$board['uri']]))
|
||
11 years ago
|
$args['posts'][$board['uri']] = array('board' => $board, 'posts' => array());
|
||
|
$args['posts'][$board['uri']]['posts'][] = $po->build(true);
|
||
11 years ago
|
}
|
||
|
}
|
||
|
|
||
11 years ago
|
$args['boards'] = $boards;
|
||
11 years ago
|
$args['token'] = make_secure_link_token('ban');
|
||
11 years ago
|
|
||
11 years ago
|
if (hasPermission($config['mod']['view_ban'])) {
|
||
10 years ago
|
$query = prepare("SELECT ``bans``.*, `username` FROM ``bans`` LEFT JOIN ``mods`` ON `mod` = ``mods``.`id` WHERE `ip` = :ip ORDER BY `set` DESC");
|
||
11 years ago
|
$query->bindValue(':ip', $ip);
|
||
|
$query->execute() or error(db_error($query));
|
||
|
$args['bans'] = $query->fetchAll(PDO::FETCH_ASSOC);
|
||
|
}
|
||
|
|
||
|
if (hasPermission($config['mod']['view_notes'])) {
|
||
10 years ago
|
$query = prepare("SELECT ``ip_notes``.*, `username` FROM ``ip_notes`` LEFT JOIN ``mods`` ON `mod` = ``mods``.`id` WHERE `ip` = :ip ORDER BY `time` DESC");
|
||
11 years ago
|
$query->bindValue(':ip', $ip);
|
||
|
$query->execute() or error(db_error($query));
|
||
|
$args['notes'] = $query->fetchAll(PDO::FETCH_ASSOC);
|
||
|
}
|
||
11 years ago
|
|
||
10 years ago
|
if (hasPermission($config['mod']['modlog_ip'])) {
|
||
10 years ago
|
$query = prepare("SELECT `username`, `mod`, `ip`, `board`, `time`, `text` FROM ``modlogs`` LEFT JOIN ``mods`` ON `mod` = ``mods``.`id` WHERE `text` LIKE :search ORDER BY `time` DESC LIMIT 50");
|
||
10 years ago
|
$query->bindValue(':search', '%' . $ip . '%');
|
||
|
$query->execute() or error(db_error($query));
|
||
|
$args['logs'] = $query->fetchAll(PDO::FETCH_ASSOC);
|
||
|
} else {
|
||
|
$args['logs'] = array();
|
||
|
}
|
||
|
|
||
11 years ago
|
mod_page(sprintf('%s: %s', _('IP'), $ip), 'mod/view_ip.html', $args, $args['hostname']);
|
||
11 years ago
|
}
|
||
|
|
||
11 years ago
|
function mod_ban() {
|
||
11 years ago
|
global $config;
|
||
|
|
||
|
if (!hasPermission($config['mod']['ban']))
|
||
|
error($config['error']['noaccess']);
|
||
|
|
||
11 years ago
|
if (!isset($_POST['ip'], $_POST['reason'], $_POST['length'], $_POST['board'])) {
|
||
11 years ago
|
mod_page(_('New ban'), 'mod/ban_form.html', array('token' => make_secure_link_token('ban')));
|
||
11 years ago
|
return;
|
||
|
}
|
||
|
|
||
11 years ago
|
require_once 'inc/mod/ban.php';
|
||
|
|
||
|
ban($_POST['ip'], $_POST['reason'], parse_time($_POST['length']), $_POST['board'] == '*' ? false : $_POST['board']);
|
||
|
|
||
11 years ago
|
if (isset($_POST['redirect']))
|
||
11 years ago
|
header('Location: ' . $_POST['redirect'], true, $config['redirect_http']);
|
||
|
else
|
||
|
header('Location: ?/', true, $config['redirect_http']);
|
||
|
}
|
||
|
|
||
11 years ago
|
function mod_bans($page_no = 1) {
|
||
11 years ago
|
global $config;
|
||
|
|
||
11 years ago
|
if ($page_no < 1)
|
||
|
error($config['error']['404']);
|
||
|
|
||
11 years ago
|
if (!hasPermission($config['mod']['view_banlist']))
|
||
|
error($config['error']['noaccess']);
|
||
|
|
||
|
if (isset($_POST['unban'])) {
|
||
|
if (!hasPermission($config['mod']['unban']))
|
||
|
error($config['error']['noaccess']);
|
||
|
|
||
|
$unban = array();
|
||
|
foreach ($_POST as $name => $unused) {
|
||
|
if (preg_match('/^ban_(\d+)$/', $name, $match))
|
||
|
$unban[] = $match[1];
|
||
|
}
|
||
10 years ago
|
if (isset($config['mod']['unban_limit'])){
|
||
|
if (count($unban) <= $config['mod']['unban_limit'] || $config['mod']['unban_limit'] == -1){
|
||
11 years ago
|
if (!empty($unban)) {
|
||
10 years ago
|
query('DELETE FROM ``bans`` WHERE `id` = ' . implode(' OR `id` = ', $unban)) or error(db_error());
|
||
11 years ago
|
|
||