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 "&"; 38 case '<' => 39 yield "<"; 40 case '>' => 41 yield ">"; 42 case '"' => 43 yield """; 44 case '\'' => 45 yield "'"; 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) == ""hello world!""); 64 65 let sink = strio::dynamic(); 66 defer io::close(&sink)!; 67 html_escape(&sink, "<hello & 'world'!>")!; 68 assert(strio::string(&sink) == "<hello & 'world'!>"); 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 };