// file      : inverse/driver.cxx
// author    : Boris Kolpackov <boris@codesynthesis.com>
// copyright : not copyrighted - public domain

#include <memory>   // std::auto_ptr
#include <iostream>

#include <odb/database.hxx>
#include <odb/session.hxx>
#include <odb/transaction.hxx>

#include "database.hxx" // create_database

#include "employee.hxx"
#include "employee-odb.hxx"

using namespace std;
using namespace odb::core;

void
print (const employee& e)
{
  cout << e.first () << " " << e.last () << endl
       << "  employer: " << e.employer ().load ()->name () << endl
       << "  position: " << e.position ().load ()->title () << endl;

  const projects& ps (e.projects ());

  for (projects::const_iterator i (ps.begin ()); i != ps.end (); ++i)
  {
    const lazy_shared_ptr<project>& p (*i);
    p.load ();

    cout << "  project: " << p->name () << endl;
  }

  cout << endl;
}

int
main (int argc, char* argv[])
{
  try
  {
    auto_ptr<database> db (create_database (argc, argv));

    // Create a few persistent objects.
    //
    {
      // Simple Tech Ltd.
      //
      {
        shared_ptr<employer> er (new employer ("Simple Tech Ltd"));

        shared_ptr<position> he (new position ("Hardware Engineer"));
        shared_ptr<position> se (new position ("Software Engineer"));

        shared_ptr<project> sh (new project ("Simple Hardware"));
        shared_ptr<project> ss (new project ("Simple Software"));

        shared_ptr<employee> john (new employee ("John", "Doe", er, he));
        shared_ptr<employee> jane (new employee ("Jane", "Doe", er, se));

        // Set the inverse side of the employee-employer relationship.
        //
        er->employees ().push_back (john);
        er->employees ().push_back (jane);

        // Set the inverse side of the employee-position relationship.
        //
        he->employee (john);
        se->employee (jane);

        // Set the employee-project relationship (both directions).
        //
        john->projects ().push_back (sh);
        john->projects ().push_back (ss);
        jane->projects ().push_back (ss);

        sh->employees ().push_back (john);
        ss->employees ().push_back (john);
        ss->employees ().push_back (jane);

        transaction t (db->begin ());

        db->persist (er);

        db->persist (he);
        db->persist (se);

        db->persist (sh);
        db->persist (ss);

        db->persist (john);
        db->persist (jane);

        t.commit ();
      }

      // Complex Systems Inc.
      //
      {
        shared_ptr<employer> er (new employer ("Complex Systems Inc"));

        shared_ptr<position> he (new position ("Hardware Engineer"));
        shared_ptr<position> se (new position ("Software Engineer"));

        shared_ptr<project> ch (new project ("Complex Hardware"));
        shared_ptr<project> cs (new project ("Complex Software"));

        shared_ptr<employee> john (new employee ("John", "Smith", er, se));
        shared_ptr<employee> jane (new employee ("Jane", "Smith", er, he));

        // Set the inverse side of the employee-employer relationship.
        //
        er->employees ().push_back (john);
        er->employees ().push_back (jane);

        // Set the inverse side of the employee-position relationship.
        //
        he->employee (john);
        se->employee (jane);

        // Set the employee-project relationship (both directions).
        //
        john->projects ().push_back (cs);
        jane->projects ().push_back (ch);
        jane->projects ().push_back (cs);

        ch->employees ().push_back (jane);
        cs->employees ().push_back (john);
        cs->employees ().push_back (jane);

        transaction t (db->begin ());

        db->persist (er);

        db->persist (he);
        db->persist (se);

        db->persist (ch);
        db->persist (cs);

        db->persist (john);
        db->persist (jane);

        t.commit ();
      }
    }

    // Load Simple Tech Ltd and print its employees. We use a session in this
    // and subsequent transactions to make sure that a single instance of any
    // particular object (e.g., employer) is shared among all objects (e.g.,
    // employee) that relate to it.
    //
    {
      session s;
      transaction t (db->begin ());

      shared_ptr<employer> stl (db->load<employer> ("Simple Tech Ltd"));

      employees& es (stl->employees ());

      for (employees::iterator i (es.begin ()); i != es.end (); ++i)
      {
        lazy_weak_ptr<employee>& lwp (*i);
        shared_ptr<employee> p (lwp.load ()); // Load and lock.
        print (*p);
      }

      t.commit ();
    }

    // Find all Software Engineers.
    //
    {
      typedef odb::query<position> query;
      typedef odb::result<position> result;

      session s;
      transaction t (db->begin ());

      result r (db->query<position> (query::title == "Software Engineer"));

      for (result::iterator i (r.begin ()); i != r.end (); ++i)
      {
        const lazy_weak_ptr<employee>& lwp (i->employee ());
        shared_ptr<employee> p (lwp.load ()); // Load and lock.

        // Employee can be NULL if the position is vacant.
        //
        if (p)
          print (*p);
      }

      t.commit ();
    }

    // John Doe has moved to Complex Systems Inc and is now working as
    // a Software Engineer on Complex Software.
    //
    {
      typedef odb::query<employee> query;
      typedef odb::result<employee> result;

      session s;
      transaction t (db->begin ());

      // Create "unloaded" pointers to the employer and project objects.
      //
      lazy_shared_ptr<employer> csi (*db, std::string ("Complex Systems Inc"));
      lazy_shared_ptr<project> cs (*db, std::string ("Complex Software"));

      // Create a new Software Engineer position.
      //
      shared_ptr<position> se (new position ("Software Engineer"));

      result r (db->query<employee> (query::first == "John" &&
                                     query::last == "Doe"));

      shared_ptr<employee> john (r.begin ().load ());

      john->employer (csi);
      john->position (se);
      john->projects ().clear ();
      john->projects ().push_back (cs);

      db->persist (se);
      db->update (john);

      t.commit ();
    }

    // Print Complex Systems Inc's employees. This time, instead of loading
    // the employer object, we use a query which shows how we can use members
    // of the pointed-to objects in the queries.
    //
    {
      typedef odb::query<employee> query;
      typedef odb::result<employee> result;

      session s;
      transaction t (db->begin ());

      result r (db->query<employee> (
                  query::employer->name == "Complex Systems Inc"));

      for (result::iterator i (r.begin ()); i != r.end (); ++i)
        print (*i);

      t.commit ();
    }
  }
  catch (const odb::exception& e)
  {
    cerr << e.what () << endl;
    return 1;
  }
}