1:   2:   3:   4:   5:   6:   7:   8:   9:  10:  11:  12:  13:  14:  15:  16:  17:  18:  19:  20:  21:  22:  23:  24:  25:  26:  27:  28:  29:  30:  31:  32:  33:  34:  35:  36:  37:  38:  39:  40:  41:  42:  43:  44:  45:  46:  47:  48:  49:  50:  51:  52:  53:  54:  55:  56:  57:  58:  59:  60:  61:  62:  63:  64:  65:  66:  67:  68:  69:  70:  71:  72:  73:  74:  75:  76:  77:  78:  79:  80:  81:  82:  83:  84:  85:  86:  87:  88:  89:  90:  91:  92:  93:  94:  95:  96:  97:  98:  99: 100: 101: 102: 103: 104: 105: 106: 107: 108: 109: 110: 111: 112: 113: 114: 115: 116: 117: 118: 119: 120: 121: 122: 123: 124: 125: 126: 127: 128: 129: 130: 131: 132: 133: 134: 135: 136: 137: 138: 139: 140: 141: 142: 143: 144: 145: 146: 147: 148: 149: 150: 151: 152: 153: 154: 155: 156: 157: 158: 159: 160: 161: 162: 163: 164: 165: 166: 167: 168: 169: 170: 171: 172: 173: 174: 175: 176: 177: 178: 179: 180: 181: 182: 183: 184: 185: 186: 187: 188: 189: 190: 191: 192: 193: 194: 195: 196: 197: 198: 199: 200: 201: 202: 203: 204: 205: 206: 207: 208: 209: 210: 211: 212: 213: 214: 215: 216: 217: 218: 219: 220: 221: 222: 223: 224: 225: 226: 227: 228: 229: 230: 231: 232: 233: 234: 235: 236: 237: 238: 239: 240: 241: 242: 243: 244: 245: 246: 247: 248: 249: 250: 251: 252: 253: 254: 255: 256: 257: 258: 259: 260: 261: 262: 263: 264: 265: 266: 267: 268: 269: 270: 271: 272: 273: 274: 275: 276: 277: 278: 279: 280: 281: 282: 283: 284: 285: 286: 287: 288: 289: 290: 291: 292: 293: 294: 295: 296: 297: 298: 299: 300: 301: 302: 303: 304: 305: 306: 307: 308: 309: 310: 311: 312: 313: 314: 315: 316: 317: 318: 319: 320: 321: 322: 323: 324: 325: 326: 327: 328: 329: 330: 331: 332: 333: 334: 335: 336: 337: 338: 339: 340: 341: 342: 343: 344: 345: 346: 347: 348: 349: 350: 351: 352: 353: 354: 355: 356: 357: 358: 359: 360: 361: 362: 363: 364: 365: 366: 367: 368: 369: 370: 371: 372: 373: 374: 375: 376: 377: 378: 379: 380: 381: 382: 383: 384: 385: 386: 387: 388: 389: 390: 391: 392: 393: 394: 395: 396: 397: 398: 399: 400: 401: 402: 403: 404: 405: 406: 407: 408: 409: 410: 411: 412: 413: 414: 415: 416: 417: 418: 419: 420: 421: 422: 423: 424: 425: 426: 427: 428: 429: 430: 431: 432: 433: 434: 435: 436: 437: 438: 439: 440: 441: 442: 443: 444: 445: 446: 447: 448: 449: 450: 451: 452: 453: 454: 455: 456: 457: 458: 459: 460: 461: 462: 463: 464: 465: 466: 467: 468: 469: 470: 471: 472: 473: 474: 475: 476: 477: 478: 479: 480: 481: 482: 483: 484: 485: 486: 487: 488: 489: 490: 491: 492: 493: 494: 495: 496: 497: 498: 499: 500: 501: 502: 503: 504: 505: 506: 507: 508: 509: 510: 511: 512: 513: 514: 515: 516: 517: 518: 519: 520: 521: 522: 523: 524: 525: 526: 527: 528: 529: 530: 531: 532: 533: 534: 535: 536: 537: 538: 539: 540: 541: 542: 543: 544: 545: 546: 547: 548: 549: 550: 551: 552: 553: 554: 555: 556: 557: 558: 559: 560: 561: 562: 563: 564: 565: 566: 567: 568: 569: 570: 571: 572: 573: 574: 575: 576: 577: 578: 579: 580: 581: 582: 583: 584: 585: 586: 587: 588: 589: 590: 591: 592: 593: 594: 595: 596: 597: 598: 599: 600: 601: 602: 603: 604: 605: 606: 607: 608: 609: 610: 611: 612: 613: 614: 615: 616: 617: 618: 619: 620: 621: 622: 623: 624: 625: 626: 627: 628: 629: 630: 631: 632: 633: 634: 635: 636: 637: 638: 639: 640: 641: 642: 643: 644: 645: 646: 647: 648: 649: 650: 651: 652: 653: 654: 655: 656: 657: 658: 659: 660: 661: 662: 663: 664: 665: 666: 667: 668: 669: 670: 671: 672: 673: 674: 675: 676: 677: 
<?php

