Error\nPHP's GD extension seems to be missing; ". "you can live without it if you have imagemagick installed..."; } else { $gd = ""; } print << Shimmie2 Installer

Shimmie Installer

$gd

Note

Shimmie is developed with MySQL, and support for it is included. Other databases may work, but you'll need to add the appropriate ADOdb drivers yourself

Install

Database
Admin Name:
Admin Pass:

Databases should be specified like so:
ie: protocol://username:password@host/database?options
eg: mysql://shimmie:pw123@localhost/shimmie?persist

Upgrade

Old Database:
New Database:
Old Data Folder:

Data folder is where the "images" and "thumbs" folders are stored

EOD; } // }}} // common {{{ function set_admin_cookie($admin_name, $admin_pass) { // {{{ $addr = $_SERVER['REMOTE_ADDR']; $hash = md5(strtolower($admin_name) . $admin_pass); setcookie("shm_user", $admin_name, time()+60*60*24*365); setcookie("shm_session", md5($hash.$addr), time()+60*60*24*7, "/"); } // }}} function create_tables($dsn) { // {{{ $db = NewADOConnection($dsn); if(!$db) { die("Couldn't connect to \"$dsn\""); } else { if(substr($dsn, 0, 5) == "mysql") { if(create_tables_mysql($db)) { $_SESSION['tables_created'] = true; } } else if(substr($dsn, 0, 5) == "pgsql" || substr($dsn, 0, 8) == "postgres") { if(create_tables_pgsql($db)) { $_SESSION['tables_created'] = true; } } else { die("This database format isn't currently supported. Please use either MySQL or PostgreSQL."); } if(!isset($_SESSION['tables_created']) || !$_SESSION['tables_created']) { die("Error creating tables"); } } $db->Close(); } // }}} function build_dirs() { // {{{ if(!file_exists("images")) @mkdir("images"); // *try* and make default dirs. Ignore any errors -- if(!file_exists("thumbs")) @mkdir("thumbs"); // if something is amiss, we'll tell the user later if(!file_exists("data")) @mkdir("data"); if( ((!file_exists("images") || !file_exists("thumbs") || !file_exists("data")) && !is_writable("./")) || (!is_writable("images") || !is_writable("thumbs") || !is_writable("data")) ) { print "Shimmie needs three folders in it's directory, 'images', 'thumbs', and 'data', and they need to be writable by the PHP user (if you see this error, if probably means the folders are owned by you, and they need to be writable by the web server).

Once you have created these folders, hit 'refresh' to continue."; exit; } else { assert(file_exists("images") && is_writable("images")); assert(file_exists("thumbs") && is_writable("thumbs")); assert(file_exists("data") && is_writable("data")); if(!file_exists("images/ff")) { for($i=0; $i<256; $i++) { mkdir(sprintf("images/%02x", $i)); mkdir(sprintf("thumbs/%02x", $i)); } } } } // }}} function write_config($dsn) { // {{{ $file_content = ""; if(is_writable("./") && installer_write_file("config.php", $file_content)) { assert(file_exists("config.php")); session_destroy(); } else { $h_file_content = htmlentities($file_content); print << Shimmie2 Installer The web server isn't allowed to write to the config file; please copy the text below, save it as 'config.php', and upload it into the shimmie folder manually. Make sure that when you save it, there is no whitespace before the "<?php" or after the "?>"

