streams/Code/Module/Item.php

2093 lines
80 KiB
PHP

<?php
namespace Code\Module;
/**
*
* As a GET request, this module answers to activitypub and zot6 item fetches and
* acts as a permalink for local content.
*
* Otherwise this is the POST destination for most all locally posted
* text stuff. This function handles status, wall-to-wall status,
* local comments, and remote coments that are posted on this site
* (as opposed to being delivered in a feed).
* Also processed here are posts and comments coming through the API.
* All of these become an "item" which is our basic unit of
* information.
* Posts that originate externally or do not fall into the above
* posting categories go through item_store() instead of this function.
*
*/
use Code\Lib\Libsync;
use Code\Lib\Activity;
use Code\Lib\ActivityStreams;
use Code\Lib\JcsEddsa2022;
use Code\Web\HTTPSig;
use Code\Web\Controller;
use Code\Lib\Libzot;
use Code\Lib\Share;
use Code\Lib\ThreadListener;
use Code\Lib\ServiceClass;
use Code\Lib\Config;
use Code\Lib\Channel;
use Code\Lib\IConfig;
use Code\Lib\PConfig;
use Code\Lib\Enotify;
use Code\Lib\Apps;
use Code\Access\PermissionLimits;
use Code\Access\PermissionRoles;
use Code\Access\AccessControl;
use Code\Extend\Hook;
use Code\Daemon\Run;
use App;
use URLify;
require_once('include/attach.php');
require_once('include/photos.php');
require_once('include/bbcode.php');
require_once('include/security.php');
require_once('include/conversation.php');
class Item extends Controller
{
public $return_404 = false;
public function init()
{
if (ActivityStreams::is_as_request()) {
$item_uuid = argv(1);
if (!$item_uuid) {
http_status_exit(404, 'Not found');
}
$portable_id = EMPTY_STR;
$item_normal = " and item.item_hidden = 0 and item.item_type = 0 and item.item_unpublished = 0 and item.item_delayed = 0 and item.item_blocked = 0 ";
$i = null;
// do we have the item (at all)?
// add preferential bias to item owners (item_wall = 1)
$r = q(
"select * from item where (mid = '%s' or uuid = '%s') $item_normal order by item_wall desc limit 1",
dbesc(z_root() . '/item/' . $item_uuid),
dbesc($item_uuid)
);
if (!$r) {
http_status_exit(404, 'Not found');
}
// process an authenticated fetch
$sigdata = HTTPSig::verify(EMPTY_STR);
if ($sigdata['portable_id'] && $sigdata['header_valid']) {
$portable_id = $sigdata['portable_id'];
if (!check_channelallowed($portable_id)) {
http_status_exit(403, 'Permission denied');
}
if (!check_siteallowed($sigdata['signer'])) {
http_status_exit(403, 'Permission denied');
}
observer_auth($portable_id);
$i = q(
"select id as item_id from item where mid = '%s' $item_normal and owner_xchan = '%s' limit 1 ",
dbesc($r[0]['parent_mid']),
dbesc($portable_id)
);
} elseif (Config::Get('system', 'require_authenticated_fetch', false)) {
http_status_exit(403, 'Permission denied');
}
// if we don't have a parent id belonging to the signer see if we can obtain one as a visitor that we have permission to access
// and which contains our requested uri as a child.
$sql_extra = item_permissions_sql(0);
if (!$i) {
// First find every instance of the parent_mid that we can access.
$j = q(
"select id as item_id from item where mid = '%s' $item_normal $sql_extra",
dbesc($r[0]['parent_mid'])
);
// If any of these have our request uri as a child, make that the head of the conversation we are going to return.
// Otherwise keep looking.
foreach ($j as $test) {
$candidate = q("select id as item_id from item where (mid = '%s' or uuid = '%s') and parent = %d ",
dbesc(z_root() . '/item/' . $item_uuid),
dbesc($item_uuid),
intval($test['item_id'])
);
if ($candidate) {
$i = [$test];
break;
}
}
}
$bear = Activity::token_from_request();
if ($bear) {
logger('bear: ' . $bear, LOGGER_DEBUG);
if (!$i) {
$t = q(
"select * from iconfig where cat = 'ocap' and k = 'relay' and v = '%s'",
dbesc($bear)
);
if ($t) {
$i = q(
"select id as item_id from item where uuid = '%s' and id = %d $item_normal limit 1",
dbesc($item_uuid),
intval($t[0]['iid'])
);
}
}
}
if (!$i) {
http_status_exit(403, 'Forbidden');
}
// If we get to this point we have determined we can access the original in $r (fetched much further above), so use it.
xchan_query($r, true);
$items = fetch_post_tags($r);
$chan = Channel::from_id($items[0]['uid']);
if (!$chan) {
http_status_exit(404, 'Not found');
}
if (!perm_is_allowed($chan['channel_id'], get_observer_hash(), 'view_stream')) {
http_status_exit(403, 'Forbidden');
}
$i = Activity::encode_item($items[0], true);
if (!$i) {
http_status_exit(404, 'Not found');
}
if ($portable_id && (!intval($items[0]['item_private']))) {
$c = q(
"select abook_id from abook where abook_channel = %d and abook_xchan = '%s'",
intval($items[0]['uid']),
dbesc($portable_id)
);
if (!$c) {
ThreadListener::store(z_root() . '/item/' . $item_uuid, $portable_id);
}
}
as_return_and_die($i, $chan);
}
if (Libzot::is_nomad_request()) {
$item_uuid = argv(1);
if (!$item_uuid) {
http_status_exit(404, 'Not found');
}
$portable_id = EMPTY_STR;
$item_normal = " and item.item_hidden = 0 and item.item_type = 0 and item.item_unpublished = 0 and item.item_delayed = 0 and item.item_blocked = 0 and not verb in ( 'Follow', 'Ignore' ) ";
$i = null;
// do we have the item (at all)?
$r = q(
"select * from item where (mid = '%s' or uuid = '%s') $item_normal limit 1",
dbesc(z_root() . '/item/' . $item_uuid),
dbesc($item_uuid)
);
$bear = Activity::token_from_request();
if ($bear) {
logger('bear: ' . $bear, LOGGER_DEBUG);
if (!$i) {
$t = q(
"select * from iconfig where cat = 'ocap' and k = 'relay' and v = '%s'",
dbesc($bear)
);
if ($t) {
$r = q(
"select id as item_id from item where uuid = '%s' and id = %d $item_normal limit 1",
dbesc($item_uuid),
intval($t[0]['iid'])
);
}
}
}
if (!$r) {
http_status_exit(404, 'Not found');
}
// process an authenticated fetch
$sigdata = HTTPSig::verify(($_SERVER['REQUEST_METHOD'] === 'POST') ? file_get_contents('php://input') : EMPTY_STR);
if ($sigdata['portable_id'] && $sigdata['header_valid']) {
$portable_id = $sigdata['portable_id'];
if (!check_channelallowed($portable_id)) {
http_status_exit(403, 'Permission denied');
}
if (!check_siteallowed($sigdata['signer'])) {
http_status_exit(403, 'Permission denied');
}
observer_auth($portable_id);
$i = q(
"select id as item_id from item where mid = '%s' $item_normal and owner_xchan = '%s' limit 1",
dbesc($r[0]['parent_mid']),
dbesc($portable_id)
);
}
elseif (Config::Get('system', 'require_authenticated_fetch', false)) {
http_status_exit(403, 'Permission denied');
}
// if we don't have a parent id belonging to the signer see if we can obtain one as a visitor that we have permission to access
// with a bias towards those items owned by channels on this site (item_wall = 1)
$sql_extra = item_permissions_sql(0);
if (!$i) {
$i = q(
"select id as item_id, item_private from item where mid = '%s' $item_normal $sql_extra order by item_wall desc limit 1",
dbesc($r[0]['parent_mid'])
);
}
if (!$i) {
http_status_exit(403, 'Forbidden');
}
$parents_str = ids_to_querystr($i, 'item_id');
// We won't need to check for privacy mismatches if the verified observer is also owner
$parent_item_private = isset($i[0]['item_private']) ? " and item_private = " . (int) $i[0]['item_private'] . " " : '';
$items = q(
"SELECT item.*, item.id AS item_id FROM item WHERE item.parent IN ( %s ) $parent_item_private $item_normal order by item.id asc",
dbesc($parents_str)
);
if (!$items) {
http_status_exit(404, 'Not found');
}
xchan_query($items, true);
$items = fetch_post_tags($items);
if (!$items) {
http_status_exit(404, 'Not found');
}
$chan = Channel::from_id($items[0]['uid']);
if (!$chan) {
http_status_exit(404, 'Not found');
}
if (!perm_is_allowed($chan['channel_id'], get_observer_hash(), 'view_stream')) {
http_status_exit(403, 'Forbidden');
}
$i = Activity::encode_item_collection($items, 'conversation/' . $item_uuid, 'OrderedCollection', true, z_root() . '/channel/' . $chan['channel_address'], count($items));
if ($portable_id && (!intval($items[0]['item_private']))) {
ThreadListener::store(z_root() . '/item/' . $item_uuid, $portable_id);
}
if (!$i) {
http_status_exit(404, 'Not found');
}
$x = array_merge(Activity::ap_context(), $i);
$headers = [];
$headers['Content-Type'] = 'application/x-nomad+json';
$x['proof'] = (new JcsEddsa2022)->sign($x, $chan);
$ret = json_encode($x, JSON_UNESCAPED_SLASHES);
$headers['Digest'] = HTTPSig::generate_digest_header($ret);
$headers['(request-target)'] = strtolower($_SERVER['REQUEST_METHOD']) . ' ' . $_SERVER['REQUEST_URI'];
$h = HTTPSig::create_sig($headers, $chan['channel_prvkey'], Channel::url($chan));
HTTPSig::set_headers($h);
echo $ret;
killme();
}
// if it isn't a drop command and isn't a post method and wasn't handled already,
// the default action is a browser request for a persistent uri and this should return
// the text/html page of the item.
if (argc() > 1 && argv(1) !== 'drop') {
$x = q(
"select uid, item_wall, llink, mid from item where mid = '%s' or mid = '%s' or uuid = '%s'",
dbesc(z_root() . '/item/' . argv(1)),
dbesc(z_root() . '/activity/' . argv(1)),
dbesc(argv(1))
);
if ($x) {
foreach ($x as $xv) {
if (intval($xv['item_wall'])) {
$c = Channel::from_id($xv['uid']);
if ($c) {
goaway($c['xchan_url'] . '?mid=' . gen_link_id($xv['mid']));
}
}
}
goaway($x[0]['llink']);
}
// save this state and catch it in the get() function
$this->return_404 = true;
}
}
public function post()
{
if ((!local_channel()) && (!remote_channel()) && (!isset($_REQUEST['anonname']))) {
return;
}
// drop an array of items.
if (isset($_REQUEST['dropitems'])) {
$arr_drop = explode(',', $_REQUEST['dropitems']);
drop_items($arr_drop);
$json = ['success' => 1];
echo json_encode($json);
killme();
}
$uid = local_channel();
$channel = null;
$observer = null;
$token = EMPTY_STR;
$datarray = [];
$item_starred = false;
$item_uplink = false;
$item_notshown = false;
$item_nsfw = false;
$item_relay = false;
$item_mentionsme = false;
$item_verified = false;
$item_retained = false;
$item_rss = false;
$item_deleted = false;
$item_hidden = false;
$item_delayed = false;
$item_pending_remove = false;
$item_blocked = false;
$post_tags = false;
$pub_copy = false;
// logger('args: ' . print_r($_REQUEST,true));
/**
* Is this a reply to something?
*/
$parent = ((isset($_REQUEST['parent'])) ? intval($_REQUEST['parent']) : 0);
$parent_mid = ((isset($_REQUEST['parent_mid'])) ? trim($_REQUEST['parent_mid']) : '');
$hidden_mentions = ((isset($_REQUEST['hidden_mentions'])) ? trim($_REQUEST['hidden_mentions']) : '');
/**
* Who is viewing this page and posting this thing
*/
$remote_xchan = ((isset($_REQUEST['remote_xchan'])) ? trim($_REQUEST['remote_xchan']) : false);
$remote_observer = xchan_match(['xchan_hash' => $remote_xchan]);
if (!$remote_observer) {
$remote_xchan = $remote_observer = false;
}
// This is the local channel representing who the posted item will belong to.
$profile_uid = ((isset($_REQUEST['profile_uid'])) ? intval($_REQUEST['profile_uid']) : 0);
// *If* you are logged in as the site admin you are allowed to create top-level items for the sys channel.
// This would typically be a webpage or webpage element.
// Comments and replies are excluded because further below we also check for sys channel ownership and
// will make a copy of the parent that you can interact with in your own stream
$sys = Channel::get_system();
if ($sys && $profile_uid && ($sys['channel_id'] == $profile_uid) && is_site_admin() && !$parent) {
$uid = intval($sys['channel_id']);
$channel = $sys;
$observer = $sys;
}
Hook::call('post_local_start', $_REQUEST);
// logger('postvars ' . print_r($_REQUEST,true), LOGGER_DATA);
$api_source = ((isset($_REQUEST['api_source']) && $_REQUEST['api_source']) ? true : false);
$nocomment = 0;
if (isset($_REQUEST['comments_enabled'])) {
$nocomment = 1 - intval($_REQUEST['comments_enabled']);
}
// this is in days, convert to absolute time
$channel_comments_closed = get_pconfig($profile_uid, 'system', 'close_comments');
if (intval($channel_comments_closed)) {
$channel_comments_closed = datetime_convert(date_Default_timezone_get(), 'UTC', 'now + ' . intval($channel_comments_closed) . ' days');
} else {
$channel_comments_closed = NULL_DATE;
}
$comments_closed = ((isset($_REQUEST['comments_closed']) && $_REQUEST['comments_closed']) ? datetime_convert(date_default_timezone_get(), 'UTC', $_REQUEST['comments_closed']) : $channel_comments_closed);
$is_poll = ((trim($_REQUEST['poll_answers'][0]) != '' && trim($_REQUEST['poll_answers'][1]) != '') ? true : false);
// 'origin' (if non-zero) indicates that this network is where the message originated,
// for the purpose of relaying comments to other conversation members.
// If using the API from a device (leaf node) you must set origin to 1 (default) or leave unset.
// If the API is used from another network with its own distribution
// and deliveries, you may wish to set origin to 0 or false and allow the other
// network to relay comments.
// If you are unsure, it is prudent (and important) to leave it unset.
$origin = (($api_source && array_key_exists('origin', $_REQUEST)) ? intval($_REQUEST['origin']) : 1);
// To represent message-ids on other networks - this will create an iconfig record
$namespace = (($api_source && array_key_exists('namespace', $_REQUEST)) ? strip_tags($_REQUEST['namespace']) : '');
$remote_id = (($api_source && array_key_exists('remote_id', $_REQUEST)) ? strip_tags($_REQUEST['remote_id']) : '');
$owner_hash = null;
$message_id = ((x($_REQUEST, 'message_id') && $api_source) ? strip_tags($_REQUEST['message_id']) : '');
$created = ((x($_REQUEST, 'created')) ? datetime_convert(date_default_timezone_get(), 'UTC', $_REQUEST['created']) : datetime_convert());
// Because somebody will probably try this and create a mess
if ($created <= NULL_DATE) {
$created = datetime_convert();
}
$post_id = ((x($_REQUEST, 'post_id')) ? intval($_REQUEST['post_id']) : 0);
$app = ((x($_REQUEST, 'source')) ? strip_tags($_REQUEST['source']) : '');
$return_path = ((x($_REQUEST, 'return')) ? $_REQUEST['return'] : '');
$preview = ((x($_REQUEST, 'preview')) ? intval($_REQUEST['preview']) : 0);
$categories = ((x($_REQUEST, 'category')) ? escape_tags($_REQUEST['category']) : '');
$webpage = ((x($_REQUEST, 'webpage')) ? intval($_REQUEST['webpage']) : 0);
$item_obscured = ((x($_REQUEST, 'obscured')) ? intval($_REQUEST['obscured']) : 0);
$pagetitle = ((x($_REQUEST, 'pagetitle')) ? escape_tags(urlencode($_REQUEST['pagetitle'])) : '');
$layout_mid = ((x($_REQUEST, 'layout_mid')) ? escape_tags($_REQUEST['layout_mid']) : '');
$plink = ((x($_REQUEST, 'permalink')) ? escape_tags($_REQUEST['permalink']) : '');
$obj_type = ((x($_REQUEST, 'obj_type')) ? escape_tags($_REQUEST['obj_type']) : ACTIVITY_OBJ_NOTE);
$checkin = ((x($_REQUEST, 'checkin')) ? 1 : 0);
$checkout = ((x($_REQUEST, 'checkout')) ? 1 : 0);
$item_unpublished = ((isset($_REQUEST['draft'])) ? intval($_REQUEST['draft']) : 0);
// allow API to bulk load a bunch of imported items without sending out a bunch of posts.
$nopush = ((x($_REQUEST, 'nopush')) ? intval($_REQUEST['nopush']) : 0);
/*
* Check service class limits
*/
if ($uid && !(x($_REQUEST, 'parent')) && !(x($_REQUEST, 'post_id'))) {
$ret = $this->item_check_service_class($uid, (($_REQUEST['webpage'] == ITEM_TYPE_WEBPAGE) ? true : false));
if (!$ret['success']) {
notice(t($ret['message']) . EOL);
if ($api_source) {
return (['success' => false, 'message' => 'service class exception']);
}
if (x($_REQUEST, 'return')) {
goaway(z_root() . "/" . $return_path);
}
killme();
}
}
if ($pagetitle) {
$pagetitle = strtolower(URLify::transliterate($pagetitle));
}
$item_flags = $item_restrict = 0;
$expires = NULL_DATE;
$parent_item = null;
$parent_contact = null;
$thr_parent = '';
$parid = 0;
$r = false;
// If this is a comment, find the parent and preset some stuff
if ($parent || $parent_mid) {
if (!x($_REQUEST, 'type')) {
$_REQUEST['type'] = 'net-comment';
}
if ($obj_type == ACTIVITY_OBJ_NOTE) {
$obj_type = ACTIVITY_OBJ_COMMENT;
}
// fetch the parent item
if ($parent) {
$r = q(
"SELECT * FROM item WHERE id = %d LIMIT 1",
intval($parent)
);
} elseif ($parent_mid && $uid) {
// This is coming from an API source, and we are logged in
$r = q(
"SELECT * FROM item WHERE mid = '%s' AND uid = %d LIMIT 1",
dbesc($parent_mid),
intval($uid)
);
}
// if this isn't the real parent of the conversation, find it
if ($r) {
$parid = $r[0]['parent'];
$parent_mid = $r[0]['mid'];
if ($r[0]['id'] != $r[0]['parent']) {
$r = q(
"SELECT * FROM item WHERE id = parent AND parent = %d LIMIT 1",
intval($parid)
);
}
// if interacting with a pubstream item (owned by the sys channel),
// create a copy of the parent in your stream
// $r may have changed. Check it again before trying to use it.
if ($r && local_channel() && (!Channel::is_system(local_channel()))) {
$old_id = $r[0]['id'];
$r = [copy_of_pubitem(App::get_channel(), $r[0]['mid'])];
if ($r[0]['id'] !== $old_id) {
// keep track that a copy was made to display a special status notice that is unique to this condition
$pub_copy = true;
}
}
}
if (!$r) {
notice(t('Unable to locate original post.') . EOL);
if ($api_source) {
return (['success' => false, 'message' => 'invalid post id']);
}
if (x($_REQUEST, 'return')) {
goaway(z_root() . "/" . $return_path);
}
killme();
}
xchan_query($r, true);
$parent_item = $r[0];
$parent = $r[0]['id'];
// multi-level threading - preserve the info but re-parent to our single level threading
$thr_parent = $parent_mid;
}
if ($parent_item && isset($parent_item['replyto']) && $parent_item['replyto']) {
$replyto = unserialise($parent_item['replyto']);
}
$moderated = false;
if (!$observer) {
$observer = App::get_observer();
if (!$observer) {
// perhaps we're allowing moderated comments from anonymous viewers
$observer = Channel::anon_identity_init($_REQUEST);
if ($observer) {
$moderated = true;
$remote_xchan = $remote_observer = $observer;
}
}
}
if (!$observer) {
notice(t('Permission denied.') . EOL);
if ($api_source) {
return (['success' => false, 'message' => 'permission denied']);
}
if (x($_REQUEST, 'return')) {
goaway(z_root() . "/" . $return_path);
}
killme();
}
if ($parent) {
logger('mod_item: item_post parent=' . $parent);
$can_comment = false;
$can_comment = can_comment_on_post($observer['xchan_hash'], $parent_item);
if (!$can_comment) {
if ((array_key_exists('owner', $parent_item)) && intval($parent_item['owner']['abook_self']) == 1) {
$can_comment = perm_is_allowed($profile_uid, $observer['xchan_hash'], 'post_comments');
}
}
if (!$can_comment) {
notice(t('Permission denied.') . EOL);
if ($api_source) {
return (['success' => false, 'message' => 'permission denied']);
}
if (x($_REQUEST, 'return')) {
goaway(z_root() . "/" . $return_path);
}
killme();
}
} else {
// fixme - $webpage could also be a wiki page or article and require a different permission to be checked.
if (!perm_is_allowed($profile_uid, $observer['xchan_hash'], ($webpage) ? 'write_pages' : 'post_wall')) {
notice(t('Permission denied.') . EOL);
if ($api_source) {
return (['success' => false, 'message' => 'permission denied']);
}
if (x($_REQUEST, 'return')) {
goaway(z_root() . "/" . $return_path);
}
killme();
}
}
// check if this author is being moderated through the 'moderated' (negative) permission
// when posting wall-to-wall
if ($moderated === false && intval($uid) !== intval($profile_uid)) {
$moderated = perm_is_allowed($profile_uid, $observer['xchan_hash'], 'moderated');
}
// If this is a comment, check the moderated permission of the parent; who may be on another site
$remote_moderated = (($parent) ? their_perms_contains($profile_uid, $parent_item['owner_xchan'], 'moderated') : false);
if ($remote_moderated) {
notice(t('Comment may be moderated.') . EOL);
}
// is this an edited post?
$orig_post = null;
if ($namespace && $remote_id) {
// It wasn't an internally generated post - see if we've got an item matching this remote service id
$i = q(
"select iid from iconfig where cat = 'system' and k = '%s' and v = '%s' limit 1",
dbesc($namespace),
dbesc($remote_id)
);
if ($i) {
$post_id = $i[0]['iid'];
}
}
$iconfig = null;
if ($post_id) {
$i = q(
"SELECT * FROM item WHERE uid = %d AND id = %d LIMIT 1",
intval($profile_uid),
intval($post_id)
);
if (!count($i)) {
killme();
}
$orig_post = $i[0];
$iconfig = q(
"select * from iconfig where iid = %d",
intval($post_id)
);
}
if (!$channel) {
if ($uid && $uid == $profile_uid) {
$channel = App::get_channel();
} else {
// posting as yourself but not necessarily to a channel you control
$r = q(
"select * from channel left join account on channel_account_id = account_id where channel_id = %d LIMIT 1",
intval($profile_uid)
);
if ($r) {
$channel = $r[0];
}
}
}
if (!$channel) {
logger("mod_item: no channel.");
if ($api_source) {
return (['success' => false, 'message' => 'no channel']);
}
if (x($_REQUEST, 'return')) {
goaway(z_root() . "/" . $return_path);
}
killme();
}
$owner_xchan = null;
$r = q(
"select * from xchan where xchan_hash = '%s' limit 1",
dbesc($channel['channel_hash'])
);
if ($r && count($r)) {
$owner_xchan = array_shift($r);
} else {
logger("mod_item: no owner.");
if ($api_source) {
return (['success' => false, 'message' => 'no owner']);
}
if (x($_REQUEST, 'return')) {
goaway(z_root() . "/" . $return_path);
}
killme();
}
$walltowall = false;
$walltowall_comment = false;
if ($remote_xchan && !$moderated) {
$observer = $remote_observer;
}
if ($observer) {
logger('mod_item: post accepted from ' . $observer['xchan_name'] . ' for ' . $owner_xchan['xchan_name'], LOGGER_DEBUG);
// wall-to-wall detection.
// For top-level posts, if the author and owner are different it's a wall-to-wall
// For comments, We need to additionally look at the parent and see if it's a wall post that originated locally.
if ($observer['xchan_name'] != $owner_xchan['xchan_name']) {
if (($parent_item) && ($parent_item['item_wall'] && $parent_item['item_origin'])) {
$walltowall_comment = true;
$walltowall = true;
}
if (!$parent) {
$walltowall = true;
}
}
}
$acl = new AccessControl($channel);
$view_policy = PermissionLimits::Get($channel['channel_id'], 'view_stream');
$comment_policy = ((isset($_REQUEST['comments_from']) && intval($_REQUEST['comments_from'])) ? intval($_REQUEST['comments_from']) : PermissionLimits::Get($channel['channel_id'], 'post_comments'));
$public_policy = ((x($_REQUEST, 'public_policy')) ? escape_tags($_REQUEST['public_policy']) : map_scope($view_policy, true));
if ($webpage) {
$public_policy = '';
}
if ($public_policy) {
$private = 1;
}
if ($orig_post) {
$private = 0;
// webpages and unpublished drafts are allowed to change ACLs after the fact. Normal conversation items aren't.
if ($webpage || intval($orig_post['item_unpublished'])) {
$acl->set_from_array($_REQUEST);
// However, draft comments should inherit the original item_private since the ACL is empty.
if ($orig_post['item_private'] && $orig_post['mid'] !== $orig_post['parent_mid']) {
$private = $orig_post['item_private'];
}
} else {
$acl->set($orig_post);
$public_policy = $orig_post['public_policy'];
$private = $orig_post['item_private'];
}
if ($public_policy || $acl->is_private()) {
$private = (($private) ? $private : 1);
}
$location = $orig_post['location'];
$lat = $orig_post['lat'];
$lon = $orig_post['lon'];
$verb = $orig_post['verb'];
$app = $orig_post['app'];
$title = escape_tags(trim($_REQUEST['title']));
$summary = trim($_REQUEST['summary']);
$body = trim($_REQUEST['body']);
$item_flags = $orig_post['item_flags'];
$item_origin = $orig_post['item_origin'];
$item_unseen = $orig_post['item_unseen'];
$item_starred = $orig_post['item_starred'];
$item_uplink = $orig_post['item_uplink'];
$item_wall = $orig_post['item_wall'];
$item_thread_top = $orig_post['item_thread_top'];
$item_notshown = $orig_post['item_notshown'];
$item_nsfw = $orig_post['item_nsfw'];
$item_relay = $orig_post['item_relay'];
$item_mentionsme = $orig_post['item_mentionsme'];
$item_nocomment = $orig_post['item_nocomment'];
$item_obscured = $orig_post['item_obscured'];
$item_verified = $orig_post['item_verified'];
$item_retained = $orig_post['item_retained'];
$item_rss = $orig_post['item_rss'];
$item_deleted = $orig_post['item_deleted'];
$item_type = $orig_post['item_type'];
$item_hidden = $orig_post['item_hidden'];
$item_delayed = $orig_post['item_delayed'];
$item_pending_remove = $orig_post['item_pending_remove'];
$item_blocked = $orig_post['item_blocked'];
$postopts = $orig_post['postopts'];
$created = ((intval($orig_post['item_unpublished'])) ? $created : $orig_post['created']);
$expires = ((intval($orig_post['item_unpublished'])) ? NULL_DATE : $orig_post['expires']);
$mid = $orig_post['mid'];
$thr_parent = $orig_post['thr_parent'];
$parent_mid = $orig_post['parent_mid'];
$plink = $orig_post['plink'];
} else {
if (!$walltowall) {
if (
(array_key_exists('contact_allow', $_REQUEST))
|| (array_key_exists('group_allow', $_REQUEST))
|| (array_key_exists('contact_deny', $_REQUEST))
|| (array_key_exists('group_deny', $_REQUEST))
) {
$acl->set_from_array($_REQUEST);
} elseif (!$api_source) {
// if no ACL has been defined and we aren't using the API, the form
// didn't send us any parameters. This means there's no ACL or it has
// been reset to the default audience.
// If $api_source is set and there are no ACL parameters, we default
// to the channel permissions which were set in the ACL contructor.
$acl->set(['allow_cid' => '', 'allow_gid' => '', 'deny_cid' => '', 'deny_gid' => '']);
}
}
$location = ((isset($_REQUEST['location'])) ? notags(trim($_REQUEST['location'])) : EMPTY_STR);
$lat = ((isset($_REQUEST['lat'])) ? floatval($_REQUEST['lat']) : 0.0);
$lon = ((isset($_REQUEST['lon'])) ? floatval($_REQUEST['lon']) : 0.0);
$verb = ((isset($_REQUEST['verb'])) ? notags(trim($_REQUEST['verb'])) : EMPTY_STR);
$title = ((isset($_REQUEST['title'])) ? escape_tags(trim($_REQUEST['title'])) : EMPTY_STR);
$summary = ((isset($_REQUEST['summary'])) ? trim($_REQUEST['summary']) : EMPTY_STR);
$body = ((isset($_REQUEST['body'])) ? trim($_REQUEST['body']) : EMPTY_STR);
$body .= ((isset($_REQUEST['attachment'])) ? trim($_REQUEST['attachment']) : EMPTY_STR);
$postopts = '';
$haslocation = $lat || $lon;
$allow_empty = ((($checkin || $checkout) && $haslocation) || $_REQUEST['allow_empty']);
$private = ((isset($private) && $private) ? $private : intval($acl->is_private() || ($public_policy)));
// If this is a comment, set the permissions from the parent.
if ($parent_item) {
$private = 0;
$acl->set($parent_item);
$private = ((intval($parent_item['item_private']) ? $parent_item['item_private'] : $acl->is_private()));
$public_policy = $parent_item['public_policy'];
$owner_hash = $parent_item['owner_xchan'];
$webpage = $parent_item['item_type'];
$comment_policy = $parent_item['comment_policy'];
$item_nocomment = $parent_item['item_nocomment'];
$comments_closed = $parent_item['comments_closed'];
}
if ((!$allow_empty) && (!strlen($body))) {
if ($preview) {
killme();
}
info(t('Empty post discarded.') . EOL);
if ($api_source) {
return (['success' => false, 'message' => 'no content']);
}
if (x($_REQUEST, 'return')) {
goaway(z_root() . "/" . $return_path);
}
killme();
}
}
if (Apps::system_app_installed($profile_uid, 'Expire Posts')) {
if (x($_REQUEST, 'expire')) {
$expires = datetime_convert(date_default_timezone_get(), 'UTC', $_REQUEST['expire']);
if ($expires <= datetime_convert()) {
$expires = NULL_DATE;
}
}
}
$mimetype = notags(trim($_REQUEST['mimetype']));
if (!$mimetype) {
$mimetype = 'text/x-multicode';
}
if ($preview) {
$summary = z_input_filter($summary, $mimetype);
$body = z_input_filter($body, $mimetype);
}
$arr = ['profile_uid' => $profile_uid, 'summary' => $summary, 'content' => $body, 'mimetype' => $mimetype];
Hook::call('post_content', $arr);
$summary = $arr['summary'];
$body = $arr['content'];
$mimetype = $arr['mimetype'];
$gacl = $acl->get();
$str_contact_allow = $gacl['allow_cid'];
$str_group_allow = $gacl['allow_gid'];
$str_contact_deny = $gacl['deny_cid'];
$str_group_deny = $gacl['deny_gid'];
$groupww = false;
// if this is a wall-to-wall post to a group, turn it into a direct message
$role = get_pconfig($profile_uid, 'system', 'permissions_role');
$rolesettings = PermissionRoles::role_perms($role);
$channel_type = isset($rolesettings['channel_type']) ? $rolesettings['channel_type'] : 'normal';
$is_group = (($channel_type === 'group') ? true : false);
if (($is_group) && ($walltowall) && (!$walltowall_comment)) {
$groupww = true;
$str_contact_allow = $owner_xchan['xchan_hash'];
$str_group_allow = '';
}
if (!strlen($verb)) {
$verb = ACTIVITY_POST;
}
if ($checkin) {
$verb = 'Arrive';
}
if ($checkout) {
$verb = 'Leave';
}
if (in_array($mimetype, [ 'text/bbcode', 'text/x-multicode' ])) {
// BBCODE alert: the following functions assume bbcode input
// and will require alternatives for alternative content-types (text/html, text/markdown, text/plain, etc.)
// we may need virtual or template classes to implement the possible alternatives
if (str_contains($body, '[/summary]')) {
$match = '';
$cnt = preg_match("/\[summary](.*?)\[\/summary]/ism", $body, $match);
if ($cnt) {
$summary .= $match[1];
}
$body_content = preg_replace("/^(.*?)\[summary](.*?)\[\/summary]/ism", '', $body);
$body = trim($body_content);
}
$summary = wrap_code($summary);
$summary = cleanup_bbcode($summary);
$summary_tags = linkify_tags($summary, ($uid) ? $uid : $profile_uid);
$summary = unwrap_code($summary);
$body = wrap_code($body);
$body = cleanup_bbcode($body);
$body_tags = linkify_tags($body, ($uid) ? $uid : $profile_uid);
$body = unwrap_code($body);
$comment_tags = linkify_tags($hidden_mentions, ($uid) ? $uid : $profile_uid);
foreach ([$summary_tags, $body_tags, $comment_tags] as $results) {
if ($results) {
// Set permissions based on tag replacements
set_linkified_perms($results, $str_contact_allow, $str_group_allow, $profile_uid, $parent_item, $private);
if (!isset($post_tags)) {
$post_tags = [];
}
foreach ($results as $result) {
$success = $result['success'];
if ($success['replaced']) {
// suppress duplicate mentions/tags
$already_tagged = false;
foreach ($post_tags as $pt) {
if ($pt['term'] === $success['term'] && $pt['url'] === $success['url'] && intval($pt['ttype']) === intval($success['termtype'])) {
$already_tagged = true;
break;
}
}
if ($already_tagged) {
continue;
}
$post_tags[] = [
'uid' => $profile_uid,
'ttype' => $success['termtype'],
'otype' => TERM_OBJ_POST,
'term' => $success['term'],
'url' => $success['url']
];
// support #collection syntax to post to a collection
// this is accomplished by adding a pcategory tag for each collection target
// this is checked inside tag_deliver() to create a second delivery chain
if ($success['termtype'] === TERM_HASHTAG) {
$r = q(
"select xchan_url from channel left join xchan on xchan_hash = channel_hash where channel_address = '%s' and channel_parent = '%s' and channel_removed = 0",
dbesc($success['term']),
dbesc(get_observer_hash())
);
if ($r) {
$post_tags[] = [
'uid' => $profile_uid,
'ttype' => TERM_PCATEGORY,
'otype' => TERM_OBJ_POST,
'term' => $success['term'] . '@' . App::get_hostname(),
'url' => $r[0]['xchan_url']
];
}
}
}
}
}
}
/**
* process collections selected manually
*/
if (array_key_exists('collections', $_REQUEST) && is_array($_REQUEST['collections']) && count($_REQUEST['collections'])) {
foreach ($_REQUEST['collections'] as $clct) {
$r = q(
"select xchan_url, xchan_hash from xchan left join hubloc on hubloc_hash = xchan_hash where hubloc_addr = '%s' and hubloc_deleted = 0 order by hubloc_id desc limit 1",
dbesc($clct)
);
if ($r) {
if (!isset($post_tags)) {
$post_tags = [];
}
$post_tags[] = [
'uid' => $profile_uid,
'ttype' => TERM_PCATEGORY,
'otype' => TERM_OBJ_POST,
'term' => $clct,
'url' => $r[0]['xchan_url']
];
}
}
}
if (in_array(substr_count($str_contact_allow,'<'), [ 1, 2 ]) && (!$str_group_allow)) {
// direct message - private between individual channels but not groups
$private = 2;
}
if ($private) {
// for edited posts, re-use any existing OCAP token (if found).
// Otherwise generate a new one.
if ($iconfig) {
foreach ($iconfig as $cfg) {
if ($cfg['cat'] === 'ocap' && $cfg['k'] === 'relay') {
$token = $cfg['v'];
}
}
}
if (!$token) {
$token = new_token();
}
}
/**
*
* When a photo was uploaded into the message using the (profile wall) ajax
* uploader, The permissions are initially set to disallow anybody but the
* owner from seeing it. This is because the permissions may not yet have been
* set for the post. If it's private, the photo permissions should be set
* appropriately. But we didn't know the final permissions on the post until
* now. So now we'll look for links of uploaded photos and attachments that are in the
* post and set them to the same permissions as the post itself.
*
* If the post was end-to-end encrypted we can't find images and attachments in the body,
* use our media_str input instead which only contains these elements - but only do this
* when encrypted content exists because the photo/attachment may have been removed from
* the post and we should keep it private. If it's encrypted we have no way of knowing
* so we'll set the permissions regardless and realise that the media may not be
* referenced in the post.
*
*/
if (!$preview) {
fix_attached_permissions($profile_uid, ((strpos($body, '[/crypt]')) ? $_POST['media_str'] : $body), $str_contact_allow, $str_group_allow, $str_contact_deny, $str_group_deny, $token);
}
$attachments = '';
$match = false;
if (preg_match_all('/(\[attachment](.*?)\[\/attachment])/', $body, $match)) {
$attachments = [];
$i = 0;
foreach ($match[2] as $mtch) {
$attach_link = '';
$hash = substr($mtch, 0, strpos($mtch, ','));
$rev = intval(substr($mtch, strpos($mtch, ',')));
if (str_starts_with($mtch, 'https://')) {
$attachments[] = [ 'href' => $mtch, 'type' => 'application/activity+json', 'title' => $mtch ];
}
else {
$r = attach_by_hash_nodata($hash, $observer['xchan_hash'], $rev);
if ($r['success']) {
$ch = Channel::from_id($r['data']['uid']);
$href = z_root() . (($ch) ? '/cloud/' . $ch['channel_address'] . '/' . $r['data']['display_path'] : '/attach/' . $r['data']['hash']);
if ($r['data']['is_photo']) {
$href = z_root() . '/photo/' . $r['data']['hash'] . '-1.' . photoExtensionFromType($r['data']['filetype']);
if ($token) {
$href .= '?token=' . $token;
}
}
$attachments[] = [
'href' => $href,
'length' => $r['data']['filesize'],
'type' => $r['data']['filetype'],
'title' => urlencode($r['data']['filename']),
'revision' => $r['data']['revision']
];
}
}
$body = str_replace($match[1][$i], $attach_link, $body);
$i++;
}
}
if (preg_match_all('/(\[share=(.*?)](.*?)\[\/share])/', $body, $match)) {
// process share by id
$i = 0;
foreach ($match[2] as $mtch) {
$reshare = new Share($mtch);
$body = str_replace($match[1][$i], $reshare->bbcode(), $body);
if (! is_array($attachments)) {
$attachments = [];
}
$attachments = array_merge($attachments,$reshare->get_attach());
if (! is_array($post_tags)) {
$post_tags = [];
}
$post_tags = array_merge($post_tags,$reshare->get_tags());
$i++;
}
}
}
$hook_args = ['location' => $location, 'latitude' => $lat, 'longitude' => $lon];
Hook::call('post_location', $hook_args);
$location = $hook_args['location'];
$lat = $hook_args['latitude'];
$lon = $hook_args['longitude'];
if (in_array($verb, ['Arrive', 'Leave'])) {
$body = preg_replace('/\[map=(.*?)\]/','', $body);
$body = preg_replace('/\[map\](.*?)\[\/map\]/','', $body);
$datarray['tgt_type'] = 'Place';
$datarray['target'] = [
'type' => 'Place',
'name' => isset($location) ? $location : '',
'latitude' => $lat,
'longitude' => $lon,
];
if ($lat || $lon) {
$body .= "\n\n" . '[map=' . $lat . ',' . $lon . ']' . "\n";
}
elseif ($location) {
$body .= "\n\n" . '[map]' . $location . '[/map]' . "\n";
}
}
// BBCODE end alert
$netgroup = false;
$found_group_mention = false;
$netgroups = get_forum_channels($profile_uid, 1);
if ($post_tags) {
foreach ($post_tags as $tag) {
foreach ($netgroups as $netgroup) {
if (isset($tag['url']) && $tag['url'] === $netgroup['xchan_url']) {
$found_group_mention = true;
break;
}
}
if ($found_group_mention) {
break;
}
}
}
// if the acl contains a single contact and it's a group, add a mention. This is for compatibility
// with other groups implementations which require a mention to trigger group delivery.
if (($str_contact_allow) && (!$str_group_allow) && (!$str_contact_deny) && (!$str_group_deny)) {
$cida = expand_acl($str_contact_allow);
if (count($cida) === 1) {
$netgroups = get_forum_channels($profile_uid, 1);
if ($netgroups) {
foreach ($netgroups as $ng) {
if ($ng['xchan_hash'] == $cida[0]) {
if (!is_array($post_tags)) {
$post_tags = [];
}
$post_tags[] = [
'uid' => $profile_uid,
'ttype' => TERM_MENTION,
'otype' => TERM_OBJ_POST,
'term' => $ng['xchan_name'],
'url' => $ng['xchan_url']
];
}
}
}
}
}
if (strlen($categories)) {
if (!isset($post_tags)) {
$post_tags = [];
}
$cats = explode(',', $categories);
foreach ($cats as $cat) {
if ($webpage == ITEM_TYPE_CARD) {
$catlink = z_root() . '/cards/' . $channel['channel_address'] . '?f=&cat=' . urlencode(trim($cat));
} elseif ($webpage == ITEM_TYPE_ARTICLE) {
$catlink = z_root() . '/articles/' . $channel['channel_address'] . '?f=&cat=' . urlencode(trim($cat));
} else {
$catlink = $owner_xchan['xchan_url'] . '?f=&cat=' . urlencode(trim($cat));
}
$post_tags[] = [
'uid' => $profile_uid,
'ttype' => TERM_CATEGORY,
'otype' => TERM_OBJ_POST,
'term' => trim($cat),
'url' => $catlink
];
}
}
if ($orig_post) {
// preserve original tags
$t = q(
"select * from term where oid = %d and otype = %d and uid = %d and ttype in ( %d, %d, %d )",
intval($orig_post['id']),
intval(TERM_OBJ_POST),
intval($profile_uid),
intval(TERM_UNKNOWN),
intval(TERM_FILE),
intval(TERM_COMMUNITYTAG)
);
if ($t) {
if (!isset($post_tags)) {
$post_tags = [];
}
foreach ($t as $t1) {
$post_tags[] = [
'uid' => $profile_uid,
'ttype' => $t1['ttype'],
'otype' => TERM_OBJ_POST,
'term' => $t1['term'],
'url' => $t1['url'],
];
}
}
}
$item_unseen = ((local_channel() != $profile_uid) ? 1 : 0);
$item_wall = ((isset($_REQUEST['type']) && ($_REQUEST['type'] === 'wall' || $_REQUEST['type'] === 'wall-comment')) ? 1 : 0);
$item_origin = (($origin) ? 1 : 0);
$item_nocomment = ((isset($item_nocomment)) ? $item_nocomment : $nocomment);
// determine if this is a wall post
if ($parent) {
$item_wall = $parent_item['item_wall'];
} else {
if (!$webpage) {
$item_wall = 1;
}
}
if ($moderated) {
$item_blocked = ITEM_MODERATED;
}
$notify_type = (($parent) ? 'comment-new' : 'wall-new');
if (!(isset($mid) && $mid)) {
if ($message_id) {
$mid = $message_id;
} else {
$uuid = new_uuid();
$mid = z_root() . '/item/' . $uuid;
}
}
// Set the conversation target.
if (empty($owner_hash)) {
$owner_hash = $owner_xchan['xchan_hash'];
}
if ($owner_hash === $channel['channel_hash']) {
$conversation = isset($parent_item) ? $parent_item['mid'] : $mid;
$datarray['target'] = [
'id' => str_replace('/item/', '/conversation/', $conversation),
'type' => 'Collection',
'attributedTo' => z_root() . '/channel/' . $channel['channel_address'],
];
$datarray['tgt_type'] = 'Collection';
}
elseif (!empty($parent_item['target'])) {
$datarray['target'] = $parent_item['target'];
$datarray['tgt_type'] = $parent_item['tgt_type'];
}
if ($is_poll) {
$poll = [
'question' => $body,
'answers' => $_REQUEST['poll_answers'],
'multiple_answers' => $_REQUEST['poll_multiple_answers'],
'expire_value' => $_REQUEST['poll_expire_value'],
'expire_unit' => $_REQUEST['poll_expire_unit'],
'tags' => $post_tags
];
$obj = $this->extract_poll_data($channel, $poll, ['item_private' => $private, 'allow_cid' => $str_contact_allow, 'allow_gid' => $str_contact_deny]);
} else {
$obj = $this->extract_bb_poll_data($body, ['item_private' => $private, 'allow_cid' => $str_contact_allow, 'allow_gid' => $str_contact_deny]);
}
if ($obj) {
$obj['url'] = $obj['id'] = $mid;
$obj['attributedTo'] = Channel::url($channel);
$datarray['obj'] = $obj;
$obj_type = 'Question';
if ($obj['endTime']) {
$d = datetime_convert('UTC', 'UTC', $obj['endTime']);
if ($d > NULL_DATE) {
$comments_closed = $d;
}
}
}
if (!$parent_mid) {
$parent_mid = $mid;
}
if ($parent_item) {
$parent_mid = $parent_item['mid'];
}
// Fallback so that we alway have a thr_parent
if (!$thr_parent) {
$thr_parent = $mid;
}
$item_thread_top = ((!$parent) ? 1 : 0);
// fix permalinks for cards, etc.
if ($webpage == ITEM_TYPE_CARD) {
$plink = z_root() . '/cards/' . $channel['channel_address'] . '/' . (($pagetitle) ? $pagetitle : $uuid);
}
if (($parent_item) && ($parent_item['item_type'] == ITEM_TYPE_CARD)) {
$r = q(
"select v from iconfig where iconfig.cat = 'system' and iconfig.k = 'CARD' and iconfig.iid = %d limit 1",
intval($parent_item['id'])
);
if ($r) {
$plink = z_root() . '/cards/' . $channel['channel_address'] . '/' . $r[0]['v'];
}
}
if ($webpage == ITEM_TYPE_ARTICLE) {
$plink = z_root() . '/articles/' . $channel['channel_address'] . '/' . (($pagetitle) ? $pagetitle : $uuid);
}
if (($parent_item) && ($parent_item['item_type'] == ITEM_TYPE_ARTICLE)) {
$r = q(
"select v from iconfig where iconfig.cat = 'system' and iconfig.k = 'ARTICLE' and iconfig.iid = %d limit 1",
intval($parent_item['id'])
);
if ($r) {
$plink = z_root() . '/articles/' . $channel['channel_address'] . '/' . $r[0]['v'];
}
}
if ((!(isset($plink) && $plink)) && $item_thread_top) {
$plink = z_root() . '/item/' . $uuid;
}
if (array_path_exists('obj/id', $datarray)) {
$datarray['obj']['id'] = $mid;
}
if (! (isset($replyto) && $replyto)) {
$replyto = Activity::actorEncode($owner_xchan, false);
}
if ($private && !$parent) {
if ( intval($private) === 1 && (!$str_group_allow) && in_array(substr_count($str_contact_allow,'<'), [ 1, 2 ])) {
$private = 2;
}
}
$datarray['aid'] = $channel['channel_account_id'];
$datarray['uid'] = $profile_uid;
$datarray['uuid'] = $uuid;
$datarray['owner_xchan'] = (($owner_hash) ? $owner_hash : $owner_xchan['xchan_hash']);
$datarray['author_xchan'] = $observer['xchan_hash'];
$datarray['created'] = $created;
$datarray['edited'] = (($orig_post && (!intval($orig_post['item_unpublished']))) ? datetime_convert() : $created);
$datarray['expires'] = $expires;
$datarray['commented'] = (($orig_post && (!intval($orig_post['item_unpublished']))) ? datetime_convert() : $created);
$datarray['received'] = (($orig_post && (!intval($orig_post['item_unpublished']))) ? datetime_convert() : $created);
$datarray['changed'] = (($orig_post && (!intval($orig_post['item_unpublished']))) ? datetime_convert() : $created);
$datarray['comments_closed'] = $comments_closed;
$datarray['mid'] = $mid;
$datarray['parent_mid'] = $parent_mid;
$datarray['mimetype'] = $mimetype;
$datarray['title'] = $title;
$datarray['summary'] = $summary;
$datarray['body'] = $body;
$datarray['app'] = $app;
$datarray['location'] = $location;
$datarray['lat'] = $lat;
$datarray['lon'] = $lon;
$datarray['verb'] = $verb;
$datarray['obj_type'] = $obj_type;
$datarray['allow_cid'] = $str_contact_allow;
$datarray['allow_gid'] = $str_group_allow;
$datarray['deny_cid'] = $str_contact_deny;
$datarray['deny_gid'] = $str_group_deny;
$datarray['attach'] = $attachments;
$datarray['thr_parent'] = $thr_parent;
$datarray['postopts'] = $postopts;
$datarray['item_wall'] = intval($item_wall);
$datarray['item_origin'] = intval($item_origin);
$datarray['item_type'] = $webpage;
$datarray['item_private'] = intval($private);
$datarray['item_thread_top'] = intval($item_thread_top);
$datarray['item_unseen'] = intval($item_unseen);
$datarray['item_starred'] = intval($item_starred);
$datarray['item_uplink'] = intval($item_uplink);
$datarray['item_consensus'] = 0;
$datarray['item_notshown'] = intval($item_notshown);
$datarray['item_nsfw'] = intval($item_nsfw);
$datarray['item_relay'] = intval($item_relay);
$datarray['item_mentionsme'] = intval($item_mentionsme);
$datarray['item_nocomment'] = intval($item_nocomment);
$datarray['item_obscured'] = intval($item_obscured);
$datarray['item_verified'] = intval($item_verified);
$datarray['item_retained'] = intval($item_retained);
$datarray['item_rss'] = intval($item_rss);
$datarray['item_deleted'] = intval($item_deleted);
$datarray['item_hidden'] = intval($item_hidden);
$datarray['item_unpublished'] = intval($item_unpublished);
$datarray['item_delayed'] = intval($item_delayed);
$datarray['item_pending_remove'] = intval($item_pending_remove);
$datarray['item_blocked'] = intval($item_blocked);
$datarray['layout_mid'] = $layout_mid;
$datarray['public_policy'] = $public_policy;
$datarray['comment_policy'] = ((is_numeric($comment_policy)) ? map_scope($comment_policy) : $comment_policy); // only map scope if it is numeric, otherwise use what we have
$datarray['term'] = $post_tags;
$datarray['plink'] = $plink;
$datarray['replyto'] = $replyto;
// A specific ACL over-rides public_policy completely
if (!empty_acl($datarray)) {
$datarray['public_policy'] = '';
}
if ($iconfig) {
$datarray['iconfig'] = $iconfig;
}
if ($private) {
IConfig::set($datarray, 'ocap', 'relay', $token);
}
if (!array_key_exists('obj', $datarray)) {
$copy = $datarray;
$copy['author'] = $observer;
$datarray['obj'] = Activity::encode_item($copy, ((get_config('system', 'activitypub', ACTIVITYPUB_ENABLED)) ? true : false));
$recips = [];
$i = $datarray['obj'];
if ($i['to']) {
$recips['to'] = $i['to'];
}
if ($i['cc']) {
$recips['cc'] = $i['cc'];
}
IConfig::Set($datarray, 'activitypub', 'recips', $recips);
}
Activity::rewrite_mentions($datarray);
// preview mode - prepare the body for display and send it via json
if ($preview) {
$datarray['owner'] = $owner_xchan;
$datarray['author'] = $observer;
$datarray['attach'] = json_encode($datarray['attach']);
Hook::call('post_prestore', $datarray);
$o = conversation([$datarray], 'search', false, 'preview');
// logger('preview: ' . $o, LOGGER_DEBUG);
echo json_encode(['preview' => $o]);
killme();
}
Hook::call('post_prestore', $datarray);
// Let 'post_local' event listeners know if this is an edit.
// We will unset it immediately afterward.
if ($orig_post) {
$datarray['edit'] = true;
}
// suppress duplicates, *unless* you're editing an existing post. This could get picked up
// as a duplicate if you're editing it very soon after posting it initially and you edited
// some attribute besides the content, such as title or categories.
if (PConfig::Get($profile_uid, 'system', 'suppress_duplicates', true) && (!$orig_post)) {
$z = q(
"select created from item where uid = %d and created > %s - INTERVAL %s and body = '%s' limit 1",
intval($profile_uid),
db_utcnow(),
db_quoteinterval('2 MINUTE'),
dbesc($body)
);
if ($z) {
$datarray['cancel'] = 1;
notice(t('Duplicate post suppressed.') . EOL);
logger('Duplicate post. Cancelled.');
}
}
Hook::call('post_local', $datarray);
// This is no longer needed
unset($datarray['edit']);
if (x($datarray, 'cancel')) {
logger('mod_item: post cancelled by plugin or duplicate suppressed.');
if ($return_path) {
goaway(z_root() . "/" . $return_path);
}
if ($api_source) {
return (['success' => false, 'message' => 'operation cancelled']);
}
$json = ['cancel' => 1];
$json['reload'] = z_root() . '/' . $_REQUEST['jsreload'];
json_return_and_die($json);
}
if (mb_strlen($datarray['title']) > 255) {
$datarray['title'] = mb_substr($datarray['title'], 0, 255);
}
if ($webpage) {
IConfig::Set(
$datarray,
'system',
webpage_to_namespace($webpage),
(($pagetitle) ? $pagetitle : basename($datarray['mid'])),
true
);
} elseif ($namespace) {
IConfig::Set(
$datarray,
'system',
$namespace,
(($remote_id) ? $remote_id : basename($datarray['mid'])),
true
);
}
if (intval($datarray['item_unpublished'])) {
$draft_msg = t('Draft saved. Use <a href="/stream?draft=1">Drafts</a> app to continue editing.');
}
if ($orig_post) {
$datarray['id'] = $post_id;
q("delete from item where verb in ('Accept', 'Reject') and obj = '%s' and uid = %d",
dbesc('"', $datarray['mid'] . '"'),
intval($channel['channel_id'])
);
$x = item_store_update($datarray);
if ($x['success']) {
Hook::call('after_item_store', $x['item']);
}
// Sync packets for edited comments are generated elsewhere.
// @FIXME: where exactly?
if (!$parent) {
$r = q(
"select * from item where id = %d",
intval($post_id)
);
if ($r) {
xchan_query($r);
$sync_item = fetch_post_tags($r);
Libsync::build_sync_packet($profile_uid, ['item' => [encode_item($sync_item[0], true)]]);
}
}
if (!$nopush) {
Run::Summon(['Notifier', 'edit_post', $post_id]);
}
if ($api_source) {
return ($x);
}
if (intval($datarray['item_unpublished'])) {
info($draft_msg);
}
if ((x($_REQUEST, 'return')) && strlen($return_path)) {
logger('return: ' . $return_path);
goaway(z_root() . "/" . $return_path);
}
killme();
}
$post = item_store($datarray);
if ($post['success']) {
Hook::call('after_item_store', $post['item']);
}
if ($pub_copy) {
info(t('Your comment has been posted.') . EOL);
}
$post_id = $post['item_id'];
$datarray = $post['item'];
if ($post_id) {
logger('mod_item: saved item ' . $post_id);
if ($parent) {
// prevent conversations which you are involved from being expired
if (local_channel()) {
retain_item($parent);
}
// only send comment notification if this is a wall-to-wall comment and not a DM,
// otherwise it will happen during delivery
if (($datarray['owner_xchan'] != $datarray['author_xchan']) && (intval($parent_item['item_wall'])) && intval($datarray['item_private']) != 2) {
Enotify::submit([
'type' => NOTIFY_COMMENT,
'from_xchan' => $datarray['author_xchan'],
'to_xchan' => $datarray['owner_xchan'],
'item' => $datarray,
'link' => z_root() . '/display/?mid=' . gen_link_id($datarray['mid']),
'verb' => ACTIVITY_POST,
'otype' => 'item',
'parent' => $parent,
'parent_mid' => $parent_item['mid']
]);
}
} else {
$parent = $post_id;
if (($datarray['owner_xchan'] != $datarray['author_xchan']) && ($datarray['item_type'] == ITEM_TYPE_POST)) {
Enotify::submit([
'type' => NOTIFY_WALL,
'from_xchan' => $datarray['author_xchan'],
'to_xchan' => $datarray['owner_xchan'],
'item' => $datarray,
'link' => z_root() . '/display/?mid=' . gen_link_id($datarray['mid']),
'verb' => ACTIVITY_POST,
'otype' => 'item'
]);
}
if ($uid && $uid == $profile_uid && (is_item_normal($datarray))) {
q(
"update channel set channel_lastpost = '%s' where channel_id = %d",
dbesc(datetime_convert()),
intval($uid)
);
}
}
// photo comments turn the corresponding item visible to the profile wall
// This way we don't see every picture in your new photo album posted to your wall at once.
// They will show up as people comment on them.
if (intval($parent_item['item_hidden'])) {
$r = q(
"UPDATE item SET item_hidden = 0 WHERE id = %d",
intval($parent_item['id'])
);
}
} else {
logger('mod_item: unable to retrieve post that was just stored.');
notice(t('System error. Post not saved.') . EOL);
if ($return_path) {
goaway(z_root() . "/" . $return_path);
}
if ($api_source) {
return (['success' => false, 'message' => 'system error']);
}
killme();
}
$datarray['id'] = $post_id;
$datarray['llink'] = z_root() . '/display/?mid=' . gen_link_id($datarray['mid']);
Hook::call('post_local_end', $datarray);
if ($groupww) {
$nopush = false;
}
if (!$nopush) {
Run::Summon(['Notifier', $notify_type, $post_id]);
if (intval($post['approval_id'])) {
Run::Summon(['Notifier', $notify_type, $post['approval_id']]);
}
}
logger('post_complete');
if ($moderated) {
info(t('Your post/comment is awaiting approval.') . EOL);
}
// figure out how to return, depending on from whence we came
if ($api_source) {
return $post;
}
if (intval($datarray['item_unpublished'])) {
info($draft_msg);
}
if ($return_path) {
goaway(z_root() . "/" . $return_path);
}
$json = ['success' => 1];
if (x($_REQUEST, 'jsreload') && strlen($_REQUEST['jsreload'])) {
$json['reload'] = z_root() . '/' . $_REQUEST['jsreload'];
}
logger('post_json: ' . print_r($json, true), LOGGER_DEBUG);
json_return_and_die($json);
return;
}
public function get()
{
if ($this->return_404) {
notice(t('Not found'));
return;
}
if ((!local_channel()) && (!remote_channel())) {
return;
}
// allow pinned items to be dropped. 'pin-' was prepended to the id of these
// items so that they would have a unique html id even if the pinned item
// was also displayed in a normal conversation on the same web page.
$drop_id = str_replace('pin-', '', argv(2));
if ((argc() == 3) && (argv(1) === 'drop') && intval($drop_id)) {
$i = q(
"select * from item where id = %d limit 1",
intval($drop_id)
);
if ($i) {
$dropped_item = array_shift($i);
$can_delete = false;
$local_delete = false;
$regular_delete = false;
if (local_channel() && local_channel() == $dropped_item['uid']) {
$local_delete = true;
}
$ob_hash = get_observer_hash();
if ($ob_hash && ($ob_hash === $dropped_item['author_xchan'] || $ob_hash === $dropped_item['owner_xchan'] || $ob_hash === $dropped_item['source_xchan'])) {
$can_delete = true;
$regular_delete = true;
}
// The site admin can delete any post/item on the site.
// If the item originated on this site+channel the deletion will propagate downstream.
// Otherwise just the local copy is removed.
if (is_site_admin()) {
$local_delete = true;
if (intval($dropped_item['item_origin'])) {
$can_delete = true;
}
}
if (!($can_delete || $local_delete)) {
notice(t('Permission denied.') . EOL);
return;
}
if ($dropped_item['resource_type'] === 'event') {
// delete and sync the event separately
$r = q(
"SELECT * FROM event WHERE event_hash = '%s' AND uid = %d LIMIT 1",
dbesc($dropped_item['resource_id']),
intval($dropped_item['uid'])
);
if ($r && $regular_delete) {
$sync_event = $r[0];
q(
"delete from event WHERE event_hash = '%s' AND uid = %d LIMIT 1",
dbesc($dropped_item['resource_id']),
intval($dropped_item['uid'])
);
$sync_event['event_deleted'] = 1;
Libsync::build_sync_packet($dropped_item['uid'], ['event' => [$sync_event]]);
}
}
if ($dropped_item['resource_type'] === 'photo') {
attach_delete($dropped_item['uid'], $dropped_item['resource_id'], true);
$ch = Channel::from_id($dropped_item['uid']);
if ($ch && $regular_delete) {
$sync = attach_export_data($ch, $dropped_item['resource_id'], true);
if ($sync) {
Libsync::build_sync_packet($dropped_item['uid'], ['file' => [$sync]]);
}
}
}
// if this is a different page type or it's just a local delete
// but not by the item author or owner, do a simple deletion
$complex = false;
if (intval($dropped_item['item_type']) || ($local_delete && (!$can_delete))) {
drop_item($dropped_item['id']);
} else {
// complex deletion that needs to propagate and be performed in phases
drop_item($dropped_item['id'], DROPITEM_PHASE1);
$complex = true;
}
$sync = q(
"select * from item where id = %d",
intval($dropped_item['id'])
);
if ($sync) {
xchan_query($sync);
$sync_item = fetch_post_tags($sync);
Libsync::build_sync_packet($dropped_item['uid'], [ 'item' => [ encode_item($sync_item[0], true)]]);
}
if ($complex) {
tag_deliver($dropped_item['uid'], $dropped_item['id']);
if (intval($dropped_item['item_wall']) || $dropped_item['mid'] !== $dropped_item['parent_mid']) {
Run::Summon(['Notifier', 'drop', $dropped_item['id']]);
}
}
}
}
}
public function item_check_service_class($channel_id, $iswebpage)
{
$ret = ['success' => false, 'message' => ''];
$max = engr_units_to_bytes(ServiceClass::fetch($channel_id, ($iswebpage) ? 'total_pages' : 'total_items'));
if (!$max) {
$ret['success'] = true;
return $ret;
}
if ($iswebpage) {
$r = q(
"select count(i.id) as total from item i
right join channel c on (i.author_xchan=c.channel_hash and i.uid=c.channel_id )
and i.parent=i.id and i.item_type = %d and i.item_deleted = 0 and i.uid= %d ",
intval(ITEM_TYPE_WEBPAGE),
intval($channel_id)
);
} else {
$r = q(
"select count(id) as total from item where parent = id and item_wall = 1 and uid = %d " . item_normal(),
intval($channel_id)
);
}
if (!$r) {
$ret['message'] = t('Unable to obtain post information from database.');
return $ret;
}
if (!$iswebpage) {
if (!ServiceClass::allows($channel_id, 'total_items', $r[0]['total'])) {
$ret['message'] .= ServiceClass::upgrade_message() . sprintf(t('You have reached your limit of %1$.0f top level posts.'), $max);
return $ret;
}
} else {
if (!ServiceClass::allows($channel_id, 'total_pages', $r[0]['total'])) {
$ret['message'] .= ServiceClass::upgrade_message() . sprintf(t('You have reached your limit of %1$.0f webpages.'), $max);
return $ret;
}
}
$ret['success'] = true;
return $ret;
}
public function extract_bb_poll_data(&$body, $item)
{
if (!str_contains($body, '[/question]') && !str_contains($body, '[/answer]')) {
return false;
}
if (str_contains($body, '[nobb]')) {
return false;
}
$obj = [];
$ptr = [];
$matches = null;
$obj['type'] = 'Question';
if (preg_match_all('/\[answer](.*?)\[\/answer]/ism', $body, $matches, PREG_SET_ORDER)) {
foreach ($matches as $match) {
$ptr[] = ['name' => $match[1], 'type' => 'Note', 'replies' => ['type' => 'Collection', 'totalItems' => 0]];
$body = str_replace('[answer]' . $match[1] . '[/answer]', EMPTY_STR, $body);
}
}
$matches = null;
if (preg_match('/\[question](.*?)\[\/question]/ism', $body, $matches)) {
$obj['content'] = bbcode($matches[1]);
$body = str_replace('[question]' . $matches[1] . '[/question]', $matches[1], $body);
$obj['oneOf'] = $ptr;
}
$matches = null;
if (preg_match('/\[question=multiple](.*?)\[\/question]/ism', $body, $matches)) {
$obj['content'] = bbcode($matches[1]);
$body = str_replace('[question=multiple]' . $matches[1] . '[/question]', $matches[1], $body);
$obj['anyOf'] = $ptr;
}
$matches = null;
if (preg_match('/\[ends](.*?)\[\/ends]/ism', $body, $matches)) {
$obj['endTime'] = datetime_convert(date_default_timezone_get(), 'UTC', $matches[1], ATOM_TIME);
$body = str_replace('[ends]' . $matches[1] . '[/ends]', EMPTY_STR, $body);
}
if ($item['item_private']) {
$obj['to'] = Activity::map_acl($item);
} else {
$obj['to'] = [ACTIVITY_PUBLIC_INBOX];
}
return $obj;
}
public function extract_poll_data($channel, $poll, $item)
{
$multiple = intval($poll['multiple_answers']);
$expire_value = intval($poll['expire_value']);
$expire_unit = $poll['expire_unit'];
$question = $poll['question'];
$answers = $poll['answers'];
$term = isset($poll['tags']) ? $poll['tags'] : [];
$obj = [];
$ptr = [];
$obj['type'] = 'Question';
$obj['content'] = bbcode($question);
foreach ($answers as $answer) {
if (trim($answer)) {
$ptr[] = ['name' => escape_tags(trim($answer)), 'type' => 'Note', 'replies' => ['type' => 'Collection', 'totalItems' => 0]];
}
}
if ($multiple) {
$obj['anyOf'] = $ptr;
} else {
$obj['oneOf'] = $ptr;
}
$obj['endTime'] = datetime_convert(date_default_timezone_get(), 'UTC', 'now + ' . $expire_value . ' ' . $expire_unit, ATOM_TIME);
if ($item['item_private']) {
$obj['to'] = Activity::map_acl($item);
} else {
$obj['to'] = [ACTIVITY_PUBLIC_INBOX];
$obj['cc'] = [z_root() . '/followers/' . $channel['channel_address']];
}
$tags = Activity::encode_taxonomy(['term' => $term]);
if ($tags) {
$obj['tag'] = $tags;
}
return $obj;
}
}