hare

[hare] The Hare programming language
git clone https://git.torresjrjr.com/hare.git
Log | Files | Refs | README | LICENSE

html.ha (67907B)


      1 // License: GPL-3.0
      2 // (c) 2021-2022 Alexey Yerin <yyp@disroot.org>
      3 // (c) 2022 Byron Torres <b@torresjrjr.com>
      4 // (c) 2021-2022 Drew DeVault <sir@cmpwn.com>
      5 // (c) 2021 Eyal Sawady <ecs@d2evs.net>
      6 // (c) 2021 Thomas Bracht Laumann Jespersen <t@laumann.xyz>
      7 // (c) 2022 Umar Getagazov <umar@handlerug.me>
      8 
      9 // Note: ast::ident should never have to be escaped
     10 use bufio;
     11 use encoding::utf8;
     12 use fmt;
     13 use hare::ast;
     14 use hare::ast::{variadism};
     15 use hare::lex;
     16 use hare::module;
     17 use hare::unparse;
     18 use io;
     19 use net::ip;
     20 use net::uri;
     21 use os;
     22 use path;
     23 use strings;
     24 use strio;
     25 
     26 // Prints a string to an output handle, escaping any of HTML's reserved
     27 // characters.
     28 fn html_escape(out: io::handle, in: str) (size | io::error) = {
     29 	let z = 0z;
     30 	let iter = strings::iter(in);
     31 	for (true) {
     32 		match (strings::next(&iter)) {
     33 		case void => break;
     34 		case let rn: rune =>
     35 			z += fmt::fprint(out, switch (rn) {
     36 			case '&' =>
     37 				yield "&amp;";
     38 			case '<' =>
     39 				yield "&lt;";
     40 			case '>' =>
     41 				yield "&gt;";
     42 			case '"' =>
     43 				yield "&quot;";
     44 			case '\'' =>
     45 				yield "&apos;";
     46 			case =>
     47 				yield strings::fromutf8(utf8::encoderune(rn));
     48 			})?;
     49 		};
     50 	};
     51 	return z;
     52 };
     53 
     54 @test fn html_escape() void = {
     55        let sink = strio::dynamic();
     56        defer io::close(&sink)!;
     57        html_escape(&sink, "hello world!")!;
     58        assert(strio::string(&sink) == "hello world!");
     59 
     60        let sink = strio::dynamic();
     61        defer io::close(&sink)!;
     62        html_escape(&sink, "\"hello world!\"")!;
     63        assert(strio::string(&sink) == "&quot;hello world!&quot;");
     64 
     65        let sink = strio::dynamic();
     66        defer io::close(&sink)!;
     67        html_escape(&sink, "<hello & 'world'!>")!;
     68        assert(strio::string(&sink) == "&lt;hello &amp; &apos;world&apos;!&gt;");
     69 };
     70 
     71 // Formats output as HTML
     72 fn emit_html(ctx: *context) (void | error) = {
     73 	const decls = ctx.summary;
     74 	const ident = unparse::identstr(ctx.ident);
     75 	defer free(ident);
     76 
     77 	if (ctx.template) head(ctx.ident)?;
     78 
     79 	if (len(ident) == 0) {
     80 		fmt::fprintf(ctx.out, "<h2>The Hare standard library <span class='heading-extra'>")?;
     81 	} else {
     82 		fmt::fprintf(ctx.out, "<h2>{} <span class='heading-extra'>", ident)?;
     83 	};
     84 	for (let i = 0z; i < len(ctx.tags); i += 1) {
     85 		const mode = switch (ctx.tags[i].mode) {
     86 		case module::tag_mode::INCLUSIVE =>
     87 			yield '+';
     88 		case module::tag_mode::EXCLUSIVE =>
     89 			yield '-';
     90 		};
     91 		fmt::fprintf(ctx.out, "{}{} ", mode, ctx.tags[i].name)?;
     92 	};
     93 	fmt::fprintln(ctx.out, "</span></h2>")?;
     94 
     95 	match (ctx.readme) {
     96 	case void => void;
     97 	case let f: io::file =>
     98 		fmt::fprintln(ctx.out, "<div class='readme'>")?;
     99 		markup_html(ctx, f)?;
    100 		fmt::fprintln(ctx.out, "</div>")?;
    101 	};
    102 
    103 	let identpath = module::identpath(ctx.ident);
    104 	defer free(identpath);
    105 
    106 	let submodules: []str = [];
    107 	defer free(submodules);
    108 
    109 	for (let i = 0z; i < len(ctx.version.subdirs); i += 1) {
    110 		let dir = ctx.version.subdirs[i];
    111 		// XXX: the list of reserved directory names is not yet
    112 		// finalized. See https://todo.sr.ht/~sircmpwn/hare/516
    113 		if (dir == "contrib") continue;
    114 		if (dir == "cmd") continue;
    115 		if (dir == "docs") continue;
    116 		if (dir == "ext") continue;
    117 		if (dir == "vendor") continue;
    118 		if (dir == "scripts") continue;
    119 
    120 		let submod = [identpath, dir]: ast::ident;
    121 		if (module::lookup(ctx.mctx, submod) is module::error) {
    122 			continue;
    123 		};
    124 
    125 		append(submodules, dir);
    126 	};
    127 
    128 	if (len(submodules) != 0) {
    129 		if (len(ctx.ident) == 0) {
    130 			fmt::fprintln(ctx.out, "<h3>Modules</h3>")?;
    131 		} else {
    132 			fmt::fprintln(ctx.out, "<h3>Submodules</h3>")?;
    133 		};
    134 		fmt::fprintln(ctx.out, "<ul class='submodules'>")?;
    135 		for (let i = 0z; i < len(submodules); i += 1) {
    136 			let submodule = submodules[i];
    137 			let path = path::join("/", identpath, submodule);
    138 			defer free(path);
    139 
    140 			fmt::fprintf(ctx.out, "<li><a href='")?;
    141 			html_escape(ctx.out, path)?;
    142 			fmt::fprintf(ctx.out, "'>")?;
    143 			html_escape(ctx.out, submodule)?;
    144 			fmt::fprintfln(ctx.out, "</a></li>")?;
    145 		};
    146 		fmt::fprintln(ctx.out, "</ul>")?;
    147 	};
    148 
    149 	if (len(decls.types) == 0
    150 			&& len(decls.errors) == 0
    151 			&& len(decls.constants) == 0
    152 			&& len(decls.globals) == 0
    153 			&& len(decls.funcs) == 0) {
    154 		return;
    155 	};
    156 
    157 	fmt::fprintln(ctx.out, "<h3>Index</h3>")?;
    158 	tocentries(ctx.out, decls.types, "Types", "types")?;
    159 	tocentries(ctx.out, decls.errors, "Errors", "Errors")?;
    160 	tocentries(ctx.out, decls.constants, "Constants", "constants")?;
    161 	tocentries(ctx.out, decls.globals, "Globals", "globals")?;
    162 	tocentries(ctx.out, decls.funcs, "Functions", "functions")?;
    163 
    164 	if (len(decls.types) != 0) {
    165 		fmt::fprintln(ctx.out, "<h3>Types</h3>")?;
    166 		for (let i = 0z; i < len(decls.types); i += 1) {
    167 			details(ctx, decls.types[i])?;
    168 		};
    169 	};
    170 
    171 	if (len(decls.errors) != 0) {
    172 		fmt::fprintln(ctx.out, "<h3>Errors</h3>")?;
    173 		for (let i = 0z; i < len(decls.errors); i += 1) {
    174 			details(ctx, decls.errors[i])?;
    175 		};
    176 	};
    177 
    178 	if (len(decls.constants) != 0) {
    179 		fmt::fprintln(ctx.out, "<h3>Constants</h3>")?;
    180 		for (let i = 0z; i < len(decls.constants); i += 1) {
    181 			details(ctx, decls.constants[i])?;
    182 		};
    183 	};
    184 
    185 	if (len(decls.globals) != 0) {
    186 		fmt::fprintln(ctx.out, "<h3>Globals</h3>")?;
    187 		for (let i = 0z; i < len(decls.globals); i += 1) {
    188 			details(ctx, decls.globals[i])?;
    189 		};
    190 	};
    191 
    192 	if (len(decls.funcs) != 0) {
    193 		fmt::fprintln(ctx.out, "<h3>Functions</h3>")?;
    194 		for (let i = 0z; i < len(decls.funcs); i += 1) {
    195 			details(ctx, decls.funcs[i])?;
    196 		};
    197 	};
    198 };
    199 
    200 fn comment_html(out: io::handle, s: str) (size | io::error) = {
    201 	// TODO: handle [[references]]
    202 	let z = fmt::fprint(out, "<span class='comment'>//")?;
    203 	z += html_escape(out, s)?;
    204 	z += fmt::fprint(out, "</span><br>")?;
    205 	return z;
    206 };
    207 
    208 fn docs_html(out: io::handle, s: str, indent: size) (size | io::error) = {
    209 	const iter = strings::tokenize(s, "\n");
    210 	let z = 0z;
    211 	for (true) match (strings::next_token(&iter)) {
    212 	case let s: str =>
    213 		if (!(strings::peek_token(&iter) is void)) {
    214 			z += comment_html(out, s)?;
    215 			for (let i = 0z; i < indent; i += 1) {
    216 				z += fmt::fprint(out, "\t")?;
    217 			};
    218 		};
    219 	case void => break;
    220 	};
    221 
    222 	return z;
    223 };
    224 
    225 fn tocentries(
    226 	out: io::handle,
    227 	decls: []ast::decl,
    228 	name: str,
    229 	lname: str,
    230 ) (void | error) = {
    231 	if (len(decls) == 0) {
    232 		return;
    233 	};
    234 	fmt::fprintfln(out, "<h4>{}</h4>", name)?;
    235 	fmt::fprintln(out, "<pre>")?;
    236 	let undoc = false;
    237 	for (let i = 0z; i < len(decls); i += 1) {
    238 		if (!undoc && decls[i].docs == "") {
    239 			fmt::fprintfln(
    240 				out,
    241 				"{}<span class='comment'>// Undocumented {}:</span>",
    242 				if (i == 0) "" else "\n",
    243 				lname)?;
    244 			undoc = true;
    245 		};
    246 		tocentry(out, decls[i])?;
    247 	};
    248 	fmt::fprint(out, "</pre>")?;
    249 	return;
    250 };
    251 
    252 fn tocentry(out: io::handle, decl: ast::decl) (void | error) = {
    253 	fmt::fprintf(out, "{} ",
    254 		match (decl.decl) {
    255 		case ast::decl_func =>
    256 			yield "fn";
    257 		case []ast::decl_type =>
    258 			yield "type";
    259 		case []ast::decl_const =>
    260 			yield "const";
    261 		case []ast::decl_global =>
    262 			yield "let";
    263 		})?;
    264 	fmt::fprintf(out, "<a href='#")?;
    265 	unparse::ident(out, decl_ident(decl))?;
    266 	fmt::fprintf(out, "'>")?;
    267 	unparse::ident(out, decl_ident(decl))?;
    268 	fmt::fprint(out, "</a>")?;
    269 
    270 	match (decl.decl) {
    271 	case let t: []ast::decl_type => void;
    272 	case let g: []ast::decl_global =>
    273 		let g = g[0];
    274 		fmt::fprint(out, ": ")?;
    275 		type_html(out, 0, g._type, true)?;
    276 	case let c: []ast::decl_const =>
    277 		let c = c[0];
    278 		fmt::fprint(out, ": ")?;
    279 		type_html(out, 0, c._type, true)?;
    280 	case let f: ast::decl_func =>
    281 		prototype_html(out, 0,
    282 			f.prototype.repr as ast::func_type,
    283 			true)?;
    284 	};
    285 	fmt::fprintln(out, ";")?;
    286 	return;
    287 };
    288 
    289 fn details(ctx: *context, decl: ast::decl) (void | error) = {
    290 	fmt::fprintln(ctx.out, "<section class='member'>")?;
    291 	fmt::fprint(ctx.out, "<h4 id='")?;
    292 	unparse::ident(ctx.out, decl_ident(decl))?;
    293 	fmt::fprint(ctx.out, "'>")?;
    294 	fmt::fprintf(ctx.out, "{} ", match (decl.decl) {
    295 		case ast::decl_func =>
    296 			yield "fn";
    297 		case []ast::decl_type =>
    298 			yield "type";
    299 		case []ast::decl_const =>
    300 			yield "def";
    301 		case []ast::decl_global =>
    302 			yield "let";
    303 		})?;
    304 	unparse::ident(ctx.out, decl_ident(decl))?;
    305 	// TODO: Add source URL
    306 	fmt::fprint(ctx.out, "<span class='heading-extra'><a href='#")?;
    307 	unparse::ident(ctx.out, decl_ident(decl))?;
    308 	fmt::fprint(ctx.out, "'>[link]</a>
    309 	</span>")?;
    310 	fmt::fprintln(ctx.out, "</h4>")?;
    311 
    312 	if (len(decl.docs) == 0) {
    313 		fmt::fprintln(ctx.out, "<details>")?;
    314 		fmt::fprintln(ctx.out, "<summary>Show undocumented member</summary>")?;
    315 	};
    316 
    317 	fmt::fprintln(ctx.out, "<pre class='decl'>")?;
    318 	unparse_html(ctx.out, decl)?;
    319 	fmt::fprintln(ctx.out, "</pre>")?;
    320 
    321 	if (len(decl.docs) != 0) {
    322 		const trimmed = trim_comment(decl.docs);
    323 		defer free(trimmed);
    324 		const buf = strings::toutf8(trimmed);
    325 		markup_html(ctx, &bufio::fixed(buf, io::mode::READ))?;
    326 	} else {
    327 		fmt::fprintln(ctx.out, "</details>")?;
    328 	};
    329 
    330 	fmt::fprintln(ctx.out, "</section>")?;
    331 	return;
    332 };
    333 
    334 fn htmlref(ctx: *context, ref: ast::ident) (void | io::error) = {
    335 	const ik =
    336 		match (resolve(ctx, ref)) {
    337 		case let ik: (ast::ident, symkind) =>
    338 			yield ik;
    339 		case void =>
    340 			const ident = unparse::identstr(ref);
    341 			fmt::errorfln("Warning: Unresolved reference: {}", ident)?;
    342 			fmt::fprintf(ctx.out, "<a href='#' "
    343 				"class='ref invalid' "
    344 				"title='This reference could not be found'>{}</a>",
    345 				ident)?;
    346 			free(ident);
    347 			return;
    348 		};
    349 
    350 	// TODO: The reference is not necessarily in the stdlib
    351 	const kind = ik.1, id = ik.0;
    352 	const ident = unparse::identstr(id);
    353 	switch (kind) {
    354 	case symkind::LOCAL =>
    355 		fmt::fprintf(ctx.out, "<a href='#{0}' class='ref'>{0}</a>", ident)?;
    356 	case symkind::MODULE =>
    357 		let ipath = module::identpath(id);
    358 		defer free(ipath);
    359 		fmt::fprintf(ctx.out, "<a href='/{}' class='ref'>{}</a>",
    360 			ipath, ident)?;
    361 	case symkind::SYMBOL =>
    362 		let ipath = module::identpath(id[..len(id) - 1]);
    363 		defer free(ipath);
    364 		fmt::fprintf(ctx.out, "<a href='/{}#{}' class='ref'>{}</a>",
    365 			ipath, id[len(id) - 1], ident)?;
    366 	case symkind::ENUM_LOCAL =>
    367 		fmt::fprintf(ctx.out, "<a href='#{}' class='ref'>{}</a>",
    368 			id[len(id) - 2], ident)?;
    369 	case symkind::ENUM_REMOTE =>
    370 		let ipath = module::identpath(id[..len(id) - 2]);
    371 		defer free(ipath);
    372 		fmt::fprintf(ctx.out, "<a href='/{}#{}' class='ref'>{}</a>",
    373 			ipath, id[len(id) - 2], ident)?;
    374 	};
    375 	free(ident);
    376 };
    377 
    378 fn markup_html(ctx: *context, in: io::handle) (void | io::error) = {
    379 	let parser = parsedoc(in);
    380 	let waslist = false;
    381 	for (true) {
    382 		const tok = match (scandoc(&parser)) {
    383 			case void =>
    384 				if (waslist) {
    385 					fmt::fprintln(ctx.out, "</ul>")?;
    386 				};
    387 				break;
    388 			case let tok: token =>
    389 				yield tok;
    390 			};
    391 		match (tok) {
    392 		case paragraph =>
    393 			if (waslist) {
    394 				fmt::fprintln(ctx.out, "</ul>")?;
    395 				waslist = false;
    396 			};
    397 			fmt::fprintln(ctx.out)?;
    398 			fmt::fprint(ctx.out, "<p>")?;
    399 		case let tx: text =>
    400 			defer free(tx);
    401 			match (uri::parse(strings::trim(tx))) {
    402 			case let uri: uri::uri =>
    403 				defer uri::finish(&uri);
    404 				if (uri.host is net::ip::addr || len(uri.host as str) > 0) {
    405 					fmt::fprint(ctx.out, "<a rel='nofollow noopener' href='")?;
    406 					uri::fmt(ctx.out, &uri)?;
    407 					fmt::fprint(ctx.out, "'>")?;
    408 					html_escape(ctx.out, tx)?;
    409 					fmt::fprint(ctx.out, "</a>")?;
    410 				} else {
    411 					html_escape(ctx.out, tx)?;
    412 				};
    413 			case uri::invalid =>
    414 				html_escape(ctx.out, tx)?;
    415 			};
    416 		case let re: reference =>
    417 			htmlref(ctx, re)?;
    418 		case let sa: sample =>
    419 			if (waslist) {
    420 				fmt::fprintln(ctx.out, "</ul>")?;
    421 				waslist = false;
    422 			};
    423 			fmt::fprint(ctx.out, "<pre class='sample'>")?;
    424 			html_escape(ctx.out, sa)?;
    425 			fmt::fprint(ctx.out, "</pre>")?;
    426 			free(sa);
    427 		case listitem =>
    428 			if (!waslist) {
    429 				fmt::fprintln(ctx.out, "<ul>")?;
    430 				waslist = true;
    431 			};
    432 			fmt::fprint(ctx.out, "<li>")?;
    433 		};
    434 	};
    435 	fmt::fprintln(ctx.out)?;
    436 	return;
    437 };
    438 
    439 // Forked from [[hare::unparse]]
    440 fn unparse_html(out: io::handle, d: ast::decl) (size | io::error) = {
    441 	let n = 0z;
    442 	match (d.decl) {
    443 	case let c: []ast::decl_const =>
    444 		n += fmt::fprintf(out, "<span class='keyword'>def</span> ")?;
    445 		for (let i = 0z; i < len(c); i += 1) {
    446 			n += unparse::ident(out, c[i].ident)?;
    447 			n += fmt::fprint(out, ": ")?;
    448 			n += type_html(out, 0, c[i]._type, false)?;
    449 			if (i + 1 < len(c)) {
    450 				n += fmt::fprint(out, ", ")?;
    451 			};
    452 		};
    453 	case let g: []ast::decl_global =>
    454 		n += fmt::fprintf(out, "<span class='keyword'>{}</span>",
    455 			if (g[0].is_const) "const " else "let ")?;
    456 		for (let i = 0z; i < len(g); i += 1) {
    457 			n += unparse::ident(out, g[i].ident)?;
    458 			n += fmt::fprint(out, ": ")?;
    459 			n += type_html(out, 0, g[i]._type, false)?;
    460 			if (i + 1 < len(g)) {
    461 				n += fmt::fprint(out, ", ")?;
    462 			};
    463 		};
    464 	case let t: []ast::decl_type =>
    465 		n += fmt::fprint(out, "<span class='keyword'>type</span> ")?;
    466 		for (let i = 0z; i < len(t); i += 1) {
    467 			n += unparse::ident(out, t[i].ident)?;
    468 			n += fmt::fprint(out, " = ")?;
    469 			n += type_html(out, 0, t[i]._type, false)?;
    470 			if (i + 1 < len(t)) {
    471 				n += fmt::fprint(out, ", ")?;
    472 			};
    473 		};
    474 	case let f: ast::decl_func =>
    475 		n += fmt::fprint(out, switch (f.attrs) {
    476 		case ast::fndecl_attrs::NONE =>
    477 			yield "";
    478 		case ast::fndecl_attrs::FINI =>
    479 			yield "@fini ";
    480 		case ast::fndecl_attrs::INIT =>
    481 			yield "@init ";
    482 		case ast::fndecl_attrs::TEST =>
    483 			yield "@test ";
    484 		})?;
    485 		let p = f.prototype.repr as ast::func_type;
    486 		if (p.attrs & ast::func_attrs::NORETURN != 0) {
    487 			n += fmt::fprint(out, "@noreturn ")?;
    488 		};
    489 		n += fmt::fprint(out, "<span class='keyword'>fn</span> ")?;
    490 		n += unparse::ident(out, f.ident)?;
    491 		n += prototype_html(out, 0,
    492 			f.prototype.repr as ast::func_type,
    493 			false)?;
    494 	};
    495 	n += fmt::fprint(out, ";")?;
    496 	return n;
    497 };
    498 
    499 fn enum_html(
    500 	out: io::handle,
    501 	indent: size,
    502 	t: ast::enum_type
    503 ) (size | io::error) = {
    504 	let z = 0z;
    505 
    506 	z += fmt::fprint(out, "<span class='type'>enum</span> ")?;
    507 	if (t.storage != ast::builtin_type::INT) {
    508 		z += fmt::fprintf(out, "<span class='type'>{}</span> ",
    509 			unparse::builtin_type(t.storage))?;
    510 	};
    511 	z += fmt::fprintln(out, "{")?;
    512 	indent += 1;
    513 	for (let i = 0z; i < len(t.values); i += 1) {
    514 		for (let i = 0z; i < indent; i += 1) {
    515 			z += fmt::fprint(out, "\t")?;
    516 		};
    517 		const val = t.values[i];
    518 		let wrotedocs = false;
    519 		if (val.docs != "") {
    520 			// Check if comment should go above or next to field
    521 			if (multiline_comment(val.docs)) {
    522 				z += docs_html(out, val.docs, indent)?;
    523 				wrotedocs = true;
    524 			};
    525 		};
    526 
    527 		z += fmt::fprint(out, val.name)?;
    528 
    529 		match (val.value) {
    530 		case null => void;
    531 		case let expr: *ast::expr =>
    532 			z += fmt::fprint(out, " = ")?;
    533 			z += unparse::expr(out, indent, *expr)?;
    534 		};
    535 
    536 		z += fmt::fprint(out, ",")?;
    537 
    538 		if (val.docs != "" && !wrotedocs) {
    539 			z += fmt::fprint(out, " ")?;
    540 			z += docs_html(out, val.docs, 0)?;
    541 		} else {
    542 			z += fmt::fprintln(out)?;
    543 		};
    544 	};
    545 	indent -= 1;
    546 	for (let i = 0z; i < indent; i += 1) {
    547 		z += fmt::fprint(out, "\t")?;
    548 	};
    549 	z += newline(out, indent)?;
    550 	z += fmt::fprint(out, "}")?;
    551 	return z;
    552 };
    553 
    554 fn struct_union_html(
    555 	out: io::handle,
    556 	indent: size,
    557 	t: ast::_type,
    558 	brief: bool,
    559 ) (size | io::error) = {
    560 	let z = 0z;
    561 	let members = match (t.repr) {
    562 	case let t: ast::struct_type =>
    563 		z += fmt::fprint(out, "<span class='keyword'>struct</span> {")?;
    564 		yield t: []ast::struct_member;
    565 	case let t: ast::union_type =>
    566 		z += fmt::fprint(out, "<span class='keyword'>union</span> {")?;
    567 		yield t: []ast::struct_member;
    568 	};
    569 
    570 	indent += 1;
    571 	for (let i = 0z; i < len(members); i += 1) {
    572 		const member = members[i];
    573 
    574 		z += newline(out, indent)?;
    575 		if (member.docs != "" && !brief) {
    576 			z += docs_html(out, member.docs, indent)?;
    577 		};
    578 		match (member._offset) {
    579 		case null => void;
    580 		case let expr: *ast::expr =>
    581 			z += fmt::fprint(out, "@offset(")?;
    582 			z += unparse::expr(out, indent, *expr)?;
    583 			z += fmt::fprint(out, ") ")?;
    584 		};
    585 
    586 		match (member.member) {
    587 		case let f: ast::struct_field =>
    588 			z += fmt::fprintf(out, "{}: ", f.name)?;
    589 			z += type_html(out, indent, *f._type, brief)?;
    590 		case let embed: ast::struct_embedded =>
    591 			z += type_html(out, indent, *embed, brief)?;
    592 		case let indent: ast::struct_alias =>
    593 			z += unparse::ident(out, indent)?;
    594 		};
    595 		z += fmt::fprint(out, ",")?;
    596 	};
    597 
    598 	indent -= 1;
    599 	z += newline(out, indent)?;
    600 	z += fmt::fprint(out, "}")?;
    601 
    602 	return z;
    603 };
    604 
    605 fn type_html(
    606 	out: io::handle,
    607 	indent: size,
    608 	_type: ast::_type,
    609 	brief: bool,
    610 ) (size | io::error) = {
    611 	if (brief) {
    612 		let buf = strio::dynamic();
    613 		defer io::close(&buf)!;
    614 		unparse::_type(&buf, indent, _type)?;
    615 		return html_escape(out, strio::string(&buf))?;
    616 	};
    617 
    618 	// TODO: More detailed formatter which can find aliases nested deeper in
    619 	// other types and highlight more keywords, like const
    620 	let z = 0z;
    621 
    622 	if (_type.flags & ast::type_flags::CONST != 0
    623 			&& !(_type.repr is ast::func_type)) {
    624 		z += fmt::fprint(out, "<span class='keyword'>const</span> ")?;
    625 	};
    626 
    627 	if (_type.flags & ast::type_flags::ERROR != 0) {
    628 		if (_type.repr is ast::builtin_type) {
    629 			z += fmt::fprint(out, "<span class='type'>!</span>")?;
    630 		} else {
    631 			z += fmt::fprint(out, "!")?;
    632 		};
    633 	};
    634 
    635 	match (_type.repr) {
    636 	case let a: ast::alias_type =>
    637 		if (a.unwrap) {
    638 			z += fmt::fprint(out, "...")?;
    639 		};
    640 		z += unparse::ident(out, a.ident)?;
    641 	case let t: ast::builtin_type =>
    642 		z += fmt::fprintf(out, "<span class='type'>{}</span>",
    643 			unparse::builtin_type(t))?;
    644 	case let t: ast::tagged_type =>
    645 		// rough estimate of current line length
    646 		let linelen: size = z + (indent + 1) * 8;
    647 		z = 0;
    648 		linelen += fmt::fprint(out, "(")?;
    649 		for (let i = 0z; i < len(t); i += 1) {
    650 			linelen += type_html(out, indent, *t[i], brief)?;
    651 			if (i + 1 == len(t)) break;
    652 			linelen += fmt::fprint(out, " |")?;
    653 			// use 72 instead of 80 to give a bit of leeway for long
    654 			// type names
    655 			if (linelen > 72) {
    656 				z += linelen;
    657 				linelen = (indent + 1) * 8;
    658 				z += fmt::fprintln(out)?;
    659 				for (let i = 0z; i < indent; i += 1) {
    660 					z += fmt::fprint(out, "\t")?;
    661 				};
    662 			} else {
    663 				linelen += fmt::fprint(out, " ")?;
    664 			};
    665 		};
    666 		z += linelen;
    667 		z += fmt::fprint(out, ")")?;
    668 	case let t: ast::tuple_type =>
    669 		// rough estimate of current line length
    670 		let linelen: size = z + (indent + 1) * 8;
    671 		z = 0;
    672 		linelen += fmt::fprint(out, "(")?;
    673 		for (let i = 0z; i < len(t); i += 1) {
    674 			linelen += type_html(out, indent, *t[i], brief)?;
    675 			if (i + 1 == len(t)) break;
    676 			linelen += fmt::fprint(out, ",")?;
    677 			// use 72 instead of 80 to give a bit of leeway for long
    678 			// type names
    679 			if (linelen > 72) {
    680 				z += linelen;
    681 				linelen = (indent + 1) * 8;
    682 				z += fmt::fprintln(out)?;
    683 				for (let i = 0z; i < indent; i += 1) {
    684 					z += fmt::fprint(out, "\t")?;
    685 				};
    686 			} else {
    687 				linelen += fmt::fprint(out, " ")?;
    688 			};
    689 		};
    690 		z += linelen;
    691 		z += fmt::fprint(out, ")")?;
    692 	case let t: ast::pointer_type =>
    693 		if (t.flags & ast::pointer_flags::NULLABLE != 0) {
    694 			z += fmt::fprint(out, "<span class='type'>nullable</span> ")?;
    695 		};
    696 		z += fmt::fprint(out, "*")?;
    697 		z += type_html(out, indent, *t.referent, brief)?;
    698 	case let t: ast::func_type =>
    699 		if (t.attrs & ast::func_attrs::NORETURN == ast::func_attrs::NORETURN) {
    700 			z += fmt::fprint(out, "@noreturn ")?;
    701 		};
    702 
    703 		z += fmt::fprint(out, "<span class='keyword'>fn</span>(")?;
    704 		for (let i = 0z; i < len(t.params); i += 1) {
    705 			const param = t.params[i];
    706 			z += fmt::fprintf(out, "{}: ",
    707 				if (len(param.name) == 0) "_" else param.name)?;
    708 			z += type_html(out, indent, *param._type, brief)?;
    709 
    710 			if (i + 1 == len(t.params)
    711 					&& t.variadism == ast::variadism::HARE) {
    712 				// TODO: Highlight that as well
    713 				z += fmt::fprint(out, "...")?;
    714 			};
    715 			if (i + 1 < len(t.params)) {
    716 				z += fmt::fprint(out, ", ")?;
    717 			};
    718 		};
    719 		if (t.variadism == ast::variadism::C) {
    720 			z += fmt::fprint(out, ", ...")?;
    721 		};
    722 		z += fmt::fprint(out, ") ")?;
    723 		z += type_html(out, indent, *t.result, brief)?;
    724 	case let t: ast::enum_type =>
    725 		z += enum_html(out, indent, t)?;
    726 	case let t: ast::list_type =>
    727 		z += fmt::fprint(out, "[")?;
    728 		match (t.length) {
    729 		case let expr: *ast::expr =>
    730 			z += unparse::expr(out, indent, *expr)?;
    731 		case ast::len_slice =>
    732 			z += 0;
    733 		case ast::len_unbounded =>
    734 			z += fmt::fprintf(out, "*")?;
    735 		case ast::len_contextual =>
    736 			z += fmt::fprintf(out, "_")?;
    737 		};
    738 		z += fmt::fprint(out, "]")?;
    739 
    740 		z += type_html(out, indent, *t.members, brief)?;
    741 	case let t: ast::struct_type =>
    742 		z += struct_union_html(out, indent, _type, brief)?;
    743 	case let t: ast::union_type =>
    744 		z += struct_union_html(out, indent, _type, brief)?;
    745 	};
    746 
    747 	return z;
    748 };
    749 
    750 fn prototype_html(
    751 	out: io::handle,
    752 	indent: size,
    753 	t: ast::func_type,
    754 	brief: bool,
    755 ) (size | io::error) = {
    756 	let n = 0z;
    757 	n += fmt::fprint(out, "(")?;
    758 
    759 	// estimate length of prototype to determine if it should span multiple
    760 	// lines
    761 	const linelen = if (len(t.params) == 0 || brief) {
    762 		yield 0z; // If no parameters or brief, only use one line.
    763 	} else {
    764 		let linelen = indent * 8 + 5;
    765 		linelen += if (len(t.params) != 0) len(t.params) * 3 - 1 else 0;
    766 		for (let i = 0z; i < len(t.params); i += 1) {
    767 			const param = t.params[i];
    768 			linelen += unparse::_type(io::empty, indent,
    769 				*param._type)?;
    770 			linelen += if (param.name == "") 1 else len(param.name);
    771 		};
    772 		switch (t.variadism) {
    773 		case variadism::NONE => void;
    774 		case variadism::HARE =>
    775 			linelen += 3;
    776 		case variadism::C =>
    777 			linelen += 5;
    778 		};
    779 		linelen += unparse::_type(io::empty, indent, *t.result)?;
    780 		yield linelen;
    781 	};
    782 
    783 	// use 72 instead of 80 to give a bit of leeway for preceding text
    784 	if (linelen > 72) {
    785 		indent += 1;
    786 		for (let i = 0z; i < len(t.params); i += 1) {
    787 			const param = t.params[i];
    788 			n += newline(out, indent)?;
    789 			n += fmt::fprintf(out, "{}: ",
    790 				if (param.name == "") "_" else param.name)?;
    791 			n += type_html(out, indent, *param._type, brief)?;
    792 			if (i + 1 == len(t.params)
    793 					&& t.variadism == variadism::HARE) {
    794 				n += fmt::fprint(out, "...")?;
    795 			} else {
    796 				n += fmt::fprint(out, ",")?;
    797 			};
    798 		};
    799 		if (t.variadism == variadism::C) {
    800 			n += newline(out, indent)?;
    801 			n += fmt::fprint(out, "...")?;
    802 		};
    803 		indent -= 1;
    804 		n += newline(out, indent)?;
    805 	} else for (let i = 0z; i < len(t.params); i += 1) {
    806 		const param = t.params[i];
    807 		if (!brief) {
    808 			n += fmt::fprintf(out, "{}: ",
    809 				if (param.name == "") "_" else param.name)?;
    810 		};
    811 		n += type_html(out, indent, *param._type, brief)?;
    812 		if (i + 1 == len(t.params)) {
    813 			switch (t.variadism) {
    814 			case variadism::NONE => void;
    815 			case variadism::HARE =>
    816 				n += fmt::fprint(out, "...")?;
    817 			case variadism::C =>
    818 				n += fmt::fprint(out, ", ...")?;
    819 			};
    820 		} else {
    821 			n += fmt::fprint(out, ", ")?;
    822 		};
    823 	};
    824 
    825 	n += fmt::fprint(out, ") ")?;
    826 	n += type_html(out, indent, *t.result, brief)?;
    827 	return n;
    828 };
    829 
    830 fn breadcrumb(ident: ast::ident) str = {
    831 	if (len(ident) == 0) {
    832 		return "";
    833 	};
    834 	let buf = strio::dynamic();
    835 	fmt::fprintf(&buf, "<a href='/'>stdlib</a> » ")!;
    836 	for (let i = 0z; i < len(ident) - 1; i += 1) {
    837 		let ipath = module::identpath(ident[..i+1]);
    838 		defer free(ipath);
    839 		fmt::fprintf(&buf, "<a href='/{}'>{}</a>::", ipath, ident[i])!;
    840 	};
    841 	fmt::fprint(&buf, ident[len(ident) - 1])!;
    842 	return strio::string(&buf);
    843 };
    844 
    845 fn head(ident: ast::ident) (void | error) = {
    846 	const id = unparse::identstr(ident);
    847 	defer free(id);
    848 
    849 	let breadcrumb = breadcrumb(ident);
    850 	defer free(breadcrumb);
    851 
    852 	const title =
    853 		if (len(id) == 0)
    854 			fmt::asprintf("Hare documentation")
    855 		else
    856 			fmt::asprintf("{} — Hare documentation", id);
    857 	defer free(title);
    858 
    859 	// TODO: Move bits to +embed?
    860 	fmt::printfln("<!doctype html>
    861 <html lang='en'>
    862 <meta charset='utf-8' />
    863 <meta name='viewport' content='width=device-width, initial-scale=1' />
    864 <title>{}</title>", title)?;
    865 	fmt::println("<style>
    866 body {
    867 	font-family: sans-serif;
    868 	line-height: 1.3;
    869 	margin: 0 auto;
    870 	padding: 0 1rem;
    871 }
    872 
    873 nav:not(#TableOfContents) {
    874 	max-width: calc(800px + 128px + 128px);
    875 	margin: 1rem auto 0;
    876 	display: grid;
    877 	grid-template-rows: auto auto 1fr;
    878 	grid-template-columns: auto 1fr;
    879 	grid-template-areas:
    880 		'logo header'
    881 		'logo nav'
    882 		'logo none';
    883 }
    884 
    885 nav:not(#TableOfContents) img {
    886 	grid-area: logo;
    887 }
    888 
    889 nav:not(#TableOfContents) h1 {
    890 	grid-area: header;
    891 	margin: 0;
    892 	padding: 0;
    893 }
    894 
    895 nav:not(#TableOfContents) ul {
    896 	grid-area: nav;
    897 	margin: 0.5rem 0 0 0;
    898 	padding: 0;
    899 	list-style: none;
    900 	display: flex;
    901 	flex-direction: row;
    902 	justify-content: left;
    903 	flex-wrap: wrap;
    904 }
    905 
    906 nav:not(#TableOfContents) li:not(:first-child) {
    907 	margin-left: 2rem;
    908 }
    909 
    910 #TableOfContents {
    911 	font-size: 1.1rem;
    912 }
    913 
    914 main {
    915 	padding: 0 128px;
    916 	max-width: 800px;
    917 	margin: 0 auto;
    918 
    919 }
    920 
    921 pre {
    922 	background-color: #eee;
    923 	padding: 0.25rem 1rem;
    924 	margin: 0 -1rem 1rem;
    925 	font-size: 1.2rem;
    926 	max-width: calc(100% + 1rem);
    927 	overflow-x: auto;
    928 }
    929 
    930 pre .keyword {
    931     color: #008;
    932 }
    933 
    934 pre .type {
    935 	color: #44F;
    936 }
    937 
    938 ol {
    939 	padding-left: 0;
    940 	list-style: none;
    941 }
    942 
    943 ol li {
    944 	padding-left: 0;
    945 }
    946 
    947 h2, h3, h4 {
    948 	display: flex;
    949 }
    950 
    951 h3 {
    952 	border-bottom: 1px solid #ccc;
    953 	padding-bottom: 0.25rem;
    954 }
    955 
    956 .invalid {
    957 	color: red;
    958 }
    959 
    960 .heading-extra {
    961 	align-self: flex-end;
    962 	flex-grow: 1;
    963 	text-align: right;
    964 	font-size: 0.8rem;
    965 	color: #444;
    966 }
    967 
    968 h4:target + pre {
    969 	background: #ddf;
    970 }
    971 
    972 details {
    973 	background: #eee;
    974 	margin: 1rem -1rem 1rem;
    975 }
    976 
    977 summary {
    978 	cursor: pointer;
    979 	padding: 0.5rem 1rem;
    980 }
    981 
    982 details pre {
    983 	margin: 0;
    984 }
    985 
    986 .comment {
    987 	color: #000;
    988 	font-weight: bold;
    989 }
    990 
    991 @media(max-width: 1000px) {
    992 	main {
    993 		padding: 0;
    994 	}
    995 }
    996 
    997 @media(prefers-color-scheme: dark) {
    998 	body {
    999 		background: #121415;
   1000 		color: #e1dfdc;
   1001 	}
   1002 
   1003 	img.mascot {
   1004 		filter: invert(.92);
   1005 	}
   1006 
   1007 	a {
   1008 		color: #78bef8;
   1009 	}
   1010 
   1011 	a:visited {
   1012 		color: #48a7f5;
   1013 	}
   1014 
   1015 	summary {
   1016 		background: #16191c;
   1017 	}
   1018 
   1019 	h3 {
   1020 		border-bottom: solid #16191c;
   1021 	}
   1022 
   1023 	h4:target + pre {
   1024 		background: #162329;
   1025 	}
   1026 
   1027 	pre {
   1028 		background-color: #16191c;
   1029 	}
   1030 
   1031 	pre .keyword {
   1032 		color: #69f;
   1033 	}
   1034 
   1035 	pre .type {
   1036 		color: #3cf;
   1037 	}
   1038 
   1039 	.comment {
   1040 		color: #fff;
   1041 	}
   1042 
   1043 	.heading-extra {
   1044 		color: #9b9997;
   1045 	}
   1046 }
   1047 </style>
   1048 <nav>
   1049 	<img
   1050 		src=''
   1051 		class='mascot'
   1052 		alt='An inked drawing of the Hare mascot, a fuzzy rabbit'
   1053 		width='128' height='128' />
   1054 	<h1>Hare documentation</h1>
   1055 	<ul>
   1056 		<li>
   1057 			<a href='https://harelang.org'>Home</a>
   1058 		</li>")?;
   1059 	fmt::printf("<li>{}</li>", breadcrumb)?;
   1060 	fmt::print("</ul>
   1061 </nav>
   1062 <main>")?;
   1063 	return;
   1064 };