Commit 0794e784 by Scott

Coding style (db posts/users)

parent f5aaabc6
...@@ -20,88 +20,109 @@ ...@@ -20,88 +20,109 @@
More about this license: http://www.question2answer.org/license.php More about this license: http://www.question2answer.org/license.php
*/ */
if (!defined('QA_VERSION')) { // don't allow this page to be requested directly from browser if (!defined('QA_VERSION')) { // don't allow this page to be requested directly from browser
header('Location: ../'); header('Location: ../');
exit; exit;
} }
function qa_db_post_create($type, $parentid, $userid, $cookieid, $ip, $title, $content, $format, $tagstring, $notify, $categoryid=null, $name=null) /**
/* * Create a new post in the database and return its ID (based on auto-incrementing)
Create a new post in the database and return its ID (based on auto-incrementing) * @param $type
*/ * @param $parentid
{ * @param $userid
* @param $cookieid
* @param $ip
* @param $title
* @param $content
* @param $format
* @param $tagstring
* @param $notify
* @param $categoryid
* @param $name
* @return mixed
*/
function qa_db_post_create($type, $parentid, $userid, $cookieid, $ip, $title, $content, $format, $tagstring, $notify, $categoryid = null, $name = null)
{
qa_db_query_sub( qa_db_query_sub(
'INSERT INTO ^posts (categoryid, type, parentid, userid, cookieid, createip, title, content, format, tags, notify, name, created) '. 'INSERT INTO ^posts (categoryid, type, parentid, userid, cookieid, createip, title, content, format, tags, notify, name, created) ' .
'VALUES (#, $, #, $, #, $, $, $, $, $, $, $, NOW())', 'VALUES (#, $, #, $, #, $, $, $, $, $, $, $, NOW())',
$categoryid, $type, $parentid, $userid, $cookieid, @inet_pton($ip), $title, $content, $format, $tagstring, $notify, $name $categoryid, $type, $parentid, $userid, $cookieid, @inet_pton($ip), $title, $content, $format, $tagstring, $notify, $name
); );
return qa_db_last_insert_id(); return qa_db_last_insert_id();
} }
function qa_db_posts_calc_category_path($firstpostid, $lastpostid=null) /**
/* * Recalculate the full category path (i.e. columns catidpath1/2/3) for posts from $firstpostid to $lastpostid (if specified)
Recalculate the full category path (i.e. columns catidpath1/2/3) for posts from $firstpostid to $lastpostid (if specified) * @param $firstpostid
*/ * @param $lastpostid
{ */
function qa_db_posts_calc_category_path($firstpostid, $lastpostid = null)
{
if (!isset($lastpostid)) if (!isset($lastpostid))
$lastpostid=$firstpostid; $lastpostid = $firstpostid;
qa_db_query_sub( qa_db_query_sub(
"UPDATE ^posts AS x, (SELECT ^posts.postid, ". "UPDATE ^posts AS x, (SELECT ^posts.postid, " .
"COALESCE(parent2.parentid, parent1.parentid, parent0.parentid, parent0.categoryid) AS catidpath1, ". "COALESCE(parent2.parentid, parent1.parentid, parent0.parentid, parent0.categoryid) AS catidpath1, " .
"IF (parent2.parentid IS NOT NULL, parent1.parentid, IF (parent1.parentid IS NOT NULL, parent0.parentid, IF (parent0.parentid IS NOT NULL, parent0.categoryid, NULL))) AS catidpath2, ". "IF (parent2.parentid IS NOT NULL, parent1.parentid, IF (parent1.parentid IS NOT NULL, parent0.parentid, IF (parent0.parentid IS NOT NULL, parent0.categoryid, NULL))) AS catidpath2, " .
"IF (parent2.parentid IS NOT NULL, parent0.parentid, IF (parent1.parentid IS NOT NULL, parent0.categoryid, NULL)) AS catidpath3 ". "IF (parent2.parentid IS NOT NULL, parent0.parentid, IF (parent1.parentid IS NOT NULL, parent0.categoryid, NULL)) AS catidpath3 " .
"FROM ^posts LEFT JOIN ^categories AS parent0 ON ^posts.categoryid=parent0.categoryid LEFT JOIN ^categories AS parent1 ON parent0.parentid=parent1.categoryid LEFT JOIN ^categories AS parent2 ON parent1.parentid=parent2.categoryid WHERE ^posts.postid BETWEEN # AND #) AS a SET x.catidpath1=a.catidpath1, x.catidpath2=a.catidpath2, x.catidpath3=a.catidpath3 WHERE x.postid=a.postid", "FROM ^posts LEFT JOIN ^categories AS parent0 ON ^posts.categoryid=parent0.categoryid LEFT JOIN ^categories AS parent1 ON parent0.parentid=parent1.categoryid LEFT JOIN ^categories AS parent2 ON parent1.parentid=parent2.categoryid WHERE ^posts.postid BETWEEN # AND #) AS a SET x.catidpath1=a.catidpath1, x.catidpath2=a.catidpath2, x.catidpath3=a.catidpath3 WHERE x.postid=a.postid",
$firstpostid, $lastpostid $firstpostid, $lastpostid
); // requires QA_CATEGORY_DEPTH=4 ); // requires QA_CATEGORY_DEPTH=4
} }
function qa_db_post_get_category_path($postid) /**
/* * Get the full category path (including categoryid) for $postid
Get the full category path (including categoryid) for $postid * @param $postid
*/ * @return array|null
{ */
function qa_db_post_get_category_path($postid)
{
return qa_db_read_one_assoc(qa_db_query_sub( return qa_db_read_one_assoc(qa_db_query_sub(
'SELECT categoryid, catidpath1, catidpath2, catidpath3 FROM ^posts WHERE postid=#', 'SELECT categoryid, catidpath1, catidpath2, catidpath3 FROM ^posts WHERE postid=#',
$postid $postid
)); // requires QA_CATEGORY_DEPTH=4 )); // requires QA_CATEGORY_DEPTH=4
} }
function qa_db_post_acount_update($questionid) /**
/* * Update the cached number of answers for $questionid in the database, along with the highest netvotes of any of its answers
Update the cached number of answers for $questionid in the database, along with the highest netvotes of any of its answers * @param $questionid
*/ */
{ function qa_db_post_acount_update($questionid)
if (qa_should_update_counts()) {
if (qa_should_update_counts()) {
qa_db_query_sub( qa_db_query_sub(
"UPDATE ^posts AS x, (SELECT COUNT(*) AS acount, COALESCE(GREATEST(MAX(netvotes), 0), 0) AS amaxvote FROM ^posts WHERE parentid=# AND type='A') AS a SET x.acount=a.acount, x.amaxvote=a.amaxvote WHERE x.postid=#", "UPDATE ^posts AS x, (SELECT COUNT(*) AS acount, COALESCE(GREATEST(MAX(netvotes), 0), 0) AS amaxvote FROM ^posts WHERE parentid=# AND type='A') AS a SET x.acount=a.acount, x.amaxvote=a.amaxvote WHERE x.postid=#",
$questionid, $questionid $questionid, $questionid
); );
} }
}
function qa_db_category_path_qcount_update($path) /**
/* * Recalculate the number of questions for each category in $path retrieved via qa_db_post_get_category_path()
Recalculate the number of questions for each category in $path retrieved via qa_db_post_get_category_path() * @param $path
*/ */
{ function qa_db_category_path_qcount_update($path)
{
qa_db_ifcategory_qcount_update($path['categoryid']); // requires QA_CATEGORY_DEPTH=4 qa_db_ifcategory_qcount_update($path['categoryid']); // requires QA_CATEGORY_DEPTH=4
qa_db_ifcategory_qcount_update($path['catidpath1']); qa_db_ifcategory_qcount_update($path['catidpath1']);
qa_db_ifcategory_qcount_update($path['catidpath2']); qa_db_ifcategory_qcount_update($path['catidpath2']);
qa_db_ifcategory_qcount_update($path['catidpath3']); qa_db_ifcategory_qcount_update($path['catidpath3']);
} }
function qa_db_ifcategory_qcount_update($categoryid) /**
/* * Update the cached number of questions for category $categoryid in the database, including its subcategories
Update the cached number of questions for category $categoryid in the database, including its subcategories * @param $categoryid
*/ */
{ function qa_db_ifcategory_qcount_update($categoryid)
{
if (qa_should_update_counts() && isset($categoryid)) { if (qa_should_update_counts() && isset($categoryid)) {
// This seemed like the most sensible approach which avoids explicitly calculating the category's depth in the hierarchy // This seemed like the most sensible approach which avoids explicitly calculating the category's depth in the hierarchy
...@@ -110,182 +131,208 @@ ...@@ -110,182 +131,208 @@
$categoryid, $categoryid, $categoryid, $categoryid, $categoryid $categoryid, $categoryid, $categoryid, $categoryid, $categoryid
); // requires QA_CATEGORY_DEPTH=4 ); // requires QA_CATEGORY_DEPTH=4
} }
} }
function qa_db_titlewords_add_post_wordids($postid, $wordids) /**
/* * Add rows into the database title index, where $postid contains the words $wordids - this does the same sort
Add rows into the database title index, where $postid contains the words $wordids - this does the same sort * of thing as qa_db_posttags_add_post_wordids() in a different way, for no particularly good reason.
of thing as qa_db_posttags_add_post_wordids() in a different way, for no particularly good reason. * @param $postid
*/ * @param $wordids
{ */
function qa_db_titlewords_add_post_wordids($postid, $wordids)
{
if (count($wordids)) { if (count($wordids)) {
$rowstoadd=array(); $rowstoadd = array();
foreach ($wordids as $wordid) foreach ($wordids as $wordid)
$rowstoadd[]=array($postid, $wordid); $rowstoadd[] = array($postid, $wordid);
qa_db_query_sub( qa_db_query_sub(
'INSERT INTO ^titlewords (postid, wordid) VALUES #', 'INSERT INTO ^titlewords (postid, wordid) VALUES #',
$rowstoadd $rowstoadd
); );
} }
} }
function qa_db_contentwords_add_post_wordidcounts($postid, $type, $questionid, $wordidcounts) /**
/* * Add rows into the database content index, where $postid (of $type, with the antecedent $questionid)
Add rows into the database content index, where $postid (of $type, with the antecedent $questionid) * has words as per the keys of $wordidcounts, and the corresponding number of those words in the values.
has words as per the keys of $wordidcounts, and the corresponding number of those words in the values. * @param $postid
*/ * @param $type
{ * @param $questionid
* @param $wordidcounts
*/
function qa_db_contentwords_add_post_wordidcounts($postid, $type, $questionid, $wordidcounts)
{
if (count($wordidcounts)) { if (count($wordidcounts)) {
$rowstoadd=array(); $rowstoadd = array();
foreach ($wordidcounts as $wordid => $count) foreach ($wordidcounts as $wordid => $count)
$rowstoadd[]=array($postid, $wordid, $count, $type, $questionid); $rowstoadd[] = array($postid, $wordid, $count, $type, $questionid);
qa_db_query_sub( qa_db_query_sub(
'INSERT INTO ^contentwords (postid, wordid, count, type, questionid) VALUES #', 'INSERT INTO ^contentwords (postid, wordid, count, type, questionid) VALUES #',
$rowstoadd $rowstoadd
); );
} }
} }
function qa_db_tagwords_add_post_wordids($postid, $wordids) /**
/* * Add rows into the database index of individual tag words, where $postid contains the words $wordids
Add rows into the database index of individual tag words, where $postid contains the words $wordids * @param $postid
*/ * @param $wordids
{ */
function qa_db_tagwords_add_post_wordids($postid, $wordids)
{
if (count($wordids)) { if (count($wordids)) {
$rowstoadd=array(); $rowstoadd = array();
foreach ($wordids as $wordid) foreach ($wordids as $wordid)
$rowstoadd[]=array($postid, $wordid); $rowstoadd[] = array($postid, $wordid);
qa_db_query_sub( qa_db_query_sub(
'INSERT INTO ^tagwords (postid, wordid) VALUES #', 'INSERT INTO ^tagwords (postid, wordid) VALUES #',
$rowstoadd $rowstoadd
); );
} }
} }
function qa_db_posttags_add_post_wordids($postid, $wordids) /**
/* * Add rows into the database index of whole tags, where $postid contains the tags $wordids
Add rows into the database index of whole tags, where $postid contains the tags $wordids * @param $postid
*/ * @param $wordids
{ */
if (count($wordids)) function qa_db_posttags_add_post_wordids($postid, $wordids)
{
if (count($wordids)) {
qa_db_query_sub( qa_db_query_sub(
'INSERT INTO ^posttags (postid, wordid, postcreated) SELECT postid, wordid, created FROM ^words, ^posts WHERE postid=# AND wordid IN ($)', 'INSERT INTO ^posttags (postid, wordid, postcreated) SELECT postid, wordid, created FROM ^words, ^posts WHERE postid=# AND wordid IN ($)',
$postid, $wordids $postid, $wordids
); );
} }
}
function qa_db_word_mapto_ids($words) /**
/* * Return an array mapping each word in $words to its corresponding wordid in the database
Return an array mapping each word in $words to its corresponding wordid in the database * @param $words
*/ * @return array
{ */
if (count($words)) function qa_db_word_mapto_ids($words)
{
if (count($words)) {
return qa_db_read_all_assoc(qa_db_query_sub( return qa_db_read_all_assoc(qa_db_query_sub(
'SELECT wordid, word FROM ^words WHERE word IN ($)', $words 'SELECT wordid, word FROM ^words WHERE word IN ($)', $words
), 'word', 'wordid'); ), 'word', 'wordid');
else
return array();
} }
return array();
}
function qa_db_word_mapto_ids_add($words)
/*
Return an array mapping each word in $words to its corresponding wordid in the database, adding any that are missing
*/
{
$wordtoid=qa_db_word_mapto_ids($words);
$wordstoadd=array(); /**
foreach ($words as $word) * Return an array mapping each word in $words to its corresponding wordid in the database, adding any that are missing
* @param $words
* @return array
*/
function qa_db_word_mapto_ids_add($words)
{
$wordtoid = qa_db_word_mapto_ids($words);
$wordstoadd = array();
foreach ($words as $word) {
if (!isset($wordtoid[$word])) if (!isset($wordtoid[$word]))
$wordstoadd[]=$word; $wordstoadd[] = $word;
}
if (count($wordstoadd)) { if (count($wordstoadd)) {
qa_db_query_sub('LOCK TABLES ^words WRITE'); // to prevent two requests adding the same word qa_db_query_sub('LOCK TABLES ^words WRITE'); // to prevent two requests adding the same word
$wordtoid=qa_db_word_mapto_ids($words); // map it again in case table content changed before it was locked $wordtoid = qa_db_word_mapto_ids($words); // map it again in case table content changed before it was locked
$rowstoadd=array(); $rowstoadd = array();
foreach ($words as $word) foreach ($words as $word) {
if (!isset($wordtoid[$word])) if (!isset($wordtoid[$word]))
$rowstoadd[]=array($word); $rowstoadd[] = array($word);
}
qa_db_query_sub('INSERT IGNORE INTO ^words (word) VALUES $', $rowstoadd); qa_db_query_sub('INSERT IGNORE INTO ^words (word) VALUES $', $rowstoadd);
qa_db_query_sub('UNLOCK TABLES'); qa_db_query_sub('UNLOCK TABLES');
$wordtoid=qa_db_word_mapto_ids($words); // do it one last time $wordtoid = qa_db_word_mapto_ids($words); // do it one last time
} }
return $wordtoid; return $wordtoid;
} }
function qa_db_word_titlecount_update($wordids) /**
/* * Update the titlecount column in the database for the words in $wordids, based on how many posts they appear in the title of
Update the titlecount column in the database for the words in $wordids, based on how many posts they appear in the title of * @param $wordids
*/ */
{ function qa_db_word_titlecount_update($wordids)
if (qa_should_update_counts() && count($wordids)) {
if (qa_should_update_counts() && count($wordids)) {
qa_db_query_sub( qa_db_query_sub(
'UPDATE ^words AS x, (SELECT ^words.wordid, COUNT(^titlewords.wordid) AS titlecount FROM ^words LEFT JOIN ^titlewords ON ^titlewords.wordid=^words.wordid WHERE ^words.wordid IN (#) GROUP BY wordid) AS a SET x.titlecount=a.titlecount WHERE x.wordid=a.wordid', 'UPDATE ^words AS x, (SELECT ^words.wordid, COUNT(^titlewords.wordid) AS titlecount FROM ^words LEFT JOIN ^titlewords ON ^titlewords.wordid=^words.wordid WHERE ^words.wordid IN (#) GROUP BY wordid) AS a SET x.titlecount=a.titlecount WHERE x.wordid=a.wordid',
$wordids $wordids
); );
} }
}
function qa_db_word_contentcount_update($wordids) /**
/* * Update the contentcount column in the database for the words in $wordids, based on how many posts they appear in the content of
Update the contentcount column in the database for the words in $wordids, based on how many posts they appear in the content of * @param $wordids
*/ */
{ function qa_db_word_contentcount_update($wordids)
if (qa_should_update_counts() && count($wordids)) {
if (qa_should_update_counts() && count($wordids)) {
qa_db_query_sub( qa_db_query_sub(
'UPDATE ^words AS x, (SELECT ^words.wordid, COUNT(^contentwords.wordid) AS contentcount FROM ^words LEFT JOIN ^contentwords ON ^contentwords.wordid=^words.wordid WHERE ^words.wordid IN (#) GROUP BY wordid) AS a SET x.contentcount=a.contentcount WHERE x.wordid=a.wordid', 'UPDATE ^words AS x, (SELECT ^words.wordid, COUNT(^contentwords.wordid) AS contentcount FROM ^words LEFT JOIN ^contentwords ON ^contentwords.wordid=^words.wordid WHERE ^words.wordid IN (#) GROUP BY wordid) AS a SET x.contentcount=a.contentcount WHERE x.wordid=a.wordid',
$wordids $wordids
); );
} }
}
function qa_db_word_tagwordcount_update($wordids) /**
/* * Update the tagwordcount column in the database for the individual tag words in $wordids, based on how many posts they appear in the tags of
Update the tagwordcount column in the database for the individual tag words in $wordids, based on how many posts they appear in the tags of * @param $wordids
*/ */
{ function qa_db_word_tagwordcount_update($wordids)
if (qa_should_update_counts() && count($wordids)) {
if (qa_should_update_counts() && count($wordids)) {
qa_db_query_sub( qa_db_query_sub(
'UPDATE ^words AS x, (SELECT ^words.wordid, COUNT(^tagwords.wordid) AS tagwordcount FROM ^words LEFT JOIN ^tagwords ON ^tagwords.wordid=^words.wordid WHERE ^words.wordid IN (#) GROUP BY wordid) AS a SET x.tagwordcount=a.tagwordcount WHERE x.wordid=a.wordid', 'UPDATE ^words AS x, (SELECT ^words.wordid, COUNT(^tagwords.wordid) AS tagwordcount FROM ^words LEFT JOIN ^tagwords ON ^tagwords.wordid=^words.wordid WHERE ^words.wordid IN (#) GROUP BY wordid) AS a SET x.tagwordcount=a.tagwordcount WHERE x.wordid=a.wordid',
$wordids $wordids
); );
} }
}
function qa_db_word_tagcount_update($wordids) /**
/* * Update the tagcount column in the database for the whole tags in $wordids, based on how many posts they appear as tags of
Update the tagcount column in the database for the whole tags in $wordids, based on how many posts they appear as tags of * @param $wordids
*/ */
{ function qa_db_word_tagcount_update($wordids)
if (qa_should_update_counts() && count($wordids)) {
if (qa_should_update_counts() && count($wordids)) {
qa_db_query_sub( qa_db_query_sub(
'UPDATE ^words AS x, (SELECT ^words.wordid, COUNT(^posttags.wordid) AS tagcount FROM ^words LEFT JOIN ^posttags ON ^posttags.wordid=^words.wordid WHERE ^words.wordid IN (#) GROUP BY wordid) AS a SET x.tagcount=a.tagcount WHERE x.wordid=a.wordid', 'UPDATE ^words AS x, (SELECT ^words.wordid, COUNT(^posttags.wordid) AS tagcount FROM ^words LEFT JOIN ^posttags ON ^posttags.wordid=^words.wordid WHERE ^words.wordid IN (#) GROUP BY wordid) AS a SET x.tagcount=a.tagcount WHERE x.wordid=a.wordid',
$wordids $wordids
); );
} }
}
function qa_db_qcount_update() /**
/* * Update the cached count in the database of the number of questions (excluding hidden/queued)
Update the cached count in the database of the number of questions (excluding hidden/queued) */
*/ function qa_db_qcount_update()
{ {
if (qa_should_update_counts()) { if (qa_should_update_counts()) {
qa_db_query_sub( qa_db_query_sub(
"INSERT INTO ^options (title, content) " . "INSERT INTO ^options (title, content) " .
...@@ -294,14 +341,14 @@ ...@@ -294,14 +341,14 @@
"ON DUPLICATE KEY UPDATE content = VALUES(content)" "ON DUPLICATE KEY UPDATE content = VALUES(content)"
); );
} }
} }
function qa_db_acount_update() /**
/* * Update the cached count in the database of the number of answers (excluding hidden/queued)
Update the cached count in the database of the number of answers (excluding hidden/queued) */
*/ function qa_db_acount_update()
{ {
if (qa_should_update_counts()) { if (qa_should_update_counts()) {
qa_db_query_sub( qa_db_query_sub(
"INSERT INTO ^options (title, content) " . "INSERT INTO ^options (title, content) " .
...@@ -310,14 +357,14 @@ ...@@ -310,14 +357,14 @@
"ON DUPLICATE KEY UPDATE content = VALUES(content)" "ON DUPLICATE KEY UPDATE content = VALUES(content)"
); );
} }
} }
function qa_db_ccount_update() /**
/* * Update the cached count in the database of the number of comments (excluding hidden/queued)
Update the cached count in the database of the number of comments (excluding hidden/queued) */
*/ function qa_db_ccount_update()
{ {
if (qa_should_update_counts()) { if (qa_should_update_counts()) {
qa_db_query_sub( qa_db_query_sub(
"INSERT INTO ^options (title, content) " . "INSERT INTO ^options (title, content) " .
...@@ -326,14 +373,14 @@ ...@@ -326,14 +373,14 @@
"ON DUPLICATE KEY UPDATE content = VALUES(content)" "ON DUPLICATE KEY UPDATE content = VALUES(content)"
); );
} }
} }
function qa_db_tagcount_update() /**
/* * Update the cached count in the database of the number of different tags used
Update the cached count in the database of the number of different tags used */
*/ function qa_db_tagcount_update()
{ {
if (qa_should_update_counts()) { if (qa_should_update_counts()) {
qa_db_query_sub( qa_db_query_sub(
"INSERT INTO ^options (title, content) " . "INSERT INTO ^options (title, content) " .
...@@ -342,14 +389,14 @@ ...@@ -342,14 +389,14 @@
"ON DUPLICATE KEY UPDATE content = VALUES(content)" "ON DUPLICATE KEY UPDATE content = VALUES(content)"
); );
} }
} }
function qa_db_unaqcount_update() /**
/* * Update the cached count in the database of the number of unanswered questions (excluding hidden/queued)
Update the cached count in the database of the number of unanswered questions (excluding hidden/queued) */
*/ function qa_db_unaqcount_update()
{ {
if (qa_should_update_counts()) { if (qa_should_update_counts()) {
qa_db_query_sub( qa_db_query_sub(
"INSERT INTO ^options (title, content) " . "INSERT INTO ^options (title, content) " .
...@@ -358,14 +405,14 @@ ...@@ -358,14 +405,14 @@
"ON DUPLICATE KEY UPDATE content = VALUES(content)" "ON DUPLICATE KEY UPDATE content = VALUES(content)"
); );
} }
} }
function qa_db_unselqcount_update() /**
/* * Update the cached count in the database of the number of questions with no answer selected (excluding hidden/queued)
Update the cached count in the database of the number of questions with no answer selected (excluding hidden/queued) */
*/ function qa_db_unselqcount_update()
{ {
if (qa_should_update_counts()) { if (qa_should_update_counts()) {
qa_db_query_sub( qa_db_query_sub(
"INSERT INTO ^options (title, content) " . "INSERT INTO ^options (title, content) " .
...@@ -374,14 +421,14 @@ ...@@ -374,14 +421,14 @@
"ON DUPLICATE KEY UPDATE content = VALUES(content)" "ON DUPLICATE KEY UPDATE content = VALUES(content)"
); );
} }
} }
function qa_db_unupaqcount_update() /**
/* * Update the cached count in the database of the number of questions with no upvoted answers (excluding hidden/queued)
Update the cached count in the database of the number of questions with no upvoted answers (excluding hidden/queued) */
*/ function qa_db_unupaqcount_update()
{ {
if (qa_should_update_counts()) { if (qa_should_update_counts()) {
qa_db_query_sub( qa_db_query_sub(
"INSERT INTO ^options (title, content) " . "INSERT INTO ^options (title, content) " .
...@@ -390,14 +437,14 @@ ...@@ -390,14 +437,14 @@
"ON DUPLICATE KEY UPDATE content = VALUES(content)" "ON DUPLICATE KEY UPDATE content = VALUES(content)"
); );
} }
} }
function qa_db_queuedcount_update() /**
/* * Update the cached count in the database of the number of posts which are queued for moderation
Update the cached count in the database of the number of posts which are queued for moderation */
*/ function qa_db_queuedcount_update()
{ {
if (qa_should_update_counts()) { if (qa_should_update_counts()) {
qa_db_query_sub( qa_db_query_sub(
"INSERT INTO ^options (title, content) " . "INSERT INTO ^options (title, content) " .
...@@ -406,8 +453,4 @@ ...@@ -406,8 +453,4 @@
"ON DUPLICATE KEY UPDATE content = VALUES(content)" "ON DUPLICATE KEY UPDATE content = VALUES(content)"
); );
} }
} }
/*
Omit PHP closing tag to help avoid accidental output
*/
...@@ -20,23 +20,27 @@ ...@@ -20,23 +20,27 @@
More about this license: http://www.question2answer.org/license.php More about this license: http://www.question2answer.org/license.php
*/ */
if (!defined('QA_VERSION')) { // don't allow this page to be requested directly from browser if (!defined('QA_VERSION')) { // don't allow this page to be requested directly from browser
header('Location: ../'); header('Location: ../');
exit; exit;
} }
require_once QA_INCLUDE_DIR.'app/updates.php'; require_once QA_INCLUDE_DIR . 'app/updates.php';
function qa_db_post_set_selchildid($questionid, $selchildid, $lastuserid=null, $lastip=null) /**
/* * Update the selected answer in the database for $questionid to $selchildid, and optionally record that $lastuserid did it from $lastip
Update the selected answer in the database for $questionid to $selchildid, and optionally record that $lastuserid did it from $lastip * @param $questionid
*/ * @param $selchildid
{ * @param $lastuserid
* @param $lastip
*/
function qa_db_post_set_selchildid($questionid, $selchildid, $lastuserid = null, $lastip = null)
{
qa_db_query_sub( qa_db_query_sub(
"UPDATE ^posts AS x, (SELECT selchildid FROM ^posts WHERE postid=#) AS a ". "UPDATE ^posts AS x, (SELECT selchildid FROM ^posts WHERE postid=#) AS a " .
"SET x.updated=NULL, x.updatetype=NULL, x.lastuserid=NULL, x.lastip=NULL WHERE ". // if previous answer's last edit was to be selected, remove that "SET x.updated=NULL, x.updatetype=NULL, x.lastuserid=NULL, x.lastip=NULL WHERE " . // if previous answer's last edit was to be selected, remove that
"x.postid=a.selchildid AND x.updatetype=$", "x.postid=a.selchildid AND x.updatetype=$",
$questionid, QA_UPDATE_SELECTED $questionid, QA_UPDATE_SELECTED
); );
...@@ -46,313 +50,378 @@ ...@@ -46,313 +50,378 @@
$selchildid, $questionid $selchildid, $questionid
); );
if (isset($selchildid) && isset($lastuserid) && isset($lastip)) if (isset($selchildid) && isset($lastuserid) && isset($lastip)) {
qa_db_query_sub( qa_db_query_sub(
"UPDATE ^posts SET updated=NOW(), updatetype=$, lastuserid=$, lastip=$ WHERE postid=#", "UPDATE ^posts SET updated=NOW(), updatetype=$, lastuserid=$, lastip=$ WHERE postid=#",
QA_UPDATE_SELECTED, $lastuserid, @inet_pton($lastip), $selchildid QA_UPDATE_SELECTED, $lastuserid, @inet_pton($lastip), $selchildid
); );
} }
}
function qa_db_post_set_closed($questionid, $closedbyid, $lastuserid=null, $lastip=null)
/* /**
Set $questionid to be closed by post $closedbyid (null if not closed) in the database, and optionally record that * Set $questionid to be closed by post $closedbyid (null if not closed) in the database, and optionally record that
$lastuserid did it from $lastip * $lastuserid did it from $lastip
*/ * @param $questionid
{ * @param $closedbyid
* @param $lastuserid
* @param $lastip
*/
function qa_db_post_set_closed($questionid, $closedbyid, $lastuserid = null, $lastip = null)
{
if (isset($lastuserid) || isset($lastip)) { if (isset($lastuserid) || isset($lastip)) {
qa_db_query_sub( qa_db_query_sub(
"UPDATE ^posts SET closedbyid=#, updated=NOW(), updatetype=$, lastuserid=$, lastip=$ WHERE postid=#", "UPDATE ^posts SET closedbyid=#, updated=NOW(), updatetype=$, lastuserid=$, lastip=$ WHERE postid=#",
$closedbyid, QA_UPDATE_CLOSED, $lastuserid, @inet_pton($lastip), $questionid $closedbyid, QA_UPDATE_CLOSED, $lastuserid, @inet_pton($lastip), $questionid
); );
} else } else {
qa_db_query_sub( qa_db_query_sub(
'UPDATE ^posts SET closedbyid=# WHERE postid=#', 'UPDATE ^posts SET closedbyid=# WHERE postid=#',
$closedbyid, $questionid $closedbyid, $questionid
); );
} }
}
function qa_db_post_set_type($postid, $type, $lastuserid=null, $lastip=null, $updatetype=QA_UPDATE_TYPE)
/* /**
Set the type in the database of $postid to $type, and optionally record that $lastuserid did it from $lastip * Set the type in the database of $postid to $type, and optionally record that $lastuserid did it from $lastip
*/ * @param $postid
{ * @param $type
* @param $lastuserid
* @param $lastip
* @param string $updatetype
*/
function qa_db_post_set_type($postid, $type, $lastuserid = null, $lastip = null, $updatetype = QA_UPDATE_TYPE)
{
if (isset($lastuserid) || isset($lastip)) { if (isset($lastuserid) || isset($lastip)) {
qa_db_query_sub( qa_db_query_sub(
'UPDATE ^posts SET type=$, updated=NOW(), updatetype=$, lastuserid=$, lastip=$ WHERE postid=#', 'UPDATE ^posts SET type=$, updated=NOW(), updatetype=$, lastuserid=$, lastip=$ WHERE postid=#',
$type, $updatetype, $lastuserid, @inet_pton($lastip), $postid $type, $updatetype, $lastuserid, @inet_pton($lastip), $postid
); );
} else } else {
qa_db_query_sub( qa_db_query_sub(
'UPDATE ^posts SET type=$ WHERE postid=#', 'UPDATE ^posts SET type=$ WHERE postid=#',
$type, $postid $type, $postid
); );
} }
}
function qa_db_post_set_parent($postid, $parentid, $lastuserid=null, $lastip=null)
/* /**
Set the parent in the database of $postid to $parentid, and optionally record that $lastuserid did it from $lastip * Set the parent in the database of $postid to $parentid, and optionally record that $lastuserid did it from $lastip
(if at least one is specified) * (if at least one is specified)
*/ * @param $postid
{ * @param $parentid
if (isset($lastuserid) || isset($lastip)) * @param $lastuserid
* @param $lastip
*/
function qa_db_post_set_parent($postid, $parentid, $lastuserid = null, $lastip = null)
{
if (isset($lastuserid) || isset($lastip)) {
qa_db_query_sub( qa_db_query_sub(
"UPDATE ^posts SET parentid=#, updated=NOW(), updatetype=$, lastuserid=$, lastip=$ WHERE postid=#", "UPDATE ^posts SET parentid=#, updated=NOW(), updatetype=$, lastuserid=$, lastip=$ WHERE postid=#",
$parentid, QA_UPDATE_PARENT, $lastuserid, @inet_pton($lastip), $postid $parentid, QA_UPDATE_PARENT, $lastuserid, @inet_pton($lastip), $postid
); );
else } else {
qa_db_query_sub( qa_db_query_sub(
'UPDATE ^posts SET parentid=# WHERE postid=#', 'UPDATE ^posts SET parentid=# WHERE postid=#',
$parentid, $postid $parentid, $postid
); );
} }
}
function qa_db_post_set_content($postid, $title, $content, $format, $tagstring, $notify, $lastuserid=null, $lastip=null, $updatetype=QA_UPDATE_CONTENT, $name=null)
/* /**
Set the text fields in the database of $postid to $title, $content, $tagstring, $notify and $name, and record that * Set the text fields in the database of $postid to $title, $content, $tagstring, $notify and $name, and record that
$lastuserid did it from $lastip (if at least one is specified) with $updatetype. For backwards compatibility if $name * $lastuserid did it from $lastip (if at least one is specified) with $updatetype. For backwards compatibility if $name
is null then the name will not be changed. * is null then the name will not be changed.
*/ * @param $postid
{ * @param $title
if (isset($lastuserid) || isset($lastip)) // use COALESCE() for name since $name=null means it should not be modified (for backwards compatibility) * @param $content
* @param $format
* @param $tagstring
* @param $notify
* @param $lastuserid
* @param $lastip
* @param string $updatetype
* @param $name
*/
function qa_db_post_set_content($postid, $title, $content, $format, $tagstring, $notify, $lastuserid = null, $lastip = null, $updatetype = QA_UPDATE_CONTENT, $name = null)
{
if (isset($lastuserid) || isset($lastip)) {
// use COALESCE() for name since $name=null means it should not be modified (for backwards compatibility)
qa_db_query_sub( qa_db_query_sub(
'UPDATE ^posts SET title=$, content=$, format=$, tags=$, name=COALESCE($, name), notify=$, updated=NOW(), updatetype=$, lastuserid=$, lastip=$ WHERE postid=#', 'UPDATE ^posts SET title=$, content=$, format=$, tags=$, name=COALESCE($, name), notify=$, updated=NOW(), updatetype=$, lastuserid=$, lastip=$ WHERE postid=#',
$title, $content, $format, $tagstring, $name, $notify, $updatetype, $lastuserid, @inet_pton($lastip), $postid $title, $content, $format, $tagstring, $name, $notify, $updatetype, $lastuserid, @inet_pton($lastip), $postid
); );
else } else {
qa_db_query_sub( qa_db_query_sub(
'UPDATE ^posts SET title=$, content=$, format=$, tags=$, name=COALESCE($, name), notify=$ WHERE postid=#', 'UPDATE ^posts SET title=$, content=$, format=$, tags=$, name=COALESCE($, name), notify=$ WHERE postid=#',
$title, $content, $format, $tagstring, $name, $notify, $postid $title, $content, $format, $tagstring, $name, $notify, $postid
); );
} }
}
function qa_db_post_set_userid($postid, $userid) /**
/* * Set the author in the database of $postid to $userid, and set the lastuserid to $userid as well if appropriate
Set the author in the database of $postid to $userid, and set the lastuserid to $userid as well if appropriate * @param $postid
*/ * @param $userid
{ */
function qa_db_post_set_userid($postid, $userid)
{
qa_db_query_sub( qa_db_query_sub(
'UPDATE ^posts SET userid=$, lastuserid=IF(updated IS NULL, lastuserid, COALESCE(lastuserid,$)) WHERE postid=#', 'UPDATE ^posts SET userid=$, lastuserid=IF(updated IS NULL, lastuserid, COALESCE(lastuserid,$)) WHERE postid=#',
$userid, $userid, $postid $userid, $userid, $postid
); );
} }
function qa_db_post_set_category($postid, $categoryid, $lastuserid=null, $lastip=null) /**
/* * Set the (exact) category in the database of $postid to $categoryid, and optionally record that $lastuserid did it from
Set the (exact) category in the database of $postid to $categoryid, and optionally record that $lastuserid did it from * $lastip (if at least one is specified)
$lastip (if at least one is specified) * @param $postid
*/ * @param $categoryid
{ * @param $lastuserid
if (isset($lastuserid) || isset($lastip)) * @param $lastip
*/
function qa_db_post_set_category($postid, $categoryid, $lastuserid = null, $lastip = null)
{
if (isset($lastuserid) || isset($lastip)) {
qa_db_query_sub( qa_db_query_sub(
"UPDATE ^posts SET categoryid=#, updated=NOW(), updatetype=$, lastuserid=$, lastip=$ WHERE postid=#", "UPDATE ^posts SET categoryid=#, updated=NOW(), updatetype=$, lastuserid=$, lastip=$ WHERE postid=#",
$categoryid, QA_UPDATE_CATEGORY, $lastuserid, @inet_pton($lastip), $postid $categoryid, QA_UPDATE_CATEGORY, $lastuserid, @inet_pton($lastip), $postid
); );
else } else {
qa_db_query_sub( qa_db_query_sub(
'UPDATE ^posts SET categoryid=# WHERE postid=#', 'UPDATE ^posts SET categoryid=# WHERE postid=#',
$categoryid, $postid $categoryid, $postid
); );
} }
}
function qa_db_posts_set_category_path($postids, $path)
/* /**
Set the category path in the database of each of $postids to $path retrieved via qa_db_post_get_category_path() * Set the category path in the database of each of $postids to $path retrieved via qa_db_post_get_category_path()
*/ * @param $postids
{ * @param $path
if (count($postids)) */
function qa_db_posts_set_category_path($postids, $path)
{
if (count($postids)) {
// requires QA_CATEGORY_DEPTH=4
qa_db_query_sub( qa_db_query_sub(
'UPDATE ^posts SET categoryid=#, catidpath1=#, catidpath2=#, catidpath3=# WHERE postid IN (#)', 'UPDATE ^posts SET categoryid=#, catidpath1=#, catidpath2=#, catidpath3=# WHERE postid IN (#)',
$path['categoryid'], $path['catidpath1'], $path['catidpath2'], $path['catidpath3'], $postids $path['categoryid'], $path['catidpath1'], $path['catidpath2'], $path['catidpath3'], $postids
); // requires QA_CATEGORY_DEPTH=4 );
} }
}
function qa_db_post_set_created($postid, $created) /**
/* * Set the created date of $postid to $created, which is a unix timestamp. If created is null, set to now.
Set the created date of $postid to $created, which is a unix timestamp. If created is null, set to now. * @param $postid
*/ * @param $created
{ */
if (isset($created)) function qa_db_post_set_created($postid, $created)
{
if (isset($created)) {
qa_db_query_sub( qa_db_query_sub(
'UPDATE ^posts SET created=FROM_UNIXTIME(#) WHERE postid=#', 'UPDATE ^posts SET created=FROM_UNIXTIME(#) WHERE postid=#',
$created, $postid $created, $postid
); );
else } else {
qa_db_query_sub( qa_db_query_sub(
'UPDATE ^posts SET created=NOW() WHERE postid=#', 'UPDATE ^posts SET created=NOW() WHERE postid=#',
$postid $postid
); );
} }
}
function qa_db_post_set_updated($postid, $updated) /**
/* * Set the last updated date of $postid to $updated, which is a unix timestamp. If updated is null, set to now.
Set the last updated date of $postid to $updated, which is a unix timestamp. If updated is null, set to now. * @param $postid
*/ * @param $updated
{ */
if (isset($updated)) function qa_db_post_set_updated($postid, $updated)
{
if (isset($updated)) {
qa_db_query_sub( qa_db_query_sub(
'UPDATE ^posts SET updated=FROM_UNIXTIME(#) WHERE postid=#', 'UPDATE ^posts SET updated=FROM_UNIXTIME(#) WHERE postid=#',
$updated, $postid $updated, $postid
); );
else } else {
qa_db_query_sub( qa_db_query_sub(
'UPDATE ^posts SET updated=NOW() WHERE postid=#', 'UPDATE ^posts SET updated=NOW() WHERE postid=#',
$postid $postid
); );
} }
}
function qa_db_post_delete($postid) /**
/* * Deletes post $postid from the database (will also delete any votes on the post due to foreign key cascading)
Deletes post $postid from the database (will also delete any votes on the post due to foreign key cascading) * @param $postid
*/ */
{ function qa_db_post_delete($postid)
{
qa_db_query_sub( qa_db_query_sub(
'DELETE FROM ^posts WHERE postid=#', 'DELETE FROM ^posts WHERE postid=#',
$postid $postid
); );
} }
function qa_db_titlewords_get_post_wordids($postid) /**
/* * Return an array of wordids that were indexed in the database for the title of $postid
Return an array of wordids that were indexed in the database for the title of $postid * @param $postid
*/ * @return array
{ */
function qa_db_titlewords_get_post_wordids($postid)
{
return qa_db_read_all_values(qa_db_query_sub( return qa_db_read_all_values(qa_db_query_sub(
'SELECT wordid FROM ^titlewords WHERE postid=#', 'SELECT wordid FROM ^titlewords WHERE postid=#',
$postid $postid
)); ));
} }
function qa_db_titlewords_delete_post($postid) /**
/* * Remove all entries in the database index of title words for $postid
Remove all entries in the database index of title words for $postid * @param $postid
*/ */
{ function qa_db_titlewords_delete_post($postid)
{
qa_db_query_sub( qa_db_query_sub(
'DELETE FROM ^titlewords WHERE postid=#', 'DELETE FROM ^titlewords WHERE postid=#',
$postid $postid
); );
} }
function qa_db_contentwords_get_post_wordids($postid) /**
/* * Return an array of wordids that were indexed in the database for the content of $postid
Return an array of wordids that were indexed in the database for the content of $postid * @param $postid
*/ * @return array
{ */
function qa_db_contentwords_get_post_wordids($postid)
{
return qa_db_read_all_values(qa_db_query_sub( return qa_db_read_all_values(qa_db_query_sub(
'SELECT wordid FROM ^contentwords WHERE postid=#', 'SELECT wordid FROM ^contentwords WHERE postid=#',
$postid $postid
)); ));
} }
function qa_db_contentwords_delete_post($postid) /**
/* * Remove all entries in the database index of content words for $postid
Remove all entries in the database index of content words for $postid * @param $postid
*/ */
{ function qa_db_contentwords_delete_post($postid)
{
qa_db_query_sub( qa_db_query_sub(
'DELETE FROM ^contentwords WHERE postid=#', 'DELETE FROM ^contentwords WHERE postid=#',
$postid $postid
); );
} }
function qa_db_tagwords_get_post_wordids($postid) /**
/* * Return an array of wordids that were indexed in the database for the individual words in tags of $postid
Return an array of wordids that were indexed in the database for the individual words in tags of $postid * @param $postid
*/ * @return array
{ */
function qa_db_tagwords_get_post_wordids($postid)
{
return qa_db_read_all_values(qa_db_query_sub( return qa_db_read_all_values(qa_db_query_sub(
'SELECT wordid FROM ^tagwords WHERE postid=#', 'SELECT wordid FROM ^tagwords WHERE postid=#',
$postid $postid
)); ));
} }
function qa_db_tagwords_delete_post($postid) /**
/* * Remove all entries in the database index of individual words in tags of $postid
Remove all entries in the database index of individual words in tags of $postid * @param $postid
*/ */
{ function qa_db_tagwords_delete_post($postid)
{
qa_db_query_sub( qa_db_query_sub(
'DELETE FROM ^tagwords WHERE postid=#', 'DELETE FROM ^tagwords WHERE postid=#',
$postid $postid
); );
} }
function qa_db_posttags_get_post_wordids($postid) /**
/* * Return an array of wordids that were indexed in the database for the whole tags of $postid
Return an array of wordids that were indexed in the database for the whole tags of $postid * @param $postid
*/ * @return array
{ */
function qa_db_posttags_get_post_wordids($postid)
{
return qa_db_read_all_values(qa_db_query_sub( return qa_db_read_all_values(qa_db_query_sub(
'SELECT wordid FROM ^posttags WHERE postid=#', 'SELECT wordid FROM ^posttags WHERE postid=#',
$postid $postid
)); ));
} }
function qa_db_posttags_delete_post($postid) /**
/* * Remove all entries in the database index of whole tags for $postid
Remove all entries in the database index of whole tags for $postid * @param $postid
*/ */
{ function qa_db_posttags_delete_post($postid)
{
qa_db_query_sub( qa_db_query_sub(
'DELETE FROM ^posttags WHERE postid=#', 'DELETE FROM ^posttags WHERE postid=#',
$postid $postid
); );
} }
function qa_db_posts_filter_q_postids($postids) /**
/* * Return the array $postids containing only those elements which are the postid of a question in the database
Return the array $postids containing only those elements which are the postid of a question in the database * @param $postids
*/ * @return array
{ */
if (count($postids)) function qa_db_posts_filter_q_postids($postids)
{
if (count($postids)) {
return qa_db_read_all_values(qa_db_query_sub( return qa_db_read_all_values(qa_db_query_sub(
"SELECT postid FROM ^posts WHERE type='Q' AND postid IN (#)", "SELECT postid FROM ^posts WHERE type='Q' AND postid IN (#)",
$postids $postids
)); ));
else
return array();
} }
return array();
}
function qa_db_posts_get_userids($postids)
/* /**
Return an array of all the userids of authors of posts in the array $postids * Return an array of all the userids of authors of posts in the array $postids
*/ * @param $postids
{ * @return array
if (count($postids)) */
function qa_db_posts_get_userids($postids)
{
if (count($postids)) {
return qa_db_read_all_values(qa_db_query_sub( return qa_db_read_all_values(qa_db_query_sub(
"SELECT DISTINCT userid FROM ^posts WHERE postid IN (#) AND userid IS NOT NULL", "SELECT DISTINCT userid FROM ^posts WHERE postid IN (#) AND userid IS NOT NULL",
$postids $postids
)); ));
else
return array();
} }
return array();
}
function qa_db_flaggedcount_update()
/* /**
Update the cached count of the number of flagged posts in the database * Update the cached count of the number of flagged posts in the database
*/ */
{ function qa_db_flaggedcount_update()
{
if (qa_should_update_counts()) { if (qa_should_update_counts()) {
qa_db_query_sub( qa_db_query_sub(
"INSERT INTO ^options (title, content) " . "INSERT INTO ^options (title, content) " .
...@@ -361,8 +430,4 @@ ...@@ -361,8 +430,4 @@
"ON DUPLICATE KEY UPDATE content = VALUES(content)" "ON DUPLICATE KEY UPDATE content = VALUES(content)"
); );
} }
} }
/*
Omit PHP closing tag to help avoid accidental output
*/
...@@ -20,35 +20,44 @@ ...@@ -20,35 +20,44 @@
More about this license: http://www.question2answer.org/license.php More about this license: http://www.question2answer.org/license.php
*/ */
if (!defined('QA_VERSION')) { // don't allow this page to be requested directly from browser if (!defined('QA_VERSION')) { // don't allow this page to be requested directly from browser
header('Location: ../'); header('Location: ../');
exit; exit;
} }
function qa_db_calc_passcheck($password, $salt) /**
/* * Return the expected value for the passcheck column given the $password and password $salt
Return the expected value for the passcheck column given the $password and password $salt * @param $password
*/ * @param $salt
{ * @return mixed|string
*/
function qa_db_calc_passcheck($password, $salt)
{
if (qa_to_override(__FUNCTION__)) { $args=func_get_args(); return qa_call_override(__FUNCTION__, $args); } if (qa_to_override(__FUNCTION__)) { $args=func_get_args(); return qa_call_override(__FUNCTION__, $args); }
return sha1(substr($salt, 0, 8).$password.substr($salt, 8)); return sha1(substr($salt, 0, 8) . $password . substr($salt, 8));
} }
function qa_db_user_create($email, $password, $handle, $level, $ip) /**
/* * Create a new user in the database with $email, $password, $handle, privilege $level, and $ip address
Create a new user in the database with $email, $password, $handle, privilege $level, and $ip address * @param $email
*/ * @param $password
{ * @param $handle
require_once QA_INCLUDE_DIR.'util/string.php'; * @param $level
* @param $ip
* @return mixed
*/
function qa_db_user_create($email, $password, $handle, $level, $ip)
{
require_once QA_INCLUDE_DIR . 'util/string.php';
$ipbin = @inet_pton($ip); $ipbin = @inet_pton($ip);
if (QA_PASSWORD_HASH) { if (QA_PASSWORD_HASH) {
qa_db_query_sub( qa_db_query_sub(
'INSERT INTO ^users (created, createip, email, passhash, level, handle, loggedin, loginip) '. 'INSERT INTO ^users (created, createip, email, passhash, level, handle, loggedin, loginip) ' .
'VALUES (NOW(), $, $, $, #, $, NOW(), $)', 'VALUES (NOW(), $, $, $, #, $, NOW(), $)',
$ipbin, $email, isset($password) ? password_hash($password, PASSWORD_BCRYPT) : null, (int)$level, $handle, $ipbin $ipbin, $email, isset($password) ? password_hash($password, PASSWORD_BCRYPT) : null, (int)$level, $handle, $ipbin
); );
...@@ -56,7 +65,7 @@ ...@@ -56,7 +65,7 @@
$salt = isset($password) ? qa_random_alphanum(16) : null; $salt = isset($password) ? qa_random_alphanum(16) : null;
qa_db_query_sub( qa_db_query_sub(
'INSERT INTO ^users (created, createip, email, passsalt, passcheck, level, handle, loggedin, loginip) '. 'INSERT INTO ^users (created, createip, email, passsalt, passcheck, level, handle, loggedin, loginip) ' .
'VALUES (NOW(), $, $, $, UNHEX($), #, $, NOW(), $)', 'VALUES (NOW(), $, $, $, UNHEX($), #, $, NOW(), $)',
$ipbin, $email, $salt, isset($password) ? qa_db_calc_passcheck($password, $salt) : null, (int)$level, $handle, $ipbin $ipbin, $email, $salt, isset($password) ? qa_db_calc_passcheck($password, $salt) : null, (int)$level, $handle, $ipbin
); );
...@@ -64,14 +73,15 @@ ...@@ -64,14 +73,15 @@
return qa_db_last_insert_id(); return qa_db_last_insert_id();
} }
function qa_db_user_delete($userid) /**
/* * Delete user $userid from the database, along with everything they have ever done (to the extent that it's possible)
Delete user $userid from the database, along with everything they have ever done (to the extent that it's possible) * @param $userid
*/ */
{ function qa_db_user_delete($userid)
{
qa_db_query_sub('UPDATE ^posts SET lastuserid=NULL WHERE lastuserid=$', $userid); qa_db_query_sub('UPDATE ^posts SET lastuserid=NULL WHERE lastuserid=$', $userid);
qa_db_query_sub('DELETE FROM ^userpoints WHERE userid=$', $userid); qa_db_query_sub('DELETE FROM ^userpoints WHERE userid=$', $userid);
qa_db_query_sub('DELETE FROM ^blobs WHERE blobid=(SELECT avatarblobid FROM ^users WHERE userid=$)', $userid); qa_db_query_sub('DELETE FROM ^blobs WHERE blobid=(SELECT avatarblobid FROM ^users WHERE userid=$)', $userid);
...@@ -87,83 +97,99 @@ ...@@ -87,83 +97,99 @@
qa_db_query_sub('DELETE FROM ^userevents WHERE userid=$', $userid); qa_db_query_sub('DELETE FROM ^userevents WHERE userid=$', $userid);
qa_db_query_sub('DELETE FROM ^uservotes WHERE userid=$', $userid); qa_db_query_sub('DELETE FROM ^uservotes WHERE userid=$', $userid);
qa_db_query_sub('DELETE FROM ^userlimits WHERE userid=$', $userid); qa_db_query_sub('DELETE FROM ^userlimits WHERE userid=$', $userid);
} }
function qa_db_user_find_by_email($email) /**
/* * Return the ids of all users in the database which match $email (should be one or none)
Return the ids of all users in the database which match $email (should be one or none) * @param $email
*/ * @return array
{ */
function qa_db_user_find_by_email($email)
{
return qa_db_read_all_values(qa_db_query_sub( return qa_db_read_all_values(qa_db_query_sub(
'SELECT userid FROM ^users WHERE email=$', 'SELECT userid FROM ^users WHERE email=$',
$email $email
)); ));
} }
function qa_db_user_find_by_handle($handle) /**
/* * Return the ids of all users in the database which match $handle (=username), should be one or none
Return the ids of all users in the database which match $handle (=username), should be one or none * @param $handle
*/ * @return array
{ */
function qa_db_user_find_by_handle($handle)
{
return qa_db_read_all_values(qa_db_query_sub( return qa_db_read_all_values(qa_db_query_sub(
'SELECT userid FROM ^users WHERE handle=$', 'SELECT userid FROM ^users WHERE handle=$',
$handle $handle
)); ));
} }
function qa_db_user_get_userid_handles($userids) /**
/* * Return an array mapping each userid in $userids that can be found to that user's handle
Return an array mapping each userid in $userids that can be found to that user's handle * @param $userids
*/ * @return array
{ */
if (count($userids)) function qa_db_user_get_userid_handles($userids)
{
if (count($userids)) {
return qa_db_read_all_assoc(qa_db_query_sub( return qa_db_read_all_assoc(qa_db_query_sub(
'SELECT userid, handle FROM ^users WHERE userid IN (#)', 'SELECT userid, handle FROM ^users WHERE userid IN (#)',
$userids $userids
), 'userid', 'handle'); ), 'userid', 'handle');
else
return array();
} }
return array();
}
function qa_db_user_get_handle_userids($handles)
/* /**
Return an array mapping mapping each handle in $handle that can be found to that user's userid * Return an array mapping mapping each handle in $handle that can be found to that user's userid
*/ * @param $handles
{ * @return array
if (count($handles)) */
function qa_db_user_get_handle_userids($handles)
{
if (count($handles)) {
return qa_db_read_all_assoc(qa_db_query_sub( return qa_db_read_all_assoc(qa_db_query_sub(
'SELECT handle, userid FROM ^users WHERE handle IN ($)', 'SELECT handle, userid FROM ^users WHERE handle IN ($)',
$handles $handles
), 'handle', 'userid'); ), 'handle', 'userid');
else
return array();
} }
return array();
}
function qa_db_user_set($userid, $field, $value)
/* /**
Set $field of $userid to $value in the database users table * Set $field of $userid to $value in the database users table
*/ * @param $userid
{ * @param $field
* @param $value
*/
function qa_db_user_set($userid, $field, $value)
{
qa_db_query_sub( qa_db_query_sub(
'UPDATE ^users SET '.qa_db_escape_string($field).'=$ WHERE userid=$', 'UPDATE ^users SET ' . qa_db_escape_string($field) . '=$ WHERE userid=$',
$value, $userid $value, $userid
); );
} }
function qa_db_user_set_password($userid, $password) /**
/* * Set the password of $userid to $password, and reset their salt at the same time
Set the password of $userid to $password, and reset their salt at the same time * @param $userid
*/ * @param $password
{ * @return mixed
*/
function qa_db_user_set_password($userid, $password)
{
if (qa_to_override(__FUNCTION__)) { $args=func_get_args(); return qa_call_override(__FUNCTION__, $args); } if (qa_to_override(__FUNCTION__)) { $args=func_get_args(); return qa_call_override(__FUNCTION__, $args); }
require_once QA_INCLUDE_DIR.'util/string.php'; require_once QA_INCLUDE_DIR . 'util/string.php';
if (QA_PASSWORD_HASH) { if (QA_PASSWORD_HASH) {
qa_db_query_sub( qa_db_query_sub(
...@@ -178,135 +204,155 @@ ...@@ -178,135 +204,155 @@
$salt, qa_db_calc_passcheck($password, $salt), $userid $salt, qa_db_calc_passcheck($password, $salt), $userid
); );
} }
} }
function qa_db_user_set_flag($userid, $flag, $set) /**
/* * Switch on the $flag bit of the flags column for $userid if $set is true, or switch off otherwise
Switch on the $flag bit of the flags column for $userid if $set is true, or switch off otherwise * @param $userid
*/ * @param $flag
{ * @param $set
*/
function qa_db_user_set_flag($userid, $flag, $set)
{
qa_db_query_sub( qa_db_query_sub(
'UPDATE ^users SET flags=flags'.($set ? '|' : '&~').'# WHERE userid=$', 'UPDATE ^users SET flags=flags' . ($set ? '|' : '&~') . '# WHERE userid=$',
$flag, $userid $flag, $userid
); );
} }
function qa_db_user_rand_emailcode() /**
/* * Return a random string to be used for a user's emailcode column
Return a random string to be used for a user's emailcode column */
*/ function qa_db_user_rand_emailcode()
{ {
if (qa_to_override(__FUNCTION__)) { $args=func_get_args(); return qa_call_override(__FUNCTION__, $args); } if (qa_to_override(__FUNCTION__)) { $args=func_get_args(); return qa_call_override(__FUNCTION__, $args); }
require_once QA_INCLUDE_DIR.'util/string.php'; require_once QA_INCLUDE_DIR . 'util/string.php';
return qa_random_alphanum(8); return qa_random_alphanum(8);
} }
function qa_db_user_rand_sessioncode() /**
/* * Return a random string to be used for a user's sessioncode column (for browser session cookies)
Return a random string to be used for a user's sessioncode column (for browser session cookies) */
*/ function qa_db_user_rand_sessioncode()
{ {
if (qa_to_override(__FUNCTION__)) { $args=func_get_args(); return qa_call_override(__FUNCTION__, $args); } if (qa_to_override(__FUNCTION__)) { $args=func_get_args(); return qa_call_override(__FUNCTION__, $args); }
require_once QA_INCLUDE_DIR.'util/string.php'; require_once QA_INCLUDE_DIR . 'util/string.php';
return qa_random_alphanum(8); return qa_random_alphanum(8);
} }
function qa_db_user_profile_set($userid, $field, $value) /**
/* * Set a row in the database user profile table to store $value for $field for $userid
Set a row in the database user profile table to store $value for $field for $userid * @param $userid
*/ * @param $field
{ * @param $value
*/
function qa_db_user_profile_set($userid, $field, $value)
{
qa_db_query_sub( qa_db_query_sub(
'INSERT INTO ^userprofile (userid, title, content) VALUES ($, $, $) ' . 'INSERT INTO ^userprofile (userid, title, content) VALUES ($, $, $) ' .
'ON DUPLICATE KEY UPDATE content = VALUES(content)', 'ON DUPLICATE KEY UPDATE content = VALUES(content)',
$userid, $field, $value $userid, $field, $value
); );
} }
function qa_db_user_logged_in($userid, $ip) /**
/* * Note in the database that $userid just logged in from $ip address
Note in the database that $userid just logged in from $ip address * @param $userid
*/ * @param $ip
{ */
function qa_db_user_logged_in($userid, $ip)
{
qa_db_query_sub( qa_db_query_sub(
'UPDATE ^users SET loggedin=NOW(), loginip=$ WHERE userid=$', 'UPDATE ^users SET loggedin=NOW(), loginip=$ WHERE userid=$',
@inet_pton($ip), $userid @inet_pton($ip), $userid
); );
} }
function qa_db_user_written($userid, $ip) /**
/* * Note in the database that $userid just performed a write operation from $ip address
Note in the database that $userid just performed a write operation from $ip address * @param $userid
*/ * @param $ip
{ */
function qa_db_user_written($userid, $ip)
{
qa_db_query_sub( qa_db_query_sub(
'UPDATE ^users SET written=NOW(), writeip=$ WHERE userid=$', 'UPDATE ^users SET written=NOW(), writeip=$ WHERE userid=$',
@inet_pton($ip), $userid @inet_pton($ip), $userid
); );
} }
function qa_db_user_login_add($userid, $source, $identifier) /**
/* * Add an external login in the database for $source and $identifier for user $userid
Add an external login in the database for $source and $identifier for user $userid * @param $userid
*/ * @param $source
{ * @param $identifier
*/
function qa_db_user_login_add($userid, $source, $identifier)
{
qa_db_query_sub( qa_db_query_sub(
'INSERT INTO ^userlogins (userid, source, identifier, identifiermd5) '. 'INSERT INTO ^userlogins (userid, source, identifier, identifiermd5) ' .
'VALUES ($, $, $, UNHEX($))', 'VALUES ($, $, $, UNHEX($))',
$userid, $source, $identifier, md5($identifier) $userid, $source, $identifier, md5($identifier)
); );
} }
function qa_db_user_login_find($source, $identifier) /**
/* * Return some information about the user with external login $source and $identifier in the database, if a match is found
Return some information about the user with external login $source and $identifier in the database, if a match is found * @param $source
*/ * @param $identifier
{ * @return array
*/
function qa_db_user_login_find($source, $identifier)
{
return qa_db_read_all_assoc(qa_db_query_sub( return qa_db_read_all_assoc(qa_db_query_sub(
'SELECT ^userlogins.userid, handle, email FROM ^userlogins LEFT JOIN ^users ON ^userlogins.userid=^users.userid '. 'SELECT ^userlogins.userid, handle, email FROM ^userlogins LEFT JOIN ^users ON ^userlogins.userid=^users.userid ' .
'WHERE source=$ AND identifiermd5=UNHEX($) AND identifier=$', 'WHERE source=$ AND identifiermd5=UNHEX($) AND identifier=$',
$source, md5($identifier), $identifier $source, md5($identifier), $identifier
)); ));
} }
function qa_db_user_login_sync($sync) /**
/* * Lock all tables if $sync is true, otherwise unlock them. Used to synchronize creation of external login mappings.
Lock all tables if $sync is true, otherwise unlock them. Used to synchronize creation of external login mappings. * @param $sync
*/ */
{ function qa_db_user_login_sync($sync)
{
if ($sync) { // need to lock all tables since any could be used by a plugin's event module if ($sync) { // need to lock all tables since any could be used by a plugin's event module
$tables=qa_db_list_tables(); $tables = qa_db_list_tables();
$locks=array(); $locks = array();
foreach ($tables as $table) foreach ($tables as $table)
$locks[]=$table.' WRITE'; $locks[] = $table . ' WRITE';
qa_db_query_sub('LOCK TABLES '.implode(', ', $locks)); qa_db_query_sub('LOCK TABLES ' . implode(', ', $locks));
} else } else {
qa_db_query_sub('UNLOCK TABLES'); qa_db_query_sub('UNLOCK TABLES');
} }
}
function qa_db_user_levels_set($userid, $userlevels) /**
/* * Reset the full set of context-specific (currently, per category) user levels for user $userid to $userlevels, where
Reset the full set of context-specific (currently, per category) user levels for user $userid to $userlevels, where * $userlevels is an array of arrays, the inner arrays containing items 'entitytype', 'entityid' and 'level'.
$userlevels is an array of arrays, the inner arrays containing items 'entitytype', 'entityid' and 'level'. * @param $userid
*/ * @param $userlevels
{ */
function qa_db_user_levels_set($userid, $userlevels)
{
qa_db_query_sub( qa_db_query_sub(
'DELETE FROM ^userlevels WHERE userid=$', 'DELETE FROM ^userlevels WHERE userid=$',
$userid $userid
...@@ -319,27 +365,30 @@ ...@@ -319,27 +365,30 @@
$userid, $userlevel['entitytype'], $userlevel['entityid'], $userlevel['level'] $userid, $userlevel['entitytype'], $userlevel['entityid'], $userlevel['level']
); );
} }
} }
function qa_db_users_get_mailing_next($lastuserid, $count) /**
/* * Get the information required for sending a mailing to the next $count users with userids greater than $lastuserid
Get the information required for sending a mailing to the next $count users with userids greater than $lastuserid * @param $lastuserid
*/ * @param $count
{ * @return array
*/
function qa_db_users_get_mailing_next($lastuserid, $count)
{
return qa_db_read_all_assoc(qa_db_query_sub( return qa_db_read_all_assoc(qa_db_query_sub(
'SELECT userid, email, handle, emailcode, flags FROM ^users WHERE userid># ORDER BY userid LIMIT #', 'SELECT userid, email, handle, emailcode, flags FROM ^users WHERE userid># ORDER BY userid LIMIT #',
$lastuserid, $count $lastuserid, $count
)); ));
} }
function qa_db_uapprovecount_update() /**
/* * Update the cached count of the number of users who are awaiting approval after registration
Update the cached count of the number of users who are awaiting approval after registration */
*/ function qa_db_uapprovecount_update()
{ {
if ( qa_should_update_counts() && !QA_FINAL_EXTERNAL_USERS ) { if (qa_should_update_counts() && !QA_FINAL_EXTERNAL_USERS) {
qa_db_query_sub( qa_db_query_sub(
"INSERT INTO ^options (title, content) " . "INSERT INTO ^options (title, content) " .
"SELECT 'cache_uapprovecount', COUNT(*) FROM ^users " . "SELECT 'cache_uapprovecount', COUNT(*) FROM ^users " .
...@@ -348,9 +397,4 @@ ...@@ -348,9 +397,4 @@
QA_USER_LEVEL_APPROVED, QA_USER_FLAGS_USER_BLOCKED QA_USER_LEVEL_APPROVED, QA_USER_FLAGS_USER_BLOCKED
); );
} }
} }
/*
Omit PHP closing tag to help avoid accidental output
*/
...@@ -20,145 +20,159 @@ ...@@ -20,145 +20,159 @@
More about this license: http://www.question2answer.org/license.php More about this license: http://www.question2answer.org/license.php
*/ */
if (!defined('QA_VERSION')) { // don't allow this page to be requested directly from browser if (!defined('QA_VERSION')) { // don't allow this page to be requested directly from browser
header('Location: ../'); header('Location: ../');
exit; exit;
} }
function qa_db_uservote_set($postid, $userid, $vote) /**
/* * Set the vote for $userid on $postid to $vote in the database
Set the vote for $userid on $postid to $vote in the database * @param $postid
*/ * @param $userid
{ * @param $vote
$vote=max(min(($vote), 1), -1); */
function qa_db_uservote_set($postid, $userid, $vote)
{
$vote = max(min(($vote), 1), -1);
qa_db_query_sub( qa_db_query_sub(
'INSERT INTO ^uservotes (postid, userid, vote, flag) VALUES (#, #, #, 0) ON DUPLICATE KEY UPDATE vote=#', 'INSERT INTO ^uservotes (postid, userid, vote, flag) VALUES (#, #, #, 0) ON DUPLICATE KEY UPDATE vote=#',
$postid, $userid, $vote, $vote $postid, $userid, $vote, $vote
); );
} }
function qa_db_uservote_get($postid, $userid) /**
/* * Get the vote for $userid on $postid from the database (or NULL if none)
Get the vote for $userid on $postid from the database (or NULL if none) * @param $postid
*/ * @param $userid
{ * @return mixed|null
*/
function qa_db_uservote_get($postid, $userid)
{
return qa_db_read_one_value(qa_db_query_sub( return qa_db_read_one_value(qa_db_query_sub(
'SELECT vote FROM ^uservotes WHERE postid=# AND userid=#', 'SELECT vote FROM ^uservotes WHERE postid=# AND userid=#',
$postid, $userid $postid, $userid
), true); ), true);
} }
function qa_db_userflag_set($postid, $userid, $flag) /**
/* * Set the flag for $userid on $postid to $flag (true or false) in the database
Set the flag for $userid on $postid to $flag (true or false) in the database * @param $postid
*/ * @param $userid
{ * @param $flag
$flag=$flag ? 1 : 0; */
function qa_db_userflag_set($postid, $userid, $flag)
{
$flag = $flag ? 1 : 0;
qa_db_query_sub( qa_db_query_sub(
'INSERT INTO ^uservotes (postid, userid, vote, flag) VALUES (#, #, 0, #) ON DUPLICATE KEY UPDATE flag=#', 'INSERT INTO ^uservotes (postid, userid, vote, flag) VALUES (#, #, 0, #) ON DUPLICATE KEY UPDATE flag=#',
$postid, $userid, $flag, $flag $postid, $userid, $flag, $flag
); );
} }
function qa_db_userflags_clear_all($postid) /**
/* * Clear all flags for $postid in the database
Clear all flags for $postid in the database * @param $postid
*/ */
{ function qa_db_userflags_clear_all($postid)
{
qa_db_query_sub( qa_db_query_sub(
'UPDATE ^uservotes SET flag=0 WHERE postid=#', 'UPDATE ^uservotes SET flag=0 WHERE postid=#',
$postid $postid
); );
} }
function qa_db_post_recount_votes($postid) /**
/* * Recalculate the cached count of upvotes, downvotes and netvotes for $postid in the database
Recalculate the cached count of upvotes, downvotes and netvotes for $postid in the database * @param $postid
*/ */
{ function qa_db_post_recount_votes($postid)
{
if (qa_should_update_counts()) if (qa_should_update_counts())
qa_db_query_sub( qa_db_query_sub(
'UPDATE ^posts AS x, (SELECT COALESCE(SUM(GREATEST(0,vote)),0) AS upvotes, -COALESCE(SUM(LEAST(0,vote)),0) AS downvotes FROM ^uservotes WHERE postid=#) AS a SET x.upvotes=a.upvotes, x.downvotes=a.downvotes, x.netvotes=a.upvotes-a.downvotes WHERE x.postid=#', 'UPDATE ^posts AS x, (SELECT COALESCE(SUM(GREATEST(0,vote)),0) AS upvotes, -COALESCE(SUM(LEAST(0,vote)),0) AS downvotes FROM ^uservotes WHERE postid=#) AS a SET x.upvotes=a.upvotes, x.downvotes=a.downvotes, x.netvotes=a.upvotes-a.downvotes WHERE x.postid=#',
$postid, $postid $postid, $postid
); );
} }
function qa_db_post_recount_flags($postid) /**
/* * Recalculate the cached count of flags for $postid in the database
Recalculate the cached count of flags for $postid in the database * @param $postid
*/ */
{ function qa_db_post_recount_flags($postid)
{
if (qa_should_update_counts()) if (qa_should_update_counts())
qa_db_query_sub( qa_db_query_sub(
'UPDATE ^posts AS x, (SELECT COALESCE(SUM(IF(flag, 1, 0)),0) AS flagcount FROM ^uservotes WHERE postid=#) AS a SET x.flagcount=a.flagcount WHERE x.postid=#', 'UPDATE ^posts AS x, (SELECT COALESCE(SUM(IF(flag, 1, 0)),0) AS flagcount FROM ^uservotes WHERE postid=#) AS a SET x.flagcount=a.flagcount WHERE x.postid=#',
$postid, $postid $postid, $postid
); );
} }
function qa_db_uservote_post_get($postid) /**
/* * Returns all non-zero votes on post $postid from the database as an array of [userid] => [vote]
Returns all non-zero votes on post $postid from the database as an array of [userid] => [vote] * @param $postid
*/ * @return array
{ */
function qa_db_uservote_post_get($postid)
{
return qa_db_read_all_assoc(qa_db_query_sub( return qa_db_read_all_assoc(qa_db_query_sub(
'SELECT userid, vote FROM ^uservotes WHERE postid=# AND vote!=0', 'SELECT userid, vote FROM ^uservotes WHERE postid=# AND vote!=0',
$postid $postid
), 'userid', 'vote'); ), 'userid', 'vote');
} }
function qa_db_uservoteflag_user_get($userid) /**
/* * Returns all the postids from the database for posts that $userid has voted on or flagged
Returns all the postids from the database for posts that $userid has voted on or flagged * @param $userid
*/ * @return array
{ */
function qa_db_uservoteflag_user_get($userid)
{
return qa_db_read_all_values(qa_db_query_sub( return qa_db_read_all_values(qa_db_query_sub(
'SELECT postid FROM ^uservotes WHERE userid=# AND (vote!=0) OR (flag!=0)', 'SELECT postid FROM ^uservotes WHERE userid=# AND (vote!=0) OR (flag!=0)',
$userid $userid
)); ));
} }
function qa_db_uservoteflag_posts_get($postids) /**
/* * Return information about all the non-zero votes and/or flags on the posts in postids, including user handles for internal user management
Return information about all the non-zero votes and/or flags on the posts in postids, including user handles for internal user management * @param $postids
*/ * @return array
{ */
if (QA_FINAL_EXTERNAL_USERS) function qa_db_uservoteflag_posts_get($postids)
{
if (QA_FINAL_EXTERNAL_USERS) {
return qa_db_read_all_assoc(qa_db_query_sub( return qa_db_read_all_assoc(qa_db_query_sub(
'SELECT postid, userid, vote, flag FROM ^uservotes WHERE postid IN (#) AND ((vote!=0) OR (flag!=0))', 'SELECT postid, userid, vote, flag FROM ^uservotes WHERE postid IN (#) AND ((vote!=0) OR (flag!=0))',
$postids $postids
)); ));
} else {
else
return qa_db_read_all_assoc(qa_db_query_sub( return qa_db_read_all_assoc(qa_db_query_sub(
'SELECT postid, handle, vote, flag FROM ^uservotes LEFT JOIN ^users ON ^uservotes.userid=^users.userid WHERE postid IN (#) AND ((vote!=0) OR (flag!=0))', 'SELECT postid, handle, vote, flag FROM ^uservotes LEFT JOIN ^users ON ^uservotes.userid=^users.userid WHERE postid IN (#) AND ((vote!=0) OR (flag!=0))',
$postids $postids
)); ));
} }
}
/** /**
* Remove all votes assigned to a post that had been cast by the owner of the post. * Remove all votes assigned to a post that had been cast by the owner of the post.
* *
* @param int $postid The post ID from which the owner's votes will be removed. * @param int $postid The post ID from which the owner's votes will be removed.
*/ */
function qa_db_uservote_remove_own($postid) function qa_db_uservote_remove_own($postid)
{ {
qa_db_query_sub( qa_db_query_sub(
'DELETE uv FROM ^uservotes uv JOIN ^posts p ON uv.postid=p.postid AND uv.userid=p.userid WHERE uv.postid=#', $postid 'DELETE uv FROM ^uservotes uv JOIN ^posts p ON uv.postid=p.postid AND uv.userid=p.userid WHERE uv.postid=#', $postid
); );
} }
/*
Omit PHP closing tag to help avoid accidental output
*/
\ No newline at end of file
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment