Annotation of win32/apache22/include/mod_dav.h, revision 1.1

1.1     ! moko        1: /* Licensed to the Apache Software Foundation (ASF) under one or more
        !             2:  * contributor license agreements.  See the NOTICE file distributed with
        !             3:  * this work for additional information regarding copyright ownership.
        !             4:  * The ASF licenses this file to You under the Apache License, Version 2.0
        !             5:  * (the "License"); you may not use this file except in compliance with
        !             6:  * the License.  You may obtain a copy of the License at
        !             7:  *
        !             8:  *     http://www.apache.org/licenses/LICENSE-2.0
        !             9:  *
        !            10:  * Unless required by applicable law or agreed to in writing, software
        !            11:  * distributed under the License is distributed on an "AS IS" BASIS,
        !            12:  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
        !            13:  * See the License for the specific language governing permissions and
        !            14:  * limitations under the License.
        !            15:  */
        !            16: 
        !            17: /**
        !            18:  * @file  mod_dav.h 
        !            19:  * @brief DAV extension module for Apache 2.0.*
        !            20:  *
        !            21:  * @defgroup MOD_DAV mod_dav
        !            22:  * @ingroup APACHE_MODS
        !            23:  * @{
        !            24:  */
        !            25: 
        !            26: #ifndef _MOD_DAV_H_
        !            27: #define _MOD_DAV_H_
        !            28: 
        !            29: #include "apr_hooks.h"
        !            30: #include "apr_hash.h"
        !            31: #include "apr_dbm.h"
        !            32: #include "apr_tables.h"
        !            33: 
        !            34: #include "httpd.h"
        !            35: #include "util_filter.h"
        !            36: #include "util_xml.h"
        !            37: 
        !            38: #include <limits.h>     /* for INT_MAX */
        !            39: #include <time.h>       /* for time_t */
        !            40: 
        !            41: #ifdef __cplusplus
        !            42: extern "C" {
        !            43: #endif
        !            44: 
        !            45: 
        !            46: #define DAV_VERSION             AP_SERVER_BASEREVISION
        !            47: 
        !            48: #define DAV_XML_HEADER          "<?xml version=\"1.0\" encoding=\"utf-8\"?>"
        !            49: #define DAV_XML_CONTENT_TYPE    "text/xml; charset=\"utf-8\""
        !            50: 
        !            51: #define DAV_READ_BLOCKSIZE      2048    /* used for reading input blocks */
        !            52: 
        !            53: #define DAV_RESPONSE_BODY_1     "<!DOCTYPE HTML PUBLIC \"-//IETF//DTD HTML 2.0//EN\">\n<html><head>\n<title>"
        !            54: #define DAV_RESPONSE_BODY_2     "</title>\n</head><body>\n<h1>"
        !            55: #define DAV_RESPONSE_BODY_3     "</h1>\n<p>"
        !            56: #define DAV_RESPONSE_BODY_4     "</p>\n"
        !            57: #define DAV_RESPONSE_BODY_5     "</body></html>\n"
        !            58: 
        !            59: #define DAV_DO_COPY             0
        !            60: #define DAV_DO_MOVE             1
        !            61: 
        !            62: 
        !            63: #if 1
        !            64: #define DAV_DEBUG        1
        !            65: #define DEBUG_CR         "\n"
        !            66: #define DBG0(f)          ap_log_error(APLOG_MARK, \
        !            67:                                 APLOG_ERR, 0, NULL, (f))
        !            68: #define DBG1(f,a1)       ap_log_error(APLOG_MARK, \
        !            69:                                 APLOG_ERR, 0, NULL, f, a1)
        !            70: #define DBG2(f,a1,a2)    ap_log_error(APLOG_MARK, \
        !            71:                                 APLOG_ERR, 0, NULL, f, a1, a2)
        !            72: #define DBG3(f,a1,a2,a3) ap_log_error(APLOG_MARK, \
        !            73:                                 APLOG_ERR, 0, NULL, f, a1, a2, a3)
        !            74: #else
        !            75: #undef DAV_DEBUG
        !            76: #define DEBUG_CR        ""
        !            77: #endif
        !            78: 
        !            79: #define DAV_INFINITY    INT_MAX    /* for the Depth: header */
        !            80: 
        !            81: /* Create a set of DAV_DECLARE(type), DAV_DECLARE_NONSTD(type) and 
        !            82:  * DAV_DECLARE_DATA with appropriate export and import tags for the platform
        !            83:  */
        !            84: #if !defined(WIN32)
        !            85: #define DAV_DECLARE(type)            type
        !            86: #define DAV_DECLARE_NONSTD(type)     type
        !            87: #define DAV_DECLARE_DATA
        !            88: #elif defined(DAV_DECLARE_STATIC)
        !            89: #define DAV_DECLARE(type)            type __stdcall
        !            90: #define DAV_DECLARE_NONSTD(type)     type
        !            91: #define DAV_DECLARE_DATA
        !            92: #elif defined(DAV_DECLARE_EXPORT)
        !            93: #define DAV_DECLARE(type)            __declspec(dllexport) type __stdcall
        !            94: #define DAV_DECLARE_NONSTD(type)     __declspec(dllexport) type
        !            95: #define DAV_DECLARE_DATA             __declspec(dllexport)
        !            96: #else
        !            97: #define DAV_DECLARE(type)            __declspec(dllimport) type __stdcall
        !            98: #define DAV_DECLARE_NONSTD(type)     __declspec(dllimport) type
        !            99: #define DAV_DECLARE_DATA             __declspec(dllimport)
        !           100: #endif
        !           101: 
        !           102: /* --------------------------------------------------------------------
        !           103: **
        !           104: ** ERROR MANAGEMENT
        !           105: */
        !           106: 
        !           107: /*
        !           108: ** dav_error structure.
        !           109: **
        !           110: ** In most cases, mod_dav uses a pointer to a dav_error structure. If the
        !           111: ** pointer is NULL, then no error has occurred.
        !           112: **
        !           113: ** In certain cases, a dav_error structure is directly used. In these cases,
        !           114: ** a status value of 0 means that an error has not occurred.
        !           115: **
        !           116: ** Note: this implies that status != 0 whenever an error occurs.
        !           117: **
        !           118: ** The desc field is optional (it may be NULL). When NULL, it typically
        !           119: ** implies that Apache has a proper description for the specified status.
        !           120: */
        !           121: typedef struct dav_error {
        !           122:     int status;                 /* suggested HTTP status (0 for no error) */
        !           123:     int error_id;               /* DAV-specific error ID */
        !           124:     const char *desc;           /* DAV:responsedescription and error log */
        !           125: 
        !           126:     int save_errno;             /* copy of errno causing the error */
        !           127: 
        !           128:     const char *namespace;      /* [optional] namespace of error */
        !           129:     const char *tagname;        /* name of error-tag */
        !           130: 
        !           131:     struct dav_error *prev;     /* previous error (in stack) */
        !           132: 
        !           133: } dav_error;
        !           134: 
        !           135: /*
        !           136: ** Create a new error structure. save_errno will be filled with the current
        !           137: ** errno value.
        !           138: */
        !           139: DAV_DECLARE(dav_error*) dav_new_error(apr_pool_t *p, int status, 
        !           140:                                       int error_id, const char *desc);
        !           141: 
        !           142: 
        !           143: /*
        !           144: ** Create a new error structure with tagname and (optional) namespace;
        !           145: ** namespace may be NULL, which means "DAV:". save_errno will be
        !           146: ** filled with the current errno value.
        !           147: */
        !           148: DAV_DECLARE(dav_error*) dav_new_error_tag(apr_pool_t *p, int status, 
        !           149:                                           int error_id, const char *desc,
        !           150:                                           const char *namespace,
        !           151:                                           const char *tagname);
        !           152: 
        !           153: 
        !           154: /*
        !           155: ** Push a new error description onto the stack of errors.
        !           156: **
        !           157: ** This function is used to provide an additional description to an existing
        !           158: ** error.
        !           159: **
        !           160: ** <status> should contain the caller's view of what the current status is,
        !           161: ** given the underlying error. If it doesn't have a better idea, then the
        !           162: ** caller should pass prev->status.
        !           163: **
        !           164: ** <error_id> can specify a new error_id since the topmost description has
        !           165: ** changed.
        !           166: */
        !           167: DAV_DECLARE(dav_error*) dav_push_error(apr_pool_t *p, int status, int error_id,
        !           168:                                        const char *desc, dav_error *prev);
        !           169: 
        !           170: 
        !           171: /* error ID values... */
        !           172: 
        !           173: /* IF: header errors */
        !           174: #define DAV_ERR_IF_PARSE                100    /* general parsing error */
        !           175: #define DAV_ERR_IF_MULTIPLE_NOT         101    /* multiple "Not" found */
        !           176: #define DAV_ERR_IF_UNK_CHAR             102    /* unknown char in header */
        !           177: #define DAV_ERR_IF_ABSENT               103    /* no locktokens given */
        !           178: #define DAV_ERR_IF_TAGGED               104    /* in parsing tagged-list */
        !           179: #define DAV_ERR_IF_UNCLOSED_PAREN       105    /* in no-tagged-list */
        !           180: 
        !           181: /* Prop DB errors */
        !           182: #define DAV_ERR_PROP_BAD_MAJOR          200    /* major version was wrong */
        !           183: #define DAV_ERR_PROP_READONLY           201    /* prop is read-only */
        !           184: #define DAV_ERR_PROP_NO_DATABASE        202    /* writable db not avail */
        !           185: #define DAV_ERR_PROP_NOT_FOUND          203    /* prop not found */
        !           186: #define DAV_ERR_PROP_BAD_LOCKDB         204    /* could not open lockdb */
        !           187: #define DAV_ERR_PROP_OPENING            205    /* problem opening propdb */
        !           188: #define DAV_ERR_PROP_EXEC               206    /* problem exec'ing patch */
        !           189: 
        !           190: /* Predefined DB errors */
        !           191: /* ### any to define?? */
        !           192: 
        !           193: /* Predefined locking system errors */
        !           194: #define DAV_ERR_LOCK_OPENDB             400    /* could not open lockdb */
        !           195: #define DAV_ERR_LOCK_NO_DB              401    /* no database defined */
        !           196: #define DAV_ERR_LOCK_CORRUPT_DB         402    /* DB is corrupt */
        !           197: #define DAV_ERR_LOCK_UNK_STATE_TOKEN    403    /* unknown State-token */
        !           198: #define DAV_ERR_LOCK_PARSE_TOKEN        404    /* bad opaquelocktoken */
        !           199: #define DAV_ERR_LOCK_SAVE_LOCK          405    /* err saving locks */
        !           200: 
        !           201: /*
        !           202: ** Some comments on Error ID values:
        !           203: **
        !           204: ** The numbers do not necessarily need to be unique. Uniqueness simply means
        !           205: ** that two errors that have not been predefined above can be distinguished
        !           206: ** from each other. At the moment, mod_dav does not use this distinguishing
        !           207: ** feature, but it could be used in the future to collapse <response> elements
        !           208: ** into groups based on the error ID (and associated responsedescription).
        !           209: **
        !           210: ** If a compute_desc is provided, then the error ID should be unique within
        !           211: ** the context of the compute_desc function (so the function can figure out
        !           212: ** what to filled into the desc).
        !           213: **
        !           214: ** Basically, subsystems can ignore defining new error ID values if they want
        !           215: ** to. The subsystems *do* need to return the predefined errors when
        !           216: ** appropriate, so that mod_dav can figure out what to do. Subsystems can
        !           217: ** simply leave the error ID field unfilled (zero) if there isn't an error
        !           218: ** that must be placed there.
        !           219: */
        !           220: 
        !           221: 
        !           222: /* --------------------------------------------------------------------
        !           223: **
        !           224: ** HOOK STRUCTURES
        !           225: **
        !           226: ** These are here for forward-declaration purposes. For more info, see
        !           227: ** the section title "HOOK HANDLING" for more information, plus each
        !           228: ** structure definition.
        !           229: */
        !           230: 
        !           231: /* forward-declare this structure */
        !           232: typedef struct dav_hooks_propdb dav_hooks_propdb;
        !           233: typedef struct dav_hooks_locks dav_hooks_locks;
        !           234: typedef struct dav_hooks_vsn dav_hooks_vsn;
        !           235: typedef struct dav_hooks_repository dav_hooks_repository;
        !           236: typedef struct dav_hooks_liveprop dav_hooks_liveprop;
        !           237: typedef struct dav_hooks_binding dav_hooks_binding;
        !           238: typedef struct dav_hooks_search dav_hooks_search;
        !           239: 
        !           240: /* ### deprecated name */
        !           241: typedef dav_hooks_propdb dav_hooks_db;
        !           242: 
        !           243: 
        !           244: /* --------------------------------------------------------------------
        !           245: **
        !           246: ** RESOURCE HANDLING
        !           247: */
        !           248: 
        !           249: /*
        !           250: ** Resource Types:
        !           251: ** The base protocol defines only file and collection resources.
        !           252: ** The versioning protocol defines several additional resource types
        !           253: ** to represent artifacts of a version control system.
        !           254: **
        !           255: ** This enumeration identifies the type of URL used to identify the
        !           256: ** resource. Since the same resource may have more than one type of
        !           257: ** URL which can identify it, dav_resource_type cannot be used
        !           258: ** alone to determine the type of the resource; attributes of the
        !           259: ** dav_resource object must also be consulted.
        !           260: */
        !           261: typedef enum {
        !           262:     DAV_RESOURCE_TYPE_UNKNOWN,
        !           263: 
        !           264:     DAV_RESOURCE_TYPE_REGULAR,          /* file or collection; could be
        !           265:                                          * unversioned, or version selector,
        !           266:                                          * or baseline selector */
        !           267: 
        !           268:     DAV_RESOURCE_TYPE_VERSION,          /* version or baseline URL */
        !           269: 
        !           270:     DAV_RESOURCE_TYPE_HISTORY,          /* version or baseline history URL */
        !           271: 
        !           272:     DAV_RESOURCE_TYPE_WORKING,          /* working resource URL */
        !           273: 
        !           274:     DAV_RESOURCE_TYPE_WORKSPACE,        /* workspace URL */
        !           275: 
        !           276:     DAV_RESOURCE_TYPE_ACTIVITY,         /* activity URL */
        !           277: 
        !           278:     DAV_RESOURCE_TYPE_PRIVATE           /* repository-private type */
        !           279: 
        !           280: } dav_resource_type;
        !           281: 
        !           282: /*
        !           283: ** Opaque, repository-specific information for a resource.
        !           284: */
        !           285: typedef struct dav_resource_private dav_resource_private;
        !           286: 
        !           287: /*
        !           288: ** Resource descriptor, generated by a repository provider.
        !           289: **
        !           290: ** Note: the lock-null state is not explicitly represented here,
        !           291: ** since it may be expensive to compute. Use dav_get_resource_state()
        !           292: ** to determine whether a non-existent resource is a lock-null resource.
        !           293: **
        !           294: ** A quick explanation of how the flags can apply to different resources:
        !           295: **
        !           296: ** unversioned file or collection:
        !           297: **     type       = DAV_RESOURCE_TYPE_REGULAR
        !           298: **     exists     = ? (1 if exists)
        !           299: **     collection = ? (1 if collection)
        !           300: **     versioned  = 0
        !           301: **     baselined  = 0
        !           302: **     working    = 0
        !           303: **
        !           304: ** version-controlled resource or configuration:
        !           305: **     type       = DAV_RESOURCE_TYPE_REGULAR
        !           306: **     exists     = 1
        !           307: **     collection = ? (1 if collection)
        !           308: **     versioned  = 1
        !           309: **     baselined  = ? (1 if configuration)
        !           310: **     working    = ? (1 if checked out)
        !           311: **
        !           312: ** version/baseline history:
        !           313: **     type       = DAV_RESOURCE_TYPE_HISTORY
        !           314: **     exists     = 1
        !           315: **     collection = 0
        !           316: **     versioned  = 0
        !           317: **     baselined  = 0
        !           318: **     working    = 0
        !           319: **
        !           320: ** version/baseline:
        !           321: **     type       = DAV_RESOURCE_TYPE_VERSION
        !           322: **     exists     = 1
        !           323: **     collection = ? (1 if collection)
        !           324: **     versioned  = 1
        !           325: **     baselined  = ? (1 if baseline)
        !           326: **     working    = 0
        !           327: **
        !           328: ** working resource:
        !           329: **     type       = DAV_RESOURCE_TYPE_WORKING
        !           330: **     exists     = 1
        !           331: **     collection = ? (1 if collection)
        !           332: **     versioned  = 1
        !           333: **     baselined  = 0
        !           334: **     working    = 1
        !           335: **
        !           336: ** workspace:
        !           337: **     type       = DAV_RESOURCE_TYPE_WORKSPACE
        !           338: **     exists     = ? (1 if exists)
        !           339: **     collection = 1
        !           340: **     versioned  = ? (1 if version-controlled)
        !           341: **     baselined  = ? (1 if baseline-controlled)
        !           342: **     working    = ? (1 if checked out)
        !           343: **
        !           344: ** activity:
        !           345: **     type       = DAV_RESOURCE_TYPE_ACTIVITY
        !           346: **     exists     = ? (1 if exists)
        !           347: **     collection = 0
        !           348: **     versioned  = 0
        !           349: **     baselined  = 0
        !           350: **     working    = 0
        !           351: */
        !           352: typedef struct dav_resource {
        !           353:     dav_resource_type type;
        !           354: 
        !           355:     int exists;         /* 0 => null resource */
        !           356: 
        !           357:     int collection;     /* 0 => file; can be 1 for
        !           358:                          * REGULAR, VERSION, and WORKING resources,
        !           359:                          * and is always 1 for WORKSPACE */
        !           360: 
        !           361:     int versioned;      /* 0 => unversioned; can be 1 for
        !           362:                          * REGULAR and WORKSPACE resources,
        !           363:                          * and is always 1 for VERSION and WORKING */
        !           364: 
        !           365:     int baselined;      /* 0 => not baselined; can be 1 for
        !           366:                          * REGULAR, VERSION, and WORKSPACE resources;
        !           367:                          * versioned == 1 when baselined == 1 */
        !           368: 
        !           369:     int working;        /* 0 => not checked out; can be 1 for
        !           370:                          * REGULAR and WORKSPACE resources,
        !           371:                          * and is always 1 for WORKING */
        !           372: 
        !           373:     const char *uri;    /* the URI for this resource */
        !           374: 
        !           375:     dav_resource_private *info;         /* the provider's private info */
        !           376: 
        !           377:     const dav_hooks_repository *hooks;  /* hooks used for this resource */
        !           378: 
        !           379:     /* When allocating items related specifically to this resource, the
        !           380:        following pool should be used. Its lifetime will be at least as
        !           381:        long as the dav_resource structure. */
        !           382:     apr_pool_t *pool;
        !           383: 
        !           384: } dav_resource;
        !           385: 
        !           386: /*
        !           387: ** Lock token type. Lock providers define the details of a lock token.
        !           388: ** However, all providers are expected to at least be able to parse
        !           389: ** the "opaquelocktoken" scheme, which is represented by a uuid_t.
        !           390: */
        !           391: typedef struct dav_locktoken dav_locktoken;
        !           392: 
        !           393: 
        !           394: /* --------------------------------------------------------------------
        !           395: **
        !           396: ** BUFFER HANDLING
        !           397: **
        !           398: ** These buffers are used as a lightweight buffer reuse mechanism. Apache
        !           399: ** provides sub-pool creation and destruction to much the same effect, but
        !           400: ** the sub-pools are a bit more general and heavyweight than these buffers.
        !           401: */
        !           402: 
        !           403: /* buffer for reuse; can grow to accomodate needed size */
        !           404: typedef struct
        !           405: {
        !           406:     apr_size_t alloc_len;       /* how much has been allocated */
        !           407:     apr_size_t cur_len;         /* how much is currently being used */
        !           408:     char *buf;                  /* buffer contents */
        !           409: } dav_buffer;
        !           410: #define DAV_BUFFER_MINSIZE      256    /* minimum size for buffer */
        !           411: #define DAV_BUFFER_PAD          64     /* amount of pad when growing */
        !           412: 
        !           413: /* set the cur_len to the given size and ensure space is available */
        !           414: DAV_DECLARE(void) dav_set_bufsize(apr_pool_t *p, dav_buffer *pbuf, 
        !           415:                                   apr_size_t size);
        !           416: 
        !           417: /* initialize a buffer and copy the specified (null-term'd) string into it */
        !           418: DAV_DECLARE(void) dav_buffer_init(apr_pool_t *p, dav_buffer *pbuf, 
        !           419:                                   const char *str);
        !           420: 
        !           421: /* check that the buffer can accomodate <extra_needed> more bytes */
        !           422: DAV_DECLARE(void) dav_check_bufsize(apr_pool_t *p, dav_buffer *pbuf, 
        !           423:                                     apr_size_t extra_needed);
        !           424: 
        !           425: /* append a string to the end of the buffer, adjust length */
        !           426: DAV_DECLARE(void) dav_buffer_append(apr_pool_t *p, dav_buffer *pbuf, 
        !           427:                                     const char *str);
        !           428: 
        !           429: /* place a string on the end of the buffer, do NOT adjust length */
        !           430: DAV_DECLARE(void) dav_buffer_place(apr_pool_t *p, dav_buffer *pbuf, 
        !           431:                                    const char *str);
        !           432: 
        !           433: /* place some memory on the end of a buffer; do NOT adjust length */
        !           434: DAV_DECLARE(void) dav_buffer_place_mem(apr_pool_t *p, dav_buffer *pbuf, 
        !           435:                                        const void *mem, apr_size_t amt, 
        !           436:                                        apr_size_t pad);
        !           437: 
        !           438: 
        !           439: /* --------------------------------------------------------------------
        !           440: **
        !           441: ** HANDY UTILITIES
        !           442: */
        !           443: 
        !           444: /* contains results from one of the getprop functions */
        !           445: typedef struct
        !           446: {
        !           447:     apr_text * propstats;       /* <propstat> element text */
        !           448:     apr_text * xmlns;           /* namespace decls for <response> elem */
        !           449: } dav_get_props_result;
        !           450: 
        !           451: /* holds the contents of a <response> element */
        !           452: typedef struct dav_response
        !           453: {
        !           454:     const char *href;           /* always */
        !           455:     const char *desc;           /* optional description at <response> level */
        !           456: 
        !           457:     /* use status if propresult.propstats is NULL. */
        !           458:     dav_get_props_result propresult;
        !           459: 
        !           460:     int status;
        !           461: 
        !           462:     struct dav_response *next;
        !           463: } dav_response;
        !           464: 
        !           465: typedef struct
        !           466: {
        !           467:     request_rec *rnew;          /* new subrequest */
        !           468:     dav_error err;              /* potential error response */
        !           469: } dav_lookup_result;
        !           470: 
        !           471: 
        !           472: DAV_DECLARE(dav_lookup_result) dav_lookup_uri(const char *uri, request_rec *r,
        !           473:                                               int must_be_absolute);
        !           474: 
        !           475: /* defines type of property info a provider is to return */
        !           476: typedef enum {
        !           477:     DAV_PROP_INSERT_NOTDEF,     /* property is defined by this provider,
        !           478:                                    but nothing was inserted because the
        !           479:                                    (live) property is not defined for this
        !           480:                                    resource (it may be present as a dead
        !           481:                                    property). */
        !           482:     DAV_PROP_INSERT_NOTSUPP,    /* property is recognized by this provider,
        !           483:                                    but it is not supported, and cannot be
        !           484:                                    treated as a dead property */
        !           485:     DAV_PROP_INSERT_NAME,       /* a property name (empty elem) was
        !           486:                                    inserted into the text block */
        !           487:     DAV_PROP_INSERT_VALUE,      /* a property name/value pair was inserted
        !           488:                                    into the text block */
        !           489:     DAV_PROP_INSERT_SUPPORTED   /* a supported live property was added to
        !           490:                                    the text block as a
        !           491:                                    <DAV:supported-live-property> element */
        !           492: } dav_prop_insert;
        !           493: 
        !           494: /* ### this stuff is private to dav/fs/repos.c; move it... */
        !           495: /* format a time string (buf must be at least DAV_TIMEBUF_SIZE chars) */
        !           496: #define DAV_STYLE_ISO8601       1
        !           497: #define DAV_STYLE_RFC822        2
        !           498: #define DAV_TIMEBUF_SIZE        30
        !           499: 
        !           500: DAV_DECLARE(int) dav_get_depth(request_rec *r, int def_depth);
        !           501: 
        !           502: DAV_DECLARE(int) dav_validate_root(const apr_xml_doc *doc,
        !           503:                                    const char *tagname);
        !           504: DAV_DECLARE(apr_xml_elem *) dav_find_child(const apr_xml_elem *elem,
        !           505:                                            const char *tagname);
        !           506: 
        !           507: /* gather up all the CDATA into a single string */
        !           508: DAV_DECLARE(const char *) dav_xml_get_cdata(const apr_xml_elem *elem, apr_pool_t *pool,
        !           509:                               int strip_white);
        !           510: 
        !           511: /*
        !           512: ** XML namespace handling
        !           513: **
        !           514: ** This structure tracks namespace declarations (xmlns:prefix="URI").
        !           515: ** It maintains a one-to-many relationship of URIs-to-prefixes. In other
        !           516: ** words, one URI may be defined by many prefixes, but any specific
        !           517: ** prefix will specify only one URI.
        !           518: **
        !           519: ** Prefixes using the "g###" pattern can be generated automatically if
        !           520: ** the caller does not have specific prefix requirements.
        !           521: */
        !           522: typedef struct {
        !           523:     apr_pool_t *pool;
        !           524:     apr_hash_t *uri_prefix;     /* map URIs to an available prefix */
        !           525:     apr_hash_t *prefix_uri;     /* map all prefixes to their URIs */
        !           526:     int count;                  /* counter for "g###" prefixes */
        !           527: } dav_xmlns_info;
        !           528: 
        !           529: /* create an empty dav_xmlns_info structure */
        !           530: DAV_DECLARE(dav_xmlns_info *) dav_xmlns_create(apr_pool_t *pool);
        !           531: 
        !           532: /* add a specific prefix/URI pair. the prefix/uri should have a lifetime
        !           533:    at least that of xmlns->pool */
        !           534: DAV_DECLARE(void) dav_xmlns_add(dav_xmlns_info *xi,
        !           535:                                 const char *prefix, const char *uri);
        !           536: 
        !           537: /* add a URI (if not present); any prefix is acceptable and is returned.
        !           538:    the uri should have a lifetime at least that xmlns->pool */
        !           539: DAV_DECLARE(const char *) dav_xmlns_add_uri(dav_xmlns_info *xi,
        !           540:                                             const char *uri);
        !           541: 
        !           542: /* return the URI for a specified prefix (or NULL if the prefix is unknown) */
        !           543: DAV_DECLARE(const char *) dav_xmlns_get_uri(dav_xmlns_info *xi,
        !           544:                                             const char *prefix);
        !           545: 
        !           546: /* return an available prefix for a specified URI (or NULL if the URI
        !           547:    is unknown) */
        !           548: DAV_DECLARE(const char *) dav_xmlns_get_prefix(dav_xmlns_info *xi,
        !           549:                                                const char *uri);
        !           550: 
        !           551: /* generate xmlns declarations (appending into the given text) */
        !           552: DAV_DECLARE(void) dav_xmlns_generate(dav_xmlns_info *xi,
        !           553:                                      apr_text_header *phdr);
        !           554: 
        !           555: /* --------------------------------------------------------------------
        !           556: **
        !           557: ** DAV PLUGINS
        !           558: */
        !           559: 
        !           560: /* ### docco ... */
        !           561: 
        !           562: /*
        !           563: ** dav_provider
        !           564: **
        !           565: ** This structure wraps up all of the hooks that a mod_dav provider can
        !           566: ** supply. The provider MUST supply <repos> and <propdb>. The rest are
        !           567: ** optional and should contain NULL if that feature is not supplied.
        !           568: **
        !           569: ** Note that a provider cannot pick and choose portions from various
        !           570: ** underlying implementations (which was theoretically possible in
        !           571: ** mod_dav 1.0). There are too many dependencies between a dav_resource
        !           572: ** (defined by <repos>) and the other functionality.
        !           573: **
        !           574: ** Live properties are not part of the dav_provider structure because they
        !           575: ** are handled through the APR_HOOK interface (to allow for multiple liveprop
        !           576: ** providers). The core always provides some properties, and then a given
        !           577: ** provider will add more properties.
        !           578: **
        !           579: ** Some providers may need to associate a context with the dav_provider
        !           580: ** structure -- the ctx field is available for storing this context. Just
        !           581: ** leave it NULL if it isn't required.
        !           582: */
        !           583: typedef struct {
        !           584:     const dav_hooks_repository *repos;
        !           585:     const dav_hooks_propdb *propdb;
        !           586:     const dav_hooks_locks *locks;
        !           587:     const dav_hooks_vsn *vsn;
        !           588:     const dav_hooks_binding *binding;
        !           589:     const dav_hooks_search *search;
        !           590: 
        !           591:     void *ctx;
        !           592: } dav_provider;
        !           593: 
        !           594: /*
        !           595: ** gather_propsets: gather all live property propset-URIs
        !           596: **
        !           597: ** The hook implementor should push one or more URIs into the specified
        !           598: ** array. These URIs are returned in the DAV: header to let clients know
        !           599: ** what sets of live properties are supported by the installation. mod_dav
        !           600: ** will place open/close angle brackets around each value (much like
        !           601: ** a Coded-URL); quotes and brackets should not be in the value.
        !           602: **
        !           603: ** Example:    http://apache.org/dav/props/
        !           604: **
        !           605: ** (of course, use your own domain to ensure a unique value)
        !           606: */
        !           607: APR_DECLARE_EXTERNAL_HOOK(dav, DAV, void, gather_propsets, 
        !           608:                          (apr_array_header_t *uris))
        !           609: 
        !           610: /*
        !           611: ** find_liveprop: find a live property, returning a non-zero, unique,
        !           612: **                opaque identifier.
        !           613: **
        !           614: ** If the hook implementor determines the specified URI/name refers to
        !           615: ** one of its properties, then it should fill in HOOKS and return a
        !           616: ** non-zero value. The returned value is the "property ID" and will
        !           617: ** be passed to the various liveprop hook functions.
        !           618: **
        !           619: ** Return 0 if the property is not defined by the hook implementor.
        !           620: */
        !           621: APR_DECLARE_EXTERNAL_HOOK(dav, DAV, int, find_liveprop,
        !           622:                          (const dav_resource *resource,
        !           623:                           const char *ns_uri, const char *name,
        !           624:                           const dav_hooks_liveprop **hooks))
        !           625: 
        !           626: /*
        !           627: ** insert_all_liveprops: insert all (known) live property names/values.
        !           628: **
        !           629: ** The hook implementor should append XML text to PHDR, containing liveprop
        !           630: ** names. If INSVALUE is true, then the property values should also be
        !           631: ** inserted into the output XML stream.
        !           632: **
        !           633: ** The liveprop provider should insert *all* known and *defined* live
        !           634: ** properties on the specified resource. If a particular liveprop is
        !           635: ** not defined for this resource, then it should not be inserted.
        !           636: */
        !           637: APR_DECLARE_EXTERNAL_HOOK(dav, DAV, void, insert_all_liveprops, 
        !           638:                          (request_rec *r, const dav_resource *resource,
        !           639:                           dav_prop_insert what, apr_text_header *phdr))
        !           640: 
        !           641: DAV_DECLARE(const dav_hooks_locks *) dav_get_lock_hooks(request_rec *r);
        !           642: DAV_DECLARE(const dav_hooks_propdb *) dav_get_propdb_hooks(request_rec *r);
        !           643: DAV_DECLARE(const dav_hooks_vsn *) dav_get_vsn_hooks(request_rec *r);
        !           644: DAV_DECLARE(const dav_hooks_binding *) dav_get_binding_hooks(request_rec *r);
        !           645: DAV_DECLARE(const dav_hooks_search *) dav_get_search_hooks(request_rec *r);
        !           646: 
        !           647: DAV_DECLARE(void) dav_register_provider(apr_pool_t *p, const char *name,
        !           648:                                         const dav_provider *hooks);
        !           649: DAV_DECLARE(const dav_provider *) dav_lookup_provider(const char *name);
        !           650: 
        !           651: 
        !           652: /* ### deprecated */
        !           653: #define DAV_GET_HOOKS_PROPDB(r)         dav_get_propdb_hooks(r)
        !           654: #define DAV_GET_HOOKS_LOCKS(r)          dav_get_lock_hooks(r)
        !           655: #define DAV_GET_HOOKS_VSN(r)            dav_get_vsn_hooks(r)
        !           656: #define DAV_GET_HOOKS_BINDING(r)        dav_get_binding_hooks(r)
        !           657: #define DAV_GET_HOOKS_SEARCH(r)         dav_get_search_hooks(r)
        !           658: 
        !           659: 
        !           660: /* --------------------------------------------------------------------
        !           661: **
        !           662: ** IF HEADER PROCESSING
        !           663: **
        !           664: ** Here is the definition of the If: header from RFC 2518, S9.4:
        !           665: **
        !           666: **    If = "If" ":" (1*No-tag-list | 1*Tagged-list)
        !           667: **    No-tag-list = List
        !           668: **    Tagged-list = Resource 1*List
        !           669: **    Resource = Coded-URL
        !           670: **    List = "(" 1*(["Not"](State-token | "[" entity-tag "]")) ")"
        !           671: **    State-token = Coded-URL
        !           672: **    Coded-URL = "<" absoluteURI ">"        ; absoluteURI from RFC 2616
        !           673: **
        !           674: ** List corresponds to dav_if_state_list. No-tag-list corresponds to
        !           675: ** dav_if_header with uri==NULL. Tagged-list corresponds to a sequence of
        !           676: ** dav_if_header structures with (duplicate) uri==Resource -- one
        !           677: ** dav_if_header per state_list. A second Tagged-list will start a new
        !           678: ** sequence of dav_if_header structures with the new URI.
        !           679: **
        !           680: ** A summary of the semantics, mapped into our structures:
        !           681: **    - Chained dav_if_headers: OR
        !           682: **    - Chained dav_if_state_lists: AND
        !           683: **    - NULL uri matches all resources
        !           684: */
        !           685: 
        !           686: typedef enum
        !           687: {
        !           688:     dav_if_etag,
        !           689:     dav_if_opaquelock,
        !           690:     dav_if_unknown /* the "unknown" state type; always matches false. */   
        !           691: } dav_if_state_type;
        !           692: 
        !           693: typedef struct dav_if_state_list
        !           694: {
        !           695:     dav_if_state_type type;
        !           696: 
        !           697:     int condition;
        !           698: #define DAV_IF_COND_NORMAL      0
        !           699: #define DAV_IF_COND_NOT         1    /* "Not" was applied */
        !           700: 
        !           701:     const char *etag;
        !           702:     dav_locktoken *locktoken;
        !           703: 
        !           704:     struct dav_if_state_list *next;
        !           705: } dav_if_state_list;
        !           706: 
        !           707: typedef struct dav_if_header
        !           708: {
        !           709:     const char *uri;
        !           710:     apr_size_t uri_len;
        !           711:     struct dav_if_state_list *state;
        !           712:     struct dav_if_header *next;
        !           713: 
        !           714:     int dummy_header;   /* used internally by the lock/etag validation */
        !           715: } dav_if_header;
        !           716: 
        !           717: typedef struct dav_locktoken_list 
        !           718: {
        !           719:     dav_locktoken *locktoken;
        !           720:     struct dav_locktoken_list *next;
        !           721: } dav_locktoken_list;
        !           722: 
        !           723: DAV_DECLARE(dav_error *) dav_get_locktoken_list(request_rec *r,
        !           724:                                                 dav_locktoken_list **ltl);
        !           725: 
        !           726: 
        !           727: /* --------------------------------------------------------------------
        !           728: **
        !           729: ** LIVE PROPERTY HANDLING
        !           730: */
        !           731: 
        !           732: /* opaque type for PROPPATCH rollback information */
        !           733: typedef struct dav_liveprop_rollback dav_liveprop_rollback;
        !           734: 
        !           735: struct dav_hooks_liveprop
        !           736: {
        !           737:     /*
        !           738:     ** Insert property information into a text block. The property to
        !           739:     ** insert is identified by the propid value. The information to insert
        !           740:     ** is identified by the "what" argument, as follows:
        !           741:     **   DAV_PROP_INSERT_NAME
        !           742:     **      property name, as an empty XML element
        !           743:     **   DAV_PROP_INSERT_VALUE
        !           744:     **      property name/value, as an XML element
        !           745:     **   DAV_PROP_INSERT_SUPPORTED
        !           746:     **      if the property is defined on the resource, then
        !           747:     **      a DAV:supported-live-property element, as defined
        !           748:     **      by the DeltaV extensions to RFC2518.
        !           749:     **                      
        !           750:     ** Providers should return DAV_PROP_INSERT_NOTDEF if the property is
        !           751:     ** known and not defined for this resource, so should be handled as a
        !           752:     ** dead property. If a provider recognizes, but does not support, a
        !           753:     ** property, and does not want it handled as a dead property, it should
        !           754:     ** return DAV_PROP_INSERT_NOTSUPP.
        !           755:     **
        !           756:     ** Returns one of DAV_PROP_INSERT_* based on what happened.
        !           757:     **
        !           758:     ** ### we may need more context... ie. the lock database
        !           759:     */
        !           760:     dav_prop_insert (*insert_prop)(const dav_resource *resource,
        !           761:                                    int propid, dav_prop_insert what,
        !           762:                                    apr_text_header *phdr);
        !           763: 
        !           764:     /*
        !           765:     ** Determine whether a given property is writable.
        !           766:     **
        !           767:     ** ### we may want a different semantic. i.e. maybe it should be
        !           768:     ** ### "can we write <value> into this property?"
        !           769:     **
        !           770:     ** Returns 1 if the live property can be written, 0 if read-only.
        !           771:     */
        !           772:     int (*is_writable)(const dav_resource *resource, int propid);
        !           773: 
        !           774:     /*
        !           775:     ** This member defines the set of namespace URIs that the provider
        !           776:     ** uses for its properties. When insert_all is called, it will be
        !           777:     ** passed a list of integers that map from indices into this list
        !           778:     ** to namespace IDs for output generation.
        !           779:     **
        !           780:     ** The last entry in this list should be a NULL value (sentinel).
        !           781:     */
        !           782:     const char * const * namespace_uris;
        !           783: 
        !           784:     /*
        !           785:     ** ### this is not the final design. we want an open-ended way for
        !           786:     ** ### liveprop providers to attach *new* properties. To this end,
        !           787:     ** ### we'll have a "give me a list of the props you define", a way
        !           788:     ** ### to check for a prop's existence, a way to validate a set/remove
        !           789:     ** ### of a prop, and a way to execute/commit/rollback that change.
        !           790:     */
        !           791: 
        !           792:     /*
        !           793:     ** Validate that the live property can be assigned a value, and that
        !           794:     ** the provided value is valid.
        !           795:     **
        !           796:     ** elem will point to the XML element that names the property. For
        !           797:     ** example:
        !           798:     **     <lp1:executable>T</lp1:executable>
        !           799:     **
        !           800:     ** The provider can access the cdata fields and the child elements
        !           801:     ** to extract the relevant pieces.
        !           802:     **
        !           803:     ** operation is one of DAV_PROP_OP_SET or _DELETE.
        !           804:     **
        !           805:     ** The provider may return a value in *context which will be passed
        !           806:     ** to each of the exec/commit/rollback functions. For example, this
        !           807:     ** may contain an internal value which has been processed from the
        !           808:     ** input element.
        !           809:     **
        !           810:     ** The provider must set defer_to_dead to true (non-zero) or false.
        !           811:     ** If true, then the set/remove is deferred to the dead property
        !           812:     ** database. Note: it will be set to zero on entry.
        !           813:     */
        !           814:     dav_error * (*patch_validate)(const dav_resource *resource,
        !           815:                                   const apr_xml_elem *elem,
        !           816:                                   int operation,
        !           817:                                   void **context,
        !           818:                                   int *defer_to_dead);
        !           819: 
        !           820:     /* ### doc... */
        !           821:     dav_error * (*patch_exec)(const dav_resource *resource,
        !           822:                               const apr_xml_elem *elem,
        !           823:                               int operation,
        !           824:                               void *context,
        !           825:                               dav_liveprop_rollback **rollback_ctx);
        !           826: 
        !           827:     /* ### doc... */
        !           828:     void (*patch_commit)(const dav_resource *resource,
        !           829:                          int operation,
        !           830:                          void *context,
        !           831:                          dav_liveprop_rollback *rollback_ctx);
        !           832: 
        !           833:     /* ### doc... */
        !           834:     dav_error * (*patch_rollback)(const dav_resource *resource,
        !           835:                                   int operation,
        !           836:                                   void *context,
        !           837:                                   dav_liveprop_rollback *rollback_ctx);
        !           838: 
        !           839:     /*
        !           840:     ** If a provider needs a context to associate with this hooks structure,
        !           841:     ** then this field may be used. In most cases, it will just be NULL.
        !           842:     */
        !           843:     void *ctx;
        !           844: };
        !           845: 
        !           846: /*
        !           847: ** dav_liveprop_spec: specify a live property
        !           848: **
        !           849: ** This structure is used as a standard way to determine if a particular
        !           850: ** property is a live property. Its use is not part of the mandated liveprop
        !           851: ** interface, but can be used by liveprop providers in conjuction with the
        !           852: ** utility routines below.
        !           853: **
        !           854: ** spec->name == NULL is the defined end-sentinel for a list of specs.
        !           855: */
        !           856: typedef struct {
        !           857:     int ns;             /* provider-local namespace index */
        !           858:     const char *name;   /* name of the property */
        !           859: 
        !           860:     int propid;         /* provider-local property ID */
        !           861: 
        !           862:     int is_writable;    /* is the property writable? */
        !           863: 
        !           864: } dav_liveprop_spec;
        !           865: 
        !           866: /*
        !           867: ** dav_liveprop_group: specify a group of liveprops
        !           868: **
        !           869: ** This structure specifies a group of live properties, their namespaces,
        !           870: ** and how to handle them.
        !           871: */
        !           872: typedef struct {
        !           873:     const dav_liveprop_spec *specs;
        !           874:     const char * const *namespace_uris;
        !           875:     const dav_hooks_liveprop *hooks;
        !           876: 
        !           877: } dav_liveprop_group;
        !           878: 
        !           879: /* ### docco */
        !           880: DAV_DECLARE(int) dav_do_find_liveprop(const char *ns_uri, const char *name,
        !           881:                                       const dav_liveprop_group *group,
        !           882:                                       const dav_hooks_liveprop **hooks);
        !           883: 
        !           884: /* ### docco */
        !           885: DAV_DECLARE(long) dav_get_liveprop_info(int propid,
        !           886:                                         const dav_liveprop_group *group,
        !           887:                                         const dav_liveprop_spec **info);
        !           888: 
        !           889: /* ### docco */
        !           890: DAV_DECLARE(void) dav_register_liveprop_group(apr_pool_t *pool, 
        !           891:                                               const dav_liveprop_group *group);
        !           892: 
        !           893: /* ### docco */
        !           894: DAV_DECLARE(long) dav_get_liveprop_ns_index(const char *uri);
        !           895: 
        !           896: /* ### docco */
        !           897: DAV_DECLARE(long) dav_get_liveprop_ns_count(void);
        !           898: 
        !           899: /* ### docco */
        !           900: DAV_DECLARE(void) dav_add_all_liveprop_xmlns(apr_pool_t *p,
        !           901:                                              apr_text_header *phdr);
        !           902: 
        !           903: /*
        !           904: ** The following three functions are part of mod_dav's internal handling
        !           905: ** for the core WebDAV properties. They are not part of mod_dav's API.
        !           906: */
        !           907: DAV_DECLARE_NONSTD(int) dav_core_find_liveprop(
        !           908:     const dav_resource *resource,
        !           909:     const char *ns_uri,
        !           910:     const char *name,
        !           911:     const dav_hooks_liveprop **hooks);
        !           912: DAV_DECLARE_NONSTD(void) dav_core_insert_all_liveprops(
        !           913:     request_rec *r,
        !           914:     const dav_resource *resource,
        !           915:     dav_prop_insert what,
        !           916:     apr_text_header *phdr);
        !           917: DAV_DECLARE_NONSTD(void) dav_core_register_uris(apr_pool_t *p);
        !           918: 
        !           919: 
        !           920: /*
        !           921: ** Standard WebDAV Property Identifiers
        !           922: **
        !           923: ** A live property provider does not need to use these; they are simply
        !           924: ** provided for convenience.
        !           925: **
        !           926: ** Property identifiers need to be unique within a given provider, but not
        !           927: ** *across* providers (note: this uniqueness constraint was different in
        !           928: ** older versions of mod_dav).
        !           929: **
        !           930: ** The identifiers start at 20000 to make it easier for providers to avoid
        !           931: ** conflicts with the standard properties. The properties are arranged
        !           932: ** alphabetically, and may be reordered from time to time (as properties
        !           933: ** are introduced).
        !           934: **
        !           935: ** NOTE: there is no problem with reordering (e.g. binary compat) since the
        !           936: ** identifiers are only used within a given provider, which would pick up
        !           937: ** the entire set of changes upon a recompile.
        !           938: */
        !           939: enum {
        !           940:     DAV_PROPID_BEGIN = 20000,
        !           941: 
        !           942:     /* Standard WebDAV properties (RFC 2518) */
        !           943:     DAV_PROPID_creationdate,
        !           944:     DAV_PROPID_displayname,
        !           945:     DAV_PROPID_getcontentlanguage,
        !           946:     DAV_PROPID_getcontentlength,
        !           947:     DAV_PROPID_getcontenttype,
        !           948:     DAV_PROPID_getetag,
        !           949:     DAV_PROPID_getlastmodified,
        !           950:     DAV_PROPID_lockdiscovery,
        !           951:     DAV_PROPID_resourcetype,
        !           952:     DAV_PROPID_source,
        !           953:     DAV_PROPID_supportedlock,
        !           954: 
        !           955:     /* DeltaV properties (from the I-D (#14)) */
        !           956:     DAV_PROPID_activity_checkout_set,
        !           957:     DAV_PROPID_activity_set,
        !           958:     DAV_PROPID_activity_version_set,
        !           959:     DAV_PROPID_auto_merge_set,
        !           960:     DAV_PROPID_auto_version,
        !           961:     DAV_PROPID_baseline_collection,
        !           962:     DAV_PROPID_baseline_controlled_collection,
        !           963:     DAV_PROPID_baseline_controlled_collection_set,
        !           964:     DAV_PROPID_checked_in,
        !           965:     DAV_PROPID_checked_out,
        !           966:     DAV_PROPID_checkin_fork,
        !           967:     DAV_PROPID_checkout_fork,
        !           968:     DAV_PROPID_checkout_set,
        !           969:     DAV_PROPID_comment,
        !           970:     DAV_PROPID_creator_displayname,
        !           971:     DAV_PROPID_current_activity_set,
        !           972:     DAV_PROPID_current_workspace_set,
        !           973:     DAV_PROPID_default_variant,
        !           974:     DAV_PROPID_eclipsed_set,
        !           975:     DAV_PROPID_label_name_set,
        !           976:     DAV_PROPID_merge_set,
        !           977:     DAV_PROPID_precursor_set,
        !           978:     DAV_PROPID_predecessor_set,
        !           979:     DAV_PROPID_root_version,
        !           980:     DAV_PROPID_subactivity_set,
        !           981:     DAV_PROPID_subbaseline_set,
        !           982:     DAV_PROPID_successor_set,
        !           983:     DAV_PROPID_supported_method_set,
        !           984:     DAV_PROPID_supported_live_property_set,
        !           985:     DAV_PROPID_supported_report_set,
        !           986:     DAV_PROPID_unreserved,
        !           987:     DAV_PROPID_variant_set,
        !           988:     DAV_PROPID_version_controlled_binding_set,
        !           989:     DAV_PROPID_version_controlled_configuration,
        !           990:     DAV_PROPID_version_history,
        !           991:     DAV_PROPID_version_name,
        !           992:     DAV_PROPID_workspace,
        !           993:     DAV_PROPID_workspace_checkout_set,
        !           994: 
        !           995:     DAV_PROPID_END
        !           996: };
        !           997: 
        !           998: /*
        !           999: ** Property Identifier Registration
        !          1000: **
        !          1001: ** At the moment, mod_dav requires live property providers to ensure that
        !          1002: ** each property returned has a unique value. For now, this is done through
        !          1003: ** central registration (there are no known providers other than the default,
        !          1004: ** so this remains manageable).
        !          1005: **
        !          1006: ** WARNING: the TEST ranges should never be "shipped".
        !          1007: */
        !          1008: #define DAV_PROPID_CORE         10000   /* ..10099. defined by mod_dav */
        !          1009: #define DAV_PROPID_FS           10100   /* ..10299.
        !          1010:                                            mod_dav filesystem provider. */
        !          1011: #define DAV_PROPID_TEST1        10300   /* ..10399 */
        !          1012: #define DAV_PROPID_TEST2        10400   /* ..10499 */
        !          1013: #define DAV_PROPID_TEST3        10500   /* ..10599 */
        !          1014: /* Next: 10600 */
        !          1015: 
        !          1016: 
        !          1017: /* --------------------------------------------------------------------
        !          1018: **
        !          1019: ** DATABASE FUNCTIONS
        !          1020: */
        !          1021: 
        !          1022: typedef struct dav_db dav_db;
        !          1023: typedef struct dav_namespace_map dav_namespace_map;
        !          1024: typedef struct dav_deadprop_rollback dav_deadprop_rollback;
        !          1025: 
        !          1026: typedef struct {
        !          1027:     const char *ns;     /* "" signals "no namespace" */
        !          1028:     const char *name;
        !          1029: } dav_prop_name;
        !          1030: 
        !          1031: /* hook functions to enable pluggable databases */
        !          1032: struct dav_hooks_propdb
        !          1033: {
        !          1034:     dav_error * (*open)(apr_pool_t *p, const dav_resource *resource, int ro,
        !          1035:                         dav_db **pdb);
        !          1036:     void (*close)(dav_db *db);
        !          1037: 
        !          1038:     /*
        !          1039:     ** In bulk, define any namespaces that the values and their name
        !          1040:     ** elements may need.
        !          1041:     **
        !          1042:     ** Note: sometimes mod_dav will defer calling this until output_value
        !          1043:     ** returns found==1. If the output process needs the dav_xmlns_info
        !          1044:     ** filled for its work, then it will need to fill it on demand rather
        !          1045:     ** than depending upon this hook to fill in the structure.
        !          1046:     **
        !          1047:     ** Note: this will *always* be called during an output sequence. Thus,
        !          1048:     ** the provider may rely solely on using this to fill the xmlns info.
        !          1049:     */
        !          1050:     dav_error * (*define_namespaces)(dav_db *db, dav_xmlns_info *xi);
        !          1051: 
        !          1052:     /*
        !          1053:     ** Output the value from the database (i.e. add an element name and
        !          1054:     ** the value into *phdr). Set *found based on whether the name/value
        !          1055:     ** was found in the propdb.
        !          1056:     **
        !          1057:     ** Note: it is NOT an error for the key/value pair to not exist.
        !          1058:     **
        !          1059:     ** The dav_xmlns_info passed to define_namespaces() is also passed to
        !          1060:     ** each output_value() call so that namespaces can be added on-demand.
        !          1061:     ** It can also be used to look up prefixes or URIs during the output
        !          1062:     ** process.
        !          1063:     */
        !          1064:     dav_error * (*output_value)(dav_db *db, const dav_prop_name *name,
        !          1065:                                 dav_xmlns_info *xi,
        !          1066:                                 apr_text_header *phdr, int *found);
        !          1067: 
        !          1068:     /*
        !          1069:     ** Build a mapping from "global" namespaces (stored in apr_xml_*)
        !          1070:     ** into provider-local namespace identifiers.
        !          1071:     **
        !          1072:     ** This mapping should be done once per set of namespaces, and the
        !          1073:     ** resulting mapping should be passed into the store() hook function.
        !          1074:     **
        !          1075:     ** Note: usually, there is just a single document/namespaces for all
        !          1076:     ** elements passed. However, the generality of creating multiple
        !          1077:     ** mappings and passing them to store() is provided here.
        !          1078:     **
        !          1079:     ** Note: this is only in preparation for a series of store() calls.
        !          1080:     ** As a result, the propdb must be open for read/write access when
        !          1081:     ** this function is called.
        !          1082:     */
        !          1083:     dav_error * (*map_namespaces)(dav_db *db,
        !          1084:                                   const apr_array_header_t *namespaces,
        !          1085:                                   dav_namespace_map **mapping);
        !          1086:     
        !          1087:     /*
        !          1088:     ** Store a property value for a given name. The value->combined field
        !          1089:     ** MUST be set for this call.
        !          1090:     **
        !          1091:     ** ### WARNING: current providers will quote the text within ELEM.
        !          1092:     ** ### this implies you can call this function only once with a given
        !          1093:     ** ### element structure (a second time will quote it again).
        !          1094:     */
        !          1095:     dav_error * (*store)(dav_db *db, const dav_prop_name *name,
        !          1096:                          const apr_xml_elem *elem,
        !          1097:                          dav_namespace_map *mapping);
        !          1098: 
        !          1099:     /* remove a given property */
        !          1100:     dav_error * (*remove)(dav_db *db, const dav_prop_name *name);
        !          1101: 
        !          1102:     /* returns 1 if the record specified by "key" exists; 0 otherwise */
        !          1103:     int (*exists)(dav_db *db, const dav_prop_name *name);
        !          1104: 
        !          1105:     /*
        !          1106:     ** Iterate over the property names in the database.
        !          1107:     **
        !          1108:     ** iter->name.ns == iter->name.name == NULL when there are no more names.
        !          1109:     **
        !          1110:     ** Note: only one iteration may occur over the propdb at a time.
        !          1111:     */
        !          1112:     dav_error * (*first_name)(dav_db *db, dav_prop_name *pname);
        !          1113:     dav_error * (*next_name)(dav_db *db, dav_prop_name *pname);
        !          1114: 
        !          1115:     /*
        !          1116:     ** Rollback support: get rollback context, and apply it.
        !          1117:     **
        !          1118:     ** struct dav_deadprop_rollback is a provider-private structure; it
        !          1119:     ** should remember the name, and the name's old value (or the fact that
        !          1120:     ** the value was not present, and should be deleted if a rollback occurs).
        !          1121:     */
        !          1122:     dav_error * (*get_rollback)(dav_db *db, const dav_prop_name *name,
        !          1123:                                 dav_deadprop_rollback **prollback);
        !          1124:     dav_error * (*apply_rollback)(dav_db *db,
        !          1125:                                   dav_deadprop_rollback *rollback);
        !          1126: 
        !          1127:     /*
        !          1128:     ** If a provider needs a context to associate with this hooks structure,
        !          1129:     ** then this field may be used. In most cases, it will just be NULL.
        !          1130:     */
        !          1131:     void *ctx;
        !          1132: };
        !          1133: 
        !          1134: 
        !          1135: /* --------------------------------------------------------------------
        !          1136: **
        !          1137: ** LOCK FUNCTIONS
        !          1138: */
        !          1139: 
        !          1140: /* Used to represent a Timeout header of "Infinity" */
        !          1141: #define DAV_TIMEOUT_INFINITE 0
        !          1142: 
        !          1143: DAV_DECLARE(time_t) dav_get_timeout(request_rec *r);
        !          1144: 
        !          1145: /*
        !          1146: ** Opaque, provider-specific information for a lock database.
        !          1147: */
        !          1148: typedef struct dav_lockdb_private dav_lockdb_private;
        !          1149: 
        !          1150: /*
        !          1151: ** Opaque, provider-specific information for a lock record.
        !          1152: */
        !          1153: typedef struct dav_lock_private dav_lock_private;
        !          1154: 
        !          1155: /*
        !          1156: ** Lock database type. Lock providers are urged to implement a "lazy" open, so
        !          1157: ** doing an "open" is cheap until something is actually needed from the DB.
        !          1158: */
        !          1159: typedef struct
        !          1160: {
        !          1161:     const dav_hooks_locks *hooks;   /* the hooks used for this lockdb */
        !          1162:     int ro;                         /* was it opened readonly? */
        !          1163: 
        !          1164:     dav_lockdb_private *info;
        !          1165: 
        !          1166: } dav_lockdb;
        !          1167: 
        !          1168: typedef enum {
        !          1169:     DAV_LOCKSCOPE_UNKNOWN,
        !          1170:     DAV_LOCKSCOPE_EXCLUSIVE,
        !          1171:     DAV_LOCKSCOPE_SHARED
        !          1172: } dav_lock_scope;
        !          1173: 
        !          1174: typedef enum {
        !          1175:     DAV_LOCKTYPE_UNKNOWN,
        !          1176:     DAV_LOCKTYPE_WRITE
        !          1177: } dav_lock_type;
        !          1178: 
        !          1179: typedef enum {
        !          1180:     DAV_LOCKREC_DIRECT,             /* lock asserted on this resource */
        !          1181:     DAV_LOCKREC_INDIRECT,           /* lock inherited from a parent */
        !          1182:     DAV_LOCKREC_INDIRECT_PARTIAL    /* most info is not filled in */
        !          1183: } dav_lock_rectype;
        !          1184: 
        !          1185: /*
        !          1186: ** dav_lock: hold information about a lock on a resource.
        !          1187: **
        !          1188: ** This structure is used for both direct and indirect locks. A direct lock
        !          1189: ** is a lock applied to a specific resource by the client. An indirect lock
        !          1190: ** is one that is inherited from a parent resource by virtue of a non-zero
        !          1191: ** Depth: header when the lock was applied.
        !          1192: **
        !          1193: ** mod_dav records both types of locks in the lock database, managing their
        !          1194: ** addition/removal as resources are moved about the namespace.
        !          1195: **
        !          1196: ** Note that the lockdb is free to marshal this structure in any form that
        !          1197: ** it likes.
        !          1198: **
        !          1199: ** For a "partial" lock, the <rectype> and <locktoken> fields must be filled
        !          1200: ** in. All other (user) fields should be zeroed. The lock provider will
        !          1201: ** usually fill in the <info> field, and the <next> field may be used to
        !          1202: ** construct a list of partial locks.
        !          1203: **
        !          1204: ** The lock provider MUST use the info field to store a value such that a
        !          1205: ** dav_lock structure can locate itself in the underlying lock database.
        !          1206: ** This requirement is needed for refreshing: when an indirect dav_lock is
        !          1207: ** refreshed, its reference to the direct lock does not specify the direct's
        !          1208: ** resource, so the only way to locate the (refreshed, direct) lock in the
        !          1209: ** database is to use the info field.
        !          1210: **
        !          1211: ** Note that <is_locknull> only refers to the resource where this lock was
        !          1212: ** found.
        !          1213: ** ### hrm. that says the abstraction is wrong. is_locknull may disappear.
        !          1214: */
        !          1215: typedef struct dav_lock
        !          1216: {
        !          1217:     dav_lock_rectype rectype;   /* type of lock record */
        !          1218:     int is_locknull;            /* lock establishes a locknull resource */
        !          1219: 
        !          1220:     /* ### put the resource in here? */
        !          1221: 
        !          1222:     dav_lock_scope scope;       /* scope of the lock */
        !          1223:     dav_lock_type type;         /* type of lock */
        !          1224:     int depth;                  /* depth of the lock */
        !          1225:     time_t timeout;             /* when the lock will timeout */
        !          1226: 
        !          1227:     const dav_locktoken *locktoken;  /* the token that was issued */
        !          1228: 
        !          1229:     const char *owner;          /* (XML) owner of the lock */
        !          1230:     const char *auth_user;      /* auth'd username owning lock */
        !          1231: 
        !          1232:     dav_lock_private *info;     /* private to the lockdb */
        !          1233: 
        !          1234:     struct dav_lock *next;      /* for managing a list of locks */
        !          1235: } dav_lock;
        !          1236: 
        !          1237: /* Property-related public lock functions */
        !          1238: DAV_DECLARE(const char *)dav_lock_get_activelock(request_rec *r,
        !          1239:                                                  dav_lock *locks,
        !          1240:                                                  dav_buffer *pbuf);
        !          1241: 
        !          1242: /* LockDB-related public lock functions */
        !          1243: DAV_DECLARE(dav_error *) dav_lock_parse_lockinfo(request_rec *r,
        !          1244:                                                  const dav_resource *resrouce,
        !          1245:                                                  dav_lockdb *lockdb,
        !          1246:                                                  const apr_xml_doc *doc,
        !          1247:                                                  dav_lock **lock_request);
        !          1248: DAV_DECLARE(int) dav_unlock(request_rec *r,
        !          1249:                             const dav_resource *resource,
        !          1250:                             const dav_locktoken *locktoken);
        !          1251: DAV_DECLARE(dav_error *) dav_add_lock(request_rec *r,
        !          1252:                                       const dav_resource *resource,
        !          1253:                                       dav_lockdb *lockdb, dav_lock *request,
        !          1254:                                       dav_response **response);
        !          1255: DAV_DECLARE(dav_error *) dav_notify_created(request_rec *r,
        !          1256:                                             dav_lockdb *lockdb,
        !          1257:                                             const dav_resource *resource,
        !          1258:                                             int resource_state,
        !          1259:                                             int depth);
        !          1260: 
        !          1261: DAV_DECLARE(dav_error*) dav_lock_query(dav_lockdb *lockdb, 
        !          1262:                                        const dav_resource *resource,
        !          1263:                                        dav_lock **locks);
        !          1264: 
        !          1265: DAV_DECLARE(dav_error *) dav_validate_request(request_rec *r,
        !          1266:                                               dav_resource *resource,
        !          1267:                                               int depth,
        !          1268:                                               dav_locktoken *locktoken,
        !          1269:                                               dav_response **response,
        !          1270:                                               int flags,
        !          1271:                                               dav_lockdb *lockdb);
        !          1272: /*
        !          1273: ** flags:
        !          1274: **    0x0F -- reserved for <dav_lock_scope> values
        !          1275: **
        !          1276: **    other flags, detailed below
        !          1277: */
        !          1278: #define DAV_VALIDATE_RESOURCE   0x0010  /* validate just the resource */
        !          1279: #define DAV_VALIDATE_PARENT     0x0020  /* validate resource AND its parent */
        !          1280: #define DAV_VALIDATE_ADD_LD     0x0040  /* add DAV:lockdiscovery into
        !          1281:                                            the 424 DAV:response */
        !          1282: #define DAV_VALIDATE_USE_424    0x0080  /* return 424 status, not 207 */
        !          1283: #define DAV_VALIDATE_IS_PARENT  0x0100  /* for internal use */
        !          1284: 
        !          1285: /* Lock-null related public lock functions */
        !          1286: DAV_DECLARE(int) dav_get_resource_state(request_rec *r,
        !          1287:                                         const dav_resource *resource);
        !          1288: 
        !          1289: /* Lock provider hooks. Locking is optional, so there may be no
        !          1290:  * lock provider for a given repository.
        !          1291:  */
        !          1292: struct dav_hooks_locks
        !          1293: {
        !          1294:     /* Return the supportedlock property for a resource */
        !          1295:     const char * (*get_supportedlock)(
        !          1296:         const dav_resource *resource
        !          1297:     );
        !          1298: 
        !          1299:     /* Parse a lock token URI, returning a lock token object allocated
        !          1300:      * in the given pool.
        !          1301:      */
        !          1302:     dav_error * (*parse_locktoken)(
        !          1303:         apr_pool_t *p,
        !          1304:         const char *char_token,
        !          1305:         dav_locktoken **locktoken_p
        !          1306:     );
        !          1307: 
        !          1308:     /* Format a lock token object into a URI string, allocated in
        !          1309:      * the given pool.
        !          1310:      *
        !          1311:      * Always returns non-NULL.
        !          1312:      */
        !          1313:     const char * (*format_locktoken)(
        !          1314:         apr_pool_t *p,
        !          1315:         const dav_locktoken *locktoken
        !          1316:     );
        !          1317: 
        !          1318:     /* Compare two lock tokens.
        !          1319:      *
        !          1320:      * Result < 0  => lt1 < lt2
        !          1321:      * Result == 0 => lt1 == lt2
        !          1322:      * Result > 0  => lt1 > lt2
        !          1323:      */
        !          1324:     int (*compare_locktoken)(
        !          1325:         const dav_locktoken *lt1,
        !          1326:         const dav_locktoken *lt2
        !          1327:     );
        !          1328: 
        !          1329:     /* Open the provider's lock database.
        !          1330:      *
        !          1331:      * The provider may or may not use a "real" database for locks
        !          1332:      * (a lock could be an attribute on a resource, for example).
        !          1333:      *
        !          1334:      * The provider may choose to use the value of the DAVLockDB directive
        !          1335:      * (as returned by dav_get_lockdb_path()) to decide where to place
        !          1336:      * any storage it may need.
        !          1337:      *
        !          1338:      * The request storage pool should be associated with the lockdb,
        !          1339:      * so it can be used in subsequent operations.
        !          1340:      *
        !          1341:      * If ro != 0, only readonly operations will be performed.
        !          1342:      * If force == 0, the open can be "lazy"; no subsequent locking operations
        !          1343:      * may occur.
        !          1344:      * If force != 0, locking operations will definitely occur.
        !          1345:      */
        !          1346:     dav_error * (*open_lockdb)(
        !          1347:         request_rec *r,
        !          1348:         int ro,
        !          1349:         int force,
        !          1350:         dav_lockdb **lockdb
        !          1351:     );
        !          1352: 
        !          1353:     /* Indicates completion of locking operations */
        !          1354:     void (*close_lockdb)(
        !          1355:         dav_lockdb *lockdb
        !          1356:     );
        !          1357: 
        !          1358:     /* Take a resource out of the lock-null state. */
        !          1359:     dav_error * (*remove_locknull_state)(
        !          1360:         dav_lockdb *lockdb,
        !          1361:         const dav_resource *resource
        !          1362:     );
        !          1363: 
        !          1364:     /*
        !          1365:     ** Create a (direct) lock structure for the given resource. A locktoken
        !          1366:     ** will be created.
        !          1367:     **
        !          1368:     ** The lock provider may store private information into lock->info.
        !          1369:     */
        !          1370:     dav_error * (*create_lock)(dav_lockdb *lockdb,
        !          1371:                                const dav_resource *resource,
        !          1372:                                dav_lock **lock);
        !          1373: 
        !          1374:     /*
        !          1375:     ** Get the locks associated with the specified resource.
        !          1376:     **
        !          1377:     ** If resolve_locks is true (non-zero), then any indirect locks are
        !          1378:     ** resolved to their actual, direct lock (i.e. the reference to followed
        !          1379:     ** to the original lock).
        !          1380:     **
        !          1381:     ** The locks, if any, are returned as a linked list in no particular
        !          1382:     ** order. If no locks are present, then *locks will be NULL.
        !          1383:     */
        !          1384:     dav_error * (*get_locks)(dav_lockdb *lockdb,
        !          1385:                              const dav_resource *resource,
        !          1386:                              int calltype,
        !          1387:                              dav_lock **locks);
        !          1388: 
        !          1389: #define DAV_GETLOCKS_RESOLVED   0    /* resolve indirects to directs */
        !          1390: #define DAV_GETLOCKS_PARTIAL    1    /* leave indirects partially filled */
        !          1391: #define DAV_GETLOCKS_COMPLETE   2    /* fill out indirect locks */
        !          1392: 
        !          1393:     /*
        !          1394:     ** Find a particular lock on a resource (specified by its locktoken).
        !          1395:     **
        !          1396:     ** *lock will be set to NULL if the lock is not found.
        !          1397:     **
        !          1398:     ** Note that the provider can optimize the unmarshalling -- only one
        !          1399:     ** lock (or none) must be constructed and returned.
        !          1400:     **
        !          1401:     ** If partial_ok is true (non-zero), then an indirect lock can be
        !          1402:     ** partially filled in. Otherwise, another lookup is done and the
        !          1403:     ** lock structure will be filled out as a DAV_LOCKREC_INDIRECT.
        !          1404:     */
        !          1405:     dav_error * (*find_lock)(dav_lockdb *lockdb,
        !          1406:                              const dav_resource *resource,
        !          1407:                              const dav_locktoken *locktoken,
        !          1408:                              int partial_ok,
        !          1409:                              dav_lock **lock);
        !          1410: 
        !          1411:     /*
        !          1412:     ** Quick test to see if the resource has *any* locks on it.
        !          1413:     **
        !          1414:     ** This is typically used to determine if a non-existent resource
        !          1415:     ** has a lock and is (therefore) a locknull resource.
        !          1416:     **
        !          1417:     ** WARNING: this function may return TRUE even when timed-out locks
        !          1418:     **          exist (i.e. it may not perform timeout checks).
        !          1419:     */
        !          1420:     dav_error * (*has_locks)(dav_lockdb *lockdb,
        !          1421:                              const dav_resource *resource,
        !          1422:                              int *locks_present);
        !          1423: 
        !          1424:     /*
        !          1425:     ** Append the specified lock(s) to the set of locks on this resource.
        !          1426:     **
        !          1427:     ** If "make_indirect" is true (non-zero), then the specified lock(s)
        !          1428:     ** should be converted to an indirect lock (if it is a direct lock)
        !          1429:     ** before appending. Note that the conversion to an indirect lock does
        !          1430:     ** not alter the passed-in lock -- the change is internal the
        !          1431:     ** append_locks function.
        !          1432:     **
        !          1433:     ** Multiple locks are specified using the lock->next links.
        !          1434:     */
        !          1435:     dav_error * (*append_locks)(dav_lockdb *lockdb,
        !          1436:                                 const dav_resource *resource,
        !          1437:                                 int make_indirect,
        !          1438:                                 const dav_lock *lock);
        !          1439: 
        !          1440:     /*
        !          1441:     ** Remove any lock that has the specified locktoken.
        !          1442:     **
        !          1443:     ** If locktoken == NULL, then ALL locks are removed.
        !          1444:     */
        !          1445:     dav_error * (*remove_lock)(dav_lockdb *lockdb,
        !          1446:                                const dav_resource *resource,
        !          1447:                                const dav_locktoken *locktoken);
        !          1448: 
        !          1449:     /*
        !          1450:     ** Refresh all locks, found on the specified resource, which has a
        !          1451:     ** locktoken in the provided list.
        !          1452:     **
        !          1453:     ** If the lock is indirect, then the direct lock is referenced and
        !          1454:     ** refreshed.
        !          1455:     **
        !          1456:     ** Each lock that is updated is returned in the <locks> argument.
        !          1457:     ** Note that the locks will be fully resolved.
        !          1458:     */
        !          1459:     dav_error * (*refresh_locks)(dav_lockdb *lockdb,
        !          1460:                                  const dav_resource *resource,
        !          1461:                                  const dav_locktoken_list *ltl,
        !          1462:                                  time_t new_time,
        !          1463:                                  dav_lock **locks);
        !          1464: 
        !          1465:     /*
        !          1466:     ** Look up the resource associated with a particular locktoken.
        !          1467:     **
        !          1468:     ** The search begins at the specified <start_resource> and the lock
        !          1469:     ** specified by <locktoken>.
        !          1470:     **
        !          1471:     ** If the resource/token specifies an indirect lock, then the direct
        !          1472:     ** lock will be looked up, and THAT resource will be returned. In other
        !          1473:     ** words, this function always returns the resource where a particular
        !          1474:     ** lock (token) was asserted.
        !          1475:     **
        !          1476:     ** NOTE: this function pointer is allowed to be NULL, indicating that
        !          1477:     **       the provider does not support this type of functionality. The
        !          1478:     **       caller should then traverse up the repository hierarchy looking
        !          1479:     **       for the resource defining a lock with this locktoken.
        !          1480:     */
        !          1481:     dav_error * (*lookup_resource)(dav_lockdb *lockdb,
        !          1482:                                    const dav_locktoken *locktoken,
        !          1483:                                    const dav_resource *start_resource,
        !          1484:                                    const dav_resource **resource);
        !          1485: 
        !          1486:     /*
        !          1487:     ** If a provider needs a context to associate with this hooks structure,
        !          1488:     ** then this field may be used. In most cases, it will just be NULL.
        !          1489:     */
        !          1490:     void *ctx;
        !          1491: };
        !          1492: 
        !          1493: /* what types of resources can be discovered by dav_get_resource_state() */
        !          1494: #define DAV_RESOURCE_LOCK_NULL  10    /* resource lock-null */
        !          1495: #define DAV_RESOURCE_NULL       11    /* resource null */
        !          1496: #define DAV_RESOURCE_EXISTS     12    /* resource exists */
        !          1497: #define DAV_RESOURCE_ERROR      13    /* an error occurred */
        !          1498: 
        !          1499: 
        !          1500: /* --------------------------------------------------------------------
        !          1501: **
        !          1502: ** PROPERTY HANDLING
        !          1503: */
        !          1504: 
        !          1505: typedef struct dav_propdb dav_propdb;
        !          1506: 
        !          1507: 
        !          1508: DAV_DECLARE(dav_error *) dav_open_propdb(
        !          1509:     request_rec *r,
        !          1510:     dav_lockdb *lockdb,
        !          1511:     const dav_resource *resource,
        !          1512:     int ro,
        !          1513:     apr_array_header_t *ns_xlate,
        !          1514:     dav_propdb **propdb);
        !          1515: 
        !          1516: DAV_DECLARE(void) dav_close_propdb(dav_propdb *db);
        !          1517: 
        !          1518: DAV_DECLARE(dav_get_props_result) dav_get_props(
        !          1519:     dav_propdb *db,
        !          1520:     apr_xml_doc *doc);
        !          1521: 
        !          1522: DAV_DECLARE(dav_get_props_result) dav_get_allprops(
        !          1523:     dav_propdb *db,
        !          1524:     dav_prop_insert what);
        !          1525: 
        !          1526: DAV_DECLARE(void) dav_get_liveprop_supported(
        !          1527:     dav_propdb *propdb,
        !          1528:     const char *ns_uri,
        !          1529:     const char *propname,
        !          1530:     apr_text_header *body);
        !          1531: 
        !          1532: /*
        !          1533: ** 3-phase property modification.
        !          1534: **
        !          1535: **   1) validate props. readable? unlocked? ACLs allow access?
        !          1536: **   2) execute operation (set/delete)
        !          1537: **   3) commit or rollback
        !          1538: **
        !          1539: ** ### eventually, auth must be available. a ref to the request_rec (which
        !          1540: ** ### contains the auth info) should be in the shared context struct.
        !          1541: **
        !          1542: ** Each function may alter the error values and information contained within
        !          1543: ** the context record. This should be done as an "increasing" level of
        !          1544: ** error, rather than overwriting any previous error.
        !          1545: **
        !          1546: ** Note that commit() cannot generate errors. It should simply free the
        !          1547: ** rollback information.
        !          1548: **
        !          1549: ** rollback() may generate additional errors because the rollback operation
        !          1550: ** can sometimes fail(!).
        !          1551: **
        !          1552: ** The caller should allocate an array of these, one per operation. It should
        !          1553: ** be zero-initialized, then the db, operation, and prop fields should be
        !          1554: ** filled in before calling dav_prop_validate. Note that the set/delete
        !          1555: ** operations are order-dependent. For a given (logical) context, the same
        !          1556: ** pointer must be passed to each phase.
        !          1557: **
        !          1558: ** error_type is an internal value, but will have the same numeric value
        !          1559: ** for each possible "desc" value. This allows the caller to group the
        !          1560: ** descriptions via the error_type variable, rather than through string
        !          1561: ** comparisons. Note that "status" does not provide enough granularity to
        !          1562: ** differentiate/group the "desc" values.
        !          1563: **
        !          1564: ** Note that the propdb will maintain some (global) context across all
        !          1565: ** of the property change contexts. This implies that you can have only
        !          1566: ** one open transaction per propdb.
        !          1567: */
        !          1568: typedef struct dav_prop_ctx
        !          1569: {
        !          1570:     dav_propdb *propdb;
        !          1571: 
        !          1572:     int operation;
        !          1573: #define DAV_PROP_OP_SET        1    /* set a property value */
        !          1574: #define DAV_PROP_OP_DELETE     2    /* delete a prop value */
        !          1575: /* ### add a GET? */
        !          1576: 
        !          1577:     apr_xml_elem *prop;             /* property to affect */
        !          1578: 
        !          1579:     dav_error *err;                 /* error (if any) */
        !          1580: 
        !          1581:     /* private items to the propdb */
        !          1582:     int is_liveprop;
        !          1583:     void *liveprop_ctx;
        !          1584:     struct dav_rollback_item *rollback;  /* optional rollback info */
        !          1585: 
        !          1586:     /* private to mod_dav.c */
        !          1587:     request_rec *r;
        !          1588: 
        !          1589: } dav_prop_ctx;
        !          1590: 
        !          1591: DAV_DECLARE_NONSTD(void) dav_prop_validate(dav_prop_ctx *ctx);
        !          1592: DAV_DECLARE_NONSTD(void) dav_prop_exec(dav_prop_ctx *ctx);
        !          1593: DAV_DECLARE_NONSTD(void) dav_prop_commit(dav_prop_ctx *ctx);
        !          1594: DAV_DECLARE_NONSTD(void) dav_prop_rollback(dav_prop_ctx *ctx);
        !          1595: 
        !          1596: #define DAV_PROP_CTX_HAS_ERR(dpc)  ((dpc).err && (dpc).err->status >= 300)
        !          1597: 
        !          1598: 
        !          1599: /* --------------------------------------------------------------------
        !          1600: **
        !          1601: ** WALKER STRUCTURE
        !          1602: */
        !          1603: 
        !          1604: enum {
        !          1605:     DAV_CALLTYPE_MEMBER = 1,    /* called for a member resource */
        !          1606:     DAV_CALLTYPE_COLLECTION,    /* called for a collection */
        !          1607:     DAV_CALLTYPE_LOCKNULL       /* called for a locknull resource */
        !          1608: };
        !          1609: 
        !          1610: typedef struct
        !          1611: {
        !          1612:     /* the client-provided context */
        !          1613:     void *walk_ctx;
        !          1614: 
        !          1615:     /* pool to use for allocations in the callback */
        !          1616:     apr_pool_t *pool;
        !          1617: 
        !          1618:     /* the current resource */
        !          1619:     const dav_resource *resource;
        !          1620: 
        !          1621:     /* OUTPUT: add responses to this */
        !          1622:     dav_response *response;
        !          1623: 
        !          1624: } dav_walk_resource;
        !          1625: 
        !          1626: typedef struct
        !          1627: {
        !          1628:     int walk_type;
        !          1629: #define DAV_WALKTYPE_AUTH       0x0001  /* limit to authorized files */
        !          1630: #define DAV_WALKTYPE_NORMAL     0x0002  /* walk normal files */
        !          1631: #define DAV_WALKTYPE_LOCKNULL   0x0004  /* walk locknull resources */
        !          1632: 
        !          1633:     /* callback function and a client context for the walk */
        !          1634:     dav_error * (*func)(dav_walk_resource *wres, int calltype);
        !          1635:     void *walk_ctx;
        !          1636: 
        !          1637:     /* what pool to use for allocations needed by walk logic */
        !          1638:     apr_pool_t *pool;
        !          1639: 
        !          1640:     /* beginning root of the walk */
        !          1641:     const dav_resource *root;
        !          1642: 
        !          1643:     /* lock database to enable walking LOCKNULL resources */
        !          1644:     dav_lockdb *lockdb;
        !          1645: 
        !          1646: } dav_walk_params;
        !          1647: 
        !          1648: /* directory tree walking context */
        !          1649: typedef struct dav_walker_ctx
        !          1650: {
        !          1651:     /* input: */
        !          1652:     dav_walk_params w;
        !          1653: 
        !          1654: 
        !          1655:     /* ### client data... phasing out this big glom */
        !          1656: 
        !          1657:     /* this brigade buffers data being sent to r->output_filters */ 
        !          1658:     apr_bucket_brigade *bb;
        !          1659: 
        !          1660:     /* a scratch pool, used to stream responses and iteratively cleared. */
        !          1661:     apr_pool_t *scratchpool;
        !          1662: 
        !          1663:     request_rec *r;                 /* original request */
        !          1664: 
        !          1665:     /* for PROPFIND operations */
        !          1666:     apr_xml_doc *doc;
        !          1667:     int propfind_type;
        !          1668: #define DAV_PROPFIND_IS_ALLPROP     1
        !          1669: #define DAV_PROPFIND_IS_PROPNAME    2
        !          1670: #define DAV_PROPFIND_IS_PROP        3
        !          1671: 
        !          1672:     apr_text *propstat_404;         /* (cached) propstat giving a 404 error */
        !          1673: 
        !          1674:     const dav_if_header *if_header; /* for validation */
        !          1675:     const dav_locktoken *locktoken; /* for UNLOCK */
        !          1676:     const dav_lock *lock;           /* for LOCK */
        !          1677:     int skip_root;                  /* for dav_inherit_locks() */
        !          1678: 
        !          1679:     int flags;
        !          1680: 
        !          1681:     dav_buffer work_buf;            /* for dav_validate_request() */
        !          1682: 
        !          1683: } dav_walker_ctx;
        !          1684: 
        !          1685: DAV_DECLARE(void) dav_add_response(dav_walk_resource *wres,
        !          1686:                                    int status,
        !          1687:                                    dav_get_props_result *propstats);
        !          1688: 
        !          1689: 
        !          1690: /* --------------------------------------------------------------------
        !          1691: **
        !          1692: ** "STREAM" STRUCTURE
        !          1693: **
        !          1694: ** mod_dav uses this abstraction for interacting with the repository
        !          1695: ** while fetching/storing resources. mod_dav views resources as a stream
        !          1696: ** of bytes.
        !          1697: **
        !          1698: ** Note that the structure is opaque -- it is private to the repository
        !          1699: ** that created the stream in the repository's "open" function.
        !          1700: **
        !          1701: ** ### THIS STUFF IS GOING AWAY ... GET/read requests are handled by
        !          1702: ** ### having the provider jam stuff straight into the filter stack.
        !          1703: ** ### this is only left for handling PUT/write requests.
        !          1704: */
        !          1705: 
        !          1706: typedef struct dav_stream dav_stream;
        !          1707: 
        !          1708: typedef enum {
        !          1709:     DAV_MODE_WRITE_TRUNC,      /* truncate and open for writing */
        !          1710:     DAV_MODE_WRITE_SEEKABLE    /* open for writing; random access */
        !          1711: } dav_stream_mode;
        !          1712: 
        !          1713: 
        !          1714: /* --------------------------------------------------------------------
        !          1715: **
        !          1716: ** REPOSITORY FUNCTIONS
        !          1717: */
        !          1718: 
        !          1719: /* Repository provider hooks */
        !          1720: struct dav_hooks_repository
        !          1721: {
        !          1722:     /* Flag for whether repository requires special GET handling.
        !          1723:      * If resources in the repository are not visible in the
        !          1724:      * filesystem location which URLs map to, then special handling
        !          1725:      * is required to first fetch a resource from the repository,
        !          1726:      * respond to the GET request, then free the resource copy.
        !          1727:      */
        !          1728:     int handle_get;
        !          1729: 
        !          1730:     /* Get a resource descriptor for the URI in a request. A descriptor
        !          1731:      * should always be returned even if the resource does not exist. This
        !          1732:      * repository has been identified as handling the resource given by
        !          1733:      * the URI, so an answer must be given. If there is a problem with the
        !          1734:      * URI or accessing the resource or whatever, then an error should be
        !          1735:      * returned.
        !          1736:      *
        !          1737:      * root_dir:
        !          1738:      *   the root of the directory for which this repository is configured.
        !          1739:      *
        !          1740:      * label:
        !          1741:      *   if a Label: header is present (and allowed), this is the label
        !          1742:      *   to use to identify a version resource from the resource's
        !          1743:      *   corresponding version history. Otherwise, it will be NULL.
        !          1744:      *
        !          1745:      * use_checked_in:
        !          1746:      *   use the DAV:checked-in property of the resource identified by the
        !          1747:      *   Request-URI to identify and return a version resource
        !          1748:      *
        !          1749:      * The provider may associate the request storage pool with the resource
        !          1750:      * (in the resource->pool field), to use in other operations on that
        !          1751:      * resource. 
        !          1752:      */
        !          1753:     dav_error * (*get_resource)(
        !          1754:         request_rec *r,
        !          1755:         const char *root_dir,
        !          1756:         const char *label,
        !          1757:         int use_checked_in,
        !          1758:         dav_resource **resource
        !          1759:     );
        !          1760: 
        !          1761:     /* Get a resource descriptor for the parent of the given resource.
        !          1762:      * The resources need not exist.  NULL is returned if the resource 
        !          1763:      * is the root collection.
        !          1764:      *
        !          1765:      * An error should be returned only if there is a fatal error in
        !          1766:      * fetching information about the parent resource.
        !          1767:      */
        !          1768:     dav_error * (*get_parent_resource)(
        !          1769:         const dav_resource *resource,
        !          1770:         dav_resource **parent_resource
        !          1771:     );
        !          1772: 
        !          1773:     /* Determine whether two resource descriptors refer to the same resource.
        !          1774:     *
        !          1775:      * Result != 0 => the resources are the same.
        !          1776:      */
        !          1777:     int (*is_same_resource)(
        !          1778:         const dav_resource *res1,
        !          1779:         const dav_resource *res2
        !          1780:     );
        !          1781: 
        !          1782:     /* Determine whether one resource is a parent (immediate or otherwise)
        !          1783:      * of another.
        !          1784:      *
        !          1785:      * Result != 0 => res1 is a parent of res2.
        !          1786:      */
        !          1787:     int (*is_parent_resource)(
        !          1788:         const dav_resource *res1,
        !          1789:         const dav_resource *res2
        !          1790:     );
        !          1791: 
        !          1792:     /*
        !          1793:     ** Open a stream for this resource, using the specified mode. The
        !          1794:     ** stream will be returned in *stream.
        !          1795:     */
        !          1796:     dav_error * (*open_stream)(const dav_resource *resource,
        !          1797:                                dav_stream_mode mode,
        !          1798:                                dav_stream **stream);
        !          1799: 
        !          1800:     /*
        !          1801:     ** Close the specified stream.
        !          1802:     **
        !          1803:     ** mod_dav will (ideally) make sure to call this. For safety purposes,
        !          1804:     ** a provider should (ideally) register a cleanup function with the
        !          1805:     ** request pool to get this closed and cleaned up.
        !          1806:     **
        !          1807:     ** Note the possibility of an error from the close -- it is entirely
        !          1808:     ** feasible that the close does a "commit" of some kind, which can
        !          1809:     ** produce an error.
        !          1810:     **
        !          1811:     ** commit should be TRUE (non-zero) or FALSE (0) if the stream was
        !          1812:     ** opened for writing. This flag states whether to retain the file
        !          1813:     ** or not.
        !          1814:     ** Note: the commit flag is ignored for streams opened for reading.
        !          1815:     */
        !          1816:     dav_error * (*close_stream)(dav_stream *stream, int commit);
        !          1817: 
        !          1818:     /*
        !          1819:     ** Write data to the stream.
        !          1820:     **
        !          1821:     ** All of the bytes must be written, or an error should be returned.
        !          1822:     */
        !          1823:     dav_error * (*write_stream)(dav_stream *stream,
        !          1824:                                 const void *buf, apr_size_t bufsize);
        !          1825: 
        !          1826:     /*
        !          1827:     ** Seek to an absolute position in the stream. This is used to support
        !          1828:     ** Content-Range in a GET/PUT.
        !          1829:     **
        !          1830:     ** NOTE: if this function is NULL (which is allowed), then any
        !          1831:     **       operations using Content-Range will be refused.
        !          1832:     */
        !          1833:     dav_error * (*seek_stream)(dav_stream *stream, apr_off_t abs_position);
        !          1834: 
        !          1835:     /*
        !          1836:     ** If a GET is processed using a stream (open_stream, read_stream)
        !          1837:     ** rather than via a sub-request (on get_pathname), then this function
        !          1838:     ** is used to provide the repository with a way to set the headers
        !          1839:     ** in the response.
        !          1840:     **
        !          1841:     ** This function may be called without a following deliver(), to
        !          1842:     ** handle a HEAD request.
        !          1843:     **
        !          1844:     ** This may be NULL if handle_get is FALSE.
        !          1845:     */
        !          1846:     dav_error * (*set_headers)(request_rec *r,
        !          1847:                                const dav_resource *resource);
        !          1848: 
        !          1849:     /*
        !          1850:     ** The provider should deliver the resource into the specified filter.
        !          1851:     ** Basically, this is the response to the GET method.
        !          1852:     **
        !          1853:     ** Note that this is called for all resources, including collections.
        !          1854:     ** The provider should determine what has content to deliver or not.
        !          1855:     **
        !          1856:     ** set_headers will be called prior to this function, allowing the
        !          1857:     ** provider to set the appropriate response headers.
        !          1858:     **
        !          1859:     ** This may be NULL if handle_get is FALSE.
        !          1860:     ** ### maybe toss handle_get and just use this function as the marker
        !          1861:     */
        !          1862:     dav_error * (*deliver)(const dav_resource *resource,
        !          1863:                            ap_filter_t *output);
        !          1864: 
        !          1865:     /* Create a collection resource. The resource must not already exist.
        !          1866:      *
        !          1867:      * Result == NULL if the collection was created successfully. Also, the
        !          1868:      * resource object is updated to reflect that the resource exists, and
        !          1869:      * is a collection.
        !          1870:      */
        !          1871:     dav_error * (*create_collection)(
        !          1872:         dav_resource *resource
        !          1873:     );
        !          1874: 
        !          1875:     /* Copy one resource to another. The destination may exist, if it is
        !          1876:      * versioned.
        !          1877:      * Handles both files and collections. Properties are copied as well.
        !          1878:      * If the destination exists and is versioned, the provider must update
        !          1879:      * the destination to have identical content to the source,
        !          1880:      * recursively for collections.
        !          1881:      * The depth argument is ignored for a file, and can be either 0 or
        !          1882:      * DAV_INFINITY for a collection.
        !          1883:      * If an error occurs in a child resource, then the return value is
        !          1884:      * non-NULL, and *response is set to a multistatus response.
        !          1885:      * If the copy is successful, the dst resource object is
        !          1886:      * updated to reflect that the resource exists.
        !          1887:      */
        !          1888:     dav_error * (*copy_resource)(
        !          1889:         const dav_resource *src,
        !          1890:         dav_resource *dst,
        !          1891:         int depth,
        !          1892:         dav_response **response
        !          1893:     );
        !          1894: 
        !          1895:     /* Move one resource to another. The destination must not exist.
        !          1896:      * Handles both files and collections. Properties are moved as well.
        !          1897:      * If an error occurs in a child resource, then the return value is
        !          1898:      * non-NULL, and *response is set to a multistatus response.
        !          1899:      * If the move is successful, the src and dst resource objects are
        !          1900:      * updated to reflect that the source no longer exists, and the
        !          1901:      * destination does.
        !          1902:      */
        !          1903:     dav_error * (*move_resource)(
        !          1904:         dav_resource *src,
        !          1905:         dav_resource *dst,
        !          1906:         dav_response **response
        !          1907:     );
        !          1908: 
        !          1909:     /* Remove a resource. Handles both files and collections.
        !          1910:      * Removes any associated properties as well.
        !          1911:      * If an error occurs in a child resource, then the return value is
        !          1912:      * non-NULL, and *response is set to a multistatus response.
        !          1913:      * If the delete is successful, the resource object is updated to
        !          1914:      * reflect that the resource no longer exists.
        !          1915:      */
        !          1916:     dav_error * (*remove_resource)(
        !          1917:         dav_resource *resource,
        !          1918:         dav_response **response
        !          1919:     );
        !          1920: 
        !          1921:     /* Walk a resource hierarchy.
        !          1922:      *
        !          1923:      * Iterates over the resource hierarchy specified by params->root.
        !          1924:      * Control of the walk and the callback are specified by 'params'.
        !          1925:      *
        !          1926:      * An error may be returned. *response will contain multistatus
        !          1927:      * responses (if any) suitable for the body of the error. It is also
        !          1928:      * possible to return NULL, yet still have multistatus responses.
        !          1929:      * In this case, typically the caller should return a 207 (Multistatus)
        !          1930:      * and the responses (in the body) as the HTTP response.
        !          1931:      */
        !          1932:     dav_error * (*walk)(const dav_walk_params *params, int depth,
        !          1933:                         dav_response **response);
        !          1934: 
        !          1935:     /* Get the entity tag for a resource */
        !          1936:     const char * (*getetag)(const dav_resource *resource);
        !          1937: 
        !          1938:     /*
        !          1939:     ** If a provider needs a context to associate with this hooks structure,
        !          1940:     ** then this field may be used. In most cases, it will just be NULL.
        !          1941:     */
        !          1942:     void *ctx;
        !          1943: };
        !          1944: 
        !          1945: 
        !          1946: /* --------------------------------------------------------------------
        !          1947: **
        !          1948: ** VERSIONING FUNCTIONS
        !          1949: */
        !          1950: 
        !          1951: 
        !          1952: /* dav_add_vary_header
        !          1953:  *
        !          1954:  * If there were any headers in the request which require a Vary header
        !          1955:  * in the response, add it.
        !          1956:  */
        !          1957: DAV_DECLARE(void) dav_add_vary_header(request_rec *in_req,
        !          1958:                                       request_rec *out_req,
        !          1959:                                       const dav_resource *resource);
        !          1960: 
        !          1961: /*
        !          1962: ** Flags specifying auto-versioning behavior, returned by
        !          1963: ** the auto_versionable hook. The value returned depends
        !          1964: ** on both the state of the resource and the value of the
        !          1965: ** DAV:auto-versioning property for the resource.
        !          1966: **
        !          1967: ** If the resource does not exist (null or lock-null),
        !          1968: ** DAV_AUTO_VERSION_ALWAYS causes creation of a new version-controlled resource
        !          1969: **
        !          1970: ** If the resource is checked in,
        !          1971: ** DAV_AUTO_VERSION_ALWAYS causes it to be checked out always,
        !          1972: ** DAV_AUTO_VERSION_LOCKED causes it to be checked out only when locked
        !          1973: **
        !          1974: ** If the resource is checked out,
        !          1975: ** DAV_AUTO_VERSION_ALWAYS causes it to be checked in always,
        !          1976: ** DAV_AUTO_VERSION_LOCKED causes it to be checked in when unlocked
        !          1977: ** (note: a provider should allow auto-checkin only for resources which
        !          1978: ** were automatically checked out)
        !          1979: **
        !          1980: ** In all cases, DAV_AUTO_VERSION_NEVER results in no auto-versioning behavior.
        !          1981: */
        !          1982: typedef enum {
        !          1983:     DAV_AUTO_VERSION_NEVER,
        !          1984:     DAV_AUTO_VERSION_ALWAYS,
        !          1985:     DAV_AUTO_VERSION_LOCKED
        !          1986: } dav_auto_version;
        !          1987: 
        !          1988: /*
        !          1989: ** This structure is used to record what auto-versioning operations
        !          1990: ** were done to make a resource writable, so that they can be undone
        !          1991: ** at the end of a request.
        !          1992: */
        !          1993: typedef struct {
        !          1994:     int resource_versioned;             /* 1 => resource was auto-version-controlled */
        !          1995:     int resource_checkedout;            /* 1 => resource was auto-checked-out */
        !          1996:     int parent_checkedout;              /* 1 => parent was auto-checked-out */
        !          1997:     dav_resource *parent_resource;      /* parent resource, if it was needed */
        !          1998: } dav_auto_version_info;
        !          1999: 
        !          2000: /* Ensure that a resource is writable. If there is no versioning
        !          2001:  * provider, then this is essentially a no-op. Versioning repositories
        !          2002:  * require explicit resource creation and checkout before they can
        !          2003:  * be written to. If a new resource is to be created, or an existing
        !          2004:  * resource deleted, the parent collection must be checked out as well.
        !          2005:  *
        !          2006:  * Set the parent_only flag to only make the parent collection writable.
        !          2007:  * Otherwise, both parent and child are made writable as needed. If the
        !          2008:  * child does not exist, then a new versioned resource is created and
        !          2009:  * checked out.
        !          2010:  *
        !          2011:  * If auto-versioning is not enabled for a versioned resource, then an error is
        !          2012:  * returned, since the resource cannot be modified.
        !          2013:  *
        !          2014:  * The dav_auto_version_info structure is filled in with enough information
        !          2015:  * to restore both parent and child resources to the state they were in
        !          2016:  * before the auto-versioning operations occurred.
        !          2017:  */
        !          2018: DAV_DECLARE(dav_error *) dav_auto_checkout(
        !          2019:     request_rec *r,
        !          2020:     dav_resource *resource,
        !          2021:     int parent_only,
        !          2022:     dav_auto_version_info *av_info);
        !          2023: 
        !          2024: /* Revert the writability of resources back to what they were
        !          2025:  * before they were modified. If undo == 0, then the resource
        !          2026:  * modifications are maintained (i.e. they are checked in).
        !          2027:  * If undo != 0, then resource modifications are discarded
        !          2028:  * (i.e. they are unchecked out).
        !          2029:  *
        !          2030:  * Set the unlock flag to indicate that the resource is about
        !          2031:  * to be unlocked; it will be checked in if the resource
        !          2032:  * auto-versioning property indicates it should be. In this case,
        !          2033:  * av_info is ignored, so it can be NULL.
        !          2034:  *
        !          2035:  * The resource argument may be NULL if only the parent resource
        !          2036:  * was checked out (i.e. the parent_only was != 0 in the
        !          2037:  * dav_auto_checkout call).
        !          2038:  */
        !          2039: DAV_DECLARE(dav_error *) dav_auto_checkin(
        !          2040:     request_rec *r,
        !          2041:     dav_resource *resource,
        !          2042:     int undo,
        !          2043:     int unlock,
        !          2044:     dav_auto_version_info *av_info);
        !          2045: 
        !          2046: /*
        !          2047: ** This structure is used to describe available reports
        !          2048: **
        !          2049: ** "nmspace" should be valid XML and URL-quoted. mod_dav will place
        !          2050: ** double-quotes around it and use it in an xmlns declaration.
        !          2051: */
        !          2052: typedef struct {
        !          2053:     const char *nmspace;        /* namespace of the XML report element */
        !          2054:     const char *name;           /* element name for the XML report */
        !          2055: } dav_report_elem;
        !          2056: 
        !          2057: 
        !          2058: /* Versioning provider hooks */
        !          2059: struct dav_hooks_vsn
        !          2060: {
        !          2061:     /*
        !          2062:     ** MANDATORY HOOKS
        !          2063:     ** The following hooks are mandatory for all versioning providers;
        !          2064:     ** they define the functionality needed to implement "core" versioning.
        !          2065:     */
        !          2066: 
        !          2067:     /* Return supported versioning options.
        !          2068:      * Each dav_text item in the list will be returned as a separate
        !          2069:      * DAV header. Providers are advised to limit the length of an
        !          2070:      * individual text item to 63 characters, to conform to the limit
        !          2071:      * used by MS Web Folders.
        !          2072:      */
        !          2073:     void (*get_vsn_options)(apr_pool_t *p, apr_text_header *phdr);
        !          2074: 
        !          2075:     /* Get the value of a specific option for an OPTIONS request.
        !          2076:      * The option being requested is given by the parsed XML
        !          2077:      * element object "elem". The value of the option should be
        !          2078:      * appended to the "option" text object.
        !          2079:      */
        !          2080:     dav_error * (*get_option)(const dav_resource *resource,
        !          2081:                               const apr_xml_elem *elem,
        !          2082:                               apr_text_header *option);
        !          2083: 
        !          2084:     /* Determine whether a non-versioned (or non-existent) resource
        !          2085:      * is versionable. Returns != 0 if resource can be versioned.
        !          2086:      */
        !          2087:     int (*versionable)(const dav_resource *resource);
        !          2088: 
        !          2089:     /* Determine whether auto-versioning is enabled for a resource
        !          2090:      * (which may not exist, or may not be versioned). If the resource
        !          2091:      * is a checked-out resource, the provider must only enable
        !          2092:      * auto-checkin if the resource was automatically checked out.
        !          2093:      *
        !          2094:      * The value returned depends on both the state of the resource
        !          2095:      * and the value of its DAV:auto-version property. See the description
        !          2096:      * of the dav_auto_version enumeration above for the details.
        !          2097:      */
        !          2098:     dav_auto_version (*auto_versionable)(const dav_resource *resource);
        !          2099: 
        !          2100:     /* Put a resource under version control. If the resource already
        !          2101:      * exists unversioned, then it becomes the initial version of the
        !          2102:      * new version history, and it is replaced by a version selector
        !          2103:      * which targets the new version.
        !          2104:      *
        !          2105:      * If the resource does not exist, then a new version-controlled
        !          2106:      * resource is created which either targets an existing version (if the
        !          2107:      * "target" argument is not NULL), or the initial, empty version
        !          2108:      * in a new history resource (if the "target" argument is NULL).
        !          2109:      *
        !          2110:      * If successful, the resource object state is updated appropriately
        !          2111:      * (that is, changed to refer to the new version-controlled resource).
        !          2112:      */
        !          2113:     dav_error * (*vsn_control)(dav_resource *resource,
        !          2114:                                const char *target);
        !          2115: 
        !          2116:     /* Checkout a resource. If successful, the resource
        !          2117:      * object state is updated appropriately.
        !          2118:      *
        !          2119:      * The auto_checkout flag will be set if this checkout is being
        !          2120:      * done automatically, as part of some method which modifies
        !          2121:      * the resource. The provider must remember that the resource
        !          2122:      * was automatically checked out, so it can determine whether it
        !          2123:      * can be automatically checked in. (Auto-checkin should only be
        !          2124:      * enabled for resources which were automatically checked out.)
        !          2125:      *
        !          2126:      * If the working resource has a different URL from the
        !          2127:      * target resource, a dav_resource descriptor is returned
        !          2128:      * for the new working resource. Otherwise, the original
        !          2129:      * resource descriptor will refer to the working resource.
        !          2130:      * The working_resource argument can be NULL if the caller
        !          2131:      * is not interested in the working resource.
        !          2132:      *
        !          2133:      * If the client has specified DAV:unreserved or DAV:fork-ok in the
        !          2134:      * checkout request, then the corresponding flags are set. If
        !          2135:      * DAV:activity-set has been specified, then create_activity is set
        !          2136:      * if DAV:new was specified; otherwise, the DAV:href elements' CDATA
        !          2137:      * (the actual href text) is passed in the "activities" array (each
        !          2138:      * element of the array is a const char *). activities will be NULL
        !          2139:      * no DAV:activity-set was provided or when create_activity is set.
        !          2140:      */
        !          2141:     dav_error * (*checkout)(dav_resource *resource,
        !          2142:                             int auto_checkout,
        !          2143:                             int is_unreserved, int is_fork_ok,
        !          2144:                             int create_activity,
        !          2145:                             apr_array_header_t *activities,
        !          2146:                             dav_resource **working_resource);
        !          2147: 
        !          2148:     /* Uncheckout a checked-out resource. If successful, the resource
        !          2149:      * object state is updated appropriately.
        !          2150:      */
        !          2151:     dav_error * (*uncheckout)(dav_resource *resource);
        !          2152: 
        !          2153:     /* Checkin a checked-out resource. If successful, the resource
        !          2154:      * object state is updated appropriately, and the
        !          2155:      * version_resource descriptor will refer to the new version.
        !          2156:      * The version_resource argument can be NULL if the caller
        !          2157:      * is not interested in the new version resource.
        !          2158:      *
        !          2159:      * If the client has specified DAV:keep-checked-out in the checkin
        !          2160:      * request, then the keep_checked_out flag is set. The provider
        !          2161:      * should create a new version, but keep the resource in the
        !          2162:      * checked-out state.
        !          2163:      */
        !          2164:     dav_error * (*checkin)(dav_resource *resource,
        !          2165:                            int keep_checked_out,
        !          2166:                            dav_resource **version_resource);
        !          2167: 
        !          2168:     /*
        !          2169:     ** Return the set of reports available at this resource.
        !          2170:     **
        !          2171:     ** An array of report elements should be returned, with an end-marker
        !          2172:     ** element containing namespace==NULL. The value of the
        !          2173:     ** DAV:supported-report-set property will be constructed and
        !          2174:     ** returned.
        !          2175:     */
        !          2176:     dav_error * (*avail_reports)(const dav_resource *resource,
        !          2177:                                  const dav_report_elem **reports);
        !          2178: 
        !          2179:     /*
        !          2180:     ** Determine whether a Label header can be used
        !          2181:     ** with a particular report. The dav_xml_doc structure
        !          2182:     ** contains the parsed report request body.
        !          2183:     ** Returns 0 if the Label header is not allowed.
        !          2184:     */
        !          2185:     int (*report_label_header_allowed)(const apr_xml_doc *doc);
        !          2186: 
        !          2187:     /*
        !          2188:     ** Generate a report on a resource. Since a provider is free
        !          2189:     ** to define its own reports, and the value of request headers
        !          2190:     ** may affect the interpretation of a report, the request record
        !          2191:     ** must be passed to this routine.
        !          2192:     **
        !          2193:     ** The dav_xml_doc structure contains the parsed report request
        !          2194:     ** body. The report response should be generated into the specified
        !          2195:     ** output filter.
        !          2196:     **
        !          2197:     ** If an error occurs, and a response has not yet been generated,
        !          2198:     ** then an error can be returned from this function. mod_dav will
        !          2199:     ** construct an appropriate error response. Once some output has
        !          2200:     ** been placed into the filter, however, the provider should not
        !          2201:     ** return an error -- there is no way that mod_dav can deliver it
        !          2202:     ** properly.
        !          2203:     **
        !          2204:     ** ### maybe we need a way to signal an error anyways, and then
        !          2205:     ** ### apache can abort the connection?
        !          2206:     */
        !          2207:     dav_error * (*deliver_report)(request_rec *r,
        !          2208:                                   const dav_resource *resource,
        !          2209:                                   const apr_xml_doc *doc,
        !          2210:                                   ap_filter_t *output);
        !          2211: 
        !          2212:     /*
        !          2213:     ** OPTIONAL HOOKS
        !          2214:     ** The following hooks are optional; if not defined, then the
        !          2215:     ** corresponding protocol methods will be unsupported.
        !          2216:     */
        !          2217: 
        !          2218:     /*
        !          2219:     ** Set the state of a checked-in version-controlled resource.
        !          2220:     **
        !          2221:     ** If the request specified a version, the version resource
        !          2222:     ** represents that version. If the request specified a label,
        !          2223:     ** then "version" is NULL, and "label" is the label.
        !          2224:     **
        !          2225:     ** The depth argument is ignored for a file, and can be 0, 1, or
        !          2226:     ** DAV_INFINITY for a collection. The depth argument only applies
        !          2227:     ** with a label, not a version.
        !          2228:     **
        !          2229:     ** If an error occurs in a child resource, then the return value is
        !          2230:     ** non-NULL, and *response is set to a multistatus response.
        !          2231:     **
        !          2232:     ** This hook is optional; if not defined, then the UPDATE method
        !          2233:     ** will not be supported.
        !          2234:     */
        !          2235:     dav_error * (*update)(const dav_resource *resource,
        !          2236:                           const dav_resource *version,
        !          2237:                           const char *label,
        !          2238:                           int depth,
        !          2239:                           dav_response **response);
        !          2240: 
        !          2241:     /*
        !          2242:     ** Add a label to a version. The resource is either a specific
        !          2243:     ** version, or a version selector, in which case the label should
        !          2244:     ** be added to the current target of the version selector. The
        !          2245:     ** version selector cannot be checked out.
        !          2246:     **
        !          2247:     ** If replace != 0, any existing label by the same name is
        !          2248:     ** effectively deleted first. Otherwise, it is an error to
        !          2249:     ** attempt to add a label which already exists on some version
        !          2250:     ** of the same history resource.
        !          2251:     **
        !          2252:     ** This hook is optional; if not defined, then the LABEL method
        !          2253:     ** will not be supported. If it is defined, then the remove_label
        !          2254:     ** hook must be defined also.
        !          2255:     */
        !          2256:     dav_error * (*add_label)(const dav_resource *resource,
        !          2257:                              const char *label,
        !          2258:                              int replace);
        !          2259: 
        !          2260:     /*
        !          2261:     ** Remove a label from a version. The resource is either a specific
        !          2262:     ** version, or a version selector, in which case the label should
        !          2263:     ** be added to the current target of the version selector. The
        !          2264:     ** version selector cannot be checked out.
        !          2265:     **
        !          2266:     ** It is an error if no such label exists on the specified version.
        !          2267:     **
        !          2268:     ** This hook is optional, but if defined, the add_label hook
        !          2269:     ** must be defined also.
        !          2270:     */
        !          2271:     dav_error * (*remove_label)(const dav_resource *resource,
        !          2272:                                 const char *label);
        !          2273: 
        !          2274:     /*
        !          2275:     ** Determine whether a null resource can be created as a workspace.
        !          2276:     ** The provider may restrict workspaces to certain locations.
        !          2277:     ** Returns 0 if the resource cannot be a workspace.
        !          2278:     **
        !          2279:     ** This hook is optional; if the provider does not support workspaces,
        !          2280:     ** it should be set to NULL.
        !          2281:     */
        !          2282:     int (*can_be_workspace)(const dav_resource *resource);
        !          2283: 
        !          2284:     /*
        !          2285:     ** Create a workspace resource. The resource must not already
        !          2286:     ** exist. Any <DAV:mkworkspace> element is passed to the provider
        !          2287:     ** in the "doc" structure; it may be empty.
        !          2288:     **
        !          2289:     ** If workspace creation is succesful, the state of the resource
        !          2290:     ** object is updated appropriately.
        !          2291:     **
        !          2292:     ** This hook is optional; if the provider does not support workspaces,
        !          2293:     ** it should be set to NULL.
        !          2294:     */
        !          2295:     dav_error * (*make_workspace)(dav_resource *resource,
        !          2296:                                   apr_xml_doc *doc);
        !          2297: 
        !          2298:     /*
        !          2299:     ** Determine whether a null resource can be created as an activity.
        !          2300:     ** The provider may restrict activities to certain locations.
        !          2301:     ** Returns 0 if the resource cannot be an activity.
        !          2302:     **
        !          2303:     ** This hook is optional; if the provider does not support activities,
        !          2304:     ** it should be set to NULL.
        !          2305:     */
        !          2306:     int (*can_be_activity)(const dav_resource *resource);
        !          2307: 
        !          2308:     /*
        !          2309:     ** Create an activity resource. The resource must not already
        !          2310:     ** exist.
        !          2311:     **
        !          2312:     ** If activity creation is succesful, the state of the resource
        !          2313:     ** object is updated appropriately.
        !          2314:     **
        !          2315:     ** This hook is optional; if the provider does not support activities,
        !          2316:     ** it should be set to NULL.
        !          2317:     */
        !          2318:     dav_error * (*make_activity)(dav_resource *resource);
        !          2319: 
        !          2320:     /*
        !          2321:     ** Merge a resource (tree) into target resource (tree).
        !          2322:     **
        !          2323:     ** ### more doc...
        !          2324:     **
        !          2325:     ** This hook is optional; if the provider does not support merging,
        !          2326:     ** then this should be set to NULL.
        !          2327:     */
        !          2328:     dav_error * (*merge)(dav_resource *target, dav_resource *source,
        !          2329:                          int no_auto_merge, int no_checkout,
        !          2330:                          apr_xml_elem *prop_elem,
        !          2331:                          ap_filter_t *output);
        !          2332: 
        !          2333:     /*
        !          2334:     ** If a provider needs a context to associate with this hooks structure,
        !          2335:     ** then this field may be used. In most cases, it will just be NULL.
        !          2336:     */
        !          2337:     void *ctx;
        !          2338: };
        !          2339: 
        !          2340: 
        !          2341: /* --------------------------------------------------------------------
        !          2342: **
        !          2343: ** BINDING FUNCTIONS
        !          2344: */
        !          2345: 
        !          2346: /* binding provider hooks */
        !          2347: struct dav_hooks_binding {
        !          2348: 
        !          2349:     /* Determine whether a resource can be the target of a binding.
        !          2350:      * Returns 0 if the resource cannot be a binding target.
        !          2351:      */
        !          2352:     int (*is_bindable)(const dav_resource *resource);
        !          2353: 
        !          2354:     /* Create a binding to a resource.
        !          2355:      * The resource argument is the target of the binding;
        !          2356:      * the binding argument must be a resource which does not already
        !          2357:      * exist.
        !          2358:      */
        !          2359:     dav_error * (*bind_resource)(const dav_resource *resource,
        !          2360:                                  dav_resource *binding);
        !          2361: 
        !          2362:     /*
        !          2363:     ** If a provider needs a context to associate with this hooks structure,
        !          2364:     ** then this field may be used. In most cases, it will just be NULL.
        !          2365:     */
        !          2366:     void *ctx;
        !          2367: 
        !          2368: };
        !          2369: 
        !          2370: 
        !          2371: /* --------------------------------------------------------------------
        !          2372: **
        !          2373: ** SEARCH(DASL) FUNCTIONS
        !          2374: */
        !          2375: 
        !          2376: /* search provider hooks */
        !          2377: struct dav_hooks_search {
        !          2378:     /* Set header for a OPTION method
        !          2379:      * An error may be returned.
        !          2380:      * To set a hadder, this function might call
        !          2381:      * apr_table_setn(r->headers_out, "DASL", dasl_optin1);
        !          2382:      *
        !          2383:      * Examples:
        !          2384:      * DASL: <DAV:basicsearch>
        !          2385:      * DASL: <http://foo.bar.com/syntax1>
        !          2386:      * DASL: <http://akuma.com/syntax2>
        !          2387:      */
        !          2388:     dav_error * (*set_option_head)(request_rec *r);
        !          2389: 
        !          2390:     /* Search resources
        !          2391:      * An error may be returned. *response will contain multistatus
        !          2392:      * responses (if any) suitable for the body of the error. It is also
        !          2393:      * possible to return NULL, yet still have multistatus responses.
        !          2394:      * In this case, typically the caller should return a 207 (Multistatus)
        !          2395:      * and the responses (in the body) as the HTTP response.
        !          2396:      */
        !          2397:     dav_error * (*search_resource)(request_rec *r,
        !          2398:                                    dav_response **response);
        !          2399: 
        !          2400:     /*
        !          2401:     ** If a provider needs a context to associate with this hooks structure,
        !          2402:     ** then this field may be used. In most cases, it will just be NULL.
        !          2403:     */
        !          2404:     void *ctx;
        !          2405: 
        !          2406: };
        !          2407: 
        !          2408: 
        !          2409: /* --------------------------------------------------------------------
        !          2410: **
        !          2411: ** MISCELLANEOUS STUFF
        !          2412: */
        !          2413: 
        !          2414: typedef struct {
        !          2415:     int propid;                          /* live property ID */
        !          2416:     const dav_hooks_liveprop *provider;  /* the provider defining this prop */
        !          2417: } dav_elem_private;    
        !          2418: 
        !          2419: #ifdef __cplusplus
        !          2420: }
        !          2421: #endif
        !          2422: 
        !          2423: #endif /* _MOD_DAV_H_ */
        !          2424: /** @} */
        !          2425: 

E-mail: