e79aa3c0ed
Former-commit-id: a2155e9bd80020e49e72e86c44da02a8ac0e57a4
397 lines
9.3 KiB
C#
397 lines
9.3 KiB
C#
//
|
|
// TimerTest.cs - NUnit Test Cases for System.Timers.Timer
|
|
//
|
|
// Author:
|
|
// Kornél Pál <http://www.kornelpal.hu/>
|
|
// Robert Jordan <robertj@gmx.net>
|
|
//
|
|
// Copyright (C) 2005 Kornél Pál
|
|
//
|
|
|
|
//
|
|
// Permission is hereby granted, free of charge, to any person obtaining
|
|
// a copy of this software and associated documentation files (the
|
|
// "Software"), to deal in the Software without restriction, including
|
|
// without limitation the rights to use, copy, modify, merge, publish,
|
|
// distribute, sublicense, and/or sell copies of the Software, and to
|
|
// permit persons to whom the Software is furnished to do so, subject to
|
|
// the following conditions:
|
|
//
|
|
// The above copyright notice and this permission notice shall be
|
|
// included in all copies or substantial portions of the Software.
|
|
//
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
|
// EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
|
// NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
|
// LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
|
// OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
|
// WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
//
|
|
|
|
using NUnit.Framework;
|
|
using System;
|
|
using System.Timers;
|
|
using ST = System.Threading;
|
|
|
|
namespace MonoTests.System.Timers
|
|
{
|
|
[TestFixture]
|
|
public class TimerTest
|
|
{
|
|
Timer timer;
|
|
|
|
[SetUp]
|
|
public void SetUp ()
|
|
{
|
|
timer = new Timer ();
|
|
}
|
|
|
|
[TearDown]
|
|
public void TearDown ()
|
|
{
|
|
timer.Close ();
|
|
}
|
|
|
|
[Test]
|
|
public void Constructor0 ()
|
|
{
|
|
Assert.IsTrue (timer.AutoReset, "#1");
|
|
Assert.IsFalse (timer.Enabled, "#2");
|
|
Assert.AreEqual (100, timer.Interval, "#3");
|
|
Assert.IsNull (timer.SynchronizingObject, "#4");
|
|
}
|
|
|
|
[Test]
|
|
public void Constructor1 ()
|
|
{
|
|
timer = new Timer (1);
|
|
Assert.IsTrue (timer.AutoReset, "#A1");
|
|
Assert.IsFalse (timer.Enabled, "#A2");
|
|
Assert.AreEqual (1, timer.Interval, "#A3");
|
|
Assert.IsNull (timer.SynchronizingObject, "#A4");
|
|
|
|
timer = new Timer (int.MaxValue);
|
|
Assert.IsTrue (timer.AutoReset, "#B1");
|
|
Assert.IsFalse (timer.Enabled, "#B2");
|
|
Assert.AreEqual (int.MaxValue, timer.Interval, "#B3");
|
|
Assert.IsNull (timer.SynchronizingObject, "#B4");
|
|
}
|
|
|
|
[Test]
|
|
public void Constructor1_Interval_Negative ()
|
|
{
|
|
try {
|
|
new Timer (-1);
|
|
Assert.Fail ("#1");
|
|
} catch (ArgumentException ex) {
|
|
// Invalid value -1 for parameter interval
|
|
Assert.AreEqual (typeof (ArgumentException), ex.GetType (), "#2");
|
|
Assert.IsNull (ex.InnerException, "#3");
|
|
Assert.IsNotNull (ex.Message, "#4");
|
|
}
|
|
}
|
|
|
|
[Test]
|
|
public void Constructor1_Interval_Zero ()
|
|
{
|
|
try {
|
|
new Timer (0);
|
|
Assert.Fail ("#1");
|
|
} catch (ArgumentException ex) {
|
|
// Invalid value 0 for parameter interval
|
|
Assert.AreEqual (typeof (ArgumentException), ex.GetType (), "#2");
|
|
Assert.IsNull (ex.InnerException, "#3");
|
|
Assert.IsNotNull (ex.Message, "#4");
|
|
}
|
|
}
|
|
|
|
[Test]
|
|
public void Constructor1_Interval_Max ()
|
|
{
|
|
try {
|
|
new Timer (0x80000000);
|
|
Assert.Fail ("#A1");
|
|
} catch (ArgumentException ex) {
|
|
// Invalid value 2147483648 for parameter interval
|
|
Assert.AreEqual (typeof (ArgumentException), ex.GetType (), "#A2");
|
|
Assert.IsNull (ex.InnerException, "#A3");
|
|
Assert.IsNotNull (ex.Message, "#A4");
|
|
}
|
|
|
|
try {
|
|
new Timer (double.MaxValue);
|
|
Assert.Fail ("#B1");
|
|
} catch (ArgumentException ex) {
|
|
// Invalid value 1.79769313486232E+308 for parameter interval
|
|
Assert.AreEqual (typeof (ArgumentException), ex.GetType (), "#B2");
|
|
Assert.IsNull (ex.InnerException, "#B3");
|
|
Assert.IsNotNull (ex.Message, "#B4");
|
|
}
|
|
}
|
|
|
|
[Test]
|
|
[ExpectedException (typeof (ArgumentException))]
|
|
public void Constructor1_Interval_Max_2 ()
|
|
{
|
|
timer = new Timer (double.MaxValue);
|
|
}
|
|
|
|
[Test]
|
|
[ExpectedException (typeof (ArgumentException))]
|
|
public void Constructor1_Interval_Min_1 ()
|
|
{
|
|
timer = new Timer (0);
|
|
}
|
|
|
|
[Test]
|
|
[ExpectedException (typeof (ArgumentException))]
|
|
public void Constructor1_Interval_Min_2 ()
|
|
{
|
|
timer = new Timer (-5);
|
|
}
|
|
|
|
[Test]
|
|
public void Interval_TooHigh_Disabled_NoThrow ()
|
|
{
|
|
timer.Interval = double.MaxValue;
|
|
Assert.AreEqual (double.MaxValue, timer.Interval, "#3");
|
|
}
|
|
|
|
[Test]
|
|
public void Interval_TooHigh_ThrowOnEnabled ()
|
|
{
|
|
timer.Interval = 0x80000000;
|
|
Assert.AreEqual (0x80000000, timer.Interval, "#1");
|
|
try {
|
|
timer.Enabled = true;
|
|
Assert.Fail ("#2");
|
|
} catch (Exception ex) {
|
|
Assert.AreEqual (typeof (ArgumentOutOfRangeException), ex.GetType (), "#3");
|
|
Assert.IsTrue (timer.Enabled);
|
|
}
|
|
}
|
|
|
|
[Test]
|
|
[ExpectedException (typeof (ArgumentOutOfRangeException))]
|
|
public void Interval_TooHigh_Enabled_Throw ()
|
|
{
|
|
timer.Interval = 100;
|
|
timer.Enabled = true;
|
|
timer.Interval = double.MaxValue;
|
|
}
|
|
|
|
[Test]
|
|
public void DoubleClose_NoThrow ()
|
|
{
|
|
timer.Interval = 100;
|
|
timer.Start ();
|
|
timer.Close ();
|
|
timer.Close ();
|
|
}
|
|
|
|
[Test]
|
|
public void DisposedMeansDisabled_NoThrow ()
|
|
{
|
|
timer.Interval = 100;
|
|
timer.Start ();
|
|
timer.Close ();
|
|
Assert.IsFalse (timer.Enabled);
|
|
}
|
|
|
|
[Test]
|
|
public void Disposed_ThrowOnEnabled ()
|
|
{
|
|
timer.Interval = 100;
|
|
timer.Start ();
|
|
timer.Close ();
|
|
timer.Enabled = false;
|
|
}
|
|
|
|
[Test]
|
|
public void Elapsed_DontFireIfDisposed ()
|
|
{
|
|
timer.Interval = 500;
|
|
var countElapsedCalls = 0;
|
|
timer.Elapsed += (_, __) => { countElapsedCalls++; };
|
|
timer.Start ();
|
|
timer.Close ();
|
|
ST.Thread.Sleep (500);
|
|
Assert.AreEqual (countElapsedCalls, 0);
|
|
}
|
|
|
|
[Test]
|
|
public void AutoReset ()
|
|
{
|
|
Assert.IsTrue (timer.AutoReset, "#1");
|
|
timer.AutoReset = false;
|
|
Assert.IsFalse (timer.AutoReset, "#2");
|
|
}
|
|
|
|
[Test]
|
|
public void Interval ()
|
|
{
|
|
timer.Interval = 1;
|
|
Assert.AreEqual (1, timer.Interval, "#1");
|
|
timer.Interval = 500;
|
|
Assert.AreEqual (500, timer.Interval, "#2");
|
|
timer.Interval = double.MaxValue;
|
|
Assert.AreEqual (double.MaxValue, timer.Interval, "#3");
|
|
}
|
|
|
|
[Test]
|
|
public void Interval_Negative ()
|
|
{
|
|
try {
|
|
timer.Interval = -1;
|
|
Assert.Fail ("#1");
|
|
} catch (ArgumentException ex) {
|
|
// '0' is not a valid value for 'Interval'. 'Interval' must be greater than 0
|
|
Assert.AreEqual (typeof (ArgumentException), ex.GetType (), "#2");
|
|
Assert.IsNull (ex.InnerException, "#3");
|
|
Assert.IsNotNull (ex.Message, "#4");
|
|
}
|
|
}
|
|
|
|
[Test]
|
|
public void Interval_Zero ()
|
|
{
|
|
try {
|
|
timer.Interval = 0;
|
|
Assert.Fail ("#1");
|
|
} catch (ArgumentException ex) {
|
|
// '0' is not a valid value for 'Interval'. 'Interval' must be greater than 0
|
|
Assert.AreEqual (typeof (ArgumentException), ex.GetType (), "#2");
|
|
Assert.IsNull (ex.InnerException, "#3");
|
|
Assert.IsNotNull (ex.Message, "#4");
|
|
}
|
|
}
|
|
|
|
[Test]
|
|
public void StartStopEnabled ()
|
|
{
|
|
timer.Start ();
|
|
Assert.IsTrue (timer.Enabled, "#1");
|
|
timer.Stop ();
|
|
Assert.IsFalse (timer.Enabled, "#2");
|
|
}
|
|
|
|
[Test]
|
|
public void CloseEnabled ()
|
|
{
|
|
Assert.IsFalse (timer.Enabled, "#1");
|
|
timer.Enabled = true;
|
|
Assert.IsTrue (timer.Enabled, "#2");
|
|
timer.Close ();
|
|
Assert.IsFalse (timer.Enabled, "#3");
|
|
}
|
|
|
|
[Test] // bug https://bugzilla.novell.com/show_bug.cgi?id=325368
|
|
public void EnabledInElapsed ()
|
|
{
|
|
var elapsedCount = 0;
|
|
var mre = new ST.ManualResetEventSlim ();
|
|
timer = new Timer (50);
|
|
timer.AutoReset = false;
|
|
timer.Elapsed += (s, e) =>
|
|
{
|
|
elapsedCount++;
|
|
if (elapsedCount == 1)
|
|
timer.Enabled = true;
|
|
else if (elapsedCount == 2)
|
|
mre.Set ();
|
|
};
|
|
timer.Start ();
|
|
|
|
Assert.IsTrue (mre.Wait (1000), "#1 re-enabling timer in Elapsed didn't work");
|
|
Assert.AreEqual (2, elapsedCount, "#2 wrong elapsedCount");
|
|
timer.Stop ();
|
|
}
|
|
|
|
[Test]
|
|
public void AutoResetEventFalseStopsFiringElapsed ()
|
|
{
|
|
var elapsedCount = 0;
|
|
var mre = new ST.ManualResetEventSlim ();
|
|
timer = new Timer (50);
|
|
timer.AutoReset = false;
|
|
timer.Elapsed += (s, e) =>
|
|
{
|
|
elapsedCount++;
|
|
if (elapsedCount > 1)
|
|
mre.Set ();
|
|
};
|
|
timer.Start ();
|
|
|
|
Assert.IsFalse (mre.Wait (1000), "#1 AutoReset=false didn't stop firing Elapsed, elapsedCount=" + elapsedCount);
|
|
Assert.AreEqual (1, elapsedCount, "#2 wrong elapsedCount");
|
|
timer.Stop ();
|
|
}
|
|
|
|
[Test]
|
|
public void TestRaceCondition ()
|
|
{
|
|
Assert.IsTrue (new RaceTest (true).Success, "#1");
|
|
Assert.IsTrue (new RaceTest (false).Success, "#2");
|
|
}
|
|
}
|
|
|
|
class RaceTest
|
|
{
|
|
const int Threads = 2;
|
|
const int Loops = 100;
|
|
|
|
object locker = new object ();
|
|
Timer timer;
|
|
int counter;
|
|
|
|
public bool Success {
|
|
get { return counter > Loops * Threads; }
|
|
}
|
|
|
|
public RaceTest (bool autoReset)
|
|
{
|
|
timer = new Timer ();
|
|
timer.AutoReset = autoReset;
|
|
timer.Interval = 100;
|
|
timer.Elapsed += new ElapsedEventHandler (Tick);
|
|
timer.Start ();
|
|
|
|
ST.Thread[] tl = new ST.Thread [Threads];
|
|
|
|
for (int i = 0; i < Threads; i++) {
|
|
tl [i] = new ST.Thread (new ST.ThreadStart (Run));
|
|
tl [i].Start ();
|
|
}
|
|
|
|
for (int i = 0; i < Threads; i++) {
|
|
tl [i].Join ();
|
|
}
|
|
|
|
ST.Thread.Sleep (1000);
|
|
}
|
|
|
|
void Restart ()
|
|
{
|
|
lock (locker) {
|
|
timer.Stop ();
|
|
timer.Start ();
|
|
}
|
|
ST.Interlocked.Increment (ref counter);
|
|
}
|
|
|
|
void Tick (object sender, ElapsedEventArgs e)
|
|
{
|
|
Restart ();
|
|
}
|
|
|
|
void Run ()
|
|
{
|
|
for (int i = 0; i < Loops; i++) {
|
|
ST.Thread.Sleep (0);
|
|
Restart ();
|
|
}
|
|
}
|
|
}
|
|
}
|