]> git.tdb.fi Git - builder.git/blobdiff - source/copy.cpp
Add class InternalAction for actions that use a thread to do their work
[builder.git] / source / copy.cpp
index 23cd6f1af358cebb11e9e526a7ede7ef1af4ef0b..eea40fcdae6eda6f63dc30f09988736fa88a7fbc 100644 (file)
@@ -1,5 +1,15 @@
+/* $Id$
+
+This file is part of builder
+Copyright © 2006-2007 Mikko Rasa, Mikkosoft Productions
+Distributed under the LGPL
+*/
+
+#include <errno.h>
 #include <fstream>
+#include <iostream>
 #include <msp/path/utils.h>
+#include "builder.h"
 #include "copy.h"
 #include "package.h"
 
@@ -7,32 +17,55 @@ using namespace std;
 using namespace Msp;
 
 Copy::Copy(Builder &b, const Package &pkg, const Path::Path &s, const Path::Path &d):
-       Action(b),
+       InternalAction(b),
        src(s),
-       dest(d),
-       worker(*this)
+       dest(d)
 {
-       announce(pkg.get_name(), "INST", dest[-1]);
+       announce(pkg.get_name(), "COPY", dest[-1]);
+       if(builder.get_verbose()>=2)
+               cout<<s<<" -> "<<d<<'\n';
+
+       if(!builder.get_dry_run())
+               worker=new Worker(*this);
 }
 
-int Copy::check()
+
+Copy::Worker::Worker(Copy &c):
+       copy(c)
 {
-       if(worker.get_done())
-       {
-               signal_done.emit();
-               worker.join();
-               return 0;
-       }
-       return -1;
+       launch();
 }
 
 void Copy::Worker::main()
 {
-       Path::mkpath(copy.src.subpath(0, copy.src.size()-1), 0755);
-       unlink(copy.dest.str().c_str());
+       Path::mkpath(copy.dest.subpath(0, copy.dest.size()-1), 0755);
+
+       // Remove old file.  Not doing this would cause Bad Stuff when installing libraries.
+       if(unlink(copy.dest.str().c_str())<0 && errno!=ENOENT)
+       {
+               int err=errno;
+               cerr<<"Can't unlink "<<copy.dest<<": "<<strerror(err)<<'\n';
+               done=error=true;
+               return;
+       }
+
        ifstream in(copy.src.str().c_str());
+       if(!in)
+       {
+               cerr<<"Can't open "<<copy.src<<" for reading\n";
+               done=error=true;
+               return;
+       }
+
        ofstream out(copy.dest.str().c_str());
+       if(!out)
+       {
+               cerr<<"Can't open "<<copy.dest<<" for writing\n";
+               done=error=true;
+               return;
+       }
 
+       // Actual transfer loop
        char buf[16384];
        while(!in.eof())
        {
@@ -40,6 +73,7 @@ void Copy::Worker::main()
                out.write(buf, in.gcount());
        }
 
+       // Preserve file permissions
        struct stat st;
        Path::stat(copy.src, st);
        chmod(copy.dest.str().c_str(), st.st_mode&0777);