aboutsummaryrefslogtreecommitdiff
path: root/hatask.ha
blob: 291f17e99b6526deb2f67a609491d5378a8bb4bf (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
use fmt;
use dirs;
use bufio;
use memio;
use os;
use fs;
use path;
use io;
use strings;
use format::ini;
use strconv;
use encoding::utf8;

type task = struct {
	name: str,
	context: (str | void),
	tags: ([]str | void),
	priority: uint,
};

def METADATASEP: str = "----";

type rtaskerror = !(fs::error | ini::error | strconv::error | io::error |
	utf8::invalid);

fn listtasks(root: str = "tasks", context: str = "*") ([]task | rtaskerror |
	path::error) = {
	const dirents = os::readdir(root)?;
	defer fs::dirents_free(dirents);
	let tasks: []task = [];
	for (const dirent .. dirents) {
		if (context == "*" || dirent.name == context) {
			if (fs::isdir(dirent.ftype)) {
				let buf = path::init()?;
				const p = path::push(&buf, root, dirent.name)?;
				listtasks(p)?;
			} else {
				fmt::println(dirent.name)!;
				let buf = path::init()?;
				const p = path::push(&buf, root, dirent.name)?;
				let t = readtask(p)?;
				defer freetask(&t);
				fmt::println(t.name)!;
			};
		};
	};
	return tasks;
};

fn readtask(taskpath: str) (task | rtaskerror) = {
	const f = os::open(taskpath)?;
	defer io::close(f)!;
	const sc = bufio::newscanner(f);
	defer bufio::finish(&sc);
	const meta = memio::dynamic();
	defer io::close(&meta)!;
	for (let line => bufio::scan_line(&sc)?) {
		line = strings::trim(line);
		if (line == METADATASEP) {
			break;
		};
		memio::concat(&meta, line, "\n")?;
		//append(inilines, strings::dup(line));
	};
	// seek to start of memio buffer
	io::seek(&meta, 0, io::whence::SET)?;
	const sc = ini::scan(&meta);
	defer ini::finish(&sc);
	let t = task {
		context = void,
		tags = void,
		...
	};
	for (let entry: ini::entry => ini::next(&sc)?) {
		switch (entry.1) {
		case "name" =>
			t.name = strings::dup(entry.2);
		case "priority" =>
			t.priority = strconv::stou(entry.2)?;
		case =>
			void;
		};
	};
	return t;
};

fn freetask(t: *task) void = {
	free(t.name);
	if (t.context is str) {
		free(t.context as str);
	};
	if (t.tags is []str) {
		strings::freeall(t.tags as []str);
	};
};

export fn main() void = {
	match (listtasks()) {
	case let e: fs::error =>
		fmt::fatal(fs::strerror(e));
	case let e: path::error =>
		fmt::fatal(path::strerror(e));
	case =>
		void;
	};
};