Source code for luigi.contrib.target

# -*- coding: utf-8 -*-
#
# Copyright 2012-2015 Spotify AB
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
#

import logging
from types import MethodType

import luigi.target

logger = logging.getLogger('luigi-interface')


[docs] class CascadingClient: """ A FilesystemClient that will cascade failing function calls through a list of clients. Which clients are used are specified at time of construction. """ # This constant member is supposed to include all methods, feel free to add # methods here. If you want full control of which methods that should be # created, pass the kwarg to the constructor. ALL_METHOD_NAMES = ['exists', 'rename', 'remove', 'chmod', 'chown', 'count', 'copy', 'get', 'put', 'mkdir', 'list', 'listdir', 'getmerge', 'isdir', 'rename_dont_move', 'touchz', ] def __init__(self, clients, method_names=None): self.clients = clients if method_names is None: method_names = self.ALL_METHOD_NAMES for method_name in method_names: new_method = self._make_method(method_name) real_method = MethodType(new_method, self) setattr(self, method_name, real_method) @classmethod def _make_method(cls, method_name): def new_method(self, *args, **kwargs): return self._chained_call(method_name, *args, **kwargs) return new_method def _chained_call(self, method_name, *args, **kwargs): for i in range(len(self.clients)): client = self.clients[i] try: result = getattr(client, method_name)(*args, **kwargs) return result except luigi.target.FileSystemException: # For exceptions that are semantical, we must throw along raise except BaseException: is_last_iteration = (i + 1) >= len(self.clients) if is_last_iteration: raise else: logger.warning('The %s failed to %s, using fallback class %s', client.__class__.__name__, method_name, self.clients[i + 1].__class__.__name__)