New rename-project command.

The `rename-project` command will rename a project and update all of the action
items associated with that project.

* Fills in some missing information in the online help.
* Fixes a bug in the finGtdRootDir function. It was not properly handling
  relative paths.
* Fixes a bug where the reconfigure command was referring to the wrong
  configuration file.
This commit is contained in:
Joanthan Bernard 2014-12-01 12:05:55 -06:00
parent 2c8180d9b2
commit f95dc91707
4 changed files with 96 additions and 9 deletions

View File

@ -1,8 +1,8 @@
#Wed, 19 Nov 2014 12:50:01 -0600 #Mon, 01 Dec 2014 12:03:37 -0600
lib.local=true lib.local=true
name=jdb-gtd name=jdb-gtd
version=1.10 version=1.11
nailgun.classpath.dir=/home/jdbernard/programs/nailgun/classpath nailgun.classpath.dir=/home/jdbernard/programs/nailgun/classpath
executable.jar=true executable.jar=true
main.class=com.jdblabs.gtd.cli.GTDCLI main.class=com.jdblabs.gtd.cli.GTDCLI
build.number=1 build.number=10

View File

@ -19,6 +19,7 @@ package com.jdblabs.gtd
* calendar to "schedule" items, but only to represent items which must be * calendar to "schedule" items, but only to represent items which must be
* done by or on that day. * done by or on that day.
* * `details`: more information related to this item. * * `details`: more information related to this item.
* * `project`: the name of the project with which this item is associated.
* @org gtd.jdb-labs.com/Item * @org gtd.jdb-labs.com/Item
*/ */
public class Item { public class Item {

View File

@ -104,7 +104,7 @@ public class Util {
def gtdDirs = [:] def gtdDirs = [:]
/// Start by considering the current directory as a candidate. /// Start by considering the current directory as a candidate.
File currentDir = givenDir File currentDir = givenDir.canonicalFile
while (currentDir != null) { while (currentDir != null) {
/// We recognize the GTD root directory when it contains all of the /// We recognize the GTD root directory when it contains all of the
/// GTD top-level directories. /// GTD top-level directories.

View File

@ -17,6 +17,9 @@ import com.jdblabs.gtd.Item
import com.jdblabs.gtd.PropertyHelp import com.jdblabs.gtd.PropertyHelp
import com.jdbernard.util.LightOptionParser import com.jdbernard.util.LightOptionParser
import com.martiansoftware.nailgun.NGContext import com.martiansoftware.nailgun.NGContext
import java.io.FileFilter
import java.nio.file.Files
import java.nio.file.Path
import java.security.MessageDigest import java.security.MessageDigest
import org.joda.time.DateMidnight import org.joda.time.DateMidnight
import org.joda.time.DateTime import org.joda.time.DateTime
@ -24,6 +27,7 @@ import org.slf4j.Logger as SFL4JLogger
import org.slf4j.LoggerFactory import org.slf4j.LoggerFactory
import static com.jdblabs.gtd.Util.* import static com.jdblabs.gtd.Util.*
import static java.nio.file.StandardCopyOption.*
/** /**
* Command-line helper for working with this implementation of the Getting * Command-line helper for working with this implementation of the Getting
@ -107,7 +111,7 @@ public class GTDCLI {
/// read afresh the configuration file. /// read afresh the configuration file.
nailgunInst = null nailgunInst = null
nailgunInst = new GTDCLI(new File( nailgunInst = new GTDCLI(new File(
System.getProperty("user.home"), ".gritterrc")) System.getProperty("user.home"), ".gtdclirc"))
nailgunInst.run(args) } } nailgunInst.run(args) } }
@ -243,6 +247,7 @@ public class GTDCLI {
case ~/ls|list/: ls(parsedArgs); break; case ~/ls|list/: ls(parsedArgs); break;
case ~/debug/: debug(parsedArgs); break; case ~/debug/: debug(parsedArgs); break;
case ~/delegate/: delegateAction(parsedArgs); break; case ~/delegate/: delegateAction(parsedArgs); break;
case ~/rp|rename-project/: renameProject(parsedArgs); break;
default: default:
log.error "Unrecognized command: ${command}" log.error "Unrecognized command: ${command}"
break } } } break } } }
@ -787,6 +792,56 @@ public class GTDCLI {
/// Delete the original file. /// Delete the original file.
oldFile.delete() } } oldFile.delete() } }
/** #### `rename-project`
* Implement the `rename-project` command. This will rename the project
* directory in TODO as well as change the project reference in any of the
* items from the `next-actions` contexts.
*
* `gtd help rename-project`.
*
* [help-rename-project]: jlp://gtd.jdb-labs.com/cli/GTDCLI/help/rename-project
*/
protected void renameProject(LinkedList args) {
def projectName = args.poll()
def newName = args.poll()
if (!projectName || !newName) {
log.error "The 'gtd rename-project' command requires two " +
"parameters: <existing-project-name> and a <new-name>."
return }
def projectDir = new File(gtdDirs.projects, projectName)
if (!projectDir.exists() || !projectDir.isDirectory()) {
log.error "There is no directory named '$projectName' in the " +
"'projects' directory."
return }
def newDir = new File(gtdDirs.projects, newName)
if (newDir.exists()) {
log.error "There is already a project named '$newName'."
return }
// Perform the rename of the directory itself.
try { Files.move(projectDir.toPath(), newDir.toPath(), REPLACE_EXISTING) }
catch (Exception e) {
log.error "Unable to rename the project: ${e.localizedMessage}."
return }
// Update all of the items associated with this project.
def projectFiles = newDir.
listFiles({ File f ->
f.exists() && !f.isHidden() &&
f.isFile() && !f.name.startsWith('.') } as FileFilter)
def allProjectItems = projectFiles.collectMany { f ->
findAllCopies(f, gtdDirs.root).collect { new Item(it) } }
allProjectItems.each {
it.project = newName
it.save() }
println "Project renamed. ${allProjectItems.size()} items updated." }
private void print(String msg) { log.info(msg) } private void print(String msg) { log.info(msg) }
private void println(String line) { log.info(line + EOL) } private void println(String line) { log.info(line + EOL) }
@ -809,20 +864,41 @@ options are:
top-level commands: top-level commands:
help <command> Print detailed help about a command. help <command> Print detailed help about a command.
process Process inbox items systematically. process Process inbox items systematically.
done <action-file> Mark an action as done. This will automatically done <action-file> Mark an action as done. This will automatically
take care of duplicates of the action in project take care of duplicates of the action in project
or next-actions sub-folders. or next-actions sub-folders.
calendar Show the tasks with specific days assigned to calendar Show the tasks with specific days assigned to
them, sorted by date. them, sorted by date.
list-copies <action-file> Given an action item, list all the other places list-copies <action-file> Given an action item, list all the other places
there the same item is filed (cross-reference there the same item is filed (cross-reference
with a project folder, for example). with a project folder, for example).
new Interactively create a new action item in the new Interactively create a new action item in the
current folder. current folder.
tickler Search the tickler file for items that need to be tickler Search the tickler file for items that need to be
delivered and move them to the *next-actions* delivered and move them to the *next-actions*
folder.""" folder.
list, ls [<context> ...] List all the tasks for a given set of contexts
projects.
debug n
delegate Move an item from a next-action context or a
project folder to a waiting context and attach
the name of the party now responsible for the
item.
rename-project, rp <existing-project> <new-name>
Rename a project directory and update any task
items that reference it."""
} else { } else {
def command = args.poll() def command = args.poll()
@ -926,10 +1002,11 @@ file for any items that should become active (based on their <tickle> property)
and moves them out of the tickler file and into the next-actions file.""" and moves them out of the tickler file and into the next-actions file."""
break break
/// Online help for the `ls`/`list-context` command. /// Online help for the `ls`/`list` command.
/// @org gtd.jdb-labs.com/cli/GTDCLI/help/ls /// @org gtd.jdb-labs.com/cli/GTDCLI/help/ls
case ~/ls|list-context/: println """\ case ~/ls|list/: println """\
usage gtd ls [<context> ...] usage gtd list [<context> ...]
or gtd ls [<context> ...]
This command lists all the tasks for a given context or project. The purpose is This command lists all the tasks for a given context or project. The purpose is
to list in one place items that are sitting in the next-actions folder or the to list in one place items that are sitting in the next-actions folder or the
@ -945,6 +1022,15 @@ This command moves an action item from a next-action context or project folder
to the delegate folder. It allows the user to attach the name of the newly to the delegate folder. It allows the user to attach the name of the newly
responsible party and optionally rename the item.""" responsible party and optionally rename the item."""
/// Online help for the `rename-project` command.
/// @org gtd.jdb-labs.com/cli/GTDCLI/help/rename-project
case ~/delegate/: println """\
usage gtd rename-project <existing-project> <new-name>
or gtd rp <existing-project> <new-name>
This command renames a project directory and updates any items that reference
the project."""
} }
} }
} }