xref: /dflybsd-src/contrib/lvm2/dist/lib/format_text/archive.c (revision 86d7f5d305c6adaa56ff4582ece9859d73106103)
1*86d7f5d3SJohn Marino /*	$NetBSD: archive.c,v 1.1.1.3 2009/12/02 00:26:28 haad Exp $	*/
2*86d7f5d3SJohn Marino 
3*86d7f5d3SJohn Marino /*
4*86d7f5d3SJohn Marino  * Copyright (C) 2001-2004 Sistina Software, Inc. All rights reserved.
5*86d7f5d3SJohn Marino  * Copyright (C) 2004-2007 Red Hat, Inc. All rights reserved.
6*86d7f5d3SJohn Marino  *
7*86d7f5d3SJohn Marino  * This file is part of LVM2.
8*86d7f5d3SJohn Marino  *
9*86d7f5d3SJohn Marino  * This copyrighted material is made available to anyone wishing to use,
10*86d7f5d3SJohn Marino  * modify, copy, or redistribute it subject to the terms and conditions
11*86d7f5d3SJohn Marino  * of the GNU Lesser General Public License v.2.1.
12*86d7f5d3SJohn Marino  *
13*86d7f5d3SJohn Marino  * You should have received a copy of the GNU Lesser General Public License
14*86d7f5d3SJohn Marino  * along with this program; if not, write to the Free Software Foundation,
15*86d7f5d3SJohn Marino  * Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
16*86d7f5d3SJohn Marino  */
17*86d7f5d3SJohn Marino 
18*86d7f5d3SJohn Marino #include "lib.h"
19*86d7f5d3SJohn Marino #include "format-text.h"
20*86d7f5d3SJohn Marino 
21*86d7f5d3SJohn Marino #include "config.h"
22*86d7f5d3SJohn Marino #include "import-export.h"
23*86d7f5d3SJohn Marino #include "lvm-string.h"
24*86d7f5d3SJohn Marino #include "lvm-file.h"
25*86d7f5d3SJohn Marino #include "toolcontext.h"
26*86d7f5d3SJohn Marino 
27*86d7f5d3SJohn Marino #include <dirent.h>
28*86d7f5d3SJohn Marino #include <unistd.h>
29*86d7f5d3SJohn Marino #include <sys/stat.h>
30*86d7f5d3SJohn Marino #include <sys/file.h>
31*86d7f5d3SJohn Marino #include <fcntl.h>
32*86d7f5d3SJohn Marino #include <time.h>
33*86d7f5d3SJohn Marino 
34*86d7f5d3SJohn Marino #define SECS_PER_DAY 86400	/* 24*60*60 */
35*86d7f5d3SJohn Marino 
36*86d7f5d3SJohn Marino /*
37*86d7f5d3SJohn Marino  * The format instance is given a directory path upon creation.
38*86d7f5d3SJohn Marino  * Each file in this directory whose name is of the form
39*86d7f5d3SJohn Marino  * '(.*)_[0-9]*.vg' is a config file (see lib/config.[hc]), which
40*86d7f5d3SJohn Marino  * contains a description of a single volume group.
41*86d7f5d3SJohn Marino  *
42*86d7f5d3SJohn Marino  * The prefix ($1 from the above regex) of the config file gives
43*86d7f5d3SJohn Marino  * the volume group name.
44*86d7f5d3SJohn Marino  *
45*86d7f5d3SJohn Marino  * Backup files that have expired will be removed.
46*86d7f5d3SJohn Marino  */
47*86d7f5d3SJohn Marino 
48*86d7f5d3SJohn Marino /*
49*86d7f5d3SJohn Marino  * A list of these is built up for our volume group.  Ordered
50*86d7f5d3SJohn Marino  * with the least recent at the head.
51*86d7f5d3SJohn Marino  */
52*86d7f5d3SJohn Marino struct archive_file {
53*86d7f5d3SJohn Marino 	struct dm_list list;
54*86d7f5d3SJohn Marino 
55*86d7f5d3SJohn Marino 	char *path;
56*86d7f5d3SJohn Marino 	uint32_t index;
57*86d7f5d3SJohn Marino };
58*86d7f5d3SJohn Marino 
59*86d7f5d3SJohn Marino /*
60*86d7f5d3SJohn Marino  * Extract vg name and version number from a filename.
61*86d7f5d3SJohn Marino  */
_split_vg(const char * filename,char * vgname,size_t vgsize,uint32_t * ix)62*86d7f5d3SJohn Marino static int _split_vg(const char *filename, char *vgname, size_t vgsize,
63*86d7f5d3SJohn Marino 		     uint32_t *ix)
64*86d7f5d3SJohn Marino {
65*86d7f5d3SJohn Marino 	size_t len, vg_len;
66*86d7f5d3SJohn Marino 	const char *dot, *underscore;
67*86d7f5d3SJohn Marino 
68*86d7f5d3SJohn Marino 	len = strlen(filename);
69*86d7f5d3SJohn Marino 	if (len < 7)
70*86d7f5d3SJohn Marino 		return 0;
71*86d7f5d3SJohn Marino 
72*86d7f5d3SJohn Marino 	dot = (filename + len - 3);
73*86d7f5d3SJohn Marino 	if (strcmp(".vg", dot))
74*86d7f5d3SJohn Marino 		return 0;
75*86d7f5d3SJohn Marino 
76*86d7f5d3SJohn Marino 	if (!(underscore = strrchr(filename, '_')))
77*86d7f5d3SJohn Marino 		return 0;
78*86d7f5d3SJohn Marino 
79*86d7f5d3SJohn Marino 	if (sscanf(underscore + 1, "%u", ix) != 1)
80*86d7f5d3SJohn Marino 		return 0;
81*86d7f5d3SJohn Marino 
82*86d7f5d3SJohn Marino 	vg_len = underscore - filename;
83*86d7f5d3SJohn Marino 	if (vg_len + 1 > vgsize)
84*86d7f5d3SJohn Marino 		return 0;
85*86d7f5d3SJohn Marino 
86*86d7f5d3SJohn Marino 	strncpy(vgname, filename, vg_len);
87*86d7f5d3SJohn Marino 	vgname[vg_len] = '\0';
88*86d7f5d3SJohn Marino 
89*86d7f5d3SJohn Marino 	return 1;
90*86d7f5d3SJohn Marino }
91*86d7f5d3SJohn Marino 
_insert_archive_file(struct dm_list * head,struct archive_file * b)92*86d7f5d3SJohn Marino static void _insert_archive_file(struct dm_list *head, struct archive_file *b)
93*86d7f5d3SJohn Marino {
94*86d7f5d3SJohn Marino 	struct archive_file *bf = NULL;
95*86d7f5d3SJohn Marino 
96*86d7f5d3SJohn Marino 	if (dm_list_empty(head)) {
97*86d7f5d3SJohn Marino 		dm_list_add(head, &b->list);
98*86d7f5d3SJohn Marino 		return;
99*86d7f5d3SJohn Marino 	}
100*86d7f5d3SJohn Marino 
101*86d7f5d3SJohn Marino 	/* index reduces through list */
102*86d7f5d3SJohn Marino 	dm_list_iterate_items(bf, head) {
103*86d7f5d3SJohn Marino 		if (b->index > bf->index) {
104*86d7f5d3SJohn Marino 			dm_list_add(&bf->list, &b->list);
105*86d7f5d3SJohn Marino 			return;
106*86d7f5d3SJohn Marino 		}
107*86d7f5d3SJohn Marino 	}
108*86d7f5d3SJohn Marino 
109*86d7f5d3SJohn Marino 	dm_list_add_h(&bf->list, &b->list);
110*86d7f5d3SJohn Marino }
111*86d7f5d3SJohn Marino 
_join_file_to_dir(struct dm_pool * mem,const char * dir,const char * name)112*86d7f5d3SJohn Marino static char *_join_file_to_dir(struct dm_pool *mem, const char *dir, const char *name)
113*86d7f5d3SJohn Marino {
114*86d7f5d3SJohn Marino 	if (!dm_pool_begin_object(mem, 32) ||
115*86d7f5d3SJohn Marino 	    !dm_pool_grow_object(mem, dir, strlen(dir)) ||
116*86d7f5d3SJohn Marino 	    !dm_pool_grow_object(mem, "/", 1) ||
117*86d7f5d3SJohn Marino 	    !dm_pool_grow_object(mem, name, strlen(name)) ||
118*86d7f5d3SJohn Marino 	    !dm_pool_grow_object(mem, "\0", 1))
119*86d7f5d3SJohn Marino 		return_NULL;
120*86d7f5d3SJohn Marino 
121*86d7f5d3SJohn Marino 	return dm_pool_end_object(mem);
122*86d7f5d3SJohn Marino }
123*86d7f5d3SJohn Marino 
124*86d7f5d3SJohn Marino /*
125*86d7f5d3SJohn Marino  * Returns a list of archive_files.
126*86d7f5d3SJohn Marino  */
_scan_archive(struct dm_pool * mem,const char * vgname,const char * dir)127*86d7f5d3SJohn Marino static struct dm_list *_scan_archive(struct dm_pool *mem,
128*86d7f5d3SJohn Marino 				  const char *vgname, const char *dir)
129*86d7f5d3SJohn Marino {
130*86d7f5d3SJohn Marino 	int i, count;
131*86d7f5d3SJohn Marino 	uint32_t ix;
132*86d7f5d3SJohn Marino 	char vgname_found[64], *path;
133*86d7f5d3SJohn Marino 	struct dirent **dirent;
134*86d7f5d3SJohn Marino 	struct archive_file *af;
135*86d7f5d3SJohn Marino 	struct dm_list *results;
136*86d7f5d3SJohn Marino 
137*86d7f5d3SJohn Marino 	if (!(results = dm_pool_alloc(mem, sizeof(*results))))
138*86d7f5d3SJohn Marino 		return_NULL;
139*86d7f5d3SJohn Marino 
140*86d7f5d3SJohn Marino 	dm_list_init(results);
141*86d7f5d3SJohn Marino 
142*86d7f5d3SJohn Marino 	/* Sort fails beyond 5-digit indexes */
143*86d7f5d3SJohn Marino 	if ((count = scandir(dir, &dirent, NULL, alphasort)) < 0) {
144*86d7f5d3SJohn Marino 		log_error("Couldn't scan the archive directory (%s).", dir);
145*86d7f5d3SJohn Marino 		return 0;
146*86d7f5d3SJohn Marino 	}
147*86d7f5d3SJohn Marino 
148*86d7f5d3SJohn Marino 	for (i = 0; i < count; i++) {
149*86d7f5d3SJohn Marino 		if (!strcmp(dirent[i]->d_name, ".") ||
150*86d7f5d3SJohn Marino 		    !strcmp(dirent[i]->d_name, ".."))
151*86d7f5d3SJohn Marino 			continue;
152*86d7f5d3SJohn Marino 
153*86d7f5d3SJohn Marino 		/* check the name is the correct format */
154*86d7f5d3SJohn Marino 		if (!_split_vg(dirent[i]->d_name, vgname_found,
155*86d7f5d3SJohn Marino 			       sizeof(vgname_found), &ix))
156*86d7f5d3SJohn Marino 			continue;
157*86d7f5d3SJohn Marino 
158*86d7f5d3SJohn Marino 		/* is it the vg we're interested in ? */
159*86d7f5d3SJohn Marino 		if (strcmp(vgname, vgname_found))
160*86d7f5d3SJohn Marino 			continue;
161*86d7f5d3SJohn Marino 
162*86d7f5d3SJohn Marino 		if (!(path = _join_file_to_dir(mem, dir, dirent[i]->d_name)))
163*86d7f5d3SJohn Marino 			goto_out;
164*86d7f5d3SJohn Marino 
165*86d7f5d3SJohn Marino 		/*
166*86d7f5d3SJohn Marino 		 * Create a new archive_file.
167*86d7f5d3SJohn Marino 		 */
168*86d7f5d3SJohn Marino 		if (!(af = dm_pool_alloc(mem, sizeof(*af)))) {
169*86d7f5d3SJohn Marino 			log_error("Couldn't create new archive file.");
170*86d7f5d3SJohn Marino 			results = NULL;
171*86d7f5d3SJohn Marino 			goto out;
172*86d7f5d3SJohn Marino 		}
173*86d7f5d3SJohn Marino 
174*86d7f5d3SJohn Marino 		af->index = ix;
175*86d7f5d3SJohn Marino 		af->path = path;
176*86d7f5d3SJohn Marino 
177*86d7f5d3SJohn Marino 		/*
178*86d7f5d3SJohn Marino 		 * Insert it to the correct part of the list.
179*86d7f5d3SJohn Marino 		 */
180*86d7f5d3SJohn Marino 		_insert_archive_file(results, af);
181*86d7f5d3SJohn Marino 	}
182*86d7f5d3SJohn Marino 
183*86d7f5d3SJohn Marino       out:
184*86d7f5d3SJohn Marino 	for (i = 0; i < count; i++)
185*86d7f5d3SJohn Marino 		free(dirent[i]);
186*86d7f5d3SJohn Marino 	free(dirent);
187*86d7f5d3SJohn Marino 
188*86d7f5d3SJohn Marino 	return results;
189*86d7f5d3SJohn Marino }
190*86d7f5d3SJohn Marino 
_remove_expired(struct dm_list * archives,uint32_t archives_size,uint32_t retain_days,uint32_t min_archive)191*86d7f5d3SJohn Marino static void _remove_expired(struct dm_list *archives, uint32_t archives_size,
192*86d7f5d3SJohn Marino 			    uint32_t retain_days, uint32_t min_archive)
193*86d7f5d3SJohn Marino {
194*86d7f5d3SJohn Marino 	struct archive_file *bf;
195*86d7f5d3SJohn Marino 	struct stat sb;
196*86d7f5d3SJohn Marino 	time_t retain_time;
197*86d7f5d3SJohn Marino 
198*86d7f5d3SJohn Marino 	/* Make sure there are enough archives to even bother looking for
199*86d7f5d3SJohn Marino 	 * expired ones... */
200*86d7f5d3SJohn Marino 	if (archives_size <= min_archive)
201*86d7f5d3SJohn Marino 		return;
202*86d7f5d3SJohn Marino 
203*86d7f5d3SJohn Marino 	/* Convert retain_days into the time after which we must retain */
204*86d7f5d3SJohn Marino 	retain_time = time(NULL) - (time_t) retain_days *SECS_PER_DAY;
205*86d7f5d3SJohn Marino 
206*86d7f5d3SJohn Marino 	/* Assume list is ordered newest first (by index) */
207*86d7f5d3SJohn Marino 	dm_list_iterate_back_items(bf, archives) {
208*86d7f5d3SJohn Marino 		/* Get the mtime of the file and unlink if too old */
209*86d7f5d3SJohn Marino 		if (stat(bf->path, &sb)) {
210*86d7f5d3SJohn Marino 			log_sys_error("stat", bf->path);
211*86d7f5d3SJohn Marino 			continue;
212*86d7f5d3SJohn Marino 		}
213*86d7f5d3SJohn Marino 
214*86d7f5d3SJohn Marino 		if (sb.st_mtime > retain_time)
215*86d7f5d3SJohn Marino 			return;
216*86d7f5d3SJohn Marino 
217*86d7f5d3SJohn Marino 		log_very_verbose("Expiring archive %s", bf->path);
218*86d7f5d3SJohn Marino 		if (unlink(bf->path))
219*86d7f5d3SJohn Marino 			log_sys_error("unlink", bf->path);
220*86d7f5d3SJohn Marino 
221*86d7f5d3SJohn Marino 		/* Don't delete any more if we've reached the minimum */
222*86d7f5d3SJohn Marino 		if (--archives_size <= min_archive)
223*86d7f5d3SJohn Marino 			return;
224*86d7f5d3SJohn Marino 	}
225*86d7f5d3SJohn Marino }
226*86d7f5d3SJohn Marino 
archive_vg(struct volume_group * vg,const char * dir,const char * desc,uint32_t retain_days,uint32_t min_archive)227*86d7f5d3SJohn Marino int archive_vg(struct volume_group *vg,
228*86d7f5d3SJohn Marino 	       const char *dir, const char *desc,
229*86d7f5d3SJohn Marino 	       uint32_t retain_days, uint32_t min_archive)
230*86d7f5d3SJohn Marino {
231*86d7f5d3SJohn Marino 	int i, fd, renamed = 0;
232*86d7f5d3SJohn Marino 	uint32_t ix = 0;
233*86d7f5d3SJohn Marino 	struct archive_file *last;
234*86d7f5d3SJohn Marino 	FILE *fp = NULL;
235*86d7f5d3SJohn Marino 	char temp_file[PATH_MAX], archive_name[PATH_MAX];
236*86d7f5d3SJohn Marino 	struct dm_list *archives;
237*86d7f5d3SJohn Marino 
238*86d7f5d3SJohn Marino 	/*
239*86d7f5d3SJohn Marino 	 * Write the vg out to a temporary file.
240*86d7f5d3SJohn Marino 	 */
241*86d7f5d3SJohn Marino 	if (!create_temp_name(dir, temp_file, sizeof(temp_file), &fd,
242*86d7f5d3SJohn Marino 			      &vg->cmd->rand_seed)) {
243*86d7f5d3SJohn Marino 		log_error("Couldn't create temporary archive name.");
244*86d7f5d3SJohn Marino 		return 0;
245*86d7f5d3SJohn Marino 	}
246*86d7f5d3SJohn Marino 
247*86d7f5d3SJohn Marino 	if (!(fp = fdopen(fd, "w"))) {
248*86d7f5d3SJohn Marino 		log_error("Couldn't create FILE object for archive.");
249*86d7f5d3SJohn Marino 		if (close(fd))
250*86d7f5d3SJohn Marino 			log_sys_error("close", temp_file);
251*86d7f5d3SJohn Marino 		return 0;
252*86d7f5d3SJohn Marino 	}
253*86d7f5d3SJohn Marino 
254*86d7f5d3SJohn Marino 	if (!text_vg_export_file(vg, desc, fp)) {
255*86d7f5d3SJohn Marino 		if (fclose(fp))
256*86d7f5d3SJohn Marino 			log_sys_error("fclose", temp_file);
257*86d7f5d3SJohn Marino 		return_0;
258*86d7f5d3SJohn Marino 	}
259*86d7f5d3SJohn Marino 
260*86d7f5d3SJohn Marino 	if (lvm_fclose(fp, temp_file))
261*86d7f5d3SJohn Marino 		return_0; /* Leave file behind as evidence of failure */
262*86d7f5d3SJohn Marino 
263*86d7f5d3SJohn Marino 	/*
264*86d7f5d3SJohn Marino 	 * Now we want to rename this file to <vg>_index.vg.
265*86d7f5d3SJohn Marino 	 */
266*86d7f5d3SJohn Marino 	if (!(archives = _scan_archive(vg->cmd->mem, vg->name, dir)))
267*86d7f5d3SJohn Marino 		return_0;
268*86d7f5d3SJohn Marino 
269*86d7f5d3SJohn Marino 	if (dm_list_empty(archives))
270*86d7f5d3SJohn Marino 		ix = 0;
271*86d7f5d3SJohn Marino 	else {
272*86d7f5d3SJohn Marino 		last = dm_list_item(dm_list_first(archives), struct archive_file);
273*86d7f5d3SJohn Marino 		ix = last->index + 1;
274*86d7f5d3SJohn Marino 	}
275*86d7f5d3SJohn Marino 
276*86d7f5d3SJohn Marino 	for (i = 0; i < 10; i++) {
277*86d7f5d3SJohn Marino 		if (dm_snprintf(archive_name, sizeof(archive_name),
278*86d7f5d3SJohn Marino 				 "%s/%s_%05u.vg", dir, vg->name, ix) < 0) {
279*86d7f5d3SJohn Marino 			log_error("Archive file name too long.");
280*86d7f5d3SJohn Marino 			return 0;
281*86d7f5d3SJohn Marino 		}
282*86d7f5d3SJohn Marino 
283*86d7f5d3SJohn Marino 		if ((renamed = lvm_rename(temp_file, archive_name)))
284*86d7f5d3SJohn Marino 			break;
285*86d7f5d3SJohn Marino 
286*86d7f5d3SJohn Marino 		ix++;
287*86d7f5d3SJohn Marino 	}
288*86d7f5d3SJohn Marino 
289*86d7f5d3SJohn Marino 	if (!renamed)
290*86d7f5d3SJohn Marino 		log_error("Archive rename failed for %s", temp_file);
291*86d7f5d3SJohn Marino 
292*86d7f5d3SJohn Marino 	_remove_expired(archives, dm_list_size(archives) + renamed, retain_days,
293*86d7f5d3SJohn Marino 			min_archive);
294*86d7f5d3SJohn Marino 
295*86d7f5d3SJohn Marino 	return 1;
296*86d7f5d3SJohn Marino }
297*86d7f5d3SJohn Marino 
_display_archive(struct cmd_context * cmd,struct archive_file * af)298*86d7f5d3SJohn Marino static void _display_archive(struct cmd_context *cmd, struct archive_file *af)
299*86d7f5d3SJohn Marino {
300*86d7f5d3SJohn Marino 	struct volume_group *vg = NULL;
301*86d7f5d3SJohn Marino 	struct format_instance *tf;
302*86d7f5d3SJohn Marino 	time_t when;
303*86d7f5d3SJohn Marino 	char *desc;
304*86d7f5d3SJohn Marino 	void *context;
305*86d7f5d3SJohn Marino 
306*86d7f5d3SJohn Marino 	log_print(" ");
307*86d7f5d3SJohn Marino 	log_print("File:\t\t%s", af->path);
308*86d7f5d3SJohn Marino 
309*86d7f5d3SJohn Marino 	if (!(context = create_text_context(cmd, af->path, NULL)) ||
310*86d7f5d3SJohn Marino 	    !(tf = cmd->fmt_backup->ops->create_instance(cmd->fmt_backup, NULL,
311*86d7f5d3SJohn Marino 							 NULL, context))) {
312*86d7f5d3SJohn Marino 		log_error("Couldn't create text instance object.");
313*86d7f5d3SJohn Marino 		return;
314*86d7f5d3SJohn Marino 	}
315*86d7f5d3SJohn Marino 
316*86d7f5d3SJohn Marino 	/*
317*86d7f5d3SJohn Marino 	 * Read the archive file to ensure that it is valid, and
318*86d7f5d3SJohn Marino 	 * retrieve the archive time and description.
319*86d7f5d3SJohn Marino 	 */
320*86d7f5d3SJohn Marino 	/* FIXME Use variation on _vg_read */
321*86d7f5d3SJohn Marino 	if (!(vg = text_vg_import_file(tf, af->path, &when, &desc))) {
322*86d7f5d3SJohn Marino 		log_print("Unable to read archive file.");
323*86d7f5d3SJohn Marino 		tf->fmt->ops->destroy_instance(tf);
324*86d7f5d3SJohn Marino 		return;
325*86d7f5d3SJohn Marino 	}
326*86d7f5d3SJohn Marino 
327*86d7f5d3SJohn Marino 	log_print("VG name:    \t%s", vg->name);
328*86d7f5d3SJohn Marino 	log_print("Description:\t%s", desc ? : "<No description>");
329*86d7f5d3SJohn Marino 	log_print("Backup Time:\t%s", ctime(&when));
330*86d7f5d3SJohn Marino 
331*86d7f5d3SJohn Marino 	vg_release(vg);
332*86d7f5d3SJohn Marino 	tf->fmt->ops->destroy_instance(tf);
333*86d7f5d3SJohn Marino }
334*86d7f5d3SJohn Marino 
archive_list(struct cmd_context * cmd,const char * dir,const char * vgname)335*86d7f5d3SJohn Marino int archive_list(struct cmd_context *cmd, const char *dir, const char *vgname)
336*86d7f5d3SJohn Marino {
337*86d7f5d3SJohn Marino 	struct dm_list *archives;
338*86d7f5d3SJohn Marino 	struct archive_file *af;
339*86d7f5d3SJohn Marino 
340*86d7f5d3SJohn Marino 	if (!(archives = _scan_archive(cmd->mem, vgname, dir)))
341*86d7f5d3SJohn Marino 		return_0;
342*86d7f5d3SJohn Marino 
343*86d7f5d3SJohn Marino 	if (dm_list_empty(archives))
344*86d7f5d3SJohn Marino 		log_print("No archives found in %s.", dir);
345*86d7f5d3SJohn Marino 
346*86d7f5d3SJohn Marino 	dm_list_iterate_back_items(af, archives)
347*86d7f5d3SJohn Marino 		_display_archive(cmd, af);
348*86d7f5d3SJohn Marino 
349*86d7f5d3SJohn Marino 	dm_pool_free(cmd->mem, archives);
350*86d7f5d3SJohn Marino 
351*86d7f5d3SJohn Marino 	return 1;
352*86d7f5d3SJohn Marino }
353*86d7f5d3SJohn Marino 
archive_list_file(struct cmd_context * cmd,const char * file)354*86d7f5d3SJohn Marino int archive_list_file(struct cmd_context *cmd, const char *file)
355*86d7f5d3SJohn Marino {
356*86d7f5d3SJohn Marino 	struct archive_file af;
357*86d7f5d3SJohn Marino 
358*86d7f5d3SJohn Marino 	af.path = (char *)file;
359*86d7f5d3SJohn Marino 
360*86d7f5d3SJohn Marino 	if (!path_exists(af.path)) {
361*86d7f5d3SJohn Marino 		log_error("Archive file %s not found.", af.path);
362*86d7f5d3SJohn Marino 		return 0;
363*86d7f5d3SJohn Marino 	}
364*86d7f5d3SJohn Marino 
365*86d7f5d3SJohn Marino 	_display_archive(cmd, &af);
366*86d7f5d3SJohn Marino 
367*86d7f5d3SJohn Marino 	return 1;
368*86d7f5d3SJohn Marino }
369*86d7f5d3SJohn Marino 
backup_list(struct cmd_context * cmd,const char * dir,const char * vgname)370*86d7f5d3SJohn Marino int backup_list(struct cmd_context *cmd, const char *dir, const char *vgname)
371*86d7f5d3SJohn Marino {
372*86d7f5d3SJohn Marino 	struct archive_file af;
373*86d7f5d3SJohn Marino 
374*86d7f5d3SJohn Marino 	if (!(af.path = _join_file_to_dir(cmd->mem, dir, vgname)))
375*86d7f5d3SJohn Marino 		return_0;
376*86d7f5d3SJohn Marino 
377*86d7f5d3SJohn Marino 	if (path_exists(af.path))
378*86d7f5d3SJohn Marino 		_display_archive(cmd, &af);
379*86d7f5d3SJohn Marino 
380*86d7f5d3SJohn Marino 	return 1;
381*86d7f5d3SJohn Marino }
382