Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
T
topydo
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
Kirill Smelkov
topydo
Commits
5bcf70e6
Commit
5bcf70e6
authored
May 31, 2015
by
Bram Schoenmakers
Browse files
Options
Browse Files
Download
Plain Diff
Merge pull request #30 from mruwek/add-todos-from-file
Add todos from file with '-f' flag.
parents
f31b49f1
4380d872
Changes
3
Hide whitespace changes
Inline
Side-by-side
Showing
3 changed files
with
113 additions
and
46 deletions
+113
-46
test/AddCommandTest.py
test/AddCommandTest.py
+24
-0
test/data/AddCommandTest-from_file.txt
test/data/AddCommandTest-from_file.txt
+2
-0
topydo/commands/AddCommand.py
topydo/commands/AddCommand.py
+87
-46
No files found.
test/AddCommandTest.py
View file @
5bcf70e6
...
...
@@ -16,7 +16,16 @@
from
datetime
import
date
import
unittest
# We're searching for 'mock'
# pylint: disable=no-name-in-module
try
:
from
unittest
import
mock
except
ImportError
:
import
mock
from
six
import
u
from
io
import
StringIO
from
topydo.commands
import
AddCommand
from
topydo.commands
import
ListCommand
...
...
@@ -242,6 +251,21 @@ class AddCommandTest(CommandTest):
self
.
assertEqual
(
self
.
output
,
utf8
(
u
(
"| 1| {} Special
\
u25c4
\
n
"
).
format
(
self
.
today
)))
self
.
assertEqual
(
self
.
errors
,
""
)
@
mock
.
patch
(
"topydo.commands.AddCommand.stdin"
,
StringIO
(
u
(
"Fo
\
u00f3
due:tod id:1
\
n
B
\
u0105
r before:1"
)))
def
test_add_from_stdin
(
self
):
command
=
AddCommand
.
AddCommand
([
"-f"
,
"-"
],
self
.
todolist
,
self
.
out
,
self
.
error
)
command
.
execute
()
self
.
assertEqual
(
self
.
output
,
utf8
(
u
(
"| 1| {tod} Fo
\
u00f3
due:{tod} id:1
\
n
| 2| {tod} B
\
u0105
r p:1
\
n
"
.
format
(
tod
=
self
.
today
))))
self
.
assertEqual
(
self
.
errors
,
""
)
def
test_add_from_file
(
self
):
command
=
AddCommand
.
AddCommand
([
"-f"
,
"test/data/AddCommandTest-from_file.txt"
],
self
.
todolist
,
self
.
out
,
self
.
error
)
command
.
execute
()
self
.
assertEqual
(
self
.
output
,
utf8
(
u
(
"| 1| {tod} Foo @fo
\
u00f3
b
\
u0105
r due:{tod} id:1
\
n
| 2| {tod} Bar +baz t:{tod} p:1
\
n
"
.
format
(
tod
=
self
.
today
))))
self
.
assertEqual
(
self
.
errors
,
""
)
def
test_help
(
self
):
command
=
AddCommand
.
AddCommand
([
"help"
],
self
.
todolist
,
self
.
out
,
self
.
error
)
command
.
execute
()
...
...
test/data/AddCommandTest-from_file.txt
0 → 100644
View file @
5bcf70e6
Foo @foóbąr due:tod id:1
Bar +baz t:tod before:1
topydo/commands/AddCommand.py
View file @
5bcf70e6
...
...
@@ -18,6 +18,9 @@
from
datetime
import
date
import
re
from
sys
import
stdin
import
codecs
from
os.path
import
expanduser
from
topydo.lib.Config
import
config
from
topydo.lib.Command
import
Command
...
...
@@ -33,73 +36,109 @@ class AddCommand(Command):
super
(
AddCommand
,
self
).
__init__
(
p_args
,
p_todolist
,
p_out
,
p_err
,
p_prompt
)
self
.
text
=
' '
.
join
(
p_args
)
self
.
todo
=
None
self
.
from_file
=
None
def
_preprocess_input_todo
(
self
):
"""
Preprocesses user input when adding a task.
def
_process_flags
(
self
):
opts
,
args
=
self
.
getopt
(
'f:'
)
It detects a priority mid-sentence and puts it at the start.
"""
self
.
text
=
re
.
sub
(
r'^(.+) (\
([A-Z]
\))(.*)$'
,
r'\2 \1\3'
,
self
.
text
)
for
opt
,
value
in
opts
:
if
opt
==
'-f'
:
self
.
from_file
=
expanduser
(
value
)
def
_postprocess_input_todo
(
self
):
"""
Post-processes a parsed todo when adding it to the list.
self
.
args
=
args
* It converts relative dates to absolute ones.
* Automatically inserts a creation date if not present.
* Handles more user-friendly dependencies with before:, partof: and
after: tags
"""
def
convert_date
(
p_tag
):
value
=
self
.
todo
.
tag_value
(
p_tag
)
if
value
:
dateobj
=
relative_date_to_date
(
value
)
if
dateobj
:
self
.
todo
.
set_tag
(
p_tag
,
dateobj
.
isoformat
())
def
get_todos_from_file
(
self
):
if
self
.
from_file
==
'-'
:
f
=
stdin
else
:
f
=
codecs
.
open
(
self
.
from_file
,
'r'
,
encoding
=
'utf-8'
)
todos
=
f
.
read
().
splitlines
()
return
todos
def
_add_todo
(
self
,
p_todo_text
):
def
_preprocess_input_todo
(
p_todo_text
):
"""
Preprocesses user input when adding a task.
It detects a priority mid-sentence and puts it at the start.
"""
todo_text
=
re
.
sub
(
r'^(.+) (\
([A-Z]
\))(.*)$'
,
r'\2 \1\3'
,
p_todo_text
)
return
todo_text
def
_postprocess_input_todo
(
p_todo
):
"""
Post-processes a parsed todo when adding it to the list.
def
add_dependencies
(
p_tag
):
for
value
in
self
.
todo
.
tag_values
(
p_tag
):
try
:
dep
=
self
.
todolist
.
todo
(
value
)
* It converts relative dates to absolute ones.
* Automatically inserts a creation date if not present.
* Handles more user-friendly dependencies with before:, partof: and
after: tags
"""
def
convert_date
(
p_tag
):
value
=
p_todo
.
tag_value
(
p_tag
)
if
p_tag
==
'after'
:
self
.
todolist
.
add_dependency
(
self
.
todo
,
dep
)
elif
p_tag
==
'before'
or
p_tag
==
'partof'
:
self
.
todolist
.
add_dependency
(
dep
,
self
.
todo
)
except
InvalidTodoException
:
pass
if
value
:
dateobj
=
relative_date_to_date
(
value
)
if
dateobj
:
p_todo
.
set_tag
(
p_tag
,
dateobj
.
isoformat
())
self
.
todo
.
remove_tag
(
p_tag
,
value
)
def
add_dependencies
(
p_tag
):
for
value
in
p_todo
.
tag_values
(
p_tag
):
try
:
dep
=
self
.
todolist
.
todo
(
value
)
convert_date
(
config
().
tag_start
())
convert_date
(
config
().
tag_due
())
if
p_tag
==
'after'
:
self
.
todolist
.
add_dependency
(
p_todo
,
dep
)
elif
p_tag
==
'before'
or
p_tag
==
'partof'
:
self
.
todolist
.
add_dependency
(
dep
,
p_todo
)
except
InvalidTodoException
:
pass
add_dependencies
(
'partof'
)
add_dependencies
(
'before'
)
add_dependencies
(
'after'
)
p_todo
.
remove_tag
(
p_tag
,
value
)
self
.
todo
.
set_creation_date
(
date
.
today
())
convert_date
(
config
().
tag_start
())
convert_date
(
config
().
tag_due
())
add_dependencies
(
'partof'
)
add_dependencies
(
'before'
)
add_dependencies
(
'after'
)
p_todo
.
set_creation_date
(
date
.
today
())
todo_text
=
_preprocess_input_todo
(
p_todo_text
)
todo
=
self
.
todolist
.
add
(
todo_text
)
_postprocess_input_todo
(
todo
)
self
.
out
(
self
.
printer
.
print_todo
(
todo
))
def
execute
(
self
):
""" Adds a todo item to the list. """
if
not
super
(
AddCommand
,
self
).
execute
():
return
False
if
self
.
text
:
self
.
_preprocess_input_todo
()
self
.
todo
=
self
.
todolist
.
add
(
self
.
text
)
self
.
_postprocess_input_todo
()
self
.
printer
.
add_filter
(
PrettyPrinterNumbers
(
self
.
todolist
))
self
.
_process_flags
()
if
self
.
from_file
:
new_todos
=
self
.
get_todos_from_file
()
self
.
printer
.
add_filter
(
PrettyPrinterNumbers
(
self
.
todolist
))
self
.
out
(
self
.
printer
.
print_todo
(
self
.
todo
)
)
for
todo
in
new_todos
:
self
.
_add_todo
(
todo
)
else
:
self
.
error
(
self
.
usage
())
if
self
.
text
:
self
.
_add_todo
(
self
.
text
)
else
:
self
.
error
(
self
.
usage
())
def
usage
(
self
):
return
"""Synopsis: add <text>"""
return
"""Synopsis:
add <text>
add -f <file>
add -f -"""
def
help
(
self
):
return
"""
\
...
...
@@ -114,4 +153,6 @@ This subcommand automatically adds the creation date to the added item.
todo number (not the dependency number).
Example: add "Subtask partof:1"
-f : Add todo items from specified <file> or from standard input.
"""
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment