aboutsummaryrefslogtreecommitdiff
path: root/gerboweb/job_processor.py
blob: 6c54d644fa56b62bfef5698442f517f686228c7c (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
import os
import sys
import signal
import subprocess
import logging
import itertools
import tempfile
from pathlib import Path

from job_queue import JobQueue

def run_resvg(*args):
        if 'RESVG' in os.environ:
            subprocess.run([os.environ['RESVG'], *args], check=True)

        else:
            # By default, try four options:
            for candidate in [
                    # somewhere in $PATH
                    'resvg',

                    # in user-local cargo installation
                    Path.home() / '.cargo' / 'bin' / 'resvg',

                    # somewhere in $PATH
                    'wasi-resvg',

                    # in user-local pip installation
                    Path.home() / '.local' / 'bin' / 'wasi-resvg',

                    # next to our current python interpreter (e.g. in virtualenv)
                    str(Path(sys.executable).parent / 'resvg'),
                    str(Path(sys.executable).parent / 'wasi-resvg') ]:

                try:
                    subprocess.run([candidate, *args], check=True)
                    print('used svg-flatten at', candidate)
                    break

                except (FileNotFoundError, ModuleNotFoundError):
                    continue

            else:
                raise SystemError('svg-flatten executable not found')

def process_job(job_queue):
    logging.debug('Checking for jobs')
    for job in job_queue.job_iter('render'):
        logging.info(f'Processing {job.type} job {job.id} session {job["session_id"]} from {job.client} submitted {job.created}')
        with job:
            try:
                with tempfile.NamedTemporaryFile(suffix='.svg') as svg:
                    subprocess.run(['python3', '-m', 'gerbonara', '--top', job['infile'], svg.name], check=True)
                    run_resvg('--dpi', '300', svg.name, job['preview_top_out'])
                with tempfile.NamedTemporaryFile(suffix='.svg') as svg:
                    subprocess.run(['python3', '-m', 'gerbonara', '--bottom', job['infile'], svg.name], check=True)
                    run_resvg('--dpi', '300', svg.name, job['preview_bottom_out'])
                subprocess.run(['python3', '-m', 'gerbolyze', 'template', '--top', job['infile'], job['template_top_out']], check=True)
                subprocess.run(['python3', '-m', 'gerbolyze', 'template', '--bottom', job['infile'], job['template_bottom_out']], check=True)
                logging.info(f'Finishied processing {job.type} job {job.id}')
                job.result = True
            except:
                logging.exception('Error during job processing')
                job.result = False

    for job in job_queue.job_iter('vector'):
        logging.info(f'Processing {job.type} job {job.id} session {job["session_id"]} from {job.client} submitted {job.created}')
        with job:
            try:
                subprocess.run(['python3', '-m', 'gerbolyze', 'paste', job['gerber_in'], job['overlay'], job['gerber_out']], check=True)
                logging.info(f'Finishied processing {job.type} job {job.id}')
                job.result = True
            except:
                logging.exception('Error during job processing')
                job.result = False

if __name__ == '__main__':
    import argparse
    parser = argparse.ArgumentParser()
    parser.add_argument('queue', help='job queue sqlite3 database file')
    parser.add_argument('--loglevel', '-l', default='info')
    args = parser.parse_args()

    numeric_level = getattr(logging, args.loglevel.upper(), None)
    if not isinstance(numeric_level, int):
        raise ValueError('Invalid log level: %s' % loglevel)
    logging.basicConfig(level=numeric_level)

    job_queue = JobQueue(args.queue)
    print('Job processor online')
    
    signal.signal(signal.SIGALRM, lambda *args: None) # Ignore incoming alarm signals while processing jobs
    signal.setitimer(signal.ITIMER_REAL, 0.001, 1)
    while signal.sigwait([signal.SIGALRM, signal.SIGINT]) == signal.SIGALRM:
        process_job(job_queue)
    logging.info('Caught SIGINT. Exiting.')