Customizing DS#
Distributed-Something is a template. It is not fully functional software but is intended to serve as an editable source so that you can quickly and easily implement a distributed workflow for your own Dockerized software.
Examples of sophisticated implementations can be found at Distributed-CellProfiler, Distributed-Fiji, and Distributed-OmeZarrCreator. We have also created a minimal, fully functional example at Distributed-HelloWorld.
Customization overview#
Before starting to customize Distributed-Something code, do some research on your desired implementation.
- Ask how splittable is the function you want to distribute? Distributed-Something only works on “perfectly parallel” tasks, or tasks that do not communicate with each other while running. If the end product you envision cannot easily be split into perfectly parallel tasks, then it may not be a good fit for Distributed-Something. - Scale has a large impact on how splittable your function is. For example, if you want to stitch together a set of images into one larger image, that set that you are stitching is the smallest unit you can make your job. Because jobs must be “perfectly parallel”, you cannot distribute the images any further. If you’re generally working with datasets that only require a few stitching jobs, Distributed-Something may not be a good fit for your general use case. However, if you often work with very large datasets where you need to stitch many sets of images, even though you cannot further parallelize your jobs, distributing stitching tasks with Distributed-Something may still provide a significant savings in time and compute cost. 
- Make or find a Docker of the software you want to distribute. You can find over 1000 scientific softwares already Dockerized at Biocontainers and many open-source softwares provide Docker files within their GitHub repositories. See Implementing Distributed-Something for more details. 
- Figure out how to make your software run from the command line. What parameters do you need to pass to it? Are there optional program parameters that you want to require in your Distributed-Something implementation? What is generic to how you like to run the application and what is different for each job? 
- Think about how you will set up/access your data so that it is batchable/parallelizeable. Because Distributed-Something is so application specific, there are many approaches one can take to parse a dataset into batches that can be parallelized. Implemented examples you can reference are: 
- In Distributed-CellProfiler, we use LoadData.csvs to pass to CellProfiler the exact list of files with their S3 file paths that we want it to access/download for processing. 
- In Distributed-FIJI, we tell it what folder to access and pass upload and download filters for it to select specific files within that folder. 
- In Distributed-OMEZARRCreator, the job unit is always the same (one plate of images) so less flexibility is required and the S3 path and plate name passed in the job file is sufficient. 
Using the Distributed-Something template#
Distributed-Something is a template repository. Read more about Github template repositories and follow the instructions to create your own project repository from the template.
Required customization details#
There are many points at which you will need to customize Distributed-Something for your own implementation; These customization points are summarized below. Files that do not require any customization are not listed.
files/#
exampleFleet.json#
exampleFleet.json does not need to be changed depending on your implementation of Distributed-Something. However, each AWS account running your implementation will need to update the Fleet file with configuration specific to their account as detailed in Step 3: Start Cluster.
exampleJob.json#
exampleJob.json needs to be entirely customized for your implementation of Distributed-Something.
When you submit your jobs in Step 2: Submit Jobs, Distributed-Something adds a job to your SQS queue for each item in groups.
Each job contains the shared variables common to all jobs, listed in the exampleJob.json above the groups key.
These variables are passed to your worker as the message and should include any metadata that may possibly change between runs of your Distributed-Something implementation.
Some common variables used in Job files include:
- input location 
- output location 
- output structure 
- script/pipeline name 
- flags to pass to your program 
worker/#
Dockerfile#
The Dockerfile is used to create the Distributed-Something Docker.
You will need to edit the FROM to point to your own docker.
No further edits to the Dockerfile should be necessary, though advanced users make additional customizations based on the docker they are FROMing.
Additionally, you may remove the section # Install S3FS if your workflow doesn’t require mounting an S3 bucket.
You will still be able to upload and download from an S3 bucket using AWS CLI without mounting it with S3FS.
generic-worker.py#
The majority of code customization for your implementation of Distributed-Something happens in the worker file.
It is responsible for parsing the input message, creating the call command, and figuring out how to determine if your task succeeded or failed.
The generic-worker.py code is thoroughly documented with customization details.
Makefile#
Update user and project to match you and your Distributed-Something implementation, respectively.
Decide on a tag to use.
Note that you an use latest for the tag while your are developing/testing your software (it will continually overwrite) and give it a numbered tag when you have a functional version.
run-worker.sh#
You do not need to make any modifications to run-worker.sh.
You might want to remove 2. MOUNT S3 if your workflow doesn’t require mounting an S3 bucket.
If you change the name of the generic-worker.py , and/or if you plan to change the Python versions, reflect those changes here.
other files#
config.py#
DOCKERHUB_TAG needs to match the user and project set in Makefile.
We recommend adjusting EC2 AND ECS INFORMATION and DOCKER INSTANCE RUNNING ENVIRONMENT variables to reasonable defaults for your Distributed-Something implementation.
Suggestions for determining optimal parameters can be found in Implementing Distributed-Something.
If there are any variables you would like to pass to your program as part of configuration, you can add them at the bottom and they will be passed as system variables to the Docker.
Note that any additional variables added to config.py need to also be added to CONSTANT PATHS IN THE CONTAINER in generic-worker.py.
AWS GENERAL SETTINGS are specific to your account. All other sections are variable specific to each batch/run of your Distributed-Something implementation and will need to be adjusted at each run time.
More configuration information is available in Step 1: Configuration
run.py#
You need to customize the Dashboard creation function by changing ‘start run’ to whatever your run command is. If you have changed anything in config.py, you will need to edit the section on Task Definitions to match.
Optional customization details#
Adding an additional script#
If you want to add an additional script that you will trigger from within your worker.py file, you need to do the following.
See Distributed-OMEZarrCreator for an example of this in action with add_downsampling.py.
- Add the file into your - workerfolder.
- In - Dockerfile, add- COPY your_script_name.py .after line 56 (where similar copy commands are performed for the other scripts).
- In - Dockerfile, install any additional dependencies required by the new script. Follow the format of other install commands (e.g.- RUN python3.8 -m pip install,- RUN apt install -y)
- Call the script from within your - worker.pyscript using a similar format as your main call. e.g.
cmd = (f"python3 your_script_name.py input_to_script --flag-for-script")
printandlog(f"Running your_script_name", logger)
logger.info(cmd)
subp = subprocess.Popen(cmd.split(), stdout=subprocess.PIPE, stderr=subprocess.STDOUT)
monitorAndLog(subp, logger)