/**
 * Perform CRUD actions for reported posts and moderation comments.
 *
 * Simple Machines Forum (SMF)
 *
 * @package SMF
 * @author Simple Machines http://www.simplemachines.org
 * @copyright 2019 Simple Machines and individual contributors
 * @license http://www.simplemachines.org/about/smf/license.php BSD
 *
 * @version 2.1 RC1
 */

if (!defined('SMF'))
    die('No direct access...');

/**
 * Updates a report with the given parameters. Logs each action via logAction()
 *
 * @param string $action The action to perform. Accepts "closed" and "ignore".
 * @param integer $value The new value to update.
 * @param integer|array $report_id The affected report(s).
 *
 * @return bool
 */
function updateReport($action, $value, $report_id)
{
    global $smcFunc, $user_info, $context;

    // Don't bother.
    if (empty($action) || empty($report_id))
        return false;

    // Add the "_all" thingy.
    if ($action == 'ignore')
        $action = 'ignore_all';

    // We don't need the board query for reported members
    if ($context['report_type'] == 'members')
    {
        $board_query = '';
    }
    else
    {
        $board_query = ' AND ' . $user_info['mod_cache']['bq'];
    }

    // Update the report...
    $smcFunc['db_query']('', '
        UPDATE {db_prefix}log_reported
        SET  {raw:action} = {string:value}
        ' . (is_array($report_id) ? 'WHERE id_report IN ({array_int:id_report})' : 'WHERE id_report = {int:id_report}') . '
            ' . $board_query,
        array(
            'action' => $action,
            'value' => $value,
            'id_report' => $report_id,
        )
    );

    // From now on, lets work with arrays, makes life easier.
    $report_id = (array) $report_id;

    // Set up the data for the log...
    $extra = array();

    if ($context['report_type'] == 'posts')
    {
        // Get the board, topic and message for this report
        $request = $smcFunc['db_query']('', '
            SELECT id_board, id_topic, id_msg, id_report
            FROM {db_prefix}log_reported
            WHERE id_report IN ({array_int:id_report})',
            array(
                'id_report' => $report_id,
            )
        );

        while ($row = $smcFunc['db_fetch_assoc']($request))
            $extra[$row['id_report']] = array(
                'report' => $row['id_report'],
                'board' => $row['id_board'],
                'message' => $row['id_msg'],
                'topic' => $row['id_topic'],
            );

        $smcFunc['db_free_result']($request);
    }
    else
    {
        $request = $smcFunc['db_query']('', '
            SELECT id_report, id_member, membername
            FROM {db_prefix}log_reported
            WHERE id_report IN ({array_int:id_report})',
            array(
                'id_report' => $report_id,
            )
        );

        while ($row = $smcFunc['db_fetch_assoc']($request))
            $extra[$row['id_report']] = array(
                'report' => $row['id_report'],
                'member' => $row['id_member'],
            );

        $smcFunc['db_free_result']($request);
    }

    // Back to "ignore".
    if ($action == 'ignore_all')
        $action = 'ignore';

    $log_report = $action == 'ignore' ? (!empty($value) ? 'ignore' : 'unignore') : (!empty($value) ? 'close' : 'open');

    if ($context['report_type'] == 'members')
        $log_report .= '_user';

    // Log this action.
    if (!empty($extra))
        foreach ($extra as $report)
            logAction($log_report . '_report', $report);

    // Time to update.
    updateSettings(array('last_mod_report_action' => time()));
    recountOpenReports($context['report_type']);
}

