Fixed locale support
[captive.git] / src / client / cmdline / main.c
1 /* $Id$
2  * client cmdline interface for libcaptive
3  * Copyright (C) 2002-2003 Jan Kratochvil <project-captive@jankratochvil.net>
4  * 
5  * This program is free software; you can redistribute it and/or modify
6  * it under the terms of the GNU General Public License as published by
7  * the Free Software Foundation; exactly version 2 of June 1991 is required
8  * 
9  * This program is distributed in the hope that it will be useful,
10  * but WITHOUT ANY WARRANTY; without even the implied warranty of
11  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12  * GNU General Public License for more details.
13  * 
14  * You should have received a copy of the GNU General Public License
15  * along with this program; if not, write to the Free Software
16  * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
17  */
18
19
20 #include "config.h"
21
22 #include <glib/gmessages.h>
23 #include <stdlib.h>
24 #include <glib/giochannel.h>
25 #include <glib/gerror.h>
26 #include <popt.h>
27 #include <string.h>
28 #include <stdio.h>
29 #include <locale.h>
30
31 #include <captive/client.h>     /* for captive_init() */
32
33 #include "main.h"       /* self */
34 #include "cmd_shell.h"
35 #include "cmd_cd.h"
36 #include "cmd_lcd.h"
37 #include "cmd_ls.h"
38 #include "cmd_get.h"
39 #include "cmd_put.h"
40 #include "cmd_info.h"
41 #include "cmd_rm.h"
42 #include "cmd_mv.h"
43 #include "cmd_mkdir.h"
44 #include "cmd_rmdir.h"
45 #include "cmd_quit.h"
46 #include "cmd_help.h"
47
48
49 GQuark cmdline_main_error_quark(void)
50 {
51 GQuark r=0;
52
53         if (!r)
54                 r=g_quark_from_static_string("cmdline-main");
55
56         return r;
57 }
58
59
60 static const struct poptOption popt_table[]={
61                 CAPTIVE_POPT_INCLUDE,
62                 POPT_AUTOHELP
63                 POPT_TABLEEND
64                 };
65
66 const struct cmdline_command cmdline_command_table[]={
67                 /* First entry is the default if no command name was specified. */
68                 { "shell",N_("Interactive commands shell.")                        ,cmd_shell_table,cmd_shell,0,0 },
69                 { "cd"   ,N_("Print or change current guest-os directory[1].")     ,cmd_cd_table   ,cmd_cd   ,0,1 },
70                 { "lcd"  ,N_("Print or change current host-os  directory[1].")     ,cmd_lcd_table  ,cmd_lcd  ,0,1 },
71                 { "ls"   ,N_("Directory[1] listing.")                              ,cmd_ls_table   ,cmd_ls   ,0,1 },
72                 { "get"  ,N_("Copy guest-os file[1] to host-os (opt. file[2]).")   ,cmd_get_table  ,cmd_get  ,1,2 },
73                 { "put"  ,N_("Copy host-os file[1] to guest-os (opt. file[2]).")   ,cmd_put_table  ,cmd_put  ,1,2 },
74                 { "info" ,N_("Query information about guest-os item[1].")          ,cmd_info_table ,cmd_info ,1,1 },
75                 { "rm"   ,N_("Remove guest-os file[1].")                           ,cmd_rm_table   ,cmd_rm   ,1,1 },
76                 { "mv"   ,N_("Move (rename) guest-os item[1] to guest-os item[2]."),cmd_mv_table   ,cmd_mv   ,2,2 },
77                 { "mkdir",N_("Create guest-os directory[1].")                      ,cmd_mkdir_table,cmd_mkdir,1,1 },
78                 { "rmdir",N_("Remove guest-os directory[1].")                      ,cmd_rmdir_table,cmd_rmdir,1,1 },
79                 { "quit" ,N_("Quit this program.")                                 ,cmd_quit_table ,cmd_quit ,0,0 },
80                 { "help" ,N_("Show this list of commands.")                        ,cmd_help_table ,cmd_help ,0,0 },
81                 { NULL },       /* G_N_ELEMENTS() not usable as sizeof() is not visible for 'extern' */
82                 };
83
84
85 static void invoke_cmd_err(int cmd_argc,const char **cmd_argv,GError **errp)
86 {
87 const struct cmdline_command *commandp;
88 const char *cmd_name=NULL;
89 poptContext cmd_context;
90 int errint;
91 const char **cmdarg_argv;
92 int cmdarg_argc;
93 const char **csp;
94 const char *emptyargv_NULL=NULL;
95
96         g_return_if_fail(cmd_argc>=0);
97         g_return_if_fail(!errp || !*errp);
98
99         /* poptGetContext() cannot be passed argc==0 even if we lass POPT_CONTEXT_KEEP_FIRST
100          * as it is buggy. Workaround it by keeping the command name as argv[0].
101          */
102         if (!cmd_argc) {
103 const char *stub_shell[]={ cmdline_command_table[0].name,NULL };
104                 
105                 cmd_argc=1;
106                 cmd_argv=stub_shell;
107                 }
108
109         cmd_name=*cmd_argv;
110         for (commandp=cmdline_command_table;commandp->name;commandp++) {
111                 if (!cmd_name   /* NULL cmd_name fallback to the first table entry - "shell" */
112                                 || !strcasecmp(cmd_name,commandp->name))
113                         break;
114                 }
115         if (!commandp->name) {
116                 g_set_error(errp,CMDLINE_MAIN_ERROR,CMDLINE_MAIN_ERROR_UNKNOWN_COMMAND,
117                                 _("Unknown command, try 'help': %s"),cmd_name);
118                 return;
119                 }
120         cmd_context=poptGetContext(
121                         PACKAGE,        /* name */
122                         cmd_argc,cmd_argv,      /* argc,argv */
123                         commandp->table,        /* options */
124                         POPT_CONTEXT_POSIXMEHARDER);    /* flags; !POPT_CONTEXT_KEEP_FIRST */
125         if (cmd_context==NULL) {
126                 g_set_error(errp,CMDLINE_MAIN_ERROR,CMDLINE_MAIN_ERROR_INVALID_COMMAND_ARGUMENTS,
127                                 _("Invalid arguments for command: %s"),cmd_name);
128                 return;
129                 }
130         errint=poptReadDefaultConfig(cmd_context,
131                         TRUE);  /* useEnv */
132         if (errint!=0) {
133                 g_set_error(errp,CMDLINE_MAIN_ERROR,CMDLINE_MAIN_ERROR_READING_COMMAND_CONFIG,
134                                 _("Error '%s' reading default configuration for command: %s"),poptStrerror(errint),cmd_name);
135                 return;
136                 }
137         errint=poptGetNextOpt(cmd_context);
138         if (errint!=-1) {
139                 g_set_error(errp,CMDLINE_MAIN_ERROR,CMDLINE_MAIN_ERROR_EXCEEDING_COMMAND_OPTION,
140                                 _("Exceeding command option for command: %s"),cmd_name);
141                 return;
142                 }
143         if (!(cmdarg_argv=poptGetArgs(cmd_context)))
144                 cmdarg_argv=&emptyargv_NULL;
145         
146         for (csp=cmdarg_argv,cmdarg_argc=0;*csp;csp++)
147                 cmdarg_argc++;
148         if (cmdarg_argc<commandp->argsn_min || cmdarg_argc>commandp->argsn_max) {
149                 g_set_error(errp,CMDLINE_MAIN_ERROR,CMDLINE_MAIN_ERROR_INVALID_COMMAND_ARGUMENT_COUNT,
150                                 _("Invalid number of command '%s' arguments: %d; expected from %d to %d incl."),
151                                 cmd_name,cmdarg_argc,commandp->argsn_min,commandp->argsn_max);
152                 return;
153                 }
154
155         (*commandp->func)(cmdarg_argv,errp);
156
157         poptFreeContext(cmd_context);
158 }
159
160
161 void err_cleanup(GError **errp)
162 {
163         g_return_if_fail(errp!=NULL);
164
165         if (!*errp)
166                 return;
167         printf("\nERROR: %s\n",(*errp)->message);
168         g_clear_error(errp);
169 }
170
171
172 void invoke_cmd(int cmd_argc,const char **cmd_argv)
173 {
174 GError *gerr=NULL;
175
176         invoke_cmd_err(cmd_argc,cmd_argv,&gerr);
177         err_cleanup(&gerr);
178 }
179
180
181 /* Returns: Success (no error occured). */
182 gboolean errvfsresult_to_gerr(GError **errp,GnomeVFSResult errvfsresult)
183 {
184         g_return_val_if_fail(!errp || !*errp,FALSE);
185
186         if (errvfsresult==GNOME_VFS_OK)
187                 return TRUE;
188
189         g_set_error(errp,CMDLINE_MAIN_ERROR,CMDLINE_MAIN_ERROR_GENERIC_ERROR,
190                         _("Generic error: %s"),gnome_vfs_result_to_string(errvfsresult));
191         return FALSE;
192 }
193
194 int main(int argc,char **argv)
195 {
196 poptContext context;
197 int errint;
198 const char **cmd_argv,**csp;
199 int cmd_argc;
200 GError *gerr=NULL;
201 const char *cmd_cd_root_args[]={"/",NULL};
202
203         /* Initialize the i18n stuff */
204         setlocale(LC_ALL,"");
205         bindtextdomain(PACKAGE,LOCALEDIR);
206         textdomain(PACKAGE);
207
208         context=poptGetContext(
209                         PACKAGE,        /* name */
210                         argc,(/*en-const*/const char **)argv,   /* argc,argv */
211                         popt_table,     /* options */
212                         POPT_CONTEXT_POSIXMEHARDER);    /* flags; && !POPT_CONTEXT_KEEP_FIRST */
213         if (context==NULL) {
214                 g_assert_not_reached(); /* argument recognization args_error */
215                 return EXIT_FAILURE;
216                 }
217         errint=poptReadDefaultConfig(context,
218                         TRUE);  /* useEnv */
219         if (errint!=0) {
220                 g_assert_not_reached(); /* argument recognization args_error */
221                 return EXIT_FAILURE;
222                 }
223         errint=poptGetNextOpt(context);
224         if (errint!=-1) {
225                 g_assert_not_reached(); /* some non-callbacked argument reached */
226                 return EXIT_FAILURE;
227                 }
228         cmd_argv=poptGetArgs(context);
229         for (csp=cmd_argv,cmd_argc=0;csp && *csp;csp++)
230                 cmd_argc++;
231
232         if (TRUE!=captive_init(NULL,    /* captive_args; already parsed above */
233                         (               /* image_iochannel */
234                                         !cmd_argc ? NULL : g_io_channel_new_file(       /* FIXME: g_io_channel_new_file() is NOT 64-bit compliant! */
235                                                         cmd_argv[0],    /* filename */
236                                                         (captive_option_rwmode==CAPTIVE_OPTION_RWMODE_RW ? "r+" : "r"), /* mode */
237                                                         NULL))))        /* error */
238                 g_error(_("captive_init image_iochannel FAIL"));
239         if (cmd_argc>0) {
240                 /* image file */
241                 cmd_argc--;
242                 cmd_argv++;
243                 }
244
245         cmd_cd(cmd_cd_root_args,&gerr);
246         if (gerr) {
247                 err_cleanup(&gerr);
248                 return EXIT_FAILURE;
249                 }
250
251         invoke_cmd(cmd_argc,cmd_argv);
252
253         /* 'cmd_argv' gets cleared by 'poptFreeContext(context);' below */
254         poptFreeContext(context);
255
256         return EXIT_SUCCESS;
257 }