symuid-sync 3.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106
  1. #!/usr/bin/env python3
  2. # -*- coding: utf-8 -*-
  3. import mutagen
  4. import mutagen.id3
  5. import mutagen.mp4
  6. import os
  7. import re
  8. import subprocess
  9. import sys
  10. import symuid
  11. # freeform keys start with '----'
  12. # http://mutagen.readthedocs.io/en/latest/api/mp4.html
  13. TRACK_UUID_MP4_TAG = '----:symuid:uuid'
  14. PATH_DEFAULT_IGNORE_REGEX = r'\.(itdb|itc2|itl|jpg|midi?|plist|xml|zip)$'
  15. def generate_uuid():
  16. return subprocess.check_output(['uuid', '-v', '4', '-F', 'BIN']).strip()
  17. def get_or_assign_uuid_id3(track):
  18. uuid = track._iface.get_uuid()
  19. if uuid is None:
  20. id3_tags = track._iface._mutagen_file.tags
  21. # mutagen.id3._specs.EncodedTextSpec.write encodes 'owner'
  22. id3_tags.add(mutagen.id3.UFID(
  23. owner=track._iface._UFID_OWNER_ID,
  24. data=generate_uuid(),
  25. ))
  26. id3_tags.save()
  27. id3_tags.load(filename=id3_tags.filename)
  28. uuid = track._iface.get_uuid()
  29. print("{!r}: assigned uuid {!r}".format(id3_tags.filename, uuid))
  30. assert uuid is not None
  31. return uuid
  32. def get_or_assign_uuid_mp4(mp4_file):
  33. if not TRACK_UUID_MP4_TAG in mp4_file:
  34. mp4_file[TRACK_UUID_MP4_TAG] = mutagen.mp4.MP4FreeForm(
  35. data=generate_uuid(),
  36. # https://mutagen.readthedocs.io/en/latest/api/mp4.html#mutagen.mp4.AtomDataType.UUID
  37. dataformat=mutagen.mp4.AtomDataType.UUID,
  38. )
  39. mp4_file.save()
  40. mp4_file.load(filename=mp4_file.filename)
  41. print("{!r}: assigned uuid {!r}".format(
  42. mp4_file.filename, mp4_file[TRACK_UUID_MP4_TAG][0],
  43. ))
  44. return mp4_file[TRACK_UUID_MP4_TAG][0]
  45. def log_path(track_path, msg, stream=sys.stdout):
  46. stream.write("{!r}: {}\n".format(track_path, msg))
  47. def log_path_error(track_path, msg):
  48. log_path(track_path, msg, stream=sys.stderr)
  49. def symuid_sync(path, path_ignore_regex, show_ignored=False):
  50. for track in symuid.Track.walk(
  51. root_path=path,
  52. path_ignore_regex=path_ignore_regex,
  53. ignored_cb=lambda p: show_ignored and log_path(p, 'ignored'),
  54. unsupported_cb=lambda p, e: log_path_error(p, 'unsupported type, skipped'),
  55. ):
  56. if isinstance(track._iface, symuid.tag_interface.MP4):
  57. get_or_assign_uuid_mp4(track._iface._mutagen_file)
  58. elif isinstance(track._iface, symuid.tag_interface.ID3):
  59. get_or_assign_uuid_id3(track)
  60. else:
  61. raise Exception(track)
  62. def _init_argparser():
  63. import argparse
  64. argparser = argparse.ArgumentParser(description=None)
  65. argparser.add_argument('path')
  66. argparser.add_argument(
  67. '--path-ignore-regex',
  68. default=PATH_DEFAULT_IGNORE_REGEX,
  69. nargs=1,
  70. metavar='pattern',
  71. dest='path_ignore_regex',
  72. type=lambda pattern: re.compile(pattern),
  73. help='(default: %(default)s)',
  74. )
  75. argparser.add_argument(
  76. '--show-ignored',
  77. action='store_true',
  78. )
  79. return argparser
  80. def main(argv):
  81. argparser = _init_argparser()
  82. args = argparser.parse_args(argv[1:])
  83. symuid_sync(**vars(args))
  84. return 0
  85. if __name__ == "__main__":
  86. sys.exit(main(sys.argv))