/* * Copyright (c) 2022 Egor Tensin * This file is part of the "cimple" project. * For details, see https://github.com/egor-tensin/cimple. * Distributed under the MIT License. */ #include "worker.h" #include "ci.h" #include "command.h" #include "compiler.h" #include "const.h" #include "event_loop.h" #include "git.h" #include "log.h" #include "net.h" #include "process.h" #include "protocol.h" #include "run_queue.h" #include "signal.h" #include #include #include struct worker { struct settings *settings; int stopping; struct cmd_dispatcher *cmd_dispatcher; struct event_loop *event_loop; int signalfd; struct run *run; }; static struct settings *worker_settings_copy(const struct settings *src) { struct settings *result = malloc(sizeof(*src)); if (!result) { log_errno("malloc"); return NULL; } result->host = strdup(src->host); if (!result->host) { log_errno("strdup"); goto free_result; } result->port = strdup(src->port); if (!result->port) { log_errno("strdup"); goto free_host; } return result; free_host: free((void *)result->host); free_result: free(result); return NULL; } static void worker_settings_destroy(struct settings *settings) { free((void *)settings->port); free((void *)settings->host); free(settings); } static int worker_set_stopping(UNUSED struct event_loop *loop, UNUSED int fd, UNUSED short revents, void *_worker) { struct worker *worker = (struct worker *)_worker; worker->stopping = 1; return 0; } static int worker_handle_cmd_start_run(const struct jsonrpc_request *request, UNUSED struct jsonrpc_response **response, void *_ctx) { struct cmd_conn_ctx *ctx = (struct cmd_conn_ctx *)_ctx; struct worker *worker = (struct worker *)ctx->arg; int ret = 0; ret = request_parse_start_run(request, &worker->run); if (ret < 0) return ret; return ret; } static struct cmd_desc commands[] = { {CMD_START_RUN, worker_handle_cmd_start_run}, }; static const size_t numof_commands = sizeof(commands) / sizeof(commands[0]); int worker_create(struct worker **_worker, const struct settings *settings) { int ret = 0; struct worker *worker = malloc(sizeof(struct worker)); if (!worker) { log_errno("malloc"); return -1; } worker->settings = worker_settings_copy(settings); if (!worker->settings) { ret = -1; goto free; } worker->stopping = 0; ret = cmd_dispatcher_create(&worker->cmd_dispatcher, commands, numof_commands, worker); if (ret < 0) goto free_settings; ret = event_loop_create(&worker->event_loop); if (ret < 0) goto destroy_cmd_dispatcher; ret = signalfd_create_sigterms(); if (ret < 0) goto destroy_event_loop; worker->signalfd = ret; ret = event_loop_add(worker->event_loop, worker->signalfd, POLLIN, worker_set_stopping, worker); if (ret < 0) goto close_signalfd; ret = libgit_init(); if (ret < 0) goto close_signalfd; *_worker = worker; return ret; close_signalfd: signalfd_destroy(worker->signalfd); destroy_event_loop: event_loop_destroy(worker->event_loop); destroy_cmd_dispatcher: cmd_dispatcher_destroy(worker->cmd_dispatcher); free_settings: worker_settings_destroy(worker->settings); free: free(worker); return ret; } void worker_destroy(struct worker *worker) { log("Shutting down\n"); libgit_shutdown(); signalfd_destroy(worker->signalfd); event_loop_destroy(worker->event_loop); cmd_dispatcher_destroy(worker->cmd_dispatcher); worker_settings_destroy(worker->settings); free(worker); } static int worker_do_run(struct worker *worker) { int ret = 0; struct process_output *result = NULL; ret = process_output_create(&result); if (ret < 0) return ret; ret = ci_run_git_repo(run_get_repo_url(worker->run), run_get_repo_rev(worker->run), result); if (ret < 0) { log_err("Run failed with an error\n"); goto free_output; } process_output_dump(result); struct jsonrpc_request *finished_request = NULL; ret = request_create_finished_run(&finished_request, run_get_id(worker->run), result); if (ret < 0) goto free_output; ret = net_connect(worker->settings->host, worker->settings->port); if (ret < 0) goto free_request; int fd = ret; ret = jsonrpc_request_send(finished_request, fd); if (ret < 0) goto close_conn; close_conn: net_close(fd); free_request: jsonrpc_request_destroy(finished_request); free_output: process_output_destroy(result); run_destroy(worker->run); return ret; } static int worker_get_run(struct worker *worker) { int ret = 0, fd = -1; ret = net_connect(worker->settings->host, worker->settings->port); if (ret < 0) return ret; fd = ret; struct jsonrpc_request *new_worker_request = NULL; ret = request_create_new_worker(&new_worker_request); if (ret < 0) goto close; ret = jsonrpc_request_send(new_worker_request, fd); jsonrpc_request_destroy(new_worker_request); if (ret < 0) goto close; ret = event_loop_add_once(worker->event_loop, fd, POLLIN, cmd_dispatcher_handle_event, worker->cmd_dispatcher); if (ret < 0) goto close; log("Waiting for a new command\n"); ret = event_loop_run(worker->event_loop); if (ret < 0) goto close; close: net_close(fd); return ret; } int worker_main(struct worker *worker) { int ret = 0; while (1) { ret = worker_get_run(worker); if (ret < 0) return ret; if (worker->stopping) break; ret = worker_do_run(worker); if (ret < 0) return ret; } return ret; }