forked from friendica/friendica-addons
		
	Exporting and Importing ICS-Files
This commit is contained in:
		
					parent
					
						
							
								a96c8ce670
							
						
					
				
			
			
				commit
				
					
						66384e2e2c
					
				
			
		
					 16 changed files with 245 additions and 47 deletions
				
			
		|  | @ -31,6 +31,8 @@ | ||||||
| 	* Fixed: Uploaded VCards without a UID are now rejected. (thanks Dominik!) | 	* Fixed: Uploaded VCards without a UID are now rejected. (thanks Dominik!) | ||||||
| 	* Fixed: Rejecting calendar objects if they are not in the | 	* Fixed: Rejecting calendar objects if they are not in the | ||||||
| 	  supported-calendar-component list. (thanks Armin!) | 	  supported-calendar-component list. (thanks Armin!) | ||||||
|  | 	* Fixed: Workaround for 10.8 Mountain Lion vCards, as it needs \r line | ||||||
|  | 	  endings to parse them correctly. | ||||||
| 
 | 
 | ||||||
| 1.6.4-stable (2012-??-??) | 1.6.4-stable (2012-??-??) | ||||||
| 	* Fixed: Issue 220: Calendar-query filters may fail when filtering on | 	* Fixed: Issue 220: Calendar-query filters may fail when filtering on | ||||||
|  | @ -40,6 +42,13 @@ | ||||||
| 	  requests. | 	  requests. | ||||||
| 	* Fixed: Problem with POST requests to the outbox if mailto: was not lower | 	* Fixed: Problem with POST requests to the outbox if mailto: was not lower | ||||||
| 	  cased. | 	  cased. | ||||||
|  | 	* Fixed: Yearly recurrence rule expansion on leap-days no behaves | ||||||
|  | 	  correctly. | ||||||
|  | 	* Fixed: Correctly checking if recurring, all-day events with no dtstart | ||||||
|  | 	  fall in a timerange if the start of the time-range exceeds the start of | ||||||
|  | 	  the instance of an event, but not the end. | ||||||
|  | 	* Fixed: All-day recurring events wouldn't match if an occurence ended | ||||||
|  | 	  exactly on the start of a time-range. | ||||||
| 
 | 
 | ||||||
