diff --git a/LICENSE b/LICENSE
index 3c10b3ca..3a742f99 100644
--- a/LICENSE
+++ b/LICENSE
@@ -1,13 +1,21 @@
-Copyright (c) 2010 by Omega Software Development Group
-
-Permission to use, copy, modify, and/or distribute this software for any
-purpose with or without fee is hereby granted, provided that the above copyright
-notice and this permission notice appear in all copies.
-
-THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
-WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
-MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
-ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
-WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
-ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
-OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
\ No newline at end of file
+Copyright (c) 2010-2011, Omega Software Development Group
+
+Permission to use, copy, modify, and/or distribute this software is granted,
+provided that the following terms are obeyed.
+
+1. The above copyright notice, this permission notice, and these terms must
+appear in all copies and modifications of this software.
+
+2. This software must not be utilized in any manner that is primarily intended
+for or directed toward commercial advantage or private monetary compensation.
+
+3. Any changes to this software must be made easily publicly available in the
+form of source code.
+
+THIS SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH
+REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND
+FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT,
+INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS
+OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER
+TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF
+THIS SOFTWARE.
diff --git a/README.md b/README.md
index 815e3a4a..bad204dc 100644
--- a/README.md
+++ b/README.md
@@ -5,23 +5,25 @@ Tinyboard is an imageboard software package written in PHP. It aims to maintain
Tinyboard is not currently at a stable state.
-[o]: http://omegadev.org/
+[o]: http://omegasdg.com/
+
+## Requirements
+ 1. PHP >= 5.2.0
+ 2. php-gd
+ 3. php-pdo with appropriate driver for your database
## Installation
- 1. Tinyboard requires a MySQL database and a user to work. Create one.
+ 1. Tinyboard requires an SQL database and a user to work. Create one.
2. Import 'install.sql' into the database. There are several ways to do this.
- using phpMyAdmin
- `mysql -uUSERNAME -pPASSWORD DATABASE < install.sql`
3. Edit 'instance-config.php' to suit your installation. You should copy some values from '[inc/config.php][c]' to redefine them in the instance-config.
- 4. Make sure that the directories used by Tinyboard are writable. Depending on your setup, you may need to `chmod` the directories to 777.
- The default directories are:
- - ./res
- - ./src
- - ./thumb
- - . (document root)
+ 4. Make sure that the Tinyboard directory is writable. Depending on your setup, you may need to `chmod` "." to 777, with `chmod 777 .`
5. Ensure everything is okay by running [test.php][t] in a browser. The script will try and help you correct your errors.
6. Run the [post.php][p] script. It should create an index.html and redirect you to it if everything is okay.
7. Optional (highly recommended): Either delete or chmod as unreadable the following files: [test.php][t], [install.sql][i], and this [README][r].
+
+An automated installation script will be completed soon.
[t]: http://github.com/savetheinternet/Tinyboard/blob/master/test.php
[p]: http://github.com/savetheinternet/Tinyboard/blob/master/post.php
@@ -30,17 +32,24 @@ Tinyboard is not currently at a stable state.
[r]: http://github.com/savetheinternet/Tinyboard/blob/master/README.md
## License
-Copyright (c) 2010 by Omega Software Development Group
+Copyright (c) 2010-2011, Omega Software Development Group
-Permission to use, copy, modify, and/or distribute this software for any
-purpose with or without fee is hereby granted, provided that the above copyright
-notice and this permission notice appear in all copies.
+Permission to use, copy, modify, and/or distribute this software is granted,
+provided that the following terms are obeyed.
-THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
-WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
-MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
-ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
-WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
-ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
-OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
+1. The above copyright notice, this permission notice, and these terms must
+appear in all copies and modifications of this software.
+2. This software must not be utilized in any manner that is primarily intended
+for or directed toward commercial advantage or private monetary compensation.
+
+3. Any changes to this software must be made easily publicly available in the
+form of source code.
+
+THIS SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH
+REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND
+FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT,
+INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS
+OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER
+TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF
+THIS SOFTWARE.
\ No newline at end of file
diff --git a/default.css b/default.css
new file mode 100644
index 00000000..e69de29b
diff --git a/img/fade-yotsuba.png b/img/fade-yotsuba.png
new file mode 100644
index 00000000..80c566b9
Binary files /dev/null and b/img/fade-yotsuba.png differ
diff --git a/inc/config.php b/inc/config.php
index 4d481e8c..4fe7a988 100644
--- a/inc/config.php
+++ b/inc/config.php
@@ -8,177 +8,326 @@
* your instance-config.php
*
*/
-
+
+ $config = Array(
+ 'db' => Array(),
+ 'cookies' => Array(),
+ 'error' => Array(),
+ 'dir' => Array(),
+ 'mod' => Array()
+ );
// Database stuff
- define('MY_SERVER', 'localhost', true);
- define('MY_USER', '', true);
- define('MY_PASSWORD', '', true);
- define('MY_DATABASE', '', true);
+
+
+ // SQL driver ("mysql", "pgsql", "sqlite", "dblib", etc)
+ // http://www.php.net/manual/en/pdo.drivers.php
+ $config['db']['type'] = 'mysql';
+ // Hostname or IP address
+ $config['db']['server'] = 'localhost';
+ // Login
+ $config['db']['user'] = '';
+ $config['db']['password'] = '';
+ // Tinyboard database
+ $config['db']['database'] = '';
+ // Anything more to add to the DSN string (eg. port=xxx;foo=bar)
+ $config['db']['dsn'] = '';
// The name of the session cookie (PHP's $_SESSION)
- define('SESS_COOKIE', 'imgboard', true);
+ $config['cookies']['session']= 'imgboard';
- // Used to safely determine when the user was first seen, to prevent floods.
- // time()
- define('TIME_COOKIE', 'arrived', true);
- // HASH_COOKIE contains an MD5 hash of TIME_COOKIE+SALT for verification.
- define('HASH_COOKIE', 'hash', true);
+ // Used to safely determine when the user was first seen, to prevent floods. Contains a UNIX timestamp.
+ $config['cookies']['time'] = 'arrived';
+ // Contains an MD5 hash of $config['cookies']['time'] for verification.
+ $config['cookies']['hash'] = 'hash';
// Used for moderation login
- define('MOD_COOKIE', 'mod', true);
- // Where to set the 'path' parameter to ROOT when creating cookies. Recommended.
- define('JAIL_COOKIES', true, true);
-
- // Whether or not to lock moderator sessions to the IP address that was logged in with.
- define('MOD_LOCK_IP', true, true);
- // Mod
-
+ $config['cookies']['mod'] = 'mod';
+ // Where to set the 'path' parameter to $config['root'] when creating cookies. Recommended.
+ $config['cookies']['jail'] = true;
// How long should the cookies last (in seconds)
- define('COOKIE_EXPIRE', 15778463, true); //6 months
-
- define('SALT', 'wefaw98YHEWUFuo', true);
-
+ $config['cookies']['expire']= 15778463; //6 months
+ // Make this something long and random for security
+ $config['cookies']['salt'] = 'wefaw98YHEWUFuo';
+ // How long should moderators should remain logged in (0=browser session) (in seconds)
+ $config['mod']['expire'] = 15778463; //6 months
+ // Used to salt secure tripcodes (##trip)
+ $config['secure_trip_salt'] = '@#$&^@#)$(*&@!_$(&329-8347';
+
// How many seconds before you can post, after the first visit
- define('LURKTIME', 30, true);
+ $config['lurktime'] = 30;
+ // How many seconds between each post
+ $config['flood_time'] = 10;
+ // How many seconds between each post with exactly the same content and same IP
+ $config['flood_time_ip'] = 120;
+ // Same as above but different IP address
+ $config['flood_time_same'] = 30;
+ // Do you need a body for your non-OP posts?
+ $config['force_body'] = true;
// Max body length
- define('MAX_BODY', 1800, true);
-
- define('THREADS_PER_PAGE', 10, true);
- define('MAX_PAGES', 5, true);
- define('THREADS_PREVIEW', 5, true);
+ $config['max_body'] = 1800;
+
+ $config['threads_per_page'] = 10;
+ $config['max_pages'] = 10;
+ $config['threads_preview'] = 5;
// For development purposes. Turns 'display_errors' on. Not recommended for production.
- define('VERBOSE_ERRORS', true, true);
+ $config['verbose_errors'] = true;
// Error messages
- define('ERROR_LURK', 'Lurk some more before posting.', true);
- define('ERROR_BOT', 'You look like a bot.', true);
- define('ERROR_TOOLONG', 'The %s field was too long.', true);
- define('ERROR_TOOLONGBODY', 'The body was too long.', true);
- define('ERROR_TOOSHORTBODY', 'The body was too short or empty.', true);
- define('ERROR_NOIMAGE', 'You must upload an image.', true);
- define('ERROR_NOMOVE', 'The server failed to handle your upload.', true);
- define('ERROR_FILEEXT', 'Unsupported image format.', true);
- define('ERROR_NOBOARD', 'Invalid board!', true);
- define('ERROR_NONEXISTANT', 'Thread specified does not exist.', true);
- define('ERROR_NOPOST', 'You didn\'t make a post.', true);
- define('ERR_INVALIDIMG','Invalid image.', true);
- define('ERR_FILESIZE', 'Maximum file size: %maxsz% bytes Your file\'s size: %filesz% bytes', true);
- define('ERR_MAXSIZE', 'The file was too big.', true);
- define('ERR_INVALIDZIP','Invalid archive!', true);
+ $config['error']['lurk'] = 'Lurk some more before posting.';
+ $config['error']['bot'] = 'You look like a bot.';
+ $config['error']['toolong'] = 'The %s field was too long.';
+ $config['error']['toolong_body'] = 'The body was too long.';
+ $config['error']['tooshort_body'] = 'The body was too short or empty.';
+ $config['error']['noimage'] = 'You must upload an image.';
+ $config['error']['nomove'] = 'The server failed to handle your upload.';
+ $config['error']['fileext'] = 'Unsupported image format.';
+ $config['error']['noboard'] = 'Invalid board!';
+ $config['error']['nonexistant'] = 'Thread specified does not exist.';
+ $config['error']['locked'] = 'Thread locked. You may not reply at this time.';
+ $config['error']['nopost'] = 'You didn\'t make a post.';
+ $config['error']['flood'] = 'Flood detected; Post discared.';
+ $config['error']['unoriginal'] = 'Unoriginal content!';
+ $config['error']['muted'] = 'Unoriginal content! You have been muted for %d seconds.';
+ $config['error']['youaremuted'] = 'You are muted! Expires in %d seconds.';
+ $config['error']['tor'] = 'Hmm… That looks like a Tor exit node.';
+ $config['error']['toomanylinks'] = 'Too many links; flood detected.';
+ $config['error']['nodelete'] = 'You didn\'t select anything to delete.';
+ $config['error']['invalidpassword'] = 'Wrong password…';
+ $config['error']['invalidimg'] = 'Invalid image.';
+ $config['error']['filesize'] = 'Maximum file size: %maxsz% bytes Your file\'s size: %filesz% bytes';
+ $config['error']['maxsize'] = 'The file was too big.';
+ $config['error']['invalidzip'] = 'Invalid archive!';
// Moderator errors
- define('ERROR_INVALID', 'Invalid username and/or password.', true);
- define('ERROR_INVALIDAFTER', 'Invalid username and/or password. Your user may have been deleted or changed.');
- define('ERROR_MALFORMED','Invalid/malformed cookies.', true);
+ $config['error']['invalid'] = 'Invalid username and/or password.';
+ $config['error']['notamod'] = 'You are not a mod…';
+ $config['error']['invalidafter'] = 'Invalid username and/or password. Your user may have been deleted or changed.';
+ $config['error']['malformed'] = 'Invalid/malformed cookies.';
+ $config['error']['missedafield'] = 'Your browser didn\'t submit an input when it should have.';
+ $config['error']['required'] = 'The %s field is required.';
+ $config['error']['invalidfield'] = 'The %s field was invalid.';
+ $config['error']['boardexists'] = 'There is already a %s board.';
+ $config['error']['noaccess'] = 'You don\'t have permission to do that.';
+ $config['error']['invalidpost'] = 'That post doesn\'t exist…';
+ $config['error']['404'] = 'Page not found.';
+
+ // Reply limit (deletes thread when this is reached)
+ $config['reply_limit'] = 250;
// For resizing, max values
- define('THUMB_WIDTH', 200, true);
- define('THUMB_HEIGHT', 200, true);
-
+ $config['thumb_width'] = 255;
+ $config['thumb_height'] = 255;
+
+ // Store image hash in the database for r9k-like boards implementation soon
+ // Function name for hashing
+ // sha1_file, md5_file, etc.
+ $config['file_hash'] = 'sha1_file';
+
+ $config['block_tor'] = true;
+ // Typically spambots try to post a lot of links. Refuse a post with X standalone links?
+ $config['max_links'] = 20;
+
// Maximum image upload size in bytes
- define('MAX_FILESIZE', 10*1024*1024, true); // 10MB
+ $config['max_filesize'] = 10*1024*1024; // 10MB
// Maximum image dimensions
- define('MAX_WIDTH', 10000, true);
- define('MAX_HEIGHT', MAX_WIDTH, true);
-
- /* When you upload a ZIP as a file, all the images inside the archive
- * get dumped into the thread as replies.
- * Extremely beta and not recommended yet.
- */
- define('ALLOW_ZIP', false, true);
- define('ZIP_IMAGE', 'src/zip.png', true);
-
+ $config['max_width'] = 1000;
+ $config['max_height'] = $config['max_width']; // 1:1
+
/**
Redraw the image using GD functions to strip any excess data (commonly ZIP archives)
- WARNING: Very beta. Currently strips animated GIFs too :(
+ WARNING: Currently strips animated GIFs too :(
**/
- define('REDRAW_IMAGE', false, true);
+ $config['redraw_image'] = false;
// Redrawing configuration
- define('JPEG_QUALITY', 100, true);
- define('REDRAW_GIF', false, true);
-
- // Display the aspect ratio in a post's file info
- define('SHOW_RATIO', true, true);
-
- define('DIR_IMG', 'src/', true);
- define('DIR_THUMB', 'thumb/', true);
- define('DIR_RES', 'res/', true);
+ $config['jpeg_quality'] = 100;
+ // Temporary fix for the animation-stripping bug
+ $config['redraw_gifs'] = false;
- // Where to store the .html templates. This folder and templates must exist or fatal errors will be thrown.
- define('DIR_TEMPLATE', getcwd() . '/templates', true);
+ // Display the aspect ratio in a post's file info
+ $config['show_ratio'] = true;
// The root directory, including the trailing slash, for Tinyboard.
// examples: '/', 'http://boards.chan.org/', '/chan/'
- define('ROOT', '/', true);
+ $config['root'] = '/';
+
+ $config['dir']['img'] = 'src/';
+ $config['dir']['thumb'] = 'thumb/';
+ $config['dir']['res'] = 'res/';
+ // For load balancing, having a seperate server (and domain/subdomain) for serving static content is possible.
+ // This can either be a directory or a URL (eg. http://static.example.org/)
+ $config['dir']['static'] = $config['root'] . 'static/';
+ // Where to store the .html templates. This folder and templates must exist or fatal errors will be thrown.
+ $config['dir']['template'] = getcwd() . '/templates';
+ // Static images
+ // These can be URLs OR base64 (data URI scheme)
+ $config['image_sticky'] = $config['dir']['static'] . 'sticky.gif';
+ $config['image_locked'] = $config['dir']['static'] . 'locked.gif';
+ $config['image_deleted'] = $config['dir']['static'] . 'deleted.png';
+ $config['image_zip'] = $config['dir']['static'] . 'zip.png';
// If for some reason the folders and static HTML index files aren't in the current working direcotry,
// enter the directory path here. Otherwise, keep it false.
- define('ROOT_FILE', false, true);
+ $config['root_file'] = false;
- define('POST_URL', ROOT . 'post.php', true);
- define('FILE_INDEX', 'index.html', true);
- define('FILE_PAGE', '%d.html', true);
+ $config['file_index'] = 'index.html';
+ $config['file_page'] = '%d.html';
+ $config['file_mod'] = 'mod.php';
// Multi-board (%s is board abbreviation)
- define('BOARD_PATH', '%s/', true);
+ $config['board_path'] = '%s/';
+
+ // The HTTP status code to use when redirecting.
+ // Should be 3xx (redirection). http://www.w3.org/Protocols/rfc2616/rfc2616-sec10.html
+ // "302" is recommended.
+ $config['redirect_http'] = 302;
+
+ // TODO: Put this in per-board instance-config instead
+ // Robot stuff
+ // Strip repeating characters when making hashes
+ $config['robot_enable'] = false;
+ $config['robot_strip_repeating'] = true;
+
+ // Enable mutes
+ // Tinyboard uses ROBOT9000's original 2^x implementation
+ $config['robot_mute'] = true;
+ // How many mutes x hours ago to include in the algorithm
+ $config['robot_mute_hour'] = 50;
+ // If you want to alter the algorithm a bit. Default value is 2. n^x
+ $config['robot_mute_multiplier'] = 2;
+ $config['robot_mute_descritpion'] = 'You have been muted for unoriginal content.';
+
+ /*
+ Mod stuff
+ */
+ // Whether or not to lock moderator sessions to the IP address that was logged in with.
+ $config['mod']['lock_ip'] = true;
+ // The page that is first shown when a moderator logs in. Defaults to the dashboard.
+ $config['mod']['default'] = '/';
+ // Don't even display MySQL password to administrators (in the configuration page).
+ $config['mod']['never_reveal_password'] = true;
+ // Do a DNS lookup on IP addresses to get their hostname on the IP summary page
+ $config['mod']['dns_lookup'] = true;
+ // Show ban form on the IP summary page
+ $config['mod']['ip_banform'] = true;
+ // How many recent posts, per board, to show in the IP summary page
+ $config['mod']['ip_recentposts'] = 5;
+
+ // Probably best not to change these:
+ define('JANITOR', 0, true);
+ define('MOD', 1, true);
+ define('ADMIN', 2, true);
+
+ // Permissions
+ // What level of administration you need to:
+
+ /* Post Controls */
+ // View IP addresses
+ $config['mod']['show_ip'] = MOD;
+ // Delete a post
+ $config['mod']['delete'] = JANITOR;
+ // Ban a user for a post
+ $config['mod']['ban'] = MOD;
+ // Ban and delete (one click; instant)
+ $config['mod']['bandelete'] = MOD;
+ // Delete file (and keep post)
+ $config['mod']['deletefile'] = JANITOR;
+ // Delete all posts by IP
+ $config['mod']['deletebyip'] = MOD;
+ // Sticky a thread
+ $config['mod']['sticky'] = MOD;
+ // Lock a thread
+ $config['mod']['lock'] = MOD;
+ // Post in a locked thread
+ $config['mod']['postinlocked'] = MOD;
+ // Post bypass unoriginal content check
+ $config['mod']['postunoriginal'] = MOD;
+ // Raw HTML posting
+ $config['mod']['rawhtml'] = MOD;
+
+ /* Administration */
+ // Display the contents of instance-config.php
+ $config['mod']['show_config'] = ADMIN;
+ // View list of bans
+ $config['mod']['view_banlist'] = MOD;
+ // View the username of the mod who made a ban
+ $config['mod']['view_banstaff'] = MOD;
+ // If the moderator doesn't fit the $config['mod']['view_banstaff''] (previous) permission,
+ // show him just a "?" instead. Otherwise, it will be "Mod" or "Admin"
+ $config['mod']['view_banquestionmark'] = false;
+ // Show expired bans in the ban list (they are kept in cache until the culprit returns)
+ $config['mod']['view_banexpired'] = true;
+ // Create a new board
+ $config['mod']['newboard'] = ADMIN;
+
+ // Mod links (full HTML)
+ // Correspond to above permission directives
+ $config['mod']['link_delete'] = '[D]';
+ $config['mod']['link_ban'] = '[B]';
+ $config['mod']['link_bandelete'] = '[B&D]';
+ $config['mod']['link_deletefile'] = '[F]';
+ $config['mod']['link_deletebyip'] = '[D+]';
+ $config['mod']['link_sticky'] = '[Sticky]';
+ $config['mod']['link_desticky'] = '[-Sticky]';
+ $config['mod']['link_lock'] = '[Lock]';
+ $config['mod']['link_unlock'] = '[-Lock]';
// A small file in the main directory indicating that the script has been ran and the board(s) have been generated.
// This keeps the script from querying the database and causing strain when not needed.
- define('HAS_INSTALLED', '.installed', true);
+ $config['has_installed'] = '.installed';
+
+ // Name of the boards. Usually '/%s/' (/b/, /mu/, etc)
+ // $config['board_abbreviation'] - BOARD_TITLE
+ $config['board_abbreviation'] = '/%s/';
- // Name of the boards. Typically '/%s/' (/b/, /mu/, etc)
- // BOARD_ABBREVIATION - BOARD_TITLE
- define('BOARD_ABBREVIATION', '/%s/', true);
-
// Automatically convert things like "..." to Unicode characters ("…")
- define('AUTO_UNICODE', true, true);
+ $config['auto_unicode'] = true;
// Use some Wiki-like syntax (''em'', '''strong''', ==Heading==, etc)
- define('WIKI_MARKUP', true, true);
+ $config['wiki_markup'] = true;
// Whether to turn URLs into functional links
- define('MARKUP_URLS', true, true);
+ $config['markup_urls'] = true;
// Complex regular expression to catch URLs
- define('URL_REGEX', '/' . '(https?|ftp):\/\/' . '([\w\-]+\.)+[a-zA-Z]{2,6}' . '(\/([\w\-~\.#\/?=&;:+%]+))?' . '/', true);
+ $config['url_regex'] = '/' . '(https?|ftp):\/\/' . '(([\w\-]+\.)+[a-zA-Z]{2,6}|\d{1,3}\.\d{1,3}\.\d{1,3}\.\d{1,3})' . '(\/([\w\-~\.#\/?=&;:+%]+)?)?' . '/';
// Allowed file extensions
- $allowed_ext = Array('jpg', 'jpeg', 'bmp', 'gif', 'png', true);
-
- define('BUTTON_NEWTOPIC', 'New Topic', true);
- define('BUTTON_REPLY', 'New Reply', true);
+ $config['allowed_ext'] = Array('jpg', 'jpeg', 'bmp', 'gif', 'png');
- define('ALWAYS_NOKO', false, true);
+ // The names on the post buttons. (On most imageboards, these are both "Post".)
+ $config['button_newtopic'] = 'New Topic';
+ $config['button_reply'] = 'New Reply';
- define('URL_MATCH', '/^' .
- (@$_SERVER['HTTPS']?'https':'http').':\/\/'.$_SERVER['HTTP_HOST'] .
- '(' .
- preg_quote(ROOT, '/') .
- str_replace('%s', '\w{1,8}', preg_quote(BOARD_PATH, '/')) .
- '|' .
- preg_quote(ROOT, '/') .
- str_replace('%s', '\w{1,8}', preg_quote(BOARD_PATH, '/')) .
- preg_quote(FILE_INDEX, '/') .
- '|' .
- preg_quote(ROOT, '/') .
- str_replace('%s', '\w{1,8}', preg_quote(BOARD_PATH, '/')) .
- str_replace('%d', '\d+', preg_quote(FILE_PAGE, '/')) .
- ')$/', true);
+ // The string passed to date() for post times
+ // http://php.net/manual/en/function.date.php
+ $config['post_date'] = 'm/d/y (D) H:i:s';
- if(ROOT_FILE) {
- chdir(ROOT_FILE);
+ // Always act as if they had typed "noko" in the email field no mattter what
+ $config['always_noko'] = false;
+
+ $config['url_match'] = '/^' .
+ (preg_match($config['url_regex'], $config['root']) ? '' :
+ (@$_SERVER['HTTPS']?'https':'http') .
+ ':\/\/'.$_SERVER['HTTP_HOST']) .
+ preg_quote($config['root'], '/') .
+ '(' .
+ str_replace('%s', '\w{1,8}', preg_quote($config['board_path'], '/')) .
+ '|' .
+ str_replace('%s', '\w{1,8}', preg_quote($config['board_path'], '/')) .
+ preg_quote($config['file_index'], '/') .
+ '|' .
+ str_replace('%s', '\w{1,8}', preg_quote($config['board_path'], '/')) .
+ str_replace('%d', '\d+', preg_quote($config['file_page'], '/')) .
+ '|' .
+ preg_quote($config['file_mod'], '/') .
+ '\?\/.+' .
+ ')$/i';
+
+ if($config['root_file']) {
+ chdir($config['root_file']);
}
- if(VERBOSE_ERRORS) {
+ if($config['verbose_errors']) {
error_reporting(E_ALL);
ini_set('display_errors', 1);
}
- /*
- Multi-board support removes any use for this.
-
- if(!defined('IS_INSTALLATION')) {
- if(!file_exists(DIR_IMG)) @mkdir(DIR_IMG, 0777) or error("Couldn't create " . DIR_IMG . ". Install manually.", true);
- if(!file_exists(DIR_THUMB)) @mkdir(DIR_THUMB, 0777) or error("Couldn't create " . DIR_IMG . ". Install manually.", true);
- if(!file_exists(DIR_RES)) @mkdir(DIR_RES, 0777) or error("Couldn't create " . DIR_IMG . ". Install manually.", true);
- }
- */
?>
\ No newline at end of file
diff --git a/inc/database.php b/inc/database.php
new file mode 100644
index 00000000..21fabe71
--- /dev/null
+++ b/inc/database.php
@@ -0,0 +1,49 @@
+getMessage();
+
+ // Remove any sensitive information
+ $message = str_replace($config['db']['user'], 'hidden ', $message);
+ $message = str_replace($config['db']['password'], 'hidden ', $message);
+
+ // Print error
+ error('Database error: ' . $message);
+ }
+ }
+
+ function sql_close() {
+ global $pdo;
+ $pdo = NULL;
+ }
+
+ function prepare($query) {
+ global $pdo;
+ return $pdo->prepare($query);
+ }
+
+ function query($query) {
+ global $pdo;
+ return $pdo->query($query);
+ }
+
+ function db_error($PDOStatement=null) {
+ global $pdo;
+ if(isset($PDOStatement)) {
+ $err = $PDOStatement->errorInfo();
+ return $err[2];
+ } else {
+ $err = $pdo->errorInfo();
+ return $err[2];
+ }
+ }
+?>
\ No newline at end of file
diff --git a/inc/display.php b/inc/display.php
index f7e70290..acf77e39 100644
--- a/inc/display.php
+++ b/inc/display.php
@@ -20,27 +20,32 @@
}
function error($message) {
- global $board;
+ global $board, $mod, $config;
if(function_exists('sql_close')) sql_close();
die(Element('page.html', Array(
- 'index'=>ROOT,
+ 'index'=>$config['root'],
'title'=>'Error',
'subtitle'=>'An error has occured.',
'body'=>"
" .
"$message " .
- (isset($board) ? "Go back .
" : '').
+ (isset($board) ?
+ "Go back .
" : '').
" "
)));
}
function loginForm($error=false, $username=false) {
+ global $config;
+
if(function_exists('sql_close')) sql_close();
die(Element('page.html', Array(
- 'index'=>ROOT,
+ 'index'=>$config['root'],
'title'=>'Login',
'body'=>Element('login.html', Array(
- 'index'=>ROOT,
+ 'index'=>$config['root'],
'error'=>$error,
'username'=>$username
)
@@ -49,7 +54,10 @@
}
class Post {
- public function __construct($id, $thread, $subject, $email, $name, $trip, $body, $time, $thumb, $thumbx, $thumby, $file, $filex, $filey, $filesize, $filename) {
+ public function __construct($id, $thread, $subject, $email, $name, $trip, $body, $time, $thumb, $thumbx, $thumby, $file, $filex, $filey, $filesize, $filename, $ip, $root=null, $mod=false) {
+ global $config;
+ if(!isset($root)) $root = $config['root'];
+
$this->id = $id;
$this->thread = $thread;
$this->subject = utf8tohtml($subject);
@@ -66,12 +74,59 @@
$this->filey = $filey;
$this->filesize = $filesize;
$this->filename = $filename;
+ $this->ip = $ip;
+ $this->root = $root;
+ $this->mod = $mod;
+
+ if($this->mod)
+ // Fix internal links
+ // Very complicated regex
+ $this->body = preg_replace(
+ '/';
+
+ // Delete
+ if($this->mod['type'] >= $config['mod']['delete'])
+ $built .= ' ' . $config['mod']['link_delete'] . ' ';
+
+ // Delete all posts by IP
+ if($this->mod['type'] >= $config['mod']['deletebyip'])
+ $built .= ' ' . $config['mod']['link_deletebyip'] . ' ';
+
+ // Ban
+ if($this->mod['type'] >= $config['mod']['ban'])
+ $built .= ' ' . $config['mod']['link_ban'] . ' ';
+
+ // Ban & Delete
+ if($this->mod['type'] >= $config['mod']['bandelete'])
+ $built .= ' ' . $config['mod']['link_bandelete'] . ' ';
+
+ // Delete file (keep post)
+ if(!empty($this->file) && $this->mod['type'] >= $config['mod']['deletefile'])
+ $built .= ' ' . $config['mod']['link_deletefile'] . ' ';
+
+ $built .= '';
+ }
+ return $built;
+ }
+
public function build($index=false) {
- global $board;
+ global $board, $config;
$built = '' .
- '
';
+ '
' .
+ // Delete
+ '';
// Subject
if(!empty($this->subject))
@@ -84,39 +139,52 @@
// Trip
. (!empty($this->trip) ? ' '.$this->trip.' ':'');
+ // IP Address
+ if($this->mod && $this->mod['type'] >= $config['mod']['show_ip']) {
+ $built .= ' [' . $this->ip . ' ]';
+ }
+
// End email
if(!empty($this->email))
$built .= '';
// Date/time
- $built .= ' ' . date('m/d/y (D) H:i:s', $this->time);
+ $built .= ' ' . date($config['post_date'], $this->time);
+
+ // End delete
+ $built .= ' ';
$built .= ' No. ' .
+ ' href="' . $this->root . $board['dir'] . $config['dir']['res'] . $this->thread . '.html' . '#' . $this->id . '">No.' .
// JavaScript cite
- 'id . ');"') . 'href="' . ($index?ROOT . $board['dir'] . DIR_RES . $this->thread . '.html' . '#q' . $this->id:'javascript:void(0);') . '">'.$this->id.' ' .
+ ''.$this->id.' ' .
'
';
// File info
- if(!empty($this->file)) {
- $built .= '
File: ' . $this->file . ' (' .
+ if(!empty($this->file) && $this->file != 'deleted') {
+ $built .= 'File: ' . $this->file . ' (' .
// Filesize
format_bytes($this->filesize) . ', ' .
// File dimensions
$this->filex . 'x' . $this->filey;
// Aspect Ratio
- if(SHOW_RATIO) {
+ if($config['show_ratio']) {
$fraction = fraction($this->filex, $this->filey, ':');
$built .= ', ' . $fraction;
}
// Filename
- $built .= ', ' . $this->filename . ')
' .
+ $built .= ', ' . $this->filename . ')
' .
+
// Thumbnail
- '
';
+ '
';
+ } elseif($this->file == 'deleted') {
+ $built .= '
';
}
+ $built .= $this->postControls();
+
// Body
$built .= '
' . $this->body . '
';
@@ -126,7 +194,10 @@
class Thread {
public $omitted = 0;
- public function __construct($id, $subject, $email, $name, $trip, $body, $time, $thumb, $thumbx, $thumby, $file, $filex, $filey, $filesize, $filename) {
+ public function __construct($id, $subject, $email, $name, $trip, $body, $time, $thumb, $thumbx, $thumby, $file, $filex, $filey, $filesize, $filename, $ip, $sticky, $locked, $root=null, $mod=false) {
+ global $config;
+ if(!isset($root)) $root = $config['root'];
+
$this->id = $id;
$this->subject = utf8tohtml($subject);
$this->email = $email;
@@ -144,32 +215,91 @@
$this->filename = $filename;
$this->omitted = 0;
$this->posts = Array();
+ $this->ip = $ip;
+ $this->sticky = $sticky;
+ $this->locked = $locked;
+ $this->root = $root;
+ $this->mod = $mod;
+
+ if($this->mod)
+ // Fix internal links
+ // Very complicated regex
+ $this->body = preg_replace(
+ '/';
+
+ // Delete
+ if($this->mod['type'] >= $config['mod']['delete'])
+ $built .= ' ' . $config['mod']['link_delete'] . ' ';
+
+ // Delete all posts by IP
+ if($this->mod['type'] >= $config['mod']['deletebyip'])
+ $built .= ' ' . $config['mod']['link_deletebyip'] . ' ';
+
+ // Ban
+ if($this->mod['type'] >= $config['mod']['ban'])
+ $built .= ' ' . $config['mod']['link_ban'] . ' ';
+
+ // Ban & Delete
+ if($this->mod['type'] >= $config['mod']['bandelete'])
+ $built .= ' ' . $config['mod']['link_bandelete'] . ' ';
+
+ // Stickies
+ if($this->mod['type'] >= $config['mod']['sticky'])
+ if($this->sticky)
+ $built .= ' ' . $config['mod']['link_desticky'] . ' ';
+ else
+ $built .= ' ' . $config['mod']['link_sticky'] . ' ';
+
+ // Lock
+ if($this->mod['type'] >= $config['mod']['lock'])
+ if($this->locked)
+ $built .= ' ' . $config['mod']['link_unlock'] . ' ';
+ else
+ $built .= ' ' . $config['mod']['link_lock'] . ' ';
+
+
+ $built .= '';
+ }
+ return $built;
+ }
public function build($index=false) {
- global $board;
+ global $board, $config;
- $built = 'File: ' . $this->file . ' (' .
+ $built = 'File: ' . $this->file . ' (' .
// Filesize
format_bytes($this->filesize) . ', ' .
// File dimensions
$this->filex . 'x' . $this->filey;
// Aspect Ratio
- if(SHOW_RATIO) {
+ if($config['show_ratio']) {
$fraction = fraction($this->filex, $this->filey, ':');
$built .= ', ' . $fraction;
}
// Filename
$built .= ', ' . $this->filename . ')
' .
// Thumbnail
- ' ';
+ ' ';
$built .= '';
+ // Delete
+ $built .= '';
+
// Subject
if(!empty($this->subject))
$built .= '' . $this->subject . ' ';
@@ -181,23 +311,38 @@
// Trip
. (!empty($this->trip) ? ' '.$this->trip.' ':'');
+ // IP Address
+ if($this->mod && $this->mod['type'] >= $config['mod']['show_ip']) {
+ $built .= ' [' . $this->ip . ' ]';
+ }
+
// End email
if(!empty($this->email))
$built .= '';
// Date/time
- $built .= ' ' . date('m/d/y (D) H:i:s', $this->time);
+ $built .= ' ' . date($config['post_date'], $this->time);
+
+ // End delete
+ $built .= ' ';
$built .= ' No. ' .
+ ' href="' . $this->root . $board['dir'] . $config['dir']['res'] . $this->id . '.html' . '#' . $this->id . '">No.' .
// JavaScript cite
- 'id . ');"') . 'href="' . ($index?ROOT . $board['dir'] . DIR_RES . $this->id . '.html' . '#q' . $this->id:'javascript:void(0);') . '">'.$this->id.' ' .
+ ''.$this->id.' ' .
+ // Sticky
+ ($this->sticky ? ' ' : '') .
+ // Locked
+ ($this->locked ? ' ' : '') .
// [Reply]
- ($index ? '[Reply] ' : '') .
+ ($index ? '[Reply] ' : '') .
+
+ // Mod controls
+ $this->postControls() .
'
';
-
+
// Body
$built .= $this->body .
diff --git a/inc/functions.php b/inc/functions.php
index 7cb934fc..cc2a02f8 100644
--- a/inc/functions.php
+++ b/inc/functions.php
@@ -7,26 +7,9 @@
return str_replace(array_keys($replaces),
array_values($replaces), $str);
}
-
- function sql_open() {
- global $sql;
- $sql = @mysql_connect(MY_SERVER, MY_USER, MY_PASSWORD) or error('Database error.');
- @mysql_select_db(MY_DATABASE, $sql) or error('Database error.');
- }
-
- function sql_close() {
- global $sql;
- @mysql_close($sql);
- }
-
- function mysql_safe_array(&$array) {
- foreach($array as &$item) {
- $item = mysql_real_escape_string($item);
- }
- }
function setupBoard($array) {
- global $board;
+ global $board, $config;
$board = Array(
'id' => $array['id'],
@@ -34,166 +17,514 @@
'name' => $array['title'],
'title' => $array['subtitle']);
- $board['dir'] = sprintf(BOARD_PATH, $board['uri']);
- $board['url'] = sprintf(BOARD_ABBREVIATION, $board['uri']);
+ $board['dir'] = sprintf($config['board_path'], $board['uri']);
+ $board['url'] = sprintf($config['board_abbreviation'], $board['uri']);
if(!file_exists($board['dir'])) mkdir($board['dir'], 0777);
- if(!file_exists($board['dir'] . DIR_IMG)) @mkdir($board['dir'] . DIR_IMG, 0777) or error("Couldn't create " . DIR_IMG . ". Check permissions.", true);
- if(!file_exists($board['dir'] . DIR_THUMB)) @mkdir($board['dir'] . DIR_THUMB, 0777) or error("Couldn't create " . DIR_THUMB . ". Check permissions.", true);
- if(!file_exists($board['dir'] . DIR_RES)) @mkdir($board['dir'] . DIR_RES, 0777) or error("Couldn't create " . DIR_RES . ". Check permissions.", true);
+ if(!file_exists($board['dir'] . $config['dir']['img'])) @mkdir($board['dir'] . $config['dir']['img'], 0777) or error("Couldn't create " . $config['dir']['img'] . ". Check permissions.", true);
+ if(!file_exists($board['dir'] . $config['dir']['thumb'])) @mkdir($board['dir'] . $config['dir']['thumb'], 0777) or error("Couldn't create " . $config['dir']['thumb'] . ". Check permissions.", true);
+ if(!file_exists($board['dir'] . $config['dir']['res'])) @mkdir($board['dir'] . $config['dir']['res'], 0777) or error("Couldn't create " . $config['dir']['res'] . ". Check permissions.", true);
}
function openBoard($uri) {
- global $sql;
- $boards_res = mysql_query(sprintf(
- "SELECT * FROM `boards` WHERE `uri` = '%s' LIMIT 1",
- mysql_real_escape_string($uri)
- ), $sql) or error(mysql_error($sql));
+ sql_open();
- if($_board = mysql_fetch_array($boards_res)) {
- setupBoard($_board);
+ $query = prepare("SELECT * FROM `boards` WHERE `uri` = :uri LIMIT 1");
+ $query->bindValue(':uri', $uri);
+ $query->execute() or error(db_error($query));
+
+ if($board = $query->fetch()) {
+ setupBoard($board);
return true;
} else return false;
}
- function threadExists($id) {
- global $sql;
- $thread_res = mysql_query(sprintf(
- "SELECT 1 FROM `posts` WHERE `id` = '%d' AND `thread` IS NULL LIMIT 1",
- $id
- ), $sql) or error(mysql_error($sql));
+ function listBoards() {
+ $query = query("SELECT * FROM `boards` ORDER BY `uri`") or error(db_error());
+ $boards = $query->fetchAll();
+ return $boards;
+ }
+
+ function checkFlood($post) {
+ global $board, $config;
- if(mysql_num_rows($thread_res) > 0) {
+ $query = prepare(sprintf("SELECT * FROM `posts_%s` WHERE (`ip` = :ip AND `time` >= :floodtime) OR (`ip` = :ip AND `body` = :body AND `time` >= :floodsameiptime) OR (`body` = :body AND `time` >= :floodsametime) LIMIT 1", $board['uri']));
+ $query->bindValue(':ip', $_SERVER['REMOTE_ADDR']);
+ $query->bindValue(':body', $post['body'], PDO::PARAM_INT);
+ $query->bindValue(':floodtime', time()-$config['flood_time'], PDO::PARAM_INT);
+ $query->bindValue(':floodsameiptime', time()-$config['flood_time_ip'], PDO::PARAM_INT);
+ $query->bindValue(':floodsametime', time()-$config['flood_time_same'], PDO::PARAM_INT);
+ $query->execute() or error(db_error($query));
+
+ return (bool)$query->fetch();
+ }
+
+ function until($timestamp) {
+ $difference = $timestamp - time();
+ if($difference < 60) {
+ return $difference . ' second' . ($difference != 1 ? 's' : '');
+ } elseif($difference < 60*60) {
+ return ($num = round($difference/(60))) . ' minute' . ($num != 1 ? 's' : '');
+ } elseif($difference < 60*60*24) {
+ return ($num = round($difference/(60*60))) . ' hour' . ($num != 1 ? 's' : '');
+ } elseif($difference < 60*60*24*7) {
+ return ($num = round($difference/(60*60*24))) . ' day' . ($num != 1 ? 's' : '');
+ } elseif($difference < 60*60*24*365) {
+ return ($num = round($difference/(60*60*24*7))) . ' week' . ($num != 1 ? 's' : '');
+ } else {
+ return ($num = round($difference/(60*60*24*365))) . ' year' . ($num != 1 ? 's' : '');
+ }
+ }
+
+ function formatDate($timestamp) {
+ return date('jS F, Y', $timestamp);
+ }
+
+ function checkBan() {
+ global $config;
+
+ if(!isset($_SERVER['REMOTE_ADDR'])) {
+ // Server misconfiguration
+ return;
+ }
+
+ $query = prepare("SELECT * FROM `bans` WHERE `ip` = :ip LIMIT 1");
+ $query->bindValue(':ip', $_SERVER['REMOTE_ADDR']);
+ $query->execute() or error(db_error($query));
+
+ if($ban = $query->fetch()) {
+ if($ban['expires'] && $ban['expires'] < time()) {
+ // Ban expired
+ $query = prepare("DELETE FROM `bans` WHERE `ip` = :ip AND `expires` = :expires LIMIT 1");
+ $query->bindValue(':ip', $_SERVER['REMOTE_ADDR']);
+ $query->bindValue(':expires', $ban['expires'], PDO::PARAM_INT);
+ $query->execute() or error(db_error($query));
+ return;
+ }
+ $body = '
+
You are banned! ;_;
+
You have been banned ' .
+ ($ban['reason'] ? 'for the following reason:' : 'for an unspecified reason.') .
+ '
' .
+ ($ban['reason'] ?
+ '
' .
+ $ban['reason'] .
+ '
'
+ : '') .
+ '
Your ban was filed on ' .
+ formatDate($ban['set']) .
+ ' , and ' .
+ ($ban['expires'] ?
+ 'expires ' . until($ban['expires']) . ' from now, which is on ' .
+ formatDate($ban['expires']) .
+ '
+ '
+ : 'will not expire .' ) .
+ '
+
Your IP address is ' . $_SERVER['REMOTE_ADDR'] . ' .
+
';
+
+ // Show banned page and exit
+ die(Element('page.html', Array(
+ 'index' => $config['root'],
+ 'title' => 'Banned',
+ 'subtitle' => 'You are banned!',
+ 'body' => $body
+ )
+ ));
+ }
+ }
+
+ function threadLocked($id) {
+ global $board;
+
+ $query = prepare(sprintf("SELECT `locked` FROM `posts_%s` WHERE `id` = :id AND `thread` IS NULL LIMIT 1", $board['uri']));
+ $query->bindValue(':id', $id, PDO::PARAM_INT);
+ $query->execute() or error(db_error());
+
+ if(!$post = $query->fetch()) {
+ // Non-existant, so it can't be locked...
+ return false;
+ }
+
+ return (bool) $post['locked'];
+ }
+
+ function threadExists($id) {
+ global $board;
+
+ $query = prepare(sprintf("SELECT 1 FROM `posts_%s` WHERE `id` = :id AND `thread` IS NULL LIMIT 1", $board['uri']));
+ $query->bindValue(':id', $id, PDO::PARAM_INT);
+ $query->execute() or error(db_error());
+
+ if($query->rowCount()) {
return true;
} else return false;
}
function post($post, $OP) {
- global $sql, $board;
- if($OP) {
- mysql_query(
- sprintf("INSERT INTO `posts` VALUES ( NULL, '%d', NULL, '%s', '%s', '%s', '%s', '%s', '%d', '%d', '%s', '%d', '%d', '%s', '%d', '%d', '%d', '%s', '%s', '%s', '%s' )",
- $board['id'],
- $post['subject'],
- $post['email'],
- $post['name'],
- $post['trip'],
- $post['body'],
- time(),
- time(),
- $post['thumb'],
- $post['thumbwidth'],
- $post['thumbheight'],
- $post['file'],
- $post['width'],
- $post['height'],
- $post['filesize'],
- $post['filename'],
- $post['filehash'],
- $post['password'],
- mysql_real_escape_string($_SERVER['REMOTE_ADDR'])
- ), $sql) or error(mysql_error($sql));
- return mysql_insert_id($sql);
+ global $pdo, $board;
+
+ $query = prepare(sprintf("INSERT INTO `posts_%s` VALUES ( NULL, :thread, :subject, :email, :name, :trip, :body, :time, :time, :thumb, :thumbwidth, :thumbheight, :file, :width, :height, :filesize, :filename, :filehash, :password, :ip, :sticky, :locked)", $board['uri']));
+
+ // Basic stuff
+ $query->bindValue(':subject', $post['subject']);
+ $query->bindValue(':email', $post['email']);
+ $query->bindValue(':name', $post['name']);
+ $query->bindValue(':trip', $post['trip']);
+ $query->bindValue(':body', $post['body']);
+ $query->bindValue(':time', time(), PDO::PARAM_INT);
+ $query->bindValue(':password', $post['password']);
+ $query->bindValue(':ip', $_SERVER['REMOTE_ADDR']);
+
+ if($post['mod'] && $post['sticky']) {
+ $query->bindValue(':sticky', 1, PDO::PARAM_INT);
} else {
- mysql_query(
- sprintf("INSERT INTO `posts` VALUES ( NULL, '%d', '%d', '%s', '%s', '%s', '%s', '%s', '%d', '%d', '%s', '%d', '%d', '%s', '%d', '%d', '%d', '%s', '%s', '%s', '%s' )",
- $board['id'],
- $post['thread'],
- $post['subject'],
- $post['email'],
- $post['name'],
- $post['trip'],
- $post['body'],
- time(),
- time(),
- $post['has_file']?$post['thumb']:null,
- $post['has_file']?$post['thumbwidth']:null,
- $post['has_file']?$post['thumbheight']:null,
- $post['has_file']?$post['file']:null,
- $post['has_file']?$post['width']:null,
- $post['has_file']?$post['height']:null,
- $post['has_file']?$post['filesize']:null,
- $post['has_file']?$post['filename']:null,
- $post['has_file']?$post['filehash']:null,
- $post['password'],
- mysql_real_escape_string($_SERVER['REMOTE_ADDR'])
- ), $sql) or error(mysql_error($sql));
- return mysql_insert_id($sql);
+ $query->bindValue(':sticky', 0, PDO::PARAM_INT);
+ }
+
+ if($post['mod'] && $post['locked']) {
+ $query->bindValue(':locked', 1, PDO::PARAM_INT);
+ } else {
+ $query->bindValue(':locked', 0, PDO::PARAM_INT);
+ }
+
+ if($OP) {
+ // No parent thread, image
+ $query->bindValue(':thread', null, PDO::PARAM_NULL);
+ } else {
+ $query->bindValue(':thread', $post['thread'], PDO::PARAM_INT);
+ }
+
+ if($post['has_file']) {
+ $query->bindValue(':thumb', $post['thumb']);
+ $query->bindValue(':thumbwidth', $post['thumbwidth'], PDO::PARAM_INT);
+ $query->bindValue(':thumbheight', $post['thumbheight'], PDO::PARAM_INT);
+ $query->bindValue(':file', $post['file']);
+ $query->bindValue(':width', $post['width'], PDO::PARAM_INT);
+ $query->bindValue(':height', $post['height'], PDO::PARAM_INT);
+ $query->bindValue(':filesize', $post['filesize'], PDO::PARAM_INT);
+ $query->bindValue(':filename', $post['filename']);
+ $query->bindValue(':filehash', $post['filehash']);
+ } else {
+ $query->bindValue(':thumb', null, PDO::PARAM_NULL);
+ $query->bindValue(':thumbwidth', null, PDO::PARAM_NULL);
+ $query->bindValue(':thumbheight', null, PDO::PARAM_NULL);
+ $query->bindValue(':file', null, PDO::PARAM_NULL);
+ $query->bindValue(':width', null, PDO::PARAM_NULL);
+ $query->bindValue(':height', null, PDO::PARAM_NULL);
+ $query->bindValue(':filesize', null, PDO::PARAM_NULL);
+ $query->bindValue(':filename', null, PDO::PARAM_NULL);
+ $query->bindValue(':filehash', null, PDO::PARAM_NULL);
+ }
+
+ $query->execute() or error(db_error($query));
+
+ return $pdo->lastInsertId();
+ }
+
+ function bumpThread($id) {
+ global $board;
+ $query = prepare(sprintf("UPDATE `posts_%s` SET `bump` = :time WHERE `id` = :id AND `thread` IS NULL", $board['uri']));
+ $query->bindValue(':time', time(), PDO::PARAM_INT);
+ $query->bindValue(':id', $id, PDO::PARAM_INT);
+ $query->execute() or error(db_error($query));
+ }
+
+ // Remove file from post
+ function deleteFile($id, $remove_entirely_if_already=true) {
+ global $board, $config;
+
+ $query = prepare(sprintf("SELECT `thread`,`thumb`,`file` FROM `posts_%s` WHERE `id` = :id AND `thread` IS NOT NULL LIMIT 1", $board['uri']));
+ $query->bindValue(':id', $id, PDO::PARAM_INT);
+ $query->execute() or error(db_error($query));
+
+ if($query->rowCount() < 1) {
+ error($config['error']['invalidpost']);
+ }
+
+ $post = $query->fetch();
+
+ $query = prepare(sprintf("UPDATE `posts_%s` SET `thumb` = NULL, `thumbwidth` = NULL, `thumbheight` = NULL, `filewidth` = NULL, `fileheight` = NULL, `filesize` = NULL, `filename` = NULL, `filehash` = NULL, `file` = :file WHERE `id` = :id OR `thread` = :id", $board['uri']));
+ if($post['file'] == 'deleted' && $remove_entirely_if_already) {
+ // Already deleted; remove file fully
+ $query->bindValue(':file', null, PDO::PARAM_NULL);
+ } else {
+ // Delete thumbnail
+ @unlink($board['dir'] . $config['dir']['thumb'] . $post['thumb']);
+
+ // Delete file
+ @unlink($board['dir'] . $config['dir']['img'] . $post['file']);
+
+ // Set file to 'deleted'
+ $query->bindValue(':file', 'deleted', PDO::PARAM_INT);
+ }
+ // Update database
+
+ $query->bindValue(':id', $id, PDO::PARAM_INT);
+ $query->execute() or error(db_error($query));
+
+ buildThread($post['thread']);
+ }
+
+ // Delete a post (reply or thread)
+ function deletePost($id, $error_if_doesnt_exist=true) {
+ global $board, $config;
+
+ // Select post and replies (if thread) in one query
+ $query = prepare(sprintf("SELECT `id`,`thread`,`thumb`,`file` FROM `posts_%s` WHERE `id` = :id OR `thread` = :id", $board['uri']));
+ $query->bindValue(':id', $id, PDO::PARAM_INT);
+ $query->execute() or error(db_error($query));
+
+ if($query->rowCount() < 1) {
+ if($error_if_doesnt_exist)
+ error($config['error']['invalidpost']);
+ else return false;
+ }
+
+ // Delete posts and maybe replies
+ while($post = $query->fetch()) {
+ if(!$post['thread']) {
+ // Delete thread HTML page
+ @unlink($board['dir'] . $config['dir']['res'] . sprintf($config['file_page'], $post['id']));
+ } elseif($query->rowCount() == 1) {
+ // Rebuild thread
+ $rebuild = $post['thread'];
+ }
+ if($post['thumb']) {
+ // Delete thumbnail
+ @unlink($board['dir'] . $config['dir']['thumb'] . $post['thumb']);
+ }
+ if($post['file']) {
+ // Delete file
+ @unlink($board['dir'] . $config['dir']['img'] . $post['file']);
+ }
+ }
+
+ $query = prepare(sprintf("DELETE FROM `posts_%s` WHERE `id` = :id OR `thread` = :id", $board['uri']));
+ $query->bindValue(':id', $id, PDO::PARAM_INT);
+ $query->execute() or error(db_error($query));
+
+ if(isset($rebuild)) {
+ buildThread($rebuild);
+ }
+
+ return true;
+ }
+
+ function clean() {
+ global $board, $config;
+ $offset = round($config['max_pages']*$config['threads_per_page']);
+
+ // I too wish there was an easier way of doing this...
+ $query = prepare(sprintf("SELECT `id` FROM `posts_%s` WHERE `thread` IS NULL ORDER BY `sticky` DESC, `bump` DESC LIMIT :offset, 9001", $board['uri']));
+ $query->bindValue(':offset', $offset, PDO::PARAM_INT);
+
+ $query->execute() or error(db_error($query));
+ while($post = $query->fetch()) {
+ deletePost($post['id']);
}
}
-
- function index($page) {
- global $sql, $board;
+
+ function index($page, $mod=false) {
+ global $board, $config;
$body = '';
- $offset = round($page*THREADS_PER_PAGE-THREADS_PER_PAGE);
+ $offset = round($page*$config['threads_per_page']-$config['threads_per_page']);
sql_open();
- $query = mysql_query(sprintf(
- "SELECT * FROM `posts` WHERE `thread` IS NULL AND `board` = '%d' ORDER BY `bump` DESC LIMIT %d,%d",
- $board['id'],
- $offset,
- THREADS_PER_PAGE
- ), $sql) or error(mysql_error($sql));
-
- if(mysql_num_rows($query) < 1 && $page > 1) return false;
- while($th = mysql_fetch_array($query)) {
- $thread = new Thread($th['id'], $th['subject'], $th['email'], $th['name'], $th['trip'], $th['body'], $th['time'], $th['thumb'], $th['thumbwidth'], $th['thumbheight'], $th['file'], $th['filewidth'], $th['fileheight'], $th['filesize'], $th['filename']);
+
+ $query = prepare(sprintf("SELECT * FROM `posts_%s` WHERE `thread` IS NULL ORDER BY `sticky` DESC, `bump` DESC LIMIT ?,?", $board['uri']));
+ $query->bindValue(1, $offset, PDO::PARAM_INT);
+ $query->bindValue(2, $config['threads_per_page'], PDO::PARAM_INT);
+ $query->execute() or error(db_error($query));
+
+ if($query->rowcount() < 1 && $page > 1) return false;
+ while($th = $query->fetch()) {
+ $thread = new Thread($th['id'], $th['subject'], $th['email'], $th['name'], $th['trip'], $th['body'], $th['time'], $th['thumb'], $th['thumbwidth'], $th['thumbheight'], $th['file'], $th['filewidth'], $th['fileheight'], $th['filesize'], $th['filename'], $th['ip'], $th['sticky'], $th['locked'], $mod ? '?/' : $config['root'], $mod);
- $newposts = mysql_query(sprintf(
- "SELECT `id`, `subject`, `email`, `name`, `trip`, `body`, `time`, `thumb`, `thumbwidth`, `thumbheight`, `file`, `filewidth`, `fileheight`, `filesize`, `filename` FROM `posts` WHERE `board` = '%d' AND `thread` = '%s' ORDER BY `time` DESC LIMIT %d",
- $board['id'],
- $th['id'],
- THREADS_PREVIEW
- ), $sql) or error(mysql_error($sql));
- if(mysql_num_rows($newposts) == THREADS_PREVIEW) {
- $count_query = mysql_query(sprintf(
- "SELECT COUNT(`id`) as `num` FROM `posts` WHERE `board` = '%d' AND `thread` = '%s'",
- $board['id'],
- $th['id']
- ), $sql) or error(mysql_error($sql));
- $count = mysql_fetch_array($count_query);
- $omitted = $count['num'] - THREADS_PREVIEW;
+ $posts = prepare(sprintf("SELECT `id`, `subject`, `email`, `name`, `trip`, `body`, `time`, `thumb`, `thumbwidth`, `thumbheight`, `file`, `filewidth`, `fileheight`, `filesize`, `filename`,`ip` FROM `posts_%s` WHERE `thread` = ? ORDER BY `id` DESC LIMIT ?", $board['uri']));
+ $posts->bindValue(1, $th['id']);
+ $posts->bindValue(2, $config['threads_preview'], PDO::PARAM_INT);
+ $posts->execute() or error(db_error($posts));
+
+ if($posts->rowCount() == $config['threads_preview']) {
+ $count = prepare(sprintf("SELECT COUNT(`id`) as `num` FROM `posts_%s` WHERE `thread` = ?", $board['uri']));
+ $count->bindValue(1, $th['id']);
+ $count->execute() or error(db_error($count));
+
+ $count = $count->fetch();
+ $omitted = $count['num'] - $config['threads_preview'];
$thread->omitted = $omitted;
- mysql_free_result($count_query);
unset($count);
unset($omitted);
}
- while($po = mysql_fetch_array($newposts)) {
- $thread->add(new Post($po['id'], $th['id'], $po['subject'], $po['email'], $po['name'], $po['trip'], $po['body'], $po['time'], $po['thumb'], $po['thumbwidth'], $po['thumbheight'], $po['file'], $po['filewidth'], $po['fileheight'], $po['filesize'], $po['filename']));
+
+ while($po = $posts->fetch()) {
+ $thread->add(new Post($po['id'], $th['id'], $po['subject'], $po['email'], $po['name'], $po['trip'], $po['body'], $po['time'], $po['thumb'], $po['thumbwidth'], $po['thumbheight'], $po['file'], $po['filewidth'], $po['fileheight'], $po['filesize'], $po['filename'], $po['ip'], $mod ? '?/' : $config['root'], $mod));
}
- mysql_free_result($newposts);
$thread->posts = array_reverse($thread->posts);
$body .= $thread->build(true);
}
- mysql_free_result($query);
- return Array('button'=>BUTTON_NEWTOPIC, 'board'=>$board, 'body'=>$body, 'post_url' => POST_URL, 'index' => ROOT);
+
+ return Array('button'=>$config['button_newtopic'], 'board'=>$board, 'body'=>$body, 'post_url' => $config['post_url'], 'index' => $config['root']);
}
-
- function buildIndex() {
- global $sql, $board;
- sql_open();
-
- $res = mysql_query(sprintf(
- "SELECT COUNT(`id`) as `num` FROM `posts` WHERE `board` = '%d' AND `thread` IS NULL",
- $board['id']
- ), $sql) or error(mysql_error($sql));
- $arr = mysql_fetch_array($res);
- $count = floor((THREADS_PER_PAGE + $arr['num'] - 1) / THREADS_PER_PAGE);
+
+ function getPages($mod=false) {
+ global $board, $config;
+
+ // Count threads
+ $query = query(sprintf("SELECT COUNT(`id`) as `num` FROM `posts_%s` WHERE `thread` IS NULL", $board['uri'])) or error(db_error());
+
+ $count = current($query->fetch());
+ $count = floor(($config['threads_per_page'] + $count - 1) / $config['threads_per_page']);
$pages = Array();
- for($x=0;$x<$count && $x
$x+1, 'link' => $x==0 ? ROOT . $board['dir'] . FILE_INDEX : ROOT . $board['dir'] . sprintf(FILE_PAGE, $x+1));
- }
+ for($x=0;$x<$count && $x<$config['max_pages'];$x++) {
+ $pages[] = Array('num' => $x+1, 'link' => $x==0 ? ($mod ? '?/' : $config['root']) . $board['dir'] . $config['file_index'] : ($mod ? '?/' : $config['root']) . $board['dir'] . sprintf($config['file_page'], $x+1));
+ }
+
+ return $pages;
+ }
+
+ function makerobot($body) {
+ global $config;
+ $body = strtolower($body);
+
+ // Leave only letters
+ $body = preg_replace('/[^a-z]/i', '', $body);
+ // Remove repeating characters
+ if($config['robot_strip_repeating'])
+ $body = preg_replace('/(.)\\1+/', '$1', $body);
+
+ return sha1($body);
+ }
+
+ function checkRobot($body) {
+ /* CREATE TABLE `robot` (
+`hash` VARCHAR( 40 ) NOT NULL COMMENT 'SHA1'
+) ENGINE = INNODB; */
+ /* CREATE TABLE `mutes` (
+`ip` VARCHAR( 15 ) NOT NULL ,
+`time` INT NOT NULL
+) ENGINE = MYISAM ; */
- mysql_free_result($res);
- unset($arr);
- unset($count);
+ $body = makerobot($body);
+ $query = prepare("SELECT 1 FROM `robot` WHERE `hash` = :hash LIMIT 1");
+ $query->bindValue(':hash', $body);
+ $query->execute() or error(db_error($query));
+
+ if($query->fetch()) {
+ return true;
+ } else {
+ // Insert new hash
+
+ $query = prepare("INSERT INTO `robot` VALUES (:hash)");
+ $query->bindValue(':hash', $body);
+ $query->execute() or error(db_error($query));
+ return false;
+ }
+ }
+
+ function numPosts($id) {
+ global $board;
+ $query = prepare(sprintf("SELECT COUNT(*) as `count` FROM `posts_%s` WHERE `thread` = :thread", $board['uri']));
+ $query->bindValue(':thread', $id, PDO::PARAM_INT);
+ $query->execute() or error(db_error($query));
+
+ $result = $query->fetch();
+ return $result['count'];
+ }
+
+ function muteTime() {
+ global $config;
+ // Find number of mutes in the past X hours
+ $query = prepare("SELECT COUNT(*) as `count` FROM `mutes` WHERE `time` >= :time AND `ip` = :ip");
+ $query->bindValue(':time', time()-(ROBOT_MUTE_HOUR*3600), PDO::PARAM_INT);
+ $query->bindValue(':ip', $_SERVER['REMOTE_ADDR']);
+ $query->execute() or error(db_error($query));
+
+ $result = $query->fetch();
+ if($result['count'] == 0) return 0;
+ return pow($config['robot_mute_multiplier'], $result['count']);
+ }
+
+ function mute() {
+ // Insert mute
+ $query = prepare("INSERT INTO `mutes` VALUES (:ip, :time)");
+ $query->bindValue(':time', time(), PDO::PARAM_INT);
+ $query->bindValue(':ip', $_SERVER['REMOTE_ADDR']);
+ $query->execute() or error(db_error($query));
+
+ return muteTime();
+ }
+
+ function checkMute() {
+ $mutetime = muteTime();
+ if($mutetime > 0) {
+ // Find last mute time
+ $query = prepare("SELECT `time` FROM `mutes` WHERE `ip` = :ip ORDER BY `time` DESC LIMIT 1");
+ $query->bindValue(':ip', $_SERVER['REMOTE_ADDR']);
+ $query->execute() or error(db_error($query));
+
+ if(!$mute = $query->fetch()) {
+ // What!? He's muted but he's not muted...
+ return;
+ }
+
+ if($mute['time'] + $mutetime > time()) {
+ // Not expired yet
+ error(sprintf($config['error']['youaremuted'], $mute['time'] + $mutetime - time()));
+ } else {
+ // Already expired
+ return;
+ }
+ }
+ }
+
+ function buildIndex() {
+ global $board, $config;
+ sql_open();
+
+ $pages = getPages();
$page = 1;
- while($page <= MAX_PAGES && $content = index($page)) {
- $filename = $board['dir'] . ($page==1 ? FILE_INDEX : sprintf(FILE_PAGE, $page));
+ while($page <= $config['max_pages'] && $content = index($page)) {
+ $filename = $board['dir'] . ($page==1 ? $config['file_index'] : sprintf($config['file_page'], $page));
if(file_exists($filename)) $md5 = md5_file($filename);
$content['pages'] = $pages;
@@ -204,18 +535,52 @@
}
$page++;
}
- if($page < MAX_PAGES) {
- for(;$page<=MAX_PAGES;$page++) {
- $filename = $page==1 ? FILE_INDEX : sprintf(FILE_PAGE, $page);
+ if($page < $config['max_pages']) {
+ for(;$page<=$config['max_pages'];$page++) {
+ $filename = $page==1 ? $config['file_index'] : sprintf($config['file_page'], $page);
@unlink($filename);
}
}
}
+
+ function isDNSBL() {
+ $dns_black_lists = file('./dnsbl.txt', FILE_IGNORE_NEW_LINES);
+
+ // Reverse the IP
+ $rev_ip = implode(array_reverse(explode('.', $_SERVER['REMOTE_ADDR'])), '.');
+ $response = array();
+ foreach ($dns_black_lists as $dns_black_list) {
+ $response = (gethostbynamel($rev_ip . '.' . $dns_black_list));
+ if(!empty($response))
+ return true;
+ }
+
+ return false;
+ }
+
+ function isTor() {
+ return gethostbyname(
+ ReverseIPOctets($_SERVER['REMOTE_ADDR']) . '.' . $_SERVER['SERVER_PORT'] . '.' . ReverseIPOctets($_SERVER['SERVER_ADDR']) . '.ip-port.exitlist.torproject.org'
+ ) == '127.0.0.2';
+ }
+
+ function ReverseIPOctets($inputip) {
+ $ipoc = explode('.', $inputip);
+ return $ipoc[3] . '.' . $ipoc[2] . '.' . $ipoc[1] . '.' . $ipoc[0];
+ }
function markup(&$body) {
- global $sql, $board;
-
- if(AUTO_UNICODE) {
+ global $board, $config;
+
+ $body = utf8tohtml($body, true);
+
+ if($config['markup_urls']) {
+ $body = preg_replace($config['url_regex'], "$0 ", $body, -1, $num_links);
+ if($num_links > $config['max_links'])
+ error($config['error']['toomanylinks']);
+ }
+
+ if($config['auto_unicode']) {
$body = str_replace('...', '…', $body);
$body = str_replace('<--', '←', $body);
@@ -226,8 +591,6 @@
$body = str_replace('--', '–', $body); // en dash
}
- $body = utf8tohtml($body, true);
-
// Cites
if(preg_match_all('/(^|\s)>>([0-9]+?)(\s|$)/', $body, $cites)) {
$previousPosition = 0;
@@ -239,43 +602,43 @@
strlen($cites[1][$index]),
strlen($cites[3][$index]),
);
-
- $result = mysql_query(sprintf(
- "SELECT `thread`,`id` FROM `posts` WHERE `board` = '%d' AND `id` = '%d' LIMIT 1",
- $board['id'],
- $cite
- ), $sql) or error(mysql_error($sql));
- if($post = mysql_fetch_array($result)) {
- $replacement = '>>' . $cite . ' ';
+ $query = prepare(sprintf("SELECT `thread`,`id` FROM `posts_%s` WHERE `id` = :id LIMIT 1", $board['uri']));
+ $query->bindValue(':id', $cite);
+ $query->execute() or error(db_error($query));
+
+ if($post = $query->fetch()) {
+ $replacement = '>>' . $cite . ' ';
} else {
$replacement = ">>{$cite}";
}
- mysql_free_result($result);
// Find the position of the cite
$position = strpos($body, $cites[0][$index]);
+
+
+
// Replace the found string with "xxxx[...]". (allows duplicate tags). Keeps whitespace.
$body = substr_replace($body, str_repeat('x', strlen($cites[0][$index]) - $whitespace[0] - $whitespace[1]), $position + $whitespace[0], strlen($cites[0][$index]) - $whitespace[0] - $whitespace[1]);
-
+
$temp .= substr($body, $previousPosition, $position-$previousPosition) . $cites[1][$index] . $replacement . $cites[3][$index];
$previousPosition = $position+strlen($cites[0][$index]);
}
+
// The rest
$temp .= substr($body, $previousPosition);
-
+
$body = $temp;
}
$body = str_replace("\r", '', $body);
-
- if(MARKUP_URLS)
- $body = preg_replace(URL_REGEX, "$0 ", $body);
+
$body = preg_replace("/(^|\n)([\s]+)?(>)([^\n]+)?($|\n)/m", '$1$2$3$4 $5', $body);
- if(WIKI_MARKUP) {
+ if($config['wiki_markup']) {
$body = preg_replace("/(^|\n)==(.+?)==\n?/m", "$2 ", $body);
$body = preg_replace("/'''(.+?)'''/m", "$1 ", $body);
$body = preg_replace("/''(.+?)''/m", "$1 ", $body);
+ $body = preg_replace("/\*\*(.+?)\*\*/m", "$1 ", $body);
}
$body = preg_replace("/\n/", ' ', $body);
}
@@ -319,41 +682,44 @@
return $result;
}
- function buildThread($id, $return=false) {
- global $sql, $board;
+ function buildThread($id, $return=false, $mod=false) {
+ global $board, $config;
$id = round($id);
-
- $query = mysql_query(sprintf(
- "SELECT `id`,`thread`,`subject`,`name`,`email`,`trip`,`body`,`time`,`thumb`,`thumbwidth`,`thumbheight`,`file`,`filewidth`,`fileheight`,`filesize`,`filename` FROM `posts` WHERE `board` = '%d' AND ((`thread` IS NULL AND `id` = '%s') OR `thread` = '%s') ORDER BY `thread`,`time`",
- $board['id'],
- $id,
- $id
- ), $sql) or error(mysql_error($sql));
-
- while($post = mysql_fetch_array($query)) {
+
+ $query = prepare(sprintf("SELECT `id`,`thread`,`subject`,`name`,`email`,`trip`,`body`,`time`,`thumb`,`thumbwidth`,`thumbheight`,`file`,`filewidth`,`fileheight`,`filesize`,`filename`,`ip`,`sticky`,`locked` FROM `posts_%s` WHERE (`thread` IS NULL AND `id` = :id) OR `thread` = :id ORDER BY `thread`,`time`", $board['uri']));
+ $query->bindValue(':id', $id, PDO::PARAM_INT);
+ $query->execute() or error(db_error($query));
+
+ while($post = $query->fetch()) {
if(!isset($thread)) {
- $thread = new Thread($post['id'], $post['subject'], $post['email'], $post['name'], $post['trip'], $post['body'], $post['time'], $post['thumb'], $post['thumbwidth'], $post['thumbheight'], $post['file'], $post['filewidth'], $post['fileheight'], $post['filesize'], $post['filename'], false);
+ $thread = new Thread($post['id'], $post['subject'], $post['email'], $post['name'], $post['trip'], $post['body'], $post['time'], $post['thumb'], $post['thumbwidth'], $post['thumbheight'], $post['file'], $post['filewidth'], $post['fileheight'], $post['filesize'], $post['filename'], $post['ip'], $post['sticky'], $post['locked'], $mod ? '?/' : $config['root'], $mod);
} else {
- $thread->add(new Post($post['id'], $thread->id, $post['subject'], $post['email'], $post['name'], $post['trip'], $post['body'], $post['time'], $post['thumb'], $post['thumbwidth'], $post['thumbheight'], $post['file'], $post['filewidth'], $post['fileheight'], $post['filesize'], $post['filename']));
+ $thread->add(new Post($post['id'], $thread->id, $post['subject'], $post['email'], $post['name'], $post['trip'], $post['body'], $post['time'], $post['thumb'], $post['thumbwidth'], $post['thumbheight'], $post['file'], $post['filewidth'], $post['fileheight'], $post['filesize'], $post['filename'], $post['ip'], $mod ? '?/' : $config['root'], $mod));
}
- $body = Element('thread.html', Array(
- 'button'=>BUTTON_REPLY,
- 'board'=>$board,
- 'body'=>$thread->build(),
- 'post_url' => POST_URL,
- 'index' => ROOT,
- 'id' => $id
- ));
-
- if($return)
- return $body;
- else
- @file_put_contents($board['dir'] . DIR_RES . $id . '.html', $body) or error("Couldn't write to file.");
}
- mysql_free_result($query);
+
+ // Check if any posts were found
+ if(!isset($thread)) error($config['error']['nonexistant']);
+
+ $body = Element('thread.html', Array(
+ 'button'=>$config['button_reply'],
+ 'board'=>$board,
+ 'body'=>$thread->build(),
+ 'post_url' => $config['post_url'],
+ 'index' => $config['root'],
+ 'id' => $id,
+ 'mod' => $mod,
+ 'return' => ($mod ? '?' . $board['url'] . $config['file_index'] : $config['root'] . $board['uri'] . '/' . $config['file_index'])
+ ));
+
+ if($return)
+ return $body;
+ else
+ @file_put_contents($board['dir'] . $config['dir']['res'] . sprintf($config['file_page'], $id), $body) or error("Couldn't write to file.");
}
function generate_tripcode ( $name, $length = 10 ) {
+ global $config;
$name = stripslashes ( $name );
$t = explode('#', $name);
$nameo = $t[0];
@@ -369,7 +735,7 @@
$salt = strtr ( $salt, ':;<=>?@[\]^_`', 'ABCDEFGabcdef' );
if ( isset ( $t[2] ) ) {
// secure
- $trip = '!!' . substr ( crypt ( $trip, '@#$%^&*()' ), ( -1 * $length ) );
+ $trip = '!!' . substr ( crypt ( $trip, $config['secure_trip_salt'] ), ( -1 * $length ) );
} else {
// insecure
$trip = '!' . substr ( crypt ( $trip, $salt ), ( -1 * $length ) );
@@ -504,25 +870,25 @@
case 'jpeg':
if(!$image = @imagecreatefromjpeg($source_pic)) {
unlink($source_pic);
- error(ERR_INVALIDIMG);
+ error($config['error']['invalidimg']);
}
break;
case 'png':
if(!$image = @imagecreatefrompng($source_pic)) {
unlink($source_pic);
- error(ERR_INVALIDIMG);
+ error($config['error']['invalidimg']);
}
break;
case 'gif':
if(!$image = @imagecreatefromgif($source_pic)) {
unlink($source_pic);
- error(ERR_INVALIDIMG);
+ error($config['error']['invalidimg']);
}
break;
case 'bmp':
if(!$image = @imagecreatefrombmp($source_pic)) {
unlink($source_pic);
- error(ERR_INVALIDIMG);
+ error($config['error']['invalidimg']);
}
break;
default:
@@ -644,4 +1010,4 @@
function int_to_word($n) {
return chr($n & 255).chr(($n >> 8) & 255);
}
-?>
\ No newline at end of file
+?>
diff --git a/inc/instance-config.php b/inc/instance-config.php
index 408e9b85..864a956c 100644
--- a/inc/instance-config.php
+++ b/inc/instance-config.php
@@ -7,17 +7,45 @@
*
* You can copy values from config.php (defaults) and paste them here.
*/
-
-
- /*
+
+
+
// Database stuff
- define('MY_SERVER', '127.0.0.1');
- define('MY_USER', '');
- define('MY_PASSWORD', '');
- define('MY_DATABASE', '');
+ $config['db']['type'] = 'mysql';
+ $config['db']['server'] = 'localhost';
+ $config['db']['user'] = '';
+ $config['db']['password'] = '';
+ $config['db']['database'] = '';
- define('ROOT', '/');
+ $config['root'] = '/';
-// define('FOO', 'bar');
- */
+
+
+ // The following looks ugly. I will find a better place to put this code soon.
+ $config['post_url'] = $config['root'] . 'post.php';
+
+ $config['url_match'] = '/^' .
+ (preg_match($config['url_regex'], $config['root']) ? '' :
+ (@$_SERVER['HTTPS']?'https':'http') .
+ ':\/\/'.$_SERVER['HTTP_HOST']) .
+ preg_quote($config['root'], '/') .
+ '(' .
+ str_replace('%s', '\w{1,8}', preg_quote($config['board_path'], '/')) .
+ '|' .
+ str_replace('%s', '\w{1,8}', preg_quote($config['board_path'], '/')) .
+ preg_quote($config['file_index'], '/') .
+ '|' .
+ str_replace('%s', '\w{1,8}', preg_quote($config['board_path'], '/')) .
+ str_replace('%d', '\d+', preg_quote($config['file_page'], '/')) .
+ '|' .
+ preg_quote($config['file_mod'], '/') .
+ '\?\/.+' .
+ ')$/i';
+
+ $config['dir']['static'] = $config['root'] . 'static/';
+
+ $config['image_sticky'] = $config['dir']['static'] . 'sticky.gif';
+ $config['image_locked'] = $config['dir']['static'] . 'locked.gif';
+ $config['image_deleted'] = $config['dir']['static'] . 'deleted.png';
+ $config['image_zip'] = $config['dir']['static'] . 'zip.png';
?>
\ No newline at end of file
diff --git a/inc/mod.php b/inc/mod.php
new file mode 100644
index 00000000..4e82aaeb
--- /dev/null
+++ b/inc/mod.php
@@ -0,0 +1,187 @@
+bindValue(':username', $username);
+ $query->bindValue(':password', $password);
+ $query->execute() or error(db_error($query));
+
+ if($user = $query->fetch()) {
+ return $mod = Array(
+ 'id' => $user['id'],
+ 'type' => $user['type'],
+ 'username' => $username,
+ 'password' => $password,
+ 'hash' => isset($_SESSION['mod']['hash']) ? $_SESSION['mod']['hash'] : mkhash()
+ );
+ } else return false;
+ }
+
+ function setCookies() {
+ global $mod, $config;
+ if(!$mod) error('setCookies() was called for a non-moderator!');
+
+ // $config['cookies']['mod'] contains username:hash
+ setcookie($config['cookies']['mod'], $mod['username'] . ':' . $mod['hash'], time()+$config['cookies']['expire'], $config['cookies']['jail']?$config['root']:'/', null, false, true);
+
+ // Put $mod in the session
+ $_SESSION['mod'] = $mod;
+
+ // Lock sessions to IP addresses
+ if($mod['lock_ip'])
+ $_SESSION['mod']['ip'] = $_SERVER['REMOTE_ADDR'];
+ }
+
+ function destroyCookies() {
+ // Delete the cookies
+ setcookie($config['cookies']['mod'], 'deleted', time()-$config['cookies']['expire'], $config['cookies']['jail']?$config['root']:'/', null, false, true);
+
+ // Unset the session
+ unset($_SESSION['mod']);
+ }
+
+ function modLog($action) {
+ global $mod;
+ $query = prepare("INSERT INTO `modlogs` VALUES (:id, :ip, :time, :text)");
+ $query->bindValue(':id', $mod['id'], PDO::PARAM_INT);
+ $query->bindValue(':ip', $_SERVER['REMOTE_ADDR']);
+ $query->bindValue(':time', time(), PDO::PARAM_INT);
+ $query->bindValue(':text', $action);
+ $query->execute() or error(db_error($query));
+ }
+
+ if(isset($_COOKIE['mod']) && isset($_SESSION['mod']) && is_array($_SESSION['mod'])) {
+ // Should be username:session hash
+ $cookie = explode(':', $_COOKIE['mod']);
+ if(count($cookie) != 2) {
+ destroyCookies();
+ error($config['error']['malformed']);
+ }
+
+ // Validate session
+ if( $cookie[0] != $_SESSION['mod']['username'] ||
+ $cookie[1] != $_SESSION['mod']['hash']) {
+ // Malformed cookies
+ destroyCookies();
+ error($config['error']['malformed']);
+ }
+
+ // Open connection
+ sql_open();
+
+ // Check username/password
+ if(!login($_SESSION['mod']['username'], $_SESSION['mod']['password'], false)) {
+ destroyCookies();
+ error($config['error']['invalidafter']);
+ }
+
+ }
+
+ // Generates a
+
+ Database
+ Type:
+ ';
+
+ $drivers = PDO::getAvailableDrivers();
+
+ foreach($drivers as &$driver) {
+ $driver_txt = $driver;
+ switch($driver) {
+ case 'cubrid':
+ $driver_txt = 'Cubrid';
+ break;
+ case 'dblib':
+ $driver_txt = 'FreeTDS / Microsoft SQL Server / Sybase';
+ break;
+ case 'firebird':
+ $driver_txt = 'Firebird/Interbase 6';
+ break;
+ case 'ibm':
+ $driver_txt = 'IBM DB2';
+ break;
+ case 'informix':
+ $driver_txt = 'IBM Informix Dynamic Server';
+ break;
+ case 'mysql':
+ $driver_txt = 'MySQL';
+ break;
+ case 'oci':
+ $driver_txt = 'OCI';
+ break;
+ case 'odbc':
+ $driver_txt = 'ODBC v3 (IBM DB2, unixODBC)';
+ break;
+ case 'pgsql':
+ $driver_txt = 'PostgreSQL';
+ break;
+ case 'sqlite':
+ $driver_txt = 'SQLite 3';
+ break;
+ case 'sqlite2':
+ $driver_txt = 'SQLite 2';
+ break;
+ }
+ $page['body'] .= '' . $driver_txt . ' ';
+ }
+
+ $page['body'] .= '
+
+
+ Database:
+
+
+ Username:
+
+
+ Password:
+
+
+
+
+ Cookies
+ Name of session cookie:
+
+
+ Cookie containing a timestamp of first arrival:
+
+
+ Cookie containing a hash for verification purposes:
+
+
+ Moderator cookie:
+
+
+ Secure salt:
+
+
+
+
+ Flood control
+ Seconds before each post:
+
+
+ Seconds before you can repost something (post the exact same text):
+
+
+ Same as above, but with a different IP address:
+
+
+ Maximum post body length:
+
+
+ Replies in a thread before it can no longer be bumped:
+
+
+ Maximum number of links in a single post:
+
+
+
+
+ Images
+ Maximum image filesize:
+
+
+ Thumbnail width:
+
+
+ Thumbnail height:
+
+
+ Maximum image width:
+
+
+ Maximum image height:
+
+
+
+
+ Display
+ Threads per page:
+
+
+ Page limit:
+
+
+ Number of replies to show per thread on the index page:
+
+
+
+
+ Directories
+ Root URI (include trailing slash):
+
+
+ Image directory:
+
+
+ Thumbnail directory:
+
+
+ Thread directory:
+
+
+
+
+ Miscellaneous
+ Secure trip (##) salt:
+
+
+
+
+
+
+
+ ';
+
+
+ echo Element('page.html', $page);
+ }
+?>
\ No newline at end of file
diff --git a/install.sql b/install.sql
index c517bdfc..1141fc9c 100644
--- a/install.sql
+++ b/install.sql
@@ -45,10 +45,10 @@ INSERT INTO `boards` (`id`, `uri`, `title`, `subtitle`) VALUES
-- --------------------------------------------------------
--
--- Table structure for table `posts`
+-- Table structure for table `posts_b`
--
-CREATE TABLE IF NOT EXISTS `posts` (
+CREATE TABLE IF NOT EXISTS `posts_b` (
`id` int(11) NOT NULL AUTO_INCREMENT,
`board` smallint(6) NOT NULL,
`thread` int(11) DEFAULT NULL,
@@ -70,10 +70,67 @@ CREATE TABLE IF NOT EXISTS `posts` (
`filehash` varchar(32) DEFAULT NULL,
`password` varchar(20) DEFAULT NULL,
`ip` varchar(15) NOT NULL,
+ `sticky` int(1) NOT NULL,
+ `locked` int(1) NOT NULL,
UNIQUE KEY `id` (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8 AUTO_INCREMENT=1 ;
--
--- Dumping data for table `posts`
+-- Dumping data for table `posts_b`
--
+-- --------------------------------------------------------
+
+--
+-- Table structure for table `mods`
+--
+
+CREATE TABLE IF NOT EXISTS `mods` (
+ `id` smallint(6) NOT NULL AUTO_INCREMENT,
+ `username` varchar(30) NOT NULL,
+ `password` char(40) NOT NULL COMMENT 'SHA1',
+ `type` smallint(1) NOT NULL COMMENT '0: janitor, 1: mod, 2: admin',
+ PRIMARY KEY (`id`),
+ UNIQUE KEY `id` (`id`,`username`)
+) ENGINE=InnoDB DEFAULT CHARSET=utf8 AUTO_INCREMENT=2 ;
+
+--
+-- Dumping data for table `mods`
+--
+
+INSERT INTO `mods` (`id`, `username`, `password`, `type`) VALUES
+(1, 'admin', '5baa61e4c9b93f3f0682250b6cf8331b7ee68fd8', 2);
+
+
+-- --------------------------------------------------------
+
+--
+-- Table structure for table `bans`
+--
+
+
+CREATE TABLE IF NOT EXISTS `bans` (
+ `ip` varchar( 15 ) NOT NULL ,
+ `mod` int NOT NULL COMMENT 'which mod made the ban',
+ `set` int NOT NULL,
+ `expires` int NULL,
+ `reason` text NULL
+) ENGINE = InnoDB;
+
+
+CREATE TABLE `robot` (
+`hash` VARCHAR( 40 ) NOT NULL COMMENT 'SHA1'
+) ENGINE = INNODB; */
+
+
+CREATE TABLE `mutes` (
+`ip` VARCHAR( 15 ) NOT NULL ,
+`time` INT NOT NULL
+) ENGINE = MYISAM ;
+
+CREATE TABLE `modlogs` (
+`mod` INT NOT NULL ,
+`ip` VARCHAR( 15 ) NOT NULL ,
+`time` INT NOT NULL ,
+`text` TEXT NOT NULL
+) ENGINE = INNODB;
diff --git a/main.js b/main.js
index d0b0e0d6..038713cd 100644
--- a/main.js
+++ b/main.js
@@ -6,17 +6,32 @@ function highlightReply(id)
if (divs[i].className.indexOf('post') != -1)
divs[i].className = divs[i].className.replace(/highlighted/, '');
}
- if (id)
- document.getElementById('reply_'+id).className += ' highlighted';
+ if (id) {
+ post = document.getElementById('reply_'+id);
+ if(post)
+ post.className += ' highlighted';
+ }
}
function focusId(id)
{
document.getElementById(id).focus();
init();
}
+
+function generatePassword() {
+ pass = '';
+ chars = 'abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789!@#$%^&*()_+';
+ for(i=0;i<8;i++) {
+ rnd = Math.floor(Math.random() * chars.length);
+ pass += chars.substring(rnd,rnd + 1);
+ }
+ return pass;
+}
+
function dopost(form) {
- localStorage.name = form.name.value;
- localStorage.email = form.email.value;
+ localStorage.name = form.name.value.replace(/ ##.+$/, '');
+ if(form.email.value != 'sage')
+ localStorage.email = form.email.value;
return form.body.value != "" || (typeof form.thread != "undefined" && form.file.value != "");
}
@@ -24,8 +39,56 @@ function citeReply(id) {
document.getElementById('body').value += '>>' + id + '\n';
}
+var selectedstyle = 'Yotsuba B';
+var styles = [
+ ['Yotsuba B', '/default.css'],
+ ['Yotsuba', '/yotsuba.css']
+];
+
+function changeStyle(x) {
+ localStorage.stylesheet = styles[x][1];
+ document.getElementById('stylesheet').href = styles[x][1];
+ selectedstyle = styles[x][0];
+}
+
+newLink = document.createElement('link');
+newLink.rel = 'stylesheet';
+newLink.type = 'text/css';
+newLink.id = 'stylesheet';
+document.getElementsByTagName('head')[0].insertBefore(newLink, document.getElementsByTagName('link')[0].lastChild)
+
+if(localStorage.stylesheet) {
+ for(x=0;x str_replace('%s', '(\w{1,8})', preg_quote($config['board_path'], '/')),
+ 'page' => str_replace('%d', '(\d+)', preg_quote($config['file_page'], '/')),
+ 'img' => preg_quote($config['dir']['img'], '/'),
+ 'thumb' => preg_quote($config['dir']['thumb'], '/'),
+ 'res' => preg_quote($config['dir']['res'], '/'),
+ 'index' => preg_quote($config['file_index'], '/')
+ );
+
+ if(preg_match('/^\/?$/', $query)) {
+ // Dashboard
+ $fieldset = Array(
+ 'Boards' => '',
+ 'Administration' => ''
+ );
+
+ // Boards
+ $fieldset['Boards'] .= ulBoards();
+
+ if($mod['type'] >= $config['mod']['view_banlist']) {
+ $fieldset['Administration'] .= 'Ban list ';
+ }
+ if($mod['type'] >= $config['mod']['show_config']) {
+ $fieldset['Administration'] .= 'Show configuration ';
+ }
+
+ // TODO: Statistics, etc, in the dashboard.
+
+ $body = '';
+ foreach($fieldset as $title => $data) {
+ if($data)
+ $body .= "{$title} ";
+ }
+
+ echo Element('page.html', Array(
+ 'index'=>$config['root'],
+ 'title'=>'Dashboard',
+ 'body'=>$body
+ //,'mod'=>true /* All 'mod' does, at this point, is put the "Return to dashboard" link in. */
+ )
+ );
+ } elseif(preg_match('/^\/bans$/', $query)) {
+ if($mod['type'] < $config['mod']['view_banlist']) error($config['error']['noaccess']);
+
+ if($config['mod']['view_banexpired']) {
+ $query = prepare("SELECT * FROM `bans` INNER JOIN `mods` ON `mod` = `id` GROUP BY `ip` ORDER BY `expires` < :time, `set` DESC");
+ $query->bindValue(':time', time(), PDO::PARAM_INT);
+ $query->execute() or error(db_error($query));
+ } else {
+ // Filter out expired bans
+ $query = prepare("SELECT * FROM `bans` INNER JOIN `mods` ON `mod` = `id` GROUP BY `ip` WHERE `expires` = 0 OR `expires` > :time ORDER BY `set` DESC");
+ $query->bindValue(':time', time(), PDO::PARAM_INT);
+ $query->execute() or error(db_error($query));
+ }
+
+ if($query->rowCount() < 1) {
+ $body = '(There are no active bans.)';
+ } else {
+ $body = '';
+ $body .= ' ';
+ }
+
+ echo Element('page.html', Array(
+ 'index'=>$config['root'],
+ 'title'=>'Ban list',
+ 'body'=>$body,
+ 'mod'=>true
+ )
+ );
+ } elseif(preg_match('/^\/config$/', $query)) {
+ if($mod['type'] < $config['mod']['show_config']) error($config['error']['noaccess']);
+
+ // Show instance-config.php
+
+ //$data = highlight_file('inc/instance-config.php', true);
+ //if(MOD_NEVER_REAL_PASSWORD) {
+ // // Rough and dirty removal of password
+ // $data = str_replace(MY_PASSWORD, '*******', $data);
+ //}
+
+ $constants = get_defined_constants(true);
+ $constants = $constants['user'];
+
+ $data = '';
+ foreach($constants as $name => $value) {
+ if(MOD_NEVER_REAL_PASSWORD && $name == 'DB_PASSWORD')
+ $value = 'hidden ';
+ else {
+ // For some reason PHP is only giving me the first defined value (the default), so use constant()
+ $value = constant($name);
+ if(gettype($value) == 'boolean') {
+ $value = $value ? 'On ' : 'Off ';
+ } elseif(gettype($value) == 'string') {
+ if(empty($value))
+ $value = 'empty ';
+ else
+ $value = '' . utf8tohtml(substr($value, 0, 110) . (strlen($value) > 110 ? '…' : '')) . ' ';
+ } elseif(gettype($value) == 'integer') {
+ // Show permissions in a cleaner way
+ if(preg_match('/^MOD_/', $name) && $name != 'MOD_JANITOR' && $name != 'MOD_MOD' && $name != 'MOD_ADMIN') {
+ if($value == MOD_JANITOR)
+ $value = 'Janitor';
+ elseif($value == MOD_MOD)
+ $value = 'Mod';
+ elseif($value == MOD_ADMIN)
+ $value = 'Admin';
+ }
+ $value = '' . $value . ' ';
+ }
+ }
+
+ $data .=
+ '' .
+ $name .
+ ' ' .
+ $value .
+ ' ';
+ }
+
+ $body = 'Configuration ';
+
+ echo Element('page.html', Array(
+ 'index'=>$config['root'],
+ 'title'=>'Configuration',
+ 'body'=>$body,
+ 'mod'=>true
+ )
+ );
+ } elseif(preg_match('/^\/new$/', $query)) {
+ if($mod['type'] < $config['mod']['newboard']) error($config['error']['noaccess']);
+
+ // New board
+ $body = '';
+
+ if(isset($_POST['new_board'])) {
+ // Create new board
+ if( !isset($_POST['uri']) ||
+ !isset($_POST['title']) ||
+ !isset($_POST['subtitle'])
+ ) error($config['error']['missedafield']);
+
+ $b = Array(
+ 'uri' => $_POST['uri'],
+ 'title' => $_POST['title'],
+ 'subtitle' => $_POST['subtitle']
+ );
+
+ // Check required fields
+ if(empty($b['uri']))
+ error(sprintf($config['error']['required'], 'URI'));
+ if(empty($b['title']))
+ error(sprintf($config['error']['required'], 'title'));
+
+ // Check string lengths
+ if(strlen($b['uri']) > 8)
+ error(sprintf($config['error']['toolong'], 'URI'));
+ if(strlen($b['title']) > 20)
+ error(sprintf($config['error']['toolong'], 'title'));
+ if(strlen($b['subtitle']) > 40)
+ error(sprintf($config['error']['toolong'], 'subtitle'));
+
+ if(!preg_match('/^\w+$/', $b['uri']))
+ error(sprintf($config['error']['invalidfield'], 'URI'));
+
+ if(openBoard($b['uri'])) {
+ unset($board);
+ error(sprintf($config['error']['boardexists'], sprintf($config['board_abbreviation'], $b['uri'])));
+ }
+
+ $query = prepare("INSERT INTO `boards` VALUES (NULL, :uri, :title, :subtitle)");
+ $query->bindValue(':uri', $b['uri']);
+ $query->bindValue(':title', $b['title']);
+ if(!empty($b['subtitle'])) {
+ $query->bindValue(':subtitle', $b['subtitle']);
+ } else {
+ $query->bindValue(':subtitle', null, PDO::PARAM_NULL);
+ }
+ $query->execute() or error(db_error($query));
+
+ // Record the action
+ modLog("Created a new board: {$b['title']}");
+
+ // Open the board
+ openBoard($b['uri']) or error("Couldn't open board after creation.");
+
+ // Create the posts table
+ query(Element('posts.sql', Array('board' => $board['uri']))) or error(db_error());
+
+ // Build the board
+ buildIndex();
+ }
+
+ $body .= form_newBoard();
+
+ // TODO: Statistics, etc, in the dashboard.
+
+ echo Element('page.html', Array(
+ 'index'=>$config['root'],
+ 'title'=>'New board',
+ 'body'=>$body,
+ 'mod'=>true
+ )
+ );
+ } elseif(preg_match('/^\/' . $regex['board'] . '(' . $regex['index'] . '|' . $regex['page'] . ')?$/', $query, $matches)) {
+ // Board index
+
+ $boardName = $matches[1];
+
+ // Open board
+ if(!openBoard($boardName))
+ error($config['error']['noboard']);
+
+ if(!$page = index(empty($matches[2]) || $matches[2] == $config['file_index'] ? 1 : $matches[2], $mod)) {
+ error($config['error']['404']);
+ }
+ $page['pages'] = getPages(true);
+ $page['mod'] = true;
+
+ echo Element('index.html', $page);
+ } elseif(preg_match('/^\/' . $regex['board'] . $regex['res'] . $regex['page'] . '$/', $query, $matches)) {
+ // View thread
+
+ $boardName = $matches[1];
+ $thread = $matches[2];
+ // Open board
+ if(!openBoard($boardName))
+ error($config['error']['noboard']);
+
+ $page = buildThread($thread, true, $mod);
+
+ echo $page;
+ } elseif(preg_match('/^\/' . $regex['board'] . 'deletefile\/(\d+)$/', $query, $matches)) {
+ if($mod['type'] < $config['mod']['deletefile']) error($config['error']['noaccess']);
+ // Delete file from post
+
+ $boardName = $matches[1];
+ $post = $matches[2];
+ // Open board
+ if(!openBoard($boardName))
+ error($config['error']['noboard']);
+
+ // Delete post
+ deleteFile($post);
+
+ // Record the action
+ modLog("Removed file from post #{$post}");
+
+ // Rebuild board
+ buildIndex();
+
+
+ // Redirect
+ if(isset($_SERVER['HTTP_REFERER']))
+ header('Location: ' . $_SERVER['HTTP_REFERER'], true, $config['redirect_http']);
+ else
+ header('Location: ?/' . sprintf($config['board_path'], $boardName) . $config['file_index'], true, $config['redirect_http']);
+ } elseif(preg_match('/^\/' . $regex['board'] . 'delete\/(\d+)$/', $query, $matches)) {
+ if($mod['type'] < $config['mod']['delete']) error($config['error']['noaccess']);
+ // Delete post
+
+ $boardName = $matches[1];
+ $post = $matches[2];
+ // Open board
+ if(!openBoard($boardName))
+ error($config['error']['noboard']);
+
+ // Delete post
+ deletePost($post);
+
+ // Record the action
+ modLog("Deleted post #{$post}");
+
+ // Rebuild board
+ buildIndex();
+
+ // Redirect
+ if(isset($_SERVER['HTTP_REFERER']))
+ header('Location: ' . $_SERVER['HTTP_REFERER'], true, $config['redirect_http']);
+ else
+ header('Location: ?/' . sprintf($config['board_path'], $boardName) . $config['file_index'], true, $config['redirect_http']);
+ } elseif(preg_match('/^\/' . $regex['board'] . '(un)?sticky\/(\d+)$/', $query, $matches)) {
+ if($mod['type'] < $config['mod']['sticky']) error($config['error']['noaccess']);
+ // Add/remove sticky
+
+ $boardName = $matches[1];
+ $post = $matches[3];
+ // Open board
+ if(!openBoard($boardName))
+ error($config['error']['noboard']);
+
+ $query = prepare(sprintf("UPDATE `posts_%s` SET `sticky` = :sticky WHERE `id` = :id AND `thread` IS NULL", $board['uri']));
+ $query->bindValue(':id', $post, PDO::PARAM_INT);
+
+ if($matches[2] == 'un') {
+ // Record the action
+ modLog("Unstickied post #{$post}");
+ $query->bindValue(':sticky', 0, PDO::PARAM_INT);
+ } else {
+ // Record the action
+ modLog("Stickied post #{$post}");
+ $query->bindValue(':sticky', 1, PDO::PARAM_INT);
+ }
+
+ $query->execute() or error(db_error($query));
+
+ buildIndex();
+ buildThread($post);
+
+
+ // Redirect
+ if(isset($_SERVER['HTTP_REFERER']))
+ header('Location: ' . $_SERVER['HTTP_REFERER'], true, $config['redirect_http']);
+ else
+ header('Location: ?/' . sprintf($config['board_path'], $boardName) . $config['file_index'], true, $config['redirect_http']);
+ } elseif(preg_match('/^\/' . $regex['board'] . '(un)?lock\/(\d+)$/', $query, $matches)) {
+ if($mod['type'] < $config['mod']['lock']) error($config['error']['noaccess']);
+ // Lock/Unlock
+
+ $boardName = $matches[1];
+ $post = $matches[3];
+ // Open board
+ if(!openBoard($boardName))
+ error($config['error']['noboard']);
+
+ $query = prepare(sprintf("UPDATE `posts_%s` SET `locked` = :locked WHERE `id` = :id AND `thread` IS NULL", $board['uri']));
+ $query->bindValue(':id', $post, PDO::PARAM_INT);
+
+ if($matches[2] == 'un') {
+ // Record the action
+ modLog("Unlocked post #{$post}");
+ $query->bindValue(':locked', 0, PDO::PARAM_INT);
+ } else {
+ // Record the action
+ modLog("Locked post #{$post}");
+ $query->bindValue(':locked', 1, PDO::PARAM_INT);
+ }
+
+ $query->execute() or error(db_error($query));
+
+ buildIndex();
+ buildThread($post);
+
+
+ // Redirect
+ if(isset($_SERVER['HTTP_REFERER']))
+ header('Location: ' . $_SERVER['HTTP_REFERER'], true, $config['redirect_http']);
+ else
+ header('Location: ?/' . sprintf($config['board_path'], $boardName) . $config['file_index'], true, $config['redirect_http']);
+ } elseif(preg_match('/^\/' . $regex['board'] . 'deletebyip\/(\d+)$/', $query, $matches)) {
+ // Delete all posts by an IP
+
+ $boardName = $matches[1];
+ $post = $matches[2];
+ // Open board
+ if(!openBoard($boardName))
+ error($config['error']['noboard']);
+
+ $query = prepare(sprintf("SELECT `ip` FROM `posts_%s` WHERE `id` = :id", $board['uri']));
+ $query->bindValue(':id', $post);
+ $query->execute() or error(db_error($query));
+
+ if(!$post = $query->fetch())
+ error($config['error']['invalidpost']);
+
+ $ip = $post['ip'];
+
+ // Record the action
+ modLog("Deleted all posts by IP address: #{$ip}");
+
+ $query = prepare(sprintf("SELECT `id` FROM `posts_%s` WHERE `ip` = :ip", $board['uri']));
+ $query->bindValue(':ip', $ip);
+ $query->execute() or error(db_error($query));
+
+ if($query->rowCount() < 1)
+ error($config['error']['invalidpost']);
+
+ while($post = $query->fetch()) {
+ deletePost($post['id'], false);
+ }
+
+ if(isset($_SERVER['HTTP_REFERER']))
+ header('Location: ' . $_SERVER['HTTP_REFERER'], true, $config['redirect_http']);
+ else
+ header('Location: ?/' . sprintf($config['board_path'], $boardName) . $config['file_index'], true, $config['redirect_http']);
+ } elseif(preg_match('/^\/ban$/', $query)) {
+ // Ban page
+
+ if(isset($_POST['new_ban'])) {
+ if( !isset($_POST['ip']) ||
+ !isset($_POST['reason']) ||
+ !isset($_POST['length'])
+ ) error($config['error']['missedafield']);
+
+ // Check required fields
+ if(empty($_POST['ip']))
+ error(sprintf($config['error']['required'], 'IP address'));
+
+ $query = prepare("INSERT INTO `bans` VALUES (:ip, :mod, :set, :expires, :reason)");
+
+ // 1yr2hrs30mins
+ // 1y2h30m
+ $expire = 0;
+ if(preg_match('/^((\d+)\s?ye?a?r?s?)?\s?+((\d+)\s?we?e?k?s?)?\s?+((\d+)\s?da?y?s?)?((\d+)\s?ho?u?r?s?)?\s?+((\d+)\s?mi?n?u?t?e?s?)?\s?+((\d+)\s?se?c?o?n?d?s?)?$/', $_POST['length'], $m)) {
+ if(isset($m[2])) {
+ // Years
+ $expire += $m[2]*60*60*24*365;
+ }
+ if(isset($m[4])) {
+ // Weeks
+ $expire += $m[4]*60*60*24*7;
+ }
+ if(isset($m[6])) {
+ // Days
+ $expire += $m[6]*60*60*24;
+ }
+ if(isset($m[8])) {
+ // Hours
+ $expire += $m[8]*60*60;
+ }
+ if(isset($m[10])) {
+ // Minutes
+ $expire += $m[10]*60;
+ }
+ if(isset($m[12])) {
+ // Seconds
+ $expire += $m[12];
+ }
+ }
+ if($expire) {
+ $query->bindValue(':expires', time()+$expire, PDO::PARAM_INT);
+ } else {
+ // Never expire
+ $query->bindValue(':expires', null, PDO::PARAM_NULL);
+ }
+
+ $query->bindValue(':ip', $_POST['ip'], PDO::PARAM_STR);
+ $query->bindValue(':mod', $mod['id'], PDO::PARAM_INT);
+ $query->bindValue(':set', time(), PDO::PARAM_INT);
+
+ if(isset($_POST['reason'])) {
+ $query->bindValue(':reason', $_POST['reason'], PDO::PARAM_STR);
+ } else {
+ $query->bindValue(':reason', null, PDO::PARAM_NULL);
+ }
+
+ // Record the action
+ modLog("Created a ban for {$_POST['ip']} with reason {$_POST['reason']}");
+
+ $query->execute() or error(db_error($query));
+
+ // Delete too
+ if($mod['type'] >= $config['mod']['delete'] && isset($_POST['delete']) && isset($_POST['board'])) {
+ openBoard($_POST['board']);
+ deletePost(round($_POST['delete']));
+ }
+
+ // Redirect
+ if(isset($_POST['continue']))
+ header('Location: ' . $_POST['continue'], true, $config['redirect_http']);
+ else
+ header('Location: ?/' . sprintf($config['board_path'], $boardName) . $config['file_index'], true, $config['redirect_http']);
+ }
+ } elseif(preg_match('/^\/' . $regex['board'] . 'ban(&delete)?\/(\d+)$/', $query, $matches)) {
+ if($mod['type'] < $config['mod']['delete']) error($config['error']['noaccess']);
+ // Ban by post
+
+ $boardName = $matches[1];
+ $delete = isset($matches[2]) && $matches[2] == '&delete';
+ $post = $matches[3];
+ // Open board
+ if(!openBoard($boardName))
+ error($config['error']['noboard']);
+
+ $query = prepare(sprintf("SELECT `ip`,`id` FROM `posts_%s` WHERE `id` = :id LIMIT 1", $board['uri']));
+ $query->bindValue(':id', $post, PDO::PARAM_INT);
+ $query->execute() or error(db_error($query));
+
+ if($query->rowCount() < 1) {
+ error($config['error']['invalidpost']);
+ }
+
+ $post = $query->fetch();
+
+ $body = form_newBan($post['ip'], null, isset($_SERVER['HTTP_REFERER']) ? $_SERVER['HTTP_REFERER'] : false, $delete ? $post['id'] : false, $delete ? $boardName : false);
+
+ echo Element('page.html', Array(
+ 'index'=>$config['root'],
+ 'title'=>'New ban',
+ 'body'=>$body,
+ 'mod'=>true
+ )
+ );
+ } elseif(preg_match('/^\/IP\/(\d+\.\d+\.\d+\.\d+)$/', $query, $matches)) {
+ // View information on an IP address
+
+ $ip = $matches[1];
+ $host = $config['mod']['dns_lookup'] ? gethostbyaddr($ip) : false;
+
+ $body = '';
+ $boards = listBoards();
+ foreach($boards as &$_board) {
+ openBoard($_board['uri']);
+
+ $temp = '';
+ $query = prepare(sprintf("SELECT * FROM `posts_%s` WHERE `ip` = :ip ORDER BY `sticky` DESC, `time` 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()) {
+ $po = new Post($post['id'], $post['thread'], $post['subject'], $post['email'], $post['name'], $post['trip'], $post['body'], $post['time'], $post['thumb'], $post['thumbwidth'], $post['thumbheight'], $post['file'], $post['filewidth'], $post['fileheight'], $post['filesize'], $post['filename'], $post['ip'], $mod ? '?/' : $config['root'], $mod);
+ $temp .= $po->build();
+ }
+ if(!empty($temp))
+ $body .= 'Last ' . $query->rowCount() . ' posts on ' .
+ sprintf($config['board_abbreviation'], $_board['uri']) . ' - ' . $_board['title'] .
+ ' ' . $temp . ' ';
+ }
+
+ if($config['mod']['ip_banform'])
+ $body .= form_newBan($ip, null, isset($_SERVER['HTTP_REFERER']) ? $_SERVER['HTTP_REFERER'] : false);
+
+ echo Element('page.html', Array(
+ 'index'=>$config['root'],
+ 'title'=>'IP: ' . $ip,
+ 'subtitle' => $host,
+ 'body'=>$body,
+ 'mod'=>true
+ )
+ );
+ } else {
+ error($config['error']['404']);
+ }
+ }
+
+ // Close the connection in-case it's still open
+ sql_close();
+?>
+
diff --git a/post.php b/post.php
index 8653a2c3..431be0c0 100644
--- a/post.php
+++ b/post.php
@@ -1,11 +1,12 @@
- $value) {
+ if(preg_match('/^delete_(\d+)$/', $post, $m)) {
+ $delete[] = (int)$m[1];
+ }
+ }
+
+ sql_open();
+
+ // Check if banned
+ checkBan();
+
+ if($config['block_tor'] && isTor())
+ error($config['error']['tor']);
+
+ // Check if board exists
+ if(!openBoard($_POST['board']))
+ error($config['error']['noboard']);
+
+ if(empty($delete))
+ error($config['error']['nodelete']);
+
+ foreach($delete as &$id) {
+ $query = prepare(sprintf("SELECT `password` FROM `posts_%s` WHERE `id` = :id", $board['uri']));
+ $query->bindValue(':id', $id, PDO::PARAM_INT);
+ $query->execute() or error(db_error($query));
+
+ if($post = $query->fetch()) {
+ if(!empty($password) && $post['password'] != $password)
+ error($config['error']['invalidpassword']);
+
+ if(isset($_POST['file'])) {
+ // Delete just the file
+ deleteFile($id);
+ } else {
+ // Delete entire post
+ deletePost($id);
+ }
+ }
+ }
+
+ buildIndex();
+
+ sql_close();
+
+ $is_mod = isset($_POST['mod']) && $_POST['mod'];
+ $root = $is_mod ? $config['root'] . $config['file_mod'] . '?/' : $config['root'];
+
+ header('Location: ' . $root . $board['dir'] . $config['file_index'], true, $config['redirect_http']);
+
+ } elseif(isset($_POST['post'])) {
if( !isset($_POST['name']) ||
!isset($_POST['email']) ||
!isset($_POST['subject']) ||
!isset($_POST['body']) ||
!isset($_POST['board']) ||
!isset($_POST['password'])
- ) error(ERROR_BOT);
+ ) error($config['error']['bot']);
$post = Array('board' => $_POST['board']);
@@ -35,13 +99,13 @@
$post['thread'] = round($_POST['thread']);
} else $OP = true;
- if(!(($OP && $_POST['post'] == BUTTON_NEWTOPIC) ||
- (!$OP && $_POST['post'] == BUTTON_REPLY)))
- error(ERROR_BOT);
+ if(!(($OP && $_POST['post'] == $config['button_newtopic']) ||
+ (!$OP && $_POST['post'] == $config['button_reply'])))
+ error($config['error']['bot']);
// Check the referrer
if($OP) {
- if(!isset($_SERVER['HTTP_REFERER']) || !preg_match(URL_MATCH, $_SERVER['HTTP_REFERER'])) error(ERROR_BOT);
+ if(!isset($_SERVER['HTTP_REFERER']) || !preg_match($config['url_match'], $_SERVER['HTTP_REFERER'])) error($config['error']['bot']);
}
// TODO: Since we're now using static HTML files, we can't give them cookies on their first page view
@@ -49,7 +113,7 @@
/*
// Check if he has a valid cookie.
- if(!$user['valid']) error(ERROR_BOT);
+ if(!$user['valid']) error($config['error']['bot']);
// Check how long he has been here.
if(time()-$user['appeared'] MAX_FILESIZE)
- error(sprintf3(ERR_FILESIZE, array(
+ if($size > $config['max_filesize'])
+ error(sprintf3($config['error']['filesize'], array(
'sz'=>commaize($size),
'filesz'=>commaize($size),
- 'maxsz'=>commaize(MAX_FILESIZE))));
+ 'maxsz'=>commaize($config['max_filesize']))));
+ }
+
+ if($mod && $mod['type'] >= MOD && preg_match('/^((.+) )?## (.+)$/', $post['name'], $match)) {
+ if(($mod['type'] == MOD && $match[3] == 'Mod') || $mod['type'] >= ADMIN) {
+ $post['mod_tag'] = $match[3];
+ $post['name'] = !empty($match[2])?$match[2]:'Anonymous';
+ }
+ } else {
+ $post['mod_tag'] = false;
}
$trip = generate_tripcode($post['name']);
@@ -100,40 +210,37 @@
if($post['has_file']) {
$post['extension'] = strtolower(substr($post['filename'], strrpos($post['filename'], '.') + 1));
- $post['file_id'] = rand(0, 1000000000);
- $post['file'] = $board['dir'] . DIR_IMG . $post['file_id'] . '.' . $post['extension'];
- $post['thumb'] = $board['dir'] . DIR_THUMB . $post['file_id'] . '.png';
- $post['zip'] = $OP && $post['has_file'] && ALLOW_ZIP && $post['extension'] == 'zip' ? $post['file'] : false;
- if(!($post['zip'] || in_array($post['extension'], $allowed_ext))) error(ERROR_FILEEXT);
+ $post['file_id'] = time() . rand(100, 999);
+ $post['file'] = $board['dir'] . $config['dir']['img'] . $post['file_id'] . '.' . $post['extension'];
+ $post['thumb'] = $board['dir'] . $config['dir']['thumb'] . $post['file_id'] . '.png';
}
// Check string lengths
- if(strlen($post['name']) > 25) error(sprintf(ERROR_TOOLONG, 'name'));
- if(strlen($post['email']) > 30) error(sprintf(ERROR_TOOLONG, 'email'));
- if(strlen($post['subject']) > 40) error(sprintf(ERROR_TOOLONG, 'subject'));
- if(strlen($post['body']) > MAX_BODY) error(ERROR_TOOLONGBODY);
- if(!(!$OP && $post['has_file']) && strlen($post['body']) < 1) error(ERROR_TOOSHORTBODY);
- if(strlen($post['password']) > 20) error(sprintf(ERROR_TOOLONG, 'password'));
+ if(strlen($post['name']) > 50) error(sprintf($config['error']['toolong'], 'name'));
+ if(strlen($post['email']) > 30) error(sprintf($config['error']['toolong'], 'email'));
+ if(strlen($post['subject']) > 40) error(sprintf($config['error']['toolong'], 'subject'));
+ if(!$mod && strlen($post['body']) > $config['max_body']) error($config['error']['toolongbody']);
+ if(!(!$OP && $post['has_file']) && strlen($post['body']) < 1) error($config['error']['tooshortbody']);
+ if(strlen($post['password']) > 20) error(sprintf($config['error']['toolong'], 'password'));
- markup($post['body']);
+ if($post['mod_tag'])
+ $post['trip'] .= ' ## ' . $post['mod_tag'] . ' ';
+
+ $post['body_nomarkup'] = $post['body'];
+
+ if(!($mod && $post['raw']))
+ markup($post['body']);
+
+ // Check for a flood
+ if(checkFlood($post)) {
+ error($config['error']['flood']);
+ }
if($post['has_file']) {
// Just trim the filename if it's too long
if(strlen($post['filename']) > 30) $post['filename'] = substr($post['filename'], 0, 27).'…';
// Move the uploaded file
- if(!@move_uploaded_file($_FILES['file']['tmp_name'], $post['file'])) error(ERROR_NOMOVE);
-
- if($post['zip']) {
- // Validate ZIP file
- if(is_resource($zip = zip_open($post['zip'])))
- // TODO: Check if it's not empty and has at least one (valid) image
- zip_close($zip);
- else
- error(ERR_INVALIDZIP);
-
- $post['file'] = ZIP_IMAGE;
- $post['extension'] = strtolower(substr($post['file'], strrpos($post['file'], '.') + 1));
- }
+ if(!@move_uploaded_file($_FILES['file']['tmp_name'], $post['file'])) error($config['error']['nomove']);
$size = @getimagesize($post['file']);
$post['width'] = $size[0];
@@ -142,189 +249,79 @@
// Check if the image is valid
if($post['width'] < 1 || $post['height'] < 1) {
unlink($post['file']);
- error(ERR_INVALIDIMG);
+ error($config['error']['invalidimg']);
}
- if($post['width'] > MAX_WIDTH || $post['height'] > MAX_HEIGHT) {
+ if($post['width'] > $config['max_width'] || $post['height'] > $config['max_height']) {
unlink($post['file']);
- error(ERR_MAXSIZE);
+ error($config['error']['maxsize']);
}
- $post['filehash'] = md5_file($post['file']);
+ $post['filehash'] = $config['file_hash']($post['file']);
$post['filesize'] = filesize($post['file']);
$image = createimage($post['extension'], $post['file']);
- if(REDRAW_IMAGE && !$post['zip']) {
- switch($post['extension']) {
- case 'jpg':
- case 'jpeg':
- imagejpeg($image, $post['file'], JPEG_QUALITY);
- break;
- case 'png':
- imagepng($image, $post['file'], 7);
- break;
- case 'gif':
- if(REDRAW_GIF)
- imagegif($image, $post['file']);
- break;
- case 'bmp':
- imagebmp($image, $post['file']);
- break;
- default:
- error('Unknwon file extension.');
- }
- }
-
// Create a thumbnail
- $thumb = resize($image, $post['width'], $post['height'], $post['thumb'], THUMB_WIDTH, THUMB_HEIGHT);
+ $thumb = resize($image, $post['width'], $post['height'], $post['thumb'], $config['thumb_width'], $config['thumb_height']);
$post['thumbwidth'] = $thumb['width'];
$post['thumbheight'] = $thumb['height'];
}
+ /*
+ if(!($mod && $mod['type'] >= $config['mod']['postunoriginal']) && ROBOT_ENABLE && $board['uri'] == ROBOT_BOARD && checkRobot($post['body_nomarkup'])) {
+ if(ROBOT_MUTE) {
+ error(sprintf($config['error']['muted'], mute()));
+ } else {
+ error($config['error']['unoriginal']);
+ }
+ }
+ */
+
// Remove DIR_* before inserting them into the database.
if($post['has_file']) {
- $post['file'] = substr_replace($post['file'], '', 0, strlen($board['dir'] . DIR_IMG));
- $post['thumb'] = substr_replace($post['thumb'], '', 0, strlen($board['dir'] . DIR_THUMB));
+ $post['file'] = substr_replace($post['file'], '', 0, strlen($board['dir'] . $config['dir']['img']));
+ $post['thumb'] = substr_replace($post['thumb'], '', 0, strlen($board['dir'] . $config['dir']['thumb']));
}
// Todo: Validate some more, remove messy code, allow more specific configuration
-
- // MySQLify
- mysql_safe_array($post);
-
$id = post($post, $OP);
- if($post['has_file'] && $post['zip']) {
- // Open ZIP
- $zip = zip_open($post['zip']);
- // Read files
- while($entry = zip_read($zip)) {
- $filename = basename(zip_entry_name($entry));
- $extension = strtolower(substr($filename, strrpos($filename, '.') + 1));
-
- if(in_array($extension, $allowed_ext)) {
- if (zip_entry_open($zip, $entry, 'r')) {
- // Fake post
- $dump_post = Array(
- 'subject' => $post['subject'],
- 'email' => $post['email'],
- 'name' => $post['name'],
- 'trip' => $post['trip'],
- 'body' => '',
- 'thread' => $id,
- 'password' => '',
- 'has_file' => true,
- 'file_id' => rand(0, 1000000000),
- 'filename' => $filename
- );
-
- $dump_post['file'] = $board['dir'] . DIR_IMG . $dump_post['file_id'] . '.' . $extension;
- $dump_post['thumb'] = $board['dir'] . DIR_THUMB . $dump_post['file_id'] . '.png';
-
- // Extract the image from the ZIP
- $fp = fopen($dump_post['file'], 'w+');
- fwrite($fp, zip_entry_read($entry, zip_entry_filesize($entry)));
- fclose($fp);
-
- $size = @getimagesize($dump_post['file']);
- $dump_post['width'] = $size[0];
- $dump_post['height'] = $size[1];
-
- // Check if the image is valid
- if($dump_post['width'] < 1 || $dump_post['height'] < 1) {
- unlink($dump_post['file']);
- } else {
- if($dump_post['width'] > MAX_WIDTH || $dump_post['height'] > MAX_HEIGHT) {
- unlink($dump_post['file']);
- error(ERR_MAXSIZE);
- } else {
- $dump_post['filehash'] = md5_file($dump_post['file']);
- $dump_post['filesize'] = filesize($dump_post['file']);
-
- $image = createimage($extension, $dump_post['file']);
-
- $success = true;
- if(REDRAW_IMAGE) {
- switch($extension) {
- case 'jpg':
- case 'jpeg':
- imagejpeg($image, $dump_post['file'], JPEG_QUALITY);
- break;
- case 'png':
- imagepng($image, $dump_post['file'], 7);
- break;
- case 'gif':
- if(REDRAW_GIF)
- imagegif($image, $dump_post['file']);
- break;
- case 'bmp':
- imagebmp($image, $dump_post['file']);
- break;
- default:
- $success = false;
- }
- }
-
-
- // Create a thumbnail
- $thumb = resize($image, $dump_post['width'], $dump_post['height'], $dump_post['thumb'], THUMB_WIDTH, THUMB_HEIGHT);
-
- $dump_post['thumbwidth'] = $thumb['width'];
- $dump_post['thumbheight'] = $thumb['height'];
-
- // Remove DIR_* before inserting them into the database.
- $dump_post['file'] = substr_replace($dump_post['file'], '', 0, strlen($board['dir'] . DIR_IMG));
- $dump_post['thumb'] = substr_replace($dump_post['thumb'], '', 0, strlen($board['dir'] . DIR_THUMB));
-
- // Create the post
- post($dump_post, false);
- }
- }
-
- // Close the ZIP
- zip_entry_close($entry);
- }
- }
- }
- zip_close($zip);
- unlink($post['zip']);
- }
-
buildThread(($OP?$id:$post['thread']));
- if(!$OP) {
- mysql_query(
- sprintf("UPDATE `posts` SET `bump` = '%d' WHERE `id` = '%s' AND `thread` IS NULL",
- time(),
- $post['thread']
- ), $sql) or error(mysql_error($sql));
+ if(!$OP && strtolower($post['email']) != 'sage' && ($config['reply_limit'] == 0 || numPosts($post['thread']) < $config['reply_limit'])) {
+ bumpThread($post['thread']);
}
+ if($OP)
+ clean();
+
buildIndex();
sql_close();
- if(ALWAYS_NOKO || $noko) {
- header('Location: ' . ROOT . $board['dir'] . DIR_RES . ($OP?$id:$post['thread']) . '.html' . (!$OP?'#'.$id:''), true, 302);
+ $root = $post['mod'] ? $config['root'] . $config['file_mod'] . '?/' : $config['root'];
+
+ if($config['always_noko'] || $noko) {
+ header('Location: ' . $root . $board['dir'] . $config['dir']['res'] . ($OP?$id:$post['thread']) . '.html' . (!$OP?'#'.$id:''), true, $config['redirect_http']);
} else {
- header('Location: ' . ROOT . $board['dir'] . FILE_INDEX, true, 302);
+ header('Location: ' . $root . $board['dir'] . $config['file_index'], true, $config['redirect_http']);
}
exit;
} else {
- if(!file_exists(HAS_INSTALLED)) {
+ if(!file_exists($config['has_installed'])) {
sql_open();
// Build all boards
- $boards_res = mysql_query('SELECT * FROM `boards`', $sql) or error(mysql_error($sql));
- while($_board = mysql_fetch_array($boards_res)) {
+ $boards = listBoards();
+ foreach($boards as &$_board) {
setupBoard($_board);
buildIndex();
}
sql_close();
- touch(HAS_INSTALLED, 0777);
+ touch($config['has_installed'], 0777);
die(Element('page.html', Array(
'index'=>ROOT,
@@ -336,7 +333,7 @@
} else {
// They opened post.php in their browser manually.
// Possible TODO: Redirect back to homepage.
- error(ERROR_NOPOST);
+ error($config['error']['nopost']);
}
}
?>
diff --git a/res/index.php b/res/index.php
deleted file mode 100644
index c5619a58..00000000
--- a/res/index.php
+++ /dev/null
@@ -1,3 +0,0 @@
-
\ No newline at end of file
diff --git a/src/index.php b/src/index.php
deleted file mode 100644
index c5619a58..00000000
--- a/src/index.php
+++ /dev/null
@@ -1,3 +0,0 @@
-
\ No newline at end of file
diff --git a/static/deleted.png b/static/deleted.png
new file mode 100644
index 00000000..269090ec
Binary files /dev/null and b/static/deleted.png differ
diff --git a/src/error.png b/static/error.png
similarity index 100%
rename from src/error.png
rename to static/error.png
diff --git a/static/locked.gif b/static/locked.gif
new file mode 100644
index 00000000..d178c7b6
Binary files /dev/null and b/static/locked.gif differ
diff --git a/src/ok.png b/static/ok.png
similarity index 100%
rename from src/ok.png
rename to static/ok.png
diff --git a/static/sticky.gif b/static/sticky.gif
new file mode 100644
index 00000000..da4ec3a3
Binary files /dev/null and b/static/sticky.gif differ
diff --git a/src/warning.png b/static/warning.png
similarity index 100%
rename from src/warning.png
rename to static/warning.png
diff --git a/src/zip.png b/static/zip.png
old mode 100755
new mode 100644
similarity index 100%
rename from src/zip.png
rename to static/zip.png
diff --git a/style.css b/style.css
index 8d8bae1c..4aec2b96 100644
--- a/style.css
+++ b/style.css
@@ -80,6 +80,18 @@ form table tr th {
form table tr th {
background: #98E;
}
+form table tr td div {
+ text-align: cetner;
+ float: left;
+ padding-left: 3px;
+}
+form table tr td div input {
+ display: block;
+ margin: 2px auto 0 auto;
+}
+form table tr td div label {
+ font-size: 10px;
+}
.unimportant, .unimportant * {
font-size: 10px;
}
@@ -101,6 +113,10 @@ div.banner a:hover {
color: #EEF2FF;
text-decoration: none;
}
+img.banner {
+ float: none;
+ margin: 4px auto 0 auto;
+}
img {
display: block;
float: left;
@@ -111,6 +127,12 @@ div.post img {
padding: 5px;
margin: 5px 20px 0 0;
}
+div.post img.icon {
+ display: inline;
+ float: none;
+ margin: 0 5px;
+ padding: 0;
+}
div.post.op {
margin-right: 20px;
margin-bottom: 5px;
@@ -120,6 +142,10 @@ p.intro {
padding: 0;
padding-bottom: 0.2em;
}
+input.delete {
+ float: left;
+ margin: 1px 6px 0 0;
+}
p.intro span.subject {
color: #0F0C5D;
font-weight: bold;
@@ -128,9 +154,16 @@ p.intro span.name {
color: #117743;
font-weight: bold;
}
+p.intro a.nametag {
+ color: #F00000;
+ margin-left: 0;
+}
p.intro a {
margin-left: 8px;
}
+div.delete {
+ float: right;
+}
div.post.reply p {
margin: 0.3em 0 0 0;
}
@@ -167,4 +200,74 @@ span.omitted {
}
br.clear {
clear: left;
+}
+span.controls {
+ float: right;
+ margin: 0;
+ padding: 0;
+ font-size: 80%;
+}
+span.controls.op {
+ float: none;
+ margin-left: 10px;
+}
+span.controls a {
+ margin: 0;
+}
+div#wrap {
+ width: 900px;
+ margin:0 auto;
+}
+div.ban {
+ background: white;
+ border: 1px solid #98E;
+ max-width: 700px;
+ margin: 30px auto;
+}
+div.ban p, div.ban h2 {
+ padding: 3px 7px;
+}
+div.ban h2 {
+ background: #98E;
+ color: black;
+ font-size: 12pt;
+}
+div.ban p {
+ font-size: 12px;
+ margin-bottom: 12px;
+}
+div.ban p.reason {
+ font-weight: bold;
+}
+span.spoiler {
+ background: black;
+}
+span.spoiler:hover {
+ color: white;
+}
+div.styles {
+ float: right;
+ padding-bottom: 20px;
+}
+div.styles a {
+ margin: 0 10px;
+}
+div.styles a.selected {
+ text-decoration: none;
+}
+table.test {
+ width: 100%;
+}
+table.test td, table.test th {
+ text-align: left;
+ padding: 5px;
+}
+table.test tr.h th {
+ background: #98E;
+}
+table.test td img {
+ margin: 0;
+}
+fieldset label {
+ display: block;
}
\ No newline at end of file
diff --git a/templates/index.html b/templates/index.html
index 048a3dd9..424cf21b 100644
--- a/templates/index.html
+++ b/templates/index.html
@@ -1,26 +1,27 @@
-
+
{board[url]} - {board[name]}
-
-
+
{board[url]} - {board[name]}
- {board[title]?{board[title]}}
+
+
+ {mod? }
+
+
+
+ {mod? }
{body}
+
+ Delete Post [
+ File ] Password
+
+
+
+
Pages: {pages:
[{pages[num]} ]{!%last? }
}
- Copyright © 2010 OmegaSDG
+ Tinyboard Software Copyright © 2010-2011 OmegaSDG
\ No newline at end of file
diff --git a/templates/login.html b/templates/login.html
new file mode 100644
index 00000000..312bd6e9
--- /dev/null
+++ b/templates/login.html
@@ -0,0 +1,28 @@
+
+ {error?{error} }
+
+
+
+
\ No newline at end of file
diff --git a/templates/page.html b/templates/page.html
index f47fbca4..41f0d4e6 100644
--- a/templates/page.html
+++ b/templates/page.html
@@ -1,16 +1,16 @@
-
+
{title}
-
+
{title}
- {subtitle?{subtitle}}
+
{body}
- Copyright © 2010 OmegaSDG
+ Tinyboard Software Copyright © 2010-2011 OmegaSDG
-
\ No newline at end of file
+