/**
 * Counts how many reports are in total. Used for creating pagination.
 *
 * @param int $closed 1 for counting closed reports, 0 for open ones.
 * @return integer How many reports.
 */
function countReports($closed = 0)
{
    global $smcFunc, $user_info, $context;

    // Skip entries with id_board = 0 if we're viewing member reports
    if ($context['report_type'] == 'members')
    {
        $and = 'lr.id_board = 0';
    }
    else
    {
        if ($user_info['mod_cache']['bq'] == '1=1' || $user_info['mod_cache']['bq'] == '0=1')
        {
            $bq = $user_info['mod_cache']['bq'];
        }
        else
        {
            $bq = 'lr.' . $user_info['mod_cache']['bq'];
        }

        $and = $bq . ' AND lr.id_board != 0';
    }

    // How many entries are we viewing?
    $request = $smcFunc['db_query']('', '
        SELECT COUNT(*)
        FROM {db_prefix}log_reported AS lr
        WHERE lr.closed = {int:view_closed}
            AND ' . $and,
        array(
            'view_closed' => (int) $closed,
        )
    );
    list ($total_reports) = $smcFunc['db_fetch_row']($request);
    $smcFunc['db_free_result']($request);

    return $total_reports;
}

/**
 * Get all possible reports the current user can see.
 *
 * @param int $closed 1 for closed reports, 0 for open ones.
 * @return array the reports data with the report ID as key.
 */