One done, Continue EOD; session_destroy(); exit; } } // }}} // }}} // install {{{ function install_process() { // {{{ if(!isset($_POST['database_dsn']) || !isset($_POST["admin_name"]) || !isset($_POST["admin_pass"])) { die("Install is missing some paramaters (database_dsn, admin_name, or admin_pass)"); } else if(strlen($_POST["admin_name"]) < 1 || strlen($_POST["admin_pass"]) < 1) { die("Admin name and password must be at least one character each"); } else { $database_dsn = $_POST['database_dsn']; $admin_name = $_POST["admin_name"]; $admin_pass = $_POST["admin_pass"]; } set_admin_cookie($admin_name, $admin_pass); create_tables($database_dsn); insert_defaults($database_dsn, $admin_name, $admin_pass); build_dirs(); write_config($database_dsn); header("Location: index.php?q=setup"); } // }}} function insert_defaults($dsn, $admin_name, $admin_pass) { // {{{ $db = NewADOConnection($dsn); if(!$db) { die("Couldn't connect to \"$dsn\""); } else { $config_insert = $db->Prepare("INSERT INTO config(name, value) VALUES(?, ?)"); $user_insert = $db->Prepare("INSERT INTO users(name, pass, joindate, admin) VALUES(?, ?, now(), ?)"); if(!$db->GetOne("SELECT * FROM users WHERE name=?", Array('Anonymous'))) { $db->Execute($user_insert, Array('Anonymous', null, 'N')); $db->Execute("DELETE FROM config WHERE name=?", Array('anon_id')); $db->Execute($config_insert, Array('anon_id', $db->Insert_ID())); } # we can safely delete the user and recreate, hence changing the ID, # because insert_defaults is only called during first installation if($db->GetOne("SELECT * FROM users WHERE name=?", Array($admin_name))) { $db->Execute("DELETE FROM users WHERE name=?", Array($admin_name)); } $admin_pass = md5(strtolower($admin_name).$admin_pass); $db->Execute($user_insert, Array($admin_name, $admin_pass, 'Y')); if(!ini_get('safe_mode')) { $convert_check = exec("convert"); if(!empty($convert_check)) { $db->Execute($config_insert, Array('thumb_engine', 'convert')); } } $db->Close(); } } // }}} // }}} // upgrade {{{ function upgrade_process() { // {{{ if(!isset($_POST['old_dsn']) || !isset($_POST["new_dsn"]) || !isset($_POST["old_data"])) { die("Install is missing some paramaters (old_dsn, new_dsn, or old_data)"); } else { $old_dsn = $_POST['old_dsn']; $new_dsn = $_POST['new_dsn']; $old_data = $_POST['old_data']; } if(!is_readable($old_data)) {die("Can't find \"$old_data\"");} if(!is_readable("$old_data/images")) {die("Can't find \"$old_data/images\"");} if(!is_readable("$old_data/thumbs")) {die("Can't find \"$old_data/thumbs\"");} // set_admin_cookie($admin_name, $admin_pass); create_tables($new_dsn); build_dirs(); move_data($old_dsn, $new_dsn, $old_data); write_config($new_dsn); // header("Location: index.php?q=setup"); print "

