Source code for descarteslabs.compute.function

# Copyright 2018-2023 Descartes Labs.
#
# 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 builtins
import dis
import glob
import gzip
import importlib
import inspect
import io
import itertools
import os
import re
import sys
import time
import uuid
import warnings
import zipfile
from datetime import datetime
from tempfile import NamedTemporaryFile
from typing import (
    Any,
    Callable,
    Dict,
    Iterable,
    List,
    Mapping,
    Optional,
    Set,
    Type,
    Union,
)

import pkg_resources
from strenum import StrEnum

import descarteslabs.exceptions as exceptions

from ..client.deprecation import deprecate
from ..client.services.service import ThirdPartyService
from ..common.client import (
    Attribute,
    DatetimeAttribute,
    Document,
    DocumentState,
    Search,
)
from .compute_client import ComputeClient
from .exceptions import BoundGlobalError
from .job import Job, JobSearch, JobStatus
from .result import Serializable

MAX_FUNCTION_IDS_PER_REQUEST = 128


def batched(iterable, n):
    """Batch an iterable into lists of size n"""
    if n < 1:
        raise ValueError("n must be at least one")
    it = iter(iterable)
    while batch := list(itertools.islice(it, n)):
        yield batch


[docs]class FunctionStatus(StrEnum): "The status of the Function." AWAITING_BUNDLE = "awaiting_bundle" BUILDING = "building" BUILD_FAILED = "build_failed" READY = "ready" STOPPED = "stopped"
class Cpus(float): """Validates CPUs for a Function""" NON_NUMERIC = r"[^0-9.]" def __new__(cls, value): if isinstance(value, str): value = re.sub(cls.NON_NUMERIC, "", value) return super().__new__(cls, value) class Memory(int): """Validates Memory for a Function""" MEMORY_MB = re.compile(r"[\s]*(?:mb|mi)$", flags=re.IGNORECASE) MEMORY_GB = re.compile(r"[\s]*(?:gb|gi)$", flags=re.IGNORECASE) def __new__(cls, memory: Union[str, int, float]) -> None: if isinstance(memory, str): if memory.isnumeric(): pass elif re.search(cls.MEMORY_MB, memory): memory = re.sub(cls.MEMORY_MB, "", memory) elif re.search(cls.MEMORY_GB, memory): memory = re.sub(cls.MEMORY_GB, "", memory) memory = int(float(memory) * 1024) else: raise ValueError(f"Unable to convert memory to megabytes: {memory}") return super().__new__(cls, memory) class JobBulkCreateError: """An error that occurred while submitting a bulk job.""" def __init__( self, function: "Function", args, kwargs, environments, exception: Exception, reference_id: str, ): self.function = function self.args = args self.kwargs = kwargs self.environments = environments self.exception = exception self.reference_id = reference_id def __str__(self): return f"{self.reference_id}: {self.exception}" def __repr__(self): return ( f"JobBulkCreateError(" f"function={self.function}, " f"reference_id={self.reference_id}, " f"exception={repr(self.exception)})" ) class JobBulkCreateResult(List[Job]): """The result of a bulk job submission.""" def __init__(self): super().__init__() self.errors: List[JobBulkCreateError] = [] def append_error(self, error: JobBulkCreateError): """Append an error to the result.""" self.errors.append(error) @property def is_success(self) -> bool: """Returns true if all jobs were successfully submitted.""" return len(self.errors) == 0
[docs]class Function(Document): """The serverless cloud function that you can call directly or submit many jobs to.""" id: str = Attribute( str, filterable=True, readonly=True, sortable=True, doc="The ID of the Function.", ) creation_date: datetime = DatetimeAttribute( filterable=True, readonly=True, sortable=True, doc="""The date the Function was created.""", ) name: str = Attribute( str, filterable=True, sortable=True, doc="The name of the Function.", ) image: str = Attribute( str, filterable=True, mutable=False, doc="The base image used to create the Function.", ) cpus: float = Attribute( Cpus, filterable=True, sortable=True, doc="The number of cpus to request when executing the Function.", ) memory: int = Attribute( Memory, filterable=True, sortable=True, doc="The amount of memory, in megabytes, to request when executing the Function.", ) maximum_concurrency: int = Attribute( int, filterable=True, sortable=True, doc="The maximum number of Jobs that execute at the same time for this Function.", ) namespace: str = Attribute( str, filterable=True, readonly=True, doc="The storage namespace for the Function.", ) owner: str = Attribute( str, filterable=True, readonly=True, doc="The owner of the Function.", ) status: FunctionStatus = Attribute( FunctionStatus, filterable=True, readonly=True, sortable=True, doc="The status of the Function.", ) timeout: int = Attribute( int, filterable=True, sortable=True, doc="The number of seconds Jobs can run before timing out for this Function.", ) retry_count: int = Attribute( int, filterable=True, sortable=True, doc="The total number of retries requested for a Job before it failed.", ) enabled: bool = Attribute( bool, filterable=True, sortable=True, doc="Whether the Function accepts job submissions and reruns.", ) auto_start: bool = Attribute( bool, filterable=True, sortable=True, doc="Whether the Function will be placed into READY status once building is complete.", ) environment: Dict[str, str] = Attribute( dict, doc="The environment variables to set for Jobs run by the Function.", ) modified_date: datetime = DatetimeAttribute( filterable=True, readonly=True, sortable=True, doc="""The date the Function was last modified or processed a job submission.""", ) job_statistics: Dict = Attribute( dict, readonly=True, doc=( "Statistics about the Job statuses for this Function. This attribute will only be " "available if includes='job.statistics' is specified in the request." ), ) _ENTRYPOINT_TEMPLATE = "{source}\n\n\nmain = {function_name}\n" _IMPORT_TEMPLATE = "from {module} import {obj}" _SYS_PACKAGES = ".*(?:site|dist)-packages" _ENTRYPOINT = "__dlentrypoint__.py" _REQUIREMENTS = "requirements.txt" def __init__( self, function: Callable = None, requirements: List[str] = None, include_data: List[str] = None, include_modules: List[str] = None, name: str = None, image: str = None, cpus: Cpus = None, memory: Memory = None, maximum_concurrency: int = None, timeout: int = None, retry_count: int = None, client: ComputeClient = None, **extra, ): # check to see if we need more validation here (type conversions) """ Parameters ---------- function : Callable The function to be called in a Compute Job. requirements : List[str], optional A list of Python dependencies required by this function. include_data : List[str], optional Non-Python data files to include in the compute function. name : str, optional Name of the function, will take name of function if not provided. image : str The location of a docker image to be used for the environment where the function will be executed. cpus : Cpus The number of CPUs requested for a single Job. CPUs can be specified as an integer, float, or string. Supported CPU options include: ``0.25, 0.5, 1, 2, 4, 8, 16, '0.25vCPU', '0.5vCPU', '1vCPU', '2vCPU', '4vCPU', '8vCPU', '16vCPU'``. memory : Memory The maximum memory requirement for a single Job. Memory can be specified as an integer or string. If an integer is provided, it is assumed that the units are megabytes. For instance, ``1024`` is equivalent to one ``1GB`` or ``1024MB`` of memory. Alternatively, memory can be specified as a case-insensitive memory string, such as ``'1GB'``, ``'1Gi'``, ``'1024MB'``, or ``'1024Mi'``, all of which are equivalent. The allowable memory for a `Function` is determined by the number of CPUs requested. Supported memory options per CPUs requested include: ``0.25vCPU: 0.5GB, 1GB, or 2GB``, ``0.5vCPU: 1 - 4GB in 1GB increments``, ``1vCPU: 2 - 8GB in 1GB increments``, ``2vCPU: 4 - 16GB in 1GB increments``, ``4vCPU: 8 - 30GB in 1GB increments``, ``8vCPU: 16 - 60GB in 4GB increments``, ``16vCPU: 32 - 120GB in 8GB increments``. maximum_concurrency : int The maximum number of jobs to run in parallel. timeout : int, optional Maximum runtime for a single job in seconds. Job will be killed if it exceeds this limit. retry_count : int, optional Number of times to retry a job if it fails. client : ComputeClient, optional If set, operations on the function will be performed using the configured client. Otherwise, the default client will be used. Examples -------- Retrieving an existing function and executing it. >>> fn = Function.get(<function-id>) >>> fn() Job <job id>: "pending" Creating a new function. >>> from descarteslabs.compute import Function >>> def test_func(): ... print("Hello :)") >>> fn = Function( ... test_func, ... requirements=[], ... name="my_func", ... image="test_image", ... cpus=1, ... memory=16, ... maximum_concurrency=5, ... timeout=3600, ... retry_count=1, ... ) >>> fn() Job <job id>: "pending" """ self._client = client or ComputeClient.get_default_client() self._function = function self._requirements = requirements self._include_data = include_data self._include_modules = include_modules # if name is not provided and function is a string, use the name of the function # if name is not provided and function is a callable, set the name to __name__ if not name and self._function: if isinstance(self._function, str): name = self._function.split(".")[-1] else: name = self._function.__name__ # When a Function is hydrated from the server, register the namespace # with the client so that it can be used for subsequent calls. if "id" in extra and "namespace" in extra: self._client.set_namespace(extra["id"], extra["namespace"]) super().__init__( name=name, image=image, cpus=cpus, memory=memory, maximum_concurrency=maximum_concurrency, timeout=timeout, retry_count=retry_count, **extra, ) def __call__( self, *args, tags: List[str] = None, environment: Dict[str, str] = None, **kwargs, ): """Execute the function with the given arguments. This call will return a Job object that can be used to monitor the state of the job. Returns ------- Job The Job that was submitted. Parameters ---------- args : Any, optional Positional arguments to pass to the function. tags : List[str], optional A list of tags to apply to the Job. kwargs : Any, optional Keyword arguments to pass to the function. environment : Dict[str, str], optional Environment variables to be set in the environment of the running Job. Will be merged with environment variables set on the Function, with the Job environment variables taking precedence. """ self.save() job = Job( function_id=self.id, args=args, kwargs=kwargs, environment=environment, tags=tags, ) job.save() return job def _sys_paths(self): """Get the system paths.""" if not hasattr(self, "_cached_sys_paths"): # use longest matching path entries. self._cached_sys_paths = sorted( map(os.path.abspath, sys.path), key=len, reverse=True ) return self._cached_sys_paths def _get_globals(self, func) -> Set[str]: """Get the globals for a function.""" instructions: List[dis.Instruction] = dis.get_instructions(func) globals = set() for i in instructions: if inspect.iscode(i.argval): # The function can have nested functions with their own globals # so we need to recursively disassemble those as well globals.update(self._get_globals(i.argval)) elif i.opname == "LOAD_GLOBAL" and not hasattr(builtins, i.argval): globals.add(i.argval) return globals def _find_object(self, name): """Search for an object as specified by a fully qualified name. The fully qualified name must refer to an object that can be resolved through the module search path. Parameters ---------- name : str Fully qualified name of the object to search for. Must refer to an object that can be resolved through the module search path. Returns ------- object The object specified by the fully qualified name. module_path : list The fully qualified module path. object_path : list The fully qualified object path. """ module_path = [] object_path = [] obj = None parts = name.split(".") for part in parts: error = None if hasattr(obj, part): # Could be any object (module, class, etc.) obj = getattr(obj, part) if inspect.ismodule(obj) and not object_path: module_path.append(part) else: object_path.append(part) else: # If not found, assume it's a module that must be loaded if object_path: error = "'{}' has no attribute '{}'".format(type(obj), part) raise NameError( "Cannot resolve function name '{}': {}".format(name, error) ) else: module_path.append(part) current_module_path = ".".join(module_path) try: obj = importlib.import_module(current_module_path) except Exception as ex: traceback = sys.exc_info()[2] raise NameError( "Cannot resolve function name '{}', error importing module {}: {}".format( name, current_module_path, ex ) ).with_traceback(traceback) # When we're at the end, we should have found a valid object return obj, module_path, object_path def _bundle(self): """Bundle the function and its dependencies into a zip file.""" function = self._function include_modules = self._include_modules requirements = self._requirements if not function: raise ValueError("Function not provided!") data_files = self._data_globs_to_paths() try: with NamedTemporaryFile(delete=False, suffix=".zip", mode="wb") as f: with zipfile.ZipFile( f, mode="w", compression=zipfile.ZIP_DEFLATED ) as bundle: self._write_main_function(function, bundle) self._write_data_files(data_files, bundle) if include_modules: self._write_include_modules(self._include_modules, bundle) if requirements: bundle.writestr( self._REQUIREMENTS, self._bundle_requirements(requirements) ) return f.name except Exception: if os.path.exists(f.name): os.remove(f.name) raise def _write_main_function(self, f, archive): """Write the main function to the archive.""" is_named_function = isinstance(f, str) if is_named_function: f, module_path, function_path = self._find_object(f) if not callable(f): raise ValueError( "Compute main function must be a callable: `{}`".format(f) ) # Simply import the module source = self._IMPORT_TEMPLATE.format( module=".".join(module_path), obj=function_path[0] ) function_name = ".".join(function_path) else: # make sure function_name is set function_name = f.__name__ if not inspect.isfunction(f): raise ValueError( "Compute main function must be user-defined function: `{}`".format( f ) ) # We can't get the code for a given lambda if f.__name__ == "<lambda>": raise ValueError( "Compute main function cannot be a lambda expression: `{}`".format( f ) ) # Furthermore, the given function cannot refer to globals bound_globals = self._get_globals(f) if bound_globals: raise BoundGlobalError( "Illegal reference to one or more global variables in your " "function: {}".format(bound_globals) ) try: source = inspect.getsource(f).strip() except Exception: try: import dill source = dill.source.getsource(f).strip() except ImportError: raise ValueError( "Unable to retrieve the source of interactively defined functions." " To support this install dill: pip install dill" ) entrypoint_source = self._ENTRYPOINT_TEMPLATE.format( source=source, function_name=function_name ) archive.writestr(self._ENTRYPOINT, entrypoint_source) def _write_data_files(self, data_files, archive): """Write the data files to the archive.""" for path, archive_path in data_files: archive.write(path, archive_path) def _find_module_file(self, mod_name): """Search for module file in python path. Raise ImportError if not found.""" try: mod = importlib.import_module(mod_name) mod_file = mod.__file__.replace(".pyc", ".py", 1) return mod_file except ImportError as ie: # Search for possible pyx file mod_basename = "{}.pyx".format(mod_name.replace(".", "/")) for s in sys.path: mod_file_option = os.path.join(s, mod_basename) if os.path.isfile(mod_file_option): # Check that found cython source not in CWD (causes build problems) if os.getcwd() == os.path.dirname(os.path.abspath(mod_file_option)): raise ValueError( "Cannot include cython modules from working directory: `{}`.".format( mod_file_option ) ) else: return mod_file_option # Raise caught ImportError if we still haven't found the module raise ie def _write_include_modules(self, include_modules, archive): """Write the included modules to the archive.""" for mod_name in include_modules: mod_file = self._find_module_file(mod_name) # detect system packages from distribution or virtualenv locations. if re.match(self._SYS_PACKAGES, mod_file) is not None: raise ValueError( "Cannot include system modules: `{}`.".format(mod_file) ) if not os.path.exists(mod_file): raise IOError( "Source code for module is missing, only byte code exists: `{}`.".format( mod_name ) ) sys_path = self._sys_path_prefix(mod_file) self._include_init_files(os.path.dirname(mod_file), archive, sys_path) archive_names = archive.namelist() # this is a package, get all decendants if they exist. if os.path.basename(mod_file) == "__init__.py": for dirpath, dirnames, filenames in os.walk(os.path.dirname(mod_file)): for file_ in [f for f in filenames if f.endswith((".py", ".pyx"))]: path = os.path.join(dirpath, file_) arcname = self._archive_path(path, None, sys_path) if arcname not in archive_names: archive.write(path, arcname=arcname) else: archive.write( mod_file, arcname=self._archive_path(mod_file, None, sys_path) ) def _include_init_files(self, dir_path, archive, sys_path): """Include __init__.py files for all parent directories.""" relative_dir_path = os.path.relpath(dir_path, sys_path) archive_names = archive.namelist() # have we walked this path before? if os.path.join(relative_dir_path, "__init__.py") not in archive_names: partial_path = "" for path_part in relative_dir_path.split(os.sep): partial_path = os.path.join(partial_path, path_part) rel_init_location = os.path.join(partial_path, "__init__.py") abs_init_location = os.path.join(sys_path, rel_init_location) if not os.path.exists(abs_init_location): raise IOError( "Source code for module is missing: `{}`.".format( abs_init_location ) ) if rel_init_location not in archive_names: archive.write(abs_init_location, arcname=rel_init_location) def _bundle_requirements(self, requirements): """Bundle the requirements into the archive.""" if not pkg_resources: warnings.warn( "Your Python does not have a recent version of `setuptools`. " "For a better experience update your environment by running `pip install -U setuptools`." ) if isinstance(requirements, str): return self._requirements_file(requirements) else: return self._requirements_list(requirements) def _requirements_file(self, requirements): """Read the requirements file and validate it.""" if not os.path.isfile(requirements): raise ValueError( "Requirements file at {} not found. Did you mean to specify a single requirement? " "Pass it wrapped in a list.".format(requirements) ) with open(requirements) as f: return self._requirements_list([line.strip() for line in f.readlines()]) def _requirements_list(self, requirements): """Validate the requirements list.""" if pkg_resources: bad_requirements = [] for requirement in requirements: try: pkg_resources.Requirement.parse(requirement) except ValueError: # comment or pip-specific option not understood by pkg_resources if requirement.startswith("#") or requirement.startswith("-"): continue if requirement.startswith("https://"): continue # e.g. torch-2.0.1+cpu which pkg_resource doesn't understand if "+" in requirement: try: pkg_resources.Requirement.parse(requirement.rsplit("+")[0]) continue except ValueError: pass bad_requirements.append(requirement) if bad_requirements: raise ValueError( "Invalid Python requirements: {}".format(",".join(bad_requirements)) ) return "\n".join(requirements) def _sys_path_prefix(self, path): """Get the system path prefix for a given path.""" absolute_path = os.path.abspath(path) for sys_path in self._sys_paths(): if absolute_path.startswith(sys_path): return sys_path else: raise IOError("Location is not on system path: `{}`".format(path)) def _archive_path(self, path, archive_prefix, sys_path): """Get the archive path for a given path.""" if archive_prefix: return os.path.join(archive_prefix, os.path.relpath(path, sys_path)) else: return os.path.relpath(path, sys_path) def _data_globs_to_paths(self) -> List[str]: """Convert data globs to absolute paths.""" data_files = [] # if there are no data files, return empty list if not self._include_data: return data_files for pattern in self._include_data: is_glob = glob.has_magic(pattern) matches = glob.glob(pattern) if not matches: if is_glob: warnings.warn(f"Include data pattern had no matches: {pattern}") else: raise ValueError(f"No data file found for path: {pattern}") for relative_path in matches: path = os.path.abspath(relative_path) if os.path.exists(path): if os.path.isdir(path): relative_path = relative_path.rstrip("/") raise ValueError( "Cannot accept directories as include data." " Use globs instead: {} OR {}".format( f"{relative_path}/*.*", f"{relative_path}/**/*.*" ) ) else: archive_path = self._archive_path( path, None, sys_path=self._sys_path_prefix(path) ) if archive_path == "__main__.py": raise ValueError( f"{pattern} includes a file with the forbidden relative path `__main__.py`" ) data_files.append((path, archive_path)) else: raise ValueError(f"Data file does not exist: {path}") return data_files
[docs] @classmethod def get(cls, id: str, client: ComputeClient = None, **params): """Get Function by id. Parameters ---------- id : str Id of function to get. client: ComputeClient, optional If set, the result will be retrieved using the configured client. Otherwise, the default client will be used. include : List[str], optional List of additional attributes to include in the response. Allowed values are: - "job.statistics": Include statistics about the Job statuses for this Function. Example ------- >>> from descarteslabs.compute import Function >>> fn = Function.get(<func_id>) <Function name="test_name" image=test_image cpus=1 memory=16 maximum_concurrency=5 timeout=3 retries=1 """ client = client or ComputeClient.get_default_client() response = client.session.get(f"/functions/{id}", params=params) return cls(**response.json(), client=client, saved=True)
[docs] @classmethod def list( cls, page_size: int = 100, client: ComputeClient = None, **params ) -> Search["Function"]: """Lists all Functions for a user. If you would like to filter Functions, use :py:meth:`Function.search`. Parameters ---------- page_size : int, default=100 Maximum number of results per page. client: ComputeClient, optional If set, the result will be retrieved using the configured client. Otherwise, the default client will be used. include : List[str], optional List of additional attributes to include in the response. Allowed values are: - "job.statistics": Include statistics about the Job statuses for this Function. Example ------- >>> from descarteslabs.compute import Function >>> fn = Function.list() """ params = {"page_size": page_size, **params} return cls.search(client=client).param(**params)
[docs] @classmethod def search(cls, client: ComputeClient = None) -> Search["Function"]: """Creates a search for Functions. The search is lazy and will be executed when the search is iterated over or :py:meth:`Search.collect` is called. Parameters ---------- client: ComputeClient, optional If set, the result will be retrieved using the configured client. Otherwise, the default client will be used. Example ------- >>> from descarteslabs.compute import Function, FunctionStatus >>> fns: List[Function] = ( ... Function.search() ... .filter(Function.status.in_([ ... FunctionStatus.BUILDING, FunctionStatus.AWAITING_BUNDLE ... ]) ... .collect() ... ) Collection([Function <fn-id1>: building, Function <fn-id2>: awaiting_bundle]) """ client = client or ComputeClient.get_default_client() return Search(Function, client, url="/functions")
[docs] @classmethod def update_credentials(cls, client: ComputeClient = None): """Updates the credentials for the Functions and Jobs run by this user. These credentials are used by other Descarteslabs services. If the user invalidates existing credentials and needs to update them, you should call this method. Notes ----- Credentials are automatically updated when a new Function is created. Parameters ---------- client: ComputeClient, optional If set, the operation will be performed using the configured client. Otherwise, the default client will be used. """ client = client or ComputeClient.get_default_client() client.set_credentials()
@property def jobs(self) -> JobSearch: """Returns a JobSearch for all the Jobs for the Function.""" if self.state != DocumentState.SAVED: raise ValueError( "Cannot search for jobs for a Function that has not been saved" ) return Job.search(client=self._client).filter(Job.function_id == self.id)
[docs] def build_log(self): """Retrieves the build log for the Function.""" if self.state != DocumentState.SAVED: raise ValueError( "Cannot retrieve logs for a Function that has not been saved" ) response = self._client.session.get(f"/functions/{self.id}/log") return gzip.decompress(response.content).decode()
[docs] def delete(self, delete_results: bool = False): """Deletes the Function and all associated Jobs. If any jobs are in a running state, the deletion will fail. Please see the `:meth:~descarteslabs.compute.Function.delete_jobs` method for more information on deleting large numbers of jobs. Parameters ---------- delete_results : bool, default=False If True, deletes the job result blobs as well. """ if self.state == DocumentState.NEW: raise ValueError("Cannot delete a Function that has not been saved") self.delete_jobs(delete_results=delete_results) self._client.session.delete(f"/functions/{self.id}") self._deleted = True
[docs] def disable(self): """Disables the Function so that new jobs cannot be submitted.""" self.enabled = False if self.state != DocumentState.NEW: self.save()
[docs] def enable(self): """Enables the Function so that new jobs may be submitted.""" self.enabled = True if self.state != DocumentState.NEW: self.save()
[docs] def save(self): """Creates the Function if it does not already exist. If the Function already exists, it will be updated on the server if the Function instance was modified. Examples -------- Create a Function without creating jobs: >>> from descarteslabs.compute import Function >>> def test_func(): ... print("Hello :)") >>> fn = Function( ... test_func, ... requirements=[], ... name="my_func", ... image="test_image", ... cpus=1, ... memory=16, ... maximum_concurrency=5, ... timeout=3600, ... retry_count=1, ... ) >>> fn.save() Updating a Function: >>> from descarteslabs.compute import Function >>> fn = Function.get(<func_id>) >>> fn.memory = 4096 # 4 Gi >>> fn.save() """ if self.state == DocumentState.SAVED: # Document already exists on the server without changes locally return if self.state == DocumentState.NEW: self.update_credentials() code_bundle_path = self._bundle() response = self._client.session.post( "/functions", json=self.to_dict(exclude_none=True) ) response_json = response.json() self._load_from_remote(response_json["function"]) # Upload the bundle to s3 s3_client = ThirdPartyService() upload_url = response_json["bundle_upload_url"] code_bundle = io.open(code_bundle_path, "rb") headers = { "content-type": "application/octet-stream", } s3_client.session.put(upload_url, data=code_bundle, headers=headers) # Complete the upload with compute response = self._client.session.post(f"/functions/{self.id}/bundle") self._load_from_remote(response.json()) elif self.state == DocumentState.MODIFIED: response = self._client.session.patch( f"/functions/{self.id}", json=self.to_dict(only_modified=True) ) self._load_from_remote(response.json()) else: raise ValueError( f"Unexpected Function state {self.state}." f'Reload the function from the server: Function.get("{self.id}")' )
[docs] def start(self): """Starts Function so that pending jobs can be executed.""" if self.state != DocumentState.SAVED: raise ValueError("Cannot start a Function that has not been saved") response = self._client.session.patch( f"/functions/{self.id}", json={"status": FunctionStatus.READY.value} ) self._load_from_remote(response.json())
[docs] def stop(self): """Stops Function so that pending jobs cannot be executed.""" if self.state != DocumentState.SAVED: raise ValueError("Cannot start a Function that has not been saved") response = self._client.session.patch( f"/functions/{self.id}", json={"status": FunctionStatus.STOPPED.value} ) self._load_from_remote(response.json())
[docs] @deprecate(renamed={"iterargs": "kwargs"}) def map( self, args: Iterable[Iterable[Any]], kwargs: Iterable[Mapping[str, Any]] = None, tags: List[str] = None, batch_size: int = 1000, environments: Iterable[Mapping[str, str]] = None, ) -> JobBulkCreateResult: """Submits multiple jobs efficiently with positional args to each function call. Preferred over repeatedly calling the function, such as in a loop, when submitting multiple jobs. If supplied, the length of ``kwargs`` must match the length of ``args``. All parameter values must be JSON serializable. As an example, if the function takes two positional arguments and has a keyword argument ``x``, you can submit multiple jobs like this: >>> async_func.map([['a', 'b'], ['c', 'd']], [{'x': 1}, {'x': 2}]) # doctest: +SKIP is equivalent to: >>> async_func('a', 'b', x=1) # doctest: +SKIP >>> async_func('c', 'd', x=2) # doctest: +SKIP Notes ----- Map is idempotent for the initial call such that request errors that occur once started, will not cause duplicate jobs to be submitted. However, if the method is called again with the same arguments, it will submit duplicate jobs. You should always check the return value to ensure all jobs were submitted successfully and handle any errors that may have occurred. Parameters ---------- args : Iterable[Iterable[Any]] An iterable of iterables of arguments. For each outer element, a job will be submitted with each of its elements as the positional arguments to the function. The length of each element of the outer iterable must match the number of positional arguments to the function. kwargs : Iterable[Mapping[str, Any]], optional An iterable of Mappings with keyword arguments. For each outer element, the Mapping will be expanded into keyword arguments for the function. environments : Iterable[Mapping[str, str]], optional AN iterable of Mappings of Environment variables to be set in the environment of the running Jobs. The values for each job will be merged with environment variables set on the Function, with the Job environment variables taking precedence. tags : List[str], optional A list of tags to apply to all jobs submitted. batch_size : int, default=1000 The number of jobs to submit in each batch. The maximum batch size is 1000. Returns ------_ JobBulkCreateResult An object containing the jobs that were submitted and any errors that occurred. This object is compatible with a list of Job objects for backwards compatibility. If the value of `JobBulkCreateResult.is_success` is False, you should check `JobBulkCreateResult.errors` and handle any errors that occurred. Raises ------ ClientError, ServerError If the request to create the first batch of jobs, fails after all retries have been exhausted. Otherwise, any errors will be available in the returned JobBulkCreateResult. """ if self.state != DocumentState.SAVED: raise ValueError("Cannot execute a Function that has not been saved") if batch_size < 1 or batch_size > 1000: raise ValueError("Batch size must between 1 and 1000") args = [list(iterable) for iterable in args] if kwargs is not None: kwargs = [dict(mapping) for mapping in kwargs] if len(kwargs) != len(args): raise ValueError( "The number of kwargs must match the number of args. " f"Got {len(args)} args and {len(kwargs)} kwargs." ) if environments is not None: environments = [dict(mapping) for mapping in environments] if len(environments) != len(args): raise ValueError( "The number of environments must match the number of args. " f"Got {len(args)} args and {len(environments)} environments." ) result = JobBulkCreateResult() # Send the jobs in batches of batch_size for index, (positional, named, env) in enumerate( itertools.zip_longest( batched(args, batch_size), batched(kwargs or [], batch_size), batched(environments or [], batch_size), ) ): payload = { "function_id": self.id, "bulk_args": positional, "bulk_kwargs": named, "bulk_environments": env, "reference_id": str(uuid.uuid4()), } if tags: payload["tags"] = tags # This implementation uses a `reference_id` to ensure that the request is idempotent # and duplicate jobs are not submitted in a retry scenario. try: response = self._client.session.post("/jobs/bulk", json=payload) result.extend([Job(**data, saved=True) for data in response.json()]) except exceptions.NotFoundError: # If one of these errors occurs, we cannot continue submitting any jobs at all raise except Exception as exc: if index == 0: # The first batch failed, let the user deal with the exception as all # the batches would likely fail. raise result.append_error( JobBulkCreateError( function=self, args=payload["bulk_args"], kwargs=payload["bulk_kwargs"], environments=payload["bulk_environments"], reference_id=payload["reference_id"], exception=exc, ) ) return result
[docs] def cancel_jobs(self, query: Optional[JobSearch] = None, job_ids: List[str] = None): """Cancels all jobs for the Function matching the given query. If both `query` and `job_ids` are None, all jobs for the Function will be canceled. If both are provided, they will be combined with an AND operator. Any jobs matched by `query` or `job_ids` which are not associated with this function will be ignored. Parameters ---------- query : JobSearch, optional Query to filter jobs to cancel. job_ids : List[str], optional List of job ids to cancel. """ if self.state != DocumentState.SAVED: raise ValueError( "Cannot cancel jobs for a Function that has not been saved" ) if query is None: query = self.jobs else: query = query.filter(Job.function_id == self.id) if job_ids: query = query.filter(Job.id.in_(job_ids)) return query.cancel()
[docs] def rerun(self, query: Optional[JobSearch] = None, job_ids: List[str] = None): """Submits all the unsuccessful jobs matching the query to be rerun. If both `query` and `job_ids` are None, all rerunnable jobs for the Function will be rerun. If both are provided, they will be combined with an AND operator. Any jobs matched by `query` or `job_ids` which are not associated with this function will be ignored. Parameters ---------- query : JobSearch, optional Query to filter jobs to rerun. job_ids : List[str], optional List of job ids to rerun. """ if self.state != DocumentState.SAVED: raise ValueError("Cannot execute a Function that has not been saved") if query is None: query = self.jobs else: query = query.filter(Job.function_id == self.id) if job_ids: query = query.filter(Job.id.in_(job_ids)) return query.rerun()
[docs] def delete_jobs( self, query: Optional[JobSearch] = None, job_ids: List[str] = None, delete_results: bool = False, ) -> List[str]: """Deletes all non-running jobs for the Function matching the given query. If both `query` and `job_ids` are None, all jobs for the Function will be deleted. If both are provided, they will be combined with an AND operator. Any jobs matched by `query` or `job_ids` which are not associated with this function will be ignored. Also deletes any job log blobs for the jobs. Use `delete_results=True` to delete the job result blobs as well. There is a limit to how many jobs can be deleted in a single request before the request times out. If you need to delete a large number of jobs and experience timeouts, consider using a loop to delete batches, using the `query` parameter with a limit (e.g. ``async_func.delete_jobs(async_func.jobs.limit(10000))``, or use the `job_ids` parameter to limit the number of jobs to delete. Parameters ---------- query : JobSearch, optional Query to filter jobs to delete. job_ids : List[str], optional List of job ids to delete. delete_results : bool, default=False If True, deletes the job result blobs as well. Returns ------- List[str] List of job ids that were deleted. """ if self.state != DocumentState.SAVED: raise ValueError( "Cannot delete jobs for a Function that has not been saved" ) if query is None: query = self.jobs else: query = query.filter(Job.function_id == self.id) if job_ids: query = query.filter(Job.id.in_(job_ids)) return query.delete(delete_results=delete_results)
[docs] def refresh(self, includes: Optional[str] = None): """Updates the Function instance with data from the server. Parameters ---------- includes : Optional[str], optional List of additional attributes to include in the response. Allowed values are: - "job.statistics": Include statistics about the Job statuses for this Function. """ if self.state == DocumentState.NEW: raise ValueError("Cannot refresh a Function that has not been saved") if includes: params = {"include": includes} elif includes is None and self.job_statistics: params = {"include": ["job.statistics"]} else: params = {} response = self._client.session.get(f"/functions/{self.id}", params=params) self._load_from_remote(response.json())
[docs] def iter_results(self, cast_type: Type[Serializable] = None): """Iterates over all successful job results. Parameters ---------- cast_type : Type[Serializable], optional If set, the results will be cast to the given type. """ if self.state != DocumentState.SAVED: raise ValueError( "Cannot retrieve results for a Function that has not been saved" ) for job in self.jobs.filter(Job.status == JobStatus.SUCCESS): yield job.result(cast_type=cast_type)
[docs] def results(self, cast_type: Type[Serializable] = None): """Retrieves all the job results for the Function as a list. Notes ----- This immediately downloads all results into a list and could run out of memory. If the result set is large, strongly consider using :py:meth:`Function.iter_results` instead. Parameters ---------- cast_type : Type[Serializable], optional If set, the results will be cast to the given type. """ return list(self.iter_results(cast_type=cast_type))
[docs] def as_completed(self, jobs: Iterable[Job], timeout=None, interval=10): """Yields jobs as they complete. Completion includes success, failure, timeout, and canceled. Can be used in any iterable context. Parameters ---------- jobs : Iterable[Job] The jobs to wait for completion. Jobs must be associated with this Function. timeout : int, default=None Maximum time to wait before timing out. If not set, this will continue polling until all jobs have completed. interval : int, default=10 Interval in seconds for how often to check if jobs have been completed. """ if self.state != DocumentState.SAVED: raise ValueError( "Cannot retrieve jobs for a Function that has not been saved" ) job_ids = set() for job in jobs: if job.function_id != self.id: raise ValueError( f"Job {job.id} is not associated with Function {self.id}" ) job_ids.add(job.id) current_interval = interval chunk_size = MAX_FUNCTION_IDS_PER_REQUEST start_time = time.time() while job_ids: self.refresh() if self.status == FunctionStatus.BUILD_FAILED: raise RuntimeError( f"Function {self.name} failed to build. Check the build log for more details." ) hits = set() query_ids = {job_id for job_id in job_ids} # refresh the job state, but only a chunk at a time while query_ids: chunk_ids = set( itertools.islice(query_ids, min(chunk_size, len(query_ids))) ) query_ids -= chunk_ids for job in Job.search(client=self._client).filter( Job.id.in_(list(chunk_ids)) & Job.status.in_(JobStatus.terminal()) ): hits.add(job.id) yield job job_ids -= hits if hits: current_interval = interval if timeout: t = timeout - (time.time() - start_time) if t <= 0: raise TimeoutError( f"Function {self.name} did not complete before timeout!" ) t = min(t, current_interval) else: t = current_interval # Don't sleep as long as we are picking up hits if not hits: time.sleep(t) current_interval = min(current_interval * 2, 60)
[docs] def wait_for_completion(self, timeout=None, interval=10): """Waits until all submitted jobs for a given Function are completed. Completion includes success, failure, timeout, and canceled. Parameters ---------- timeout : int, default=None Maximum time to wait before timing out. If not set, this method will block indefinitely. interval : int, default=10 Interval in seconds for how often to check if jobs have been completed. """ if self.state != DocumentState.SAVED: raise ValueError("Cannot wait for a Function that has not been saved") start_time = time.time() while True: self.refresh(includes=["job.statistics"]) if self.status == FunctionStatus.BUILD_FAILED: raise RuntimeError( f"Function {self.name} failed to build. Check the build log for more details." ) if self.status in [FunctionStatus.READY, FunctionStatus.STOPPED]: job_statistics = getattr(self, "job_statistics", {}) if ( job_statistics.get("pending", 0) == 0 and job_statistics.get("running", 0) == 0 and job_statistics.get("cancel", 0) == 0 and job_statistics.get("canceling", 0) == 0 ): break if timeout: t = timeout - (time.time() - start_time) if t <= 0: raise TimeoutError( f"Function {self.name} did not complete before timeout!" ) t = min(t, interval) else: t = interval time.sleep(t)