notmuch restore was skipping entries with no new tags
[notmuch] / notmuch-restore.c
1 /* notmuch - Not much of an email program, (just index and search)
2  *
3  * Copyright © 2009 Carl Worth
4  *
5  * This program is free software: you can redistribute it and/or modify
6  * it under the terms of the GNU General Public License as published by
7  * the Free Software Foundation, either version 3 of the License, or
8  * (at your option) any later version.
9  *
10  * This program is distributed in the hope that it will be useful,
11  * but WITHOUT ANY WARRANTY; without even the implied warranty of
12  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
13  * GNU General Public License for more details.
14  *
15  * You should have received a copy of the GNU General Public License
16  * along with this program.  If not, see http://www.gnu.org/licenses/ .
17  *
18  * Author: Carl Worth <cworth@cworth.org>
19  */
20
21 #include "notmuch-client.h"
22
23 int
24 notmuch_restore_command (unused (void *ctx), int argc, char *argv[])
25 {
26     notmuch_config_t *config;
27     notmuch_database_t *notmuch;
28     FILE *input;
29     char *line = NULL;
30     size_t line_size;
31     ssize_t line_len;
32     regex_t regex;
33     int rerr;
34
35     config = notmuch_config_open (ctx, NULL, NULL);
36     if (config == NULL)
37         return 1;
38
39     notmuch = notmuch_database_open (notmuch_config_get_database_path (config));
40     if (notmuch == NULL)
41         return 1;
42
43     if (argc) {
44         input = fopen (argv[0], "r");
45         if (input == NULL) {
46             fprintf (stderr, "Error opening %s for reading: %s\n",
47                      argv[0], strerror (errno));
48             return 1;
49         }
50     } else {
51         printf ("No filename given. Reading dump from stdin.\n");
52         input = stdin;
53     }
54
55     /* Dump output is one line per message. We match a sequence of
56      * non-space characters for the message-id, then one or more
57      * spaces, then a list of space-separated tags as a sequence of
58      * characters within literal '(' and ')'. */
59     xregcomp (&regex,
60               "^([^ ]+) \\(([^)]*)\\)$",
61               REG_EXTENDED);
62
63     while ((line_len = getline (&line, &line_size, input)) != -1) {
64         regmatch_t match[3];
65         char *message_id, *tags, *tag, *next;
66         notmuch_message_t *message;
67         notmuch_status_t status;
68
69         chomp_newline (line);
70
71         rerr = xregexec (&regex, line, 3, match, 0);
72         if (rerr == REG_NOMATCH)
73         {
74             fprintf (stderr, "Warning: Ignoring invalid input line: %s\n",
75                      line);
76             continue;
77         }
78
79         message_id = xstrndup (line + match[1].rm_so,
80                                match[1].rm_eo - match[1].rm_so);
81         tags = xstrndup (line + match[2].rm_so,
82                          match[2].rm_eo - match[2].rm_so);
83
84         message = notmuch_database_find_message (notmuch, message_id);
85         if (message == NULL) {
86             fprintf (stderr, "Warning: Cannot apply tags to missing message: %s\n",
87                      message_id);
88             goto NEXT_LINE;
89         }
90
91         notmuch_message_freeze (message);
92
93         notmuch_message_remove_all_tags (message);
94
95         next = tags;
96         while (next) {
97             tag = strsep (&next, " ");
98             if (*tag == '\0')
99                 continue;
100             status = notmuch_message_add_tag (message, tag);
101             if (status) {
102                 fprintf (stderr,
103                          "Error applying tag %s to message %s:\n",
104                          tag, message_id);
105                 fprintf (stderr, "%s\n",
106                          notmuch_status_to_string (status));
107             }
108         }
109
110         notmuch_message_thaw (message);
111         notmuch_message_destroy (message);
112       NEXT_LINE:
113         free (message_id);
114         free (tags);
115     }
116
117     regfree (&regex);
118
119     if (line)
120         free (line);
121
122     notmuch_database_close (notmuch);
123     if (input != stdin)
124         fclose (input);
125
126     return 0;
127 }