{ "info": { "author": "Acrisel Team", "author_email": "support@acrisel.com", "bugtrack_url": null, "classifiers": [ "Development Status :: 5 - Production/Stable", "Environment :: Other Environment", "Intended Audience :: Developers", "License :: OSI Approved :: MIT License", "Operating System :: OS Independent", "Programming Language :: Python", "Programming Language :: Python :: 3", "Programming Language :: Python :: 3.2", "Programming Language :: Python :: 3.3", "Programming Language :: Python :: 3.4", "Programming Language :: Python :: 3.5", "Programming Language :: Python :: 3.6", "Topic :: Software Development :: Libraries :: Application Frameworks" ], "description": "=======\nSequent\n=======\n\n------------------------------------------------------------\nProgramming interface to use in programs to create task flow\n------------------------------------------------------------\n\n.. contents:: Table of Contents\n :depth: 1\n\nOverview\n========\n\n *Sequent* provides programmer with interface to create flow. *Sequent* is useful when there are multiple batch processes that should link together in processing flow.\n \n For example, programs A1, A2 extract data from database, program B process the extracted files, and programs C1 and C2 transmit process information to remote client.\n \n *Sequent* can be used to create the flow where A1, A2 would run concurrently, then program B would run. When B is successful, programs C1 and C2 would run.\n \n *Sequent* uses *Eventor* (https://github.com/Acrisel/eventor) as its underline infrastructure. With that, it gains *Eventor*'s recovery capabilities. When running in recovery, successful steps can be skipped.\n \n Considering the above example, this can be useful when C2 transmission program fails. Recovery run would only execute it with out redoing potentially expensive work that was already completed (A1, A2, B, and C1)\n \n It would be easier to show an example. In this example, *Step s2* would run after *Step s1* would loop twice. When they are done, *Step s3* will run. This flow would be repeat twice.\n\nSimple Example\n--------------\n \n .. code-block:: python\n :number-lines:\n \n import sequent as seq\n import logging\n\n def prog(progname, success=True):\n logger = logging.getLogger(os.getenv(\"SEQUENT_LOGGER_NAME\"))\n logger.info(\"doing what {} is doing\".format(progname))\n if not success:\n raise Exception(\"{} failed\".format(progname))\n return progname\n\n myflow = seq.Sequent(config={'sleep_between_loops': 0.05, \n 'LOGGING': {'logging_level':logging.DEBUG}})\n\n s1 = myflow.add_step('s1', repeats=[1,2])\n\n s11 = s1.add_step('s11', repeats=[1,2,])\n\n s111 = s11.add_step('s111', func=prog, kwargs={'progname': 'prog1'}) \n s112 = s11.add_step('s112', func=prog, kwargs={'progname': 'prog2'}, \n requires=( (s111, seq.STEP_SUCCESS), )) \n\n s12 = s1.add_step('s12', func=prog, kwargs={'progname': 'prog3'}, \n require=s( (s11, seq.STEP_SUCCESS), )) \n\n s2 = myflow.add_step('s2', func=prog, kwargs={'progname': 'prog4'}, \n requires=((s1, seq.STEP_SUCCESS),)) \n\n myflow.run() \n \nExample Output\n--------------\n\n The above example with provide the following log output. Note more detailed logging activities was stripped off. Only actual shows actual program activity is shown.\n \n .. code-block:: python\n :number-lines:\n\n [ 2016-12-07 11:14:29,761 ][ INFO ][ Eventor store file: /sequent/example/runly00.run.db ]\n ...\n [ 2016-12-07 11:14:29,962 ][ INFO ][ doing what prog1 is doing ]\n ...\n [ 2016-12-07 11:14:30,124 ][ INFO ][ doing what prog2 is doing ]\n ...\n [ 2016-12-07 11:14:30,358 ][ INFO ][ doing what prog1 is doing ]\n ...\n [ 2016-12-07 11:14:30,587 ][ INFO ][ doing what prog2 is doing ]\n ...\n [ 2016-12-07 11:14:30,908 ][ INFO ][ doing what prog3 is doing ]\n ...\n [ 2016-12-07 11:14:31,234 ][ INFO ][ doing what prog1 is doing ]\n ...\n [ 2016-12-07 11:14:31,407 ][ INFO ][ doing what prog2 is doing ]\n ...\n [ 2016-12-07 11:14:31,657 ][ INFO ][ doing what prog1 is doing ]\n ...\n [ 2016-12-07 11:14:31,894 ][ INFO ][ doing what prog2 is doing ]\n ...\n [ 2016-12-07 11:14:32,240 ][ INFO ][ doing what prog3 is doing ]\n ...\n [ 2016-12-07 11:14:32,565 ][ INFO ][ doing what prog4 is doing ]\n ...\n [ 2016-12-07 11:14:32,713 ][ INFO ][ Processing finished with: success ]\n\nCode Highlights\n---------------\n\n Flow diagram:\n \n .. code-block:: python\n \n \n +--S1----------------------+\n | |\n | +--S11----------+ |\n | | | |\n | | S111 -> S112 | -> S12 | -> S2\n | +---------------+ |\n +--------------------------+\n\n For simplicity, code definition of prog (line 6) serves as reusable activity for all the steps in this example.\n \n A *Sequent* object is defined (line 12) to host myflow. By default, Sequent's Eventor loops on events and steps. By defaults it sleeps one second between loops. Here '*sleep_between_loops*' changes this setting to 0.05 seconds. \n \n myflow contains two steps, *s1* and *s2*. *s1* is a container step that would repeat twice (defined on line 15). *s2* is a processing step (defined on line 26).\n \n *s1* contains two steps. *s11* (line 17) is *container* step and *s12* is a processing step. \n \n *s11* contains two processing steps *s111* and *s112* (lines 19-20). \n \n Finally, on line 29 the flow is executed using *myflow()*.\n \n *logger* is set with in step program (line 5) to direct step logging into its dedicated log.\n \nSequent Interface\n=================\n\nSequent Class Initiator\n-----------------------\n\n Sequent signature in its most simplistic format:\n \n .. code-block:: python\n \n Sequent(name='', store='', run_mode=SEQ.SEQUENT_RESTART, recovery_run=None, config={}, config_tag='')\n\nDescription\n```````````\n\n Sequent, when instantiated, provides interface to build program flow. When called upon, *Sequent* steps are translated to *Eventor* steps and *Step*'s *requires* are translated to *Eventor*'s *Events* and *Steps'* *triggers*.\n \n Sequent instantiation arguments are the same as *Eventor*'s. \n\nArgs\n````\n\n name: string id for Sequent object initiated.\n \n store: path to file that would store runnable (sqlite) information; if ':memory:' is used, in-memory temporary \n storage will be created. If not provided, calling module path and name will be used \n with db extension instead of 'py'.\n \n run_mode: can be either *RUN_RESTART* (default) or *RUN_RECOVER*; in restart, new instance or the run \n will be created. In recovery, \n \n recovery_run: if *RUN_RECOVER* is used, *recovery_run* will indicate specific instance of previously recovery \n run that would be executed.If not provided, latest run would be used.\n \n config: keyword dictionary of default configurations. Available keywords and their default values:\n \n +---------------------+------------------+--------------------------------------------------+\n | Name | Default | Description |\n | | Value | |\n +=====================+==================+==================================================+\n | workdir | /tmp | place to create necessry artifacts (not in use) |\n +---------------------+------------------+--------------------------------------------------+\n | logdir | /var/log/eventor | place to create debug and error log files |\n +---------------------+------------------+--------------------------------------------------+\n | task_construct | 'process' | method to use for execution of steps |\n +---------------------+------------------+--------------------------------------------------+\n | max_concurrent | 1 | maximum concurrent processing, if value <1, no |\n | | | limit will be pose |\n +---------------------+------------------+--------------------------------------------------+\n | stop_on_exception | True | if an exception occurs in a step, stop |\n | | | all processes. If True, new processes will not |\n | | | start. But running processes will be permitted |\n | | | to finish |\n +---------------------+------------------+--------------------------------------------------+\n | sleep_between_loops | 1 | seconds to sleep between iteration of checking |\n | | | triggers and tasks |\n +---------------------+------------------+--------------------------------------------------+\n \n config_tag: key with in config where Sequent configuration starts.\n \nSequent add_event method\n------------------------\n\n .. code-block:: python\n \n add_event(require=None)\n\nArgs\n````\n\n *requires*: logical expression 'sqlalchemy' style to automatically raise this expression.\n syntax: \n \n .. code-block:: python\n \n requires : (requires, requires, ...)\n | or_(requires, requires, ...) \n | event\n \n - if expression is of the first style, logical *and* will apply.\n - the second expression will apply logical *or*.\n - the basic atom in expression is *even* which is the product of add_event.\n \nReturns\n```````\n\n Event object to use are require in *add_step*.\n \nSequent add_step method\n-----------------------\n\n .. code-block:: python\n \n add_step(name, func, args=(), kwargs={}, hosts=[], requires={}, delay=0, acquires=[], releases=None, recovery={}, config={})\n\nArgs\n````\n\n *name*: string unique id for step \n \n *func*: callable object that would be call at time if step execution\n \n *args*: tuple of values that will be passed to *func* at calling\n \n *kwargs*: keywords arguments that will be passed to *func* at calling\n \n *hosts*: list of hosts step should run on. If not provided, *localhost* will be used.\n if \n \n *requires*: mapping of step statuses such that when set of events, added step will be launched:\n \n +---------------+-------------------------------------------+\n | status | description |\n +===============+===========================================+\n | STEP_READY | set when task is ready to run (triggered) |\n +---------------+-------------------------------------------+\n | STEP_ACTIVE | set when task is running |\n +---------------+-------------------------------------------+\n | STEP_SUCCESS | set when task is successful |\n +---------------+-------------------------------------------+\n | STEP_FAILURE | set when task fails |\n +---------------+-------------------------------------------+\n | STEP_COMPLETE | stands for success or failure of task |\n +---------------+-------------------------------------------+\n \n *delay*: seconds to wait before executing step once is requires are available. Actual execution \n may be delayed further if resources needs to be acquired.\n \n *acquires*: list of tuples of resource pool and amount of resources to acquire before starting. \n \n *releases*: list of tuples of resources pool and amount of resources to release once completed.\n If None, defaults to *acquires*. If set to empty list, none of the acquired resources would \n be released.\n \n *recovery*: mapping of state status to how step should be handled in recovery:\n \n +--------------+-----------+------------------------------------------------------+\n | status | default | description |\n +==============+===========+======================================================+\n | STEP_READY | STP_RERUN | if in recovery and previous status is ready, rerun |\n +--------------+-----------+------------------------------------------------------+\n | STEP_ACTIVE | STP_RERUN | if in recovery and previous status is active, rerun |\n +--------------+-----------+------------------------------------------------------+\n | STEP_FAILURE | STP_RERUN | if in recovery and previous status is failure, rerun |\n +--------------+-----------+------------------------------------------------------+\n | STEP_SUCCESS | STP_SKIP | if in recovery and previous status is success, skip |\n +--------------+-----------+------------------------------------------------------+\n \n *config*: keywords mapping overrides for step configuration.\n \n +-------------------+---------------+---------------------------------------+\n | name | default | description |\n +===================+===============+=======================================+\n | stop_on_exception | True | stop flow if step ends with Exception | \n +-------------------+---------------+---------------------------------------+\n \nReturns\n```````\n\n Step object to use in add_assoc method.\n\nSequent run method\n------------------\n\n .. code-block:: python\n \n run(max_loops=-1)\n \nwhen calling *run* method, information is built and loops evaluating events and task starts are executed. \nIn each loop events are raised and tasks are performed. max_loops parameters allows control of how many\nloops to execute.\n\nIn simple example, **myflow.run()** engage Sequent's run() method.\n \nArgs\n````\n\n *max_loops*: max_loops: number of loops to run. If positive, limits number of loops.\n defaults to negative, which would run loops until there are no events to raise and\n no task to run. \n \nReturns\n```````\n\n If there was a failure that was not followed by event triggered, result will be False.\n\nDistributed Operation \n=====================\n\n*Sequent* can operate Steps on distributed environment. A step can be associated with hosts using *hosts* argument in *add_step*. *Sequent* uses SSH to submit steps to remote host. This means that cluster needs to be configured with SSH keys. To set up the environment for *Sequent* distributed operation:\n\n1. Host from which *Sequent* program would be initiated, should be able to SSH to participating hosts without only using keys.\n#. SSH authorized_keys on each target host should has proper *command* to initiate the right operation environment. This may include activating the correct virtualenv.\n#. Optionally, set SSH backdoor to originated host. In the future *Sequent* may use this backdoor, as callback.\n#. Software needs to be uniformly installed on all participating machines.\n#. *Sequent* must be initiated with database configuration that is accessible from all participating hosts. *Sequent* and its remote agents would use that database to share operation information. The database user needs to have permissions to create schema (if the associated schema is not created.) It also needs to have create table permissions.\n#. Anything passed to Sequent, predominately with *add_step*, needs to be importable. For example in simple example:\n\n .. code-block:: python\n \n import example_progs as example\n \n s111 = s11.add_step('s111', func=example.prog, kwargs={'progname': 'prog1'})\n\n\nRecovery\n========\n\n Recovery allows rerun of a program in a way that it will skip successful steps. To use recovery, store mast be physical (cannot use in-memory). \n \n According to step recovery setup, when in recovery, step may be skipped or rerun. By default, only success statuses are skipped.\n \n Here is an example for recovery program and run.\n \nRecovery Example\n----------------\n\n .. code-block:: python\n :number-lines:\n \n import sequent as sqnt\n import logging\n\n appname = os.path.basename(__file__)\n logger = logging.getLogger(appname)\n\n def prog(flow, progname, step_to_fail=None, iteration_to_fail=''):\n logger = logging.getLogger(os.getenv(\"SEQUENT_LOGGER_NAME\"))\n step_name = flow.get_step_name() \n step_sequence = flow.get_step_sequence()\n logger.info(\"doing what {} is doing (}/{})\".format(progname, step_name, step_sequence))\n if step_to_fail == step_name and step_sequence== iteration_to_fail:\n raise Exception(\"{} failed ({}/{})\".format(progname, step_name, step_sequence))\n return progname\n\n def build_flow(run_mode = sqnt.RUN_RESTART, run_id=None, step_to_fail=None, iteration_to_fail=''):\n myflow = sqnt.Sequent(name=appname, run_mode=run_mode, run_id=run_id, config={'sleep_between_loops': 0.05,}, )\n\n s1 = myflow.add_step('s1', repeats=[1,2])\n \n s11 = s1.add_step('s11', repeats=[1,2,])\n \n s111 = s11.add_step('s111', func=prog, kwargs={'flow': myflow, 'progname': 'prog1', \n 'step_to_fail':step_to_fail, \n 'iteration_to_fail':iteration_to_fail,}) \n s112 = s11.add_step('s112', func=prog, kwargs={'flow': myflow, 'progname': 'prog2', \n 'step_to_fail':step_to_fail, \n 'iteration_to_fail':iteration_to_fail,}, \n requires=((s111, sqnt.STEP_SUCEESS),)) \n \n s12 = s1.add_step('s12', func=prog, kwargs={'flow': myflow, 'progname': 'prog3', \n 'step_to_fail':step_to_fail, \n 'iteration_to_fail':iteration_to_fail,}, \n requires=((s11, sqnt.STEP_SUCEESS),)) \n \n s2 = myflow.add_step('s2', func=prog, kwargs={'flow': myflow, 'progname': 'prog4', \n 'step_to_fail':step_to_fail, \n 'iteration_to_fail':iteration_to_fail,}, \n requires=((s1, sqnt.STEP_SUCEESS),)) \n return myflow\n\n # creating flow simulating failure\n myflow = build_flow(step_to_fail='s1_s11_s111', iteration_to_fail='1.2.2')\n myflow.run()\n \n run_id = myflow.run_id\n\n # creating recovery flow\n myflow = build_flow(run_mode=RUN_RECOVER, run_id=run_id)\n myflow.run()\n \nExample Output\n--------------\n\n .. code-block:: python\n :number-lines:\n \n [ 2016-12-07 14:49:24,437 ][ INFO ][ Eventor store file: /sequent/example/runly04.run.db ]\n ...\n [ 2016-12-07 14:49:24,645 ][ INFO ][ doing what prog1 is doing (s1_s11_s111/1.1.1) ]\n ...\n [ 2016-12-07 14:49:24,805 ][ INFO ][ doing what prog2 is doing (s1_s11_s112/1.1.1) ]\n ...\n [ 2016-12-07 14:49:25,047 ][ INFO ][ doing what prog1 is doing (s1_s11_s111/1.1.2) ]\n ...\n [ 2016-12-07 14:49:25,272 ][ INFO ][ doing what prog2 is doing (s1_s11_s112/1.1.2) ]\n ...\n [ 2016-12-07 14:49:25,587 ][ INFO ][ doing what prog3 is doing (s1_s12/1.1) ]\n ...\n [ 2016-12-07 14:49:25,909 ][ INFO ][ doing what prog1 is doing (s1_s11_s111/1.2.1) ]\n ...\n [ 2016-12-07 14:49:26,073 ][ INFO ][ doing what prog2 is doing (s1_s11_s112/1.2.1) ]\n ...\n [ 2016-12-07 14:49:26,321 ][ INFO ][ doing what prog1 is doing (s1_s11_s111/1.2.2) ]\n [ 2016-12-07 14:49:26,323 ][ INFO ][ [ Step s1_s11_s111/1.2.2 ] Completed, status: TaskStatus.failure ]\n [ 2016-12-07 14:49:26,397 ][ ERROR ][ Exception in run_action: \n ]\n [ 2016-12-07 14:49:26,397 ][ ERROR ][ Exception('prog1 failed (s1_s11_s111/1.2.2)',) ]\n [ 2016-12-07 14:49:26,397 ][ ERROR ][ File \"/eventor/eventor/main.py\", line 63, in task_wrapper\n result=step(seq_path=task.sequence)\n File \"/eventor/eventor/step.py\", line 82, in __call__\n result=func(*func_args, **func_kwargs)\n File \"/sequent/example/runly04.py\", line 34, in prog\n raise Exception(\"%s failed (%s/%s)\" % (progname, step_name, step_sequence)) ]\n [ 2016-12-07 14:49:26,397 ][ INFO ][ Stopping running processes ]\n [ 2016-12-07 14:49:26,401 ][ INFO ][ Processing finished with: failure ]\n [ 2016-12-07 14:49:26,404 ][ INFO ][ Eventor store file: /sequent/example/runly04.run.db ]\n ...\n [ 2016-12-07 14:49:27,921 ][ INFO ][ doing what prog1 is doing (s1_s11_s111/1.2.2) ]\n ...\n [ 2016-12-07 14:49:28,159 ][ INFO ][ doing what prog2 is doing (s1_s11_s112/1.2.2) ]\n ...\n [ 2016-12-07 14:49:28,494 ][ INFO ][ doing what prog3 is doing (s1_s12/1.2) ]\n ...\n [ 2016-12-07 14:49:28,844 ][ INFO ][ doing what prog4 is doing (s2/1) ]\n [ 2016-12-07 14:49:28,845 ][ INFO ][ [ Step s2/1 ] Completed, status: TaskStatus.success ]\n [ 2016-12-07 14:49:29,002 ][ INFO ][ Processing finished with: success ]\n\nExample Highlights\n------------------\n \n The function *build_flow* (code line 14) build a Sequent flow similarly to simple example above. Since no specific store is provided in Sequent instantiation, a default runner store is assigned (code line 15). In this build, steps will use default recovery directives whereby successful steps are skipped. \n \n The first build and run is done in lines 42-43. In this run, a parameter is passed to cause step *s111* in its fourth iteration to fail. As a result, flow fails. Output lines 1-29 is associated with the first run. \n \n The second build and run is then initiated. In this run, parameter is set to a value that would pass step *s111* and run mode is set to recovery (code lines 45-46). Eventor skips successful steps and start executing from failed steps onwards. Output lines 30-40 reflects successful second run.\n \n Note that the second run required a **run_id** of the run that is reactivated. *run_id* is fetched from its corresponding attribute in *Sequent* Objects.\n \n For prog to know when to default, it uses the following methods flow.get_step_name() and flow.get_step_sequence() (lines 7-8). Those Sequent methods allow access to Eventor step attributes. Another way\n to access these attributes is via os.environ:\n \n .. code-block:: python\n \n name = os.getenv('SEQUENT_STEP_NAME')\n sequence = os.getenv('SEQUENT_STEP_SEQUENCE')\n recovery = os.getenv('SEQUENT_STEP_RECOVERY')\n logger_name = os.getenv('SEQUENT_LOGGER_NAME')\n \nDistributed Example\n-------------------\n\nResources\n=========\n\n *add_step* allows association of step with resources. If acquires argument is provided, before step starts, *Eventor* \n will attempt to reserve resources. Step will be executed only when resources are secured.\n \n When *release* argument is provided, resources resources listed as its value will be released when step is done. If \n release is None, whatever resources stated by *acquires* would be released. If the empty list is set as value, no \n resource would be released.\n \n To use resources, program to use Resource and ResourcePool from acris.virtual_resource_pool. Example for such definitions are below.\n \nExample for resources definitions\n---------------------------------\n\n .. code-block:: python\n :number-lines:\n \n import sequent as sqnt\n from acris import virtual_resource_pool as vrp\n\n class Resources1(vrp.Resource): pass\n class Resources2(vrp.Resource): pass\n \n rp1 = vrp.ResourcePool('RP1', resource_cls=Resources1, policy={'resource_limit': 2, }).load() \n rp2 = vrp.ResourcePool('RP2', resource_cls=Resources2, policy={'resource_limit': 2, }).load()\n \n myflow = sqnt.Sequent(config={'sleep_between_loops': 0.05,}, )\n s1 = myflow.add_step('s1', repeats=[1,2], acquires=[(rp1, 2), ])\n \nAdditional Information\n======================\n\n Sequent github project (https://github.com/Acrisel/sequent) has additional examples with more complicated flows.", "description_content_type": null, "docs_url": null, "download_url": "", "downloads": { "last_day": -1, "last_month": -1, "last_week": -1 }, "home_page": "https://github.com/Acrisel/sequent", "keywords": "framework workflow flow program dependency dependencies procedural procedure sequential recovery restart", "license": "MIT", "maintainer": "", "maintainer_email": "", "name": "sequent", "package_url": "https://pypi.org/project/sequent/", "platform": "", "project_url": "https://pypi.org/project/sequent/", "project_urls": { "Homepage": "https://github.com/Acrisel/sequent" }, "release_url": "https://pypi.org/project/sequent/3.1.7/", "requires_dist": null, "requires_python": "", "summary": "sequent is a programming interface based on eventor providing simple way to write program flows", "version": "3.1.7" }, "last_serial": 3594020, "releases": { "0.9": [], "1.0.1": [ { "comment_text": "", "digests": { "md5": "6e1158823ab15090e6789d462a5c30fa", "sha256": "90a8f6445d4f150a1b74e543c13ed6d7334cbf9f4212c3e086be57d3d704ec09" }, "downloads": -1, "filename": "sequent-1.0.1.tar.gz", "has_sig": false, "md5_digest": "6e1158823ab15090e6789d462a5c30fa", "packagetype": "sdist", "python_version": "source", "requires_python": null, "size": 24717, "upload_time": "2016-12-07T22:12:27", "url": "https://files.pythonhosted.org/packages/50/fa/a137dfe0429eb464326b3582a04182606a32a2e2d814f1b2c78a1b942df3/sequent-1.0.1.tar.gz" } ], "1.0.2": [ { "comment_text": "", "digests": { "md5": "ef32d0b00e6a720f790f9c421a2f48d4", "sha256": "640d7fdab18ee6f641e74b99f04709412ed272ccd62aa2c21e9f7545d3f7b920" }, "downloads": -1, "filename": "sequent-1.0.2.tar.gz", "has_sig": false, "md5_digest": "ef32d0b00e6a720f790f9c421a2f48d4", "packagetype": "sdist", "python_version": "source", "requires_python": null, "size": 24709, "upload_time": "2016-12-07T22:36:14", "url": "https://files.pythonhosted.org/packages/d6/9d/6f35f68a06eac12c2760483a9f267c24d97fb72ade76013bb5a7f4025a2f/sequent-1.0.2.tar.gz" } ], "1.0.3": [ { "comment_text": "", "digests": { "md5": "81055a885b45cf0acaaeb07d664e738a", "sha256": "943c599abfe1564df06ce181fbbc17735dcc80070b7089cbce2dd5ef84afe646" }, "downloads": -1, "filename": "sequent-1.0.3.tar.gz", "has_sig": false, "md5_digest": "81055a885b45cf0acaaeb07d664e738a", "packagetype": "sdist", "python_version": "source", "requires_python": null, "size": 21388, "upload_time": "2016-12-20T00:59:00", "url": "https://files.pythonhosted.org/packages/e2/f4/759970ba80d0346d7fa6ea99bacde6ff4e28f58defc5d86e16191d33d1c1/sequent-1.0.3.tar.gz" } ], "2.0.1": [ { "comment_text": "", "digests": { "md5": "a04fea0193496802f9405a81e6c019e8", "sha256": "880739ab06d413951e5d3b716090509a24578d142bcbd24271778506dbeea87b" }, "downloads": -1, "filename": "sequent-2.0.1.tar.gz", "has_sig": false, "md5_digest": "a04fea0193496802f9405a81e6c019e8", "packagetype": "sdist", "python_version": "source", "requires_python": null, "size": 21455, "upload_time": "2017-01-13T00:59:49", "url": "https://files.pythonhosted.org/packages/17/76/cc65a4a29f08be366a76d41fd2621b5ec837391e64e4569749398826821b/sequent-2.0.1.tar.gz" } ], "2.0.2": [ { "comment_text": "", "digests": { "md5": "6cc7855272afb68de6673690081193f8", "sha256": "08a02cf5877bf0f8f39d00d27fefd72168bb0babe3ccd235719fdd18b2b7e08d" }, "downloads": -1, "filename": "sequent-2.0.2.tar.gz", "has_sig": false, "md5_digest": "6cc7855272afb68de6673690081193f8", "packagetype": "sdist", "python_version": "source", "requires_python": null, "size": 21445, "upload_time": "2017-01-13T01:02:18", "url": "https://files.pythonhosted.org/packages/9e/16/41f4780049cbb434b1287ea2129a2e15c5f5d1a89b0344ce2adb11807c90/sequent-2.0.2.tar.gz" } ], "2.0.4": [ { "comment_text": "", "digests": { "md5": "56d1c7fe30396099f8193884ee054db1", "sha256": "ea0b1bffb5fe48fb9239c966344190f794fc950991c0071c0ed5507ff6e2a97a" }, "downloads": -1, "filename": "sequent-2.0.4.tar.gz", "has_sig": false, "md5_digest": "56d1c7fe30396099f8193884ee054db1", "packagetype": "sdist", "python_version": "source", "requires_python": null, "size": 22297, "upload_time": "2017-01-30T23:21:03", "url": "https://files.pythonhosted.org/packages/3f/2d/76574f1c7fbb6aba80d36c1aad5325004051a1a385685bd9e448e08bff5f/sequent-2.0.4.tar.gz" } ], "2.0.5": [ { "comment_text": "", "digests": { "md5": "6a84dd33fa64bfd962fc072c73c712d0", "sha256": "814f8876ac0c6771d23e8f00e248e23a965bbd8f9e64786bf03f7462e936bb2f" }, "downloads": -1, "filename": "sequent-2.0.5.tar.gz", "has_sig": false, "md5_digest": "6a84dd33fa64bfd962fc072c73c712d0", "packagetype": "sdist", "python_version": "source", "requires_python": null, "size": 22229, "upload_time": "2017-01-30T23:31:40", "url": "https://files.pythonhosted.org/packages/f1/90/7140c9bbbf3451239f6424c2b32d3993959d27e80d3de55dabc95d2178ce/sequent-2.0.5.tar.gz" } ], "2.0.6": [ { "comment_text": "", "digests": { "md5": "87655c9b462c787ac0bc3f41dcef2879", "sha256": "82aa6198bc707e8e787da75ce4612cc5aeab918ae4f0aa098eeb887ff9fbd471" }, "downloads": -1, "filename": "sequent-2.0.6.tar.gz", "has_sig": false, "md5_digest": "87655c9b462c787ac0bc3f41dcef2879", "packagetype": "sdist", "python_version": "source", "requires_python": null, "size": 22221, "upload_time": "2017-02-13T15:58:36", "url": "https://files.pythonhosted.org/packages/ab/13/b489c034422816f9f50f130edc3045f40990ff8527beb75801e4ce31dc6b/sequent-2.0.6.tar.gz" } ], "2.0.7": [ { "comment_text": "", "digests": { "md5": "badb9738bdddcf5da1f491a39afd13a7", "sha256": "3b9cb44dc7ae89daa37ea02748cd0e8c8b886e27651a12efb5425d29ceb29d3b" }, "downloads": -1, "filename": "sequent-2.0.7.tar.gz", "has_sig": false, "md5_digest": "badb9738bdddcf5da1f491a39afd13a7", "packagetype": "sdist", "python_version": "source", "requires_python": null, "size": 22235, "upload_time": "2017-02-13T16:13:12", "url": "https://files.pythonhosted.org/packages/c6/68/e94d9348f8d893f5fba8af0fbe2312939aca9df7d6f78811f6a0d33ae24a/sequent-2.0.7.tar.gz" } ], "2.0.8": [ { "comment_text": "", "digests": { "md5": "6ec0c7a2d7dc14d9f0d8e7dd8bb5ac4d", "sha256": "fb01ac3ed1219456203093c142e3ad30d636b23d49ad5004714c4300f1e23af0" }, "downloads": -1, "filename": "sequent-2.0.8.tar.gz", "has_sig": false, "md5_digest": "6ec0c7a2d7dc14d9f0d8e7dd8bb5ac4d", "packagetype": "sdist", "python_version": "source", "requires_python": null, "size": 22294, "upload_time": "2017-02-13T16:29:36", "url": "https://files.pythonhosted.org/packages/52/c6/64da7d7b07ec0be5aec7a5d9254763902ce6bb30e009e43fd1c531acf870/sequent-2.0.8.tar.gz" } ], "2.1.0": [ { "comment_text": "", "digests": { "md5": "8ad9724054ccf8fb5610b2eb0aa7f720", "sha256": "c4bc276f24bfd8aa2b55ed32ffd713bb23ac75377ac5f6f3b8f1a23baa75d7aa" }, "downloads": -1, "filename": "sequent-2.1.0.tar.gz", "has_sig": false, "md5_digest": "8ad9724054ccf8fb5610b2eb0aa7f720", "packagetype": "sdist", "python_version": "source", "requires_python": null, "size": 22590, "upload_time": "2017-02-14T16:14:41", "url": "https://files.pythonhosted.org/packages/6f/59/9d53257087e08ad919017cb7a6559cd4e38f5918b4a2863ab8a81d18759b/sequent-2.1.0.tar.gz" } ], "3.0.0": [ { "comment_text": "", "digests": { "md5": "e6aad748cdb373354ff2522ad9bbf8d1", "sha256": "152fa83aba74053e3e4315921a1e82ce334457d650e03e514398ed1e88db39f1" }, "downloads": -1, "filename": "sequent-3.0.0.tar.gz", "has_sig": false, "md5_digest": "e6aad748cdb373354ff2522ad9bbf8d1", "packagetype": "sdist", "python_version": "source", "requires_python": null, "size": 22913, "upload_time": "2017-02-15T00:07:47", "url": "https://files.pythonhosted.org/packages/48/6f/3e4a7c9f7f4a429c80692247a6916a17861a4082ab0c48cae7f8ba29837c/sequent-3.0.0.tar.gz" } ], "3.0.1": [ { "comment_text": "", "digests": { "md5": "412e2591967294724cbcdac6124bded2", "sha256": "dbd060674d86b9fbbb460b504cc857aac505dc4eae850dd88fa6d969c1f95dca" }, "downloads": -1, "filename": "sequent-3.0.1.tar.gz", "has_sig": false, "md5_digest": "412e2591967294724cbcdac6124bded2", "packagetype": "sdist", "python_version": "source", "requires_python": null, "size": 22994, "upload_time": "2017-02-15T00:27:42", "url": "https://files.pythonhosted.org/packages/36/00/4b1e9741747b14974cf687cd041299f0d361205311d228504e728e31acbd/sequent-3.0.1.tar.gz" } ], "3.0.2": [ { "comment_text": "", "digests": { "md5": "766e32a65c8d1bf30e5af5d153bdee61", "sha256": "57971754038de40efcb55038a536ade3acfa1c9363285e30458f7f046c935c9f" }, "downloads": -1, "filename": "sequent-3.0.2.tar.gz", "has_sig": false, "md5_digest": "766e32a65c8d1bf30e5af5d153bdee61", "packagetype": "sdist", "python_version": "source", "requires_python": null, "size": 23013, "upload_time": "2017-02-15T13:06:10", "url": "https://files.pythonhosted.org/packages/6f/33/e8fc68a7ffa7d15e979135976b94dd7c61b51efae285969712e898038b88/sequent-3.0.2.tar.gz" } ], "3.0.3": [ { "comment_text": "", "digests": { "md5": "a9038bfef31cdb01234005157e9d07bc", "sha256": "2ec872099a924edbb28c94a61efd787d6721158d24c1ce7e5f1acc8e28c07c6c" }, "downloads": -1, "filename": "sequent-3.0.3.tar.gz", "has_sig": false, "md5_digest": "a9038bfef31cdb01234005157e9d07bc", "packagetype": "sdist", "python_version": "source", "requires_python": null, "size": 22990, "upload_time": "2017-02-15T13:14:27", "url": "https://files.pythonhosted.org/packages/a6/ab/50a105a7aa1753a7221242d4f94b4db85924bce0cd352d32a643b3b7b8ec/sequent-3.0.3.tar.gz" } ], "3.0.4": [ { "comment_text": "", "digests": { "md5": "427fd9da21ae426848051a8d903022c5", "sha256": "d720d4682be47b19032ac624ebf2de50cab3e1662bc69260eeb90a4566f51338" }, "downloads": -1, "filename": "sequent-3.0.4.tar.gz", "has_sig": false, "md5_digest": "427fd9da21ae426848051a8d903022c5", "packagetype": "sdist", "python_version": "source", "requires_python": null, "size": 22997, "upload_time": "2017-02-22T15:20:27", "url": "https://files.pythonhosted.org/packages/0a/15/578178ef9d4d0b129591bf545ddbc83163e028b908446f330628f67077d7/sequent-3.0.4.tar.gz" } ], "3.1.0": [ { "comment_text": "", "digests": { "md5": "46a7726012d75721d130dd814d2fe204", "sha256": "96a7e6d00843d39e9c9257d4748f643ebf90c2d701ec79b33b285388aa92814d" }, "downloads": -1, "filename": "sequent-3.1.0.tar.gz", "has_sig": true, "md5_digest": "46a7726012d75721d130dd814d2fe204", "packagetype": "sdist", "python_version": "source", "requires_python": null, "size": 33958, "upload_time": "2018-01-02T01:55:23", "url": "https://files.pythonhosted.org/packages/7f/fd/84494ad86e31016c1a742f5bc62f5625bfac0034a6dcd2399c340441cf09/sequent-3.1.0.tar.gz" } ], "3.1.1": [ { "comment_text": "", "digests": { "md5": "329f8dd001b4d97d7e5b5bde98671105", "sha256": "3b9c4b9c4e3bf5a10121ecd06be1e202ebc74ace5b9a7b584e078071b8bdf3f8" }, "downloads": -1, "filename": "sequent-3.1.1.tar.gz", "has_sig": true, "md5_digest": "329f8dd001b4d97d7e5b5bde98671105", "packagetype": "sdist", "python_version": "source", "requires_python": null, "size": 34245, "upload_time": "2018-01-05T03:56:18", "url": "https://files.pythonhosted.org/packages/a5/31/4ffc1a8120d3298fc2fc8bf2e0ffd5e77dd3eed3b2cc71d62882c65ecea1/sequent-3.1.1.tar.gz" } ], "3.1.2": [ { "comment_text": "", "digests": { "md5": "ad45ca5c98db504792032c32fefdd0a7", "sha256": "81375454f22e4e3c173b079d3349b97442a78511d8ece3cb727104fdb2681aae" }, "downloads": -1, "filename": "sequent-3.1.2.tar.gz", "has_sig": true, "md5_digest": "ad45ca5c98db504792032c32fefdd0a7", "packagetype": "sdist", "python_version": "source", "requires_python": null, "size": 34165, "upload_time": "2018-01-21T00:34:39", "url": "https://files.pythonhosted.org/packages/d1/42/1fbf2fbcfefce50e10ed03fa26c151ad1e646f5a4b0d1bff3e6857fe9d03/sequent-3.1.2.tar.gz" } ], "3.1.3": [ { "comment_text": "", "digests": { "md5": "59054789144d3dcb1c06247c98fe6ed5", "sha256": "a1c5ca05a3ac9f8d142d14c9e7841085f9b57ff3bc994c085ff7ab75feae756e" }, "downloads": -1, "filename": "sequent-3.1.3.tar.gz", "has_sig": true, "md5_digest": "59054789144d3dcb1c06247c98fe6ed5", "packagetype": "sdist", "python_version": "source", "requires_python": null, "size": 33993, "upload_time": "2018-01-21T01:06:24", "url": "https://files.pythonhosted.org/packages/0d/ed/4a7a7aab0a1a356315b926e80b2a344404097126744b3e73b1d8c5f26f6c/sequent-3.1.3.tar.gz" } ], "3.1.4": [ { "comment_text": "", "digests": { "md5": "8f166f52eef7a63d46a12564aab916c0", "sha256": "f94af24b621fbd7479a2494121c9f5823b224ed94b5d275361d97eaa9f4e598e" }, "downloads": -1, "filename": "sequent-3.1.4.tar.gz", "has_sig": true, "md5_digest": "8f166f52eef7a63d46a12564aab916c0", "packagetype": "sdist", "python_version": "source", "requires_python": null, "size": 34048, "upload_time": "2018-01-21T01:11:31", "url": "https://files.pythonhosted.org/packages/26/f7/dadf0c13378a5d14ada73d46a7c75a22b359670083d527dfa51baf5046e7/sequent-3.1.4.tar.gz" } ], "3.1.5": [ { "comment_text": "", "digests": { "md5": "7438cf0003dfeffe89cf464eeaef8c08", "sha256": "fe82bc7744265b16fc7314c6464df39a40e9b2b8577390ca286f0c4e7452292f" }, "downloads": -1, "filename": "sequent-3.1.5.tar.gz", "has_sig": true, "md5_digest": "7438cf0003dfeffe89cf464eeaef8c08", "packagetype": "sdist", "python_version": "source", "requires_python": null, "size": 37159, "upload_time": "2018-01-21T01:22:59", "url": "https://files.pythonhosted.org/packages/85/0e/8e8a66f5d07c8ed55d1ee13f54d55c7a92aa53be984786962b7d563b6036/sequent-3.1.5.tar.gz" } ], "3.1.6": [ { "comment_text": "", "digests": { "md5": "e98ee34de47acca67b9d3b4c9ad84bfd", "sha256": "21b1b6540d0d83386c9401951f669778d0700dabe0061dbe30a3bfd5ecbea1e3" }, "downloads": -1, "filename": "sequent-3.1.6.tar.gz", "has_sig": true, "md5_digest": "e98ee34de47acca67b9d3b4c9ad84bfd", "packagetype": "sdist", "python_version": "source", "requires_python": null, "size": 37165, "upload_time": "2018-01-21T01:35:00", "url": "https://files.pythonhosted.org/packages/d9/60/cfe4fef485d56aced067dbe114093696cc29cd8f0b05f04360d9e8928585/sequent-3.1.6.tar.gz" } ], "3.1.7": [ { "comment_text": "", "digests": { "md5": "a610ae7a40f05499a5388bf794124b9c", "sha256": "502544f41464aa524375013f1574a5a7121eec3f1d33bc0143900b082aab1a38" }, "downloads": -1, "filename": "sequent-3.1.7.tar.gz", "has_sig": true, "md5_digest": "a610ae7a40f05499a5388bf794124b9c", "packagetype": "sdist", "python_version": "source", "requires_python": null, "size": 34118, "upload_time": "2018-02-18T20:12:02", "url": "https://files.pythonhosted.org/packages/50/53/19b51d01c1f62a873871d7f5d505e0a80ef25a75a46651bc0510f658bb43/sequent-3.1.7.tar.gz" } ] }, "urls": [ { "comment_text": "", "digests": { "md5": "a610ae7a40f05499a5388bf794124b9c", "sha256": "502544f41464aa524375013f1574a5a7121eec3f1d33bc0143900b082aab1a38" }, "downloads": -1, "filename": "sequent-3.1.7.tar.gz", "has_sig": true, "md5_digest": "a610ae7a40f05499a5388bf794124b9c", "packagetype": "sdist", "python_version": "source", "requires_python": null, "size": 34118, "upload_time": "2018-02-18T20:12:02", "url": "https://files.pythonhosted.org/packages/50/53/19b51d01c1f62a873871d7f5d505e0a80ef25a75a46651bc0510f658bb43/sequent-3.1.7.tar.gz" } ] }