mirror of
https://github.com/GothenburgBitFactory/taskwarrior.git
synced 2025-06-26 10:54:26 +02:00

- When creating a new .taskrc file, no newlines were included at EOL. This needs a 1.5.1 regression test.
320 lines
12 KiB
C++
320 lines
12 KiB
C++
///////////////////////////////////////////////////////////////////////////////
|
|
// task - a command line task list manager.
|
|
//
|
|
// Copyright 2006 - 2009, Paul Beckingham.
|
|
// 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
|
|
//
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
#include <iostream>
|
|
#include <fstream>
|
|
#include <sstream>
|
|
#include <sys/types.h>
|
|
#include <sys/stat.h>
|
|
#include <unistd.h>
|
|
#include <stdlib.h>
|
|
#include <pwd.h>
|
|
#include "task.h"
|
|
#include "Config.h"
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
// These are default (but overridable) reports. These entries are necessary
|
|
// because these three reports were converted from hard-coded reports to custom
|
|
// reports, and therefore need these config file entries. However, users are
|
|
// already used to seeing these five reports, but do not have these entries.
|
|
// The choice was a) make users edit their .taskrc files, b) write a .taskrc
|
|
// upgrade program to make the change, or c) this.
|
|
Config::Config ()
|
|
{
|
|
(*this)["report.long.description"] = "Lists all task, all data, matching the specified criteria";
|
|
(*this)["report.long.columns"] = "id,project,priority,entry,start,due,recur,age,tags,description";
|
|
(*this)["report.long.sort"] = "due+,priority-,project+";
|
|
|
|
(*this)["report.list.description"] = "Lists all tasks matching the specified criteria";
|
|
(*this)["report.list.columns"] = "id,project,priority,due,active,age,description";
|
|
(*this)["report.list.sort"] = "due+,priority-,project+";
|
|
|
|
(*this)["report.ls.description"] = "Minimal listing of all tasks matching the specified criteria";
|
|
(*this)["report.ls.columns"] = "id,project,priority,description";
|
|
(*this)["report.ls.sort"] = "priority-,project+";
|
|
|
|
(*this)["report.newest.description"] = "Shows the newest tasks";
|
|
(*this)["report.newest.columns"] = "id,project,priority,due,active,age,description";
|
|
(*this)["report.newest.sort"] = "id-";
|
|
(*this)["report.newest.limit"] = "10";
|
|
|
|
(*this)["report.oldest.description"] = "Shows the oldest tasks";
|
|
(*this)["report.oldest.columns"] = "id,project,priority,due,active,age,description";
|
|
(*this)["report.oldest.sort"] = "id+";
|
|
(*this)["report.oldest.limit"] = "10";
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
Config::Config (const std::string& file)
|
|
{
|
|
load (file);
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
// Read the Configuration file and populate the *this map. The file format is
|
|
// simply lines with name=value pairs. Whitespace between name, = and value is
|
|
// not tolerated, but blank lines and comments starting with # are allowed.
|
|
bool Config::load (const std::string& file)
|
|
{
|
|
std::ifstream in;
|
|
in.open (file.c_str (), std::ifstream::in);
|
|
if (in.good ())
|
|
{
|
|
std::string line;
|
|
while (getline (in, line))
|
|
{
|
|
// Remove comments.
|
|
size_type pound = line.find ("#");
|
|
if (pound != std::string::npos)
|
|
line = line.substr (0, pound);
|
|
|
|
line = trim (line, " \t");
|
|
|
|
// Skip empty lines.
|
|
if (line.length () > 0)
|
|
{
|
|
size_type equal = line.find ("=");
|
|
if (equal != std::string::npos)
|
|
{
|
|
std::string key = trim (line.substr (0, equal), " \t");
|
|
std::string value = trim (line.substr (equal+1, line.length () - equal), " \t");
|
|
(*this)[key] = value;
|
|
}
|
|
}
|
|
}
|
|
|
|
in.close ();
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
void Config::createDefault (const std::string& home)
|
|
{
|
|
// Strip trailing slash off home directory, if necessary.
|
|
std::string terminatedHome = home;
|
|
if (home[home.length () - 1] == '/')
|
|
terminatedHome = home.substr (0, home.length () - 1);
|
|
|
|
// Determine default names of init file and task directory.
|
|
std::string rcFile = terminatedHome + "/.taskrc";
|
|
std::string dataDir = terminatedHome + "/.task";;
|
|
|
|
// If rcFile is not found, offer to create one.
|
|
if (-1 == access (rcFile.c_str (), F_OK))
|
|
{
|
|
if (confirm (
|
|
"A configuration file could not be found in "
|
|
+ rcFile
|
|
+ "\n\n"
|
|
+ "Would you like a sample .taskrc created, so task can proceed?"))
|
|
{
|
|
// Create a sample .taskrc file.
|
|
FILE* out;
|
|
if ((out = fopen (rcFile.c_str (), "w")))
|
|
{
|
|
fprintf (out, "data.location=%s\n", dataDir.c_str ());
|
|
fprintf (out, "confirmation=yes\n");
|
|
fprintf (out, "next=2\n");
|
|
fprintf (out, "dateformat=m/d/Y\n");
|
|
fprintf (out, "#monthsperline=2\n");
|
|
fprintf (out, "curses=on\n");
|
|
fprintf (out, "color=on\n");
|
|
fprintf (out, "due=7\n");
|
|
fprintf (out, "nag=You have higher priority tasks.\n");
|
|
fprintf (out, "locking=on\n");
|
|
|
|
fprintf (out, "color.overdue=bold_red\n");
|
|
fprintf (out, "color.due=bold_yellow\n");
|
|
fprintf (out, "color.pri.H=bold\n");
|
|
fprintf (out, "#color.pri.M=on_yellow\n");
|
|
fprintf (out, "#color.pri.L=on_green\n");
|
|
fprintf (out, "#color.pri.none=white on_blue\n");
|
|
fprintf (out, "color.active=bold_cyan\n");
|
|
fprintf (out, "color.tagged=yellow\n");
|
|
fprintf (out, "#color.tag.bug=yellow\n");
|
|
fprintf (out, "#color.project.garden=on_green\n");
|
|
fprintf (out, "#color.keyword.car=on_blue\n");
|
|
fprintf (out, "#color.recurring=on_red\n");
|
|
fprintf (out, "#shadow.file=%s/shadow.txt\n", dataDir.c_str ());
|
|
fprintf (out, "#shadow.command=list\n");
|
|
fprintf (out, "#shadow.notify=on\n");
|
|
fprintf (out, "#default.project=foo\n");
|
|
fprintf (out, "#default.priority=M\n");
|
|
fprintf (out, "default.command=list\n");
|
|
|
|
// Custom reports.
|
|
fprintf (out, "# Fields: id,uuid,project,priority,entry,start,due,recur,age,active,tags,description\n");
|
|
fprintf (out, "# Description: This report is ...\n");
|
|
fprintf (out, "# Sort: due+,priority-,project+\n");
|
|
fprintf (out, "# Filter: pro:x pri:H +bug\n");
|
|
fprintf (out, "# Limit: 10\n");
|
|
|
|
fprintf (out, "report.long.description=Lists all task, all data, matching the specified criteria\n");
|
|
fprintf (out, "report.long.columns=id,project,priority,entry,start,due,recur,age,tags,description\n");
|
|
fprintf (out, "report.long.sort=due+,priority-,project+\n");
|
|
|
|
fprintf (out, "report.list.description=Lists all tasks matching the specified criteria\n");
|
|
fprintf (out, "report.list.columns=id,project,priority,due,active,age,description\n");
|
|
fprintf (out, "report.list.sort=due+,priority-,project+\n");
|
|
|
|
fprintf (out, "report.ls.description=Minimal listing of all tasks matching the specified criteria\n");
|
|
fprintf (out, "report.ls.columns=id,project,priority,description\n");
|
|
fprintf (out, "report.ls.sort=priority-,project+\n");
|
|
|
|
fprintf (out, "report.newest.description=Shows the newest tasks\n");
|
|
fprintf (out, "report.newest.columns=id,project,priority,due,active,age,description\n");
|
|
fprintf (out, "report.newest.sort=id-\n");
|
|
fprintf (out, "report.newest.limit=10\n");
|
|
|
|
fprintf (out, "report.oldest.description=Shows the oldest tasks\n");
|
|
fprintf (out, "report.oldest.columns=id,project,priority,due,active,age,description\n");
|
|
fprintf (out, "report.oldest.sort=id+\n");
|
|
fprintf (out, "report.oldest.limit=10\n");
|
|
|
|
fclose (out);
|
|
|
|
std::cout << "Done." << std::endl;
|
|
}
|
|
}
|
|
}
|
|
|
|
this->load (rcFile);
|
|
|
|
if (-1 == access (dataDir.c_str (), F_OK))
|
|
mkdir (dataDir.c_str (), S_IRWXU);
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
// Return the configuration value given the specified key.
|
|
const std::string Config::get (const char* key)
|
|
{
|
|
return this->get (std::string (key));
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
// Return the configuration value given the specified key. If a default_value
|
|
// is present, it will be the returned value in the event of a missing key.
|
|
const std::string Config::get (
|
|
const char* key,
|
|
const char* default_value)
|
|
{
|
|
return this->get (std::string (key), std::string (default_value));
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
// Return the configuration value given the specified key.
|
|
const std::string Config::get (const std::string& key)
|
|
{
|
|
return (*this)[key];
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
// Return the configuration value given the specified key. If a default_value
|
|
// is present, it will be the returned value in the event of a missing key.
|
|
const std::string Config::get (
|
|
const std::string& key,
|
|
const std::string& default_value)
|
|
{
|
|
if ((*this).find (key) != (*this).end ())
|
|
return (*this)[key];
|
|
|
|
return default_value;
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
bool Config::get (const std::string& key, bool default_value)
|
|
{
|
|
if ((*this).find (key) != (*this).end ())
|
|
{
|
|
std::string value = lowerCase ((*this)[key]);
|
|
|
|
if (value == "t" ||
|
|
value == "true" ||
|
|
value == "1" ||
|
|
value == "yes" ||
|
|
value == "on" ||
|
|
value == "enable" ||
|
|
value == "enabled")
|
|
return true;
|
|
|
|
return false;
|
|
}
|
|
|
|
return default_value;
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
int Config::get (const std::string& key, const int default_value)
|
|
{
|
|
if ((*this).find (key) != (*this).end ())
|
|
return ::atoi ((*this)[key].c_str ());
|
|
|
|
return default_value;
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
double Config::get (const std::string& key, const double default_value)
|
|
{
|
|
if ((*this).find (key) != (*this).end ())
|
|
return ::atof ((*this)[key].c_str ());
|
|
|
|
return default_value;
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
void Config::set (const std::string& key, const int value)
|
|
{
|
|
char v[24];
|
|
sprintf (v, "%d", value);
|
|
(*this)[key] = v;
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
void Config::set (const std::string& key, const double value)
|
|
{
|
|
char v[32];
|
|
sprintf (v, "%f", value);
|
|
(*this)[key] = v;
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
void Config::set (const std::string& key, const std::string& value)
|
|
{
|
|
(*this)[key] = value;
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
// Provide a vector of all configuration keys.
|
|
void Config::all (std::vector<std::string>& items)
|
|
{
|
|
foreach (i, *this)
|
|
items.push_back (i->first);
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|