Skip to content

Latest commit

 

History

History
255 lines (211 loc) · 11.5 KB

README.md

File metadata and controls

255 lines (211 loc) · 11.5 KB

License Build Status Total alerts Codecov

What is this?

Gaplan is a simple but functional toolset for constructing and analyzing Gaperton's (aka Vlad Balin's) declarative plans.

It supports plan analysis (mainly checking for common mistakes) and generation of various artifacts (e.g PERT diagrams, TaskJuggler plans and burndown charts).

This is very much a work-in-progress, driven by my own planning needs. I'm happy to improve the tool so ping me if you consider using it.

About declarative planning

Declarative planning is a goal-based approach to building plans, backed up with an easy-to-use notation for writing them down.

Declarative plannig can be roughly described as modelling projects via PERT-like activity-on-edges diagrams with focus on well-defined events (called goals) rather than activities. Goals are defined through their accomplishment criteria. Declarative plans are often called goal maps (probably to pun on Alistair Cockburn's project maps) or project networks.

Declarative plans are

  • easy to construct and verify (compared to traditional activity-based plans)
  • scalable (can be written or collapsed to desired level of detail)
  • stable against (inevitable) business and technical risks (e.g. changes in requirements or architecture)

For more information on declarative planning see Vlad's presentation on SoftwarePeople 2009 or his numerous blogposts (sadly all in Russian):

Install

gaplan is based on setuptools so to install simply run

$ pip3 install .

in gaplan's folder.

For tooltips in TaskJuggler download wz_tooltip.js to scripts/ subfolder (note that it's distributed under LGPL).

How to use

To use the toolset you first need to write a declarative plan. Take a look at example plans in exampes/ folder (e.g. toolchain plan).

Now to build a PERT chart, do

# Requires Graphviz dot.exe in PATH
$ python3 -mgaplan pert plan.txt

To generate TaskJuggler project:

# Requires TaskJuggler in PATH
$ python3 -mgaplan tj plan.txt

To compute project schedule (if defined for the project):

$ python3 -mgaplan schedule plan.txt

To generate a burndown chart:

$ python3 -mgaplan burn --phase 'Iteration 1 completed' burndown plan.txt

To display generated files (instead of passing it to Graphviz or TaskJuggler) use --dump flag e.g.

$ python3 -mgaplan --dump pert plan.txt > plan.gv
$ dot -Tpdf < plan.gv > plan.pdf

or

$ python3 -mgaplan --dump tj plan.txt > plan.tjp
$ mkdir -p tjdir
$ tj3 plan.tjp -o tjdir

All commands support -W (emit warnings for common errors) and -v (add diagnostic prints) switches.

For additional details run

$ python3 -mgaplan --help

Notation

The core of the syntax is Vlad's text notation for graphs:

|Ready for NY celebration
|<-
   |Bought food
   |[X] Bread
   |[] Cheese
   |<-
      |Bought alcohol  // !3
      |[] Wine
      |[] Alcohol
|<-  // @me, 1h-2h
   |Invited friends  // !3
   |[] Alex
   |[] Max

Example above illustrates few core concepts:

  • only goals have names (Invited friends), priorities (!3) and internal structure (checklists)
  • only activities (arrows) have duration (1h-2h) and assignees (@me) See more examples in examples/ subfoler.

Canonical notation has been extended with additional features which turned out to be useful in practice:

  • Tool tries to infer hierarchical connections between tasks (i.e. WBS) as they allow to produce more readable plans for TaskJuggler.
# A hierarchy of 3 nested goals
|Quality requirements fullfilled
|<-
   |Results of regression testing acceptable for beta
   |[] Regression testing logs ready
   |[] Number of regressions against prev. version <= 1%
   |[] All regressions against prev. version fixed or explained
   |<-
      |Results of regression testing acceptable for alpha
      |[] Regression testing logs ready
      |[] Number of regressions against prev. version <= 5%
  • Goals can be annotated with various attributes (priority, risk, assignees, etc.) e.g.
# This goal has to be reached by the end of November, has max risk and priority and has to be scheduled in first iteration
|Symbol visibility in TZ 3.0 reduced  // deadline 2016-11-30, !3, ?3, I0
  • In addition to normal dependencies (|<-, |->) tools supports global dependencies (marked with global). Globality causes all hierarchical children of a goal to depend on RHS. It's useful for splitting plan into disjoint phases, where task in depending phase can not start until their global dependency completes. This is an experimental feature.
  • Some goals many be unnamed (the so called "dummy PERT goals"):
|Feature X added
|<-  // 2d-1w
   |<-
      |Prerequisite 1
   |<-
      |Prerequisite 2

Note that activity depends on unnamed implicit goal (which itself depends on Prerequisites 1 and 2).

  • Plan can specify a custom time-boxed schedule (see SCHEDULE.md for details).
  • Special syntax is used for describing project attributes e.g. duration or resources (see PROJECT.md for details).

Attributes

Goals can be annotated with special attributes listed in table below.

Attribute Syntax Example Comments
Priority !PRIO !3 1, 2 or 3
Risk ?RISK ?3 1, 2 or 3
Deadline deadline YYYY-MM-DD deadline 2016-09-30
Accomplishment (or planned) date YYYY-MM-DD 2016-08-31
Identifier id symbolic_name id mod2_test Short name
External issue task name task PRJ-123

Priorities use a 3-level scale:

Prio Description
1 Default
2 Desirable but not required for project completion
3 Project requirement (probably stated in project proposal)

Same applies to risks:

Risk Description
1 Know-how, have done similar stuff before
2 No prior experience but known to be doable (e.g. already done by someone else)
3 No prior experience, not clear whether doable at all

Priorities propagate to predecessors (i.e. if some goal has high priority, it's predecessors will inherit it).

For example this statement

|Compiler rebuilds full distro  // deadline 2016-09-01, !3, ?1
...

says that the goal "Compiler rebuilds full distro" must be completed by September (deadline attribute), is high-prio (!3) and low-risk (?1).

There is a distinct set of attributes for activities:

Attribute Syntax Example Comment
Effort estimate min-max 1h-3d, 1w-1m Activity effort estimate (in "ideal hours")
Actual effort min-max (real) 1h-3d (1d), 1w-1m (2w) Actual observed effort (used for tracking)
Completion min-max (X%) 1h-3d (50%) Percentage of completion
Actual duration YYYY-MM-DD-YYYY-MM-DD 2016-05-31-2016-06-02 Actual observed duration (used for tracking)
Assignees @dev1/dev2/... @yura/slava Developers assigned to the task
Actual assignees @dev1/dev2/... (real) @yura/slava (max) Developers who actually accomplished the task
Parallel impl. || || Notes that developers can work on task in parallel
Identifier id symbolic_name id enable-jenkins-job Gives symbolic name to activity
Fast tracking over id X% over enable-jenkins-job 15% How much activity can be overlapped with it's predecessor(s) (multiple overs can be specified)

The exact meaning of resource assignment attribute (@) depends on presense of "parallel" attribute (denoted with ||):

  • with || (or || NUMBER) - developers can work on task in parallel (e.g. it consists of many similar unrelated chunks)
  • without - only one of specified engineers will be selected to work on a task

For example, this activity

|<-  // @yura/slava/max, 2h-1d

can be done either by "yura", "slava" or by "max" (but not both simultaneously) and may require 2 hours to 2 days of effort. Adding || (or || 3) would mean that all three developers will be able to work on parallel. Adding || 2 would mean that any two of them will be able to work in parallel.

Note that you can not

  • assign names to activities (this forces you to focus on goals rather than tasks)
  • assign resources or efforts to goals (these are instantaneous events so they do not take any time to "accomplish")

Development

Install in editable mode via

$ pip3 install -e .

To test, install pytest-3 and run

$ pytest-3 gaplan

TODO