]> git.meshlink.io Git - catta/blob - avahi-ui/bssh.c
9712ee89ceea7c114084e373009a7eb3f11ce716
[catta] / avahi-ui / bssh.c
1 /* $Id$ */
2
3 /***
4   This file is part of avahi.
5
6   avahi is free software; you can redistribute it and/or modify it
7   under the terms of the GNU Lesser General Public License as
8   published by the Free Software Foundation; either version 2.1 of the
9   License, or (at your option) any later version.
10
11   avahi is distributed in the hope that it will be useful, but WITHOUT
12   ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
13   or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General
14   Public License for more details.
15
16   You should have received a copy of the GNU Lesser General Public
17   License along with avahi; if not, write to the Free Software
18   Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
19   USA.
20 ***/
21
22 #ifdef HAVE_CONFIG_H
23 #include <config.h>
24 #endif
25
26 #include <string.h>
27 #include <unistd.h>
28 #include <errno.h>
29 #include <stdlib.h>
30 #include <locale.h>
31 #include <getopt.h>
32
33 #include <gtk/gtk.h>
34
35 #include <avahi-client/client.h>
36 #include <avahi-common/strlst.h>
37 #include <avahi-common/malloc.h>
38 #include <avahi-common/domain.h>
39 #include <avahi-common/i18n.h>
40
41 #include "avahi-ui.h"
42
43 typedef enum {
44     COMMAND_HELP,
45     COMMAND_SSH,
46     COMMAND_VNC,
47     COMMAND_SHELL
48 } Command;
49
50 typedef struct Config {
51     char *domain;
52     Command command;
53 } Config;
54
55 static void help(FILE *f, const char *argv0) {
56     fprintf(f,
57             _("%s [options]\n\n"
58               "    -h --help            Show this help\n"
59               "    -s --ssh             Browse SSH servers\n"
60               "    -v --vnc             Browse VNC servers\n"
61               "    -S --shell           Browse both SSH and VNC\n"
62               "    -d --domain=DOMAIN   The domain to browse in\n"),
63             argv0);
64 }
65
66 static int parse_command_line(Config *c, int argc, char *argv[]) {
67     int o;
68
69     static const struct option long_options[] = {
70         { "help",           no_argument,       NULL, 'h' },
71         { "ssh",            no_argument,       NULL, 's' },
72         { "vnc",            no_argument,       NULL, 'v' },
73         { "shell",          no_argument,       NULL, 'S' },
74         { "domain",         required_argument, NULL, 'd' },
75         { NULL, 0, NULL, 0 }
76     };
77
78     while ((o = getopt_long(argc, argv, "hVd:svS", long_options, NULL)) >= 0) {
79
80         switch(o) {
81             case 'h':
82                 c->command = COMMAND_HELP;
83                 break;
84             case 's':
85                 c->command = COMMAND_SSH;
86                 break;
87             case 'v':
88                 c->command = COMMAND_VNC;
89                 break;
90             case 'S':
91                 c->command = COMMAND_SHELL;
92                 break;
93             case 'd':
94                 avahi_free(c->domain);
95                 c->domain = avahi_strdup(optarg);
96                 break;
97             default:
98                 return -1;
99         }
100     }
101
102     if (optind < argc) {
103         fprintf(stderr, _("Too many arguments\n"));
104         return -1;
105     }
106
107     return 0;
108 }
109
110 int main(int argc, char*argv[]) {
111     GtkWidget *d;
112     Config config;
113     const char *argv0;
114
115     avahi_init_i18n();
116     setlocale(LC_ALL, "");
117
118     if ((argv0 = strrchr(argv[0], '/')))
119         argv0++;
120     else
121         argv0 = argv[0];
122
123     if (g_str_has_suffix(argv[0], "bshell"))
124         config.command = COMMAND_SHELL;
125     else if (g_str_has_suffix(argv[0], "bvnc"))
126         config.command = COMMAND_VNC;
127     else
128         config.command = COMMAND_SSH;
129
130     /* defaults to local */
131     config.domain = NULL;
132
133     if (parse_command_line(&config, argc, argv) < 0) {
134         help(stderr, argv0);
135         return 1;
136     }
137
138     bindtextdomain (GETTEXT_PACKAGE, GNOMELOCALEDIR);
139     bind_textdomain_codeset (GETTEXT_PACKAGE, "UTF-8");
140     textdomain (GETTEXT_PACKAGE);
141
142     gtk_init(&argc, &argv);
143
144     switch (config.command) {
145         case COMMAND_HELP:
146             help(stdout, argv0);
147             return 0;
148             break;
149
150         case COMMAND_SHELL:
151             d = aui_service_dialog_new(_("Choose Shell Server"), NULL, GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL, GTK_STOCK_CONNECT, GTK_RESPONSE_ACCEPT, NULL);
152             aui_service_dialog_set_browse_service_types(AUI_SERVICE_DIALOG(d), "_rfb._tcp", "_ssh._tcp", NULL);
153             aui_service_dialog_set_service_type_name(AUI_SERVICE_DIALOG(d), "_rfb._tcp", _("Desktop"));
154             aui_service_dialog_set_service_type_name(AUI_SERVICE_DIALOG(d), "_ssh._tcp", _("Terminal"));
155             break;
156
157         case COMMAND_VNC:
158             d = aui_service_dialog_new(_("Choose VNC server"), NULL, GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL, GTK_STOCK_CONNECT, GTK_RESPONSE_ACCEPT, NULL);
159             aui_service_dialog_set_browse_service_types(AUI_SERVICE_DIALOG(d), "_rfb._tcp", NULL);
160             break;
161
162         case COMMAND_SSH:
163             d = aui_service_dialog_new(_("Choose SSH server"), NULL, GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL, GTK_STOCK_CONNECT, GTK_RESPONSE_ACCEPT, NULL);
164             aui_service_dialog_set_browse_service_types(AUI_SERVICE_DIALOG(d), "_ssh._tcp", NULL);
165             break;
166     }
167
168     aui_service_dialog_set_domain (AUI_SERVICE_DIALOG(d), config.domain);
169     aui_service_dialog_set_resolve_service(AUI_SERVICE_DIALOG(d), TRUE);
170     aui_service_dialog_set_resolve_host_name(AUI_SERVICE_DIALOG(d), !avahi_nss_support());
171
172     gtk_window_present(GTK_WINDOW(d));
173
174     if (gtk_dialog_run(GTK_DIALOG(d)) == GTK_RESPONSE_ACCEPT) {
175         char a[AVAHI_ADDRESS_STR_MAX], *u = NULL, *n = NULL;
176         char *h = NULL, *t = NULL;
177         const AvahiStringList *txt;
178
179         t = g_strdup(aui_service_dialog_get_service_type(AUI_SERVICE_DIALOG(d)));
180         n = g_strdup(aui_service_dialog_get_service_name(AUI_SERVICE_DIALOG(d)));
181
182         if (avahi_nss_support())
183             h = g_strdup(aui_service_dialog_get_host_name(AUI_SERVICE_DIALOG(d)));
184         else
185             h = g_strdup(avahi_address_snprint(a, sizeof(a), aui_service_dialog_get_address(AUI_SERVICE_DIALOG(d))));
186
187         g_print(_("Connecting to '%s' ...\n"), n);
188
189         if (avahi_domain_equal(t, "_rfb._tcp")) {
190             char p[AVAHI_DOMAIN_NAME_MAX+16];
191             snprintf(p, sizeof(p), "%s:%u", h, aui_service_dialog_get_port(AUI_SERVICE_DIALOG(d))-5900);
192
193             gtk_widget_destroy(d);
194
195             g_print("vncviewer %s\n", p);
196             execlp("xvncviewer", "xvncviewer", p, NULL);
197             execlp("vncviewer", "vncviewer", p, NULL);
198
199         } else {
200             char p[16];
201
202             snprintf(p, sizeof(p), "%u", aui_service_dialog_get_port(AUI_SERVICE_DIALOG(d)));
203
204             for (txt = aui_service_dialog_get_txt_data(AUI_SERVICE_DIALOG(d)); txt; txt = txt->next) {
205                 char *key, *value;
206
207                 if (avahi_string_list_get_pair((AvahiStringList*) txt, &key, &value, NULL) < 0)
208                     break;
209
210                 if (strcmp(key, "u") == 0)
211                     u = g_strdup(value);
212
213                 avahi_free(key);
214                 avahi_free(value);
215             }
216
217             gtk_widget_destroy(d);
218
219             if (u) {
220                 g_print("ssh -p %s -l %s %s\n", p, u, h);
221
222                 if (isatty(0) || !getenv("DISPLAY"))
223                     execlp("ssh", "ssh", "-p", p, "-l", u, h, NULL);
224                 else {
225                     execlp("x-terminal-emulator", "x-terminal-emulator", "-T", n, "-e", "ssh", "-p", p, "-l", u, h, NULL);
226                     execlp("gnome-terminal", "gnome-terminal", "-t", n, "-x", "ssh", "-p", p, "-l", u, h, NULL);
227                     execlp("xterm", "xterm", "-T", n, "-e", "ssh", "-p", p, "-l", u, h, NULL);
228                 }
229             } else {
230                 g_print("ssh -p %s %s\n", p, h);
231
232                 if (isatty(0) || !getenv("DISPLAY"))
233                     execlp("ssh", "ssh", "-p", p, h, NULL);
234                 else {
235                     execlp("x-terminal-emulator", "x-terminal-emulator", "-T", n, "-e", "ssh", "-p", p, h, NULL);
236                     execlp("gnome-terminal", "gnome-terminal", "-t", n, "-x", "ssh", "-p", p, h, NULL);
237                     execlp("xterm", "xterm", "-T", n, "-e", "ssh", "-p", p, h, NULL);
238                 }
239             }
240         }
241
242         g_warning(_("execlp() failed: %s\n"), strerror(errno));
243
244         g_free(h);
245         g_free(u);
246         g_free(t);
247         g_free(n);
248
249     } else {
250         gtk_widget_destroy(d);
251
252         g_print(_("Canceled.\n"));
253     }
254
255     g_free(config.domain);
256
257     return 1;
258 }