Vim: Error format matches everything

2.2k views Asked by At

I am trying to match the following error with efm:

AssertionError: 1 == 2
    at /home/strager/projects/blah/blah.js:13:37

The error message can be anything (i.e. it doesn't always match the AssertionError: .* or .*Error: .* formats). The general format is:

errormessage
    at filename:line:column

My problem is that the error message matches any line; I need to restrict the error message to one line only, and only match if it's followed by a matching "at" line.

I have tried the following efm:

set efm=%Z\ \ \ \ at\ %f:%l:%c,%E%m
" %Z    at %f:%l:%c,%E%m

This almost works, but it matches status lines (e.g. non-errors before and after the error) in addition to the errors. How can I force %E%m ... %Z to be only two lines total (one for the error message, and one for the at line)? I have access to the standard UNIX tools for makeprg if needed.

4

There are 4 answers

4
Andrew On

Do you really want to spend your time learning an obscure pattern language that has no application anywhere else? Unless someone is paying you to write vim compiler plug-ins, I wouldn't (and I love vim!). Since you're willing to open the toolbox, just write a wrapper around your checker and spit out an easy-to-parse format. Eg:

#!/usr/bin/perl
use warnings;
use strict;

open my $fh, '-|', 'compiler', @_ or die $!;

my $last_line = <$fh> // exit;
while (defined(my $line = <$fh>)) {
    my($file, $l, $c) = $line =~ /^    at (.+?):(\d+):(\d+)$/;
    print "$file:$l:$c: $last_line" if defined($file);
    $last_line = $line;
}
0
Starfish On

Does this work?

set efm=%Z\ \ \ \ at\ %f:%l:%c,%E%m,%-G%.%#

The %-G%.%# tells vim to ignore entire lines that do not match the other patterns.

1
Junru Zhu On

Maybe

set efm=%E%>%m,%Z\ \ \ \ at\ %f:%l:%c

Check this

:help efm%>
1
too much php On

What about ...

set efm=%E%m,%Z\ \ \ \ at\ %f:%l:%c