boinc/dcapi/examples/message-example/master.c

185 lines
4.2 KiB
C
Raw Normal View History

/**************************************************
* Message-examle Master program
*
* Created by: Gabor Vida
*
* The aim of the program is to test message passing between the master and
* the client in a Public Resource Computing environment. The
* program follows the client server model, therefore having a server-side and
* a client-side part.
*
* This is the server-side part of the program. It creates a WorkUnit (WU) for
* a Boinc project using the DC Application Programming Interface. The program
* takes a text file (input.txt) and submits it into the PRC-infrastructure,
* where it get processed by a client. After the submittion, the master waits
* for a user interaction (press ENTER), when the client downloaded the WU.
* After this, the master sends a sample message to the client, which is
* running around 5 mins on the client. If everything goes well, the client
* will get the message, and sends back a reply. After the client has returned
* the results of the WU, the program assimilates it.
*
*****************************************************/
#ifdef HAVE_CONFIG_H
#include <config.h>
#endif
#include <dc.h>
#include <stdlib.h>
#include <getopt.h>
#include <string.h>
#include <limits.h>
#include <stdio.h>
#include <errno.h>
#include "common.h"
/* Command line options */
static const struct option longopts[] =
{
{ "config", required_argument, NULL, 'c' },
{ "help", no_argument, NULL, 'h' },
{ NULL }
};
static int do_exit;
static void process_result(DC_Workunit *wu, DC_Result *result)
{
char *tag;
/* Extract our own private identifier */
tag = DC_getWUTag(wu);
DC_log(LOG_NOTICE, "Work unit %s has completed", tag);
/* We no longer need the work unit */
DC_destroyWU(wu);
free(tag);
/* This will cause the main loop to quit */
do_exit = 1;
}
static void process_message(DC_Workunit *wu, const char *message)
{
char *tag;
tag = DC_getWUTag(wu);
DC_log(LOG_NOTICE, "Message '%s' has arrived from work unit %s", message, tag);
free(tag);
}
static DC_Workunit *create_work(void)
{
DC_Workunit *wu;
wu = DC_createWU("message-example", NULL, 0, "message-test");
if (!wu)
{
DC_log(LOG_ERR, "Work unit creation has failed");
exit(1);
}
if (DC_addWUInput(wu, INPUT_LABEL, "input.txt", DC_FILE_PERSISTENT))
{
DC_log(LOG_ERR, "Failed to register WU input file");
exit(1);
}
if (DC_addWUOutput(wu, OUTPUT_LABEL))
{
DC_log(LOG_ERR, "Failed to register WU output file");
exit(1);
}
if (DC_submitWU(wu))
{
DC_log(LOG_ERR, "Failed to submit WU");
exit(1);
}
return wu;
}
static void print_help(const char *prog) __attribute__((noreturn));
static void print_help(const char *prog)
{
const char *p;
/* Strip the path component if present */
p = strrchr(prog, '/');
if (p)
prog = p;
printf("Usage: %s {-c|--config} <config file>\n", prog);
printf("Available options:\n");
printf("\t-c <file>, --config <file>\tUse the specified config. file\n");
printf("\t-h, --help\t\t\tThis help text\n");
exit(0);
}
int main(int argc, char *argv[])
{
char *config_file = NULL;
DC_Workunit *wu;
int c;
while ((c = getopt_long(argc, argv, "ch", longopts, NULL)) != -1)
{
switch (c)
{
case 'c':
config_file = optarg;
break;
case 'h':
print_help(argv[0]);
break;
default:
exit(1);
}
}
if (optind != argc)
{
fprintf(stderr, "Extra arguments on the command line\n");
exit(1);
}
/* Specifying the config file is mandatory, otherwise the master can't
* run as a BOINC daemon */
if (!config_file)
{
config_file = strdup("dc-api.conf");
fprintf(stdout, "You didn't specified the config file, use 'dc-api.conf'\n");
}
/* Initialize the DC-API */
if (DC_initMaster(config_file))
{
fprintf(stderr, "Master: DC_initMaster failed, exiting.\n");
exit(1);
}
/* Set the callbacks for receiving messages & the result */
DC_setMasterCb(process_result, NULL, process_message);
wu = create_work();
DC_log(LOG_NOTICE, "Master: Work unit has been created. Waiting "
"for result/message");
/* Wait for the work units to finish */
while (!do_exit)
{
DC_processMasterEvents(60);
/* The result processing destroys the WU, so make sure we
* only try to send a message if the WU is still alive */
if (!do_exit)
DC_sendWUMessage(wu, "What's up?");
}
return 0;
}