about summary refs log blame commit diff
path: root/src/nix.cc
blob: 0c786527082c078ff1b274c0b2c34cda8777e8b3 (plain) (tree)


























































































































































































































                                                                           
#include <iostream>
#include <memory>
#include <string>

#include <unistd.h>
#include <sys/stat.h>
#include <sys/types.h>
#include <sys/wait.h>

#include <db4/db_cxx.h>

using namespace std;


#define PKGINFO_PATH "/pkg/sys/var/pkginfo"


static string prog;
static string dbfile = PKGINFO_PATH;


class Db2 : public Db 
{
public:
    Db2(DbEnv *env, u_int32_t flags)
        : Db(env, flags)
    {
    }

    ~Db2()
    {
        close(0);
    }
};


auto_ptr<Db2> openDB(const string & dbname, bool readonly)
{
    auto_ptr<Db2> db;

    db = auto_ptr<Db2>(new Db2(0, 0));

    db->open(dbfile.c_str(), dbname.c_str(),
        DB_HASH, readonly ? DB_RDONLY : DB_CREATE, 0666);

    return db;
}


bool queryDB(const string & dbname, const string & key, string & data)
{
    int err;
    auto_ptr<Db2> db = openDB(dbname, true);

    Dbt kt((void *) key.c_str(), key.length());
    Dbt dt;

    err = db->get(0, &kt, &dt, 0);
    if (err) return false;

    data = string((char *) dt.get_data(), dt.get_size());
    
    return true;
}


void setDB(const string & dbname, const string & key, const string & data)
{
    auto_ptr<Db2> db = openDB(dbname, false);
    Dbt kt((void *) key.c_str(), key.length());
    Dbt dt((void *) data.c_str(), data.length());
    db->put(0, &kt, &dt, 0);
}


void delDB(const string & dbname, const string & key)
{
    auto_ptr<Db2> db = openDB(dbname, false);
    Dbt kt((void *) key.c_str(), key.length());
    db->del(0, &kt, 0);
}


void getPkg(int argc, char * * argv)
{
    string pkg;
    string src;
    string inst;
    string cmd;
    int res;

    if (argc != 1)
        throw string("arguments missing in get-pkg");

    pkg = argv[0];

    if (queryDB("pkginst", pkg, inst)) {
        cout << inst << endl;
        return;
    }
    
    cerr << "package " << pkg << " is not yet installed\n";
        
    if (!queryDB("pkgsrc", pkg, src)) 
        throw string("source of package " + string(pkg) + " is not known");

    inst = "/pkg/" + pkg;

    cmd = "rsync -a \"" + src + "\"/ \"" + inst + "\"";
        
    res = system(cmd.c_str());
    if (!WIFEXITED(res) || WEXITSTATUS(res) != 0)
        throw string("unable to copy sources");

    if (chdir(inst.c_str()))
        throw string("unable to chdir to package directory");

    /* Prepare for building.  Clean the environment so that the
       build process does not inherit things it shouldn't. */
    setenv("PATH", "/pkg/sys/bin", 1);

    res = system("./buildme");
    if (!WIFEXITED(res) || WEXITSTATUS(res) != 0)
        throw string("unable to build package");

    setDB("pkginst", pkg, inst);

    cout << inst << endl;
}


void registerPkg(int argc, char * * argv)
{
    char * pkg;
    char * src;
    
    if (argc != 2)
        throw string("arguments missing in register-pkg");

    pkg = argv[0];
    src = argv[1];

    setDB("pkgsrc", pkg, src);
}


/* This is primarily used for bootstrapping. */
void registerInstalledPkg(int argc, char * * argv)
{
    string pkg;
    string inst;
    
    if (argc != 2)
        throw string("arguments missing in register-installed-pkg");

    pkg = argv[0];
    inst = argv[1];
    
    if (inst == "")
        delDB("pkginst", pkg);
    else
        setDB("pkginst", pkg, inst);
}


void run(int argc, char * * argv)
{
    string cmd;

    if (argc < 1)
        throw string("command not specified\n");

    cmd = argv[0];
    argc--, argv++;

    if (cmd == "get-pkg")
        getPkg(argc, argv);
    else if (cmd == "register-pkg")
        registerPkg(argc, argv);
    else if (cmd == "register-installed-pkg")
        registerInstalledPkg(argc, argv);
    else
        throw string("unknown command: " + string(cmd));
}
    
    
int main(int argc, char * * argv)
{
    int c;

    prog = argv[0];

    while ((c = getopt(argc, argv, "d:")) != EOF) {
        
        switch (c) {

        case 'd':
            dbfile = optarg;
            break;

        default:
            cerr << "unknown option\n";
            break;
        }

    }

    argc -= optind, argv += optind;

    try {
        run(argc, argv);
        return 0;
    } catch (DbException e) {
        cerr << "db exception: " << e.what() << endl;
        return 1;
    } catch (string s) {
        cerr << s << endl;
        return 1;
    }
}