2012-02-16 14:05:28 +00:00
|
|
|
/*
|
|
|
|
Serval Distributed Numbering Architecture (DNA)
|
|
|
|
Copyright (C) 2010 Paul Gardner-Stephen
|
|
|
|
|
|
|
|
This program is free software; you can redistribute it and/or
|
|
|
|
modify it under the terms of the GNU General Public License
|
|
|
|
as published by the Free Software Foundation; either version 2
|
|
|
|
of the License, or (at your option) any later version.
|
|
|
|
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
GNU General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
along with this program; if not, write to the Free Software
|
|
|
|
Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "mphlr.h"
|
|
|
|
|
|
|
|
typedef struct command_line_option {
|
|
|
|
int (*function)(int argc,char **argv,struct command_line_option *o);
|
|
|
|
char *words[32]; // 32 words should be plenty!
|
|
|
|
unsigned long long flags;
|
|
|
|
#define CLIFLAG_NONOVERLAY (1<<0) /* Uses a legacy IPv4 DNA call instead of overlay mnetwork */
|
|
|
|
#define CLIFLAG_STANDALONE (1<<1) /* Cannot be issued to a running instance */
|
|
|
|
char *description; // describe this invocation
|
|
|
|
} command_line_option;
|
|
|
|
|
|
|
|
extern command_line_option command_line_options[];
|
|
|
|
|
2012-02-23 01:25:39 +00:00
|
|
|
int servalNodeRunning(int *pid,char *instancepath)
|
|
|
|
{
|
|
|
|
char filename[1024];
|
|
|
|
char line[1024];
|
|
|
|
if (!instancepath) instancepath=DEFAULT_INSTANCE_PATH;
|
|
|
|
|
|
|
|
int running=0;
|
2012-02-23 02:08:19 +00:00
|
|
|
snprintf(filename,1023,"%s/serval.pid",instancepath); filename[1023]=0;
|
2012-02-23 01:25:39 +00:00
|
|
|
FILE *f=fopen(filename,"r");
|
|
|
|
if (f) {
|
|
|
|
line[0]=0; fgets(line,1024,f);
|
|
|
|
*pid = strtoll(line,NULL,10);
|
|
|
|
running=*pid;
|
|
|
|
if (running) {
|
|
|
|
/* Check that process is really running.
|
|
|
|
Some systems don't have /proc (including mac),
|
|
|
|
so we need to find out some otherway.*/
|
|
|
|
running=0;
|
|
|
|
WHY("need to test if process running");
|
|
|
|
}
|
|
|
|
fclose(f);
|
2012-02-23 02:08:19 +00:00
|
|
|
}
|
2012-02-23 01:25:39 +00:00
|
|
|
|
|
|
|
return running;
|
|
|
|
}
|
|
|
|
|
2012-02-16 14:05:28 +00:00
|
|
|
int cli_usage() {
|
|
|
|
fprintf(stderr,"\nServal Mesh version <version>.\n");
|
|
|
|
fprintf(stderr,"Usage:\n");
|
|
|
|
int i,j;
|
|
|
|
for(i=0;command_line_options[i].function;i++)
|
|
|
|
{
|
|
|
|
for(j=0;command_line_options[i].words[j];j++)
|
|
|
|
fprintf(stderr," %s",command_line_options[i].words[j]);
|
|
|
|
fprintf(stderr,"\n %s\n",command_line_options[i].description);
|
|
|
|
}
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2012-02-16 14:36:08 +00:00
|
|
|
char *cli_arg(int argc,char **argv,command_line_option *o,
|
|
|
|
char *argname,char *defaultvalue)
|
|
|
|
{
|
|
|
|
int arglen=strlen(argname)+2;
|
|
|
|
int i;
|
|
|
|
for(i=0;o->words[i];i++)
|
|
|
|
if ((strlen(o->words[i])==arglen)
|
|
|
|
&&(o->words[i][0]=='<')
|
|
|
|
&&(o->words[i][arglen-1]=='>')
|
|
|
|
&&(!strncasecmp(&o->words[i][1],argname,arglen-2)))
|
|
|
|
{
|
|
|
|
/* Found the arg, so return the corresponding argument */
|
|
|
|
return argv[i];
|
|
|
|
}
|
|
|
|
|
|
|
|
/* No matching argument was found, so return default value.
|
|
|
|
It might seem that this should never happen, but it can because more than
|
|
|
|
one version of a command line optiom may exist, one with a given argument
|
|
|
|
and another without, and allowing a default value means we can have a single
|
|
|
|
function handle both in a fairly simple manner. */
|
|
|
|
return defaultvalue;
|
|
|
|
}
|
|
|
|
|
2012-02-16 14:05:28 +00:00
|
|
|
/* args[] excludes command name (unless hardlinks are used to use first words
|
|
|
|
of command sequences as alternate names of the command. */
|
|
|
|
int parseCommandLine(int argc, char **args)
|
|
|
|
{
|
|
|
|
int i,j;
|
|
|
|
int ambiguous=0;
|
|
|
|
int cli_call=-1;
|
2012-02-23 02:08:19 +00:00
|
|
|
|
2012-02-16 14:05:28 +00:00
|
|
|
for(i=0;command_line_options[i].function;i++)
|
|
|
|
{
|
|
|
|
for(j=0;(j<argc)&&command_line_options[i].words[j];j++)
|
|
|
|
if ((command_line_options[i].words[j][0]!='<')&&
|
|
|
|
strcasecmp(command_line_options[i].words[j],args[j])) {
|
|
|
|
/* Words don't match, and word is not a place-holder for an argument,
|
|
|
|
so it isn't this command line call. */
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((j==argc)&&(!command_line_options[i].words[j])) {
|
|
|
|
/* We used up all words in args and command line call sequence, so we have
|
|
|
|
a match. If we have multiple matches, then note that the call is
|
|
|
|
ambiguous. */
|
|
|
|
if (cli_call>=0) ambiguous++;
|
|
|
|
if (ambiguous==1) {
|
|
|
|
fprintf(stderr,"Ambiguous command line call:\n ");
|
|
|
|
for(j=0;j<argc;j++) fprintf(stderr," %s",args[j]);
|
|
|
|
fprintf(stderr,"\nMatches the following known command line calls:\n");
|
|
|
|
}
|
|
|
|
if (ambiguous) {
|
|
|
|
fprintf(stderr," ");
|
|
|
|
for(j=0;j<argc;j++) fprintf(stderr," %s",command_line_options[i].words[j]);
|
|
|
|
fprintf(stderr,"\n");
|
|
|
|
}
|
|
|
|
cli_call=i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Don't process ambiguous calls */
|
|
|
|
if (ambiguous) return -1;
|
|
|
|
/* Complain if we found no matching calls */
|
|
|
|
if (cli_call<0) return cli_usage();
|
|
|
|
|
|
|
|
/* Otherwise, make call */
|
2012-02-23 02:08:19 +00:00
|
|
|
return command_line_options[cli_call].function(argc,args,
|
|
|
|
&command_line_options[cli_call]);
|
2012-02-16 14:05:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int app_dna_lookup(int argc,char **argv,struct command_line_option *o)
|
|
|
|
{
|
2012-02-16 14:36:08 +00:00
|
|
|
return WHY("Not implemented");
|
|
|
|
}
|
|
|
|
|
|
|
|
int app_server_start(int argc,char **argv,struct command_line_option *o)
|
|
|
|
{
|
|
|
|
return WHY("Not implemented");
|
|
|
|
}
|
|
|
|
|
|
|
|
int app_server_stop(int argc,char **argv,struct command_line_option *o)
|
|
|
|
{
|
|
|
|
return WHY("Not implemented");
|
|
|
|
}
|
|
|
|
|
|
|
|
int app_server_status(int argc,char **argv,struct command_line_option *o)
|
|
|
|
{
|
|
|
|
char *instancepath
|
|
|
|
=cli_arg(argc,argv,o,"instance path",getenv("SERVALINSTANCE_PATH"));
|
|
|
|
if (!instancepath) instancepath=DEFAULT_INSTANCE_PATH;
|
|
|
|
|
2012-02-23 01:25:39 +00:00
|
|
|
char filename[1024];
|
|
|
|
char line[1024];
|
|
|
|
FILE *f;
|
|
|
|
|
|
|
|
/* Display configuration information */
|
|
|
|
snprintf(filename,1023,"%s/serval.conf",instancepath); filename[1023]=0;
|
|
|
|
f=fopen(filename,"r");
|
|
|
|
if (f) {
|
|
|
|
line[0]=0; fgets(line,1024,f);
|
|
|
|
printf("\nServal Mesh configuration:\n");
|
|
|
|
while(line[0]) {
|
|
|
|
printf(" %s",line);
|
|
|
|
line[0]=0; fgets(line,1024,f);
|
|
|
|
}
|
|
|
|
fclose(f);
|
|
|
|
}
|
|
|
|
/* Display running status of daemon from serval.pid file */
|
|
|
|
int pid=-1;
|
|
|
|
int running = servalNodeRunning(&pid,instancepath);
|
|
|
|
|
2012-02-23 02:08:19 +00:00
|
|
|
printf("For Serval Mesh instance %s:\n",instancepath);
|
2012-02-23 01:25:39 +00:00
|
|
|
if (running)
|
2012-02-23 02:08:19 +00:00
|
|
|
printf(" Serval mesh process is running (pid=%d)\n",pid);
|
|
|
|
else
|
|
|
|
printf(" Serval Mesh process not running\n");
|
|
|
|
|
2012-02-23 01:25:39 +00:00
|
|
|
return 0;
|
2012-02-16 14:05:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* NULL marks ends of command structure.
|
|
|
|
"<anystring>" marks an arg that can take any value.
|
|
|
|
Only exactly matching prototypes will be used.
|
|
|
|
Together with the description, this makes it easy for us to auto-generate the
|
|
|
|
list of valid command line formats for display to the user if they try an
|
|
|
|
invalid one. It also means we can do away with getopt() etc.
|
|
|
|
|
|
|
|
Keep this list alphabetically sorted for user convenience.
|
|
|
|
*/
|
|
|
|
command_line_option command_line_options[]={
|
|
|
|
{app_dna_lookup,{"dna","lookup","<did>",NULL},CLIFLAG_NONOVERLAY,"Lookup the SIP/MDP address of the supplied telephone number (DID)."},
|
2012-02-23 01:25:39 +00:00
|
|
|
{cli_usage,{"help",NULL},0,
|
|
|
|
"Display command usage."},
|
|
|
|
{app_server_start,{"node","start"},CLIFLAG_STANDALONE,
|
|
|
|
"Start Serval Mesh node process. Instance path is read from SERVALINSTANCE_PATH environment variable."},
|
|
|
|
{app_server_start,{"node","start","in","<instance path>"},CLIFLAG_STANDALONE,
|
|
|
|
"Start Serval Mesh node process. Instance path is as specified."},
|
|
|
|
{app_server_stop,{"node","stop"},0,
|
|
|
|
"Ask running Serval Mesh node process to stop. Instance path is read from SERVALINSTANCE_PATH environment variable."},
|
|
|
|
{app_server_stop,{"node","stop","in","<instance path>"},0,
|
|
|
|
"Ask running Serval Mesh node process to stop. Instance path as specified."},
|
|
|
|
{app_server_status,{"node","status"},0,
|
|
|
|
"Display information about any running Serval Mesh node."},
|
2012-02-16 14:05:28 +00:00
|
|
|
{NULL,{NULL}}
|
|
|
|
};
|