function getReports($closed = 0)
{
    global $smcFunc, $context, $user_info, $scripturl, $txt;

    // Lonely, standalone var.
    $reports = array();

    // By George, that means we are in a position to get the reports, golly good.
    if ($context['report_type'] == 'members')
    {
        $request = $smcFunc['db_query']('', '
            SELECT lr.id_report, lr.id_member,
                lr.time_started, lr.time_updated, lr.num_reports, lr.closed, lr.ignore_all,
                COALESCE(mem.real_name, lr.membername) AS user_name, COALESCE(mem.id_member, 0) AS id_user
            FROM {db_prefix}log_reported AS lr
                LEFT JOIN {db_prefix}members AS mem ON (mem.id_member = lr.id_member)
            WHERE lr.closed = {int:view_closed}
                AND lr.id_board = 0
            ORDER BY lr.time_updated DESC
            LIMIT {int:start}, {int:max}',
            array(
                'view_closed' => (int) $closed,
                'start' => $context['start'],
                'max' => 10,
            )
        );
    }
    else
    {
        $request = $smcFunc['db_query']('', '
            SELECT lr.id_report, lr.id_msg, lr.id_topic, lr.id_board, lr.id_member, lr.subject, lr.body,
                lr.time_started, lr.time_updated, lr.num_reports, lr.closed, lr.ignore_all,
                COALESCE(mem.real_name, lr.membername) AS author_name, COALESCE(mem.id_member, 0) AS id_author
            FROM {db_prefix}log_reported AS lr
                LEFT JOIN {db_prefix}members AS mem ON (mem.id_member = lr.id_member)
            WHERE lr.closed = {int:view_closed}
                AND lr.id_board != 0
                AND ' . ($user_info['mod_cache']['bq'] == '1=1' || $user_info['mod_cache']['bq'] == '0=1' ? $user_info['mod_cache']['bq'] : 'lr.' . $user_info['mod_cache']['bq']) . '
            ORDER BY lr.time_updated DESC
            LIMIT {int:start}, {int:max}',
            array(
                'view_closed' => (int) $closed,
                'start' => $context['start'],
                'max' => 10,
            )
        );
    }

    $report_ids = array();
    $report_boards_ids = array();
    $i = 0;
    while ($row = $smcFunc['db_fetch_assoc']($request))
    {
        $report_ids[] = $row['id_report'];
        $reports[$row['id_report']] = array(
            'id' => $row['id_report'],
            'report_href' => $scripturl . '?action=moderate;area=reported' . $context['report_type'] . ';sa=details;rid=' . $row['id_report'],
            'comments' => array(),
            'time_started' => timeformat($row['time_started']),
            'last_updated' => timeformat($row['time_updated']),
            'num_reports' => $row['num_reports'],
            'closed' => $row['closed'],
            'ignore' => $row['ignore_all']
        );

        if ($context['report_type'] == 'members')
        {
            $extraDetails = array(
                'user' => array(
                    'id' => $row['id_user'],
                    'name' => $row['user_name'],
                    'link' => $row['id_user'] ? '<a href="' . $scripturl . '?action=profile;u=' . $row['id_user'] . '">' . $row['user_name'] . '</a>' : $row['user_name'],
                    'href' => $scripturl . '?action=profile;u=' . $row['id_user'],
                ),
            );
        }
        else
        {
            $report_boards_ids[] = $row['id_board'];
            $extraDetails = array(
                'topic' => array(
                    'id' => $row['id_topic'],
                    'id_msg' => $row['id_msg'],
                    'id_board' => $row['id_board'],
                    'href' => $scripturl . '?topic=' . $row['id_topic'] . '.msg' . $row['id_msg'] . '#msg' . $row['id_msg'],
                ),
                'author' => array(
                    'id' => $row['id_author'],
                    'name' => $row['author_name'],
                    'link' => $row['id_author'] ? '<a href="' . $scripturl . '?action=profile;u=' . $row['id_author'] . '">' . $row['author_name'] . '</a>' : $row['author_name'],
                    'href' => $scripturl . '?action=profile;u=' . $row['id_author'],
                ),
                'subject' => $row['subject'],
                'body' => parse_bbc($row['body']),
            );
        }

        $reports[$row['id_report']] = array_merge($reports[$row['id_report']], $extraDetails);
        $i++;
    }
    $smcFunc['db_free_result']($request);

    // Get the names of boards those topics are in. Slightly faster this way.
    if (!empty($report_boards_ids))
    {
        $report_boards_ids = array_unique($report_boards_ids);
        $board_names = array();
        $request = $smcFunc['db_query']('', '
            SELECT id_board, name
            FROM {db_prefix}boards
            WHERE id_board IN ({array_int:boards})',
            array(
                'boards' => $report_boards_ids,
            )
        );

        while ($row = $smcFunc['db_fetch_assoc']($request))
            $board_names[$row['id_board']] = $row['name'];

        $smcFunc['db_free_result']($request);

        foreach ($reports as $id_report => $report)
            if (!empty($board_names[$report['topic']['id_board']]))
                $reports[$id_report]['topic']['board_name'] = $board_names[$report['topic']['id_board']];
    }

    // Now get all the people who reported it.
    if (!empty($report_ids))
    {
        $request = $smcFunc['db_query']('', '
            SELECT lrc.id_comment, lrc.id_report, lrc.time_sent, lrc.comment,
                COALESCE(mem.id_member, 0) AS id_member, COALESCE(mem.real_name, lrc.membername) AS reporter
            FROM {db_prefix}log_reported_comments AS lrc
                LEFT JOIN {db_prefix}members AS mem ON (mem.id_member = lrc.id_member)
            WHERE lrc.id_report IN ({array_int:report_list})',
            array(
                'report_list' => $report_ids,
            )
        );
        while ($row = $smcFunc['db_fetch_assoc']($request))
        {
            $reports[$row['id_report']]['comments'][] = array(
                'id' => $row['id_comment'],
                'message' => $row['comment'],
                'time' => timeformat($row['time_sent']),
                'member' => array(
                    'id' => $row['id_member'],
                    'name' => empty($row['reporter']) ? $txt['guest'] : $row['reporter'],
                    'link' => $row['id_member'] ? '<a href="' . $scripturl . '?action=profile;u=' . $row['id_member'] . '">' . $row['reporter'] . '</a>' : (empty($row['reporter']) ? $txt['guest'] : $row['reporter']),
                    'href' => $row['id_member'] ? $scripturl . '?action=profile;u=' . $row['id_member'] : '',
                ),
            );
        }
        $smcFunc['db_free_result']($request);
    }

    // Get the boards where the current user can remove any message.
    $context['report_remove_any_boards'] = $user_info['is_admin'] ? $report_boards_ids : array_intersect($report_boards_ids, boardsAllowedTo('remove_any'));
    $context['report_manage_bans'] = allowedTo('manage_bans');

    return $reports;
}

