$config, 'mod' => $mod, 'hide_dashboard_link' => $template == 'mod/dashboard.html', 'title' => $title, 'subtitle' => $subtitle, 'nojavascript' => true, 'body' => Element($template, array_merge( array('config' => $config, 'mod' => $mod), $args ) ) ) ); } function mod_login($redirect = false) { global $config; $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 { modLog('Logged in'); // Login successful // Set cookies setCookies(); if ($redirect) header('Location: ?' . $redirect, true, $config['redirect_http']); else header('Location: ?/', true, $config['redirect_http']); } } if (isset($_POST['username'])) $args['username'] = $_POST['username']; mod_page(_('Login'), 'mod/login.html', $args); } function mod_confirm($request) { mod_page(_('Confirm action'), 'mod/confirm.html', array('request' => $request, 'token' => make_secure_link_token($request))); } function mod_logout() { global $config; destroyCookies(); header('Location: ?/', true, $config['redirect_http']); } function mod_dashboard() { global $config, $mod; $args = array(); $args['boards'] = listBoards(); if (hasPermission($config['mod']['noticeboard'])) { if (!$config['cache']['enabled'] || !$args['noticeboard'] = cache::get('noticeboard_preview')) { $query = prepare("SELECT ``noticeboard``.*, `username` FROM ``noticeboard`` LEFT JOIN ``mods`` ON ``mods``.`id` = `mod` ORDER BY `id` DESC LIMIT :limit"); $query->bindValue(':limit', $config['mod']['noticeboard_dashboard'], PDO::PARAM_INT); $query->execute() or error(db_error($query)); $args['noticeboard'] = $query->fetchAll(PDO::FETCH_ASSOC); if ($config['cache']['enabled']) cache::set('noticeboard_preview', $args['noticeboard']); } } if (!$config['cache']['enabled'] || ($args['unread_pms'] = cache::get('pm_unreadcount_' . $mod['id'])) === false) { $query = prepare('SELECT COUNT(*) FROM ``pms`` WHERE `to` = :id AND `unread` = 1'); $query->bindValue(':id', $mod['id']); $query->execute() or error(db_error($query)); $args['unread_pms'] = $query->fetchColumn(); if ($config['cache']['enabled']) cache::set('pm_unreadcount_' . $mod['id'], $args['unread_pms']); } $query = prepare('SELECT COUNT(*) AS `total_reports` FROM ``reports``' . (($mod["type"] < GLOBALVOLUNTEER) ? " WHERE board = :board" : "")); if ($mod['type'] < GLOBALVOLUNTEER) { $query->bindValue(':board', $mod['boards'][0]); } else { $query = prepare('SELECT (SELECT COUNT(id) FROM reports WHERE global = 0) AS total_reports, (SELECT COUNT(id) FROM reports WHERE global = 1) AS global_reports'); } $query->execute() or error(db_error($query)); $row = $query->fetch(); $args['reports'] = $row['total_reports']; $args['global_reports'] = isset($row['global_reports']) ? $row['global_reports'] : false; $args['logout_token'] = make_secure_link_token('logout'); modLog('Looked at dashboard', false); mod_page(_('Dashboard'), 'mod/dashboard.html', $args); } function mod_search_redirect() { global $config; if (!hasPermission($config['mod']['search'])) error($config['error']['noaccess']); if (isset($_POST['query'], $_POST['type']) && in_array($_POST['type'], array('posts', 'IP_notes', 'bans', 'log'))) { $query = $_POST['query']; $query = urlencode($query); $query = str_replace('_', '%5F', $query); $query = str_replace('+', '_', $query); if ($query === '') { header('Location: ?/', true, $config['redirect_http']); return; } header('Location: ?/search/' . $_POST['type'] . '/' . $query, true, $config['redirect_http']); } else { header('Location: ?/', true, $config['redirect_http']); } } function mod_search($type, $search_query_escaped, $page_no = 1) { global $pdo, $config; if (!hasPermission($config['mod']['search'])) error($config['error']['noaccess']); // Unescape query $query = str_replace('_', ' ', $search_query_escaped); $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); $query = str_replace('`', '!`', $query); // Array of phrases to match $match = array(); // Exact phrases ("like this") if (preg_match_all('/"(.+?)"/', $query, $exact_phrases)) { $exact_phrases = $exact_phrases[1]; foreach ($exact_phrases as $phrase) { $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') $sql_field = array('body_nomarkup', 'files', 'subject', 'filehash', 'ip', 'name', 'trip'); if ($type == 'IP_notes') $sql_field = 'body'; if ($type == 'bans') $sql_field = 'reason'; if ($type == 'log') $sql_field = 'text'; // 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); 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 \'!\''; } } // Compile SQL query if ($type == 'posts') { $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 '; $query .= sprintf("SELECT *, '%s' AS `board` FROM ``posts_%s`` WHERE %s", $board['uri'], $board['uri'], $sql_like); } // You weren't allowed to search any boards if (empty($query)) error($config['error']['noaccess']); $query .= ' ORDER BY `sticky` DESC, `id` DESC'; } if ($type == 'IP_notes') { $query = 'SELECT * FROM ``ip_notes`` LEFT JOIN ``mods`` ON `mod` = ``mods``.`id` WHERE ' . $sql_like . ' ORDER BY `time` DESC'; $sql_table = 'ip_notes'; if (!hasPermission($config['mod']['view_notes']) || !hasPermission($config['mod']['show_ip'])) error($config['error']['noaccess']); } if ($type == 'bans') { $query = 'SELECT ``bans``.*, `username` FROM ``bans`` LEFT JOIN ``mods`` ON `creator` = ``mods``.`id` WHERE ' . $sql_like . ' ORDER BY (`expires` IS NOT NULL AND `expires` < UNIX_TIMESTAMP()), `created` DESC'; $sql_table = 'bans'; if (!hasPermission($config['mod']['view_banlist'])) error($config['error']['noaccess']); } if ($type == 'log') { $query = 'SELECT `username`, `mod`, `ip`, `board`, `time`, `text` FROM ``modlogs`` LEFT JOIN ``mods`` ON `mod` = ``mods``.`id` WHERE ' . $sql_like . ' ORDER BY `time` DESC'; $sql_table = 'modlogs'; if (!hasPermission($config['mod']['modlog'])) error($config['error']['noaccess']); } // 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); // Get total result count 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(); } if ($type == 'bans') { foreach ($results as &$ban) { $ban['mask'] = Bans::range_to_string(array($ban['ipstart'], $ban['ipend'])); if (filter_var($ban['mask'], FILTER_VALIDATE_IP) !== false) $ban['single_addr'] = true; } } if ($type == 'posts') { foreach ($results as &$post) { $post['snippet'] = pm_snippet($post['body']); } } // $results now contains the search results mod_page(_('Search results'), 'mod/search_results.html', array( 'search_type' => $type, 'search_query' => $search_query, 'search_query_escaped' => $search_query_escaped, 'result_count' => $result_count, 'results' => $results )); } 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']); $query = prepare('DELETE FROM ``boards`` WHERE `uri` = :uri'); $query->bindValue(':uri', $board['uri']); $query->execute() or error(db_error($query)); if ($config['cache']['enabled']) { cache::delete('board_' . $board['uri']); cache::delete('all_boards'); cache::delete('all_boards_uri'); } modLog('Deleted board: ' . sprintf($config['board_abbreviation'], $board['uri']), false); // Delete posting table $query = query(sprintf('DROP TABLE IF EXISTS ``posts_%s``', $board['uri'])) or error(db_error()); // Clear reports $query = prepare('DELETE FROM ``reports`` WHERE `board` = :id'); $query->bindValue(':id', $board['uri'], PDO::PARAM_INT); $query->execute() or error(db_error($query)); // Delete from table $query = prepare('DELETE FROM ``boards`` WHERE `uri` = :uri'); $query->bindValue(':uri', $board['uri'], PDO::PARAM_INT); $query->execute() or error(db_error($query)); $query = prepare("SELECT `board`, `post` FROM ``cites`` WHERE `target_board` = :board ORDER BY `board`"); $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']); } } if (isset($tmp_board)) $board = $tmp_board; $query = prepare('DELETE FROM ``cites`` WHERE `board` = :board OR `target_board` = :board'); $query->bindValue(':board', $board['uri']); $query->execute() or error(db_error($query)); $query = prepare('DELETE FROM ``antispam`` WHERE `board` = :board'); $query->bindValue(':board', $board['uri']); $query->execute() or error(db_error($query)); // Remove board from users/permissions table $query = query('SELECT `id`,`boards` FROM ``mods``') or error(db_error()); 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)]); $_query = prepare('UPDATE ``mods`` SET `boards` = :boards WHERE `id` = :id'); $_query->bindValue(':boards', implode(',', $user_boards)); $_query->bindValue(':id', $user['id']); $_query->execute() or error(db_error($_query)); } } // Delete entire board directory rrmdir($board['uri'] . '/'); // To reiterate: HAAAAAX if($config['dir']['img_root'] != '') rrmdir($config['dir']['img_root'] . $board['uri']); } else { $query = prepare('UPDATE ``boards`` SET `title` = :title, `subtitle` = :subtitle WHERE `uri` = :uri'); $query->bindValue(':uri', $board['uri']); $query->bindValue(':title', $_POST['title']); $query->bindValue(':subtitle', $_POST['subtitle']); $query->execute() or error(db_error($query)); modLog('Edited board information for ' . sprintf($config['board_abbreviation'], $board['uri']), false); } if ($config['cache']['enabled']) { cache::delete('board_' . $board['uri']); cache::delete('all_boards'); cache::delete('all_boards_uri'); } rebuildThemes('boards'); header('Location: ?/', true, $config['redirect_http']); } else { mod_page(sprintf('%s: ' . $config['board_abbreviation'], _('Edit board'), $board['uri']), 'mod/board.html', array( 'board' => $board, 'token' => make_secure_link_token('edit/' . $board['uri']) )); } } 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')); if (!preg_match('/^' . $config['board_regex'] . '$/u', $_POST['uri'])) error(sprintf($config['error']['invalidfield'], 'URI')); $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; } if (openBoard($_POST['uri'])) { error(sprintf($config['error']['boardexists'], $board['url'])); } foreach ($config['banned_boards'] as $i => $w) { if ($w[0] !== '/') { if (strpos($_POST['uri'],$w) !== false) error(_("Cannot create board with banned word $w")); } else { if (preg_match($w,$_POST['uri'])) error(_("Cannot create board matching banned pattern $w")); } } $query = prepare('INSERT INTO ``boards`` (``uri``, ``title``, ``subtitle``) VALUES (:uri, :title, :subtitle)'); $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.")); $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()); if ($config['cache']['enabled']) cache::delete('all_boards'); // Build the board buildIndex(); rebuildThemes('boards'); header('Location: ?/' . $board['uri'] . '/' . $config['file_index'], true, $config['redirect_http']); } mod_page(_('New board'), 'mod/board.html', array('new' => true, 'token' => make_secure_link_token('new-board'))); } function mod_noticeboard($page_no = 1) { global $config, $pdo, $mod; if ($page_no < 1) error($config['error']['404']); 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']); $_POST['body'] = escape_markup_modifiers($_POST['body']); markup($_POST['body']); $query = prepare('INSERT INTO ``noticeboard`` VALUES (NULL, :mod, :time, :subject, :body)'); $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)); if ($config['cache']['enabled']) cache::delete('noticeboard_preview'); modLog('Posted a noticeboard entry'); header('Location: ?/noticeboard#' . $pdo->lastInsertId(), true, $config['redirect_http']); } $query = prepare("SELECT ``noticeboard``.*, `username` FROM ``noticeboard`` LEFT JOIN ``mods`` ON ``mods``.`id` = `mod` ORDER BY `id` DESC LIMIT :offset, :limit"); $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); if (empty($noticeboard) && $page_no > 1) error($config['error']['404']); foreach ($noticeboard as &$entry) { $entry['delete_token'] = make_secure_link_token('noticeboard/delete/' . $entry['id']); } $query = prepare("SELECT COUNT(*) FROM ``noticeboard``"); $query->execute() or error(db_error($query)); $count = $query->fetchColumn(); mod_page(_('Noticeboard'), 'mod/noticeboard.html', array( 'noticeboard' => $noticeboard, 'count' => $count, 'token' => make_secure_link_token('noticeboard') )); } function mod_noticeboard_delete($id) { global $config; if (!hasPermission($config['mod']['noticeboard_delete'])) error($config['error']['noaccess']); $query = prepare('DELETE FROM ``noticeboard`` WHERE `id` = :id'); $query->bindValue(':id', $id); $query->execute() or error(db_error($query)); modLog('Deleted a noticeboard entry'); if ($config['cache']['enabled']) cache::delete('noticeboard_preview'); 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']); $_POST['body'] = escape_markup_modifiers($_POST['body']); markup($_POST['body']); $query = prepare('INSERT INTO ``news`` VALUES (NULL, :name, :time, :subject, :body)'); $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: ?/edit_news#' . $pdo->lastInsertId(), true, $config['redirect_http']); } $query = prepare("SELECT * FROM ``news`` ORDER BY `id` DESC LIMIT :offset, :limit"); $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']); foreach ($news as &$entry) { $entry['delete_token'] = make_secure_link_token('edit_news/delete/' . $entry['id']); } $query = prepare("SELECT COUNT(*) FROM ``news``"); $query->execute() or error(db_error($query)); $count = $query->fetchColumn(); mod_page(_('News'), 'mod/news.html', array('news' => $news, 'count' => $count, 'token' => make_secure_link_token('edit_news'))); } function mod_news_delete($id) { global $config; if (!hasPermission($config['mod']['news_delete'])) error($config['error']['noaccess']); $query = prepare('DELETE FROM ``news`` WHERE `id` = :id'); $query->bindValue(':id', $id); $query->execute() or error(db_error($query)); modLog('Deleted a news entry'); header('Location: ?/edit_news', true, $config['redirect_http']); } function mod_log($page_no = 1) { global $config; if ($page_no < 1) error($config['error']['404']); if (!hasPermission($config['mod']['modlog'])) error($config['error']['noaccess']); $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"); $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']); $query = prepare("SELECT COUNT(*) FROM ``modlogs``"); $query->execute() or error(db_error($query)); $count = $query->fetchColumn(); mod_page(_('Board log'), 'mod/log.html', array('logs' => $logs, 'count' => $count)); } 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']); $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"); $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']); $query = prepare("SELECT COUNT(*) FROM ``modlogs`` LEFT JOIN ``mods`` ON `mod` = ``mods``.`id` WHERE `username` = :username"); $query->bindValue(':username', $username); $query->execute() or error(db_error($query)); $count = $query->fetchColumn(); mod_page(_('Board log'), 'mod/log.html', array('logs' => $logs, 'count' => $count, 'username' => $username)); } function mod_board_log($board, $page_no = 1, $hide_names = false, $public = false) { global $config; if ($page_no < 1) error($config['error']['404']); if (!hasPermission($config['mod']['mod_board_log'], $board) && !$public) error($config['error']['noaccess']); $query = prepare("SELECT `username`, `mod`, `ip`, `board`, `time`, `text` FROM ``modlogs`` LEFT JOIN ``mods`` ON `mod` = ``mods``.`id` WHERE `board` = :board ORDER BY `time` DESC LIMIT :offset, :limit"); $query->bindValue(':board', $board); $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']); if (!hasPermission($config['mod']['show_ip'])) { // Supports ipv4 only! foreach ($logs as $i => &$log) { $log['text'] = preg_replace_callback('/(?:)?(\d{1,3}\.\d{1,3}\.\d{1,3}\.\d{1,3})(?:<\/a>)?/', function($matches) { global $board; return less_ip($matches[1], $board['uri']); }, $log['text']); } } $query = prepare("SELECT COUNT(*) FROM ``modlogs`` LEFT JOIN ``mods`` ON `mod` = ``mods``.`id` WHERE `board` = :board"); $query->bindValue(':board', $board); $query->execute() or error(db_error($query)); $count = $query->fetchColumn(); mod_page(_('Board log'), 'mod/log.html', array('logs' => $logs, 'count' => $count, 'board' => $board, 'hide_names' => $hide_names, 'public' => $public)); } 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; 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; } function mod_view_thread50($boardName, $thread) { global $config, $mod; if (!openBoard($boardName)) error($config['error']['noboard']); $page = buildThread50($thread, true, $mod); echo $page; } function mod_ip_remove_note($ip, $id) { global $config, $mod; if (!hasPermission($config['mod']['remove_notes'])) error($config['error']['noaccess']); if (filter_var($ip, FILTER_VALIDATE_IP) === false) error("Invalid IP address."); $query = prepare('DELETE FROM ``ip_notes`` WHERE `ip` = :ip AND `id` = :id'); $query->bindValue(':ip', $ip); $query->bindValue(':id', $id); $query->execute() or error(db_error($query)); modLog("Removed a note for {$ip}"); header('Location: ?/IP/' . $ip . '#notes', true, $config['redirect_http']); } function mod_page_ip($ip) { global $config, $mod; if (!hasPermission($config['mod']['show_ip'])) error($config['error']['noaccess']); if (filter_var($ip, FILTER_VALIDATE_IP) === false) error("Invalid IP address."); if (isset($_POST['ban_id'], $_POST['unban'])) { if (!hasPermission($config['mod']['unban'])) error($config['error']['noaccess']); Bans::delete($_POST['ban_id'], true, $mod['boards']); header('Location: ?/IP/' . $ip . '#bans', true, $config['redirect_http']); return; } if (isset($_POST['note'])) { if (!hasPermission($config['mod']['create_notes'])) error($config['error']['noaccess']); $_POST['note'] = escape_markup_modifiers($_POST['note']); markup($_POST['note']); $query = prepare('INSERT INTO ``ip_notes`` VALUES (NULL, :ip, :mod, :time, :body)'); $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)); modLog("Added a note for {$ip}"); header('Location: ?/IP/' . $ip . '#notes', true, $config['redirect_http']); return; } $args = array(); $args['ip'] = $ip; $args['posts'] = array(); if ($config['mod']['dns_lookup']) $args['hostname'] = rDNS($ip); $boards = listBoards(); foreach ($boards as $board) { openBoard($board['uri']); if (!hasPermission($config['mod']['show_ip'], $board['uri'])) continue; $query = prepare(sprintf('SELECT * FROM ``posts_%s`` WHERE `ip` = :ip ORDER BY `sticky` DESC, `id` DESC LIMIT :limit', $board['uri'])); $query->bindValue(':ip', $ip); $query->bindValue(':limit', $config['mod']['ip_recentposts'], PDO::PARAM_INT); $query->execute() or error(db_error($query)); while ($post = $query->fetch(PDO::FETCH_ASSOC)) { if (!$post['thread']) { $po = new Thread($post, '?/', $mod, false); } else { $po = new Post($post, '?/', $mod); } if (!isset($args['posts'][$board['uri']])) $args['posts'][$board['uri']] = array('board' => $board, 'posts' => array()); $args['posts'][$board['uri']]['posts'][] = $po->build(true); } } $args['boards'] = $boards; $args['token'] = make_secure_link_token('ban'); if (hasPermission($config['mod']['view_ban'])) { $args['bans'] = Bans::find($ip, false, true); } if (hasPermission($config['mod']['view_notes'])) { $query = prepare("SELECT ``ip_notes``.*, `username` FROM ``ip_notes`` LEFT JOIN ``mods`` ON `mod` = ``mods``.`id` WHERE `ip` = :ip ORDER BY `time` DESC"); $query->bindValue(':ip', $ip); $query->execute() or error(db_error($query)); $args['notes'] = $query->fetchAll(PDO::FETCH_ASSOC); } if (hasPermission($config['mod']['modlog_ip'])) { $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"); $query->bindValue(':search', '%' . $ip . '%'); $query->execute() or error(db_error($query)); $args['logs'] = $query->fetchAll(PDO::FETCH_ASSOC); } else { $args['logs'] = array(); } $args['security_token'] = make_secure_link_token('IP/' . $ip); mod_page(sprintf('%s: %s', _('IP'), $ip), 'mod/view_ip.html', $args, $args['hostname']); } function mod_page_ip_less($b, $id) { global $config, $mod; if (!openBoard($b)) error('No board.'); $query = prepare(sprintf('SELECT `ip` FROM ``posts_%s`` WHERE `id` = :id', $b)); $query->bindValue(':id', $id); $query->execute() or error(db_error($query)); $result = $query->fetch(PDO::FETCH_ASSOC); if ($result) { $ip = $result['ip']; } else { error(_('Could not find that post.')); } if (filter_var($ip, FILTER_VALIDATE_IP) === false) error("Invalid IP address."); if (isset($_POST['ban_id'], $_POST['unban'])) { if (!hasPermission($config['mod']['unban'])) error($config['error']['noaccess']); Bans::delete($_POST['ban_id'], true); header('Location: ?/IP_less/' . $b . '/' . $id . '#bans', true, $config['redirect_http']); return; } if (isset($_POST['note'])) { if (!hasPermission($config['mod']['create_notes'])) error($config['error']['noaccess']); $_POST['note'] = escape_markup_modifiers($_POST['note']); markup($_POST['note']); $query = prepare('INSERT INTO ``ip_notes`` VALUES (NULL, :ip, :mod, :time, :body)'); $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)); modLog("Added a note for {$ip}"); header('Location: ?/IP_less/' . $b . '/' . $id . '#notes', true, $config['redirect_http']); return; } $args = array(); $args['ip'] = $ip; $args['posts'] = array(); if ($config['mod']['dns_lookup']) $args['hostname'] = rDNS($ip); $query = prepare(sprintf('SELECT * FROM ``posts_%s`` WHERE `ip` = :ip ORDER BY `sticky` DESC, `id` DESC LIMIT :limit', $b)); $query->bindValue(':ip', $ip); $query->bindValue(':limit', $config['mod']['ip_less_recentposts'], PDO::PARAM_INT); $query->execute() or error(db_error($query)); while ($post = $query->fetch(PDO::FETCH_ASSOC)) { if (!$post['thread']) { $po = new Thread($post, '?/', $mod, false); } else { $po = new Post($post, '?/', $mod); } if (!isset($args['posts'][$b])) $args['posts'][$b] = array('board' => $b, 'posts' => array()); $args['posts'][$b]['posts'][] = $po->build(true); } $args['boards'] = listBoards(); $args['token'] = make_secure_link_token('ban'); if (hasPermission($config['mod']['view_ban'])) { $args['bans'] = Bans::find($ip, false, true); } if (hasPermission($config['mod']['view_notes'])) { $query = prepare("SELECT ``ip_notes``.*, `username` FROM ``ip_notes`` LEFT JOIN ``mods`` ON `mod` = ``mods``.`id` WHERE `ip` = :ip ORDER BY `time` DESC"); $query->bindValue(':ip', $ip); $query->execute() or error(db_error($query)); $args['notes'] = $query->fetchAll(PDO::FETCH_ASSOC); } if (hasPermission($config['mod']['modlog_ip'])) { $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"); $query->bindValue(':search', '%' . $ip . '%'); $query->execute() or error(db_error($query)); $args['logs'] = $query->fetchAll(PDO::FETCH_ASSOC); } else { $args['logs'] = array(); } $args['security_token'] = make_secure_link_token('IP_less/' . $b . '/' . $id); mod_page(sprintf('%s: %s', _('IP'), less_ip($ip, $b)), 'mod/view_ip_less.html', $args); } function mod_ban() { global $config, $mod; if (!hasPermission($config['mod']['ban'])) error($config['error']['noaccess']); if (!isset($_POST['ip'], $_POST['reason'], $_POST['length'], $_POST['board'])) { mod_page(_('New ban'), 'mod/ban_form.html', array('token' => make_secure_link_token('ban'))); return; } require_once 'inc/mod/ban.php'; Bans::new_ban($_POST['ip'], $_POST['reason'], $_POST['length'], $_POST['board'] == '*' ? false : $_POST['board']); if (isset($_POST['redirect'])) header('Location: ' . $_POST['redirect'], true, $config['redirect_http']); else header('Location: ?/', true, $config['redirect_http']); } function mod_bans() { global $config; global $mod; 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]; } if (isset($config['mod']['unban_limit']) && $config['mod']['unban_limit'] && count($unban) > $config['mod']['unban_limit']) error(sprintf($config['error']['toomanyunban'], $config['mod']['unban_limit'], count($unban))); foreach ($unban as $id) { Bans::delete($id, true, $mod['boards'], true); } rebuildThemes('bans'); header('Location: ?/bans', true, $config['redirect_http']); return; } mod_page(_('Ban list'), 'mod/ban_list.html', array( 'mod' => $mod, 'boards' => json_encode($mod['boards']), 'token' => make_secure_link_token('bans'), 'token_json' => make_secure_link_token('bans.json') )); } function mod_bans_json() { global $config, $mod; if (!hasPermission($config['mod']['ban'])) error($config['error']['noaccess']); // Compress the json for faster loads if (isset($_SERVER['HTTP_ACCEPT_ENCODING']) && substr_count($_SERVER['HTTP_ACCEPT_ENCODING'], 'gzip')) ob_start("ob_gzhandler"); Bans::stream_json(false, !hasPermission($config['mod']['show_ip']), !hasPermission($config['mod']['view_banstaff']), $mod['boards']); } function mod_ban_appeals() { global $config, $board, $mod; if (!hasPermission($config['mod']['view_ban_appeals'])) error($config['error']['noaccess']); // Remove stale ban appeals query("DELETE FROM ``ban_appeals`` WHERE NOT EXISTS (SELECT 1 FROM ``bans`` WHERE `ban_id` = ``bans``.`id`)") or error(db_error()); if (isset($_POST['appeal_id']) && (isset($_POST['unban']) || isset($_POST['deny']))) { if (!hasPermission($config['mod']['ban_appeals'])) error($config['error']['noaccess']); $query = query("SELECT *, ``ban_appeals``.`id` AS `id` FROM ``ban_appeals`` LEFT JOIN ``bans`` ON `ban_id` = ``bans``.`id` WHERE ``ban_appeals``.`id` = " . (int)$_POST['appeal_id']) or error(db_error()); if (!$ban = $query->fetch(PDO::FETCH_ASSOC)) { error(_('Ban appeal not found!')); } if (!in_array($ban['board'], $mod['boards']) && $mod['boards'][0] != '*') error($config['error']['noaccess']); $ban['mask'] = Bans::range_to_string(array($ban['ipstart'], $ban['ipend'])); if (isset($_POST['unban'])) { modLog('Accepted ban appeal #' . $ban['id'] . ' for ' . $ban['mask']); Bans::delete($ban['ban_id'], true); query("DELETE FROM ``ban_appeals`` WHERE `id` = " . $ban['id']) or error(db_error()); } else { modLog('Denied ban appeal #' . $ban['id'] . ' for ' . $ban['mask']); query("UPDATE ``ban_appeals`` SET `denied` = 1 WHERE `id` = " . $ban['id']) or error(db_error()); } header('Location: ?/ban-appeals', true, $config['redirect_http']); return; } $local = ($mod['type'] < GLOBALVOLUNTEER); $query = prepare("SELECT *, ``ban_appeals``.`id` AS `id` FROM ``ban_appeals`` LEFT JOIN ``bans`` ON `ban_id` = ``bans``.`id` LEFT JOIN ``mods`` ON ``bans``.`creator` = ``mods``.`id` WHERE `denied` != 1 ".($local ? " AND ``bans``.`board` = :board " : "")." ORDER BY `time`"); if ($local) { $query->bindValue(':board', $mod['boards'][0]); } $query->execute() or error(db_error()); $ban_appeals = $query->fetchAll(PDO::FETCH_ASSOC); foreach ($ban_appeals as &$ban) { if ($ban['post']) $ban['post'] = json_decode($ban['post'], true); $ban['mask'] = Bans::range_to_string(array($ban['ipstart'], $ban['ipend'])); if ($ban['post'] && isset($ban['post']['board'], $ban['post']['id'])) { openBoard($ban['post']['board']); if ($ban['post']['thread']) { $po = new Post($ban['post']); $ban['post'] = $po->build(true); } else { $po = new Thread($ban['post'], null, false, false); $ban['post'] = $po->build(true); } } } mod_page(_('Ban appeals'), 'mod/ban_appeals.html', array( 'ban_appeals' => $ban_appeals, 'token' => make_secure_link_token('ban-appeals') )); } function mod_lock($board, $unlock, $post) { global $config; if (!openBoard($board)) error($config['error']['noboard']); if (!hasPermission($config['mod']['lock'], $board)) error($config['error']['noaccess']); $query = prepare(sprintf('UPDATE ``posts_%s`` SET `locked` = :locked WHERE `id` = :id AND `thread` IS NULL', $board)); $query->bindValue(':id', $post); $query->bindValue(':locked', $unlock ? 0 : 1); $query->execute() or error(db_error($query)); if ($query->rowCount()) { modLog(($unlock ? 'Unlocked' : 'Locked') . " thread #{$post}"); buildThread($post); buildIndex(); } if ($config['mod']['dismiss_reports_on_lock']) { $query = prepare('DELETE FROM ``reports`` WHERE `board` = :board AND `post` = :id'); $query->bindValue(':board', $board); $query->bindValue(':id', $post); $query->execute() or error(db_error($query)); } header('Location: ?/' . sprintf($config['board_path'], $board) . $config['file_index'], true, $config['redirect_http']); if ($unlock) event('unlock', $post); else event('lock', $post); } function mod_sticky($board, $unsticky, $post) { global $config; if (!openBoard($board)) error($config['error']['noboard']); if (!hasPermission($config['mod']['sticky'], $board)) error($config['error']['noaccess']); $query = prepare(sprintf('UPDATE ``posts_%s`` SET `sticky` = :sticky WHERE `id` = :id AND `thread` IS NULL', $board)); $query->bindValue(':id', $post); $query->bindValue(':sticky', $unsticky ? 0 : 1); $query->execute() or error(db_error($query)); if ($query->rowCount()) { modLog(($unsticky ? 'Unstickied' : 'Stickied') . " thread #{$post}"); buildThread($post); buildIndex(); } header('Location: ?/' . sprintf($config['board_path'], $board) . $config['file_index'], true, $config['redirect_http']); } function mod_cycle($board, $uncycle, $post) { global $config; if (!openBoard($board)) error($config['error']['noboard']); if (!hasPermission($config['mod']['cycle'], $board)) error($config['error']['noaccess']); $query = prepare(sprintf('UPDATE ``posts_%s`` SET `cycle` = :cycle WHERE `id` = :id AND `thread` IS NULL', $board)); $query->bindValue(':id', $post); $query->bindValue(':cycle', $uncycle ? 0 : 1); $query->execute() or error(db_error($query)); if ($query->rowCount()) { modLog(($uncycle ? 'Made not cyclical' : 'Made cyclical') . " thread #{$post}"); buildThread($post); buildIndex(); } header('Location: ?/' . sprintf($config['board_path'], $board) . $config['file_index'], true, $config['redirect_http']); } function mod_bumplock($board, $unbumplock, $post) { global $config; if (!openBoard($board)) error($config['error']['noboard']); if (!hasPermission($config['mod']['bumplock'], $board)) error($config['error']['noaccess']); $query = prepare(sprintf('UPDATE ``posts_%s`` SET `sage` = :bumplock WHERE `id` = :id AND `thread` IS NULL', $board)); $query->bindValue(':id', $post); $query->bindValue(':bumplock', $unbumplock ? 0 : 1); $query->execute() or error(db_error($query)); if ($query->rowCount()) { modLog(($unbumplock ? 'Unbumplocked' : 'Bumplocked') . " thread #{$post}"); buildThread($post); buildIndex(); } header('Location: ?/' . sprintf($config['board_path'], $board) . $config['file_index'], true, $config['redirect_http']); } function mod_move_reply($originBoard, $postID) { global $board, $config, $mod; if (!openBoard($originBoard)) error($config['error']['noboard']); if (!hasPermission($config['mod']['move'], $originBoard)) error($config['error']['noaccess']); $query = prepare(sprintf('SELECT * FROM ``posts_%s`` WHERE `id` = :id', $originBoard)); $query->bindValue(':id', $postID); $query->execute() or error(db_error($query)); if (!$post = $query->fetch(PDO::FETCH_ASSOC)) error($config['error']['404']); if (isset($_POST['board'])) { $targetBoard = $_POST['board']; if ($_POST['target_thread']) { $query = prepare(sprintf('SELECT * FROM ``posts_%s`` WHERE `id` = :id', $targetBoard)); $query->bindValue(':id', $_POST['target_thread']); $query->execute() or error(db_error($query)); // If it fails, thread probably does not exist $post['op'] = false; $post['thread'] = $_POST['target_thread']; } else { $post['op'] = true; } if ($post['files']) { $post['files'] = json_decode($post['files'], TRUE); $post['has_file'] = true; foreach ($post['files'] as $i => &$file) { $file['file_path'] = sprintf($config['board_path'], $config['dir']['img_root'] . $board['uri']) . $config['dir']['img'] . $file['file']; $file['thumb_path'] = sprintf($config['board_path'], $config['dir']['img_root'] . $board['uri']) . $config['dir']['thumb'] . $file['thumb']; } } else { $post['has_file'] = false; } // allow thread to keep its same traits (stickied, locked, etc.) $post['mod'] = true; if (!openBoard($targetBoard)) error($config['error']['noboard']); // create the new post $newID = post($post); if ($post['has_file']) { foreach ($post['files'] as $i => &$file) { // move the image rename($file['file_path'], sprintf($config['board_path'], $config['dir']['img_root'] . $board['uri']) . $config['dir']['img'] . $file['file']); if ($file['thumb'] != 'spoiler') { //trying to move/copy the spoiler thumb raises an error rename($file['thumb_path'], sprintf($config['board_path'], $config['dir']['img_root'] . $board['uri']) . $config['dir']['thumb'] . $file['thumb']); } } } // build index buildIndex(); // build new thread buildThread($newID); // trigger themes rebuildThemes('post', $targetBoard); // mod log modLog("Moved post #${postID} to " . sprintf($config['board_abbreviation'], $targetBoard) . " (#${newID})", $originBoard); // return to original board openBoard($originBoard); // delete original post deletePost($postID); buildIndex(); // open target board for redirect openBoard($targetBoard); // Find new thread on our target board $query = prepare(sprintf('SELECT thread FROM ``posts_%s`` WHERE `id` = :id', $targetBoard)); $query->bindValue(':id', $newID); $query->execute() or error(db_error($query)); $post = $query->fetch(PDO::FETCH_ASSOC); // redirect header('Location: ?/' . sprintf($config['board_path'], $board['uri']) . $config['dir']['res'] . sprintf($config['file_page'], $post['thread'] ? $post['thread'] : $newID) . '#' . $newID, true, $config['redirect_http']); } else { $boards = listBoards(); $security_token = make_secure_link_token($originBoard . '/move_reply/' . $postID); mod_page(_('Move reply'), 'mod/move_reply.html', array('post' => $postID, 'board' => $originBoard, 'boards' => $boards, 'token' => $security_token)); } } function mod_move($originBoard, $postID) { global $board, $config, $mod, $pdo; if (!openBoard($originBoard)) error($config['error']['noboard']); if (!hasPermission($config['mod']['move'], $originBoard)) error($config['error']['noaccess']); $query = prepare(sprintf('SELECT * FROM ``posts_%s`` WHERE `id` = :id AND `thread` IS NULL', $originBoard)); $query->bindValue(':id', $postID); $query->execute() or error(db_error($query)); if (!$post = $query->fetch(PDO::FETCH_ASSOC)) error($config['error']['404']); if (isset($_POST['board'])) { $targetBoard = $_POST['board']; $shadow = isset($_POST['shadow']); if ($targetBoard === $originBoard) error(_('Target and source board are the same.')); // copy() if leaving a shadow thread behind; else, rename(). $clone = $shadow ? 'copy' : 'rename'; // indicate that the post is a thread $post['op'] = true; if ($post['files']) { $post['files'] = json_decode($post['files'], TRUE); $post['has_file'] = true; foreach ($post['files'] as $i => &$file) { if ($file['file'] === 'deleted') continue; $file['file_path'] = sprintf($config['board_path'], $config['dir']['img_root'] . $board['uri']) . $config['dir']['img'] . $file['file']; $file['thumb_path'] = sprintf($config['board_path'], $config['dir']['img_root'] . $board['uri']) . $config['dir']['thumb'] . $file['thumb']; } } else { $post['has_file'] = false; } // allow thread to keep its same traits (stickied, locked, etc.) $post['mod'] = true; if (!openBoard($targetBoard)) error($config['error']['noboard']); // create the new thread $newID = post($post); if ($post['has_file']) { // copy image foreach ($post['files'] as $i => &$file) { if ($file['file'] !== 'deleted') $clone($file['file_path'], sprintf($config['board_path'], $config['dir']['img_root'] . $board['uri']) . $config['dir']['img'] . $file['file']); if (isset($file['thumb']) && !in_array($file['thumb'], array('spoiler', 'deleted', 'file'))) $clone($file['thumb_path'], sprintf($config['board_path'], $config['dir']['img_root'] . $board['uri']) . $config['dir']['thumb'] . $file['thumb']); } } // go back to the original board to fetch replies openBoard($originBoard); $query = prepare(sprintf('SELECT * FROM ``posts_%s`` WHERE `thread` = :id ORDER BY `id`', $originBoard)); $query->bindValue(':id', $postID, PDO::PARAM_INT); $query->execute() or error(db_error($query)); $replies = array(); while ($post = $query->fetch(PDO::FETCH_ASSOC)) { $post['mod'] = true; $post['thread'] = $newID; if ($post['files']) { $post['files'] = json_decode($post['files'], TRUE); $post['has_file'] = true; foreach ($post['files'] as $i => &$file) { $file['file_path'] = sprintf($config['board_path'], $config['dir']['img_root'] . $board['uri']) . $config['dir']['img'] . $file['file']; $file['thumb_path'] = sprintf($config['board_path'], $config['dir']['img_root'] . $board['uri']) . $config['dir']['thumb'] . $file['thumb']; } } else { $post['has_file'] = false; } $replies[] = $post; } $newIDs = array($postID => $newID); openBoard($targetBoard); foreach ($replies as &$post) { $query = prepare('SELECT `target` FROM ``cites`` WHERE `target_board` = :board AND `board` = :board AND `post` = :post'); $query->bindValue(':board', $originBoard); $query->bindValue(':post', $post['id'], PDO::PARAM_INT); $query->execute() or error(db_error($qurey)); // correct >>X links while ($cite = $query->fetch(PDO::FETCH_ASSOC)) { if (isset($newIDs[$cite['target']])) { $post['body_nomarkup'] = preg_replace( '/(>>(>\/' . preg_quote($originBoard, '/') . '\/)?)' . preg_quote($cite['target'], '/') . '/', '>>' . $newIDs[$cite['target']], $post['body_nomarkup']); $post['body'] = $post['body_nomarkup']; } } $post['body'] = $post['body_nomarkup']; $post['op'] = false; $post['tracked_cites'] = markup($post['body'], true); if ($post['has_file']) { // copy image foreach ($post['files'] as $i => &$file) { if ($file['file'] !== 'deleted') $clone($file['file_path'], sprintf($config['board_path'], $config['dir']['img_root'] . $board['uri']) . $config['dir']['img'] . $file['file']); if (isset($file['thumb']) && !in_array($file['thumb'], array('spoiler', 'deleted', 'file'))) $clone($file['thumb_path'], sprintf($config['board_path'], $config['dir']['img_root'] . $board['uri']) . $config['dir']['thumb'] . $file['thumb']); } } // insert reply $newIDs[$post['id']] = $newPostID = post($post); if (!empty($post['tracked_cites'])) { $insert_rows = array(); foreach ($post['tracked_cites'] as $cite) { $insert_rows[] = '(' . $pdo->quote($board['uri']) . ', ' . $newPostID . ', ' . $pdo->quote($cite[0]) . ', ' . (int)$cite[1] . ')'; } query('INSERT INTO ``cites`` VALUES ' . implode(', ', $insert_rows)) or error(db_error()); } } modLog("Moved thread #${postID} to " . sprintf($config['board_abbreviation'], $targetBoard) . " (#${newID})", $originBoard); // build new thread buildThread($newID); clean(); buildIndex(); // trigger themes rebuildThemes('post', $targetBoard); // return to original board openBoard($originBoard); if ($shadow) { // lock old thread $query = prepare(sprintf('UPDATE ``posts_%s`` SET `locked` = 1 WHERE `id` = :id', $originBoard)); $query->bindValue(':id', $postID, PDO::PARAM_INT); $query->execute() or error(db_error($query)); // leave a reply, linking to the new thread $post = array( 'mod' => true, 'subject' => '', 'email' => '', 'name' => (!$config['mod']['shadow_name'] ? $config['anonymous'] : $config['mod']['shadow_name']), 'capcode' => $config['mod']['shadow_capcode'], 'trip' => '', 'password' => '', 'has_file' => false, // attach to original thread 'thread' => $postID, 'op' => false ); $post['body'] = $post['body_nomarkup'] = sprintf($config['mod']['shadow_mesage'], '>>>/' . $targetBoard . '/' . $newID); markup($post['body']); $botID = post($post); buildThread($postID); buildIndex(); header('Location: ?/' . sprintf($config['board_path'], $originBoard) . $config['dir']['res'] .sprintf($config['file_page'], $postID) . '#' . $botID, true, $config['redirect_http']); } else { deletePost($postID); buildIndex(); openBoard($targetBoard); header('Location: ?/' . sprintf($config['board_path'], $board['uri']) . $config['dir']['res'] . sprintf($config['file_page'], $newID), true, $config['redirect_http']); } } $boards = listBoards(); if (count($boards) <= 1) error(_('Impossible to move thread; there is only one board.')); $security_token = make_secure_link_token($originBoard . '/move/' . $postID); mod_page(_('Move thread'), 'mod/move.html', array('post' => $postID, 'board' => $originBoard, 'boards' => $boards, 'token' => $security_token)); } function mod_ban_post($board, $delete, $post, $token = false) { global $config, $mod; if (!openBoard($board)) error($config['error']['noboard']); if (!hasPermission($config['mod']['delete'], $board)) error($config['error']['noaccess']); $security_token = make_secure_link_token($board . '/ban/' . $post); $query = prepare(sprintf('SELECT ' . ($config['ban_show_post'] ? '*' : '`ip`, `thread`') . ' FROM ``posts_%s`` WHERE `id` = :id', $board)); $query->bindValue(':id', $post); $query->execute() or error(db_error($query)); if (!$_post = $query->fetch(PDO::FETCH_ASSOC)) error($config['error']['404']); $thread = $_post['thread']; $ip = $_post['ip']; $tor = checkDNSBL($ip); if (isset($_POST['new_ban'], $_POST['reason'], $_POST['length'], $_POST['board'])) { require_once 'inc/mod/ban.php'; if (isset($_POST['ip'])) $ip = $_POST['ip']; if (isset($_POST['range'])) $ip = $ip . $_POST['range']; Bans::new_ban($ip, $_POST['reason'], $_POST['length'], $_POST['board'] == '*' ? false : $_POST['board'], false, $config['ban_show_post'] ? $_post : false); if (isset($_POST['public_message'], $_POST['message'])) { // public ban message $length_english = Bans::parse_time($_POST['length']) ? 'for ' . until(Bans::parse_time($_POST['length'])) : 'permanently'; $_POST['message'] = preg_replace('/[\r\n]/', '', $_POST['message']); $_POST['message'] = str_replace('%length%', $length_english, $_POST['message']); $_POST['message'] = str_replace('%LENGTH%', strtoupper($length_english), $_POST['message']); $query = prepare(sprintf('UPDATE ``posts_%s`` SET `body_nomarkup` = CONCAT(`body_nomarkup`, :body_nomarkup) WHERE `id` = :id', $board)); $query->bindValue(':id', $post); $query->bindValue(':body_nomarkup', sprintf("\n%s", utf8tohtml($_POST['message']))); $query->execute() or error(db_error($query)); rebuildPost($post); modLog("Attached a public ban message to post #{$post}: " . utf8tohtml($_POST['message'])); buildThread($thread ? $thread : $post); buildIndex(); } elseif (isset($_POST['delete']) && (int) $_POST['delete']) { // Delete post deletePost($post); modLog("Deleted post #{$post}"); // Rebuild board buildIndex(); // Rebuild themes rebuildThemes('post-delete', $board); } header('Location: ?/' . sprintf($config['board_path'], $board) . $config['file_index'], true, $config['redirect_http']); } $args = array( 'ip' => $ip, 'hide_ip' => !hasPermission($config['mod']['show_ip'], $board), 'post' => $post, 'board' => $board, 'tor' => $tor, 'delete' => (bool)$delete, 'boards' => listBoards(), 'token' => $security_token ); mod_page(_('New ban'), 'mod/ban_form.html', $args); } function mod_edit_post($board, $edit_raw_html, $postID) { global $config, $mod; if (!openBoard($board)) error($config['error']['noboard']); if (!hasPermission($config['mod']['editpost'], $board)) error($config['error']['noaccess']); if ($edit_raw_html && !hasPermission($config['mod']['rawhtml'], $board)) error($config['error']['noaccess']); $security_token = make_secure_link_token($board . '/edit' . ($edit_raw_html ? '_raw' : '') . '/' . $postID); $query = prepare(sprintf('SELECT * FROM ``posts_%s`` WHERE `id` = :id', $board)); $query->bindValue(':id', $postID); $query->execute() or error(db_error($query)); if (!$post = $query->fetch(PDO::FETCH_ASSOC)) error($config['error']['404']); if (isset($_POST['name'], $_POST['email'], $_POST['subject'], $_POST['body'])) { $trip = isset($_POST['remove_trip']) ? ' `trip` = NULL,' : ''; // Remove any modifiers they may have put in $_POST['body'] = remove_modifiers($_POST['body']); // Add back modifiers in the original post $modifiers = extract_modifiers($post['body_nomarkup']); foreach ($modifiers as $key => $value) { $_POST['body'] .= "$value"; } // Handle embed edits... foreach ($config['embedding'] as &$embed) { if (preg_match($embed[0], $_POST['embed'])) { $embed_link = $_POST['embed']; } } if ($edit_raw_html) $query = prepare(sprintf('UPDATE ``posts_%s`` SET `name` = :name,'. $trip .' `email` = :email, `subject` = :subject, `body` = :body, `body_nomarkup` = :body_nomarkup, `embed` = :embed `edited_at` = UNIX_TIMESTAMP(NOW()) WHERE `id` = :id', $board)); else $query = prepare(sprintf('UPDATE ``posts_%s`` SET `name` = :name,'. $trip .' `email` = :email, `subject` = :subject, `body_nomarkup` = :body, `embed` = :embed, `edited_at` = UNIX_TIMESTAMP(NOW()) WHERE `id` = :id', $board)); $query->bindValue(':id', $postID); $query->bindValue(':name', $_POST['name'] ? $_POST['name'] : $config['anonymous']); $query->bindValue(':email', $_POST['email']); $query->bindValue(':subject', $_POST['subject']); $query->bindValue(':body', $_POST['body']); if ($edit_raw_html) { $body_nomarkup = $_POST['body'] . "\n1"; $query->bindValue(':body_nomarkup', $body_nomarkup); } if (isset($embed_link)) { $query->bindValue(':embed', $embed_link); } else { $query->bindValue(':embed', NULL, PDO::PARAM_NULL); } $query->execute() or error(db_error($query)); if( $config['clean']['edits_remove_local'] || $config['clean']['edits_remove_global'] ) { $query_global = "`clean_global` = :clean"; $query_global_mod = "`clean_global_mod_id` = :mod"; $query_local = "`clean_local` = :clean"; $query_local_mod = "`clean_local_mod_id` = :mod"; if( $config['clean']['edits_remove_local'] && $config['clean']['edits_remove_global'] ) { $query = prepare("UPDATE `post_clean` SET {$query_global}, {$query_global_mod}, {$query_local}, {$query_local_mod} WHERE `board_id` = :board AND `post_id` = :post"); } else if( $config['clean']['edits_remove_global'] ) { $query = prepare("UPDATE `post_clean` SET {$query_global}, {$query_global_mod} WHERE `board_id` = :board AND `post_id` = :post"); } else { $query = prepare("UPDATE `post_clean` SET {$query_local}, {$query_local_mod} WHERE `board_id` = :board AND `post_id` = :post"); } $query->bindValue( ':clean', false ); $query->bindValue( ':mod', NULL ); $query->bindValue( ':board', $board ); $query->bindValue( ':post', $postID ); $query->execute() or error(db_error($query)); // Finally, run a query to tidy up our records. $cleanup = prepare("DELETE FROM `post_clean` WHERE `clean_local` = FALSE AND `clean_global` = FALSE"); $query->execute() or error(db_error($query)); } if ($edit_raw_html) { modLog("Edited raw HTML of post #{$postID}"); } else { modLog("Edited post #{$postID}"); rebuildPost($postID); } buildIndex(); rebuildThemes('post', $board); header('Location: ?/' . sprintf($config['board_path'], $board) . $config['dir']['res'] . sprintf($config['file_page'], $post['thread'] ? $post['thread'] : $postID) . '#' . $postID, true, $config['redirect_http']); } else { // Remove modifiers $post['body_nomarkup'] = remove_modifiers($post['body_nomarkup']); $post['body_nomarkup'] = utf8tohtml($post['body_nomarkup']); $post['body'] = utf8tohtml($post['body']); if ($config['minify_html']) { $post['body_nomarkup'] = str_replace("\n", ' ', $post['body_nomarkup']); $post['body'] = str_replace("\n", ' ', $post['body']); $post['body_nomarkup'] = str_replace("\r", '', $post['body_nomarkup']); $post['body'] = str_replace("\r", '', $post['body']); $post['body_nomarkup'] = str_replace("\t", ' ', $post['body_nomarkup']); $post['body'] = str_replace("\t", ' ', $post['body']); } $preview = new Post($post); $html = $preview->build(true); mod_page(_('Edit post'), 'mod/edit_post_form.html', array('token' => $security_token, 'board' => $board, 'raw' => $edit_raw_html, 'post' => $post, 'preview' => $html)); } } function mod_delete($board, $post) { global $config, $mod; if (!openBoard($board)) error($config['error']['noboard']); if (!hasPermission($config['mod']['delete'], $board)) error($config['error']['noaccess']); // Delete post deletePost($post); // Record the action modLog("Deleted post #{$post}"); // Rebuild board buildIndex(); // Rebuild themes rebuildThemes('post-delete', $board); // Redirect header('Location: ?/' . sprintf($config['board_path'], $board) . $config['file_index'], true, $config['redirect_http']); } function mod_deletefile($board, $post, $file) { global $config, $mod; if (!openBoard($board)) error($config['error']['noboard']); if (!hasPermission($config['mod']['deletefile'], $board)) error($config['error']['noaccess']); // Delete file deleteFile($post, TRUE, $file); // Record the action modLog("Deleted file from post #{$post}"); // Rebuild board buildIndex(); // Rebuild themes rebuildThemes('post-delete', $board); // Redirect header('Location: ?/' . sprintf($config['board_path'], $board) . $config['file_index'], true, $config['redirect_http']); } function mod_spoiler_image($board, $post, $file) { global $config, $mod; if (!openBoard($board)) error($config['error']['noboard']); if (!hasPermission($config['mod']['spoilerimage'], $board)) error($config['error']['noaccess']); // Delete file thumbnail $query = prepare(sprintf("SELECT `files`, `thread` FROM ``posts_%s`` WHERE id = :id", $board)); $query->bindValue(':id', $post, PDO::PARAM_INT); $query->execute() or error(db_error($query)); $result = $query->fetch(PDO::FETCH_ASSOC); $files = json_decode($result['files']); $size_spoiler_image = @getimagesize($config['spoiler_image']); file_unlink($config['dir']['img_root'] . $board . '/' . $config['dir']['thumb'] . $files[$file]->thumb); $files[$file]->thumb = 'spoiler'; $files[$file]->thumbwidth = $size_spoiler_image[0]; $files[$file]->thumbheight = $size_spoiler_image[1]; // Make thumbnail spoiler $query = prepare(sprintf("UPDATE ``posts_%s`` SET `files` = :files WHERE `id` = :id", $board)); $query->bindValue(':files', json_encode($files)); $query->bindValue(':id', $post, PDO::PARAM_INT); $query->execute() or error(db_error($query)); // Record the action modLog("Spoilered file from post #{$post}"); // Rebuild thread buildThread($result['thread'] ? $result['thread'] : $post); // Rebuild board buildIndex(); // Rebuild themes rebuildThemes('post-delete', $board); // Redirect header('Location: ?/' . sprintf($config['board_path'], $board) . $config['file_index'], true, $config['redirect_http']); } function mod_spoiler_images($board, $post) { global $config, $mod; if (!openBoard($board)) error($config['error']['noboard']); if (!hasPermission($config['mod']['spoilerimage'], $board)) error($config['error']['noaccess']); // Delete file thumbnails $query = prepare(sprintf("SELECT `files`, `thread` FROM ``posts_%s`` WHERE id = :id", $board)); $query->bindValue(':id', $post, PDO::PARAM_INT); $query->execute() or error(db_error($query)); $result = $query->fetch(PDO::FETCH_ASSOC); $files = json_decode($result['files']); foreach ($files as $file => $name) { $size_spoiler_image = @getimagesize($config['spoiler_image']); file_unlink($config['dir']['img_root'] . $board . '/' . $config['dir']['thumb'] . $files[$file]->thumb); $files[$file]->thumb = 'spoiler'; $files[$file]->thumbwidth = $size_spoiler_image[0]; $files[$file]->thumbheight = $size_spoiler_image[1]; }; // Make thumbnail spoiler $query = prepare(sprintf("UPDATE ``posts_%s`` SET `files` = :files WHERE `id` = :id", $board)); $query->bindValue(':files', json_encode($files)); $query->bindValue(':id', $post, PDO::PARAM_INT); $query->execute() or error(db_error($query)); // Record the action modLog("Spoilered file from post #{$post}"); // Rebuild thread buildThread($result['thread'] ? $result['thread'] : $post); // Rebuild board buildIndex(); // Rebuild themes rebuildThemes('post-delete', $board); // Redirect header('Location: ?/' . sprintf($config['board_path'], $board) . $config['file_index'], true, $config['redirect_http']); } function mod_deletebyip($boardName, $post, $global = false) { global $config, $mod, $board; $global = (bool)$global; if (!openBoard($boardName)) error($config['error']['noboard']); if (!$global && !hasPermission($config['mod']['deletebyip'], $boardName)) error($config['error']['noaccess']); if ($global && !hasPermission($config['mod']['deletebyip_global'], $boardName)) error($config['error']['noaccess']); // Find IP address $query = prepare(sprintf('SELECT `ip` FROM ``posts_%s`` WHERE `id` = :id', $boardName)); $query->bindValue(':id', $post); $query->execute() or error(db_error($query)); if (!$ip = $query->fetchColumn()) error($config['error']['invalidpost']); $boards = $global ? listBoards() : array(array('uri' => $boardName)); $query = ''; foreach ($boards as $_board) { $query .= sprintf("SELECT `thread`, `id`, '%s' AS `board` FROM ``posts_%s`` WHERE `ip` = :ip UNION ALL ", $_board['uri'], $_board['uri']); } $query = preg_replace('/UNION ALL $/', '', $query); $query = prepare($query); $query->bindValue(':ip', $ip); $query->execute() or error(db_error($query)); if ($query->rowCount() < 1) error($config['error']['invalidpost']); @set_time_limit($config['mod']['rebuild_timelimit']); $threads_to_rebuild = array(); $threads_deleted = array(); while ($post = $query->fetch(PDO::FETCH_ASSOC)) { openBoard($post['board']); deletePost($post['id'], false, false); rebuildThemes('post-delete', $board['uri']); if ($post['thread']) $threads_to_rebuild[$post['board']][$post['thread']] = true; else $threads_deleted[$post['board']][$post['id']] = true; } foreach ($threads_to_rebuild as $_board => $_threads) { openBoard($_board); foreach ($_threads as $_thread => $_dummy) { if ($_dummy && !isset($threads_deleted[$_board][$_thread])) buildThread($_thread); } buildIndex(); } if ($global) { $board = false; } // Record the action modLog("Deleted all posts by IP address: $ip"); // Redirect header('Location: ?/' . sprintf($config['board_path'], $boardName) . $config['file_index'], true, $config['redirect_http']); } function mod_user($uid) { global $config, $mod; if (!hasPermission($config['mod']['editusers']) && !(hasPermission($config['mod']['edit_profile']) && $uid == $mod['id'])) error($config['error']['noaccess']); if (in_array($mod['boards'][0], array('infinity', 'z'))) error('This board has profile changing disabled.'); $query = prepare('SELECT * FROM ``mods`` WHERE `id` = :id'); $query->bindValue(':id', $uid); $query->execute() or error(db_error($query)); if (!$user = $query->fetch(PDO::FETCH_ASSOC)) error($config['error']['404']); if (hasPermission($config['mod']['editusers']) && isset($_POST['username'], $_POST['password'])) { if (isset($_POST['allboards'])) { $boards = array('*'); } else { $_boards = listBoards(); foreach ($_boards as &$board) { $board = $board['uri']; } $boards = array(); foreach ($_POST as $name => $value) { if (preg_match('/^board_(' . $config['board_regex'] . ')$/u', $name, $matches) && in_array($matches[1], $_boards)) $boards[] = $matches[1]; } } if (isset($_POST['delete'])) { if (!hasPermission($config['mod']['deleteusers'])) error($config['error']['noaccess']); $query = prepare('DELETE FROM ``mods`` WHERE `id` = :id'); $query->bindValue(':id', $uid); $query->execute() or error(db_error($query)); modLog('Deleted user ' . utf8tohtml($user['username']) . ' (#' . $user['id'] . ')'); header('Location: ?/users', true, $config['redirect_http']); return; } if ($_POST['username'] == '') error(sprintf($config['error']['required'], 'username')); $query = prepare('UPDATE ``mods`` SET `username` = :username, `boards` = :boards WHERE `id` = :id'); $query->bindValue(':id', $uid); $query->bindValue(':username', $_POST['username']); $query->bindValue(':boards', implode(',', $boards)); $query->execute() or error(db_error($query)); if ($user['username'] !== $_POST['username']) { // account was renamed modLog('Renamed user "' . utf8tohtml($user['username']) . '" (#' . $user['id'] . ') to "' . utf8tohtml($_POST['username']) . '"'); } if ($_POST['password'] != '') { $salt = generate_salt(); $password = hash('sha256', $salt . sha1($_POST['password'])); $query = prepare('UPDATE ``mods`` SET `password` = :password, `salt` = :salt WHERE `id` = :id'); $query->bindValue(':id', $uid); $query->bindValue(':password', $password); $query->bindValue(':salt', $salt); $query->execute() or error(db_error($query)); modLog('Changed password for ' . utf8tohtml($_POST['username']) . ' (#' . $user['id'] . ')'); if ($uid == $mod['id']) { login($_POST['username'], $_POST['password']); setCookies(); } } if (hasPermission($config['mod']['manageusers'])) header('Location: ?/users', true, $config['redirect_http']); else header('Location: ?/', true, $config['redirect_http']); return; } if (hasPermission($config['mod']['edit_profile']) && $uid == $mod['id']) { if (isset($_POST['password']) && $_POST['password'] != '') { $salt = generate_salt(); $password = hash('sha256', $salt . sha1($_POST['password'])); $query = prepare('UPDATE ``mods`` SET `password` = :password, `salt` = :salt WHERE `id` = :id'); $query->bindValue(':id', $uid); $query->bindValue(':password', $password); $query->bindValue(':salt', $salt); $query->execute() or error(db_error($query)); modLog('Changed own password'); login($user['username'], $_POST['password']); setCookies(); } if (isset($_POST['username']) && $user['username'] !== $_POST['username']) { if ($_POST['username'] == '') error(sprintf($config['error']['required'], 'username')); if (!preg_match('/^[a-zA-Z0-9._]{1,30}$/', $_POST['username'])) error(_('Invalid username')); $query = prepare('SELECT `username` FROM ``mods``'); $query->execute() or error(db_error($query)); $users = $query->fetchAll(PDO::FETCH_ASSOC); foreach ($users as $i => $v) { if (strtolower($_POST['username']) == strtolower($v['username'])) { error(_('Refusing to change your username because another user is already using it.')); } } $query = prepare('UPDATE ``mods`` SET `username` = :username WHERE `id` = :id'); $query->bindValue(':id', $uid); $query->bindValue(':username', $_POST['username']); $query->execute() or error(db_error($query)); modLog('Renamed user "' . utf8tohtml($user['username']) . '" (#' . $user['id'] . ') to "' . utf8tohtml($_POST['username']) . '"'); } if (isset($_POST['email']) && $user['email'] !== $_POST['email'] && (empty($_POST['email']) || filter_var($_POST['email'], FILTER_VALIDATE_EMAIL))) { // account was renamed $query = prepare('UPDATE ``mods`` SET `email` = :email WHERE `id` = :id'); $query->bindValue(':id', $uid); $query->bindValue(':email', $_POST['email']); $query->execute() or error(db_error($query)); modLog('Changed user\'s email "' . utf8tohtml($user['email']) . '" (#' . $user['id'] . ') to "' . utf8tohtml($_POST['email']) . '"'); } if ($_SERVER['REQUEST_METHOD'] === 'POST') { if (hasPermission($config['mod']['manageusers'])) header('Location: ?/users', true, $config['redirect_http']); else header('Location: ?/', true, $config['redirect_http']); return; } } if (hasPermission($config['mod']['modlog'])) { $query = prepare('SELECT * FROM ``modlogs`` WHERE `mod` = :id ORDER BY `time` DESC LIMIT 5'); $query->bindValue(':id', $uid); $query->execute() or error(db_error($query)); $log = $query->fetchAll(PDO::FETCH_ASSOC); } else { $log = array(); } if ($mod['type'] >= ADMIN){ $boards = listBoards(); } else { $boards2 = explode(',', $user['boards']); foreach ($boards2 as $string){ $boards[] = array("uri"=>$string, "title" => _("My board")); } } $user['boards'] = explode(',', $user['boards']); mod_page(_('Edit user profile'), 'mod/user.html', array( 'user' => $user, 'logs' => $log, 'boards' => $boards, 'token' => make_secure_link_token('users/' . $user['id']) )); } function mod_user_new() { global $pdo, $config; if (!hasPermission($config['mod']['createusers'])) error($config['error']['noaccess']); if (isset($_POST['username'], $_POST['password'], $_POST['type'])) { if ($_POST['username'] == '') error(sprintf($config['error']['required'], 'username')); if ($_POST['password'] == '') error(sprintf($config['error']['required'], 'password')); if (isset($_POST['allboards'])) { $boards = array('*'); } else { $_boards = listBoards(); foreach ($_boards as &$board) { $board = $board['uri']; } $boards = array(); foreach ($_POST as $name => $value) { if (preg_match('/^board_(' . $config['board_regex'] . ')$/u', $name, $matches) && in_array($matches[1], $_boards)) $boards[] = $matches[1]; } } $type = (int)$_POST['type']; if (!isset($config['mod']['groups'][$type]) || $type == DISABLED) error(sprintf($config['error']['invalidfield'], 'type')); $salt = generate_salt(); $password = hash('sha256', $salt . sha1($_POST['password'])); $query = prepare('INSERT INTO ``mods`` VALUES (NULL, :username, :password, :salt, :type, :boards)'); $query->bindValue(':username', $_POST['username']); $query->bindValue(':password', $password); $query->bindValue(':salt', $salt); $query->bindValue(':type', $type); $query->bindValue(':boards', implode(',', $boards)); $query->execute() or error(db_error($query)); $userID = $pdo->lastInsertId(); modLog('Created a new user: ' . utf8tohtml($_POST['username']) . ' (#' . $userID . ')'); header('Location: ?/users', true, $config['redirect_http']); return; } mod_page(_('New user'), 'mod/user.html', array('new' => true, 'boards' => listBoards(), 'token' => make_secure_link_token('users/new'))); } function mod_users() { global $config; if (!hasPermission($config['mod']['manageusers'])) error($config['error']['noaccess']); $query = query("SELECT ``m``.`id`, ``m``.`username`, ``m``.`boards`, ``m``.`type`, ``m``.`email`, ``ml``.`time` last, ``ml``.`text` action FROM ``mods`` AS m LEFT JOIN ( SELECT ml1.* FROM ``modlogs`` AS ml1 JOIN ( SELECT `mod`, MAX(time) AS time FROM ``modlogs`` GROUP BY `mod` ) AS ml2 USING (`mod`, time) ) AS ml ON m.id = ml.`mod` GROUP BY ``m``.`id` ORDER BY ``m``.`type` DESC;") or error(db_error()); $users = $query->fetchAll(PDO::FETCH_ASSOC); foreach ($users as &$user) { $user['promote_token'] = make_secure_link_token("users/{$user['id']}/promote"); $user['demote_token'] = make_secure_link_token("users/{$user['id']}/demote"); } mod_page(sprintf('%s (%d)', _('Manage users'), count($users)), 'mod/users.html', array('users' => $users)); } function mod_user_promote($uid, $action) { global $config; if (!hasPermission($config['mod']['promoteusers'])) error($config['error']['noaccess']); $query = prepare("SELECT `type`, `username` FROM ``mods`` WHERE `id` = :id"); $query->bindValue(':id', $uid); $query->execute() or error(db_error($query)); if (!$mod = $query->fetch(PDO::FETCH_ASSOC)) error($config['error']['404']); $new_group = false; $groups = $config['mod']['groups']; if ($action == 'demote') $groups = array_reverse($groups, true); foreach ($groups as $group_value => $group_name) { if ($action == 'promote' && $group_value > $mod['type']) { $new_group = $group_value; break; } elseif ($action == 'demote' && $group_value < $mod['type']) { $new_group = $group_value; break; } } if ($new_group === false || $new_group == DISABLED) error(_('Impossible to promote/demote user.')); $query = prepare("UPDATE ``mods`` SET `type` = :group_value WHERE `id` = :id"); $query->bindValue(':id', $uid); $query->bindValue(':group_value', $new_group); $query->execute() or error(db_error($query)); modLog(($action == 'promote' ? 'Promoted' : 'Demoted') . ' user "' . utf8tohtml($mod['username']) . '" to ' . $config['mod']['groups'][$new_group]); header('Location: ?/users', true, $config['redirect_http']); } function mod_pm($id, $reply = false) { global $mod, $config; if ($reply && !hasPermission($config['mod']['create_pm'])) error($config['error']['noaccess']); $query = prepare("SELECT ``mods``.`username`, `mods_to`.`username` AS `to_username`, ``pms``.* FROM ``pms`` LEFT JOIN ``mods`` ON ``mods``.`id` = `sender` LEFT JOIN ``mods`` AS `mods_to` ON `mods_to`.`id` = `to` WHERE ``pms``.`id` = :id"); $query->bindValue(':id', $id); $query->execute() or error(db_error($query)); if ((!$pm = $query->fetch(PDO::FETCH_ASSOC)) || ($pm['to'] != $mod['id'] && !hasPermission($config['mod']['master_pm']))) error($config['error']['404']); if (isset($_POST['delete'])) { $query = prepare("DELETE FROM ``pms`` WHERE `id` = :id"); $query->bindValue(':id', $id); $query->execute() or error(db_error($query)); if ($config['cache']['enabled']) { cache::delete('pm_unread_' . $mod['id']); cache::delete('pm_unreadcount_' . $mod['id']); } header('Location: ?/', true, $config['redirect_http']); return; } if ($pm['unread'] && $pm['to'] == $mod['id']) { $query = prepare("UPDATE ``pms`` SET `unread` = 0 WHERE `id` = :id"); $query->bindValue(':id', $id); $query->execute() or error(db_error($query)); if ($config['cache']['enabled']) { cache::delete('pm_unread_' . $mod['id']); cache::delete('pm_unreadcount_' . $mod['id']); } modLog('Read a PM'); } if ($reply) { if (!$pm['to_username']) error($config['error']['404']); // deleted? mod_page(sprintf('%s %s', _('New PM for'), $pm['to_username']), 'mod/new_pm.html', array( 'username' => $pm['username'], 'id' => $pm['sender'], 'message' => quote($pm['message']), 'token' => make_secure_link_token('new_PM/' . $pm['username']) )); } else { mod_page(sprintf('%s – #%d', _('Private message'), $id), 'mod/pm.html', $pm); } } function mod_inbox() { global $config, $mod; $query = prepare('SELECT `unread`,``pms``.`id`, `time`, `sender`, `to`, `message`, `username` FROM ``pms`` LEFT JOIN ``mods`` ON ``mods``.`id` = `sender` WHERE `to` = :mod ORDER BY `unread` DESC, `time` DESC'); $query->bindValue(':mod', $mod['id']); $query->execute() or error(db_error($query)); $messages = $query->fetchAll(PDO::FETCH_ASSOC); $query = prepare('SELECT COUNT(*) FROM ``pms`` WHERE `to` = :mod AND `unread` = 1'); $query->bindValue(':mod', $mod['id']); $query->execute() or error(db_error($query)); $unread = $query->fetchColumn(); foreach ($messages as &$message) { $message['snippet'] = pm_snippet($message['message']); } mod_page(sprintf('%s (%s)', _('PM inbox'), count($messages) > 0 ? $unread . ' unread' : 'empty'), 'mod/inbox.html', array( 'messages' => $messages, 'unread' => $unread )); } function mod_new_pm($username) { global $config, $mod; if (!hasPermission($config['mod']['create_pm'])) error($config['error']['noaccess']); $query = prepare("SELECT `id` FROM ``mods`` WHERE `username` = :username"); $query->bindValue(':username', $username); $query->execute() or error(db_error($query)); if (!$id = $query->fetchColumn()) { // Old style ?/PM: by user ID $query = prepare("SELECT `username` FROM ``mods`` WHERE `id` = :username"); $query->bindValue(':username', $username); $query->execute() or error(db_error($query)); if ($username = $query->fetchColumn()) header('Location: ?/new_PM/' . $username, true, $config['redirect_http']); else error($config['error']['404']); } if (isset($_POST['message'])) { $_POST['message'] = escape_markup_modifiers($_POST['message']); markup($_POST['message']); $query = prepare("INSERT INTO ``pms`` VALUES (NULL, :me, :id, :message, :time, 1)"); $query->bindValue(':me', $mod['id']); $query->bindValue(':id', $id); $query->bindValue(':message', $_POST['message']); $query->bindValue(':time', time()); $query->execute() or error(db_error($query)); if ($config['cache']['enabled']) { cache::delete('pm_unread_' . $id); cache::delete('pm_unreadcount_' . $id); } modLog('Sent a PM to ' . utf8tohtml($username)); header('Location: ?/', true, $config['redirect_http']); } mod_page(sprintf('%s %s', _('New PM for'), $username), 'mod/new_pm.html', array( 'username' => $username, 'id' => $id, 'token' => make_secure_link_token('new_PM/' . $username) )); } function mod_rebuild() { global $config, $twig; if (!hasPermission($config['mod']['rebuild'])) error($config['error']['noaccess']); if (isset($_POST['rebuild'])) { @set_time_limit($config['mod']['rebuild_timelimit']); $log = array(); $boards = listBoards(); $rebuilt_scripts = array(); if (isset($_POST['rebuild_cache'])) { if ($config['cache']['enabled']) { $log[] = 'Flushing cache'; Cache::flush(); } $log[] = 'Clearing template cache'; load_twig(); $twig->clearCacheFiles(); } if (isset($_POST['rebuild_themes'])) { $log[] = 'Regenerating theme files'; rebuildThemes('all'); } if (isset($_POST['rebuild_javascript'])) { $log[] = 'Rebuilding ' . $config['file_script'] . ''; buildJavascript(); $rebuilt_scripts[] = $config['file_script']; } foreach ($boards as $board) { if (!(isset($_POST['boards_all']) || isset($_POST['board_' . $board['uri']]))) continue; openBoard($board['uri']); $config['try_smarter'] = false; if (isset($_POST['rebuild_index'])) { buildIndex(); $log[] = '' . sprintf($config['board_abbreviation'], $board['uri']) . ': Creating index pages'; } if (isset($_POST['rebuild_javascript']) && !in_array($config['file_script'], $rebuilt_scripts)) { $log[] = '' . sprintf($config['board_abbreviation'], $board['uri']) . ': Rebuilding ' . $config['file_script'] . ''; buildJavascript(); $rebuilt_scripts[] = $config['file_script']; } if (isset($_POST['rebuild_thread'])) { $query = query(sprintf("SELECT `id` FROM ``posts_%s`` WHERE `thread` IS NULL", $board['uri'])) or error(db_error()); while ($post = $query->fetch(PDO::FETCH_ASSOC)) { $log[] = '' . sprintf($config['board_abbreviation'], $board['uri']) . ': Rebuilding thread #' . $post['id']; buildThread($post['id']); } } } mod_page(_('Rebuild'), 'mod/rebuilt.html', array('logs' => $log)); return; } mod_page(_('Rebuild'), 'mod/rebuild.html', array( 'boards' => listBoards(), 'token' => make_secure_link_token('rebuild') )); } function mod_reports() { global $config, $mod; // Parse arguments. $urlArgs = func_get_args(); $global = in_array( "global", $urlArgs ); $json = in_array( "json", $urlArgs ); if( !hasPermission($config['mod']['reports']) ) { error($config['error']['noaccess']); } if( ($mod['type'] < GLOBALVOLUNTEER) and $global) { error($config['error']['noaccess']); } // Limit reports to ONLY those in our scope. $report_scope = $global ? "global" : "local"; // Get REPORTS. $query = prepare("SELECT * FROM ``reports`` WHERE " . (($mod["type"] < GLOBALVOLUNTEER) ? "board = :board AND" : "") . " ``".($global ? "global" : "local")."`` = TRUE LIMIT :limit"); // Limit reports by board if the moderator is local. if( $mod['type'] < GLOBALVOLUNTEER ) { $query->bindValue(':board', $mod['boards'][0]); } // Limit by config ceiling. $query->bindValue( ':limit', $config['mod']['recent_reports'], PDO::PARAM_INT ); $query->execute() or error(db_error($query)); $reports = $query->fetchAll(PDO::FETCH_ASSOC); // Cut off here if we don't have any reports. $reportCount = 0; $reportHTML = ''; if( count( $reports ) > 0 ) { // Build queries to fetch content. $report_queries = array(); foreach ($reports as $report) { if (!isset($report_queries[$report['board']])) $report_queries[$report['board']] = array(); $report_queries[$report['board']][] = $report['post']; } // Get reported CONTENT. $report_posts = array(); foreach ($report_queries as $board => $posts) { $report_posts[$board] = array(); $query = query(sprintf('SELECT * FROM ``posts_%s`` WHERE `id` = ' . implode(' OR `id` = ', $posts), $board)) or error(db_error()); while ($post = $query->fetch(PDO::FETCH_ASSOC)) { $report_posts[$board][$post['id']] = $post; } } // Develop an associative array of posts to reports. $report_index = array(); foreach( $reports as &$report ) { // Delete reports which are for removed content. if( !isset( $report_posts[ $report['board'] ][ $report['post'] ] ) ) { // Invalid report (post has since been deleted) $query = prepare("DELETE FROM ``reports`` WHERE `post` = :id AND `board` = :board"); $query->bindValue(':id', $report['post'], PDO::PARAM_INT); $query->bindValue(':board', $report['board']); $query->execute() or error(db_error($query)); continue; } // Build a unique ID. $content_key = "{$report['board']}.{$report['post']}"; // Create a dummy array if it doesn't already exist. if( !isset( $report_index[ $content_key ] ) ) { $report_index[ $content_key ] = array( "board_id" => $report['board'], "post_id" => $report['post'], "content" => $report_posts[ $report['board'] ][ $report['post'] ], "reports" => array(), ); } // Add the report to the list of reports. $report_index[ $content_key ]['reports'][ $report['id'] ] = $report; // Increment the total report count. ++$reportCount; } // Only continue if we have something to do. // If there are no valid reports left, we're done. if( $reportCount > 0 && !$json ) { // Sort this report index by number of reports, desc. usort( $report_index, function( $a, $b ) { $ra = count( $a['reports'] ); $rb = count( $b['reports'] ); if( $ra < $rb ) { return 1; } else if( $rb > $ra ) { return -1; } else { return 0; } } ); // Loop through the custom index. foreach( $report_index as &$report_item ) { $content = $report_item['content']; // Load board content. openBoard($report_item['board_id']); // Load the reported content. if( !$content['thread'] ) { // Still need to fix this: $po = new Thread($content, '?/', $mod, false); } else { $po = new Post($content, '?/', $mod); } // Fetch clean status. $po->getClean(true); $clean = $po->clean; // Add each report's template to this container. $report_html = ""; $reports_can_demote = false; $reports_can_promote = false; $content_reports = 0; foreach( $report_item['reports'] as $report ) { $uri_report_base = "reports/" . ($global ? "global/" : "" ) . $report['id']; $report_html .= Element('mod/report.html', array( 'report' => $report, 'config' => $config, 'mod' => $mod, 'global' => $global, 'clean' => $clean, 'uri_dismiss' => "?/{$uri_report_base}/dismiss", 'uri_ip' => "?/{$uri_report_base}/dismissall", 'uri_demote' => "?/{$uri_report_base}/demote", 'uri_promote' => "?/{$uri_report_base}/promote", 'token_dismiss' => make_secure_link_token( $uri_report_base . '/dismiss' ), 'token_ip' => make_secure_link_token( $uri_report_base . '/dismissall' ), 'token_demote' => make_secure_link_token( $uri_report_base . '/demote' ), 'token_promote' => make_secure_link_token( $uri_report_base . '/promote' ), )); // Determines if we can "Demote All" / "Promote All" // This logic only needs one instance of a demotable or promotable report to work. // DEMOTE can occur when we're global and the report has a 1 for local (meaning locally, it's not dismissed) // PROMOTE can occur when we're local and the report has a 0 for global (meaning it's not global). if( $global && $report['local'] == "1" ) { $reports_can_demote = true; } else if( !$global && $report['global'] != "1" ) { $reports_can_promote = true; } ++$content_reports; } // Build the ">>>/b/ thread reported 3 times" title. $report_title = sprintf( _('>>>/%s/ %s reported %d time(s).'), "?/{$report_item['board_id']}/res/" . ( $content['thread'] ?: $content['id'] ) . ".html#{$content['thread']}", $report_item['board_id'], _( $content['thread'] ? "reply" : "thread" ), $content_reports ); // Figure out some stuff we need for the page. $reports_can_demote = ( $clean['clean_local'] ? false : $reports_can_demote ); $reports_can_promote = ( $clean['clean_global'] ? false : $reports_can_promote ); $uri_content_base = "reports/" . ($global ? "global/" : "" ) . "content/"; $uri_clean_base = "reports/" . ($global ? "global/" : "" ) . "{$report_item['board_id']}/clean/{$content['id']}"; // Build the actions page. $content_html = Element('mod/report_content.html', array( 'reports_html' => $report_html, 'reports_can_demote' => $reports_can_demote, 'reports_can_promote' => $reports_can_promote, 'report_count' => $content_reports, 'report_title' => $report_title, 'content_html' => $po->build(true), 'content_board' => $report_item['board_id'], 'content' => (array) $content, 'clean' => $clean, 'uri_content_demote' => "?/{$uri_content_base}{$report_item['board_id']}/{$content['id']}/demote", 'uri_content_promote' => "?/{$uri_content_base}{$report_item['board_id']}/{$content['id']}/promote", 'uri_content_dismiss' => "?/{$uri_content_base}{$report_item['board_id']}/{$content['id']}/dismiss", 'token_content_demote' => make_secure_link_token( "{$uri_content_base}{$report_item['board_id']}/{$content['id']}/demote" ), 'token_content_promote' => make_secure_link_token( "{$uri_content_base}{$report_item['board_id']}/{$content['id']}/promote" ), 'token_content_dismiss' => make_secure_link_token( "{$uri_content_base}{$report_item['board_id']}/{$content['id']}/dismiss" ), 'uri_clean' => "?/{$uri_clean_base}/local", 'uri_clean_global' => "?/{$uri_clean_base}/global", 'uri_clean_both' => "?/{$uri_clean_base}/global+local", 'token_clean' => make_secure_link_token( $uri_clean_base . '/local' ), 'token_clean_global' => make_secure_link_token( $uri_clean_base . '/global' ), 'token_clean_both' => make_secure_link_token( $uri_clean_base . '/global+local' ), 'global' => $global, 'config' => $config, 'mod' => $mod, )); $reportHTML .= $content_html; } } if ( $reportCount > 0 && $json ) { array_walk($reports, function(&$v, $k, $ud) { $global = $ud['global']; $report_posts = $ud['report_posts']; $board = ($v['board'] ? $v['board'] : NULL); if (isset($v['ip']) && !$global) { $v['ip'] = less_ip($v['ip'], ($board?$board:'')); } if (isset($report_posts[ $v['board'] ][ $v['post'] ])) { $post_content = $report_posts[ $v['board'] ][ $v['post'] ]; unset($post_content['password']); if (!$global) { $post_content['ip'] = less_ip($post_content['ip'], ($board?$board:'')); } $v['post_content'] = $post_content; } }, array('global' => $global, 'report_posts' => $report_posts)); } } $pageArgs = array( 'count' => $reportCount, 'reports' => $reportHTML, 'global' => $global, ); if ($json) { header('Content-Type: application/json'); echo json_encode($reports); } else { mod_page( sprintf('%s (%d)', _( ( $global ? 'Global report queue' : 'Report queue' ) ), $reportCount), 'mod/reports.html', $pageArgs ); } } function mod_report_dismiss() { global $config, $mod; // Parse arguments. $arguments = func_get_args(); $global = in_array( "global", $arguments ); $content = in_array( "content", $arguments ); if( ($mod['type'] < GLOBALVOLUNTEER ) and $global ) { error($config['error']['noaccess']); } if( $content ) { $board = @$arguments[2]; $post = @$arguments[3]; if( !hasPermission($config['mod']['report_dismiss_content'], $board) ) { error($config['error']['noaccess']); } if( $board != "" && $post != "" ) { $query = prepare("SELECT `id` FROM `reports` WHERE `board` = :board AND `post` = :post"); $query->bindValue(':board', $board); $query->bindValue(':post', $post); $query->execute() or error(db_error($query)); if( count( $reports = $query->fetchAll(PDO::FETCH_ASSOC) ) > 0 ) { $report_ids = array(); foreach( $reports as $report ) { $report_ids[ $report['id'] ] = $report['id']; } if( $global ) { $scope = "``global`` = FALSE AND ``local`` = FALSE"; } else { $scope = "``local`` = FALSE"; } $query = prepare("UPDATE ``reports`` SET {$scope} WHERE `id` IN (".implode(',', array_map('intval', $report_ids)).")"); $query->execute() or error(db_error($query)); // Cleanup - Remove reports that have been completely dismissed. $query = prepare("DELETE FROM `reports` WHERE `local` = FALSE AND `global` = FALSE"); $query->execute() or error(db_error($query)); modLog("Dismissed " . count($report_ids) . " local report(s) for post #{$post}", $board); } else { error($config['error']['404']); } } else { error($config['error']['404']); } } else { $report = @$arguments[1]; $all = in_array( "all", $arguments ); if( $report != "" ) { $query = prepare("SELECT `post`, `board`, `ip` FROM ``reports`` WHERE `id` = :id"); $query->bindValue(':id', $report); $query->execute() or error(db_error($query)); if ($reportobj = $query->fetch(PDO::FETCH_ASSOC)) { $ip = $reportobj['ip']; $board = $reportobj['board']; $post = $reportobj['post']; if( !$all && !hasPermission($config['mod']['report_dismiss'], $board) ) { error($config['error']['noaccess']); } if( $all && !hasPermission($config['mod']['report_dismiss_ip'], $board) ) { error($config['error']['noaccess']); } // Determine scope (local and global or just local) based on /global/ being in URI. if( $global ) { $scope = "`global` = FALSE"; $boards = ""; } else { $scope = "`local` = FALSE"; $boards = "AND `board` = '{$board}'"; } // Prepare query. // We don't delete reports, only modify scope. if( $all ) { $query = prepare("UPDATE ``reports`` SET {$scope} WHERE `ip` = :ip {$boards}"); $query->bindValue(':ip', $ip); } else { $query = prepare("UPDATE ``reports`` SET {$scope} WHERE `id` = :id {$boards}"); $query->bindValue(':id', $report); } $query->execute() or error(db_error($query)); // Cleanup - Remove reports that have been completely dismissed. $query = prepare("DELETE FROM `reports` WHERE `local` = FALSE AND `global` = FALSE"); $query->execute() or error(db_error($query)); if( $all ) { modLog("Dismissed all reports by {$ip}"); } else { modLog("Dismissed a report for post #{$post}", $board); } } else { error($config['error']['404']); } } else { error($config['error']['404']); } } if( $global ) { header('Location: ?/reports/global', true, $config['redirect_http']); } else { header('Location: ?/reports', true, $config['redirect_http']); } } function mod_report_demote() { global $config, $mod; if( $mod['type'] < GLOBALVOLUNTEER ) { error($config['error']['noaccess']); } // Parse arguments. $arguments = func_get_args(); $content = in_array( "content", $arguments ); if( $content ) { $board = @$arguments[2]; $post = @$arguments[3]; if( !hasPermission($config['mod']['report_demote'], $board) ) { error($config['error']['noaccess']); } if( $board != "" && $post != "" ) { $query = prepare("SELECT `id` FROM `reports` WHERE `global` = TRUE AND `board` = :board AND `post` = :post"); $query->bindValue(':board', $board); $query->bindValue(':post', $post); $query->execute() or error(db_error($query)); if( count( $reports = $query->fetchAll(PDO::FETCH_ASSOC) ) > 0 ) { $report_ids = array(); foreach( $reports as $report ) { $report_ids[ $report['id'] ] = $report['id']; } $query = prepare("UPDATE ``reports`` SET ``global`` = FALSE WHERE `id` IN (".implode(',', array_map('intval', $report_ids)).")"); $query->execute() or error(db_error($query)); modLog("Demoted " . count($report_ids) . " global report(s) for post #{$post}", $board); } else { error($config['error']['404']); } } else { error($config['error']['404']); } } else { $report = @$arguments[1]; if( $report != "" ) { $query = prepare("SELECT `post`, `board`, `ip` FROM ``reports`` WHERE `id` = :id AND ``global`` = TRUE"); $query->bindValue(':id', $report); $query->execute() or error(db_error($query)); if( $reportobj = $query->fetch(PDO::FETCH_ASSOC) ) { $ip = $reportobj['ip']; $board = $reportobj['board']; $post = $reportobj['post']; if( !hasPermission($config['mod']['report_demote'], $board) ) { error($config['error']['noaccess']); } $query = prepare("UPDATE ``reports`` SET ``global`` = FALSE WHERE `id` = :id"); $query->bindValue(':id', $report); $query->execute() or error(db_error($query)); modLog("Demoted a global report for post #{$report}", $board); } else { error($config['error']['404']); } } else { error($config['error']['404']); } } header('Location: ?/reports/global', true, $config['redirect_http']); } function mod_report_promote() { global $config, $mod; // Parse arguments. $arguments = func_get_args(); $content = in_array( "content", $arguments ); if( $content ) { $board = @$arguments[2]; $post = @$arguments[3]; if( !hasPermission($config['mod']['report_promote'], $board) ) { error($config['error']['noaccess']); } if( $board != "" && $post != "" ) { $query = prepare("SELECT `id` FROM `reports` WHERE `global` = FALSE AND `board` = :board AND `post` = :post"); $query->bindValue(':board', $board); $query->bindValue(':post', $post); $query->execute() or error(db_error($query)); if( count( $reports = $query->fetchAll(PDO::FETCH_ASSOC) ) > 0 ) { $report_ids = array(); foreach( $reports as $report ) { $report_ids[ $report['id'] ] = $report['id']; } $query = prepare("UPDATE ``reports`` SET ``global`` = TRUE WHERE `id` IN (".implode(',', array_map('intval', $report_ids)).")"); $query->execute() or error(db_error($query)); modLog("Promoted " . count($report_ids) . " local report(s) for post #{$post}", $board); } else { error($config['error']['404']); } } else { error($config['error']['404']); } } else { $report = @$arguments[1]; if( $report != "" ) { $query = prepare("SELECT `post`, `board`, `ip` FROM ``reports`` WHERE `id` = :id AND ``global`` = FALSE"); $query->bindValue(':id', $report); $query->execute() or error(db_error($query)); if ($reportobj = $query->fetch(PDO::FETCH_ASSOC)) { $ip = $reportobj['ip']; $board = $reportobj['board']; $post = $reportobj['post']; if( !hasPermission($config['mod']['report_promote'], $board) ) { error($config['error']['noaccess']); } $query = prepare("UPDATE ``reports`` SET ``global`` = TRUE WHERE `id` = :id"); $query->bindValue(':id', $report); $query->execute() or error(db_error($query)); modLog("Promoted a local report for post #{$report}", $board); } else { error($config['error']['404']); } } else { error($config['error']['404']); } } header('Location: ?/reports', true, $config['redirect_http']); } function mod_recent_posts($lim) { global $config, $mod, $pdo; if (!hasPermission($config['mod']['recent'])) error($config['error']['noaccess']); $limit = (is_numeric($lim))? $lim : 25; $last_time = (isset($_GET['last']) && is_numeric($_GET['last'])) ? $_GET['last'] : 0; if ($limit > 100) $limit = 100; $mod_boards = array(); $boards = listBoards(); //if not all boards if ($mod['boards'][0]!='*') { foreach ($boards as $board) { if (in_array($board['uri'], $mod['boards'])) $mod_boards[] = $board; } } else { $mod_boards = $boards; } // Manually build an SQL query $query = 'SELECT * FROM ('; foreach ($mod_boards as $board) { $query .= sprintf('SELECT *, %s AS `board` FROM ``posts_%s`` UNION ALL ', $pdo->quote($board['uri']), $board['uri']); } // Remove the last "UNION ALL" seperator and complete the query $query = preg_replace('/UNION ALL $/', ') AS `all_posts` WHERE (`time` < :last_time OR NOT :last_time) ORDER BY `time` DESC LIMIT ' . $limit, $query); $query = prepare($query); $query->bindValue(':last_time', $last_time); $query->execute() or error(db_error($query)); $posts = $query->fetchAll(PDO::FETCH_ASSOC); foreach ($posts as &$post) { openBoard($post['board']); if (!$post['thread']) { // Still need to fix this: $po = new Thread($post, '?/', $mod, false); $post['built'] = $po->build(true); } else { $po = new Post($post, '?/', $mod); $post['built'] = $po->build(true); } $last_time = $post['time']; } echo mod_page(_('Recent posts'), 'mod/recent_posts.html', array( 'posts' => $posts, 'limit' => $limit, 'last_time' => $last_time ) ); } function mod_report_clean( $global_reports, $board, $unclean, $post, $global, $local ) { global $config, $mod; if( !openBoard($board) ) { error($config['error']['noboard']); } $query_global = ""; $query_global_mod = ""; if( $global ) { if( !hasPermission($config['mod']['clean_global'], $board) ) { error($config['error']['noaccess']); } $query_global = "`clean_global` = :clean"; $query_global_mod = "`clean_global_mod_id` = :mod"; } $query_local = ""; $query_local_mod = ""; if( $local ) { if( !hasPermission($config['mod']['clean'], $board) ) { error($config['error']['noaccess']); } $query_local = "`clean_local` = :clean"; $query_local_mod = "`clean_local_mod_id` = :mod"; } // Marking this post as "Clean" (report immune?) if( !$unclean ) { // Attempt to find a `post_clean` row for this content. $query = prepare("SELECT * FROM `post_clean` WHERE `board_id` = :board AND `post_id` = :post"); $query->bindValue( ':board', $board ); $query->bindValue( ':post', $post ); $query->execute() or error(db_error($query)); // If the $clean object doesn't exist we need to insert a row for this post. if( !($cleanRecord = $query->fetch(PDO::FETCH_ASSOC)) ) { $query = prepare("INSERT INTO `post_clean` (`post_id`, `board_id`) VALUES ( :post, :board )"); $query->bindValue( ':board', $board ); $query->bindValue( ':post', $post ); $query->execute() or error(db_error($query)); if( $query->rowCount() == 0 ) { error("The database failed to create a record for this content in `post_clean` to record clean status."); } $cleanRecord = true; } } // Revoking clean status (open it to reports?) else { // Attempt to find a `post_clean` row for this content. $query = prepare("SELECT * FROM `post_clean` WHERE `board_id` = :board AND `post_id` = :post"); $query->bindValue( ':board', $board ); $query->bindValue( ':post', $post ); $query->execute() or error(db_error($query)); if( !($cleanRecord = $query->fetch(PDO::FETCH_ASSOC)) ) { error($config['error']['404']); } } // Update the `post_clean` row represented by $clean. if( $cleanRecord ) { // Build our query based on the URI arguments. if( $global && $local ) { $query = prepare("UPDATE `post_clean` SET {$query_global}, {$query_global_mod}, {$query_local}, {$query_local_mod} WHERE `board_id` = :board AND `post_id` = :post"); } else if( $global ) { $query = prepare("UPDATE `post_clean` SET {$query_global}, {$query_global_mod} WHERE `board_id` = :board AND `post_id` = :post"); } else { $query = prepare("UPDATE `post_clean` SET {$query_local}, {$query_local_mod} WHERE `board_id` = :board AND `post_id` = :post"); } $query->bindValue( ':clean', !$unclean ); $query->bindValue( ':mod', $unclean ? NULL : $mod['id'] ); $query->bindValue( ':board', $board ); $query->bindValue( ':post', $post ); $query->execute() or error(db_error($query)); // Finally, run a query to tidy up our records. if( $unclean ) { // Query is removing clean status from content. // Remove any clean records that are now null. $cleanup = prepare("DELETE FROM `post_clean` WHERE `clean_local` = FALSE AND `clean_global` = FALSE"); $query->execute() or error(db_error($query)); } else { // Content is clean, auto-handle all reports. // If this is a total clean, we don't need to update records first. if( !($global && $local) ) { $query = prepare("UPDATE `reports` SET `" . ($local ? "local" : "global") . "` = FALSE WHERE `board` = :board AND `post` = :post"); $query->bindValue( ':board', $board ); $query->bindValue( ':post', $post ); $query->execute() or error(db_error($query)); // If we didn't hit anything, this content doesn't have reports, so don't run the delete query. $require_delete = ($query->rowCount() > 0); if( $require_delete ) { $query = prepare("DELETE FROM `reports` WHERE `local` = FALSE and `global` = FALSE"); $query->execute() or error(db_error($query)); } } // This is a total clean, so delete content by ID rather than via cleanup. else { $query = prepare("DELETE FROM `reports` WHERE `board` = :board AND `post` = :post"); $query->bindValue( ':board', $board ); $query->bindValue( ':post', $post ); $query->execute() or error(db_error($query)); } } // Log the action. // Having clear wording of ths log is very important because of the nature of clean status. $log_action = ($unclean ? "Closed" : "Re-opened" ); $log_scope = ($local && $global ? "local and global" : ($local ? "local" : "global" ) ); modLog( "{$log_action} reports for post #{$post} in {$log_scope}.", $board); if ($config['cache']['enabled']) { cache::delete("post_clean_{$board}_{$post}"); } rebuildPost( $post ); } // Redirect if( $global_reports ) { header('Location: ?/reports/global', true, $config['redirect_http']); } else { header('Location: ?/reports', true, $config['redirect_http']); } } function mod_config($board_config = false) { global $config, $mod, $board; if ($board_config && !openBoard($board_config)) error($config['error']['noboard']); if (!hasPermission($config['mod']['edit_config'], $board_config)) error($config['error']['noaccess']); $config_file = $board_config ? $board['dir'] . 'config.php' : 'inc/instance-config.php'; if ($config['mod']['config_editor_php']) { $readonly = !(is_file($config_file) ? is_writable($config_file) : is_writable(dirname($config_file))); if (!$readonly && isset($_POST['code'])) { $code = $_POST['code']; // Save previous instance_config if php_check_syntax fails $old_code = file_get_contents($config_file); file_put_contents($config_file, $code); $resp = shell_exec_error('php -l ' . $config_file); if (preg_match('/No syntax errors detected/', $resp)) { header('Location: ?/config' . ($board_config ? '/' . $board_config : ''), true, $config['redirect_http']); return; } else { file_put_contents($config_file, $old_code); error($config['error']['badsyntax'] . $resp); } } $instance_config = @file_get_contents($config_file); if ($instance_config === false) { $instance_config = " $instance_config, 'readonly' => $readonly, 'boards' => listBoards(), 'board' => $board_config, 'file' => $config_file, 'token' => make_secure_link_token('config' . ($board_config ? '/' . $board_config : '')) )); return; } require_once 'inc/mod/config-editor.php'; $conf = config_vars(); foreach ($conf as &$var) { if (is_array($var['name'])) { $c = &$config; foreach ($var['name'] as $n) $c = &$c[$n]; } else { $c = @$config[$var['name']]; } $var['value'] = $c; } unset($var); if (isset($_POST['save'])) { $config_append = ''; foreach ($conf as $var) { $field_name = 'cf_' . (is_array($var['name']) ? implode('/', $var['name']) : $var['name']); if ($var['type'] == 'boolean') $value = isset($_POST[$field_name]); elseif (isset($_POST[$field_name])) $value = $_POST[$field_name]; else continue; // ??? if (!settype($value, $var['type'])) continue; // invalid if ($value != $var['value']) { // This value has been changed. $config_append .= '$config'; if (is_array($var['name'])) { foreach ($var['name'] as $name) $config_append .= '[' . var_export($name, true) . ']'; } else { $config_append .= '[' . var_export($var['name'], true) . ']'; } $config_append .= ' = '; if (@$var['permissions'] && isset($config['mod']['groups'][$value])) { $config_append .= $config['mod']['groups'][$value]; } else { $config_append .= var_export($value, true); } $config_append .= ";\n"; } } if (!empty($config_append)) { $config_append = "\n// Changes made via web editor by \"" . $mod['username'] . "\" @ " . date('r') . ":\n" . $config_append . "\n"; if (!is_file($config_file)) $config_append = "Tinyboard could not write to ' . $config_file . ' with the ammended configuration, probably due to a permissions error.

You may proceed with these changes manually by copying and pasting the following code to the end of ' . $config_file . ':

'; echo Element('page.html', $page); exit; } } header('Location: ?/config' . ($board_config ? '/' . $board_config : ''), true, $config['redirect_http']); exit; } mod_page(_('Config editor') . ($board_config ? ': ' . sprintf($config['board_abbreviation'], $board_config) : ''), 'mod/config-editor.html', array( 'boards' => listBoards(), 'board' => $board_config, 'conf' => $conf, 'file' => $config_file, 'token' => make_secure_link_token('config' . ($board_config ? '/' . $board_config : '')) )); } function mod_themes_list() { global $config; if (!hasPermission($config['mod']['themes'])) error($config['error']['noaccess']); if (!is_dir($config['dir']['themes'])) error(_('Themes directory doesn\'t exist!')); if (!$dir = opendir($config['dir']['themes'])) error(_('Cannot open themes directory; check permissions.')); $query = query('SELECT `theme` FROM ``theme_settings`` WHERE `name` IS NULL AND `value` IS NULL') or error(db_error()); $themes_in_use = $query->fetchAll(PDO::FETCH_COLUMN); // Scan directory for themes $themes = array(); while ($file = readdir($dir)) { if ($file[0] != '.' && is_dir($config['dir']['themes'] . '/' . $file)) { $themes[$file] = loadThemeConfig($file); } } closedir($dir); foreach ($themes as $theme_name => &$theme) { $theme['rebuild_token'] = make_secure_link_token('themes/' . $theme_name . '/rebuild'); $theme['uninstall_token'] = make_secure_link_token('themes/' . $theme_name . '/uninstall'); } mod_page(_('Manage themes'), 'mod/themes.html', array( 'themes' => $themes, 'themes_in_use' => $themes_in_use, )); } function mod_theme_configure($theme_name) { global $config; if (!hasPermission($config['mod']['themes'])) error($config['error']['noaccess']); if (!$theme = loadThemeConfig($theme_name)) { error($config['error']['invalidtheme']); } if (isset($_POST['install'])) { // Check if everything is submitted foreach ($theme['config'] as &$conf) { if (!isset($_POST[$conf['name']]) && $conf['type'] != 'checkbox') error(sprintf($config['error']['required'], $c['title'])); } // Clear previous settings $query = prepare("DELETE FROM ``theme_settings`` WHERE `theme` = :theme"); $query->bindValue(':theme', $theme_name); $query->execute() or error(db_error($query)); foreach ($theme['config'] as &$conf) { $query = prepare("INSERT INTO ``theme_settings`` VALUES(:theme, :name, :value)"); $query->bindValue(':theme', $theme_name); $query->bindValue(':name', $conf['name']); if ($conf['type'] == 'checkbox') $query->bindValue(':value', isset($_POST[$conf['name']]) ? 1 : 0); else $query->bindValue(':value', $_POST[$conf['name']]); $query->execute() or error(db_error($query)); } $query = prepare("INSERT INTO ``theme_settings`` VALUES(:theme, NULL, NULL)"); $query->bindValue(':theme', $theme_name); $query->execute() or error(db_error($query)); // Clean cache Cache::delete("themes"); Cache::delete("theme_settings_".$theme_name); $result = true; $message = false; if (isset($theme['install_callback'])) { $ret = $theme['install_callback'](themeSettings($theme_name)); if ($ret && !empty($ret)) { if (is_array($ret) && count($ret) == 2) { $result = $ret[0]; $message = $ret[1]; } } } if (!$result) { // Install failed $query = prepare("DELETE FROM ``theme_settings`` WHERE `theme` = :theme"); $query->bindValue(':theme', $theme_name); $query->execute() or error(db_error($query)); } // Build themes rebuildThemes('all'); mod_page(sprintf(_($result ? 'Installed theme: %s' : 'Installation failed: %s'), $theme['name']), 'mod/theme_installed.html', array( 'theme_name' => $theme_name, 'theme' => $theme, 'result' => $result, 'message' => $message )); return; } $settings = themeSettings($theme_name); mod_page(sprintf(_('Configuring theme: %s'), $theme['name']), 'mod/theme_config.html', array( 'theme_name' => $theme_name, 'theme' => $theme, 'settings' => $settings, 'token' => make_secure_link_token('themes/' . $theme_name) )); } function mod_theme_uninstall($theme_name) { global $config; if (!hasPermission($config['mod']['themes'])) error($config['error']['noaccess']); $query = prepare("DELETE FROM ``theme_settings`` WHERE `theme` = :theme"); $query->bindValue(':theme', $theme_name); $query->execute() or error(db_error($query)); // Clean cache Cache::delete("themes"); Cache::delete("theme_settings_".$theme); header('Location: ?/themes', true, $config['redirect_http']); } function mod_theme_rebuild($theme_name) { global $config; if (!hasPermission($config['mod']['themes'])) error($config['error']['noaccess']); rebuildTheme($theme_name, 'all'); mod_page(sprintf(_('Rebuilt theme: %s'), $theme_name), 'mod/theme_rebuilt.html', array( 'theme_name' => $theme_name, )); } // This needs to be done for `secure` CSRF prevention compatibility, otherwise the $board will be read in as the token if editing global pages. function delete_page_base($page = '', $board = false) { global $config, $mod; if (empty($board)) $board = false; if (!$board && $mod['boards'][0] !== '*') error($config['error']['noaccess']); if (!hasPermission($config['mod']['edit_pages'], $board)) error($config['error']['noaccess']); if ($board !== FALSE && !openBoard($board)) error($config['error']['noboard']); if (preg_match('/^[a-z0-9]{1,255}$/', $page) && !preg_match('/^(index|catalog|index\+50)|(\d+)$/', $page)) { if ($board) { $query = prepare('DELETE FROM ``pages`` WHERE `board` = :board AND `name` = :name'); $query->bindValue(':board', ($board ? $board : NULL)); } else { $query = prepare('DELETE FROM ``pages`` WHERE `board` IS NULL AND `name` = :name'); } $query->bindValue(':name', $page); $query->execute() or error(db_error($query)); @file_unlink(($board ? ($board . '/') : '') . $page . '.html'); } header('Location: ?/edit_pages' . ($board ? ('/' . $board) : ''), true, $config['redirect_http']); } function mod_delete_page($page = '') { delete_page_base($page); } function mod_delete_page_board($page = '', $board = false) { delete_page_base($page, $board); } function mod_edit_page($id) { global $config, $mod, $board; $query = prepare('SELECT * FROM ``pages`` WHERE `id` = :id'); $query->bindValue(':id', $id); $query->execute() or error(db_error($query)); $page = $query->fetch(); if (!$page) error(_('Could not find the page you are trying to edit.')); if (!$page['board'] && $mod['boards'][0] !== '*') error($config['error']['noaccess']); if (!hasPermission($config['mod']['edit_pages'], $page['board'])) error($config['error']['noaccess']); if ($page['board'] && !openBoard($page['board'])) error($config['error']['noboard']); if (isset($_POST['method'], $_POST['content'])) { $content = $_POST['content']; $method = $_POST['method']; $page['type'] = $method; if (!in_array($method, array('markdown', 'html', 'infinity'))) error(_('Unrecognized page markup method.')); switch ($method) { case 'markdown': $write = purify_html(markdown($content)); break; case 'html': if (hasPermission($config['mod']['rawhtml'])) { $write = $content; } else { $write = purify_html($content); } break; case 'infinity': $c = $content; markup($content); $write = $content; $content = $c; } if (!isset($write) or !$write) error(_('Failed to mark up your input for some reason...')); $query = prepare('UPDATE ``pages`` SET `type` = :method, `content` = :content WHERE `id` = :id'); $query->bindValue(':method', $method); $query->bindValue(':content', $content); $query->bindValue(':id', $id); $query->execute() or error(db_error($query)); $fn = ($board['uri'] ? ($board['uri'] . '/') : '') . $page['name'] . '.html'; $body = "
$write
"; $html = Element('page.html', array('config' => $config, 'body' => $body, 'title' => utf8tohtml($page['title']))); file_write($fn, $html); modLog("Edited page {$page['name']} (#{$page['id']})"); } if (!isset($content)) { $query = prepare('SELECT `content` FROM ``pages`` WHERE `id` = :id'); $query->bindValue(':id', $id); $query->execute() or error(db_error($query)); $content = $query->fetchColumn(); } mod_page(sprintf(_('Editing static page: %s'), $page['name']), 'mod/edit_page.html', array('page' => $page, 'token' => make_secure_link_token("edit_page/$id"), 'content' => prettify_textarea($content), 'board' => $board)); } function mod_pages($board = false) { global $config, $mod, $pdo; if (empty($board)) $board = false; if (!$board && $mod['boards'][0] !== '*') error($config['error']['noaccess']); if (!hasPermission($config['mod']['edit_pages'], $board)) error($config['error']['noaccess']); if ($board !== FALSE && !openBoard($board)) error($config['error']['noboard']); if ($board) { $query = prepare('SELECT * FROM ``pages`` WHERE `board` = :board'); $query->bindValue(':board', $board); } else { $query = query('SELECT * FROM ``pages`` WHERE `board` IS NULL'); } $query->execute() or error(db_error($query)); $pages = $query->fetchAll(PDO::FETCH_ASSOC); if (isset($_POST['page'])) { if ($board and sizeof($pages) > $config['pages_max']) error(sprintf(_('Sorry, this site only allows %d pages per board.'), $config['pages_max'])); if (!preg_match('/^[a-z0-9]{1,255}$/', $_POST['page'])) error(_('Page names must be < 255 chars and may only contain lowercase letters A-Z and digits 1-9.')); if (preg_match('/^(index|catalog|index\+50)|(\d+)$/', $_POST['page'])) error(_('Nope.')); foreach ($pages as $i => $p) { if ($_POST['page'] === $p['name']) error(_('Refusing to create a new page with the same name as an existing one.')); } $title = ($_POST['title'] ? $_POST['title'] : NULL); $query = prepare('INSERT INTO ``pages``(board, title, name) VALUES(:board, :title, :name)'); $query->bindValue(':board', ($board ? $board : NULL)); $query->bindValue(':title', $title); $query->bindValue(':name', $_POST['page']); $query->execute() or error(db_error($query)); $pages[] = array('id' => $pdo->lastInsertId(), 'name' => $_POST['page'], 'board' => $board, 'title' => $title); } foreach ($pages as $i => &$p) { $p['delete_token'] = make_secure_link_token('edit_pages/delete/' . $p['name'] . ($board ? ('/' . $board) : '')); } mod_page(_('Pages'), 'mod/pages.html', array('pages' => $pages, 'token' => make_secure_link_token('edit_pages' . ($board ? ('/' . $board) : '')), 'board' => $board)); } function mod_debug_antispam() { global $pdo, $config; if (!hasPermission($config['mod']['debug_antispam'])) error($config['error']['noaccess']); $args = array(); if (isset($_POST['board'], $_POST['thread'])) { $where = '`board` = ' . $pdo->quote($_POST['board']); if ($_POST['thread'] != '') $where .= ' AND `thread` = ' . $pdo->quote($_POST['thread']); if (isset($_POST['purge'])) { $query = prepare(', DATE ``antispam`` SET `expires` = UNIX_TIMESTAMP() + :expires WHERE' . $where); $query->bindValue(':expires', $config['spam']['hidden_inputs_expire']); $query->execute() or error(db_error()); } $args['board'] = $_POST['board']; $args['thread'] = $_POST['thread']; } else { $where = ''; } $query = query('SELECT COUNT(*) FROM ``antispam``' . ($where ? " WHERE $where" : '')) or error(db_error()); $args['total'] = number_format($query->fetchColumn()); $query = query('SELECT COUNT(*) FROM ``antispam`` WHERE `expires` IS NOT NULL' . ($where ? " AND $where" : '')) or error(db_error()); $args['expiring'] = number_format($query->fetchColumn()); $query = query('SELECT * FROM ``antispam`` ' . ($where ? "WHERE $where" : '') . ' ORDER BY `passed` DESC LIMIT 40') or error(db_error()); $args['top'] = $query->fetchAll(PDO::FETCH_ASSOC); $query = query('SELECT * FROM ``antispam`` ' . ($where ? "WHERE $where" : '') . ' ORDER BY `created` DESC LIMIT 20') or error(db_error()); $args['recent'] = $query->fetchAll(PDO::FETCH_ASSOC); mod_page(_('Debug: Anti-spam'), 'mod/debug/antispam.html', $args); } function mod_debug_recent_posts() { global $pdo, $config; if (!hasPermission($config['mod']['debug_recent'])) error($config['error']['noaccess']); $limit = 500; $boards = listBoards(); // Manually build an SQL query $query = 'SELECT * FROM ('; foreach ($boards as $board) { $query .= sprintf('SELECT *, %s AS `board` FROM ``posts_%s`` UNION ALL ', $pdo->quote($board['uri']), $board['uri']); } // Remove the last "UNION ALL" seperator and complete the query $query = preg_replace('/UNION ALL $/', ') AS `all_posts` ORDER BY `time` DESC LIMIT ' . $limit, $query); $query = query($query) or error(db_error()); $posts = $query->fetchAll(PDO::FETCH_ASSOC); // Fetch recent posts from flood prevention cache $query = query("SELECT * FROM ``flood`` ORDER BY `time` DESC") or error(db_error()); $flood_posts = $query->fetchAll(PDO::FETCH_ASSOC); foreach ($posts as &$post) { $post['snippet'] = pm_snippet($post['body']); foreach ($flood_posts as $flood_post) { if ($flood_post['time'] == $post['time'] && $flood_post['posthash'] == make_comment_hex($post['body_nomarkup']) && $flood_post['filehash'] == $post['filehash']) $post['in_flood_table'] = true; } } mod_page(_('Debug: Recent posts'), 'mod/debug/recent_posts.html', array('posts' => $posts, 'flood_posts' => $flood_posts)); } function mod_debug_sql() { global $config; if (!hasPermission($config['mod']['debug_sql'])) error($config['error']['noaccess']); $args['security_token'] = make_secure_link_token('debug/sql'); if (isset($_POST['query'])) { $args['query'] = $_POST['query']; if ($query = query($_POST['query'])) { $args['result'] = $query->fetchAll(PDO::FETCH_ASSOC); if (!empty($args['result'])) $args['keys'] = array_keys($args['result'][0]); else $args['result'] = 'empty'; } else { $args['error'] = db_error(); } } mod_page(_('Debug: SQL'), 'mod/debug/sql.html', $args); } function mod_debug_apc() { global $config; if (!hasPermission($config['mod']['debug_apc'])) error($config['error']['noaccess']); if ($config['cache']['enabled'] != 'apc') error('APC is not enabled.'); $cache_info = apc_cache_info('user'); // $cached_vars = new APCIterator('user', '/^' . $config['cache']['prefix'] . '/'); $cached_vars = array(); foreach ($cache_info['cache_list'] as $var) { if ($config['cache']['prefix'] != '' && strpos(isset($var['key']) ? $var['key'] : $var['info'], $config['cache']['prefix']) !== 0) continue; $cached_vars[] = $var; } mod_page(_('Debug: APC'), 'mod/debug/apc.html', array('cached_vars' => $cached_vars)); }