.. DO NOT EDIT. .. THIS FILE WAS AUTOMATICALLY GENERATED BY SPHINX-GALLERY. .. TO MAKE CHANGES, EDIT THE SOURCE PYTHON FILE: .. "examples\gallery_examples\004_isolate_orders.py" .. LINE NUMBERS ARE GIVEN BELOW. .. only:: html .. note:: :class: sphx-glr-download-link-note :ref:`Go to the end ` to download the full example code. .. rst-class:: sphx-glr-example-title .. _sphx_glr_examples_gallery_examples_004_isolate_orders.py: .. _isolate_orders_example: Isolate orders -------------- Orders are harmonic and partial components in the sound related to the speed of a rotating machine. This example shows how to isolate orders in a signal containing an RPM profile. It also uses additional classes from PyAnsys Sound to compute spectrograms and the loudness of the isolated signals. .. GENERATED FROM PYTHON SOURCE LINES 35-39 .. code-block:: Python # Maximum frequency for STFT plots, change according to your need MAX_FREQUENCY_PLOT_STFT = 2000.0 .. GENERATED FROM PYTHON SOURCE LINES 40-45 Set up analysis ~~~~~~~~~~~~~~~ Setting up the analysis consists of loading Ansys libraries, connecting to the DPF server, and retrieving the example files. .. GENERATED FROM PYTHON SOURCE LINES 45-68 .. code-block:: Python # Load Ansys libraries. import os import pathlib import matplotlib.pyplot as plt import numpy as np from ansys.sound.core.examples_helpers import ( download_accel_with_rpm_2_wav, download_accel_with_rpm_3_wav, download_accel_with_rpm_wav, ) from ansys.sound.core.psychoacoustics import LoudnessISO532_1_Stationary from ansys.sound.core.server_helpers import connect_to_or_start_server from ansys.sound.core.signal_utilities import LoadWav, WriteWav from ansys.sound.core.spectrogram_processing import IsolateOrders, Stft # Connect to a remote or start a local server my_server = connect_to_or_start_server(use_license_context=True) .. GENERATED FROM PYTHON SOURCE LINES 72-78 Define a custom function for STFT plots ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ Defining a custom function for STFT plots lets you have more control over what you are displaying. While you could use the ``Stft.plot()`` method, the custom function defined here restricts the frequency range of the plot. .. GENERATED FROM PYTHON SOURCE LINES 78-121 .. code-block:: Python def plot_stft(stft_class, vmax): out = stft_class.get_output_as_nparray() # Extract first half of the STFT (second half is symmetrical) half_nfft = int(out.shape[0] / 2) + 1 magnitude = stft_class.get_stft_magnitude_as_nparray() # Voluntarily ignore a numpy warning np.seterr(divide="ignore") magnitude = 20 * np.log10(magnitude[0:half_nfft, :]) np.seterr(divide="warn") # Obtain sampling frequency, time steps, and number of time samples fs = 1.0 / ( stft_class.signal.time_freq_support.time_frequencies.data[1] - stft_class.signal.time_freq_support.time_frequencies.data[0] ) time_step = np.floor(stft_class.fft_size * (1.0 - stft_class.window_overlap) + 0.5) / fs num_time_index = len(stft_class.get_output().get_available_ids_for_label("time")) # Define boundaries of the plot extent = [0, time_step * num_time_index, 0.0, fs / 2.0] # Plot plt.imshow( magnitude, origin="lower", aspect="auto", cmap="jet", extent=extent, vmin=vmax - 70.0, vmax=vmax, ) plt.colorbar(label="Amplitude (dB SPL)") plt.ylabel("Frequency (Hz)") plt.xlabel("Time (s)") plt.ylim( [0.0, MAX_FREQUENCY_PLOT_STFT] ) # Change the value of MAX_FREQUENCY_PLOT_STFT if needed plt.title("STFT") plt.show() .. GENERATED FROM PYTHON SOURCE LINES 122-130 Load a signal with an RPM profile ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ Load a signal that has been generated with Ansys Sound Analysis and Specification (SAS) from a WAV file using the ``LoadWav`` class. This class contains two channels: - The actual signal (an acceleration recording) - The associated RPM profile .. GENERATED FROM PYTHON SOURCE LINES 130-158 .. code-block:: Python # Return the input data of the example file path_accel_wav = download_accel_with_rpm_wav(server=my_server) # Load the WAV file. wav_loader = LoadWav(path_accel_wav) wav_loader.process() fc_signal = wav_loader.get_output() # Extract the audio signal and the RPM profile wav_signal, rpm_signal = wav_loader.get_output_as_nparray() # Extract time support associated with the signal time_support = fc_signal[0].time_freq_support.time_frequencies.data # Plot the signal and its associated RPM profile fig, ax = plt.subplots(nrows=2, sharex=True) ax[0].plot(time_support, wav_signal) ax[0].set_title("Audio Signal") ax[0].set_ylabel("Amplitude (Pa)") ax[0].grid(True) ax[1].plot(time_support, rpm_signal, color="red") ax[1].set_title("RPM profile") ax[1].set_ylabel("rpm") ax[1].grid(True) plt.xlabel("Time (s)") plt.show() .. image-sg:: /examples/gallery_examples/images/sphx_glr_004_isolate_orders_001.png :alt: Audio Signal, RPM profile :srcset: /examples/gallery_examples/images/sphx_glr_004_isolate_orders_001.png :class: sphx-glr-single-img .. GENERATED FROM PYTHON SOURCE LINES 159-162 Plot spectrogram of the original signal ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ Plot the spectrogram of the original signal. .. GENERATED FROM PYTHON SOURCE LINES 162-168 .. code-block:: Python stft = Stft(signal=fc_signal[0], window_overlap=0.9, fft_size=8192) stft.process() max_stft = 20 * np.log10(np.max(stft.get_stft_magnitude_as_nparray())) plot_stft(stft, max_stft) .. image-sg:: /examples/gallery_examples/images/sphx_glr_004_isolate_orders_002.png :alt: STFT :srcset: /examples/gallery_examples/images/sphx_glr_004_isolate_orders_002.png :class: sphx-glr-single-img .. GENERATED FROM PYTHON SOURCE LINES 169-172 Isolate orders ~~~~~~~~~~~~~~~~ Isolate orders 2, 4, and 6 with the ``IsolateOrders`` class. .. GENERATED FROM PYTHON SOURCE LINES 172-202 .. code-block:: Python field_wav, field_rpm = wav_loader.get_output() # Define parameters for order isolation field_wav.unit = "Pa" order_to_isolate = [2, 4, 6] # Orders indexes to isolate as a list fft_size = 8192 # FFT Size (in samples) window_type = "HANN" # Window type window_overlap = 0.9 # Window overlap width_selection = 3 # Width of the order selection in Hz # Instantiate the ``IsolateOrders`` class with the parameters isolate_orders = IsolateOrders( signal=field_wav, rpm_profile=field_rpm, orders=order_to_isolate, fft_size=fft_size, window_type=window_type, window_overlap=window_overlap, width_selection=width_selection, ) # Isolate orders isolate_orders.process() # Plot the spectrogram of the isolated orders stft.signal = isolate_orders.get_output() stft.process() plot_stft(stft, max_stft) .. image-sg:: /examples/gallery_examples/images/sphx_glr_004_isolate_orders_003.png :alt: STFT :srcset: /examples/gallery_examples/images/sphx_glr_004_isolate_orders_003.png :class: sphx-glr-single-img .. GENERATED FROM PYTHON SOURCE LINES 203-206 Isolate different orders ~~~~~~~~~~~~~~~~~~~~~~~~ Change FFT size, order indexes, and window type. Then re-isolate the orders. .. GENERATED FROM PYTHON SOURCE LINES 206-219 .. code-block:: Python # Change some parameters directly using the setters of the class isolate_orders.orders = [2, 6] isolate_orders.window_type = "BLACKMAN" # Reprocess (Must be called explicitly. Otherwise, the output won't be updated.) isolate_orders.process() # Plot the spectrogram of the isolated orders stft.signal = isolate_orders.get_output() stft.process() plot_stft(stft, max_stft) .. image-sg:: /examples/gallery_examples/images/sphx_glr_004_isolate_orders_004.png :alt: STFT :srcset: /examples/gallery_examples/images/sphx_glr_004_isolate_orders_004.png :class: sphx-glr-single-img .. GENERATED FROM PYTHON SOURCE LINES 220-223 Work with the isolated signal ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ Plot the signal containing the isolated orders and compute its loudness. .. GENERATED FROM PYTHON SOURCE LINES 223-244 .. code-block:: Python # Plot the signal directly using the method from the ``IsolateOrders`` class isolate_orders.plot() # Use the ``Loudness`` class to compute the loudness of the isolate signal input_loudness = isolate_orders.get_output() input_loudness.unit = "Pa" loudness = LoudnessISO532_1_Stationary(signal=input_loudness) loudness.process() loudness_isolated_signal = loudness.get_loudness_level_phon() # Compute the loudness for the original signal loudness.signal = field_wav loudness.process() loudness_original_signal = loudness.get_loudness_level_phon() print(f"Loudness of the original signal is {loudness_original_signal: .1f} phons.") print(f"Loudness of the isolated signal is {loudness_isolated_signal: .1f} phons.") .. image-sg:: /examples/gallery_examples/images/sphx_glr_004_isolate_orders_005.png :alt: 004 isolate orders :srcset: /examples/gallery_examples/images/sphx_glr_004_isolate_orders_005.png :class: sphx-glr-single-img .. rst-class:: sphx-glr-script-out .. code-block:: none Loudness of the original signal is 76.6 phons. Loudness of the isolated signal is 61.4 phons. .. GENERATED FROM PYTHON SOURCE LINES 245-248 Isolate orders of several signals in a loop ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ Loop over a list of given signals and write them as a WAV file. .. GENERATED FROM PYTHON SOURCE LINES 248-278 .. code-block:: Python # Obtain parent folder of the 'accel_with_rpm.wav' file parent_folder = pathlib.Path(path_accel_wav).parent.absolute() path_accel_wav_2 = download_accel_with_rpm_2_wav(server=my_server) path_accel_wav_3 = download_accel_with_rpm_3_wav(server=my_server) paths = (path_accel_wav, path_accel_wav_2, path_accel_wav_3) fft_sizes = [256, 2048, 4096] wav_writer = WriteWav() # Isolate orders for all the files containing RPM profiles in this folder for file, fft_sz in zip(paths, fft_sizes): # Loading the file wav_loader.path_to_wav = file wav_loader.process() # Set parameters for order isolation isolate_orders.signal = wav_loader.get_output()[0] isolate_orders.rpm_profile = wav_loader.get_output()[1] isolate_orders.fft_size = fft_sz isolate_orders.process() # Write as a WAV file on the disk out_name = os.path.basename(file)[:-4] + "_isolated_fft_size_" + str(fft_sz) + ".wav" path_to_write = parent_folder / out_name wav_writer.path_to_write = str(path_to_write) wav_writer.signal = isolate_orders.get_output() wav_writer.process() .. rst-class:: sphx-glr-timing **Total running time of the script:** (7 minutes 49.695 seconds) .. _sphx_glr_download_examples_gallery_examples_004_isolate_orders.py: .. only:: html .. container:: sphx-glr-footer sphx-glr-footer-example .. container:: sphx-glr-download sphx-glr-download-jupyter :download:`Download Jupyter notebook: 004_isolate_orders.ipynb <004_isolate_orders.ipynb>` .. container:: sphx-glr-download sphx-glr-download-python :download:`Download Python source code: 004_isolate_orders.py <004_isolate_orders.py>` .. container:: sphx-glr-download sphx-glr-download-zip :download:`Download zipped: 004_isolate_orders.zip <004_isolate_orders.zip>` .. only:: html .. rst-class:: sphx-glr-signature `Gallery generated by Sphinx-Gallery `_