--- /dev/null
+*~
+*.o
+*.cmi
+*.cmo
+*.cmx
+
+Makefile
+Makefile.in
+
+/.depend
+/META
+/aclocal.m4
+/autom4te.cache
+/compile
+/config.guess
+/config.h
+/config.h.in
+/config.log
+/config.status
+/config.sub
+/configure
+/dummy.c
+/install-sh
+/libtool
+/ltmain.sh
+/m4/libtool.m4
+/m4/ltoptions.m4
+/m4/ltsugar.m4
+/m4/ltversion.m4
+/m4/lt~obsolete.m4
+/missing
+/stamp-h1
+/todo
+/todo.spec
+/todo_config.ml
--- /dev/null
+ GNU GENERAL PUBLIC LICENSE
+ Version 2, June 1991
+
+ Copyright (C) 1989, 1991 Free Software Foundation, Inc.,
+ 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
+ Everyone is permitted to copy and distribute verbatim copies
+ of this license document, but changing it is not allowed.
+
+ Preamble
+
+ The licenses for most software are designed to take away your
+freedom to share and change it. By contrast, the GNU General Public
+License is intended to guarantee your freedom to share and change free
+software--to make sure the software is free for all its users. This
+General Public License applies to most of the Free Software
+Foundation's software and to any other program whose authors commit to
+using it. (Some other Free Software Foundation software is covered by
+the GNU Lesser General Public License instead.) You can apply it to
+your programs, too.
+
+ When we speak of free software, we are referring to freedom, not
+price. Our General Public Licenses are designed to make sure that you
+have the freedom to distribute copies of free software (and charge for
+this service if you wish), that you receive source code or can get it
+if you want it, that you can change the software or use pieces of it
+in new free programs; and that you know you can do these things.
+
+ To protect your rights, we need to make restrictions that forbid
+anyone to deny you these rights or to ask you to surrender the rights.
+These restrictions translate to certain responsibilities for you if you
+distribute copies of the software, or if you modify it.
+
+ For example, if you distribute copies of such a program, whether
+gratis or for a fee, you must give the recipients all the rights that
+you have. You must make sure that they, too, receive or can get the
+source code. And you must show them these terms so they know their
+rights.
+
+ We protect your rights with two steps: (1) copyright the software, and
+(2) offer you this license which gives you legal permission to copy,
+distribute and/or modify the software.
+
+ Also, for each author's protection and ours, we want to make certain
+that everyone understands that there is no warranty for this free
+software. If the software is modified by someone else and passed on, we
+want its recipients to know that what they have is not the original, so
+that any problems introduced by others will not reflect on the original
+authors' reputations.
+
+ Finally, any free program is threatened constantly by software
+patents. We wish to avoid the danger that redistributors of a free
+program will individually obtain patent licenses, in effect making the
+program proprietary. To prevent this, we have made it clear that any
+patent must be licensed for everyone's free use or not licensed at all.
+
+ The precise terms and conditions for copying, distribution and
+modification follow.
+
+ GNU GENERAL PUBLIC LICENSE
+ TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION
+
+ 0. This License applies to any program or other work which contains
+a notice placed by the copyright holder saying it may be distributed
+under the terms of this General Public License. The "Program", below,
+refers to any such program or work, and a "work based on the Program"
+means either the Program or any derivative work under copyright law:
+that is to say, a work containing the Program or a portion of it,
+either verbatim or with modifications and/or translated into another
+language. (Hereinafter, translation is included without limitation in
+the term "modification".) Each licensee is addressed as "you".
+
+Activities other than copying, distribution and modification are not
+covered by this License; they are outside its scope. The act of
+running the Program is not restricted, and the output from the Program
+is covered only if its contents constitute a work based on the
+Program (independent of having been made by running the Program).
+Whether that is true depends on what the Program does.
+
+ 1. You may copy and distribute verbatim copies of the Program's
+source code as you receive it, in any medium, provided that you
+conspicuously and appropriately publish on each copy an appropriate
+copyright notice and disclaimer of warranty; keep intact all the
+notices that refer to this License and to the absence of any warranty;
+and give any other recipients of the Program a copy of this License
+along with the Program.
+
+You may charge a fee for the physical act of transferring a copy, and
+you may at your option offer warranty protection in exchange for a fee.
+
+ 2. You may modify your copy or copies of the Program or any portion
+of it, thus forming a work based on the Program, and copy and
+distribute such modifications or work under the terms of Section 1
+above, provided that you also meet all of these conditions:
+
+ a) You must cause the modified files to carry prominent notices
+ stating that you changed the files and the date of any change.
+
+ b) You must cause any work that you distribute or publish, that in
+ whole or in part contains or is derived from the Program or any
+ part thereof, to be licensed as a whole at no charge to all third
+ parties under the terms of this License.
+
+ c) If the modified program normally reads commands interactively
+ when run, you must cause it, when started running for such
+ interactive use in the most ordinary way, to print or display an
+ announcement including an appropriate copyright notice and a
+ notice that there is no warranty (or else, saying that you provide
+ a warranty) and that users may redistribute the program under
+ these conditions, and telling the user how to view a copy of this
+ License. (Exception: if the Program itself is interactive but
+ does not normally print such an announcement, your work based on
+ the Program is not required to print an announcement.)
+
+These requirements apply to the modified work as a whole. If
+identifiable sections of that work are not derived from the Program,
+and can be reasonably considered independent and separate works in
+themselves, then this License, and its terms, do not apply to those
+sections when you distribute them as separate works. But when you
+distribute the same sections as part of a whole which is a work based
+on the Program, the distribution of the whole must be on the terms of
+this License, whose permissions for other licensees extend to the
+entire whole, and thus to each and every part regardless of who wrote it.
+
+Thus, it is not the intent of this section to claim rights or contest
+your rights to work written entirely by you; rather, the intent is to
+exercise the right to control the distribution of derivative or
+collective works based on the Program.
+
+In addition, mere aggregation of another work not based on the Program
+with the Program (or with a work based on the Program) on a volume of
+a storage or distribution medium does not bring the other work under
+the scope of this License.
+
+ 3. You may copy and distribute the Program (or a work based on it,
+under Section 2) in object code or executable form under the terms of
+Sections 1 and 2 above provided that you also do one of the following:
+
+ a) Accompany it with the complete corresponding machine-readable
+ source code, which must be distributed under the terms of Sections
+ 1 and 2 above on a medium customarily used for software interchange; or,
+
+ b) Accompany it with a written offer, valid for at least three
+ years, to give any third party, for a charge no more than your
+ cost of physically performing source distribution, a complete
+ machine-readable copy of the corresponding source code, to be
+ distributed under the terms of Sections 1 and 2 above on a medium
+ customarily used for software interchange; or,
+
+ c) Accompany it with the information you received as to the offer
+ to distribute corresponding source code. (This alternative is
+ allowed only for noncommercial distribution and only if you
+ received the program in object code or executable form with such
+ an offer, in accord with Subsection b above.)
+
+The source code for a work means the preferred form of the work for
+making modifications to it. For an executable work, complete source
+code means all the source code for all modules it contains, plus any
+associated interface definition files, plus the scripts used to
+control compilation and installation of the executable. However, as a
+special exception, the source code distributed need not include
+anything that is normally distributed (in either source or binary
+form) with the major components (compiler, kernel, and so on) of the
+operating system on which the executable runs, unless that component
+itself accompanies the executable.
+
+If distribution of executable or object code is made by offering
+access to copy from a designated place, then offering equivalent
+access to copy the source code from the same place counts as
+distribution of the source code, even though third parties are not
+compelled to copy the source along with the object code.
+
+ 4. You may not copy, modify, sublicense, or distribute the Program
+except as expressly provided under this License. Any attempt
+otherwise to copy, modify, sublicense or distribute the Program is
+void, and will automatically terminate your rights under this License.
+However, parties who have received copies, or rights, from you under
+this License will not have their licenses terminated so long as such
+parties remain in full compliance.
+
+ 5. You are not required to accept this License, since you have not
+signed it. However, nothing else grants you permission to modify or
+distribute the Program or its derivative works. These actions are
+prohibited by law if you do not accept this License. Therefore, by
+modifying or distributing the Program (or any work based on the
+Program), you indicate your acceptance of this License to do so, and
+all its terms and conditions for copying, distributing or modifying
+the Program or works based on it.
+
+ 6. Each time you redistribute the Program (or any work based on the
+Program), the recipient automatically receives a license from the
+original licensor to copy, distribute or modify the Program subject to
+these terms and conditions. You may not impose any further
+restrictions on the recipients' exercise of the rights granted herein.
+You are not responsible for enforcing compliance by third parties to
+this License.
+
+ 7. If, as a consequence of a court judgment or allegation of patent
+infringement or for any other reason (not limited to patent issues),
+conditions are imposed on you (whether by court order, agreement or
+otherwise) that contradict the conditions of this License, they do not
+excuse you from the conditions of this License. If you cannot
+distribute so as to satisfy simultaneously your obligations under this
+License and any other pertinent obligations, then as a consequence you
+may not distribute the Program at all. For example, if a patent
+license would not permit royalty-free redistribution of the Program by
+all those who receive copies directly or indirectly through you, then
+the only way you could satisfy both it and this License would be to
+refrain entirely from distribution of the Program.
+
+If any portion of this section is held invalid or unenforceable under
+any particular circumstance, the balance of the section is intended to
+apply and the section as a whole is intended to apply in other
+circumstances.
+
+It is not the purpose of this section to induce you to infringe any
+patents or other property right claims or to contest validity of any
+such claims; this section has the sole purpose of protecting the
+integrity of the free software distribution system, which is
+implemented by public license practices. Many people have made
+generous contributions to the wide range of software distributed
+through that system in reliance on consistent application of that
+system; it is up to the author/donor to decide if he or she is willing
+to distribute software through any other system and a licensee cannot
+impose that choice.
+
+This section is intended to make thoroughly clear what is believed to
+be a consequence of the rest of this License.
+
+ 8. If the distribution and/or use of the Program is restricted in
+certain countries either by patents or by copyrighted interfaces, the
+original copyright holder who places the Program under this License
+may add an explicit geographical distribution limitation excluding
+those countries, so that distribution is permitted only in or among
+countries not thus excluded. In such case, this License incorporates
+the limitation as if written in the body of this License.
+
+ 9. The Free Software Foundation may publish revised and/or new versions
+of the General Public License from time to time. Such new versions will
+be similar in spirit to the present version, but may differ in detail to
+address new problems or concerns.
+
+Each version is given a distinguishing version number. If the Program
+specifies a version number of this License which applies to it and "any
+later version", you have the option of following the terms and conditions
+either of that version or of any later version published by the Free
+Software Foundation. If the Program does not specify a version number of
+this License, you may choose any version ever published by the Free Software
+Foundation.
+
+ 10. If you wish to incorporate parts of the Program into other free
+programs whose distribution conditions are different, write to the author
+to ask for permission. For software which is copyrighted by the Free
+Software Foundation, write to the Free Software Foundation; we sometimes
+make exceptions for this. Our decision will be guided by the two goals
+of preserving the free status of all derivatives of our free software and
+of promoting the sharing and reuse of software generally.
+
+ NO WARRANTY
+
+ 11. BECAUSE THE PROGRAM IS LICENSED FREE OF CHARGE, THERE IS NO WARRANTY
+FOR THE PROGRAM, TO THE EXTENT PERMITTED BY APPLICABLE LAW. EXCEPT WHEN
+OTHERWISE STATED IN WRITING THE COPYRIGHT HOLDERS AND/OR OTHER PARTIES
+PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESSED
+OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
+MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE. THE ENTIRE RISK AS
+TO THE QUALITY AND PERFORMANCE OF THE PROGRAM IS WITH YOU. SHOULD THE
+PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF ALL NECESSARY SERVICING,
+REPAIR OR CORRECTION.
+
+ 12. IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING
+WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MAY MODIFY AND/OR
+REDISTRIBUTE THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES,
+INCLUDING ANY GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING
+OUT OF THE USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED
+TO LOSS OF DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY
+YOU OR THIRD PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER
+PROGRAMS), EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE
+POSSIBILITY OF SUCH DAMAGES.
+
+ END OF TERMS AND CONDITIONS
+
+ How to Apply These Terms to Your New Programs
+
+ If you develop a new program, and you want it to be of the greatest
+possible use to the public, the best way to achieve this is to make it
+free software which everyone can redistribute and change under these terms.
+
+ To do so, attach the following notices to the program. It is safest
+to attach them to the start of each source file to most effectively
+convey the exclusion of warranty; and each file should have at least
+the "copyright" line and a pointer to where the full notice is found.
+
+ <one line to give the program's name and a brief idea of what it does.>
+ Copyright (C) <year> <name of author>
+
+ 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.
+
+Also add information on how to contact you by electronic and paper mail.
+
+If the program is interactive, make it output a short notice like this
+when it starts in an interactive mode:
+
+ Gnomovision version 69, Copyright (C) year name of author
+ Gnomovision comes with ABSOLUTELY NO WARRANTY; for details type `show w'.
+ This is free software, and you are welcome to redistribute it
+ under certain conditions; type `show c' for details.
+
+The hypothetical commands `show w' and `show c' should show the appropriate
+parts of the General Public License. Of course, the commands you use may
+be called something other than `show w' and `show c'; they could even be
+mouse-clicks or menu items--whatever suits your program.
+
+You should also get your employer (if you work as a programmer) or your
+school, if any, to sign a "copyright disclaimer" for the program, if
+necessary. Here is a sample; alter the names:
+
+ Yoyodyne, Inc., hereby disclaims all copyright interest in the program
+ `Gnomovision' (which makes passes at compilers) written by James Hacker.
+
+ <signature of Ty Coon>, 1 April 1989
+ Ty Coon, President of Vice
+
+This General Public License does not permit incorporating your program into
+proprietary programs. If your program is a subroutine library, you may
+consider it more useful to permit linking proprietary applications with the
+library. If this is what you want to do, use the GNU Lesser General
+Public License instead of this License.
--- /dev/null
+# http://projects.camlcity.org/projects/dl/findlib-1.3.3/doc/guide-html/x412.html
+name = "todo"
+version = "@PACKAGE_VERSION@"
+description = "To do list management"
+
+requires = "unix"
--- /dev/null
+# todo
+# Copyright (C) 2016 Red Hat Inc.
+#
+# 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 common-rules.mk
+
+ACLOCAL_AMFLAGS = -I m4
+
+EXTRA_DIST = \
+ COPYING \
+ META.in \
+ README
+
+all_sources = $(TODO_SOURCES_ML)
+
+# You must adjust this to point to the right database host. The
+# database itself is hard-coded as 'todo'. Setting the environment
+# variable here merely tells PG'OCaml that.
+#export PGHOST=todo
+export PGDATABASE=todo
+
+bin_PROGRAMS = todo
+
+TODO_SOURCES_ML = \
+ todo_config.ml \
+ todo_types.ml \
+ todo_utils.ml \
+ todo_tag_utils.ml \
+ todo_add.ml \
+ todo_list.ml \
+ todo_move.ml \
+ todo_retire.ml \
+ todo_tag.ml \
+ todo_cmdline.ml \
+ todo.ml
+if !HAVE_OCAMLOPT
+TODO_OBJECTS = $(TODO_SOURCES_ML:.ml=.cmo)
+else
+TODO_OBJECTS = $(TODO_SOURCES_ML:.ml=.cmx)
+endif
+
+todo_SOURCES = dummy.c
+todo_DEPENDENCIES = $(TODO_OBJECTS) $(top_srcdir)/ocaml-link.sh
+todo_LINK = \
+ $(top_srcdir)/ocaml-link.sh -- \
+ $(OCAMLFIND) $(OCAMLBEST) $(OCAMLFLAGS) $(OCAMLPACKAGES) \
+ $(OCAMLLINKFLAGS) $(TODO_OBJECTS) -o $@
+
+BUILT_SOURCES = dummy.c
+dummy.c:
+ rm -f $@
+ touch $@
+
+# Dependencies.
+depend: .depend
+
+.depend: $(all_sources)
+ rm -f $@ $@-t
+ $(OCAMLFIND) ocamldep $(OCAMLPACKAGES) -I $(abs_srcdir) $^ | \
+ $(SED) 's/ *$$//' | \
+ $(SED) -e :a -e '/ *\\$$/N; s/ *\\\n */ /; ta' | \
+ $(SED) -e 's,$(abs_srcdir)/,$(builddir)/,g' | \
+ sort > $@-t
+ mv $@-t $@
+
+-include .depend
+
+# RPM package.
+rpm: dist
+ rpmbuild -ta $(PACKAGE_NAME)-$(PACKAGE_VERSION).tar.gz
+
+if HAVE_POD2MAN
+
+#man_MANS =
+
+#goaljobs.1: goaljobs.pod
+# $(POD2MAN) -c "Goaljobs" --release $(PACKAGE)-$(VERSION) $< > $@-t
+# mv $@-t $@
+
+endif
--- /dev/null
+This is the software I now use to manage my to-do list. It is
+probably not useful or appropriate for anyone else. If it's useful
+for you, that's great, but I don't want to be contacted about it.
+
+- Richard W.M. Jones, 2016-11-10
--- /dev/null
+# todo
+# Copyright (C) 2016 Red Hat Inc.
+#
+# 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.
+
+CLEANFILES = *~ *.cmi *.cmo *.cmx *.cma *.cmxa *.o
+
+OCAMLFLAGS = -g -warn-error CDEFLMPSUVYZX-3
+OCAMLPACKAGES = \
+ -package unix,calendar,pgocaml.syntax \
+ -syntax camlp4o \
+ -I $(top_builddir)
+
+%.cmi: %.mli
+ $(OCAMLFIND) ocamlc $(OCAMLFLAGS) $(OCAMLPACKAGES) -c $< -o $@
+%.cmo: %.ml
+ $(OCAMLFIND) ocamlc $(OCAMLFLAGS) $(OCAMLPACKAGES) -c $< -o $@
+%.cmx: %.ml
+ $(OCAMLFIND) ocamlopt $(OCAMLFLAGS) $(OCAMLPACKAGES) -c $< -o $@
+
+SUFFIXES = .cmo .cmi .cmx .ml .mli .mll .mly
--- /dev/null
+# todo
+# Copyright (C) 2016 Red Hat Inc.
+#
+# 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.
+
+AC_INIT([todo],[0.1])
+AM_INIT_AUTOMAKE([foreign])
+
+AC_SUBST([RPM_RELEASE],[1])
+
+AC_CONFIG_MACRO_DIR([m4])
+
+dnl Allow all GNU/Linux functions.
+dnl autoconf complains unless this is very early in the file.
+AC_USE_SYSTEM_EXTENSIONS
+
+AC_PROG_LIBTOOL
+
+dnl Check for basic C environment.
+AC_PROG_CC_STDC
+AC_PROG_INSTALL
+AC_PROG_CPP
+
+AC_C_PROTOTYPES
+test "x$U" != "x" && AC_MSG_ERROR([Compiler not ANSI compliant])
+
+AM_PROG_CC_C_O
+
+AC_ARG_ENABLE([gcc-warnings],
+ [AS_HELP_STRING([--enable-gcc-warnings],
+ [turn on lots of GCC warnings (for developers)])],
+ [case $enableval in
+ yes|no) ;;
+ *) AC_MSG_ERROR([bad value $enableval for gcc-warnings option]) ;;
+ esac
+ gcc_warnings=$enableval],
+ [gcc_warnings=no]
+)
+
+if test "$gcc_warnings" = yes; then
+ # XXX With gnulib we can improve this in future.
+ WARN_CFLAGS="-Wall"
+ AC_SUBST([WARN_CFLAGS])
+ WERROR_CFLAGS="-Werror"
+ AC_SUBST([WERROR_CFLAGS])
+fi
+
+dnl Check support for 64 bit file offsets.
+AC_SYS_LARGEFILE
+
+dnl OCaml compiler.
+AC_PROG_OCAML
+if test "$OCAMLC" = "no"; then
+ AC_MSG_ERROR([You must install the OCaml compiler])
+fi
+
+AM_CONDITIONAL([HAVE_OCAMLOPT], [test "x$OCAMLOPT" != "xno"])
+
+dnl OCaml findlib ("ocamlfind") is required.
+AC_PROG_FINDLIB
+if test "x$OCAMLFIND" = "xno"; then
+ AC_MSG_ERROR([You must install OCaml findlib (the ocamlfind command)])
+fi
+
+dnl Check for OCaml package Calendar (required).
+AC_CHECK_OCAML_PKG(calendar)
+if test "x$OCAML_PKG_calendar" = "xno"; then
+ AC_MSG_ERROR([You must install the 'ocaml-calendar' library])
+fi
+
+dnl Check for OCaml package PG'OCaml (required).
+AC_CHECK_OCAML_PKG(pgocaml)
+if test "x$OCAML_PKG_pgocaml" = "xno"; then
+ AC_MSG_ERROR([You must install the 'ocaml-pgocaml' library])
+fi
+
+dnl Check for POD (for manual pages).
+AC_CHECK_PROG(POD2MAN,pod2man,pod2man,no)
+AM_CONDITIONAL([HAVE_POD2MAN],
+ [test "x$POD2MAN" != "xno"])
+
+AC_CONFIG_HEADERS([config.h])
+AC_CONFIG_FILES([todo_config.ml
+ todo.spec
+ Makefile
+ META])
+AC_OUTPUT
--- /dev/null
+dnl autoconf macros for OCaml
+dnl
+dnl Copyright © 2009 Richard W.M. Jones
+dnl Copyright © 2009 Stefano Zacchiroli
+dnl Copyright © 2000-2005 Olivier Andrieu
+dnl Copyright © 2000-2005 Jean-Christophe Filliâtre
+dnl Copyright © 2000-2005 Georges Mariano
+dnl
+dnl For documentation, please read the ocaml.m4 man page.
+
+AC_DEFUN([AC_PROG_OCAML],
+[dnl
+ # checking for ocamlc
+ AC_CHECK_TOOL([OCAMLC],[ocamlc],[no])
+
+ if test "$OCAMLC" != "no"; then
+ OCAMLVERSION=`$OCAMLC -v | sed -n -e 's|.*version* *\(.*\)$|\1|p'`
+ AC_MSG_RESULT([OCaml version is $OCAMLVERSION])
+ OCAMLLIB=`$OCAMLC -where 2>/dev/null || $OCAMLC -v|tail -1|cut -d ' ' -f 4`
+ AC_MSG_RESULT([OCaml library path is $OCAMLLIB])
+
+ AC_SUBST([OCAMLVERSION])
+ AC_SUBST([OCAMLLIB])
+
+ # checking for ocamlopt
+ AC_CHECK_TOOL([OCAMLOPT],[ocamlopt],[no])
+ OCAMLBEST=byte
+ if test "$OCAMLOPT" = "no"; then
+ AC_MSG_WARN([Cannot find ocamlopt; bytecode compilation only.])
+ else
+ TMPVERSION=`$OCAMLOPT -v | sed -n -e 's|.*version* *\(.*\)$|\1|p' `
+ if test "$TMPVERSION" != "$OCAMLVERSION" ; then
+ AC_MSG_RESULT([versions differs from ocamlc; ocamlopt discarded.])
+ OCAMLOPT=no
+ else
+ OCAMLBEST=opt
+ fi
+ fi
+
+ AC_SUBST([OCAMLBEST])
+
+ # checking for ocamlc.opt
+ AC_CHECK_TOOL([OCAMLCDOTOPT],[ocamlc.opt],[no])
+ if test "$OCAMLCDOTOPT" != "no"; then
+ TMPVERSION=`$OCAMLCDOTOPT -v | sed -n -e 's|.*version* *\(.*\)$|\1|p' `
+ if test "$TMPVERSION" != "$OCAMLVERSION" ; then
+ AC_MSG_RESULT([versions differs from ocamlc; ocamlc.opt discarded.])
+ else
+ OCAMLC=$OCAMLCDOTOPT
+ fi
+ fi
+
+ # checking for ocamlopt.opt
+ if test "$OCAMLOPT" != "no" ; then
+ AC_CHECK_TOOL([OCAMLOPTDOTOPT],[ocamlopt.opt],[no])
+ if test "$OCAMLOPTDOTOPT" != "no"; then
+ TMPVERSION=`$OCAMLOPTDOTOPT -v | sed -n -e 's|.*version* *\(.*\)$|\1|p' `
+ if test "$TMPVERSION" != "$OCAMLVERSION" ; then
+ AC_MSG_RESULT([version differs from ocamlc; ocamlopt.opt discarded.])
+ else
+ OCAMLOPT=$OCAMLOPTDOTOPT
+ fi
+ fi
+ fi
+
+ AC_SUBST([OCAMLOPT])
+ fi
+
+ AC_SUBST([OCAMLC])
+
+ # checking for ocamldep
+ AC_CHECK_TOOL([OCAMLDEP],[ocamldep],[no])
+
+ # checking for ocamlmktop
+ AC_CHECK_TOOL([OCAMLMKTOP],[ocamlmktop],[no])
+
+ # checking for ocamlmklib
+ AC_CHECK_TOOL([OCAMLMKLIB],[ocamlmklib],[no])
+
+ # checking for ocamldoc
+ AC_CHECK_TOOL([OCAMLDOC],[ocamldoc],[no])
+
+ # checking for ocamlbuild
+ AC_CHECK_TOOL([OCAMLBUILD],[ocamlbuild],[no])
+])
+
+
+AC_DEFUN([AC_PROG_OCAMLLEX],
+[dnl
+ # checking for ocamllex
+ AC_CHECK_TOOL([OCAMLLEX],[ocamllex],[no])
+ if test "$OCAMLLEX" != "no"; then
+ AC_CHECK_TOOL([OCAMLLEXDOTOPT],[ocamllex.opt],[no])
+ if test "$OCAMLLEXDOTOPT" != "no"; then
+ OCAMLLEX=$OCAMLLEXDOTOPT
+ fi
+ fi
+ AC_SUBST([OCAMLLEX])
+])
+
+AC_DEFUN([AC_PROG_OCAMLYACC],
+[dnl
+ AC_CHECK_TOOL([OCAMLYACC],[ocamlyacc],[no])
+ AC_SUBST([OCAMLYACC])
+])
+
+
+AC_DEFUN([AC_PROG_CAMLP4],
+[dnl
+ AC_REQUIRE([AC_PROG_OCAML])dnl
+
+ # checking for camlp4
+ AC_CHECK_TOOL([CAMLP4],[camlp4],[no])
+ if test "$CAMLP4" != "no"; then
+ TMPVERSION=`$CAMLP4 -v 2>&1| sed -n -e 's|.*version *\(.*\)$|\1|p'`
+ if test "$TMPVERSION" != "$OCAMLVERSION" ; then
+ AC_MSG_RESULT([versions differs from ocamlc])
+ CAMLP4=no
+ fi
+ fi
+ AC_SUBST([CAMLP4])
+
+ # checking for companion tools
+ AC_CHECK_TOOL([CAMLP4BOOT],[camlp4boot],[no])
+ AC_CHECK_TOOL([CAMLP4O],[camlp4o],[no])
+ AC_CHECK_TOOL([CAMLP4OF],[camlp4of],[no])
+ AC_CHECK_TOOL([CAMLP4OOF],[camlp4oof],[no])
+ AC_CHECK_TOOL([CAMLP4ORF],[camlp4orf],[no])
+ AC_CHECK_TOOL([CAMLP4PROF],[camlp4prof],[no])
+ AC_CHECK_TOOL([CAMLP4R],[camlp4r],[no])
+ AC_CHECK_TOOL([CAMLP4RF],[camlp4rf],[no])
+ AC_SUBST([CAMLP4BOOT])
+ AC_SUBST([CAMLP4O])
+ AC_SUBST([CAMLP4OF])
+ AC_SUBST([CAMLP4OOF])
+ AC_SUBST([CAMLP4ORF])
+ AC_SUBST([CAMLP4PROF])
+ AC_SUBST([CAMLP4R])
+ AC_SUBST([CAMLP4RF])
+])
+
+
+AC_DEFUN([AC_PROG_FINDLIB],
+[dnl
+ AC_REQUIRE([AC_PROG_OCAML])dnl
+
+ # checking for ocamlfind
+ AC_CHECK_TOOL([OCAMLFIND],[ocamlfind],[no])
+ AC_SUBST([OCAMLFIND])
+])
+
+
+dnl Thanks to Jim Meyering for working this next bit out for us.
+dnl XXX We should define AS_TR_SH if it's not defined already
+dnl (eg. for old autoconf).
+AC_DEFUN([AC_CHECK_OCAML_PKG],
+[dnl
+ AC_REQUIRE([AC_PROG_FINDLIB])dnl
+
+ AC_MSG_CHECKING([for OCaml findlib package $1])
+
+ unset found
+ unset pkg
+ found=no
+ for pkg in $1 $2 ; do
+ if $OCAMLFIND query $pkg >/dev/null 2>/dev/null; then
+ AC_MSG_RESULT([found])
+ AS_TR_SH([OCAML_PKG_$1])=$pkg
+ found=yes
+ break
+ fi
+ done
+ if test "$found" = "no" ; then
+ AC_MSG_RESULT([not found])
+ AS_TR_SH([OCAML_PKG_$1])=no
+ fi
+
+ AC_SUBST(AS_TR_SH([OCAML_PKG_$1]))
+])
+
+
+AC_DEFUN([AC_CHECK_OCAML_MODULE],
+[dnl
+ AC_MSG_CHECKING([for OCaml module $2])
+
+ cat > conftest.ml <<EOF
+open $3
+EOF
+ unset found
+ for $1 in $$1 $4 ; do
+ if $OCAMLC -c -I "$$1" conftest.ml >&5 2>&5 ; then
+ found=yes
+ break
+ fi
+ done
+
+ if test "$found" ; then
+ AC_MSG_RESULT([$$1])
+ else
+ AC_MSG_RESULT([not found])
+ $1=no
+ fi
+ AC_SUBST([$1])
+])
+
+
+dnl XXX Cross-compiling
+AC_DEFUN([AC_CHECK_OCAML_WORD_SIZE],
+[dnl
+ AC_MSG_CHECKING([for OCaml compiler word size])
+ cat > conftest.ml <<EOF
+ print_endline (string_of_int Sys.word_size)
+ EOF
+ OCAML_WORD_SIZE=`ocaml conftest.ml`
+ AC_MSG_RESULT([$OCAML_WORD_SIZE])
+ AC_SUBST([OCAML_WORD_SIZE])
+])
--- /dev/null
+#!/bin/bash -
+# (C) Copyright 2015-2016 Red Hat Inc.
+#
+# 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., 675 Mass Ave, Cambridge, MA 02139, USA.
+
+# See guestfs-hacking(1) section "HOW OCAML PROGRAMS ARE COMPILED AND LINKED"
+
+# Hack automake to link OCaml-based binaries properly.
+# There is no other way to add the -cclib parameter to the end of
+# the command line.
+
+# Usage:
+# ./ocaml-link.sh -cclib '...' -- ARGS
+# Pass the cclib argument separately, and the rest as separated
+# arguments.
+
+TEMP=`getopt -a -o '' --long 'cclib:' \
+ -n "$(basename $0)" -- "$@"`
+if [ $? != 0 ] ; then echo "Terminating..." >&2 ; exit 1 ; fi
+eval set -- "$TEMP"
+
+cclib=
+
+while true ; do
+ case "$1" in
+ -cclib|--cclib) cclib="$2" ; shift 2 ;;
+ --) shift ; break ;;
+ *) echo "Internal error!" ; exit 1 ;;
+ esac
+done
+
+exec "$@" -linkpkg -cclib "${cclib}"
--- /dev/null
+-- PostgreSQL database schema.
+
+begin work;
+
+-- List of all tasks.
+--
+-- Note that unsorted tasks (which need to be moved to 'imminent',
+-- 'todo', 'ideas', 'retired' lists) are any item which appears in
+-- this table but not in one of the other tables.
+create table tasks (
+ id serial not null primary key,
+ description text not null,
+ created timestamp without time zone not null default now(),
+
+ -- task source
+ rhbz integer, -- if task comes from RHBZ
+ message_id text -- if task comes from email
+);
+
+-- Tasks which must be completed immediately and should take no more
+-- than a few minutes (else they should be moved to another list).
+create table imminent (
+ taskid integer not null references tasks (id),
+ added timestamp without time zone not null default now()
+);
+
+-- Regular to-do items with deadlines.
+create table todo (
+ taskid integer not null references tasks (id),
+ added timestamp without time zone not null default now(),
+ deadline timestamp without time zone not null,
+ estimate interval -- optional
+);
+
+-- Ideas. These have no committed deadline.
+create table ideas (
+ taskid integer not null references tasks (id),
+ added timestamp without time zone not null default now()
+);
+
+-- Retired/done items.
+create table retired (
+ taskid integer not null references tasks (id),
+ added timestamp without time zone not null default now()
+);
+
+-- Tags (eg 'urgent', 'personal', 'work') which can be added to any
+-- task.
+create table tags (
+ id serial not null primary key,
+ name text not null,
+ colour text not null references colours (colour)
+);
+
+create table tags_tasks (
+ tagid integer not null references tags (id),
+ taskid integer not null references tasks (id)
+);
+
+create table colours (
+ colour text not null,
+ unique (colour)
+);
+insert into colours (colour) values ('black');
+insert into colours (colour) values ('blue');
+insert into colours (colour) values ('green');
+insert into colours (colour) values ('red');
+insert into colours (colour) values ('purple');
+insert into colours (colour) values ('cyan');
+insert into colours (colour) values ('yellow');
+
+commit;
--- /dev/null
+(* 'todo' command, for adding, listing, deleting etc tasks from the
+ * command line.
+ *)
+
+open CalendarLib
+
+open Todo_types
+open Todo_utils
+
+open Printf
+
+(* Parse the command line. *)
+let subcommand, anon_params, list_retired, list_all, tag_del, todo_estimate =
+ Todo_cmdline.parse_command_line ()
+
+(* Connect to the database. *)
+let dbh = PGOCaml.connect ~database:"todo" ()
+
+(* Run the subcommand. *)
+let () =
+ match subcommand with
+ | Idea -> (* todo idea *)
+ Todo_add.cmd_idea dbh anon_params
+ | List -> (* todo list *)
+ Todo_list.cmd_list dbh anon_params list_retired list_all
+ | Move -> (* todo move *)
+ Todo_move.cmd_move dbh anon_params
+ | Retire -> (* todo retire *)
+ Todo_retire.cmd_retire dbh anon_params
+ | Tag -> (* todo tag *)
+ Todo_tag.cmd_tag dbh anon_params tag_del
+ | Tag_add -> (* todo tag-add *)
+ Todo_tag.cmd_tag_add dbh anon_params
+ | Tag_colour -> (* todo tag-colour *)
+ Todo_tag.cmd_tag_colour dbh anon_params
+ | Tag_del -> (* todo tag-del *)
+ Todo_tag.cmd_tag_del dbh anon_params
+ | Tag_list -> (* todo tag-list *)
+ Todo_tag.cmd_tag_list dbh anon_params
+ | Today -> (* todo today *)
+ Todo_add.cmd_today dbh anon_params
+ | Todo -> (* todo todo|task *)
+ Todo_add.cmd_todo dbh anon_params todo_estimate
--- /dev/null
+%global opt %(test -x %{_bindir}/ocamlopt && echo 1 || echo 0)
+
+Name: @PACKAGE_NAME@
+Version: @PACKAGE_VERSION@
+Release: @RPM_RELEASE@%{?dist}
+Summary: To do list management
+License: GPLv2+
+
+#URL:
+Source0: %{name}-%{version}.tar.gz
+
+BuildRequires: ocaml >= 3.12.0
+BuildRequires: ocaml-findlib-devel
+BuildRequires: ocaml-sqlite-devel
+
+# For building manual pages.
+BuildRequires: perl-podlators
+
+
+%description
+
+
+%prep
+%setup -q
+
+
+%build
+%configure
+make %{?_smp_mflags}
+
+
+%install
+make DESTDIR=$RPM_BUILD_ROOT install
+
+%if %opt
+# Remove bytecode library.
+rm $RPM_BUILD_ROOT%{_libdir}/ocaml/todo/todo.cma
+%endif
+
+
+%files
+%doc COPYING README
+
+
+%changelog
+* Thu Nov 10 2016 Richard W.M. Jones <rjones@redhat.com> - @PACKAGE_VERSION@-@RPM_RELEASE@
+- Initial release.
+
+# Local variables:
+# mode: shell-script
+# sh-shell: rpm
+# end:
--- /dev/null
+(* Implement subcommands for adding new tasks. *)
+
+open CalendarLib
+
+open Todo_types
+open Todo_utils
+
+open Printf
+
+let parse_tags tags = nsplit "," tags
+
+let insert_tags dbh taskid tags =
+ List.iter (
+ fun tag ->
+ let rows = PGSQL(dbh) "select id from tags where name = $tag" in
+ let id =
+ match rows with
+ | [] -> error "unknown tag: %s" tag
+ | [id] -> id
+ | _ -> assert false in
+ PGSQL(dbh) "insert into tags_tasks (tagid, taskid)
+ values ($id, $taskid)"
+ ) tags
+
+let cmd_todo dbh anon_params estimate =
+ let deadline, description, tags =
+ match anon_params with
+ | [deadline; description] ->
+ Printer.Date.from_string deadline, description, []
+ | [deadline; description; tags] ->
+ Printer.Date.from_string deadline, description, parse_tags tags
+ | _ -> error "incorrect number of parameters to 'todo' subcommand" in
+
+ PGOCaml.begin_work dbh;
+
+ PGSQL(dbh) "insert into tasks (description) values ($description)";
+ let taskid = PGOCaml.serial4 dbh "tasks_id_seq" in
+ PGSQL(dbh) "insert into todo (taskid, deadline, estimate)
+ values ($taskid, $deadline :: date, $?estimate)";
+ insert_tags dbh taskid tags;
+
+ PGOCaml.commit dbh
+
+let cmd_today dbh anon_params =
+ let description, tags =
+ match anon_params with
+ | [description] -> description, []
+ | [description; tags] ->
+ description, parse_tags tags
+ | _ -> error "incorrect number of parameters to 'today' subcommand" in
+
+ PGOCaml.begin_work dbh;
+
+ PGSQL(dbh) "insert into tasks (description) values ($description)";
+ let taskid = PGOCaml.serial4 dbh "tasks_id_seq" in
+ PGSQL(dbh) "insert into imminent (taskid) values ($taskid)";
+ insert_tags dbh taskid tags;
+
+ PGOCaml.commit dbh
+
+let cmd_idea dbh anon_params =
+ let description, tags =
+ match anon_params with
+ | [description] -> description, []
+ | [description; tags] ->
+ description, parse_tags tags
+ | _ -> error "incorrect number of parameters to 'idea' subcommand" in
+
+ PGOCaml.begin_work dbh;
+
+ PGSQL(dbh) "insert into tasks (description) values ($description)";
+ let taskid = PGOCaml.serial4 dbh "tasks_id_seq" in
+ PGSQL(dbh) "insert into ideas (taskid) values ($taskid)";
+ insert_tags dbh taskid tags;
+
+ PGOCaml.commit dbh
--- /dev/null
+(* Parse the command line. *)
+
+open CalendarLib
+
+open Todo_types
+open Todo_utils
+
+open Printf
+
+let usage = "\
+todo -- A simple to-do list
+
+ todo [--global-options] [subcommand] [--subcommand-options]
+
+Subcommands:
+ todo list [--all] [--retired]
+ - List all tasks
+ todo today \"description\" [tag[,tag...]]
+ - Add a simple task for today, with an optional list of tags
+ todo idea \"description\" [tag[,tag...]]
+ - Add an long-term idea
+ todo [todo|task] deadline [--estimate nr_days] \"description\" [tag[,tag...]]
+ - Add a to-do task with a deadline and optional time estimate
+ todo retire ID [ID ...]
+ - Retire one or more tasks using the task #ID
+ todo move ID [today | [todo|task] deadline | ideas | retired | unsorted]
+ - Move a task to another list
+ todo tag ID tag [--del tag]
+ - Add or remove tags from a task
+ todo tag-list
+ - List all tags and their usage
+ todo tag-add tag colour
+ - Create a new tag, with a colour
+ todo tag-del tag
+ - Delete a tag (must be unused)
+ todo tag-colour tag colour
+ - Change the colour of a tag
+
+Tag colours may be: black blue green cyan red purple yellow
+
+Examples:
+ todo today \"Eat breakfast\" personal,food
+ todo task 2019-31-12 \"Prepare for next decade\"
+ todo idea \"Write a to-do manager\" work
+ todo list
+
+Options:"
+
+(* Parse the command line. *)
+let parse_command_line () =
+ let print_version_and_exit () =
+ printf "%s %s\n" Todo_config.package_name Todo_config.package_version;
+ exit 0
+ in
+
+ let list_all = ref false in
+ let list_retired = ref false in
+ let tag_del = ref [] in
+ let add_tag_del tag = tag_del := tag :: !tag_del in
+ let todo_estimate = ref None in
+ let set_todo_estimate days =
+ todo_estimate := Some (Calendar.Period.day days)
+ in
+
+ (* Per-subcommand specs. *)
+ let idea_spec =
+ Arg.align [
+ ] in
+ let list_spec =
+ Arg.align [
+ "--all", Arg.Set list_all, " List all tasks including retired";
+ "--retired", Arg.Set list_retired, " List only retired tasks";
+ ] in
+ let move_spec =
+ Arg.align [
+ ] in
+ let retire_spec =
+ Arg.align [
+ ] in
+ let tag_spec =
+ Arg.align [
+ "--del", Arg.String add_tag_del, "tag Remove tag";
+ "--delete", Arg.String add_tag_del, "tag Remove tag";
+ "--remove", Arg.String add_tag_del, "tag Remove tag";
+ "--rm", Arg.String add_tag_del, "tag Remove tag";
+ ] in
+ let tag_add_spec =
+ Arg.align [
+ ] in
+ let tag_colour_spec =
+ Arg.align [
+ ] in
+ let tag_del_spec =
+ Arg.align [
+ ] in
+ let tag_list_spec =
+ Arg.align [
+ ] in
+ let today_spec =
+ Arg.align [
+ ] in
+ let todo_spec =
+ Arg.align [
+ "--estimate", Arg.Int set_todo_estimate, "days Estimated time taken";
+ ] in
+
+ (* This parses the global parameters before the subcommand. *)
+ let global_spec =
+ Arg.align [
+ "--version", Arg.Unit print_version_and_exit, " Display version and exit";
+ ] in
+ let spec = ref global_spec in
+ let subcommand = ref None in
+ let anon_params = ref [] in
+ let anon s =
+ if !subcommand = None then (
+ (* It's the subcommand, so switch the spec. *)
+ let sc =
+ try subcommand_of_string s
+ with Invalid_argument _ ->
+ raise (Arg.Bad "unknown subcommand") in
+ subcommand := Some sc;
+ match sc with
+ | Idea -> spec := idea_spec
+ | List -> spec := list_spec
+ | Move -> spec := move_spec
+ | Retire -> spec := retire_spec
+ | Tag -> spec := tag_spec
+ | Tag_add -> spec := tag_add_spec
+ | Tag_colour -> spec := tag_colour_spec
+ | Tag_del -> spec := tag_del_spec
+ | Tag_list -> spec := tag_list_spec
+ | Today -> spec := today_spec
+ | Todo -> spec := todo_spec
+ )
+ else
+ (* We've seen the subcommand already, just add it to anon_params. *)
+ anon_params := s :: !anon_params
+ in
+ Arg.parse_dynamic spec anon usage;
+
+ (* Return the parsed command line. *)
+ let subcommand =
+ match !subcommand with
+ | None -> error "no subcommand given"
+ | Some c -> c in
+ let anon_params = List.rev !anon_params in
+
+ let list_retired = !list_retired in
+ let list_all = !list_all in
+ let tag_del = List.rev !tag_del in
+ let todo_estimate = !todo_estimate in
+
+ subcommand, anon_params, list_retired, list_all, tag_del, todo_estimate
--- /dev/null
+(* todo
+ * Copyright (C) 2016 Red Hat Inc.
+ * @configure_input@
+ *
+ * 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.
+ *)
+
+let package_name = "@PACKAGE_NAME@"
+let package_version = "@PACKAGE_VERSION@"
--- /dev/null
+(* Implement 'list' subcommand. *)
+
+open CalendarLib
+
+open Todo_types
+open Todo_utils
+open Todo_tag_utils
+
+open Printf
+
+let tags_of_task dbh id =
+ let rows = PGSQL(dbh) "select tags.name, tags.colour
+ from tags_tasks, tags
+ where tags_tasks.taskid = $id
+ and tags_tasks.tagid = tags.id
+ order by tags.name" in
+ if rows = [] then ""
+ else
+ " " ^
+ String.concat " "
+ (List.map (
+ fun (name, colour) ->
+ string_of_tag name colour
+ ) rows)
+
+let cmd_list dbh anon_params list_retired list_all =
+ if anon_params <> [] then
+ error "extra parameters to 'list' subcommand";
+
+ let show_unretired, show_retired =
+ match list_retired, list_all with
+ | _, true -> true, true
+ | false, false -> true, false
+ | true, false -> false, true in
+
+ if show_unretired then (
+ let rows = PGSQL(dbh) "select imminent.taskid, tasks.description
+ from imminent, tasks
+ where imminent.taskid = tasks.id
+ order by tasks.description" in
+ if rows <> [] then printf "Today:\n";
+ List.iter (
+ fun (id, desc) ->
+ printf " %s #%ld%s\n" desc id (tags_of_task dbh id)
+ ) rows;
+
+ let rows = PGSQL(dbh) "select todo.taskid, todo.deadline, todo.estimate,
+ tasks.description
+ from todo, tasks
+ where todo.taskid = tasks.id
+ order by todo.deadline" in
+ if rows <> [] then printf "To-do list:\n";
+ List.iter (
+ fun (id, deadline, estimate, desc) ->
+ let deadline =
+ if Time.equal (Time.midnight ()) (Calendar.to_time deadline) then
+ Printer.Calendar.sprint "%F" deadline
+ else
+ Printer.Calendar.to_string deadline in
+ let estimate =
+ match estimate with
+ | None -> ""
+ | Some period -> string_of_estimate period ^ " " in
+ printf " %s %s%s #%ld%s\n"
+ deadline estimate desc id (tags_of_task dbh id)
+ ) rows;
+
+ let rows = PGSQL(dbh) "
+ select tasks.id, tasks.description
+ from tasks
+ where not exists (select 1 from imminent where taskid = tasks.id)
+ and not exists (select 1 from todo where taskid = tasks.id)
+ and not exists (select 1 from ideas where taskid = tasks.id)
+ and not exists (select 1 from retired where taskid = tasks.id)
+ order by tasks.description" in
+ if rows <> [] then printf "Unsorted (use 'todo move' to move these):\n";
+ List.iter (
+ fun (id, desc) ->
+ printf " %s #%ld%s\n" desc id (tags_of_task dbh id)
+ ) rows;
+
+ let rows = PGSQL(dbh) "select ideas.taskid, tasks.description
+ from ideas, tasks
+ where ideas.taskid = tasks.id
+ order by tasks.description" in
+ if rows <> [] then printf "Ideas:\n";
+ List.iter (
+ fun (id, desc) ->
+ printf " %s #%ld%s\n" desc id (tags_of_task dbh id)
+ ) rows
+ ); (* unretired *)
+
+ if show_retired then (
+ let rows = PGSQL(dbh) "select retired.taskid, tasks.description
+ from retired, tasks
+ where retired.taskid = tasks.id
+ order by tasks.description" in
+ if rows <> [] then printf "Retired:\n";
+ List.iter (
+ fun (id, desc) ->
+ printf " %s #%ld%s\n" desc id (tags_of_task dbh id)
+ ) rows
+ )
--- /dev/null
+(* Implement 'move' subcommand. *)
+
+open CalendarLib
+
+open Todo_types
+open Todo_utils
+
+open Printf
+
+(* The form of the command is one of:
+ * move ID today
+ * move ID [todo|task] deadline
+ * move ID ideas
+ * move ID retired
+ * move ID unsorted
+ *)
+let cmd_move dbh anon_params =
+ let id, anon_params =
+ match anon_params with
+ | id :: rest -> Int32.of_string id, rest
+ | _ -> error "incorrect parameters to 'move' subcommand" in
+ let dest =
+ match anon_params with
+ | ["today"|"imminent"] -> `Today
+ | ["todo"|"task"; deadline] ->
+ `Todo (Printer.Calendar.from_string deadline)
+ | ["idea"|"ideas"] -> `Ideas
+ | ["retire"|"retired"] -> `Retired
+ | ["none"|"unsorted"] -> `Unsorted
+ | _ -> error "incorrect destination for 'move' subcommand" in
+
+ PGOCaml.begin_work dbh;
+
+ (* Remove it from any other table. *)
+ PGSQL(dbh) "delete from imminent where taskid = $id";
+ PGSQL(dbh) "delete from todo where taskid = $id";
+ PGSQL(dbh) "delete from ideas where taskid = $id";
+ PGSQL(dbh) "delete from retired where taskid = $id";
+
+ (* Add it to the destination table. *)
+ (match dest with
+ | `Today ->
+ PGSQL(dbh) "insert into imminent (taskid) values ($id)"
+ | `Todo deadline ->
+ PGSQL(dbh) "insert into todo (taskid, deadline) values ($id, $deadline)"
+ | `Ideas ->
+ PGSQL(dbh) "insert into ideas (taskid) values ($id)"
+ | `Retired ->
+ PGSQL(dbh) "insert into retired (taskid) values ($id)"
+ | `Unsorted ->
+ () (* Nothing - unsorted items are not present in any table. *)
+ );
+
+ PGOCaml.commit dbh
--- /dev/null
+(* Implement 'retire' subcommand. *)
+
+open CalendarLib
+
+open Todo_types
+open Todo_utils
+
+open Printf
+
+let cmd_retire dbh anon_params =
+ let ids =
+ match anon_params with
+ | [] -> error "give a list of IDs (#ID) to retire"
+ | ids -> List.map Int32.of_string ids in
+
+ PGOCaml.begin_work dbh;
+
+ (* It's an error to retire a task which is already retired. *)
+ List.iter (
+ fun taskid ->
+ let rows = PGSQL(dbh) "select 1 from retired where taskid = $taskid" in
+ if rows = [Some 1_l] then
+ error "task %ld is already retired" taskid;
+ ) ids;
+
+ (* Drop the tasks from every other table. *)
+ PGSQL(dbh) "delete from imminent where taskid in $@ids";
+ PGSQL(dbh) "delete from todo where taskid in $@ids";
+ PGSQL(dbh) "delete from ideas where taskid in $@ids";
+
+ (* Add them to the retired table. *)
+ List.iter (
+ fun taskid ->
+ PGSQL(dbh) "insert into retired (taskid) values ($taskid)"
+ ) ids;
+
+ PGOCaml.commit dbh
--- /dev/null
+(* Implement subcommands for listing and adding tags. *)
+
+open CalendarLib
+
+open Todo_types
+open Todo_utils
+open Todo_tag_utils
+
+open Printf
+
+let cmd_tag_list dbh anon_params =
+ if anon_params <> [] then
+ error "extra parameters to 'tag-list' subcommand";
+
+ (* Used tags. *)
+ let rows = PGSQL(dbh) "
+ select tags.name, tags.colour, tasks.id
+ from tags_tasks, tags, tasks
+ where tags_tasks.tagid = tags.id
+ and tags_tasks.taskid = tasks.id
+ order by tags.name, tasks.id" in
+ if rows <> [] then printf "Tags used:\n";
+ let prev = ref None in
+ List.iter (
+ fun (name, colour, taskid) ->
+ let cur = name, colour in
+ (match !prev with
+ | None -> printf " %s used by" (string_of_tag name colour)
+ | Some prev when prev <> cur ->
+ printf "\n %s used by" (string_of_tag name colour)
+ | Some _ -> ()
+ );
+ prev := Some cur;
+ printf " #%ld" taskid
+ ) rows;
+ if !prev <> None then printf "\n";
+
+ (* Unused tags. *)
+ let rows = PGSQL(dbh) "
+ select tags.name, tags.colour
+ from tags
+ where not exists (select 1 from tags_tasks where tagid = tags.id)
+ order by tags.name" in
+ if rows <> [] then printf "Tags not used (delete with 'todo tag-del'):\n";
+ List.iter (
+ fun (name, colour) ->
+ printf " %s\n" (string_of_tag name colour)
+ ) rows
+
+let cmd_tag dbh anon_params tag_del =
+ let id, tag_add =
+ match anon_params with
+ | id :: rest -> Int32.of_string id, rest
+ | _ -> error "incorrect parameters to 'tag' subcommand" in
+ if anon_params = [] && tag_del = [] then
+ error "no tags to add or delete from this task";
+
+ (* Map the tag_add and tag_del names to tag IDs. *)
+ let tag_add =
+ if tag_add <> [] then
+ PGSQL(dbh) "select id from tags where name in $@tag_add"
+ else [] in
+ let tag_del =
+ if tag_del <> [] then
+ PGSQL(dbh) "select id from tags where name in $@tag_del"
+ else [] in
+
+ PGOCaml.begin_work dbh;
+
+ if tag_del <> [] then
+ PGSQL(dbh) "delete from tags_tasks
+ where taskid = $id and tagid in $@tag_del";
+
+ List.iter (
+ fun tag ->
+ PGSQL(dbh) "insert into tags_tasks (taskid, tagid)
+ values ($id, $tag)"
+ ) tag_add;
+
+ PGOCaml.commit dbh
+
+let cmd_tag_add dbh anon_params =
+ let name, colour =
+ match anon_params with
+ | [name; colour] -> name, colour
+ | _ -> error "incorrect parameters to 'tag-add' subcommand" in
+
+ PGOCaml.begin_work dbh;
+
+ PGSQL(dbh) "insert into tags (name, colour) values ($name, $colour)";
+
+ PGOCaml.commit dbh
+
+let cmd_tag_del dbh anon_params =
+ let name =
+ match anon_params with
+ | [name] -> name
+ | _ -> error "incorrect parameters to 'tag-del' subcommand" in
+
+ PGOCaml.begin_work dbh;
+
+ PGSQL(dbh) "delete from tags where name = $name";
+
+ PGOCaml.commit dbh
+
+let cmd_tag_colour dbh anon_params =
+ let name, colour =
+ match anon_params with
+ | [name; colour] -> name, colour
+ | _ -> error "incorrect parameters to 'tag-colour' subcommand" in
+
+ PGOCaml.begin_work dbh;
+
+ PGSQL(dbh) "update tags set colour = $colour where name = $name";
+
+ PGOCaml.commit dbh
--- /dev/null
+(* Utility functions related to tags. *)
+
+open Printf
+
+let string_of_tag name colour =
+ let bg, fg =
+ match colour with
+ (* white fg *)
+ | "black" -> 40, 37
+ | "blue" -> 44, 37
+ | "green" -> 42, 37
+ | "red" -> 41, 37
+ | "purple" -> 45, 37
+
+ (* black fg *)
+ | "cyan" -> 46, 30
+ | "yellow" -> 43, 30
+ | _ -> assert false in
+
+ sprintf "\x1b[%d;%dm\x1b[%d;%dm %s \x1b[0m" 1 fg 1 bg name
--- /dev/null
+type subcommand =
+ | Idea
+ | List
+ | Move
+ | Retire
+ | Tag
+ | Tag_add
+ | Tag_colour
+ | Tag_del
+ | Tag_list
+ | Today
+ | Todo
+
+let subcommand_of_string = function
+ | "idea" -> Idea
+ | "list" -> List
+ | "move" -> Move
+ | "retire" -> Retire
+ | "tag" -> Tag
+ | "tag-add" -> Tag_add
+ | "tag-color" | "tag-colour" -> Tag_colour
+ | "tag-del" | "tag-delete" | "tag-rm" -> Tag_del
+ | "tag-list" -> Tag_list
+ | "task" | "todo" -> Todo
+ | "today" -> Today
+ | _ -> invalid_arg "subcommand_of_string"
--- /dev/null
+(* Utility functions. *)
+
+open CalendarLib
+
+open Printf
+
+let rec find s sub =
+ let len = String.length s in
+ let sublen = String.length sub in
+ let rec loop i =
+ if i <= len-sublen then (
+ let rec loop2 j =
+ if j < sublen then (
+ if s.[i+j] = sub.[j] then loop2 (j+1)
+ else -1
+ ) else
+ i (* found *)
+ in
+ let r = loop2 0 in
+ if r = -1 then loop (i+1) else r
+ ) else
+ -1 (* not found *)
+ in
+ loop 0
+
+let rec nsplit sep str =
+ let len = String.length str in
+ let seplen = String.length sep in
+ let i = find str sep in
+ if i = -1 then [str]
+ else (
+ let s' = String.sub str 0 i in
+ let s'' = String.sub str (i+seplen) (len-i-seplen) in
+ s' :: nsplit sep s''
+ )
+
+let error fs =
+ let display str =
+ prerr_string "todo: error: ";
+ prerr_endline str;
+ exit 1
+ in
+ ksprintf display fs
+
+let string_of_estimate period =
+ (* We ignore the seconds, and assume every estimate is >= 1 day. *)
+ let (years, months, days, _) = Calendar.Period.ymds period in
+ let buf = Buffer.create 13 in
+ let append str =
+ if Buffer.length buf > 0 then
+ Buffer.add_string buf ", ";
+ Buffer.add_string buf str
+ in
+ if years >= 2 then append (sprintf "%d years" years)
+ else if years = 1 then append "1 year";
+ if months >= 2 then append (sprintf "%d months" months)
+ else if months = 1 then append "1 month";
+ if days >= 2 then append (sprintf "%d days" days)
+ else if days = 1 then append "1 day";
+ let str = Buffer.contents buf in
+ if str <> "" then str else "-"