# HG changeset patch
# User Olaf Wintermann <olaf.wintermann@gmail.com>
# Date 1404395413 -7200
# Node ID c5759ac76c1b7216706ea1ef2aa6ce5d0368eb2d
# Parent  08d5544c92fb3ad2c28b12614608d21a58129334
dav-sync detects locally removed files

diff -r 08d5544c92fb -r c5759ac76c1b dav/db.c
--- a/dav/db.c	Sun Jun 15 20:12:48 2014 +0200
+++ b/dav/db.c	Thu Jul 03 15:50:13 2014 +0200
@@ -46,19 +46,23 @@
 #define ENV_HOME getenv("HOME")
 #endif /* _WIN32 */
 
-UcxMap* load_db(char *name) {
+SyncDatabase* load_db(char *name) {
     char *dav_dir = util_concat_path(ENV_HOME, ".dav");
     char *db_file = util_concat_path(dav_dir, name);
     free(dav_dir);
     
+    SyncDatabase *db = malloc(sizeof(SyncDatabase));
     xmlTextReaderPtr reader = xmlReaderForFile(db_file, NULL, 0);
     if(!reader) {
         xmlDoc *doc =  doc = xmlNewDoc(BAD_CAST "1.0");
         xmlNode *root = xmlNewNode(NULL, BAD_CAST "directory");
         xmlDocSetRootElement(doc, root);
-        UcxMap *db = NULL;
         if(xmlSaveFormatFileEnc(db_file, doc, "UTF-8", 1) != -1) {
-            db = ucx_map_new(2048);
+            db->resources = ucx_map_new(2048);
+            db->remove = ucx_map_new(8);
+        } else {
+            free(db);
+            db = NULL;
         }
         xmlFreeDoc(doc);
         free(db_file);
@@ -66,7 +70,8 @@
     }
     free(db_file);
     
-    UcxMap *map = ucx_map_new(2048);
+    db->resources = ucx_map_new(2048);
+    db->remove = ucx_map_new(16);
     int error = 0;
     while(xmlTextReaderRead(reader)) {
         int type = xmlTextReaderNodeType(reader);
@@ -76,7 +81,15 @@
             if(xstreq(name, "resource")) {
                 LocalResource *res = process_resource(reader);
                 if(res) {
-                    ucx_map_cstr_put(map, res->path, res);
+                    ucx_map_cstr_put(db->resources, res->path, res);
+                } else {
+                    error = 1;
+                    break;
+                }
+            } else if(xstreq(name, "remove")) {
+                LocalResource *res = process_remove(reader);
+                if(res) {
+                    ucx_map_cstr_put(db->remove, res->path, res);
                 } else {
                     error = 1;
                     break;
@@ -90,7 +103,7 @@
         // TODO: free resources and map
         return NULL;
     } else {
-        return map;
+        return db;
     }
 }
 
@@ -160,7 +173,39 @@
     }
 }
 
-int store_db(UcxMap *db, char *name) {
+LocalResource* process_remove(xmlTextReaderPtr reader) {
+    LocalResource *res = calloc(1, sizeof(LocalResource));
+    
+    int path = 0;
+    while(xmlTextReaderRead(reader)) {
+        int type = xmlTextReaderNodeType(reader);
+        const xmlChar *name = xmlTextReaderConstName(reader);
+        
+        if(type == XML_READER_TYPE_ELEMENT) {
+            if(xstreq(name, "path")) {
+                path = 1;
+            }
+        } else if(type == XML_READER_TYPE_TEXT && path) {
+            const xmlChar *value = xmlTextReaderConstValue(reader);
+            res->path = strdup((char*)value);
+        } else if(XML_READER_TYPE_END_ELEMENT) {
+            if(xstreq(name, "remove")) {
+                break;
+            } else {
+                path = 0;
+            }
+        }
+    }
+    
+    if(!res->path) {
+        // TODO: free res
+        return NULL;
+    } else {
+        return res;
+    }
+}
+
+int store_db(SyncDatabase *db, char *name) {
     // open writer
     char *dav_dir = util_concat_path(ENV_HOME, ".dav");
     char *db_file = util_concat_path(dav_dir, name);
@@ -183,7 +228,7 @@
     xmlTextWriterStartElement(writer, BAD_CAST "directory");
     
     // write all resource entries
-    UcxMapIterator i = ucx_map_iterator(db);
+    UcxMapIterator i = ucx_map_iterator(db->resources);
     LocalResource *res;
     UCX_MAP_FOREACH(key, res, i) {
         // <resource>
@@ -237,6 +282,21 @@
         xmlTextWriterEndElement(writer);
     }
     
+    // write all remove entries
+    i = ucx_map_iterator(db->remove);
+    UCX_MAP_FOREACH(key, res, i) {
+        // <remove>
+        xmlTextWriterStartElement(writer, BAD_CAST "remove");
+        
+        xmlTextWriterWriteElement(
+                    writer,
+                    BAD_CAST "path",
+                    BAD_CAST res->path);
+        
+        // </remove>
+        xmlTextWriterEndElement(writer);
+    }
+    
     // end
     xmlTextWriterEndElement(writer);
     r = xmlTextWriterEndDocument(writer);
diff -r 08d5544c92fb -r c5759ac76c1b dav/db.h
--- a/dav/db.h	Sun Jun 15 20:12:48 2014 +0200
+++ b/dav/db.h	Thu Jul 03 15:50:13 2014 +0200
@@ -39,7 +39,8 @@
 extern "C" {
 #endif
 
-typedef struct LocalResource LocalResource;    
+typedef struct LocalResource LocalResource;   
+typedef struct SyncDatabase  SyncDatabase;
 
 struct LocalResource {
     char    *name;
@@ -49,10 +50,16 @@
     off_t  size;
 };
 
-UcxMap* load_db(char *name);
-int store_db(UcxMap *db, char *name);
+struct SyncDatabase {
+    UcxMap *resources;
+    UcxMap *remove;
+};
+
+SyncDatabase* load_db(char *name);
+int store_db(SyncDatabase *db, char *name);
 
 LocalResource* process_resource(xmlTextReaderPtr reader);
+LocalResource* process_remove(xmlTextReaderPtr reader);
 
 
 #ifdef	__cplusplus
diff -r 08d5544c92fb -r c5759ac76c1b dav/sync.c
--- a/dav/sync.c	Sun Jun 15 20:12:48 2014 +0200
+++ b/dav/sync.c	Thu Jul 03 15:50:13 2014 +0200
@@ -109,7 +109,7 @@
         return -1;
     }
     
-    UcxMap *db = load_db(dir->database);
+    SyncDatabase *db = load_db(dir->database);
     if(!db) {
         fprintf(stderr, "Cannot load database file: %s\n", dir->database);
         return -1;
@@ -134,12 +134,12 @@
         // TODO: free
         return 0; // empty repository
     }
-    
+      
     UcxList *stack = ucx_list_prepend(NULL, ls->children);
     while(stack) {
         DavResource *res = stack->data;
         stack = ucx_list_remove(stack, stack);
-        
+         
         while(res) {
             if(sync_get_resource(dir, res, db)) {
                 fprintf(stderr, "sync_get_resource failed for resource: %s\n", res->path);
@@ -163,10 +163,28 @@
     return 0;
 }
 
-int sync_get_resource(SyncDirectory *dir, DavResource *res, UcxMap *db) {
-    LocalResource *local = ucx_map_cstr_get(db, res->path);
+int sync_get_resource(SyncDirectory *dir, DavResource *res, SyncDatabase *db) {
+    LocalResource *local = ucx_map_cstr_get(db->resources, res->path);
+    char *local_path = util_concat_path(dir->path, res->path);
+    
     char *etag = dav_get_property(res, "D:getetag");
+    struct stat s;
     if(local) {
+        if(stat(local_path, &s)) {
+            if(errno == ENOENT) {
+                printf("removed %s\n", res->path);
+                // the file is in the database, but doesn't exists
+                // mark the file as removed to delete it on next push
+                ucx_map_cstr_remove(db->resources, local->path);
+                ucx_map_cstr_put(db->remove, local->path, local);
+                return 0;
+            } else {
+                fprintf(stderr, "stat failed: %s\n", local_path);
+                free(local_path);
+                return -1;
+            }
+        }
+        
         if(local->etag) {
             sstr_t e = sstr(etag);
             if(sstrprefix(e, S("W/"))) {
@@ -179,7 +197,6 @@
         }
     }
     
-    char *local_path = util_concat_path(dir->path, res->path);
     int ret = 0;
     if(res->iscollection) {
         mode_t mode = S_IRWXU | S_IRGRP | S_IXGRP | S_IROTH | S_IXOTH;
@@ -201,22 +218,17 @@
         fclose(out);
         
         if(ret == 0) {
-            // get file informations for db update
-            struct stat s;
-            if(stat(local_path, &s)) {
-                fprintf(stderr, "stat failed: %s\n", local_path);
-                ret = -1;
-            }
-            
             if(!local) {
+                // new local resource
                 local = calloc(1, sizeof(LocalResource));
                 local->path = strdup(res->path);
-                ucx_map_cstr_put(db, local->path, local);
+                ucx_map_cstr_put(db->resources, local->path, local);
             }
             
             if(local->etag) {
                 free(local->etag);
             }
+            // set metadata from stat
             local->etag = etag;
             local->last_modified = s.st_mtim.tv_sec;
             local->size = s.st_size;
@@ -245,7 +257,7 @@
         return -1;
     }
     
-    UcxMap *db = load_db(dir->database);
+    SyncDatabase *db = load_db(dir->database);
     if(!db) {
         fprintf(stderr, "Cannot load database file: %s\n", dir->database);
         return -1;
@@ -279,7 +291,7 @@
     return 0;
 }
 
-UcxList* local_scan(SyncDirectory *dir, UcxMap *db) {
+UcxList* local_scan(SyncDirectory *dir, SyncDatabase *db) {
     UcxList *resources = NULL;
     
     char *path = strdup("/");
@@ -322,7 +334,9 @@
                 if(S_ISDIR(s.st_mode)) {
                     stack = ucx_list_prepend(stack, new_path);
                 } else {
-                    LocalResource *res = ucx_map_cstr_get(db, new_path);
+                    LocalResource *res = ucx_map_cstr_get(
+                            db->resources,
+                            new_path);
                     if(res) {
                         // the file is already in the database
                         // compare length and lastmodified date
@@ -349,7 +363,7 @@
                         res->etag = NULL;
                         res->last_modified = s.st_mtim.tv_sec;
                         res->size = s.st_size;
-                        ucx_map_cstr_put(db, res->path, res);
+                        ucx_map_cstr_put(db->resources, res->path, res);
                         resources = ucx_list_append(resources, new_path);
                     }
                 }
@@ -364,7 +378,7 @@
     return resources;
 }
 
-int sync_put_resource(SyncDirectory *dir, DavResource *res, UcxMap *db) {
+int sync_put_resource(SyncDirectory *dir, DavResource *res, SyncDatabase *db) {
     char *local_path = util_concat_path(dir->path, res->path);
     FILE *in = fopen(local_path, "r");
     if(!in) {
@@ -389,25 +403,22 @@
     }
     
     if(ret == 0) {
-        LocalResource *local_res = ucx_map_cstr_get(db, res->path);
+        LocalResource *local_res = ucx_map_cstr_get(db->resources, res->path);
         if(local_res->etag) {
             free(local_res->etag);
         }
         
         DavResource *up_res = dav_get(res->session, res->path, "D:getetag");
-        char *etag_str = dav_get_property(up_res, "D:getetag");
-        sstr_t etag;
-        etag.ptr = NULL;
-        if(etag_str) {
-            etag = sstr(etag_str);
-        }
-        if(sstrprefix(etag, S("W/"))) {
-            etag = sstrsubs(etag, 2);
+        char *etag = dav_get_property(up_res, "D:getetag");
+        if(etag) {
+            if(strlen(etag) > 2 && etag[0] == 'W' && etag[1] == '/') {
+                etag = etag + 2;
+            } 
         }
         
         
-        if(etag.ptr) {
-            local_res->etag = strdup(etag.ptr);
+        if(etag) {
+            local_res->etag = strdup(etag);
         } else {
             local_res->etag = NULL;
         }
diff -r 08d5544c92fb -r c5759ac76c1b dav/sync.h
--- a/dav/sync.h	Sun Jun 15 20:12:48 2014 +0200
+++ b/dav/sync.h	Thu Jul 03 15:50:13 2014 +0200
@@ -45,11 +45,11 @@
 int cmd_push(CmdArgs *args);
 int cmd_sync(CmdArgs *args);
 
-int sync_get_resource(SyncDirectory *dir, DavResource *res, UcxMap *db);
+int sync_get_resource(SyncDirectory *dir, DavResource *res, SyncDatabase *db);
 
-UcxList* local_scan(SyncDirectory *dir, UcxMap *db);
+UcxList* local_scan(SyncDirectory *dir, SyncDatabase *db);
 
-int sync_put_resource(SyncDirectory *dir, DavResource *res, UcxMap *db);
+int sync_put_resource(SyncDirectory *dir, DavResource *res, SyncDatabase *db);
 
 
 #ifdef	__cplusplus