summaryrefslogtreecommitdiff
path: root/src/main.zig
blob: 335e80c63dfe584f927d63dc3fd50b26eb903c17 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
const std = @import("std");

// extremely basic http file server
pub fn main() !void {
  var allocator = std.heap.GeneralPurposeAllocator(.{}){};
  defer _ = allocator.deinit();
  const alloc = allocator.allocator();
  var svr = std.http.Server.init(alloc, .{.reuse_address = true});
  defer svr.deinit();
  const addr = try std.net.Address.resolveIp("127.0.0.1", 8080);
  
  try svr.listen(addr);
  while (true) {
    var res = try svr.accept(.{.allocator = alloc });
    defer res.deinit();
    try res.wait();
    const target = res.request.target;
    const path = try std.fs.path.join(alloc, &[_][]const u8{".", target});
    defer alloc.free(path);
    if (std.fs.cwd().openFile(path, .{})) |file| {
      const md = try file.metadata();
      if (md.kind() == .directory) {
        const index_path = try std.fs.path.join(alloc, &[_][]const u8{path, "index.html"});
        defer alloc.free(index_path);
        if (std.fs.cwd().openFile(index_path, .{})) |index_file| {
          const index_md = try index_file.metadata();
          try serve_file(&res, index_file, index_md);
        } else |_| {
          try serve_error(&res, .not_found);
        }
      } else {
        try serve_file(&res, file, md);
      }
    } else |err| {
      switch (err) {
        error.FileNotFound => try serve_error(&res, .not_found),
        else => try serve_error(&res, .bad_request),
      }
    }
    try res.finish();
  }
}

fn serve_file(res: *std.http.Server.Response, file: std.fs.File, md: std.fs.File.Metadata) !void {
  res.transfer_encoding = .{ .content_length = md.size() };
  try res.do();
  var buf = [_]u8{0} ** 1024;
  while (true) {
    const read = try file.read(&buf);
    if (read == 0) break;
    _ = try res.write(buf[0..read]);
  }
}

fn serve_error(res: *std.http.Server.Response, status: std.http.Status) !void {
  res.status = status;
  const msg = 
    \\ <!doctype html><html><body>error!</body></html>
    ;
  res.transfer_encoding = . { .content_length = msg.len };
   try res.do();
  _ = try res.write(msg);
}