/* MObjectRef(Test) - validating basic properties of the external MObject reference tag Copyright (C) Lumiera.org 2009, Hermann Vosseler 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" #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" #include "proc/mobject/test-dummy-mobject.hpp" #include "lib/test/test-helper.hpp" #include using lib::test::showSizeof; using std::string; using std::cout; using std::endl; namespace mobject { namespace test { using lumiera::Time; using session::Clip; using session::SessionServiceMockIndex; using session::PPIdx; /*************************************************************************** * @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. * @see mobject::Placement * @see mobject::MObject * @see mobject::PlacementRef_test */ class MObjectRef_test : public Test { typedef Placement PMObj; typedef Placement PClip; typedef TestPlacement PSub1; 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 id2 = pClip2.recastID(); // explicit unchecked re-declaration of target type LumieraUid luid = id1.get(); ///////////////////////////////////////////TODO: check the C-API representation here PlacementRef ref1 (id1); PlacementRef ref2 (pClip2); #if false //////////////////////////////////////////////////////////////////////////////////////////////////////////TICKET 384 !!!!!!!!! // -----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); checkLifecycle (rP1,rP2); checkTypeHandling (luid); // -----Tests------------------ #endif ////////////////////////////////////////////////////////////////////////////////////////TODO lots of things unimplemented.....!!!!! // 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(); } template void checkBuildMObjectRef (REF refObj, void* placementAdr) { MORef 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 ASSERT (3 == rMO.use_count()); // now rMO shares ownership with the Placement #if false //////////////////////////////////////////////////////////////////////////////////////////////////////////TICKET 384 !!!!!!!!! ASSERT (0 < rMO.getStartTime()); // (internally, this resolves to an ExplicitPlacement) /////////TICKET #332 ASSERT ( rMO.isCompatible()); ASSERT ( rMO.isCompatible()); ASSERT (!rMO.isCompatible()); Time start = rMO.getStartTime(); #endif ////////////////////////////////////////////////////////////////////////////////////////TODO lots of things unimplemented.....!!!!! // re-link to the Placement (note we get the Clip API!) Placement & refP = rMO.getPlacement(); ASSERT (refP); ASSERT (3 == refP.use_count()); ASSERT (&refP == placementAdr); // actually denotes the address of the original Placement in the "session" cout << string(refP) << endl; ExplicitPlacement exPla = refP.resolve(); // ASSERT (exPla.time == start); // recovered Placement resolves to the same time as provided by the proxied API ASSERT (4 == refP.use_count()); // but now we've indeed created an additional owner (exPla) ASSERT (4 == rMO.use_count()); } void checkLifecylce (PMObj const& p1, PMObj const& p2) { ASSERT (2 == p1.use_count()); ASSERT (2 == p2.use_count()); MORef rMO; ASSERT (!rMO); ASSERT (0 == rMO.use_count()); #if false //////////////////////////////////////////////////////////////////////////////////////////////////////////TICKET 384 !!!!!!!!! rMO.activate(p1); #endif ////////////////////////////////////////////////////////////////////////////////////////TODO lots of things unimplemented.....!!!!! ASSERT (rMO); ASSERT (rMO->getMedia()->getFilename() == "test-1"); ASSERT (3 == rMO.use_count()); ASSERT (3 == p1.use_count()); ASSERT (2 == p2.use_count()); #if false //////////////////////////////////////////////////////////////////////////////////////////////////////////TICKET 384 !!!!!!!!! rMO.activate(p2); #endif ////////////////////////////////////////////////////////////////////////////////////////TODO lots of things unimplemented.....!!!!! ASSERT (rMO); ASSERT (rMO->getMedia()->getFilename() == "test-2"); ASSERT (3 == rMO.use_count()); ASSERT (2 == p1.use_count()); ASSERT (3 == p2.use_count()); #if false //////////////////////////////////////////////////////////////////////////////////////////////////////////TICKET 384 !!!!!!!!! rMO.activate(p2); #endif ////////////////////////////////////////////////////////////////////////////////////////TODO lots of things unimplemented.....!!!!! 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 (INVALID_MOBJECTREF, rMO->getMedia() ); } void checkTypeHandling (LumieraUid luid) { MObjectRef rMObj; MORef rClip; MORef rSub1; ASSERT (0 == rMObj.use_count()); ASSERT (0 == rClip.use_count()); ASSERT (0 == rSub1.use_count()); #if false //////////////////////////////////////////////////////////////////////////////////////////////////////////TICKET 384 !!!!!!!!! rMObj.activate(luid); #endif ////////////////////////////////////////////////////////////////////////////////////////TODO lots of things unimplemented.....!!!!! ASSERT (3 == rMObj.use_count()); ASSERT (0 == rClip.use_count()); ASSERT (0 == rSub1.use_count()); #if false //////////////////////////////////////////////////////////////////////////////////////////////////////////TICKET 384 !!!!!!!!! rClip.activate(rMObj); // attach on existing MObjectRef #endif ////////////////////////////////////////////////////////////////////////////////////////TODO lots of things unimplemented.....!!!!! ASSERT (4 == rMObj.use_count()); ASSERT (4 == rClip.use_count()); ASSERT (0 == rSub1.use_count()); // impossible, because Clip isn't a subclass of TestSubMO1: #if false //////////////////////////////////////////////////////////////////////////////////////////////////////////TICKET 384 !!!!!!!!! VERIFY_ERROR (INVALID_PLACEMENTREF, rSub1.activate(luid) ); // VERIFY_ERROR (INVALID_PLACEMENTREF, rSub1 = rMObj ); #endif ////////////////////////////////////////////////////////////////////////////////////////TODO lots of things unimplemented.....!!!!! 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(); // can assign, because the actual type is relevant: rClip.close(); ASSERT (3 == rMObj.use_count()); ASSERT (0 == rClip.use_count()); // rClip = rMObj; ASSERT (4 == rMObj.use_count()); ASSERT (4 == rClip.use_count()); cout << rClip << endl; cout << string(rClip->getMedia()->ident) << endl; //////////TICKET #520 } }; /** Register this test class... */ LAUNCHER (MObjectRef_test, "unit session"); }} // namespace mobject::test