1 /*
2  * Licensed to the Apache Software Foundation (ASF) under one or more
3  * contributor license agreements.  See the NOTICE file distributed with
4  * this work for additional information regarding copyright ownership.
5  * The ASF licenses this file to You under the Apache License, Version 2.0
6  * (the "License"); you may not use this file except in compliance with
7  * the License.  You may obtain a copy of the License at
8  *
9  *   http://www.apache.org/licenses/LICENSE-2.0
10  *
11  * Unless required by applicable law or agreed to in writing, software
12  * distributed under the License is distributed on an "AS IS" BASIS,
13  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14  * See the License for the specific language governing permissions and
15  * limitations under the License.
16  */
17 
18 package org.apache.harmony.tests.java.util;
19 
20 import java.lang.Thread.UncaughtExceptionHandler;
21 import java.util.Date;
22 import java.util.Timer;
23 import java.util.TimerTask;
24 import java.util.concurrent.CountDownLatch;
25 import java.util.concurrent.TimeUnit;
26 import java.util.concurrent.atomic.AtomicBoolean;
27 import java.util.concurrent.atomic.AtomicLong;
28 import java.util.concurrent.atomic.AtomicReference;
29 import junit.framework.TestCase;
30 
31 public class TimerTest extends TestCase {
32 
33     int timerCounter = 0;
34 
35     private final Object sync = new Object();
36 
37     /**
38      * Warning: These tests have the possibility to leave a VM hanging if the
39      * Timer is not cancelled.
40      */
41     class TimerTestTask extends TimerTask {
42         int wasRun = 0;
43 
44         // Should we sleep for 200 ms each run()?
45         boolean sleepInRun = false;
46 
47         // Should we increment the timerCounter?
48         boolean incrementCount = false;
49 
50         // Should we terminate the timer at a specific timerCounter?
51         int terminateCount = -1;
52 
53         // The timer we belong to
54         Timer timer = null;
55 
TimerTestTask()56         public TimerTestTask() {
57         }
58 
TimerTestTask(Timer t)59         public TimerTestTask(Timer t) {
60             timer = t;
61         }
62 
run()63         public void run() {
64             synchronized (this) {
65                 wasRun++;
66             }
67             if (incrementCount) {
68                 timerCounter++;
69             }
70             if (terminateCount == timerCounter && timer != null) {
71                 timer.cancel();
72             }
73             if (sleepInRun) {
74                 try {
75                     Thread.sleep(200);
76                 } catch (InterruptedException e) {
77                     throw new RuntimeException(e);
78                 }
79             }
80             synchronized (sync) {
81                 sync.notify();
82             }
83         }
84 
wasRun()85         public synchronized int wasRun() {
86             return wasRun;
87         }
88 
sleepInRun(boolean sleepInRun)89         public void sleepInRun(boolean sleepInRun) {
90             this.sleepInRun = sleepInRun;
91         }
92 
incrementCount(boolean incrementCount)93         public void incrementCount(boolean incrementCount) {
94             this.incrementCount = incrementCount;
95         }
96 
terminateCount(int terminateCount)97         public void terminateCount(int terminateCount) {
98             this.terminateCount = terminateCount;
99         }
100     }
101 
awaitRun(TimerTestTask task)102     private void awaitRun(TimerTestTask task) throws Exception {
103         while (task.wasRun() == 0) {
104             Thread.sleep(150);
105         }
106     }
107 
108     /**
109      * java.util.Timer#Timer(boolean)
110      */
test_ConstructorZ()111     public void test_ConstructorZ() throws Exception {
112         Timer t = null;
113         try {
114             // Ensure a task is run
115             t = new Timer(true);
116             TimerTestTask testTask = new TimerTestTask();
117             t.schedule(testTask, 200);
118             awaitRun(testTask);
119             t.cancel();
120         } finally {
121             if (t != null)
122                 t.cancel();
123         }
124 
125     }
126 
127     /**
128      * java.util.Timer#Timer()
129      */
test_Constructor()130     public void test_Constructor() throws Exception {
131         Timer t = null;
132         try {
133             // Ensure a task is run
134             t = new Timer();
135             TimerTestTask testTask = new TimerTestTask();
136             t.schedule(testTask, 200);
137             awaitRun(testTask);
138             t.cancel();
139         } finally {
140             if (t != null)
141                 t.cancel();
142         }
143 
144     }
145 
146     /**
147      * java.util.Timer#Timer(String, boolean)
148      */
test_ConstructorSZ()149     public void test_ConstructorSZ() throws Exception {
150         Timer t = null;
151         try {
152             // Ensure a task is run
153             t = new Timer("test_ConstructorSZThread", true);
154             TimerTestTask testTask = new TimerTestTask();
155             t.schedule(testTask, 200);
156             awaitRun(testTask);
157             t.cancel();
158         } finally {
159             if (t != null)
160                 t.cancel();
161         }
162 
163         try {
164             new Timer(null, true);
165             fail();
166         } catch (NullPointerException expected) {
167         }
168 
169         try {
170             new Timer(null, false);
171             fail();
172         } catch (NullPointerException expected) {
173         }
174     }
175 
176     /**
177      * java.util.Timer#Timer(String)
178      */
test_ConstructorS()179     public void test_ConstructorS() throws Exception {
180         Timer t = null;
181         try {
182             // Ensure a task is run
183             t = new Timer("test_ConstructorSThread");
184             TimerTestTask testTask = new TimerTestTask();
185             t.schedule(testTask, 200);
186             awaitRun(testTask);
187             t.cancel();
188         } finally {
189             if (t != null)
190                 t.cancel();
191         }
192 
193         try {
194             new Timer(null);
195             fail();
196         } catch (NullPointerException expected) {
197         }
198     }
199 
200     /**
201      * java.util.Timer#cancel()
202      */
test_cancel()203     public void test_cancel() throws Exception {
204         Timer t = null;
205         try {
206             // Ensure a task throws an IllegalStateException after cancelled
207             t = new Timer();
208             TimerTestTask testTask = new TimerTestTask();
209             t.cancel();
210             try {
211                 t.schedule(testTask, 100, 200);
212                 fail("Scheduling a task after Timer.cancel() should throw exception");
213             } catch (IllegalStateException expected) {
214             }
215 
216             // Ensure a task is run but not after cancel
217             t = new Timer();
218             testTask = new TimerTestTask();
219             t.schedule(testTask, 100, 500);
220             awaitRun(testTask);
221             t.cancel();
222             synchronized (sync) {
223                 sync.wait(500);
224             }
225             assertEquals("TimerTask.run() method should not have been called after cancel",
226                     1, testTask.wasRun());
227 
228             // Ensure you can call cancel more than once
229             t = new Timer();
230             testTask = new TimerTestTask();
231             t.schedule(testTask, 100, 500);
232             awaitRun(testTask);
233             t.cancel();
234             t.cancel();
235             t.cancel();
236             synchronized (sync) {
237                 sync.wait(500);
238             }
239             assertEquals("TimerTask.run() method should not have been called after cancel",
240                     1, testTask.wasRun());
241 
242             // Ensure that a call to cancel from within a timer ensures no more
243             // run
244             t = new Timer();
245             testTask = new TimerTestTask(t);
246             testTask.incrementCount(true);
247             testTask.terminateCount(5); // Terminate after 5 runs
248             t.schedule(testTask, 100, 100);
249             synchronized (sync) {
250                 sync.wait(200);
251                 assertEquals(1, testTask.wasRun());
252                 sync.wait(200);
253                 assertEquals(2, testTask.wasRun());
254                 sync.wait(200);
255                 assertEquals(3, testTask.wasRun());
256                 sync.wait(200);
257                 assertEquals(4, testTask.wasRun());
258                 sync.wait(200);
259                 assertEquals(5, testTask.wasRun());
260                 sync.wait(200);
261                 assertEquals(5, testTask.wasRun());
262             }
263             t.cancel();
264             Thread.sleep(200);
265         } finally {
266             if (t != null)
267                 t.cancel();
268         }
269 
270     }
271 
272     /**
273      * java.util.Timer#purge()
274      */
test_purge()275     public void test_purge() throws Exception {
276         Timer t = null;
277         try {
278             t = new Timer();
279             assertEquals(0, t.purge());
280 
281             TimerTestTask[] tasks = new TimerTestTask[100];
282             int[] delayTime = { 50, 80, 20, 70, 40, 10, 90, 30, 60 };
283 
284             int j = 0;
285             for (int i = 0; i < 100; i++) {
286                 tasks[i] = new TimerTestTask();
287                 t.schedule(tasks[i], delayTime[j++], 200);
288                 if (j == 9) {
289                     j = 0;
290                 }
291             }
292 
293             for (int i = 0; i < 50; i++) {
294                 tasks[i].cancel();
295             }
296 
297             assertTrue(t.purge() <= 50);
298             assertEquals(0, t.purge());
299         } finally {
300             if (t != null) {
301                 t.cancel();
302             }
303         }
304     }
305 
306     /**
307      * java.util.Timer#schedule(java.util.TimerTask, java.util.Date)
308      */
test_scheduleLjava_util_TimerTaskLjava_util_Date()309     public void test_scheduleLjava_util_TimerTaskLjava_util_Date() throws Exception {
310         Timer t = null;
311         try {
312             // Ensure a Timer throws an IllegalStateException after cancelled
313             t = new Timer();
314             TimerTestTask testTask = new TimerTestTask();
315             Date d = new Date(System.currentTimeMillis() + 100);
316             t.cancel();
317             try {
318                 t.schedule(testTask, d);
319                 fail("Scheduling a task after Timer.cancel() should throw exception");
320             } catch (IllegalStateException expected) {
321             }
322 
323             // Ensure a Timer throws an IllegalStateException if task already
324             // cancelled
325             t = new Timer();
326             testTask = new TimerTestTask();
327             d = new Date(System.currentTimeMillis() + 100);
328             testTask.cancel();
329             try {
330                 t.schedule(testTask, d);
331                 fail("Scheduling a task after cancelling it should throw exception");
332             } catch (IllegalStateException expected) {
333             }
334             t.cancel();
335 
336             // Ensure a Timer throws an IllegalArgumentException if delay is
337             // negative
338             t = new Timer();
339             testTask = new TimerTestTask();
340             d = new Date(-100);
341             try {
342                 t.schedule(testTask, d);
343                 fail("Scheduling a task with negative date should throw IllegalArgumentException");
344             } catch (IllegalArgumentException expected) {
345             }
346             t.cancel();
347 
348             // Ensure a Timer throws a NullPointerException if the task is null
349             t = new Timer();
350             d = new Date(System.currentTimeMillis() + 100);
351             try {
352                 t.schedule(null, d);
353                 fail("Scheduling a null task should throw NullPointerException");
354             } catch (NullPointerException expected) {
355             }
356             t.cancel();
357 
358             // Ensure a Timer throws a NullPointerException if the date is null
359             t = new Timer();
360             testTask = new TimerTestTask();
361             try {
362                 t.schedule(testTask, null);
363                 fail("Scheduling a null date should throw NullPointerException");
364             } catch (NullPointerException expected) {
365             }
366             t.cancel();
367 
368             // Ensure proper sequence of exceptions
369             t = new Timer();
370             d = new Date(-100);
371             try {
372                 t.schedule(null, d);
373                 fail("Scheduling a null task with negative date should throw IllegalArgumentException first");
374             } catch (IllegalArgumentException expected) {
375             }
376             t.cancel();
377 
378             // Ensure a task is run
379             t = new Timer();
380             testTask = new TimerTestTask();
381             d = new Date(System.currentTimeMillis() + 200);
382             t.schedule(testTask, d);
383             awaitRun(testTask);
384             t.cancel();
385 
386             // Ensure multiple tasks are run
387             t = new Timer();
388             testTask = new TimerTestTask();
389             testTask.incrementCount(true);
390             d = new Date(System.currentTimeMillis() + 100);
391             t.schedule(testTask, d);
392             testTask = new TimerTestTask();
393             testTask.incrementCount(true);
394             d = new Date(System.currentTimeMillis() + 150);
395             t.schedule(testTask, d);
396             testTask = new TimerTestTask();
397             testTask.incrementCount(true);
398             d = new Date(System.currentTimeMillis() + 70);
399             t.schedule(testTask, d);
400             testTask = new TimerTestTask();
401             testTask.incrementCount(true);
402             d = new Date(System.currentTimeMillis() + 10);
403             t.schedule(testTask, d);
404             Thread.sleep(400);
405             assertTrue("Multiple tasks should have incremented counter 4 times not "
406                     + timerCounter, timerCounter == 4);
407             t.cancel();
408         } finally {
409             if (t != null)
410                 t.cancel();
411         }
412     }
413 
414     /**
415      * java.util.Timer#schedule(java.util.TimerTask, long)
416      */
test_scheduleLjava_util_TimerTaskJ()417     public void test_scheduleLjava_util_TimerTaskJ() throws Exception {
418         Timer t = null;
419         try {
420             // Ensure a Timer throws an IllegalStateException after cancelled
421             t = new Timer();
422             TimerTestTask testTask = new TimerTestTask();
423             t.cancel();
424             try {
425                 t.schedule(testTask, 100);
426                 fail("Scheduling a task after Timer.cancel() should throw exception");
427             } catch (IllegalStateException expected) {
428             }
429 
430             // Ensure a Timer throws an IllegalStateException if task already
431             // cancelled
432             t = new Timer();
433             testTask = new TimerTestTask();
434             testTask.cancel();
435             try {
436                 t.schedule(testTask, 100);
437                 fail("Scheduling a task after cancelling it should throw exception");
438             } catch (IllegalStateException expected) {
439             }
440             t.cancel();
441 
442             // Ensure a Timer throws an IllegalArgumentException if delay is
443             // negative
444             t = new Timer();
445             testTask = new TimerTestTask();
446             try {
447                 t.schedule(testTask, -100);
448                 fail("Scheduling a task with negative delay should throw IllegalArgumentException");
449             } catch (IllegalArgumentException expected) {
450             }
451             t.cancel();
452 
453             // Ensure a Timer throws a NullPointerException if the task is null
454             t = new Timer();
455             try {
456                 t.schedule(null, 10);
457                 fail("Scheduling a null task should throw NullPointerException");
458             } catch (NullPointerException expected) {
459             }
460             t.cancel();
461 
462             // Ensure proper sequence of exceptions
463             t = new Timer();
464             try {
465                 t.schedule(null, -10);
466                 fail("Scheduling a null task with negative delays should throw IllegalArgumentException first");
467             } catch (IllegalArgumentException expected) {
468             }
469             t.cancel();
470 
471             // Ensure a task is run
472             t = new Timer();
473             testTask = new TimerTestTask();
474             t.schedule(testTask, 200);
475             awaitRun(testTask);
476             t.cancel();
477 
478             // Ensure multiple tasks are run
479             t = new Timer();
480             testTask = new TimerTestTask();
481             testTask.incrementCount(true);
482             t.schedule(testTask, 100);
483             testTask = new TimerTestTask();
484             testTask.incrementCount(true);
485             t.schedule(testTask, 150);
486             testTask = new TimerTestTask();
487             testTask.incrementCount(true);
488             t.schedule(testTask, 70);
489             testTask = new TimerTestTask();
490             testTask.incrementCount(true);
491             t.schedule(testTask, 10);
492             Thread.sleep(400);
493             assertTrue("Multiple tasks should have incremented counter 4 times not "
494                     + timerCounter, timerCounter == 4);
495             t.cancel();
496         } finally {
497             if (t != null)
498                 t.cancel();
499         }
500     }
501 
502     /**
503      * java.util.Timer#schedule(java.util.TimerTask, long, long)
504      */
test_scheduleLjava_util_TimerTaskJJ()505     public void test_scheduleLjava_util_TimerTaskJJ() throws Exception {
506         Timer t = null;
507         try {
508             // Ensure a Timer throws an IllegalStateException after cancelled
509             t = new Timer();
510             TimerTestTask testTask = new TimerTestTask();
511             t.cancel();
512             try {
513                 t.schedule(testTask, 100, 100);
514                 fail("Scheduling a task after Timer.cancel() should throw exception");
515             } catch (IllegalStateException expected) {
516             }
517 
518             // Ensure a Timer throws an IllegalStateException if task already
519             // cancelled
520             t = new Timer();
521             testTask = new TimerTestTask();
522             testTask.cancel();
523             try {
524                 t.schedule(testTask, 100, 100);
525                 fail("Scheduling a task after cancelling it should throw exception");
526             } catch (IllegalStateException expected) {
527             }
528             t.cancel();
529 
530             // Ensure a Timer throws an IllegalArgumentException if delay is
531             // negative
532             t = new Timer();
533             testTask = new TimerTestTask();
534             try {
535                 t.schedule(testTask, -100, 100);
536                 fail("Scheduling a task with negative delay should throw IllegalArgumentException");
537             } catch (IllegalArgumentException expected) {
538             }
539             t.cancel();
540 
541             // Ensure a Timer throws an IllegalArgumentException if period is
542             // negative
543             t = new Timer();
544             testTask = new TimerTestTask();
545             try {
546                 t.schedule(testTask, 100, -100);
547                 fail("Scheduling a task with negative period should throw IllegalArgumentException");
548             } catch (IllegalArgumentException expected) {
549             }
550             t.cancel();
551 
552             // Ensure a Timer throws an IllegalArgumentException if period is
553             // zero
554             t = new Timer();
555             testTask = new TimerTestTask();
556             try {
557                 t.schedule(testTask, 100, 0);
558                 fail("Scheduling a task with 0 period should throw IllegalArgumentException");
559             } catch (IllegalArgumentException expected) {
560             }
561             t.cancel();
562 
563             // Ensure a Timer throws a NullPointerException if the task is null
564             t = new Timer();
565             try {
566                 t.schedule(null, 10, 10);
567                 fail("Scheduling a null task should throw NullPointerException");
568             } catch (NullPointerException expected) {
569             }
570             t.cancel();
571 
572             // Ensure proper sequence of exceptions
573             t = new Timer();
574             try {
575                 t.schedule(null, -10, -10);
576                 fail("Scheduling a null task with negative delays should throw IllegalArgumentException first");
577             } catch (IllegalArgumentException expected) {
578             }
579             t.cancel();
580 
581             // Ensure a task is run at least twice
582             t = new Timer();
583             testTask = new TimerTestTask();
584             t.schedule(testTask, 100, 100);
585             Thread.sleep(400);
586             assertTrue("TimerTask.run() method should have been called at least twice ("
587                     + testTask.wasRun() + ")", testTask.wasRun() >= 2);
588             t.cancel();
589 
590             // Ensure multiple tasks are run
591             t = new Timer();
592             testTask = new TimerTestTask();
593             testTask.incrementCount(true);
594             t.schedule(testTask, 100, 100); // at least 9 times
595             testTask = new TimerTestTask();
596             testTask.incrementCount(true);
597             t.schedule(testTask, 200, 100); // at least 7 times
598             testTask = new TimerTestTask();
599             testTask.incrementCount(true);
600             t.schedule(testTask, 300, 200); // at least 4 times
601             testTask = new TimerTestTask();
602             testTask.incrementCount(true);
603             t.schedule(testTask, 100, 200); // at least 4 times
604             Thread.sleep(1200); // Allowed more room for error
605             assertTrue("Multiple tasks should have incremented counter 24 times not "
606                     + timerCounter, timerCounter >= 24);
607             t.cancel();
608         } finally {
609             if (t != null)
610                 t.cancel();
611         }
612     }
613 
614     /**
615      * java.util.Timer#schedule(java.util.TimerTask, java.util.Date,
616      *        long)
617      */
test_scheduleLjava_util_TimerTaskLjava_util_DateJ()618     public void test_scheduleLjava_util_TimerTaskLjava_util_DateJ() throws Exception {
619         Timer t = null;
620         try {
621             // Ensure a Timer throws an IllegalStateException after cancelled
622             t = new Timer();
623             TimerTestTask testTask = new TimerTestTask();
624             Date d = new Date(System.currentTimeMillis() + 100);
625             t.cancel();
626             try {
627                 t.schedule(testTask, d, 100);
628                 fail("Scheduling a task after Timer.cancel() should throw exception");
629             } catch (IllegalStateException expected) {
630             }
631 
632             // Ensure a Timer throws an IllegalStateException if task already
633             // cancelled
634             t = new Timer();
635             d = new Date(System.currentTimeMillis() + 100);
636             testTask = new TimerTestTask();
637             testTask.cancel();
638             try {
639                 t.schedule(testTask, d, 100);
640                 fail("Scheduling a task after cancelling it should throw exception");
641             } catch (IllegalStateException expected) {
642             }
643             t.cancel();
644 
645             // Ensure a Timer throws an IllegalArgumentException if delay is
646             // negative
647             t = new Timer();
648             d = new Date(-100);
649             testTask = new TimerTestTask();
650             try {
651                 t.schedule(testTask, d, 100);
652                 fail("Scheduling a task with negative delay should throw IllegalArgumentException");
653             } catch (IllegalArgumentException expected) {
654             }
655             t.cancel();
656 
657             // Ensure a Timer throws an IllegalArgumentException if period is
658             // negative
659             t = new Timer();
660             d = new Date(System.currentTimeMillis() + 100);
661             testTask = new TimerTestTask();
662             try {
663                 t.schedule(testTask, d, -100);
664                 fail("Scheduling a task with negative period should throw IllegalArgumentException");
665             } catch (IllegalArgumentException expected) {
666             }
667             t.cancel();
668 
669             // Ensure a Timer throws a NullPointerException if the task is null
670             t = new Timer();
671             d = new Date(System.currentTimeMillis() + 100);
672             try {
673                 t.schedule(null, d, 10);
674                 fail("Scheduling a null task should throw NullPointerException");
675             } catch (NullPointerException expected) {
676             }
677             t.cancel();
678 
679             // Ensure a Timer throws a NullPointerException if the date is null
680             t = new Timer();
681             testTask = new TimerTestTask();
682             try {
683                 t.schedule(testTask, null, 10);
684                 fail("Scheduling a null task should throw NullPointerException");
685             } catch (NullPointerException expected) {
686             }
687             t.cancel();
688 
689             // Ensure proper sequence of exceptions
690             t = new Timer();
691             d = new Date(-100);
692             try {
693                 t.schedule(null, d, 10);
694                 fail("Scheduling a null task with negative dates should throw IllegalArgumentException first");
695             } catch (IllegalArgumentException expected) {
696             }
697             t.cancel();
698 
699             // Ensure a task is run at least twice
700             t = new Timer();
701             d = new Date(System.currentTimeMillis() + 100);
702             testTask = new TimerTestTask();
703             t.schedule(testTask, d, 100);
704             Thread.sleep(800);
705             assertTrue("TimerTask.run() method should have been called at least twice ("
706                     + testTask.wasRun() + ")", testTask.wasRun() >= 2);
707             t.cancel();
708 
709             // Ensure multiple tasks are run
710             t = new Timer();
711             testTask = new TimerTestTask();
712             testTask.incrementCount(true);
713             d = new Date(System.currentTimeMillis() + 100);
714             t.schedule(testTask, d, 200); // at least 4 times
715             testTask = new TimerTestTask();
716             testTask.incrementCount(true);
717             d = new Date(System.currentTimeMillis() + 300);
718             t.schedule(testTask, d, 200); // at least 4 times
719             testTask = new TimerTestTask();
720             testTask.incrementCount(true);
721             d = new Date(System.currentTimeMillis() + 500);
722             t.schedule(testTask, d, 400); // at least 2 times
723             testTask = new TimerTestTask();
724             testTask.incrementCount(true);
725             d = new Date(System.currentTimeMillis() + 100);
726             t.schedule(testTask, d, 400); // at least 2 times
727             Thread.sleep(3000);
728             assertTrue("Multiple tasks should have incremented counter 12 times not "
729                     + timerCounter, timerCounter >= 12);
730             t.cancel();
731         } finally {
732             if (t != null)
733                 t.cancel();
734         }
735     }
736 
737     /**
738      * java.util.Timer#scheduleAtFixedRate(java.util.TimerTask, long,
739      *        long)
740      */
test_scheduleAtFixedRateLjava_util_TimerTaskJJ()741     public void test_scheduleAtFixedRateLjava_util_TimerTaskJJ() throws Exception {
742         Timer t = null;
743         try {
744             // Ensure a Timer throws an IllegalStateException after cancelled
745             t = new Timer();
746             TimerTestTask testTask = new TimerTestTask();
747             t.cancel();
748             try {
749                 t.scheduleAtFixedRate(testTask, 100, 100);
750                 fail("scheduleAtFixedRate after Timer.cancel() should throw exception");
751             } catch (IllegalStateException expected) {
752             }
753 
754             // Ensure a Timer throws an IllegalArgumentException if delay is
755             // negative
756             t = new Timer();
757             testTask = new TimerTestTask();
758             try {
759                 t.scheduleAtFixedRate(testTask, -100, 100);
760                 fail("scheduleAtFixedRate with negative delay should throw IllegalArgumentException");
761             } catch (IllegalArgumentException expected) {
762             }
763             t.cancel();
764 
765             // Ensure a Timer throws an IllegalArgumentException if period is
766             // negative
767             t = new Timer();
768             testTask = new TimerTestTask();
769             try {
770                 t.scheduleAtFixedRate(testTask, 100, -100);
771                 fail("scheduleAtFixedRate with negative period should throw IllegalArgumentException");
772             } catch (IllegalArgumentException expected) {
773             }
774             t.cancel();
775 
776             // Ensure a task is run at least twice
777             t = new Timer();
778             testTask = new TimerTestTask();
779             t.scheduleAtFixedRate(testTask, 100, 100);
780             Thread.sleep(400);
781             assertTrue("TimerTask.run() method should have been called at least twice ("
782                     + testTask.wasRun() + ")", testTask.wasRun() >= 2);
783             t.cancel();
784 
785             class SlowThenFastTask extends TimerTask {
786                 int wasRun = 0;
787 
788                 long startedAt;
789 
790                 long lastDelta;
791 
792                 public void run() {
793                     if (wasRun == 0)
794                         startedAt = System.currentTimeMillis();
795                     lastDelta = System.currentTimeMillis()
796                             - (startedAt + (100 * wasRun));
797                     wasRun++;
798                     if (wasRun == 2) {
799                         try {
800                             Thread.sleep(200);
801                         } catch (InterruptedException e) {
802                             throw new RuntimeException(e);
803                         }
804                     }
805                 }
806 
807                 public long lastDelta() {
808                     return lastDelta;
809                 }
810 
811                 public int wasRun() {
812                     return wasRun;
813                 }
814             }
815 
816             // Ensure multiple tasks are run
817             t = new Timer();
818             SlowThenFastTask slowThenFastTask = new SlowThenFastTask();
819 
820             // at least 9 times even when asleep
821             t.scheduleAtFixedRate(slowThenFastTask, 100, 100);
822             Thread.sleep(1000);
823             long lastDelta = slowThenFastTask.lastDelta();
824             assertTrue("Fixed Rate Schedule should catch up, but is off by "
825                     + lastDelta + " ms", slowThenFastTask.lastDelta < 300);
826             t.cancel();
827         } finally {
828             if (t != null)
829                 t.cancel();
830         }
831     }
832 
833     /**
834      * java.util.Timer#scheduleAtFixedRate(java.util.TimerTask,
835      *        java.util.Date, long)
836      */
test_scheduleAtFixedRateLjava_util_TimerTaskLjava_util_DateJ()837     public void test_scheduleAtFixedRateLjava_util_TimerTaskLjava_util_DateJ() throws Exception {
838         Timer t = null;
839         try {
840             // Ensure a Timer throws an IllegalStateException after cancelled
841             t = new Timer();
842             TimerTestTask testTask = new TimerTestTask();
843             t.cancel();
844             Date d = new Date(System.currentTimeMillis() + 100);
845             try {
846                 t.scheduleAtFixedRate(testTask, d, 100);
847                 fail("scheduleAtFixedRate after Timer.cancel() should throw exception");
848             } catch (IllegalStateException expected) {
849             }
850 
851             // Ensure a Timer throws an IllegalArgumentException if delay is
852             // negative
853             t = new Timer();
854             testTask = new TimerTestTask();
855             d = new Date(-100);
856             try {
857                 t.scheduleAtFixedRate(testTask, d, 100);
858                 fail("scheduleAtFixedRate with negative Date should throw IllegalArgumentException");
859             } catch (IllegalArgumentException expected) {
860             }
861             t.cancel();
862 
863             // Ensure a Timer throws an IllegalArgumentException if period is
864             // negative
865             t = new Timer();
866             testTask = new TimerTestTask();
867             try {
868                 t.scheduleAtFixedRate(testTask, d, -100);
869                 fail("scheduleAtFixedRate with negative period should throw IllegalArgumentException");
870             } catch (IllegalArgumentException expected) {
871             }
872             t.cancel();
873 
874             // Ensure a Timer throws an NullPointerException if date is Null
875             t = new Timer();
876             testTask = new TimerTestTask();
877             try {
878                 t.scheduleAtFixedRate(testTask, null, 100);
879                 fail("scheduleAtFixedRate with null date should throw NullPointerException");
880             } catch (NullPointerException expected) {
881             }
882             t.cancel();
883 
884             // Ensure proper sequence of exceptions
885             t = new Timer();
886             d = new Date(-100);
887             try {
888                 t.scheduleAtFixedRate(null, d, 10);
889                 fail("Scheduling a null task with negative date should throw IllegalArgumentException first");
890             } catch (IllegalArgumentException expected) {
891             }
892             t.cancel();
893 
894             // Ensure proper sequence of exceptions
895             t = new Timer();
896             try {
897                 t.scheduleAtFixedRate(null, null, -10);
898                 fail("Scheduling a null task & null date & negative period should throw IllegalArgumentException first");
899             } catch (IllegalArgumentException expected) {
900             }
901             t.cancel();
902 
903             // Ensure a task is run at least twice
904             t = new Timer();
905             testTask = new TimerTestTask();
906             d = new Date(System.currentTimeMillis() + 100);
907             t.scheduleAtFixedRate(testTask, d, 100);
908             Thread.sleep(400);
909             assertTrue("TimerTask.run() method should have been called at least twice ("
910                     + testTask.wasRun() + ")", testTask.wasRun() >= 2);
911             t.cancel();
912 
913             class SlowThenFastTask extends TimerTask {
914                 int wasRun = 0;
915 
916                 long startedAt;
917 
918                 long lastDelta;
919 
920                 public void run() {
921                     if (wasRun == 0)
922                         startedAt = System.currentTimeMillis();
923                     lastDelta = System.currentTimeMillis()
924                             - (startedAt + (100 * wasRun));
925                     wasRun++;
926                     if (wasRun == 2) {
927                         try {
928                             Thread.sleep(200);
929                         } catch (InterruptedException e) {
930                             throw new RuntimeException(e);
931                         }
932                     }
933                 }
934 
935                 public long lastDelta() {
936                     return lastDelta;
937                 }
938 
939                 public int wasRun() {
940                     return wasRun;
941                 }
942             }
943 
944             // Ensure multiple tasks are run
945             t = new Timer();
946             SlowThenFastTask slowThenFastTask = new SlowThenFastTask();
947             d = new Date(System.currentTimeMillis() + 100);
948 
949             // at least 9 times even when asleep
950             t.scheduleAtFixedRate(slowThenFastTask, d, 100);
951             Thread.sleep(1000);
952             long lastDelta = slowThenFastTask.lastDelta();
953             assertTrue("Fixed Rate Schedule should catch up, but is off by "
954                     + lastDelta + " ms", lastDelta < 300);
955             t.cancel();
956         } finally {
957             if (t != null)
958                 t.cancel();
959         }
960     }
961 
962     /**
963      * We used to swallow RuntimeExceptions thrown by tasks. Instead, we need to
964      * let those exceptions bubble up, where they will both notify the thread's
965      * uncaught exception handler and terminate the timer's thread.
966      */
testThrowingTaskKillsTimerThread()967     public void testThrowingTaskKillsTimerThread() throws Exception {
968         final AtomicReference<Thread> threadRef = new AtomicReference<Thread>();
969         new Timer().schedule(new TimerTask() {
970             @Override
971             public void run() {
972                 Thread.currentThread().setUncaughtExceptionHandler(new UncaughtExceptionHandler() {
973                     @Override public void uncaughtException(Thread thread, Throwable ex) {}
974                 });
975                 threadRef.set(Thread.currentThread());
976                 throw new RuntimeException("task failure!");
977             }
978         }, 1);
979 
980         Thread.sleep(400);
981         Thread timerThread = threadRef.get();
982 
983         // Check if the timer thread is still alive every 10ms for 2 seconds
984         for (int i = 0; i < 200; i++) {
985             if (!timerThread.isAlive()) {
986                 break;
987             }
988             Thread.sleep(10);
989         }
990         assertFalse(timerThread.isAlive());
991     }
992 
993     private class CheckIfExecutedOnTime extends TimerTask {
994         private static final int TOLERANCE_TIME = 300;
995         private final AtomicBoolean executedOnTime;
996 
997         static final int SLEEPING_TIME = 10 * TOLERANCE_TIME;
998 
CheckIfExecutedOnTime(AtomicBoolean executedOnTime)999         private CheckIfExecutedOnTime(AtomicBoolean executedOnTime) {
1000             this.executedOnTime = executedOnTime;
1001         }
1002 
1003         @Override
run()1004         public void run() {
1005             // We'll schedule one after the other to execute immediately, the first one with
1006             // {@code executedOnTime == null}. Ensure that the second
1007             // is delayed by at most the time spent by the first one, plus some tolerance.
1008             if (executedOnTime != null &&
1009                     System.currentTimeMillis()
1010                             <= scheduledExecutionTime() + SLEEPING_TIME + TOLERANCE_TIME) {
1011                 executedOnTime.set(true);
1012             } else {
1013                 try {
1014                     Thread.sleep(SLEEPING_TIME);
1015                 } catch (InterruptedException e) {
1016                     throw new IllegalStateException(e);
1017                 }
1018             }
1019         }
1020     };
1021 
testOverdueTaskExecutesImmediately()1022     public void testOverdueTaskExecutesImmediately() throws Exception {
1023         Timer t = new Timer();
1024         Date date = new Date(System.currentTimeMillis());
1025         t.schedule(new CheckIfExecutedOnTime(null), date);
1026         AtomicBoolean actuallyExecutedOnTime = new AtomicBoolean();
1027         // Scheduled to execute right now but won't do as the other task is sleeping. Check that
1028         // this one executes as soon as the other one finishes.
1029         t.schedule(new CheckIfExecutedOnTime(actuallyExecutedOnTime), date);
1030         // Only the first one sleeps, this will be the two tasks plenty of time to finish.
1031         Thread.sleep(2 * CheckIfExecutedOnTime.SLEEPING_TIME);
1032         t.cancel();
1033         assertTrue(actuallyExecutedOnTime.get());
1034     }
1035 
testCanBeCancelledEvenIfTaskKeepsItPermanentlyBusy()1036     public void testCanBeCancelledEvenIfTaskKeepsItPermanentlyBusy() throws Exception {
1037         final int timeSleeping = 200;
1038         Timer t = new Timer();
1039         final AtomicLong counter = new AtomicLong();
1040         TimerTask task = new TimerTask() {
1041             @Override
1042             public void run() {
1043                 try {
1044                     counter.incrementAndGet();
1045                     Thread.sleep(timeSleeping);
1046                 } catch (InterruptedException e) {
1047                     throw new IllegalStateException(e);
1048                 }
1049             }
1050         };
1051         // Keep the thread busy by scheduling execution twice as fast than the task can execute.
1052         t.scheduleAtFixedRate(task, 1 /* delay */, timeSleeping / 2 /* rate */);
1053         Thread.sleep(timeSleeping * 8);
1054         // Check the task was actually running.
1055         assertTrue(counter.get() > 0);
1056         t.cancel();
1057         // Allow some time to finish.
1058         Thread.sleep(2 * timeSleeping);
1059         try {
1060             t.schedule(
1061                     new TimerTask() {
1062                         @Override
1063                         public void run() {
1064 
1065                         }
1066                     },
1067                     1 /* delay */);
1068             fail("timer should be cancelled, and not accept new schedulings");
1069         } catch (IllegalStateException expected) {
1070             // Expected.
1071         }
1072     }
1073 
testTaskNotCancelledWhenTimerCancelled()1074     public void testTaskNotCancelledWhenTimerCancelled() throws Exception {
1075         final int timeSleeping = 200;
1076         Timer t = new Timer();
1077         final AtomicLong counter = new AtomicLong();
1078         TimerTask task = new TimerTask() {
1079             @Override
1080             public void run() {
1081                 try {
1082                     counter.incrementAndGet();
1083                     Thread.sleep(timeSleeping);
1084                 } catch (InterruptedException e) {
1085                     throw new IllegalStateException(e);
1086                 }
1087             }
1088         };
1089         t.scheduleAtFixedRate(task, 1 /* delay */, 100 /* rate */);
1090         Thread.sleep(1000);
1091         t.cancel();
1092         // Returns true as the task wasn't cancelled before.
1093         assertTrue(task.cancel());
1094     }
1095 
testTaskNotCancelledWhenTimerCancelledAndPurged()1096     public void testTaskNotCancelledWhenTimerCancelledAndPurged() throws Exception {
1097         final int timeSleeping = 200;
1098         Timer t = new Timer();
1099         final AtomicLong counter = new AtomicLong();
1100         TimerTask task = new TimerTask() {
1101             @Override
1102             public void run() {
1103                 try {
1104                     counter.incrementAndGet();
1105                     Thread.sleep(timeSleeping);
1106                 } catch (InterruptedException e) {
1107                     throw new IllegalStateException(e);
1108                 }
1109             }
1110         };
1111         t.scheduleAtFixedRate(task, 1 /* delay */, 100 /* rate */);
1112         Thread.sleep(1000);
1113         t.cancel();
1114         t.purge();
1115         // Returns true as the task wasn't cancelled before.
1116         assertTrue(task.cancel());
1117     }
1118 
1119     private static class IncrementCounterTaskAndPossiblyThrowAfter extends TimerTask {
1120 
1121         private final AtomicLong counter;
1122         private final int incrementAmount;
1123         private final boolean willThrow;
1124 
1125 
IncrementCounterTaskAndPossiblyThrowAfter( AtomicLong counter, int incrementAmount, boolean willThrow)1126         IncrementCounterTaskAndPossiblyThrowAfter(
1127                 AtomicLong counter, int incrementAmount, boolean willThrow) {
1128             this.counter = counter;
1129             this.incrementAmount = incrementAmount;
1130             this.willThrow = willThrow;
1131         }
1132 
1133         @Override
run()1134         public void run() {
1135             counter.addAndGet(incrementAmount);
1136             if (willThrow) {
1137                 throw new IllegalStateException("TimerTask runtime exception from run()");
1138             }
1139         }
1140     }
1141 
1142     private static class SwallowUncaughtExceptionHandler implements UncaughtExceptionHandler {
1143         CountDownLatch latch = new CountDownLatch(1);
1144         @Override
uncaughtException(Thread thread, Throwable ex)1145         public void uncaughtException(Thread thread, Throwable ex) {
1146             latch.countDown();
1147         }
1148 
waitForException(long millis)1149         void waitForException(long millis) throws InterruptedException {
1150             if(!latch.await(millis, TimeUnit.MILLISECONDS)) {
1151                 throw new AssertionError("Expected exception thrown from timer thread");
1152             }
1153         }
1154     }
1155 
testTimerCancelledAfterException()1156     public void testTimerCancelledAfterException() throws Exception {
1157         UncaughtExceptionHandler excHandler = Thread.getDefaultUncaughtExceptionHandler();
1158         // Install an uncaught exception handler because we are
1159         // deliberately causing the timer thread to die in this test (which will cause CTS tests
1160         // to fail).
1161         SwallowUncaughtExceptionHandler swallowUncaughtExceptionHandler =
1162                 new SwallowUncaughtExceptionHandler();
1163         Thread.setDefaultUncaughtExceptionHandler(swallowUncaughtExceptionHandler);
1164         try {
1165             Timer t = new Timer();
1166             final AtomicLong counter = new AtomicLong();
1167 
1168             // Schedule tasks to run:
1169             // A) {In 1 millis} Increment a counter by 1 and throw an exception
1170             // B) {In 100 millis} Increment a counter by 1000 (but it's not intended to be executed
1171             //        because of the previous exception).
1172             // We want A and B to be scheduled before A runs.
1173             // We add them in reverse order.
1174             // We have ~99 millis after scheduling B to schedule A. If A ran before we scheduled B
1175             // we would get an exception when we came to schedule B.
1176             TimerTask taskThatDoesntThrow = new IncrementCounterTaskAndPossiblyThrowAfter(
1177                     counter,
1178                     1000,  /* incrementAmount */
1179                     false /* willThrow */);
1180             TimerTask taskThatThrows = new IncrementCounterTaskAndPossiblyThrowAfter(
1181                     counter,
1182                     1,    /* incrementAmount */
1183                     true  /* willThrow */);
1184             t.schedule(taskThatDoesntThrow, 100 /* delay */);
1185             t.scheduleAtFixedRate(taskThatThrows, 1 /* delay */, 100 /* period */);
1186 
1187             swallowUncaughtExceptionHandler.waitForException(1000);
1188             // Check the counter wasn't increased more than once (ie, the exception killed the
1189             // execution thread).
1190             assertEquals("Counter should be 1, and is: " + counter.get(), 1, counter.get());
1191 
1192             assertTrue("The timer should not cancel the tasks", taskThatDoesntThrow.cancel());
1193             assertTrue("The timer should not cancel the tasks", taskThatThrows.cancel());
1194 
1195             TimerTask otherTask = new TimerTask() {
1196                 @Override
1197                 public void run() {
1198                     counter.incrementAndGet();
1199                 }
1200             };
1201 
1202             try {
1203                 t.schedule(otherTask, 1);
1204                 fail("Timer should be cancelled and no new tasks should be allowed");
1205             } catch (Exception expected) {
1206                 // Expected.
1207             }
1208         } finally {
1209             Thread.setDefaultUncaughtExceptionHandler(excHandler);
1210         }
1211     }
1212 
testTimerCancelledAfterExceptionAndTasksNotCancelledAfterPurge()1213     public void testTimerCancelledAfterExceptionAndTasksNotCancelledAfterPurge() throws Exception {
1214         UncaughtExceptionHandler excHandler = Thread.getDefaultUncaughtExceptionHandler();
1215         // Install an uncaught exception handler because we are
1216         // deliberately causing the timer thread to die in this test (which will cause CTS tests
1217         // to fail).
1218         SwallowUncaughtExceptionHandler swallowUncaughtExceptionHandler =
1219                 new SwallowUncaughtExceptionHandler();
1220         Thread.setDefaultUncaughtExceptionHandler(swallowUncaughtExceptionHandler);
1221         try {
1222             Timer t = new Timer();
1223             final AtomicLong counter = new AtomicLong();
1224 
1225             // Schedule tasks to run:
1226             // A) {In 1 millis} Increment a counter by 1 and throw an exception
1227             // B) {In 100 millis} Increment a counter by 1000 (but it's not intended to be executed
1228             //        because of the previous exception).
1229             // We want A and B to be scheduled before A runs.
1230             // We add them in reverse order.
1231             // We have ~99 millis after scheduling B to schedule A. If A ran before we scheduled B
1232             // we would get an exception when we came to schedule B.
1233 
1234             TimerTask taskThatDoesntThrow = new IncrementCounterTaskAndPossiblyThrowAfter(
1235                     counter,
1236                     1000, /* incrementAmount */
1237                     false /* willThrow */);
1238             TimerTask taskThatThrows = new IncrementCounterTaskAndPossiblyThrowAfter(
1239                     counter,
1240                     1,    /* incrementAmount */
1241                     true  /* willThrow */);
1242             t.schedule(taskThatDoesntThrow, 100 /* delay */);
1243             t.scheduleAtFixedRate(taskThatThrows, 1 /* delay */, 100 /* period */);
1244             swallowUncaughtExceptionHandler.waitForException(1000);
1245             // Check the counter wasn't increased more than once (ie, the exception killed the
1246             // execution thread).
1247             assertEquals("Counter should be 1, and is: " + counter.get(), 1, counter.get());
1248             t.purge();
1249             assertTrue("The timer should not cancel the tasks", taskThatDoesntThrow.cancel());
1250             assertTrue("The timer should not cancel the tasks", taskThatThrows.cancel());
1251 
1252             TimerTask otherTask = new TimerTask() {
1253                 @Override
1254                 public void run() {
1255                     counter.incrementAndGet();
1256                 }
1257             };
1258 
1259             try {
1260                 t.schedule(otherTask, 1);
1261                 fail("Timer should be cancelled and no new tasks should be allowed");
1262             } catch (Exception expected) {
1263                 // Expected.
1264             }
1265         } finally {
1266             Thread.setDefaultUncaughtExceptionHandler(excHandler);
1267         }
1268     }
1269 
testTimerCancelledTasksRemovedFromQueue()1270     public void testTimerCancelledTasksRemovedFromQueue() throws Exception {
1271         Timer t = new Timer();
1272         TimerTask task1 = new TimerTask() {
1273             @Override
1274             public void run() {
1275             }
1276         };
1277         t.scheduleAtFixedRate(task1, 1 /* delay */, 10 /* period */);
1278 
1279         task1.cancel();
1280         // As the rate is 10, the timer will try to schedule it before the purge and remove it.
1281         Thread.sleep(500);
1282         assertEquals(0, t.purge());
1283     }
1284 
setUp()1285     protected void setUp() {
1286         timerCounter = 0;
1287     }
1288 
tearDown()1289     protected void tearDown() {
1290     }
1291 }
1292