From aa7c8ad2913754b2a572ffe3602d6ee7d0e22804 Mon Sep 17 00:00:00 2001 From: thebigmunch Date: Thu, 17 Oct 2019 18:57:54 -0400 Subject: [PATCH] Add MP4 load(s) support [#14] --- docs/api.rst | 11 + src/audio_metadata/api.py | 7 + src/audio_metadata/formats/__init__.py | 2 + src/audio_metadata/formats/mp4.py | 675 +++++++++++++++++++++++++ src/audio_metadata/formats/mp4_tags.py | 378 ++++++++++++++ src/audio_metadata/formats/tables.py | 113 +++++ tests/audio/mp4-aac.m4a | Bin 0 -> 6667 bytes tests/audio/mp4-ac3.m4a | Bin 0 -> 121911 bytes tests/audio/mp4-alac.m4a | Bin 0 -> 7061 bytes 9 files changed, 1186 insertions(+) create mode 100644 src/audio_metadata/formats/mp4.py create mode 100644 src/audio_metadata/formats/mp4_tags.py create mode 100644 tests/audio/mp4-aac.m4a create mode 100644 tests/audio/mp4-ac3.m4a create mode 100644 tests/audio/mp4-alac.m4a diff --git a/docs/api.rst b/docs/api.rst index 58a7afa..52b1982 100644 --- a/docs/api.rst +++ b/docs/api.rst @@ -87,6 +87,17 @@ MP3 .. autoclass:: XingToC +MP4 +--- + +.. autoclass:: MP4 + +.. autoclass:: MP4Atom +.. autoclass:: MP4Atoms +.. autoclass:: MP4StreamInfo +.. autoclass:: MP4Tags + + Ogg --- diff --git a/src/audio_metadata/api.py b/src/audio_metadata/api.py index 94cb497..513d203 100644 --- a/src/audio_metadata/api.py +++ b/src/audio_metadata/api.py @@ -19,6 +19,7 @@ from .formats import ( FLAC, MP3, + MP4, WAV, ID3v2, MP3StreamInfo, @@ -68,6 +69,12 @@ def determine_format(data): if d.startswith(b'RIFF'): return WAV + if ( + d[4:8].lower() == b'ftyp' + and d[8:].lower().startswith((b'dash', b'm4a', b'mp4')) + ): + return MP4 + if d.startswith(b'ID3'): ID3v2.parse(data) diff --git a/src/audio_metadata/formats/__init__.py b/src/audio_metadata/formats/__init__.py index afd3d3d..971ada1 100644 --- a/src/audio_metadata/formats/__init__.py +++ b/src/audio_metadata/formats/__init__.py @@ -3,6 +3,7 @@ from .id3v2 import * from .id3v2_frames import * from .mp3 import * +from .mp4 import * from .ogg import * from .oggopus import * from .oggvorbis import * @@ -17,6 +18,7 @@ *id3v2_frames.__all__, *id3v2.__all__, *mp3.__all__, + *mp4.__all__, *ogg.__all__, *oggopus.__all__, *oggvorbis.__all__, diff --git a/src/audio_metadata/formats/mp4.py b/src/audio_metadata/formats/mp4.py new file mode 100644 index 0000000..680e206 --- /dev/null +++ b/src/audio_metadata/formats/mp4.py @@ -0,0 +1,675 @@ +__all__ = [ + 'MP4', + 'MP4Atom', + 'MP4Atoms', + 'MP4StreamInfo', + 'MP4Tags', +] + +import os +import struct +from collections import defaultdict + +from attr import attrib, attrs +from bidict import frozenbidict +from tbm_utils import ( + AttrMapping, + LabelList, + datareader, +) + +from .mp4_tags import MP4Tag +from .tables import ( + MP4AudioObjectTypes, + MP4SamplingFrequencies, +) +from ..exceptions import ( + FormatError, + UnsupportedFormat, +) +from ..models import ( + Format, + StreamInfo, + Tags, +) + +try: # pragma: nocover + import bitstruct.c as bitstruct +except ImportError: # pragma: nocover + import bitstruct + +PARENT_ATOMS = { + 'ilst', + 'mdia', + 'meta', + 'minf', + 'moof', + 'moov', + 'stbl', + 'traf', + 'trak', + 'udta' +} +EXT_DESCRIPTOR_TYPES = {b'\x80', b'\x81', b'\xFE'} + + +# TODO: Custom type? +@attrs( + repr=False, + kw_only=True, +) +class MP4Atom(AttrMapping): + _start = attrib() + _size = attrib() + _data_start = attrib() + type = attrib() # noqa + _children = attrib(default=[]) + + def __repr__(self): + repr_dict = {k: v for k, v in self.__dict__.items() if not k.startswith('_')} + + return super().__repr__(repr_dict=repr_dict) + + @datareader + @classmethod + def parse(cls, data, level=0): + children = [] + + atom_start = data_start = data.tell() + atom_header = data.read(8) + + try: + atom_size, atom_type = struct.unpack('>I4s', atom_header[0:8]) + data_start += 8 + except struct.error: + raise FormatError("Invalid MP4 atom.") + + atom_type = atom_type.decode('iso-8859-1') + + if atom_size == 1: + try: + atom_size = struct.unpack('>Q', data.read(8))[0] + data_start += 8 + except struct.error: + raise FormatError("Invalid MP4 atom.") + + if atom_size < 16: + raise FormatError("Invalid MP4 atom.") + elif atom_size == 0: + if level != 0: + raise FormatError("Invalid MP4 atom.") + + data.seek(0, os.SEEK_END) + atom_size = data.tell() - atom_start + data.seek(atom_start + 8, os.SEEK_SET) + elif atom_size < 8: + raise FormatError("Invalid MP4 atom.") + + if atom_type in PARENT_ATOMS: + if atom_type == 'meta': + data.seek(4, os.SEEK_CUR) + + while data.tell() < atom_start + atom_size: + children.append(MP4Atom.parse(data, level + 1)) + else: + data.seek(atom_start + atom_size, os.SEEK_SET) + + return cls( + start=atom_start, + size=atom_size, + data_start=data_start, + type=atom_type, + children=children, + ) + + def get_child(self, path): + if not path: + return self + + if not self._children: + raise KeyError("No children found.") + + if isinstance(path, str): + path = path.split('.') + + for child in self._children: + if child.type == path[0]: + return child.get_child(path[1:]) + else: + raise KeyError('Path not found.') + + @datareader + def read_data(self, data): + data.seek(self._data_start, os.SEEK_SET) + atom_data = data.read(self._size - (self._data_start - self._start)) + + return atom_data + + +# TODO: Custom type? +class MP4Atoms(LabelList): + item_label = ('atom', 'atoms') + + def __init__(self, items): + super().__init__(items) + + def __getitem__(self, path): + if isinstance(path, str): + path = path.split('.') + + for atom in self.data: + if atom.type == path[0]: + return atom.get_child(path[1:]) + else: + raise KeyError(f'No atom of type {path[0]} found.') + + return list.__getitem__(self.data, path) + + @datareader + @classmethod + def parse(cls, data): + atoms = [] + while True: + try: + atoms.append(MP4Atom.parse(data, level=0)) + except (FormatError, struct.error): + break + + return cls(atoms) + + +class MP4Tags(Tags): + FIELD_MAP = frozenbidict({ + 'album': '©alb', + 'albumsort': 'soal', + 'albumartist': 'aART', + 'albumartistsort': 'soaa', + 'artist': '©ART', + 'artistsort': 'soar', + 'bpm': 'tmpo', + 'category': 'catg', + 'comment': '©cmt', + 'compilation': 'cpil', + 'composer': '©wrt', + 'composersort': 'soco', + 'copyright': 'cprt', + 'date': '©day', + 'description': 'desc', + 'discnumber': 'disk', + 'encodedby': '©too', + 'freeform': '----', + 'gapless': 'pgap', + 'genre': '©gen', + 'genre_id3': 'gnre', + 'grouping': '©grp', + 'keyword': 'keyw', + 'lyrics': '©lyr', + 'pictures': 'covr', + 'podcast': 'pcst', + 'podcasturl': 'purl', + 'rating': 'rtng', + 'title': '©nam', + 'titlesort': 'sonm', + 'tracknumber': 'trkn' + }) + + @datareader + @classmethod + def parse(cls, data, ilst): + fields = defaultdict(list) + + for child in ilst._children: + tag = MP4Tag.parse(data, child) + if tag is None: # TODO + continue + + fields[tag.name] = tag.value + + return cls(**fields) + + +@attrs( + repr=False, + kw_only=True, +) +class MP4StreamInfo(StreamInfo): + _start = attrib() + _size = attrib() + bit_depth = attrib() + bitrate = attrib() + channels = attrib() + codec = attrib() + codec_description = attrib() + duration = attrib() + sample_rate = attrib() + + @staticmethod + def _parse_audio_sample_entry(ase_data): + channels = struct.unpack( + '>H', + ase_data[16:18] + )[0] + + bit_depth = struct.unpack( + '>H', + ase_data[18:20] + )[0] + + sample_rate = struct.unpack( + '>I', + ase_data[22:26] + )[0] + + return channels, bit_depth, sample_rate + + @datareader + @staticmethod + def _parse_esds(data): + def _parse_audio_object_type(data): + audio_object_type_index = data.readbits(5) + audio_object_type_ext = None + + if audio_object_type_index == 31: + data.readbits(5) + + audio_object_type_ext = data.readbits(6) + audio_object_type_index = 32 + audio_object_type_ext + + return audio_object_type_index, audio_object_type_ext + + def _parse_sample_rate(data): + sampling_frequency_index = data.readbits(4) + if sampling_frequency_index == 15: + sample_rate = data.readbits(24) + else: + try: + sample_rate = MP4SamplingFrequencies[sampling_frequency_index] + except IndexError: + sample_rate = None + + return sample_rate + version = data.readbits(8) + if version != 0: + raise Exception + + data.seek(3, os.SEEK_CUR) + if data.readbits(8) == 3: + while True: + b = data.read(1) + if b not in EXT_DESCRIPTOR_TYPES: + break + + descriptor_type_length = b + es_id = data.readbits(16) + + stream_dependence_flag, url_flag, ocr_stream_flag = bitstruct.unpack( + 'b1 b1 b1', + data.read(1) + ) # TODO: Stream priority + + if stream_dependence_flag: + stream_dependence = data.readbits(16) + if url_flag: + url_length = data.readbits(8) + url = data.read(url_length) + if ocr_stream_flag: + ocr = data.readbits(16) + + if data.readbits(8) == 4: + while True: + b = data.read(1) + if b not in EXT_DESCRIPTOR_TYPES: + break + + object_type_indication = data.readbits(8) + stream_type, up_stream, reserved = bitstruct.unpack( + 'u6 b1 u1', + data.read(1) + ) + + if ( + object_type_indication != 64 + or stream_type != 5 + ): + raise Exception + + buffer_size = data.readbits(24) + max_bitrate = data.readbits(32) + average_bitrate = data.readbits(32) + + if data.readbits(8) == 5: + while True: + b = data.read(1) + if b not in EXT_DESCRIPTOR_TYPES: + break + + audio_object_type_index, audio_object_type_ext = _parse_audio_object_type(data) + + try: + codec_description = MP4AudioObjectTypes[audio_object_type_index] + except IndexError: + codec_description = None + + sample_rate = _parse_sample_rate(data) + + channel_config = data.readbits(4) # TODO: Channels + + spectral_band_replication = False + parametric_stereo = False + ext_sample_rate = None + + if audio_object_type_index in [5, 29]: + audio_object_type_ext = 5 + spectral_band_replication = True + + if audio_object_type_index == 29: + parametric_stereo = True + + ext_sample_rate = _parse_sample_rate(data) + audio_object_type_index, _ = _parse_audio_object_type(data) + + if audio_object_type_index == 22: + ext_channel_config = data.readbits(4) + else: + audio_object_type_ext = None + + if audio_object_type_index in [1, 2, 3, 4, 6, 7, 17, 19, 20, 21, 22, 23]: + try: + data.readbits(1) + + core_coder_dependence = data.readbits(1) + if core_coder_dependence: + data.readbits(14) + + extension_flag = data.readbits(1) + + if not channel_config: # TODO + element_instance_tag = data.readbits(4) + object_type = data.readbits(2) + sample_rate_index = data.readbits(4) + num_front_channel_elements = data.readbits(4) + num_side_channel_elements = data.readbits(4) + num_back_channel_elements = data.readbits(4) + num_lfe_channel_elements = data.readbits(2) + num_associated_data_elements = data.readbits(3) + num_valid_cc_elements = data.readbits(4) + + mono_mixdown_present = data.readbits(1) + mono_mixdown = data.readbits(4) + + stereo_mixdown_present = data.readbits(1) + stereo_mixdown = data.readbits(4) + + matrix_mixdown_present = data.readbits(1) + matrix_mixdown = data.readbits(3) + + elements = ( + num_front_channel_elements + num_side_channel_elements + num_back_channel_elements + ) + channels = 0 + for i in range(elements): + channels += 1 + element_is_cpe = data.readbits(1) + if element_is_cpe: + channels += 1 + + data.readbits(4) + + channels += num_lfe_channel_elements + data.readbits(4 * num_lfe_channel_elements) + data.readbits(4 * num_associated_data_elements) + data.readbits(5 * num_valid_cc_elements) + data.readbits(data.bit_count) + comment_field_bytes = data.readbits(8) + data.read(comment_field_bytes) + + if audio_object_type_index in [6, 20]: + data.readbits(3) + + if extension_flag: + if audio_object_type_index == 22: + data.readbits(16) + + if audio_object_type_index in [17, 19, 20, 23]: + data.readbits(3) + + extension_flag_3 = data.readbits(1) + if extension_flag_3 != 0: + raise Exception # TODO + except Exception: + raise + else: + raise UnsupportedFormat("Not a supported MP4 audio object type.") + + if audio_object_type_index in [17, 19, 20, 21, 22, 23, 24, 25, 26, 27, 39]: + ep_config = data.readbits(2) + if ep_config in [2, 3]: + raise UnsupportedFormat + + # TODO: Finish/check. + if ( + audio_object_type_ext != 5 + and (len(data.peek()) * 8) - data.bit_count >= 16 + ): + sync_extension_type = data.readbits(11) + if sync_extension_type == 695: + audio_object_type_ext, _ = _parse_audio_object_type(data) + + if audio_object_type_ext == 5: + spectral_band_replication = bool(data.readbits(1)) + if spectral_band_replication: + ext_sample_rate = _parse_sample_rate(data) + + if (len(data.peek()) * 8) - data.bit_count >= 12: + sync_extension_type = data.readbits(11) + if sync_extension_type == 1352: + parametric_stereo = bool(data.readbits(1)) + + if audio_object_type_ext == 22: + spectral_band_replication = bool(data.readbits(1)) + if spectral_band_replication: + ext_sample_rate = _parse_sample_rate(data) + ext_channel_config = data.readbits(4) + + if spectral_band_replication and parametric_stereo: + codec_description = "AAC HE (v2)" + elif spectral_band_replication: + codec_description = "AAC HE (v1)" + + return ( + average_bitrate, + sample_rate, + codec_description, + ) + + @staticmethod + def _parse_alac(alac_data): + version = alac_data[0] + if version != 0: + raise Exception + + compatible_version = alac_data[8] + if compatible_version != 0: + raise UnsupportedFormat + + bit_depth = alac_data[9] + channels = alac_data[13] + bitrate = bitstruct.unpack('u32', alac_data[20:24])[0] + sample_rate = bitstruct.unpack('u32', alac_data[24:28])[0] + + return bit_depth, channels, bitrate, sample_rate + + @staticmethod + def _parse_ac3(ac3_data): + _, _, _, acmod, lfeon, bitrate_index = bitstruct.unpack( + 'u2 u5 u3 u3 u1 u5', + ac3_data[0:3] + ) + + channels = [2, 1, 2, 3, 3, 4, 4, 5][acmod] + lfeon + + try: + bitrate = [ + 32, + 40, + 48, + 56, + 64, + 80, + 96, + 112, + 128, + 160, + 192, + 224, + 256, + 320, + 384, + 448, + 512, + 576, + 640 + ][bitrate_index] * 1000 + except IndexError: + bitrate = None + + return channels, bitrate + + @staticmethod + def _parse_mdhd(mdhd_data): + if len(mdhd_data) < 4: + raise Exception + + version = mdhd_data[0] + flags = int.from_bytes(mdhd_data[1:4], 'big') + + if version == 0: + offset = 8 + struct_pattern = '>2I' + elif version == 1: + offset = 16 + struct_pattern = '>IQ' + else: + raise Exception + + end = offset + struct.calcsize(struct_pattern) + unit, size = struct.unpack(struct_pattern, mdhd_data[4:][offset:end]) + + try: + duration = size / unit + except ZeroDivisionError: + duration = 0 + + return version, flags, size, duration + + @datareader + @classmethod + def parse(cls, data, atoms): + try: + moov = atoms['moov'] + except KeyError: + raise + + for child in moov._children: + if child.type == 'trak': + trak = child + + hdlr = trak.get_child('mdia.hdlr') + hdlr_data = hdlr.read_data(data) + + if hdlr_data[8:12] == b'soun': + break + else: + raise Exception + + mdhd = trak.get_child('mdia.mdhd') + version, flags, size, duration = cls._parse_mdhd(mdhd.read_data(data)) + + if version != 0: + raise Exception + + try: + stsd = trak.get_child('mdia.minf.stbl.stsd') + except KeyError: + raise + else: + stsd_data = stsd.read_data(data) + + num_entries = struct.unpack( + '>I', + stsd_data[4:8] + )[0] + + if num_entries == 0: + raise Exception + + audio_sample_entry = MP4Atom.parse(stsd_data[8:]) + ase_data = audio_sample_entry.read_data(stsd_data[8:]) + codec = audio_sample_entry.type + + channels, bit_depth, sample_rate = ( + cls._parse_audio_sample_entry(ase_data) + ) + + extra = MP4Atom.parse(ase_data[28:]) + + bitrate = None + # TODO: Other formats. + if codec == 'mp4a' and extra.type == 'esds': + esds_data = extra.read_data(ase_data[28:]) + average_bitrate, sample_rate, codec_description = cls._parse_esds(esds_data) + elif codec == 'alac' and extra.type == 'alac': + alac_data = extra.read_data(ase_data[28:]) + bit_depth, channels, bitrate, sample_rate = cls._parse_alac(alac_data) + codec_description = 'ALAC' + elif codec == 'ac-3' and extra.type == 'dac3': + ac3_data = extra.read_data(ase_data[28:]) + channels, bitrate_ = cls._parse_ac3(ac3_data) + + if bitrate_: + bitrate = bitrate_ + + codec_description = 'AC-3' + else: + raise UnsupportedFormat("Not a supported MP4 audio codec.") + + audio_start = atoms['mdat']._start + audio_size = atoms['mdat']._size + + if not bitrate: + bitrate = audio_size * 8 / duration + + return cls( + start=audio_start, + size=audio_size, + bit_depth=bit_depth, + bitrate=bitrate, + channels=channels, + codec=codec, + codec_description=codec_description, + duration=duration, + sample_rate=sample_rate, + ) + + +class MP4(Format): + @classmethod + def parse(cls, data): + self = super()._load(data) + + atoms = MP4Atoms.parse(self._obj) + + self.streaminfo = MP4StreamInfo.parse(data, atoms) + + try: + ilst = atoms['moov.udta.meta.ilst'] + except KeyError: + self.tags = MP4Tags() + else: + self.tags = MP4Tags.parse(data, ilst) + + self.pictures = self.tags.pop('pictures', []) + + self._obj.close() + + return self diff --git a/src/audio_metadata/formats/mp4_tags.py b/src/audio_metadata/formats/mp4_tags.py new file mode 100644 index 0000000..44e588a --- /dev/null +++ b/src/audio_metadata/formats/mp4_tags.py @@ -0,0 +1,378 @@ +__all__ = [ + 'MP4BooleanTag', + 'MP4Cover', + 'MP4CoverTag', + 'MP4FloatTag', + 'MP4Freeform', + 'MP4FreeformDecoders', + 'MP4FreeformTag', + 'MP4GenreTag', + 'MP4IntegerTag', + 'MP4NumberTag', + 'MP4TextTag', + 'MP4Tag', + 'MP4TagTypes', +] + +import os +import struct + +from attr import attrib, attrs +from tbm_utils import ( + AttrMapping, + datareader, +) + +from .tables import ( + ID3v1Genres, + MP4AtomDataType, + MP4CoverFormat, +) +from ..models import ( + Picture, + Tag, +) +from ..utils import get_image_size + + +# https://developer.apple.com/library/archive/documentation/QuickTime/QTFF/Metadata/Metadata.html#//apple_ref/doc/uid/TP40000939-CH1-SW34 +MP4FreeformDecoders = { + MP4AtomDataType.IMPLICIT: lambda d: d, + MP4AtomDataType.UTF8: lambda d: d.decode('utf-8', 'replace'), + MP4AtomDataType.UTF16: lambda d: d.decode('utf-16', 'replace'), + MP4AtomDataType.SJIS: lambda d: d.decode('s/jis', 'replace'), + 4: lambda d: d.decode('utf-8', 'replace'), + 5: lambda d: d.decode('utf-16', 'replace'), + MP4AtomDataType.URL: lambda d: d.decode('utf-8', 'replace'), + MP4AtomDataType.DURATION: lambda d: struct.unpack('>L', d)[0], + MP4AtomDataType.DATETIME: lambda d: struct.unpack('>L', d)[0], # TODO: Can be 32-bit or 64-bit. + MP4AtomDataType.GIF: lambda d: d, + MP4AtomDataType.JPEG: lambda d: d, + MP4AtomDataType.PNG: lambda d: d, + MP4AtomDataType.SIGNED_INT_BE: lambda d: struct.unpack('>b', d)[0], # TODO: Can be 1, 2, 3, 4, 8 bytes. + MP4AtomDataType.UNSIGNED_INT_BE: lambda d: struct.unpack('>B', d)[0], # TODO: Can be 1, 2, 3, 4, 8 bytes. + MP4AtomDataType.FLOAT_BE_32: lambda d: struct.unpack('>f', d)[0], + MP4AtomDataType.FLOAT_BE_64: lambda d: struct.unpack('>d', d)[0], + MP4AtomDataType.RIAA_PA: lambda d: struct.unpack('>d'. d)[0], + MP4AtomDataType.UPC: lambda d: d, + MP4AtomDataType.BMP: lambda d: d, + 28: lambda d: d, + MP4AtomDataType.SIGNED_INT: lambda d: struct.unpack('b', d)[0], + MP4AtomDataType.SIGNED_INT_BE_16: lambda d: struct.unpack('>h', d)[0], + MP4AtomDataType.SIGNED_INT_BE_32: lambda d: struct.unpack('>i', d)[0], + MP4AtomDataType.SIGNED_INT_BE_64: lambda d: struct.unpack('>q', d)[0], + MP4AtomDataType.UNSIGNED_INT: lambda d: struct.unpack('B', d)[0], + MP4AtomDataType.UNSIGNED_INT_BE_16: lambda d: struct.unpack('>H', d)[0], + MP4AtomDataType.UNSIGNED_INT_BE_32: lambda d: struct.unpack('>I', d)[0], + MP4AtomDataType.UNSIGNED_INT_BE_64: lambda d: struct.unpack('>Q', d)[0] +} + + +class MP4Cover(Picture): + @datareader + @classmethod + def parse(cls, data): + size = struct.unpack('>I', data.read(4))[0] + data.seek(4, os.SEEK_CUR) + format_ = MP4CoverFormat(struct.unpack('>I', data.read(4))[0]) + data.seek(4, os.SEEK_CUR) + image_data = data.read(size - 16) + width, height = get_image_size(image_data) + + return cls(format=format_, width=width, height=height, data=image_data) + + +@attrs( + repr=False, + kw_only=True, +) +class MP4Freeform(AttrMapping): # TODO: Other attributes. + description = attrib() + name = attrib() + data_type = attrib(converter=MP4AtomDataType) + value = attrib() + + +@attrs( + repr=False, + kw_only=True, +) +class MP4Tag(Tag): + @datareader + @classmethod + def _parse_atom(cls, data, atom): + atom_data = atom.read_data(data) + position = 0 + + while position < atom._size - 8: + size, atom_name = struct.unpack('>I4s', atom_data[position : position + 8]) + if atom_name != b'data': + raise Exception # TODO + + version = atom_data[position + 8] + flags = struct.unpack('>I', b'\x00' + atom_data[position + 9: position + 12]) + + yield version, flags, atom_data[position + 16 : position + size] + + position += size + + @datareader + @classmethod + def parse(cls, data, atom): # TODO: Move tag parsing logic into tag classes. + if atom.type in MP4TagTypes: + return MP4TagTypes[atom.type]._parse(data, atom) + + +@attrs( + repr=False, + kw_only=True, +) +class MP4BooleanTag(MP4Tag): + @datareader + @classmethod + def _parse(cls, data, atom): + _, _, atom_data = next(cls._parse_atom(data, atom)) + if len(atom_data) != 1: + raise Exception # TODO + + return cls( + name=atom.type, + value=bool(atom_data[0]), + ) + + +@attrs( + repr=False, + kw_only=True, +) +class MP4CoverTag(MP4Tag): + @datareader + @classmethod + def _parse(cls, data, atom): + atom_data = atom.read_data(data) + + covers = [] + remaining = atom_data + while remaining: + size, name = struct.unpack('>I4s', remaining[:8]) + if name != b'data': + if name == b'name': + remaining = remaining[size:] + continue + else: + raise Exception # TODO + + covers.append(MP4Cover.parse(remaining)) + remaining = remaining[size:] + + return cls( + name='covr', + value=covers, + ) + + +@attrs( + repr=False, + kw_only=True, +) +class MP4FloatTag(MP4Tag): + value = attrib(converter=float) + + +@attrs( + repr=False, + kw_only=True, +) +class MP4FreeformTag(MP4Tag): + @datareader + @classmethod + def _parse(cls, data, atom): + atom_data = atom.read_data(data) + + size = struct.unpack('>I', atom_data[:4])[0] + description = atom_data[12:size].decode('iso-8859-1') + position = size + size = struct.unpack('>I', atom_data[position : position + 4])[0] + name = atom_data[position + 12 : position + size].decode('iso-8859-1') + position += size + value = [] + + while position < atom._size - 8: + size, atom_name = struct.unpack('>I4s', atom_data[position : position + 8]) + if atom_name != b'data': + raise Exception # TODO + + version = atom_data[position + 8] + data_type = MP4AtomDataType(struct.unpack('>I', b'\x00' + atom_data[position + 9 : position + 12])[0]) + + value.append( + MP4Freeform( + description=description, + name=name, + data_type=data_type, + value=atom_data[position + 16 : position + size], + ) + ) + + position += size + + return cls( + name=atom.type, + value=value, + ) + + +@attrs( + repr=False, + kw_only=True, +) +class MP4GenreTag(MP4Tag): + @datareader + @classmethod + def _parse(cls, data, atom): + values = [] + for _, _, atom_data in cls._parse_atom(data, atom): + if len(atom_data) != 2: + raise Exception # TODO + + genre_index = struct.unpack('>H', atom_data)[0] + try: + genre = ID3v1Genres[genre_index] + except IndexError: + raise Exception # TODO + else: + values.append(genre) + + +@attrs( + repr=False, + kw_only=True, +) +class MP4IntegerTag(MP4Tag): + @datareader + @classmethod + def _parse(cls, data, atom): + values = [] + for _, _, atom_data in cls._parse_atom(data, atom): + if len(atom_data) == 1: + value = struct.unpack('>b', atom_data)[0] + elif len(atom_data) == 2: + value = struct.unpack('>h', atom_data)[0] + elif len(atom_data) == 3: + value = struct.unpack('>i', atom_data + b'\x00')[0] + elif len(atom_data) == 4: + value = struct.unpack('>i', atom_data)[0] + elif len(atom_data) == 8: + value = struct.unpack('>q', atom_data)[0] + else: + raise Exception # TODO + + values.append(value) + + return cls( + name=atom.type, + value=values, + ) +@attrs( + repr=False, + kw_only=True, +) +class MP4NumberTag(MP4Tag): + @datareader + @classmethod + def _parse(cls, data, atom): + atom_data = atom.read_data(data) + + number, total = struct.unpack('>HH', atom_data[18:22]) + + return cls( + name=atom.type, + value=f"{number}/{total}" + ) + + @property + def number(self): + return self.value.split('/')[0] + + @property + def total(self): + try: + tot = self.value.split('/')[1] + except IndexError: + tot = None + + return tot + + +@attrs( + repr=False, + kw_only=True, +) +class MP4TextTag(MP4Tag): + @datareader + @classmethod + def _parse(cls, data, atom): + values = [] + for _, _, atom_data in cls._parse_atom(data, atom): + try: + values.append(atom_data.decode('utf-8')) + except UnicodeDecodeError: + raise # TODO + + return cls( + name=atom.type, + value=values, + ) + + +MP4TagTypes = { + '----': MP4FreeformTag, + '©ART': MP4TextTag, + '©alb': MP4TextTag, + '©cmt': MP4TextTag, + '©day': MP4TextTag, + '©gen': MP4TextTag, + '©grp': MP4TextTag, + '©lyr': MP4TextTag, + '©mvc': MP4IntegerTag, + '©mvi': MP4IntegerTag, + '©nam': MP4TextTag, + '©too': MP4TextTag, + '©wrt': MP4TextTag, + 'aART': MP4TextTag, + 'akID': MP4IntegerTag, + 'atID': MP4IntegerTag, + 'catg': MP4TextTag, + 'cmID': MP4IntegerTag, + 'cnID': MP4IntegerTag, + 'covr': MP4CoverTag, + 'cpil': MP4BooleanTag, + 'cprt': MP4TextTag, + 'desc': MP4TextTag, + 'disk': MP4NumberTag, + 'egid': MP4TextTag, + 'geID': MP4IntegerTag, + 'gnre': MP4GenreTag, + 'hdvd': MP4IntegerTag, + 'keyw': MP4TextTag, + 'pcst': MP4BooleanTag, + 'pgap': MP4BooleanTag, + 'plID': MP4IntegerTag, + 'purd': MP4TextTag, + 'purl': MP4TextTag, + 'rtng': MP4IntegerTag, + 'sfID': MP4IntegerTag, + 'shwm': MP4IntegerTag, + 'soaa': MP4TextTag, + 'soal': MP4TextTag, + 'soar': MP4TextTag, + 'soco': MP4TextTag, + 'sonm': MP4TextTag, + 'sosn': MP4TextTag, + 'stik': MP4IntegerTag, + 'tmpo': MP4IntegerTag, + 'trkn': MP4NumberTag, + 'tves': MP4IntegerTag, + 'tvsh': MP4TextTag, + 'tvsn': MP4IntegerTag, +} diff --git a/src/audio_metadata/formats/tables.py b/src/audio_metadata/formats/tables.py index e619cab..21a8c42 100644 --- a/src/audio_metadata/formats/tables.py +++ b/src/audio_metadata/formats/tables.py @@ -20,6 +20,10 @@ 'MP3ChannelMode', 'MP3SampleRates', 'MP3SamplesPerFrame', + 'MP4AtomDataType', + 'MP4AudioObjectTypes', + 'MP4CoverFormat', + 'MP4SamplingFrequencies' ] from enum import ( @@ -651,3 +655,112 @@ class MP3ChannelMode(_BaseIntEnum): (2.5, 2): (1152, 1), (2.5, 3): (576, 1), } + +MP4AudioObjectTypes = [ + None, + "AAC MAIN", + "AAC LC", + "AAC SSR", + "AAC LTP", + "SBR", + "AAC Scalable", + "TwinVQ", + "CELP", + "HVXC", + None, + None, + "TTSI", + "Main synthetic", + "Wavetable sample-based synthesis", + "General MIDI", + "Algorithmic Synthesis and Audio Effects", + "ER AAC LC", + None, + "ER AAC LTP", + "ER AAC Scalable", + "ER Twin VQ", + "ER BSAC", + "ER AAC LD", + "ER CELP", + "ER HVXC", + "ER HILN", + "ER Parametric", + "SSC", + "PS", + "MPEG Surround", + None, + "MPEG-1/2 Layer-1", + "MPEG-1/2 Layer-2", + "MPEG-1/2 Layer-3", + "DST", + "ALS", + "SLS", + "SLS non-core", + "ER AAC ELD", + "SMR Simple", + "SMR Main", + "USAC", + "SAOC", + "LD MPEG Surround", + "USAC" +] + + +class MP4AtomDataType(_BaseIntEnum): + IMPLICIT = 0 + UTF8 = 1 + UTF16 = 2 + SJIS = 3 + HTML = 6 + XML = 7 + UUID = 8 + ISRC = 9 + MI3P = 10 + GIF = 12 + JPEG = 13 + PNG = 14 + URL = 15 + DURATION = 16 + DATETIME = 17 + GENRES = 18 + SIGNED_INT_BE = 21 + UNSIGNED_INT_BE = 22 + FLOAT_BE_32 = 23 + FLOAT_BE_64 = 23 + RIAA_PA = 24 + UPC = 25 + BMP = 27 + SIGNED_INT = 65 + SIGNED_INT_BE_16 = 66 + SIGNED_INT_BE_32 = 67 + SIGNED_INT_BE_64 = 74 + UNSIGNED_INT = 75 + UNSIGNED_INT_BE_16 = 76 + UNSIGNED_INT_BE_32 = 77 + UNSIGNED_INT_BE_64 = 78 + UNDEFINED = 255 + + +class MP4CoverFormat(_BaseIntEnum): + UNKNOWN = 0 + GIF = 12 + JPEG = 13 + PNG = 14 + BMP = 27 + + +MP4SamplingFrequencies = [ + 96000, + 88200, + 64000, + 48000, + 44100, + 32000, + 24000, + 22050, + 16000, + 12000, + 11025, + 8000, + 7350 +] diff --git a/tests/audio/mp4-aac.m4a b/tests/audio/mp4-aac.m4a new file mode 100644 index 0000000000000000000000000000000000000000..194c23041903ea049bbb45f688464693735116ae GIT binary patch literal 6667 zcmeHL&rcIU6rL>>5JehN2x6m4>A{%Nlu~|pvZW8a7rJ4$TXv@ncre6h zqKW^7iAjy|FYw^On-`-eO+0(_;K7Rtes8*z^5bT_%}clMoA13h^X8k`9?BStt@EAg zVm7Zb7Lb$S?kQ)C@w#uV|PP}49zWJkuiL&0N% z5N7+tJ#7<_2YgeksT{0HtHEg9HWTELb%M-RR<4BzD7Ir*8i48;C-3X_2T?FmHf&EI zeEDvzvX29=j5=0j9fOGPG$#rI38Tr!49}!E8MSz}=Gkh1fB1aO2HxJ5vXf1izTp$-*xlWY!=HkW<^~w+WUPAzPc#fB)F0`DQ=lsw@;MhJz%=bQ z6v445mVZb9hGOE`WpFwXFsh~eWZ=W_R|1^yx$Xj!fsg)VY~VLzgL_!6i?Q>4DBr6Y z9NP(faZK{U4(W=s@U&``q}aF6E6esdMk@Cfso_pdqyUmPJ`1M+@;OkZ;x$hnyzM2{{? zb?r5GTb)jgr&H6ZbW+Xd3u8A5>Rd&44bw}ir8~1L>ehG$RX1yzZSfuT>J+AqH!2$5 zvlANR7PqOvcVjU4m}6?<_UW#ZBDKtvRjO4?I?*U5pu#N1(`qouIbimDdFgs@&u|Z5@4|d>1)koZNH<;bOTYH$E}pK=--Ci@ z`sq=kP1izqyqm{fpEgX Xp93STBUg^eZ>fM(Kq?>=XjkAb@x`^l literal 0 HcmV?d00001 diff --git a/tests/audio/mp4-ac3.m4a b/tests/audio/mp4-ac3.m4a new file mode 100644 index 0000000000000000000000000000000000000000..97d465f334220341d2fd0c9122ea2d2a29a416d3 GIT binary patch literal 121911 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 J{_5Jw&EIbGw~qh- literal 0 HcmV?d00001