Tue, 25 Feb 2025 18:46:17 +0100
add total commits counters
fixes #605
0 | 1 | /* Copyright 2025 Mike Becker. All rights reserved. |
2 | * | |
3 | * Redistribution and use in source and binary forms, with or without | |
4 | * modification, are permitted provided that the following conditions are met: | |
5 | * | |
6 | * 1. Redistributions of source code must retain the above copyright | |
7 | * notice, this list of conditions and the following disclaimer. | |
8 | * | |
9 | * 2. Redistributions in binary form must reproduce the above copyright | |
10 | * notice, this list of conditions and the following disclaimer in the | |
11 | * documentation and/or other materials provided with the distribution. | |
12 | * | |
13 | * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" | |
14 | * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE | |
15 | * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | |
16 | * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE | |
17 | * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL | |
18 | * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR | |
19 | * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER | |
20 | * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, | |
21 | * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE | |
22 | * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | |
23 | */ | |
24 | ||
1 | 25 | #include "settings.h" |
3 | 26 | #include "repositories.h" |
4
82680ce258d6
add automatic pull/udate of repositories
Mike Becker <universe@uap-core.de>
parents:
3
diff
changeset
|
27 | #include "process.h" |
5 | 28 | #include "heatmap.h" |
29 | #include "html.h" | |
1 | 30 | |
5 | 31 | #include <chrono> |
1 | 32 | #include <cstdlib> |
33 | #include <cstdio> | |
34 | #include <cstring> | |
35 | #include <cerrno> | |
36 | ||
44
de22ded6d50a
add total commits counters
Mike Becker <universe@uap-core.de>
parents:
41
diff
changeset
|
37 | #include <numeric> |
de22ded6d50a
add total commits counters
Mike Becker <universe@uap-core.de>
parents:
41
diff
changeset
|
38 | |
13 | 39 | using namespace std::chrono; |
5 | 40 | |
41
19cc90878968
fix wrong escape in raw string
Mike Becker <universe@uap-core.de>
parents:
39
diff
changeset
|
41 | static constexpr auto program_version = "1.1.0 (dev)"; |
29 | 42 | |
1 | 43 | static void print_help() { |
44 | fputs( | |
23
b4979c194dc8
replace working title with official project title
Mike Becker <universe@uap-core.de>
parents:
21
diff
changeset
|
45 | "Usage: repoheat [OPTION]... [PATH]...\n\n" |
1 | 46 | "Options:\n" |
14 | 47 | " -a, --author <name> Only report this author\n" |
48 | " (repeat option to report multiple authors)\n" | |
49 | " -A, --authormap <file> Apply an author mapping file\n" | |
20
8639ccd855ba
implement --fragment option
Mike Becker <universe@uap-core.de>
parents:
19
diff
changeset
|
50 | " -d, --depth <num> The search depth (default: 1, max: 255)\n" |
8639ccd855ba
implement --fragment option
Mike Becker <universe@uap-core.de>
parents:
19
diff
changeset
|
51 | " -f, --fragment Output as fragment\n" |
14 | 52 | " -h, --help Print this help message\n" |
27
d2eee642a31b
do not pull repos by default - fixes #587
Mike Becker <universe@uap-core.de>
parents:
24
diff
changeset
|
53 | " -p, --pull Try to pull the repositories\n" |
14 | 54 | " -s, --separate Output a separate heat map for each repository\n" |
29 | 55 | " -V, --version Output the version of this program and exit\n" |
14 | 56 | " -y, --year <year> The year for which to create the heat map\n" |
57 | " --hg <path> Path to hg binary (default: /usr/bin/hg)\n" | |
58 | " --git <path> Path to git binary (default: /usr/bin/git)\n\n" | |
1 | 59 | "Scans all specified paths recursively for Mercurial and Git repositories and\n" |
60 | "creates a commit heat map for the specified \033[1myear\033[22m or the current year.\n" | |
61 | "By default, the recursion \033[1mdepth\033[22m is one, meaning that this tool assumes that\n" | |
62 | "each \033[1mpath\033[22m is either a repository or contains repositories as subdirectories.\n" | |
63 | "You can change the \033[1mdepth\033[22m to support other directory structures.\n\n" | |
27
d2eee642a31b
do not pull repos by default - fixes #587
Mike Becker <universe@uap-core.de>
parents:
24
diff
changeset
|
64 | "When you specify the \033[1m--pull\033[22m option, this tool will execute the pull command\n" |
1 | 65 | "(and for hg the update command) before retrieving the commit log, assuming\n" |
9
98312f94dbdd
skip authorization requests when pulling
Mike Becker <universe@uap-core.de>
parents:
8
diff
changeset
|
66 | "to be on the default branch with \033[4mno uncommitted changes\033[24m. If pulling leads to\n" |
98312f94dbdd
skip authorization requests when pulling
Mike Becker <universe@uap-core.de>
parents:
8
diff
changeset
|
67 | "an error, an error message is written to stderr and the process continues\n" |
98312f94dbdd
skip authorization requests when pulling
Mike Becker <universe@uap-core.de>
parents:
8
diff
changeset
|
68 | "with the repository in its current state. This is also the case when pulling\n" |
27
d2eee642a31b
do not pull repos by default - fixes #587
Mike Becker <universe@uap-core.de>
parents:
24
diff
changeset
|
69 | "would require authorization.\n\n" |
14 | 70 | "By default, this tool reports commits from all authors. If you want to include\n" |
71 | "only specific authors in the report, you can use the \033[1m--author\033[22m option with as\n" | |
72 | "many authors as you like. You can specify either the full author strings with\n" | |
73 | "name and mail address, just the mail address, or even just the local-part of\n" | |
74 | "the mail address. In case your repository contains commits from an author who\n" | |
75 | "used different names or mail addresses, you can use the \033[1m--authormap\033[22m option\n" | |
76 | "to specify a file that contains pairs of author strings, like in the following\n" | |
77 | "example:\n\n" | |
78 | " Full Name <full.name@example.org> = New Name <new.name@example.org>\n" | |
16 | 79 | " just.mail@example.org = Jus Mail <just.mail@example.org>\n" |
80 | " jane = Jane Doe <jane.doe@example.org>\n\n" | |
14 | 81 | "The different variants of full string, only mail address, and only local-part\n" |
16 | 82 | "should \033[4monly\033[24m be used on the left-hand side. When you use the \033[1m--author\033[22m option at\n" |
83 | "the same time, you only need to specify the new author names.\n\n" | |
14 | 84 | "Finally, this tool prints an HTML page to stdout. A separate heap map is\n" |
3 | 85 | "generated for each author showing commits across all repositories, unless the\n" |
86 | "\033[1m--separate\033[22m option is specified in which case each repository is displayed with\n" | |
20
8639ccd855ba
implement --fragment option
Mike Becker <universe@uap-core.de>
parents:
19
diff
changeset
|
87 | "its own heat map. By using the \033[1m--fragment\033[22m option, the tool only outputs a\n" |
8639ccd855ba
implement --fragment option
Mike Becker <universe@uap-core.de>
parents:
19
diff
changeset
|
88 | "single HTML div container without any header or footer that can be embedded in\n" |
8639ccd855ba
implement --fragment option
Mike Becker <universe@uap-core.de>
parents:
19
diff
changeset
|
89 | "your custom web page.\n" |
1 | 90 | , stderr); |
91 | } | |
92 | ||
3 | 93 | static bool chk_arg(const char *arg, const char *opt1, const char *opt2) { |
1 | 94 | return strcmp(arg, opt1) == 0 || (opt2 != nullptr && strcmp(arg, opt2) == 0); |
95 | } | |
96 | ||
97 | template<typename T> | |
98 | static bool parse_unsigned(const char *str, T *result, unsigned long max) { | |
99 | char *endptr; | |
100 | errno = 0; | |
101 | unsigned long d = strtoul(str, &endptr, 10); | |
102 | if (*endptr != '\0' || errno == ERANGE) return true; | |
103 | if (d < max) { | |
104 | *result = d; | |
105 | return false; | |
106 | } else { | |
107 | return true; | |
108 | } | |
109 | } | |
110 | ||
111 | static int parse_args(fm::settings &settings, int argc, char *argv[]) { | |
112 | for (int i = 1; i < argc; i++) { | |
113 | if (chk_arg(argv[i], "-h", "--help")) { | |
114 | print_help(); | |
29 | 115 | return 1; |
1 | 116 | } else if (chk_arg(argv[i], "-d", "--depth")) { |
117 | if (i + 1 >= argc || parse_unsigned(argv[++i], &settings.depth, 256)) { | |
118 | fputs("missing or invalid depth\n", stderr); | |
119 | return -1; | |
120 | } | |
121 | } else if (chk_arg(argv[i], "-y", "--year")) { | |
122 | if (i + 1 >= argc || parse_unsigned(argv[++i], &settings.year, 9999)) { | |
123 | fputs("missing or invalid year\n", stderr); | |
124 | return -1; | |
125 | } | |
14 | 126 | } else if (chk_arg(argv[i], "-a", "--author")) { |
127 | if (i + 1 < argc) { | |
128 | settings.authors.emplace_back(argv[++i]); | |
129 | } else { | |
130 | fputs("missing author name\n", stderr); | |
131 | return -1; | |
132 | } | |
27
d2eee642a31b
do not pull repos by default - fixes #587
Mike Becker <universe@uap-core.de>
parents:
24
diff
changeset
|
133 | } else if (chk_arg(argv[i], "-p", "--pull")) { |
d2eee642a31b
do not pull repos by default - fixes #587
Mike Becker <universe@uap-core.de>
parents:
24
diff
changeset
|
134 | settings.update_repos = true; |
20
8639ccd855ba
implement --fragment option
Mike Becker <universe@uap-core.de>
parents:
19
diff
changeset
|
135 | } else if (chk_arg(argv[i], "-f", "--fragment")) { |
8639ccd855ba
implement --fragment option
Mike Becker <universe@uap-core.de>
parents:
19
diff
changeset
|
136 | settings.fragment = true; |
1 | 137 | } else if (chk_arg(argv[i], "-s", "--separate")) { |
138 | settings.separate = true; | |
16 | 139 | } else if (chk_arg(argv[i], "-A", "--authormap")) { |
140 | if (i + 1 < argc) { | |
141 | if (settings.parse_authormap(argv[++i])) { | |
142 | fputs("parsing authormap failed\n", stderr); | |
143 | return -1; | |
144 | } | |
145 | } else { | |
146 | fputs("missing filename for authormap\n", stderr); | |
147 | return -1; | |
148 | } | |
29 | 149 | } else if (chk_arg(argv[i], "-V", "--version")) { |
150 | printf("repoheat version %s\n", program_version); | |
151 | return 1; | |
1 | 152 | } else if (chk_arg(argv[i], "--hg", nullptr)) { |
153 | if (i + 1 < argc) { | |
154 | settings.hg.assign(argv[++i]); | |
155 | } else { | |
156 | fputs("--hg is expecting a path\n", stderr); | |
157 | return -1; | |
158 | } | |
159 | } else if (chk_arg(argv[i], "--git", nullptr)) { | |
160 | if (i + 1 < argc) { | |
161 | settings.git.assign(argv[++i]); | |
162 | } else { | |
163 | fputs("--git is expecting a path\n", stderr); | |
164 | return -1; | |
165 | } | |
166 | } else if (argv[i][0] == '-') { | |
167 | fprintf(stderr, "Unknown option: %s\n", argv[i]); | |
168 | return -1; | |
169 | } else { | |
170 | settings.paths.emplace_back(argv[i]); | |
171 | } | |
172 | } | |
173 | ||
174 | if (settings.paths.empty()) { | |
175 | settings.paths.emplace_back("./"); | |
176 | } | |
177 | ||
0 | 178 | return 0; |
179 | } | |
180 | ||
1 | 181 | int main(int argc, char *argv[]) { |
4
82680ce258d6
add automatic pull/udate of repositories
Mike Becker <universe@uap-core.de>
parents:
3
diff
changeset
|
182 | // parse settings |
1 | 183 | fm::settings settings; |
29 | 184 | if (int result = parse_args(settings, argc, argv); result != 0) { |
185 | return result < 0 ? EXIT_FAILURE : EXIT_SUCCESS; | |
1 | 186 | } |
187 | ||
4
82680ce258d6
add automatic pull/udate of repositories
Mike Becker <universe@uap-core.de>
parents:
3
diff
changeset
|
188 | // check hg and git |
82680ce258d6
add automatic pull/udate of repositories
Mike Becker <universe@uap-core.de>
parents:
3
diff
changeset
|
189 | fm::process proc; |
82680ce258d6
add automatic pull/udate of repositories
Mike Becker <universe@uap-core.de>
parents:
3
diff
changeset
|
190 | proc.setbin(settings.hg); |
82680ce258d6
add automatic pull/udate of repositories
Mike Becker <universe@uap-core.de>
parents:
3
diff
changeset
|
191 | if (proc.exec({"--version"})) { |
82680ce258d6
add automatic pull/udate of repositories
Mike Becker <universe@uap-core.de>
parents:
3
diff
changeset
|
192 | fprintf(stderr, "Testing hg binary '%s' failed!\n", settings.hg.c_str()); |
82680ce258d6
add automatic pull/udate of repositories
Mike Becker <universe@uap-core.de>
parents:
3
diff
changeset
|
193 | return EXIT_FAILURE; |
82680ce258d6
add automatic pull/udate of repositories
Mike Becker <universe@uap-core.de>
parents:
3
diff
changeset
|
194 | } |
82680ce258d6
add automatic pull/udate of repositories
Mike Becker <universe@uap-core.de>
parents:
3
diff
changeset
|
195 | proc.setbin(settings.git); |
82680ce258d6
add automatic pull/udate of repositories
Mike Becker <universe@uap-core.de>
parents:
3
diff
changeset
|
196 | if (proc.exec({"--version"})) { |
82680ce258d6
add automatic pull/udate of repositories
Mike Becker <universe@uap-core.de>
parents:
3
diff
changeset
|
197 | fprintf(stderr, "Testing git binary '%s' failed!\n", settings.git.c_str()); |
82680ce258d6
add automatic pull/udate of repositories
Mike Becker <universe@uap-core.de>
parents:
3
diff
changeset
|
198 | return EXIT_FAILURE; |
82680ce258d6
add automatic pull/udate of repositories
Mike Becker <universe@uap-core.de>
parents:
3
diff
changeset
|
199 | } |
82680ce258d6
add automatic pull/udate of repositories
Mike Becker <universe@uap-core.de>
parents:
3
diff
changeset
|
200 | |
82680ce258d6
add automatic pull/udate of repositories
Mike Becker <universe@uap-core.de>
parents:
3
diff
changeset
|
201 | // scan for repos |
3 | 202 | fm::repositories repos; |
203 | for (auto &&path: settings.paths) { | |
204 | repos.scan(path, settings.depth); | |
205 | } | |
4
82680ce258d6
add automatic pull/udate of repositories
Mike Becker <universe@uap-core.de>
parents:
3
diff
changeset
|
206 | |
82680ce258d6
add automatic pull/udate of repositories
Mike Becker <universe@uap-core.de>
parents:
3
diff
changeset
|
207 | // update repos, if not disabled |
82680ce258d6
add automatic pull/udate of repositories
Mike Becker <universe@uap-core.de>
parents:
3
diff
changeset
|
208 | if (settings.update_repos) { |
82680ce258d6
add automatic pull/udate of repositories
Mike Becker <universe@uap-core.de>
parents:
3
diff
changeset
|
209 | for (auto &&repo : repos.list()) { |
82680ce258d6
add automatic pull/udate of repositories
Mike Becker <universe@uap-core.de>
parents:
3
diff
changeset
|
210 | proc.chdir(repo.path); |
82680ce258d6
add automatic pull/udate of repositories
Mike Becker <universe@uap-core.de>
parents:
3
diff
changeset
|
211 | if (repo.type == fm::HG) { |
82680ce258d6
add automatic pull/udate of repositories
Mike Becker <universe@uap-core.de>
parents:
3
diff
changeset
|
212 | proc.setbin(settings.hg); |
9
98312f94dbdd
skip authorization requests when pulling
Mike Becker <universe@uap-core.de>
parents:
8
diff
changeset
|
213 | if (proc.exec({"pull", "-y"})) { |
21
bc8b76ca9ee9
fix possible misleading error output
Mike Becker <universe@uap-core.de>
parents:
20
diff
changeset
|
214 | fprintf(stderr, "Pulling repo '%s' failed - continue without pull.\n", repo.path.c_str()); |
4
82680ce258d6
add automatic pull/udate of repositories
Mike Becker <universe@uap-core.de>
parents:
3
diff
changeset
|
215 | } else if (proc.exec({"update"})) { |
82680ce258d6
add automatic pull/udate of repositories
Mike Becker <universe@uap-core.de>
parents:
3
diff
changeset
|
216 | fprintf(stderr, "Updating repo '%s' failed!\nMaybe there are local changes?\n", repo.path.c_str()); |
82680ce258d6
add automatic pull/udate of repositories
Mike Becker <universe@uap-core.de>
parents:
3
diff
changeset
|
217 | } |
82680ce258d6
add automatic pull/udate of repositories
Mike Becker <universe@uap-core.de>
parents:
3
diff
changeset
|
218 | } else { |
82680ce258d6
add automatic pull/udate of repositories
Mike Becker <universe@uap-core.de>
parents:
3
diff
changeset
|
219 | proc.setbin(settings.git); |
9
98312f94dbdd
skip authorization requests when pulling
Mike Becker <universe@uap-core.de>
parents:
8
diff
changeset
|
220 | if (proc.exec({"pull", "-q"})) { |
21
bc8b76ca9ee9
fix possible misleading error output
Mike Becker <universe@uap-core.de>
parents:
20
diff
changeset
|
221 | fprintf(stderr, "Pulling repo '%s' failed - continue without pull.\n", repo.path.c_str()); |
4
82680ce258d6
add automatic pull/udate of repositories
Mike Becker <universe@uap-core.de>
parents:
3
diff
changeset
|
222 | } |
82680ce258d6
add automatic pull/udate of repositories
Mike Becker <universe@uap-core.de>
parents:
3
diff
changeset
|
223 | } |
82680ce258d6
add automatic pull/udate of repositories
Mike Becker <universe@uap-core.de>
parents:
3
diff
changeset
|
224 | } |
3 | 225 | } |
5 | 226 | |
227 | // determine our reporting range | |
13 | 228 | year report_year{ |
229 | settings.year == fm::settings_current_year | |
230 | ? year_month_day{floor<days>(system_clock::now())}.year() | |
231 | : year{settings.year} | |
232 | }; | |
233 | year_month_day report_begin{report_year, January, 1d}; | |
234 | year_month_day report_end{report_year, December, 31d}; | |
5 | 235 | |
236 | // read the commit logs | |
237 | fm::heatmap heatmap; | |
238 | for (auto &&repo : repos.list()) { | |
239 | if (settings.separate) { | |
37
d7e9a1200e21
improve headings in separate repository view
Mike Becker <universe@uap-core.de>
parents:
36
diff
changeset
|
240 | heatmap.set_repo(repo.name); |
5 | 241 | } |
242 | proc.chdir(repo.path); | |
243 | if (repo.type == fm::HG) { | |
244 | proc.setbin(settings.hg); | |
245 | if (proc.exec_log({"log", | |
36 | 246 | "--date", std::format("{0}-01-01 00:00:00 to {0}-12-31 23:59:59", report_year), |
5 | 247 | "--template", "{author}#{date|shortdate}\n"})) { |
248 | fprintf(stderr, "Reading commit log for repo '%s' failed!\n", repo.path.c_str()); | |
249 | return EXIT_FAILURE; | |
250 | } | |
16 | 251 | heatmap.add(settings, proc.output()); |
5 | 252 | } else { |
253 | proc.setbin(settings.git); | |
8
6a2e20a4a8ff
fix log not being captured from git
Mike Becker <universe@uap-core.de>
parents:
7
diff
changeset
|
254 | if (proc.exec_log({"log", |
36 | 255 | "--since", std::format("{0}-01-01 00:00:00", report_year), |
256 | "--until", std::format("{0}-12-31 23:59:59", report_year), | |
5 | 257 | "--format=tformat:%an <%ae>#%cs"})) { |
258 | fprintf(stderr, "Reading commit log for repo '%s' failed!\n", repo.path.c_str()); | |
259 | return EXIT_FAILURE; | |
260 | } | |
16 | 261 | heatmap.add(settings, proc.output()); |
5 | 262 | } |
263 | } | |
264 | ||
20
8639ccd855ba
implement --fragment option
Mike Becker <universe@uap-core.de>
parents:
19
diff
changeset
|
265 | html::open(settings.fragment); |
5 | 266 | for (const auto &[repo, authors] : heatmap.data()) { |
24
2a4e97fc5ce5
fix repo h1 generated even when no author has commits - fixes #585
Mike Becker <universe@uap-core.de>
parents:
23
diff
changeset
|
267 | bool h1_rendered = false; |
5 | 268 | for (const auto &[author, entries] : authors) { |
14 | 269 | if (settings.exclude_author(author)) continue; |
24
2a4e97fc5ce5
fix repo h1 generated even when no author has commits - fixes #585
Mike Becker <universe@uap-core.de>
parents:
23
diff
changeset
|
270 | if (!h1_rendered) { |
44
de22ded6d50a
add total commits counters
Mike Becker <universe@uap-core.de>
parents:
41
diff
changeset
|
271 | html::heading_repo(repo); |
24
2a4e97fc5ce5
fix repo h1 generated even when no author has commits - fixes #585
Mike Becker <universe@uap-core.de>
parents:
23
diff
changeset
|
272 | h1_rendered = true; |
2a4e97fc5ce5
fix repo h1 generated even when no author has commits - fixes #585
Mike Becker <universe@uap-core.de>
parents:
23
diff
changeset
|
273 | } |
44
de22ded6d50a
add total commits counters
Mike Becker <universe@uap-core.de>
parents:
41
diff
changeset
|
274 | |
de22ded6d50a
add total commits counters
Mike Becker <universe@uap-core.de>
parents:
41
diff
changeset
|
275 | const auto commits_per_month = heatmap.commits_per_month(repo, author, report_year); |
de22ded6d50a
add total commits counters
Mike Becker <universe@uap-core.de>
parents:
41
diff
changeset
|
276 | const auto total_commits = std::accumulate(commits_per_month.begin(), commits_per_month.end(), 0u); |
de22ded6d50a
add total commits counters
Mike Becker <universe@uap-core.de>
parents:
41
diff
changeset
|
277 | html::heading_author(author, total_commits); |
de22ded6d50a
add total commits counters
Mike Becker <universe@uap-core.de>
parents:
41
diff
changeset
|
278 | html::table_begin(report_year, commits_per_month); |
5 | 279 | |
280 | // initialize counters | |
281 | unsigned column = 0, row = 0; | |
3 | 282 | |
5 | 283 | // initialize first day (which must be a Monday, possibly the year before) |
13 | 284 | sys_days day_to_check = January / Monday[1] / report_year; |
19
2c128952f198
fix accidentally breaking the start of year
Mike Becker <universe@uap-core.de>
parents:
16
diff
changeset
|
285 | if (year_month_day{day_to_check}.day() != 1d) { |
2c128952f198
fix accidentally breaking the start of year
Mike Becker <universe@uap-core.de>
parents:
16
diff
changeset
|
286 | day_to_check -= days{7}; |
2c128952f198
fix accidentally breaking the start of year
Mike Becker <universe@uap-core.de>
parents:
16
diff
changeset
|
287 | } |
5 | 288 | |
289 | // remember the starting point | |
290 | auto start = day_to_check; | |
291 | ||
292 | // now add all entries for Monday, Tuesdays, etc. always starting back in january | |
293 | while (true) { | |
294 | html::row_begin(row); | |
295 | ||
296 | // check if we need to add blank cells | |
297 | while (day_to_check < report_begin) { | |
298 | html::cell_out_of_range(); | |
13 | 299 | day_to_check += days{7}; |
5 | 300 | column++; |
301 | } | |
302 | ||
303 | while (day_to_check <= report_end) { | |
304 | // get the entry from the heatmap | |
305 | auto find_result = entries.find(day_to_check); | |
306 | if (find_result == entries.end()) { | |
7 | 307 | html::cell(day_to_check, 0); |
5 | 308 | } else { |
7 | 309 | html::cell(day_to_check, find_result->second); |
5 | 310 | } |
311 | // advance seven days and one column | |
13 | 312 | day_to_check += days{7}; |
5 | 313 | column++; |
314 | } | |
315 | // fill remaining columns with blank cells | |
316 | for (unsigned i = column ; i < html::columns ; i++) { | |
317 | html::cell_out_of_range(); | |
318 | } | |
319 | ||
320 | // terminate the row | |
321 | html::row_end(); | |
322 | ||
323 | // if we have seen all seven weekdays, that's it | |
324 | if (++row == 7) break; | |
325 | ||
326 | // otherwise, advance the starting point by one day, reset, and begin a new row | |
13 | 327 | start += days{1}; |
5 | 328 | day_to_check = start; |
329 | column =0; | |
330 | } | |
331 | ||
332 | html::table_end(); | |
333 | } | |
334 | } | |
20
8639ccd855ba
implement --fragment option
Mike Becker <universe@uap-core.de>
parents:
19
diff
changeset
|
335 | html::close(settings.fragment); |
3 | 336 | |
1 | 337 | return EXIT_SUCCESS; |
338 | } |