tinytag


Nametinytag JSON
Version 2.0.0 PyPI version JSON
download
home_pageNone
SummaryRead audio file metadata
upload_time2024-11-03 15:25:06
maintainerNone
docs_urlNone
authorMat (mathiascode)
requires_python>=3.7
licenseNone
keywords metadata audio music mp3 m4a wav ogg opus flac wma aiff
VCS
bugtrack_url
requirements No requirements were recorded.
Travis-CI No Travis.
coveralls test coverage No coveralls.
            <!--
  SPDX-FileCopyrightText: 2014-2024 tinytag Contributors
  SPDX-License-Identifier: MIT
-->

# tinytag 

tinytag is a Python library for reading audio file metadata

[![Build Status](https://img.shields.io/github/actions/workflow/status/tinytag/tinytag/tests.yml
)](https://github.com/tinytag/tinytag/actions?query=workflow:%22Tests%22)
[![Coverage Status](https://img.shields.io/coverallsCoverage/github/tinytag/tinytag
)](https://coveralls.io/r/tinytag/tinytag)
[![PyPI Version](https://img.shields.io/pypi/v/tinytag
)](https://pypi.org/project/tinytag/)
[![PyPI Downloads](https://img.shields.io/pypi/dm/tinytag
)](https://pypistats.org/packages/tinytag)


## Install

```
python3 -m pip install tinytag
```


## Features

  * Read tags, images and properties of audio files
  * Supported formats:
    * MP3 / MP2 / MP1 (ID3 v1, v1.1, v2.2, v2.3+)
    * M4A (AAC / ALAC)
    * WAVE / WAV
    * OGG (FLAC / Opus / Speex / Vorbis)
    * FLAC
    * WMA
    * AIFF / AIFF-C
  * Same API for all formats
  * Small, portable library
  * High code coverage
  * Pure Python, no dependencies
  * Supports Python 3.7 or higher


## Usage

tinytag only provides the minimum needed for _reading_ metadata, and presents
it in a simple format. It can determine track number, total tracks, title,
artist, album, year, duration and more.

```python
from tinytag import TinyTag
tag: TinyTag = TinyTag.get('/some/music.mp3')

print(f'This track is by {tag.artist}.')
print(f'It is {tag.duration:.2f} seconds long.')
```

> [!WARNING]  
> The `ignore_errors` parameter of `TinyTag.get()` is obsolete as of tinytag
> 2.0.0, and will be removed in the future.

Alternatively you can use tinytag directly on the command line:

    $ python3 -m tinytag /some/music.mp3
    {
      "filename": "/some/music.mp3",
      "filesize": 3243226,
      "duration": 173.52,
      "channels": 2,
      "bitrate": 128,
      "samplerate": 44100,
      "artist": [
        "artist name"
      ],
      "album": [
        "album name"
      ],
      "title": [
        "track name"
      ],
      "track": 4,
      "genre": [
        "Jazz"
      ],
      "year": [
        "2010"
      ],
      "comment": [
        "Some comment here"
      ]
    }

Check `python3 -m tinytag --help` for all CLI options, for example other
output formats.

Support for changing/writing metadata will not be added. Use another library
such as [Mutagen](https://mutagen.readthedocs.io/) for this.

### Supported Files

To receive a tuple of file extensions tinytag supports, use the
`SUPPORTED_FILE_EXTENSIONS` constant:

```python
TinyTag.SUPPORTED_FILE_EXTENSIONS
```

Alternatively, check if a file is supported by providing its path:

```python
is_supported = TinyTag.is_supported('/some/music.mp3')
```

### Common Metadata

tinytag provides some common attributes, which always contain a single value.
These are helpful when you need quick access to common metadata.

#### File/Audio Properties

    tag.bitdepth      # bit depth as integer (for lossless audio)
    tag.bitrate       # bitrate in kBits/s as float
    tag.duration      # audio duration in seconds as float
    tag.filename      # filename as string
    tag.filesize      # file size in bytes as integer
    tag.samplerate    # samples per second as integer

> [!WARNING]  
> The `tag.audio_offset` attribute is obsolete as of tinytag 2.0.0, and will
> be removed in the future.

#### Metadata Fields

    tag.album         # album as string
    tag.albumartist   # album artist as string
    tag.artist        # artist name as string
    tag.comment       # file comment as string
    tag.composer      # composer as string
    tag.disc          # disc number as integer
    tag.disc_total    # total number of discs as integer
    tag.genre         # genre as string
    tag.title         # title of the song as string
    tag.track         # track number as integer
    tag.track_total   # total number of tracks as integer
    tag.year          # year or date as string

### Additional Metadata

For additional values of the same field type, non-common metadata fields, or
metadata specific to certain file formats, use `other`:

    tag.other         # a dictionary of additional fields

> [!WARNING]  
> The `other` dictionary has replaced the `extra` dictionary in tinytag 2.0.0.
> The latter will be removed in a future release.

The following `other` field names are standardized in tinytag, and optionally
present when files provide such metadata:

    barcode
    bpm
    catalog_number
    conductor
    copyright
    director
    encoded_by
    encoder_settings
    initial_key
    isrc
    language
    license
    lyricist
    lyrics
    media
    publisher
    set_subtitle
    url

Additional `other` field names not documented above may be present, but are
format-specific and may change or disappear in future tinytag releases. If
tinytag does not expose metadata you need, or you wish to standardize more
field names, open a feature request on GitHub for discussion.

`other` values are always provided as strings, and are not guaranteed to be
valid. Should e.g. the `bpm` value in the file contain non-numeric characters,
tinytag will provide the string as-is. It is your responsibility to handle
possible exceptions, e.g. when converting the value to an integer.

Multiple values of the same field type are provided if a file contains them.
Values are always provided as a list, even when only a single value exists.

Example:

```python
from tinytag import OtherFields, TinyTag

tag: TinyTag = TinyTag.get('/some/music.mp3')
other_fields: OtherFields = tag.other
catalog_numbers: list[str] | None = other_fields.get('catalog_number')

if catalog_numbers:
    catalog_number: str = catalog_numbers[0]
    print(catalog_number)

print(catalog_numbers)
```

Output:

    > 10
    > ['10']

When a file contains multiple values for a [common metadata field](#common-metadata)
(e.g. `artist`), the primary value is accessed through the common attribute
(`tag.artist`), and any additional values through the `other` dictionary
(`tag.other['artist']`).

Example:

```python
from tinytag import TinyTag

tag: TinyTag = TinyTag.get('/some/music.mp3')
artist: str | None = tag.artist
additional_artists: list[str] | None = tag.other.get('artist')

print(artist)
print(additional_artists)
```

Output:

    > main artist
    > ['another artist', 'yet another artist']

### All Metadata

If you need to receive all available metadata as key-value pairs in a flat
dictionary, use the `as_dict()` method. This combines the common attributes
and `other` dictionary, which can be more convenient in some cases.

    from tinytag import TinyTag

    tag: TinyTag = TinyTag.get('/some/music.mp3')
    metadata: dict = tag.as_dict()

### Images

Additionally, you can also read embedded images by passing a `image=True`
keyword argument to `TinyTag.get()`.

If you need to receive an image of a specific kind, including its description,
use `images`:

    tag.images        # available embedded images

The following common image attributes are available, providing the first
located image of each kind:

    tag.images.front_cover  # front cover as 'Image' object
    tag.images.back_cover   # back cover as 'Image' object
    tag.images.media        # media (e.g. CD label) as 'Image' object

When present, any additional images are available in an `images.other`
dictionary, using the following standardized key names:

    generic
    icon
    alt_icon
    front_cover
    back_cover
    media
    leaflet
    lead_artist
    artist
    conductor
    band
    composer
    lyricist
    recording_location
    during_recording
    during_performance
    screen_capture
    bright_colored_fish
    illustration
    band_logo
    publisher_logo
    unknown

Provided values are always lists containing at least one `Image` object.

The `Image` object provides the following attributes:

    data           # image data as bytes
    name           # image name/kind as string
    mime_type      # image MIME type as string
    description    # image description as string

To receive any available image, prioritizing the front cover, use `images.any`:

```python
from tinytag import Image, TinyTag

tag: TinyTag = TinyTag.get('/some/music.ogg', image=True)
image: Image | None = tag.images.any

if image is not None:
    data: bytes = image.data
    name: str = image.name
    mime_type: str = image.mime_type
    description: str = image.description

    print(len(data))
    print(name)
    print(mime_type)
    print(description)
```

Output:

    > 74452
    > front_cover
    > image/jpeg
    > some image description

> [!WARNING]  
> `tag.images.any` has replaced `tag.get_image()` in tinytag 2.0.0.
> `tag.get_image()` will be removed in a future tinytag 2.x release.

To receive a common image, e.g. `front_cover`:

```python
from tinytag import Image, Images, TinyTag

tag: TinyTag = TinyTag.get('/some/music.ogg', image=True)
images: Images = tag.images
cover_image: Image = images.front_cover

if cover_image is not None:
    data: bytes = cover_image.data
    description: str = cover_image.description
```

To receive an additional image, e.g. `bright_colored_fish`:

```python
from tinytag import Image, OtherImages, TinyTag

tag: TinyTag = TinyTag.get('/some/music.ogg', image=True)
other_images: OtherImages = tag.images.other
fish_images: list[Image] | None = other_images.get('bright_colored_fish')

if fish_images:
    image = fish_images[0]  # Use first image
    data = image.data
    description = image.description
```

### Encoding

To open files using a specific encoding, you can use the `encoding` parameter.
This parameter is however only used for formats where the encoding is not
explicitly specified.

```python
TinyTag.get('a_file_with_gbk_encoding.mp3', encoding='gbk')
```

### File-like Objects

To use a file-like object (e.g. BytesIO) instead of a file path, pass a
`file_obj` keyword argument:

```python
TinyTag.get(file_obj=your_file_obj)
```

### Exceptions

    TinyTagException        # Base class for exceptions
    ParseError              # Parsing an audio file failed
    UnsupportedFormatError  # File format is not supported


## Changelog

### 2.0.0  (2024-11-03)

- **BREAKING:** Store 'disc', 'disc_total', 'track' and 'track_total' values as int instead of str
- **BREAKING:** 'as_dict()' method (previously undocumented) returns tag field values in list form
- **BREAKING:** TinyTagException no longer inherits LookupError
- **BREAKING:** TinyTag subclasses are now private
- **BREAKING:** Remove function to use custom audio file samples in tests
- **BREAKING:** Remove support for Python 2
- **DEPRECATION:** Mark 'ignore_errors' parameter for TinyTag.get() as obsolete
- **DEPRECATION:** Mark 'audio_offset' attribute as obsolete
- **DEPRECATION:** Deprecate 'extra' dict in favor of 'other' dict with values in list form
- **DEPRECATION:** Deprecate 'get_image()' method in favor of 'images.any' property
- Add type hints to codebase
- Provide access to custom metadata fields through the 'other' dict
- Provide access to all available images
- Add more standard 'other' fields
- Use Flit as Python build backend instead of Setuptools
- ID3: Fix invalid sample rate/duration in some cases
- ID3: Fix reading of UTF-16 strings without BOM
- FLAC: Apply ID3 tags after Vorbis
- OGG/WMA: Set missing 'channels' field
- WMA: Set missing 'other.copyright' field
- WMA: Raise exception if file is invalid
- Various optimizations

### 1.10.1  (2023-10-26)

- Update 'extra' fields with data from other tags #188
- ID3: Add missing 'extra.copyright' field

### 1.10.0  (2023-10-18)

- Add support for OGG FLAC format #182
- Add support for OGG Speex format #181
- Wave: support image loading
- Add support for file-like objects (BytesIO) #178
- Add list of supported file extensions #177
- Fix deprecations related to setuptools #176
- Fix pathlib support in TinyTag.is_supported()
- Only remove zero bytes at the end of strings
- Stricter conditions in while loops
- OGG: Add stricter magic byte matching for OGG files
- Compatibility with Python 3.4 and 3.5 is no longer tested

### 1.9.0  (2023-04-23)

- Add bitdepth attribute for lossless audio #157
- Add recognition of Audible formats #163 (thanks to snowskeleton)
- Add .m4v to list of supported file extensions #142
- Aiff: Implement replacement for Python's aifc module #164
- ID3: Only check for language in COMM and USLT frames #147
- ID3: Read the correct number of bytes from Xing header #154
- ID3: Add support for ID3v2.4 TDRC frame #156 (thanks to Uninen)
- M4A: Add description fields #168 (thanks to snowskeleton)
- RIFF: Handle tags containing extra zero-byte #141
- Vorbis: Parse OGG cover art #144 (thanks to Pseurae)
- Vorbis: Support standard disctotal/tracktotal comments #171
- Wave: Add proper support for padded IFF chunks

### 1.8.1  (2022-03-12) [still mathiascode-edition]

- MP3 ID3: Set correct file position if tag reading is disabled #119 (thanks to mathiascode)
- MP3: Fix incorrect calculation of duration for VBR encoded MP3s #128 (thanks to mathiascode)

### 1.8.0  (2022-03-05) [mathiascode-edition]

- Add support for ALAC audio files #130 (thanks to mathiascode)
- AIFF: Fixed bitrate calculation for certain files #129 (thanks to mathiascode)
- MP3: Do not round MP3 bitrates #131 (thanks to mathiascode)
- MP3 ID3: Support any language in COMM and USLT frames #135 (thanks to mathiascode)
- Performance: Don't use regex when parsing genre #136 (thanks to mathiascode)
- Disable tag parsing for all formats when requested #137 (thanks to mathiascode)
- M4A: Fix invalid bitrates in certain files #132 (thanks to mathiascode)
- WAV: Fix metadata parsing for certain files #133 (thanks to mathiascode)

### 1.7.0. (2021-12-14)

- fixed rare occasion of ID3v2 tags missing their first character, #106
- allow overriding the default encoding of ID3 tags (e.g. `TinyTag.get(..., encoding='gbk'))`)
- fixed calculation of bitrate for very short mp3 files, #99
- utf-8 support for AIFF files, #123
- fixed image parsing for id3v2 with images containing utf-16LE descriptions, #117
- fixed ID3v1 tags overwriting ID3v2 tags, #121
- Set correct file position if tag reading is disabled for ID3 (thanks to mathiascode)

### 1.6.0  (2021-08-28) [aw-edition]

- fixed handling of non-latin encoding types for images (thanks to aw-was-here)
- added support for ISRC data, available in `extra['isrc']` field (thanks to aw-was-here)
- added support for AIFF/AIFF-C (thanks to aw-was-here)
- fixed import deprecation warnings (thanks to idotobi)
- fixed exception for TinyTag misuse being different in different python versions (thanks to idotobi)
- added support for ID3 initial key tonality hint, available in `extra['initial_key']`
- added support for ID3 unsynchronized lyrics, available in `extra['lyrics']`
- added `extra` field, which may contain additional metadata not available in all file formats

### 1.5.0  (2020-11-05)

- fixed data type to always return str for disc, disc_total, track, track_total #97 (thanks to kostalski)
- fixed package install being reported as UNKNOWN for some python/pip variations #90 (thanks to russpoutine)
- Added automatic detection for certain MP4 file headers

### 1.4.0  (2020-04-23)

- detecting file types based on their magic header bytes, #85
- fixed opus duration being wrong for files with lower sample rate #81
- implemented support for binary paths #72
- always cast mp3 bitrates to int, so that CBR and VBR output behaves the sam
- made __str__ deterministic and use json as output format

### 1.3.0  (2020-03-09)

- added option to ignore encoding errors `ignore_errors` #73
- Improved text decoding for many malformed files

### 1.2.2  (2019-04-13)

- Improved stability when reading corrupted mp3 files

### 1.2.1  (2019-04-13)

- fixed wav files not correctly reporting the number of channels #61

### 1.2.0  (2019-04-13)

- using setup.cfg instead of setup.py (thanks to scivision)
- added support for calling TinyTag.get with pathlib.Path (thanks to scivision)
- added appveyor windows test CI (thanks to scivision)
- using pytest instead of nosetest (thanks to scivision)

### 1.1.0  (2019-04-13)

- added new field "composer" (Thanks to Phil Borman)

### 1.0.1  (2019-04-13)

- fixed ID3 loading for files with corrupt header (thanks to Ian Homer)
- fixed parsing of duration in wav file (thanks to Ian Homer)

### 1.0.0  (2018-12-12)

- added comment field
- added wav-riff format support
- use MP4 parser for m4b files
- added simple cli tool
- fix parsing of FLAC files with ID3 header (thanks to minus7)
- added method `TinyTag.is_supported(filename)`

### 0.19.0 (2018-02-11)

- fixed corrupted images for some mp3s (#45)

### 0.18.0 (2017-04-29)

- fixed wrong bitrate and crash when parsing xing header

### 0.17.0 (2016-10-02)

- supporting ID3v2.2 images

### 0.16.0 (2016-08-06)

- MP4 cover image support

### 0.15.2 (2016-08-06)

- fixed crash for malformed MP4 files (#34)

### 0.15.0 (2016-08-06)

- fixed decoding of UTF-16LE ID3v2 Tags, improved overall stability

### 0.14.0 (2016-06-05):

- MP4/M4A and Opus support


            

Raw data

            {
    "_id": null,
    "home_page": null,
    "name": "tinytag",
    "maintainer": null,
    "docs_url": null,
    "requires_python": ">=3.7",
    "maintainer_email": null,
    "keywords": "metadata, audio, music, mp3, m4a, wav, ogg, opus, flac, wma, aiff",
    "author": "Mat (mathiascode)",
    "author_email": "Tom Wallroth <tomwallroth@gmail.com>",
    "download_url": "https://files.pythonhosted.org/packages/21/7a/04f50e2c660862b9929a06d710968b2520ae1bf215ed57a9972065d0aeef/tinytag-2.0.0.tar.gz",
    "platform": null,
    "description": "<!--\n  SPDX-FileCopyrightText: 2014-2024 tinytag Contributors\n  SPDX-License-Identifier: MIT\n-->\n\n# tinytag \n\ntinytag is a Python library for reading audio file metadata\n\n[![Build Status](https://img.shields.io/github/actions/workflow/status/tinytag/tinytag/tests.yml\n)](https://github.com/tinytag/tinytag/actions?query=workflow:%22Tests%22)\n[![Coverage Status](https://img.shields.io/coverallsCoverage/github/tinytag/tinytag\n)](https://coveralls.io/r/tinytag/tinytag)\n[![PyPI Version](https://img.shields.io/pypi/v/tinytag\n)](https://pypi.org/project/tinytag/)\n[![PyPI Downloads](https://img.shields.io/pypi/dm/tinytag\n)](https://pypistats.org/packages/tinytag)\n\n\n## Install\n\n```\npython3 -m pip install tinytag\n```\n\n\n## Features\n\n  * Read tags, images and properties of audio files\n  * Supported formats:\n    * MP3 / MP2 / MP1 (ID3 v1, v1.1, v2.2, v2.3+)\n    * M4A (AAC / ALAC)\n    * WAVE / WAV\n    * OGG (FLAC / Opus / Speex / Vorbis)\n    * FLAC\n    * WMA\n    * AIFF / AIFF-C\n  * Same API for all formats\n  * Small, portable library\n  * High code coverage\n  * Pure Python, no dependencies\n  * Supports Python 3.7 or higher\n\n\n## Usage\n\ntinytag only provides the minimum needed for _reading_ metadata, and presents\nit in a simple format. It can determine track number, total tracks, title,\nartist, album, year, duration and more.\n\n```python\nfrom tinytag import TinyTag\ntag: TinyTag = TinyTag.get('/some/music.mp3')\n\nprint(f'This track is by {tag.artist}.')\nprint(f'It is {tag.duration:.2f} seconds long.')\n```\n\n> [!WARNING]  \n> The `ignore_errors` parameter of `TinyTag.get()` is obsolete as of tinytag\n> 2.0.0, and will be removed in the future.\n\nAlternatively you can use tinytag directly on the command line:\n\n    $ python3 -m tinytag /some/music.mp3\n    {\n      \"filename\": \"/some/music.mp3\",\n      \"filesize\": 3243226,\n      \"duration\": 173.52,\n      \"channels\": 2,\n      \"bitrate\": 128,\n      \"samplerate\": 44100,\n      \"artist\": [\n        \"artist name\"\n      ],\n      \"album\": [\n        \"album name\"\n      ],\n      \"title\": [\n        \"track name\"\n      ],\n      \"track\": 4,\n      \"genre\": [\n        \"Jazz\"\n      ],\n      \"year\": [\n        \"2010\"\n      ],\n      \"comment\": [\n        \"Some comment here\"\n      ]\n    }\n\nCheck `python3 -m tinytag --help` for all CLI options, for example other\noutput formats.\n\nSupport for changing/writing metadata will not be added. Use another library\nsuch as [Mutagen](https://mutagen.readthedocs.io/) for this.\n\n### Supported Files\n\nTo receive a tuple of file extensions tinytag supports, use the\n`SUPPORTED_FILE_EXTENSIONS` constant:\n\n```python\nTinyTag.SUPPORTED_FILE_EXTENSIONS\n```\n\nAlternatively, check if a file is supported by providing its path:\n\n```python\nis_supported = TinyTag.is_supported('/some/music.mp3')\n```\n\n### Common Metadata\n\ntinytag provides some common attributes, which always contain a single value.\nThese are helpful when you need quick access to common metadata.\n\n#### File/Audio Properties\n\n    tag.bitdepth      # bit depth as integer (for lossless audio)\n    tag.bitrate       # bitrate in kBits/s as float\n    tag.duration      # audio duration in seconds as float\n    tag.filename      # filename as string\n    tag.filesize      # file size in bytes as integer\n    tag.samplerate    # samples per second as integer\n\n> [!WARNING]  \n> The `tag.audio_offset` attribute is obsolete as of tinytag 2.0.0, and will\n> be removed in the future.\n\n#### Metadata Fields\n\n    tag.album         # album as string\n    tag.albumartist   # album artist as string\n    tag.artist        # artist name as string\n    tag.comment       # file comment as string\n    tag.composer      # composer as string\n    tag.disc          # disc number as integer\n    tag.disc_total    # total number of discs as integer\n    tag.genre         # genre as string\n    tag.title         # title of the song as string\n    tag.track         # track number as integer\n    tag.track_total   # total number of tracks as integer\n    tag.year          # year or date as string\n\n### Additional Metadata\n\nFor additional values of the same field type, non-common metadata fields, or\nmetadata specific to certain file formats, use `other`:\n\n    tag.other         # a dictionary of additional fields\n\n> [!WARNING]  \n> The `other` dictionary has replaced the `extra` dictionary in tinytag 2.0.0.\n> The latter will be removed in a future release.\n\nThe following `other` field names are standardized in tinytag, and optionally\npresent when files provide such metadata:\n\n    barcode\n    bpm\n    catalog_number\n    conductor\n    copyright\n    director\n    encoded_by\n    encoder_settings\n    initial_key\n    isrc\n    language\n    license\n    lyricist\n    lyrics\n    media\n    publisher\n    set_subtitle\n    url\n\nAdditional `other` field names not documented above may be present, but are\nformat-specific and may change or disappear in future tinytag releases. If\ntinytag does not expose metadata you need, or you wish to standardize more\nfield names, open a feature request on GitHub for discussion.\n\n`other` values are always provided as strings, and are not guaranteed to be\nvalid. Should e.g. the `bpm` value in the file contain non-numeric characters,\ntinytag will provide the string as-is. It is your responsibility to handle\npossible exceptions, e.g. when converting the value to an integer.\n\nMultiple values of the same field type are provided if a file contains them.\nValues are always provided as a list, even when only a single value exists.\n\nExample:\n\n```python\nfrom tinytag import OtherFields, TinyTag\n\ntag: TinyTag = TinyTag.get('/some/music.mp3')\nother_fields: OtherFields = tag.other\ncatalog_numbers: list[str] | None = other_fields.get('catalog_number')\n\nif catalog_numbers:\n    catalog_number: str = catalog_numbers[0]\n    print(catalog_number)\n\nprint(catalog_numbers)\n```\n\nOutput:\n\n    > 10\n    > ['10']\n\nWhen a file contains multiple values for a [common metadata field](#common-metadata)\n(e.g. `artist`), the primary value is accessed through the common attribute\n(`tag.artist`), and any additional values through the `other` dictionary\n(`tag.other['artist']`).\n\nExample:\n\n```python\nfrom tinytag import TinyTag\n\ntag: TinyTag = TinyTag.get('/some/music.mp3')\nartist: str | None = tag.artist\nadditional_artists: list[str] | None = tag.other.get('artist')\n\nprint(artist)\nprint(additional_artists)\n```\n\nOutput:\n\n    > main artist\n    > ['another artist', 'yet another artist']\n\n### All Metadata\n\nIf you need to receive all available metadata as key-value pairs in a flat\ndictionary, use the `as_dict()` method. This combines the common attributes\nand `other` dictionary, which can be more convenient in some cases.\n\n    from tinytag import TinyTag\n\n    tag: TinyTag = TinyTag.get('/some/music.mp3')\n    metadata: dict = tag.as_dict()\n\n### Images\n\nAdditionally, you can also read embedded images by passing a `image=True`\nkeyword argument to `TinyTag.get()`.\n\nIf you need to receive an image of a specific kind, including its description,\nuse `images`:\n\n    tag.images        # available embedded images\n\nThe following common image attributes are available, providing the first\nlocated image of each kind:\n\n    tag.images.front_cover  # front cover as 'Image' object\n    tag.images.back_cover   # back cover as 'Image' object\n    tag.images.media        # media (e.g. CD label) as 'Image' object\n\nWhen present, any additional images are available in an `images.other`\ndictionary, using the following standardized key names:\n\n    generic\n    icon\n    alt_icon\n    front_cover\n    back_cover\n    media\n    leaflet\n    lead_artist\n    artist\n    conductor\n    band\n    composer\n    lyricist\n    recording_location\n    during_recording\n    during_performance\n    screen_capture\n    bright_colored_fish\n    illustration\n    band_logo\n    publisher_logo\n    unknown\n\nProvided values are always lists containing at least one `Image` object.\n\nThe `Image` object provides the following attributes:\n\n    data           # image data as bytes\n    name           # image name/kind as string\n    mime_type      # image MIME type as string\n    description    # image description as string\n\nTo receive any available image, prioritizing the front cover, use `images.any`:\n\n```python\nfrom tinytag import Image, TinyTag\n\ntag: TinyTag = TinyTag.get('/some/music.ogg', image=True)\nimage: Image | None = tag.images.any\n\nif image is not None:\n    data: bytes = image.data\n    name: str = image.name\n    mime_type: str = image.mime_type\n    description: str = image.description\n\n    print(len(data))\n    print(name)\n    print(mime_type)\n    print(description)\n```\n\nOutput:\n\n    > 74452\n    > front_cover\n    > image/jpeg\n    > some image description\n\n> [!WARNING]  \n> `tag.images.any` has replaced `tag.get_image()` in tinytag 2.0.0.\n> `tag.get_image()` will be removed in a future tinytag 2.x release.\n\nTo receive a common image, e.g. `front_cover`:\n\n```python\nfrom tinytag import Image, Images, TinyTag\n\ntag: TinyTag = TinyTag.get('/some/music.ogg', image=True)\nimages: Images = tag.images\ncover_image: Image = images.front_cover\n\nif cover_image is not None:\n    data: bytes = cover_image.data\n    description: str = cover_image.description\n```\n\nTo receive an additional image, e.g. `bright_colored_fish`:\n\n```python\nfrom tinytag import Image, OtherImages, TinyTag\n\ntag: TinyTag = TinyTag.get('/some/music.ogg', image=True)\nother_images: OtherImages = tag.images.other\nfish_images: list[Image] | None = other_images.get('bright_colored_fish')\n\nif fish_images:\n    image = fish_images[0]  # Use first image\n    data = image.data\n    description = image.description\n```\n\n### Encoding\n\nTo open files using a specific encoding, you can use the `encoding` parameter.\nThis parameter is however only used for formats where the encoding is not\nexplicitly specified.\n\n```python\nTinyTag.get('a_file_with_gbk_encoding.mp3', encoding='gbk')\n```\n\n### File-like Objects\n\nTo use a file-like object (e.g. BytesIO) instead of a file path, pass a\n`file_obj` keyword argument:\n\n```python\nTinyTag.get(file_obj=your_file_obj)\n```\n\n### Exceptions\n\n    TinyTagException        # Base class for exceptions\n    ParseError              # Parsing an audio file failed\n    UnsupportedFormatError  # File format is not supported\n\n\n## Changelog\n\n### 2.0.0  (2024-11-03)\n\n- **BREAKING:** Store 'disc', 'disc_total', 'track' and 'track_total' values as int instead of str\n- **BREAKING:** 'as_dict()' method (previously undocumented) returns tag field values in list form\n- **BREAKING:** TinyTagException no longer inherits LookupError\n- **BREAKING:** TinyTag subclasses are now private\n- **BREAKING:** Remove function to use custom audio file samples in tests\n- **BREAKING:** Remove support for Python 2\n- **DEPRECATION:** Mark 'ignore_errors' parameter for TinyTag.get() as obsolete\n- **DEPRECATION:** Mark 'audio_offset' attribute as obsolete\n- **DEPRECATION:** Deprecate 'extra' dict in favor of 'other' dict with values in list form\n- **DEPRECATION:** Deprecate 'get_image()' method in favor of 'images.any' property\n- Add type hints to codebase\n- Provide access to custom metadata fields through the 'other' dict\n- Provide access to all available images\n- Add more standard 'other' fields\n- Use Flit as Python build backend instead of Setuptools\n- ID3: Fix invalid sample rate/duration in some cases\n- ID3: Fix reading of UTF-16 strings without BOM\n- FLAC: Apply ID3 tags after Vorbis\n- OGG/WMA: Set missing 'channels' field\n- WMA: Set missing 'other.copyright' field\n- WMA: Raise exception if file is invalid\n- Various optimizations\n\n### 1.10.1  (2023-10-26)\n\n- Update 'extra' fields with data from other tags #188\n- ID3: Add missing 'extra.copyright' field\n\n### 1.10.0  (2023-10-18)\n\n- Add support for OGG FLAC format #182\n- Add support for OGG Speex format #181\n- Wave: support image loading\n- Add support for file-like objects (BytesIO) #178\n- Add list of supported file extensions #177\n- Fix deprecations related to setuptools #176\n- Fix pathlib support in TinyTag.is_supported()\n- Only remove zero bytes at the end of strings\n- Stricter conditions in while loops\n- OGG: Add stricter magic byte matching for OGG files\n- Compatibility with Python 3.4 and 3.5 is no longer tested\n\n### 1.9.0  (2023-04-23)\n\n- Add bitdepth attribute for lossless audio #157\n- Add recognition of Audible formats #163 (thanks to snowskeleton)\n- Add .m4v to list of supported file extensions #142\n- Aiff: Implement replacement for Python's aifc module #164\n- ID3: Only check for language in COMM and USLT frames #147\n- ID3: Read the correct number of bytes from Xing header #154\n- ID3: Add support for ID3v2.4 TDRC frame #156 (thanks to Uninen)\n- M4A: Add description fields #168 (thanks to snowskeleton)\n- RIFF: Handle tags containing extra zero-byte #141\n- Vorbis: Parse OGG cover art #144 (thanks to Pseurae)\n- Vorbis: Support standard disctotal/tracktotal comments #171\n- Wave: Add proper support for padded IFF chunks\n\n### 1.8.1  (2022-03-12) [still mathiascode-edition]\n\n- MP3 ID3: Set correct file position if tag reading is disabled #119 (thanks to mathiascode)\n- MP3: Fix incorrect calculation of duration for VBR encoded MP3s #128 (thanks to mathiascode)\n\n### 1.8.0  (2022-03-05) [mathiascode-edition]\n\n- Add support for ALAC audio files #130 (thanks to mathiascode)\n- AIFF: Fixed bitrate calculation for certain files #129 (thanks to mathiascode)\n- MP3: Do not round MP3 bitrates #131 (thanks to mathiascode)\n- MP3 ID3: Support any language in COMM and USLT frames #135 (thanks to mathiascode)\n- Performance: Don't use regex when parsing genre #136 (thanks to mathiascode)\n- Disable tag parsing for all formats when requested #137 (thanks to mathiascode)\n- M4A: Fix invalid bitrates in certain files #132 (thanks to mathiascode)\n- WAV: Fix metadata parsing for certain files #133 (thanks to mathiascode)\n\n### 1.7.0. (2021-12-14)\n\n- fixed rare occasion of ID3v2 tags missing their first character, #106\n- allow overriding the default encoding of ID3 tags (e.g. `TinyTag.get(..., encoding='gbk'))`)\n- fixed calculation of bitrate for very short mp3 files, #99\n- utf-8 support for AIFF files, #123\n- fixed image parsing for id3v2 with images containing utf-16LE descriptions, #117\n- fixed ID3v1 tags overwriting ID3v2 tags, #121\n- Set correct file position if tag reading is disabled for ID3 (thanks to mathiascode)\n\n### 1.6.0  (2021-08-28) [aw-edition]\n\n- fixed handling of non-latin encoding types for images (thanks to aw-was-here)\n- added support for ISRC data, available in `extra['isrc']` field (thanks to aw-was-here)\n- added support for AIFF/AIFF-C (thanks to aw-was-here)\n- fixed import deprecation warnings (thanks to idotobi)\n- fixed exception for TinyTag misuse being different in different python versions (thanks to idotobi)\n- added support for ID3 initial key tonality hint, available in `extra['initial_key']`\n- added support for ID3 unsynchronized lyrics, available in `extra['lyrics']`\n- added `extra` field, which may contain additional metadata not available in all file formats\n\n### 1.5.0  (2020-11-05)\n\n- fixed data type to always return str for disc, disc_total, track, track_total #97 (thanks to kostalski)\n- fixed package install being reported as UNKNOWN for some python/pip variations #90 (thanks to russpoutine)\n- Added automatic detection for certain MP4 file headers\n\n### 1.4.0  (2020-04-23)\n\n- detecting file types based on their magic header bytes, #85\n- fixed opus duration being wrong for files with lower sample rate #81\n- implemented support for binary paths #72\n- always cast mp3 bitrates to int, so that CBR and VBR output behaves the sam\n- made __str__ deterministic and use json as output format\n\n### 1.3.0  (2020-03-09)\n\n- added option to ignore encoding errors `ignore_errors` #73\n- Improved text decoding for many malformed files\n\n### 1.2.2  (2019-04-13)\n\n- Improved stability when reading corrupted mp3 files\n\n### 1.2.1  (2019-04-13)\n\n- fixed wav files not correctly reporting the number of channels #61\n\n### 1.2.0  (2019-04-13)\n\n- using setup.cfg instead of setup.py (thanks to scivision)\n- added support for calling TinyTag.get with pathlib.Path (thanks to scivision)\n- added appveyor windows test CI (thanks to scivision)\n- using pytest instead of nosetest (thanks to scivision)\n\n### 1.1.0  (2019-04-13)\n\n- added new field \"composer\" (Thanks to Phil Borman)\n\n### 1.0.1  (2019-04-13)\n\n- fixed ID3 loading for files with corrupt header (thanks to Ian Homer)\n- fixed parsing of duration in wav file (thanks to Ian Homer)\n\n### 1.0.0  (2018-12-12)\n\n- added comment field\n- added wav-riff format support\n- use MP4 parser for m4b files\n- added simple cli tool\n- fix parsing of FLAC files with ID3 header (thanks to minus7)\n- added method `TinyTag.is_supported(filename)`\n\n### 0.19.0 (2018-02-11)\n\n- fixed corrupted images for some mp3s (#45)\n\n### 0.18.0 (2017-04-29)\n\n- fixed wrong bitrate and crash when parsing xing header\n\n### 0.17.0 (2016-10-02)\n\n- supporting ID3v2.2 images\n\n### 0.16.0 (2016-08-06)\n\n- MP4 cover image support\n\n### 0.15.2 (2016-08-06)\n\n- fixed crash for malformed MP4 files (#34)\n\n### 0.15.0 (2016-08-06)\n\n- fixed decoding of UTF-16LE ID3v2 Tags, improved overall stability\n\n### 0.14.0 (2016-06-05):\n\n- MP4/M4A and Opus support\n\n",
    "bugtrack_url": null,
    "license": null,
    "summary": "Read audio file metadata",
    "version": "2.0.0",
    "project_urls": {
        "Homepage": "https://github.com/tinytag/tinytag"
    },
    "split_keywords": [
        "metadata",
        " audio",
        " music",
        " mp3",
        " m4a",
        " wav",
        " ogg",
        " opus",
        " flac",
        " wma",
        " aiff"
    ],
    "urls": [
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "78605440472b05e761d703240e222f7451c86eaca9ae66bf7c66defb26e76772",
                "md5": "073509257443fb73677eca07da17f1ba",
                "sha256": "971b9dceae2d1de73b5e8300639ea0b41454633b899426e702aed15f0e72a9b4"
            },
            "downloads": -1,
            "filename": "tinytag-2.0.0-py3-none-any.whl",
            "has_sig": false,
            "md5_digest": "073509257443fb73677eca07da17f1ba",
            "packagetype": "bdist_wheel",
            "python_version": "py3",
            "requires_python": ">=3.7",
            "size": 30734,
            "upload_time": "2024-11-03T15:25:04",
            "upload_time_iso_8601": "2024-11-03T15:25:04.598774Z",
            "url": "https://files.pythonhosted.org/packages/78/60/5440472b05e761d703240e222f7451c86eaca9ae66bf7c66defb26e76772/tinytag-2.0.0-py3-none-any.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "217a04f50e2c660862b9929a06d710968b2520ae1bf215ed57a9972065d0aeef",
                "md5": "144a35c5733709c87b34da9d555d26ac",
                "sha256": "d041f53d15553bb148549bfbc7feab445caf7105ba95fa2ecb9827bb06b62275"
            },
            "downloads": -1,
            "filename": "tinytag-2.0.0.tar.gz",
            "has_sig": false,
            "md5_digest": "144a35c5733709c87b34da9d555d26ac",
            "packagetype": "sdist",
            "python_version": "source",
            "requires_python": ">=3.7",
            "size": 35367,
            "upload_time": "2024-11-03T15:25:06",
            "upload_time_iso_8601": "2024-11-03T15:25:06.262998Z",
            "url": "https://files.pythonhosted.org/packages/21/7a/04f50e2c660862b9929a06d710968b2520ae1bf215ed57a9972065d0aeef/tinytag-2.0.0.tar.gz",
            "yanked": false,
            "yanked_reason": null
        }
    ],
    "upload_time": "2024-11-03 15:25:06",
    "github": true,
    "gitlab": false,
    "bitbucket": false,
    "codeberg": false,
    "github_user": "tinytag",
    "github_project": "tinytag",
    "travis_ci": false,
    "coveralls": false,
    "github_actions": true,
    "lcname": "tinytag"
}
        
Elapsed time: 0.34387s