mirror of
https://github.com/SickGear/SickGear.git
synced 2025-01-22 09:33:37 +00:00
980e05cc99
Backported 400 revisions from rev 1de4961-8897c5b (2018-2014). Move core/benchmark, core/cmd_line, core/memory, core/profiler and core/timeout to core/optional/* Remove metadata/qt* PORT: Version 2.0a3 (inline with 3.0a3 @ f80c7d5). Basic Support for XMP Packets. tga: improvements to adhere more closely to the spec. pdf: slightly improved parsing. rar: fix TypeError on unknown block types. Add MacRoman win32 codepage. tiff/exif: support SubIFDs and tiled images. Add method to export metadata in dictionary. mpeg_video: don't attempt to parse Stream past length. mpeg_video: parse ESCR correctly, add SCR value. Change centralise CustomFragments. field: don't set parser class if class is None, to enable autodetect. field: add value/display for CustomFragment. parser: inline warning to enable tracebacks in debug mode. Fix empty bytestrings in makePrintable. Fix contentSize in jpeg.py to account for image_data blocks. Fix the ELF parser. Enhance the AR archive parser. elf parser: fix wrong wrong fields order in parsing little endian section flags. elf parser: add s390 as a machine type. Flesh out mp4 parser. PORT: Version 2.0a1 (inline with 3.0a1). Major refactoring and PEP8. Fix ResourceWarning warnings on files. Add a close() method and support for the context manager protocol ("with obj: ...") to parsers, input and output streams. metadata: get comment from ZIP. Support for InputIOStream.read(0). Fix sizeGe when size is None. Remove unused new_seekable_field_set file. Remove parser Mapsforge .map. Remove parser Parallel Realities Starfighter .pak files. sevenzip: fix for newer archives. java: update access flags and modifiers for Java 1.7 and update description text for most recent Java. Support ustar prefix field in tar archives. Remove file_system* parsers. Remove misc parsers 3d0, 3ds, gnome_keyring, msoffice*, mstask, ole*, word*. Remove program parsers macho, nds, prc. Support non-8bit Character subclasses. Python parser supports Python 3.7. Enhance mpeg_ts parser to support MTS/M2TS. Support for creation date in tiff. Change don't hardcode errno constant. PORT: 1.9.1 Internal Only: The following are legacy reference to upstream commit messages. Relevant changes up to b0a115f8. Use integer division. Replace HACHOIR_ERRORS with Exception. Fix metadata.Data: make it sortable. Import fixes from e7de492. PORT: Version 2.0a1 (inline with 3.0a1 @ e9f8fad). Replace hachoir.core.field with hachoir.field Replace hachoir.core.stream with hachoir.stream Remove the compatibility module for PY1.5 to PY2.5. metadata: support TIFF picture. metadata: fix string normalization. metadata: fix datetime regex Fix hachoir bug #57. FileFromInputStream: fix comparison between None and an int. InputIOStream: open the file in binary mode.
135 lines
4 KiB
Python
135 lines
4 KiB
Python
"""
|
|
Audio Interchange File Format (AIFF) parser.
|
|
|
|
Author: Victor Stinner
|
|
Creation: 27 december 2006
|
|
"""
|
|
|
|
from hachoir.parser import Parser
|
|
from hachoir.field import (FieldSet,
|
|
UInt16, UInt32, Float80, TimestampMac32,
|
|
RawBytes, NullBytes,
|
|
String, Enum, PascalString32)
|
|
from hachoir.core.endian import BIG_ENDIAN
|
|
from hachoir.core.text_handler import filesizeHandler
|
|
from hachoir.core.tools import alignValue
|
|
from hachoir.parser.audio.id3 import ID3v2
|
|
|
|
CODEC_NAME = {
|
|
'ACE2': u"ACE 2-to-1",
|
|
'ACE8': u"ACE 8-to-3",
|
|
'MAC3': u"MAC 3-to-1",
|
|
'MAC6': u"MAC 6-to-1",
|
|
'NONE': u"None",
|
|
'sowt': u"Little-endian, no compression",
|
|
}
|
|
|
|
|
|
class Comment(FieldSet):
|
|
def createFields(self):
|
|
yield TimestampMac32(self, "timestamp")
|
|
yield PascalString32(self, "text")
|
|
|
|
|
|
def parseText(self):
|
|
yield String(self, "text", self["size"].value)
|
|
|
|
|
|
def parseID3(self):
|
|
yield ID3v2(self, "id3v2", size=self["size"].value * 8)
|
|
|
|
|
|
def parseComment(self):
|
|
yield UInt16(self, "nb_comment")
|
|
for index in xrange(self["nb_comment"].value):
|
|
yield Comment(self, "comment[]")
|
|
|
|
|
|
def parseCommon(self):
|
|
yield UInt16(self, "nb_channel")
|
|
yield UInt32(self, "nb_sample")
|
|
yield UInt16(self, "sample_size")
|
|
yield Float80(self, "sample_rate")
|
|
yield Enum(String(self, "codec", 4, strip="\0", charset="ASCII"), CODEC_NAME)
|
|
|
|
|
|
def parseVersion(self):
|
|
yield TimestampMac32(self, "timestamp")
|
|
|
|
|
|
def parseSound(self):
|
|
yield UInt32(self, "offset")
|
|
yield UInt32(self, "block_size")
|
|
size = (self.size - self.current_size) // 8
|
|
if size:
|
|
yield RawBytes(self, "data", size)
|
|
|
|
|
|
class Chunk(FieldSet):
|
|
TAG_INFO = {
|
|
'COMM': ('common', "Common chunk", parseCommon),
|
|
'COMT': ('comment', "Comment", parseComment),
|
|
'NAME': ('name', "Name", parseText),
|
|
'AUTH': ('author', "Author", parseText),
|
|
'FVER': ('version', "Version", parseVersion),
|
|
'SSND': ('sound', "Sound data", parseSound),
|
|
'ID3 ': ('id3', "ID3", parseID3),
|
|
}
|
|
|
|
def __init__(self, *args):
|
|
FieldSet.__init__(self, *args)
|
|
self._size = (8 + alignValue(self["size"].value, 2)) * 8
|
|
tag = self["type"].value
|
|
if tag in self.TAG_INFO:
|
|
self._name, self._description, self._parser = self.TAG_INFO[tag]
|
|
else:
|
|
self._parser = None
|
|
|
|
def createFields(self):
|
|
yield String(self, "type", 4, "Signature (FORM)", charset="ASCII")
|
|
yield filesizeHandler(UInt32(self, "size"))
|
|
size = self["size"].value
|
|
if size:
|
|
if self._parser:
|
|
for field in self._parser(self):
|
|
yield field
|
|
if size % 2:
|
|
yield NullBytes(self, "padding", 1)
|
|
else:
|
|
yield RawBytes(self, "data", size)
|
|
|
|
|
|
class AiffFile(Parser):
|
|
PARSER_TAGS = {
|
|
"id": "aiff",
|
|
"category": "audio",
|
|
"file_ext": ("aif", "aiff", "aifc"),
|
|
"mime": (u"audio/x-aiff",),
|
|
"magic_regex": (("FORM.{4}AIF[CF]", 0),),
|
|
"min_size": 12 * 8,
|
|
"description": "Audio Interchange File Format (AIFF)"
|
|
}
|
|
endian = BIG_ENDIAN
|
|
|
|
def validate(self):
|
|
if self.stream.readBytes(0, 4) != "FORM":
|
|
return "Invalid signature"
|
|
if self.stream.readBytes(8 * 8, 4) not in ("AIFF", "AIFC"):
|
|
return "Invalid type"
|
|
return True
|
|
|
|
def createFields(self):
|
|
yield String(self, "signature", 4, "Signature (FORM)", charset="ASCII")
|
|
yield filesizeHandler(UInt32(self, "filesize"))
|
|
yield String(self, "type", 4, "Form type (AIFF or AIFC)", charset="ASCII")
|
|
while not self.eof:
|
|
yield Chunk(self, "chunk[]")
|
|
|
|
def createDescription(self):
|
|
if self["type"].value == "AIFC":
|
|
return "Audio Interchange File Format Compressed (AIFC)"
|
|
else:
|
|
return "Audio Interchange File Format (AIFF)"
|
|
|
|
def createContentSize(self):
|
|
return self["filesize"].value * 8
|