2 * viking -- GPS Data and Topo Analyzer, Explorer, and Manager
4 * Copyright (C) 2003-2005, Evan Battaglia <gtoevan@gmx.net>
6 * This program is free software; you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License as published by
8 * the Free Software Foundation; either version 2 of the License, or
9 * (at your option) any later version.
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
16 * You should have received a copy of the GNU General Public License
17 * along with this program; if not, write to the Free Software
18 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
22 /* babel.c: running external programs and redirecting to TRWLayers.
23 * GPSBabel may not be necessary for everything -- for instance,
24 * use a_babel_convert_from_shellcommand with input_file_type == NULL
25 * for an external program that outputs GPX.
36 #ifdef HAVE_SYS_WAIT_H
43 #include <glib/gstdio.h>
45 /* in the future we could have support for other shells (change command strings), or not use a shell at all */
46 #define BASH_LOCATION "/bin/bash"
48 gboolean
a_babel_convert( VikTrwLayer
*vt
, const char *babelargs
, BabelStatusFunc cb
, gpointer user_data
)
54 gchar
*bargs
= g_strconcat(babelargs
, " -i gpx", NULL
);
56 if ((fd_src
= g_file_open_tmp("tmp-viking.XXXXXX", &name_src
, NULL
)) >= 0) {
57 f
= fdopen(fd_src
, "w");
58 a_gpx_write_file(vt
, f
);
61 ret
= a_babel_convert_from ( vt
, bargs
, cb
, name_src
, user_data
);
70 /* Runs args[0] with the arguments and uses the GPX module
71 * to import the GPX data into layer vt. Assumes that upon
72 * running the command, the data will appear in the (usually
73 * temporary) file name_dst.
75 * cb: callback that is run upon new data from STDOUT (?)
76 * (TODO: STDERR would be nice since we usually redirect STDOUT)
77 * user_data: passed along to cb
79 * returns TRUE on success
81 gboolean
babel_general_convert_from( VikTrwLayer
*vt
, BabelStatusFunc cb
, gchar
**args
, const gchar
*name_dst
, gpointer user_data
)
90 if (!g_spawn_async_with_pipes (NULL
, args
, NULL
, 0, NULL
, NULL
, &pid
, NULL
, &babel_stdout
, NULL
, &error
)) {
91 g_warning("Error : %s", error
->message
);
97 diag
= fdopen(babel_stdout
, "r");
98 setvbuf(diag
, NULL
, _IONBF
, 0);
100 while (fgets(line
, sizeof(line
), diag
)) {
102 cb(BABEL_DIAG_OUTPUT
, line
, user_data
);
105 cb(BABEL_DONE
, NULL
, user_data
);
108 waitpid(pid
, NULL
, 0);
109 g_spawn_close_pid(pid
);
111 f
= g_fopen(name_dst
, "r");
113 a_gpx_read_file ( vt
, f
);
123 gboolean
a_babel_convert_from( VikTrwLayer
*vt
, const char *babelargs
, BabelStatusFunc cb
, const char *from
, gpointer user_data
)
128 gboolean ret
= FALSE
;
132 if ((fd_dst
= g_file_open_tmp("tmp-viking.XXXXXX", &name_dst
, NULL
)) >= 0) {
136 gpsbabel_loc
= g_find_program_in_path("gpsbabel");
139 gchar
*unbuffer_loc
= g_find_program_in_path("unbuffer");
140 cmd
= g_strdup_printf ( "%s%s%s %s -o gpx %s %s",
141 unbuffer_loc
? unbuffer_loc
: "",
142 unbuffer_loc
? " " : "",
149 g_free ( unbuffer_loc
);
151 if ( g_shell_parse_argv(cmd
, &nb_args
, &args
, NULL
) ) {
152 ret
= babel_general_convert_from ( vt
, cb
, args
, name_dst
, user_data
);
164 /* Runs the input command in a shell (bash) and optionally uses GPSBabel to convert from input_file_type.
165 * If input_file_type is NULL, doesn't use GPSBabel. Input must be GPX (or Geocaching *.loc)
167 * Uses babel_general_convert_from to actually run the command. This function
168 * prepares the command and temporary file, and sets up the arguments for bash.
170 gboolean
a_babel_convert_from_shellcommand ( VikTrwLayer
*vt
, const char *input_cmd
, const char *input_file_type
, BabelStatusFunc cb
, gpointer user_data
)
174 gboolean ret
= FALSE
;
177 if ((fd_dst
= g_file_open_tmp("tmp-viking.XXXXXX", &name_dst
, NULL
)) >= 0) {
178 gchar
*shell_command
;
179 if ( input_file_type
)
180 shell_command
= g_strdup_printf("%s | gpsbabel -i %s -f - -o gpx -F %s", input_cmd
, input_file_type
, name_dst
);
182 shell_command
= g_strdup_printf("%s > %s", input_cmd
, name_dst
);
184 g_debug("%s: %s", __FUNCTION__
, shell_command
);
187 args
= g_malloc(sizeof(gchar
*)*4);
188 args
[0] = BASH_LOCATION
;
190 args
[2] = shell_command
;
193 ret
= babel_general_convert_from ( vt
, cb
, args
, name_dst
, user_data
);
195 g_free ( shell_command
);
203 gboolean
a_babel_convert_from_url ( VikTrwLayer
*vt
, const char *url
, const char *input_type
, BabelStatusFunc cb
, gpointer user_data
)
207 gboolean ret
= FALSE
;
211 g_debug("%s: input_type=%s url=%s", __FUNCTION__
, input_type
, url
);
213 if ((fd_src
= g_file_open_tmp("tmp-viking.XXXXXX", &name_src
, NULL
)) >= 0) {
217 babelargs
= g_strdup_printf(" -i %s", input_type
);
219 fetch_ret
= a_http_download_get_url(url
, "", name_src
, NULL
);
221 ret
= a_babel_convert_from( vt
, babelargs
, NULL
, name_src
, NULL
);
231 gboolean
babel_general_convert_to( VikTrwLayer
*vt
, BabelStatusFunc cb
, gchar
**args
, const gchar
*name_src
, gpointer user_data
)
233 gboolean ret
= FALSE
;
235 GError
*error
= NULL
;
238 if (!a_file_export(vt
, name_src
, FILE_TYPE_GPX
)) {
239 g_warning("%s(): error exporting to %s", __FUNCTION__
, name_src
);
243 if (!g_spawn_async_with_pipes (NULL
, args
, NULL
, 0, NULL
, NULL
, &pid
, NULL
, &babel_stdout
, NULL
, &error
)) {
244 g_warning("Error : %s", error
->message
);
250 diag
= fdopen(babel_stdout
, "r");
251 setvbuf(diag
, NULL
, _IONBF
, 0);
253 while (fgets(line
, sizeof(line
), diag
)) {
255 cb(BABEL_DIAG_OUTPUT
, line
, user_data
);
258 cb(BABEL_DONE
, NULL
, user_data
);
261 waitpid(pid
, NULL
, 0);
262 g_spawn_close_pid(pid
);
270 gboolean
a_babel_convert_to( VikTrwLayer
*vt
, const char *babelargs
, BabelStatusFunc cb
, const char *to
, gpointer user_data
)
275 gboolean ret
= FALSE
;
279 if ((fd_src
= g_file_open_tmp("tmp-viking.XXXXXX", &name_src
, NULL
)) >= 0) {
283 gpsbabel_loc
= g_find_program_in_path("gpsbabel");
286 gchar
*unbuffer_loc
= g_find_program_in_path("unbuffer");
287 cmd
= g_strdup_printf ( "%s%s%s %s -i gpx %s %s",
288 unbuffer_loc
? unbuffer_loc
: "",
289 unbuffer_loc
? " " : "",
296 g_free ( unbuffer_loc
);
297 g_debug ( "%s: %s", __FUNCTION__
, cmd
);
298 if ( g_shell_parse_argv(cmd
, &nb_args
, &args
, NULL
) ) {
299 ret
= babel_general_convert_to ( vt
, cb
, args
, name_src
, user_data
);