mirror of
https://github.com/GothenburgBitFactory/taskwarrior.git
synced 2025-06-26 10:54:26 +02:00
Import YAML
- Implemented the import-yaml.pl external script. - Added unit tests. - Fixed problem where the pending tasks were not loaded prior to uuid verification.
This commit is contained in:
parent
fa973f734b
commit
751e8e7f90
4 changed files with 195 additions and 56 deletions
143
scripts/add-ons/import-yaml.pl
Executable file
143
scripts/add-ons/import-yaml.pl
Executable file
|
@ -0,0 +1,143 @@
|
||||||
|
#! /usr/bin/perl
|
||||||
|
################################################################################
|
||||||
|
## taskwarrior - a command line task list manager.
|
||||||
|
##
|
||||||
|
## Copyright 2006 - 2011, Paul Beckingham, Federico Hernandez.
|
||||||
|
## All rights reserved.
|
||||||
|
##
|
||||||
|
## This program is free software; you can redistribute it and/or modify it under
|
||||||
|
## the terms of the GNU General Public License as published by the Free Software
|
||||||
|
## Foundation; either version 2 of the License, or (at your option) any later
|
||||||
|
## version.
|
||||||
|
##
|
||||||
|
## This program is distributed in the hope that it will be useful, but WITHOUT
|
||||||
|
## ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
|
||||||
|
## FOR A PARTICULAR PURPOSE. See the GNU General Public License for more
|
||||||
|
## details.
|
||||||
|
##
|
||||||
|
## You should have received a copy of the GNU General Public License along with
|
||||||
|
## this program; if not, write to the
|
||||||
|
##
|
||||||
|
## Free Software Foundation, Inc.,
|
||||||
|
## 51 Franklin Street, Fifth Floor,
|
||||||
|
## Boston, MA
|
||||||
|
## 02110-1301
|
||||||
|
## USA
|
||||||
|
##
|
||||||
|
################################################################################
|
||||||
|
|
||||||
|
use strict;
|
||||||
|
use warnings;
|
||||||
|
use Time::Local;
|
||||||
|
|
||||||
|
my @tasks;
|
||||||
|
|
||||||
|
my $status = '';
|
||||||
|
my $uuid = '';
|
||||||
|
my $priority = '';
|
||||||
|
my $entry = '';
|
||||||
|
my $end = '';
|
||||||
|
my $project = '';
|
||||||
|
my $tags = '';
|
||||||
|
my $description = '';
|
||||||
|
my $due = '';
|
||||||
|
|
||||||
|
my %annotations;
|
||||||
|
my $mid_anno = 0;
|
||||||
|
my $anno_entry = '';
|
||||||
|
my $anno_desc = '';
|
||||||
|
|
||||||
|
while (my $yaml = <>)
|
||||||
|
{
|
||||||
|
chomp $yaml;
|
||||||
|
next if $yaml =~ /\sid:/;
|
||||||
|
next if $yaml =~ /\sannotations:/;
|
||||||
|
|
||||||
|
if ($yaml =~ /task:|\.\.\./ && $description ne '')
|
||||||
|
{
|
||||||
|
# Compose the JSON
|
||||||
|
my $json = "{\"status\":\"${status}\"";
|
||||||
|
$json .= ",\"uuid\":\"${uuid}\"" if $uuid ne '';
|
||||||
|
$json .= ",\"priority\":\"${priority}\"" if $priority ne '';
|
||||||
|
$json .= ",\"project\":\"${project}\"" if $project ne '';
|
||||||
|
$json .= ",\"entry\":\"${entry}\"" if $entry ne '';
|
||||||
|
$json .= ",\"end\":\"${end}\"" if $end ne '';
|
||||||
|
$json .= ",\"due\":\"${due}\"" if $due ne '';
|
||||||
|
$json .= ",\"tags\":\"${tags}\"" if $tags ne '';
|
||||||
|
$json .= ",\"description\":\"${description}\"}";
|
||||||
|
|
||||||
|
for my $key (sort keys %annotations)
|
||||||
|
{
|
||||||
|
$json .= ",\"${key}\":\"" . $annotations{$key} . "\""
|
||||||
|
}
|
||||||
|
|
||||||
|
push @tasks, $json;
|
||||||
|
|
||||||
|
$status = $uuid = $priority = $entry = $end = $project = $tags =
|
||||||
|
$description = $due = $anno_entry = $anno_desc = '';
|
||||||
|
$mid_anno = 0;
|
||||||
|
%annotations = ();
|
||||||
|
$yaml = '';
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
$mid_anno = 1 if $yaml =~ /annotation:/;
|
||||||
|
$anno_entry = $1 if $mid_anno && $yaml =~ /entry:\s*(\S+)/;
|
||||||
|
if ($mid_anno)
|
||||||
|
{
|
||||||
|
if ($yaml =~ /description:\s*(.+)/)
|
||||||
|
{
|
||||||
|
$anno_desc = $1;
|
||||||
|
$mid_anno = 0;
|
||||||
|
|
||||||
|
$annotations{'annotation_' . epoch ($anno_entry)} = $anno_desc;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
$entry = $1 if $yaml =~ /entry:\s*(\S+)/;
|
||||||
|
$description = $1 if $yaml =~ /description:\s*(.+)/;
|
||||||
|
}
|
||||||
|
|
||||||
|
$status = $1 if $yaml =~ /status:\s*(\S+)/;
|
||||||
|
$uuid = $1 if $yaml =~ /uuid:\s*(\S+)/;
|
||||||
|
$priority = $1 if $yaml =~ /priority:\s*(\S+)/;
|
||||||
|
$project = $1 if $yaml =~ /project:\s*(\S+)/;
|
||||||
|
$end = $1 if $yaml =~ /end:\s*(\S+)/;
|
||||||
|
$due = $1 if $yaml =~ /due:\s*(\S+)/;
|
||||||
|
$tags = $1 if $yaml =~ /tags:\s*(\S+)/;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
print "[\n", join (",\n", @tasks), "\n]\n";
|
||||||
|
exit 0;
|
||||||
|
|
||||||
|
################################################################################
|
||||||
|
sub epoch
|
||||||
|
{
|
||||||
|
my ($input) = @_;
|
||||||
|
|
||||||
|
my ($Y, $M, $D, $h, $m, $s) = $input =~ /(\d{4})(\d{2})(\d{2})T(\d{2})(\d{2})(\d{2})Z/;
|
||||||
|
|
||||||
|
return timegm ($s, $m, $h, $D, $M-1, $Y-1900);
|
||||||
|
}
|
||||||
|
|
||||||
|
################################################################################
|
||||||
|
|
||||||
|
__DATA__
|
||||||
|
%YAML 1.1
|
||||||
|
---
|
||||||
|
task:
|
||||||
|
annotations:
|
||||||
|
annotation:
|
||||||
|
entry: 20100706T025311Z
|
||||||
|
description: Also needs to ignore control codes
|
||||||
|
description: text.cpp/extractLines needs to calculate string length in a way that supports UTF8
|
||||||
|
entry: 20090319T151633Z
|
||||||
|
id: 9
|
||||||
|
project: task-2.0
|
||||||
|
status: pending
|
||||||
|
tags: bug,utf8,next
|
||||||
|
uuid: 0c4cf066-9413-4862-9dc8-0793f158a649
|
||||||
|
...
|
||||||
|
|
|
@ -1771,13 +1771,11 @@ int TDB2::id (const std::string& uuid)
|
||||||
}
|
}
|
||||||
|
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
|
// Make sure the specified UUID does not already exist in the data.
|
||||||
bool TDB2::verifyUniqueUUID (const std::string& uuid)
|
bool TDB2::verifyUniqueUUID (const std::string& uuid)
|
||||||
{
|
{
|
||||||
if (pending.id (uuid) != 0 ||
|
pending.get_tasks ();
|
||||||
completed.id (uuid) != 0)
|
return pending.id (uuid) != 0 ? false : true;
|
||||||
return false;
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
|
@ -174,19 +174,12 @@ int CmdImport::execute (std::string& output)
|
||||||
task.setAnnotations (annos);
|
task.setAnnotations (annos);
|
||||||
}
|
}
|
||||||
|
|
||||||
// TODO Implement
|
// Attributes without columns are simply added.
|
||||||
else if (i->first == "parent")
|
else if (i->first == "parent" ||
|
||||||
{
|
i->first == "mask" ||
|
||||||
}
|
i->first == "imask")
|
||||||
|
|
||||||
// TODO Implement
|
|
||||||
else if (i->first == "mask")
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
// TODO Implement
|
|
||||||
else if (i->first == "imask")
|
|
||||||
{
|
{
|
||||||
|
task.set (i->first, unquoteText (i->second->dump ()));
|
||||||
}
|
}
|
||||||
|
|
||||||
else
|
else
|
||||||
|
|
|
@ -28,7 +28,7 @@
|
||||||
|
|
||||||
use strict;
|
use strict;
|
||||||
use warnings;
|
use warnings;
|
||||||
use Test::More tests => 20;
|
use Test::More tests => 16;
|
||||||
|
|
||||||
# Create the rc file.
|
# Create the rc file.
|
||||||
if (open my $fh, '>', 'import.rc')
|
if (open my $fh, '>', 'import.rc')
|
||||||
|
@ -49,19 +49,19 @@ if (open my $fh, '>', 'import.txt')
|
||||||
description: zero
|
description: zero
|
||||||
project: A
|
project: A
|
||||||
status: pending
|
status: pending
|
||||||
entry: 1234567889
|
entry: 20110901T120000Z
|
||||||
task:
|
task:
|
||||||
uuid: 11111111-1111-1111-1111-111111111111
|
uuid: 11111111-1111-1111-1111-111111111111
|
||||||
description: one
|
description: one
|
||||||
project: B
|
project: B
|
||||||
status: pending
|
status: pending
|
||||||
entry: 1234567889
|
entry: 20110902T120000Z
|
||||||
task:
|
task:
|
||||||
uuid: 22222222-2222-2222-2222-222222222222
|
uuid: 22222222-2222-2222-2222-222222222222
|
||||||
description: two
|
description: two
|
||||||
status: completed
|
status: completed
|
||||||
entry: 1234524689
|
entry: 20110903T010000Z
|
||||||
end: 1234524690
|
end: 20110904T120000Z
|
||||||
...
|
...
|
||||||
EOF
|
EOF
|
||||||
|
|
||||||
|
@ -69,9 +69,12 @@ EOF
|
||||||
ok (-r 'import.txt', 'Created sample import data');
|
ok (-r 'import.txt', 'Created sample import data');
|
||||||
}
|
}
|
||||||
|
|
||||||
my $output = qx{../src/task rc:import.rc import import.txt};
|
# Convert YAML --> task JSON.
|
||||||
like ($output, qr/Imported 3 tasks successfully./, 'no errors');
|
qx{../scripts/add-ons/import-yaml.pl <import.txt >import.json};
|
||||||
# Imported 3 tasks successfully.
|
|
||||||
|
# Import the JSON.
|
||||||
|
my $output = qx{../src/task rc:import.rc import import.json};
|
||||||
|
like ($output, qr/Imported 3 tasks\./, '3 tasks imported');
|
||||||
|
|
||||||
$output = qx{../src/task rc:import.rc list};
|
$output = qx{../src/task rc:import.rc list};
|
||||||
# ID Project Pri Due Active Age Description
|
# ID Project Pri Due Active Age Description
|
||||||
|
@ -94,55 +97,57 @@ $output = qx{../src/task rc:import.rc completed};
|
||||||
|
|
||||||
unlike ($output, qr/1.+A.+zero/, 't1 missing');
|
unlike ($output, qr/1.+A.+zero/, 't1 missing');
|
||||||
unlike ($output, qr/2.+B.+one/, 't2 missing');
|
unlike ($output, qr/2.+B.+one/, 't2 missing');
|
||||||
like ($output, qr/2\/13\/2009.+two/, 't3 present');
|
like ($output, qr/9\/4\/2011.+two/, 't3 present');
|
||||||
|
|
||||||
# Make sure that a duplicate task cannot be imported.
|
# Make sure that a duplicate task cannot be imported.
|
||||||
$output = qx{../src/task rc:import.rc import import.txt};
|
$output = qx{../src/task rc:import.rc import import.json};
|
||||||
like ($output, qr/Cannot add task because the uuid .+ is not unique\./, 'error on duplicate uuid');
|
like ($output, qr/Cannot add task because the uuid '.{36}' is not unique\./, 'error on duplicate uuid');
|
||||||
|
|
||||||
# Create import file.
|
# Create import file.
|
||||||
if (open my $fh, '>', 'import2.txt')
|
if (open my $fh, '>', 'import.txt')
|
||||||
{
|
{
|
||||||
print $fh <<EOF;
|
print $fh <<EOF;
|
||||||
|
%YAML 1.1
|
||||||
|
---
|
||||||
task:
|
task:
|
||||||
uuid: 44444444-4444-4444-4444-444444444444
|
uuid: 44444444-4444-4444-4444-444444444444
|
||||||
description: three
|
description: three
|
||||||
status: pending
|
status: pending
|
||||||
entry: 1234567889
|
entry: 1234567889
|
||||||
|
...
|
||||||
EOF
|
EOF
|
||||||
|
|
||||||
close $fh;
|
close $fh;
|
||||||
ok (-r 'import2.txt', 'Created second sample import data');
|
ok (-r 'import2.txt', 'Created second sample import data');
|
||||||
}
|
}
|
||||||
|
|
||||||
$output = qx{../src/task rc:import.rc import import2.txt};
|
# Convert YAML --> task JSON.
|
||||||
like ($output, qr/Imported 1 tasks successfully./, 'no errors');
|
qx{../scripts/add-ons/import-yaml.pl <import.txt >import.json};
|
||||||
# Imported 1 tasks successfully.
|
|
||||||
|
# Import the JSON.
|
||||||
|
$output = qx{../src/task rc:import.rc import import.json};
|
||||||
|
like ($output, qr/Imported 1 tasks\./, '1 task imported');
|
||||||
|
|
||||||
|
# Verify.
|
||||||
|
$output = qx{../src/task rc:import.rc list};
|
||||||
|
# ID Project Pri Due Active Age Description
|
||||||
|
# -- ------- --- --- ------ ---- -----------
|
||||||
|
# 3 2.6y three
|
||||||
|
# 1 A 3d zero
|
||||||
|
# 2 B 2d one
|
||||||
|
like ($output, qr/1.+A.+zero/, 't1 present');
|
||||||
|
like ($output, qr/2.+B.+one/, 't2 present');
|
||||||
|
like ($output, qr/3.+three/, 't3 present');
|
||||||
|
|
||||||
# Cleanup.
|
# Cleanup.
|
||||||
unlink 'import.txt';
|
unlink qw(pending.data completed.data undo.data backlog.data synch.key import.rc import.txt import.json);
|
||||||
ok (!-r 'import.txt', 'Removed import.txt');
|
ok (! -r 'pending.data' &&
|
||||||
|
! -r 'completed.data' &&
|
||||||
unlink 'import2.txt';
|
! -r 'undo.data' &&
|
||||||
ok (!-r 'import2.txt', 'Removed import2.txt');
|
! -r 'backlog.data' &&
|
||||||
|
! -r 'synch_key.data' &&
|
||||||
unlink 'pending.data';
|
! -r 'import.rc' &&
|
||||||
ok (!-r 'pending.data', 'Removed pending.data');
|
! -r 'import.txt' &&
|
||||||
|
! -r 'import.json', 'Cleanup');
|
||||||
unlink 'completed.data';
|
|
||||||
ok (!-r 'completed.data', 'Removed completed.data');
|
|
||||||
|
|
||||||
unlink 'undo.data';
|
|
||||||
ok (!-r 'undo.data', 'Removed undo.data');
|
|
||||||
|
|
||||||
unlink 'backlog.data';
|
|
||||||
ok (!-r 'backlog.data', 'Removed backlog.data');
|
|
||||||
|
|
||||||
unlink 'synch.key';
|
|
||||||
ok (!-r 'synch.key', 'Removed synch.key');
|
|
||||||
|
|
||||||
unlink 'import.rc';
|
|
||||||
ok (!-r 'import.rc', 'Removed import.rc');
|
|
||||||
|
|
||||||
exit 0;
|
exit 0;
|
||||||
|
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue