1#!/usr/bin/env python3 2# 3# Copyright 2017 - The Android Open Source Project 4# 5# Licensed under the Apache License, Version 2.0 (the "License"); 6# you may not use this file except in compliance with the License. 7# 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# Note: This test has been labelled as an integration test due to its use of 18# real data, and the 12+ second execution time. It also generates sine waves 19# during the test, rather than using data that has been pre-calculated. 20 21import math 22import numpy 23import unittest 24 25# TODO(markdr): Remove this after soundfile is added to setup.py 26import sys 27import mock 28sys.modules['soundfile'] = mock.Mock() 29 30import acts_contrib.test_utils.audio_analysis_lib.audio_quality_measurement as audio_quality_measurement 31 32 33class NoiseLevelTest(unittest.TestCase): 34 def setUp(self): 35 """Uses the same seed to generate noise for each test.""" 36 numpy.random.seed(0) 37 38 def test_noise_level(self): 39 # Generates the standard sin wave with standard_noise portion of noise. 40 rate = 48000 41 length_in_secs = 2 42 frequency = 440 43 amplitude = 1 44 standard_noise = 0.05 45 46 wave = [] 47 for index in range(0, rate * length_in_secs): 48 phase = 2.0 * math.pi * frequency * float(index) / float(rate) 49 sine_wave = math.sin(phase) 50 noise = standard_noise * numpy.random.standard_normal() 51 wave.append(float(amplitude) * (sine_wave + noise)) 52 53 # Calculates the average value after applying teager operator. 54 teager_value_of_wave, length = 0, len(wave) 55 for i in range(1, length - 1): 56 ith_teager_value = abs(wave[i] * wave[i] - wave[i - 1] * wave[i + 57 1]) 58 ith_teager_value *= max(1, abs(wave[i])) 59 teager_value_of_wave += ith_teager_value 60 teager_value_of_wave /= float(length * (amplitude**2)) 61 62 noise = audio_quality_measurement.noise_level( 63 amplitude, frequency, rate, teager_value_of_wave) 64 65 self.assertTrue(abs(noise - standard_noise) < 0.01) 66 67 68class ErrorTest(unittest.TestCase): 69 def test_error(self): 70 value1 = [0.2, 0.4, 0.1, 0.01, 0.01, 0.01] 71 value2 = [0.3, 0.3, 0.08, 0.0095, 0.0098, 0.0099] 72 error = [0.5, 0.25, 0.2, 0.05, 0.02, 0.01] 73 for i in range(len(value1)): 74 ret = audio_quality_measurement.error(value1[i], value2[i]) 75 self.assertTrue(abs(ret - error[i]) < 0.001) 76 77 78class QualityMeasurementTest(unittest.TestCase): 79 def setUp(self): 80 """Creates a test signal of sine wave.""" 81 numpy.random.seed(0) 82 83 self.rate = 48000 84 self.freq = 440 85 self.amplitude = 1 86 length_in_secs = 2 87 self.samples = length_in_secs * self.rate 88 self.y = [] 89 for index in range(self.samples): 90 phase = 2.0 * math.pi * self.freq * float(index) / float(self.rate) 91 sine_wave = math.sin(phase) 92 self.y.append(float(self.amplitude) * sine_wave) 93 94 def add_noise(self): 95 """Adds noise to the test signal.""" 96 noise_amplitude = 0.01 * self.amplitude 97 for index in range(self.samples): 98 noise = noise_amplitude * numpy.random.standard_normal() 99 self.y[index] += noise 100 101 def generate_delay(self): 102 """Generates some delays during playing.""" 103 self.delay_start_time = [0.200, 0.375, 0.513, 0.814, 1.000, 1.300] 104 self.delay_end_time = [0.201, 0.377, 0.516, 0.824, 1.100, 1.600] 105 106 for i in range(len(self.delay_start_time)): 107 start_index = int(self.delay_start_time[i] * self.rate) 108 end_index = int(self.delay_end_time[i] * self.rate) 109 for j in range(start_index, end_index): 110 self.y[j] = 0 111 112 def generate_artifacts_before_playback(self): 113 """Generates artifacts before playing.""" 114 silence_before_playback_end_time = 0.2 115 end_index = int(silence_before_playback_end_time * self.rate) 116 for i in range(0, end_index): 117 self.y[i] = 0 118 noise_start_index = int(0.1 * self.rate) 119 noise_end_index = int(0.1005 * self.rate) 120 for i in range(noise_start_index, noise_end_index): 121 self.y[i] = 3 * self.amplitude 122 123 def generate_artifacts_after_playback(self): 124 """Generates artifacts after playing.""" 125 silence_after_playback_start_time = int(1.9 * self.rate) 126 noise_start_index = int(1.95 * self.rate) 127 noise_end_index = int((1.95 + 0.02) * self.rate) 128 129 for i in range(silence_after_playback_start_time, self.samples): 130 self.y[i] = 0 131 for i in range(noise_start_index, noise_end_index): 132 self.y[i] = self.amplitude 133 134 def generate_burst_during_playback(self): 135 """Generates bursts during playing.""" 136 self.burst_start_time = [0.300, 0.475, 0.613, 0.814, 1.300] 137 self.burst_end_time = [0.301, 0.476, 0.614, 0.815, 1.301] 138 139 for i in range(len(self.burst_start_time)): 140 start_index = int(self.burst_start_time[i] * self.rate) 141 end_index = int(self.burst_end_time[i] * self.rate) 142 for j in range(start_index, end_index): 143 self.y[j] = self.amplitude * (3 + numpy.random.uniform(-1, 1)) 144 145 def generate_volume_changing(self): 146 """Generates volume changing during playing.""" 147 start_time = [0.300, 1.400] 148 end_time = [0.600, 1.700] 149 for i in range(len(start_time)): 150 start_index = int(start_time[i] * self.rate) 151 end_index = int(end_time[i] * self.rate) 152 for j in range(start_index, end_index): 153 self.y[j] *= 1.4 154 self.volume_changing = [+1, -1, +1, -1] 155 self.volume_changing_time = [0.3, 0.6, 1.4, 1.7] 156 157 def test_good_signal(self): 158 """Sine wave signal with no noise or artifacts.""" 159 result = audio_quality_measurement.quality_measurement(self.y, 160 self.rate) 161 self.assertTrue(len(result['artifacts']['noise_before_playback']) == 0) 162 self.assertTrue(len(result['artifacts']['noise_after_playback']) == 0) 163 self.assertTrue(len(result['artifacts']['delay_during_playback']) == 0) 164 self.assertTrue(len(result['artifacts']['burst_during_playback']) == 0) 165 self.assertTrue(len(result['volume_changes']) == 0) 166 self.assertTrue(result['equivalent_noise_level'] < 0.005) 167 168 def test_good_signal_with_noise(self): 169 """Sine wave signal with noise.""" 170 self.add_noise() 171 result = audio_quality_measurement.quality_measurement(self.y, 172 self.rate) 173 self.assertTrue(len(result['artifacts']['noise_before_playback']) == 0) 174 self.assertTrue(len(result['artifacts']['noise_after_playback']) == 0) 175 self.assertTrue(len(result['artifacts']['delay_during_playback']) == 0) 176 self.assertTrue(len(result['artifacts']['burst_during_playback']) == 0) 177 self.assertTrue(len(result['volume_changes']) == 0) 178 self.assertTrue(0.009 < result['equivalent_noise_level'] < 0.011) 179 180 def test_delay(self): 181 """Sine wave with delay during playing.""" 182 self.generate_delay() 183 result = audio_quality_measurement.quality_measurement(self.y, 184 self.rate) 185 self.assertTrue(len(result['artifacts']['noise_before_playback']) == 0) 186 self.assertTrue(len(result['artifacts']['noise_after_playback']) == 0) 187 self.assertTrue( 188 len(result['volume_changes']) == 2 * len(self.delay_start_time)) 189 self.assertTrue(result['equivalent_noise_level'] < 0.005) 190 191 self.assertTrue( 192 len(result['artifacts']['delay_during_playback']) == 193 len(self.delay_start_time)) 194 for i in range(len(result['artifacts']['delay_during_playback'])): 195 delta = abs(result['artifacts']['delay_during_playback'][i][0] - 196 self.delay_start_time[i]) 197 self.assertTrue(delta < 0.001) 198 duration = self.delay_end_time[i] - self.delay_start_time[i] 199 delta = abs(result['artifacts']['delay_during_playback'][i][1] - 200 duration) 201 self.assertTrue(delta < 0.001) 202 203 def test_artifacts_before_playback(self): 204 """Sine wave with artifacts before playback.""" 205 self.generate_artifacts_before_playback() 206 result = audio_quality_measurement.quality_measurement(self.y, 207 self.rate) 208 self.assertTrue(len(result['artifacts']['noise_before_playback']) == 1) 209 delta = abs(result['artifacts']['noise_before_playback'][0][0] - 0.1) 210 self.assertTrue(delta < 0.01) 211 delta = abs(result['artifacts']['noise_before_playback'][0][1] - 0.005) 212 self.assertTrue(delta < 0.004) 213 self.assertTrue(len(result['artifacts']['noise_after_playback']) == 0) 214 self.assertTrue(len(result['artifacts']['delay_during_playback']) == 0) 215 self.assertTrue(len(result['artifacts']['burst_during_playback']) == 0) 216 self.assertTrue(len(result['volume_changes']) == 0) 217 self.assertTrue(result['equivalent_noise_level'] < 0.005) 218 219 def test_artifacts_after_playback(self): 220 """Sine wave with artifacts after playback.""" 221 self.generate_artifacts_after_playback() 222 result = audio_quality_measurement.quality_measurement(self.y, 223 self.rate) 224 self.assertTrue(len(result['artifacts']['noise_before_playback']) == 0) 225 self.assertTrue(len(result['artifacts']['noise_after_playback']) == 1) 226 delta = abs(result['artifacts']['noise_after_playback'][0][0] - 1.95) 227 self.assertTrue(delta < 0.01) 228 delta = abs(result['artifacts']['noise_after_playback'][0][1] - 0.02) 229 self.assertTrue(delta < 0.001) 230 self.assertTrue(len(result['artifacts']['delay_during_playback']) == 0) 231 self.assertTrue(len(result['artifacts']['burst_during_playback']) == 0) 232 self.assertTrue(len(result['volume_changes']) == 0) 233 self.assertTrue(result['equivalent_noise_level'] < 0.005) 234 235 def test_burst_during_playback(self): 236 """Sine wave with burst during playback.""" 237 self.generate_burst_during_playback() 238 result = audio_quality_measurement.quality_measurement(self.y, 239 self.rate) 240 self.assertTrue(len(result['artifacts']['noise_before_playback']) == 0) 241 self.assertTrue(len(result['artifacts']['noise_after_playback']) == 0) 242 self.assertTrue(len(result['artifacts']['delay_during_playback']) == 0) 243 self.assertTrue(len(result['artifacts']['burst_during_playback']) == 5) 244 self.assertTrue(len(result['volume_changes']) == 10) 245 self.assertTrue(result['equivalent_noise_level'] > 0.02) 246 for i in range(len(result['artifacts']['burst_during_playback'])): 247 delta = abs(self.burst_start_time[i] - result['artifacts'][ 248 'burst_during_playback'][i]) 249 self.assertTrue(delta < 0.002) 250 251 def test_volume_changing(self): 252 """Sine wave with volume changing during playback.""" 253 self.generate_volume_changing() 254 result = audio_quality_measurement.quality_measurement(self.y, 255 self.rate) 256 self.assertTrue(len(result['artifacts']['noise_before_playback']) == 0) 257 self.assertTrue(len(result['artifacts']['noise_after_playback']) == 0) 258 self.assertTrue(len(result['artifacts']['delay_during_playback']) == 0) 259 self.assertTrue(len(result['artifacts']['burst_during_playback']) == 0) 260 self.assertTrue(result['equivalent_noise_level'] < 0.005) 261 self.assertTrue( 262 len(result['volume_changes']) == len(self.volume_changing)) 263 for i in range(len(self.volume_changing)): 264 self.assertTrue( 265 abs(self.volume_changing_time[i] - result['volume_changes'][i][ 266 0]) < 0.01) 267 self.assertTrue( 268 self.volume_changing[i] == result['volume_changes'][i][1]) 269 270 271if __name__ == '__main__': 272 unittest.main() 273