/**
 * Recount all open reports. Sets a SESSION var with the updated info.
 *
 * @param string $type the type of reports to count
 * @return int the update open report count.
 */
function recountOpenReports($type)
{
    global $user_info, $smcFunc;

    if ($type == 'members')
        $bq = '';
    else
        $bq = ' AND ' . $user_info['mod_cache']['bq'];

    $request = $smcFunc['db_query']('', '
        SELECT COUNT(*)
        FROM {db_prefix}log_reported
        WHERE closed = {int:not_closed}
            AND ignore_all = {int:not_ignored}
            AND id_board' . ($type == 'members' ? '' : '!') . '= {int:not_a_reported_post}'
            . $bq,
        array(
            'not_closed' => 0,
            'not_ignored' => 0,
            'not_a_reported_post' => 0,
        )
    );
    list ($open_reports) = $smcFunc['db_fetch_row']($request);
    $smcFunc['db_free_result']($request);

    $arr = ($type == 'members' ? 'member_reports' : 'reports');
    $_SESSION['rc'] = array_merge(!empty($_SESSION['rc']) ? $_SESSION['rc'] : array(),
        array(
            'id' => $user_info['id'],
            'time' => time(),
            $arr => $open_reports,
        ));

    return $open_reports;
}

/**
 * Gets additional information for a specific report.
 *
 * @param int $report_id The report ID to get the info from.
 * @return array|bool the report data. Boolean false if no report_id was provided.
 */
function getReportDetails($report_id)
{
    global $smcFunc, $user_info, $context;

    if (empty($report_id))
        return false;

    // We don't need all this info if we're only getting user info
    if ($context['report_type'] == 'members')
    {
        $request = $smcFunc['db_query']('', '
            SELECT lr.id_report, lr.id_member,
                lr.time_started, lr.time_updated, lr.num_reports, lr.closed, lr.ignore_all,
                COALESCE(mem.real_name, lr.membername) AS user_name, COALESCE(mem.id_member, 0) AS id_user
            FROM {db_prefix}log_reported AS lr
                LEFT JOIN {db_prefix}members AS mem ON (mem.id_member = lr.id_member)
            WHERE lr.id_report = {int:id_report}
                AND lr.id_board = 0
            LIMIT 1',
            array(
                'id_report' => $report_id,
            )
        );
    }
    else
    {
        // Get the report details, need this so we can limit access to a particular board.
        $request = $smcFunc['db_query']('', '
            SELECT lr.id_report, lr.id_msg, lr.id_topic, lr.id_board, lr.id_member, lr.subject, lr.body,
                lr.time_started, lr.time_updated, lr.num_reports, lr.closed, lr.ignore_all,
                COALESCE(mem.real_name, lr.membername) AS author_name, COALESCE(mem.id_member, 0) AS id_author
            FROM {db_prefix}log_reported AS lr
                LEFT JOIN {db_prefix}members AS mem ON (mem.id_member = lr.id_member)
            WHERE lr.id_report = {int:id_report}
                AND ' . ($user_info['mod_cache']['bq'] == '1=1' || $user_info['mod_cache']['bq'] == '0=1' ? $user_info['mod_cache']['bq'] : 'lr.' . $user_info['mod_cache']['bq']) . '
            LIMIT 1',
            array(
                'id_report' => $report_id,
            )
        );
    }

    // So did we find anything?
    if (!$smcFunc['db_num_rows']($request))
        return false;

    // Woohoo we found a report and they can see it!
    $row = $smcFunc['db_fetch_assoc']($request);
    $smcFunc['db_free_result']($request);

    return $row;
}

