import os from subprocess import Popen, PIPE, STDOUT import dirutil import regex from buffer.emul import XTermBuffer from method import Method, arg from term import XTerm from window import Window class Exec(Method): '''Execute a command in a shell and put the output in a new buffer''' show_success = True args = [arg('cmd', dt='shell', p="Exec: ")] def _doit(self, w, path, cmd, cmdname=None, bufname=None, cmddir=None, opts={}): if cmddir: cmd = "cd %r && %s" % (cmddir, cmd) d = dict(opts) if path: d['path'] = path try: cmd = cmd % d except: pass if bufname is None: bufname = '*%s*' % self.name.title() if cmdname is None: cmdname = cmd.split(None, 1)[0] p = Popen(cmd, shell=True, stdout=PIPE, stderr=STDOUT) output = p.stdout.read() result = p.wait() status = os.WEXITSTATUS(result) if not os.WIFEXITED(result): err = True msg = "%s: killed by signal %r" % (cmdname, os.WTERMSIG(result)) elif status != 0: err = True msg = "%s: failed with status %r" % (cmdname, status) else: err = False msg = "%s: ok" % (cmdname,) if output: switch_to = err or self.show_success w.application.data_buffer(bufname, output, switch_to=switch_to) w.set_error(msg) def _execute(self, w, **vargs): if w.buffer.btype == 'dir': name, path = dirutil.resolve_name_path(w) self._doit(w, path, vargs['cmd']) dirutil.find_name(w, name) elif hasattr(w.buffer, 'path'): path = w.buffer.path self._doit(w, path, vargs['cmd']) else: self._doit(w, None, vargs['cmd']) class Man(Exec): '''Execute a command in a shell and put the output in a new buffer''' args = [arg('name', p="Program: ")] def _execute(self, w, **vargs): name = vargs['name'] cmd = 'man %r' % name p = Popen(cmd, shell=True, stdout=PIPE, stderr=PIPE) output = p.stdout.read() result = p.wait() status = os.WEXITSTATUS(result) if not os.WIFEXITED(result): err = True errmsg = "man: killed by signal %r" % os.WTERMSIG(result) elif status != 0: err = True errmsg = "man: failed with status %r" % status else: err = False errmsg = "man: ok" if output: xterm = XTerm(cbuf=True) s = xterm.term_filter(output) switch = err or self.show_success w.application.color_data_buffer('*Manpage*', s, switch_to=switch) w.set_error(errmsg) class Pipe(Method): '''Pipe the buffer's contents through the command, and display the output in a new buffer''' args = [arg('cmd', dt="shell", p="Pipe: ")] def _parse(self, w, **vargs): # return 3 things: prog name, cmd, and whether to use the shell m = regex.shell_command.match(vargs['cmd']) if m: prog = m.group(0) return (prog, vargs['cmd'], True) else: return (None, None, False) def _display(self, w, data, status, prog, cmd, shell): lines = data.split('\n') if lines and lines[-1] == '': lines = lines[:-1] bufname = '*%s*' % self.name.title() b = w.application.data_buffer(bufname, data, switch_to=True, modename='error') b.orig_path = w.buffer.path w.set_error("%s exited with status %d" % (prog, status)) def _execute(self, w, **vargs): (prog, cmd, shell) = self._parse(w, **vargs) if prog is None or not cmd: return self._dopipe(w, prog, cmd, shell) def _dopipe(self, w, prog, cmd, shell): pipe = Popen(cmd, shell=shell, stdin=PIPE, stdout=PIPE, stderr=STDOUT) pid = pipe.pid indata = w.buffer.make_string() pipe.stdin.write(indata) pipe.stdin.close() outdata = pipe.stdout.read() status = pipe.wait() >> 8 self._display(w, outdata, status, prog, cmd, shell) class Grep(Pipe): '''Grep the buffer's contents for instances of a pattern, and display them in a new buffer''' args = [arg('pattern', dt="str", p="Pattern: ")] def _parse(self, w, **vargs): return ('grep', ('grep', '-E', '-n', vargs['pattern']), False) class Sed(Pipe): '''Push the buffer's contents through a sed expression''' args = [arg('expression', dt="str", p="Expression: ")] def _parse(self, w, **vargs): return ('sed', ('sed', '-r', '-e', vargs['expression']), False) class Interact(Method): '''Interact with a program via a PTY''' args = [arg('bname', dt="str", p="Buffer Name: ", dv=lambda w: '*Interact*'), arg('cmd', dt="shell", p="Command: ", dv=lambda w: 'bash')] modename = None reuse = False def _execute(self, w, **vargs): bname = vargs['bname'] cmd = vargs['cmd'] a = w.application if self.reuse and a.has_buffer_name(bname): a.switch_buffer(a.get_buffer_by_name(bname)) return a.close_buffer_by_name(bname) b = XTermBuffer(a, 'bash', ['-c', cmd], name=bname, modename=self.modename) a.add_buffer(b) Window(b, a) if a.window().buffer is not b: a.switch_buffer(b)