const std = @import("std"); const kmd = @import("kmd.zig"); const Allocator = std.mem.Allocator; // Globals const stdout_file = std.io.getStdOut().writer(); var bw = std.io.bufferedWriter(stdout_file); const stdout = bw.writer(); var output: []const u8 = undefined; var input: []const u8 = undefined; var template: []const u8 = undefined; var debug = false; fn print(comptime fmt: []const u8, args: anytype) !void { try stdout.print(fmt, args); try bw.flush(); } fn usage() noreturn { print("TODO: add usage\n", .{}) catch { std.process.exit(1); }; std.process.exit(0); } const Jezup = struct { const Mode = enum { none, p, ul, ol, }; title: []const u8, date: []const u8, contents: []const u8, pub fn init(self: *Jezup, text: []const u8) void { var it = std.mem.splitAny(u8, text, "\n"); self.title = it.first(); self.date = it.next().?; self.contents = it.rest(); } pub fn calculateSizeOfContentsAsHtml(self: *Jezup) usize { var mode: Mode = .none; var it = std.mem.splitAny(u8, self.contents, "\n"); var line: []const u8 = it.first(); var additional: usize = 0; var handled = false; // if !handled and not empty, default is to handle as

while (true) { // handle

,

, etc. if (std.mem.startsWith(u8, line, "# ")) { handled = true; additional += 8; //

= 9 minus 2 from "# " } if (std.mem.startsWith(u8, line, "## ")) { handled = true; additional += 7; } if (std.mem.startsWith(u8, line, "### ")) { handled = true; additional += 6; } if (std.mem.startsWith(u8, line, "- ")) { handled = true; additional += if (mode == .ul) 8 else 7; if (mode != .ul) mode = .ul; } if (std.mem.startsWith(u8, line, "~ ")) { handled = true; additional += if (mode == .ol) 8 else 7; if (mode != .ol) mode = .ol; } if (mode == .none and !handled and !std.mem.eql(u8, line, "")) { mode = .p; additional += 4; //

\n } // handle // @@[][] becomes -> 6 bytes becomes 15 -> 9 add bytes additional += std.mem.count(u8, line, "@@[") * 9; // handle > and < - they become > and < so +3 bytes each additional += std.mem.count(u8, line, "<") * 3; additional += std.mem.count(u8, line, ">") * 3; // peek next and if empty, that means \n\n chain - cancel out cond. if (it.peek() == null or std.mem.eql(u8, it.peek().?, "")) { switch (mode) { .p => { additional += 6; // \n

\n }, // add

\n .ul => { additional += 11; }, // add \n (11 bytes) .ol => { additional += 11; }, // add .none => {}, // do nothing } mode = .none; break; } else { line = it.next().?; } } // handle
additional += std.mem.count(u8, self.contents, ";;") * 2; //
(4) - ;; (2) -> 2 add chars return additional + self.contents.len; } fn append(buf: *[256]u8, pos: *usize, data: []const u8) void { std.mem.copyForwards(u8, buf[pos.*..], data); pos.* += data.len; } pub fn contentsAsHtml(self: *Jezup, allocator: Allocator, required_capacity_param: ?usize) ![]const u8 { var mode: Mode = .none; var it = std.mem.splitAny(u8, self.contents, "\n"); var line: []const u8 = it.first(); const required_capacity = if (required_capacity_param) |req| req else calculateSizeOfContentsAsHtml(self); var handled = false; // if !handled and not empty, default is to handle as

// allocate the new buffer - text + additional_bytes_req // this will hold the full output var buf: []u8 = try allocator.alloc(u8, required_capacity - 1); var pos: usize = 0; // offset to memcpy to // hold the buffer for the entire line var linebuf: [256]u8 = undefined; var linepos: usize = 0; std.debug.print("allocated_capacity: {}\n", .{required_capacity - 1}); std.debug.print("The length of buf is: {}\n", .{buf.len}); std.debug.print("buf = {s};\nbuf[pos..] = {s};\n", .{ buf, buf[pos..] }); // Part two - actually run the changes { // TODO: handle em, b, br, ul, ol, p, <, > // NOTE: handling hN a while (true) { // reset @memset(&linebuf, 0); linepos = 0; std.debug.print("While iteration with line: {s}\n", .{line}); // handle

,

, etc. // these are "interrupting handlers" which disrupt the normal flow of

// included are: hN, ul, ol. if (std.mem.startsWith(u8, line, "# ")) { handled = true; append(&linebuf, &linepos, "

"); append(&linebuf, &linepos, line[2..]); append(&linebuf, &linepos, "

\n"); } if (std.mem.startsWith(u8, line, "## ")) { handled = true; append(&linebuf, &linepos, "

"); append(&linebuf, &linepos, line[3..]); append(&linebuf, &linepos, "

\n"); } if (std.mem.startsWith(u8, line, "### ")) { handled = true; append(&linebuf, &linepos, "

"); append(&linebuf, &linepos, line[4..]); append(&linebuf, &linepos, "

\n"); } if (std.mem.startsWith(u8, line, "- ") or std.mem.startsWith(u8, line, "~ ")) { handled = true; if (mode != .ul and mode != .ol) { mode = if (line[0] == '-') .ul else .ol; append(&linebuf, &linepos, if (line[0] == '-') ""), .ol => append(&linebuf, &linepos, "\n"), .none => {}, // do nothing } mode = .none; } else {} // copy linebuf into buf std.debug.print("copied linebuf = {s}; buf.len = {}; pos = {}; linebuf.len = {}; linepos = {}\n", .{ linebuf, buf.len, pos, linebuf.len, linepos }); std.mem.copyForwards(u8, buf[pos..], linebuf[0..linepos]); pos += linepos; // linebuf.len is always 256 line = it.next().?; handled = false; std.debug.print("---- CLOSE LINE ----\n", .{}); if (it.peek() == null) { std.debug.print("---- QUITTING ----\n", .{}); break; } } } return buf; } }; test "jezup.contentsToHtml" { var gpa = std.heap.GeneralPurposeAllocator(.{}){}; const allocator = gpa.allocator(); var jz1: Jezup = undefined; const str = "Title\n2024-10-01\n# My header\nThis line has @@[a link][https://gabbott.dev] in it\n\nThis line has @@[two][https://kagi.com] and @@[three][https://boodle.mon] awesome links."; const out = "

My header

\nThis line has a link in it\n\nThis line has two and three awesome links."; jz1.init(str); const html = try jz1.contentsAsHtml(allocator, null); defer allocator.free(html); std.debug.print("HTML\n----\n{s}\n| len = {}\n", .{ html, html.len }); try std.testing.expect(std.mem.eql(u8, html, out)); } pub fn main() !void { var gpa = std.heap.GeneralPurposeAllocator(.{}){}; const allocator = gpa.allocator(); const args = try std.process.argsAlloc(allocator); var i: usize = 1; while (i < args.len) : (i += 1) { if (std.mem.eql(u8, args[i], "-o")) { i += 1; output = args[i]; } else if (std.mem.eql(u8, args[i], "-i")) { i += 1; input = args[i]; } else if (std.mem.eql(u8, args[i], "-t")) { i += 1; template = args[i]; } else if (std.mem.eql(u8, args[i], "-d")) { debug = true; } else if (std.mem.eql(u8, args[i], "-h")) { usage(); } else { try print("not a valid arg\n", .{}); } } }