/**
 * Gets both report comments as well as any moderator comment.
 *
 * @param int $report_id The report ID to get the info from.
 * @return array|bool an associative array with 2 keys comments and mod_comments. Boolean false if no report_id was provided.
 */
function getReportComments($report_id)
{
    global $smcFunc, $scripturl, $user_info, $txt;

    if (empty($report_id))
        return false;

    $report = array(
        'comments' => array(),
        'mod_comments' => array()
    );

    // So what bad things do the reporters have to say about it?
    $request = $smcFunc['db_query']('', '
        SELECT lrc.id_comment, lrc.id_report, lrc.time_sent, lrc.comment, lrc.member_ip,
            COALESCE(mem.id_member, 0) AS id_member, COALESCE(mem.real_name, lrc.membername) AS reporter
        FROM {db_prefix}log_reported_comments AS lrc
            LEFT JOIN {db_prefix}members AS mem ON (mem.id_member = lrc.id_member)
        WHERE lrc.id_report = {int:id_report}',
        array(
            'id_report' => $report_id,
        )
    );

    while ($row = $smcFunc['db_fetch_assoc']($request))
    {
        $report['comments'][] = array(
            'id' => $row['id_comment'],
            'message' => strtr($row['comment'], array("\n" => '<br>')),
            'time' => timeformat($row['time_sent']),
            'member' => array(
                'id' => $row['id_member'],
                'name' => empty($row['reporter']) ? $txt['guest'] : $row['reporter'],
                'link' => $row['id_member'] ? '<a href="' . $scripturl . '?action=profile;u=' . $row['id_member'] . '">' . $row['reporter'] . '</a>' : (empty($row['reporter']) ? $txt['guest'] : $row['reporter']),
                'href' => $row['id_member'] ? $scripturl . '?action=profile;u=' . $row['id_member'] : '',
                'ip' => !empty($row['member_ip']) && allowedTo('moderate_forum') ? '<a href="' . $scripturl . '?action=trackip;searchip=' . inet_dtop($row['member_ip']) . '">' . inet_dtop($row['member_ip']) . '</a>' : '',
            ),
        );
    }
    $smcFunc['db_free_result']($request);

    // Hang about old chap, any comments from moderators on this one?
    $request = $smcFunc['db_query']('', '
        SELECT lc.id_comment, lc.id_notice, lc.log_time, lc.body,
            COALESCE(mem.id_member, 0) AS id_member, COALESCE(mem.real_name, lc.member_name) AS moderator
        FROM {db_prefix}log_comments AS lc
            LEFT JOIN {db_prefix}members AS mem ON (mem.id_member = lc.id_member)
        WHERE lc.id_notice = {int:id_report}
            AND lc.comment_type = {literal:reportc}',
        array(
            'id_report' => $report_id,
        )
    );

    while ($row = $smcFunc['db_fetch_assoc']($request))
    {
        $report['mod_comments'][] = array(
            'id' => $row['id_comment'],
            'message' => parse_bbc($row['body']),
            'time' => timeformat($row['log_time']),
            'can_edit' => allowedTo('admin_forum') || (($user_info['id'] == $row['id_member'])),
            'member' => array(
                'id' => $row['id_member'],
                'name' => $row['moderator'],
                'link' => $row['id_member'] ? '<a href="' . $scripturl . '?action=profile;u=' . $row['id_member'] . '">' . $row['moderator'] . '</a>' : $row['moderator'],
                'href' => $scripturl . '?action=profile;u=' . $row['id_member'],
            ),
        );
    }

    $smcFunc['db_free_result']($request);

    return $report;
}

/**
 * Gets specific details about a moderator comment. It also adds a permission for editing/deleting the comment,
 * by default only admins and the author of the comment can edit/delete it.
 *
 * @param int $comment_id The moderator comment ID to get the info from.
 * @return array|bool an array with the fetched data. Boolean false if no report_id was provided.
 */
