Advertisement
theoriginalbit

DoorCode v4.1

Dec 16th, 2012
416
0
Never
Not a member of Pastebin yet? Sign Up, it unlocks many cool features!
Lua 28.48 KB | None | 0 0
  1. --[[
  2. Author: TheOriginalBIT
  3. Version: 4.1
  4. Created: 17 Dec 2012
  5. Last Update: 22 Feb 2013
  6.  
  7. License:
  8.  
  9. COPYRIGHT NOTICE
  10. Copyright © 2012-2013 Joshua Asbury a.k.a TheOriginalBIT [[email protected]]
  11.  
  12. Permission is hereby granted, free of charge, to any person obtaining a copy of this software and
  13. associated documentation files (the "Software"), to deal in the Software without restriction,
  14. including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense,
  15. copies of the Software, and to permit persons to whom the Software is furnished to do so,
  16. subject to the following conditions:
  17.  
  18. -The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
  19. -Visible credit is given to the original author.
  20. -The software is distributed in a non-profit way.
  21.  
  22. THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE
  23. WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR
  24. COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
  25. ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
  26. ]]--
  27.  
  28. -- changing anything beyond this point could break the program
  29.  
  30. --[[ EXTERNAL LIBRARY BY GRAVITYSCORE ]]--
  31. --  
  32. --  Adaptation of the Secure Hashing Algorithm (SHA-244/256)
  33. --  Found Here: http://lua-users.org/wiki/SecureHashAlgorithm
  34. --  
  35. --  Using an adapted version of the bit library
  36. --  Found Here: https://bitbucket.org/Boolsheet/bslf/src/1ee664885805/bit.lua
  37. --  
  38. --  Adapted into a single function by GravityScore
  39. --  
  40.  
  41. local hash = function(msg) local function band(int1, int2, int3, ...) local ret = ((int1%0x00000002>=0x00000001 and int2%0x00000002>=0x00000001 and 0x00000001) or 0) + ((int1%0x00000004>=0x00000002 and int2%0x00000004>=0x00000002 and 0x00000002) or 0) + ((int1%0x00000008>=0x00000004 and int2%0x00000008>=0x00000004 and 0x00000004) or 0) + ((int1%0x00000010>=0x00000008 and int2%0x00000010>=0x00000008 and 0x00000008) or 0) + ((int1%0x00000020>=0x00000010 and int2%0x00000020>=0x00000010 and 0x00000010) or 0) + ((int1%0x00000040>=0x00000020 and int2%0x00000040>=0x00000020 and 0x00000020) or 0) + ((int1%0x00000080>=0x00000040 and int2%0x00000080>=0x00000040 and 0x00000040) or 0) + ((int1%0x00000100>=0x00000080 and int2%0x00000100>=0x00000080 and 0x00000080) or 0) + ((int1%0x00000200>=0x00000100 and int2%0x00000200>=0x00000100 and 0x00000100) or 0) + ((int1%0x00000400>=0x00000200 and int2%0x00000400>=0x00000200 and 0x00000200) or 0) + ((int1%0x00000800>=0x00000400 and int2%0x00000800>=0x00000400 and 0x00000400) or 0) + ((int1%0x00001000>=0x00000800 and int2%0x00001000>=0x00000800 and 0x00000800) or 0) + ((int1%0x00002000>=0x00001000 and int2%0x00002000>=0x00001000 and 0x00001000) or 0) + ((int1%0x00004000>=0x00002000 and int2%0x00004000>=0x00002000 and 0x00002000) or 0) + ((int1%0x00008000>=0x00004000 and int2%0x00008000>=0x00004000 and 0x00004000) or 0) + ((int1%0x00010000>=0x00008000 and int2%0x00010000>=0x00008000 and 0x00008000) or 0) + ((int1%0x00020000>=0x00010000 and int2%0x00020000>=0x00010000 and 0x00010000) or 0) + ((int1%0x00040000>=0x00020000 and int2%0x00040000>=0x00020000 and 0x00020000) or 0) + ((int1%0x00080000>=0x00040000 and int2%0x00080000>=0x00040000 and 0x00040000) or 0) + ((int1%0x00100000>=0x00080000 and int2%0x00100000>=0x00080000 and 0x00080000) or 0) + ((int1%0x00200000>=0x00100000 and int2%0x00200000>=0x00100000 and 0x00100000) or 0) + ((int1%0x00400000>=0x00200000 and int2%0x00400000>=0x00200000 and 0x00200000) or 0) + ((int1%0x00800000>=0x00400000 and int2%0x00800000>=0x00400000 and 0x00400000) or 0) + ((int1%0x01000000>=0x00800000 and int2%0x01000000>=0x00800000 and 0x00800000) or 0) + ((int1%0x02000000>=0x01000000 and int2%0x02000000>=0x01000000 and 0x01000000) or 0) + ((int1%0x04000000>=0x02000000 and int2%0x04000000>=0x02000000 and 0x02000000) or 0) + ((int1%0x08000000>=0x04000000 and int2%0x08000000>=0x04000000 and 0x04000000) or 0) + ((int1%0x10000000>=0x08000000 and int2%0x10000000>=0x08000000 and 0x08000000) or 0) + ((int1%0x20000000>=0x10000000 and int2%0x20000000>=0x10000000 and 0x10000000) or 0) + ((int1%0x40000000>=0x20000000 and int2%0x40000000>=0x20000000 and 0x20000000) or 0) + ((int1%0x80000000>=0x40000000 and int2%0x80000000>=0x40000000 and 0x40000000) or 0) + ((int1>=0x80000000 and int2>=0x80000000 and 0x80000000) or 0) return (int3 and band(ret, int3, ...)) or ret end local function bxor(int1, int2, int3, ...) local ret = ((int1%0x00000002>=0x00000001 ~= (int2%0x00000002>=0x00000001) and 0x00000001) or 0) + ((int1%0x00000004>=0x00000002 ~= (int2%0x00000004>=0x00000002) and 0x00000002) or 0) + ((int1%0x00000008>=0x00000004 ~= (int2%0x00000008>=0x00000004) and 0x00000004) or 0) + ((int1%0x00000010>=0x00000008 ~= (int2%0x00000010>=0x00000008) and 0x00000008) or 0) + ((int1%0x00000020>=0x00000010 ~= (int2%0x00000020>=0x00000010) and 0x00000010) or 0) + ((int1%0x00000040>=0x00000020 ~= (int2%0x00000040>=0x00000020) and 0x00000020) or 0) + ((int1%0x00000080>=0x00000040 ~= (int2%0x00000080>=0x00000040) and 0x00000040) or 0) + ((int1%0x00000100>=0x00000080 ~= (int2%0x00000100>=0x00000080) and 0x00000080) or 0) + ((int1%0x00000200>=0x00000100 ~= (int2%0x00000200>=0x00000100) and 0x00000100) or 0) + ((int1%0x00000400>=0x00000200 ~= (int2%0x00000400>=0x00000200) and 0x00000200) or 0) + ((int1%0x00000800>=0x00000400 ~= (int2%0x00000800>=0x00000400) and 0x00000400) or 0) + ((int1%0x00001000>=0x00000800 ~= (int2%0x00001000>=0x00000800) and 0x00000800) or 0) + ((int1%0x00002000>=0x00001000 ~= (int2%0x00002000>=0x00001000) and 0x00001000) or 0) + ((int1%0x00004000>=0x00002000 ~= (int2%0x00004000>=0x00002000) and 0x00002000) or 0) + ((int1%0x00008000>=0x00004000 ~= (int2%0x00008000>=0x00004000) and 0x00004000) or 0) + ((int1%0x00010000>=0x00008000 ~= (int2%0x00010000>=0x00008000) and 0x00008000) or 0) + ((int1%0x00020000>=0x00010000 ~= (int2%0x00020000>=0x00010000) and 0x00010000) or 0) + ((int1%0x00040000>=0x00020000 ~= (int2%0x00040000>=0x00020000) and 0x00020000) or 0) + ((int1%0x00080000>=0x00040000 ~= (int2%0x00080000>=0x00040000) and 0x00040000) or 0) + ((int1%0x00100000>=0x00080000 ~= (int2%0x00100000>=0x00080000) and 0x00080000) or 0) + ((int1%0x00200000>=0x00100000 ~= (int2%0x00200000>=0x00100000) and 0x00100000) or 0) + ((int1%0x00400000>=0x00200000 ~= (int2%0x00400000>=0x00200000) and 0x00200000) or 0) + ((int1%0x00800000>=0x00400000 ~= (int2%0x00800000>=0x00400000) and 0x00400000) or 0) + ((int1%0x01000000>=0x00800000 ~= (int2%0x01000000>=0x00800000) and 0x00800000) or 0) + ((int1%0x02000000>=0x01000000 ~= (int2%0x02000000>=0x01000000) and 0x01000000) or 0) + ((int1%0x04000000>=0x02000000 ~= (int2%0x04000000>=0x02000000) and 0x02000000) or 0) + ((int1%0x08000000>=0x04000000 ~= (int2%0x08000000>=0x04000000) and 0x04000000) or 0) + ((int1%0x10000000>=0x08000000 ~= (int2%0x10000000>=0x08000000) and 0x08000000) or 0) + ((int1%0x20000000>=0x10000000 ~= (int2%0x20000000>=0x10000000) and 0x10000000) or 0) + ((int1%0x40000000>=0x20000000 ~= (int2%0x40000000>=0x20000000) and 0x20000000) or 0) + ((int1%0x80000000>=0x40000000 ~= (int2%0x80000000>=0x40000000) and 0x40000000) or 0) + ((int1>=0x80000000 ~= (int2>=0x80000000) and 0x80000000) or 0) return (int3 and bxor(ret, int3, ...)) or ret end local function bnot(int) return 4294967295 - int end local function rshift(int, by) local shifted = int / (2 ^ by) return shifted - shifted % 1 end local function rrotate(int, by) local shifted = int / (2 ^ by) local fraction = shifted % 1 return (shifted - fraction) + fraction * (2 ^ 32) end local k = { 0x428a2f98, 0x71374491, 0xb5c0fbcf, 0xe9b5dba5, 0x3956c25b, 0x59f111f1, 0x923f82a4, 0xab1c5ed5, 0xd807aa98, 0x12835b01, 0x243185be, 0x550c7dc3, 0x72be5d74, 0x80deb1fe, 0x9bdc06a7, 0xc19bf174, 0xe49b69c1, 0xefbe4786, 0x0fc19dc6, 0x240ca1cc, 0x2de92c6f, 0x4a7484aa, 0x5cb0a9dc, 0x76f988da, 0x983e5152, 0xa831c66d, 0xb00327c8, 0xbf597fc7, 0xc6e00bf3, 0xd5a79147, 0x06ca6351, 0x14292967, 0x27b70a85, 0x2e1b2138, 0x4d2c6dfc, 0x53380d13, 0x650a7354, 0x766a0abb, 0x81c2c92e, 0x92722c85, 0xa2bfe8a1, 0xa81a664b, 0xc24b8b70, 0xc76c51a3, 0xd192e819, 0xd6990624, 0xf40e3585, 0x106aa070, 0x19a4c116, 0x1e376c08, 0x2748774c, 0x34b0bcb5, 0x391c0cb3, 0x4ed8aa4a, 0x5b9cca4f, 0x682e6ff3, 0x748f82ee, 0x78a5636f, 0x84c87814, 0x8cc70208, 0x90befffa, 0xa4506ceb, 0xbef9a3f7, 0xc67178f2 } local function str2hexa(s) local h = string.gsub(s, ".", function(c) return string.format("%02x", string.byte(c)) end) return h end local function num2s(l, n) local s = "" for i = 1, n do local rem = l % 256 s = string.char(rem) .. s l = (l - rem) / 256 end return s end local function s232num(s, i) local n = 0 for i = i, i + 3 do n = n*256 + string.byte(s, i) end return n end local function preproc(msg, len) local extra = 64 - ((len + 1 + 8) % 64) len = num2s(8 * len, 8) msg = msg .. "\128" .. string.rep("\0", extra) .. len return msg end local function initH256(H) H[1] = 0x6a09e667 H[2] = 0xbb67ae85 H[3] = 0x3c6ef372 H[4] = 0xa54ff53a H[5] = 0x510e527f H[6] = 0x9b05688c H[7] = 0x1f83d9ab H[8] = 0x5be0cd19 return H end local function digestblock(msg, i, H) local w = {} for j = 1, 16 do w[j] = s232num(msg, i + (j - 1) * 4) end for j = 17, 64 do local v = w[j - 15] local s0 = bxor(rrotate(v, 7), rrotate(v, 18), rshift(v, 3)) v = w[j - 2] local s1 = bxor(rrotate(v, 17), rrotate(v, 19), rshift(v, 10)) w[j] = w[j - 16] + s0 + w[j - 7] + s1 end local a, b, c, d, e, f, g, h = H[1], H[2], H[3], H[4], H[5], H[6], H[7], H[8] for i = 1, 64 do local s0 = bxor(rrotate(a, 2), rrotate(a, 13), rrotate(a, 22)) local maj = bxor(band(a, b), band(a, c), band(b, c)) local t2 = s0 + maj local s1 = bxor(rrotate(e, 6), rrotate(e, 11), rrotate(e, 25)) local ch = bxor (band(e, f), band(bnot(e), g)) local t1 = h + s1 + ch + k[i] + w[i] h, g, f, e, d, c, b, a = g, f, e, d + t1, c, b, a, t1 + t2 end H[1] = band(H[1], a) H[2] = band(H[2], b) H[3] = band(H[3], c) H[4] = band(H[4], d) H[5] = band(H[5], e) H[6] = band(H[6], f) H[7] = band(H[7], g) H[8] = band(H[8], h) end msg = preproc(msg, #msg) local H = initH256({}) for i = 1, #msg, 64 do digestblock(msg, i, H) end return str2hexa(num2s(H[1], 4)..num2s(H[2], 4)..num2s(H[3], 4)..num2s(H[4], 4)..num2s(H[5], 4)..num2s(H[6], 4)..num2s(H[7], 4)..num2s(H[8], 4)) end
  42.  
  43. --[[ END EXTERNAL LIBRARY ]]--
  44.  
  45. local version = "4.0"
  46. local salt = [[T0B1td2j@<?s3.&8)uXOC1(E>25-~;o2{MJ+mSB=1Ye9SSS|[~7_t195l6Q2Cg/Awb"(53cUr1Ty]]
  47. local supportLoc = ".theoriginalbit/DoorLock/"
  48. local passFile = supportLoc.."access.dat"
  49. local configFile = supportLoc.."access.cfg"
  50. local adv = term.isColor and term.isColor()
  51.  
  52. local accounts = {}
  53. local terminatePass = ""
  54. local rsOutput
  55. local welcomeMessage
  56. local text = ""
  57.  
  58. do
  59.  --[[
  60.  Author: TheOriginalBIT
  61.  Version: 1.4.5
  62.  Created: 5 Feb 2013
  63.  Last Update: 22 Feb 2013
  64.  ]]--
  65.  
  66.  local configHeader = [[
  67. +-------------------------------------------+
  68. +--          CONFIGURATION FILE           --+
  69. +-- Generated by TheOriginalBIT's CConfig --+
  70. +-------------------------------------------+
  71.  
  72. ]]
  73.  
  74.  local startsWith = function(str, prefix) prefix = tostring(prefix) return str:sub( 1, prefix:len()) == prefix end
  75.  local contains = function(str, seq) str = tostring(str) local sStart, sEnd, sStr = str:find(seq, 1) return sStart ~= nil end
  76.  local split = function(str, pat) local t = {} local fpat = "(.-)"..pat local last_end = 1 local s, e, cap = str:find(fpat, 1) while s do if s ~= 1 or cap ~= "" then table.insert(t,cap) end last_end = e+1 s, e, cap = str:find(fpat, last_end) end if last_end <= #str then cap = str:sub(last_end) table.insert(t, cap) end return t end
  77.  local trim = function(str) return (str:gsub("^%s*(.-)%s*$", "%1")) end
  78.  local log = function(...) if not emulog then return end local details = table.concat({...}," ") emulog.log(details) end
  79.  local clearLog = function() if not emulog then return end emulog.clear() end
  80.  
  81.  -- If you're reading the code, normally I do not suggest overwriting system functions without restoring them! Do as I say, not as I do. I've made sure this 100% won't break other programs!
  82.  _G.assert = function(condition, errMsg, level) if not condition then error(errMsg, (tonumber(level) or 1) + 1) end return condition end
  83.  
  84.  local openedConfigs = {}
  85.  Configuration = {}
  86.  local debugCount = 0
  87.  
  88.  Configuration.__index = Configuration
  89.  
  90.  local function vaidateInitParams(path)
  91.    assert(type(p) ~= "string", "Invalid file path supplied", 3)
  92.    assert(not openedConfigs[p], "A configuration file object already exists for this file", 3)
  93.  end
  94.  
  95.  local function validateParam(param, value)
  96.    assert(type(param) == value, "Invalid parameter, expected "..value.." got "..type(param), 3)
  97.  end
  98.  
  99.  local function validBoolean(b)
  100.    return (b == "true" or b == "false" or b == "1" or b == "0" or b == 1 or b == 0 or b == true or b == false or b == "yes" or b == "no")
  101.  end
  102.  
  103.  local colorStrings = {["colors.white"] = colors.white, ["colours.white"] = colors.white, ["white"] = colors.white, ["colors.orange"] = colors.orange, ["colours.orange"] = colors.orange, ["orange"] = colors.orange, ["colors.magenta"] = colors.magenta, ["colours.magenta"] = colors.magenta, ["magenta"] = colors.magenta, ["colors.lightblue"] = colors.lightBlue, ["colours.lightblue"] = colors.lightBlue, ["lightblue"] = colors.lightBlue, ["light blue"] = colors.lightBlue, ["colors.yellow"] = colors.yellow, ["colours.yellow"] = colors.yellow, ["yellow"] = colors.yellow, ["colors.lime"] = colors.lime, ["colours.lime"] = colors.lime, ["lime"] = colors.lime, ["colors.pink"] = colors.pink, ["colours.pink"] = colors.pink, ["pink"] = colors.pink, ["colors.gray"] = colors.gray, ["colours.gray"] = colors.gray, ["gray"] = colors.gray, ["colors.grey"] = colors.gray, ["colours.grey"] = colors.gray, ["grey"] = colors.gray, ["colors.lightgray"] = colors.lightGray, ["colours.lightgray"] = colors.lightGray, ["lightgray"] = colors.lightGray, ["light gray"] = colors.lightGray, ["colors.lightgrey"] = colors.lightGray, ["colours.lightgrey"] = colors.lightGray, ["lightgrey"] = colors.lightGray, ["light grey"] = colors.lightGray, ["colors.cyan"] = colors.cyan, ["colours.cyan"] = colors.cyan, ["cyan"] = colors.cyan, ["colors.purple"] = colors.purple, ["colours.purple"] = colors.purple, ["purple"] = colors.purple, ["colors.blue"] = colors.blue, ["colours.blue"] = colors.blue, ["blue"] = colors.blue, ["colors.brown"] = colors.brown, ["colours.brown"] = colors.brown, ["brown"] = colors.brown, ["colors.green"] = colors.green, ["colours.green"] = colors.green, ["green"] = colors.green, ["colors.red"] = colors.red, ["colours.red"] = colors.red, ["red"] = colors.red, ["colors.black"] = colors.black, ["colours.black"] = colors.black, ["black"] = colors.black}
  104.  
  105.  local function validateColor(col)
  106.    if tonumber(col) then
  107.      local col = tonumber(col)
  108.      if not (col >= 2^0 and col <= (2^16)-1) then return false end
  109.      return true, col
  110.    else
  111.      if colorStrings[col:lower()] then
  112.        return true, colorStrings[col:lower()]
  113.      else
  114.        return false
  115.      end
  116.    end
  117.  end
  118.  
  119.  function Configuration.new(p)
  120.    vaidateInitParams(p)
  121.    openedConfigs[p] = true
  122.    return setmetatable({ path = p, properties = {} }, Configuration)
  123.  end
  124.  
  125.  local function get(self, key, defaultValue)
  126.    if type(self.properties[key]) == "table" then
  127.      self.properties[key].default = defaultValue
  128.      return self.properties[key].value
  129.    else
  130.      self.properties[key] = {}
  131.      self.properties[key].value = defaultValue
  132.      self.properties[key].default = defaultValue
  133.      self.properties[key].comment = ""
  134.      self.properties[key].restrict = ""
  135.      return defaultValue
  136.    end
  137.  end
  138.  
  139.  function Configuration:getBoolean(key, defaultBoolean)
  140.    validateParam(key, "string")
  141.    assert(validBoolean(defaultBoolean), "Boolean value expected, got "..type(defaultBoolean).." of type "..type(defaultBoolean), 2)
  142.    local v = get(self, key, defaultBoolean)
  143.    assert(validBoolean(v), "No boolean value found under the key \'"..tostring(key).."\', found "..tostring(v).." of type "..type(v), 2)
  144.    return v == "true" or v == true or v == "1" or v == 1 or v == "yes"
  145.  end
  146.  
  147.  function Configuration:getNumber(key, defaultNumber)
  148.    validateParam(key, "string")
  149.    return assert(tonumber(get(self, key, defaultNumber)), "Number expected, got "..type(defaultNumber), 2)
  150.  end
  151.  
  152.  function Configuration:getString(key, defaultString)
  153.    validateParam(key, "string")
  154.    return assert(tostring(get(self, key, defaultString)), "Invalid parameter, expected string got "..type(key), 2)
  155.  end
  156.  
  157.  function Configuration:getColor(key, defaultColor)
  158.    validateParam(key, "string")
  159.    assert(validateColor(defaultColor), "Invalid parameter, color expected", 2)
  160.    local ok, col = validateColor(get(self, key, defaultColor))
  161.    assert(ok, "Config file does not contain color for value "..tostring(key), 2)
  162.    self.properties[key].restrict = "A number between 1 and 65535 or the string colors.<color>"
  163.    return col
  164.  end
  165.  
  166.  Configuration.getColour = Configuration.getColor
  167.  
  168.  function Configuration:getTable(key, defaultTable)
  169.    validateParam(key, "string")
  170.    validateParam(defaultTable, "table")
  171.    local v = self:getString(key, textutils.serialize(defaultTable))
  172.    v = textutils.unserialize(v)
  173.    return assert(v, "No table in config", 2)
  174.  end
  175.  
  176.  function Configuration:containsKey(key)
  177.    return (self.properties[key] ~= nil)
  178.  end
  179.  
  180.  function Configuration:load()
  181.    if not fs.exists(self.path) then return false end
  182.    local handle = fs.open(self.path,"r")
  183.    assert(handle, "Cannot open configuration file \'"..self.path.."\' for read")
  184.    local contents = handle.readAll()
  185.    handle.close()
  186.  
  187.    if not contents or contents == "" or contents == "\n" then return end
  188.    local count = 1
  189.    contents = split(contents,"\n")
  190.    for i = 1, #contents do
  191.      local v = contents[i]
  192.      if v and v ~= "" and not startsWith(v,"+--") and contains(v,"=") then
  193.        local prop = split(v, "%=")
  194.        key = trim(prop[1])
  195.        value = trim(prop[2])
  196.        self.properties[key] = {}
  197.        self.properties[key]["value"]   = value
  198.        self.properties[key]["comment"] = ""
  199.        self.properties[key]["default"] = value
  200.      elseif v and v ~= "" and not startsWith(v,"+--") and not contains(v,"=") then
  201.        print("Error: \""..self.path.."\": Cannot parse line #"..i.." in configuration file")
  202.        error()
  203.      end
  204.    end
  205.    return true
  206.  end
  207.  
  208.  function Configuration:reset(key)
  209.    if type(self) ~= "table" or type(self.properties) ~= "table" then error("Error resetting config, have you forgotten to load the config?",2) end
  210.    if key and not self.properties[key] then error("No property for key "..tostring(key)..", have you forgotten to load the config?",2) end
  211.  
  212.    if key then
  213.      self.properties[key].value = self.properties[key].default
  214.    else
  215.      for k,_ in pairs(self.properties) do
  216.        self.properties[k].value = self.properties[k].default
  217.      end
  218.    end
  219.  end
  220.  
  221.  function Configuration:save()
  222.    local handle = fs.open(self.path,"w")
  223.    handle.write(configHeader)
  224.    for k,v in pairs(self.properties) do
  225.      handle.write("+-- ")
  226.      if v.comment and v.comment ~= "" then
  227.        handle.write(tostring(v.comment).."; ")
  228.      end
  229.      if v.restrict and v.restrict ~= "" then
  230.        handle.write("restrictions: "..tostring(v.restrict).."; ")
  231.      end
  232.      handle.write("default: "..tostring(v.default).."\n")
  233.      handle.write(k.."="..tostring(v.value).."\n\n")
  234.    end
  235.    handle.close()
  236.  end
  237.  
  238.  function Configuration:addCommentForKey(key,com)
  239.    assert(self.properties[key], "No property with key \'"..tostring(key).."\' to add comment for", 2)
  240.    self.properties[key].comment = com
  241.  end
  242.  
  243.  function Configuration:addRestrictionForKey(key,res)
  244.    assert(self.properties[key], "No property with key \'"..tostring(key).."\' to add comment for", 2)
  245.    self.properties[key].restrict = res
  246.  end
  247.  
  248.  function Configuration:debug(toFile)
  249.    toFile = (toFile == true)
  250.    local handle = toFile and fs.open("debug"..debugCount,"w") or nil
  251.    for k,v in pairs(self.properties) do
  252.      if toFile then
  253.        handle.write(k..":"..v.value.."\n")
  254.      else
  255.        print(k..":"..v.value)
  256.      end
  257.    end
  258.    if toFile then
  259.      handle.close()
  260.      debugCount = debugCount + 1
  261.    end
  262.  end
  263. end
  264.  
  265. term.clear = function() if adv then term.setBackgroundColor(colors.black) term.setTextColor(colors.white) end term.native.clear() term.setCursorPos(1,1) end
  266.  
  267. local waitForKey = function() os.pullEventRaw("key") end
  268. local checkPass = function(pass) for _,v in pairs(accounts) do if v == pass then return true end end return false end
  269. local validateSideWithoutFail = function(side) for _,v in pairs(rs.getSides()) do if v == side then return v end end return "left" end
  270. local printRecognition = function() local sw, sh = term.getSize();local pX,pY=term.getCursorPos();term.setCursorPos(sw - string.len("By TheOriginalBIT "), sh);write("By TheOriginalBIT");term.setCursorPos(pX, pY);end
  271. local split = function(str, pat) local t = {} local fpat = "(.-)"..pat local last_end = 1 local s, e, cap = str:find(fpat, 1) while s do if s ~= 1 or cap ~= "" then table.insert(t,cap) end last_end = e+1 s, e, cap = str:find(fpat, last_end) end if last_end <= #str then cap = str:sub(last_end) table.insert(t, cap) end return t end
  272. local cwrite = function(msg,y) local w,h = term.getSize() term.setCursorPos(w/2-#msg/2+(#msg%2 and 1 or 0), y) write(msg) end
  273.  
  274. local config = Configuration.new(configFile)
  275.  
  276. local function loadAccounts()
  277.  local file = io.open(passFile,"r")
  278.  local contents = file:read("*a")
  279.  file:close()
  280.  contents = split(contents, "\n")
  281.  terminatePass = contents[1]
  282.  for i = 2, #contents do
  283.    if #contents[i] == 64 then
  284.      table.insert(accounts, contents[i])
  285.    end
  286.  end
  287. end
  288.  
  289. local function saveAccounts()
  290.  local file = io.open(passFile,"w")
  291.  file:write(terminatePass.."\n")
  292.  file:write(table.concat(accounts,"\n"))
  293.  file:close()
  294. end
  295.  
  296. local function loadConfig()
  297.  config:load()
  298.  
  299.  welcomeMessage = config:getString("welcomeMessage", "Advanced DoorLock v4.0")
  300.  config:addCommentForKey("welcomeMessage", "the message to display at the top of the screen")
  301.  rsOutput = validateSideWithoutFail(config:getString("redstoneSide", "left"))
  302.  config:addCommentForKey("redstoneSide", "side to output when password is correct")
  303.  canDiskStay = config:getBoolean("allowDisks", false)
  304.  config:addCommentForKey("allowDisks", "allows disks to be inserted into any disk drives found")
  305.  
  306.  config:save()
  307. end
  308.  
  309. local function getInput(msg, char)
  310.  x, y = term.getCursorPos()
  311.  input = ""
  312.  print(msg)
  313.  while true do
  314.    term.setCursorPos(x, y)
  315.    term.clearLine()
  316.    
  317.    write(msg.." ")
  318.    if char then
  319.      for i = 1, #input do
  320.        write(char)
  321.      end
  322.    else
  323.      write(input)
  324.    end
  325.    
  326.    event = { os.pullEventRaw() }
  327.    
  328.    if event[1] == "char" then
  329.      input = input..event[2]
  330.    elseif event[1] == "key" then
  331.      if event[2] == keys.backspace then
  332.        input = input:sub(1, input:len() - 1)
  333.      elseif event[2] == keys.enter then
  334.        break
  335.      end
  336.    end
  337.  end
  338.  print()
  339.  return input
  340. end
  341.  
  342. local function initialSetup()
  343.  term.clear()
  344.  print("Setup...")
  345.  
  346.  if not fs.exists(supportLoc) then
  347.    fs.makeDir(supportLoc)
  348.  end
  349.  
  350.  if not fs.exists(passFile) then
  351.    local pfile = io.open(passFile,"w")
  352.    pfile:write()
  353.    pfile:close()
  354.  end
  355.  
  356.  while true do
  357.    local ok, err = pcall(loadConfig)
  358.    print("Prepare to review the config file... ")
  359.    os.startTimer(3)
  360.    os.pullEvent("timer")
  361.  
  362.    local ope = os.pullEvent
  363.    os.pullEvent = os.pullEventRaw
  364.    shell.run("edit "..configFile)
  365.    os.pullEvent = ope
  366.    local ok, err = pcall(loadConfig)
  367.    if not ok then
  368.      term.setTextColor(colors.red)
  369.      print("Error: Cannot parse the config file. Please check syntax for what you changed.\n\n")
  370.      term.setTextColor(colors.white)
  371.      os.startTimer(3)
  372.      os.pullEvent("timer")
  373.    else
  374.      break
  375.    end
  376.  end
  377.  
  378.  print("Prepare to setup passwords... ")
  379.  os.startTimer(3)
  380.  os.pullEvent("timer")
  381.  
  382.  while true do
  383.    term.clear()
  384.    print("Termination password setup.")
  385.    local pw = getInput("Password:","*")
  386.    if pw == "finished" then break end
  387.    local pw2 = getInput("Again:","*")
  388.    if pw2 == "finished" then break end
  389.    if pw == pw2 then
  390.      terminatePass = hash(pw..salt)
  391.      saveAccounts()
  392.      print("Password added")
  393.      os.startTimer(2)
  394.      os.pullEvent("timer")
  395.      break
  396.    else
  397.      print("Passwords do not match")
  398.      os.startTimer(2)
  399.      os.pullEvent("timer")
  400.    end
  401.  end
  402.  
  403.  while true do
  404.    term.clear()
  405.    print("Type 'finished' to stop account entry")
  406.    local pw = getInput("Password:","*")
  407.    if pw == "finished" then break end
  408.    local pw2 = getInput("Again:","*")
  409.    if pw2 == "finished" then break end
  410.    if pw == pw2 then
  411.      table.insert(accounts,hash(pw..salt))
  412.      saveAccounts()
  413.      print("Password added")
  414.      os.startTimer(2)
  415.      os.pullEvent("timer")
  416.    else
  417.      print("Passwords do not match")
  418.      os.startTimer(2)
  419.      os.pullEvent("timer")
  420.    end
  421.  end
  422. end
  423.  
  424. local tArgs = {...}
  425.  
  426. if #tArgs == 1 and tArgs[1] == "-setup" then
  427.  initialSetup()
  428. elseif #tArgs ~= 0 then
  429.  print("Usage: "..fs.getName(shell.getRunningProgram()))
  430.  print("Usage: "..fs.getName(shell.getRunningProgram()).." -setup")
  431.  error()
  432. end
  433.  
  434. if not fs.exists(passFile) or not fs.exists(configFile) then
  435.  initialSetup()
  436. else
  437.  loadConfig()
  438.  loadAccounts()
  439. end
  440.  
  441. local sw, sh = term.getSize()
  442. local text
  443. local errorString
  444. local errorStringTimer
  445.  
  446. local function reset()
  447.  text = ""
  448.  trollString = ""
  449.  errorString = ""
  450.  errorStringTimer = os.startTimer(0)
  451.  rs.setOutput(rsOutput, false)
  452. end
  453.  
  454. local function checkDisk()
  455.  for _,v in pairs(rs.getSides()) do
  456.    if peripheral.getType(v) == "drive" and disk.isPresent(v) and disk.hasData(v) then
  457.      local path = disk.getMountPath(v)
  458.      if fs.exists(path.."/startup") then
  459.        if fs.exists(path.."/goodTry_startup") then fs.delete(path.."/goodTry_startup") end
  460.        fs.move(path.."/startup", path.."/goodTry_startup")
  461.        disk.eject(v)
  462.        return true
  463.      end
  464.    end
  465.  end
  466.  
  467.  return false
  468. end
  469.  
  470. reset()
  471. while true do
  472.  local w,h = term.getSize()
  473.  term.clear()
  474.  
  475.  if adv then
  476.    term.setTextColor(colors.yellow)
  477.  end
  478.  cwrite(welcomeMessage or "None found", 2)
  479.  printRecognition()
  480.  if adv then
  481.    term.setTextColor(colors.white)
  482.  end
  483.  
  484.  term.setCursorPos(6, h / 2)
  485.  write("Password: "..string.rep( "*", math.min(#text, w-22)))
  486.  
  487.  if not canDiskStay and checkDisk() then
  488.    trollString = "Error: Disks with a startup file are not allowed!"
  489.    errsorStringTimer = os.startTimer(4)
  490.  end
  491.  
  492.  if adv then
  493.    term.setTextColor(colors.red)
  494.  end
  495.  
  496.  term.setCursorPos(2, h)
  497.  write(errorString)
  498.  
  499.  term.setCursorPos(2, h - 2)
  500.  write(trollString)
  501.  
  502.  if adv then
  503.    term.setTextColor(colors.white)
  504.  end
  505.  
  506.  local event = { os.pullEventRaw() }
  507.  
  508.  if event[1] == "char" then
  509.    text = text..event[2]
  510.  elseif event[1] == "key" then
  511.    if errorString ~= "Error: Cannot terminate!" then errorString = "" end
  512.    if event[2] == keys.backspace then
  513.      text = text:sub(1, text:len() - 1)
  514.    elseif event[2] == keys.enter then
  515.      local attempt = hash(text..salt)
  516.      if checkPass( attempt ) then
  517.        rs.setOutput(rsOutput, true)
  518.        local waitTimer = os.startTimer(3)
  519.        term.setCursorPos(6, h/2)
  520.        term.clearLine()
  521.        if adv then
  522.          term.setTextColor(colors.lime)
  523.        end
  524.        cwrite("ACCESS GRANTED",h/2)
  525.        while true do
  526.          event = { os.pullEventRaw() }
  527.          if event[1] == "timer" and event[2] == waitTimer then break end
  528.        end
  529.        reset()
  530.      else
  531.        if attempt == terminatePass then
  532.          term.clear()
  533.          if adv then
  534.            term.setTextColor(colors.yellow)
  535.          end
  536.          print("Terminal Access")
  537.          term.clear = term.native.clear
  538.          error()
  539.        else
  540.          text = ""
  541.          errorString = "Error: Incorrect Password"
  542.          errorStringTimer = os.startTimer(4)
  543.        end
  544.      end
  545.    end
  546.  elseif event[1] == "timer" and event[2] == errorStringTimer then
  547.    errorString = ""
  548.    trollString = ""
  549.  elseif event[1] == "terminate" then
  550.    text = ""
  551.    errorString = "Error: Cannot terminate!"
  552.    errorStringTimer = os.startTimer(2)
  553.  end
  554. end
Advertisement
Add Comment
Please, Sign In to add comment
Advertisement