noapi.py 4.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126
  1. #
  2. # This file is part of stov, written by Helmut Pozimski 2012-2014.
  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. from __future__ import unicode_literals
  17. import subprocess
  18. import sys
  19. import lxml.html
  20. if sys.version_info >= (3,):
  21. import urllib.request as urllib2
  22. else:
  23. import urllib2
  24. from lib_stov import youtubeAPI
  25. from lib_stov import stov_exceptions
  26. class Connector(object):
  27. """This class will retrieve all the necessary data from youtube using
  28. youtube-dl, thus bypassing the API.
  29. """
  30. def __init__(self, type, name, conf, search=""):
  31. """Populates the object with all necessary data."""
  32. self._type = type
  33. self._name = name
  34. self._search = search
  35. self._conf = conf
  36. self._title = ""
  37. self._url = ""
  38. self._construct_url()
  39. def _construct_url(self):
  40. if self._type == "channel":
  41. self._url = "https://www.youtube.com/user/%s" \
  42. % urllib2.quote(self._name)
  43. elif self._type == "search":
  44. self._url = "https://www.youtube.com/results?search_query=%s"\
  45. % urllib2.quote(self._search)
  46. elif self._type == "playlist":
  47. self._url = "https://www.youtube.com/playlist?list=%s" \
  48. % urllib2.quote(self._name)
  49. def _fetch_title(self):
  50. """Retrieves the title of the HTML page to use as a title for the
  51. subscription."""
  52. data = urllib2.urlopen(self._url)
  53. parsed_html = lxml.html.parse(data)
  54. data.close()
  55. i = 0
  56. for item in parsed_html.iter("title"):
  57. if i == 0:
  58. self._title = item.text_content().strip().replace("\n", "")
  59. i += 1
  60. if self._search != "" and self._type == "channel":
  61. self._title += _(" search %s") % self._search
  62. def _fetch_videos(self):
  63. """Retrieves all the relevant videos in a subscription."""
  64. videos_list = []
  65. if self._type == "channel" and self._search != "":
  66. try:
  67. video_ids = subprocess.check_output([
  68. self._conf.values["youtube-dl"],
  69. "--max-downloads",
  70. self._conf.values["maxvideos"],
  71. "--match-title",
  72. self._search,
  73. "--get-id",
  74. self._url]).strip()
  75. except subprocess.CalledProcessError as e:
  76. video_ids = e.output.strip()
  77. else:
  78. try:
  79. video_ids = subprocess.check_output([
  80. self._conf.values["youtube-dl"], "--max-downloads",
  81. self._conf.values["maxvideos"], "--get-id",
  82. self._url]).strip()
  83. except subprocess.CalledProcessError as e:
  84. video_ids = e.output.strip()
  85. for video_id in video_ids.split("\n"):
  86. try:
  87. video_title = subprocess.check_output([
  88. self._conf.values["youtube-dl"], "--get-title",
  89. "https://www.youtube.com/watch?v=%s"
  90. % video_id]).strip()
  91. video_description = subprocess.check_output([
  92. self._conf.values["youtube-dl"], "--get-description",
  93. "https://www.youtube.com/watch?v=%s"
  94. % video_id]).strip()
  95. except subprocess.CalledProcessError:
  96. raise stov_exceptions.YoutubeDlCallFailed()
  97. else:
  98. videos_list.append(youtubeAPI.YtVideo(
  99. unicode(video_title, "utf-8"),
  100. unicode(video_description, "utf-8"),
  101. unicode(video_id)))
  102. return videos_list
  103. def ParseAPIData(self):
  104. """This method calls all necessary methods to retrieve the data
  105. and assembles them into a Channel object. The naming of this
  106. method is set according to the method in youtubeAPI to be
  107. compatible.
  108. """
  109. self._fetch_title()
  110. videos = self._fetch_videos()
  111. channel = youtubeAPI.YtChannel()
  112. channel.title = unicode(self._title)
  113. channel.videos = videos
  114. return channel