tag_interface.py 5.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157
  1. # -*- coding: utf-8 -*-
  2. import mutagen.id3
  3. import mutagen.mp4
  4. import re
  5. class _mutagen:
  6. @property
  7. def track_path(self):
  8. return self._mutagen_file.filename
  9. def save(self):
  10. self._mutagen_file.save()
  11. class ID3(_mutagen):
  12. # http://id3.org/id3v2.4.0-frames#4.1.
  13. _UFID_OWNER_ID = 'symuid'
  14. def __init__(self, mutagen_file):
  15. assert mutagen_file.tags, mutagen_file
  16. assert isinstance(mutagen_file.tags, mutagen.id3.ID3), \
  17. mutagen_file.tags
  18. self._mutagen_file = mutagen_file
  19. def _get_single_text(self, tag_label):
  20. tag = self._mutagen_file.tags.get(tag_label, None)
  21. if tag is None:
  22. # {}.get('a') == None
  23. return None
  24. elif len(tag.text) == 1:
  25. return tag.text[0]
  26. else:
  27. raise ValueError(tag)
  28. def get_free_ints(self, tag_label_prefix):
  29. for t in self._mutagen_file.tags.getall('TXXX:' + tag_label_prefix):
  30. assert len(t.text) == 1, t
  31. yield (t.desc, int(t.text[0]))
  32. def get_free_int(self, tag_label):
  33. text = self._get_single_text('TXXX:' + tag_label)
  34. return int(text) if text else None
  35. def set_free_int(self, tag_label, data):
  36. # mutagen.id3._specs.EncodedTextSpec.write encodes 'desc' and 'text'
  37. tag = mutagen.id3.TXXX(
  38. encoding=mutagen.id3.Encoding.LATIN1,
  39. desc=tag_label,
  40. text=[str(data)],
  41. )
  42. # TODO overwrite instead of add() ?
  43. self._mutagen_file.tags.add(tag)
  44. return tag
  45. def get_comment(self):
  46. return self._get_single_text('COMM::eng')
  47. def get_track_uuid(self):
  48. for ufid in self._mutagen_file.tags.getall('UFID'):
  49. if ufid.owner == self._UFID_OWNER_ID:
  50. return ufid.data
  51. return None
  52. def set_track_uuid(self, uuid):
  53. # mutagen.id3._specs.EncodedTextSpec.write encodes 'owner'
  54. tag = mutagen.id3.UFID(owner=self._UFID_OWNER_ID, data=uuid)
  55. self._mutagen_file.tags.add(tag)
  56. return tag
  57. class MP4(_mutagen):
  58. _UUID_TAG_KEY = 'symuid:uuid'
  59. def __init__(self, mutagen_file):
  60. assert mutagen_file.tags, mutagen_file
  61. assert isinstance(mutagen_file.tags, mutagen.mp4.MP4Tags), \
  62. mutagen_file.tags
  63. self._mutagen_file = mutagen_file
  64. def _get_single(self, tag_label):
  65. tag = self._mutagen_file.tags.get(tag_label, None)
  66. if tag is None:
  67. # {}.get('a') == None
  68. return None
  69. elif len(tag) == 1:
  70. return tag[0]
  71. else:
  72. raise ValueError(tag)
  73. @staticmethod
  74. def _freeform_to_int(freeform):
  75. # "a signed big-endian integer with length one of { 1,2,3,4,8 } bytes"
  76. assert freeform.dataformat == mutagen.mp4.AtomDataType.INTEGER, freeform
  77. return int.from_bytes(freeform, byteorder='big', signed=True)
  78. def get_free_ints(self, tag_label_prefix):
  79. label_pattern = re.compile(r'^----:{}(:|$)'.format(
  80. re.escape(tag_label_prefix),
  81. ))
  82. for label, values in self._mutagen_file.tags.items():
  83. # TODO overwrite instead of add() ?
  84. if label_pattern.match(label):
  85. assert len(values) == 1, (label, values)
  86. value = MP4._freeform_to_int(values[0])
  87. yield (re.sub(r'^----:', '', label), value)
  88. def _get_free(self, tag_label):
  89. # freeform keys start with '----'
  90. # http://mutagen.readthedocs.io/en/latest/api/mp4.html
  91. return self._get_single('----:' + tag_label)
  92. def get_free_int(self, tag_label):
  93. t = self._get_free(tag_label)
  94. return None if t is None else MP4._freeform_to_int(t)
  95. def _get_free_uuid(self, tag_label):
  96. tag = self._get_free(tag_label)
  97. assert tag is None or tag.dataformat == mutagen.mp4.AtomDataType.UUID, tag.dataformat
  98. return tag
  99. def _set_free(self, tag_label, dataformat, data):
  100. assert isinstance(data, bytes)
  101. tag = mutagen.mp4.MP4FreeForm(dataformat=dataformat, data=data)
  102. self._mutagen_file.tags['----:' + tag_label] = [tag]
  103. return tag
  104. def set_free_int(self, tag_label, data):
  105. assert isinstance(data, int)
  106. return self._set_free(
  107. tag_label=tag_label,
  108. # "a signed big-endian integer with length one of { 1,2,3,4,8 } bytes"
  109. dataformat=mutagen.mp4.AtomDataType.INTEGER,
  110. # TODO set byte length properly
  111. data=data.to_bytes(1, byteorder='big', signed=True),
  112. )
  113. def _set_free_uuid(self, tag_label, data):
  114. return self._set_free(
  115. tag_label=tag_label,
  116. # https://mutagen.readthedocs.io/en/latest/api/mp4.html#mutagen.mp4.AtomDataType.UUID
  117. dataformat=mutagen.mp4.AtomDataType.UUID,
  118. data=data,
  119. )
  120. def get_comment(self):
  121. return self._get_single('\xa9cmt')
  122. def get_track_uuid(self):
  123. return self._get_free_uuid(self._UUID_TAG_KEY)
  124. def set_track_uuid(self, uuid):
  125. return self._set_free_uuid(self._UUID_TAG_KEY, uuid)