Intial revision.

This commit is contained in:
root 2010-03-21 14:50:44 -04:00
commit f87492d73e
19 changed files with 607 additions and 0 deletions

34
README Normal file
View file

@ -0,0 +1,34 @@
___ __ _ __ ___ __ ___ __
/ | ____/ /___ ___ (_)___ / |/ /_ _______/ /_/__ \/ /
/ /| |/ __ / __ `__ \/ / __ \ / /|_/ / / / / ___/ __ \/ _/ /
/ ___ / /_/ / / / / / / / / / / / / / / /_/ / /__/ / / /_//_/
/_/ |_\__,_/_/ /_/ /_/_/_/ /_/ /_/ /_/\__,_/\___/_/ /_(_)(_)
==================================================================
About
-----
Admin Much?! is a bad ass PHP back-end system for quickly and easily
administering submission-based photo sites running a WordPress front-end.
Intentions
----------
Hone the system and release it to the masses.
Requirements
------------
* WordPress - http://www.wordpress.com
* PICKLES - http://www.phpwithpickles.org
Who's using it?
---------------
Sites currently using Admin Much?! include:
* ParkMuch.com:
Photos of Bad Parking Jobs and other Automotive Shortcomings
* COMING SOON - LitterMuch.com

9
TODO Normal file
View file

@ -0,0 +1,9 @@
* Port system to run on the LATEST version of PICKLES.
* Move all hardcoded / site specific values to the config file.
* Add better spam detection to the check email script.
* Build set up scripts possibly automatically downloading WordPress.
* Add Geocoding support.

19
config.xml Normal file
View file

@ -0,0 +1,19 @@
<config>
<database>
<hostname>localhost</hostname>
<username>**********</username>
<password>**********</password>
<database>parkmuch</database>
</database>
<modules>
<display>Smarty</display>
</modules>
<templates>
<main>index.tpl</main>
</templates>
<admin>
<username>**********</username>
<password>********************************</password>
<salt>**********</salt>
</admin>
</config>

1
incoming/README Normal file
View file

@ -0,0 +1 @@
This is where images that are picked up by the check email script are stored before being promoted.

12
modules/admin.php Normal file
View file

@ -0,0 +1,12 @@
<?php
class admin extends Module
{
protected $authentication = true;
public function __default() {
$this->setPublic('messages', $this->db->getArray('SELECT * FROM incoming ORDER BY received_at'));
}
}
?>

26
modules/expunge.php Normal file
View file

@ -0,0 +1,26 @@
<?php
class expunge extends admin
{
public function __default()
{
$this->db->execute('DELETE FROM incoming WHERE id = "' . $_REQUEST['id'] . '";');
$path = getcwd() . '/../incoming/' . $_REQUEST['id'] . '/';
$files = scandir($path);
foreach ($files as $file)
{
if ($file != '.' && $file != '..')
{
unlink($path . $file);
}
}
rmdir($path);
header('Location: /admin');
}
}
?>

18
modules/file.php Normal file
View file

@ -0,0 +1,18 @@
<?php
class file extends admin
{
public function __default()
{
$path = getcwd() . '/../incoming/' . $_REQUEST['id'] . '/';
$files = scandir($path);
$file = $path . $files[2];
header('Content-Type: ' . mime_content_type($file));
$handle = fopen($file, 'rb');
fpassthru($handle);
exit();
}
}
?>

11
modules/home.php Normal file
View file

@ -0,0 +1,11 @@
<?php
class home extends Module
{
public function __default()
{
header('Location: http://parkmuch.com');
}
}
?>

28
modules/message.php Normal file
View file

@ -0,0 +1,28 @@
<?php
class message extends admin
{
public function __default()
{
$message = $this->db->getRow('SELECT * FROM incoming WHERE id = "' . $_REQUEST['id'] . '";');
$path = getcwd() . '/../incoming/' . $_REQUEST['id'] . '/';
$files = scandir($path);
foreach ($files as $file)
{
if ($file != '.' && $file != '..')
{
$filename = $path . $file;
}
}
$size = @getimagesize($filename);
$message['attachment'] = $files[2];
$message['details'] = $size;
$this->setPublic('message', $message);
}
}
?>

11
modules/post.php Normal file
View file

@ -0,0 +1,11 @@
<?php
class post extends message
{
public function __default()
{
parent::__default();
}
}
?>

61
modules/promote.php Normal file
View file

@ -0,0 +1,61 @@
<?php
class promote extends expunge
{
public function __default()
{
// Inserts the post into the database as a draft
$data = array(
'post_author' => '1',
'post_date' => date('Y-m-d H:i:s'),
'post_date_gmt' => gmdate('Y-m-d H:i:s'),
'post_title' => $_REQUEST['title'],
'post_status' => 'draft',
'post_name' => str_replace(' ' , '-', strtolower($_REQUEST['title'])),
'post_modified' => date('Y-m-d H:i:s'),
'post_modified_gmt' => gmdate('Y-m-d H:i:s')
);
$id = $this->db->insert('wp_posts', $data);
// Finds the image and extract the extension
$path = getcwd() . '/../incoming/' . $_REQUEST['id'] . '/';
$files = scandir($path);
foreach ($files as $file)
{
if ($file != '.' && $file != '..')
{
$filename = $path . $file;
$parts = explode('.', $file);
end($parts);
$extension = current($parts);
}
}
// Creates the directory for the image and moves the original
$public_path = '/submissions/' . date('Y/m/') . $id . '/';
$new_path = getcwd() . $public_path;
$original = $new_path . 'original' . $extension;
mkdir($new_path, 0777, true);
copy($filename, $original);
// Scales the image down to 500px wide
$thumb = new Imagick($original);
$thumb->thumbnailImage(500, 500, true);
$thumb->writeImage($new_path . 'scaled_500.' . $extension);
// Updates the post content and marks it as published
$data = array(
'post_content' => '<img src="http://images.parkmuch.com' . $public_path . 'scaled_500.' . $extension . '" /><br /><br />' . $_REQUEST['content'],
'post_status' => 'publish',
'guid' => 'http://parkmuch.com/?p=' . $id
);
$this->db->update('wp_posts', $data, array('ID' => $id));
// Expunges the data
parent::__default();
}
}
?>

43
public/.htaccess Normal file
View file

@ -0,0 +1,43 @@
# Sets up ETags
FileETag MTime Size
# Sets up the mod_rewrite engine
RewriteEngine on
#RewriteCond %{REQUEST_FILENAME}/$ -f [NC,OR]
#RewriteCond %{REQUEST_FILENAME}/$ -d [NC]
#RewriteRule .* - [L]
# Sets the base path (document root)
RewriteBase /
# Strips the trailing slash
RewriteRule ^(.+)/$ $1 [R]
#RewriteCond %{HTTP_HOST} ^(admin).thatgirljen.com$ [NC]
#RewriteRule ^(.*)$ http://thatgirljen.com/admin [R=301,L]
# Strips the preceeding subdomain
#RewriteCond %{HTTP_HOST} ^(.+).thatgirljen.com$ [NC]
#RewriteRule ^(.*)$ http://thatgirljen.com/$1 [R=301,L]
# Rewrite Rules for the PICKLES Quaternity
RewriteRule ^(template/edit)/([a-z-/]+)$ index.php?module=$1&template=$2 [NC,QSA]
RewriteRule ^(weblog)/([0-9]+)$ index.php?module=$1&page=$2 [NC,QSA]
RewriteCond %{DOCUMENT_ROOT}%{REQUEST_URI} !-f
RewriteRule ^([a-z-/]+)/([0-9/]{10})/([a-z-]+)$ index.php?module=$1&date=$2&title=$3 [NC,QSA]
RewriteRule ^([a-z-/]+)/([0-9]+)$ index.php?module=$1&id=$2 [NC,QSA]
RewriteRule ^([a-z-/]+)$ index.php?module=$1 [NC,QSA]
# Set up the error documents
ErrorDocument 400 /
ErrorDocument 401 /
ErrorDocument 403 /
ErrorDocument 404 /
ErrorDocument 500 /
# Blocks access to .htaccess
<Files .htaccess>
order allow,deny
deny from all
</Files>

6
public/index.php Normal file
View file

@ -0,0 +1,6 @@
<?php
require_once 'pickles.php';
new Controller();
?>

View file

@ -0,0 +1 @@
This is where images that are promoted from the upcoming directory are stored.

227
scripts/check_mail.php Normal file
View file

@ -0,0 +1,227 @@
<?php
// Sets the server name manually as the value isn't present from the CLI
$_SERVER['SERVER_NAME'] = 'parkmuch.com';
$path = '/home/gravityboulevard/domains/images.parkmuch.com/';
// Sets up our PICKLES environment
require '/usr/share/pickles/pickles.php';
$config = new Config($path . 'config.xml');
$error = new Error();
$db = new DB($config, $error);
Logger::write('check_mail', 'Check mail job has started');
// Connects to the mail bo
$mailbox = imap_open('{mail.parkmuch.com:143/novalidate-cert}INBOX', 'submission@parkmuch.com', '**********');
// Pulls the number of messages
$count = imap_num_msg($mailbox);
$thin_line = "\n" . '----------------------------------------------------------------' . "\n";
$thick_line = "\n" . '================================================================' . "\n";
$alerted = false;
// Loops through the messages
for ($i = 1; $i <= $count; $i++)
{
echo "\n" . $thick_line . 'Message ' . $i . ' of ' . $count . ': ' . $thin_line;
// Pulls information from the message header
$header = imap_headerinfo($mailbox, $i);
$time = date('Y-m-d H:i:s', strtotime($header->date));
$from = $header->fromaddress;
$subject = $header->subject;
// Gets the message structure
$structure = imap_fetchstructure($mailbox, $i);
// Pulls the plain text and HTML message body
$message = get_part($mailbox, $i, 'TEXT/PLAIN', $structure);
// Pulls the attachment if any
$attachments = array();
if (isset($structure->parts) && count($structure->parts))
{
$part_count = count($structure->parts);
for ($j = 0; $j < $part_count; $j++)
{
if ($structure->parts[$j]->ifdparameters)
{
foreach ($structure->parts[$j]->dparameters as $object)
{
if (strtolower($object->attribute) == 'filename')
{
$attachments[$j]['is_attachment'] = true;
$attachments[$j]['filename'] = $object->value;
}
}
}
if ($structure->parts[$j]->ifparameters)
{
foreach ($structure->parts[$j]->parameters as $object)
{
if (strtolower($object->attribute) == 'name')
{
$attachments[$j]['is_attachment'] = true;
$attachments[$j]['name'] = $object->value;
}
}
}
if ($attachments[$j]['is_attachment'])
{
$attachments[$j]['attachment'] = imap_fetchbody($mailbox, $i, $j + 1);
// 3 = BASE64
if ($structure->parts[$j]->encoding == 3)
{
$attachments[$j]['attachment'] = base64_decode($attachments[$j]['attachment']);
}
// 4 = QUOTED-PRINTABLE
elseif ($structure->parts[$j]->encoding == 4)
{
$attachments[$j]['attachment'] = quoted_printable_decode($attachments[$j]['attachment']);
}
}
}
}
echo 'Time: ' . $time . "\n";
echo 'From: ' . $from . "\n";
echo 'Subject: ' . $subject;
echo $thin_line . 'Message: ' . $thin_line . "\n" . $message;
if (count($attachments) > 0)
{
$displayed = false;
foreach ($attachments as $attachment)
{
if ($displayed == false)
{
echo $thin_line . 'Attachments:';
$displayed = true;
}
$data = array(
'sender' => $from,
'subject' => $subject,
'message' => $message,
'received_at' => $time
);
$id = $db->insert('incoming', $data);
$incoming_path = $path . 'incoming/' . $id . '/';
if (!file_exists($incoming_path))
{
mkdir($incoming_path, 0777, true);
}
// Places each attachment in the appropriate location
file_put_contents($incoming_path . $attachment['filename'], $attachment['attachment']);
//chmod($incoming_path, 0777);
chgrp($incoming_path, 'www-data');
chown($incoming_path, 'www-data');
//chmod($incoming_path . $attachment['filename'], 0777);
chgrp($incoming_path . $attachment['filename'], 'www-data');
chown($incoming_path . $attachment['filename'], 'www-data');
echo "\n" . $attachment['name'];
}
// Marks the message for deletion
imap_delete($mailbox, $i);
// Sends an alert if one hasn't been sent already
if ($alerted == false)
{
mail('8134952668@tmomail.net', 'NEW PARK MUCH?! SUBMISSION', 'Get on that shit!');
$alerted = true;
}
}
echo $thick_line;
}
// Closes the mail box
imap_expunge($mailbox);
imap_close($mailbox);
Logger::write('check_mail', 'Check mail job has completed');
function get_mime_type($structure)
{
$primary_mime_type = array('TEXT', 'MULTIPART','MESSAGE', 'APPLICATION', 'AUDIO','IMAGE', 'VIDEO', 'OTHER');
if ($structure->subtype)
{
return $primary_mime_type[(int)$structure->type] . '/' . $structure->subtype;
}
return 'TEXT/PLAIN';
}
function get_part($stream, $msg_number, $mime_type, $structure = false, $part_number = false)
{
if (!$structure)
{
$structure = imap_fetchstructure($stream, $msg_number);
}
if ($structure)
{
if ($mime_type == get_mime_type($structure))
{
if(!$part_number)
{
$part_number = "1";
}
$text = imap_fetchbody($stream, $msg_number, $part_number);
if ($structure->encoding == 3)
{
return imap_base64($text);
}
elseif ($structure->encoding == 4)
{
return imap_qprint($text);
}
else
{
return $text;
}
}
// Multi-part
if($structure->type == 1)
{
while (list($index, $sub_structure) = each($structure->parts))
{
if ($part_number)
{
$prefix = $part_number . '.';
}
$data = get_part($stream, $msg_number, $mime_type, $sub_structure, $prefix . ($index + 1));
if ($data)
{
return $data;
}
}
}
}
return false;
}
?>

20
templates/admin.tpl Normal file
View file

@ -0,0 +1,20 @@
{foreach from=$module.messages item=message name=messages}
{if $smarty.foreach.messages.first}
<table border="1">
<tr>
<th>ID</th>
<th>From</th>
<th>Subject</th>
<th>Received At</th>
</tr>
{/if}
<tr>
<td><a href="/message/{$message.id}">{$message.id}</a></td>
<td>{$message.sender|htmlentities}</td>
<td>{$message.subject}</td>
<td>{$message.received_at}</td>
</tr>
{if $smarty.foreach.messages.last}</table>{/if}
{foreachelse}
<em>No incoming submissions at this time</em>
{/foreach}

9
templates/index.tpl Normal file
View file

@ -0,0 +1,9 @@
<html>
<head>
<title>Park Much?! Admin</title>
</head>
<body>
<h1>Park Much?! Admin</h1>
{include file="$template"}
</body>
</html>

36
templates/message.tpl Normal file
View file

@ -0,0 +1,36 @@
{literal}<style>th { text-align: right; vertical-align: top }</style>{/literal}
<table>
<tr>
<th>Received At:</th>
<td>{$module.message.received_at}</td>
</tr>
<tr>
<th>From:</th>
<td>{$module.message.sender|htmlentities}</td>
</tr>
<tr>
<th>Subject:</th>
<td>{$module.message.subject}</td>
</tr>
<tr>
<th>Message:</th>
<td>{$module.message.message|nl2br}</td>
</tr>
<tr>
<th>Attachment:</th>
<td>
<a href="/file/{$module.message.id}" target="_blank">{$module.message.attachment}</a>
{if $module.message.details !== false}
<br />
<img src="/file/{$module.message.id}" width="200" />
{/if}
</td>
</tr>
<tr>
<td colspan="2">
<br />
<button onclick="if (confirm('Are you sure? Seriously, there\'s no undo')) document.location.href='/expunge/{$module.message.id}';">Expunge</button>
<button onclick="document.location.href='/post/{$module.message.id}'">Promote</button>
</td>
</tr>
</table>

35
templates/post.tpl Normal file
View file

@ -0,0 +1,35 @@
{literal}<style>th { text-align: right; vertical-align: top }</style>{/literal}
<form method="post" action="/promote" onsubmit="if (!confirm('Did you double-check everything? Are you sure??')) return false;">
<table>
<tr>
<th>Title:</th>
<td><input type="text" name="title" value="{$module.message.subject}" /></td>
</tr>
<tr>
<th>Content:</th>
<td>
<textarea name="content" cols="100" rows="10">
{$module.message.message}<br /><br />
Submitted by {$module.message.sender}
</textarea>
</td>
</tr>
<tr>
<th>Attachment:</th>
<td>
<a href="/file/{$module.message.id}" target="_blank">{$module.message.attachment}</a>
{if $module.message.details !== false}
<br />
<img src="/file/{$module.message.id}" width="200" />
{/if}
</td>
</tr>
<tr>
<td colspan="2">
<br />
<input type="hidden" name="id" value="{$module.message.id}" />
<button>Promote!</button>
</td>
</tr>
</table>
</form>