// file      : common/composite-id/driver.cxx
// copyright : Copyright (c) 2009-2015 Code Synthesis Tools CC
// license   : GNU GPL v2; see accompanying LICENSE file

// Test composite object ids.
//

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

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

#include <common/common.hxx>

#include "test.hxx"
#include "test-odb.hxx"

using namespace std;
using namespace odb::core;

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

    // Test 1.
    //
    {
      using namespace test1;

      object o1 (scomp ("aaa", "bbb", "ccc"), 123);
      o1.vec.push_back (scomp ("xxx", "xxx", "xxx"));
      o1.vec.push_back (scomp ("yyy", "yyy", "yyy"));

      object o2 (scomp ("aaa", "bbb", "ccd"), 234);
      o2.vec.push_back (scomp ("zzz", "", "zzz"));

      object o3 (scomp ("baa", "bbb", "ccc"), 345);

      // Persist.
      //
      {
        transaction t (db->begin ());
        db->persist (o1);
        db->persist (o2);
        db->persist (o3);
        t.commit ();
      }

      // Load.
      //
      {
        transaction t (db->begin ());
        auto_ptr<object> p1 (db->load<object> (o1.id));
        auto_ptr<object> p2 (db->load<object> (o2.id));
        auto_ptr<object> p3 (db->load<object> (o3.id));
        t.commit ();

        assert (*p1 == o1);
        assert (*p2 == o2);
        assert (*p3 == o3);
      }

      // Update.
      //
      {
        transaction t (db->begin ());
        auto_ptr<object> p (db->load<object> (o1.id));
        p->num++;
        db->update (*p);
        t.commit ();
      }

      {
        transaction t (db->begin ());
        auto_ptr<object> p (db->load<object> (o1.id));
        t.commit ();

        assert (p->num == o1.num + 1);
      }

      // Erase.
      //
      {
        transaction t (db->begin ());
        db->erase<object> (o1.id);
        t.commit ();
      }

      {
        transaction t (db->begin ());
        auto_ptr<object> p (db->find<object> (o1.id));
        assert (p.get () == 0);
        t.commit ();
      }
    }

    // Test 2.
    //
    {
      using namespace test2;

      object2 o2 (ncomp (2, 0, 1));
      o2.o1 = new object1 (scomp ("o1", "o2", "aaa"));

      object3 o3 (ncomp (3, 0, 1));
      o3.o1.push_back (new object1 (scomp ("o1", "o3", "aaa")));
      o3.o1.push_back (new object1 (scomp ("o1", "o3", "bbb")));

      object4 o4 (ncomp (4, 0, 1));
      o4.c.o2 = new object2 (ncomp (2, 4, 1));
      o4.c.o2->o1 = new object1 (scomp ("o1", "o2", "ccc"));

      // Persist.
      //
      {
        transaction t (db->begin ());
        db->persist (o2.o1);
        db->persist (o2);
        db->persist (o3.o1[0]);
        db->persist (o3.o1[1]);
        db->persist (o3);
        db->persist (o4.c.o2->o1);
        db->persist (o4.c.o2);
        db->persist (o4);
        t.commit ();
      }

      // Load.
      //
      {
        transaction t (db->begin ());
        auto_ptr<object2> p2 (db->load<object2> (o2.id));
        auto_ptr<object3> p3 (db->load<object3> (o3.id));
        auto_ptr<object4> p4 (db->load<object4> (o4.id));
        t.commit ();

        assert (p2->o1->id == o2.o1->id);
        assert (p3->o1.size () == o3.o1.size ());
        assert (p3->o1[0]->id == o3.o1[0]->id);
        assert (p3->o1[1]->id == o3.o1[1]->id);
        assert (p4->c.o2->id == o4.c.o2->id);
        assert (p4->c.o2->o1->id == o4.c.o2->o1->id);
      }

      // Update.
      //
      {
        scomp id2, id3;

        {
          transaction t (db->begin ());

          auto_ptr<object2> p2 (db->load<object2> (o2.id));
          delete p2->o1;
          p2->o1 = new object1 (scomp ("o1", "o2", "bbb"));
          id2 = db->persist (p2->o1);
          db->update (*p2);

          auto_ptr<object3> p3 (db->load<object3> (o3.id));
          delete p3->o1.back ();
          p3->o1.pop_back ();
          p3->o1.push_back (new object1 (scomp ("o1", "o3", "ccc")));
          id3 = db->persist (p3->o1.back ());
          db->update (*p3);

          t.commit ();
        }

        {
          transaction t (db->begin ());
          auto_ptr<object2> p2 (db->load<object2> (o2.id));
          auto_ptr<object3> p3 (db->load<object3> (o3.id));
          t.commit ();

          assert (p2->o1->id == id2);
          assert (p3->o1.back ()->id == id3);
        }
      }

      // Query.
      //
      {
        {
          typedef odb::query<object2> query;
          typedef odb::result<object2> result;

          transaction t (db->begin ());

          {
            result r (db->query<object2> (query::o1->id.str3 == "bbb"));
            result::iterator i (r.begin ());
            assert (i != r.end ());
            assert (i->id == o2.id);
            assert (++i == r.end ());
          }

          {
            // As id (dual interface).
            //
            result r (db->query<object2> (query::o1.str3 == "bbb"));
            result::iterator i (r.begin ());
            assert (i != r.end ());
            assert (i->id == o2.id);
            assert (++i == r.end ());
          }

          t.commit ();
        }

        // Second level composite object pointer.
        //
        {
          typedef odb::query<object4> query;
          typedef odb::result<object4> result;

          transaction t (db->begin ());

          result r (db->query<object4> (query::c.o2->o1.str3 == "ccc"));
          result::iterator i (r.begin ());
          assert (i != r.end ());
          assert (i->id == o4.id);
          assert (++i == r.end ());

          t.commit ();
        }
      }

      // View.
      //
      {
        transaction t (db->begin ());

        {
          typedef odb::query<view2> query;
          typedef odb::result<view2> result;

          result r (db->query<view2> (query::object2::id.num2 == 0));
          result::iterator i (r.begin ());
          assert (i != r.end ());
          assert (i->num == 1 && i->str == "bbb");
          assert (++i == r.end ());
        }

        {
          typedef odb::query<view3> query;
          typedef odb::result<view3> result;

          result r (db->query<view3> ((query::object3::id.num2 == 0) +
                                      "ORDER BY" + query::object1::id.str3));
          result::iterator i (r.begin ());
          assert (i != r.end ());
          assert (i->num == 1 && i->str == "aaa");
          assert (++i != r.end ());
          assert (i->num == 1 && i->str == "ccc");
          assert (++i == r.end ());
        }

        {
          typedef odb::query<view4> query;
          typedef odb::result<view4> result;

          result r (db->query<view4> (query::object4::id.num2 == 0));
          result::iterator i (r.begin ());
          assert (i != r.end ());
          assert (i->num4 == 1 && i->num2 == 1 && i->str == "ccc");
          assert (++i == r.end ());
        }

        t.commit ();
      }
    }

    // Test 3.
    //
    {
      using namespace test3;

      object2 o2 (ncomp (2, 0, 1));
      o2.o1 = new object1 (scomp ("o1", "o2", "aaa"));
      o2.o1->o2 = &o2;

      // Persist.
      //
      {
        transaction t (db->begin ());
        db->persist (o2.o1);
        db->persist (o2);
        t.commit ();
      }

      // Load.
      //
      {
        session s;
        transaction t (db->begin ());
        auto_ptr<object2> p2 (db->load<object2> (o2.id));
        t.commit ();

        assert (p2->o1->o2->id == o2.id);
      }

      // Query.
      //
      {
        typedef odb::query<object1> query;
        typedef odb::result<object1> result;

        transaction t (db->begin ());

        {
          session s;

          result r (db->query<object1> (query::o2->id.num2 == 0));
          result::iterator i (r.begin ());
          assert (i != r.end ());
          assert (i->id == o2.o1->id);

          i->o2->o1 = 0;
          delete i->o2;

          assert (++i == r.end ());
        }

        t.commit ();
      }

      // View.
      //
      {
        typedef odb::query<view> query;
        typedef odb::result<view> result;

        transaction t (db->begin ());

        result r (db->query<view> (query::object1::id.str2 == "o2"));
        result::iterator i (r.begin ());
        assert (i != r.end ());
        assert (i->num == 1 && i->str == "aaa");
        assert (++i == r.end ());

        t.commit ();
      }
    }

    // Test 4.
    //
    {
      using namespace test4;

      object2 o2 (ncomp (2, 0, 1));

      o2.o1.push_back (new object1 (scomp ("o1", "o2", "aaa")));
      o2.o1.back ()->o2 = &o2;

      o2.o1.push_back (new object1 (scomp ("o1", "o2", "bbb")));
      o2.o1.back ()->o2 = &o2;

      // Persist.
      //
      {
        transaction t (db->begin ());
        db->persist (o2.o1[0]);
        db->persist (o2.o1[1]);
        db->persist (o2);
        t.commit ();
      }

      // Load.
      //
      {
        session s;
        transaction t (db->begin ());
        auto_ptr<object2> p2 (db->load<object2> (o2.id));
        t.commit ();

        assert (p2->o1.size () == 2);
        assert (p2->o1[0]->o2->id == o2.id);
        assert (p2->o1[1]->o2->id == o2.id);
      }

      // Query.
      //
      {
        typedef odb::query<object1> query;
        typedef odb::result<object1> result;

        transaction t (db->begin ());

        {
          session s;

          result r (db->query<object1> (query::o2->id.num2 == 0));
          result::iterator i (r.begin ());
          assert (i != r.end ());
          assert (i->id == o2.o1[0]->id);
          i->o2->o1.clear ();

          assert (++i != r.end ());
          assert (i->id == o2.o1[1]->id);

          i->o2->o1.clear ();
          delete i->o2;

          assert (++i == r.end ());
        }

        t.commit ();
      }

      // View.
      //
      {
        typedef odb::query<view> query;
        typedef odb::result<view> result;

        transaction t (db->begin ());

        result r (db->query<view> (query::object1::id.str3 == "bbb"));
        result::iterator i (r.begin ());
        assert (i != r.end ());
        assert (i->num == 1 && i->str == "bbb");
        assert (++i == r.end ());

        t.commit ();
      }
    }

    // Test 5.
    //
    {
      using namespace test5;

      object2 o2 (ncomp (2, 0, 1));

      o2.o1.push_back (new object1 (scomp ("o1", "o2", "aaa")));
      o2.o1.back ()->o2 = &o2;

      o2.o1.push_back (new object1 (scomp ("o1", "o2", "bbb")));
      o2.o1.back ()->o2 = &o2;

      // Persist.
      //
      {
        transaction t (db->begin ());
        db->persist (o2.o1[0]);
        db->persist (o2.o1[1]);
        db->persist (o2);
        t.commit ();
      }

      // Load.
      //
      {
        session s;
        transaction t (db->begin ());
        auto_ptr<object2> p2 (db->load<object2> (o2.id));
        t.commit ();

        assert (p2->o1.size () == 2);

        assert (p2->o1[0]->id == o2.o1[0]->id);
        assert (p2->o1[0]->o2->id == o2.id);

        assert (p2->o1[1]->id == o2.o1[1]->id);
        assert (p2->o1[1]->o2->id == o2.id);
      }

      // View.
      //
      {
        typedef odb::query<view> query;
        typedef odb::result<view> result;

        transaction t (db->begin ());

        result r (db->query<view> ((query::object2::id.num2 == 0) +
                                   "ORDER BY" + query::object1::id.str3));
        result::iterator i (r.begin ());
        assert (i != r.end ());
        assert (i->num == 1 && i->str == "aaa");
        assert (++i != r.end ());
        assert (i->num == 1 && i->str == "bbb");
        assert (++i == r.end ());

        t.commit ();
      }
    }

    // Test 6.
    //
    {
      using namespace test6;

      object2 o2 (ncomp (2, 0, 1));

      o2.o1.push_back (new object1 (scomp ("o1", "o2", "aaa")));
      o2.o1.back ()->o2.push_back (&o2);

      o2.o1.push_back (new object1 (scomp ("o1", "o2", "bbb")));
      o2.o1.back ()->o2.push_back (&o2);

      // Persist.
      //
      {
        transaction t (db->begin ());
        db->persist (o2.o1[0]);
        db->persist (o2.o1[1]);
        db->persist (o2);
        t.commit ();
      }

      // Load.
      //
      {
        session s;
        transaction t (db->begin ());
        auto_ptr<object2> p2 (db->load<object2> (o2.id));
        t.commit ();

        assert (p2->o1.size () == 2);

        assert (p2->o1[0]->id == o2.o1[0]->id);
        assert (p2->o1[0]->o2[0]->id == o2.id);

        assert (p2->o1[1]->id == o2.o1[1]->id);
        assert (p2->o1[1]->o2[0]->id == o2.id);
      }

      // View.
      //
      {
        typedef odb::query<view> query;
        typedef odb::result<view> result;

        transaction t (db->begin ());

        result r (db->query<view> ((query::object2::id.num2 == 0) +
                                   "ORDER BY" + query::object1::id.str3));
        result::iterator i (r.begin ());
        assert (i != r.end ());
        assert (i->num == 1 && i->str == "aaa");
        assert (++i != r.end ());
        assert (i->num == 1 && i->str == "bbb");
        assert (++i == r.end ());

        t.commit ();
      }
    }

    // Test 7.
    //
    {
      using namespace test7;

      object o (scomp ("aaa", "bbb", "ccc"), 123);

      // Persist.
      //
      {
        transaction t (db->begin ());
        db->persist (o);
        t.commit ();
      }

      // Load.
      //
      {
        transaction t (db->begin ());
        auto_ptr<object> p (db->load<object> (o.id));
        t.commit ();

        assert (*p == o);
      }

      // Update.
      //
      {
        transaction t (db->begin ());
        auto_ptr<object> p (db->load<object> (o.id));
        p->num++;
        db->update (*p);

        try
        {
          db->update (o);
          assert (false);
        }
        catch (const object_changed&)
        {
        }

        t.commit ();
      }

      {
        transaction t (db->begin ());
        auto_ptr<object> p (db->load<object> (o.id));
        t.commit ();

        assert (p->num == o.num + 1);
      }

      // Erase.
      //
      {
        transaction t (db->begin ());

        try
        {
          db->update (o);
          assert (false);
        }
        catch (const object_changed&)
        {
        }

        t.commit ();
      }
    }

    // Test 8.
    //
    {
      using namespace test8;

      object2 o2a, o2b;
      object3 o3;

      o2b.o1 = new object1 (scomp ("222", "aaa", "bbb"), 123);
      o3.o1.push_back (0);
      o3.o1.push_back (new object1 (scomp ("333", "aaa", "bbb"), 234));

      // Persist.
      //
      {
        transaction t (db->begin ());
        db->persist (o2a);
        db->persist (o2b);
        db->persist (o2b.o1);
        db->persist (o3);
        db->persist (o3.o1[1]);
        t.commit ();
      }

      // Load.
      //
      {
        transaction t (db->begin ());
        auto_ptr<object2> p2a (db->load<object2> (o2a.id));
        auto_ptr<object2> p2b (db->load<object2> (o2b.id));
        auto_ptr<object3> p3 (db->load<object3> (o3.id));
        t.commit ();

        assert (p2a->o1 == 0);
        assert (p2b->o1 != 0 && *p2b->o1 == *o2b.o1);
        assert (p3->o1[0] == 0);
        assert (p3->o1[1] != 0 && *p3->o1[1] == *o3.o1[1]);
      }

      // Update.
      //
      {
        object1* o1 (o3.o1[1]);

        o3.o1.clear ();
        o3.o1.push_back (o2b.o1);
        o3.o1.push_back (0);

        o2a.o1 = o1;
        o2b.o1 = 0;

        transaction t (db->begin ());
        db->update (o2a);
        db->update (o2b);
        db->update (o3);
        t.commit ();
      }

      {
        transaction t (db->begin ());
        auto_ptr<object2> p2a (db->load<object2> (o2a.id));
        auto_ptr<object2> p2b (db->load<object2> (o2b.id));
        auto_ptr<object3> p3 (db->load<object3> (o3.id));
        t.commit ();

        assert (p2a->o1 != 0 && *p2a->o1 == *o2a.o1);
        assert (p2b->o1 == 0);
        assert (p3->o1[0] != 0 && *p3->o1[0] == *o3.o1[0]);
        assert (p3->o1[1] == 0);
      }
    }

    // Test 9.
    {
      using namespace test9;

      object o (123, "abc");
      o.v.push_back (123);

      // persist
      //
      {
        transaction t (db->begin ());
        db->persist (o);
        t.commit ();
      }

      // load & check
      //
      {
        transaction t (db->begin ());
        result<object> r (db->query<object> ());
        result<object>::iterator i (r.begin ());
        assert (i != r.end () && o == *i && ++i == r.end ());
        t.commit ();
      }
    }
  }
  catch (const odb::exception& e)
  {
    cerr << e.what () << endl;
    return 1;
  }
}