1
Fork 0
turbonss/src/User.zig

200 lines
5.5 KiB
Zig
Raw Normal View History

const std = @import("std");
const mem = std.mem;
2022-04-10 21:58:43 +03:00
const fmt = std.fmt;
const maxInt = std.math.maxInt;
const Allocator = mem.Allocator;
2022-04-08 20:05:30 +03:00
const ArrayList = std.ArrayList;
2022-04-10 21:58:43 +03:00
const BoundedArray = std.BoundedArray;
2022-04-19 13:17:23 +03:00
const pw_passwd = "x\x00";
const User = @This();
2022-06-15 12:56:19 +03:00
const ErrCtx = @import("ErrCtx.zig");
2022-04-10 21:58:43 +03:00
const PackedUser = @import("PackedUser.zig");
const validate = @import("validate.zig");
uid: u32,
gid: u32,
name: []const u8,
gecos: []const u8,
home: []const u8,
shell: []const u8,
// deep-clones a User record with a given Allocator.
pub fn clone(
self: *const User,
allocator: Allocator,
) error{OutOfMemory}!User {
2022-06-15 12:56:19 +03:00
const name = try allocator.dupe(u8, self.name);
errdefer allocator.free(name);
const gecos = try allocator.dupe(u8, self.gecos);
errdefer allocator.free(gecos);
const home = try allocator.dupe(u8, self.home);
errdefer allocator.free(home);
const shell = try allocator.dupe(u8, self.shell);
errdefer allocator.free(shell);
return User{
.uid = self.uid,
.gid = self.gid,
2022-06-15 12:56:19 +03:00
.name = name,
.gecos = gecos,
.home = home,
.shell = shell,
};
}
2022-04-10 21:58:43 +03:00
// fromLine accepts a line of /etc/passwd (with or without the EOL) and makes a
// User.
2022-06-15 13:34:37 +03:00
fn fromLine(allocator: Allocator, err_ctx: *ErrCtx, line: []const u8) error{ InvalidRecord, OutOfMemory }!User {
2022-04-10 21:58:43 +03:00
var it = mem.split(u8, line, ":");
2022-06-15 13:34:37 +03:00
const name = it.next();
_ = it.next(); // password
const uids = it.next();
const gids = it.next();
const gecos = it.next();
const home = it.next();
const shell = it.next();
// all fields are set
if (shell == null) {
2022-06-22 20:59:58 +03:00
err_ctx.print("too few user fields in line: {s}", .{line});
2022-06-15 13:34:37 +03:00
return error.InvalidRecord;
}
2022-04-10 21:58:43 +03:00
// the line must be exhaustive.
2022-06-15 13:34:37 +03:00
if (it.next() != null)
return error.InvalidRecord;
2022-06-22 21:06:20 +03:00
const uid = fmt.parseInt(u32, uids.?, 10) catch
2022-06-22 20:59:58 +03:00
return err_ctx.returnf("bad uid: {s}", .{uids.?}, error.InvalidRecord);
2022-06-15 13:34:37 +03:00
2022-06-22 21:06:20 +03:00
const gid = fmt.parseInt(u32, gids.?, 10) catch
2022-06-22 20:59:58 +03:00
return err_ctx.returnf("bad uid: {s}", .{gids.?}, error.InvalidRecord);
2022-04-10 21:58:43 +03:00
2022-06-22 21:06:20 +03:00
validate.utf8(name.?) catch
return err_ctx.returnf("name is invalid utf8: {s}", .{name.?}, error.InvalidRecord);
validate.utf8(gecos.?) catch
return err_ctx.returnf("gecos is invalid utf8: {s}", .{gecos.?}, error.InvalidRecord);
validate.utf8(home.?) catch
return err_ctx.returnf("home is invalid utf8: {s}", .{home.?}, error.InvalidRecord);
validate.utf8(shell.?) catch
return err_ctx.returnf("shell is invalid utf8: {s}", .{shell.?}, error.InvalidRecord);
2022-04-10 21:58:43 +03:00
const user = User{
.uid = uid,
.gid = gid,
2022-06-15 13:34:37 +03:00
.name = name.?,
.gecos = gecos.?,
.home = home.?,
.shell = shell.?,
2022-04-10 21:58:43 +03:00
};
return try user.clone(allocator);
}
fn strlen(self: *const User) usize {
return self.name.len +
self.gecos.len +
self.home.len +
self.shell.len;
}
2022-06-07 05:32:21 +03:00
const line_fmt = "{s}:x:{d}:{d}:{s}:{s}:{s}\n";
2022-06-05 23:56:45 +03:00
const max_line_len = fmt.count(line_fmt, .{
max_user.name,
max_user.uid,
max_user.gid,
max_user.gecos,
max_user.home,
max_user.shell,
});
2022-06-07 13:14:50 +03:00
// toLine formats the user in /etc/passwd format, including the EOL
pub fn toLine(self: *const User) BoundedArray(u8, max_line_len) {
var result = BoundedArray(u8, max_line_len).init(max_line_len) catch unreachable;
_ = fmt.bufPrint(result.slice(), line_fmt, .{
self.name,
2022-06-05 23:56:45 +03:00
self.uid,
self.gid,
self.gecos,
self.home,
self.shell,
2022-06-07 13:14:50 +03:00
}) catch unreachable;
2022-06-05 23:56:45 +03:00
return result;
}
2022-04-18 13:11:20 +03:00
// length of all string-data fields, assuming they are zero-terminated.
// Does not include password, since that's always static 'x\00'.
2022-04-18 13:11:20 +03:00
pub fn strlenZ(self: *const User) usize {
return self.strlen() + 4; // '\0' of name, gecos, home and shell
2022-04-18 13:11:20 +03:00
}
pub fn deinit(self: *User, allocator: Allocator) void {
2022-06-12 13:38:10 +03:00
allocator.free(self.name);
allocator.free(self.gecos);
allocator.free(self.home);
allocator.free(self.shell);
self.* = undefined;
}
2022-06-15 13:34:37 +03:00
pub fn fromReader(allocator: Allocator, err_ctx: *ErrCtx, reader: anytype) ![]User {
2022-04-08 20:05:30 +03:00
var users = ArrayList(User).init(allocator);
2022-06-15 12:56:19 +03:00
errdefer {
for (users.items) |*user| user.deinit(allocator);
users.deinit();
}
2022-06-07 05:32:21 +03:00
var buf: [max_line_len + 1]u8 = undefined;
2022-06-05 23:56:45 +03:00
while (try reader.readUntilDelimiterOrEof(buf[0..], '\n')) |line| {
2022-06-15 12:56:19 +03:00
var user = try fromLine(allocator, err_ctx, line);
2022-06-05 23:56:45 +03:00
try users.append(user);
2022-04-08 20:05:30 +03:00
}
2022-06-05 23:56:45 +03:00
return users.toOwnedSlice();
2022-04-08 20:05:30 +03:00
}
2022-04-19 13:17:23 +03:00
pub const CUser = extern struct {
pw_name: [*]const u8,
pw_passwd: [*]const u8 = pw_passwd,
pw_uid: u32,
pw_gid: u32,
pw_gecos: [*]const u8,
pw_dir: [*]const u8,
pw_shell: [*]const u8,
};
const testing = std.testing;
2022-04-10 21:58:43 +03:00
pub const max_user = User{
.uid = maxInt(u32),
.gid = maxInt(u32),
.name = "Name" ** 8,
.gecos = "realname" ** 255 ++ "realnam",
.home = "Home" ** 16,
.shell = "She.LllL" ** 32,
};
test "max_user and max_str_len are consistent" {
const total_len = max_user.name.len +
max_user.gecos.len +
max_user.home.len +
max_user.shell.len;
try testing.expectEqual(total_len, PackedUser.max_str_len);
}
test "User.clone" {
var allocator = testing.allocator;
const user = User{
.uid = 1000,
.gid = 1000,
.name = "vidmantas",
.gecos = "Vidmantas Kaminskas",
.home = "/home/vidmantas",
.shell = "/bin/bash",
};
var user2 = try user.clone(allocator);
defer user2.deinit(allocator);
try testing.expectEqualStrings(user.shell, "/bin/bash");
}