LUMIERA.clone/tests/components/proc/mobject/mobject-ref-test.cpp

344 lines
12 KiB
C++
Raw Normal View History

2009-06-02 03:31:52 +02:00
/*
MObjectRef(Test) - validating basic properties of the external MObject reference tag
Copyright (C) Lumiera.org
2009, Hermann Vosseler <Ichthyostega@web.de>
This program is free software; you can redistribute it and/or
modify it under the terms of the GNU General Public License as
published by the Free Software Foundation; either version 2 of the
License, or (at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program; if not, write to the Free Software
Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
* *****************************************************/
#include "lib/test/run.hpp"
#include "lib/lumitime.hpp"
#include "proc/asset/media.hpp"
2009-06-02 03:31:52 +02:00
#include "proc/mobject/mobject.hpp"
#include "proc/mobject/mobject-ref.hpp"
#include "proc/mobject/placement.hpp"
#include "proc/mobject/placement-ref.hpp"
#include "proc/mobject/session/placement-index.hpp"
#include "proc/mobject/session/session-service-mock-index.hpp"
#include "proc/mobject/session/clip.hpp"
#include "proc/mobject/explicitplacement.hpp"
2009-06-02 03:31:52 +02:00
#include "proc/mobject/test-dummy-mobject.hpp"
2009-06-07 03:12:07 +02:00
#include "lib/test/test-helper.hpp"
2009-06-02 03:31:52 +02:00
#include <iostream>
using lib::test::showSizeof;
2009-06-02 03:31:52 +02:00
using std::string;
using std::cout;
using std::endl;
2009-06-02 03:31:52 +02:00
namespace mobject {
namespace test {
using lumiera::Time;
using session::Clip;
using session::SessionServiceMockIndex;
using session::PPIdx;
2009-06-02 03:31:52 +02:00
/***************************************************************************
* @test properties and behaviour of the external reference-mechanism for
* MObjects placed into the session. We create a simple, simulated
* "session" (no real session datastructure) and a mock PlacementIndex.
* Then we add two Placements, which can be used then to build
* MObjectRef instances to validate their behaviour.
2009-06-02 03:31:52 +02:00
* @see mobject::Placement
* @see mobject::MObject
* @see mobject::PlacementRef_test
*/
class MObjectRef_test : public Test
{
typedef Placement<MObject> PMObj;
typedef Placement<Clip> PClip;
typedef TestPlacement<TestSubMO1> PSub1;
2009-06-02 03:31:52 +02:00
virtual void
run (Arg)
{
// create data simulating a "Session"
PMObj testClip1 = asset::Media::create("test-1", asset::VIDEO)->createClip();
PMObj testClip2 = asset::Media::create("test-2", asset::VIDEO)->createClip();
// set up a tie to fixed start positions (i.e. "properties of placement")
testClip1.chain(Time(10));
testClip2.chain(Time(20));
ASSERT (testClip1->isValid());
ASSERT (testClip2->isValid());
ASSERT (2 == testClip1.use_count()); // one by the placement and one by the clip-Asset
ASSERT (2 == testClip2.use_count());
// Prepare an (test)Index
PPIdx index = SessionServiceMockIndex::install();
PMO& root = index->getRoot();
// Add the Clips to "session" (here: dummy index)
PMObj::ID clip1ID = index->insert (testClip1, root);
PMObj::ID clip2ID = index->insert (testClip2, root);
ASSERT (2 == index->size());
// use the IDs returned on insertion to fetch
// language references to the placement instance within the index
// we'll use these language refs as base to create MObejctRef handles.
PMObj& pClip1 = index->find(clip1ID);
PMObj& pClip2 = index->find(clip2ID);
// extract various kinds of IDs and refs
PMObj & rP1 (pClip1);
PMObj const& rP2 (pClip2);
PMObj::ID id1 = pClip1.getID();
PMObj::Id<Clip> id2 = pClip2.recastID<Clip>(); // explicit unchecked re-declaration of target type
LumieraUid luid = id1.get();
///////////////////////////////////////////TODO: check the C-API representation here
PlacementRef<Clip> ref1 (id1);
PlacementRef<MObject> ref2 (pClip2);
// -----Tests------------------
checkBuildMObjectRef (rP1, &pClip1);
checkBuildMObjectRef (rP2, &pClip2);
checkBuildMObjectRef (id1, &pClip1);
checkBuildMObjectRef (id2, &pClip2);
checkBuildMObjectRef (luid, &pClip1);
///////////////////////////////////////////TODO: check the C-API representation here
checkBuildMObjectRef (ref1, &pClip1);
checkBuildMObjectRef (ref2, &pClip2);
2009-06-07 03:12:07 +02:00
checkComparison(rP1,rP2);
2009-06-07 03:12:07 +02:00
checkLifecycle (rP1,rP2);
checkTypeHandling (luid);
// -----Tests------------------
// verify clean state
index->remove (pClip1);
index->remove (pClip2); // note: this invalidates pClip1 and pClip2;
ASSERT (!ref1);
ASSERT (!ref2);
ASSERT (0 == index->size());
ASSERT (1 == index.use_count());
ASSERT (2 == testClip1.use_count());
ASSERT (2 == testClip2.use_count());
index.reset();
2009-06-02 03:31:52 +02:00
}
template<typename REF>
void
checkBuildMObjectRef (REF refObj, void* placementAdr)
{
MORef<Clip> rMO;
ASSERT (!rMO); // still empty (not bound)
cout << rMO << endl;
cout << showSizeof(rMO) << endl;
// activate by binding to provided ref
rMO.activate(refObj);
ASSERT (rMO); // now bound
cout << rMO << endl;
// access MObject (Clip API)
// cout << rMO->operator string() << endl; /////////////////////TICKET #428
cout << string(rMO->getMedia()->ident) << endl; /////////////////////TICKET #520
ASSERT (rMO->isValid());
// access the Placement-API
2009-06-07 03:12:07 +02:00
ASSERT (3 == rMO.use_count()); // now rMO shares ownership with the Placement
ASSERT (Time(0) < rMO.getStartTime()); // (internally, this resolves to an ExplicitPlacement) /////////TICKET #332
ASSERT ( rMO.isCompatible<MObject>());
ASSERT ( rMO.isCompatible<Clip>());
ASSERT (!rMO.isCompatible<TestSubMO1>());
Time start = rMO.getStartTime();
// re-link to the Placement (note we get the Clip API!)
Placement<Clip> & refP = rMO.getPlacement();
ASSERT (refP.isValid());
2009-06-07 03:12:07 +02:00
ASSERT (3 == refP.use_count());
ASSERT (&refP == placementAdr); // actually denotes the address of the original Placement in the "session"
cout << string(refP) << endl;
2009-06-07 03:12:07 +02:00
ExplicitPlacement exPla = refP.resolve();
ASSERT (exPla.time == start); // recovered Placement resolves to the same time as provided by the proxied API
2009-06-07 03:12:07 +02:00
ASSERT (4 == refP.use_count()); // but now we've indeed created an additional owner (exPla)
ASSERT (4 == rMO.use_count());
}
void
checkComparison (PMO const& p1, PMO const& p2)
{
PlacementRef<Clip> pRef1 (p1);
PlacementRef<MObject> pRef2 (p2);
MORef<MObject> rM;
MORef<Clip> rC;
rM.activate (p1);
rC.activate (p2);
ASSERT (rM && rC);
ASSERT (!(rM == rC) && !(rC == rM));
ASSERT ( (rM != rC) && (rC != rM));
// mixed comparisons
ASSERT ( (rM == pRef1) && (pRef1 == rM));
ASSERT ( (rC == pRef2) && (pRef2 == rC));
ASSERT (!(rM != pRef1) && !(pRef1 != rM));
ASSERT (!(rC != pRef2) && !(pRef2 != rC));
ASSERT ( (rM != pRef2) && (pRef2 != rM));
ASSERT ( (rC != pRef1) && (pRef1 != rC));
ASSERT (!(rM == pRef2) && !(pRef2 == rM));
ASSERT (!(rC == pRef1) && !(pRef1 == rC));
ASSERT ( (rM == p1.getID()) );
ASSERT ( (rC == p2.getID()) );
ASSERT (!(rM != p1.getID()) );
ASSERT (!(rC != p2.getID()) );
ASSERT ( (rM != p2.getID()) );
ASSERT ( (rC != p1.getID()) );
ASSERT (!(rM == p2.getID()) );
ASSERT (!(rC == p1.getID()) );
rC.activate (pRef1);
ASSERT ( (rM == rC) && (rC == rM));
ASSERT (!(rM != rC) && !(rC != rM));
ASSERT ( (rC == pRef1) && (pRef1 == rC));
ASSERT (!(rC != pRef1) && !(pRef1 != rC));
ASSERT ( (rC != pRef2) && (pRef2 != rC));
ASSERT (!(rC == pRef2) && !(pRef2 == rC));
ASSERT ( (rC == p1.getID()) );
ASSERT (!(rC != p1.getID()) );
ASSERT ( (rC != p2.getID()) );
ASSERT (!(rC == p2.getID()) );
rM.close();
ASSERT (!rM);
ASSERT (!(rM == rC) && !(rC == rM));
ASSERT ( (rM != rC) && (rC != rM));
ASSERT (!(rM == pRef1) && !(pRef1 == rM));
ASSERT ( (rM != pRef1) && (pRef1 != rM));
ASSERT (!(rM != pRef2) && !(pRef2 != rM));
ASSERT ( (rM == pRef2) && (pRef2 == rM));
ASSERT (!(rM == p1.getID()) );
ASSERT ( (rM != p1.getID()) );
ASSERT (!(rM != p2.getID()) );
ASSERT ( (rM == p2.getID()) );
}
2009-06-07 03:12:07 +02:00
void
checkLifecycle (PMObj const& p1, PMObj const& p2)
2009-06-07 03:12:07 +02:00
{
ASSERT (2 == p1.use_count());
ASSERT (2 == p2.use_count());
MORef<Clip> rMO;
ASSERT (!rMO);
ASSERT (0 == rMO.use_count());
rMO.activate(p1);
ASSERT (rMO);
ASSERT (rMO->getMedia()->getFilename() == "test-1");
ASSERT (3 == rMO.use_count());
2009-06-07 03:12:07 +02:00
ASSERT (3 == p1.use_count());
ASSERT (2 == p2.use_count());
rMO.activate(p2);
ASSERT (rMO);
ASSERT (rMO->getMedia()->getFilename() == "test-2");
ASSERT (3 == rMO.use_count());
ASSERT (2 == p1.use_count());
ASSERT (3 == p2.use_count());
rMO.activate(p2);
ASSERT (3 == rMO.use_count());
rMO.close();
ASSERT (!rMO);
ASSERT (2 == p1.use_count());
ASSERT (2 == p2.use_count());
VERIFY_ERROR (INVALID_PLACEMENTREF, rMO.getPlacement() );
VERIFY_ERROR (BOTTOM_MOBJECTREF, rMO->getMedia() );
2009-06-07 03:12:07 +02:00
}
void
checkTypeHandling (LumieraUid luid)
{
MObjectRef rMObj;
MORef<Clip> rClip;
MORef<TestSubMO1> rSub1;
2009-06-07 03:12:07 +02:00
ASSERT (0 == rMObj.use_count());
ASSERT (0 == rClip.use_count());
ASSERT (0 == rSub1.use_count());
rMObj.activate(luid);
ASSERT (3 == rMObj.use_count());
ASSERT (0 == rClip.use_count());
ASSERT (0 == rSub1.use_count());
rClip.activate(rMObj); // attach on existing MObjectRef
ASSERT (4 == rMObj.use_count());
ASSERT (4 == rClip.use_count());
ASSERT (0 == rSub1.use_count());
// impossible, because Clip isn't a subclass of TestSubMO1:
VERIFY_ERROR (INVALID_PLACEMENTREF, rSub1.activate(luid) );
VERIFY_ERROR (INVALID_PLACEMENTREF, rSub1 = rMObj );
ASSERT (rMObj->isValid());
ASSERT (rClip->isValid());
ASSERT (rMObj.getPlacement().getID() == rClip.getPlacement().getID());
// doesn't compile, because the function isn't on MObject API:
// rMObj->getMedia();
rClip.close();
ASSERT (3 == rMObj.use_count());
ASSERT (0 == rClip.use_count());
// can assign, because the actual type checked:
rClip = rMObj;
ASSERT (4 == rMObj.use_count());
ASSERT (4 == rClip.use_count());
cout << rClip << endl;
cout << string(rClip->getMedia()->ident) << endl; //////////TICKET #520
}
2009-06-02 03:31:52 +02:00
};
/** Register this test class... */
LAUNCHER (MObjectRef_test, "unit session");
}} // namespace mobject::test