Release changes for version 1.1.2
[davical.git] / htdocs / caldav.php
blobf6ddf918816b1599ad338ef754d43acc79929464
1 <?php
2 /**
3 * CalDAV Server - main program
5 * @package davical
6 * @subpackage caldav
7 * @author Andrew McMillan <andrew@mcmillan.net.nz>
8 * @copyright Catalyst .Net Ltd, Morphoss Ltd <http://www.morphoss.com/>
9 * @license http://gnu.org/copyleft/gpl.html GNU GPL v2 or later
11 if ( isset($_SERVER['PATH_INFO']) && preg_match( '{^(/favicon.ico|davical.css|(images|js|css)/.+)$}', $_SERVER['PATH_INFO'], $matches ) ) {
12 $filename = $_SERVER['DOCUMENT_ROOT'] . preg_replace('{(\.\.|\\\\)}', '', $matches[1]);
13 $fh = @fopen($matches[1],'r');
14 if ( ! $fh ) {
15 @header( sprintf("HTTP/1.1 %d %s", 404, 'Not found') );
17 else {
18 fpassthru($fh);
20 @ob_flush(); exit(0);
22 require_once('./always.php');
24 if ( isset($_SERVER['PATH_INFO']) && preg_match( '{^/\.well-known/(.+)$}', $_SERVER['PATH_INFO'], $matches ) ) {
25 require ('well-known.php');
26 @ob_flush(); exit(0);
28 elseif ( isset($_SERVER['PATH_INFO']) && $_SERVER['PATH_INFO'] == '/autodiscover/autodiscover.xml' ) {
29 require ('autodiscover-handler.php');
30 @ob_flush(); exit(0);
33 function logRequestHeaders() {
34 global $c;
36 /** Log the request headers */
37 $lines = apache_request_headers();
38 dbg_error_log( "LOG ", "***************** Request Header ****************" );
39 dbg_error_log( "LOG ", "%s %s", $_SERVER['REQUEST_METHOD'], $_SERVER['REQUEST_URI'] );
40 foreach( $lines AS $k => $v ) {
41 if ( $k != 'Authorization' || (isset($c->dbg['password']) && $c->dbg['password'] ) )
42 dbg_error_log( "LOG headers", "-->%s: %s", $k, $v );
43 else
44 dbg_error_log( "LOG headers", "-->%s: %s", $k, 'Delicious tasty password eaten by debugging monster!' );
46 dbg_error_log( "LOG ", "******************** Request ********************" );
48 // Log the request in all it's gory detail.
49 $lines = preg_split( '#[\r\n]+#', $c->raw_post );
50 foreach( $lines AS $v ) {
51 dbg_error_log( "LOG request", "-->%s", $v );
53 unset($lines);
56 if ( !isset($c->raw_post) ) $c->raw_post = file_get_contents( 'php://input');
57 if ( (isset($c->dbg['ALL']) && $c->dbg['ALL']) || (isset($c->dbg['request']) && $c->dbg['request']) )
58 logRequestHeaders();
61 require_once('HTTPAuthSession.php');
62 $session = new HTTPAuthSession();
64 function send_dav_header() {
65 global $c;
67 /**
68 * access-control is rfc3744, we do most of it, but no way to say that.
69 * calendar-schedule is another one we do most of, but the spec is not final yet either.
71 if ( isset($c->override_dav_header) ) {
72 $dav = $c->override_dav_header;
74 else {
75 $dav = '1, 2, 3, access-control, calendar-access, calendar-schedule, extended-mkcol, bind, addressbook';
76 if ( $c->enable_auto_schedule ) $dav .= ', calendar-auto-schedule';
77 if ( !isset($c->disable_caldav_proxy) || $c->disable_caldav_proxy == false) $dav .= ', calendar-proxy';
79 $dav = explode( "\n", wordwrap( $dav ) );
80 foreach( $dav AS $v ) {
81 header( 'DAV: '.trim($v, ', '), false);
84 send_dav_header(); // Avoid polluting global namespace
86 require_once('CalDAVRequest.php');
87 $request = new CalDAVRequest();
89 $allowed = implode( ', ', array_keys($request->supported_methods) );
90 // header( 'Allow: '.$allowed);
92 if ( ! ($request->IsPrincipal() || isset($request->collection) || $request->method == 'PUT' || $request->method == 'MKCALENDAR' || $request->method == 'MKCOL' ) ) {
93 if ( preg_match( '#^/principals/users(/.*/)$#', $request->path, $matches ) ) {
94 // Although this doesn't work with the iPhone, perhaps it will with iCal...
95 /** @todo integrate handling this URL into CalDAVRequest.php */
96 $redirect_url = ConstructURL('/caldav.php'.$matches[1]);
97 dbg_error_log( 'LOG WARNING', 'Redirecting %s for "%s" to "%s"', $request->method, $request->path, $redirect_url );
98 header('Location: '.$redirect_url );
99 @ob_flush(); exit(0);
102 param_to_global('add_member','.*');
103 $add_member = isset($add_member);
105 switch ( $request->method ) {
106 case 'OPTIONS': include_once('caldav-OPTIONS.php'); break;
107 case 'REPORT': include_once('caldav-REPORT.php'); break;
108 case 'PROPFIND': include('caldav-PROPFIND.php'); break;
109 case 'GET': include('caldav-GET.php'); break;
110 case 'HEAD': include('caldav-GET.php'); break;
111 case 'PROPPATCH': include('caldav-PROPPATCH.php'); break;
112 case 'POST':
113 if ( $request->content_type != 'text/vcard' && !$add_member ) {
114 include('caldav-POST.php');
115 break;
117 case 'PUT':
118 switch( $request->content_type ) {
119 case 'text/calendar':
120 include('caldav-PUT-vcalendar.php');
121 break;
122 case 'text/vcard':
123 case 'text/x-vcard':
124 include('caldav-PUT-vcard.php');
125 break;
126 default:
127 include('caldav-PUT-default.php');
128 break;
130 break;
131 case 'MKCALENDAR': include('caldav-MKCOL.php'); break;
132 case 'MKCOL': include('caldav-MKCOL.php'); break;
133 case 'DELETE': include('caldav-DELETE.php'); break;
134 case 'MOVE': include('caldav-MOVE.php'); break;
135 case 'ACL': include('caldav-ACL.php'); break;
136 case 'LOCK': include('caldav-LOCK.php'); break;
137 case 'UNLOCK': include('caldav-LOCK.php'); break;
138 case 'MKTICKET': include('caldav-MKTICKET.php'); break;
139 case 'DELTICKET': include('caldav-DELTICKET.php'); break;
140 case 'BIND': include('caldav-BIND.php'); break;
142 case 'TESTRRULE': include('test-RRULE-v2.php'); break;
144 default:
145 dbg_error_log( 'caldav', 'Unhandled request method >>%s<<', $request->method );
146 dbg_log_array( 'caldav', '_SERVER', $_SERVER, true );
147 dbg_error_log( 'caldav', 'RAW: %s', str_replace("\n", '',str_replace("\r", '', $request->raw_post)) );
150 $request->DoResponse( 400, translate('The application program does not understand that request.') );