mirror of https://github.com/fairyglade/ly.git
556 lines
21 KiB
Zig
556 lines
21 KiB
Zig
const std = @import("std");
|
|
const build_options = @import("build_options");
|
|
const builtin = @import("builtin");
|
|
const enums = @import("enums.zig");
|
|
const Environment = @import("Environment.zig");
|
|
const interop = @import("interop.zig");
|
|
const SharedError = @import("SharedError.zig");
|
|
|
|
const Allocator = std.mem.Allocator;
|
|
const Md5 = std.crypto.hash.Md5;
|
|
const utmp = interop.utmp;
|
|
const Utmp = utmp.utmpx;
|
|
|
|
pub const AuthOptions = struct {
|
|
tty: u8,
|
|
service_name: [:0]const u8,
|
|
path: ?[:0]const u8,
|
|
session_log: []const u8,
|
|
xauth_cmd: []const u8,
|
|
setup_cmd: []const u8,
|
|
login_cmd: ?[]const u8,
|
|
x_cmd: []const u8,
|
|
session_pid: std.posix.pid_t,
|
|
};
|
|
|
|
var xorg_pid: std.posix.pid_t = 0;
|
|
pub fn xorgSignalHandler(i: c_int) callconv(.C) void {
|
|
if (xorg_pid > 0) _ = std.c.kill(xorg_pid, i);
|
|
}
|
|
|
|
var child_pid: std.posix.pid_t = 0;
|
|
pub fn sessionSignalHandler(i: c_int) callconv(.C) void {
|
|
if (child_pid > 0) _ = std.c.kill(child_pid, i);
|
|
}
|
|
|
|
pub fn authenticate(options: AuthOptions, current_environment: Environment, login: [:0]const u8, password: [:0]const u8) !void {
|
|
var tty_buffer: [3]u8 = undefined;
|
|
const tty_str = try std.fmt.bufPrintZ(&tty_buffer, "{d}", .{options.tty});
|
|
|
|
var pam_tty_buffer: [6]u8 = undefined;
|
|
const pam_tty_str = try std.fmt.bufPrintZ(&pam_tty_buffer, "tty{d}", .{options.tty});
|
|
|
|
// Set the XDG environment variables
|
|
try setXdgEnv(tty_str, current_environment);
|
|
|
|
// Open the PAM session
|
|
var credentials = [_:null]?[*:0]const u8{ login, password };
|
|
|
|
const conv = interop.pam.pam_conv{
|
|
.conv = loginConv,
|
|
.appdata_ptr = @ptrCast(&credentials),
|
|
};
|
|
var handle: ?*interop.pam.pam_handle = undefined;
|
|
|
|
var status = interop.pam.pam_start(options.service_name, null, &conv, &handle);
|
|
if (status != interop.pam.PAM_SUCCESS) return pamDiagnose(status);
|
|
defer _ = interop.pam.pam_end(handle, status);
|
|
|
|
// Set PAM_TTY as the current TTY. This is required in case it isn't being set by another PAM module
|
|
status = interop.pam.pam_set_item(handle, interop.pam.PAM_TTY, pam_tty_str.ptr);
|
|
if (status != interop.pam.PAM_SUCCESS) return pamDiagnose(status);
|
|
|
|
// Do the PAM routine
|
|
status = interop.pam.pam_authenticate(handle, 0);
|
|
if (status != interop.pam.PAM_SUCCESS) return pamDiagnose(status);
|
|
|
|
status = interop.pam.pam_acct_mgmt(handle, 0);
|
|
if (status != interop.pam.PAM_SUCCESS) return pamDiagnose(status);
|
|
|
|
status = interop.pam.pam_setcred(handle, interop.pam.PAM_ESTABLISH_CRED);
|
|
if (status != interop.pam.PAM_SUCCESS) return pamDiagnose(status);
|
|
defer status = interop.pam.pam_setcred(handle, interop.pam.PAM_DELETE_CRED);
|
|
|
|
status = interop.pam.pam_open_session(handle, 0);
|
|
if (status != interop.pam.PAM_SUCCESS) return pamDiagnose(status);
|
|
defer status = interop.pam.pam_close_session(handle, 0);
|
|
|
|
var pwd: *interop.pwd.passwd = undefined;
|
|
{
|
|
defer interop.pwd.endpwent();
|
|
|
|
// Get password structure from username
|
|
pwd = interop.pwd.getpwnam(login) orelse return error.GetPasswordNameFailed;
|
|
}
|
|
|
|
// Set user shell if it hasn't already been set
|
|
if (pwd.pw_shell == null) {
|
|
interop.unistd.setusershell();
|
|
pwd.pw_shell = interop.unistd.getusershell();
|
|
interop.unistd.endusershell();
|
|
}
|
|
|
|
var shared_err = try SharedError.init();
|
|
defer shared_err.deinit();
|
|
|
|
child_pid = try std.posix.fork();
|
|
if (child_pid == 0) {
|
|
startSession(options, tty_str, pwd, handle, current_environment) catch |e| {
|
|
shared_err.writeError(e);
|
|
std.process.exit(1);
|
|
};
|
|
std.process.exit(0);
|
|
}
|
|
|
|
var entry = std.mem.zeroes(Utmp);
|
|
|
|
{
|
|
// If an error occurs here, we can send SIGTERM to the session
|
|
errdefer cleanup: {
|
|
_ = std.posix.kill(child_pid, std.posix.SIG.TERM) catch break :cleanup;
|
|
_ = std.posix.waitpid(child_pid, 0);
|
|
}
|
|
|
|
// If we receive SIGTERM, forward it to child_pid
|
|
const act = std.posix.Sigaction{
|
|
.handler = .{ .handler = &sessionSignalHandler },
|
|
.mask = std.posix.empty_sigset,
|
|
.flags = 0,
|
|
};
|
|
std.posix.sigaction(std.posix.SIG.TERM, &act, null);
|
|
|
|
try addUtmpEntry(&entry, pwd.pw_name.?, child_pid);
|
|
}
|
|
// Wait for the session to stop
|
|
_ = std.posix.waitpid(child_pid, 0);
|
|
|
|
removeUtmpEntry(&entry);
|
|
|
|
if (shared_err.readError()) |err| return err;
|
|
}
|
|
|
|
fn startSession(
|
|
options: AuthOptions,
|
|
tty_str: [:0]u8,
|
|
pwd: *interop.pwd.passwd,
|
|
handle: ?*interop.pam.pam_handle,
|
|
current_environment: Environment,
|
|
) !void {
|
|
if (builtin.os.tag == .freebsd) {
|
|
// FreeBSD has initgroups() in unistd
|
|
const status = interop.unistd.initgroups(pwd.pw_name, pwd.pw_gid);
|
|
if (status != 0) return error.GroupInitializationFailed;
|
|
|
|
// FreeBSD sets the GID and UID with setusercontext()
|
|
const result = interop.pwd.setusercontext(null, pwd, pwd.pw_uid, interop.pwd.LOGIN_SETALL);
|
|
if (result != 0) return error.SetUserUidFailed;
|
|
} else {
|
|
const status = interop.grp.initgroups(pwd.pw_name, pwd.pw_gid);
|
|
if (status != 0) return error.GroupInitializationFailed;
|
|
|
|
std.posix.setgid(pwd.pw_gid) catch return error.SetUserGidFailed;
|
|
std.posix.setuid(pwd.pw_uid) catch return error.SetUserUidFailed;
|
|
}
|
|
|
|
// Set up the environment
|
|
try initEnv(pwd, options.path);
|
|
|
|
// Reset the XDG environment variables
|
|
try setXdgEnv(tty_str, current_environment);
|
|
|
|
// Set the PAM variables
|
|
const pam_env_vars: ?[*:null]?[*:0]u8 = interop.pam.pam_getenvlist(handle);
|
|
if (pam_env_vars == null) return error.GetEnvListFailed;
|
|
|
|
const env_list = std.mem.span(pam_env_vars.?);
|
|
for (env_list) |env_var| _ = interop.stdlib.putenv(env_var);
|
|
|
|
// Change to the user's home directory
|
|
std.posix.chdirZ(pwd.pw_dir.?) catch return error.ChangeDirectoryFailed;
|
|
|
|
// Signal to the session process to give up control on the TTY
|
|
_ = std.posix.kill(options.session_pid, std.posix.SIG.CHLD) catch return error.TtyControlTransferFailed;
|
|
|
|
// Execute what the user requested
|
|
switch (current_environment.display_server) {
|
|
.wayland => try executeWaylandCmd(pwd.pw_shell.?, options, current_environment.cmd),
|
|
.shell => try executeShellCmd(pwd.pw_shell.?, options),
|
|
.xinitrc, .x11 => if (build_options.enable_x11_support) {
|
|
var vt_buf: [5]u8 = undefined;
|
|
const vt = try std.fmt.bufPrint(&vt_buf, "vt{d}", .{options.tty});
|
|
try executeX11Cmd(pwd.pw_shell.?, pwd.pw_dir.?, options, current_environment.cmd, vt);
|
|
},
|
|
.custom => try executeCustomCmd(pwd.pw_shell.?, options, current_environment.is_terminal, current_environment.cmd),
|
|
}
|
|
}
|
|
|
|
fn initEnv(pwd: *interop.pwd.passwd, path_env: ?[:0]const u8) !void {
|
|
_ = interop.stdlib.setenv("HOME", pwd.pw_dir, 1);
|
|
_ = interop.stdlib.setenv("PWD", pwd.pw_dir, 1);
|
|
_ = interop.stdlib.setenv("SHELL", pwd.pw_shell, 1);
|
|
_ = interop.stdlib.setenv("USER", pwd.pw_name, 1);
|
|
_ = interop.stdlib.setenv("LOGNAME", pwd.pw_name, 1);
|
|
|
|
if (path_env) |path| {
|
|
const status = interop.stdlib.setenv("PATH", path, 1);
|
|
if (status != 0) return error.SetPathFailed;
|
|
}
|
|
}
|
|
|
|
fn setXdgEnv(tty_str: [:0]u8, environment: Environment) !void {
|
|
_ = interop.stdlib.setenv("XDG_SESSION_TYPE", switch (environment.display_server) {
|
|
.wayland => "wayland",
|
|
.shell => "tty",
|
|
.xinitrc, .x11 => "x11",
|
|
.custom => "unspecified",
|
|
}, 0);
|
|
|
|
// The "/run/user/%d" directory is not available on FreeBSD. It is much
|
|
// better to stick to the defaults and let applications using
|
|
// XDG_RUNTIME_DIR to fall back to directories inside user's home
|
|
// directory.
|
|
if (builtin.os.tag != .freebsd) {
|
|
const uid = interop.unistd.getuid();
|
|
var uid_buffer: [10 + @sizeOf(u32) + 1]u8 = undefined;
|
|
const uid_str = try std.fmt.bufPrintZ(&uid_buffer, "/run/user/{d}", .{uid});
|
|
|
|
_ = interop.stdlib.setenv("XDG_RUNTIME_DIR", uid_str, 0);
|
|
}
|
|
|
|
if (environment.xdg_desktop_names) |xdg_desktop_names| _ = interop.stdlib.setenv("XDG_CURRENT_DESKTOP", xdg_desktop_names, 0);
|
|
_ = interop.stdlib.setenv("XDG_SESSION_CLASS", "user", 0);
|
|
_ = interop.stdlib.setenv("XDG_SESSION_ID", "1", 0);
|
|
if (environment.xdg_session_desktop) |desktop_name| _ = interop.stdlib.setenv("XDG_SESSION_DESKTOP", desktop_name, 0);
|
|
_ = interop.stdlib.setenv("XDG_SEAT", "seat0", 0);
|
|
_ = interop.stdlib.setenv("XDG_VTNR", tty_str, 0);
|
|
}
|
|
|
|
fn loginConv(
|
|
num_msg: c_int,
|
|
msg: ?[*]?*const interop.pam.pam_message,
|
|
resp: ?*?[*]interop.pam.pam_response,
|
|
appdata_ptr: ?*anyopaque,
|
|
) callconv(.C) c_int {
|
|
const message_count: u32 = @intCast(num_msg);
|
|
const messages = msg.?;
|
|
|
|
const allocator = std.heap.c_allocator;
|
|
const response = allocator.alloc(interop.pam.pam_response, message_count) catch return interop.pam.PAM_BUF_ERR;
|
|
|
|
// Initialise allocated memory to 0
|
|
// This ensures memory can be freed by pam on success
|
|
@memset(response, std.mem.zeroes(interop.pam.pam_response));
|
|
|
|
var username: ?[:0]u8 = null;
|
|
var password: ?[:0]u8 = null;
|
|
var status: c_int = interop.pam.PAM_SUCCESS;
|
|
|
|
for (0..message_count) |i| set_credentials: {
|
|
switch (messages[i].?.msg_style) {
|
|
interop.pam.PAM_PROMPT_ECHO_ON => {
|
|
const data: [*][*:0]u8 = @ptrCast(@alignCast(appdata_ptr));
|
|
username = allocator.dupeZ(u8, std.mem.span(data[0])) catch {
|
|
status = interop.pam.PAM_BUF_ERR;
|
|
break :set_credentials;
|
|
};
|
|
response[i].resp = username.?;
|
|
},
|
|
interop.pam.PAM_PROMPT_ECHO_OFF => {
|
|
const data: [*][*:0]u8 = @ptrCast(@alignCast(appdata_ptr));
|
|
password = allocator.dupeZ(u8, std.mem.span(data[1])) catch {
|
|
status = interop.pam.PAM_BUF_ERR;
|
|
break :set_credentials;
|
|
};
|
|
response[i].resp = password.?;
|
|
},
|
|
interop.pam.PAM_ERROR_MSG => {
|
|
status = interop.pam.PAM_CONV_ERR;
|
|
break :set_credentials;
|
|
},
|
|
else => {},
|
|
}
|
|
}
|
|
|
|
if (status != interop.pam.PAM_SUCCESS) {
|
|
// Memory is freed by pam otherwise
|
|
allocator.free(response);
|
|
if (username != null) allocator.free(username.?);
|
|
if (password != null) allocator.free(password.?);
|
|
} else {
|
|
resp.?.* = response.ptr;
|
|
}
|
|
|
|
return status;
|
|
}
|
|
|
|
fn getFreeDisplay() !u8 {
|
|
var buf: [15]u8 = undefined;
|
|
var i: u8 = 0;
|
|
while (i < 200) : (i += 1) {
|
|
const xlock = try std.fmt.bufPrint(&buf, "/tmp/.X{d}-lock", .{i});
|
|
std.posix.access(xlock, std.posix.F_OK) catch break;
|
|
}
|
|
return i;
|
|
}
|
|
|
|
fn getXPid(display_num: u8) !i32 {
|
|
var buf: [15]u8 = undefined;
|
|
const file_name = try std.fmt.bufPrint(&buf, "/tmp/.X{d}-lock", .{display_num});
|
|
const file = try std.fs.openFileAbsolute(file_name, .{});
|
|
defer file.close();
|
|
|
|
var file_buf: [20]u8 = undefined;
|
|
var fbs = std.io.fixedBufferStream(&file_buf);
|
|
|
|
_ = try file.reader().streamUntilDelimiter(fbs.writer(), '\n', 20);
|
|
const line = fbs.getWritten();
|
|
|
|
return std.fmt.parseInt(i32, std.mem.trim(u8, line, " "), 10);
|
|
}
|
|
|
|
fn createXauthFile(pwd: [:0]const u8) ![:0]const u8 {
|
|
var xauth_buf: [100]u8 = undefined;
|
|
var xauth_dir: [:0]const u8 = undefined;
|
|
const xdg_rt_dir = std.posix.getenv("XDG_RUNTIME_DIR");
|
|
var xauth_file: []const u8 = "lyxauth";
|
|
|
|
if (xdg_rt_dir == null) {
|
|
const xdg_cfg_home = std.posix.getenv("XDG_CONFIG_HOME");
|
|
var sb: std.c.Stat = undefined;
|
|
if (xdg_cfg_home == null) {
|
|
xauth_dir = try std.fmt.bufPrintZ(&xauth_buf, "{s}/.config", .{pwd});
|
|
_ = std.c.stat(xauth_dir, &sb);
|
|
const mode = sb.mode & std.posix.S.IFMT;
|
|
if (mode == std.posix.S.IFDIR) {
|
|
xauth_dir = try std.fmt.bufPrintZ(&xauth_buf, "{s}/ly", .{xauth_dir});
|
|
} else {
|
|
xauth_dir = pwd;
|
|
xauth_file = ".lyxauth";
|
|
}
|
|
} else {
|
|
xauth_dir = try std.fmt.bufPrintZ(&xauth_buf, "{s}/ly", .{xdg_cfg_home.?});
|
|
}
|
|
|
|
_ = std.c.stat(xauth_dir, &sb);
|
|
const mode = sb.mode & std.posix.S.IFMT;
|
|
if (mode != std.posix.S.IFDIR) {
|
|
std.posix.mkdir(xauth_dir, 777) catch {
|
|
xauth_dir = pwd;
|
|
xauth_file = ".lyxauth";
|
|
};
|
|
}
|
|
} else {
|
|
xauth_dir = xdg_rt_dir.?;
|
|
}
|
|
|
|
// Trim trailing slashes
|
|
var i = xauth_dir.len - 1;
|
|
while (xauth_dir[i] == '/') i -= 1;
|
|
const trimmed_xauth_dir = xauth_dir[0 .. i + 1];
|
|
|
|
var buf: [256]u8 = undefined;
|
|
const xauthority: [:0]u8 = try std.fmt.bufPrintZ(&buf, "{s}/{s}", .{ trimmed_xauth_dir, xauth_file });
|
|
const file = try std.fs.createFileAbsoluteZ(xauthority, .{});
|
|
file.close();
|
|
|
|
return xauthority;
|
|
}
|
|
|
|
fn mcookie() [Md5.digest_length * 2]u8 {
|
|
var buf: [4096]u8 = undefined;
|
|
std.crypto.random.bytes(&buf);
|
|
|
|
var out: [Md5.digest_length]u8 = undefined;
|
|
Md5.hash(&buf, &out, .{});
|
|
|
|
return std.fmt.bytesToHex(&out, .lower);
|
|
}
|
|
|
|
fn xauth(display_name: [:0]u8, shell: [*:0]const u8, pw_dir: [*:0]const u8, options: AuthOptions) !void {
|
|
var pwd_buf: [100]u8 = undefined;
|
|
const pwd = try std.fmt.bufPrintZ(&pwd_buf, "{s}", .{pw_dir});
|
|
|
|
const xauthority = try createXauthFile(pwd);
|
|
_ = interop.stdlib.setenv("XAUTHORITY", xauthority, 1);
|
|
_ = interop.stdlib.setenv("DISPLAY", display_name, 1);
|
|
|
|
const magic_cookie = mcookie();
|
|
|
|
const pid = try std.posix.fork();
|
|
if (pid == 0) {
|
|
var cmd_buffer: [1024]u8 = undefined;
|
|
const cmd_str = std.fmt.bufPrintZ(&cmd_buffer, "{s} add {s} . {s}", .{ options.xauth_cmd, display_name, magic_cookie }) catch std.process.exit(1);
|
|
const args = [_:null]?[*:0]const u8{ shell, "-c", cmd_str };
|
|
std.posix.execveZ(shell, &args, std.c.environ) catch {};
|
|
std.process.exit(1);
|
|
}
|
|
|
|
const status = std.posix.waitpid(pid, 0);
|
|
if (status.status != 0) return error.XauthFailed;
|
|
}
|
|
|
|
fn executeShellCmd(shell: [*:0]const u8, options: AuthOptions) !void {
|
|
// We don't want to redirect stdout and stderr in a shell session
|
|
|
|
var cmd_buffer: [1024]u8 = undefined;
|
|
const cmd_str = try std.fmt.bufPrintZ(&cmd_buffer, "{s} {s} {s}", .{ options.setup_cmd, options.login_cmd orelse "", shell });
|
|
const args = [_:null]?[*:0]const u8{ shell, "-c", cmd_str };
|
|
return std.posix.execveZ(shell, &args, std.c.environ);
|
|
}
|
|
|
|
fn executeWaylandCmd(shell: [*:0]const u8, options: AuthOptions, desktop_cmd: []const u8) !void {
|
|
const log_file = try redirectStandardStreams(options.session_log, true);
|
|
defer log_file.close();
|
|
|
|
var cmd_buffer: [1024]u8 = undefined;
|
|
const cmd_str = try std.fmt.bufPrintZ(&cmd_buffer, "{s} {s} {s}", .{ options.setup_cmd, options.login_cmd orelse "", desktop_cmd });
|
|
const args = [_:null]?[*:0]const u8{ shell, "-c", cmd_str };
|
|
return std.posix.execveZ(shell, &args, std.c.environ);
|
|
}
|
|
|
|
fn executeX11Cmd(shell: [*:0]const u8, pw_dir: [*:0]const u8, options: AuthOptions, desktop_cmd: []const u8, vt: []const u8) !void {
|
|
const display_num = try getFreeDisplay();
|
|
var buf: [5]u8 = undefined;
|
|
const display_name = try std.fmt.bufPrintZ(&buf, ":{d}", .{display_num});
|
|
try xauth(display_name, shell, pw_dir, options);
|
|
|
|
const pid = try std.posix.fork();
|
|
if (pid == 0) {
|
|
var cmd_buffer: [1024]u8 = undefined;
|
|
const cmd_str = std.fmt.bufPrintZ(&cmd_buffer, "{s} {s} {s}", .{ options.x_cmd, display_name, vt }) catch std.process.exit(1);
|
|
const args = [_:null]?[*:0]const u8{ shell, "-c", cmd_str };
|
|
std.posix.execveZ(shell, &args, std.c.environ) catch {};
|
|
std.process.exit(1);
|
|
}
|
|
|
|
var ok: c_int = undefined;
|
|
var xcb: ?*interop.xcb.xcb_connection_t = null;
|
|
while (ok != 0) {
|
|
xcb = interop.xcb.xcb_connect(null, null);
|
|
ok = interop.xcb.xcb_connection_has_error(xcb);
|
|
std.posix.kill(pid, 0) catch |e| {
|
|
if (e == error.ProcessNotFound and ok != 0) return error.XcbConnectionFailed;
|
|
};
|
|
}
|
|
|
|
// X Server detaches from the process.
|
|
// PID can be fetched from /tmp/X{d}.lock
|
|
const x_pid = try getXPid(display_num);
|
|
|
|
xorg_pid = try std.posix.fork();
|
|
if (xorg_pid == 0) {
|
|
var cmd_buffer: [1024]u8 = undefined;
|
|
const cmd_str = std.fmt.bufPrintZ(&cmd_buffer, "{s} {s} {s}", .{ options.setup_cmd, options.login_cmd orelse "", desktop_cmd }) catch std.process.exit(1);
|
|
const args = [_:null]?[*:0]const u8{ shell, "-c", cmd_str };
|
|
std.posix.execveZ(shell, &args, std.c.environ) catch {};
|
|
std.process.exit(1);
|
|
}
|
|
|
|
// If we receive SIGTERM, clean up by killing the xorg_pid process
|
|
const act = std.posix.Sigaction{
|
|
.handler = .{ .handler = &xorgSignalHandler },
|
|
.mask = std.posix.empty_sigset,
|
|
.flags = 0,
|
|
};
|
|
std.posix.sigaction(std.posix.SIG.TERM, &act, null);
|
|
|
|
_ = std.posix.waitpid(xorg_pid, 0);
|
|
interop.xcb.xcb_disconnect(xcb);
|
|
|
|
std.posix.kill(x_pid, 0) catch return;
|
|
std.posix.kill(x_pid, std.posix.SIG.KILL) catch {};
|
|
|
|
var status: c_int = 0;
|
|
_ = std.c.waitpid(x_pid, &status, 0);
|
|
}
|
|
|
|
fn executeCustomCmd(shell: [*:0]const u8, options: AuthOptions, is_terminal: bool, exec_cmd: []const u8) !void {
|
|
var maybe_log_file: ?std.fs.File = null;
|
|
if (!is_terminal) {
|
|
// For custom desktop entries, the "Terminal" value here determines if
|
|
// we redirect standard output & error or not. That is, we redirect only
|
|
// if it's equal to false (so if it's not running in a TTY).
|
|
maybe_log_file = try redirectStandardStreams(options.session_log, true);
|
|
}
|
|
defer if (maybe_log_file) |log_file| log_file.close();
|
|
|
|
var cmd_buffer: [1024]u8 = undefined;
|
|
const cmd_str = try std.fmt.bufPrintZ(&cmd_buffer, "{s} {s} {s}", .{ options.setup_cmd, options.login_cmd orelse "", exec_cmd });
|
|
const args = [_:null]?[*:0]const u8{ shell, "-c", cmd_str };
|
|
return std.posix.execveZ(shell, &args, std.c.environ);
|
|
}
|
|
|
|
fn redirectStandardStreams(session_log: []const u8, create: bool) !std.fs.File {
|
|
const log_file = if (create) (try std.fs.cwd().createFile(session_log, .{ .mode = 0o666 })) else (try std.fs.cwd().openFile(session_log, .{ .mode = .read_write }));
|
|
|
|
try std.posix.dup2(std.posix.STDOUT_FILENO, std.posix.STDERR_FILENO);
|
|
try std.posix.dup2(log_file.handle, std.posix.STDOUT_FILENO);
|
|
|
|
return log_file;
|
|
}
|
|
|
|
fn addUtmpEntry(entry: *Utmp, username: [*:0]const u8, pid: c_int) !void {
|
|
entry.ut_type = utmp.USER_PROCESS;
|
|
entry.ut_pid = pid;
|
|
|
|
var buf: [4096]u8 = undefined;
|
|
const ttyname = try std.os.getFdPath(std.posix.STDIN_FILENO, &buf);
|
|
|
|
var ttyname_buf: [@sizeOf(@TypeOf(entry.ut_line))]u8 = undefined;
|
|
_ = try std.fmt.bufPrintZ(&ttyname_buf, "{s}", .{ttyname["/dev/".len..]});
|
|
|
|
entry.ut_line = ttyname_buf;
|
|
entry.ut_id = ttyname_buf["tty".len..7].*;
|
|
|
|
var username_buf: [@sizeOf(@TypeOf(entry.ut_user))]u8 = undefined;
|
|
_ = try std.fmt.bufPrintZ(&username_buf, "{s}", .{username});
|
|
|
|
entry.ut_user = username_buf;
|
|
|
|
var host: [@sizeOf(@TypeOf(entry.ut_host))]u8 = undefined;
|
|
host[0] = 0;
|
|
entry.ut_host = host;
|
|
|
|
var tv: interop.system_time.timeval = undefined;
|
|
_ = interop.system_time.gettimeofday(&tv, null);
|
|
|
|
entry.ut_tv = .{
|
|
.tv_sec = @intCast(tv.tv_sec),
|
|
.tv_usec = @intCast(tv.tv_usec),
|
|
};
|
|
entry.ut_addr_v6[0] = 0;
|
|
|
|
utmp.setutxent();
|
|
_ = utmp.pututxline(entry);
|
|
utmp.endutxent();
|
|
}
|
|
|
|
fn removeUtmpEntry(entry: *Utmp) void {
|
|
entry.ut_type = utmp.DEAD_PROCESS;
|
|
entry.ut_line[0] = 0;
|
|
entry.ut_user[0] = 0;
|
|
utmp.setutxent();
|
|
_ = utmp.pututxline(entry);
|
|
utmp.endutxent();
|
|
}
|
|
|
|
fn pamDiagnose(status: c_int) anyerror {
|
|
return switch (status) {
|
|
interop.pam.PAM_ACCT_EXPIRED => return error.PamAccountExpired,
|
|
interop.pam.PAM_AUTH_ERR => return error.PamAuthError,
|
|
interop.pam.PAM_AUTHINFO_UNAVAIL => return error.PamAuthInfoUnavailable,
|
|
interop.pam.PAM_BUF_ERR => return error.PamBufferError,
|
|
interop.pam.PAM_CRED_ERR => return error.PamCredentialsError,
|
|
interop.pam.PAM_CRED_EXPIRED => return error.PamCredentialsExpired,
|
|
interop.pam.PAM_CRED_INSUFFICIENT => return error.PamCredentialsInsufficient,
|
|
interop.pam.PAM_CRED_UNAVAIL => return error.PamCredentialsUnavailable,
|
|
interop.pam.PAM_MAXTRIES => return error.PamMaximumTries,
|
|
interop.pam.PAM_NEW_AUTHTOK_REQD => return error.PamNewAuthTokenRequired,
|
|
interop.pam.PAM_PERM_DENIED => return error.PamPermissionDenied,
|
|
interop.pam.PAM_SESSION_ERR => return error.PamSessionError,
|
|
interop.pam.PAM_SYSTEM_ERR => return error.PamSystemError,
|
|
interop.pam.PAM_USER_UNKNOWN => return error.PamUserUnknown,
|
|
else => return error.PamAbort,
|
|
};
|
|
}
|