const UserListList = db.List(SavedUserList);
fn parse_enum(comptime E: type, buf: []const u8, base: u8) !E {
- return @enumFromInt(try std.fmt.parseUnsigned(@typeInfo(E).Enum.tag_type, buf, base));
+ return @enumFromInt(try std.fmt.parseUnsigned(@typeInfo(E).@"enum".tag_type, buf, base));
}
// https://developer.mozilla.org/en-US/docs/Glossary/Percent-encoding
const Chirp = struct {
const PostsPerPage = 10;
const UsersPerPage = 10;
+ var HashBuffer = std.mem.zeroes([1024 * 1024 * 50]u8);
pub fn hash_password(password: []const u8) !PasswordHash {
var hash_buffer = try PasswordHash.init(128);
// TODO: choose buffer size
- // TODO: dont allocate on stack, maybe zero memory?
- var buffer: [1024 * 10]u8 = undefined;
- var alloc = std.heap.FixedBufferAllocator.init(&buffer);
+ var alloc = std.heap.FixedBufferAllocator.init(&HashBuffer);
// TODO: choose limits
const result = try std.crypto.pwhash.argon2.strHash(password, .{
.allocator = alloc.allocator(),
- .params = std.crypto.pwhash.argon2.Params.fromLimits(1000, 1024),
+ .params = std.crypto.pwhash.argon2.Params.owasp_2id,
}, hash_buffer.slice());
try hash_buffer.resize(result.len);
}
pub fn verify_password(password: []const u8, hash: PasswordHash) bool {
- var buffer: [1024 * 10]u8 = undefined;
- var alloc = std.heap.FixedBufferAllocator.init(&buffer);
+ var alloc = std.heap.FixedBufferAllocator.init(&HashBuffer);
if (std.crypto.pwhash.argon2.strVerify(hash.constSlice(), password, .{
.allocator = alloc.allocator(),
.starting_idx = it.idx,
};
}
- pub fn next(self: *Self) IterateResult {
+ pub fn next(self: *Self) ?IterateResult {
if (self.it.next()) |kv| {
if (self.count < self.per_page) {
self.count += 1;
inline for (inputs) |input| {
switch (@typeInfo(@TypeOf(input))) {
- .Struct => {
- try res.write("<input ", .{});
- try res.write(input[0], input[1]);
- try res.write(" />", .{});
+ .@"struct" => |s| {
+ if (s.fields.len == 3) {
+ try res.write("<{s} ", .{input[0]});
+ try res.write(input[1], input[2]);
+ try res.write("></{s}>", .{input[0]});
+ } else {
+ try res.write("<input ", .{});
+ try res.write(input[0], input[1]);
+ try res.write(" />", .{});
+ }
},
else => {
try res.write("<input ", .{});
\\ form {
\\ display: inline-block;
\\ }
+ \\ body {
+ \\ margin:40px auto;max-width:650px;line-height:1.6;font-size:18px;color:#444;padding:0 10px;
+ \\ }
+ \\ h1,h2,h3{line-height:1.2}
\\</style>
\\</head>
\\<body>
try res.write("<option value=\"{x}\">{s}{s}</option>", .{ id, name.constSlice(), if (list_view.has(post_id) catch false) " *" else "" });
}
try res.write("</select>", .{});
- try res.write("<input type=\"hidden\" name=\"post_id\" value=\"{x}\"></input>", .{@intFromEnum(post_id)});
- try res.write("<input type=\"submit\" value=\"Save\"></input>", .{});
+ try res.write("<input type=\"hidden\" name=\"post_id\" value=\"{x}\" />", .{@intFromEnum(post_id)});
+ try res.write("<input type=\"submit\" value=\"Save\" />", .{});
try res.write("</form>", .{});
}
try res.write("<br /><br />", .{});
try html_form(res, "/comment", .{
.{ "type=\"hidden\" value=\"{x}\" name=\"post_id\"", .{@intFromEnum(post.id)} },
- "type=\"text\" name=\"text\" placeholder=\"Text\"",
+ .{ "textarea", "type=\"text\" name=\"text\" placeholder=\"Text\"", .{} },
"type=\"submit\" value=\"Comment\"",
});
try res.write("<br />", .{});
try res.write("<option value=\"{x}\">{s}{s}</option>", .{ id, name.constSlice(), if (list_view.has(user.id) catch false) " *" else "" });
}
try res.write("</select>", .{});
- try res.write("<input type=\"hidden\" name=\"user_id\" value=\"{x}\"></input>", .{@intFromEnum(user.id)});
- try res.write("<input type=\"submit\" value=\"Add to feed\"></input>", .{});
+ try res.write("<input type=\"hidden\" name=\"user_id\" value=\"{x}\" />", .{@intFromEnum(user.id)});
+ try res.write("<input type=\"submit\" value=\"Add to feed\" />", .{});
try res.write("</form>", .{});
}
try res.write(
var prev_newest_post: ?Post = null;
const following = try user_list.open(txn);
+ if (following.len() == 0) {
+ try res.write("Empty timeline (no users)", .{});
+ return;
+ }
while (true) {
var newest_post: ?Post = null;
fn handle(self: Self) !bool {
const ti = @typeInfo(Self);
- inline for (ti.Struct.decls) |f_decl| {
+ inline for (ti.@"struct".decls) |f_decl| {
const has_arg = f_decl.name.len > 1 and f_decl.name[f_decl.name.len - 1] == '/';
const match = if (has_arg) std.mem.startsWith(u8, self.req.target, f_decl.name) else std.mem.eql(u8, self.req.target, f_decl.name);
if (match) {
const f = @field(Self, f_decl.name);
const fi = @typeInfo(@TypeOf(f));
- if (fi.Fn.params.len == 1) {
+ if (fi.@"fn".params.len == 1) {
try @call(.auto, f, .{self});
} else {
- const arg_type = fi.Fn.params[1].type.?;
+ const arg_type = fi.@"fn".params[1].type.?;
const arg_info = @typeInfo(arg_type);
var arg: arg_type = undefined;
- const field = arg_info.Struct.fields[0];
+ const field = arg_info.@"struct".fields[0];
if (self.req.target.len <= f_decl.name.len) {
return error.NoArgProvided;
}
const field_ti = @typeInfo(field.type);
switch (field_ti) {
// TODO: maybe handle BoundedArray?
- .Int => {
+ .int => {
@field(arg, field.name) = try std.fmt.parseUnsigned(field.type, str, 16);
},
- .Enum => {
+ .@"enum" => {
@field(arg, field.name) = try parse_enum(field.type, str, 16);
},
else => {
try html_form(self.res, "/quote", .{
.{ "type=\"hidden\" name=\"referer\" value=\"{s}\"", .{referer} },
.{ "type=\"hidden\" name=\"post_id\" value=\"{x}\"", .{@intFromEnum(post.id)} },
- "type=\"text\" name=\"text\" placeholder=\"Text\" autofocus",
+ .{ "textarea", "type=\"text\" name=\"text\" placeholder=\"Text\" autofocus", .{} },
"type=\"submit\" value=\"Quote\"",
});
try self.res.write("<br />", .{});
try html_form(self.res, "/post", .{
.{ "type=\"hidden\" name=\"referer\" value=\"{s}\"", .{referer} },
- "type=\"text\" name=\"text\" placeholder=\"Text\" autofocus",
+ .{ "textarea", "type=\"text\" name=\"text\" placeholder=\"Text\" autofocus", .{} },
"type=\"submit\" value=\"Post\"",
});
} else {
});
try self.res.write("<br />Description: ", .{});
try html_form(self.res, "/set_description", .{
- .{ "type=\"text\" name=\"description\" placeholder=\"{s}\"", .{login.user.description.constSlice()} },
+ .{ "textarea", "type=\"text\" name=\"description\" placeholder=\"{s}\"", .{login.user.description.constSlice()} },
"type=\"submit\" value=\"Change\"",
});
try self.res.write("<br />Password: ", .{});
pub fn handle(self: Self) !bool {
const ti = @typeInfo(Self);
- inline for (ti.Struct.decls) |f_decl| {
+ inline for (ti.@"struct".decls) |f_decl| {
if (std.mem.eql(u8, f_decl.name, self.req.target)) {
const f = @field(Self, f_decl.name);
const fi = @typeInfo(@TypeOf(f));
- if (fi.Fn.params.len == 1) {
+ if (fi.@"fn".params.len == 1) {
_ = try @call(.auto, f, .{self});
} else {
- const args_type = fi.Fn.params[fi.Fn.params.len - 1].type.?;
+ const args_type = fi.@"fn".params[fi.@"fn".params.len - 1].type.?;
const argsi = @typeInfo(args_type);
var args: args_type = undefined;
- inline for (argsi.Struct.fields) |field| {
+ inline for (argsi.@"struct".fields) |field| {
const str = self.req.get_value(field.name) orelse return error.ArgNotFound;
const field_ti = @typeInfo(field.type);
switch (field_ti) {
- .Int => {
+ .int => {
@field(args, field.name) = try std.fmt.parseUnsigned(field.type, str, 16);
},
- .Enum => {
+ .@"enum" => {
@field(args, field.name) = try parse_enum(field.type, str, 16);
},
else => {