2005-06-14 23:54:17 +02:00
< ? php
2007-03-09 12:39:47 +01:00
/**
2011-04-06 21:26:10 +02:00
* EGroupware - Calendar ' s buisness - object - access + update
2007-03-09 12:39:47 +01:00
*
* @ link http :// www . egroupware . org
* @ package calendar
* @ author Ralf Becker < RalfBecker - AT - outdoor - training . de >
2009-07-15 22:35:56 +02:00
* @ author Joerg Lehrke < jlehrke @ noc . de >
2016-05-01 19:47:59 +02:00
* @ copyright ( c ) 2005 - 16 by RalfBecker - At - outdoor - training . de
2007-03-09 12:39:47 +01:00
* @ license http :// opensource . org / licenses / gpl - license . php GPL - GNU General Public License
* @ version $Id $
*/
2005-06-14 23:54:17 +02:00
2016-05-01 19:47:59 +02:00
use EGroupware\Api ;
use EGroupware\Api\Acl ;
2020-03-11 16:25:24 +01:00
use EGroupware\Api\Link ;
2016-05-01 19:47:59 +02:00
2009-08-04 19:14:16 +02:00
// types of messsages send by calendar_boupdate::send_update
2005-11-09 00:15:14 +01:00
define ( 'MSG_DELETED' , 0 );
define ( 'MSG_MODIFIED' , 1 );
define ( 'MSG_ADDED' , 2 );
define ( 'MSG_REJECTED' , 3 );
define ( 'MSG_TENTATIVE' , 4 );
define ( 'MSG_ACCEPTED' , 5 );
define ( 'MSG_ALARM' , 6 );
define ( 'MSG_DISINVITE' , 7 );
2010-02-17 14:29:28 +01:00
define ( 'MSG_DELEGATED' , 8 );
2021-02-11 19:43:47 +01:00
define ( 'MSG_REQUEST' , 9 );
2005-11-09 00:15:14 +01:00
2005-06-14 23:54:17 +02:00
/**
2005-11-09 00:15:14 +01:00
* Class to access AND manipulate all calendar data ( business object )
2005-06-14 23:54:17 +02:00
*
* The new UI , BO and SO classes have a strikt definition , in which time - zone they operate :
* UI only operates in user - time , so there have to be no conversation at all !!!
* BO ' s functions take and return user - time only ( ! ), they convert internaly everything to servertime , because
2005-11-09 00:15:14 +01:00
* SO operates only on server - time
2005-06-14 23:54:17 +02:00
*
* As this BO class deals with dates / times of several types and timezone , each variable should have a postfix
* appended , telling with type it is : _s = seconds , _su = secs in user - time , _ss = secs in server - time , _h = hours
*
* All new BO code ( should be true for eGW in general ) NEVER use any $_REQUEST ( $_POST or $_GET ) vars itself .
2005-11-09 00:15:14 +01:00
* Nor does it store the state of any UI - elements ( eg . cat - id selectbox ) . All this is the task of the UI class ( es ) !!!
2005-06-14 23:54:17 +02:00
*
2005-11-09 00:15:14 +01:00
* All permanent debug messages of the calendar - code should done via the debug - message method of the bocal class ! ! !
2005-06-14 23:54:17 +02:00
*/
2008-06-07 19:45:33 +02:00
class calendar_boupdate extends calendar_bo
2005-06-14 23:54:17 +02:00
{
2009-12-03 19:24:19 +01:00
/**
* Category ACL allowing to add a given category
*/
const CAT_ACL_ADD = 512 ;
/**
* Category ACL allowing to change status of a participant
*/
const CAT_ACL_STATUS = 1024 ;
2006-10-02 07:36:55 +02:00
/**
* name of method to debug or level of debug - messages :
* False = Off as higher as more messages you get ; - )
* 1 = function - calls incl . parameters to general functions like search , read , write , delete
* 2 = function - calls to exported helper - functions like check_perms
* 4 = function - calls to exported conversation - functions like date2ts , date2array , ...
* 5 = function - calls to private functions
* @ var mixed
*/
var $debug ;
2008-05-08 00:12:25 +02:00
2005-11-09 00:15:14 +01:00
/**
2010-01-29 22:42:54 +01:00
* Set Logging
*
* @ var boolean
2005-11-09 00:15:14 +01:00
*/
2010-01-29 22:42:54 +01:00
var $log = false ;
2010-03-09 18:03:41 +01:00
var $logfile = '/tmp/log-calendar-boupdate' ;
2005-11-09 00:15:14 +01:00
2010-02-09 22:56:39 +01:00
/**
* Cached timezone data
*
* @ var array id => data
*/
protected static $tz_cache = array ();
2018-06-27 20:22:04 +02:00
2005-06-14 23:54:17 +02:00
/**
* Constructor
*/
2008-06-07 19:45:33 +02:00
function __construct ()
2005-06-14 23:54:17 +02:00
{
2009-08-04 19:14:16 +02:00
if ( $this -> debug > 0 ) $this -> debug_message ( 'calendar_boupdate::__construct() started' , True );
2005-06-14 23:54:17 +02:00
2008-06-07 19:45:33 +02:00
parent :: __construct (); // calling the parent constructor
2018-06-27 20:22:04 +02:00
2009-08-04 19:14:16 +02:00
if ( $this -> debug > 0 ) $this -> debug_message ( 'calendar_boupdate::__construct() finished' , True );
2005-06-14 23:54:17 +02:00
}
/**
2022-06-03 09:57:33 +02:00
* updates or creates an event , it ( optionally ) checks for conflicts and sends the necessary notifications
2005-06-14 23:54:17 +02:00
*
* @ param array & $event event - array , on return some values might be changed due to set defaults
2015-06-25 22:39:53 +02:00
* @ param boolean $ignore_conflicts = false just ignore conflicts or do a conflict check and return the conflicting events
* @ param boolean $touch_modified = true NOT USED ANYMORE ( was only used in old csv - import ), modified & modifier is always updated !
* @ param boolean $ignore_acl = false should we ignore the acl
2017-08-18 09:49:02 +02:00
* @ param boolean $updateTS = true update the content history of the event ( ignored , as updating timestamps is required for sync ! )
2009-12-03 19:24:19 +01:00
* @ param array & $messages = null messages about because of missing ACL removed participants or categories
2022-06-03 09:57:33 +02:00
* @ param boolean | " NOPUSH " $skip_notification = false true : send NO notifications , default false = send them ,
* " NOPUSH " : also do NOT send push notifications / call Link :: notifiy (), which still happens for true
2005-06-14 23:54:17 +02:00
* @ return mixed on success : int $cal_id > 0 , on error false or array with conflicting events ( only if $check_conflicts )
2022-06-03 09:57:33 +02:00
* Please note : the events are not guarantied to be readable by the user ( no read grant or private ) !
2010-11-11 09:51:13 +01:00
*
* @ ToDo current conflict checking code does NOT cope quantity - wise correct with multiple non - overlapping
* events overlapping the event to store : the quantity sum is used , even as the events dont overlap !
*
* ++++++++ ++++++++
* + + + B + If A get checked for conflicts , the check used for resource quantity is
* + + ++++++++
* + A + quantity ( A , resource ) + quantity ( B , resource ) + quantity ( C , resource ) > maxBookable ( resource )
* + + ++++++++
* + + + C + which is clearly wrong for everything with a maximum quantity > 1
* ++++++++ ++++++++
2005-06-14 23:54:17 +02:00
*/
2011-03-09 18:26:20 +01:00
function update ( & $event , $ignore_conflicts = false , $touch_modified = true , $ignore_acl = false , $updateTS = true , & $messages = null , $skip_notification = false )
2005-06-14 23:54:17 +02:00
{
2020-07-14 21:39:45 +02:00
$update_type = 'edit' ;
2017-08-18 09:49:02 +02:00
unset ( $updateTS ); // ignored, as updating timestamps is required for sync!
2008-05-17 15:00:34 +02:00
//error_log(__METHOD__."(".array2string($event).",$ignore_conflicts,$touch_modified,$ignore_acl)");
2012-04-24 16:37:32 +02:00
if ( ! is_array ( $messages )) $messages = $messages ? ( array ) $messages : array ();
2010-11-11 09:51:13 +01:00
2005-11-09 00:15:14 +01:00
if ( $this -> debug > 1 || $this -> debug == 'update' )
{
2009-08-04 19:14:16 +02:00
$this -> debug_message ( 'calendar_boupdate::update(%1,ignore_conflict=%2,touch_modified=%3,ignore_acl=%4)' ,
2005-11-09 00:15:14 +01:00
false , $event , $ignore_conflicts , $touch_modified , $ignore_acl );
}
2005-06-14 23:54:17 +02:00
// check some minimum requirements:
// - new events need start, end and title
// - updated events cant set start, end or title to empty
2023-06-09 18:27:28 +02:00
if ( empty ( $event [ 'id' ]) && ( ! $event [ 'start' ] || ! $event [ 'end' ] || ! $event [ 'title' ]) ||
! empty ( $event [ 'id' ]) && ( isset ( $event [ 'start' ]) && ! $event [ 'start' ] || isset ( $event [ 'end' ]) && ! $event [ 'end' ] ||
2005-11-23 22:39:37 +01:00
isset ( $event [ 'title' ]) && ! $event [ 'title' ]))
2005-06-14 23:54:17 +02:00
{
2012-04-24 16:37:32 +02:00
$messages [] = lang ( 'Required information (start, end, title, ...) missing!' );
2005-06-14 23:54:17 +02:00
return false ;
}
2008-01-17 16:39:22 +01:00
2010-06-23 16:52:55 +02:00
$status_reset_to_unknown = false ;
2010-11-11 09:51:13 +01:00
2023-06-09 18:27:28 +02:00
if (( $new_event = empty ( $event [ 'id' ]))) // some defaults for new entries
2005-06-14 23:54:17 +02:00
{
// if no owner given, set user to owner
if ( ! $event [ 'owner' ]) $event [ 'owner' ] = $this -> user ;
// set owner as participant if none is given
2009-03-20 09:17:54 +01:00
if ( ! is_array ( $event [ 'participants' ]) || ! count ( $event [ 'participants' ]))
2005-06-14 23:54:17 +02:00
{
2015-06-25 22:39:53 +02:00
$status = calendar_so :: combine_status ( $event [ 'owner' ] == $this -> user ? 'A' : 'U' , 1 , 'CHAIR' );
2009-11-26 19:36:19 +01:00
$event [ 'participants' ] = array ( $event [ 'owner' ] => $status );
2008-05-08 00:12:25 +02:00
}
2020-07-14 21:39:45 +02:00
$update_type = 'add' ;
2009-12-03 19:24:19 +01:00
}
2009-11-26 19:36:19 +01:00
2005-11-09 00:15:14 +01:00
// check if user has the permission to update / create the event
2016-05-01 19:47:59 +02:00
if ( ! $ignore_acl && ( ! $new_event && ! $this -> check_perms ( Acl :: EDIT , $event [ 'id' ]) ||
$new_event && ! $this -> check_perms ( Acl :: EDIT , 0 , $event [ 'owner' ])) &&
! $this -> check_perms ( Acl :: ADD , 0 , $event [ 'owner' ]))
2005-11-09 00:15:14 +01:00
{
2016-05-01 19:47:59 +02:00
$messages [] = lang ( 'Access to calendar of %1 denied!' , Api\Accounts :: username ( $event [ 'owner' ]));
2005-11-09 00:15:14 +01:00
return false ;
}
2009-11-26 19:36:19 +01:00
if ( $new_event )
{
2010-03-16 10:26:01 +01:00
$old_event = array ();
2009-11-26 19:36:19 +01:00
}
else
2009-11-19 19:56:04 +01:00
{
$old_event = $this -> read (( int ) $event [ 'id' ], null , $ignore_acl );
}
2009-11-26 19:36:19 +01:00
2009-11-19 19:56:04 +01:00
// do we need to check, if user is allowed to invite the invited participants
if ( $this -> require_acl_invite && ( $removed = $this -> remove_no_acl_invite ( $event , $old_event )))
{
2009-12-03 19:24:19 +01:00
// report removed participants back to user
foreach ( $removed as $key => $account_id )
{
$removed [ $key ] = $this -> participant_name ( $account_id );
}
$messages [] = lang ( '%1 participants removed because of missing invite grants' , count ( $removed )) .
': ' . implode ( ', ' , $removed );
}
// check category based ACL
2023-06-09 18:27:28 +02:00
if ( ! empty ( $event [ 'category' ]))
2009-12-03 19:24:19 +01:00
{
if ( ! is_array ( $event [ 'category' ])) $event [ 'category' ] = explode ( ',' , $event [ 'category' ]);
2010-03-16 10:26:01 +01:00
if ( ! $old_event || ! isset ( $old_event [ 'category' ]))
2009-12-03 19:24:19 +01:00
{
2010-03-16 10:26:01 +01:00
$old_event [ 'category' ] = array ();
}
elseif ( ! is_array ( $old_event [ 'category' ]))
{
$old_event [ 'category' ] = explode ( ',' , $old_event [ 'category' ]);
2009-12-03 19:24:19 +01:00
}
foreach ( $event [ 'category' ] as $key => $cat_id )
{
// check if user is allowed to update event categories
2009-12-07 14:31:51 +01:00
if (( ! $old_event || ! in_array ( $cat_id , $old_event [ 'category' ])) &&
self :: has_cat_right ( self :: CAT_ACL_ADD , $cat_id , $this -> user ) === false )
2009-12-03 19:24:19 +01:00
{
unset ( $event [ 'category' ][ $key ]);
// report removed category to user
$removed_cats [ $cat_id ] = $this -> categories -> id2name ( $cat_id );
2009-12-07 14:31:51 +01:00
continue ; // no further check, as cat was removed
2009-12-03 19:24:19 +01:00
}
// for new or moved events check status of participants, if no category status right --> set all status to 'U' = unknown
2009-12-07 14:31:51 +01:00
if ( ! $status_reset_to_unknown &&
self :: has_cat_right ( self :: CAT_ACL_STATUS , $cat_id , $this -> user ) === false &&
2009-12-03 19:24:19 +01:00
( ! $old_event || $old_event [ 'start' ] != $event [ 'start' ] || $old_event [ 'end' ] != $event [ 'end' ]))
{
2010-03-07 00:06:43 +01:00
foreach (( array ) $event [ 'participants' ] as $uid => $status )
2009-12-03 19:24:19 +01:00
{
2015-06-25 22:39:53 +02:00
$q = $r = null ;
2009-12-03 19:24:19 +01:00
calendar_so :: split_status ( $status , $q , $r );
if ( $status != 'U' )
{
$event [ 'participants' ][ $uid ] = calendar_so :: combine_status ( 'U' , $q , $r );
// todo: report reset status to user
}
}
$status_reset_to_unknown = true ; // once is enough
}
}
if ( $removed_cats )
{
$messages [] = lang ( 'Category %1 removed because of missing rights' , implode ( ', ' , $removed_cats ));
}
if ( $status_reset_to_unknown )
{
$messages [] = lang ( 'Status of participants set to unknown because of missing category rights' );
}
2009-11-19 19:56:04 +01:00
}
2020-11-26 12:10:59 +01:00
// generate a video-room-url, if we need one and not already have one
2023-06-09 18:27:28 +02:00
if ( ! empty ( $event [ 'videoconference' ]) && empty ( $event [ '##videoconference' ]) && ! calendar_hooks :: isVideoconferenceDisabled ())
2020-11-26 12:10:59 +01:00
{
$event [ '##videoconference' ] = EGroupware\Status\Videoconference\Call :: genUniqueRoomID ();
}
elseif ( isset ( $event [ 'videoconference' ]) && ! $event [ 'videoconference' ])
{
$event [ '##videoconference' ] = '' ;
}
// update videoconference resource amounts based on number of participants
2023-06-09 18:27:28 +02:00
if ( ! empty ( $event [ 'videoconference' ]) && ! empty ( $event [ '##videoconference' ]) && ! calendar_hooks :: isVideoconferenceDisabled ()
2021-05-17 12:42:24 +02:00
&& ( $videoconferenceResId = \EGroupware\Status\Hooks :: getVideoconferenceResourceId ()))
2020-11-26 12:10:59 +01:00
{
2021-01-15 17:44:28 +01:00
$participant_total = 0 ;
foreach ([ 'u' , 'e' , 'c' ] as $p_type )
{
if ( is_array ( $event [ 'participant_types' ][ $p_type ]))
{
$participant_total += count ( $event [ 'participant_types' ][ $p_type ]);
}
}
2020-11-26 12:10:59 +01:00
$event [ 'participant_types' ][ 'r' ][ $videoconferenceResId ] =
2021-01-15 17:44:28 +01:00
$event [ 'participants' ][ 'r' . $videoconferenceResId ] = 'A' . $participant_total ;
2020-11-26 12:10:59 +01:00
}
2005-06-14 23:54:17 +02:00
// check for conflicts only happens !$ignore_conflicts AND if start + end date are given
2016-07-06 12:45:22 +02:00
$checked_excluding = null ;
if ( ! $ignore_conflicts && ! $event [ 'non_blocking' ] && isset ( $event [ 'start' ]) && isset ( $event [ 'end' ]) &&
(( $conflicts = $this -> conflicts ( $event , $checked_excluding )) || $checked_excluding ))
2005-06-14 23:54:17 +02:00
{
2016-07-06 12:45:22 +02:00
if ( $checked_excluding ) // warn user if not all recurrences have been checked
2005-06-14 23:54:17 +02:00
{
2016-07-06 12:45:22 +02:00
$conflicts [ 'warning' ] = array (
'start' => $checked_excluding ,
'title' => lang ( 'Only recurrences until %1 (excluding) have been checked!' , $checked_excluding -> format ( true )),
);
2005-06-14 23:54:17 +02:00
}
2016-07-06 12:45:22 +02:00
return $conflicts ;
}
2020-11-26 12:10:59 +01:00
2020-04-03 10:28:34 +02:00
2016-07-06 12:45:22 +02:00
//echo "saving $event[id]="; _debug_array($event);
$event2save = $event ;
2017-01-18 18:29:44 +01:00
if ( ! ( $cal_id = $this -> save ( $event , $ignore_acl )))
2016-07-06 12:45:22 +02:00
{
return $cal_id ;
}
2016-09-15 18:45:45 +02:00
// we re-read the event, in case only partial information was update and we need the full info for the notifies
// The check for new private events is to at least show the private version,
// otherwise we get FALSE
$event = $this -> read ( $cal_id , null , $ignore_acl , 'ts' , $new_event && ! $event [ 'public' ] ? $this -> user : null );
//error_log("new $cal_id=". array2string($event));
2016-07-06 12:45:22 +02:00
2023-06-09 18:27:28 +02:00
if ( ! empty ( $old_event [ 'deleted' ]) && ! isset ( $event [ 'deleted' ]))
2016-07-06 12:45:22 +02:00
{
// Restored, bring back links
Link :: restore ( 'calendar' , $cal_id );
2020-07-14 21:39:45 +02:00
$update_type = 'add' ;
2016-07-06 12:45:22 +02:00
}
2023-06-09 18:27:28 +02:00
if ( ! empty ( $this -> log_file ))
2016-07-06 12:45:22 +02:00
{
$this -> log2file ( $event2save , $event , $old_event );
}
2019-06-03 19:28:39 +02:00
// send notifications if the event is in the future
2024-07-02 15:13:42 +02:00
if ( ! $skip_notification && $this -> eventInFuture ( $event ))
2016-07-06 12:45:22 +02:00
{
if ( $new_event )
2005-06-14 23:54:17 +02:00
{
2023-06-15 16:15:10 +02:00
$this -> send_update ( MSG_ADDED , $event [ 'participants' ], null , $event );
2016-07-06 12:45:22 +02:00
}
else // update existing event
{
$this -> check4update ( $event , $old_event );
}
}
2008-05-08 00:12:25 +02:00
2020-07-14 21:39:45 +02:00
// notify the link-class about the update, as other apps may be subscribed to it
2022-06-03 09:57:33 +02:00
if ( $skip_notification !== " NOPUSH " )
{
Link :: notify_update ( 'calendar' , $cal_id , $event , $update_type );
}
2016-07-06 12:45:22 +02:00
return $cal_id ;
}
/**
* Check given event for conflicts and return them
*
* For recurring events we check a configurable fixed number of recurrences
2016-07-06 16:44:06 +02:00
* and for a fixed maximum time ( default 3 s ) .
*
* Conflict check skips past events / recurrences and is always limited by recurrence horizont ,
* as it would only report non - recurring events after .
2016-07-06 12:45:22 +02:00
*
* @ param array $event
* @ param Api\DateTime & $checked_excluding = null time until which ( excluding ) recurrences have been checked
* @ return array or events
*/
function conflicts ( array $event , & $checked_excluding = null )
{
$types_with_quantity = array ();
foreach ( $this -> resources as $type => $data )
{
if ( $data [ 'max_quantity' ]) $types_with_quantity [] = $type ;
}
// get all NOT rejected participants and evtl. their quantity
$quantity = $users = array ();
foreach ( $event [ 'participants' ] as $uid => $status )
{
2017-08-18 09:49:02 +02:00
$q = $role = null ;
calendar_so :: split_status ( $status , $q , $role );
if ( $status == 'R' || $role == 'NON-PARTICIPANT' ) continue ; // ignore rejected or non-participants
2016-07-06 12:45:22 +02:00
if ( $uid < 0 ) // group, check it's members too
{
$users = array_unique ( array_merge ( $users , ( array ) $GLOBALS [ 'egw' ] -> accounts -> members ( $uid , true )));
}
$users [] = $uid ;
if ( in_array ( $uid [ 0 ], $types_with_quantity ))
{
$quantity [ $uid ] = $q ;
}
}
$max_quantity = $possible_quantity_conflicts = $conflicts = array ();
if ( $event [ 'recur_type' ])
{
$recurences = calendar_rrule :: event2rrule ( $event );
}
else
{
$recurences = array ( new Api\DateTime (( int ) $event [ 'start' ]));
}
$checked_excluding = null ;
$max_checked = $GLOBALS [ 'egw_info' ][ 'server' ][ 'conflict_max_checked' ];
if (( $max_check_time = ( float ) $GLOBALS [ 'egw_info' ][ 'server' ][ 'conflict_max_check_time' ]) < 1.0 )
{
$max_check_time = 3.0 ;
}
$checked = 0 ;
$start = microtime ( true );
2021-11-09 08:39:21 +01:00
$duration = Api\DateTime :: to ( $event [ 'end' ], 'ts' ) - Api\DateTime :: to ( $event [ 'start' ], 'ts' );
2024-08-06 08:54:12 +02:00
// as floating, whole-day events are defined as whole-day in user timezone (in fact in every TZ!), but they are
// always stored in server-timezone, we have to account for the difference, to not miss conflicts!
$start_tz_diff = Api\DateTime :: $server_timezone -> getOffset ( new Api\DateTime ( $event [ 'start' ])) -
Api\DateTime :: $user_timezone -> getOffset ( new Api\DateTime ( $event [ 'start' ]));
// might be different from $start_tz_diff, if TZ transition is during the event
$end_tz_diff = Api\DateTime :: $server_timezone -> getOffset ( new Api\DateTime ( $event [ 'end' ])) -
Api\DateTime :: $user_timezone -> getOffset ( new Api\DateTime ( $event [ 'end' ]));
2016-07-06 12:45:22 +02:00
foreach ( $recurences as $date )
{
$startts = $date -> format ( 'ts' );
2016-07-06 16:44:06 +02:00
// skip past events or recurrences
if ( $startts + $duration < $this -> now_su ) continue ;
2016-07-06 12:45:22 +02:00
// abort check if configured limits are exceeded
if ( $event [ 'recur_type' ] &&
2016-07-06 16:44:06 +02:00
( ++ $checked > $max_checked && $max_checked > 0 || // maximum number of checked recurrences exceeded
2016-07-06 12:45:22 +02:00
microtime ( true ) > $start + $max_check_time || // max check time exceeded
2016-07-06 16:44:06 +02:00
$startts > $this -> config [ 'horizont' ])) // we are behind horizon for which recurrences are rendered
2016-07-06 12:45:22 +02:00
{
if ( $this -> debug > 2 || $this -> debug == 'conflicts' )
2005-06-14 23:54:17 +02:00
{
2016-07-06 12:45:22 +02:00
$this -> debug_message ( __METHOD__ . '() conflict check limited to %1 recurrences, %2 seconds, until (excluding) %3' ,
$checked , microtime ( true ) - $start , $date );
2005-06-14 23:54:17 +02:00
}
2016-07-06 12:45:22 +02:00
$checked_excluding = $date ;
break ;
2005-06-14 23:54:17 +02:00
}
$overlapping_events =& $this -> search ( array (
2024-08-06 08:54:12 +02:00
'start' => $startts - abs ( $start_tz_diff ),
'end' => $startts + $duration + abs ( $end_tz_diff ),
2005-06-14 23:54:17 +02:00
'users' => $users ,
2005-11-09 00:15:14 +01:00
'ignore_acl' => true , // otherwise we get only events readable by the user
2006-03-16 19:03:43 +01:00
'enum_groups' => true , // otherwise group-events would not block time
2015-01-14 20:41:01 +01:00
'query' => array (
'cal_non_blocking' => 0 ,
),
2015-01-19 20:32:26 +01:00
'no_integration' => true , // do NOT use integration of other apps
2005-06-14 23:54:17 +02:00
));
2016-07-06 12:45:22 +02:00
if ( $this -> debug > 2 || $this -> debug == 'conflicts' )
2005-11-09 00:15:14 +01:00
{
2016-07-06 12:45:22 +02:00
$this -> debug_message ( __METHOD__ . '() checking for potential overlapping events for users %1 from %2 to %3' , false , $users , $startts , $startts + $duration );
2005-11-09 00:15:14 +01:00
}
2005-06-14 23:54:17 +02:00
foreach (( array ) $overlapping_events as $k => $overlap )
{
2005-11-09 00:15:14 +01:00
if ( $overlap [ 'id' ] == $event [ 'id' ] || // that's the event itself
2005-11-23 22:39:37 +01:00
$overlap [ 'id' ] == $event [ 'reference' ] || // event is an exception of overlap
2005-11-09 00:15:14 +01:00
$overlap [ 'non_blocking' ]) // that's a non_blocking event
{
continue ;
}
2024-08-06 08:54:12 +02:00
// check if event is really blocking, as whole-day events use floating time and might NOT block in user's timezone
if ( $overlap [ 'end' ] <= $startts || $overlap [ 'start' ] >= $startts + $duration )
{
continue ;
}
2016-07-06 12:45:22 +02:00
if ( $this -> debug > 3 || $this -> debug == 'conflicts' )
2005-11-09 00:15:14 +01:00
{
2016-07-06 12:45:22 +02:00
$this -> debug_message ( __METHOD__ . '() checking overlapping event %1' , false , $overlap );
2005-11-09 00:15:14 +01:00
}
2005-06-14 23:54:17 +02:00
// check if the overlap is with a rejected participant or within the allowed quantity
$common_parts = array_intersect ( $users , array_keys ( $overlap [ 'participants' ]));
foreach ( $common_parts as $n => $uid )
{
2010-11-11 09:51:13 +01:00
$status = $overlap [ 'participants' ][ $uid ];
2017-08-18 09:49:02 +02:00
calendar_so :: split_status ( $status , $q , $role );
if ( $status == 'R' || $role == 'NON-PARTICIPANT' )
2005-06-14 23:54:17 +02:00
{
2010-03-25 21:23:22 +01:00
unset ( $common_parts [ $n ]);
2005-06-14 23:54:17 +02:00
continue ;
}
2008-05-08 17:02:35 +02:00
if ( is_numeric ( $uid ) || ! in_array ( $uid [ 0 ], $types_with_quantity ))
2005-06-14 23:54:17 +02:00
{
continue ; // no quantity check: quantity allways 1 ==> conflict
}
if ( ! isset ( $max_quantity [ $uid ]))
{
2006-03-05 11:26:03 +01:00
$res_info = $this -> resource_info ( $uid );
2008-05-08 17:02:35 +02:00
$max_quantity [ $uid ] = $res_info [ $this -> resources [ $uid [ 0 ]][ 'max_quantity' ]];
2005-06-14 23:54:17 +02:00
}
2010-11-11 09:51:13 +01:00
$quantity [ $uid ] += $q ;
2006-03-05 11:26:03 +01:00
if ( $quantity [ $uid ] <= $max_quantity [ $uid ])
2005-06-14 23:54:17 +02:00
{
$possible_quantity_conflicts [ $uid ][] =& $overlapping_events [ $k ]; // an other event can give the conflict
2006-03-05 11:26:03 +01:00
unset ( $common_parts [ $n ]);
2005-06-14 23:54:17 +02:00
continue ;
}
// now we have a quantity conflict for $uid
}
2005-11-09 00:15:14 +01:00
if ( count ( $common_parts ))
2005-06-14 23:54:17 +02:00
{
2016-07-06 12:45:22 +02:00
if ( $this -> debug > 3 || $this -> debug == 'conflicts' )
2005-11-09 00:15:14 +01:00
{
2016-07-06 12:45:22 +02:00
$this -> debug_message ( __METHOD__ . '() conflicts with the following participants found %1' , false , $common_parts );
2005-11-09 00:15:14 +01:00
}
$conflicts [ $overlap [ 'id' ] . '-' . $this -> date2ts ( $overlap [ 'start' ])] =& $overlapping_events [ $k ];
2005-06-14 23:54:17 +02:00
}
}
2016-07-06 12:45:22 +02:00
}
//error_log(__METHOD__."() conflict check took ".number_format(microtime(true)-$start, 3).'s');
// check if we are withing the allowed quantity and if not add all events using that resource
// seems this function is doing very strange things, it gives empty conflicts
foreach ( $max_quantity as $uid => $max )
{
if ( $quantity [ $uid ] > $max )
2005-06-14 23:54:17 +02:00
{
2016-07-06 12:45:22 +02:00
foreach (( array ) $possible_quantity_conflicts [ $uid ] as $conflict )
2005-11-09 00:15:14 +01:00
{
2016-07-06 12:45:22 +02:00
$conflicts [ $conflict [ 'id' ] . '-' . $this -> date2ts ( $conflict [ 'start' ])] =& $possible_quantity_conflicts [ $k ];
2005-11-09 00:15:14 +01:00
}
2008-05-08 00:12:25 +02:00
}
2005-06-14 23:54:17 +02:00
}
2016-07-06 12:45:22 +02:00
unset ( $possible_quantity_conflicts );
2008-01-17 16:39:22 +01:00
2016-07-06 12:45:22 +02:00
if ( count ( $conflicts ))
2005-11-09 00:15:14 +01:00
{
2016-07-06 12:45:22 +02:00
foreach ( $conflicts as $key => $conflict )
2011-03-09 18:26:20 +01:00
{
2016-07-06 12:45:22 +02:00
$conflict [ 'participants' ] = array_intersect_key (( array ) $conflict [ 'participants' ], $event [ 'participants' ]);
if ( ! $this -> check_perms ( Acl :: READ , $conflict ))
{
$conflicts [ $key ] = array (
'id' => $conflict [ 'id' ],
'title' => lang ( 'busy' ),
'participants' => $conflict [ 'participants' ],
'start' => $conflict [ 'start' ],
'end' => $conflict [ 'end' ],
);
}
2011-03-09 18:26:20 +01:00
}
2016-07-06 12:45:22 +02:00
if ( $this -> debug > 2 || $this -> debug == 'conflicts' )
2011-03-09 18:26:20 +01:00
{
2016-07-06 12:45:22 +02:00
$this -> debug_message ( __METHOD__ . '() %1 conflicts found %2' , false , count ( $conflicts ), $conflicts );
2011-03-09 18:26:20 +01:00
}
2005-11-09 00:15:14 +01:00
}
2016-07-06 12:45:22 +02:00
return $conflicts ;
2005-06-14 23:54:17 +02:00
}
2009-11-19 19:56:04 +01:00
/**
* Remove participants current user has no right to invite
*
* @ param array & $event new event
2015-06-25 22:39:53 +02:00
* @ param array $old_event = null old event with already invited participants
2009-11-19 19:56:04 +01:00
* @ return array removed participants because of missing invite grants
*/
public function remove_no_acl_invite ( array & $event , array $old_event = null )
{
if ( ! $this -> require_acl_invite )
{
return array (); // nothing to check, everyone can invite everyone else
}
if ( $event [ 'id' ] && is_null ( $old_event ))
{
$old_event = $this -> read ( $event [ 'id' ]);
}
$removed = array ();
2015-06-25 22:39:53 +02:00
foreach ( array_keys (( array ) $event [ 'participants' ]) as $uid )
2009-11-19 19:56:04 +01:00
{
2009-12-07 14:31:51 +01:00
if (( is_null ( $old_event ) || ! isset ( $old_event [ 'participants' ][ $uid ])) && ! $this -> check_acl_invite ( $uid ))
2009-11-19 19:56:04 +01:00
{
unset ( $event [ 'participants' ][ $uid ]); // remove participant
$removed [] = $uid ;
}
}
2009-11-24 14:13:41 +01:00
//echo "<p>".__METHOD__."($event[title],".($old_event?'$old_event':'NULL').") returning ".array2string($removed)."</p>";
2009-11-19 19:56:04 +01:00
return $removed ;
}
/**
* Check if current user is allowed to invite a given participant
*
* @ param int | string $uid
* @ return boolean
*/
public function check_acl_invite ( $uid )
{
2018-10-01 18:25:55 +02:00
if ( ! is_numeric ( $uid ) && $this -> resources [ $uid [ 0 ]][ 'check_invite' ])
2018-03-19 16:58:44 +01:00
{
2018-10-01 18:25:55 +02:00
// Resource specific ACL check
return call_user_func ( $this -> resources [ $uid [ 0 ]][ 'check_invite' ], $uid );
2018-10-09 18:09:20 +02:00
}
2018-03-19 16:58:44 +01:00
elseif ( ! $this -> require_acl_invite )
2009-11-24 14:13:41 +01:00
{
$ret = true ; // no grant required
}
2009-12-07 14:31:51 +01:00
elseif ( $this -> require_acl_invite == 'groups' && $GLOBALS [ 'egw' ] -> accounts -> get_type ( $uid ) != 'g' )
2009-11-24 14:13:41 +01:00
{
$ret = true ; // grant only required for groups
}
else
2009-11-19 19:56:04 +01:00
{
2016-05-01 19:47:59 +02:00
$ret = $this -> check_perms ( self :: ACL_INVITE , 0 , $uid );
2009-11-19 19:56:04 +01:00
}
2009-11-24 14:13:41 +01:00
//error_log(__METHOD__."($uid) = ".array2string($ret));
2009-12-07 14:31:51 +01:00
//echo "<p>".__METHOD__."($uid) require_acl_invite=$this->require_acl_invite returning ".array2string($ret)."</p>\n";
2009-11-24 14:13:41 +01:00
return $ret ;
2009-11-19 19:56:04 +01:00
}
2005-11-09 00:15:14 +01:00
/**
2009-03-20 09:17:54 +01:00
* Check for added , modified or deleted participants AND notify them
2005-11-09 00:15:14 +01:00
*
* @ param array $new_event the updated event
* @ param array $old_event the event before the update
2024-07-09 21:39:40 +02:00
* @ param int $notify_max_recurrences notify only about this number of single recurrences at maximum
2008-05-08 00:12:25 +02:00
*/
2024-07-09 21:39:40 +02:00
function check4update ( $new_event , $old_event , int $notify_max_recurrences = 3 )
2005-11-09 00:15:14 +01:00
{
$modified = $added = $deleted = array ();
2008-05-08 00:12:25 +02:00
2009-08-04 19:14:16 +02:00
//echo "<p>calendar_boupdate::check4update() new participants = ".print_r($new_event['participants'],true).", old participants =".print_r($old_event['participants'],true)."</p>\n";
2024-07-04 16:22:06 +02:00
foreach ([ 'start' , 'end' , 'tz_id' , 'owner' , 'category' , 'priority' , 'public' , 'title' , 'description' , 'location' ,
'recur_exception' , 'recur_rdates' , 'recur_enddate' ] as $field )
2021-05-05 22:44:14 +02:00
{
if ( $new_event [ $field ] !== $old_event [ $field ])
{
2024-07-09 21:39:40 +02:00
$n = 0 ;
switch ( $field )
{
case 'recur_exception' :
foreach ( $added_exceptions = array_diff ( $new_event [ $field ] ? ? [], $old_event [ $field ] ? ? []) as $key => $recurrence )
{
if ( $this -> read ( $new_event [ 'uid' ], $recurrence ))
{
unset ( $added_exceptions [ $key ]);
continue ; // explicit exception --> no need to notify, already done for the exception itself
}
// send a cancel for each RECURRENCE-ID
$this -> send_update ( MSG_DELETED , $new_event [ 'participants' ], [
'recurrence' => $recurrence ,
'start' => $recurrence ,
'end' => $recurrence + $new_event [ 'end' ] - $new_event [ 'start' ],
] + $new_event );
// limit number of notifications to the first N recurrences
if ( ++ $n >= $notify_max_recurrences ) break ;
}
$n = 0 ;
foreach ( $removed_exceptions = array_diff ( $old_event [ $field ] ? ? [], $new_event [ $field ] ? ? []) as $key => $recurrence )
{
if ( $this -> read ( $new_event [ 'uid' ], $recurrence ))
{
unset ( $removed_exceptions [ $key ]);
continue ; // explicit exception --> no need to notify, already done for the exception itself
}
// send an add for each RECURRENCE-ID
$this -> send_update ( MSG_ADDED , $new_event [ 'participants' ], null , [
'recurrence' => $recurrence ,
'start' => $recurrence ,
'end' => $recurrence + $new_event [ 'end' ] - $new_event [ 'start' ],
] + $new_event );
// limit number of notifications to the first N recurrences
if ( ++ $n >= $notify_max_recurrences ) break ;
}
continue 2 ;
case 'recur_rdates' :
foreach ( array_diff ( $new_event [ $field ] ? ? [], $old_event [ $field ] ? ? []) as $recurrence )
{
// send an add for each RECURRENCE-ID
$this -> send_update ( MSG_ADDED , $new_event [ 'participants' ], null , [
'recurrence' => $recurrence ,
'start' => $recurrence ,
'end' => $recurrence + $new_event [ 'end' ] - $new_event [ 'start' ],
] + $new_event );
// limit number of notifications to the first N recurrences
if ( ++ $n >= $notify_max_recurrences ) break ;
}
continue 2 ;
case 'recur_enddate' :
if ( $new_event [ 'recur_type' ] == MCAL_RECUR_RDATE ) continue 2 ; // already handled above
break ;
}
2021-12-08 18:59:18 +01:00
$modified = $new_event [ 'participants' ];
2021-05-05 22:44:14 +02:00
break ;
}
}
2005-11-09 00:15:14 +01:00
// Find modified and deleted participants ...
2020-03-11 16:25:24 +01:00
foreach (( array ) $old_event [ 'participants' ] as $old_userid => $old_status )
2005-11-09 00:15:14 +01:00
{
2021-05-05 22:44:14 +02:00
if ( isset ( $new_event [ 'participants' ][ $old_userid ]) && $old_status !== $new_event [ 'participants' ][ $old_userid ])
2005-11-09 00:15:14 +01:00
{
$modified [ $old_userid ] = $new_event [ 'participants' ][ $old_userid ];
}
2021-05-05 22:44:14 +02:00
else if ( ! isset ( $new_event [ 'participants' ][ $old_userid ]))
2005-11-09 00:15:14 +01:00
{
$deleted [ $old_userid ] = $old_status ;
}
}
2008-07-11 10:17:22 +02:00
// Find new participants ...
2015-06-25 22:39:53 +02:00
foreach ( array_keys (( array ) $new_event [ 'participants' ]) as $new_userid )
2005-11-09 00:15:14 +01:00
{
if ( ! isset ( $old_event [ 'participants' ][ $new_userid ]))
{
$added [ $new_userid ] = 'U' ;
}
}
2009-08-04 19:14:16 +02:00
//echo "<p>calendar_boupdate::check4update() added=".print_r($added,true).", modified=".print_r($modified,true).", deleted=".print_r($deleted,true)."</p>\n";
2005-11-09 00:15:14 +01:00
if ( count ( $added ) || count ( $modified ) || count ( $deleted ))
{
if ( count ( $added ))
{
$this -> send_update ( MSG_ADDED , $added , $old_event , $new_event );
}
if ( count ( $modified ))
{
$this -> send_update ( MSG_MODIFIED , $modified , $old_event , $new_event );
}
if ( count ( $deleted ))
{
$this -> send_update ( MSG_DISINVITE , $deleted , $new_event );
}
}
}
/**
* checks if $userid has requested ( in $part_prefs ) updates for $msg_type
*
* @ param int $userid numerical user - id
2024-07-04 16:22:06 +02:00
* @ param array $part_prefs preferences of the user $userid
2014-02-20 17:14:32 +01:00
* @ param int & $msg_type type of the notification : MSG_ADDED , MSG_MODIFIED , MSG_ACCEPTED , ...
2005-11-09 00:15:14 +01:00
* @ param array $old_event Event before the change
* @ param array $new_event Event after the change
2012-01-05 06:47:02 +01:00
* @ param string $role we treat CHAIR like event owners
2014-02-20 17:14:32 +01:00
* @ param string $status of current user
2012-01-05 06:47:02 +01:00
* @ return boolean true = update requested , false otherwise
2005-11-09 00:15:14 +01:00
*/
2023-06-15 14:13:55 +02:00
public function update_requested ( $userid , $part_prefs , & $msg_type , $old_event , $new_event , $role , $status = null )
2005-11-09 00:15:14 +01:00
{
if ( $msg_type == MSG_ALARM )
{
return True ; // always True for now
}
$want_update = 0 ;
// the following switch falls through all cases, as each included the following too
//
2010-02-17 14:29:28 +01:00
$msg_is_response = $msg_type == MSG_REJECTED || $msg_type == MSG_ACCEPTED || $msg_type == MSG_TENTATIVE || $msg_type == MSG_DELEGATED ;
2005-11-09 00:15:14 +01:00
2019-11-04 18:51:39 +01:00
// Check if user is not participating, and does not want notifications
2023-06-15 16:15:10 +02:00
if ( $msg_is_response && is_array ( $old_event ) && empty ( $part_prefs [ 'calendar' ][ 'receive_not_participating' ]) &&
2023-06-15 14:13:55 +02:00
// userid is the email address for non-user and NOT necessary the uid used as key in participants ("rb@egroupware.org" vs "eRalf Becker <rb@egroupware.org>")
! in_array ( is_numeric ( $userid ) ? $userid : strtolower ( $userid ), array_map ( function ( $uid )
{
return is_numeric ( $uid ) ? $uid : strtolower ( $this -> resource_info ( $uid )[ 'email' ] ? ? '' );
}, array_keys ( $old_event [ 'participants' ])), false ))
2019-11-04 18:51:39 +01:00
{
2023-06-15 14:13:55 +02:00
error_log ( __METHOD__ . " (userid= $userid , receive_not_participating=' { $part_prefs [ 'calendar' ][ 'receive_not_participating' ] } ', msg_type= $msg_type , { participants: " . json_encode ( $old_event [ 'participants' ]) . " , ...}, role=' $role ') msg_is_response= $msg_is_response --> user $userid is NOT participating " );
2019-11-04 18:51:39 +01:00
return false ;
}
2018-06-26 15:57:09 +02:00
// always notify externals chairs
// EGroupware owner only get notified about responses, if pref is NOT "no"
if ( ! is_numeric ( $userid ) && $role == 'CHAIR' &&
( $msg_is_response || in_array ( $msg_type , array ( MSG_ADDED , MSG_DELETED ))))
2005-11-09 00:15:14 +01:00
{
2018-06-26 15:57:09 +02:00
switch ( $msg_type )
{
case MSG_DELETED : // treat deleting event as rejection to organizer
$msg_type = MSG_REJECTED ;
break ;
case MSG_ADDED : // new events use added, but organizer needs status
switch ( $status [ 0 ])
{
case 'A' : $msg_type = MSG_ACCEPTED ; break ;
case 'R' : $msg_type = MSG_REJECTED ; break ;
case 'T' : $msg_type = MSG_TENTATIVE ; break ;
case 'D' : $msg_type = MSG_DELEGATED ; break ;
}
break ;
}
++ $want_update ;
}
else
{
switch ( $ru = $part_prefs [ 'calendar' ][ 'receive_updates' ])
{
case 'responses' :
2005-11-09 00:15:14 +01:00
++ $want_update ;
2018-06-26 15:57:09 +02:00
case 'modifications' :
if ( ! $msg_is_response )
2012-07-02 10:14:27 +02:00
{
++ $want_update ;
}
2018-06-26 15:57:09 +02:00
case 'time_change_4h' :
case 'time_change' :
default :
if ( is_array ( $new_event ) && is_array ( $old_event ))
2014-02-20 17:14:32 +01:00
{
2024-02-08 10:24:39 +01:00
$diff = max ( abs ( self :: date2ts ( $old_event [ 'start' ] ? ? null ) - self :: date2ts ( $new_event [ 'start' ] ? ? null )),
abs ( self :: date2ts ( $old_event [ 'end' ] ? ? null ) - self :: date2ts ( $new_event [ 'end' ] ? ? null )));
2018-06-26 15:57:09 +02:00
$check = $ru == 'time_change_4h' ? 4 * 60 * 60 - 1 : 0 ;
2024-07-04 16:22:06 +02:00
if ( $msg_type == MSG_MODIFIED && ( $diff > $check ||
// also notify if recurrences where added or removed
$old_event [ 'recur_exception' ] != $new_event [ 'recur_exception' ] ||
$old_event [ 'recur_rdates' ] != $new_event [ 'recur_rdates' ] ||
$old_event [ 'recur_enddate' ] != $new_event [ 'recur_enddate' ]))
2018-06-26 15:57:09 +02:00
{
++ $want_update ;
}
2014-02-20 17:14:32 +01:00
}
2018-06-26 15:57:09 +02:00
case 'add_cancel' :
2024-02-08 10:24:39 +01:00
if ( $msg_is_response && is_array ( $old_event ) && ( isset ( $old_event [ 'owner' ]) && $old_event [ 'owner' ] == $userid || $role == 'CHAIR' ) ||
2018-06-26 15:57:09 +02:00
$msg_type == MSG_DELETED || $msg_type == MSG_ADDED || $msg_type == MSG_DISINVITE )
{
++ $want_update ;
}
break ;
2018-07-17 16:02:11 +02:00
case 'no' :
break ;
2018-06-26 15:57:09 +02:00
}
2005-11-09 00:15:14 +01:00
}
2023-06-15 16:15:10 +02:00
//error_log(__METHOD__."(userid=$userid, receive_updates='$ru', msg_type=$msg_type, ".json_encode($old_event).", ..., role='$role') msg_is_response=$msg_is_response --> want_update=$want_update");
2005-11-09 00:15:14 +01:00
return $want_update > 0 ;
}
2012-06-12 17:44:55 +02:00
/**
* Check calendar prefs , if a given user ( integer account_id ) or email ( user or externals ) should get notified
*
* @ param int | string $user_or_email
2015-06-25 22:39:53 +02:00
* @ param string $ical_method = 'REQUEST'
* @ param string $role = 'REQ-PARTICIPANT'
2020-04-03 10:28:34 +02:00
* @ param string $notify_externals = null event - specific overwrite , default preferences
2012-06-12 17:44:55 +02:00
* @ return boolean true if user requested to be notified , false if not
*/
2020-04-03 10:28:34 +02:00
static public function email_update_requested ( $user_or_email , $ical_method = 'REQUEST' , $role = 'REQ-PARTICIPANT' , $notify_externals = null )
2012-06-12 17:44:55 +02:00
{
// check if email is from a user
if ( is_numeric ( $user_or_email ))
{
$account_id = $user_or_email ;
}
else
{
2012-06-12 18:17:25 +02:00
$account_id = $GLOBALS [ 'egw' ] -> accounts -> name2id ( $user_or_email , 'account_email' );
2012-06-12 17:44:55 +02:00
}
if ( $account_id )
{
2016-05-01 19:47:59 +02:00
$pref_obj = new Api\Preferences ( $account_id );
2012-06-12 18:17:25 +02:00
$prefs = $pref_obj -> read_repository ();
2012-06-12 17:44:55 +02:00
}
else
{
$prefs = array (
'calendar' => array (
2020-04-03 10:28:34 +02:00
'receive_updates' => $notify_externals ? : $GLOBALS [ 'egw_info' ][ 'user' ][ 'preferences' ][ 'calendar' ][ 'notify_externals' ],
2012-06-12 17:44:55 +02:00
)
);
}
switch ( $ical_method )
{
default :
case 'REQUEST' :
$msg_type = MSG_ADDED ;
break ;
case 'REPLY' :
$msg_type = MSG_ACCEPTED ;
break ;
case 'CANCEL' :
$msg_type = MSG_DELETED ;
break ;
}
2024-02-01 09:32:18 +01:00
static $calendar_boupdate = null ;
if ( ! isset ( $calendar_boupdate )) $calendar_boupdate = new calendar_boupdate ();
$ret = $calendar_boupdate -> update_requested ( $account_id , $prefs , $msg_type , array (), array (), $role );
2012-06-12 17:44:55 +02:00
//error_log(__METHOD__."('$user_or_email', '$ical_method', '$role') account_id=$account_id --> updated_requested returned ".array2string($ret));
return $ret ;
}
2014-02-20 17:14:32 +01:00
/**
* Get iCal / iMip method from internal nummeric msg - type plus optional notification message and verbose name
*
* @ param int $msg_type see MSG_ * defines
* @ param string & $action = null on return verbose name
* @ param string & $msg = null on return notification message
*/
2019-11-20 23:16:16 +01:00
function msg_type2ical_method ( $msg_type , & $action = null , & $msg = null , $prefs = null )
2014-02-20 17:14:32 +01:00
{
switch ( $msg_type )
{
case MSG_DELETED :
$action = 'Canceled' ;
$pref = 'Canceled' ;
$method = 'CANCEL' ;
break ;
case MSG_MODIFIED :
$action = 'Modified' ;
$pref = 'Modified' ;
$method = 'REQUEST' ;
break ;
case MSG_DISINVITE :
$action = 'Disinvited' ;
$pref = 'Disinvited' ;
$method = 'CANCEL' ;
break ;
case MSG_ADDED :
$action = 'Added' ;
$pref = 'Added' ;
$method = 'REQUEST' ;
break ;
case MSG_REJECTED :
$action = 'Rejected' ;
$pref = 'Response' ;
$method = 'REPLY' ;
break ;
case MSG_TENTATIVE :
$action = 'Tentative' ;
$pref = 'Response' ;
$method = 'REPLY' ;
break ;
case MSG_ACCEPTED :
$action = 'Accepted' ;
$pref = 'Response' ;
$method = 'REPLY' ;
break ;
case MSG_DELEGATED :
$action = 'Delegated' ;
$pref = 'Response' ;
$method = 'REPLY' ;
break ;
case MSG_ALARM :
$action = 'Alarm' ;
$pref = 'Alarm' ;
break ;
2021-02-11 19:43:47 +01:00
case MSG_REQUEST :
$method = 'REQUEST' ;
break ;
2014-02-20 17:14:32 +01:00
default :
$method = 'PUBLISH' ;
}
2019-11-20 23:16:16 +01:00
if ( is_null ( $prefs ))
{
$prefs = $this -> cal_prefs ;
}
$msg = $prefs [ 'notify' . $pref ];
2014-02-20 17:14:32 +01:00
if ( empty ( $msg ))
{
2019-11-20 23:16:16 +01:00
$msg = $prefs [ 'notifyAdded' ]; // use a default
2014-02-20 17:14:32 +01:00
}
//error_log(__METHOD__."($msg_type) action='$action', $msg='$msg' returning '$method'");
2022-04-26 21:04:16 +02:00
return $method ? ? null ;
2014-02-20 17:14:32 +01:00
}
2005-11-09 00:15:14 +01:00
/**
* sends update - messages to certain participants of an event
*
* @ param int $msg_type type of the notification : MSG_ADDED , MSG_MODIFIED , MSG_ACCEPTED , ...
* @ param array $to_notify numerical user - ids as keys ( ! ) ( value is not used )
* @ param array $old_event Event before the change
2015-06-25 22:39:53 +02:00
* @ param array $new_event = null Event after the change
2020-04-03 10:28:34 +02:00
* @ param int | string $user = 0 User / participant who started the notify , default current user
2020-04-07 16:16:31 +02:00
* @ param array $alarm = null values for " offset " , " start " , etc .
2024-07-02 15:13:42 +02:00
* @ parqm boolean $ignore_prefs Ignore the user ' s preferences about when they want to be notified and send it
2007-11-22 09:29:16 +01:00
* @ return bool true / false
2005-11-09 00:15:14 +01:00
*/
2024-07-02 15:13:42 +02:00
function send_update ( $msg_type , $to_notify , $old_event , $new_event = null , $user = 0 , ? array $alarm = null , $ignore_prefs = false )
2020-07-26 15:04:22 +02:00
{
Api\Egw :: on_shutdown ([ $this , '_send_update' ], func_get_args ());
}
2024-07-02 15:13:42 +02:00
/**
* Check event is ( ending ) in the future
*
* @ param array $event
* @ param int $grace_time
* @ return bool
*/
public function eventInFuture ( array $event , int $grace_time = 10 ) : bool
{
if ( $event [ 'recur_type' ] != MCAL_RECUR_NONE )
{
return empty ( $event [ 'recur_enddate' ]) || $event [ 'recur_enddate' ] > $this -> now_su - $grace_time ;
}
return $event [ 'end' ] > $this -> now_su - $grace_time ;
}
2020-07-26 15:04:22 +02:00
/**
* sends update - messages to certain participants of an event
*
* @ param int $msg_type type of the notification : MSG_ADDED , MSG_MODIFIED , MSG_ACCEPTED , ...
* @ param array $to_notify numerical user - ids as keys ( ! ) ( value is not used )
* @ param array $old_event Event before the change
* @ param array $new_event = null Event after the change
* @ param int | string $user = 0 User / participant who started the notify , default current user
* @ param array $alarm = null values for " offset " , " start " , etc .
2021-02-09 21:42:51 +01:00
* @ parqm boolean $ignore_prefs Ignore the user ' s preferences about when they want to be notified and send it
2020-07-26 15:04:22 +02:00
* @ return bool true / false
*/
2024-07-02 15:13:42 +02:00
function _send_update ( $msg_type , $to_notify , $old_event , $new_event = null , $user = 0 , ? array $alarm = null , $ignore_prefs = false )
2005-11-09 00:15:14 +01:00
{
2024-07-06 19:39:21 +02:00
try {
//error_log(__METHOD__."($msg_type, ".json_encode($to_notify).", ..., ".json_encode($new_event).", ...)");
if ( ! is_array ( $to_notify ))
{
$to_notify = array ();
}
$notify_externals = $new_event ? ( $new_event [ '##notify_externals' ] ? ? null ) : ( $old_event [ '##notify_externals' ] ? ? null );
$disinvited = $msg_type == MSG_DISINVITE ? array_keys ( $to_notify ) : array ();
2005-11-09 00:15:14 +01:00
2024-07-06 19:39:21 +02:00
$owner = $old_event ? $old_event [ 'owner' ] : $new_event [ 'owner' ];
if ( $owner && ! isset ( $to_notify [ $owner ]) && $msg_type != MSG_ALARM )
{
$to_notify [ $owner ] = 'OCHAIR' ; // always include the event-owner
}
2005-11-09 00:15:14 +01:00
2024-07-06 19:39:21 +02:00
// ignore events in the past (give a tolerance of 10 seconds for the script)
if ( $new_event && ! $this -> eventInFuture ( $new_event ) || ! $new_event && $old_event && ! $this -> eventInFuture ( $old_event ))
{
error_log ( __METHOD__ . " ( $msg_type , " . json_encode ( $to_notify ) . " , ..., " . json_encode ( $new_event ) . " , ...) --> ignoring event in the past: start= " .
date ( 'Y-m-d H:i:s' , ( $new_event ? : $old_event )[ 'start' ]) . " < " . date ( 'Y-m-d H:i:s' , $this -> now_su - 10 ));
return False ;
}
// check if default timezone is set correctly to server-timezone (ical-parser messes with it!!!)
if ( $GLOBALS [ 'egw_info' ][ 'server' ][ 'server_timezone' ] && ( $tz = date_default_timezone_get ()) != $GLOBALS [ 'egw_info' ][ 'server' ][ 'server_timezone' ])
{
$restore_tz = $tz ;
date_default_timezone_set ( $GLOBALS [ 'egw_info' ][ 'server' ][ 'server_timezone' ]);
}
$temp_user = $GLOBALS [ 'egw_info' ][ 'user' ]; // save user-date of the enviroment to restore it after
2019-12-18 19:54:39 +01:00
2024-07-06 19:39:21 +02:00
if ( ! $user )
2019-12-18 19:54:39 +01:00
{
2024-07-06 19:39:21 +02:00
$user = $temp_user [ 'account_id' ];
2019-12-18 19:54:39 +01:00
}
2024-07-06 19:39:21 +02:00
$lang = $GLOBALS [ 'egw_info' ][ 'user' ][ 'preferences' ][ 'common' ][ 'lang' ];
if ( $GLOBALS [ 'egw' ] -> preferences -> account_id != $user )
{
// Get correct preferences
$GLOBALS [ 'egw' ] -> preferences -> __construct ( is_numeric ( $user ) ? $user : $temp_user [ 'account_id' ]);
$GLOBALS [ 'egw_info' ][ 'user' ][ 'preferences' ] = $GLOBALS [ 'egw' ] -> preferences -> read_repository ();
2019-12-18 19:54:39 +01:00
2024-07-06 19:39:21 +02:00
// If target user/participant is not an account, try to get good notification message
if ( ! is_numeric ( $user ))
{
$res_info = $this -> resource_info ( $user );
$title = $res_info [ 'name' ] ? : Link :: title ( $res_info [ 'app' ], $res_info [ 'res_id' ]) ? : $res_info [ 'res_id' ] ? : $user ;
$GLOBALS [ 'egw' ] -> preferences -> values [ 'fullname' ] = $GLOBALS [ 'egw' ] -> preferences -> values [ 'lastname' ] = $title ;
$GLOBALS [ 'egw' ] -> preferences -> values [ 'firstname' ] = '' ;
$msg = $GLOBALS [ 'egw' ] -> preferences -> user [ 'calendar' ][ 'notifyResponse' ] ? : $GLOBALS [ 'egw' ] -> preferences -> default [ 'calendar' ][ 'notifyResponse' ] ? : $GLOBALS [ 'egw' ] -> preferences -> forced [ 'calendar' ][ 'notifyResponse' ];
$GLOBALS [ 'egw_info' ][ 'user' ][ 'preferences' ][ 'calendar' ][ 'notifyResponse' ] = $GLOBALS [ 'egw' ] -> preferences -> parse_notify (
$msg
);
}
2005-11-09 00:15:14 +01:00
2024-07-06 19:39:21 +02:00
}
$senderid = $this -> user ;
$event = $msg_type == MSG_ADDED || $msg_type == MSG_MODIFIED ? $new_event : $old_event ;
// add all group-members to the notification, unless they are already participants
foreach ( $to_notify as $userid => $statusid )
2005-11-23 15:21:20 +01:00
{
2024-07-06 19:39:21 +02:00
if ( is_numeric ( $userid ) && $GLOBALS [ 'egw' ] -> accounts -> get_type ( $userid ) == 'g' &&
( $members = $GLOBALS [ 'egw' ] -> accounts -> members ( $userid , true )))
2005-11-23 15:21:20 +01:00
{
2024-07-06 19:39:21 +02:00
foreach ( $members as $member )
2005-11-23 15:21:20 +01:00
{
2024-07-06 19:39:21 +02:00
if ( ! isset ( $to_notify [ $member ]))
{
$to_notify [ $member ] = 'G' ; // Group-invitation
}
2005-11-23 15:21:20 +01:00
}
}
}
2024-07-06 19:39:21 +02:00
// unless we notify externals about everything aka 'responses'
// we will notify only an external chair, if only one exists
if (( $notify_externals ? : $GLOBALS [ 'egw_info' ][ 'user' ][ 'calendar' ][ 'notify_externals' ] ? ? null ) !== 'responses' )
2018-06-26 15:57:09 +02:00
{
2024-07-06 19:39:21 +02:00
// check if we have *only* an external chair
$chair = null ;
foreach ( $to_notify as $userid => $statusid )
{
$res_info = $quantity = $role = null ;
calendar_so :: split_status ( $statusid , $quantity , $role );
if ( $role == 'CHAIR' && ( empty ( $chair ) || ! is_numeric ( $chair )))
{
$chair = $userid ;
}
}
// *only* an external chair --> do not notify anyone, but the external chair and the current user
if ( ! empty ( $chair ) && ! is_numeric ( $chair ))
2018-06-26 15:57:09 +02:00
{
2024-07-06 19:39:21 +02:00
$to_notify = array ( $chair => $to_notify [ $chair ]) +
( isset ( $to_notify [ $user ]) ? array ( $user => $to_notify [ $user ]) : array ());
2018-06-26 15:57:09 +02:00
}
}
2024-07-06 19:39:21 +02:00
// Event is passed in user time, make sure that's taken into account for date calculations
$user_prefs = $GLOBALS [ 'egw_info' ][ 'user' ][ 'preferences' ];
$date = new Api\DateTime ( 'now' , new DateTimeZone ( $user_prefs [ 'common' ][ 'tz' ]));
$startdate = new Api\DateTime ( $event [ 'start' ], new DateTimeZone ( $user_prefs [ 'common' ][ 'tz' ]));
$enddate = new Api\DateTime ( $event [ 'end' ], new DateTimeZone ( $user_prefs [ 'common' ][ 'tz' ]));
$modified = new Api\DateTime ( $event [ 'modified' ], new DateTimeZone ( $user_prefs [ 'common' ][ 'tz' ]));
if ( $old_event ) $olddate = new Api\DateTime ( $old_event [ 'start' ], new DateTimeZone ( $user_prefs [ 'common' ][ 'tz' ]));
$rdates = array_map ( static function ( $rdate ) use ( $user_prefs ) {
return new Api\DateTime ( $rdate , new DateTimeZone ( $user_prefs [ 'common' ][ 'tz' ]));
}, $event [ 'recur_rdates' ] ? ? []);
$recur_date = isset ( $event [ 'recur_date' ]) ? new Api\DateTime ( $event [ 'recur_date' ], new DateTimeZone ( $user_prefs [ 'common' ][ 'tz' ])) : null ;
$recurrence = isset ( $event [ 'recurrence' ]) ? new Api\DateTime ( $event [ 'recurrence' ], new DateTimeZone ( $user_prefs [ 'common' ][ 'tz' ])) : null ;
//error_log(__METHOD__."() date_default_timezone_get()=".date_default_timezone_get().", user-timezone=".Api\DateTime::$user_timezone->getName().", startdate=".$startdate->format().", enddate=".$enddate->format().", updated=".$modified->format().", olddate=".($olddate ? $olddate->format() : ''));
$owner_prefs = $ics = null ;
foreach ( $to_notify as $userid => $statusid )
2018-06-26 15:57:09 +02:00
{
2024-07-06 19:39:21 +02:00
$res_info = $quantity = $role = null ;
calendar_so :: split_status ( $statusid , $quantity , $role );
if ( $this -> debug > 0 ) error_log ( __METHOD__ . " trying to notify $userid , with $statusid ( $role ) " );
2021-01-14 22:19:45 +01:00
2024-07-06 19:39:21 +02:00
// hack to add videoconference in event description, by always setting $cleared_event
// Can't re-load, if we're notifying of a cancelled recurrence we'll load the next event in the series
$cleared_event = $event ;
2021-01-14 22:19:45 +01:00
2024-07-06 19:39:21 +02:00
if ( ! is_numeric ( $userid ))
{
$res_info = $this -> resource_info ( $userid );
2011-05-27 22:32:46 +02:00
2024-07-06 19:39:21 +02:00
// check if responsible for a resource has read rights on event (might be private!)
if ( $res_info [ 'app' ] == 'resources' && $res_info [ 'responsible' ] &&
! $this -> check_perms ( Acl :: READ , $event , 0 , 'ts' , null , $res_info [ 'responsible' ]))
{
// --> use only details from (private-)cleared event only containing resource ($userid)
// reading server timezone, to be able to use cleared event for iCal generation further down
//$cleared_event = $this->read($event['id'], null, true, 'server');
$this -> clear_private_infos ( $cleared_event , array ( $userid ));
}
$userid = $res_info [ 'responsible' ] ? ? null ;
2020-04-03 10:28:34 +02:00
2024-07-06 19:39:21 +02:00
if ( empty ( $userid )) // no resource responsible: $userid===0
{
if ( empty ( $res_info [ 'email' ])) continue ; // no way to notify
// check if event-owner wants non-EGroupware users notified
if ( is_null ( $owner_prefs ))
{
$preferences = new Api\Preferences ( $owner );
$owner_prefs = $preferences -> read_repository ();
if ( ! empty ( $notify_externals )) $owner_prefs [ 'calendar' ][ 'notify_externals' ] = $notify_externals ;
}
if ( $role != 'CHAIR' && // always notify externals CHAIRs
( empty ( $owner_prefs [ 'calendar' ][ 'notify_externals' ]) ||
$owner_prefs [ 'calendar' ][ 'notify_externals' ] == 'no' ))
{
continue ;
}
$userid = $res_info [ 'email' ];
}
}
2012-11-13 14:31:44 +01:00
2024-07-06 19:39:21 +02:00
if ( $statusid == 'R' || $GLOBALS [ 'egw' ] -> accounts -> get_type ( $userid ) == 'g' )
2012-11-13 14:31:44 +01:00
{
2024-07-06 19:39:21 +02:00
continue ; // dont notify rejected participants or groups
2012-11-13 14:31:44 +01:00
}
2024-07-06 19:39:21 +02:00
if ( $userid != $GLOBALS [ 'egw_info' ][ 'user' ][ 'account_id' ] ||
( $userid == $GLOBALS [ 'egw_info' ][ 'user' ][ 'account_id' ] &&
$user_prefs [ 'calendar' ][ 'receive_own_updates' ] == 1 ) ||
$msg_type == MSG_ALARM )
2012-01-05 06:47:02 +01:00
{
2024-07-06 19:39:21 +02:00
$tfn = $tln = $lid = null ; //cleanup of lastname and fullname (in case they are set in a previous loop)
if ( is_numeric ( $userid ))
{
$preferences = new Api\Preferences ( $userid );
$GLOBALS [ 'egw_info' ][ 'user' ][ 'preferences' ] = $part_prefs = $preferences -> read_repository ();
$fullname = Api\Accounts :: username ( $userid );
$tfn = Api\Accounts :: id2name ( $userid , 'account_firstname' );
$tln = Api\Accounts :: id2name ( $userid , 'account_lastname' );
}
else // external email address: use Api\Preferences of event-owner, plus some hardcoded settings (eg. ical notification)
2012-01-05 06:47:02 +01:00
{
2024-07-06 19:39:21 +02:00
if ( is_null ( $owner_prefs ))
{
$preferences = new Api\Preferences ( $owner );
$GLOBALS [ 'egw_info' ][ 'user' ][ 'preferences' ] = $owner_prefs = $preferences -> read_repository ();
if ( ! empty ( $notify_externals )) $owner_prefs [ 'calendar' ][ 'notify_externals' ] = $notify_externals ;
}
$part_prefs = $owner_prefs ;
$part_prefs [ 'calendar' ][ 'receive_updates' ] = $owner_prefs [ 'calendar' ][ 'notify_externals' ];
$part_prefs [ 'calendar' ][ 'update_format' ] = 'ical' ; // use ical format
$fullname = $res_info && ! empty ( $res_info [ 'name' ]) ? $res_info [ 'name' ] : $userid ;
2012-01-05 06:47:02 +01:00
}
2024-07-06 19:39:21 +02:00
$m_type = $msg_type ;
if ( ! $ignore_prefs && ! $this -> update_requested ( $userid , $part_prefs , $m_type , $old_event , $new_event , $role ,
$event [ 'participants' ][ $GLOBALS [ 'egw_info' ][ 'user' ][ 'account_id' ]]))
2012-01-05 06:47:02 +01:00
{
2024-07-06 19:39:21 +02:00
//error_log("--> Update/notification NOT requested / ignored");
2012-01-05 06:47:02 +01:00
continue ;
}
2024-07-06 19:39:21 +02:00
$action = $notify_msg = null ;
$method = $this -> msg_type2ical_method ( $m_type , $action , $notify_msg , $user_prefs [ 'calendar' ]);
2010-03-03 12:59:55 +01:00
2024-07-06 19:39:21 +02:00
if ( $lang !== $part_prefs [ 'common' ][ 'lang' ])
2012-01-05 06:47:02 +01:00
{
2024-07-06 19:39:21 +02:00
Api\Translation :: init ();
$lang = $part_prefs [ 'common' ][ 'lang' ];
2012-01-05 06:47:02 +01:00
}
2017-01-10 19:42:18 +01:00
2024-07-06 19:39:21 +02:00
// Since we're running from cron, make sure notifications uses user's theme (for images)
$GLOBALS [ 'egw_info' ][ 'server' ][ 'template_set' ] = $GLOBALS [ 'egw_info' ][ 'user' ][ 'preferences' ][ 'common' ][ 'template_set' ];
2017-01-18 18:29:44 +01:00
2024-07-06 19:39:21 +02:00
$event_arr = null ;
$details = $this -> _get_event_details ( isset ( $cleared_event ) ? $cleared_event : $event ,
$action , $event_arr , $disinvited );
$details [ 'fullname' ] = is_numeric ( $user ) ? Api\Accounts :: username ( $user ) : $fullname ;
$details [ 'to-fullname' ] = $fullname ;
$details [ 'to-firstname' ] = isset ( $tfn ) ? $tfn : '' ;
$details [ 'to-lastname' ] = isset ( $tln ) ? $tln : '' ;
2012-11-13 14:31:44 +01:00
2024-07-06 19:39:21 +02:00
// event is in user-time of current user, now we need to calculate the tz-difference to the notified user and take it into account
if ( ! isset ( $part_prefs [ 'common' ][ 'tz' ])) $part_prefs [ 'common' ][ 'tz' ] = $GLOBALS [ 'egw_info' ][ 'server' ][ 'server_timezone' ];
try
{
$timezone = new DateTimeZone ( $part_prefs [ 'common' ][ 'tz' ]);
} catch ( Exception $e )
{
$timezone = new DateTimeZone ( $GLOBALS [ 'egw_info' ][ 'server' ][ 'server_timezone' ]);
}
$timeformat = $part_prefs [ 'common' ][ 'timeformat' ];
switch ( $timeformat )
{
case '24' :
$timeformat = 'H:i' ;
break ;
case '12' :
$timeformat = 'h:i a' ;
break ;
}
$timeformat = $part_prefs [ 'common' ][ 'dateformat' ] . ', ' . $timeformat ;
2010-11-11 09:51:13 +01:00
2024-07-06 19:39:21 +02:00
// Set dates:
// $details in "preference" format, $cleared_event as DateTime so calendar_ical->exportVCal() gets
// the times right, since it assumes a timestamp is in server time
$cleared_event [ 'start' ] = $startdate -> setTimezone ( $timezone );
$details [ 'startdate' ] = $startdate -> format ( $timeformat );
2010-11-11 09:51:13 +01:00
2024-07-06 19:39:21 +02:00
$cleared_event [ 'end' ] = $enddate -> setTimezone ( $timezone );
$details [ 'enddate' ] = $enddate -> format ( $timeformat );
2010-11-11 09:51:13 +01:00
2024-07-06 19:39:21 +02:00
$cleared_event [ 'updated' ] = $modified -> setTimezone ( $timezone );
$details [ 'updated' ] = $modified -> format ( $timeformat ) . ', ' . Api\Accounts :: username ( $event [ 'modifier' ]);
2024-07-02 15:13:42 +02:00
2024-07-06 19:39:21 +02:00
// we also need to "fix" timezone for rdates, to not get wrong times!
$cleared_event [ 'recur_rdates' ] = array_map ( static function ( $rdate ) use ( $timezone ) {
return $rdate -> setTimezone ( $timezone );
}, $rdates );
2010-11-11 09:51:13 +01:00
2024-07-06 19:39:21 +02:00
if ( isset ( $recur_date ))
{
$cleared_event [ 'recur_date' ] = $recur_date -> setTimezone ( $timezone );
$details [ 'recur_date' ] = $recur_date -> format ( $timeformat );
}
if ( isset ( $recurrence ))
{
$cleared_event [ 'recurrence' ] = $recurrence -> setTimezone ( $timezone );
}
2024-07-04 12:41:21 +02:00
2024-07-06 19:39:21 +02:00
// Current date doesn't need to go into the cleared event, just for details
$date -> setTimezone ( $timezone );
$details [ 'date' ] = $date -> format ( $timeformat );
2021-12-08 18:59:18 +01:00
2024-07-06 19:39:21 +02:00
if ( $old_event != False )
2020-04-03 10:28:34 +02:00
{
2024-07-06 19:39:21 +02:00
$olddate -> setTimezone ( $timezone );
$details [ 'olddate' ] = $olddate -> format ( $timeformat );
2020-04-03 10:28:34 +02:00
}
2024-07-06 19:39:21 +02:00
// generate a personal videoconference url, if we need one
if ( ! empty ( $event [ '##videoconference' ]) && ! calendar_hooks :: isVideoconferenceDisabled ())
{
$avatar = new Api\Contacts\Photo ( is_numeric ( $userid ) ? " account: $userid " :
( isset ( $res_info ) && $res_info [ 'type' ] === 'c' ? $res_info [ 'res_id' ] : $userid ),
// disable sharing links currently, as sharing links from a different EGroupware user destroy the session
true );
$details [ 'videoconference' ] = EGroupware\Status\Videoconference\Call :: genMeetingUrl ( $event [ '##videoconference' ], [
'title' => $event [ 'title' ],
'name' => $fullname ,
'email' => is_numeric ( $userid ) ? Api\Accounts :: id2name ( $userid , 'account_email' ) : $userid ,
'avatar' => ( string ) $avatar ,
'account_id' => $userid ,
'cal_id' => $details [ 'id' ],
'notify_only' => true
], [ 'participants' => array_filter ( $event [ 'participants' ], function ( $key ) {
return is_numeric ( $key );
}, ARRAY_FILTER_USE_KEY )], $startdate , $enddate );
$event_arr [ 'videoconference' ] = [
'field' => lang ( 'Video Conference' ),
'data' => $details [ 'videoconference' ],
];
// hack to add videoconference-url to ical, only if description was NOT cleared
if ( isset ( $cleared_event [ 'description' ]))
2009-10-18 14:46:18 +02:00
{
2024-07-06 19:39:21 +02:00
$cleared_event [ 'description' ] = lang ( 'Video conference' ) . ': ' . $details [ 'videoconference' ] . " \n \n " . $event [ 'description' ];
2009-10-18 14:46:18 +02:00
}
2024-07-06 19:39:21 +02:00
}
//error_log(__METHOD__."() userid=$userid, timezone=".$timezone->getName().", startdate=$details[startdate], enddate=$details[enddate], updated=$details[updated], olddate=$details[olddate]");
list ( $subject , $notify_body ) = explode ( " \n " , $GLOBALS [ 'egw' ] -> preferences -> parse_notify ( $notify_msg , $details ), 2 );
// alarm is NOT an iCal method, therefore we have to use extened (no iCal)
switch ( $msg_type == MSG_ALARM ? 'extended' : $part_prefs [ 'calendar' ][ 'update_format' ])
{
case 'ical' :
if ( is_null ( $ics ) || $m_type != $msg_type || $event [ '##videoconference' ]) // need different ical for organizer notification or videoconference join urls
{
$calendar_ical = new calendar_ical ();
$calendar_ical -> setSupportedFields ( 'full' ); // full iCal fields+event TZ
// we need to pass $event[id] so iCal class reads event again,
// as event is in user TZ, but iCal class expects server TZ!
$ics = $calendar_ical -> exportVCal ([ $cleared_event ],
'2.0' , $method , $cleared_event [ 'recur_date' ] ? ? null ,
'' , 'utf-8' , $method == 'REPLY' ? $user : 0
);
unset ( $calendar_ical );
}
$attachment = array (
'string' => $ics ,
'filename' => 'cal.ics' ,
'encoding' => '8bit' ,
'type' => 'text/calendar; method=' . $method ,
);
if ( $m_type != $msg_type ) unset ( $ics );
$subject = isset ( $cleared_event ) ? $cleared_event [ 'title' ] : $event [ 'title' ];
2012-07-17 11:48:43 +02:00
// fall through
2024-07-06 19:39:21 +02:00
case 'extended' :
2012-08-21 16:49:26 +02:00
2024-07-06 19:39:21 +02:00
$details_body = lang ( 'Event Details follow' ) . " : \n " ;
foreach ( $event_arr as $key => $val )
2011-05-27 22:32:46 +02:00
{
2024-07-06 19:39:21 +02:00
if ( ! empty ( $details [ $key ]))
2012-07-25 00:23:32 +02:00
{
2024-07-06 19:39:21 +02:00
switch ( $key )
{
case 'access' :
case 'priority' :
case 'link' :
case 'description' :
case 'title' :
break ;
default :
$details_body .= sprintf ( " %-20s %s \n " , $val [ 'field' ] . ':' , $details [ $key ]);
break ;
}
}
2005-11-09 00:15:14 +01:00
}
2024-07-06 19:39:21 +02:00
break ;
}
// send via notification_app
if ( $GLOBALS [ 'egw_info' ][ 'apps' ][ 'notifications' ][ 'enabled' ])
{
try
2012-08-21 16:49:26 +02:00
{
2024-07-06 19:39:21 +02:00
//error_log(__METHOD__."() notifying $userid from $senderid: $subject");
$notification = new notifications ();
$notification -> set_receivers ( array ( $userid ));
$notification -> set_sender ( $senderid );
$notification -> set_subject ( $subject );
// as we want ical body to be just description, we can NOT set links, as they get appended to body
if ( $part_prefs [ 'calendar' ][ 'update_format' ] != 'ical' )
{
$notification -> set_message ( $notify_body . " \n \n " . $details [ 'description' ] . " \n \n " . $details_body );
$notification -> set_links ( array ( $details [ 'link_arr' ]));
}
else
{
// iCal: description need to be separated from body by fancy separator
$notification -> set_message ( $notify_body . " \n \n " . $details_body . " \n *~*~*~*~*~*~*~*~*~* \n \n " . $details [ 'description' ]);
}
// popup notifiactions: set subject, different message (without separator) and (always) links
$notification -> set_popupsubject ( $subject );
2017-05-23 16:54:20 +02:00
2024-07-06 19:39:21 +02:00
if ( $method == 'REQUEST' )
{
// Add ACCEPT|REJECT|TENTATIVE actions
$notification -> set_popupdata ( 'calendar' , array (
'event_id' => $event [ 'id' ],
'user_id' => $userid ,
2020-04-07 18:01:55 +02:00
'type' => $m_type ,
2024-07-06 19:39:21 +02:00
'id' => $event [ 'id' ],
'app' => 'calendar' ,
2020-04-08 11:28:09 +02:00
'videoconference' => $details [ 'videoconference' ],
2024-07-06 19:39:21 +02:00
), $event [ 'id' ]);
}
2024-07-09 21:39:40 +02:00
elseif ( $m_type === MSG_ALARM )
2024-07-06 19:39:21 +02:00
{
$notification -> set_popupdata ( 'calendar' ,
array ( 'egw_pr_notify' => 1 ,
'type' => $m_type ,
'videoconference' => $details [ 'videoconference' ],
'account_id' => $senderid ,
'name' => Api\Accounts :: username ( $senderid )
)
+ ( $alarm ? [ 'alarm-offset' => ( int ) $alarm [ 'offset' ]] : []), $event [ 'id' ]);
}
2024-07-09 21:39:40 +02:00
else
{
$notification -> set_popupdata ( 'calendar' , null , $event [ 'id' ]);
}
2024-07-06 19:39:21 +02:00
$notification -> set_popupmessage ( $subject . " \n \n " . $notify_body . " \n \n " . $details [ 'description' ] . " \n \n " . $details_body . " \n \n " );
$notification -> set_popuplinks ( array ( $details [ 'link_arr' ] + array ( 'app' => 'calendar' )));
2012-09-04 14:10:37 +02:00
2024-07-06 19:39:21 +02:00
if ( ! empty ( $attachment ))
{
$notification -> set_attachments ( array ( $attachment ));
}
$notification -> send ();
$errors = notifications :: errors ( true );
} catch ( Exception $exception )
{
$errors = [ $exception -> getMessage ()];
continue ;
}
2007-11-29 19:31:24 +01:00
}
2024-07-06 19:39:21 +02:00
else
{
$errors = [ lang ( 'Can not send any notifications because notifications app is not installed!' )];
}
foreach ( $errors as $error )
{
error_log ( __METHOD__ . " () Error notifying $userid from $senderid : $subject : $error " );
// send notification errors via push to current user (not session, as alarms send via async job have none!)
( new Api\Json\Push ( $GLOBALS [ 'egw_info' ][ 'user' ][ 'account_id' ])) -> message (
lang ( 'Error notifying %1' , ! is_numeric ( $userid ) ? $userid :
Api\Accounts :: id2name ( $userid , 'account_fullname' ) . ' <' . Api\Accounts :: id2name ( $userid , 'account_email' ) . '>' ) .
" \n " . $subject . " \n " . $error , 'error' );
2007-11-29 19:31:24 +01:00
}
2006-09-25 12:29:03 +02:00
}
2005-11-09 00:15:14 +01:00
}
2024-07-06 19:39:21 +02:00
// restore the enviroment (preferences->read_repository() sets the timezone!)
$GLOBALS [ 'egw_info' ][ 'user' ] = $temp_user ;
if ( $GLOBALS [ 'egw' ] -> preferences -> account_id != $temp_user [ 'account_id' ])
{
$GLOBALS [ 'egw' ] -> preferences -> __construct ( $temp_user [ 'account_id' ]);
$GLOBALS [ 'egw_info' ][ 'user' ][ 'preferences' ] = $GLOBALS [ 'egw' ] -> preferences -> read_repository ();
//echo "<p>".__METHOD__."() restored enviroment of #$temp_user[account_id] $temp_user[account_fullname]: tz={$GLOBALS['egw_info']['user']['preferences']['common']['tz']}</p>\n";
}
else
{
// Loading other user's preferences can change current user's tz
$GLOBALS [ 'egw' ] -> preferences -> check_set_tz_offset ();
}
if ( $lang !== $GLOBALS [ 'egw_info' ][ 'user' ][ 'preferences' ][ 'common' ][ 'lang' ])
{
Api\Translation :: init ();
}
// restore timezone, in case we had to reset it to server-timezone
if ( ! empty ( $restore_tz )) date_default_timezone_set ( $restore_tz );
}
catch ( Throwable $e ) {
// logging all exceptions and errors to the error_log AND pushing them to user
$message = null ;
_egw_log_exception ( $e , $message );
$response = new Api\Json\Push ( $GLOBALS [ 'egw_info' ][ 'user' ][ 'account_id' ]);
$message .= ( $message ? " \n \n " : '' ) . $e -> getMessage ();
$message .= " \n \n " . $e -> getFile () . ' (' . $e -> getLine () . ')' ;
// only show trace (incl. function arguments) if explicitly enabled, eg. on a development system
if ( $GLOBALS [ 'egw_info' ][ 'server' ][ 'exception_show_trace' ])
{
$message .= " \n \n " . $e -> getTraceAsString ();
}
$response -> message ( $message , 'error' );
2013-01-31 16:15:48 +01:00
2024-07-06 19:39:21 +02:00
// under PHP 8 the destructor is called to late and the response is not send
$GLOBALS [ 'egw' ] -> __destruct ();
exit ;
}
2007-11-22 09:29:16 +01:00
return true ;
2005-11-09 00:15:14 +01:00
}
function get_update_message ( $event , $added )
{
2015-06-25 22:39:53 +02:00
$nul = null ;
2005-11-09 00:15:14 +01:00
$details = $this -> _get_event_details ( $event , $added ? lang ( 'Added' ) : lang ( 'Modified' ), $nul );
$notify_msg = $this -> cal_prefs [ $added || empty ( $this -> cal_prefs [ 'notifyModified' ]) ? 'notifyAdded' : 'notifyModified' ];
return explode ( " \n " , $GLOBALS [ 'egw' ] -> preferences -> parse_notify ( $notify_msg , $details ), 2 );
}
2006-03-07 20:18:15 +01:00
/**
* Function called via async service , when an alarm is to be send
*
* @ param array $alarm array with keys owner , cal_id , all
2008-05-08 00:12:25 +02:00
* @ return boolean
2006-03-07 20:18:15 +01:00
*/
2005-11-09 00:15:14 +01:00
function send_alarm ( $alarm )
{
//echo "<p>bocalendar::send_alarm("; print_r($alarm); echo ")</p>\n";
$GLOBALS [ 'egw_info' ][ 'user' ][ 'account_id' ] = $this -> owner = $alarm [ 'owner' ];
2016-05-01 19:47:59 +02:00
$event_time_user = Api\DateTime :: server2user ( $alarm [ 'time' ] + $alarm [ 'offset' ]); // alarm[time] is in server-time, read requires user-time
2006-03-07 20:18:15 +01:00
if ( ! $alarm [ 'owner' ] || ! $alarm [ 'cal_id' ] || ! ( $event = $this -> read ( $alarm [ 'cal_id' ], $event_time_user )))
2005-11-09 00:15:14 +01:00
{
return False ; // event not found
}
if ( $alarm [ 'all' ])
{
$to_notify = $event [ 'participants' ];
}
2016-05-01 19:47:59 +02:00
elseif ( $this -> check_perms ( Acl :: READ , $event )) // checks agains $this->owner set to $alarm[owner]
2005-11-09 00:15:14 +01:00
{
$to_notify [ $alarm [ 'owner' ]] = 'A' ;
}
else
{
return False ; // no rights
}
2012-05-15 16:41:41 +02:00
// need to load calendar translations and set currentapp, so calendar can reload a different lang
2016-05-01 19:47:59 +02:00
Api\Translation :: add_app ( 'calendar' );
2012-05-15 16:41:41 +02:00
$GLOBALS [ 'egw_info' ][ 'flags' ][ 'currentapp' ] = 'calendar' ;
2020-07-26 16:00:38 +02:00
$ret = $this -> _send_update ( MSG_ALARM , $to_notify , $event , False , $alarm [ 'owner' ], $alarm );
2006-03-07 20:18:15 +01:00
// create a new alarm for recuring events for the next event, if one exists
2024-02-08 10:24:39 +01:00
if ( ! empty ( $event [ 'recur_type' ]) && ( $event = $this -> read ( $alarm [ 'cal_id' ], $event_time_user + 1 )))
2006-03-07 20:18:15 +01:00
{
$alarm [ 'time' ] = $this -> date2ts ( $event [ 'start' ]) - $alarm [ 'offset' ];
2012-12-07 15:10:51 +01:00
unset ( $alarm [ 'times' ]);
unset ( $alarm [ 'next' ]);
2019-04-26 15:13:12 +02:00
unset ( $alarm [ 'keep_time' ]); // need to remove the keep_time, as otherwise the alarm would be deleted automatically
2015-06-25 22:39:53 +02:00
//error_log(__METHOD__."() moving alarm to next recurrence ".array2string($alarm));
$this -> save_alarm ( $alarm [ 'cal_id' ], $alarm , false ); // false = do NOT update timestamp, as nothing changed for iCal clients
2006-03-07 20:18:15 +01:00
}
return $ret ;
2005-11-09 00:15:14 +01:00
}
2005-06-14 23:54:17 +02:00
/**
2009-08-04 19:14:16 +02:00
* saves an event to the database , does NOT do any notifications , see calendar_boupdate :: update for that
2005-06-14 23:54:17 +02:00
*
* This methode converts from user to server time and handles the insertion of users and dates of repeating events
*
* @ param array $event
2015-06-25 22:39:53 +02:00
* @ param boolean $ignore_acl = false should we ignore the acl
* @ param boolean $updateTS = true update the content history of the event
2017-01-18 18:29:44 +01:00
* Please note : you should ALLWAYS update timestamps , as they are required for sync !
2009-08-04 19:14:16 +02:00
* @ return int | boolean $cal_id > 0 or false on error ( eg . permission denied )
2005-06-14 23:54:17 +02:00
*/
2009-10-25 19:20:58 +01:00
function save ( $event , $ignore_acl = false , $updateTS = true )
2005-06-14 23:54:17 +02:00
{
2011-04-07 10:01:48 +02:00
//error_log(__METHOD__.'('.array2string($event).", $ignore_acl, $updateTS)");
2010-11-11 09:51:13 +01:00
2005-11-09 00:15:14 +01:00
// check if user has the permission to update / create the event
2023-06-09 18:27:28 +02:00
if ( ! $ignore_acl && ( ! empty ( $event [ 'id' ]) && ! $this -> check_perms ( Acl :: EDIT , $event [ 'id' ]) ||
empty ( $event [ 'id' ]) && ! $this -> check_perms ( Acl :: EDIT , 0 , $event [ 'owner' ]) &&
2016-05-01 19:47:59 +02:00
! $this -> check_perms ( Acl :: ADD , 0 , $event [ 'owner' ])))
2005-11-09 00:15:14 +01:00
{
return false ;
}
2008-01-17 16:39:22 +01:00
2023-06-09 18:27:28 +02:00
if ( ! empty ( $event [ 'id' ]))
2010-02-17 14:29:28 +01:00
{
// invalidate the read-cache if it contains the event we store now
if ( $event [ 'id' ] == self :: $cached_event [ 'id' ]) self :: $cached_event = array ();
2019-05-28 00:17:53 +02:00
$old_event = $this -> read ( $event [ 'id' ], $event [ 'recurrence' ], $ignore_acl , 'server' );
2010-02-17 14:29:28 +01:00
}
else
{
$old_event = null ;
}
2017-05-31 17:55:41 +02:00
if ( ! isset ( $event [ 'whole_day' ])) $event [ 'whole_day' ] = $this -> isWholeDay ( $event );
2005-11-09 00:15:14 +01:00
2020-05-19 18:44:56 +02:00
$this -> check_reset_statuses ( $event , $old_event );
2017-03-29 19:31:53 +02:00
// set recur-enddate/range-end to real end-date of last recurrence
2024-07-02 15:13:42 +02:00
if ( ! empty ( $event [ 'recur_type' ]) && ( ! empty ( $event [ 'recur_enddate' ]) || $event [ 'recur_type' ] == calendar_rrule :: RDATE ) && $event [ 'start' ])
2017-03-29 19:31:53 +02:00
{
2017-05-31 17:55:41 +02:00
$event [ 'recur_enddate' ] = new Api\DateTime ( $event [ 'recur_enddate' ], calendar_timezones :: DateTimeZone ( $event [ 'tzid' ]));
2017-06-06 23:03:14 +02:00
$event [ 'recur_enddate' ] -> setTime ( 23 , 59 , 59 );
$rrule = calendar_rrule :: event2rrule ( $event , true , Api\DateTime :: $user_timezone -> getName ());
2017-03-29 19:31:53 +02:00
$rrule -> rewind ();
$enddate = $rrule -> current ();
do
{
$rrule -> next_no_exception ();
$occurrence = $rrule -> current ();
}
2022-04-24 18:15:47 +02:00
while ( $rrule -> validDate ( $event [ 'whole_day' ]) && ( $enddate = $occurrence ));
2024-06-04 16:44:59 +02:00
if ( $enddate )
{
$enddate -> modify (( $event [ 'end' ] - $event [ 'start' ]) . ' second' );
$event [ 'recur_enddate' ] = $enddate -> format ( 'ts' );
}
2017-03-29 19:31:53 +02:00
//error_log(__METHOD__."($event[title]) start=".Api\DateTime::to($event['start'],'string').', end='.Api\DateTime::to($event['end'],'string').', range_end='.Api\DateTime::to($event['recur_enddate'],'string'));
}
2010-05-01 18:25:10 +02:00
$save_event = $event ;
2010-04-23 15:52:45 +02:00
if ( $event [ 'whole_day' ])
2010-04-23 08:15:18 +02:00
{
2010-06-23 16:52:55 +02:00
if ( ! empty ( $event [ 'start' ]))
{
2016-05-01 19:47:59 +02:00
$time = $this -> so -> startOfDay ( new Api\DateTime ( $event [ 'start' ], Api\DateTime :: $user_timezone ));
$event [ 'start' ] = Api\DateTime :: to ( $time , 'ts' );
2010-06-23 16:52:55 +02:00
$save_event [ 'start' ] = $time ;
}
if ( ! empty ( $event [ 'end' ]))
{
2024-08-02 16:09:15 +02:00
// as EGroupware end of whole-day events is 1sec (or sometimes 60sec) shorter, we have to add 60sec,
// before calling startOfDay to NOT lose one day
$time = $this -> so -> startOfDay (( new Api\DateTime ( $event [ 'end' ], Api\DateTime :: $user_timezone )) -> add ( '60seconds' ));
2024-07-31 19:20:31 +02:00
$time -> add ( '-1second' );
2016-05-01 19:47:59 +02:00
$event [ 'end' ] = Api\DateTime :: to ( $time , 'ts' );
2013-06-07 13:53:45 +02:00
$save_event [ 'end' ] = $time ;
2010-06-23 16:52:55 +02:00
}
2010-05-01 18:25:10 +02:00
if ( ! empty ( $event [ 'recurrence' ]))
{
2016-05-01 19:47:59 +02:00
$time = $this -> so -> startOfDay ( new Api\DateTime ( $event [ 'recurrence' ], Api\DateTime :: $user_timezone ));
$event [ 'recurrence' ] = Api\DateTime :: to ( $time , 'ts' );
2010-05-01 18:25:10 +02:00
}
if ( ! empty ( $event [ 'recur_enddate' ]))
{
2017-06-07 21:19:20 +02:00
// all-day events are handled in server time, but here (BO) it's in user time
$time = new Api\DateTime ( $event [ 'recur_enddate' ], Api\DateTime :: $user_timezone );
$time -> setTime ( 23 , 59 , 59 );
// Check to see if switching timezones changes the date, we'll need to adjust for that
$enddate_event_timezone = clone $time ;
$time -> setServer ();
$delta = ( int ) $enddate_event_timezone -> format ( 'z' ) - ( int ) $time -> format ( 'z' );
$time -> add ( " $delta days " );
//$time->setServer();
$time -> setTime ( 23 , 59 , 59 );
2017-08-18 09:49:02 +02:00
2017-06-01 19:20:38 +02:00
$event [ 'recur_enddate' ] = $save_event [ 'recur_enddate' ] = $time ;
2010-05-01 18:25:10 +02:00
}
2010-04-23 08:15:18 +02:00
$timestamps = array ( 'modified' , 'created' );
2010-05-01 18:25:10 +02:00
// all-day events are handled in server time
2017-05-31 17:55:41 +02:00
// $event['tzid'] = $save_event['tzid'] = Api\DateTime::$server_timezone->getName();
2010-04-23 08:15:18 +02:00
}
else
{
$timestamps = array ( 'start' , 'end' , 'modified' , 'created' , 'recur_enddate' , 'recurrence' );
}
2005-11-09 00:15:14 +01:00
// we run all dates through date2ts, to adjust to server-time and the possible date-formats
2024-07-29 15:57:08 +02:00
foreach ( array_merge ( $timestamps , $this -> getCfTtimestamps ()) as $ts )
2005-06-14 23:54:17 +02:00
{
// we convert here from user-time to timestamps in server-time!
2020-06-24 23:07:52 +02:00
if ( isset ( $event [ $ts ])) $event [ $ts ] = $event [ $ts ] ? $this -> date2ts ( $event [ $ts ], true ) : 0 ;
2005-06-14 23:54:17 +02:00
}
2009-11-04 16:00:08 +01:00
// convert tzid name to integer tz_id, of set user default
if ( empty ( $event [ 'tzid' ]) || ! ( $event [ 'tz_id' ] = calendar_timezones :: tz2id ( $event [ 'tzid' ])))
{
2016-05-01 19:47:59 +02:00
$event [ 'tz_id' ] = calendar_timezones :: tz2id ( $event [ 'tzid' ] = Api\DateTime :: $user_timezone -> getName ());
2009-11-04 16:00:08 +01:00
}
2024-06-04 15:30:54 +02:00
// same with the recur exceptions and rdates
foreach ([ 'recur_exception' , 'recur_rdates' ] as $name )
2005-11-09 00:15:14 +01:00
{
2024-06-04 15:30:54 +02:00
if ( ! is_array ( $event [ $name ] ? ? null )) continue ;
foreach ( $event [ $name ] as & $date )
2005-11-09 00:15:14 +01:00
{
2010-04-23 08:15:18 +02:00
if ( $event [ 'whole_day' ])
{
2024-07-31 19:20:31 +02:00
// we use so->startOfDay(new Api\DateTime($time, Api\DateTime::$user_time)) as we not yet converted to server-time!
$date = $this -> so -> startOfDay ( new Api\DateTime ( $date , Api\DateTime :: $user_timezone )) -> format ( 'server' );
2010-04-23 08:15:18 +02:00
}
else
{
2013-08-15 14:23:38 +02:00
$date = $this -> date2ts ( $date , true );
2010-04-23 08:15:18 +02:00
}
2005-11-09 00:15:14 +01:00
}
2013-08-15 14:23:38 +02:00
unset ( $date );
2005-11-09 00:15:14 +01:00
}
// same with the alarms
2011-11-11 13:46:34 +01:00
if ( isset ( $event [ 'alarm' ]) && is_array ( $event [ 'alarm' ]) && isset ( $event [ 'start' ]))
2005-11-09 00:15:14 +01:00
{
2018-11-22 19:35:35 +01:00
// Expand group invitations so we don't lose individual alarms
$expanded = $event ;
$this -> enum_groups ( $expanded );
2015-06-25 22:39:53 +02:00
foreach ( $event [ 'alarm' ] as $id => & $alarm )
2005-11-09 00:15:14 +01:00
{
2023-06-09 18:27:28 +02:00
if ( ! empty ( $alarm [ 'time' ]))
2021-04-28 19:20:43 +02:00
{
$alarm [ 'time' ] = $this -> date2ts ( $alarm [ 'time' ], true ); // user to server-time
}
2021-01-27 19:04:08 +01:00
2024-07-31 19:20:31 +02:00
// remove alarms belonging to no longer existing or rejected participants
2023-06-09 18:27:28 +02:00
if ( ! empty ( $alarm [ 'owner' ]) && isset ( $expanded [ 'participants' ]))
2013-08-15 16:56:34 +02:00
{
2018-12-14 19:51:46 +01:00
// Don't auto-delete alarm if for all users
2023-06-09 18:27:28 +02:00
if ( ! empty ( $alarm [ 'all' ])) continue ;
2018-12-14 19:50:15 +01:00
2018-11-22 19:35:35 +01:00
$status = $expanded [ 'participants' ][ $alarm [ 'owner' ]];
2013-08-15 16:56:34 +02:00
if ( ! isset ( $status ) || calendar_so :: split_status ( $status ) === 'R' )
{
unset ( $event [ 'alarm' ][ $id ]);
$this -> so -> delete_alarm ( $id );
2017-10-09 16:59:38 +02:00
//error_log(__LINE__.': '.__METHOD__."(".array2string($event).") deleting alarm=".array2string($alarm).", $status=".array2string($alarm));
2013-08-15 16:56:34 +02:00
}
}
2023-06-09 18:27:28 +02:00
else if ( empty ( $alarm [ 'owner' ]))
2021-04-28 19:20:43 +02:00
{
$alarm [ 'owner' ] = $event [ 'owner' ];
}
2011-03-07 14:49:08 +01:00
}
}
// update all existing alarm times, in case alarm got moved and alarms are not include in $event
2011-11-11 13:46:34 +01:00
if ( $old_event && is_array ( $old_event [ 'alarm' ]) && isset ( $event [ 'start' ]))
2011-03-07 14:49:08 +01:00
{
2015-06-25 22:39:53 +02:00
foreach ( $old_event [ 'alarm' ] as $id => & $alarm )
2011-03-07 14:49:08 +01:00
{
2011-11-11 13:46:34 +01:00
if ( ! isset ( $event [ 'alarm' ][ $id ]))
{
$alarm [ 'time' ] = $event [ 'start' ] - $alarm [ 'offset' ];
2015-06-25 22:39:53 +02:00
if ( $alarm [ 'time' ] < time ()) calendar_so :: shift_alarm ( $event , $alarm );
// remove (not store) alarms belonging to not longer existing or rejected participants
2013-08-15 16:56:34 +02:00
$status = isset ( $event [ 'participants' ]) ? $event [ 'participants' ][ $alarm [ 'owner' ]] :
$old_event [ 'participants' ][ $alarm [ 'owner' ]];
if ( ! $alarm [ 'owner' ] || isset ( $status ) && calendar_so :: split_status ( $status ) !== 'R' )
{
2015-06-25 22:39:53 +02:00
$this -> so -> save_alarm ( $event [ 'id' ], $alarm );
2017-10-09 16:59:38 +02:00
//error_log(__LINE__.': '.__METHOD__."() so->save_alarm($event[id], ".array2string($alarm).")");
2013-08-15 16:56:34 +02:00
}
else
{
$this -> so -> delete_alarm ( $id );
2017-10-09 16:59:38 +02:00
//error_log(__LINE__.': '.__METHOD__."(".array2string($event).") deleting alarm=".array2string($alarm).", $status=".array2string($alarm));
2013-08-15 16:56:34 +02:00
}
2011-11-11 13:46:34 +01:00
}
2005-11-09 00:15:14 +01:00
}
}
2011-04-07 10:01:48 +02:00
2017-01-18 18:29:44 +01:00
// you should always update modification time (ctag depends on it!)
if ( $updateTS )
{
2018-11-21 21:50:29 +01:00
$event [ 'modified' ] = $save_event [ 'modified' ] = $this -> now ;
$event [ 'modifier' ] = $save_event [ 'modifier' ] = $this -> user ;
2017-01-18 18:29:44 +01:00
}
2011-04-07 10:01:48 +02:00
2010-09-12 22:14:46 +02:00
if ( empty ( $event [ 'id' ]) && ( ! isset ( $event [ 'created' ]) || $event [ 'created' ] > $this -> now ))
2010-09-11 20:08:48 +02:00
{
2018-11-21 21:50:29 +01:00
$event [ 'created' ] = $save_event [ 'created' ] = $this -> now ;
$event [ 'creator' ] = $save_event [ 'creator' ] = $this -> user ;
2010-09-11 20:08:48 +02:00
}
2024-07-01 11:39:28 +02:00
$set_recurrences = ! $old_event ? false :
abs ( Api\DateTime :: to ( $event [ 'recur_enddate' ] ? ? null , 'utc' ) - Api\DateTime :: to ( $old_event [ 'recur_enddate' ] ? ? null , 'utc' )) > 1 ||
count ( $old_event [ 'recur_exception' ] ? ? []) != count ( $event [ 'recur_exception' ] ? ? []) ||
count ( $old_event [ 'recur_rdates' ] ? ? []) != count ( $event [ 'recur_rdates' ] ? ? []);
2009-08-10 11:24:39 +02:00
$set_recurrences_start = 0 ;
2024-02-08 10:24:39 +01:00
if (( $cal_id = $this -> so -> save ( $event , $set_recurrences , $set_recurrences_start , 0 , $event [ 'etag' ])) && $set_recurrences && ! empty ( $event [ 'recur_type' ]))
2005-06-14 23:54:17 +02:00
{
$save_event [ 'id' ] = $cal_id ;
2009-08-10 11:24:39 +02:00
// unset participants to enforce the default stati for all added recurrences
unset ( $save_event [ 'participants' ]);
$this -> set_recurrences ( $save_event , $set_recurrences_start );
2005-06-14 23:54:17 +02:00
}
2013-06-04 20:00:39 +02:00
// create links for new participants from addressbook, if configured
2024-02-08 10:24:39 +01:00
if ( $cal_id && ! empty ( $GLOBALS [ 'egw_info' ][ 'server' ][ 'link_contacts' ]) && ! empty ( $save_event [ 'participants' ]))
2013-06-04 20:00:39 +02:00
{
2023-07-11 23:17:37 +02:00
foreach ( $save_event [ 'participants' ] as $uid => $status )
2013-06-04 20:00:39 +02:00
{
2015-06-25 22:39:53 +02:00
$user_type = $user_id = null ;
2013-06-04 20:00:39 +02:00
calendar_so :: split_user ( $uid , $user_type , $user_id );
2023-07-11 23:17:37 +02:00
if ( $user_type == 'c' && ( ! $old_event || ! isset ( $old_event [ 'participants' ][ $uid ])))
2013-06-04 20:00:39 +02:00
{
2016-05-01 19:47:59 +02:00
Link :: link ( 'calendar' , $cal_id , 'addressbook' , $user_id );
2013-06-04 20:00:39 +02:00
}
}
}
2010-02-12 21:18:45 +01:00
// Update history
$tracking = new calendar_tracking ( $this );
2022-03-07 18:12:58 +01:00
if ( empty ( $event [ 'id' ]) && ! empty ( $cal_id ))
{
$event [ 'id' ] = $cal_id ;
$save_event [ 'id' ] = $cal_id ;
}
2020-07-17 14:53:09 +02:00
// we run all dates through date2ts, to adjust to server-time and the possible date-formats
// This is done here to avoid damaging the actual event when saving, but the old event is in server-time
foreach ( $timestamps as $ts )
2020-06-24 23:07:52 +02:00
{
2020-07-17 14:53:09 +02:00
// we convert here from user-time to timestamps in server-time!
if ( isset ( $save_event [ $ts ])) $save_event [ $ts ] = $save_event [ $ts ] ? calendar_bo :: date2ts ( $save_event [ $ts ], true ) : 0 ;
}
2022-03-07 18:12:58 +01:00
foreach ([ 'start' , 'end' , 'recur_enddate' ] as $ts )
2021-03-17 16:01:53 +01:00
{
2023-06-09 18:27:28 +02:00
if ( isset ( $save_event [ $ts ]) && is_object ( $save_event [ $ts ]))
2021-03-17 16:01:53 +01:00
{
$save_event [ $ts ] = $save_event [ $ts ] -> format ( 'ts' );
}
}
2021-11-18 17:35:26 +01:00
$tracking -> track ( $save_event , $old_event ? : null );
2010-02-12 21:18:45 +01:00
2005-06-14 23:54:17 +02:00
return $cal_id ;
}
2008-05-08 00:12:25 +02:00
2006-12-22 20:51:56 +01:00
/**
* Check if the current user has the necessary ACL rights to change the status of $uid
2008-05-08 00:12:25 +02:00
*
2006-12-22 20:51:56 +01:00
* For contacts we use edit rights of the owner of the event ( aka . edit rights of the event ) .
*
2009-08-04 19:14:16 +02:00
* @ param int | string $uid account_id or 1 - char type - identifer plus id ( eg . c15 for addressbook entry #15)
* @ param array | int $event event array or id of the event
2006-12-22 20:51:56 +01:00
* @ return boolean
*/
function check_status_perms ( $uid , $event )
{
2009-12-07 14:31:51 +01:00
if ( $uid [ 0 ] == 'c' || $uid [ 0 ] == 'e' ) // for contact we use the owner of the event
2006-12-22 20:51:56 +01:00
{
if ( ! is_array ( $event ) && ! ( $event = $this -> read ( $event ))) return false ;
2016-05-01 19:47:59 +02:00
return $this -> check_perms ( Acl :: EDIT , 0 , $event [ 'owner' ]);
2006-12-22 20:51:56 +01:00
}
2016-05-01 19:47:59 +02:00
// check if we have a category Acl for the event or not (null)
2009-12-07 14:31:51 +01:00
$access = $this -> check_cat_acl ( self :: CAT_ACL_STATUS , $event );
if ( ! is_null ( $access ))
2009-12-03 19:24:19 +01:00
{
2009-12-07 14:31:51 +01:00
return $access ;
2009-12-03 19:24:19 +01:00
}
2009-12-07 14:31:51 +01:00
// no access or denied access because of category acl --> regular check
2006-12-22 20:51:56 +01:00
if ( ! is_numeric ( $uid )) // this is eg. for resources (r123)
{
$resource = $this -> resource_info ( $uid );
2016-05-01 19:47:59 +02:00
return Acl :: EDIT & $resource [ 'rights' ];
2006-12-22 20:51:56 +01:00
}
2011-11-09 10:44:28 +01:00
if ( ! is_array ( $event ) && ! ( $event = $this -> read ( $event ))) return false ;
2011-11-21 15:25:44 +01:00
// regular user and groups (need to check memberships too)
if ( ! isset ( $event [ 'participants' ][ $uid ]))
{
$memberships = $GLOBALS [ 'egw' ] -> accounts -> memberships ( $uid , true );
}
$memberships [] = $uid ;
2021-10-08 14:58:27 +02:00
return array_intersect ( $memberships , array_keys ( $event [ 'participants' ] ? ? [])) && $this -> check_perms ( Acl :: EDIT , 0 , $uid );
2006-12-22 20:51:56 +01:00
}
2009-12-03 19:24:19 +01:00
/**
* Check if current user has a certain right on the categories of an event
*
* Not having the given right for a single category , means not having it !
*
* @ param int $right self :: CAT_ACL_ { ADD | STATUS }
* @ param int | array $event
* @ return boolean true if use has the right , false if not
2009-12-07 14:31:51 +01:00
* @ return boolean false = access denied because of cat acl , true access granted because of cat acl ,
* null = cat has no acl
2009-12-03 19:24:19 +01:00
*/
function check_cat_acl ( $right , $event )
{
if ( ! is_array ( $event )) $event = $this -> read ( $event );
2009-12-07 14:31:51 +01:00
$ret = null ;
2009-12-03 19:24:19 +01:00
if ( $event [ 'category' ])
{
foreach ( is_array ( $event [ 'category' ]) ? $event [ 'category' ] : explode ( ',' , $event [ 'category' ]) as $cat_id )
{
2009-12-07 14:31:51 +01:00
$access = self :: has_cat_right ( $right , $cat_id , $this -> user );
if ( $access === true )
2009-12-03 19:24:19 +01:00
{
2009-12-07 14:31:51 +01:00
$ret = true ;
2009-12-03 19:24:19 +01:00
break ;
}
2009-12-07 14:31:51 +01:00
if ( $access === false )
{
$ret = false ; // cat denies access --> check further cats
}
2009-12-03 19:24:19 +01:00
}
}
//echo "<p>".__METHOD__."($event[id]: $event[title], $right) = ".array2string($ret)."</p>\n";
return $ret ;
}
/**
* Array with $cat_id => $rights pairs for current user ( no entry means , cat is not limited by ACL ! )
*
* @ var array
*/
private static $cat_rights_cache ;
/**
* Get rights for a given category id
*
2015-06-25 22:39:53 +02:00
* @ param int $cat_id = null null to return array with all cats
2009-12-03 19:24:19 +01:00
* @ return array with account_id => right pairs
*/
public static function get_cat_rights ( $cat_id = null )
{
if ( ! isset ( self :: $cat_rights_cache ))
{
2016-05-01 19:47:59 +02:00
self :: $cat_rights_cache = Api\Cache :: getSession ( 'calendar' , 'cat_rights' ,
2009-12-03 19:24:19 +01:00
array ( $GLOBALS [ 'egw' ] -> acl , 'get_location_grants' ), array ( 'L%' , 'calendar' ));
}
//echo "<p>".__METHOD__."($cat_id) = ".array2string($cat_id ? self::$cat_rights_cache['L'.$cat_id] : self::$cat_rights_cache)."</p>\n";
return $cat_id ? self :: $cat_rights_cache [ 'L' . $cat_id ] : self :: $cat_rights_cache ;
}
/**
* Set rights for a given single category and user
*
* @ param int $cat_id
* @ param int $user
* @ param int $rights self :: CAT_ACL_ { ADD | STATUS } or ' ed together
*/
public static function set_cat_rights ( $cat_id , $user , $rights )
{
//echo "<p>".__METHOD__."($cat_id,$user,$rights)</p>\n";
if ( ! isset ( self :: $cat_rights_cache )) self :: get_cat_rights ( $cat_id );
if (( int ) $rights != ( int ) self :: $cat_rights_cache [ 'L' . $cat_id ][ $user ])
{
if ( $rights )
{
self :: $cat_rights_cache [ 'L' . $cat_id ][ $user ] = $rights ;
$GLOBALS [ 'egw' ] -> acl -> add_repository ( 'calendar' , 'L' . $cat_id , $user , $rights );
}
else
{
unset ( self :: $cat_rights_cache [ 'L' . $cat_id ][ $user ]);
if ( ! self :: $cat_rights_cache [ 'L' . $cat_id ]) unset ( self :: $cat_rights_cache [ 'L' . $cat_id ]);
$GLOBALS [ 'egw' ] -> acl -> delete_repository ( 'calendar' , 'L' . $cat_id , $user );
}
2016-05-01 19:47:59 +02:00
Api\Cache :: setSession ( 'calendar' , 'cat_rights' , self :: $cat_rights_cache );
2009-12-03 19:24:19 +01:00
}
}
/**
* Check if current user has a given right on a category ( if it ' s restricted ! )
*
* @ param int $cat_id
2009-12-07 14:31:51 +01:00
* @ return boolean false = access denied because of cat acl , true access granted because of cat acl ,
* null = cat has no acl
2009-12-03 19:24:19 +01:00
*/
public static function has_cat_right ( $right , $cat_id , $user )
{
2015-06-25 22:39:53 +02:00
static $cache = null ;
2009-12-03 19:24:19 +01:00
2009-12-07 14:31:51 +01:00
if ( ! isset ( $cache [ $cat_id ]))
2009-12-03 19:24:19 +01:00
{
$all = $own = 0 ;
$cat_rights = self :: get_cat_rights ( $cat_id );
if ( ! is_null ( $cat_rights ))
{
2015-06-25 22:39:53 +02:00
static $memberships = null ;
2009-12-03 19:24:19 +01:00
if ( is_null ( $memberships ))
{
2021-11-20 08:40:35 +01:00
$memberships = $GLOBALS [ 'egw' ] -> accounts -> memberships ( $user , true ) ? : [];
2009-12-03 19:24:19 +01:00
$memberships [] = $user ;
}
foreach ( $cat_rights as $uid => $value )
{
$all |= $value ;
if ( in_array ( $uid , $memberships )) $own |= $value ;
}
}
foreach ( array ( self :: CAT_ACL_ADD , self :: CAT_ACL_STATUS ) as $mask )
{
2009-12-07 14:31:51 +01:00
$cache [ $cat_id ][ $mask ] = ! ( $all & $mask ) ? null : !! ( $own & $mask );
2009-12-03 19:24:19 +01:00
}
}
//echo "<p>".__METHOD__."($right,$cat_id) all=$all, own=$own returning ".array2string($cache[$cat_id][$right])."</p>\n";
return $cache [ $cat_id ][ $right ];
}
2005-06-14 23:54:17 +02:00
/**
2005-11-09 00:15:14 +01:00
* set the status of one participant for a given recurrence or for all recurrences since now ( includes recur_date = 0 )
2005-06-14 23:54:17 +02:00
*
2009-08-04 19:14:16 +02:00
* @ param int | array $event event - array or id of the event
* @ param string | int $uid account_id or 1 - char type - identifer plus id ( eg . c15 for addressbook entry #15)
* @ param int | char $status numeric status ( defines ) or 1 - char code : 'R' , 'U' , 'T' or 'A'
2015-10-14 18:39:39 +02:00
* @ param int $recur_date = 0 date to change , or 0 = all since now
* @ param boolean $ignore_acl = false do not check the permisions for the $uid , if true
* @ param boolean $updateTS = true update the content history of the event
2022-06-03 09:57:33 +02:00
* DEPRECATED : we always ( have to ) update timestamp , as they are required for sync !
* @ param boolean | " NOPUSH " $skip_notification = false true : send NO notifications , default false = send them ,
* " NOPUSH " : also do NOT send push notifications / call Link :: notifiy (), which still happens for true
2024-07-06 19:39:21 +02:00
* @ param ? string $comment Comment to send with notification to organizer ( as iCal COMMENT )
2005-11-09 00:15:14 +01:00
* @ return int number of changed recurrences
*/
2024-07-06 19:39:21 +02:00
function set_status ( $event , $uid , $status , $recur_date = 0 , $ignore_acl = false , $updateTS = true , $skip_notification = false , ? string $comment = null )
2005-11-09 00:15:14 +01:00
{
2015-10-14 18:39:39 +02:00
unset ( $updateTS );
2006-12-22 20:51:56 +01:00
$cal_id = is_array ( $event ) ? $event [ 'id' ] : $event ;
2009-08-04 19:14:16 +02:00
//echo "<p>calendar_boupdate::set_status($cal_id,$uid,$status,$recur_date)</p>\n";
2009-07-15 22:35:56 +02:00
if ( ! $cal_id || ( ! $ignore_acl && ! $this -> check_status_perms ( $uid , $event )))
2005-11-09 00:15:14 +01:00
{
return false ;
}
2024-07-08 16:39:40 +02:00
// make sure to not set a recur_date for a non-recurring event (as recur_date has to be 0, for non-recurring events!)
2024-08-22 11:56:34 +02:00
if ( $recur_date && ( is_array ( $event ) || ( $event = $this -> read ( $cal_id , null , $ignore_acl ))) && empty ( $event [ 'recur_type' ]))
2024-07-08 16:39:40 +02:00
{
$recur_date = 0 ;
}
2015-06-25 22:39:53 +02:00
$quantity = $role = null ;
2009-11-26 19:36:19 +01:00
calendar_so :: split_status ( $status , $quantity , $role );
2010-01-29 22:42:54 +01:00
if ( $this -> log )
{
error_log ( __FILE__ . '[' . __LINE__ . '] ' . __METHOD__ .
2010-03-09 18:03:41 +01:00
" ( $cal_id , $uid , $status , $recur_date ) \n " , 3 , $this -> logfile );
2010-01-29 22:42:54 +01:00
}
2017-01-19 11:11:01 +01:00
$old_event = $this -> read ( $cal_id , $recur_date , $ignore_acl , 'server' );
2010-03-07 00:06:43 +01:00
if (( $Ok = $this -> so -> set_status ( $cal_id , is_numeric ( $uid ) ? 'u' : $uid [ 0 ],
is_numeric ( $uid ) ? $uid : substr ( $uid , 1 ), $status ,
$recur_date ? $this -> date2ts ( $recur_date , true ) : 0 , $role )))
2005-11-09 00:15:14 +01:00
{
2013-08-15 16:56:34 +02:00
if ( $status == 'R' ) // remove alarms belonging to rejected participants
{
2017-01-19 11:11:01 +01:00
foreach ( is_array ( $event ) && isset ( $event [ 'alarm' ]) ? $event [ 'alarm' ] : $old_event [ 'alarm' ] as $id => $alarm )
2013-08-15 16:56:34 +02:00
{
if (( string ) $alarm [ 'owner' ] === ( string ) $uid )
{
$this -> so -> delete_alarm ( $id );
2014-02-20 16:04:47 +01:00
//error_log(__LINE__.': '.__METHOD__."(".array2string($event).", '$uid', '$status', ...) deleting alarm=".array2string($alarm).", $status=".array2string($alarm));
2013-08-15 16:56:34 +02:00
}
}
}
2010-11-11 09:51:13 +01:00
2005-11-09 00:15:14 +01:00
static $status2msg = array (
'R' => MSG_REJECTED ,
'T' => MSG_TENTATIVE ,
'A' => MSG_ACCEPTED ,
2010-02-17 14:29:28 +01:00
'D' => MSG_DELEGATED ,
2005-11-09 00:15:14 +01:00
);
2017-10-11 23:10:59 +02:00
// Reset cached event
static :: $cached_event = array ();
2017-12-06 21:47:54 +01:00
2011-03-09 18:26:20 +01:00
if ( isset ( $status2msg [ $status ]) && ! $skip_notification )
2005-11-09 00:15:14 +01:00
{
2006-12-22 20:51:56 +01:00
if ( ! is_array ( $event )) $event = $this -> read ( $cal_id );
2007-02-07 14:51:07 +01:00
if ( isset ( $recur_date )) $event = $this -> read ( $event [ 'id' ], $recur_date ); //re-read the actually edited recurring event
2024-07-06 19:39:21 +02:00
if ( ! empty ( $comment )) $event [ 'comment' ] = $comment ;
2019-12-18 19:54:39 +01:00
$user_id = is_numeric ( $uid ) ? ( int ) $uid : $uid ;
2019-11-20 23:16:16 +01:00
$this -> send_update ( $status2msg [ $status ], $event [ 'participants' ], $event , null , $user_id );
2005-11-09 00:15:14 +01:00
}
2010-02-12 21:18:45 +01:00
// Update history
2021-11-10 13:00:59 +01:00
if (( $event = $this -> read ( $cal_id , $recur_date , $ignore_acl , 'server' )))
{
$tracking = new calendar_tracking ( $this );
2021-11-18 17:35:26 +01:00
$tracking -> track ( $event , $old_event ? : null );
2021-11-10 13:00:59 +01:00
}
2020-07-15 21:32:10 +02:00
// notify the link-class about the update, as other apps may be subscribed to it
2022-06-03 09:57:33 +02:00
if ( $skip_notification !== " NOPUSH " )
{
Link :: notify_update ( 'calendar' , $cal_id , $event , " update " );
}
2005-11-09 00:15:14 +01:00
}
2005-11-23 15:21:20 +01:00
return $Ok ;
2005-11-09 00:15:14 +01:00
}
2010-02-26 13:37:07 +01:00
/**
* update the status of all participant for a given recurrence or for all recurrences since now ( includes recur_date = 0 )
*
* @ param array $new_event event - array with the new stati
* @ param array $old_event event - array with the old stati
2015-06-25 22:39:53 +02:00
* @ param int $recur_date = 0 date to change , or 0 = all since now
2022-06-03 09:57:33 +02:00
* @ param boolean | " NOPUSH " $skip_notification = false true : send NO notifications , default false = send them ,
* " NOPUSH " : also do NOT send push notifications / call Link :: notifiy (), which still happens for true
2010-02-26 13:37:07 +01:00
*/
2012-12-18 17:00:29 +01:00
function update_status ( $new_event , $old_event , $recur_date = 0 , $skip_notification = false )
2010-02-26 13:37:07 +01:00
{
if ( ! isset ( $new_event [ 'participants' ])) return ;
// check the old list against the new list
foreach ( $old_event [ 'participants' ] as $userid => $status )
{
if ( ! isset ( $new_event [ 'participants' ][ $userid ])){
// Attendee will be deleted this way
$new_event [ 'participants' ][ $userid ] = 'G' ;
}
elseif ( $new_event [ 'participants' ][ $userid ] == $status )
{
// Same status -- nothing to do.
unset ( $new_event [ 'participants' ][ $userid ]);
}
}
// write the changes
foreach ( $new_event [ 'participants' ] as $userid => $status )
{
2012-12-18 17:00:29 +01:00
$this -> set_status ( $old_event , $userid , $status , $recur_date , true , false , $skip_notification );
2010-02-26 13:37:07 +01:00
}
2020-07-14 21:39:45 +02:00
// notify the link-class about the update, as other apps may be subscribed to it
2022-06-03 09:57:33 +02:00
if ( $skip_notification !== " NOPUSH " )
{
Link :: notify_update ( 'calendar' , $new_event [ 'id' ], $new_event , " update " );
}
2020-07-14 21:39:45 +02:00
}
2010-02-26 13:37:07 +01:00
2005-11-09 00:15:14 +01:00
/**
* deletes an event
*
* @ param int $cal_id id of the event to delete
2015-06-25 22:39:53 +02:00
* @ param int $recur_date = 0 if a single event from a series should be deleted , its date
* @ param boolean $ignore_acl = false true for no ACL check , default do ACL check
2022-06-03 09:57:33 +02:00
* @ param boolean | array | " NOPUSH " $skip_notification = false or array with uid to skip eg . [ 5 , 'eemail@example.org' ]
* " NOPUSH " : also do NOT send push notifications / call Link :: notifiy (), which still happens for true
2015-06-25 22:39:53 +02:00
* @ param boolean $delete_exceptions = true true : delete , false : keep exceptions ( with new UID )
2012-10-22 15:15:40 +02:00
* @ param int & $exceptions_kept = null on return number of kept exceptions
2005-11-09 00:15:14 +01:00
* @ return boolean true on success , false on error ( usually permission denied )
2005-06-14 23:54:17 +02:00
*/
2012-10-22 15:15:40 +02:00
function delete ( $cal_id , $recur_date = 0 , $ignore_acl = false , $skip_notification = false ,
$delete_exceptions = true , & $exceptions_kept = null )
2005-06-14 23:54:17 +02:00
{
2012-08-11 12:01:02 +02:00
//error_log(__METHOD__."(cal_id=$cal_id, recur_date=$recur_date, ignore_acl=$ignore_acl, skip_notifications=$skip_notification)");
2005-11-09 00:15:14 +01:00
if ( ! ( $event = $this -> read ( $cal_id , $recur_date )) ||
2016-05-01 19:47:59 +02:00
! $ignore_acl && ! $this -> check_perms ( Acl :: DELETE , $event ))
2005-11-09 00:15:14 +01:00
{
return false ;
}
2010-06-28 13:15:25 +02:00
// Don't send notification if the event has already been deleted
2020-06-16 16:53:43 +02:00
if ( ! $event [ 'deleted' ] && ( ! $skip_notification || is_array ( $skip_notification )))
2010-06-28 13:15:25 +02:00
{
2020-06-16 16:53:43 +02:00
$to_notify = ! is_array ( $skip_notification ) ? $event [ 'participants' ] :
array_diff_key ( $event [ 'participants' ], array_flip ( $skip_notification ));
$this -> send_update ( MSG_DELETED , $to_notify , $event );
2010-06-28 13:15:25 +02:00
}
2008-05-08 00:12:25 +02:00
2024-02-08 10:24:39 +01:00
if ( ! $recur_date || empty ( $event [ 'recur_type' ]))
2005-11-09 00:15:14 +01:00
{
2016-05-01 19:47:59 +02:00
$config = Api\Config :: read ( 'phpgwapi' );
2021-05-18 13:11:14 +02:00
if ( $event [ 'deleted' ])
2010-04-22 18:09:36 +02:00
{
2010-03-05 21:01:44 +01:00
$this -> so -> delete ( $cal_id );
2010-09-02 22:55:37 +02:00
// delete all links to the event
2016-05-01 19:47:59 +02:00
Link :: unlink ( 0 , 'calendar' , $cal_id );
2010-03-05 21:01:44 +01:00
}
2021-07-07 11:21:45 +02:00
else
2010-04-22 18:09:36 +02:00
{
2010-09-27 20:03:46 +02:00
// mark all links to the event as deleted, but keep them
2016-05-01 19:47:59 +02:00
Link :: unlink ( 0 , 'calendar' , $cal_id , '' , '' , '' , true );
2010-09-27 20:03:46 +02:00
2011-04-07 10:01:48 +02:00
$event [ 'deleted' ] = $this -> now ;
2010-06-01 09:36:14 +02:00
$this -> save ( $event , $ignore_acl );
2010-05-28 09:24:34 +02:00
// Actually delete alarms
if ( isset ( $event [ 'alarm' ]) && is_array ( $event [ 'alarm' ]))
{
foreach ( $event [ 'alarm' ] as $id => $alarm )
{
2023-06-06 19:59:43 +02:00
$this -> delete_alarm ( $id , true );
2010-05-28 09:24:34 +02:00
}
}
2010-04-22 18:09:36 +02:00
}
2012-10-22 15:15:40 +02:00
// delete or keep (with new uid) exceptions of a recurring event
2024-02-08 10:24:39 +01:00
if ( ! empty ( $event [ 'recur_type' ]))
2012-10-22 15:15:40 +02:00
{
$exceptions_kept = 0 ;
foreach ( $this -> so -> get_related ( $event [ 'uid' ]) as $id )
{
if ( $delete_exceptions )
{
$this -> delete ( $id , 0 , $ignore_acl , true );
}
else
{
if ( ! ( $exception = $this -> read ( $id ))) continue ;
2016-05-01 19:47:59 +02:00
$exception [ 'uid' ] = Api\CalDAV :: generate_uid ( 'calendar' , $id );
2012-10-22 15:15:40 +02:00
$exception [ 'reference' ] = $exception [ 'recurrence' ] = 0 ;
2021-11-05 13:52:20 +01:00
$msg = null ;
$this -> update ( $exception , true , true , false , true , $msg , true );
2012-10-22 15:15:40 +02:00
++ $exceptions_kept ;
}
}
}
2005-11-09 00:15:14 +01:00
}
2011-04-07 10:01:48 +02:00
else // delete an exception
2005-11-09 00:15:14 +01:00
{
2012-08-14 19:02:34 +02:00
// check if deleted recurrance has alarms (because it's the next recurrance) --> move it to next recurrance
if ( $event [ 'alarm' ])
{
2015-06-25 22:39:53 +02:00
$next_recurrance = null ;
2012-08-14 19:02:34 +02:00
foreach ( $event [ 'alarm' ] as & $alarm )
{
2012-12-07 15:10:51 +01:00
if (( $alarm [ 'time' ] == $recur_date ) || ( $alarm [ 'time' ] + $alarm [ 'offset' ] == $recur_date ))
2012-08-14 19:02:34 +02:00
{
2012-12-07 15:10:51 +01:00
//error_log(__METHOD__.__LINE__.'->'.array2string($recur_date));
//error_log(__METHOD__.__LINE__.array2string($event));
2012-08-14 19:02:34 +02:00
if ( is_null ( $next_recurrance ))
{
2012-12-07 15:10:51 +01:00
$checkdate = $recur_date ;
//if ($alarm['time']+$alarm['offset'] == $recur_date) $checkdate = $recur_date + $alarm['offset'];
2015-06-25 22:39:53 +02:00
if (( $e = $this -> read ( $cal_id , $checkdate + 1 )))
2012-12-07 15:10:51 +01:00
{
$next_recurrance = $this -> date2ts ( $e [ 'start' ]);
}
2012-08-14 19:02:34 +02:00
}
$alarm [ 'time' ] = $this -> date2ts ( $next_recurrance , true ); // user to server-time
2012-12-07 15:10:51 +01:00
$alarm [ 'cal_id' ] = $cal_id ;
unset ( $alarm [ 'times' ]);
unset ( $alarm [ 'next' ]);
2012-08-14 19:02:34 +02:00
$this -> so -> save_alarm ( $event [ 'id' ], $alarm );
}
}
}
2012-08-11 12:01:02 +02:00
// need to read series master, as otherwise existing exceptions will be lost!
$recur_date = $this -> date2ts ( $event [ 'start' ]);
2012-12-07 15:10:51 +01:00
//if ($event['alarm']) $alarmbuffer = $event['alarm'];
2012-08-11 12:01:02 +02:00
$event = $this -> read ( $cal_id );
2012-12-07 15:10:51 +01:00
//if (isset($alarmbuffer)) $event['alarm'] = $alarmbuffer;
2012-08-11 12:01:02 +02:00
$event [ 'recur_exception' ][] = $recur_date ;
2012-12-07 15:10:51 +01:00
$this -> save ( $event ); // updates the content-history
2020-08-19 09:59:53 +02:00
2022-06-03 09:57:33 +02:00
// for "real" push, need to push delete of recurrence
if ( ! Api\Json\Push :: onlyFallback () && $skip_notification !== " NOPUSH " )
2020-08-19 09:59:53 +02:00
{
Api\Link :: notify_update ( 'calendar' , $cal_id . ':' . $recur_date , $event , 'delete' );
}
2005-11-09 00:15:14 +01:00
}
2009-07-23 18:14:22 +02:00
if ( $event [ 'reference' ])
{
// evtl. delete recur_exception $event['recurrence'] from event with cal_id=$event['reference']
}
2005-06-14 23:54:17 +02:00
return true ;
}
2005-11-09 00:15:14 +01:00
/**
* helper for send_update and get_update_message
* @ internal
2015-06-25 22:39:53 +02:00
* @ param array $event
* @ param string $action
* @ param array $event_arr
* @ param array $disinvited
* @ return array
2005-11-09 00:15:14 +01:00
*/
function _get_event_details ( $event , $action , & $event_arr , $disinvited = array ())
{
$details = array ( // event-details for the notify-msg
'id' => $event [ 'id' ],
2012-05-15 14:43:32 +02:00
'action' => lang ( $action ),
2005-11-09 00:15:14 +01:00
);
$event_arr = $this -> event2array ( $event );
foreach ( $event_arr as $key => $val )
{
2024-07-02 15:13:42 +02:00
if ( $key == 'recur_type' ) $details [ 'repetition' ] = $val [ 'data' ];
2005-11-09 00:15:14 +01:00
$details [ $key ] = $val [ 'data' ];
}
$details [ 'participants' ] = $details [ 'participants' ] ? implode ( " \n " , $details [ 'participants' ]) : '' ;
2006-03-21 12:25:31 +01:00
$event_arr [ 'link' ][ 'field' ] = lang ( 'URL' );
2007-02-10 19:49:55 +01:00
$eventStart_arr = $this -> date2array ( $event [ 'start' ]); // give this as 'date' to the link to pick the right recurrence for the participants state
2017-06-22 19:03:28 +02:00
$link = $GLOBALS [ 'egw_info' ][ 'server' ][ 'webserver_url' ] . '/index.php?menuaction=calendar.calendar_uiforms.edit&cal_id=' . $event [ 'id' ] . '&date=' . $eventStart_arr [ 'full' ] . '&no_popup=1&ajax=true' ;
2005-11-09 00:15:14 +01:00
// if url is only a path, try guessing the rest ;-)
2018-03-21 12:40:59 +01:00
if ( $link [ 0 ] == '/' ) $link = Api\Framework :: getUrl ( $link );
2006-07-28 18:30:16 +02:00
$event_arr [ 'link' ][ 'data' ] = $details [ 'link' ] = $link ;
2008-05-08 00:12:25 +02:00
2007-11-22 09:29:16 +01:00
/* this is needed for notification - app
* notification - app creates the link individual for
* every user , so we must provide a neutral link - style
2008-01-30 19:58:00 +01:00
* if calendar implements tracking in near future , this part can be deleted
2007-11-22 09:29:16 +01:00
*/
$link_arr = array ();
$link_arr [ 'text' ] = $event [ 'title' ];
2008-06-12 09:06:29 +02:00
$link_arr [ 'view' ] = array ( 'menuaction' => 'calendar.calendar_uiforms.edit' ,
2008-01-30 19:58:00 +01:00
'cal_id' => $event [ 'id' ],
'date' => $eventStart_arr [ 'full' ],
2017-09-28 18:38:18 +02:00
'ajax' => true
2008-01-30 19:58:00 +01:00
);
$link_arr [ 'popup' ] = '750x400' ;
2007-11-22 09:29:16 +01:00
$details [ 'link_arr' ] = $link_arr ;
2008-05-08 00:12:25 +02:00
2005-11-09 00:15:14 +01:00
$dis = array ();
foreach ( $disinvited as $uid )
{
$dis [] = $this -> participant_name ( $uid );
}
$details [ 'disinvited' ] = implode ( ', ' , $dis );
return $details ;
}
/**
* create array with name , translated name and readable content of each attributes of an event
*
* old function , so far only used by send_update ( therefor it ' s in bocalupdate and not bocal )
*
* @ param array $event event to use
* @ returns array of attributes with fieldname as key and array with the 'field' = translated name 'data' = readable content ( for participants this is an array ! )
*/
function event2array ( $event )
{
$var [ 'title' ] = Array (
'field' => lang ( 'Title' ),
'data' => $event [ 'title' ]
);
$var [ 'description' ] = Array (
'field' => lang ( 'Description' ),
'data' => $event [ 'description' ]
);
2021-11-16 08:34:51 +01:00
foreach ( is_array ( $event [ 'category' ]) ? $event [ 'category' ] : explode ( ',' , $event [ 'category' ]) as $cat_id )
2005-11-09 00:15:14 +01:00
{
2016-05-05 18:56:25 +02:00
$cat_string [] = stripslashes ( Api\Categories :: id2name ( $cat_id ));
2005-11-09 00:15:14 +01:00
}
$var [ 'category' ] = Array (
'field' => lang ( 'Category' ),
'data' => implode ( ', ' , $cat_string )
);
$var [ 'location' ] = Array (
'field' => lang ( 'Location' ),
'data' => $event [ 'location' ]
);
$var [ 'startdate' ] = Array (
'field' => lang ( 'Start Date/Time' ),
'data' => $this -> format_date ( $event [ 'start' ]),
);
$var [ 'enddate' ] = Array (
'field' => lang ( 'End Date/Time' ),
'data' => $this -> format_date ( $event [ 'end' ]),
);
$pri = Array (
2013-01-31 15:13:45 +01:00
0 => lang ( 'None' ),
2005-11-09 00:15:14 +01:00
1 => lang ( 'Low' ),
2 => lang ( 'Normal' ),
3 => lang ( 'High' )
);
$var [ 'priority' ] = Array (
'field' => lang ( 'Priority' ),
'data' => $pri [ $event [ 'priority' ]]
);
$var [ 'owner' ] = Array (
'field' => lang ( 'Owner' ),
2016-05-01 19:47:59 +02:00
'data' => Api\Accounts :: username ( $event [ 'owner' ])
2005-11-09 00:15:14 +01:00
);
$var [ 'updated' ] = Array (
'field' => lang ( 'Updated' ),
2022-04-26 21:04:16 +02:00
'data' => $this -> format_date ( $event [ 'modtime' ] ? ? null ) . ', ' . Api\Accounts :: username ( $event [ 'modifier' ])
2005-11-09 00:15:14 +01:00
);
$var [ 'access' ] = Array (
'field' => lang ( 'Access' ),
'data' => $event [ 'public' ] ? lang ( 'Public' ) : lang ( 'Private' )
);
2014-03-20 17:10:52 +01:00
if ( isset ( $event [ 'participants' ]) && is_array ( $event [ 'participants' ]) && ! empty ( $event [ 'participants' ]))
2005-11-09 00:15:14 +01:00
{
$participants = $this -> participants ( $event , true );
2018-06-27 20:22:04 +02:00
// fix external organiser to not be included as participant and shown as organiser
foreach ( $event [ 'participants' ] as $uid => $status )
{
$role = $quantity = null ;
calendar_so :: split_status ( $status , $quantity , $role );
if ( $role == 'CHAIR' && $status == 'D' && ! is_numeric ( $uid ))
{
$var [ 'owner' ][ 'data' ] = $this -> participant_name ( $uid );
unset ( $participants [ $uid ]);
break ;
}
}
2005-11-09 00:15:14 +01:00
}
2018-06-27 20:22:04 +02:00
2005-11-09 00:15:14 +01:00
$var [ 'participants' ] = Array (
'field' => lang ( 'Participants' ),
'data' => $participants
);
// Repeated Events
2015-03-23 20:58:58 +01:00
$var [ 'recur_type' ] = Array (
'field' => lang ( 'Repetition' ),
2024-02-08 10:24:39 +01:00
'data' => ( ! empty ( $event [ 'recur_type' ])) ? $this -> recure2string ( $event ) : '' ,
2015-03-23 20:58:58 +01:00
);
2005-11-09 00:15:14 +01:00
return $var ;
}
/**
* log all updates to a file
*
* @ param array $event2save event - data before calling save
* @ param array $event_saved event - data read back from the DB
2015-06-25 22:39:53 +02:00
* @ param array $old_event = null event - data in the DB before calling save
* @ param string $type = 'update'
2005-11-09 00:15:14 +01:00
*/
function log2file ( $event2save , $event_saved , $old_event = null , $type = 'update' )
{
if ( ! ( $f = fopen ( $this -> log_file , 'a' )))
{
echo " <p>error opening ' $this->log_file ' !!!</p> \n " ;
return false ;
}
2016-05-01 19:47:59 +02:00
fwrite ( $f , $type . ': ' . Api\Accounts :: username ( $this -> user ) . ': ' . date ( 'r' ) . " \n " );
2005-11-09 00:15:14 +01:00
fwrite ( $f , " Time: time to save / saved time read back / old time before save \n " );
foreach ( array ( 'start' , 'end' ) as $name )
{
fwrite ( $f , $name . ': ' . ( isset ( $event2save [ $name ]) ? $this -> format_date ( $event2save [ $name ]) : 'not set' ) . ' / ' .
$this -> format_date ( $event_saved [ $name ]) . ' / ' .
( is_null ( $old_event ) ? 'no old event' : $this -> format_date ( $old_event [ $name ])) . " \n " );
}
foreach ( array ( 'event2save' , 'event_saved' , 'old_event' ) as $name )
{
fwrite ( $f , $name . ' = ' . print_r ( $$name , true ));
}
fwrite ( $f , " \n " );
fclose ( $f );
return true ;
}
2016-01-18 18:45:25 +01:00
/**
* Check alarms and move them if needed
*
* Used when the start time has changed , and alarms need to be updated
*
* @ param array $event
* @ param array $old_event
2022-07-26 14:13:17 +02:00
* @ param Api\DateTime | int | null $instance_date For recurring events , this is the date we are dealing with
* @ param boolean $ignore_acl = false true : no acl check
2016-01-18 18:45:25 +01:00
*/
2022-07-26 14:13:17 +02:00
function check_move_alarms ( Array & $event , Array $old_event = null , $instance_date = null , $ignore_acl = false )
2016-01-18 18:45:25 +01:00
{
if ( $old_event !== null && $event [ 'start' ] == $old_event [ 'start' ]) return ;
2016-05-01 19:47:59 +02:00
$time = new Api\DateTime ( $event [ 'start' ]);
2016-01-18 18:45:25 +01:00
if ( ! is_array ( $event [ 'alarm' ]))
{
$event [ 'alarm' ] = $this -> so -> read_alarms ( $event [ 'id' ]);
}
2018-06-27 18:26:42 +02:00
if ( is_object ( $instance_date ))
{
if ( ! is_a ( $instance_date , 'EGroupware\\Api\\DateTime' ))
{
throw new Api\Exception\WrongParameter ( '$instance_date must be integer or Api\DateTime!' );
}
$instance_date = $instance_date -> format ( 'ts' );
}
2016-05-01 19:47:59 +02:00
foreach ( $event [ 'alarm' ] as & $alarm )
2016-01-18 18:45:25 +01:00
{
2024-02-08 10:24:39 +01:00
if ( ! empty ( $event [ 'recur_type' ]) && $instance_date )
2016-01-18 18:45:25 +01:00
{
2018-06-27 18:26:42 +02:00
calendar_so :: shift_alarm ( $event , $alarm , $instance_date );
2016-01-18 18:45:25 +01:00
}
else if ( $alarm [ 'time' ] !== $time -> format ( 'ts' ) - $alarm [ 'offset' ])
{
$alarm [ 'time' ] = $time -> format ( 'ts' ) - $alarm [ 'offset' ];
2022-07-26 14:13:17 +02:00
$this -> save_alarm ( $event [ 'id' ], $alarm , true , $ignore_acl );
2016-01-18 18:45:25 +01:00
}
}
}
2005-11-09 00:15:14 +01:00
/**
* saves a new or updated alarm
*
* @ param int $cal_id Id of the calendar - entry
* @ param array $alarm array with fields : text , owner , enabled , ..
2015-06-25 22:39:53 +02:00
* @ param boolean $update_modified = true call update modified , default true
2022-07-26 14:13:17 +02:00
* @ param boolean $ignore_acl = false true : no acl check
2005-11-09 00:15:14 +01:00
* @ return string id of the alarm , or false on error ( eg . no perms )
*/
2022-07-26 14:13:17 +02:00
function save_alarm ( $cal_id , $alarm , $update_modified = true , $ignore_acl = false )
2005-11-09 00:15:14 +01:00
{
2022-07-26 14:13:17 +02:00
if ( ! $cal_id || ! $ignore_acl && ! $this -> check_perms ( Acl :: EDIT , $alarm [ 'all' ] ? $cal_id : 0 , ! $alarm [ 'all' ] ? $alarm [ 'owner' ] : 0 ))
2005-11-09 00:15:14 +01:00
{
2006-03-07 20:18:15 +01:00
//echo "<p>no rights to save the alarm=".print_r($alarm,true)." to event($cal_id)</p>";
2005-11-09 00:15:14 +01:00
return false ; // no rights to add the alarm
}
$alarm [ 'time' ] = $this -> date2ts ( $alarm [ 'time' ], true ); // user to server-time
2015-06-25 22:39:53 +02:00
return $this -> so -> save_alarm ( $cal_id , $alarm , $update_modified );
2005-11-09 00:15:14 +01:00
}
/**
* delete one alarms identified by its id
*
* @ param string $id alarm - id is a string of 'cal:' . $cal_id . ':' . $alarm_nr , it is used as the job - id too
2022-07-26 14:13:17 +02:00
* @ param boolean $ignore_acl = false true : no acl check
2005-11-09 00:15:14 +01:00
* @ return int number of alarms deleted , false on error ( eg . no perms )
*/
2022-07-26 14:13:17 +02:00
function delete_alarm ( $id , $ignore_acl = false )
2005-11-09 00:15:14 +01:00
{
list (, $cal_id ) = explode ( ':' , $id );
2022-07-26 14:13:17 +02:00
if ( ! ( $alarm = $this -> so -> read_alarm ( $id )) || ! $cal_id || ! $ignore_acl && ! $this -> check_perms ( Acl :: EDIT , $alarm [ 'all' ] ? $cal_id : 0 , ! $alarm [ 'all' ] ? $alarm [ 'owner' ] : 0 ))
2005-11-09 00:15:14 +01:00
{
return false ; // no rights to delete the alarm
}
2009-07-15 22:35:56 +02:00
2013-02-26 09:48:50 +01:00
return $this -> so -> delete_alarm ( $id );
2005-11-09 00:15:14 +01:00
}
SyncML patches from patrick.bihan-faou-AT-mindstep.com (without
logout+mbstring stuff), small modification to use the already exiting
methodes to generate full name and fileas)
The code is commited to trunk only at the moment to allow testing of it.
If everything goes well, we intend to commit it to 1.4 branch too.
Here's the original description of the patch by Patrick:
- handles the default config for current versions of funambol (i.e. the
scard/stask/snote/scal locations)
- tries to be a bit smarter on how the data content should be encoded
based on what the client specified (sif+base64/vcard, / fragmented or
not, etc.)
- workaround a bug in some versions of funambol, where funambol does not
specify the proper sif type for the type of requested data
- imported patch #117 from egw's tracker
- make sure that the logs generated by the horde code go to stderr so
they can be view in the webserver's logs
- as much as possible reduce code duplication. For example, the
categories are handled in the parent classes for both the SIF avn VCAL
formats for each type of data (addressbook,infolog,calendar).
- make sure the code can handle more than one categories in each
direction
- treat the 'sony ericsson' vendor string just like 'sonyericsson', the
newer phones apparently have a space in the vendor string... (this
touches some files in the icalsrv as well)
- handle notes: these should now work with everything (funambol or
other)
- remove more code duplication: the syncml "api" for the various data
types (calendar, contacts, infolog) is now common for both the vcard and
sif data formats (cf the files that need to be removed)
- handle the "privat" filter in infolog like the "private" filter (some
part of the code use the name without the trailing e)
- imported patch # 267 from egw's tracker
2007-09-29 12:29:48 +02:00
2009-11-18 15:46:25 +01:00
/**
* Find existing categories in database by name or add categories that do not exist yet
* currently used for ical / sif import
*
* @ param array $catname_list names of the categories which should be found or added
2015-06-25 22:39:53 +02:00
* @ param int | array $old_event = null match against existing event and expand the returned category ids
2009-11-18 15:46:25 +01:00
* by the ones the user normally does not see due to category permissions - used to preserve categories
* @ return array category ids ( found , added and preserved categories )
*/
2011-05-15 20:25:16 +02:00
function find_or_add_categories ( $catname_list , $old_event = null )
SyncML patches from patrick.bihan-faou-AT-mindstep.com (without
logout+mbstring stuff), small modification to use the already exiting
methodes to generate full name and fileas)
The code is commited to trunk only at the moment to allow testing of it.
If everything goes well, we intend to commit it to 1.4 branch too.
Here's the original description of the patch by Patrick:
- handles the default config for current versions of funambol (i.e. the
scard/stask/snote/scal locations)
- tries to be a bit smarter on how the data content should be encoded
based on what the client specified (sif+base64/vcard, / fragmented or
not, etc.)
- workaround a bug in some versions of funambol, where funambol does not
specify the proper sif type for the type of requested data
- imported patch #117 from egw's tracker
- make sure that the logs generated by the horde code go to stderr so
they can be view in the webserver's logs
- as much as possible reduce code duplication. For example, the
categories are handled in the parent classes for both the SIF avn VCAL
formats for each type of data (addressbook,infolog,calendar).
- make sure the code can handle more than one categories in each
direction
- treat the 'sony ericsson' vendor string just like 'sonyericsson', the
newer phones apparently have a space in the vendor string... (this
touches some files in the icalsrv as well)
- handle notes: these should now work with everything (funambol or
other)
- remove more code duplication: the syncml "api" for the various data
types (calendar, contacts, infolog) is now common for both the vcard and
sif data formats (cf the files that need to be removed)
- handle the "privat" filter in infolog like the "private" filter (some
part of the code use the name without the trailing e)
- imported patch # 267 from egw's tracker
2007-09-29 12:29:48 +02:00
{
2011-05-15 20:25:16 +02:00
if ( is_array ( $old_event ) || $old_event > 0 )
2009-11-18 15:46:25 +01:00
{
// preserve categories without users read access
2011-05-15 20:25:16 +02:00
if ( ! is_array ( $old_event )) $old_event = $this -> read ( $old_event );
2009-11-18 15:46:25 +01:00
$old_categories = explode ( ',' , $old_event [ 'category' ]);
$old_cats_preserve = array ();
2009-12-01 11:24:55 +01:00
if ( is_array ( $old_categories ) && count ( $old_categories ) > 0 )
2009-11-18 15:46:25 +01:00
{
2009-12-01 11:24:55 +01:00
foreach ( $old_categories as $cat_id )
2009-11-18 15:46:25 +01:00
{
2016-05-01 19:47:59 +02:00
if ( ! $this -> categories -> check_perms ( Acl :: READ , $cat_id ))
2009-11-18 15:46:25 +01:00
{
$old_cats_preserve [] = $cat_id ;
}
}
}
SyncML patches from patrick.bihan-faou-AT-mindstep.com (without
logout+mbstring stuff), small modification to use the already exiting
methodes to generate full name and fileas)
The code is commited to trunk only at the moment to allow testing of it.
If everything goes well, we intend to commit it to 1.4 branch too.
Here's the original description of the patch by Patrick:
- handles the default config for current versions of funambol (i.e. the
scard/stask/snote/scal locations)
- tries to be a bit smarter on how the data content should be encoded
based on what the client specified (sif+base64/vcard, / fragmented or
not, etc.)
- workaround a bug in some versions of funambol, where funambol does not
specify the proper sif type for the type of requested data
- imported patch #117 from egw's tracker
- make sure that the logs generated by the horde code go to stderr so
they can be view in the webserver's logs
- as much as possible reduce code duplication. For example, the
categories are handled in the parent classes for both the SIF avn VCAL
formats for each type of data (addressbook,infolog,calendar).
- make sure the code can handle more than one categories in each
direction
- treat the 'sony ericsson' vendor string just like 'sonyericsson', the
newer phones apparently have a space in the vendor string... (this
touches some files in the icalsrv as well)
- handle notes: these should now work with everything (funambol or
other)
- remove more code duplication: the syncml "api" for the various data
types (calendar, contacts, infolog) is now common for both the vcard and
sif data formats (cf the files that need to be removed)
- handle the "privat" filter in infolog like the "private" filter (some
part of the code use the name without the trailing e)
- imported patch # 267 from egw's tracker
2007-09-29 12:29:48 +02:00
}
$cat_id_list = array ();
2010-03-07 00:06:43 +01:00
foreach (( array ) $catname_list as $cat_name )
SyncML patches from patrick.bihan-faou-AT-mindstep.com (without
logout+mbstring stuff), small modification to use the already exiting
methodes to generate full name and fileas)
The code is commited to trunk only at the moment to allow testing of it.
If everything goes well, we intend to commit it to 1.4 branch too.
Here's the original description of the patch by Patrick:
- handles the default config for current versions of funambol (i.e. the
scard/stask/snote/scal locations)
- tries to be a bit smarter on how the data content should be encoded
based on what the client specified (sif+base64/vcard, / fragmented or
not, etc.)
- workaround a bug in some versions of funambol, where funambol does not
specify the proper sif type for the type of requested data
- imported patch #117 from egw's tracker
- make sure that the logs generated by the horde code go to stderr so
they can be view in the webserver's logs
- as much as possible reduce code duplication. For example, the
categories are handled in the parent classes for both the SIF avn VCAL
formats for each type of data (addressbook,infolog,calendar).
- make sure the code can handle more than one categories in each
direction
- treat the 'sony ericsson' vendor string just like 'sonyericsson', the
newer phones apparently have a space in the vendor string... (this
touches some files in the icalsrv as well)
- handle notes: these should now work with everything (funambol or
other)
- remove more code duplication: the syncml "api" for the various data
types (calendar, contacts, infolog) is now common for both the vcard and
sif data formats (cf the files that need to be removed)
- handle the "privat" filter in infolog like the "private" filter (some
part of the code use the name without the trailing e)
- imported patch # 267 from egw's tracker
2007-09-29 12:29:48 +02:00
{
2010-03-07 00:06:43 +01:00
$cat_name = trim ( $cat_name );
$cat_id = $this -> categories -> name2id ( $cat_name , 'X-' );
2010-03-02 19:01:41 +01:00
2010-03-07 00:06:43 +01:00
if ( ! $cat_id )
{
// some SyncML clients (mostly phones) add an X- to the category names
if ( strncmp ( $cat_name , 'X-' , 2 ) == 0 )
2007-12-11 10:29:50 +01:00
{
2010-03-07 00:06:43 +01:00
$cat_name = substr ( $cat_name , 2 );
2007-12-11 10:29:50 +01:00
}
2010-03-07 00:06:43 +01:00
$cat_id = $this -> categories -> add ( array ( 'name' => $cat_name , 'descr' => $cat_name , 'access' => 'private' ));
}
SyncML patches from patrick.bihan-faou-AT-mindstep.com (without
logout+mbstring stuff), small modification to use the already exiting
methodes to generate full name and fileas)
The code is commited to trunk only at the moment to allow testing of it.
If everything goes well, we intend to commit it to 1.4 branch too.
Here's the original description of the patch by Patrick:
- handles the default config for current versions of funambol (i.e. the
scard/stask/snote/scal locations)
- tries to be a bit smarter on how the data content should be encoded
based on what the client specified (sif+base64/vcard, / fragmented or
not, etc.)
- workaround a bug in some versions of funambol, where funambol does not
specify the proper sif type for the type of requested data
- imported patch #117 from egw's tracker
- make sure that the logs generated by the horde code go to stderr so
they can be view in the webserver's logs
- as much as possible reduce code duplication. For example, the
categories are handled in the parent classes for both the SIF avn VCAL
formats for each type of data (addressbook,infolog,calendar).
- make sure the code can handle more than one categories in each
direction
- treat the 'sony ericsson' vendor string just like 'sonyericsson', the
newer phones apparently have a space in the vendor string... (this
touches some files in the icalsrv as well)
- handle notes: these should now work with everything (funambol or
other)
- remove more code duplication: the syncml "api" for the various data
types (calendar, contacts, infolog) is now common for both the vcard and
sif data formats (cf the files that need to be removed)
- handle the "privat" filter in infolog like the "private" filter (some
part of the code use the name without the trailing e)
- imported patch # 267 from egw's tracker
2007-09-29 12:29:48 +02:00
2010-03-07 00:06:43 +01:00
if ( $cat_id )
{
$cat_id_list [] = $cat_id ;
2007-12-11 10:29:50 +01:00
}
SyncML patches from patrick.bihan-faou-AT-mindstep.com (without
logout+mbstring stuff), small modification to use the already exiting
methodes to generate full name and fileas)
The code is commited to trunk only at the moment to allow testing of it.
If everything goes well, we intend to commit it to 1.4 branch too.
Here's the original description of the patch by Patrick:
- handles the default config for current versions of funambol (i.e. the
scard/stask/snote/scal locations)
- tries to be a bit smarter on how the data content should be encoded
based on what the client specified (sif+base64/vcard, / fragmented or
not, etc.)
- workaround a bug in some versions of funambol, where funambol does not
specify the proper sif type for the type of requested data
- imported patch #117 from egw's tracker
- make sure that the logs generated by the horde code go to stderr so
they can be view in the webserver's logs
- as much as possible reduce code duplication. For example, the
categories are handled in the parent classes for both the SIF avn VCAL
formats for each type of data (addressbook,infolog,calendar).
- make sure the code can handle more than one categories in each
direction
- treat the 'sony ericsson' vendor string just like 'sonyericsson', the
newer phones apparently have a space in the vendor string... (this
touches some files in the icalsrv as well)
- handle notes: these should now work with everything (funambol or
other)
- remove more code duplication: the syncml "api" for the various data
types (calendar, contacts, infolog) is now common for both the vcard and
sif data formats (cf the files that need to be removed)
- handle the "privat" filter in infolog like the "private" filter (some
part of the code use the name without the trailing e)
- imported patch # 267 from egw's tracker
2007-09-29 12:29:48 +02:00
}
2009-11-19 19:56:04 +01:00
2009-12-01 11:24:55 +01:00
if ( is_array ( $old_cats_preserve ) && count ( $old_cats_preserve ) > 0 )
2009-11-18 15:46:25 +01:00
{
$cat_id_list = array_merge ( $cat_id_list , $old_cats_preserve );
}
SyncML patches from patrick.bihan-faou-AT-mindstep.com (without
logout+mbstring stuff), small modification to use the already exiting
methodes to generate full name and fileas)
The code is commited to trunk only at the moment to allow testing of it.
If everything goes well, we intend to commit it to 1.4 branch too.
Here's the original description of the patch by Patrick:
- handles the default config for current versions of funambol (i.e. the
scard/stask/snote/scal locations)
- tries to be a bit smarter on how the data content should be encoded
based on what the client specified (sif+base64/vcard, / fragmented or
not, etc.)
- workaround a bug in some versions of funambol, where funambol does not
specify the proper sif type for the type of requested data
- imported patch #117 from egw's tracker
- make sure that the logs generated by the horde code go to stderr so
they can be view in the webserver's logs
- as much as possible reduce code duplication. For example, the
categories are handled in the parent classes for both the SIF avn VCAL
formats for each type of data (addressbook,infolog,calendar).
- make sure the code can handle more than one categories in each
direction
- treat the 'sony ericsson' vendor string just like 'sonyericsson', the
newer phones apparently have a space in the vendor string... (this
touches some files in the icalsrv as well)
- handle notes: these should now work with everything (funambol or
other)
- remove more code duplication: the syncml "api" for the various data
types (calendar, contacts, infolog) is now common for both the vcard and
sif data formats (cf the files that need to be removed)
- handle the "privat" filter in infolog like the "private" filter (some
part of the code use the name without the trailing e)
- imported patch # 267 from egw's tracker
2007-09-29 12:29:48 +02:00
if ( count ( $cat_id_list ) > 1 )
{
2007-12-11 10:29:50 +01:00
$cat_id_list = array_unique ( $cat_id_list );
SyncML patches from patrick.bihan-faou-AT-mindstep.com (without
logout+mbstring stuff), small modification to use the already exiting
methodes to generate full name and fileas)
The code is commited to trunk only at the moment to allow testing of it.
If everything goes well, we intend to commit it to 1.4 branch too.
Here's the original description of the patch by Patrick:
- handles the default config for current versions of funambol (i.e. the
scard/stask/snote/scal locations)
- tries to be a bit smarter on how the data content should be encoded
based on what the client specified (sif+base64/vcard, / fragmented or
not, etc.)
- workaround a bug in some versions of funambol, where funambol does not
specify the proper sif type for the type of requested data
- imported patch #117 from egw's tracker
- make sure that the logs generated by the horde code go to stderr so
they can be view in the webserver's logs
- as much as possible reduce code duplication. For example, the
categories are handled in the parent classes for both the SIF avn VCAL
formats for each type of data (addressbook,infolog,calendar).
- make sure the code can handle more than one categories in each
direction
- treat the 'sony ericsson' vendor string just like 'sonyericsson', the
newer phones apparently have a space in the vendor string... (this
touches some files in the icalsrv as well)
- handle notes: these should now work with everything (funambol or
other)
- remove more code duplication: the syncml "api" for the various data
types (calendar, contacts, infolog) is now common for both the vcard and
sif data formats (cf the files that need to be removed)
- handle the "privat" filter in infolog like the "private" filter (some
part of the code use the name without the trailing e)
- imported patch # 267 from egw's tracker
2007-09-29 12:29:48 +02:00
sort ( $cat_id_list , SORT_NUMERIC );
}
2009-11-18 15:46:25 +01:00
SyncML patches from patrick.bihan-faou-AT-mindstep.com (without
logout+mbstring stuff), small modification to use the already exiting
methodes to generate full name and fileas)
The code is commited to trunk only at the moment to allow testing of it.
If everything goes well, we intend to commit it to 1.4 branch too.
Here's the original description of the patch by Patrick:
- handles the default config for current versions of funambol (i.e. the
scard/stask/snote/scal locations)
- tries to be a bit smarter on how the data content should be encoded
based on what the client specified (sif+base64/vcard, / fragmented or
not, etc.)
- workaround a bug in some versions of funambol, where funambol does not
specify the proper sif type for the type of requested data
- imported patch #117 from egw's tracker
- make sure that the logs generated by the horde code go to stderr so
they can be view in the webserver's logs
- as much as possible reduce code duplication. For example, the
categories are handled in the parent classes for both the SIF avn VCAL
formats for each type of data (addressbook,infolog,calendar).
- make sure the code can handle more than one categories in each
direction
- treat the 'sony ericsson' vendor string just like 'sonyericsson', the
newer phones apparently have a space in the vendor string... (this
touches some files in the icalsrv as well)
- handle notes: these should now work with everything (funambol or
other)
- remove more code duplication: the syncml "api" for the various data
types (calendar, contacts, infolog) is now common for both the vcard and
sif data formats (cf the files that need to be removed)
- handle the "privat" filter in infolog like the "private" filter (some
part of the code use the name without the trailing e)
- imported patch # 267 from egw's tracker
2007-09-29 12:29:48 +02:00
return $cat_id_list ;
}
function get_categories ( $cat_id_list )
{
2007-12-11 10:29:50 +01:00
if ( ! is_array ( $cat_id_list ))
SyncML patches from patrick.bihan-faou-AT-mindstep.com (without
logout+mbstring stuff), small modification to use the already exiting
methodes to generate full name and fileas)
The code is commited to trunk only at the moment to allow testing of it.
If everything goes well, we intend to commit it to 1.4 branch too.
Here's the original description of the patch by Patrick:
- handles the default config for current versions of funambol (i.e. the
scard/stask/snote/scal locations)
- tries to be a bit smarter on how the data content should be encoded
based on what the client specified (sif+base64/vcard, / fragmented or
not, etc.)
- workaround a bug in some versions of funambol, where funambol does not
specify the proper sif type for the type of requested data
- imported patch #117 from egw's tracker
- make sure that the logs generated by the horde code go to stderr so
they can be view in the webserver's logs
- as much as possible reduce code duplication. For example, the
categories are handled in the parent classes for both the SIF avn VCAL
formats for each type of data (addressbook,infolog,calendar).
- make sure the code can handle more than one categories in each
direction
- treat the 'sony ericsson' vendor string just like 'sonyericsson', the
newer phones apparently have a space in the vendor string... (this
touches some files in the icalsrv as well)
- handle notes: these should now work with everything (funambol or
other)
- remove more code duplication: the syncml "api" for the various data
types (calendar, contacts, infolog) is now common for both the vcard and
sif data formats (cf the files that need to be removed)
- handle the "privat" filter in infolog like the "private" filter (some
part of the code use the name without the trailing e)
- imported patch # 267 from egw's tracker
2007-09-29 12:29:48 +02:00
{
2007-12-11 10:29:50 +01:00
$cat_id_list = explode ( ',' , $cat_id_list );
SyncML patches from patrick.bihan-faou-AT-mindstep.com (without
logout+mbstring stuff), small modification to use the already exiting
methodes to generate full name and fileas)
The code is commited to trunk only at the moment to allow testing of it.
If everything goes well, we intend to commit it to 1.4 branch too.
Here's the original description of the patch by Patrick:
- handles the default config for current versions of funambol (i.e. the
scard/stask/snote/scal locations)
- tries to be a bit smarter on how the data content should be encoded
based on what the client specified (sif+base64/vcard, / fragmented or
not, etc.)
- workaround a bug in some versions of funambol, where funambol does not
specify the proper sif type for the type of requested data
- imported patch #117 from egw's tracker
- make sure that the logs generated by the horde code go to stderr so
they can be view in the webserver's logs
- as much as possible reduce code duplication. For example, the
categories are handled in the parent classes for both the SIF avn VCAL
formats for each type of data (addressbook,infolog,calendar).
- make sure the code can handle more than one categories in each
direction
- treat the 'sony ericsson' vendor string just like 'sonyericsson', the
newer phones apparently have a space in the vendor string... (this
touches some files in the icalsrv as well)
- handle notes: these should now work with everything (funambol or
other)
- remove more code duplication: the syncml "api" for the various data
types (calendar, contacts, infolog) is now common for both the vcard and
sif data formats (cf the files that need to be removed)
- handle the "privat" filter in infolog like the "private" filter (some
part of the code use the name without the trailing e)
- imported patch # 267 from egw's tracker
2007-09-29 12:29:48 +02:00
}
$cat_list = array ();
2009-12-01 11:24:55 +01:00
foreach ( $cat_id_list as $cat_id )
SyncML patches from patrick.bihan-faou-AT-mindstep.com (without
logout+mbstring stuff), small modification to use the already exiting
methodes to generate full name and fileas)
The code is commited to trunk only at the moment to allow testing of it.
If everything goes well, we intend to commit it to 1.4 branch too.
Here's the original description of the patch by Patrick:
- handles the default config for current versions of funambol (i.e. the
scard/stask/snote/scal locations)
- tries to be a bit smarter on how the data content should be encoded
based on what the client specified (sif+base64/vcard, / fragmented or
not, etc.)
- workaround a bug in some versions of funambol, where funambol does not
specify the proper sif type for the type of requested data
- imported patch #117 from egw's tracker
- make sure that the logs generated by the horde code go to stderr so
they can be view in the webserver's logs
- as much as possible reduce code duplication. For example, the
categories are handled in the parent classes for both the SIF avn VCAL
formats for each type of data (addressbook,infolog,calendar).
- make sure the code can handle more than one categories in each
direction
- treat the 'sony ericsson' vendor string just like 'sonyericsson', the
newer phones apparently have a space in the vendor string... (this
touches some files in the icalsrv as well)
- handle notes: these should now work with everything (funambol or
other)
- remove more code duplication: the syncml "api" for the various data
types (calendar, contacts, infolog) is now common for both the vcard and
sif data formats (cf the files that need to be removed)
- handle the "privat" filter in infolog like the "private" filter (some
part of the code use the name without the trailing e)
- imported patch # 267 from egw's tracker
2007-09-29 12:29:48 +02:00
{
2016-05-01 19:47:59 +02:00
if ( $cat_id && $this -> categories -> check_perms ( Acl :: READ , $cat_id ) &&
2009-11-18 15:46:25 +01:00
( $cat_name = $this -> categories -> id2name ( $cat_id )) && $cat_name != '--' )
SyncML patches from patrick.bihan-faou-AT-mindstep.com (without
logout+mbstring stuff), small modification to use the already exiting
methodes to generate full name and fileas)
The code is commited to trunk only at the moment to allow testing of it.
If everything goes well, we intend to commit it to 1.4 branch too.
Here's the original description of the patch by Patrick:
- handles the default config for current versions of funambol (i.e. the
scard/stask/snote/scal locations)
- tries to be a bit smarter on how the data content should be encoded
based on what the client specified (sif+base64/vcard, / fragmented or
not, etc.)
- workaround a bug in some versions of funambol, where funambol does not
specify the proper sif type for the type of requested data
- imported patch #117 from egw's tracker
- make sure that the logs generated by the horde code go to stderr so
they can be view in the webserver's logs
- as much as possible reduce code duplication. For example, the
categories are handled in the parent classes for both the SIF avn VCAL
formats for each type of data (addressbook,infolog,calendar).
- make sure the code can handle more than one categories in each
direction
- treat the 'sony ericsson' vendor string just like 'sonyericsson', the
newer phones apparently have a space in the vendor string... (this
touches some files in the icalsrv as well)
- handle notes: these should now work with everything (funambol or
other)
- remove more code duplication: the syncml "api" for the various data
types (calendar, contacts, infolog) is now common for both the vcard and
sif data formats (cf the files that need to be removed)
- handle the "privat" filter in infolog like the "private" filter (some
part of the code use the name without the trailing e)
- imported patch # 267 from egw's tracker
2007-09-29 12:29:48 +02:00
{
2009-11-18 15:46:25 +01:00
$cat_list [] = $cat_name ;
SyncML patches from patrick.bihan-faou-AT-mindstep.com (without
logout+mbstring stuff), small modification to use the already exiting
methodes to generate full name and fileas)
The code is commited to trunk only at the moment to allow testing of it.
If everything goes well, we intend to commit it to 1.4 branch too.
Here's the original description of the patch by Patrick:
- handles the default config for current versions of funambol (i.e. the
scard/stask/snote/scal locations)
- tries to be a bit smarter on how the data content should be encoded
based on what the client specified (sif+base64/vcard, / fragmented or
not, etc.)
- workaround a bug in some versions of funambol, where funambol does not
specify the proper sif type for the type of requested data
- imported patch #117 from egw's tracker
- make sure that the logs generated by the horde code go to stderr so
they can be view in the webserver's logs
- as much as possible reduce code duplication. For example, the
categories are handled in the parent classes for both the SIF avn VCAL
formats for each type of data (addressbook,infolog,calendar).
- make sure the code can handle more than one categories in each
direction
- treat the 'sony ericsson' vendor string just like 'sonyericsson', the
newer phones apparently have a space in the vendor string... (this
touches some files in the icalsrv as well)
- handle notes: these should now work with everything (funambol or
other)
- remove more code duplication: the syncml "api" for the various data
types (calendar, contacts, infolog) is now common for both the vcard and
sif data formats (cf the files that need to be removed)
- handle the "privat" filter in infolog like the "private" filter (some
part of the code use the name without the trailing e)
- imported patch # 267 from egw's tracker
2007-09-29 12:29:48 +02:00
}
}
return $cat_list ;
}
2009-07-15 22:35:56 +02:00
/**
* Try to find a matching db entry
*
* @ param array $event the vCalendar data we try to find
2010-02-11 08:33:32 +01:00
* @ param string filter = 'exact' exact -> find the matching entry
2024-06-04 15:30:54 +02:00
* check -> check ( consistency ) for identical matches
2010-01-29 22:42:54 +01:00
* relax -> be more tolerant
2024-06-04 15:30:54 +02:00
* master -> try to find a related series master
2010-02-09 22:56:39 +01:00
* @ return array calendar_ids of matching entries
2009-07-15 22:35:56 +02:00
*/
2010-01-29 22:42:54 +01:00
function find_event ( $event , $filter = 'exact' )
2009-07-15 22:35:56 +02:00
{
2010-01-29 22:42:54 +01:00
$matchingEvents = array ();
2009-12-01 11:24:55 +01:00
$query = array ();
2010-01-29 22:42:54 +01:00
if ( $this -> log )
{
error_log ( __FILE__ . '[' . __LINE__ . '] ' . __METHOD__ .
2010-03-09 18:03:41 +01:00
" ( $filter )[EVENT]: " . array2string ( $event ) . " \n " , 3 , $this -> logfile );
2010-01-29 22:42:54 +01:00
}
2010-09-12 01:15:05 +02:00
if ( ! isset ( $event [ 'recurrence' ])) $event [ 'recurrence' ] = 0 ;
2010-11-11 09:51:13 +01:00
2010-01-29 22:42:54 +01:00
if ( $filter == 'master' )
{
2010-03-07 00:06:43 +01:00
$query [] = 'recur_type!=' . MCAL_RECUR_NONE ;
$query [ 'cal_recurrence' ] = 0 ;
2010-02-03 13:24:42 +01:00
}
2010-09-12 22:14:46 +02:00
elseif ( $filter == 'exact' )
2010-09-12 01:15:05 +02:00
{
2024-02-08 10:24:39 +01:00
if ( ! empty ( $event [ 'recur_type' ]))
2010-09-12 22:14:46 +02:00
{
$query [] = 'recur_type=' . $event [ 'recur_type' ];
}
else
{
$query [] = 'recur_type IS NULL' ;
}
2010-11-11 09:51:13 +01:00
$query [ 'cal_recurrence' ] = $event [ 'recurrence' ];
2010-09-12 01:15:05 +02:00
}
2010-11-11 09:51:13 +01:00
2010-01-29 22:42:54 +01:00
if ( $event [ 'id' ])
{
if ( $this -> log )
{
error_log ( __FILE__ . '[' . __LINE__ . '] ' . __METHOD__ .
2010-03-09 18:03:41 +01:00
'(' . $event [ 'id' ] . " )[EventID] \n " , 3 , $this -> logfile );
2010-01-29 22:42:54 +01:00
}
2010-03-07 00:06:43 +01:00
if (( $egwEvent = $this -> read ( $event [ 'id' ], 0 , false , 'server' )))
2010-01-29 22:42:54 +01:00
{
if ( $this -> log )
{
error_log ( __FILE__ . '[' . __LINE__ . '] ' . __METHOD__ .
2010-03-09 18:03:41 +01:00
'()[FOUND]:' . array2string ( $egwEvent ) . " \n " , 3 , $this -> logfile );
2010-01-29 22:42:54 +01:00
}
2024-02-08 10:24:39 +01:00
if ( ! empty ( $egwEvent [ 'recur_type' ]) &&
2010-03-07 00:06:43 +01:00
( empty ( $event [ 'uid' ]) || $event [ 'uid' ] == $egwEvent [ 'uid' ]))
2010-01-29 22:42:54 +01:00
{
2010-03-07 00:06:43 +01:00
if ( $filter == 'master' )
{
$matchingEvents [] = $egwEvent [ 'id' ]; // we found the master
}
if ( $event [ 'recur_type' ] == $egwEvent [ 'recur_type' ])
2010-01-29 22:42:54 +01:00
{
2010-03-07 00:06:43 +01:00
$matchingEvents [] = $egwEvent [ 'id' ]; // we found the event
2010-01-29 22:42:54 +01:00
}
2024-02-08 10:24:39 +01:00
elseif ( empty ( $event [ 'recur_type' ]) &&
2010-03-07 00:06:43 +01:00
$event [ 'recurrence' ] != 0 )
{
$exceptions = $this -> so -> get_recurrence_exceptions ( $egwEvent , $event [ 'tzid' ]);
if ( in_array ( $event [ 'recurrence' ], $exceptions ))
{
$matchingEvents [] = $egwEvent [ 'id' ] . ':' . ( int ) $event [ 'recurrence' ];
}
}
2010-05-19 11:20:15 +02:00
} elseif ( $filter != 'master' && ( $filter == 'exact' ||
$event [ 'recur_type' ] == $egwEvent [ 'recur_type' ] &&
2010-03-16 10:26:01 +01:00
strpos ( $egwEvent [ 'title' ], $event [ 'title' ]) === 0 ))
2010-03-07 00:06:43 +01:00
{
$matchingEvents [] = $egwEvent [ 'id' ]; // we found the event
2010-01-29 22:42:54 +01:00
}
}
2010-03-07 00:06:43 +01:00
if ( ! empty ( $matchingEvents ) || $filter == 'exact' ) return $matchingEvents ;
2010-01-29 22:42:54 +01:00
}
unset ( $event [ 'id' ]);
2010-03-09 18:03:41 +01:00
// No chance to find a master without [U]ID
if ( $filter == 'master' && empty ( $event [ 'uid' ])) return $matchingEvents ;
2010-01-29 22:42:54 +01:00
// only query calendars of users, we have READ-grants from
$users = array ();
foreach ( array_keys ( $this -> grants ) as $user )
2009-07-15 22:35:56 +02:00
{
2010-01-29 22:42:54 +01:00
$user = trim ( $user );
2016-05-01 19:47:59 +02:00
if ( $this -> check_perms ( Acl :: READ | self :: ACL_READ_FOR_PARTICIPANTS | self :: ACL_FREEBUSY , 0 , $user ))
2010-01-29 22:42:54 +01:00
{
if ( $user && ! in_array ( $user , $users )) // already added?
{
$users [] = $user ;
}
}
elseif ( $GLOBALS [ 'egw' ] -> accounts -> get_type ( $user ) != 'g' )
2009-07-15 22:35:56 +02:00
{
2010-01-29 22:42:54 +01:00
continue ; // for non-groups (eg. users), we stop here if we have no read-rights
}
// the further code is only for real users
if ( ! is_numeric ( $user )) continue ;
2009-09-25 00:29:22 +02:00
2010-01-29 22:42:54 +01:00
// for groups we have to include the members
if ( $GLOBALS [ 'egw' ] -> accounts -> get_type ( $user ) == 'g' )
{
2016-03-06 17:09:58 +01:00
$members = $GLOBALS [ 'egw' ] -> accounts -> members ( $user , true );
2010-01-29 22:42:54 +01:00
if ( is_array ( $members ))
2009-07-15 22:35:56 +02:00
{
2010-01-29 22:42:54 +01:00
foreach ( $members as $member )
2009-07-15 22:35:56 +02:00
{
2010-01-29 22:42:54 +01:00
// use only members which gave the user a read-grant
2016-03-06 17:09:58 +01:00
if ( ! in_array ( $member , $users ) &&
2016-05-01 19:47:59 +02:00
$this -> check_perms ( Acl :: READ | self :: ACL_FREEBUSY , 0 , $member ))
2010-01-29 22:42:54 +01:00
{
2016-03-06 17:09:58 +01:00
$users [] = $member ;
2010-01-29 22:42:54 +01:00
}
2009-07-15 22:35:56 +02:00
}
}
2010-01-29 22:42:54 +01:00
}
else // for users we have to include all the memberships, to get the group-events
{
2016-03-06 17:09:58 +01:00
$memberships = $GLOBALS [ 'egw' ] -> accounts -> memberships ( $user , true );
2010-01-29 22:42:54 +01:00
if ( is_array ( $memberships ))
2009-07-15 22:35:56 +02:00
{
2010-01-29 22:42:54 +01:00
foreach ( $memberships as $group )
2009-07-15 22:35:56 +02:00
{
2016-03-06 17:09:58 +01:00
if ( ! in_array ( $group , $users ))
2009-07-15 22:35:56 +02:00
{
2016-03-06 17:09:58 +01:00
$users [] = $group ;
2009-07-15 22:35:56 +02:00
}
2009-09-25 00:29:22 +02:00
}
2010-01-29 22:42:54 +01:00
}
}
}
2010-03-07 00:06:43 +01:00
2010-02-11 22:04:10 +01:00
if ( $filter != 'master' && ( $filter != 'exact' || empty ( $event [ 'uid' ])))
2010-01-29 22:42:54 +01:00
{
2010-04-23 17:15:22 +02:00
if ( ! empty ( $event [ 'whole_day' ]))
2010-01-29 22:42:54 +01:00
{
2010-02-09 22:56:39 +01:00
if ( $filter == 'relax' )
{
$delta = 1800 ;
}
else
{
$delta = 60 ;
}
2010-01-29 22:42:54 +01:00
2010-02-09 22:56:39 +01:00
// check length with some tolerance
$length = $event [ 'end' ] - $event [ 'start' ] - $delta ;
$query [] = ( '(cal_end-cal_start)>' . $length );
$length += 2 * $delta ;
$query [] = ( '(cal_end-cal_start)<' . $length );
$query [] = ( 'cal_start>' . ( $event [ 'start' ] - 86400 ));
$query [] = ( 'cal_start<' . ( $event [ 'start' ] + 86400 ));
}
elseif ( isset ( $event [ 'start' ]))
{
2010-02-11 08:33:32 +01:00
if ( $filter == 'relax' )
2010-02-09 22:56:39 +01:00
{
2010-02-11 08:33:32 +01:00
$query [] = ( 'cal_start>' . ( $event [ 'start' ] - 3600 ));
$query [] = ( 'cal_start<' . ( $event [ 'start' ] + 3600 ));
}
else
{
// we accept a tiny tolerance
$query [] = ( 'cal_start>' . ( $event [ 'start' ] - 2 ));
$query [] = ( 'cal_start<' . ( $event [ 'start' ] + 2 ));
2010-02-09 22:56:39 +01:00
}
}
2010-03-09 18:03:41 +01:00
if ( $filter == 'relax' )
{
$matchFields = array ();
}
else
{
2010-05-14 11:28:46 +02:00
$matchFields = array ( 'priority' , 'public' );
2010-03-09 18:03:41 +01:00
}
2010-01-29 22:42:54 +01:00
foreach ( $matchFields as $key )
{
2010-02-22 11:02:33 +01:00
if ( isset ( $event [ $key ])) $query [ 'cal_' . $key ] = $event [ $key ];
2010-01-29 22:42:54 +01:00
}
}
2010-09-12 01:15:05 +02:00
2010-02-11 08:33:32 +01:00
if ( ! empty ( $event [ 'uid' ]))
2010-01-29 22:42:54 +01:00
{
$query [ 'cal_uid' ] = $event [ 'uid' ];
if ( $this -> log )
{
error_log ( __FILE__ . '[' . __LINE__ . '] ' . __METHOD__ .
2010-03-09 18:03:41 +01:00
'(' . $event [ 'uid' ] . " )[EventUID] \n " , 3 , $this -> logfile );
2010-01-29 22:42:54 +01:00
}
}
if ( $this -> log )
{
error_log ( __FILE__ . '[' . __LINE__ . '] ' . __METHOD__ .
2010-03-09 18:03:41 +01:00
'[QUERY]: ' . array2string ( $query ) . " \n " , 3 , $this -> logfile );
2010-01-29 22:42:54 +01:00
}
if ( ! count ( $users ) || ! ( $foundEvents =
2010-06-01 11:26:00 +02:00
$this -> so -> search ( null , null , $users , 0 , 'owner' , false , 0 , array ( 'query' => $query ))))
2010-01-29 22:42:54 +01:00
{
if ( $this -> log )
{
error_log ( __FILE__ . '[' . __LINE__ . '] ' . __METHOD__ .
2010-03-09 18:03:41 +01:00
" [NO MATCH] \n " , 3 , $this -> logfile );
2010-01-29 22:42:54 +01:00
}
return $matchingEvents ;
}
$pseudos = array ();
foreach ( $foundEvents as $egwEvent )
{
if ( $this -> log )
{
error_log ( __FILE__ . '[' . __LINE__ . '] ' . __METHOD__ .
2010-03-09 18:03:41 +01:00
'[FOUND]: ' . array2string ( $egwEvent ) . " \n " , 3 , $this -> logfile );
2010-01-29 22:42:54 +01:00
}
2010-11-11 09:51:13 +01:00
2010-02-17 14:29:28 +01:00
if ( in_array ( $egwEvent [ 'id' ], $matchingEvents )) continue ;
2010-11-11 09:51:13 +01:00
2010-01-29 22:42:54 +01:00
// convert timezone id of event to tzid (iCal id like 'Europe/Berlin')
if ( ! $egwEvent [ 'tz_id' ] || ! ( $egwEvent [ 'tzid' ] = calendar_timezones :: id2tz ( $egwEvent [ 'tz_id' ])))
{
2016-05-01 19:47:59 +02:00
$egwEvent [ 'tzid' ] = Api\DateTime :: $server_timezone -> getName ();
2010-01-29 22:42:54 +01:00
}
2010-06-23 16:52:55 +02:00
if ( ! isset ( self :: $tz_cache [ $egwEvent [ 'tzid' ]]))
2010-03-07 00:06:43 +01:00
{
2010-06-23 16:52:55 +02:00
self :: $tz_cache [ $egwEvent [ 'tzid' ]] = calendar_timezones :: DateTimeZone ( $egwEvent [ 'tzid' ]);
}
if ( ! $event [ 'tzid' ])
{
2016-05-01 19:47:59 +02:00
$event [ 'tzid' ] = Api\DateTime :: $server_timezone -> getName ();
2010-06-23 16:52:55 +02:00
}
if ( ! isset ( self :: $tz_cache [ $event [ 'tzid' ]]))
{
self :: $tz_cache [ $event [ 'tzid' ]] = calendar_timezones :: DateTimeZone ( $event [ 'tzid' ]);
}
2010-11-11 09:51:13 +01:00
2010-06-23 16:52:55 +02:00
if ( ! empty ( $event [ 'uid' ]))
{
if ( $filter == 'master' )
2010-03-07 00:06:43 +01:00
{
2010-06-23 16:52:55 +02:00
// We found the master
2015-06-25 22:39:53 +02:00
$matchingEvents = array ( $egwEvent [ 'id' ]);
2010-03-07 00:06:43 +01:00
break ;
2010-11-11 09:51:13 +01:00
}
if ( $filter == 'exact' )
2010-03-07 00:06:43 +01:00
{
2010-06-23 16:52:55 +02:00
// UID found
if ( empty ( $event [ 'recurrence' ]))
2010-03-07 00:06:43 +01:00
{
2016-05-01 19:47:59 +02:00
$egwstart = new Api\DateTime ( $egwEvent [ 'start' ], Api\DateTime :: $server_timezone );
2010-06-23 16:52:55 +02:00
$egwstart -> setTimezone ( self :: $tz_cache [ $egwEvent [ 'tzid' ]]);
2016-05-01 19:47:59 +02:00
$dtstart = new Api\DateTime ( $event [ 'start' ], Api\DateTime :: $server_timezone );
2010-06-23 16:52:55 +02:00
$dtstart -> setTimezone ( self :: $tz_cache [ $event [ 'tzid' ]]);
2024-02-08 10:24:39 +01:00
if ( empty ( $egwEvent [ 'recur_type' ]) &&
empty ( $event [ 'recur_type' ]) ||
! empty ( $egwEvent [ 'recur_type' ]) &&
! empty ( $event [ 'recur_type' ]))
2010-06-23 16:52:55 +02:00
{
2024-02-13 13:46:28 +01:00
if ( empty ( $egwEvent [ 'recur_type' ]) &&
2010-06-23 16:52:55 +02:00
$egwstart -> format ( 'Ymd' ) == $dtstart -> format ( 'Ymd' ) ||
2024-02-08 10:24:39 +01:00
! empty ( $egwEvent [ 'recur_type' ]))
2010-06-23 16:52:55 +02:00
{
// We found an exact match
$matchingEvents = array ( $egwEvent [ 'id' ]);
break ;
}
else
{
2010-11-11 09:51:13 +01:00
$matchingEvents [] = $egwEvent [ 'id' ];
2010-06-23 16:52:55 +02:00
}
}
continue ;
2010-03-07 00:06:43 +01:00
}
2010-06-23 16:52:55 +02:00
elseif ( $egwEvent [ 'recurrence' ] == $event [ 'recurrence' ])
2010-03-07 00:06:43 +01:00
{
2010-06-23 16:52:55 +02:00
// We found an exact match
$matchingEvents = array ( $egwEvent [ 'id' ]);
2010-03-07 00:06:43 +01:00
break ;
}
2024-02-08 10:24:39 +01:00
if ( ! empty ( $egwEvent [ 'recur_type' ]) &&
empty ( $event [ 'recur_type' ]) &&
2010-06-23 16:52:55 +02:00
! $egwEvent [ 'recurrence' ] && $event [ 'recurrence' ])
{
$exceptions = $this -> so -> get_recurrence_exceptions ( $egwEvent , $event [ 'tzid' ]);
if ( in_array ( $event [ 'recurrence' ], $exceptions ))
{
// We found a pseudo exception
$matchingEvents = array ( $egwEvent [ 'id' ] . ':' . ( int ) $event [ 'recurrence' ]);
break ;
}
}
continue ;
2010-03-07 00:06:43 +01:00
}
}
2010-01-29 22:42:54 +01:00
// check times
if ( $filter != 'relax' )
{
2010-04-23 17:15:22 +02:00
if ( empty ( $event [ 'whole_day' ]))
2010-01-29 22:42:54 +01:00
{
2010-04-23 17:15:22 +02:00
if ( abs ( $event [ 'end' ] - $egwEvent [ 'end' ]) >= 120 )
2009-09-25 00:29:22 +02:00
{
2010-01-29 22:42:54 +01:00
if ( $this -> log )
2009-07-15 22:35:56 +02:00
{
2010-01-29 22:42:54 +01:00
error_log ( __FILE__ . '[' . __LINE__ . '] ' . __METHOD__ .
2010-04-23 17:15:22 +02:00
" () egwEvent length does not match! \n " , 3 , $this -> logfile );
2009-07-15 22:35:56 +02:00
}
2010-01-29 22:42:54 +01:00
continue ;
2009-07-15 22:35:56 +02:00
}
2010-01-29 22:42:54 +01:00
}
2010-03-09 18:03:41 +01:00
else
2010-01-29 22:42:54 +01:00
{
2010-04-23 17:15:22 +02:00
if ( ! $this -> so -> isWholeDay ( $egwEvent ))
2010-01-29 22:42:54 +01:00
{
if ( $this -> log )
{
error_log ( __FILE__ . '[' . __LINE__ . '] ' . __METHOD__ .
2010-04-23 17:15:22 +02:00
" () egwEvent is not a whole-day event! \n " , 3 , $this -> logfile );
2010-01-29 22:42:54 +01:00
}
continue ;
}
}
}
2009-07-15 22:35:56 +02:00
2010-02-09 22:56:39 +01:00
// check for real match
2010-03-09 18:03:41 +01:00
$matchFields = array ( 'title' , 'description' );
if ( $filter != 'relax' )
2010-02-09 22:56:39 +01:00
{
2010-03-09 18:03:41 +01:00
$matchFields [] = 'location' ;
2010-02-09 22:56:39 +01:00
}
foreach ( $matchFields as $key )
{
if ( ! empty ( $event [ $key ]) && ( empty ( $egwEvent [ $key ])
2010-05-14 15:32:08 +02:00
|| strpos ( str_replace ( " \r \n " , " \n " , $egwEvent [ $key ]), $event [ $key ]) !== 0 ))
2010-02-09 22:56:39 +01:00
{
if ( $this -> log )
{
error_log ( __FILE__ . '[' . __LINE__ . '] ' . __METHOD__ .
" () event[ $key ] differ: ' " . $event [ $key ] .
2010-03-09 18:03:41 +01:00
" ' <> ' " . $egwEvent [ $key ] . " ' \n " , 3 , $this -> logfile );
2010-02-09 22:56:39 +01:00
}
continue 2 ; // next foundEvent
}
}
2010-03-09 18:03:41 +01:00
if ( is_array ( $event [ 'category' ]))
2010-01-29 22:42:54 +01:00
{
// check categories
$egwCategories = explode ( ',' , $egwEvent [ 'category' ]);
foreach ( $egwCategories as $cat_id )
{
2016-05-01 19:47:59 +02:00
if ( $this -> categories -> check_perms ( Acl :: READ , $cat_id ) &&
2010-01-29 22:42:54 +01:00
! in_array ( $cat_id , $event [ 'category' ]))
{
if ( $this -> log )
{
error_log ( __FILE__ . '[' . __LINE__ . '] ' . __METHOD__ .
2010-03-09 18:03:41 +01:00
" () egwEvent category $cat_id is missing! \n " , 3 , $this -> logfile );
2010-01-29 22:42:54 +01:00
}
continue 2 ;
}
}
$newCategories = array_diff ( $event [ 'category' ], $egwCategories );
if ( ! empty ( $newCategories ))
{
if ( $this -> log )
{
error_log ( __FILE__ . '[' . __LINE__ . '] ' . __METHOD__ .
2010-03-09 18:03:41 +01:00
'() event has additional categories:'
. array2string ( $newCategories ) . " \n " , 3 , $this -> logfile );
2010-01-29 22:42:54 +01:00
}
continue ;
2009-09-25 00:29:22 +02:00
}
2009-07-15 22:35:56 +02:00
}
2010-03-09 18:03:41 +01:00
if ( $filter != 'relax' )
2010-01-29 22:42:54 +01:00
{
// check participants
if ( is_array ( $event [ 'participants' ]))
2009-07-15 22:35:56 +02:00
{
2010-01-29 22:42:54 +01:00
foreach ( $event [ 'participants' ] as $attendee => $status )
{
if ( ! isset ( $egwEvent [ 'participants' ][ $attendee ]) &&
$attendee != $egwEvent [ 'owner' ]) // ||
//(!$relax && $egw_event['participants'][$attendee] != $status))
{
if ( $this -> log )
{
error_log ( __FILE__ . '[' . __LINE__ . '] ' . __METHOD__ .
2010-03-09 18:03:41 +01:00
" () additional event['participants']: $attendee\n " , 3 , $this -> logfile );
2010-01-29 22:42:54 +01:00
}
continue 2 ;
}
else
{
unset ( $egwEvent [ 'participants' ][ $attendee ]);
}
}
2010-09-12 22:14:46 +02:00
// ORGANIZER and Groups may be missing
2010-01-29 22:42:54 +01:00
unset ( $egwEvent [ 'participants' ][ $egwEvent [ 'owner' ]]);
2010-09-12 22:14:46 +02:00
foreach ( $egwEvent [ 'participants' ] as $attendee => $status )
{
if ( is_numeric ( $attendee ) && $attendee < 0 )
{
unset ( $egwEvent [ 'participants' ][ $attendee ]);
}
}
2010-01-29 22:42:54 +01:00
if ( ! empty ( $egwEvent [ 'participants' ]))
{
if ( $this -> log )
{
error_log ( __FILE__ . '[' . __LINE__ . '] ' . __METHOD__ .
'() missing event[participants]: ' .
2010-03-09 18:03:41 +01:00
array2string ( $egwEvent [ 'participants' ]) . " \n " , 3 , $this -> logfile );
2010-01-29 22:42:54 +01:00
}
continue ;
}
2009-07-15 22:35:56 +02:00
}
2010-01-29 22:42:54 +01:00
}
2009-07-15 22:35:56 +02:00
2024-02-08 10:24:39 +01:00
if ( empty ( $event [ 'recur_type' ]))
2009-07-15 22:35:56 +02:00
{
2024-02-08 10:24:39 +01:00
if ( ! empty ( $egwEvent [ 'recur_type' ]))
2010-01-29 22:42:54 +01:00
{
2010-03-09 18:03:41 +01:00
// We found a pseudo Exception
$pseudos [] = $egwEvent [ 'id' ] . ':' . $event [ 'start' ];
continue ;
2010-01-29 22:42:54 +01:00
}
2010-03-09 18:03:41 +01:00
}
elseif ( $filter != 'relax' )
{
// check exceptions
// $exceptions[$remote_ts] = $egw_ts
$exceptions = $this -> so -> get_recurrence_exceptions ( $egwEvent , $event [ '$tzid' ], 0 , 0 , 'map' );
if ( is_array ( $event [ 'recur_exception' ]))
2010-01-29 22:42:54 +01:00
{
2010-03-09 18:03:41 +01:00
foreach ( $event [ 'recur_exception' ] as $key => $day )
2010-01-29 22:42:54 +01:00
{
2010-03-09 18:03:41 +01:00
if ( isset ( $exceptions [ $day ]))
2010-01-29 22:42:54 +01:00
{
2010-03-09 18:03:41 +01:00
unset ( $exceptions [ $day ]);
2010-01-29 22:42:54 +01:00
}
2010-03-09 18:03:41 +01:00
else
2010-01-29 22:42:54 +01:00
{
if ( $this -> log )
{
error_log ( __FILE__ . '[' . __LINE__ . '] ' . __METHOD__ .
2010-03-09 18:03:41 +01:00
" () additional event['recur_exception']: $day\n " , 3 , $this -> logfile );
2010-01-29 22:42:54 +01:00
}
2010-03-09 18:03:41 +01:00
continue 2 ;
}
}
if ( ! empty ( $exceptions ))
{
if ( $this -> log )
{
error_log ( __FILE__ . '[' . __LINE__ . '] ' . __METHOD__ .
'() missing event[recur_exception]: ' .
2010-09-12 22:14:46 +02:00
array2string ( $event [ 'recur_exception' ]) . " \n " , 3 , $this -> logfile );
2010-01-29 22:42:54 +01:00
}
2010-03-09 18:03:41 +01:00
continue ;
2010-01-29 22:42:54 +01:00
}
2010-03-09 18:03:41 +01:00
}
2010-01-29 22:42:54 +01:00
2010-03-09 18:03:41 +01:00
// check recurrence information
foreach ( array ( 'recur_type' , 'recur_interval' , 'recur_enddate' ) as $key )
{
if ( isset ( $event [ $key ])
&& $event [ $key ] != $egwEvent [ $key ])
2010-01-29 22:42:54 +01:00
{
2010-03-09 18:03:41 +01:00
if ( $this -> log )
2010-01-29 22:42:54 +01:00
{
2010-03-09 18:03:41 +01:00
error_log ( __FILE__ . '[' . __LINE__ . '] ' . __METHOD__ .
" () events[ $key ] differ: " . $event [ $key ] .
' <> ' . $egwEvent [ $key ] . " \n " , 3 , $this -> logfile );
2010-01-29 22:42:54 +01:00
}
2010-03-09 18:03:41 +01:00
continue 2 ;
2010-01-29 22:42:54 +01:00
}
}
2009-07-15 22:35:56 +02:00
}
2010-01-29 22:42:54 +01:00
$matchingEvents [] = $egwEvent [ 'id' ]; // exact match
}
2010-11-11 09:51:13 +01:00
2010-09-12 22:14:46 +02:00
if ( $filter == 'exact' && ! empty ( $event [ 'uid' ]) && count ( $matchingEvents ) > 1
|| $filter != 'master' && ! empty ( $egwEvent [ 'recur_type' ]) && empty ( $event [ 'recur_type' ]))
2010-06-23 16:52:55 +02:00
{
// Unknown exception for existing series
if ( $this -> log )
{
error_log ( __FILE__ . '[' . __LINE__ . '] ' . __METHOD__ .
" () new exception for series found. \n " , 3 , $this -> logfile );
}
$matchingEvents = array ();
}
2010-11-11 09:51:13 +01:00
2010-01-29 22:42:54 +01:00
// append pseudos as last entries
2015-06-25 22:39:53 +02:00
$matches = array_merge ( $matchingEvents , $pseudos );
2010-01-29 22:42:54 +01:00
if ( $this -> log )
{
error_log ( __FILE__ . '[' . __LINE__ . '] ' . __METHOD__ .
2015-06-25 22:39:53 +02:00
'[MATCHES]:' . array2string ( $matches ) . " \n " , 3 , $this -> logfile );
2009-07-15 22:35:56 +02:00
}
2015-06-25 22:39:53 +02:00
return $matches ;
2009-07-15 22:35:56 +02:00
}
2010-01-14 18:01:30 +01:00
/**
* classifies an incoming event from the eGW point - of - view
*
* exceptions : unlike other calendar apps eGW does not create an event exception
* if just the participant state changes - therefore we have to distinguish between
* real exceptions and status only exceptions
*
* @ param array $event the event to check
*
* @ return array
* type =>
* SINGLE a single event
* SERIES - MASTER the series master
* SERIES - EXCEPTION event is a real exception
2010-01-29 22:42:54 +01:00
* SERIES - PSEUDO - EXCEPTION event is a status only exception
2010-01-14 18:01:30 +01:00
* SERIES - EXCEPTION - PROPAGATE event was a status only exception in the past and is now a real exception
* stored_event => if event already exists in the database array with event data or false
2010-01-29 22:42:54 +01:00
* master_event => for event type SERIES - EXCEPTION , SERIES - PSEUDO - EXCEPTION or SERIES - EXCEPTION - PROPAGATE
2010-01-14 18:01:30 +01:00
* the corresponding series master event array
* NOTE : this param is false if event is of type SERIES - MASTER
*/
2023-11-10 22:32:41 +01:00
function get_event_info ( & $event )
2010-02-10 20:47:39 +01:00
{
$type = 'SINGLE' ; // default
$master_event = false ; //default
$stored_event = false ;
$recurrence_event = false ;
$wasPseudo = false ;
2010-01-14 18:01:30 +01:00
2010-02-10 20:47:39 +01:00
if (( $foundEvents = $this -> find_event ( $event , 'exact' )))
{
// We found the exact match
$eventID = array_shift ( $foundEvents );
if ( strstr ( $eventID , ':' ))
2010-01-14 18:01:30 +01:00
{
2010-02-10 20:47:39 +01:00
$type = 'SERIES-PSEUDO-EXCEPTION' ;
$wasPseudo = true ;
2015-06-25 22:39:53 +02:00
list ( $eventID , $date ) = explode ( ':' , $eventID );
$recur_date = $this -> date2usertime ( $date );
2010-02-10 20:47:39 +01:00
$stored_event = $this -> read ( $eventID , $recur_date , false , 'server' );
$master_event = $this -> read ( $eventID , 0 , false , 'server' );
$recurrence_event = $stored_event ;
2010-01-14 18:01:30 +01:00
}
else
{
2010-02-10 20:47:39 +01:00
$stored_event = $this -> read ( $eventID , 0 , false , 'server' );
}
if ( ! empty ( $stored_event [ 'uid' ]) && empty ( $event [ 'uid' ]))
{
$event [ 'uid' ] = $stored_event [ 'uid' ]; // restore the UID if it was not delivered
}
}
2024-02-08 10:24:39 +01:00
if ( ! empty ( $event [ 'recur_type' ]))
2010-02-10 20:47:39 +01:00
{
$type = 'SERIES-MASTER' ;
}
if ( $type == 'SINGLE' &&
( $foundEvents = $this -> find_event ( $event , 'master' )))
{
// SINGLE, SERIES-EXCEPTION OR SERIES-EXCEPTON-STATUS
foreach ( $foundEvents as $eventID )
{
// Let's try to find a related series
if ( $this -> log )
2010-01-14 18:01:30 +01:00
{
2010-02-10 20:47:39 +01:00
error_log ( __FILE__ . '[' . __LINE__ . '] ' . __METHOD__ .
2010-03-09 18:03:41 +01:00
" ()[MASTER]: $eventID\n " , 3 , $this -> logfile );
2010-02-10 20:47:39 +01:00
}
2010-06-23 16:52:55 +02:00
$type = 'SERIES-EXCEPTION' ;
2010-02-10 20:47:39 +01:00
if (( $master_event = $this -> read ( $eventID , 0 , false , 'server' )))
{
2010-06-23 16:52:55 +02:00
if ( isset ( $stored_event [ 'id' ]) &&
$master_event [ 'id' ] != $stored_event [ 'id' ])
2010-01-29 22:42:54 +01:00
{
2010-06-23 16:52:55 +02:00
break ; // this is an existing exception
2010-01-14 18:01:30 +01:00
}
2010-02-10 20:47:39 +01:00
elseif ( isset ( $event [ 'recurrence' ]) &&
in_array ( $event [ 'recurrence' ], $master_event [ 'recur_exception' ]))
2010-01-14 18:01:30 +01:00
{
2010-02-10 20:47:39 +01:00
$type = 'SERIES-PSEUDO-EXCEPTION' ; // could also be a real one
2010-02-17 14:29:28 +01:00
$recurrence_event = $master_event ;
$recurrence_event [ 'start' ] = $event [ 'recurrence' ];
$recurrence_event [ 'end' ] -= $master_event [ 'start' ] - $event [ 'recurrence' ];
2010-02-10 20:47:39 +01:00
break ;
2010-01-29 22:42:54 +01:00
}
2010-02-10 20:47:39 +01:00
elseif ( in_array ( $event [ 'start' ], $master_event [ 'recur_exception' ]))
2010-01-29 22:42:54 +01:00
{
2010-02-10 20:47:39 +01:00
$type = 'SERIES-PSEUDO-EXCEPTION' ; // new pseudo exception?
2010-02-17 14:29:28 +01:00
$recurrence_event = $master_event ;
$recurrence_event [ 'start' ] = $event [ 'start' ];
$recurrence_event [ 'end' ] -= $master_event [ 'start' ] - $event [ 'start' ];
2010-02-10 20:47:39 +01:00
break ;
2010-01-29 22:42:54 +01:00
}
2010-02-10 20:47:39 +01:00
else
2010-01-29 22:42:54 +01:00
{
2010-02-10 20:47:39 +01:00
// try to find a suitable pseudo exception date
2023-11-10 22:32:41 +01:00
// Checks are all in server time
$egw_rrule = calendar_rrule :: event2rrule ( $master_event , false , $GLOBALS [ 'egw_info' ][ 'server' ][ 'server_timezone' ]);
2010-02-17 14:29:28 +01:00
$egw_rrule -> current = clone $egw_rrule -> time ;
2010-02-10 20:47:39 +01:00
while ( $egw_rrule -> valid ())
2010-01-29 22:42:54 +01:00
{
2016-05-01 19:47:59 +02:00
$occurrence = Api\DateTime :: to ( $egw_rrule -> current (), 'server' );
2010-02-10 20:47:39 +01:00
if ( $this -> log )
2010-02-03 13:24:42 +01:00
{
2010-02-10 20:47:39 +01:00
error_log ( __FILE__ . '[' . __LINE__ . '] ' . __METHOD__ .
2010-03-09 18:03:41 +01:00
'() try occurrence ' . $egw_rrule -> current ()
. " ( $occurrence ) \n " , 3 , $this -> logfile );
2010-01-29 22:42:54 +01:00
}
2010-02-10 20:47:39 +01:00
if ( $event [ 'start' ] == $occurrence )
2010-01-14 18:01:30 +01:00
{
2010-02-10 20:47:39 +01:00
$type = 'SERIES-PSEUDO-EXCEPTION' ; // let's try a pseudo exception
2010-02-17 14:29:28 +01:00
$recurrence_event = $master_event ;
2023-11-10 22:32:41 +01:00
// Update the event's recurrence too
$recurrence_event [ 'start' ] = $event [ 'recurrence' ] = $occurrence ;
2010-02-17 14:29:28 +01:00
$recurrence_event [ 'end' ] -= $master_event [ 'start' ] - $occurrence ;
2010-02-10 20:47:39 +01:00
break 2 ;
2010-01-29 22:42:54 +01:00
}
2010-02-10 20:47:39 +01:00
if ( isset ( $event [ 'recurrence' ]) && $event [ 'recurrence' ] == $occurrence )
2010-01-29 22:42:54 +01:00
{
2010-02-10 20:47:39 +01:00
$type = 'SERIES-EXCEPTION-PROPAGATE' ;
if ( $stored_event )
2010-01-14 18:01:30 +01:00
{
2010-02-10 20:47:39 +01:00
unset ( $stored_event [ 'id' ]); // signal the true exception
$stored_event [ 'recur_type' ] = MCAL_RECUR_NONE ;
2010-01-14 18:01:30 +01:00
}
2010-02-10 20:47:39 +01:00
break 2 ;
2010-01-14 18:01:30 +01:00
}
2010-02-10 20:47:39 +01:00
$egw_rrule -> next_no_exception ();
2010-01-14 18:01:30 +01:00
}
}
}
2010-02-10 20:47:39 +01:00
}
}
2010-01-29 22:42:54 +01:00
2010-02-10 20:47:39 +01:00
// check pseudo exception propagation
if ( $recurrence_event )
{
// default if we cannot find a proof for a fundamental change
// the recurrence_event is the master event with start and end adjusted to the recurrence
// check for changed data
foreach ( array ( 'start' , 'end' , 'uid' , 'title' , 'location' , 'description' ,
'priority' , 'public' , 'special' , 'non_blocking' ) as $key )
2010-03-16 10:26:01 +01:00
{
2010-02-10 20:47:39 +01:00
if ( ! empty ( $event [ $key ]) && $recurrence_event [ $key ] != $event [ $key ])
2010-01-14 18:01:30 +01:00
{
2010-02-10 20:47:39 +01:00
if ( $wasPseudo )
2010-01-29 22:42:54 +01:00
{
2010-02-10 20:47:39 +01:00
// We started with a pseudo exception
$type = 'SERIES-EXCEPTION-PROPAGATE' ;
}
else
{
$type = 'SERIES-EXCEPTION' ;
}
2010-02-03 13:24:42 +01:00
2010-02-10 20:47:39 +01:00
if ( $stored_event )
{
unset ( $stored_event [ 'id' ]); // signal the true exception
$stored_event [ 'recur_type' ] = MCAL_RECUR_NONE ;
2010-01-29 22:42:54 +01:00
}
2010-02-10 20:47:39 +01:00
break ;
2010-01-14 18:01:30 +01:00
}
2010-03-16 10:26:01 +01:00
}
2010-02-10 20:47:39 +01:00
// the event id here is always the id of the master event
// unset it to prevent confusion of stored event and master event
unset ( $event [ 'id' ]);
}
2010-01-14 18:01:30 +01:00
2010-02-10 20:47:39 +01:00
// check ACL
if ( is_array ( $master_event ))
{
2016-05-01 19:47:59 +02:00
$acl_edit = $this -> check_perms ( Acl :: EDIT , $master_event [ 'id' ]);
2010-02-10 20:47:39 +01:00
}
else
{
if ( is_array ( $stored_event ))
2010-01-14 18:01:30 +01:00
{
2016-05-01 19:47:59 +02:00
$acl_edit = $this -> check_perms ( Acl :: EDIT , $stored_event [ 'id' ]);
2010-01-14 18:01:30 +01:00
}
else
{
2010-02-10 20:47:39 +01:00
$acl_edit = true ; // new event
2010-01-14 18:01:30 +01:00
}
2010-02-10 20:47:39 +01:00
}
2010-01-14 18:01:30 +01:00
2010-02-10 20:47:39 +01:00
return array (
'type' => $type ,
'acl_edit' => $acl_edit ,
'stored_event' => $stored_event ,
'master_event' => $master_event ,
);
2010-01-14 18:01:30 +01:00
}
2010-01-29 22:42:54 +01:00
2010-02-09 22:56:39 +01:00
/**
2010-02-04 13:08:03 +01:00
* Translates all timestamps for a given event from server - time to user - time .
2010-01-29 22:42:54 +01:00
* The update () and save () methods expect timestamps in user - time .
* @ param & $event the event we are working on
*
*/
function server2usertime ( & $event )
{
// we run all dates through date2usertime, to adjust to user-time
foreach ( array ( 'start' , 'end' , 'recur_enddate' , 'recurrence' ) as $ts )
{
// we convert here from server-time to timestamps in user-time!
if ( isset ( $event [ $ts ])) $event [ $ts ] = $event [ $ts ] ? $this -> date2usertime ( $event [ $ts ]) : 0 ;
}
2024-06-04 15:30:54 +02:00
// same with the recur exceptions and rdates
foreach ([ 'recur_exception' , 'recur_rdates' ] as $name )
2010-01-29 22:42:54 +01:00
{
2024-06-04 15:30:54 +02:00
foreach ( $event [ $name ] ? ? [] as $n => $date )
2010-01-29 22:42:54 +01:00
{
2024-06-04 15:30:54 +02:00
$event [ $name ][ $n ] = $this -> date2usertime ( $date );
2010-01-29 22:42:54 +01:00
}
}
// same with the alarms
if ( isset ( $event [ 'alarm' ]) && is_array ( $event [ 'alarm' ]))
{
foreach ( $event [ 'alarm' ] as $id => $alarm )
{
2023-06-09 18:27:28 +02:00
$event [ 'alarm' ][ $id ][ 'time' ] = $this -> date2usertime ( $alarm [ 'time' ] ? ? null );
2010-01-29 22:42:54 +01:00
}
}
}
2024-06-04 15:30:54 +02:00
2010-04-22 22:27:14 +02:00
/**
2011-06-14 20:16:08 +02:00
* Delete events that are more than $age years old
*
* Purges old events from the database
*
* @ param int | float $age How many years old the event must be before it is deleted
*/
2010-04-22 22:27:14 +02:00
function purge ( $age )
{
2014-07-02 14:42:12 +02:00
if ( is_numeric ( $age ) && $age > 0 ) // just make sure bogus values dont delete everything
{
$this -> so -> purge ( time () - 365 * 24 * 3600 * ( float ) $age );
}
2010-04-22 22:27:14 +02:00
}
2019-06-18 23:11:53 +02:00
/**
* Check to see if we need to reset the status of any of the participants
* - Current user is never reset
* - Other users we respect their preference
* - Non - users status is always reset
*
* @ param Array $event New event
* @ param Array $old_event Event before modification
*
* @ return boolean true if any statuses were reset
*/
2020-05-19 18:44:56 +02:00
protected function check_reset_statuses ( & $event , $old_event )
2019-06-18 23:11:53 +02:00
{
2020-06-25 17:38:34 +02:00
// Event passed is still in user time at this point, convert to servertime for
// comparison, but don't modify it in event
if ( ! $old_event || ! is_array ( $old_event ) || $this -> date2ts ( $event [ 'start' ], true ) == $old_event [ 'start' ])
2019-06-18 23:11:53 +02:00
{
return false ;
}
$status_reset = false ;
$sameday = ( date ( 'Ymd' , $old_event [ 'start' ]) == date ( 'Ymd' , $event [ 'start' ]));
foreach (( array ) $event [ 'participants' ] as $uid => $status )
{
$q = $r = null ;
2023-04-19 19:00:35 +02:00
calendar_so :: split_status ( $status , $q , $r );
2019-06-18 23:11:53 +02:00
if ( $uid == $this -> user || $status == 'U' )
{
continue ;
}
2023-04-19 19:00:35 +02:00
if ( is_int ( $uid ))
2019-06-18 23:11:53 +02:00
{
2023-04-19 19:00:35 +02:00
// Just user accounts
2019-08-14 22:32:35 +02:00
$preferences = new Api\Preferences ( $uid );
2019-06-18 23:11:53 +02:00
}
else
{
2023-04-19 19:00:35 +02:00
// For non-users, use the owner's preference
$preferences = new Api\Preferences ( $event [ 'owner' ]);
}
$part_prefs = $preferences -> read_repository ();
switch ( $part_prefs [ 'calendar' ][ 'reset_stati' ])
{
case 'no' :
break ;
case 'startday' :
if ( $sameday )
{
break ;
}
default :
$status_reset = true ;
$event [ 'participants' ][ $uid ] = calendar_so :: combine_status ( 'U' , $q , $r );
2019-06-18 23:11:53 +02:00
}
2023-04-19 19:00:35 +02:00
2019-06-18 23:11:53 +02:00
}
return $status_reset ;
}
2022-04-24 18:15:47 +02:00
}