The hydra grows yet another head...
[asterisk/asterisk.git] / contrib / scripts / astcli
1 #!/usr/bin/perl -w
2
3 use strict;
4 use IO::Socket;
5 use Getopt::Long;
6
7 # Created by: David Van Ginneken
8 # Bird's the Word Technologies
9 # davevg@btwtech.com
10 #
11 # And distributed under the terms of the GPL
12 #
13 my ($user, $pw, $host, $port, $interactive, $save) = (undef, undef, 'localhost', 5038, 0, 0);
14 my $EOL = "\r\n"; # Standard End of Line
15 process_credentials('/etc/astcli.conf');
16 process_credentials("$ENV{HOME}/.astcli") if defined $ENV{HOME};
17 GetOptions("username=s" => \$user, "secret=s" => \$pw, "host=s" => \$host, "port=s" => \$port, "readline" => \$interactive, "write" => \$save);
18
19 $|++; # Auto Flush Output
20 my $action = join(" ", @ARGV);
21
22 &usage if (!defined $user || !defined $pw);
23 my $tc = new IO::Socket::INET(
24                 PeerAddr => $host,
25                 PeerPort => $port,
26                 Timeout => 30,
27                 Proto => 'tcp'
28 ) or die "Could not connect to Host: $host on port $port\n";
29 if (my $error = login()) {
30         print STDERR $error;
31         exit 1;
32 };
33
34 if ($save) {
35         if (-d $ENV{HOME}) {
36                 open DEFAULT, ">$ENV{HOME}/.astcli";
37                 print DEFAULT "username=$user\n" if $user;
38                 print DEFAULT "password=$pw\n" if $pw;
39                 print DEFAULT "hostname=$host\n" if $host;
40                 print DEFAULT "portno=$port\n" if $port;
41                 close DEFAULT;
42         }
43 }
44
45 # Send a single command to the manager connection handle (global $tc).
46 # Assumes things always work well :-)
47 sub send_command($) {
48         my $command = shift;
49         $tc->send('Action: Command' . $EOL);
50         $tc->send("Command: $command" . $EOL);
51         $tc->send($EOL);
52         my $response = '';
53         while (<$tc>) {
54                 last if $_ =~ /--END COMMAND--/;
55                 $response .= $_;
56         }
57         $response =~ s/Privilege: Command$EOL//;
58         $response =~ s/Response: Follows$EOL//;
59         print $response;
60 }
61
62 sub login {
63         my ($response, $message);
64         $tc->send("Action: Login" . $EOL);
65         $tc->send("Username: $user" . $EOL);
66         $tc->send("Secret: $pw" . $EOL);
67         $tc->send("Events: off" . $EOL);
68         $tc->send($EOL);
69         while (<$tc>) {
70                 last if $_ eq $EOL;
71                 $_ =~ s/$EOL//g;
72                 ($response) = $_ =~ /^Response: (.*?)$/ if $_ =~ /^Response:/;
73                 ($message) = $_ =~ /^Message: (.*?)$/ if $_ =~ /^Message:/;
74         }
75         return 0 if $response eq 'Success';
76         return $message;
77 }
78
79 sub logoff {
80         my ($response, $message);
81         $tc->send("Action: Logoff" . $EOL . $EOL);
82         return 1;
83 }
84
85 # If the user asked to send commands from standard input:
86 if ($action eq '-' || !defined $action || $action eq '') {
87         if ($interactive) {
88                 eval { require Term::ReadLine;};
89                 $interactive = scalar($@) ? 0 : 1;
90                 print STDERR "Falling back to standard mode, Unable to load Term::Readline for readline mode\n" unless $interactive;
91         }
92         if ($interactive) {
93                 my $term = new Term::ReadLine 'Command Line Interface';
94                 my $prompt = "$host*CLI> ";
95                 my $attribs = $term->Attribs;
96                 $attribs->{completion_function} = sub {
97                         my ($text, $line, $start) = @_;
98                         # Stub function for tab auto completion for those feeling adventurous
99                         return;
100                 };
101                 while (defined($_ = $term->readline($prompt))) {
102                         (logoff() and exit) if $_ =~ /exit|quit/; # Give them a way to exit the "terminal"
103                         send_command($_);
104                 }       
105         } else {
106                 while (<>) {
107                         chomp;
108                         (logoff() and exit) if $_ =~ /exit|quit/; # If someone accidentally ends up here, let them exit
109                         send_command($_);
110                 }
111         }
112         exit 0;
113 }
114
115 # Otherwise just send the command:
116 send_command($action);
117
118 # parses a configuration file into the global $user and $pw.
119 sub process_credentials {
120         # Process the credentials found..
121         my $file = shift;
122         # silently fail if we can't read the file:
123         return unless (-r $file);
124         open (my $fh, "<$file") or return;
125         while (<$fh>) {
126                 chomp;
127                 (undef,$user) = split(/[,=]/, $_) if $_ =~ /user(name)?[,=]/i;
128                 (undef,$pw) = split(/[,=]/, $_) if $_ =~ /(secret|passw(or)?d|pwd?)[,=]/i;
129                 (undef,$host) = split(/[,=]/, $_) if $_ =~ /host(name)?[,=]/i;
130                 (undef,$port) = split(/[,=]/, $_) if $_ =~ /port(num|no)?[,=]/i;
131         }
132         close ($fh);
133 }
134
135 sub usage {
136         print STDERR "astcli [<options>] [<cli-command>|-]\n";
137         print STDERR "       -u <name> - Connect as username <name>\n";
138         print STDERR "       -s <pw>   - Connect with secret <pw>\n";
139         print STDERR "       -h <host> - Connect to host <host> [localhost]\n";
140         print STDERR "       -p <port> - Connect on TCP port <port> [5038]\n";
141         print STDERR "       -r        - Start a readline session for interactivity\n";
142         print STDERR "       -w        - Save connection options in a configuration file\n";
143         print STDERR "   You may specify the command as '-' to take commands from stdin.\n";
144         exit;
145 }
146