9a7d363a5e2cf241f15902b1fb7bfb6f026262e3
[samba.git] / source4 / gtk / tools / gepdump.c
1 /* 
2    Unix SMB/CIFS implementation.
3    GTK+ Endpoint Mapper frontend
4    
5    Copyright (C) Jelmer Vernooij 2004
6
7    This program is free software; you can redistribute it and/or modify
8    it under the terms of the GNU General Public License as published by
9    the Free Software Foundation; either version 2 of the License, or
10    (at your option) any later version.
11    
12    This program is distributed in the hope that it will be useful,
13    but WITHOUT ANY WARRANTY; without even the implied warranty of
14    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
15    GNU General Public License for more details.
16    
17    You should have received a copy of the GNU General Public License
18    along with this program; if not, write to the Free Software
19    Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
20 */
21
22 #include "includes.h"
23 #include "dynconfig.h"
24 #include "librpc/gen_ndr/ndr_epmapper.h"
25 #include "librpc/gen_ndr/ndr_mgmt.h"
26 #include "gtk/common/select.h"
27 #include "gtk/common/gtk-smb.h"
28 #include "lib/cmdline/popt_common.h"
29
30 /* 
31  * Show: 
32  *  - RPC statistics
33  *  - Available interfaces
34  *   - Per interface: available endpoints
35  *   - Per interface auth details
36  */
37
38 static GtkWidget *mainwin;
39 static GtkTreeStore *store_eps;
40 static GtkWidget *table_statistics;
41 static GtkWidget *lbl_calls_in, *lbl_calls_out, *lbl_pkts_in, *lbl_pkts_out;
42 static GtkWidget *lbl_iface_version, *lbl_iface_uuid, *lbl_iface_name;
43 TALLOC_CTX *eps_ctx = NULL;
44 TALLOC_CTX *conn_ctx = NULL;
45
46 struct dcerpc_pipe *epmapper_pipe;
47 struct dcerpc_pipe *mgmt_pipe;
48
49 static void on_quit1_activate (GtkMenuItem *menuitem, gpointer user_data)
50 {
51         gtk_main_quit();
52 }
53
54
55 static void on_about1_activate (GtkMenuItem *menuitem, gpointer user_data)
56 {
57         GtkDialog *aboutwin = GTK_DIALOG(create_gtk_samba_about_dialog("gepdump"));
58         gtk_dialog_run(aboutwin);
59         gtk_widget_destroy(GTK_WIDGET(aboutwin));
60 }
61
62 static const char *get_protocol_name(enum epm_protocol protocol)
63 {
64         switch (protocol) {
65         case EPM_PROTOCOL_UUID: return "UUID";
66         case EPM_PROTOCOL_NCACN: return "NCACN";
67         case EPM_PROTOCOL_NCALRPC: return "NCALRPC";
68         case EPM_PROTOCOL_NCADG: return "NCADG";
69         case EPM_PROTOCOL_IP: return "IP";
70         case EPM_PROTOCOL_TCP: return "TCP";
71         case EPM_PROTOCOL_NETBIOS: return "NetBIOS";
72         case EPM_PROTOCOL_SMB: return "SMB";
73         case EPM_PROTOCOL_PIPE: return "PIPE";
74         case EPM_PROTOCOL_UNIX_DS: return "Unix";
75         default: return "Unknown";
76         }
77 }
78
79 static void add_epm_entry(TALLOC_CTX *mem_ctx, const char *annotation, struct epm_tower *t)
80 {
81         struct dcerpc_binding *bd;
82         int i;
83         NTSTATUS status;
84         GtkTreeIter toweriter;
85
86         status = dcerpc_binding_from_tower(mem_ctx, t, &bd);
87         if (!NT_STATUS_IS_OK(status)) {
88                 gtk_show_ntstatus(mainwin, status);
89                 return;
90         }
91         
92         /* Don't show UUID's */
93         ZERO_STRUCT(bd->object);
94
95         gtk_tree_store_append(store_eps, &toweriter, NULL);
96         gtk_tree_store_set(store_eps, &toweriter, 0, strdup(annotation), 1, strdup(dcerpc_binding_string(mem_ctx, bd)), 2, t, -1);
97
98         for (i = 0; i < t->num_floors; i++) {
99                 const char *data;
100                 struct GUID if_uuid;
101                 uint16_t if_version;
102                 GtkTreeIter iter;
103                 gtk_tree_store_append(store_eps, &iter, &toweriter);
104
105                 dcerpc_floor_get_lhs_data(&t->floors[i], &if_uuid, &if_version);
106                 if (t->floors[i].lhs.protocol == EPM_PROTOCOL_UUID) {
107                         data = GUID_string(mem_ctx, &if_uuid);
108                 } else {
109                         data = dcerpc_floor_get_rhs_data(mem_ctx, &t->floors[i]);
110                 }
111                 
112                 gtk_tree_store_set(store_eps, &iter, 0, get_protocol_name(t->floors[i].lhs.protocol), 1, data, -1);
113         }
114 }
115
116 static void refresh_eps(void)
117 {
118         NTSTATUS status;
119         struct epm_Lookup r;
120         struct GUID uuid;
121         struct rpc_if_id_t iface;
122         struct policy_handle handle;
123         TALLOC_CTX *mem_ctx = talloc_init("dump");
124
125         talloc_free(eps_ctx);
126
127         ZERO_STRUCT(handle);
128
129         r.in.inquiry_type = 0;
130         r.in.object = &uuid;
131         r.in.interface_id = &iface;
132         r.in.vers_option = 0;
133         r.in.entry_handle = r.out.entry_handle = &handle;
134         r.in.max_ents = 10;
135
136         gtk_tree_store_clear(store_eps);
137
138         eps_ctx = talloc_init("current endpoint list data");
139
140         do {
141                 int i;
142                 ZERO_STRUCT(uuid);
143                 ZERO_STRUCT(iface);
144
145                 status = dcerpc_epm_Lookup(epmapper_pipe, eps_ctx, &r);
146                 if (!NT_STATUS_IS_OK(status) || r.out.result != 0) {
147                         break;
148                 }
149                 for (i=0;i<r.out.num_ents;i++) {
150                         add_epm_entry(mem_ctx, r.out.entries[i].annotation, &r.out.entries[i].tower->tower);
151                 }
152
153         } while (NT_STATUS_IS_OK(status) && 
154                  r.out.result == 0 && 
155                  r.out.num_ents == r.in.max_ents);
156
157         if (!NT_STATUS_IS_OK(status)) {
158                 gtk_show_ntstatus(mainwin, status);
159                 talloc_free(mem_ctx);
160                 return;
161         }
162         talloc_free(mem_ctx);
163 }
164
165 static void on_refresh_clicked (GtkButton *btn, gpointer user_data)
166 {
167         refresh_eps();
168 }
169
170 static void on_connect_clicked(GtkButton *btn, gpointer         user_data)
171 {
172         GtkRpcBindingDialog *d;
173         const char *bs;
174         TALLOC_CTX *mem_ctx;
175         NTSTATUS status;
176         gint result;
177
178         d = GTK_RPC_BINDING_DIALOG(gtk_rpc_binding_dialog_new(TRUE, NULL));
179         result = gtk_dialog_run(GTK_DIALOG(d));
180         switch(result) {
181         case GTK_RESPONSE_ACCEPT:
182                 break;
183         default:
184                 gtk_widget_destroy(GTK_WIDGET(d));
185                 return;
186         }
187
188         mem_ctx = talloc_init("connect");
189         bs = gtk_rpc_binding_dialog_get_binding_string (d, mem_ctx);
190
191         status = dcerpc_pipe_connect(&epmapper_pipe, bs, 
192                                      DCERPC_EPMAPPER_UUID, DCERPC_EPMAPPER_VERSION, 
193                                      cmdline_credentials);
194
195         if (NT_STATUS_IS_ERR(status)) {
196                 gtk_show_ntstatus(mainwin, status);
197                 goto fail;
198         }
199
200         refresh_eps();
201
202         status = dcerpc_secondary_context(epmapper_pipe, &mgmt_pipe, DCERPC_MGMT_UUID, DCERPC_MGMT_VERSION);
203
204         if (NT_STATUS_IS_ERR(status)) {
205                 mgmt_pipe = NULL;
206                 gtk_show_ntstatus(NULL, status);
207                 goto fail;
208         }
209
210 fail:
211         gtk_widget_destroy(GTK_WIDGET(d));
212 }
213
214 static gboolean on_eps_select(GtkTreeSelection *selection,
215     GtkTreeModel *model, GtkTreePath *path, gboolean path_currently_selected, gpointer data)
216 {
217         NTSTATUS status;
218         TALLOC_CTX *mem_ctx;
219         
220         if (mgmt_pipe == NULL) 
221                 return FALSE;
222         
223         mem_ctx = talloc_init("eps");
224
225         {
226                 /* Do an InqStats call */
227                 struct mgmt_inq_stats r;
228
229                 r.in.max_count = MGMT_STATS_ARRAY_MAX_SIZE;
230                 r.in.unknown = 0;
231
232                 status = dcerpc_mgmt_inq_stats(mgmt_pipe, mem_ctx, &r);
233                 if (NT_STATUS_IS_ERR(status)) {
234                         gtk_show_ntstatus(NULL, status);
235                         return TRUE;
236                 }
237
238                 if (r.out.statistics.count != MGMT_STATS_ARRAY_MAX_SIZE) {
239                         printf("Unexpected array size %d\n", r.out.statistics.count);
240                         return False;
241                 }
242
243                 gtk_label_set_text(GTK_LABEL(lbl_calls_in), talloc_asprintf(mem_ctx, "%6d", r.out.statistics.statistics[MGMT_STATS_CALLS_IN]));
244                 gtk_label_set_text(GTK_LABEL(lbl_calls_out), talloc_asprintf(mem_ctx, "%6d", r.out.statistics.statistics[MGMT_STATS_CALLS_OUT]));
245                 gtk_label_set_text(GTK_LABEL(lbl_pkts_in), talloc_asprintf(mem_ctx, "%6d", r.out.statistics.statistics[MGMT_STATS_PKTS_IN]));
246                 gtk_label_set_text(GTK_LABEL(lbl_pkts_out), talloc_asprintf(mem_ctx, "%6d", r.out.statistics.statistics[MGMT_STATS_PKTS_OUT]));
247         }
248
249         {
250                 struct mgmt_inq_princ_name r;
251                 int i;
252
253                 for (i=0;i<100;i++) {
254                         r.in.authn_proto = i;  /* DCERPC_AUTH_TYPE_* */
255                         r.in.princ_name_size = 100;
256
257                         status = dcerpc_mgmt_inq_princ_name(mgmt_pipe, mem_ctx, &r);
258                         if (!NT_STATUS_IS_OK(status)) {
259                                 continue;
260                         }
261                         if (W_ERROR_IS_OK(r.out.result)) {
262                                 const char *name = gensec_get_name_by_authtype(i);
263                                 if (name) {
264                                         printf("\tprinciple name for proto %u (%s) is '%s'\n",
265                                                    i, name, r.out.princ_name);
266                                 } else {
267                                         printf("\tprinciple name for proto %u is '%s'\n",
268                                                    i, r.out.princ_name);
269                                 }
270                         }
271                 }
272         }
273
274         return TRUE;
275 }
276
277
278 static GtkWidget* create_mainwindow (void)
279 {
280         GtkWidget *mainwindow;
281         GtkWidget *vbox1, *vbox2, *vbox3;
282         GtkWidget *menubar1;
283         GtkWidget *menuitem1;
284         GtkWidget *menuitem1_menu;
285         GtkWidget *quit1;
286         GtkWidget *menuitem4;
287         GtkWidget *menuitem4_menu;
288         GtkWidget *mnu_connect;
289         GtkWidget *mnu_refresh;
290         GtkWidget *about1;
291         GtkWidget *hbox2;
292         GtkWidget *scrolledwindow1;
293         GtkWidget *frame1;
294         GtkWidget *tree_eps;
295         GtkTreeViewColumn *curcol;
296         GtkCellRenderer *renderer;
297         GtkWidget *statusbar;
298         GtkAccelGroup *accel_group;
299
300         accel_group = gtk_accel_group_new ();
301
302         mainwindow = gtk_window_new (GTK_WINDOW_TOPLEVEL);
303         gtk_window_set_title (GTK_WINDOW (mainwindow), "Gtk+ Endpoint Mapper Viewer");
304
305         vbox1 = gtk_vbox_new (FALSE, 0);
306         gtk_widget_show (vbox1);
307         gtk_container_add (GTK_CONTAINER (mainwindow), vbox1);
308
309         menubar1 = gtk_menu_bar_new ();
310         gtk_widget_show (menubar1);
311         gtk_box_pack_start (GTK_BOX (vbox1), menubar1, FALSE, FALSE, 0);
312
313         menuitem1 = gtk_menu_item_new_with_mnemonic ("_File");
314         gtk_widget_show (menuitem1);
315         gtk_container_add (GTK_CONTAINER (menubar1), menuitem1);
316
317         menuitem1_menu = gtk_menu_new ();
318         gtk_menu_item_set_submenu (GTK_MENU_ITEM (menuitem1), menuitem1_menu);
319
320         mnu_connect = gtk_menu_item_new_with_mnemonic ("_Connect");
321         gtk_container_add(GTK_CONTAINER(menuitem1_menu), mnu_connect);
322
323         mnu_refresh = gtk_menu_item_new_with_mnemonic ("_Refresh");
324         gtk_container_add(GTK_CONTAINER(menuitem1_menu), mnu_refresh);
325
326         quit1 = gtk_image_menu_item_new_from_stock ("gtk-quit", accel_group);
327         gtk_container_add (GTK_CONTAINER (menuitem1_menu), quit1);
328
329         menuitem4 = gtk_menu_item_new_with_mnemonic ("_Help");
330         gtk_container_add (GTK_CONTAINER (menubar1), menuitem4);
331
332         menuitem4_menu = gtk_menu_new ();
333         gtk_menu_item_set_submenu (GTK_MENU_ITEM (menuitem4), menuitem4_menu);
334
335         about1 = gtk_menu_item_new_with_mnemonic ("_About");
336         gtk_container_add (GTK_CONTAINER (menuitem4_menu), about1);
337
338         hbox2 = gtk_hbox_new (FALSE, 0);
339         gtk_container_add (GTK_CONTAINER (vbox1), hbox2);
340
341         scrolledwindow1 = gtk_scrolled_window_new (NULL, NULL);
342         gtk_box_pack_start (GTK_BOX(hbox2), scrolledwindow1, TRUE, TRUE, 0);
343
344         tree_eps = gtk_tree_view_new ();
345
346         curcol = gtk_tree_view_column_new ();
347         gtk_tree_view_column_set_title(curcol, "Name");
348         renderer = gtk_cell_renderer_text_new();
349         gtk_tree_view_column_pack_start(curcol, renderer, True);
350
351         gtk_tree_view_append_column(GTK_TREE_VIEW(tree_eps), curcol);
352         gtk_tree_view_column_add_attribute(curcol, renderer, "text", 0);
353
354         curcol = gtk_tree_view_column_new ();
355         gtk_tree_view_column_set_title(curcol, "Binding String");
356         renderer = gtk_cell_renderer_text_new();
357         gtk_tree_view_column_pack_start(curcol, renderer, True);
358         gtk_tree_view_column_add_attribute(curcol, renderer, "text", 1);
359
360
361         gtk_tree_view_append_column(GTK_TREE_VIEW(tree_eps), curcol);
362
363         store_eps = gtk_tree_store_new(3, G_TYPE_STRING, G_TYPE_STRING, G_TYPE_POINTER);
364         gtk_tree_view_set_model(GTK_TREE_VIEW(tree_eps), GTK_TREE_MODEL(store_eps));
365         g_object_unref(store_eps);
366
367         gtk_container_add (GTK_CONTAINER (scrolledwindow1), tree_eps);
368
369         gtk_tree_selection_set_select_function (gtk_tree_view_get_selection(GTK_TREE_VIEW(tree_eps)), on_eps_select, NULL, NULL);
370
371         vbox2 = gtk_vbox_new (FALSE, 0);
372         gtk_container_add (GTK_CONTAINER (hbox2), vbox2);
373
374         frame1 = gtk_frame_new("Interface");
375         gtk_container_add (GTK_CONTAINER(vbox2), frame1);
376
377         vbox3 = gtk_vbox_new (FALSE, 0);
378         gtk_container_add (GTK_CONTAINER (frame1), vbox3);
379         gtk_container_add (GTK_CONTAINER (vbox3), lbl_iface_uuid = gtk_label_new(""));
380         gtk_container_add (GTK_CONTAINER (vbox3), lbl_iface_version = gtk_label_new(""));
381         gtk_container_add (GTK_CONTAINER (vbox3), lbl_iface_name = gtk_label_new(""));
382
383         frame1 = gtk_frame_new("Statistics");
384         gtk_container_add (GTK_CONTAINER(vbox2), frame1);
385
386         table_statistics = gtk_table_new(4, 2, TRUE);
387         gtk_container_add (GTK_CONTAINER(frame1), table_statistics);
388
389         gtk_table_attach_defaults (GTK_TABLE(table_statistics), gtk_label_new("Calls In: "), 0, 1, 0, 1);
390         gtk_table_attach_defaults (GTK_TABLE(table_statistics), lbl_calls_in = gtk_label_new(""), 1, 2, 0, 1);
391         gtk_table_attach_defaults (GTK_TABLE(table_statistics), gtk_label_new("Calls Out: "), 0, 1, 1, 2);
392         gtk_table_attach_defaults (GTK_TABLE(table_statistics), lbl_calls_out = gtk_label_new(""), 1, 2, 1, 2);
393         gtk_table_attach_defaults (GTK_TABLE(table_statistics), gtk_label_new("Packets In: "), 0, 1, 2, 3);
394         gtk_table_attach_defaults (GTK_TABLE(table_statistics), lbl_pkts_in = gtk_label_new(""), 1, 2, 2, 3);
395         gtk_table_attach_defaults (GTK_TABLE(table_statistics), gtk_label_new("Packets Out: "), 0, 1, 3, 4);
396         gtk_table_attach_defaults (GTK_TABLE(table_statistics), lbl_pkts_out = gtk_label_new(""), 1, 2, 3, 4);
397         
398         frame1 = gtk_frame_new("Authentication");
399         gtk_container_add (GTK_CONTAINER(vbox2), frame1);
400
401         statusbar = gtk_statusbar_new ();
402         gtk_box_pack_start (GTK_BOX (vbox1), statusbar, FALSE, FALSE, 0);
403
404         g_signal_connect ((gpointer) quit1, "activate",
405                                           G_CALLBACK (on_quit1_activate),
406                                           NULL);
407         g_signal_connect ((gpointer) about1, "activate",
408                                           G_CALLBACK (on_about1_activate),
409                                           NULL);
410         g_signal_connect ((gpointer) mnu_connect, "activate",
411                                           G_CALLBACK (on_connect_clicked),
412                                           NULL);
413         g_signal_connect ((gpointer) mnu_refresh, "activate",
414                                           G_CALLBACK (on_refresh_clicked),
415                                           NULL);
416
417         gtk_window_add_accel_group (GTK_WINDOW (mainwindow), accel_group);
418
419         return mainwindow;
420 }
421
422
423  int main(int argc, char **argv)
424 {
425         gepdump_init_subsystems;
426         lp_load(dyn_CONFIGFILE,True,False,False);
427         load_interfaces();
428         setup_logging(argv[0], DEBUG_STDERR);
429
430         gtk_init(&argc, &argv);
431         mainwin = create_mainwindow();
432         gtk_widget_show_all(mainwin);
433
434         return gtk_event_loop();
435 }