youtubedl_wrapper.py 4.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122
  1. #
  2. # This file is part of stov, written by Helmut Pozimski 2012-2021.
  3. #
  4. # stov is free software: you can redistribute it and/or modify
  5. # it under the terms of the GNU General Public License as published by
  6. # the Free Software Foundation, version 2 of the License.
  7. #
  8. # stov is distributed in the hope that it will be useful,
  9. # but WITHOUT ANY WARRANTY; without even the implied warranty of
  10. # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  11. # GNU General Public License for more details.
  12. #
  13. # You should have received a copy of the GNU General Public License
  14. # along with stov. If not, see <http://www.gnu.org/licenses/>.
  15. # -*- coding: utf8 -*-
  16. """ Provides a wrapper around youtube-dl"""
  17. import logging
  18. import subprocess
  19. import sys
  20. from lib_stov import configuration
  21. from lib_stov import stov_exceptions
  22. LOGGER = logging.getLogger("stov")
  23. def get_ids(url, reverse=False):
  24. """
  25. Retrieves the IDs
  26. :param url: URL to pass to youtube-dl
  27. :type url: str
  28. :param reverse: look up a playlist in reverse order to get \
  29. the recent videos first
  30. :type reverse: bool
  31. :return: video IDs
  32. :rtype: list
  33. """
  34. conf = configuration.Conf.get_instance()
  35. videos_list = []
  36. if conf.outputlevel == "verbose":
  37. stderr = sys.stderr
  38. else:
  39. stderr = open("/dev/null", "w")
  40. LOGGER.debug(_("Executing command: %s %s %s %s %s %s"),
  41. conf.values["youtube-dl"], "--max-downloads",
  42. conf.values["maxvideos"], "-i", "--get-id", url)
  43. try:
  44. if reverse:
  45. video_ids = subprocess.check_output(
  46. [conf.values["youtube-dl"], "--max-downloads",
  47. conf.values["maxvideos"], "-i", "--playlist-reverse",
  48. "--get-id", url], stderr=stderr)
  49. else:
  50. video_ids = subprocess.check_output(
  51. [conf.values["youtube-dl"], "--max-downloads",
  52. conf.values["maxvideos"],
  53. "-i", "--get-id", url], stderr=stderr)
  54. except subprocess.CalledProcessError as error_message:
  55. video_ids = error_message.output
  56. video_ids = video_ids.decode(sys.stdout.encoding).strip()
  57. for video in video_ids.split("\n"):
  58. if video:
  59. videos_list.append(video)
  60. LOGGER.debug("generated list: %s", videos_list)
  61. return videos_list
  62. def get_title(url):
  63. """
  64. Retrieves the title of a specified video
  65. :param url: URL to pass to youtube-dl
  66. :type url: str
  67. """
  68. conf = configuration.Conf.get_instance()
  69. if conf.outputlevel == "verbose":
  70. stderr = sys.stderr
  71. else:
  72. stderr = open("/dev/null", "w")
  73. LOGGER.debug(_("Executing command: %s %s %s"),
  74. conf.values["youtube-dl"], "--get-title", url)
  75. video_title = subprocess.check_output([
  76. conf.values["youtube-dl"], "--get-title", url], stderr=stderr)
  77. video_title = video_title.decode(sys.stdout.encoding)
  78. return video_title
  79. def download_video(url):
  80. """
  81. Downloads a video from a specified url using youtube-dl.
  82. :param url: URL to pass to youtube-dl
  83. :type url: str
  84. """
  85. conf = configuration.Conf.get_instance()
  86. youtube_dl_format = "%s[height=%s]+%s/%s+best" % \
  87. (conf.get_value("video_codec"),
  88. conf.get_value("video_height"),
  89. conf.get_value("audio_quality"),
  90. conf.get_value("video_codec"))
  91. LOGGER.debug(_("Executing command: %s -f %s %s"),
  92. conf.values["youtube-dl"], youtube_dl_format, url)
  93. stderr = open("/dev/null", "w")
  94. stdout = sys.stdout
  95. if conf.outputlevel == "verbose":
  96. stderr = sys.stderr
  97. elif conf.outputlevel == "quiet":
  98. stdout = open("/dev/null", "w")
  99. stderr = open("/dev/null", "w")
  100. try:
  101. subprocess.check_call([conf.values["youtube-dl"], "-f %s"
  102. % youtube_dl_format,
  103. "-o", "%(title)s-%(id)s.%(ext)s", url],
  104. stderr=stderr,
  105. stdout=stdout)
  106. except subprocess.CalledProcessError as error:
  107. LOGGER.debug(_("Error while calling youtube-dl: %s"), error.output)
  108. raise stov_exceptions.YoutubeDlCallFailed()