diff --git a/README.md b/README.md index 8e17804..ff142d1 100644 --- a/README.md +++ b/README.md @@ -1,6 +1,6 @@ # whyliam.workflows.youdao -## 有道翻译 workflow v2.0.0 +## 有道翻译 workflow v2.0.1 默认快捷键 `yd`,查看翻译结果。 @@ -20,7 +20,7 @@ ### 下载 -[点击下载](https://github.com/liszd/whyliam.workflows.youdao/releases/download/2.0.0/whyliam.workflows.youdao.alfredworkflow) +[点击下载](https://github.com/liszd/whyliam.workflows.youdao/releases/download/2.0.1/whyliam.workflows.youdao.alfredworkflow) ### 安装 diff --git a/info.plist b/info.plist index b9e1832..474d756 100755 --- a/info.plist +++ b/info.plist @@ -324,7 +324,7 @@ readme - 有道翻译 Workflow v2.0.0 + 有道翻译 Workflow v2.0.1 默认快捷键 yd, 查看翻译结果。 diff --git a/version b/version index 359a5b9..10bf840 100644 --- a/version +++ b/version @@ -1 +1 @@ -2.0.0 \ No newline at end of file +2.0.1 \ No newline at end of file diff --git a/workflow/__init__.py b/workflow/__init__.py index 632f1f5..3069e51 100755 --- a/workflow/__init__.py +++ b/workflow/__init__.py @@ -14,7 +14,7 @@ # Workflow objects from .workflow import Workflow, manager -from .workflow3 import Workflow3 +from .workflow3 import Variables, Workflow3 # Exceptions from .workflow import PasswordNotFound, KeychainError @@ -67,6 +67,7 @@ __copyright__ = 'Copyright 2014 Dean Jackson' __all__ = [ + 'Variables', 'Workflow', 'Workflow3', 'manager', diff --git a/workflow/background.py b/workflow/background.py index cf883ed..7bda3f5 100755 --- a/workflow/background.py +++ b/workflow/background.py @@ -108,33 +108,31 @@ def _background(stdin='/dev/null', stdout='/dev/null', :type stderr: filepath """ + def _fork_and_exit_parent(errmsg): + try: + pid = os.fork() + if pid > 0: + os._exit(0) + except OSError as err: + wf().logger.critical('%s: (%d) %s', errmsg, err.errno, + err.strerror) + raise err + # Do first fork. - try: - pid = os.fork() - if pid > 0: - sys.exit(0) # Exit first parent. - except OSError as e: - wf().logger.critical("fork #1 failed: ({0:d}) {1}".format( - e.errno, e.strerror)) - sys.exit(1) + _fork_and_exit_parent('fork #1 failed') + # Decouple from parent environment. os.chdir(wf().workflowdir) - os.umask(0) os.setsid() + # Do second fork. - try: - pid = os.fork() - if pid > 0: - sys.exit(0) # Exit second parent. - except OSError as e: - wf().logger.critical("fork #2 failed: ({0:d}) {1}".format( - e.errno, e.strerror)) - sys.exit(1) + _fork_and_exit_parent('fork #2 failed') + # Now I am a daemon! # Redirect standard file descriptors. - si = file(stdin, 'r', 0) - so = file(stdout, 'a+', 0) - se = file(stderr, 'a+', 0) + si = open(stdin, 'r', 0) + so = open(stdout, 'a+', 0) + se = open(stderr, 'a+', 0) if hasattr(sys.stdin, 'fileno'): os.dup2(si.fileno(), sys.stdin.fileno()) if hasattr(sys.stdout, 'fileno'): diff --git a/workflow/update.py b/workflow/update.py index 468d024..bb8e9da 100755 --- a/workflow/update.py +++ b/workflow/update.py @@ -203,8 +203,8 @@ def download_workflow(url): """ filename = url.split("/")[-1] - if (not url.endswith('.alfredworkflow') or - not filename.endswith('.alfredworkflow')): + if (not filename.endswith('.alfredworkflow') and + not filename.endswith('.alfred3workflow')): raise ValueError('Attachment `{0}` not a workflow'.format(filename)) local_path = os.path.join(tempfile.gettempdir(), filename) diff --git a/workflow/version b/workflow/version index 614245e..c8d3893 100755 --- a/workflow/version +++ b/workflow/version @@ -1 +1 @@ -1.24 \ No newline at end of file +1.26 \ No newline at end of file diff --git a/workflow/workflow.py b/workflow/workflow.py index d824b1a..4fd8db4 100755 --- a/workflow/workflow.py +++ b/workflow/workflow.py @@ -21,6 +21,7 @@ from __future__ import print_function, unicode_literals +import atexit import binascii from contextlib import contextmanager import cPickle @@ -804,6 +805,7 @@ def __init__(self, protected_path, timeout=0, delay=0.05): self.timeout = timeout self.delay = delay self._locked = False + atexit.register(self.release) @property def locked(self): @@ -817,11 +819,14 @@ def acquire(self, blocking=True): ``False``. Otherwise, check every `self.delay` seconds until it acquires - lock or exceeds `self.timeout` and raises an exception. + lock or exceeds `self.timeout` and raises an `~AcquisitionError`. """ start = time.time() while True: + + self._validate_lockfile() + try: fd = os.open(self.lockfile, os.O_CREAT | os.O_EXCL | os.O_RDWR) with os.fdopen(fd, 'w') as fd: @@ -830,6 +835,7 @@ def acquire(self, blocking=True): except OSError as err: if err.errno != errno.EEXIST: # pragma: no cover raise + if self.timeout and (time.time() - start) >= self.timeout: raise AcquisitionError('Lock acquisition timed out.') if not blocking: @@ -839,10 +845,36 @@ def acquire(self, blocking=True): self._locked = True return True + def _validate_lockfile(self): + """Check existence and validity of lockfile. + + If the lockfile exists, but contains an invalid PID + or the PID of a non-existant process, it is removed. + + """ + try: + with open(self.lockfile) as fp: + s = fp.read() + except Exception: + return + + try: + pid = int(s) + except ValueError: + return self.release() + + from background import _process_exists + if not _process_exists(pid): + self.release() + def release(self): """Release the lock by deleting `self.lockfile`.""" self._locked = False - os.unlink(self.lockfile) + try: + os.unlink(self.lockfile) + except (OSError, IOError) as err: # pragma: no cover + if err.errno != 2: + raise err def __enter__(self): """Acquire lock.""" @@ -1942,26 +1974,30 @@ def filter(self, query, items, key=lambda x: x, ascending=False, By default, :meth:`filter` uses all of the following flags (i.e. :const:`MATCH_ALL`). The tests are always run in the given order: - 1. :const:`MATCH_STARTSWITH` : Item search key startswith - ``query``(case-insensitive). - 2. :const:`MATCH_CAPITALS` : The list of capital letters in item - search key starts with ``query`` (``query`` may be - lower-case). E.g., ``of`` would match ``OmniFocus``, - ``gc`` would match ``Google Chrome``. - 3. :const:`MATCH_ATOM` : Search key is split into "atoms" on - non-word characters (.,-,' etc.). Matches if ``query`` is - one of these atoms (case-insensitive). - 4. :const:`MATCH_INITIALS_STARTSWITH` : Initials are the first - characters of the above-described "atoms" (case-insensitive). - 5. :const:`MATCH_INITIALS_CONTAIN` : ``query`` is a substring of - the above-described initials. - 6. :const:`MATCH_INITIALS` : Combination of (4) and (5). - 7. :const:`MATCH_SUBSTRING` : Match if ``query`` is a substring - of item search key (case-insensitive). - 8. :const:`MATCH_ALLCHARS` : Matches if all characters in - ``query`` appear in item search key in the same order + 1. :const:`MATCH_STARTSWITH` + Item search key starts with ``query`` (case-insensitive). + 2. :const:`MATCH_CAPITALS` + The list of capital letters in item search key starts with + ``query`` (``query`` may be lower-case). E.g., ``of`` + would match ``OmniFocus``, ``gc`` would match ``Google Chrome``. + 3. :const:`MATCH_ATOM` + Search key is split into "atoms" on non-word characters + (.,-,' etc.). Matches if ``query`` is one of these atoms (case-insensitive). - 9. :const:`MATCH_ALL` : Combination of all the above. + 4. :const:`MATCH_INITIALS_STARTSWITH` + Initials are the first characters of the above-described + "atoms" (case-insensitive). + 5. :const:`MATCH_INITIALS_CONTAIN` + ``query`` is a substring of the above-described initials. + 6. :const:`MATCH_INITIALS` + Combination of (4) and (5). + 7. :const:`MATCH_SUBSTRING` + ``query`` is a substring of item search key (case-insensitive). + 8. :const:`MATCH_ALLCHARS` + All characters in ``query`` appear in item search key in + the same order (case-insensitive). + 9. :const:`MATCH_ALL` + Combination of all the above. :const:`MATCH_ALLCHARS` is considerably slower than the other @@ -2400,7 +2436,11 @@ def update_available(self): :returns: ``True`` if an update is available, else ``False`` """ - update_data = self.cached_data('__workflow_update_status', max_age=0) + # Create a new workflow object to ensure standard serialiser + # is used (update.py is called without the user's settings) + update_data = Workflow().cached_data('__workflow_update_status', + max_age=0) + self.logger.debug('update_data : {0}'.format(update_data)) if not update_data or not update_data.get('available'): diff --git a/workflow/workflow3.py b/workflow/workflow3.py index fcfb65a..4625a50 100755 --- a/workflow/workflow3.py +++ b/workflow/workflow3.py @@ -33,6 +33,79 @@ from .workflow import Workflow +class Variables(dict): + """Workflow variables for Run Script actions. + + .. versionadded: 1.26 + + This class allows you to set workflow variables from + Run Script actions. + + It is a subclass of :class:`dict`. + + >>> v = Variables(username='deanishe', password='hunter2') + >>> v.arg = u'output value' + >>> print(v) + + Attributes: + arg (unicode): Output value (``{query}``). + config (dict): Configuration for downstream workflow element. + + """ + + def __init__(self, arg=None, **variables): + """Create a new `Variables` object. + + Args: + arg (unicode, optional): Main output/``{query}``. + **variables: Workflow variables to set. + + """ + self.arg = arg + self.config = {} + super(Variables, self).__init__(**variables) + + @property + def obj(self): + """Return ``alfredworkflow`` `dict`.""" + o = {} + if self: + d2 = {} + for k, v in self.items(): + d2[k] = v + o['variables'] = d2 + + if self.config: + o['config'] = self.config + + if self.arg is not None: + o['arg'] = self.arg + + return {'alfredworkflow': o} + + def __unicode__(self): + """Convert to ``alfredworkflow`` JSON object. + + Returns: + unicode: ``alfredworkflow`` JSON object + """ + if not self and not self.config: + if self.arg: + return self.arg + else: + return u'' + + return json.dumps(self.obj) + + def __str__(self): + """Convert to ``alfredworkflow`` JSON object. + + Returns: + str: UTF-8 encoded ``alfredworkflow`` JSON object + """ + return unicode(self).encode('utf-8') + + class Modifier(object): """Modify ``Item3`` values for when specified modifier keys are pressed. @@ -213,7 +286,7 @@ def obj(self): Returns: dict: Data suitable for Alfred 3 feedback. """ - # Basic values + # Required values o = {'title': self.title, 'subtitle': self.subtitle, 'valid': self.valid} @@ -221,8 +294,13 @@ def obj(self): icon = {} # Optional values - if self.arg is not None: - o['arg'] = self.arg + + # arg & variables + v = Variables(self.arg, **self.variables) + v.config = self.config + arg = unicode(v) + if arg: + o['arg'] = arg if self.autocomplete is not None: o['autocomplete'] = self.autocomplete @@ -245,11 +323,6 @@ def obj(self): if icon: o['icon'] = icon - # Variables and config - js = self._vars_and_config() - if js: - o['arg'] = js - # Modifiers mods = self._modifiers() if mods: @@ -287,27 +360,6 @@ def _text(self): return text - def _vars_and_config(self): - """Build `arg` including workflow variables and configuration. - - Returns: - str: JSON string value for `arg` (or `None`) - """ - if self.variables or self.config: - d = {} - if self.variables: - d['variables'] = self.variables - - if self.config: - d['config'] = self.config - - if self.arg is not None: - d['arg'] = self.arg - - return json.dumps({'alfredworkflow': d}) - - return None - def _modifiers(self): """Build `mods` dictionary for JSON feedback. @@ -342,6 +394,7 @@ def __init__(self, **kwargs): Workflow.__init__(self, **kwargs) self.variables = {} self._rerun = 0 + self._session_id = None @property def _default_cachedir(self): @@ -373,6 +426,28 @@ def rerun(self, seconds): """ self._rerun = seconds + @property + def session_id(self): + """A unique session ID every time the user uses the workflow. + + .. versionadded:: 1.25 + + The session ID persists while the user is using this workflow. + It expires when the user runs a different workflow or closes + Alfred. + + """ + if not self._session_id: + sid = os.getenv('_WF_SESSION_ID') + if not sid: + from uuid import uuid4 + sid = uuid4().hex + self.setvar('_WF_SESSION_ID', sid) + + self._session_id = sid + + return self._session_id + def setvar(self, name, value): """Set a "global" workflow variable. @@ -421,6 +496,70 @@ def add_item(self, title, subtitle='', arg=None, autocomplete=None, self._items.append(item) return item + def _mk_session_name(self, name): + """New cache name/key based on session ID.""" + return '_wfsess-{0}-{1}'.format(self.session_id, name) + + def cache_data(self, name, data, session=False): + """Cache API with session-scoped expiry. + + .. versionadded:: 1.25 + + Args: + name (str): Cache key + data (object): Data to cache + session (bool, optional): Whether to scope the cache + to the current session. + + ``name`` and ``data`` are as for the + :meth:`~workflow.workflow.Workflow.cache_data` on + :class:`~workflow.workflow.Workflow`. + + If ``session`` is ``True``, the ``name`` variable is prefixed + with :attr:`session_id`. + + """ + if session: + name = self._mk_session_name(name) + + return super(Workflow3, self).cache_data(name, data) + + def cached_data(self, name, data_func=None, max_age=60, session=False): + """Cache API with session-scoped expiry. + + .. versionadded:: 1.25 + + Args: + name (str): Cache key + data_func (callable): Callable that returns fresh data. It + is called if the cache has expired or doesn't exist. + max_age (int): Maximum allowable age of cache in seconds. + session (bool, optional): Whether to scope the cache + to the current session. + + ``name``, ``data_func`` and ``max_age`` are as for the + :meth:`~workflow.workflow.Workflow.cached_data` on + :class:`~workflow.workflow.Workflow`. + + If ``session`` is ``True``, the ``name`` variable is prefixed + with :attr:`session_id`. + + """ + if session: + name = self._mk_session_name(name) + + return super(Workflow3, self).cached_data(name, data_func, max_age) + + def clear_session_cache(self): + """Remove *all* session data from the cache. + + .. versionadded:: 1.25 + """ + def _is_session_file(filename): + return filename.startswith('_wfsess-') + + self.clear_cache(_is_session_file) + @property def obj(self): """Feedback formatted for JSON serialization.