X-Git-Url: https://code.citadel.org/?a=blobdiff_plain;f=webcit%2Fdav_main.c;fp=webcit%2Fdav_main.c;h=2b8ca189d2c91e6514f1e93731d1ef2fe2ad03f9;hb=977a00909d0de9ab3f96e6f5bdac18d91e30d79f;hp=0000000000000000000000000000000000000000;hpb=168342e0ff3187e28faa4be0f17f5a3b7d043e95;p=citadel.git diff --git a/webcit/dav_main.c b/webcit/dav_main.c new file mode 100644 index 000000000..2b8ca189d --- /dev/null +++ b/webcit/dav_main.c @@ -0,0 +1,353 @@ +/* + * Entry point for GroupDAV functions + * + * Copyright (c) 2005-2010 by the citadel.org team + * + * This program is open source software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA + */ + +#include "webcit.h" +#include "webserver.h" +#include "dav.h" + +extern HashList *HandlerHash; + +HashList *DavNamespaces = NULL; + +/* + * Output HTTP headers which are common to all requests. + * + * Please observe that we don't use the usual output_headers() + * and wDumpContent() functions in the GroupDAV subsystem, so we + * do our own header stuff here. + * + */ +void groupdav_common_headers(void) { + hprintf( + "Server: %s / %s\r\n" + "Connection: close\r\n", + PACKAGE_STRING, ChrPtr(WC->serv_info->serv_software) + ); +} + + + +/* + * string conversion function + */ +void euid_escapize(char *target, const char *source) { + int i, len; + int target_length = 0; + + strcpy(target, ""); + len = strlen(source); + for (i=0; iHdr->HR.dav_ifmatch); + if (len > 0) { + StrBufTrim(WCC->Hdr->HR.dav_ifmatch); + if (ChrPtr(WCC->Hdr->HR.dav_ifmatch)[0] == '\"') { + StrBufCutLeft(WCC->Hdr->HR.dav_ifmatch, 1); + len --; + for (i=0; iHdr->HR.dav_ifmatch)[i] == '\"') { + StrBufCutAt(WCC->Hdr->HR.dav_ifmatch, i, NULL); + len = StrLength(WCC->Hdr->HR.dav_ifmatch); + } + } + } + if (!strcmp(ChrPtr(WCC->Hdr->HR.dav_ifmatch), "*")) { + FlushStrBuf(WCC->Hdr->HR.dav_ifmatch); + } + } + + switch (WCC->Hdr->HR.eReqType) + { + /* + * The OPTIONS method is not required by GroupDAV. This is an + * experiment to determine what might be involved in supporting + * other variants of DAV in the future. + */ + case eOPTIONS: + groupdav_options(); + break; + + + /* + * The PROPFIND method is basically used to list all objects in a + * room, or to list all relevant rooms on the server. + */ + case ePROPFIND: + groupdav_propfind(); + break; + + /* + * The GET method is used for fetching individual items. + */ + case eGET: + groupdav_get(); + break; + + /* + * The PUT method is used to add or modify items. + */ + case ePUT: + groupdav_put(); + break; + + /* + * The DELETE method kills, maims, and destroys. + */ + case eDELETE: + groupdav_delete(); + break; + default: + + /* + * Couldn't find what we were looking for. Die in a car fire. + */ + hprintf("HTTP/1.1 501 Method not implemented\r\n"); + groupdav_common_headers(); + hprintf("Content-Type: text/plain\r\n"); + wc_printf("GroupDAV method \"%s\" is not implemented.\r\n", + ReqStrs[WCC->Hdr->HR.eReqType]); + end_burst(); + } +} + + +/* + * Output our host prefix for globally absolute URL's. + */ +void groupdav_identify_host(void) { + wc_printf("%s", ChrPtr(site_prefix)); +} + + +void tmplput_dav.hOSTNAME(StrBuf *Target, WCTemplputParams *TP) +{ + StrBufAppendPrintf(Target, "%s", ChrPtr(site_prefix)); +} + +/* + * Output our host prefix for globally absolute URL's. + */ +void groupdav_identify_hosthdr(void) { + hprintf("%s", ChrPtr(site_prefix)); +} + + +void Header_HandleIfMatch(StrBuf *Line, ParsedHttpHdrs *hdr) +{ + hdr->HR.dav_ifmatch = Line; +} + +void Header_HandleDepth(StrBuf *Line, ParsedHttpHdrs *hdr) +{ + if (!strcasecmp(ChrPtr(Line), "infinity")) { + hdr->HR.dav_depth = 32767; + } + else if (strcmp(ChrPtr(Line), "0") == 0) { + hdr->HR.dav_depth = 0; + } + else if (strcmp(ChrPtr(Line), "1") == 0) { + hdr->HR.dav_depth = 1; + } +} +int Conditional_DAV_DEPTH(StrBuf *Target, WCTemplputParams *TP) +{ + return WC->Hdr->HR.dav_depth == GetTemplateTokenNumber(Target, TP, 2, 0); +} + + +void RegisterDAVNamespace(const char * UrlString, + long UrlSLen, + const char *DisplayName, + long dslen, + WebcitHandlerFunc F, + WebcitRESTDispatchID RID, + long Flags) +{ + void *vHandler; + + /* first put it in... */ + WebcitAddUrlHandler(UrlString, UrlSLen, DisplayName, dslen, F, Flags|PARSE_REST_URL); + /* get it out again... */ + GetHash(HandlerHash, UrlString, UrlSLen, &vHandler); + ((WebcitHandler*)vHandler)->RID = RID; + /* and keep a copy of it, so we can compare it later */ + Put(DavNamespaces, UrlString, UrlSLen, vHandler, reference_free_handler); +} + +int Conditional_DAV_NS(StrBuf *Target, WCTemplputParams *TP) +{ + wcsession *WCC = WC; + void *vHandler; + const char *NS; + long NSLen; + + GetTemplateTokenString(NULL, TP, 2, &NS, &NSLen); + GetHash(HandlerHash, NS, NSLen, &vHandler); + return WCC->Hdr->HR.Handler == vHandler; +} + + +int Conditional_DAV_NSCURRENT(StrBuf *Target, WCTemplputParams *TP) +{ + wcsession *WCC = WC; + void *vHandler; + + vHandler = CTX; + return WCC->Hdr->HR.Handler == vHandler; +} + +void tmplput_DAV_NAMESPACE(StrBuf *Target, WCTemplputParams *TP) +{ + wcsession *WCC = WC; + + if (TP->Filter.ContextType == CTX_DAVNS) { + WebcitHandler *H; + H = (WebcitHandler*) CTX; + StrBufAppendTemplate(Target, TP, H->Name, 0); + } + else if (WCC->Hdr->HR.Handler != NULL) { + StrBufAppendTemplate(Target, TP, WCC->Hdr->HR.Handler->Name, 0); + } +} + +int GroupdavDispatchREST(RESTDispatchID WhichAction, int IgnoreFloor) +{ + wcsession *WCC = WC; + void *vDir; + + switch(WhichAction){ + case ExistsID: + GetHash(WCC->Directory, IKEY(WCC->ThisRoom->nRoomNameParts + 1), &vDir); + return locate_message_by_uid(ChrPtr((StrBuf*)vDir)) != -1; + /* TODO: remember euid */ + case PutID: + case DeleteID: + break; + + + } + return 0; +} + + +void +ServerStartModule_DAV +(void) +{ + + DavNamespaces = NewHash(1, NULL); + +} + +void +ServerShutdownModule_DAV +(void) +{ + DeleteHash(&DavNamespaces); +} + + + + +void +InitModule_GROUPDAV +(void) +{ +/* + WebcitAddUrlHandler(HKEY("groupdav"), "", 0, groupdav_main, XHTTP_COMMANDS|COOKIEUNNEEDED|FORCE_SESSIONCLOSE); + */ + RegisterDAVNamespace(HKEY("groupdav"), HKEY("GroupDAV"), + groupdav_main, GroupdavDispatchREST, + XHTTP_COMMANDS|COOKIEUNNEEDED|FORCE_SESSIONCLOSE); + + RegisterNamespace("DAV:HOSTNAME", 0, 0, tmplput_dav.hOSTNAME, NULL, CTX_NONE); + + RegisterConditional(HKEY("COND:DAV:NS"), 0, Conditional_DAV_NS, CTX_NONE); + + RegisterIterator("DAV:NS", 0, DavNamespaces, NULL, + NULL, NULL, CTX_DAVNS, CTX_NONE, IT_NOFLAG); + + + RegisterConditional(HKEY("COND:DAV:NSCURRENT"), 0, Conditional_DAV_NSCURRENT, CTX_DAVNS); + RegisterNamespace("DAV:NAMESPACE", 0, 1, tmplput_DAV_NAMESPACE, NULL, CTX_NONE); + + RegisterHeaderHandler(HKEY("IF-MATCH"), Header_HandleIfMatch); + RegisterHeaderHandler(HKEY("DEPTH"), Header_HandleDepth); + RegisterConditional(HKEY("COND:DAV:DEPTH"), 1, Conditional_DAV_DEPTH, CTX_NONE); + +}