pmacs3/mode/awk.py

160 lines
6.6 KiB
Python
Raw Normal View History

2008-10-04 12:09:15 -04:00
import commands, os
2008-10-06 12:32:18 -04:00
from subprocess import Popen, PIPE, STDOUT
2008-10-04 12:09:15 -04:00
import color, default, mode, tab
from lex import Grammar, PatternRule, RegionRule
from mode.python import StringGrammar2
2008-10-02 12:49:22 -04:00
from tab import StackTabber2
2008-10-04 12:09:15 -04:00
from method import Method, Argument, arg
from method.shell import Exec, Pipe
2008-10-01 18:10:11 -04:00
class RegexGrammar(Grammar):
rules = [
PatternRule(r'escaped', r'\\.'),
PatternRule(r'data', r'[^\\/]+'),
]
class AwkGrammar(Grammar):
rules = [
2008-10-01 18:10:11 -04:00
PatternRule(r'comment', r'#.*\n$'),
PatternRule(r'spaces', r' +'),
2008-10-01 23:59:30 -04:00
RegionRule(r'awk_regex', r'/(?! )', RegexGrammar, r'/'),
2008-10-01 18:10:11 -04:00
PatternRule(r'awk_global', r'(?:TEXTDOMAIN|SUBSEP|RLENGTH|RSTART|RT|RS|PROCINFO|ORS|OFS|OFMT|NR|NF|LINT|IGNORECASE|FS|FNR|FILENAME|FIELDWIDTHS|ERRNO|ENVIRON|CONVFMT|BINMODE|ARGV|ARGIND|ARGC)(?![a-zA-Z0-9_])'),
2008-10-01 23:59:30 -04:00
PatternRule(r'delimiter', r'(?:[\{\}()\[\]?:;,]|=(?!=)|\+=|-=|\*=|/=|\%=|\^=)'),
2008-10-02 09:35:28 -04:00
PatternRule(r'keyword', r'(?:BEGIN|END|if|else|while|do|for|break|continue|delete|exit)(?![a-zA-Z0-9_])'),
PatternRule(r'builtin', r'(?:close|getline|nextfile|next|printf|print|system|fflush|atan2|cos|exp|int|log|rand|sin|sqrt|srand|asorti|asort|gensub|gsub|index|length|match|split|sprintf|strtonum|substr|sub|tolower|toupper|mktime|strftime|systime|and|compl|lshift|or|xor|rshift|bindtextdomain|dcgettext|dcngettext|function|extension)(?![a-zA-Z0-9_])'),
2008-10-01 23:59:30 -04:00
PatternRule(r'awk_field', r'\$\d*'),
2008-10-01 18:15:52 -04:00
PatternRule(r'number', r'-?0x[0-9A-Fa-f]+'),
PatternRule(r'number', r'-?0[0-7]*'),
PatternRule(r'number', r'-?[0-9]+\.?[0-9]*'),
PatternRule(r'number', r'-?\.[0-9]+'),
PatternRule(r'unop', r'!(?![=~])|--|\+\+'),
PatternRule(r'binop', r'(?:&&|\|\||<=|>=|!=|!~|==|\^|%|[-~/+*<>])'),
RegionRule(r'string', r'"', StringGrammar2, r'"'),
2008-10-01 23:59:30 -04:00
PatternRule(r'awk_function', r'[a-zA-Z_][a-zA-Z0-9_]*(?=\()'),
PatternRule(r'awk_identifier', r'[a-zA-Z_][a-zA-Z0-9_]*'),
2008-10-01 18:15:52 -04:00
2008-10-01 23:59:30 -04:00
PatternRule(r'continuation', r'\\\n$'),
2008-10-01 18:10:11 -04:00
PatternRule(r'eol', r'\n'),
]
2008-10-02 12:49:22 -04:00
class AwkTabber(StackTabber2):
open_tokens = {'delimiter': {'{': '}', '(': ')', '[': ']'}}
close_tokens = {'delimiter': {'}': '{', ')': '(', ']': '['}}
control_tokens = {
'keyword': {'if': 1, 'else': 1, 'while': 1, 'do': 1, 'for': 1},
}
end_at_eof = True
end_at_tokens = {}
2008-10-01 23:59:30 -04:00
def _is_base(self, y):
2008-10-02 09:35:28 -04:00
if y == 0:
return True
t = self._get_tokens(y)[0]
if t.fqname() == 'awk_regex.start':
return True
elif t.name in ('awk_field', 'awk_global'):
return True
elif t.name == 'keyword' and t.string in ('BEGIN', 'END'):
return True
else:
return False
2008-10-02 12:49:22 -04:00
def _is_indent(self, t):
return t.name == 'spaces'
def _is_ignored(self, t):
return t.name in ('spaces', 'eol', 'comment')
2008-10-02 09:35:28 -04:00
2008-10-04 12:09:15 -04:00
class AwkFilterFile(Exec):
show_success = True
2008-10-06 12:32:18 -04:00
args = [arg('path', dt="path", p="Filter File: ", dv=default.path_dirname, ld=True, h="file to open")]
2008-10-04 12:09:15 -04:00
def _execute(self, w, **vargs):
if not hasattr(w.buffer, 'path'):
2008-10-06 12:32:18 -04:00
w.set_error("Buffer %r has no program" % w.buffer.name())
2008-10-04 12:09:15 -04:00
return
elif w.buffer.changed():
2008-10-06 12:32:18 -04:00
w.set_error("Buffer %r is not saved" % w.buffer.name())
2008-10-04 12:09:15 -04:00
return
2008-10-06 12:32:18 -04:00
elif not os.path.exists(vargs['path']):
w.set_error("File %r not found" % vargs['path'])
return
cmd = 'awk -f %r %r' % (w.buffer.path, vargs['path'])
pipe = Popen(cmd, shell=True, stdout=PIPE, stderr=STDOUT)
outdata = pipe.stdout.read()
status = pipe.wait() >> 8
w.application.data_buffer('*Awk-Output*', outdata, switch_to=True)
w.set_error("awk exited with status %d" % status)
2008-10-04 12:09:15 -04:00
2008-10-06 00:55:38 -04:00
class AwkFilterBuffer(Pipe):
2008-10-06 12:32:18 -04:00
args = [arg('name', dt="buffer", p="Filter Buffer: ", h="name of the buffer to switch to")]
def _execute(self, w, **vargs):
if not hasattr(w.buffer, 'path'):
w.set_error("Buffer %r has no program" % w.buffer.name())
return
elif w.buffer.changed():
w.set_error("Buffer %r is not saved" % w.buffer.name())
return
elif not w.application.has_buffer_name(vargs['name']):
w.set_error("Buffer %r not found" % vargs['name'])
return
b = w.application.bufferlist.get_buffer_by_name(vargs['name'])
opts = w.application.config['awk.cmd-opts']
cmd = 'awk %s -f %r' % (opts, w.buffer.path)
pipe = Popen(cmd, shell=True, stdin=PIPE, stdout=PIPE, stderr=STDOUT)
pipe.stdin.write(b.make_string())
pipe.stdin.close()
output = pipe.stdout.read()
status = pipe.wait() >> 8
w.application.data_buffer('*Awk-Output*', output, switch_to=True)
w.set_error("awk exited with status %d" % status)
2008-10-04 12:09:15 -04:00
class AwkFilterInput(Method):
2008-10-06 12:32:18 -04:00
args = [arg('input', p="Data To Filter: ", h="data to filter")]
def _execute(self, w, **vargs):
if not hasattr(w.buffer, 'path'):
w.set_error("Buffer %r has no program" % w.buffer.name())
return
elif w.buffer.changed():
w.set_error("Buffer %r is not saved" % w.buffer.name())
return
opts = w.application.config['awk.cmd-opts']
cmd = 'awk %s -f %r' % (opts, w.buffer.path)
pipe = Popen(cmd, shell=True, stdin=PIPE, stdout=PIPE, stderr=STDOUT)
pipe.stdin.write(vargs['input'])
pipe.stdin.close()
output = pipe.stdout.read()
status = pipe.wait() >> 8
w.application.data_buffer('*Awk-Output*', output, switch_to=True)
w.set_error("awk exited with status %d" % status)
2008-10-01 23:59:30 -04:00
class Awk(mode.Fundamental):
2008-10-02 09:35:28 -04:00
tabbercls = AwkTabber
2009-02-15 12:06:35 -05:00
modename = 'awk'
extensions = ['.awk']
grammar = AwkGrammar
opentokens = ('delimiter',)
opentags = {'(': ')', '[': ']', '{': '}'}
closetokens = ('delimiter',)
closetags = {')': '(', ']': '[', '}': '{'}
config = {'awk.cmd-opts': ""}
actions = [AwkFilterFile, AwkFilterBuffer, AwkFilterInput]
colors = {
2008-10-01 23:59:30 -04:00
'awk_global': ('yellow', 'default', 'bold'),
'awk_function': ('magenta', 'default', 'bold'),
'awk_field': ('yellow', 'default', 'bold'),
'awk_identifier': ('yellow', 'default', 'bold'),
'awk_regex.start': ('cyan', 'default', 'bold'),
'awk_regex.null': ('cyan', 'default', 'bold'),
'awk_regex.data': ('cyan', 'default', 'bold'),
'awk_regex.end': ('cyan', 'default', 'bold'),
}
2009-02-15 12:06:35 -05:00
_bindings = {
'close-paren': (')',),
'close-brace': ('}',),
'close-bracket': (']',),
}
install = Awk.install