local data = { users = {} } -- Schema: -- data = { -- users = { -- ["username"] = { -- password = "hashed password", -- groups = { -- ["group"] = true -- }, -- }, -- } function save_data() db = fs.open("auth.db", "w") db.write(textutils.serialize(data)) db.close() end function load_data() if fs.exists("auth.db") then db = fs.open("auth.db", "r") db_contents = db.readAll() db.close() data = textutils.unserialize(db_contents) end end function generate_token(user) local token = hash(user .. os.time() .. math.random()) data.users[user].token = token return token end function convert_password(password) return hash(password .. "3m&LmNm7") end -- Does a complex hash of a string to make it harder to guess function hash(str) local hash = 0 local len = string.len(str) local byte = 0 for i = 1, len do byte = string.byte(str, i) hash = bit32.band(hash * 31 + byte, 0xFFFFFFFF) end return hash end function log(str) io.write("[" .. os.time() .. "] " .. str .. "\n") end function countElements(table) local count = 0 for key, value in pairs(table) do count = count + 1 end return count end load_data() math.randomseed(os.time()) while true do client_id, msg = rednet.receive("auth") request = msg if request.action == nil then rednet.send(client_id, "invalid request", "auth") end if request.action == "register" then if request.username == nil or request.password == nil then rednet.send(client_id, "invalid request", "auth") end if data.users[request.username] ~= nil then rednet.send(client_id, "user already exists", "auth") end data.users[request.username] = { password = convert_password(request.password) } local user_count = countElements(data.users) if user_count == 1 then log("Promoting " .. request.username .. " to admin") data.users[request.username].groups = { ["admin"] = true } end save_data() rednet.send(client_id, "ok", "auth") log(request.username .. " registered") end if request.action == "login" then if request.username == nil or request.password == nil then rednet.send(client_id, "invalid request", "auth") elseif data.users[request.username] == nil then rednet.send(client_id, "user not found", "auth") log(request.username .. " failed log in attempt") elseif convert_password(request.password) == data.users[request.username].password then local token = generate_token(request.username) rednet.send(client_id, token, "auth") log(request.username .. " logged in") else rednet.send(client_id, "invalid password", "auth") end end if request.action == "token" then if request.token == nil or request.token == "" or request.token == -1 then rednet.send(client_id, "invalid request", "auth") else local found = false for user, userdata in pairs(data.users) do if userdata.token == request.token then rednet.send(client_id, user, "auth") found = true end end if not found then rednet.send(client_id, "invalid token", "auth") end end end if request.action == "profile" then if request.username == nil then rednet.send(client_id, "invalid request", "auth") end local profile = {} if data.users[request.username].profile_data ~= nil then profile = data.users[request.username].profile_data else profile.display_name = request.username end profile.username = request.username rednet.send(client_id, data.users[request.username], "auth") end if request.action == "update_profile" then if request.username == nil or request.token == nil then rednet.send(client_id, "invalid request", "auth") end if data.users[request.username].token == request.token then if request.profile_data ~= nil then data.users[request.username].profile_data = request.profile_data end save_data() rednet.send(client_id, "ok") log(request.username .. " updated their profile", "auth") else rednet.send(client_id, "invalid token", "auth") end end if request.action == "check_group" then if request.username == nil or request.group == nil then rednet.send(client_id, "invalid request", "auth") end if data.users[request.username].groups == nil then rednet.send(client_id, false, "auth") else rednet.send(client_id, data.users[request.username].groups[request.group] ~= nil, "auth") end end if request.action == "list_groups" then if request.username == nil then rednet.send(client_id, "invalid request", "auth") end if data.users[request.username].groups == nil then rednet.send(client_id, {}, "auth") else rednet.send(client_id, data.users[request.username].groups, "auth") end end -- Only if in group admin if request.action == "add_group" then if request.username == nil or request.group == nil or request.token == nil then rednet.send(client_id, "invalid request", "auth") else local found = false local acting_user = nil for user, userdata in pairs(data.users) do if userdata.token == request.token then acting_user = user found = true end end if not found then rednet.send(client_id, "invalid token", "auth") elseif data.users[acting_user]["groups"] == nil or data.users[acting_user].groups["admin"] == nil then rednet.send(client_id, "invalid privileges", "auth") else if data.users[request.username].groups == nil then data.users[request.username].groups = {} end data.users[request.username].groups[request.group] = true save_data() rednet.send(client_id, "ok", "auth") log(request.username .. " added to group " .. request.group, "auth") end end end end