diff --git a/.github/workflows/addon-check.yml b/.github/workflows/addon-check.yml index c6229a3..c3eafd9 100644 --- a/.github/workflows/addon-check.yml +++ b/.github/workflows/addon-check.yml @@ -1,39 +1,27 @@ name: Kodi on: - - pull_request - - push + # Run action when pushed to master, or for commits in a pull request. + push: + branches: + - master + pull_request: + branches: + - master jobs: - tests: + kodi-addon-checker: name: Addon checker runs-on: ubuntu-latest strategy: fail-fast: false matrix: - kodi-branch: [leia, matrix] + kodi-version: [ leia, matrix ] steps: - - uses: actions/checkout@v2 - with: - path: ${{ github.repository }} - - name: Set up Python 3.8 - uses: actions/setup-python@v1 - with: - python-version: 3.8 - - name: Install dependencies - run: | - sudo apt-get install xmlstarlet - python -m pip install --upgrade pip - # FIXME: Requires changes from xbmc/addon-check#217 - #pip install kodi-addon-checker - pip install git+git://github.com/xbmc/addon-check.git@master - - name: Remove unwanted files - run: awk '/export-ignore/ { print $1 }' .gitattributes | xargs rm -rf -- - working-directory: ${{ github.repository }} - - name: Rewrite addon.xml for Matrix - run: | - xmlstarlet ed -L -u '/addon/requires/import[@addon="xbmc.python"]/@version' -v "3.0.0" addon.xml - version=$(xmlstarlet sel -t -v 'string(/addon/@version)' addon.xml) - xmlstarlet ed -L -u '/addon/@version' -v "${version}+matrix.99" addon.xml - working-directory: ${{ github.repository }} - if: matrix.kodi-branch == 'matrix' + - name: Check out ${{ github.sha }} from repository ${{ github.repository }} + uses: actions/checkout@v2 + - name: Run kodi-addon-checker - run: kodi-addon-checker --branch=${{ matrix.kodi-branch }} ${{ github.repository }}/ \ No newline at end of file + uses: xbmc/action-kodi-addon-checker@v1.2 + with: + kodi-version: ${{ matrix.kodi-version }} + rewrite-for-matrix: true + addon-id: ${{ github.event.repository.name }} \ No newline at end of file diff --git a/resources/lib/modules/iptvsimple.py b/resources/lib/modules/iptvsimple.py index 90ca9ac..e88d256 100644 --- a/resources/lib/modules/iptvsimple.py +++ b/resources/lib/modules/iptvsimple.py @@ -51,6 +51,7 @@ def check(cls): try: addon = kodiutils.get_addon(IPTV_SIMPLE_ID) except Exception as exc: # pylint: disable=broad-except + _LOGGER.exception(exc) return True # It might be restarting # Validate IPTV Simple configuration diff --git a/resources/lib/modules/sources/__init__.py b/resources/lib/modules/sources/__init__.py index ab36d4e..2cb3e58 100644 --- a/resources/lib/modules/sources/__init__.py +++ b/resources/lib/modules/sources/__init__.py @@ -18,6 +18,9 @@ class Sources: """Helper class for Source updating""" + def __init__(self): + """ Initialise object """ + @classmethod def refresh(cls, show_progress=False): """Update channels and EPG data""" @@ -88,9 +91,12 @@ def refresh(cls, show_progress=False): progress.close() -class Source: +class Source(object): # pylint: disable=useless-object-inheritance """ Base class for a Source """ + def __init__(self): + """ Initialise object """ + @staticmethod def detect_sources(): """ Detect available sources. """ @@ -117,18 +123,17 @@ def _load_url(self, url): """ Load the specified URL. """ response = requests.get(url) response.raise_for_status() - data = response.content if url.lower().endswith('.gz'): - return self._decompress_gz(data) + return self._decompress_gz(response.content) if url.lower().endswith('.bz2'): - return self._decompress_bz2(data) + return self._decompress_bz2(response.content) - return data + return response.text def _load_file(self, filename): """ Load the specified file. """ - with open(filename, 'r') as fdesc: + with open(filename, 'rb') as fdesc: data = fdesc.read() if filename.lower().endswith('.gz'): @@ -136,12 +141,12 @@ def _load_file(self, filename): if filename.lower().endswith('.bz2'): return self._decompress_bz2(data) - return data + return data.decode() @staticmethod def _extract_m3u(data): """ Extract the m3u content """ - return data.replace('#EXTM3U\n', '') + return data.replace('#EXTM3U', '').strip() @staticmethod def _extract_xmltv(data): @@ -151,8 +156,14 @@ def _extract_xmltv(data): @staticmethod def _decompress_gz(data): """ Decompress gzip data. """ - from gzip import decompress - return decompress(data).decode() + try: # Python 3 + from gzip import decompress + return decompress(data).decode() + except ImportError: # Python 2 + from gzip import GzipFile + from StringIO import StringIO + with GzipFile(fileobj=StringIO(data)) as fdesc: + return fdesc.read().decode() @staticmethod def _decompress_bz2(data): diff --git a/resources/lib/modules/sources/addon.py b/resources/lib/modules/sources/addon.py index cbe4344..7441419 100644 --- a/resources/lib/modules/sources/addon.py +++ b/resources/lib/modules/sources/addon.py @@ -35,6 +35,8 @@ class AddonSource(Source): EPG_VERSION = 1 def __init__(self, addon_id, enabled=False, channels_uri=None, epg_uri=None): + """ Initialise object """ + super(AddonSource, self).__init__() self.addon_id = addon_id self.enabled = enabled self.channels_uri = channels_uri diff --git a/resources/lib/modules/sources/custom.py b/resources/lib/modules/sources/custom.py index cf1992d..cc669a9 100644 --- a/resources/lib/modules/sources/custom.py +++ b/resources/lib/modules/sources/custom.py @@ -23,6 +23,8 @@ class CustomSource(Source): TYPE_FILE = 2 def __init__(self, uuid, name, enabled, playlist_uri=None, playlist_type=TYPE_NONE, epg_uri=None, epg_type=TYPE_NONE): + """ Initialise object """ + super(CustomSource, self).__init__() self.uuid = uuid self.name = name self.enabled = enabled @@ -106,8 +108,12 @@ def get_epg(self): def save(self): """ Save this source. """ + output_path = kodiutils.addon_profile() try: - with open(os.path.join(kodiutils.addon_profile(), CustomSource.SOURCES_FILE), 'r') as fdesc: + if not os.path.exists(output_path): + os.mkdir(output_path) + + with open(os.path.join(output_path, CustomSource.SOURCES_FILE), 'r') as fdesc: sources = json.loads(fdesc.read()) except (IOError, TypeError, ValueError): sources = {} @@ -115,13 +121,14 @@ def save(self): # Update the element with my uuid sources[self.uuid] = self.__dict__ - with open(os.path.join(kodiutils.addon_profile(), CustomSource.SOURCES_FILE), 'w') as fdesc: + with open(os.path.join(output_path, CustomSource.SOURCES_FILE), 'w') as fdesc: json.dump(sources, fdesc) def delete(self): """ Delete this source. """ + output_path = kodiutils.addon_profile() try: - with open(os.path.join(kodiutils.addon_profile(), CustomSource.SOURCES_FILE), 'r') as fdesc: + with open(os.path.join(output_path, CustomSource.SOURCES_FILE), 'r') as fdesc: sources = json.loads(fdesc.read()) except (IOError, TypeError, ValueError): sources = {} @@ -129,5 +136,5 @@ def delete(self): # Remove the element with my uuid sources.pop(self.uuid) - with open(os.path.join(kodiutils.addon_profile(), CustomSource.SOURCES_FILE), 'w') as fdesc: + with open(os.path.join(output_path, CustomSource.SOURCES_FILE), 'w') as fdesc: json.dump(sources, fdesc) diff --git a/tests/data/custom_playlist.m3u.gz b/tests/data/custom_playlist.m3u.gz new file mode 100644 index 0000000..a1f2f6f Binary files /dev/null and b/tests/data/custom_playlist.m3u.gz differ diff --git a/tests/home/addons/plugin.video.example.raw/plugin.py b/tests/home/addons/plugin.video.example.raw/plugin.py index bf40589..4c220a9 100644 --- a/tests/home/addons/plugin.video.example.raw/plugin.py +++ b/tests/home/addons/plugin.video.example.raw/plugin.py @@ -44,16 +44,16 @@ def send(self): @via_socket def send_channels(): # pylint: disable=no-method-argument """Return JSON-STREAMS formatted information to IPTV Manager""" - with open(os.path.dirname(__file__) + '/resources/raw_playlist.m3u', 'r') as fdesc: + with open(os.path.dirname(__file__) + '/resources/raw_playlist.m3u', 'rb') as fdesc: channels = fdesc.read() - return channels + return channels.decode() @via_socket def send_epg(): # pylint: disable=no-method-argument """Return JSON-EPG formatted information to IPTV Manager""" - with open(os.path.dirname(__file__) + '/resources/raw_epg.xml', 'r') as fdesc: + with open(os.path.dirname(__file__) + '/resources/raw_epg.xml', 'rb') as fdesc: epg = fdesc.read() - return epg + return epg.decode() if __name__ == "__main__": diff --git a/tests/test_sources.py b/tests/test_sources.py index fd084e1..0059c8d 100644 --- a/tests/test_sources.py +++ b/tests/test_sources.py @@ -53,14 +53,15 @@ def test_create(self): self.assertNotIn(key, [source.uuid for source in sources]) def test_fetch_none(self): - source = CustomSource(uuid=str(uuid4()), - name='Test Source', - enabled=True, - playlist_uri=None, - playlist_type=CustomSource.TYPE_NONE, - epg_uri=None, - epg_type=CustomSource.TYPE_NONE, - ) + source = CustomSource( + uuid=str(uuid4()), + name='Test Source', + enabled=True, + playlist_uri=None, + playlist_type=CustomSource.TYPE_NONE, + epg_uri=None, + epg_type=CustomSource.TYPE_NONE, + ) channels = source.get_channels() self.assertEqual(channels, '') @@ -69,20 +70,30 @@ def test_fetch_none(self): self.assertEqual(epg, '') def test_fetch_file(self): - source = CustomSource(uuid=str(uuid4()), - name='Test Source', - enabled=True, - playlist_uri=os.path.realpath('tests/data/custom_playlist.m3u'), - playlist_type=CustomSource.TYPE_FILE, - epg_uri=os.path.realpath('tests/data/custom_epg.xml'), - epg_type=CustomSource.TYPE_FILE, - ) + source = CustomSource( + uuid=str(uuid4()), + name='Test Source', + enabled=True, + playlist_uri=os.path.realpath('tests/data/custom_playlist.m3u'), + playlist_type=CustomSource.TYPE_FILE, + epg_uri=os.path.realpath('tests/data/custom_epg.xml'), + epg_type=CustomSource.TYPE_FILE, + ) + expected_channels = Source._extract_m3u(open('tests/data/custom_playlist.m3u', 'r').read()) + expected_epg = Source._extract_xmltv(open('tests/data/custom_epg.xml', 'r').read()) + + # Test channels + channels = source.get_channels() + self.assertEqual(channels.replace('\r\n', '\n'), expected_channels) + # Test channels (gzip) + source.playlist_uri = os.path.realpath('tests/data/custom_playlist.m3u.gz') channels = source.get_channels() - self.assertEqual(channels, Source._extract_m3u(open('tests/data/custom_playlist.m3u').read())) + self.assertEqual(channels.replace('\r\n', '\n'), expected_channels) + # Test EPG epg = source.get_epg() - self.assertEqual(epg, Source._extract_xmltv(open('tests/data/custom_epg.xml').read())) + self.assertEqual(epg.replace('\r\n', '\n'), expected_epg) def test_fetch_url(self): @@ -99,14 +110,26 @@ def raise_for_status(self): def content(self): return self.data + @property + def text(self): + return self.data.decode() + if args[0].endswith('m3u'): - data = open('tests/data/custom_playlist.m3u', 'r').read() + data = open('tests/data/custom_playlist.m3u', 'rb').read() return MockResponse(data, 200) if args[0].endswith('m3u.gz'): - from gzip import compress data = open('tests/data/custom_playlist.m3u', 'rb').read() - return MockResponse(compress(data), 200) + try: # Python 3 + from gzip import compress + return MockResponse(compress(data), 200) + except ImportError: # Python 2 + from gzip import GzipFile + from StringIO import StringIO + buf = StringIO() + with GzipFile(fileobj=buf, mode='wb') as f: + f.write(data) + return MockResponse(buf.getvalue(), 200) if args[0].endswith('m3u.bz2'): from bz2 import compress @@ -114,40 +137,41 @@ def content(self): return MockResponse(compress(data), 200) if args[0].endswith('xml'): - data = open('tests/data/custom_epg.xml', 'r').read() + data = open('tests/data/custom_epg.xml', 'rb').read() return MockResponse(data, 200) return MockResponse(None, 404) - with patch('requests.get', side_effect=mocked_requests_get): - source = CustomSource(uuid=str(uuid4()), - name='Test Source', - enabled=True, - playlist_uri='https://example.com/playlist.m3u', - playlist_type=CustomSource.TYPE_URL, - epg_uri='https://example.com/xmltv.xml', - epg_type=CustomSource.TYPE_URL, - ) + source = CustomSource( + uuid=str(uuid4()), + name='Test Source', + enabled=True, + playlist_uri='https://example.com/playlist.m3u', + playlist_type=CustomSource.TYPE_URL, + epg_uri='https://example.com/xmltv.xml', + epg_type=CustomSource.TYPE_URL, + ) + expected_channels = Source._extract_m3u(open('tests/data/custom_playlist.m3u', 'r').read()) + expected_epg = Source._extract_xmltv(open('tests/data/custom_epg.xml', 'r').read()) + with patch('requests.get', side_effect=mocked_requests_get): # Test channels channels = source.get_channels() - self.assertEqual(channels, Source._extract_m3u(open('tests/data/custom_playlist.m3u').read())) - - # Test EPG - epg = source.get_epg() - self.assertEqual(epg, Source._extract_xmltv(open('tests/data/custom_epg.xml').read())) + self.assertEqual(channels.replace('\r\n', '\n'), expected_channels) # Test channels (gzip) source.playlist_uri = 'https://example.com/playlist.m3u.gz' channels = source.get_channels() - self.assertEqual(channels, Source._extract_m3u(open('tests/data/custom_playlist.m3u').read())) + self.assertEqual(channels.replace('\r\n', '\n'), expected_channels) # Test channels (bzip2) source.playlist_uri = 'https://example.com/playlist.m3u.bz2' channels = source.get_channels() - self.assertEqual(channels, Source._extract_m3u(open('tests/data/custom_playlist.m3u').read())) - + self.assertEqual(channels.replace('\r\n', '\n'), expected_channels) + # Test EPG + epg = source.get_epg() + self.assertEqual(epg.replace('\r\n', '\n'), expected_epg) if __name__ == '__main__':