Informatics Matters Job Tester ("jote")
=======================================
.. image:: https://badge.fury.io/py/im-jote.svg
:target: https://badge.fury.io/py/im-jote
:alt: PyPI package (latest)
.. image:: https://github.com/InformaticsMatters/squonk2-data-manager-job-tester/actions/workflows/build.yaml/badge.svg
:target: https://github.com/InformaticsMatters/squonk2-data-manager-job-tester/actions/workflows/build.yaml
:alt: Build
.. image:: https://github.com/InformaticsMatters/squonk2-data-manager-job-tester/actions/workflows/publish.yaml/badge.svg
:target: https://github.com/InformaticsMatters/squonk2-data-manager-job-tester/actions/workflows/publish.yaml
:alt: Publish
The **Squonk2 Job Tester** (``jote``) is a Python utility used to run *unit tests*
that are defined in Data Manager *job implementation repositories* against
the job's container image, images that are typically built from the same
repository.
``jote`` is designed to run job implementations in a file-system
environment that replicates what they find when they're run by the Data Manager.
But jobs *are not* running in the same operating-system environment, e.g. they
are not bound by the same processor and memory constraints they'll encounter in
the Data Manager, which runs in `Kubernetes`_.
To use ``jote`` you will need to install ``docker-compose`` (v1 or v2).
A successful test should give you confidence that it *should* work in the
Data Manger but without writing a lot of tests you'll never be completely
confident that it will always run successfully.
``jote`` is a tool we designed to provide *us* with confidence that we can
deploy jobs to a Data Manager instance and know that they're basically fit
for purpose. Jobs that have no tests will not normally be deployed to the
Data Manager.
To use a job in Squonk you need to create at least one **manifest file** and
one **job definition file**. These reside in the ``data-manager``
directory of the repository you're going to test. ``jote`` expects the
default manifest file to be called ``manifest.yaml`` but you can use a
different name and have more than one.
If you want to provide your own Squonk jobs and corresponding
job definitions our **Virtual Screening** repository
(https://github.com/InformaticsMatters/virtual-screening) is a good
place to start. The repository is host to a number of job-based
container images and several *manifests* and *job definition files*.
Here's an example **manifest** from a recent **Virtual Screening** repository::
---
kind: DataManagerManifest
kind-version: '2021.1'
job-definition-files:
- im-virtual-screening.yaml
- rdkit.yaml
- xchem.yaml
Each Manifest must list at least one file. To be included in Squonk every
job must contain at least one test. ``jote`` runs the tests but also ensures
the repository structure is as expected and applies strict rules for the
formatting of the YAML files.
Both ``jote`` and the Data Manager rely on the schemas that can be found
in our **Job Decoder** repository
(https://github.com/InformaticsMatters/data-manager-job-decoder).
Here's a snippet from a job definition file illustrating a
job (``max-min-picker``) that has a test called ``simple-execution``.
The test defines an input option (a file) and some other command options.
The ``checks`` section is used to define the exit criteria of the test.
In this case the container must exit with code ``0`` and the file
``diverse.smi`` must be found in the generated test directory, i.e
it must *exist* and contain ``100`` lines. ``jote`` will fail the test unless
these checks are satisfied::
jobs:
[...]
max-min-picker:
[...]
tests:
simple-execution:
inputs:
inputFile: data/100000.smi
options:
outputFile: diverse.smi
count: 100
checks:
exitCode: 0
outputs:
- name: diverse.smi
checks:
- exists: true
- lineCount: 100
.. _kubernetes: https://kubernetes.io/
Running tests
-------------
Run ``jote`` from the root of a clone of the Data Manager Job implementation
repository that you want to test::
jote
You can display the utility's help with::
jote --help
Jote container network
----------------------
``jote`` tests are executed on the network ``data-manager_jote``. This is
defined in the docker-compose file that it generates to run your tests.
Built-in variables
------------------
Job definition command-expansion provided by the **job decoder**
relies on a number of *built in* variables. Some are provided by the
Data Manager when the job runs under its control
(i.e. ``DM_INSTANCE_DIRECTORY``) others are provided by ``jote`` to simplify
testing.
The set of variables injected into the command expansion by ``jote``
are: -
- ``DM_INSTANCE_DIRECTORY``. Set to the path of the simulated instance
directory created by ``jote``, normally created by the Data Manager
- ``CODE_DIRECTORY``. Set to the root of the repository that you're running
the tests in. This is a convenient variable to locate your out-of-container
nextflow workflow file, which is likely to be in the root of your repository
Ignoring tests
--------------
Occasionally you may want to disable some tests because they need some work
before they're complete. To allow you to continue testing other jobs under
these circumstances you can mark individual tests and have them excluded
by adding an ``ignore`` declaration::
jobs:
[...]
max-min-picker:
[...]
tests:
simple-execution:
ignore:
[...]
You don't have to remove the ``ignore`` declaration to run the test in ``jote``.
If you want to see whether an ignored test now works you can run ``jote``
for specific tests by using ``--test`` and naming the ignored test you want
to run. When a test is named explicitly it is run, regardless of whether
``ignore`` has been set or not.
Test run levels
---------------
Tests can be assigned a ``run-level``. Run-levels are numerical value (1..100)
that can be used to group your tests. You can use the ``run-level``
as an indication of execution time, with short tests having low values and
time-consuming tests with higher values.
By default all tests that have no run-level defined and those with
a run-level of ``1`` are executed. If you set the run-level for longer-running
tests to a higher value, e.g. ``5``, these will be skipped. To run these more
time-consuming tests you specify the run-level when running ``jote``
using ``--run-level 5``.
When you give ``jote`` a run-level only tests up to and including the
level, and those without any run-level, will be run.
You define the run-level in the root block of the job's test specification::
jobs:
[...]
max-min-picker:
[...]
tests:
simple-execution:
run-level: 5
[...]
Test timeouts
-------------
``jote`` lets each test run for 10 minutes before cancelling (and failing) them.
If you expect that your test needs to run for more than 10 minutes you must
use the ``timeout-minutes`` property in the job definition to define your own
test-specific value::
jobs:
[...]
max-min-picker:
[...]
tests:
simple-execution:
timeout-minutes: 120
[...]
You should try and avoid creating too many long-running tests. If you cannot,
consider whether it's a appropriate to use ``run-level`` to avoid ``jote``
running them by default.
Test groups
-----------
Tests are normally executed and the environment torn-down between them.
If you have tests that depend on the results from a prior test you can run
tests as a **group**, which preserves the project directory between the tests.
To run a sequence of test (as a **group**) you need to define a ``test-group``
in your Job Definition file and then refer to that group in your test. Here,
we define a test group called ``experiment-a``, at the top of the
definition file::
test-groups:
- name: experiment-a
We then place a test in that group with a ``run-group`` declaration
in the corresponding test block::
jobs:
max-min-picker:
[...]
tests:
test-a:
run-groups:
- name: experiment-a
ordinal: 1
We need to provide an ``ordinal`` value. This numeric value (from 1 ..N)
puts the test in a specific position in the test sequence. When tests are
placed in a ``run-group`` you have to order your tests, i.e. declare that
``test-a`` follows ``test-b``. This is done with unique ordinals for each
test in the ``run-group``. A test with ordinal ``1`` will run before a test
with ordinal ``2``. Ordinals have to be unique within a ``run-group``.
You can run the tests for a specific group by using the ``--run-group``
option::
jote --run-group experiment-a
Running additional containers (group testing)
---------------------------------------------
Test groups provide an ability to launch additional support containers during
testing. You might want to start a background database for example, that can
be used by tests in your ``test-group``. To take advantage of this feature
you just need to provide a ``docker-compose`` file (in the Job definition
``data-manager`` directory) and name that file in you r``test-groups``
declaration.
Here we declare a docker-compose file called
``docker-compose-experiment-a.yaml``::
test-groups:
- name: experiment-a
compose:
file: docker-compose-experiment-a.yaml
The compose filename must begin ``docker-compose`` and end ``.yaml``.
The compose file is run before any tests in the corresponding test group
have been run and will be stopped after the last test in the group.
The compose file you provide is run in a *detached* state so ``jote`` does
not wait for the containers to start (or initialise). As the first test
in the test group can begin very soon after the compose file is started
you can minimise the risk that your containers are not ready for the tests
by adding a fixed delay between ``jote`` starting the compose file and
running the first test::
test-groups:
- name: experiment-a
compose:
file: docker-compose-experiment-a.yaml
delay-seconds: 10
Nextflow test execution
-----------------------
Job image types can be ``simple`` or ``nextflow``. Simple jobs are executed in
the container image you've built and should behave much the same as they do
when run within the Data Manager. Nextflow jobs on the other hand are executed
using the shell, relying on Docker as the execution run-time for the processes
in your workflow.
Be aware that nextflow tests run by ``jote`` run under different conditions
compared to when it runs under the Data Manager's control. In the Data Manager
nextflow jobs will be executed within a Kubernetes environment. When run by ``jote``
nextflow is expected using the operating system shell. This introduces a
variability that you need to take into account - i.e. under ``jote`` the
nextflow controller runs in the shell, and *are not* executed in the same
environment or under the same memory or processor constraints.
You might need to provide a custom nextflow configuration file
for your tests to run successfully. You do this by adding a ``nextflow-config-file``
declaration in the test. Here, we name the file ``nextflow-test.config``::
jobs:
max-min-picker:
[...]
tests:
simple-load:
nextflow-config-file: nextflow-test.config
[...]
The config file must be located in the Job repository's ``data-manager``
directory.
Prior to running the corresponding test ``jote`` copies it to the
Job's project directory as the file ``nextflow.config`` (a standard file
expected by nextflow).
``jote`` *will not* let you have a nextflow config in your home directory
as any settings found there would be merged with the file ``jote`` writes,
potentially disturbing the execution behaviour.
.. note::
It's your responsibility to install a suitable nextflow that's available
for shell execution. ``jote`` expects to be able to run nextflow when
executing the corresponding ``command`` that's defined in the job
definition.
Installation
============
``jote`` is published on `PyPI`_ and can be installed from there::
pip install im-jote
This is a Python 3 utility, so try to run it from a recent (ideally 3.10)
Python environment.
To use the utility you will need to have installed `Docker`_, `docker-compose`,
and, if you want to test nextflow jobs, `nextflow`_.
.. _PyPI: https://pypi.org/project/im-jote/
.. _Docker: https://docs.docker.com/get-docker/
.. _nextflow: https://www.nextflow.io/
Get in touch
------------
- Report bugs, suggest features or view the source code `on GitHub`_.
.. _on GitHub: https://github.com/informaticsmatters/squonk2-data-manager-job-tester
Raw data
{
"_id": null,
"home_page": "https://github.com/informaticsmatters/squonk2-data-manager-job-tester",
"name": "im-jote",
"maintainer": null,
"docs_url": null,
"requires_python": ">=3.10.0",
"maintainer_email": null,
"keywords": "configuration",
"author": "Alan Christie",
"author_email": "achristie@informaticsmatters.com",
"download_url": "https://files.pythonhosted.org/packages/1a/21/1f976845e9d30dc86123bb3f4544500df413d1f232813745c6ba176dca2a/im_jote-0.10.7.tar.gz",
"platform": "any",
"description": "Informatics Matters Job Tester (\"jote\")\n=======================================\n\n.. image:: https://badge.fury.io/py/im-jote.svg\n :target: https://badge.fury.io/py/im-jote\n :alt: PyPI package (latest)\n\n.. image:: https://github.com/InformaticsMatters/squonk2-data-manager-job-tester/actions/workflows/build.yaml/badge.svg\n :target: https://github.com/InformaticsMatters/squonk2-data-manager-job-tester/actions/workflows/build.yaml\n :alt: Build\n\n.. image:: https://github.com/InformaticsMatters/squonk2-data-manager-job-tester/actions/workflows/publish.yaml/badge.svg\n :target: https://github.com/InformaticsMatters/squonk2-data-manager-job-tester/actions/workflows/publish.yaml\n :alt: Publish\n\nThe **Squonk2 Job Tester** (``jote``) is a Python utility used to run *unit tests*\nthat are defined in Data Manager *job implementation repositories* against\nthe job's container image, images that are typically built from the same\nrepository.\n\n``jote`` is designed to run job implementations in a file-system\nenvironment that replicates what they find when they're run by the Data Manager.\nBut jobs *are not* running in the same operating-system environment, e.g. they\nare not bound by the same processor and memory constraints they'll encounter in\nthe Data Manager, which runs in `Kubernetes`_.\n\nTo use ``jote`` you will need to install ``docker-compose`` (v1 or v2).\n\nA successful test should give you confidence that it *should* work in the\nData Manger but without writing a lot of tests you'll never be completely\nconfident that it will always run successfully.\n\n``jote`` is a tool we designed to provide *us* with confidence that we can\ndeploy jobs to a Data Manager instance and know that they're basically fit\nfor purpose. Jobs that have no tests will not normally be deployed to the\nData Manager.\n\nTo use a job in Squonk you need to create at least one **manifest file** and\none **job definition file**. These reside in the ``data-manager``\ndirectory of the repository you're going to test. ``jote`` expects the\ndefault manifest file to be called ``manifest.yaml`` but you can use a\ndifferent name and have more than one.\n\n If you want to provide your own Squonk jobs and corresponding\n job definitions our **Virtual Screening** repository\n (https://github.com/InformaticsMatters/virtual-screening) is a good\n place to start. The repository is host to a number of job-based\n container images and several *manifests* and *job definition files*.\n\nHere's an example **manifest** from a recent **Virtual Screening** repository::\n\n ---\n kind: DataManagerManifest\n kind-version: '2021.1'\n\n job-definition-files:\n - im-virtual-screening.yaml\n - rdkit.yaml\n - xchem.yaml\n\nEach Manifest must list at least one file. To be included in Squonk every\njob must contain at least one test. ``jote`` runs the tests but also ensures\nthe repository structure is as expected and applies strict rules for the\nformatting of the YAML files.\n\nBoth ``jote`` and the Data Manager rely on the schemas that can be found\nin our **Job Decoder** repository\n(https://github.com/InformaticsMatters/data-manager-job-decoder).\n\nHere's a snippet from a job definition file illustrating a\njob (``max-min-picker``) that has a test called ``simple-execution``.\n\nThe test defines an input option (a file) and some other command options.\nThe ``checks`` section is used to define the exit criteria of the test.\nIn this case the container must exit with code ``0`` and the file\n``diverse.smi`` must be found in the generated test directory, i.e\nit must *exist* and contain ``100`` lines. ``jote`` will fail the test unless\nthese checks are satisfied::\n\n jobs:\n [...]\n max-min-picker:\n [...]\n tests:\n simple-execution:\n inputs:\n inputFile: data/100000.smi\n options:\n outputFile: diverse.smi\n count: 100\n checks:\n exitCode: 0\n outputs:\n - name: diverse.smi\n checks:\n - exists: true\n - lineCount: 100\n\n.. _kubernetes: https://kubernetes.io/\n\nRunning tests\n-------------\n\nRun ``jote`` from the root of a clone of the Data Manager Job implementation\nrepository that you want to test::\n\n jote\n\nYou can display the utility's help with::\n\n jote --help\n\nJote container network\n----------------------\n\n``jote`` tests are executed on the network ``data-manager_jote``. This is\ndefined in the docker-compose file that it generates to run your tests.\n\nBuilt-in variables\n------------------\n\nJob definition command-expansion provided by the **job decoder**\nrelies on a number of *built in* variables. Some are provided by the\nData Manager when the job runs under its control\n(i.e. ``DM_INSTANCE_DIRECTORY``) others are provided by ``jote`` to simplify\ntesting.\n\nThe set of variables injected into the command expansion by ``jote``\nare: -\n\n- ``DM_INSTANCE_DIRECTORY``. Set to the path of the simulated instance\n directory created by ``jote``, normally created by the Data Manager\n- ``CODE_DIRECTORY``. Set to the root of the repository that you're running\n the tests in. This is a convenient variable to locate your out-of-container\n nextflow workflow file, which is likely to be in the root of your repository\n\nIgnoring tests\n--------------\n\nOccasionally you may want to disable some tests because they need some work\nbefore they're complete. To allow you to continue testing other jobs under\nthese circumstances you can mark individual tests and have them excluded\nby adding an ``ignore`` declaration::\n\n jobs:\n [...]\n max-min-picker:\n [...]\n tests:\n simple-execution:\n ignore:\n [...]\n\nYou don't have to remove the ``ignore`` declaration to run the test in ``jote``.\nIf you want to see whether an ignored test now works you can run ``jote``\nfor specific tests by using ``--test`` and naming the ignored test you want\nto run. When a test is named explicitly it is run, regardless of whether\n``ignore`` has been set or not.\n\nTest run levels\n---------------\n\nTests can be assigned a ``run-level``. Run-levels are numerical value (1..100)\nthat can be used to group your tests. You can use the ``run-level``\nas an indication of execution time, with short tests having low values and\ntime-consuming tests with higher values.\n\nBy default all tests that have no run-level defined and those with\na run-level of ``1`` are executed. If you set the run-level for longer-running\ntests to a higher value, e.g. ``5``, these will be skipped. To run these more\ntime-consuming tests you specify the run-level when running ``jote``\nusing ``--run-level 5``.\n\n When you give ``jote`` a run-level only tests up to and including the\n level, and those without any run-level, will be run.\n\nYou define the run-level in the root block of the job's test specification::\n\n jobs:\n [...]\n max-min-picker:\n [...]\n tests:\n simple-execution:\n run-level: 5\n [...]\n\nTest timeouts\n-------------\n\n``jote`` lets each test run for 10 minutes before cancelling (and failing) them.\nIf you expect that your test needs to run for more than 10 minutes you must\nuse the ``timeout-minutes`` property in the job definition to define your own\ntest-specific value::\n\n jobs:\n [...]\n max-min-picker:\n [...]\n tests:\n simple-execution:\n timeout-minutes: 120\n [...]\n\nYou should try and avoid creating too many long-running tests. If you cannot,\nconsider whether it's a appropriate to use ``run-level`` to avoid ``jote``\nrunning them by default.\n\nTest groups\n-----------\n\nTests are normally executed and the environment torn-down between them.\nIf you have tests that depend on the results from a prior test you can run\ntests as a **group**, which preserves the project directory between the tests.\n\nTo run a sequence of test (as a **group**) you need to define a ``test-group``\nin your Job Definition file and then refer to that group in your test. Here,\nwe define a test group called ``experiment-a``, at the top of the\ndefinition file::\n\n test-groups:\n - name: experiment-a\n\n\nWe then place a test in that group with a ``run-group`` declaration\nin the corresponding test block::\n\n jobs:\n max-min-picker:\n [...]\n tests:\n test-a:\n run-groups:\n - name: experiment-a\n ordinal: 1\n\nWe need to provide an ``ordinal`` value. This numeric value (from 1 ..N)\nputs the test in a specific position in the test sequence. When tests are\nplaced in a ``run-group`` you have to order your tests, i.e. declare that\n``test-a`` follows ``test-b``. This is done with unique ordinals for each\ntest in the ``run-group``. A test with ordinal ``1`` will run before a test\nwith ordinal ``2``. Ordinals have to be unique within a ``run-group``.\n\nYou can run the tests for a specific group by using the ``--run-group``\noption::\n\n jote --run-group experiment-a\n\nRunning additional containers (group testing)\n---------------------------------------------\n\nTest groups provide an ability to launch additional support containers during\ntesting. You might want to start a background database for example, that can\nbe used by tests in your ``test-group``. To take advantage of this feature\nyou just need to provide a ``docker-compose`` file (in the Job definition\n``data-manager`` directory) and name that file in you r``test-groups``\ndeclaration.\n\nHere we declare a docker-compose file called\n``docker-compose-experiment-a.yaml``::\n\n test-groups:\n - name: experiment-a\n compose:\n file: docker-compose-experiment-a.yaml\n\nThe compose filename must begin ``docker-compose`` and end ``.yaml``.\n\nThe compose file is run before any tests in the corresponding test group\nhave been run and will be stopped after the last test in the group.\n\nThe compose file you provide is run in a *detached* state so ``jote`` does\nnot wait for the containers to start (or initialise). As the first test\nin the test group can begin very soon after the compose file is started\nyou can minimise the risk that your containers are not ready for the tests\nby adding a fixed delay between ``jote`` starting the compose file and\nrunning the first test::\n\n test-groups:\n - name: experiment-a\n compose:\n file: docker-compose-experiment-a.yaml\n delay-seconds: 10\n\nNextflow test execution\n-----------------------\n\nJob image types can be ``simple`` or ``nextflow``. Simple jobs are executed in\nthe container image you've built and should behave much the same as they do\nwhen run within the Data Manager. Nextflow jobs on the other hand are executed\nusing the shell, relying on Docker as the execution run-time for the processes\nin your workflow.\n\nBe aware that nextflow tests run by ``jote`` run under different conditions\ncompared to when it runs under the Data Manager's control. In the Data Manager\nnextflow jobs will be executed within a Kubernetes environment. When run by ``jote``\nnextflow is expected using the operating system shell. This introduces a\nvariability that you need to take into account - i.e. under ``jote`` the\nnextflow controller runs in the shell, and *are not* executed in the same\nenvironment or under the same memory or processor constraints.\n\nYou might need to provide a custom nextflow configuration file\nfor your tests to run successfully. You do this by adding a ``nextflow-config-file``\ndeclaration in the test. Here, we name the file ``nextflow-test.config``::\n\n jobs:\n max-min-picker:\n [...]\n tests:\n simple-load:\n nextflow-config-file: nextflow-test.config\n [...]\n\nThe config file must be located in the Job repository's ``data-manager``\ndirectory.\n\nPrior to running the corresponding test ``jote`` copies it to the\nJob's project directory as the file ``nextflow.config`` (a standard file\nexpected by nextflow).\n\n``jote`` *will not* let you have a nextflow config in your home directory\nas any settings found there would be merged with the file ``jote`` writes,\npotentially disturbing the execution behaviour.\n\n.. note::\n It's your responsibility to install a suitable nextflow that's available\n for shell execution. ``jote`` expects to be able to run nextflow when\n executing the corresponding ``command`` that's defined in the job\n definition.\n\nInstallation\n============\n\n``jote`` is published on `PyPI`_ and can be installed from there::\n\n pip install im-jote\n\nThis is a Python 3 utility, so try to run it from a recent (ideally 3.10)\nPython environment.\n\nTo use the utility you will need to have installed `Docker`_, `docker-compose`,\n and, if you want to test nextflow jobs, `nextflow`_.\n\n.. _PyPI: https://pypi.org/project/im-jote/\n.. _Docker: https://docs.docker.com/get-docker/\n.. _nextflow: https://www.nextflow.io/\n\nGet in touch\n------------\n\n- Report bugs, suggest features or view the source code `on GitHub`_.\n\n.. _on GitHub: https://github.com/informaticsmatters/squonk2-data-manager-job-tester\n",
"bugtrack_url": null,
"license": "MIT",
"summary": "The IM Data Manager Job Tester (jote)",
"version": "0.10.7",
"project_urls": {
"Homepage": "https://github.com/informaticsmatters/squonk2-data-manager-job-tester"
},
"split_keywords": [
"configuration"
],
"urls": [
{
"comment_text": "",
"digests": {
"blake2b_256": "ac3039a65bbaa5ac83bd9e2b49ea68830f55a8d70a6713677cadf4aa7e508246",
"md5": "b4b5777c559bc03736e8265e3527405c",
"sha256": "35d1352ffe8f482b34b8e8d1e69a028f866bb9a648a95acf1df64cecec45d825"
},
"downloads": -1,
"filename": "im_jote-0.10.7-py3-none-any.whl",
"has_sig": false,
"md5_digest": "b4b5777c559bc03736e8265e3527405c",
"packagetype": "bdist_wheel",
"python_version": "py3",
"requires_python": ">=3.10.0",
"size": 25828,
"upload_time": "2024-07-31T08:49:59",
"upload_time_iso_8601": "2024-07-31T08:49:59.211185Z",
"url": "https://files.pythonhosted.org/packages/ac/30/39a65bbaa5ac83bd9e2b49ea68830f55a8d70a6713677cadf4aa7e508246/im_jote-0.10.7-py3-none-any.whl",
"yanked": false,
"yanked_reason": null
},
{
"comment_text": "",
"digests": {
"blake2b_256": "1a211f976845e9d30dc86123bb3f4544500df413d1f232813745c6ba176dca2a",
"md5": "09fbc33ba0e7cc1c75c680a77d3afa93",
"sha256": "ea2eaaead4ed9fb83934e4ce7f516e66c16c1cfb2cd0d4c096704ac6f47450cc"
},
"downloads": -1,
"filename": "im_jote-0.10.7.tar.gz",
"has_sig": false,
"md5_digest": "09fbc33ba0e7cc1c75c680a77d3afa93",
"packagetype": "sdist",
"python_version": "source",
"requires_python": ">=3.10.0",
"size": 25352,
"upload_time": "2024-07-31T08:50:22",
"upload_time_iso_8601": "2024-07-31T08:50:22.836931Z",
"url": "https://files.pythonhosted.org/packages/1a/21/1f976845e9d30dc86123bb3f4544500df413d1f232813745c6ba176dca2a/im_jote-0.10.7.tar.gz",
"yanked": false,
"yanked_reason": null
}
],
"upload_time": "2024-07-31 08:50:22",
"github": true,
"gitlab": false,
"bitbucket": false,
"codeberg": false,
"github_user": "informaticsmatters",
"github_project": "squonk2-data-manager-job-tester",
"travis_ci": false,
"coveralls": false,
"github_actions": true,
"requirements": [],
"lcname": "im-jote"
}