mirror of
https://github.com/evilhero/mylar
synced 2024-12-23 00:02:38 +00:00
21eee17344
torrents will now properly hide torrent information, IMP: Specified daemon port for deluge as an on-screen tip for more detail, IMP: Added 100,200,ALL as viewable watchlist views, FIX: When viewing pullist and annual integration enabled, if annual was present would incorrectly link to invalid annual series instead of the actual series itself, IMP: Added more detail error messages to metatagging errors and better handling of stranded files during cleanup, IMP: Improved some handling for weekly pull-list one-off's and refactored the nzb/oneoff post-processing into a seperate function for future callables, Moved all the main url locations for public torrent sites to the init module so that it can be cascaded down for use in other modules instead as a global, IMP: Added a 'deep_search_32p' variable in the config.ini for specific usage with 32p, where if there is more than one result will dig deeper into each result to try and figure out if there are series matches, as opposed to the default where it will only use ref32p table if available or just the first hit in a multiple series search results and ignore the remainder, FIX:Fixed some unknown characters appearing in the pullist due to unicode-related conversion problems, FIX: fixed some special cases of file parsing errors due to Volume label being named different than expected, FIX: Added a 3s pause between experimental searches to try and not hit their frequency limitation, IMP: Weekly Pullist One-off's will now show status of Snatched/Downloaded as required, FIX: Fixed some deluge parameter problems when using auto-snatch torrent script/option, IMP: Changed the downlocation in the auto-snatch option to an env variable instead of being passed to avoid unicode-related problems, FIX: Fixed some magnet-related issues for torrents when using a watchdir + TPSE, FIX: Added more verbose error message for rtorrent connection issues, FIX: Could not connect to rtorrent client if no username/password were provided, IMP: Set the db updater to run every 5 minutes on the watchlist, automatically refreshing the oldest updated series each time that is more than 5 hours old (force db update from the activity/job schedulers page will run the db updater against the entire watchlist in sequence), IMP: Attempt to handle long paths in windows (ie. > 256c) by prepending the unicode windows api character to the import a directory path (windows only), IMP: When manual metatagging a series, will update the series after all the metatagging has been completed as opposed to after each issue, IMP: Will now display available inkdrops on Config/Search Providers tab when using 32P (future will utilize/indicate inkdrop threshold when downloading)
143 lines
4.4 KiB
Python
143 lines
4.4 KiB
Python
from abc import ABCMeta, abstractmethod
|
|
import logging
|
|
|
|
import six
|
|
|
|
|
|
class JobLookupError(KeyError):
|
|
"""Raised when the job store cannot find a job for update or removal."""
|
|
|
|
def __init__(self, job_id):
|
|
super(JobLookupError, self).__init__(u'No job by the id of %s was found' % job_id)
|
|
|
|
|
|
class ConflictingIdError(KeyError):
|
|
"""Raised when the uniqueness of job IDs is being violated."""
|
|
|
|
def __init__(self, job_id):
|
|
super(ConflictingIdError, self).__init__(
|
|
u'Job identifier (%s) conflicts with an existing job' % job_id)
|
|
|
|
|
|
class TransientJobError(ValueError):
|
|
"""
|
|
Raised when an attempt to add transient (with no func_ref) job to a persistent job store is
|
|
detected.
|
|
"""
|
|
|
|
def __init__(self, job_id):
|
|
super(TransientJobError, self).__init__(
|
|
u'Job (%s) cannot be added to this job store because a reference to the callable '
|
|
u'could not be determined.' % job_id)
|
|
|
|
|
|
class BaseJobStore(six.with_metaclass(ABCMeta)):
|
|
"""Abstract base class that defines the interface that every job store must implement."""
|
|
|
|
_scheduler = None
|
|
_alias = None
|
|
_logger = logging.getLogger('apscheduler.jobstores')
|
|
|
|
def start(self, scheduler, alias):
|
|
"""
|
|
Called by the scheduler when the scheduler is being started or when the job store is being
|
|
added to an already running scheduler.
|
|
|
|
:param apscheduler.schedulers.base.BaseScheduler scheduler: the scheduler that is starting
|
|
this job store
|
|
:param str|unicode alias: alias of this job store as it was assigned to the scheduler
|
|
"""
|
|
|
|
self._scheduler = scheduler
|
|
self._alias = alias
|
|
self._logger = logging.getLogger('apscheduler.jobstores.%s' % alias)
|
|
|
|
def shutdown(self):
|
|
"""Frees any resources still bound to this job store."""
|
|
|
|
def _fix_paused_jobs_sorting(self, jobs):
|
|
for i, job in enumerate(jobs):
|
|
if job.next_run_time is not None:
|
|
if i > 0:
|
|
paused_jobs = jobs[:i]
|
|
del jobs[:i]
|
|
jobs.extend(paused_jobs)
|
|
break
|
|
|
|
@abstractmethod
|
|
def lookup_job(self, job_id):
|
|
"""
|
|
Returns a specific job, or ``None`` if it isn't found..
|
|
|
|
The job store is responsible for setting the ``scheduler`` and ``jobstore`` attributes of
|
|
the returned job to point to the scheduler and itself, respectively.
|
|
|
|
:param str|unicode job_id: identifier of the job
|
|
:rtype: Job
|
|
"""
|
|
|
|
@abstractmethod
|
|
def get_due_jobs(self, now):
|
|
"""
|
|
Returns the list of jobs that have ``next_run_time`` earlier or equal to ``now``.
|
|
The returned jobs must be sorted by next run time (ascending).
|
|
|
|
:param datetime.datetime now: the current (timezone aware) datetime
|
|
:rtype: list[Job]
|
|
"""
|
|
|
|
@abstractmethod
|
|
def get_next_run_time(self):
|
|
"""
|
|
Returns the earliest run time of all the jobs stored in this job store, or ``None`` if
|
|
there are no active jobs.
|
|
|
|
:rtype: datetime.datetime
|
|
"""
|
|
|
|
@abstractmethod
|
|
def get_all_jobs(self):
|
|
"""
|
|
Returns a list of all jobs in this job store.
|
|
The returned jobs should be sorted by next run time (ascending).
|
|
Paused jobs (next_run_time == None) should be sorted last.
|
|
|
|
The job store is responsible for setting the ``scheduler`` and ``jobstore`` attributes of
|
|
the returned jobs to point to the scheduler and itself, respectively.
|
|
|
|
:rtype: list[Job]
|
|
"""
|
|
|
|
@abstractmethod
|
|
def add_job(self, job):
|
|
"""
|
|
Adds the given job to this store.
|
|
|
|
:param Job job: the job to add
|
|
:raises ConflictingIdError: if there is another job in this store with the same ID
|
|
"""
|
|
|
|
@abstractmethod
|
|
def update_job(self, job):
|
|
"""
|
|
Replaces the job in the store with the given newer version.
|
|
|
|
:param Job job: the job to update
|
|
:raises JobLookupError: if the job does not exist
|
|
"""
|
|
|
|
@abstractmethod
|
|
def remove_job(self, job_id):
|
|
"""
|
|
Removes the given job from this store.
|
|
|
|
:param str|unicode job_id: identifier of the job
|
|
:raises JobLookupError: if the job does not exist
|
|
"""
|
|
|
|
@abstractmethod
|
|
def remove_all_jobs(self):
|
|
"""Removes all jobs from this store."""
|
|
|
|
def __repr__(self):
|
|
return '<%s>' % self.__class__.__name__
|