evlauate api calls passed in as arguments to plcsh
[plcapi.git] / plcsh
1 #!/usr/bin/python
2 #
3 # Interactive shell for testing PLCAPI
4 #
5 # Mark Huang <mlhuang@cs.princeton.edu>
6 # Copyright (C) 2005 The Trustees of Princeton University
7 #
8 # $Id$
9 #
10
11 import os
12 import sys
13 from socket import gethostname
14 from optparse import OptionParser
15 from getpass import getpass
16 from traceback import print_exc
17
18 sys.path.append(os.path.dirname(os.path.realpath(sys.argv[0])))
19 from PLC.Shell import Shell
20
21 parser = OptionParser(add_help_option = False)
22 parser.add_option("-f", "--config", help = "PLC configuration file")
23 parser.add_option("-h", "--url", help = "API URL")
24 parser.add_option("-c", "--cacert", help = "API SSL certificate")
25 parser.add_option("-k", "--insecure", help = "Do not check SSL certificate")
26 parser.add_option("-m", "--method", help = "API authentication method")
27 parser.add_option("-s", "--session", help = "API session key")
28 parser.add_option("-u", "--user", help = "API user name")
29 parser.add_option("-p", "--password", help = "API password")
30 parser.add_option("-r", "--role", help = "API role")
31 parser.add_option("-x", "--xmlrpc", action = "store_true", default = False, help = "Use XML-RPC interface")
32 parser.add_option("--help", action = "help", help = "show this help message and exit")
33 (options, args) = parser.parse_args()
34
35 # If user is specified but password is not
36 if options.user is not None and options.password is None:
37     try:
38         options.password = getpass()
39     except (EOFError, KeyboardInterrupt):
40         print
41         sys.exit(0)
42
43 # Initialize a single global instance (scripts may re-initialize
44 # this instance and/or create additional instances).
45 try:
46     shell = Shell(globals = globals(),
47                   config = options.config,
48                   url = options.url, xmlrpc = options.xmlrpc, cacert = options.cacert,
49                   method = options.method, role = options.role,
50                   user = options.user, password = options.password,
51                   session = options.session)
52     # Register a few more globals for backward compatibility
53     auth = shell.auth
54     api = shell.api
55     config = shell.config
56 except Exception, err:
57     print "Error:", err
58     print
59     parser.print_help()
60     sys.exit(1)
61
62 # If called by a script or api call is passed
63 if len(sys.argv) > 1:
64     # Pop us off the argument stack
65     sys.argv.pop(0)
66     if os.path.exists(sys.argv[0]) and os.path.isfile(sys.argv[0]):
67         execfile(sys.argv[0])
68     else:
69         print eval(sys.argv[0])
70         sys.exit(0)
71
72 # Otherwise, run an interactive shell environment
73 else:
74     if shell.server is None:
75         print "PlanetLab Central Direct API Access"
76         prompt = ""
77     elif shell.auth['AuthMethod'] == "anonymous":
78         prompt = "[anonymous]"
79         print "Connected anonymously"
80     elif shell.auth['AuthMethod'] == "session":
81         # XXX No way to tell node and user sessions apart from the
82         # client point of view.
83         prompt = "[%s]" % gethostname()
84         print "%s connected using session authentication" % gethostname()
85     else:
86         prompt = "[%s]" % shell.auth['Username']
87         print "%s connected using %s authentication" % \
88               (shell.auth['Username'], shell.auth['AuthMethod'])
89
90     # Readline and tab completion support
91     import atexit
92     import readline
93     import rlcompleter
94
95     print 'Type "system.listMethods()" or "help(method)" for more information.'
96     # Load command history
97     history_path = os.path.join(os.environ["HOME"], ".plcapi_history")
98     try:
99         file(history_path, 'a').close()
100         readline.read_history_file(history_path)
101         atexit.register(readline.write_history_file, history_path)
102     except IOError:
103         pass
104
105     # Enable tab completion
106     readline.parse_and_bind("tab: complete")
107
108     try:
109         while True:
110             command = ""
111             while True:
112                 # Get line
113                 try:
114                     if command == "":
115                         sep = ">>> "
116                     else:
117                         sep = "... "
118                     line = raw_input(prompt + sep)
119                 # Ctrl-C
120                 except KeyboardInterrupt:
121                     command = ""
122                     print
123                     break
124
125                 # Build up multi-line command
126                 command += line
127
128                 # Blank line or first line does not end in :
129                 if line == "" or (command == line and line[-1] != ':'):
130                     break
131
132                 command += os.linesep
133
134             # Blank line
135             if command == "":
136                 continue
137             # Quit
138             elif command in ["q", "quit", "exit"]:
139                 break
140
141             try:
142                 try:
143                     # Try evaluating as an expression and printing the result
144                     result = eval(command)
145                     if result is not None:
146                         print result
147                 except SyntaxError:
148                     # Fall back to executing as a statement
149                     exec command
150             except Exception, err:
151                 print_exc()
152
153     except EOFError:
154         print
155         pass