blob: 6b6582679a8d4a1e1df5551d75514743cbffe287 [file] [log] [blame]
paul718e3742002-12-13 20:15:29 +00001/* Virtual terminal interface shell.
2 * Copyright (C) 2000 Kunihiro Ishiguro
3 *
4 * This file is part of GNU Zebra.
5 *
6 * GNU Zebra is free software; you can redistribute it and/or modify it
7 * under the terms of the GNU General Public License as published by the
8 * Free Software Foundation; either version 2, or (at your option) any
9 * later version.
10 *
11 * GNU Zebra is distributed in the hope that it will be useful, but
12 * WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * General Public License for more details.
15 *
16 * You should have received a copy of the GNU General Public License
17 * along with GNU Zebra; see the file COPYING. If not, write to the Free
18 * Software Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA
19 * 02111-1307, USA.
20 */
21
22#include <zebra.h>
23
24#include <sys/un.h>
25#include <setjmp.h>
26#include <sys/wait.h>
27#include <sys/resource.h>
28#include <sys/stat.h>
29
30#include <readline/readline.h>
31#include <readline/history.h>
32
33#include "command.h"
34#include "memory.h"
35#include "vtysh/vtysh.h"
36
37/* Struct VTY. */
38struct vty *vty;
39
40/* VTY shell pager name. */
41char *vtysh_pager_name = NULL;
42
43/* VTY shell client structure. */
44struct vtysh_client
45{
46 int fd;
47} vtysh_client[VTYSH_INDEX_MAX];
48
49/* When '^Z' is received from vty, move down to the enable mode. */
50int
51vtysh_end ()
52{
53 switch (vty->node)
54 {
55 case VIEW_NODE:
56 case ENABLE_NODE:
57 /* Nothing to do. */
58 break;
59 default:
60 vty->node = ENABLE_NODE;
61 break;
62 }
63 return CMD_SUCCESS;
64}
65
66DEFUNSH (VTYSH_ALL,
67 vtysh_end_all,
68 vtysh_end_all_cmd,
69 "end",
70 "End current mode and down to previous mode\n")
71{
72 return vtysh_end (vty);
73}
74
75DEFUNSH (VTYSH_ALL,
76 vtysh_log_stdout,
77 vtysh_log_stdout_cmd,
78 "log stdout",
79 "Logging control\n"
80 "Logging goes to stdout\n")
81{
82 return CMD_SUCCESS;
83}
84
85DEFUNSH (VTYSH_ALL,
86 no_vtysh_log_stdout,
87 no_vtysh_log_stdout_cmd,
88 "no log stdout",
89 NO_STR
90 "Logging control\n"
91 "Logging goes to stdout\n")
92{
93 return CMD_SUCCESS;
94}
95
96DEFUNSH (VTYSH_ALL,
97 vtysh_log_file,
98 vtysh_log_file_cmd,
99 "log file FILENAME",
100 "Logging control\n"
101 "Logging to file\n"
102 "Logging filename\n")
103{
104 return CMD_SUCCESS;
105}
106
107DEFUNSH (VTYSH_ALL,
108 no_vtysh_log_file,
109 no_vtysh_log_file_cmd,
110 "no log file [FILENAME]",
111 NO_STR
112 "Logging control\n"
113 "Cancel logging to file\n"
114 "Logging file name\n")
115{
116 return CMD_SUCCESS;
117}
118
119DEFUNSH (VTYSH_ALL,
120 vtysh_log_syslog,
121 vtysh_log_syslog_cmd,
122 "log syslog",
123 "Logging control\n"
124 "Logging goes to syslog\n")
125{
126 return CMD_SUCCESS;
127}
128
129DEFUNSH (VTYSH_ALL,
130 no_vtysh_log_syslog,
131 no_vtysh_log_syslog_cmd,
132 "no log syslog",
133 NO_STR
134 "Logging control\n"
135 "Cancel logging to syslog\n")
136{
137 return CMD_SUCCESS;
138}
139
140DEFUNSH (VTYSH_ALL,
141 vtysh_log_trap,
142 vtysh_log_trap_cmd,
143 "log trap (emergencies|alerts|critical|errors|warnings|notifications|informational|debugging)",
144 "Logging control\n"
145 "Limit logging to specifed level\n")
146{
147 return CMD_SUCCESS;
148}
149
150DEFUNSH (VTYSH_ALL,
151 no_vtysh_log_trap,
152 no_vtysh_log_trap_cmd,
153 "no log trap",
154 NO_STR
155 "Logging control\n"
156 "Permit all logging information\n")
157{
158 return CMD_SUCCESS;
159}
160
161DEFUNSH (VTYSH_ALL,
162 vtysh_log_record_priority,
163 vtysh_log_record_priority_cmd,
164 "log record-priority",
165 "Logging control\n"
166 "Log the priority of the message within the message\n")
167{
168 return CMD_SUCCESS;
169}
170
171DEFUNSH (VTYSH_ALL,
172 no_vtysh_log_record_priority,
173 no_vtysh_log_record_priority_cmd,
174 "no log record-priority",
175 NO_STR
176 "Logging control\n"
177 "Do not log the priority of the message within the message\n")
178{
179 return CMD_SUCCESS;
180}
181
182void
183vclient_close (struct vtysh_client *vclient)
184{
185 if (vclient->fd > 0)
186 close (vclient->fd);
187 vclient->fd = -1;
188}
189
190
191/* Following filled with debug code to trace a problematic condition
192 under load - it SHOULD handle it.
193*/
194#define ERR_WHERE_STRING "vtysh(): vtysh_client_config(): "
195int
196vtysh_client_config (struct vtysh_client *vclient, char *line)
197{
198 int ret;
199 char *buf;
200 size_t bufsz;
201 char *pbuf;
202 size_t left;
203 char *eoln;
204 int nbytes;
205 int i;
206 int readln;
207
208 if (vclient->fd < 0)
209 return CMD_SUCCESS;
210
211 ret = write (vclient->fd, line, strlen (line) + 1);
212 if (ret <= 0)
213 {
214 vclient_close (vclient);
215 return CMD_SUCCESS;
216 }
217
218 /* Allow enough room for buffer to read more than a few pages from socket
219 */
paule3d29b52003-01-23 18:05:42 +0000220 bufsz = 5 * getpagesize() + 1;
paul718e3742002-12-13 20:15:29 +0000221 buf = XMALLOC(MTYPE_TMP, bufsz);
222 memset(buf, 0, bufsz);
223 pbuf = buf;
224
225 while (1)
226 {
227 if (pbuf >= ((buf + bufsz) -1))
228 {
229 fprintf (stderr, ERR_WHERE_STRING \
230 "warning - pbuf beyond buffer end.\n");
231 return CMD_WARNING;
232 }
233
234 readln = (buf + bufsz) - pbuf - 1;
235 nbytes = read (vclient->fd, pbuf, readln);
236
237 if (nbytes <= 0)
238 {
239
240 if (errno == EINTR)
241 continue;
242
243 fprintf(stderr, ERR_WHERE_STRING "(%u)", errno);
244 perror("");
245
246 if (errno == EAGAIN || errno == EIO)
247 continue;
248
249 vclient_close (vclient);
250 XFREE(MTYPE_TMP, buf);
251 return CMD_SUCCESS;
252 }
253
254 pbuf[nbytes] = '\0';
255
256 if (nbytes >= 4)
257 {
258 i = nbytes - 4;
259 if (pbuf[i] == '\0' && pbuf[i + 1] == '\0' && pbuf[i + 2] == '\0')
260 {
261 ret = pbuf[i + 3];
262 break;
263 }
264 }
265 pbuf += nbytes;
266
267 /* See if a line exists in buffer, if so parse and consume it, and
268 reset read position */
269 if ((eoln = strrchr(buf, '\n')) == NULL)
270 continue;
271
272 if (eoln >= ((buf + bufsz) - 1))
273 {
274 fprintf (stderr, ERR_WHERE_STRING \
275 "warning - eoln beyond buffer end.\n");
276 }
277 vtysh_config_parse(buf);
278
279 eoln++;
280 left = (size_t)(buf + bufsz - eoln);
281 memmove(buf, eoln, left);
282 buf[bufsz-1] = '\0';
283 pbuf = buf + strlen(buf);
284 }
285
286 /* parse anything left in the buffer */
287 vtysh_config_parse (buf);
288
289 XFREE(MTYPE_TMP, buf);
290 return ret;
291}
292
293int
294vtysh_client_execute (struct vtysh_client *vclient, char *line, FILE *fp)
295{
296 int ret;
297 char buf[1001];
298 int nbytes;
299 int i;
300
301 if (vclient->fd < 0)
302 return CMD_SUCCESS;
303
304 ret = write (vclient->fd, line, strlen (line) + 1);
305 if (ret <= 0)
306 {
307 vclient_close (vclient);
308 return CMD_SUCCESS;
309 }
310
311 while (1)
312 {
313 nbytes = read (vclient->fd, buf, sizeof(buf)-1);
314
315 if (nbytes <= 0 && errno != EINTR)
316 {
317 vclient_close (vclient);
318 return CMD_SUCCESS;
319 }
320
321 if (nbytes > 0)
322 {
323 buf[nbytes] = '\0';
324 fprintf (fp, "%s", buf);
325 fflush (fp);
326
327 if (nbytes >= 4)
328 {
329 i = nbytes - 4;
330 if (buf[i] == '\0' && buf[i + 1] == '\0' && buf[i + 2] == '\0')
331 {
332 ret = buf[i + 3];
333 break;
334 }
335 }
336 }
337 }
338 return ret;
339}
340
341void
342vtysh_exit_ripd_only ()
343{
344 vtysh_client_execute (&vtysh_client[VTYSH_INDEX_RIP], "exit", stdout);
345}
346
347
348void
349vtysh_pager_init ()
350{
351 vtysh_pager_name = getenv ("VTYSH_PAGER");
352 if (! vtysh_pager_name)
paula805cc22003-05-01 14:29:48 +0000353 vtysh_pager_name = getenv ("PAGER");
354 if (! vtysh_pager_name)
paul718e3742002-12-13 20:15:29 +0000355 vtysh_pager_name = "more";
356}
357
358/* Command execution over the vty interface. */
359void
360vtysh_execute_func (char *line, int pager)
361{
362 int ret, cmd_stat;
363 vector vline;
364 struct cmd_element *cmd;
365 FILE *fp = NULL;
paula805cc22003-05-01 14:29:48 +0000366 int closepager=0;
paul718e3742002-12-13 20:15:29 +0000367
368 /* Split readline string up into the vector */
369 vline = cmd_make_strvec (line);
370
371 if (vline == NULL)
372 return;
373
374 ret = cmd_execute_command (vline, vty, &cmd);
375
376 cmd_free_strvec (vline);
377
378 switch (ret)
379 {
380 case CMD_WARNING:
381 if (vty->type == VTY_FILE)
paul4fc01e62002-12-13 20:49:00 +0000382 fprintf (stdout,"Warning...\n");
paul718e3742002-12-13 20:15:29 +0000383 break;
384 case CMD_ERR_AMBIGUOUS:
paul4fc01e62002-12-13 20:49:00 +0000385 fprintf (stdout,"%% Ambiguous command.\n");
paul718e3742002-12-13 20:15:29 +0000386 break;
387 case CMD_ERR_NO_MATCH:
paul4fc01e62002-12-13 20:49:00 +0000388 fprintf (stdout,"%% Unknown command.\n");
paul718e3742002-12-13 20:15:29 +0000389 break;
390 case CMD_ERR_INCOMPLETE:
paul4fc01e62002-12-13 20:49:00 +0000391 fprintf (stdout,"%% Command incomplete.\n");
paul718e3742002-12-13 20:15:29 +0000392 break;
393 case CMD_SUCCESS_DAEMON:
394 {
395 if (pager && vtysh_pager_name)
396 {
paul4fc01e62002-12-13 20:49:00 +0000397 fp = popen (vtysh_pager_name, "w");
paul718e3742002-12-13 20:15:29 +0000398 if (fp == NULL)
399 {
paula805cc22003-05-01 14:29:48 +0000400 perror ("popen failed for pager");
401 fp = stdout;
paul718e3742002-12-13 20:15:29 +0000402 }
paula805cc22003-05-01 14:29:48 +0000403 else
404 closepager=1;
paul718e3742002-12-13 20:15:29 +0000405 }
406 else
407 fp = stdout;
408
409 if (! strcmp(cmd->string,"configure terminal"))
410 {
411 cmd_stat = vtysh_client_execute (&vtysh_client[VTYSH_INDEX_ZEBRA],
412 line, fp);
413 if (cmd_stat != CMD_WARNING)
414 cmd_stat = vtysh_client_execute (&vtysh_client[VTYSH_INDEX_RIP],
415 line, fp);
416 if (cmd_stat != CMD_WARNING)
417 cmd_stat = vtysh_client_execute (&vtysh_client[VTYSH_INDEX_RIPNG], line, fp);
418 if (cmd_stat != CMD_WARNING)
419 cmd_stat = vtysh_client_execute (&vtysh_client[VTYSH_INDEX_OSPF],
420 line, fp);
421 if (cmd_stat != CMD_WARNING)
422 cmd_stat = vtysh_client_execute (&vtysh_client[VTYSH_INDEX_OSPF6], line, fp);
423 if (cmd_stat != CMD_WARNING)
424 cmd_stat = vtysh_client_execute (&vtysh_client[VTYSH_INDEX_BGP],
425 line, fp);
426 if (cmd_stat)
427 {
428 line = "end";
429 vline = cmd_make_strvec (line);
430
431 if (vline == NULL)
432 {
paula805cc22003-05-01 14:29:48 +0000433 if (pager && vtysh_pager_name && fp && closepager)
paul718e3742002-12-13 20:15:29 +0000434 {
435 if (pclose (fp) == -1)
436 {
paula805cc22003-05-01 14:29:48 +0000437 perror ("pclose failed for pager");
paul718e3742002-12-13 20:15:29 +0000438 }
439 fp = NULL;
440 }
441 return;
442 }
443
444 ret = cmd_execute_command (vline, vty, &cmd);
445 cmd_free_strvec (vline);
446 if (ret != CMD_SUCCESS_DAEMON)
447 break;
448 }
449 else
450 if (cmd->func)
451 {
452 (*cmd->func) (cmd, vty, 0, NULL);
453 break;
454 }
455 }
456
457 if (cmd->daemon & VTYSH_ZEBRA)
458 if (vtysh_client_execute (&vtysh_client[VTYSH_INDEX_ZEBRA], line, fp)
459 != CMD_SUCCESS)
460 break;
461 if (cmd->daemon & VTYSH_RIPD)
462 if (vtysh_client_execute (&vtysh_client[VTYSH_INDEX_RIP], line, fp)
463 != CMD_SUCCESS)
464 break;
465 if (cmd->daemon & VTYSH_RIPNGD)
466 if (vtysh_client_execute (&vtysh_client[VTYSH_INDEX_RIPNG], line, fp)
467 != CMD_SUCCESS)
468 break;
469 if (cmd->daemon & VTYSH_OSPFD)
470 if (vtysh_client_execute (&vtysh_client[VTYSH_INDEX_OSPF], line, fp)
471 != CMD_SUCCESS)
472 break;
473 if (cmd->daemon & VTYSH_OSPF6D)
474 if (vtysh_client_execute (&vtysh_client[VTYSH_INDEX_OSPF6], line, fp)
475 != CMD_SUCCESS)
476 break;
477 if (cmd->daemon & VTYSH_BGPD)
478 if (vtysh_client_execute (&vtysh_client[VTYSH_INDEX_BGP], line, fp)
479 != CMD_SUCCESS)
480 break;
481 if (cmd->func)
482 (*cmd->func) (cmd, vty, 0, NULL);
483 }
484 }
paula805cc22003-05-01 14:29:48 +0000485 if (pager && vtysh_pager_name && fp && closepager)
paul718e3742002-12-13 20:15:29 +0000486 {
487 if (pclose (fp) == -1)
488 {
paula805cc22003-05-01 14:29:48 +0000489 perror ("pclose failed for pager");
paul718e3742002-12-13 20:15:29 +0000490 }
491 fp = NULL;
492 }
493}
494
495void
496vtysh_execute_no_pager (char *line)
497{
498 vtysh_execute_func (line, 0);
499}
500
501void
502vtysh_execute (char *line)
503{
504 vtysh_execute_func (line, 1);
505}
506
507/* Configration make from file. */
508int
509vtysh_config_from_file (struct vty *vty, FILE *fp)
510{
511 int ret;
512 vector vline;
513 struct cmd_element *cmd;
514
515 while (fgets (vty->buf, VTY_BUFSIZ, fp))
516 {
517 if (vty->buf[0] == '!' || vty->buf[1] == '#')
518 continue;
519
520 vline = cmd_make_strvec (vty->buf);
521
522 /* In case of comment line */
523 if (vline == NULL)
524 continue;
525
526 /* Execute configuration command : this is strict match */
527 ret = cmd_execute_command_strict (vline, vty, &cmd);
528
529 /* Try again with setting node to CONFIG_NODE */
530 if (ret != CMD_SUCCESS
531 && ret != CMD_SUCCESS_DAEMON
532 && ret != CMD_WARNING)
533 {
534 if (vty->node == KEYCHAIN_KEY_NODE)
535 {
536 vty->node = KEYCHAIN_NODE;
537 vtysh_exit_ripd_only ();
538 ret = cmd_execute_command_strict (vline, vty, &cmd);
539
540 if (ret != CMD_SUCCESS
541 && ret != CMD_SUCCESS_DAEMON
542 && ret != CMD_WARNING)
543 {
544 vtysh_exit_ripd_only ();
545 vty->node = CONFIG_NODE;
546 ret = cmd_execute_command_strict (vline, vty, &cmd);
547 }
548 }
549 else
550 {
551 vtysh_execute ("end");
552 vtysh_execute ("configure terminal");
553 vty->node = CONFIG_NODE;
554 ret = cmd_execute_command_strict (vline, vty, &cmd);
555 }
556 }
557
558 cmd_free_strvec (vline);
559
560 switch (ret)
561 {
562 case CMD_WARNING:
563 if (vty->type == VTY_FILE)
paul4fc01e62002-12-13 20:49:00 +0000564 fprintf (stdout,"Warning...\n");
paul718e3742002-12-13 20:15:29 +0000565 break;
566 case CMD_ERR_AMBIGUOUS:
paul4fc01e62002-12-13 20:49:00 +0000567 fprintf (stdout,"%% Ambiguous command.\n");
paul718e3742002-12-13 20:15:29 +0000568 break;
569 case CMD_ERR_NO_MATCH:
paul4fc01e62002-12-13 20:49:00 +0000570 fprintf (stdout,"%% Unknown command: %s", vty->buf);
paul718e3742002-12-13 20:15:29 +0000571 break;
572 case CMD_ERR_INCOMPLETE:
paul4fc01e62002-12-13 20:49:00 +0000573 fprintf (stdout,"%% Command incomplete.\n");
paul718e3742002-12-13 20:15:29 +0000574 break;
575 case CMD_SUCCESS_DAEMON:
576 {
577 if (cmd->daemon & VTYSH_ZEBRA)
578 if (vtysh_client_execute (&vtysh_client[VTYSH_INDEX_ZEBRA],
579 vty->buf, stdout) != CMD_SUCCESS)
580 break;
581 if (cmd->daemon & VTYSH_RIPD)
582 if (vtysh_client_execute (&vtysh_client[VTYSH_INDEX_RIP],
583 vty->buf, stdout) != CMD_SUCCESS)
584 break;
585 if (cmd->daemon & VTYSH_RIPNGD)
586 if (vtysh_client_execute (&vtysh_client[VTYSH_INDEX_RIPNG],
587 vty->buf, stdout) != CMD_SUCCESS)
588 break;
589 if (cmd->daemon & VTYSH_OSPFD)
590 if (vtysh_client_execute (&vtysh_client[VTYSH_INDEX_OSPF],
591 vty->buf, stdout) != CMD_SUCCESS)
592 break;
593 if (cmd->daemon & VTYSH_OSPF6D)
594 if (vtysh_client_execute (&vtysh_client[VTYSH_INDEX_OSPF6],
595 vty->buf, stdout) != CMD_SUCCESS)
596 break;
597 if (cmd->daemon & VTYSH_BGPD)
598 if (vtysh_client_execute (&vtysh_client[VTYSH_INDEX_BGP],
599 vty->buf, stdout) != CMD_SUCCESS)
600 break;
601 if (cmd->func)
602 (*cmd->func) (cmd, vty, 0, NULL);
603 }
604 }
605 }
606 return CMD_SUCCESS;
607}
608
609/* We don't care about the point of the cursor when '?' is typed. */
610int
611vtysh_rl_describe ()
612{
613 int ret;
614 int i;
615 vector vline;
616 vector describe;
617 int width;
618 struct desc *desc;
619
620 vline = cmd_make_strvec (rl_line_buffer);
621
622 /* In case of '> ?'. */
623 if (vline == NULL)
624 {
625 vline = vector_init (1);
626 vector_set (vline, '\0');
627 }
628 else
629 if (rl_end && isspace ((int) rl_line_buffer[rl_end - 1]))
630 vector_set (vline, '\0');
631
632 describe = cmd_describe_command (vline, vty, &ret);
633
paul4fc01e62002-12-13 20:49:00 +0000634 fprintf (stdout,"\n");
paul718e3742002-12-13 20:15:29 +0000635
636 /* Ambiguous and no match error. */
637 switch (ret)
638 {
639 case CMD_ERR_AMBIGUOUS:
640 cmd_free_strvec (vline);
paul4fc01e62002-12-13 20:49:00 +0000641 fprintf (stdout,"%% Ambiguous command.\n");
paul718e3742002-12-13 20:15:29 +0000642 rl_on_new_line ();
643 return 0;
644 break;
645 case CMD_ERR_NO_MATCH:
646 cmd_free_strvec (vline);
paul4fc01e62002-12-13 20:49:00 +0000647 fprintf (stdout,"%% There is no matched command.\n");
paul718e3742002-12-13 20:15:29 +0000648 rl_on_new_line ();
649 return 0;
650 break;
651 }
652
653 /* Get width of command string. */
654 width = 0;
655 for (i = 0; i < vector_max (describe); i++)
656 if ((desc = vector_slot (describe, i)) != NULL)
657 {
658 int len;
659
660 if (desc->cmd[0] == '\0')
661 continue;
662
663 len = strlen (desc->cmd);
664 if (desc->cmd[0] == '.')
665 len--;
666
667 if (width < len)
668 width = len;
669 }
670
671 for (i = 0; i < vector_max (describe); i++)
672 if ((desc = vector_slot (describe, i)) != NULL)
673 {
674 if (desc->cmd[0] == '\0')
675 continue;
676
677 if (! desc->str)
paul4fc01e62002-12-13 20:49:00 +0000678 fprintf (stdout," %-s\n",
paul718e3742002-12-13 20:15:29 +0000679 desc->cmd[0] == '.' ? desc->cmd + 1 : desc->cmd);
680 else
paul4fc01e62002-12-13 20:49:00 +0000681 fprintf (stdout," %-*s %s\n",
paul718e3742002-12-13 20:15:29 +0000682 width,
683 desc->cmd[0] == '.' ? desc->cmd + 1 : desc->cmd,
684 desc->str);
685 }
686
687 cmd_free_strvec (vline);
688 vector_free (describe);
689
690 rl_on_new_line();
691
692 return 0;
693}
694
695/* result of cmd_complete_command() call will be stored here
696 and used in new_completion() in order to put the space in
697 correct places only */
698int complete_status;
699
700char *
pauldfc0d9b2003-04-18 23:55:29 +0000701command_generator (const char *text, int state)
paul718e3742002-12-13 20:15:29 +0000702{
703 vector vline;
704 static char **matched = NULL;
705 static int index = 0;
706
707 /* First call. */
708 if (! state)
709 {
710 index = 0;
711
712 if (vty->node == AUTH_NODE || vty->node == AUTH_ENABLE_NODE)
713 return NULL;
714
715 vline = cmd_make_strvec (rl_line_buffer);
716 if (vline == NULL)
717 return NULL;
718
719 if (rl_end && isspace ((int) rl_line_buffer[rl_end - 1]))
720 vector_set (vline, '\0');
721
722 matched = cmd_complete_command (vline, vty, &complete_status);
723 }
724
725 if (matched && matched[index])
726 return matched[index++];
727
728 return NULL;
729}
730
731char **
732new_completion (char *text, int start, int end)
733{
734 char **matches;
735
pauldfc0d9b2003-04-18 23:55:29 +0000736 matches = rl_completion_matches (text, command_generator);
paul718e3742002-12-13 20:15:29 +0000737
738 if (matches)
739 {
740 rl_point = rl_end;
741 if (complete_status == CMD_COMPLETE_FULL_MATCH)
742 rl_pending_input = ' ';
743 }
744
745 return matches;
746}
747
748char **
749vtysh_completion (char *text, int start, int end)
750{
751 int ret;
752 vector vline;
753 char **matched = NULL;
754
755 if (vty->node == AUTH_NODE || vty->node == AUTH_ENABLE_NODE)
756 return NULL;
757
758 vline = cmd_make_strvec (rl_line_buffer);
759 if (vline == NULL)
760 return NULL;
761
762 /* In case of 'help \t'. */
763 if (rl_end && isspace ((int) rl_line_buffer[rl_end - 1]))
764 vector_set (vline, '\0');
765
766 matched = cmd_complete_command (vline, vty, &ret);
767
768 cmd_free_strvec (vline);
769
770 return (char **) matched;
771}
772
773/* BGP node structure. */
774struct cmd_node bgp_node =
775{
776 BGP_NODE,
777 "%s(config-router)# ",
778};
779
780/* BGP node structure. */
781struct cmd_node rip_node =
782{
783 RIP_NODE,
784 "%s(config-router)# ",
785};
786
787struct cmd_node interface_node =
788{
789 INTERFACE_NODE,
790 "%s(config-if)# ",
791};
792
793DEFUNSH (VTYSH_BGPD,
794 router_bgp,
795 router_bgp_cmd,
796 "router bgp <1-65535>",
797 ROUTER_STR
798 BGP_STR
799 AS_STR)
800{
801 vty->node = BGP_NODE;
802 return CMD_SUCCESS;
803}
804
805DEFUNSH (VTYSH_BGPD,
806 address_family_vpnv4,
807 address_family_vpnv4_cmd,
808 "address-family vpnv4",
809 "Enter Address Family command mode\n"
810 "Address family\n")
811{
812 vty->node = BGP_VPNV4_NODE;
813 return CMD_SUCCESS;
814}
815
816DEFUNSH (VTYSH_BGPD,
817 address_family_vpnv4_unicast,
818 address_family_vpnv4_unicast_cmd,
819 "address-family vpnv4 unicast",
820 "Enter Address Family command mode\n"
821 "Address family\n"
822 "Address Family Modifier\n")
823{
824 vty->node = BGP_VPNV4_NODE;
825 return CMD_SUCCESS;
826}
827
828DEFUNSH (VTYSH_BGPD,
829 address_family_ipv4_unicast,
830 address_family_ipv4_unicast_cmd,
831 "address-family ipv4 unicast",
832 "Enter Address Family command mode\n"
833 "Address family\n"
834 "Address Family Modifier\n")
835{
836 vty->node = BGP_IPV4_NODE;
837 return CMD_SUCCESS;
838}
839
840DEFUNSH (VTYSH_BGPD,
841 address_family_ipv4_multicast,
842 address_family_ipv4_multicast_cmd,
843 "address-family ipv4 multicast",
844 "Enter Address Family command mode\n"
845 "Address family\n"
846 "Address Family Modifier\n")
847{
848 vty->node = BGP_IPV4M_NODE;
849 return CMD_SUCCESS;
850}
851
852DEFUNSH (VTYSH_BGPD,
853 address_family_ipv6,
854 address_family_ipv6_cmd,
855 "address-family ipv6",
856 "Enter Address Family command mode\n"
857 "Address family\n")
858{
859 vty->node = BGP_IPV6_NODE;
860 return CMD_SUCCESS;
861}
862
863DEFUNSH (VTYSH_BGPD,
864 address_family_ipv6_unicast,
865 address_family_ipv6_unicast_cmd,
866 "address-family ipv6 unicast",
867 "Enter Address Family command mode\n"
868 "Address family\n"
869 "Address Family Modifier\n")
870{
871 vty->node = BGP_IPV6_NODE;
872 return CMD_SUCCESS;
873}
874
875DEFUNSH (VTYSH_RIPD,
876 key_chain,
877 key_chain_cmd,
878 "key chain WORD",
879 "Authentication key management\n"
880 "Key-chain management\n"
881 "Key-chain name\n")
882{
883 vty->node = KEYCHAIN_NODE;
884 return CMD_SUCCESS;
885}
886
887DEFUNSH (VTYSH_RIPD,
888 key,
889 key_cmd,
890 "key <0-2147483647>",
891 "Configure a key\n"
892 "Key identifier number\n")
893{
894 vty->node = KEYCHAIN_KEY_NODE;
895 return CMD_SUCCESS;
896}
897
898DEFUNSH (VTYSH_RIPD,
899 router_rip,
900 router_rip_cmd,
901 "router rip",
902 ROUTER_STR
903 "RIP")
904{
905 vty->node = RIP_NODE;
906 return CMD_SUCCESS;
907}
908
909DEFUNSH (VTYSH_RIPNGD,
910 router_ripng,
911 router_ripng_cmd,
912 "router ripng",
913 ROUTER_STR
914 "RIPng")
915{
916 vty->node = RIPNG_NODE;
917 return CMD_SUCCESS;
918}
919
920DEFUNSH (VTYSH_OSPFD,
921 router_ospf,
922 router_ospf_cmd,
923 "router ospf",
924 "Enable a routing process\n"
925 "Start OSPF configuration\n")
926{
927 vty->node = OSPF_NODE;
928 return CMD_SUCCESS;
929}
930
931DEFUNSH (VTYSH_OSPF6D,
932 router_ospf6,
933 router_ospf6_cmd,
934 "router ospf6",
935 OSPF6_ROUTER_STR
936 OSPF6_STR)
937{
938 vty->node = OSPF6_NODE;
939 return CMD_SUCCESS;
940}
941
942DEFUNSH (VTYSH_RMAP,
943 route_map,
944 route_map_cmd,
945 "route-map WORD (deny|permit) <1-65535>",
946 "Create route-map or enter route-map command mode\n"
947 "Route map tag\n"
948 "Route map denies set operations\n"
949 "Route map permits set operations\n"
950 "Sequence to insert to/delete from existing route-map entry\n")
951{
952 vty->node = RMAP_NODE;
953 return CMD_SUCCESS;
954}
955
956/* Enable command */
957DEFUNSH (VTYSH_ALL,
958 vtysh_enable,
959 vtysh_enable_cmd,
960 "enable",
961 "Turn on privileged mode command\n")
962{
963 vty->node = ENABLE_NODE;
964 return CMD_SUCCESS;
965}
966
967/* Disable command */
968DEFUNSH (VTYSH_ALL,
969 vtysh_disable,
970 vtysh_disable_cmd,
971 "disable",
972 "Turn off privileged mode command\n")
973{
974 if (vty->node == ENABLE_NODE)
975 vty->node = VIEW_NODE;
976 return CMD_SUCCESS;
977}
978
979/* Configration from terminal */
980DEFUNSH (VTYSH_ALL,
981 vtysh_config_terminal,
982 vtysh_config_terminal_cmd,
983 "configure terminal",
984 "Configuration from vty interface\n"
985 "Configuration terminal\n")
986{
987 vty->node = CONFIG_NODE;
988 return CMD_SUCCESS;
989}
990
991int
992vtysh_exit (struct vty *vty)
993{
994 switch (vty->node)
995 {
996 case VIEW_NODE:
997 case ENABLE_NODE:
998 exit (0);
999 break;
1000 case CONFIG_NODE:
1001 vty->node = ENABLE_NODE;
1002 break;
1003 case INTERFACE_NODE:
1004 case ZEBRA_NODE:
1005 case BGP_NODE:
1006 case RIP_NODE:
1007 case RIPNG_NODE:
1008 case OSPF_NODE:
1009 case OSPF6_NODE:
1010 case MASC_NODE:
1011 case RMAP_NODE:
1012 case VTY_NODE:
1013 case KEYCHAIN_NODE:
1014 vty->node = CONFIG_NODE;
1015 break;
1016 case BGP_VPNV4_NODE:
1017 case BGP_IPV4_NODE:
1018 case BGP_IPV4M_NODE:
1019 case BGP_IPV6_NODE:
1020 vty->node = BGP_NODE;
1021 break;
1022 case KEYCHAIN_KEY_NODE:
1023 vty->node = KEYCHAIN_NODE;
1024 break;
1025 default:
1026 break;
1027 }
1028 return CMD_SUCCESS;
1029}
1030
1031DEFUNSH (VTYSH_ALL,
1032 vtysh_exit_all,
1033 vtysh_exit_all_cmd,
1034 "exit",
1035 "Exit current mode and down to previous mode\n")
1036{
1037 return vtysh_exit (vty);
1038}
1039
1040ALIAS (vtysh_exit_all,
1041 vtysh_quit_all_cmd,
1042 "quit",
1043 "Exit current mode and down to previous mode\n")
1044
1045DEFUNSH (VTYSH_BGPD,
1046 exit_address_family,
1047 exit_address_family_cmd,
1048 "exit-address-family",
1049 "Exit from Address Family configuration mode\n")
1050{
1051 if (vty->node == BGP_IPV4_NODE
1052 || vty->node == BGP_IPV4M_NODE
1053 || vty->node == BGP_VPNV4_NODE
1054 || vty->node == BGP_IPV6_NODE)
1055 vty->node = BGP_NODE;
1056 return CMD_SUCCESS;
1057}
1058
1059DEFUNSH (VTYSH_ZEBRA,
1060 vtysh_exit_zebra,
1061 vtysh_exit_zebra_cmd,
1062 "exit",
1063 "Exit current mode and down to previous mode\n")
1064{
1065 return vtysh_exit (vty);
1066}
1067
1068ALIAS (vtysh_exit_zebra,
1069 vtysh_quit_zebra_cmd,
1070 "quit",
1071 "Exit current mode and down to previous mode\n")
1072
1073DEFUNSH (VTYSH_RIPD,
1074 vtysh_exit_ripd,
1075 vtysh_exit_ripd_cmd,
1076 "exit",
1077 "Exit current mode and down to previous mode\n")
1078{
1079 return vtysh_exit (vty);
1080}
1081
1082ALIAS (vtysh_exit_ripd,
1083 vtysh_quit_ripd_cmd,
1084 "quit",
1085 "Exit current mode and down to previous mode\n")
1086
paul68980082003-03-25 05:07:42 +00001087DEFUNSH (VTYSH_RIPNGD,
1088 vtysh_exit_ripngd,
1089 vtysh_exit_ripngd_cmd,
1090 "exit",
1091 "Exit current mode and down to previous mode\n")
1092{
1093 return vtysh_exit (vty);
1094}
1095
1096ALIAS (vtysh_exit_ripngd,
1097 vtysh_quit_ripngd_cmd,
1098 "quit",
1099 "Exit current mode and down to previous mode\n")
1100
paul718e3742002-12-13 20:15:29 +00001101DEFUNSH (VTYSH_RMAP,
1102 vtysh_exit_rmap,
1103 vtysh_exit_rmap_cmd,
1104 "exit",
1105 "Exit current mode and down to previous mode\n")
1106{
1107 return vtysh_exit (vty);
1108}
1109
1110ALIAS (vtysh_exit_rmap,
1111 vtysh_quit_rmap_cmd,
1112 "quit",
1113 "Exit current mode and down to previous mode\n")
1114
1115DEFUNSH (VTYSH_BGPD,
1116 vtysh_exit_bgpd,
1117 vtysh_exit_bgpd_cmd,
1118 "exit",
1119 "Exit current mode and down to previous mode\n")
1120{
1121 return vtysh_exit (vty);
1122}
1123
1124ALIAS (vtysh_exit_bgpd,
1125 vtysh_quit_bgpd_cmd,
1126 "quit",
1127 "Exit current mode and down to previous mode\n")
1128
1129DEFUNSH (VTYSH_OSPFD,
1130 vtysh_exit_ospfd,
1131 vtysh_exit_ospfd_cmd,
1132 "exit",
1133 "Exit current mode and down to previous mode\n")
1134{
1135 return vtysh_exit (vty);
1136}
1137
1138ALIAS (vtysh_exit_ospfd,
1139 vtysh_quit_ospfd_cmd,
1140 "quit",
1141 "Exit current mode and down to previous mode\n")
1142
paul68980082003-03-25 05:07:42 +00001143DEFUNSH (VTYSH_OSPF6D,
1144 vtysh_exit_ospf6d,
1145 vtysh_exit_ospf6d_cmd,
1146 "exit",
1147 "Exit current mode and down to previous mode\n")
1148{
1149 return vtysh_exit (vty);
1150}
1151
1152ALIAS (vtysh_exit_ospf6d,
1153 vtysh_quit_ospf6d_cmd,
1154 "quit",
1155 "Exit current mode and down to previous mode\n")
1156
paul111bd7a2003-05-23 10:43:59 +00001157DEFUNSH (VTYSH_ZEBRA|VTYSH_RIPD|VTYSH_RIPNGD|VTYSH_OSPFD|VTYSH_OSPF6D,
paul718e3742002-12-13 20:15:29 +00001158 vtysh_interface,
1159 vtysh_interface_cmd,
1160 "interface IFNAME",
1161 "Select an interface to configure\n"
1162 "Interface's name\n")
1163{
1164 vty->node = INTERFACE_NODE;
1165 return CMD_SUCCESS;
1166}
1167
paul32d24632003-05-23 09:25:20 +00001168DEFSH (VTYSH_ZEBRA|VTYSH_RIPD|VTYSH_RIPNGD|VTYSH_OSPFD|VTYSH_OSPF6D,
1169 vtysh_no_interface_cmd,
1170 "no interface IFNAME",
1171 NO_STR
1172 "Delete a pseudo interface's configuration\n"
1173 "Interface's name\n")
1174
paul338a9912003-03-01 15:44:10 +00001175DEFSH (VTYSH_ZEBRA|VTYSH_RIPD|VTYSH_OSPFD,
1176 interface_desc_cmd,
1177 "description .LINE",
1178 "Interface specific description\n"
1179 "Characters describing this interface\n")
paul464dc8d2003-03-28 02:25:45 +00001180
1181DEFSH (VTYSH_ZEBRA|VTYSH_RIPD|VTYSH_OSPFD,
1182 no_interface_desc_cmd,
1183 "no description",
1184 NO_STR
1185 "Interface specific description\n")
paul338a9912003-03-01 15:44:10 +00001186
paul111bd7a2003-05-23 10:43:59 +00001187DEFUNSH (VTYSH_ZEBRA|VTYSH_RIPD|VTYSH_RIPNGD|VTYSH_OSPFD|VTYSH_OSPF6D,
paul718e3742002-12-13 20:15:29 +00001188 vtysh_exit_interface,
1189 vtysh_exit_interface_cmd,
1190 "exit",
1191 "Exit current mode and down to previous mode\n")
1192{
1193 return vtysh_exit (vty);
1194}
1195
1196ALIAS (vtysh_exit_interface,
1197 vtysh_quit_interface_cmd,
1198 "quit",
1199 "Exit current mode and down to previous mode\n")
1200
1201DEFUN (vtysh_write_terminal,
1202 vtysh_write_terminal_cmd,
1203 "write terminal",
1204 "Write running configuration to memory, network, or terminal\n"
1205 "Write to terminal\n")
1206{
1207 int ret;
1208 char line[] = "write terminal\n";
1209 FILE *fp = NULL;
1210
1211 if (vtysh_pager_name)
1212 {
paul4fc01e62002-12-13 20:49:00 +00001213 fp = popen (vtysh_pager_name, "w");
paul718e3742002-12-13 20:15:29 +00001214 if (fp == NULL)
1215 {
1216 perror ("popen");
1217 exit (1);
1218 }
1219 }
1220 else
1221 fp = stdout;
1222
1223 vty_out (vty, "Building configuration...%s", VTY_NEWLINE);
1224 vty_out (vty, "%sCurrent configuration:%s", VTY_NEWLINE,
1225 VTY_NEWLINE);
1226
1227 vtysh_config_write (fp);
1228
1229 ret = vtysh_client_config (&vtysh_client[VTYSH_INDEX_ZEBRA], line);
1230 ret = vtysh_client_config (&vtysh_client[VTYSH_INDEX_RIP], line);
1231 ret = vtysh_client_config (&vtysh_client[VTYSH_INDEX_RIPNG], line);
1232 ret = vtysh_client_config (&vtysh_client[VTYSH_INDEX_OSPF], line);
1233 ret = vtysh_client_config (&vtysh_client[VTYSH_INDEX_OSPF6], line);
1234 ret = vtysh_client_config (&vtysh_client[VTYSH_INDEX_BGP], line);
1235
1236 vtysh_config_dump (fp);
1237
1238 if (vtysh_pager_name && fp)
1239 {
1240 fflush (fp);
1241 if (pclose (fp) == -1)
1242 {
1243 perror ("pclose");
1244 exit (1);
1245 }
1246 fp = NULL;
1247 }
1248
1249 return CMD_SUCCESS;
1250}
1251
paul4fc01e62002-12-13 20:49:00 +00001252struct vtysh_writeconfig_t {
1253 int daemon;
1254 int integrated;
1255} vtysh_wc = {-1,0};
1256
1257DEFUN (vtysh_write_config,
1258 vtysh_write_config_cmd,
1259 "write-config (daemon|integrated)",
1260 "Specify config files to write to\n"
1261 "Write per daemon file\n"
1262 "Write integrated file\n"
1263)
1264{
1265 if (!strncmp(argv[0],"d",1)) {
1266 vtysh_wc.daemon = 1;
1267 } else if (!strncmp(argv[0],"i",1)) {
1268 vtysh_wc.integrated = 1;
1269 }
1270 return CMD_SUCCESS;
1271}
1272
1273DEFUN (no_vtysh_write_config,
1274 no_vtysh_write_config_cmd,
1275 "no write-config (daemon|integrated)",
1276 "Negate per daemon and/or integrated config files\n"
1277)
1278{
1279 if (!strncmp(argv[0],"d",1)) {
1280 vtysh_wc.daemon = 0;
1281 } else if (!strncmp(argv[0],"i",1)) {
1282 vtysh_wc.integrated = 0;
1283 }
1284 return CMD_SUCCESS;
1285}
1286
1287int write_config_integrated(void)
paul718e3742002-12-13 20:15:29 +00001288{
1289 int ret;
1290 mode_t old_umask;
1291 char line[] = "write terminal\n";
1292 FILE *fp;
1293 char *integrate_sav = NULL;
1294
1295 /* config files have 0600 perms... */
1296 old_umask = umask (0077);
1297
1298 integrate_sav = malloc (strlen (integrate_default)
1299 + strlen (CONF_BACKUP_EXT) + 1);
1300 strcpy (integrate_sav, integrate_default);
1301 strcat (integrate_sav, CONF_BACKUP_EXT);
1302
1303
paul4fc01e62002-12-13 20:49:00 +00001304 fprintf (stdout,"Building Configuration...\n");
paul718e3742002-12-13 20:15:29 +00001305
1306 /* Move current configuration file to backup config file */
1307 unlink (integrate_sav);
1308 rename (integrate_default, integrate_sav);
paul5087df52003-01-25 06:56:09 +00001309 free (integrate_sav);
paul4fc01e62002-12-13 20:49:00 +00001310
paul718e3742002-12-13 20:15:29 +00001311 fp = fopen (integrate_default, "w");
1312 if (fp == NULL)
1313 {
paul4fc01e62002-12-13 20:49:00 +00001314 fprintf (stdout,"%% Can't open configuration file %s.\n", integrate_default);
paul718e3742002-12-13 20:15:29 +00001315 umask (old_umask);
1316 return CMD_SUCCESS;
1317 }
paul718e3742002-12-13 20:15:29 +00001318
1319 vtysh_config_write (fp);
1320
1321 ret = vtysh_client_config (&vtysh_client[VTYSH_INDEX_ZEBRA], line);
1322 ret = vtysh_client_config (&vtysh_client[VTYSH_INDEX_RIP], line);
1323 ret = vtysh_client_config (&vtysh_client[VTYSH_INDEX_RIPNG], line);
1324 ret = vtysh_client_config (&vtysh_client[VTYSH_INDEX_OSPF], line);
1325 ret = vtysh_client_config (&vtysh_client[VTYSH_INDEX_OSPF6], line);
1326 ret = vtysh_client_config (&vtysh_client[VTYSH_INDEX_BGP], line);
1327
1328 vtysh_config_dump (fp);
1329
1330 fclose (fp);
1331
paul4fc01e62002-12-13 20:49:00 +00001332 fprintf(stdout,"Integrated configuration saved to %s\n",integrate_default);
1333
1334 fprintf (stdout,"[OK]\n");
1335
paul718e3742002-12-13 20:15:29 +00001336 umask (old_umask);
1337 return CMD_SUCCESS;
1338}
1339
paul4fc01e62002-12-13 20:49:00 +00001340DEFUN (vtysh_write_memory,
1341 vtysh_write_memory_cmd,
1342 "write memory",
1343 "Write running configuration to memory, network, or terminal\n"
1344 "Write configuration to the file (same as write file)\n")
1345{
pauldfc0d9b2003-04-18 23:55:29 +00001346 int ret = CMD_SUCCESS;
paul4fc01e62002-12-13 20:49:00 +00001347 char line[] = "write memory\n";
1348
1349 /* if integrated Zebra.conf explicitely set */
1350 if (vtysh_wc.integrated == 1) {
1351 ret = write_config_integrated();
1352 }
1353
1354 if (!vtysh_wc.daemon) {
1355 return ret;
1356 }
1357
1358 fprintf (stdout,"Building Configuration...\n");
1359
1360 ret = vtysh_client_execute (&vtysh_client[VTYSH_INDEX_ZEBRA], line, stdout);
1361 ret = vtysh_client_execute (&vtysh_client[VTYSH_INDEX_RIP], line, stdout);
1362 ret = vtysh_client_execute (&vtysh_client[VTYSH_INDEX_RIPNG], line, stdout);
1363 ret = vtysh_client_execute (&vtysh_client[VTYSH_INDEX_OSPF], line, stdout);
1364 ret = vtysh_client_execute (&vtysh_client[VTYSH_INDEX_OSPF6], line, stdout);
1365 ret = vtysh_client_execute (&vtysh_client[VTYSH_INDEX_BGP], line, stdout);
1366
1367 fprintf (stdout,"[OK]\n");
1368
pauldfc0d9b2003-04-18 23:55:29 +00001369 return ret;
paul4fc01e62002-12-13 20:49:00 +00001370}
1371
paul718e3742002-12-13 20:15:29 +00001372ALIAS (vtysh_write_memory,
1373 vtysh_copy_runningconfig_startupconfig_cmd,
1374 "copy running-config startup-config",
1375 "Copy from one file to another\n"
1376 "Copy from current system configuration\n"
1377 "Copy to startup configuration\n")
1378
1379ALIAS (vtysh_write_memory,
1380 vtysh_write_file_cmd,
1381 "write file",
1382 "Write running configuration to memory, network, or terminal\n"
1383 "Write configuration to the file (same as write memory)\n")
1384
1385ALIAS (vtysh_write_terminal,
1386 vtysh_show_running_config_cmd,
1387 "show running-config",
1388 SHOW_STR
1389 "Current operating configuration\n")
1390
1391/* Execute command in child process. */
1392int
1393execute_command (char *command, int argc, char *arg1, char *arg2)
1394{
1395 int ret;
1396 pid_t pid;
1397 int status;
1398
1399 /* Call fork(). */
1400 pid = fork ();
1401
1402 if (pid < 0)
1403 {
1404 /* Failure of fork(). */
1405 fprintf (stderr, "Can't fork: %s\n", strerror (errno));
1406 exit (1);
1407 }
1408 else if (pid == 0)
1409 {
1410 /* This is child process. */
1411 switch (argc)
1412 {
1413 case 0:
1414 ret = execlp (command, command, NULL);
1415 break;
1416 case 1:
1417 ret = execlp (command, command, arg1, NULL);
1418 break;
1419 case 2:
1420 ret = execlp (command, command, arg1, arg2, NULL);
1421 break;
1422 }
1423
1424 /* When execlp suceed, this part is not executed. */
1425 fprintf (stderr, "Can't execute %s: %s\n", command, strerror (errno));
1426 exit (1);
1427 }
1428 else
1429 {
1430 /* This is parent. */
1431 execute_flag = 1;
1432 ret = wait4 (pid, &status, 0, NULL);
1433 execute_flag = 0;
1434 }
1435 return 0;
1436}
1437
1438DEFUN (vtysh_ping,
1439 vtysh_ping_cmd,
1440 "ping WORD",
1441 "send echo messages\n"
1442 "Ping destination address or hostname\n")
1443{
1444 execute_command ("ping", 1, argv[0], NULL);
1445 return CMD_SUCCESS;
1446}
1447
1448DEFUN (vtysh_traceroute,
1449 vtysh_traceroute_cmd,
1450 "traceroute WORD",
1451 "Trace route to destination\n"
1452 "Trace route to destination address or hostname\n")
1453{
1454 execute_command ("traceroute", 1, argv[0], NULL);
1455 return CMD_SUCCESS;
1456}
1457
1458DEFUN (vtysh_telnet,
1459 vtysh_telnet_cmd,
1460 "telnet WORD",
1461 "Open a telnet connection\n"
1462 "IP address or hostname of a remote system\n")
1463{
1464 execute_command ("telnet", 1, argv[0], NULL);
1465 return CMD_SUCCESS;
1466}
1467
1468DEFUN (vtysh_telnet_port,
1469 vtysh_telnet_port_cmd,
1470 "telnet WORD PORT",
1471 "Open a telnet connection\n"
1472 "IP address or hostname of a remote system\n"
1473 "TCP Port number\n")
1474{
1475 execute_command ("telnet", 2, argv[0], argv[1]);
1476 return CMD_SUCCESS;
1477}
1478
paul5087df52003-01-25 06:56:09 +00001479DEFUN (vtysh_ssh,
1480 vtysh_ssh_cmd,
1481 "ssh WORD",
1482 "Open an ssh connection\n"
1483 "[user@]host\n")
1484{
1485 execute_command ("ssh", 1, argv[0], NULL);
1486 return CMD_SUCCESS;
1487}
1488
paul718e3742002-12-13 20:15:29 +00001489DEFUN (vtysh_start_shell,
1490 vtysh_start_shell_cmd,
1491 "start-shell",
1492 "Start UNIX shell\n")
1493{
1494 execute_command ("sh", 0, NULL, NULL);
1495 return CMD_SUCCESS;
1496}
1497
1498DEFUN (vtysh_start_bash,
1499 vtysh_start_bash_cmd,
1500 "start-shell bash",
1501 "Start UNIX shell\n"
1502 "Start bash\n")
1503{
1504 execute_command ("bash", 0, NULL, NULL);
1505 return CMD_SUCCESS;
1506}
1507
1508DEFUN (vtysh_start_zsh,
1509 vtysh_start_zsh_cmd,
1510 "start-shell zsh",
1511 "Start UNIX shell\n"
1512 "Start Z shell\n")
1513{
1514 execute_command ("zsh", 0, NULL, NULL);
1515 return CMD_SUCCESS;
1516}
1517
1518/* Route map node structure. */
1519struct cmd_node rmap_node =
1520{
1521 RMAP_NODE,
1522 "%s(config-route-map)# "
1523};
1524
1525/* Zebra node structure. */
1526struct cmd_node zebra_node =
1527{
1528 ZEBRA_NODE,
1529 "%s(config-router)# "
1530};
1531
1532struct cmd_node bgp_vpnv4_node =
1533{
1534 BGP_VPNV4_NODE,
1535 "%s(config-router-af)# "
1536};
1537
1538struct cmd_node bgp_ipv4_node =
1539{
1540 BGP_IPV4_NODE,
1541 "%s(config-router-af)# "
1542};
1543
1544struct cmd_node bgp_ipv4m_node =
1545{
1546 BGP_IPV4M_NODE,
1547 "%s(config-router-af)# "
1548};
1549
1550struct cmd_node bgp_ipv6_node =
1551{
1552 BGP_IPV6_NODE,
1553 "%s(config-router-af)# "
1554};
1555
1556struct cmd_node ospf_node =
1557{
1558 OSPF_NODE,
1559 "%s(config-router)# "
1560};
1561
1562/* RIPng node structure. */
1563struct cmd_node ripng_node =
1564{
1565 RIPNG_NODE,
1566 "%s(config-router)# "
1567};
1568
1569/* OSPF6 node structure. */
1570struct cmd_node ospf6_node =
1571{
1572 OSPF6_NODE,
1573 "%s(config-ospf6)# "
1574};
1575
1576struct cmd_node keychain_node =
1577{
1578 KEYCHAIN_NODE,
1579 "%s(config-keychain)# "
1580};
1581
1582struct cmd_node keychain_key_node =
1583{
1584 KEYCHAIN_KEY_NODE,
1585 "%s(config-keychain-key)# "
1586};
1587
1588void
1589vtysh_install_default (enum node_type node)
1590{
1591 install_element (node, &config_list_cmd);
1592}
1593
1594/* Making connection to protocol daemon. */
1595int
1596vtysh_connect (struct vtysh_client *vclient, char *path)
1597{
1598 int ret;
1599 int sock, len;
1600 struct sockaddr_un addr;
1601 struct stat s_stat;
1602 uid_t euid;
1603 gid_t egid;
1604
1605 memset (vclient, 0, sizeof (struct vtysh_client));
1606 vclient->fd = -1;
1607
1608 /* Stat socket to see if we have permission to access it. */
1609 euid = geteuid();
1610 egid = getegid();
1611 ret = stat (path, &s_stat);
1612 if (ret < 0 && errno != ENOENT)
1613 {
1614 fprintf (stderr, "vtysh_connect(%s): stat = %s\n",
1615 path, strerror(errno));
1616 exit(1);
1617 }
1618
1619 if (ret >= 0)
1620 {
1621 if (! S_ISSOCK(s_stat.st_mode))
1622 {
1623 fprintf (stderr, "vtysh_connect(%s): Not a socket\n",
1624 path);
1625 exit (1);
1626 }
1627
paul718e3742002-12-13 20:15:29 +00001628 }
1629
1630 sock = socket (AF_UNIX, SOCK_STREAM, 0);
1631 if (sock < 0)
1632 {
1633#ifdef DEBUG
1634 fprintf(stderr, "vtysh_connect(%s): socket = %s\n", path, strerror(errno));
1635#endif /* DEBUG */
1636 return -1;
1637 }
1638
1639 memset (&addr, 0, sizeof (struct sockaddr_un));
1640 addr.sun_family = AF_UNIX;
1641 strncpy (addr.sun_path, path, strlen (path));
1642#ifdef HAVE_SUN_LEN
1643 len = addr.sun_len = SUN_LEN(&addr);
1644#else
1645 len = sizeof (addr.sun_family) + strlen (addr.sun_path);
1646#endif /* HAVE_SUN_LEN */
1647
1648 ret = connect (sock, (struct sockaddr *) &addr, len);
1649 if (ret < 0)
1650 {
1651#ifdef DEBUG
1652 fprintf(stderr, "vtysh_connect(%s): connect = %s\n", path, strerror(errno));
1653#endif /* DEBUG */
1654 close (sock);
1655 return -1;
1656 }
1657 vclient->fd = sock;
1658
1659 return 0;
1660}
1661
1662void
1663vtysh_connect_all()
1664{
1665 /* Clear each daemons client structure. */
paulfe067782003-04-07 16:10:05 +00001666 vtysh_connect (&vtysh_client[VTYSH_INDEX_ZEBRA], ZEBRA_VTYSH_PATH);
1667 vtysh_connect (&vtysh_client[VTYSH_INDEX_RIP], RIP_VTYSH_PATH);
1668 vtysh_connect (&vtysh_client[VTYSH_INDEX_RIPNG], RIPNG_VTYSH_PATH);
1669 vtysh_connect (&vtysh_client[VTYSH_INDEX_OSPF], OSPF_VTYSH_PATH);
1670 vtysh_connect (&vtysh_client[VTYSH_INDEX_OSPF6], OSPF6_VTYSH_PATH);
1671 vtysh_connect (&vtysh_client[VTYSH_INDEX_BGP], BGP_VTYSH_PATH);
paul718e3742002-12-13 20:15:29 +00001672}
1673
1674
1675/* To disable readline's filename completion */
pauldfc0d9b2003-04-18 23:55:29 +00001676char *
1677vtysh_completion_entry_function (const char *ignore, int invoking_key)
paul718e3742002-12-13 20:15:29 +00001678{
pauldfc0d9b2003-04-18 23:55:29 +00001679 return NULL;
paul718e3742002-12-13 20:15:29 +00001680}
1681
1682void
1683vtysh_readline_init ()
1684{
1685 /* readline related settings. */
1686 rl_bind_key ('?', vtysh_rl_describe);
paul68980082003-03-25 05:07:42 +00001687 rl_completion_entry_function = vtysh_completion_entry_function;
paul718e3742002-12-13 20:15:29 +00001688 rl_attempted_completion_function = (CPPFunction *)new_completion;
1689 /* do not append space after completion. It will be appended
1690 in new_completion() function explicitly */
1691 rl_completion_append_character = '\0';
1692}
1693
1694char *
1695vtysh_prompt ()
1696{
1697 struct utsname names;
1698 static char buf[100];
1699 const char*hostname;
1700 extern struct host host;
1701
1702 hostname = host.name;
1703
1704 if (!hostname)
1705 {
1706 uname (&names);
1707 hostname = names.nodename;
1708 }
1709
1710 snprintf (buf, sizeof buf, cmd_prompt (vty->node), hostname);
1711
1712 return buf;
1713}
1714
1715void
1716vtysh_init_vty ()
1717{
1718 /* Make vty structure. */
1719 vty = vty_new ();
1720 vty->type = VTY_SHELL;
1721 vty->node = VIEW_NODE;
1722
1723 /* Initialize commands. */
1724 cmd_init (0);
1725
1726 /* Install nodes. */
1727 install_node (&bgp_node, NULL);
1728 install_node (&rip_node, NULL);
1729 install_node (&interface_node, NULL);
1730 install_node (&rmap_node, NULL);
1731 install_node (&zebra_node, NULL);
1732 install_node (&bgp_vpnv4_node, NULL);
1733 install_node (&bgp_ipv4_node, NULL);
1734 install_node (&bgp_ipv4m_node, NULL);
1735/* #ifdef HAVE_IPV6 */
1736 install_node (&bgp_ipv6_node, NULL);
1737/* #endif */
1738 install_node (&ospf_node, NULL);
1739/* #ifdef HAVE_IPV6 */
1740 install_node (&ripng_node, NULL);
1741 install_node (&ospf6_node, NULL);
1742/* #endif */
1743 install_node (&keychain_node, NULL);
1744 install_node (&keychain_key_node, NULL);
1745
1746 vtysh_install_default (VIEW_NODE);
1747 vtysh_install_default (ENABLE_NODE);
1748 vtysh_install_default (CONFIG_NODE);
1749 vtysh_install_default (BGP_NODE);
1750 vtysh_install_default (RIP_NODE);
1751 vtysh_install_default (INTERFACE_NODE);
1752 vtysh_install_default (RMAP_NODE);
1753 vtysh_install_default (ZEBRA_NODE);
1754 vtysh_install_default (BGP_VPNV4_NODE);
1755 vtysh_install_default (BGP_IPV4_NODE);
1756 vtysh_install_default (BGP_IPV4M_NODE);
1757 vtysh_install_default (BGP_IPV6_NODE);
1758 vtysh_install_default (OSPF_NODE);
1759 vtysh_install_default (RIPNG_NODE);
1760 vtysh_install_default (OSPF6_NODE);
1761 vtysh_install_default (KEYCHAIN_NODE);
1762 vtysh_install_default (KEYCHAIN_KEY_NODE);
1763
1764 install_element (VIEW_NODE, &vtysh_enable_cmd);
1765 install_element (ENABLE_NODE, &vtysh_config_terminal_cmd);
1766 install_element (ENABLE_NODE, &vtysh_disable_cmd);
1767
1768 /* "exit" command. */
1769 install_element (VIEW_NODE, &vtysh_exit_all_cmd);
1770 install_element (VIEW_NODE, &vtysh_quit_all_cmd);
1771 install_element (CONFIG_NODE, &vtysh_exit_all_cmd);
1772 /* install_element (CONFIG_NODE, &vtysh_quit_all_cmd); */
1773 install_element (ENABLE_NODE, &vtysh_exit_all_cmd);
1774 install_element (ENABLE_NODE, &vtysh_quit_all_cmd);
1775 install_element (RIP_NODE, &vtysh_exit_ripd_cmd);
1776 install_element (RIP_NODE, &vtysh_quit_ripd_cmd);
paul68980082003-03-25 05:07:42 +00001777 install_element (RIPNG_NODE, &vtysh_exit_ripngd_cmd);
1778 install_element (RIPNG_NODE, &vtysh_quit_ripngd_cmd);
paul718e3742002-12-13 20:15:29 +00001779 install_element (OSPF_NODE, &vtysh_exit_ospfd_cmd);
1780 install_element (OSPF_NODE, &vtysh_quit_ospfd_cmd);
paul68980082003-03-25 05:07:42 +00001781 install_element (OSPF6_NODE, &vtysh_exit_ospf6d_cmd);
1782 install_element (OSPF6_NODE, &vtysh_quit_ospf6d_cmd);
paul718e3742002-12-13 20:15:29 +00001783 install_element (BGP_NODE, &vtysh_exit_bgpd_cmd);
1784 install_element (BGP_NODE, &vtysh_quit_bgpd_cmd);
1785 install_element (BGP_VPNV4_NODE, &vtysh_exit_bgpd_cmd);
1786 install_element (BGP_VPNV4_NODE, &vtysh_quit_bgpd_cmd);
1787 install_element (BGP_IPV4_NODE, &vtysh_exit_bgpd_cmd);
1788 install_element (BGP_IPV4_NODE, &vtysh_quit_bgpd_cmd);
1789 install_element (BGP_IPV4M_NODE, &vtysh_exit_bgpd_cmd);
1790 install_element (BGP_IPV4M_NODE, &vtysh_quit_bgpd_cmd);
1791 install_element (BGP_IPV6_NODE, &vtysh_exit_bgpd_cmd);
1792 install_element (BGP_IPV6_NODE, &vtysh_quit_bgpd_cmd);
1793 install_element (KEYCHAIN_NODE, &vtysh_exit_ripd_cmd);
1794 install_element (KEYCHAIN_NODE, &vtysh_quit_ripd_cmd);
1795 install_element (KEYCHAIN_KEY_NODE, &vtysh_exit_ripd_cmd);
1796 install_element (KEYCHAIN_KEY_NODE, &vtysh_quit_ripd_cmd);
1797 install_element (RMAP_NODE, &vtysh_exit_rmap_cmd);
1798 install_element (RMAP_NODE, &vtysh_quit_rmap_cmd);
1799
1800 /* "end" command. */
1801 install_element (CONFIG_NODE, &vtysh_end_all_cmd);
1802 install_element (ENABLE_NODE, &vtysh_end_all_cmd);
1803 install_element (RIP_NODE, &vtysh_end_all_cmd);
1804 install_element (RIPNG_NODE, &vtysh_end_all_cmd);
1805 install_element (OSPF_NODE, &vtysh_end_all_cmd);
1806 install_element (OSPF6_NODE, &vtysh_end_all_cmd);
1807 install_element (BGP_NODE, &vtysh_end_all_cmd);
1808 install_element (BGP_IPV4_NODE, &vtysh_end_all_cmd);
1809 install_element (BGP_IPV4M_NODE, &vtysh_end_all_cmd);
1810 install_element (BGP_VPNV4_NODE, &vtysh_end_all_cmd);
1811 install_element (BGP_IPV6_NODE, &vtysh_end_all_cmd);
1812 install_element (KEYCHAIN_NODE, &vtysh_end_all_cmd);
1813 install_element (KEYCHAIN_KEY_NODE, &vtysh_end_all_cmd);
1814 install_element (RMAP_NODE, &vtysh_end_all_cmd);
1815
paul338a9912003-03-01 15:44:10 +00001816 install_element (INTERFACE_NODE, &interface_desc_cmd);
paul464dc8d2003-03-28 02:25:45 +00001817 install_element (INTERFACE_NODE, &no_interface_desc_cmd);
paul718e3742002-12-13 20:15:29 +00001818 install_element (INTERFACE_NODE, &vtysh_end_all_cmd);
1819 install_element (INTERFACE_NODE, &vtysh_exit_interface_cmd);
1820 install_element (INTERFACE_NODE, &vtysh_quit_interface_cmd);
1821 install_element (CONFIG_NODE, &router_rip_cmd);
1822#ifdef HAVE_IPV6
1823 install_element (CONFIG_NODE, &router_ripng_cmd);
1824#endif
1825 install_element (CONFIG_NODE, &router_ospf_cmd);
1826#ifdef HAVE_IPV6
1827 install_element (CONFIG_NODE, &router_ospf6_cmd);
1828#endif
1829 install_element (CONFIG_NODE, &router_bgp_cmd);
1830 install_element (BGP_NODE, &address_family_vpnv4_cmd);
1831 install_element (BGP_NODE, &address_family_vpnv4_unicast_cmd);
1832 install_element (BGP_NODE, &address_family_ipv4_unicast_cmd);
1833 install_element (BGP_NODE, &address_family_ipv4_multicast_cmd);
1834#ifdef HAVE_IPV6
1835 install_element (BGP_NODE, &address_family_ipv6_cmd);
1836 install_element (BGP_NODE, &address_family_ipv6_unicast_cmd);
1837#endif
1838 install_element (BGP_VPNV4_NODE, &exit_address_family_cmd);
1839 install_element (BGP_IPV4_NODE, &exit_address_family_cmd);
1840 install_element (BGP_IPV4M_NODE, &exit_address_family_cmd);
1841 install_element (BGP_IPV6_NODE, &exit_address_family_cmd);
1842 install_element (CONFIG_NODE, &key_chain_cmd);
1843 install_element (CONFIG_NODE, &route_map_cmd);
1844 install_element (KEYCHAIN_NODE, &key_cmd);
1845 install_element (KEYCHAIN_NODE, &key_chain_cmd);
1846 install_element (KEYCHAIN_KEY_NODE, &key_chain_cmd);
1847 install_element (CONFIG_NODE, &vtysh_interface_cmd);
paul32d24632003-05-23 09:25:20 +00001848 install_element (CONFIG_NODE, &vtysh_no_interface_cmd);
paul718e3742002-12-13 20:15:29 +00001849 install_element (ENABLE_NODE, &vtysh_show_running_config_cmd);
1850 install_element (ENABLE_NODE, &vtysh_copy_runningconfig_startupconfig_cmd);
1851 install_element (ENABLE_NODE, &vtysh_write_file_cmd);
1852
1853 /* write terminal command */
1854 install_element (ENABLE_NODE, &vtysh_write_terminal_cmd);
1855 install_element (CONFIG_NODE, &vtysh_write_terminal_cmd);
1856 install_element (BGP_NODE, &vtysh_write_terminal_cmd);
1857 install_element (BGP_VPNV4_NODE, &vtysh_write_terminal_cmd);
1858 install_element (BGP_IPV4_NODE, &vtysh_write_terminal_cmd);
1859 install_element (BGP_IPV4M_NODE, &vtysh_write_terminal_cmd);
1860 install_element (BGP_IPV6_NODE, &vtysh_write_terminal_cmd);
1861 install_element (RIP_NODE, &vtysh_write_terminal_cmd);
1862 install_element (RIPNG_NODE, &vtysh_write_terminal_cmd);
1863 install_element (OSPF_NODE, &vtysh_write_terminal_cmd);
1864 install_element (OSPF6_NODE, &vtysh_write_terminal_cmd);
1865 install_element (INTERFACE_NODE, &vtysh_write_terminal_cmd);
1866 install_element (RMAP_NODE, &vtysh_write_terminal_cmd);
1867 install_element (KEYCHAIN_NODE, &vtysh_write_terminal_cmd);
1868 install_element (KEYCHAIN_KEY_NODE, &vtysh_write_terminal_cmd);
1869
1870 /* write memory command */
1871 install_element (ENABLE_NODE, &vtysh_write_memory_cmd);
1872 install_element (CONFIG_NODE, &vtysh_write_memory_cmd);
1873 install_element (BGP_NODE, &vtysh_write_memory_cmd);
1874 install_element (BGP_VPNV4_NODE, &vtysh_write_memory_cmd);
1875 install_element (BGP_IPV4_NODE, &vtysh_write_memory_cmd);
1876 install_element (BGP_IPV4M_NODE, &vtysh_write_memory_cmd);
1877 install_element (BGP_IPV6_NODE, &vtysh_write_memory_cmd);
1878 install_element (RIP_NODE, &vtysh_write_memory_cmd);
1879 install_element (RIPNG_NODE, &vtysh_write_memory_cmd);
1880 install_element (OSPF_NODE, &vtysh_write_memory_cmd);
1881 install_element (OSPF6_NODE, &vtysh_write_memory_cmd);
1882 install_element (INTERFACE_NODE, &vtysh_write_memory_cmd);
1883 install_element (RMAP_NODE, &vtysh_write_memory_cmd);
1884 install_element (KEYCHAIN_NODE, &vtysh_write_memory_cmd);
1885 install_element (KEYCHAIN_KEY_NODE, &vtysh_write_memory_cmd);
1886
1887 install_element (VIEW_NODE, &vtysh_ping_cmd);
1888 install_element (VIEW_NODE, &vtysh_traceroute_cmd);
1889 install_element (VIEW_NODE, &vtysh_telnet_cmd);
1890 install_element (VIEW_NODE, &vtysh_telnet_port_cmd);
paul5087df52003-01-25 06:56:09 +00001891 install_element (VIEW_NODE, &vtysh_ssh_cmd);
paul718e3742002-12-13 20:15:29 +00001892 install_element (ENABLE_NODE, &vtysh_ping_cmd);
1893 install_element (ENABLE_NODE, &vtysh_traceroute_cmd);
1894 install_element (ENABLE_NODE, &vtysh_telnet_cmd);
1895 install_element (ENABLE_NODE, &vtysh_telnet_port_cmd);
1896 install_element (ENABLE_NODE, &vtysh_start_shell_cmd);
1897 install_element (ENABLE_NODE, &vtysh_start_bash_cmd);
1898 install_element (ENABLE_NODE, &vtysh_start_zsh_cmd);
1899
paul718e3742002-12-13 20:15:29 +00001900 install_element (CONFIG_NODE, &vtysh_log_stdout_cmd);
1901 install_element (CONFIG_NODE, &no_vtysh_log_stdout_cmd);
1902 install_element (CONFIG_NODE, &vtysh_log_file_cmd);
1903 install_element (CONFIG_NODE, &no_vtysh_log_file_cmd);
1904 install_element (CONFIG_NODE, &vtysh_log_syslog_cmd);
1905 install_element (CONFIG_NODE, &no_vtysh_log_syslog_cmd);
1906 install_element (CONFIG_NODE, &vtysh_log_trap_cmd);
1907 install_element (CONFIG_NODE, &no_vtysh_log_trap_cmd);
1908 install_element (CONFIG_NODE, &vtysh_log_record_priority_cmd);
1909 install_element (CONFIG_NODE, &no_vtysh_log_record_priority_cmd);
paul4fc01e62002-12-13 20:49:00 +00001910 install_element (CONFIG_NODE, &vtysh_write_config_cmd);
1911 install_element (CONFIG_NODE, &no_vtysh_write_config_cmd);
paul718e3742002-12-13 20:15:29 +00001912}