Skip to content

A Python library allowing to create powerful audio players in no time.

License

Notifications You must be signed in to change notification settings

colin-guyon/py-audio-player-base

Repository files navigation

py-audio-player-base

This Python library allows to bootstrap an audio player in no time.

Basically you can quickly create an efficient player using FFmpeg under the hood (thanks to PyAv), with hooks and methods allowing you to do things like:

  • Playing local audio files and web streams (web radios ...) (all things supported by FFmpeg)
  • Updating a UI (such as updating a player status, a progress bar ...)
  • Processing audio data chunks while the playback goes on, for example to calculate and display a power spectrum using Fast Fourier Transform.
  • ...

I initially wrote this library to embed it on a Raspberry Pi for my alarm clock project: http://wakepiup.colinguyon.com/.

The most simple usage example

Install PyAv (/~https://github.com/mikeboers/PyAV) (requires FFmpeg to be installed on your system):

pip install av

Install pyalsaaudio (http://larsimmisch.github.io/pyalsaaudio/)

pip install pyalsaaudio
from time import sleep
from audio_player_base.alsa import AlsaAudioPlayer
from audio_player_base.pyav import PyAVPlayObject

class MyAudioPlayer(AlsaAudioPlayer):
    # A real PlayObjectClass implementing ``PlayObjectInterface``
    PlayObjectClass = PyAVPlayObject

player = MyAudioPlayer()  # playlist will be audio files of the current dir
player.play()  # Non-blocking call

try:
    while player.status != 'stopped':
        sleep(1)
except KeyboardInterrupt:
    player.stop()

Other simple usage example

from audio_player_base.alsa import AlsaAudioPlayer
from audio_player_base.pyav import PyAVPlayObject

class MyAudioPlayer(AlsaAudioPlayer):
    # ALSA mixer name for volume control
    mixer_name = 'SomeAlsaMixer'
    # A real PlayObjectClass implementing ``PlayObjectInterface``
    PlayObjectClass = PyAVPlayObject

    def _do_write_data_chunk(self, data, context):
        # Write the data to the ALSA audio output,
        # calling the parent class method
        super(MyAudioPlayer, self)._do_write_data_chunk(data, context)
        # Call some custom function that analyzes the data
        process_fft_and_update_power_spectrum(data, context)

    def _notify_progression(self, context):
        # Regularly called, for example to update a progress bar
        current_percent_pos = self.play_object.get_percentage_pos()
        total_duration_seconds = self.play_object.duration
        current_pos_seconds = int(current_percent_pos / 100. * total_duration_seconds)
        print("Current play pos is {:.1f}% (-{:d}s)".\
                format(current_percent_pos,
                       total_duration_seconds - current_pos_seconds))

root_dir = '/my/usb/key/'
audio_dir = os.path.join(root_dir, 'music')
audio_trash_dir = os.path.join(root_dir, 'music_trash')

player = MyAudioPlayer(default_files_dir=audio_dir,
                       removed_files_backup_dir=audio_trash_dir)

player.play_pause()
# ...
player.play_next()
# ...
player.remove_current()
# ...
player.search_and_play("some pattern")
# ...
player.stop()

Main content

  • AudioPlayerInterface: Interface class for an audio player.

    Main methods:

    • play: Play a single file/stream or a whole folder, or a list of files/streams.
    • search_and_play: Search among a default search dir and play the matching results.
    • play_pause: Pause if currently playing, else play.
    • play_next
    • play_prev
    • stop
    • seek
    • set_volume
    • start_volume_fade_in
    • stop_volume_fade
    • remove_current: Remove the currently played track.

    Main readonly attributes and properties:

    • status: Current play status ("playing", "paused", "stopped")
    • volume: Current volume level (between 0 and 100)
    • current: Current playing track path
    • current_display_name: Current playing track pretty name

    Main hooks / methods to implement in real player classes:

    • _do_open_path_for_play: Open an audio track so that it can be played. An instance of a class implementing PlayObjectInterface must be returned, after having called its open method. Overriding this method you can also notify the new audio track name that will be played for example ...
    • _do_open_output: Open the audio output that will be used to hear the sound (called before the play queue is started).
    • _do_close_output: Close the audio output after a playback (called when the playback is finished, after a manual stop or the end of a playlist). The _on_playback_stopped method is called after _do_close_output and can be implemented for things that are not related to the audio output.
    • _do_write_data_chunk: Write an audio data chunk to the audio output. You can also process the data chunk in this method for example to display a power spectrum using FFT (but do not do too long actions, otherwise the playback would be degraded).
    • _notify_progression: Override this method that is regularly called to update a progress bar for example (not for heavy actions because called in the playback thread).
    • _on_track_removed: Called after the remove_current method has been called, and has effectively removed a file, for example to also remove the file path from a database.
  • PlayObjectInterface: Interface class for a play object used by a AudioPlayerInterface.

Main implementations that you would certainly like to use:

  • PyAVPlayObject: Play object class implenting PlayObjectInterface and using PyAV (that is to say the FFmpeg or libav lib). Compared to SubprocessDecoderPlayObject it allows to have a much faster seek, and faster track previous/next operations.
  • AlsaAudioPlayer: A player class implementing AudioPlayerInterface that uses ALSA for audio output and volume control.

Other implementations (not very well maintained, to be fixed):

  • SubprocessAudioPlayer: A player class implementing AudioPlayerInterface that uses a player in a sub process, such as mplayer. It does not allow to retrieve audio data chunks and process them.
  • SubprocessDecoderPlayObject: Play object class implenting PlayObjectInterface and using decoder.py (https://pypi.org/project/decoder.py/) for decoding (it uses lame, avconv or FFmpeg in a subprocess).

TODO

  • Add a setup.py
  • Document undocumented things + generate a sphinx doc
  • Write unit tests
  • Do adjustments to make this package useful for other people, if needed...

Links

Many thanks to the following projects that are used under the hood:

... and all those that I forget to mention !

About

A Python library allowing to create powerful audio players in no time.

Topics

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Languages