function getCommentModDetails($comment_id)
{
    global $smcFunc, $user_info;

    if (empty($comment_id))
        return false;

    $request = $smcFunc['db_query']('', '
        SELECT id_comment, id_notice, log_time, body, id_member
        FROM {db_prefix}log_comments
        WHERE id_comment = {int:id_comment}
            AND comment_type = {literal:reportc}',
        array(
            'id_comment' => $comment_id,
        )
    );

    $comment = $smcFunc['db_fetch_assoc']($request);

    $smcFunc['db_free_result']($request);

    // Add the permission
    if (!empty($comment))
        $comment['can_edit'] = allowedTo('admin_forum') || (($user_info['id'] == $comment['id_member']));

    return $comment;
}

/**
 * Inserts a new moderator comment to the DB.
 *
 * @param int $report_id The report ID is used to fire a notification about the event.
 * @param array $data a formatted array of data to be inserted. Should be already properly sanitized.
 * @return bool  Boolean false if no data was provided.
 */
function saveModComment($report_id, $data)
{
    global $smcFunc, $user_info, $context;

    if (empty($data))
        return false;

    $data = array_merge(array($user_info['id'], $user_info['name'], 'reportc', ''), $data);

    $last_comment = $smcFunc['db_insert']('',
        '{db_prefix}log_comments',
        array(
            'id_member' => 'int', 'member_name' => 'string', 'comment_type' => 'string', 'recipient_name' => 'string',
            'id_notice' => 'int', 'body' => 'string', 'log_time' => 'int',
        ),
        $data,
        array('id_comment'),
        1
    );

    $report = getReportDetails($report_id);

    if ($context['report_type'] == 'members')
    {
        $prefix = 'Member';
        $data = array(
            'report_id' => $report_id,
            'user_id' => $report['id_user'],
            'user_name' => $report['user_name'],
            'sender_id' => $context['user']['id'],
            'sender_name' => $context['user']['name'],
            'comment_id' => $last_comment,
            'time' => time(),
        );
    }
    else
    {
        $prefix = 'Msg';
        $data = array(
            'report_id' => $report_id,
            'comment_id' => $last_comment,
            'msg_id' => $report['id_msg'],
            'topic_id' => $report['id_topic'],
            'board_id' => $report['id_board'],
            'sender_id' => $user_info['id'],
            'sender_name' => $user_info['name'],
            'time' => time(),
        );
    }

    // And get ready to notify people.
    if (!empty($report))
        $smcFunc['db_insert']('insert',
            '{db_prefix}background_tasks',
            array('task_file' => 'string', 'task_class' => 'string', 'task_data' => 'string', 'claimed_time' => 'int'),
            array('$sourcedir/tasks/' . $prefix . 'ReportReply-Notify.php', $prefix . 'ReportReply_Notify_Background', $smcFunc['json_encode']($data), 0),
            array('id_task')
        );
}

/**
 * Saves the new information whenever a moderator comment is edited.
 *
 * @param int $comment_id The edited moderator comment ID.
 * @param string $edited_comment The edited moderator comment text.
 * @return bool  Boolean false if no data or no comment ID was provided.
 */
function editModComment($comment_id, $edited_comment)
{
    global $smcFunc;

    if (empty($comment_id) || empty($edited_comment))
        return false;

    $smcFunc['db_query']('', '
        UPDATE {db_prefix}log_comments
        SET  body = {string:body}
        WHERE id_comment = {int:id_comment}',
        array(
            'body' => $edited_comment,
            'id_comment' => $comment_id,
        )
    );
    return true;
}

/**
 * Deletes a moderator comment from the DB.
 *
 * @param int $comment_id The moderator comment ID used to identify which report will be deleted.
 * @return bool  Boolean false if no data was provided.
 */
function deleteModComment($comment_id)
{
    global $smcFunc;

    if (empty($comment_id))
        return false;

    $smcFunc['db_query']('', '
        DELETE FROM {db_prefix}log_comments
        WHERE id_comment = {int:comment_id}',
        array(
            'comment_id' => $comment_id,
        )
    );

}

?>