From fae75072f4503467ff450737c65fe26b9ee12cb1 Mon Sep 17 00:00:00 2001 From: Jakub Doka Date: Sat, 14 Dec 2024 13:17:58 +0100 Subject: [PATCH] removing hardcoded html files and replacing them with markdown the markdown gets transpiled on build and built files are then included in the server executable Signed-off-by: Jakub Doka --- .gitignore | 1 + Cargo.lock | 16 ++++ Cargo.toml | 2 - depell/src/index.css | 6 ++ depell/src/index.js | 5 +- depell/src/main.rs | 90 +++++++++++++++++--- depell/src/post-page.html | 21 ----- depell/src/profile-page.html | 0 depell/src/static-pages/developing-hblang.md | 31 +++++++ depell/src/static-pages/post.md | 8 ++ depell/src/static-pages/welcome.md | 11 +++ depell/src/welcome-page.html | 17 ---- foo.hb | 15 ---- lang/src/son.rs | 3 +- offense.hb | 0 smh.hb | 10 --- xtask/Cargo.toml | 1 + xtask/src/main.rs | 40 ++++++++- 18 files changed, 195 insertions(+), 82 deletions(-) delete mode 100644 depell/src/post-page.html delete mode 100644 depell/src/profile-page.html create mode 100644 depell/src/static-pages/developing-hblang.md create mode 100644 depell/src/static-pages/post.md create mode 100644 depell/src/static-pages/welcome.md delete mode 100644 depell/src/welcome-page.html delete mode 100644 foo.hb delete mode 100644 offense.hb delete mode 100644 smh.hb diff --git a/.gitignore b/.gitignore index f53b832d5..0fa010816 100644 --- a/.gitignore +++ b/.gitignore @@ -9,5 +9,6 @@ db.sqlite-journal # assets /depell/src/*.gz /depell/src/*.wasm +/depell/src/static-pages/*.html #**/*-sv.rs /bytecode/src/instrs.rs diff --git a/Cargo.lock b/Cargo.lock index 59b7f1cd5..29754c39b 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -844,6 +844,15 @@ version = "0.4.22" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "a7a70ba024b9dc04c27ea2f0c0548feb474ec5c54bba33a7f72f873a39d07b24" +[[package]] +name = "markdown" +version = "1.0.0-alpha.21" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a6491e6c702bf7e3b24e769d800746d5f2c06a6c6a2db7992612e0f429029e81" +dependencies = [ + "unicode-id", +] + [[package]] name = "matchit" version = "0.7.3" @@ -1446,6 +1455,12 @@ version = "1.17.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "42ff0bf0c66b8238c6f3b578df37d0b7848e55df8577b3f74f92a69acceeb825" +[[package]] +name = "unicode-id" +version = "0.3.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "10103c57044730945224467c09f71a4db0071c123a0648cc3e818913bde6b561" + [[package]] name = "unicode-ident" version = "1.0.13" @@ -1648,6 +1663,7 @@ name = "xtask" version = "0.1.0" dependencies = [ "anyhow", + "markdown", "walrus", ] diff --git a/Cargo.toml b/Cargo.toml index d4c33a3bc..48734ae9f 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -17,9 +17,7 @@ members = [ [workspace.dependencies] hbbytecode = { path = "bytecode", default-features = false } hbvm = { path = "vm", default-features = false } -hbxrt = { path = "xrt" } hblang = { path = "lang", default-features = false } -hbjit = { path = "jit" } [profile.release] lto = true diff --git a/depell/src/index.css b/depell/src/index.css index 3e19d6b54..1a6e97afc 100644 --- a/depell/src/index.css +++ b/depell/src/index.css @@ -1,5 +1,6 @@ * { font-family: var(--font); + line-height: 1.3; } body { @@ -135,6 +136,11 @@ button:hover:not(:active) { background: var(--primary); } +code { + font-family: var(--monospace); + line-height: 1; +} + div#code-editor { display: flex; position: relative; diff --git a/depell/src/index.js b/depell/src/index.js index 46e47321e..8e7a77979 100644 --- a/depell/src/index.js +++ b/depell/src/index.js @@ -433,9 +433,12 @@ function cacheInputs(target) { /** @param {string} [path] */ function updateTab(path) { + console.log(path); for (const elem of document.querySelectorAll("button[hx-push-url]")) { if (elem instanceof HTMLButtonElement) - elem.disabled = elem.getAttribute("hx-push-url") === (path ?? window.location.pathname); + elem.disabled = + elem.getAttribute("hx-push-url") === path + || elem.getAttribute("hx-push-url") === window.location.pathname; } } diff --git a/depell/src/main.rs b/depell/src/main.rs index 7de08ccde..eaf60110b 100644 --- a/depell/src/main.rs +++ b/depell/src/main.rs @@ -1,4 +1,4 @@ -#![feature(iter_collect_into)] +#![feature(iter_collect_into, macro_metavar_expr)] use { argon2::{password_hash::SaltString, PasswordVerifier}, axum::{ @@ -52,12 +52,15 @@ async fn amain() { db::init(); let router = axum::Router::new() - .route("/", get(Index::page)) .route("/index.css", static_asset!("text/css", "index.css")) .route("/index.js", static_asset!("text/javascript", "index.js")) .route("/hbfmt.wasm", static_asset!("application/wasm", "hbfmt.wasm")) .route("/hbc.wasm", static_asset!("application/wasm", "hbc.wasm")) - .route("/index-view", get(Index::get)) + .route("/", get(Index::page)) + .route("/index-view", get(Index::get_with_blog)) + .route("/blogs/index-view", get(Index::get)) + .route("/blogs/developing-hblang", get(DevelopingHblang::page)) + .route("/blogs/developing-hblang-view", get(DevelopingHblang::get)) .route("/feed", get(Feed::page)) .route("/feed-view", get(Feed::get)) .route("/feed-more", post(Feed::more)) @@ -199,12 +202,45 @@ impl Page for Feed { } } -#[derive(Default)] -struct Index; +macro_rules! decl_static_pages { + ($( + #[derive(PublicPage)] + #[page(static = $file:literal)] + struct $name:ident; + )*) => { + const ALL_STATIC_PAGES: [&str; ${count($file)}] = [$($file),*]; -impl PublicPage for Index { - fn render_to_buf(self, buf: &mut String) { - buf.push_str(include_str!("welcome-page.html")); + $( + #[derive(Default)] + struct $name; + + impl PublicPage for $name { + fn render_to_buf(self, buf: &mut String) { + buf.push_str(include_str!(concat!("static-pages/", $file, ".html"))); + } + + async fn page(session: Option) -> Html { + base(|s| blog_base(s, |s| Self::default().render_to_buf(s)), session.as_ref()) + } + } + )* + }; +} + +decl_static_pages! { + #[derive(PublicPage)] + #[page(static = "welcome")] + struct Index; + #[derive(PublicPage)] + #[page(static = "developing-hblang")] + struct DevelopingHblang; +} + +impl Index { + async fn get_with_blog() -> Html { + let mut buf = String::new(); + blog_base(&mut buf, |s| Index.render_to_buf(s)); + Html(buf) } } @@ -242,7 +278,17 @@ impl Page for Post {

             
-            !{include_str!("post-page.html")}
+
+            
+ +
+ "results show here..." +
+
+ +
+ !{include_str!("static-pages/post.html")} +
} } } @@ -468,7 +514,6 @@ impl Page for Profile { } else { "no posts" } - !{include_str!("profile-page.html")} } }) } @@ -631,6 +676,29 @@ impl Signup { } } +fn blog_base(s: &mut String, body: impl FnOnce(&mut String)) { + let nav_button = |f: &mut String, name: &str| { + write_html! {(f) + + } + }; + + write_html! {(*s) + +
+
|f|{body(f)}
+ } +} + fn base(body: impl FnOnce(&mut String), session: Option<&Session>) -> Html { let username = session.map(|s| &s.name); @@ -648,7 +716,7 @@ fn base(body: impl FnOnce(&mut String), session: Option<&Session>) -> Html - + "depell" diff --git a/depell/src/post-page.html b/depell/src/post-page.html deleted file mode 100644 index cf73e356d..000000000 --- a/depell/src/post-page.html +++ /dev/null @@ -1,21 +0,0 @@ -
- -
- results show here... -
-
- -
- -

About posting code

-

- If you are unfammiliar with hblang, refer to the - hblang/README.md or - vizit mlokis'es posts. Preferably don't edit the code here. -

- -

Extra textarea features

-
    -
  • proper tab behaviour
  • -
  • snap to previous tab boundary on "empty" lines
  • -
diff --git a/depell/src/profile-page.html b/depell/src/profile-page.html deleted file mode 100644 index e69de29bb..000000000 diff --git a/depell/src/static-pages/developing-hblang.md b/depell/src/static-pages/developing-hblang.md new file mode 100644 index 000000000..617a45c90 --- /dev/null +++ b/depell/src/static-pages/developing-hblang.md @@ -0,0 +1,31 @@ +# The journey to optimizing compiler + +It's been years since I was continuously trying to make a compiler to implement language of my dreams. Problem was tho that I wanted something similar to Rust, which if you did not know, `rustc` far exceeded the one million lines of code mark some time ago, so implementing such language would take me years if not decades, but I still tired it. + +Besides being extremely ambitions, the problem with my earliest attempts at making a compiler, is that literally nobody, not even me, was using the language, and so retroactively I am confident, what I implemented was a complex test-case implementation, and not a compiler. I often fall into a trap of implementing edge cases instead of an algorithm that would handle not only the very few thing the tests do but also all the other stuff that users of the language would try. + +Another part of why I was failing for all that time, is that I did the hardest thing first without understanding the core concepts involved in translating written language to IR, god forbid assembly. I wasted a lot of time like this, but at least I learned Rust well. At some point I found a job where I started developing a decentralized network and that fully drawn me away from language development. + +## Completely new approach + +At some point the company I was working for started having financial issues and they were unable to pay me. During that period, I discovered that my love for networking was majorly fueled by the monetary gains associated with it. I burned out, and started to look for things to do with the free time. + +One could say timing was perfect because [`ableos`](https://git.ablecorp.us/AbleOS/ableos) was desperately in need of a sane programming language that compiles to the home made VM ISA used for all software ran in `ableos`, but there was nobody crazy enough to do this. I got terribly nerd sniped, tho I don't regret it. Process of making a language for `ableos` was completely different. Firstly, it needed to be done asap, the lack of a good language blocked everyone form writing drivers for `ableos`, secondly, the moment the language is at least a little bit usable, people other then me will start using it, and lastly, the ISA the language compiles to very simple to emit, understand, and run. + +### Urgency is a bliss + +I actually managed to make the language somewhat work in one week, mainly because my mind set changed. I no longer spent a lot of time designing syntax for elegance, I designed it so that it incredibly easy to parse, meaning I can spent minimal effort implementing the parser, and fully focus on the hard problem of translating AST to instructions. Surprisingly, making everything an expression and not enforcing any arbitrary rules, makes the code you can write incredibly flexible and (most) people love it. One of the decisions I made to save time (or maybe it was an accident) was to make `,;` not enforced, meaning, you are allowed to write delimiters in lists but, as long as it does not change the intent of the code, you can leave them out. In practice, you actually don't need semicolons, unless the next line starts with something sticky like `*x`, int that case you put a semicolon on the previous line to tell the parser where the current expression ends. + +### Only the problem I care about + +Its good to note that writing a parser is no longer interesting for me. I wrote many parsers before and writing one no longer feel rewarding, but more like a chore. The real problem I was excited about was translating AST to instructions, I always ended up overcomplicating this step wit edge cases for every possible scenario that can happen in code, for which there are infinite. But why did I succeed this time? Well all the friction related to getting something that I can execute was so low, I could iterate quickly and realize what I am doing wrong before I burn out. In a week I managed to understand what I was failing to do for years, partly because of all the previous suffering, but mainly because it was so easy to pivot and try new things. And so I managed to make my first single pass compiler, and people immediately started using it. + +### Don't implement features nobody asked for + +Immediately after someone else then me wrote something in `hb` stuff started breaking, over the course of a month I kept fixing bugs and adding new features just fine, and more people started to use the language. All was good and well until I looked into the code. It was incredibly cursed, full of tricks to work around the compiler not doing any optimizations. At that moment I realized the whole compiler after parser needs to be rewritten, I had to implement optimizations, otherwise people wont be able to write readable code that runs fast. All of the features I have added up until now, were a technical dept now. Unless they are all working with optimizations, can't compile the existing code. Yes, if feature exists, be sure as hell it will be used. + +It took around 4 months to reimplement everything make make the optimal code look like what you are used to in other languages. I am really thankful for [sea of nodes](https://github.com/SeaOfNodes), and all the amazing work Cliff Click and others do to make demystify optimizers, It would have taken much longer to for me to figure all the principles out without the exhaustive [tutorial](https://github.com/SeaOfNodes/Simple?tab=readme-ov-file). + +## How my understanding of optimizations changed + +I need to admit, before writing a single pass compiler and later upgrading it to optimizing one, I took optimizations as some magic that makes code faster and honestly believed they are optional and most of the hard work is done in the process of translating readable text to the machine code. That is almost true with minus the readable part. If you want the code you write to perform well, with a compiler that translates your code from text to instructions as its written, you will be forced to do everything modern optimizers do, by hand in your code. TODO... diff --git a/depell/src/static-pages/post.md b/depell/src/static-pages/post.md new file mode 100644 index 000000000..55c8ff5a6 --- /dev/null +++ b/depell/src/static-pages/post.md @@ -0,0 +1,8 @@ +### About posting code + +If you are unfammiliar with [hblang](https://git.ablecorp.us/AbleOS/holey-bytes), refer to the **hblang/README.md** or vizit [mlokis'es posts](/profile/mlokis). Preferably don't edit the code here. + +### Extra textarea features + +- proper tab behaviour +- snap to previous tab boundary on "empty" lines diff --git a/depell/src/static-pages/welcome.md b/depell/src/static-pages/welcome.md new file mode 100644 index 000000000..cbcd73268 --- /dev/null +++ b/depell/src/static-pages/welcome.md @@ -0,0 +1,11 @@ +## Welcome to depell + +Depell (dependency hell) is a simple "social" media site, except that all you can post is [hblang](https://git.ablecorp.us/AbleOS/holey-bytes) code. Instead of likes you run the program, and instead of mentions you import the program as dependency. Run counts even when ran indirectly. + +The backend only serves the code and frontend compiles and runs it locally. All posts are immutable. + +## Security? + +All code runs in WASM (inside a holey-bytes VM until hblang compiles to wasm) and is controlled by JavaScript. WASM +cant do any form of IO without going trough JavaScript so as long as JS import does not allow wasm to execute +arbitrary JS code, WASM can act as a container inside the JS. diff --git a/depell/src/welcome-page.html b/depell/src/welcome-page.html deleted file mode 100644 index 2f3908fbf..000000000 --- a/depell/src/welcome-page.html +++ /dev/null @@ -1,17 +0,0 @@ -

Welcome to depell

-

- Depell (dependency hell) is a simple "social" media site best compared to twitter, except that all you can post is - hblang code with no comments allowed. Instead of likes you - run the program, and instead of retweets you import the program as dependency. Run counts even when ran indirectly. -

- -

- The backend only serves the code and frontend compiles and runs it locally. All posts are immutable. -

- -

Security?

-

- All code runs in WASM (inside a holey-bytes VM until hblang compiles to wasm) and is controlled by JavaScript. WASM - cant do any form of IO without going trough JavaScript so as long as JS import does not allow wasm to execute - arbitrary JS code, WASM can act as a container inside the JS. -

diff --git a/foo.hb b/foo.hb deleted file mode 100644 index 8047a5bfd..000000000 --- a/foo.hb +++ /dev/null @@ -1,15 +0,0 @@ - -Res := fn($O: type, $E: type): type return union(enum) { - ok: O, - err: E, -} - -main := fn(): uint { - r := do_something() - if r == .err return v.err - return v.ok -} - -do_something := fn(): Res(uint, uint) { - return .{ok: 0} -} diff --git a/lang/src/son.rs b/lang/src/son.rs index 8199c0eed..c38679eba 100644 --- a/lang/src/son.rs +++ b/lang/src/son.rs @@ -31,7 +31,6 @@ use { }, hashbrown::hash_map, hbbytecode::DisasmError, - std::panic, }; pub const VOID: Nid = 0; @@ -2978,7 +2977,7 @@ impl<'a> Codegen<'a> { Expr::Comment { .. } => Some(Value::VOID), Expr::String { pos, literal } => { let literal = &literal[1..literal.len() - 1]; - let mut data = std::mem::take(&mut self.pool.lit_buf); + let mut data = core::mem::take(&mut self.pool.lit_buf); debug_assert!(data.is_empty()); let report = |bytes: &core::str::Bytes, message: &str| { diff --git a/offense.hb b/offense.hb deleted file mode 100644 index e69de29bb..000000000 diff --git a/smh.hb b/smh.hb deleted file mode 100644 index 86275bc58..000000000 --- a/smh.hb +++ /dev/null @@ -1,10 +0,0 @@ -main := fn(): int { - a := 10 - b := 20 - - if &a == &a { - return 10 - } else { - return 20 - } -} diff --git a/xtask/Cargo.toml b/xtask/Cargo.toml index 095818fff..b6f081bd5 100644 --- a/xtask/Cargo.toml +++ b/xtask/Cargo.toml @@ -5,5 +5,6 @@ edition = "2021" [dependencies] anyhow = "1.0.89" +markdown = "1.0.0-alpha.21" walrus = "0.22.0" diff --git a/xtask/src/main.rs b/xtask/src/main.rs index 6e733419a..f3d6b7432 100644 --- a/xtask/src/main.rs +++ b/xtask/src/main.rs @@ -4,13 +4,42 @@ use { crate::utils::IterExt, anyhow::Context, std::{ - fs::File, + fs::{self, File}, io::{self, BufRead, BufReader, BufWriter, Seek, Write}, - path::Path, + path::{Path, PathBuf}, }, walrus::{ir::Value, ConstExpr, GlobalKind, ValType}, }; +pub fn convert_md_to_html(dir: &str) -> anyhow::Result<()> { + let dir_path = Path::new(dir); + + if !dir_path.is_dir() { + anyhow::bail!("{} is not a valid directory", dir); + } + + for entry in fs::read_dir(dir_path)? { + let entry = entry?; + let path = entry.path(); + + if path.extension().is_none_or(|s| s != "md") { + continue; + } + + let markdown_content = fs::read_to_string(&path)?; + let html_content = + markdown::to_html_with_options(&markdown_content, &markdown::Options::gfm()) + .map_err(anyhow::Error::msg)?; + + let mut output_path = PathBuf::from(&path); + output_path.set_extension("html"); + + fs::write(output_path, html_content)?; + } + + Ok(()) +} + fn root() -> &'static Path { Path::new(env!("CARGO_MANIFEST_DIR")).parent().unwrap() } @@ -73,6 +102,7 @@ fn main() -> anyhow::Result<()> { build_wasm_blob("hbc", true)?; exec(build_cmd("gzip -k -f depell/src/index.js"))?; exec(build_cmd("gzip -k -f depell/src/index.css"))?; + convert_md_to_html("depell/src/static-pages")?; Ok(()) } "build-depell" => { @@ -80,6 +110,7 @@ fn main() -> anyhow::Result<()> { build_wasm_blob("hbc", false)?; exec(build_cmd("gzip -k -f depell/src/index.js"))?; exec(build_cmd("gzip -k -f depell/src/index.css"))?; + convert_md_to_html("depell/src/static-pages")?; Ok(()) } "watch-depell-debug" => { @@ -102,7 +133,10 @@ fn main() -> anyhow::Result<()> { ))?; Ok(()) } - _ => Ok(()), + unknown => { + anyhow::bail!("command {unknown:?} not found, availale commands:\ + fmt, build-depell-debug, build-depell, watch-depell-debug, watch-depell, release-depell"); + } } }