2017-12-09 18:31:00 +00:00
< ? php
/**
* @ file src / Model / Group . php
*/
2019-07-28 13:39:45 +00:00
2017-12-09 18:31:00 +00:00
namespace Friendica\Model ;
2018-10-17 19:30:41 +00:00
use Friendica\BaseModule ;
2018-02-26 00:58:23 +00:00
use Friendica\Core\L10n ;
2018-10-29 21:20:46 +00:00
use Friendica\Core\Logger ;
2019-07-17 00:23:19 +00:00
use Friendica\Core\Protocol ;
2018-10-31 14:35:50 +00:00
use Friendica\Core\Renderer ;
2018-07-20 12:19:26 +00:00
use Friendica\Database\DBA ;
2017-12-09 18:31:00 +00:00
/**
2020-01-19 06:05:23 +00:00
* functions for interacting with the group database table
2017-12-09 18:31:00 +00:00
*/
2019-12-15 22:28:01 +00:00
class Group
2017-12-09 18:31:00 +00:00
{
2019-07-15 01:48:35 +00:00
const FOLLOWERS = '~' ;
const MUTUALS = '&' ;
public static function getByUserId ( $uid , $includesDeleted = false )
{
$conditions = [ 'uid' => $uid ];
if ( ! $includesDeleted ) {
$conditions [ 'deleted' ] = false ;
}
2019-07-28 13:39:45 +00:00
return DBA :: selectToArray ( 'group' , [], $conditions );
2019-07-15 01:48:35 +00:00
}
2019-02-23 20:33:55 +00:00
/**
* @ param int $group_id
* @ return bool
* @ throws \Exception
*/
public static function exists ( $group_id , $uid = null )
{
$condition = [ 'id' => $group_id , 'deleted' => false ];
if ( isset ( $uid )) {
$condition = [
'uid' => $uid
];
}
return DBA :: exists ( 'group' , $condition );
}
2017-12-09 18:31:00 +00:00
/**
2020-01-19 06:05:23 +00:00
* Create a new contact group
2017-12-09 18:31:00 +00:00
*
* Note : If we found a deleted group with the same name , we restore it
*
2019-01-06 21:06:53 +00:00
* @ param int $uid
2017-12-09 18:31:00 +00:00
* @ param string $name
* @ return boolean
2019-01-06 21:06:53 +00:00
* @ throws \Exception
2017-12-09 18:31:00 +00:00
*/
public static function create ( $uid , $name )
{
$return = false ;
2018-11-30 14:06:22 +00:00
if ( ! empty ( $uid ) && ! empty ( $name )) {
2017-12-09 18:31:00 +00:00
$gid = self :: getIdByName ( $uid , $name ); // check for dupes
if ( $gid !== false ) {
// This could be a problem.
// Let's assume we've just created a group which we once deleted
// all the old members are gone, but the group remains so we don't break any security
// access lists. What we're doing here is reviving the dead group, but old content which
// was restricted to this group may now be seen by the new group members.
2018-07-20 12:19:26 +00:00
$group = DBA :: selectFirst ( 'group' , [ 'deleted' ], [ 'id' => $gid ]);
2018-07-21 12:46:04 +00:00
if ( DBA :: isResult ( $group ) && $group [ 'deleted' ]) {
2018-07-20 12:19:26 +00:00
DBA :: update ( 'group' , [ 'deleted' => 0 ], [ 'id' => $gid ]);
2020-01-18 19:52:34 +00:00
notice ( DI :: l10n () -> t ( 'A deleted group with this name was revived. Existing item permissions <strong>may</strong> apply to this group and any future members. If this is not what you intended, please create another group with a different name.' ) . EOL );
2017-12-09 18:31:00 +00:00
}
return true ;
}
2018-07-20 12:19:26 +00:00
$return = DBA :: insert ( 'group' , [ 'uid' => $uid , 'name' => $name ]);
2017-12-13 01:52:50 +00:00
if ( $return ) {
2018-07-20 12:19:26 +00:00
$return = DBA :: lastInsertId ();
2017-12-13 01:52:50 +00:00
}
2017-12-09 18:31:00 +00:00
}
return $return ;
}
2018-04-07 13:54:26 +00:00
/**
* Update group information .
*
2019-07-28 13:39:45 +00:00
* @ param int $id Group ID
* @ param string $name Group name
2018-04-07 13:54:26 +00:00
*
* @ return bool Was the update successful ?
2019-01-06 21:06:53 +00:00
* @ throws \Exception
2018-04-07 13:54:26 +00:00
*/
public static function update ( $id , $name )
{
2018-07-20 12:19:26 +00:00
return DBA :: update ( 'group' , [ 'name' => $name ], [ 'id' => $id ]);
2018-04-07 13:54:26 +00:00
}
2017-12-09 18:31:00 +00:00
/**
2020-01-19 06:05:23 +00:00
* Get a list of group ids a contact belongs to
2017-12-09 18:31:00 +00:00
*
* @ param int $cid
* @ return array
2019-01-06 21:06:53 +00:00
* @ throws \Exception
2017-12-09 18:31:00 +00:00
*/
2017-12-17 00:21:56 +00:00
public static function getIdsByContactId ( $cid )
2017-12-09 18:31:00 +00:00
{
$return = [];
2017-12-10 20:12:23 +00:00
2019-07-27 22:06:29 +00:00
$stmt = DBA :: select ( 'group_member' , [ 'gid' ], [ 'contact-id' => $cid ]);
2018-07-20 12:19:26 +00:00
while ( $group = DBA :: fetch ( $stmt )) {
2017-12-10 20:12:23 +00:00
$return [] = $group [ 'gid' ];
2017-12-09 18:31:00 +00:00
}
2019-07-27 22:06:29 +00:00
DBA :: close ( $stmt );
2017-12-09 18:31:00 +00:00
return $return ;
}
/**
2020-01-19 06:05:23 +00:00
* count unread group items
2017-12-09 18:31:00 +00:00
*
* Count unread items of each groups of the local user
*
* @ return array
2019-01-06 21:06:53 +00:00
* 'id' => group id
* 'name' => group name
* 'count' => counted unseen group items
* @ throws \Exception
2017-12-09 18:31:00 +00:00
*/
public static function countUnseen ()
{
2018-07-20 12:19:26 +00:00
$stmt = DBA :: p ( " SELECT `group`.`id`, `group`.`name`,
2017-12-09 18:31:00 +00:00
( SELECT COUNT ( * ) FROM `item` FORCE INDEX ( `uid_unseen_contactid` )
WHERE `uid` = ?
AND `unseen`
AND `contact-id` IN
( SELECT `contact-id`
FROM `group_member`
2017-12-15 03:47:58 +00:00
WHERE `group_member` . `gid` = `group` . `id` )
2017-12-09 18:31:00 +00:00
) AS `count`
FROM `group`
WHERE `group` . `uid` = ? ; " ,
local_user (),
local_user ()
);
2018-07-21 02:03:40 +00:00
return DBA :: toArray ( $stmt );
2017-12-09 18:31:00 +00:00
}
/**
2020-01-19 06:05:23 +00:00
* Get the group id for a user / name couple
2017-12-09 18:31:00 +00:00
*
* Returns false if no group has been found .
*
2019-01-06 21:06:53 +00:00
* @ param int $uid
2017-12-09 18:31:00 +00:00
* @ param string $name
* @ return int | boolean
2019-01-06 21:06:53 +00:00
* @ throws \Exception
2017-12-09 18:31:00 +00:00
*/
public static function getIdByName ( $uid , $name )
{
2017-12-10 06:07:48 +00:00
if ( ! $uid || ! strlen ( $name )) {
2017-12-09 18:31:00 +00:00
return false ;
}
2018-07-20 12:19:26 +00:00
$group = DBA :: selectFirst ( 'group' , [ 'id' ], [ 'uid' => $uid , 'name' => $name ]);
2018-07-21 12:46:04 +00:00
if ( DBA :: isResult ( $group )) {
2017-12-09 18:31:00 +00:00
return $group [ 'id' ];
}
return false ;
}
/**
2020-01-19 06:05:23 +00:00
* Mark a group as deleted
2017-12-09 18:31:00 +00:00
*
2017-12-17 20:27:50 +00:00
* @ param int $gid
2017-12-09 18:31:00 +00:00
* @ return boolean
2019-01-06 21:06:53 +00:00
* @ throws \Exception
2017-12-09 18:31:00 +00:00
*/
2019-07-28 13:39:45 +00:00
public static function remove ( $gid )
{
if ( ! $gid ) {
2017-12-09 18:31:00 +00:00
return false ;
}
2018-07-20 12:19:26 +00:00
$group = DBA :: selectFirst ( 'group' , [ 'uid' ], [ 'id' => $gid ]);
2018-07-21 12:46:04 +00:00
if ( ! DBA :: isResult ( $group )) {
2017-12-17 20:31:37 +00:00
return false ;
}
2017-12-09 18:31:00 +00:00
// remove group from default posting lists
2018-07-20 12:19:26 +00:00
$user = DBA :: selectFirst ( 'user' , [ 'def_gid' , 'allow_gid' , 'deny_gid' ], [ 'uid' => $group [ 'uid' ]]);
2018-07-21 12:46:04 +00:00
if ( DBA :: isResult ( $user )) {
2017-12-09 18:31:00 +00:00
$change = false ;
if ( $user [ 'def_gid' ] == $gid ) {
$user [ 'def_gid' ] = 0 ;
$change = true ;
}
if ( strpos ( $user [ 'allow_gid' ], '<' . $gid . '>' ) !== false ) {
$user [ 'allow_gid' ] = str_replace ( '<' . $gid . '>' , '' , $user [ 'allow_gid' ]);
$change = true ;
}
if ( strpos ( $user [ 'deny_gid' ], '<' . $gid . '>' ) !== false ) {
$user [ 'deny_gid' ] = str_replace ( '<' . $gid . '>' , '' , $user [ 'deny_gid' ]);
$change = true ;
}
if ( $change ) {
2018-07-20 12:19:26 +00:00
DBA :: update ( 'user' , $user , [ 'uid' => $group [ 'uid' ]]);
2017-12-09 18:31:00 +00:00
}
}
// remove all members
2018-07-20 12:19:26 +00:00
DBA :: delete ( 'group_member' , [ 'gid' => $gid ]);
2017-12-09 18:31:00 +00:00
// remove group
2018-07-20 12:19:26 +00:00
$return = DBA :: update ( 'group' , [ 'deleted' => 1 ], [ 'id' => $gid ]);
2017-12-09 18:31:00 +00:00
return $return ;
}
/**
2020-01-19 06:05:23 +00:00
* Mark a group as deleted based on its name
2017-12-09 18:31:00 +00:00
*
2019-01-06 21:06:53 +00:00
* @ param int $uid
2017-12-17 20:27:50 +00:00
* @ param string $name
* @ return bool
2019-01-06 21:06:53 +00:00
* @ throws \Exception
2019-07-28 13:39:45 +00:00
* @ deprecated Use Group :: remove instead
*
2017-12-09 18:31:00 +00:00
*/
2019-07-28 13:39:45 +00:00
public static function removeByName ( $uid , $name )
{
2017-12-09 18:31:00 +00:00
$return = false ;
2018-11-30 14:06:22 +00:00
if ( ! empty ( $uid ) && ! empty ( $name )) {
2017-12-09 18:31:00 +00:00
$gid = self :: getIdByName ( $uid , $name );
$return = self :: remove ( $gid );
}
return $return ;
}
/**
2020-01-19 06:05:23 +00:00
* Adds a contact to a group
2017-12-09 18:31:00 +00:00
*
* @ param int $gid
* @ param int $cid
* @ return boolean
2019-01-06 21:06:53 +00:00
* @ throws \Exception
2017-12-09 18:31:00 +00:00
*/
public static function addMember ( $gid , $cid )
{
2017-12-10 06:06:12 +00:00
if ( ! $gid || ! $cid ) {
2017-12-09 18:31:00 +00:00
return false ;
}
2018-07-20 12:19:26 +00:00
$row_exists = DBA :: exists ( 'group_member' , [ 'gid' => $gid , 'contact-id' => $cid ]);
2017-12-09 18:31:00 +00:00
if ( $row_exists ) {
// Row already existing, nothing to do
$return = true ;
} else {
2018-07-20 12:19:26 +00:00
$return = DBA :: insert ( 'group_member' , [ 'gid' => $gid , 'contact-id' => $cid ]);
2017-12-09 18:31:00 +00:00
}
return $return ;
}
/**
2020-01-19 06:05:23 +00:00
* Removes a contact from a group
2017-12-09 18:31:00 +00:00
*
* @ param int $gid
* @ param int $cid
* @ return boolean
2019-01-06 21:06:53 +00:00
* @ throws \Exception
2017-12-09 18:31:00 +00:00
*/
public static function removeMember ( $gid , $cid )
{
2017-12-10 06:06:12 +00:00
if ( ! $gid || ! $cid ) {
2017-12-09 18:31:00 +00:00
return false ;
}
2018-07-20 12:19:26 +00:00
$return = DBA :: delete ( 'group_member' , [ 'gid' => $gid , 'contact-id' => $cid ]);
2017-12-09 18:31:00 +00:00
return $return ;
}
/**
2020-01-19 06:05:23 +00:00
* Removes a contact from a group based on its name
2017-12-09 18:31:00 +00:00
*
2019-01-06 21:06:53 +00:00
* @ param int $uid
2017-12-09 18:31:00 +00:00
* @ param string $name
2019-01-06 21:06:53 +00:00
* @ param int $cid
2017-12-09 18:31:00 +00:00
* @ return boolean
2019-01-06 21:06:53 +00:00
* @ throws \Exception
2019-07-28 13:39:45 +00:00
* @ deprecated Use Group :: removeMember instead
*
2017-12-09 18:31:00 +00:00
*/
public static function removeMemberByName ( $uid , $name , $cid )
{
$gid = self :: getIdByName ( $uid , $name );
$return = self :: removeMember ( $gid , $cid );
return $return ;
}
/**
2020-01-19 06:05:23 +00:00
* Returns the combined list of contact ids from a group id list
2017-12-09 18:31:00 +00:00
*
2019-07-15 01:48:35 +00:00
* @ param int $uid
2019-01-06 21:06:53 +00:00
* @ param array $group_ids
2017-12-09 18:31:00 +00:00
* @ param boolean $check_dead
* @ return array
2019-01-06 21:06:53 +00:00
* @ throws \Exception
2017-12-09 18:31:00 +00:00
*/
2019-07-15 01:48:35 +00:00
public static function expand ( $uid , array $group_ids , $check_dead = false )
2017-12-09 18:31:00 +00:00
{
2017-12-10 06:06:12 +00:00
if ( ! is_array ( $group_ids ) || ! count ( $group_ids )) {
2017-12-09 18:31:00 +00:00
return [];
}
2017-12-10 06:06:12 +00:00
$return = [];
2019-12-02 23:04:11 +00:00
$pubmail = false ;
$networks = Protocol :: SUPPORT_PRIVATE ;
$mailacct = DBA :: selectFirst ( 'mailacct' , [ 'pubmail' ], [ '`uid` = ? AND `server` != ""' , $uid ]);
if ( DBA :: isResult ( $mailacct )) {
$pubmail = $mailacct [ 'pubmail' ];
}
if ( ! $pubmail ) {
$networks = array_diff ( $networks , [ Protocol :: MAIL ]);
}
2019-07-15 01:48:35 +00:00
$key = array_search ( self :: FOLLOWERS , $group_ids );
if ( $key !== false ) {
2019-07-27 22:06:29 +00:00
$followers = Contact :: selectToArray ([ 'id' ], [
2019-07-17 00:23:19 +00:00
'uid' => $uid ,
'rel' => [ Contact :: FOLLOWER , Contact :: FRIEND ],
2019-12-02 23:04:11 +00:00
'network' => $networks ,
'contact-type' => [ Contact :: TYPE_UNKNOWN , Contact :: TYPE_PERSON ],
'archive' => false ,
'pending' => false ,
'blocked' => false ,
2019-07-17 00:23:19 +00:00
]);
2019-07-15 01:48:35 +00:00
2019-07-27 22:06:29 +00:00
foreach ( $followers as $follower ) {
2019-07-15 01:48:35 +00:00
$return [] = $follower [ 'id' ];
}
unset ( $group_ids [ $key ]);
}
$key = array_search ( self :: MUTUALS , $group_ids );
if ( $key !== false ) {
2019-07-27 22:06:29 +00:00
$mutuals = Contact :: selectToArray ([ 'id' ], [
2019-07-17 00:23:19 +00:00
'uid' => $uid ,
'rel' => [ Contact :: FRIEND ],
2019-12-02 23:04:11 +00:00
'network' => $networks ,
'contact-type' => [ Contact :: TYPE_UNKNOWN , Contact :: TYPE_PERSON ],
'archive' => false ,
'pending' => false ,
'blocked' => false ,
2019-07-17 00:23:19 +00:00
]);
2019-07-15 01:48:35 +00:00
2019-07-27 22:06:29 +00:00
foreach ( $mutuals as $mutual ) {
2019-07-15 01:48:35 +00:00
$return [] = $mutual [ 'id' ];
}
unset ( $group_ids [ $key ]);
}
$stmt = DBA :: select ( 'group_member' , [ 'contact-id' ], [ 'gid' => $group_ids ]);
2019-07-28 13:39:45 +00:00
while ( $group_member = DBA :: fetch ( $stmt )) {
2017-12-10 06:06:12 +00:00
$return [] = $group_member [ 'contact-id' ];
2017-12-09 18:31:00 +00:00
}
2019-07-27 22:06:29 +00:00
DBA :: close ( $stmt );
2017-12-09 18:31:00 +00:00
2018-08-23 13:51:58 +00:00
if ( $check_dead ) {
2020-01-05 22:07:33 +00:00
$return = Contact :: pruneUnavailable ( $return );
2017-12-09 18:31:00 +00:00
}
2018-08-23 13:51:58 +00:00
2017-12-09 18:31:00 +00:00
return $return ;
}
/**
2020-01-19 06:05:23 +00:00
* Returns a templated group selection list
2017-12-09 18:31:00 +00:00
*
2019-01-06 21:06:53 +00:00
* @ param int $uid
* @ param int $gid An optional pre - selected group
2017-12-09 18:31:00 +00:00
* @ param string $label An optional label of the list
* @ return string
2019-07-15 01:48:35 +00:00
* @ throws \Exception
2017-12-09 18:31:00 +00:00
*/
public static function displayGroupSelection ( $uid , $gid = 0 , $label = '' )
{
$display_groups = [
[
'name' => '' ,
'id' => '0' ,
'selected' => ''
]
];
2019-07-27 22:06:29 +00:00
$stmt = DBA :: select ( 'group' , [], [ 'deleted' => 0 , 'uid' => $uid ], [ 'order' => [ 'name' ]]);
2018-07-20 12:19:26 +00:00
while ( $group = DBA :: fetch ( $stmt )) {
2017-12-09 18:31:00 +00:00
$display_groups [] = [
'name' => $group [ 'name' ],
'id' => $group [ 'id' ],
'selected' => $gid == $group [ 'id' ] ? 'true' : ''
];
}
2019-07-27 22:06:29 +00:00
DBA :: close ( $stmt );
2019-07-28 13:39:45 +00:00
Logger :: info ( 'Got groups' , $display_groups );
2017-12-09 18:31:00 +00:00
if ( $label == '' ) {
2020-01-18 19:52:34 +00:00
$label = DI :: l10n () -> t ( 'Default privacy group for new contacts' );
2017-12-09 18:31:00 +00:00
}
2018-10-31 14:44:06 +00:00
$o = Renderer :: replaceMacros ( Renderer :: getMarkupTemplate ( 'group_selection.tpl' ), [
2017-12-09 18:31:00 +00:00
'$label' => $label ,
'$groups' => $display_groups
2018-01-15 13:05:12 +00:00
]);
2017-12-09 18:31:00 +00:00
return $o ;
}
/**
2020-01-19 06:05:23 +00:00
* Create group sidebar widget
2017-12-09 18:31:00 +00:00
*
* @ param string $every
* @ param string $each
* @ param string $editmode
2019-01-06 21:06:53 +00:00
* 'standard' => include link 'Edit groups'
* 'extended' => include link 'Create new group'
* 'full' => include link 'Create new group' and provide for each group a link to edit this group
* @ param string $group_id
* @ param int $cid
2017-12-09 18:31:00 +00:00
* @ return string
2019-07-15 01:48:35 +00:00
* @ throws \Exception
2017-12-09 18:31:00 +00:00
*/
2018-10-14 22:02:54 +00:00
public static function sidebarWidget ( $every = 'contact' , $each = 'group' , $editmode = 'standard' , $group_id = '' , $cid = 0 )
2017-12-09 18:31:00 +00:00
{
if ( ! local_user ()) {
return '' ;
}
$display_groups = [
[
2020-01-18 19:52:34 +00:00
'text' => DI :: l10n () -> t ( 'Everybody' ),
2017-12-09 18:31:00 +00:00
'id' => 0 ,
2018-07-09 22:36:50 +00:00
'selected' => (( $group_id === 'everyone' ) ? 'group-selected' : '' ),
2017-12-09 18:31:00 +00:00
'href' => $every ,
]
];
2018-01-15 13:05:12 +00:00
$member_of = [];
2017-12-09 18:31:00 +00:00
if ( $cid ) {
2017-12-16 16:50:58 +00:00
$member_of = self :: getIdsByContactId ( $cid );
2017-12-09 18:31:00 +00:00
}
2019-07-27 22:06:29 +00:00
$stmt = DBA :: select ( 'group' , [], [ 'deleted' => 0 , 'uid' => local_user ()], [ 'order' => [ 'name' ]]);
2018-07-20 12:19:26 +00:00
while ( $group = DBA :: fetch ( $stmt )) {
2017-12-10 06:06:12 +00:00
$selected = (( $group_id == $group [ 'id' ]) ? ' group-selected' : '' );
2017-12-09 18:31:00 +00:00
2017-12-10 06:06:12 +00:00
if ( $editmode == 'full' ) {
$groupedit = [
'href' => 'group/' . $group [ 'id' ],
2020-01-18 19:52:34 +00:00
'title' => DI :: l10n () -> t ( 'edit' ),
2017-12-09 18:31:00 +00:00
];
2017-12-10 06:06:12 +00:00
} else {
$groupedit = null ;
2017-12-09 18:31:00 +00:00
}
2017-12-10 06:06:12 +00:00
$display_groups [] = [
'id' => $group [ 'id' ],
'cid' => $cid ,
'text' => $group [ 'name' ],
'href' => $each . '/' . $group [ 'id' ],
'edit' => $groupedit ,
'selected' => $selected ,
'ismember' => in_array ( $group [ 'id' ], $member_of ),
];
2017-12-09 18:31:00 +00:00
}
2019-07-27 22:06:29 +00:00
DBA :: close ( $stmt );
2017-12-09 18:31:00 +00:00
2018-12-14 06:33:57 +00:00
// Don't show the groups on the network page when there is only one
if (( count ( $display_groups ) <= 2 ) && ( $each == 'network' )) {
2018-11-24 12:10:30 +00:00
return '' ;
}
2018-10-31 14:44:06 +00:00
$tpl = Renderer :: getMarkupTemplate ( 'group_side.tpl' );
2018-10-31 14:35:50 +00:00
$o = Renderer :: replaceMacros ( $tpl , [
2020-01-18 19:52:34 +00:00
'$add' => DI :: l10n () -> t ( 'add' ),
'$title' => DI :: l10n () -> t ( 'Groups' ),
2017-12-09 18:31:00 +00:00
'$groups' => $display_groups ,
'newgroup' => $editmode == 'extended' || $editmode == 'full' ? 1 : '' ,
'grouppage' => 'group/' ,
2020-01-18 19:52:34 +00:00
'$edittext' => DI :: l10n () -> t ( 'Edit group' ),
'$ungrouped' => $every === 'contact' ? DI :: l10n () -> t ( 'Contacts not in any group' ) : '' ,
2018-07-09 22:36:50 +00:00
'$ungrouped_selected' => (( $group_id === 'none' ) ? 'group-selected' : '' ),
2020-01-18 19:52:34 +00:00
'$createtext' => DI :: l10n () -> t ( 'Create a new group' ),
'$creategroup' => DI :: l10n () -> t ( 'Group Name: ' ),
'$editgroupstext' => DI :: l10n () -> t ( 'Edit groups' ),
2018-10-17 19:30:41 +00:00
'$form_security_token' => BaseModule :: getFormSecurityToken ( 'group_edit' ),
2017-12-09 18:31:00 +00:00
]);
return $o ;
}
}