nopaque/app/scheduler_functions.py

79 lines
3.0 KiB
Python
Raw Normal View History

from datetime import datetime
from . import create_minimal_app, db
from .models import Job
2019-07-17 13:34:20 +02:00
import docker
import json
2019-08-06 14:54:00 +02:00
import os
2019-07-17 13:34:20 +02:00
2019-08-13 14:27:02 +02:00
def checkout_jobs():
app = create_minimal_app(os.getenv('FLASK_CONFIG') or 'default')
app.app_context().push()
client = docker.from_env()
jobs = db.session.query(Job)
submitted_jobs = jobs.filter_by(status='submitted').all()
foo_jobs = jobs.filter(Job.status != 'complete',
Job.status != 'failed',
Job.status != 'submitted').all()
for job in submitted_jobs:
_command = (job.service
2019-08-06 14:54:00 +02:00
+ ' -i /files'
+ ' -o /files/output'
+ ' ' + ' '.join(json.loads(job.service_args)))
_constraints = ['node.role==worker']
_image = 'gitlab.ub.uni-bielefeld.de:4567/sfb1288inf/{}:{}'.format(
job.service,
2019-08-09 11:48:43 +02:00
job.service_version
)
_labels = {'service': job.service}
2019-08-06 14:54:00 +02:00
_mounts = [os.path.join('/home/compute/mnt/opaque',
str(job.user_id),
'jobs',
str(job.id))
+ ':/files:rw']
2019-08-09 09:13:24 +02:00
_name = str(job.id)
2019-07-17 13:34:20 +02:00
'''
' The Docker SDK for Python expects the cpu_reservation value to be
' scaled to nanos (10^9). Because the job object contains unscaled
' (10^0) values, it must be conveted.
'
' While the cpu_reservation value has to be in nanos, the
2019-07-31 09:10:37 +02:00
' mem_reservation value must be presented in an unscaled form
' (intuitive right?). Bacause the job object provides the memory value
' in megabytes, it is also necessary to convert the value.
2019-07-17 13:34:20 +02:00
'''
_resources = docker.types.Resources(
2019-08-09 11:48:43 +02:00
cpu_reservation=job.n_cores * (10 ** 9),
mem_reservation=job.mem_mb * (10 ** 6)
)
_restart_policy = docker.types.RestartPolicy(condition='none')
2019-07-17 13:34:20 +02:00
'''
' Create the service with the prepared values.
'
' Note: A service reserves hardware ressources. In case no worker node
' has the required ressources available (not reserved), the
' service gets queued by the Docker engine until a node is able
' to meet the requirements.
2019-07-17 13:34:20 +02:00
'''
service = client.services.create(
_image,
command=_command,
constraints=_constraints,
labels=_labels,
mounts=_mounts,
2019-08-08 12:01:55 +02:00
name=_name,
resources=_resources,
restart_policy=_restart_policy
)
job.status = 'scheduled'
for job in foo_jobs:
'''
' TODO: Handle service not found error.
'''
service = client.services.get(str(job.id))
job.status = service.tasks()[0].get('Status').get('State')
if job.status == 'complete' or job.status == 'failed':
job.end_date = datetime.utcnow()
service.remove()
db.session.commit()