/* cmd-mv.c * **************************************************************** * Copyright (C) 2001, 2002, 2003 Tom Lord, Rob Weir * * See the file "COPYING" for further information about * the copyright and warranty status of this work. */ #include "config-options.h" #include "po/gettext.h" #include "hackerlab/cmd/main.h" #include "hackerlab/fs/file-names.h" #include "libarch/inv-ids.h" #include "commands/mv.h" #include "commands/version.h" static t_uchar * usage = N_("[options] from to OR from1 from2 ... dir"); #define OPTS(OP) \ OP (opt_help_msg, "h", "help", 0, \ N_("Display a help message and exit.")) \ OP (opt_long_help, "H", 0, 0, \ N_("Display a verbose help message and exit.")) \ OP (opt_version, "V", "version", 0, \ N_("Display a release identifier string\n" \ "and exit.")) t_uchar arch_cmd_mv_help[] = N_("move a file (or dir, or symlink) and its explicit inventory tag (if any)\n" "This command is suitable as a replacement for mv(1) within any baz\n" "tree for any move operation, regardless of id tagging method.\n"); enum options { OPTS (OPT_ENUM) }; static struct opt_desc opts[] = { OPTS (OPT_DESC) {-1, 0, 0, 0, 0} }; int arch_cmd_mv (t_uchar * program_name, int argc, char * argv[]) { int o; int i; struct opt_parsed * option; t_uchar * dest_name = 0; int dest_is_dir = 0; struct stat stat_buf; safe_buffer_fd (1, 0, O_WRONLY, 0); option = 0; while (1) { o = opt_standard (lim_use_must_malloc, &option, opts, &argc, argv, program_name, usage, libarch_version_string, arch_cmd_mv_help, opt_help_msg, opt_long_help, opt_version); if (o == opt_none) break; switch (o) { default: safe_printfmt (2, "unhandled option `%s'\n", option->opt_string); panic ("internal error parsing arguments"); usage_error: opt_usage (2, argv[0], program_name, usage, 1); exit (1); /* bogus_arg: */ safe_printfmt (2, "ill-formed argument for `%s' (`%s')\n", option->opt_string, option->arg_string); goto usage_error; } } if (argc < 3) goto usage_error; dest_name = argv[argc-1]; if (safe_access (dest_name, F_OK) != -1) { safe_stat (dest_name, &stat_buf); if (stat_buf.st_mode & S_IFDIR) dest_is_dir = 1; } /* if the destination is not a directory, then there should be only 2 args for a simple rename */ if (!dest_is_dir && (argc != 3)) goto usage_error; for (i=1; i < argc-1; ++i) { int move_id = 0; t_uchar * new_name; /* if argv[i] is NOT a directory AND it has an explicit tag, we need to move this tag */ safe_stat (argv[i], &stat_buf); if (!(stat_buf.st_mode & S_IFDIR) && (safe_access (arch_explicit_id_file_for (argv[i]), F_OK) != -1)) move_id = 1; if (dest_is_dir) { /* if dest_name is a directory, move arg into it */ new_name = file_name_in_vicinity (0, dest_name, file_name_tail (0, argv[i])); } else { /* otherwise, simply rename the file */ new_name = dest_name; } safe_rename (argv[i], new_name); if (move_id) arch_move_explicit_id (argv[i], new_name); } return 0; } /* tag: Rob Weir Sun Sep 21 15:22:53 EST 2003 (cmd-mv.c) */