PHP package to parse and update audio files metadata, with JamesHeinrich/getID3
.
[!NOTE]
You can check formats supported on Supported formats section.
Audio files can use different formats, this package aims to provide a simple way to read them with JamesHeinrich/getID3
. The JamesHeinrich/getID3
package is excellent to read metadata from audio files, but output is just an array, current package aims to provide a simple way to read audio files with a beautiful API.
8.1
minimumFLAC
: flac
(with apt
, brew
or scoop
)OGG
: vorbis-tools
(with apt
or brew
) / extras/icecast
(with scoop
)You can install the package via composer:
composer require kiwilan/php-audio
Core metadata:
use Kiwilan\Audio\Audio;
$audio = Audio::read('path/to/audio.mp3');
$audio->getTitle(); // `?string` to get title
$audio->getArtist(); // `?string` to get artist
$audio->getAlbum(); // `?string` to get album
$audio->getGenre(); // `?string` to get genre
$audio->getYear(); // `?int` to get year
$audio->getTrackNumber(); // `?string` to get track number
$audio->getComment(); // `?string` to get comment
$audio->getAlbumArtist(); // `?string` to get album artist
$audio->getComposer(); // `?string` to get composer
$audio->getDiscNumber(); // `?string` to get disc number
$audio->isCompilation(); // `bool` to know if is compilation
$audio->getCreationDate(); // `?string` to get creation date
$audio->getCopyright(); // `?string` to get copyright
$audio->getEncoding(); // `?string` to get encoding
$audio->getDescription(); // `?string` to get description
$audio->getSynopsis(); // `?string` to get synopsis
$audio->getLanguage(); // `?string` to get language
$audio->getLyrics(); // `?string`
$audio->getDuration(); // `?float` to get duration in seconds
$audio->getDurationHuman(); // `?string` to get duration in human readable format
Raw tags:
use Kiwilan\Audio\Audio;
$audio = Audio::read('path/to/audio.mp3');
$raw_all = $audio->getRawAll(); // `array` with all tags
$raw = $audio->getRaw(); // `array` with main tag
$title = $audio->getRawKey('title'); // `?string` to get title same as `$audio->getTitle()`
$format = $audio->getRaw('id3v2'); // `?array` with all tags with format `id3v2`
$title = $audio->getRawKey('title', 'id3v2'); // `?string` to get title with format `id3v2`
Additional metadata:
use Kiwilan\Audio\Audio;
$audio = Audio::read('path/to/audio.mp3');
$audio->getPath(); // `string` to get path
$audio->getExtension(); // `string` to get extension
$audio->hasCover(); // `bool` to know if has cover
$audio->isValid(); // `bool` to know if file is valid audio file
$audio->isWritable(); // `bool` to know if file is writable
$audio->getFormat(); // `AudioFormatEnum` to get format (mp3, m4a, ...)
$audio->getType(); // `?AudioTypeEnum` ID3 type (id3, riff, asf, quicktime, matroska, ape, vorbiscomment)
You can use toArray()
method to get raw info:
use Kiwilan\Audio\Audio;
$audio = Audio::read('path/to/audio.mp3');
$audio->toArray(); // `array` with all metadata
Advanced properties:
use Kiwilan\Audio\Audio;
$audio = Audio::read('path/to/audio.mp3');
$audio->getId3Reader(); // `?Id3Reader` reader based on `getID3`
$audio->getMetadata(); // `?AudioMetadata` with audio metadata
$audio->getCover(); // `?AudioCover` with cover metadata
You can update audio files metadata with Audio::class
, but not all formats are supported. See supported formats
[!WARNING]
You can use any property of
Audio::class
but if you use a property not supported by the format, it will be ignored.
use Kiwilan\Audio\Audio;
$audio = Audio::read('path/to/audio.mp3');
$audio->getTitle(); // `Title`
$tag = $audio->write()
->title('New Title')
->artist('New Artist')
->album('New Album')
->genre('New Genre')
->year('2022')
->trackNumber('2/10')
->albumArtist('New Album Artist')
->comment('New Comment')
->composer('New Composer')
->creationDate('2021-01-01')
->description('New Description')
->synopsis('New Synopsis')
->discNumber('2/2')
->encodingBy('New Encoding By')
->encoding('New Encoding')
->isCompilation()
->lyrics('New Lyrics')
->cover('path/to/cover.jpg') // you can use file content `file_get_contents('path/to/cover.jpg')`
->save();
$audio = Audio::read('path/to/audio.mp3');
$audio->getTitle(); // `New Title`
$audio->getCreationDate(); // `null` because `creationDate` is not supported by `MP3`
Some properties are not supported by all formats, for example MP3
can't handle some properties like lyrics
or stik
, if you try to update these properties, they will be ignored.
You can set tags manually with tag()
or tags()
methods, but you need to know the format of the tag, you could use tagFormats
to set formats of tags (if you don't know the format, it will be automatically detected).
[!WARNING]
If you use
tags
method, you have to use key used by metadata container. For example, if you want to set album artist inid3v2
, you have to useband
key. If you want to know which key to use checksrc/Core/AudioCore.php
file.If your key is not supported,
save
method will throw an exception, unless you useskipErrors
.
use Kiwilan\Audio\Audio;
$audio = Audio::read('path/to/audio.mp3');
$audio->getAlbumArtist(); // `Band`
$tag = $audio->write()
->tag('composer', 'New Composer')
->tag('genre', 'New Genre') // can be chained
->tags([
'title' => 'New Title',
'band' => 'New Band', // `band` is used by `id3v2` to set album artist, method is `albumArtist` but `albumArtist` key will throw an exception with `id3v2`
])
->tagFormats(['id3v1', 'id3v2.4']) // optional
->save();
$audio = Audio::read('path/to/audio.mp3');
$audio->getAlbumArtist(); // `New Band`
use Kiwilan\Audio\Audio;
$audio = Audio::read('path/to/audio.mp3');
$audio->getAlbumArtist(); // `Band`
$tag = $audio->write()
->title('New Title')
->albumArtist('New Band') // `albumArtist` will set `band` for `id3v2`, exception safe
->save();
$audio = Audio::read('path/to/audio.mp3');
$audio->getAlbumArtist(); // `New Band`
You can use skipErrors
to prevent exception if you use unsupported format.
use Kiwilan\Audio\Audio;
$audio = Audio::read('path/to/audio.mp3');
$tag = $audio->write()
->tags([
'title' => 'New Title',
'title2' => 'New title', // not supported by `id3v2`, will throw an exception
])
->skipErrors() // will prevent exception
->save();
[!NOTE]
Arrow functions are exception safe for properties but not for unsupported formats.
Audio files format metadata with different methods, JamesHeinrich/getID3
offer to check these metadatas by different methods. In raw_all
property of Audio::class
, you will find raw metadata from JamesHeinrich/getID3
package, like id3v2
, id3v1
, riff
, asf
, quicktime
, matroska
, ape
, vorbiscomment
...
If you want to extract specific field which can be skipped by Audio::class
, you can use raw_all
property.
use Kiwilan\Audio\Audio;
$audio = Audio::read('path/to/audio.mp3');
$raw_all = $audio->getRawAll(); // all formats
$raw = $audio->getRaw(); // main format
use Kiwilan\Audio\Audio;
$audio = Audio::read('path/to/audio.mp3');
$metadata = $audio->getMetadata();
$metadata->getFileSize(); // `?int` in bytes
$metadata->getSizeHuman(); // `?string` (1.2 MB, 1.2 GB, ...)
$metadata->getExtension(); // `?string` (mp3, m4a, ...)
$metadata->getEncoding(); // `?string` (UTF-8...)
$metadata->getMimeType(); // `?string` (audio/mpeg, audio/mp4, ...)
$metadata->getDurationSeconds(); // `?float` in seconds
$metadata->getDurationReadable(); // `?string` (00:00:00)
$metadata->getBitrate(); // `?int` in kbps
$metadata->getBitrateMode(); // `?string` (cbr, vbr, ...)
$metadata->getSampleRate(); // `?int` in Hz
$metadata->getChannels(); // `?int` (1, 2, ...)
$metadata->getChannelMode(); // `?string` (mono, stereo, ...)
$metadata->isLossless(); // `bool` to know if is lossless
$metadata->getCompressionRatio(); // `?float`
$metadata->getFilesize(); // `?int` in bytes
$metadata->getSizeHuman(); // `?string` (1.2 MB, 1.2 GB, ...)
$metadata->getDataFormat(); // `?string` (mp3, m4a, ...)
$metadata->getWarning(); // `?array`
$metadata->getQuicktime(); // `?Id3AudioQuicktime
$metadata->getCodec(); // `?string` (mp3, aac, ...)
$metadata->getEncoderOptions(); // `?string`
$metadata->getVersion(); // `?string`
$metadata->getAvDataOffset(); // `?int` in bytes
$metadata->getAvDataEnd(); // `?int` in bytes
$metadata->getFilePath(); // `?string`
$metadata->getFilename(); // `?string`
$metadata->getLastAccessAt(); // `?DateTime`
$metadata->getCreatedAt(); // `?DateTime`
$metadata->getModifiedAt(); // `?DateTime`
$metadata->toArray();
For quicktime
type, like for M4B audiobook, you can use Id3TagQuicktime
to get more informations.
use Kiwilan\Audio\Audio;
$audio = Audio::read('path/to/audio.m4b');
$quicktime = $audio->getMetadata()->getQuicktime();
$quicktime->getHinting();
$quicktime->getController();
$quicktime->getFtyp();
$quicktime->getTimestampsUnix();
$quicktime->getTimeScale();
$quicktime->getDisplayScale();
$quicktime->getVideo();
$quicktime->getAudio();
$quicktime->getSttsFramecount();
$quicktime->getComments();
$quicktime->getFree();
$quicktime->getWide();
$quicktime->getMdat();
$quicktime->getEncoding();
$quicktime->getChapters(); // ?Id3AudioQuicktimeChapter[]
use Kiwilan\Audio\Audio;
$audio = Audio::read('path/to/audio.mp3');
$cover = $audio->getCover();
$cover->getContents(); // `?string` raw file
$cover->getMimeType(); // `?string` (image/jpeg, image/png, ...)
$cover->getWidth(); // `?int` in pixels
$cover->getHeight(); // `?int` in pixels
id3v2
will be selected before id3v1
or riff
if both are available.Format | Supported | About | ID3 type | Notes |
---|---|---|---|---|
AAC | ❌ | Advanced Audio Coding | ||
ALAC | ✅ | Apple Lossless Audio Codec | quicktime |
|
AIF | ✅ | Audio Interchange File Format (aif) | id3v2 ,riff |
|
AIFC | ✅ | Audio Interchange File Format (aifc) | id3v2 ,riff |
|
AIFF | ✅ | Audio Interchange File Format (aiff) | id3v2 ,riff |
|
DSF | ❌ | Direct Stream Digital Audio | ||
FLAC | ✅ | Free Lossless Audio Codec | vorbiscomment |
|
MKA | ✅ | Matroska | matroska |
Cover not supported |
MKV | ✅ | Matroska | matroska |
Cover not supported |
APE | ❌ | Monkey's Audio | ||
MP3 | ✅ | MPEG audio layer 3 | id3v2 ,id3v1 |
|
MP4 | ✅ | Digital multimedia container format | quicktime |
Partially supported |
M4A | ✅ | mpeg-4 audio | quicktime |
|
M4B | ✅ | Audiobook | quicktime |
|
M4V | ✅ | mpeg-4 video | quicktime |
|
MPC | ❌ | Musepack | ||
OGG | ✅ | Open container format | vorbiscomment |
|
OPUS | ✅ | IETF Opus audio | vorbiscomment |
|
OFR | ❌ | OptimFROG | ||
OFS | ❌ | OptimFROG | ||
SPX | ✅ | Speex | vorbiscomment |
Cover not supported |
TAK | ❌ | Tom's Audio Kompressor | ||
TTA | ✅ | True Audio | ape |
Cover not supported |
WMA | ✅ | Windows Media Audio | asf |
Cover not supported |
WV | ✅ | WavPack | ape |
|
WAV | ✅ | Waveform Audio | id3v2 ,riff |
|
WEBM | ✅ | WebM | matroska |
Cover not supported |
You want to add a format? See FAQ
JamesHeinrich/getID3
can update some formats, but not all.
- ID3v1 (v1 & v1.1)
- ID3v2 (v2.3, v2.4)
- APE (v2)
- Ogg Vorbis comments (need
vorbis-tools
)- FLAC comments (need
flac
)
Format | Notes | Requires |
---|---|---|
FLAC | Cover not supported | flac |
MP3 | ||
OGG | Cover not supported | vorbis-tools |
flac
: with apt
, brew
or scoop
vorbis-tools
: with apt
, brew
or scoop
scoop
, vorbis-tools
is not available, you can use extras/icecast
instead.Audio::class
convert some properties to be more readable.
ape
format: Id3TagApe
asf
format: Id3TagAsf
id3v1
format: Id3TagAudioV1
id3v2
format: Id3TagAudioV2
matroska
format: Id3TagMatroska
quicktime
format: Id3TagQuicktime
vorbiscomment
format: Id3TagVorbisComment
riff
format: Id3TagRiff
unknown
format: Id3TagVorbisComment
ID3 type | Original | New property |
---|---|---|
id3v2 |
band |
album_artist |
id3v2 |
part_of_a_set |
disc_number |
id3v2 |
part_of_a_compilation |
is_compilation |
quicktime |
compilation |
is_compilation |
quicktime |
encoded_by |
encoding_by |
quicktime |
encoding_tool |
encoding |
quicktime |
description_long |
synopsis |
asf |
albumartist |
album_artist |
asf |
partofset |
disc_number |
asf |
encodingsettings |
encoding |
vorbiscomment |
encoder |
encoding |
vorbiscomment |
albumartist |
album_artist |
vorbiscomment |
discnumber |
disc_number |
vorbiscomment |
compilation |
is_compilation |
vorbiscomment |
tracknumber |
track_number |
matroska |
disc |
disc_number |
matroska |
part_number |
track_number |
matroska |
date |
year |
matroska |
compilation |
is_compilation |
matroska |
encoder |
encoding |
ape |
disc |
disc_number |
ape |
compilation |
is_compilation |
ape |
track |
track_number |
ape |
date |
year |
ape |
encoder |
encoding |
composer test
In Audio::class
, you have a property raw_all
which contains all raw metadata, if JamesHeinrich/getID3
support this field, you will find it in this property.
use Kiwilan\Audio\Audio;
$audio = Audio::read('path/to/audio.mp3');
$raw_all = $audio->getRawAll());
$custom = null;
$id3v2 = $raw_all['id3v2'] ?? [];
if ($id3v2) {
$custom = $id3v2['custom'] ?? null;
}
If your field could be added to global properties of Audio::class
, you could create an an issue.
null
, what can I do?You can check extras
property to know if some metadata are available.
use Kiwilan\Audio\Audio;
$audio = Audio::read('path/to/audio.mp3');
$raw_all = $audio->getRawAll();
var_dump($raw_all);
If you find metadata which are not parsed by Audio::class
, you can create an issue, otherwise JamesHeinrich/getID3
doesn't support this metadata.z
You can create an issue with the format name and a link to the format documentation. If JamesHeinrich/getID3
support this format, I will add it to this package but if you want to contribute, you can create a pull request with the format implementation.
Please give me an example file to test the format.
You can create an issue with informations.
This package doesn't provide a way to convert audio files, but you can use ffmpeg to convert audio files and PHP-FFMpeg/PHP-FFMpeg.
Please see CHANGELOG for more information on what has changed recently.
ewilan-riviere
: package authorJamesHeinrich/getID3
: parser used to read audio filesspatie/package-skeleton-php
: package skeleton used to create this packageThe MIT License (MIT). Please see License File for more information.