Test calling wxFileSystemWatcher::{Add,Remove}Tree().
[wxWidgets.git] / tests / fswatcher / fswatchertest.cpp
1 ///////////////////////////////////////////////////////////////////////////////
2 // Name: tests/fswatcher/fswatchertest.cpp
3 // Purpose: wxFileSystemWatcher unit test
4 // Author: Bartosz Bekier
5 // Created: 2009-06-11
6 // RCS-ID: $Id$
7 // Copyright: (c) 2009 Bartosz Bekier
8 ///////////////////////////////////////////////////////////////////////////////
9
10 // ----------------------------------------------------------------------------
11 // headers
12 // ----------------------------------------------------------------------------
13
14 #include "testprec.h"
15
16 #ifdef __BORLANDC__
17 #pragma hdrstop
18 #endif
19
20 #ifndef WX_PRECOMP
21 #include "wx/timer.h"
22 #endif
23
24 #include "wx/evtloop.h"
25 #include "wx/filename.h"
26 #include "wx/filefn.h"
27 #include "wx/stdpaths.h"
28 #include "wx/fswatcher.h"
29
30 #include "testfile.h"
31
32 // ----------------------------------------------------------------------------
33 // local functions
34 // ----------------------------------------------------------------------------
35
36 // class generating file system events
37 class EventGenerator
38 {
39 public:
40 static EventGenerator& Get()
41 {
42 if (!ms_instance)
43 ms_instance = new EventGenerator(GetWatchDir());
44
45 return *ms_instance;
46 }
47
48 EventGenerator(const wxFileName& path) : m_base(path)
49 {
50 m_old = wxFileName();
51 m_file = RandomName();
52 m_new = RandomName();
53 }
54
55 // operations
56 bool CreateFile()
57 {
58 wxFile file(m_file.GetFullPath(), wxFile::write);
59 return file.IsOpened() && m_file.FileExists();
60 }
61
62 bool RenameFile()
63 {
64 CPPUNIT_ASSERT(m_file.FileExists());
65
66 wxLogDebug("Renaming %s=>%s", m_file.GetFullPath(), m_new.GetFullPath());
67
68 bool ret = wxRenameFile(m_file.GetFullPath(), m_new.GetFullPath());
69 if (ret)
70 {
71 m_old = m_file;
72 m_file = m_new;
73 m_new = RandomName();
74 }
75
76 return ret;
77 }
78
79 bool DeleteFile()
80 {
81 CPPUNIT_ASSERT(m_file.FileExists());
82
83 bool ret = wxRemoveFile(m_file.GetFullPath());
84 if (ret)
85 {
86 m_old = m_file;
87 m_file = m_new;
88 m_new = RandomName();
89 }
90
91 return ret;
92 }
93
94 bool TouchFile()
95 {
96 return m_file.Touch();
97 }
98
99 bool ReadFile()
100 {
101 wxFile f(m_file.GetFullPath());
102 CPPUNIT_ASSERT(f.IsOpened());
103
104 char buf[1];
105 ssize_t count = f.Read(buf, sizeof(buf));
106 CPPUNIT_ASSERT(count > 0);
107
108 return true;
109 }
110
111 bool ModifyFile()
112 {
113 CPPUNIT_ASSERT(m_file.FileExists());
114
115 wxFile file(m_file.GetFullPath(), wxFile::write_append);
116 CPPUNIT_ASSERT(file.IsOpened());
117
118 CPPUNIT_ASSERT(file.Write("Words of Wisdom, Lloyd. Words of wisdom\n"));
119 return file.Close();
120 }
121
122 // helpers
123 wxFileName RandomName(int length = 10)
124 {
125 return RandomName(m_base, length);
126 }
127
128 // static helpers
129 static const wxFileName& GetWatchDir()
130 {
131 static wxFileName dir;
132
133 if (dir.DirExists())
134 return dir;
135
136 wxString tmp = wxStandardPaths::Get().GetTempDir();
137 dir.AssignDir(tmp);
138
139 // XXX look for more unique name? there is no function to generate
140 // unique filename, the file always get created...
141 dir.AppendDir("fswatcher_test");
142 CPPUNIT_ASSERT(!dir.DirExists());
143 CPPUNIT_ASSERT(dir.Mkdir());
144
145 return dir;
146 }
147
148 static void RemoveWatchDir()
149 {
150 wxFileName dir = GetWatchDir();
151 CPPUNIT_ASSERT(dir.DirExists());
152
153 // just to be really sure we know what we remove
154 CPPUNIT_ASSERT_EQUAL( "fswatcher_test", dir.GetDirs().Last() );
155
156 // FIXME-VC6: using non-static Rmdir() results in ICE
157 CPPUNIT_ASSERT( wxFileName::Rmdir(dir.GetFullPath(), wxPATH_RMDIR_RECURSIVE) );
158 }
159
160 static wxFileName RandomName(const wxFileName& base, int length = 10)
161 {
162 static int ALFA_CNT = 'z' - 'a';
163
164 wxString s;
165 for (int i = 0 ; i < length; ++i)
166 {
167 char c = 'a' + (rand() % ALFA_CNT);
168 s += c;
169 }
170
171 return wxFileName(base.GetFullPath(), s);
172 }
173
174 public:
175 wxFileName m_base; // base dir for doing operations
176 wxFileName m_file; // current file name
177 wxFileName m_old; // previous file name
178 wxFileName m_new; // name after renaming
179
180 protected:
181 static EventGenerator* ms_instance;
182 };
183
184 EventGenerator* EventGenerator::ms_instance = 0;
185
186
187 // custom event handler
188 class EventHandler : public wxEvtHandler
189 {
190 public:
191 enum { WAIT_DURATION = 3 };
192
193 EventHandler() :
194 eg(EventGenerator::Get()), m_loop(0), m_count(0), m_watcher(0)
195 {
196 m_loop = new wxEventLoop();
197 Connect(wxEVT_IDLE, wxIdleEventHandler(EventHandler::OnIdle));
198 Connect(wxEVT_FSWATCHER, wxFileSystemWatcherEventHandler(
199 EventHandler::OnFileSystemEvent));
200 }
201
202 virtual ~EventHandler()
203 {
204 delete m_watcher;
205 if (m_loop)
206 {
207 if (m_loop->IsRunning())
208 m_loop->Exit();
209 delete m_loop;
210 }
211 }
212
213 void Exit()
214 {
215 m_loop->Exit();
216 }
217
218 // sends idle event, so we get called in a moment
219 void SendIdle()
220 {
221 wxIdleEvent* e = new wxIdleEvent();
222 QueueEvent(e);
223 }
224
225 void Run()
226 {
227 SendIdle();
228 m_loop->Run();
229 }
230
231 void OnIdle(wxIdleEvent& /*evt*/)
232 {
233 bool more = Action();
234 m_count++;
235
236 if (more)
237 {
238 SendIdle();
239 }
240 }
241
242 // returns whether we should produce more idle events
243 virtual bool Action()
244 {
245 switch (m_count)
246 {
247 case 0:
248 CPPUNIT_ASSERT(Init());
249 break;
250 case 1:
251 GenerateEvent();
252 break;
253 case 2:
254 // actual test
255 CheckResult();
256 Exit();
257 break;
258
259 // TODO a mechanism that will break the loop in case we
260 // don't receive a file system event
261 // this below doesn't quite work, so all tests must pass :-)
262 #if 0
263 case 2:
264 m_loop.Yield();
265 m_loop.WakeUp();
266 CPPUNIT_ASSERT(KeepWaiting());
267 m_loop.Yield();
268 break;
269 case 3:
270 break;
271 case 4:
272 CPPUNIT_ASSERT(AfterWait());
273 break;
274 #endif
275 } // switch (m_count)
276
277 return m_count <= 0;
278 }
279
280 virtual bool Init()
281 {
282 // test we're good to go
283 CPPUNIT_ASSERT(wxEventLoopBase::GetActive());
284
285 // XXX only now can we construct Watcher, because we need
286 // active loop here
287 m_watcher = new wxFileSystemWatcher();
288 m_watcher->SetOwner(this);
289
290 // add dir to be watched
291 wxFileName dir = EventGenerator::GetWatchDir();
292 CPPUNIT_ASSERT(m_watcher->Add(dir, wxFSW_EVENT_ALL));
293
294 return true;
295 }
296
297 virtual bool KeepWaiting()
298 {
299 // did we receive event already?
300 if (!tested)
301 {
302 // well, let's wait a bit more
303 wxSleep(WAIT_DURATION);
304 }
305
306 return true;
307 }
308
309 virtual bool AfterWait()
310 {
311 // fail if still no events
312 WX_ASSERT_MESSAGE
313 (
314 ("No events during %d seconds!", static_cast<int>(WAIT_DURATION)),
315 tested
316 );
317
318 return true;
319 }
320
321 virtual void OnFileSystemEvent(wxFileSystemWatcherEvent& evt)
322 {
323 wxLogDebug("--- %s ---", evt.ToString());
324 m_lastEvent = wxDynamicCast(evt.Clone(), wxFileSystemWatcherEvent);
325 m_events.Add(m_lastEvent);
326
327 // test finished
328 SendIdle();
329 tested = true;
330 }
331
332 virtual void CheckResult()
333 {
334 CPPUNIT_ASSERT_MESSAGE( "No events received", !m_events.empty() );
335
336 const wxFileSystemWatcherEvent * const e = m_events.front();
337
338 // this is our "reference event"
339 const wxFileSystemWatcherEvent expected = ExpectedEvent();
340
341 CPPUNIT_ASSERT_EQUAL( expected.GetChangeType(), e->GetChangeType() );
342
343 CPPUNIT_ASSERT_EQUAL((int)wxEVT_FSWATCHER, e->GetEventType());
344
345 // XXX this needs change
346 CPPUNIT_ASSERT_EQUAL(wxEVT_CATEGORY_UNKNOWN, e->GetEventCategory());
347
348 CPPUNIT_ASSERT_EQUAL(expected.GetPath(), e->GetPath());
349 CPPUNIT_ASSERT_EQUAL(expected.GetNewPath(), e->GetNewPath());
350
351 // Under MSW extra modification events are sometimes reported after a
352 // rename and we just can't get rid of them, so ignore them in this
353 // test if they do happen.
354 if ( e->GetChangeType() == wxFSW_EVENT_RENAME &&
355 m_events.size() == 2 )
356 {
357 const wxFileSystemWatcherEvent* const e2 = m_events.back();
358 if ( e2->GetChangeType() == wxFSW_EVENT_MODIFY &&
359 e2->GetPath() == e->GetNewPath() )
360 {
361 // This is a modify event for the new file, ignore it.
362 return;
363 }
364 }
365
366 WX_ASSERT_EQUAL_MESSAGE
367 (
368 (
369 "Extra events received, last one is of type %x, path=\"%s\" "
370 "(the original event was for \"%s\" (\"%s\")",
371 m_events.back()->GetChangeType(),
372 m_events.back()->GetPath().GetFullPath(),
373 e->GetPath().GetFullPath(),
374 e->GetNewPath().GetFullPath()
375 ),
376 1, m_events.size()
377 );
378
379 }
380
381 virtual void GenerateEvent() = 0;
382
383 virtual wxFileSystemWatcherEvent ExpectedEvent() = 0;
384
385
386 protected:
387 EventGenerator& eg;
388 wxEventLoopBase* m_loop; // loop reference
389 int m_count; // idle events count
390
391 wxFileSystemWatcher* m_watcher;
392 bool tested; // indicates, whether we have already passed the test
393
394 #include "wx/arrimpl.cpp"
395 WX_DEFINE_ARRAY_PTR(wxFileSystemWatcherEvent*, wxArrayEvent);
396 wxArrayEvent m_events;
397 wxFileSystemWatcherEvent* m_lastEvent;
398 };
399
400
401 // ----------------------------------------------------------------------------
402 // test class
403 // ----------------------------------------------------------------------------
404
405 class FileSystemWatcherTestCase : public CppUnit::TestCase
406 {
407 public:
408 FileSystemWatcherTestCase() { }
409
410 virtual void setUp();
411 virtual void tearDown();
412
413 protected:
414 wxEventLoopBase* m_loop;
415
416 private:
417 CPPUNIT_TEST_SUITE( FileSystemWatcherTestCase );
418 CPPUNIT_TEST( TestEventCreate );
419 CPPUNIT_TEST( TestEventDelete );
420 CPPUNIT_TEST( TestTrees );
421
422 // kqueue-based implementation doesn't collapse create/delete pairs in
423 // renames and doesn't detect neither modifications nor access to the
424 // files reliably currently so disable these tests
425 //
426 // FIXME: fix the code and reenable them
427 #ifndef wxHAS_KQUEUE
428 CPPUNIT_TEST( TestEventRename );
429 CPPUNIT_TEST( TestEventModify );
430
431 // MSW implementation doesn't detect file access events currently
432 #ifndef __WINDOWS__
433 CPPUNIT_TEST( TestEventAccess );
434 #endif // __WINDOWS__
435 #endif // !wxHAS_KQUEUE
436
437 CPPUNIT_TEST( TestNoEventsAfterRemove );
438 CPPUNIT_TEST_SUITE_END();
439
440 void TestEventCreate();
441 void TestEventDelete();
442 void TestEventRename();
443 void TestEventModify();
444 void TestEventAccess();
445 void TestTrees();
446
447 void TestNoEventsAfterRemove();
448
449 DECLARE_NO_COPY_CLASS(FileSystemWatcherTestCase)
450 };
451
452 // the test currently hangs under OS X for some reason and this prevents tests
453 // ran by buildbot from completing so disable it until someone has time to
454 // debug it
455 //
456 // FIXME: debug and fix this!
457 #ifndef __WXOSX__
458 // register in the unnamed registry so that these tests are run by default
459 CPPUNIT_TEST_SUITE_REGISTRATION( FileSystemWatcherTestCase );
460 #endif
461
462 // also include in its own registry so that these tests can be run alone
463 CPPUNIT_TEST_SUITE_NAMED_REGISTRATION( FileSystemWatcherTestCase,
464 "FileSystemWatcherTestCase" );
465
466 void FileSystemWatcherTestCase::setUp()
467 {
468 wxLog::AddTraceMask(wxTRACE_FSWATCHER);
469 EventGenerator::Get().GetWatchDir();
470 }
471
472 void FileSystemWatcherTestCase::tearDown()
473 {
474 EventGenerator::Get().RemoveWatchDir();
475 }
476
477 // ----------------------------------------------------------------------------
478 // TestEventCreate
479 // ----------------------------------------------------------------------------
480 void FileSystemWatcherTestCase::TestEventCreate()
481 {
482 wxLogDebug("TestEventCreate()");
483
484 class EventTester : public EventHandler
485 {
486 public:
487 virtual void GenerateEvent()
488 {
489 CPPUNIT_ASSERT(eg.CreateFile());
490 }
491
492 virtual wxFileSystemWatcherEvent ExpectedEvent()
493 {
494 wxFileSystemWatcherEvent event(wxFSW_EVENT_CREATE);
495 event.SetPath(eg.m_file);
496 event.SetNewPath(eg.m_file);
497 return event;
498 }
499 };
500
501 EventTester tester;
502
503 wxLogTrace(wxTRACE_FSWATCHER, "TestEventCreate tester created()");
504
505 tester.Run();
506 }
507
508 // ----------------------------------------------------------------------------
509 // TestEventDelete
510 // ----------------------------------------------------------------------------
511 void FileSystemWatcherTestCase::TestEventDelete()
512 {
513 wxLogDebug("TestEventDelete()");
514
515 class EventTester : public EventHandler
516 {
517 public:
518 virtual void GenerateEvent()
519 {
520 CPPUNIT_ASSERT(eg.DeleteFile());
521 }
522
523 virtual wxFileSystemWatcherEvent ExpectedEvent()
524 {
525 wxFileSystemWatcherEvent event(wxFSW_EVENT_DELETE);
526 event.SetPath(eg.m_old);
527
528 // CHECK maybe new path here could be NULL or sth?
529 event.SetNewPath(eg.m_old);
530 return event;
531 }
532 };
533
534 // we need to create a file now, so we can delete it
535 EventGenerator::Get().CreateFile();
536
537 EventTester tester;
538 tester.Run();
539 }
540
541 // ----------------------------------------------------------------------------
542 // TestEventRename
543 // ----------------------------------------------------------------------------
544 void FileSystemWatcherTestCase::TestEventRename()
545 {
546 wxLogDebug("TestEventRename()");
547
548 class EventTester : public EventHandler
549 {
550 public:
551 virtual void GenerateEvent()
552 {
553 CPPUNIT_ASSERT(eg.RenameFile());
554 }
555
556 virtual wxFileSystemWatcherEvent ExpectedEvent()
557 {
558 wxFileSystemWatcherEvent event(wxFSW_EVENT_RENAME);
559 event.SetPath(eg.m_old);
560 event.SetNewPath(eg.m_file);
561 return event;
562 }
563 };
564
565 // need a file to rename later
566 EventGenerator::Get().CreateFile();
567
568 EventTester tester;
569 tester.Run();
570 }
571
572 // ----------------------------------------------------------------------------
573 // TestEventModify
574 // ----------------------------------------------------------------------------
575 void FileSystemWatcherTestCase::TestEventModify()
576 {
577 wxLogDebug("TestEventModify()");
578
579 class EventTester : public EventHandler
580 {
581 public:
582 virtual void GenerateEvent()
583 {
584 CPPUNIT_ASSERT(eg.ModifyFile());
585 }
586
587 virtual wxFileSystemWatcherEvent ExpectedEvent()
588 {
589 wxFileSystemWatcherEvent event(wxFSW_EVENT_MODIFY);
590 event.SetPath(eg.m_file);
591 event.SetNewPath(eg.m_file);
592 return event;
593 }
594 };
595
596 // we need to create a file to modify
597 EventGenerator::Get().CreateFile();
598
599 EventTester tester;
600 tester.Run();
601 }
602
603 // ----------------------------------------------------------------------------
604 // TestEventAccess
605 // ----------------------------------------------------------------------------
606 void FileSystemWatcherTestCase::TestEventAccess()
607 {
608 wxLogDebug("TestEventAccess()");
609
610 class EventTester : public EventHandler
611 {
612 public:
613 virtual void GenerateEvent()
614 {
615 CPPUNIT_ASSERT(eg.ReadFile());
616 }
617
618 virtual wxFileSystemWatcherEvent ExpectedEvent()
619 {
620 wxFileSystemWatcherEvent event(wxFSW_EVENT_ACCESS);
621 event.SetPath(eg.m_file);
622 event.SetNewPath(eg.m_file);
623 return event;
624 }
625 };
626
627 // we need to create a file to read from it and write sth to it
628 EventGenerator::Get().CreateFile();
629 EventGenerator::Get().ModifyFile();
630
631 EventTester tester;
632 tester.Run();
633 }
634
635 // ----------------------------------------------------------------------------
636 // TestTrees
637 // ----------------------------------------------------------------------------
638 void FileSystemWatcherTestCase::TestTrees()
639 {
640 class TreeTester : public EventHandler
641 {
642 const size_t subdirs;
643 const size_t files;
644
645 public:
646 TreeTester() : subdirs(5), files(3) {}
647
648 void GrowTree(wxFileName dir)
649 {
650 CPPUNIT_ASSERT(dir.Mkdir());
651
652 // Create a branch of 5 numbered subdirs, each containing 3
653 // numbered files
654 for ( unsigned d = 0; d < subdirs; ++d )
655 {
656 dir.AppendDir(wxString::Format("subdir%u", d+1));
657 CPPUNIT_ASSERT(dir.Mkdir());
658
659 const wxString prefix = dir.GetPathWithSep();
660 for ( unsigned f = 0; f < files; ++f )
661 {
662 // Just create the files.
663 wxFile(prefix + wxString::Format("file%u", f+1),
664 wxFile::write);
665 }
666 }
667 }
668
669 void RmDir(wxFileName dir)
670 {
671 CPPUNIT_ASSERT(dir.DirExists());
672
673 CPPUNIT_ASSERT(dir.Rmdir(wxPATH_RMDIR_RECURSIVE));
674 }
675
676 void WatchDir(wxFileName dir)
677 {
678 CPPUNIT_ASSERT(m_watcher);
679
680 // Store the initial count; there may already be some watches
681 const int initial = m_watcher->GetWatchedPathsCount();
682
683 m_watcher->Add(dir);
684 CPPUNIT_ASSERT_EQUAL(initial + 1,
685 m_watcher->GetWatchedPathsCount());
686 }
687
688 void RemoveSingleWatch(wxFileName dir)
689 {
690 CPPUNIT_ASSERT(m_watcher);
691
692 const int initial = m_watcher->GetWatchedPathsCount();
693
694 m_watcher->Remove(dir);
695 CPPUNIT_ASSERT_EQUAL(initial - 1,
696 m_watcher->GetWatchedPathsCount());
697 }
698
699 void WatchTree(const wxFileName& dir)
700 {
701 CPPUNIT_ASSERT(m_watcher);
702
703 const size_t
704 treeitems = (subdirs*files) + subdirs + 1; // +1 for the trunk
705
706 // Store the initial count; there may already be some watches
707 const int initial = m_watcher->GetWatchedPathsCount();
708
709 GrowTree(dir);
710
711 m_watcher->AddTree(dir);
712 const int plustree = m_watcher->GetWatchedPathsCount();
713
714 CPPUNIT_ASSERT_EQUAL(initial + treeitems, plustree);
715
716 m_watcher->RemoveTree(dir);
717 CPPUNIT_ASSERT_EQUAL(initial, m_watcher->GetWatchedPathsCount());
718 }
719
720 void RemoveAllWatches()
721 {
722 CPPUNIT_ASSERT(m_watcher);
723
724 m_watcher->RemoveAll();
725 CPPUNIT_ASSERT_EQUAL(0, m_watcher->GetWatchedPathsCount());
726 }
727
728 virtual void GenerateEvent()
729 {
730 // We don't use this function for events. Just run the tests
731
732 wxFileName watchdir = EventGenerator::GetWatchDir();
733 CPPUNIT_ASSERT(watchdir.DirExists());
734
735 wxFileName treedir(watchdir);
736 treedir.AppendDir("treetrunk");
737 CPPUNIT_ASSERT(!treedir.DirExists());
738
739 wxFileName singledir(watchdir);
740 singledir.AppendDir("single");
741 CPPUNIT_ASSERT(!singledir.DirExists());
742 CPPUNIT_ASSERT(singledir.Mkdir());
743
744 WatchDir(singledir);
745 WatchTree(treedir);
746
747 RemoveSingleWatch(singledir);
748 // Add it back again, ready to test RemoveAll()
749 WatchDir(singledir);
750
751 RemoveAllWatches();
752
753 // Clean up
754 RmDir(singledir);
755 RmDir(treedir);
756
757 Exit();
758 }
759
760 virtual wxFileSystemWatcherEvent ExpectedEvent()
761 {
762 CPPUNIT_FAIL("Shouldn't be called");
763
764 return wxFileSystemWatcherEvent(wxFSW_EVENT_ERROR);
765 }
766
767 virtual void CheckResult()
768 {
769 // Do nothing. We override this to prevent receiving events in
770 // ExpectedEvent()
771 }
772 };
773
774 TreeTester tester;
775 tester.Run();
776 }
777
778 namespace
779 {
780
781 // We can't define this class locally inside TestNoEventsAfterRemove() for some
782 // reason with g++ 4.0 under OS X 10.5, it results in the following mysterious
783 // error:
784 //
785 // /var/tmp//ccTkNCkc.s:unknown:Non-global symbol:
786 // __ZThn80_ZN25FileSystemWatcherTestCase23TestNoEventsAfterRemoveEvEN11EventTester6NotifyEv.eh
787 // can't be a weak_definition
788 //
789 // So define this class outside the function instead.
790 class NoEventsAfterRemoveEventTester : public EventHandler,
791 public wxTimer
792 {
793 public:
794 NoEventsAfterRemoveEventTester()
795 {
796 // We need to use an inactivity timer as we never get any file
797 // system events in this test, so we consider that the test is
798 // finished when this 1s timeout expires instead of, as usual,
799 // stopping after getting the file system events.
800 Start(1000, true);
801 }
802
803 virtual void GenerateEvent()
804 {
805 m_watcher->Remove(EventGenerator::GetWatchDir());
806 CPPUNIT_ASSERT(eg.CreateFile());
807 }
808
809 virtual void CheckResult()
810 {
811 CPPUNIT_ASSERT( m_events.empty() );
812 }
813
814 virtual wxFileSystemWatcherEvent ExpectedEvent()
815 {
816 CPPUNIT_FAIL( "Shouldn't be called" );
817
818 return wxFileSystemWatcherEvent(wxFSW_EVENT_ERROR);
819 }
820
821 virtual void Notify()
822 {
823 SendIdle();
824 }
825 };
826
827 } // anonymous namespace
828
829 void FileSystemWatcherTestCase::TestNoEventsAfterRemove()
830 {
831 NoEventsAfterRemoveEventTester tester;
832 tester.Run();
833 }