If everything looks OK, continue"; } // }}} function move_data($old_dsn, $new_dsn, $old_data) { print("
Upping PHP resource limits..."); set_time_limit(600); ini_set("memory_limit", "32M"); print("
Fetching old data..."); $old_db = NewADOConnection($old_dsn); $old_db->SetFetchMode(ADODB_FETCH_ASSOC); # tmpfile & serialize? $anon_id = -1; $users = $old_db->GetAll("SELECT id, name, pass, joindate FROM users ORDER BY id"); $admins = $old_db->GetCol("SELECT owner_id FROM user_configs WHERE name='isadmin' AND value='true'"); $images = $old_db->GetAll("SELECT id, owner_id, owner_ip, filename, hash, ext FROM images ORDER BY id"); $comments = $old_db->GetAll("SELECT id, image_id, owner_id, owner_ip, posted, comment FROM comments ORDER BY id"); $tags = $old_db->GetAll("SELECT image_id, tag FROM tags"); $old_db->Close(); $new_db = NewADOConnection($new_dsn); $new_db->SetFetchMode(ADODB_FETCH_ASSOC); if($users) { print("
Moving users..."); $new_db->Execute("DELETE FROM users"); $new_db->Execute(" INSERT INTO users(id, name, pass, joindate, enabled, admin, email) VALUES(?, ?, ?, ?, 'Y', 'N', '')", $users); } if($admins) { print("
Setting account flags"); $new_db->Execute("UPDATE users SET admin='Y' WHERE id=?", array($admins)); } if(true) { print("
Updating anonymous account..."); $anon_id = $new_db->GetOne("SELECT id FROM users WHERE name='Anonymous'"); if(!$anon_id) { print("
Warning: 'Anonymous' not found; creating one"); $new_db->Execute("INSERT INTO users(name, pass, joindate) VALUES ('Anonymous', NULL, now())"); $anon_id = $new_db->Insert_ID(); } $new_db->Execute("DELETE FROM config WHERE name=?", array("anon_id")); $new_db->Execute("INSERT INTO config(name, value) VALUES(?, ?)", array('anon_id', $anon_id)); } if($images) { print("
Moving images..."); $new_db->Execute("DELETE FROM images"); $new_db->Execute(" INSERT INTO images(id, owner_id, owner_ip, filename, hash, ext, filesize, width, height, source, posted) VALUES (?, ?, ?, ?, ?, ?, 0, 0, 0, NULL, now())", $images); print("
Setting orphan images to anonymous..."); $orphans = $new_db->GetCol(" SELECT images.id FROM images LEFT JOIN users ON users.id = images.owner_id WHERE isnull(users.name)"); if($orphans) { foreach($orphans as $orphan) { $new_db->Execute("UPDATE images SET owner_id=? WHERE id=?", array($anon_id, $orphan)); } } } if($comments) { print("
Moving comments..."); // HAAAAAAAX! // the comments table is installed by an extension, so it won't be // ready when we need it... $new_db->Execute("DROP TABLE comments"); $new_db->Execute("DELETE FROM config WHERE name=?", array('ext_comments_version')); $new_db->Execute("CREATE TABLE `comments` ( `id` int(11) NOT NULL auto_increment, `image_id` int(11) NOT NULL, `owner_id` int(11) NOT NULL, `owner_ip` char(16) NOT NULL, `posted` datetime default NULL, `comment` text NOT NULL, PRIMARY KEY (`id`), KEY `comments_image_id` (`image_id`) )"); $new_db->Execute("INSERT INTO config(name, value) VALUES(?, ?)", array("ext_comments_version", 1)); $new_db->Execute("DELETE FROM comments"); $new_db->Execute(" INSERT INTO comments(id, image_id, owner_id, owner_ip, posted, comment) VALUES (?, ?, ?, ?, ?, ?)", $comments); print("
Setting orphan comments to anonymous..."); $orphans = $new_db->GetCol(" SELECT comments.id FROM comments LEFT JOIN users ON users.id = comments.owner_id WHERE isnull(users.name)"); if($orphans) { foreach($orphans as $orphan) { $new_db->Execute("UPDATE comments SET owner_id=? WHERE id=?", array($anon_id, $orphan)); } } } if($tags) { print("
Moving tags.."); $new_db->Execute("CREATE TABLE old_tags(image_id int, tag varchar(64))"); $new_db->Execute("DELETE FROM old_tags"); $new_db->Execute("INSERT INTO old_tags(image_id, tag) VALUES (?, ?)", $tags); $database->Execute("DELETE FROM tags"); $database->Execute("INSERT INTO tags(tag) SELECT DISTINCT tag FROM old_tags"); $database->Execute("DELETE FROM image_tags"); $database->Execute("INSERT INTO image_tags(image_id, tag_id) SELECT old_tags.image_id, tags.id FROM old_tags JOIN tags ON old_tags.tag = tags.tag"); $database->Execute("UPDATE tags SET count=(SELECT COUNT(image_id) FROM image_tags WHERE tag_id=tags.id GROUP BY tag_id)"); $new_db->Execute("DROP TABLE tags_tmp"); } print("
Moving files..."); $result = $new_db->Execute("SELECT * FROM images"); while(!$result->EOF) { $fields = $result->fields; $id = $fields['id']; $hash = $fields['hash']; $ext = $fields['ext']; $ab = substr($hash, 0, 2); if(file_exists("images/$ab/$hash")) { unlink("images/$ab/$hash"); } $fname = "$old_data/images/$id.$ext"; if(file_exists($fname)) { $size = filesize($fname); $sizekb = (int)($size/1024); $info = getimagesize($fname); if($info) { $width = $info[0]; $height = $info[1]; // print "
{$id}: {$width}x{$height}, {$sizekb}KB\n"; // noise $new_db->Execute("UPDATE images SET width=?, height=?, filesize=? WHERE id=?", array($width, $height, $size, $id)); } copy("$old_data/thumbs/$id.jpg", "thumbs/$ab/$hash"); copy("$old_data/images/$id.$ext", "images/$ab/$hash"); } else { print "
Warning: $fname not found; dropped from new database"; $new_db->Execute("DELETE FROM images WHERE id=?", array($id)); } $result->MoveNext(); } $new_db->Close(); } // }}} // table creation {{{ /* * Note: try and keep this as ANSI SQL compliant as possible, * so that we can (in theory) support other databases */ function create_tables_common($db, $auto_incrementing_id, $boolean, $true, $false, $ip) { $db->Execute("CREATE TABLE aliases ( oldtag VARCHAR(255) NOT NULL PRIMARY KEY, newtag VARCHAR(255) NOT NULL )"); $db->Execute("CREATE TABLE config ( name VARCHAR(255) NOT NULL PRIMARY KEY, value TEXT )"); $db->Execute("CREATE TABLE images ( id $auto_incrementing_id, owner_id INTEGER NOT NULL, owner_ip $ip, filename VARCHAR(64) NOT NULL DEFAULT '', filesize INTEGER NOT NULL, hash CHAR(32) NOT NULL UNIQUE, ext CHAR(4) NOT NULL, source VARCHAR(255), width INTEGER NOT NULL, height INTEGER NOT NULL, posted TIMESTAMP NOT NULL )"); $db->Execute("CREATE TABLE users ( id $auto_incrementing_id, name VARCHAR(32) NOT NULL UNIQUE, pass CHAR(32), joindate DATETIME NOT NULL, enabled $boolean NOT NULL DEFAULT $true, admin $boolean NOT NULL DEFAULT $false, email VARCHAR(255) )"); $db->Execute("CREATE TABLE layout ( title VARCHAR(64) PRIMARY KEY NOT NULL, section VARCHAR(32) NOT NULL DEFAULT 'left', position INTEGER NOT NULL DEFAULT 50, visible $boolean DEFAULT $true )"); $db->Execute("CREATE TABLE tags ( id $auto_incrementing_id, tag VARCHAR(64) NOT NULL UNIQUE, count INTEGER NOT NULL DEFAULT 0 )"); $db->Execute("CREATE INDEX tags__count ON tags(count)"); $db->Execute("CREATE TABLE image_tags ( image_id INTEGER NOT NULL DEFAULT 0, tag_id INTEGER NOT NULL DEFAULT 0, UNIQUE (image_id, tag_id) )"); $db->Execute("CREATE INDEX image_tags__tag_id ON image_tags(tag_id)"); $db->Execute("CREATE INDEX image_tags__image_id ON image_tags(image_id)"); $db->Execute("INSERT INTO config(name, value) VALUES(?, ?)", Array('db_version', 5)); } function create_tables_mysql($db) { $db->StartTrans(); $db->Execute("SET NAMES utf8"); create_tables_common($db, "INTEGER NOT NULL AUTO_INCREMENT PRIMARY KEY", "ENUM('Y', 'N')", "'Y'", "'N'", "CHAR(15)" ); return $db->CommitTrans(); } function create_tables_pgsql($db) { $db->StartTrans(); create_tables_common($db, "SERIAL NOT NULL PRIMARY KEY", "BOOLEAN", "True", "False", "INET" ); return $db->CommitTrans(); } function create_tables_sqlite($db) { $db->StartTrans(); create_tables_common($db, "INTEGER AUTOINCREMENT PRIMARY KEY NOT NULL", "CHAR(1)", "'Y'", "'N'", "CHAR(15)" ); return $db->CommitTrans(); } // }}} ?>