blob: 61d864a64857f7ab90d7a5fd3ff16db584988340 [file] [log] [blame]
Guido van Rossum2512d6d2000-04-24 14:01:51 +00001# Ridiculously simple test of the winsound module for Windows.
Guido van Rossumcdd092f2000-04-21 21:28:47 +00002
Walter Dörwald7fd94242003-05-18 00:47:47 +00003import unittest
Benjamin Petersonee8712c2008-05-20 21:35:26 +00004from test import support
5support.requires('audio')
R. David Murraya21e4ca2009-03-31 23:16:50 +00006import time
Thomas Wouters49fd7fa2006-04-21 10:40:58 +00007import os
8import subprocess
9
R. David Murraya21e4ca2009-03-31 23:16:50 +000010winsound = support.import_module('winsound')
Antoine Pitrou817c9df2010-08-10 00:45:32 +000011ctypes = support.import_module('ctypes')
Mark Dickinsona09543c2010-04-13 11:43:10 +000012import winreg
R. David Murraya21e4ca2009-03-31 23:16:50 +000013
Brian Curtin36067602010-04-13 02:32:40 +000014def has_sound(sound):
15 """Find out if a particular event is configured with a default sound"""
16 try:
Brian Curtinae16af22010-06-01 13:49:19 +000017 # Ask the mixer API for the number of devices it knows about.
18 # When there are no devices, PlaySound will fail.
Benjamin Petersona5119352012-10-09 11:14:59 -040019 if ctypes.windll.winmm.mixerGetNumDevs() == 0:
Brian Curtinae16af22010-06-01 13:49:19 +000020 return False
21
Brian Curtinf1407182010-04-13 12:43:07 +000022 key = winreg.OpenKeyEx(winreg.HKEY_CURRENT_USER,
Brian Curtin36067602010-04-13 02:32:40 +000023 "AppEvents\Schemes\Apps\.Default\{0}\.Default".format(sound))
Benjamin Petersona5119352012-10-09 11:14:59 -040024 return winreg.EnumValue(key, 0)[1] != ""
Andrew Svetlov2606a6f2012-12-19 14:33:35 +020025 except OSError:
Brian Curtin36067602010-04-13 02:32:40 +000026 return False
Walter Dörwald7fd94242003-05-18 00:47:47 +000027
28class BeepTest(unittest.TestCase):
Christian Heimesdd15f6c2008-03-16 00:07:10 +000029 # As with PlaySoundTest, incorporate the _have_soundcard() check
30 # into our test methods. If there's no audio device present,
31 # winsound.Beep returns 0 and GetLastError() returns 127, which
32 # is: ERROR_PROC_NOT_FOUND ("The specified procedure could not
33 # be found"). (FWIW, virtual/Hyper-V systems fall under this
34 # scenario as they have no sound devices whatsoever (not even
35 # a legacy Beep device).)
Walter Dörwald7fd94242003-05-18 00:47:47 +000036
37 def test_errors(self):
38 self.assertRaises(TypeError, winsound.Beep)
39 self.assertRaises(ValueError, winsound.Beep, 36, 75)
40 self.assertRaises(ValueError, winsound.Beep, 32768, 75)
41
42 def test_extremes(self):
Christian Heimesd5e2b6f2008-03-19 21:50:51 +000043 self._beep(37, 75)
44 self._beep(32767, 75)
Walter Dörwald7fd94242003-05-18 00:47:47 +000045
46 def test_increasingfrequency(self):
Amaury Forgeot d'Arc0a665ce2008-03-20 01:02:48 +000047 for i in range(100, 2000, 100):
Christian Heimesd5e2b6f2008-03-19 21:50:51 +000048 self._beep(i, 75)
49
50 def _beep(self, *args):
51 # these tests used to use _have_soundcard(), but it's quite
52 # possible to have a soundcard, and yet have the beep driver
53 # disabled. So basically, we have no way of knowing whether
54 # a beep should be produced or not, so currently if these
55 # tests fail we're ignoring them
56 #
57 # XXX the right fix for this is to define something like
58 # _have_enabled_beep_driver() and use that instead of the
59 # try/except below
60 try:
61 winsound.Beep(*args)
62 except RuntimeError:
63 pass
Walter Dörwald7fd94242003-05-18 00:47:47 +000064
65class MessageBeepTest(unittest.TestCase):
66
67 def tearDown(self):
68 time.sleep(0.5)
69
70 def test_default(self):
71 self.assertRaises(TypeError, winsound.MessageBeep, "bad")
72 self.assertRaises(TypeError, winsound.MessageBeep, 42, 42)
73 winsound.MessageBeep()
74
75 def test_ok(self):
76 winsound.MessageBeep(winsound.MB_OK)
77
78 def test_asterisk(self):
79 winsound.MessageBeep(winsound.MB_ICONASTERISK)
80
81 def test_exclamation(self):
82 winsound.MessageBeep(winsound.MB_ICONEXCLAMATION)
83
84 def test_hand(self):
85 winsound.MessageBeep(winsound.MB_ICONHAND)
86
87 def test_question(self):
88 winsound.MessageBeep(winsound.MB_ICONQUESTION)
89
Thomas Wouters49fd7fa2006-04-21 10:40:58 +000090
Walter Dörwald7fd94242003-05-18 00:47:47 +000091class PlaySoundTest(unittest.TestCase):
92
93 def test_errors(self):
94 self.assertRaises(TypeError, winsound.PlaySound)
95 self.assertRaises(TypeError, winsound.PlaySound, "bad", "bad")
96 self.assertRaises(
97 RuntimeError,
98 winsound.PlaySound,
99 "none", winsound.SND_ASYNC | winsound.SND_MEMORY
100 )
101
Brian Curtin36067602010-04-13 02:32:40 +0000102 @unittest.skipUnless(has_sound("SystemAsterisk"),
103 "No default SystemAsterisk")
Walter Dörwald7fd94242003-05-18 00:47:47 +0000104 def test_alias_asterisk(self):
Thomas Wouters49fd7fa2006-04-21 10:40:58 +0000105 if _have_soundcard():
106 winsound.PlaySound('SystemAsterisk', winsound.SND_ALIAS)
107 else:
108 self.assertRaises(
109 RuntimeError,
110 winsound.PlaySound,
111 'SystemAsterisk', winsound.SND_ALIAS
112 )
Walter Dörwald7fd94242003-05-18 00:47:47 +0000113
Brian Curtin36067602010-04-13 02:32:40 +0000114 @unittest.skipUnless(has_sound("SystemExclamation"),
115 "No default SystemExclamation")
Walter Dörwald7fd94242003-05-18 00:47:47 +0000116 def test_alias_exclamation(self):
Thomas Wouters49fd7fa2006-04-21 10:40:58 +0000117 if _have_soundcard():
118 winsound.PlaySound('SystemExclamation', winsound.SND_ALIAS)
119 else:
120 self.assertRaises(
121 RuntimeError,
122 winsound.PlaySound,
123 'SystemExclamation', winsound.SND_ALIAS
124 )
Walter Dörwald7fd94242003-05-18 00:47:47 +0000125
Brian Curtin36067602010-04-13 02:32:40 +0000126 @unittest.skipUnless(has_sound("SystemExit"), "No default SystemExit")
Walter Dörwald7fd94242003-05-18 00:47:47 +0000127 def test_alias_exit(self):
Thomas Wouters49fd7fa2006-04-21 10:40:58 +0000128 if _have_soundcard():
129 winsound.PlaySound('SystemExit', winsound.SND_ALIAS)
130 else:
131 self.assertRaises(
132 RuntimeError,
133 winsound.PlaySound,
134 'SystemExit', winsound.SND_ALIAS
135 )
Walter Dörwald7fd94242003-05-18 00:47:47 +0000136
Brian Curtin36067602010-04-13 02:32:40 +0000137 @unittest.skipUnless(has_sound("SystemHand"), "No default SystemHand")
Walter Dörwald7fd94242003-05-18 00:47:47 +0000138 def test_alias_hand(self):
Thomas Wouters49fd7fa2006-04-21 10:40:58 +0000139 if _have_soundcard():
140 winsound.PlaySound('SystemHand', winsound.SND_ALIAS)
141 else:
142 self.assertRaises(
143 RuntimeError,
144 winsound.PlaySound,
145 'SystemHand', winsound.SND_ALIAS
146 )
Walter Dörwald7fd94242003-05-18 00:47:47 +0000147
Brian Curtin36067602010-04-13 02:32:40 +0000148 @unittest.skipUnless(has_sound("SystemQuestion"),
149 "No default SystemQuestion")
Walter Dörwald7fd94242003-05-18 00:47:47 +0000150 def test_alias_question(self):
Thomas Wouters49fd7fa2006-04-21 10:40:58 +0000151 if _have_soundcard():
152 winsound.PlaySound('SystemQuestion', winsound.SND_ALIAS)
153 else:
154 self.assertRaises(
155 RuntimeError,
156 winsound.PlaySound,
157 'SystemQuestion', winsound.SND_ALIAS
158 )
Walter Dörwald7fd94242003-05-18 00:47:47 +0000159
160 def test_alias_fallback(self):
Tim Peters086e5622003-09-22 18:38:53 +0000161 # This test can't be expected to work on all systems. The MS
162 # PlaySound() docs say:
163 #
164 # If it cannot find the specified sound, PlaySound uses the
165 # default system event sound entry instead. If the function
166 # can find neither the system default entry nor the default
167 # sound, it makes no sound and returns FALSE.
168 #
169 # It's known to return FALSE on some real systems.
170
171 # winsound.PlaySound('!"$%&/(#+*', winsound.SND_ALIAS)
172 return
Walter Dörwald7fd94242003-05-18 00:47:47 +0000173
174 def test_alias_nofallback(self):
Thomas Wouters49fd7fa2006-04-21 10:40:58 +0000175 if _have_soundcard():
176 # Note that this is not the same as asserting RuntimeError
177 # will get raised: you cannot convert this to
178 # self.assertRaises(...) form. The attempt may or may not
179 # raise RuntimeError, but it shouldn't raise anything other
180 # than RuntimeError, and that's all we're trying to test
181 # here. The MS docs aren't clear about whether the SDK
182 # PlaySound() with SND_ALIAS and SND_NODEFAULT will return
183 # True or False when the alias is unknown. On Tim's WinXP
184 # box today, it returns True (no exception is raised). What
185 # we'd really like to test is that no sound is played, but
186 # that requires first wiring an eardrum class into unittest
187 # <wink>.
188 try:
189 winsound.PlaySound(
190 '!"$%&/(#+*',
191 winsound.SND_ALIAS | winsound.SND_NODEFAULT
192 )
193 except RuntimeError:
194 pass
195 else:
196 self.assertRaises(
197 RuntimeError,
198 winsound.PlaySound,
199 '!"$%&/(#+*', winsound.SND_ALIAS | winsound.SND_NODEFAULT
Tim Petersad9a7c42004-05-16 05:36:30 +0000200 )
Walter Dörwald7fd94242003-05-18 00:47:47 +0000201
202 def test_stopasync(self):
Thomas Wouters49fd7fa2006-04-21 10:40:58 +0000203 if _have_soundcard():
Walter Dörwald8bcbe6a2003-06-30 11:57:52 +0000204 winsound.PlaySound(
205 'SystemQuestion',
Thomas Wouters49fd7fa2006-04-21 10:40:58 +0000206 winsound.SND_ALIAS | winsound.SND_ASYNC | winsound.SND_LOOP
Walter Dörwald8bcbe6a2003-06-30 11:57:52 +0000207 )
Thomas Wouters49fd7fa2006-04-21 10:40:58 +0000208 time.sleep(0.5)
209 try:
210 winsound.PlaySound(
211 'SystemQuestion',
212 winsound.SND_ALIAS | winsound.SND_NOSTOP
213 )
214 except RuntimeError:
215 pass
216 else: # the first sound might already be finished
217 pass
218 winsound.PlaySound(None, winsound.SND_PURGE)
219 else:
Stefan Krahaf9c3cf2010-04-12 15:33:12 +0000220 # Issue 8367: PlaySound(None, winsound.SND_PURGE)
221 # does not raise on systems without a sound card.
222 pass
Thomas Wouters49fd7fa2006-04-21 10:40:58 +0000223
224
225def _get_cscript_path():
226 """Return the full path to cscript.exe or None."""
227 for dir in os.environ.get("PATH", "").split(os.pathsep):
228 cscript_path = os.path.join(dir, "cscript.exe")
229 if os.path.exists(cscript_path):
230 return cscript_path
231
232__have_soundcard_cache = None
233def _have_soundcard():
234 """Return True iff this computer has a soundcard."""
235 global __have_soundcard_cache
236 if __have_soundcard_cache is None:
237 cscript_path = _get_cscript_path()
238 if cscript_path is None:
239 # Could not find cscript.exe to run our VBScript helper. Default
240 # to True: most computers these days *do* have a soundcard.
241 return True
242
243 check_script = os.path.join(os.path.dirname(__file__),
244 "check_soundcard.vbs")
245 p = subprocess.Popen([cscript_path, check_script],
246 stdout=subprocess.PIPE)
247 __have_soundcard_cache = not p.wait()
Brian Curtin7ef28e82010-12-29 02:41:07 +0000248 p.stdout.close()
Thomas Wouters49fd7fa2006-04-21 10:40:58 +0000249 return __have_soundcard_cache
250
Walter Dörwald7fd94242003-05-18 00:47:47 +0000251
252def test_main():
Benjamin Petersonee8712c2008-05-20 21:35:26 +0000253 support.run_unittest(BeepTest, MessageBeepTest, PlaySoundTest)
Walter Dörwald7fd94242003-05-18 00:47:47 +0000254
255if __name__=="__main__":
256 test_main()