source: python/tests/test_fft.py @ 8ff31fb

feature/autosinkfeature/cnnfeature/cnn_orgfeature/constantqfeature/crepefeature/crepe_orgfeature/pitchshiftfeature/timestretchfix/ffmpeg5
Last change on this file since 8ff31fb was 319edae, checked in by Paul Brossier <piem@piem.org>, 6 years ago

[tests] avoid some imports, move main to end

  • Property mode set to 100755
File size: 6.5 KB
Line 
1#! /usr/bin/env python
2
3from numpy.testing import TestCase
4from numpy.testing import assert_equal, assert_almost_equal
5import numpy as np
6from aubio import fvec, fft, cvec
7from math import pi, floor
8from random import random
9
10class aubio_fft_test_case(TestCase):
11
12    def test_members(self):
13        """ check members are set correctly """
14        win_s = 2048
15        f = fft(win_s)
16        assert_equal (f.win_s, win_s)
17
18    def test_output_dimensions(self):
19        """ check the dimensions of output """
20        win_s = 1024
21        timegrain = fvec(win_s)
22        f = fft (win_s)
23        fftgrain = f (timegrain)
24        del f
25        assert_equal (fftgrain.norm.shape, (win_s/2+1,))
26        assert_equal (fftgrain.phas.shape, (win_s/2+1,))
27
28    def test_zeros(self):
29        """ check the transform of zeros is all zeros """
30        win_s = 512
31        timegrain = fvec(win_s)
32        f = fft (win_s)
33        fftgrain = f (timegrain)
34        assert_equal ( fftgrain.norm, 0 )
35        try:
36            assert_equal ( fftgrain.phas, 0 )
37        except AssertionError:
38            assert_equal (fftgrain.phas[fftgrain.phas > 0], +pi)
39            assert_equal (fftgrain.phas[fftgrain.phas < 0], -pi)
40            assert_equal (np.abs(fftgrain.phas[np.abs(fftgrain.phas) != pi]), 0)
41            self.skipTest('fft(fvec(%d)).phas != +0, ' % win_s \
42                    + 'This is expected when using fftw3 on powerpc.')
43
44    def test_impulse(self):
45        """ check the transform of one impulse at a random place """
46        win_s = 256
47        i = int(floor(random()*win_s))
48        impulse = pi * random()
49        f = fft(win_s)
50        timegrain = fvec(win_s)
51        timegrain[i] = impulse
52        fftgrain = f ( timegrain )
53        #self.plot_this ( fftgrain.phas )
54        assert_almost_equal ( fftgrain.norm, impulse, decimal = 6 )
55        assert_equal ( fftgrain.phas <= pi, True)
56        assert_equal ( fftgrain.phas >= -pi, True)
57
58    def test_impulse_negative(self):
59        """ check the transform of a negative impulse at a random place """
60        win_s = 256
61        i = int(floor(random()*win_s))
62        impulse = -.1
63        f = fft(win_s)
64        timegrain = fvec(win_s)
65        timegrain[0] = 0
66        timegrain[i] = impulse
67        fftgrain = f ( timegrain )
68        #self.plot_this ( fftgrain.phas )
69        assert_almost_equal ( fftgrain.norm, abs(impulse), decimal = 5 )
70        if impulse < 0:
71            # phase can be pi or -pi, as it is not unwrapped
72            #assert_almost_equal ( abs(fftgrain.phas[1:-1]) , pi, decimal = 6 )
73            assert_almost_equal ( fftgrain.phas[0], pi, decimal = 6)
74            assert_almost_equal ( np.fmod(fftgrain.phas[-1], pi), 0, decimal = 6)
75        else:
76            #assert_equal ( fftgrain.phas[1:-1] == 0, True)
77            assert_equal ( fftgrain.phas[0], 0)
78            assert_almost_equal ( np.fmod(fftgrain.phas[-1], pi), 0, decimal = 6)
79        # now check the resynthesis
80        synthgrain = f.rdo ( fftgrain )
81        #self.plot_this ( fftgrain.phas.T )
82        assert_equal ( fftgrain.phas <= pi, True)
83        assert_equal ( fftgrain.phas >= -pi, True)
84        #self.plot_this ( synthgrain - timegrain )
85        assert_almost_equal ( synthgrain, timegrain, decimal = 6 )
86
87    def test_impulse_at_zero(self):
88        """ check the transform of one impulse at a index 0 """
89        win_s = 1024
90        impulse = pi
91        f = fft(win_s)
92        timegrain = fvec(win_s)
93        timegrain[0] = impulse
94        fftgrain = f ( timegrain )
95        #self.plot_this ( fftgrain.phas )
96        assert_equal ( fftgrain.phas[0], 0)
97        # could be 0 or -0 depending on fft implementation (0 for fftw3, -0 for ooura)
98        assert_almost_equal ( fftgrain.phas[1], 0)
99        assert_almost_equal ( fftgrain.norm[0], impulse, decimal = 6 )
100
101    def test_rdo_before_do(self):
102        """ check running fft.rdo before fft.do works """
103        win_s = 1024
104        f = fft(win_s)
105        fftgrain = cvec(win_s)
106        t = f.rdo( fftgrain )
107        assert_equal ( t, 0 )
108
109    def plot_this(self, this):
110        from pylab import plot, show
111        plot ( this )
112        show ()
113
114    def test_local_fftgrain(self):
115        """ check aubio.fft() result can be accessed after deletion """
116        def compute_grain(impulse):
117            win_s = 1024
118            timegrain = fvec(win_s)
119            timegrain[0] = impulse
120            f = fft(win_s)
121            fftgrain = f ( timegrain )
122            return fftgrain
123        impulse = pi
124        fftgrain = compute_grain(impulse)
125        assert_equal ( fftgrain.phas[0], 0)
126        assert_almost_equal ( fftgrain.phas[1], 0)
127        assert_almost_equal ( fftgrain.norm[0], impulse, decimal = 6 )
128
129    def test_local_reconstruct(self):
130        """ check aubio.fft.rdo() result can be accessed after deletion """
131        def compute_grain(impulse):
132            win_s = 1024
133            timegrain = fvec(win_s)
134            timegrain[0] = impulse
135            f = fft(win_s)
136            fftgrain = f ( timegrain )
137            r = f.rdo(fftgrain)
138            return r
139        impulse = pi
140        r = compute_grain(impulse)
141        assert_almost_equal ( r[0], impulse, decimal = 6)
142        assert_almost_equal ( r[1:], 0)
143
144    def test_large_input_timegrain(self):
145        win_s = 1024
146        f = fft(win_s)
147        t = fvec(win_s + 1)
148        with self.assertRaises(ValueError):
149            f(t)
150
151    def test_small_input_timegrain(self):
152        win_s = 1024
153        f = fft(win_s)
154        t = fvec(1)
155        with self.assertRaises(ValueError):
156            f(t)
157
158    def test_large_input_fftgrain(self):
159        win_s = 1024
160        f = fft(win_s)
161        s = cvec(win_s + 5)
162        with self.assertRaises(ValueError):
163            f.rdo(s)
164
165    def test_small_input_fftgrain(self):
166        win_s = 1024
167        f = fft(win_s)
168        s = cvec(16)
169        with self.assertRaises(ValueError):
170            f.rdo(s)
171
172class aubio_fft_wrong_params(TestCase):
173
174    def test_wrong_buf_size(self):
175        win_s = -1
176        with self.assertRaises(ValueError):
177            fft(win_s)
178
179    def test_buf_size_not_power_of_two(self):
180        # when compiled with fftw3, aubio supports non power of two fft sizes
181        win_s = 320
182        try:
183            with self.assertRaises(RuntimeError):
184                fft(win_s)
185        except AssertionError:
186            self.skipTest('creating aubio.fft with size %d did not fail' % win_s)
187
188    def test_buf_size_too_small(self):
189        win_s = 1
190        with self.assertRaises(RuntimeError):
191            fft(win_s)
192
193if __name__ == '__main__':
194    from unittest import main
195    main()
Note: See TracBrowser for help on using the repository browser.