| 1.6.3-stable (2012-06-12) | 1.6.3-stable (2012-06-12) | ||||||
| 	* Added: It's now possible to specify in Sabre_DAV_Client which type of | 	* Added: It's now possible to specify in Sabre_DAV_Client which type of | ||||||
|  |  | ||||||
|  | @ -16,8 +16,13 @@ class Sabre_CalDAV_Backend_PDO extends Sabre_CalDAV_Backend_Abstract { | ||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
|      * We need to specify a max date, because we need to stop *somewhere* |      * We need to specify a max date, because we need to stop *somewhere* | ||||||
|  |      * | ||||||
|  |      * On 32 bit system the maximum for a signed integer is 2147483647, so | ||||||
|  |      * MAX_DATE cannot be higher than date('Y-m-d', 2147483647) which results | ||||||
|  |      * in 2038-01-19 to avoid problems when the date is converted | ||||||
|  |      * to a unix timestamp. | ||||||
|      */ |      */ | ||||||
|     const MAX_DATE = '2040-01-01'; |     const MAX_DATE = '2038-01-01'; | ||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
|      * pdo |      * pdo | ||||||
|  |  | ||||||
|  | @ -154,8 +154,7 @@ class Sabre_CardDAV_Plugin extends Sabre_DAV_ServerPlugin { | ||||||
|                 if (is_resource($val)) |                 if (is_resource($val)) | ||||||
|                     $val = stream_get_contents($val); |                     $val = stream_get_contents($val); | ||||||
| 
 | 
 | ||||||
|                 // Taking out \r to not screw up the xml output
 |                 $returnedProperties[200][$addressDataProp] = $val; | ||||||
|                 $returnedProperties[200][$addressDataProp] = str_replace("\r","", $val); |  | ||||||
| 
 | 
 | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|  |  | ||||||
|  | @ -13,8 +13,6 @@ | ||||||
|  */ |  */ | ||||||
| abstract class Sabre_DAV_Property implements Sabre_DAV_PropertyInterface { | abstract class Sabre_DAV_Property implements Sabre_DAV_PropertyInterface { | ||||||
| 
 | 
 | ||||||
|     abstract function serialize(Sabre_DAV_Server $server, DOMElement $prop); |  | ||||||
| 
 |  | ||||||
|     static function unserialize(DOMElement $prop) { |     static function unserialize(DOMElement $prop) { | ||||||
| 
 | 
 | ||||||
|         throw new Sabre_DAV_Exception('Unserialize has not been implemented for this class'); |         throw new Sabre_DAV_Exception('Unserialize has not been implemented for this class'); | ||||||
|  |  | ||||||
|  | @ -337,6 +337,8 @@ class Sabre_VObject_RecurrenceIterator implements Iterator { | ||||||
|             $this->endDate = clone $this->startDate; |             $this->endDate = clone $this->startDate; | ||||||
|             if (isset($this->baseEvent->DURATION)) { |             if (isset($this->baseEvent->DURATION)) { | ||||||
|                 $this->endDate->add(Sabre_VObject_DateTimeParser::parse($this->baseEvent->DURATION->value)); |                 $this->endDate->add(Sabre_VObject_DateTimeParser::parse($this->baseEvent->DURATION->value)); | ||||||
|  |             } elseif ($this->baseEvent->DTSTART->getDateType()===Sabre_VObject_Property_DateTime::DATE) { | ||||||
|  |                 $this->endDate->modify('+1 day'); | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|         $this->currentDate = clone $this->startDate; |         $this->currentDate = clone $this->startDate; | ||||||
|  | @ -565,7 +567,7 @@ class Sabre_VObject_RecurrenceIterator implements Iterator { | ||||||
|      */ |      */ | ||||||
|     public function fastForward(DateTime $dt) { |     public function fastForward(DateTime $dt) { | ||||||
| 
 | 
 | ||||||
|         while($this->valid() && $this->getDTEnd() < $dt) { |         while($this->valid() && $this->getDTEnd() <= $dt) { | ||||||
|             $this->next(); |             $this->next(); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|  | @ -838,9 +840,40 @@ class Sabre_VObject_RecurrenceIterator implements Iterator { | ||||||
|      */ |      */ | ||||||
|     protected function nextYearly() { |     protected function nextYearly() { | ||||||
| 
 | 
 | ||||||
|  |         $currentMonth = $this->currentDate->format('n'); | ||||||
|  |         $currentYear = $this->currentDate->format('Y'); | ||||||
|  |         $currentDayOfMonth = $this->currentDate->format('j'); | ||||||
|  | 
 | ||||||
|  |         // No sub-rules, so we just advance by year
 | ||||||
|         if (!$this->byMonth) { |         if (!$this->byMonth) { | ||||||
|  | 
 | ||||||
|  |             // Unless it was a leap day!
 | ||||||
|  |             if ($currentMonth==2 && $currentDayOfMonth==29) { | ||||||
|  | 
 | ||||||
|  |                 $counter = 0; | ||||||
|  |                 do { | ||||||
|  |                     $counter++; | ||||||
|  |                     // Here we increase the year count by the interval, until
 | ||||||
|  |                     // we hit a date that's also in a leap year.
 | ||||||
|  |                     //
 | ||||||
|  |                     // We could just find the next interval that's dividable by
 | ||||||
|  |                     // 4, but that would ignore the rule that there's no leap
 | ||||||
|  |                     // year every year that's dividable by a 100, but not by
 | ||||||
|  |                     // 400. (1800, 1900, 2100). So we just rely on the datetime
 | ||||||
|  |                     // functions instead.
 | ||||||
|  |                     $nextDate = clone $this->currentDate; | ||||||
|  |                     $nextDate->modify('+ ' . ($this->interval*$counter) . ' years'); | ||||||
|  |                 } while ($nextDate->format('n')!=2); | ||||||
|  |                 $this->currentDate = $nextDate; | ||||||
|  | 
 | ||||||
|  |                 return; | ||||||
|  | 
 | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             // The easiest form
 | ||||||
|             $this->currentDate->modify('+' . $this->interval . ' years'); |             $this->currentDate->modify('+' . $this->interval . ' years'); | ||||||
|             return; |             return; | ||||||
|  | 
 | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         $currentMonth = $this->currentDate->format('n'); |         $currentMonth = $this->currentDate->format('n'); | ||||||
|  | @ -892,8 +925,8 @@ class Sabre_VObject_RecurrenceIterator implements Iterator { | ||||||
| 
 | 
 | ||||||
|         } else { |         } else { | ||||||
| 
 | 
 | ||||||
|             // no byDay or byMonthDay, so we can just loop through the
 |             // These are the 'byMonth' rules, if there are no byDay or
 | ||||||
|             // months.
 |             // byMonthDay sub-rules.
 | ||||||
|             do { |             do { | ||||||
| 
 | 
 | ||||||
|                 $currentMonth++; |                 $currentMonth++; | ||||||
|  | @ -903,6 +936,7 @@ class Sabre_VObject_RecurrenceIterator implements Iterator { | ||||||
|                 } |                 } | ||||||
|             } while (!in_array($currentMonth, $this->byMonth)); |             } while (!in_array($currentMonth, $this->byMonth)); | ||||||
|             $this->currentDate->setDate($currentYear, $currentMonth, $currentDayOfMonth); |             $this->currentDate->setDate($currentYear, $currentMonth, $currentDayOfMonth); | ||||||
|  | 
 | ||||||
|             return; |             return; | ||||||
| 
 | 
 | ||||||
|         } |         } | ||||||
|  |  | ||||||
|  | @ -14,7 +14,7 @@ class Sabre_VObject_Version { | ||||||
|     /** |     /** | ||||||
|      * Full version number |      * Full version number | ||||||
|      */ |      */ | ||||||
|     const VERSION = '1.3.3'; |     const VERSION = '1.3.4'; | ||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
|      * Stability : alpha, beta, stable |      * Stability : alpha, beta, stable | ||||||
|  |  | ||||||
|  | @ -343,6 +343,14 @@ DURATION:PT1H | ||||||
| RRULE:FREQ=YEARLY | RRULE:FREQ=YEARLY | ||||||
| END:VEVENT | END:VEVENT | ||||||
| END:VCALENDAR | END:VCALENDAR | ||||||
|  | yow; | ||||||
|  |         $blob33 = <<<yow | ||||||
|  | BEGIN:VCALENDAR | ||||||
|  | BEGIN:VEVENT | ||||||
|  | DTSTART;VALUE=DATE:20120628 | ||||||
|  | RRULE:FREQ=DAILY | ||||||
|  | END:VEVENT | ||||||
|  | END:VCALENDAR | ||||||
| yow; | yow; | ||||||
| 
 | 
 | ||||||
|         $filter1 = array( |         $filter1 = array( | ||||||
|  | @ -604,8 +612,16 @@ yow; | ||||||
|             'time-range' => null, |             'time-range' => null, | ||||||
|         ); |         ); | ||||||
| 
 | 
 | ||||||
|         // Time-range with RRULE
 |         $filter38 = array( | ||||||
| 
 |             'name' => 'VEVENT', | ||||||
|  |             'comp-filters' => array(), | ||||||
|  |             'prop-filters' => array(), | ||||||
|  |             'is-not-defined' => false, | ||||||
|  |             'time-range' => array( | ||||||
|  |                 'start' => new DateTime('2012-07-01 00:00:00', new DateTimeZone('UTC')), | ||||||
|  |                 'end' => new DateTime('2012-08-01 00:00:00', new DateTimeZone('UTC')), | ||||||
|  |             ) | ||||||
|  |         ); | ||||||
| 
 | 
 | ||||||
|         return array( |         return array( | ||||||
|             // Component check
 |             // Component check
 | ||||||
|  | @ -741,6 +757,9 @@ yow; | ||||||
|             array($blob31, $filter20, 1), |             array($blob31, $filter20, 1), | ||||||
|             array($blob32, $filter20, 0), |             array($blob32, $filter20, 0), | ||||||
| 
 | 
 | ||||||
|  |             // Bug reported on mailing list, related to all-day events.
 | ||||||
|  |             array($blob33, $filter38, 1), | ||||||
|  | 
 | ||||||
|         ); |         ); | ||||||
| 
 | 
 | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  | @ -53,9 +53,14 @@ class Sabre_VObject_Component_VEventTest extends PHPUnit_Framework_TestCase { | ||||||
|         $vevent6->DTEND['VALUE'] = 'DATE'; |         $vevent6->DTEND['VALUE'] = 'DATE'; | ||||||
|         $tests[] = array($vevent6, new DateTime('2011-01-01'), new DateTime('2012-01-01'), true); |         $tests[] = array($vevent6, new DateTime('2011-01-01'), new DateTime('2012-01-01'), true); | ||||||
|         $tests[] = array($vevent6, new DateTime('2011-01-01'), new DateTime('2011-11-01'), false); |         $tests[] = array($vevent6, new DateTime('2011-01-01'), new DateTime('2011-11-01'), false); | ||||||
|         // Event with no end date should be treated as lasting the entire day.
 |  | ||||||
|         $tests[] = array($vevent6, new DateTime('2011-12-25 16:00:00'), new DateTime('2011-12-25 17:00:00'), true); |  | ||||||
| 
 | 
 | ||||||
|  |         // Added this test to ensure that recurrence rules with no DTEND also 
 | ||||||
|  |         // get checked for the entire day.
 | ||||||
|  |         $vevent7 = clone $vevent; | ||||||
|  |         $vevent7->DTSTART = '20120101'; | ||||||
|  |         $vevent7->DTSTART['VALUE'] = 'DATE'; | ||||||
|  |         $vevent7->RRULE = 'FREQ=MONTHLY'; | ||||||
|  |         $tests[] = array($vevent7, new DateTime('2012-02-01 15:00:00'), new DateTime('2012-02-02'), true); | ||||||
|         return $tests; |         return $tests; | ||||||
| 
 | 
 | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  | @ -707,6 +707,51 @@ class Sabre_VObject_RecurrenceIteratorTest extends PHPUnit_Framework_TestCase { | ||||||
| 
 | 
 | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     /** | ||||||
|  |      * @depends testValues | ||||||
|  |      */ | ||||||
|  |     function testYearlyLeapYear() { | ||||||
|  | 
 | ||||||
|  |         $ev = new Sabre_VObject_Component('VEVENT'); | ||||||
|  |         $ev->UID = 'bla'; | ||||||
|  |         $ev->RRULE = 'FREQ=YEARLY;COUNT=3'; | ||||||
|  |         $dtStart = new Sabre_VObject_Property_DateTime('DTSTART'); | ||||||
|  |         $dtStart->setDateTime(new DateTime('2012-02-29'),Sabre_VObject_Property_DateTime::UTC); | ||||||
|  | 
 | ||||||
|  |         $ev->add($dtStart); | ||||||
|  | 
 | ||||||
|  |         $vcal = Sabre_VObject_Component::create('VCALENDAR'); | ||||||
|  |         $vcal->add($ev); | ||||||
|  | 
 | ||||||
|  |         $it = new Sabre_VObject_RecurrenceIterator($vcal,(string)$ev->uid); | ||||||
|  | 
 | ||||||
|  |         $this->assertEquals('yearly', $it->frequency); | ||||||
|  |         $this->assertEquals(3, $it->count); | ||||||
|  | 
 | ||||||
|  |         $max = 20; | ||||||
|  |         $result = array(); | ||||||
|  |         foreach($it as $k=>$item) { | ||||||
|  | 
 | ||||||
|  |             $result[] = $item; | ||||||
|  |             $max--; | ||||||
|  | 
 | ||||||
|  |             if (!$max) break; | ||||||
|  | 
 | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         $tz = new DateTimeZone('UTC'); | ||||||
|  | 
 | ||||||
|  |         $this->assertEquals( | ||||||
|  |             array( | ||||||
|  |                 new DateTime('2012-02-29', $tz), | ||||||
|  |                 new DateTime('2016-02-29', $tz), | ||||||
|  |                 new DateTime('2020-02-29', $tz), | ||||||
|  |             ), | ||||||
|  |             $result | ||||||
|  |         ); | ||||||
|  | 
 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     /** |     /** | ||||||
|      * @depends testValues |      * @depends testValues | ||||||
|      */ |      */ | ||||||
|  |  | ||||||
|  | @ -186,19 +186,22 @@ function wdcal_create_std_calendars() | ||||||
| 	$a = get_app(); | 	$a = get_app(); | ||||||
| 	if (!local_user()) return; | 	if (!local_user()) return; | ||||||
| 
 | 
 | ||||||
|  | 	$privates = q("SELECT COUNT(*) num FROM %s%scalendars WHERE `namespace` = %d AND `namespace_id` = %d", CALDAV_SQL_DB, CALDAV_SQL_PREFIX, CALDAV_NAMESPACE_PRIVATE, IntVal($a->user["uid"])); | ||||||
|  | 	if ($privates[0]["num"] > 0) return; | ||||||
|  | 
 | ||||||
| 	$uris = array( | 	$uris = array( | ||||||
| 		'private'                 => t("Private Calendar"), | 		'private'                 => t("Private Calendar"), | ||||||
| 		CALDAV_FRIENDICA_MINE     => t("Friendica Events: Mine"), | 		CALDAV_FRIENDICA_MINE     => t("Friendica Events: Mine"), | ||||||
| 		CALDAV_FRIENDICA_CONTACTS => t("Friendica Events: Contacts"), | 		CALDAV_FRIENDICA_CONTACTS => t("Friendica Events: Contacts"), | ||||||
| 	); | 	); | ||||||
| 	foreach ($uris as $uri => $name) { | 	foreach ($uris as $uri => $name) { | ||||||
| 
 |  | ||||||
| 		$cals = q("SELECT * FROM %s%scalendars WHERE `namespace` = %d AND `namespace_id` = %d AND `uri` = '%s'", | 		$cals = q("SELECT * FROM %s%scalendars WHERE `namespace` = %d AND `namespace_id` = %d AND `uri` = '%s'", | ||||||
| 			CALDAV_SQL_DB, CALDAV_SQL_PREFIX, CALDAV_NAMESPACE_PRIVATE, $a->user["uid"], dbesc($uri)); | 			CALDAV_SQL_DB, CALDAV_SQL_PREFIX, CALDAV_NAMESPACE_PRIVATE, $a->user["uid"], dbesc($uri)); | ||||||
| 		if (count($cals) == 0) { | 		if (count($cals) == 0) { | ||||||
| 			q("INSERT INTO %s%scalendars (`namespace`, `namespace_id`, `displayname`, `timezone`, `ctag`, `uri`, `has_vevent`, `has_vtodo`) VALUES (%d, %d, %d, '%s', '%s', 1, '%s', 1, 0)", | 			q("INSERT INTO %s%scalendars (`namespace`, `namespace_id`, `displayname`, `timezone`, `ctag`, `uri`, `has_vevent`, `has_vtodo`) VALUES (%d, %d, '%s', '%s', 1, '%s', 1, 0)", | ||||||
| 				CALDAV_SQL_DB, CALDAV_SQL_PREFIX, CALDAV_NAMESPACE_PRIVATE, IntVal($a->user["uid"]), dbesc($name), dbesc($a->timezone), dbesc($uri) | 				CALDAV_SQL_DB, CALDAV_SQL_PREFIX, CALDAV_NAMESPACE_PRIVATE, IntVal($a->user["uid"]), dbesc($name), dbesc($a->timezone), dbesc($uri) | ||||||
| 			); | 			); | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  | 
 | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -185,12 +185,7 @@ function wdcal_mySql2icalTime($myqlDate) | ||||||
|  */ |  */ | ||||||
| function icalendar_sanitize_string($str = "") | function icalendar_sanitize_string($str = "") | ||||||
| { | { | ||||||
| 	$str = str_replace("\r\n", "\n", $str); | 	return preg_replace("/[\\r\\n]+/siu", "\r\n", $str); | ||||||
| 	$str = str_replace("\n\r", "\n", $str); |  | ||||||
| 	$str = str_replace("\r", "\n", $str); |  | ||||||
| 	$str = str_replace("\n\n", "\n", $str); |  | ||||||
| 	$str = str_replace("\n\n", "\n", $str); |  | ||||||
| 	return $str; |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  | @ -361,10 +356,8 @@ function dav_create_empty_vevent($uid = "") | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
| /** | /** | ||||||
|  * @param Sabre_VObject_Component_VEvent $vObject |  * @param Sabre_VObject_Component_VCalendar $vObject | ||||||
|  * @return Sabre_VObject_Component_VEvent|null |  * @return Sabre_VObject_Component_VEvent|null | ||||||
|  */ |  */ | ||||||
| function dav_get_eventComponent(&$vObject) | function dav_get_eventComponent(&$vObject) | ||||||
|  |  | ||||||
|  | @ -130,6 +130,7 @@ abstract class Sabre_CalDAV_Backend_Common extends Sabre_CalDAV_Backend_Abstract | ||||||
| 		$component      = null; | 		$component      = null; | ||||||
| 		$firstOccurence = null; | 		$firstOccurence = null; | ||||||
| 		$lastOccurence  = null; | 		$lastOccurence  = null; | ||||||
|  | 
 | ||||||
| 		foreach ($vObject->getComponents() as $component) { | 		foreach ($vObject->getComponents() as $component) { | ||||||
| 			if ($component->name !== 'VTIMEZONE') { | 			if ($component->name !== 'VTIMEZONE') { | ||||||
| 				$componentType = $component->name; | 				$componentType = $component->name; | ||||||
|  |  | ||||||
|  | @ -431,7 +431,6 @@ class Sabre_CalDAV_Backend_Private extends Sabre_CalDAV_Backend_Common | ||||||
| 	 */ | 	 */ | ||||||
| 	function createCalendarObject($calendarId, $objectUri, $calendarData) | 	function createCalendarObject($calendarId, $objectUri, $calendarData) | ||||||
| 	{ | 	{ | ||||||
| 
 |  | ||||||
| 		$calendarData = icalendar_sanitize_string($calendarData); | 		$calendarData = icalendar_sanitize_string($calendarData); | ||||||
| 
 | 
 | ||||||
| 		$extraData = $this->getDenormalizedData($calendarData); | 		$extraData = $this->getDenormalizedData($calendarData); | ||||||
|  |  | ||||||
|  | @ -325,7 +325,7 @@ function wdcal_getEditPage_str(&$localization, $baseurl, $uid, $calendar_id, $ur | ||||||
| 	$out .= "</div>\n"; | 	$out .= "</div>\n"; | ||||||
| 
 | 
 | ||||||
| 	if ($recurrence->frequency == "yearly") { | 	if ($recurrence->frequency == "yearly") { | ||||||
| 		if ($recurrence->byMonth != IntVal(date("m", $event["StartTime"]))) notice("The recurrence of this event cannot be parsed"); | 		if (count($recurrence->byMonth) != 1 || $recurrence->byMonth[0] != date("n", $event["StartTime"])) notice("The recurrence of this event cannot be parsed!"); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	$out .= "<div class='rec_yearly'>"; | 	$out .= "<div class='rec_yearly'>"; | ||||||
|  |  | ||||||
|  | @ -39,26 +39,19 @@ function wdcal_addRequiredHeaders() | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| /** | /** | ||||||
|  * @param array|int[] $calendars |  * @param int $calendar_id | ||||||
|  */ |  */ | ||||||
| function wdcal_print_user_ics($calendars = array()) | function wdcal_print_user_ics($calendar_id) | ||||||
| { | { | ||||||
| 	$add = ""; | 	$calendar_id = IntVal($calendar_id); | ||||||
| 	if (count($calendars) > 0) { |  | ||||||
| 		$c = array(); |  | ||||||
| 		foreach ($calendars as $i) $c[] = IntVal($i); |  | ||||||
| 		$add = " AND `id` IN (" . implode(", ", $c) . ")"; |  | ||||||
| 	} |  | ||||||
| 
 | 
 | ||||||
| 	$a = get_app(); | 	$a = get_app(); | ||||||
| 	header("Content-type: text/plain"); | 	header("Content-type: text/plain"); | ||||||
| 
 | 
 | ||||||
| 	$str  = "BEGIN:VCALENDAR\r\nVERSION:2.0\r\nPRODID:-//Friendica//DAV-Plugin//EN\r\n"; | 	$str  = "BEGIN:VCALENDAR\r\nVERSION:2.0\r\nPRODID:-//Friendica//DAV-Plugin//EN\r\n"; | ||||||
| 	$cals = q("SELECT * FROM %s%scalendars WHERE `namespace` = %d AND `namespace_id` = %d %s", CALDAV_SQL_DB, CALDAV_SQL_PREFIX, CALDAV_NAMESPACE_PRIVATE, $a->user["uid"], $add); | 	$cals = q("SELECT * FROM %s%scalendars WHERE `id` = %d AND `namespace` = %d AND `namespace_id` = %d", CALDAV_SQL_DB, CALDAV_SQL_PREFIX, $calendar_id, CALDAV_NAMESPACE_PRIVATE, $a->user["uid"]); | ||||||
| 	if (count($cals) > 0) { | 	if (count($cals) > 0) { | ||||||
| 		$ids = array(); | 		$objs = q("SELECT * FROM %s%scalendarobjects WHERE `calendar_id` = %d ORDER BY `firstOccurence`", CALDAV_SQL_DB, CALDAV_SQL_PREFIX, $calendar_id); | ||||||
| 		foreach ($cals as $c) $ids[] = IntVal($c["id"]); |  | ||||||
| 		$objs = q("SELECT * FROM %s%scalendarobjects WHERE `calendar_id` IN (" . implode(", ", $ids) . ") ORDER BY `firstOccurence`", CALDAV_SQL_DB, CALDAV_SQL_PREFIX); |  | ||||||
| 
 | 
 | ||||||
| 		foreach ($objs as $obj) { | 		foreach ($objs as $obj) { | ||||||
| 			preg_match("/BEGIN:VEVENT(.*)END:VEVENT/siu", $obj["calendardata"], $matches); | 			preg_match("/BEGIN:VEVENT(.*)END:VEVENT/siu", $obj["calendardata"], $matches); | ||||||
|  | @ -74,6 +67,95 @@ function wdcal_print_user_ics($calendars = array()) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  | /** | ||||||
|  |  * @param int $calendar_id | ||||||
|  |  * @return string | ||||||
|  |  */ | ||||||
|  | function wdcal_import_user_ics($calendar_id) { | ||||||
|  | 	$a = get_app(); | ||||||
|  | 	$calendar_id = IntVal($calendar_id); | ||||||
|  | 	$o = ""; | ||||||
|  | 
 | ||||||
|  | 	$server = dav_create_server(true, true, false); | ||||||
|  | 	$calendar = dav_get_current_user_calendar_by_id($server, $calendar_id, DAV_ACL_WRITE); | ||||||
|  | 	if (!$calendar) goaway($a->get_baseurl() . "/dav/wdcal/"); | ||||||
|  | 
 | ||||||
|  | 	if (isset($_REQUEST["save"])) { | ||||||
|  | 		check_form_security_token_redirectOnErr('/dav/settings/', 'icsimport'); | ||||||
|  | 
 | ||||||
|  | 		if ($_FILES["ics_file"]["tmp_name"] != "" && is_uploaded_file($_FILES["ics_file"]["tmp_name"])) try { | ||||||
|  | 			$text = file_get_contents($_FILES["ics_file"]["tmp_name"]); | ||||||
|  | 
 | ||||||
|  | 			/** @var Sabre_VObject_Component_VCalendar $vObject  */ | ||||||
|  | 			$vObject        = Sabre_VObject_Reader::read($text); | ||||||
|  | 			$comp = $vObject->getComponents(); | ||||||
|  | 			$imported = array(); | ||||||
|  | 			foreach ($comp as $c) try { | ||||||
|  | 				/** @var Sabre_VObject_Component_VEvent $c */ | ||||||
|  | 				$uid = $c->__get("UID")->value; | ||||||
|  | 				if (!isset($imported[$uid])) $imported[$uid] = ""; | ||||||
|  | 				$imported[$uid] .= $c->serialize(); | ||||||
|  | 			} catch (Exception $e) { | ||||||
|  | 				notice(t("Something went wrong when trying to import the file. Sorry. Maybe some events were imported anyway.")); | ||||||
|  | 			} | ||||||
|  | 
 | ||||||
|  | 			if (isset($_REQUEST["overwrite"])) { | ||||||
|  | 				$children = $calendar->getChildren(); | ||||||
|  | 				foreach ($children as $child) { | ||||||
|  | 					/** @var Sabre_CalDAV_CalendarObject $child */ | ||||||
|  | 					$child->delete(); | ||||||
|  | 				} | ||||||
|  | 				$i = 1; | ||||||
|  | 			} else { | ||||||
|  | 				$i = 0; | ||||||
|  | 				$children = $calendar->getChildren(); | ||||||
|  | 				foreach ($children as $child) { | ||||||
|  | 					/** @var Sabre_CalDAV_CalendarObject $child */ | ||||||
|  | 					$name = $child->getName(); | ||||||
|  | 					if (preg_match("/import\-([0-9]+)\.ics/siu", $name, $matches)) { | ||||||
|  | 						if ($matches[1] > $i) $i = $matches[1]; | ||||||
|  | 					}; | ||||||
|  | 				} | ||||||
|  | 				$i++; | ||||||
|  | 			} | ||||||
|  | 
 | ||||||
|  | 			foreach ($imported as $object) try { | ||||||
|  | 
 | ||||||
|  | 				$str = "BEGIN:VCALENDAR\r\nVERSION:2.0\r\nPRODID:-//Friendica//DAV-Plugin//EN\r\n"; | ||||||
|  | 				$str .= trim($object); | ||||||
|  | 				$str .= "\r\nEND:VCALENDAR\r\n"; | ||||||
|  | 
 | ||||||
|  | 				$calendar->createFile("import-" . $i . ".ics", $str); | ||||||
|  | 				$i++; | ||||||
|  | 			} catch (Exception $e) { | ||||||
|  | 				notice(t("Something went wrong when trying to import the file. Sorry.")); | ||||||
|  | 			} | ||||||
|  | 
 | ||||||
|  | 			$o = t("The ICS-File has been imported."); | ||||||
|  | 		} catch (Exception $e) { | ||||||
|  | 			notice(t("Something went wrong when trying to import the file. Sorry. Maybe some events were imported anyway.")); | ||||||
|  | 		} else { | ||||||
|  | 			notice(t("No file was uploaded.")); | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | 	$o .= "<a href='" . $a->get_baseurl() . "/dav/wdcal/'>" . t("Go back to the calendar") . "</a><br><br>"; | ||||||
|  | 
 | ||||||
|  | 	$num = q("SELECT COUNT(*) num FROM %s%scalendarobjects WHERE `calendar_id` = %d", CALDAV_SQL_DB, CALDAV_SQL_PREFIX, $calendar_id); | ||||||
|  | 
 | ||||||
|  | 	$o .= "<h2>" . t("Import a ICS-file") . "</h2>"; | ||||||
|  | 	$o .= '<form method="POST" action="' . $a->get_baseurl() . '/dav/wdcal/' . $calendar_id . '/ics-import/" enctype="multipart/form-data">'; | ||||||
|  | 	$o .= "<input type='hidden' name='form_security_token' value='" . get_form_security_token('icsimport') . "'>\n"; | ||||||
|  | 	$o .= "<label for='ics_file'>" . t("ICS-File") . "</label><input type='file' name='ics_file' id='ics_file'><br>\n"; | ||||||
|  | 	if ($num[0]["num"] > 0) $o .= "<label for='overwrite'>" . str_replace("#num#", $num[0]["num"], t("Overwrite all #num# existing events")) . "</label> <input name='overwrite' id='overwrite' type='checkbox'><br>\n"; | ||||||
|  | 	$o .= "<input type='submit' name='save' value='" . t("Upload") . "'>"; | ||||||
|  | 	$o .= '</form>'; | ||||||
|  | 
 | ||||||
|  | 	return $o; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
| /** | /** | ||||||
|  * @param array|Sabre_CalDAV_Calendar[] $calendars |  * @param array|Sabre_CalDAV_Calendar[] $calendars | ||||||
|  * @param array|int[] $calendars_selected |  * @param array|int[] $calendars_selected | ||||||
|  | @ -276,13 +358,13 @@ function wdcal_getSettingsPage(&$a) | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	if (isset($_REQUEST["save"])) { | 	if (isset($_REQUEST["save"])) { | ||||||
| 		check_form_security_token_redirectOnErr($a->get_baseurl() . '/dav/settings/', 'calprop'); | 		check_form_security_token_redirectOnErr('/dav/settings/', 'calprop'); | ||||||
| 		set_pconfig($a->user["uid"], "dav", "dateformat", $_REQUEST["wdcal_date_format"]); | 		set_pconfig($a->user["uid"], "dav", "dateformat", $_REQUEST["wdcal_date_format"]); | ||||||
| 		info(t('The new values have been saved.')); | 		info(t('The new values have been saved.')); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	if (isset($_REQUEST["save_cals"])) { | 	if (isset($_REQUEST["save_cals"])) { | ||||||
| 		check_form_security_token_redirectOnErr($a->get_baseurl() . '/dav/settings/', 'calprop'); | 		check_form_security_token_redirectOnErr('/dav/settings/', 'calprop'); | ||||||
| 
 | 
 | ||||||
| 		$r = q("SELECT * FROM %s%scalendars WHERE `namespace` = " . CALDAV_NAMESPACE_PRIVATE . " AND `namespace_id` = %d", CALDAV_SQL_DB, CALDAV_SQL_PREFIX, IntVal($a->user["uid"])); | 		$r = q("SELECT * FROM %s%scalendars WHERE `namespace` = " . CALDAV_NAMESPACE_PRIVATE . " AND `namespace_id` = %d", CALDAV_SQL_DB, CALDAV_SQL_PREFIX, IntVal($a->user["uid"])); | ||||||
| 		foreach ($r as $cal) { | 		foreach ($r as $cal) { | ||||||
|  | @ -315,7 +397,7 @@ function wdcal_getSettingsPage(&$a) | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	if (isset($_REQUEST["remove_cal"])) { | 	if (isset($_REQUEST["remove_cal"])) { | ||||||
| 		check_form_security_token_redirectOnErr($a->get_baseurl() . '/dav/settings/', 'del_cal', 't'); | 		check_form_security_token_redirectOnErr('/dav/settings/', 'del_cal', 't'); | ||||||
| 
 | 
 | ||||||
| 		$c = q("SELECT * FROM %s%scalendars WHERE `id` = %d AND `namespace_id` = %d AND `namespace_id` = %d", | 		$c = q("SELECT * FROM %s%scalendars WHERE `id` = %d AND `namespace_id` = %d AND `namespace_id` = %d", | ||||||
| 			CALDAV_SQL_DB, CALDAV_SQL_PREFIX, IntVal($_REQUEST["remove_cal"]), CALDAV_NAMESPACE_PRIVATE, IntVal($a->user["uid"])); | 			CALDAV_SQL_DB, CALDAV_SQL_PREFIX, IntVal($_REQUEST["remove_cal"]), CALDAV_NAMESPACE_PRIVATE, IntVal($a->user["uid"])); | ||||||
|  | @ -365,7 +447,7 @@ function wdcal_getSettingsPage(&$a) | ||||||
| 	$o .= '<br><br><h3>' . t('Calendars') . '</h3>'; | 	$o .= '<br><br><h3>' . t('Calendars') . '</h3>'; | ||||||
| 	$o .= '<form method="POST" action="' . $a->get_baseurl() . '/dav/settings/">'; | 	$o .= '<form method="POST" action="' . $a->get_baseurl() . '/dav/settings/">'; | ||||||
| 	$o .= "<input type='hidden' name='form_security_token' value='" . get_form_security_token('calprop') . "'>\n"; | 	$o .= "<input type='hidden' name='form_security_token' value='" . get_form_security_token('calprop') . "'>\n"; | ||||||
| 	$o .= "<table><tr><th>Type</th><th>Color</th><th>Name</th><th>URI (for CalDAV)</th></tr>"; | 	$o .= "<table><tr><th>Type</th><th>Color</th><th>Name</th><th>URI (for CalDAV)</th><th>ICS</th></tr>"; | ||||||
| 
 | 
 | ||||||
| 	$r = q("SELECT * FROM %s%scalendars WHERE `namespace` = " . CALDAV_NAMESPACE_PRIVATE . " AND `namespace_id` = %d", CALDAV_SQL_DB, CALDAV_SQL_PREFIX, IntVal($a->user["uid"])); | 	$r = q("SELECT * FROM %s%scalendars WHERE `namespace` = " . CALDAV_NAMESPACE_PRIVATE . " AND `namespace_id` = %d", CALDAV_SQL_DB, CALDAV_SQL_PREFIX, IntVal($a->user["uid"])); | ||||||
| 	$private_max = 0; | 	$private_max = 0; | ||||||
|  | @ -385,7 +467,10 @@ function wdcal_getSettingsPage(&$a) | ||||||
| 		$o .= "<td style='padding: 2px; text-align: center;'><input style='margin-left: 10px; width: 70px;' class='cal_color' name='color[" . $x["id"] . "]' id='cal_color_" . $x["id"] . "' value='#" . (strlen($x["calendarcolor"]) != 6 ? "5858ff" : escape_tags($x["calendarcolor"])) . "'></td>"; | 		$o .= "<td style='padding: 2px; text-align: center;'><input style='margin-left: 10px; width: 70px;' class='cal_color' name='color[" . $x["id"] . "]' id='cal_color_" . $x["id"] . "' value='#" . (strlen($x["calendarcolor"]) != 6 ? "5858ff" : escape_tags($x["calendarcolor"])) . "'></td>"; | ||||||
| 		$o .= "<td style='padding: 2px;'><input style='margin-left: 10px;' name='name[" . $x["id"] . "]' value='" . escape_tags($x["displayname"]) . "' $disabled></td>"; | 		$o .= "<td style='padding: 2px;'><input style='margin-left: 10px;' name='name[" . $x["id"] . "]' value='" . escape_tags($x["displayname"]) . "' $disabled></td>"; | ||||||
| 		$o .= "<td style='padding: 2px;'><input style='margin-left: 10px; width: 150px;' name='uri[" . $x["id"] . "]' value='" . escape_tags($x["uri"]) . "' $disabled></td>"; | 		$o .= "<td style='padding: 2px;'><input style='margin-left: 10px; width: 150px;' name='uri[" . $x["id"] . "]' value='" . escape_tags($x["uri"]) . "' $disabled></td>"; | ||||||
| 		$o .= "<td style='padding: 2px;'>"; | 		$o .= "<td style='padding: 2px;'><a href='" . $a->get_baseurl() . "/dav/wdcal/" . $x["id"] . "/ics-export/'>Export</a>"; | ||||||
|  | 		if (!is_subclass_of($backend, "Sabre_CalDAV_Backend_Virtual") && $num_non_virtual > 1) $o .= " / <a href='" . $a->get_baseurl() . "/dav/wdcal/" . $x["id"] . "/ics-import/'>Import</a>"; | ||||||
|  | 		$o .= "</td>"; | ||||||
|  | 		$o .= "<td style='padding: 2px; padding-left: 50px;'>"; | ||||||
| 		if (!is_subclass_of($backend, "Sabre_CalDAV_Backend_Virtual") && $num_non_virtual > 1) $o .= "<a href='" . $a->get_baseurl() . "/dav/settings/?remove_cal=" . $x["id"] . "&t=" . get_form_security_token("del_cal") . "' class='delete_cal'>Delete</a>"; | 		if (!is_subclass_of($backend, "Sabre_CalDAV_Backend_Virtual") && $num_non_virtual > 1) $o .= "<a href='" . $a->get_baseurl() . "/dav/settings/?remove_cal=" . $x["id"] . "&t=" . get_form_security_token("del_cal") . "' class='delete_cal'>Delete</a>"; | ||||||
| 		$o .= "</td>\n"; | 		$o .= "</td>\n"; | ||||||
| 		$o .= "</tr>\n"; | 		$o .= "</tr>\n"; | ||||||
|  | @ -397,6 +482,7 @@ function wdcal_getSettingsPage(&$a) | ||||||
| 	$o .= "<td style='padding: 2px; text-align: center;'><input style='margin-left: 10px; width: 70px;' class='cal_color' name='color[new]' id='cal_color_new' value='#5858ff'></td>"; | 	$o .= "<td style='padding: 2px; text-align: center;'><input style='margin-left: 10px; width: 70px;' class='cal_color' name='color[new]' id='cal_color_new' value='#5858ff'></td>"; | ||||||
| 	$o .= "<td style='padding: 2px;'><input style='margin-left: 10px;' name='name[new]' value='Another calendar'></td>"; | 	$o .= "<td style='padding: 2px;'><input style='margin-left: 10px;' name='name[new]' value='Another calendar'></td>"; | ||||||
| 	$o .= "<td style='padding: 2px;'><input style='margin-left: 10px; width: 150px;' name='uri[new]' value='private-${private_max}'></td>"; | 	$o .= "<td style='padding: 2px;'><input style='margin-left: 10px; width: 150px;' name='uri[new]' value='private-${private_max}'></td>"; | ||||||
|  | 	$o .= "<td></td><td></td>"; | ||||||
| 	$o .= "</tr>\n"; | 	$o .= "</tr>\n"; | ||||||
| 
 | 
 | ||||||
| 	$o .= "</table>"; | 	$o .= "</table>"; | ||||||
|  |  | ||||||
							
								
								
									
										10
									
								
								dav/main.php
									
										
									
									
									
								
							
							
						
						
									
										10
									
								
								dav/main.php
									
										
									
									
									
								
							|  | @ -127,9 +127,7 @@ function dav_content() | ||||||
| 		return wdcal_getSettingsPage($a); | 		return wdcal_getSettingsPage($a); | ||||||
| 	} elseif ($a->argv[1] == "wdcal") { | 	} elseif ($a->argv[1] == "wdcal") { | ||||||
| 		if (isset($a->argv[2]) && strlen($a->argv[2]) > 0) { | 		if (isset($a->argv[2]) && strlen($a->argv[2]) > 0) { | ||||||
| 			if ($a->argv[2] == "ics") { | 			if ($a->argv[2] == "new") { | ||||||
| 				wdcal_print_user_ics(); |  | ||||||
| 			} elseif ($a->argv[2] == "new") { |  | ||||||
| 				$o = ""; | 				$o = ""; | ||||||
| 				if (isset($_REQUEST["save"])) { | 				if (isset($_REQUEST["save"])) { | ||||||
| 					check_form_security_token_redirectOnErr($a->get_baseurl() . "/dav/wdcal/", "caledit"); | 					check_form_security_token_redirectOnErr($a->get_baseurl() . "/dav/wdcal/", "caledit"); | ||||||
|  | @ -142,7 +140,11 @@ function dav_content() | ||||||
| 				return $o; | 				return $o; | ||||||
| 			} else { | 			} else { | ||||||
| 				$calendar_id = IntVal($a->argv[2]); | 				$calendar_id = IntVal($a->argv[2]); | ||||||
| 				if (isset($a->argv[3]) && $a->argv[3] > 0) { | 				if (isset($a->argv[3]) && $a->argv[3] == "ics-export") { | ||||||
|  | 					wdcal_print_user_ics($calendar_id); | ||||||
|  | 				} elseif (isset($a->argv[3]) && $a->argv[3] == "ics-import") { | ||||||
|  | 					return wdcal_import_user_ics($calendar_id); | ||||||
|  | 				} elseif (isset($a->argv[3]) && $a->argv[3] > 0) { | ||||||
| 					$recurr_uri = ""; // @TODO
 | 					$recurr_uri = ""; // @TODO
 | ||||||
| 					if (isset($a->argv[4]) && $a->argv[4] == "edit") { | 					if (isset($a->argv[4]) && $a->argv[4] == "edit") { | ||||||
| 						$o = ""; | 						$o = ""; | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue