# coding=UTF-8 # Author: Dennis Lutter # URL: http://code.google.com/p/sickbeard/ # # This file is part of SickGear. # # SickGear is free software: you can redistribute it and/or modify # it under the terms of the GNU General Public License as published by # the Free Software Foundation, either version 3 of the License, or # (at your option) any later version. # # SickGear is distributed in the hope that it will be useful, # but WITHOUT ANY WARRANTY; without even the implied warranty of # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the # GNU General Public License for more details. # # You should have received a copy of the GNU General Public License # along with SickGear. If not, see . from __future__ import print_function from __future__ import with_statement import glob import os.path import shutil import sqlite3 import sys import unittest sys.path.insert(1, os.path.abspath('..')) sys.path.insert(1, os.path.abspath('../lib')) import sickbeard from sickbeard import db, providers, tvcache from sickbeard.databases import cache_db, failed_db, mainDB # ================= # test globals # ================= TESTDIR = os.path.abspath('.') TESTDBNAME = 'sickbeard.db' TESTCACHEDBNAME = 'cache.db' TESTFAILEDDBNAME = 'failed.db' SHOWNAME = u'show name' SEASON = 4 EPISODE = 2 FILENAME = u'show name - s0' + str(SEASON) + 'e0' + str(EPISODE) + '.mkv' FILEDIR = os.path.join(TESTDIR, SHOWNAME) FILEPATH = os.path.join(FILEDIR, FILENAME) SHOWDIR = os.path.join(TESTDIR, SHOWNAME + ' final') # sickbeard.logger.sb_log_instance = sickbeard.logger.SBRotatingLogHandler( # os.path.join(TESTDIR, 'sickgear.log'), sickbeard.logger.NUM_LOGS, sickbeard.logger.LOG_SIZE) sickbeard.logger.SBRotatingLogHandler.log_file = os.path.join(os.path.join(TESTDIR, 'Logs'), 'test_sickgear.log') # ================= # prepare env functions # ================= def create_test_log_folder(): if not os.path.isdir(sickbeard.LOG_DIR): os.mkdir(sickbeard.LOG_DIR) def create_test_cache_folder(): if not os.path.isdir(sickbeard.CACHE_DIR): os.mkdir(sickbeard.CACHE_DIR) # call env functions at appropriate time during sickbeard var setup # ================= # sickbeard globals # ================= sickbeard.SYS_ENCODING = 'UTF-8' sickbeard.showList = [] sickbeard.QUALITY_DEFAULT = 4 # hdtv sickbeard.FLATTEN_FOLDERS_DEFAULT = 0 sickbeard.NAMING_PATTERN = '' sickbeard.NAMING_ABD_PATTERN = '' sickbeard.NAMING_SPORTS_PATTERN = '' sickbeard.NAMING_MULTI_EP = 1 sickbeard.PROVIDER_ORDER = ['sick_beard_index'] sickbeard.newznabProviderList = providers.getNewznabProviderList( "'Sick Beard Index|http://lolo.sickbeard.com/|0|5030,5040,5060|0|eponly|0'") sickbeard.providerList = providers.makeProviderList() sickbeard.PROG_DIR = os.path.abspath('..') # sickbeard.DATA_DIR = os.path.join(sickbeard.PROG_DIR, 'tests') sickbeard.DATA_DIR = os.path.join(TESTDIR, 'data') sickbeard.LOG_DIR = os.path.join(TESTDIR, 'Logs') create_test_log_folder() sickbeard.logger.sb_log_instance.init_logging(False) sickbeard.CACHE_DIR = os.path.join(TESTDIR, 'cache') sickbeard.ZONEINFO_DIR = os.path.join(TESTDIR, 'zoneinfo') create_test_cache_folder() # ================= # dummy functions # ================= def _dummy_save_config(): return True # this overrides the save_config which gets called during a db upgrade # this might be considered a hack mainDB.sickbeard.save_config = _dummy_save_config # the real one tries to contact tvdb just stop it from getting more info on the ep # noinspection PyUnusedLocal def _fake_specify_ep(self, season, episode, show_sql=None): pass sickbeard.tv.TVEpisode.specify_episode = _fake_specify_ep # ================= # test classes # ================= class SickbeardTestDBCase(unittest.TestCase): def setUp(self): sickbeard.showList = [] setup_test_db() setup_test_episode_file() setup_test_show_dir() def tearDown(self): sickbeard.showList = [] teardown_test_db() teardown_test_episode_file() teardown_test_show_dir() class TestDBConnection(db.DBConnection, object): def __init__(self, db_file_name=TESTDBNAME): db_file_name = os.path.join(TESTDIR, db_file_name) super(TestDBConnection, self).__init__(db_file_name) class TestCacheDBConnection(TestDBConnection, object): def __init__(self, provider_name): db.DBConnection.__init__(self, os.path.join(TESTDIR, TESTCACHEDBNAME)) # Create the table if it's not already there try: sql = 'CREATE TABLE ' + provider_name + \ ' (name TEXT, season NUMERIC, episodes TEXT,' \ ' indexerid NUMERIC, url TEXT, time NUMERIC, quality TEXT);' self.connection.execute(sql) self.connection.commit() except sqlite3.OperationalError as e: if 'table %s already exists' % provider_name != str(e): raise # Create the table if it's not already there try: sql = 'CREATE TABLE lastUpdate (provider TEXT, time NUMERIC);' self.connection.execute(sql) self.connection.commit() except sqlite3.OperationalError as e: if 'table lastUpdate already exists' != str(e): raise # this will override the normal db connection sickbeard.db.DBConnection = TestDBConnection sickbeard.tvcache.CacheDBConnection = TestCacheDBConnection # ================= # test functions # ================= def setup_test_db(): """upgrades the db to the latest version """ # upgrading the db db.MigrationCode(db.DBConnection()) # fix up any db problems db.sanityCheckDatabase(db.DBConnection(), mainDB.MainSanityCheck) # and for cachedb too db.upgradeDatabase(db.DBConnection('cache.db'), cache_db.InitialSchema) # and for faileddb too db.upgradeDatabase(db.DBConnection('failed.db'), failed_db.InitialSchema) def teardown_test_db(): """Deletes the test db although this seams not to work on my system it leaves me with an zero kb file """ # uncomment next line so leave the db intact between test and at the end # return False try: sickbeard.db.DBConnection().close() except (BaseException, Exception): pass try: sickbeard.tvcache.CacheDBConnection().close() except (BaseException, Exception): pass for filename in glob.glob(os.path.join(TESTDIR, TESTDBNAME) + '*'): os.remove(filename) for filename in glob.glob(os.path.join(TESTDIR, TESTCACHEDBNAME) + '*'): os.remove(filename) for filename in glob.glob(os.path.join(TESTDIR, TESTFAILEDDBNAME) + '*'): os.remove(filename) def setup_test_episode_file(): if not os.path.exists(FILEDIR): os.makedirs(FILEDIR) try: with open(FILEPATH, 'w') as f: f.write('foo bar') except EnvironmentError: print('Unable to set up test episode') raise def teardown_test_episode_file(): if os.path.exists(FILEDIR): shutil.rmtree(FILEDIR) def setup_test_show_dir(): if not os.path.exists(SHOWDIR): os.makedirs(SHOWDIR) def teardown_test_show_dir(): if os.path.exists(SHOWDIR): shutil.rmtree(SHOWDIR) teardown_test_db() if '__main__' == __name__: print('=========================') print('Dont call this directly') print('=========================') print('you might want to call') dirList = os.listdir(TESTDIR) for fname in dirList: if (0 < fname.find('_test')) and (0 > fname.find('pyc')): print('- ' + fname) print('=========================') print('or just call all_tests.py')