mirror of
https://github.com/GothenburgBitFactory/taskwarrior.git
synced 2025-07-07 20:06:36 +02:00
Filter Bugs
- A substitution arg is categorized as "subst", but was tested for "substitution". - The Arguments::is_multipart method was triggering on '/', which is wrong, but I can't remember why I put it in there. Hmmm. - When extracting a write-filter, substitutions after the command were generating an error, whereas they should be ignored. - Task::substitute was modifying data even when the patterns were not matching. Yikes. - Added unit tests for en-passant modification of all types during done, delete, start, stop.
This commit is contained in:
parent
69e70889be
commit
f0d2af2651
7 changed files with 369 additions and 22 deletions
|
@ -154,7 +154,7 @@ void Arguments::capture (int argc, const char** argv)
|
|||
// be an absolute path, and Expression::expand_tokens would make a dog's
|
||||
// dinner out of it.
|
||||
std::vector <std::string> parts;
|
||||
if (is_multipart (argv[i], parts) && i != 0)
|
||||
if (i && is_multipart (argv[i], parts))
|
||||
{
|
||||
std::vector <std::string>::iterator part;
|
||||
for (part = parts.begin (); part != parts.end (); ++part)
|
||||
|
@ -731,7 +731,7 @@ bool Arguments::is_multipart (
|
|||
std::string part;
|
||||
while (n.getQuoted ('"', part) ||
|
||||
n.getQuoted ('\'', part) ||
|
||||
n.getQuoted ('/', part) ||
|
||||
// n.getQuoted ('/', part) || <--- this line breaks subst.
|
||||
n.getUntilWS (part))
|
||||
{
|
||||
n.skipWS ();
|
||||
|
@ -864,14 +864,14 @@ bool Arguments::is_subst (const std::string& input)
|
|||
Nibbler n (input);
|
||||
if (n.skip ('/') &&
|
||||
n.getUntil ('/', from) &&
|
||||
from.length () &&
|
||||
n.skip ('/') &&
|
||||
n.getUntil ('/', to) &&
|
||||
n.skip ('/'))
|
||||
{
|
||||
n.skip ('g');
|
||||
if (n.depleted () &&
|
||||
! Directory (input).exists () && // Ouch - expensive call.
|
||||
from.length ())
|
||||
if (n.depleted () &&
|
||||
! Directory (input).exists ()) // Ouch - expensive call.
|
||||
return true;
|
||||
}
|
||||
|
||||
|
@ -1376,6 +1376,14 @@ bool Arguments::extract_operator (
|
|||
}
|
||||
|
||||
////////////////////////////////////////////////////////////////////////////////
|
||||
// Almost all arguments are filters, except:
|
||||
// subst
|
||||
// program
|
||||
// command
|
||||
// rc
|
||||
// override
|
||||
//
|
||||
// Special case: attr "limit" is ignored.
|
||||
Arguments Arguments::extract_read_only_filter ()
|
||||
{
|
||||
Arguments filter;
|
||||
|
@ -1389,7 +1397,6 @@ Arguments Arguments::extract_read_only_filter ()
|
|||
arg->_third == "rc" ||
|
||||
arg->_third == "override")
|
||||
{
|
||||
;
|
||||
}
|
||||
|
||||
// Included.
|
||||
|
@ -1424,6 +1431,10 @@ Arguments Arguments::extract_read_only_filter ()
|
|||
}
|
||||
|
||||
////////////////////////////////////////////////////////////////////////////////
|
||||
// A write filter includes id/uuid anywhere on the command line, but any other
|
||||
// filter elements must occur before the command.
|
||||
//
|
||||
// Special case: attr "limit" is ignored.
|
||||
Arguments Arguments::extract_write_filter ()
|
||||
{
|
||||
Arguments filter;
|
||||
|
@ -1441,7 +1452,6 @@ Arguments Arguments::extract_write_filter ()
|
|||
arg->_third == "rc" ||
|
||||
arg->_third == "override")
|
||||
{
|
||||
;
|
||||
}
|
||||
|
||||
// Included regardless of position.
|
||||
|
@ -1471,10 +1481,13 @@ Arguments Arguments::extract_write_filter ()
|
|||
// Error.
|
||||
else
|
||||
{
|
||||
// substitution
|
||||
throw std::string ("A substitution '")
|
||||
+ arg->_first
|
||||
+ "' is not allowed in a read-only command filter.";
|
||||
if (before_command)
|
||||
{
|
||||
// substitution
|
||||
throw std::string ("A substitution '")
|
||||
+ arg->_first
|
||||
+ "' is not allowed in a write command filter.";
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -1574,13 +1587,13 @@ Arguments Arguments::extract_simple_words ()
|
|||
}
|
||||
|
||||
// Included.
|
||||
else if (arg->_third == "tag" ||
|
||||
arg->_third == "pattern" ||
|
||||
arg->_third == "substitution" ||
|
||||
arg->_third == "id" ||
|
||||
arg->_third == "uuid" ||
|
||||
arg->_third == "op" ||
|
||||
arg->_third == "exp" ||
|
||||
else if (arg->_third == "tag" ||
|
||||
arg->_third == "pattern" ||
|
||||
arg->_third == "subst" ||
|
||||
arg->_third == "id" ||
|
||||
arg->_third == "uuid" ||
|
||||
arg->_third == "op" ||
|
||||
arg->_third == "exp" ||
|
||||
arg->_third == "word")
|
||||
{
|
||||
// "limit" is special - it is recognized but not included in filters.
|
||||
|
|
|
@ -119,8 +119,7 @@ int Context::initialize (int argc, const char** argv)
|
|||
// Instantiate built-in column objects.
|
||||
Column::factory (columns);
|
||||
|
||||
// Categorize all arguments one more time.
|
||||
// TODO This may not be necessary.
|
||||
// Categorize all arguments one more time. THIS IS NECESSARY.
|
||||
args.categorize ();
|
||||
|
||||
// Handle default command and assumed 'info' command.
|
||||
|
|
|
@ -861,8 +861,11 @@ void Task::substitute (
|
|||
}
|
||||
}
|
||||
|
||||
set ("description", description);
|
||||
setAnnotations (annotations);
|
||||
if (changes)
|
||||
{
|
||||
set ("description", description);
|
||||
setAnnotations (annotations);
|
||||
}
|
||||
}
|
||||
|
||||
////////////////////////////////////////////////////////////////////////////////
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue