home *** CD-ROM | disk | FTP | other *** search
- /*Kopy - Copy or Find and optionally Delete files Everywhere
- by Stephen R. Davis, 1987
- 214-454-2426
-
- Search all subdirectories for a particular file pattern. All files
- found matching that pattern are copied to the target path. The
- arguments to KOPY are explained in ERRMSG below.
-
- (Note: compiling with the label SWITCH defined in the
- Options/Compiler/Defines menu reverses the sense of
- the "/S" flag for those that prefer it the other way)
- */
- char *banner = {
- "This program was developed for Borland's Turbo C (Ver 1.0) by \n"
- "Stephen R. Davis for the book:\n\n"
- " Turbo C:\n"
- " The Art of Advanced Program Design, Optimization and Debugging\n"
- " M&T Books\n"
- " 501 Galveston Drive\n"
- " Redwood City, CA 94063\n\n"
- "This program is released into the public domain without charge for\n"
- "the use and enjoyment of the public with the single provision that this\n"
- "banner remain intact. Anyone wishing to learn more about getting the\n"
- "most out of the IBM PC and its clones using Turbo C, can order a copy of\n"
- "this book by calling:\n\n"
- " 1-800-533-4372\n"
- " (in CA, 1-800-356-2002)\n"
- " 8 am to 5 pm PST\n\n"};
-
- #include <stdio.h>
- #include <dir.h>
- #include <io.h>
- #include <dos.h>
- #include <process.h>
- #include <fcntl.h>
- #include <conio.h>
- #include <ctype.h>
- #include <string.h>
- #include <stat.h>
-
- #define TRUE 1
- #define FALSE 0
-
- /*define error message*/
- char *errmsg = {
- "This program copies all files from the source path and all\n"
- "of its subdirectories which match the source pattern to the\n"
- "target path. If no target path is given, the current directory\n"
- "is assumed. The following switches are defined:\n"
- " /D - delete after copying/finding (prompts operator for okay)\n"
- " /F - simply find (do not copy)\n"
- " /S - search single directory only (no subdirectories)\n"
- " /V - requests operator verification before continuing\n"
- " /R - copy to the console (may be redirected with >)\n\n"
- "For example:\n\n"
- "KOPY /D C:\\*.DAT copy and delete all .DAT files from\n"
- " all of the directories on disk C to\n"
- " the current directory.\n\n"
- "KOPY /S \\USER\\*.DAT B:\\COPY copy all the .DAT files from \\USER\n"
- " to directory COPY on drive B.\n\n"
- "KOPY /F/D \\*.BAK find and delete all *.BAK files on\n"
- " current disk\n\n"
- "If the target disk fills you will be prompted to insert a new one.\n"
- };
-
- /*prototyping definitions*/
- void main (unsigned, char **);
- void copyall (char *, char *, char *);
- void copy (char *, char *);
- void find (char *);
- void append (char *, char *, char *);
- void addslash (char *);
- void closefiles (int, int);
- void arg_error (void);
- int getyval (char *);
- int opennew (char *);
-
- /*define global flags*/
-
- char found_one = FALSE, delete = FALSE, find_only = FALSE;
- char verify = FALSE, standardoutput = FALSE;
- int thandle, fhandle;
-
- char include_subdirs =
- #ifdef SWITCH
- FALSE; /*default subdirectory search to off*/
- #else
- TRUE; /* " " " " on*/
- #endif
-
- /*Main - parse user input and start the ball rolling*/
- void main (argc, argv)
- unsigned argc;
- char *argv [];
- {
- char sourcedisk [MAXDRIVE], sourcedir [MAXDIR];
- char sourcefile [MAXFILE], sourceext [MAXEXT];
- char fromdir [MAXPATH], pattern [MAXFILE+MAXEXT];
- char *secondarg, *switchptr, *srcfile;
-
- /*first check for switches -- ignore case and extra '/'s*/
-
- while (*(switchptr = argv [1]) == '/') {
- while (*++switchptr) {
- switch (tolower (*switchptr)) {
- case 'd': delete = TRUE;
- break;
- case 'f': find_only = TRUE;
- break;
- case 's': include_subdirs = !include_subdirs;
- break;
- case 'v': verify = TRUE;
- break;
- case 'r': standardoutput = TRUE;
- }
- }
-
- /*now skip over this argument*/
- argc--;
- argv [1] = argv [0];
- argv++;
- }
-
- /*check the argument count*/
- if (argc == 1 || argc > 3)
- arg_error ();
-
- /*parse argument 1 into its two halves
- (if no source path given, assume "*.*") */
- fnsplit (argv [1], sourcedisk, sourcedir, sourcefile, sourceext);
- if (!*(srcfile = sourcefile))
- srcfile = "*.*";
-
- /*now reconstruct the two halves*/
- fnmerge (fromdir, sourcedisk, sourcedir, 0, 0);
- fnmerge (pattern, 0, 0, srcfile, sourceext);
-
- /*if no second argument and not redirected, assume the
- current directory*/
- if (standardoutput) {
- secondarg = " ";
- thandle = fileno (stdout);
- } else {
- if (!(secondarg = argv [2]))
- secondarg = ".\\";
- addslash (secondarg);
- }
-
- /*now just copy/find them everywhere*/
- copyall (fromdir, pattern, secondarg);
- if (!found_one) {
- fprintf (stderr, "No files found");
- if (include_subdirs && (sourcedir [0] != '\\'))
- fprintf (stderr, " (use %s\\%s%s to search entire disk)\n",
- sourcedisk, sourcefile, sourceext);
- fprintf (stderr, "\n");
- }
-
- /*exit normally*/
- exit (0);
- }
-
- /*Copyall - copy/find all files matching a given pattern from the
- current directory and all subdirectories*/
- void copyall (fromdir, pattern, todir)
- char *fromdir, *pattern, *todir;
- {
- char spath [MAXPATH], tpath [MAXPATH];
- struct ffblk block;
-
- /*first copy/find all files patching the pattern*/
- append (spath, fromdir, pattern);
- if (!findfirst (spath, &block, 0))
- do {
- append (spath, fromdir, block.ff_name);
- append (tpath, todir, block.ff_name);
- if (!find_only)
- copy (spath, tpath);
- else
- find (spath);
- } while (!findnext (&block));
-
- /*now check all subdirectories, if desired*/
- if (include_subdirs) {
- append (spath, fromdir, "*.*");
- if (!findfirst (spath, &block, FA_DIREC))
- do {
-
- /*only pay attention to directories*/
- if (block.ff_attrib & FA_DIREC)
-
- /*ignore directories '.' and '..'*/
- if (block.ff_name [0] != '.') {
-
- /*now tack on name of directory + '\'*/
- append (spath, fromdir, block.ff_name);
- addslash (spath);
-
- /*and copy its contents too*/
- copyall (spath, pattern, todir);
- }
- } while (!findnext (&block));
- }
- }
-
- /*Copy - given two patterns, copy the source to the destination file*/
- #define NSECT 64
- void copy (from, to)
- char *from, *to;
- {
- int number;
- char buffer [NSECT*512], failure;
-
- /*don't copy a file to itself*/
- if (!stricmp (to, from))
- return;
- found_one = TRUE;
-
- do {
- /*check for verification before copying*/
- fprintf (stderr, "\nCopying %s -> %s", from, to);
- if (verify)
- if (!getyval (" - copy?"))
- return;
-
- /*open the source for reading binary*/
- _fmode = O_BINARY;
- if ((fhandle = open (from, O_RDONLY)) == -1) {
- perror ("\nError opening source file");
- return;
- }
-
- /*now open the destination, if not standard output*/
- if (!standardoutput)
- if ((thandle = opennew (to)) == -1) {
- if (!getyval (" - overwrite target?")) {
- close (fhandle);
- return;
- }
- if ((thandle = open (to, O_RDWR + O_TRUNC,
- S_IWRITE)) == -1) {
- perror ("\nError opening target file");
- return;
- }
- }
-
- /*now perform the copy*/
- failure = FALSE;
- while (number = read (fhandle, buffer, NSECT*512))
- if (number != _write (thandle, buffer, number))
- if (!standardoutput) {
-
- /*disk full, close source and delete target*/
- closefiles (fhandle, thandle);
- unlink (to);
- getyval ("\nDisk full - "
- "insert new disk and hit any key");
- failure = TRUE;
- break;
- }
- } while (failure);
- fprintf (stderr, " - copied");
- closefiles (fhandle, thandle);
-
- /*delete source, if requested*/
- if (delete)
- if (getyval (" - delete source?"))
- unlink (from);
- }
-
- /*Find - in case of find, ask user permission to delete file if
- "/d" switch and whether we should continue if verify enabled*/
- void find (fname)
- char *fname;
- {
- found_one = TRUE;
- printf ("\nFound %s", fname);
- if (delete)
- if (getyval (" - delete?"))
- if (unlink (fname))
- perror ("\nError on delete");
-
- if (verify)
- if (!getyval (" - continue?"))
- exit (0);
- }
-
- /*Append - concatenate two strings together*/
- void append (to, from1, from2)
- char *to, *from1, *from2;
- {
- /*copy the first string*/
- while (*from1)
- *to++ = *from1++;
-
- /*now the second*/
- while (*from2)
- *to++ = *from2++;
-
- /*and then tack on a terminator*/
- *to = '\0';
- }
-
- /*Addslash - add a slash onto a directory name which doesn't
- already end in '\' or ':'*/
- void addslash (dirptr)
- char *dirptr;
- {
- /*skip to next to last character in path*/
- while (*dirptr)
- dirptr++;
- dirptr--;
-
- /*now check last character*/
- if (*dirptr != '\\' && *dirptr != ':') {
- *++dirptr = '\\';
- *++dirptr = '\0';
- }
- }
-
- /*closefiles - close the source and, if not stdout, the target*/
- void closefiles (fhandle, thandle)
- int fhandle, thandle;
- {
- close (fhandle);
- if (!standardoutput)
- close (thandle);
- }
-
- /*Opennew - use creatnew() to create a new file. Since this system
- call is not present under DOS 2.x, its function must
- be emulated by normal open() calls.*/
- int opennew (name)
- char *name;
- {
- int handle;
-
- if (_osmajor >= 3)
- return creatnew (name, 0);
-
- if ((handle = open (name, O_RDONLY)) != -1) {
- close (handle);
- return -1;
- }
- return open (name, O_RDWR + O_CREAT, S_IWRITE);
- }
-
- /*Arg_error - print error message and then abort*/
- void arg_error (void)
- {
- printf (errmsg);
- if (!include_subdirs)
- printf ("(Note: This version has the meaning of the "
- "/S switch inverted.)");
- getyval ("\nEnter any key to continue");
- printf ("\n\n\n\n\n\n\n%s", banner);
- exit (1);
- }
-
- /*Getyval - out a string and then await a response.
- Exit program if Break (Control-C). Return a 1
- if entered 'y', else 0*/
- int getyval (msg)
- char *msg;
- {
- char entered;
-
- fprintf (stderr, msg);
- if ((entered = getch ()) == 0x03) exit (1);
- return (tolower (entered) == 'y');
- }