symuid-import-itunes 3.2 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697
  1. #!/usr/bin/env python3
  2. # -*- coding: utf-8 -*-
  3. import datetime as dt
  4. import mutagen
  5. import mutagen.id3
  6. import mutagen.mp4
  7. import os
  8. import re
  9. import symuid.library.itunes
  10. import sys
  11. def generate_play_count_tag_label(player, library_id, reg_dt):
  12. return 'symuid:pcnt:{}:{}:{}'.format(player, library_id, int(reg_dt.timestamp()))
  13. def set_play_count_tag(track_path, player, library_id, reg_dt, play_count):
  14. assert isinstance(reg_dt, dt.datetime), reg_dt
  15. tag_label = generate_play_count_tag_label(
  16. player=player,
  17. library_id=library_id,
  18. reg_dt=reg_dt,
  19. )
  20. track = mutagen.File(filename=track_path)
  21. if isinstance(track.tags, mutagen.id3.ID3):
  22. tag_label_id3 = 'TXXX:' + tag_label
  23. if not tag_label_id3 in track.tags:
  24. # mutagen.id3._specs.EncodedTextSpec.write encodes
  25. # 'desc' and 'text'
  26. tag = mutagen.id3.TXXX(
  27. encoding=mutagen.id3.Encoding.LATIN1,
  28. desc=tag_label,
  29. text=[str(play_count)],
  30. )
  31. track.tags.add(tag)
  32. track.save()
  33. print('{!r}: set ID3 tag {!r}'.format(track_path, tag))
  34. elif isinstance(track.tags, mutagen.mp4.MP4Tags):
  35. tag_label_mp4 = '----:' + tag_label
  36. if not tag_label_mp4 in track.tags:
  37. track.tags[tag_label_mp4] = tag = mutagen.mp4.MP4FreeForm(
  38. # "a signed big-endian integer with length one of { 1,2,3,4,8 } bytes"
  39. # TODO set byte length properly
  40. data=play_count.to_bytes(1, byteorder='big'),
  41. dataformat=mutagen.mp4.AtomDataType.INTEGER,
  42. )
  43. track.save()
  44. print('{!r}: set MP4 tag {!r}'.format(track_path, tag))
  45. else:
  46. raise Exception(track_path)
  47. def symuid_import_itunes(xml_library_path, path_regex_sub):
  48. lib = symuid.library.itunes.XmlLibrary(xml_library_path)
  49. for track in lib.tracks:
  50. # TODO create tag if last_play_dt is None
  51. if track.last_play_dt and track.local:
  52. track_path = track.local_path
  53. for pattern, repl in path_regex_sub:
  54. track_path = re.sub(pattern, repl, track_path)
  55. if not os.path.exists(track_path):
  56. sys.stderr.write('{!r}: not found\n'.format(track_path))
  57. else:
  58. # TODO dt=dt.datetime.now()
  59. set_play_count_tag(
  60. track_path=track_path,
  61. player='itunes',
  62. library_id=lib.id,
  63. reg_dt=track.last_play_dt,
  64. play_count=track.play_count,
  65. )
  66. def _init_argparser():
  67. import argparse
  68. argparser = argparse.ArgumentParser(description=None)
  69. argparser.add_argument('xml_library_path')
  70. argparser.add_argument(
  71. '--path-regex-sub',
  72. nargs=2,
  73. action='append',
  74. metavar=('regex', 'replacement'),
  75. default=[],
  76. help='(default: %(default)r)',
  77. )
  78. return argparser
  79. def main(argv):
  80. argparser = _init_argparser()
  81. args = argparser.parse_args(argv[1:])
  82. symuid_import_itunes(**vars(args))
  83. return 0
  84. if __name__ == "__main__":
  85. sys.exit(main(sys.argv))